From 55d2e864ee28ec26911d88539a8f1bab106e9d57 Mon Sep 17 00:00:00 2001
From: joris <joris.dauphin@gmail.com>
Date: Sat, 18 Feb 2012 09:59:57 +0100
Subject: [PATCH] New unit-type flag "Wall", which will treat unit as wall.
 This will use the unit's Direction field to show different wall connections.
 The unit should be a building and have NumDirections = 16 and Flip = false.
 The wall unit didn't count in GetNumOpponents. When it attacked, AI didn't
 fight back and no warnings are shown.

NOTE: need to modify triggers, so they won't count the wall units when calculating player's remaining units. Not needed for those games which don't use this feature.

Wall mapping:
 Frame 0 - wall without connections.
 Frame 1 - wall with north connection.
 Frame 2 - wall with west connection.
 Frame 3 - wall with north and west connections.
 Frame 4 - wall with south connection.
 Frame 5 - wall with south and north connections.
 Frame 6 - wall with south and west connections.
 Frame 7 - wall with south, north and west connections.
 Frame 8 - wall with east connection.
 Frame 9 - wall with east and north connections.
 Frame 10 - wall with east and west connections.
 Frame 11 - wall with east, north and west connections.
 Frame 12 - wall with east and south connections.
 Frame 13 - wall with east, south and north connections.
 Frame 14 - wall with east, south and west connections.
 Frame 15 - wall with all connections.

patch from Cybermind.
---
 src/action/action_build.cpp  |   7 +-
 src/game/trigger.cpp         |  14 ++--
 src/include/unit.h           |   5 ++
 src/include/unittype.h       |   6 +-
 src/unit/script_unittype.cpp |  11 ++--
 src/unit/unit.cpp            | 121 ++++++++++++++++++++++++++++++++---
 src/unit/unittype.cpp        |   5 +-
 7 files changed, 147 insertions(+), 22 deletions(-)

diff --git a/src/action/action_build.cpp b/src/action/action_build.cpp
index fae3a1e78..a49f80e00 100644
--- a/src/action/action_build.cpp
+++ b/src/action/action_build.cpp
@@ -593,7 +593,12 @@ void HandleActionBuilt(COrder& order, CUnit &unit)
 
 		// Set the direction of the building if it supports them
 		if (unit.Type->NumDirections > 1) {
-			unit.Direction = (MyRand() >> 8) & 0xFF; // random heading
+			if (unit.Type->Wall) { // Special logic for walls
+				CorrectWallDirections(unit);
+				CorrectWallNeighBours(unit);
+			} else {
+				unit.Direction = (MyRand() >> 8) & 0xFF; // random heading
+			}
 			UnitUpdateHeading(unit);
 		}
 
diff --git a/src/game/trigger.cpp b/src/game/trigger.cpp
index 518f1cb46..249128e13 100644
--- a/src/game/trigger.cpp
+++ b/src/game/trigger.cpp
@@ -443,13 +443,19 @@ int GetNumOpponents(int player)
 
 	// Check the player opponents
 	for (int i = 0; i < PlayerMax; ++i) {
+		const int unitCount = Players[i].TotalNumUnits;
+
 		// This player is our enemy and has units left.
-		if (((Players[player].Enemy & (1 << i)) || (Players[i].Enemy & (1 << player))) &&
-				Players[i].TotalNumUnits) {
-			++n;
+		if ((Players[player].Enemy & (1 << i)) || (Players[i].Enemy & (1 << player))) {
+			// Don't count walls
+			for (int j = 0; j < unitCount; ++j) {
+				if (Players[i].Units[j]->Type->Wall == false) {
+					++n;
+					break;
+				}
+			}
 		}
 	}
-
 	return n;
 }
 
diff --git a/src/include/unit.h b/src/include/unit.h
index 55a8b3167..8f0157176 100644
--- a/src/include/unit.h
+++ b/src/include/unit.h
@@ -1046,6 +1046,11 @@ extern int DirectionToHeading(const Vec2i &dir);
 	/// Convert direction (dx,dy) to heading (0-255)
 extern int DirectionToHeading(const PixelDiff &dir);
 
+	///Correct directions for placed wall.
+extern void CorrectWallDirections(CUnit &unit);
+	/// Correct the surrounding walls.
+extern void CorrectWallNeighBours(CUnit &unit);
+
 	/// Update frame from heading
 extern void UnitUpdateHeading(CUnit &unit);
 	/// Heading and frame from delta direction
diff --git a/src/include/unittype.h b/src/include/unittype.h
index 86b3bcdf5..fbc30cd79 100644
--- a/src/include/unittype.h
+++ b/src/include/unittype.h
@@ -645,6 +645,7 @@ enum {
 	TELEPORTER_INDEX,
 	SHIELDPIERCE_INDEX,
 	SAVECARGO_INDEX,
+	WALL_INDEX,
 	NBARALREADYDEFINED
 };
 
@@ -1007,8 +1008,9 @@ public:
 	unsigned Decoration : 1;            /// Unit is a decoration (act as tile).
 	unsigned Indestructible : 1;        /// Unit is indestructible (take no damage).
 	unsigned Teleporter : 1;            /// Can teleport other units.
-	unsigned ShieldPiercing : 1;            /// Can directly damage shield-protected units, without shield damaging.
-	unsigned SaveCargo : 1;            /// Unit unloads his passengers after death.
+	unsigned ShieldPiercing : 1;        /// Can directly damage shield-protected units, without shield damaging.
+	unsigned SaveCargo : 1;             /// Unit unloads his passengers after death.
+	unsigned Wall : 1;                  /// Use special logic for Direction field.
 
 	CVariable *Variable;            /// Array of user defined variables.
 	struct BoolFlags {
diff --git a/src/unit/script_unittype.cpp b/src/unit/script_unittype.cpp
index 68d3ee245..1f7542ce7 100644
--- a/src/unit/script_unittype.cpp
+++ b/src/unit/script_unittype.cpp
@@ -106,6 +106,7 @@ static const char INDESTRUCTIBLE_KEY[] = "Indestructible";
 static const char TELEPORTER_KEY[] = "Teleporter";
 static const char SHIELDPIERCE_KEY[] = "ShieldPiercing";
 static const char SAVECARGO_KEY[] = "LoseCargo";
+static const char WALL_KEY[] = "Wall";
 // names of the variable.
 static const char HITPOINTS_KEY[] = "HitPoints";
 static const char BUILD_KEY[] = "Build";
@@ -153,7 +154,7 @@ CUnitTypeVar::CBoolKeys::CBoolKeys() {
 		SHOREBUILDING_KEY, CANATTACK_KEY,BUILDEROUTSIDE_KEY,
 		BUILDERLOST_KEY,CANHARVEST_KEY,HARVESTER_KEY,SELECTABLEBYRECTANGLE_KEY,
 		ISNOTSELECTABLE_KEY,DECORATION_KEY,INDESTRUCTIBLE_KEY,TELEPORTER_KEY,SHIELDPIERCE_KEY,
-		SAVECARGO_KEY};
+		SAVECARGO_KEY, WALL_KEY};
 
 	for (int i = 0; i < NBARALREADYDEFINED; ++i) {
 		buildin[i].offset = i;
@@ -1005,6 +1006,8 @@ static int CclDefineUnitType(lua_State *l)
 			type->ShieldPiercing = LuaToBoolean(l, -1);
 		} else if (!strcmp(value, "SaveCargo")) {
 			type->SaveCargo = LuaToBoolean(l, -1);
+		} else if (!strcmp(value, "Wall")) {
+			type->Wall = LuaToBoolean(l, -1);
 		} else if (!strcmp(value, "Sounds")) {
 			if (!lua_istable(l, -1)) {
 				LuaError(l, "incorrect argument");
@@ -2095,9 +2098,9 @@ void UpdateUnitVariables(const CUnit &unit)
 	type->BoolFlag[DECORATION_INDEX].value            = type->Decoration;
 	type->BoolFlag[INDESTRUCTIBLE_INDEX].value        = type->Indestructible;
 	type->BoolFlag[TELEPORTER_INDEX].value            = type->Teleporter;
-	type->BoolFlag[SHIELDPIERCE_INDEX].value            = type->ShieldPiercing;
-	type->BoolFlag[SAVECARGO_INDEX].value            = type->SaveCargo;
-
+	type->BoolFlag[SHIELDPIERCE_INDEX].value          = type->ShieldPiercing;
+	type->BoolFlag[SAVECARGO_INDEX].value             = type->SaveCargo;
+	type->BoolFlag[WALL_INDEX].value                  = type->Wall;
 }
 
 
diff --git a/src/unit/unit.cpp b/src/unit/unit.cpp
index 2967b9213..fbaf3c338 100644
--- a/src/unit/unit.cpp
+++ b/src/unit/unit.cpp
@@ -428,7 +428,7 @@ void CUnit::AssignToPlayer(CPlayer *player)
 			// don't count again
 			if (type->Building) {
 				// FIXME: support more races
-				if (type != UnitTypeOrcWall && type != UnitTypeHumanWall) {
+				if (!type->Wall && type != UnitTypeOrcWall && type != UnitTypeHumanWall) {
 					player->TotalBuildings++;
 				}
 			} else {
@@ -445,7 +445,7 @@ void CUnit::AssignToPlayer(CPlayer *player)
 	// Don't Add the building if it's dieing, used to load a save game
 	if (type->Building && CurrentAction() != UnitActionDie) {
 		// FIXME: support more races
-		if (type != UnitTypeOrcWall && type != UnitTypeHumanWall) {
+		if (!type->Wall && type != UnitTypeOrcWall && type != UnitTypeHumanWall) {
 			player->NumBuildings++;
 		}
 	}
@@ -860,6 +860,12 @@ void CUnit::Place(const Vec2i &pos)
 	UnitCountSeen(*this);
 	// Vision
 	MapMarkUnitSight(*this);
+	// Correct directions for wall units
+	if (this->Type->Wall && this->CurrentAction() != UnitActionBuilt){
+		CorrectWallDirections(*this);
+		UnitUpdateHeading(*this);
+		CorrectWallNeighBours(*this);
+	}
 }
 
 /**
@@ -912,6 +918,11 @@ void CUnit::Remove(CUnit *host)
 
 	Removed = 1;
 
+	// Correct surrounding walls directions
+	if (this->Type->Wall){
+		CorrectWallNeighBours(*this);
+	}
+
 	//  Remove unit from the current selection
 	if (Selected) {
 		if (NumSelected == 1) { //  Remove building cursor
@@ -974,7 +985,7 @@ void UnitLost(CUnit &unit)
 
 		if (unit.Type->Building) {
 			// FIXME: support more races
-			if (type != UnitTypeOrcWall && type != UnitTypeHumanWall) {
+			if (!type->Wall && type != UnitTypeOrcWall && type != UnitTypeHumanWall) {
 				player->NumBuildings--;
 			}
 		}
@@ -1152,6 +1163,97 @@ static void UnitFillSeenValues(CUnit &unit)
 	unit.CurrentOrder()->FillSeenValues(unit);
 }
 
+class SamePlayerAndTypeAs
+{
+public:
+	explicit SamePlayerAndTypeAs(const CUnit &unit) :
+		player(unit.Player), type(unit.Type)
+	{}
+
+	bool operator() (const CUnit *unit) const
+	{
+		return (unit->Player == player && unit->Type == type);
+	}
+
+private:
+	const CPlayer *player;
+	const CUnitType *type;
+};
+
+// Wall unit positions
+enum {
+	W_NORTH = 0x10,
+	W_WEST = 0x20,
+	W_SOUTH = 0x40,
+	W_EAST = 0x80
+};
+
+/**
+**  Correct direction for placed wall.
+**
+**  @param unit    The wall unit.
+*/
+void CorrectWallDirections(CUnit &unit)
+{
+	Assert(unit.Type->Wall);
+	Assert(unit.Type->NumDirections == 16);
+	Assert(!unit.Type->Flip);
+
+	if (!Map.Info.IsPointOnMap(unit.tilePos)) {
+		return ;
+	}
+	const struct {
+		Vec2i offset;
+		const int dirFlag;
+	} configs[] = {{{0, -1}, W_NORTH}, {{1, 0}, W_EAST},
+		{{0, 1}, W_SOUTH}, {{-1, 0}, W_WEST}};
+	int flags = 0;
+
+	for (int i = 0; i != sizeof (configs) / sizeof (*configs); ++i) {
+		const Vec2i pos = unit.tilePos + configs[i].offset;
+		const int dirFlag = configs[i].dirFlag;
+
+		if (Map.Info.IsPointOnMap(pos) == false) {
+			flags |= dirFlag;
+		} else {
+			const CUnitCache &unitCache = Map.Field(pos)->UnitCache;
+			const CUnit *neighboor = unitCache.find(SamePlayerAndTypeAs(unit));
+
+			if (neighboor != NULL) {
+				flags |= dirFlag;
+			}
+		}
+	}
+	unit.Direction = flags;
+}
+
+/**
+** Correct the surrounding walls.
+**
+** @param unit The wall unit.
+*/
+void CorrectWallNeighBours(CUnit &unit)
+{
+	Assert(unit.Type->Wall);
+
+	const Vec2i offset[] = {{1, 0}, {-1, 0}, {0, 1}, {0, -1}};
+
+	for (unsigned int i = 0; i < sizeof (offset) / sizeof (*offset); ++i) {
+		const Vec2i pos = unit.tilePos + offset[i];
+
+		if (Map.Info.IsPointOnMap(pos) == false) {
+			continue;
+		}
+		CUnitCache &unitCache = Map.Field(pos)->UnitCache;
+		CUnit *neighboor = unitCache.find(SamePlayerAndTypeAs(unit));
+
+		if (neighboor != NULL) {
+			CorrectWallDirections(*neighboor);
+			UnitUpdateHeading(*neighboor);
+		}
+	}
+}
+
 /**
 **  This function should get called when a unit goes under fog of war.
 **
@@ -1554,9 +1656,10 @@ void CUnit::ChangeOwner(CPlayer &newplayer)
 
 	PlayerSlot = newplayer.Units + newplayer.TotalNumUnits++;
 	if (Type->Building) {
-		newplayer.TotalBuildings++;
-	}
-	else {
+		if (!Type->Wall) {
+			newplayer.TotalBuildings++;
+		}
+	} else {
 		newplayer.TotalUnits++;
 	}
 	*PlayerSlot = this;
@@ -1581,7 +1684,7 @@ void CUnit::ChangeOwner(CPlayer &newplayer)
 			newplayer.MaxResources[i] += Type->_Storing[i];
 		}
 	}
-	if (Type->Building) {
+	if (Type->Building && !Type->Wall) {
 		newplayer.NumBuildings++;
 	}
 	newplayer.UnitTypesCount[Type->Slot]++;
@@ -2842,7 +2945,7 @@ void HitUnit(CUnit *attacker, CUnit &target, int damage)
 	if (attacker) {
 		target.DamagedType = ExtraDeathIndex(attacker->Type->DamageType.c_str());
 	}
-	if (!lastattack || lastattack + 2 * CYCLES_PER_SECOND < GameCycle) {
+	if (!target.Type->Wall && (!lastattack || lastattack + 2 * CYCLES_PER_SECOND < GameCycle)) {
 		// NOTE: perhaps this should also be moved into the notify?
 		if (target.Player == ThisPlayer) {
 			// FIXME: Problem with load+save.
@@ -2875,7 +2978,7 @@ void HitUnit(CUnit *attacker, CUnit &target, int damage)
 		}
 	}
 
-	if (attacker && target.Type->Building && target.Player->AiEnabled) {
+	if (attacker && !target.Type->Wall && target.Type->Building && target.Player->AiEnabled) {
 		AiHelpMe(attacker, target);
 	}
 
diff --git a/src/unit/unittype.cpp b/src/unit/unittype.cpp
index fd73acc9a..478202e52 100644
--- a/src/unit/unittype.cpp
+++ b/src/unit/unittype.cpp
@@ -113,7 +113,7 @@ CUnitType::CUnitType() :
 	ShadowWidth(0), ShadowHeight(0), ShadowOffsetX(0), ShadowOffsetY(0),
 	Animations(NULL), StillFrame(0),
 	DeathExplosion(NULL), CorpseType(NULL),
-	Construction(NULL),  RepairHP(0), TileWidth(0), TileHeight(0),
+	Construction(NULL), RepairHP(0), TileWidth(0), TileHeight(0),
 	BoxWidth(0), BoxHeight(0), NumDirections(0), MinAttackRange(0),
 	ReactRangeComputer(0), ReactRangePerson(0), Priority(0),
 	BurnPercent(0), BurnDamageRate(0), RepairRange(0),
@@ -129,7 +129,8 @@ CUnitType::CUnitType() :
 	Vanishes(0), GroundAttack(0), ShoreBuilding(0), CanAttack(0),
 	BuilderOutside(0), BuilderLost(0), CanHarvest(0), Harvester(0),
 	Neutral(0), SelectableByRectangle(0), IsNotSelectable(0), Decoration(0),
-	Indestructible(0), Teleporter(0), ShieldPiercing(0), SaveCargo(0), Variable(NULL),
+	Indestructible(0), Teleporter(0), ShieldPiercing(0), SaveCargo(0),
+	Wall(0), Variable(NULL),
 	GivesResource(0), Supply(0), Demand(0), FieldFlags(0), MovementMask(0),
 	Sprite(NULL), ShadowSprite(NULL)
 {