• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

Return-To-The-Roots / s25client / 25985372463

17 May 2026 08:03AM UTC coverage: 50.362% (+0.08%) from 50.284%
25985372463

Pull #1917

github

web-flow
Merge 5d7afa219 into 57b082981
Pull Request #1917: Fix sea path finding (for ships)

118 of 166 new or added lines in 13 files covered. (71.08%)

5 existing lines in 3 files now uncovered.

23215 of 46096 relevant lines covered (50.36%)

47651.7 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

72.18
/libs/s25main/GamePlayer.cpp
1
// Copyright (C) 2005 - 2026 Settlers Freaks (sf-team at siedler25.org)
2
//
3
// SPDX-License-Identifier: GPL-2.0-or-later
4

5
#include "GamePlayer.h"
6
#include "AddonHelperFunctions.h"
7
#include "Cheats.h"
8
#include "EventManager.h"
9
#include "FindWhConditions.h"
10
#include "GameInterface.h"
11
#include "GlobalGameSettings.h"
12
#include "LeatherLoader.h"
13
#include "RoadSegment.h"
14
#include "SerializedGameData.h"
15
#include "TradePathCache.h"
16
#include "Ware.h"
17
#include "WineLoader.h"
18
#include "addons/const_addons.h"
19
#include "buildings/noBuildingSite.h"
20
#include "buildings/nobHQ.h"
21
#include "buildings/nobHarborBuilding.h"
22
#include "buildings/nobMilitary.h"
23
#include "buildings/nobUsual.h"
24
#include "figures/nofCarrier.h"
25
#include "figures/nofFlagWorker.h"
26
#include "helpers/containerUtils.h"
27
#include "helpers/mathFuncs.h"
28
#include "lua/LuaInterfaceGame.h"
29
#include "notifications/ToolNote.h"
30
#include "pathfinding/RoadPathFinder.h"
31
#include "postSystem/DiplomacyPostQuestion.h"
32
#include "postSystem/PostManager.h"
33
#include "random/Random.h"
34
#include "world/GameWorld.h"
35
#include "world/TradeRoute.h"
36
#include "nodeObjs/noFlag.h"
37
#include "nodeObjs/noShip.h"
38
#include "gameTypes/BuildingCount.h"
39
#include "gameTypes/GoodTypes.h"
40
#include "gameTypes/JobTypes.h"
41
#include "gameTypes/PactTypes.h"
42
#include "gameTypes/VisualSettings.h"
43
#include "gameData/BuildingConsts.h"
44
#include "gameData/BuildingProperties.h"
45
#include "gameData/GoodConsts.h"
46
#include "gameData/SettingTypeConv.h"
47
#include "gameData/ShieldConsts.h"
48
#include "gameData/ToolConsts.h"
49
#include "s25util/Log.h"
50
#include <limits>
51
#include <numeric>
52

53
GamePlayer::GamePlayer(unsigned playerId, const PlayerInfo& playerInfo, GameWorld& world)
423✔
54
    : GamePlayerInfo(playerId, playerInfo), world(world), hqPos(MapPoint::Invalid()), emergency(false)
423✔
55
{
56
    std::fill(building_enabled.begin(), building_enabled.end(), true);
423✔
57

58
    LoadStandardDistribution();
423✔
59
    useCustomBuildOrder_ = false;
423✔
60
    build_order = GetStandardBuildOrder();
423✔
61
    transportPrio = STD_TRANSPORT_PRIO;
423✔
62
    LoadStandardMilitarySettings();
423✔
63
    LoadStandardToolSettings();
423✔
64

65
    // Inventur nullen
66
    global_inventory.clear();
423✔
67

68
    // Statistiken mit 0en füllen
69
    statistic = {};
423✔
70
    statisticCurrentData = {};
423✔
71
    statisticCurrentMerchandiseData = {};
423✔
72

73
    RecalcDistribution();
423✔
74
}
423✔
75

76
void GamePlayer::LoadStandardToolSettings()
423✔
77
{
78
    // metalwork tool request
79

80
    // manually
81
    std::fill(tools_ordered.begin(), tools_ordered.end(), 0u);
423✔
82
    std::fill(tools_ordered_delta.begin(), tools_ordered_delta.end(), 0);
423✔
83

84
    // percentage (tool-settings-window-slider, in 10th percent)
85
    toolsSettings_[Tool::Tongs] = 1;
423✔
86
    toolsSettings_[Tool::Hammer] = 4;
423✔
87
    toolsSettings_[Tool::Axe] = 2;
423✔
88
    toolsSettings_[Tool::Saw] = 5;
423✔
89
    toolsSettings_[Tool::PickAxe] = 7;
423✔
90
    toolsSettings_[Tool::Shovel] = 1;
423✔
91
    toolsSettings_[Tool::Crucible] = 3;
423✔
92
    toolsSettings_[Tool::RodAndLine] = 1;
423✔
93
    toolsSettings_[Tool::Scythe] = 2;
423✔
94
    toolsSettings_[Tool::Cleaver] = 1;
423✔
95
    toolsSettings_[Tool::Rollingpin] = 2;
423✔
96
    toolsSettings_[Tool::Bow] = 1;
423✔
97
}
423✔
98

99
void GamePlayer::LoadStandardMilitarySettings()
423✔
100
{
101
    // military settings (military-window-slider, in 10th percent)
102
    militarySettings_[0] = MILITARY_SETTINGS_SCALE[0]; //-V525
423✔
103
    militarySettings_[1] = 3;
423✔
104
    militarySettings_[2] = MILITARY_SETTINGS_SCALE[2];
423✔
105
    militarySettings_[3] = 3;
423✔
106
    militarySettings_[4] = 0;
423✔
107
    militarySettings_[5] = 1;
423✔
108
    militarySettings_[6] = MILITARY_SETTINGS_SCALE[6];
423✔
109
    militarySettings_[7] = MILITARY_SETTINGS_SCALE[7];
423✔
110
}
423✔
111

112
BuildOrders GamePlayer::GetStandardBuildOrder()
425✔
113
{
114
    BuildOrders ordering;
115

116
    // Baureihenfolge füllen
117
    unsigned curPrio = 0;
425✔
118
    for(const auto bld : helpers::enumRange<BuildingType>())
35,700✔
119
    {
120
        if(bld == BuildingType::Headquarters || !BuildingProperties::IsValid(bld))
17,000✔
121
            continue;
850✔
122

123
        RTTR_Assert(curPrio < ordering.size());
16,150✔
124
        ordering[curPrio] = bld;
16,150✔
125
        ++curPrio;
16,150✔
126
    }
127
    RTTR_Assert(curPrio == ordering.size());
425✔
128
    return ordering;
425✔
129
}
130

131
void GamePlayer::LoadStandardDistribution()
423✔
132
{
133
    // Verteilung mit Standardwerten füllen bei Waren mit nur einem Ziel (wie z.B. Mehl, Holz...)
134
    distribution[GoodType::Flour].client_buildings.push_back(BuildingType::Bakery);
423✔
135
    distribution[GoodType::Gold].client_buildings.push_back(BuildingType::Mint);
423✔
136
    distribution[GoodType::IronOre].client_buildings.push_back(BuildingType::Ironsmelter);
423✔
137
    distribution[GoodType::Ham].client_buildings.push_back(BuildingType::Slaughterhouse);
423✔
138
    distribution[GoodType::Stones].client_buildings.push_back(
423✔
139
      BuildingType::Headquarters); // BuildingType::Headquarters = Baustellen!
423✔
140
    distribution[GoodType::Stones].client_buildings.push_back(BuildingType::Catapult);
423✔
141
    distribution[GoodType::Grapes].client_buildings.push_back(BuildingType::Winery);
423✔
142
    distribution[GoodType::Wine].client_buildings.push_back(BuildingType::Temple);
423✔
143
    distribution[GoodType::Skins].client_buildings.push_back(BuildingType::Tannery);
423✔
144
    distribution[GoodType::Leather].client_buildings.push_back(BuildingType::LeatherWorks);
423✔
145

146
    // Waren mit mehreren möglichen Zielen erstmal nullen, kann dann im Fenster eingestellt werden
147
    for(const auto i : helpers::enumRange<GoodType>())
35,532✔
148
    {
149
        std::fill(distribution[i].percent_buildings.begin(), distribution[i].percent_buildings.end(), 0);
16,920✔
150
        distribution[i].selected_goal = 0;
16,920✔
151
    }
152

153
    // Standardverteilung der Waren
154
    for(const DistributionMapping& mapping : distributionMap)
12,690✔
155
    {
156
        distribution[std::get<0>(mapping)].percent_buildings[std::get<1>(mapping)] = std::get<2>(mapping);
12,267✔
157
    }
158
}
423✔
159

160
GamePlayer::~GamePlayer() = default;
854✔
161

162
void GamePlayer::Serialize(SerializedGameData& sgd) const
17✔
163
{
164
    // PlayerStatus speichern, ehemalig
165
    sgd.PushEnum<uint8_t>(ps);
17✔
166

167
    // Nur richtige Spieler serialisieren
168
    if(ps != PlayerState::Occupied && ps != PlayerState::AI)
17✔
169
        return;
3✔
170

171
    sgd.PushBool(isDefeated);
14✔
172

173
    buildings.Serialize(sgd);
14✔
174

175
    sgd.PushObjectContainer(roads, true);
14✔
176

177
    sgd.PushUnsignedInt(jobs_wanted.size());
14✔
178
    for(const JobNeeded& job : jobs_wanted)
20✔
179
    {
180
        sgd.PushEnum<uint8_t>(job.job);
6✔
181
        sgd.PushObject(job.workplace);
6✔
182
    }
183

184
    sgd.PushObjectContainer(ware_list, true);
14✔
185
    sgd.PushObjectContainer(flagworkers);
14✔
186
    sgd.PushObjectContainer(ships, true);
14✔
187

188
    helpers::pushContainer(sgd, shouldSendDefenderList);
14✔
189
    helpers::pushPoint(sgd, hqPos);
14✔
190

191
    for(const Distribution& dist : distribution)
574✔
192
    {
193
        helpers::pushContainer(sgd, dist.percent_buildings);
560✔
194
        helpers::pushContainer(sgd, dist.client_buildings);
560✔
195
        helpers::pushContainer(sgd, dist.goals);
560✔
196
        sgd.PushUnsignedInt(dist.selected_goal);
560✔
197
    }
198

199
    sgd.PushBool(useCustomBuildOrder_);
14✔
200
    helpers::pushContainer(sgd, build_order);
14✔
201
    helpers::pushContainer(sgd, transportPrio);
14✔
202
    helpers::pushContainer(sgd, militarySettings_);
14✔
203
    helpers::pushContainer(sgd, toolsSettings_);
14✔
204
    helpers::pushContainer(sgd, tools_ordered);
14✔
205
    helpers::pushContainer(sgd, global_inventory.goods);
14✔
206
    helpers::pushContainer(sgd, global_inventory.people);
14✔
207
    helpers::pushContainer(sgd, global_inventory.armoredSoldiers);
14✔
208

209
    // für Statistik
210
    for(const Statistic& curStatistic : statistic)
70✔
211
    {
212
        // normale Statistik
213
        for(const auto& curData : curStatistic.data)
560✔
214
            helpers::pushContainer(sgd, curData);
504✔
215

216
        // Warenstatistik
217
        for(unsigned j = 0; j < NUM_STAT_MERCHANDISE_TYPES; ++j)
840✔
218
            helpers::pushContainer(sgd, curStatistic.merchandiseData[j]);
784✔
219

220
        sgd.PushUnsignedShort(curStatistic.currentIndex);
56✔
221
        sgd.PushUnsignedShort(curStatistic.counter);
56✔
222
    }
223
    helpers::pushContainer(sgd, statisticCurrentData);
14✔
224
    helpers::pushContainer(sgd, statisticCurrentMerchandiseData);
14✔
225

226
    // Serialize Pacts:
227
    for(const auto& playerPacts : pacts)
126✔
228
    {
229
        for(const Pact& pact : playerPacts)
336✔
230
            pact.Serialize(sgd);
224✔
231
    }
232

233
    sgd.PushBool(emergency);
14✔
234
}
235

236
void GamePlayer::Deserialize(SerializedGameData& sgd)
8✔
237
{
238
    std::fill(building_enabled.begin(), building_enabled.end(), true);
8✔
239

240
    // Ehemaligen PS auslesen
241
    auto origin_ps = sgd.Pop<PlayerState>();
8✔
242
    // Nur richtige Spieler serialisieren
243
    if(origin_ps != PlayerState::Occupied && origin_ps != PlayerState::AI)
8✔
244
        return;
1✔
245

246
    isDefeated = sgd.PopBool();
7✔
247
    buildings.Deserialize(sgd);
7✔
248

249
    sgd.PopObjectContainer(roads, GO_Type::Roadsegment);
7✔
250

251
    jobs_wanted.resize(sgd.PopUnsignedInt());
7✔
252
    for(JobNeeded& job : jobs_wanted)
10✔
253
    {
254
        job.job = sgd.Pop<Job>();
3✔
255
        job.workplace = sgd.PopObject<noRoadNode>();
3✔
256
    }
257

258
    if(sgd.GetGameDataVersion() < 2)
7✔
259
        buildings.Deserialize2(sgd);
×
260

261
    sgd.PopObjectContainer(ware_list, GO_Type::Ware);
7✔
262
    sgd.PopObjectContainer(flagworkers);
7✔
263
    sgd.PopObjectContainer(ships, GO_Type::Ship);
7✔
264

265
    sgd.PopContainer(shouldSendDefenderList);
7✔
266

267
    hqPos = sgd.PopMapPoint();
7✔
268

269
    for(const auto i : helpers::enumRange<GoodType>())
588✔
270
    {
271
        if(sgd.GetGameDataVersion() < 11 && wineaddon::isWineAddonGoodType(i))
280✔
272
            continue;
×
273

274
        if(sgd.GetGameDataVersion() < 12 && leatheraddon::isLeatherAddonGoodType(i))
280✔
275
            continue;
×
276

277
        Distribution& dist = distribution[i];
280✔
278
        helpers::popContainer(sgd, dist.percent_buildings);
280✔
279
        // Set standard value otherwise its zero and Slaughterhouse never gets ham
280
        // because the ham distribution was not there in earlier versions
281
        if(sgd.GetGameDataVersion() < 12 && i == GoodType::Ham)
280✔
282
            dist.percent_buildings[BuildingType::Slaughterhouse] = 8;
×
283

284
        if(sgd.GetGameDataVersion() < 7)
280✔
285
        {
286
            dist.client_buildings.resize(sgd.PopUnsignedInt());
×
287
            helpers::popContainer(sgd, dist.client_buildings, true);
×
288
            dist.goals.resize(sgd.PopUnsignedInt());
×
289
            helpers::popContainer(sgd, dist.goals, true);
×
290
        } else
291
        {
292
            helpers::popContainer(sgd, dist.client_buildings);
280✔
293
            helpers::popContainer(sgd, dist.goals);
280✔
294
        }
295
        dist.selected_goal = sgd.PopUnsignedInt();
280✔
296
    }
297

298
    useCustomBuildOrder_ = sgd.PopBool();
7✔
299

300
    if(sgd.GetGameDataVersion() < 12)
7✔
301
    {
302
        auto countOfNotAvailableBuildingsInSaveGame =
303
          sgd.GetGameDataVersion() < 11 ? numWineAndLeatherAddonBuildings : numLeatherAddonBuildings;
×
304
        std::vector<BuildingType> build_order_raw(build_order.size() - countOfNotAvailableBuildingsInSaveGame);
×
305
        helpers::popContainer(sgd, build_order_raw, true);
×
306

307
        if(sgd.GetGameDataVersion() < 11)
×
308
        {
309
            build_order_raw.insert(build_order_raw.end(),
×
310
                                   {BuildingType::Vineyard, BuildingType::Winery, BuildingType::Temple});
×
311
        }
312

313
        if(sgd.GetGameDataVersion() < 12)
×
314
        {
315
            build_order_raw.insert(build_order_raw.end(),
×
316
                                   {BuildingType::Skinner, BuildingType::Tannery, BuildingType::LeatherWorks});
×
317
        }
318

319
        std::copy(build_order_raw.begin(), build_order_raw.end(), build_order.begin());
×
320

321
        auto countOfNotAvailableGoodsInSaveGame =
322
          sgd.GetGameDataVersion() < 11 ? numWineAndLeatherAddonGoods : numLeatherAddonGoods;
×
323
        std::vector<uint8_t> transportPrio_raw(transportPrio.size() - countOfNotAvailableGoodsInSaveGame);
×
324
        helpers::popContainer(sgd, transportPrio_raw, true);
×
325
        std::copy(transportPrio_raw.begin(), transportPrio_raw.end(), transportPrio.begin());
×
326
        std::transform(transportPrio.begin(), transportPrio.end() - countOfNotAvailableGoodsInSaveGame,
×
327
                       transportPrio.begin(),
328
                       [](uint8_t& prio) { return prio < transportPrioOfLeatherworks ? prio : prio + 1; });
×
329
    } else
330
    {
331
        helpers::popContainer(sgd, build_order);
7✔
332
        helpers::popContainer(sgd, transportPrio);
7✔
333
    }
334

335
    helpers::popContainer(sgd, militarySettings_);
7✔
336
    helpers::popContainer(sgd, toolsSettings_);
7✔
337

338
    // qx:tools
339
    helpers::popContainer(sgd, tools_ordered);
7✔
340
    tools_ordered_delta = {};
7✔
341

342
    if(sgd.GetGameDataVersion() < 12)
7✔
343
    {
344
        auto countOfNotAvailableGoodsInSaveGame =
345
          sgd.GetGameDataVersion() < 11 ? numWineAndLeatherAddonGoods : numLeatherAddonGoods;
×
346
        std::vector<unsigned int> global_inventory_good_raw(global_inventory.goods.size()
×
347
                                                            - countOfNotAvailableGoodsInSaveGame);
×
348
        helpers::popContainer(sgd, global_inventory_good_raw, true);
×
349
        std::copy(global_inventory_good_raw.begin(), global_inventory_good_raw.end(), global_inventory.goods.begin());
×
350

351
        auto countOfNotAvailableJobsInSaveGame =
352
          sgd.GetGameDataVersion() < 11 ? numWineAndLeatherAddonJobs : numLeatherAddonJobs;
×
353
        std::vector<unsigned int> global_inventory_people_raw(global_inventory.people.size()
×
354
                                                              - countOfNotAvailableJobsInSaveGame);
×
355
        helpers::popContainer(sgd, global_inventory_people_raw, true);
×
356
        std::copy(global_inventory_people_raw.begin(), global_inventory_people_raw.end(),
×
357
                  global_inventory.people.begin());
358

359
        std::fill(global_inventory.armoredSoldiers.begin(), global_inventory.armoredSoldiers.end(), 0);
×
360
    } else
361
    {
362
        helpers::popContainer(sgd, global_inventory.goods);
7✔
363
        helpers::popContainer(sgd, global_inventory.people);
7✔
364
        helpers::popContainer(sgd, global_inventory.armoredSoldiers);
7✔
365
    }
366

367
    // Visuelle Einstellungen festlegen
368

369
    // für Statistik
370
    for(Statistic& curStatistic : statistic)
35✔
371
    {
372
        // normale Statistik
373
        for(auto& curData : curStatistic.data)
280✔
374
            helpers::popContainer(sgd, curData);
252✔
375

376
        // Warenstatistik
377
        for(unsigned j = 0; j < NUM_STAT_MERCHANDISE_TYPES; ++j)
420✔
378
            helpers::popContainer(sgd, curStatistic.merchandiseData[j]);
392✔
379

380
        curStatistic.currentIndex = sgd.PopUnsignedShort();
28✔
381
        curStatistic.counter = sgd.PopUnsignedShort();
28✔
382
    }
383
    helpers::popContainer(sgd, statisticCurrentData);
7✔
384
    helpers::popContainer(sgd, statisticCurrentMerchandiseData);
7✔
385

386
    // Deserialize Pacts:
387
    for(auto& playerPacts : pacts)
63✔
388
    {
389
        for(Pact& pact : playerPacts)
168✔
390
            pact = GamePlayer::Pact(sgd);
112✔
391
    }
392

393
    emergency = sgd.PopBool();
7✔
394
}
395

396
template<class T_IsWarehouseGood>
397
nobBaseWarehouse* GamePlayer::FindWarehouse(const noRoadNode& start, const T_IsWarehouseGood& isWarehouseGood,
4,728✔
398
                                            bool to_wh, bool use_boat_roads, unsigned* length,
399
                                            const RoadSegment* forbidden) const
400
{
401
    nobBaseWarehouse* best = nullptr;
4,728✔
402

403
    unsigned best_length = std::numeric_limits<unsigned>::max();
4,728✔
404

405
    for(nobBaseWarehouse* wh : buildings.GetStorehouses())
15,751✔
406
    {
407
        // Lagerhaus geeignet?
408
        RTTR_Assert(wh);
11,028✔
409
        if(!isWarehouseGood(*wh))
11,028✔
410
            continue;
10,566✔
411

412
        if(start.GetPos() == wh->GetPos())
476✔
413
        {
414
            // We are already there -> Take it
415
            if(length)
5✔
416
                *length = 0;
×
417
            return wh;
5✔
418
        }
419

420
        // now check if there is at least a chance that the next wh is closer than current best because pathfinding
421
        // takes time
422
        if(world.CalcDistance(start.GetPos(), wh->GetPos()) > best_length)
471✔
423
            continue;
14✔
424
        // Bei der erlaubten Benutzung von Bootsstraßen Waren-Pathfinding benutzen wenns zu nem Lagerhaus gehn soll
425
        // start <-> ziel tauschen bei der wegfindung
426
        unsigned tlength;
427
        if(world.GetRoadPathFinder().FindPath(to_wh ? start : *wh, to_wh ? *wh : start, use_boat_roads, best_length,
457✔
428
                                              forbidden, &tlength))
429
        {
430
            if(tlength < best_length || !best)
368✔
431
            {
432
                best_length = tlength;
364✔
433
                best = wh;
364✔
434
            }
435
        }
436
    }
437

438
    if(length)
4,723✔
439
        *length = best_length;
538✔
440

441
    return best;
4,723✔
442
}
443

444
void GamePlayer::AddBuildingSite(noBuildingSite* bldSite)
28✔
445
{
446
    RTTR_Assert(bldSite->GetPlayer() == GetPlayerId());
28✔
447
    buildings.Add(bldSite);
28✔
448
}
28✔
449

450
void GamePlayer::RemoveBuildingSite(noBuildingSite* bldSite)
12✔
451
{
452
    RTTR_Assert(bldSite->GetPlayer() == GetPlayerId());
12✔
453
    buildings.Remove(bldSite);
12✔
454
}
12✔
455

456
bool GamePlayer::IsHQTent() const
8✔
457
{
458
    if(const nobHQ* hq = GetHQ())
8✔
459
        return hq->IsTent();
8✔
460
    return false;
×
461
}
462

463
void GamePlayer::SetHQIsTent(bool isTent)
6✔
464
{
465
    if(nobHQ* hq = GetHQ())
6✔
466
        hq->SetIsTent(isTent);
5✔
467
}
6✔
468

469
void GamePlayer::AddBuilding(noBuilding* bld, BuildingType bldType)
676✔
470
{
471
    RTTR_Assert(bld->GetPlayer() == GetPlayerId());
676✔
472
    buildings.Add(bld, bldType);
676✔
473
    ChangeStatisticValue(StatisticType::Buildings, 1);
676✔
474

475
    // Order a worker if needed
476
    const auto& description = BLD_WORK_DESC[bldType];
676✔
477
    if(description.job && !isSoldierJob(*description.job))
676✔
478
    {
479
        AddJobWanted(*description.job, bld);
110✔
480
    }
481

482
    if(bldType == BuildingType::HarborBuilding)
676✔
483
    {
484
        // Schiff durchgehen und denen Bescheid sagen
485
        for(noShip* ship : ships)
62✔
486
            ship->NewHarborBuilt(static_cast<nobHarborBuilding*>(bld));
13✔
487
    } else if(bldType == BuildingType::Headquarters)
627✔
488
    {
489
        // If there is more than one HQ, keep the original position.
490
        if(!hqPos.isValid())
378✔
491
            hqPos = bld->GetPos();
369✔
492
    } else if(BuildingProperties::IsMilitary(bldType))
249✔
493
    {
494
        auto* milBld = static_cast<nobMilitary*>(bld);
128✔
495
        // New built? -> Calculate frontier distance
496
        if(milBld->IsNewBuilt())
128✔
497
            milBld->LookForEnemyBuildings();
118✔
498
    }
499
}
676✔
500

501
void GamePlayer::RemoveBuilding(noBuilding* bld, BuildingType bldType)
117✔
502
{
503
    RTTR_Assert(bld->GetPlayer() == GetPlayerId());
117✔
504
    buildings.Remove(bld, bldType);
117✔
505
    ChangeStatisticValue(StatisticType::Buildings, -1);
117✔
506
    if(bldType == BuildingType::HarborBuilding)
117✔
507
    { // Schiffen Bescheid sagen
508
        for(noShip* ship : ships)
31✔
509
            ship->HarborDestroyed(static_cast<nobHarborBuilding*>(bld));
14✔
510
    } else if(bldType == BuildingType::Headquarters && bld->GetPos() == hqPos)
100✔
511
    {
512
        hqPos = MapPoint::Invalid();
30✔
513
        for(const noBaseBuilding* bld : buildings.GetStorehouses())
30✔
514
        {
515
            if(bld->GetBuildingType() == BuildingType::Headquarters)
×
516
            {
517
                hqPos = bld->GetPos();
×
518
                break;
×
519
            }
520
        }
521
    }
522
    if(BuildingProperties::IsWareHouse(bldType) || BuildingProperties::IsMilitary(bldType))
117✔
523
        TestDefeat();
84✔
524
}
117✔
525

526
void GamePlayer::NewRoadConnection(RoadSegment* rs)
160✔
527
{
528
    // Zu den Straßen hinzufgen, da's ja ne neue ist
529
    roads.push_back(rs);
160✔
530

531
    // Alle Straßen müssen nun gucken, ob sie einen Weg zu einem Warehouse finden
532
    FindCarrierForAllRoads();
160✔
533

534
    // Alle Straßen müssen gucken, ob sie einen Esel bekommen können
535
    for(RoadSegment* rs : roads)
347✔
536
        rs->TryGetDonkey();
187✔
537

538
    // Alle Arbeitsplätze müssen nun gucken, ob sie einen Weg zu einem Lagerhaus mit entsprechender Arbeitskraft finden
539
    FindWarehouseForAllJobs();
160✔
540

541
    // Alle Baustellen müssen nun gucken, ob sie ihr benötigtes Baumaterial bekommen (evtl war vorher die Straße zum
542
    // Lagerhaus unterbrochen
543
    FindMaterialForBuildingSites();
160✔
544

545
    // Alle Lost-Wares müssen gucken, ob sie ein Lagerhaus finden
546
    FindClientForLostWares();
160✔
547

548
    // Alle Militärgebäude müssen ihre Truppen überprüfen und können nun ggf. neue bestellen
549
    // und müssen prüfen, ob sie evtl Gold bekommen
550
    for(nobMilitary* mil : buildings.GetMilitaryBuildings())
178✔
551
    {
552
        mil->RegulateTroops();
18✔
553
        mil->SearchCoins();
18✔
554
    }
555
}
160✔
556

557
void GamePlayer::AddRoad(RoadSegment* rs)
23✔
558
{
559
    roads.push_back(rs);
23✔
560
}
23✔
561

562
void GamePlayer::DeleteRoad(RoadSegment* rs)
103✔
563
{
564
    RTTR_Assert(helpers::contains(roads, rs));
103✔
565
    roads.remove(rs);
103✔
566
}
103✔
567

568
void GamePlayer::FindClientForLostWares()
166✔
569
{
570
    // Alle Lost-Wares müssen gucken, ob sie ein Lagerhaus finden
571
    for(Ware* ware : ware_list)
176✔
572
    {
573
        if(ware->IsLostWare())
10✔
574
        {
575
            if(ware->FindRouteToWarehouse() && ware->IsWaitingAtFlag())
×
576
                ware->CallCarrier();
×
577
        }
578
    }
579
}
166✔
580

581
void GamePlayer::RoadDestroyed()
226✔
582
{
583
    // Alle Waren, die an Flagge liegen und in Lagerhäusern, müssen gucken, ob sie ihr Ziel noch erreichen können, jetzt
584
    // wo eine Straße fehlt
585
    for(auto it = ware_list.begin(); it != ware_list.end();)
238✔
586
    {
587
        Ware* ware = *it;
12✔
588
        if(ware->IsWaitingAtFlag()) // Liegt die Flagge an einer Flagge, muss ihr Weg neu berechnet werden
12✔
589
        {
590
            RoadPathDirection last_next_dir = ware->GetNextDir();
2✔
591
            ware->RecalcRoute();
2✔
592
            // special case: ware was lost some time ago and the new goal is at this flag and not a warehouse,hq,harbor
593
            // and the "flip-route" picked so a carrier would pick up the ware carry it away from goal then back and
594
            // drop  it off at the goal was just destroyed?
595
            // -> try to pick another flip route or tell the goal about failure.
596
            noRoadNode& wareLocation = *ware->GetLocation();
2✔
597
            noBaseBuilding* wareGoal = ware->GetGoal();
2✔
598
            if(wareGoal && ware->GetNextDir() == RoadPathDirection::NorthWest
2✔
599
               && wareLocation.GetPos() == wareGoal->GetFlagPos()
2✔
600
               && ((wareGoal->GetBuildingType() != BuildingType::Storehouse
6✔
601
                    && wareGoal->GetBuildingType() != BuildingType::Headquarters
2✔
602
                    && wareGoal->GetBuildingType() != BuildingType::HarborBuilding)
×
603
                   || wareGoal->GetType() == NodalObjectType::Buildingsite))
2✔
604
            {
605
                Direction newWareDir = Direction::NorthWest;
×
606
                for(auto dir : helpers::EnumRange<Direction>{})
×
607
                {
608
                    dir += 2u; // Need to skip Direction::NorthWest and we used to start with an offset of 2. TODO:
×
609
                               // Increase gameDataVersion and just skip NW
610
                    if(wareLocation.GetRoute(dir))
×
611
                    {
612
                        newWareDir = dir;
×
613
                        break;
×
614
                    }
615
                }
616
                if(newWareDir != Direction::NorthWest)
×
617
                {
618
                    ware->SetNextDir(toRoadPathDirection(newWareDir));
×
619
                } else // no route to goal -> notify goal, try to send ware to a warehouse
620
                {
621
                    ware->NotifyGoalAboutLostWare();
×
622
                    ware->FindRouteToWarehouse();
×
623
                }
624
            }
625
            // end of special case
626

627
            // notify carriers/flags about news if there are any
628
            if(ware->GetNextDir() != last_next_dir)
2✔
629
            {
630
                // notify current flag that transport in the old direction might not longer be required
631
                ware->RemoveWareJobForDir(last_next_dir);
×
632
                if(ware->GetNextDir() != RoadPathDirection::None)
×
633
                    ware->CallCarrier();
×
634
            }
635
        } else if(ware->IsWaitingInWarehouse())
10✔
636
        {
637
            if(!ware->IsRouteToGoal())
×
638
            {
639
                // Das Ziel wird nun nich mehr beliefert
640
                ware->NotifyGoalAboutLostWare();
×
641
                // Ware aus der Warteliste des Lagerhauses entfernen
642
                static_cast<nobBaseWarehouse*>(ware->GetLocation())->CancelWare(ware);
×
643
                // Ware aus der Liste raus
644
                it = ware_list.erase(it);
×
645
                continue;
×
646
            }
647
        } else if(ware->IsWaitingForShip())
10✔
648
        {
649
            // Weg neu berechnen
650
            ware->RecalcRoute();
1✔
651
        }
652

653
        ++it;
12✔
654
    }
655

656
    // Alle Häfen müssen ihre Figuren den Weg überprüfen lassen
657
    for(nobHarborBuilding* hb : buildings.GetHarbors())
246✔
658
    {
659
        hb->ExamineShipRouteOfPeople();
20✔
660
    }
661
}
226✔
662

663
bool GamePlayer::FindCarrierForRoad(RoadSegment& rs) const
268✔
664
{
665
    RTTR_Assert(rs.GetF1() != nullptr && rs.GetF2() != nullptr);
268✔
666
    std::array<unsigned, 2> length;
667
    std::array<nobBaseWarehouse*, 2> best;
668

669
    // Braucht der ein Boot?
670
    if(rs.GetRoadType() == RoadType::Water)
268✔
671
    {
672
        // dann braucht man Träger UND Boot
673
        best[0] = FindWarehouse(*rs.GetF1(), FW::HasWareAndFigure(GoodType::Boat, Job::Helper, false), false, false,
×
674
                                length.data(), &rs);
675
        // 2. Flagge des Weges
676
        best[1] = FindWarehouse(*rs.GetF2(), FW::HasWareAndFigure(GoodType::Boat, Job::Helper, false), false, false,
×
677
                                &length[1], &rs);
×
678
    } else
679
    {
680
        // 1. Flagge des Weges
681
        best[0] = FindWarehouse(*rs.GetF1(), FW::HasFigure(Job::Helper, false), false, false, length.data(), &rs);
268✔
682
        // 2. Flagge des Weges
683
        best[1] = FindWarehouse(*rs.GetF2(), FW::HasFigure(Job::Helper, false), false, false, &length[1], &rs);
268✔
684
    }
685

686
    // überhaupt nen Weg gefunden?
687
    // Welche Flagge benutzen?
688
    if(best[0] && (!best[1] || length[0] < length[1]))
268✔
689
        best[0]->OrderCarrier(*rs.GetF1(), rs);
28✔
690
    else if(best[1])
240✔
691
        best[1]->OrderCarrier(*rs.GetF2(), rs);
35✔
692
    else
693
        return false;
205✔
694
    return true;
63✔
695
}
696

697
bool GamePlayer::IsWarehouseValid(nobBaseWarehouse* wh) const
×
698
{
699
    return helpers::contains(buildings.GetStorehouses(), wh);
×
700
}
701

702
void GamePlayer::RecalcDistribution()
429✔
703
{
704
    GoodType lastWare = GoodType::Nothing;
429✔
705
    for(const DistributionMapping& mapping : distributionMap)
12,870✔
706
    {
707
        if(lastWare == std::get<0>(mapping))
12,441✔
708
            continue;
9,009✔
709
        lastWare = std::get<0>(mapping);
3,432✔
710
        RecalcDistributionOfWare(std::get<0>(mapping));
3,432✔
711
    }
712
}
429✔
713

714
void GamePlayer::RecalcDistributionOfWare(const GoodType ware)
3,432✔
715
{
716
    // Punktesystem zur Verteilung, in der Liste alle Gebäude sammeln, die die Ware wollen
717
    distribution[ware].client_buildings.clear();
3,432✔
718

719
    // 1. Anteile der einzelnen Waren ausrechnen
720

721
    /// Mapping of buildings that want the current ware to its percentage
722
    using BldEntry = std::pair<BuildingType, uint8_t>;
723
    std::vector<BldEntry> bldPercentageMap;
3,432✔
724

725
    unsigned goal_count = 0;
3,432✔
726

727
    for(const auto bld : helpers::enumRange<BuildingType>())
288,288✔
728
    {
729
        uint8_t percentForCurBld = distribution[ware].percent_buildings[bld];
137,280✔
730
        if(percentForCurBld)
137,280✔
731
        {
732
            distribution[ware].client_buildings.push_back(bld);
12,441✔
733
            goal_count += percentForCurBld;
12,441✔
734
            bldPercentageMap.emplace_back(bld, percentForCurBld);
12,441✔
735
        }
736
    }
737

738
    // TODO: evtl noch die counts miteinander kürzen (ggt berechnen)
739

740
    // Array für die Gebäudtypen erstellen
741

742
    std::vector<BuildingType>& wareGoals = distribution[ware].goals;
3,432✔
743
    wareGoals.clear();
3,432✔
744
    wareGoals.reserve(goal_count);
3,432✔
745

746
    // just drop them in the list, the distribution will be handled by going through this list using a prime as step
747
    // (see GameClientPlayer::FindClientForWare)
748
    for(const BldEntry& bldEntry : bldPercentageMap)
15,873✔
749
    {
750
        for(unsigned char i = 0; i < bldEntry.second; ++i)
81,405✔
751
            wareGoals.push_back(bldEntry.first);
68,964✔
752
    }
753

754
    distribution[ware].selected_goal = 0;
3,432✔
755
}
3,432✔
756

757
void GamePlayer::FindCarrierForAllRoads()
272✔
758
{
759
    for(RoadSegment* rs : roads)
629✔
760
    {
761
        if(!rs->hasCarrier(0))
357✔
762
            FindCarrierForRoad(*rs);
240✔
763
    }
764
}
272✔
765

766
void GamePlayer::FindMaterialForBuildingSites()
311✔
767
{
768
    for(noBuildingSite* bldSite : buildings.GetBuildingSites())
335✔
769
        bldSite->OrderConstructionMaterial();
24✔
770
}
311✔
771

772
void GamePlayer::AddJobWanted(const Job job, noRoadNode* workplace)
146✔
773
{
774
    if(!FindWarehouseForJob(job, *workplace))
146✔
775
    {
776
        JobNeeded jn = {job, workplace};
136✔
777
        jobs_wanted.push_back(jn);
136✔
778
    }
779
}
146✔
780

781
void GamePlayer::JobNotWanted(noRoadNode* workplace, bool all)
58✔
782
{
783
    for(auto it = jobs_wanted.begin(); it != jobs_wanted.end();)
62✔
784
    {
785
        if(it->workplace == workplace)
45✔
786
        {
787
            it = jobs_wanted.erase(it);
41✔
788
            if(!all)
41✔
789
                return;
41✔
790
        } else
791
        {
792
            ++it;
4✔
793
        }
794
    }
795
}
796

797
void GamePlayer::OneJobNotWanted(const Job job, noRoadNode* workplace)
2✔
798
{
799
    const auto it = helpers::find_if(
800
      jobs_wanted, [workplace, job](const auto& it) { return it.workplace == workplace && it.job == job; });
2✔
801
    if(it != jobs_wanted.end())
2✔
802
        jobs_wanted.erase(it);
×
803
}
2✔
804

805
void GamePlayer::SendPostMessage(std::unique_ptr<PostMsg> msg)
35✔
806
{
807
    world.GetPostMgr().SendMsg(GetPlayerId(), std::move(msg));
35✔
808
}
35✔
809

810
unsigned GamePlayer::GetToolsOrderedVisual(Tool tool) const
×
811
{
812
    return std::max(0, int(tools_ordered[tool] + tools_ordered_delta[tool]));
×
813
}
814

815
unsigned GamePlayer::GetToolsOrdered(Tool tool) const
369✔
816
{
817
    return tools_ordered[tool];
369✔
818
}
819

820
bool GamePlayer::ChangeToolOrderVisual(Tool tool, int changeAmount) const
×
821
{
822
    if(std::abs(changeAmount) > 100)
×
823
        return false;
×
824
    int newOrderAmount = int(GetToolsOrderedVisual(tool)) + changeAmount;
×
825
    if(newOrderAmount < 0 || newOrderAmount > 100)
×
826
        return false;
×
827
    tools_ordered_delta[tool] += changeAmount;
×
828
    return true;
×
829
}
830

831
unsigned GamePlayer::GetToolPriority(Tool tool) const
271✔
832
{
833
    return toolsSettings_[tool];
271✔
834
}
835

836
void GamePlayer::ToolOrderProcessed(Tool tool)
4✔
837
{
838
    if(tools_ordered[tool])
4✔
839
    {
840
        --tools_ordered[tool];
4✔
841
        world.GetNotifications().publish(ToolNote(ToolNote::OrderCompleted, GetPlayerId()));
4✔
842
    }
843
}
4✔
844

845
bool GamePlayer::FindWarehouseForJob(const Job job, noRoadNode& goal) const
221✔
846
{
847
    // Optimization: return early if building is isolated
848
    if(goal.GetType() == NodalObjectType::Building || goal.GetType() == NodalObjectType::Buildingsite)
221✔
849
    {
850
        if(!static_cast<noBaseBuilding&>(goal).IsConnected())
221✔
851
            return false;
136✔
852
    }
853

854
    nobBaseWarehouse* wh = FindWarehouse(goal, FW::HasFigure(job, true), false, false);
85✔
855

856
    if(wh)
85✔
857
    {
858
        // Es wurde ein Lagerhaus gefunden, wo es den geforderten Beruf gibt, also den Typen zur Arbeit rufen
859
        wh->OrderJob(job, goal, true);
18✔
860
        return true;
18✔
861
    }
862

863
    return false;
67✔
864
}
865

866
void GamePlayer::FindWarehouseForAllJobs()
271✔
867
{
868
    for(auto it = jobs_wanted.begin(); it != jobs_wanted.end();)
346✔
869
    {
870
        if(FindWarehouseForJob(it->job, *it->workplace))
75✔
871
            it = jobs_wanted.erase(it);
8✔
872
        else
873
            ++it;
67✔
874
    }
875
}
271✔
876

877
void GamePlayer::FindWarehouseForAllJobs(const Job job)
2,673✔
878
{
879
    for(auto it = jobs_wanted.begin(); it != jobs_wanted.end();)
2,673✔
880
    {
881
        if(it->job == job)
×
882
        {
883
            if(FindWarehouseForJob(it->job, *it->workplace))
×
884
                it = jobs_wanted.erase(it);
×
885
            else
886
                ++it;
×
887
        } else
888
            ++it;
×
889
    }
890
}
2,673✔
891

892
Ware* GamePlayer::OrderWare(const GoodType ware, noBaseBuilding& goal)
112✔
893
{
894
    /// Gibt es ein Lagerhaus mit dieser Ware?
895
    nobBaseWarehouse* wh = FindWarehouse(goal, FW::HasMinWares(ware, 1), false, true);
112✔
896

897
    if(wh)
112✔
898
    {
899
        // Prüfe ob Notfallprogramm aktiv
900
        if(!emergency)
84✔
901
            return wh->OrderWare(ware, goal);
84✔
902
        else
903
        {
904
            // Wenn Notfallprogramm aktiv nur an Holzfäller und Sägewerke Bretter/Steine liefern
905
            if((ware != GoodType::Boards && ware != GoodType::Stones)
×
906
               || goal.GetBuildingType() == BuildingType::Woodcutter || goal.GetBuildingType() == BuildingType::Sawmill)
×
907
                return wh->OrderWare(ware, goal);
×
908
            else
909
                return nullptr;
×
910
        }
911
    } else // no warehouse can deliver the ware -> check all our wares for lost wares that might match the order
912
    {
913
        unsigned bestLength = std::numeric_limits<unsigned>::max();
28✔
914
        Ware* bestWare = nullptr;
28✔
915
        for(Ware* curWare : ware_list)
28✔
916
        {
917
            if(curWare->IsLostWare() && curWare->type == ware)
×
918
            {
919
                // got a lost ware with a road to goal -> find best
920
                unsigned curLength = curWare->CheckNewGoalForLostWare(goal);
×
921
                if(curLength < bestLength)
×
922
                {
923
                    bestLength = curLength;
×
924
                    bestWare = curWare;
×
925
                }
926
            }
927
        }
928
        if(bestWare)
28✔
929
        {
930
            bestWare->SetNewGoalForLostWare(goal);
×
931
            return bestWare;
×
932
        }
933
    }
934
    return nullptr;
28✔
935
}
936

937
nofCarrier* GamePlayer::OrderDonkey(RoadSegment& road) const
1✔
938
{
939
    std::array<unsigned, 2> length;
940
    std::array<nobBaseWarehouse*, 2> best;
941

942
    // 1. Flagge des Weges
943
    best[0] = FindWarehouse(*road.GetF1(), FW::HasFigure(Job::PackDonkey, false), false, false, length.data(), &road);
1✔
944
    // 2. Flagge des Weges
945
    best[1] = FindWarehouse(*road.GetF2(), FW::HasFigure(Job::PackDonkey, false), false, false, &length[1], &road);
1✔
946

947
    // überhaupt nen Weg gefunden?
948
    // Welche Flagge benutzen?
949
    if(best[0] && (!best[1] || length[0] < length[1]))
1✔
950
        return best[0]->OrderDonkey(road, *road.GetF1());
×
951
    else if(best[1])
1✔
952
        return best[1]->OrderDonkey(road, *road.GetF2());
×
953
    else
954
        return nullptr;
1✔
955
}
956

957
RoadSegment* GamePlayer::FindRoadForDonkey(noRoadNode& start, noRoadNode** goal)
75✔
958
{
959
    // Bisher höchste Trägerproduktivität und die entsprechende Straße dazu
960
    unsigned best_productivity = 0;
75✔
961
    RoadSegment* best_road = nullptr;
75✔
962
    // Beste Flagge dieser Straße
963
    *goal = nullptr;
75✔
964

965
    for(RoadSegment* roadSeg : roads)
88✔
966
    {
967
        // Braucht die Straße einen Esel?
968
        if(roadSeg->NeedDonkey())
13✔
969
        {
970
            // Beste Flagge von diesem Weg, und beste Wegstrecke
971
            noRoadNode* current_best_goal = nullptr;
×
972
            // Weg zu beiden Flaggen berechnen
973
            unsigned length1, length2;
974
            bool isF1Reachable = world.FindHumanPathOnRoads(start, *roadSeg->GetF1(), &length1, nullptr, roadSeg)
×
975
                                 != RoadPathDirection::None;
×
976
            bool isF2Reachable = world.FindHumanPathOnRoads(start, *roadSeg->GetF2(), &length2, nullptr, roadSeg)
×
977
                                 != RoadPathDirection::None;
×
978

979
            // Wenn man zu einer Flagge nich kommt, die jeweils andere nehmen
980
            if(!isF1Reachable)
×
981
                current_best_goal = (isF2Reachable) ? roadSeg->GetF2() : nullptr;
×
982
            else if(!isF2Reachable)
×
983
                current_best_goal = roadSeg->GetF1();
×
984
            else
985
            {
986
                // ansonsten die kürzeste von beiden
987
                current_best_goal = (length1 < length2) ? roadSeg->GetF1() : roadSeg->GetF2();
×
988
            }
989

990
            // Kein Weg führt hin, nächste Straße bitte
991
            if(!current_best_goal)
×
992
                continue;
×
993

994
            // Jeweiligen Weg bestimmen
995
            unsigned current_best_way = (roadSeg->GetF1() == current_best_goal) ? length1 : length2;
×
996

997
            // Produktivität ausrechnen, *10 die Produktivität + die Wegstrecke, damit die
998
            // auch noch mit einberechnet wird
999
            unsigned current_productivity = 10 * roadSeg->getCarrier(0)->GetProductivity() + current_best_way;
×
1000

1001
            // Besser als der bisher beste?
1002
            if(current_productivity > best_productivity)
×
1003
            {
1004
                // Dann wird der vom Thron gestoßen
1005
                best_productivity = current_productivity;
×
1006
                best_road = roadSeg;
×
1007
                *goal = current_best_goal;
×
1008
            }
1009
        }
1010
    }
1011

1012
    return best_road;
75✔
1013
}
1014

1015
struct ClientForWare
1016
{
1017
    noBaseBuilding* bld;
1018
    unsigned estimate; // points minus half the optimal distance
1019
    unsigned points;
1020

1021
    ClientForWare(noBaseBuilding* bld, unsigned estimate, unsigned points)
×
1022
        : bld(bld), estimate(estimate), points(points)
×
1023
    {}
×
1024

1025
    bool operator<(const ClientForWare& b) const noexcept
×
1026
    {
1027
        // use estimate, points and object id (as tie breaker) for sorting
1028
        if(estimate != b.estimate)
×
1029
            return estimate > b.estimate;
×
1030
        else if(points != b.points)
×
1031
            return points > b.points;
×
1032
        else
1033
            return bld->GetObjId() > b.bld->GetObjId();
×
1034
    }
1035
};
1036

1037
noBaseBuilding* GamePlayer::FindClientForWare(const Ware& ware)
12✔
1038
{
1039
    // If the ware is a coin or an armor, the goal is determined by another logic
1040
    if(ware.type == GoodType::Coins)
12✔
1041
        return FindClientForCoin(ware);
×
1042
    else if(ware.type == GoodType::Armor)
12✔
1043
        return FindClientForArmor(ware);
×
1044

1045
    // Warentyp herausfinden
1046
    GoodType gt = ware.type;
12✔
1047
    // All food is considered fish in the distribution table
1048
    Distribution& wareDistribution =
1049
      (gt == GoodType::Bread || gt == GoodType::Meat) ? distribution[GoodType::Fish] : distribution[gt];
12✔
1050

1051
    std::vector<ClientForWare> possibleClients;
12✔
1052

1053
    const noRoadNode* start = ware.GetLocation();
12✔
1054

1055
    // Bretter und Steine können evtl. auch Häfen für Expeditionen gebrauchen
1056
    if(gt == GoodType::Stones || gt == GoodType::Boards)
12✔
1057
    {
1058
        for(nobHarborBuilding* harbor : buildings.GetHarbors())
8✔
1059
        {
1060
            unsigned points = harbor->CalcDistributionPoints(gt);
×
1061
            if(!points)
×
1062
                continue;
×
1063

1064
            points += 10 * 30; // Verteilung existiert nicht, Expeditionen haben allerdings hohe Priorität
×
1065
            unsigned distance = world.CalcDistance(start->GetPos(), harbor->GetPos()) / 2;
×
1066
            possibleClients.push_back(ClientForWare(harbor, points > distance ? points - distance : 0, points));
×
1067
        }
1068
    }
1069

1070
    for(const auto bldType : wareDistribution.client_buildings)
49✔
1071
    {
1072
        // BuildingType::Headquarters sind Baustellen!!, da HQs ja sowieso nicht gebaut werden können
1073
        if(bldType == BuildingType::Headquarters)
37✔
1074
        {
1075
            // Bei Baustellen die Extraliste abfragen
1076
            for(noBuildingSite* bldSite : buildings.GetBuildingSites())
8✔
1077
            {
1078
                // Optimization: Ignore if unconnected
1079
                if(!bldSite->IsConnected())
×
1080
                    continue;
×
1081

1082
                unsigned points = bldSite->CalcDistributionPoints(gt);
×
1083
                if(!points)
×
1084
                    continue;
×
1085

1086
                points += wareDistribution.percent_buildings[BuildingType::Headquarters] * 30;
×
1087
                unsigned distance = world.CalcDistance(start->GetPos(), bldSite->GetPos()) / 2;
×
1088
                possibleClients.push_back(ClientForWare(bldSite, points > distance ? points - distance : 0, points));
×
1089
            }
1090
        } else
1091
        {
1092
            // Für übrige Gebäude
1093
            for(nobUsual* bld : buildings.GetBuildings(bldType))
29✔
1094
            {
1095
                // Optimization: Ignore if unconnected
1096
                if(!bld->IsConnected())
×
1097
                    continue;
×
1098

1099
                unsigned points = bld->CalcDistributionPoints(gt);
×
1100
                if(!points)
×
1101
                    continue; // Ware not needed
×
1102

1103
                if(!wareDistribution.goals.empty())
×
1104
                {
1105
                    if(bld->GetBuildingType()
×
1106
                       == static_cast<BuildingType>(wareDistribution.goals[wareDistribution.selected_goal]))
×
1107
                        points += 300;
×
1108
                    else if(points >= 300) // avoid overflows (async!)
×
1109
                        points -= 300;
×
1110
                    else
1111
                        points = 0;
×
1112
                }
1113

1114
                unsigned distance = world.CalcDistance(start->GetPos(), bld->GetPos()) / 2;
×
1115
                possibleClients.push_back(ClientForWare(bld, points > distance ? points - distance : 0, points));
×
1116
            }
1117
        }
1118
    }
1119

1120
    // sort our clients, highest score first
1121
    helpers::sort(possibleClients);
12✔
1122

1123
    noBaseBuilding* lastBld = nullptr;
12✔
1124
    noBaseBuilding* bestBld = nullptr;
12✔
1125
    unsigned best_points = 0;
12✔
1126
    for(auto& possibleClient : possibleClients)
12✔
1127
    {
1128
        unsigned path_length;
1129

1130
        // If our estimate is worse (or equal) best_points, the real value cannot be better.
1131
        // As our list is sorted, further entries cannot be better either, so stop searching.
1132
        if(possibleClient.estimate <= best_points)
×
1133
            break;
×
1134

1135
        // get rid of double building entries. TODO: why are there double entries!?
1136
        if(possibleClient.bld == lastBld)
×
1137
            continue;
×
1138

1139
        lastBld = possibleClient.bld;
×
1140

1141
        // Just to be sure no underflow happens...
1142
        if(possibleClient.points < best_points + 1)
×
1143
            continue;
×
1144

1145
        // Find path ONLY if it may be better. Pathfinding is limited to the worst path score that would lead to a
1146
        // better score. This eliminates the worst case scenario where all nodes in a split road network would be hit by
1147
        // the pathfinding only to conclude that there is no possible path.
1148
        if(world.FindPathForWareOnRoads(*start, *possibleClient.bld, &path_length, nullptr,
×
1149
                                        (possibleClient.points - best_points) * 2 - 1)
×
1150
           != RoadPathDirection::None)
×
1151
        {
1152
            unsigned score = possibleClient.points - (path_length / 2);
×
1153

1154
            // As we have limited our pathfinding to take a maximum of (points - best_points) * 2 - 1 steps,
1155
            // path_length / 2 can at most be points - best_points - 1, so the score will be greater than best_points.
1156
            // :)
1157
            RTTR_Assert(score > best_points);
×
1158

1159
            best_points = score;
×
1160
            bestBld = possibleClient.bld;
×
1161
        }
1162
    }
1163

1164
    if(bestBld && !wareDistribution.goals.empty())
12✔
1165
        wareDistribution.selected_goal =
×
1166
          (wareDistribution.selected_goal + 907) % unsigned(wareDistribution.goals.size());
×
1167

1168
    // Wenn kein Abnehmer gefunden wurde, muss es halt in ein Lagerhaus
1169
    if(!bestBld)
12✔
1170
        bestBld = FindWarehouseForWare(ware);
12✔
1171

1172
    return bestBld;
12✔
1173
}
1174

1175
nobBaseWarehouse* GamePlayer::FindWarehouseForWare(const Ware& ware) const
16✔
1176
{
1177
    // Check whs that collect this ware
1178
    nobBaseWarehouse* wh = FindWarehouse(*ware.GetLocation(), FW::CollectsWare(ware.type), true, true);
16✔
1179
    // If there is none, check those that accept it
1180
    if(!wh)
16✔
1181
    {
1182
        // First find the ones, that do not send it right away (IMPORTANT: This avoids sending a ware to the wh that is
1183
        // sending the ware out)
1184
        wh = FindWarehouse(*ware.GetLocation(), FW::AcceptsWareButNoSend(ware.type), true, true);
16✔
1185
        // The others only if this fails
1186
        if(!wh)
16✔
1187
            wh = FindWarehouse(*ware.GetLocation(), FW::AcceptsWare(ware.type), true, true);
5✔
1188
    }
1189
    return wh;
16✔
1190
}
1191

1192
template<class T_GetPriority, class T_Buildings>
1193
typename T_Buildings::value_type GamePlayer::FindClientImpl(const Ware& ware, T_GetPriority&& getPriority,
×
1194
                                                            const T_Buildings& buildings) const
1195
{
1196
    // TODO(replay): Unify with FindClientForWare: Sort by estimated priority first for less path finding.
1197
    // See PR #1720
1198
    typename T_Buildings::value_type bestClient = nullptr;
×
1199
    unsigned bestPoints = 0;
×
1200
    const auto& wareLocation = *ware.GetLocation();
×
1201
    const MapPoint warePos = wareLocation.GetPos();
×
1202
    for(auto* bld : buildings)
×
1203
    {
1204
        // Optimization: Ignore if unconnected
1205
        if(!bld->IsConnected())
×
1206
            continue;
×
1207
        const auto points = getPriority(*bld);
×
1208

1209
        // Consider costs for reaching the building
1210
        // We want only: points - pathCosts > bestPoints, i.e. 0 <= pathCosts < points - bestPoints
1211
        // So only check if difference is strictly positive, which covers the points==0 case
1212
        if(points > bestPoints)
×
1213
        {
1214
            const unsigned maxPathCosts = points - bestPoints - 1;
×
1215
            const unsigned distance = world.CalcDistance(warePos, bld->GetPos());
×
1216
            if(distance > maxPathCosts)
×
1217
                continue;
×
1218
            unsigned pathCosts;
1219
            if(world.FindPathForWareOnRoads(wareLocation, *bld, &pathCosts, nullptr, maxPathCosts)
×
1220
               != RoadPathDirection::None)
×
1221
            {
1222
                RTTR_Assert(points > pathCosts && points - pathCosts > bestPoints);
×
1223
                bestPoints = points - pathCosts;
×
1224
                bestClient = bld;
×
1225
            }
1226
        }
1227
    }
1228
    return bestClient;
×
1229
}
1230

1231
nobBaseMilitary* GamePlayer::FindClientForCoin(const Ware& ware) const
×
1232
{
1233
    nobBaseMilitary* result = FindClientImpl(
×
1234
      ware, [](const nobMilitary& bld) { return bld.CalcCoinsPoints(); }, buildings.GetMilitaryBuildings());
×
1235
    // Send to warehouse if no military building needs the coins
1236
    if(!result)
×
1237
        result = FindWarehouseForWare(ware);
×
1238

1239
    return result;
×
1240
}
1241

1242
nobBaseMilitary* GamePlayer::FindClientForArmor(const Ware& ware) const
×
1243
{
1244
    nobBaseMilitary* result = FindClientImpl(
×
1245
      ware, [](const nobMilitary& bld) { return bld.CalcArmorPoints(); }, buildings.GetMilitaryBuildings());
×
1246
    // Send to warehouse if no military building needs the coins
1247
    if(!result)
×
1248
        result = FindWarehouseForWare(ware);
×
1249

1250
    return result;
×
1251
}
1252

1253
unsigned GamePlayer::GetBuidingSitePriority(const noBuildingSite* building_site)
×
1254
{
1255
    if(useCustomBuildOrder_)
×
1256
    {
1257
        // Spezielle Reihenfolge
1258

1259
        // Typ in der Reihenfolge suchen und Position als Priorität zurückgeben
1260
        for(unsigned i = 0; i < build_order.size(); ++i)
×
1261
        {
1262
            if(building_site->GetBuildingType() == build_order[i])
×
1263
                return i;
×
1264
        }
1265
    } else
1266
    {
1267
        // Reihenfolge der Bauaufträge, also was zuerst in Auftrag gegeben wurde, wird zuerst gebaut
1268
        unsigned i = 0;
×
1269
        for(noBuildingSite* bldSite : buildings.GetBuildingSites())
×
1270
        {
1271
            if(building_site == bldSite)
×
1272
                return i;
×
1273
            i++;
×
1274
        }
1275
    }
1276

1277
    LOG.write("GameClientPlayer::GetBuidingSitePriority: ERROR: Buildingsite or type of it not found in the list!\n");
×
1278
    RTTR_Assert(false);
×
1279
    // We may want to multiply this value so don't return the absolute max value
1280
    return std::numeric_limits<unsigned>::max() / 1000;
1281
}
1282

1283
void GamePlayer::ConvertTransportData(const TransportOrders& transport_data)
2✔
1284
{
1285
    for(const auto ware : helpers::EnumRange<GoodType>{})
168✔
1286
        transportPrio[ware] = GetTransportPrioFromOrdering(transport_data, ware);
80✔
1287
}
2✔
1288

1289
bool GamePlayer::IsAlly(const unsigned char playerId) const
12,362✔
1290
{
1291
    // Der Spieler ist ja auch zu sich selber verbündet
1292
    if(GetPlayerId() == playerId)
12,362✔
1293
        return true;
4,029✔
1294
    else
1295
        return GetPactState(PactType::TreatyOfAlliance, playerId) == PactState::Accepted;
8,333✔
1296
}
1297

1298
bool GamePlayer::IsAttackable(const unsigned char playerId) const
933✔
1299
{
1300
    // Verbündete dürfen nicht angegriffen werden
1301
    if(IsAlly(playerId))
933✔
1302
        return false;
44✔
1303
    else
1304
        // Ansonsten darf bei bestehendem Nichtangriffspakt ebenfalls nicht angegriffen werden
1305
        return GetPactState(PactType::NonAgressionPact, playerId) != PactState::Accepted;
889✔
1306
}
1307

1308
void GamePlayer::OrderTroops(nobMilitary& goal, std::array<unsigned, NUM_SOLDIER_RANKS> counts,
36✔
1309
                             unsigned total_max) const
1310
{
1311
    // Solange Lagerhäuser nach Soldaten absuchen, bis entweder keins mehr übrig ist oder alle Soldaten bestellt sind
1312
    nobBaseWarehouse* wh;
1313
    unsigned sum = 0;
36✔
1314
    do
5✔
1315
    {
1316
        std::array<bool, NUM_SOLDIER_RANKS> desiredRanks;
1317
        for(unsigned i = 0; i < NUM_SOLDIER_RANKS; i++)
246✔
1318
            desiredRanks[i] = counts[i] > 0;
205✔
1319

1320
        wh = FindWarehouse(goal, FW::HasAnyMatchingSoldier(desiredRanks), false, false);
41✔
1321
        if(wh)
41✔
1322
        {
1323
            wh->OrderTroops(goal, counts, total_max);
19✔
1324
            sum = std::accumulate(counts.begin(), counts.end(), 0u);
19✔
1325
        }
1326
    } while(total_max && sum && wh);
41✔
1327
}
36✔
1328

1329
void GamePlayer::RegulateAllTroops()
93✔
1330
{
1331
    for(nobMilitary* milBld : buildings.GetMilitaryBuildings())
100✔
1332
        milBld->RegulateTroops();
7✔
1333
}
93✔
1334

1335
void GamePlayer::RecalcMilitaryFlags()
48✔
1336
{
1337
    for(nobMilitary* milBld : buildings.GetMilitaryBuildings())
72✔
1338
        milBld->LookForEnemyBuildings(nullptr);
24✔
1339
}
48✔
1340

1341
void GamePlayer::NewSoldiersAvailable(const unsigned& soldier_count)
192✔
1342
{
1343
    RTTR_Assert(soldier_count > 0);
192✔
1344
    // solange laufen lassen, bis soldier_count = 0, d.h. der Soldat irgendwohin geschickt wurde
1345
    // Zuerst nach unbesetzten Militärgebäude schauen
1346
    for(nobMilitary* milBld : buildings.GetMilitaryBuildings())
216✔
1347
    {
1348
        if(milBld->IsNewBuilt())
24✔
1349
        {
1350
            milBld->RegulateTroops();
6✔
1351
            // Used that soldier? Go out
1352
            if(!soldier_count)
6✔
1353
                return;
×
1354
        }
1355
    }
1356

1357
    // Als nächstes Gebäude in Grenznähe
1358
    for(nobMilitary* milBld : buildings.GetMilitaryBuildings())
216✔
1359
    {
1360
        if(milBld->GetFrontierDistance() == FrontierDistance::Near)
24✔
1361
        {
1362
            milBld->RegulateTroops();
24✔
1363
            // Used that soldier? Go out
1364
            if(!soldier_count)
24✔
1365
                return;
×
1366
        }
1367
    }
1368

1369
    // Und den Rest ggf.
1370
    for(nobMilitary* milBld : buildings.GetMilitaryBuildings())
216✔
1371
    {
1372
        // already checked? -> skip
1373
        if(milBld->GetFrontierDistance() == FrontierDistance::Near || milBld->IsNewBuilt())
24✔
1374
            continue;
24✔
1375
        milBld->RegulateTroops();
×
1376
        if(!soldier_count) // used the soldier?
×
1377
            return;
×
1378
    }
1379
}
1380

1381
void GamePlayer::CallFlagWorker(const MapPoint pt, const Job job)
32✔
1382
{
1383
    auto* flag = world.GetSpecObj<noFlag>(pt);
32✔
1384
    if(!flag)
32✔
1385
        return;
2✔
1386
    /// Find wh with given job type (e.g. geologist, scout, ...)
1387
    nobBaseWarehouse* wh = FindWarehouse(*flag, FW::HasFigure(job, true), false, false);
30✔
1388

1389
    if(wh)
30✔
1390
        wh->OrderJob(job, *flag, true);
26✔
1391
}
1392

1393
bool GamePlayer::IsFlagWorker(const nofFlagWorker* flagworker)
×
1394
{
1395
    return helpers::contains(flagworkers, flagworker);
×
1396
}
1397

1398
void GamePlayer::FlagDestroyed(noFlag* flag)
777✔
1399
{
1400
    // Alle durchgehen und ggf. sagen, dass sie keine Flagge mehr haben, wenn das ihre Flagge war, die zerstört wurde
1401
    for(auto it = flagworkers.begin(); it != flagworkers.end();)
777✔
1402
    {
1403
        if((*it)->GetFlag() == flag)
×
1404
        {
1405
            (*it)->LostWork();
×
1406
            it = flagworkers.erase(it);
×
1407
        } else
1408
            ++it;
×
1409
    }
1410
}
777✔
1411

1412
void GamePlayer::RefreshDefenderList()
78✔
1413
{
1414
    shouldSendDefenderList.clear();
78✔
1415
    // Add as many true values as set in the settings, the rest will be false
1416
    for(unsigned i = 0; i < MILITARY_SETTINGS_SCALE[2]; ++i)
468✔
1417
        shouldSendDefenderList.push_back(i < militarySettings_[2]);
390✔
1418
    // und ordentlich schütteln
1419
    RANDOM_SHUFFLE2(shouldSendDefenderList, 0);
78✔
1420
}
78✔
1421

1422
void GamePlayer::ChangeMilitarySettings(const MilitarySettings& military_settings)
78✔
1423
{
1424
    for(unsigned i = 0; i < military_settings.size(); ++i)
702✔
1425
    {
1426
        // Sicherstellen, dass im validen Bereich
1427
        RTTR_Assert(military_settings[i] <= MILITARY_SETTINGS_SCALE[i]);
624✔
1428
        this->militarySettings_[i] = military_settings[i];
624✔
1429
    }
1430
    /// Truppen müssen neu kalkuliert werden
1431
    RegulateAllTroops();
78✔
1432
    /// Die Verteidigungsliste muss erneuert werden
1433
    RefreshDefenderList();
78✔
1434
}
78✔
1435

1436
void GamePlayer::ChangeToolsSettings(const ToolSettings& tools_settings,
6✔
1437
                                     const helpers::EnumArray<int8_t, Tool>& orderChanges)
1438
{
1439
    const bool settingsChanged = toolsSettings_ != tools_settings;
6✔
1440
    toolsSettings_ = tools_settings;
6✔
1441
    if(settingsChanged)
6✔
1442
        world.GetNotifications().publish(ToolNote(ToolNote::SettingsChanged, GetPlayerId()));
4✔
1443

1444
    for(const auto tool : helpers::enumRange<Tool>())
168✔
1445
    {
1446
        tools_ordered[tool] = helpers::clamp(tools_ordered[tool] + orderChanges[tool], 0, 100);
72✔
1447
        tools_ordered_delta[tool] -= orderChanges[tool];
72✔
1448

1449
        if(orderChanges[tool] != 0)
72✔
1450
        {
1451
            LOG.write(">> Committing an order of %1% for tool #%2%(%3%)\n", LogTarget::File) % (int)orderChanges[tool]
8✔
1452
              % static_cast<unsigned>(tool) % _(WARE_NAMES[TOOL_TO_GOOD[tool]]);
8✔
1453
            world.GetNotifications().publish(ToolNote(ToolNote::OrderPlaced, GetPlayerId()));
4✔
1454
        }
1455
    }
1456
}
6✔
1457

1458
void GamePlayer::ChangeDistribution(const Distributions& distribution_settings)
6✔
1459
{
1460
    unsigned idx = 0;
6✔
1461
    for(const DistributionMapping& mapping : distributionMap)
180✔
1462
    {
1463
        distribution[std::get<0>(mapping)].percent_buildings[std::get<1>(mapping)] = distribution_settings[idx++];
174✔
1464
    }
1465

1466
    RecalcDistribution();
6✔
1467
}
6✔
1468

1469
void GamePlayer::ChangeBuildOrder(bool useCustomBuildOrder, const BuildOrders& order_data)
2✔
1470
{
1471
    this->useCustomBuildOrder_ = useCustomBuildOrder;
2✔
1472
    this->build_order = order_data;
2✔
1473
}
2✔
1474

1475
bool GamePlayer::ShouldSendDefender()
46✔
1476
{
1477
    // Wenn wir schon am Ende sind, muss die Verteidgungsliste erneuert werden
1478
    if(shouldSendDefenderList.empty())
46✔
1479
        RefreshDefenderList();
×
1480

1481
    bool result = shouldSendDefenderList.back();
46✔
1482
    shouldSendDefenderList.pop_back();
46✔
1483
    return result;
46✔
1484
}
1485

1486
void GamePlayer::TestDefeat()
84✔
1487
{
1488
    // Nicht schon besiegt?
1489
    // Keine Militärgebäude, keine Lagerhäuser (HQ,Häfen) -> kein Land --> verloren
1490
    if(!isDefeated && buildings.GetMilitaryBuildings().empty() && buildings.GetStorehouses().empty())
84✔
1491
        Surrender();
29✔
1492
}
84✔
1493

1494
const nobHQ* GamePlayer::GetHQ() const
71✔
1495
{
1496
    return hqPos.isValid() ? GetGameWorld().GetSpecObj<nobHQ>(hqPos) : nullptr;
71✔
1497
}
1498

1499
nobHQ* GamePlayer::GetHQ()
63✔
1500
{
1501
    return const_cast<nobHQ*>(std::as_const(*this).GetHQ());
63✔
1502
}
1503

1504
void GamePlayer::Surrender()
32✔
1505
{
1506
    if(isDefeated)
32✔
1507
        return;
1✔
1508

1509
    isDefeated = true;
31✔
1510

1511
    // GUI Bescheid sagen
1512
    if(world.GetGameInterface())
31✔
1513
        world.GetGameInterface()->GI_PlayerDefeated(GetPlayerId());
×
1514
}
1515

1516
void GamePlayer::SetStatisticValue(StatisticType type, unsigned value)
×
1517
{
1518
    statisticCurrentData[type] = value;
×
1519
}
×
1520

1521
void GamePlayer::ChangeStatisticValue(StatisticType type, int change)
2,203✔
1522
{
1523
    RTTR_Assert(change >= 0 || statisticCurrentData[type] >= static_cast<unsigned>(-change));
2,203✔
1524
    statisticCurrentData[type] += change;
2,203✔
1525
}
2,203✔
1526

1527
void GamePlayer::IncreaseMerchandiseStatistic(GoodType type)
4✔
1528
{
1529
    // Einsortieren...
1530
    switch(type)
4✔
1531
    {
1532
        case GoodType::Wood: statisticCurrentMerchandiseData[0]++; break;
×
1533
        case GoodType::Boards: statisticCurrentMerchandiseData[1]++; break;
×
1534
        case GoodType::Stones: statisticCurrentMerchandiseData[2]++; break;
×
1535
        case GoodType::Fish:
×
1536
        case GoodType::Bread:
1537
        case GoodType::Meat: statisticCurrentMerchandiseData[3]++; break;
×
1538
        case GoodType::Water: statisticCurrentMerchandiseData[4]++; break;
×
1539
        case GoodType::Beer: statisticCurrentMerchandiseData[5]++; break;
×
1540
        case GoodType::Coal: statisticCurrentMerchandiseData[6]++; break;
×
1541
        case GoodType::IronOre: statisticCurrentMerchandiseData[7]++; break;
×
1542
        case GoodType::Gold: statisticCurrentMerchandiseData[8]++; break;
×
1543
        case GoodType::Iron: statisticCurrentMerchandiseData[9]++; break;
×
1544
        case GoodType::Coins: statisticCurrentMerchandiseData[10]++; break;
×
1545
        case GoodType::Tongs:
3✔
1546
        case GoodType::Axe:
1547
        case GoodType::Saw:
1548
        case GoodType::PickAxe:
1549
        case GoodType::Hammer:
1550
        case GoodType::Shovel:
1551
        case GoodType::Crucible:
1552
        case GoodType::RodAndLine:
1553
        case GoodType::Scythe:
1554
        case GoodType::Cleaver:
1555
        case GoodType::Rollingpin:
1556
        case GoodType::Bow: statisticCurrentMerchandiseData[11]++; break;
3✔
1557
        case GoodType::ShieldVikings:
×
1558
        case GoodType::ShieldAfricans:
1559
        case GoodType::ShieldRomans:
1560
        case GoodType::ShieldJapanese:
1561
        case GoodType::Sword: statisticCurrentMerchandiseData[12]++; break;
×
1562
        case GoodType::Boat: statisticCurrentMerchandiseData[13]++; break;
×
1563
        default: break;
1✔
1564
    }
1565
}
4✔
1566

1567
void GamePlayer::CalcStatistics()
6✔
1568
{
1569
    // Waren aus der Inventur zählen
1570
    statisticCurrentData[StatisticType::Merchandise] = 0;
6✔
1571
    for(const auto i : helpers::enumRange<GoodType>())
504✔
1572
        statisticCurrentData[StatisticType::Merchandise] += global_inventory[i];
240✔
1573

1574
    // Bevölkerung aus der Inventur zählen
1575
    statisticCurrentData[StatisticType::Inhabitants] = 0;
6✔
1576
    for(const auto i : helpers::enumRange<Job>())
480✔
1577
        statisticCurrentData[StatisticType::Inhabitants] += global_inventory[i];
228✔
1578

1579
    // Militär aus der Inventur zählen
1580
    statisticCurrentData[StatisticType::Military] =
12✔
1581
      global_inventory.people[Job::Private] + global_inventory.people[Job::PrivateFirstClass] * 2
6✔
1582
      + global_inventory.people[Job::Sergeant] * 3 + global_inventory.people[Job::Officer] * 4
6✔
1583
      + global_inventory.people[Job::General] * 5;
6✔
1584

1585
    // Produktivität berechnen
1586
    statisticCurrentData[StatisticType::Productivity] = buildings.CalcAverageProductivity();
6✔
1587

1588
    // Total points for tournament games
1589
    statisticCurrentData[StatisticType::Tournament] =
12✔
1590
      statisticCurrentData[StatisticType::Military] + 3 * statisticCurrentData[StatisticType::Vanquished];
6✔
1591
}
6✔
1592

1593
void GamePlayer::StatisticStep()
6✔
1594
{
1595
    CalcStatistics();
6✔
1596

1597
    // 15-min-Statistik ein Feld weiterschieben
1598
    for(const auto i : helpers::enumRange<StatisticType>())
132✔
1599
    {
1600
        statistic[StatisticTime::T15Minutes].data[i][incrStatIndex(statistic[StatisticTime::T15Minutes].currentIndex)] =
54✔
1601
          statisticCurrentData[i];
54✔
1602
    }
1603
    for(unsigned i = 0; i < NUM_STAT_MERCHANDISE_TYPES; ++i)
90✔
1604
    {
1605
        statistic[StatisticTime::T15Minutes]
84✔
1606
          .merchandiseData[i][incrStatIndex(statistic[StatisticTime::T15Minutes].currentIndex)] =
168✔
1607
          statisticCurrentMerchandiseData[i];
84✔
1608
    }
1609
    statistic[StatisticTime::T15Minutes].currentIndex =
6✔
1610
      incrStatIndex(statistic[StatisticTime::T15Minutes].currentIndex);
6✔
1611

1612
    statistic[StatisticTime::T15Minutes].counter++;
6✔
1613

1614
    // Prüfen ob 4mal 15-min-Statistik weitergeschoben wurde, wenn ja: 1-h-Statistik weiterschieben
1615
    // und aktuellen Wert der 15min-Statistik benutzen
1616
    // gleiches für die 4h und 16h Statistik
1617
    for(const auto t : helpers::enumRange<StatisticTime>())
60✔
1618
    {
1619
        if(t == StatisticTime(helpers::MaxEnumValue_v<StatisticTime>))
24✔
1620
            break;
6✔
1621
        const auto nextT = StatisticTime(rttr::enum_cast(t) + 1);
18✔
1622
        if(statistic[t].counter == 4)
18✔
1623
        {
1624
            statistic[t].counter = 0;
×
1625
            for(const auto i : helpers::enumRange<StatisticType>())
×
1626
            {
1627
                statistic[nextT].data[i][incrStatIndex(statistic[nextT].currentIndex)] = statisticCurrentData[i];
×
1628
            }
1629

1630
            // Summe für den Zeitraum berechnen (immer 4 Zeitschritte der jeweils kleineren Statistik)
1631
            for(unsigned i = 0; i < NUM_STAT_MERCHANDISE_TYPES; ++i)
×
1632
            {
1633
                statistic[nextT].merchandiseData[i][incrStatIndex(statistic[nextT].currentIndex)] =
×
1634
                  statisticCurrentMerchandiseData[i]
×
1635
                  + statistic[t].merchandiseData[i][decrStatIndex(statistic[t].currentIndex, 1)]
×
1636
                  + statistic[t].merchandiseData[i][decrStatIndex(statistic[t].currentIndex, 2)]
×
1637
                  + statistic[t].merchandiseData[i][decrStatIndex(statistic[t].currentIndex, 3)];
×
1638
            }
1639

1640
            statistic[nextT].currentIndex = incrStatIndex(statistic[nextT].currentIndex);
×
1641
            statistic[nextT].counter++;
×
1642
        }
1643
    }
1644

1645
    // Warenstatistikzähler nullen
1646
    statisticCurrentMerchandiseData.fill(0);
6✔
1647
}
6✔
1648

1649
GamePlayer::Pact::Pact(SerializedGameData& sgd)
112✔
1650
    : duration(sgd.PopUnsignedInt()), start(sgd.PopUnsignedInt()), accepted(sgd.PopBool()), want_cancel(sgd.PopBool())
112✔
1651
{}
112✔
1652

1653
void GamePlayer::Pact::Serialize(SerializedGameData& sgd) const
224✔
1654
{
1655
    sgd.PushUnsignedInt(duration);
224✔
1656
    sgd.PushUnsignedInt(start);
224✔
1657
    sgd.PushBool(accepted);
224✔
1658
    sgd.PushBool(want_cancel);
224✔
1659
}
224✔
1660

1661
void GamePlayer::PactChanged(const PactType pt)
24✔
1662
{
1663
    // Recheck military flags as the border (to an enemy) might have changed
1664
    RecalcMilitaryFlags();
24✔
1665

1666
    // Ggf. den GUI Bescheid sagen, um Sichtbarkeiten etc. neu zu berechnen
1667
    if(pt == PactType::TreatyOfAlliance)
24✔
1668
    {
1669
        if(world.GetGameInterface())
6✔
1670
            world.GetGameInterface()->GI_TreatyOfAllianceChanged(GetPlayerId());
×
1671
    }
1672
}
24✔
1673

1674
void GamePlayer::SuggestPact(const unsigned char targetPlayerId, const PactType pt, const unsigned duration)
10✔
1675
{
1676
    // Don't try to make pact with self
1677
    if(targetPlayerId == GetPlayerId())
10✔
1678
        return;
1✔
1679

1680
    if(!pacts[targetPlayerId][pt].accepted && duration > 0)
9✔
1681
    {
1682
        pacts[targetPlayerId][pt].duration = duration;
8✔
1683
        pacts[targetPlayerId][pt].start = world.GetEvMgr().GetCurrentGF();
8✔
1684
        GamePlayer targetPlayer = world.GetPlayer(targetPlayerId);
16✔
1685
        if(targetPlayer.isHuman())
8✔
1686
            targetPlayer.SendPostMessage(std::make_unique<DiplomacyPostQuestion>(
6✔
1687
              world.GetEvMgr().GetCurrentGF(), pt, pacts[targetPlayerId][pt].start, *this, duration));
12✔
1688
        else if(world.HasLua())
2✔
1689
            world.GetLua().EventSuggestPact(pt, GetPlayerId(), targetPlayerId, duration);
2✔
1690
    }
1691
}
1692

1693
void GamePlayer::AcceptPact(const unsigned id, const PactType pt, const unsigned char targetPlayer)
21✔
1694
{
1695
    if(!pacts[targetPlayer][pt].accepted && pacts[targetPlayer][pt].duration > 0 && pacts[targetPlayer][pt].start == id)
21✔
1696
    {
1697
        MakePact(pt, targetPlayer, pacts[targetPlayer][pt].duration);
8✔
1698
        world.GetPlayer(targetPlayer).MakePact(pt, GetPlayerId(), pacts[targetPlayer][pt].duration);
8✔
1699
        PactChanged(pt);
8✔
1700
        world.GetPlayer(targetPlayer).PactChanged(pt);
8✔
1701
        if(world.HasLua())
8✔
1702
            world.GetLua().EventPactCreated(pt, GetPlayerId(), targetPlayer, pacts[targetPlayer][pt].duration);
3✔
1703
    }
1704
}
21✔
1705

1706
void GamePlayer::MakePact(const PactType pt, const unsigned char other_player, const unsigned duration)
16✔
1707
{
1708
    pacts[other_player][pt].accepted = true;
16✔
1709
    pacts[other_player][pt].start = world.GetEvMgr().GetCurrentGF();
16✔
1710
    pacts[other_player][pt].duration = duration;
16✔
1711
    pacts[other_player][pt].want_cancel = false;
16✔
1712

1713
    SendPostMessage(
16✔
1714
      std::make_unique<PostMsg>(world.GetEvMgr().GetCurrentGF(), pt, world.GetPlayer(other_player), true));
32✔
1715
}
16✔
1716

1717
PactState GamePlayer::GetPactState(const PactType pt, const unsigned char other_player) const
9,372✔
1718
{
1719
    // Prüfen, ob Bündnis in Kraft ist
1720
    if(pacts[other_player][pt].duration)
9,372✔
1721
    {
1722
        if(!pacts[other_player][pt].accepted)
5,040✔
1723
            return PactState::InProgress;
16✔
1724

1725
        if(pacts[other_player][pt].duration == DURATION_INFINITE
5,024✔
1726
           || world.GetEvMgr().GetCurrentGF() < pacts[other_player][pt].start + pacts[other_player][pt].duration)
5,024✔
1727
            return PactState::Accepted;
5,022✔
1728
    }
1729

1730
    return PactState::None;
4,334✔
1731
}
1732

1733
void GamePlayer::NotifyAlliesOfLocation(const MapPoint pt)
6✔
1734
{
1735
    for(unsigned i = 0; i < world.GetNumPlayers(); ++i)
24✔
1736
    {
1737
        if(i != GetPlayerId() && IsAlly(i))
18✔
1738
            world.GetPlayer(i).SendPostMessage(std::make_unique<PostMsg>(
10✔
1739
              world.GetEvMgr().GetCurrentGF(), _("Your ally wishes to notify you of this location"),
5✔
1740
              PostCategory::Diplomacy, pt));
10✔
1741
    }
1742
}
6✔
1743

1744
unsigned GamePlayer::GetRemainingPactTime(const PactType pt, const unsigned char other_player) const
125✔
1745
{
1746
    if(pacts[other_player][pt].duration)
125✔
1747
    {
1748
        if(pacts[other_player][pt].accepted)
88✔
1749
        {
1750
            if(pacts[other_player][pt].duration == DURATION_INFINITE)
76✔
1751
                return DURATION_INFINITE;
×
1752
            else if(world.GetEvMgr().GetCurrentGF() <= pacts[other_player][pt].start + pacts[other_player][pt].duration)
76✔
1753
                return ((pacts[other_player][pt].start + pacts[other_player][pt].duration)
76✔
1754
                        - world.GetEvMgr().GetCurrentGF());
76✔
1755
        }
1756
    }
1757

1758
    return 0;
49✔
1759
}
1760

1761
void GamePlayer::CancelPact(const PactType pt, const unsigned char otherPlayerIdx)
8✔
1762
{
1763
    // Don't try to cancel pact with self
1764
    if(otherPlayerIdx == GetPlayerId())
8✔
1765
        return;
2✔
1766

1767
    // Besteht bereits ein Bündnis?
1768
    if(pacts[otherPlayerIdx][pt].accepted)
6✔
1769
    {
1770
        // Vermerken, dass der Spieler das Bündnis auflösen will
1771
        pacts[otherPlayerIdx][pt].want_cancel = true;
6✔
1772

1773
        // Will der andere Spieler das Bündnis auch auflösen?
1774
        GamePlayer& otherPlayer = world.GetPlayer(otherPlayerIdx);
6✔
1775
        if(otherPlayer.pacts[GetPlayerId()][pt].want_cancel)
6✔
1776
        {
1777
            // Dann wird das Bündnis aufgelöst
1778
            pacts[otherPlayerIdx][pt].accepted = false;
2✔
1779
            pacts[otherPlayerIdx][pt].duration = 0;
2✔
1780
            pacts[otherPlayerIdx][pt].want_cancel = false;
2✔
1781

1782
            otherPlayer.pacts[GetPlayerId()][pt].accepted = false;
2✔
1783
            otherPlayer.pacts[GetPlayerId()][pt].duration = 0;
2✔
1784
            otherPlayer.pacts[GetPlayerId()][pt].want_cancel = false;
2✔
1785

1786
            // Den Spielern eine Informationsnachricht schicken
1787
            world.GetPlayer(otherPlayerIdx)
2✔
1788
              .SendPostMessage(std::make_unique<PostMsg>(world.GetEvMgr().GetCurrentGF(), pt, *this, false));
2✔
1789
            SendPostMessage(
2✔
1790
              std::make_unique<PostMsg>(world.GetEvMgr().GetCurrentGF(), pt, world.GetPlayer(otherPlayerIdx), false));
4✔
1791
            PactChanged(pt);
2✔
1792
            otherPlayer.PactChanged(pt);
2✔
1793
            if(world.HasLua())
2✔
1794
                world.GetLua().EventPactCanceled(pt, GetPlayerId(), otherPlayerIdx);
1✔
1795
        } else
1796
        {
1797
            // Ansonsten den anderen Spieler fragen, ob der das auch so sieht
1798
            if(otherPlayer.isHuman())
4✔
1799
                otherPlayer.SendPostMessage(std::make_unique<DiplomacyPostQuestion>(
3✔
1800
                  world.GetEvMgr().GetCurrentGF(), pt, pacts[otherPlayerIdx][pt].start, *this));
6✔
1801
            else if(!world.HasLua() || world.GetLua().EventCancelPactRequest(pt, GetPlayerId(), otherPlayerIdx))
1✔
1802
            {
1803
                // AI accepts cancels, if there is no lua-interace
1804
                pacts[otherPlayerIdx][pt].accepted = false;
1✔
1805
                pacts[otherPlayerIdx][pt].duration = 0;
1✔
1806
                pacts[otherPlayerIdx][pt].want_cancel = false;
1✔
1807

1808
                otherPlayer.pacts[GetPlayerId()][pt].accepted = false;
1✔
1809
                otherPlayer.pacts[GetPlayerId()][pt].duration = 0;
1✔
1810
                otherPlayer.pacts[GetPlayerId()][pt].want_cancel = false;
1✔
1811

1812
                if(world.HasLua())
1✔
1813
                    world.GetLua().EventPactCanceled(pt, GetPlayerId(), otherPlayerIdx);
1✔
1814
            }
1815
        }
1816
    } else
1817
    {
1818
        // Es besteht kein Bündnis, also unseren Bündnisvorschlag wieder zurücknehmen
1819
        pacts[otherPlayerIdx][pt].duration = 0;
×
1820
    }
1821
}
1822

1823
void GamePlayer::MakeStartPacts()
66✔
1824
{
1825
    // Reset pacts
1826
    for(unsigned i = 0; i < world.GetNumPlayers(); ++i)
262✔
1827
    {
1828
        for(const auto z : helpers::enumRange<PactType>())
1,568✔
1829
            pacts[i][z] = Pact();
392✔
1830
    }
1831

1832
    // No team -> No pacts
1833
    if(team == Team::None)
66✔
1834
        return;
4✔
1835
    RTTR_Assert(isTeam(team));
62✔
1836

1837
    // Create ally- and non-aggression-pact for all players of same team
1838
    for(unsigned i = 0; i < world.GetNumPlayers(); ++i)
245✔
1839
    {
1840
        if(team != world.GetPlayer(i).team)
183✔
1841
            continue;
75✔
1842
        for(const auto z : helpers::enumRange<PactType>())
864✔
1843
        {
1844
            pacts[i][z].duration = DURATION_INFINITE;
216✔
1845
            pacts[i][z].start = 0;
216✔
1846
            pacts[i][z].accepted = true;
216✔
1847
            pacts[i][z].want_cancel = false;
216✔
1848
        }
1849
    }
1850
}
1851

1852
bool GamePlayer::IsWareRegistred(const Ware& ware)
160✔
1853
{
1854
    return helpers::contains(ware_list, &ware);
160✔
1855
}
1856

1857
bool GamePlayer::IsWareDependent(const Ware& ware)
21✔
1858
{
1859
    for(nobBaseWarehouse* wh : buildings.GetStorehouses())
62✔
1860
    {
1861
        if(wh->IsWareDependent(ware))
41✔
1862
            return true;
×
1863
    }
1864

1865
    return false;
21✔
1866
}
1867

1868
void GamePlayer::IncreaseInventoryWare(const GoodType ware, const unsigned count)
1,676✔
1869
{
1870
    global_inventory.Add(ConvertShields(ware), count);
1,676✔
1871
}
1,676✔
1872

1873
void GamePlayer::DecreaseInventoryWare(const GoodType ware, const unsigned count)
2,467✔
1874
{
1875
    global_inventory.Remove(ConvertShields(ware), count);
2,467✔
1876
}
2,467✔
1877

1878
void GamePlayer::RegisterShip(noShip& ship)
23✔
1879
{
1880
    ships.push_back(&ship);
23✔
1881
    GetJobForShip(ship);
23✔
1882
}
23✔
1883

1884
struct ShipForHarbor
1885
{
1886
    noShip* ship;
1887
    uint32_t estimate;
1888

1889
    ShipForHarbor(noShip* ship, uint32_t estimate) : ship(ship), estimate(estimate) {}
20✔
1890

1891
    bool operator<(const ShipForHarbor& b) const noexcept
×
1892
    {
1893
        return (estimate < b.estimate) || (estimate == b.estimate && ship->GetObjId() < b.ship->GetObjId());
×
1894
    }
1895
};
1896

1897
bool GamePlayer::OrderShip(nobHarborBuilding& hb)
41✔
1898
{
1899
    std::vector<ShipForHarbor> sfh;
82✔
1900

1901
    // we need more ships than those that are already on their way? limit search to idle ships
1902
    if(GetShipsToHarbor(hb) < hb.GetNumNeededShips())
41✔
1903
    {
1904
        for(noShip* ship : ships)
82✔
1905
        {
1906
            if(ship->IsIdling() && world.IsHarborAtSea(world.GetHarborPointID(hb.GetPos()), ship->GetSeaID()))
41✔
1907
                sfh.push_back(ShipForHarbor(ship, world.CalcDistance(hb.GetPos(), ship->GetPos())));
20✔
1908
        }
1909
    } else
1910
    {
1911
        for(noShip* ship : ships)
×
1912
        {
1913
            if((ship->IsIdling() && world.IsHarborAtSea(world.GetHarborPointID(hb.GetPos()), ship->GetSeaID()))
×
1914
               || ship->IsGoingToHarbor(hb))
×
1915
            {
1916
                sfh.push_back(ShipForHarbor(ship, world.CalcDistance(hb.GetPos(), ship->GetPos())));
×
1917
            }
1918
        }
1919
    }
1920

1921
    helpers::sort(sfh);
41✔
1922

1923
    noShip* best_ship = nullptr;
41✔
1924
    uint32_t best_distance = std::numeric_limits<uint32_t>::max();
41✔
1925
    std::vector<Direction> best_route;
82✔
1926

1927
    for(auto& it : sfh)
52✔
1928
    {
1929
        uint32_t distance;
1930
        std::vector<Direction> route;
20✔
1931

1932
        // the estimate (air-line distance) for this and all other ships in the list is already worse than what we
1933
        // found? disregard the rest
1934
        if(it.estimate >= best_distance)
20✔
1935
            break;
×
1936

1937
        noShip& ship = *it.ship;
20✔
1938

1939
        if(world.FindShipPathToHarbor(ship.GetPos(), hb.GetHarborPosID(), ship.GetSeaID(), &route, &distance))
20✔
1940
        {
1941
            // ship already there?
1942
            if(distance == 0)
20✔
1943
            {
1944
                hb.ShipArrived(ship);
9✔
1945
                return true;
9✔
1946
            }
1947

1948
            if(distance < best_distance)
11✔
1949
            {
1950
                best_ship = &ship;
11✔
1951
                best_distance = distance;
11✔
1952
                best_route = route;
11✔
1953
            }
1954
        }
1955
    }
1956

1957
    // only order ships not already on their way
1958
    if(best_ship && best_ship->IsIdling())
32✔
1959
    {
1960
        best_ship->GoToHarbor(hb, best_route);
11✔
1961

1962
        return true;
11✔
1963
    }
1964

1965
    return false;
21✔
1966
}
1967

UNCOV
1968
void GamePlayer::RemoveShip(noShip* ship)
×
1969
{
1970
    auto it = helpers::find(ships, ship);
×
1971
    RTTR_Assert(it != ships.end());
×
1972
    ships.erase(it);
×
1973
}
×
1974

1975
void GamePlayer::GetJobForShip(noShip& ship)
33✔
1976
{
1977
    nobHarborBuilding* best = nullptr;
33✔
1978
    int best_points = 0;
33✔
1979
    std::vector<Direction> best_route;
33✔
1980

1981
    // Beste Weglänge, die ein Schiff zurücklegen muss, welches gerade nichts zu tun hat
1982
    for(nobHarborBuilding* harbor : buildings.GetHarbors())
68✔
1983
    {
1984
        // Braucht der Hafen noch Schiffe?
1985
        if(harbor->GetNumNeededShips() == 0)
35✔
1986
            continue;
35✔
1987

1988
        // Anzahl der Schiffe ermitteln, die diesen Hafen bereits anfahren
1989
        unsigned ships_coming = GetShipsToHarbor(*harbor);
×
1990

1991
        // Evtl. kommen schon genug?
1992
        if(harbor->GetNumNeededShips() <= ships_coming)
×
1993
            continue;
×
1994

1995
        // liegen wir am gleichen Meer?
1996
        if(world.IsHarborAtSea(harbor->GetHarborPosID(), ship.GetSeaID()))
×
1997
        {
1998
            unsigned length;
1999
            std::vector<Direction> route;
×
2000

2001
            if(world.FindShipPathToHarbor(ship.GetPos(), harbor->GetHarborPosID(), ship.GetSeaID(), &route, &length))
×
2002
            {
2003
                // ship already there?
2004
                if(length == 0)
×
2005
                {
2006
                    harbor->ShipArrived(ship);
×
2007
                    return;
×
2008
                }
2009

2010
                // Punkte ausrechnen
2011
                int points = harbor->GetNeedForShip(ships_coming) - length;
×
2012
                if(points > best_points || !best)
×
2013
                {
2014
                    best = harbor;
×
2015
                    best_points = points;
×
2016
                    best_route = route;
×
2017
                }
2018
            }
2019
        }
2020
    }
2021

2022
    // Einen Hafen gefunden?
2023
    if(best)
33✔
UNCOV
2024
        ship.GoToHarbor(*best, best_route);
×
2025
}
2026

2027
unsigned GamePlayer::GetShipID(const noShip& ship) const
16✔
2028
{
2029
    return static_cast<unsigned>(helpers::indexOf(ships, &ship));
16✔
2030
}
2031

2032
noShip* GamePlayer::GetShipByID(const unsigned ship_id) const
25✔
2033
{
2034
    if(ship_id >= ships.size())
25✔
2035
        return nullptr;
×
2036
    else
2037
        return ships[ship_id];
25✔
2038
}
2039

2040
unsigned GamePlayer::GetShipsToHarbor(const nobHarborBuilding& hb) const
90✔
2041
{
2042
    unsigned count = 0;
90✔
2043
    for(const auto* ship : ships)
180✔
2044
    {
2045
        if(ship->IsGoingToHarbor(hb))
90✔
2046
            ++count;
6✔
2047
    }
2048

2049
    return count;
90✔
2050
}
2051

2052
bool GamePlayer::FindHarborForUnloading(noShip* ship, const MapPoint start, HarborId* goalHarborId,
6✔
2053
                                        std::vector<Direction>* route, nobHarborBuilding* exception)
2054
{
2055
    nobHarborBuilding* best = nullptr;
6✔
2056
    unsigned best_distance = 0xffffffff;
6✔
2057

2058
    for(nobHarborBuilding* hb : buildings.GetHarbors())
9✔
2059
    {
2060
        // Bestimmten Hafen ausschließen
2061
        if(hb == exception)
3✔
2062
            continue;
×
2063

2064
        // Prüfen, ob Hafen an das Meer, wo sich das Schiff gerade befindet, angrenzt
2065
        if(!world.IsHarborAtSea(hb->GetHarborPosID(), ship->GetSeaID()))
3✔
2066
            continue;
×
2067

2068
        // Distanz ermitteln zwischen Schiff und Hafen, Schiff kann natürlich auch über Kartenränder fahren
2069
        unsigned distance = world.CalcDistance(ship->GetPos(), hb->GetPos());
3✔
2070

2071
        // Kürzerer Weg als bisher bestes Ziel?
2072
        if(distance < best_distance)
3✔
2073
        {
2074
            best_distance = distance;
3✔
2075
            best = hb;
3✔
2076
        }
2077
    }
2078

2079
    // Hafen gefunden?
2080
    if(best)
6✔
2081
    {
2082
        // Weg dorthin suchen
2083
        route->clear();
3✔
2084
        *goalHarborId = best->GetHarborPosID();
3✔
2085
        if(world.FindShipPathToHarbor(start, best->GetHarborPosID(), ship->GetSeaID(), route, nullptr))
3✔
2086
            return true;
3✔
2087
    }
2088

2089
    return false;
3✔
2090
}
2091

2092
void GamePlayer::TestForEmergencyProgramm()
×
2093
{
2094
    // we are already defeated, do not even think about an emergency program - it's too late :-(
2095
    if(isDefeated)
×
2096
        return;
×
2097

2098
    // In Lagern vorhandene Bretter und Steine zählen
2099
    unsigned boards = 0;
×
2100
    unsigned stones = 0;
×
2101
    for(nobBaseWarehouse* wh : buildings.GetStorehouses())
×
2102
    {
2103
        boards += wh->GetInventory().goods[GoodType::Boards];
×
2104
        stones += wh->GetInventory().goods[GoodType::Stones];
×
2105
    }
2106

2107
    // Emergency happens, if we have less than 10 boards or stones...
2108
    bool isNewEmergency = boards <= 10 || stones <= 10;
×
2109
    // ...and no woddcutter or sawmill
2110
    isNewEmergency &=
×
2111
      buildings.GetBuildings(BuildingType::Woodcutter).empty() || buildings.GetBuildings(BuildingType::Sawmill).empty();
×
2112

2113
    // Wenn nötig, Notfallprogramm auslösen
2114
    if(isNewEmergency)
×
2115
    {
2116
        if(!emergency)
×
2117
        {
2118
            emergency = true;
×
2119
            SendPostMessage(std::make_unique<PostMsg>(
×
2120
              world.GetEvMgr().GetCurrentGF(), _("The emergency program has been activated."), PostCategory::Economy));
×
2121
        }
2122
    } else
2123
    {
2124
        // Sobald Notfall vorbei, Notfallprogramm beenden, evtl. Baustellen wieder mit Kram versorgen
2125
        if(emergency)
×
2126
        {
2127
            emergency = false;
×
2128
            SendPostMessage(std::make_unique<PostMsg>(world.GetEvMgr().GetCurrentGF(),
×
2129
                                                      _("The emergency program has been deactivated."),
×
2130
                                                      PostCategory::Economy));
×
2131
            FindMaterialForBuildingSites();
×
2132
        }
2133
    }
2134
}
2135

2136
void GamePlayer::TestPacts()
20✔
2137
{
2138
    for(unsigned i = 0; i < world.GetNumPlayers(); ++i)
80✔
2139
    {
2140
        if(i == GetPlayerId())
60✔
2141
            continue;
20✔
2142

2143
        for(const auto pact : helpers::enumRange<PactType>())
320✔
2144
        {
2145
            // Pact not running
2146
            if(pacts[i][pact].duration == 0)
80✔
2147
                continue;
60✔
2148
            if(GetPactState(pact, i) == PactState::None)
20✔
2149
            {
2150
                // Pact was running but is expired -> Cancel for both players
2151
                pacts[i][pact].duration = 0;
2✔
2152
                pacts[i][pact].accepted = false;
2✔
2153
                GamePlayer& otherPlayer = world.GetPlayer(i);
2✔
2154
                RTTR_Assert(otherPlayer.pacts[GetPlayerId()][pact].duration);
2✔
2155
                RTTR_Assert(otherPlayer.pacts[GetPlayerId()][pact].accepted);
2✔
2156
                otherPlayer.pacts[GetPlayerId()][pact].duration = 0;
2✔
2157
                otherPlayer.pacts[GetPlayerId()][pact].accepted = false;
2✔
2158
                // And notify
2159
                PactChanged(pact);
2✔
2160
                otherPlayer.PactChanged(pact);
2✔
2161
            }
2162
        }
2163
    }
2164
}
20✔
2165

2166
bool GamePlayer::CanBuildCatapult() const
1✔
2167
{
2168
    // Wenn AddonId::LIMIT_CATAPULTS nicht aktiv ist, bauen immer erlaubt
2169
    if(!world.GetGGS().isEnabled(AddonId::LIMIT_CATAPULTS)) //-V807
1✔
2170
        return true;
1✔
2171

2172
    BuildingCount bc = buildings.GetBuildingNums();
×
2173

2174
    unsigned max = 0;
×
2175
    // proportional?
2176
    if(world.GetGGS().getSelection(AddonId::LIMIT_CATAPULTS) == 1)
×
2177
    {
2178
        max = int(bc.buildings[BuildingType::Barracks] * 0.125 + bc.buildings[BuildingType::Guardhouse] * 0.25
×
2179
                  + bc.buildings[BuildingType::Watchtower] * 0.5 + bc.buildings[BuildingType::Fortress]
×
2180
                  + 0.111); // to avoid rounding errors
×
2181
    } else if(world.GetGGS().getSelection(AddonId::LIMIT_CATAPULTS) < 8)
×
2182
    {
2183
        const std::array<unsigned, 6> limits = {{0, 3, 5, 10, 20, 30}};
×
2184
        max = limits[world.GetGGS().getSelection(AddonId::LIMIT_CATAPULTS) - 2];
×
2185
    }
2186

2187
    return bc.buildings[BuildingType::Catapult] + bc.buildingSites[BuildingType::Catapult] < max;
×
2188
}
2189

2190
bool GamePlayer::ShipDiscoveredHostileTerritory(const MapPoint location)
25✔
2191
{
2192
    // Prüfen, ob Abstand zu bisherigen Punkten nicht zu klein
2193
    for(const auto& enemies_discovered_by_ship : enemies_discovered_by_ships)
25✔
2194
    {
2195
        if(world.CalcDistance(enemies_discovered_by_ship, location) < 30)
24✔
2196
            return false;
24✔
2197
    }
2198

2199
    // Nein? Dann haben wir ein neues Territorium gefunden
2200
    enemies_discovered_by_ships.push_back(location);
1✔
2201

2202
    return true;
1✔
2203
}
2204

2205
bool GamePlayer::IsDependentFigure(const noFigure& fig)
258✔
2206
{
2207
    for(const nobBaseWarehouse* wh : buildings.GetStorehouses())
623✔
2208
    {
2209
        if(wh->IsDependentFigure(fig))
365✔
2210
            return true;
×
2211
    }
2212
    return false;
258✔
2213
}
2214

2215
std::vector<nobBaseWarehouse*> GamePlayer::GetWarehousesForTrading(const nobBaseWarehouse& goalWh) const
18✔
2216
{
2217
    std::vector<nobBaseWarehouse*> result;
18✔
2218

2219
    // Don't try to trade with us!
2220
    if(goalWh.GetPlayer() == GetPlayerId())
18✔
2221
        return result;
6✔
2222

2223
    const MapPoint goalFlagPos = goalWh.GetFlagPos();
12✔
2224

2225
    TradePathCache& tradePathCache = world.GetTradePathCache();
12✔
2226
    for(nobBaseWarehouse* wh : buildings.GetStorehouses())
24✔
2227
    {
2228
        // Is there a trade path from this warehouse to wh? (flag to flag)
2229
        if(tradePathCache.pathExists(wh->GetFlagPos(), goalFlagPos, GetPlayerId()))
12✔
2230
            result.push_back(wh);
4✔
2231
    }
2232

2233
    return result;
12✔
2234
}
2235

2236
struct WarehouseDistanceComparator
2237
{
2238
    // Reference warehouse position, to which we want to calc the distance
2239
    const MapPoint refWareHousePos_;
2240
    /// GameWorld
2241
    const GameWorld& gwg_;
2242

2243
    WarehouseDistanceComparator(const nobBaseWarehouse& refWareHouse, const GameWorld& world)
15✔
2244
        : refWareHousePos_(refWareHouse.GetPos()), gwg_(world)
15✔
2245
    {}
15✔
2246

2247
    bool operator()(nobBaseWarehouse* const wh1, nobBaseWarehouse* const wh2) const
×
2248
    {
2249
        unsigned dist1 = gwg_.CalcDistance(wh1->GetPos(), refWareHousePos_);
×
2250
        unsigned dist2 = gwg_.CalcDistance(wh2->GetPos(), refWareHousePos_);
×
2251
        return (dist1 < dist2) || (dist1 == dist2 && wh1->GetObjId() < wh2->GetObjId());
×
2252
    }
2253
};
2254

2255
void GamePlayer::Trade(nobBaseWarehouse* goalWh, const boost_variant2<GoodType, Job>& what, unsigned count) const
23✔
2256
{
2257
    if(!world.GetGGS().isEnabled(AddonId::TRADE))
23✔
2258
        return;
11✔
2259

2260
    if(count == 0)
20✔
2261
        return;
3✔
2262

2263
    // Don't try to trade with us!
2264
    if(goalWh->GetPlayer() == GetPlayerId())
17✔
2265
        return;
1✔
2266

2267
    // No trades with enemies
2268
    if(!IsAlly(goalWh->GetPlayer()))
16✔
2269
        return;
1✔
2270

2271
    const MapPoint goalFlagPos = goalWh->GetFlagPos();
15✔
2272

2273
    std::vector<nobBaseWarehouse*> whs(buildings.GetStorehouses().begin(), buildings.GetStorehouses().end());
15✔
2274
    helpers::sort(whs, WarehouseDistanceComparator(*goalWh, world));
15✔
2275
    TradePathCache& tradePathCache = world.GetTradePathCache();
15✔
2276
    for(nobBaseWarehouse* wh : whs)
27✔
2277
    {
2278
        // Get available wares
2279
        const unsigned available =
2280
          boost::variant2::visit(composeVisitor([wh](GoodType gt) { return wh->GetAvailableWaresForTrading(gt); },
22✔
2281
                                                [wh](Job job) { return wh->GetAvailableFiguresForTrading(job); }),
23✔
2282
                                 what);
15✔
2283
        if(available == 0)
15✔
2284
            continue;
×
2285

2286
        const unsigned actualCount = std::min(available, count);
15✔
2287

2288
        // Find a trade path from flag to flag
2289
        TradeRoute tr(world, GetPlayerId(), wh->GetFlagPos(), goalFlagPos);
15✔
2290

2291
        // Found a path?
2292
        if(tr.IsValid())
15✔
2293
        {
2294
            // Add to cache for future searches
2295
            tradePathCache.addEntry(tr.GetTradePath(), GetPlayerId());
14✔
2296

2297
            wh->StartTradeCaravane(what, actualCount, tr, goalWh);
14✔
2298
            count -= available;
14✔
2299
            if(count == 0)
14✔
2300
                return;
3✔
2301
        }
2302
    }
2303
}
2304

2305
bool GamePlayer::IsBuildingEnabled(BuildingType type) const
1,036✔
2306
{
2307
    return building_enabled[type] || (isHuman() && world.GetGameInterface()->GI_GetCheats().areAllBuildingsEnabled());
1,036✔
2308
}
2309

2310
void GamePlayer::FillVisualSettings(VisualSettings& visualSettings) const
2✔
2311
{
2312
    Distributions& visDistribution = visualSettings.distribution;
2✔
2313
    unsigned visIdx = 0;
2✔
2314
    for(const DistributionMapping& mapping : distributionMap)
60✔
2315
    {
2316
        visDistribution[visIdx++] = distribution[std::get<0>(mapping)].percent_buildings[std::get<1>(mapping)];
58✔
2317
    }
2318

2319
    visualSettings.useCustomBuildOrder = useCustomBuildOrder_;
2✔
2320
    visualSettings.build_order = build_order;
2✔
2321

2322
    visualSettings.transport_order = GetOrderingFromTransportPrio(transportPrio);
2✔
2323

2324
    visualSettings.military_settings = militarySettings_;
2✔
2325
    visualSettings.tools_settings = toolsSettings_;
2✔
2326
}
2✔
2327

2328
#define INSTANTIATE_FINDWH(Cond)                                                                                \
2329
    template nobBaseWarehouse* GamePlayer::FindWarehouse(const noRoadNode&, const Cond&, bool, bool, unsigned*, \
2330
                                                         const RoadSegment*) const
2331

2332
INSTANTIATE_FINDWH(FW::HasMinWares);
2333
INSTANTIATE_FINDWH(FW::HasFigure);
2334
INSTANTIATE_FINDWH(FW::HasWareAndFigure);
2335
INSTANTIATE_FINDWH(FW::HasAnyMatchingSoldier);
2336
INSTANTIATE_FINDWH(FW::AcceptsWare);
2337
INSTANTIATE_FINDWH(FW::AcceptsFigure);
2338
INSTANTIATE_FINDWH(FW::CollectsWare);
2339
INSTANTIATE_FINDWH(FW::CollectsFigure);
2340
INSTANTIATE_FINDWH(FW::HasWareButNoCollect);
2341
INSTANTIATE_FINDWH(FW::HasFigureButNoCollect);
2342
INSTANTIATE_FINDWH(FW::AcceptsFigureButNoSend);
2343
INSTANTIATE_FINDWH(FW::NoCondition);
2344

2345
#undef INSTANTIATE_FINDWH
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc