using System.Collections.ObjectModel; namespace MultiversalDiplomacy.Model; /// /// The global game state. /// public class World { /// /// The game map. /// public ReadOnlyCollection Provinces { get; } /// /// The game powers. /// public ReadOnlyCollection Powers { get; } /// /// The state of the multiverse. /// public ReadOnlyCollection Seasons { get; } /// /// All units in the multiverse. /// public ReadOnlyCollection Units { get; } /// /// Immutable game options. /// public Options Options { get; } private World( ReadOnlyCollection provinces, ReadOnlyCollection powers, ReadOnlyCollection seasons, ReadOnlyCollection units, Options options) { this.Provinces = provinces; this.Powers = powers; this.Seasons = seasons; this.Units = units; this.Options = options; } /// /// Create a new world with specified provinces and powers. /// public static World WithMap(IEnumerable provinces, IEnumerable powers) => new World( new(provinces.ToList()), new(powers.ToList()), new(new List()), new(new List()), new Options()); /// /// Create a new world with the standard Diplomacy provinces and powers. /// public static World WithStandardMap() => WithMap(StandardProvinces, StandardPowers); /// /// Create a new world with new seasons. /// public World WithSeasons(IEnumerable seasons) => new World(this.Provinces, this.Powers, new(seasons.ToList()), this.Units, this.Options); /// /// Create a new world with an initial season. /// public World WithInitialSeason() => WithSeasons(new List { Season.MakeRoot() }); /// /// Create a new world with new units. /// public World WithUnits(IEnumerable units) => new World(this.Provinces, this.Powers, this.Seasons, new(units.ToList()), this.Options); /// /// Create a new world with new units created from unit specs. Units specs are in the format /// " []". If the province or coast name has a space in it, the /// abbreviation should be used. /// public World WithUnits(params string[] unitSpecs) { IEnumerable units = unitSpecs.Select(spec => { string[] splits = spec.Split(' ', 4); Power power = this.GetPower(splits[0]); UnitType type = splits[1] switch { "A" => UnitType.Army, "F" => UnitType.Fleet, _ => throw new ApplicationException($"Unknown unit type {splits[1]}") }; Location location = type == UnitType.Army ? this.GetLand(splits[2]) : splits.Length == 3 ? this.GetWater(splits[2]) : this.GetWater(splits[2], splits[3]); Unit unit = Unit.Build(location, this.Seasons.First(), power, type); return unit; }); return this.WithUnits(units); } /// /// Create a new world with standard Diplomacy initial unit placements. /// public World WithStandardUnits() { return this.WithUnits( "Austria A Bud", "Austria A Vir", "Austria F Tri", "England A Lvp", "England F Edi", "England F Lon", "France A Mar", "France A Par", "France F Bre", "Germany A Ber", "Germany A Mun", "Germany F Kie", "Italy A Rom", "Italy A Ven", "Italy F Nap", "Russia A Mos", "Russia A War", "Russia F Sev", "Russia F Stp wc", "Turkey A Con", "Turkey A Smy", "Turkey F Ank" ); } /// /// A standard Diplomacy game setup. /// public static World Standard => World .WithStandardMap() .WithInitialSeason() .WithStandardUnits(); /// /// Get a province by name. Throws if the province is not found. /// private Province GetProvince(string provinceName) => GetProvince(provinceName, this.Provinces); /// /// Get a province by name. Throws if the province is not found. /// private static Province GetProvince(string provinceName, IEnumerable provinces) { string provinceNameUpper = provinceName.ToUpperInvariant(); Province? foundProvince = provinces.SingleOrDefault( p => p != null && (p.Name.ToUpperInvariant() == provinceNameUpper || p.Abbreviations.Any(a => a.ToUpperInvariant() == provinceNameUpper)), null); if (foundProvince == null) throw new KeyNotFoundException( $"Province {provinceName} not found"); return foundProvince; } /// /// Get the location in a province matching a predicate. Throws if there is not exactly one /// such location. /// private Location GetLocation(string provinceName, Func predicate) { Location? foundLocation = GetProvince(provinceName).Locations.SingleOrDefault( l => l != null && predicate(l), null); if (foundLocation == null) throw new KeyNotFoundException( $"No such location in {provinceName}"); return foundLocation; } /// /// Get the sole land location of a province. /// public Location GetLand(string provinceName) => GetLocation(provinceName, l => l.Type == LocationType.Land); /// /// Get the sole water location of a province, optionally specifying a named coast. /// public Location GetWater(string provinceName, string? coastName = null) => coastName == null ? GetLocation(provinceName, l => l.Type == LocationType.Water) : GetLocation(provinceName, l => l.Name == coastName || l.Abbreviation == coastName); /// /// Get a power by name. Throws if there is not exactly one such power. /// public Power GetPower(string powerName) { Power? foundPower = this.Powers.SingleOrDefault( p => p != null && (p.Name == powerName || p.Name.StartsWith(powerName)), null); if (foundPower == null) throw new KeyNotFoundException( $"Power {powerName} not found"); return foundPower; } /// /// Returns a unit in a province. Throws if there is not exactly one such unit. /// public Unit? GetUnitAt(string provinceName) { Province province = GetProvince(provinceName); Unit? foundUnit = this.Units.SingleOrDefault( u => u != null && u.Location.Province == province, null); return foundUnit; } /// /// The standard Diplomacy provinces. /// public static ReadOnlyCollection StandardProvinces { get { // Define the provinces of the standard world map. List standardProvinces = new List { Province.Empty("North Africa", "NAF") .AddLandLocation() .AddCoastLocation(), Province.Supply("Tunis", "TUN") .AddLandLocation() .AddCoastLocation(), Province.Empty("Bohemia", "BOH") .AddLandLocation(), Province.Supply("Budapest", "BUD") .AddLandLocation(), Province.Empty("Galacia", "GAL") .AddLandLocation(), Province.Supply("Trieste", "TRI") .AddLandLocation() .AddCoastLocation(), Province.Empty("Tyrolia", "TYR") .AddLandLocation(), Province.Time("Vienna", "VIE") .AddLandLocation(), Province.Empty("Albania", "ALB") .AddLandLocation() .AddCoastLocation(), Province.Supply("Bulgaria", "BUL") .AddLandLocation() .AddCoastLocation("east coast", "ec") .AddCoastLocation("south coast", "sc"), Province.Supply("Greece", "GRE") .AddLandLocation() .AddCoastLocation(), Province.Supply("Rumania", "RUM", "RMA") .AddLandLocation() .AddCoastLocation(), Province.Supply("Serbia", "SER") .AddLandLocation(), Province.Empty("Clyde", "CLY") .AddLandLocation() .AddCoastLocation(), Province.Supply("Edinburgh", "EDI") .AddLandLocation() .AddCoastLocation(), Province.Supply("Liverpool", "LVP", "LPL") .AddLandLocation() .AddCoastLocation(), Province.Time("London", "LON") .AddLandLocation() .AddCoastLocation(), Province.Empty("Wales", "WAL") .AddLandLocation() .AddCoastLocation(), Province.Empty("Yorkshire", "YOR") .AddLandLocation() .AddCoastLocation(), Province.Supply("Brest", "BRE") .AddLandLocation() .AddCoastLocation(), Province.Empty("Burgundy", "BUR") .AddLandLocation(), Province.Empty("Gascony", "GAS") .AddLandLocation() .AddCoastLocation(), Province.Supply("Marseilles", "MAR") .AddLandLocation() .AddCoastLocation(), Province.Time("Paris", "PAR") .AddLandLocation(), Province.Empty("Picardy", "PIC") .AddLandLocation() .AddCoastLocation(), Province.Time("Berlin", "BER") .AddLandLocation() .AddCoastLocation(), Province.Supply("Kiel", "KIE") .AddLandLocation() .AddCoastLocation(), Province.Supply("Munich", "MUN") .AddLandLocation(), Province.Empty("Prussia", "PRU") .AddLandLocation() .AddCoastLocation(), Province.Empty("Ruhr", "RUH", "RHR") .AddLandLocation(), Province.Empty("Silesia", "SIL") .AddLandLocation(), Province.Supply("Spain", "SPA") .AddLandLocation() .AddCoastLocation("north coast", "nc") .AddCoastLocation("south coast", "sc"), Province.Supply("Portugal", "POR") .AddLandLocation() .AddCoastLocation(), Province.Empty("Apulia", "APU") .AddLandLocation() .AddCoastLocation(), Province.Supply("Naples", "NAP") .AddLandLocation() .AddCoastLocation(), Province.Empty("Piedmont", "PIE") .AddLandLocation() .AddCoastLocation(), Province.Time("Rome", "ROM", "RME") .AddLandLocation() .AddCoastLocation(), Province.Empty("Tuscany", "TUS") .AddLandLocation() .AddCoastLocation(), Province.Supply("Venice", "VEN") .AddLandLocation() .AddCoastLocation(), Province.Supply("Belgium", "BEL") .AddLandLocation() .AddCoastLocation(), Province.Supply("Holland", "HOL") .AddLandLocation() .AddCoastLocation(), Province.Empty("Finland", "FIN") .AddLandLocation() .AddCoastLocation(), Province.Empty("Livonia", "LVN", "LVA") .AddLandLocation() .AddCoastLocation(), Province.Time("Moscow", "MOS") .AddLandLocation() .AddCoastLocation(), Province.Supply("Sevastopol", "SEV") .AddLandLocation() .AddCoastLocation(), Province.Supply("Saint Petersburg", "STP") .AddLandLocation() .AddCoastLocation("north coast", "nc") .AddCoastLocation("west coast", "wc"), Province.Empty("Ukraine", "UKR") .AddLandLocation(), Province.Supply("Warsaw", "WAR") .AddLandLocation(), Province.Supply("Denmark", "DEN") .AddLandLocation() .AddCoastLocation(), Province.Supply("Norway", "NWY") .AddLandLocation() .AddCoastLocation(), Province.Supply("Sweden", "SWE") .AddLandLocation() .AddCoastLocation(), Province.Supply("Ankara", "ANK") .AddLandLocation() .AddCoastLocation(), Province.Empty("Armenia", "ARM") .AddLandLocation() .AddCoastLocation(), Province.Time("Constantinople", "CON") .AddLandLocation() .AddCoastLocation(), Province.Supply("Smyrna", "SMY") .AddLandLocation() .AddCoastLocation(), Province.Empty("Syria", "SYR") .AddLandLocation() .AddCoastLocation(), Province.Empty("Barents Sea", "BAR") .AddOceanLocation(), Province.Empty("English Channel", "ENC", "ECH") .AddOceanLocation(), Province.Empty("Heligoland Bight", "HEL", "HGB") .AddOceanLocation(), Province.Empty("Irish Sea", "IRS", "IRI") .AddOceanLocation(), Province.Empty("Mid-Atlantic Ocean", "MAO", "MID") .AddOceanLocation(), Province.Empty("North Atlantic Ocean", "NAO", "NAT") .AddOceanLocation(), Province.Empty("North Sea", "NTH", "NTS") .AddOceanLocation(), Province.Empty("Norwegian Sea", "NWS", "NWG") .AddOceanLocation(), Province.Empty("Skagerrak", "SKA", "SKG") .AddOceanLocation(), Province.Empty("Baltic Sea", "BAL") .AddOceanLocation(), Province.Empty("Guld of Bothnia", "GOB", "BOT") .AddOceanLocation(), Province.Empty("Adriatic Sea", "ADS", "ADR") .AddOceanLocation(), Province.Empty("Aegean Sea", "AEG") .AddOceanLocation(), Province.Empty("Black Sea", "BLA") .AddOceanLocation(), Province.Empty("Eastern Mediterranean Sea", "EMS", "EAS") .AddOceanLocation(), Province.Empty("Gulf of Lyons", "GOL", "LYO") .AddOceanLocation(), Province.Empty("Ionian Sea", "IOS", "ION", "INS") .AddOceanLocation(), Province.Empty("Tyrrhenian Sea", "TYS", "TYN") .AddOceanLocation(), Province.Empty("Western Mediterranean Sea", "WMS", "WES") .AddOceanLocation(), }; // Declare some helpers for border definitions Location Land(string provinceName) => GetProvince(provinceName, standardProvinces) .Locations.Single(l => l.Type == LocationType.Land); Location Water(string provinceName) => GetProvince(provinceName, standardProvinces) .Locations.Single(l => l.Type == LocationType.Water); Location Coast(string provinceName, string coastName) => GetProvince(provinceName, standardProvinces) .Locations.Single(l => l.Name == coastName || l.Abbreviation == coastName); static void AddBordersTo(Location location, Func LocationType, params string[] borders) { foreach (string bordering in borders) { location.AddBorder(LocationType(bordering)); } } void AddBorders(string provinceName, Func LocationType, params string[] borders) => AddBordersTo(LocationType(provinceName), LocationType, borders); AddBorders("NAF", Land, "TUN"); AddBorders("NAF", Water, "MAO", "WES", "TUN"); AddBorders("TUN", Land, "NAF"); AddBorders("TUN", Water, "NAF", "WES", "TYS", "ION"); AddBorders("BOH", Land, "MUN", "SIL", "GAL", "VIE", "TYR"); AddBorders("BUD", Land, "VIE", "GAL", "RUM", "SER", "TRI"); AddBorders("GAL", Land, "BOH", "SIL", "WAR", "UKR", "RUM", "BUD", "VIE"); AddBorders("TRI", Land, "TYR", "VIE", "BUD", "SER", "ALB"); AddBorders("TRI", Water, "ALB", "ADR", "VEN"); AddBorders("TYR", Land, "MUN", "BOH", "VIE", "TRI", "VEN", "PIE"); AddBorders("VIE", Land, "TYR", "BOH", "GAL", "BUD", "TRI"); AddBorders("ALB", Land, "TRI", "SER", "GRE"); AddBorders("ALB", Water, "TRI", "ADR", "ION", "GRE"); AddBorders("BUL", Land, "GRE", "SER", "RUM", "CON"); AddBordersTo(Coast("BUL", "ec"), Water, "BLA", "CON"); AddBordersTo(Coast("BUL", "sc"), Water, "CON", "AEG", "GRE"); AddBorders("GRE", Land, "ALB", "SER", "BUL"); AddBorders("GRE", Water, "ALB", "ION", "AEG"); Water("GRE").AddBorder(Coast("BUL", "sc")); AddBorders("RUM", Land, "BUL", "SER", "BUD", "GAL", "UKR", "SEV"); AddBorders("RUM", Water, "SEV", "BLA"); Water("RUM").AddBorder(Coast("BUL", "ec")); AddBorders("SER", Land, "BUD", "RUM", "BUL", "GRE", "ALB", "TRI"); AddBorders("CLY", Land, "EDI", "LVP"); AddBorders("CLY", Water, "LVP", "NAO", "NWG", "EDI"); AddBorders("EDI", Land, "YOR", "LVP", "CLY"); AddBorders("EDI", Water, "CLY", "NWG", "NTH", "YOR"); AddBorders("LVP", Land, "CLY", "EDI", "YOR", "WAL"); AddBorders("LVP", Water, "WAL", "IRS", "NAO", "CLY"); AddBorders("LON", Land, "WAL", "YOR"); AddBorders("LON", Water, "WAL", "ENC", "NTH", "YOR"); AddBorders("WAL", Land, "LVP", "YOR", "LON"); AddBorders("WAL", Water, "LON", "ENC", "IRS", "LVP"); AddBorders("YOR", Land, "LON", "WAL", "LVP", "EDI"); AddBorders("YOR", Water, "EDI", "NTH", "LON"); AddBorders("BRE", Land, "PIC", "PAR", "GAS"); AddBorders("BRE", Water, "GAS", "MAO", "ENC", "PIC"); AddBorders("BUR", Land, "BEL", "RUH", "MUN", "MAR", "GAS", "PAR", "PIC"); AddBorders("GAS", Land, "BRE", "PAR", "BUR", "MAR", "SPA"); AddBorders("GAS", Water, "MAO", "BRE"); Water("GAS").AddBorder(Coast("SPA", "nc")); AddBorders("MAR", Land, "SPA", "GAS", "BUR", "PIE"); AddBorders("MAR", Water, "LYO", "PIE"); Water("MAR").AddBorder(Coast("SPA", "sc")); AddBorders("PAR", Land, "PIC", "BUR", "GAS", "BRE"); AddBorders("PIC", Land, "BEL", "BUR", "PAR", "BRE"); AddBorders("PIC", Water, "BRE", "ENC", "BEL"); AddBorders("BER", Land, "PRU", "SIL", "MUN", "KIE"); AddBorders("BER", Water, "KIE", "BAL", "PRU"); AddBorders("KIE", Land, "BER", "MUN", "RUH", "HOL", "DEN"); AddBorders("KIE", Water, "HOL", "HEL", "DEN", "BAL", "BER"); AddBorders("MUN", Land, "BUR", "RUH", "KIE", "BER", "SIL", "BOH", "TYR"); AddBorders("PRU", Land, "LVN", "WAR", "SIL", "BER"); AddBorders("PRU", Water, "BER", "BAL", "LVN"); AddBorders("RUH", Land, "KIE", "MUN", "BUR", "BEL", "HOL"); AddBorders("SIL", Land, "PRU", "WAR", "GAL", "BOH", "MUN", "BER"); AddBorders("SPA", Land, "POR", "GAS", "MAR"); AddBordersTo(Coast("SPA", "nc"), Water, "POR", "MAO", "GAS"); AddBordersTo(Coast("SPA", "sc"), Water, "POR", "MAO", "WES", "LYO", "MAR"); AddBorders("POR", Land, "SPA"); AddBorders("POR", Water, "MAO"); Water("POR").AddBorder(Coast("SPA", "nc")); Water("POR").AddBorder(Coast("SPA", "sc")); AddBorders("APU", Land, "NAP", "ROM", "VEN"); AddBorders("APU", Water, "VEN", "ADR", "IOS", "NAP"); AddBorders("NAP", Land, "ROM", "APU"); AddBorders("NAP", Water, "APU", "IOS", "TYS", "ROM"); AddBorders("PIE", Land, "MAR", "TYR", "VEN", "TUS"); AddBorders("PIE", Water, "TUS", "LYO", "MAR"); AddBorders("ROM", Land, "TUS", "VEN", "APU", "NAP"); AddBorders("ROM", Water, "NAP", "TYS", "TUS"); AddBorders("TUS", Land, "PIE", "VEN", "ROM"); AddBorders("TUS", Water, "ROM", "TYS", "LYO", "PIE"); AddBorders("VEN", Land, "APU", "ROM", "TUS", "PIE", "TYR", "TRI"); AddBorders("VEN", Water, "TRI", "ADR", "APU"); AddBorders("BEL", Land, "HOL", "RUH", "BUR", "PIC"); AddBorders("BEL", Water, "PIC", "ENC", "NTH", "HOL"); AddBorders("HOL", Land, "BEL", "RUH", "KIE"); AddBorders("HOL", Water, "NTH", "HEL"); AddBorders("FIN", Land, "SWE", "NWY", "STP"); AddBorders("FIN", Water, "SWE", "BOT"); Water("FIN").AddBorder(Coast("STP", "wc")); AddBorders("LVN", Land, "STP", "MOS", "WAR", "PRU"); AddBorders("LVN", Water, "PRU", "BAL", "BOT"); Water("LVN").AddBorder(Coast("STP", "wc")); AddBorders("MOS", Land, "SEV", "UKR", "WAR", "LVN", "STP"); AddBorders("SEV", Land, "RUM", "UKR", "MOS", "ARM"); AddBorders("SEV", Water, "ARM", "BLA", "RUM"); AddBorders("STP", Land, "MOS", "LVN", "FIN"); AddBordersTo(Coast("STP", "nc"), Water, "BAR", "NWY"); AddBordersTo(Coast("STP", "wc"), Water, "LVN", "BOT", "FIN"); AddBorders("UKR", Land, "MOS", "SEV", "RUM", "GAL", "WAR"); AddBorders("WAR", Land, "PRU", "LVN", "MOS", "UKR", "GAL", "SIL"); AddBorders("DEN", Land, "KIE", "SWE"); AddBorders("DEN", Water, "KIE", "HEL", "NTH", "SKA", "BAL", "SWE"); AddBorders("NWY", Land, "STP", "FIN", "SWE"); AddBorders("NWY", Water, "BAR", "NWG", "NTH", "SKA", "SWE"); Water("NWY").AddBorder(Coast("STP", "nc")); AddBorders("SWE", Land, "NWY", "FIN", "DEN"); AddBorders("SWE", Water, "FIN", "BOT", "BAL", "DEN", "SKA", "NWY"); AddBorders("ANK", Land, "ARM", "SMY", "CON"); AddBorders("ANK", Water, "CON", "BLA", "ARM"); AddBorders("ARM", Land, "SEV", "SYR", "SMY", "ANK"); AddBorders("ARM", Water, "ANK", "BLA", "SEV"); AddBorders("CON", Land, "BUL", "ANK", "SMY"); AddBorders("CON", Water, "BLA", "ANK", "SMY", "AEG"); Water("CON").AddBorder(Coast("BUL", "ec")); Water("CON").AddBorder(Coast("BUL", "sc")); AddBorders("SMY", Land, "CON", "ANK", "ARM", "SYR"); AddBorders("SMY", Water, "SYR", "EAS", "AEG", "CON"); AddBorders("SYR", Land, "SMY", "ARM"); AddBorders("SYR", Water, "EAS", "SMY"); AddBorders("BAR", Water, "NWG", "NWY"); Water("BAR").AddBorder(Coast("STP", "nc")); AddBorders("ENC", Water, "LON", "NTH", "BEL", "PIC", "BRE", "MAO", "IRS", "WAL"); AddBorders("HEL", Water, "NTH", "DEN", "BAL", "KIE", "HOL"); AddBorders("IRS", Water, "NAO", "LVP", "WAL", "ENC", "MAO"); AddBorders("MAO", Water, "NAO", "IRS", "ENC", "BRE", "GAS", "POR", "NAF"); Water("MAO").AddBorder(Coast("SPA", "nc")); Water("MAO").AddBorder(Coast("SPA", "sc")); AddBorders("NAO", Water, "NWG", "CLY", "LVP", "IRS", "MAO"); AddBorders("NTH", Water, "NWG", "NWY", "SKA", "DEN", "HEL", "HOL", "BEL", "ENC", "LON", "YOR", "EDI"); AddBorders("NWG", Water, "BAR", "NWY", "NTH", "EDI", "CLY", "NAO"); AddBorders("SKA", Water, "NWY", "SWE", "BAL", "DEN", "NTH"); AddBorders("BAL", Water, "BOT", "LVN", "PRU", "BER", "KIE", "HEL", "DEN", "SWE"); AddBorders("BOT", Water, "LVN", "BAL", "SWE", "FIN"); Water("BOT").AddBorder(Coast("STP", "wc")); AddBorders("ADR", Water, "IOS", "APU", "VEN", "TRI", "ALB"); AddBorders("AEG", Water, "CON", "SMY", "EAS", "IOS", "GRE"); Water("AEG").AddBorder(Coast("BUL", "sc")); AddBorders("BLA", Water, "RUM", "SEV", "ARM", "ANK", "CON"); Water("BLA").AddBorder(Coast("BUL", "ec")); AddBorders("EAS", Water, "IOS", "AEG", "SMY", "SYR"); AddBorders("LYO", Water, "MAR", "PIE", "TUS", "TYS", "WES"); Water("LYO").AddBorder(Coast("SPA", "sc")); AddBorders("IOS", Water, "TUN", "TYS", "NAP", "APU", "ADR", "ALB", "GRE", "AEG"); AddBorders("TYS", Water, "LYO", "TUS", "ROM", "NAP", "IOS", "TUN", "WES"); AddBorders("WES", Water, "LYO", "TYS", "TUN", "NAF", "MAO"); Water("WES").AddBorder(Coast("SPA", "sc")); return new(standardProvinces); } } /// /// The standard Diplomacy powers. /// public static ReadOnlyCollection StandardPowers { get => new(new List { new Power("Austria"), new Power("England"), new Power("France"), new Power("Germany"), new Power("Italy"), new Power("Russia"), new Power("Turkey"), }); } }