用JavaFX編寫圖塊引擎

隨著JavaFX嵌入式版本的問世,我們的框架對于游戲開發變得越來越有趣,因為我們現在可以瞄準平板電腦和智能手機等小型消費類設備。 因此,我決定對JavaFX進行更多的游戲編寫實驗。 這次,我想使用Canvas對渲染進行更多控制,以便能夠在較小的設備上優化性能。 這些是我編寫Tile Engine時的經驗。

早期,游戲機和計算機的資源非常有限。 因此,為了使游戲具有成千上萬的大屏幕,開發人員需要想出一種方法來以每個屏幕的位圖以外的格式存儲屏幕。 因此,發明了Tile Engine,它們可以從有限的一組可重復使用的較小圖形(標題)中生成大屏幕。 這樣可以節省內存并提高渲染性能。

如何生成屏幕的說明存儲在TileMaps中。 這些地圖通常組織為Tile ID的二維矩陣。 通常,磁貼按層進行組織,以實現簡單的Z順序,并在組合具有不同背景的圖形時具有更大的靈活性。 通常,TileMaps還支持存儲元數據,例如,如果某些圖塊被阻止或敵人的生成點。

帶有多個圖層的TileMap,使用

映射中引用的圖塊通常存儲在TileSet中,該圖塊由單個位圖和有關如何將其劃分為圖塊的元信息組成。 這是來自opengameart.com的此類圖像的示例,該網站托管具有開放源代碼許可的游戲資產。 在我的示例中,我使用了其中一些圖形。

典型的TileSet圖像,尺寸為1024 x 1024(^ 2 =適用于圖形卡)

TMX格式的另一項功能是對象層。 這些特殊層可用于定義自由形狀和折線并為其指定屬性。 其背后的基本思想是,我們可以使用它們來定義創建精靈(生成點),出口,門戶和非矩形碰撞形狀的區域。 取決于TileEngine的創建者或使用它來構建游戲的開發者來定義如何處理ObjectGroup。 我打算廣泛使用它們,它們是用于聲明性定義游戲玩法的很好的擴展點。 例如,您可以使用它們來定義動畫,skript對話框等。

tilemap的想法也允許一個很好的工作流。 圖形設計師可以創建資產,游戲設計師可以將其導入“ Tiled”等關卡編輯器,并通過拖放來設計關卡。 地圖以機器可讀的TileMap格式存儲。 例如Tiled使用TMX Map格式存儲TileMap。 那是一種非常簡單的XML格式,然后可以由TileEngine加載。 對于我的實現,我決定使用TMX格式,因此可以使用“ Tiled ”來設計級別。

對于實現,我決定在使用單個節點時使用JavaFX Canvas立即模式渲染,而不是保留模式渲染。 這使我有了更多控制權,可以優化Raspberry Pi等小型設備的性能。

我們需要的第一件事是讀取TileMap(TMX)和TileSet(TSX)文件的方法 。 使用JAXB,創建可以從文件創建POJO的TileMapReader非常簡單。 因此,如果您使用引擎,則只需調用:

TileMap map = TileMapReader.readMap(“path/to/my/map.tmx”);

由于在大多數游戲中,“ TileMaps”將比屏幕大,因此僅渲染“ Map”的一部分。 通常,地圖以英雄為中心。 您只需跟蹤屏幕左上角的地圖位置即可。 我們將此稱為我們的相機位置。 然后,在像這樣渲染TileMap之前,從英雄的位置更新位置:

// the center of the screen is the preferred location of our herodouble centerX = screenWidth / 2;double centerY = screenHeight / 2;cameraX = hero.getX() - centerX;cameraY = hero.getY() - centerY;

我們只需要確保相機沒有離開圖塊地圖即可:

// if we get too close to the bordersif (cameraX >= cameraMaxX) {cameraX = cameraMaxX;}if (cameraY >= cameraMaxY) {cameraY = cameraMaxY;}

使用Canvas渲染TileMap

然后,渲染圖塊非常容易。 我們只需遍歷圖層,并要求tilemap在當前位置渲染正確的圖像。 首先,我們需要找出當前可見的圖塊以及偏移量,因為我們的英雄逐像素而不是逐圖地移動:

// x,y index of first tile to be shownint startX = (int) (cameraX / tileWidth);int startY = (int) (cameraY / tileHeight);// the offset in pixelsint offX = (int) (cameraX % tileWidth);int offY = (int) (cameraY % tileHeight);Then we loop through the visible layers and draw the tile:for (int y = 0; y < screenHeightInTiles; y++) {for (int x = 0; x < screenWidthInTiles; x++) {// get the tile id of the tile at this positionint gid = layer.getGid((x + startX) + ((y + startY) * tileMap.getWidth()));graphicsContext2D.save();// position the graphicscontext for drawinggraphicsContext2D.translate((x * tileWidth) - offX, (y * tileHeight) - offY);// ask the tilemap to draw the tiletileMap.drawTile(graphicsContext2D, gid);// restore the old stategraphicsContext2D.restore();}}

然后,TileMap將找出該Tile屬于哪個Tileset,并要求TileSet將其繪制到Context。 繪制本身就像在TileSets圖像中找到正確的坐標一樣簡單:

public void drawTile(GraphicsContext graphicsContext2D, int tileIndex) {int x = tileIndex % cols;int y = tileIndex / cols;// TODO support for margin and spacinggraphicsContext2D.drawImage(tileImage, x * tilewidth, y* tileheight, tilewidth, tileheight, 0, 0, tilewidth, tileheight);}

游戲循環。 因此,我們可以將其簡化為:

游戲循環再次非常簡單。 我正在使用時間軸和關鍵幀以特定幀率(FPS)為游戲觸發脈沖:

final Duration oneFrameAmt = Duration.millis(1000 / FPS);final KeyFrame oneFrame = new KeyFrame(oneFrameAmt,new EventHandler() {@Overridepublic void handle(Event t) {update();render();}});TimelineBuilder.create().cycleCount(Animation.INDEFINITE).keyFrames(oneFrame).build().play();

TileMapCanvas中的每個update更新都循環遍歷所有Sprites并對其進行更新。 基本Sprite當前包含一個帶有行走周期的TileSet,如下所示:

由于子畫面通常在其周圍有很多透明空間,因此為了為動畫行為(例如揮劍)提供一些額外的空間,為方便起見,我決定允許添加MoveBox和CollisionBox。 CollisionBox可以用于定義我們的英雄可能受到傷害的區域。 MoveBox應該放在腿周圍,這樣它就可以在上半身與瓷磚重疊的情況下通過禁止的瓷磚前面。 我們的“英雄”周圍的藍色區域是精靈邊界:

https://www.youtube.com/watch?v=08H6LZkcqXw

子畫面也可以具有定時行為。 在每次更新時,Sprite都會循環遍歷其行為,并檢查是否該觸發。 如果是這樣,則調用“行為”方法。 如果我們有一個敵人,例如示例應用程序中的骨架,我們可以在此處添加它為AI。 例如,我們的骷髏具有非常簡單的行為,可以使其跟隨我們的英雄。 它還會檢查碰撞并像這樣對我們的英雄造成傷害:

monsterSprite.addBehaviour(new Sprite.Behavior() {@Overridepublic void behave(Sprite sprite, TileMapCanvas playingField) {if (sprite.getCollisionBox().intersects(hero.getCollisionBox())) {hero.hurt(1);}}});

默認間隔是一秒鐘。 如果需要其他間隔,可以設置它們。 行為是可重用的,不同的Sprite可以共享相同的Behavior實例。 行為與KeyFrames相似,并且我目前還使用它們來為Animations計時(增加下一個渲染調用的tile索引)。

如開頭所述,ObjectGroup是方便的擴展點。 在我的示例游戲中,我使用它們來定義英雄和怪物的生成點。 當前,您只需添加一個ObjectGroupHandler,然后使用ObjectGroup中的信息來創建Hero和Monster精靈并將行為添加到它們:

class MonsterHandler implements ObjectGroupHandler {Sprite hero;@Overridepublic void handle(ObjectGroup group, final TileMapCanvas field) {if (group.getName().equals('sprites')) {for (TObject tObject : group.getObjectLIst()) {if (tObject.getName().equals('MonsterSpawner')) {try {double x = tObject.getX();double y = tObject.getY();TileSet monster = TileMapReader.readSet('/de/eppleton/tileengine/resources/maps/BODY_skeleton.tsx');Sprite monsterSprite = new Sprite(monster, 9, x, y, 'monster');monsterSprite.setMoveBox(new Rectangle2D(18, 42, 28, 20));field.addSprite(monsterSprite);monsterSprite.addBehaviour(new Sprite.Behavior() {@Overridepublic void behave(Sprite sprite, TileMapCanvas playingField) {if (sprite.getCollisionBox().intersects(hero.getCollisionBox())) {hero.hurt(1);}}});}

放在一起

要創建一個示例游戲,您需要做的就是創建TileMaps,TileSets,一個或多個ObjectGroupHandler來創建Sprites并添加Behavior,然后就可以開始游戲了:

// create the worldTileMap tileMap = TileMapReader.readMap('/de/eppleton/tileengine/resources/maps/sample.tmx');// initialize the TileMapCanvasTileMapCanvas playingField = new TileMapCanvas(tileMap, 0, 0, 500, 500);// add Handlers, can also be done declaratively.playingField.addObjectGroupHandler(new MonsterHandler());// display the TileMapCanvasStackPane root = new StackPane();root.getChildren().add(playingField);Scene scene = new Scene(root, 500, 500);playingField.requestFocus();primaryStage.setTitle('Tile Engine Sample');primaryStage.setScene(scene);primaryStage.show();

那是我的Tile Engine的起點。 同時,它已經發展成為更通用的2D引擎,因此還支持不使用TileSet的Sprite和自由渲染的Layers。 到目前為止,它仍然運行良好。

參考: Eppleton博客上的JCG合作伙伴 Toni Epple 用JavaFX編寫了一個Tile Engine 。

翻譯自: https://www.javacodegeeks.com/2013/01/writing-a-tile-engine-in-javafx.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/370457.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/370457.shtml
英文地址,請注明出處:http://en.pswp.cn/news/370457.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

python命令行運行模式_[Python] 命令行模式閱讀博客園的博文

1 #-*- coding:UTF-8 -*-2 importrequests3 from lxml importetree4 importsys5 importio6 importos789 sys.stdout io.TextIOWrapper(sys.stdout.buffer, encodinggb18030)101112 classCnBlogs:13 """"14 Auth&#xff1a;reader15 發表地址&#xff1a;…

HTML5--應用網頁模板

因為剛開始寫博客,只想著把知識點記錄在這,也想給你們一些參考,在布局上有些沒有思考太多;回過頭來看,實在是不忍直視,對不住之前閱讀的100 ,既然昨天的事無法挽回,那就從現在開始從新整改吧!也希望大家看了,能對你們有所幫助 1.先給大家看看效果圖,好讓大家有點興趣 2.大家再來…

企業集成模式簡介

在此博客文章中&#xff0c;我們將介紹一些企業集成模式。 這些是旨在解決集成挑戰的已知設計模式。 閱讀此書后&#xff0c;您將可以設計集成解決方案。 EIP&#xff08;簡而言之&#xff09;是已知的設計模式&#xff0c;可為應用程序集成過程中遇到的問題/問題提供解決方案…

手把手教你Chrome瀏覽器安裝Postman(含下載云盤鏈接)【轉載】

轉載自&#xff1a;http://www.ljwit.com/archives/php/278.html 說明&#xff1a; Postman不多介紹&#xff0c;是一款功能強大的網頁調試與發送網頁HTTP請求的Chrome插件。本文主要介紹下安裝過程。 本文使用的是解壓文件直接進行安裝。是比較快速有效的安裝方式&#xff0c;…

C語言博客作業--數據類型

題目1&#xff1a;7-4 打印菱形圖案 1. 本題PTA提交列表 2. 設計思路 1.定義變量i,j,k,n;且聲明i為要打印的行數&#xff0c;j是控制輸出打印空格和星星&#xff0c;n是菱形為菱形的高 2.輸入n 3.i1&#xff0c;j1 4.先打印上半部分&#xff0c;第一行到n/21行&#xff0c;輸出…

信息隱藏將txt文件合并到jpg文件中_使用Kali Linux在圖像內隱藏機密消息—可在任何Linux發行版使用

歡迎回到“Esn技術社區”&#xff01;今天&#xff0c;我們將演示如何使用Steghide(一種可在Kali Linux上使用的流行隱寫工具)在圖像內隱藏消息。在計算機科學中&#xff0c;將信息隱藏在文件內(例如圖像&#xff0c;文檔&#xff0c;程序&#xff0c;有用數據&#xff0c;消息…

Spring 3.1,Cloud Foundry和本地開發

這篇文章將幫助您在Cloud Foundry上使用MongoDB構建Spring 3.1 Web應用程序。 除了推動Cloud Foundry之外&#xff0c;您還可以使用MongoDB實例在本地環境中進行開發。 目標 此博客發布的目標是在本地構建應用程序&#xff0c;然后發布到本地Cloud Foundry實例。 我們將利用C…

Spring MVC 簡述:從MVC框架普遍關注的問題說起

任何一個完備的MVC框架都需要解決Web開發過程中的一些共性的問題&#xff0c;比如請求的收集與分發、數據前后臺流轉與轉換&#xff0c;當前最流行的SpringMVC和Struts2也不例外。本文首先概述MVC模式的分層思想與MVC框架普遍關注的問題&#xff0c;并以此為契機結合SpringMVC的…

java方法調用機制_Java方法調用機制 - osc_bkdv2it5的個人空間 - OSCHINA - 中文開源技術交流社區...

最近在編程時&#xff0c;修改方法傳入對象的對象引用&#xff0c;并沒有將修改反映到調用方法中。奇怪為什么結果沒有變化&#xff0c;原因是遺忘了Java對象引用和內存分配機制。本文介紹3個點&#xff1a;① 該問題舉例說明② 簡要闡述Java內存區域③ 介紹JVM中方法調用的機制…

CSS染色圖標(圖片)

之前一直以為用background引入的圖標無法染色&#xff08;非字體圖標&#xff09;&#xff0c;現在才知道有黑科技可以用&#xff0c;就是利用drop-shadow。 代碼示例 <!DOCTYPE html> <html> <head lang"en"><meta charset"UTF-8"&…

eclipse安裝java web插件

1 查看eclipse版本 找到eclipse的安裝目錄&#xff0c;找到readme文件&#xff0c;打開其中的html文件&#xff0c;我的是4.6版本的,代號是oxygen 2 安裝 打開eclipse,點擊help-Install new software-單擊add&#xff0c;在彈出窗口中輸入網址&#xff1a; http://download.ecl…

python正則表達式指南_Python正則表達式指南

1. 正則表達式基礎1.1. 簡單介紹正則表達式并不是Python的一部分。正則表達式是用于處理字符串的強大工具&#xff0c;擁有自己獨特的語法以及一個獨立的處理引擎&#xff0c;效率上可能不如str自帶的方法&#xff0c;但功能十分強大。得益于這一點&#xff0c;在提供了正則表達…

Google Guava EventBus用于事件編程

在任何軟件應用程序中都是如此&#xff0c;有些對象需要共享信息才能完成工作。 在Java應用程序中&#xff0c;實現信息共享的一種方法是擁有事件偵聽器&#xff0c;其唯一目的是在發生所需事件時采取某些措施。 在大多數情況下&#xff0c;此過程有效&#xff0c;并且最有經驗…

system類

package system.cn; /** system類的方法 都是靜態方法&#xff0c;可以直接用類名直接調用* 常用的方法&#xff1a;* static long currentTimeMillis() 返回以毫秒為單位的當前時間。 static void exit(int status) 終止當前正在運行的 Java 虛擬機。 static void gc…

c await和java_blog/java/test/awaitility.zh.md at master · c-rainstorm/blog · GitHub

javaAtomicInteger atomic new AtomicInteger(0);// Do some async stuff that eventually updates the atomic integerawait().untilAtomic(atomic, equalTo(1));等待一個 AtomicBoolean 更簡單&#xff1a;javaAtomicBoolean atomic new AtomicBoolean(false);// Do some a…

實現輸入框小數多 自動進位展示,編輯時實際值不變

今天遇到個業務需求&#xff0c;要求輸入框&#xff0c;輸入數字的小數位數可以很多位&#xff0c;但移開后顯示&#xff0c;只顯示小數點后兩位 &#xff08;四舍五入&#xff09;&#xff0c;當要編輯的時候&#xff0c;展現其原來的輸入數據。 閑話不多說&#xff0c;當時也…

使用Jasper Reports以Java創建報告

上周&#xff0c;我試圖使用Jasper創建報告。 在這篇文章中&#xff0c;我將記錄一些資源和鏈接&#xff0c;以便對任何尋求類似信息的人都有用。 我將介紹Jasper報告&#xff0c;示例和Dynamic Jasper的生命周期。 Jasper Reports是世界上最受歡迎的開源報告引擎。 它完全用…

CentOS7 安裝NodeJS

一、切換目錄到/usr/local/src 命令行&#xff1a;cd /usr/local/src 二、下載node.js&#xff08;我這里下載的是二進制的源碼&#xff09; 命令行&#xff1a; wget https://nodejs.org/dist/v8.9.1/node-v8.9.1-linux-x64.tar.xz 圖片&#xff1a; 三、解壓壓縮包 命令行&am…

CSS3基礎2(變形與動畫)

<!DOCTYPE html5><html lang"en"><head> <meta charset"UTF-8"> <title>CSS3基礎知識&#xff08;動畫&#xff09;</title> <style> /*div{*/ /*width: 150px;*/ /*hei…

java對hashmap迭代_Java:通過HashMap迭代,這樣更有效率?

第二個選項肯定更有效&#xff0c;因為在第一個選項中只進行一次查找&#xff0c;次數為n次。但是&#xff0c;沒有什么比嘗試它更好&#xff0c;當你可以。所以這里 –(不完美&#xff0c;但足夠好驗證假設和我的機器)public static void main(String args[]) {Map map new H…