토이 프로젝트/Spring&Java 갖고놀기
자바 리듬 게임 만들기(4)
TerianP
2022. 3. 23. 15:04
728x90
이번 글은 4번째 Note, KeyListener 클래스에 대한 부분입니다.
1. 구현 기능, 목표 및 Main, Dynamic 클래스
2. Game 클래스
3. Track, Music, Beat클래스
4. Note, KeyListener 클래스
1. Note
- note 클래스는 곡의 Note 들에 대한 정보와 노트 판정 정보를 저장하는 클래스로 다른 클래스들과 마찬가지로 Thread 를 상속받는다.
- 생성자 Note 메소드에서는 게임에서의 키보드 위치에따라 note의 x 좌표가 어디에 위치해야하는지를 알려준다.
- screenDraw 는 많이 보아왔던 메소드로 그래픽적 요소를 그려내기 위한 메소드이다. drawImage 메소드를 사용해 어떤 그림을 어떤 위치에 그리고 싶은지 정해준다.
- drop 메소드는 노트를 떨어뜨리는 메소드이다. 노트를 떨어뜨린다 라는 행위의 의미는 노트의 y 좌표가 일정한 양만큼 지속적으로 증가한다 라는 것을 의미한다.
- 이에따라서 drop 메소드가 실행될 때마다 y+=Main.Note_SPEED 를 해준 것이다.
- 만약 y 좌표가 620 을 넘어간다면, 즉 노트 판정 위치를 넘어간다면 Close 를 통해 proceeded = false 로 바뀌고, 해당 노트 스레드를 지워주게 된다.
- run 메소드는 Thread 를 실행시키는 메소드로 drop 메소드를 지속적을 실행시킨다. 이때 proceeded 가 true 인 동안에만 노트를 떨어지고, false 인 경우 스레드가 종료된다.
- 이때 노트가 미친듯이 떨어지지않게 하기 위해서 Thread.sleep 를 통해 노트가 정해진 Main.SLEEP_TIME 만큼 딜레이를 주면서 떨어지게 된다.
- judge 메소드에서는 버튼을 눌렀을 때의 y 좌표에 따라서 해당 판정을 String 타입으로 리턴한다. 이때 판정된 노트의 스레드는 종료된다.
- 여기서 y 좌표를 조절해주면 판정이 좀더 여유로워지거나 빡빡해지거나 하는듯 하다.
package DynamicMusic;
import javax.swing.*;
import java.awt.*;
// 각각의 떨어지는 노트 모두 Thread
/*
1. 노트 판정 과정
노트판정은 Queue 형식으로 구현한다. 즉 first in first out 형식.
이는 당연히도 먼저 출현 - 들어온 - 노트가 가장 먼저 판정이 되어야하기 때문이다.
큐는 특별한 코드로 만들기보다는 ArrayList 로 구현하면 된다.
*/
public class Note extends Thread{
private Image noteBasicImage = new ImageIcon(getClass().getResource("/menu_images/noteBasic_new.jpg")).getImage();
// 현재 노트 위치 확인을 위한 x, y 좌표
// 이 중 y 값의 시작위치를 특정값으로 고정으로 해두어서
// 노트가 생성된 지 1초 뒤 노트 판정위치인 580에 도달 할 수 있도록 함
private int x;
private int y = 580 - 1000 / Main.SLEEP_TIME * Main.NOTE_SPEED;
private String noteType;
// 현재 노트의 진행 여부 -> 즉 노트가 판정이 필요한 범위를 넘어가는지 확인할 수 있또록
private boolean proceeded = true;
// 노트 판정 -> 현재 노트 타입 - 키보드 어떤 버튼인지 - 반환
public String getNoteType(){
return noteType;
}
public boolean isProceeded(){
return proceeded;
}
// 노트를 더이상 사용할 필요가 없다면 false
public void close(){
proceeded = false;
}
public Note(String noteType){
if(noteType.equals("A")){
x = 228;
}else if(noteType.equals("S")){
x = 332;
}else if(noteType.equals("D")){
x = 436;
}else if(noteType.equals("Space")){
x = 540;
}else if(noteType.equals("J")){
x = 744;
}else if(noteType.equals("K")){
x = 848;
}else if(noteType.equals("L")){
x = 952;
}
this.noteType = noteType;
}
// 노트가 내려오는 그래픽을 그리기 위한 screenDraw
public void screenDraw(Graphics2D g){
// 짧은 노트라면 현재 x, y 좌표에 노트를 그려냄
// 노트 타입 - 키 - 에 따라서 다른 그래픽 그림을 그려냄 출력함
if(!noteType.equals("Space")){
g.drawImage(noteBasicImage, x, y, null);
// 긴 노트라면 노트 이미지 2개를 그려냄, 이때 겹치면 안되니까 한쪽은 x+100
}else{
g.drawImage(noteBasicImage, x, y, null);
g.drawImage(noteBasicImage, x+100, y, null);
}
}
// 노트가 떨어지도록 만드는 함수
public void drop(){
// 노트가 떨어진다 -> 노트가 아래로 움직인다 -> y 축으로 일정한 좌표만큼 움직인다
y += Main.NOTE_SPEED;
if(y>620){ // y 좌표가 620 이상이라면 Miss 판정
// System.out.println("Miss");
close();
}
}
// 스레드 실행 함수
@Override
public void run(){
try{
// 노트 떨어지는것은 무한반복
// 1초에 Main.NOTE_SPEED * 100 정도만큼 움직임
while(true) {
drop();
if (proceeded == true) {
// 떨어질때 Main.Sleep_Time 에 설정된 시간만큼 딜레이를 주면서 떨어짐
// 현재 노트가 계속해서 움직이고 있다면 반복적으로 내려옴
// 해당 노트에대한 작업처리가 끝나면 proceeded 가 false 로 변경
Thread.sleep(Main.SLEEP_TIME);
}else{
// 노트에 대한 전반적 작업 - 판정, 입력 등- 끝나서
// 더이상 해당 노트가 필요없어지면 스레드 종료(interrupt)
// proceeded = false
interrupt();
break;
}
}
}catch (Exception e){
System.err.println(e.getMessage());
}
}
public String judge() {
if(y >= 613){
// System.out.println("Late");
close();
return "Late";
}else if(y >= 600){
// System.out.println("Good");
close();
return "Good";
}else if(y>= 587){
// System.out.println("Great");
close();
return "Great";
}else if(y>= 573){
// System.out.println("Perfect");
close();
return "Perfect";
}else if(y>= 565){
// System.out.println("Great");
close();
return "Great";
}else if(y >= 550){
// System.out.println("Good");
close();
return "Good";
}else if(y>= 535){
// System.out.println("Early");
close();
return "Early";
}else{
// 범위 안에서 안누르고 막 눌렀을 경우 miss
return "Miss";
}
}
public int getY(){
return y;
}
}
2. KeyListener
- 지금까지 보았던 모든 클래스와 메소드 중에서 가장 최고로 간단한 클래스로 말 그대로 내가 키보드를 눌렀을 때 실행할 이벤트를 정의하는 클래스이다.
- KeyAdapter 을 상속받았으며, 각각 버튼을 누를때 - keyPressed - 와 버튼을 뗄 때 - key released - 를 정의한다.
- 만약 게임이 실행되고 있지 않은 상태인 game = null 인 경우라면 아래 if ~ else 문을 실행하지 않고 그냥 return 한다.
package DynamicMusic;
import java.awt.event.KeyAdapter;
import java.awt.event.KeyEvent;
// 키보드 이벤트 감지
public class KeyListener extends KeyAdapter {
@Override
public void keyPressed(KeyEvent e) {
// e.getKeyCode 는 사용자가 누르는 키
// KeyEvent.VK_@ 컴퓨터가 갖는 키
// static 으로 설정된 game 이 null 값을 갖는다면 => 어떤 게임도 실행되고 있지 않는다면
// return 해서 아래 키보드 이벤트 작업 실행되지 않도록 함
if(DynamicMusic.game == null){
return;
}
if(e.getKeyCode() == KeyEvent.VK_A){
DynamicMusic.game.pressA();
}
else if(e.getKeyCode() == KeyEvent.VK_S){
DynamicMusic.game.pressS();
}
else if(e.getKeyCode() == KeyEvent.VK_D){
DynamicMusic.game.pressD();
}
else if(e.getKeyCode() == KeyEvent.VK_SPACE){
DynamicMusic.game.pressSpace();
}
else if(e.getKeyCode() == KeyEvent.VK_J){
DynamicMusic.game.pressJ();
}
else if(e.getKeyCode() == KeyEvent.VK_K){
DynamicMusic.game.pressK();
}
else if(e.getKeyCode() == KeyEvent.VK_L){
DynamicMusic.game.pressL();
}
}
@Override
public void keyReleased(KeyEvent e) {
if(DynamicMusic.game == null){
return;
}
// 여기서 사용되는 e.getKeyCode() 는 키보드의 아스키 코드값을 가져옴
// KeyEvent.VK_% 도 출력해보면 해당 글자, 숫자의 아스키 코드값이 출력됨
if(e.getKeyCode() == KeyEvent.VK_A){
DynamicMusic.game.releaseA();
}
else if(e.getKeyCode() == KeyEvent.VK_S){
DynamicMusic.game.releaseS();
}
else if(e.getKeyCode() == KeyEvent.VK_D){
DynamicMusic.game.releaseD();
}
else if(e.getKeyCode() == KeyEvent.VK_SPACE){
DynamicMusic.game.releaseSpace();
}
else if(e.getKeyCode() == KeyEvent.VK_J){
DynamicMusic.game.releaseJ();
}
else if(e.getKeyCode() == KeyEvent.VK_K){
DynamicMusic.game.releaseK();
}
else if(e.getKeyCode() == KeyEvent.VK_L){
DynamicMusic.game.releaseL();
}
}
}