Still prototype-ish. Made things look fancier when setting control options and even has a little help text! Still missing rewind...

This commit is contained in:
fatratknight 2010-01-07 16:08:14 +00:00
parent 34a6afbdef
commit 70f353fbc6
1 changed files with 195 additions and 98 deletions

View File

@ -1,3 +1,6 @@
-- Multitrack v2 for FCEUX by FatRatKnight
local players= 2 -- You may tweak the number of players here. local players= 2 -- You may tweak the number of players here.
@ -25,6 +28,13 @@ local key = {"right", "left", "down", "up", "L", "O", "J", "K"}
local btn = {"right", "left", "down", "up", "start", "select", "B", "A"} local btn = {"right", "left", "down", "up", "start", "select", "B", "A"}
local shade= 0x00000080
local white= "#FFFFFFFF"
local red= "#FF2000FF"
local green= 0x00FF00FF
local blue= 0x0040FFFF
local orange="#FFC000FF"
local plmin , plmax= 1 , 1 local plmin , plmax= 1 , 1
@ -172,7 +182,6 @@ Draw[5],Draw[6],Draw[7],Draw[8],Draw[9]=Draw.D5,Draw.D6,Draw.D7,Draw.D8,Draw.D9
--***************************************************************************** --*****************************************************************************
function DrawNum(right, top, Number, color, bkgnd) function DrawNum(right, top, Number, color, bkgnd)
--***************************************************************************** --*****************************************************************************
--FatRatKnight
-- Paints the input number as right-aligned. -- Paints the input number as right-aligned.
-- Returns the x position where it would paint another digit. -- Returns the x position where it would paint another digit.
-- It only works with integers. Rounds fractions toward zero. -- It only works with integers. Rounds fractions toward zero.
@ -213,29 +222,45 @@ end
--***************************************************************************** --*****************************************************************************
function limits( value , low , high ) function limits( value , low , high ) -- Expects numbers
--***************************************************************************** --*****************************************************************************
-- Returns value, low, or high. high is returned if value exceeds high,
-- and low is returned if value is beneath low.
return math.max(math.min(value,high),low) return math.max(math.min(value,high),low)
end end
--***************************************************************************** --*****************************************************************************
function within( value , low , high ) function within( value , low , high ) -- Expects numbers
--***************************************************************************** --*****************************************************************************
-- Returns true if value is between low and high. False otherwise.
return ( value >= low ) and ( value <= high ) return ( value >= low ) and ( value <= high )
end end
local keys, lastkeys= {}, {} local keys, lastkeys= {}, {}
--***************************************************************************** --*****************************************************************************
function press(button) function press(button) -- Expects a key
--***************************************************************************** --*****************************************************************************
-- Returns true or false.
-- Generally, if keys is pressed, the next loop around will set lastkeys,
-- and thus prevent returning true more than once if held.
return (keys[button] and not lastkeys[button]) return (keys[button] and not lastkeys[button])
end end
local repeater= 0 local repeater= 0
--***************************************************************************** --*****************************************************************************
function pressrepeater(button) function pressrepeater(button) -- Expects a key
--***************************************************************************** --*****************************************************************************
--DarkKobold & FatRatKnight
-- Returns true or false.
-- Acts much like press if you don't hold the key down. Once held long enough,
-- it will repeatedly return true.
-- Try not to call this function more than once per main loop, since I've only
-- set one repeater variable. Inside a for loop is real bad.
if keys[button] then if keys[button] then
if not lastkeys[button] or repeater >= 3 then if not lastkeys[button] or repeater >= 3 then
return true return true
@ -250,8 +275,11 @@ function pressrepeater(button)
end end
--***************************************************************************** --*****************************************************************************
function JoyToNum(Joys) function JoyToNum(Joys) -- Expects a table containing joypad buttons
--***************************************************************************** --*****************************************************************************
-- Returns a number from 0 to 255, representing button presses.
-- These numbers are the primary storage for this version of this script.
local joynum= 0 local joynum= 0
for i= 1, 8 do for i= 1, 8 do
@ -263,28 +291,22 @@ function JoyToNum(Joys)
return joynum return joynum
end end
local MsgTmr, Message= 0, "Activated multitrack script. Have fun!"
--***************************************************************************** --*****************************************************************************
function NewMsg(text) function ReadJoynum(input, button) -- Expects... Certain numbers!
--***************************************************************************** --*****************************************************************************
MsgTmr= 0 -- Returns true or false. True if the indicated button is pressed
Message= text -- according to the input. False otherwise.
end -- Helper function
return ( bit.band(input , bit.rshift( 0x100,button )) ~= 0 )
local MsgDuration= 60
--*****************************************************************************
function DispMsg()
--*****************************************************************************
if MsgTmr < MsgDuration then
MsgTmr= MsgTmr + 1
gui.text(0,20,Message)
end
end end
--***************************************************************************** --*****************************************************************************
function ShowOnePlayer(x,y,color,button,pl) function ShowOnePlayer(x,y,color,button,pl)
--***************************************************************************** --*****************************************************************************
-- Displays an individual button.
-- Helper function for DisplayInput.
x= x + 4*button - 3 x= x + 4*button - 3
Draw[btn[button]](x,y,color) Draw[btn[button]](x,y,color)
end end
@ -292,6 +314,9 @@ end
--***************************************************************************** --*****************************************************************************
function ShowManyPlayers(x,y,color,button,pl) function ShowManyPlayers(x,y,color,button,pl)
--***************************************************************************** --*****************************************************************************
-- Displays an individual button. Uses 2x3 boxes instead of button graphics.
-- Helper function for DisplayInput.
x= x + (2*players + 1)*(button - 1) + 2*pl - 1 x= x + (2*players + 1)*(button - 1) + 2*pl - 1
gui.box(x,y,x+1,y+2,0,color) gui.box(x,y,x+1,y+2,0,color)
end end
@ -301,8 +326,11 @@ local DispX, DispY= 190, 70
local Past, Future= -12, 20 local Past, Future= -12, 20
local Opaque= 1 local Opaque= 1
--***************************************************************************** --*****************************************************************************
function DisplayOptions(width) function DisplayOptions(width) -- Expects width calculated by DisplayInput
--***************************************************************************** --*****************************************************************************
-- Returns true if Opaque is 0, as a signal to don't bother painting.
-- Separated from DisplayInput to make it clear which half is doing what.
-- Change opacity? -- Change opacity?
if pressrepeater(solid) then Opaque= Opaque + 1/8 end if pressrepeater(solid) then Opaque= Opaque + 1/8 end
@ -355,12 +383,7 @@ end
--***************************************************************************** --*****************************************************************************
function DisplayInput() function DisplayInput()
--***************************************************************************** --*****************************************************************************
local shade= 0x00000080 -- Paints on the screen the current input stored within the script's list.
local white= "#FFFFFFFF"
local red= "#FF2000FF"
local green= 0x00FF00FF
local blue= 0x0040FFFF
local yellow="#FFC000FF"
--Are we showing all players or just one? --Are we showing all players or just one?
local HighlightButton= ShowOnePlayer local HighlightButton= ShowOnePlayer
@ -411,7 +434,7 @@ local yellow="#FFC000FF"
color= white color= white
elseif ReadJoynum(scanz,button) then elseif ReadJoynum(scanz,button) then
if (i > 0) and not (ReadList[pl][button]) then if (i > 0) and not (ReadList[pl][button]) then
color= yellow color= orange
else else
color= green color= green
end end
@ -426,13 +449,6 @@ local yellow="#FFC000FF"
end end
--*****************************************************************************
function ReadJoynum(input, button)
--*****************************************************************************
return ( bit.band(input , bit.rshift( 0x100,button )) ~= 0 )
end
--***************************************************************************** --*****************************************************************************
function SetInput() function SetInput()
--***************************************************************************** --*****************************************************************************
@ -448,7 +464,7 @@ function SetInput()
if temp and ReadJoynum(temp,i) and ReadList[pl][i] then if temp and ReadJoynum(temp,i) and ReadList[pl][i] then
ThisInput[pl][btn[i]]= TrueSwitch[pl][i] ThisInput[pl][btn[i]]= TrueSwitch[pl][i]
else else
ThisInput[pl][btn[i]]= FalseSwitch[pl][button] ThisInput[pl][btn[i]]= FalseSwitch[pl][i]
end end
end end
end end
@ -464,86 +480,119 @@ function ApplyInput()
end end
end end
local XStart, XDiff= 30, 25
local YStart, YDiff= 12, 15
local Status= { {},{},{},{} }
local TargA= {FalseSwitch,TrueSwitch,ReadList,ScriptEdit}
local TrueA= { nil , "inv" , true , true }
local FalsA= { false , true , false , false }
local BasicText= {"Activate: Lets the joypad activate input.",
"Remove: Allows the joypad to remove input.",
"List: Should the script remember button presses?",
"Keys: Allow the script-specific keys to work?"}
--***************************************************************************** --*****************************************************************************
function HandleOptions() function HandleOptions()
--***************************************************************************** --*****************************************************************************
Draw.B( 32,2,"red") -- Lets the user make adjustments on the various controls of this script.
Draw.right(37,2,"white") -- The interface here still needs some work. Bleh.
Draw.B( 42,2,"green")
Draw.B( 52,2,"green") Draw.B( XStart + XDiff ,YStart,red)
Draw.right(57,2,"white") Draw.right(XStart + XDiff + 5,YStart,white)
Draw.B( 62,2,"red") Draw.B( XStart + XDiff +10,YStart,green)
Draw.right(72,2,"green") Draw.B( XStart + XDiff*2 ,YStart,green)
Draw.right(72,6,"green") Draw.right(XStart + XDiff*2+ 5,YStart,white)
Draw.left( 76,2,"green") Draw.B( XStart + XDiff*2+10,YStart,red)
Draw.left( 76,6,"red")
Draw.down( 80,2,"red")
Draw.down( 80,6,"green")
for i= 1, 8 do Draw.right(XStart + XDiff*3 ,YStart ,green)
gui.text( 1, 12*i, btn[i]) Draw.right(XStart + XDiff*3 ,YStart+4,green)
Draw.left( XStart + XDiff*3+ 4,YStart ,green)
Draw.left( XStart + XDiff*3+ 4,YStart+4,red)
Draw.down( XStart + XDiff*3+ 8,YStart ,red)
Draw.down( XStart + XDiff*3+ 8,YStart+4,green)
if FalseSwitch[plmin][i] == nil then local BtnX= math.floor((keys.xmouse- XStart+4)/XDiff)
gui.text(31, 12*i, "Yes") local BtnY= math.floor((keys.ymouse- YStart-6)/YDiff)
else
gui.text(31, 12*i, "No") for i= 1, 4 do
for j= 1, 8 do
Status[i][j]= TargA[i][plmin][j]
local pl= plmin+1
while (pl <= plmax) do
if Status[i][j] ~= TargA[i][pl][j] then
Status[i][j]= "???"
break
end
pl= pl+1
end
if Status[i][j] == TrueA[i] then Status[i][j]= "Yes"
elseif Status[i][j] == FalsA[i] then Status[i][j]= "No" end
end end
if TrueSwitch[plmin][i] == "inv" then
gui.text(51, 12*i, "Yes")
else
gui.text(51, 12*i, "No")
end end
if ReadList[plmin][i] then if within( BtnY , 1 , 9 ) then
gui.text(71, 12*i, "Yes") if within( BtnX , 1 , 4 ) then
else local LowX=BtnX*XDiff +XStart -4
gui.text(71, 12*i, "No") local LowY=BtnY*YDiff +YStart -3
gui.box(LowX,LowY,LowX+XDiff-2,LowY+YDiff-2,blue,blue)
gui.text(1,170,BasicText[BtnX])
end end
if ScriptEdit[plmin][i] then
gui.text(91, 12*i, "Yes")
else
gui.text(91, 12*i, "No")
end end
for i= 1, 5 do
local Xpos= XStart + XDiff*i -5
gui.line(Xpos,0,Xpos,YStart + YDiff*10,green)
end
for j= 1, 8 do
local Ypos= YStart + YDiff*j
gui.line( 0, Ypos-4, XStart + XDiff*5, Ypos-4, green)
gui.text( 1, Ypos, btn[j])
Draw[btn[j]](XStart + XDiff - 12, Ypos, green)
for i= 1, 4 do
gui.text(XStart + XDiff*i, Ypos, Status[i][j])
end
end
for i= 1, 4 do
gui.text(XStart + XDiff*i, YStart + YDiff*9, "All")
end end
if press("leftclick") then if press("leftclick") then
local BtnX= math.floor((keys.xmouse- 8)/20) if within( BtnY , 1 , 9 ) then
local BtnY= math.floor((keys.ymouse- 4)/12) if within( BtnX , 1 , 4 ) then
local LoopS, LoopF= 1, 8
if within( BtnY , 1 , 8 ) then if within( BtnY , 1 , 8 ) then
if BtnX == 1 then LoopS, LoopF= BtnY, BtnY
if FalseSwitch[plmin][BtnY] == nil then end
FalseSwitch[plmin][BtnY]= false
local Target, TstT, TstF= TargA[BtnX], TrueA[BtnX], FalsA[BtnX]
for pl= plmin, plmax do
for Loop= LoopS, LoopF do
if Target[plmax][LoopF] == TstT then
Target[pl][Loop]= TstF
else else
FalseSwitch[plmin][BtnY]= nil Target[pl][Loop]= TstT
end
elseif BtnX == 2 then
if TrueSwitch[plmin][BtnY] == "inv" then
TrueSwitch[plmin][BtnY]= true
else
TrueSwitch[plmin][BtnY]= "inv"
end
elseif BtnX == 3 then
if ReadList[plmin][BtnY] then
ReadList[plmin][BtnY]= false
else
ReadList[plmin][BtnY]= true
end
elseif BtnX == 4 then
if ScriptEdit[plmin][BtnY] then
ScriptEdit[plmin][BtnY]= false
else
ScriptEdit[plmin][BtnY]= true
end end
end end
end end
end end
end
end
end end
--***************************************************************************** --*****************************************************************************
function CatchInput() function CatchInput()
--***************************************************************************** --*****************************************************************************
-- For use with registerbefore. It will get the input and paste it into
-- the input list, no questions asked.
fc= movie.framecount() fc= movie.framecount()
for pl= 1, players do for pl= 1, players do
InputList[pl][fc-1]= JoyToNum(joypad.get(pl)) InputList[pl][fc-1]= JoyToNum(joypad.get(pl))
@ -556,6 +605,9 @@ emu.registerbefore(CatchInput)
--***************************************************************************** --*****************************************************************************
function OnLoad() function OnLoad()
--***************************************************************************** --*****************************************************************************
-- For use with registerload. It updates ThisInput so we're not stuck with
-- junk being carried over when we load something.
fc= movie.framecount() fc= movie.framecount()
SetInput() SetInput()
ApplyInput() -- Sanity versus unpaused stateload ApplyInput() -- Sanity versus unpaused stateload
@ -566,12 +618,14 @@ savestate.registerload(OnLoad)
--***************************************************************************** --*****************************************************************************
function ItIsYourTurn() function ItIsYourTurn()
--***************************************************************************** --*****************************************************************************
-- For use with gui.register. I need to catch input while paused!
-- Needed for a half-decent interface.
local fc_= movie.framecount() local fc_= movie.framecount()
keys= input.get() keys= input.get()
if fc ~= fc_ then -- Sanity versus unusual jump (Reset cycle?) if fc ~= fc_ then -- Sanity versus unusual jump (Reset cycle?)
OnLoad() OnLoad()
fc= fc_
end end
if press(PlayerSwitch) then if press(PlayerSwitch) then
@ -594,7 +648,7 @@ function ItIsYourTurn()
if keys[opt] then if keys[opt] then
gui.opacity(1) gui.opacity(1)
HandleOptions() HandleOptions()
SetInput()
else else
if pressrepeater(Insert) then if pressrepeater(Insert) then
for pl= plmin, plmax do for pl= plmin, plmax do
@ -627,13 +681,56 @@ function ItIsYourTurn()
ApplyInput() ApplyInput()
lastkeys= keys lastkeys= keys
gui.pixel(0,0,"clear")
end end
gui.register(ItIsYourTurn) gui.register(ItIsYourTurn)
emu.pause() emu.pause()
while true do --*****************************************************************************
while true do -- Main loop
--*****************************************************************************
-- Currently does nothing. Everything that needs to be done are in registers.
-- I could just remove this, if I can't make reason to keep it.
emu.frameadvance() emu.frameadvance()
end end
--Someday, I'll implement these texts!
--With both joypad options on, you have full control.
--Held buttons will toggle the script's stored input.
--With this on-off set-up, auto-hold will not
--accidentally cancel input, such as from load state.
--With this off-on set, you can use auto-hold to
--elegantly strip off unwanted input.
--With both joypad options off, you ensure you can't
--accidentally change what's stored in the script.
--List on: Script will apply input stored within.
--This is the whole point of the script, ya know.
--List off: Stored input is ignored. A good idea if
--you don't want the script interfering.
--Keys on: Script keys will toggle the button press
--for the current frame. Meant for precise edits.
--Keys off: Script-specific keys are no longer a
--factor. Long-term control is meant for joypad!
--Apparently, you've selected "All players" and
--they have different options set.
--This is the 'All' button. Selecting this will set
--all 'Yes' or all 'No' for this particular option.