메뉴 건너뛰기

창작에 관련된 질문이나 간단한 팁, 예제를 올리는 곳

9편 : 체계화된 동작

지금까지 열심히 이 강의를 따라오신 분들이라면
지금 게임의 여러 버그가 산재해 있음을 깨달으셨을겁니다.
그걸 왜 그냥 내버려 뒀나면... 귀찮아서~
~는 훼이크고 일단 가르치는 주제에서 벗어나면 집중력이 떨어지고
어느 정도 이해력이 있으신 분들이면 스스로 고쳤을 거라 봤기 때문입니다.

어쨌든 오늘은 그동안 미뤄왔던 동작 체계의 버그를 모두 고쳐봅시다.
공중을 밟고 계속 점프하고, 점프 중에 발차기하면 바닥에 갑자기 나타나고
말도 아니었죠...

일단 동작을 잘 생각해 봅시다.
현재는
평상시 = 0
펀치 = 1
킥 = 2
점프 = 3
식으로 동작이 정해져 있습니다.
그리고 키를 누르면 무조건 그 상태로 넘어가죠.

하지만 동작에는 계층 구조를 둬야 된다고 봅니다.
펀치를 날리면서 달려나가는 것은 좀 아니라고 보지 않습니까?
물론 펀치 치면서 나가는 게임도 괜찮지만 게임 밸런스 상
공격할 때는 이동 하지 않는 게 좋다고 봅니다.
공격하면서 좌, 우 키를 흔들면 이거 뭐... 사방 공격이고...

그럼 점프 중에 펀치나 킥은?
일단 지금의 점프로 봤을 때 공중 체류시간이 짧은 데다가 복잡하니
점프 중에는 이동만 가능하게 합시다.
즉, 방향을 움직일 수 있게.

그리고 하나 더 추가하고 싶은 상태가 있습니다.
지금은 평상시라고 되어 있지만
평상시는 이동중+가만히 있는 중이 합쳐진 것이죠.
왜 이걸 구분하려고 하냐면
가만히 있는데도 다리를 움직이면 어색하기 때문입니다.

자, Control 부분을 대폭 고쳐봅시다.
어디가 어떻게 변했는지 표시하기 난감하니 그냥 봐주세요.

void Unit::Control()
{
      if(state==1 || state==2)return;

      if(state!=3)
      {
            //펀치
            if(GetKey(vkey_z))
            {
                  state=1;
                  delay=0;

                  if( left_punch)left_punch=false;
                        else left_punch=true;

                  return;
            }
            //킥
            else if(GetKey(vkey_x))
            {
                  state=2;
                  delay=0;

                  return;
            }
            //점프
            else if(GetKey(vkey_c))
            {
                  state=3;
                  jump_power=100;
            }
      }

      bool move = false;

      //상하 이동
      if(GetKey(vkey_up,0))
      {
            y-=speed;
            move = true;
      }
      else if(GetKey(vkey_down,0))
      {
            y+=speed;
            move = true;
      }

      //좌우 이동
      if(GetKey(vkey_left,0))
      {
            x-=speed;
            move = true;
      }
      else if(GetKey(vkey_right,0))
      {
            x+=speed;
            move = true;
      }

      //if(!move)state = 0;
}

일단 곳곳에  else가 추가되었습니다.
위로 이동하면서 아래로 이동하는 건 이상하죠.
그러므로 위, 아래 중 한 군데만 이동할 수 있도록
if(Getkey(vkey_down,0)) 앞부분에 else가 붙었습니다.
마찬가지로 좌우 이동도 한 군데만 가능하도록 else.

그 다음은 move라는 bool 변수를 썼습니다.
아까 전에 말했던 이동시와 이동을 하지 않을 때를 구분하기 위해서입니다.
이동하는 키를 눌렀을 때는 move가 true가 되고, 아무것도 안 눌렀을 때는 초기값 false로 남습니다.

밑에 주석해 놓은 if는 나중에 이야기하기로 하고, 위로 가봅시다.

펀치, 킥, 점프도 한번에 하나만 입력할 수 있게 해 두었습니다.
그리고 맨 위에는

if(state==1 || state==2)return;

를 넣어서 킥이나 펀치를 하고 있을 때는 아무 키도 입력받지 않고
바로 return 해버리게 해 두었습니다.
점프(state == 3)의 경우는 조금 특별한 것이 점프 중에 이동은 가능해야 하므로
점프 중에는 바로 return을 시키지 않고 펀치, 킥, 점프 키 입력만 받지 않게
if(state != 3) 이라는 조건식을 넣었습니다.
점프 상태가 아닐 때만 펀치, 킥, 점프를 할 수 있게 한 거죠.

이제 다시 마지막 주석(//)을 달아놓은 if를 봅시다.
왜 제가 저렇게 했냐면 만들다가 문제를 발견했기 때문입니다.
가만히 있을 때와 이동할 때를 구분하려고 했는데
현재 평상시는 0이기 때문에 숫자 하나가 더 필요한 거죠.
그러니 이동을 뜬금없이 4번 상태로 하려니 왠지 찜찜하고(결벽증?)
이동을 1로 하고 나머지를 하나씩 밀려고 하니 귀찮은 거죠.

뭔 소리를 하는고 하니 매크로의 유용성을 설명하고자 일단 여기서 멈춘 겁니다.
지금 현재 알아보기 힘들게 되어 있는 state==3 같은 것을 매크로를 사용해서 알기 쉽게 바꿔 봅시다.

#define STATE_NORMAL 0
#define STATE_PUNCH  1
#define STATE_KICK  2
#define STATE_JUMP  3

다음과 같이 코드 윗부분에 선언을 해주고
아까 전 코드는 다음과 같이 바꿉니다.

void Unit::Control()
{
      if(state==STATE_PUNCH || state==STATE_KICK)return;

      if(state!=STATE_JUMP)
      {
            //펀치
            if(GetKey(vkey_z))
            {
                  state=STATE_PUNCH;
                  delay=0;

                  if(left_punch)left_punch=false;
                        else left_punch=true;

                  return;
            }
            //킥
            else if(GetKey(vkey_x))
            {
                  state=STATE_KICK;
                  delay=0;

                  return;
            }
            //점프
            else if(GetKey(vkey_c))
            {
                  state=STATE_JUMP;
                  jump_power=100;
            }
      }

      bool move = false;

      //상하 이동
      if(GetKey(vkey_up,0))
      {
            y-=speed;
            move = true;
      }
      else if(GetKey(vkey_down,0))
      {
            y+=speed;
            move = true;
      }

      //좌우 이동
      if(GetKey(vkey_left,0))
      {
            x-=speed;
            move = true;
      }
      else if(GetKey(vkey_right,0))
      {
            x+=speed;
            move = true;
      }

      //if(!move)state = STATE_NORMAL;
}

코드를 이해하기 한결 편해졌죠?
하지만 매크로의 장점은 가독성을 높이는 경우 외에 또 하나 있습니다.
아까 전에 말한 값의 변경의 경우 중간에 새로운 값이 하나 추가되는 경우
모든 코드를 손 볼 필요 없이

#define STATE_NORMAL 0
#define STATE_MOVE 1
#define STATE_PUNCH 2
#define STATE_KICK  3
#define STATE_JUMP  4

선언 부분에서 살짝 고치는 것만으로 손쉽게 수정이 가능합니다.
주의 할 것은 Control 부분 외에도 state를 쓰던 부분이 있으니
그 쪽도 모두 0, 1, 2, 3 같은 자신만 아는 숫자가 아닌 매크로로 바꿔 주십시오.

이제 주석 처리 해놓았던 부분을 제대로 만들어 봅시다.

if(state!=STATE_JUMP)
{
      if(!move)state = STATE_NORMAL;
      else state = STATE_MOVE;
}

일단 점프 중에 이동할 수 있습니다.
점프 중에 이동을 하거나 안 하는 것에 의해서 상태가 변하면 안 되니
점프 중에는 점프 상태가 유지 되도록 점프가 아닐때만 상태를 변화시킵니다.

만약 이동이 없었다면 평상시 : if(!move)state = STATE_NORMAL;
그렇지 않은 경우(이동한 경우)는 이동 상태 : else state = STATE_MOVE;

이걸 실행시켜보면 아마 주인공이 이동할 때 투명인간이 되는 것을 볼 수 있을 겁니다.
당연히 이동시의 상태에 대한  Action을 정의하지 않았기 때문입니다.
이번엔 Action을 바꿔 봅시다.

//평상시
if(state==STATE_NORMAL)
{
      jdd->DrawPicture(backbuffer, "걷기2", x, y-2, NULL);

      ++delay;
      if(delay >= 120)delay=0;

      attack.left = -999;
}
//이동
else if(state==STATE_MOVE)
{
      if(delay<30)jdd->DrawPicture(backbuffer, "걷기2", x, y-2, NULL);
            else if(delay<60)jdd->DrawPicture(backbuffer, "걷기1", x, y, NULL);
            else if(delay<90)jdd->DrawPicture(backbuffer, "걷기3", x, y, NULL);
            else jdd->DrawPicture(backbuffer, "걷기1", x, y, NULL);

      ++delay;
      if(delay >= 120)delay=0;

      attack.left = -999;
}
//펀치
else if(state==STATE_PUNCH)
{
      if(left_punch)jdd->DrawPicture(backbuffer, "펀치1", x-8, y-2, NULL);
            else jdd->DrawPicture(backbuffer, "펀치2", x-8, y-2, NULL);

      ++delay;
      if(delay>=10)state=STATE_NORMAL;

      SetRect(&attack, x-8, y+16, x+8, y+30);
}
//킥
else if(state==STATE_KICK)
{
      jdd->DrawPicture(backbuffer, "킥", x-6, y, NULL);

      ++delay;
      if(delay>=10)state=STATE_NORMAL;

      SetRect(&attack, x-6, y+30, x+18, y+44);
}
//점프
else if(state==STATE_JUMP)
{
      jdd->DrawPicture(backbuffer, "걷기1", x, y-z, NULL);

      z+=jump_power/10;
      jump_power-=GRAVITY;
     
      //착지
      if(z <= 0)
      {
            z=0;
            state=STATE_NORMAL;
      }

      attack.left = -999;
}

전에 없던 이동 동작이 추가되었습니다.
하지만 사실상 새로 추가된 이동 동작 = 저번의 평상시 동작 이고,
평상시에서 애니메이션을 없앴을 뿐입니다.
이제 실행시켜 보면 이동시에는 발을 움직이고, 가만히 있을 때는 멈춰있는 주인공을 볼 수 있을 겁니다.
조금 더 노력하면 가만히 있을 때는 어깨를 들썩이거나 하는 애니를 넣으면 좋겠죠.

적이 멈춰서 쫒아오는 것은 일단 무시합시다.
AI를 손보는 것은 나중에...

 

조회 수 :
2889
등록일 :
2008.05.17
00:15:52 (*.239.144.2)
엮인글 :
게시글 주소 :
https://hondoom.com/zbxe/index.php?mid=study&document_srl=199648
List of Articles
번호 제목 글쓴이 조회 수sort 추천 수 날짜 최근 수정일
165 멘탈붕괴의 절정을 부르는 파일입출력 노루발 273   2013-09-17 2013-09-17 08:18
Love2D의 love.filesystem.load는 훼이크입니다. 파일을 로드하는 것 같지만 사실 lua 파일을 불러옴. love.filesystem.read도 그리 믿음직하지 못함. (고정 크기의 파일만 읽어옴) 우리가 믿을 수 있는 최후의 보루는 love.filesystem.lines 이었던 것입니다....  
164 [Lua] Split 노루발 361   2013-09-17 2013-09-17 08:16
상황을 가정해보자. 웹 서버에 부탁해서 유희왕 덱의 리스트를 가져왔다. 원문은 대략 이렇다고 가정하자.   셰이프스내치x3 모린팬x3 트렌트x3 ...    보기엔 이렇지만 이걸 가져오면 아래와 같이 뜰 것이다. (웹 서버니까 HTML로 줌) 셰이프스내치x3<br>모...  
163 Love2d 게임 중간에 광고 표시 [1] 노루발 377   2015-11-12 2015-11-17 00:16
http://love2d.org/forums/viewtopic.php?f=11&t=81224  
162 김프로 이미지 맵 만들기 노루발 393   2015-11-11 2015-11-11 08:05
https://docs.gimp.org/en/plug-in-imagemap.html  
161 love.graphics.print 한국어 출력 노루발 431   2013-09-17 2013-09-17 08:24
Love2D 사용 시 한국어가 ㅁㅁㅁ이나 ??? 등으로 깨져 나와 해결책을 수소문하여 겨우 찾았다. 원래 중국어 등을 출력하기 위한 것이었으니 적절한 폰트를 쓴다면 한자나 히라가나등도 출력할 수 있을 것이다. function love.load() korfont = love.graphics....  
160 턴 기반 시스템 구현에 대한 글 [4] 노루발 440   2020-11-14 2020-11-18 05:49
플레이어와 NPC들의 모든 행동에 1턴이 소요된다면 턴 기반 시스템의 구현이 쉽겠지만 (사실 엄청 쉽지는 않다... 게임엔진은 실시간으로 돌아가는데 행동은 턴으로 제약해야 하니) 전략성을 요구하기 위해 행동에 소모되는 턴을 다르게 설정한다면 다소 생각...  
159 cocos2d-x CCMenuItem 자신을 지웠을 때 생기는 에러 똥똥배 442   2013-09-13 2013-09-13 07:28
CCMenuItem으로 메뉴를 생성하고 타겟과 셀렉터를 정해서 그 명령 안에서 자신을 지우게 되었을 경우 activate() 안에서 에러가 발생한다. 아직 activate()를 실행해야 되는데 remove 당했기 때문에 생긴 문제인데, 구글에서 검색해 본 결과, https://github.c...  
158 Love2D를 사용하기 위한 Lua 강좌: 변수형 노루발 446   2013-09-17 2013-09-17 08:23
안녕하세요, 노루발입니다. 저번에 변수에 대해 다뤄보았습니다. 변수는 값을 담는 그릇이라고 했습니다. 그리고 루아는 변수 안에 있는 값이 숫자던 문자던 그냥 변수로 취급한다고 했고 "몇개를 살까요?" 라는 질문에 "ㄹ개" 라고 답할 수도 있다고 했습니...  
157 [Love2D] 일단 뭔가 해보자 노루발 457   2013-09-17 2013-09-17 08:27
감기를 딛고 다시 재기... Lua로 초 간단한 원시적인 게임을 만들려고 합니다. 뭐냐하면 소코반이라고, 이렇게 말하시면 모르지만 푸쉬푸쉬라면 아시는 분들이 엄청 많으실겁니다. 이걸 그래픽조차 씌우지 않은 아스키로. ########## #    ..      # #    o  o...  
156 Love2D? 개요, 설치, 실행. 노루발 465   2013-09-17 2013-09-17 08:20
love2d.org/Love2D 홈페이지. Lua로 2D 게임을 만드는 프레임워크입니다. 무료고, 오픈 소스이며, 윈도우, 맥, 리눅스에서 동작하며, 안드로이드에서 동작하게 하는 논의도 활발하다 합니다. 그럼 일단 받아봅시다. 아무리 좋아도 내 컴퓨터에 못 깔면 소용없...  
155 Love2d 여러 플랫폼으로 빌드 자동화 노루발 470   2014-11-12 2014-11-12 17:35
https://github.com/MisterDA/love-release/blob/master/README.md http://www.ambience.sk/lua-love2d-game-distribution/  
154 PHP로 웹게임 만드는 영상 [1] 노루발 472   2021-06-25 2022-01-28 03:40
Simple PHP Strategy Game - YouTube  
153 cocos2d-x에서 schedule_selector 정의 변화 [1] 똥똥배 522   2013-08-19 2013-09-13 07:28
예전에 적은 글에는 터치와 업데이트를 활성화 시키는 법을 적어놨다. http://hondoom.com/zbxe/index.php?mid=study&document_srl=385031 이글은 cocos2d-1.0.1-x-0.9.1를 기준으로 적은 글인데, 최신버전으로 오면서 update의 정의에 대해서 변화가 생겼...  
152 Love2D: 콜백 함수들 노루발 538   2013-09-17 2013-09-17 08:25
Love의 콜백 함수들은 love.run(가장 메인이 되는 겁니다.)에 의해 호출되며, 많은 작업들을 수행합니다. 이것들이 없다고 게임이 실행되지 않는다거나 하지는 않지만 좋은 게임을 만들기 위해선 이것들을 적재적소에 활용해야 합니다. 이미 아시는 분들도 있...  
151 cocos2d-x 게임을 iOS에 이식할 때 생기는 문제들 똥똥배 543   2013-08-26 2013-09-13 07:28
1. 윈도우 한정 명령어는 사용할 수 없다. 너무도 당연한 문제. 2. 한글 인코딩 문제 최신 VS을 사용하면 겪지 않을 문제일지 모르겠지만, 보통은 h나 cpp를 ANISI형식 문서로 만들 것이다. XCode에서는 Unicode나 UTF-8 형식을 쓰므로 소스에 박아둔 한글은 ...  
150 Love2d 안드로이드 게임 패키징하기 [3] 노루발 548   2014-12-15 2021-01-11 12:11
이 문서는 개발 환경이 갖추어져 있는 상태이고, 빌드를 무사히 마친 뒤라고 가정합니다. 또한 이 문서는 https://bitbucket.org/MartinFelis/love-android-sdl2/wiki/Game%20Packaging#markdown-header-how-to-package-the-apk-with-your-own-love-090-game ...  
149 Love2d 안드로이드 빌드하기 노루발 572   2014-12-15 2014-12-15 00:59
Love2d의 안드로이드 포트는 알파 단계입니다. 차차 개선되어 나가긴 하겠지만 아직 불안정한 부분이 많으며, 일어날 수 있는 오작동과 그로 인한 결과는 일절 책임지지 않습니다.   이 문서는 Windows 사용자 기준입니다. Linux나 Mac을 사용할 정도의 내공...  
148 구글 인앱 구매 Soomla로 구현해본 후 팁 똥똥배 577   2014-09-20 2014-09-20 18:45
1. 일단 알파 테스트라도 앱을 출시시켜야 인앱이 작동한다. 그리고 APK 업로드 후에 바로 인앱이 적용되지 않는다. 구글에 게시되는 데, 시간이 걸리므로 인앱 등록 - APK 등록을 마친 후 다음날부터 구현하는 게 깔끔하다. 이걸로 모르고 하루종일 왜 안되나...  
147 Love2D를 사용하기 위한 Lua 강좌: 코딩 스타일, 변수 노루발 585   2013-09-17 2013-09-17 08:22
안녕하세요, 노루발입니다. 어제는 Lua에 대해 대강 알아보고 입력과 출력을 다뤄봤습니다. 출력은 print([출력할 것]), 입력은 io.read()였습니다. 일단 본격적인 프로그램 작성에 앞서 Lua의 코딩 스타일에 대해 알아보겠습니다. 코드를 작성하는 것을 코딩...  
146 소코반: 맵을 초기화하자. 노루발 588   2013-09-17 2013-09-17 08:27
8*8 맵의 데이터가 담길 2차원 배열(테이블)을 초기화하는 코드이다. Map = {} for a = 1, 8 do     Map[a] = {}     for b = 1, 8 do         Map[a][b] = 0     end end Map 테이블의 [1][1]부터 [8][8]까지 모두 0이라는 값이 담기고 Map[0][0]이나 Map[1][...