-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtic-tac-toe.tal
269 lines (210 loc) · 4.63 KB
/
tic-tac-toe.tal
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
(
@|constants )
|0e @System/debug |0f @System/state |18 @Console/write
|80 @Controller/vector |83 @Controller/key
|09 @t |0a @n |20 @s
( 9 constant #squares )
|09 @squares
( 48 constant zero )
|31 @zero
( 113 constant q-char )
|71 @q-char
( 1 constant X 2 constant O )
|01 @X |02 @O
|0000
(
@|variables )
( variable unplayed )
@unplayed $1 @user-input $1
|0100
(
@|main )
@on-reset ( -> )
;on-controller .Controller/vector DEO2
full-game
print-stacks
BRK
@on-controller
.Controller/key DEI .user-input STZ
BRK
( : square! ( square # --- ) board rot 1- cells ! ; )
@set-square ( i val -- ) SWP #00 SWP ;board ADD2 STA JMP2r
( : square@ ( square --- # ) board swap 1- cells @ ; )
@get-square ( i -- val ) #00 SWP ;board ADD2 LDA JMP2r
( : 3-cr ( n --- ) 3 mod 0= if cr then ; )
@3-cr ( i -- ) #03 AND #00 NEQ ?{ cr } JMP2r
( : dashes ( --- ) cr tab ." ---------" cr ; )
@dashes ( -- ) cr tab { "--------- $1 } print cr JMP2r
(
: .square ( n --- )
dup square@
dup 0 = if
swap .
then dup 1 = if
." X " drop
then dup 2 = if
." O " drop
then
drop
)
@show-square ( i -- )
DUP get-square
DUP #00 EQU not ?{ SWP print-nibble spc !&end }
DUP #01 EQU not ?{ { "X -s $1 } print POP !&end }
DUP #02 EQU not ?{ { "O -s $1 } print POP !&end }
&end
POP
JMP2r
(
: 3numbers ( a b c --- )
tab .square ." | "
.square ." | "
.square ;
)
@3numbers ( a b c -- )
tab show-square { "| -s $1 } print
show-square { "| -s $1 } print
show-square
JMP2r
(
: .board ( --- )
9 8 7 6 5 4 3 2 1
3numbers dashes
3numbers dashes
3numbers cr ;
)
@show-board ( -- )
#0908 #0706 #0504 #0302 #01
3numbers dashes
3numbers dashes
3numbers cr
JMP2r
(
: clear-game ( --- )
#squares 1+ 1
do
i 0 square!
loop ;
)
@clear-game ( -- )
.squares INC #01
&for
DUP #00 set-square
INC GTHk ?&for
POP2
JMP2r
( : X! ( i --- ) X square! ; )
@set-X ( i -- ) .X set-square JMP2r
( : O! ( i --- ) O square! ; )
@set-O ( i -- ) .O set-square JMP2r
( : start ( --- ) clear-game #squares unplayed ! ; )
@start ( -- ) clear-game .squares .unplayed STZ JMP2r
( : current-player ( --- f ) unplayed @ 1 and ; )
@current-player ( -- f ) .unplayed LDZ #01 AND JMP2r
( : ascii># ( char --- n ) zero - ; )
@ascii># ( char -- n ) .zero SUB JMP2r
( : range? ( n --- ) dup 1 < swap 9 > or 0= ; )
@range? ( n -- ) DUP #00 LTH SWP #0a GTH ORA #00 EQU JMP2r
( : empty? ( n --- ) square@ 0= )
@empty? ( n -- ) get-square #00 EQU JMP2r
(
: place-symbol ( square --- )
current-player if
X!
else
O!
then
unplayed @ 1+ unplayed ! ;
)
@place-symbol ( square -- )
current-player not ?{ set-X !&end }
set-O
&end
.unplayed LDZ INC .unplayed STZ
JMP2r
( : ps ( --- ) place-symbol ; )
@ps ( -- ) place-symbol JMP2r
(
: player-input ( --- )
begin
cr ." Square number for "
current-player
if
." X: "
else
." O: "
then
key dup q-char =
if
drop cr ." Exiting " bye
then
ascii># dup range? over empty? and
if
place-symbol .board
else
drop ." Pick another square. "
then
while
true
repeat ;
)
@player-input ( -- )
&while
cr { "Square -s "number -s "for -s $1 } print
current-player not ?{ { "X: -s $1 } print !&continue }
{ "O: -s $1 } print
&continue
key DUP .q-char EQU not ?{ POP cr { "Exiting $1 } print exit }
ascii># DUP range? OVR empty? AND not ?{ place-symbol show-board !&end }
POP { "Pick -s "another -s "square. $1 } print
&end
( !&while )
JSR2
( : next ( --- ) player-input ; )
@next ( -- ) player-input JMP2r
(
: full-game ( --- )
start cr ." Enter 'q' to exit. "
begin
.game player-input
if exit then
unplayed
0= until ;
)
@full-game ( -- )
start { "Enter -s ""q" -s "to -s "exit. $1 } print cr cr
show-board player-input
JMP2r
(
@|stdlib )
@print ( {str}* -- )
SWP2r STH2r
&while
LDAk emit
INC2 LDAk ?&while
POP2
JMP2r
@print-nibble ( u -- )
#0f AND DUP #09 GTH #27 MUL ADD
[ LIT "0 ] ADD emit
JMP2r
@emit ( char -- ) .Console/write DEO JMP2r
@cr ( -- ) .n emit JMP2r
@tab ( -- ) .t emit JMP2r
@spc ( -- ) .s emit JMP2r
@not ( f -- ) #00 EQU JMP2r
@key ( -- char ) .user-input LDZ JMP2r
@exit ( -> ) [ LIT2 80 -System/state ] DEO BRK
@print-stacks ( -- )
STH2r ,&rtn STR2
cr [ LIT2 01 -System/debug ] DEO
[ LIT2 &rtn $2 ] STH2
JMP2r
(
@|data )
( create board squares cells allot )
@board [
00 00 00
00 00 00
00 00 00
]