diff --git a/css/futureSocieties/fsPassage.css b/css/futureSocieties/fsPassage.css new file mode 100644 index 0000000000000000000000000000000000000000..6c61e92f93fa9d49e9469076acd6114a9eedec96 --- /dev/null +++ b/css/futureSocieties/fsPassage.css @@ -0,0 +1,26 @@ +.fs-recommend { + border: 1px solid; + padding: 2px; + border-radius: 3px; + font-size: smaller; +} + +.fs-recommend-great { + border-color: green; + color: green; +} + +.fs-recommend-good { + border-color: greenyellow; + color: greenyellow; +} + +.fs-recommend-neutral { + border-color: gold; + color: gold; +} + +.fs-recommend-bad { + border-color: red; + color: red; +} \ No newline at end of file diff --git a/src/endWeek/saLongTermEffects.js b/src/endWeek/saLongTermEffects.js index 5aeafe07e879219c5f4927614212ee6fbace6970..55f93ffb0b979ec9d38667f6a232c4ca3bb439ac 100644 --- a/src/endWeek/saLongTermEffects.js +++ b/src/endWeek/saLongTermEffects.js @@ -59,7 +59,7 @@ App.SlaveAssignment.longTermEffects = function saLongTermEffects(slave) { solidSlaveFoodEffects(slave); } } - r.push(App.SlaveAssignment.saSocialEffects(slave)); + r.push(App.SlaveAssignment.saSocialEffects(slave).report()); if (slave.fuckdoll === 0) { // swap to fuckdoll suit in the future brandEffects(slave); if (slave.fetish !== "mindbroken") { diff --git a/src/endWeek/saSocialEffects.js b/src/endWeek/saSocialEffects.js index 8fb51f0f20c9d530a2341f230f2a62972781e734..d4b42dfaca0d33773e942f325b0a863f1caf3677 100644 --- a/src/endWeek/saSocialEffects.js +++ b/src/endWeek/saSocialEffects.js @@ -1,6 +1,5 @@ /** Apply and return description of social effects * @param {FC.ReportSlave} slave - * @returns {DocumentFragment} */ App.SlaveAssignment.saSocialEffects = function(slave) { const {His, his, him, he, girl, wife} = getPronouns(slave); @@ -993,65 +992,92 @@ App.SlaveAssignment.saSocialEffects = function(slave) { return el; } - function displayCompressed() { - const positiveSum = socialEffects.filter(e => e.magnitude > 0).reduce((acc, cur) => acc += cur.magnitude, 0); - const negativeSum = socialEffects.filter(e => e.magnitude < 0).reduce((acc, cur) => acc += cur.magnitude, 0); + function report() { + function displayCompressed() { + const positiveSum = socialEffects.filter(e => e.magnitude > 0).reduce((acc, cur) => acc += cur.magnitude, 0); + const negativeSum = socialEffects.filter(e => e.magnitude < 0).reduce((acc, cur) => acc += cur.magnitude, 0); - const sumFrag = document.createDocumentFragment(); - sumFrag.append(`(`); - App.UI.DOM.appendNewElement("span", sumFrag, '+' + positiveSum.toString(), positiveSum > 0 ? "green" : "gray"); - sumFrag.append(`/`); - App.UI.DOM.appendNewElement("span", sumFrag, '-' + Math.abs(negativeSum).toString(), negativeSum < 0 ? "red" : "gray"); // literal '-' + Math.abs needed to handle 0 case - sumFrag.append(`)`); + const sumFrag = document.createDocumentFragment(); + sumFrag.append(`(`); + App.UI.DOM.appendNewElement("span", sumFrag, '+' + positiveSum.toString(), positiveSum > 0 ? "green" : "gray"); + sumFrag.append(`/`); + App.UI.DOM.appendNewElement("span", sumFrag, '-' + Math.abs(negativeSum).toString(), negativeSum < 0 ? "red" : "gray"); // literal '-' + Math.abs needed to handle 0 case + sumFrag.append(`)`); - const sum = positiveSum + negativeSum; - frag.append(`Society has a `); - if (sum > 0) { - const opinion = App.UI.DOM.makeElement("span", "positive", ["green", "major-link", "has-tooltip"]); - tippy(opinion, {content: renderTooltip(), placement: "right"}); - frag.append(opinion, ` overall view of ${slave.slaveName} `, sumFrag, `, which improves your reputation and advances social progress.`); - } else if (sum === 0) { - const opinion = App.UI.DOM.makeElement("span", "neutral", ["yellow", "major-link", "has-tooltip"]); - tippy(opinion, {content: renderTooltip(), placement: "right"}); - frag.append(opinion, ` overall view of ${slave.slaveName} `, sumFrag, `; ${he} had no net impact on your reputation or social progress this week.`); - } else { - const opinion = App.UI.DOM.makeElement("span", "negative", ["red", "major-link", "has-tooltip"]); - tippy(opinion, {content: renderTooltip(), placement: "right"}); - frag.append(opinion, ` overall view of ${slave.slaveName} `, sumFrag, `, which decreases your reputation and retards social progress.`); + const sum = positiveSum + negativeSum; + frag.append(`Society has a `); + if (sum > 0) { + const opinion = App.UI.DOM.makeElement("span", "positive", ["green", "major-link", "has-tooltip"]); + tippy(opinion, {content: renderTooltip(), placement: "right"}); + frag.append(opinion, ` overall view of ${slave.slaveName} `, sumFrag, `, which improves your reputation and advances social progress.`); + } else if (sum === 0) { + const opinion = App.UI.DOM.makeElement("span", "neutral", ["yellow", "major-link", "has-tooltip"]); + tippy(opinion, {content: renderTooltip(), placement: "right"}); + frag.append(opinion, ` overall view of ${slave.slaveName} `, sumFrag, `; ${he} had no net impact on your reputation or social progress this week.`); + } else { + const opinion = App.UI.DOM.makeElement("span", "negative", ["red", "major-link", "has-tooltip"]); + tippy(opinion, {content: renderTooltip(), placement: "right"}); + frag.append(opinion, ` overall view of ${slave.slaveName} `, sumFrag, `, which decreases your reputation and retards social progress.`); + } } - } - function displayLong() { - $(frag).append(socialEffects.map(e => e.longDesc).join(" ")); - } + function displayLong() { + $(frag).append(socialEffects.map(e => e.longDesc).join(" ")); + } - const frag = document.createDocumentFragment(); + const frag = document.createDocumentFragment(); - let socialEffects = []; - if (V.FSAnnounced > 0) { - if (V.studio === 1) { - if (slave.porn.viewerCount > 0) { - slave.pornFameBonus += (Math.ceil(slave.porn.viewerCount / 100000)); - if (slave.porn.viewerCount >= 100000) { - frag.append(`${His} near-ubiquitous presence in arcology pornography greatly increases ${his} impact on society. `); - } else if (slave.porn.viewerCount >= 10000) { - frag.append(`${His} presence in arcology pornography increases ${his} impact on society. `); - } else { - frag.append(`${His} occasional presence in arcology pornography slightly increases ${his} impact on society. `); + let socialEffects = []; + if (V.FSAnnounced > 0) { + if (V.studio === 1) { + if (slave.porn.viewerCount > 0) { + slave.pornFameBonus += (Math.ceil(slave.porn.viewerCount / 100000)); + if (slave.porn.viewerCount >= 100000) { + frag.append(`${His} near-ubiquitous presence in arcology pornography greatly increases ${his} impact on society. `); + } else if (slave.porn.viewerCount >= 10000) { + frag.append(`${His} presence in arcology pornography increases ${his} impact on society. `); + } else { + frag.append(`${His} occasional presence in arcology pornography slightly increases ${his} impact on society. `); + } } } } - socialEffects.push(...makeSocialEffects()); + if (socialEffects.length > 0) { + applySocialEffects(); + if (!V.UI.compressSocialEffects) { + displayLong(); + } else { + displayCompressed(); + } + } + + return frag; } - socialEffects.push(...makeShelterGirlEffects()); - if (socialEffects.length > 0) { - applySocialEffects(); - if (!V.UI.compressSocialEffects) { - displayLong(); - } else { - displayCompressed(); + + /** Test what new social effects this slave would gain if a new FS were added + * Note that this function does not currently take into account the social effects + * that would be *lost* if a new FS was enacted; this is a rare but extant case. + * @param {FC.FutureSociety} proposedFS + */ + function newForFS(proposedFS) { + // make sure the FS is not currently enacted + if (V.arcologies[0][proposedFS] !== "unset") { + throw new Error(`Cannot test new FS social effects for existing FS ${proposedFS}`); } + // temporarily enact the FS at full strength + V.arcologies[0][proposedFS] = 100; + // see what the slave's social effects would be under the proposed FS + const newSocialEffects = [...makeSocialEffects(), ...makeShelterGirlEffects()]; + // undo the temporary FS enactment + V.arcologies[0][proposedFS] = "unset"; + // return the differences between the current and new states + return _.differenceBy(newSocialEffects, socialEffects, s => s.shortDesc); } - return frag; + const socialEffects = [...makeSocialEffects(), ...makeShelterGirlEffects()]; + + return { + report, + newForFS + } }; diff --git a/src/futureSocieties/fsPassage.js b/src/futureSocieties/fsPassage.js index a9c23ed21eb8fb66c293ea386883dfdb6f9d3070..9187330b5f2ce9d6d0aacfadd25df8d6e297b24b 100644 --- a/src/futureSocieties/fsPassage.js +++ b/src/futureSocieties/fsPassage.js @@ -342,6 +342,72 @@ App.UI.fsPassage = function() { } } + /** Test what new social effects you'd get if a new FS were added + * This is a good proxy for evaluating whether you'll be able to make it stick or not + * @param {FC.FutureSociety} proposedFS + */ + function evaluation(proposedFS) { + const effectCounts = new Map(); + for (const slave of App.SlaveAssignment.reportSlaves(V.slaves)) { + const slaveEffects = App.SlaveAssignment.saSocialEffects(slave).newForFS(proposedFS); + for (const effect of slaveEffects) { + const curVal = effectCounts.get(effect.shortDesc); + if (curVal) { + effectCounts.set(effect.shortDesc, curVal + effect.magnitude); + } else { + effectCounts.set(effect.shortDesc, effect.magnitude); + } + } + } + + const grid = document.createElement("div"); + grid.classList.add("grid-2columns-auto"); + let avg = 0; + for (const [key, count] of effectCounts) { + const className = count < 0 ? "red" : "green"; + App.UI.DOM.appendNewElement("div", grid, count.toString(), [className]); + App.UI.DOM.appendNewElement("div", grid, key); + avg += count; + } + avg /= V.slaves.length; + + const tip = document.createElement('div'); + tip.classList.add("tip-details"); + if (avg > 1.5) { + tip.appendChild(document.createTextNode(`Adopting ${FutureSocieties.displayName(proposedFS)} is likely to be straightforward, even against social pressure from your neighbors. Your arcology is already primed to move this direction thanks to your actions, and your slaves are well-aligned with it.`)); + } else if (avg > 0.5) { + tip.appendChild(document.createTextNode(`Adopting ${FutureSocieties.displayName(proposedFS)} should be relatively painless. Your arcology is already receptive to moving this direction thanks to your actions, and your slaves are aligned with it.`)); + } else if (avg > -0.1) { + tip.appendChild(document.createTextNode(`Adopting ${FutureSocieties.displayName(proposedFS)} will require investment and effort. Your arcology is not opposed to moving this direction, but it's also not expecting it based on your past actions. To improve your chances, consider first aligning your slaves with ${FutureSocieties.displayAdj(proposedFS)} goals.`)); + } else { + tip.appendChild(document.createTextNode(`Attempting to adopt ${FutureSocieties.displayName(proposedFS)} with your arcology in its current state will likely result in failure. You should strongly consider aligning your slaves with ${FutureSocieties.displayAdj(proposedFS)} goals before endorsing it.`)); + } + tip.append(grid); + + const span = document.createElement("span"); + span.classList.add("fs-recommend"); + if (avg > 1.5) { + span.textContent = "Primed"; + span.classList.add("fs-recommend-great"); + } else if (avg > 0.5) { + span.textContent = "Receptive"; + span.classList.add("fs-recommend-good"); + } else if (avg > -0.1) { + span.textContent = "Neutral"; + span.classList.add("fs-recommend-neutral"); + } else { + span.textContent = "Resistant"; + span.classList.add("fs-recommend-bad"); + } + span.tabIndex = 0; + span.classList.add("has-tooltip"); + tippy(span, { + content: tip, + placement: "right", + }); + return span; + } + function selectFS() { const el = new DocumentFragment(); let r; @@ -380,6 +446,7 @@ App.UI.fsPassage = function() { arc.FSSupremacistRace = /** @type {FC.Race} */ (race); App.UI.reload(); })); + r.push(evaluation("FSSupremacist")); } else { /* <span class="note"><span style="font-weight:Bold">Racial Supremacism:</span> a belief in the superiority of a chosen race.</span>*/ } @@ -408,15 +475,15 @@ App.UI.fsPassage = function() { r.push(`${arc.FSSubjugationistRace} inferiority.`); } r.push(`Select race:`); - const options = []; for (const race of App.Utils.getRaceArrayWithoutParamRace(arc.FSSupremacistRace)) { // Superior race cannot be subj, so remove options.push({key: race, name: capFirstChar(race)}); } r.push(App.UI.DOM.makeSelect(options, arc.FSSubjugationistRace, race => { - arc.FSSubjugationistRace = race; + arc.FSSubjugationistRace = /** @type {FC.Race} */ (race); App.UI.reload(); })); + r.push(evaluation("FSSubjugationist")); } else { /* <span class="note"><span style="font-weight:Bold">Racial Subjugationism:</span> a belief in the inferiority of a subject race.</span>*/ } @@ -447,6 +514,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a focus on mass breeding in order to repopulate the future world.`); + r.push(evaluation("FSRepopulationFocus")); } else { /* <span class="note"><span style="font-weight:Bold">Repopulation Efforts:</span> societal fetishization of pregnancy.</span>*/ } @@ -480,6 +548,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is rebuilding society using restrictive breeding programs reserved solely for society's finest.`); + r.push(evaluation("FSRestart")); } else { /* <span class="note"><span style="font-weight:Bold">Complete Societal Reconstruction:</span> rebuilding society based off the elite.</span>*/ } @@ -509,6 +578,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a belief that slaves should be airheaded, horny and fully dependent on their owners.`); + r.push(evaluation("FSIntellectualDependency")); } else { /* <span class="note"><span style="font-weight:Bold">Intellectual Dependency:</span> a belief that slaves should be airheaded, horny and fully dependent on their owners.</span>*/ } @@ -533,6 +603,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is increased interest in smart, refined, altogether perfect slaves.`); + r.push(evaluation("FSSlaveProfessionalism")); } else { /* <span class="note"><span style="font-weight:Bold">Slave Professionalism:</span> increased interest in smart, refined, altogether perfect slaves.</span>*/ } @@ -562,6 +633,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a radical redefinition of gender that identifies powerful people as male, and everyone else as female.`); + r.push(evaluation("FSGenderRadicalist")); } else { /* <span class="note"><span style="font-weight:Bold">Gender radicalism:</span> a radical redefinition of gender that identifies powerful people as male, and everyone else as female.</span>*/ } @@ -587,6 +659,7 @@ App.UI.fsPassage = function() { ) ); r.push(`gender traditionalism, including a societal preference for feminine slaves ${(V.seePreg !== 0) ? ` and support for slave pregnancy` : ``}.`); + r.push(evaluation("FSGenderFundamentalist")); } else { /* <span class="note"><span style="font-weight:Bold">Gender traditionalism:</span> a societal preference for feminine slaves if (V.seePreg !== 0) { @@ -619,6 +692,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a vision of slave improvement, including slaves' health, mental well-being, and education.`); + r.push(evaluation("FSPaternalist")); } else { /* <span class="note"><span style="font-weight:Bold">Paternalism:</span> a vision of slave improvement, including slaves' health, mental well-being, and education.</span>*/ } @@ -643,6 +717,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a belief that slaves are not human and should not be treated decently.`); + r.push(evaluation("FSDegradationist")); } else { /* <span class="note"><span style="font-weight:Bold">Degradation:</span> a belief that slaves are not human and should not be treated decently.</span>*/ } @@ -671,6 +746,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is societal disapproval of implant surgery.`); + r.push(evaluation("FSBodyPurist")); } else { /* <span class="note"><span style="font-weight:Bold">Body Purism:</span> societal disapproval of implant surgery.</span>*/ } @@ -695,6 +771,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is societal fetishization of implant surgery.`); + r.push(evaluation("FSTransformationFetishist")); } else { /* <span class="note"><span style="font-weight:Bold">Transformation Fetishism:</span> societal fetishization of implant surgery.</span>*/ } @@ -732,6 +809,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is increased interest in girls just past their majority.`); + r.push(evaluation("FSYouthPreferentialist")); } else { /* <span class="note"><span style="font-weight:Bold">Youth Preferentialism:</span> increased interest in girls just past their majority.</span>*/ } @@ -763,6 +841,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is increased interest in mature slaves.`); + r.push(evaluation("FSMaturityPreferentialist")); } else { /* <span class="note"><span style="font-weight:Bold">Maturity Preferentialism:</span> increased interest in mature slaves.</span>*/ } @@ -791,6 +870,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is increased interest in short slaves.`); + r.push(evaluation("FSPetiteAdmiration")); } else { /* <span class="note"><span style="font-weight:Bold">Petite Admiration:</span> increased interest in short slaves.</span>*/ } @@ -815,6 +895,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is societal fixation on tallness.`); + r.push(evaluation("FSStatuesqueGlorification")); } else { /* <span class="note"><span style="font-weight:Bold">Statuesque Glorification:</span> societal fixation on tallness.</span>*/ } @@ -843,6 +924,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a fashion for slaves with girlish figures.`); + r.push(evaluation("FSSlimnessEnthusiast")); } else { /* <span class="note"><span style="font-weight:Bold">Slimness Enthusiasm:</span> a fashion for slaves with girlish figures.</span>*/ } @@ -867,6 +949,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is societal hunger for huge assets of whatever origin.`); + r.push(evaluation("FSAssetExpansionist")); } else { /* <span class="note"><span style="font-weight:Bold">Asset Expansionism:</span> societal hunger for huge assets of whatever origin.</span>*/ } @@ -893,6 +976,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is societal acceptance of slave products like milk.`); + r.push(evaluation("FSPastoralist")); } else { /* <span class="note"><span style="font-weight:Bold">Slave Pastoralism:</span> societal acceptance of slave products like milk.</span>*/ } @@ -919,6 +1003,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is societal reverence for the idealized human form, including height, health and muscle.`); + r.push(evaluation("FSPhysicalIdealist")); } else { /* <span class="note"><span style="font-weight:Bold">Physical Idealism:</span> societal reverence for the idealized human form, including height, health and muscle.</span>*/ } @@ -943,6 +1028,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is societal acceptance of overindulgence and immediate gratification. Be it food, drink, sex, drugs or whatever one's desire may be.`); + r.push(evaluation("FSHedonisticDecadence")); } else { /* <span class="note"><span style="font-weight:Bold">HedonisticDecadence:</span> societal acceptance of over indulgence, particularly of food, drink, sex and drugs.</span>*/ } @@ -971,6 +1057,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a new strain of religion that emphasizes the slaveholding portions of religious history.`); + r.push(evaluation("FSChattelReligionist")); } else { /* <span class="note"><span style="font-weight:Bold">Chattel Religionism:</span> a new strain of religion that emphasizes the slaveholding portions of religious history.</span>*/ } @@ -1142,6 +1229,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a vision of a new Rome.`); + r.push(evaluation("FSRomanRevivalist")); } else { /* <span class="note"><span style="font-weight:Bold">Roman Revivalism:</span> a vision of a new Rome.</span>*/ } @@ -1166,6 +1254,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a vision of a new Imperial society, integrating high technology and old-world culture under the iron fist of your absolute rule.`); + r.push(evaluation("FSNeoImperialist")); } else { /* <span class="note"><span style="font-weight:Bold">Neo-Imperialism:</span> a vision of a new Imperial Society, integrating high technology and old-world culture under the iron fist of your absolute rule.</span>*/ } @@ -1190,6 +1279,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a vision of a new Aztec Empire.`); + r.push(evaluation("FSAztecRevivalist")); } else { /* <span class="note"><span style="font-weight:Bold">Aztec Revivalism:</span> a vision of a new Aztec Empire.</span>*/ } @@ -1215,6 +1305,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a vision of a Pharaoh's Egypt.`); + r.push(evaluation("FSEgyptianRevivalist")); } else { /* <span class="note"><span style="font-weight:Bold">Egyptian Revivalism:</span> a vision of Pharaoh's Egypt.</span>*/ } @@ -1239,6 +1330,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a vision of Edo Japan.`); + r.push(evaluation("FSEdoRevivalist")); } else { /* <span class="note"><span style="font-weight:Bold">Edo Revivalism:</span> a vision of Edo Japan.</span>*/ } @@ -1263,6 +1355,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a vision of the Sultanate of old.`); + r.push(evaluation("FSArabianRevivalist")); } else { /* <span class="note"><span style="font-weight:Bold">Arabian Revivalism:</span> a vision of the Sultanate of old.</span>*/ } @@ -1287,6 +1380,7 @@ App.UI.fsPassage = function() { ) ); r.push(`is a vision of ancient China.`); + r.push(evaluation("FSChineseRevivalist")); } else { /* <span class="note"><span style="font-weight:Bold">Chinese Revivalism:</span> a vision of ancient China.</span>*/ }