메뉴 건너뛰기

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

4편 : 애니메이션2

여러분 혹시 그거 아십니까?
지금까지 가르쳐 드린 것은 비효율적인 코딩이란 것을...
하지만 효율적으로 짜면 이해도가 떨어지기 때문에 풀어서 썼습니다.
최적화는 모든 것을 이해한 다음에 하기로 하죠.

그럼 애니메이션에 대해 이야기 하기 전에
저번에 했던 애니메이션에서 아주 사소한 부분을 수정하겠습니다.

걷기 그림을 보면 걷기2.gif만이 세로 크기가 29입니다.
이것을 그냥 0,0에 출력했는데 사실 이 그림은 바닥을 중심으로 해서 그려놓은 거라
걷는 모습이 약간 부자연스럽게 보였습니다.

if(delay<30)jdd->DrawPicture(backbuffer, "걷기2", x, y-1, NULL);

다음과 같이 걷기2일 경우는 y좌표를 1을 뺀 상태에서 출력하게 합시다.
조금 어깨를 들썩이는 것 같아서 괜찮아 보이죠.
그래도 어색하기는 하지만 일단 넘어가고...

현재는 아무 입력없이 걷기 애니메이션만 하고 있지만
상황에 따라 여러가지 애니메이션을 하도록 바꿔 봅시다.
일단 애니메이션 상태를 기억할 변수 하나를 만듭니다.

int state = 0;

이것을 임의로
0 : 기본 애니메이션
1 : 펀치
2 : 킥

이라고 생각합시다.
그렇다면 일단 평상시에는 0입니다.
펀치에 해당하는 키(임의로 Z를 쓰겠습니다.)를 누르면 1의 상태가 됩니다.
하지만 펀치는 누르는 동안 계속 내밀고 있으면 안 되니까
내민 후 일정 시간이 지나면 자동으로 0의 상태로 돌아가게 합니다.
킥도 펀치와 마찬가지.
이것을 코드로 표현해 볼까요?

int main(char* arg[])
{
     //창 생성
     if(!MainInitialize("Sample", TRUE, FALSE, window_mode))return 0;

     //윈도우창 이동
     if(window_mode)
     {
          jdd->OnMove(100, 100);
          SetCursor(LoadCursor(0, IDC_ARROW));
     }

     JPictureInfo jpi;
     jpi.SetColorKey(JColor(0,0,255));

     jdd->LoadPicture("배경", "던젼6.jpg", NULL, true);
     jdd->LoadPicture("걷기1", "w1.gif", &jpi, true);
     jdd->LoadPicture("걷기2", "w2.gif", &jpi, true);
     jdd->LoadPicture("걷기3", "w3.gif", &jpi, true);
     jdd->LoadPicture("펀치1", "p1.gif", &jpi, true);
     jdd->LoadPicture("펀치2", "p2.gif", &jpi, true);
     jdd->LoadPicture("킥", "k1.gif", &jpi, true);

     int x = 100;
     int y = 100;
     int delay = 0;
     int state=0;

     //메인 실행
     while(!GetKey(vkey_esc))
     {
          if(!ProcessMessage())break;

          if(GetKey(vkey_up,1))y-=5;
          if(GetKey(vkey_down,1))y+=5;
          if(GetKey(vkey_left,1))x-=5;
          if(GetKey(vkey_right,1))x+=5;
          if(GetKey(vkey_z))
          {
               state=1;
               delay=0;
          }
          if(GetKey(vkey_x))
          {
               state=2;
               delay=0;
          }

          if(x<0)x=0;
          if(x>SCREEN_X-19)x=SCREEN_X-19;
          if(y<0)y=0;
          if(y>SCREEN_Y-29)y=SCREEN_Y-29;

          jdd->DrawPicture(backbuffer, "배경", 0, 0, NULL);

          //평상시
          if(state==0)
          {
               if(delay<30)jdd->DrawPicture(backbuffer, "걷기2", x, y-1, 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;
          }
          //펀치
          else if(state==1)
          {
               jdd->DrawPicture(backbuffer, "펀치1", x-4, y-1, NULL);

               ++delay;
               if(delay>=10)state=0;
          }
          //킥
          else if(state==2)
          {
               jdd->DrawPicture(backbuffer, "킥", x-3, y, NULL);

               ++delay;
               if(delay>=10)state=0;
          }

          jdd->Render();
     }

     //정리하고 끝내기
     jdd->DeleteFont(global_font);

     return 0;
}

제대로 작성하셨다면
Z나 X키에 의해 펀치, 발차기를 날리는 주인공을 볼 수 있습니다.




그럼 새로 추가된 코드를 하나하나 봅시다.
그림 불러 오는 것과 state 변수는 설명할 필요가 없겠죠.
다음은 키 입력 부분.

if(GetKey(vkey_z))

Z키를 누르면 작동이 되게 해두었습니다.
여기서 두번째 파라미터가 설정되어 있지 않기 때문에
방향키와 다르게 한 번 누르면 한번만 작동하게 됩니다.
아무리 누르고 있어도 두번 공격하지 않는 것이죠.

Z키를 누르면 어떤 일이 일어나는지 봅시다.

state=1;
delay=0;

state를 1로 만듭니다. 그리고 delay도 0으로 만드는 군요.
왜 delay를 0으로 만드냐면 뒤에 보면 아시겠지만 10을 센 후
시간이 지나면 원래 동작으로 복귀하기 위해서입니다.
만약 0으로 초기화 시키지 않았을 때 delay가 7이라면
10/100초 동안 동작이 지속되지 않고 3/100초 만에 끝나게 될 겁니다.

=====================================================
추가 팁 - 다양한 키 입력
GetKey에서 Z키는 vkey_z로 정의되어 있습니다.
X는 vkey_x로 정의되어 있습니다.
그럼 다른 키를 쓰기 위해서 어떻게 해야 할까요?
바로 vkey_a ~ vkey_z 식으로 알파벳 키는 모두 입력 받을 수 있습니다.
덧붙여 숫자키는 vkey_0~vkey_9로 숫자키를 입력받을 수도 있습니다.
하지만 나머지 키들은 vkey_enter, vkey_esc 같은 특유의 이름이 있으니
쓰려면 donglib.h에 있는 정의 값을 봐야 할 겁니다.
======================================================

그 다음으로 애니메이션 부분은 확 달라졌다는 것을 아실겁니다.
하지만 자세히 보면 그전에 있던 애니메이션이

if(state==0)
{
}

이 안으로 들어가고 나머지 상태에 대한 정의가 추가된 것입니다.
일단 평상시는 지금까지 했으니 펀치를 봅시다.

그림출력은 x좌표를 x-4로 출력하는 것 빼고는 별 것 없습니다.
4를 빼서 출력하는 이유는 펀치의 동작이 다른 그림보다 가로가 4픽셀 크기 때문입니다.
만약 주인공이 오른쪽을 보고 있다면 상관없지만, 지금은 왼쪽을 바라보고 있습니다.
그러므로 주먹을 내미는데 출력좌표를 똑같이 0,0으로 한다면 주먹을 내민만큼 몸이 뒤로 물러나는 것 처럼 보이게 됩니다.
이건 직접 값을 바꿔보며 눈으로 보시면 편이 좋을 거라고 봅니다.


그럼 이걸로 주인공의 공격 애니메이션도 끝났습니다.
하지만 펀치2를 불러놓고 안 쓰면 허전하니 펀치2도 써봅시다.
이건 어떨까요? Z키를 누르면 누를 때마다 교대로 왼팔, 오른팔 식으로 나가는 겁니다.
그럼 일단 지금이 왼팔이 나갈 차례인지 오른팔이 나갈 차례인지를 확인하는 변수를 만듭니다.

bool left_punch = false;

int가 아니고 bool형이군요.
int는 정수형으로 -2147483649~2147483648의 영역을 표현할 수 있다면
bool은 0, 1. 단 두 값밖에 가지지 못 합니다.
스위치처럼 ON, OFF 상태를 기억할 뿐이지요.
보통은 참, 거짓이라고 부릅니다.
위의 선에서는 거짓(false)으로 선언한 것입니다.

이것을 적용하려면 애니메이션 부분과 키 부분을 고쳐야 겠습니다.
일단 키 입력 부분은 이렇게 고칩니다.

if(GetKey(vkey_z))
{
     state=1;
     delay=0;

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

if(left_punch)는 if(left_punch==true)와 같은 뜻입니다.
만약 left_punch가 참이면 거짓상태로 바꾸고(left_punch=false)
아니면(left_punch가 거짓이면) 참 상태로 바꾸라는 거죠.(else left_punch=true)

애니메이션 부분도 마찬가지입니다.

if(left_punch)jdd->DrawPicture(backbuffer, "펀치1", x-4, y-1, NULL);
     else jdd->DrawPicture(backbuffer, "펀치2", x-4, y-1, NULL);

left_punch가 참이면 펀치1을 보여주고, 아니면 펀치2를 보여주는 겁니다.
실행 후 z버튼을 여러번 눌러보면 확인 할 수 있으실 겁니다.
하지만 빨라서 잘 안 보인다면 주먹 애니메이션의 delay>=10 부분의 10의 숫자를 높게 바꿔 봅시다.


조회 수 :
3213
등록일 :
2008.04.20
19:43:00 (*.193.78.73)
엮인글 :
게시글 주소 :
https://hondoom.com/zbxe/index.php?mid=study&document_srl=191151

A.미스릴

2008.04.21
06:19:59
(*.234.10.203)

if(GetKey(vkey_z) && state == 0 )
if(GetKey(vkey_x) && state == 0 )
가 좋지 않을까영
공격이 끝나기 전에 delay가 초기화되서 공격이 연장되는 버그가 있음

똥똥배

2008.04.21
07:20:46
(*.239.144.2)
그게 좋겠죠.
말씀드렸듯이 지금은 비효율적이라도 알기쉽게 가고 있는 중입니다.
실력이 되시는 분은 맘대로 변형해서 쓰십시오.
List of Articles
번호 제목 글쓴이 조회 수 추천 수 날짜 최근 수정일
85 문D라이브로 더블드래곤을 만들자(6) [2] file 똥똥배 3114   2008-04-23 2008-04-25 05:01
 
84 문D라이브로 더블드래곤을 만들자(5) [6] file 똥똥배 3072   2008-04-21 2008-04-24 03:23
 
83 C++ 데이터의 바이트 용량 임의로 정의할수 없나영 [1] A.미스릴 3059   2008-04-21 2008-04-21 07:18
int는 4바이트로 정해져 있는데 약간의 수만 있으면 되는 수도 있는데 괜히 많은 숫자를 사용해서 메모리를 많이 사용하는 건 아닐지 ㅡㅡ; 3바이트라던지 4비트라던지... 데이터의 바이트 사용량을 임의로 바꿀수 없나요  
» 문D라이브로 더블드래곤을 만들자(4) [2] file 똥똥배 3213   2008-04-20 2008-04-21 07:20
4편 : 애니메이션2 여러분 혹시 그거 아십니까? 지금까지 가르쳐 드린 것은 비효율적인 코딩이란 것을... 하지만 효율적으로 짜면 이해도가 떨어지기 때문에 풀어서 썼습니다. 최적화는 모든 것을 이해한 다음에 하기로 하죠. 그럼 애니메이션에 대해 이야기 ...  
81 문D라이브로 더블드래곤을 만들자(3) file 똥똥배 3430   2008-04-18 2008-04-18 18:27
 
80 문D라이브로 더블드래곤을 만들자(2) [6] file 똥똥배 3611   2008-04-18 2013-11-23 08:43
 
79 VC++ 2008 Express Edition에서 문D라이브 링크 [2] A.미스릴 13090   2008-04-17 2014-04-15 17:28
SDK도 최신버전으로 깔아주었고 도구 설정도 적절하게 했습니다 모든 셋팅이 완료되고 링크를 해보니까 아래와 같은 문장이 쭉쭉나오는군요 warning은 굉장히 많이 나오는데 실제로 컴파일을 막아버린 부분은 LINK : fatal error LNK1104: 'LIBC.lib' 파일을 ...  
78 문D라이브로 더블드래곤을 만들자(1) [2] file 똥똥배 4479   2008-04-16 2008-04-17 08:29
 
77 임의의 점이 다각형 내부에 있는지 검사하는 함수 똥똥배 5358   2008-04-14 2008-04-14 02:21
수학도 기하학도 허접한 제가 짠 거라서 허접하지만 넓은 아량으로 봐 주십시오. struct Point { double x; double y; }; bool LineIn(Point p1, Point p2, Point p3) { double top, bottom; //위 아래 범위 결정 if(p1.y < p2.y) { top = p1.y; bottom = p2.y...  
76 명령문 질문드립니다. [1] X-tra 2244   2008-03-26 2008-03-27 00:40
변해서 게시판 찾기 어려울것 같았는데 이상하게 한번에 찾아 버렸네요. 연구소라고 적혀 있어서요... 이번건 질문이라고 하기에는 좀 애매합니다. 예전에 흥크립트의 모든 명령어가 적혀 있던 파일을 본 기억이 납니다. 하지만 아무리 찾아도 없더군요. 블로...  
75 저는 사실 이걸 잘 못해요. [3] 앟랄 2282   2008-03-26 2008-04-18 07:45
알만툴 액알. 원리가 도저히 이해가 안 갑니다. 그림넣어서 설명해주세요. 그 뭐시기 그냥 근접공격만 일단. 나중가서 원거리를 합시다.  
74 OgreOde 사용기 똥똥배 3184   2008-03-25 2008-03-25 21:37
OgreSDK 버전 1.4.7 OgreOde 버전 0.95(아마도) 사실 오우거 엔진 쓴지도 얼마 안 되고 물리엔진은 처음 만져봤습니다. 처음에 Ogre Wiki에 나온대로 따라서 만들었는데 crateCube.mesh와 plane.mesh 때문에 에러가 나서 바닥은 직접 만들고 crateCube 대신 오...  
73 웹 프로그래밍을 배우려고 합니다. [2] Kadalin 2171   2008-03-22 2008-03-22 23:37
사용하는 형식에 따라 꽤 분야가 많은 것으로 알고 있습니다. 각각의 특징을 설명해 주세요.  
72 질문요.C++ [7] 질문자01 2729   2007-05-10 2008-04-03 05:32
도대체 파라미터가 뭔지 모르겠어요.  
71 흥크립트 클릭명령 질문입니다. [1] 카시 2009   2008-03-18 2008-03-18 20:18
으음 제가 몇가지 실험해본 결과. 1. @영역선택과 @클릭 명령은 한 스크립트 내에서 혼합해서 쓸 수 없다. 2. @영역선택은 ~영역선택으로 쓸 수 없다. (대화창과 동시 처리 불가능) 3. @클릭명령 사용시 오른쪽 버튼을 누르면 세이브화면이 뜬다. 이 세가진데...  
70 흥크립트 키입력 질문 [1] A.미스릴 2363   2008-03-15 2008-03-17 04:37
키 클릭을 누르고 누른상태로 있는것도 인식할 수 있나영? 어떻게 하나요 ㅡ,.ㅡ  
69 그림 출력의 순서를 알고 싶습니다. [3] X-tra 2466   2008-03-12 2008-03-17 04:37
제 글이 연구소로 가서 이번에는 신경써서 제목을 적었습니다 ㅡ.ㅡb 채색을 제외하면 전투도 70%정도 완성되었는데 자잘한 버그가 발생해서요. 제 스스로 알아 내려 했지만 확실하게 하기 위해서 물어 봅니다. 정확히 어떤 그림이 앞에 출력이 되고 어떤 그...  
68 '@클릭'이거 어떻게 사용하는거죠? [4] 네모상자 2244   2008-01-26 2008-03-17 04:37
'@영역선택'사용법은 역전심판 뜯어봐서 알 수 있겠는데, 이건 던전 앤 러버를 뜯을 수도 없고 해서 잘 모르겠네요. * 똥똥배님에 의해서 게시물 이동되었습니다 (2008-03-11 17:11)  
67 퀴즈소스입니다 허클베리핀님 라컨 2546   2005-08-23 2008-03-17 04:37
Private Sub Form_Click() If 시작 <= 1 Then MsgBox ("이제 왔군 학생") MsgBox ("이번이 시험인건 알고 있겠지?") MsgBox ("문제를 낼테니 잘 풀어라") 문제 = InputBox("흑곰의 이전 닉네임은 무엇이였나?") If 문제 = "답" Then MsgBox ("맞았군") Cls Pri...  
66 게임 만들때는 게임 기획을 해야 합니다. [7] 똥똥배 2106   2008-02-12 2008-03-17 04:37
당연한 소리지만 간혹 이것을 잘못하는 사람도 있고 저 역시 이런 실수를 많이 저질렀습니다. 마왕이 용사를 무찌르는 게임을 만들자. 이건 게임 기획이 아닙니다. 시나리오 기획입니다. 이 결과 게임이 어떻게 만들어지는지 봅시다. 일단 마왕이 용사를 무찌...