2 * This file is part of OpenTTD.
3 * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
4 * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
5 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
8 /** @file group_cmd.cpp Handling of the engine groups */
11 #include "command_func.h"
13 #include "vehiclelist.h"
14 #include "vehicle_func.h"
15 #include "autoreplace_base.h"
16 #include "autoreplace_func.h"
17 #include "string_func.h"
18 #include "company_func.h"
19 #include "core/pool_func.hpp"
20 #include "order_backup.h"
21 #include "group_cmd.h"
23 #include "table/strings.h"
25 #include "safeguards.h"
27 GroupPool
_group_pool("Group");
28 INSTANTIATE_POOL_METHODS(Group
)
33 void GroupStatistics::Clear()
35 this->num_vehicle
= 0;
36 this->profit_last_year
= 0;
37 this->num_vehicle_min_age
= 0;
38 this->profit_last_year_min_age
= 0;
40 /* This is also called when NewGRF change. So the number of engines might have changed. Reset. */
41 this->num_engines
.clear();
45 * Get number of vehicles of a specific engine ID.
46 * @param engine Engine ID.
47 * @returns number of vehicles of this engine ID.
49 uint16_t GroupStatistics::GetNumEngines(EngineID engine
) const
51 auto found
= this->num_engines
.find(engine
);
52 if (found
!= std::end(this->num_engines
)) return found
->second
;
57 * Returns the GroupStatistics for a specific group.
58 * @param company Owner of the group.
59 * @param id_g GroupID of the group.
60 * @param type VehicleType of the vehicles in the group.
61 * @return Statistics for the group.
63 /* static */ GroupStatistics
&GroupStatistics::Get(CompanyID company
, GroupID id_g
, VehicleType type
)
65 if (Group::IsValidID(id_g
)) {
66 Group
*g
= Group::Get(id_g
);
67 assert(g
->owner
== company
);
68 assert(g
->vehicle_type
== type
);
72 if (IsDefaultGroupID(id_g
)) return Company::Get(company
)->group_default
[type
];
73 if (IsAllGroupID(id_g
)) return Company::Get(company
)->group_all
[type
];
79 * Returns the GroupStatistic for the group of a vehicle.
81 * @return GroupStatistics for the group of the vehicle.
83 /* static */ GroupStatistics
&GroupStatistics::Get(const Vehicle
*v
)
85 return GroupStatistics::Get(v
->owner
, v
->group_id
, v
->type
);
89 * Returns the GroupStatistic for the ALL_GROUPO of a vehicle type.
91 * @return GroupStatistics for the ALL_GROUP of the vehicle type.
93 /* static */ GroupStatistics
&GroupStatistics::GetAllGroup(const Vehicle
*v
)
95 return GroupStatistics::Get(v
->owner
, ALL_GROUP
, v
->type
);
99 * Update all caches after loading a game, changing NewGRF, etc.
101 /* static */ void GroupStatistics::UpdateAfterLoad()
103 /* Set up the engine count for all companies */
104 for (Company
*c
: Company::Iterate()) {
105 for (VehicleType type
= VEH_BEGIN
; type
< VEH_COMPANY_END
; type
++) {
106 c
->group_all
[type
].Clear();
107 c
->group_default
[type
].Clear();
112 for (Group
*g
: Group::Iterate()) {
113 g
->statistics
.Clear();
116 for (const Vehicle
*v
: Vehicle::Iterate()) {
117 if (!v
->IsEngineCountable()) continue;
119 GroupStatistics::CountEngine(v
, 1);
120 if (v
->IsPrimaryVehicle()) GroupStatistics::CountVehicle(v
, 1);
123 for (const Company
*c
: Company::Iterate()) {
124 GroupStatistics::UpdateAutoreplace(c
->index
);
129 * Update num_vehicle when adding or removing a vehicle.
130 * @param v Vehicle to count.
131 * @param delta +1 to add, -1 to remove.
133 /* static */ void GroupStatistics::CountVehicle(const Vehicle
*v
, int delta
)
135 assert(delta
== 1 || delta
== -1);
137 GroupStatistics
&stats_all
= GroupStatistics::GetAllGroup(v
);
138 GroupStatistics
&stats
= GroupStatistics::Get(v
);
140 stats_all
.num_vehicle
+= delta
;
141 stats_all
.profit_last_year
+= v
->GetDisplayProfitLastYear() * delta
;
142 stats
.num_vehicle
+= delta
;
143 stats
.profit_last_year
+= v
->GetDisplayProfitLastYear() * delta
;
145 if (v
->economy_age
> VEHICLE_PROFIT_MIN_AGE
) {
146 stats_all
.num_vehicle_min_age
+= delta
;
147 stats_all
.profit_last_year_min_age
+= v
->GetDisplayProfitLastYear() * delta
;
148 stats
.num_vehicle_min_age
+= delta
;
149 stats
.profit_last_year_min_age
+= v
->GetDisplayProfitLastYear() * delta
;
154 * Update num_engines when adding/removing an engine.
155 * @param v Engine to count.
156 * @param delta +1 to add, -1 to remove.
158 /* static */ void GroupStatistics::CountEngine(const Vehicle
*v
, int delta
)
160 assert(delta
== 1 || delta
== -1);
161 GroupStatistics::GetAllGroup(v
).num_engines
[v
->engine_type
] += delta
;
162 GroupStatistics::Get(v
).num_engines
[v
->engine_type
] += delta
;
166 * Add a vehicle's last year profit to the profit sum of its group.
168 /* static */ void GroupStatistics::AddProfitLastYear(const Vehicle
*v
)
170 GroupStatistics
&stats_all
= GroupStatistics::GetAllGroup(v
);
171 GroupStatistics
&stats
= GroupStatistics::Get(v
);
173 stats_all
.profit_last_year
+= v
->GetDisplayProfitLastYear();
174 stats
.profit_last_year
+= v
->GetDisplayProfitLastYear();
178 * Add a vehicle to the profit sum of its group.
180 /* static */ void GroupStatistics::VehicleReachedMinAge(const Vehicle
*v
)
182 GroupStatistics
&stats_all
= GroupStatistics::GetAllGroup(v
);
183 GroupStatistics
&stats
= GroupStatistics::Get(v
);
185 stats_all
.num_vehicle_min_age
++;
186 stats_all
.profit_last_year_min_age
+= v
->GetDisplayProfitLastYear();
187 stats
.num_vehicle_min_age
++;
188 stats
.profit_last_year_min_age
+= v
->GetDisplayProfitLastYear();
192 * Recompute the profits for all groups.
194 /* static */ void GroupStatistics::UpdateProfits()
196 /* Set up the engine count for all companies */
197 for (Company
*c
: Company::Iterate()) {
198 for (VehicleType type
= VEH_BEGIN
; type
< VEH_COMPANY_END
; type
++) {
199 c
->group_all
[type
].ClearProfits();
200 c
->group_default
[type
].ClearProfits();
205 for (Group
*g
: Group::Iterate()) {
206 g
->statistics
.ClearProfits();
209 for (const Vehicle
*v
: Vehicle::Iterate()) {
210 if (v
->IsPrimaryVehicle()) {
211 GroupStatistics::AddProfitLastYear(v
);
212 if (v
->economy_age
> VEHICLE_PROFIT_MIN_AGE
) GroupStatistics::VehicleReachedMinAge(v
);
218 * Update autoreplace_defined and autoreplace_finished of all statistics of a company.
219 * @param company Company to update statistics for.
221 /* static */ void GroupStatistics::UpdateAutoreplace(CompanyID company
)
223 /* Set up the engine count for all companies */
224 Company
*c
= Company::Get(company
);
225 for (VehicleType type
= VEH_BEGIN
; type
< VEH_COMPANY_END
; type
++) {
226 c
->group_all
[type
].ClearAutoreplace();
227 c
->group_default
[type
].ClearAutoreplace();
231 for (Group
*g
: Group::Iterate()) {
232 if (g
->owner
!= company
) continue;
233 g
->statistics
.ClearAutoreplace();
236 for (EngineRenewList erl
= c
->engine_renew_list
; erl
!= nullptr; erl
= erl
->next
) {
237 const Engine
*e
= Engine::Get(erl
->from
);
238 GroupStatistics
&stats
= GroupStatistics::Get(company
, erl
->group_id
, e
->type
);
239 if (!stats
.autoreplace_defined
) {
240 stats
.autoreplace_defined
= true;
241 stats
.autoreplace_finished
= true;
243 if (GetGroupNumEngines(company
, erl
->group_id
, erl
->from
) > 0) stats
.autoreplace_finished
= false;
248 * Update the num engines of a groupID. Decrease the old one and increase the new one
249 * @note called in SetTrainGroupID and UpdateTrainGroupID
250 * @param v Vehicle we have to update
251 * @param old_g index of the old group
252 * @param new_g index of the new group
254 static inline void UpdateNumEngineGroup(const Vehicle
*v
, GroupID old_g
, GroupID new_g
)
256 if (old_g
!= new_g
) {
257 /* Decrease the num engines in the old group */
258 GroupStatistics::Get(v
->owner
, old_g
, v
->type
).num_engines
[v
->engine_type
]--;
260 /* Increase the num engines in the new group */
261 GroupStatistics::Get(v
->owner
, new_g
, v
->type
).num_engines
[v
->engine_type
]++;
266 const Livery
*GetParentLivery(const Group
*g
)
268 if (g
->parent
== INVALID_GROUP
) {
269 const Company
*c
= Company::Get(g
->owner
);
270 return &c
->livery
[LS_DEFAULT
];
273 const Group
*pg
= Group::Get(g
->parent
);
279 * Propagate a livery change to a group's children, and optionally update cached vehicle colourmaps.
280 * @param g Group to propagate colours to children.
281 * @param reset_cache Reset colourmap of vehicles in this group.
283 static void PropagateChildLivery(const Group
*g
, bool reset_cache
)
286 /* Company colour data is indirectly cached. */
287 for (Vehicle
*v
: Vehicle::Iterate()) {
288 if (v
->group_id
== g
->index
&& (!v
->IsGroundVehicle() || v
->IsFrontEngine())) {
289 for (Vehicle
*u
= v
; u
!= nullptr; u
= u
->Next()) {
290 u
->colourmap
= PAL_NONE
;
291 u
->InvalidateNewGRFCache();
297 for (Group
*cg
: Group::Iterate()) {
298 if (cg
->parent
== g
->index
) {
299 if (!HasBit(cg
->livery
.in_use
, 0)) cg
->livery
.colour1
= g
->livery
.colour1
;
300 if (!HasBit(cg
->livery
.in_use
, 1)) cg
->livery
.colour2
= g
->livery
.colour2
;
301 PropagateChildLivery(cg
, reset_cache
);
307 * Update group liveries for a company. This is called when the LS_DEFAULT scheme is changed, to update groups with
308 * colours set to default.
309 * @param c Company to update.
311 void UpdateCompanyGroupLiveries(const Company
*c
)
313 for (Group
*g
: Group::Iterate()) {
314 if (g
->owner
== c
->index
&& g
->parent
== INVALID_GROUP
) {
315 if (!HasBit(g
->livery
.in_use
, 0)) g
->livery
.colour1
= c
->livery
[LS_DEFAULT
].colour1
;
316 if (!HasBit(g
->livery
.in_use
, 1)) g
->livery
.colour2
= c
->livery
[LS_DEFAULT
].colour2
;
317 PropagateChildLivery(g
, false);
322 Group::Group(Owner owner
)
325 this->folded
= false;
330 * Create a new vehicle group.
331 * @param flags type of operation
332 * @param vt vehicle type
333 * @param parent_group parent groupid
334 * @return the cost of this operation or an error
336 std::tuple
<CommandCost
, GroupID
> CmdCreateGroup(DoCommandFlag flags
, VehicleType vt
, GroupID parent_group
)
338 if (!IsCompanyBuildableVehicleType(vt
)) return { CMD_ERROR
, INVALID_GROUP
};
340 if (!Group::CanAllocateItem()) return { CMD_ERROR
, INVALID_GROUP
};
342 const Group
*pg
= Group::GetIfValid(parent_group
);
344 if (pg
->owner
!= _current_company
) return { CMD_ERROR
, INVALID_GROUP
};
345 if (pg
->vehicle_type
!= vt
) return { CMD_ERROR
, INVALID_GROUP
};
348 if (flags
& DC_EXEC
) {
349 Group
*g
= new Group(_current_company
);
350 g
->vehicle_type
= vt
;
351 g
->parent
= INVALID_GROUP
;
353 Company
*c
= Company::Get(g
->owner
);
354 g
->number
= c
->freegroups
.UseID(c
->freegroups
.NextID());
356 g
->livery
.colour1
= c
->livery
[LS_DEFAULT
].colour1
;
357 g
->livery
.colour2
= c
->livery
[LS_DEFAULT
].colour2
;
358 if (c
->settings
.renew_keep_length
) SetBit(g
->flags
, GroupFlags::GF_REPLACE_WAGON_REMOVAL
);
360 g
->parent
= pg
->index
;
361 g
->livery
.colour1
= pg
->livery
.colour1
;
362 g
->livery
.colour2
= pg
->livery
.colour2
;
363 g
->flags
= pg
->flags
;
366 InvalidateWindowData(GetWindowClassForVehicleType(vt
), VehicleListIdentifier(VL_GROUP_LIST
, vt
, _current_company
).Pack());
367 InvalidateWindowData(WC_COMPANY_COLOUR
, g
->owner
, g
->vehicle_type
);
369 return { CommandCost(), g
->index
};
372 return { CommandCost(), INVALID_GROUP
};
377 * Add all vehicles in the given group to the default group and then deletes the group.
378 * @param flags type of operation
379 * @param group_id index of group
380 * @return the cost of this operation or an error
382 CommandCost
CmdDeleteGroup(DoCommandFlag flags
, GroupID group_id
)
384 Group
*g
= Group::GetIfValid(group_id
);
385 if (g
== nullptr || g
->owner
!= _current_company
) return CMD_ERROR
;
387 /* Remove all vehicles from the group */
388 Command
<CMD_REMOVE_ALL_VEHICLES_GROUP
>::Do(flags
, group_id
);
390 /* Delete sub-groups */
391 for (const Group
*gp
: Group::Iterate()) {
392 if (gp
->parent
== g
->index
) {
393 Command
<CMD_DELETE_GROUP
>::Do(flags
, gp
->index
);
397 if (flags
& DC_EXEC
) {
398 /* Update backupped orders if needed */
399 OrderBackup::ClearGroup(g
->index
);
401 if (g
->owner
< MAX_COMPANIES
) {
402 Company
*c
= Company::Get(g
->owner
);
404 /* If we set an autoreplace for the group we delete, remove it. */
405 for (EngineRenew
*er
: EngineRenew::Iterate()) {
406 if (er
->group_id
== g
->index
) RemoveEngineReplacementForCompany(c
, er
->from
, g
->index
, flags
);
409 c
->freegroups
.ReleaseID(g
->number
);
412 VehicleType vt
= g
->vehicle_type
;
414 /* Delete the Replace Vehicle Windows */
415 CloseWindowById(WC_REPLACE_VEHICLE
, g
->vehicle_type
);
418 InvalidateWindowData(GetWindowClassForVehicleType(vt
), VehicleListIdentifier(VL_GROUP_LIST
, vt
, _current_company
).Pack());
419 InvalidateWindowData(WC_COMPANY_COLOUR
, _current_company
, vt
);
422 return CommandCost();
427 * @param flags type of operation
428 * @param mode Operation to perform.
429 * @param group_id GroupID
430 * @param parent_id parent group index
431 * @param text the new name or an empty string when resetting to the default
432 * @return the cost of this operation or an error
434 CommandCost
CmdAlterGroup(DoCommandFlag flags
, AlterGroupMode mode
, GroupID group_id
, GroupID parent_id
, const std::string
&text
)
436 Group
*g
= Group::GetIfValid(group_id
);
437 if (g
== nullptr || g
->owner
!= _current_company
) return CMD_ERROR
;
439 if (mode
== AlterGroupMode::Rename
) {
441 bool reset
= text
.empty();
444 if (Utf8StringLength(text
) >= MAX_LENGTH_GROUP_NAME_CHARS
) return CMD_ERROR
;
447 if (flags
& DC_EXEC
) {
448 /* Assign the new one */
455 } else if (mode
== AlterGroupMode::SetParent
) {
456 /* Set group parent */
457 const Group
*pg
= Group::GetIfValid(parent_id
);
460 if (pg
->owner
!= _current_company
) return CMD_ERROR
;
461 if (pg
->vehicle_type
!= g
->vehicle_type
) return CMD_ERROR
;
463 /* Ensure request parent isn't child of group.
464 * This is the only place that infinite loops are prevented. */
465 if (GroupIsInGroup(pg
->index
, g
->index
)) return_cmd_error(STR_ERROR_GROUP_CAN_T_SET_PARENT_RECURSION
);
468 if (flags
& DC_EXEC
) {
469 g
->parent
= (pg
== nullptr) ? INVALID_GROUP
: pg
->index
;
470 GroupStatistics::UpdateAutoreplace(g
->owner
);
472 if (!HasBit(g
->livery
.in_use
, 0) || !HasBit(g
->livery
.in_use
, 1)) {
473 /* Update livery with new parent's colours if either colour is default. */
474 const Livery
*livery
= GetParentLivery(g
);
475 if (!HasBit(g
->livery
.in_use
, 0)) g
->livery
.colour1
= livery
->colour1
;
476 if (!HasBit(g
->livery
.in_use
, 1)) g
->livery
.colour2
= livery
->colour2
;
478 PropagateChildLivery(g
, true);
479 MarkWholeScreenDirty();
486 if (flags
& DC_EXEC
) {
487 InvalidateWindowData(WC_REPLACE_VEHICLE
, g
->vehicle_type
, 1);
488 InvalidateWindowData(GetWindowClassForVehicleType(g
->vehicle_type
), VehicleListIdentifier(VL_GROUP_LIST
, g
->vehicle_type
, _current_company
).Pack());
489 InvalidateWindowData(WC_COMPANY_COLOUR
, g
->owner
, g
->vehicle_type
);
490 InvalidateWindowClassesData(WC_VEHICLE_VIEW
);
491 InvalidateWindowClassesData(WC_VEHICLE_DETAILS
);
494 return CommandCost();
499 * Do add a vehicle to a group.
500 * @param v Vehicle to add.
501 * @param new_g Group to add to.
503 static void AddVehicleToGroup(Vehicle
*v
, GroupID new_g
)
505 GroupStatistics::CountVehicle(v
, -1);
508 default: NOT_REACHED();
510 SetTrainGroupID(Train::From(v
), new_g
);
516 if (v
->IsEngineCountable()) UpdateNumEngineGroup(v
, v
->group_id
, new_g
);
518 for (Vehicle
*u
= v
; u
!= nullptr; u
= u
->Next()) {
519 u
->colourmap
= PAL_NONE
;
520 u
->InvalidateNewGRFCache();
521 u
->UpdateViewport(true);
526 GroupStatistics::CountVehicle(v
, 1);
530 * Add a vehicle to a group
531 * @param flags type of operation
532 * @param group_id index of group
533 * @param veh_id vehicle to add to a group
534 * @param add_shared Add shared vehicles as well.
535 * @return the cost of this operation or an error
537 std::tuple
<CommandCost
, GroupID
> CmdAddVehicleGroup(DoCommandFlag flags
, GroupID group_id
, VehicleID veh_id
, bool add_shared
, const VehicleListIdentifier
&vli
)
539 GroupID new_g
= group_id
;
540 if (!Group::IsValidID(new_g
) && !IsDefaultGroupID(new_g
) && new_g
!= NEW_GROUP
) return { CMD_ERROR
, INVALID_GROUP
};
543 if (veh_id
== INVALID_VEHICLE
&& vli
.Valid()) {
544 if (!GenerateVehicleSortList(&list
, vli
) || list
.empty()) return { CMD_ERROR
, INVALID_GROUP
};
546 Vehicle
*v
= Vehicle::GetIfValid(veh_id
);
547 if (v
== nullptr) return { CMD_ERROR
, INVALID_GROUP
};
551 VehicleType vtype
= list
.front()->type
;
552 for (const Vehicle
*v
: list
) {
553 if (v
->owner
!= _current_company
|| !v
->IsPrimaryVehicle()) return { CMD_ERROR
, INVALID_GROUP
};
556 if (Group::IsValidID(new_g
)) {
557 Group
*g
= Group::Get(new_g
);
558 if (g
->owner
!= _current_company
|| g
->vehicle_type
!= vtype
) return { CMD_ERROR
, INVALID_GROUP
};
561 if (new_g
== NEW_GROUP
) {
562 /* Create new group. */
563 auto [ret
, new_group_id
] = CmdCreateGroup(flags
, vtype
, INVALID_GROUP
);
564 if (ret
.Failed()) return { ret
, new_group_id
};
566 new_g
= new_group_id
;
569 if (flags
& DC_EXEC
) {
570 for (const Vehicle
*vc
: list
) {
571 /* VehicleList is const but we need to modify the vehicle. */
572 Vehicle
*v
= Vehicle::Get(vc
->index
);
573 AddVehicleToGroup(v
, new_g
);
576 /* Add vehicles in the shared order list as well. */
577 for (Vehicle
*v2
= v
->FirstShared(); v2
!= nullptr; v2
= v2
->NextShared()) {
578 if (v2
->group_id
!= new_g
) AddVehicleToGroup(v2
, new_g
);
582 SetWindowDirty(WC_VEHICLE_DEPOT
, v
->tile
);
583 SetWindowDirty(WC_VEHICLE_VIEW
, v
->index
);
584 SetWindowDirty(WC_VEHICLE_DETAILS
, v
->index
);
585 InvalidateWindowData(WC_VEHICLE_VIEW
, v
->index
);
586 InvalidateWindowData(WC_VEHICLE_DETAILS
, v
->index
);
589 GroupStatistics::UpdateAutoreplace(_current_company
);
591 /* Update the Replace Vehicle Windows */
592 SetWindowDirty(WC_REPLACE_VEHICLE
, vtype
);
593 InvalidateWindowData(GetWindowClassForVehicleType(vtype
), VehicleListIdentifier(VL_GROUP_LIST
, vtype
, _current_company
).Pack());
596 return { CommandCost(), new_g
};
600 * Add all shared vehicles of all vehicles from a group
601 * @param flags type of operation
602 * @param id_g index of group
603 * @param type type of vehicles
604 * @return the cost of this operation or an error
606 CommandCost
CmdAddSharedVehicleGroup(DoCommandFlag flags
, GroupID id_g
, VehicleType type
)
608 if (!Group::IsValidID(id_g
) || !IsCompanyBuildableVehicleType(type
)) return CMD_ERROR
;
610 if (flags
& DC_EXEC
) {
611 /* Find the first front engine which belong to the group id_g
612 * then add all shared vehicles of this front engine to the group id_g */
613 for (const Vehicle
*v
: Vehicle::Iterate()) {
614 if (v
->type
== type
&& v
->IsPrimaryVehicle()) {
615 if (v
->group_id
!= id_g
) continue;
617 /* For each shared vehicles add it to the group */
618 for (Vehicle
*v2
= v
->FirstShared(); v2
!= nullptr; v2
= v2
->NextShared()) {
619 if (v2
->group_id
!= id_g
) Command
<CMD_ADD_VEHICLE_GROUP
>::Do(flags
, id_g
, v2
->index
, false, VehicleListIdentifier
{});
624 InvalidateWindowData(GetWindowClassForVehicleType(type
), VehicleListIdentifier(VL_GROUP_LIST
, type
, _current_company
).Pack());
627 return CommandCost();
632 * Remove all vehicles from a group
633 * @param flags type of operation
634 * @param group_id index of group
635 * @return the cost of this operation or an error
637 CommandCost
CmdRemoveAllVehiclesGroup(DoCommandFlag flags
, GroupID group_id
)
639 Group
*g
= Group::GetIfValid(group_id
);
641 if (g
== nullptr || g
->owner
!= _current_company
) return CMD_ERROR
;
643 if (flags
& DC_EXEC
) {
644 /* Find each Vehicle that belongs to the group old_g and add it to the default group */
645 for (const Vehicle
*v
: Vehicle::Iterate()) {
646 if (v
->IsPrimaryVehicle()) {
647 if (v
->group_id
!= group_id
) continue;
649 /* Add The Vehicle to the default group */
650 Command
<CMD_ADD_VEHICLE_GROUP
>::Do(flags
, DEFAULT_GROUP
, v
->index
, false, VehicleListIdentifier
{});
654 InvalidateWindowData(GetWindowClassForVehicleType(g
->vehicle_type
), VehicleListIdentifier(VL_GROUP_LIST
, g
->vehicle_type
, _current_company
).Pack());
657 return CommandCost();
661 * Set the livery for a vehicle group.
662 * @param flags Command flags.
663 * @param group_id Group ID.
664 * @param primary Set primary instead of secondary colour
665 * @param colour Colour.
667 CommandCost
CmdSetGroupLivery(DoCommandFlag flags
, GroupID group_id
, bool primary
, Colours colour
)
669 Group
*g
= Group::GetIfValid(group_id
);
671 if (g
== nullptr || g
->owner
!= _current_company
) return CMD_ERROR
;
673 if (colour
>= COLOUR_END
&& colour
!= INVALID_COLOUR
) return CMD_ERROR
;
675 if (flags
& DC_EXEC
) {
677 AssignBit(g
->livery
.in_use
, 0, colour
!= INVALID_COLOUR
);
678 if (colour
== INVALID_COLOUR
) colour
= GetParentLivery(g
)->colour1
;
679 g
->livery
.colour1
= colour
;
681 AssignBit(g
->livery
.in_use
, 1, colour
!= INVALID_COLOUR
);
682 if (colour
== INVALID_COLOUR
) colour
= GetParentLivery(g
)->colour2
;
683 g
->livery
.colour2
= colour
;
686 PropagateChildLivery(g
, true);
687 MarkWholeScreenDirty();
690 return CommandCost();
694 * Set group flag for a group and its sub-groups.
695 * @param g initial group.
696 * @param set 1 to set or 0 to clear protection.
698 static void SetGroupFlag(Group
*g
, GroupFlags flag
, bool set
, bool children
)
701 SetBit(g
->flags
, flag
);
703 ClrBit(g
->flags
, flag
);
706 if (!children
) return;
708 for (Group
*pg
: Group::Iterate()) {
709 if (pg
->parent
== g
->index
) SetGroupFlag(pg
, flag
, set
, true);
714 * (Un)set group flag from a group
715 * @param flags type of operation
716 * @param group_id index of group array
717 * @param flag flag to set, by value not bit.
718 * @param value value to set the flag to.
719 * @param recursive to apply to sub-groups.
720 * @return the cost of this operation or an error
722 CommandCost
CmdSetGroupFlag(DoCommandFlag flags
, GroupID group_id
, GroupFlags flag
, bool value
, bool recursive
)
724 Group
*g
= Group::GetIfValid(group_id
);
725 if (g
== nullptr || g
->owner
!= _current_company
) return CMD_ERROR
;
727 if (flag
>= GroupFlags::GF_END
) return CMD_ERROR
;
729 if (flags
& DC_EXEC
) {
730 SetGroupFlag(g
, flag
, value
, recursive
);
732 SetWindowDirty(GetWindowClassForVehicleType(g
->vehicle_type
), VehicleListIdentifier(VL_GROUP_LIST
, g
->vehicle_type
, _current_company
).Pack());
733 InvalidateWindowData(WC_REPLACE_VEHICLE
, g
->vehicle_type
);
736 return CommandCost();
740 * Affect the groupID of a train to new_g.
741 * @note called in CmdAddVehicleGroup and CmdMoveRailVehicle
742 * @param v First vehicle of the chain.
743 * @param new_g index of array group
745 void SetTrainGroupID(Train
*v
, GroupID new_g
)
747 if (!Group::IsValidID(new_g
) && !IsDefaultGroupID(new_g
)) return;
749 assert(v
->IsFrontEngine() || IsDefaultGroupID(new_g
));
751 for (Vehicle
*u
= v
; u
!= nullptr; u
= u
->Next()) {
752 if (u
->IsEngineCountable()) UpdateNumEngineGroup(u
, u
->group_id
, new_g
);
755 u
->colourmap
= PAL_NONE
;
756 u
->InvalidateNewGRFCache();
757 u
->UpdateViewport(true);
760 /* Update the Replace Vehicle Windows */
761 GroupStatistics::UpdateAutoreplace(v
->owner
);
762 SetWindowDirty(WC_REPLACE_VEHICLE
, VEH_TRAIN
);
767 * Recalculates the groupID of a train. Should be called each time a vehicle is added
768 * to/removed from the chain,.
769 * @note this needs to be called too for 'wagon chains' (in the depot, without an engine)
770 * @note Called in CmdBuildRailVehicle, CmdBuildRailWagon, CmdMoveRailVehicle, CmdSellRailWagon
771 * @param v First vehicle of the chain.
773 void UpdateTrainGroupID(Train
*v
)
775 assert(v
->IsFrontEngine() || v
->IsFreeWagon());
777 GroupID new_g
= v
->IsFrontEngine() ? v
->group_id
: (GroupID
)DEFAULT_GROUP
;
778 for (Vehicle
*u
= v
; u
!= nullptr; u
= u
->Next()) {
779 if (u
->IsEngineCountable()) UpdateNumEngineGroup(u
, u
->group_id
, new_g
);
782 u
->colourmap
= PAL_NONE
;
783 u
->InvalidateNewGRFCache();
786 /* Update the Replace Vehicle Windows */
787 GroupStatistics::UpdateAutoreplace(v
->owner
);
788 SetWindowDirty(WC_REPLACE_VEHICLE
, VEH_TRAIN
);
792 * Get the number of engines with EngineID id_e in the group with GroupID
793 * id_g and its sub-groups.
794 * @param company The company the group belongs to
795 * @param id_g The GroupID of the group used
796 * @param id_e The EngineID of the engine to count
797 * @return The number of engines with EngineID id_e in the group
799 uint
GetGroupNumEngines(CompanyID company
, GroupID id_g
, EngineID id_e
)
802 const Engine
*e
= Engine::Get(id_e
);
803 for (const Group
*g
: Group::Iterate()) {
804 if (g
->parent
== id_g
) count
+= GetGroupNumEngines(company
, g
->index
, id_e
);
806 return count
+ GroupStatistics::Get(company
, id_g
, e
->type
).GetNumEngines(id_e
);
810 * Get the number of vehicles in the group with GroupID
811 * id_g and its sub-groups.
812 * @param company The company the group belongs to
813 * @param id_g The GroupID of the group used
814 * @param type The vehicle type of the group
815 * @return The number of vehicles in the group
817 uint
GetGroupNumVehicle(CompanyID company
, GroupID id_g
, VehicleType type
)
820 for (const Group
*g
: Group::Iterate()) {
821 if (g
->parent
== id_g
) count
+= GetGroupNumVehicle(company
, g
->index
, type
);
823 return count
+ GroupStatistics::Get(company
, id_g
, type
).num_vehicle
;
827 * Get the number of vehicles above profit minimum age in the group with GroupID
828 * id_g and its sub-groups.
829 * @param company The company the group belongs to
830 * @param id_g The GroupID of the group used
831 * @param type The vehicle type of the group
832 * @return The number of vehicles above profit minimum age in the group
834 uint
GetGroupNumVehicleMinAge(CompanyID company
, GroupID id_g
, VehicleType type
)
837 for (const Group
*g
: Group::Iterate()) {
838 if (g
->parent
== id_g
) count
+= GetGroupNumVehicleMinAge(company
, g
->index
, type
);
840 return count
+ GroupStatistics::Get(company
, id_g
, type
).num_vehicle_min_age
;
844 * Get last year's profit of vehicles above minimum age
845 * for the group with GroupID id_g and its sub-groups.
846 * @param company The company the group belongs to
847 * @param id_g The GroupID of the group used
848 * @param type The vehicle type of the group
849 * @return Last year's profit of vehicles above minimum age for the group
851 Money
GetGroupProfitLastYearMinAge(CompanyID company
, GroupID id_g
, VehicleType type
)
854 for (const Group
*g
: Group::Iterate()) {
855 if (g
->parent
== id_g
) sum
+= GetGroupProfitLastYearMinAge(company
, g
->index
, type
);
857 return sum
+ GroupStatistics::Get(company
, id_g
, type
).profit_last_year_min_age
;
860 void RemoveAllGroupsForCompany(const CompanyID company
)
862 for (Group
*g
: Group::Iterate()) {
863 if (company
== g
->owner
) delete g
;
869 * Test if GroupID group is a descendant of (or is) GroupID search
870 * @param search The GroupID to search in
871 * @param group The GroupID to search for
872 * @return True iff group is search or a descendant of search
874 bool GroupIsInGroup(GroupID search
, GroupID group
)
876 if (!Group::IsValidID(search
)) return search
== group
;
879 if (search
== group
) return true;
880 search
= Group::Get(search
)->parent
;
881 } while (search
!= INVALID_GROUP
);