메뉴 건너뛰기

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

3편 : 애니메이션

저번에 팩맨을 설명할 때는 애니메이션을 이야기하려다 그만 뒀는데
지금 이야기하도록 하겠습니다.
뭐, 그때 애니메이션과 지금 애니메이션은 다르지만... 그렇다 치고.

어제까지 제대로 하셨다면 아래와 같은 코드가 되어 있을 겁니다.

 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));

     }


     jdd->LoadPicture("배경", "던젼6.jpg", NULL, true);

     jdd->LoadPicture("걷기1", "w1.gif", NULL, true);

     jdd->LoadPicture("걷기2", "w2.gif", NULL, true);

     jdd->LoadPicture("걷기3", "w3.gif", NULL, true);


     int x = 100;

     int y = 100;


     //메인 실행

     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(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);

          jdd->DrawPicture(backbuffer, "걷기1", x, y, NULL);


          jdd->Render();

     }


     //정리하고 끝내기

     jdd->DeleteFont(global_font);


     return 0;

}

오늘은 눈에 거슬리는 두가지를 고쳐봅시다.
1) 주위의 파란 박스
2) 움직이지 않는 캐릭터

일단 파란박스의 경우는 투명색으로 바꿔서 없애야 합니다.
투명색이란 뭐냐면 화면에 찍을 때 표시하지 않는 색입니다.
즉, 파란 색을 투명색으로 하면 파란 부분은 화면에 표시되지 않기 때문에 캐릭터는 자연스럽게 찍히게 됩니다.

여기서 문라이브의 그림찍기의 중요한 원리를 알고 넘어갑시다.
몰라도 일단 그림은 찍을 수 있지만 나중에 응용에 도움이 될 겁니다.
문라이브에서는 2가지 정보를 이용해서 그림을 출력합니다.

1) 그림 데이터
2) 그림 정보

그림 데이터는 말그대로 눈에 보고 있는 그림이죠.
이건 사실 별로 상관할 것도 아니고 직접 접근하기도 어렵습니다.
중요한 것은 그림 정보(JPictureInfo)입니다.
그림 정보는 어떻게 그림을 찍을까를 정하는 정보입니다.
이 정보는 기본적으로 한 그림당 하나씩 가지지만 보통 설정하지 않으면
투명색 없음, 보통 레이어, 불투명도 100%로 자동 설정됩니다.
즉, 이것을 건드리는 것 만으로 그림 데이터 자체에는 아무 변화없이
화면에 출력되는 그림의 투명색, 레이어 종류, 불투명도를 바꿀 수 있는 겁니다.

그럼 한번 그림 정보를 만들어 봅시다.
그림 로드 부분을 아래와 같이 고칩니다.

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);


jpi가 바로 그림 정보 변수입니다.
다시 변수 이야기가 나오는데 int가 정수형 변수를 선언하는 거라면
JPictureInfo는 문라이브용 그림 정보의 형태로 jpi를 선언하는 겁니다.
물론 int는 기본 제공이지만 JPictureInfo는 프로그래머가 임의로 만들어낸 형식이죠.

그건 그렇다치고 둘째 줄은 SetColorKey를 이용해서 투명색을 지정하고 있습니다.
그 값은 JColor(0,0,255), 즉 RGB값으로 R:0 G:0 B:255 = 파랑색인 셈입니다.

그리고 로딩시에 전에는 NULL로 해두었던 3번째 파라미터에 &jpi라고 써넣으면
그 그림에 그 정보가 적용됩니다.
그림 정보를 넣을 때는 &를 앞에 붙여야 하는데 일단 이건 넘어갑시다.
그냥 붙이세요.

제대로 적었다면 아래와 같이 파란 상자가 없어져서 출력될 겁니다.



이 투명색에 관해서 하나 덧붙일 게 있는데
문라이브는 윈도우 모드에서 투명색이나 투명도에 관해서 선이 남거나 제대로 표시되지 않는 현상이 발생합니다.
그건 윈도우가 32비트 컬러로 설정되어 있을 때 일어나는 현상인데 16비트 컬러로 바꾸면 정상 작동합니다.
원래 문라이브는 전체화면 게임을 기본으로 한 것이기 때문에 전체화면 모드로 바꾸면 이 문제는 전혀 일어나지 않습니다.
하지만 일단 테스트나 만들어 볼 때는 윈도우 모드가 좋으니 이대로 갑시다.

그럼 다음은 애니메이션을 해봅시다.
걷기1, 걷기2, 걷기3 세가지를 불러놓고 걷기1만 썼는데 이젠 다 써보겠습니다.
일단 애니메이션을 하려면 변수 하나가 필요합니다.
x와 y를 선언한 아래에 다음과 같은 변수를 만들어 봅시다.

int delay = 0;

이건 시간 재기 용입니다.
무조건 애니메이션을 1프레임당 한번씩 하면 1초에 100프레임
즉, 100이나 바뀌기 때문에 눈으로 보기엔 뭔가 파바바박 하는 느낌이 되버리죠.
보고 싶으면 해봐도 되지만 일단 이걸로 가겠습니다.

애니메이션 속도는 0.3초로 하겠습니다.
0.3초마다 그림이 바뀌도록 아래와 같이 출력부분을 바꿉니다.

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

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

일단 밑부분 부터 봅시다.
++delay는 delay를 1씩 증가시키란 말입니다.
우리는 delay의 30 간격으로 그림을 바꿀 테니 그림은 총 3장.
즉, 총 90/100초의 애니메이션입니다.
그러므로 90/100초가 넘어간 후에는 다시 처음으로 리셋해줘야 겠죠?

if(delay >= 90)delay=0;
만약 delay가 90보다 크거나 같으면 delay를 0으로 만들어라.

이런 뜻입니다. 사실 ==(같으면)으로 해도 되는데 만약을 대비해서 크거나 같으면(>=)으로 비교했습니다.

그럼 다시 위쪽을 볼까요?
delay가 30보다 작을 때, 즉 0~29인 동안은 걷기1을 보여줍니다.
그리고 delay가 30~59인 동안은 걷기2를 보여줍니다.
마지막으로 그 외의 경우에는 걷기3을 보여줍니다.

여기서 if-else를 모르시는 분이 있을까봐 설명하자면
if가 만약이라면 else는 그렇지 않은 경우를 말합니다.
단순히 else라고 하면 앞의 if에 해당하지 않는 경우 무조건 실행하지만
앞의 경우는 아니지만 또 다른 경우를 검사해보고 싶을 때는 else if가 됩니다.
이런 if-else 구조는 상당히 중요한 것이 숙지하고 계십시오.

제대로 실행하면 그림이 움직이는 것을 볼 수 있을 겁니다.
그런데 뭔가 이상합니다. 다리를 벌리더니 바로 오므려 버립니다.
어색하죠. 왜냐면 걷기3에서 바로 걷기1이 되었기 때문입니다.
자연스럽게 걷는 동작을 하려면

걷기1 - 걷기2 - 걷기3 - 걷기2

순서로 보여줘야 겠죠?
그럼 그렇게 바꿔 봅시다.
아주 간단합니다.

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

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

그냥 그림을 4장이라고 생각하는 겁니다.
똑같은 그림 걷기2를 2번째 4번째에 넣는 것이죠.
전체 시간도 4장이니 120으로 늘렸습니다.

이제 자연스럽게 애니메이션하는 캐릭터가 보일... 아니군요.
이거 원본 그림이 순서가 잘못되어서 걷기2 - 걷기1 - 걷기3 - 걷기1 순으로 출력해야 하는군요.
그럼 그렇게 바꿔서 해보십시오.
자연스럽게 보일 겁니다.

애니메이션 속도도 조절해보면서 이것 저것 해보십시오.
가능하다면 그림 한장 더 넣어서 4장 애니메이션에 도전해보시는 것도?

조회 수 :
3430
등록일 :
2008.04.18
18:21:19 (*.193.78.73)
엮인글 :
게시글 주소 :
https://hondoom.com/zbxe/index.php?mid=study&document_srl=190574
List of Articles
번호 제목 글쓴이 조회 수 추천 수sort 날짜 최근 수정일
85 임의의 점이 다각형 내부에 있는지 검사하는 함수 똥똥배 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...  
84 MFC 더블 버퍼링 질문 [2] A.미스릴 6603   2008-06-13 2013-11-23 08:43
void CPingpongView::OnDraw(CDC* pDC) { CPingpongDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here // 현재 창 크기와 같은 DC를 만듬 CRect rect; GetClientRect(&rect); int int_client_width = (rect.ri...  
83 저는 사실 이걸 잘 못해요. [3] 앟랄 2282   2008-03-26 2008-04-18 07:45
알만툴 액알. 원리가 도저히 이해가 안 갑니다. 그림넣어서 설명해주세요. 그 뭐시기 그냥 근접공격만 일단. 나중가서 원거리를 합시다.  
82 OgreOde 사용기 똥똥배 3184   2008-03-25 2008-03-25 21:37
OgreSDK 버전 1.4.7 OgreOde 버전 0.95(아마도) 사실 오우거 엔진 쓴지도 얼마 안 되고 물리엔진은 처음 만져봤습니다. 처음에 Ogre Wiki에 나온대로 따라서 만들었는데 crateCube.mesh와 plane.mesh 때문에 에러가 나서 바닥은 직접 만들고 crateCube 대신 오...  
81 명령문 질문드립니다. [1] X-tra 2244   2008-03-26 2008-03-27 00:40
변해서 게시판 찾기 어려울것 같았는데 이상하게 한번에 찾아 버렸네요. 연구소라고 적혀 있어서요... 이번건 질문이라고 하기에는 좀 애매합니다. 예전에 흥크립트의 모든 명령어가 적혀 있던 파일을 본 기억이 납니다. 하지만 아무리 찾아도 없더군요. 블로...  
80 웹 프로그래밍을 배우려고 합니다. [2] Kadalin 2171   2008-03-22 2008-03-22 23:37
사용하는 형식에 따라 꽤 분야가 많은 것으로 알고 있습니다. 각각의 특징을 설명해 주세요.  
79 [수정]이거왜이러는거죠;; [4] file 상상악수 3254   2008-08-19 2008-08-21 03:24
 
78 문D라이브로 더블드래곤을 만들자(8) [1] file 똥똥배 2973   2008-05-16 2009-01-07 22:05
 
77 흥크립트에 궁금한점 [4] 상상악수 3835   2008-08-21 2008-08-27 04:13
tile1에서 앞쪽에 18칸말고는 사용할수없는 칸인가요? 그림을 그리면 맵에디터에서는 잘나오는데 게임에서는 까만색에 움직일수없는..  
76 문D 라이브 질문 [5] 대슬 2889   2008-05-15 2008-05-16 06:13
1. 음악이나 사운드는 어떻게 불러와서 출력하나요. 배웠었는데 까먹었음. 2. 그림을 원하는 각도로 자유롭게 회전시켜서 출력하는 기능은 없나요?  
75 문D라이브로 더블드래곤을 만들자(10) file 똥똥배 4065   2008-05-17 2008-05-17 03:07
 
74 또질문 [2] 쿠로쇼우 2854   2008-09-29 2008-09-29 06:43
#include <stdio.h> int main (void) { int i=1,a=1,b=0,c=0,d=0; char ch[50]; char ca=0; FILE * file = fopen("a.txt", "rt"); if(file==NULL){ printf("NULL"); return 1; } while(1) { fgets(ch, 50, file); if(feof(file)!=0){ break;} else { fscanf(fi...  
73 이상하군요. [2] 대슬 2490   2008-05-16 2009-01-07 22:05
MIDI 음악을 메인 루프 앞에서 재생시킨 후 다른 음악을 재생시키려고 했더니 그 다음부턴 음악이 정지만 하고 재생이 되지를 않네요. 예전에 배고파요 만들었을 때는 비슷하게 해도 별 문제가 없었던 것 같은데, 왜 이러는지 잘 모르겠습니다. void bgm(int ...  
72 문D라이브로 더블드래곤을 만들자(9) 똥똥배 2889   2008-05-17 2008-05-17 03:07
9편 : 체계화된 동작 지금까지 열심히 이 강의를 따라오신 분들이라면 지금 게임의 여러 버그가 산재해 있음을 깨달으셨을겁니다. 그걸 왜 그냥 내버려 뒀나면... 귀찮아서~ ~는 훼이크고 일단 가르치는 주제에서 벗어나면 집중력이 떨어지고 어느 정도 이해력...  
71 그럼 질문으로... [1] 쿠로쇼우 3243   2008-09-26 2009-01-07 22:05
c++에서 텍스트파일을 입,출력 하는것은 어느정도 알겠습니다. 문자라든가, 띄어쓰기, <┘ 세고, 출력하는것도 어느정도 알겠고요,, 근데 한줄 단위로 입력하고, 한줄 단위로 출력하는건 어떻해야 하나요? 오늘 아침 질문(??) 올린거에서 텍스트 파일 입력한거 ...  
70 흥크립트 클릭명령 질문입니다. [1] 카시 2009   2008-03-18 2008-03-18 20:18
으음 제가 몇가지 실험해본 결과. 1. @영역선택과 @클릭 명령은 한 스크립트 내에서 혼합해서 쓸 수 없다. 2. @영역선택은 ~영역선택으로 쓸 수 없다. (대화창과 동시 처리 불가능) 3. @클릭명령 사용시 오른쪽 버튼을 누르면 세이브화면이 뜬다. 이 세가진데...  
69 흥크립트 키입력 질문 [1] A.미스릴 2363   2008-03-15 2008-03-17 04:37
키 클릭을 누르고 누른상태로 있는것도 인식할 수 있나영? 어떻게 하나요 ㅡ,.ㅡ  
68 그림 출력의 순서를 알고 싶습니다. [3] X-tra 2466   2008-03-12 2008-03-17 04:37
제 글이 연구소로 가서 이번에는 신경써서 제목을 적었습니다 ㅡ.ㅡb 채색을 제외하면 전투도 70%정도 완성되었는데 자잘한 버그가 발생해서요. 제 스스로 알아 내려 했지만 확실하게 하기 위해서 물어 봅니다. 정확히 어떤 그림이 앞에 출력이 되고 어떤 그...  
67 '@클릭'이거 어떻게 사용하는거죠? [4] 네모상자 2244   2008-01-26 2008-03-17 04:37
'@영역선택'사용법은 역전심판 뜯어봐서 알 수 있겠는데, 이건 던전 앤 러버를 뜯을 수도 없고 해서 잘 모르겠네요. * 똥똥배님에 의해서 게시물 이동되었습니다 (2008-03-11 17:11)  
66 퀴즈소스입니다 허클베리핀님 라컨 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...