1) 요구사항
1) 브라우저를 켠다. => DOMContentLoaded 이벤트, new Game()실행
2) 캔버스가 실행된다.(고객은 모름) => Game엔진에서 캔버스 API실행
3) 이미지가 캔버스에 나타난다. => Render, update, Sprite 클래스 실행
4) 이미지가 반복적으로 움직인다. => sprite 설정하기
빠른 구현을 위해서는 프로토타입 방식과 비슷하지만 기초에서 점차 살을 붙여나가는 예광탄 방식(테스트주도)이 좋을 듯하다. 참고로 예광탄 방식은 최소한의 기능만 구현되게끔 A-Z까지 빠르게 진행한 뒤에 살을 조금씩 붙여나가는 기법이다. 이를 위해, 아래와 같은 코스로 진행해보자.
"최소 클래스 생성" => "최소 기능 구현" => "살 붙이기" => "완성"
1 최소 클래스 생성
constructor(){
this.canvas = document.querySelector('#canvas');
this.context = this.canvas.getContext('2d');
const game = this;
this.loadJSON();
}
//게임엔진 메서드 시작
loadJSON(확장자, callback함수){ }
init(){ //게임설정 세팅 }
update(){ //이미지의 상태변화 점검 }
spawn(){ //이미지 생성하기 }
refresh(){ //시간변화, 반복설정하기 }
render(){ //이미지 캔버스에 띄우기 }
//게임 클래스 End
}
class Sprite{
constructor(options){
}
get offset(){ //게임엔진이 정한 이미지 세부사항 설정 }
update(dt){ //시간변화에 따른 이미지 움직임 구현 }
render(){ //캔버스API 구현 }
}
프로퍼티 설정같은 경우, 굳이 초기에 작성할 필요가 없다. 요구사항을 충족하는 메서드를 하나씩 작성하다보면 프로퍼티는 자동으로 늘어난다. 따라서 최소 구현을 위한 요구사항의 메시지에 집중할 필요가 있다. 위에서는 2개 클래스로부터 7~8개의 메서드만 붙이는 것만으로 최소 도메인을 설계했다.
2-1 최소 구현
constructor(){
this.canvas = document.querySelector('#canvas');
this.context = this.canvas.getContext('2d');
const game = this;
this.sprites = [];
this.lastTime;
this.loadJSON('이미지 경로',function(data,game){
game.spriteData = JSON.parse(data); --- ②
game.spriteImage = new Image(); --- ③
game.spriteImage.src = game.spriteData.meta.image;
game.spriteImage.onload = function(){
game.init(); --- ④
}
});
}
//게임엔진 메서드 시작
loadJSON('이미지 경로', callback){
let url = '확장자';
const game = this;
var xobj = new XMLHttpRequest();
xobj.overrideMimetype("application/json");
xobj.open('GET', url+'.json', true );
xobj.onreadystatechange = function(){
if(xobj.readyState == 4 && xobj.status == '200'){
callback(xobj.responseText, game); ---- ①
}
}
xobj.send(null);
}
자체서버를 사용하고 있다면 loadJSON 메서드가 필요하지 않겠지만 클라우드 서버를 사용하고 있다면 ajax를 통해 이미지 파일을 불러들여야 한다. 위의 코드 ①에서는 loadJSON메서드를 통해 이미지 경로를 받고, 결과값을 callback함수로 return하고 있다.
Game클래스는 loadJSON메서드에게 다음 3가지 할일을 지시한다.
1) ajax로 받아온 text데이터를 json형태로 파싱한 뒤 저장하라. --- ②
2) 자바스크립트 내장객체 new Image()를 통해 이미지 객체를 생성하라. --- ③
3) 게임을 실행하라. --- ④
2-2 Game Class 구현
loadJSON()메서드를 통해 실행된 game.init()메서드는 어떤 역할을 해야할까?
(....중략)
init(){ //게임설정 세팅
this.lastTime = Date.now();
this.spawn();
this.refresh();
}
init()메서드는 initialize란 의미 그대로 실행을 해야한다. 캔버스에 이미지를 띄우고, 이미지 정보를 이미지에 보내야하고, 시간이 바뀔 때마다 이미지를 갱신해줘야 한다.
1) 이미지 띄우기 => render()메서드
2) 이미지 정보전달 => spawn()메서드
3) 이미지 갱신 => update(), refresh()메서드
init()메서드의 역할은 Game클래스의 책임과 같다. 시간이 바뀔 때마다 화면을 갱신하고, 화면이 갱신될 때 이미지를 띄우며, 또 바뀐 정보를 이미지객체에 전달한다. 해당 프로젝트처럼 간단한 프로그램에서는 따로 view역할이 없지만 만일 view가 있다면 인터페이스 클래스를 따로 만들고 추상화를 진행하는 역할까지 Game클래스가 담당한다.
update(delta){
//이미지의 상태변화 점검
for(let sprite of this.sprites){
if(sprite == null) continue;
sprite.update(delta);
}
}
spawn(){
//이미지 생성하기
const game = this;
let options = {
context : this.context,
image : this.spriteImage
}
let sprite = new Sprite(options);
this.sprites.push(sprite);
}
refresh(){
//시간변화, 반복설정하기
const game = this;
let now = new Date();
let delta = (now - this.sinceTime)/1000.0;
this.update(delta);
this.render();
this.lastTime = now;
requestAnimationFrame(function(){
game.refresh();
})
}
render(){
//이미지 캔버스에 띄우기
this.context.clearRect(0, 0, this.canvas.width, this.canvas.height )
for(let sprite of this.sprites.sprite){
sprite.render();
}
}
//게임 클래스 End
}
3 Sprite Class 구현
Sprite클래스의 역할은 Game엔진이 지시한 이미지 세팅값에 따라 캔버스에 이미지를 구현해내는 것이다. 최소한의 구현을 위해서는 canvas의 drawImage()메서드를 실행하는 동작이 가장 중요하다.이를 위해서 게임 클래스에서 받은 options객체값을 통해 drawImage()를 구현하는 부분이 핵심이다.
class Sprite{
constructor(options){
this.context = options.context;
this.image = options.image;
}
update(dt){ //시간변화에 따른 이미지 움직임 구현
this.currentTime = dt;
}
render(){ //캔버스API 구현
this.context.drawImage(this.image, 10, 10, 450, 220);
}
}
가장 기본적인 구현을 위해서 drawImage의 설정값은 this.image를 제외하고 임의로 설정했다. 최소구현 단계에서는 무엇보다도 '구현'이 가장 중요하기 때문에 세부적인 부분은 모두 '살 붙이기'단계로 넘긴다.
Game클래스의 loadJSON에 파이어베이스에 올려놓은 이미지경로를 넣으면 아래와 같은 반복되는 이미지가 그려져 있는 이미지를 로딩할 수 있다. 여기까지 진행을 마무리했다면 '살 붙이기' 단계에서는 시간변화(프레임)에 따라 특정동작이 그려진 영역만 캔버스에 표현하는 작업을 진행한다.
'웹개발 자료실 > 웹게임제작 Code' 카테고리의 다른 글
캔버스 루프 애니메이션 3편 『프레임 계산』 (0) | 2020.02.05 |
---|---|
캔버스 루프 애니메이팅 2편 『canvas 좌표축 이해』 (1) | 2020.02.04 |
캔버스 게임만들기 3편『state구현하기』 (0) | 2019.11.04 |
캔버스 게임만들기 2편『멀티 이미지 넣기』 (0) | 2019.11.02 |
캔버스 게임만들기 1편『캔버스에 이미지 띄우기』 (0) | 2019.10.24 |
댓글