/* * The ManaPlus Client * Copyright (C) 2004-2009 The Mana World Development Team * Copyright (C) 2009-2010 The Mana Developers * Copyright (C) 2011-2012 The ManaPlus Developers * * This file is part of The ManaPlus Client. * * 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 * 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, see <http://www.gnu.org/licenses/>. */ #include "gui/widgets/button.h" #include "client.h" #include "configuration.h" #include "graphics.h" #include "graphicsvertexes.h" #include "keydata.h" #include "keyevent.h" #include "logger.h" #include "gui/palette.h" #include "gui/theme.h" #include "resources/image.h" #include "resources/imageset.h" #include "utils/dtor.h" #include <guichan/exception.hpp> #include <guichan/font.hpp> #include "debug.h" int Button::mInstances = 0; float Button::mAlpha = 1.0; enum { BUTTON_STANDARD = 0, // 0 BUTTON_HIGHLIGHTED, // 1 BUTTON_PRESSED, // 2 BUTTON_DISABLED, // 3 BUTTON_COUNT // 4 - Must be last. }; static std::string const data[BUTTON_COUNT] = { "button.xml", "button_highlighted.xml", "button_pressed.xml", "button_disabled.xml" }; ImageRect Button::button[BUTTON_COUNT]; Button::Button() : mDescription(""), mClickCount(0), mTag(0), mVertexes(new GraphicsVertexes()), mRedraw(true), mMode(0), mXOffset(0), mYOffset(0), mImages(nullptr), mImageSet(nullptr), mStick(false), mPressed(false) { init(); adjustSize(); } Button::Button(const std::string &caption, const std::string &actionEventId, gcn::ActionListener *listener) : gcn::Button(caption), mDescription(""), mClickCount(0), mTag(0), mVertexes(new GraphicsVertexes()), mRedraw(true), mMode(0), mXOffset(0), mYOffset(0), mImages(nullptr), mImageSet(nullptr), mImageWidth(0), mImageHeight(0), mStick(false), mPressed(false) { init(); adjustSize(); setActionEventId(actionEventId); if (listener) addActionListener(listener); } Button::Button(const std::string &caption, const std::string &imageName, int imageWidth, int imageHeight, const std::string &actionEventId, gcn::ActionListener *listener) : gcn::Button(caption), mDescription(""), mClickCount(0), mTag(0), mVertexes(new GraphicsVertexes()), mRedraw(true), mMode(0), mXOffset(0), mYOffset(0), mImages(nullptr), mImageSet(nullptr), mImageWidth(imageWidth), mImageHeight(imageHeight), mStick(false), mPressed(false) { init(); loadImage(imageName); adjustSize(); setActionEventId(actionEventId); if (listener) addActionListener(listener); } Button::Button(const std::string &imageName, int imageWidth, int imageHeight, const std::string &actionEventId, gcn::ActionListener *listener) : gcn::Button(""), mDescription(""), mClickCount(0), mTag(0), mVertexes(new GraphicsVertexes()), mRedraw(true), mMode(0), mXOffset(0), mYOffset(0), mImages(nullptr), mImageSet(nullptr), mImageWidth(imageWidth), mImageHeight(imageHeight), mStick(false), mPressed(false) { init(); loadImage(imageName); adjustSize(); setActionEventId(actionEventId); if (listener) addActionListener(listener); } void Button::init() { setFrameSize(0); addWidgetListener(this); if (mInstances == 0) { if (Theme::instance()) { for (int mode = 0; mode < BUTTON_COUNT; mode ++) Theme::instance()->loadRect(button[mode], data[mode]); } updateAlpha(); } mEnabledColor = Theme::getThemeColor(Theme::BUTTON); mDisabledColor = Theme::getThemeColor(Theme::BUTTON_DISABLED); mInstances++; } Button::~Button() { mInstances--; delete mVertexes; mVertexes = nullptr; if (mImageSet) { mImageSet->decRef(); mImageSet = nullptr; } if (mImages) { for (int f = 0; f < BUTTON_COUNT; f ++) mImages[f] = nullptr; delete [] mImages; mImages = nullptr; } } void Button::loadImage(const std::string &imageName) { if (mImageSet) { mImageSet->decRef(); mImageSet = nullptr; } mImageSet = Theme::getImageSetFromTheme(imageName, mImageWidth, mImageHeight); if (!mImageSet) return; mImages = new Image*[BUTTON_COUNT]; mImages[0] = nullptr; for (int f = 0; f < BUTTON_COUNT; f ++) { Image *img = mImageSet->get(f); if (img) mImages[f] = img; else mImages[f] = mImages[0]; } } void Button::updateAlpha() { float alpha = std::max(Client::getGuiAlpha(), Theme::instance()->getMinimumOpacity()); if (mAlpha != alpha) { mAlpha = alpha; for (int a = 0; a < 9; a++) { if (button[BUTTON_DISABLED].grid[a]) button[BUTTON_DISABLED].grid[a]->setAlpha(mAlpha); if (button[BUTTON_PRESSED].grid[a]) button[BUTTON_PRESSED].grid[a]->setAlpha(mAlpha); if (button[BUTTON_HIGHLIGHTED].grid[a]) button[BUTTON_HIGHLIGHTED].grid[a]->setAlpha(mAlpha); if (button[BUTTON_STANDARD].grid[a]) button[BUTTON_STANDARD].grid[a]->setAlpha(mAlpha); } } } void Button::draw(gcn::Graphics *graphics) { int mode; if (!isEnabled()) mode = BUTTON_DISABLED; else if (isPressed2()) mode = BUTTON_PRESSED; else if (mHasMouse || isFocused()) mode = BUTTON_HIGHLIGHTED; else mode = BUTTON_STANDARD; updateAlpha(); Graphics *g2 = static_cast<Graphics*>(graphics); bool recalc = false; if (mRedraw) { recalc = true; } else { // because we don't know where parent windows was moved, // need recalc vertexes gcn::ClipRectangle &rect = g2->getTopClip(); if (rect.xOffset != mXOffset || rect.yOffset != mYOffset) { recalc = true; mXOffset = rect.xOffset; mYOffset = rect.yOffset; } else if (mMode != mode) { recalc = true; mMode = mode; } else if (g2->getRedraw()) { recalc = true; } } if (recalc) { mRedraw = false; mMode = mode; g2->calcWindow(mVertexes, 0, 0, getWidth(), getHeight(), button[mode]); } g2->drawImageRect2(mVertexes, button[mode]); // g2->drawImageRect(0, 0, getWidth(), getHeight(), button[mode]); if (mode == BUTTON_DISABLED) graphics->setColor(mDisabledColor); else graphics->setColor(mEnabledColor); int textX = 0; int textY = getHeight() / 2 - getFont()->getHeight() / 2; int imageX = 0; int imageY = 0; if (mImages) imageY = getHeight() / 2 - mImageHeight / 2; // need move calculation from draw!!! switch (getAlignment()) { default: case gcn::Graphics::LEFT: if (mImages) { imageX = 4; textX = 4 + mImageWidth + 2; } else { textX = 4; } break; case gcn::Graphics::CENTER: if (mImages) { int width = getFont()->getWidth(mCaption) + mImageWidth + 2; imageX = getWidth() / 2 - width / 2; textX = imageX + mImageWidth + 2; } else { textX = getWidth() / 2; } break; case gcn::Graphics::RIGHT: textX = getWidth() - 4; imageX = textX - getFont()->getWidth(mCaption) - 2; break; } graphics->setFont(getFont()); if (isPressed()) { if (mImages) g2->drawImage(mImages[mode], imageX + 1, imageY + 1); g2->drawText(getCaption(), textX + 1, textY + 1, getAlignment()); } else { if (mImages) g2->drawImage(mImages[mode], imageX, imageY); g2->drawText(getCaption(), textX, textY, getAlignment()); } } void Button::mouseReleased(gcn::MouseEvent& mouseEvent) { if (mouseEvent.getButton() == gcn::MouseEvent::LEFT) { if (mStick) mPressed = !mPressed; if (mMousePressed && mHasMouse) { mMousePressed = false; mClickCount = mouseEvent.getClickCount(); distributeActionEvent(); } else { mMousePressed = false; mClickCount = 0; } mouseEvent.consume(); } } void Button::widgetResized(const gcn::Event &event A_UNUSED) { mRedraw = true; } void Button::widgetMoved(const gcn::Event &event A_UNUSED) { mRedraw = true; } void Button::adjustSize() { if (mImages) { setWidth(getFont()->getWidth(mCaption) + mImageWidth + 2 + 2 * mSpacing); int height = getFont()->getHeight(); if (height < mImageHeight) height = mImageHeight; setHeight(height + 2 * mSpacing); } else { setWidth(getFont()->getWidth(mCaption) + 2 * mSpacing); setHeight(getFont()->getHeight() + 2 * mSpacing); } } void Button::setCaption(const std::string& caption) { mCaption = caption; } void Button::keyPressed(gcn::KeyEvent& keyEvent) { int action = static_cast<KeyEvent*>(&keyEvent)->getActionId(); if (action == Input::KEY_GUI_SELECT) { mKeyPressed = true; keyEvent.consume(); } } void Button::keyReleased(gcn::KeyEvent& keyEvent) { int action = static_cast<KeyEvent*>(&keyEvent)->getActionId(); if (action == Input::KEY_GUI_SELECT && mKeyPressed) { mKeyPressed = false; if (mStick) mPressed = !mPressed; distributeActionEvent(); keyEvent.consume(); } } bool Button::isPressed2() { return (mPressed || isPressed()); }