wuxianshengcong/Library/PackageCache/com.unity.2d.tilemap@1.0.0/Editor/TileDragAndDrop.cs
2025-01-02 14:50:41 +08:00

510 lines
21 KiB
C#

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<Sprite> GetSpritesFromTexture(Texture2D texture)
{
string path = AssetDatabase.GetAssetPath(texture);
Object[] assets = AssetDatabase.LoadAllAssetsAtPath(path);
List<Sprite> sprites = new List<Sprite>();
foreach (Object asset in assets)
{
if (asset is Sprite)
{
sprites.Add(asset as Sprite);
}
}
return sprites;
}
private static bool AllSpritesAreSameSizeOrMultiples(List<Sprite> 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;
}
/// <summary>
/// Converts Objects that can be laid out in the Tile Palette and organises them for placement into a given CellLayout
/// </summary>
/// <param name="sheetTextures">Textures containing 2-N equal sized Sprites</param>
/// <param name="singleSprites">All the leftover Sprites that were in same texture but different sizes or just dragged in as Sprite</param>
/// <param name="tiles">Just plain tiles</param>
/// <param name="cellLayout">Cell Layout to place objects on</param>
/// <returns>Dictionary mapping the positions of the Objects on the Grid Layout with details of how to place the Objects</returns>
public static Dictionary<Vector2Int, TileDragAndDropHoverData> CreateHoverData(List<Texture2D> sheetTextures, List<Sprite> singleSprites, List<TileBase> tiles, GridLayout.CellLayout cellLayout)
{
Dictionary<Vector2Int, TileDragAndDropHoverData> result = new Dictionary<Vector2Int, TileDragAndDropHoverData>();
Vector2Int currentPosition = new Vector2Int(0, 0);
int width = 0;
if (sheetTextures != null)
{
foreach (Texture2D sheetTexture in sheetTextures)
{
Dictionary<Vector2Int, TileDragAndDropHoverData> sheet = CreateHoverData(sheetTexture, cellLayout);
foreach (KeyValuePair<Vector2Int, TileDragAndDropHoverData> 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<Texture2D> GetValidSpritesheets(Object[] objects)
{
List<Texture2D> result = new List<Texture2D>();
foreach (Object obj in objects)
{
if (obj is Texture2D)
{
Texture2D texture = obj as Texture2D;
List<Sprite> 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<Sprite> GetValidSingleSprites(Object[] objects)
{
List<Sprite> result = new List<Sprite>();
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<Sprite> sprites = GetSpritesFromTexture(texture);
if (sprites.Count == 1 || !AllSpritesAreSameSizeOrMultiples(sprites))
{
result.AddRange(sprites);
}
}
}
return result;
}
public static List<TileBase> GetValidTiles(Object[] objects)
{
List<TileBase> result = new List<TileBase>();
foreach (Object obj in objects)
{
if (obj is TileBase)
{
result.Add(obj as TileBase);
}
}
return result;
}
private static Vector2Int GetMinimum(List<Sprite> sprites, Func<Sprite, float> minX, Func<Sprite, float> 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<Sprite> 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<Sprite> 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<Sprite> 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<Vector2Int, TileDragAndDropHoverData> CreateHoverData(Texture2D sheet, GridLayout.CellLayout cellLayout)
{
Dictionary<Vector2Int, TileDragAndDropHoverData> result = new Dictionary<Vector2Int, TileDragAndDropHoverData>();
List<Sprite> 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<Sprite> 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<string> 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<TileBase> ConvertToTileSheet(Dictionary<Vector2Int, TileDragAndDropHoverData> sheet)
{
var result = new List<TileBase>();
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<String> uniqueNames = new HashSet<string>();
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<Vector2Int, TileDragAndDropHoverData> 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<TileBase>(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<Vector2Int> 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();
}
}
}