Adds Link's Awakening: DX. Fully imports and forks LADXR, with permission - https://github.com/daid/LADXR
		
			
				
	
	
		
			318 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			318 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
from ..assembler import ASM
 | 
						|
from ..roomEditor import RoomEditor, Object, ObjectVertical, ObjectHorizontal, ObjectWarp
 | 
						|
from ..utils import formatText
 | 
						|
 | 
						|
 | 
						|
def setRequiredInstrumentCount(rom, count):
 | 
						|
    rom.texts[0x1A3] = formatText("You need %d instruments" % (count))
 | 
						|
    if count >= 8:
 | 
						|
        return
 | 
						|
    if count < 0:
 | 
						|
        rom.patch(0x00, 0x31f5, ASM("ld a, [$D806]\nand $10\njr z, $25"), ASM(""), fill_nop=True)
 | 
						|
        rom.patch(0x20, 0x2dea, ASM("ld a, [$D806]\nand $10\njr z, $29"), ASM(""), fill_nop=True)
 | 
						|
        count = 0
 | 
						|
 | 
						|
    # TODO: Music bugs out at the end, unless you have all instruments.
 | 
						|
    rom.patch(0x19, 0x0B79, None, "0000")  # always spawn all instruments, we need the last one as that handles opening the egg.
 | 
						|
    rom.patch(0x19, 0x0BF4, ASM("jp $3BC0"), ASM("jp $7FE0")) # instead of rendering the instrument, jump to the code below.
 | 
						|
    rom.patch(0x19, 0x0BFE, ASM("""
 | 
						|
        ; Normal check fo all instruments
 | 
						|
        ld   e, $08
 | 
						|
        ld   hl, $DB65
 | 
						|
    loop:
 | 
						|
        ldi  a, [hl]
 | 
						|
        and  $02
 | 
						|
        jr   z, $12
 | 
						|
        dec  e
 | 
						|
        jr   nz, loop
 | 
						|
    """), ASM("""
 | 
						|
        jp   $7F2B ; jump to the end of the bank, where there is some space for code.
 | 
						|
    """), fill_nop=True)
 | 
						|
    # Add some code at the end of the bank, as we do not have enough space to do this "in place"
 | 
						|
    rom.patch(0x19, 0x3F2B, "0000000000000000000000000000000000000000000000000000", ASM("""
 | 
						|
        ld   d, $00
 | 
						|
        ld   e, $08
 | 
						|
        ld   hl, $DB65 ; start of has instrument memory
 | 
						|
loop:
 | 
						|
        ld   a, [hl]
 | 
						|
        and  $02
 | 
						|
        jr   z, noinc
 | 
						|
        inc  d
 | 
						|
noinc:
 | 
						|
        inc  hl
 | 
						|
        dec  e
 | 
						|
        jr   nz, loop
 | 
						|
        ld   a, d
 | 
						|
        cp   $%02x    ; check if we have a minimal of this amount of instruments.
 | 
						|
        jp   c, $4C1A ; not enough instruments
 | 
						|
        jp   $4C0B    ; enough instruments
 | 
						|
    """ % (count)), fill_nop=True)
 | 
						|
    rom.patch(0x19, 0x3FE0, "0000000000000000000000000000000000000000000000000000", ASM("""
 | 
						|
    ; Entry point of render code
 | 
						|
        ld   hl, $DB65  ; table of having instruments
 | 
						|
        push bc
 | 
						|
        ldh  a, [$F1]
 | 
						|
        ld   c, a
 | 
						|
        add  hl, bc
 | 
						|
        pop  bc
 | 
						|
        ld   a, [hl]
 | 
						|
        and  $02        ; check if we have this instrument
 | 
						|
        ret  z
 | 
						|
        jp   $3BC0 ; jump to render code
 | 
						|
    """), fill_nop=True)
 | 
						|
 | 
						|
 | 
						|
def setSeashellGoal(rom, count):
 | 
						|
    rom.texts[0x1A3] = formatText("You need %d {SEASHELL}s" % (count))
 | 
						|
 | 
						|
    # Remove the seashell mansion handler (as it will take your seashells) but put a heartpiece instead
 | 
						|
    re = RoomEditor(rom, 0x2E9)
 | 
						|
    re.entities = [(4, 4, 0x35)]
 | 
						|
    re.store(rom)
 | 
						|
 | 
						|
    rom.patch(0x19, 0x0ACB, 0x0C21, ASM("""
 | 
						|
        ldh  a, [$F8] ; room status
 | 
						|
        and  $10
 | 
						|
        ret  nz
 | 
						|
        ldh  a, [$F0] ; active entity state
 | 
						|
        rst  0
 | 
						|
        dw   state0, state1, state2, state3, state4
 | 
						|
 | 
						|
state0:
 | 
						|
        ld   a, [$C124] ; room transition state
 | 
						|
        and  a
 | 
						|
        ret  nz
 | 
						|
        ldh  a, [$99]  ; link position Y
 | 
						|
        cp   $70
 | 
						|
        ret  nc
 | 
						|
        jp   $3B12  ; increase entity state
 | 
						|
 | 
						|
state1:
 | 
						|
        call $0C05 ; get entity transition countdown
 | 
						|
        jr   nz, renderShells
 | 
						|
        ld   [hl], $10
 | 
						|
        call renderShells
 | 
						|
 | 
						|
        ld   hl, $C2B0 ; private state 1 table
 | 
						|
        add  hl, bc
 | 
						|
        ld   a, [wSeashellsCount]
 | 
						|
        cp   [hl]
 | 
						|
        jp   z, $3B12  ; increase entity state
 | 
						|
        ld   a, [hl]   ; increase the amount of compared shells
 | 
						|
        inc  a
 | 
						|
        daa
 | 
						|
        ld   [hl], a
 | 
						|
        ld   hl, $C2C0 ; private state 2 table
 | 
						|
        add  hl, bc
 | 
						|
        inc  [hl] ; increase amount of displayed shells
 | 
						|
        ld   a, $2B
 | 
						|
        ldh  [$F4], a ; SFX
 | 
						|
        ret
 | 
						|
 | 
						|
state2:
 | 
						|
        ld   a, [wSeashellsCount]
 | 
						|
        cp   $%02d
 | 
						|
        jr   c, renderShells
 | 
						|
        ; got enough shells
 | 
						|
        call $3B12 ; increase entity state
 | 
						|
        call $0C05 ; get entity transition countdown
 | 
						|
        ld   [hl], $40
 | 
						|
        jp   renderShells
 | 
						|
 | 
						|
state3:
 | 
						|
        ld   a, $23
 | 
						|
        ldh  [$F2], a ; SFX: Dungeon opened
 | 
						|
        ld   hl, $D806 ; egg room status
 | 
						|
        set  4, [hl]
 | 
						|
        ld   a, [hl]
 | 
						|
        ldh  [$F8], a ; current room status
 | 
						|
        call $3B12 ; increase entity state
 | 
						|
 | 
						|
        ld   a, $00
 | 
						|
        jp   $4C2E
 | 
						|
 | 
						|
state4:
 | 
						|
        ret
 | 
						|
 | 
						|
renderShells:
 | 
						|
        ld   hl, $C2C0 ; private state 2 table
 | 
						|
        add  hl, bc
 | 
						|
        ld   a, [hl]
 | 
						|
        cp   $14
 | 
						|
        jr   c, .noMax
 | 
						|
        ld   a, $14
 | 
						|
.noMax:
 | 
						|
        and  a
 | 
						|
        ret  z
 | 
						|
        ld   c, a
 | 
						|
        ld   hl, spriteRect
 | 
						|
        call $3CE6 ; RenderActiveEntitySpritesRect
 | 
						|
        ret
 | 
						|
 | 
						|
spriteRect:
 | 
						|
        db $10, $1E, $1E, $0C
 | 
						|
        db $10, $2A, $1E, $0C
 | 
						|
        db $10, $36, $1E, $0C
 | 
						|
        db $10, $42, $1E, $0C
 | 
						|
        db $10, $4E, $1E, $0C
 | 
						|
 | 
						|
        db $10, $5A, $1E, $0C
 | 
						|
        db $10, $66, $1E, $0C
 | 
						|
        db $10, $72, $1E, $0C
 | 
						|
        db $10, $7E, $1E, $0C
 | 
						|
        db $10, $8A, $1E, $0C
 | 
						|
 | 
						|
        db $24, $1E, $1E, $0C
 | 
						|
        db $24, $2A, $1E, $0C
 | 
						|
        db $24, $36, $1E, $0C
 | 
						|
        db $24, $42, $1E, $0C
 | 
						|
        db $24, $4E, $1E, $0C
 | 
						|
 | 
						|
        db $24, $5A, $1E, $0C
 | 
						|
        db $24, $66, $1E, $0C
 | 
						|
        db $24, $72, $1E, $0C
 | 
						|
        db $24, $7E, $1E, $0C
 | 
						|
        db $24, $8A, $1E, $0C
 | 
						|
    """ % (count), 0x4ACB), fill_nop=True)
 | 
						|
 | 
						|
 | 
						|
def setRaftGoal(rom):
 | 
						|
    rom.texts[0x1A3] = formatText("Just sail away.")
 | 
						|
 | 
						|
    # Remove the egg and egg event handler.
 | 
						|
    re = RoomEditor(rom, 0x006)
 | 
						|
    for x in range(4, 7):
 | 
						|
        for y in range(0, 4):
 | 
						|
            re.removeObject(x, y)
 | 
						|
    re.objects.append(ObjectHorizontal(4, 1, 0x4d, 3))
 | 
						|
    re.objects.append(ObjectHorizontal(4, 2, 0x03, 3))
 | 
						|
    re.objects.append(ObjectHorizontal(4, 3, 0x03, 3))
 | 
						|
    re.entities = []
 | 
						|
    re.updateOverlay()
 | 
						|
    re.store(rom)
 | 
						|
 | 
						|
    re = RoomEditor(rom, 0x08D)
 | 
						|
    re.objects[6].count = 4
 | 
						|
    re.objects[7].x += 2
 | 
						|
    re.objects[7].type_id = 0x2B
 | 
						|
    re.objects[8].x += 2
 | 
						|
    re.objects[8].count = 2
 | 
						|
    re.objects[9].x += 1
 | 
						|
    re.objects[11] = ObjectVertical(7, 5, 0x37, 2)
 | 
						|
    re.objects[12].x -= 1
 | 
						|
    re.objects[13].x -= 1
 | 
						|
    re.objects[14].x -= 1
 | 
						|
    re.objects[14].type_id = 0x34
 | 
						|
    re.objects[17].x += 3
 | 
						|
    re.objects[17].count -= 3
 | 
						|
    re.updateOverlay()
 | 
						|
    re.overlay[7 + 60] = 0x33
 | 
						|
    re.store(rom)
 | 
						|
 | 
						|
    re = RoomEditor(rom, 0x0E9)
 | 
						|
    re.objects[30].count = 1
 | 
						|
    re.objects[30].x += 2
 | 
						|
    re.overlay[7 + 70] = 0x0E
 | 
						|
    re.overlay[8 + 70] = 0x0E
 | 
						|
    re.store(rom)
 | 
						|
    re = RoomEditor(rom, 0x0F9)
 | 
						|
    re.objects = [
 | 
						|
        ObjectHorizontal(4, 0, 0x0E, 6),
 | 
						|
        ObjectVertical(9, 0, 0xCA, 8),
 | 
						|
        ObjectVertical(8, 0, 0x0E, 8),
 | 
						|
 | 
						|
        Object(3, 0, 0x38),
 | 
						|
        Object(3, 1, 0x32),
 | 
						|
        ObjectHorizontal(4, 1, 0x2C, 3),
 | 
						|
        Object(7, 1, 0x2D),
 | 
						|
        ObjectVertical(7, 2, 0x38, 5),
 | 
						|
        Object(7, 7, 0x34),
 | 
						|
        ObjectHorizontal(0, 7, 0x2F, 7),
 | 
						|
 | 
						|
        ObjectVertical(2, 3, 0xE8, 4),
 | 
						|
        ObjectVertical(3, 2, 0xE8, 5),
 | 
						|
        ObjectVertical(4, 2, 0xE8, 2),
 | 
						|
 | 
						|
        ObjectVertical(4, 4, 0x5C, 3),
 | 
						|
        ObjectVertical(5, 2, 0x5C, 5),
 | 
						|
        ObjectVertical(6, 2, 0x5C, 5),
 | 
						|
 | 
						|
        Object(6, 4, 0xC6),
 | 
						|
        ObjectWarp(1, 0x1F, 0xF6, 136, 112)
 | 
						|
    ]
 | 
						|
    re.updateOverlay(True)
 | 
						|
    re.entities.append((0, 0, 0x41))
 | 
						|
    re.store(rom)
 | 
						|
    re = RoomEditor(rom, 0x1F6)
 | 
						|
    re.objects[-1].target_x -= 16
 | 
						|
    re.store(rom)
 | 
						|
 | 
						|
    # Fix the raft graphics (this overrides some unused graphic tiles)
 | 
						|
    rom.banks[0x31][0x21C0:0x2200] = rom.banks[0x2E][0x07C0:0x0800]
 | 
						|
 | 
						|
    # Patch the owl entity to run our custom end handling.
 | 
						|
    rom.patch(0x06, 0x27F5, 0x2A77, ASM("""
 | 
						|
        ld  a, [$DB95]
 | 
						|
        cp  $0B
 | 
						|
        ret nz
 | 
						|
        ; If map is not fully loaded, return
 | 
						|
        ld  a, [$C124]
 | 
						|
        and a
 | 
						|
        ret nz
 | 
						|
        ; Check if we are moving off the bottom of the map
 | 
						|
        ldh a, [$99]
 | 
						|
        cp  $7D
 | 
						|
        ret c
 | 
						|
        ; Move link back so it does not move off the map
 | 
						|
        ld  a, $7D
 | 
						|
        ldh [$99], a
 | 
						|
        
 | 
						|
        xor a
 | 
						|
        ld  e, a
 | 
						|
        ld  d, a
 | 
						|
 | 
						|
raftSearchLoop:
 | 
						|
        ld  hl, $C280
 | 
						|
        add hl, de
 | 
						|
        ld  a, [hl]
 | 
						|
        and a
 | 
						|
        jr  z, .skipEntity
 | 
						|
        
 | 
						|
        ld  hl, $C3A0
 | 
						|
        add hl, de
 | 
						|
        ld  a, [hl]
 | 
						|
        cp  $6A
 | 
						|
        jr  nz, .skipEntity
 | 
						|
 | 
						|
        ; Raft found, check if near the bottom of the screen.
 | 
						|
        ld  hl, $C210
 | 
						|
        add hl, de
 | 
						|
        ld  a, [hl]
 | 
						|
        cp  $70
 | 
						|
        jr  nc, raftOffWorld
 | 
						|
 | 
						|
.skipEntity:
 | 
						|
        inc e
 | 
						|
        ld  a, e
 | 
						|
        cp  $10
 | 
						|
        jr  nz, raftSearchLoop
 | 
						|
        ret
 | 
						|
 | 
						|
raftOffWorld:
 | 
						|
        ; Switch to the end credits
 | 
						|
        ld  a, $01
 | 
						|
        ld  [$DB95], a
 | 
						|
        ld  a, $00
 | 
						|
        ld  [$DB96], a
 | 
						|
        ret
 | 
						|
    """), fill_nop=True)
 | 
						|
 | 
						|
    # We need to run quickly trough part of the credits, or else it bugs out
 | 
						|
    # Skip the whole windfish part.
 | 
						|
    rom.patch(0x17, 0x0D39, None, ASM("ld a, $18\nld [$D00E], a\nret"))
 | 
						|
    # And skip the zoomed out laying on the log
 | 
						|
    rom.patch(0x17, 0x20ED, None, ASM("ld a, $00"))
 | 
						|
    # Finally skip some waking up on the log.
 | 
						|
    rom.patch(0x17, 0x23BC, None, ASM("jp $4CD9"))
 | 
						|
    rom.patch(0x17, 0x2476, None, ASM("jp $4CD9"))
 |