Enjl

anotherwalljump

Oct 19th, 2019 (edited)
3,536
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 10.81 KB | None | 0 0
  1. -- How to use: Read the comments while having a basic understanding of code.
  2. -- By Enjl, October 2019 - May 2020.
  3. -- 1.3
  4. -- Recent changes:
  5. -- Added blacklist
  6. -- Added whitelist
  7. -- Fixed invisible blocks and hurt blocks being walljumpable.
  8. -- Fixed tail spins
  9.  
  10. local aw = {}
  11. local registeredCharacters = {}
  12.  
  13. -- Global slide speed. Can be overridden per-registration.
  14. aw.slideSpeed = 3
  15. -- Global spinjump setting. Can be overridden per-registration. If false, spinjump key triggers regular jump.
  16. aw.allowSpinjump = true
  17. -- Whether to prevent consecutive jumps off the same wall.
  18. aw.preventLastDirection = false
  19. -- Whether to prevent consecutive jumps off the same wall ONLY when the powerup state is one that permits flight.
  20. aw.preventLastDirectionWhenFlying = true
  21. -- Horizontal speed upon leaving the wall.
  22. aw.xForce = 6
  23. -- Frames of rising momentum after jumping off a wall.
  24. aw.yForce = 12
  25. -- Number of frames you need to stop holding toward the wall before the player drops off.
  26. aw.gracePeriod = 12
  27. -- Whether the character can walljump while holding items
  28. aw.canHoldItem = true
  29. -- Whether the character can perform toad's double jump when equipped with a leaf or tanooki suit. More useful in per-id registration.
  30. aw.canDoubleJump = false
  31. -- Whether the character's walljump is enabled. If set after initialization, it must be set through the enable/disable functions below.
  32. aw.enabled = true
  33.  
  34. -- Wraps frame IDs into a table. Arguments correspond to powerup states. Values correspond to this: http://i.imgur.com/1dnW3g3.png
  35. -- Example: aw.createFrameTable(6, 6, 6, 4, 6, 6, 6) - Uses frame 6 for powerup states 1-7, except for state 4 (leaf).
  36. -- This function is used simply for clarity in the table's purpose. Defining {6, 6, 6, 4, 6, 6, 6} has the same effect.
  37. function aw.createFrameTable(...)
  38.     return {...}
  39. end
  40.  
  41. local function checkOverride(a, b)
  42.     if a == nil then
  43.         return b
  44.     end
  45.     return a
  46. end
  47.  
  48. local blockList = Block.SOLID
  49. local whitelist = {}
  50. local blacklist = {}
  51.  
  52. -- Registers a character ID to the walljump system.
  53. -- The second argument is a frameTable, corresponding to the "wall slide" frames the character should use in each powerup state.
  54. -- The third argument defines further settings overrides. Arguments are: slideSpeed, allowSpinjump, preventLastDirection, xForce, yForce, gracePeriod, canHoldItem, canDoubleJump, enabled
  55. -- To overwrite a character's settings later on, just register them again.
  56. function aw.registerCharacter(id, frameTable, settings)
  57.     settings = settings or {}
  58.     registeredCharacters[id] = {
  59.         frames = frameTable or aw.createFrameTable(-6, -6, -6, -6, -6, -6, -6),
  60.         speed = checkOverride(settings.slideSpeed, aw.slideSpeed),
  61.         spinjump = checkOverride(settings.allowSpinjump, aw.allowSpinjump),
  62.         preventLast = checkOverride(settings.preventLastDirection, aw.preventLastDirection),
  63.         preventLastFly = checkOverride(settings.preventLastDirectionWhenFlying, aw.preventLastDirectionWhenFlying),
  64.         xForce = checkOverride(settings.xForce, aw.xForce),
  65.         yForce = checkOverride(settings.yForce, aw.yForce),
  66.         grace = checkOverride(settings.gracePeriod, aw.gracePeriod),
  67.         item = checkOverride(settings.canHoldItem, aw.canHoldItem),
  68.         canDoubleJump = checkOverride(settings.canDoubleJump, aw.canDoubleJump),
  69.         enabled = checkOverride(settings.enabled, aw.enabled),
  70.     }
  71. end
  72.  
  73. -- Deregisters a previously registered character. If you wish to only temporarily disable a character, consider using enable/disable below.
  74. function aw.deregisterCharacter(id)
  75.     registeredCharacters[id] = nil
  76. end
  77.  
  78. -- Shorthand for automatically registering the five base characters with their default settings.
  79. function aw.registerAllPlayersDefault()
  80.     aw.registerCharacter(CHARACTER_MARIO, aw.createFrameTable(-4, -6, -6, -6, -6, -6, -6)) -- The brothers' skid frames for the small state are in position 4.
  81.     aw.registerCharacter(CHARACTER_LUIGI, aw.createFrameTable(-4, -6, -6, -6, -6, -6, -6))
  82.     aw.registerCharacter(CHARACTER_PEACH, aw.createFrameTable(-6, -6, -6, -6, -6, -6, -6), {allowSpinjump = false, preventLastDirection = true})
  83.     aw.registerCharacter(CHARACTER_TOAD, aw.createFrameTable(-6, -6, -6, -6, -6, -6, -6), {canDoubleJump = true})
  84.     aw.registerCharacter(CHARACTER_LINK, aw.createFrameTable(-5, -5, -5, -5, -5, -5, -5), {allowSpinjump = false}) -- Looks kinda rad.
  85. end
  86.  
  87. local atWall = {0, 0}
  88. local lastDir = {0, 0}
  89. local cantThisFrame = {false,false}
  90.  
  91. -- Manually check wall slide status from outside the code
  92. function aw.isWallSliding(p)
  93.     return atWall[p.idx]
  94. end
  95.  
  96. -- Manually disable wall slide from outside the code. Disables sliding for one frame. To prevent until something happens, call continuously or disable.
  97. function aw.preventWallSlide(p)
  98.     cantThisFrame[p.idx] = true
  99. end
  100.  
  101. -- Enables a character's walljump.
  102. function aw.enable(p)
  103.     registeredCharacters[p.character].enabled = true
  104. end
  105.  
  106. -- Disables a character's walljump.
  107. function aw.disable(p)
  108.     registeredCharacters[p.character].enabled = false
  109. end
  110.  
  111. -- Whitelists a block, making it walljumpable even if it's a lava, hurt, semisolid, nonsolid or sizable block
  112. function aw.whitelist(id)
  113.     if (not whitelist[id]) then
  114.         table.insert(blockList, id)
  115.         whitelist[id] = true
  116.     end
  117. end
  118.  
  119. -- Blacklists a block, making it unwalljumpable even if it would otherwise qualify
  120. function aw.blacklist(id)
  121.     blacklist[id] = true
  122. end
  123. -- If you want to change the blacklist or whitelist at runtime,
  124. -- I recommend instead switching the ID of the block to one that carries different walljump properties.
  125. -- Make sure to maybe also use different sprites, to ensure players know what's going on.
  126.  
  127. -- Below here is nothing of interest for those seeking exposed customizability.
  128. -- For those seeking to alter the code, be my guest, but don't @ me.
  129.  
  130. local directionTable = {
  131.     [-1] = function (p) return p.keys.left end,
  132.     [1] = function (p) return p.keys.right end
  133. }
  134. local collider = Colliders.Box(0,0,1,0)
  135.  
  136. local function walljump(p)
  137.     local idx = p.idx
  138.     local cfg = registeredCharacters[p.character]
  139.     local i = math.sign(atWall[idx])
  140.  
  141.     local dir = i
  142.     if dir == 0 then dir = p.direction end
  143.  
  144.     if dir == lastDir[idx] and (cfg.preventLast or (cfg.preventLastFly and (p.powerup == 4 or p.powerup == 5))) then return end
  145.  
  146.     local xCoord = p.x + p.width * 0.5 + dir * (p.width * 0.5)
  147.    
  148.     collider.x = xCoord + (dir - 1) * 0.5
  149.     collider.y = p.y
  150.     collider.height = p.height * 0.75
  151.  
  152.     local b = Colliders.getColliding{
  153.         a = collider,
  154.         b = blockList,
  155.         btype = Colliders.BLOCK,
  156.         filter = function(o)
  157.             if o.invisible or o:mem(0x5A, FIELD_BOOL) or o:mem(0x5C, FIELD_BOOL) then
  158.                 return false
  159.             end
  160.  
  161.             if (not whitelist[o.id]) and (blacklist[o.id] or Block.LAVA_MAP[o.id] or Block.HURT_MAP[o.id]) then
  162.                 return false
  163.             end
  164.  
  165.             return true
  166.         end
  167.     }
  168.     for k,v in ipairs(b) do
  169.         if directionTable[-dir](p) and atWall[idx] ~= 0 then --letting go of the wall
  170.             atWall[idx] = atWall[idx] - dir
  171.             if atWall[idx] == 0 then
  172.                 p.speedX = -dir
  173.                 return
  174.             end
  175.         elseif not directionTable[-dir](p) then --latching onto a wall
  176.             atWall[idx] = math.abs(cfg.grace) * dir
  177.             p.direction = dir
  178.             p:mem(0x50, FIELD_WORD, 0) --spinjump
  179.         end
  180.         break
  181.         if k == #b then
  182.             atWall[idx] = 0
  183.             if p:mem(0x164, FIELD_WORD) == -1 then
  184.                 p:mem(0x164, FIELD_WORD, 0)
  185.             end
  186.         end
  187.     end
  188.     if #b == 0 then
  189.         atWall[idx] = 0
  190.         if p:mem(0x164, FIELD_WORD) == -1 then
  191.             p:mem(0x164, FIELD_WORD, 0)
  192.         end
  193.     end
  194.    
  195.     if atWall[idx] ~= 0 then --movement handling while at a wall
  196.         p.keys.down = false
  197.         p.keys.right = false
  198.         p.keys.left = false
  199.         p:mem(0x160, FIELD_WORD, 2) -- Projectile timer
  200.         p:mem(0x164, FIELD_WORD, -1) -- Tail timer
  201.         local absspeed = math.abs(registeredCharacters[p.character].speed)
  202.         p.speedY = math.clamp(p.speedY, -absspeed, absspeed) -- If you somehow make the player rise in your own code... :)
  203.         if RNG.randomInt(0, 2) == 2 then
  204.             Animation.spawn(74, xCoord + 8 * ((dir - 1) * 0.5), p.y + 0.75 * p.height)
  205.         end
  206.         if p.keys.jump == KEYS_PRESSED or p.keys.altJump == KEYS_PRESSED then
  207.             p.speedX = -math.abs(cfg.xForce) * dir
  208.             p:mem(0x11C, FIELD_WORD, cfg.yForce)
  209.             p:mem(0x00, FIELD_BOOL, cfg.canDoubleJump) -- Toad Jump
  210.             p:mem(0x18, FIELD_BOOL, true) -- Peach Hover
  211.             Animation.spawn(75, xCoord - 16 , p.y + 0.25 * p.height)
  212.             p.direction = -dir
  213.             lastDir[idx] = dir
  214.             atWall[idx] = 0
  215.             p:mem(0x164, FIELD_WORD, 0) -- Tail timer
  216.             if p.keys.altJump == KEYS_PRESSED and cfg.spinjump then
  217.                 p:mem(0x50, FIELD_WORD, -1) -- spinjump
  218.                 SFX.play(33)
  219.             end
  220.             SFX.play(2)
  221.         end
  222.     end
  223. end
  224.  
  225. function aw.onTick()
  226.     if Level.winState() > 0 then return end
  227.     for k,p in ipairs(Player.get()) do
  228.         if registeredCharacters[p.character] and registeredCharacters[p.character].enabled then
  229.             if p.deathTimer == 0
  230.             and p.mount == 0
  231.             and p.forcedState == 0
  232.             and not cantThisFrame[p.idx]
  233.             and p:mem(0x36, FIELD_BOOL) == false -- Underwater
  234.             and p:isGroundTouching() == false
  235.             and p:mem(0x0C, FIELD_BOOL) == false -- Fairy
  236.             and p:mem(0x40, FIELD_WORD) == 0 -- Climbing
  237.             and p:mem(0x4A, FIELD_BOOL) == false -- Tanooki Statue
  238.             and (registeredCharacters[p.character].item or ((not registeredCharacters[p.character].item) and p:mem(0x154, FIELD_WORD) == 0)) then
  239.                 walljump(p)
  240.             else
  241.                 lastDir[p.idx] = 0
  242.                 atWall[p.idx] = 0
  243.                 if p:mem(0x164, FIELD_WORD) == -1 then
  244.                     p:mem(0x164, FIELD_WORD, 0)
  245.                 end
  246.             end
  247.         else
  248.             lastDir[p.idx] = 0
  249.             atWall[p.idx] = 0
  250.             if p:mem(0x164, FIELD_WORD) == -1 then
  251.                 p:mem(0x164, FIELD_WORD, 0)
  252.             end
  253.         end
  254.         cantThisFrame[p.idx] = false
  255.     end
  256. end
  257.  
  258. function aw.onDraw()
  259.     for k,p in ipairs(Player.get()) do
  260.         if registeredCharacters[p.character] then
  261.             if atWall[k] ~= 0 then
  262.                 p:mem(0x114, FIELD_WORD, registeredCharacters[p.character].frames[p.powerup])
  263.             end
  264.         end
  265.     end
  266. end
  267.  
  268. function aw.onInitAPI()
  269.     registerEvent(aw, "onTick")
  270.     registerEvent(aw, "onDraw")
  271. end
  272.  
  273. return aw
Add Comment
Please, Sign In to add comment