<template> <Controls :layer="tiles" :depth="0" /> </template> <script setup lang="ts"> import config from '@/config' import { useScene } from 'phavuer' import { useZoneStore } from '@/stores/zone' import { onBeforeMount, onBeforeUnmount, ref } from 'vue' import { storeToRefs } from 'pinia' import { placeTile, setAllTiles } from '@/composables/zoneComposable' import Controls from '@/components/utilities/Controls.vue' const emit = defineEmits(['tilemap:create']) const zoneStore = useZoneStore() const scene = useScene() const { zone } = storeToRefs(zoneStore) const zoneTilemap = createTilemap() const tiles = createTileLayer() let tileArray = createTileArray() function createTilemap() { const zoneData = new Phaser.Tilemaps.MapData({ width: zone.value?.width, height: zone.value?.height, tileWidth: config.tile_size.x, tileHeight: config.tile_size.y, orientation: Phaser.Tilemaps.Orientation.ISOMETRIC, format: Phaser.Tilemaps.Formats.ARRAY_2D }) const tilemap = new Phaser.Tilemaps.Tilemap(scene, zoneData) emit('tilemap:create', tilemap) return tilemap } function createTileLayer() { const tilesFromZone = zone.value?.tiles || [] const uniqueTiles = new Set(tilesFromZone.flat().filter(Boolean)) const tilesetImages = Array.from(uniqueTiles).map((tile, index) => { return zoneTilemap.addTilesetImage(tile, tile, config.tile_size.x, config.tile_size.y, 0, 0, index + 1, { x: 0, y: -config.tile_size.y }) }) as any // Add blank tile tilesetImages.push(zoneTilemap.addTilesetImage('blank_tile', 'blank_tile', config.tile_size.x, config.tile_size.y, 0, 0, 0, { x: 0, y: -config.tile_size.y })) const layer = zoneTilemap.createBlankLayer('tiles', tilesetImages, 0, config.tile_size.y) as Phaser.Tilemaps.TilemapLayer layer.setDepth(0) return layer } function createTileArray() { return Array.from({ length: zone.value?.width ?? 0 }, () => Array.from({ length: zone.value?.height ?? 0 }, () => 'blank_tile')) } onBeforeMount(() => { if (zone.value?.tiles) { setAllTiles(zoneTilemap, tiles, zone.value.tiles) tileArray = zone.value.tiles.map((row) => row.map((tileId) => tileId || 'blank_tile')) } else { tileArray.forEach((row, y) => row.forEach((_, x) => placeTile(zoneTilemap, tiles, x, y, 'blank_tile'))) } }) onBeforeUnmount(() => { zoneTilemap.destroyLayer('tiles') zoneTilemap.removeAllLayers() zoneTilemap.destroy() }) </script>