Compare commits

..

No commits in common. "dc33b46260aafb493f1e1d64710e10f1379b0343" and "5110c6c48760b667e448002ff4dd651d8ddcb18c" have entirely different histories.

View File

@ -117,7 +117,7 @@ function _rast(
if (x0<x1) sx=1 if (x0<x1) sx=1
if (y0<y1) sy=1 if (y0<y1) sy=1
local done,err local done=false,err
if dx>dy then if dx>dy then
err=dx/2.0 err=dx/2.0
while x0!=x1 do while x0!=x1 do
@ -207,7 +207,7 @@ function title:draw()
print(lvlstr,51,90,blinkcol) print(lvlstr,51,90,blinkcol)
end end
start_level=0 start_level=11
max_level=31 max_level=31
function title:update() function title:update()
@ -234,10 +234,10 @@ function level:reinit(n)
self.ix=n self.ix=n
self.todo={} self.todo={}
self.bigx,self.bigy=n%8,n\8 self.bigx,self.bigy=n%8,n\8
self.dirty=false
self:load_dynobjs() self:load_dynobjs()
self:recollide_reanchor() self:recollide()
self:reanchor(true)
self:spawn_exit() self:spawn_exit()
end end
@ -252,7 +252,8 @@ end
normpal = {[1]=0,[8]=0,[14]=0} normpal = {[1]=0,[8]=0,[14]=0}
pitpal = {[8]=0,[13]=3,[4]=3,[7]=3} pitpal = {[8]=0,[13]=3,[4]=3,[7]=3}
function level:draw() function level:draw()
cls(5) fillp(0b1111111110111111)
rectfill(0,0,127,127,0x55)
fillp() fillp()
pal(normpal) pal(normpal)
map( map(
@ -288,7 +289,7 @@ function level:busy()
for _,crate in pairs(self._crates) do for _,crate in pairs(self._crates) do
if (#crate.todo>0) return true if (#crate.todo>0) return true
end end
return return false
end end
function level:update() function level:update()
@ -309,12 +310,10 @@ function level:update()
end end
for cix in all(remove) do for cix in all(remove) do
self._crates[cix]=nil self._crates[cix]=nil
self.dirty=true
end end
if #remove>0 then
if self.dirty then self:recollide()
self:recollide_reanchor() self:reanchor()
self.dirty=false
end end
end end
@ -346,17 +345,19 @@ function level:load_dynobjs()
end end
end end
function level:recollide_reanchor() function level:recollide()
self._coll={} self._coll={}
for mx=0,15 do for mx=0,15 do
for my=0,15 do for my=0,15 do
local mxy=_mix(mx,my) local mxy=_mix(mx,my)
self._coll[mxy]= self._coll[mxy]=
fget(self:_mget(mx,my),7) or fget(self:_mget(mx,my),7) or
self._crates[mxy] self._crates[mxy]!=nil
end
end end
end end
function level:reanchor()
local anch_new={} local anch_new={}
for dxy in all{{-1,-1},{1,-1},{-1,1},{1,1}} do for dxy in all{{-1,-1},{1,-1},{-1,1},{1,1}} do
local dx,dy=unpack(dxy) local dx,dy=unpack(dxy)
@ -389,12 +390,18 @@ function level:recollide_reanchor()
end end
end end
local moves={} local anch_old=self._anch
if (anch_old==nil) anch_old={}
for _,old in pairs(anch_old) do
old.dropped=true
end
for k,new in pairs(anch_new) do for k,new in pairs(anch_new) do
local old=(self._anch or {})[k] local old=anch_old[k]
if old then if old then
anch_new[k]=new anch_new[k]=old
if (old.ax!=new.ax or old.ay!=new.ay) add(moves,{old.ax,old.ay,new.ax,new.ay,old,key=k}) old.ax_old,old.ay_old,old.ax,old.ay,old.adx,old.ady=old.ax,old.ay,new.ax,new.ay,new.adx,new.ady
old.dropped=nil
end end
end end
self._anch=anch_new self._anch=anch_new
@ -403,16 +410,15 @@ function level:recollide_reanchor()
add(self._anch_keys,{key=k}) add(self._anch_keys,{key=k})
end end
shellsort(self._anch_keys) shellsort(self._anch_keys)
shellsort(moves)
--printh("!!STARTING!!")
if player.rope then
player.rope:experience_anchor_moves(moves)
end
for point in self:anchor_points() do for point in self:anchor_points() do
point.moved=nil if point.ax_old and player.rope and (point.ax_old != point.ax or point.ay_old != point.ay) then
-- printh("moving: "..tostring({point.ax_old,point.ay_old}).."=>"..tostring({point.ax,point.ay}))
player.rope:be_pushed_by(point,point.ax_old,point.ay_old)
end end
point.ax_old,point.ay_old=nil,nil
end
if (player.rope) player.rope:relax()
end end
function level:win_at(mx,my) function level:win_at(mx,my)
@ -420,28 +426,22 @@ function level:win_at(mx,my)
end end
function level:anchor_points() function level:anchor_points()
-- TODO: Return this to using all() keys=all(self._anch_keys)
local keys=all(self._anch_keys)
return function() return function()
local k=keys() local k=keys()
if (k) return self._anch[k.key] if (k==nil) return nil
end return self._anch[k.key]
end
function level:anchor_at(ax,ay)
for i in self:anchor_points() do
if (i.ax==ax and i.ay==ay) return i
end end
end end
function level:get_open_pit(mx,my) function level:get_open_pit(mx,my)
local pit=self._pits[_mix(mx,my)] local pit=self._pits[_mix(mx,my)]
if (pit and not pit.contents) return pit if (pit and pit.contents==nil) return pit
end end
function level:spawn_exit() function level:spawn_exit()
self._wins={} self._wins={}
local spawned local spawned=false
local spawn_at=function(x,y) local spawn_at=function(x,y)
if (self:_mget(x,y)!=1) return if (self:_mget(x,y)!=1) return
assert(not spawned,x..","..y) assert(not spawned,x..","..y)
@ -470,8 +470,11 @@ function level:spawn_exit()
end end
function level:mcoll(mx,my) function level:mcoll(mx,my)
if ((mx | my) & 0xFFF0!=0) return true return self._coll[_mix(mx,my)]!=false
return self._coll[_mix(mx,my)] end
function level:pcoll(px,py)
return self:mcoll(px\8,py\8)
end end
function level:get_crate(mx,my) function level:get_crate(mx,my)
@ -554,12 +557,12 @@ function level:can_move(
if (is_player and self:win_at(mx1,my1)) return true if (is_player and self:win_at(mx1,my1)) return true
if (is_player and self:get_open_pit(mx1,my1)) return wrongbleep:adequately_warned() if (is_player and self:get_open_pit(mx1,my1)) return wrongbleep:adequately_warned()
if (self:mcoll(mx1,my1) or player.x==mx1 and player.y==my1) return if (self:mcoll(mx1,my1) or player.x==mx1 and player.y==my1) return false
if player.rope then if player.rope then
local w,h=1.2,0.2 local w,h=1.2,0.2
if (dmx==0) w,h=0.2,1.2 if (dmx==0) w,h=0.2,1.2
if (player.rope:collide_mrect(mx0+ropecheck[dmx+2],my0+ropecheck[dmy+2],w,h,exclude_src,exclude_dst)) return if (player.rope:collide_mrect(mx0+ropecheck[dmx+2],my0+ropecheck[dmy+2],w,h,exclude_src,exclude_dst)) return false
end end
return true return true
@ -575,7 +578,8 @@ function level:tug_crate(mx0,my0,dmx,dmy)
local px1,py1=mx1*8,my1*8 local px1,py1=mx1*8,my1*8
existing.todo={ existing.todo={
{px=px1+dmx,py=py1+dmy,mx=mx1,my=my1,update=function() {px=px1+dmx,py=py1+dmy,mx=mx1,my=my1,update=function()
self.dirty=true self:recollide()
self:reanchor()
return true return true
end}, end},
{px=px1,py=py1} {px=px1,py=py1}
@ -599,8 +603,6 @@ function player:reinit(x,y)
self.x,self.y=x,y self.x,self.y=x,y
self.px,self.py=0,0 self.px,self.py=0,0
self.todo={} self.todo={}
self.cooldown=4
self.rope=nil -- don't elide, needs to be cleared if present
self.fall_frame=0 self.fall_frame=0
@ -634,10 +636,6 @@ function player:update()
self.orienty=1 self.orienty=1
end end
function btncd(x)
return (kbd:btn(x) and self.cooldown==0) or kbd:btnp(x)
end
if kbd:btn(4) then if kbd:btn(4) then
if kbd:btnp(4) and self.rope then if kbd:btnp(4) and self.rope then
self.rope:destroy() self.rope:destroy()
@ -645,30 +643,23 @@ function player:update()
end end
-- wait for user to release it -- wait for user to release it
else else
local x,y=self.x,self.y
local function try_move(dx,dy,f) local function try_move(dx,dy,f)
if level:can_move(true,x,y,dx,dy,0,2) then if (level:can_move(true,self.x,self.y,dx,dy,0,2)) self.todo=f return
self.todo=f
self.cooldown=3
local t=f[#f]
t.x=x+dx
t.y=y+dy
return
end
wrongbleep:bleep() wrongbleep:bleep()
end end
if btncd(0) then if kbd:btn(0) then
try_move(-1,0,{{orientx=-1,orienty=0,px=-2,py=0},{px=1}}) try_move(-1,0,{{orientx=-1,orienty=0,px=-2},{px=-7},{x=self.x-1,px=0}})
elseif btncd(1) then elseif kbd:btn(1) then
try_move(1,0,{{orientx=1,orienty=0,px=2,py=0},{px=-1}}) try_move(1,0,{{orientx=1,orienty=0,px=2},{px=7},{x=self.x+1,px=0}})
elseif btncd(2) then elseif kbd:btn(2) then
try_move(0,-1,{{orienty=-1,px=0,py=-2},{py=1}}) try_move(0,-1,{{orienty=-1,py=-2},{py=-7},{y=self.y-1,py=0}})
elseif btncd(3) then elseif kbd:btn(3) then
try_move(0,1,{{orienty=1,px=0,py=2},{py=-1}}) try_move(0,1,{{orienty=1,py=2},{py=7},{y=self.y+1,py=0}})
elseif not self.rope and kbd:btnr(4) then elseif not self.rope and kbd:btnr(4) then
local dx,dy=self.orientx,self.orienty local dx,dy=self.orientx,self.orienty
if (dy!=0) dx=0 if (dy!=0) dx=0
local x,y=self.x,self.y
while not level:mcoll(x,y) do x+=dx y+=dy end while not level:mcoll(x,y) do x+=dx y+=dy end
self.rope=rope:new( self.rope=rope:new(
@ -688,9 +679,7 @@ function player:update()
end end
end end
if (#self.todo==0) self.px=0 self.py=0
_apply(self,self.todo) _apply(self,self.todo)
if (self.cooldown>0) self.cooldown-=1
if self.rope then if self.rope then
self.rope:update() self.rope:update()
@ -775,7 +764,7 @@ function player:draw()
-- vanish colors -- vanish colors
local vanish=split"13,15,14,5,4,12,2,3,9,10" local vanish=split"13,15,14,5,4,12,2,3,9,10"
for i,ilc in ipairs(vanish) do for i,ilc in ipairs(vanish) do
if (vanish_level>i/#vanish) pal(ilc,5) if (vanish_level>i/#vanish) pal(ilc,1)
end end
if (self.fall_frame>3) local zc=@0x5f00&0xf0 for i=0x5f00,0x5f0c,4 do poke4(i,0x0101.0101) end poke(0x5f00,zc|0x01) if (self.fall_frame>3) local zc=@0x5f00&0xf0 for i=0x5f00,0x5f0c,4 do poke4(i,0x0101.0101) end poke(0x5f00,zc|0x01)
@ -852,9 +841,9 @@ function rope:update()
if (self.state.frame>=3) self.state={name="latched"} if (self.state.frame>=3) self.state={name="latched"}
elseif self.state.name=="latched" then elseif self.state.name=="latched" then
if (not self.latch) wrongbleep:bleep(5) self:destroy() return if (self.latch==nil) wrongbleep:bleep(5) self:destroy() return
if self.latch.rec then if self.latch and self.latch.rec then
self:drag_src( self:drag_src(
self.latch.rec.mx+0.5+self.latch.ax_offset, self.latch.rec.mx+0.5+self.latch.ax_offset,
self.latch.rec.my+0.5+self.latch.ay_offset self.latch.rec.my+0.5+self.latch.ay_offset
@ -869,7 +858,7 @@ function rope:update()
elseif self.state.name=="destroy" then -- destroy elseif self.state.name=="destroy" then -- destroy
self.state.frame+=1 self.state.frame+=1
if (self.state.frame>=4) self.state={name="done",reelin=self.state.reelin} if (self.state.frame>=5) self.state={name="done",reelin=self.state.reelin}
else else
-- done state -- done state
end end
@ -916,10 +905,10 @@ end
function rope:draw(artificial_px,artificial_py) function rope:draw(artificial_px,artificial_py)
local points,highlight=self:_tug(true) local points,highlight=self:_tug(true)
local n,perc_to_show,from_end = self.state.name,1.0 local n,perc_to_show,from_end = self.state.name,1.0,false
if (n=="done") return if (n=="done") return
if (n=="cast") perc_to_show=self.state.frame/2 if (n=="cast") perc_to_show=self.state.frame/2
if (n=="destroy") perc_to_show=(1.0-self.state.frame/4)^2 if (n=="destroy") perc_to_show=(1.0-self.state.frame/8)^2
if (self.state.reelin) from_end=true if (self.state.reelin) from_end=true
points[#points]={x=artificial_px,y=artificial_py} points[#points]={x=artificial_px,y=artificial_py}
@ -933,11 +922,6 @@ function rope:draw(artificial_px,artificial_py)
local len_cumulative=0 local len_cumulative=0
local ia,iz,istep=#points-1,1,-1 local ia,iz,istep=#points-1,1,-1
if (from_end) ia,iz,istep=1,#points-1,1 if (from_end) ia,iz,istep=1,#points-1,1
local function colorh(ix)
color(8)
if (highlight==ix) color(12)
end
for i=ia,iz,istep do for i=ia,iz,istep do
local src=points[i] local src=points[i]
local dst=points[i+1] local dst=points[i+1]
@ -954,35 +938,36 @@ function rope:draw(artificial_px,artificial_py)
if from_end then if from_end then
x,y=x+dx-dx*coef,y+dy-dy*coef x,y=x+dx-dx*coef,y+dy-dy*coef
end
dx,dy=dx*coef,dy*coef dx,dy=dx*coef,dy*coef
else
colorh(i) dx,dy=dx*coef,dy*coef
local function lf(d0,d1,w)
linefill(x+d0*dx,y+d0*dy,x+d1*dx,y+d1*dy,w)
end end
lf(0,0.25,1.0) local color=8
lf(0.25,1,0.5) if (highlight==i) color=12
lf(0.9,1,1.0)
circfill(x+dx+0.5,y+dy+0.5,1.0) linefill(x,y,x+0.25*dx,y+0.25*dy,1.0,color)
linefill(x+0.25*dx,y+0.25*dy,x+1*dx,y+1*dy,0.5,color)
linefill(x+0.9*dx,y+0.9*dy,x+dx,y+dy,1.0,color)
circfill(x+dx+0.5,y+dy+0.5,1.0,color)
end end
end end
-- draw latch -- draw latch
local l=self.latch if self.latch!=nil and self.latch.rec and (perc_to_show>=1.0 or from_end) then
if l and l.rec and (perc_to_show>=1.0 or from_end) then local x,y=self.latch.rec.px,self.latch.rec.py
local function rfsplit(x) rectfill(unpack(split(x))) end local ldx,ldy=self.latch.dx,self.latch.dy
local ldx,ldy=l.dx,l.dy local color=8
colorh(0) if (highlight==0) color=12
camera(-l.rec.px,-l.rec.py) if self.latch.dx==-1 and self.latch.dy==0 then
if (ldx==-1) rfsplit"0,3,2,4" rectfill(x,y+3,x+2,y+4,color)
if (ldx==1) rfsplit"5,3,7,4" elseif self.latch.dx==1 and self.latch.dy==0 then
if (ldy==-1) rfsplit"3,0,4,2" rectfill(x+5,y+3,x+7,y+4,color)
if (ldy==1) rfsplit"3,5,4,7" elseif self.latch.dx==0 and self.latch.dy==-1 then
camera() rectfill(x+3,y,x+4,y+2,color)
color() elseif self.latch.dx==0 and self.latch.dy==1 then
rectfill(x+3,y+5,x+4,y+7,color)
end
end end
-- debug -- debug
@ -991,26 +976,31 @@ function rope:draw(artificial_px,artificial_py)
local sy=0 local sy=0
while true do while true do
if (n1==nil) break if (n1==nil) break
local anch=level:anchor_at(n1.ax,n1.ay)
local x=n1.ax*8 local x=n1.ax*8
local y=n1.ay*8 local y=n1.ay*8
if anch then if n1.associated_with then
if (anch.adx>0) x-=1 if (n1.associated_with.adx>0) x-=1
if (anch.ady>0) y-=1 if (n1.associated_with.ady>0) y-=1
end end
rectfill(x-1,y-1,x+1,y+1,12) rectfill(x-1,y-1,x+1,y+1,12)
print("ax="..n1.ax..",ay="..n1.ay,72,sy) print("ax="..n1.ax..",ay="..n1.ay,72,sy)
sy+=7 print(tostring(n1.associated_with and (not n1.associated_with.dropped and n1.associated_with.ax==n1.ax and n1.associated_with.ay==n1.ay)),76,sy+7)
sy+=14
local n0=n1.prev local n0=n1.prev
local n2=n1.next local n2=n1.next
if n0!=nil and n2!=nil then if n0!=nil and n2!=nil then
if anch then if n1.associated_with then
local _,_,_=would_stick(anch,n0.ax,n0.ay,n1.ax,n1.ay,n2.ax,n2.ay) local _,_,_,adx,ady=would_stick(n0.ax,n0.ay,n1.associated_with,n2.ax,n2.ay)
assert(adx==-1 or adx==0 or adx==1)
assert(ady==-1 or ady==0 or ady==1)
--assert(not (adx==0 and ady==0))
local c=3 local c=3
if (anch.dropped) c=4 if (n1.associated_with.dropped) c=4
rectfill(x+2,y+2,x+4,y+4,c) rectfill(x+2,y+2,x+4,y+4,c)
pset(x+adx*2,y,9)
pset(x,y+ady*2,9)
else else
rectfill(x+2,y+2,x+4,y+4,2) rectfill(x+2,y+2,x+4,y+4,2)
end end
@ -1050,33 +1040,56 @@ end
function rope:relax() function rope:relax()
local n0=self.src local n0=self.src
while true do while true do
local n1=n0.next if (n0==nil) break
if (not n1) break if n0.associated_with and n0.associated_with.dropped then
local n2=n1.next for i in level:anchor_points() do
if i.ax==n0.ax and i.ay==n0.ay then
n0.associated_with=i
break
end
end
end
n0=n0.next
end
local n0=self.src
while true do
local n1=n0.next
if (n1==nil) break
local n2=n1.next
if n0.ax==n1.ax and n0.ay==n1.ay then if n0.ax==n1.ax and n0.ay==n1.ay then
n0.next=n2 n0.next=n2
if (n2) n2.prev=n0 if (n2!=nil) n2.prev=n0
else else
n0=n0.next n0=n0.next
end end
end end
local n0=self.src local n0=self.src
while n0 do while true do
if (n0==nil) return
local n1=n0.next local n1=n0.next
if (not n1) return if (n1==nil) return
local n2=n1.next local n2=n1.next
if (not n2) return if (n2==nil) return
if n1.associated_with!=nil then
local x0,y0=n0.ax,n0.ay local x0,y0=n0.ax,n0.ay
local x1,y1=n1.ax,n1.ay local x1,y1=n1.ax,n1.ay
local x2,y2=n2.ax,n2.ay local x2,y2=n2.ax,n2.ay
local anch=level:anchor_at(n1.ax,n1.ay) if x1!=n1.associated_with.ax or y1!=n1.associated_with.ay then
local would,x1_new,y1_new=would_stick(anch,x0,y0,x1,y1,x2,y2) -- printh("dragging home: "..tostring{n1.ax,n1.ay}.."->"..tostring(n1.associated_with))
self:_drag(n1,n1.associated_with.ax,n1.associated_with.ay)
end
local would,x1_new,y1_new=would_stick(x0,y0,n1.associated_with,x2,y2)
if not would and not (n1.ax==x1_new and n1.ay==y1_new) then if not would and not (n1.ax==x1_new and n1.ay==y1_new) then
-- printh("dragging: "..tostring{n1.associated_with, {x1_new, y1_new}})
-- printh("relaxing: "..tostring(n0.associated_with).."->"..tostring(n1.associated_with).."->"..tostring(n2.associated_with))
self:_drag(n1,x1_new,y1_new,n1.ax,n1.ay) self:_drag(n1,x1_new,y1_new,n1.ax,n1.ay)
n0=n1.prev n0=n1.prev
n2=n1.next n2=n1.next
@ -1085,6 +1098,7 @@ function rope:relax()
n1.next=nil n1.next=nil
n1.prev=nil n1.prev=nil
else n0=n0.next end else n0=n0.next end
else n0=n0.next end
end end
end end
@ -1097,7 +1111,7 @@ function rope:_check_sane()
local qxs,qys={},{} local qxs,qys={},{}
while true do while true do
local n1=n0.next local n1=n0.next
if (not n1) break if (n1==nil) break
_rast(qxs,qys,flr(n0.ax*2),flr(n0.ay*2),flr(n1.ax*2),flr(n1.ay*2)) _rast(qxs,qys,flr(n0.ax*2),flr(n0.ay*2),flr(n1.ax*2),flr(n1.ay*2))
n0=n1 n0=n1
@ -1119,21 +1133,21 @@ function rope:_check_sane()
end end
local function _blocked(qx,qy) local function _blocked(qx,qy)
for i in all(_possible_tiles(qx,qy)) do for i in all(_possible_tiles(qx,qy)) do
if (not level:mcoll(i.mx,i.my)) return if (not level:mcoll(i.mx,i.my)) return false
end end
return true return true
end end
-- find cases where i move through an impassable zone -- find cases where i move through an impassable zone
for i=1,#qxs do for i=1,#qxs do
if (_blocked(qxs[i],qys[i])) return if (_blocked(qxs[i],qys[i])) return false
end end
-- find cases where i am cut off diagonally -- find cases where i am cut off diagonally
for i=3,#qxs do for i=3,#qxs do
local qx1,qy1=qxs[i-1],qys[i-1] local qx1,qy1=qxs[i-1],qys[i-1]
if qx1%2==0 and qy1%2==0 then if qx1%2==0 and qy1%2==0 then
local ok local ok=false
for m0 in all(_possible_tiles(qxs[i-2],qys[i-2])) do for m0 in all(_possible_tiles(qxs[i-2],qys[i-2])) do
for m2 in all(_possible_tiles(qxs[i],qys[i])) do for m2 in all(_possible_tiles(qxs[i],qys[i])) do
local mx0,my0=m0.mx,m0.my local mx0,my0=m0.mx,m0.my
@ -1150,75 +1164,49 @@ function rope:_check_sane()
end end
end end
if (not ok) return if (not ok) return false
end end
end end
return true return true
end end
function would_stick(anchor,x0,y0,x1,y1,x2,y2) function would_stick(x0,y0,anchor,x2,y2)
x1,y1=x1 or anchor.ax,y1 or anchor.ay local x1,y1=anchor.ax,anchor.ay
local dx,dy=x2-x0,y2-y0 local dx=x2-x0
if (x1==x0 and y1==y0) return local dy=y2-y0
if (x1==x2 and y1==y2) return
local function switch_ends() if (x1==x0 and y1==y0) return false
dx,dy,x0,y0,x2,y2=-dx,-dy,x2,y2,x0,y0 if (x1==x2 and y1==y2) return false
end
local function signs(douter,dinner) local adx,ady
local adb=sgn0(dinner) local x1_new,y1_new
return -sgn0(douter)*adb,adb
end
local adx,ady,x1_new,y1_new
if abs(dx)>abs(dy) then if abs(dx)>abs(dy) then
if (x0>x2) switch_ends() if (x0>x2) dx,x0,y0,x2,y2=-dx,x2,y2,x0,y0
x1_new,y1_new=x1,y0+(x1-x0)/dx*dy local dprop=(x1-x0)/dx
adx,ady=signs(dy,y1_new-y1) x1_new,y1_new=x1,y0+dprop*(y2-y0)
ady=sgn0(y1_new-y1)
adx=0
if (y0>y2) adx=ady
if (y0<y2) adx=-ady
else else
if (y0>y2) switch_ends() if (y0>y2) dy,x0,y0,x2,y2=-dy,x2,y2,x0,y0
x1_new,y1_new=x0+(y1-y0)/dy*dx,y1 local dprop=(y1-y0)/dy
ady,adx=signs(dx,x1_new-x1) x1_new,y1_new=x0+dprop*(x2-x0),y1
adx=sgn0(x1_new-x1)
ady=0
if (x0>x2) ady=adx
if (x0<x2) ady=-adx
end end
return local wouldnt=anchor.dropped or (anchor.adx!=adx or anchor.ady!=ady)
anchor and anchor.adx==adx and anchor.ady==ady,
x1_new,y1_new return not wouldnt,x1_new,y1_new,adx,ady
end end
function rope:experience_anchor_moves(moves) function rope:be_pushed_by(anchor,ax_old,ay_old)
self:_be_dragged_by(moves)
self:_be_pushed_by(moves)
self:relax()
end
function rope:_be_dragged_by(moves)
local n=self.src
while n do
for t in all(moves) do
local ax_old,ay_old,ax_new,ay_new=unpack(t)
if (ax_old==n.ax and ay_old==n.ay) n.dest={ax_new,ay_new} break
end
n=n.next
end
n=self.src
while n do
if (n.dest) self:_drag(n,unpack(n.dest)) n.dest=nil
n=n.next
end
end
function rope:_be_pushed_by(moves)
for i in all(moves) do
self:_be_pushed_by1(unpack(i))
end
end
function rope:_be_pushed_by1(ax_old,ay_old,ax_new,ay_new,anch)
local n0=self.src local n0=self.src
local ax_new,ay_new=anchor.ax,anchor.ay
while true do while true do
n1=n0.next n1=n0.next
if (not n1) return if (not n1) return
@ -1228,30 +1216,37 @@ function rope:_be_pushed_by1(ax_old,ay_old,ax_new,ay_new,anch)
local nxmn,nxmx = _mnmx(nx0,nx1) local nxmn,nxmx = _mnmx(nx0,nx1)
local nymn,nymx = _mnmx(ny0,ny1) local nymn,nymx = _mnmx(ny0,ny1)
-- printh(tostring({anchor,nxmn,nxmx,nymn,nymx}))
if if
(ax_new!=ax_old or (nxmn<ax_new and ax_new<nxmx)) and (ax_new!=ax_old or (nxmn<=anchor.ax and anchor.ax<=nxmx)) and
(ay_new!=ay_old or (nymn<ay_new and ay_new<nymx)) and (ay_new!=ay_old or (nymn<=anchor.ay and anchor.ay<=nymx)) and
(_which_side(ax_old,ay_old,nx0,ny0,nx1,ny1)!= (_which_side(ax_old,ay_old,nx0,ny0,nx1,ny1)!=
_which_side(ax_new,ay_new,nx0,ny0,nx1,ny1) _which_side(ax_new,ay_new,nx0,ny0,nx1,ny1)
) and would_stick(anch,nx0,ny0,nil,nil,nx1,ny1) ) and would_stick(nx0,ny0,anchor,nx1,ny1)
then then
-- printh("found (in): "..tostring({{nx0,ny0},{nx1,ny1}, anchor}))
local nx05,ny05 local nx05,ny05
if ax_new==ax_old then if ax_new==ax_old then
nx05=ax_new nx05=anchor.ax
ny05=ny0+(nx05-nx0)/(nx1-nx0) * (ny1-ny0) ny05=ny0+(nx05-nx0)/(nx1-nx0) * (ny1-ny0)
-- printh("found (x): "..tostring({nx05,ny05}))
elseif ay_new==ay_old then elseif ay_new==ay_old then
ny05=ay_new ny05=anchor.ay
nx05=nx0+(ny05-ny0)/(ny1-ny0) * (nx1-nx0) nx05=nx0+(ny05-ny0)/(ny1-ny0) * (nx1-nx0)
-- printh("found (y): "..tostring({nx05,ny05}))
else
assert(false,"wtf?")
end end
local n05={ax=nx05,ay=ny05,prev=n0,next=n1} local n05={ax=nx05,ay=ny05,associated_with=anchor,prev=n0,next=n1}
--printh("adding: "..tostring({nx05,ny05,anchor}))
n0.next=n05 n0.next=n05
n1.prev=n05 n1.prev=n05
-- printh("creating: "..tostring{anch,{n0.ax,n0.ay},{n05.ax,n05.ay},{n1.ax,n1.ay}}) self:_drag(n05,anchor.ax,anchor.ay)
-- printh("dragging: "..tostring{anch,{n05.ax,n05.ay},{ax_new,ay_new}}) -- printh("dragged: "..tostring({n05.ax,n05.ay,anchor}))
self:_drag(n05,ax_new,ay_new) -- printh("local: "..tostring(n0.associated_with).."->"..tostring(n05.associated_with).."->"..tostring(n1.associated_with))
else else
n0=n0.next n0=n0.next
end end
@ -1266,7 +1261,7 @@ function rope:_drag(n1,ax1_new,ay1_new,ax_removing,ay_removing)
(anchor.ax==ax_removing and anchor.ay==ay_removing) (anchor.ax==ax_removing and anchor.ay==ay_removing)
end end
if (ax_far0==ax_far1 and ay_far0==ay_far1) return if (ax_far0==ax_far1 and ay_far0==ay_far1) return nil
if ax_far0==ax_far1 then if ax_far0==ax_far1 then
local ax_far=ax_far0 local ax_far=ax_far0
@ -1278,7 +1273,7 @@ function rope:_drag(n1,ax1_new,ay1_new,ax_removing,ay_removing)
if if
not _uncreatable(anchor) and not _uncreatable(anchor) and
(ax0<=anchor.ax and anchor.ax<=ax1) and (ax0<=anchor.ax and anchor.ax<=ax1) and
would_stick(anchor,ax_pivot,ay_pivot,nil,nil,ax_far,ay_far_new) and would_stick(ax_pivot,ay_pivot,anchor,ax_far,ay_far_new) and
( (
_which_side(anchor.ax,anchor.ay,ax_pivot,ay_pivot,ax_far,ay_far_old) != _which_side(anchor.ax,anchor.ay,ax_pivot,ay_pivot,ax_far,ay_far_old) !=
_which_side(anchor.ax,anchor.ay,ax_pivot,ay_pivot,ax_far,ay_far_new) _which_side(anchor.ax,anchor.ay,ax_pivot,ay_pivot,ax_far,ay_far_new)
@ -1299,7 +1294,7 @@ function rope:_drag(n1,ax1_new,ay1_new,ax_removing,ay_removing)
if if
not _uncreatable(anchor) and not _uncreatable(anchor) and
(ay0<=anchor.ay and anchor.ay<=ay1) and (ay0<=anchor.ay and anchor.ay<=ay1) and
would_stick(anchor,ax_pivot,ay_pivot,nil,nil,ax_far_new,ay_far) and would_stick(ax_pivot,ay_pivot,anchor,ax_far_new,ay_far) and
( (
_which_side(anchor.ax,anchor.ay,ax_pivot,ay_pivot,ax_far_old,ay_far) != _which_side(anchor.ax,anchor.ay,ax_pivot,ay_pivot,ax_far_old,ay_far) !=
_which_side(anchor.ax,anchor.ay,ax_pivot,ay_pivot,ax_far_new,ay_far) _which_side(anchor.ax,anchor.ay,ax_pivot,ay_pivot,ax_far_new,ay_far)
@ -1318,26 +1313,30 @@ function rope:_drag(n1,ax1_new,ay1_new,ax_removing,ay_removing)
n1.ay=ay1_new n1.ay=ay1_new
local n0=n1.prev local n0=n1.prev
while n0 do while true do
if (n0==nil) break
local anch=_sweep_radar( local anch=_sweep_radar(
n0.ax,n0.ay,n1.ax,n1.ay, n0.ax,n0.ay,n1.ax,n1.ay,
n0.ax,n0.ay,ax1_old,ay1_old,ax1_new,ay1_new n0.ax,n0.ay,ax1_old,ay1_old,ax1_new,ay1_new
) )
if (not anch) break if (anch==nil) break
local n05={ax=anch.ax,ay=anch.ay,prev=n0,next=n1} local n05={ax=anch.ax,ay=anch.ay,associated_with=anch,prev=n0,next=n1}
-- printh("creating pre: "..tostring(n0.associated_with).."->"..tostring(n05.associated_with).."->"..tostring(n1.associated_with))
n0.next=n05 n0.next=n05
n1.prev=n05 n1.prev=n05
n0=n05 n0=n05
end end
local n2=n1.next local n2=n1.next
while n2 do while true do
if (n2==nil) break
local anch=_sweep_radar( local anch=_sweep_radar(
n1.ax,n1.ay,n2.ax,n2.ay, n1.ax,n1.ay,n2.ax,n2.ay,
n2.ax,n2.ay,ax1_old,ay1_old,ax1_new,ay1_new n2.ax,n2.ay,ax1_old,ay1_old,ax1_new,ay1_new
) )
if (not anch) break if (anch==nil) break
local n15={ax=anch.ax,ay=anch.ay,prev=n1,next=n2} local n15={ax=anch.ax,ay=anch.ay,associated_with=anch,prev=n1,next=n2}
-- printh("creating post: "..tostring(n1.associated_with).."->"..tostring(n15.associated_with).."->"..tostring(n2.associated_with))
n1.next=n15 n1.next=n15
n2.prev=n15 n2.prev=n15
n2=n15 n2=n15
@ -1345,7 +1344,7 @@ function rope:_drag(n1,ax1_new,ay1_new,ax_removing,ay_removing)
end end
function _stepfrom(x0,x1) function _stepfrom(x0,x1)
local done local done=false
if x0==x1 then if x0==x1 then
return function() return function()
if (done) return if (done) return
@ -1360,12 +1359,12 @@ function _stepfrom(x0,x1)
local i=flr(x0) local i=flr(x0)
local top=flr(x1) local top=flr(x1)
return function() return function()
if (done) return if (done) return nil
i+=1 i+=1
if i>top then if i>top then
done = true done = true
if (x1!=flr(x1)) return mul*x1 if (x1!=flr(x1)) return mul*x1
return return nil
end end
return mul*i return mul*i
end end
@ -1391,25 +1390,25 @@ function rope:collide_mrect(mx0,my0,mw,mh,exclude_src,exclude_dst)
while true do while true do
local n1=n0.next local n1=n0.next
if (not n1) return if (n1==nil) return false
local nd=n0 local nd=n0
for i=1,exclude_dst do for i=1,exclude_dst do
nd=nd.next nd=nd.next
if (not nd) return if (nd==nil) return false
end end
local x1,y1,x2,y2=n0.ax,n0.ay,n1.ax,n1.ay local x1,y1,x2,y2=n0.ax,n0.ay,n1.ax,n1.ay
local function _line_line(x3,y3,x4,y4) local function _line_line(x3,y3,x4,y4)
local denom=(y4-y3)*(x2-x1)-(x4-x3)*(y2-y1) local denom=((y4-y3)*(x2-x1)-(x4-x3)*(y2-y1))
local ua= local ua=
((x4-x3)*(y1-y3)-(y4-y3)*(x1-x3))/denom ((x4-x3)*(y1-y3)-(y4-y3)*(x1-x3))/denom
if (ua<0 or ua>1) return if (ua<0 or ua>1) return false
local ub= local ub=
((x2-x1)*(y1-y3)-(y2-y1)*(x1-x3))/denom ((x2-x1)*(y1-y3)-(y2-y1)*(x1-x3))/denom
if (ub<0 or ub>1) return if (ub<0 or ub>1) return false
return true return true
end end
@ -1430,12 +1429,12 @@ function rope:tug_orientxy()
local a1=self.dst local a1=self.dst
local a0=self.dst.prev local a0=self.dst.prev
local dx=a0.ax-a1.ax local dx=a0.ax-a1.ax
local tdx local tdx=nil
if (dx>3/8) tdx=1 if (dx>3/8) tdx=1
if (dx<-3/8) tdx=-1 if (dx<-3/8) tdx=-1
local dy=a0.ay-a1.ay local dy=a0.ay-a1.ay
local tdy local tdy=nil
if abs(dy)>abs(dx)/2 then if abs(dy)>abs(dx)/2 then
if (dy>3/8) tdy=1 if (dy>3/8) tdy=1
if (dy<-3/8) tdy=-1 if (dy<-3/8) tdy=-1
@ -1476,11 +1475,11 @@ function rope:_tug(hypothetically)
end end
local latch=self.latch local latch=self.latch
if latch and latch.el=="eyehook" then if latch!=nil and latch.el=="eyehook" then
if (hypothetically) return ancs,0 if (hypothetically) return ancs,0
player.todo={{ player.todo={{
update=function(s) update=function(s)
if not s.rope or s.rope:done() then if s.rope==nil or s.rope:done() then
return true return true
end end
end end
@ -1488,7 +1487,7 @@ function rope:_tug(hypothetically)
self:destroy(true) self:destroy(true)
return true return true
end end
if latch and latch.el=="crate" then if latch!=nil and latch.el=="crate" then
local dmx,dmy= local dmx,dmy=
sgn0(latch.dx), sgn0(latch.dx),
sgn0(latch.dy) sgn0(latch.dy)
@ -1504,7 +1503,7 @@ function rope:_tug(hypothetically)
local mxa=(pull_anc.x+dmx)\8 local mxa=(pull_anc.x+dmx)\8
local mya=(pull_anc.y+dmy)\8 local mya=(pull_anc.y+dmy)\8
local invalid_move local invalid_move=false
if if
(dmx!=0 and sgn0(pull_dx)!=dmx) or (dmx!=0 and sgn0(pull_dx)!=dmx) or
(dmy!=0 and sgn0(pull_dy)!=dmy) or (dmy!=0 and sgn0(pull_dy)!=dmy) or
@ -1528,15 +1527,15 @@ function rope:_tug(hypothetically)
end end
end end
if (hypothetically) return ancs if (hypothetically) return ancs,nil
return return false
end end
function rope:_calc_push( function rope:_calc_push(
an,a0,a1,af an,a0,a1,af
) )
local ops={} local ops={}
if (not an) return ops if (an==nil) return ops
if a0.x==a1.x then if a0.x==a1.x then
-- no far side applying pressure? -- no far side applying pressure?
@ -1598,12 +1597,12 @@ function rope:_calc_push(
-- great! -- great!
else else
local crate=level:get_crate(o.mx,o.my) local crate=level:get_crate(o.mx,o.my)
if crate then if crate==nil then
break
else
if not level:can_move(false,o.mx,o.my,o.dmx,o.dmy,0,0) then if not level:can_move(false,o.mx,o.my,o.dmx,o.dmy,0,0) then
break break
end end
else
break
end end
add(ops2,o) add(ops2,o)
end end
@ -1618,23 +1617,23 @@ function rope:_anchors_simplified()
return atan2(p1.y-p0.y,p1.x-p0.x) return atan2(p1.y-p0.y,p1.x-p0.x)
end end
a=self.src a=self.src
while a do while a!=nil do
local point={ local point={
x=flr(a.ax*8+0.5),y=flr(a.ay*8+0.5), x=flr(a.ax*8+0.5),y=flr(a.ay*8+0.5),
ax=a.ax,ay=a.ay ax=a.ax,ay=a.ay
} }
local aw=level:anchor_at(a.ax,a.ay) local aw=a.associated_with
local l=self.latch local l=self.latch
if aw then if aw then
if (aw.adx==1) point.x-=1 if (aw.adx==1) point.x-=1
if (aw.ady==1) point.y-=1 if (aw.ady==1) point.y-=1
elseif not a.prev and l then elseif a.prev==nil and l then
if (l.ax_offset<0) point.x-=1 if (l.ax_offset<0) point.x-=1
if (l.ay_offset<0) point.y-=1 if (l.ay_offset<0) point.y-=1
end end
local p0,p1=points[#points-1],points[#points] local p0,p1=points[#points-1],points[#points]
if not p0 then if p0==nil then
add(points,point) add(points,point)
elseif _slope(p0,p1)==_slope(p1,point) then -- epsilon? elseif _slope(p0,p1)==_slope(p1,point) then -- epsilon?
points[#points]=point points[#points]=point
@ -1973,22 +1972,22 @@ __map__
0d0d0d0d0d0d0d0d0d080d0d0d0d0d0d0d0d230024242425141300000000150d0d0d134f1e00151414141400140d0d0d0d0d0d0d0d13000006240024070000150d2324240024242513000040441500140d0d0d0d0d0d0d0d0d080d0d0d0d0d0d0100000000000000062424242514140d0100001d1d483f230024242424242425 0d0d0d0d0d0d0d0d0d080d0d0d0d0d0d0d0d230024242425141300000000150d0d0d134f1e00151414141400140d0d0d0d0d0d0d0d13000006240024070000150d2324240024242513000040441500140d0d0d0d0d0d0d0d0d080d0d0d0d0d0d0100000000000000062424242514140d0100001d1d483f230024242424242425
0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d1400140d0d0d0d2324242424250d0d0d13000000000000000000140d0d0d0d0d0d0d0d23242425140014232424250d0d0d1400140d0d13000000001500120d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d1423242424243c250d0d0d0d0d0d0d0d14131f1f00000000140d0d0d0d0d0d 0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d1400140d0d0d0d2324242424250d0d0d13000000000000000000140d0d0d0d0d0d0d0d23242425140014232424250d0d0d1400140d0d13000000001500120d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d1423242424243c250d0d0d0d0d0d0d0d14131f1f00000000140d0d0d0d0d0d
0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d120d0d0d0d0d0d0d0d0d0d0d0d0d0d23242424251414141414140d0d0d0d0d0d0d0d0d0d0d0d0d010d0d0d0d0d0d0d0d0d010d0d0d232424242425140d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d23242424251414140d0d0d0d0d0d 0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d120d0d0d0d0d0d0d0d0d0d0d0d0d0d23242424251414141414140d0d0d0d0d0d0d0d0d0d0d0d0d010d0d0d0d0d0d0d0d0d010d0d0d232424242425140d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d23242424251414140d0d0d0d0d0d
0d0d0d0d0d0d0d0d030404040404050d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d010d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d030404040404050d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d000000000000000000000000000000000d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d03040404040405133d00001c00150d0d0d0d0d0d0d0d0d0304043e0404050d0d0d0d0d030404000404050d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d13001c00003f15030404040404050d0d0304043e0404050d0d0d0d0d0d0d0d000000000000000000000000000000000d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d130000000000151300001d1f00150d0d0d0d0d0d0d0d0d150000000000150d0d0d0d0317420000004816050d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d13001f1d000015130000000000150d0d130000000000150d0d0d0d0d0d0d0d000000000000000000000000000000000d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d13001d1d1e4f000000001c000000010d03040404040405150000000000150d0d0d141300191b1d191b0015140d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 010000001c000000004f1e1d1d00150d0d13000000000015030404040404050d000000000000000000000000000000000d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d13001f1f000015131e1e1f1d1d150d0d13460044004c15150000000000150d0d3d001e0042001f0048001e003f0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d131d1d1f1e1e151300001f1f00150d0d1300000000001513460044004c150d000000000000000000000000000000000d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d13000000000015130000001c1c150d0d131d1d1e1d1d152324241d2424250d0d140023240700410006242500140d0d0d0d0d0d030404040404050d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d131c1c00000015130000000000150d0d2324241d242425131d1d1e1d1d150d000000000000000000000000000000000d0d0d0d030404040404050d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0927271b1e1e0015130014001c1c150d0d131c3d001c1c000000001f040404050d1400000023243c2425000000140d0d0d0d0d0d130000001c000000000d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d131c1c0014001513001e1e1927270b030404041f000000001c1c003f1c150d000000000000000000000000000000000d0d0d0d130000001c000000000d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d131c1c00000000000000001c1c150d0d131f1f1e1f1f1514141300280000150d14000000001f1f1f00000000140d0d0d141414130028001c4516050014140d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d131c1c00000000000000001c1c150d1300000000151414131f1f1e1f1f150d000000000000000000000000000000000d141414130028001c4516050014140d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d131c1c0014001513001e1e1927270b0d1300004100001514141300000000010d144f47411400000014000000140d0d01000000000000001f0000150000001200000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0927271b1e1e0015130014001c1c150d0100000000151414130000410000150d0000000000000000000000000000000001000000000000001f0000150000001200000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d131c1c00000015130000000000150d0927271b001927270b031741280000150d14434b491400000014000000140d0d0d14141413002800000000151414140d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d13000000000015130000001c1c150d130000284116050927271b001927270b000000000000000000000000000000000d14141413002800000000151414140d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d131f1f1d1e1e151300001d1d00150d0d131f1c001c1c151f130000004900150d03040404040000000304040405140d0d0d0d0d13000000000006250d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d13001d1d000015131e1e1d1f1f150d130043000000151f131c1c001c1f150d000000000000000000000000000000000d0d0d0d13000000000006250d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
120000001c000000004f1e1f1f00150d0d13001f001f1f0000131d1d000000150d1300000000000003170000000000120d0d0d0d232424242424250d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d13001f1f1e4f000000001c00000012130000001d1d1500001f1f001f00150d000000000000000000000000000000000d0d0d0d232424242424250d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d13001d1f000015130000000000150d0d1300000000001500232424242424250d13000000001d1d1d0000000015140d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d130000000000151300001f1d00150d2324242424242500130000000000150d000000000000000000000000000000000d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d13001c00003f15232424242424250d0d1300000000001500140d0d0d0d0d0d0d130000000625031700000000150d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d23242424242425133d00001c00150d0d0d0d0d0d0d1400130000000000150d000000000000000000000000000000000d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d232424242424250d0d0d0d0d0d0d0d0d2324242424242500140d0d0d0d0d0d0d130000062503170000000000150d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d0d0d0d0d0d0d0d232424242424250d0d0d0d0d0d0d1400232424242424250d000000000000000000000000000000000d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d120d0d0d0d0d0d0d0d232424250d23242424242424250d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d120d0d0d0d0d0d0d0d000000000000000000000000000000000d0d0d0d0d0d0d0d0d0d0d0d0d0d0d0d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
__sfx__ __sfx__
00110000250002500025030250302503000000230350000023030230302303023030230302303023030230302103021030210302103021030210300000000000177401774017740177311e7501e7501e7501e750 00110000250002500025030250302503000000230350000023030230302303023030230302303023030230302103021030210302103021030210300000000000177401774017740177311e7501e7501e7501e750
00110000290202a0212a0202a0202a0202a02025030250052503025030250302503025030250302503025030250302503025030250302503225030230322504526030260302a040000002d0402d0402d0402d040 00110000290202a0212a0202a0202a0202a02025030250052503025030250302503025030250302503025030250302503025030250302503225030230322504526030260302a040000002d0402d0402d0402d040