123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347 |
- -- Hangglider mod for Minetest
- -- Original code by Piezo_ (orderofthefourthwall@gmail.com)
- -- 2018-11-14
- -- Modifications by David G (kestral246@gmail.com)
- -- 2018-11-24
- -- For Minetest 5.x, glider's set_attach needs to be offset by 1 node
- -- Switch to alternate commented line below with correct offset.
- -- Additional tuning of parameters.
- -- Commented out debug hud display code, prefixed with "--debug:".
- -- 2018-11-22
- -- Give visual indication that hangglider is equiped.
- -- Display simple overlay with blurred struts when equiped.
- -- Issue: don't know how to disable overlay in third person view.
- -- Also Unequip hangglider when landing on water.
- -- Attempt to linearize parabolic flight path.
- -- Start gravity stronger, but gradually reduce it as descent velocity increases.
- -- Don't use airstopper when equipped from the ground (descent velocity is low).
- -- Slightly increase flight speed to 1.25.
- -- Unequip/equip cycling mid-flight should not fly farther than continuous flight.
- -- When equipping mid-air (descent velocity higher), use airstopper but increase descent slope afterwards.
- -- Create airbreak flag so all equips mid-flight use faster descent.
- -- Reset airbreak flag only when land (canExist goes false).
- -- Issue: it wouldn't reset if land in water, use fly, and launch from air, before I added test for water,
- -- not sure if there are other such cases.
- -- Temporarily add hud debug display to show descent velocity, gravity override, and airbreak flag.
- -- Still in process of tuning all the parameters.
- -- Modifications by Piezo_
- -- 2018-11-25
- -- hud overlay and debug can be enabled/disabled
- -- Added blender-rendered overlay for struts using the actual model.
- -- Reduced airbreak penalty severity
- -- gave glider limited durability.
- -- Improved gravity adjustment function.
- -- Changed airbreaking process
- -- Removed airbreak penalty, as any 'advantage' seems minimal after new adjustments
- -- Removed airbreak until minetest devs are smart enough to implement better serverside players.
- -- Simplified liquid check.
- -- Modifications by gpcf
- -- 2018-12-09
- -- get shot down while flying over protected areas marked as no-fly-zones (flak, from German Flugabwehrkanone)
- -- set these areas with the /area_flak command
- -- Modifications by SpaghettiToastBook
- -- 2018-12-29
- -- Physics overrides use player_monoids mod if available
- local HUD_Overlay = true --show glider struts as overlay on HUD
- local debug = false --show debug info in top-center of hud
- local moveModelUp = false
- if tonumber(string.sub(minetest.get_version().string, 1, 1)) and tonumber(string.sub(minetest.get_version().string, 1, 1)) > 4 then
- moveModelUp = true
- end
- hangglider = {} --Make this global, so other mods can tell if hangglider exists.
- hangglider.use = {}
- if HUD_Overlay then
- hangglider.id = {} -- hud id for displaying overlay with struts
- end
- hangglider.players = {}
- if debug then hangglider.debug = {} end -- hud id for debug data
- --hangglider.airbreak = {} -- true if falling fast when equip
- --[[minetest.register_entity("hangglider:airstopper", { --A one-instant entity that catches the player and stops them.
- is_visible = false,
- physical = false,
- immortal = true,
- attach = nil,
- on_step = function(self, _)
- local canExist = false
- if self.attach then
- local player = self.attach
- if player:is_player() then
- local pname = player:get_player_name()
- canExist = true
- if player:get_player_velocity().y < 0.5 and player:get_player_velocity().y > -0.5 then
- --Let go when the player actually stops, as that's the whole point.
- if hangglider.use[pname] then
- if moveModelUp then
- minetest.add_entity(player:get_pos(), "hangglider:glider"):set_attach(player, "", {x=0,y=10,z=0}, {x=0,y=0,z=0})
- else
- minetest.add_entity(player:get_pos(), "hangglider:glider"):set_attach(player, "", {x=0,y=0,z=0}, {x=0,y=0,z=0})
- end
- end
- canExist = false
- end
- end
- if not canExist then
- player:set_detach()
- end
- end
- if not canExist then
- self.object:remove()
- end
- end
- })
- if core.global_exists("areas") then
- hangglider.flak = false
- -- chat command definition essentially copied from areas mod.
- minetest.register_chatcommand("area_flak",{
- params = "<ID>",
- description = "Toggle airspace restrictions for area <ID>",
- func = function(name, param)
- local id = tonumber(param)
- if not id then
- return false, "Invalid usage, see /help area_flak."
- end
- if not areas:isAreaOwner(id, name) then
- return false, "Area "..id.." does not exist"
- .." or is not owned by you."
- end
- local open = not areas.areas[id].flak
- -- Save false as nil to avoid inflating the DB.
- areas.areas[id].flak = open or nil
- areas:save()
- return true, ("Area's airspace %s."):format(open and "closed" or "opened")
- end
- })
- end
- if core.global_exists("minetestd") and minetestd.services.physicsctl.enabled then
- minetestd.physicsctl.register_physics_effect("hangglider",
- function(player) -- check
- return hangglider.use[player:get_player_name()]
- end,
- function(phys, player) -- blend
- local vel_y = player:get_player_velocity().y
- if debug then player:hud_change(hangglider.debug[pname].id, "text", vel_y..', '..player:get_physics_override().gravity..', '..tostring(hangglider.airbreak[pname])) end
- phys.gravity = phys.gravity*((vel_y + 3)/20)
- if vel_y < 0 and vel_y > -3 then
- phys.speed = (math.abs(vel_y/2) + 0.75)
- elseif vel_y <= -3 then --Cap our gliding movement speed.
- phys.speed = 2.25
- end
- phys.jump = 0
- end,
- 7 -- effect order
- )
- end
- hangglider.shot_sound = function (pos)
- minetest.sound_play("hangglider_flak_shot", {
- pos = pos,
- max_hear_distance = 30,
- gain = 10.0,
- })
- end
- ]]
- local physics_attrs = {"jump", "speed", "gravity"}
- local function apply_physics_override(player, overrides)
- if player_monoids then
- for _, attr in pairs(physics_attrs) do
- if overrides[attr] then
- player_monoids[attr]:add_change(player, overrides[attr], "hangglider:glider")
- end
- end
- else
- player:set_physics_override(overrides)
- end
- end
- local function remove_physics_override(player, overrides)
- for _, attr in pairs(physics_attrs) do
- if overrides[attr] then
- if core.global_exists("player_monoids") then
- player_monoids[attr]:del_change(player, "hangglider:glider")
- else
- player:set_physics_override({[attr] = 1})
- end
- end
- end
- end
- local step_v
- minetest.register_entity("hangglider:glider", {
- visual = "mesh",
- visual_size = {x = 12, y = 12},
- collisionbox = {0,0,0,0,0,0},
- mesh = "glider.obj",
- immortal = true,
- static_save = false,
- textures = {"wool.png","default_wood.png"},
- on_step = function(self, dtime)
- local canExist = false
- if self.object:get_attach() then
- local player = self.object:get_attach("parent")
- if player then
- local pos = player:getpos()
- local pname = player:get_player_name()
- if hangglider.use[pname] then
- local mrn_name = minetest.registered_nodes[minetest.get_node(vector.new(pos.x, pos.y-0.5, pos.z)).name]
- if mrn_name then
- if not (mrn_name.walkable or mrn_name.liquidtype ~= "none") then
- canExist = true
- if not minetestd then
- step_v = player:get_player_velocity().y
- if step_v < 0 and step_v > -3 then
- apply_physics_override(player, {speed=math.abs(step_v/2) + 0.75})
- elseif step_v <= -3 then --Cap our gliding movement speed.
- apply_physics_override(player, {speed=2.25})
- else
- remove_physics_override(player, {speed=1})
- end
- if debug then player:hud_change(hangglider.debug[pname].id, "text", step_v..', '..player:get_physics_override().gravity..', '..tostring(hangglider.airbreak[pname])) end
- apply_physics_override(player, {gravity=((step_v + 3)/20)})
- end
- --[[local vel = player:get_player_velocity()
- if debug then player:hud_change(hangglider.debug[pname].id, "text", vel.y..', '..grav..', '..tostring(hangglider.airbreak[pname])) end
- player:set_physics_override({gravity = (vel.y + 2.0)/20})
- ]]end
- end
- end
- if not canExist then
- if not minetestd then
- local player_physics = hangglider.players[pname]
- local player_gravity = player_physics.gravity
- local player_jump = player_physics.jump
- local player_speed = player_physics.speed
- if player_gravity < .9 then
- player:set_physics_override({gravity=1})
- end
- if player_jump < .8 then
- player:set_physics_override({jump=1})
- end
- if player_speed < .9 or player_speed > 2 then
- player:set_physics_override({speed=1})
- end
- if player_jump > .7 and player_gravity > .8 then
- player:set_physics_override(hangglider.players[pname])
- end
- end
- hangglider.use[pname] = false
- if HUD_Overlay then
- player:hud_change(hangglider.id[pname], "text", "blank.png")
- end
- --hangglider.airbreak[pname] = false
- end
- end
- end
- if not canExist then
- self.object:set_detach()
- self.object:remove()
- end
- end
- })
- minetest.register_on_dieplayer(function(player)
- remove_physics_override(player, {
- gravity = 1,
- jump = 1,
- })
- hangglider.use[player:get_player_name()] = false
- end)
- minetest.register_on_joinplayer(function(player)
- local pname = player:get_player_name()
- remove_physics_override(player, {
- gravity = 1,
- jump = 1,
- })
- hangglider.use[pname] = false
- if HUD_Overlay then
- hangglider.id[pname] = player:hud_add({
- hud_elem_type = "image",
- text = "blank.png",
- position = {x=0, y=0},
- scale = {x=-100, y=-100},
- alignment = {x=1, y=1},
- offset = {x=0, y=0}
- }) end
- if debug then
- hangglider.debug[pname] = {id = player:hud_add({hud_elem_type = "text",
- position = {x=0.5, y=0.1},
- text = "-",
- number = 0xFF0000}), -- red text
- -- ht = {50,50,50},
- }
- end
- --hangglider.airbreak[pname] = false
- end)
- minetest.register_on_leaveplayer(function(player)
- local pname = player:get_player_name()
- hangglider.use[pname] = nil
- if HUD_Overlay then hangglider.id[pname] = nil end
- if debug then hangglider.debug[pname] = nil end
- --hangglider.airbreak[pname] = nil
- end)
- minetest.register_tool("hangglider:hangglider", {
- description = "Glider",
- inventory_image = "glider_item.png",
- stack_max=1,
- on_use = function(itemstack, player, pointed_thing)
- if not player then
- return
- end
- local pos = player:get_pos()
- local pname = player:get_player_name()
- hangglider.players[pname] = player:get_physics_override()
- if not hangglider.use[pname] then --Equip
- minetest.sound_play("bedsheet", {pos=pos, max_hear_distance = 8, gain = 1.0})
- if HUD_Overlay then player:hud_change(hangglider.id[pname], "text", "glider_struts.png") end
- local airbreak = false
- local vel = player:get_player_velocity().y
- --[[if vel < -1.5 then -- engage mid-air, falling fast, so stop but ramp velocity more quickly
- --hangglider.airbreak[pname] = true
- airbreak = true
- local stopper = minetest.add_entity(pos, "hangglider:airstopper")
- minetest.after(0, function(stopper, player) --"Extreme Measures"
- stopper:set_pos(player:get_pos())
- stopper:get_luaentity().attach = player
- player:set_attach( stopper, "", {x=0,y=0,z=0}, {x=0,y=0,z=0})
- end, stopper, player)
- end]]
- if not airbreak then
- if moveModelUp then
- minetest.add_entity(pos, "hangglider:glider"):set_attach(player, "", {x=0,y=10,z=0}, {x=0,y=0,z=0})
- else
- minetest.add_entity(pos, "hangglider:glider"):set_attach(player, "", {x=0,y=0,z=0}, {x=0,y=0,z=0})
- end
- end
- hangglider.use[pname] = true
- apply_physics_override(player, {jump = 0})
- -- if minetest 0.4.x use this:
- -- if minetest 5.x use this:
- -- minetest.add_entity(player:get_pos(), "hangglider:glider"):set_attach(player, "", {x=0,y=10,z=0}, {x=0,y=0,z=0})
- itemstack:set_wear(itemstack:get_wear() + 255)
- return itemstack
- elseif hangglider.use[pname] then --Unequip
- if HUD_Overlay then player:hud_change(hangglider.id[pname], "text", "default_wood.png^[colorize:#0000:255") end
- hangglider.use[pname] = false
- end
- end,
- sound = {breaks = "default_tool_breaks"},
- })
|