#pragma region Copyright (c) 2014-2017 OpenRCT2 Developers /***************************************************************************** * OpenRCT2, an open source clone of Roller Coaster Tycoon 2. * * OpenRCT2 is the work of many authors, a full list can be found in contributors.md * For more information, visit https://github.com/OpenRCT2/OpenRCT2 * * OpenRCT2 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, either version 3 of the License, or * (at your option) any later version. * * A full copy of the GNU General Public License can be found in licence.txt *****************************************************************************/ #pragma endregion #include #include #include #include #include #include "CursorRepository.h" using namespace OpenRCT2::Ui; CursorRepository::~CursorRepository() { _scaledCursors.clear(); _currentCursor = CURSOR_UNDEFINED; _currentCursorScale = 1; } void CursorRepository::LoadCursors() { SetCursorScale(static_cast(round(gConfigGeneral.window_scale))); SetCurrentCursor(CURSOR_ARROW); } CURSOR_ID CursorRepository::GetCurrentCursor() { return _currentCursor; } void CursorRepository::SetCurrentCursor(CURSOR_ID cursorId) { if (_currentCursor != cursorId) { SDL_Cursor * cursor = _scaledCursors.at(_currentCursorScale).getScaledCursor(cursorId); SDL_SetCursor(cursor); _currentCursor = cursorId; } } static bool getBit(const uint8 * data, size_t x, size_t y, size_t width) { size_t position = y * width + x; return (data[position / 8] & (1 << (7 - (x % 8)))) != 0; } static void setBit(uint8 * data, size_t x, size_t y, size_t width) { size_t position = y * width + x; data[position / 8] |= (1 << (7 - (position % 8))); } static void drawRect(uint8 * data, size_t x, size_t y, size_t width, size_t scale) { for (size_t outY = (y * scale); outY < ((1 + y) * scale); outY++) { for (size_t outX = (x * scale); outX < ((1 + x) * scale); outX++) { setBit(data, outX, outY, width * scale); } } } static uint8 * scaleDataArray(const uint8 data[], size_t width, size_t height, size_t scale) { size_t length = width * height; auto * ret = static_cast(calloc(sizeof(uint8), length * scale * scale)); for (size_t y = 0; y < height * 8; y++) { for (size_t x = 0; x < width; x++) { bool value = getBit(data, x, y, width); if (!value) continue; drawRect(ret, x, y, width, scale); } } return ret; } SDL_Cursor * CursorRepository::Create(const CursorData * cursorInfo, uint8 scale) { SDL_Cursor * cursor; auto integer_scale = (int) round(scale); auto data = scaleDataArray(cursorInfo->Data, CURSOR_BIT_WIDTH, CURSOR_HEIGHT, static_cast(integer_scale)); auto mask = scaleDataArray(cursorInfo->Mask, CURSOR_BIT_WIDTH, CURSOR_HEIGHT, static_cast(integer_scale)); cursor = SDL_CreateCursor( data, mask, BASE_CURSOR_WIDTH * integer_scale, BASE_CURSOR_HEIGHT * integer_scale, cursorInfo->HotSpot.X * integer_scale, cursorInfo->HotSpot.Y * integer_scale); free(data); free(mask); return cursor; } void CursorRepository::SetCursorScale(uint8 cursorScale) { if (cursorScale > 0.0) { _currentCursorScale = cursorScale; GenerateScaledCursorSetHolder(_currentCursorScale); } } void CursorRepository::GenerateScaledCursorSetHolder(uint8 scale) { if (_scaledCursors.find(scale) == _scaledCursors.end()) { std::function cursorGenerator = [this, scale](CURSOR_ID cursorId) { switch (cursorId) { // We can't scale the system cursors, but they should be appropriately scaled anyway case CURSOR_ARROW: return SDL_CreateSystemCursor(SDL_SYSTEM_CURSOR_ARROW); case CURSOR_HAND_POINT: return SDL_CreateSystemCursor(SDL_SYSTEM_CURSOR_HAND); default: return this->Create(GetCursorData(cursorId), scale); } }; _scaledCursors.emplace(scale, cursorGenerator); } }