diff --git a/css/rulesAssistant/activationConditions.css b/css/rulesAssistant/activationConditions.css
index 2acfd18cb8dbd87a79e6d275fb3c96e26fd45511..30433fac3bd515130016db82b527fe0ea4c8cc4f 100644
--- a/css/rulesAssistant/activationConditions.css
+++ b/css/rulesAssistant/activationConditions.css
@@ -37,3 +37,28 @@
     min-width: 10em;
     min-height: 2em;
 }
+
+/* Encyclopedia help entry */
+
+.rule-help-getter-table {
+    display: grid;
+    grid-template-columns: max-content auto auto;
+    /*grid-column-gap: 1em;*/
+}
+
+.rule-help-getter-table .head {
+    font-weight: bold;
+    position: sticky;
+    top: -1em;
+    background-color: #111;
+}
+
+.rule-help-getter-table > div {
+    padding: 0 0.5em
+}
+
+.rule-help-getter-table > :nth-child(6n+4),
+.rule-help-getter-table > :nth-child(6n+5),
+.rule-help-getter-table > :nth-child(6n+6) {
+    background-color: #222;
+}
diff --git a/devTools/types/FC/RA.d.ts b/devTools/types/FC/RA.d.ts
index 775bfe4b32a5653f001aa8d265b38fc69d6bcd0e..d9c75fff29ffea78e0b574b757ee501e71f33834 100644
--- a/devTools/types/FC/RA.d.ts
+++ b/devTools/types/FC/RA.d.ts
@@ -10,7 +10,6 @@ declare namespace FC {
 
 		interface RuleConditions {
 			activation: PostFixRule;
-			assignment: Assignment[];
 			selectedSlaves: number[];
 			excludedSlaves: number[];
 			applyRuleOnce: boolean;
diff --git a/src/data/backwardsCompatibility/datatypeCleanup.js b/src/data/backwardsCompatibility/datatypeCleanup.js
index 433523bd7030119ecb475ee930452fe4eadc945e..4024db4a2190f215a8d620065b342ae66a979bf8 100644
--- a/src/data/backwardsCompatibility/datatypeCleanup.js
+++ b/src/data/backwardsCompatibility/datatypeCleanup.js
@@ -2229,6 +2229,9 @@ App.Entity.Utils.RARuleDatatypeCleanup = function() {
 						case "chem":
 							count += addBetween("vchem");
 							break;
+						case "lactation":
+							count += addBetween("vlactation");
+							break;
 					}
 					if (count === 0) {
 						count++;
@@ -2242,12 +2245,154 @@ App.Entity.Utils.RARuleDatatypeCleanup = function() {
 					cond.activation = [false, 1, "and"];
 				}
 			} catch (e) {
-				console.log("everything broke", e.message, JSON.parse(JSON.stringify(cond)));
+				console.log("condition broke", e.message, JSON.parse(JSON.stringify(cond)));
 				cond.activation = [false, 1, "and"];
 			} finally {
 				delete cond.function;
 				delete cond.data;
 			}
+
+			// assignments
+			try {
+				if (cond.assignment.length > 0) {
+					const rule = [];
+					for (const assignment of cond.assignment) {
+						switch (assignment) {
+							case Job.REST:
+								rule.push("vrest");
+								break;
+							case Job.CHOICE:
+								rule.push("vchoice");
+								break;
+							case Job.FUCKTOY:
+								rule.push("vfucktoy");
+								break;
+							case Job.CLASSES:
+								rule.push("vclasses");
+								break;
+							case Job.HOUSE:
+								rule.push("vhouse");
+								break;
+							case Job.WHORE:
+								rule.push("vwhore");
+								break;
+							case Job.PUBLIC:
+								rule.push("vpublic");
+								break;
+							case Job.SUBORDINATE:
+								rule.push("vsubordinate");
+								break;
+							case Job.MILKED:
+								rule.push("vmilked");
+								break;
+							case Job.GLORYHOLE:
+								rule.push("vgloryhole");
+								break;
+							case Job.CONFINEMENT:
+								rule.push("vconfinement");
+								break;
+							case Job.BODYGUARD:
+								rule.push("vbodyguard");
+								break;
+							case Job.RECRUITER:
+								rule.push("vrecruiter");
+								break;
+							case Job.HEADGIRL:
+								rule.push("vheadgirl");
+								break;
+							case Job.ARCADE:
+								rule.push("varcade");
+								break;
+							case Job.MADAM:
+								rule.push("vmadam");
+								break;
+							case Job.BROTHEL:
+								rule.push("vbrothel");
+								break;
+							case Job.WARDEN:
+								rule.push("vwarden");
+								break;
+							case Job.CELLBLOCK:
+								rule.push("vcellblock");
+								break;
+							case Job.DJ:
+								rule.push("vdj");
+								break;
+							case Job.CLUB:
+								rule.push("vclub");
+								break;
+							case Job.NURSE:
+								rule.push("vnurse");
+								break;
+							case Job.CLINIC:
+								rule.push("vclinic");
+								break;
+							case Job.MILKMAID:
+								rule.push("vmilkmaid");
+								break;
+							case Job.DAIRY:
+								rule.push("vdairy");
+								break;
+							case Job.FARMER:
+								rule.push("vfarmer");
+								break;
+							case Job.FARMYARD:
+								rule.push("vfarmyard");
+								break;
+							case Job.HEADGIRLSUITE:
+								rule.push("vheadgirlsuite");
+								break;
+							case Job.CONCUBINE:
+								rule.push("vconcubine");
+								break;
+							case Job.MASTERSUITE:
+								rule.push("vmastersuite");
+								break;
+							case Job.MATRON:
+								rule.push("vmatron");
+								break;
+							case Job.NURSERY:
+								rule.push("vnursery");
+								break;
+							case Job.TEACHER:
+								rule.push("vteacher");
+								break;
+							case Job.SCHOOL:
+								rule.push("vschool");
+								break;
+							case Job.STEWARD:
+								rule.push("vsteward");
+								break;
+							case Job.QUARTER:
+								rule.push("vquarter");
+								break;
+							case Job.ATTENDANT:
+								rule.push("vattendant");
+								break;
+							case Job.SPA:
+								rule.push("vspa");
+								break;
+						}
+					}
+					rule.push(rule.length, "or");
+
+					if (!App.RA.Activation.Editor.validateRule(rule)) {
+						rule = [false, 1, "or"];
+					}
+
+					cond.activation.pop();
+					const length = cond.activation.pop();
+					cond.activation.push(...rule);
+					cond.activation.push(length + 1, "and");
+					if (!App.RA.Activation.Editor.validateRule(cond.activation)) {
+						cond.activation = [false, 1, "and"];
+					}
+				}
+			} catch (e) {
+				console.log("assignments broke", e.message, JSON.parse(JSON.stringify(cond)));
+			} finally {
+				delete cond.assignment;
+			}
 		}
 
 		function addBetween(key) {
diff --git a/src/gui/Encyclopedia/encyclopediaRAActivationEditor.js b/src/gui/Encyclopedia/encyclopediaRAActivationEditor.js
new file mode 100644
index 0000000000000000000000000000000000000000..3154391870ae7ee099e613dc8430ca8436a94a0b
--- /dev/null
+++ b/src/gui/Encyclopedia/encyclopediaRAActivationEditor.js
@@ -0,0 +1,61 @@
+App.Encyclopedia.addArticle("RA Condition Editor", function() {
+	/**
+	 * @param {ReadonlyMap<string, Getter<*>>} getters
+	 */
+	function getterTable(getters) {
+		const container = document.createElement("p");
+		container.classList.add("rule-help-getter-table");
+
+		App.UI.DOM.appendNewElement("div", container, "Name", "head");
+		App.UI.DOM.appendNewElement("div", container, "Description", "head");
+		App.UI.DOM.appendNewElement("div", container, "Requirements", "head");
+
+		for (const getter of getters.values()) {
+			if (getter.visible && !getter.visible()) {
+				continue;
+			}
+			App.UI.DOM.appendNewElement("div", container, getter.name);
+			App.UI.DOM.appendNewElement("div", container, getter.description);
+			const div = document.createElement("div");
+			if (getter.requirements) {
+				div.append(getter.requirements);
+			}
+			container.append(div);
+		}
+
+		return container;
+	}
+
+	const acc = new SpacedTextAccumulator();
+
+	acc.push("The rule conditions handle 3 different data types: Boolean, Numeric and String. Not all aggregators accept all data types.",
+		"Numeric and Boolean can be used interchangeably, the conversion is as follows:",
+		"Putting a Numeric value into a boolean aggregator will interpret 0 as false and all other values as true.",
+		"Putting a Boolean value into a numeric aggregator will interpret false as 0 and true as 1.");
+	acc.toParagraph();
+
+	App.UI.DOM.appendNewElement("h2", acc.container(), "Predefined data getters");
+
+	acc.push("There are a number of predefined getters which read values either from a slave or from the global state. They always have a predefined data type.");
+	acc.toParagraph();
+
+	let c = acc.container();
+	App.UI.DOM.appendNewElement("h3", c, "Boolean getters");
+	c.append(getterTable(App.RA.Activation.getterManager.booleanGetters));
+	App.UI.DOM.appendNewElement("h3", c, "Assignment getters");
+	acc.push("A special type of boolean getters checking if the slave has the given assignment");
+	acc.toParagraph();
+	c = acc.container();
+	c.append(getterTable(App.RA.Activation.getterManager.assignmentGetters));
+	App.UI.DOM.appendNewElement("h3", c, "Number getters");
+	c.append(getterTable(App.RA.Activation.getterManager.numberGetters));
+	App.UI.DOM.appendNewElement("h3", c, "String getters");
+	c.append(getterTable(App.RA.Activation.getterManager.stringGetters));
+	App.UI.DOM.appendNewElement("h3", c, "Custom getters");
+	acc.push("If greater freedom is required for the conditions needed, a custom data getter can be used.",
+		"It operates on a context object with the following properties: slave: The slave currently tested against.",
+		"It is required to explicitly set the return type. If the actual return type does not match the set type, the condition evaluation will fail!");
+	acc.toParagraph();
+
+	return acc.container();
+}, "obtainingSlaves");
diff --git a/src/js/rulesAssistant.js b/src/js/rulesAssistant.js
index 63bcdcb02133888f37d803cbd59c3819502847e1..3a3709ff5f93f2066a7df7f8a9341592641f28fb 100644
--- a/src/js/rulesAssistant.js
+++ b/src/js/rulesAssistant.js
@@ -123,10 +123,8 @@ globalThis.ruleAppliesP = function(rule, slave) {
 		}
 	}
 
-	// assignment / facility / special slaves / specific slaves check
-	if (cond.assignment.length > 0 && !cond.assignment.includes(slave.assignment)) {
-		return false;
-	} else if (cond.selectedSlaves.length > 0 && !cond.selectedSlaves.includes(slave.ID)) {
+	//  special slaves / specific slaves check
+	if (cond.selectedSlaves.length > 0 && !cond.selectedSlaves.includes(slave.ID)) {
 		return false;
 	} else if (cond.excludedSlaves.includes(slave.ID)) {
 		return false;
@@ -173,7 +171,6 @@ App.RA.newRule = function() {
 	function emptyConditions() {
 		return {
 			activation: [true, 1, "and"],
-			assignment: [],
 			selectedSlaves: [],
 			excludedSlaves: [],
 			applyRuleOnce: false,
diff --git a/src/js/rulesAssistantActivationCondition.js b/src/js/rulesAssistantActivationCondition.js
index 50a46072b8fcb0d3d7f10cf2d7748ed54fdecb46..1ca6a25dd150fde5b5eb51700b1e2a897aeab85c 100644
--- a/src/js/rulesAssistantActivationCondition.js
+++ b/src/js/rulesAssistantActivationCondition.js
@@ -1,9 +1,6 @@
 /**
  * TODO add encyclopedia entry
  *
- * Numeric and Boolean checks can be used interchangeably.
- * Putting a Numeric value into a boolean aggregator will interpret 0 as false and all other values as true.
- * Putting a Boolean value into a numeric aggregator will interpret false as 0 and true as 1.
  */
 
 App.RA.Activation.Editor = (function() {
@@ -76,6 +73,7 @@ App.RA.Activation.Editor = (function() {
 		} else {
 			ruleDiv.append("Rule saved");
 		}
+		ruleDiv.append(" ", App.Encyclopedia.Dialog.linkDOM("Help", "RA Condition Editor"));
 		ruleDiv.append(currentRule.render());
 		outerDiv.append(ruleDiv);
 
@@ -95,6 +93,7 @@ App.RA.Activation.Editor = (function() {
 		div.append(new RulePartProvider(() => new RulePair("eq")).render());
 		div.append(new RulePartProvider(() => new RuleNegate()).render());
 		div.append(new RulePartProvider(() => new RuleMapCheck(App.RA.Activation.getterManager.booleanDefault)).render());
+		div.append(new RulePartProvider(() => new RuleMapCheck(App.RA.Activation.getterManager.assignmentDefault)).render());
 		div.append(new RulePartProvider(() => new RuleMapCheck(App.RA.Activation.getterManager.numberDefault)).render());
 		div.append(new RulePartProvider(() => new RuleMapCheck(App.RA.Activation.getterManager.stringDefault)).render());
 		div.append(new RulePartProvider(() => new RuleConstant(0)).render());
@@ -770,11 +769,12 @@ App.RA.Activation.Editor = (function() {
 		constructor(key) {
 			super();
 			/**
-			 * @type {"boolean"|"number"|"string"}
+			 * @type {"boolean"|"assignment"|"number"|"string"}
 			 */
 			this.mode = App.RA.Activation.getterManager.isBoolean(key) ? "boolean"
-				: App.RA.Activation.getterManager.isNumber(key) ? "number"
-					: "string";
+				: App.RA.Activation.getterManager.isAssignment(key) ? "assignment"
+					: App.RA.Activation.getterManager.isNumber(key) ? "number"
+						: "string";
 			this.key = key;
 		}
 
@@ -784,14 +784,24 @@ App.RA.Activation.Editor = (function() {
 			span.classList.add("rule-condition");
 			makeDraggable(span, this);
 			// fill container
-			span.append("Slave");
+			if (this.mode === "assignment") {
+				span.append("Assignment");
+			} else {
+				span.append("Slave");
+			}
 			let matchFound = false;
 			let select = document.createElement("select");
 
 			for (const [key, value] of this._getterMap) {
+				if (value.visible && !value.visible()) {
+					continue;
+				}
 				let el = document.createElement("option");
 				el.value = key;
 				el.textContent = value.name;
+				if (value.enabled) {
+					el.disabled = value.enabled();
+				}
 				if (this.key === key) {
 					el.selected = true;
 					matchFound = true;
@@ -813,7 +823,7 @@ App.RA.Activation.Editor = (function() {
 		}
 
 		validate() {
-			if (this.mode === "boolean") {
+			if (this.mode === "boolean" || this.mode === "assignment") {
 				return "number";
 			}
 			return this.mode;
@@ -824,9 +834,10 @@ App.RA.Activation.Editor = (function() {
 		 * @private
 		 */
 		get _getterMap() {
-			return this.mode === "boolean" ? App.RA.Activation.getterManager.booleanGetter
-				: this.mode === "number" ? App.RA.Activation.getterManager.numberGetter
-					: App.RA.Activation.getterManager.stringGetter;
+			return this.mode === "boolean" ? App.RA.Activation.getterManager.booleanGetters
+				: this.mode === "assignment" ? App.RA.Activation.getterManager.assignmentGetters
+					: this.mode === "number" ? App.RA.Activation.getterManager.numberGetters
+						: App.RA.Activation.getterManager.stringGetters;
 		}
 	}
 
diff --git a/src/js/rulesAssistantActivationEvaluation.js b/src/js/rulesAssistantActivationEvaluation.js
index 1264a682765c5c37f5ccfc4f1ce47fcc02b1d767..1a05cdecd1f73880054128542b43eabeb5f093fa 100644
--- a/src/js/rulesAssistantActivationEvaluation.js
+++ b/src/js/rulesAssistantActivationEvaluation.js
@@ -14,10 +14,13 @@ App.RA.Activation.Context = class {
 };
 
 /**
- * TODO: add availability checks
  * @template {boolean|number|string} T
  * @typedef {object} Getter
  * @property {string} name
+ * @property {string} description Should include possible values if applicable.
+ * @property {string} [requirements] Plaintext description of requirements to use this getter.
+ * @property {()=>boolean} [enabled] Whether the getter can be used.
+ * @property {()=>boolean} [visible] Whether the getter should be shown. Mainly intended for disabled mods.
  * @property {(s: App.RA.Activation.Context) =>T} val
  */
 
@@ -28,17 +31,22 @@ App.RA.Activation.getterManager = (function() {
 			 * @private
 			 * @type {Map<string, Getter<boolean>>}
 			 */
-			this._booleanGetter = new Map();
+			this._booleanGetters = new Map();
+			/**
+			 * @private
+			 * @type {Map<string, Getter<boolean>>}
+			 */
+			this._assignmentGetters = new Map();
 			/**
 			 * @private
 			 * @type {Map<string, Getter<number>>}
 			 */
-			this._numberGetter = new Map();
+			this._numberGetters = new Map();
 			/**
 			 * @private
 			 * @type {Map<string, Getter<string>>}
 			 */
-			this._stringGetter = new Map();
+			this._stringGetters = new Map();
 
 			/**
 			 * Adding vanilla or custom getters?
@@ -61,7 +69,15 @@ App.RA.Activation.getterManager = (function() {
 		 * @param {Getter<boolean>} getter
 		 */
 		addBoolean(key, getter) {
-			this._booleanGetter.set(this._makePrefix() + key, getter);
+			this._booleanGetters.set(this._makePrefix() + key, getter);
+		}
+
+		/**
+		 * @param {string} key
+		 * @param {Getter<boolean>} getter
+		 */
+		addAssignment(key, getter) {
+			this._assignmentGetters.set(this._makePrefix() + key, getter);
 		}
 
 		/**
@@ -69,7 +85,7 @@ App.RA.Activation.getterManager = (function() {
 		 * @param {Getter<number>} getter
 		 */
 		addNumber(key, getter) {
-			this._numberGetter.set(this._makePrefix() + key, getter);
+			this._numberGetters.set(this._makePrefix() + key, getter);
 		}
 
 		/**
@@ -77,7 +93,7 @@ App.RA.Activation.getterManager = (function() {
 		 * @param {Getter<string>} getter
 		 */
 		addString(key, getter) {
-			this._stringGetter.set(this._makePrefix() + key, getter);
+			this._stringGetters.set(this._makePrefix() + key, getter);
 		}
 
 		/**
@@ -85,7 +101,8 @@ App.RA.Activation.getterManager = (function() {
 		 * @returns {boolean}
 		 */
 		has(key) {
-			return this._booleanGetter.has(key) || this._numberGetter.has(key) || this._stringGetter.has(key);
+			return this._booleanGetters.has(key) || this._assignmentGetters.has(key) ||
+				this._numberGetters.has(key) || this._stringGetters.has(key);
 		}
 
 		/**
@@ -93,7 +110,15 @@ App.RA.Activation.getterManager = (function() {
 		 * @returns {boolean}
 		 */
 		isBoolean(key) {
-			return this._booleanGetter.has(key);
+			return this._booleanGetters.has(key);
+		}
+
+		/**
+		 * @param {string} key
+		 * @returns {boolean}
+		 */
+		isAssignment(key) {
+			return this._assignmentGetters.has(key);
 		}
 
 		/**
@@ -101,7 +126,7 @@ App.RA.Activation.getterManager = (function() {
 		 * @returns {boolean}
 		 */
 		isNumber(key) {
-			return this._numberGetter.has(key);
+			return this._numberGetters.has(key);
 		}
 
 		/**
@@ -109,49 +134,63 @@ App.RA.Activation.getterManager = (function() {
 		 * @returns {boolean}
 		 */
 		isString(key) {
-			return this._stringGetter.has(key);
+			return this._stringGetters.has(key);
+		}
+
+		/**
+		 * @returns {ReadonlyMap<string, Getter<boolean>>}
+		 */
+		get booleanGetters() {
+			return this._booleanGetters;
 		}
 
 		/**
 		 * @returns {ReadonlyMap<string, Getter<boolean>>}
 		 */
-		get booleanGetter() {
-			return this._booleanGetter;
+		get assignmentGetters() {
+			return this._assignmentGetters;
 		}
 
 		/**
 		 * @returns {ReadonlyMap<string, Getter<number>>}
 		 */
-		get numberGetter() {
-			return this._numberGetter;
+		get numberGetters() {
+			return this._numberGetters;
 		}
 
 		/**
 		 * @returns {ReadonlyMap<string, Getter<string>>}
 		 */
-		get stringGetter() {
-			return this._stringGetter;
+		get stringGetters() {
+			return this._stringGetters;
 		}
 
 		/**
 		 * @returns {string}
 		 */
 		get booleanDefault() {
-			return this._booleanGetter.keys().next().value;
+			return this._booleanGetters.keys().next().value;
+		}
+
+		/**
+		 * @returns {string}
+		 */
+		get assignmentDefault() {
+			return this._assignmentGetters.keys().next().value;
 		}
 
 		/**
 		 * @returns {string}
 		 */
 		get numberDefault() {
-			return this._numberGetter.keys().next().value;
+			return this._numberGetters.keys().next().value;
 		}
 
 		/**
 		 * @returns {string}
 		 */
 		get stringDefault() {
-			return this._stringGetter.keys().next().value;
+			return this._stringGetters.keys().next().value;
 		}
 
 		/**
@@ -161,17 +200,22 @@ App.RA.Activation.getterManager = (function() {
 		 * @returns {boolean} True, if a getter exists for the given key
 		 */
 		read(stack, context, key) {
-			const getterB = this._booleanGetter.get(key);
+			let getterB = this._booleanGetters.get(key);
 			if (getterB !== undefined) {
 				stack.pushNumber(getterB.val(context) ? 1 : 0);
 				return true;
 			}
-			const getterN = this._numberGetter.get(key);
+			getterB = this._assignmentGetters.get(key);
+			if (getterB !== undefined) {
+				stack.pushNumber(getterB.val(context) ? 1 : 0);
+				return true;
+			}
+			const getterN = this._numberGetters.get(key);
 			if (getterN !== undefined) {
 				stack.pushNumber(getterN.val(context));
 				return true;
 			}
-			const getterS = this._stringGetter.get(key);
+			const getterS = this._stringGetters.get(key);
 			if (getterS !== undefined) {
 				stack.pushString(getterS.val(context));
 				return true;
@@ -186,32 +230,306 @@ App.RA.Activation.getterManager = (function() {
 App.RA.Activation.populateGetters = function() {
 	const gm = App.RA.Activation.getterManager;
 	// Note: The first value of each type being added is taken as the default.
-	gm.addBoolean("isfertile", {name: "Is Fertile?", val: c => isFertile(c.slave)});
-	gm.addBoolean("isamputee", {name: "Is Amputee?", val: c => isAmputee(c.slave)});
-	gm.addBoolean("ispregnant", {name: "Is Pregnant?", val: c => c.slave.preg > 0});
-	gm.addNumber("devotion", {name: "Devotion", val: c => c.slave.devotion});
-	gm.addNumber("trust", {name: "Trust", val: c => c.slave.trust});
-	gm.addNumber("health", {name: "Health", val: c => c.slave.health.condition});
-	gm.addNumber("fatigue", {name: "Fatigue", val: c => c.slave.health.tired});
-	gm.addNumber("energy", {name: "Sex drive", val: c => c.slave.energy});
-	gm.addNumber("weight", {name: "Weight", val: c => c.slave.weight});
-	gm.addNumber("height", {name: "Height", val: c => c.slave.height});
-	gm.addNumber("age", {name: "Age", val: c => c.slave.actualAge});
-	gm.addNumber("physicalAge", {name: "Body Age", val: c => c.slave.physicalAge});
-	gm.addNumber("visualAge", {name: "Visible Age", val: c => c.slave.visualAge});
-	gm.addNumber("muscles", {name: "Muscles", val: c => c.slave.muscles});
-	gm.addNumber("lactation", {name: "Lactation", val: c => c.slave.lactation});
-	gm.addNumber("pregType", {name: "Pregnancy Multiples", val: c => c.slave.pregType});
-	gm.addNumber("bellyImplant", {name: "Belly Implant", val: c => c.slave.bellyImplant});
-	gm.addNumber("belly", {name: "Belly Size", val: c => c.slave.belly});
-	gm.addNumber("intelligenceImplant", {name: "Education", val: c => c.slave.intelligenceImplant});
-	gm.addNumber("intelligence", {name: "Intelligence", val: c => c.slave.intelligence});
-	gm.addNumber("accent", {name: "Accent", val: c => c.slave.accent});
-	gm.addNumber("waist", {name: "Waist", val: c => c.slave.waist});
-	gm.addNumber("chem", {name: "Carcinogen Buildup", val: c => c.slave.chem});
-	gm.addString("label", {name: "Label", val: c => c.slave.custom.label});
-	gm.addString("genes", {name: "Sex", val: c => c.slave.genes});
-	gm.addString("fetish", {name: "Fetish", val: c => c.slave.fetish});
+
+	// Booleans
+	gm.addBoolean("isfertile", {
+		name: "Is Fertile?", description: "Whether or not the slave is fertile.",
+		val: c => isFertile(c.slave)
+	});
+	gm.addBoolean("isamputee", {
+		name: "Is Amputee?", description: "Whether or not the slave has no limbs.",
+		val: c => isAmputee(c.slave)
+	});
+	gm.addBoolean("ispregnant", {
+		name: "Is Pregnant?", description: "Whether or not the slave is pregnant.",
+		val: c => c.slave.preg > 0
+	});
+
+	// Assignments
+	// Penthouse Assignments
+	gm.addAssignment("rest", {
+		name: "Resting", description: "Resting in the penthouse.",
+		val: c => c.slave.assignment === Job.REST
+	});
+	gm.addAssignment("fucktoy", {
+		name: "Fucktoy", description: "Pleasing the master.",
+		val: c => c.slave.assignment === Job.FUCKTOY
+	});
+	gm.addAssignment("classes", {
+		name: "Taking classes", description: "Taking classes to better serve.",
+		val: c => c.slave.assignment === Job.CLASSES
+	});
+	gm.addAssignment("house", {
+		name: "Cleaning", description: "Cleaning the penthouse.",
+		val: c => c.slave.assignment === Job.HOUSE
+	});
+	gm.addAssignment("whore", {
+		name: "Whoring", description: "Whoring themself out.",
+		val: c => c.slave.assignment === Job.WHORE
+	});
+	gm.addAssignment("public", {
+		name: "Serving public", description: "Serving the public.",
+		val: c => c.slave.assignment === Job.PUBLIC
+	});
+	gm.addAssignment("subordinate", {
+		name: "Subordinate", description: "Subordinate to other slaves.",
+		val: c => c.slave.assignment === Job.SUBORDINATE
+	});
+	gm.addAssignment("milked", {
+		name: "Milked", description: "Getting milked.",
+		val: c => c.slave.assignment === Job.MILKED
+	});
+	gm.addAssignment("gloryhole", {
+		name: "Glory hole", description: "Working as a glory hole.",
+		val: c => c.slave.assignment === Job.GLORYHOLE
+	});
+	gm.addAssignment("confinement", {
+		name: "Confined", description: "Confined at the penthouse.",
+		val: c => c.slave.assignment === Job.CONFINEMENT
+	});
+	gm.addAssignment("choice", {
+		name: "Choose own", description: "Allowed to choose their own job.",
+		val: c => c.slave.assignment === Job.CHOICE
+	});
+	// Leadership Assignments
+	gm.addAssignment("bodyguard", {
+		name: "Bodyguard", description: "Serving as Bodyguard.",
+		requirements: "Armory is built.", enabled: ()=>App.Entity.facilities.armory.established,
+		val: c => c.slave.assignment === Job.BODYGUARD
+	});
+	gm.addAssignment("headgirl", {
+		name: "Head Girl", description: "Serving as Head Girl",
+		val: c => c.slave.assignment === Job.HEADGIRL
+	});
+	gm.addAssignment("recruiter", {
+		name: "Recruiter", description: "Recruiting new slaves.",
+		val: c => c.slave.assignment === Job.RECRUITER
+	});
+	gm.addAssignment("agent", {
+		name: "Agent", description: "Serving as an Agent in another arcology.",
+		val: c => c.slave.assignment === Job.AGENT
+	});
+	gm.addAssignment("agentpartner", {
+		name: "Agent partner", description: "Serving an agent living in another arcology.",
+		val: c => c.slave.assignment === Job.AGENTPARTNER
+	});
+	// Facility Assignments
+	gm.addAssignment("arcade", {
+		name: "Confined in arcade", description: "Confined in the arcade.",
+		requirements: "Arcade is built.", enabled: ()=>App.Entity.facilities.arcade.established,
+		val: c => c.slave.assignment === Job.ARCADE
+	});
+	gm.addAssignment("madam", {
+		name: "Madam", description: "Serving as Madam.",
+		requirements: "Brothel is built.", enabled: ()=>App.Entity.facilities.brothel.established,
+		val: c => c.slave.assignment === Job.MADAM
+	});
+	gm.addAssignment("brothel", {
+		name: "Brothel whoring?", description: "Working in the brothel.",
+		requirements: "Brothel is built.", enabled: ()=>App.Entity.facilities.brothel.established,
+		val: c => c.slave.assignment === Job.BROTHEL
+	});
+	gm.addAssignment("warden", {
+		name: "Wardeness", description: "Serving as Wardeness.",
+		requirements: "Cellblock is built.", enabled: ()=>App.Entity.facilities.cellblock.established,
+		val: c => c.slave.assignment === Job.WARDEN
+	});
+	gm.addAssignment("cellblock", {
+		name: "Confined in cellblock?", description: "Confined in the cellblock.",
+		requirements: "Cellblock is built.", enabled: ()=>App.Entity.facilities.cellblock.established,
+		val: c => c.slave.assignment === Job.CELLBLOCK
+	});
+	gm.addAssignment("dj", {
+		name: "DJ", description: "Serving as DJ.",
+		requirements: "Club is built.", enabled: ()=>App.Entity.facilities.club.established,
+		val: c => c.slave.assignment === Job.DJ
+	});
+	gm.addAssignment("club", {
+		name: "Serving club", description: "Serving in the club.",
+		requirements: "Club is built.", enabled: ()=>App.Entity.facilities.club.established,
+		val: c => c.slave.assignment === Job.CLUB
+	});
+	gm.addAssignment("nurse", {
+		name: "Nurse", description: "Serving as Nurse.",
+		requirements: "Clinic is built.", enabled: ()=>App.Entity.facilities.clinic.established,
+		val: c => c.slave.assignment === Job.NURSE
+	});
+	gm.addAssignment("clinic", {
+		name: "Getting treatment", description: "Getting treatment in the clinic.",
+		requirements: "Clinic is built.", enabled: ()=>App.Entity.facilities.clinic.established,
+		val: c => c.slave.assignment === Job.CLINIC
+	});
+	gm.addAssignment("milkmaid", {
+		name: "Milkmaid", description: "Serving as Milkmaid",
+		requirements: "Dairy is built.", enabled: ()=>App.Entity.facilities.dairy.established,
+		val: c => c.slave.assignment === Job.MILKMAID
+	});
+	gm.addAssignment("dairy", {
+		name: "Work dairy", description: "Working in the dairy",
+		requirements: "Dairy is built.", enabled: ()=>App.Entity.facilities.dairy.established,
+		val: c => c.slave.assignment === Job.DAIRY
+	});
+	gm.addAssignment("farmer", {
+		name: "Farmer", description: "Serving as Farmer",
+		requirements: "Farmyard is built.", enabled: ()=>App.Entity.facilities.farmyard.established,
+		val: c => c.slave.assignment === Job.FARMER
+	});
+	gm.addAssignment("farmyard", {
+		name: "Farmhand", description: "Working as a farmhand.",
+		requirements: "Farmyard is built.", enabled: ()=>App.Entity.facilities.farmyard.established,
+		val: c => c.slave.assignment === Job.FARMYARD
+	});
+	gm.addAssignment("headgirlsuite", {
+		name: "Head Girl Servant", description: "Living with the Head Girl.",
+		requirements: "Head Girl Suite is built.", enabled: ()=>App.Entity.facilities.headGirlSuite.established,
+		val: c => c.slave.assignment === Job.HEADGIRLSUITE
+	});
+	gm.addAssignment("concubine", {
+		name: "Concubine", description: "Serving as Concubine.",
+		requirements: "Master suite is built.", enabled: ()=>App.Entity.facilities.masterSuite.established,
+		val: c => c.slave.assignment === Job.CONCUBINE
+	});
+	gm.addAssignment("mastersuite", {
+		name: "Master suite servant", description: "Serving in the master suite.",
+		requirements: "Master suite is built.", enabled: ()=>App.Entity.facilities.masterSuite.established,
+		val: c => c.slave.assignment === Job.MASTERSUITE
+	});
+	gm.addAssignment("matron", {
+		name: "Matron", description: "Serving as Matron.",
+		requirements: "Nursery is built.", enabled: ()=>App.Entity.facilities.nursery.established,
+		visible: () => V.experimental.nursery > 0,
+		val: c => c.slave.assignment === Job.MATRON
+	});
+	gm.addAssignment("nursery", {
+		name: "Nanny", description: "Working as a nanny.",
+		requirements: "Nursery is built.", enabled: ()=>App.Entity.facilities.nursery.established,
+		visible: () => V.experimental.nursery > 0,
+		val: c => c.slave.assignment === Job.NURSERY
+	});
+	gm.addAssignment("teacher", {
+		name: "Schoolteacher", description: "Serving as Schoolteacher.",
+		requirements: "Schoolroom is built.", enabled: ()=>App.Entity.facilities.schoolroom.established,
+		val: c => c.slave.assignment === Job.TEACHER
+	});
+	gm.addAssignment("school", {
+		name: "Learning", description: "Learning in the schoolroom.",
+		requirements: "Schoolroom is built.", enabled: ()=>App.Entity.facilities.schoolroom.established,
+		val: c => c.slave.assignment === Job.SCHOOL
+	});
+	gm.addAssignment("steward", {
+		name: "Stewardess", description: "Serving as Stewardess.",
+		requirements: "Servants Quarters are built.",
+		enabled: ()=>App.Entity.facilities.servantsQuarters.established,
+		val: c => c.slave.assignment === Job.STEWARD
+	});
+	gm.addAssignment("quarter", {
+		name: "Servant", description: "Working as a servant in the Servants Quarters.",
+		requirements: "Servants Quarters are built.",
+		enabled: ()=>App.Entity.facilities.servantsQuarters.established,
+		val: c => c.slave.assignment === Job.QUARTER
+	});
+	gm.addAssignment("attendant", {
+		name: "Attendant", description: "Serving as Attendant.",
+		requirements: "Spa is built.", enabled: ()=>App.Entity.facilities.spa.established,
+		val: c => c.slave.assignment === Job.ATTENDANT
+	});
+	gm.addAssignment("spa", {
+		name: "Spa resting", description: "Resting in the spa.",
+		requirements: "Spa is built.", enabled: () => App.Entity.facilities.spa.established,
+		val: c => c.slave.assignment === Job.SPA
+	});
+
+	// Numbers
+	gm.addNumber("devotion", {
+		name: "Devotion",
+		description: "Very Hateful: (-∞, -95), Hateful: [-95, -50), Resistant: [-50, -20), Ambivalent: [-20, 20], Accepting: (20, 50], Devoted: (50, 95], Worshipful: (95, ∞)",
+		val: c => c.slave.devotion
+	});
+	gm.addNumber("trust", {
+		name: "Trust",
+		description: "Extremely terrified: (-∞, -95), Terrified: [-95, -50), Frightened: [-50, -20), Fearful: [-20, 20], Careful: (20, 50], Trusting: (50, 95], Total trust: (95, ∞)",
+		val: c => c.slave.trust
+	});
+	gm.addNumber("health", {
+		name: "Health",
+		description: "Death: (-∞, -100), Near Death: [-100, -90), Extremely Unhealthy: [-90, -50), Unhealthy: [-50, -20), Healthy: [-20, 20], Very Healthy: (20, 50], Extremely Healthy: (50, 90], Unnaturally Healthy: (90, ∞)",
+		val: c => c.slave.health.condition
+	});
+	gm.addNumber("fatigue", {
+		name: "Fatigue",
+		description: "Energetic: (-∞, 0], Rested: (0, 30], Tired: (30, 60], Fatigued: (60, 90], Exhausted: (90, ∞)",
+		val: c => c.slave.health.tired
+	});
+	gm.addNumber("energy", {
+		name: "Sex drive",
+		description: "Frigid: (-∞, 20], Poor: (20, 40], Average: (40, 60], Powerful: (60, 80], Sex Addict: (80, 100), Nympho: 100",
+		val: c => c.slave.energy
+	});
+	gm.addNumber("weight", {
+		name: "Weight",
+		description: "Emaciated: (-∞, -95), Skinny: [-95, -30), Thin: [-30, -10), Average: [-10, 10], Plush: (10, 30], Overweight: (30, 95], Fat: (95, 130], Obese: (130, 160], Super Obese: (160, 190], Dangerously Obese: (190, ∞)",
+		val: c => c.slave.weight
+	});
+	gm.addNumber("height", {name: "Height", description: "Slave height in cm.", val: c => c.slave.height});
+	gm.addNumber("age", {name: "Age", description: "Real slave age", val: c => c.slave.actualAge});
+	gm.addNumber("physicalAge", {
+		name: "Body Age", description: "Age of the slave's body.",
+		val: c => c.slave.physicalAge
+	});
+	gm.addNumber("visualAge", {
+		name: "Visible Age", description: "How old the slave looks.",
+		val: c => c.slave.visualAge
+	});
+	gm.addNumber("muscles", {
+		name: "Muscles",
+		description: "Frail: (-∞, -96), Very weak: [-96, -31], Weak: [-31, -6), Soft: [-6, 5), Toned: [5, 30), Fit: [30, 50), Muscular: [50, 95), Hugely muscular: [95, ∞)",
+		val: c => c.slave.muscles
+	});
+	gm.addNumber("lactation", {
+		name: "Lactation", description: "0: None, 1: Natural, 2: Lactation implant",
+		val: c => c.slave.lactation
+	});
+	gm.addNumber("pregType", {
+		name: "Pregnancy Multiples", description: "Fetus count, known only after the 10th week of pregnancy",
+		val: c => c.slave.pregType
+	});
+	gm.addNumber("bellyImplant", {
+		name: "Belly Implant", description: "Volume in CCs. None: -1",
+		val: c => c.slave.bellyImplant
+	});
+	gm.addNumber("belly", {name: "Belly Size", description: "Volume in CCs, any source", val: c => c.slave.belly});
+	gm.addNumber("intelligenceImplant", {
+		name: "Education",
+		description: "Education level. 0: uneducated, 15: educated, 30: advanced education, (0, 15): incomplete education.",
+		val: c => c.slave.intelligenceImplant
+	});
+	gm.addNumber("intelligence", {
+		name: "Intelligence", description: "From moronic to brilliant: [-100, 100]",
+		val: c => c.slave.intelligence
+	});
+	gm.addNumber("accent", {
+		name: "Accent", description: "No accent: 0, Nice accent: 1, Bad accent: 2, Can't speak language: 3 and above",
+		val: c => c.slave.accent
+	});
+	gm.addNumber("waist", {
+		name: "Waist",
+		description: "Masculine waist: (95, ∞), Ugly waist: (40, 95], Unattractive waist: (10, 40], Average waist: [-10, 10], Feminine waist: [-40, -10), Wasp waist: [-95, -40), Absurdly narrow: (-∞, -95)",
+		val: c => c.slave.waist
+	});
+	gm.addNumber("chem", {
+		name: "Carcinogen Buildup",
+		description: "Side effects from drug use. If greater than 10 triggers will have negative consequences.",
+		val: c => c.slave.chem
+	});
+
+	// Strings
+	gm.addString("label", {name: "Label", description: "Assigned Label", val: c => c.slave.custom.label});
+	gm.addString("genes", {name: "Sex", description: "Genetic sex: Male: XX, Female: XY", val: c => c.slave.genes});
+	gm.addString("fetish", {
+		name: "Fetish",
+		description: "One of buttslut, cumslut, masochist, sadist, dom, submissive, boobs, pregnancy, none (AKA vanilla)",
+		val: c => c.slave.fetish
+	});
 
 	// Do this last:
 	gm.vanillaDone();
diff --git a/src/js/rulesAssistantOptions.js b/src/js/rulesAssistantOptions.js
index e3d1039d85ae2392e3b2f0ee86c066c3db5f068d..c41aa7ff14e7f7d890e1d9730bf22cb3aa28b109 100644
--- a/src/js/rulesAssistantOptions.js
+++ b/src/js/rulesAssistantOptions.js
@@ -1287,8 +1287,6 @@ App.RA.options = (function() {
 		constructor() {
 			super("Activation Condition");
 			this.appendChild(new ConditionBuilder());
-			this.appendChild(new AssignmentInclusion());
-			this.appendChild(new FacilityHeadAssignmentInclusion());
 			this.appendChild(new SpecificInclusionExclusion());
 			this.appendChild(new ApplyRuleOnce());
 		}
@@ -1300,71 +1298,6 @@ App.RA.options = (function() {
 		}
 	}
 
-	class AssignmentInclusionBase extends ButtonList {
-		/**
-		 * @param {string} label
-		 * @param {FC.Data.JobDesc[]} [jobs]
-		 * @param {App.Entity.Facilities.SingleJobFacility[]} [facilities]
-		 */
-		constructor(label, jobs, facilities) {
-			super(label);
-			this._attributes = {};
-			if (jobs !== undefined) {
-				jobs.forEach(job => {
-					this._attributes[capFirstChar(job.position)] = job.assignment;
-				});
-			}
-			if (facilities !== undefined) {
-				facilities.forEach(f => {
-					if (f.established && f.desc.defaultJob != null) { /* eslint-disable-line eqeqeq */
-						const displayName = f.name === "the " + f.genericName ? f.genericName : f.name;
-						this._attributes[displayName] = f.desc.jobs[f.desc.defaultJob].assignment;
-					}
-				});
-			}
-			for (const i in this._attributes) {
-				this.appendChild(new ButtonItem(i, this.getAttribute(i), current_rule.condition.assignment.includes(this.getAttribute(i))));
-			}
-		}
-
-		onchange() {
-			const allValues = this.getAllValues();
-			current_rule.condition.assignment = this.getSelection().concat(current_rule.condition.assignment.filter(a => !allValues.includes(a)));
-		}
-
-		getAttribute(what) {
-			return this._attributes[what];
-		}
-	}
-
-
-	class AssignmentInclusion extends AssignmentInclusionBase {
-		constructor() {
-			let facilities = [];
-			for (const f of Object.values(App.Entity.facilities)) {
-				if (f === App.Entity.facilities.penthouse) {
-					continue;
-				}
-				if (f.established) {
-					facilities.push(f);
-				}
-			}
-			super("Apply to assignments and facilities", Object.values(App.Data.Facilities.penthouse.jobs), facilities);
-		}
-	}
-
-	class FacilityHeadAssignmentInclusion extends AssignmentInclusionBase {
-		constructor() {
-			const jobs = [];
-			for (const f of Object.values(App.Entity.facilities)) {
-				if (f.established && f.desc.manager !== null) {
-					jobs.push(f.desc.manager);
-				}
-			}
-			super("Apply to facility heads", jobs);
-		}
-	}
-
 	class SpecificInclusionExclusion extends Options {
 		constructor() {
 			super();