diff --git a/devTools/FC.d.ts b/devTools/FC.d.ts
index bbaee44babb1af4a845284118b9112eaf7a0a492..e3e9a5ef4190bf9e15bc27882b2b5abf89c863d8 100644
--- a/devTools/FC.d.ts
+++ b/devTools/FC.d.ts
@@ -277,7 +277,7 @@ declare namespace App {
 		namespace DOM {
 			namespace Widgets { }
 
-			function makeElement<K extends keyof HTMLElementTagNameMap>(tag: K, content: string | Node, classNames: string | string[]): HTMLElementTagNameMap[K];
+			function makeElement<K extends keyof HTMLElementTagNameMap>(tag: K, content: string | Node, classNames?: string | string[]): HTMLElementTagNameMap[K];
 			function appendNewElement<K extends keyof HTMLElementTagNameMap>(tag: K, parent: ParentNode, content?: string | Node, classNames?: string | string[]): HTMLElementTagNameMap[K];
 		}
 		namespace View { }
diff --git a/js/002-config/fc-js-init.js b/js/002-config/fc-js-init.js
index f0f4a2670cc5bd8f4dee378273743a86e4e14c58..550ee01672b9960ed4cbf4b99e439ee30785b1ff 100644
--- a/js/002-config/fc-js-init.js
+++ b/js/002-config/fc-js-init.js
@@ -43,6 +43,7 @@ App.Facilities.ServantsQuarters = {};
 App.Facilities.Spa = {};
 App.Interact = {};
 App.Intro = {};
+App.Neighbor = {};
 App.MainView = {};
 App.Medicine = {};
 App.Medicine.Modification = {};
diff --git a/src/js/futureSocietyJS.js b/src/js/futureSocietyJS.js
index d0f12b4330a61b4ab894c712a0fc33057445d4bf..cfba77e9bdf49917629d0406c1d60cd714e0ce15 100644
--- a/src/js/futureSocietyJS.js
+++ b/src/js/futureSocietyJS.js
@@ -89,6 +89,7 @@ globalThis.FutureSocieties = (function() {
 	};
 
 	return {
+		activeFSes: activeFSes,
 		activeCount: activeCount,
 		applyBroadProgress: applyBroadProgress,
 		influenceSources: influenceSources,
diff --git a/src/js/generateMarketSlave.js b/src/js/generateMarketSlave.js
index 8d025903825d3d9d7f6a741d0f5b6a743464d02b..1088be2f554d0a8959f61628fa5d9273b9566c30 100644
--- a/src/js/generateMarketSlave.js
+++ b/src/js/generateMarketSlave.js
@@ -470,7 +470,7 @@ globalThis.generateMarketSlave = function(market = "kidnappers", numArcology = 1
 			} else {
 				market = 1;
 			}
-			opinion = ArcologyDiplomacy.opinion(0, market);
+			opinion = App.Neighbor.opinion(0, market);
 			opinion = Math.trunc(opinion/20);
 			opinion = Math.clamp(opinion, -10, 10);
 
diff --git a/src/neighbor/arcologyDiplomacy.js b/src/neighbor/arcologyDiplomacy.js
index 9283acf2496456789cd073622f8130437668d837..1df56958aefbae6d09ccb4983468eba2eb3560f8 100644
--- a/src/neighbor/arcologyDiplomacy.js
+++ b/src/neighbor/arcologyDiplomacy.js
@@ -1,78 +1,71 @@
-globalThis.ArcologyDiplomacy = (function() {
-	return {
-		opinion: opinion,
-		selectInfluenceTarget: selectInfluenceTarget
-	};
-
-	/** get one arcology's opinion of another
-	 * @param {number} activeID index
-	 * @param {number} targetID index
-	 * @returns {number} opinion
-	 */
-	function opinion(activeID, targetID) {
-		const activeArcology = V.arcologies[activeID];
-		const targetArcology = V.arcologies[targetID];
-		if (typeof activeArcology.FSNull === "undefined") { activeArcology.FSNull = "unset"; }
-		if (typeof targetArcology.FSNull === "undefined") { targetArcology.FSNull = "unset"; }
-
-		let opinion = 0;
-
-		const {shared, conflicting} = FutureSocieties.diplomaticFSes(activeID, targetID);
-
-		for (const fs of shared) {
-			opinion += activeArcology[fs];
-			opinion += targetArcology[fs];
-		}
+/** get one arcology's opinion of another
+ * @param {number} activeID index
+ * @param {number} targetID index
+ * @returns {number} opinion
+ */
+App.Neighbor.opinion = function(activeID, targetID) {
+	const activeArcology = V.arcologies[activeID];
+	const targetArcology = V.arcologies[targetID];
+	if (typeof activeArcology.FSNull === "undefined") { activeArcology.FSNull = "unset"; }
+	if (typeof targetArcology.FSNull === "undefined") { targetArcology.FSNull = "unset"; }
+
+	let opinion = 0;
+
+	const {shared, conflicting} = FutureSocieties.diplomaticFSes(activeID, targetID);
+
+	for (const fs of shared) {
+		opinion += activeArcology[fs];
+		opinion += targetArcology[fs];
+	}
 
-		for (const [activeFS, targetFS] of conflicting) {
-			opinion -= activeArcology[activeFS];
-			opinion -= targetArcology[targetFS];
-		}
+	for (const [activeFS, targetFS] of conflicting) {
+		opinion -= activeArcology[activeFS];
+		opinion -= targetArcology[targetFS];
+	}
 
-		// unshared but uncontested multiculturalism gets a relationship bonus
-		if (!shared.includes("FSNull") && !conflicting.some((pair) => pair.includes("FSNull"))) {
-			if (activeArcology.FSNull !== "unset") {
-				opinion += activeArcology.FSNull;
-			} else if (targetArcology.FSNull !== "unset") {
-				opinion += targetArcology.FSNull;
-			}
+	// unshared but uncontested multiculturalism gets a relationship bonus
+	if (!shared.includes("FSNull") && !conflicting.some((pair) => pair.includes("FSNull"))) {
+		if (activeArcology.FSNull !== "unset") {
+			opinion += activeArcology.FSNull;
+		} else if (targetArcology.FSNull !== "unset") {
+			opinion += targetArcology.FSNull;
 		}
-
-		return opinion = Number(opinion) || 0;
 	}
 
-	/** set a new influence target for a given arcology
-	 * @param {number} arcID
-	 */
-	function selectInfluenceTarget(arcID) {
-		const notMulticulturalism = (f) => f !== "FSNull"; // multiculturalism can neither influence nor be influenced
-		const influenceSources = FutureSocieties.influenceSources(arcID);
-		const arcology = V.arcologies[arcID];
-		if (influenceSources.length > 0) {
-			let eligibleTargets = [];
-			const obdedient = (arcology.government === "your trustees" || arcology.government === "your agent");
-
-			for (let targetID = 0; targetID < V.arcologies.length; ++targetID) {
-				const target = V.arcologies[targetID];
-				if (arcology.direction !== target.direction) {
-					if (!obdedient || target.direction !== 0) {
-						const {shared, conflicting} = FutureSocieties.diplomaticFSes(arcID, targetID);
-						let count = 0;
-						count += shared.filter(notMulticulturalism).length;
-						count += conflicting.filter((pair) => pair.every(notMulticulturalism)).length;
-						eligibleTargets.push(...Array(count).fill(target.direction));
-					}
+	return opinion = Number(opinion) || 0;
+};
+
+/** set a new influence target for a given arcology
+ * @param {number} arcID
+ */
+App.Neighbor.selectInfluenceTarget = function(arcID) {
+	const notMulticulturalism = (f) => f !== "FSNull"; // multiculturalism can neither influence nor be influenced
+	const influenceSources = FutureSocieties.influenceSources(arcID);
+	const arcology = V.arcologies[arcID];
+	if (influenceSources.length > 0) {
+		let eligibleTargets = [];
+		const obdedient = (arcology.government === "your trustees" || arcology.government === "your agent");
+
+		for (let targetID = 0; targetID < V.arcologies.length; ++targetID) {
+			const target = V.arcologies[targetID];
+			if (arcology.direction !== target.direction) {
+				if (!obdedient || target.direction !== 0) {
+					const {shared, conflicting} = FutureSocieties.diplomaticFSes(arcID, targetID);
+					let count = 0;
+					count += shared.filter(notMulticulturalism).length;
+					count += conflicting.filter((pair) => pair.every(notMulticulturalism)).length;
+					eligibleTargets.push(...Array(count).fill(target.direction));
 				}
 			}
+		}
 
-			if (eligibleTargets.length > 0) {
-				arcology.influenceTarget = eligibleTargets.random();
-			}
+		if (eligibleTargets.length > 0) {
+			arcology.influenceTarget = eligibleTargets.random();
 		}
 	}
-})();
+};
 
-globalThis.ArcologyFSPassiveInfluence = class {
+App.Neighbor.PassiveFSInfluence = class {
 	/** pick up social hints from an arcology's neighbors
 	 * @param {number} arcID
 	 */
@@ -106,7 +99,6 @@ globalThis.ArcologyFSPassiveInfluence = class {
 		let conflicting = new Map();
 		const arcology = V.arcologies[this._arcID];
 
-		debugger;
 		for (const [i, rel] of this._relationships) {
 			if (rel.shared.some((s) => s === fs)) {
 				if (V.arcologies[i][fs] > arcology[fs] + this._thresh) {
diff --git a/src/neighbor/neighborGrid.css b/src/neighbor/neighborGrid.css
new file mode 100644
index 0000000000000000000000000000000000000000..dc20b25dbf10e2750d32b46bd72d5007a069f338
--- /dev/null
+++ b/src/neighbor/neighborGrid.css
@@ -0,0 +1,35 @@
+.neighborgrid-container {
+  display: grid;
+  grid-template-columns: auto auto auto;
+  background-color: rgba(17,17,17,1);
+  padding: 5px;
+  max-width: fit-content;
+  margin: 0 auto;
+}
+
+.neighborgrid-item {
+  margin: 3px;
+  border: 5px solid;
+  padding: 10px;
+  text-align: center;
+}
+
+.neighborgrid-item-selected {
+  background-color: rgba(255, 255, 255, 0.2);
+}
+
+.neighborgrid-item-empty {
+  border-color: rgba(17,17,17,1);
+}
+
+.neighborgrid-item-self {
+  border-color: white;
+}
+
+.neighborgrid-item-owned {
+  border-color: steelblue;
+}
+
+.neighborgrid-item-unowned {
+  border-color: red;
+}
diff --git a/src/neighbor/neighborGrid.js b/src/neighbor/neighborGrid.js
new file mode 100644
index 0000000000000000000000000000000000000000..04582e12aeba86edcc6fe0ca05b3f2000f972404
--- /dev/null
+++ b/src/neighbor/neighborGrid.js
@@ -0,0 +1,146 @@
+App.Neighbor.Grid = class {
+	/** Create a neighbor grid controller
+	 * @param {function(number):void} onSelection function to be called when selection changes
+	 */
+	constructor(onSelection) {
+		this._onSelection = onSelection;
+	}
+
+	/** Render the neighbor grid to DOM
+	 * @returns {Element}
+	 */
+	render() {
+		const container = App.UI.DOM.makeElement("div", null, "neighborgrid-container");
+
+		this._renderCell(container, "northwest");
+		this._renderCell(container, "north");
+		this._renderCell(container, "northeast");
+		this._renderCell(container, "west");
+		this._renderCell(container, "self");
+		this._renderCell(container, "east");
+		this._renderCell(container, "southwest");
+		this._renderCell(container, "south");
+		this._renderCell(container, "southeast");
+
+		return container;
+	}
+
+	/** Render a single cell to the grid
+	 * @param {Element} container
+	 * @param {string} direction
+	 */
+	_renderCell(container, direction) {
+		const arcID = direction === "self" ? 0 : V.arcologies.findIndex((a) => a.direction === direction);
+		const arcology = V.arcologies[arcID];
+
+		/** render styled text with a tooltip
+		 * @param {string} text
+		 * @param {string} tooltipText
+		 * @param {string} className
+		 * @returns {HTMLSpanElement}
+		 */
+		function withTooltip(text, tooltipText, className) {
+			let tooltip = App.UI.DOM.makeElement("span", tooltipText, "tooltip");
+			let res = App.UI.DOM.makeElement("span", text, ["textWithTooltip", className]);
+			res.appendChild(tooltip);
+			return res;
+		}
+
+		function nameFrag() {
+			return App.UI.DOM.makeElement("div", App.UI.DOM.link(arcology.name, () => this.select(arcID)), "name");
+		}
+
+		function govGSPFrag() {
+			let frag = document.createDocumentFragment();
+			let gov;
+			if (arcID === 0) {
+				gov = App.UI.DOM.appendNewElement("div", frag, "Your arcology; ");
+			} else {
+				gov = App.UI.DOM.appendNewElement("div", frag, capFirstChar(arcology.government) + '; ');
+			}
+			const estimatedGSP = Math.trunc((0.1 * arcology.prosperity * random(100 - V.economicUncertainty, 100 + V.economicUncertainty))/100);
+			App.UI.DOM.appendNewElement("span", gov, cashFormat(estimatedGSP) + "m GSP", "cash");
+			return frag;
+		}
+
+		function fsFrag() {
+			let frag = document.createDocumentFragment();
+			frag.appendChild(document.createTextNode("FS: "));
+			if (arcID === 0) {
+				const fses = FutureSocieties.activeFSes(0);
+				for (const fs of fses) {
+					frag.appendChild(withTooltip("⯁", FutureSocieties.displayName(fs), "steelblue"));
+				}
+			} else {
+				const fses = FutureSocieties.activeFSes(arcID);
+				const diplo = FutureSocieties.diplomaticFSes(arcID, 0);
+				let style = "white";
+				for (const fs of fses) {
+					if (diplo.shared.includes(fs)) {
+						style = "steelblue";
+					} else {
+						const conflict = diplo.conflicting.find((f) => f[0] === fs);
+						if (conflict) {
+							style = "red";
+						}
+					}
+					frag.appendChild(withTooltip("⯁", FutureSocieties.displayName(fs), style));
+				}
+			}
+			return App.UI.DOM.makeElement("div", frag);
+		}
+
+		function ownershipFrag() {
+			let frag = document.createDocumentFragment();
+			const owned = arcID === 0 || arcology.government === "your trustees" || arcology.government === "your agent";
+			const yourPercent = owned ? `${arcology.ownership}%` : `${arcology.PCminority}%`;
+			const hostilePercent = owned ? `${arcology.minority}%` : `${arcology.ownership}%`;
+			const challengerPercent = owned ? null : `${arcology.minority}%`;
+			const publicPercent = `${100 - (arcology.ownership + arcology.minority + arcology.PCminority)}%`;
+			App.UI.DOM.appendNewElement("li", frag, withTooltip(yourPercent, "Your ownership", "steelblue"));
+			if (arcID !== 0) {
+				App.UI.DOM.appendNewElement("li", frag, withTooltip(publicPercent, "Public ownership", "yellow"));
+			}
+			if (challengerPercent) {
+				App.UI.DOM.appendNewElement("li", frag, withTooltip(challengerPercent, "Minority challenger ownership", "orange"));
+			}
+			App.UI.DOM.appendNewElement("li", frag, withTooltip(hostilePercent, "Hostile ownership", "red"));
+			return App.UI.DOM.makeElement("ul", frag, "choicesStrip");
+		}
+
+		let frag = document.createDocumentFragment();
+		let classNames = ["neighborgrid-item"];
+		if (!arcology) {
+			// empty block
+			classNames.push("neighborgrid-item-empty");
+		} else {
+			frag.appendChild(nameFrag());
+			frag.appendChild(govGSPFrag());
+			frag.appendChild(fsFrag());
+			frag.appendChild(ownershipFrag());
+			if (arcID === 0) {
+				classNames.push("neighborgrid-item-self");
+			} else if (arcology.government === "your trustees" || arcology.government === "your agent") {
+				classNames.push("neighborgrid-item-owned");
+			} else {
+				classNames.push("neighborgrid-item-unowned");
+			}
+		}
+
+		const element = App.UI.DOM.appendNewElement("div", container, frag, classNames);
+		element.id = `neighborgrid-cell-${direction}`;
+	}
+
+	/** Set the selection to a particular arcology
+	 * @param {number} arcID
+	 */
+	select(arcID) {
+		const direction = arcID === 0 ? "self" : V.arcologies[arcID].direction;
+		const elementID = `#neighborgrid-cell-${direction}`;
+
+		$("div[id^='neighborgrid-cell']").each((i, e) => { $(e).removeClass("neighborgrid-item-selected"); });
+		$(elementID).addClass("neighborgrid-item-selected");
+
+		this._onSelection(arcID);
+	}
+};
diff --git a/src/uncategorized/arcmgmt.tw b/src/uncategorized/arcmgmt.tw
index 95d9db643b04013c722a4955dc2aa24065daf650..6210c09c70d1be8ae36286540c86633283df19dd 100644
--- a/src/uncategorized/arcmgmt.tw
+++ b/src/uncategorized/arcmgmt.tw
@@ -1501,7 +1501,7 @@ You own
 	<<set _desc = []>>
 	<<set _descNeg = []>>
 	<<for $i = 1; $i < $arcologies.length; $i++>>
-		<<set _opinion = ArcologyDiplomacy.opinion(0, $i)>>
+		<<set _opinion = App.Neighbor.opinion(0, $i)>>
 		<<if _opinion >= 100>>
 			<<set _desc.push($arcologies[$i].name)>>
 		<<elseif _opinion <= -100>>
diff --git a/src/uncategorized/bulkSlaveGenerate.tw b/src/uncategorized/bulkSlaveGenerate.tw
index c7f3d0b90a52cf310d757cab9f145ddec3263cb2..eb87587f3c2dbd68742c0321995bab72f9c8cacb 100644
--- a/src/uncategorized/bulkSlaveGenerate.tw
+++ b/src/uncategorized/bulkSlaveGenerate.tw
@@ -65,7 +65,7 @@
 	<<if $numArcology >= $arcologies.length>>
 		<<set $numArcology = 1>>
 	<</if>>
-	<<set _opinion = ArcologyDiplomacy.opinion(0, $numArcology)>>
+	<<set _opinion = App.Neighbor.opinion(0, $numArcology)>>
 	<<set _opinion = Math.clamp(Math.trunc(_opinion/20), -10, 10)>>
 	<<set $discount -= (_opinion * 25)>>
 
diff --git a/src/uncategorized/neighborInteract.tw b/src/uncategorized/neighborInteract.tw
index 97c8bba3c3bfdc2a219b6b7d6a0cc76d4eb46a7f..52cad24e8d65493be51808bfbc13476b1fe2b30c 100644
--- a/src/uncategorized/neighborInteract.tw
+++ b/src/uncategorized/neighborInteract.tw
@@ -169,7 +169,12 @@ You have <<print $arcologies.length-1>> neighbors. <br><br>
 	<<set $desc.push("Chinese Revivalism")>>
 <</if>>
 
-<br> <span id="Security"> <br>
+<<set _selFunc = (f) => {}>>
+<<set _grid = new App.Neighbor.Grid(_selFunc)>>
+<<includeDOM _grid.render()>>
+<<run _grid.select(0)>>
+
+<span id="Security">
 <<if $desc.length == 0>>
 	Your arcology's culture has not developed to the point where it can meaningfully influence other arcologies.
 <<elseif $desc.length > 2>>
@@ -190,12 +195,12 @@ You have <<print $arcologies.length-1>> neighbors. <br><br>
 
 <<for _currentNeighbor = 1; _currentNeighbor < $arcologies.length; _currentNeighbor++>>
 <<set _Agent = App.currentAgent(_currentNeighbor)>>
-<<capture _currentNeighbor, _Agent>>
+<<capture _currentNeighbor, _Agent, _grid>>
 	<<set $arcologies[_currentNeighbor].prosperity = Math.clamp($arcologies[_currentNeighbor].prosperity, 1, 300)>>
 	<br>You own $arcologies[_currentNeighbor].PCminority% of
 	<<link "$arcologies[_currentNeighbor].name">>
 	<<replace "#Security">> <<set $activeArcologyIdx = _currentNeighbor>>
-	<br>[[Back to the main diplomacy page|Neighbor Interact]]
+	[[Back to the main diplomacy page|Neighbor Interact]]
 	<<set $i = _currentNeighbor>> <<include "Neighbor Description">> <br>
 
 		<<set _ownershipCost = 500*Math.trunc($arcologies[_currentNeighbor].prosperity*(1+($arcologies[_currentNeighbor].demandFactor/100)))>>
@@ -402,7 +407,7 @@ You have <<print $arcologies.length-1>> neighbors. <br><br>
 	<<elseif $arcologies[_currentNeighbor].direction === $arcologies[0].embargoTarget>>
 		Due to your active embargo, trade with $arcologies[_currentNeighbor].name is not possible.
 	<<else>>
-		<<set _opinion = ArcologyDiplomacy.opinion(_currentNeighbor, 0)>>
+		<<set _opinion = App.Neighbor.opinion(_currentNeighbor, 0)>>
 		<<set _prices = _opinion*10>>
 	<</if>>
 	<<if $arcologies[_currentNeighbor].FSRomanRevivalist > 95>>
diff --git a/src/uncategorized/neighborsDevelopment.tw b/src/uncategorized/neighborsDevelopment.tw
index 266655e7b0318633d9df4b046d44f3e34125f816..2d7dd91affb1419de1dbddb58b861885418fe3a0 100644
--- a/src/uncategorized/neighborsDevelopment.tw
+++ b/src/uncategorized/neighborsDevelopment.tw
@@ -426,7 +426,7 @@ has an estimated GSP of @@.yellowgreen;<<print cashFormat(_prosperity)>><<if $sh
 <<if $arcologies[$i].direction != 0>>
 	<<run FutureSocieties.applyBroadProgress($i, _efficiency)>>
 <</if>>
-<<set _passive = new ArcologyFSPassiveInfluence($i)>>
+<<set _passive = new App.Neighbor.PassiveFSInfluence($i)>>
 <<if $arcologies[$i].FSSupremacist != "unset">>
 	<<= _passive.output("FSSupremacist")>>
 	<<if $arcologies[$i].direction != 0>>
@@ -1780,7 +1780,7 @@ has an estimated GSP of @@.yellowgreen;<<print cashFormat(_prosperity)>><<if $sh
 
 <<if $arcologies[$i].direction != 0>>
 	<<if $arcologies[$i].influenceTarget == -1>>
-		<<run ArcologyDiplomacy.selectInfluenceTarget($i)>>
+		<<run App.Neighbor.selectInfluenceTarget($i)>>
 	<</if>>
 <</if>>
 
diff --git a/src/uncategorized/neighborsFSAdoption.tw b/src/uncategorized/neighborsFSAdoption.tw
index b044c7c478eac4cdf90486a6556225abfe662720..69cf45e9b1ea942ca78a741aed904a6aae5c5fea 100644
--- a/src/uncategorized/neighborsFSAdoption.tw
+++ b/src/uncategorized/neighborsFSAdoption.tw
@@ -871,7 +871,7 @@ societal development.
 			<<set _influenceBonus = 20>>
 		<</if>>
 
-		<<set _opinion = ArcologyDiplomacy.opinion($i, $j)>>
+		<<set _opinion = App.Neighbor.opinion($i, $j)>>
 		<<if _opinion >= 50>>
 			$arcologies[$i].name is aligned with $arcologies[$j].name socially, encouraging it to consider adopting all its cultural values.
 			<<set _influenceBonus += _opinion-50>>
diff --git a/src/uncategorized/slaveMarkets.tw b/src/uncategorized/slaveMarkets.tw
index 070a4756e2f9d0c05ab9e7c037e7528d400f61ad..afa58266093315505f82989dc9f89f1a8981c3c5 100644
--- a/src/uncategorized/slaveMarkets.tw
+++ b/src/uncategorized/slaveMarkets.tw
@@ -139,7 +139,7 @@ You visit the slave markets off the arcology plaza. It's always preferable to ex
 
 <<case "neighbor">>
 	You're in the area of the slave market that specializes in slaves from within the Free City, viewing slaves from ''<<print "$arcologies[$numArcology].name">>''. Some were trained there, specifically for sale, while others are simply being sold.
-	<<set _opinion = ArcologyDiplomacy.opinion(0, $numArcology)>>
+	<<set _opinion = App.Neighbor.opinion(0, $numArcology)>>
 	<<if _opinion != 0>>
 		<<set _slaveCost -= Math.trunc(_slaveCost*_opinion*0.05)>>
 		<<if _opinion > 2>>