메뉴 건너뛰기

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

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
번호 제목 글쓴이 조회 수 추천 수sort 날짜 최근 수정일
165 클릭 문제.... ㄱ- [1] 장펭돌 1801   2008-01-31 2008-03-17 04:37
~클릭 100, 123, 0, 276, 186 @계속 ^100 이벤트 발생! 그림판에서 좌표수를 확인하고나서 123,0 부터 276,186의 위치를 위 처럼 정한뒤 흥크립트 실행후 아무리 클릭해봐도 안되네요... 또하나 궁금증은 창모드가 아닌 전체화 모드에선 마우스가 안나오는데.....  
164 [re] 흥크립트 질문 똥똥배 1782   2008-02-01 2008-03-17 04:37
조금 복잡하지만 이렇습니다... ====================================== [분기] ([레벨]==16)?91:[분기] [분기] ([레벨]==15)?92:[분기] [분기] ([레벨]==14)?93:[분기] [분기] ([레벨]<=13)?94:[분기] @이동 [분기] ^91 당신의 랭크 : S @이동 95 ^92 당신의...  
163 흥크립트 질문 [1] 장펭돌 1733   2008-02-01 2008-03-17 04:37
흥크립트에서 if처럼 변수1이 1이면 어떻게되고 변수1이 2면 어떻게되고 이렇게 어떻게하나요? 역전심판에서 1+1 답 얘기할때, 정답을 말하느냐 오답을 말하느냐에 따라서 달라지는것 처럼 말이죠... 봐도 잘 모르겠네요... * 똥똥배님에 의해서 게시물 이동되...  
162 흥크립트, 이건 답변 안해주신듯... [1] 장펭돌 1669   2008-02-02 2008-03-17 04:37
마우스를 지원하게 하고싶은곳과 지원 안하게 하고싶은곳... 이 두곳을 어떻게 구분하죠? 저같은경우엔 특별한 경우를 제외하곤 마우스가 안나오게 할껀데... 어떻게 해야... 하는지 자세히 설명좀 부탁... ㅈㅅ.. * 똥똥배님에 의해서 게시물 이동되었습니다 ...  
161 아아, 그렇다면... 흥크립트에서 [1] 장펭돌 1712   2008-02-02 2008-03-17 04:37
제가 타이틀을 만들고 나서 @마우스지원 해놓고 나니까 선택지에서 키보드로 선택할수 있는 커서가 사라지더라구요? 이건 원래 그런건지... 키보드로 선택해도 되긴 되는데 커서가 없어서 지금 뭘 고르고 있는건지 알수가 없는데.. 원래 이런건가요? 으음, 그...  
160 흥크립트 AVI 지원에서... [2] file 장펭돌 1823   2008-02-02 2008-03-17 04:37
 
159 흥크립트 기본 바꾸기. [3] 장펭돌 1855   2008-02-03 2008-03-17 04:37
흥크립트 마우스 포인터를 바꾸려는데, system 의 "커서"를 바꾸면 된다구 하셨짢습니까? 그럼, "DATA\XXX.mlc" 이렇게 해야 되는겁니까, 그리구, 기본 커서 크기좀 알려줏메. ps. 또 문제가 생김... 왠일인지 동영상이 그냥 곰플레이어로 재생하면 멀쩡한데, ...  
158 흥크립트에대해. [16] 궤레브 1711   2008-02-06 2008-03-17 04:37
음.. 그냥 몇가지 간단한 질문인데요 그림과 그림이 맞닿앗을때 라는 액션을 만들수 있나요? 만약 그림과 그림이 맞닿앗을때 액션이 없으면 턴알피지밖에 불가능할거같은데 마우스로 눌러서 공격하는 버츄얼캅같은 게임이 아니고선.. 이건 제가 직접 실험해볼...  
157 [re] 흥크립트 질문.. [2] 똥똥배 1797   2008-02-07 2008-03-17 04:37
>흔히 알만툴에서 쓰이는 >조건분기를 어떻게 쓰나요? 펭돌님 말로는 if문이 있다던데.. >그리고.. >흥크립트 테스트인가? 슬라임같이 생긴 사람이 >돌아다니면서 사람들 만나면 대화 나오게 만드신게 있더군요? >거기서처럼 그냥 아무 추가 이벤트가 없어도 ...  
156 흥크립트 질문.. [3] 궤레브 1747   2008-02-07 2008-03-17 04:37
흔히 알만툴에서 쓰이는 조건분기를 어떻게 쓰나요? 펭돌님 말로는 if문이 있다던데.. 그리고.. 흥크립트 테스트인가? 슬라임같이 생긴 사람이 돌아다니면서 사람들 만나면 대화 나오게 만드신게 있더군요? 거기서처럼 그냥 아무 추가 이벤트가 없어도 꺼지지 ...  
155 [re] 흥크립트 개선점 [3] 똥똥배 1928   2008-02-09 2008-03-17 04:37
1. 글자변수 외에 숫자변수도 화면에 표시할 수 있으면 좋겠음 이건 내부적인 문제이기도 한데, 화면에 표시할 때는 위치정보, 색 등 엄청난 데이터가 필요합니다. 글자는 길이가 있고 하니 어차피 덩치가 커서 상관없다고 생각했지만, 숫자변수는 4바이트면 ...  
154 흥크립트 개선점 흑곰 1578   2008-02-08 2008-03-17 04:37
글자변수 외에 숫자변수도 화면에 표시할 수 있으면 좋겠음 글자입력시 아무 것도 안썼으면 다시 입력받을 수 있도록 하면 좋겠음 글자변수도 조건문을 할 수 있으면 좋겠음 if문이 있으면 좋겠음 아이콘 변경이 되었으면 좋겠음(실은 이건 컴파일 문제이지만)...  
153 Divx6를 받아서 인코딩을 하려는데... 에러뜸... [3] 장펭돌 2154   2008-02-09 2008-03-17 04:37
Divx 공식 홈페이지를 가보니, 6가 있길래 받아서 동영상파일 (AVI)를 열면 이렇게 에러가 뜨는군여... ----------------------------------------------------------------- Invalid VIdeo Stream DivX Converter cannot encode the in put file: C:\Document...  
152 조건분기할 때 [5] 흑곰 1745   2008-02-12 2008-03-17 04:37
&& and || or == 같다 != 같지 않다 원래 vc++ 에서 쓰는 것인데 흥크립트에도 쓰여요. (마지막 것은 확인 못해봤음) 흥크립트도 조건문을 활용하면 if문이 가능합니다. if문을 여러개 구현하다보면 그렇게 빠르진 않은 것 같습니다만 * 똥똥배님에 의해서 게...  
151 흥크립트 if-else 문 흉내내기 [4] 똥똥배 1933   2008-02-12 2008-03-17 04:37
이곳 저곳으로 이동다니면 나중에 스크립트를 읽을 때 이해하기 어려우므로 일직선으로 가는 흐름이 이해하기 쉽습니다. 그래서 C에서는 goto는 금기하고 if-else, while등을 씁니다. 뭐, 서론은 이 정도로 하고 흥크립트에서 if-else 문을 흉내내려면 //첫 번...  
150 흥크립트 win모드와 전체화면의 속도 차이 있나요? [4] 흑곰 1698   2008-02-13 2008-03-17 04:37
추억의 냉무 * 똥똥배님에 의해서 게시물 이동되었습니다 (2008-03-11 14:06)  
149 질문! [4] file DOH 1616   2008-02-14 2008-03-17 04:37
 
148 흥크립트 AVI 재생.. [2] 장펭돌 1851   2008-02-14 2008-03-17 04:37
흥크립트 AVI 재생시에 가끔 전체화면으로 게임을 할때, 재생하는부분에서 갑자기 창모드 화면 크기로 영상이 나온 다음 전체화면인 게임은 아래로 최소화 되어 있을때가 가끔있는데요... (이것도 어떨때는 그러고 또 어떤때는 안그럼) 이거 뭐 대체 왜이러는...  
147 흥크립트 원인불명 버그... [3] file 네모상자 738   2008-02-16 2008-03-17 04:37
 
146 그러고보니, 흥크립트에서.. [1] 장펭돌 1693   2008-03-10 2008-03-17 04:37
왠지는 모르겠으나, 저희집 컴퓨터에서는 효과음은 들리지만, 배경음악이 안들렸었습니다.. (엉? 왠지 예전에도 말한적 있는것 같기두 하고...) 아무튼, 어째서 그런겁니까...? 사실 그래서 공룡돌의 추리쩡 배경음악.. 저는 들을수가 없어서 다른분들한테 들...