diff options
Diffstat (limited to 'data/scripts/objects')
-rw-r--r-- | data/scripts/objects/test/client.lua | 137 | ||||
-rw-r--r-- | data/scripts/objects/test/server.lua | 322 |
2 files changed, 459 insertions, 0 deletions
diff --git a/data/scripts/objects/test/client.lua b/data/scripts/objects/test/client.lua new file mode 100644 index 000000000..a685721a1 --- /dev/null +++ b/data/scripts/objects/test/client.lua @@ -0,0 +1,137 @@ +-- Client-side code of the test lua object + +-- +-- Some helper functions and classes +-- + +function split(str, pat) + local t = {} -- NOTE: use {n = 0} in Lua-5.0 + local fpat = "(.-)" .. pat + local last_end = 1 + local s, e, cap = str:find(fpat, 1) + while s do + if s ~= 1 or cap ~= "" then + table.insert(t,cap) + end + last_end = e+1 + s, e, cap = str:find(fpat, last_end) + end + if last_end <= #str then + cap = str:sub(last_end) + table.insert(t, cap) + end + return t +end + +-- For debugging +function dump(o) + if type(o) == 'table' then + local s = '{ ' + for k,v in pairs(o) do + if type(k) ~= 'number' then k = '"'..k..'"' end + s = s .. '['..k..'] = ' .. dump(v) .. ',' + end + return s .. '} ' + else + return tostring(o) + end +end + +function vector_subtract(a, b) + return {X=a.X-b.X, Y=a.Y-b.Y, Z=a.Z-b.Z} +end + +function vector_add(a, b) + return {X=a.X+b.X, Y=a.Y+b.Y, Z=a.Z+b.Z} +end + +function vector_multiply(a, d) + return {X=a.X*d, Y=a.Y*d, Z=a.Z*d} +end + +SmoothTranslator = {} +SmoothTranslator.__index = SmoothTranslator + +function SmoothTranslator.create() + local obj = {} + setmetatable(obj, SmoothTranslator) + obj.vect_old = {X=0, Y=0, Z=0} + obj.anim_counter = 0 + obj.anim_time = 0 + obj.anim_time_counter = 0 + obj.vect_show = {X=0, Y=0, Z=0} + obj.vect_aim = {X=0, Y=0, Z=0} + return obj +end + +function SmoothTranslator:update(vect_new) + self.vect_old = self.vect_show + self.vect_aim = vect_new + if self.anim_time < 0.001 or self.anim_time > 1.0 then + self.anim_time = self.anim_time_counter + else + self.anim_time = self.anim_time * 0.9 + self.anim_time_counter * 0.1 + end + self.anim_time_counter = 0 + self.anim_counter = 0 +end + +function SmoothTranslator:translate(dtime) + self.anim_time_counter = self.anim_time_counter + dtime + self.anim_counter = self.anim_counter + dtime + vect_move = vector_subtract(self.vect_aim, self.vect_old) + moveratio = 1.0 + if self.anim_time > 0.001 then + moveratio = self.anim_time_counter / self.anim_time + end + -- Move a bit less than should, to avoid oscillation + moveratio = moveratio * 0.8 + if moveratio > 1.5 then + moveratio = 1.5 + end + self.vect_show = vector_add(self.vect_old, vector_multiply(vect_move, moveratio)) +end + +-- +-- Actual code +-- + +pos_trans = SmoothTranslator.create() +rot_trans = SmoothTranslator.create() + +-- Callback functions + +function on_step(self, dtime) + pos_trans:translate(dtime) + rot_trans:translate(dtime) + object_set_position(self, pos_trans.vect_show) + object_set_rotation(self, rot_trans.vect_show) +end + +function on_process_message(self, data) + --print("client got message: " .. data) + + -- Receive our custom messages + + sp = split(data, " ") + if sp[1] == "pos" then + pos_trans:update({X=sp[2], Y=sp[3], Z=sp[4]}) + end + if sp[1] == "rot" then + rot_trans:update({X=sp[2], Y=sp[3], Z=sp[4]}) + end +end + +function on_initialize(self, data) + print("client object got initialization: " .. data) + + corners = { + {-1/2,-1/2, 0}, + { 1/2,-1/2, 0}, + { 1/2, 0, 0}, + {-1/2, 0, 0}, + } + object_add_to_mesh(self, "rat.png", corners, false) + +end + diff --git a/data/scripts/objects/test/server.lua b/data/scripts/objects/test/server.lua new file mode 100644 index 000000000..1213e2fe6 --- /dev/null +++ b/data/scripts/objects/test/server.lua @@ -0,0 +1,322 @@ +-- Server-side code of the test lua object + +-- +-- Some helper functions and classes +-- + +-- For debugging +function dump(o) + if type(o) == 'table' then + local s = '{ ' + for k,v in pairs(o) do + if type(k) ~= 'number' then k = '"'..k..'"' end + s = s .. '['..k..'] = ' .. dump(v) .. ',' + end + return s .. '} ' + else + return tostring(o) + end +end + +function table.copy(t) + local t2 = {} + for k,v in pairs(t) do + t2[k] = v + end + return t2 +end + +function vector_zero() + return {X=0,Y=0,Z=0} +end + +function vector_subtract(a, b) + return {X=a.X-b.X, Y=a.Y-b.Y, Z=a.Z-b.Z} +end + +function vector_add(a, b) + return {X=a.X+b.X, Y=a.Y+b.Y, Z=a.Z+b.Z} +end + +function vector_multiply(a, d) + return {X=a.X*d, Y=a.Y*d, Z=a.Z*d} +end + +function vector_copy(a) + return {X=a.X, Y=a.Y, Z=a.Z} +end + +function vector_length(a) + return math.sqrt(a.X*a.X + a.Y*a.Y + a.Z*a.Z) +end + +function vector_eq(a, b) + return (a.X==b.X and a.Y==b.Y and a.Z==b.Z) +end + +function round(num, idp) + local mult = 10^(idp or 0) + return math.floor(num * mult + 0.5) / mult +end + +function vector_snap(a) + return {X=round(a.X, 0), Y=round(a.Y, 0), Z=round(a.Z, 0)} +end + +-- +-- Actual code +-- + +CONTENT_STONE = 0 + +is_digger = false +counter = 0 +counter2 = 0 +counter3 = 0 +counter4 = 0 +counter_move = 0 +death_counter = 0 +-- This is set in on_initialize() +position = {X=0,Y=0,Z=0} +starting_position = {X=0,Y=0,Z=0} +rotation = {X=0, Y=math.random(0,360), Z=0} +y_dir = 1 +temp1 = 0 +speed = 1.5 +main_dir = {X=0,Y=0,Z=0} + +function dir_goodness(env, pos, dir) + if vector_eq(dir, vector_zero()) then + return -1 + end + p = vector_add(pos, dir) + n = env_get_node(env, p) + f = get_content_features(n.content) + if f.walkable then + p.Y = p.Y + 1 + n = env_get_node(env, p) + f = get_content_features(n.content) + if f.walkable then + -- Too high + return -1 + end + -- Hill + return 2 + end + p.Y = p.Y - 1 + n = env_get_node(env, p) + f = get_content_features(n.content) + if f.walkable then + -- Flat + return 1 + end + -- Drop + return 0 +end + +function on_step(self, dtime) + -- Limit step to a sane value; it jumps a lot while the map generator + -- is in action + if dtime > 0.5 then + dtime = 0.5 + end + + env = object_get_environment(self) + + --[[ + -- Returned value has these fields: + -- * int content + -- * int param1 + -- * int param2 + p = {X=position.X, Y=position.Y-0.35, Z=position.Z} + n = env_get_node(env, p) + f = get_content_features(n.content) + if f.walkable then + y_dir = 1 + else + y_dir = -1 + end + -- Keep the object approximately at ground level + position.Y = position.Y + dtime * 2.0 * y_dir + + -- Move the object around + position.X = position.X + math.cos(math.pi+rotation.Y/180*math.pi) + * dtime * speed + position.Z = position.Z + math.sin(math.pi+rotation.Y/180*math.pi) + * dtime * speed + + -- Rotate the object if it is too far from the starting point + counter3 = counter3 - dtime + if counter3 < 0 then + counter3 = counter3 + 1 + diff = vector_subtract(position, starting_position) + d = vector_length(diff) + --print("pos="..dump(position).." starting="..dump(starting_position)) + --print("diff=" .. dump(diff)) + --print("d=" .. d) + if d > 3 then + rotation.Y = rotation.Y + 90 + --rotation.Y = rotation.Y + math.random(-180, 180) + end + end + + -- This value has to be set; it determines to which player the + -- object is near to and such + object_set_base_position(self, position) + + counter4 = counter4 - dtime + if counter4 < 0 then + --counter4 = counter4 + math.random(0.5,8) + counter4 = counter4 + 0.6/speed + -- Mess around with the map + if is_digger == true then + np = vector_add(position, {X=0,Y=-0.6,Z=0}) + env_dig_node(env, np) + else + np = vector_add(position, {X=0,Y=0,Z=0}) + env_place_node(env, np, {content=0}) + end + end + --]] + + counter_move = counter_move - dtime + if counter_move < 0 then + counter_move = counter_move + 1/speed + if counter_move < 0 then counter_move = 0 end + + old_position = vector_copy(position) + + dirs = { + {X=1, Y=0, Z=0}, + {X=-1, Y=0, Z=0}, + {X=0, Y=0, Z=1}, + {X=0, Y=0, Z=-1} + } + + best_dir = main_dir + best_goodness = dir_goodness(env, position, main_dir) + + for k,v in ipairs(dirs) do + -- Don't go directly backwards + if not vector_eq(vector_subtract(vector_zero(), v), main_dir) then + goodness = dir_goodness(env, position, v) + if goodness > best_goodness then + best_dir = v + goodness = best_goodness + end + end + end + + -- Place stone block when dir changed + if not vector_eq(main_dir, best_dir) then + np = vector_add(position, {X=0,Y=0,Z=0}) + env_place_node(env, np, {content=CONTENT_STONE}) + end + + main_dir = best_dir + + position = vector_add(position, main_dir) + + pos_diff = vector_subtract(position, old_position) + rotation.Y = math.atan2(pos_diff.Z, pos_diff.X)/math.pi*180-180 + + -- Returned value has these fields: + -- * int content + -- * int param1 + -- * int param2 + p = {X=position.X, Y=position.Y, Z=position.Z} + n = env_get_node(env, p) + f = get_content_features(n.content) + if f.walkable then + position.Y = position.Y + 1 + end + p = {X=position.X, Y=position.Y-1, Z=position.Z} + n = env_get_node(env, p) + f = get_content_features(n.content) + if not f.walkable then + position.Y = position.Y - 1 + end + + -- Center in the middle of the node + position = vector_snap(position) + end + + -- This value has to be set; it determines to which player the + -- object is near to and such + object_set_base_position(self, position) + + --[[ + counter4 = counter4 - dtime + if counter4 < 0 then + --counter4 = counter4 + math.random(0.5,8) + counter4 = counter4 + 0.6/speed + -- Mess around with the map + if is_digger == true then + np = vector_add(position, {X=0,Y=-0.6,Z=0}) + env_dig_node(env, np) + else + np = vector_add(position, {X=0,Y=0,Z=0}) + env_place_node(env, np, {content=0}) + end + end + --]] + + ---[[ + -- Send some custom messages at a custom interval + + counter = counter - dtime + if counter < 0 then + counter = counter + 0.25 + if counter < 0 then + counter = 0 + end + + message = "pos " .. position.X .. " " .. position.Y .. " " .. position.Z + object_add_message(self, message) + + message = "rot " .. rotation.X .. " " .. rotation.Y .. " " .. rotation.Z + object_add_message(self, message) + end + --]] + + -- Remove the object after some time + death_counter = death_counter + dtime + if death_counter > 40 then + object_remove(self) + end + +end + +-- This stuff is passed to a newly created client-side counterpart of this object +function on_get_client_init_data(self) + -- Just return some data for testing + return "result of get_client_init_data" +end + +-- This should return some data that mostly saves the state of this object +-- Not completely implemented yet +function on_get_server_init_data(self) + -- Just return some data for testing + return "result of get_server_init_data" +end + +-- When the object is loaded from scratch, this is called before the first +-- on_step(). Data is an empty string or the output of an object spawner +-- hook, but such things are not yet implemented. +-- +-- At reload time, the last output of get_server_init_data is passed as data. +-- +-- This should initialize the position of the object to the value returned +-- by object_get_base_position(self) +-- +-- Not completely implemented yet +-- +function on_initialize(self, data) + print("server object got initialization: " .. data) + position = object_get_base_position(self) + starting_position = vector_copy(position); + if math.random() < 0.5 then + is_digger = true + end +end + |