/* ScummVM - Graphic Adventure Engine * * ScummVM is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * This program 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 2 * of the License, or (at your option) any later version. * * This program 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. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include "backends/networking/curl/cloudicon.h" #include "backends/cloud/cloudmanager.h" #include "common/memstream.h" #include "gui/ThemeEngine.h" #include "gui/gui-manager.h" #include "image/png.h" namespace Networking { const float CloudIcon::ALPHA_STEP = 0.025; const float CloudIcon::ALPHA_MAX = 1; const float CloudIcon::ALPHA_MIN = 0.6; CloudIcon::CloudIcon(): _wasVisible(false), _iconsInited(false), _showingDisabled(false), _currentAlpha(0), _alphaRising(true), _disabledFrames(0) { initIcons(); } CloudIcon::~CloudIcon() {} bool CloudIcon::draw() { bool stop = false; initIcons(); if (CloudMan.isWorking() || _disabledFrames > 0) { if (g_system) { if (!_wasVisible) { g_system->clearOSD(); _wasVisible = true; } --_disabledFrames; if (_alphaRising) { if (_currentAlpha < ALPHA_MIN) _currentAlpha += 5 * ALPHA_STEP; else _currentAlpha += ALPHA_STEP; if (_currentAlpha > ALPHA_MAX) { _currentAlpha = ALPHA_MAX; _alphaRising = false; } } else { _currentAlpha -= ALPHA_STEP; if (_currentAlpha < ALPHA_MIN) { _currentAlpha = ALPHA_MIN; _alphaRising = true; } } } else { _wasVisible = false; } } else { _wasVisible = false; _currentAlpha -= 5 * ALPHA_STEP; if (_currentAlpha <= 0) { _currentAlpha = 0; stop = true; } } if (g_system) { Graphics::TransparentSurface *surface = &_icon; makeAlphaIcon((_showingDisabled ? _disabledIcon : _icon), _currentAlpha); if (_alphaIcon.getPixels()) surface = &_alphaIcon; if (surface && surface->getPixels()) { int x = g_system->getOverlayWidth() - surface->w - 10, y = 10; g_system->copyRectToOSD(surface->getPixels(), surface->pitch, x, y, surface->w, surface->h); } } if (stop) _showingDisabled = false; return stop; } void CloudIcon::showDisabled() { _showingDisabled = true; _disabledFrames = 20 * 3; //3 seconds 20 fps } #include "backends/networking/curl/cloudicon_data.h" #include "backends/networking/curl/cloudicon_disabled_data.h" void CloudIcon::initIcons() { if (_iconsInited) return; loadIcon(_icon, cloudicon_data, ARRAYSIZE(cloudicon_data)); loadIcon(_disabledIcon, cloudicon_disabled_data, ARRAYSIZE(cloudicon_disabled_data)); _iconsInited = true; } void CloudIcon::loadIcon(Graphics::TransparentSurface &icon, byte *data, uint32 size) { Image::PNGDecoder decoder; Common::MemoryReadStream stream(data, size); if (!decoder.loadStream(stream)) error("CloudIcon::loadIcon: error decoding PNG"); Graphics::TransparentSurface *s = new Graphics::TransparentSurface(*decoder.getSurface(), true); if (s) { Graphics::PixelFormat f = g_system->getOSDFormat(); if (f != s->format) { // Graphics::TransparentSurface::convertTo(f) errors out if the format is not 2Bpp or 4Bpp. // We don't need to error out as we can recover from it. So check the format before calling convertTo(f); Graphics::TransparentSurface *s2 = nullptr; if (f.bytesPerPixel == 2 || f.bytesPerPixel == 4) s2 = s->convertTo(f); if (s2) icon.copyFrom(*s2); else warning("CloudIcon::loadIcon: failed converting TransparentSurface"); delete s2; } else { icon.copyFrom(*s); } delete s; } else { warning("CloudIcon::loadIcon: failed reading TransparentSurface from PNGDecoder"); } } void CloudIcon::makeAlphaIcon(Graphics::TransparentSurface &icon, float alpha) { _alphaIcon.copyFrom(icon); byte *pixels = (byte *)_alphaIcon.getPixels(); for (int y = 0; y < _alphaIcon.h; y++) { byte *row = pixels + y * _alphaIcon.pitch; for (int x = 0; x < _alphaIcon.w; x++) { uint32 srcColor; if (_alphaIcon.format.bytesPerPixel == 2) srcColor = READ_UINT16(row); else if (_alphaIcon.format.bytesPerPixel == 3) srcColor = READ_UINT24(row); else srcColor = READ_UINT32(row); // Update color's alpha byte r, g, b, a; _alphaIcon.format.colorToARGB(srcColor, a, r, g, b); a = (byte)(a * alpha); uint32 color = _alphaIcon.format.ARGBToColor(a, r, g, b); if (_alphaIcon.format.bytesPerPixel == 2) *((uint16 *)row) = color; else *((uint32 *)row) = color; row += _alphaIcon.format.bytesPerPixel; } } } } // End of namespace Networking