화요일의 Tutorial이 다시 돌아왔습니다. 오늘의 튜토리얼은
Brent
Sorrentino 가 작성한 건데요. Northern Colorado 에서 Corona Ambassador 로 활동하고 있습니다. Brent는 2년여 코로나 커뮤니팅서 활발히 활동하고 있습니다. 그는 프리랜서 여행사진작가이고 코로나 개발자이며 그래픽 디자이너 입니다. 그만의 앱을 개발하기 위해 코로나를 사용하고 있습니다. 또한 정기적으로 포럼에서 여러사람에게 도움을 주고 있고 여러 이슈들을 해결해 주고 있습니다 그의 웹사이트를 보시려면 여기를 클릭하세요.
오늘의 튜토리얼은 어떻게 animated sprite를 implement 할 것인가와 그와 관련된 API 들에 관한 겁니다. 아마 이전 튜토리얼에서 다룬 내용도 있을 겁니다. 올해 초 현재의 스프라이트 시스템이 소개 된 이후 몇가지 주요 개선사항이 있었습니다. 그리고 많은 개발자들이 아직도 코로나에서 어떻게 sprite를 implement 하는지에 대해 혼동을 하고 있습니다. 이 sprite를 사용하는 방법에는 두가지가 있습니다.
- The old (and now depreciated) sprite.* library.
- The current sprite APIs that work cohesively with image sheets and the display.* library.
이렇게 코로나의 스프라이트 라이브러리가 변화를 보였는데요. 그 중에서 현재 버전을 사용할 것을 강력히 추천합니다. 현재 방법을 사용하지 않았거나 코로나에서 스프라이트를 한번도 사용해 보지 않았다면 이 튜토리얼이 많이 도움이 될 겁니다.
코로나의 basic animation에 이미 익숙하신 분이라면 이 튜토리얼이 sprite event listeners에 대한 모든 정보를 제공하고 어떻게 implement 할 것인지를 가이드 해 드릴 겁니다.
Tutorial Contents
- Configuring a Basic Image Sheet or “Sprite Sheet”
- Defining Animation Sequences
- Declaring a Sprite Object
- Sprite Control Methods — managing playback and sequences
- Sprite Properties — accessing and setting various sprite properties
- Sprite Event Listeners — detecting sequence events such as “ended”, “loop”, and “next”
- Frame Trimming Support
Configuring a Basic Image Sheet or “Sprite Sheet”
정확히 image sheet 이 뭘까요? 한번 상상해 보세요. 여러분들의 animated object 들을 위해 각각 의 프레임들을 그린 한장의 종이가 있다고요. 코로나에서 다른 전문적인 용어로는 texture atlas, image map, or sprite sheet 라고 합니다. 이걸 그냥 간단히 image sheet라고 부를께요. 이것은 static 이든 animated object 이든지 상관없이 활용할 수 있습니다.
graphics.newImageSheet() API에 대한 자세한 사용법과 샘플들은 여기에 있습니다. 이 튜토리얼에서는 그중에서 어떻게 animated sprite를 위해 image sheets를 사용할지에 대해 다루겠습니다.
아래 그림은 running cat 을 표현하는 샘플 image sheet 입니다. 이 튜토리얼을 공부하면서 이 이미지를 사용하시려면 여기에 hi-resolution version 이미지가 있습니다. 이 sheet 에는 8개의 프레임이 순서대로 있습니다. 디폴트로 애니메이션은 top-left frame 부터 시작해서 오른쪽으로 진행하죠. 오른쪽 끝까지 가면 그 다음 줄로 갑니다. 그리고 전체 프레임을 다 돌면 중지합니다.
코로나에서 이 image sheet 을 어떻게 다루는지 보겠습니다. 우선 uniform-sized frame 인 basic image sheet 를 위해 indexed table을 setup 합니다.
local sheetData = { width = 512,
--the width of each frame
height = 256,
--the height of each frame
numFrames = 8,
--the total number of frames on the sheet
sheetContentWidth = 1024,
--the total width of the image sheet (see note below)
sheetContentHeight = 1024
--the total height of the image sheet (see note below)
}
IMPORTANT: sheetContentWidth 와 sheetContentHeight는 overall 1:1 dimensions of the image sheet (전체 가로 세로)를 나타냅니다.1:1 content scale 은 여러분이 앱의 config.lua file에 set up 한 내용을 기준으로 합니다. 이렇게 하면 다른 디바이스별로 다른 이미지를 사용할 수 있도록 합니다. 예를 들어 @1 sheet는 original iPad 에 맞는 이미지를 사용하고 high-resolution @2
sheet 는 Retina iPad 에 맞는 이미지를 사용할 수 있습니다. 그러니까 1:1 sheet이 1024×1024 pixels 이라면 2:1 sheet 는 2048×2048 인 이미지가 되는거죠. image sheet setup에는 항상 1:1 dimensions를 사용하세요. config.lua file 세팅이 정확하다면 그 다음부터는 코로나가 알아서 처리할 겁니다.
이제 실제 image sheet 를 정의하세요. 괄호 안에 image file 과 위에 정의한 data table을 넣어주시면 됩니다. 아래처럼 파일이름만 넣으면 그 이미지 파일을 여러분 프로젝트 디렉토리의 root 에서 찾을 겁니다.
local mySheet = graphics.newImageSheet( "runningcat.png", sheetData )
Defining Animation Sequences
여러분은 두가지 방법으로 animated frame 순서를 정하실 수 있습니다.
- consecutively using a starting frame index and frame count
- non-consecutively using a specific order of frames
이 두 가지 방법으로 코로나에서 sprite system 을 아주 flexible 하게 사용하실 수 있습니다. 하나의 image sheet 으로 여러 animation sequences를 사용해서 활용할 수가 있죠. 두 경우 모두 sub-table에서 comma-separated array 로 sequences를 define 하시면 됩니다.
Consecutive frames
local sequenceData = {
{ name = "normalRun", --name of animation sequence
start = 1, --starting frame index
count = 8, --total number of frames to animate consecutively before stopping or looping
time = 800, --optional, in milliseconds; if not supplied, the sprite is frame-based
loopCount = 0, --optional. 0 (default) repeats forever; a positive integer specifies the number of loops
loopDirection = "forward" --optional, either "forward" (default) or "bounce" which will play forward then backwards through the sequence of frames
} --if defining more sequences, place a comma here and proceed to the next sequence sub-table
}
Non-consecutive frames
local sequenceData = {
{ name = "fastRun",
frames = { 1,2,4,5,6,7 }, --specific order of frame indexes from the image sheet
time = 250,
loopCount = 0
} --if defining more sequences, place a comma here and proceed to the next sequence sub-table
}
Mixed sequences (both consecutive and non-consecutive sequences)
local sequenceData = {
{ name="normalRun", start=1, count=8, time=800 },
{ name="fastRun", frames={ 1,2,4,5,6,7 }, time=250, loopCount=0 }
}
Declaring the Sprite Object
현재의 sprite method를 사용하려면 display.newSprite() API 로 sprite를 선언하셔야 합니다. 신택스는 간단합니다.
display.newSprite( [parent,] imageSheet, sequenceData )
- parent = The parent display group in which to insert the sprite (optional).
- imageSheet = The image sheet which the sprite should utilize.
- sequenceData = The array of animation sequences
which you set up. The sprite will default to the first sequence in the
array unless you specify otherwise (see Sprite Control Methods below).
이 튜토리얼에 나온 예제로 한다면 sprite 선언은 아래와 같을 겁니다.
local animation = display.newSprite( mySheet, sequenceData )
이제 이 sprite 는 display object가 됐습니다. 그냥 일반 static image, vector objects 같은 것들과 같게 됐죠. 이제 이 sprite 는 움직일 수도 있고 manipulated 될 수도 있고 physics body 를 입힐 수도 있고... 등등을 할 수 있습니다. 이 sprite object 를 remove 시키려면object:removeSelf() 나 display.remove( object ) 를 사용하시면 됩니다. remove 한 다음에 nil로 세팅하는 것을 잊지 마세요.
Sprite Control Methods
이 스프라이트 시스템은 4가지 주요 control methods를 제공합니다. 여러분들은 이것으로 스프라이트의 playback과 sequence 를 control 하실 수 있습니다.
- animation:play()
Start the animation playing. Animations do not begin playing when you create them — you must start each animation manually using this command. - animation:pause()
Pauses the animation. There is no “stop” control method; instead, pause the animation using this method. - animation:setFrame( frame )
Immediately set or skip to the indicated frame index. If you want to
“stop and reset” an animation sometime after you have started playing
it, use the :pause() and :setFrame( frame ) commands consecutively, setting the frame back to the beginning of the sequence. - animation:setSequence( sequence )
Set the sprite to a specific sequence that you declared in your sequence
array. For example, if you want to change your cat animation from
“normalRun” to “fastRun”, you would use animation:setSequence( “fastRun” ) and use animation:play() to begin playing it, since the animation will not play automatically after you change the sequence.
Putting It Together
전체 애니메이션을 함께 넣어보죠. 달리는 고양이를 normalRun 과 fastRun 두가지로 셋업할 겁니다. 아래 예제가 있습니다.
local sheetData = { width=512, height=256, numFrames=8,
sheetContentWidth=1024, sheetContentHeight=1024 }
local mySheet = graphics.newImageSheet( "runningcat.png", sheetData )
local sequenceData = {
{ name = "normalRun", start=1, count=8, time=800 },
{ name = "fastRun", frames={ 1,2,4,5,6,7 }, time=250 }
}
local animation = display.newSprite( mySheet, sequenceData )
animation.x = display.contentWidth/2 --center the sprite horizontally
animation.y = display.contentHeight/2 --center the sprite vertically
animation:play()
코로나 시뮬레이터로 테스트 해 보세요. 저 고양이가 화면 중앙에 나올겁니다. 그리고 normalRun sequence로 animating 되겠죠. (sequence를 따로 선언하지 않으면 위에 얘기했던 디폴트 순수대로 진행합니다.)
테스트를 위해 animation:play()앞에 animation:setSequence( “fastRun” )를 넣어 보세요.
Sprite Properties
Corona provides several properties which can yield information about existing sprites. You can even modify the timeScale (relative speed) of a particular sprite. These properties are as follows:
코로나에는 현재의 스프라이트에 적용할 수 있는 몇가지 프로퍼티들을 제공합니다. 특정 스프라이트의 timeScale (relative speed)를 modify 할 수도 있습니다. 아래 프로퍼티들을
- object.frame
A read-only property that represents the currently shown frame index of the loaded sequence. This does not set the frame — use the :setFrame() command to explicitly set an animation frame. - object.isPlaying
Returns true if the animation is currently playing; false if it is not. - object.numFrames
A read-only property that represents the number of frames in currently loaded sequence. - object.sequence
A read-only property that returns the name of the currently playing sequence. - object.timeScale
Gets or sets the scale to be applied
to the animation time. This is used to control a sprite’s animation
speed dynamically. For example, a time scale of 1.0 (default) runs the
animation at normal speed. A time scale of 2.0 runs the animation twice
as fast. A time scale of 0.5 runs the animation at half speed. The
maximum allowed value is 20.0 and the minimum allowed value is 0.05. The
value supports up to 2 decimal places.
Sprite Event Listeners
이제 basic sprite 선언과 두개의 sequences (“normalRun” , “fastRun”) 가 생겼습니다.
이제 sprite event listener를 살펴보죠. 그리고 그것을 어떻게 implement 하는지에 대해 알아보겠습니다. sprite event listener 의 정의는 'sprite의 activity를 "listens" 하고 그 정보를 listener function에 전달하는 것' 입니다.
예를 들어 여러분의 달리는 고양이를 "normalRun" sequence로 4번(4 cycles)를 돌게 한 다음에 "fastRun" sequence로 바꿀 수 있습니다. 이것은 standard timer 로 표현하기에는 불가능한 효과죠. 그래서 그 대신에 sprite event listener를 implement 하는 겁니다.
예제를 보기 전에 sprite에서 사용 가능한 5가지 를 event phases보겠습니다.
이 phases 들은 최근의 코로나 Public Release 인 (2012.894) 버전 이후부터 사용하실 수 있습니다.
- began = The sprite has started playing.
- ended = The sprite has finished playing.
- bounce = The sprite has bounced from forward to backward while playing.
- loop = The sprite has looped to the beginning of the sequence.
- next = The sprite has played a subsequent frame that’s not one of the above phases.
이 phases 를 어떻게 listen 할지에 대해 다루겠습니다. 근데 우선 달리는 고양이에 loopCount = 4를 추가해서 normalRun 의 sequence를 바꾸고 시작하도록 하죠. 이렇게 하면 4번의 loop가 끝나면 ended phase를 받을 수 있도록 해 줍니다.
local sequenceData = {
{ name = "normalRun", start=1, count=8, time=800, loopCount=4 }, --add loopCount=4
{ name = "fastRun", frames={ 1,2,4,5,6,7 }, time=250 }
}
Now, let’s write the listener function and add the actual event
listener to the running cat. You can place this at the end of your
sample code, after the sequences are declared and the sprite object
placed on the screen.
이제 listener function을 만들어 봅시다. 그리고 달리는 고양이에 실제 event listener를 달아보죠. 아래 내용을 위 샘플 코드의 마지막 부분에 sequence 가 선언된 다음에 추가하시면 됩니다.
local function mySpriteListener( event )
if ( event.phase == "ended" ) then
local thisSprite = event.target --"event.target" references the sprite
thisSprite:setSequence( "fastRun" ) --switch to "fastRun" sequence
thisSprite:play() --play the new sequence; it won't play automatically!
end
end
animation:addEventListener( "sprite", mySpriteListener ) --add a sprite listener to your sprite
sprite listener 가 이제 모든 phases를 listener 함수에 전달하게 됩니다. if-then 구문을 사용해서 이 phases를 사용하는 것은 개발자가 코딩해야 할 부분입니다. 특정 시점에 우리는
ended phase를 listen 하게 됩니다. sequence의
loopCount parameter에 의해 4번의 루프가 끝나면 이
ended phase가 발생하도록 했죠. 이
ended phase 를 받으면 이 cat animation을 fastRun sequence로 바꾸고 play 하게 됩니다.
하나의 sprite listener를 모든 sprite 에 대해 사용하실 수 있습니다. listener 함수에서 event.target을 사용해서 원하는 스프라이트를 catch 해서 사용하실 수 있는겁니다.
Frame Trimming Supported
최근의 Public Build (2012.894) 에서는 frame trimming 도 지원합니다. 3rd-party sprite utilities 인 SpriteLoq 와 TexturePacker
같은 곳에서 이 기능을 제공하고 있습니다. 그리고 코로나에서도 이 어플리케이션과 호환성 있게 이 기능을 사용할 수 있도록 하고 있습니다.
frame trimming 예제는 current sprite methods 를 사용하고 있는데 이것은 여러분 시스템의 코로나 어플리케이션에 있는 “HorseAnimation” sample
project에서 확인 하실 수 있습니다.
CoronaSDK → SampleCode → Sprites → HorseAnimation
이와 관련해서는 다음 튜토리얼에서 다뤄지게 될 것 같습니다. 그동안 여러분은 샘플 프로젝트를 보셔도 되고 여기에서
imageSheet documentation를 보시면 많은 도움이 되실 겁니다.
In Summary
이 튜토리얼은 current sprite methods 의 대부분을 다뤘습니다. 여기에 basic image sheets, animation sequences 정의하기, sprite playback 다루기, 다양한 sprite property들 다루기 그리고 sprite event listener 사용하기 등을 추가로 다루고 있습니다. 모쪼록 이 튜토리얼이 개발자 여러분에게 스프라이트를 이용해서 앱을 만드는데 도움을 드릴 수 있기를 바라겠습니다. 특히 이전 버전의 sprite library 를 사용하시던 분들에게 새로 바뀐 sprite library 를 사용하시는데 도움이 되기를 바라구요.