mirror of
https://github.com/CPunch/gopenfusion.git
synced 2024-12-04 22:46:32 +00:00
Compare commits
2 Commits
3e04103ae4
...
459b71a109
Author | SHA1 | Date | |
---|---|---|---|
459b71a109 | |||
06f4a4d33f |
@ -7,23 +7,29 @@ import (
|
|||||||
|
|
||||||
type Chunk struct {
|
type Chunk struct {
|
||||||
Position ChunkPosition
|
Position ChunkPosition
|
||||||
Entities map[Entity]struct{}
|
entities map[Entity]struct{}
|
||||||
lock sync.Mutex
|
lock sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewChunk(position ChunkPosition) *Chunk {
|
func NewChunk(position ChunkPosition) *Chunk {
|
||||||
return &Chunk{
|
return &Chunk{
|
||||||
Position: position,
|
Position: position,
|
||||||
Entities: make(map[Entity]struct{}),
|
entities: make(map[Entity]struct{}),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Chunk) AddEntity(entity Entity) {
|
func (c *Chunk) AddEntity(entity Entity) {
|
||||||
c.Entities[entity] = struct{}{}
|
c.lock.Lock()
|
||||||
|
defer c.lock.Unlock()
|
||||||
|
|
||||||
|
c.entities[entity] = struct{}{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Chunk) RemoveEntity(entity Entity) {
|
func (c *Chunk) RemoveEntity(entity Entity) {
|
||||||
delete(c.Entities, entity)
|
c.lock.Lock()
|
||||||
|
defer c.lock.Unlock()
|
||||||
|
|
||||||
|
delete(c.entities, entity)
|
||||||
}
|
}
|
||||||
|
|
||||||
// send packet to all peers in this chunk and kill each peer if error
|
// send packet to all peers in this chunk and kill each peer if error
|
||||||
@ -31,14 +37,23 @@ func (c *Chunk) SendPacket(typeID uint32, pkt ...interface{}) {
|
|||||||
c.SendPacketExclude(nil, typeID, pkt...)
|
c.SendPacketExclude(nil, typeID, pkt...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Chunk) SendPacketExclude(exclude Entity, typeID uint32, pkt ...interface{}) {
|
// calls f for each entity in this chunk, if f returns true, stop iterating
|
||||||
|
func (c *Chunk) ForEachEntity(f func(entity Entity) bool) {
|
||||||
c.lock.Lock()
|
c.lock.Lock()
|
||||||
defer c.lock.Unlock()
|
defer c.lock.Unlock()
|
||||||
|
|
||||||
for entity := range c.Entities {
|
for entity := range c.entities {
|
||||||
|
if f(entity) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Chunk) SendPacketExclude(exclude Entity, typeID uint32, pkt ...interface{}) {
|
||||||
|
c.ForEachEntity(func(entity Entity) bool {
|
||||||
// only send to players, and exclude the player that sent the packet
|
// only send to players, and exclude the player that sent the packet
|
||||||
if entity.GetKind() != ENTITY_KIND_PLAYER || entity == exclude {
|
if entity.GetKind() != ENTITY_KIND_PLAYER || entity == exclude {
|
||||||
continue
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
plr, ok := entity.(*Player)
|
plr, ok := entity.(*Player)
|
||||||
@ -51,7 +66,9 @@ func (c *Chunk) SendPacketExclude(exclude Entity, typeID uint32, pkt ...interfac
|
|||||||
log.Printf("Error sending packet to peer %p: %v", peer, err)
|
log.Printf("Error sending packet to peer %p: %v", peer, err)
|
||||||
peer.Kill()
|
peer.Kill()
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
return false
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Chunk) GetAdjacentPositions() []ChunkPosition {
|
func (c *Chunk) GetAdjacentPositions() []ChunkPosition {
|
||||||
|
@ -12,11 +12,11 @@ const (
|
|||||||
type Entity interface {
|
type Entity interface {
|
||||||
GetKind() EntityKind
|
GetKind() EntityKind
|
||||||
|
|
||||||
GetChunk() ChunkPosition
|
GetChunkPos() ChunkPosition
|
||||||
GetPosition() (x int, y int, z int)
|
GetPosition() (x int, y int, z int)
|
||||||
GetAngle() int
|
GetAngle() int
|
||||||
|
|
||||||
SetChunk(chunk ChunkPosition)
|
SetChunkPos(chunk ChunkPosition)
|
||||||
SetPosition(x, y, z int)
|
SetPosition(x, y, z int)
|
||||||
SetAngle(angle int)
|
SetAngle(angle int)
|
||||||
|
|
||||||
|
@ -36,7 +36,7 @@ func (npc *NPC) GetKind() EntityKind {
|
|||||||
return ENTITY_KIND_NPC
|
return ENTITY_KIND_NPC
|
||||||
}
|
}
|
||||||
|
|
||||||
func (npc *NPC) GetChunk() ChunkPosition {
|
func (npc *NPC) GetChunkPos() ChunkPosition {
|
||||||
return npc.Chunk
|
return npc.Chunk
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -48,7 +48,7 @@ func (npc *NPC) GetAngle() int {
|
|||||||
return npc.Angle
|
return npc.Angle
|
||||||
}
|
}
|
||||||
|
|
||||||
func (npc *NPC) SetChunk(chunk ChunkPosition) {
|
func (npc *NPC) SetChunkPos(chunk ChunkPosition) {
|
||||||
npc.Chunk = chunk
|
npc.Chunk = chunk
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,7 +44,7 @@ func (plr *Player) GetKind() EntityKind {
|
|||||||
return ENTITY_KIND_PLAYER
|
return ENTITY_KIND_PLAYER
|
||||||
}
|
}
|
||||||
|
|
||||||
func (plr *Player) GetChunk() ChunkPosition {
|
func (plr *Player) GetChunkPos() ChunkPosition {
|
||||||
return plr.Chunk
|
return plr.Chunk
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -56,7 +56,7 @@ func (plr *Player) GetAngle() int {
|
|||||||
return plr.Angle
|
return plr.Angle
|
||||||
}
|
}
|
||||||
|
|
||||||
func (plr *Player) SetChunk(chunk ChunkPosition) {
|
func (plr *Player) SetChunkPos(chunk ChunkPosition) {
|
||||||
plr.Chunk = chunk
|
plr.Chunk = chunk
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,15 +5,15 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func (server *ShardServer) addEntity(e entity.Entity) {
|
func (server *ShardServer) addEntity(e entity.Entity) {
|
||||||
pos := e.GetChunk()
|
pos := e.GetChunkPos()
|
||||||
server.addEntityToChunks(server.getViewableChunks(pos), e)
|
server.addEntityToChunks(e, server.getViewableChunks(pos))
|
||||||
server.getChunk(pos).AddEntity(e)
|
server.getChunk(pos).AddEntity(e)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (server *ShardServer) removeEntity(e entity.Entity) {
|
func (server *ShardServer) removeEntity(e entity.Entity) {
|
||||||
// TODO: chunk cleanup
|
// TODO: chunk cleanup
|
||||||
pos := e.GetChunk()
|
pos := e.GetChunkPos()
|
||||||
server.removeEntityFromChunks(server.getViewableChunks(pos), e)
|
server.removeEntityFromChunks(e, server.getViewableChunks(pos))
|
||||||
server.getChunk(pos).RemoveEntity(e)
|
server.getChunk(pos).RemoveEntity(e)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -56,11 +56,11 @@ func (server *ShardServer) sendAllPacket(plr *entity.Player, typeID uint32, pkt
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (server *ShardServer) removeEntityFromChunks(chunks []*entity.Chunk, this entity.Entity) {
|
func (server *ShardServer) removeEntityFromChunks(this entity.Entity, chunks []*entity.Chunk) {
|
||||||
for _, chunk := range chunks {
|
for _, chunk := range chunks {
|
||||||
for e := range chunk.Entities {
|
chunk.ForEachEntity(func(e entity.Entity) bool {
|
||||||
if e == this {
|
if e == this {
|
||||||
continue
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// notify other players we're leaving
|
// notify other players we're leaving
|
||||||
@ -74,15 +74,17 @@ func (server *ShardServer) removeEntityFromChunks(chunks []*entity.Chunk, this e
|
|||||||
thisPlr := this.(*entity.Player)
|
thisPlr := this.(*entity.Player)
|
||||||
e.DisappearFromViewOf(thisPlr.Peer)
|
e.DisappearFromViewOf(thisPlr.Peer)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
return false
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (server *ShardServer) addEntityToChunks(chunks []*entity.Chunk, this entity.Entity) {
|
func (server *ShardServer) addEntityToChunks(this entity.Entity, chunks []*entity.Chunk) {
|
||||||
for _, chunk := range chunks {
|
for _, chunk := range chunks {
|
||||||
for e := range chunk.Entities {
|
chunk.ForEachEntity(func(e entity.Entity) bool {
|
||||||
if e == this {
|
if e == this {
|
||||||
continue
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// notify other players we're entering
|
// notify other players we're entering
|
||||||
@ -96,7 +98,9 @@ func (server *ShardServer) addEntityToChunks(chunks []*entity.Chunk, this entity
|
|||||||
thisPlr := this.(*entity.Player)
|
thisPlr := this.(*entity.Player)
|
||||||
e.EnterIntoViewOf(thisPlr.Peer)
|
e.EnterIntoViewOf(thisPlr.Peer)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
return false
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -114,9 +118,9 @@ func (server *ShardServer) updateEntityChunk(e entity.Entity, from entity.ChunkP
|
|||||||
toEnter := entity.ChunkSliceDifference(newViewables, oldViewables)
|
toEnter := entity.ChunkSliceDifference(newViewables, oldViewables)
|
||||||
|
|
||||||
// update chunks
|
// update chunks
|
||||||
server.removeEntityFromChunks(toExit, e)
|
server.removeEntityFromChunks(e, toExit)
|
||||||
server.addEntityToChunks(toEnter, e)
|
server.addEntityToChunks(e, toEnter)
|
||||||
server.getChunk(from).RemoveEntity(e)
|
server.getChunk(from).RemoveEntity(e)
|
||||||
server.getChunk(to).AddEntity(e)
|
server.getChunk(to).AddEntity(e)
|
||||||
e.SetChunk(to)
|
e.SetChunkPos(to)
|
||||||
}
|
}
|
||||||
|
@ -10,7 +10,7 @@ func (server *ShardServer) updatePlayerPosition(plr *entity.Player, X, Y, Z, Ang
|
|||||||
plr.Y = Y
|
plr.Y = Y
|
||||||
plr.Z = Z
|
plr.Z = Z
|
||||||
plr.Angle = Angle
|
plr.Angle = Angle
|
||||||
server.updateEntityChunk(plr, plr.GetChunk(), entity.MakeChunkPosition(X, Y))
|
server.updateEntityChunk(plr, plr.GetChunkPos(), entity.MakeChunkPosition(X, Y))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (server *ShardServer) playerMove(peer *protocol.CNPeer, pkt protocol.Packet) error {
|
func (server *ShardServer) playerMove(peer *protocol.CNPeer, pkt protocol.Packet) error {
|
||||||
|
Loading…
Reference in New Issue
Block a user