Posts pythongame
Post
Cancel

pythongame


파이썬을 이용해서 간단하고 쉬운 게임 만들어보기

작품 소개

: 1학년 2학기, 공학도를 위한 창의적 컴퓨팅이라는 강의를 들으면서 파이썬이라는 컴퓨팅 언어를 배웠는데, 이를 이용해서 각종 복잡한 활동들을 할 수 있으며 게임도 만들어낼 수 있다는 교수님의 말씀을 들었다. 예전부터 게임에 관심도 많고 좋아하기도 했으며, 실제 파이썬을 통해 게임을 만들었을 때, 내가 배운 내용이 실제로 사용되는지 궁금해서 한 번 파이썬을 이용해서 게임을 만들어보았다. 이 코드는 유튜브 “초보 코딩”이라는 채널, 이수안 컴퓨터 연구소에서 배포한 pyshooting이라는 pdf 파일, 저서 “혼자 공부하는 파이썬”, 강의 시간에 배운 내용으로 Anaconda3의 jupiter notbook을 통해 제작했다. 그리고 과거에 내가 했던 드래곤 플라이트라는 게임을 모티브로 삼아서 이를 비슷하게 만들어보고 싶었다.

기존 목표

: 플레이어가 조종하는 드래곤 형태의 캐릭터를 화살표 상, 하, 좌, 우를 이용해서 움직이고 스페이스 바를 눌러서 불을 뿜게 마시며 위에서부터 아래로 적들이 출몰하고 적과 충돌하면 게임 오버되게 만드는 것이 기본적인 게임의 구조이다. 거기다가 시간, 점수, 놓친 적의 숫자, 좌우에서부터 발사되는 장애물, 획득 시 몇 가지 좋은 효과를 내는 소모성 효과를 지닌 것과 나쁜 효과를 내는 소모성 효과를 지닌 것이 드문 확률로 맵 어딘가에 등장하게 만들어보는 것

결과

: 코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
import pygame
import random
import time
from datetime import datetime

# 1. 게임 초기화
pygame.init()

# 2. 게임창 옵션 설정
size = [400, 800]
screen = pygame.display.set_mode(size)

title = "Dragon Flight"
pygame.display.set_caption(title)

# 3. 게임 내 필요한 설정
clock = pygame.time.Clock()

class obj:
    def __init__(self):
        self.x = 0
        self.y = 0
        self.mobe = 0
    def put_img(self, address):
        if address[-3:] == "png":
            self.img = pygame.image.load(address).convert_alpha()
        else :
            self.img = pygame.image.load(address)
        self.sx, self.sy = self.img.get_size()
    def change_size(self, sx, sy):
        self.img = pygame.transform.scale(self.img, (sx, sy))
        self.sx, self.sy = self.img.get_size()
    def show(self):
        screen.blit(self.img, (self.x,self.y))

# a.x-b.sx <= b.x <= a.x+a.sx
# a.y-b.sy <= b.y <= a.y+a.sy

def crash(a, b):
    if (a.x-b.sx <= b.x) and (b.x <= a.x+a.sx):
        if (a.y-b.sy <= b.y) and (b.y <= a.y+a.sy):
            return True
        else:
            return False
    else:
        return False
        
ss = obj()
ss.put_img("C:/Python39/images/ss.png")
ss.change_size(60,80)
ss.x = round(size[0]/2- ss.sx/2)
ss.y = size[1] -ss.sy -100
ss.move = 6     

left_go = False
right_go = False
up_go = False
down_go = False
space_go = False

m_list = []
a_list = []

black = (0,0,0)
white = (255,255,255)
k = 0

kill = 0
loss = 0

#4-0. 게임 시작 대기 화면
SB = 0
while SB == 0:
    clock.tick(60)
    for event in pygame.event.get():
        if event.type == pygame.KEYDOWN:
            if event.key == pygame.K_SPACE:
                SB = 1
    screen.fill(black)
    font = pygame.font.Font("C:/Python39/Fonts/나눔 글꼴/나눔고딕/NanumGothic.otf", 15)
    text = font.render("PRESS SPACE KEY TO START", True, (255,255,255))
    screen.blit(text, (40, size[1]/2-50))
    pygame.display.flip()
        
    

# 4. 메인 이벤트
start_time = datetime.now()
SB = 0
while SB == 0:

    # 4-1. FPS 설정
    clock.tick(60)

    # 4-2. 각종 입력 감지
    # ss: 인물 키보드 입력 상하좌우 이동
    # ss1: 던져지는 교재 마우스 클릭, 마우스 위치
    # ss2: 과제 빌런 랜덤 발생 랜덤 일자 이동
    for event in pygame.event.get():
        if event.type == pygame.QUIT:
            SB = 1
        if event.type == pygame.KEYDOWN:
            if event.key == pygame.K_LEFT:
                left_go = True
            elif event.key == pygame.K_RIGHT:
                right_go = True
            elif event.key == pygame.K_UP:
                up_go = True
            elif event.key == pygame.K_DOWN:
                down_go = True
            elif event.key == pygame.K_SPACE:
                space_go = True   
                k = 0
        elif event.type == pygame.KEYUP:
            if event.key == pygame.K_LEFT:
                left_go = False
            elif event.key == pygame.K_RIGHT:
                right_go = False
            elif event.key == pygame.K_UP:
                up_go = False
            elif event.key == pygame.K_DOWN:
                down_go = False
            elif event.key == pygame.K_SPACE:
                space_go = False    
                

    # 4-3. 입력, 시간에 따른 변화
    now_time = datetime.now()
    delta_time = (now_time - start_time).total_seconds()
    
    if left_go == True:
        ss.x -= ss.move
        if ss.x <= 0:
            ss.x = 0
    elif right_go == True:
        ss.x += ss.move
        if ss.x >= size[0] - ss.sx:
            ss.x = size[0] - ss.sx
    elif up_go == True:
        ss.y -= ss.move
        if ss.y <= 0:
            ss.y = 0
    elif down_go == True:
        ss.y += ss.move
        if ss.y >= size[0] + 3*ss.sy -10:
            ss.y = size[0] + 3*ss.sy -10
               
    if space_go == True and k % 6 == 0:
        mm = obj()
        mm.put_img("C:/Python39/images/mm.png")
        mm.change_size(30,30)
        mm.x = round(ss.x + ss.sx/2 - mm.sx/2)
        mm.y = ss.y - mm.sy
        mm.move = 15
        m_list.append(mm)
    k += 1    
    d_list = []
    for i in range(len(m_list)):
        m = m_list[i]
        m.y-= m.move
        if m.y <= -m.sy:
            d_list.append(i)
    for d in d_list:
        del m_list[d]

    if random.random() > 0.98 : 
        aa = obj()
        aa.put_img("C:/Python39/images/aa.png")
        aa.change_size(60,80)
        aa.x = random.randrange(0, size[0]-aa.sx-round(ss.sx/2))
        aa.y = 10
        aa.move = 3
        a_list.append(aa)
    d_list = []
    for i in range(len(a_list)):
        a = a_list[i]
        a.y += a.move
        if a.y >= size[1]:
            d_list.append(i)
    for d in d_list:
        del a_list[d]
        loss += 1
            
    dm_list = []
    da_list = []
    for i in range(len(m_list)):
        for j in range(len(a_list)):
            m = m_list[i]
            a = a_list[j]
            if crash(m,a) == True:
                dm_list.append(i)
                da_list.append(j)
    dm_list = list(set(dm_list))
    da_list = list(set(da_list))

    for dm in dm_list:
        del m_list[dm]
    for da in da_list:
        del a_list[da]
        kill += 1
        
    for i in range(len(a_list)):
        a = a_list[i]
        if crash(a, ss) == True:
            SB = 1
            GO = 1
    
    # 4-4. 그리기
    screen.fill(black)
    ss.show()
    for m in m_list:
        m.show()
    for a in a_list:
        a.show()
        
    font = pygame.font.Font("C:/Python39/Fonts/나눔 글꼴/나눔고딕/NanumGothic.otf", 20)
    text_kill = font.render("killed: {}, loss: {}".format(kill, loss), True, (0,255,0))
    screen.blit(text_kill, (10, 5))
    
    text_time = font.render("time : {}".format(delta_time), True, (255,255,255))
    screen.blit(text_time, (size[0]-100, 5))
    
    # 4-5. 업데이트
    pygame.display.flip()

# 5. 게임 종료
while GO == 1:
    clock.tick(60)
    for event in pygame.event.get():
        if event.type == pygame.QUIT:
            GO = 0
    font = pygame.font.Font("C:/Python39/Fonts/나눔 글꼴/나눔고딕/NanumGothic.otf", 15)
    text = font.render("GAME OVER", True, (255,0,0))
    screen.blit(text, (80, size[1]/2-50))
    pygame.display.flip()
pygame.quit() 

이미지:

개선해야 할 점

: 기본적인 구조는 내가 처음에 구상한 것처럼 만들 수 있었고, 시간, 점수, 놓친 횟수 같은 것은 구현했지만 내가 창의적으로 만들어냈다기 보다는 유튜브, 책 등의 도움을 많이 받으며 이를 거의 따라 만든 것과 다를 바가 없다. 캐릭터를 움직이게 하거나 게임을 시작하고 종료할 때 등장하는 텍스트 등, 그리고 일부 내용들은 내가 배운 내용을 응용해서 제작해보았다. 또한 내가 처음에 추가하고자 했던 랜덤성 아이템의 등장과 추가적인 장애물의 기능은 추가하지 못했으며 플레이어 캐릭터가 발사하는 불덩이 파일에서 하얀 부분을 없애는 방법을 적용하지 못해서 발사체가 조금 이상하다. 그리고 텍스트가 화면의 중간보다 살짝 치우쳐진 곳에 출력되게 되었고, 적들에게 플레이어가 충돌하는 범위가 캐릭터의 크기보다 살짝 더 커서 충돌 판정이 굉장히 애매하고 확실하지 않았다.

This post is licensed under CC BY 4.0 by the author.

카메라 캘리브레이션

탈부착형 분실물 찾기 시스템