[pygame] Cannon 게임 프로젝트 4 - 스프라이트 기본

 제작로그 4 - 스프라이트 기본 

스프라이트는 화면에 움직이는 오브젝트들(캐릭터들?) 이라고 생각하면 됩니다. 제가 만드는 게임이라면 총알이라던가 적 캐릭터라던가 하는것들입니다. 

pygame 에는 스프라이트 클래스가 준비되어 있고 중요한 기능중 하나는 캐릭터 간의 충돌체크 입니다. 

충돌체크는 다음에 다루기로 하고 여기서는 간단히 스프라이트 이용의 기본 내용만 보겠습니다. 

import pygame
from robot import *
 
SCREEN_X = 800    #게임창 x 크기
SCREEN_Y = 600    #게임창 y 크기
 
screen = pygame.display.set_mode((SCREEN_X,SCREEN_Y)) #게임 윈도우 생성
background = pygame.Surface(screen.get_size())        #게임 화면을 그릴 surface

pygame.init() #pygame 초기화
robot_image = pygame.image.load('robot.png').convert_alpha() #이미지를 로드하고 변환
robot_image = pygame.transform.scale(robot_image,(40,40)) #이미지의 크기 변환
robot = Robot(robot_image,300,300)    #Robot 스프라이트 객체 생성

done=True
 
while done:        #이 루프를 벗어나면 게임이 종료됨
    ev = pygame.event.get()
    for event in ev:
        if event.type ==pygame.QUIT:     #창의 종료버튼이 눌렸을때 게임종료
            done = False     
        if event.type == pygame.MOUSEMOTION:    #마우스의 위치 얻음
            robot.rect.center = pygame.mouse.get_pos() #마우스 위치를 center 에 넣음
        break
 
    #적과 아군 캐릭터 표현. 점수 표현은 이곳에서
    background.fill((0,0,0))    #화면 지움
    background.blit(robot.image , robot.rect) #로봇캐릭터 표시
 
    screen.blit(background,(0,0)) #background 의 화면을 스크린에 복사
    pygame.display.flip()         #화면에 표시함

화면에 Robot 을 표시하고 마우스를 움직이면 마우스 포인터를 따라 Robot 이 그려지는 예제 입니다.

>

Robot 클래스는 다음과 같습니다. 

import pygame

class Robot(pygame.sprite.Sprite):

    def __init__(self,img,x,y):
        super().__init__()
        self.image = img    #스프라이트 이미지 surface
        self.rect = img.get_rect()  #이미지의 rect 구해서 저장
        self.rect.center = (x,y)    #캐릭터 위치 좌표
        self.mask = pygame.mask.from_surface(img) #충돌체크용 마스크 생성
        self.vecpos = pygame.math.Vector2(x,y)  #벡터 값 저장
        self.x = x
        self.y = y
        self.alive = 1  #로봇의 상태

Robot 클래스는 pygame 의 스프라이트 클래스를 상속받아 만듭니다. Vector2 클래스는 실제 이 게임의 robot 의 움직임에 쓰이지 않기는 하는데 만약에 쓰일 일이 있을까 싶어서 넣어둔 것입니다. 

다른건 보시면 대충 아실거고 중요한건 rect.center 와 pygame.mask.from_surface(img) 부분입니다. 

mask 생성은 스프라이트의 충돌체크에 쓰이기 때문에 꼭 넣어주어야 합니다. 이는 다음의 충돌체크부분에서 이야기 하도록 하겠습니다.

>

background.blit 함수 부분을 보면 robot 클래스의 image 와 rect 를 인자로 넣었는데 sprite 클래스의 rect 에는 center 라는 변수가 존재합니다. 만일 center 변수에 값이 없으면 rect의 x,y 값을 좌표로 사용합니다. 

일반적으론 게임에서는 center 변수를 이용하는게 좋은데 왜냐하면 center 의 좌표가 캐릭터의 중심이 되기 때문입니다. 


위는 center 변수를 사용하지 않았을때인데 캐릭터가 상단 왼쪽위치를 중심으로 그려지는걸 알 수 있습니다. 

 

center 변수에 값을 넣어주면 위와 같이 그려집니다. 마우스 포인터를 중심으로 하여 캐릭터를 그려주는 걸 확인 할 수 있습니다. 

소스에서는 "robot.rect.center = pygame.mouse.get_pos()" 의 방법으로 마우스를 움직였을때 마우스 포인터의 위치 값을 Robot 클래스의 rect.center 에 넣어줌으로서 마우스 포인터의 위치를 중심으로 캐릭터를 그리도록 하고 있습니다.  

>

"pygame.image.load('robot.png').convert_alpha()" 부분은 캐릭터인 Robot 의 이미지를 로드하는 부분인데요. convert_alpha() 로 변환을 해주는게 중요합니다. 이미지의 투명부분인 알파채널 때문일걸로 생각됩니다. 

이 부분도 충돌체크에서 중요한 부분입니다. 저 투명부분은 충돌로 판정되지 않고 캐릭터 부분만 충돌로 판정되어야 하기 때문입니다.

 

위의 이미지는 gimp 에서 캐릭터 이미지를 로드 한것 입니다. 캐릭터인 초록색 부분을 제외한 체크 격자 부분이 바로 이미지의 투명한 부분 입니다. 충돌 체크시 저 부분은 충돌 체크에서 제외됩니다. 따라서 정확한 충돌체크를 할 수가 있습니다.

>

다음에 간단히 충돌체크를 적어 보겠습니다. 충돌체크 방법은 사각형, 원 등의 방법이 있지만 알파채널을 이용한 mask 방법만 적어 볼 생각입니다. 

 

댓글