diff --git a/_maps/map_files/RandomRuins/LavaRuins/lavaland_surface_ash_walker1.dmm b/_maps/map_files/RandomRuins/LavaRuins/lavaland_surface_ash_walker1.dmm index 1fad6ae212c..af39fb0195c 100644 --- a/_maps/map_files/RandomRuins/LavaRuins/lavaland_surface_ash_walker1.dmm +++ b/_maps/map_files/RandomRuins/LavaRuins/lavaland_surface_ash_walker1.dmm @@ -1416,6 +1416,14 @@ }, /turf/simulated/floor/plating/asteroid/basalt/lava_land_surface, /area/lavaland/surface/outdoors) +"Pd" = ( +/obj/structure/stone_tile/cracked{ + dir = 4 + }, +/obj/effect/mapping_helpers/no_lava, +/obj/effect/decal/cleanable/ashrune, +/turf/simulated/floor/plating/asteroid/basalt/lava_land_surface, +/area/lavaland/surface/outdoors) (1,1,1) = {" aa @@ -1606,7 +1614,7 @@ ak ak cg cb -cg +Pd cn bL dr diff --git a/code/__DEFINES/mobs.dm b/code/__DEFINES/mobs.dm index 0e8c0873ed8..089ea970b09 100644 --- a/code/__DEFINES/mobs.dm +++ b/code/__DEFINES/mobs.dm @@ -207,6 +207,8 @@ #define isskeleton(A) (is_species(A, /datum/species/skeleton)) #define ishumanbasic(A) (is_species(A, /datum/species/human)) #define isunathi(A) (is_species(A, /datum/species/unathi)) +#define isashwalker(A) (is_species(A, /datum/species/unathi/ashwalker)) +#define isashwalkershaman(A) (is_species(A, /datum/species/unathi/ashwalker/shaman)) #define istajaran(A) (is_species(A, /datum/species/tajaran)) #define isvulpkanin(A) (is_species(A, /datum/species/vulpkanin)) #define isskrell(A) (is_species(A, /datum/species/skrell)) diff --git a/code/__DEFINES/rituals.dm b/code/__DEFINES/rituals.dm new file mode 100644 index 00000000000..c33d531225c --- /dev/null +++ b/code/__DEFINES/rituals.dm @@ -0,0 +1,18 @@ +/// Used in ritual variables +#define DEFAULT_RITUAL_RANGE_FIND 1 +#define DEFAULT_RITUAL_COOLDOWN (100 SECONDS) +#define DEFAULT_RITUAL_DISASTER_PROB 10 +#define DEFAULT_RITUAL_FAIL_PROB 10 +/// Ritual object bitflags +#define RITUAL_STARTED (1<<0) +#define RITUAL_ENDED (1<<1) +#define RITUAL_FAILED (1<<2) +/// Ritual datum bitflags +#define RITUAL_SUCCESSFUL (1<<0) +/// Invocation checks, should not be used in extra checks. +#define RITUAL_FAILED_INVALID_SPECIES (1<<1) +#define RITUAL_FAILED_EXTRA_INVOKERS (1<<2) +#define RITUAL_FAILED_MISSED_REQUIREMENTS (1<<3) +#define RITUAL_FAILED_ON_PROCEED (1<<4) +#define RITUAL_FAILED_INVALID_SPECIAL_ROLE (1<<5) + diff --git a/code/__DEFINES/status_effects.dm b/code/__DEFINES/status_effects.dm index 29e33ed9265..b48227307a6 100644 --- a/code/__DEFINES/status_effects.dm +++ b/code/__DEFINES/status_effects.dm @@ -94,6 +94,8 @@ #define STATUS_EFFECT_CRUSHERMARK /datum/status_effect/crusher_mark //if struck with a proto-kinetic crusher, takes a ton of damage +#define STATUS_EFFECT_FANG_EXHAUSTION /datum/status_effect/fang_exhaust // called by poison fang (crusher trophy) + #define STATUS_EFFECT_SAWBLEED /datum/status_effect/saw_bleed //if the bleed builds up enough, takes a ton of damage #define STATUS_EFFECT_BLOODLETTING /datum/status_effect/saw_bleed/bloodletting //nerfed version diff --git a/code/__HELPERS/global_lists.dm b/code/__HELPERS/global_lists.dm index f8907ab3667..539019a41c8 100644 --- a/code/__HELPERS/global_lists.dm +++ b/code/__HELPERS/global_lists.dm @@ -313,6 +313,7 @@ prize_list["Miscellaneous"] = list( EQUIPMENT("Absinthe", /obj/item/reagent_containers/food/drinks/bottle/absinthe/premium, 500), EQUIPMENT("Alien Toy", /obj/item/clothing/mask/facehugger/toy, 300), + EQUIPMENT("Richard & Co cigarettes", /obj/item/storage/fancy/cigarettes/cigpack_richard, 400), EQUIPMENT("Cigar", /obj/item/clothing/mask/cigarette/cigar/havana, 300), EQUIPMENT("GAR Meson Scanners", /obj/item/clothing/glasses/meson/gar, 800), EQUIPMENT("GPS upgrade", /obj/item/gpsupgrade, 1500), @@ -342,6 +343,7 @@ EQUIPMENT("Absinthe", /obj/item/reagent_containers/food/drinks/bottle/absinthe/premium, 250), EQUIPMENT("Cigarettes", /obj/item/storage/fancy/cigarettes, 100), EQUIPMENT("Medical Marijuana", /obj/item/storage/fancy/cigarettes/cigpack_med, 250), + EQUIPMENT("Richard & Co cigarettes", /obj/item/storage/fancy/cigarettes/cigpack_richard, 400), EQUIPMENT("Cigar", /obj/item/clothing/mask/cigarette/cigar/havana, 150), EQUIPMENT("Box of matches", /obj/item/storage/box/matches, 50), EQUIPMENT("Cheeseburger", /obj/item/reagent_containers/food/snacks/cheeseburger, 150), diff --git a/code/datums/components/ritual_object.dm b/code/datums/components/ritual_object.dm new file mode 100644 index 00000000000..31064e438db --- /dev/null +++ b/code/datums/components/ritual_object.dm @@ -0,0 +1,119 @@ +/datum/component/ritual_object + /// Pre-defined rituals list + var/list/rituals = list() + /// We define rituals from this. + var/list/allowed_categories + /// Required species to activate ritual object + var/list/allowed_species + /// Required special role to activate ritual object + var/list/allowed_special_role + /// Prevents from multiple uses + var/active_ui = FALSE + +/datum/component/ritual_object/Initialize( + allowed_categories = /datum/ritual, + list/allowed_species, + list/allowed_special_role +) + + if(!isobj(parent)) + return COMPONENT_INCOMPATIBLE + + src.allowed_categories = allowed_categories + src.allowed_species = allowed_species + src.allowed_special_role = allowed_special_role + get_rituals() + +/datum/component/ritual_object/RegisterWithParent() + RegisterSignal(parent, COMSIG_ATOM_ATTACK_HAND, PROC_REF(attackby)) + +/datum/component/ritual_object/UnregisterFromParent() + UnregisterSignal(parent, COMSIG_ATOM_ATTACK_HAND) + +/datum/component/ritual_object/proc/get_rituals() // We'll get all rituals for flexibility. + LAZYCLEARLIST(rituals) + + for(var/datum/ritual/ritual as anything in typecacheof(allowed_categories)) + if(!ritual.name) + continue + + rituals += new ritual + + for(var/datum/ritual/ritual as anything in rituals) + ritual.ritual_object = parent + + return + +/datum/component/ritual_object/Destroy(force) + LAZYNULL(rituals) + return ..() + +/datum/component/ritual_object/proc/attackby(datum/source, mob/user) + SIGNAL_HANDLER + + if(active_ui) + return + + if(!ishuman(user)) + return + + var/mob/living/carbon/human/human = user + + if(allowed_species && !is_type_in_list(human.dna.species, allowed_species)) + return + + if(allowed_special_role && !is_type_in_list(human.mind?.special_role, allowed_special_role)) + return + + active_ui = TRUE + INVOKE_ASYNC(src, PROC_REF(open_ritual_ui), human) + + return COMPONENT_CANCEL_ATTACK_CHAIN + +/datum/component/ritual_object/proc/open_ritual_ui(mob/living/carbon/human/human) + var/list/rituals_list = get_available_rituals(human) + + if(!LAZYLEN(rituals_list)) + active_ui = FALSE + to_chat(human, "Не имеется доступных для выполнения ритуалов.") + return + + var/choosen_ritual = tgui_input_list(human, "Выберите ритуал", "Ритуалы", rituals_list) + + if(!choosen_ritual) + active_ui = FALSE + return + + var/ritual_status + + for(var/datum/ritual/ritual as anything in rituals) + if(choosen_ritual != ritual.name) + continue + + ritual_status = ritual.pre_ritual_check(human) + break + + if(ritual_status) + active_ui = FALSE + + return FALSE + +/datum/component/ritual_object/proc/get_available_rituals(mob/living/carbon/human/human) + var/list/rituals_list = list() + + for(var/datum/ritual/ritual as anything in rituals) + if(ritual.charges == 0) + continue + + if(!COOLDOWN_FINISHED(ritual, ritual_cooldown)) + continue + + if(ritual.allowed_species && !is_type_in_list(human.dna.species, ritual.allowed_species)) + continue + + if(ritual.allowed_special_role && !is_type_in_list(human.mind?.special_role, ritual.allowed_special_role)) + continue + + LAZYADD(rituals_list, ritual.name) + + return rituals_list diff --git a/code/datums/rituals.dm b/code/datums/rituals.dm new file mode 100644 index 00000000000..4c3049d00fe --- /dev/null +++ b/code/datums/rituals.dm @@ -0,0 +1,1255 @@ +/datum/ritual + /// Linked object + var/obj/ritual_object + /// Name of our ritual + var/name + /// If ritual requires more than one invoker + var/extra_invokers = 0 + /// If invoker species isn't in allowed - he won't do ritual. + var/list/allowed_species + /// If invoker special role isn't in allowed - he won't do ritual. + var/list/allowed_special_role + /// Required to ritual invoke things are located here + var/list/required_things + /// If true - only whitelisted species will be added as invokers + var/require_allowed_species = TRUE + /// Same as require_allowed_species, but requires special role to be counted as invoker. + var/require_allowed_special_role = FALSE + /// We search for humans in that radius + var/finding_range = DEFAULT_RITUAL_RANGE_FIND + /// Amount of maximum ritual uses. + var/charges = -1 + /// Cooldown for one ritual + COOLDOWN_DECLARE(ritual_cooldown) + /// Our cooldown after we casted ritual. + var/cooldown_after_cast = DEFAULT_RITUAL_COOLDOWN + /// If our ritual failed on proceed - we'll try to cause disaster. + var/disaster_prob = DEFAULT_RITUAL_DISASTER_PROB + /// A chance of failing our ritual. + var/fail_chance = DEFAULT_RITUAL_FAIL_PROB + /// After successful ritual we'll destroy used things. + var/ritual_should_del_things = TRUE + /// After failed ritual proceed - we'll delete items. + var/ritual_should_del_things_on_fail = FALSE + /// Temporary list of objects, which we will delete. Or use in transformations! Then clear list. + var/list/used_things = list() + /// Temporary list of invokers. + var/list/invokers = list() + /// If defined - do_after will be added to your ritual + var/cast_time + +/datum/ritual/Destroy(force) + ritual_object = null + LAZYNULL(used_things) + LAZYNULL(required_things) + LAZYNULL(invokers) + return ..() + +/datum/ritual/proc/pre_ritual_check(mob/living/carbon/human/invoker) + var/failed = FALSE + var/cause_disaster = FALSE + + var/del_things = FALSE + var/start_cooldown = FALSE + + handle_ritual_object(RITUAL_STARTED) + + . = ritual_invoke_check(invoker) + switch(.) + if(RITUAL_SUCCESSFUL) + start_cooldown = TRUE + addtimer(CALLBACK(src, PROC_REF(handle_ritual_object), RITUAL_ENDED), 1 SECONDS) + charges-- + if(RITUAL_FAILED_INVALID_SPECIES) + failed = TRUE + if(RITUAL_FAILED_EXTRA_INVOKERS) + failed = TRUE + if(RITUAL_FAILED_MISSED_REQUIREMENTS) + failed = TRUE + if(RITUAL_FAILED_INVALID_SPECIAL_ROLE) + failed = TRUE + if(RITUAL_FAILED_ON_PROCEED) + failed = TRUE + cause_disaster = TRUE + start_cooldown = TRUE + if(NONE) + failed = TRUE + + if(start_cooldown) + COOLDOWN_START(src, ritual_cooldown, cooldown_after_cast) + + if(cause_disaster && prob(disaster_prob)) + disaster(invoker) + + if((. & RITUAL_SUCCESSFUL) && (ritual_should_del_things)) + del_things = TRUE + + if((. & RITUAL_FAILED_ON_PROCEED) && (ritual_should_del_things_on_fail)) + del_things = TRUE + + if(del_things) + del_things() + + if(failed) + addtimer(CALLBACK(src, PROC_REF(handle_ritual_object), RITUAL_FAILED), 2 SECONDS) + + /// We use pre-defines + LAZYCLEARLIST(invokers) + LAZYCLEARLIST(used_things) + + return . + +/datum/ritual/proc/handle_ritual_object(bitflags, silent = FALSE) + switch(bitflags) + if(RITUAL_STARTED) + . = RITUAL_STARTED + if(!silent) + playsound(ritual_object.loc, 'sound/effects/ghost2.ogg', 50, TRUE) + if(RITUAL_ENDED) + . = RITUAL_ENDED + if(!silent) + playsound(ritual_object.loc, 'sound/effects/phasein.ogg', 50, TRUE) + if(RITUAL_FAILED) + . = RITUAL_FAILED + if(!silent) + playsound(ritual_object.loc, 'sound/effects/empulse.ogg', 50, TRUE) + + return . + +/datum/ritual/proc/del_things() // This is a neutral variant with item delete. Override it to change. + for(var/obj/item/thing in used_things) + qdel(thing) + + return + +/datum/ritual/proc/ritual_invoke_check(mob/living/carbon/human/invoker) + if(!COOLDOWN_FINISHED(src, ritual_cooldown)) + return NONE + + if(charges == 0) + return NONE + + if(allowed_special_role && !is_type_in_list(invoker.mind?.special_role, allowed_special_role)) + return RITUAL_FAILED_INVALID_SPECIAL_ROLE + + if(allowed_species && !is_type_in_list(invoker.dna.species, allowed_species)) // double check to avoid funny situations + return RITUAL_FAILED_INVALID_SPECIES + + if(!check_invokers(invoker)) + return RITUAL_FAILED_EXTRA_INVOKERS + + if(required_things && !check_contents(invoker)) + return RITUAL_FAILED_MISSED_REQUIREMENTS + + if(prob(fail_chance)) + return RITUAL_FAILED_ON_PROCEED + + if(cast_time && !cast(invoker)) + return RITUAL_FAILED_ON_PROCEED + + return do_ritual(invoker) + +/datum/ritual/proc/cast(mob/living/carbon/human/invoker) + . = TRUE + LAZYADD(invokers, invoker) + + for(var/mob/living/carbon/human/human as anything in invokers) + if(!do_after(human, cast_time, ritual_object, extra_checks = CALLBACK(src, PROC_REF(action_check_contents)))) + . = FALSE + + LAZYREMOVE(invokers, invoker) + + return . + +/datum/ritual/proc/check_invokers(mob/living/carbon/human/invoker) + if(!extra_invokers) + return TRUE + + for(var/mob/living/carbon/human/human in range(finding_range, ritual_object)) + if(human == invoker) + continue + + if(require_allowed_species && !is_type_in_list(human.dna.species, allowed_species)) + continue + + if(require_allowed_special_role && !is_type_in_list(human.mind?.special_role, allowed_special_role)) + continue + + LAZYADD(invokers, human) + + if(LAZYLEN(invokers) >= extra_invokers) + break + + if(LAZYLEN(invokers) < extra_invokers) + ritual_object.balloon_alert(invoker, "требуется больше участников!") + return FALSE + + return TRUE + +/datum/ritual/proc/check_contents(mob/living/carbon/human/invoker) + var/list/atom/movable/atoms = list() + + for(var/atom/obj as anything in range(finding_range, ritual_object)) + if(isitem(obj)) + var/obj/item/close_item = obj + if(close_item.item_flags & ABSTRACT) + continue + + if(obj.invisibility) + continue + + if(obj == invoker) + continue + + if(obj == ritual_object) + continue + + if(locate(obj) in invokers) + continue + + LAZYADD(atoms, obj) + + var/list/requirements = required_things.Copy() + for(var/atom/atom as anything in atoms) + for(var/req_type in requirements) + if(requirements[req_type] <= 0) + continue + + if(!istype(atom, req_type)) + continue + + LAZYADD(used_things, atom) + + if(isstack(atom)) + var/obj/item/stack/picked_stack = atom + LAZYREMOVE(requirements[req_type], picked_stack.amount) + else + requirements[req_type]-- + + var/list/what_are_we_missing = list() + for(var/req_type in requirements) + var/number_of_things = requirements[req_type] + + if(number_of_things <= 0) + continue + + LAZYADD(what_are_we_missing, req_type) + + if(LAZYLEN(what_are_we_missing)) + ritual_object.balloon_alert(invoker, "требуется больше компонентов!") + return FALSE + + return TRUE + +/datum/ritual/proc/action_check_contents() + for(var/atom/atom as anything in used_things) + if(QDELETED(atom)) + return FALSE + + if(!(atom in range(finding_range, ritual_object))) + return FALSE + + return TRUE + +/datum/ritual/proc/do_ritual(mob/living/carbon/human/invoker) // Do ritual stuff. + return RITUAL_SUCCESSFUL + +/datum/ritual/proc/disaster(mob/living/carbon/human/invoker) + return + +/datum/ritual/ashwalker + /// If ritual requires extra shaman invokers + var/extra_shaman_invokers = 0 + /// If ritual can be invoked only by shaman + var/shaman_only = FALSE + allowed_species = list(/datum/species/unathi/ashwalker, /datum/species/unathi/draconid) + +/datum/ritual/ashwalker/check_invokers(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + if(shaman_only && !isashwalkershaman(invoker)) + to_chat(invoker, span_warning("Только шаман может выполнить данный ритуал!")) + return FALSE + + var/list/shaman_invokers = list() + + if(extra_shaman_invokers) + for(var/mob/living/carbon/human/human as anything in invokers) + if(human == invoker) + continue + + if(isashwalkershaman(human)) + LAZYADD(shaman_invokers, human) + + if(LAZYLEN(shaman_invokers) >= extra_shaman_invokers) + break + + if(LAZYLEN(shaman_invokers) < extra_shaman_invokers) + ritual_object.balloon_alert(invoker, "требуется больше шаманов!") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/summon_ashstorm + name = "Ash storm summon" + shaman_only = TRUE + disaster_prob = 20 + charges = 2 + cooldown_after_cast = 1200 SECONDS + cast_time = 100 SECONDS + fail_chance = 20 + extra_invokers = 2 + required_things = list( + /mob/living/simple_animal/hostile/asteroid/goldgrub = 1 + ) + +/datum/ritual/ashwalker/summon_ashstorm/check_contents(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + for(var/mob/living/living in used_things) + if(living.stat != DEAD) + to_chat(invoker, "Существа должны быть мертвы") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/summon_ashstorm/del_things() + . = ..() + + for(var/mob/living/living in used_things) + living.gib() + + return + +/datum/ritual/ashwalker/summon_ashstorm/check_invokers(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + if(!invoker.fire_stacks) + to_chat(invoker, "Инициатор ритуала должнен быть в воспламеняемой субстанции.") + return FALSE + + for(var/mob/living/carbon/human/human as anything in invokers) + if(!human.fire_stacks) + to_chat(invoker, "Участники ритуала должны быть в воспламеняемой субстанции.") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/summon_ashstorm/do_ritual(mob/living/carbon/human/invoker) + SSweather.run_weather(/datum/weather/ash_storm) + message_admins("[key_name(invoker)] accomplished ashstorm ritual and summoned ashstorm") + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/summon_ashstorm/disaster(mob/living/carbon/human/invoker) + var/list/targets = list() + + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(isashwalker(human)) + LAZYADD(targets, human) + + if(!LAZYLEN(targets)) + return + + var/mob/living/carbon/human/human = pick(targets) + var/datum/disease/virus/cadaver/cadaver = new + cadaver.Contract(human) + + return + +/datum/ritual/ashwalker/summon_ashstorm/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/magic/fleshtostone.ogg', 50, TRUE) + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/magic/invoke_general.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/magic/castsummon.ogg', 50, TRUE) + + return . + +/datum/ritual/ashwalker/transformation + name = "Transformation ritual" + disaster_prob = 30 + fail_chance = 50 + extra_invokers = 1 + cooldown_after_cast = 480 SECONDS + cast_time = 70 SECONDS + ritual_should_del_things_on_fail = TRUE + required_things = list( + /obj/item/twohanded/spear = 3, + /obj/item/organ/internal/regenerative_core = 1, + /mob/living/carbon/human = 1 + ) + +/datum/ritual/ashwalker/transformation/do_ritual(mob/living/carbon/human/invoker) + var/mob/living/carbon/human/human = locate() in used_things + + if(!human || !human.mind || !human.ckey) + return RITUAL_FAILED_ON_PROCEED // Your punishment + + human.set_species(/datum/species/unathi/ashwalker) + human.mind.store_memory("Теперь вы пеплоходец, вы часть племени! Вы довольно смутно помните о прошлой жизни, и вы не помните, как пользоваться технологиями!") + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/transformation/disaster(mob/living/carbon/human/invoker) + invoker.adjustBrainLoss(15) + invoker.SetKnockdown(5 SECONDS) + + var/mob/living/carbon/human/human = locate() in used_things + + if(QDELETED(human)) + return + + var/list/destinations = list() + + for(var/obj/item/radio/beacon/beacon in GLOB.global_radios) + LAZYADD(destinations, get_turf(beacon)) + + human.forceMove(safepick(destinations)) + playsound(get_turf(human), 'sound/magic/invoke_general.ogg', 50, TRUE) + + return + +/datum/ritual/ashwalker/transformation/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + if(. == RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/effects/clone_jutsu.ogg', 50, TRUE) + return + + . = ..(bitflags) + return . + +/datum/ritual/ashwalker/summon + name = "Summoning ritual" + disaster_prob = 30 + fail_chance = 30 + shaman_only = TRUE + cooldown_after_cast = 900 SECONDS + cast_time = 50 SECONDS + extra_invokers = 1 + +/datum/ritual/ashwalker/summon/do_ritual(mob/living/carbon/human/invoker) + var/list/ready_for_summoning = list() + + for(var/mob/living/carbon/human/human in GLOB.mob_list) + if(isashwalker(human)) + LAZYADD(ready_for_summoning, human) + + if(!LAZYLEN(ready_for_summoning)) + return RITUAL_FAILED_ON_PROCEED + + var/mob/living/carbon/human/human = tgui_input_list(invoker, "Who will be summoned?", "Summon ritual", ready_for_summoning) + + if(!human) + return RITUAL_FAILED_ON_PROCEED + + LAZYADD(invokers, invoker) + + for(var/mob/living/carbon/human/summoner as anything in invokers) + summoner.blood_volume -= (summoner.blood_volume * 0.20) + summoner.apply_damage(25, def_zone = pick(BODY_ZONE_L_ARM, BODY_ZONE_R_ARM)) + + human.forceMove(ritual_object) + human.vomit() + human.Weaken(10 SECONDS) + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/summon/disaster(mob/living/carbon/human/invoker) + if(!prob(70)) + return + + var/obj/item/organ/external/limb = invoker.get_organ(pick(BODY_ZONE_L_ARM, BODY_ZONE_R_ARM, BODY_ZONE_L_LEG, BODY_ZONE_R_LEG)) + limb?.droplimb() + + return + +/datum/ritual/ashwalker/summon/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/weapons/zapbang.ogg', 50, TRUE) + var/datum/effect_system/smoke_spread/smoke = new + smoke.set_up(5, FALSE, ritual_object.loc) + smoke.start() + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/magic/forcewall.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/magic/invoke_general.ogg', 50, TRUE) + + return . + +/datum/ritual/ashwalker/curse + name = "Curse ritual" + disaster_prob = 30 + fail_chance = 30 + cooldown_after_cast = 600 SECONDS + cast_time = 60 SECONDS + charges = 3 + shaman_only = TRUE + extra_invokers = 2 + required_things = list( + /mob/living/carbon/human = 3 + ) + +/datum/ritual/ashwalker/curse/del_things() + for(var/mob/living/carbon/human/human as anything in used_things) + human.gib() + + return + +/datum/ritual/ashwalker/curse/check_contents(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + for(var/mob/living/carbon/human/human as anything in used_things) + if(human.stat != DEAD) + to_chat(invoker, "Гуманоиды должны быть мертвы.") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/curse/do_ritual(mob/living/carbon/human/invoker) + var/list/humans = list() + + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(!isashwalker(human)) + LAZYADD(humans, human) + + if(!LAZYLEN(humans)) + return RITUAL_FAILED_ON_PROCEED + + var/mob/living/carbon/human/human = pick(humans) + var/datum/disease/vampire/disease = new + + if(!disease.Contract(human)) + return RITUAL_FAILED_ON_PROCEED + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/curse/disaster(mob/living/carbon/human/invoker) + var/list/targets = list() + + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(isashwalker(human)) + LAZYADD(targets, human) + + if(!LAZYLEN(targets)) + return + + var/mob/living/carbon/human/human = pick(targets) + human.monkeyize() + + return + +/datum/ritual/ashwalker/power + name = "Power ritual" + disaster_prob = 40 + fail_chance = 40 + charges = 1 + cooldown_after_cast = 800 SECONDS + cast_time = 80 SECONDS + shaman_only = TRUE + extra_invokers = 4 + required_things = list( + /mob/living/simple_animal/hostile/asteroid/goliath = 3, + /obj/item/organ/internal/regenerative_core = 3 + ) + +/datum/ritual/ashwalker/power/del_things() + for(var/mob/living/living in used_things) + living.gib() + + return + +/datum/ritual/ashwalker/power/check_contents(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + for(var/mob/living/living in used_things) + if(living.stat != DEAD) + to_chat(invoker, "Существа должны быть мертвы.") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/power/do_ritual(mob/living/carbon/human/invoker) + LAZYADD(invokers, invoker) + + for(var/mob/living/carbon/human/human as anything in invokers) + if(LAZYIN(human.dna?.default_blocks, GLOB.weakblock)) + human.force_gene_block(GLOB.weakblock) + + human.force_gene_block(GLOB.strongblock, TRUE) + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/power/disaster(mob/living/carbon/human/invoker) + var/list/targets = list() + + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(isashwalker(human)) + LAZYADD(targets, human) + + if(!LAZYLEN(targets)) + return + + invoker.force_gene_block(pick(GLOB.bad_blocks), TRUE) + for(var/mob/living/carbon/human/human as anything in invokers) + human.force_gene_block(pick(GLOB.bad_blocks), TRUE) + + var/mob/living/carbon/human/human = pick(targets) + human.force_gene_block(pick(GLOB.bad_blocks), TRUE) + + return + +/datum/ritual/ashwalker/power/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/magic/castsummon.ogg', 50, TRUE) + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/magic/smoke.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/magic/strings.ogg', 50, TRUE) + + return . + +/datum/ritual/ashwalker/resurrection + name = "Resurrection ritual" + charges = 3 + extra_invokers = 2 + cooldown_after_cast = 180 SECONDS + cast_time = 100 SECONDS + shaman_only = TRUE + disaster_prob = 25 + fail_chance = 35 + required_things = list( + /obj/item/organ/internal/regenerative_core = 2, + /mob/living/carbon/human = 1, + /obj/item/reagent_containers/food/snacks/grown/ash_flora/fireblossom = 4, + /obj/item/reagent_containers/food/snacks/grown/ash_flora/cactus_fruit = 1 + ) + +/datum/ritual/ashwalker/resurrection/check_contents(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + for(var/mob/living/living in used_things) + if(living.stat != DEAD) + to_chat(invoker, "Существа должны быть мертвы.") + return FALSE + + var/mob/living/carbon/human/human = locate() in used_things + + if(!human.mind || !human.ckey) + return FALSE + + if(!isashwalker(human)) + fail_chance = 15 + + return TRUE + +/datum/ritual/ashwalker/resurrection/do_ritual(mob/living/carbon/human/invoker) + var/mob/living/carbon/human/human = locate() in used_things + human.revive() + human.adjustBrainLoss(20) + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/resurrection/disaster(mob/living/carbon/human/invoker) + for(var/mob/living/carbon/human/human in range(10, ritual_object)) + if(!isashwalker(human) || human.stat == DEAD) + continue + + human.adjustBrainLoss(15) + + return + +/datum/ritual/ashwalker/resurrection/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/magic/clockwork/reconstruct.ogg', 50, TRUE) + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/magic/disable_tech.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/magic/invoke_general.ogg', 50, TRUE) + + return . + +/datum/ritual/ashwalker/recharge + name = "Recharge rituals" + extra_invokers = 3 + disaster_prob = 30 + fail_chance = 50 + cooldown_after_cast = 360 SECONDS + cast_time = 90 SECONDS + shaman_only = TRUE + required_things = list( + /mob/living/simple_animal/hostile/asteroid/basilisk/watcher = 1, + /mob/living/simple_animal/hostile/asteroid/goliath = 1, + /obj/item/organ/internal/regenerative_core = 1, + /mob/living/simple_animal/hostile/asteroid/goldgrub = 1 + ) + var/list/blacklisted_rituals = list(/datum/ritual/ashwalker/power) + +/datum/ritual/ashwalker/recharge/del_things() + . = ..() + + for(var/mob/living/living in used_things) + living.gib() + + return + +/datum/ritual/ashwalker/recharge/check_contents(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + for(var/mob/living/living in used_things) + if(living.stat != DEAD) + to_chat(invoker, "Существа должны быть мертвы.") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/recharge/do_ritual(mob/living/carbon/human/invoker) + var/datum/component/ritual_object/component = ritual_object.GetComponent(/datum/component/ritual_object) + + if(!component) + return RITUAL_FAILED_ON_PROCEED + + for(var/datum/ritual/ritual as anything in component.rituals) + if(is_type_in_list(ritual, blacklisted_rituals)) + continue + + if(ritual.charges < 0) + continue + + ritual.charges++ + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/recharge/disaster(mob/living/carbon/human/invoker) + var/list/targets = list() + + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(isashwalker(human)) + LAZYADD(targets, human) + + if(!LAZYLEN(targets)) + return + + var/mob/living/carbon/human/human = pick(targets) + new /obj/item/organ/internal/legion_tumour(human) + + return + +/datum/ritual/ashwalker/recharge/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/magic/castsummon.ogg', 50, TRUE) + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/magic/cult_spell.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/magic/invoke_general.ogg', 50, TRUE) + + return . + +/datum/ritual/ashwalker/population + name = "Population ritual" + extra_invokers = 2 + charges = 1 + cooldown_after_cast = 120 SECONDS + cast_time = 40 SECONDS + ritual_should_del_things_on_fail = TRUE + required_things = list( + /obj/item/reagent_containers/food/snacks/grown/ash_flora/cactus_fruit = 1, + /obj/item/reagent_containers/food/snacks/grown/ash_flora/fireblossom = 1, + /obj/item/reagent_containers/food/snacks/grown/ash_flora/mushroom_stem = 1, + /obj/item/reagent_containers/food/snacks/grown/ash_flora/mushroom_leaf = 1, + /obj/item/reagent_containers/food/snacks/grown/ash_flora/mushroom_cap = 1, + /obj/item/reagent_containers/food/snacks/grown/ash_flora/shavings = 1 + ) + +/datum/ritual/ashwalker/population/check_invokers(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + if(!isashwalkershaman(invoker)) + disaster_prob = 40 + fail_chance = 40 + + return TRUE + +/datum/ritual/ashwalker/population/del_things() + for(var/mob/living/living as anything in used_things) + living.gib() + + return + +/datum/ritual/ashwalker/population/check_contents(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + for(var/mob/living/living as anything in used_things) + if(living.stat != DEAD) + to_chat(invoker, "Существа должны быть мертвы.") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/population/do_ritual(mob/living/carbon/human/invoker) + new /obj/effect/mob_spawn/human/ash_walker/shaman(ritual_object.loc) + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/population/disaster(mob/living/carbon/human/invoker) + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(!isashwalker(human) || !prob(disaster_prob)) + continue + + if(!isturf(human.loc)) + continue + + var/datum/effect_system/smoke_spread/smoke = new + smoke.set_up(5, FALSE, get_turf(human.loc)) + smoke.start() + + for(var/obj/item/obj as anything in human.get_equipped_items(TRUE, TRUE)) + human.drop_item_ground(obj) + + return + +/datum/ritual/ashwalker/population/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/magic/demon_consume.ogg', 50, TRUE) + var/datum/effect_system/smoke_spread/smoke = new + smoke.set_up(5, FALSE, get_turf(ritual_object.loc)) + smoke.start() + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/magic/cult_spell.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/magic/teleport_diss.ogg', 50, TRUE) + + return . + +/datum/ritual/ashwalker/soul + name = "Soul ritual" + extra_invokers = 3 + cooldown_after_cast = 1200 SECONDS + cast_time = 60 SECONDS + required_things = list( + /mob/living/carbon/human = 3, + /obj/item/stack/sheet/animalhide/ashdrake = 1 + ) + +/datum/ritual/ashwalker/soul/check_invokers(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + if(!isashwalkershaman(invoker)) + disaster_prob = 40 + fail_chance = 70 + + return TRUE + +/datum/ritual/ashwalker/population/del_things() + var/obj/item/stack/sheet/animalhide/ashdrake/stack = locate() in used_things + stack.use(1) + + for(var/mob/living/living in used_things) + living.gib() + + return + +/datum/ritual/ashwalker/soul/check_contents(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + for(var/mob/living/living in used_things) + if(living.stat != DEAD) + to_chat(invoker, "Существа должны быть мертвы.") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/soul/do_ritual(mob/living/carbon/human/invoker) + var/datum/effect_system/smoke_spread/smoke = new + smoke.set_up(5, FALSE, get_turf(invoker.loc)) + smoke.start() + invoker.set_species(/datum/species/unathi/draconid) + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/soul/disaster(mob/living/carbon/human/invoker) + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(!isashwalker(human) || !prob(disaster_prob)) + continue + + if(!isturf(human.loc)) + continue + + human.SetKnockdown(10 SECONDS) + var/turf/turf = human.loc + new /obj/effect/hotspot(turf) + turf.hotspot_expose(700, 50, 1) + + return + +/datum/ritual/ashwalker/soul/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/effects/whoosh.ogg', 50, TRUE) + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/effects/bamf.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/effects/blobattack.ogg', 50, TRUE) + + return . + +/datum/ritual/ashwalker/transmutation + name = "Transmutation ritual" + cooldown_after_cast = 120 SECONDS + cast_time = 10 SECONDS + required_things = list( + /obj/item/stack/ore = 10 + ) + +/datum/ritual/ashwalker/transmutation/check_invokers(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + if(!isashwalkershaman(invoker)) + disaster_prob = 30 + fail_chance = 50 + + return TRUE + +/datum/ritual/ashwalker/transmutation/do_ritual(mob/living/carbon/human/invoker) + var/list/ore_types = list() + + for(var/obj/item/stack/ore/ore as anything in subtypesof(/obj/item/stack/ore)) + LAZYADD(ore_types, ore) + + var/obj/item/stack/ore/ore = pick(ore_types) + ore = new(get_turf(ritual_object)) + ore.add(10) + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/transmutation/disaster(mob/living/carbon/human/invoker) + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(!isashwalker(human) || !prob(disaster_prob)) + continue + + if(!isturf(human.loc)) + continue + + human.SetKnockdown(10 SECONDS) + var/turf/turf = human.loc + new /obj/effect/hotspot(turf) + turf.hotspot_expose(700, 50, 1) + + return + +/datum/ritual/ashwalker/transmutation/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/effects/bin_close.ogg', 50, TRUE) + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/magic/cult_spell.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/magic/knock.ogg', 50, TRUE) + + return . + +/datum/ritual/ashwalker/interrogation + name = "Interrogation ritual" + cooldown_after_cast = 50 SECONDS + shaman_only = TRUE + cast_time = 10 SECONDS + required_things = list( + /mob/living/carbon/human = 1 + ) + +/datum/ritual/ashwalker/interrogation/check_invokers(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + if(invoker.health > 10) + disaster_prob = 30 + fail_chance = 30 + + return TRUE + +/datum/ritual/ashwalker/interrogation/check_contents(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + var/mob/living/carbon/human/human = locate() in used_things + if(!human || QDELETED(human)) + return RITUAL_FAILED_ON_PROCEED + + if(human.stat == DEAD || !human.mind) + to_chat(invoker, "Гуманоид должен быть жив и иметь разум.") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/interrogation/do_ritual(mob/living/carbon/human/invoker) + var/obj/effect/proc_holder/spell/empath/empath = new + if(!empath.cast(used_things, invoker)) + return RITUAL_FAILED_ON_PROCEED + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/interrogation/disaster(mob/living/carbon/human/invoker) + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(!isashwalker(human)) + continue + + if(!isturf(human.loc)) + continue + + var/turf/turf = human.loc + to_chat(human, "HONK") + SEND_SOUND(turf, sound('sound/items/airhorn.ogg')) + human.AdjustHallucinate(150 SECONDS) + human.EyeBlind(5 SECONDS) + var/datum/effect_system/smoke_spread/smoke = new + smoke.set_up(5, FALSE, turf) + smoke.start() + + return + +/datum/ritual/ashwalker/interrogation/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/effects/anvil_start.ogg', 50, TRUE) + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/effects/hulk_hit_airlock.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/effects/forge_destroy.ogg', 50, TRUE) + + return . + +/datum/ritual/ashwalker/creation + name = "Creation ritual" + cooldown_after_cast = 150 SECONDS + shaman_only = TRUE + extra_invokers = 2 + cast_time = 60 SECONDS + required_things = list( + /mob/living/carbon/human = 2 + ) + +/datum/ritual/ashwalker/creation/check_invokers(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + for(var/mob/living/carbon/human/human as anything in invokers) + if(human.stat != UNCONSCIOUS) + disaster_prob += 20 + fail_chance += 20 + + return TRUE + +/datum/ritual/ashwalker/creation/check_contents(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + for(var/mob/living/carbon/human/human as anything in used_things) + if(human.stat != DEAD) + to_chat(invoker, "Гуманоиды должны быть мертвы.") + return FALSE + + if(!isashwalker(human)) + to_chat(invoker, "Гуманоиды должны быть пеплоходцами.") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/creation/do_ritual(mob/living/carbon/human/invoker) + for(var/mob/living/mob as anything in subtypesof(/mob/living/simple_animal/hostile/asteroid)) + if(prob(30)) + mob = new(get_turf(ritual_object)) + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/creation/disaster(mob/living/carbon/human/invoker) + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(!isashwalker(human) || !prob(disaster_prob)) + continue + + if(!isturf(human.loc)) + continue + + human.SetKnockdown(10 SECONDS) + var/turf/turf = human.loc + new /obj/effect/hotspot(turf) + turf.hotspot_expose(700, 50, 1) + + return + +/datum/ritual/ashwalker/creation/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/magic/demon_consume.ogg', 50, TRUE) + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/magic/blind.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/magic/castsummon.ogg', 50, TRUE) + + return . + +/datum/ritual/ashwalker/command + name = "Command ritual" + cooldown_after_cast = 150 SECONDS + shaman_only = TRUE + disaster_prob = 35 + extra_invokers = 1 + cast_time = 60 SECONDS + required_things = list( + /mob/living/simple_animal = 1, + /obj/item/organ/internal/regenerative_core = 1, + /obj/item/reagent_containers/food/snacks/monstermeat/spiderleg = 1 + ) + +/datum/ritual/ashwalker/command/check_contents(mob/living/carbon/human/invoker) + . = ..() + + if(!.) + return FALSE + + for(var/mob/living/simple_animal/living as anything in used_things) + if(living.client) + to_chat(invoker, "Существо должно быть бездушным.") + return FALSE + + if(living.sentience_type == SENTIENCE_BOSS) + to_chat(invoker, "Ритуал не может воздействовать на мегафауну.") + return FALSE + + if(living.stat != DEAD) + to_chat(invoker, "Существа должны быть мертвы.") + return FALSE + + return TRUE + +/datum/ritual/ashwalker/command/do_ritual(mob/living/carbon/human/invoker) + var/mob/living/simple_animal/animal = locate() in used_things + + if(QDELETED(animal)) + return RITUAL_FAILED_ON_PROCEED + + animal.faction = invoker.faction + animal.revive() + var/list/candidates = SSghost_spawns.poll_candidates("Вы хотите сыграть за раба пеплоходцев?", ROLE_SENTIENT, TRUE, source = animal) + + if(!LAZYLEN(candidates) || QDELETED(animal)) // no travelling into nullspace + return RITUAL_FAILED_ON_PROCEED // no mercy guys. But you got friendly creature + + var/mob/mob = pick(candidates) + animal.key = mob.key + animal.universal_speak = 1 + animal.sentience_act() + animal.can_collar = 1 + animal.maxHealth = max(animal.maxHealth, 200) + animal.del_on_death = FALSE + animal.master_commander = invoker + + animal.mind.store_memory("Мой хозяин [invoker.name], выполню [genderize_ru(invoker.gender, "его", "её", "этого", "их")] цели любой ценой!") + to_chat(animal, chat_box_green("Вы - раб пеплоходцев. Всегда подчиняйтесь и помогайте им.")) + add_game_logs("стал питомцем игрока [key_name(invoker)]", animal) + + return RITUAL_SUCCESSFUL + +/datum/ritual/ashwalker/command/disaster(mob/living/carbon/human/invoker) + for(var/mob/living/carbon/human/human in SSmobs.clients_by_zlevel[invoker.z]) + if(!isashwalker(human) || !prob(disaster_prob)) + continue + + if(!isturf(human.loc)) + continue + + var/datum/effect_system/smoke_spread/smoke = new + smoke.set_up(5, FALSE, get_turf(human.loc)) + smoke.start() + + var/mob/living/simple_animal/mob = locate() in used_things + qdel(mob) + + new /mob/living/simple_animal/hostile/asteroid/goliath/beast/ancient(get_turf(ritual_object)) + + return + +/datum/ritual/ashwalker/command/handle_ritual_object(bitflags, silent = FALSE) + . = ..(bitflags, TRUE) + + switch(.) + if(RITUAL_ENDED) + playsound(ritual_object.loc, 'sound/magic/demon_consume.ogg', 50, TRUE) + if(RITUAL_STARTED) + playsound(ritual_object.loc, 'sound/magic/invoke_general.ogg', 50, TRUE) + if(RITUAL_FAILED) + playsound(ritual_object.loc, 'sound/magic/castsummon.ogg', 50, TRUE) + + return . + diff --git a/code/datums/status_effects/debuffs.dm b/code/datums/status_effects/debuffs.dm index 839a3cdf909..d95b87db1f1 100644 --- a/code/datums/status_effects/debuffs.dm +++ b/code/datums/status_effects/debuffs.dm @@ -57,6 +57,40 @@ /datum/status_effect/pacifism/on_remove() REMOVE_TRAIT(owner, TRAIT_PACIFISM, id) +/datum/status_effect/fang_exhaust + id = "fang_exhaust" + alert_type = null + duration = 2 SECONDS + var/modifier + +/datum/status_effect/fang_exhaust/on_creation(mob/living/simple_animal/new_owner, modifier = 1.1) + if(!istype(new_owner)) + return FALSE + + src.modifier = modifier + return ..() + +/datum/status_effect/fang_exhaust/on_apply() + var/mob/living/simple_animal/new_owner = owner + + for(var/thing in new_owner.damage_coeff) + if(!new_owner.damage_coeff[thing]) + continue + + new_owner.damage_coeff[thing] *= modifier + + return ..() + +/datum/status_effect/fang_exhaust/on_remove() + var/mob/living/simple_animal/new_owner = owner + + for(var/thing in new_owner.damage_coeff) + if(!new_owner.damage_coeff[thing]) + continue + + new_owner.damage_coeff[thing] /= modifier + + return ..() /datum/status_effect/shadow_boxing id = "shadow barrage" diff --git a/code/game/objects/effects/decals/Cleanable/misc.dm b/code/game/objects/effects/decals/Cleanable/misc.dm index 4c9df1cec23..82e17d98a83 100644 --- a/code/game/objects/effects/decals/Cleanable/misc.dm +++ b/code/game/objects/effects/decals/Cleanable/misc.dm @@ -257,3 +257,21 @@ /obj/effect/decal/cleanable/glass/plasma icon_state = "plasmatiny" + +/obj/effect/decal/cleanable/ashrune + name = "Ash rune" + desc = "A rune drawn in ash." + icon = 'icons/effects/ashwalker_rune.dmi' + icon_state = "AshRuneFilled" + anchored = TRUE + mergeable_decal = FALSE + mouse_opacity = MOUSE_OPACITY_ICON + +/obj/effect/decal/cleanable/ashrune/ComponentInitialize() + AddComponent( \ + /datum/component/ritual_object, \ + /datum/ritual/ashwalker, \ + ) + +/obj/effect/decal/cleanable/ashrune/is_cleanable() + return FALSE diff --git a/code/game/objects/items/weapons/cigs.dm b/code/game/objects/items/weapons/cigs.dm index 3b1b3cc915c..a548b051206 100644 --- a/code/game/objects/items/weapons/cigs.dm +++ b/code/game/objects/items/weapons/cigs.dm @@ -331,6 +331,9 @@ LIGHTERS ARE IN LIGHTERS.DM /obj/item/clothing/mask/cigarette/shadyjims list_reagents = list("nicotine" = 40, "lipolicide" = 7.5, "ammonia" = 2, "atrazine" = 1, "toxin" = 1.5) +/obj/item/clothing/mask/cigarette/richard + list_reagents = list("nicotine" = 40, "epinephrine" = 5, "absinthe" = 5) + /obj/item/clothing/mask/cigarette/rollie name = "rollie" desc = "A roll of dried plant matter wrapped in thin paper." diff --git a/code/game/objects/items/weapons/storage/fancy.dm b/code/game/objects/items/weapons/storage/fancy.dm index c491bd96b2a..b1c8f7b30f9 100644 --- a/code/game/objects/items/weapons/storage/fancy.dm +++ b/code/game/objects/items/weapons/storage/fancy.dm @@ -335,6 +335,11 @@ item_state = "upliftpacket" cigarette_type = /obj/item/clothing/mask/cigarette/menthol +/obj/item/storage/fancy/cigarettes/cigpack_richard + name = "\improper Richard & Co cigarettes" + desc = "Курят только отчаянные." + cigarette_type = /obj/item/clothing/mask/cigarette/richard + /obj/item/storage/fancy/cigarettes/cigpack_robust name = "\improper Robust packet" desc = "Smoked by the robust." diff --git a/code/modules/crafting/recipes.dm b/code/modules/crafting/recipes.dm index cd2fd7af4fd..4dcd10981ba 100644 --- a/code/modules/crafting/recipes.dm +++ b/code/modules/crafting/recipes.dm @@ -1519,6 +1519,39 @@ subcategory = CAT_WEAPON always_availible = FALSE +/datum/crafting_recipe/pickaxe + name = "Iron pickaxe" + reqs = list( + /obj/item/stack/sheet/wood = 2, + /obj/item/stack/sheet/metal = 5 + ) + result = list(/obj/item/pickaxe) + category = CAT_PRIMAL + +/datum/crafting_recipe/pickaxe/silver + name = "Silver pickaxe" + reqs = list( + /obj/item/stack/sheet/wood = 2, + /obj/item/stack/sheet/mineral/silver = 5 + ) + result = list(/obj/item/pickaxe/silver) + +/datum/crafting_recipe/pickaxe/golden + name = "Golden pickaxe" + reqs = list( + /obj/item/stack/sheet/wood = 2, + /obj/item/stack/sheet/mineral/gold = 5 + ) + result = list(/obj/item/pickaxe/gold) + +/datum/crafting_recipe/pickaxe/diamond + name = "Diamond pickaxe" + reqs = list( + /obj/item/stack/sheet/wood = 2, + /obj/item/stack/sheet/mineral/diamond = 5 + ) + result = list(/obj/item/pickaxe/diamond) + /datum/crafting_recipe/drone name = "Inactive Drone" result = list(/obj/item/inactive_drone) diff --git a/code/modules/mining/equipment/kinetic_crusher.dm b/code/modules/mining/equipment/kinetic_crusher.dm index 978864203c5..822765b8858 100644 --- a/code/modules/mining/equipment/kinetic_crusher.dm +++ b/code/modules/mining/equipment/kinetic_crusher.dm @@ -361,6 +361,70 @@ if(.) H.charge_time += bonus_value +/// Massive eyed tentacle +/obj/item/crusher_trophy/eyed_tentacle + name = "Massive eyed tentacle" + desc = "Большое и глазастое щупальце древнего голиафа. Может быть установлено как трофей крашера." + icon_state = "ancient_goliath_tentacle" + denied_type = /obj/item/crusher_trophy/eyed_tentacle + bonus_value = 1 + +/obj/item/crusher_trophy/eyed_tentacle/effect_desc() + return "causes kinetic crusher to deal 50% more damage if target has more than 90% HP" + +/obj/item/crusher_trophy/eyed_tentacle/on_melee_hit(mob/living/target, mob/living/user) + var/procent = (target.health / target.maxHealth) * 100 + if(procent < 90) + return + + var/obj/item/twohanded/kinetic_crusher/crusher = user.get_active_hand() + if(!crusher) + return + + target.apply_damage(crusher.force * bonus_value, crusher.damtype, user.zone_selected) + +/// Poison fang +/obj/item/crusher_trophy/fang + name = "Poison fang" + desc = "Уродливый и отравленный коготь. Может быть установлен как трофей крашера." + icon_state = "ob_gniga" + denied_type = /obj/item/crusher_trophy/fang + bonus_value = 1.1 + +/obj/item/crusher_trophy/fang/effect_desc() + return "causes fauna to get 10% more damage after mark destroyed for 2 seconds" + +/obj/item/crusher_trophy/fang/on_mark_detonation(mob/living/target, mob/living/user) + target.apply_status_effect(STATUS_EFFECT_FANG_EXHAUSTION, bonus_value) + +/// Frost gland +/obj/item/crusher_trophy/gland + name = "Frost gland" + desc = "Замороженная железа. Может быть установлена как трофей крашера." + icon_state = "ice_gniga" + denied_type = /obj/item/crusher_trophy/gland + bonus_value = 0.9 + +/obj/item/crusher_trophy/gland/effect_desc() + return "causes fauna to deal 10% less damage when marked" + +/obj/item/crusher_trophy/gland/on_mark_application(mob/living/simple_animal/target, datum/status_effect/crusher_mark/mark, had_mark) + if(had_mark) + return + + if(!istype(target)) + return + + target.melee_damage_lower *= bonus_value + target.melee_damage_upper *= bonus_value + +/obj/item/crusher_trophy/gland/on_mark_detonation(mob/living/simple_animal/target, mob/living/user) + if(!istype(target)) // double check + return + + target.melee_damage_lower /= bonus_value + target.melee_damage_upper /= bonus_value + //blood-drunk hunter /obj/item/crusher_trophy/miner_eye name = "eye of a blood-drunk hunter" diff --git a/code/modules/mob/living/simple_animal/hostile/mining/goliath.dm b/code/modules/mob/living/simple_animal/hostile/mining/goliath.dm index da1aa23923b..b9a6169ab4b 100644 --- a/code/modules/mob/living/simple_animal/hostile/mining/goliath.dm +++ b/code/modules/mob/living/simple_animal/hostile/mining/goliath.dm @@ -244,6 +244,7 @@ speed = 4 pre_attack_icon = "Goliath_preattack" throw_message = "does nothing to the rocky hide of the" + crusher_loot = /obj/item/crusher_trophy/eyed_tentacle loot = list(/obj/item/stack/sheet/animalhide/goliath_hide) //A throwback to the asteroid days butcher_results = list(/obj/item/reagent_containers/food/snacks/monstermeat/goliath = 2, /obj/item/stack/sheet/bone = 2) crusher_drop_mod = 30 diff --git a/code/modules/mob/living/simple_animal/hostile/mining/marrow_weaver.dm b/code/modules/mob/living/simple_animal/hostile/mining/marrow_weaver.dm index c70328f1574..2fd8b9fe08b 100644 --- a/code/modules/mob/living/simple_animal/hostile/mining/marrow_weaver.dm +++ b/code/modules/mob/living/simple_animal/hostile/mining/marrow_weaver.dm @@ -7,6 +7,7 @@ icon_aggro = "weaver" icon_dead = "weaver_dead" throw_message = "bounces harmlessly off the" + crusher_loot = /obj/item/crusher_trophy/fang butcher_results = list(/obj/item/stack/ore/uranium = 2, /obj/item/stack/sheet/bone = 2, /obj/item/stack/sheet/sinew = 1, /obj/item/stack/sheet/animalhide/weaver_chitin = 3, /obj/item/reagent_containers/food/snacks/monstermeat/spiderleg = 2) loot = list() attacktext = "кусает" //can we revert all translation in our code? @@ -136,15 +137,20 @@ /mob/living/simple_animal/hostile/asteroid/marrowweaver/frost name = "frostbite weaver" desc = "A big, angry, venomous ice spider. It likes to snack on bone marrow. Its preferred food source is you." + icon_state = "weaver_ice" icon_living = "weaver_ice" icon_aggro = "weaver_ice" icon_dead = "weaver_ice_dead" + melee_damage_lower = 10 //stronger venom, but weaker attack. melee_damage_upper = 13 + poison_type = "frostoil" poison_per_bite = 5 + crusher_loot = /obj/item/crusher_trophy/gland + /mob/living/simple_animal/hostile/asteroid/marrowweaver/tendril fromtendril = TRUE diff --git a/icons/effects/ashwalker_rune.dmi b/icons/effects/ashwalker_rune.dmi new file mode 100644 index 00000000000..3c703403738 Binary files /dev/null and b/icons/effects/ashwalker_rune.dmi differ diff --git a/icons/obj/lavaland/artefacts.dmi b/icons/obj/lavaland/artefacts.dmi index 178610b7424..4fc1f5cba29 100644 Binary files a/icons/obj/lavaland/artefacts.dmi and b/icons/obj/lavaland/artefacts.dmi differ diff --git a/paradise.dme b/paradise.dme index 7834783ec6b..c99eb0ff96d 100644 --- a/paradise.dme +++ b/paradise.dme @@ -39,6 +39,7 @@ #include "code\__DEFINES\blob.dm" #include "code\__DEFINES\borer.dm" #include "code\__DEFINES\bots.dm" +#include "code\__DEFINES\rituals.dm" #include "code\__DEFINES\byond_tracy.dm" #include "code\__DEFINES\callbacks.dm" #include "code\__DEFINES\cargo_quests.dm" @@ -371,6 +372,7 @@ #include "code\controllers\subsystem\tickets\mentor_tickets.dm" #include "code\controllers\subsystem\tickets\tickets.dm" #include "code\datums\action.dm" +#include "code\datums\rituals.dm" #include "code\datums\ai_law_sets.dm" #include "code\datums\ai_laws.dm" #include "code\datums\armor.dm" @@ -448,6 +450,7 @@ #include "code\datums\components\conveyor_movement.dm" #include "code\datums\components\cross_shock.dm" #include "code\datums\components\decal.dm" +#include "code\datums\components\ritual_object.dm" #include "code\datums\components\defibrillator.dm" #include "code\datums\components\drift.dm" #include "code\datums\components\ducttape.dm"