2017-10-21 15:40:52 +02:00
-- SOKOBAN GAME, by rnd, robots port
if not sokoban then
sokoban = { } ;
2017-12-18 11:19:05 +01:00
local players = find_player ( 5 ) ;
2017-10-21 15:40:52 +02:00
if not players then error ( " sokoban: no player near " ) end
name = players [ 1 ] ;
self.show_form ( name ,
" size[2,1.25] " ..
" label[0,0;SELECT LEVEL 1-90] " ..
" field[0.25,1;1,1;LVL;LEVEL;1] " ..
" button_exit[1.25,0.75;1,1;OK;OK] "
)
state = 1 -- will wait for form receive otherwise game play
player_ = puzzle.get_player ( name ) ; -- get player entity - player must be present in area
player_ : set_eye_offset ( { x = 0 , y = 0 , z = 0 } , { x = 0 , y = 0 , z = 0 } ) ; player_ : set_physics_override ( { jump = 1 } ) -- reset player
self.spam ( 1 )
sokoban.push_time = 0
sokoban.blocks = 0 ; sokoban.level = 0 ; sokoban.moves = 0 ;
imax = 0 ; jmax = 0
sokoban.load = 0 ; sokoban.playername = " " ; sokoban.pos = { } ;
SOKOBAN_WALL = " default:brick "
SOKOBAN_FLOOR = " default:stonebrick "
SOKOBAN_GOAL = " default:diamondblock "
SOKOBAN_BOX = " basic_robot:button8080FF "
load_level = function ( lvl )
local pos = self.spawnpos ( ) ; pos.x = pos.x + 1 ; pos.y = pos.y + 1 ;
sokoban.pos = pos ;
sokoban.playername = name
if lvl == nil then return end
if lvl < 0 or lvl > 89 then return end
2017-12-18 11:19:05 +01:00
local file = _G.io . open ( minetest.get_modpath ( " basic_robot " ) .. " \\ scripts \\ sokoban.txt " , " r " )
2017-10-21 15:40:52 +02:00
if not file then return end
local str = " " ; local s ; local p = { x = pos.x , y = pos.y , z = pos.z } ; local i , j ; i = 0 ;
local lvl_found = false
while str ~= nil do
str = file : read ( " *line " ) ;
if str ~= nil and str == " ; " .. lvl then lvl_found = true break end
end
if not lvl_found then file : close ( ) ; return end
sokoban.blocks = 0 ; sokoban.level = lvl + 1 ; sokoban.moves = 0 ;
imax = 0 ; jmax = 0 ;
while str ~= nil do
str = file : read ( " *line " ) ;
if str ~= nil then
if string.sub ( str , 1 , 1 ) == " ; " then
imax = i ;
file : close ( ) ;
player_ : set_physics_override ( { jump = 0 } )
player_ : set_eye_offset ( { x = 0 , y = 20 , z = 0 } , { x = 0 , y = 0 , z = 0 } )
say ( " games: sokoban level " .. sokoban.level .. " loaded by " .. name .. " . It has " .. sokoban.blocks .. " boxes to push. " ) ; return
end
i = i + 1 ;
if string.len ( str ) > jmax then jmax = string.len ( str ) end -- determine max dimensions
for j = 1 , string.len ( str ) do
p.x = pos.x + i ; p.y = pos.y ; p.z = pos.z + j ; s = string.sub ( str , j , j ) ;
p.y = p.y - 1 ;
if puzzle.get_node ( p ) . name ~= SOKOBAN_FLOOR then puzzle.set_node ( p , { name = SOKOBAN_FLOOR } ) ; end -- clear floor
p.y = p.y + 1 ;
if s == " " and puzzle.get_node ( p ) . name ~= " air " then puzzle.set_node ( p , { name = " air " } ) end
if s == " # " and puzzle.get_node ( p ) . name ~= SOKOBAN_WALL then puzzle.set_node ( p , { name = SOKOBAN_WALL } ) end
if s == " $ " then puzzle.set_node ( p , { name = SOKOBAN_BOX } ) ; sokoban.blocks = sokoban.blocks + 1 end
if s == " . " then p.y = p.y - 1 ; puzzle.set_node ( p , { name = SOKOBAN_GOAL } ) ; p.y = p.y + 1 ; puzzle.set_node ( p , { name = " air " } ) end
--starting position
if s == " @ " then
player_ : setpos ( { x = p.x , y = p.y + 1 , z = p.z } ) ; -- move player to start position
--p.y=p.y-1;puzzle.set_node(p,{name="default:glass"});
puzzle.set_node ( p , { name = " air " } )
p.y = p.y + 1 ; puzzle.set_node ( p , { name = " air " } )
--p.y=p.y+2;puzzle.set_node(p,{name="default:ladder"})
end
if s ~= " @ " then p.y = pos.y + 2 ; puzzle.set_node ( p , { name = " air " } ) ; -- ceiling default:obsidian_glass
else --p.y=pos.y+2;puzzle.set_node(p,{name="default:ladder"})
end -- roof above to block jumps
end
end
end
file : close ( ) ;
end
end
if state == 1 then
sender , fields = self.read_form ( ) ; -- get fields from form submittal
if sender then
--say(serialize(fields))
if fields.LVL then
load_level ( ( tonumber ( fields.LVL ) or 1 ) - 1 )
state = 0
self.label ( " stand close to blue box and punch it one time to push it. you can only push 1 box \n and cant pull. goal is to get all boxes pushed on diamond blocks " )
end
end
else
local ppos = player_ : getpos ( )
if math.abs ( ppos.y - sokoban.pos . y ) ~= 0.5 then say ( colorize ( " red " , " SOKOBAN: " .. name .. " CHEATS ! " ) ) ; self.remove ( ) end
event = keyboard.get ( ) ;
if event then
local pname = event.puncher
if pname ~= name then goto quit end
local pos = { x = event.x , y = event.y , z = event.z } ;
local p = player.getpos ( pname ) ; local q = { x = pos.x , y = pos.y , z = pos.z }
p.x = p.x - q.x ; p.y = p.y - q.y ; p.z = p.z - q.z
if math.abs ( p.y + 0.5 ) > 0 then goto quit end
if math.abs ( p.x ) > math.abs ( p.z ) then -- determine push direction
if p.z <- 0.5 or p.z > 0.5 or math.abs ( p.x ) > 1.5 then goto quit end
if p.x + q.x > q.x then q.x = q.x - 1
else q.x = q.x + 1
end
else
if p.x <- 0.5 or p.x > 0.5 or math.abs ( p.z ) > 1.5 then goto quit end
if p.z + q.z > q.z then q.z = q.z - 1
else q.z = q.z + 1
end
end
if minetest.get_node ( q ) . name == " air " then -- push crate
sokoban.moves = sokoban.moves + 1
local old_infotext = minetest.get_meta ( pos ) : get_string ( " infotext " ) ;
minetest.set_node ( pos , { name = " air " } )
minetest.set_node ( q , { name = SOKOBAN_BOX } )
minetest.sound_play ( " default_dig_dig_immediate " , { pos = q , gain = 1.0 , max_hear_distance = 24 , } ) -- sound of pushing
local meta = minetest.get_meta ( q ) ;
q.y = q.y - 1 ;
if minetest.get_node ( q ) . name == SOKOBAN_GOAL then
if old_infotext ~= " GOAL REACHED " then
sokoban.blocks = sokoban.blocks - 1 ;
end
meta : set_string ( " infotext " , " GOAL REACHED " )
else
if old_infotext == " GOAL REACHED " then
sokoban.blocks = sokoban.blocks + 1
end
--meta:set_string("infotext", "push crate on top of goal block")
end
end
if sokoban.blocks ~= 0 then -- how many blocks left
--say("move " .. sokoban.moves .. " : " ..sokoban.blocks .. " crates left ");
else
say ( " games: " .. name .. " just solved sokoban level " .. sokoban.level .. " in " .. sokoban.moves .. " moves. " ) ;
player_ : set_physics_override ( { jump = 1 } )
player_ : set_eye_offset ( { x = 0 , y = 0 , z = 0 } , { x = 0 , y = 0 , z = 0 } )
local i , j ;
for i = 1 , imax do
for j = 1 , jmax do
minetest.set_node ( { x = sokoban.pos . x + i , y = sokoban.pos . y , z = sokoban.pos . z + j } , { name = " air " } ) ; -- clear level
end
end
sokoban.playername = " " ; sokoban.level = 1
end
:: quit ::
end
end