You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					201 lines
				
				5.8 KiB
			
		
		
			
		
	
	
					201 lines
				
				5.8 KiB
			| 
											2 years ago
										 | using System;
 | ||
|  | using System.Collections.Generic;
 | ||
|  | using Mapbox.Map;
 | ||
|  | using Mapbox.Unity.Utilities;
 | ||
|  | using Mapbox.Utils;
 | ||
|  | using UnityEngine;
 | ||
|  | 
 | ||
|  | namespace Mapbox.Unity.Map.TileProviders
 | ||
|  | {
 | ||
|  | 	public class QuadTreeTileProvider : AbstractTileProvider
 | ||
|  | 	{
 | ||
|  | 		private static readonly int HIT_POINTS_COUNT = 4;
 | ||
|  | 
 | ||
|  | 		private Plane _groundPlane;
 | ||
|  | 		private bool _shouldUpdate;
 | ||
|  | 		[SerializeField] private CameraBoundsTileProviderOptions _cbtpOptions;
 | ||
|  | 
 | ||
|  | 		private Vector2dBounds _viewPortWebMercBounds;
 | ||
|  | 
 | ||
|  | 		#region Tile decision and raycasting fields
 | ||
|  | 		private HashSet<UnwrappedTileId> _tiles;
 | ||
|  | 		private HashSet<CanonicalTileId> _canonicalTiles;
 | ||
|  | 
 | ||
|  | 		private Ray _ray00;
 | ||
|  | 		private Ray _ray01;
 | ||
|  | 		private Ray _ray10;
 | ||
|  | 		private Ray _ray11;
 | ||
|  | 		private Vector3[] _hitPnt = new Vector3[HIT_POINTS_COUNT];
 | ||
|  | 		private Vector2d[] _hitPntGeoPos = new Vector2d[HIT_POINTS_COUNT];
 | ||
|  | 		private bool _isFirstLoad;
 | ||
|  | 		#endregion
 | ||
|  | 
 | ||
|  | 		public override void OnInitialized()
 | ||
|  | 		{
 | ||
|  | 			_tiles = new HashSet<UnwrappedTileId>();
 | ||
|  | 			_canonicalTiles = new HashSet<CanonicalTileId>();
 | ||
|  | 			if (Options != null)
 | ||
|  | 			{
 | ||
|  | 				_cbtpOptions = (CameraBoundsTileProviderOptions)_options;
 | ||
|  | 			}
 | ||
|  | 			else
 | ||
|  | 			{
 | ||
|  | 				_cbtpOptions = new CameraBoundsTileProviderOptions();
 | ||
|  | 			}
 | ||
|  | 
 | ||
|  | 			if (_cbtpOptions.camera == null)
 | ||
|  | 			{
 | ||
|  | 				_cbtpOptions.camera = Camera.main;
 | ||
|  | 			}
 | ||
|  | 			_cbtpOptions.camera.transform.hasChanged = false;
 | ||
|  | 			_groundPlane = new Plane(Vector3.up, 0);
 | ||
|  | 			_shouldUpdate = true;
 | ||
|  | 			_currentExtent.activeTiles = new HashSet<UnwrappedTileId>();
 | ||
|  | 		}
 | ||
|  | 
 | ||
|  | 		public override void UpdateTileExtent()
 | ||
|  | 		{
 | ||
|  | 			if (!_shouldUpdate)
 | ||
|  | 			{
 | ||
|  | 				return;
 | ||
|  | 			}
 | ||
|  | 
 | ||
|  | 			//update viewport in case it was changed by switching zoom level
 | ||
|  | 			_viewPortWebMercBounds = getcurrentViewPortWebMerc();
 | ||
|  | 			_currentExtent.activeTiles = GetWithWebMerc(_viewPortWebMercBounds, _map.AbsoluteZoom);
 | ||
|  | 
 | ||
|  | 			OnExtentChanged();
 | ||
|  | 		}
 | ||
|  | 
 | ||
|  | 		public HashSet<UnwrappedTileId> GetWithWebMerc(Vector2dBounds bounds, int zoom)
 | ||
|  | 		{
 | ||
|  | 			_tiles.Clear();
 | ||
|  | 			_canonicalTiles.Clear();
 | ||
|  | 
 | ||
|  | 			if (bounds.IsEmpty()) { return _tiles; }
 | ||
|  | 
 | ||
|  | 			//stay within WebMerc bounds
 | ||
|  | 			Vector2d swWebMerc = new Vector2d(Math.Max(bounds.SouthWest.x, -Utils.Constants.WebMercMax), Math.Max(bounds.SouthWest.y, -Utils.Constants.WebMercMax));
 | ||
|  | 			Vector2d neWebMerc = new Vector2d(Math.Min(bounds.NorthEast.x, Utils.Constants.WebMercMax), Math.Min(bounds.NorthEast.y, Utils.Constants.WebMercMax));
 | ||
|  | 
 | ||
|  | 			UnwrappedTileId swTile = WebMercatorToTileId(swWebMerc, zoom);
 | ||
|  | 			UnwrappedTileId neTile = WebMercatorToTileId(neWebMerc, zoom);
 | ||
|  | 
 | ||
|  | 			for (int x = swTile.X; x <= neTile.X; x++)
 | ||
|  | 			{
 | ||
|  | 				for (int y = neTile.Y; y <= swTile.Y; y++)
 | ||
|  | 				{
 | ||
|  | 					UnwrappedTileId uwtid = new UnwrappedTileId(zoom, x, y);
 | ||
|  | 					//hack: currently too many tiles are created at lower zoom levels
 | ||
|  | 					//investigate formulas, this worked before
 | ||
|  | 					if (!_canonicalTiles.Contains(uwtid.Canonical))
 | ||
|  | 					{
 | ||
|  | 						_tiles.Add(uwtid);
 | ||
|  | 						_canonicalTiles.Add(uwtid.Canonical);
 | ||
|  | 					}
 | ||
|  | 				}
 | ||
|  | 			}
 | ||
|  | 
 | ||
|  | 			return _tiles;
 | ||
|  | 		}
 | ||
|  | 
 | ||
|  | 		public UnwrappedTileId WebMercatorToTileId(Vector2d webMerc, int zoom)
 | ||
|  | 		{
 | ||
|  | 			var tileCount = Math.Pow(2, zoom);
 | ||
|  | 
 | ||
|  | 			var dblX = webMerc.x / Utils.Constants.WebMercMax;
 | ||
|  | 			var dblY = webMerc.y / Utils.Constants.WebMercMax;
 | ||
|  | 
 | ||
|  | 			int x = (int)Math.Floor((1 + dblX) / 2 * tileCount);
 | ||
|  | 			int y = (int)Math.Floor((1 - dblY) / 2 * tileCount);
 | ||
|  | 			return new UnwrappedTileId(zoom, x, y);
 | ||
|  | 		}
 | ||
|  | 		private Vector2dBounds getcurrentViewPortWebMerc(bool useGroundPlane = true)
 | ||
|  | 		{
 | ||
|  | 			if (useGroundPlane)
 | ||
|  | 			{
 | ||
|  | 				// rays from camera to groundplane: lower left and upper right
 | ||
|  | 				_ray00 = _cbtpOptions.camera.ViewportPointToRay(new Vector3(0, 0));
 | ||
|  | 				_ray01 = _cbtpOptions.camera.ViewportPointToRay(new Vector3(0, 1));
 | ||
|  | 				_ray10 = _cbtpOptions.camera.ViewportPointToRay(new Vector3(1, 0));
 | ||
|  | 				_ray11 = _cbtpOptions.camera.ViewportPointToRay(new Vector3(1, 1));
 | ||
|  | 				_hitPnt[0] = getGroundPlaneHitPoint(_ray00);
 | ||
|  | 				_hitPnt[1] = getGroundPlaneHitPoint(_ray01);
 | ||
|  | 				_hitPnt[2] = getGroundPlaneHitPoint(_ray10);
 | ||
|  | 				_hitPnt[3] = getGroundPlaneHitPoint(_ray11);
 | ||
|  | 			}
 | ||
|  | 
 | ||
|  | 			// Find min max bounding box.
 | ||
|  | 			// TODO : Find a better way of doing this.
 | ||
|  | 			double minLat = double.MaxValue;
 | ||
|  | 			double minLong = double.MaxValue;
 | ||
|  | 			double maxLat = double.MinValue;
 | ||
|  | 			double maxLong = double.MinValue;
 | ||
|  | 
 | ||
|  | 			for (int pointIndex = 0; pointIndex < HIT_POINTS_COUNT; ++pointIndex)
 | ||
|  | 			{
 | ||
|  | 				_hitPntGeoPos[pointIndex] = _map.WorldToGeoPosition(_hitPnt[pointIndex]);
 | ||
|  | 			}
 | ||
|  | 
 | ||
|  | 			for (int i = 0; i < HIT_POINTS_COUNT; i++)
 | ||
|  | 			{
 | ||
|  | 				if (_hitPnt[i] == Vector3.zero)
 | ||
|  | 				{
 | ||
|  | 					continue;
 | ||
|  | 				}
 | ||
|  | 				else
 | ||
|  | 				{
 | ||
|  | 					if (minLat > _hitPntGeoPos[i].x)
 | ||
|  | 					{
 | ||
|  | 						minLat = _hitPntGeoPos[i].x;
 | ||
|  | 					}
 | ||
|  | 
 | ||
|  | 					if (minLong > _hitPntGeoPos[i].y)
 | ||
|  | 					{
 | ||
|  | 						minLong = _hitPntGeoPos[i].y;
 | ||
|  | 					}
 | ||
|  | 
 | ||
|  | 					if (maxLat < _hitPntGeoPos[i].x)
 | ||
|  | 					{
 | ||
|  | 						maxLat = _hitPntGeoPos[i].x;
 | ||
|  | 					}
 | ||
|  | 
 | ||
|  | 					if (maxLong < _hitPntGeoPos[i].y)
 | ||
|  | 					{
 | ||
|  | 						maxLong = _hitPntGeoPos[i].y;
 | ||
|  | 					}
 | ||
|  | 				}
 | ||
|  | 			}
 | ||
|  | 
 | ||
|  | 			Vector2d hitPntSWGeoPos = new Vector2d(minLat, minLong);
 | ||
|  | 			Vector2d hitPntNEGeoPos = new Vector2d(maxLat, maxLong);
 | ||
|  | 			Vector2dBounds tileBounds = new Vector2dBounds(Conversions.LatLonToMeters(hitPntSWGeoPos), Conversions.LatLonToMeters(hitPntNEGeoPos));         // Bounds debugging.
 | ||
|  | #if UNITY_EDITOR
 | ||
|  | 			Debug.DrawLine(_cbtpOptions.camera.transform.position, _map.GeoToWorldPosition(hitPntSWGeoPos), Color.blue);
 | ||
|  | 			Debug.DrawLine(_cbtpOptions.camera.transform.position, _map.GeoToWorldPosition(hitPntNEGeoPos), Color.red);
 | ||
|  | #endif
 | ||
|  | 			return tileBounds;
 | ||
|  | 		}
 | ||
|  | 		private Vector3 getGroundPlaneHitPoint(Ray ray)
 | ||
|  | 		{
 | ||
|  | 			float distance;
 | ||
|  | 			if (!_groundPlane.Raycast(ray, out distance)) { return Vector3.zero; }
 | ||
|  | 			return ray.GetPoint(distance);
 | ||
|  | 		}
 | ||
|  | 
 | ||
|  | 		public override void UpdateTileProvider()
 | ||
|  | 		{
 | ||
|  | 			if (_cbtpOptions != null && _cbtpOptions.camera != null && _cbtpOptions.camera.transform.hasChanged)
 | ||
|  | 			{
 | ||
|  | 				UpdateTileExtent();
 | ||
|  | 				_cbtpOptions.camera.transform.hasChanged = false;
 | ||
|  | 			}
 | ||
|  | 		}
 | ||
|  | 
 | ||
|  | 		public override bool Cleanup(UnwrappedTileId tile)
 | ||
|  | 		{
 | ||
|  | 			return (!_currentExtent.activeTiles.Contains(tile));
 | ||
|  | 		}
 | ||
|  | 	}
 | ||
|  | }
 |