Vince Weaver 5def0952d4 riven: fix annoying bug where keypress occasionally ignored
was a race condition if you pressed a key after the key handling
started but before it unconditionally cleared the strobe
2024-07-26 13:29:57 -04:00

536 lines
7.9 KiB
ArmAsm

;==============================
; Handle Keypress
;==============================
handle_keypress:
.if 0
; first handle joystick
lda JOYSTICK_ENABLED
beq actually_handle_keypress
; only check joystick every-other frame
lda FRAMEL
and #$1
beq actually_handle_keypress
check_button:
lda PADDLE_BUTTON0
bpl button_clear
lda JS_BUTTON_STATE
bne js_check
lda #1
sta JS_BUTTON_STATE
lda #' '
jmp handle_input
button_clear:
lda #0
sta JS_BUTTON_STATE
js_check:
; jsr handle_joystick
js_check_left:
; lda value0
cmp #$20
bcs js_check_right ; if less than 32, left
lda #'A'
bne handle_input
js_check_right:
cmp #$40
bcc js_check_up
lda #'D'
bne handle_input
js_check_up:
; lda value1
cmp #$20
bcs js_check_down
lda #'W'
bne handle_input
js_check_down:
cmp #$40
bcc done_joystick
lda #'S'
bne handle_input
done_joystick:
.endif
actually_handle_keypress:
; check for keypress
lda KEYPRESS
bmi keypress
; no keypress, just exit
rts
; jmp done_keypress
keypress:
; adjust weird Apple II stuff
and #$7f ; clear high bit
cmp #' '
beq handle_input ; make sure not to lose space
and #$df ; convert uppercase to lower case
handle_input:
;==============================
; assume pointer will be moved
; so restore background
pha ; save keypress info
jsr restore_bg_14x14 ; restore old background
inc UPDATE_POINTER ; keypress so assume need move pointer
pla ; restore keypress info
check_sound:
cmp #$13 ; control-S
bne check_left
; toggle sound
lda SOUND_STATUS
eor #SOUND_DISABLED
sta SOUND_STATUS
jmp done_keypress
; can't be ^J as that's the same as down
;check_joystick:
; cmp #'J'
; bne check_left
; lda JOYSTICK_ENABLED
; eor #1
; sta JOYSTICK_ENABLED
; jmp done_keypress
check_left:
cmp #'A'
beq left_pressed
cmp #8 ; left key
bne check_right
left_pressed:
lda CURSOR_X ; check Xpos
beq done_left_pressed ; if Xpos==0 don't move left
do_dec_cursor_x:
dec CURSOR_X ; move left one 3.5 pixel column
done_left_pressed:
jmp done_keypress_moved ; done checking input
check_right:
cmp #'D'
beq right_pressed
cmp #$15 ; right key
bne check_up
right_pressed:
lda CURSOR_X ; load Xpos
cmp #38 ; if Xpos > 38 don't increment
; bcc do_inc_cursor_x ; blt
bcs done_right_pressed ; bge
do_inc_cursor_x:
inc CURSOR_X ; move right one 3.5 pixel column
done_right_pressed:
jmp done_keypress_moved ; done checking input
check_up:
cmp #'W'
beq up_pressed
cmp #$0B ; up key
bne check_down
up_pressed:
lda CURSOR_Y ; load YPos
cmp #4 ; is YPos>4 then we can decrement
bcc done_up_pressed ; blt
do_dec_cursor_y:
dec CURSOR_Y ; subtract 4
dec CURSOR_Y ; 1 byte shorter to sec/lda/sbc/sta?
dec CURSOR_Y
dec CURSOR_Y
done_up_pressed:
jmp done_keypress_moved ; done checking input
check_down:
cmp #'S'
beq down_pressed
cmp #$0A
bne check_return
down_pressed:
lda CURSOR_Y ; load Ypos
cmp #177 ; if Ypos<177 move down (pointer 14 high)
bcs done_down_pressed ; bge
do_inc_cursor_y:
inc CURSOR_Y ; add 4
inc CURSOR_Y
inc CURSOR_Y
inc CURSOR_Y
done_down_pressed:
jmp done_keypress_moved ; done checking input
;check_escape:
; cmp #27
; bne check_return
;escape_pressed:
; inc ESCAPE_PRESSED
; jmp done_keypress
check_return:
cmp #' '
beq return_pressed
cmp #13
bne done_keypress
;======================================
; extra action if space/return pressed
;======================================
return_pressed:
lda IN_SPECIAL
beq not_special_return
special_return:
jsr handle_special
; special case, don't make cursor visible
jmp done_keypress
not_special_return:
lda IN_RIGHT
beq not_right_return
cmp #1
beq right_return
right_uturn:
jsr uturn
jmp done_keypress
right_return:
jsr turn_right
jmp done_keypress
not_right_return:
lda IN_LEFT
beq not_left_return
cmp #1
beq left_return
left_uturn:
jsr uturn
jmp done_keypress
left_return:
jsr turn_left
jmp done_keypress
not_left_return:
jsr go_forward
jmp done_keypress
done_keypress_moved:
lda #1 ; make cursor visible
sta CURSOR_VISIBLE
done_keypress:
bit KEYRESET
rts
;============================
; handle_special
;===========================
; set up jump table fakery
handle_special:
ldy #LOCATION_SPECIAL_FUNC+1
lda (LOCATION_STRUCT_L),Y
pha
dey
lda (LOCATION_STRUCT_L),Y
pha
rts
;=============================
; change direction
;=============================
change_direction:
; load background
lda DIRECTION
bpl no_split
; split text/graphics
bit TEXTGR
; also change sprite cutoff
; not needed for HGR
; ldx #40
; stx psc_smc1+1
;; stx psc_smc2+1
jmp done_split
no_split:
bit FULLGR
; also change sprite cutoff
; ldx #48
; stx psc_smc1+1
;; stx psc_smc2+1
done_split:
and #$f ; mask off special flags
tay
lda log2_table,Y
asl
clc
adc #LOCATION_NORTH_BG
tay
lda (LOCATION_STRUCT_L),Y
sta ZX0_src
iny
lda (LOCATION_STRUCT_L),Y
sta ZX0_src+1
lda #$20 ; decompress to hgr page1
jsr full_decomp
rts
;=============================
; change location
;=============================
;
change_location:
; reset graphics
bit SET_GR
; clear IN_SPECIAL
lda #0
sta IN_SPECIAL
sta IN_RIGHT
sta IN_LEFT
; reset pointer to not visible, centered
sta ANIMATE_FRAME
sta CURSOR_VISIBLE
lda #20
sta CURSOR_X
lda #89
sta CURSOR_Y
lda LOCATION
asl
tay
lda (LOCATIONS_L),Y
sta LOCATION_STRUCT_L
iny
lda (LOCATIONS_L),Y
sta LOCATION_STRUCT_H
jsr change_direction
rts
;==========================
; go forward
;===========================
go_forward:
; update new location
lda DIRECTION
and #$f
tay
lda log2_table,Y
clc
adc #LOCATION_NORTH_EXIT
tay
lda (LOCATION_STRUCT_L),Y
; A has new destination
; FF = can't go forward
; otherwise if top 4 bits set, new level
; otherwise, in current
cmp #$ff
beq cant_go_forward
cmp #$10
bcs new_level
same_level:
sta LOCATION
; update new direction
lda DIRECTION
and #$f
tay
lda log2_table,Y
clc
adc #LOCATION_NORTH_EXIT_DIR
tay
lda (LOCATION_STRUCT_L),Y
sta DIRECTION
jsr change_location
cant_go_forward:
rts
;=========================================
; need to load new "level"/area from disk
;=========================================
; level from disk = top 4 bits
; location = bottom 4 bits
; 12 possible levels per disk. 1..13
; $0 special, means same level
; $E special, means "new disk"
; bottom 4 bits = exit info
; $F special, means "can't go here" when clicked
new_level:
pha
lsr
lsr
lsr
lsr
cmp #$E
beq new_disk
sta WHICH_LOAD ; which level to load from disk
pla
and #$0f
sta LOCATION
; update new direction
lda DIRECTION
and #$f
tay
lda log2_table,Y
clc
adc #LOCATION_NORTH_EXIT_DIR
tay
lda (LOCATION_STRUCT_L),Y
sta DIRECTION
lda #1
sta LEVEL_OVER
rts
new_disk:
pla
and #$f
ora #$80
sta LEVEL_OVER
rts
;==========================
; turn left
;===========================
turn_left:
lda DIRECTION
and #$f
cmp #DIRECTION_N
beq go_west
cmp #DIRECTION_W
beq go_south
cmp #DIRECTION_S
beq go_east
bne go_north
;==========================
; turn right
;===========================
turn_right:
lda DIRECTION
and #$f
cmp #DIRECTION_N
beq go_east
cmp #DIRECTION_E
beq go_south
cmp #DIRECTION_S
beq go_west
bne go_north
;==========================
; uturn
;===========================
uturn:
lda DIRECTION
and #$f
cmp #DIRECTION_N
beq go_south
cmp #DIRECTION_W
beq go_east
cmp #DIRECTION_S
beq go_north
bne go_west
go_north:
lda #DIRECTION_N
jmp done_turning
go_east:
lda #DIRECTION_E
jmp done_turning
go_south:
lda #DIRECTION_S
jmp done_turning
go_west:
lda #DIRECTION_W
jmp done_turning
done_turning:
sta DIRECTION
jsr change_direction
rts