OpenRCT2/src/openrct2-ui/windows/StaffList.cpp

768 lines
30 KiB
C++
Raw Normal View History

/*****************************************************************************
* Copyright (c) 2014-2024 OpenRCT2 developers
*
* For a complete list of all authors, please refer to contributors.md
* Interested in contributing? Visit https://github.com/OpenRCT2/OpenRCT2
*
* OpenRCT2 is licensed under the GNU General Public License version 3.
*****************************************************************************/
2017-10-16 00:13:22 +02:00
#include <limits>
2018-06-22 23:21:44 +02:00
#include <openrct2-ui/interface/Dropdown.h>
#include <openrct2-ui/interface/Viewport.h>
#include <openrct2-ui/interface/Widget.h>
2017-08-12 23:06:12 +02:00
#include <openrct2-ui/windows/Window.h>
#include <openrct2/Context.h>
2017-11-30 18:17:06 +01:00
#include <openrct2/Game.h>
#include <openrct2/GameState.h>
2017-12-12 14:52:57 +01:00
#include <openrct2/Input.h>
#include <openrct2/actions/PeepPickupAction.h>
Split actions hpp files into separate h and cpp files (#13548) * Split up SmallSceneryPlace/Remove Added undo function for Remove Scenery * Refactor: Balloon and Banner actions hpp=>h/cpp * Refactor: rename all action *.hpp files to *.cpp This is preparation for separation in later commits. Note that without the complete set of commits in this branch, the code will not build. * Refactor Clear, Climate, Custom, and Footpath actions hpp=>h/cpp * VSCode: add src subdirectories to includePath * Refactor Guest actions hpp=>h/cpp * Refactor Land actions hpp=>h/cpp * Refactor LargeScenery actions hpp=>h/cpp * Refactor Load, Maze, Network actions hpp=>h/cpp * Refactor Park actions hpp=>h/cpp * Refactor/style: move private function declarations in actions *.h Previous action .h files included private function declarations with private member variables, before public function declarations. This commit re-orders the header files to the following order: - public member variables - private member variables - public functions - private functions * Refactor Pause action hpp=>h/cpp * Refactor Peep, Place, Player actions hpp=>h/cpp * Refactor Ride actions hpp=>h/cpp * Refactor Scenario, Set*, Sign* actions hpp=>h/cpp * Refactor SmallScenerySetColourAction hpp=>h/cpp * Refactor Staff actions hpp=>h/cpp * Refactor Surface, Tile, Track* actions hpp=>h/cpp * Refactor Wall and Water actions hpp=>h/cpp * Fix various includes and other compile errors Update includes for tests. Move static function declarations to .h files Add explicit includes to various files that were previously implicit (the required header was a nested include in an action hpp file, and the action .h file does not include that header) Move RideSetStatus string enum to the cpp file to avoid unused imports * Xcode: modify project file for actions refactor * Cleanup whitespace and end-of-file newlines Co-authored-by: duncanspumpkin <duncans_pumpkin@hotmail.co.uk>
2020-12-10 07:39:10 +01:00
#include <openrct2/actions/StaffFireAction.h>
2021-12-03 16:37:39 +01:00
#include <openrct2/actions/StaffHireNewAction.h>
Split actions hpp files into separate h and cpp files (#13548) * Split up SmallSceneryPlace/Remove Added undo function for Remove Scenery * Refactor: Balloon and Banner actions hpp=>h/cpp * Refactor: rename all action *.hpp files to *.cpp This is preparation for separation in later commits. Note that without the complete set of commits in this branch, the code will not build. * Refactor Clear, Climate, Custom, and Footpath actions hpp=>h/cpp * VSCode: add src subdirectories to includePath * Refactor Guest actions hpp=>h/cpp * Refactor Land actions hpp=>h/cpp * Refactor LargeScenery actions hpp=>h/cpp * Refactor Load, Maze, Network actions hpp=>h/cpp * Refactor Park actions hpp=>h/cpp * Refactor/style: move private function declarations in actions *.h Previous action .h files included private function declarations with private member variables, before public function declarations. This commit re-orders the header files to the following order: - public member variables - private member variables - public functions - private functions * Refactor Pause action hpp=>h/cpp * Refactor Peep, Place, Player actions hpp=>h/cpp * Refactor Ride actions hpp=>h/cpp * Refactor Scenario, Set*, Sign* actions hpp=>h/cpp * Refactor SmallScenerySetColourAction hpp=>h/cpp * Refactor Staff actions hpp=>h/cpp * Refactor Surface, Tile, Track* actions hpp=>h/cpp * Refactor Wall and Water actions hpp=>h/cpp * Fix various includes and other compile errors Update includes for tests. Move static function declarations to .h files Add explicit includes to various files that were previously implicit (the required header was a nested include in an action hpp file, and the action .h file does not include that header) Move RideSetStatus string enum to the cpp file to avoid unused imports * Xcode: modify project file for actions refactor * Cleanup whitespace and end-of-file newlines Co-authored-by: duncanspumpkin <duncans_pumpkin@hotmail.co.uk>
2020-12-10 07:39:10 +01:00
#include <openrct2/actions/StaffSetColourAction.h>
2018-06-22 23:21:44 +02:00
#include <openrct2/config/Config.h>
#include <openrct2/drawing/Drawing.h>
2021-11-24 15:48:33 +01:00
#include <openrct2/entity/EntityList.h>
2021-11-24 15:58:01 +01:00
#include <openrct2/entity/EntityRegistry.h>
2022-03-08 01:14:52 +01:00
#include <openrct2/entity/PatrolArea.h>
2021-11-25 22:47:24 +01:00
#include <openrct2/entity/Staff.h>
2021-12-12 00:06:06 +01:00
#include <openrct2/localisation/Formatter.h>
2018-01-06 18:32:25 +01:00
#include <openrct2/localisation/Localisation.h>
2018-06-22 23:21:44 +02:00
#include <openrct2/management/Finance.h>
2024-04-14 16:51:54 +02:00
#include <openrct2/peep/PeepAnimationData.h>
#include <openrct2/sprites.h>
2017-12-13 13:02:24 +01:00
#include <openrct2/util/Util.h>
2018-06-22 23:21:44 +02:00
#include <openrct2/windows/Intent.h>
2018-01-11 10:59:26 +01:00
#include <openrct2/world/Footpath.h>
2018-03-19 23:28:40 +01:00
#include <openrct2/world/Park.h>
#include <vector>
namespace OpenRCT2::Ui::Windows
{
enum
{
WINDOW_STAFF_LIST_TAB_HANDYMEN,
WINDOW_STAFF_LIST_TAB_MECHANICS,
WINDOW_STAFF_LIST_TAB_SECURITY,
WINDOW_STAFF_LIST_TAB_ENTERTAINERS
};
enum WindowStaffListWidgetIdx
{
WIDX_STAFF_LIST_BACKGROUND,
WIDX_STAFF_LIST_TITLE,
WIDX_STAFF_LIST_CLOSE,
WIDX_STAFF_LIST_TAB_CONTENT_PANEL,
WIDX_STAFF_LIST_HANDYMEN_TAB,
WIDX_STAFF_LIST_MECHANICS_TAB,
WIDX_STAFF_LIST_SECURITY_TAB,
WIDX_STAFF_LIST_ENTERTAINERS_TAB,
WIDX_STAFF_LIST_LIST,
WIDX_STAFF_LIST_UNIFORM_COLOUR_PICKER,
WIDX_STAFF_LIST_HIRE_BUTTON,
WIDX_STAFF_LIST_QUICK_FIRE,
WIDX_STAFF_LIST_SHOW_PATROL_AREA_BUTTON,
WIDX_STAFF_LIST_MAP,
};
static constexpr StringId WINDOW_TITLE = STR_STAFF;
static constexpr int32_t WW = 320;
static constexpr int32_t WH = 270;
constexpr int32_t MAX_WW = 500;
constexpr int32_t MAX_WH = 450;
// clang-format off
static Widget _staffListWidgets[] = {
2020-05-09 16:44:21 +02:00
WINDOW_SHIM(WINDOW_TITLE, WW, WH),
MakeWidget({ 0, 43}, { WW, WH - 43}, WindowWidgetType::Resize, WindowColour::Secondary ), // tab content panel
MakeTab ({ 3, 17}, STR_STAFF_HANDYMEN_TAB_TIP ), // handymen tab
MakeTab ({ 34, 17}, STR_STAFF_MECHANICS_TAB_TIP ), // mechanics tab
MakeTab ({ 65, 17}, STR_STAFF_SECURITY_TAB_TIP ), // security guards tab
MakeTab ({ 96, 17}, STR_STAFF_ENTERTAINERS_TAB_TIP), // entertainers tab
MakeWidget({ 3, 72}, {WW - 6, 195}, WindowWidgetType::Scroll, WindowColour::Secondary, SCROLL_VERTICAL ), // staff list
MakeWidget({130, 58}, { 12, 12}, WindowWidgetType::ColourBtn, WindowColour::Secondary, STR_NONE, STR_UNIFORM_COLOUR_TIP ), // uniform colour picker
MakeWidget({165, 17}, { 145, 13}, WindowWidgetType::Button, WindowColour::Primary , STR_NONE, STR_HIRE_STAFF_TIP ), // hire button
MakeWidget({243, 46}, { 24, 24}, WindowWidgetType::FlatBtn, WindowColour::Secondary, ImageId(SPR_DEMOLISH), STR_QUICK_FIRE_STAFF ), // quick fire staff
MakeWidget({267, 46}, { 24, 24}, WindowWidgetType::FlatBtn, WindowColour::Secondary, ImageId(SPR_PATROL_BTN), STR_SHOW_PATROL_AREA_TIP ), // show staff patrol area tool
MakeWidget({291, 46}, { 24, 24}, WindowWidgetType::FlatBtn, WindowColour::Secondary, ImageId(SPR_MAP), STR_SHOW_STAFF_ON_MAP_TIP ), // show staff on map button
kWidgetsEnd,
};
// clang-format on
class StaffListWindow final : public Window
{
private:
struct StaffNamingConvention
{
StringId Plural;
StringId Singular;
StringId ActionHire;
};
struct StaffEntry
{
EntityId Id;
u8string Name;
};
std::vector<StaffEntry> _staffList;
bool _quickFireMode{};
std::optional<size_t> _highlightedIndex{};
int32_t _selectedTab{};
uint32_t _tabAnimationIndex{};
public:
void OnOpen() override
{
widgets = _staffListWidgets;
WindowInitScrollWidgets(*this);
widgets[WIDX_STAFF_LIST_UNIFORM_COLOUR_PICKER].type = WindowWidgetType::Empty;
min_width = WW;
min_height = WH;
max_width = MAX_WW;
max_height = MAX_WH;
RefreshList();
}
void OnClose() override
{
CancelTools();
}
void OnMouseUp(WidgetIndex widgetIndex) override
{
switch (widgetIndex)
{
case WIDX_STAFF_LIST_CLOSE:
Close();
break;
case WIDX_STAFF_LIST_HIRE_BUTTON:
{
auto staffType = GetSelectedStaffType();
auto costume = EntertainerCostume::Count;
if (staffType == StaffType::Entertainer)
{
costume = GetRandomEntertainerCostume();
}
HireNewMember(staffType, costume);
break;
}
case WIDX_STAFF_LIST_SHOW_PATROL_AREA_BUTTON:
if (!ToolSet(*this, WIDX_STAFF_LIST_SHOW_PATROL_AREA_BUTTON, Tool::Crosshair))
{
ShowGridlines();
SetPatrolAreaToRender(GetSelectedStaffType());
GfxInvalidateScreen();
}
break;
case WIDX_STAFF_LIST_MAP:
ContextOpenWindow(WindowClass::Map);
break;
case WIDX_STAFF_LIST_QUICK_FIRE:
_quickFireMode = !_quickFireMode;
Invalidate();
break;
}
}
void OnResize() override
{
min_width = WW;
min_height = WH;
if (width < min_width)
{
width = min_width;
Invalidate();
}
if (height < min_height)
{
height = min_height;
Invalidate();
}
ResizeFrameWithPage();
}
void OnUpdate() override
{
_tabAnimationIndex++;
if (_tabAnimationIndex >= 24)
{
_tabAnimationIndex = 0;
}
else
{
InvalidateWidget(WIDX_STAFF_LIST_HANDYMEN_TAB + _selectedTab);
// Enable highlighting of these staff members in map window
if (WindowFindByClass(WindowClass::Map) != nullptr)
{
gWindowMapFlashingFlags |= MapFlashingFlags::StaffListOpen;
for (auto peep : EntityList<Staff>())
{
EntitySetFlashing(peep, false);
if (peep->AssignedStaffType == GetSelectedStaffType())
{
EntitySetFlashing(peep, true);
}
}
}
}
// Note this may be slow if number of staff increases a large amount.
// See GuestList for fix (more intents) if required.
RefreshList();
}
void OnMouseDown(WidgetIndex widgetIndex) override
{
switch (widgetIndex)
2018-06-22 23:21:44 +02:00
{
case WIDX_STAFF_LIST_HANDYMEN_TAB:
case WIDX_STAFF_LIST_MECHANICS_TAB:
case WIDX_STAFF_LIST_SECURITY_TAB:
case WIDX_STAFF_LIST_ENTERTAINERS_TAB:
{
auto newSelectedTab = widgetIndex - WIDX_STAFF_LIST_HANDYMEN_TAB;
if (_selectedTab != newSelectedTab)
{
_selectedTab = static_cast<uint8_t>(newSelectedTab);
Invalidate();
scrolls[0].v_top = 0;
CancelTools();
}
break;
}
case WIDX_STAFF_LIST_UNIFORM_COLOUR_PICKER:
WindowDropdownShowColour(this, &widgets[widgetIndex], colours[1], StaffGetColour(GetSelectedStaffType()));
break;
2018-06-22 23:21:44 +02:00
}
}
void OnDropdown(WidgetIndex widgetIndex, int32_t dropdownIndex) override
{
if (dropdownIndex == -1)
{
return;
}
if (widgetIndex == WIDX_STAFF_LIST_UNIFORM_COLOUR_PICKER)
{
auto action = StaffSetColourAction(GetSelectedStaffType(), ColourDropDownIndexToColour(dropdownIndex));
GameActions::Execute(&action);
}
}
void OnPrepareDraw() override
{
// Set selected tab
SetWidgetPressed(WIDX_STAFF_LIST_HANDYMEN_TAB, false);
SetWidgetPressed(WIDX_STAFF_LIST_MECHANICS_TAB, false);
SetWidgetPressed(WIDX_STAFF_LIST_SECURITY_TAB, false);
SetWidgetPressed(WIDX_STAFF_LIST_ENTERTAINERS_TAB, false);
SetWidgetPressed(_selectedTab + WIDX_STAFF_LIST_HANDYMEN_TAB, true);
widgets[WIDX_STAFF_LIST_HIRE_BUTTON].text = GetStaffNamingConvention(GetSelectedStaffType()).ActionHire;
widgets[WIDX_STAFF_LIST_UNIFORM_COLOUR_PICKER].type = WindowWidgetType::Empty;
if (GetSelectedStaffType() != StaffType::Entertainer)
{
widgets[WIDX_STAFF_LIST_UNIFORM_COLOUR_PICKER].type = WindowWidgetType::ColourBtn;
widgets[WIDX_STAFF_LIST_UNIFORM_COLOUR_PICKER].image = GetColourButtonImage(
StaffGetColour(GetSelectedStaffType()));
}
SetWidgetPressed(WIDX_STAFF_LIST_QUICK_FIRE, _quickFireMode);
ResizeFrameWithPage();
widgets[WIDX_STAFF_LIST_LIST].right = width - 4;
widgets[WIDX_STAFF_LIST_LIST].bottom = height - 15;
widgets[WIDX_STAFF_LIST_QUICK_FIRE].left = width - 77;
widgets[WIDX_STAFF_LIST_QUICK_FIRE].right = width - 54;
widgets[WIDX_STAFF_LIST_SHOW_PATROL_AREA_BUTTON].left = width - 53;
widgets[WIDX_STAFF_LIST_SHOW_PATROL_AREA_BUTTON].right = width - 30;
widgets[WIDX_STAFF_LIST_MAP].left = width - 29;
widgets[WIDX_STAFF_LIST_MAP].right = width - 6;
widgets[WIDX_STAFF_LIST_HIRE_BUTTON].left = width - 155;
widgets[WIDX_STAFF_LIST_HIRE_BUTTON].right = width - 11;
}
void OnDraw(DrawPixelInfo& dpi) override
{
DrawWidgets(dpi);
DrawTabImages(dpi);
if (!(GetGameState().Park.Flags & PARK_FLAGS_NO_MONEY))
{
auto ft = Formatter();
ft.Add<money64>(GetStaffWage(GetSelectedStaffType()));
DrawTextBasic(dpi, windowPos + ScreenCoordsXY{ width - 155, 32 }, STR_COST_PER_MONTH, ft);
}
if (GetSelectedStaffType() != StaffType::Entertainer)
{
DrawTextBasic(
dpi, windowPos + ScreenCoordsXY{ 6, widgets[WIDX_STAFF_LIST_UNIFORM_COLOUR_PICKER].top + 1 },
STR_UNIFORM_COLOUR);
}
auto namingConvention = GetStaffNamingConvention(GetSelectedStaffType());
auto staffTypeStringId = _staffList.size() == 1 ? namingConvention.Singular : namingConvention.Plural;
auto ft = Formatter();
ft.Add<uint32_t>(_staffList.size());
ft.Add<StringId>(staffTypeStringId);
DrawTextBasic(
dpi, windowPos + ScreenCoordsXY{ 4, widgets[WIDX_STAFF_LIST_LIST].bottom + 2 }, STR_STAFF_LIST_COUNTER, ft);
}
ScreenSize OnScrollGetSize(int32_t scrollIndex) override
{
if (_highlightedIndex)
{
_highlightedIndex = {};
Invalidate();
}
auto scrollHeight = static_cast<int32_t>(_staffList.size()) * SCROLLABLE_ROW_HEIGHT;
auto i = scrollHeight - widgets[WIDX_STAFF_LIST_LIST].bottom + widgets[WIDX_STAFF_LIST_LIST].top + 21;
if (i < 0)
i = 0;
if (i < scrolls[0].v_top)
{
scrolls[0].v_top = i;
Invalidate();
}
auto scrollWidth = widgets[WIDX_STAFF_LIST_LIST].width() - 15;
return { scrollWidth, scrollHeight };
}
void OnScrollMouseOver(int32_t scrollIndex, const ScreenCoordsXY& screenCoords) override
{
auto i = static_cast<size_t>(screenCoords.y / SCROLLABLE_ROW_HEIGHT);
if (i != _highlightedIndex)
{
_highlightedIndex = static_cast<size_t>(i);
Invalidate();
}
}
void OnScrollMouseDown(int32_t scrollIndex, const ScreenCoordsXY& screenCoords) override
2018-06-22 23:21:44 +02:00
{
int32_t i = screenCoords.y / SCROLLABLE_ROW_HEIGHT;
for (const auto& entry : _staffList)
2018-06-22 23:21:44 +02:00
{
if (i == 0)
{
if (_quickFireMode)
{
auto staffFireAction = StaffFireAction(entry.Id);
GameActions::Execute(&staffFireAction);
}
else
{
auto peep = GetEntity<Staff>(entry.Id);
if (peep != nullptr)
{
auto intent = Intent(WindowClass::Peep);
intent.PutExtra(INTENT_EXTRA_PEEP, peep);
ContextOpenIntent(&intent);
}
}
break;
}
i--;
}
}
void OnScrollDraw(int32_t scrollIndex, DrawPixelInfo& dpi) override
{
auto dpiCoords = ScreenCoordsXY{ dpi.x, dpi.y };
GfxFillRect(
dpi, { dpiCoords, dpiCoords + ScreenCoordsXY{ dpi.width - 1, dpi.height - 1 } },
ColourMapA[colours[1]].mid_light);
// How much space do we have for the name and action columns? (Discount scroll area and icons.)
const int32_t nonIconSpace = widgets[WIDX_STAFF_LIST_LIST].width() - 15 - 68;
const int32_t nameColumnSize = nonIconSpace * 0.42;
const int32_t actionColumnSize = nonIconSpace * 0.58;
const int32_t actionOffset = widgets[WIDX_STAFF_LIST_LIST].right - actionColumnSize - 15;
auto y = 0;
size_t i = 0;
for (const auto& entry : _staffList)
{
if (y > dpi.y + dpi.height)
{
break;
}
if (y + 11 >= dpi.y)
{
const auto* peep = GetEntity<Staff>(entry.Id);
if (peep == nullptr)
{
continue;
}
int32_t format = (_quickFireMode ? STR_RED_STRINGID : STR_BLACK_STRING);
if (i == _highlightedIndex)
{
GfxFilterRect(dpi, { 0, y, 800, y + (SCROLLABLE_ROW_HEIGHT - 1) }, FilterPaletteID::PaletteDarken1);
format = (_quickFireMode ? STR_LIGHTPINK_STRINGID : STR_WINDOW_COLOUR_2_STRINGID);
}
auto ft = Formatter();
peep->FormatNameTo(ft);
DrawTextEllipsised(dpi, { 0, y }, nameColumnSize, format, ft);
ft = Formatter();
peep->FormatActionTo(ft);
DrawTextEllipsised(dpi, { actionOffset, y }, actionColumnSize, format, ft);
// True if a patrol path is set for the worker
if (peep->HasPatrolArea())
{
GfxDrawSprite(dpi, ImageId(SPR_STAFF_PATROL_PATH), { nameColumnSize + 5, y });
}
auto staffOrderIcon_x = nameColumnSize + 20;
if (peep->AssignedStaffType != StaffType::Entertainer)
{
auto staffOrders = peep->StaffOrders;
auto staffOrderSprite = GetStaffOrderBaseSprite(GetSelectedStaffType());
while (staffOrders != 0)
{
if (staffOrders & 1)
{
GfxDrawSprite(dpi, ImageId(staffOrderSprite), { staffOrderIcon_x, y });
}
staffOrders = staffOrders >> 1;
staffOrderIcon_x += 9;
// TODO: Remove sprite ID addition
staffOrderSprite++;
}
}
else
{
GfxDrawSprite(dpi, ImageId(GetEntertainerCostumeSprite(peep->SpriteType)), { staffOrderIcon_x, y });
}
}
y += SCROLLABLE_ROW_HEIGHT;
i++;
}
}
void OnToolDown(WidgetIndex widgetIndex, const ScreenCoordsXY& screenCoords) override
{
if (widgetIndex == WIDX_STAFF_LIST_SHOW_PATROL_AREA_BUTTON)
{
auto closestStaffMember = GetClosestStaffMemberTo(screenCoords);
if (closestStaffMember != nullptr)
{
ToolCancel();
auto* staffWindow = StaffOpen(closestStaffMember);
staffWindow->OnDropdown(WC_PEEP__WIDX_PATROL, 0);
}
else
{
auto ft = Formatter();
ft.Add<StringId>(GetStaffNamingConvention(GetSelectedStaffType()).Plural);
ContextShowError(STR_NO_THING_IN_PARK_YET, STR_NONE, ft);
}
}
}
void OnToolAbort(WidgetIndex widgetIndex) override
{
if (widgetIndex == WIDX_STAFF_LIST_SHOW_PATROL_AREA_BUTTON)
2018-06-22 23:21:44 +02:00
{
HideGridlines();
ToolCancel();
ClearPatrolAreaToRender();
GfxInvalidateScreen();
}
}
void RefreshList()
2018-06-22 23:21:44 +02:00
{
_staffList.clear();
for (auto* peep : EntityList<Staff>())
{
EntitySetFlashing(peep, false);
if (peep->AssignedStaffType == GetSelectedStaffType())
{
EntitySetFlashing(peep, true);
2023-02-28 05:11:45 +01:00
StaffEntry entry;
entry.Id = peep->Id;
entry.Name = peep->GetName();
2023-02-28 05:11:45 +01:00
_staffList.push_back(std::move(entry));
}
}
std::sort(_staffList.begin(), _staffList.end(), [](const auto& a, const auto& b) {
return StrLogicalCmp(a.Name.c_str(), b.Name.c_str()) < 0;
});
2021-12-03 16:37:39 +01:00
}
private:
/**
* Hires a new staff member of the given type.
*/
void HireNewMember(StaffType staffType, EntertainerCostume entertainerType)
2021-12-03 16:37:39 +01:00
{
bool autoPosition = gConfigGeneral.AutoStaffPlacement;
if (gInputPlaceObjectModifier & PLACE_OBJECT_MODIFIER_SHIFT_Z)
2021-12-03 16:37:39 +01:00
{
autoPosition = autoPosition ^ 1;
2021-12-03 16:37:39 +01:00
}
uint32_t staffOrders = 0;
if (staffType == StaffType::Handyman)
{
staffOrders = STAFF_ORDERS_SWEEPING | STAFF_ORDERS_WATER_FLOWERS | STAFF_ORDERS_EMPTY_BINS;
if (gConfigGeneral.HandymenMowByDefault)
{
staffOrders |= STAFF_ORDERS_MOWING;
}
}
else if (staffType == StaffType::Mechanic)
{
staffOrders = STAFF_ORDERS_INSPECT_RIDES | STAFF_ORDERS_FIX_RIDES;
}
2021-12-03 16:37:39 +01:00
auto hireStaffAction = StaffHireNewAction(autoPosition, staffType, entertainerType, staffOrders);
hireStaffAction.SetCallback([=](const GameAction*, const GameActions::Result* res) -> void {
if (res->Error != GameActions::Status::Ok)
return;
2021-12-03 16:37:39 +01:00
auto actionResult = res->GetData<StaffHireNewActionResult>();
auto* staff = GetEntity<Staff>(actionResult.StaffEntityId);
if (staff == nullptr)
return;
// If autoposition of staff is disabled, pickup peep and then open the staff window
if (staff->State == PeepState::Picked)
{
CoordsXYZ nullLoc{};
nullLoc.SetNull();
PeepPickupAction pickupAction{ PeepPickupType::Pickup, staff->Id, nullLoc, NetworkGetCurrentPlayerId() };
pickupAction.SetCallback([staffId = staff->Id](const GameAction* ga, const GameActions::Result* result) {
if (result->Error != GameActions::Status::Ok)
return;
auto* staff2 = GetEntity<Staff>(staffId);
auto intent = Intent(WindowClass::Peep);
intent.PutExtra(INTENT_EXTRA_PEEP, staff2);
auto* wind = ContextOpenIntent(&intent);
if (wind != nullptr)
{
ToolSet(*wind, WC_STAFF__WIDX_PICKUP, Tool::Picker);
}
});
GameActions::Execute(&pickupAction);
}
else
{
// Open window for new staff.
auto intent = Intent(WindowClass::Peep);
intent.PutExtra(INTENT_EXTRA_PEEP, staff);
ContextOpenIntent(&intent);
}
});
GameActions::Execute(&hireStaffAction);
}
StaffType GetSelectedStaffType() const
{
return static_cast<StaffType>(_selectedTab);
}
void DrawTabImages(DrawPixelInfo& dpi) const
{
const auto& gameState = GetGameState();
DrawTabImage(dpi, WINDOW_STAFF_LIST_TAB_HANDYMEN, PeepSpriteType::Handyman, gameState.StaffHandymanColour);
DrawTabImage(dpi, WINDOW_STAFF_LIST_TAB_MECHANICS, PeepSpriteType::Mechanic, gameState.StaffMechanicColour);
DrawTabImage(dpi, WINDOW_STAFF_LIST_TAB_SECURITY, PeepSpriteType::Security, gameState.StaffSecurityColour);
DrawTabImage(dpi, WINDOW_STAFF_LIST_TAB_ENTERTAINERS, PeepSpriteType::EntertainerElephant);
}
void DrawTabImage(DrawPixelInfo& dpi, int32_t tabIndex, PeepSpriteType type, colour_t colour) const
{
auto widgetIndex = WIDX_STAFF_LIST_HANDYMEN_TAB + tabIndex;
const auto& widget = widgets[widgetIndex];
auto imageId = (_selectedTab == tabIndex ? (_tabAnimationIndex & ~3) : 0);
imageId += GetPeepAnimation(type).base_image + 1;
GfxDrawSprite(
dpi, ImageId(imageId, colour),
windowPos + ScreenCoordsXY{ (widget.left + widget.right) / 2, widget.bottom - 6 });
}
void DrawTabImage(DrawPixelInfo& dpi, int32_t tabIndex, PeepSpriteType type) const
{
auto widgetIndex = WIDX_STAFF_LIST_HANDYMEN_TAB + tabIndex;
const auto& widget = widgets[widgetIndex];
DrawPixelInfo clippedDpi;
if (ClipDrawPixelInfo(
clippedDpi, dpi, windowPos + ScreenCoordsXY{ widget.left + 1, widget.top + 1 },
widget.right - widget.left - 1, widget.bottom - widget.top - 1))
{
auto imageId = (_selectedTab == 3 ? (_tabAnimationIndex & ~3) : 0);
imageId += GetPeepAnimation(type).base_image + 1;
GfxDrawSprite(clippedDpi, ImageId(imageId), { 15, 23 });
}
}
void CancelTools()
{
if (InputTestFlag(INPUT_FLAG_TOOL_ACTIVE))
{
if (classification == gCurrentToolWidget.window_classification && number == gCurrentToolWidget.window_number)
{
ToolCancel();
}
}
}
Peep* GetClosestStaffMemberTo(const ScreenCoordsXY& screenCoords)
{
int32_t direction{};
TileElement* tileElement{};
auto footpathCoords = FootpathGetCoordinatesFromPos(screenCoords, &direction, &tileElement);
if (footpathCoords.IsNull())
return nullptr;
auto isPatrolAreaSet = IsPatrolAreaSetForStaffType(GetSelectedStaffType(), footpathCoords);
Peep* closestPeep = nullptr;
auto closestPeepDistance = std::numeric_limits<int32_t>::max();
for (auto peep : EntityList<Staff>())
{
if (peep->AssignedStaffType != GetSelectedStaffType())
continue;
if (isPatrolAreaSet)
{
if (!peep->HasPatrolArea())
{
continue;
}
if (!peep->IsLocationInPatrol(footpathCoords))
{
continue;
}
}
if (peep->x == LOCATION_NULL)
{
continue;
}
auto distance = std::abs(footpathCoords.x - peep->x) + std::abs(footpathCoords.y - peep->y);
if (distance < closestPeepDistance)
{
closestPeepDistance = distance;
closestPeep = peep;
}
}
return closestPeep;
}
static EntertainerCostume GetRandomEntertainerCostume()
{
auto result = EntertainerCostume::Panda;
EntertainerCostume costumeList[EnumValue(EntertainerCostume::Count)];
int32_t numCostumes = StaffGetAvailableEntertainerCostumeList(costumeList);
if (numCostumes > 0)
{
int32_t index = UtilRand() % numCostumes;
result = costumeList[index];
}
return result;
}
static constexpr StaffNamingConvention GetStaffNamingConvention(StaffType type)
{
switch (type)
{
default:
case StaffType::Handyman:
return { STR_HANDYMAN_PLURAL, STR_HANDYMAN_SINGULAR, STR_HIRE_HANDYMAN };
case StaffType::Mechanic:
return { STR_MECHANIC_PLURAL, STR_MECHANIC_SINGULAR, STR_HIRE_MECHANIC };
case StaffType::Security:
return { STR_SECURITY_GUARD_PLURAL, STR_SECURITY_GUARD_SINGULAR, STR_HIRE_SECURITY_GUARD };
case StaffType::Entertainer:
return { STR_ENTERTAINER_PLURAL, STR_ENTERTAINER_SINGULAR, STR_HIRE_ENTERTAINER };
}
}
static uint32_t GetStaffOrderBaseSprite(StaffType type)
{
switch (type)
{
case StaffType::Handyman:
return SPR_STAFF_ORDERS_SWEEPING;
case StaffType::Mechanic:
return SPR_STAFF_ORDERS_INSPECT_RIDES;
default:
return 0;
}
}
static uint32_t GetEntertainerCostumeSprite(PeepSpriteType type)
{
switch (type)
{
default:
case PeepSpriteType::EntertainerPanda:
return SPR_STAFF_COSTUME_PANDA;
case PeepSpriteType::EntertainerTiger:
return SPR_STAFF_COSTUME_TIGER;
case PeepSpriteType::EntertainerElephant:
return SPR_STAFF_COSTUME_ELEPHANT;
case PeepSpriteType::EntertainerRoman:
return SPR_STAFF_COSTUME_ROMAN;
case PeepSpriteType::EntertainerGorilla:
return SPR_STAFF_COSTUME_GORILLA;
case PeepSpriteType::EntertainerSnowman:
return SPR_STAFF_COSTUME_SNOWMAN;
case PeepSpriteType::EntertainerKnight:
return SPR_STAFF_COSTUME_KNIGHT;
case PeepSpriteType::EntertainerAstronaut:
return SPR_STAFF_COSTUME_ASTRONAUT;
case PeepSpriteType::EntertainerBandit:
return SPR_STAFF_COSTUME_BANDIT;
case PeepSpriteType::EntertainerSheriff:
return SPR_STAFF_COSTUME_SHERIFF;
case PeepSpriteType::EntertainerPirate:
return SPR_STAFF_COSTUME_PIRATE;
}
}
};
WindowBase* StaffListOpen()
{
return WindowFocusOrCreate<StaffListWindow>(WindowClass::StaffList, WW, WH, WF_10 | WF_RESIZABLE);
}
void WindowStaffListRefresh()
{
auto* window = WindowFindByClass(WindowClass::StaffList);
if (window != nullptr)
{
static_cast<StaffListWindow*>(window)->RefreshList();
}
}
} // namespace OpenRCT2::Ui::Windows