mirror of
https://github.com/Bubberstation/Bubberstation.git
synced 2026-02-08 23:39:32 +00:00
## About The Pull Request This PR does many things, I'll try to explain the basic/background stuff to the main thing first: 1. Adds a new remote that allows a human to function like an AI. It controls a fly that will fly around the station slowly, and when it reaches a machine then the person can interact with it as if they were an AI. This required changing a lot of silicon/AI checks with one that also checks for this remote, and some messing with shared ui state. 2. Moves req_access from the obj and bot to ``/atom/movable`` which lets it be shared between the two, no more copy-paste and one side lacking features/checks/signals the other has. 3. Adds a check for AI config for AI-related station traits, which was lacking prior Now for the good part... Adds a new station trait that replaces the AI with a Human. This person is equipped with an AI headset (including Binary), an advanced camera console, an omni door wand, the machine controller, and their laws. They are immune to the SAT's turrets (even if set to target borgs) and are slow outside of the SAT, mimicing the actions of the AI. They interact with the world through their advanced camera console, which allows them to do most AI stuff needed, and the holopad they can connect to without having to ring first (like Command can). They are given a paper with the laws they must follow, but since they are human they are able to bend it. Cyborgs that run the default lawset are "slaved" to them via an unremovable law 0, so the Human AI can bend the laws if they really need to (for their own survival n such), and make the cyborgs obey their commands above laws, but in general this shouldn't be a frequent occurrence. This does take into account the unique AI trait, so it's not guaranteed Asimov. When this station trait rolls, all Intellicards, AI uploads, and AI core boards are destroyed and are unresearchable. They can be spawned by admins in-game if necessary. Maybe in the future we can also exclude Oldstation from this but I haven't really decided. Extra perks: Human AI spawns with a Robotic voicebox (unless they are a body purist) and teleport blocking implant, so they can't use teleporters to bypass their on-station slowdown. They also have an infinite laser pointer that can be used to blind through their camera console. This is unfortunately nerfed from the recent borg balance PR that removed its stun. This was meant to be the alternative to no longer being able to permanently lock borgs down like AIs can (or more than one, for that matter). They aren't affected by Roburgers, Acid, and Fuel's toxicity. Bots salute them like they do Beepsky (which is now a trait) They spawn with SyndEye to replace the AI's tracking ability They do not have a bank account ### The machine remote The machine remote has a little fly in it that flies to the machines it is pointed to, working as the arms and legs of the Human AI. It scans the machine and punches in the action the AI does, and is how the AI accesses basically anything. This fly slowly moves from one machine to the next, and can be recalled with Alt Click. It works on machines and bots. ### Video (Low quality to fit Github) https://github.com/tgstation/tgstation/assets/53777086/e16509f8-8bed-42b5-9fbf-7e37165a11e8 ## Why It's Good For The Game I've seen a funny screenshot one day of a person replacing the AI by using a bunch of door remotes, camera console, crew monitoring console, and a few other things. I've been thinking about that for a few years and really wanted to make it official if not easier to make possible, because it is an incredibly funny interaction. This makes it a reality, and while they aren't as powerful as regular AIs, I think it makes for better and funnier in-game moments. With the same weight as Cargorilla (1), I hope this wouldn't be rolling too often and ruin rounds, but instead show off the different capabilities that Humans and AIs can do, to do the job of an AI. You win some you lose some. ## Changelog 🆑 JohnFulpWillard, Tattax add: Adds a new station trait job: The Human AI. /🆑 --------- Co-authored-by: MrMelbert <51863163+MrMelbert@users.noreply.github.com>
1326 lines
52 KiB
Plaintext
1326 lines
52 KiB
Plaintext
|
|
/obj/machinery/hydroponics
|
|
name = "hydroponics tray"
|
|
desc = "A basin used to grow plants in."
|
|
icon = 'icons/obj/service/hydroponics/equipment.dmi'
|
|
icon_state = "hydrotray"
|
|
density = TRUE
|
|
pass_flags_self = PASSMACHINE | LETPASSTHROW
|
|
pixel_z = 8
|
|
obj_flags = CAN_BE_HIT | UNIQUE_RENAME
|
|
circuit = /obj/item/circuitboard/machine/hydroponics
|
|
use_power = NO_POWER_USE
|
|
///The amount of water in the tray (max 100)
|
|
var/waterlevel = 0
|
|
///The maximum amount of water in the tray
|
|
var/maxwater = 100
|
|
///How many units of nutrients will be drained in the tray.
|
|
var/nutridrain = 1
|
|
///The maximum nutrient reagent container size of the tray.
|
|
var/maxnutri = 20
|
|
///The amount of pests in the tray (max 10)
|
|
var/pestlevel = 0
|
|
///The amount of weeds in the tray (max 10)
|
|
var/weedlevel = 0
|
|
///Nutriment's effect on yield
|
|
var/yieldmod = 1
|
|
///Nutriment's effect on mutations
|
|
var/mutmod = 1
|
|
///Toxicity in the tray?
|
|
var/toxic = 0
|
|
///Current age
|
|
var/age = 0
|
|
///The status of the plant in the tray. Whether it's harvestable, alive, missing or dead.
|
|
var/plant_status = HYDROTRAY_NO_PLANT
|
|
///Its health
|
|
var/plant_health
|
|
///Last time it was harvested
|
|
var/lastproduce = 0
|
|
///Used for timing of cycles.
|
|
var/lastcycle = 0
|
|
///About 10 seconds / cycle
|
|
var/cycledelay = HYDROTRAY_CYCLE_DELAY
|
|
///The currently planted seed
|
|
var/obj/item/seeds/myseed
|
|
///Obtained from the quality of the parts used in the tray, determines nutrient drain rate.
|
|
var/rating = 1
|
|
///Can it be unwrenched to move?
|
|
var/unwrenchable = TRUE
|
|
///Have we been visited by a bee recently, so bees dont overpollinate one plant
|
|
var/recent_bee_visit = FALSE
|
|
///The last user to add a reagent to the tray, mostly for logging purposes.
|
|
var/datum/weakref/lastuser
|
|
///If the tray generates nutrients and water on its own
|
|
var/self_sustaining = FALSE
|
|
///The icon state for the overlay used to represent that this tray is self-sustaining.
|
|
var/self_sustaining_overlay_icon_state = "gaia_blessing"
|
|
|
|
/obj/machinery/hydroponics/Initialize(mapload)
|
|
//ALRIGHT YOU DEGENERATES. YOU HAD REAGENT HOLDERS FOR AT LEAST 4 YEARS AND NONE OF YOU MADE HYDROPONICS TRAYS HOLD NUTRIENT CHEMS INSTEAD OF USING "Points".
|
|
//SO HERE LIES THE "nutrilevel" VAR. IT'S DEAD AND I PUT IT OUT OF IT'S MISERY. USE "reagents" INSTEAD. ~ArcaneMusic, accept no substitutes.
|
|
create_reagents(maxnutri, INJECTABLE)
|
|
if(mapload)
|
|
reagents.add_reagent(/datum/reagent/plantnutriment/eznutriment, max(maxnutri / 2, 10)) //Half filled nutrient trays for dirt trays to have more to grow with in prison/lavaland.
|
|
waterlevel = maxwater
|
|
. = ..()
|
|
|
|
var/static/list/hovering_item_typechecks = list(
|
|
/obj/item/plant_analyzer = list(
|
|
SCREENTIP_CONTEXT_LMB = "Scan tray stats",
|
|
SCREENTIP_CONTEXT_RMB = "Scan tray chemicals"
|
|
),
|
|
/obj/item/cultivator = list(
|
|
SCREENTIP_CONTEXT_LMB = "Remove weeds",
|
|
),
|
|
/obj/item/shovel = list(
|
|
SCREENTIP_CONTEXT_LMB = "Clear tray",
|
|
),
|
|
)
|
|
|
|
AddElement(/datum/element/contextual_screentip_item_typechecks, hovering_item_typechecks)
|
|
register_context()
|
|
|
|
/obj/machinery/hydroponics/add_context(
|
|
atom/source,
|
|
list/context,
|
|
obj/item/held_item,
|
|
mob/living/user,
|
|
)
|
|
|
|
// If we don't have a seed, we can't do much.
|
|
|
|
// The only option is to plant a new seed.
|
|
if(!myseed)
|
|
if(istype(held_item, /obj/item/seeds))
|
|
context[SCREENTIP_CONTEXT_LMB] = "Plant seed"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
return NONE
|
|
|
|
// If we DO have a seed, we can do a few things!
|
|
|
|
// With a hand we can harvest or remove dead plants
|
|
// If the plant's not in either state, we can't do much else, so early return.
|
|
if(isnull(held_item))
|
|
// Silicons can't interact with trays :frown:
|
|
if(HAS_SILICON_ACCESS(user))
|
|
return NONE
|
|
|
|
switch(plant_status)
|
|
if(HYDROTRAY_PLANT_DEAD)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Remove dead plant"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
if(HYDROTRAY_PLANT_HARVESTABLE)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Harvest plant"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
return NONE
|
|
|
|
// If the plant is harvestable, we can graft it with secateurs or harvest it with a plant bag.
|
|
if(plant_status == HYDROTRAY_PLANT_HARVESTABLE)
|
|
if(istype(held_item, /obj/item/secateurs))
|
|
context[SCREENTIP_CONTEXT_LMB] = "Graft plant"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
if(istype(held_item, /obj/item/storage/bag/plants))
|
|
context[SCREENTIP_CONTEXT_LMB] = "Harvest plant"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
// If the plant's in good health, we can shear it.
|
|
if(istype(held_item, /obj/item/geneshears) && plant_health > GENE_SHEAR_MIN_HEALTH)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Remove plant gene"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
// If we've got a charged somatoray, we can mutation lock it.
|
|
if(istype(held_item, /obj/item/gun/energy/floragun) && myseed.endurance > FLORA_GUN_MIN_ENDURANCE && LAZYLEN(myseed.mutatelist))
|
|
var/obj/item/gun/energy/floragun/flower_gun = held_item
|
|
if(flower_gun.cell.charge >= flower_gun.cell.maxcharge)
|
|
context[SCREENTIP_CONTEXT_LMB] = "Lock mutation"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
// Edibles and pills can be composted.
|
|
if(IS_EDIBLE(held_item) || istype(held_item, /obj/item/reagent_containers/pill))
|
|
context[SCREENTIP_CONTEXT_LMB] = "Compost"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
// And if a reagent container has water or plant fertilizer in it, we can use it on the plant.
|
|
if(is_reagent_container(held_item) && length(held_item.reagents.reagent_list))
|
|
var/datum/reagent/most_common_reagent = held_item.reagents.get_master_reagent()
|
|
context[SCREENTIP_CONTEXT_LMB] = "[istype(most_common_reagent, /datum/reagent/water) ? "Water" : "Feed"] plant"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
return NONE
|
|
|
|
/obj/machinery/hydroponics/constructable
|
|
name = "hydroponics tray"
|
|
icon = 'icons/obj/service/hydroponics/equipment.dmi'
|
|
icon_state = "hydrotray3"
|
|
|
|
/obj/machinery/hydroponics/constructable/Initialize(mapload)
|
|
. = ..()
|
|
AddComponent(/datum/component/simple_rotation)
|
|
AddComponent(/datum/component/plumbing/hydroponics)
|
|
AddComponent(/datum/component/usb_port, list(/obj/item/circuit_component/hydroponics))
|
|
AddComponent(/datum/component/fishing_spot, /datum/fish_source/hydro_tray)
|
|
|
|
/obj/machinery/hydroponics/constructable/RefreshParts()
|
|
. = ..()
|
|
var/tmp_capacity = 0
|
|
for (var/datum/stock_part/matter_bin/matter_bin in component_parts)
|
|
tmp_capacity += matter_bin.tier
|
|
for (var/datum/stock_part/servo/servo in component_parts)
|
|
rating = servo.tier
|
|
maxwater = tmp_capacity * 50 // Up to 300
|
|
maxnutri = (tmp_capacity * 5) + STATIC_NUTRIENT_CAPACITY // Up to 50 Maximum
|
|
reagents.maximum_volume = maxnutri
|
|
nutridrain = 1/rating
|
|
|
|
/obj/machinery/hydroponics/constructable/examine(mob/user)
|
|
. = ..()
|
|
. += span_notice("Use <b>Ctrl-Click</b> to activate autogrow. <b>RMB</b> to empty the tray's nutrients.")
|
|
if(in_range(user, src) || isobserver(user))
|
|
. += span_notice("The status display reads: Tray efficiency at <b>[rating*100]%</b>.")
|
|
|
|
/obj/machinery/hydroponics/constructable/add_context(
|
|
atom/source,
|
|
list/context,
|
|
obj/item/held_item,
|
|
mob/living/user,
|
|
)
|
|
|
|
// Constructible trays will always show that you can activate auto-grow with ctrl+click
|
|
. = ..()
|
|
context[SCREENTIP_CONTEXT_CTRL_LMB] = "Activate auto-grow"
|
|
return CONTEXTUAL_SCREENTIP_SET
|
|
|
|
/obj/machinery/hydroponics/Destroy()
|
|
if(myseed)
|
|
QDEL_NULL(myseed)
|
|
return ..()
|
|
|
|
/obj/machinery/hydroponics/Exited(atom/movable/gone)
|
|
. = ..()
|
|
if(!QDELETED(src) && gone == myseed)
|
|
set_seed(null, FALSE)
|
|
|
|
/obj/machinery/hydroponics/constructable/attackby(obj/item/I, mob/living/user, params)
|
|
if (!user.combat_mode)
|
|
// handle opening the panel
|
|
if(default_deconstruction_screwdriver(user, icon_state, icon_state, I))
|
|
return
|
|
if(default_deconstruction_crowbar(I))
|
|
return
|
|
|
|
return ..()
|
|
|
|
/// Special demand connector that consumes as normal, but redirects water into the magical water space.
|
|
/datum/component/plumbing/hydroponics
|
|
demand_connects = SOUTH
|
|
/// Alternate reagents container to buffer incoming water
|
|
var/datum/reagents/water_reagents
|
|
/// Actual parent reagents that has nutrients
|
|
var/datum/reagents/nutri_reagents
|
|
|
|
/datum/component/plumbing/hydroponics/Initialize(start=TRUE, _ducting_layer, _turn_connects=TRUE, datum/reagents/custom_receiver)
|
|
. = ..()
|
|
|
|
if(!istype(parent, /obj/machinery/hydroponics/constructable))
|
|
return COMPONENT_INCOMPATIBLE
|
|
|
|
var/obj/machinery/hydroponics/constructable/hydro_parent = parent
|
|
|
|
water_reagents = new(hydro_parent.maxwater)
|
|
water_reagents.my_atom = hydro_parent
|
|
|
|
nutri_reagents = reagents
|
|
|
|
/datum/component/plumbing/hydroponics/Destroy()
|
|
qdel(water_reagents)
|
|
nutri_reagents = null
|
|
return ..()
|
|
|
|
/datum/component/plumbing/hydroponics/send_request(dir)
|
|
var/obj/machinery/hydroponics/constructable/hydro_parent = parent
|
|
|
|
var/initial_nutri_amount = nutri_reagents.total_volume
|
|
if(initial_nutri_amount < nutri_reagents.maximum_volume)
|
|
// Well boy howdy, we have no way to tell a supply to not mix the water with everything else,
|
|
// So we'll let it leak in, and move the water over.
|
|
set_recipient_reagents_holder(nutri_reagents)
|
|
reagents = nutri_reagents
|
|
process_request(dir = dir)
|
|
|
|
// Move the leaked water from nutrients to... water
|
|
var/leaking_water_amount = nutri_reagents.get_reagent_amount(/datum/reagent/water)
|
|
if(leaking_water_amount)
|
|
nutri_reagents.trans_to(water_reagents, leaking_water_amount, target_id = /datum/reagent/water)
|
|
|
|
// We should only take MACHINE_REAGENT_TRANSFER every tick; this is the remaining amount we can take
|
|
var/remaining_transfer_amount = max(MACHINE_REAGENT_TRANSFER - (nutri_reagents.total_volume - initial_nutri_amount), 0)
|
|
|
|
// How much extra water we should gather this tick to try to fill the water tray.
|
|
var/extra_water_to_gather = clamp(hydro_parent.maxwater - hydro_parent.waterlevel - water_reagents.total_volume, 0, remaining_transfer_amount)
|
|
if(extra_water_to_gather > 0)
|
|
set_recipient_reagents_holder(water_reagents)
|
|
reagents = water_reagents
|
|
process_request(
|
|
amount = extra_water_to_gather,
|
|
reagent = /datum/reagent/water,
|
|
dir = dir,
|
|
)
|
|
|
|
// Now transfer all remaining water in that buffer and clear it out.
|
|
var/final_water_amount = water_reagents.total_volume
|
|
if(final_water_amount)
|
|
hydro_parent.adjust_waterlevel(round(final_water_amount))
|
|
// Using a pipe doesn't afford you extra water storage and the baseline behavior for trays is that excess water goes into the shadow realm.
|
|
water_reagents.del_reagent(/datum/reagent/water)
|
|
|
|
// Plumbing pauses if reagents is full.. so let's cheat and make sure it ticks unless both trays are happy
|
|
reagents = hydro_parent.waterlevel < hydro_parent.maxwater ? water_reagents : nutri_reagents
|
|
|
|
/obj/machinery/hydroponics/bullet_act(obj/projectile/Proj) //Works with the Somatoray to modify plant variables.
|
|
if(!myseed)
|
|
return ..()
|
|
if(istype(Proj , /obj/projectile/energy/flora/mut))
|
|
mutate()
|
|
else if(istype(Proj , /obj/projectile/energy/flora/yield))
|
|
return myseed.bullet_act(Proj)
|
|
else if(istype(Proj , /obj/projectile/energy/flora/evolution))
|
|
if(myseed)
|
|
if(LAZYLEN(myseed.mutatelist))
|
|
myseed.set_instability(myseed.instability/2)
|
|
mutatespecie()
|
|
else
|
|
return ..()
|
|
|
|
/obj/machinery/hydroponics/power_change()
|
|
. = ..()
|
|
if((machine_stat & NOPOWER) && self_sustaining)
|
|
set_self_sustaining(FALSE)
|
|
|
|
/obj/machinery/hydroponics/process(seconds_per_tick)
|
|
var/needs_update = FALSE // Checks if the icon needs updating so we don't redraw empty trays every time
|
|
|
|
if(self_sustaining)
|
|
if(powered())
|
|
adjust_waterlevel(rand(1,2) * seconds_per_tick * 0.5)
|
|
adjust_weedlevel(-0.5 * seconds_per_tick)
|
|
adjust_pestlevel(-0.5 * seconds_per_tick)
|
|
else
|
|
set_self_sustaining(FALSE)
|
|
visible_message(span_warning("[name]'s auto-grow functionality shuts off!"))
|
|
|
|
if(world.time > (lastcycle + cycledelay))
|
|
lastcycle = world.time
|
|
if(myseed && plant_status != HYDROTRAY_PLANT_DEAD)
|
|
// Advance age
|
|
age++
|
|
if(age < myseed.maturation)
|
|
lastproduce = age
|
|
|
|
needs_update = TRUE
|
|
|
|
|
|
//Nutrients//////////////////////////////////////////////////////////////
|
|
// Nutrients deplete at a constant rate, since new nutrients can boost stats far easier.
|
|
apply_chemicals(lastuser?.resolve())
|
|
if(self_sustaining)
|
|
reagents.remove_all(min(0.5, nutridrain))
|
|
else
|
|
reagents.remove_all(nutridrain)
|
|
|
|
// Lack of nutrients hurts non-weeds
|
|
if(reagents.total_volume <= 0 && !myseed.get_gene(/datum/plant_gene/trait/plant_type/weed_hardy))
|
|
adjust_plant_health(-rand(1,3))
|
|
|
|
//Photosynthesis/////////////////////////////////////////////////////////
|
|
// Lack of light hurts non-mushrooms
|
|
if(isturf(loc))
|
|
var/turf/currentTurf = loc
|
|
var/lightAmt = currentTurf.get_lumcount()
|
|
var/is_fungus = myseed.get_gene(/datum/plant_gene/trait/plant_type/fungal_metabolism)
|
|
if(lightAmt < (is_fungus ? 0.2 : 0.4))
|
|
adjust_plant_health((is_fungus ? -1 : -2) / rating)
|
|
|
|
//Water//////////////////////////////////////////////////////////////////
|
|
// Drink random amount of water
|
|
adjust_waterlevel(-rand(1,6) / rating)
|
|
|
|
// If the plant is dry, it loses health pretty fast, unless mushroom
|
|
if(waterlevel <= 10 && !myseed.get_gene(/datum/plant_gene/trait/plant_type/fungal_metabolism))
|
|
adjust_plant_health(-rand(0,1) / rating)
|
|
if(waterlevel <= 0)
|
|
adjust_plant_health(-rand(0,2) / rating)
|
|
|
|
// Sufficient water level and nutrient level = plant healthy but also spawns weeds
|
|
else if(waterlevel > 10 && reagents.total_volume > 0)
|
|
adjust_plant_health(rand(1,2) / rating)
|
|
if(myseed && prob(myseed.weed_chance))
|
|
adjust_weedlevel(myseed.weed_rate)
|
|
else if(prob(5)) //5 percent chance the weed population will increase
|
|
adjust_weedlevel(1 / rating)
|
|
|
|
//Toxins/////////////////////////////////////////////////////////////////
|
|
|
|
// Too much toxins cause harm, but when the plant drinks the contaiminated water, the toxins disappear slowly
|
|
if(toxic >= 40 && toxic < 80)
|
|
adjust_plant_health(-1 / rating)
|
|
adjust_toxic(-rating * 2)
|
|
else if(toxic >= 80) // I don't think it ever gets here tbh unless above is commented out
|
|
adjust_plant_health(-3)
|
|
adjust_toxic(-rating * 3)
|
|
|
|
//Pests & Weeds//////////////////////////////////////////////////////////
|
|
|
|
if(pestlevel >= 8)
|
|
if(!myseed.get_gene(/datum/plant_gene/trait/carnivory))
|
|
if(myseed.potency >= 30)
|
|
myseed.adjust_potency(-rand(2,6)) //Pests eat leaves and nibble on fruit, lowering potency.
|
|
myseed.set_potency(min((myseed.potency), CARNIVORY_POTENCY_MIN, MAX_PLANT_POTENCY))
|
|
else
|
|
adjust_plant_health(2 / rating)
|
|
adjust_pestlevel(-1 / rating)
|
|
|
|
else if(pestlevel >= 4)
|
|
if(!myseed.get_gene(/datum/plant_gene/trait/carnivory))
|
|
if(myseed.potency >= 30)
|
|
myseed.adjust_potency(-rand(1,4))
|
|
myseed.set_potency(min((myseed.potency), CARNIVORY_POTENCY_MIN, MAX_PLANT_POTENCY))
|
|
|
|
else
|
|
adjust_plant_health(1 / rating)
|
|
if(prob(50))
|
|
adjust_pestlevel(-1 / rating)
|
|
|
|
else if(pestlevel < 4 && myseed.get_gene(/datum/plant_gene/trait/carnivory))
|
|
if(prob(5))
|
|
adjust_pestlevel(-1 / rating)
|
|
|
|
// If it's a weed, it doesn't stunt the growth
|
|
if(weedlevel >= 5 && !myseed.get_gene(/datum/plant_gene/trait/plant_type/weed_hardy))
|
|
if(myseed.yield >= 3)
|
|
myseed.adjust_yield(-rand(1,2)) //Weeds choke out the plant's ability to bear more fruit.
|
|
myseed.set_yield(min((myseed.yield), WEED_HARDY_YIELD_MIN, MAX_PLANT_YIELD))
|
|
|
|
//This is the part with pollination
|
|
pollinate()
|
|
|
|
//This is where stability mutations exist now.
|
|
if(myseed.instability >= 80)
|
|
var/mutation_chance = myseed.instability - 75
|
|
mutate(0, 0, 0, 0, 0, 0, 0, mutation_chance, 0) //Scaling odds of a random trait or chemical
|
|
if(myseed.instability >= 60)
|
|
if(prob((myseed.instability)/2) && !self_sustaining && LAZYLEN(myseed.mutatelist) && !myseed.get_gene(/datum/plant_gene/trait/never_mutate)) //Minimum 30%, Maximum 50% chance of mutating every age tick when not on autogrow or having Prosophobic Inclination trait.
|
|
mutatespecie()
|
|
myseed.set_instability(myseed.instability/2)
|
|
if(myseed.instability >= 40)
|
|
if(prob(myseed.instability) && !myseed.get_gene(/datum/plant_gene/trait/stable_stats)) //No hardmutation if Symbiotic Resilience trait is present.
|
|
hardmutate()
|
|
if(myseed.instability >= 20 )
|
|
if(prob(myseed.instability) && !myseed.get_gene(/datum/plant_gene/trait/stable_stats)) //No mutation if Symbiotic Resilience trait is present.
|
|
mutate()
|
|
|
|
//Health & Age///////////////////////////////////////////////////////////
|
|
|
|
// Plant dies if plant_health <= 0
|
|
if(plant_health <= 0)
|
|
plantdies()
|
|
adjust_weedlevel(1 / rating) // Weeds flourish
|
|
|
|
// If the plant is too old, lose health fast
|
|
if(age > myseed.lifespan)
|
|
adjust_plant_health(-rand(1,5) / rating)
|
|
|
|
// Harvest code
|
|
if(age > myseed.production && (age - lastproduce) > myseed.production && plant_status == HYDROTRAY_PLANT_GROWING)
|
|
if(myseed && myseed.yield != -1) // Unharvestable shouldn't be harvested
|
|
set_plant_status(HYDROTRAY_PLANT_HARVESTABLE)
|
|
else
|
|
lastproduce = age
|
|
if(prob(5)) // On each tick, there's a 5 percent chance the pest population will increase
|
|
adjust_pestlevel(1 / rating)
|
|
else
|
|
if(waterlevel > 10 && reagents.total_volume > 0 && prob(10)) // If there's no plant, the percentage chance is 10%
|
|
adjust_weedlevel(1 / rating)
|
|
|
|
// Weeeeeeeeeeeeeeedddssss
|
|
if(weedlevel >= 10 && prob(50) && !self_sustaining) // At this point the plant is kind of fucked. Weeds can overtake the plant spot.
|
|
if(myseed && myseed.yield >= 3)
|
|
myseed.adjust_yield(-rand(1,2)) //Loses even more yield per tick, quickly dropping to 3 minimum.
|
|
myseed.set_yield(min((myseed.yield), WEED_HARDY_YIELD_MIN, MAX_PLANT_YIELD))
|
|
if(!myseed)
|
|
weedinvasion()
|
|
needs_update = 1
|
|
if (needs_update)
|
|
update_appearance()
|
|
|
|
if(myseed)
|
|
SEND_SIGNAL(myseed, COMSIG_SEED_ON_GROW, src)
|
|
|
|
return
|
|
|
|
/obj/machinery/hydroponics/update_appearance(updates)
|
|
. = ..()
|
|
if(self_sustaining)
|
|
set_light(3)
|
|
return
|
|
if(myseed?.get_gene(/datum/plant_gene/trait/glow)) // Hydroponics needs a refactor, badly.
|
|
var/datum/plant_gene/trait/glow/G = myseed.get_gene(/datum/plant_gene/trait/glow)
|
|
set_light(G.glow_range(myseed), G.glow_power(myseed), G.glow_color)
|
|
return
|
|
set_light(0)
|
|
|
|
/obj/machinery/hydroponics/update_name(updates)
|
|
. = ..()
|
|
if(myseed)
|
|
name = "[initial(name)] ([myseed.plantname])"
|
|
else
|
|
name = initial(name)
|
|
|
|
/obj/machinery/hydroponics/update_overlays()
|
|
. = ..()
|
|
if(myseed)
|
|
. += update_plant_overlay()
|
|
. += update_status_light_overlays()
|
|
|
|
if(self_sustaining && self_sustaining_overlay_icon_state)
|
|
. += mutable_appearance(icon, self_sustaining_overlay_icon_state)
|
|
|
|
/obj/machinery/hydroponics/proc/update_plant_overlay()
|
|
var/mutable_appearance/plant_overlay = mutable_appearance(myseed.growing_icon, layer = OBJ_LAYER + 0.01)
|
|
switch(plant_status)
|
|
if(HYDROTRAY_PLANT_DEAD)
|
|
plant_overlay.icon_state = myseed.icon_dead
|
|
if(HYDROTRAY_PLANT_HARVESTABLE)
|
|
if(!myseed.icon_harvest)
|
|
plant_overlay.icon_state = "[myseed.icon_grow][myseed.growthstages]"
|
|
else
|
|
plant_overlay.icon_state = myseed.icon_harvest
|
|
else
|
|
var/t_growthstate = clamp(round((age / myseed.maturation) * myseed.growthstages), 1, myseed.growthstages)
|
|
plant_overlay.icon_state = "[myseed.icon_grow][t_growthstate]"
|
|
plant_overlay.pixel_y = myseed.plant_icon_offset
|
|
return plant_overlay
|
|
|
|
/obj/machinery/hydroponics/proc/update_status_light_overlays()
|
|
. = list()
|
|
if(waterlevel <= 10)
|
|
. += mutable_appearance('icons/obj/service/hydroponics/equipment.dmi', "over_lowwater3")
|
|
if(reagents.total_volume <= 2)
|
|
. += mutable_appearance('icons/obj/service/hydroponics/equipment.dmi', "over_lownutri3")
|
|
if(plant_health <= (myseed.endurance / 2))
|
|
. += mutable_appearance('icons/obj/service/hydroponics/equipment.dmi', "over_lowhealth3")
|
|
if(weedlevel >= 5 || pestlevel >= 5 || toxic >= 40)
|
|
. += mutable_appearance('icons/obj/service/hydroponics/equipment.dmi', "over_alert3")
|
|
if(plant_status == HYDROTRAY_PLANT_HARVESTABLE)
|
|
. += mutable_appearance('icons/obj/service/hydroponics/equipment.dmi', "over_harvest3")
|
|
|
|
///Sets a new value for the myseed variable, which is the seed of the plant that's growing inside the tray.
|
|
/obj/machinery/hydroponics/proc/set_seed(obj/item/seeds/new_seed, delete_old_seed = TRUE)
|
|
var/old_seed = myseed
|
|
myseed = new_seed
|
|
if(old_seed && delete_old_seed)
|
|
qdel(old_seed)
|
|
set_plant_status(new_seed ? HYDROTRAY_PLANT_GROWING : HYDROTRAY_NO_PLANT) //To make sure they can't just put in another seed and insta-harvest it
|
|
if(myseed && myseed.loc != src)
|
|
myseed.forceMove(src)
|
|
SEND_SIGNAL(src, COMSIG_HYDROTRAY_SET_SEED, new_seed)
|
|
update_appearance()
|
|
if(isnull(myseed))
|
|
particles = null
|
|
|
|
/*
|
|
* Setter proc to set a tray to a new self_sustaining state and update all values associated with it.
|
|
*
|
|
* new_value - true / false value that self_sustaining is being set to
|
|
*/
|
|
/obj/machinery/hydroponics/proc/set_self_sustaining(new_value)
|
|
if(self_sustaining == new_value)
|
|
return
|
|
|
|
self_sustaining = new_value
|
|
|
|
update_use_power(self_sustaining ? ACTIVE_POWER_USE : NO_POWER_USE)
|
|
update_appearance()
|
|
|
|
SEND_SIGNAL(src, COMSIG_HYDROTRAY_SET_SELFSUSTAINING, new_value)
|
|
|
|
/obj/machinery/hydroponics/proc/set_weedlevel(new_weedlevel, update_icon = TRUE)
|
|
if(weedlevel == new_weedlevel)
|
|
return
|
|
SEND_SIGNAL(src, COMSIG_HYDROTRAY_SET_WEEDLEVEL, new_weedlevel)
|
|
weedlevel = new_weedlevel
|
|
if(update_icon)
|
|
update_appearance()
|
|
|
|
/obj/machinery/hydroponics/proc/set_pestlevel(new_pestlevel, update_icon = TRUE)
|
|
if(pestlevel == new_pestlevel)
|
|
return
|
|
SEND_SIGNAL(src, COMSIG_HYDROTRAY_SET_PESTLEVEL, new_pestlevel)
|
|
pestlevel = new_pestlevel
|
|
if(update_icon)
|
|
update_appearance()
|
|
|
|
/obj/machinery/hydroponics/proc/set_waterlevel(new_waterlevel, update_icon = TRUE)
|
|
if(waterlevel == new_waterlevel)
|
|
return
|
|
SEND_SIGNAL(src, COMSIG_HYDROTRAY_SET_WATERLEVEL, new_waterlevel)
|
|
waterlevel = new_waterlevel
|
|
if(update_icon)
|
|
update_appearance()
|
|
|
|
var/difference = new_waterlevel - waterlevel
|
|
if(difference > 0)
|
|
adjust_toxic(-round(difference/4))//Toxicity dilutation code. The more water you put in, the lesser the toxin concentration.
|
|
|
|
/obj/machinery/hydroponics/proc/set_plant_health(new_plant_health, update_icon = TRUE, forced = FALSE)
|
|
if(plant_health == new_plant_health || ((!myseed || plant_status == HYDROTRAY_PLANT_DEAD) && !forced))
|
|
return
|
|
SEND_SIGNAL(src, COMSIG_HYDROTRAY_SET_PLANT_HEALTH, new_plant_health)
|
|
plant_health = new_plant_health
|
|
if(update_icon)
|
|
update_appearance()
|
|
|
|
/obj/machinery/hydroponics/proc/set_toxic(new_toxic, update_icon = TRUE)
|
|
if(toxic == new_toxic)
|
|
return
|
|
SEND_SIGNAL(src, COMSIG_HYDROTRAY_SET_TOXIC, new_toxic)
|
|
toxic = new_toxic
|
|
if(update_icon)
|
|
update_appearance()
|
|
|
|
/obj/machinery/hydroponics/proc/set_plant_status(new_plant_status)
|
|
if(plant_status == new_plant_status)
|
|
return
|
|
SEND_SIGNAL(src, COMSIG_HYDROTRAY_SET_PLANT_STATUS, new_plant_status)
|
|
plant_status = new_plant_status
|
|
|
|
// The following procs adjust the hydroponics tray variables, and make sure that the stat doesn't go out of bounds.
|
|
|
|
/**
|
|
* Adjust water.
|
|
* Raises or lowers tray water values by a set value. Adding water will dillute toxicity from the tray.
|
|
* Returns the amount of water actually added/taken
|
|
* * adjustamt - determines how much water the tray will be adjusted upwards or downwards.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/adjust_waterlevel(amt)
|
|
var/initial_waterlevel = waterlevel
|
|
set_waterlevel(clamp(waterlevel+amt, 0, maxwater), FALSE)
|
|
return waterlevel-initial_waterlevel
|
|
|
|
/**
|
|
* Adjust Health.
|
|
* Raises the tray's plant_health stat by a given amount, with total health determined by the seed's endurance.
|
|
* * adjustamt - Determines how much the plant_health will be adjusted upwards or downwards.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/adjust_plant_health(amt)
|
|
set_plant_health(clamp(plant_health + amt, 0, myseed?.endurance), FALSE)
|
|
|
|
/**
|
|
* Adjust toxicity.
|
|
* Raises the plant's toxic stat by a given amount.
|
|
* * adjustamt - Determines how much the toxic will be adjusted upwards or downwards.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/adjust_toxic(amt)
|
|
set_toxic(clamp(toxic + amt, 0, MAX_TRAY_TOXINS), FALSE)
|
|
|
|
/**
|
|
* Adjust Pests.
|
|
* Raises the tray's pest level stat by a given amount.
|
|
* * adjustamt - Determines how much the pest level will be adjusted upwards or downwards.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/adjust_pestlevel(amt)
|
|
set_pestlevel(clamp(pestlevel + amt, 0, MAX_TRAY_PESTS), FALSE)
|
|
|
|
|
|
/**
|
|
* Adjust Weeds.
|
|
* Raises the plant's weed level stat by a given amount.
|
|
* * adjustamt - Determines how much the weed level will be adjusted upwards or downwards.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/adjust_weedlevel (amt)
|
|
set_weedlevel(clamp(weedlevel + amt, 0, MAX_TRAY_WEEDS), FALSE)
|
|
|
|
/obj/machinery/hydroponics/examine(user)
|
|
. = ..()
|
|
if(myseed)
|
|
. += span_info("It has [span_name("[myseed.plantname]")] planted.")
|
|
if (plant_status == HYDROTRAY_PLANT_DEAD)
|
|
. += span_warning("It's dead!")
|
|
else if (plant_status == HYDROTRAY_PLANT_HARVESTABLE)
|
|
. += span_info("It's ready to harvest.")
|
|
else if (plant_health <= (myseed.endurance / 2))
|
|
. += span_warning("It looks unhealthy.")
|
|
else
|
|
. += span_info("It's empty.")
|
|
|
|
. += span_info("Water: [waterlevel]/[maxwater].")
|
|
. += span_info("Nutrient: [reagents.total_volume]/[maxnutri].")
|
|
if(self_sustaining)
|
|
. += span_info("The tray's autogrow is active, protecting it from species mutations, weeds, and pests.")
|
|
|
|
if(weedlevel >= 5)
|
|
. += span_warning("It's filled with weeds!")
|
|
if(pestlevel >= 5)
|
|
. += span_warning("It's filled with tiny worms!")
|
|
|
|
/**
|
|
* What happens when a tray's weeds grow too large.
|
|
* Plants a new weed in an empty tray, then resets the tray.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/weedinvasion()
|
|
var/oldPlantName
|
|
if(myseed) // In case there's nothing in the tray beforehand
|
|
oldPlantName = myseed.plantname
|
|
else
|
|
oldPlantName = "empty tray"
|
|
var/obj/item/seeds/new_seed
|
|
switch(rand(1,18)) // randomly pick predominative weed
|
|
if(16 to 18)
|
|
new_seed = new /obj/item/seeds/reishi(src)
|
|
if(14 to 15)
|
|
new_seed = new /obj/item/seeds/nettle(src)
|
|
if(12 to 13)
|
|
new_seed = new /obj/item/seeds/harebell(src)
|
|
if(10 to 11)
|
|
new_seed = new /obj/item/seeds/amanita(src)
|
|
if(8 to 9)
|
|
new_seed = new /obj/item/seeds/chanter(src)
|
|
if(6 to 7)
|
|
new_seed = new /obj/item/seeds/tower(src)
|
|
if(4 to 5)
|
|
new_seed = new /obj/item/seeds/plump(src)
|
|
else
|
|
new_seed = new /obj/item/seeds/starthistle(src)
|
|
set_seed(new_seed)
|
|
age = 0
|
|
lastcycle = world.time
|
|
set_plant_health(myseed.endurance, update_icon = FALSE)
|
|
set_weedlevel(0, update_icon = FALSE) // Reset
|
|
set_pestlevel(0) // Reset
|
|
visible_message(span_warning("The [oldPlantName] is overtaken by some [myseed.plantname]!"))
|
|
|
|
/obj/machinery/hydroponics/proc/mutate(lifemut = 2, endmut = 5, productmut = 1, yieldmut = 2, potmut = 25, wrmut = 2, wcmut = 5, traitmut = 0, stabmut = 3) // Mutates the current seed
|
|
if(!myseed)
|
|
return
|
|
myseed.mutate(lifemut, endmut, productmut, yieldmut, potmut, wrmut, wcmut, traitmut, stabmut)
|
|
|
|
/obj/machinery/hydroponics/proc/hardmutate(lifemut = 4, endmut = 10, productmut = 2, yieldmut = 4, potmut = 50, wrmut = 4, wcmut = 10, traitmut = 0, stabmut = 4)
|
|
mutate(lifemut, endmut, productmut, yieldmut, potmut, wrmut, wcmut, traitmut, stabmut)
|
|
|
|
/obj/machinery/hydroponics/proc/mutatespecie() // Mutagent produced a new plant!
|
|
if(!myseed || plant_status == HYDROTRAY_PLANT_DEAD || !LAZYLEN(myseed.mutatelist))
|
|
return
|
|
|
|
var/oldPlantName = myseed.plantname
|
|
var/mutantseed = pick(myseed.mutatelist)
|
|
set_seed(new mutantseed(src))
|
|
|
|
hardmutate()
|
|
age = 0
|
|
set_plant_health(myseed.endurance, update_icon = FALSE)
|
|
lastcycle = world.time
|
|
set_weedlevel(0, update_icon = FALSE)
|
|
|
|
var/message = span_warning("[oldPlantName] suddenly mutates into [myseed.plantname]!")
|
|
addtimer(CALLBACK(src, PROC_REF(after_mutation), message), 0.5 SECONDS)
|
|
|
|
/obj/machinery/hydroponics/proc/polymorph() // Polymorph a plant into another plant
|
|
if(!myseed || plant_status == HYDROTRAY_PLANT_DEAD)
|
|
return
|
|
|
|
var/oldPlantName = myseed.plantname
|
|
var/polymorph_seed = pick(subtypesof(/obj/item/seeds))
|
|
set_seed(new polymorph_seed(src))
|
|
|
|
hardmutate()
|
|
age = 0
|
|
set_plant_health(myseed.endurance, update_icon = FALSE)
|
|
lastcycle = world.time
|
|
set_weedlevel(0, update_icon = FALSE)
|
|
|
|
var/message = span_warning("[oldPlantName] suddenly polymorphs into [myseed.plantname]!")
|
|
addtimer(CALLBACK(src, PROC_REF(after_mutation), message), 0.5 SECONDS)
|
|
|
|
/obj/machinery/hydroponics/proc/mutateweed() // If the weeds gets the mutagent instead. Mind you, this pretty much destroys the old plant
|
|
if( weedlevel > 5 )
|
|
set_seed(null)
|
|
var/newWeed = pick(/obj/item/seeds/liberty, /obj/item/seeds/angel, /obj/item/seeds/nettle/death, /obj/item/seeds/kudzu)
|
|
set_seed(new newWeed(src))
|
|
hardmutate()
|
|
age = 0
|
|
set_plant_health(myseed.endurance, update_icon = FALSE)
|
|
lastcycle = world.time
|
|
set_weedlevel(0, update_icon = FALSE) // Reset
|
|
|
|
var/message = span_warning("The mutated weeds in [src] spawn some [myseed.plantname]!")
|
|
addtimer(CALLBACK(src, PROC_REF(after_mutation), message), 0.5 SECONDS)
|
|
else
|
|
to_chat(usr, span_warning("The few weeds in [src] seem to react, but only for a moment..."))
|
|
/**
|
|
* Called after plant mutation, update the appearance of the tray content and send a visible_message()
|
|
*/
|
|
/obj/machinery/hydroponics/proc/after_mutation(message)
|
|
visible_message(message)
|
|
update_appearance()
|
|
|
|
/**
|
|
* Plant Death Proc.
|
|
* Cleans up various stats for the plant upon death, including pests, harvestability, and plant health.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/plantdies()
|
|
set_plant_health(0, update_icon = FALSE, forced = TRUE)
|
|
set_plant_status(HYDROTRAY_PLANT_DEAD)
|
|
set_pestlevel(0, update_icon = FALSE) // Pests die
|
|
lastproduce = 0
|
|
update_appearance()
|
|
SEND_SIGNAL(src, COMSIG_HYDROTRAY_PLANT_DEATH)
|
|
|
|
/**
|
|
* Plant Cross-Pollination.
|
|
* Checks all plants in the tray's oview range, then averages out the seed's potency, instability, and yield values.
|
|
* If the seed's instability is >= 20, the seed donates one of it's reagents to that nearby plant.
|
|
* * Range - The Oview range of trays to which to look for plants to donate reagents.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/pollinate(range = 1)
|
|
var/any_adjacent = FALSE
|
|
for(var/obj/machinery/hydroponics/T in oview(src, range))
|
|
//Here is where we check for window blocking.
|
|
if(!Adjacent(T) && range <= 1)
|
|
continue
|
|
if(T.myseed && T.plant_status != HYDROTRAY_PLANT_DEAD)
|
|
T.myseed.set_potency(round((T.myseed.potency+(1/10)*(myseed.potency-T.myseed.potency))))
|
|
T.myseed.set_instability(round((T.myseed.instability+(1/10)*(myseed.instability-T.myseed.instability))))
|
|
T.myseed.set_yield(round((T.myseed.yield+(1/2)*(myseed.yield-T.myseed.yield))))
|
|
any_adjacent = TRUE
|
|
if(isnull(particles))
|
|
particles = new /particles/pollen()
|
|
if(myseed.instability >= 20 && prob(70) && length(T.myseed.reagents_add))
|
|
var/list/datum/plant_gene/reagent/possible_reagents = list()
|
|
for(var/datum/plant_gene/reagent/reag in T.myseed.genes)
|
|
possible_reagents += reag
|
|
var/datum/plant_gene/reagent/reagent_gene = pick(possible_reagents) //Let this serve as a lession to delete your WIP comments before merge.
|
|
if(reagent_gene.can_add(myseed))
|
|
if(!reagent_gene.try_upgrade_gene(myseed))
|
|
myseed.genes += reagent_gene.Copy()
|
|
myseed.reagents_from_genes()
|
|
continue
|
|
if(!any_adjacent)
|
|
particles = null
|
|
|
|
/**
|
|
* Bee pollinate proc.
|
|
* Checks if the bee can pollinate the plant
|
|
*/
|
|
/obj/machinery/hydroponics/proc/can_bee_pollinate()
|
|
if(isnull(myseed))
|
|
return FALSE
|
|
if(plant_status == HYDROTRAY_PLANT_DEAD || recent_bee_visit)
|
|
return FALSE
|
|
return TRUE
|
|
|
|
/**
|
|
* Pest Mutation Proc.
|
|
* When a tray is mutated with high pest values, it will spawn spiders.
|
|
* * User - Person who last added chemicals to the tray for logging purposes.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/mutatepest(mob/user)
|
|
if(pestlevel > 5)
|
|
message_admins("[ADMIN_LOOKUPFLW(user)] last altered a hydro tray's contents which spawned spiderlings.")
|
|
user.log_message("last altered a hydro tray, which spiderlings spawned from.", LOG_GAME)
|
|
visible_message(span_warning("The pests seem to behave oddly..."))
|
|
spawn_atom_to_turf(/mob/living/basic/spider/growing/spiderling/hunter, src, 3, FALSE)
|
|
else if(myseed)
|
|
visible_message(span_warning("The pests seem to behave oddly in [myseed.name] tray, but quickly settle down..."))
|
|
|
|
/obj/machinery/hydroponics/wrench_act(mob/living/user, obj/item/tool)
|
|
. = ..()
|
|
default_unfasten_wrench(user, tool)
|
|
return ITEM_INTERACT_SUCCESS
|
|
|
|
/obj/machinery/hydroponics/attackby(obj/item/O, mob/user, params)
|
|
//Called when mob user "attacks" it with object O
|
|
if(IS_EDIBLE(O) || is_reagent_container(O)) // Syringe stuff (and other reagent containers now too)
|
|
var/obj/item/reagent_containers/reagent_source = O
|
|
|
|
if(!reagent_source.reagents.total_volume)
|
|
to_chat(user, span_warning("[reagent_source] is empty!"))
|
|
return 1
|
|
|
|
if(reagents.total_volume >= reagents.maximum_volume && !reagent_source.reagents.has_reagent(/datum/reagent/water, 1))
|
|
to_chat(user, span_notice("[src] is full."))
|
|
return
|
|
|
|
var/list/trays = list(src)//makes the list just this in cases of syringes and compost etc
|
|
var/target = myseed ? myseed.plantname : src
|
|
var/visi_msg = ""
|
|
var/transfer_amount
|
|
|
|
if(IS_EDIBLE(reagent_source) || istype(reagent_source, /obj/item/reagent_containers/pill))
|
|
visi_msg="[user] composts [reagent_source], spreading it through [target]"
|
|
transfer_amount = reagent_source.reagents.total_volume
|
|
SEND_SIGNAL(reagent_source, COMSIG_ITEM_ON_COMPOSTED, user)
|
|
else
|
|
transfer_amount = min(reagent_source.amount_per_transfer_from_this, reagent_source.reagents.total_volume)
|
|
if(istype(reagent_source, /obj/item/reagent_containers/syringe/))
|
|
var/obj/item/reagent_containers/syringe/syr = reagent_source
|
|
visi_msg="[user] injects [target] with [syr]"
|
|
// Beakers, bottles, buckets, etc.
|
|
if(reagent_source.is_drainable())
|
|
playsound(loc, 'sound/effects/slosh.ogg', 25, TRUE)
|
|
var/mutable_appearance/splash_animation = mutable_appearance('icons/effects/effects.dmi', "splash_hydroponics")
|
|
splash_animation.color = mix_color_from_reagents(reagent_source.reagents.reagent_list)
|
|
flick_overlay_view(splash_animation, 1.1 SECONDS)
|
|
|
|
if(visi_msg)
|
|
visible_message(span_notice("[visi_msg]."))
|
|
|
|
for(var/obj/machinery/hydroponics/H in trays)
|
|
//cause I don't want to feel like im juggling 15 tamagotchis and I can get to my real work of ripping flooring apart in hopes of validating my life choices of becoming a space-gardener
|
|
//This was originally in apply_chemicals, but due to apply_chemicals only holding nutrients, we handle it here now.
|
|
if(reagent_source.reagents.has_reagent(/datum/reagent/water))
|
|
var/water_amt = reagent_source.reagents.get_reagent_amount(/datum/reagent/water) * transfer_amount / reagent_source.reagents.total_volume
|
|
var/water_amt_adjusted = H.adjust_waterlevel(round(water_amt))
|
|
reagent_source.reagents.remove_reagent(/datum/reagent/water, water_amt_adjusted)
|
|
for(var/datum/reagent/not_water_reagent as anything in reagent_source.reagents.reagent_list)
|
|
if(istype(not_water_reagent,/datum/reagent/water))
|
|
continue
|
|
var/transfer_me_to_tray = reagent_source.reagents.get_reagent_amount(not_water_reagent.type) * transfer_amount / reagent_source.reagents.total_volume
|
|
reagent_source.reagents.trans_to(H.reagents, transfer_me_to_tray, target_id = not_water_reagent.type)
|
|
else
|
|
reagent_source.reagents.trans_to(H.reagents, transfer_amount, transferred_by = user)
|
|
lastuser = WEAKREF(user)
|
|
if(IS_EDIBLE(reagent_source) || istype(reagent_source, /obj/item/reagent_containers/pill))
|
|
qdel(reagent_source)
|
|
H.update_appearance()
|
|
return 1
|
|
H.update_appearance()
|
|
if(reagent_source) // If the source wasn't composted and destroyed
|
|
reagent_source.update_appearance()
|
|
return 1
|
|
|
|
else if(istype(O, /obj/item/seeds))
|
|
if(!myseed)
|
|
if(istype(O, /obj/item/seeds/kudzu))
|
|
investigate_log("had Kudzu planted in it by [key_name(user)] at [AREACOORD(src)].", INVESTIGATE_BOTANY)
|
|
if(!user.transferItemToLoc(O, src))
|
|
return
|
|
SEND_SIGNAL(O, COMSIG_SEED_ON_PLANTED, src)
|
|
to_chat(user, span_notice("You plant [O]."))
|
|
set_seed(O)
|
|
age = 1
|
|
set_plant_health(myseed.endurance)
|
|
lastcycle = world.time
|
|
return
|
|
else
|
|
to_chat(user, span_warning("[src] already has seeds in it!"))
|
|
return
|
|
|
|
else if(istype(O, /obj/item/cultivator))
|
|
if(weedlevel > 0)
|
|
user.visible_message(span_notice("[user] uproots the weeds."), span_notice("You remove the weeds from [src]."))
|
|
set_weedlevel(0)
|
|
return
|
|
else
|
|
to_chat(user, span_warning("This plot is completely devoid of weeds! It doesn't need uprooting."))
|
|
return
|
|
|
|
else if(istype(O, /obj/item/secateurs))
|
|
if(!myseed)
|
|
to_chat(user, span_notice("This plot is empty."))
|
|
return
|
|
else if(plant_status != HYDROTRAY_PLANT_HARVESTABLE)
|
|
to_chat(user, span_notice("This plant must be harvestable in order to be grafted."))
|
|
return
|
|
else if(myseed.grafted)
|
|
to_chat(user, span_notice("This plant has already been grafted."))
|
|
return
|
|
else
|
|
user.visible_message(span_notice("[user] grafts off a limb from [src]."), span_notice("You carefully graft off a portion of [src]."))
|
|
var/obj/item/graft/snip = myseed.create_graft()
|
|
if(!snip)
|
|
return // The plant did not return a graft.
|
|
|
|
snip.forceMove(drop_location())
|
|
myseed.grafted = TRUE
|
|
adjust_plant_health(-5)
|
|
return
|
|
|
|
else if(istype(O, /obj/item/geneshears))
|
|
if(!myseed)
|
|
to_chat(user, span_notice("The tray is empty."))
|
|
return
|
|
if(plant_health <= GENE_SHEAR_MIN_HEALTH)
|
|
to_chat(user, span_notice("This plant looks too unhealty to be sheared right now."))
|
|
return
|
|
|
|
var/list/current_traits = list()
|
|
for(var/datum/plant_gene/gene in myseed.genes)
|
|
if(islist(gene))
|
|
continue
|
|
if(!(gene.mutability_flags & PLANT_GENE_REMOVABLE))
|
|
continue // Don't show genes that can't be removed.
|
|
current_traits[gene.name] = gene
|
|
var/removed_trait = tgui_input_list(user, "Trait to remove from the [myseed.plantname]", "Plant Trait Removal", sort_list(current_traits))
|
|
if(isnull(removed_trait))
|
|
return
|
|
if(!user.can_perform_action(src))
|
|
return
|
|
if(!myseed)
|
|
return
|
|
if(plant_health <= GENE_SHEAR_MIN_HEALTH) //Check health again to make sure they're not keeping inputs open to get free shears.
|
|
return
|
|
for(var/datum/plant_gene/gene in myseed.genes)
|
|
if(gene.name == removed_trait)
|
|
if(myseed.genes.Remove(gene))
|
|
gene.on_removed(myseed)
|
|
qdel(gene)
|
|
break
|
|
myseed.reagents_from_genes()
|
|
adjust_plant_health(-15)
|
|
to_chat(user, span_notice("You carefully shear the genes off of the [myseed.plantname], leaving the plant looking weaker."))
|
|
update_appearance()
|
|
return
|
|
|
|
else if(istype(O, /obj/item/graft))
|
|
var/obj/item/graft/snip = O
|
|
if(!myseed)
|
|
to_chat(user, span_notice("The tray is empty."))
|
|
return
|
|
if(myseed.apply_graft(snip))
|
|
to_chat(user, span_notice("You carefully integrate the grafted plant limb onto [myseed.plantname], granting it [snip.stored_trait.get_name()]."))
|
|
else
|
|
to_chat(user, span_notice("You integrate the grafted plant limb onto [myseed.plantname], but it does not accept the [snip.stored_trait.get_name()] trait from the [snip]."))
|
|
qdel(snip)
|
|
return
|
|
|
|
else if(istype(O, /obj/item/storage/bag/plants))
|
|
if(plant_status == HYDROTRAY_PLANT_HARVESTABLE)
|
|
var/list/harvest = myseed.harvest(user)
|
|
for(var/obj/item/food/grown/G in harvest)
|
|
O.atom_storage?.attempt_insert(G, user, TRUE)
|
|
else if(plant_status == HYDROTRAY_PLANT_DEAD)
|
|
to_chat(user, span_notice("You remove the dead plant from [src]."))
|
|
set_seed(null)
|
|
return
|
|
|
|
else if(O.tool_behaviour == TOOL_SHOVEL)
|
|
if(!myseed && !weedlevel)
|
|
to_chat(user, span_warning("[src] doesn't have any plants or weeds!"))
|
|
return
|
|
user.visible_message(span_notice("[user] starts digging out [src]'s plants..."),
|
|
span_notice("You start digging out [src]'s plants..."))
|
|
if(O.use_tool(src, user, 50, volume=50) || (!myseed && !weedlevel))
|
|
user.visible_message(span_notice("[user] digs out the plants in [src]!"), span_notice("You dig out all of [src]'s plants!"))
|
|
if(myseed) //Could be that they're just using it as a de-weeder
|
|
age = 0
|
|
set_plant_health(0, update_icon = FALSE, forced = TRUE)
|
|
lastproduce = 0
|
|
set_seed(null)
|
|
set_weedlevel(0) //Has a side effect of cleaning up those nasty weeds
|
|
return
|
|
else if(istype(O, /obj/item/storage/part_replacer))
|
|
RefreshParts()
|
|
return
|
|
else if(istype(O, /obj/item/gun/energy/floragun))
|
|
var/obj/item/gun/energy/floragun/flowergun = O
|
|
if(flowergun.cell.charge < flowergun.cell.maxcharge)
|
|
to_chat(user, span_notice("[flowergun] must be fully charged to lock in a mutation!"))
|
|
return
|
|
if(!myseed)
|
|
to_chat(user, span_warning("[src] is empty!"))
|
|
return
|
|
if(myseed.endurance <= FLORA_GUN_MIN_ENDURANCE)
|
|
to_chat(user, span_warning("[myseed.plantname] isn't hardy enough to sequence it's mutation!"))
|
|
return
|
|
if(!LAZYLEN(myseed.mutatelist))
|
|
to_chat(user, span_warning("[myseed.plantname] has nothing else to mutate into!"))
|
|
return
|
|
else
|
|
var/list/fresh_mut_list = list()
|
|
for(var/muties in myseed.mutatelist)
|
|
var/obj/item/seeds/another_mut = new muties
|
|
fresh_mut_list[another_mut.plantname] = muties
|
|
var/locked_mutation = tgui_input_list(user, "Mutation to lock", "Plant Mutation Locks", sort_list(fresh_mut_list))
|
|
if(isnull(locked_mutation))
|
|
return
|
|
if(isnull(fresh_mut_list[locked_mutation]))
|
|
return
|
|
if(!user.can_perform_action(src))
|
|
return
|
|
myseed.mutatelist = list(fresh_mut_list[locked_mutation])
|
|
myseed.set_endurance(myseed.endurance/2)
|
|
flowergun.cell.use(flowergun.cell.charge)
|
|
flowergun.update_appearance()
|
|
to_chat(user, span_notice("[myseed.plantname]'s mutation was set to [locked_mutation], depleting [flowergun]'s cell!"))
|
|
return
|
|
else
|
|
return ..()
|
|
|
|
/obj/machinery/hydroponics/attackby_secondary(obj/item/weapon, mob/user, params)
|
|
if (istype(weapon, /obj/item/reagent_containers/syringe))
|
|
to_chat(user, span_warning("You can't get any extract out of this plant."))
|
|
return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
|
|
return SECONDARY_ATTACK_CALL_NORMAL
|
|
|
|
/obj/machinery/hydroponics/can_be_unfasten_wrench(mob/user, silent)
|
|
if (!unwrenchable) // case also covered by NODECONSTRUCT checks in default_unfasten_wrench
|
|
return CANT_UNFASTEN
|
|
|
|
return ..()
|
|
|
|
/obj/machinery/hydroponics/attack_hand(mob/user, list/modifiers)
|
|
. = ..()
|
|
if(.)
|
|
return
|
|
if(HAS_SILICON_ACCESS(user)) //How does AI know what plant is?
|
|
return
|
|
if(plant_status == HYDROTRAY_PLANT_HARVESTABLE)
|
|
return myseed.harvest(user)
|
|
|
|
else if(plant_status == HYDROTRAY_PLANT_DEAD)
|
|
to_chat(user, span_notice("You remove the dead plant from [src]."))
|
|
set_seed(null)
|
|
else
|
|
if(user)
|
|
user.examinate(src)
|
|
|
|
/obj/machinery/hydroponics/CtrlClick(mob/user)
|
|
. = ..()
|
|
if(!user.can_perform_action(src, FORBID_TELEKINESIS_REACH))
|
|
return
|
|
if(!powered())
|
|
to_chat(user, span_warning("[name] has no power."))
|
|
update_use_power(NO_POWER_USE)
|
|
return
|
|
if(!anchored)
|
|
return
|
|
set_self_sustaining(!self_sustaining)
|
|
to_chat(user, span_notice("You [self_sustaining ? "activate" : "deactivated"] [src]'s autogrow function[self_sustaining ? ", maintaining the tray's health while using high amounts of power" : ""]."))
|
|
|
|
/obj/machinery/hydroponics/AltClick(mob/user)
|
|
return ..() // This hotkey is BLACKLISTED since it's used by /datum/component/simple_rotation
|
|
|
|
/obj/machinery/hydroponics/attack_hand_secondary(mob/user, list/modifiers)
|
|
. = ..()
|
|
if(. == SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN)
|
|
return
|
|
if(!anchored)
|
|
return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
|
|
var/warning = tgui_alert(user, "Are you sure you wish to empty the tray's nutrient beaker?","Empty Tray Nutrients?", list("Yes", "No"))
|
|
if(warning == "Yes" && user.can_perform_action(src, FORBID_TELEKINESIS_REACH))
|
|
reagents.clear_reagents()
|
|
to_chat(user, span_warning("You empty [src]'s nutrient tank."))
|
|
update_appearance()
|
|
return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
|
|
|
|
/**
|
|
* Update Tray Proc
|
|
* Handles plant harvesting on the tray side, by clearing the seed, names, description, and dead stat.
|
|
* Shuts off autogrow if enabled.
|
|
* Sends messages to the cleaer about plants harvested, or if nothing was harvested at all.
|
|
* * User - The mob who clears the tray.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/update_tray(mob/user, product_count)
|
|
lastproduce = age
|
|
if(istype(myseed, /obj/item/seeds/replicapod))
|
|
to_chat(user, span_notice("You harvest from the [myseed.plantname]."))
|
|
else if(product_count <= 0)
|
|
to_chat(user, span_warning("You fail to harvest anything useful!"))
|
|
else
|
|
to_chat(user, span_notice("You harvest [product_count] items from the [myseed.plantname]."))
|
|
if(!myseed.get_gene(/datum/plant_gene/trait/repeated_harvest))
|
|
set_seed(null)
|
|
if(self_sustaining) //No reason to pay for an empty tray.
|
|
set_self_sustaining(FALSE)
|
|
else
|
|
set_plant_status(HYDROTRAY_PLANT_GROWING)
|
|
update_appearance()
|
|
SEND_SIGNAL(src, COMSIG_HYDROTRAY_ON_HARVEST, user, product_count)
|
|
|
|
/**
|
|
* Spawn Plant.
|
|
* Upon using strange reagent on a tray, it will spawn a killer tomato or killer tree at random.
|
|
*/
|
|
/obj/machinery/hydroponics/proc/spawnplant() // why would you put strange reagent in a hydro tray you monster I bet you also feed them blood
|
|
var/list/livingplants = list(/mob/living/basic/tree, /mob/living/basic/killer_tomato)
|
|
var/chosen = pick(livingplants)
|
|
var/mob/living/C = new chosen(get_turf(src))
|
|
C.faction = list(FACTION_PLANTS)
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
/obj/machinery/hydroponics/soil //Not actually hydroponics at all! Honk!
|
|
name = "soil"
|
|
desc = "A patch of dirt."
|
|
icon = 'icons/obj/service/hydroponics/equipment.dmi'
|
|
icon_state = "soil"
|
|
gender = PLURAL
|
|
circuit = null
|
|
density = FALSE
|
|
use_power = NO_POWER_USE
|
|
obj_flags = parent_type::obj_flags | NO_DECONSTRUCTION
|
|
unwrenchable = FALSE
|
|
self_sustaining_overlay_icon_state = null
|
|
maxnutri = 15
|
|
|
|
/obj/machinery/hydroponics/soil/update_icon(updates=ALL)
|
|
. = ..()
|
|
if(self_sustaining)
|
|
add_atom_colour(rgb(255, 175, 0), FIXED_COLOUR_PRIORITY)
|
|
|
|
/obj/machinery/hydroponics/soil/update_status_light_overlays()
|
|
return // Has no lights
|
|
|
|
/obj/machinery/hydroponics/soil/attackby_secondary(obj/item/weapon, mob/user, params)
|
|
if(weapon.tool_behaviour != TOOL_SHOVEL) //Spades can still uproot plants on left click
|
|
return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
|
|
balloon_alert(user, "clearing up soil...")
|
|
if(weapon.use_tool(src, user, 1 SECONDS, volume=50))
|
|
balloon_alert(user, "cleared")
|
|
deconstruct(disassembled = TRUE)
|
|
return SECONDARY_ATTACK_CANCEL_ATTACK_CHAIN
|
|
|
|
/obj/machinery/hydroponics/soil/CtrlClick(mob/user)
|
|
return //Soil has no electricity.
|
|
|
|
/obj/machinery/hydroponics/soil/on_deconstruction(disassembled)
|
|
new /obj/item/stack/ore/glass(drop_location(), 3)
|
|
|
|
///The usb port circuit
|
|
|
|
/obj/item/circuit_component/hydroponics
|
|
display_name = "Hydropnics Tray"
|
|
desc = "Automate the means of botanical production. Trigger to toggle auto-grow."
|
|
circuit_flags = CIRCUIT_FLAG_INPUT_SIGNAL
|
|
|
|
var/obj/machinery/hydroponics/attached_tray
|
|
///If self-sustaining (also called auto-grow) should be turned on or off when the trigger is triggered.
|
|
var/datum/port/input/selfsustaining_setting
|
|
///Whether the plant in the tray is harvestable, alive, missing or dead.
|
|
var/datum/port/output/plant_status
|
|
///Whether the self sustaining mode is on
|
|
var/datum/port/output/is_self_sustaining
|
|
///Triggered when the plant is harvested
|
|
var/datum/port/output/plant_harvested
|
|
///The product amount of the last harvest
|
|
var/datum/port/output/last_harvest
|
|
///Triggered when the plant dies
|
|
var/datum/port/output/plant_died
|
|
///Triggered when a seed is either planted by someone or takes over the tray.
|
|
var/datum/port/output/seeds_planted
|
|
///The amount of water in the tray.
|
|
var/datum/port/output/water_level
|
|
///The amount of toxins in the tray.
|
|
var/datum/port/output/toxic_level
|
|
///The amount of pests in the tray.
|
|
var/datum/port/output/pests_level
|
|
///The amount of weeds in the tray.
|
|
var/datum/port/output/weeds_level
|
|
///The health of the plant in the tray.
|
|
var/datum/port/output/plant_health
|
|
///The amount of reagents in the tray
|
|
var/datum/port/output/reagents_level
|
|
|
|
/obj/item/circuit_component/hydroponics/populate_ports()
|
|
selfsustaining_setting = add_input_port("Auto-Grow Setting", PORT_TYPE_NUMBER)
|
|
|
|
plant_status = add_output_port("Plant Status", PORT_TYPE_NUMBER)
|
|
is_self_sustaining = add_output_port("Auto-Grow Status", PORT_TYPE_NUMBER)
|
|
plant_harvested = add_output_port("Plant Harvested", PORT_TYPE_SIGNAL)
|
|
last_harvest = add_output_port("Last Harvest Amount", PORT_TYPE_NUMBER)
|
|
plant_died = add_output_port("Plant Died", PORT_TYPE_SIGNAL)
|
|
seeds_planted = add_output_port("Seeds Planted", PORT_TYPE_SIGNAL)
|
|
water_level = add_output_port("Water Level", PORT_TYPE_NUMBER)
|
|
toxic_level = add_output_port("Toxins Level", PORT_TYPE_NUMBER)
|
|
pests_level = add_output_port("Pests Level", PORT_TYPE_NUMBER)
|
|
weeds_level = add_output_port("Weeds Level", PORT_TYPE_NUMBER)
|
|
plant_health = add_output_port("Plant Health", PORT_TYPE_NUMBER)
|
|
reagents_level = add_output_port("Reagents Level", PORT_TYPE_NUMBER)
|
|
|
|
/obj/item/circuit_component/hydroponics/register_usb_parent(atom/movable/parent)
|
|
. = ..()
|
|
if(istype(parent, /obj/machinery/hydroponics))
|
|
attached_tray = parent
|
|
RegisterSignal(attached_tray, COMSIG_HYDROTRAY_SET_SEED, PROC_REF(on_set_seed))
|
|
RegisterSignal(attached_tray, COMSIG_HYDROTRAY_SET_SELFSUSTAINING, PROC_REF(on_set_selfsustaining))
|
|
RegisterSignal(attached_tray, COMSIG_HYDROTRAY_SET_WEEDLEVEL, PROC_REF(on_set_weedlevel))
|
|
RegisterSignal(attached_tray, COMSIG_HYDROTRAY_SET_PESTLEVEL, PROC_REF(on_set_pestlevel))
|
|
RegisterSignal(attached_tray, COMSIG_HYDROTRAY_SET_WATERLEVEL, PROC_REF(on_set_waterlevel))
|
|
RegisterSignal(attached_tray, COMSIG_HYDROTRAY_SET_PLANT_HEALTH, PROC_REF(on_set_plant_health))
|
|
RegisterSignal(attached_tray, COMSIG_HYDROTRAY_SET_TOXIC, PROC_REF(on_set_toxic_level))
|
|
RegisterSignal(attached_tray, COMSIG_HYDROTRAY_SET_PLANT_STATUS, PROC_REF(on_set_plant_status))
|
|
RegisterSignal(attached_tray, COMSIG_HYDROTRAY_ON_HARVEST, PROC_REF(on_harvest))
|
|
RegisterSignal(attached_tray, COMSIG_HYDROTRAY_PLANT_DEATH, PROC_REF(on_plant_death))
|
|
var/list/reagents_holder_signals = list(
|
|
COMSIG_REAGENTS_ADD_REAGENT,
|
|
COMSIG_REAGENTS_REM_REAGENT,
|
|
COMSIG_REAGENTS_NEW_REAGENT,
|
|
COMSIG_REAGENTS_DEL_REAGENT,
|
|
)
|
|
RegisterSignal(attached_tray, reagents_holder_signals, PROC_REF(update_reagents_level))
|
|
|
|
/obj/item/circuit_component/hydroponics/unregister_usb_parent(atom/movable/parent)
|
|
attached_tray = null
|
|
UnregisterSignal(parent, list(COMSIG_HYDROTRAY_SET_SEED, COMSIG_HYDROTRAY_SET_SELFSUSTAINING,
|
|
COMSIG_HYDROTRAY_SET_WEEDLEVEL, COMSIG_HYDROTRAY_SET_PESTLEVEL, COMSIG_HYDROTRAY_SET_WATERLEVEL,
|
|
COMSIG_HYDROTRAY_SET_PLANT_HEALTH, COMSIG_HYDROTRAY_SET_TOXIC, COMSIG_HYDROTRAY_SET_PLANT_STATUS,
|
|
COMSIG_HYDROTRAY_ON_HARVEST, COMSIG_HYDROTRAY_PLANT_DEATH))
|
|
if(parent.reagents)
|
|
UnregisterSignal(parent.reagents, list(COMSIG_REAGENTS_ADD_REAGENT, COMSIG_REAGENTS_REM_REAGENT,
|
|
COMSIG_REAGENTS_NEW_REAGENT, COMSIG_REAGENTS_DEL_REAGENT))
|
|
return ..()
|
|
|
|
/obj/item/circuit_component/hydroponics/get_ui_notices()
|
|
. = ..()
|
|
. += create_ui_notice("Plant Status Index: \"[HYDROTRAY_NO_PLANT]\", \"[HYDROTRAY_PLANT_GROWING]\", \"[HYDROTRAY_PLANT_DEAD]\", \"[HYDROTRAY_PLANT_HARVESTABLE]\"", "orange", "info")
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/on_set_seed(datum/source, obj/item/seeds/new_seed)
|
|
SIGNAL_HANDLER
|
|
seeds_planted.set_output(COMPONENT_SIGNAL)
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/on_set_selfsustaining(datum/source, new_value)
|
|
SIGNAL_HANDLER
|
|
is_self_sustaining.set_output(new_value)
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/on_set_weedlevel(datum/source, new_value)
|
|
SIGNAL_HANDLER
|
|
weeds_level.set_output(new_value)
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/on_set_pestlevel(datum/source, new_value)
|
|
SIGNAL_HANDLER
|
|
pests_level.set_output(new_value)
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/on_set_waterlevel(datum/source, new_value)
|
|
SIGNAL_HANDLER
|
|
water_level.set_output(new_value)
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/on_set_plant_health(datum/source, new_value)
|
|
SIGNAL_HANDLER
|
|
plant_health.set_output(new_value)
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/on_set_toxic_level(datum/source, new_value)
|
|
SIGNAL_HANDLER
|
|
toxic_level.set_output(new_value)
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/on_set_plant_status(datum/source, new_value)
|
|
SIGNAL_HANDLER
|
|
plant_status.set_output(new_value)
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/on_harvest(datum/source, product_amount)
|
|
SIGNAL_HANDLER
|
|
last_harvest.set_output(product_amount)
|
|
plant_harvested.set_output(COMPONENT_SIGNAL)
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/on_plant_death(datum/source)
|
|
SIGNAL_HANDLER
|
|
plant_died.set_output(COMPONENT_SIGNAL)
|
|
|
|
/obj/item/circuit_component/hydroponics/proc/update_reagents_level(datum/source)
|
|
SIGNAL_HANDLER
|
|
reagents_level.set_output(attached_tray.reagents.total_volume)
|
|
|
|
/obj/item/circuit_component/hydroponics/input_received(datum/port/input/port)
|
|
if(attached_tray.anchored && attached_tray.powered())
|
|
attached_tray.set_self_sustaining(!!selfsustaining_setting.value)
|