using System.Collections.Generic; using UnityEngine; namespace Pinwheel.Griffin.PaintTool { public class GElevationPainter : IGTexturePainter, IGTexturePainterWithLivePreview { public string Instruction { get { string s = string.Format( "Gradually modifying terrain geometry.\n" + " - Hold Left Mouse to raise.\n" + " - Hold {0} & Left Mouse to lower.\n" + " - Hold {1} & Left Mouse to smooth.", "Ctrl", "Shift"); return s; } } public string HistoryPrefix { get { return "Elevation Painting"; } } public List GetResourceFlagForHistory(GTexturePainterArgs args) { return GCommon.HeightMapAndFoliageResourceFlags; } private void SetupTextureGrid(GStylizedTerrain t, Material mat) { mat.SetTexture("_MainTex_Left", t.LeftNeighbor && t.LeftNeighbor.TerrainData ? t.LeftNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_TopLeft", t.LeftNeighbor && t.LeftNeighbor.TopNeighbor && t.LeftNeighbor.TopNeighbor.TerrainData ? t.LeftNeighbor.TopNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_TopLeft", t.TopNeighbor && t.TopNeighbor.LeftNeighbor && t.TopNeighbor.LeftNeighbor.TerrainData ? t.TopNeighbor.LeftNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_Top", t.TopNeighbor && t.TopNeighbor.TerrainData ? t.TopNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_TopRight", t.RightNeighbor && t.RightNeighbor.TopNeighbor && t.RightNeighbor.TopNeighbor.TerrainData ? t.RightNeighbor.TopNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_TopRight", t.TopNeighbor && t.TopNeighbor.RightNeighbor && t.TopNeighbor.RightNeighbor.TerrainData ? t.TopNeighbor.RightNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_Right", t.RightNeighbor && t.RightNeighbor.TerrainData ? t.RightNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_BottomRight", t.RightNeighbor && t.RightNeighbor.BottomNeighbor && t.RightNeighbor.BottomNeighbor.TerrainData ? t.RightNeighbor.BottomNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_BottomRight", t.BottomNeighbor && t.BottomNeighbor.RightNeighbor && t.BottomNeighbor.RightNeighbor.TerrainData ? t.BottomNeighbor.RightNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_Bottom", t.BottomNeighbor && t.BottomNeighbor.TerrainData ? t.BottomNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_BottomLeft", t.LeftNeighbor && t.LeftNeighbor.BottomNeighbor && t.LeftNeighbor.BottomNeighbor.TerrainData ? t.LeftNeighbor.BottomNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); mat.SetTexture("_MainTex_BottomLeft", t.BottomNeighbor && t.BottomNeighbor.LeftNeighbor && t.BottomNeighbor.LeftNeighbor.TerrainData ? t.BottomNeighbor.LeftNeighbor.TerrainData.Geometry.HeightMap : Texture2D.blackTexture); } public void Paint(GStylizedTerrain terrain, GTexturePainterArgs args) { if (terrain.TerrainData == null) return; if (args.MouseEventType == GPainterMouseEventType.Down) { terrain.ForceLOD(0); GRuntimeSettings.Instance.isEditingGeometry = true; } if (args.MouseEventType == GPainterMouseEventType.Up) { terrain.ForceLOD(-1); GRuntimeSettings.Instance.isEditingGeometry = false; terrain.UpdateTreesPosition(); terrain.UpdateGrassPatches(); terrain.TerrainData.Foliage.ClearTreeDirtyRegions(); terrain.TerrainData.Foliage.ClearGrassDirtyRegions(); return; } Vector2[] uvCorners = new Vector2[args.WorldPointCorners.Length]; for (int i = 0; i < uvCorners.Length; ++i) { uvCorners[i] = terrain.WorldPointToUV(args.WorldPointCorners[i]); } Rect dirtyRect = GUtilities.GetRectContainsPoints(uvCorners); if (!dirtyRect.Overlaps(new Rect(0, 0, 1, 1))) return; Texture2D bg = terrain.TerrainData.Geometry.HeightMap; int heightMapResolution = terrain.TerrainData.Geometry.HeightMapResolution; RenderTexture rt = GTerrainTexturePainter.Internal_GetRenderTexture(heightMapResolution); GCommon.CopyToRT(bg, rt); Material mat = GInternalMaterials.ElevationPainterMaterial; mat.SetTexture("_MainTex", bg); SetupTextureGrid(terrain, mat); mat.SetTexture("_Mask", args.BrushMask); mat.SetFloat("_Opacity", Mathf.Pow(args.Opacity, GTerrainTexturePainter.GEOMETRY_OPACITY_EXPONENT)); if (args.EnableTerrainMask) { mat.SetTexture("_TerrainMask", terrain.TerrainData.Mask.MaskMapOrDefault); } else { mat.SetTexture("_TerrainMask", Texture2D.blackTexture); } int pass = args.ActionType == GPainterActionType.Normal ? 0 : args.ActionType == GPainterActionType.Negative ? 1 : args.ActionType == GPainterActionType.Alternative ? 2 : 0; GCommon.DrawQuad(rt, uvCorners, mat, pass); RenderTexture.active = rt; terrain.TerrainData.Geometry.HeightMap.ReadPixels( new Rect(0, 0, heightMapResolution, heightMapResolution), 0, 0); terrain.TerrainData.Geometry.HeightMap.Apply(); RenderTexture.active = null; terrain.TerrainData.Geometry.SetRegionDirty(dirtyRect); terrain.TerrainData.Foliage.SetTreeRegionDirty(dirtyRect); terrain.TerrainData.Foliage.SetGrassRegionDirty(dirtyRect); terrain.TerrainData.SetDirty(GTerrainData.DirtyFlags.Geometry); } public void Editor_DrawLivePreview(GStylizedTerrain terrain, GTexturePainterArgs args, Camera cam) { #if UNITY_EDITOR Vector2[] uvCorners = new Vector2[args.WorldPointCorners.Length]; for (int i = 0; i < uvCorners.Length; ++i) { uvCorners[i] = terrain.WorldPointToUV(args.WorldPointCorners[i]); } Rect dirtyRect = GUtilities.GetRectContainsPoints(uvCorners); if (!dirtyRect.Overlaps(new Rect(0, 0, 1, 1))) return; Texture2D bg = terrain.TerrainData.Geometry.HeightMap; int heightMapResolution = terrain.TerrainData.Geometry.HeightMapResolution; RenderTexture rt = GTerrainTexturePainter.Internal_GetRenderTexture(terrain, heightMapResolution); GCommon.CopyToRT(bg, rt); Material mat = GInternalMaterials.ElevationPainterMaterial; mat.SetTexture("_MainTex", bg); SetupTextureGrid(terrain, mat); mat.SetTexture("_Mask", args.BrushMask); mat.SetFloat("_Opacity", Mathf.Pow(args.Opacity, GTerrainTexturePainter.GEOMETRY_OPACITY_EXPONENT)); if (args.EnableTerrainMask) { mat.SetTexture("_TerrainMask", terrain.TerrainData.Mask.MaskMapOrDefault); } else { mat.SetTexture("_TerrainMask", Texture2D.blackTexture); } int pass = args.ActionType == GPainterActionType.Normal ? 0 : args.ActionType == GPainterActionType.Negative ? 1 : args.ActionType == GPainterActionType.Alternative ? 2 : 0; GCommon.DrawQuad(rt, uvCorners, mat, pass); GLivePreviewDrawer.DrawGeometryLivePreview(terrain, cam, rt, dirtyRect); #endif } } }