WorldVoxelController.cs 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739
  1. using System.Collections;
  2. using System.Collections.Generic;
  3. using UnityEngine;
  4. using UnityEngine.UI;
  5. namespace ARLocation
  6. {
  7. public class WorldVoxelController : MonoBehaviour
  8. {
  9. // Start is called before the first frame update
  10. public PrefabDatabase PrefabDatabase;
  11. [System.Serializable]
  12. class Voxel
  13. {
  14. public string PrefabId;
  15. public int i, j, k;
  16. [System.NonSerialized]
  17. public GameObject Instance;
  18. }
  19. struct VoxelHit
  20. {
  21. public Voxel Voxel;
  22. public Vector3 Normal;
  23. public Vector3 WorldNorma;
  24. }
  25. [System.Serializable]
  26. class WorldChunk
  27. {
  28. public List<Voxel> Voxels = new List<Voxel>();
  29. public Location ChunkLocation;
  30. public float ChunkRotation;
  31. public bool HasLocation;
  32. public int Length;
  33. [System.NonSerialized]
  34. public Vector3 Origin;
  35. [System.NonSerialized]
  36. public GameObject ChunkContainer;
  37. [System.NonSerialized]
  38. public Bounds Bounds;
  39. //[System.NonSerialized]
  40. //public GameObject ChunkPlaneInstance;
  41. [System.NonSerialized]
  42. public bool IsFresh;
  43. }
  44. [System.Serializable]
  45. class World
  46. {
  47. public List<WorldChunk> Chunks = new List<WorldChunk>();
  48. }
  49. [System.Serializable]
  50. public class ElementsSettingsData
  51. {
  52. public Button ClearWorldBtn;
  53. public Button PickAxeBtn;
  54. public Button BrickBtn;
  55. public Button StoneBtn;
  56. public Button GoldBtn;
  57. public Text DebugText;
  58. public AudioClip Create;
  59. public AudioClip Destroy;
  60. public ParticleSystem BrickParticle;
  61. public GameObject IndicatorPlane;
  62. public GameObject ChunkPlanePrefab;
  63. }
  64. public enum Tools
  65. {
  66. PickAxe,
  67. Block
  68. }
  69. public enum Blocks
  70. {
  71. Brick,
  72. Stone,
  73. Gold
  74. }
  75. [System.Serializable]
  76. public class RaycastMarginSettings
  77. {
  78. public float Top;
  79. public float Bottom;
  80. public float Left;
  81. public float Right;
  82. public bool IsInside(Vector2 v)
  83. {
  84. if (v.x < Left || v.x > (1 - Right)) return false;
  85. if (v.y < Bottom || v.y > (1 - Top)) return false;
  86. return true;
  87. }
  88. }
  89. [System.Serializable]
  90. public class SettingsData
  91. {
  92. public float CunkScale = 1.0f;
  93. public Color ButtonNormalColor;
  94. public Color ButtonSelectedColor;
  95. }
  96. class StateData
  97. {
  98. public ApplicationState AppState;
  99. public GameState GameState;
  100. public Blocks CurrentBlock;
  101. public WorldChunk CurrentChunk;
  102. public Location CurrentLocation;
  103. }
  104. public ElementsSettingsData Elements;
  105. public RaycastMarginSettings RaycastMargins;
  106. public SettingsData Settings;
  107. private World world = new World();
  108. private readonly StateData state = new StateData();
  109. private void LogText(string str)
  110. {
  111. Debug.Log(str);
  112. Elements.DebugText.text = str;
  113. }
  114. enum ApplicationState
  115. {
  116. Initializing,
  117. Running
  118. };
  119. enum GameState
  120. {
  121. Destroy,
  122. Build
  123. };
  124. private IEnumerator StartWorld()
  125. {
  126. LogText($"Loading previous session file...");
  127. yield return new WaitForSeconds(0.5f);
  128. if (RestoreWorldFromLocalStorage())
  129. {
  130. LogText($"Restored world with {world.Chunks.Count} chunks");
  131. yield return new WaitForSeconds(0.5f);
  132. if (world.Chunks.Count > 0)
  133. {
  134. double distance;
  135. var closestChunk = FindClosestChunk(state.CurrentLocation, out distance, 1000.0);
  136. if (closestChunk != null)
  137. {
  138. LogText($"Found closes chunk at {closestChunk.ChunkLocation}, d = {distance}");
  139. yield return new WaitForSeconds(0.5f);
  140. SetCurrentChunk(closestChunk);
  141. LogText($"Current Chunk Set");
  142. yield return new WaitForSeconds(0.5f);
  143. yield break;
  144. }
  145. else
  146. {
  147. LogText($"No chunk nearby!");
  148. yield return new WaitForSeconds(0.5f);
  149. var i = 0;
  150. foreach (var c in world.Chunks)
  151. {
  152. var d = Location.HorizontalDistance(state.CurrentLocation, c.ChunkLocation);
  153. LogText($"Chunk {i} at {c.ChunkLocation}, d = {d}");
  154. yield return new WaitForSeconds(0.5f);
  155. i++;
  156. }
  157. }
  158. }
  159. }
  160. else
  161. {
  162. LogText($"No world to restore!");
  163. yield return new WaitForSeconds(0.5f);
  164. }
  165. LogText("Creating new chunk...");
  166. yield return new WaitForSeconds(0.5f);
  167. var chunk = CreateTestChunk();
  168. chunk.IsFresh = true;
  169. world.Chunks.Add(chunk);
  170. SetCurrentChunk(chunk);
  171. UpdateChunkLocation(chunk);
  172. LogText("Added new chunk!");
  173. yield return new WaitForSeconds(0.5f);
  174. }
  175. private IEnumerator Start()
  176. {
  177. Utils.Misc.HideGameObject(Elements.IndicatorPlane);
  178. ARLocationProvider.Instance.OnLocationUpdated.AddListener(OnLocationUpdatedListener);
  179. ChooseBrick();
  180. LogText("Starting...");
  181. LogText("Waiting for location...");
  182. yield return new WaitForSeconds(0.5f);
  183. yield return StartCoroutine(WaitForLocationServices());
  184. state.CurrentLocation = ARLocationProvider.Instance.CurrentLocation.ToLocation();
  185. LogText($"Location enabled: {state.CurrentLocation}");
  186. yield return new WaitForSeconds(0.5f);
  187. yield return StartCoroutine(StartWorld());
  188. LogText($"Starting UI Listeners...");
  189. yield return new WaitForSeconds(0.5f);
  190. InitUiListeners();
  191. LogText($"Setting app running...");
  192. yield return new WaitForSeconds(0.5f);
  193. state.AppState = ApplicationState.Running;
  194. LogText($"App is running!");
  195. }
  196. void SetCurrentChunk(WorldChunk chunk)
  197. {
  198. if ((state.CurrentChunk != null) && (state.CurrentChunk.ChunkContainer != null)) Destroy(state.CurrentChunk.ChunkContainer);
  199. state.CurrentChunk = chunk;
  200. if (chunk.ChunkContainer == null)
  201. {
  202. BuildChunk(chunk);
  203. }
  204. }
  205. void AddVoxelToChunk(WorldChunk c, string PrefabId, int i, int j, int k)
  206. {
  207. Voxel v = new Voxel { PrefabId = PrefabId, i = i, j = j, k = k };
  208. v.Instance = Instantiate(PrefabDatabase.GetEntryById(PrefabId), c.ChunkContainer.transform);
  209. v.Instance.transform.localPosition = new Vector3(v.i, v.j, v.k);
  210. c.Voxels.Add(v);
  211. }
  212. WorldChunk CreateDefaultChunk()
  213. {
  214. return new WorldChunk { Origin = new Vector3(0, -1.4f, 4), Length = 100 };
  215. }
  216. WorldChunk CreateTestChunk()
  217. {
  218. WorldChunk c = new WorldChunk
  219. {
  220. Voxels = new List<Voxel>
  221. {
  222. new Voxel
  223. {
  224. PrefabId = "Brick",
  225. Instance = null,
  226. i = 0, j = 0, k = 0
  227. }
  228. },
  229. Origin = new Vector3(0, -1.4f, 4),
  230. Length = 100
  231. };
  232. for (int i = 1; i < 10; i++)
  233. {
  234. c.Voxels.Add(new Voxel { PrefabId = "Brick", i = i, j = 0, k = 0 });
  235. }
  236. for (int i = 1; i < 10; i++)
  237. {
  238. c.Voxels.Add(new Voxel { PrefabId = "Brick", i = i, j = 0, k = 9 });
  239. }
  240. for (int i = 1; i < 10; i++)
  241. {
  242. c.Voxels.Add(new Voxel { PrefabId = "Brick", i = 0, j = 0, k = i });
  243. }
  244. for (int i = 1; i < 10; i++)
  245. {
  246. c.Voxels.Add(new Voxel { PrefabId = "Brick", i = 9, j = 0, k = i });
  247. }
  248. return c;
  249. }
  250. IEnumerator WaitForLocationServices()
  251. {
  252. while (!ARLocationProvider.Instance.IsEnabled)
  253. {
  254. yield return new WaitForSeconds(0.1f);
  255. }
  256. }
  257. bool InputIsDown(out Vector2 pos)
  258. {
  259. if (Application.isEditor)
  260. {
  261. pos = Input.mousePosition;
  262. return Input.GetMouseButtonDown(0) && RaycastMargins.IsInside(pos / new Vector2(Screen.width, Screen.height));
  263. }
  264. if (Input.touchCount == 1 && Input.touches[0].phase == TouchPhase.Began)
  265. {
  266. pos = Input.touches[0].position;
  267. return RaycastMargins.IsInside(pos / new Vector2(Screen.width, Screen.height));
  268. }
  269. pos = new Vector3();
  270. return false;
  271. }
  272. private void Update()
  273. {
  274. if (state.AppState == ApplicationState.Initializing) return;
  275. if (state.CurrentChunk == null)
  276. {
  277. FindClosestChunkOrCreateNew(state.CurrentLocation);
  278. return;
  279. }
  280. Ray ray = Camera.main.ViewportPointToRay(new Vector3(0.5f, 0.5f, 0));
  281. VoxelHit hit;
  282. if (RaycastChunk(ray, state.CurrentChunk, out hit))
  283. {
  284. Utils.Misc.ShowGameObject(Elements.IndicatorPlane);
  285. var t = Elements.IndicatorPlane.transform;
  286. t.SetParent(state.CurrentChunk.ChunkContainer.transform);
  287. t.transform.localScale = new Vector3(0.1f, 0.1f, 0.1f);
  288. t.transform.localEulerAngles = new Vector3(0, 0, 0);
  289. t.transform.localPosition = new Vector3(hit.Voxel.i + Mathf.FloorToInt(hit.Normal.x) * 0.505f, hit.Voxel.j + Mathf.FloorToInt(hit.Normal.y) * 0.505f, hit.Voxel.k + Mathf.FloorToInt(hit.Normal.z) * 0.505f);
  290. //LogText("" + hit.Normal);
  291. var Normal = hit.Normal;
  292. if (Mathf.Abs(Normal.x) < 0.0001f && Mathf.Abs(Normal.y) < 0.0001f)
  293. {
  294. float sign = Normal.z < 0 ? -1.0f : 1.0f;
  295. Elements.IndicatorPlane.transform.localEulerAngles = new Vector3(sign * 90.0f, 0, 0);
  296. }
  297. else if (Mathf.Abs(Normal.z) < 0.0001f && Mathf.Abs(Normal.y) < 0.0001f)
  298. {
  299. float sign = Normal.x < 0 ? 1.0f : -1.0f;
  300. Elements.IndicatorPlane.transform.localEulerAngles = new Vector3(0, 0, sign * 90.0f);
  301. }
  302. }
  303. else
  304. {
  305. Utils.Misc.HideGameObject(Elements.IndicatorPlane);
  306. return;
  307. }
  308. Vector2 touchPos;
  309. bool isDown = InputIsDown(out touchPos);
  310. switch (state.GameState)
  311. {
  312. case GameState.Destroy:
  313. {
  314. if (isDown && hit.Voxel.Instance != null)
  315. {
  316. var i = Instantiate(Elements.BrickParticle);
  317. i.transform.position = hit.Voxel.Instance.transform.position;
  318. i.GetComponent<ParticleSystemRenderer>().material = hit.Voxel.Instance.GetComponent<MeshRenderer>().material;
  319. i.Play();
  320. state.CurrentChunk.Voxels.Remove(hit.Voxel);
  321. Destroy(hit.Voxel.Instance);
  322. var a = Camera.main.GetComponent<AudioSource>();
  323. if (a)
  324. {
  325. a.clip = Elements.Destroy;
  326. a.Play();
  327. }
  328. }
  329. }
  330. break;
  331. case GameState.Build:
  332. {
  333. if (isDown)
  334. {
  335. AddVoxelToChunk(state.CurrentChunk, GetMeshIdForBlock(state.CurrentBlock), hit.Voxel.i + Mathf.FloorToInt(hit.Normal.x), hit.Voxel.j + Mathf.FloorToInt(hit.Normal.y), hit.Voxel.k + Mathf.FloorToInt(hit.Normal.z));
  336. var a = Camera.main.GetComponent<AudioSource>();
  337. if (a)
  338. {
  339. a.clip = Elements.Create;
  340. a.Play();
  341. }
  342. }
  343. }
  344. break;
  345. default:
  346. break;
  347. }
  348. }
  349. bool RaycastChunk(Ray ray, WorldChunk chunk, out VoxelHit hit)
  350. {
  351. //if (!chunk.Bounds.IntersectRay(ray)) {
  352. // hit = new VoxelHit();
  353. // return false;
  354. //}
  355. VoxelHit currentHit = new VoxelHit();
  356. float currentDistance = 0;
  357. bool hasHit = false;
  358. foreach (var v in chunk.Voxels)
  359. {
  360. if (v.Instance)
  361. {
  362. var collider = v.Instance.GetComponent<BoxCollider>();
  363. Debug.Assert(collider);
  364. RaycastHit h;
  365. if (collider.Raycast(ray, out h, chunk.Length))
  366. {
  367. if (!hasHit || h.distance < currentDistance)
  368. {
  369. hasHit = true;
  370. currentDistance = h.distance;
  371. currentHit = new VoxelHit { Voxel = v, WorldNorma = h.normal, Normal = chunk.ChunkContainer.transform.InverseTransformDirection(h.normal) };
  372. }
  373. }
  374. }
  375. }
  376. hit = currentHit;
  377. if (hasHit)
  378. {
  379. hit = currentHit;
  380. return true;
  381. }
  382. else
  383. {
  384. var chunkOrigin = chunk.ChunkContainer.transform.position;
  385. var plane = new Plane(new Vector3(0, 1, 0), -chunkOrigin.y + 0.5f * Settings.CunkScale);
  386. float d;
  387. if (plane.Raycast(ray, out d))
  388. {
  389. var p = chunk.ChunkContainer.transform.InverseTransformPoint(ray.GetPoint(d));
  390. var i = Mathf.FloorToInt(p.x + 0.5f);
  391. var j = -1;
  392. var k = Mathf.FloorToInt(p.z + 0.5f);
  393. hit = new VoxelHit { Voxel = new Voxel { PrefabId = "", i = i, j = j, k = k }, Normal = new Vector3(0, 1, 0) };
  394. return true;
  395. }
  396. }
  397. return false;
  398. }
  399. bool RestoreWorldFromLocalStorage()
  400. {
  401. string json = "";
  402. try
  403. {
  404. json = System.IO.File.ReadAllText(GetJsonFilename(), System.Text.Encoding.UTF8);
  405. }
  406. catch
  407. {
  408. Debug.Log("[ARLocation::WorldBuilder::RestoreWorld]: Failed to open json file for reading.");
  409. //RandomPopulateWorld();
  410. return false;
  411. }
  412. world = JsonUtility.FromJson<World>(json);
  413. Debug.Log($"Restored world from json file '{GetJsonFilename()}'");
  414. return true;
  415. }
  416. string GetJsonFilename()
  417. {
  418. var s = "WorldVoxelCraft";
  419. return Application.persistentDataPath + "/" + s + ".json";
  420. }
  421. WorldChunk FindClosestChunk(Location l, out double distance, double maxDistance)
  422. {
  423. WorldChunk current = null;
  424. double currentDistance = 0;
  425. foreach (var c in world.Chunks)
  426. {
  427. var d = Location.HorizontalDistance(c.ChunkLocation, l);
  428. if (current == null || d < currentDistance)
  429. {
  430. current = c;
  431. currentDistance = d;
  432. }
  433. }
  434. distance = currentDistance;
  435. if (currentDistance > maxDistance) return null;
  436. return current;
  437. }
  438. void BuildChunk(WorldChunk chunk)
  439. {
  440. chunk.ChunkContainer = new GameObject();
  441. chunk.ChunkContainer.transform.localScale = new Vector3(Settings.CunkScale, Settings.CunkScale, Settings.CunkScale);
  442. if (chunk.HasLocation)
  443. {
  444. PlaceAtLocation.AddPlaceAtComponent(chunk.ChunkContainer, chunk.ChunkLocation, new PlaceAtLocation.PlaceAtOptions { MaxNumberOfLocationUpdates = 2 });
  445. chunk.ChunkContainer.transform.localEulerAngles = new Vector3(0, chunk.ChunkRotation, 0);
  446. }
  447. else
  448. {
  449. chunk.ChunkContainer.transform.position = chunk.Origin;
  450. chunk.Bounds = new Bounds(chunk.Origin, new Vector3(chunk.Length, chunk.Length, chunk.Length));
  451. chunk.ChunkContainer.AddComponent<GroundHeight>();
  452. }
  453. foreach (var v in chunk.Voxels)
  454. {
  455. v.Instance = Instantiate(PrefabDatabase.GetEntryById(v.PrefabId), chunk.ChunkContainer.transform);
  456. v.Instance.transform.localPosition = new Vector3(v.i, v.j, v.k);
  457. }
  458. //if (Elements.ChunkPlanePrefab != null)
  459. //{
  460. // chunk.ChunkPlaneInstance = Instantiate(Elements.ChunkPlanePrefab, chunk.ChunkContainer.transform);
  461. // chunk.ChunkPlaneInstance.transform.localPosition = new Vector3(0, -0.5f, 0);
  462. // //chunk.ChunkPlaneInstance.transform.localScale = new Vector3(5, 1, 5);
  463. // chunk.ChunkPlaneInstance.transform.localScale = new Vector3(chunk.Length/10, 1, chunk.Length/10);
  464. // //chunk.ChunkPlaneInstance.GetComponent<MeshRenderer>().material.mainTextureScale = new Vector2(50, 50);
  465. // chunk.ChunkPlaneInstance.GetComponent<MeshRenderer>().material.mainTextureScale = new Vector2(chunk.Length, chunk.Length);
  466. // chunk.ChunkPlaneInstance.GetComponent<MeshRenderer>().material.mainTextureOffset = new Vector2(0.5f, 0.5f);
  467. //}
  468. }
  469. private void OnApplicationPause(bool pause)
  470. {
  471. if (pause) SaveWorldToLocalStorage();
  472. }
  473. private void OnDestroy()
  474. {
  475. SaveWorldToLocalStorage();
  476. }
  477. void SaveWorldToLocalStorage()
  478. {
  479. var json = JsonUtility.ToJson(world);
  480. try
  481. {
  482. System.IO.File.WriteAllText(GetJsonFilename(), json);
  483. }
  484. catch
  485. {
  486. Debug.Log("[ARLocation::WorldBuilder::SaveWorld]: Failed to open json file for writing.");
  487. return;
  488. }
  489. Debug.Log("Saved " + GetJsonFilename());
  490. }
  491. private void OnLocationUpdatedListener(Location l)
  492. {
  493. if (state.AppState != ApplicationState.Running) return;
  494. state.CurrentLocation = l;
  495. FindClosestChunkOrCreateNew(l);
  496. if (state.CurrentChunk != null)
  497. {
  498. UpdateChunkLocation(state.CurrentChunk);
  499. }
  500. }
  501. private void FindClosestChunkOrCreateNew(Location l)
  502. {
  503. double distance;
  504. var newClosestChunk = FindClosestChunk(l, out distance, 1000.0f);
  505. if (newClosestChunk != state.CurrentChunk && newClosestChunk != null)
  506. {
  507. SetCurrentChunk(newClosestChunk);
  508. }
  509. if (state.CurrentChunk == null)
  510. {
  511. SetCurrentChunk(CreateDefaultChunk());
  512. }
  513. }
  514. void UpdateChunkLocation(WorldChunk c)
  515. {
  516. if (!c.IsFresh) return;
  517. c.ChunkLocation = ARLocationManager.Instance.GetLocationForWorldPosition(c.ChunkContainer.transform.position);
  518. c.ChunkLocation.Altitude = 0;
  519. c.ChunkLocation.AltitudeMode = AltitudeMode.GroundRelative;
  520. var arLocationRoot = ARLocationManager.Instance.gameObject.transform;
  521. float angle = Vector3.SignedAngle(c.ChunkContainer.transform.forward, arLocationRoot.forward, new Vector3(0, 1, 0));
  522. c.ChunkRotation = angle;
  523. c.HasLocation = true;
  524. LogText($"Updated chunk location to {c.ChunkLocation}");
  525. }
  526. private void InitUiListeners()
  527. {
  528. Elements.ClearWorldBtn.onClick.AddListener(() =>
  529. {
  530. ClearWorld();
  531. });
  532. Elements.PickAxeBtn.onClick.AddListener(() =>
  533. {
  534. ChoosePickaxe();
  535. });
  536. Elements.BrickBtn.onClick.AddListener(() =>
  537. {
  538. ChooseBrick();
  539. });
  540. Elements.StoneBtn.onClick.AddListener(() =>
  541. {
  542. ChooseStone();
  543. });
  544. Elements.GoldBtn.onClick.AddListener(() =>
  545. {
  546. ChooseGold();
  547. });
  548. }
  549. void ChooseBrick()
  550. {
  551. state.GameState = GameState.Build;
  552. state.CurrentBlock = Blocks.Brick;
  553. Elements.BrickBtn.GetComponent<Image>().color = Settings.ButtonSelectedColor;
  554. Elements.GoldBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  555. Elements.StoneBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  556. Elements.PickAxeBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  557. }
  558. void ChooseGold()
  559. {
  560. state.GameState = GameState.Build;
  561. state.CurrentBlock = Blocks.Gold;
  562. Elements.BrickBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  563. Elements.GoldBtn.GetComponent<Image>().color = Settings.ButtonSelectedColor;
  564. Elements.StoneBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  565. Elements.PickAxeBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  566. }
  567. void ChooseStone()
  568. {
  569. state.GameState = GameState.Build;
  570. state.CurrentBlock = Blocks.Stone;
  571. Elements.BrickBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  572. Elements.GoldBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  573. Elements.StoneBtn.GetComponent<Image>().color = Settings.ButtonSelectedColor;
  574. Elements.PickAxeBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  575. }
  576. void ChoosePickaxe()
  577. {
  578. state.GameState = GameState.Destroy;
  579. Elements.BrickBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  580. Elements.GoldBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  581. Elements.StoneBtn.GetComponent<Image>().color = Settings.ButtonNormalColor;
  582. Elements.PickAxeBtn.GetComponent<Image>().color = Settings.ButtonSelectedColor;
  583. }
  584. void ClearChunk(WorldChunk chunk)
  585. {
  586. if (chunk.ChunkContainer != null)
  587. {
  588. Elements.IndicatorPlane.transform.SetParent(null);
  589. Destroy(chunk.ChunkContainer);
  590. }
  591. chunk.Voxels = new List<Voxel>();
  592. }
  593. void ClearWorld()
  594. {
  595. world.Chunks.ForEach(ClearChunk);
  596. world.Chunks = new List<WorldChunk>();
  597. state.CurrentChunk = null;
  598. LogText("World cleared!");
  599. }
  600. private string GetMeshIdForBlock(Blocks b)
  601. {
  602. switch (b)
  603. {
  604. case Blocks.Brick:
  605. return "Brick";
  606. case Blocks.Stone:
  607. return "Stone";
  608. case Blocks.Gold:
  609. return "Gold";
  610. default:
  611. return "";
  612. }
  613. }
  614. }
  615. }