메뉴 건너뛰기

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

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
165 Windows To Go와 R-Studio를 이용한 손실된 데이터 복구하기 노루발 89   2020-01-30 2020-01-30 19:52
하기의 복구 방법으로 모든 자료를 100% 복구할 수는 없으므로 자료의 손실 이전에 신뢰성 있고 주기적인 백업이 선행되어야 한다. 준비물: - 부팅 USB 혹은 외장 HDD/SSD - 복구한 자료를 저장할 외부 저장장치 (옵션) - R-Studio https://www.r-studio.com/ ...  
164 Love2d DPI 이슈 해결 [3] 노루발 97   2019-06-29 2019-07-01 06:34
이런 love 프로젝트가 있다고 하자. (conf.lua) function love.conf(t) t.window.width = 640 t.window.height = 360 end 창 크기를 640*480으로 설정한 뒤 실행하면 어떻게 보일까? 당연히 창 크기가 640*480 크기로 보여야겠지만 내 컴퓨터에서는 이렇게 보...  
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 RPG Maker MV 로컬라이징 방법 file 똥똥배 918   2015-10-27 2015-10-27 05:51
 
160 [번역] gamedev레딧의 Getting Started 문서 번역 [5] priling 1892   2014-12-26 2018-07-24 10:33
처음인 분들을 위한 '게임만들기' 가이드이 글은 [레딧 게임개발 커뮤니티의 /u/LordNed님의 포스팅]을 베이스로 작성한 것입니다. 이 글의 목적은 게임을 만들고 싶어하는 분들이 어떻게 시작할 수 있을지에 대해 명확한 가이드라인을 보여드리는 것입니다.  ...  
159 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 ...  
158 Love2d 안드로이드 빌드하기 노루발 572   2014-12-15 2014-12-15 00:59
Love2d의 안드로이드 포트는 알파 단계입니다. 차차 개선되어 나가긴 하겠지만 아직 불안정한 부분이 많으며, 일어날 수 있는 오작동과 그로 인한 결과는 일절 책임지지 않습니다.   이 문서는 Windows 사용자 기준입니다. Linux나 Mac을 사용할 정도의 내공...  
157 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/  
156 구글 인앱 구매 Soomla로 구현해본 후 팁 똥똥배 577   2014-09-20 2014-09-20 18:45
1. 일단 알파 테스트라도 앱을 출시시켜야 인앱이 작동한다. 그리고 APK 업로드 후에 바로 인앱이 적용되지 않는다. 구글에 게시되는 데, 시간이 걸리므로 인앱 등록 - APK 등록을 마친 후 다음날부터 구현하는 게 깔끔하다. 이걸로 모르고 하루종일 왜 안되나...  
155 cocos2d-x 2.2.2 문자열 출력 버그 [2] 똥똥배 601   2014-06-10 2014-08-23 22:35
cocos2d-x 2.2.2, 맥에서만 나타는 문제다. 윈도우, 안드로이드에서는 아무 이상 없다. 줄 수가 꽤 긴 문자열을 출력할 경우, 문자가 전부 다 나오지 않고 잘린다. 잘리는 기준은 줄이다. 문자 수가 어떻든 간에 줄 수에 따라서 잘려버린다. 아마 문자열의 줄 ...  
154 cocos2d-x 2.2.2 윈도우 환경 기본 메모리 누수 똥똥배 640   2014-03-10 2014-03-11 03:51
cocos2d-x 2.2.2를 윈도우에서 실행시키고 나서 종료하면 기본으로 4바이트의 메모리 누수가 발생한다. 이건 CCScriptEngineManager가 원인인데 CCObject들은 delete를 실행할 때마다 CCScriptEngineManager의 sharedManager를 호출하기 때문이다. 결국 모든 C...  
153 cocos2d-x 2.2.2 UILabelBMFont 메모리 누수 해결법 [2] 똥똥배 783   2014-03-10 2014-05-31 19:30
만약 CocoStudio로 UI를 만든 후 GUIReader를 써서 지정 폰트를 불러오면 엔진 버그로 인해서 메모리 누수가 생긴다. 이걸 고치려면 LabelBMFont.cpp를 아래와 같이 수정해야 한다. void LabelBMFont::setFntFile(const char *fileName) { if (!fileName || st...  
152 Cocostudio의 ActionNode 메모리 누수 해결법(cocos2d-x 2.2.2) [2] 똥똥배 776   2014-03-09 2014-05-31 19:26
Cocostudio로 액션을 만들고 액션매니저를 이용해서 ActionManager::shareManager()->playActionByName 아래와 같이 실행을 해주면 액션이 실행되는 경우 메모리 누수가 발생한다. (액션을 실행하지 않았을 때는 메모리 누수 없음) 이 메모리 누수를 없애려면 ...  
151 CCTextFieldTTF 0바이트 메모리 누수 버그 해결법 똥똥배 692   2014-01-13 2014-01-13 10:38
CCTextFieldTTF::textFieldWithPlaceHolder("", FONT_NAME, FONT_SIZE); 이런 식으로 텍스트필드를 생성하면 ""가 0바이트라서 해체를 못하고 0바이트 누수가 발생한다. 그냥 스페이스(" ")라도 넣어주면 누수는 발생하지 않는다. 이 버그가 발생하는 버전은 2...  
150 Lua 소코반 EX: 포팅: 3 (머나먼 여행길 안녕 친구여) 노루발 913   2013-09-18 2013-09-18 08:05
안녕하세요, 노루발입니다. 오늘로 그래픽 소코반을 끝내기로 했습니다. 아쉽지만 끝내야죠. 일단 걸음수와 이것저것을 뿌려주기로 합니다. 그런데 이미 화면이 꽉 차버려서 뿌려줄 공간이 없네요... 없으면 만들면 되지. 창을 늘립시다. conf.lua를 수정. fun...  
149 Lua 소코반 EX: 포팅: 2 (키가 눌리면 이동하게 해 보자) 노루발 832   2013-09-18 2013-09-18 08:04
-- 플레이어 위치를 저장하는 변수 -- 초기 시작지점은 5, 5로 설정. playerx = 5 playery = 5 -- 박스의 위치를 저장하는 배열. box = {} -- 박스 목표 지점의 위치를 저장하는 배열. goal = {} -- 움직인 횟수 저장 moves = 0 -- 맵 데이터를 저장하는 8*8의...  
148 Lua 소코반 EX: 포팅: 1 노루발 856   2013-09-18 2013-09-18 08:03
-- 플레이어 위치를 저장하는 변수 -- 초기 시작지점은 5, 5로 설정. playerx = 5 playery = 5 -- 박스의 위치를 저장하는 배열. box = {} -- 박스 목표 지점의 위치를 저장하는 배열. goal = {} -- 움직인 횟수 저장 moves = 0 -- 맵 데이터를 저장하는 8*8의...  
147 Lua 소코반 EX: 그래픽 준비 노루발 1084   2013-09-18 2013-09-18 08:00
안녕하세요, 노루발입니다. 얼마 전 만든 Lua 소코반을 게임다운 게임을 만들기 위해 그래픽을 씌워보려고 합니다. 저번에 한 Love2D로 살펴본 뭐가 뭔지 모르는 복잡한 그리기 함수들도 사실은 모두가 소코반을 그래픽으로 만들기 위한 훼이크였습니다. 후후....  
146 게임의 기본 설정을 담당하는 love.conf 노루발 763   2013-09-17 2013-09-17 08:39
안녕하세요, 노루발입니다. 여태까지 이것저것 많이 Love2D를 다뤄와서 제가 만든 소코반에 그래픽을 씌울 수준까지 되었지만 (물론 키 입력은 논외로 치죠, 나중에 다룰게요.) 아직 이걸로 정식 게임을 만든다면 모자라도 한참 모자랍니다. 창의 이름도 실행...