using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Reflection; using UnityEngine; using UnityEngine.Tilemaps; using Object = UnityEngine.Object; namespace UnityEditor.Tilemaps { internal static class TileDragAndDrop { private enum UserTileCreationMode { Overwrite, CreateUnique, Reuse, } private static readonly string k_TileExtension = "asset"; private static List GetSpritesFromTexture(Texture2D texture) { string path = AssetDatabase.GetAssetPath(texture); Object[] assets = AssetDatabase.LoadAllAssetsAtPath(path); List sprites = new List(); foreach (Object asset in assets) { if (asset is Sprite) { sprites.Add(asset as Sprite); } } return sprites; } private static bool AllSpritesAreSameSizeOrMultiples(List sprites) { if (sprites.Count == 0) return false; if (sprites.Count == 1) return true; var size = new Vector2(sprites[0].rect.width, sprites[0].rect.height); for (int i = 1; i < sprites.Count; i++) { var rect = sprites[i].rect; if (rect.width < size.x) size.x = rect.width; if (rect.height < size.y) size.y = rect.height; } foreach (var sprite in sprites) { var rect = sprite.rect; if (rect.width % size.x > 0) return false; if (rect.height % size.y > 0) return false; } return true; } /// /// Converts Objects that can be laid out in the Tile Palette and organises them for placement into a given CellLayout /// /// Textures containing 2-N equal sized Sprites /// All the leftover Sprites that were in same texture but different sizes or just dragged in as Sprite /// Just plain tiles /// Cell Layout to place objects on /// Dictionary mapping the positions of the Objects on the Grid Layout with details of how to place the Objects public static Dictionary CreateHoverData(List sheetTextures, List singleSprites, List tiles, GridLayout.CellLayout cellLayout) { Dictionary result = new Dictionary(); Vector2Int currentPosition = new Vector2Int(0, 0); int width = 0; if (sheetTextures != null) { foreach (Texture2D sheetTexture in sheetTextures) { Dictionary sheet = CreateHoverData(sheetTexture, cellLayout); foreach (KeyValuePair item in sheet) { result.Add(item.Key + currentPosition, item.Value); } Vector2Int min = GetMinMaxRect(sheet.Keys.ToList()).min; currentPosition += new Vector2Int(0, min.y - 1); } } if (currentPosition.x > 0) currentPosition = new Vector2Int(0, currentPosition.y - 1); if (singleSprites != null) { width = Mathf.RoundToInt(Mathf.Sqrt(singleSprites.Count)); foreach (Sprite sprite in singleSprites) { result.Add(currentPosition, new TileDragAndDropHoverData(sprite)); currentPosition += new Vector2Int(1, 0); if (currentPosition.x >= width) currentPosition = new Vector2Int(0, currentPosition.y - 1); } } if (currentPosition.x > 0) currentPosition = new Vector2Int(0, currentPosition.y - 1); if (tiles != null) { width = Math.Max(Mathf.RoundToInt(Mathf.Sqrt(tiles.Count)), width); foreach (TileBase tile in tiles) { result.Add(currentPosition, new TileDragAndDropHoverData(tile)); currentPosition += new Vector2Int(1, 0); if (currentPosition.x >= width) currentPosition = new Vector2Int(0, currentPosition.y - 1); } } return result; } // Get all textures that are valid spritesheets. More than one Sprites and all equal size. public static List GetValidSpritesheets(Object[] objects) { List result = new List(); foreach (Object obj in objects) { if (obj is Texture2D) { Texture2D texture = obj as Texture2D; List sprites = GetSpritesFromTexture(texture); if (sprites.Count > 1 && AllSpritesAreSameSizeOrMultiples(sprites)) { result.Add(texture); } } } return result; } // Get all single Sprite(s) and all Sprite(s) that are part of Texture2D that is not valid sheet (it sprites of varying sizes) public static List GetValidSingleSprites(Object[] objects) { List result = new List(); foreach (Object obj in objects) { if (obj is Sprite) { result.Add(obj as Sprite); } else if (obj is Texture2D) { Texture2D texture = obj as Texture2D; List sprites = GetSpritesFromTexture(texture); if (sprites.Count == 1 || !AllSpritesAreSameSizeOrMultiples(sprites)) { result.AddRange(sprites); } } } return result; } public static List GetValidTiles(Object[] objects) { List result = new List(); foreach (Object obj in objects) { if (obj is TileBase) { result.Add(obj as TileBase); } } return result; } private static Vector2Int GetMinimum(List sprites, Func minX, Func minY) { Vector2 minVector = new Vector2(Int32.MaxValue, Int32.MaxValue); foreach (var sprite in sprites) { minVector.x = Mathf.Min(minVector.x, minX(sprite)); minVector.y = Mathf.Min(minVector.y, minY(sprite)); } return Vector2Int.FloorToInt(minVector); } public static Vector2Int EstimateGridPixelSize(List sprites) { if (sprites.Count == 0 || sprites.Any(sprite => sprite == null)) { return Vector2Int.zero; } if (sprites.Count == 1) return Vector2Int.FloorToInt(sprites[0].rect.size); return GetMinimum(sprites, s => s.rect.width, s => s.rect.height); } public static Vector2Int EstimateGridOffsetSize(List sprites) { if (sprites.Count == 0 || sprites.Any(sprite => sprite == null)) return Vector2Int.zero; if (sprites.Count == 1) return Vector2Int.FloorToInt(sprites[0].rect.position); return GetMinimum(sprites, s => s.rect.xMin, s => s.rect.yMin); } public static Vector2Int EstimateGridPaddingSize(List sprites, Vector2Int cellSize, Vector2Int offsetSize) { if (sprites.Count < 2 || sprites.Any(sprite => sprite == null)) return Vector2Int.zero; var paddingSize = GetMinimum(sprites , (s => { var xMin = s.rect.xMin - cellSize.x - offsetSize.x; return xMin >= 0 ? xMin : Int32.MaxValue; }) , (s => { var yMin = s.rect.yMin - cellSize.y - offsetSize.y; return yMin >= 0 ? yMin : Int32.MaxValue; }) ); // Assume there is no padding if the detected padding is greater than the cell size if (paddingSize.x >= cellSize.x) paddingSize.x = 0; if (paddingSize.y >= cellSize.y) paddingSize.y = 0; return paddingSize; } // Turn texture pixel position into integer grid position based on cell size, offset size and padding private static void GetGridPosition(Sprite sprite, Vector2Int cellPixelSize, Vector2Int offsetSize, Vector2Int paddingSize, out Vector2Int cellPosition, out Vector3 positionOffset) { var spritePosition = sprite.rect.position; var spriteCenter = sprite.rect.center; var position = new Vector2( ((spriteCenter.x - offsetSize.x) / (cellPixelSize.x + paddingSize.x)), (-(sprite.texture.height - spriteCenter.y - offsetSize.y) / (cellPixelSize.y + paddingSize.y)) + 1 ); cellPosition = new Vector2Int(Mathf.FloorToInt(position.x), Mathf.FloorToInt(position.y)); positionOffset = (spriteCenter - spritePosition) / cellPixelSize; positionOffset.x = (float)(positionOffset.x - Math.Truncate(positionOffset.x)); positionOffset.y = (float)(positionOffset.y - Math.Truncate(positionOffset.y)); } // Turn texture pixel position into integer isometric grid position based on cell size and offset size private static void GetIsometricGridPosition(Sprite sprite, Vector2Int cellPixelSize, Vector2Int offsetSize, out Vector2Int cellPosition) { var offsetPosition = new Vector2(sprite.rect.center.x - offsetSize.x, sprite.rect.center.y - offsetSize.y); var cellStride = new Vector2(cellPixelSize.x, cellPixelSize.y) * 0.5f; var invCellStride = new Vector2(1.0f / cellStride.x, 1.0f / cellStride.y); var position = offsetPosition * invCellStride; position.y = (position.y - position.x) * 0.5f; position.x += position.y; cellPosition = new Vector2Int(Mathf.FloorToInt(position.x), Mathf.FloorToInt(position.y)); } // Organizes all the sprites in a single texture nicely on a 2D "table" based on their original texture position // Only call this with spritesheet with all Sprites equal size public static Dictionary CreateHoverData(Texture2D sheet, GridLayout.CellLayout cellLayout) { Dictionary result = new Dictionary(); List sprites = GetSpritesFromTexture(sheet); Vector2Int cellPixelSize = EstimateGridPixelSize(sprites); // Get Offset Vector2Int offsetSize = EstimateGridOffsetSize(sprites); // Get Padding Vector2Int paddingSize = EstimateGridPaddingSize(sprites, cellPixelSize, offsetSize); if ((cellLayout == GridLayout.CellLayout.Isometric || cellLayout == GridLayout.CellLayout.IsometricZAsY) && (HasSpriteRectOverlaps(sprites))) { foreach (Sprite sprite in sprites) { GetIsometricGridPosition(sprite, cellPixelSize, offsetSize, out Vector2Int position); result[position] = new TileDragAndDropHoverData(sprite, Vector3.zero, (Vector2)cellPixelSize / sprite.pixelsPerUnit, false); } } else { foreach (Sprite sprite in sprites) { GetGridPosition(sprite, cellPixelSize, offsetSize, paddingSize, out Vector2Int position, out Vector3 offset); result[position] = new TileDragAndDropHoverData(sprite, offset, (Vector2)cellPixelSize / sprite.pixelsPerUnit); } } return result; } private static bool HasSpriteRectOverlaps(IReadOnlyList sprites) { var count = sprites.Count; for (int i = 0; i < count; i++) { var rect = sprites[i].rect; for (int j = i + 1; j < count; j++) { if (rect.Overlaps(sprites[j].rect)) return true; } } return false; } internal static string GenerateUniqueNameForNamelessSprite(Sprite sprite, HashSet uniqueNames, ref int count) { var baseName = "Nameless"; if (sprite.texture != null) baseName = sprite.texture.name; string name; do { name = $"{baseName}_{count++}"; } while (uniqueNames.Contains(name)); return name; } public static List ConvertToTileSheet(Dictionary sheet) { var result = new List(); var defaultPath = TileDragAndDropManager.GetDefaultTileAssetDirectoryPath(); // Early out if all objects are already tiles if (sheet.Values.ToList().FindAll(data => data.hoverObject is TileBase).Count == sheet.Values.Count) { foreach (var item in sheet.Values) { result.Add(item.hoverObject as TileBase); } return result; } UserTileCreationMode userTileCreationMode = UserTileCreationMode.Overwrite; string path = ""; bool multipleTiles = sheet.Count > 1; int i = 0; HashSet uniqueNames = new HashSet(); if (multipleTiles) { bool userInterventionRequired = false; path = EditorUtility.SaveFolderPanel("Generate tiles into folder ", defaultPath, ""); path = FileUtil.GetProjectRelativePath(path); // Check if this will overwrite any existing assets foreach (var item in sheet.Values) { if (item.hoverObject is Sprite sprite) { var name = sprite.name; if (String.IsNullOrEmpty(name) || uniqueNames.Contains(name)) { name = GenerateUniqueNameForNamelessSprite(sprite, uniqueNames, ref i); } uniqueNames.Add(name); var tilePath = FileUtil.CombinePaths(path, String.Format("{0}.{1}", name, k_TileExtension)); if (File.Exists(tilePath)) { userInterventionRequired = true; break; } } } // There are existing tile assets in the folder with names matching the items to be created if (userInterventionRequired) { var option = EditorUtility.DisplayDialogComplex("Overwrite?", String.Format("Assets exist at {0}. Do you wish to overwrite existing assets?", path), "Overwrite", "Create New Copy", "Reuse"); switch (option) { case 0: // Overwrite { userTileCreationMode = UserTileCreationMode.Overwrite; } break; case 1: // Create New Copy { userTileCreationMode = UserTileCreationMode.CreateUnique; } break; case 2: // Reuse { userTileCreationMode = UserTileCreationMode.Reuse; } break; } } } else { // Do not check if this will overwrite new tile as user has explicitly selected the file to save to path = EditorUtility.SaveFilePanelInProject("Generate new tile", sheet.Values.First().hoverObject.name, k_TileExtension, "Generate new tile", defaultPath); } TileDragAndDropManager.SetUserTileAssetDirectoryPath(path); if (string.IsNullOrEmpty(path)) return result; i = 0; uniqueNames.Clear(); EditorUtility.DisplayProgressBar("Generating Tile Assets (" + i + "/" + sheet.Count + ")", "Generating tiles", 0f); AssetDatabase.StartAssetEditing(); try { MethodInfo createTileMethod = GridPaintActiveTargetsPreferences.GetCreateTileFromPaletteUsingPreferences(); if (createTileMethod == null) return null; foreach (KeyValuePair item in sheet) { TileBase tile; string tilePath = ""; if (item.Value.hoverObject is Sprite sprite) { tile = createTileMethod.Invoke(null, new object[] {sprite}) as TileBase; if (tile == null) continue; var name = tile.name; if (String.IsNullOrEmpty(name) || uniqueNames.Contains(name)) { name = GenerateUniqueNameForNamelessSprite(sprite, uniqueNames, ref i); } uniqueNames.Add(name); tilePath = multipleTiles ? FileUtil.CombinePaths(path, String.Format("{0}.{1}", name, k_TileExtension)) : path; // Case 1216101: Fix path slashes for Windows tilePath = FileUtil.NiceWinPath(tilePath); switch (userTileCreationMode) { case UserTileCreationMode.CreateUnique: { if (File.Exists(tilePath)) tilePath = AssetDatabase.GenerateUniqueAssetPath(tilePath); AssetDatabase.CreateAsset(tile, tilePath); } break; case UserTileCreationMode.Overwrite: { AssetDatabase.CreateAsset(tile, tilePath); } break; case UserTileCreationMode.Reuse: { if (File.Exists(tilePath)) tile = AssetDatabase.LoadAssetAtPath(tilePath); else AssetDatabase.CreateAsset(tile, tilePath); } break; } } else { tile = item.Value.hoverObject as TileBase; } EditorUtility.DisplayProgressBar("Generating Tile Assets (" + i + "/" + sheet.Count + ")", "Generating " + tilePath, (float)i++ / sheet.Count); result.Add(tile); } } finally { AssetDatabase.StopAssetEditing(); EditorUtility.ClearProgressBar(); } AssetDatabase.Refresh(); return result; } internal static RectInt GetMinMaxRect(IEnumerable positions) { if (positions == null) return new RectInt(); var hasValue = false; var min = new Vector2Int(Int32.MaxValue, Int32.MaxValue); var max = new Vector2Int(Int32.MinValue, Int32.MinValue); foreach (var position in positions) { min.x = Math.Min(min.x, position.x); max.x = Math.Max(max.x, position.x); min.y = Math.Min(min.y, position.y); max.y = Math.Max(max.y, position.y); hasValue = true; } return hasValue ? GridEditorUtility.GetMarqueeRect(min, max) : new RectInt(); } } }