mapbox-gl 1.13.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (460) hide show
  1. package/.flowconfig +61 -0
  2. package/CHANGELOG.md +2485 -0
  3. package/LICENSE.txt +84 -0
  4. package/README.md +34 -0
  5. package/build/banner.js +4 -0
  6. package/build/check-bundle-size.js +140 -0
  7. package/build/diff-tarball.js +18 -0
  8. package/build/generate-access-token-script.js +11 -0
  9. package/build/generate-flow-typed-style-spec.js +188 -0
  10. package/build/generate-release-list.js +21 -0
  11. package/build/generate-struct-arrays.js +243 -0
  12. package/build/generate-style-code.js +159 -0
  13. package/build/mapbox-gl.js.flow +3 -0
  14. package/build/print-release-url.js +6 -0
  15. package/build/rollup_plugin_minify_style_spec.js +24 -0
  16. package/build/rollup_plugins.js +80 -0
  17. package/build/run-node +3 -0
  18. package/build/run-tap +8 -0
  19. package/build/test/build-tape.js +19 -0
  20. package/dist/mapbox-gl-csp-worker.js +2 -0
  21. package/dist/mapbox-gl-csp-worker.js.map +1 -0
  22. package/dist/mapbox-gl-csp.js +2 -0
  23. package/dist/mapbox-gl-csp.js.map +1 -0
  24. package/dist/mapbox-gl-dev.js +65889 -0
  25. package/dist/mapbox-gl-dev.js.flow +3 -0
  26. package/dist/mapbox-gl-unminified.js +42889 -0
  27. package/dist/mapbox-gl-unminified.js.map +1 -0
  28. package/dist/mapbox-gl.css +1 -0
  29. package/dist/mapbox-gl.js +42 -0
  30. package/dist/mapbox-gl.js.flow +3 -0
  31. package/dist/mapbox-gl.js.map +1 -0
  32. package/dist/style-spec/index.es.js +15032 -0
  33. package/dist/style-spec/index.es.js.map +1 -0
  34. package/dist/style-spec/index.js +15058 -0
  35. package/dist/style-spec/index.js.map +1 -0
  36. package/flow-typed/gl.js +5 -0
  37. package/flow-typed/jsdom.js +18 -0
  38. package/flow-typed/mapbox-gl-supported.js +9 -0
  39. package/flow-typed/mapbox-unitbezier.js +14 -0
  40. package/flow-typed/offscreen-canvas.js +9 -0
  41. package/flow-typed/pbf.js +25 -0
  42. package/flow-typed/point-geometry.js +44 -0
  43. package/flow-typed/potpack.js +12 -0
  44. package/flow-typed/sinon.js +28 -0
  45. package/flow-typed/vector-tile.js +41 -0
  46. package/package.json +173 -0
  47. package/src/css/mapbox-gl.css +812 -0
  48. package/src/css/svg/mapboxgl-ctrl-attrib.svg +3 -0
  49. package/src/css/svg/mapboxgl-ctrl-compass.svg +4 -0
  50. package/src/css/svg/mapboxgl-ctrl-fullscreen.svg +3 -0
  51. package/src/css/svg/mapboxgl-ctrl-geolocate.svg +5 -0
  52. package/src/css/svg/mapboxgl-ctrl-logo.svg +20 -0
  53. package/src/css/svg/mapboxgl-ctrl-shrink.svg +3 -0
  54. package/src/css/svg/mapboxgl-ctrl-zoom-in.svg +3 -0
  55. package/src/css/svg/mapboxgl-ctrl-zoom-out.svg +3 -0
  56. package/src/data/array_types.js +1135 -0
  57. package/src/data/bucket/circle_attributes.js +9 -0
  58. package/src/data/bucket/circle_bucket.js +201 -0
  59. package/src/data/bucket/fill_attributes.js +9 -0
  60. package/src/data/bucket/fill_bucket.js +229 -0
  61. package/src/data/bucket/fill_extrusion_attributes.js +10 -0
  62. package/src/data/bucket/fill_extrusion_bucket.js +283 -0
  63. package/src/data/bucket/heatmap_bucket.js +17 -0
  64. package/src/data/bucket/line_attributes.js +10 -0
  65. package/src/data/bucket/line_attributes_ext.js +10 -0
  66. package/src/data/bucket/line_bucket.js +594 -0
  67. package/src/data/bucket/pattern_attributes.js +12 -0
  68. package/src/data/bucket/pattern_bucket_features.js +60 -0
  69. package/src/data/bucket/symbol_attributes.js +117 -0
  70. package/src/data/bucket/symbol_bucket.js +937 -0
  71. package/src/data/bucket.js +123 -0
  72. package/src/data/dem_data.js +125 -0
  73. package/src/data/evaluation_feature.js +25 -0
  74. package/src/data/extent.js +18 -0
  75. package/src/data/feature_index.js +322 -0
  76. package/src/data/feature_position_map.js +131 -0
  77. package/src/data/index_array_type.js +16 -0
  78. package/src/data/load_geometry.js +46 -0
  79. package/src/data/pos_attributes.js +6 -0
  80. package/src/data/program_configuration.js +708 -0
  81. package/src/data/raster_bounds_attributes.js +7 -0
  82. package/src/data/segment.js +76 -0
  83. package/src/geo/edge_insets.js +102 -0
  84. package/src/geo/lng_lat.js +168 -0
  85. package/src/geo/lng_lat_bounds.js +276 -0
  86. package/src/geo/mercator_coordinate.js +150 -0
  87. package/src/geo/transform.js +834 -0
  88. package/src/gl/color_mode.js +34 -0
  89. package/src/gl/context.js +302 -0
  90. package/src/gl/cull_face_mode.js +26 -0
  91. package/src/gl/depth_mode.js +29 -0
  92. package/src/gl/framebuffer.js +44 -0
  93. package/src/gl/index_buffer.js +55 -0
  94. package/src/gl/stencil_mode.js +30 -0
  95. package/src/gl/types.js +84 -0
  96. package/src/gl/value.js +520 -0
  97. package/src/gl/vertex_buffer.js +119 -0
  98. package/src/index.js +230 -0
  99. package/src/render/draw_background.js +57 -0
  100. package/src/render/draw_circle.js +113 -0
  101. package/src/render/draw_collision_debug.js +172 -0
  102. package/src/render/draw_custom.js +49 -0
  103. package/src/render/draw_debug.js +127 -0
  104. package/src/render/draw_fill.js +124 -0
  105. package/src/render/draw_fill_extrusion.js +95 -0
  106. package/src/render/draw_heatmap.js +133 -0
  107. package/src/render/draw_hillshade.js +107 -0
  108. package/src/render/draw_line.js +125 -0
  109. package/src/render/draw_raster.js +125 -0
  110. package/src/render/draw_symbol.js +392 -0
  111. package/src/render/glyph_atlas.js +71 -0
  112. package/src/render/glyph_manager.js +182 -0
  113. package/src/render/image_atlas.js +149 -0
  114. package/src/render/image_manager.js +306 -0
  115. package/src/render/line_atlas.js +210 -0
  116. package/src/render/painter.js +654 -0
  117. package/src/render/program/background_program.js +103 -0
  118. package/src/render/program/circle_program.js +69 -0
  119. package/src/render/program/clipping_mask_program.js +20 -0
  120. package/src/render/program/collision_program.js +76 -0
  121. package/src/render/program/debug_program.js +35 -0
  122. package/src/render/program/fill_extrusion_program.js +122 -0
  123. package/src/render/program/fill_program.js +126 -0
  124. package/src/render/program/heatmap_program.js +83 -0
  125. package/src/render/program/hillshade_program.js +119 -0
  126. package/src/render/program/line_program.js +211 -0
  127. package/src/render/program/pattern.js +102 -0
  128. package/src/render/program/program_uniforms.js +42 -0
  129. package/src/render/program/raster_program.js +92 -0
  130. package/src/render/program/symbol_program.js +224 -0
  131. package/src/render/program.js +188 -0
  132. package/src/render/texture.js +122 -0
  133. package/src/render/uniform_binding.js +147 -0
  134. package/src/render/vertex_array_object.js +163 -0
  135. package/src/shaders/README.md +42 -0
  136. package/src/shaders/_prelude.fragment.glsl +17 -0
  137. package/src/shaders/_prelude.vertex.glsl +73 -0
  138. package/src/shaders/background.fragment.glsl +10 -0
  139. package/src/shaders/background.vertex.glsl +7 -0
  140. package/src/shaders/background_pattern.fragment.glsl +28 -0
  141. package/src/shaders/background_pattern.vertex.glsl +20 -0
  142. package/src/shaders/circle.fragment.glsl +39 -0
  143. package/src/shaders/circle.vertex.glsl +64 -0
  144. package/src/shaders/clipping_mask.fragment.glsl +3 -0
  145. package/src/shaders/clipping_mask.vertex.glsl +7 -0
  146. package/src/shaders/collision_box.fragment.glsl +21 -0
  147. package/src/shaders/collision_box.vertex.glsl +27 -0
  148. package/src/shaders/collision_circle.fragment.glsl +17 -0
  149. package/src/shaders/collision_circle.vertex.glsl +59 -0
  150. package/src/shaders/debug.fragment.glsl +9 -0
  151. package/src/shaders/debug.vertex.glsl +12 -0
  152. package/src/shaders/encode_attribute.js +17 -0
  153. package/src/shaders/fill.fragment.glsl +13 -0
  154. package/src/shaders/fill.vertex.glsl +13 -0
  155. package/src/shaders/fill_extrusion.fragment.glsl +9 -0
  156. package/src/shaders/fill_extrusion.vertex.glsl +66 -0
  157. package/src/shaders/fill_extrusion_pattern.fragment.glsl +45 -0
  158. package/src/shaders/fill_extrusion_pattern.vertex.glsl +79 -0
  159. package/src/shaders/fill_outline.fragment.glsl +17 -0
  160. package/src/shaders/fill_outline.vertex.glsl +17 -0
  161. package/src/shaders/fill_outline_pattern.fragment.glsl +43 -0
  162. package/src/shaders/fill_outline_pattern.vertex.glsl +44 -0
  163. package/src/shaders/fill_pattern.fragment.glsl +36 -0
  164. package/src/shaders/fill_pattern.vertex.glsl +39 -0
  165. package/src/shaders/heatmap.fragment.glsl +22 -0
  166. package/src/shaders/heatmap.vertex.glsl +54 -0
  167. package/src/shaders/heatmap_texture.fragment.glsl +14 -0
  168. package/src/shaders/heatmap_texture.vertex.glsl +11 -0
  169. package/src/shaders/hillshade.fragment.glsl +52 -0
  170. package/src/shaders/hillshade.vertex.glsl +11 -0
  171. package/src/shaders/hillshade_prepare.fragment.glsl +75 -0
  172. package/src/shaders/hillshade_prepare.vertex.glsl +15 -0
  173. package/src/shaders/index.js +20 -0
  174. package/src/shaders/line.fragment.glsl +30 -0
  175. package/src/shaders/line.vertex.glsl +85 -0
  176. package/src/shaders/line_gradient.fragment.glsl +34 -0
  177. package/src/shaders/line_gradient.vertex.glsl +88 -0
  178. package/src/shaders/line_pattern.fragment.glsl +74 -0
  179. package/src/shaders/line_pattern.vertex.glsl +99 -0
  180. package/src/shaders/line_sdf.fragment.glsl +45 -0
  181. package/src/shaders/line_sdf.vertex.glsl +98 -0
  182. package/src/shaders/raster.fragment.glsl +52 -0
  183. package/src/shaders/raster.vertex.glsl +21 -0
  184. package/src/shaders/shaders.js +185 -0
  185. package/src/shaders/symbol_icon.fragment.glsl +17 -0
  186. package/src/shaders/symbol_icon.vertex.glsl +94 -0
  187. package/src/shaders/symbol_sdf.fragment.glsl +52 -0
  188. package/src/shaders/symbol_sdf.vertex.glsl +115 -0
  189. package/src/shaders/symbol_text_and_icon.fragment.glsl +68 -0
  190. package/src/shaders/symbol_text_and_icon.vertex.glsl +116 -0
  191. package/src/source/canvas_source.js +238 -0
  192. package/src/source/geojson_source.js +370 -0
  193. package/src/source/geojson_worker_source.js +366 -0
  194. package/src/source/geojson_wrapper.js +94 -0
  195. package/src/source/image_source.js +307 -0
  196. package/src/source/load_tilejson.js +39 -0
  197. package/src/source/pixels_to_tile_units.js +21 -0
  198. package/src/source/query_features.js +208 -0
  199. package/src/source/raster_dem_tile_source.js +138 -0
  200. package/src/source/raster_dem_tile_worker_source.js +62 -0
  201. package/src/source/raster_tile_source.js +169 -0
  202. package/src/source/rtl_text_plugin.js +143 -0
  203. package/src/source/source.js +137 -0
  204. package/src/source/source_cache.js +953 -0
  205. package/src/source/source_state.js +159 -0
  206. package/src/source/tile.js +458 -0
  207. package/src/source/tile_bounds.js +38 -0
  208. package/src/source/tile_cache.js +212 -0
  209. package/src/source/tile_id.js +199 -0
  210. package/src/source/vector_tile_source.js +259 -0
  211. package/src/source/vector_tile_worker_source.js +216 -0
  212. package/src/source/video_source.js +203 -0
  213. package/src/source/worker.js +240 -0
  214. package/src/source/worker_source.js +107 -0
  215. package/src/source/worker_tile.js +224 -0
  216. package/src/style/create_style_layer.js +36 -0
  217. package/src/style/evaluation_parameters.js +62 -0
  218. package/src/style/format_section_override.js +56 -0
  219. package/src/style/light.js +130 -0
  220. package/src/style/load_glyph_range.js +38 -0
  221. package/src/style/load_sprite.js +67 -0
  222. package/src/style/parse_glyph_pbf.js +44 -0
  223. package/src/style/pauseable_placement.js +132 -0
  224. package/src/style/properties.js +753 -0
  225. package/src/style/query_utils.js +43 -0
  226. package/src/style/style.js +1374 -0
  227. package/src/style/style_glyph.js +17 -0
  228. package/src/style/style_image.js +137 -0
  229. package/src/style/style_layer/background_style_layer.js +21 -0
  230. package/src/style/style_layer/background_style_layer_properties.js +40 -0
  231. package/src/style/style_layer/circle_style_layer.js +98 -0
  232. package/src/style/style_layer/circle_style_layer_properties.js +63 -0
  233. package/src/style/style_layer/custom_style_layer.js +223 -0
  234. package/src/style/style_layer/fill_extrusion_style_layer.js +224 -0
  235. package/src/style/style_layer/fill_extrusion_style_layer_properties.js +50 -0
  236. package/src/style/style_layer/fill_style_layer.js +67 -0
  237. package/src/style/style_layer/fill_style_layer_properties.js +55 -0
  238. package/src/style/style_layer/heatmap_style_layer.js +73 -0
  239. package/src/style/style_layer/heatmap_style_layer_properties.js +44 -0
  240. package/src/style/style_layer/hillshade_style_layer.js +25 -0
  241. package/src/style/style_layer/hillshade_style_layer_properties.js +46 -0
  242. package/src/style/style_layer/layer_properties.js.ejs +69 -0
  243. package/src/style/style_layer/line_style_layer.js +150 -0
  244. package/src/style/style_layer/line_style_layer_properties.js +71 -0
  245. package/src/style/style_layer/raster_style_layer.js +21 -0
  246. package/src/style/style_layer/raster_style_layer_properties.js +50 -0
  247. package/src/style/style_layer/symbol_style_layer.js +190 -0
  248. package/src/style/style_layer/symbol_style_layer_properties.js +153 -0
  249. package/src/style/style_layer/typed_style_layer.js +17 -0
  250. package/src/style/style_layer.js +283 -0
  251. package/src/style/style_layer_index.js +80 -0
  252. package/src/style/validate_style.js +42 -0
  253. package/src/style/zoom_history.js +44 -0
  254. package/src/style-spec/.eslintrc +5 -0
  255. package/src/style-spec/CHANGELOG.md +468 -0
  256. package/src/style-spec/README.md +59 -0
  257. package/src/style-spec/bin/gl-style-composite +9 -0
  258. package/src/style-spec/bin/gl-style-format +22 -0
  259. package/src/style-spec/bin/gl-style-migrate +9 -0
  260. package/src/style-spec/bin/gl-style-validate +50 -0
  261. package/src/style-spec/composite.js +50 -0
  262. package/src/style-spec/declass.js +42 -0
  263. package/src/style-spec/deref.js +52 -0
  264. package/src/style-spec/diff.js +393 -0
  265. package/src/style-spec/dist/.gitkeep +0 -0
  266. package/src/style-spec/dist/index.es.js +15032 -0
  267. package/src/style-spec/dist/index.es.js.map +1 -0
  268. package/src/style-spec/dist/index.js +15058 -0
  269. package/src/style-spec/dist/index.js.map +1 -0
  270. package/src/style-spec/empty.js +29 -0
  271. package/src/style-spec/error/parsing_error.js +16 -0
  272. package/src/style-spec/error/validation_error.js +18 -0
  273. package/src/style-spec/expression/compound_expression.js +162 -0
  274. package/src/style-spec/expression/definitions/assertion.js +130 -0
  275. package/src/style-spec/expression/definitions/at.js +70 -0
  276. package/src/style-spec/expression/definitions/case.js +85 -0
  277. package/src/style-spec/expression/definitions/coalesce.js +93 -0
  278. package/src/style-spec/expression/definitions/coercion.js +133 -0
  279. package/src/style-spec/expression/definitions/collator.js +78 -0
  280. package/src/style-spec/expression/definitions/comparison.js +184 -0
  281. package/src/style-spec/expression/definitions/format.js +144 -0
  282. package/src/style-spec/expression/definitions/image.js +52 -0
  283. package/src/style-spec/expression/definitions/in.js +72 -0
  284. package/src/style-spec/expression/definitions/index.js +565 -0
  285. package/src/style-spec/expression/definitions/index_of.js +89 -0
  286. package/src/style-spec/expression/definitions/interpolate.js +267 -0
  287. package/src/style-spec/expression/definitions/length.js +61 -0
  288. package/src/style-spec/expression/definitions/let.js +72 -0
  289. package/src/style-spec/expression/definitions/literal.js +77 -0
  290. package/src/style-spec/expression/definitions/match.js +158 -0
  291. package/src/style-spec/expression/definitions/number_format.js +142 -0
  292. package/src/style-spec/expression/definitions/slice.js +86 -0
  293. package/src/style-spec/expression/definitions/step.js +120 -0
  294. package/src/style-spec/expression/definitions/var.js +46 -0
  295. package/src/style-spec/expression/definitions/within.js +342 -0
  296. package/src/style-spec/expression/evaluation_context.js +59 -0
  297. package/src/style-spec/expression/expression.js +27 -0
  298. package/src/style-spec/expression/index.js +392 -0
  299. package/src/style-spec/expression/is_constant.js +59 -0
  300. package/src/style-spec/expression/parsing_context.js +233 -0
  301. package/src/style-spec/expression/parsing_error.js +13 -0
  302. package/src/style-spec/expression/runtime_error.js +17 -0
  303. package/src/style-spec/expression/scope.js +36 -0
  304. package/src/style-spec/expression/stops.js +39 -0
  305. package/src/style-spec/expression/types/collator.js +61 -0
  306. package/src/style-spec/expression/types/formatted.js +73 -0
  307. package/src/style-spec/expression/types/resolved_image.js +29 -0
  308. package/src/style-spec/expression/types.js +126 -0
  309. package/src/style-spec/expression/values.js +123 -0
  310. package/src/style-spec/feature_filter/README.md +55 -0
  311. package/src/style-spec/feature_filter/convert.js +208 -0
  312. package/src/style-spec/feature_filter/index.js +175 -0
  313. package/src/style-spec/format.js +51 -0
  314. package/src/style-spec/function/convert.js +270 -0
  315. package/src/style-spec/function/index.js +262 -0
  316. package/src/style-spec/group_by_layout.js +75 -0
  317. package/src/style-spec/migrate/expressions.js +39 -0
  318. package/src/style-spec/migrate/v8.js +203 -0
  319. package/src/style-spec/migrate/v9.js +26 -0
  320. package/src/style-spec/migrate.js +36 -0
  321. package/src/style-spec/package.json +41 -0
  322. package/src/style-spec/read_style.js +14 -0
  323. package/src/style-spec/reference/latest.js +3 -0
  324. package/src/style-spec/reference/v8.json +5914 -0
  325. package/src/style-spec/rollup.config.js +65 -0
  326. package/src/style-spec/style-spec.js +124 -0
  327. package/src/style-spec/types.js +432 -0
  328. package/src/style-spec/util/color.js +95 -0
  329. package/src/style-spec/util/color_spaces.js +139 -0
  330. package/src/style-spec/util/deep_equal.js +28 -0
  331. package/src/style-spec/util/extend.js +10 -0
  332. package/src/style-spec/util/get_type.js +17 -0
  333. package/src/style-spec/util/interpolate.js +22 -0
  334. package/src/style-spec/util/properties.js +15 -0
  335. package/src/style-spec/util/ref_properties.js +2 -0
  336. package/src/style-spec/util/result.js +19 -0
  337. package/src/style-spec/util/unbundle_jsonlint.js +24 -0
  338. package/src/style-spec/validate/latest.js +11 -0
  339. package/src/style-spec/validate/validate.js +75 -0
  340. package/src/style-spec/validate/validate_array.js +52 -0
  341. package/src/style-spec/validate/validate_boolean.js +15 -0
  342. package/src/style-spec/validate/validate_color.js +20 -0
  343. package/src/style-spec/validate/validate_constants.js +13 -0
  344. package/src/style-spec/validate/validate_enum.js +21 -0
  345. package/src/style-spec/validate/validate_expression.js +43 -0
  346. package/src/style-spec/validate/validate_filter.js +117 -0
  347. package/src/style-spec/validate/validate_formatted.js +11 -0
  348. package/src/style-spec/validate/validate_function.js +207 -0
  349. package/src/style-spec/validate/validate_glyphs_url.js +21 -0
  350. package/src/style-spec/validate/validate_image.js +11 -0
  351. package/src/style-spec/validate/validate_layer.js +134 -0
  352. package/src/style-spec/validate/validate_layout_property.js +6 -0
  353. package/src/style-spec/validate/validate_light.js +47 -0
  354. package/src/style-spec/validate/validate_number.js +29 -0
  355. package/src/style-spec/validate/validate_object.js +61 -0
  356. package/src/style-spec/validate/validate_paint_property.js +6 -0
  357. package/src/style-spec/validate/validate_property.js +64 -0
  358. package/src/style-spec/validate/validate_source.js +111 -0
  359. package/src/style-spec/validate/validate_string.js +15 -0
  360. package/src/style-spec/validate_mapbox_api_supported.js +171 -0
  361. package/src/style-spec/validate_style.js +39 -0
  362. package/src/style-spec/validate_style.min.js +78 -0
  363. package/src/style-spec/visit.js +77 -0
  364. package/src/symbol/anchor.js +26 -0
  365. package/src/symbol/check_max_angle.js +81 -0
  366. package/src/symbol/clip_line.js +71 -0
  367. package/src/symbol/collision_feature.js +109 -0
  368. package/src/symbol/collision_index.js +373 -0
  369. package/src/symbol/cross_tile_symbol_index.js +301 -0
  370. package/src/symbol/get_anchors.js +167 -0
  371. package/src/symbol/grid_index.js +335 -0
  372. package/src/symbol/mergelines.js +82 -0
  373. package/src/symbol/one_em.js +4 -0
  374. package/src/symbol/opacity_state.js +27 -0
  375. package/src/symbol/path_interpolator.js +61 -0
  376. package/src/symbol/placement.js +1124 -0
  377. package/src/symbol/projection.js +451 -0
  378. package/src/symbol/quads.js +334 -0
  379. package/src/symbol/shaping.js +816 -0
  380. package/src/symbol/symbol_layout.js +796 -0
  381. package/src/symbol/symbol_size.js +113 -0
  382. package/src/symbol/transform_text.js +29 -0
  383. package/src/types/callback.js +17 -0
  384. package/src/types/cancelable.js +3 -0
  385. package/src/types/tilejson.js +17 -0
  386. package/src/types/transferable.js +3 -0
  387. package/src/types/window.js +172 -0
  388. package/src/ui/anchor.js +32 -0
  389. package/src/ui/camera.js +1277 -0
  390. package/src/ui/control/attribution_control.js +212 -0
  391. package/src/ui/control/fullscreen_control.js +147 -0
  392. package/src/ui/control/geolocate_control.js +707 -0
  393. package/src/ui/control/logo_control.js +92 -0
  394. package/src/ui/control/navigation_control.js +257 -0
  395. package/src/ui/control/scale_control.js +142 -0
  396. package/src/ui/default_locale.js +22 -0
  397. package/src/ui/events.js +1301 -0
  398. package/src/ui/handler/box_zoom.js +170 -0
  399. package/src/ui/handler/click_zoom.js +52 -0
  400. package/src/ui/handler/handler_util.js +12 -0
  401. package/src/ui/handler/keyboard.js +208 -0
  402. package/src/ui/handler/map_event.js +156 -0
  403. package/src/ui/handler/mouse.js +171 -0
  404. package/src/ui/handler/scroll_zoom.js +350 -0
  405. package/src/ui/handler/shim/dblclick_zoom.js +62 -0
  406. package/src/ui/handler/shim/drag_pan.js +88 -0
  407. package/src/ui/handler/shim/drag_rotate.js +67 -0
  408. package/src/ui/handler/shim/touch_zoom_rotate.js +108 -0
  409. package/src/ui/handler/tap_drag_zoom.js +103 -0
  410. package/src/ui/handler/tap_recognizer.js +133 -0
  411. package/src/ui/handler/tap_zoom.js +93 -0
  412. package/src/ui/handler/touch_pan.js +101 -0
  413. package/src/ui/handler/touch_zoom_rotate.js +305 -0
  414. package/src/ui/handler_inertia.js +158 -0
  415. package/src/ui/handler_manager.js +531 -0
  416. package/src/ui/hash.js +148 -0
  417. package/src/ui/map.js +2874 -0
  418. package/src/ui/marker.js +672 -0
  419. package/src/ui/popup.js +640 -0
  420. package/src/util/actor.js +212 -0
  421. package/src/util/ajax.js +388 -0
  422. package/src/util/browser/web_worker.js +10 -0
  423. package/src/util/browser/window.js +6 -0
  424. package/src/util/browser.js +70 -0
  425. package/src/util/classify_rings.js +52 -0
  426. package/src/util/color_ramp.js +61 -0
  427. package/src/util/config.js +30 -0
  428. package/src/util/debug.js +28 -0
  429. package/src/util/dictionary_coder.js +30 -0
  430. package/src/util/dispatcher.js +70 -0
  431. package/src/util/dom.js +142 -0
  432. package/src/util/evented.js +174 -0
  433. package/src/util/find_pole_of_inaccessibility.js +129 -0
  434. package/src/util/global_worker_pool.js +35 -0
  435. package/src/util/image.js +142 -0
  436. package/src/util/intersection_tests.js +208 -0
  437. package/src/util/is_char_in_unicode_block.js +311 -0
  438. package/src/util/mapbox.js +491 -0
  439. package/src/util/offscreen_canvas_supported.js +14 -0
  440. package/src/util/performance.js +112 -0
  441. package/src/util/primitives.js +145 -0
  442. package/src/util/resolve_tokens.js +16 -0
  443. package/src/util/script_detection.js +328 -0
  444. package/src/util/sku_token.js +42 -0
  445. package/src/util/smart_wrap.js +55 -0
  446. package/src/util/struct_array.js +243 -0
  447. package/src/util/struct_array.js.ejs +112 -0
  448. package/src/util/struct_array_layout.js.ejs +98 -0
  449. package/src/util/task_queue.js +68 -0
  450. package/src/util/throttle.js +27 -0
  451. package/src/util/throttled_invoker.js +46 -0
  452. package/src/util/tile_request_cache.js +172 -0
  453. package/src/util/util.js +524 -0
  454. package/src/util/vectortile_to_geojson.js +50 -0
  455. package/src/util/verticalize_punctuation.js +114 -0
  456. package/src/util/web_worker.js +91 -0
  457. package/src/util/web_worker_transfer.js +266 -0
  458. package/src/util/webp_supported.js +69 -0
  459. package/src/util/window.js +102 -0
  460. package/src/util/worker_pool.js +57 -0
@@ -0,0 +1,796 @@
1
+ // @flow
2
+
3
+ import Anchor from './anchor';
4
+
5
+ import {getAnchors, getCenterAnchor} from './get_anchors';
6
+ import clipLine from './clip_line';
7
+ import {shapeText, shapeIcon, WritingMode, fitIconToText} from './shaping';
8
+ import {getGlyphQuads, getIconQuads} from './quads';
9
+ import CollisionFeature from './collision_feature';
10
+ import {warnOnce} from '../util/util';
11
+ import {
12
+ allowsVerticalWritingMode,
13
+ allowsLetterSpacing
14
+ } from '../util/script_detection';
15
+ import findPoleOfInaccessibility from '../util/find_pole_of_inaccessibility';
16
+ import classifyRings from '../util/classify_rings';
17
+ import EXTENT from '../data/extent';
18
+ import SymbolBucket from '../data/bucket/symbol_bucket';
19
+ import EvaluationParameters from '../style/evaluation_parameters';
20
+ import {SIZE_PACK_FACTOR} from './symbol_size';
21
+ import ONE_EM from './one_em';
22
+ import type {CanonicalTileID} from '../source/tile_id';
23
+ import type {Shaping, PositionedIcon, TextJustify} from './shaping';
24
+ import type {CollisionBoxArray} from '../data/array_types';
25
+ import type {SymbolFeature} from '../data/bucket/symbol_bucket';
26
+ import type {StyleImage} from '../style/style_image';
27
+ import type {StyleGlyph} from '../style/style_glyph';
28
+ import type SymbolStyleLayer from '../style/style_layer/symbol_style_layer';
29
+ import type {ImagePosition} from '../render/image_atlas';
30
+ import type {GlyphPosition} from '../render/glyph_atlas';
31
+ import type {PossiblyEvaluatedPropertyValue} from '../style/properties';
32
+
33
+ import Point from '@mapbox/point-geometry';
34
+ import murmur3 from 'murmurhash-js';
35
+
36
+ // The symbol layout process needs `text-size` evaluated at up to five different zoom levels, and
37
+ // `icon-size` at up to three:
38
+ //
39
+ // 1. `text-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `text-size`
40
+ // expressions, and to calculate the box dimensions for icon-text-fit.
41
+ // 2. `icon-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `icon-size`
42
+ // expressions.
43
+ // 3. `text-size` and `icon-size` at the zoom level of the bucket, plus one. Used to calculate collision boxes.
44
+ // 4. `text-size` at zoom level 18. Used for something line-symbol-placement-related.
45
+ // 5. For composite `*-size` expressions: two zoom levels of curve stops that "cover" the zoom level of the
46
+ // bucket. These go into a vertex buffer and are used by the shader to interpolate the size at render time.
47
+ //
48
+ // (1) and (2) are stored in `bucket.layers[0].layout`. The remainder are below.
49
+ //
50
+ type Sizes = {
51
+ layoutTextSize: PossiblyEvaluatedPropertyValue<number>, // (3)
52
+ layoutIconSize: PossiblyEvaluatedPropertyValue<number>, // (3)
53
+ textMaxSize: PossiblyEvaluatedPropertyValue<number>, // (4)
54
+ compositeTextSizes: [PossiblyEvaluatedPropertyValue<number>, PossiblyEvaluatedPropertyValue<number>], // (5)
55
+ compositeIconSizes: [PossiblyEvaluatedPropertyValue<number>, PossiblyEvaluatedPropertyValue<number>], // (5)
56
+ };
57
+
58
+ export type TextAnchor = 'center' | 'left' | 'right' | 'top' | 'bottom' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right';
59
+
60
+ // The radial offset is to the edge of the text box
61
+ // In the horizontal direction, the edge of the text box is where glyphs start
62
+ // But in the vertical direction, the glyphs appear to "start" at the baseline
63
+ // We don't actually load baseline data, but we assume an offset of ONE_EM - 17
64
+ // (see "yOffset" in shaping.js)
65
+ const baselineOffset = 7;
66
+ const INVALID_TEXT_OFFSET = Number.POSITIVE_INFINITY;
67
+
68
+ export function evaluateVariableOffset(anchor: TextAnchor, offset: [number, number]) {
69
+
70
+ function fromRadialOffset(anchor: TextAnchor, radialOffset: number) {
71
+ let x = 0, y = 0;
72
+ if (radialOffset < 0) radialOffset = 0; // Ignore negative offset.
73
+ // solve for r where r^2 + r^2 = radialOffset^2
74
+ const hypotenuse = radialOffset / Math.sqrt(2);
75
+ switch (anchor) {
76
+ case 'top-right':
77
+ case 'top-left':
78
+ y = hypotenuse - baselineOffset;
79
+ break;
80
+ case 'bottom-right':
81
+ case 'bottom-left':
82
+ y = -hypotenuse + baselineOffset;
83
+ break;
84
+ case 'bottom':
85
+ y = -radialOffset + baselineOffset;
86
+ break;
87
+ case 'top':
88
+ y = radialOffset - baselineOffset;
89
+ break;
90
+ }
91
+
92
+ switch (anchor) {
93
+ case 'top-right':
94
+ case 'bottom-right':
95
+ x = -hypotenuse;
96
+ break;
97
+ case 'top-left':
98
+ case 'bottom-left':
99
+ x = hypotenuse;
100
+ break;
101
+ case 'left':
102
+ x = radialOffset;
103
+ break;
104
+ case 'right':
105
+ x = -radialOffset;
106
+ break;
107
+ }
108
+
109
+ return [x, y];
110
+ }
111
+
112
+ function fromTextOffset(anchor: TextAnchor, offsetX: number, offsetY: number) {
113
+ let x = 0, y = 0;
114
+ // Use absolute offset values.
115
+ offsetX = Math.abs(offsetX);
116
+ offsetY = Math.abs(offsetY);
117
+
118
+ switch (anchor) {
119
+ case 'top-right':
120
+ case 'top-left':
121
+ case 'top':
122
+ y = offsetY - baselineOffset;
123
+ break;
124
+ case 'bottom-right':
125
+ case 'bottom-left':
126
+ case 'bottom':
127
+ y = -offsetY + baselineOffset;
128
+ break;
129
+ }
130
+
131
+ switch (anchor) {
132
+ case 'top-right':
133
+ case 'bottom-right':
134
+ case 'right':
135
+ x = -offsetX;
136
+ break;
137
+ case 'top-left':
138
+ case 'bottom-left':
139
+ case 'left':
140
+ x = offsetX;
141
+ break;
142
+ }
143
+
144
+ return [x, y];
145
+ }
146
+
147
+ return (offset[1] !== INVALID_TEXT_OFFSET) ? fromTextOffset(anchor, offset[0], offset[1]) : fromRadialOffset(anchor, offset[0]);
148
+ }
149
+
150
+ export function performSymbolLayout(bucket: SymbolBucket,
151
+ glyphMap: {[_: string]: {[number]: ?StyleGlyph}},
152
+ glyphPositions: {[_: string]: {[number]: GlyphPosition}},
153
+ imageMap: {[_: string]: StyleImage},
154
+ imagePositions: {[_: string]: ImagePosition},
155
+ showCollisionBoxes: boolean,
156
+ canonical: CanonicalTileID) {
157
+ bucket.createArrays();
158
+
159
+ const tileSize = 512 * bucket.overscaling;
160
+ bucket.tilePixelRatio = EXTENT / tileSize;
161
+ bucket.compareText = {};
162
+ bucket.iconsNeedLinear = false;
163
+
164
+ const layout = bucket.layers[0].layout;
165
+ const unevaluatedLayoutValues = bucket.layers[0]._unevaluatedLayout._values;
166
+
167
+ const sizes = {};
168
+
169
+ if (bucket.textSizeData.kind === 'composite') {
170
+ const {minZoom, maxZoom} = bucket.textSizeData;
171
+ sizes.compositeTextSizes = [
172
+ unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(minZoom), canonical),
173
+ unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), canonical)
174
+ ];
175
+ }
176
+
177
+ if (bucket.iconSizeData.kind === 'composite') {
178
+ const {minZoom, maxZoom} = bucket.iconSizeData;
179
+ sizes.compositeIconSizes = [
180
+ unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(minZoom), canonical),
181
+ unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), canonical)
182
+ ];
183
+ }
184
+
185
+ sizes.layoutTextSize = unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(bucket.zoom + 1), canonical);
186
+ sizes.layoutIconSize = unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(bucket.zoom + 1), canonical);
187
+ sizes.textMaxSize = unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(18));
188
+
189
+ const lineHeight = layout.get('text-line-height') * ONE_EM;
190
+ const textAlongLine = layout.get('text-rotation-alignment') === 'map' && layout.get('symbol-placement') !== 'point';
191
+ const keepUpright = layout.get('text-keep-upright');
192
+ const textSize = layout.get('text-size');
193
+
194
+ for (const feature of bucket.features) {
195
+ const fontstack = layout.get('text-font').evaluate(feature, {}, canonical).join(',');
196
+ const layoutTextSizeThisZoom = textSize.evaluate(feature, {}, canonical);
197
+ const layoutTextSize = sizes.layoutTextSize.evaluate(feature, {}, canonical);
198
+ const layoutIconSize = sizes.layoutIconSize.evaluate(feature, {}, canonical);
199
+
200
+ const shapedTextOrientations = {
201
+ horizontal: {},
202
+ vertical: undefined
203
+ };
204
+ const text = feature.text;
205
+ let textOffset: [number, number] = [0, 0];
206
+ if (text) {
207
+ const unformattedText = text.toString();
208
+ const spacing = layout.get('text-letter-spacing').evaluate(feature, {}, canonical) * ONE_EM;
209
+ const spacingIfAllowed = allowsLetterSpacing(unformattedText) ? spacing : 0;
210
+
211
+ const textAnchor = layout.get('text-anchor').evaluate(feature, {}, canonical);
212
+ const variableTextAnchor = layout.get('text-variable-anchor');
213
+
214
+ if (!variableTextAnchor) {
215
+ const radialOffset = layout.get('text-radial-offset').evaluate(feature, {}, canonical);
216
+ // Layers with variable anchors use the `text-radial-offset` property and the [x, y] offset vector
217
+ // is calculated at placement time instead of layout time
218
+ if (radialOffset) {
219
+ // The style spec says don't use `text-offset` and `text-radial-offset` together
220
+ // but doesn't actually specify what happens if you use both. We go with the radial offset.
221
+ textOffset = evaluateVariableOffset(textAnchor, [radialOffset * ONE_EM, INVALID_TEXT_OFFSET]);
222
+ } else {
223
+ textOffset = (layout.get('text-offset').evaluate(feature, {}, canonical).map(t => t * ONE_EM): any);
224
+ }
225
+ }
226
+
227
+ let textJustify = textAlongLine ?
228
+ "center" :
229
+ layout.get('text-justify').evaluate(feature, {}, canonical);
230
+
231
+ const symbolPlacement = layout.get('symbol-placement');
232
+ const maxWidth = symbolPlacement === 'point' ?
233
+ layout.get('text-max-width').evaluate(feature, {}, canonical) * ONE_EM :
234
+ 0;
235
+
236
+ const addVerticalShapingForPointLabelIfNeeded = () => {
237
+ if (bucket.allowVerticalPlacement && allowsVerticalWritingMode(unformattedText)) {
238
+ // Vertical POI label placement is meant to be used for scripts that support vertical
239
+ // writing mode, thus, default left justification is used. If Latin
240
+ // scripts would need to be supported, this should take into account other justifications.
241
+ shapedTextOrientations.vertical = shapeText(text, glyphMap, glyphPositions, imagePositions, fontstack, maxWidth, lineHeight, textAnchor,
242
+ 'left', spacingIfAllowed, textOffset, WritingMode.vertical, true, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);
243
+ }
244
+ };
245
+
246
+ // If this layer uses text-variable-anchor, generate shapings for all justification possibilities.
247
+ if (!textAlongLine && variableTextAnchor) {
248
+ const justifications = textJustify === "auto" ?
249
+ variableTextAnchor.map(a => getAnchorJustification(a)) :
250
+ [textJustify];
251
+
252
+ let singleLine = false;
253
+ for (let i = 0; i < justifications.length; i++) {
254
+ const justification: TextJustify = justifications[i];
255
+ if (shapedTextOrientations.horizontal[justification]) continue;
256
+ if (singleLine) {
257
+ // If the shaping for the first justification was only a single line, we
258
+ // can re-use it for the other justifications
259
+ shapedTextOrientations.horizontal[justification] = shapedTextOrientations.horizontal[0];
260
+ } else {
261
+ // If using text-variable-anchor for the layer, we use a center anchor for all shapings and apply
262
+ // the offsets for the anchor in the placement step.
263
+ const shaping = shapeText(text, glyphMap, glyphPositions, imagePositions, fontstack, maxWidth, lineHeight, 'center',
264
+ justification, spacingIfAllowed, textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);
265
+ if (shaping) {
266
+ shapedTextOrientations.horizontal[justification] = shaping;
267
+ singleLine = shaping.positionedLines.length === 1;
268
+ }
269
+ }
270
+ }
271
+
272
+ addVerticalShapingForPointLabelIfNeeded();
273
+ } else {
274
+ if (textJustify === "auto") {
275
+ textJustify = getAnchorJustification(textAnchor);
276
+ }
277
+
278
+ // Horizontal point or line label.
279
+ const shaping = shapeText(text, glyphMap, glyphPositions, imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify, spacingIfAllowed,
280
+ textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);
281
+ if (shaping) shapedTextOrientations.horizontal[textJustify] = shaping;
282
+
283
+ // Vertical point label (if allowVerticalPlacement is enabled).
284
+ addVerticalShapingForPointLabelIfNeeded();
285
+
286
+ // Verticalized line label.
287
+ if (allowsVerticalWritingMode(unformattedText) && textAlongLine && keepUpright) {
288
+ shapedTextOrientations.vertical = shapeText(text, glyphMap, glyphPositions, imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify,
289
+ spacingIfAllowed, textOffset, WritingMode.vertical, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);
290
+ }
291
+ }
292
+ }
293
+
294
+ let shapedIcon;
295
+ let isSDFIcon = false;
296
+ if (feature.icon && feature.icon.name) {
297
+ const image = imageMap[feature.icon.name];
298
+ if (image) {
299
+ shapedIcon = shapeIcon(
300
+ imagePositions[feature.icon.name],
301
+ layout.get('icon-offset').evaluate(feature, {}, canonical),
302
+ layout.get('icon-anchor').evaluate(feature, {}, canonical));
303
+ isSDFIcon = image.sdf;
304
+ if (bucket.sdfIcons === undefined) {
305
+ bucket.sdfIcons = image.sdf;
306
+ } else if (bucket.sdfIcons !== image.sdf) {
307
+ warnOnce('Style sheet warning: Cannot mix SDF and non-SDF icons in one buffer');
308
+ }
309
+ if (image.pixelRatio !== bucket.pixelRatio) {
310
+ bucket.iconsNeedLinear = true;
311
+ } else if (layout.get('icon-rotate').constantOr(1) !== 0) {
312
+ bucket.iconsNeedLinear = true;
313
+ }
314
+ }
315
+ }
316
+
317
+ const shapedText = getDefaultHorizontalShaping(shapedTextOrientations.horizontal) || shapedTextOrientations.vertical;
318
+ bucket.iconsInText = shapedText ? shapedText.iconsInText : false;
319
+ if (shapedText || shapedIcon) {
320
+ addFeature(bucket, feature, shapedTextOrientations, shapedIcon, imageMap, sizes, layoutTextSize, layoutIconSize, textOffset, isSDFIcon, canonical);
321
+ }
322
+ }
323
+
324
+ if (showCollisionBoxes) {
325
+ bucket.generateCollisionDebugBuffers();
326
+ }
327
+ }
328
+
329
+ // Choose the justification that matches the direction of the TextAnchor
330
+ export function getAnchorJustification(anchor: TextAnchor): TextJustify {
331
+ switch (anchor) {
332
+ case 'right':
333
+ case 'top-right':
334
+ case 'bottom-right':
335
+ return 'right';
336
+ case 'left':
337
+ case 'top-left':
338
+ case 'bottom-left':
339
+ return 'left';
340
+ }
341
+ return 'center';
342
+ }
343
+
344
+ /**
345
+ * Given a feature and its shaped text and icon data, add a 'symbol
346
+ * instance' for each _possible_ placement of the symbol feature.
347
+ * (At render timePlaceSymbols#place() selects which of these instances to
348
+ * show or hide based on collisions with symbols in other layers.)
349
+ * @private
350
+ */
351
+ function addFeature(bucket: SymbolBucket,
352
+ feature: SymbolFeature,
353
+ shapedTextOrientations: any,
354
+ shapedIcon: PositionedIcon | void,
355
+ imageMap: {[_: string]: StyleImage},
356
+ sizes: Sizes,
357
+ layoutTextSize: number,
358
+ layoutIconSize: number,
359
+ textOffset: [number, number],
360
+ isSDFIcon: boolean, canonical: CanonicalTileID) {
361
+ // To reduce the number of labels that jump around when zooming we need
362
+ // to use a text-size value that is the same for all zoom levels.
363
+ // bucket calculates text-size at a high zoom level so that all tiles can
364
+ // use the same value when calculating anchor positions.
365
+ let textMaxSize = sizes.textMaxSize.evaluate(feature, {});
366
+ if (textMaxSize === undefined) {
367
+ textMaxSize = layoutTextSize;
368
+ }
369
+ const layout = bucket.layers[0].layout;
370
+ const iconOffset = layout.get('icon-offset').evaluate(feature, {}, canonical);
371
+ const defaultHorizontalShaping = getDefaultHorizontalShaping(shapedTextOrientations.horizontal);
372
+ const glyphSize = 24,
373
+ fontScale = layoutTextSize / glyphSize,
374
+ textBoxScale = bucket.tilePixelRatio * fontScale,
375
+ textMaxBoxScale = bucket.tilePixelRatio * textMaxSize / glyphSize,
376
+ iconBoxScale = bucket.tilePixelRatio * layoutIconSize,
377
+ symbolMinDistance = bucket.tilePixelRatio * layout.get('symbol-spacing'),
378
+ textPadding = layout.get('text-padding') * bucket.tilePixelRatio,
379
+ iconPadding = layout.get('icon-padding') * bucket.tilePixelRatio,
380
+ textMaxAngle = layout.get('text-max-angle') / 180 * Math.PI,
381
+ textAlongLine = layout.get('text-rotation-alignment') === 'map' && layout.get('symbol-placement') !== 'point',
382
+ iconAlongLine = layout.get('icon-rotation-alignment') === 'map' && layout.get('symbol-placement') !== 'point',
383
+ symbolPlacement = layout.get('symbol-placement'),
384
+ textRepeatDistance = symbolMinDistance / 2;
385
+
386
+ const iconTextFit = layout.get('icon-text-fit');
387
+ let verticallyShapedIcon;
388
+ // Adjust shaped icon size when icon-text-fit is used.
389
+ if (shapedIcon && iconTextFit !== 'none') {
390
+ if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) {
391
+ verticallyShapedIcon = fitIconToText(shapedIcon, shapedTextOrientations.vertical, iconTextFit,
392
+ layout.get('icon-text-fit-padding'), iconOffset, fontScale);
393
+ }
394
+ if (defaultHorizontalShaping) {
395
+ shapedIcon = fitIconToText(shapedIcon, defaultHorizontalShaping, iconTextFit,
396
+ layout.get('icon-text-fit-padding'), iconOffset, fontScale);
397
+ }
398
+ }
399
+
400
+ const addSymbolAtAnchor = (line, anchor) => {
401
+ if (anchor.x < 0 || anchor.x >= EXTENT || anchor.y < 0 || anchor.y >= EXTENT) {
402
+ // Symbol layers are drawn across tile boundaries, We filter out symbols
403
+ // outside our tile boundaries (which may be included in vector tile buffers)
404
+ // to prevent double-drawing symbols.
405
+ return;
406
+ }
407
+
408
+ addSymbol(bucket, anchor, line, shapedTextOrientations, shapedIcon, imageMap, verticallyShapedIcon, bucket.layers[0],
409
+ bucket.collisionBoxArray, feature.index, feature.sourceLayerIndex, bucket.index,
410
+ textBoxScale, textPadding, textAlongLine, textOffset,
411
+ iconBoxScale, iconPadding, iconAlongLine, iconOffset,
412
+ feature, sizes, isSDFIcon, canonical, layoutTextSize);
413
+ };
414
+
415
+ if (symbolPlacement === 'line') {
416
+ for (const line of clipLine(feature.geometry, 0, 0, EXTENT, EXTENT)) {
417
+ const anchors = getAnchors(
418
+ line,
419
+ symbolMinDistance,
420
+ textMaxAngle,
421
+ shapedTextOrientations.vertical || defaultHorizontalShaping,
422
+ shapedIcon,
423
+ glyphSize,
424
+ textMaxBoxScale,
425
+ bucket.overscaling,
426
+ EXTENT
427
+ );
428
+ for (const anchor of anchors) {
429
+ const shapedText = defaultHorizontalShaping;
430
+ if (!shapedText || !anchorIsTooClose(bucket, shapedText.text, textRepeatDistance, anchor)) {
431
+ addSymbolAtAnchor(line, anchor);
432
+ }
433
+ }
434
+ }
435
+ } else if (symbolPlacement === 'line-center') {
436
+ // No clipping, multiple lines per feature are allowed
437
+ // "lines" with only one point are ignored as in clipLines
438
+ for (const line of feature.geometry) {
439
+ if (line.length > 1) {
440
+ const anchor = getCenterAnchor(
441
+ line,
442
+ textMaxAngle,
443
+ shapedTextOrientations.vertical || defaultHorizontalShaping,
444
+ shapedIcon,
445
+ glyphSize,
446
+ textMaxBoxScale);
447
+ if (anchor) {
448
+ addSymbolAtAnchor(line, anchor);
449
+ }
450
+ }
451
+ }
452
+ } else if (feature.type === 'Polygon') {
453
+ for (const polygon of classifyRings(feature.geometry, 0)) {
454
+ // 16 here represents 2 pixels
455
+ const poi = findPoleOfInaccessibility(polygon, 16);
456
+ addSymbolAtAnchor(polygon[0], new Anchor(poi.x, poi.y, 0));
457
+ }
458
+ } else if (feature.type === 'LineString') {
459
+ // https://github.com/mapbox/mapbox-gl-js/issues/3808
460
+ for (const line of feature.geometry) {
461
+ addSymbolAtAnchor(line, new Anchor(line[0].x, line[0].y, 0));
462
+ }
463
+ } else if (feature.type === 'Point') {
464
+ for (const points of feature.geometry) {
465
+ for (const point of points) {
466
+ addSymbolAtAnchor([point], new Anchor(point.x, point.y, 0));
467
+ }
468
+ }
469
+ }
470
+ }
471
+
472
+ const MAX_GLYPH_ICON_SIZE = 255;
473
+ const MAX_PACKED_SIZE = MAX_GLYPH_ICON_SIZE * SIZE_PACK_FACTOR;
474
+ export {MAX_PACKED_SIZE};
475
+
476
+ function addTextVertices(bucket: SymbolBucket,
477
+ anchor: Point,
478
+ shapedText: Shaping,
479
+ imageMap: {[_: string]: StyleImage},
480
+ layer: SymbolStyleLayer,
481
+ textAlongLine: boolean,
482
+ feature: SymbolFeature,
483
+ textOffset: [number, number],
484
+ lineArray: {lineStartIndex: number, lineLength: number},
485
+ writingMode: number,
486
+ placementTypes: Array<'vertical' | 'center' | 'left' | 'right'>,
487
+ placedTextSymbolIndices: {[_: string]: number},
488
+ placedIconIndex: number,
489
+ sizes: Sizes,
490
+ canonical: CanonicalTileID) {
491
+ const glyphQuads = getGlyphQuads(anchor, shapedText, textOffset,
492
+ layer, textAlongLine, feature, imageMap, bucket.allowVerticalPlacement);
493
+
494
+ const sizeData = bucket.textSizeData;
495
+ let textSizeData = null;
496
+
497
+ if (sizeData.kind === 'source') {
498
+ textSizeData = [
499
+ SIZE_PACK_FACTOR * layer.layout.get('text-size').evaluate(feature, {})
500
+ ];
501
+ if (textSizeData[0] > MAX_PACKED_SIZE) {
502
+ warnOnce(`${bucket.layerIds[0]}: Value for "text-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "text-size".`);
503
+ }
504
+ } else if (sizeData.kind === 'composite') {
505
+ textSizeData = [
506
+ SIZE_PACK_FACTOR * sizes.compositeTextSizes[0].evaluate(feature, {}, canonical),
507
+ SIZE_PACK_FACTOR * sizes.compositeTextSizes[1].evaluate(feature, {}, canonical)
508
+ ];
509
+ if (textSizeData[0] > MAX_PACKED_SIZE || textSizeData[1] > MAX_PACKED_SIZE) {
510
+ warnOnce(`${bucket.layerIds[0]}: Value for "text-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "text-size".`);
511
+ }
512
+ }
513
+
514
+ bucket.addSymbols(
515
+ bucket.text,
516
+ glyphQuads,
517
+ textSizeData,
518
+ textOffset,
519
+ textAlongLine,
520
+ feature,
521
+ writingMode,
522
+ anchor,
523
+ lineArray.lineStartIndex,
524
+ lineArray.lineLength,
525
+ placedIconIndex,
526
+ canonical);
527
+
528
+ // The placedSymbolArray is used at render time in drawTileSymbols
529
+ // These indices allow access to the array at collision detection time
530
+ for (const placementType of placementTypes) {
531
+ placedTextSymbolIndices[placementType] = bucket.text.placedSymbolArray.length - 1;
532
+ }
533
+
534
+ return glyphQuads.length * 4;
535
+ }
536
+
537
+ function getDefaultHorizontalShaping(horizontalShaping: {[_: TextJustify]: Shaping}): Shaping | null {
538
+ // We don't care which shaping we get because this is used for collision purposes
539
+ // and all the justifications have the same collision box
540
+ for (const justification: any in horizontalShaping) {
541
+ return horizontalShaping[justification];
542
+ }
543
+ return null;
544
+ }
545
+
546
+ /**
547
+ * Add a single label & icon placement.
548
+ *
549
+ * @private
550
+ */
551
+ function addSymbol(bucket: SymbolBucket,
552
+ anchor: Anchor,
553
+ line: Array<Point>,
554
+ shapedTextOrientations: any,
555
+ shapedIcon: PositionedIcon | void,
556
+ imageMap: {[_: string]: StyleImage},
557
+ verticallyShapedIcon: PositionedIcon | void,
558
+ layer: SymbolStyleLayer,
559
+ collisionBoxArray: CollisionBoxArray,
560
+ featureIndex: number,
561
+ sourceLayerIndex: number,
562
+ bucketIndex: number,
563
+ textBoxScale: number,
564
+ textPadding: number,
565
+ textAlongLine: boolean,
566
+ textOffset: [number, number],
567
+ iconBoxScale: number,
568
+ iconPadding: number,
569
+ iconAlongLine: boolean,
570
+ iconOffset: [number, number],
571
+ feature: SymbolFeature,
572
+ sizes: Sizes,
573
+ isSDFIcon: boolean,
574
+ canonical: CanonicalTileID,
575
+ layoutTextSize: number) {
576
+ const lineArray = bucket.addToLineVertexArray(anchor, line);
577
+
578
+ let textCollisionFeature, iconCollisionFeature, verticalTextCollisionFeature, verticalIconCollisionFeature;
579
+
580
+ let numIconVertices = 0;
581
+ let numVerticalIconVertices = 0;
582
+ let numHorizontalGlyphVertices = 0;
583
+ let numVerticalGlyphVertices = 0;
584
+ let placedIconSymbolIndex = -1;
585
+ let verticalPlacedIconSymbolIndex = -1;
586
+ const placedTextSymbolIndices = {};
587
+ let key = murmur3('');
588
+
589
+ let textOffset0 = 0;
590
+ let textOffset1 = 0;
591
+ if (layer._unevaluatedLayout.getValue('text-radial-offset') === undefined) {
592
+ [textOffset0, textOffset1] = (layer.layout.get('text-offset').evaluate(feature, {}, canonical).map(t => t * ONE_EM): any);
593
+ } else {
594
+ textOffset0 = layer.layout.get('text-radial-offset').evaluate(feature, {}, canonical) * ONE_EM;
595
+ textOffset1 = INVALID_TEXT_OFFSET;
596
+ }
597
+
598
+ if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) {
599
+ const textRotation = layer.layout.get('text-rotate').evaluate(feature, {}, canonical);
600
+ const verticalTextRotation = textRotation + 90.0;
601
+ const verticalShaping = shapedTextOrientations.vertical;
602
+ verticalTextCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticalShaping, textBoxScale, textPadding, textAlongLine, verticalTextRotation);
603
+
604
+ if (verticallyShapedIcon) {
605
+ verticalIconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticallyShapedIcon, iconBoxScale, iconPadding, textAlongLine, verticalTextRotation);
606
+ }
607
+ }
608
+
609
+ //Place icon first, so text can have a reference to its index in the placed symbol array.
610
+ //Text symbols can lazily shift at render-time because of variable anchor placement.
611
+ //If the style specifies an `icon-text-fit` then the icon would have to shift along with it.
612
+ // For more info check `updateVariableAnchors` in `draw_symbol.js` .
613
+ if (shapedIcon) {
614
+ const iconRotate = layer.layout.get('icon-rotate').evaluate(feature, {});
615
+ const hasIconTextFit = layer.layout.get('icon-text-fit') !== 'none';
616
+ const iconQuads = getIconQuads(shapedIcon, iconRotate, isSDFIcon, hasIconTextFit);
617
+ const verticalIconQuads = verticallyShapedIcon ? getIconQuads(verticallyShapedIcon, iconRotate, isSDFIcon, hasIconTextFit) : undefined;
618
+ iconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shapedIcon, iconBoxScale, iconPadding, /*align boxes to line*/false, iconRotate);
619
+
620
+ numIconVertices = iconQuads.length * 4;
621
+
622
+ const sizeData = bucket.iconSizeData;
623
+ let iconSizeData = null;
624
+
625
+ if (sizeData.kind === 'source') {
626
+ iconSizeData = [
627
+ SIZE_PACK_FACTOR * layer.layout.get('icon-size').evaluate(feature, {})
628
+ ];
629
+ if (iconSizeData[0] > MAX_PACKED_SIZE) {
630
+ warnOnce(`${bucket.layerIds[0]}: Value for "icon-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "icon-size".`);
631
+ }
632
+ } else if (sizeData.kind === 'composite') {
633
+ iconSizeData = [
634
+ SIZE_PACK_FACTOR * sizes.compositeIconSizes[0].evaluate(feature, {}, canonical),
635
+ SIZE_PACK_FACTOR * sizes.compositeIconSizes[1].evaluate(feature, {}, canonical)
636
+ ];
637
+ if (iconSizeData[0] > MAX_PACKED_SIZE || iconSizeData[1] > MAX_PACKED_SIZE) {
638
+ warnOnce(`${bucket.layerIds[0]}: Value for "icon-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "icon-size".`);
639
+ }
640
+ }
641
+
642
+ bucket.addSymbols(
643
+ bucket.icon,
644
+ iconQuads,
645
+ iconSizeData,
646
+ iconOffset,
647
+ iconAlongLine,
648
+ feature,
649
+ false,
650
+ anchor,
651
+ lineArray.lineStartIndex,
652
+ lineArray.lineLength,
653
+ // The icon itself does not have an associated symbol since the text isnt placed yet
654
+ -1, canonical);
655
+
656
+ placedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1;
657
+
658
+ if (verticalIconQuads) {
659
+ numVerticalIconVertices = verticalIconQuads.length * 4;
660
+
661
+ bucket.addSymbols(
662
+ bucket.icon,
663
+ verticalIconQuads,
664
+ iconSizeData,
665
+ iconOffset,
666
+ iconAlongLine,
667
+ feature,
668
+ WritingMode.vertical,
669
+ anchor,
670
+ lineArray.lineStartIndex,
671
+ lineArray.lineLength,
672
+ // The icon itself does not have an associated symbol since the text isnt placed yet
673
+ -1, canonical);
674
+
675
+ verticalPlacedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1;
676
+ }
677
+ }
678
+
679
+ for (const justification: any in shapedTextOrientations.horizontal) {
680
+ const shaping = shapedTextOrientations.horizontal[justification];
681
+
682
+ if (!textCollisionFeature) {
683
+ key = murmur3(shaping.text);
684
+ const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}, canonical);
685
+ // As a collision approximation, we can use either the vertical or any of the horizontal versions of the feature
686
+ // We're counting on all versions having similar dimensions
687
+ textCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shaping, textBoxScale, textPadding, textAlongLine, textRotate);
688
+ }
689
+
690
+ const singleLine = shaping.positionedLines.length === 1;
691
+ numHorizontalGlyphVertices += addTextVertices(
692
+ bucket, anchor, shaping, imageMap, layer, textAlongLine, feature, textOffset, lineArray,
693
+ shapedTextOrientations.vertical ? WritingMode.horizontal : WritingMode.horizontalOnly,
694
+ singleLine ? (Object.keys(shapedTextOrientations.horizontal): any) : [justification],
695
+ placedTextSymbolIndices, placedIconSymbolIndex, sizes, canonical);
696
+
697
+ if (singleLine) {
698
+ break;
699
+ }
700
+ }
701
+
702
+ if (shapedTextOrientations.vertical) {
703
+ numVerticalGlyphVertices += addTextVertices(
704
+ bucket, anchor, shapedTextOrientations.vertical, imageMap, layer, textAlongLine, feature,
705
+ textOffset, lineArray, WritingMode.vertical, ['vertical'], placedTextSymbolIndices, verticalPlacedIconSymbolIndex, sizes, canonical);
706
+ }
707
+
708
+ const textBoxStartIndex = textCollisionFeature ? textCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;
709
+ const textBoxEndIndex = textCollisionFeature ? textCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;
710
+
711
+ const verticalTextBoxStartIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;
712
+ const verticalTextBoxEndIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;
713
+
714
+ const iconBoxStartIndex = iconCollisionFeature ? iconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;
715
+ const iconBoxEndIndex = iconCollisionFeature ? iconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;
716
+
717
+ const verticalIconBoxStartIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;
718
+ const verticalIconBoxEndIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;
719
+
720
+ // Check if runtime collision circles should be used for any of the collision features.
721
+ // It is enough to choose the tallest feature shape as circles are always placed on a line.
722
+ // All measurements are in glyph metrics and later converted into pixels using proper font size "layoutTextSize"
723
+ let collisionCircleDiameter = -1;
724
+
725
+ const getCollisionCircleHeight = (feature: ?CollisionFeature, prevHeight: number): number => {
726
+ if (feature && feature.circleDiameter)
727
+ return Math.max(feature.circleDiameter, prevHeight);
728
+ return prevHeight;
729
+ };
730
+
731
+ collisionCircleDiameter = getCollisionCircleHeight(textCollisionFeature, collisionCircleDiameter);
732
+ collisionCircleDiameter = getCollisionCircleHeight(verticalTextCollisionFeature, collisionCircleDiameter);
733
+ collisionCircleDiameter = getCollisionCircleHeight(iconCollisionFeature, collisionCircleDiameter);
734
+ collisionCircleDiameter = getCollisionCircleHeight(verticalIconCollisionFeature, collisionCircleDiameter);
735
+ const useRuntimeCollisionCircles = (collisionCircleDiameter > -1) ? 1 : 0;
736
+
737
+ // Convert circle collision height into pixels
738
+ if (useRuntimeCollisionCircles)
739
+ collisionCircleDiameter *= layoutTextSize / ONE_EM;
740
+
741
+ if (bucket.glyphOffsetArray.length >= SymbolBucket.MAX_GLYPHS) warnOnce(
742
+ "Too many glyphs being rendered in a tile. See https://github.com/mapbox/mapbox-gl-js/issues/2907"
743
+ );
744
+
745
+ if (feature.sortKey !== undefined) {
746
+ bucket.addToSortKeyRanges(bucket.symbolInstances.length, feature.sortKey);
747
+ }
748
+
749
+ bucket.symbolInstances.emplaceBack(
750
+ anchor.x,
751
+ anchor.y,
752
+ placedTextSymbolIndices.right >= 0 ? placedTextSymbolIndices.right : -1,
753
+ placedTextSymbolIndices.center >= 0 ? placedTextSymbolIndices.center : -1,
754
+ placedTextSymbolIndices.left >= 0 ? placedTextSymbolIndices.left : -1,
755
+ placedTextSymbolIndices.vertical || -1,
756
+ placedIconSymbolIndex,
757
+ verticalPlacedIconSymbolIndex,
758
+ key,
759
+ textBoxStartIndex,
760
+ textBoxEndIndex,
761
+ verticalTextBoxStartIndex,
762
+ verticalTextBoxEndIndex,
763
+ iconBoxStartIndex,
764
+ iconBoxEndIndex,
765
+ verticalIconBoxStartIndex,
766
+ verticalIconBoxEndIndex,
767
+ featureIndex,
768
+ numHorizontalGlyphVertices,
769
+ numVerticalGlyphVertices,
770
+ numIconVertices,
771
+ numVerticalIconVertices,
772
+ useRuntimeCollisionCircles,
773
+ 0,
774
+ textBoxScale,
775
+ textOffset0,
776
+ textOffset1,
777
+ collisionCircleDiameter);
778
+ }
779
+
780
+ function anchorIsTooClose(bucket: any, text: string, repeatDistance: number, anchor: Point) {
781
+ const compareText = bucket.compareText;
782
+ if (!(text in compareText)) {
783
+ compareText[text] = [];
784
+ } else {
785
+ const otherAnchors = compareText[text];
786
+ for (let k = otherAnchors.length - 1; k >= 0; k--) {
787
+ if (anchor.dist(otherAnchors[k]) < repeatDistance) {
788
+ // If it's within repeatDistance of one anchor, stop looking
789
+ return true;
790
+ }
791
+ }
792
+ }
793
+ // If anchor is not within repeatDistance of any other anchor, add to array
794
+ compareText[text].push(anchor);
795
+ return false;
796
+ }