Chunk.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744
  1. #include <InMemoryBuffer.h>
  2. #include <AsynchronCall.h>
  3. #include "Chunk.h"
  4. #include "Constants.h"
  5. #include "Game.h"
  6. #include "NoBlock.h"
  7. Chunk::Chunk(Framework::Punkt location, int dimensionId)
  8. : ReferenceCounter(),
  9. dimensionId(dimensionId),
  10. location(location),
  11. added(0)
  12. {
  13. blocks = new Block * [CHUNK_SIZE * CHUNK_SIZE * WORLD_HEIGHT];
  14. blockIds = new unsigned short[CHUNK_SIZE * CHUNK_SIZE * WORLD_HEIGHT];
  15. lightData = new unsigned char[CHUNK_SIZE * CHUNK_SIZE * WORLD_HEIGHT * 6];
  16. memset(blocks, 0, CHUNK_SIZE * CHUNK_SIZE * WORLD_HEIGHT * sizeof(Block*));
  17. memset(blockIds, 0, CHUNK_SIZE * CHUNK_SIZE * WORLD_HEIGHT * sizeof(unsigned short));
  18. memset(lightData, 0, CHUNK_SIZE * CHUNK_SIZE * WORLD_HEIGHT * 6);
  19. zNeighbours[0] = 0;
  20. zNeighbours[1] = 0;
  21. zNeighbours[2] = 0;
  22. zNeighbours[3] = 0;
  23. }
  24. Chunk::Chunk(Framework::Punkt location, int dimensionId, Framework::StreamReader* zReader)
  25. : Chunk(location, dimensionId)
  26. {
  27. load(zReader);
  28. }
  29. Chunk::~Chunk()
  30. {
  31. for (int i = 0; i < CHUNK_SIZE * CHUNK_SIZE * WORLD_HEIGHT; i++)
  32. {
  33. if (blocks[i])
  34. blocks[i]->release();
  35. }
  36. delete[] blocks;
  37. delete[] blockIds;
  38. delete[] lightData;
  39. }
  40. void Chunk::addLightSource(int index)
  41. {
  42. for (int i : lightSources)
  43. {
  44. if (i == index)
  45. return;
  46. }
  47. lightSources.add(index);
  48. }
  49. void Chunk::removeLightSource(int index)
  50. {
  51. for (auto i = lightSources.begin(); i; i++)
  52. {
  53. if (i.val() == index)
  54. {
  55. i.remove();
  56. return;
  57. }
  58. }
  59. }
  60. void Chunk::sendLightToClient(Framework::StreamWriter* zWriter)
  61. {
  62. for (int z = 0; z < WORLD_HEIGHT; z++)
  63. {
  64. for (int x = 0; x < CHUNK_SIZE; x++)
  65. {
  66. for (int y = 0; y < CHUNK_SIZE; y++)
  67. {
  68. bool needSend = 0;
  69. for (int i = 0; i < 6; i++)
  70. {
  71. Vec3<int> pos = Vec3<int>(x, y, z) + getDirection(getDirectionFromIndex(i));
  72. if (pos.z >= 0 && pos.z < WORLD_HEIGHT)
  73. {
  74. if (pos.x >= 0 && pos.x < CHUNK_SIZE && pos.y >= 0 && pos.y < CHUNK_SIZE)
  75. {
  76. int bi = (pos.x * CHUNK_SIZE + pos.y) * WORLD_HEIGHT + pos.z;
  77. int type = blockIds[bi];
  78. needSend |= type != NoBlockBlockType::ID && type != AirBlockBlockType::ID;
  79. if (needSend)
  80. break;
  81. }
  82. else
  83. {
  84. needSend = 1; // TODO: check if the block is visible
  85. }
  86. }
  87. }
  88. if (needSend)
  89. {
  90. int index = (x * CHUNK_SIZE + y) * WORLD_HEIGHT + z;
  91. zWriter->schreibe((char*)&index, 4);
  92. zWriter->schreibe((char*)(lightData + index * 6), 6);
  93. }
  94. }
  95. }
  96. }
  97. int end = -1;
  98. zWriter->schreibe((char*)&end, 4);
  99. }
  100. Framework::Either<Block*, int> Chunk::zBlockNeighbor(Framework::Vec3<int> location)
  101. {
  102. if (location.x >= 0 && location.x < CHUNK_SIZE && location.y >= 0 && location.y < CHUNK_SIZE && location.z >= 0 && location.z < WORLD_HEIGHT)
  103. {
  104. int index = (location.x * CHUNK_SIZE + location.y) * WORLD_HEIGHT + location.z;
  105. if (blocks[index])
  106. return blocks[index];
  107. else
  108. return (int)blockIds[index];
  109. }
  110. if (added && location.z >= 0 && location.z < WORLD_HEIGHT)
  111. return Game::INSTANCE->zBlockAt({ location.x + this->location.x - CHUNK_SIZE / 2, location.y + this->location.y - CHUNK_SIZE / 2, location.z }, dimensionId);
  112. return 0;
  113. }
  114. void Chunk::notifyObservers(NetworkMessage& msg)
  115. {
  116. Array<int> remove;
  117. int index = 0;
  118. for (int id : observers)
  119. {
  120. Entity* zE = Game::INSTANCE->zEntity(id);
  121. if (!zE)
  122. remove.add(index, 0);
  123. else
  124. Game::INSTANCE->sendMessage(&msg, zE);
  125. index++;
  126. }
  127. for (int i : remove)
  128. observers.remove(i);
  129. }
  130. void Chunk::addObserver(Entity* zEntity, DoLaterHandler& laterHandler)
  131. {
  132. for (int id : observers)
  133. {
  134. if (id == zEntity->getId())
  135. return;
  136. }
  137. int id = zEntity->getId();
  138. observers.add(id);
  139. laterHandler.addTodo([this, id]()
  140. {
  141. InMemoryBuffer buffer;
  142. buffer.schreibe("\4", 1);
  143. buffer.schreibe((char*)&location.x, 4);
  144. buffer.schreibe((char*)&location.y, 4);
  145. sendToClient(&buffer);
  146. sendLightToClient(&buffer);
  147. NetworkMessage msg;
  148. msg.addressDimension();
  149. char* message = new char[buffer.getSize()];
  150. buffer.lese(message, (int)buffer.getSize());
  151. msg.setMessage(message, (int)buffer.getSize(), 1);
  152. msg.setUseBackground();
  153. Entity* e = Game::INSTANCE->zEntity(id);
  154. if (e)
  155. {
  156. Game::INSTANCE->sendMessage(&msg, e);
  157. }
  158. });
  159. }
  160. void Chunk::removeObserver(Entity* zEntity)
  161. {
  162. int index = 0;
  163. for (int id : observers)
  164. {
  165. if (id == zEntity->getId())
  166. {
  167. observers.remove(index);
  168. return;
  169. }
  170. index++;
  171. }
  172. }
  173. void Chunk::api(Framework::StreamReader* zRequest, Entity* zSource, DoLaterHandler& laterHandler)
  174. {
  175. // TODO: answer api messages
  176. char type;
  177. zRequest->lese(&type, 1);
  178. switch (type)
  179. {
  180. case 0:
  181. // register observer
  182. addObserver(zSource, laterHandler);
  183. break;
  184. case 1:
  185. // unsubscribe
  186. removeObserver(zSource);
  187. break;
  188. }
  189. }
  190. void Chunk::initializeLightning()
  191. {
  192. unsigned char dayLight[6] = { 255, 255, 255, 0, 0, 0 };
  193. unsigned char noLight[6] = { 0, 0, 0, 0, 0, 0 };
  194. while (true)
  195. {
  196. bool changes = false;
  197. for (int z = WORLD_HEIGHT - 1; z >= 0; z--)
  198. {
  199. for (int x = 0; x < CHUNK_SIZE; x++)
  200. {
  201. for (int y = 0; y < CHUNK_SIZE; y++)
  202. {
  203. int index = (x * CHUNK_SIZE + y) * WORLD_HEIGHT + z;
  204. unsigned char* light = getLightData(Vec3<int>(x, y, z));
  205. unsigned char newLight[6] = { 0, 0, 0, 0, 0, 0 };
  206. for (int i = 0; i < 6; i++)
  207. {
  208. unsigned char* neighborLeight;
  209. Vec3<int> neighborPos = Vec3<int>(x, y, z) + getDirection(getDirectionFromIndex(i));
  210. if (neighborPos.z < 0 || neighborPos.x < 0 || neighborPos.y < 0 || neighborPos.x >= CHUNK_SIZE || neighborPos.y >= CHUNK_SIZE)
  211. {
  212. neighborLeight = noLight;
  213. }
  214. else if (neighborPos.z >= WORLD_HEIGHT)
  215. {
  216. neighborLeight = dayLight;
  217. }
  218. else
  219. {
  220. neighborLeight = getLightData(neighborPos);
  221. }
  222. for (int j = 0; j < 3; j++)
  223. newLight[j] = (unsigned char)MAX(newLight[j], i == getDirectionIndex(TOP) ? neighborLeight[j] : (unsigned char)((float)neighborLeight[j] * 0.8f));
  224. for (int j = 3; j < 6; j++)
  225. newLight[j] = (unsigned char)MAX(newLight[j], (unsigned char)((float)neighborLeight[j] * 0.8f));
  226. }
  227. const Block* current = blocks[index] ? blocks[index] : StaticRegistry<BlockType>::INSTANCE.zElement(blockIds[index])->zDefault();
  228. // add own light emission
  229. for (int j = 3; j < 6; j++)
  230. newLight[j] = (unsigned char)MAX(newLight[j], current->getLightEmisionColor()[j - 3]);
  231. current->filterPassingLight(newLight);
  232. current->filterPassingLight(newLight + 3);
  233. for (int i = 0; i < 6; i++)
  234. {
  235. if (newLight[i] != light[i])
  236. {
  237. changes = 1;
  238. memcpy(light, newLight, 6);
  239. break;
  240. }
  241. }
  242. }
  243. }
  244. }
  245. if (!changes)
  246. break;
  247. }
  248. }
  249. Framework::Either<Block*, int> Chunk::zBlockAt(Framework::Vec3<int> location) const
  250. {
  251. int index = (location.x * CHUNK_SIZE + location.y) * WORLD_HEIGHT + location.z;
  252. assert(index < CHUNK_SIZE* CHUNK_SIZE* WORLD_HEIGHT);
  253. if (blocks[index])
  254. return blocks[index];
  255. else
  256. return (int)blockIds[index];
  257. }
  258. const Block* Chunk::zBlockConst(Framework::Vec3<int> location) const
  259. {
  260. auto b = zBlockAt(location);
  261. if (b.isA())
  262. return b;
  263. if (b.getB())
  264. return StaticRegistry<BlockType>::INSTANCE.zElement(b.getB())->zDefault();
  265. return 0;
  266. }
  267. void Chunk::instantiateBlock(Framework::Vec3<int> location)
  268. {
  269. auto b = zBlockAt(location);
  270. if (b.isA())
  271. return;
  272. if (!b.getB())
  273. generateBlock(location);
  274. b = zBlockAt(location);
  275. if (b.isB())
  276. putBlockAt(location, StaticRegistry<BlockType>::INSTANCE.zElement(b.getB())->createBlockAt({ location.x + this->location.x - CHUNK_SIZE / 2, location.y + this->location.y - CHUNK_SIZE / 2, location.z }, 0));
  277. }
  278. void Chunk::generateBlock(Framework::Vec3<int> location)
  279. {
  280. int index = (location.x * CHUNK_SIZE + location.y) * WORLD_HEIGHT + location.z;
  281. if (blockIds[index])
  282. return;
  283. auto generated = Game::INSTANCE->zGenerator()->generateSingleBlock({ location.x + this->location.x - CHUNK_SIZE / 2, location.y + this->location.y - CHUNK_SIZE / 2, location.z }, dimensionId);
  284. if (generated.isA())
  285. putBlockAt(location, generated);
  286. else
  287. putBlockTypeAt(location, generated);
  288. }
  289. void Chunk::putBlockAt(Framework::Vec3<int> location, Block* block)
  290. {
  291. int index = (location.x * CHUNK_SIZE + location.y) * WORLD_HEIGHT + location.z;
  292. assert(index < CHUNK_SIZE* CHUNK_SIZE* WORLD_HEIGHT&& index >= 0);
  293. Block* old = blocks[index];
  294. bool change = 0;
  295. bool wasLightSource = old ? old->zBlockType()->isLightSource() : StaticRegistry<BlockType>::INSTANCE.zElement(blockIds[index])->isLightSource();
  296. bool isLightSource = 0;
  297. if (block)
  298. {
  299. change = blockIds[index] != (unsigned short)block->zBlockType()->getId();
  300. blockIds[index] = (unsigned short)block->zBlockType()->getId();
  301. isLightSource = block->zBlockType()->isLightSource();
  302. }
  303. else
  304. {
  305. change = old != 0;
  306. }
  307. blocks[index] = block;
  308. Either<Block*, int> neighbor = zBlockNeighbor(location + getDirection(NORTH));
  309. if (neighbor.isA())
  310. ((Block*)neighbor)->setNeighbour(SOUTH, block);
  311. if (block)
  312. block->setNeighbour(NORTH, neighbor);
  313. neighbor = zBlockNeighbor(location + getDirection(EAST));
  314. if (neighbor.isA())
  315. ((Block*)neighbor)->setNeighbour(WEST, block);
  316. if (block)
  317. block->setNeighbour(EAST, neighbor);
  318. neighbor = zBlockNeighbor(location + getDirection(SOUTH));
  319. if (neighbor.isA())
  320. ((Block*)neighbor)->setNeighbour(NORTH, block);
  321. if (block)
  322. block->setNeighbour(SOUTH, neighbor);
  323. neighbor = zBlockNeighbor(location + getDirection(WEST));
  324. if (neighbor.isA())
  325. ((Block*)neighbor)->setNeighbour(EAST, block);
  326. if (block)
  327. block->setNeighbour(WEST, neighbor);
  328. neighbor = zBlockNeighbor(location + getDirection(TOP));
  329. if (neighbor.isA())
  330. ((Block*)neighbor)->setNeighbour(BOTTOM, block);
  331. if (block)
  332. block->setNeighbour(TOP, neighbor);
  333. neighbor = zBlockNeighbor(location + getDirection(BOTTOM));
  334. if (neighbor.isA())
  335. ((Block*)neighbor)->setNeighbour(TOP, block);
  336. if (block)
  337. block->setNeighbour(BOTTOM, neighbor);
  338. if (old)
  339. old->release();
  340. if (change)
  341. {
  342. if (isLightSource != wasLightSource)
  343. {
  344. if (isLightSource)
  345. addLightSource(index);
  346. else
  347. removeLightSource(index);
  348. }
  349. if (added)
  350. {
  351. char msg[9];
  352. msg[0] = 0; // set block
  353. *(int*)(msg + 1) = index;
  354. *(int*)(msg + 5) = block ? block->zBlockType()->getId() : NoBlockBlockType::ID;
  355. NetworkMessage message;
  356. message.addressChunck(this);
  357. message.setMessage(msg, 9, 0);
  358. notifyObservers(message);
  359. Game::INSTANCE->updateLightning(getDimensionId(), Vec3<int>(location.x + this->location.x - CHUNK_SIZE / 2, location.y + this->location.y - CHUNK_SIZE / 2, location.z));
  360. }
  361. }
  362. }
  363. void Chunk::putBlockTypeAt(Framework::Vec3<int> location, int type)
  364. {
  365. int index = (location.x * CHUNK_SIZE + location.y) * WORLD_HEIGHT + location.z;
  366. assert(index < CHUNK_SIZE* CHUNK_SIZE* WORLD_HEIGHT);
  367. bool wasLightSource = StaticRegistry<BlockType>::INSTANCE.zElement(blockIds[index])->isLightSource();
  368. bool isLightSource = StaticRegistry<BlockType>::INSTANCE.zElement(type)->isLightSource();
  369. if (blockIds[index] != (unsigned short)type)
  370. {
  371. blockIds[index] = (unsigned short)type;
  372. Either<Block*, int> neighbor = zBlockNeighbor(location + getDirection(NORTH));
  373. if (neighbor.isA())
  374. ((Block*)neighbor)->setNeighbourType(SOUTH, type);
  375. neighbor = zBlockNeighbor(location + getDirection(EAST));
  376. if (neighbor.isA())
  377. ((Block*)neighbor)->setNeighbourType(WEST, type);
  378. neighbor = zBlockNeighbor(location + getDirection(SOUTH));
  379. if (neighbor.isA())
  380. ((Block*)neighbor)->setNeighbourType(NORTH, type);
  381. neighbor = zBlockNeighbor(location + getDirection(WEST));
  382. if (neighbor.isA())
  383. ((Block*)neighbor)->setNeighbourType(EAST, type);
  384. neighbor = zBlockNeighbor(location + getDirection(TOP));
  385. if (neighbor.isA())
  386. ((Block*)neighbor)->setNeighbourType(BOTTOM, type);
  387. neighbor = zBlockNeighbor(location + getDirection(BOTTOM));
  388. if (neighbor.isA())
  389. ((Block*)neighbor)->setNeighbourType(TOP, type);
  390. if (isLightSource != wasLightSource)
  391. {
  392. if (isLightSource)
  393. addLightSource(index);
  394. else
  395. removeLightSource(index);
  396. }
  397. if (added)
  398. {
  399. char msg[9];
  400. msg[0] = 0; // set block
  401. *(int*)(msg + 1) = index;
  402. *(int*)(msg + 5) = type;
  403. NetworkMessage message;
  404. message.addressChunck(this);
  405. message.setMessage(msg, 9, 0);
  406. notifyObservers(message);
  407. Game::INSTANCE->updateLightning(getDimensionId(), Vec3<int>(location.x + this->location.x - CHUNK_SIZE / 2, location.y + this->location.y - CHUNK_SIZE / 2, location.z));
  408. }
  409. }
  410. }
  411. void Chunk::setNeighbor(Direction dir, Chunk* zChunk)
  412. {
  413. zNeighbours[getDirectionIndex(dir)] = zChunk;
  414. for (int i = 0; i < CHUNK_SIZE; i++)
  415. {
  416. for (int z = 0; z < WORLD_HEIGHT; z++)
  417. {
  418. if (dir == NORTH)
  419. {
  420. int index = i * CHUNK_SIZE * WORLD_HEIGHT + z;
  421. if (blocks[index])
  422. {
  423. int j = (i * CHUNK_SIZE + CHUNK_SIZE - 1) * WORLD_HEIGHT + z;
  424. if (zChunk && zChunk->blocks[j])
  425. blocks[index]->setNeighbour(NORTH, zChunk->blocks[j]);
  426. else
  427. {
  428. blocks[index]->setNeighbour(NORTH, 0);
  429. blocks[index]->setNeighbourType(NORTH, zChunk ? zChunk->blockIds[j] : 0);
  430. }
  431. }
  432. }
  433. else if (dir == EAST)
  434. {
  435. int index = ((CHUNK_SIZE - 1) * CHUNK_SIZE + i) * WORLD_HEIGHT + z;
  436. if (blocks[index])
  437. {
  438. int j = i * WORLD_HEIGHT + z;
  439. if (zChunk && zChunk->blocks[j])
  440. blocks[index]->setNeighbour(EAST, zChunk->blocks[j]);
  441. else
  442. {
  443. blocks[index]->setNeighbour(EAST, 0);
  444. blocks[index]->setNeighbourType(EAST, zChunk ? zChunk->blockIds[j] : 0);
  445. }
  446. }
  447. }
  448. else if (dir == SOUTH)
  449. {
  450. int index = (i * CHUNK_SIZE + CHUNK_SIZE - 1) * WORLD_HEIGHT + z;
  451. if (blocks[index])
  452. {
  453. int j = i * CHUNK_SIZE * WORLD_HEIGHT + z;
  454. if (zChunk && zChunk->blocks[j])
  455. blocks[index]->setNeighbour(SOUTH, zChunk->blocks[j]);
  456. else
  457. {
  458. blocks[index]->setNeighbour(SOUTH, 0);
  459. blocks[index]->setNeighbourType(SOUTH, zChunk ? zChunk->blockIds[j] : 0);
  460. }
  461. }
  462. }
  463. else if (dir == WEST)
  464. {
  465. int index = i * WORLD_HEIGHT + z;
  466. if (blocks[index])
  467. {
  468. int j = ((CHUNK_SIZE - 1) * CHUNK_SIZE + i) * WORLD_HEIGHT + z;
  469. if (zChunk && zChunk->blocks[j])
  470. blocks[index]->setNeighbour(WEST, zChunk->blocks[j]);
  471. else
  472. {
  473. blocks[index]->setNeighbour(WEST, 0);
  474. blocks[index]->setNeighbourType(WEST, zChunk ? zChunk->blockIds[j] : 0);
  475. }
  476. }
  477. }
  478. }
  479. }
  480. }
  481. void Chunk::load(Framework::StreamReader* zReader)
  482. {
  483. unsigned short id = 0;
  484. zReader->lese((char*)&id, 2);
  485. Framework::Vec3<int> pos;
  486. bool d = 0;
  487. while (id)
  488. {
  489. zReader->lese((char*)&pos.x, 4);
  490. zReader->lese((char*)&pos.y, 4);
  491. zReader->lese((char*)&pos.z, 4);
  492. zReader->lese((char*)&d, 1);
  493. if (d)
  494. putBlockAt(pos, StaticRegistry<BlockType>::INSTANCE.zElement(id)->loadBlock(Framework::Vec3<int>(pos.x + location.x - CHUNK_SIZE / 2, pos.y + location.y - CHUNK_SIZE / 2, pos.z), zReader));
  495. else
  496. putBlockTypeAt(pos, id);
  497. zReader->lese((char*)&id, 2);
  498. }
  499. initializeLightning();
  500. }
  501. void Chunk::save(Framework::StreamWriter* zWriter)
  502. {
  503. for (int x = 0; x < CHUNK_SIZE; x++)
  504. {
  505. for (int y = 0; y < CHUNK_SIZE; y++)
  506. {
  507. for (int z = 0; z < WORLD_HEIGHT; z++)
  508. {
  509. int index = (x * CHUNK_SIZE + y) * WORLD_HEIGHT + z;
  510. unsigned short blockType = blocks[index] ? (unsigned short)blocks[index]->zBlockType()->getId() : blockIds[index];
  511. if (blockType)
  512. {
  513. zWriter->schreibe((char*)&blockType, 2);
  514. zWriter->schreibe((char*)&x, 4);
  515. zWriter->schreibe((char*)&y, 4);
  516. zWriter->schreibe((char*)&z, 4);
  517. if (blocks[index])
  518. {
  519. bool d = 1;
  520. zWriter->schreibe((char*)&d, 1);
  521. StaticRegistry<BlockType>::INSTANCE.zElement(blockType)->saveBlock(blocks[index], zWriter);
  522. }
  523. else
  524. {
  525. bool d = 0;
  526. zWriter->schreibe((char*)&d, 1);
  527. }
  528. }
  529. }
  530. }
  531. }
  532. unsigned short end = 0;
  533. zWriter->schreibe((char*)&end, 2);
  534. }
  535. void Chunk::sendToClient(Framework::StreamWriter* zWriter)
  536. {
  537. for (int x = 0; x < CHUNK_SIZE; x++)
  538. {
  539. for (int y = 0; y < CHUNK_SIZE; y++)
  540. {
  541. for (int z = 0; z < WORLD_HEIGHT; z++)
  542. {
  543. int index = (x * CHUNK_SIZE + y) * WORLD_HEIGHT + z;
  544. unsigned short blockType = blocks[index] ? (unsigned short)blocks[index]->zBlockType()->getId() : blockIds[index];
  545. if (blockType)
  546. {
  547. bool visible = 0;
  548. if (!visible)
  549. {
  550. if (!blocks[index])
  551. {
  552. if (CONST_BLOCK(0, blockIds[index])->isTransparent() || CONST_BLOCK(0, blockIds[index])->isPassable())
  553. visible = 1;
  554. else
  555. {
  556. for (int d = 0; d < 6 && !visible; d++)
  557. {
  558. auto n = zBlockNeighbor(getDirection((Directions)getDirectionFromIndex(d)) + Framework::Vec3<int>(x, y, z));
  559. if (n.isA() && (((Block*)n)->isPassable() || ((Block*)n)->isTransparent()))
  560. visible = 1;
  561. if (n.isB() && (CONST_BLOCK(0, n)->isTransparent() || CONST_BLOCK(0, n)->isPassable()))
  562. visible = 1;
  563. }
  564. }
  565. }
  566. else
  567. visible = blocks[index]->isVisible();
  568. }
  569. if (visible && (blocks[index] || blockType != AirBlockBlockType::ID))
  570. {
  571. zWriter->schreibe((char*)&blockType, 2);
  572. zWriter->schreibe((char*)&index, 4);
  573. }
  574. }
  575. }
  576. }
  577. }
  578. unsigned short end = 0;
  579. zWriter->schreibe((char*)&end, 2);
  580. }
  581. void Chunk::removeUnusedBlocks()
  582. {
  583. for (int i = 0; i < CHUNK_SIZE * CHUNK_SIZE * WORLD_HEIGHT; i++)
  584. {
  585. if (blocks[i])
  586. {
  587. if (!blocks[i]->isVisible())
  588. {
  589. int x = (i / WORLD_HEIGHT) / CHUNK_SIZE;
  590. int y = (i / WORLD_HEIGHT) % CHUNK_SIZE;
  591. int z = i % WORLD_HEIGHT;
  592. putBlockAt({ x,y,z }, 0);
  593. putBlockTypeAt({ x, y, z }, NoBlockBlockType::ID);
  594. }
  595. }
  596. else if (blockIds[i])
  597. {
  598. int x = (i / WORLD_HEIGHT) / CHUNK_SIZE;
  599. int y = (i / WORLD_HEIGHT) % CHUNK_SIZE;
  600. int z = i % WORLD_HEIGHT;
  601. bool visible = 0;
  602. if (CONST_BLOCK(0, blockIds[i])->isTransparent() || CONST_BLOCK(0, blockIds[i])->isPassable())
  603. visible = 1;
  604. else
  605. {
  606. for (int d = 0; d < 6 && !visible; d++)
  607. {
  608. auto n = zBlockNeighbor(getDirection((Directions)getDirectionFromIndex(d)) + Framework::Vec3<int>(x, y, z));
  609. if (n.isA() && (((Block*)n)->isPassable() || ((Block*)n)->isTransparent()))
  610. visible = 1;
  611. if (n.isB() && (CONST_BLOCK(0, n)->isTransparent() || CONST_BLOCK(0, n)->isPassable()))
  612. visible = 1;
  613. }
  614. }
  615. if (!visible)
  616. {
  617. putBlockAt({ x,y,z }, 0);
  618. putBlockTypeAt({ x, y, z }, NoBlockBlockType::ID);
  619. }
  620. }
  621. }
  622. int count = 0;
  623. for (int i = 0; i < CHUNK_SIZE * CHUNK_SIZE * WORLD_HEIGHT; i++)
  624. {
  625. if (blockIds[i] && blockIds[i] != AirBlockBlockType::ID)
  626. count++;
  627. }
  628. std::cout << "chunk " << location.x << ", " << location.y << " was generated with " << count << " blocks.\n";
  629. }
  630. int Chunk::getDimensionId() const
  631. {
  632. return dimensionId;
  633. }
  634. Framework::Punkt Chunk::getCenter() const
  635. {
  636. return location;
  637. }
  638. Framework::Vec3<int> Chunk::getMin() const
  639. {
  640. return { location.x - CHUNK_SIZE / 2, location.y - CHUNK_SIZE / 2, 0 };
  641. }
  642. Framework::Vec3<int> Chunk::getMax() const
  643. {
  644. return { location.x + CHUNK_SIZE / 2, location.y + CHUNK_SIZE / 2, WORLD_HEIGHT };
  645. }
  646. void Chunk::prepareRemove()
  647. {
  648. added = 0;
  649. for (int i = 0; i < 4; i++)
  650. {
  651. if (zNeighbours[i])
  652. {
  653. zNeighbours[i]->setNeighbor(getOppositeDirection(getDirectionFromIndex(i)), 0);
  654. zNeighbours[i] = 0;
  655. }
  656. }
  657. }
  658. void Chunk::setAdded()
  659. {
  660. added = 1;
  661. }
  662. bool Chunk::hasObservers() const
  663. {
  664. return observers.getEintragAnzahl() > 0;
  665. }
  666. unsigned char* Chunk::getLightData(Framework::Vec3<int> location) const
  667. {
  668. int index = ((location.x * CHUNK_SIZE + location.y) * WORLD_HEIGHT + location.z) * 6;
  669. assert(index < CHUNK_SIZE* CHUNK_SIZE* WORLD_HEIGHT);
  670. return lightData + index;
  671. }
  672. void Chunk::setLightData(Framework::Vec3<int> location, unsigned char* data)
  673. {
  674. int index = ((location.x * CHUNK_SIZE + location.y) * WORLD_HEIGHT + location.z) * 6;
  675. memcpy(lightData + index, data, 6);
  676. // check if neighbor is a visible block and send update to clients
  677. bool needSend = 0;
  678. for (int i = 0; i < 6; i++)
  679. {
  680. Vec3<int> pos = location + getDirection(getDirectionFromIndex(i));
  681. if (pos.z >= 0 && pos.z < WORLD_HEIGHT)
  682. {
  683. if (pos.x >= 0 && pos.x < CHUNK_SIZE && pos.y >= 0 && pos.y < CHUNK_SIZE)
  684. {
  685. int bi = (pos.x * CHUNK_SIZE + pos.y) * WORLD_HEIGHT + pos.z;
  686. int type = blockIds[bi];
  687. needSend |= type != NoBlockBlockType::ID && type != AirBlockBlockType::ID;
  688. if (needSend)
  689. break;
  690. }
  691. else
  692. {
  693. needSend = 1; // TODO: check if the block is visible
  694. }
  695. }
  696. }
  697. if (needSend)
  698. {
  699. NetworkMessage msg;
  700. msg.addressChunck(this);
  701. char message[11];
  702. message[0] = 1;
  703. *(int*)(message + 1) = index / 6;
  704. memcpy(message + 5, data, 6);
  705. msg.setMessage(message, 11, 0);
  706. msg.setUseBackground();
  707. notifyObservers(msg);
  708. }
  709. }