This is a text-based Star Trek strategy game in which the player commands a starship on a 10×10 grid, scanning for, moving toward, and firing upon alien vessels (Klingons, Romulans, or Kzintzis, chosen randomly at startup). The game uses two parallel 14×14 arrays — A() for logical state and B() for display state — to track grid contents, with cell values 0 (empty), 1 (alien), 2 (player), 3 (miss), and 4 (kill). Movement and firing use a numpad-style 8-direction compass layout, and sector names are derived by multiplying the player’s row and column coordinates together and mapping the product to named star sectors. The random crew commentary subroutine at line 7500 selects from five Star Trek characters to prefix messages, adding flavour to game events. Energy decreases with every action — scanning costs 10 or 20 energy, movement costs 50, and firing costs 100 — and exhaustion triggers the end-game scoring sequence.
Program Analysis
Program Structure
The program is organised as a main loop with a dispatch mechanism, surrounded by initialisation and utility subroutines. Execution flows as follows:
- Line 9000 — initialise arrays, position, alien type, and energy
- Line 8000 — print the grid display
- Line 6950 — clear the lower screen area
- Line 7000 — print status panel
- Line 7500 — print random crew prefix
- Lines 100–160 — main menu and input loop
- Line 160 — dispatch:
GOSUB 1000*D(D=1 → scan, D=2 → move, D=3 → fire) - Lines 170–190 — brief flashing animation loop
- Line 500 — loop back to status display
Dispatch Technique
Line 160 uses the idiom GOSUB 1000*D, where D is the menu choice (1, 2, or 3). This evaluates to GOSUB 1000, GOSUB 2000, or GOSUB 3000 respectively, avoiding a chain of IF … THEN GOSUB statements. This is a classic Sinclair BASIC memory-saving technique.
Array Usage
Two 14×14 arrays are declared at line 9000, intentionally larger than the 10×10 playfield. This provides a one-cell border on each side, so boundary-adjacent lookups (e.g., A(B-1,C)) do not cause an index-out-of-range error when the player is at row or column 1 or 10. Cell values encode object type:
| Value | Meaning | Display char |
|---|---|---|
| 0 | Empty | Inverse space (% ) |
| 2 | Player ship | Inverse $ (%$) |
| 3 | Missed shot | Inverse X (%X) |
| 4 | Destroyed alien | Inverse * (%*) |
| 1 | Alien vessel | (not explicitly printed — remains empty display) |
Note that alien vessels (value 1) have no distinct display character in the B() display array; they appear as empty space. This is presumably intentional — the player must use the scanner to detect them rather than seeing them on the map.
Dual Array Design
Array A() holds logical game state while B() holds the display state. Both are updated in parallel at key events (movement at lines 2040–2050/2270–2280, firing at lines 3260–3261). However, the alien initialisation loop (lines 9060–9140) only populates A(), not B(), so alien positions are invisible on the grid from the start — consistent with the design intent above.
Directional Input
Both movement (line 2000 block) and firing (line 3000 block) use a numpad-style compass where digits 1–8 represent directions. The layout is displayed as a 3×3 grid in PRINT strings:
- 8 = north-west, 1 = north, 2 = north-east
- 7 = west, (centre = player), 3 = east
- 6 = south-west, 5 = south, 4 = south-east
Each diagonal direction is implemented as two consecutive IF statements modifying both B (row) and C (column), rather than a single compound update. This is straightforward but verbose.
Sector Naming
The subroutine at line 8500 derives a sector name by computing Q = B * C (row × column) and mapping the product to one of ten named sectors (Antares, Rigel, Procyon, Vega, Canopus, Altair, Deneb, Pollux, Sirius, Capella). Since B and C each range 1–10, Q ranges 1–100, and sectors cover bands of 10. However, this means the naming is heavily non-uniform: the player starting at (5,5) is always in “Capella” (Q=25 → Procyon), and identical products from different coordinates map to the same sector.
Energy System
Initial energy is set at line 9340: E = 1000 + 2000*RND, giving a range of 1000–3000. Energy costs:
- Close scanner: −10
- Long-range scanner: −20
- Movement: −50
- Firing: −100
- Enemy return fire hit: −100*RND (random, up to 100)
When energy drops below 1, line 7030 redirects to line 3800 where the end-game scoring loop runs. The loop at 3805–3890 uses SCROLL to push results up the screen, with an internal counter (CNT) that stops after two passes by jumping to line 5900 (STOP).
Bugs and Anomalies
- Variable name collision: The loop variable at line 9060 uses
FOR A=1 TO 20, shadowing the arrayA(). In Sinclair BASIC, scalarAand arrayA()are distinct, so this does not cause an error, but it is potentially confusing. - Line 1126 unreachable branch: Line 1126 tests for nearby aliens and, if true,
GOTO 1127— which is the very next line regardless of the branch. The conditional jump is therefore a no-op, and the real proximity check and message print happens at line 1132 unconditionally. - Return fire never reaches line 3420: Line 3340 either goes to 3400 (not defined — falls through to 3420) or falls through to line 3360. If RND > 0.5,
GOTO 3400is taken; since line 3400 does not exist, execution continues at line 3420 (“THE [alien] MISSED”). If RND ≤ 0.5, lines 3360–3390 execute (hit). This is a valid use of the non-existent-line technique for a conditional skip. - Direction 5 (south) in movement: Line 2112 implements direction “5” as
B=B+1only, correctly moving south, but direction “4” (lines 2108–2110) moves both south and east — intended as south-east. Direction “3” (line 2106 only) moves east only. The diagonal for direction “5” as labelled in the display (654bottom row, centre = south) is correct. - Initial energy used for alien type selection: Lines 9260–9320 use the raw
RNDvalue inEto pick the alien race before overwriting E with the actual energy value at line 9340. This is intentional but subtle. - Animation loop at lines 170–190 prints and erases block graphic characters (
\ :\ :\ :\ :\ :— five “▌▐” pairs) at row 20, creating a brief flashing effect between turns.
FAST/SLOW Usage
The program switches to FAST mode during initialisation (line 15) and during grid redraw (line 8001), then returns to SLOW mode (line 90, line 8225) for interactive display. This minimises screen flicker during the computationally intensive grid print loop while keeping the display readable during normal play.
Crew Commentary Subroutine
Subroutine 7500 picks a random integer 0–4 and prints a crew member’s name as a message prefix. It is called before most player-facing messages throughout the game. The five crew members are Spock, Scott, Uhura, Chekov, and Sulu, matching the original series cast. Spock’s entry (R=0) emits a full sentence opener; the others print only the name followed by a colon and space.
Content
Source Code
1 REM "STAR TREK" BY TIM HARTNELL MODIFIED BY ANTHONY WILLING WITH IMPROVED HELM/DIRECTIONAL COMMANDS
7 REM
10 CLS
15 FAST
20 GOSUB 9000
25 GOSUB 8000
30 GOSUB 6950
40 GOSUB 7000
50 GOSUB 6950
80 GOSUB 7500
90 SLOW
100 PRINT "WHAT IS YOUR ORDER,"
120 PRINT TAB 5;"SIR?",,TAB 12;"1 - SCAN";TAB 12;"2 - MOVE";TAB 12;"3 - FIRE"
140 INPUT D
150 IF D<1 OR D>3 THEN GOTO 140
155 GOSUB 6950
160 GOSUB 1000*D
170 FOR W=1 TO 40
180 PRINT AT 20,10;" : : : : :";AT 20,10;" "
190 NEXT W
500 GOTO 30
1020 PRINT TAB 4;"% % SCANNER % % "
1030 GOSUB 7500
1040 PRINT "CLOSE (1) OR ","LONG-RANGE (2), SIR?"
1060 INPUT K
1080 LET E=E-10*K
1090 GOSUB 6950
1100 IF K=2 THEN GOTO 1500
1126 IF A(B+1,C)=1 OR A(B+1,C+1)=1 OR A(B,C+1)=1 OR A(B-1,C)=1 OR A(B-1,C-1)=1 OR A(B,C-1)=1 OR A(B+1,C-1)=1 OR A(B-1,C+1)=1 THEN GOTO 1127
1127 IF B<1 THEN LET B=1
1128 IF B>10 THEN LET B=10
1129 IF C<1 THEN LET C=1
1130 IF C>10 THEN LET C=10
1132 IF A(B+1,C)=1 OR A(B+1,C+1)=1 OR A(B,C+1)=1 OR A(B-1,C)=1 OR A(B-1,C-1)=1 OR A(B,C-1)=1 OR A(B+1,C-1)=1 OR A(B-1,C+1)=1 THEN PRINT Z$;"IN VICINITY,","SIR"
1140 RETURN
1500 GOSUB 7500
1520 PRINT AT 15,0;"DIRECTION?:( 1 ) (4 3) ( 2 )"
1525 PRINT AT 16,16;"(ENTER A NUMBER)"
1530 INPUT SCAN
1540 LET Z=0
1560 IF SCAN=1 AND A(B-2,C)=1 THEN GOTO 1621
1580 IF SCAN=2 AND A(B+2,C)=1 THEN GOTO 1621
1600 IF SCAN=3 AND A(B,C+2)=1 THEN GOTO 1621
1620 IF SCAN=4 AND A(B,C-2)=1 THEN GOTO 1621
1621 IF B<1 THEN LET B=1
1622 IF B>10 THEN LET B=10
1623 IF C<1 THEN LET C=1
1624 IF C>10 THEN LET C=10
1625 IF SCAN=1 AND A(B-2,C)=1 THEN LET Z=1
1626 IF SCAN=2 AND A(B+2,C)=1 THEN LET Z=1
1627 IF SCAN=3 AND A(B,C+2)=1 THEN LET Z=1
1628 IF SCAN=4 AND A(B,C-2)=1 THEN LET Z=1
1630 GOSUB 7500
1640 PRINT "LONG-RANGE SCANNER REPORT IS"
1660 IF Z=1 THEN PRINT "POSITIVE"
1680 IF Z=0 THEN PRINT "NEGATIVE"
1700 RETURN
2020 LET E=E-50
2040 LET A(B,C)=0
2050 LET B(B,C)=0
2060 PRINT "DIRECTION (812) (7 3) (654) ?"
2080 INPUT A$
2100 IF A$="1" THEN LET B=B-1
2102 IF A$="2" THEN LET B=B-1
2104 IF A$="2" THEN LET C=C+1
2106 IF A$="3" THEN LET C=C+1
2108 IF A$="4" THEN LET B=B+1
2110 IF A$="4" THEN LET C=C+1
2112 IF A$="5" THEN LET B=B+1
2114 IF A$="6" THEN LET B=B+1
2116 IF A$="6" THEN LET C=C-1
2118 IF A$="7" THEN LET C=C-1
2120 IF A$="8" THEN LET B=B-1
2122 IF A$="8" THEN LET C=C-1
2124 IF B<1 THEN LET B=1
2126 IF B>10 THEN LET B=10
2127 IF C<1 THEN LET C=1
2128 IF C>10 THEN LET C=10
2130 GOSUB 7500
2140 PRINT "NOW AT ";B;",";C
2260 IF A(B,C)=1 THEN GOTO 5500
2270 LET A(B,C)=2
2280 LET B(B,C)=2
2285 PAUSE 100
2290 GOSUB 8000
2300 RETURN
3000 REM FIRE
3010 GOSUB 7500
3020 PRINT "DIRECTION OF FIRE? (812) (7 3) (654) "
3040 INPUT A$
3042 IF A$="1" THEN LET G=B-1
3043 IF A$="1" THEN LET F=C
3044 IF A$="2" THEN LET G=B-1
3046 IF A$="2" THEN LET F=C+1
3048 IF A$="3" THEN LET F=C+1
3049 IF A$="3" THEN LET G=B
3050 IF A$="4" THEN LET G=B+1
3052 IF A$="4" THEN LET F=C+1
3054 IF A$="5" THEN LET G=B+1
3055 IF A$="5" THEN LET F=C
3056 IF A$="6" THEN LET G=B+1
3058 IF A$="6" THEN LET F=C-1
3060 IF A$="7" THEN LET F=C-1
3061 IF A$="7" THEN LET G=B
3062 IF A$="8" THEN LET G=B-1
3064 IF A$="8" THEN LET F=C-1
3066 IF F<1 THEN LET F=1
3068 IF F>10 THEN LET F=10
3070 IF G<1 THEN LET G=1
3072 IF G>10 THEN LET G=10
3180 LET E=E-100
3190 IF A(G,F)<>1 THEN GOTO 3300
3195 GOSUB 7500
3200 PRINT "YOU HIT THE ";Z$
3220 LET AL=AL+1
3260 LET B(G,F)=4
3261 LET A(G,F)=4
3290 RETURN
3300 GOSUB 7500
3305 LET B(G,F)=3
3310 PRINT "YOU MISSED, SIR"
3315 FOR G=1 TO 30
3317 NEXT G
3320 PRINT "THE ";Z$;"ARE","RETURNING FIRE"
3330 FOR G=1 TO 30
3335 NEXT G
3337 GOSUB 6950
3340 IF RND>.5 THEN GOTO 3400
3360 PRINT "THEY HIT US, SIR"
3365 PAUSE 100
3370 GOSUB 8000
3380 LET E=E-100*RND
3390 RETURN
3420 PRINT "THE ";Z$;"MISSED, SIR"
3460 RETURN
3800 GOSUB 6590
3804 LET CNT=0
3805 SCROLL
3806 LET CNT=CNT+1
3807 IF CNT=2 THEN GOTO 5900
3808 SCROLL
3810 PRINT "ENERGY BANKS EXHAUSTED"
3811 SCROLL
3820 PRINT "YOU KILLED ";AL;" ALIEN";
3830 IF AL<>1 THEN PRINT "S"
3850 SCROLL
3860 PRINT "ON THIS MISSION"
3870 SCROLL
3880 PRINT "YOUR COMMANDER RATING IS ";INT (AL/8*100)
3883 SCROLL
3885 PRINT
3890 GOTO 3805
3895 STOP
5000 REM END
5500 GOSUB 6950
5520 FOR V=1 TO 50
5525 PRINT AT 15,0;"THE ENTERPRISE HAS BEEN HIT BY A"
5530 PRINT AT 16,0;Q$;" VESSEL AND HAS BEEN DESTROYED"
5540 PRINT AT 15,0;"%T%H%E% %E%N%T%E%R%P%R%I%S%E% %H%A%S% %B%E%E%N% %H%I%T% %B%Y% %A"
5560 NEXT V
5900 STOP
6950 PRINT AT 13,0;" "
6955 PRINT AT 13,0;
6970 RETURN
7000 REM STATUS
7020 PRINT AT 2,14;"ENERGY BANK: ";INT E;" "
7030 IF E<1 THEN GOTO 3800
7040 IF AL>0 THEN PRINT AT 3,14;Q$;" KILL";AT 4,17;"TALLY: ";AL
7050 PRINT AT 7,14;"YOU ARE AT "
7060 PRINT AT 7,14;"YOU ARE AT ";B;",";C
7070 PRINT AT 8,14;" "
7075 PRINT AT 8,14;"IN ";
7080 GOSUB 8500
7100 PRINT " SECTOR"
7120 PRINT AT 12,0;
7490 RETURN
7500 LET R=INT (RND*5)
7520 IF R=0 THEN PRINT "SPOCK: AS YOU HUMANS SAY,"
7540 IF R=1 THEN PRINT "SCOTT: ";
7560 IF R=2 THEN PRINT "UHURA: ";
7580 IF R=3 THEN PRINT "CHEKOV: ";
7600 IF R=4 THEN PRINT "SULU: ";
7900 RETURN
7999 STOP
8000 REM PRINT OUT
8001 FAST
8005 PRINT AT 0,0;
8010 PRINT " 1234567890"
8020 FOR Q=1 TO 10
8025 PRINT AT Q,13;"@@";AT Q,13;" "
8030 IF Q<10 THEN PRINT Q;" ";
8035 IF Q=10 THEN PRINT Q;
8040 FOR P=1 TO 10
8060 IF B(Q,P)=0 THEN PRINT "% ";
8080 IF B(Q,P)=2 THEN PRINT "%$";
8100 IF B(Q,P)=3 THEN PRINT "%X";
8120 IF B(Q,P)=4 THEN PRINT "%*";
8160 NEXT P
8200 NEXT Q
8210 PRINT
8220 PRINT " 1234567890"
8225 SLOW
8490 RETURN
8500 REM SECTOR
8520 LET Q=B*C
8540 IF Q<10 THEN PRINT "ANTARES";
8560 IF Q>9 AND Q<20 THEN PRINT "RIGEL";
8580 IF Q>19 AND Q<30 THEN PRINT "PROCYON";
8600 IF Q>29 AND Q<40 THEN PRINT "VEGA";
8620 IF Q>39 AND Q<50 THEN PRINT "CANOPUS";
8640 IF Q>49 AND Q<60 THEN PRINT "ALTAIR";
8660 IF Q>59 AND Q<70 THEN PRINT "DENEB";
8680 IF Q>69 AND Q<80 THEN PRINT "POLLUX";
8700 IF Q>79 AND Q<90 THEN PRINT "SIRIUS";
8720 IF Q>89 THEN PRINT "CAPELLA";
8740 RETURN
8999 STOP
9000 DIM A(14,14)
9020 DIM B(14,14)
9060 FOR A=1 TO 20
9080 LET X=INT (RND*10+1)
9100 LET Y=INT (RND*10+1)
9120 LET A(X,Y)=1
9140 NEXT A
9160 LET B=5
9180 LET C=5
9200 LET A(B,C)=2
9220 LET B(B,C)=2
9240 LET AL=0
9260 LET E=RND
9279 DIM Z$(10)
9280 IF E<.33 THEN LET Z$="KZINTZIS "
9300 IF E>.33 AND E<.66 THEN LET Z$="ROMULANS "
9320 IF E>.66 THEN LET Z$="KLINGONS "
9329 DIM Q$(7)
9330 LET Q$=Z$
9340 LET E=1000+2000*RND
9900 RETURN
9910 STOP
9920 REM RAND USR 14336
9930 REM SAVE "STREK.B1"
9935 SAVE "1021%6"
9940 RUN
Note: Type-in program listings on this website use ZMAKEBAS notation for graphics characters.
