123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495 |
- --[[
- Minetest Mod - Simple Shooter [shooter] 0.5.3
- =======================================
- License Source Code: 2013 Stuart Jones - LGPL v2.1
- License Textures: Stuart Jones - WTFPL
- Licence Models: Stuart Jones - CC-BY-SA 3.0
- License Sounds: freesound.org
- --]]
- minetest.register_alias('crossbow', 'castle_weapons:crossbow')
- minetest.register_alias('bolt', 'castle_weapons:crossbow_bolt')
- minetest.register_alias('castle:crossbow', 'castle_weapons:crossbow')
- minetest.register_alias('castle:bolt', 'castle_weapons:crossbow_bolt')
- minetest.register_alias('castle:crossbow_bolt', 'castle_weapons:crossbow_bolt')
- minetest.register_alias('castle:crossbow_loaded', 'castle_weapons:crossbow_loaded')
- -- internationalization boilerplate
- local MP = minetest.get_modpath(minetest.get_current_modname())
- local S, NS = dofile(MP..'/intllib.lua')
- local crossbow={}
- CROSSBOW_USES = 300
- CROSSBOW_BOLT_TOOL_CAPS = {damage_groups={fleshy=4}}
- CROSSBOW_BOLT_LIFETIME = 60-- 1 minute
- CROSSBOW_ENABLE_PARTICLE_FX = false
- CROSSBOW_ENABLE_PROTECTION = true
- CROSSBOW_EXPLOSION_TEXTURE = 'castle_crossbow_hit.png'
- CROSSBOW_ALLOW_NODES = true
- CROSSBOW_ALLOW_ENTITIES = true
- CROSSBOW_ALLOW_PLAYERS = true
- CROSSBOW_PLAYER_OFFSET = {x=0, y=1, z=0}
- CROSSBOW_ENTITY_OFFSET = {x=0, y=0, z=0}
- CROSSBOW_ENTITIES = {
- 'zombies:1arm',
- 'zombies:crawler',
- 'zombies:normal',
- 'scorpion:big',
- 'scorpion:boss',
- 'scorpion:little',
- 'scorpion:pet',
- 'farm_mobs:goat_he',
- 'farm_mobs:goat_she',
- 'farm_mobs:dog',
- 'fantasy_mobs:fairy',
- 'fantasy_mobs:gnome',
- 'fantasy_mobs:goblin',
- 'fantasy_mobs:mummy',
- 'fantasy_mobs:larva',
- 'fantasy_mobs:larva_pet',
- 'fantasy_mobs:cavefreak_slash',
- 'fantasy_mobs:cavefreak_fire',
- 'fantasy_mobs:lava_titan',
- 'desert_life:ostrich',
- 'desert_life:armadillo',
- 'epic:ocean_guardian',
- 'arctic_life:penguin',
- 'arctic_life:walrus',
- 'desert_life:ostrich',
- 'desert_life:armadillo',
- 'mobs_animal:bunny',
- 'mobs_animal:chicken',
- 'mobs_animal:cow',
- 'mobs_animal:kitten',
- 'mobs_animal:panda',
- 'mobs_animal:sheep_white',
- 'mobs_animal:pumba',
- 'mob_horse:horse',
- 'mobs_monster:dirt_monster',
- 'mobs_monster:dungeon_master',
- 'mobs_monster:lava_flan',
- 'mobs_monster:mese_monster',
- 'mobs_monster:oerkki',
- 'mobs_monster:sand_monster',
- 'mobs_monster:spider',
- 'mobs_monster:stone_monster',
- 'mobs_monster:tree_monster',
- 'mobs_crocs:crocodile',
- 'mobs_crocs:crocodile_swim',
- 'mobs_crocs:crocodile_float',
- 'mobs_turtles:turtle',
- 'mobs_turtles:seaturtle',
- 'viron:viron_mob',
- 'viron:viron_queen',
- 'viron:viron_larve',
- }
- if minetest.is_singleplayer() == true then
- CROSSBOW_ALLOW_ENTITIES = true
- CROSSBOW_ALLOW_PLAYERS = true
- end
- local allowed_entities = {}
- for _,v in ipairs(CROSSBOW_ENTITIES) do
- allowed_entities[v] = 1
- end
- local function get_dot_product(v1, v2)
- return v1.x * v2.x + v1.y * v2.y + v1.z * v2.z
- end
- local function get_particle_pos(p, v, d)
- return vector.add(p, vector.multiply(v, {x=d, y=d, z=d}))
- end
- function crossbow:spawn_particles(pos, texture)
- if CROSSBOW_ENABLE_PARTICLE_FX == true then
- if type(texture) ~= 'string' then
- texture = CROSSBOW_EXPLOSION_TEXTURE
- end
- local spread = {x=0.1, y=0.1, z=0.1}
- minetest.add_particlespawner({
- amount = 15,
- time = 0.3,
- minpos = vector.subtract(pos, spread),
- maxpos = vector.add(pos, spread),
- minvel = {x=-1, y=1, z=-1},
- maxvel = {x=1, y=2, z=1},
- minacc = {x=-2, y=-2, z=-2},
- maxacc = {x=2, y=-2, z=2},
- minexptime = 0.1,
- maxexptime = 0.75,
- minsize = 1,
- maxsize = 2,
- collisiondetection = false,
- texture = texture,
- })
- end
- end
- function crossbow:punch_node(pos, def)
- local node = minetest.get_node(pos)
- if not node then
- return
- end
- local item = minetest.registered_items[node.name]
- if not item then
- return
- end
- if CROSSBOW_ENABLE_PROTECTION then
- if minetest.is_protected(pos, def.name) then
- return
- end
- end
- if item.groups then
- for k, v in pairs(def.groups) do
- local level = item.groups[k] or 0
- if level >= v then
- minetest.remove_node(pos)
- if item.tiles then
- if item.tiles[1] then
- crossbow:spawn_particles(pos, item.tiles[1])
- end
- end
- break
- end
- end
- end
- end
- function crossbow:is_valid_object(object)
- if object then
- if object:is_player() == true then
- return CROSSBOW_ALLOW_PLAYERS
- end
- if CROSSBOW_ALLOW_ENTITIES == true then
- local luaentity = object:get_luaentity()
- if luaentity then
- if luaentity.name then
- if allowed_entities[luaentity.name] then
- return true
- end
- end
- end
- end
- end
- end
- function crossbow:get_intersect_pos(ray, plane, collisionbox)
- local v = vector.subtract(ray.pos, plane.pos)
- local r1 = get_dot_product(v, plane.normal)
- local r2 = get_dot_product(ray.dir, plane.normal)
- if r2 ~= 0 then
- local t = -(r1 / r2)
- local td = vector.multiply(ray.dir, {x=t, y=t, z=t})
- local pt = vector.add(ray.pos, td)
- local pd = vector.subtract(pt, plane.pos)
- if math.abs(pd.x) < collisionbox[4] and
- math.abs(pd.y) < collisionbox[5] and
- math.abs(pd.z) < collisionbox[6] then
- return pt
- end
- end
- end
- function crossbow:process_round(round)
- local target = {object=nil, distance=10000}
- local p1 = round.pos
- local v1 = round.ray
- for _,ref in ipairs(castle.objects) do
- local p2 = vector.add(ref.pos, ref.offset)
- if p1 and p2 and ref.name ~= round.name then
- local d = vector.distance(p1, p2)
- if d < round.def.step and d < target.distance then
- local ray = {pos=p1, dir=v1}
- local plane = {pos=p2, normal={x=-1, y=0, z=-1}}
- local pos = crossbow:get_intersect_pos(ray, plane, ref.collisionbox)
- if pos then
- target.object = ref.object
- target.pos = pos
- target.distance = d
- end
- end
- end
- end
- if target.object and target.pos then
- local success, pos = minetest.line_of_sight(p1, target.pos, 1)
- if success then
- local user = minetest.get_player_by_name(round.name)
- if user then
- target.object:punch(user, nil, round.def.tool_caps, v1)
- crossbow:spawn_particles(target.pos, CROSSBOW_EXPLOSION_TEXTURE)
- end
- return 1
- elseif pos and CROSSBOW_ALLOW_NODES == true then
- crossbow:punch_node(pos, round.def)
- return 1
- end
- elseif CROSSBOW_ALLOW_NODES == true then
- local d = round.def.step
- local p2 = vector.add(p1, vector.multiply(v1, {x=d, y=d, z=d}))
- local success, pos = minetest.line_of_sight(p1, p2, 1)
- if pos then
- crossbow:punch_node(pos, round.def)
- return 1
- end
- end
- end
- local function get_animation_frame(dir)
- local angle = math.atan(dir.y)
- local frame = 90 - math.floor(angle * 360 / math.pi)
- if frame < 1 then
- frame = 1
- elseif frame > 180 then
- frame = 180
- end
- return frame
- end
- local function get_target_pos(p1, p2, dir, offset)
- local d = vector.distance(p1, p2) - offset
- local td = vector.multiply(dir, {x=d, y=d, z=d})
- return vector.add(p1, td)
- end
- local function punch_object(puncher, object)
- if puncher and crossbow:is_valid_object(object) then
- if puncher ~= object then
- local dir = puncher:get_look_dir()
- local p1 = puncher:getpos()
- local p2 = object:getpos()
- local tpos = get_target_pos(p1, p2, dir, 0)
- crossbow:spawn_particles(tpos, CROSSBOW_EXPLOSION_TEXTURE)
- object:punch(puncher, nil, CROSSBOW_BOLT_TOOL_CAPS, dir)
- end
- end
- end
- local function stop_crossbow_bolt(object, pos, stuck)
- local acceleration = {x=0, y=-10, z=0}
- if stuck == true then
- pos = pos or object:getpos()
- acceleration = {x=0, y=0, z=0}
- object:moveto(pos)
- end
- object:set_properties({
- physical = true,
- collisionbox = {-1/8,-1/8,-1/8, 1/8,1/8,1/8},
- })
- object:setvelocity({x=0, y=0, z=0})
- object:setacceleration(acceleration)
- end
- minetest.register_craftitem('castle_weapons:crossbow_bolt', {
- description = S('Bolt'),
- inventory_image = 'castle_crossbow_bolt_inv.png',
- })
- minetest.register_entity('castle_weapons:crossbow_bolt_entity', {
- physical = false,
- visual = 'mesh',
- mesh = 'castle_crossbow_bolt.b3d',
- visual_size = {x=1.0, y=1.0},
- textures = {
- 'castle_crossbow_bolt_uv.png'
- },
- timer = 0,
- lifetime = CROSSBOW_BOLT_LIFETIME,
- player = nil,
- state = 'init',
- node_pos = nil,
- collisionbox = {0,0,0, 0,0,0},
- on_activate = function(self, staticdata)
- self.object:set_armor_groups({immortal=1})
- if staticdata == 'expired' then
- self.object:remove()
- end
- end,
- on_punch = function(self, puncher)
- if puncher then
- if puncher:is_player() then
- local stack = 'castle_weapons:crossbow_bolt'
- local inv = puncher:get_inventory()
- if inv:room_for_item('main', stack) then
- inv:add_item('main', stack)
- self.object:remove()
- end
- end
- end
- end,
- on_step = function(self, dtime)
- if self.state == 'init' then
- return
- end
- self.timer = self.timer + dtime
- self.lifetime = self.lifetime - dtime
- if self.lifetime < 0 then
- self.object:remove()
- return
- elseif self.state == 'dropped' then
- return
- elseif self.state == 'stuck' then
- if self.timer > 1 then
- if self.node_pos then
- local node = minetest.get_node(self.node_pos)
- if node.name then
- local item = minetest.registered_items[node.name]
- if item then
- if not item.walkable then
- self.state = 'dropped'
- stop_crossbow_bolt(self.object)
- return
- end
- end
- end
- end
- self.timer = 0
- end
- return
- end
- if self.timer > 0.2 then
- local pos = self.object:getpos()
- local dir = vector.normalize(self.object:getvelocity())
- local frame = get_animation_frame(dir)
- self.object:set_animation({x=frame, y=frame}, 0)
- local objects = minetest.get_objects_inside_radius(pos, 5)
- for _,obj in ipairs(objects) do
- if crossbow:is_valid_object(obj) then
- local collisionbox = {-0.25,-1.0,-0.25, 0.25,0.8,0.25}
- local offset = CROSSBOW_PLAYER_OFFSET
- if not obj:is_player() then
- offset = CROSSBOW_ENTITY_OFFSET
- local ent = obj:get_luaentity()
- if ent then
- local def = minetest.registered_entities[ent.name]
- collisionbox = def.collisionbox or collisionbox
- end
- end
- local opos = vector.add(obj:getpos(), offset)
- local ray = {pos=pos, dir=dir}
- local plane = {pos=opos, normal={x=-1, y=0, z=-1}}
- local ipos = crossbow:get_intersect_pos(ray, plane, collisionbox)
- if ipos then
- punch_object(self.player, obj)
- end
- end
- end
- local p = vector.add(pos, vector.multiply(dir, {x=5, y=5, z=5}))
- local _, npos = minetest.line_of_sight(pos, p, 1)
- if npos then
- local node = minetest.get_node(npos)
- local tpos = get_target_pos(pos, npos, dir, 0.66)
- self.node_pos = npos
- self.state = 'stuck'
- stop_crossbow_bolt(self.object, tpos, true)
- minetest.sound_play('castle_crossbow_bolt', {gain = 0.08, max_hear_distance = 2})
- end
- self.timer = 0
- end
- end,
- get_staticdata = function(self)
- return 'expired'
- end,
- })
- minetest.register_tool('castle_weapons:crossbow_loaded', {
- description = S('Crossbow'),
- inventory_image = 'castle_crossbow_loaded.png',
- groups = {not_in_creative_inventory=1},
- on_use = function(itemstack, user, pointed_thing)
- minetest.sound_play('castle_crossbow_click', {object=user})
- if not minetest.settings:get_bool('creative_mode') then
- itemstack:add_wear(65535/CROSSBOW_USES)
- end
- itemstack = 'castle_weapons:crossbow 1 '..itemstack:get_wear()
- local pos = user:getpos()
- local dir = user:get_look_dir()
- local yaw = user:get_look_yaw()
- if pos and dir and yaw then
- pos.y = pos.y + 1.5
- local obj = minetest.add_entity(pos, 'castle_weapons:crossbow_bolt_entity')
- local ent = nil
- if obj then
- ent = obj:get_luaentity()
- end
- if ent then
- obj:set_properties({
- textures = {'castle_crossbow_bolt_uv.png'}
- })
- minetest.sound_play('castle_crossbow_shoot', {object=obj})
- local frame = get_animation_frame(dir)
- obj:setyaw(yaw + math.pi)
- obj:set_animation({x=frame, y=frame}, 0)
- obj:setvelocity({x=dir.x * 20, y=dir.y * 20, z=dir.z * 20})
- if pointed_thing.type ~= 'nothing' then
- local ppos = minetest.get_pointed_thing_position(pointed_thing, false)
- local _, npos = minetest.line_of_sight(pos, ppos, 1)
- if npos then
- ppos = npos
- pointed_thing.type = 'node'
- end
- if pointed_thing.type == 'object' then
- punch_object(user, pointed_thing.ref)
- elseif pointed_thing.type == 'node' then
- local node = minetest.get_node(ppos)
- local tpos = get_target_pos(pos, ppos, dir, 0.66)
- minetest.after(0.2, function(object, pos, npos)
- ent.node_pos = npos
- ent.state = 'stuck'
- stop_crossbow_bolt(object, pos, true)
- minetest.sound_play('castle_crossbow_bolt', {gain = 0.08, max_hear_distance = 2})
- end, obj, tpos, ppos)
- return itemstack
- end
- end
- obj:setacceleration({x=dir.x * -3, y=-5, z=dir.z * -3})
- ent.player = ent.player or user
- ent.state = 'flight'
- end
- end
- return itemstack
- end,
- })
- minetest.register_tool('castle_weapons:crossbow', {
- description = S('Crossbow'),
- inventory_image = 'castle_crossbow_inv.png',
- on_use = function(itemstack, user, pointed_thing)
- local inv = user:get_inventory()
- if inv:contains_item('main', 'castle_weapons:crossbow_bolt') then
- minetest.sound_play('castle_crossbow_reload', {object=user})
- if not minetest.settings:get_bool('creative_mode') then
- inv:remove_item('main', 'castle_weapons:crossbow_bolt 1')
- end
- return 'castle_weapons:crossbow_loaded 1 '..itemstack:get_wear()
- end
- minetest.sound_play('castle_crossbow_click', {object=user})
- end,
- })
- -----------
- --Crafting
- -----------
- minetest.register_craft({
- output = 'castle_weapons:crossbow',
- recipe = {
- {'default:steel_ingot', 'default:stick', 'default:steel_ingot'},
- {'farming:string', 'farming:string', 'farming:string'},
- {'', 'default:stick', ''},
- }
- })
- minetest.register_craft({
- output = 'castle_weapons:crossbow_bolt',
- recipe = {
- {'epic:arrow_tip', 'default:stick'},
- }
- })
|