Gang-rework bugfixes and rebalancing. Started adding Gang API

This commit is contained in:
danielyxie 2018-10-19 19:30:02 -05:00
parent a0ebcff0aa
commit e73ebe843f
7 changed files with 471 additions and 103 deletions

@ -93,6 +93,8 @@ let CONSTANTS = {
ScriptSingularityFnRamMult: 2, // Multiplier for RAM cost outside of BN-4 ScriptSingularityFnRamMult: 2, // Multiplier for RAM cost outside of BN-4
ScriptGangApiBaseRamCost: 4,
ScriptBladeburnerApiBaseRamCost: 4, ScriptBladeburnerApiBaseRamCost: 4,
NumNetscriptPorts: 20, NumNetscriptPorts: 20,
@ -511,6 +513,8 @@ let CONSTANTS = {
*** Added a new category of upgrades for Gang Members: Augmentations *** Added a new category of upgrades for Gang Members: Augmentations
*** Non-Augmentation Gang member upgrades are now significantly weaker *** Non-Augmentation Gang member upgrades are now significantly weaker
*** Reputation for your Gang faction can no longer be gained through Infiltration *** Reputation for your Gang faction can no longer be gained through Infiltration
*** Re-worked the territory 'warfare' mechanic so that player can choose when to engage in it
*** Player's faction reputation multiplier no longer affects reputation gang from earning respect
* RAM Cost of accessing the global document object lowered from 100 GB to 25 GB * RAM Cost of accessing the global document object lowered from 100 GB to 25 GB
* RAM Cost to use Singularity Functions outside of BitNode-4 lowered by 75%. They now only cost twice as much as they do in BitNode-4 * RAM Cost to use Singularity Functions outside of BitNode-4 lowered by 75%. They now only cost twice as much as they do in BitNode-4
* b1t_flum3.exe now takes significantly less time to create * b1t_flum3.exe now takes significantly less time to create

@ -16,7 +16,6 @@ import {gangMemberUpgradesMetadata} from "./data/gangmemberupgrades"
import {Engine} from "./engine"; import {Engine} from "./engine";
import {Faction, Factions, import {Faction, Factions,
displayFactionContent} from "./Faction"; displayFactionContent} from "./Faction";
import {Player} from "./Player";
import {numeralWrapper} from "./ui/numeralFormat"; import {numeralWrapper} from "./ui/numeralFormat";
import {dialogBoxCreate} from "../utils/DialogBox"; import {dialogBoxCreate} from "../utils/DialogBox";
import {Reviver, Generic_toJSON, import {Reviver, Generic_toJSON,
@ -156,7 +155,6 @@ export function Gang(facName, hacking=false) {
this.members = []; //Array of GangMembers this.members = []; //Array of GangMembers
this.wanted = 1; this.wanted = 1;
this.respect = 1; this.respect = 1;
this.power = 0;
this.isHackingGang = hacking; this.isHackingGang = hacking;
@ -174,9 +172,19 @@ export function Gang(facName, hacking=false) {
this.territoryClashChance = 0; this.territoryClashChance = 0;
this.territoryWarfareEngaged = false; this.territoryWarfareEngaged = false;
this.notifyMemberDeath = true;
} }
Gang.prototype.process = function(numCycles=1) { Gang.prototype.power() = function() {
return AllGangs[this.facName].power;
}
Gang.prototype.getTerritory = function() {
return AllGangs[this.facName].territory;
}
Gang.prototype.process = function(numCycles=1, player) {
const CyclesPerSecond = 1000 / Engine._idleSpeed; const CyclesPerSecond = 1000 / Engine._idleSpeed;
if (isNaN(numCycles)) { if (isNaN(numCycles)) {
@ -189,7 +197,7 @@ Gang.prototype.process = function(numCycles=1) {
const cycles = Math.min(this.storedCycles, 10 * CyclesPerSecond); const cycles = Math.min(this.storedCycles, 10 * CyclesPerSecond);
try { try {
this.processGains(cycles); this.processGains(cycles, player);
this.processExperienceGains(cycles); this.processExperienceGains(cycles);
this.processTerritoryAndPowerGains(cycles); this.processTerritoryAndPowerGains(cycles);
this.storedCycles -= cycles; this.storedCycles -= cycles;
@ -198,13 +206,13 @@ Gang.prototype.process = function(numCycles=1) {
} }
} }
Gang.prototype.processGains = function(numCycles=1) { Gang.prototype.processGains = function(numCycles=1, player) {
//Get gains per cycle //Get gains per cycle
var moneyGains = 0, respectGains = 0, wantedLevelGains = 0; var moneyGains = 0, respectGains = 0, wantedLevelGains = 0;
for (var i = 0; i < this.members.length; ++i) { for (var i = 0; i < this.members.length; ++i) {
respectGains += (this.members[i].calculateRespectGain()); respectGains += (this.members[i].calculateRespectGain(this));
wantedLevelGains += (this.members[i].calculateWantedLevelGain()); wantedLevelGains += (this.members[i].calculateWantedLevelGain(this));
moneyGains += (this.members[i].calculateMoneyGain()); moneyGains += (this.members[i].calculateMoneyGain(this));
} }
this.respectGainRate = respectGains; this.respectGainRate = respectGains;
this.wantedGainRate = wantedLevelGains; this.wantedGainRate = wantedLevelGains;
@ -219,12 +227,12 @@ Gang.prototype.processGains = function(numCycles=1) {
dialogBoxCreate("ERROR: Could not get Faction associates with your gang. This is a bug, please report to game dev"); dialogBoxCreate("ERROR: Could not get Faction associates with your gang. This is a bug, please report to game dev");
} else { } else {
var favorMult = 1 + (fac.favor / 100); var favorMult = 1 + (fac.favor / 100);
fac.playerReputation += ((Player.faction_rep_mult * gain * favorMult) / GangRespectToReputationRatio); fac.playerReputation += ((player.faction_rep_mult * gain * favorMult) / GangRespectToReputationRatio);
} }
// Keep track of respect gained per member // Keep track of respect gained per member
for (let i = 0; i < this.members.length; ++i) { for (let i = 0; i < this.members.length; ++i) {
this.members[i].recordEarnedRespect(numCycles); this.members[i].recordEarnedRespect(numCycles, this);
} }
} else { } else {
console.warn("respectGains calculated to be NaN"); console.warn("respectGains calculated to be NaN");
@ -248,7 +256,7 @@ Gang.prototype.processGains = function(numCycles=1) {
console.warn("ERROR: wantedLevelGains is NaN"); console.warn("ERROR: wantedLevelGains is NaN");
} }
if (typeof moneyGains === "number") { if (typeof moneyGains === "number") {
Player.gainMoney(moneyGains * numCycles); player.gainMoney(moneyGains * numCycles);
} else { } else {
console.warn("ERROR: respectGains is NaN"); console.warn("ERROR: respectGains is NaN");
} }
@ -257,6 +265,7 @@ Gang.prototype.processGains = function(numCycles=1) {
Gang.prototype.processTerritoryAndPowerGains = function(numCycles=1) { Gang.prototype.processTerritoryAndPowerGains = function(numCycles=1) {
this.storedTerritoryAndPowerCycles += numCycles; this.storedTerritoryAndPowerCycles += numCycles;
if (this.storedTerritoryAndPowerCycles < CyclesPerTerritoryAndPowerUpdate) { return; } if (this.storedTerritoryAndPowerCycles < CyclesPerTerritoryAndPowerUpdate) { return; }
this.storedTerritoryAndPowerCycles -= CyclesPerTerritoryAndPowerUpdate;
// Process power first // Process power first
var gangName = this.facName; var gangName = this.facName;
@ -265,14 +274,22 @@ Gang.prototype.processTerritoryAndPowerGains = function(numCycles=1) {
if (name == gangName) { if (name == gangName) {
AllGangs[name].power += this.calculatePower(); AllGangs[name].power += this.calculatePower();
} else { } else {
var gain = Math.random() * 0.02; //TODO Adjust as necessary // Adjust these parameters as necessary
AllGangs[name].power += (gain); const additiveGain = Math.random() * AllGangs[name].territory;
AllGangs[name].power += (additiveGain);
AllGangs[name].power *= 1.01;
} }
} }
} }
// Determine if territory should be processed // Determine if territory should be processed
if (!this.territoryWarfareEngaged) { return; } if (this.territoryWarfareEngaged) {
this.territoryClashChance = 1;
} else if (this.territoryClashChance > 0) {
// Engagement turned off, but still a positive clash chance. So there's
// still a chance of clashing but it slowly goes down over time
this.territoryClashChance = Math.max(0, this.territoryClashChance - 0.005);
}
// Then process territory // Then process territory
for (var i = 0; i < GangNames.length; ++i) { for (var i = 0; i < GangNames.length; ++i) {
@ -287,7 +304,7 @@ Gang.prototype.processTerritoryAndPowerGains = function(numCycles=1) {
// If either of the gangs involved in this clash is the player, determine // If either of the gangs involved in this clash is the player, determine
// whether to skip or process it using the clash chance // whether to skip or process it using the clash chance
if (thisGang === gangName || otherGang === gangName) { if (thisGang === gangName || otherGang === gangName) {
if (!(Math.random() <= this.territoryClashChance)) { continue; } if (!(Math.random() < this.territoryClashChance)) { continue; }
} }
const thisPwr = AllGangs[thisGang].power; const thisPwr = AllGangs[thisGang].power;
@ -300,16 +317,28 @@ Gang.prototype.processTerritoryAndPowerGains = function(numCycles=1) {
} }
AllGangs[thisGang].territory += 0.0001; AllGangs[thisGang].territory += 0.0001;
AllGangs[otherGang].territory -= 0.0001; AllGangs[otherGang].territory -= 0.0001;
if (thisGang === gangName) {
this.clash(true); // Player won
} else if (otherGang === gangName) {
this.clash(false); // Player lost
} else {
AllGangs[otherGang].power *= (1 / 1.01);
}
} else { } else {
if (AllGangs[thisGang].territory <= 0) { if (AllGangs[thisGang].territory <= 0) {
return; return;
} }
AllGangs[thisGang].territory -= 0.0001; AllGangs[thisGang].territory -= 0.0001;
AllGangs[otherGang].territory += 0.0001; AllGangs[otherGang].territory += 0.0001;
if (thisGang === gangName) {
this.clash(false); // Player lost
} else if (otherGang === gangName) {
this.clash(true); // Player won
} else {
AllGangs[thisGang].power *= (1 / 1.01);
}
} }
} }
this.storedTerritoryAndPowerCycles -= CyclesPerTerritoryAndPowerUpdate;
} }
Gang.prototype.canRecruitMember = function() { Gang.prototype.canRecruitMember = function() {
@ -326,6 +355,24 @@ Gang.prototype.getRespectNeededToRecruitMember = function() {
return Math.round(0.7 * Math.pow(i, 3) + 0.8 * Math.pow(i, 2)); return Math.round(0.7 * Math.pow(i, 3) + 0.8 * Math.pow(i, 2));
} }
Gang.prototype.recruitMember = function(name) {
if (name === "" || !this.canRecruitMember()) { return false; }
// Check for already-existing names
let sameNames = this.members.filter((m) => {
return m.name === name;
});
if (sameNames.length >= 1) { return false; }
let member = new GangMember(name);
this.members.push(member);
if (routing.isOn(Page.Gang)) {
this.createGangMemberDisplayElement(member);
this.updateGangContent();
}
return true;
}
// Money and Respect gains multiplied by this number (< 1) // Money and Respect gains multiplied by this number (< 1)
Gang.prototype.getWantedPenalty = function() { Gang.prototype.getWantedPenalty = function() {
return (this.respect) / (this.respect + this.wanted); return (this.respect) / (this.respect + this.wanted);
@ -344,14 +391,61 @@ Gang.prototype.calculatePower = function() {
for (var i = 0; i < this.members.length; ++i) { for (var i = 0; i < this.members.length; ++i) {
if (this.members[i].task instanceof GangMemberTask && if (this.members[i].task instanceof GangMemberTask &&
this.members[i].task.name == "Territory Warfare") { this.members[i].task.name == "Territory Warfare") {
memberTotal += this.members[i].calculatePower(); const gain = this.members[i].calculatePower();
memberTotal += gain;
} }
} }
return (0.0005 * memberTotal); return (0.0005 * memberTotal);
} }
Gang.prototype.clash = function(won=false) {
// Determine if a gang member should die
let baseDeathChance;
won ? baseDeathChance = 0.05 : baseDeathChance = 0.1;
// If the clash was lost, the player loses a small percentage of power
if (!won) {
AllGangs[this.facName].power *= 0.98;
}
for (let i = this.members.length - 1; i >= 0; --i) {
const member = this.members[i];
// Only members assigned to Territory Warfare can die
if (member.task.name !== "Territory Warfare") { continue; }
// Chance to die is decreased based on defense
const modifiedDeathChance = baseDeathChance / Math.pow(def, 0.25);
if (Math.random() < modifiedDeathChance) {
this.killMember(member);
}
}
}
Gang.prototype.killMember = function(memberObj) { Gang.prototype.killMember = function(memberObj) {
// TODO const gangName = this.facName;
// Player loses a percentage of total respect, plus whatever respect that member has earned
const totalRespect = this.gang.respect;
const lostRespect = (0.05 * totalRespect) + memberObj.earnedRespect;
this.gang.respect = Math.max(0, totalRespect - lostRespect);
for (let i = 0; i < this.members.length; ++i) {
if (memberObj.name === this.members[i].name) {
this.members.splice(i, 1);
break;
}
}
// Notify of death
if (this.notifyMemberDeath) {
dialogBoxCreate(`${memberObj.name} was killed in a gang clash! You lost ${lostRespect} respect`);
}
// Update UI
if (routing.isOn(Page.Gang)) {
this.displayGangMemberList();
}
} }
Gang.prototype.ascendMember = function(memberObj) { Gang.prototype.ascendMember = function(memberObj) {
@ -380,6 +474,28 @@ Gang.prototype.ascendMember = function(memberObj) {
} }
} }
// Returns only valid tasks for this gang. Excludes 'Unassigned'
Gang.prototype.getAllTaskNames = function() {
let tasks = [];
const allTasks = Object.keys(GangMemberTasks);
if (this.isHackingGang) {
tasks = allTasks.filter((e) => {
let task = GangMemberTasks[e];
if (task == null) { return false; }
if (e === "Unassigned") { return false; }
return task.isHacking;
});
} else {
tasks = allTasks.filter((e) => {
let task = GangMemberTasks[e];
if (task == null) { return false; }
if (e === "Unassigned") { return false; }
return task.isCombat;
});
}
return tasks;
}
Gang.prototype.toJSON = function() { Gang.prototype.toJSON = function() {
return Generic_toJSON("Gang", this); return Generic_toJSON("Gang", this);
} }
@ -466,7 +582,7 @@ GangMember.prototype.unassignFromTask = function() {
} }
//Gains are per cycle //Gains are per cycle
GangMember.prototype.calculateRespectGain = function() { GangMember.prototype.calculateRespectGain = function(gang) {
var task = this.task; var task = this.task;
if (task == null || !(task instanceof GangMemberTask) || task.baseRespect === 0) {return 0;} if (task == null || !(task instanceof GangMemberTask) || task.baseRespect === 0) {return 0;}
var statWeight = (task.hackWeight/100) * this.hack + var statWeight = (task.hackWeight/100) * this.hack +
@ -477,13 +593,13 @@ GangMember.prototype.calculateRespectGain = function() {
(task.chaWeight/100) * this.cha; (task.chaWeight/100) * this.cha;
statWeight -= (3.5 * task.difficulty); statWeight -= (3.5 * task.difficulty);
if (statWeight <= 0) { return 0; } if (statWeight <= 0) { return 0; }
var territoryMult = AllGangs[Player.gang.facName].territory; var territoryMult = AllGangs[gang.facName].territory;
if (territoryMult <= 0) { return 0; } if (territoryMult <= 0) { return 0; }
var respectMult = Player.gang.getWantedPenalty(); var respectMult = gang.getWantedPenalty();
return 12 * task.baseRespect * statWeight * territoryMult * respectMult; return 12 * task.baseRespect * statWeight * territoryMult * respectMult;
} }
GangMember.prototype.calculateWantedLevelGain = function() { GangMember.prototype.calculateWantedLevelGain = function(gang) {
var task = this.task; var task = this.task;
if (task == null || !(task instanceof GangMemberTask) || task.baseWanted === 0) {return 0;} if (task == null || !(task instanceof GangMemberTask) || task.baseWanted === 0) {return 0;}
var statWeight = (task.hackWeight/100) * this.hack + var statWeight = (task.hackWeight/100) * this.hack +
@ -494,7 +610,7 @@ GangMember.prototype.calculateWantedLevelGain = function() {
(task.chaWeight/100) * this.cha; (task.chaWeight/100) * this.cha;
statWeight -= (3.5 * task.difficulty); statWeight -= (3.5 * task.difficulty);
if (statWeight <= 0) {return 0;} if (statWeight <= 0) {return 0;}
var territoryMult = AllGangs[Player.gang.facName].territory; var territoryMult = AllGangs[gang.facName].territory;
if (territoryMult <= 0) {return 0;} if (territoryMult <= 0) {return 0;}
if (task.baseWanted < 0) { if (task.baseWanted < 0) {
return task.baseWanted * statWeight * territoryMult; return task.baseWanted * statWeight * territoryMult;
@ -503,7 +619,7 @@ GangMember.prototype.calculateWantedLevelGain = function() {
} }
} }
GangMember.prototype.calculateMoneyGain = function() { GangMember.prototype.calculateMoneyGain = function(gang) {
var task = this.task; var task = this.task;
if (task == null || !(task instanceof GangMemberTask) || task.baseMoney === 0) {return 0;} if (task == null || !(task instanceof GangMemberTask) || task.baseMoney === 0) {return 0;}
var statWeight = (task.hackWeight/100) * this.hack + var statWeight = (task.hackWeight/100) * this.hack +
@ -514,9 +630,9 @@ GangMember.prototype.calculateMoneyGain = function() {
(task.chaWeight/100) * this.cha; (task.chaWeight/100) * this.cha;
statWeight -= (3.5 * task.difficulty); statWeight -= (3.5 * task.difficulty);
if (statWeight <= 0) {return 0;} if (statWeight <= 0) {return 0;}
var territoryMult = AllGangs[Player.gang.facName].territory; var territoryMult = AllGangs[gang.facName].territory;
if (territoryMult <= 0) {return 0;} if (territoryMult <= 0) {return 0;}
var respectMult = Player.gang.getWantedPenalty(); var respectMult = gang.getWantedPenalty();
return 5 * task.baseMoney * statWeight * territoryMult * respectMult; return 5 * task.baseMoney * statWeight * territoryMult * respectMult;
} }
@ -531,8 +647,8 @@ GangMember.prototype.gainExperience = function(numCycles=1) {
this.cha_exp += (task.chaWeight / 1500) * task.difficulty * numCycles; this.cha_exp += (task.chaWeight / 1500) * task.difficulty * numCycles;
} }
GangMember.prototype.recordEarnedRespect = function(numCycles=1) { GangMember.prototype.recordEarnedRespect = function(numCycles=1, gang) {
this.earnedRespect += (this.calculateRespectGain() * numCycles); this.earnedRespect += (this.calculateRespectGain(gang) * numCycles);
} }
GangMember.prototype.ascend = function() { GangMember.prototype.ascend = function() {
@ -617,6 +733,24 @@ GangMember.prototype.getAscensionResults = function() {
} }
} }
GangMember.prototype.buyUpgrade = function(upg, player, gang) {
if (!(upg instanceof GangMemberUpgrade)) {
throw new Error(`Invalid 'upg' argument passed into GangMember.buyUpgrade`);
}
if (player.money.lt(upg.cost)) { return false; }
player.loseMoney(upg.cost);
if (upg.type === "g") {
this.augmentations.push(upg.name);
} else {
this.upgrades.push(upg.name);
}
upg.apply(this);
if (routing.isOn(Page.Gang) && UIElems.gangMemberUpgradeBoxOpened) {
var initFilterValue = UIElems.gangMemberUpgradeBoxFilter.value.toString();
gang.createGangMemberUpgradeBox(player, initFilterValue);
}
}
GangMember.prototype.toJSON = function() { GangMember.prototype.toJSON = function() {
return Generic_toJSON("GangMember", this); return Generic_toJSON("GangMember", this);
} }
@ -744,7 +878,7 @@ gangMemberUpgradesMetadata.forEach((e) => {
}); });
// Create a pop-up box that lets player purchase upgrades // Create a pop-up box that lets player purchase upgrades
Gang.prototype.createGangMemberUpgradeBox = function(initialFilter="") { Gang.prototype.createGangMemberUpgradeBox = function(player, initialFilter="") {
const boxId = "gang-member-upgrade-popup-box"; const boxId = "gang-member-upgrade-popup-box";
if (UIElems.gangMemberUpgradeBoxOpened) { if (UIElems.gangMemberUpgradeBoxOpened) {
//Already opened, refreshing //Already opened, refreshing
@ -759,9 +893,9 @@ Gang.prototype.createGangMemberUpgradeBox = function(initialFilter="") {
UIElems.gangMemberUpgradeBoxElements = [UIElems.gangMemberUpgradeBoxFilter]; UIElems.gangMemberUpgradeBoxElements = [UIElems.gangMemberUpgradeBoxFilter];
var filter = UIElems.gangMemberUpgradeBoxFilter.value.toString(); var filter = UIElems.gangMemberUpgradeBoxFilter.value.toString();
for (var i = 0; i < Player.gang.members.length; ++i) { for (var i = 0; i < this.members.length; ++i) {
if (Player.gang.members[i].name.indexOf(filter) > -1 || Player.gang.members[i].task.name.indexOf(filter) > -1) { if (this.members[i].name.indexOf(filter) > -1 || this.members[i].task.name.indexOf(filter) > -1) {
var newPanel = Player.gang.members[i].createGangMemberUpgradePanel(this); var newPanel = this.members[i].createGangMemberUpgradePanel(this, player);
UIElems.gangMemberUpgradeBoxContent.appendChild(newPanel); UIElems.gangMemberUpgradeBoxContent.appendChild(newPanel);
UIElems.gangMemberUpgradeBoxElements.push(newPanel); UIElems.gangMemberUpgradeBoxElements.push(newPanel);
} }
@ -773,7 +907,7 @@ Gang.prototype.createGangMemberUpgradeBox = function(initialFilter="") {
value:initialFilter, value:initialFilter,
onkeyup:()=>{ onkeyup:()=>{
var filterValue = UIElems.gangMemberUpgradeBoxFilter.value.toString(); var filterValue = UIElems.gangMemberUpgradeBoxFilter.value.toString();
this.createGangMemberUpgradeBox(filterValue); this.createGangMemberUpgradeBox(player, filterValue);
} }
}); });
@ -782,7 +916,7 @@ Gang.prototype.createGangMemberUpgradeBox = function(initialFilter="") {
var filter = UIElems.gangMemberUpgradeBoxFilter.value.toString(); var filter = UIElems.gangMemberUpgradeBoxFilter.value.toString();
for (var i = 0; i < this.members.length; ++i) { for (var i = 0; i < this.members.length; ++i) {
if (this.members[i].name.indexOf(filter) > -1 || this.members[i].task.name.indexOf(filter) > -1) { if (this.members[i].name.indexOf(filter) > -1 || this.members[i].task.name.indexOf(filter) > -1) {
UIElems.gangMemberUpgradeBoxElements.push(this.members[i].createGangMemberUpgradePanel(this)); UIElems.gangMemberUpgradeBoxElements.push(this.members[i].createGangMemberUpgradePanel(this, player));
} }
} }
@ -793,7 +927,7 @@ Gang.prototype.createGangMemberUpgradeBox = function(initialFilter="") {
} }
//Create upgrade panels for each individual Gang Member //Create upgrade panels for each individual Gang Member
GangMember.prototype.createGangMemberUpgradePanel = function(gangObj) { GangMember.prototype.createGangMemberUpgradePanel = function(gangObj, player) {
var container = createElement("div", { var container = createElement("div", {
border:"1px solid white", border:"1px solid white",
}); });
@ -850,7 +984,7 @@ GangMember.prototype.createGangMemberUpgradePanel = function(gangObj) {
for (let upgName in GangMemberUpgrades) { for (let upgName in GangMemberUpgrades) {
if (GangMemberUpgrades.hasOwnProperty(upgName)) { if (GangMemberUpgrades.hasOwnProperty(upgName)) {
let upg = GangMemberUpgrades[upgName]; let upg = GangMemberUpgrades[upgName];
if (Player.money.lt(upg.cost)) { continue; } if (player.money.lt(upg.cost)) { continue; }
if (this.upgrades.includes(upgName) || this.augmentations.includes(upgName)) { continue; } if (this.upgrades.includes(upgName) || this.augmentations.includes(upgName)) { continue; }
switch (upg.type) { switch (upg.type) {
case "w": case "w":
@ -902,16 +1036,7 @@ GangMember.prototype.createGangMemberUpgradePanel = function(gangObj) {
class:"a-link-button", margin:"2px", padding:"2px", display:"block", class:"a-link-button", margin:"2px", padding:"2px", display:"block",
fontSize:"11px", fontSize:"11px",
clickListener:()=>{ clickListener:()=>{
if (Player.money.lt(upg.cost)) { return false; } memberObj.buyUpgrade(upg, player, gangObj);
Player.loseMoney(upg.cost);
if (upg.type === "g") {
memberObj.augmentations.push(upg.name);
} else {
memberObj.upgrades.push(upg.name);
}
upg.apply(memberObj);
var initFilterValue = UIElems.gangMemberUpgradeBoxFilter.value.toString();
gangObj.createGangMemberUpgradeBox(initFilterValue);
return false; return false;
} }
} }
@ -971,10 +1096,12 @@ const UIElems = {
gangTerritoryWarfareCheckbox: null, gangTerritoryWarfareCheckbox: null,
gangTerritoryWarfareCheckboxLabel: null, gangTerritoryWarfareCheckboxLabel: null,
gangTerritoryWarfareClashChance: null, gangTerritoryWarfareClashChance: null,
gangTerritoryDeathNotifyCheckbox: null,
gangTerritoryDeathNotifyCheckboxLabel: null,
gangTerritoryInfoText: null, gangTerritoryInfoText: null,
} }
Gang.prototype.displayGangContent = function() { Gang.prototype.displayGangContent = function(player) {
if (!UIElems.gangContentCreated || UIElems.gangContainer == null) { if (!UIElems.gangContentCreated || UIElems.gangContainer == null) {
UIElems.gangContentCreated = true; UIElems.gangContentCreated = true;
@ -1084,25 +1211,21 @@ Gang.prototype.displayGangContent = function() {
class: "std-button", class: "std-button",
clickListener: () => { clickListener: () => {
let name = nameInput.value; let name = nameInput.value;
if (name === "") {
// Check for already-existing names dialogBoxCreate("You must enter a name for your Gang member!");
let sameNames = this.members.filter((m) => { return false;
return m.name === name; }
}); if (!this.canRecruitMember()) {
if (sameNames.length >= 1) { dialogBoxCreate("You cannot recruit another Gang member!");
dialogBoxCreate("You already have a gang member with this name!");
return false; return false;
} }
if (name === "") { // At this point, the only way this can fail is if you already
dialogBoxCreate("You must enter a name for your Gang member!"); // have a gang member with the same name
} else { if (!this.recruitMember(name)) {
let member = new GangMember(name); dialogBoxCreate("You already have a gang member with this name!");
this.members.push(member);
this.createGangMemberDisplayElement(member);
this.updateGangContent();
removeElementById(popupId);
} }
return false; return false;
}, },
innerText: "Recruit Gang Member", innerText: "Recruit Gang Member",
@ -1168,7 +1291,7 @@ Gang.prototype.displayGangContent = function() {
class:"a-link-button", display:"inline-block", class:"a-link-button", display:"inline-block",
innerHTML:"Manage Equipment", innerHTML:"Manage Equipment",
clickListener: () => { clickListener: () => {
this.createGangMemberUpgradeBox(); this.createGangMemberUpgradeBox(player);
} }
}); });
UIElems.gangManagementSubpage.appendChild(UIElems.gangExpandAllButton); UIElems.gangManagementSubpage.appendChild(UIElems.gangExpandAllButton);
@ -1189,13 +1312,16 @@ Gang.prototype.displayGangContent = function() {
//Info text for territory page //Info text for territory page
UIElems.gangTerritoryDescText = createElement("p", { UIElems.gangTerritoryDescText = createElement("p", {
width:"70%", width:"70%",
innerHTML:"This page shows how much territory your Gang controls. This statistic is listed as a percentage, " + innerHTML:
"This page shows how much territory your Gang controls. This statistic is listed as a percentage, " +
"which represents how much of the total territory you control.<br><br>" + "which represents how much of the total territory you control.<br><br>" +
"Territory gain and loss is processed automatically and is updated every ~20 seconds. Your chances " + "Every ~20 seconds, your gang has a chance to 'clash' with other gangs. Your chance " +
"to gain and lose territory depend on your Gang's power, which is listed in the display below. " + "to win a clash depends on your gang's power, which is listed in the display below. " +
"Your gang's power is determined by the stats of all Gang members you have assigned to the " + "Your gang's power slowly accumulates over time. The accumulation rate is determined by the stats " +
"'Territory Warfare' task. Gang members that are not assigned to this task do not contribute to " + "of all Gang members you have assigned to the 'Territory Warfare' task. Gang members that are not " +
"your Gang's power.<br><br>" + "assigned to this task do not contribute to your gang's power.<br><br>" +
"NOTE: Gang members assigned to 'Territory Warfare' can be killed during clashes. This can happen regardless of whether you win " +
"or lose the clash. A gang member being killed results in both respect and power loss for your gang.<br><br>" +
"The amount of territory you have affects all aspects of your Gang members' production, including " + "The amount of territory you have affects all aspects of your Gang members' production, including " +
"money, respect, and wanted level. It is very beneficial to have high territory control.<br><br>" "money, respect, and wanted level. It is very beneficial to have high territory control.<br><br>"
}); });
@ -1215,20 +1341,64 @@ Gang.prototype.displayGangContent = function() {
UIElems.gangTerritoryWarfareCheckboxLabel = createElement("label", { UIElems.gangTerritoryWarfareCheckboxLabel = createElement("label", {
color: "white", color: "white",
display: "inline-block",
for: "gang-management-territory-warfare-checkbox", for: "gang-management-territory-warfare-checkbox",
innerText: "Engage in Territory Warfare", innerText: "Engage in Territory Warfare",
tooltip: "Test", tooltip: "Engaging in Territory Warfare sets your clash chance to 100%. " +
"Disengaging will cause your clash chance to gradually decrease until " +
"it reaches 0%",
}); });
UIElems.gangTerritorySubpage.appendChild(UIElems.gangTerritoryWarfareCheckbox); UIElems.gangTerritorySubpage.appendChild(UIElems.gangTerritoryWarfareCheckbox);
UIElems.gangTerritorySubpage.appendChild(UIElems.gangTerritoryWarfareCheckboxLabel); UIElems.gangTerritorySubpage.appendChild(UIElems.gangTerritoryWarfareCheckboxLabel);
// Territory Clash chance // Territory Clash chance
UIElems.gangTerritoryWarfareClashChance = createElement("p"); UIElems.gangTerritorySubpage.appendChild(createElement("br"));
UIElems.gangTerritoryWarfareClashChance = createElement("p", {display: "inline-block"});
UIElems.gangTerritorySubpage.appendChild(UIElems.gangTerritoryWarfareClashChance); UIElems.gangTerritorySubpage.appendChild(UIElems.gangTerritoryWarfareClashChance);
UIElems.gangTerritorySubpage.appendChild(createElement("div", {
class: "help-tip",
display: "inline-block",
innerText: "?",
clickListener: () => {
dialogBoxCreate("This percentage represents the chance you have of 'clashing' with " +
"with another gang. If you do not wish to gain/lose territory, " +
"then keep this percentage at 0% by not engaging in territory " +
"warfare.")
},
}));
// Checkbox for whether player wants to be notified of gang member death
UIElems.gangTerritoryDeathNotifyCheckbox = createElement("input", {
display: "inline-block",
id: "gang-management-notify-member-death-checkbox",
changeListener: () => {
this.notifyMemberDeath = UIElems.gangTerritoryDeathNotifyCheckbox.checked;
},
margin: "2px",
type: "checkbox",
});
UIElems.gangTerritoryDeathNotifyCheckbox.checked = this.notifyMemberDeath;
UIElems.gangTerritoryDeathNotifyCheckboxLabel = createElement("label", {
color: "white",
display: "inline-block",
for: "gang-management-notify-member-death-checkbox",
innerText: "Notify about Gang Member Deaths",
tooltip: "If this is enabled, then you will receive a pop-up notifying you " +
"whenever one of your Gang Members dies in a territory clash.",
});
UIElems.gangTerritorySubpage.appendChild(createElement("br"));
UIElems.gangTerritorySubpage.appendChild(UIElems.gangTerritoryDeathNotifyCheckbox);
UIElems.gangTerritorySubpage.appendChild(UIElems.gangTerritoryDeathNotifyCheckboxLabel);
// Territory info (percentages of territory owned for each gang) // Territory info (percentages of territory owned for each gang)
UIElems.gangTerritorySubpage.appendChild(createElement("br")); UIElems.gangTerritorySubpage.appendChild(createElement("br"));
var territoryBorder = createElement("fieldset", {width:"50%", display:"block"}); var territoryBorder = createElement("fieldset", {
display:"block",
margin: "6px",
width:"50%",
});
UIElems.gangTerritoryInfoText = createElement("p"); UIElems.gangTerritoryInfoText = createElement("p");
@ -1261,7 +1431,7 @@ Gang.prototype.updateGangContent = function() {
if (UIElems.gangTerritorySubpage.style.display === "block") { if (UIElems.gangTerritorySubpage.style.display === "block") {
// Territory Warfare Clash Chance // Territory Warfare Clash Chance
UIElems.gangTerritoryWarfareClashChance.innerText = UIElems.gangTerritoryWarfareClashChance.innerText =
`Territory Clash Chance: ${numeralWrapper.format(this.gangTerritoryWarfareClashChance, '0.000%')}`; `Territory Clash Chance: ${numeralWrapper.format(this.territoryClashChance, '0.000%')}`;
// Update territory information // Update territory information
UIElems.gangTerritoryInfoText.innerHTML = ""; UIElems.gangTerritoryInfoText.innerHTML = "";
@ -1496,23 +1666,7 @@ Gang.prototype.createGangMemberDisplayElement = function(memberObj) {
}); });
// Get an array of the name of all tasks that are applicable for this Gang // Get an array of the name of all tasks that are applicable for this Gang
let tasks = null; let tasks = this.getAllTaskNames();
const allTasks = Object.keys(GangMemberTasks);
if (Player.gang.isHackingGang) {
tasks = allTasks.filter((e) => {
let task = GangMemberTasks[e];
if (task == null) { return false; }
if (e === "Unassigned") { return false; }
return task.isHacking;
});
} else {
tasks = allTasks.filter((e) => {
let task = GangMemberTasks[e];
if (task == null) { return false; }
if (e === "Unassigned") { return false; }
return task.isCombat;
});
}
tasks.unshift("---"); tasks.unshift("---");
// Create selector for Gang member task // Create selector for Gang member task
@ -1567,7 +1721,7 @@ Gang.prototype.createGangMemberDisplayElement = function(memberObj) {
} }
Gang.prototype.updateGangMemberDisplayElement = function(memberObj) { Gang.prototype.updateGangMemberDisplayElement = function(memberObj) {
if (!UIElems.gangContentCreated || !Player.inGang()) {return;} if (!UIElems.gangContentCreated) { return; }
var name = memberObj.name; var name = memberObj.name;
// Update stats + exp // Update stats + exp
@ -1601,9 +1755,9 @@ Gang.prototype.updateGangMemberDisplayElement = function(memberObj) {
var gainInfo = document.getElementById(name + "gang-member-gain-info"); var gainInfo = document.getElementById(name + "gang-member-gain-info");
if (gainInfo) { if (gainInfo) {
gainInfo.innerHTML = gainInfo.innerHTML =
[`Money: $ ${formatNumber(5*memberObj.calculateMoneyGain(), 2)} / sec`, [`Money: $ ${formatNumber(5*memberObj.calculateMoneyGain(this), 2)} / sec`,
`Respect: ${formatNumber(5*memberObj.calculateRespectGain(), 6)} / sec`, `Respect: ${formatNumber(5*memberObj.calculateRespectGain(this), 6)} / sec`,
`Wanted Level: ${formatNumber(5*memberObj.calculateWantedLevelGain(), 6)} / sec`, `Wanted Level: ${formatNumber(5*memberObj.calculateWantedLevelGain(this), 6)} / sec`,
`Total Respect Earned: ${formatNumber(memberObj.earnedRespect, 6)}`].join("<br>"); `Total Respect Earned: ${formatNumber(memberObj.earnedRespect, 6)}`].join("<br>");
} }
} }

@ -9,7 +9,7 @@ function unknownBladeburnerActionErrorMessage(functionName, actionType, actionNa
} }
function unknownBladeburnerExceptionMessage(functionName, err) { function unknownBladeburnerExceptionMessage(functionName, err) {
return `Bladeburner.${functionName}() failed with exception: ` + err; return `bladeburner.${functionName}() failed with exception: ` + err;
} }
function checkBladeburnerAccess(workerScript, functionName) { function checkBladeburnerAccess(workerScript, functionName) {

@ -46,7 +46,8 @@ import {TextFile, getTextFile, createTextFile} from "./TextFile";
import {unknownBladeburnerActionErrorMessage, import {unknownBladeburnerActionErrorMessage,
unknownBladeburnerExceptionMessage, unknownBladeburnerExceptionMessage,
checkBladeburnerAccess} from "./NetscriptBladeburner.js"; checkBladeburnerAccess} from "./NetscriptBladeburner";
import * as nsGang from "./NetscriptGang";
import {WorkerScript, workerScripts, import {WorkerScript, workerScripts,
killWorkerScript, NetscriptPorts} from "./NetscriptWorker"; killWorkerScript, NetscriptPorts} from "./NetscriptWorker";
import {makeRuntimeRejectMsg, netscriptDelay, import {makeRuntimeRejectMsg, netscriptDelay,
@ -1101,7 +1102,7 @@ function NetscriptFunctions(workerScript) {
throw makeRuntimeRejectMsg(workerScript, "ps() failed. Invalid IP or hostname passed in: " + ip); throw makeRuntimeRejectMsg(workerScript, "ps() failed. Invalid IP or hostname passed in: " + ip);
} }
const processes = []; const processes = [];
for(const i in server.runningScripts) { for (const i in server.runningScripts) {
const script = server.runningScripts[i]; const script = server.runningScripts[i];
processes.push({filename:script.filename, threads: script.threads, args: script.args.slice()}) processes.push({filename:script.filename, threads: script.threads, args: script.args.slice()})
} }
@ -3534,7 +3535,196 @@ function NetscriptFunctions(workerScript) {
return true; return true;
}, },
//Bladeburner API // Gang API
gang : {
getMemberNames : function() {
if (workerScript.checkingRam) {
return updateStaticRam("getMemberNames", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("getMemberNames", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "getMemberNames");
try {
const names = [];
for (const member of Player.gang.members) {
names.push(member.name);
}
return names;
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("getMemberNames", e));
}
},
getGangInformation : function() {
if (workerScript.checkingRam) {
return updateStaticRam("getGangInformation", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("getGangInformation", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "getGangInformation");
try {
return {
faction: Player.gang.facName,
isHacking: Player.gang.isHackingGang,
moneyGainRate: Player.gang.moneyGainRate,
power: Player.gang.getPower(),
respect: Player.gang.respect,
respectGainRate: Player.gang.respectGainRate,
territory: Player.gang.getTerritory(),
territoryClashChance: Player.gang.territoryClashChance,
territoryWarfareEngaged: Player.gang.territoryWarfareEngaged,
wantedLevel: Player.gang.wanted,
wantedLevelGainRate: Player.gang.wantedGainRate,
}
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("getGangInformation", e));
}
},
getMemberInformation : function(name) {
if (workerScript.checkingRam) {
return updateStaticRam("getMemberInformation", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("getMemberInformation", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "getMemberInformation");
try {
for (const member of Player.gang.members) {
if (member.name === name) {
return {
agility: member.agi,
agilityEquipMult: member.agi_mult,
agilityAscensionMult: member.agi_asc_mult,
charisma: member.cha,
defense: member.def,
dexterity: member.dex,
hacking: member.hack,
strength: member.str
task: member.task.name,
}
}
}
return {}; // Member could not be found
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("getMemberInformation", e));
}
},
canRecruitMember : function() {
if (workerScript.checkingRam) {
return updateStaticRam("canRecruitMember", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("canRecruitMember", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "canRecruitMember");
try {
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("canRecruitMember", e));
}
},
recruitMember : function() {
if (workerScript.checkingRam) {
return updateStaticRam("recruitMember", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("recruitMember", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "recruitMember");
try {
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("recruitMember", e));
}
},
getTaskNames : function() {
if (workerScript.checkingRam) {
return updateStaticRam("getTaskNames", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("getTaskNames", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "getTaskNames");
try {
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("getTaskNames", e));
}
},
setMemberTask : function() {
if (workerScript.checkingRam) {
return updateStaticRam("setMemberTask", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("setMemberTask", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "setMemberTask");
try {
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("setMemberTask", e));
}
},
getEquipmentNames : function() {
if (workerScript.checkingRam) {
return updateStaticRam("getEquipmentNames", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("getEquipmentNames", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "getEquipmentNames");
try {
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("getEquipmentNames", e));
}
},
getEquipmentCost : function() {
if (workerScript.checkingRam) {
return updateStaticRam("getEquipmentCost", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("getEquipmentCost", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "getEquipmentCost");
try {
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("getEquipmentCost", e));
}
},
purchaseEquipment : function() {
if (workerScript.checkingRam) {
return updateStaticRam("purchaseEquipment", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("purchaseEquipment", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "purchaseEquipment");
try {
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("purchaseEquipment", e));
}
},
setTerritoryWarfare : function(engage) {
if (workerScript.checkingRam) {
return updateStaticRam("setTerritoryWarfare", CONSTANTS.ScriptGangApiBaseRamCost / 2);
}
updateDynamicRam("setTerritoryWarfare", CONSTANTS.ScriptGangApiBaseRamCost / 2);
nsGang.checkGangApiAccess(workerScript, "setTerritoryWarfare");
try {
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("setTerritoryWarfare", e));
}
},
getBonusTime : function() {
if (workerScript.checkingRam) { return 0; }
nsGang.checkGangApiAccess(workerScript, "getBonusTime");
try {
return Math.round(Player.gang.storedCycles / 5);
} catch(e) {
throw makeRuntimeRejectMsg(workerScript, nsGang.unknownGangApiExceptionMessage("getBonusTime", e));
}
},
}, // end gang namespace
// Bladeburner API
bladeburner : { bladeburner : {
getContractNames : function() { getContractNames : function() {
if (workerScript.checkingRam) { if (workerScript.checkingRam) {

17
src/NetscriptGang.js Normal file

@ -0,0 +1,17 @@
import {Player} from "./Player";
import {Gang} from "./Gang";
import {makeRuntimeRejectMsg} from "./NetscriptEvaluator";
function unknownGangApiExceptionMessage(functionName, err) {
return `gang.${functionName}() failed with exception: ` + err;
}
function checkGangApiAccess(workerScript, functionName) {
const accessDenied = `gang.${functionName}() failed because you do not currently have a Gang`;
const hasAccess = Player.gang instanceof Gang;
if (!hasAccess) {
throw makeRuntimeRejectMsg(workerScript, accessDenied);
}
}
export {unknownBladeburnerActionErrorMessage, unknownBladeburnerExceptionMessage, checkBladeburnerAccess};

@ -219,7 +219,8 @@ function scriptEditorInit() {
}); });
//Get functions from namespaces //Get functions from namespaces
if (name === "bladeburner" || name === "hacknet") { const namespaces = ["bladeburner", "hacknet", "codingcontract", "gang"];
if (namespaces.includes(name)) {
let namespace = fns[name]; let namespace = fns[name];
if (typeof namespace !== "object") {continue;} if (typeof namespace !== "object") {continue;}
let namespaceFns = Object.keys(namespace); let namespaceFns = Object.keys(namespace);
@ -557,6 +558,8 @@ async function parseOnlyRamCalculate(server, code, workerScript) {
func = workerScript.env.vars.bladeburner[ref]; func = workerScript.env.vars.bladeburner[ref];
} else if (ref in workerScript.env.vars.codingcontract) { } else if (ref in workerScript.env.vars.codingcontract) {
func = workerScript.env.vars.codingcontract[ref]; func = workerScript.env.vars.codingcontract[ref];
} else if (ref in workerScript.env.vars.gang) {
func = workerScript.env.vars.gang[ref];
} else { } else {
func = workerScript.env.get(ref); func = workerScript.env.get(ref);
} }

@ -419,7 +419,7 @@ const Engine = {
loadGangContent: function() { loadGangContent: function() {
Engine.hideAllContent(); Engine.hideAllContent();
if (document.getElementById("gang-container") || Player.inGang()) { if (document.getElementById("gang-container") || Player.inGang()) {
Player.gang.displayGangContent(); Player.gang.displayGangContent(Player);
routing.navigateTo(Page.Gang); routing.navigateTo(Page.Gang);
} else { } else {
Engine.loadTerminalContent(); Engine.loadTerminalContent();
@ -888,7 +888,7 @@ const Engine = {
//Gang, if applicable //Gang, if applicable
if (Player.bitNodeN == 2 && Player.inGang()) { if (Player.bitNodeN == 2 && Player.inGang()) {
Player.gang.process(numCycles); Player.gang.process(numCycles, Player);
} }
//Mission //Mission
@ -1313,7 +1313,7 @@ const Engine = {
//Gang progress for BitNode 2 //Gang progress for BitNode 2
if (Player.bitNodeN != null && Player.bitNodeN === 2 && Player.inGang()) { if (Player.bitNodeN != null && Player.bitNodeN === 2 && Player.inGang()) {
Player.gang.process(numCyclesOffline); Player.gang.process(numCyclesOffline, Player);
} }
//Bladeburner offline progress //Bladeburner offline progress