/* This datum should be used for handling mineral contents of machines and whatever else is supposed to hold minerals and make use of them. Variables: amount - raw amount of the mineral this container is holding, calculated by the defined value MINERAL_MATERIAL_AMOUNT=2000. max_amount - max raw amount of mineral this container can hold. sheet_type - type of the mineral sheet the container handles, used for output. owner - object that this container is being used by, used for output. MAX_STACK_SIZE - size of a stack of mineral sheets. Constant. */ /datum/material_container var/total_amount = 0 var/max_amount var/sheet_type var/obj/owner var/list/materials = list() //MAX_STACK_SIZE = 50 //MINERAL_MATERIAL_AMOUNT = 2000 /datum/material_container/New(obj/O, list/mat_list, max_amt = 0) owner = O max_amount = max(0, max_amt) if(mat_list[MAT_METAL]) materials[MAT_METAL] = new /datum/material/metal() if(mat_list[MAT_GLASS]) materials[MAT_GLASS] = new /datum/material/glass() if(mat_list[MAT_SILVER]) materials[MAT_SILVER] = new /datum/material/silver() if(mat_list[MAT_GOLD]) materials[MAT_GOLD] = new /datum/material/gold() if(mat_list[MAT_DIAMOND]) materials[MAT_DIAMOND] = new /datum/material/diamond() if(mat_list[MAT_URANIUM]) materials[MAT_URANIUM] = new /datum/material/uranium() if(mat_list[MAT_PLASMA]) materials[MAT_PLASMA] = new /datum/material/plasma() if(mat_list[MAT_BANANIUM]) materials[MAT_BANANIUM] = new /datum/material/bananium() if(mat_list[MAT_TRANQUILLITE]) materials[MAT_TRANQUILLITE] = new /datum/material/tranquillite() /datum/material_container/Destroy() owner = null return ..() //For inserting an amount of material /datum/material_container/proc/insert_amount(amt, material_type = null) if(amt > 0 && has_space(amt)) var/total_amount_saved = total_amount if(material_type) var/datum/material/M = materials[material_type] if(M) M.amount += amt total_amount += amt else for(var/datum/material/M in materials) M.amount += amt total_amount += amt return (total_amount - total_amount_saved) return 0 /datum/material_container/proc/insert_stack(obj/item/stack/S, amt = 0) if(amt <= 0) return 0 if(amt > S.amount) amt = S.amount var/material_amt = get_item_material_amount(S) if(!material_amt) return 0 amt = min(amt, round(((max_amount - total_amount) / material_amt))) if(!amt) return 0 insert_materials(S,amt) S.use(amt) return amt /datum/material_container/proc/insert_item(obj/item/I, multiplier = 1) if(!I) return 0 if(istype(I,/obj/item/stack)) var/obj/item/stack/S = I return insert_stack(I, S.amount) var/material_amount = get_item_material_amount(I) if(!material_amount || !has_space(material_amount)) return 0 insert_materials(I, multiplier) return material_amount /datum/material_container/proc/insert_materials(obj/item/I, multiplier = 1) //for internal usage only var/datum/material/M for(var/MAT in materials) M = materials[MAT] M.amount += I.materials[MAT] * multiplier total_amount += I.materials[MAT] * multiplier //For consuming material //mats is a list of types of material to use and the corresponding amounts, example: list(MAT_METAL=100, MAT_GLASS=200) /datum/material_container/proc/use_amount(list/mats, multiplier=1) if(!mats || !mats.len) return 0 var/datum/material/M for(var/MAT in materials) M = materials[MAT] if(M.amount < (mats[MAT] * multiplier)) return 0 var/total_amount_save = total_amount for(var/MAT in materials) M = materials[MAT] M.amount -= mats[MAT] * multiplier total_amount -= mats[MAT] * multiplier return total_amount_save - total_amount /datum/material_container/proc/use_amount_type(amt, material_type) var/datum/material/M M = materials[material_type] if(M) if(M.amount >= amt) M.amount -= amt total_amount -= amt return amt return 0 //For spawning mineral sheets; internal use only /datum/material_container/proc/retrieve(sheet_amt, datum/material/M) if(sheet_amt > 0) if(M.amount < (sheet_amt * MINERAL_MATERIAL_AMOUNT)) sheet_amt = round(M.amount / MINERAL_MATERIAL_AMOUNT) var/count = 0 while(sheet_amt > MAX_STACK_SIZE) new M.sheet_type(get_turf(owner), MAX_STACK_SIZE) count += MAX_STACK_SIZE use_amount_type(sheet_amt * MINERAL_MATERIAL_AMOUNT, M.material_type) sheet_amt -= MAX_STACK_SIZE if(round(M.amount / MINERAL_MATERIAL_AMOUNT)) new M.sheet_type(get_turf(owner), sheet_amt) count += sheet_amt use_amount_type(sheet_amt * MINERAL_MATERIAL_AMOUNT, M.material_type) return count return 0 /datum/material_container/proc/retrieve_sheets(sheet_amt, material_type) if(materials[material_type]) return retrieve(sheet_amt, materials[material_type]) return 0 /datum/material_container/proc/retrieve_amount(amt, material_type) return retrieve_sheets(amount2sheet(amt),material_type) /datum/material_container/proc/retrieve_all() var/result = 0 var/datum/material/M for(var/MAT in materials) M = materials[MAT] result += retrieve_sheets(amount2sheet(M.amount), MAT) return result /datum/material_container/proc/has_space(amt = 0) return (total_amount + amt) <= max_amount /datum/material_container/proc/has_materials(list/mats, multiplier=1) if(!mats || !mats.len) return 0 var/datum/material/M for(var/MAT in mats) M = materials[MAT] if(M.amount < (mats[MAT] * multiplier)) return 0 return 1 /datum/material_container/proc/amount2sheet(amt) if(amt >= MINERAL_MATERIAL_AMOUNT) return round(amt / MINERAL_MATERIAL_AMOUNT) return 0 /datum/material_container/proc/sheet2amount(sheet_amt) if(sheet_amt > 0) return sheet_amt * MINERAL_MATERIAL_AMOUNT return 0 /datum/material_container/proc/amount(material_type) var/datum/material/M = materials[material_type] return M ? M.amount : 0 //returns the amount of material relevant to this container; //if this container does not support glass, any glass in 'I' will not be taken into account /datum/material_container/proc/get_item_material_amount(obj/item/I) if(!istype(I)) return 0 var/material_amount = 0 for(var/MAT in materials) material_amount += I.materials[MAT] return material_amount /datum/material var/amount = 0 var/material_type = null var/sheet_type = null /datum/material/metal /datum/material/metal/New() ..() material_type = MAT_METAL sheet_type = /obj/item/stack/sheet/metal /datum/material/glass /datum/material/glass/New() ..() material_type = MAT_GLASS sheet_type = /obj/item/stack/sheet/glass /datum/material/silver /datum/material/silver/New() ..() material_type = MAT_SILVER sheet_type = /obj/item/stack/sheet/mineral/silver /datum/material/gold /datum/material/gold/New() ..() material_type = MAT_GOLD sheet_type = /obj/item/stack/sheet/mineral/gold /datum/material/diamond /datum/material/diamond/New() ..() material_type = MAT_DIAMOND sheet_type = /obj/item/stack/sheet/mineral/diamond /datum/material/uranium /datum/material/uranium/New() ..() material_type = MAT_URANIUM sheet_type = /obj/item/stack/sheet/mineral/uranium /datum/material/plasma /datum/material/plasma/New() ..() material_type = MAT_PLASMA sheet_type = /obj/item/stack/sheet/mineral/plasma /datum/material/bananium /datum/material/bananium/New() ..() material_type = MAT_BANANIUM sheet_type = /obj/item/stack/sheet/mineral/bananium /datum/material/tranquillite /datum/material/tranquillite/New() ..() material_type = MAT_TRANQUILLITE sheet_type = /obj/item/stack/sheet/mineral/tranquillite