-
Notifications
You must be signed in to change notification settings - Fork 67
/
Copy pathkneecapping.dm
86 lines (66 loc) · 3.78 KB
/
kneecapping.dm
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
/**
* Kneecapping element replaces the item's secondary attack with an aimed attack at the kneecaps under certain circumstances.
*
* Element is incompatible with non-items. Requires the parent item to have a force equal to or greater than WOUND_MINIMUM_DAMAGE.
* Also requires that the parent can actually get past pre_secondary_attack without the attack chain cancelling.
*
* Kneecapping attacks have a wounding bonus between severe and critical+10 wound thresholds. Without some serious wound protecting
* armour this all but guarantees a wound of some sort. The attack is directed specifically at a limb and the limb takes the damage.
*
* Requires the attacker to be aiming for either leg zone, which will be targetted specifically. They will than have a 3-second long
* do_mob before executing the attack.
*
* Kneecapping requires the target to either be on the floor, immobilised or buckled to something. And also to have an appropriate leg.
*
* Passing all the checks will cancel the entire attack chain.
*/
/datum/element/kneecapping
/datum/element/kneecapping/Attach(datum/target)
if(!isitem(target))
stack_trace("Kneecapping element added to non-item object: \[[target]\]")
return ELEMENT_INCOMPATIBLE
var/obj/item/target_item = target
if(target_item.force < WOUND_MINIMUM_DAMAGE)
stack_trace("Kneecapping element added to item with too little force to wound: \[[target]\]")
return ELEMENT_INCOMPATIBLE
. = ..()
if(. == ELEMENT_INCOMPATIBLE)
return
RegisterSignal(target, COMSIG_ITEM_ATTACK_SECONDARY , PROC_REF(try_kneecap_target))
/datum/element/kneecapping/Detach(datum/target)
UnregisterSignal(target, COMSIG_ITEM_ATTACK_SECONDARY)
return ..()
/**
* Signal handler for COMSIG_ITEM_ATTACK_SECONDARY. Does checks for pacifism, zones and target state before either returning nothing
* if the special attack could not be attempted, performing the ordinary attack procs instead - Or cancelling the attack chain if
* the attack can be started.
*/
/datum/element/kneecapping/proc/try_kneecap_target(obj/item/source, mob/living/carbon/target, mob/attacker, params)
SIGNAL_HANDLER
if((attacker.zone_selected != BODY_ZONE_L_LEG) && (attacker.zone_selected != BODY_ZONE_R_LEG))
return
if(HAS_TRAIT(attacker, TRAIT_PACIFISM))
return
if(!iscarbon(target))
return
if(!target.buckled && !HAS_TRAIT(target, TRAIT_FLOORED) && !HAS_TRAIT(target, TRAIT_IMMOBILIZED))
return
var/obj/item/bodypart/leg = target.get_bodypart(attacker.zone_selected)
if(!leg)
return
. = COMPONENT_SECONDARY_CANCEL_ATTACK_CHAIN
INVOKE_ASYNC(src, PROC_REF(do_kneecap_target), source, leg, target, attacker)
/**
* After a short do_mob, attacker applies damage to the given leg with a significant wounding bonus, applying the weapon's force as damage.
*/
/datum/element/kneecapping/proc/do_kneecap_target(obj/item/weapon, obj/item/bodypart/leg, mob/living/carbon/target, mob/attacker)
if(LAZYACCESS(attacker.do_afters, weapon))
return
attacker.visible_message(span_warning("[attacker] carefully aims [attacker.p_their()] [weapon] for a swing at [target]'s kneecaps!"), span_danger("You carefully aim \the [weapon] for a swing at [target]'s kneecaps!"))
log_combat(attacker, target, "started aiming a swing to break the kneecaps of", weapon)
if(do_after(attacker, target, 3 SECONDS, DO_PUBLIC, interaction_key = weapon, display = weapon))
attacker.visible_message(span_warning("[attacker] swings [attacker.p_their()] [weapon] at [target]'s kneecaps!"), span_danger("You swing \the [weapon] at [target]'s kneecaps!"))
leg.create_wound_easy(/datum/wound/puncture/gaping, 30)
log_combat(attacker, target, "broke the kneecaps of", weapon)
attacker.do_attack_animation(target, used_item = weapon)
playsound(source = get_turf(weapon), soundin = weapon.get_hitsound(), vol = weapon.get_clamped_volume(), vary = TRUE)