forked from fogleman/Minecraft
-
Notifications
You must be signed in to change notification settings - Fork 33
/
world.py
324 lines (284 loc) · 12.5 KB
/
world.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
# Imports, sorted alphabetically.
# Python packages
from collections import deque, defaultdict, OrderedDict
import os
from time import time
import warnings
from typing import Tuple, Set, Optional, Any, Dict, Deque, DefaultDict
# Third-party packages
import pyglet
from pyglet.gl import *
from pyglet.graphics import Batch
# Modules from this project
from blocks import *
from utils import FACES, FACES_WITH_DIAGONALS, normalize_float, normalize, sectorize, TextureGroup
import globals as G
from client import PacketReceiver
from entity import TileEntity
from custom_types import iVector, fVector
__all__ = (
'World',
)
#The Client's world
class World(dict):
batch: Batch
transparency_batch: Batch
group: TextureGroup
shown: Dict[iVector, Any]
_shown: Dict[iVector, Any]
sectors: DefaultDict[iVector, list]
sectors_shown: Dict[iVector, bool]
urgent_queue: Deque[Any]
lazy_queue: Deque[Any]
sector_queue: Dict[iVector, bool]
packetreceiver: None
sector_packets: Deque[Any]
biome_generator: None
spreading_mutations = {
dirt_block: grass_block,
}
def __init__(self):
super(World, self).__init__()
self.batch = pyglet.graphics.Batch()
self.transparency_batch = pyglet.graphics.Batch()
self.group = TextureGroup(os.path.join('resources', 'textures', 'texture.png'))
self.shown = {}
self._shown = {}
self.sectors = defaultdict(list)
self.sectors_shown = dict()
self.urgent_queue = deque()
self.lazy_queue = deque()
self.sector_queue = OrderedDict()
self.packetreceiver = None
self.sector_packets = deque()
# biome generator for colorizer, set by packet receiver
self.biome_generator = None
def get_block(self, position: iVector) -> Optional[Block]:
return self.get(position)
def get_block_above(self, position: iVector) -> Optional[Block]:
return self.get_block((position[0], position[1] + 1, position[2]))
def get_block_below(self, position: iVector) -> Optional[Block]:
return self.get_block((position[0], position[1] - 1, position[2]))
# Add the block clientside, then tell the server about the new block
def add_block(self, position: iVector, block, sync: bool = True, force: bool = True):
self._add_block(position, block) # For Prediction
if sync:
self.packetreceiver.add_block(position, block)
# Clientside, add the block
def _add_block(self, position: iVector, block):
if position in self:
self._remove_block(position, sync=True)
if hasattr(block, 'entity_type'):
# in world_server we have to create its entity to handle some tasks(growing, etc.)
# but in client's world, we only create a TileEntity that contains the position
# and the world to allow the block update itself and server will handle the task
# and tell us
self[position] = type(block)()
self[position].entity = TileEntity(self, position)
elif block.sub_id_as_metadata:
self[position] = type(block)()
self[position].set_metadata(block.get_metadata())
else:
self[position] = block
self.sectors[sectorize(position)].append(position)
if self.is_exposed(position):
self.show_block(position)
self.inform_neighbors_of_block_change(position)
def remove_block(self, player, position: iVector, sync: bool = True, sound: bool = True):
if sound and player is not None:
self[position].play_break_sound(player, position)
self._remove_block(position, sync=sync)
if sync:
self.packetreceiver.remove_block(position)
# Clientside, delete the block
def _remove_block(self, position: iVector, sync: bool = True):
del self[position]
sector_position = sectorize(position)
try:
self.sectors[sector_position].remove(position)
except ValueError:
warnings.warn('Block %s was unexpectedly not found in sector %s;'
'your save is probably corrupted'
% (position, sector_position))
if sync:
if position in self.shown:
self.hide_block(position)
self.check_neighbors(position)
self.inform_neighbors_of_block_change(position)
def is_exposed(self, position: iVector) -> bool:
x, y, z = position
for fx,fy,fz in FACES:
other_position = (fx+x, fy+y, fz+z)
if other_position not in self or self[other_position].transparent:
return True
return False
def neighbors_iterator(self, position: iVector, relative_neighbors_positions: Tuple[iVector, ...] = FACES):
x, y, z = position
for dx, dy, dz in relative_neighbors_positions:
yield x + dx, y + dy, z + dz
def check_neighbors(self, position: iVector):
for other_position in self.neighbors_iterator(position):
if other_position not in self:
continue
if self.is_exposed(other_position):
if other_position not in self.shown:
self.show_block(other_position)
else:
if other_position in self.shown:
self.hide_block(other_position)
def has_neighbors(self, position: iVector,
is_in: Optional[Set[iVector]] = None,
diagonals: bool = False,
faces: Optional[Tuple[iVector, ...]] = None) -> bool:
if faces is None:
faces = FACES_WITH_DIAGONALS if diagonals else FACES
for other_position in self.neighbors_iterator(
position, relative_neighbors_positions=faces):
if other_position in self:
if is_in is None or self[other_position] in is_in:
return True
return False
def inform_neighbors_of_block_change(self, position: iVector):
for neighbor in self.neighbors_iterator(position):
if neighbor not in self:
continue
self[neighbor].on_neighbor_change(self, position, neighbor)
def hit_test(self, position: fVector, vector: fVector, max_distance: int = 8, hitwater: bool = False):
m = 8
x, y, z = position
dx, dy, dz = vector
dx, dy, dz = dx / m, dy / m, dz / m
previous: fVector = (0.0, 0.0, 0.0)
for _ in range(max_distance * m):
key = normalize((x, y, z))
if key != previous and key in self and (self[key].density != 0.5 or hitwater):
return key, previous
previous = key
x, y, z = x + dx, y + dy, z + dz
return None, None
def hide_block(self, position: iVector, immediate: bool = True):
del self.shown[position]
if immediate:
self._hide_block(position)
else:
self.enqueue(self._hide_block, position)
def _hide_block(self, position: iVector):
self._shown.pop(position).delete()
def show_block(self, position: iVector, immediate: bool = True):
block = self[position]
self.shown[position] = block
if immediate:
self._show_block(position, block)
else:
self.enqueue(self._show_block, position, block)
def _show_block(self, position: iVector, block):
# only show exposed faces
vertex_data = list(block.get_vertices(*position))
texture_data = list(block.texture_data)
color_data = None
if hasattr(block, 'get_color') and self.biome_generator is not None:
temp = self.biome_generator.get_temperature(position[0], position[-1])
humidity = self.biome_generator.get_humidity(position[0], position[-1])
color_data = block.get_color(temp, humidity)
# FIXME: Do something of what follows.
#for neighbor in self.neighbors_iterator(position):
# if neighbor in self:
# count -= 4
# i = index * 12
# j = index * 8
# del vertex_data[i:i + 12]
# del texture_data[j:j + 8]
# if color_data is not None:
# del color_data[i:i+12]
# else:
# index += 1
count = len(texture_data) // 2
# create vertex list
batch = self.transparency_batch if block.transparent else self.batch
if color_data is not None:
self._shown[position] = batch.add(count, GL_QUADS, block.group or self.group,
('v3f/static', vertex_data),
('t2f/static', texture_data),
('c3f/static', color_data))
else:
self._shown[position] = batch.add(count, GL_QUADS, block.group or self.group,
('v3f/static', vertex_data),
('t2f/static', texture_data))
def show_sector(self, sector: iVector):
if sector in self.sectors:
self._show_sector(sector)
else:
self.sectors[sector] = [] #Initialize it so we don't keep requesting it
self.packetreceiver.request_sector(sector)
#Clientside, show a sector we've downloaded
def _show_sector(self, sector: iVector):
for position in self.sectors[sector]:
if position not in self.shown and self.is_exposed(position):
self.show_block(position)
def _hide_sector(self, sector: iVector):
if sector in self.sectors:
for position in self.sectors[sector]:
if position in self: del self[position]
if position in self.shown:
self.hide_block(position)
del self.sectors[sector]
def change_sectors(self, after: iVector):
new_sectors_shown = dict()
pad = G.VISIBLE_SECTORS_RADIUS
x, y, z = after
for distance in range(0, pad + 1):
for dx in range(-distance, distance + 1):
for dz in range(-distance, distance + 1):
if abs(dx) != distance and abs(dz) != distance:
continue
for dy in range(-4, 4):
if dx ** 2 + dy ** 2 + dz ** 2 > (pad + 1) ** 2:
continue
new_sectors_shown[(x + dx, y + dy, z + dz)] = True
#Queue the sectors to be shown, instead of rendering them in real time
for sector in new_sectors_shown.keys():
if sector not in self.sectors_shown:
self.enqueue_sector(True, sector)
self.sectors_shown = new_sectors_shown
def enqueue_sector(self, state: bool, sector: iVector): #State=True to show, False to hide
self.sector_queue[sector] = state
def dequeue_sector(self):
sector, state = self.sector_queue.popitem(False)
if state:
self.show_sector(sector)
pass
else:
self._hide_sector(sector)
def enqueue(self, func, *args, **kwargs):
task = func, args, kwargs
urgent = kwargs.pop('urgent', False)
queue = self.urgent_queue if urgent else self.lazy_queue
if task not in queue:
queue.appendleft(task)
def dequeue(self):
queue = self.urgent_queue or self.lazy_queue
func, args, kwargs = queue.pop()
func(*args, **kwargs)
def process_queue(self, dt):
stoptime = time() + G.QUEUE_PROCESS_SPEED
while (self.sector_queue or self.sector_packets or self.urgent_queue or self.lazy_queue) and time() < stoptime:
#Process as much of the queues as we can
if self.sector_queue:
self.dequeue_sector()
if self.sector_packets:
self.packetreceiver.dequeue_packet()
if self.urgent_queue or self.lazy_queue:
self.dequeue()
def process_entire_queue(self):
while self.urgent_queue or self.lazy_queue:
self.dequeue()
def hide_sectors(self, dt, player):
#TODO: This is pretty laggy, I feel an FPS drop once a second while sector changing because of this
deload = G.DELOAD_SECTORS_RADIUS
plysector = sectorize(player.position)
if player.last_sector != plysector:
px, py, pz = plysector
for sector in self.sectors:
x,y,z = sector
if abs(px-x) > deload or abs(py-y) > deload or abs(pz-z) > deload:
self.enqueue_sector(False, sector)