Re-render chunks when they change
This has some weird problems where the chunks will disappear for a moment. I spent some time investigating it and I'm still not sure why this happens.
This commit is contained in:
parent
c25f18619e
commit
512fabf0b5
TrueCraft.Client
@ -26,7 +26,7 @@ namespace TrueCraft.Client.Handlers
|
||||
// Relevant chunk is not loaded - ignore packet
|
||||
return;
|
||||
}
|
||||
chunk.SetBlockID(adjusted, packet.ID);
|
||||
chunk.SetBlockID(adjusted, (byte)packet.BlockID);
|
||||
chunk.SetMetadata(adjusted, (byte)packet.Metadata);
|
||||
client.OnChunkModified(new ChunkEventArgs(new ReadOnlyChunk(chunk)));
|
||||
}
|
||||
@ -41,10 +41,10 @@ namespace TrueCraft.Client.Handlers
|
||||
public static void HandleChunkData(IPacket _packet, MultiplayerClient client)
|
||||
{
|
||||
var packet = (ChunkDataPacket)_packet;
|
||||
var coords = new Coordinates3D(packet.X, packet.Y, packet.Z);
|
||||
var data = ZlibStream.UncompressBuffer(packet.CompressedData);
|
||||
IChunk chunk;
|
||||
var adjustedCoords = client.World.World.FindBlockPosition(
|
||||
new Coordinates3D(packet.X, packet.Y, packet.Z), out chunk);
|
||||
var adjustedCoords = client.World.World.FindBlockPosition(coords, out chunk);
|
||||
|
||||
if (packet.Width == Chunk.Width
|
||||
&& packet.Height == Chunk.Height
|
||||
|
@ -6,6 +6,8 @@ using System.Collections.Concurrent;
|
||||
using System.Collections.Generic;
|
||||
using System.ComponentModel;
|
||||
using TrueCraft.API;
|
||||
using System.Linq;
|
||||
using System.Threading;
|
||||
|
||||
namespace TrueCraft.Client.Modules
|
||||
{
|
||||
@ -15,7 +17,8 @@ namespace TrueCraft.Client.Modules
|
||||
public ChunkRenderer ChunkRenderer { get; set; }
|
||||
public int ChunksRendered { get; set; }
|
||||
|
||||
private List<Mesh> ChunkMeshes { get; set; }
|
||||
private HashSet<Coordinates2D> ActiveMeshes { get; set; }
|
||||
private List<ChunkMesh> ChunkMeshes { get; set; }
|
||||
private ConcurrentBag<Mesh> IncomingChunks { get; set; }
|
||||
|
||||
private BasicEffect OpaqueEffect { get; set; }
|
||||
@ -27,7 +30,8 @@ namespace TrueCraft.Client.Modules
|
||||
|
||||
ChunkRenderer = new ChunkRenderer(Game.Client.World, Game, Game.BlockRepository);
|
||||
Game.Client.ChunkLoaded += (sender, e) => ChunkRenderer.Enqueue(e.Chunk);
|
||||
//Client.ChunkModified += (sender, e) => ChunkRenderer.Enqueue(e.Chunk, true);
|
||||
Game.Client.ChunkUnloaded += (sender, e) => UnloadChunk(e.Chunk);
|
||||
Game.Client.ChunkModified += (sender, e) => ChunkRenderer.Enqueue(e.Chunk, true);
|
||||
ChunkRenderer.MeshCompleted += MeshCompleted;
|
||||
ChunkRenderer.Start();
|
||||
|
||||
@ -50,8 +54,9 @@ namespace TrueCraft.Client.Modules
|
||||
TransparentEffect.Texture = Game.TextureMapper.GetTexture("terrain.png");
|
||||
TransparentEffect.VertexColorEnabled = true;
|
||||
|
||||
ChunkMeshes = new List<Mesh>();
|
||||
ChunkMeshes = new List<ChunkMesh>();
|
||||
IncomingChunks = new ConcurrentBag<Mesh>();
|
||||
ActiveMeshes = new HashSet<Coordinates2D>();
|
||||
}
|
||||
|
||||
void MeshCompleted(object sender, RendererEventArgs<ReadOnlyChunk> e)
|
||||
@ -59,6 +64,15 @@ namespace TrueCraft.Client.Modules
|
||||
IncomingChunks.Add(e.Result);
|
||||
}
|
||||
|
||||
void UnloadChunk(ReadOnlyChunk chunk)
|
||||
{
|
||||
Game.PendingMainThreadActions.Add(() =>
|
||||
{
|
||||
ActiveMeshes.Remove(chunk.Coordinates);
|
||||
ChunkMeshes.RemoveAll(m => m.Chunk.Coordinates == chunk.Coordinates);
|
||||
});
|
||||
}
|
||||
|
||||
void HandleClientPropertyChanged(object sender, PropertyChangedEventArgs e)
|
||||
{
|
||||
switch (e.PropertyName)
|
||||
@ -73,10 +87,17 @@ namespace TrueCraft.Client.Modules
|
||||
|
||||
public void Update(GameTime gameTime)
|
||||
{
|
||||
Mesh mesh;
|
||||
while (IncomingChunks.TryTake(out mesh))
|
||||
Mesh _mesh;
|
||||
while (IncomingChunks.TryTake(out _mesh))
|
||||
{
|
||||
var mesh = _mesh as ChunkMesh;
|
||||
int existing = -1;
|
||||
if (ActiveMeshes.Contains(mesh.Chunk.Coordinates))
|
||||
existing = ChunkMeshes.FindIndex(m => m.Chunk.Coordinates == mesh.Chunk.Coordinates);
|
||||
ActiveMeshes.Add(mesh.Chunk.Coordinates);
|
||||
ChunkMeshes.Add(mesh);
|
||||
if (existing != -1)
|
||||
ChunkMeshes.RemoveAt(existing);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -89,6 +89,9 @@ namespace TrueCraft.Client.Modules
|
||||
string.Format(ChatColor.Gray + "Looking at {0} ({1})", Game.HighlightedBlock,
|
||||
Enum.GetName(typeof(BlockFace), Game.HighlightedBlockFace)));
|
||||
|
||||
Font.DrawText(SpriteBatch, xOrigin, yOrigin + (yOffset * 3),
|
||||
string.Format(ChatColor.Gray + "{0} pending chunks", Game.ChunkModule.ChunkRenderer.PendingChunks));
|
||||
|
||||
SpriteBatch.End();
|
||||
}
|
||||
|
||||
|
@ -109,6 +109,8 @@ namespace TrueCraft.Client
|
||||
return Chunk.GetBlockLight(coordinates);
|
||||
}
|
||||
|
||||
public Coordinates2D Coordinates { get { return Chunk.Coordinates; } }
|
||||
|
||||
public int X { get { return Chunk.X; } }
|
||||
public int Z { get { return Chunk.Z; } }
|
||||
|
||||
|
@ -39,6 +39,14 @@ namespace TrueCraft.Client.Rendering
|
||||
}
|
||||
}
|
||||
|
||||
public int PendingChunks
|
||||
{
|
||||
get
|
||||
{
|
||||
return _items.Count + _priorityItems.Count;
|
||||
}
|
||||
}
|
||||
|
||||
private ReadOnlyWorld World { get; set; }
|
||||
private TrueCraftGame Game { get; set; }
|
||||
private IBlockRepository BlockRepository { get; set; }
|
||||
|
@ -20,8 +20,8 @@ namespace TrueCraft.Client.Rendering
|
||||
|
||||
private volatile bool _isRunning;
|
||||
private Thread[] _rendererThreads;
|
||||
private ConcurrentQueue<T> _items, _priorityItems;
|
||||
private volatile bool _isDisposed;
|
||||
protected ConcurrentQueue<T> _items, _priorityItems;
|
||||
|
||||
/// <summary>
|
||||
/// Gets whether this renderer is running.
|
||||
@ -150,13 +150,10 @@ namespace TrueCraft.Client.Rendering
|
||||
throw new ObjectDisposedException(GetType().Name);
|
||||
|
||||
if (!_isRunning) return;
|
||||
lock (_syncLock)
|
||||
{
|
||||
if (hasPriority)
|
||||
_priorityItems.Enqueue(item);
|
||||
else
|
||||
_items.Enqueue(item);
|
||||
}
|
||||
if (hasPriority)
|
||||
_priorityItems.Enqueue(item);
|
||||
else
|
||||
_items.Enqueue(item);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
@ -269,9 +269,7 @@ namespace TrueCraft.Client
|
||||
// We should eventually make some means of detecing that we're on a vanilla server to enable this
|
||||
// It's a waste of bandwidth to do it on a TrueCraft server
|
||||
Client.QueuePacket(new PlayerGroundedPacket { OnGround = true });
|
||||
Client.QueuePacket(new PlayerPositionAndLookPacket(Client.Position.X, Client.Position.Y,
|
||||
Client.Position.Y + MultiplayerClient.Height, Client.Position.Z, Client.Yaw, Client.Pitch, false));
|
||||
NextPhysicsUpdate = DateTime.UtcNow.AddMilliseconds(1000 / 20);
|
||||
NextPhysicsUpdate = DateTime.UtcNow.AddMilliseconds(50);
|
||||
}
|
||||
|
||||
foreach (var module in Modules)
|
||||
|
Loading…
x
Reference in New Issue
Block a user