เขียนเกมด้วย LibGDX 7 – Simple Game - scene2d.ui
![เขียนเกมด้วย LibGDX 7 – Simple Game - scene2d.ui](/_vercel/image?url=%2Fimages%2F2014%2F08%2Flibgdx-cover.png&w=828&q=80)
วันนี้จะมาแนะนำการทำหน้า MenuScreen ด้วย scene2d.ui ครับ จากบทความที่แล้ว imple Game ภาคพิเศษ เราได้ตัวเกมที่มีหน้าเมนูก่อนเข้าเล่นเกม แต่ว่าไม่มีเมนูอะไรให้เลือกเลย เช่น กด Start กด HighScore อะไรพวกนี้ เป็นต้น วันนี้ก็เลยจะมานำเสนอวิธีการทำ Menu ดังกล่าวครับ
บทความตอนอื่นๆ ในซีรีย์ เขียนเกมด้วย LibGDX ติดตามอ่านได้ตามลิงค์ข้างล่างเลยครับ
- เขียนเกมด้วย libGDX #1 – สร้างโปรเจ็ค
- เขียนเกมด้วย LibGDX #2 – Hello World
- เขียนเกมด้วย LibGDX #3 – Render และการรับ input
- เขียนเกมด้วย LibGDX #4 – Simple Game ภาคแรก
- เขียนเกมด้วย LibGDX #5 – Simple Game ภาคจบ
- เขียนเกมด้วย LibGDX #6 – Simple Game ภาคพิเศษ
- เขียนเกมด้วย LibGDX #7 - Simple Game - scene2d.ui
- เขียนเกมด้วย LibGDX #8 - Simple Game - Actor
มาเริ่มกันเลยครับ ต่อจากโค๊ดที่บทความที่แล้วเลย หน้าเมนูของเรา ตอนนี้จะมีเพียงแค่นี้เอง ไม่มีอะไรให้เลือกเลย แค่รับ input สำหรับแตะหน้าจอเท่านั้น ถึงจะเริ่มเกม
Stage
Stage คือ InputProcessor
เปรียบเสมือนตัวรับอินพุทต่างๆ ประมาณมันคอยตรวจจับ input เช่น หากเราต้องการสร้าง Button ปุ่มกด เราก็จำเป็นจะต้องใช้ Stage ครับ
ที่ไฟล์ MainMenuScreen.java
ให้เราทำการประกาศสร้าง Stage
ชื่อว่า stage ครับ
1public class MainMenuScreen implements Screen {2 final Drop game;3 OrthographicCamera camera;4 private Stage stage;5
6 // ส่วนอื่นๆของโค๊ด
Actor
Actor
มันก็ Scene graph เป็นส่วนหนึ่งของ Stage มันสามารถที่จะ draw, render รูปภาพ หรือว่ารับ Input ก็ได้ อย่าง Widget ต่างๆเช่น Button Label Checkbox มันก็คือ Actor นั่นเอง (ถ้าผมเข้าใจไม่ผิดนะ ฮ่าๆ เดี่ยวต้องกลับไปอ่านอีกรอบ)
Add Menu Button
โอเค มาถึงวิธีการเพิ่ม Button กันเลยดีกว่า สำหรับ Button LibGDX นั้น ก็มีมาให้แล้วโดยใช้ scene2d.ui ครับ ส่วน Button จำเป็นต้องมี Skins ของ Button ครับ สามารถดาวน์โหลด Skins ได้ตามข่างล้างนี้
หรือว่าโหลดซิปนี้ครับ ผมรวมไฟล์ /assets
ที่จะใช้ทั้งหมด แตกซิปไปทับของเก่าได้เลยครับ
assets.zip บน Google Drive
สำหรับรายละเอียดไฟล์ข้างบน ตอนนี้ยังไม่ต้องสนใจมากครับ ว่ามันไว้ทำอะไร แค่เอามาใช้ได้ก็พอ โดยรวมแล้วมันก็คือ Style ของ Button นั่นเอง บางไฟล์ก็จะระบุพิกัดตำแหน่ง ของรูป ของ Button ต่างๆครับ หากสนใจ ก็ลองๆ เปิดไฟล์ดูครับ น่าจะดูไม่ยาก ฟอแมตก็ json แล้วก็ atlas ก็คล้ายๆ css ครับ
สำหรับคนที่ยังงงเรื่อง Skin อยู่ ก็ขออธิบายเพิ่มเติมแล้วกันครับ ว่า Skin มันก็เปรียบเสมือนหน้าตาของ Widget นั้นๆนั่นเอง นึกถึงเวลาคุณเขียนเว็บไซต์ ก็ต้องมี stylesheet กำหนด button ให้มีสีนั้นสีนี้ กำหนด background ให้มัน หรือว่าใน Android ก็จะมี Button โดยกำหนด attribute ให้มันเช่น ใส่ background, textSize, padding ต่างๆ เป็นต้นครับ
ใน LibGDX หากเราต้องการสร้าง Button เราจำเป็นต้องมี Skin ด้วยครับ ทำการประกาศสร้างตัวแปร skin ไว้ในคลาส MainMenuScreen
ครับ
1public class MainMenuScreen implements Screen {2
3 final Drop game;4 OrthographicCamera camera;5 private Stage stage;6 private Skin skin;7 // ส่วนอื่นๆ
จากนั้นในส่วน constructor ก็ทำการสร้างออปเจ็คใหม่ขึ้นมา ดังนี้
1 public MainMenuScreen(final MainGame game) {2 this.game = game;3
4 stage = new Stage();5 Gdx.input.setInputProcessor(stage);6
7 skin = new Skin(Gdx.files.internal("uiskin.json"));
จากโค๊ดด้านบน ผมได้ทำการสร้างออปเจ็ค Stage ขึ้นมาใหม่ จากนั้นก็ใช้ Gdx.input.setInputProcessor(stage)
เพื่อทำให้ stage เป็นตัวรับ handler ต่างๆครับ คือคอยรับ input จากผู้เล่น
ต่อมาก็สร้างออปเจ็ค Skin ขึ้นมาใหม่ โดยโหลดจาก /assets/uiskin.json
สุดท้ายที่ เมธอด render()
ก็ได้เปลี่ยนให้เหลือเพียงแค่ใช้ Stage#act()
และ Stage#draw()
สำหรับเรนเดอร์ครับ เพราะว่าเราจะใช้ Stage ในการเรนเดอร์แล้วครับ ไม่ใช้ SpriteBatch แล้ว
1 @Override2 public void render(float delta) {3 Gdx.gl.glClearColor(0, 0, 0.2f, 1);4 Gdx.gl.glClear(GL20.GL_COLOR_BUFFER_BIT);5
6 stage.act(Gdx.graphics.getDeltaTime());7 stage.draw();8 }
ต่อมา สร้าง TextButton สมมติผมสร้าง Button ชื่อ ‘START’ เพื่อเอาไว้สำหรับเมื่อกดปุ่มนี้ ก็จะไปที่หน้า MainGame เลย (แต่ว่าตอนนี้ทำให้โชว์ก่อนครับ ยังไม่ได้ให้รับ event ใดๆ) สร้างต่อจาก Skin เลย
1skin = new Skin(Gdx.files.internal("uiskin.json"));2
3TextButton buttonStart = new TextButton("START", skin);4buttonStart.setWidth(200);5buttonStart.setHeight(50);6buttonStart.setPosition(800 / 2 - 200 / 2, 300);
จากด้านบน ผมทำการสร้าง TextButton
โดยจะให้มันชื่อว่า ‘STARTและโหลด Style ต่างๆ จาก Skin ครับ ไอ้ตัว Skin มันก็ไปโหลด
uiskin.json` มาอีกทอดนึง แล้ว uiskin.json ก็จะไปโหลด uiskin.atlats แล้ว … พอๆ ฮ่าๆ ก็แค่รู้ว่า TextButton มันโหลด Skin มาครับ :)
จากนั้น เซตค่าให้มัน ความกว้าง ความสูง แล้วก็ตำแหน่งครับ ผมจัดให้มันอยู่กึ่งกลางหน้าจอ โดยแกน y = 300 ส่วน buttonStart.setPosition(800 / 2 - 200 / 2, 300);
คงไม่งงกันนะครับ จอกว้าง 800 ปุ่มมีขนาด 200 จะให้มันอยู่กึ่งกลาง ต้องจัดที่ x เท่าไหร่ (x นับจากมุมซ้ายของปุ่มนะครับ อย่าลืม)?
สุดท้าย ทดสอบ รันโปรแกรมเลย อยากเห็นปุ่มเต็มแก่แล้ว
**เอ้าเห้ย ทำไมปุ่มไม่โชว์ละเนี่ย ? **
ยังจำ Stage กันได้มั้ยครับ ตัว Stage มันก็เปรียบเสมือนฉากเกมๆหนึ่ง ที่คอยรับ input ต่างๆ รวมถึงแสดงผลออกทางจอภาพ ทุกๆอยากที่จะแสดงผล จำเป็นต้องเพิ่มเข้าไปใน Stage ด้วย เพิ่มอันนี้ต่อท้าย TextButton เมื่อกี้ครับ
1stage.addActor(buttonStart);
ทดสอบอีกรอบ มหัศจรรย์ เพิ่มบรรทัดเดียว เห็นผลเลย ^^
ว้าว ปุ่มกดโชว์แล้วครับ แต่ว่ายังไม่สามารถทำอะไรกับมันได้ ก็เพิ่ม InputListener ให้กับปุ่มกดครับ ปกติใน Android พวก Button เราสามารถ setOnClickListener
ให้มันได้ใช้มั้ยครับ ตัว scene2d.ui#Button มันก็มีเหมือนกันครับ นั่นก็คือ addListener
1buttonStart.addListener(new ClickListener() {2 @Override3 public void clicked(InputEvent event, float x, float y) {4 super.clicked(event, x, y);5 game.setScreen(new GameScreen(game));6 }7});
ด้านบน ผมทำการ addListener จากนั้นก็ override เมธอด clicked
เพื่อรอดักจับว่าเมื่อมีการกดปุ่มนี้เมื่อไหร่ ก็ให้ทำการเปิดหน้า GameScreen เลย ง่ายมั้ย?
ก่อนจบ ที่ เมธอด resize()
เพิ่มอันนี้เข้าไป เพื่อให้ Stage มันปรับขนาดเวลาหน้าจอของเรามีการเปลี่ยนแปลง
1@Override2public void resize(int width, int height) {3 stage.getViewport().update(width, height, true);4}
สุดท้าย อย่าลืม dispose สิ่งที่เราสร้างมาครับ คือ Stage และ Skin ส่วนสิ่งต่างๆที่อยู่ในออปเจ็ค Drop อย่าไป dispose มันนะครับ มันยังต้องใช้งานอยู่ ^^
1@Override2public void dispose() {3 stage.dispose();4 skin.dispose();5}
โค๊ดทั้งหมด
MainMenuScreen.java
1package com.badlogic.drop;2
3import com.badlogic.gdx.Gdx;4import com.badlogic.gdx.Screen;5import com.badlogic.gdx.graphics.GL20;6import com.badlogic.gdx.graphics.OrthographicCamera;7import com.badlogic.gdx.scenes.scene2d.InputEvent;8import com.badlogic.gdx.scenes.scene2d.Stage;9import com.badlogic.gdx.scenes.scene2d.ui.Skin;10import com.badlogic.gdx.scenes.scene2d.ui.TextButton;11import com.badlogic.gdx.scenes.scene2d.utils.ClickListener;12import com.badlogic.gdx.utils.viewport.StretchViewport;13
14public class MainMenuScreen implements Screen {15
16 final Drop game;17
18 OrthographicCamera camera;19
20 private Stage stage;21 private Skin skin;22
23 public MainMenuScreen(final Drop gam) {24 game = gam;25
26 camera = new OrthographicCamera();27 camera.setToOrtho(false, 800, 480);28
29 stage = new Stage(new StretchViewport(800, 480));30 Gdx.input.setInputProcessor(stage);31
32 skin = new Skin(Gdx.files.internal("uiskin.json"));33
34 TextButton buttonStart = new TextButton("START", skin);35 buttonStart.setWidth(200);36 buttonStart.setHeight(50);37 buttonStart.setPosition(800 / 2 - 200 / 2, 300);38
39 stage.addActor(buttonStart);40
41 buttonStart.addListener(new ClickListener() {42 @Override43 public void clicked(InputEvent event, float x, float y) {44 super.clicked(event, x, y);45 game.setScreen(new GameScreen(game));46 }47 });48
49 }50
51 @Override52 public void render(float delta) {53 Gdx.gl.glClearColor(0, 0, 0.2f, 1);54 Gdx.gl.glClear(GL20.GL_COLOR_BUFFER_BIT);55
56 stage.act(Gdx.graphics.getDeltaTime());57 stage.draw();58 }59
60 @Override61 public void resize(int width, int height) {62 stage.getViewport().update(width, height, true);63 }64
65 @Override66 public void show() {67 }68
69 @Override70 public void hide() {71 }72
73 @Override74 public void pause() {75 }76
77 @Override78 public void resume() {79 }80
81 @Override82 public void dispose() {83 stage.dispose();84 skin.dispose();85 }86}
หากใครมีปัญหาตรงไหน สอบถามได้ครับ หรือใครมีคำแนะนำ ข้อเสนอแนะ ติชม ก็พร้อมยินดีครับ อยากเห็นคนเก่งๆมาช่วยชี้แนะครับ มาแชร์ประสบการณ์กัน หากใครเห็นว่าบทความนี้มีประโยชน์ อย่าลืมบอกต่อเพื่อนๆกันด้วยนะครับ :D
- Authors
-
Chai Phonbopit
เป็น Web Dev ในบริษัทแห่งหนึ่ง ทำงานมา 10 ปีกว่าๆ ด้วยภาษาและเทคโนโลยี เช่น JavaScript, Node.js, React, Vue และปัจจุบันกำลังสนใจในเรื่องของ Blockchain และ Crypto กำลังหัดเรียนภาษา Rust