[chap 9] 자바 GUI 기초, AWT와 Swing
🔮 자바 GUI 패키지
📍 GUI 패키지 계층 구조
모든 GUI(AWT, Swing)은 Componet 클래스를 반드시 상속받으며, 스윙 컴포넌트의 클래스 명은 모두 J로 시작한다. AWT 컴포넌트는 Button, Label 등과 같이 Component를 직접 상속받는 것들과 Panel, Frame 등과 같이 Container를 상속받는 것이 있다. JApplet, JFrame, JDialog를 제외한 모든 스윙 컴포넌트들은 JComponent를 상속받는다.
📍 컨테이너와 컴포넌트
컨테이너 : 다른 컴포넌트를 포함할 수 있는 GUI 컴포넌트
- java.awt.Container를 상속받음.
다른 컨테이너에 포함될 수 있다.
- AWT 컨테이너 : Panel, Frame, Applet, Dialog, Window
- Swing 컨테이너 : JPanel, JFrame, JApplet, JDialog, JWindow
컴포넌트 : 컨테이너에 포함되어야 화면에 출력될 수 있는 GUI 객체. 컨테이너에 컴포넌트를 add 순서대로 그린다라고 생각하면 쉽다
다른 컴포넌트를 포함할 수 없는 순수 컴포넌트
모든 GUI 컴포넌트가 상속받는 클래스 : java.awt.Component
스윙 컴포넌트가 상속받는 클래스 : java.swing.JComponent
최상위 컨테이너 : 다른 컨테이너에 포함되지 않고도 화면에 출력되며 독립적으로 존재 가능한 컨테이너
- 스스로 화면에 자신을 출력하는 컨테이너 => JFrame, JDialog, JApplet
예제에서 Button, Checkbox를 그리기 위해서 무조건 f.setLayout(new FlowLayout());
을 써주었다. 이걸 보면 Frame이 컨테이너, 그 안에 컴포넌트들이 있는 것을 알 수 있다.
🔮 컨테이너와 배치
컨테이너에 부착되는 컴포넌트들의 위치와 크기는 컨테이너 내부에 있는 배치관리자(Layout Manager)에 의해 결정된다.
관계 1. 컨테이너마다 배치관리자는 하나씩 있다.
관계 2. 배치관리자는 컨테이너에 컴포넌트가 부착되는 시점에 컴포넌트의 위치와 크기를 결정한다. 컨테이너의 크기가 변하면 내부 컴포넌트들의 위치와 크기를 모두 재조절하여 재배치한다.
📍 배치관리자의 종류
-
FlowLayout : 컨테이너에 부착되는 순서를 왼쪽에서 오른쪽으로 컴포넌트를 배치. 공간없으면 아래로 내려와서 다시 왼쪽에서 오른쪽으로 배치
-
BorderLayout : 컨테이너의 공간을 동, 서, 남, 북, 중앙 5개 영역 으로 나누고 응용프로그램에서 지정한 영역에 컴포넌트를 배치 헐,,,,,,,,,,,,,HW 4이게 정답 아님,,,,,,?
-
GridLayout : 컨테이너의 공간을 응용프로그램에서 설정한 동일한 크기의 2차원 격자 로 나누고, 컴포넌트가 삽입되는 순서대로 좌에서 우 로 다시 위에서 아래 로 배치한다.
-
CardLayout : 카드 를 쌓아 놓은 듯이 컴포넌트를 포개어 배치. 컴포넌트의 크기는 컨테이너의 크기와 동일
AWT와 스윙 컨테이너 | 디폴트 배치관리자 |
---|---|
Window, JWindow | BorderLayout |
Frame, JFrame | BorderLayout |
Dialog, JDialog | BorderLayout |
Panel, JPanel | FlowLaout |
Applet, JApplet | FlowLayout |
❓ 배치관리자 맘대로 설정하려면? -> setLayout() 메소드 호출,,,
Container.setLayout(new BorderLayout);
FlowLayout 배치관리자
📐 부착 방법 : add(Component comp)
컴포넌트를 왼 -> 오, 자리 없으면 밑으로 내려와서 다시 왼 -> 오. 근데 컨테이너의 크기가 변하면, FlowLayout 배치관리자에 의해 컨테이너의 크기에 맞도록 컴포넌트가 재배치된다.
-
생성자
- FlowLayout()
- FlowLayout(int align)
- FlowLayout(int align, int hGap, int vGap)
- align: 컴포넌트의 정렬 방법. FlowLayout.LEFT, FlowLayout.RIGHT, FlowLayout.CENTER(디폴트)
- hGap : 좌우 컴포넌트 사이의 수평 간격, 픽셀 단위, 디폴트 5
- vGap : 상하 컴포넌트 사이의 수직 간격, 픽셀 단위, 디폴트 5
//ex new FlowLayout(); new FlowLayout(FlowLayout.LEFT); new FlowLayout(FlowLayout.LEFT, 10, 20);
BorderLayout 배치관리자
📐 부착 방법 : add(Component comp, int index)
- comp : 컨테이너에 부착되는 컴포넌트
- index : BorderLayout.EAST, BorderLayout.WEST, BorderLayout.SOUTH, BorderLayout.NORTH, BorderLayout.CENTER
컨테이너의 크기가 변하면 BorderLayout 역시 새로운 크기에 맞도록 컴포넌트의 크기를 재조종한다. BorderLayout 배치관리자를 사용하는 컨테이너에는 일차적으로 5개의 컴포넌트밖에 붙힐 수 없다. 5개 이상을 붙히고자한다면, 한 영역에 JPanel 등 다른 컨테이너를 부착하고 이곳에 컴포넌트를 부착하면 된다.
-
생성자
- BorderLayout()
- BorderLayout(int hGap, int vGap)
- hGap : 좌우 컴포넌트 사이의 수평 간격, 픽셀 단위, 디폴트 0
- vGap : 상하 컴포넌트 사이의 수직 간격, 픽셀 단위, 디폴트 0
생성자에서 이미 위치를 지정해 놓았으므로 align은 필요 없다.//ex new BorderLayout(); new BorderLayout(10, 20);
GridLayout 배치관리자
📐 부착 방법 : add(Component comp)
-
생성자
- GridLayout()
- GridLayout(int rows, int cols)
- GridLayout(int rows, int cols, int hGap, int vGap)
- rows : 그리드의 행 수, 디폴트 1
- cols : 그리드의 열 수, 디폴트 1
- hGap : 좌우 컴포넌트 사이의 수평 간격, 픽셀 단위. 디폴트 0
- vGap : 상하 컴포넌트 사이의 수직 간격, 픽셀 단위. 디폴트 0
📌 컨테이너가 커져도 위치 동일,,,하게 하는 방법!!! 이거다,,,,,,이거야,,,,!!!!!!!!!!!!!!!!!!!!!!!!! 아닌감,,?
배치관리자가 없는 컨테이너
배치관리자는 컴포넌트를 절대적인 위치에 배치하지 않고, 다른 컴포넌트의 오르녹이나 아래와 같이 상대적인 위치에 배치한다. 그러므로 컨테이너의 크기가 변하면 컴포넌트의 위치도 함꼐 변한다.
📌 배치관리자를 없애야되나,,,?
-
컨테이너의 배치관리자 제거
container.setLayout(null);
-
컴포넌트의 절대 위치와 절대 크기 설정
배치관리자를 제거하게 되면 모든 컴포넌트들의 값이 0이 되기 때문에 절대 위치와 절대 크기의 설정이 필요하다.
void setSize(int width, int height)
void setLovation(int x, int y)
void setBounds(int x, int y, int width, int height)
📍 CBD
Component Based Develop 또는 Class Based Develop 의 의미로, 컴포넌트를 기반으로 개발되는 프로그램을 말한다. 이 개념이 AWT를 통해서 명확히 알 수 있었다. AWT에서 구현하고 있는 클래스 들을 사용하여서 컴포넌트 를 만들고, 컴포넌트들이 모여서 컨테이너를 만들고 컨테이너들이 모여서 하나의 최상위 컨테이너를 만드는 것을 알 수 있습니다.
📍 ECA Rule
Event-Condition-Action Rule의 의미로 Event 가 발생하면 거기서 발생하는 메시지를 통해서 메소드 이름, 변수 종류, 변수 타입을 분석하여 실행할 코드를 선택(Condition) 하여 이에 맞는 메소드를 구동(Action) 시켜주는 것을 말합니다. ECA Rule의 목적은 Method Invocation 입니다.
❓ main() 메소드가 종료한 뒤에도 프레임이 살아있는 이유?
자바 응용 프로그램이 시작되면 JVM은 main 스레드를 만들고 main()을 실행시킨다. 응용프로그램이 스레드를 만들지 않는 경우, main()이 종료하면 main 스레드도 종료되고, 더이상의 스레드가 없으므로 응용프로그램이 종료된다. 하지만 AWT, Swing에서 Frame, JFrame 객체가 생성되면 main 스레드 외의 입력되는 키와 마우스의 움직임을 컴포넌트에게 이벤트로 전달하는 이벤트 처리 스레드가 자동으로 추가되어 main 스레드에 남아있게 된다. 따라서 프레임의 경우는 키와 마우스 입력을 main 스레드에서 계속 처리하기 때문에 프레임이 살아있는 것이다.
댓글남기기