1) Extract the javafx4linux.tar.bz2 file. In this example I’m placing it on my Desktop. After the installing process you can remove it.
2) Open your NetBeans 6.5 and go at Tools → Plugins and go to Downloaded tab. In a plain and new NetBeans installation there will be no plugin in this tab yet.
3) Click on the Add Plugins button and head to the directory you extracted the file and select all .nbm files.
4) You will see a list of 22 plugins selected. Click on the Install button.
5) Just keep clicking on the Next button.
6) Check the license agreement accept box.
7) You’ll see a warning because the Linux pluggin is not signed. Don’t worry, just click Continue.
8) Click on Finish to restart NetBeans.
9) Now we can test it. Go at File → New Project, select the JavaFX on Categories and JavaFX Script Application on Projects.
10) Put some code and run it. There is. JavaFX on Linux.
Considerations
This is not a official of JavaFX for Linux! This solution was tested on Ubuntu 9.04 “Jaunty Jackalope” with Java 6 update 13 and NetBeans 6.5.1, but should also work with others Linux distributions and Java versions greater than 5.
Known bugs
As a non official workaround for JavaFX for Linux you may notice some drawbacks. Some parts of the JavaFX runtime rely on native implementations on the specific operational system. You may not use some multimedia capabilities as video playback, JavaFX Mobile emulator and some performance issues in some effects. Despite that, is perfectly possible to develop applications using JavaFX on NetBeans.
Twitter is a social network and micro-blogging service that allow you to create and read tweets, 140 characters text-based posts. It’s becoming a popular tool to keep in touch with your friends, coworkers, bloggers, etc. Here we’ll create a very simple application that show us tweets related with a given word.
Twitter offers a very simple and powerfull REST API which supports XML, JSON, and the RSS and Atom formats. As we are aiming just to read tweets we’ll use just the Search API.
We do that in three steps:
Query Tweets
Parser the Atom result
Show tweets into a GUI
Let’s see them in the order of dependence beetween them.
Displaying a Tweet
var gradient = LinearGradient {
startX:0.0,
startY:0.0,
endX:0.0,
endY:150.0
proportional:false
stops:[Stop {offset:0.0 color:Color.DARKGRAY},
Stop { offset:1.0 color:Color.BLACK}]}publicclass Tweet extends CustomNode {public var image:Image;public var username:String;public var message:String;public override function create(): Node {
var txt = Text {
x:65 y:35 wrappingWidth:150 fill:Color.WHITE
content:"{message}"}returnGroup{
content:[Rectangle{
width:220 height: txt.boundsInLocal.height+40
arcHeight:10 arcWidth:10 fill: gradient
},
ImageView {
x:5 y:20 image: image
},
Text {
x:65 y:20 fill:Color.BLACK content:"{username} said"},
txt
]};}}
var gradient = LinearGradient {
startX: 0.0,
startY: 0.0,
endX: 0.0,
endY: 150.0
proportional: false
stops: [Stop {offset: 0.0 color: Color.DARKGRAY },
Stop { offset: 1.0 color: Color.BLACK }]
}
public class Tweet extends CustomNode {
public var image: Image;
public var username: String;
public var message: String;
public override function create(): Node {
var txt = Text {
x: 65 y: 35 wrappingWidth: 150 fill: Color.WHITE
content: "{message}"
}
return Group {
content: [
Rectangle {
width: 220 height: txt.boundsInLocal.height + 40
arcHeight: 10 arcWidth: 10 fill: gradient
},
ImageView {
x: 5 y: 20 image: image
},
Text {
x: 65 y: 20 fill: Color.BLACK content: "{username} said"
},
txt
]
};
}
}
In my last post about JavaFX I showed how to parse XML documents (and make sandwiches) with JavaFX. Here we’ll use the Atom format, but use any other would be almost the same. Parsing XML or JSON documents with JavaFX is both very simple using the javafx.data.pull.PullParser class.
A query output is a Atom XML document with several information. We are interested only in the fields that holds the avatar image, the message and the user name.
var tweets = VBox {}
def parser = PullParser {
var avatar;
var firstname;
var text;
documentType: PullParser.XML;
onEvent: function(event:Event){if(event.type== PullParser.START_ELEMENT){if(event.qname.name.equals("link")){if(event.getAttributeValue(QName{name:"rel"})=="image"){
avatar = event.getAttributeValue(QName{name:"href"});}}}if(event.type== PullParser.END_ELEMENT){if(event.qname.name=="title"){
text = event.text;}if((event.qname.name=="name")and(event.level==3)){
var names:String[]= event.text.split(" ");
firstname = names[0];
insert Tweet {
image:Image{
url: avatar
}
message: text
username: firstname
} into tweets.content;}}}}
var tweets = VBox {}
def parser = PullParser {
var avatar;
var firstname;
var text;
documentType: PullParser.XML;
onEvent: function(event: Event) {
if(event.type == PullParser.START_ELEMENT){
if(event.qname.name.equals("link")){
if(event.getAttributeValue(QName{name: "rel"}) == "image"){
avatar = event.getAttributeValue(QName{name:"href"});
}
}
}
if(event.type == PullParser.END_ELEMENT) {
if(event.qname.name == "title"){
text = event.text;
}
if((event.qname.name == "name")and(event.level==3)){
var names: String[] = event.text.split(" ");
firstname = names[0];
insert Tweet {
image: Image {
url: avatar
}
message: text
username: firstname
} into tweets.content;
}
}
}
}
Querying Tweets
We can get Atom results through url queries like that:
Notice that queries should be URL encoded. We will use a additional parameters &rpp=4 to receiveonly 4 results per page. To know more about search queries read the Search API Documentation. We get these results as InputStreams making asynchronous HTTP requests using the javafx.io.http.HttpRequest class, which it’s perfect to invoke RESTful Web Services.
word ="Beatles";
var request = HttpRequest {
location:"http://search.twitter.com/search.atom?q={word}&rpp=4";
onInput: function(stream: java.io.InputStream){
parser.input= stream;
parser.parse();}}
request.enqueue();
Here is the application running for the word “House”.
Is not a complete Twitter client, as it’s not intended to be, but can show you how to handle a simple asynchronous call and handle Twitter documents. There’s already a few beta JavaFX Twitter clients like Tweetbox and Twitterfx and certanly others will appears.
We can open it using java.io.FileInputStream and so use it on a javafx.data.pull.PullParser. A PullParser is a event oriented parser that works with XML and YAML files. Above a general and simple parser with a GUI that show the list of events during the parse process.
importjava.io.FileInputStream;importjavafx.data.pull.Event;importjavafx.data.pull.PullParser;importjavafx.ext.swing.SwingList;importjavafx.ext.swing.SwingListItem;importjavafx.scene.Scene;importjavafx.stage.Stage;
var list = SwingList { width:600 height:300}
var myparser = PullParser {
documentType: PullParser.XML;
onEvent: function (e:Event){
var item = SwingListItem {
text:"event {e}"};
insert item into list.items;}
input:newFileInputStream("/tmp/sandwich.xml");}
myparser.parse();
Stage {
title:"XML Sandwich"
scene: Scene { content: list }}
import java.io.FileInputStream;
import javafx.data.pull.Event;
import javafx.data.pull.PullParser;
import javafx.ext.swing.SwingList;
import javafx.ext.swing.SwingListItem;
import javafx.scene.Scene;
import javafx.stage.Stage;
var list = SwingList { width: 600 height: 300 }
var myparser = PullParser {
documentType: PullParser.XML;
onEvent: function (e: Event) {
var item = SwingListItem {
text: "event {e}"
};
insert item into list.items;
}
input: new FileInputStream("/tmp/sandwich.xml");
}
myparser.parse();
Stage {
title: "XML Sandwich"
scene: Scene { content: list }
}
Notice that white spaces like tab and escape characters like new line also produced events from type TEXT. We are not interested on them. Above a parser that looks only those events of type START_ELEMENT or END_ELEMENT, look into it’s contents, building a sandwich at runtime based on the XML file.
importjava.io.FileInputStream;importjavafx.data.pull.Event;importjavafx.data.pull.PullParser;importjavafx.data.xml.QName;importjavafx.scene.image.Image;importjavafx.scene.image.ImageView;importjavafx.scene.layout.VBox;importjavafx.scene.Scene;importjavafx.stage.Stage;// my sandwich starts as an empty VBox
var mysandwich = VBox {}// give a name and returns a ImageView with a png image like the name
function ingredient(name){return ImageView {
image:Image{
url:"{__DIR__}{name}.png"}}}// basicaly, look the event and put a ingredient at mysandwich
var myparser = PullParser {
documentType: PullParser.XML;
onEvent: function (e:Event){// starter xml elementsif(e.type== PullParser.START_ELEMENT){// breadif(e.qname.name.equals("bread")){
insert ingredient("bread_top") into mysandwich.content;}// hamburguerif(e.qname.name.equals("hamburguer")){
insert ingredient("hamburguer") into mysandwich.content;}// catchupif(e.qname.name.equals("catchup")){
insert ingredient("catchup") into mysandwich.content;}// maioneseif(e.qname.name.equals("maionese")){
insert ingredient("maionese") into mysandwich.content;}// lettuceif(e.qname.name.equals("lettuce")){
insert ingredient("lettuce") into mysandwich.content;}// cheeseif(e.qname.name.equals("cheese")){
var type= e.getAttributeValue(QName{name:"type"});if(type.equals("cheedar")){
insert ingredient("cheedar") into mysandwich.content;}else{
insert ingredient("cheese") into mysandwich.content;}}}// ending xml elements (just bread)if(e.type== PullParser.END_ELEMENT){if(e.qname.name.equals("bread")){
insert ingredient("bread_botton") into mysandwich.content;}}}
input:newFileInputStream("/tmp/sandwich.xml");}
myparser.parse();
Stage {
title:"XML Sandwich"
scene: Scene {
height:300
content: mysandwich
}}
import java.io.FileInputStream;
import javafx.data.pull.Event;
import javafx.data.pull.PullParser;
import javafx.data.xml.QName;
import javafx.scene.image.Image;
import javafx.scene.image.ImageView;
import javafx.scene.layout.VBox;
import javafx.scene.Scene;
import javafx.stage.Stage;
// my sandwich starts as an empty VBox
var mysandwich = VBox {}
// give a name and returns a ImageView with a png image like the name
function ingredient(name){
return ImageView {
image: Image {
url: "{__DIR__}{name}.png"
}
}
}
// basicaly, look the event and put a ingredient at mysandwich
var myparser = PullParser {
documentType: PullParser.XML;
onEvent: function (e: Event) {
// starter xml elements
if(e.type == PullParser.START_ELEMENT){
// bread
if(e.qname.name.equals("bread")){
insert ingredient("bread_top") into mysandwich.content;
}
// hamburguer
if(e.qname.name.equals("hamburguer")){
insert ingredient("hamburguer") into mysandwich.content;
}
// catchup
if(e.qname.name.equals("catchup")){
insert ingredient("catchup") into mysandwich.content;
}
// maionese
if(e.qname.name.equals("maionese")){
insert ingredient("maionese") into mysandwich.content;
}
// lettuce
if(e.qname.name.equals("lettuce")){
insert ingredient("lettuce") into mysandwich.content;
}
// cheese
if(e.qname.name.equals("cheese")){
var type= e.getAttributeValue(QName{name:"type"});
if(type.equals("cheedar")){
insert ingredient("cheedar") into mysandwich.content;
} else {
insert ingredient("cheese") into mysandwich.content;
}
}
}
// ending xml elements (just bread)
if(e.type == PullParser.END_ELEMENT){
if(e.qname.name.equals("bread")){
insert ingredient("bread_botton") into mysandwich.content;
}
}
}
input: new FileInputStream("/tmp/sandwich.xml");
}
myparser.parse();
Stage {
title: "XML Sandwich"
scene: Scene {
height: 300
content: mysandwich
}
}
Here’s our sandwich.
Just changing the XML file you got a new sandwich.
Tile sets are a very simple way to draw scenarios with repeated elements. From simple to complex ones using a very low footprint.
First step, load the png file that stores the tileset into a Image. The file tiles.png shoud be in the same directory of the source code. I adjusted some tiles from those tile set I’ve blogged here before into a grid of 10×10 tiles.
var tileset =Image{
url:"{__DIR__}tiles.png"}
var tileset = Image {
url: "{__DIR__}tiles.png"
}
Notice that each tile have 32 of height and 32 of width. We will assume this and use theses numbers when performing calculations to find a single tile in our tile set.
def w =32;
def h =32;
def w = 32;
def h = 32;
To display a Image in the screen we use a ImageView node. A ImageView can have a viewport property to create crop or zoom effect. A viewport is just a Rectangle2D, a object with position (minX and minY), height and width. If we want to display the first tile in the tileset we do
Notice that the minX determines the column and minY the row in the tileset. The first row is 0*32, the second row is 1*32 and so on. If we want to display the tile at the second line and third column of the tileset we do
Those properties in a Rectangle2D are for init and read only. So I created a list with all Rectangles I can need for use as a viewport.
def viewports =for(row in [0..9]){for(col in [0..9]){Rectangle2D{
minX: col * w, minY: row * h, height: w, width: h
}}}
def viewports = for (row in [0..9]) {
for (col in [0..9]) {
Rectangle2D{
minX: col * w, minY: row * h, height: w, width: h
}
}
}
The scenario map is stored in another list. The first element of the list is 7, that is, the first tile in the scenario is the 7th tile from the tile set.
Finally to create a scenario with 100 tiles, 10 per row and with 10 rows, in a list called tiles. Each iteration of this loop creates a ImageView. Each ImageView will store a single tile. We get the tile number in the map list and so use it to index the viewports list.
var tiles =for(row in [0..9]){for(col in [0..9]){
ImageView {
x: col * w, y: row * h,
viewport: bind viewports[map[row *10+ col]]
image: tileset
}}}
var tiles = for (row in [0..9]) {
for (col in [0..9]) {
ImageView {
x: col * w, y: row * h,
viewport: bind viewports[map[row * 10 + col]]
image: tileset
}
}
}
Additionally I added two things to transform this program also in a (extremely) simple map editor. At each ImageView I added a callback for onMouseClicked event. When you click on a tile, it changes its map position, ie, the tile. The next tile for the left button and the last tile for any other button.
onMouseClicked: function( e:MouseEvent):Void{
var amount =if(e.button== MouseButton.PRIMARY){1}else{-1};
map[row *10+ col]=(map[row *10+ col]+ amount) mod 100;}
We are using just a image that can handle 100 tiles, tiles.png with less than 30Kb. The map is also composed with 100 tiles. Each tile we can choose between 100 different tiles, so we can compose 10100 different maps (one googol10 ). Most of them are useless and without any sense, but some are cool. :)
Following Weiqi Gao’s steps it’s possible to already have a complete JavaFX development environment on Linux.
It’s all there for JavaFX development. Code complementation, live preview, the pallet with code snipets, templates, etc. Easier a lot my work. Those features already availiable on Windows and Mac OS X trought the regular JavaFX Kit.
JavaFX 1.0 is out and there are tons of new cool features, specially for game development.trans
I’ll show in this tutorial how to create a very simple demo that shows how to load imtrages, handle sprites, collisions and keyboard events that you can use to create a game with a old school rpg like vision.
For the background scenario I’m using the house that I drew and we’ll call as house.png.
To put animation on the character we load all sprites into four lists. Each list for each direction.
// sprites
def up =for(i in [0..2]){Image{url:"{__DIR__}up{i}.png"}}
def right =for(i in [0..2]){Image{url:"{__DIR__}right{i}.png"}}
def down =for(i in [0..2]){Image{url:"{__DIR__}down{i}.png"}}
def left =for(i in [0..2]){Image{url:"{__DIR__}left{i}.png"}}
// sprites
def up = for(i in [0..2]) { Image {url: "{__DIR__}up{i}.png" } }
def right = for(i in [0..2]) { Image {url: "{__DIR__}right{i}.png" } }
def down = for(i in [0..2]) { Image {url: "{__DIR__}down{i}.png" } }
def left = for(i in [0..2]) { Image {url: "{__DIR__}left{i}.png" } }
And create vars to store the character position and frame of animation.
var frame =0;
var posx =320;
var posy =80;
var frame = 0;
var posx = 320;
var posy = 80;
Also store the house background.
// house background
def house = ImageView{ image: Image {url: "{__DIR__}house.png"} };
I create booleans to store some key states and at each interval of time I see how they are and do something about. You can handle keyboard event with less code but I like this way because keep visual and game logics a little bit more separated.
// keyboard
var upkey =false;
var rightkey =false;
var downkey =false;
var leftkey =false;// player
var player = ImageView{
x: bind posx y: bind posy
image:Image{url:"{__DIR__}down1.png"}
onKeyPressed: function(e:KeyEvent){if(e.code== KeyCode.VK_DOWN){
downkey =true;}elseif(e.code== KeyCode.VK_UP){
upkey =true;}elseif(e.code== KeyCode.VK_LEFT){
leftkey =true;}elseif(e.code== KeyCode.VK_RIGHT){
rightkey =true;}}// onKeyPressed
onKeyReleased: function(e:KeyEvent){if(e.code== KeyCode.VK_DOWN){
downkey =false;}elseif(e.code== KeyCode.VK_UP){
upkey =false;}elseif(e.code== KeyCode.VK_LEFT){
leftkey =false;}elseif(e.code== KeyCode.VK_RIGHT){
rightkey =false;}}// onKeyReleased}
// keyboard
var upkey = false;
var rightkey = false;
var downkey = false;
var leftkey = false;
// player
var player = ImageView{
x: bind posx y: bind posy
image: Image {url: "{__DIR__}down1.png"}
onKeyPressed: function(e:KeyEvent){
if (e.code == KeyCode.VK_DOWN) {
downkey = true;
} else if (e.code == KeyCode.VK_UP) {
upkey = true;
}else if (e.code == KeyCode.VK_LEFT) {
leftkey = true;
}else if (e.code == KeyCode.VK_RIGHT) {
rightkey = true;
}
} // onKeyPressed
onKeyReleased: function(e: KeyEvent){
if (e.code == KeyCode.VK_DOWN) {
downkey = false;
} else if (e.code == KeyCode.VK_UP) {
upkey = false;
}else if (e.code == KeyCode.VK_LEFT) {
leftkey = false;
}else if (e.code == KeyCode.VK_RIGHT) {
rightkey = false;
}
} // onKeyReleased
}
See a video of the game working so far:
Now we will add collisions. In a previous post I showed some math behind bounding box game collisions. The good news are that you no longer need to worry about that. There are a lot of API improvements in JavaFX 1.0 that do all the hard work for you, specially the new classes on javafx.geometry package, Rectangle2D and Point2D.
We create rectangles that represent the obstacles in the house.
We just have to change a little bit the game logics in order to handle collisions.
We define a bounding box around the player, it’s a rectangle from (4, 25) at the player coordinates system and with width 19 and height 10. The idea is to prospect where the player will be in the next step, see if it’s bouding box don’t collide with any obstacle and so pass it to the real game position.
// game logics
var gamelogics = Timeline {
repeatCount: Timeline.INDEFINITE
keyFrames: KeyFrame {
time : 1s/8
action: function(){
var nextposx = posx;
var nextposy = posy;if(downkey){
nextposy +=5;
player.image= down[++frame mod 3];}if(upkey){
nextposy -=5;
player.image= up[++frame mod 3];}if(rightkey){
nextposx +=5;
player.image= right[++frame mod 3];}if(leftkey){
nextposx -=5;
player.image= left[++frame mod 3];}for(obst in obstacles){if(obst.boundsInLocal.intersects(nextposx +4, nextposy +25, 19, 10)){return;}}
posx = nextposx;
posy = nextposy;}}}
// game logics
var gamelogics = Timeline {
repeatCount: Timeline.INDEFINITE
keyFrames: KeyFrame {
time : 1s/8
action: function() {
var nextposx = posx;
var nextposy = posy;
if(downkey) {
nextposy += 5;
player.image = down[++frame mod 3];
}
if(upkey) {
nextposy -= 5;
player.image = up[++frame mod 3];
}
if(rightkey) {
nextposx += 5;
player.image = right[++frame mod 3];
}
if(leftkey) {
nextposx -= 5;
player.image = left[++frame mod 3];
}
for(obst in obstacles) {
if(obst.boundsInLocal.intersects(nextposx + 4, nextposy + 25, 19, 10)) {
return;
}
}
posx = nextposx;
posy = nextposy;
}
}
}
This is enough to do the trick but I also added a way to smoothly show the obstacles when pressing the space key.
Here is the complete source code.
packageGame;importjavafx.stage.Stage;importjavafx.scene.*;importjavafx.scene.image.*;importjavafx.scene.input.*;importjavafx.scene.paint.*;importjavafx.scene.shape.*;importjavafx.animation.*;
var frame =0;
var posx =320;
var posy =80;// sprites
def up =for(i in [0..2]){Image{url:"{__DIR__}up{i}.png"}}
def right =for(i in [0..2]){Image{url:"{__DIR__}right{i}.png"}}
def down =for(i in [0..2]){Image{url:"{__DIR__}down{i}.png"}}
def left =for(i in [0..2]){Image{url:"{__DIR__}left{i}.png"}}// house background
def house = ImageView{ image:Image{url:"{__DIR__}house.png"}};// keyboard
var upkey =false;
var rightkey =false;
var downkey =false;
var leftkey =false;// player
var player = ImageView{
x: bind posx y: bind posy image: down[1]
onKeyPressed: function(e:KeyEvent){if(e.code== KeyCode.VK_DOWN){
downkey =true;}elseif(e.code== KeyCode.VK_UP){
upkey =true;}elseif(e.code== KeyCode.VK_LEFT){
leftkey =true;}elseif(e.code== KeyCode.VK_RIGHT){
rightkey =true;}if(e.code== KeyCode.VK_SPACE){if(fade==0.0){
fadein.playFromStart();}if(fade==1.0){
fadeout.playFromStart();}}}// onKeyPressed
onKeyReleased: function(e:KeyEvent){if(e.code== KeyCode.VK_DOWN){
downkey =false;}elseif(e.code== KeyCode.VK_UP){
upkey =false;}elseif(e.code== KeyCode.VK_LEFT){
leftkey =false;}elseif(e.code== KeyCode.VK_RIGHT){
rightkey =false;}}// onKeyReleased}// collidable obstacles
def obstacles =[Rectangle{ x:0 y:0 width:32 height:382 stroke:Color.RED},
Rectangle{ x:0 y:0 width:414 height:64 stroke:Color.RED},
Rectangle{ x:384 y:0 width:32 height:382 stroke:Color.RED},
Rectangle{ x:0 y:192 width:128 height:64 stroke:Color.RED},
Rectangle{ x:192 y:192 width:64 height:64 stroke:Color.RED},
Rectangle{ x:224 y:0 width:32 height:288 stroke:Color.RED},
Rectangle{ x:288 y:128 width:96 height:64 stroke:Color.RED},
Rectangle{ x:0 y:352 width:128 height:32 stroke:Color.RED},
Rectangle{ x:192 y:352 width:192 height:32 stroke:Color.RED},
Rectangle{ x:224 y:320 width:32 height:32 stroke:Color.RED},
Rectangle{ x:32 y:64 width:32 height:32 stroke:Color.YELLOW},
Rectangle{ x:64 y:64 width:32 height:32 stroke:Color.YELLOW},
Rectangle{ x:96 y:64 width:32 height:32 stroke:Color.YELLOW},
Rectangle{ x:128 y:64 width:64 height:32 stroke:Color.YELLOW},
Rectangle{ x:192 y:32 width:32 height:32 stroke:Color.YELLOW},
Rectangle{ x:64 y:128 width:64 height:32 stroke:Color.YELLOW},
Rectangle{ x:32 y:250 width:32 height:32 stroke:Color.YELLOW},
Rectangle{ x:64 y:250 width:64 height:32 stroke:Color.YELLOW},
Rectangle{ x:200 y:255 width:20 height:20 stroke:Color.YELLOW},
Rectangle{ x:200 y:170 width:20 height:20 stroke:Color.YELLOW},
Rectangle{ x:257 y:32 width:32 height:32 stroke:Color.YELLOW},
Rectangle{ x:288 y:32 width:32 height:32 stroke:Color.YELLOW},
Rectangle{ x:320 y:192 width:64 height:64 stroke:Color.YELLOW},
Rectangle{ x:352 y:295 width:32 height:60 stroke:Color.YELLOW},
Rectangle{ x:32 y:327 width:64 height:23 stroke:Color.YELLOW},
];// game logics
var gamelogics = Timeline {
repeatCount: Timeline.INDEFINITE
keyFrames: KeyFrame {
time : 1s/8
action: function(){
var nextposx = posx;
var nextposy = posy;if(downkey){
nextposy +=5;
player.image= down[++frame mod 3];}if(upkey){
nextposy -=5;
player.image= up[++frame mod 3];}if(rightkey){
nextposx +=5;
player.image= right[++frame mod 3];}if(leftkey){
nextposx -=5;
player.image= left[++frame mod 3];}for(obst in obstacles){if(obst.boundsInLocal.intersects(nextposx +4, nextposy +25, 19, 10)){return;}}
posx = nextposx;
posy = nextposy;}}}
gamelogics.play();// obstacles view
var fade =0.0;
var obstacleslayer =Group{
opacity: bind fade
content:[Rectangle{ x:0 y:0 width:500 height:500 fill:Color.BLACK},
obstacles,
Rectangle{
x: bind posx +4 y: bind posy +25 width:19 height:10
fill:Color.LIME}]}
var fadein = Timeline {
keyFrames:[
at (0s){fade =>0.0}
at (1s){fade =>1.0}]}
var fadeout = Timeline {
keyFrames:[
at (0s){fade =>1.0}
at (1s){fade =>0.0}]}// game stage
Stage {
title:"RPG-like demo", width:424, height:412
visible:true
scene: Scene{
fill:Color.BLACK
content:[house, player, obstacleslayer]}}
update: The applet version and Java Web Start versions should be working now. The applet version on Linux seems to be having problems with the keyboard handling, use the Java Web Start version while I’m trying to fix it.