forked from noxious/client
142 lines
6.0 KiB
Vue
142 lines
6.0 KiB
Vue
<template>
|
|
<div class="game-container">
|
|
<GmTools />
|
|
<Game class="game" :config="gameConfig" @create="createGame" v-if="!zoneEditorStore.active">
|
|
<Scene name="main" @preload="preloadScene" @create="createScene">
|
|
<div class="top-ui">
|
|
<Hud />
|
|
</div>
|
|
<div class="center-ui">
|
|
<World />
|
|
</div>
|
|
<div class="bottom-ui">
|
|
<Chat />
|
|
<Menubar />
|
|
</div>
|
|
</Scene>
|
|
</Game>
|
|
<Game class="game" :config="gameConfig" @create="createGame" v-else>
|
|
<Scene name="main" @preload="preloadScene" @create="createScene">
|
|
<ZoneEditor />
|
|
</Scene>
|
|
</Game>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import 'phaser'
|
|
import { Game, Scene } from 'phavuer'
|
|
import World from '@/components/World.vue'
|
|
import Hud from '@/components/gui/Hud.vue'
|
|
import Chat from '@/components/gui/Chat.vue'
|
|
import Menubar from '@/components/gui/Menu.vue'
|
|
import { onUnmounted } from 'vue'
|
|
import GmTools from '@/components/utilities/GmTools.vue'
|
|
import { useSocketStore } from '@/stores/socket'
|
|
import { useZoneEditorStore } from '@/stores/zoneEditor'
|
|
import ZoneEditor from '@/components/utilities/zoneEditor/ZoneEditor.vue'
|
|
|
|
const socket = useSocketStore()
|
|
const zoneEditorStore = useZoneEditorStore()
|
|
|
|
onUnmounted(() => {
|
|
socket.disconnectSocket()
|
|
})
|
|
|
|
// on page close
|
|
addEventListener('beforeunload', () => {
|
|
socket.disconnectSocket()
|
|
})
|
|
|
|
const gameConfig = {
|
|
name: 'New Quest',
|
|
width: window.innerWidth,
|
|
height: window.innerHeight,
|
|
type: Phaser.AUTO,
|
|
pixelArt: true
|
|
}
|
|
|
|
const createGame = (game: Phaser.Game) => {
|
|
addEventListener('resize', () => {
|
|
game.scale.resize(window.innerWidth, window.innerHeight)
|
|
})
|
|
}
|
|
|
|
const preloadScene = (scene: Phaser.Scene) => {
|
|
type Asset = {
|
|
key: string
|
|
value: string
|
|
type: 'base64' | 'link'
|
|
}
|
|
|
|
socket.connection.emit('assets:download', {}, (response: Asset[]) => {
|
|
response.forEach((asset) => {
|
|
if (asset.type === 'base64') {
|
|
scene.textures.addBase64(asset.key, asset.value)
|
|
}
|
|
if (asset.type === 'link') {
|
|
scene.load.image(asset.key, asset.value)
|
|
}
|
|
})
|
|
})
|
|
scene.load.image('tiles', '/assets/zone/tiles.png')
|
|
scene.load.image('walls', '/assets/zone/walls.png')
|
|
scene.load.image('wall1', '/assets/zone/wall1.png')
|
|
scene.load.image('wall2', '/assets/zone/wall2.png')
|
|
scene.load.image('waypoint', '/assets/waypoint.png')
|
|
scene.textures.addBase64(
|
|
'character',
|
|
'data:image/png;base64,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'
|
|
)
|
|
scene.load.spritesheet('characterW', '/assets/avatar/default/walk.png', { frameWidth: 36, frameHeight: 94 })
|
|
}
|
|
|
|
const playScene = (scene: Phaser.Scene) => {}
|
|
|
|
const createScene = (scene: Phaser.Scene) => {
|
|
scene.anims.create({
|
|
key: 'walk',
|
|
frameRate: 7,
|
|
frames: scene.anims.generateFrameNumbers('characterW', { start: 0, end: 3 }),
|
|
repeat: -1
|
|
})
|
|
|
|
// const grid = scene.add.grid(0, 0, window.innerWidth, window.innerHeight, 64, 32, 0, 0, 0xff0000, 0.5).setOrigin(0, 0);
|
|
//
|
|
// window.addEventListener('resize', () => {
|
|
// grid.setSize(window.innerWidth, window.innerHeight);
|
|
// });
|
|
}
|
|
</script>
|
|
|
|
<style scoped lang="scss">
|
|
.game-container {
|
|
display: flex;
|
|
justify-content: center;
|
|
align-items: center;
|
|
height: 100vh;
|
|
padding: 30px;
|
|
position: relative;
|
|
}
|
|
|
|
.top-ui,
|
|
.bottom-ui {
|
|
display: flex;
|
|
position: absolute;
|
|
justify-content: space-between;
|
|
left: 0;
|
|
right: 0;
|
|
margin: 0 48px;
|
|
}
|
|
|
|
.top-ui {
|
|
top: 48px;
|
|
}
|
|
|
|
.bottom-ui {
|
|
align-items: center;
|
|
bottom: 100px;
|
|
height: 100px;
|
|
}
|
|
</style>
|