204 lines
8.8 KiB
C#
204 lines
8.8 KiB
C#
|
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<GTerrainResourceFlag> 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
|
||
|
}
|
||
|
}
|
||
|
}
|