using System;
using System.Linq;
using UnityEditor.SceneManagement;
using UnityEngine;
using UnityEngine.SceneManagement;
using Object = UnityEngine.Object;
namespace UnityEditor.Tilemaps
{
///
/// This Brush instances, places and manipulates GameObjects onto the scene.
/// Use this as an example to create brushes which targets objects other than tiles for manipulation.
///
[HelpURL("https://docs.unity3d.com/Packages/com.unity.2d.tilemap.extras@latest/index.html?subfolder=/manual/GameObjectBrush.html")]
[CustomGridBrush(true, false, false, "GameObject Brush")]
public class GameObjectBrush : GridBrushBase
{
[Serializable]
internal class HiddenGridLayout
{
public Vector3 cellSize = Vector3.one;
public Vector3 cellGap = Vector3.zero;
public GridLayout.CellLayout cellLayout = GridLayout.CellLayout.Rectangle;
public GridLayout.CellSwizzle cellSwizzle = GridLayout.CellSwizzle.XYZ;
}
[SerializeField]
private BrushCell[] m_Cells;
[SerializeField]
private Vector3Int m_Size;
[SerializeField]
private Vector3Int m_Pivot;
[SerializeField]
[HideInInspector]
private bool m_CanChangeZPosition;
[SerializeField]
[HideInInspector]
internal HiddenGridLayout hiddenGridLayout = new HiddenGridLayout();
///
/// GameObject used for painting onto the Scene root
///
[HideInInspector]
public GameObject hiddenGrid;
///
/// Anchor Point of the Instantiated GameObject in the cell when painting
///
public Vector3 m_Anchor = new Vector3(0.5f, 0.5f, 0.5f);
/// Size of the brush in cells.
public Vector3Int size { get { return m_Size; } set { m_Size = value; SizeUpdated(); } }
/// Pivot of the brush.
public Vector3Int pivot { get { return m_Pivot; } set { m_Pivot = value; } }
/// All the brush cells the brush holds.
public BrushCell[] cells { get { return m_Cells; } }
/// Number of brush cells in the brush.
public int cellCount { get { return m_Cells != null ? m_Cells.Length : 0; } }
/// Number of brush cells based on size.
public int sizeCount
{
get { return m_Size.x * m_Size.y * m_Size.z; }
}
/// Whether the brush can change Z Position
public bool canChangeZPosition
{
get { return m_CanChangeZPosition; }
set { m_CanChangeZPosition = value; }
}
///
/// This Brush instances, places and manipulates GameObjects onto the scene.
///
public GameObjectBrush()
{
Init(Vector3Int.one, Vector3Int.zero);
SizeUpdated();
}
private void OnEnable()
{
hiddenGrid = new GameObject();
hiddenGrid.name = "(Paint on SceneRoot)";
hiddenGrid.hideFlags = HideFlags.HideAndDontSave;
hiddenGrid.transform.position = Vector3.zero;
var grid = hiddenGrid.AddComponent();
grid.cellSize = hiddenGridLayout.cellSize;
grid.cellGap = hiddenGridLayout.cellGap;
grid.cellSwizzle = hiddenGridLayout.cellSwizzle;
grid.cellLayout = hiddenGridLayout.cellLayout;
}
private void OnDisable()
{
DestroyImmediate(hiddenGrid);
}
///
/// Initializes the content of the GameObjectBrush.
///
/// Size of the GameObjectBrush.
public void Init(Vector3Int size)
{
Init(size, Vector3Int.zero);
SizeUpdated();
}
/// Initializes the content of the GameObjectBrush.
/// Size of the GameObjectBrush.
/// Pivot point of the GameObjectBrush.
public void Init(Vector3Int size, Vector3Int pivot)
{
m_Size = size;
m_Pivot = pivot;
SizeUpdated();
}
///
/// Paints GameObjects into a given position within the selected layers.
/// The GameObjectBrush overrides this to provide GameObject painting functionality.
///
/// Grid used for layout.
/// Target of the paint operation. By default the currently selected GameObject.
/// The coordinates of the cell to paint data to.
public override void Paint(GridLayout gridLayout, GameObject brushTarget, Vector3Int position)
{
Vector3Int min = position - pivot;
BoundsInt bounds = new BoundsInt(min, m_Size);
BoxFill(gridLayout, brushTarget, bounds);
}
private void PaintCell(GridLayout grid, Vector3Int position, Transform parent, BrushCell cell)
{
if (cell.gameObject == null)
return;
var existingGO = GetObjectInCell(grid, parent, position);
if (existingGO == null)
{
SetSceneCell(grid, parent, position, cell.gameObject, cell.offset, cell.scale, cell.orientation, m_Anchor);
}
}
///
/// Erases GameObjects in a given position within the selected layers.
/// The GameObjectBrush overrides this to provide GameObject erasing functionality.
///
/// Grid used for layout.
/// Target of the erase operation. By default the currently selected GameObject.
/// The coordinates of the cell to erase data from.
public override void Erase(GridLayout gridLayout, GameObject brushTarget, Vector3Int position)
{
Vector3Int min = position - pivot;
BoundsInt bounds = new BoundsInt(min, m_Size);
GetGrid(ref gridLayout, ref brushTarget);
BoxErase(gridLayout, brushTarget, bounds);
}
private void EraseCell(GridLayout grid, Vector3Int position, Transform parent)
{
ClearSceneCell(grid, parent, position);
}
///
/// Box fills GameObjects into given bounds within the selected layers.
/// The GameObjectBrush overrides this to provide GameObject box-filling functionality.
///
/// Grid to box fill data to.
/// Target of the box fill operation. By default the currently selected GameObject.
/// The bounds to box fill data into.
public override void BoxFill(GridLayout gridLayout, GameObject brushTarget, BoundsInt position)
{
GetGrid(ref gridLayout, ref brushTarget);
foreach (Vector3Int location in position.allPositionsWithin)
{
Vector3Int local = location - position.min;
BrushCell cell = m_Cells[GetCellIndexWrapAround(local.x, local.y, local.z)];
PaintCell(gridLayout, location, brushTarget != null ? brushTarget.transform : null, cell);
}
}
///
/// Erases GameObjects from given bounds within the selected layers.
/// The GameObjectBrush overrides this to provide GameObject box-erasing functionality.
///
/// Grid to erase data from.
/// Target of the erase operation. By default the currently selected GameObject.
/// The bounds to erase data from.
public override void BoxErase(GridLayout gridLayout, GameObject brushTarget, BoundsInt position)
{
GetGrid(ref gridLayout, ref brushTarget);
foreach (Vector3Int location in position.allPositionsWithin)
{
EraseCell(gridLayout, location, brushTarget != null ? brushTarget.transform : null);
}
}
///
/// This is not supported but it should floodfill GameObjects starting from a given position within the selected layers.
///
/// Grid used for layout.
/// Target of the flood fill operation. By default the currently selected GameObject.
/// Starting position of the flood fill.
public override void FloodFill(GridLayout gridLayout, GameObject brushTarget, Vector3Int position)
{
Debug.LogWarning("FloodFill not supported");
}
///
/// Rotates the brush by 90 degrees in the given direction.
///
/// Direction to rotate by.
/// Cell Layout for rotating.
public override void Rotate(RotationDirection direction, GridLayout.CellLayout layout)
{
Vector3Int oldSize = m_Size;
BrushCell[] oldCells = m_Cells.Clone() as BrushCell[];
size = new Vector3Int(oldSize.y, oldSize.x, oldSize.z);
BoundsInt oldBounds = new BoundsInt(Vector3Int.zero, oldSize);
foreach (Vector3Int oldPos in oldBounds.allPositionsWithin)
{
int newX = direction == RotationDirection.Clockwise ? oldSize.y - oldPos.y - 1 : oldPos.y;
int newY = direction == RotationDirection.Clockwise ? oldPos.x : oldSize.x - oldPos.x - 1;
int toIndex = GetCellIndex(newX, newY, oldPos.z);
int fromIndex = GetCellIndex(oldPos.x, oldPos.y, oldPos.z, oldSize.x, oldSize.y, oldSize.z);
m_Cells[toIndex] = oldCells[fromIndex];
}
int newPivotX = direction == RotationDirection.Clockwise ? oldSize.y - pivot.y - 1 : pivot.y;
int newPivotY = direction == RotationDirection.Clockwise ? pivot.x : oldSize.x - pivot.x - 1;
pivot = new Vector3Int(newPivotX, newPivotY, pivot.z);
Matrix4x4 rotation = Matrix4x4.TRS(Vector3.zero, Quaternion.Euler(0f, 0f, direction == RotationDirection.Clockwise ? 90f : -90f), Vector3.one);
Quaternion orientation = Quaternion.Euler(0f, 0f, direction == RotationDirection.Clockwise ? 90f : -90f);
foreach (BrushCell cell in m_Cells)
{
cell.offset = rotation * cell.offset;
cell.orientation = cell.orientation * orientation;
}
}
/// Flips the brush in the given axis.
/// Axis to flip by.
/// Cell Layout for flipping.
public override void Flip(FlipAxis flip, GridLayout.CellLayout layout)
{
if (flip == FlipAxis.X)
FlipX();
else
FlipY();
}
///
/// Picks child GameObjects given the coordinates of the cells.
/// The GameObjectBrush overrides this to provide GameObject picking functionality.
///
/// Grid to pick data from.
/// Target of the picking operation. By default the currently selected GameObject.
/// The coordinates of the cells to paint data from.
/// Pivot of the picking brush.
public override void Pick(GridLayout gridLayout, GameObject brushTarget, BoundsInt position, Vector3Int pivot)
{
Reset();
UpdateSizeAndPivot(new Vector3Int(position.size.x, position.size.y, 1), new Vector3Int(pivot.x, pivot.y, 0));
GetGrid(ref gridLayout, ref brushTarget);
foreach (Vector3Int pos in position.allPositionsWithin)
{
Vector3Int brushPosition = new Vector3Int(pos.x - position.x, pos.y - position.y, 0);
PickCell(pos, brushPosition, gridLayout, brushTarget != null ? brushTarget.transform : null);
}
}
private void PickCell(Vector3Int position, Vector3Int brushPosition, GridLayout grid, Transform parent)
{
Vector3 cellCenter = grid.LocalToWorld(grid.CellToLocalInterpolated(position + m_Anchor));
GameObject go = GetObjectInCell(grid, parent, position);
if (go != null)
{
Object prefab = PrefabUtility.GetCorrespondingObjectFromSource(go);
if (prefab)
{
SetGameObject(brushPosition, (GameObject) prefab);
}
else
{
GameObject newInstance = Instantiate(go);
newInstance.hideFlags = HideFlags.HideAndDontSave;
newInstance.SetActive(false);
SetGameObject(brushPosition, newInstance);
}
SetOffset(brushPosition, go.transform.position - cellCenter);
SetScale(brushPosition, go.transform.localScale);
SetOrientation(brushPosition, go.transform.localRotation);
}
}
///
/// MoveStart is called when user starts moving the area previously selected with the selection marquee.
/// The GameObjectBrush overrides this to provide GameObject moving functionality.
///
/// Grid used for layout.
/// Target of the move operation. By default the currently selected GameObject.
/// Position where the move operation has started.
public override void MoveStart(GridLayout gridLayout, GameObject brushTarget, BoundsInt position)
{
Reset();
UpdateSizeAndPivot(new Vector3Int(position.size.x, position.size.y, 1), Vector3Int.zero);
GetGrid(ref gridLayout, ref brushTarget);
var targetTransform = brushTarget != null ? brushTarget.transform : null;
foreach (Vector3Int pos in position.allPositionsWithin)
{
Vector3Int brushPosition = new Vector3Int(pos.x - position.x, pos.y - position.y, 0);
PickCell(pos, brushPosition, gridLayout, targetTransform);
ClearSceneCell(gridLayout, targetTransform, pos);
}
}
///
/// MoveEnd is called when user has ended the move of the area previously selected with the selection marquee.
/// The GameObjectBrush overrides this to provide GameObject moving functionality.
///
/// Grid used for layout.
/// Target of the move operation. By default the currently selected GameObject.
/// Position where the move operation has ended.
public override void MoveEnd(GridLayout gridLayout, GameObject brushTarget, BoundsInt position)
{
GetGrid(ref gridLayout, ref brushTarget);
Paint(gridLayout, brushTarget, position.min);
Reset();
}
private void GetGrid(ref GridLayout gridLayout, ref GameObject brushTarget)
{
if (brushTarget == hiddenGrid)
brushTarget = null;
if (brushTarget != null)
{
var targetGridLayout = brushTarget.GetComponent();
if (targetGridLayout != null)
gridLayout = targetGridLayout;
}
}
/// Clears all data of the brush.
public void Reset()
{
foreach (var cell in m_Cells)
{
if (cell.gameObject != null && !EditorUtility.IsPersistent(cell.gameObject))
{
DestroyImmediate(cell.gameObject);
}
cell.gameObject = null;
}
UpdateSizeAndPivot(Vector3Int.one, Vector3Int.zero);
}
private void FlipX()
{
BrushCell[] oldCells = m_Cells.Clone() as BrushCell[];
BoundsInt oldBounds = new BoundsInt(Vector3Int.zero, m_Size);
foreach (Vector3Int oldPos in oldBounds.allPositionsWithin)
{
int newX = m_Size.x - oldPos.x - 1;
int toIndex = GetCellIndex(newX, oldPos.y, oldPos.z);
int fromIndex = GetCellIndex(oldPos);
m_Cells[toIndex] = oldCells[fromIndex];
}
int newPivotX = m_Size.x - pivot.x - 1;
pivot = new Vector3Int(newPivotX, pivot.y, pivot.z);
Matrix4x4 flip = Matrix4x4.TRS(Vector3.zero, Quaternion.identity, new Vector3(-1f, 1f, 1f));
Quaternion orientation = Quaternion.Euler(0f, 0f, -180f);
foreach (BrushCell cell in m_Cells)
{
Vector3 oldOffset = cell.offset;
cell.offset = flip * oldOffset;
cell.orientation = cell.orientation*orientation;
}
}
private void FlipY()
{
BrushCell[] oldCells = m_Cells.Clone() as BrushCell[];
BoundsInt oldBounds = new BoundsInt(Vector3Int.zero, m_Size);
foreach (Vector3Int oldPos in oldBounds.allPositionsWithin)
{
int newY = m_Size.y - oldPos.y - 1;
int toIndex = GetCellIndex(oldPos.x, newY, oldPos.z);
int fromIndex = GetCellIndex(oldPos);
m_Cells[toIndex] = oldCells[fromIndex];
}
int newPivotY = m_Size.y - pivot.y - 1;
pivot = new Vector3Int(pivot.x, newPivotY, pivot.z);
Matrix4x4 flip = Matrix4x4.TRS(Vector3.zero, Quaternion.identity, new Vector3(1f, -1f, 1f));
Quaternion orientation = Quaternion.Euler(0f, 0f, -180f);
foreach (BrushCell cell in m_Cells)
{
Vector3 oldOffset = cell.offset;
cell.offset = flip * oldOffset;
cell.orientation = cell.orientation * orientation;
}
}
/// Updates the size, pivot and the number of layers of the brush.
/// New size of the brush.
/// New pivot of the brush.
public void UpdateSizeAndPivot(Vector3Int size, Vector3Int pivot)
{
m_Size = size;
m_Pivot = pivot;
SizeUpdated();
}
///
/// Sets a GameObject at the position in the brush.
///
/// Position to set the GameObject in the brush.
/// GameObject to set in the brush.
public void SetGameObject(Vector3Int position, GameObject go)
{
if (ValidateCellPosition(position))
m_Cells[GetCellIndex(position)].gameObject = go;
}
///
/// Sets a position offset at the position in the brush.
///
/// Position to set the offset in the brush.
/// Offset to set in the brush.
public void SetOffset(Vector3Int position, Vector3 offset)
{
if (ValidateCellPosition(position))
m_Cells[GetCellIndex(position)].offset = offset;
}
///
/// Sets an orientation at the position in the brush.
///
/// Position to set the orientation in the brush.
/// Orientation to set in the brush.
public void SetOrientation(Vector3Int position, Quaternion orientation)
{
if (ValidateCellPosition(position))
m_Cells[GetCellIndex(position)].orientation = orientation;
}
///
/// Sets a scale at the position in the brush.
///
/// Position to set the scale in the brush.
/// Scale to set in the brush.
public void SetScale(Vector3Int position, Vector3 scale)
{
if (ValidateCellPosition(position))
m_Cells[GetCellIndex(position)].scale = scale;
}
/// Gets the index to the GameObjectBrush::ref::BrushCell based on the position of the BrushCell.
/// Position of the BrushCell.
/// The cell index for the position of the BrushCell.
public int GetCellIndex(Vector3Int brushPosition)
{
return GetCellIndex(brushPosition.x, brushPosition.y, brushPosition.z);
}
/// Gets the index to the GameObjectBrush::ref::BrushCell based on the position of the BrushCell.
/// X Position of the BrushCell.
/// Y Position of the BrushCell.
/// Z Position of the BrushCell.
/// The cell index for the position of the BrushCell.
public int GetCellIndex(int x, int y, int z)
{
return x + m_Size.x * y + m_Size.x * m_Size.y * z;
}
/// Gets the index to the GameObjectBrush::ref::BrushCell based on the position of the BrushCell.
/// X Position of the BrushCell.
/// Y Position of the BrushCell.
/// Z Position of the BrushCell.
/// X Size of Brush.
/// Y Size of Brush.
/// Z Size of Brush.
/// The cell index for the position of the BrushCell.
public int GetCellIndex(int x, int y, int z, int sizex, int sizey, int sizez)
{
return x + sizex * y + sizex * sizey * z;
}
/// Gets the index to the GameObjectBrush::ref::BrushCell based on the position of the BrushCell. Wraps each coordinate if it is larger than the size of the GameObjectBrush.
/// X Position of the BrushCell.
/// Y Position of the BrushCell.
/// Z Position of the BrushCell.
/// The cell index for the position of the BrushCell.
public int GetCellIndexWrapAround(int x, int y, int z)
{
return (x % m_Size.x) + m_Size.x * (y % m_Size.y) + m_Size.x * m_Size.y * (z % m_Size.z);
}
private GameObject GetObjectInCell(GridLayout grid, Transform parent, Vector3Int position)
{
int childCount;
GameObject[] sceneChildren = null;
if (parent == null)
{
var scene = SceneManager.GetActiveScene();
sceneChildren = scene.GetRootGameObjects();
childCount = scene.rootCount;
}
else
{
childCount = parent.childCount;
}
var anchorCellOffset = Vector3Int.FloorToInt(m_Anchor);
var cellSize = grid.cellSize;
anchorCellOffset.x = cellSize.x == 0 ? 0 : anchorCellOffset.x;
anchorCellOffset.y = cellSize.y == 0 ? 0 : anchorCellOffset.y;
anchorCellOffset.z = cellSize.z == 0 ? 0 : anchorCellOffset.z;
for (var i = 0; i < childCount; i++)
{
var child = sceneChildren == null ? parent.GetChild(i) : sceneChildren[i].transform;
if (position == grid.WorldToCell(child.position) - anchorCellOffset)
return child.gameObject;
}
return null;
}
private bool ValidateCellPosition(Vector3Int position)
{
var valid =
position.x >= 0 && position.x < size.x &&
position.y >= 0 && position.y < size.y &&
position.z >= 0 && position.z < size.z;
if (!valid)
throw new ArgumentException(string.Format("Position {0} is an invalid cell position. Valid range is between [{1}, {2}).", position, Vector3Int.zero, size));
return true;
}
internal void SizeUpdated(bool keepContents = false)
{
Array.Resize(ref m_Cells, sizeCount);
BoundsInt bounds = new BoundsInt(Vector3Int.zero, m_Size);
foreach (Vector3Int pos in bounds.allPositionsWithin)
{
if (keepContents || m_Cells[GetCellIndex(pos)] == null)
m_Cells[GetCellIndex(pos)] = new BrushCell();
}
}
private static void SetSceneCell(GridLayout grid, Transform parent, Vector3Int position, GameObject go, Vector3 offset, Vector3 scale, Quaternion orientation, Vector3 anchor)
{
if (go == null)
return;
GameObject instance;
if (PrefabUtility.IsPartOfPrefabAsset(go))
{
instance = (GameObject) PrefabUtility.InstantiatePrefab(go, parent != null ? parent.root.gameObject.scene : SceneManager.GetActiveScene());
instance.transform.parent = parent;
}
else
{
instance = Instantiate(go, parent);
instance.name = go.name;
instance.SetActive(true);
foreach (var renderer in instance.GetComponentsInChildren())
{
renderer.enabled = true;
}
}
Undo.RegisterCreatedObjectUndo(instance, "Paint GameObject");
var cellSize = grid.cellSize;
var cellStride = cellSize + grid.cellGap;
cellStride.x = Mathf.Approximately(0f, cellStride.x) ? 1f : cellStride.x;
cellStride.y = Mathf.Approximately(0f, cellStride.y) ? 1f : cellStride.y;
cellStride.z = Mathf.Approximately(0f, cellStride.z) ? 1f : cellStride.z;
var anchorRatio = new Vector3(
anchor.x * cellSize.x / cellStride.x,
anchor.y * cellSize.y / cellStride.y,
anchor.z * cellSize.z / cellStride.z
);
instance.transform.position = grid.LocalToWorld(grid.CellToLocalInterpolated(position + anchorRatio));
instance.transform.localRotation = orientation;
instance.transform.localScale = scale;
instance.transform.Translate(offset);
}
private void ClearSceneCell(GridLayout grid, Transform parent, Vector3Int position)
{
GameObject erased = GetObjectInCell(grid, parent, new Vector3Int(position.x, position.y, position.z));
if (erased != null)
Undo.DestroyObjectImmediate(erased);
}
///
/// Hashes the contents of the brush.
///
/// A hash code of the brush
public override int GetHashCode()
{
int hash = 0;
unchecked
{
foreach (var cell in cells)
{
hash = hash * 33 + cell.GetHashCode();
}
}
return hash;
}
internal void UpdateHiddenGridLayout()
{
var grid = hiddenGrid.GetComponent();
hiddenGridLayout.cellSize = grid.cellSize;
hiddenGridLayout.cellGap = grid.cellGap;
hiddenGridLayout.cellSwizzle = grid.cellSwizzle;
hiddenGridLayout.cellLayout = grid.cellLayout;
}
///
///Brush Cell stores the data to be painted in a grid cell.
///
[Serializable]
public class BrushCell
{
///
/// GameObject to be placed when painting.
///
public GameObject gameObject { get { return m_GameObject; } set { m_GameObject = value; } }
///
/// Position offset of the GameObject when painted.
///
public Vector3 offset { get { return m_Offset; } set { m_Offset = value; } }
///
/// Scale of the GameObject when painted.
///
public Vector3 scale { get { return m_Scale; } set { m_Scale = value; } }
///
/// Orientatio of the GameObject when painted.
///
public Quaternion orientation { get { return m_Orientation; } set { m_Orientation = value; } }
[SerializeField]
private GameObject m_GameObject;
[SerializeField]
Vector3 m_Offset = Vector3.zero;
[SerializeField]
Vector3 m_Scale = Vector3.one;
[SerializeField]
Quaternion m_Orientation = Quaternion.identity;
///
/// Hashes the contents of the brush cell.
///
/// A hash code of the brush cell.
public override int GetHashCode()
{
int hash;
unchecked
{
hash = gameObject != null ? gameObject.GetInstanceID() : 0;
hash = hash * 33 + offset.GetHashCode();
hash = hash * 33 + scale.GetHashCode();
hash = hash * 33 + orientation.GetHashCode();
}
return hash;
}
}
}
///
/// The Brush Editor for a GameObject Brush.
///
[CustomEditor(typeof(GameObjectBrush))]
public class GameObjectBrushEditor : GridBrushEditorBase
{
private bool hiddenGridFoldout;
private Editor hiddenGridEditor;
///
/// The GameObjectBrush for this Editor
///
public GameObjectBrush brush { get { return target as GameObjectBrush; } }
/// Whether the GridBrush can change Z Position.
public override bool canChangeZPosition
{
get { return brush.canChangeZPosition; }
set { brush.canChangeZPosition = value; }
}
///
/// Callback for painting the GUI for the GridBrush in the Scene View.
/// The GameObjectBrush Editor overrides this to draw the preview of the brush when drawing lines.
///
/// Grid that the brush is being used on.
/// Target of the GameObjectBrush::ref::Tool operation. By default the currently selected GameObject.
/// Current selected location of the brush.
/// Current GameObjectBrush::ref::Tool selected.
/// Whether brush is being used.
public override void OnPaintSceneGUI(GridLayout gridLayout, GameObject brushTarget, BoundsInt position, GridBrushBase.Tool tool, bool executing)
{
BoundsInt gizmoRect = position;
if (tool == GridBrushBase.Tool.Paint || tool == GridBrushBase.Tool.Erase)
gizmoRect = new BoundsInt(position.min - brush.pivot, brush.size);
base.OnPaintSceneGUI(gridLayout, brushTarget, gizmoRect, tool, executing);
}
///
/// Callback for painting the inspector GUI for the GameObjectBrush in the tilemap palette.
/// The GameObjectBrush Editor overrides this to show the usage of this Brush.
///
public override void OnPaintInspectorGUI()
{
EditorGUI.BeginChangeCheck();
base.OnInspectorGUI();
if (EditorGUI.EndChangeCheck() && brush.cellCount != brush.sizeCount)
{
brush.SizeUpdated(true);
}
hiddenGridFoldout = EditorGUILayout.Foldout(hiddenGridFoldout, "SceneRoot Grid");
if (hiddenGridFoldout)
{
EditorGUI.indentLevel++;
using (new EditorGUI.DisabledScope(GridPaintingState.scenePaintTarget != brush.hiddenGrid))
{
if (hiddenGridEditor == null)
{
hiddenGridEditor = Editor.CreateEditor(brush.hiddenGrid.GetComponent());
}
brush.hiddenGrid.hideFlags = HideFlags.None;
EditorGUI.BeginChangeCheck();
hiddenGridEditor.OnInspectorGUI();
if (EditorGUI.EndChangeCheck())
{
brush.UpdateHiddenGridLayout();
EditorUtility.SetDirty(brush);
SceneView.RepaintAll();
}
brush.hiddenGrid.hideFlags = HideFlags.HideAndDontSave;
}
EditorGUI.indentLevel--;
}
}
///
/// The targets that the GameObjectBrush can paint on
///
public override GameObject[] validTargets
{
get
{
StageHandle currentStageHandle = StageUtility.GetCurrentStageHandle();
var results = currentStageHandle.FindComponentsOfType().Where(x =>
{
GameObject gameObject;
return (gameObject = x.gameObject).scene.isLoaded
&& gameObject.activeInHierarchy;
}).Select(x => x.gameObject);
return results.Prepend(brush.hiddenGrid).ToArray();
}
}
}
}