1
+ extern crate js_sys;
2
+ extern crate web_sys;
3
+
4
+ mod species;
1
5
mod utils;
2
6
7
+ use rand:: { Rng , SeedableRng } ;
8
+ use rand_xoshiro:: SplitMix64 ;
9
+ use species:: Species ;
3
10
use wasm_bindgen:: prelude:: * ;
4
11
5
- // When the `wee_alloc` feature is enabled, use `wee_alloc` as the global
6
- // allocator.
7
- #[ cfg( feature = "wee_alloc" ) ]
8
- #[ global_allocator]
9
- static ALLOC : wee_alloc:: WeeAlloc = wee_alloc:: WeeAlloc :: INIT ;
10
-
11
- // #[wasm_bindgen]
12
- // extern "C" {
13
- // fn alert(s: &str);
14
- // }
15
-
16
- // #[wasm_bindgen]
17
- // pub fn greet(name: &str) {
18
- // alert(&format!("Hello, {}!", name));
19
- // }
12
+ // A macro to provide `println!(..)`-style syntax for `console.log` logging.
13
+ macro_rules! log {
14
+ ( $( $t: tt ) * ) => {
15
+ web_sys:: console:: log_1( & format!( $( $t ) * ) . into( ) ) ;
16
+ }
17
+ }
20
18
21
19
#[ wasm_bindgen]
22
- #[ repr( u8 ) ]
20
+ #[ repr( C ) ]
23
21
#[ derive( Clone , Copy , Debug , PartialEq , Eq ) ]
24
- pub enum Cell {
25
- Dead = 0 ,
26
- Alive = 1 ,
22
+ pub struct Cell {
23
+ species : Species ,
24
+ ra : u8 ,
25
+ rb : u8 ,
26
+ clock : u8 ,
27
27
}
28
28
29
29
#[ wasm_bindgen]
30
30
pub struct Universe {
31
- width : u32 ,
32
- height : u32 ,
31
+ width : i32 ,
32
+ height : i32 ,
33
33
cells : Vec < Cell > ,
34
+ rng : SplitMix64 ,
35
+ generation : u8 ,
36
+ }
37
+
38
+ // impl Cell {
39
+ // fn toggle(&mut self) {
40
+ // *self = match *self {
41
+ // Cell::Dead => Cell::Alive,
42
+ // Cell::Alive => Cell::Dead,
43
+ // }
44
+ // }
45
+ // }
46
+
47
+ impl Cell {
48
+ pub unsafe fn new ( species : Species ) -> Cell {
49
+ Cell {
50
+ species : species,
51
+ ra : 1 ,
52
+ // ra: 100 + (js_sys::Math::random() * 50.) as u8,
53
+ rb : 0 ,
54
+ clock : 0 ,
55
+ }
56
+ }
57
+ // pub fn update(&self, api: SandApi) {
58
+ // self.species.update(*self, api);
59
+ // }
34
60
}
35
61
62
+ static EMPTY_CELL : Cell = Cell {
63
+ species : Species :: Empty ,
64
+ ra : 0 ,
65
+ rb : 0 ,
66
+ clock : 0 ,
67
+ } ;
68
+
36
69
#[ wasm_bindgen]
37
70
impl Universe {
38
- fn get_index ( & self , row : u32 , column : u32 ) -> usize {
71
+ fn get_index ( & self , row : i32 , column : i32 ) -> usize {
39
72
( row * self . width + column) as usize
40
73
}
41
74
42
- fn live_neighbor_count ( & self , row : u32 , column : u32 ) -> u8 {
43
- let mut count = 0 ;
44
- for delta_row in [ self . height - 1 , 0 , 1 ] . iter ( ) . cloned ( ) {
45
- for delta_col in [ self . width - 1 , 0 , 1 ] . iter ( ) . cloned ( ) {
46
- if delta_row == 0 && delta_col == 0 {
47
- continue ;
48
- }
49
-
50
- let neighbor_row = ( row + delta_row) % self . height ;
51
- let neighbor_col = ( column + delta_col) % self . width ;
52
- let idx = self . get_index ( neighbor_row, neighbor_col) ;
53
- count += self . cells [ idx] as u8 ;
54
- }
55
- }
56
- count
75
+ fn get_cell ( & self , x : i32 , y : i32 ) -> Cell {
76
+ let i = self . get_index ( x, y) ;
77
+ return self . cells [ i] ;
57
78
}
58
79
59
80
pub fn tick ( & mut self ) {
@@ -62,70 +83,87 @@ impl Universe {
62
83
for row in 0 ..self . height {
63
84
for col in 0 ..self . width {
64
85
let idx = self . get_index ( row, col) ;
65
- let cell = self . cells [ idx] ;
66
- let live_neighbors = self . live_neighbor_count ( row, col) ;
67
-
68
- let next_cell = match ( cell, live_neighbors) {
69
- // Rule 1: Any live cell with fewer than two live neighbours
70
- // dies, as if caused by underpopulation.
71
- ( Cell :: Alive , x) if x < 2 => Cell :: Dead ,
72
- // Rule 2: Any live cell with two or three live neighbours
73
- // lives on to the next generation.
74
- ( Cell :: Alive , 2 ) | ( Cell :: Alive , 3 ) => Cell :: Alive ,
75
- // Rule 3: Any live cell with more than three live
76
- // neighbours dies, as if by overpopulation.
77
- ( Cell :: Alive , x) if x > 3 => Cell :: Dead ,
78
- // Rule 4: Any dead cell with exactly three live neighbours
79
- // becomes a live cell, as if by reproduction.
80
- ( Cell :: Dead , 3 ) => Cell :: Alive ,
81
- // All other cells remain in the same state.
82
- ( otherwise, _) => otherwise,
83
- } ;
84
-
85
- next[ idx] = next_cell;
86
+ let cell = & self . cells [ idx] ;
86
87
}
87
88
}
88
89
89
90
self . cells = next;
90
91
}
91
92
92
93
pub fn new ( ) -> Universe {
93
- let width = 64 ;
94
- let height = 64 ;
95
-
96
- let cells = ( 0 ..width * height)
97
- . map ( |i| {
98
- if i % 2 == 0 || i % 7 == 0 {
99
- Cell :: Alive
100
- } else {
101
- Cell :: Dead
102
- }
103
- } )
104
- . collect ( ) ;
94
+ let width = 150 ;
95
+ let height = 150 ;
96
+ let cells = ( 0 ..width * height) . map ( |_i| EMPTY_CELL ) . collect ( ) ;
97
+ let rng: SplitMix64 = SeedableRng :: seed_from_u64 ( 0x734f6b89de5f83cc ) ;
105
98
Universe {
106
99
width,
107
100
height,
108
101
cells,
102
+ rng,
103
+ generation : 0 ,
109
104
}
110
105
}
111
106
112
- pub fn render ( & self ) -> String {
113
- self . to_string ( )
107
+ pub fn width ( & self ) -> i32 {
108
+ self . width
114
109
}
115
- }
116
110
117
- use std:: fmt;
111
+ pub fn height ( & self ) -> i32 {
112
+ self . height
113
+ }
118
114
119
- impl fmt:: Display for Universe {
120
- fn fmt ( & self , f : & mut fmt:: Formatter ) -> fmt:: Result {
121
- for line in self . cells . as_slice ( ) . chunks ( self . width as usize ) {
122
- for & cell in line {
123
- let symbol = if cell == Cell :: Dead { '◻' } else { '◼' } ;
124
- write ! ( f, "{}" , symbol) ?;
125
- }
126
- write ! ( f, "\n " ) ?;
115
+ pub fn cells ( & self ) -> * const Cell {
116
+ self . cells . as_ptr ( )
117
+ }
118
+
119
+ pub fn paint ( & mut self , x : i32 , y : i32 , size : i32 , species : Species ) {
120
+ let size = size;
121
+ let radius: f64 = ( size as f64 ) / 2.0 ;
122
+ // let radius: f64 = (size as f64);
123
+
124
+ let floor = ( radius + 1.0 ) as i32 ;
125
+ let ciel = ( radius + 1.5 ) as i32 ;
126
+
127
+ unsafe {
128
+ log ! ( "x {}: " , x) ;
129
+ log ! ( "y {}: " , y) ;
130
+ log ! ( "size {}: " , size) ;
131
+ log ! ( "floor {}: " , floor) ;
132
+ log ! ( "ciel {}: " , ciel) ;
133
+ log ! ( "species {:?}: " , species) ;
134
+ log ! ( "self.cells {:?}: " , self . cells) ;
127
135
}
128
136
129
- Ok ( ( ) )
137
+ for dx in -floor..ciel {
138
+ for dy in -floor..ciel {
139
+ if ( ( ( dx * dx) + ( dy * dy) ) as f64 ) > ( radius * radius) {
140
+ continue ;
141
+ } ;
142
+ let px = x + dx;
143
+ let py = y + dy;
144
+ let i = self . get_index ( px, py) ;
145
+
146
+ if px < 0 || px > self . width - 1 || py < 0 || py > self . height - 1 {
147
+ continue ;
148
+ }
149
+
150
+ // let test = self.get_cell(px, py).species;
151
+ // unsafe {
152
+ // log!("test {:?}", test);
153
+ // log!("type {:?}", Species::Empty);
154
+ // }
155
+ if self . get_cell ( px, py) . species == Species :: Empty || species == Species :: Empty {
156
+ self . cells [ i] = Cell {
157
+ species : species,
158
+ ra : 60
159
+ + ( size as u8 )
160
+ + ( self . rng . gen :: < f32 > ( ) * 30. ) as u8
161
+ + ( ( self . generation % 127 ) as i8 - 60 ) . abs ( ) as u8 ,
162
+ rb : 0 ,
163
+ clock : self . generation ,
164
+ } ;
165
+ }
166
+ }
167
+ }
130
168
}
131
169
}
0 commit comments