Skip to content

Commit 18516e2

Browse files
committed
new sample
1 parent 1cf374b commit 18516e2

File tree

4 files changed

+212
-0
lines changed

4 files changed

+212
-0
lines changed

Samples/GameOfLife/GameOfLife.unoproj

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
{
2+
"InternalsVisibleTo": [],
3+
"Packages": [
4+
"Experimental.Http",
5+
"Fuse.Android",
6+
"Fuse.Android",
7+
"Fuse.Animations",
8+
"Fuse.BasicTheme",
9+
"Fuse.Controls",
10+
"FuseCore",
11+
"Fuse.Designer",
12+
"Fuse.Desktop",
13+
"Fuse.Drawing",
14+
"Fuse.Drawing.Planar",
15+
"Fuse.Drawing.Primitives",
16+
"Fuse.Effects",
17+
"Fuse.Elements",
18+
"Fuse.Entities",
19+
"Fuse.Gestures",
20+
"Fuse.iOS",
21+
"Fuse.Navigation",
22+
"Fuse.Reactive",
23+
"Fuse.Shapes",
24+
"Fuse.Triggers",
25+
],
26+
"Projects": [],
27+
"Includes": [
28+
"*"
29+
]
30+
}

Samples/GameOfLife/GameOfLife.ux

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
<App Theme="Basic" ux:Class="GameOfLife" ClearColor="1,1,1,1">
2+
<ClientPanel>
3+
<JavaScript File="Life.js"/>
4+
5+
<DockPanel Dock="Top">
6+
<StackPanel Orientation="Horizontal" Dock="Left">
7+
<Button Text="Step" Clicked="{step}"/>
8+
<Text Value="Auto-Step:" Alignment="VerticalCenter"/>
9+
<Switch Value="{autoStep}" Alignment="VerticalCenter" ValueChanged="{autoStepChanged}"/>
10+
<Button Text="Clear" Clicked="{clear}" Alignment="VerticalCenter"/>
11+
<Text Value="Speed" Alignment="VerticalCenter"/>
12+
</StackPanel>
13+
<Slider Value="{speed}" Minimum="100" Maximum="1000"/>
14+
</DockPanel>
15+
<Rectangle Height="1" Color="0,0,0,1" Dock="Top" Margin="0,2,0,0"/>
16+
17+
<Grid Clicked="{updateColorCells}" ColumnCount="{numColumns}" RowCount="{numRows}"
18+
CellSpacing="3" Color="0.6,0.6,0.7,1">
19+
<Each Items="{cells}">
20+
<Panel ux:Name="self" Color="1,1,1,1">
21+
<Rectangle ux:Name="cell" Color="0,0,0,1" CornerRadius="50"
22+
Width="0%" Height="0%"/>
23+
<WhileTrue Value="{on}">
24+
<Change cell.CornerRadius="0" Duration="0.2"/>
25+
<Change cell.Width="100" Duration="0.2"/>
26+
<Change cell.Height="100" Duration="0.2"/>
27+
</WhileTrue>
28+
<Tapped Handler="{toggleCell}"/>
29+
</Panel>
30+
</Each>
31+
</Grid>
32+
33+
</ClientPanel>
34+
</App>

Samples/GameOfLife/Life.js

+143
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,143 @@
1+
var Observable = require("FuseJS/Observable")
2+
3+
var cells = Observable()
4+
exports.cells = cells
5+
6+
var numColumns = 30
7+
var numRows = 30
8+
exports.numColumns = Observable(numColumns)
9+
exports.numRows = Observable(numRows)
10+
exports.autoStep = Observable(true)
11+
12+
function Cell(on) {
13+
this.on = Observable(on)
14+
this.count = 0
15+
}
16+
17+
for (var i=0; i < numRows; ++i) {
18+
for (var j=0; j < numColumns; ++j) {
19+
cells.add( new Cell( Math.random() > 0.8 ? true : false ) )
20+
}
21+
}
22+
23+
/**
24+
Get the cell at these coordinates (with wrap-around).
25+
*/
26+
function getCell(row,col) {
27+
if (row < 0) {
28+
row += numRows
29+
}
30+
row = row % numRows
31+
if (col < 0) {
32+
col += numColumns
33+
}
34+
col = col % numColumns
35+
36+
return cells.getAt(row * numColumns + col)
37+
}
38+
39+
/**
40+
How many neighbours of this cell are "on"
41+
*/
42+
function liveCount(row,col) {
43+
var c = 0
44+
for (var i=-1; i <= 1; ++i) {
45+
for (var j=-1; j <= 1; ++j) {
46+
if (i==0 && j==0) {
47+
continue;
48+
}
49+
50+
if (getCell(i+row,j+col).on.value)
51+
c++
52+
}
53+
}
54+
55+
return c
56+
}
57+
58+
function autoStep() {
59+
step()
60+
//check speed change now instead of with valueChagned to avoid
61+
//odd time stretching while the user is interacting
62+
updateTimer(exports.autoStep.value, exports.speed.value)
63+
}
64+
65+
function step() {
66+
67+
//count neighbours prior to changing anything
68+
for (var i=0; i < numRows; ++i) {
69+
for (var j=0; j < numColumns; ++j) {
70+
var c = getCell(i,j)
71+
c.count = liveCount(i,j)
72+
}
73+
}
74+
75+
for (var i=0; i < numRows; ++i) {
76+
for (var j=0; j < numColumns; ++j) {
77+
var c = getCell(i,j)
78+
if (!c.on.value) {
79+
c.on.value = c.count == 3
80+
} else if (c.count <2) {
81+
c.on.value = false
82+
} else if (c.count > 3) {
83+
c.on.value = false
84+
}
85+
}
86+
}
87+
}
88+
89+
/**
90+
This callback is done in the context of a cell item in the UI, thus the `data` property is the Cell object.
91+
*/
92+
exports.toggleCell = function(args) {
93+
args.data.on.value = !args.data.on.value
94+
}
95+
96+
/**
97+
Turn off all cells.
98+
*/
99+
exports.clear = function(args) {
100+
for (var i=0; i < cells.length; ++i) {
101+
cells.getAt(i).on.value = false
102+
}
103+
}
104+
105+
exports.step = step
106+
107+
108+
//timer control
109+
var intervalSpeed = 0 //0 indicates no interval setup
110+
var interval = null
111+
exports.speed = Observable(500)
112+
updateTimer(exports.autoStep.value,exports.speed.value)
113+
114+
exports.autoStepChanged = function(args) {
115+
//we must use the `args.value` since `exports.autoStep.value` may not yet be updated
116+
updateTimer(args.value, exports.speed.value)
117+
//it feels more responsive to do something immediately when turning on
118+
if (args.value) {
119+
step()
120+
}
121+
}
122+
123+
/**
124+
Match the actual interval timer to the desired timer.
125+
*/
126+
function updateTimer(on, speed) {
127+
if (!on) {
128+
speed = 0
129+
}
130+
131+
if (speed == intervalSpeed) {
132+
return
133+
}
134+
135+
if (interval != null) {
136+
clearInterval(interval)
137+
}
138+
139+
intervalSpeed = speed
140+
if (speed > 0) {
141+
interval = setInterval( autoStep, intervalSpeed )
142+
}
143+
}

Samples/GameOfLife/README.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
# Conway's Game of Life
2+
3+
This project uses an array of JavaScript Observables to produce a grid in the interface. It's a simulation of the classic game of life.
4+
5+
The UI interacts with the JavaScript in several different ways. The grid is a two-way interface: it is updated by the JavaScript, and cells can be toggled on/off by a user click. The speed slider controls the stepping time used by `setInterval`. The `Switch` allows turning on/off the auto-stepping, with a button for manual stepping.

0 commit comments

Comments
 (0)