/*
* The ManaPlus Client
* Copyright (C) 2006-2009 The Mana World Development Team
* Copyright (C) 2009-2010 The Mana Developers
* Copyright (C) 2011-2014 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/dropdown.h"
#include "client.h"
#include "input/keydata.h"
#include "input/keyevent.h"
#include "gui/widgets/extendedlistmodel.h"
#include "gui/widgets/popuplist.h"
#include "resources/image.h"
#include "gui/base/font.hpp"
#include <algorithm>
#include "debug.h"
int DropDown::instances = 0;
Image *DropDown::buttons[2][2];
ImageRect DropDown::skinRect;
float DropDown::mAlpha = 1.0;
Skin *DropDown::mSkin = nullptr;
static std::string const dropdownFiles[2] =
{
"dropdown.xml",
"dropdown_pressed.xml"
};
DropDown::DropDown(const Widget2 *const widget,
gcn::ListModel *const listModel,
const bool extended,
const bool modal,
ActionListener *const listener,
const std::string &eventId):
ActionListener(),
gcn::BasicContainer(),
gcn::KeyListener(),
gcn::MouseListener(),
gcn::FocusListener(),
gcn::SelectionListener(),
Widget2(widget),
mPopup(new PopupList(this, listModel, extended, modal)),
mShadowColor(getThemeColor(Theme::DROPDOWN_SHADOW)),
mHighlightColor(getThemeColor(Theme::HIGHLIGHT)),
mPadding(1),
mImagePadding(2),
mSpacing(0),
mFoldedUpHeight(0),
mSelectionListeners(),
mExtended(extended),
mDroppedDown(false),
mPushed(false),
mIsDragged(false)
{
mPopup->postInit();
mFrameSize = 2;
mForegroundColor2 = getThemeColor(Theme::DROPDOWN_OUTLINE);
mPopup->setHeight(100);
// Initialize graphics
if (instances == 0)
{
// Load the background skin
for (int i = 0; i < 2; i ++)
{
Skin *const skin = Theme::instance()->load(
dropdownFiles[i], "dropdown.xml");
if (skin)
{
if (!i)
mSkin = skin;
const ImageRect &rect = skin->getBorder();
for (int f = 0; f < 2; f ++)
{
if (rect.grid[f])
{
rect.grid[f]->incRef();
buttons[f][i] = rect.grid[f];
buttons[f][i]->setAlpha(mAlpha);
}
else
{
buttons[f][i] = nullptr;
}
}
if (i)
Theme::instance()->unload(skin);
}
else
{
for (int f = 0; f < 2; f ++)
buttons[f][i] = nullptr;
}
}
// get the border skin
if (Theme::instance())
{
Theme::instance()->loadRect(skinRect,
"dropdown_background.xml", "");
}
}
instances++;
setWidth(100);
setFocusable(true);
setListModel(listModel);
if (mPopup->getSelected() < 0)
mPopup->setSelected(0);
addMouseListener(this);
addKeyListener(this);
addFocusListener(this);
adjustHeight();
// mPopup->setForegroundColorAll(getThemeColor(Theme::DROPDOWN),
// getThemeColor(Theme::DROPDOWN_OUTLINE));
mForegroundColor = getThemeColor(Theme::DROPDOWN);
mForegroundColor2 = getThemeColor(Theme::DROPDOWN_OUTLINE);
if (!eventId.empty())
setActionEventId(eventId);
if (listener)
addActionListener(listener);
mPopup->adjustSize();
if (mSkin)
{
mSpacing = mSkin->getOption("spacing");
mFrameSize = mSkin->getOption("frameSize");
mPadding = mSkin->getPadding();
mImagePadding = mSkin->getOption("imagePadding");
}
adjustHeight();
}
DropDown::~DropDown()
{
if (gui)
gui->removeDragged(this);
instances--;
if (instances == 0)
{
for (int f = 0; f < 2; f ++)
{
for (int i = 0; i < 2; i ++)
{
if (buttons[f][i])
buttons[f][i]->decRef();
}
}
Theme *const theme = Theme::instance();
if (theme)
{
theme->unload(mSkin);
Theme::unloadRect(skinRect);
}
}
}
void DropDown::updateAlpha()
{
const float alpha = std::max(client->getGuiAlpha(),
Theme::instance()->getMinimumOpacity());
if (mAlpha != alpha)
{
mAlpha = alpha;
if (buttons[0][0])
buttons[0][0]->setAlpha(mAlpha);
if (buttons[0][1])
buttons[0][1]->setAlpha(mAlpha);
if (buttons[1][0])
buttons[1][0]->setAlpha(mAlpha);
if (buttons[1][1])
buttons[1][1]->setAlpha(mAlpha);
for (int a = 0; a < 9; a++)
{
if (skinRect.grid[a])
skinRect.grid[a]->setAlpha(mAlpha);
}
}
}
void DropDown::draw(Graphics* graphics)
{
BLOCK_START("DropDown::draw")
int h;
if (mDroppedDown)
h = mFoldedUpHeight;
else
h = mDimension.height;
updateAlpha();
const int alpha = static_cast<int>(mAlpha * 255.0F);
const int pad = 2 * mPadding;
mHighlightColor.a = alpha;
mShadowColor.a = alpha;
gcn::ListModel *const model = mPopup->getListModel();
if (model && mPopup->getSelected() >= 0)
{
gcn::Font *const font = getFont();
graphics->setColorAll(mForegroundColor, mForegroundColor2);
if (mExtended)
{
const int sel = mPopup->getSelected();
ExtendedListModel *const model2
= static_cast<ExtendedListModel *const>(model);
const Image *const image = model2->getImageAt(sel);
if (!image)
{
font->drawString(graphics, model->getElementAt(sel),
mPadding, mPadding);
}
else
{
graphics->drawImage2(image,
mImagePadding,
(mDimension.height - image->getHeight()) / 2 + mPadding);
font->drawString(graphics, model->getElementAt(sel),
image->getWidth() + mImagePadding + mSpacing, mPadding);
}
}
else
{
font->drawString(graphics, model->getElementAt(
mPopup->getSelected()), mPadding, mPadding);
}
}
if (isFocused())
{
graphics->setColor(mHighlightColor);
graphics->drawRectangle(gcn::Rectangle(mPadding, mPadding,
mDimension.width - h - pad, h - pad));
}
drawButton(graphics);
if (mDroppedDown)
{
// Draw two lines separating the ListBox with selected
// element view.
const int w = mDimension.width;
graphics->setColor(mHighlightColor);
graphics->drawLine(0, h, w, h);
graphics->setColor(mShadowColor);
graphics->drawLine(0, h + 1, w, h + 1);
}
BLOCK_END("DropDown::draw")
}
void DropDown::drawFrame(Graphics *graphics)
{
BLOCK_START("DropDown::drawFrame")
const int bs2 = getFrameSize();
const gcn::Rectangle &rect = mDimension;
graphics->drawImageRect(0, 0,
rect.width + bs2, rect.height + bs2,
skinRect);
BLOCK_END("DropDown::drawFrame")
}
void DropDown::drawButton(Graphics *graphics)
{
const int height = mDroppedDown ? mFoldedUpHeight : mDimension.height;
Image *image = buttons[mDroppedDown][mPushed];
if (image)
{
graphics->drawImage2(image,
mDimension.width - image->getWidth() - mImagePadding,
(height - image->getHeight()) / 2);
}
}
void DropDown::keyPressed(gcn::KeyEvent& keyEvent)
{
if (keyEvent.isConsumed())
return;
const int actionId = static_cast<KeyEvent*>(&keyEvent)->getActionId();
switch (actionId)
{
case Input::KEY_GUI_SELECT:
case Input::KEY_GUI_SELECT2:
dropDown();
break;
case Input::KEY_GUI_UP:
setSelected(getSelected() - 1);
break;
case Input::KEY_GUI_DOWN:
setSelected(getSelected() + 1);
break;
case Input::KEY_GUI_HOME:
setSelected(0);
break;
case Input::KEY_GUI_END:
if (mPopup->getListModel())
{
setSelected(mPopup->getListModel()->
getNumberOfElements() - 1);
}
break;
default:
return;
}
keyEvent.consume();
}
void DropDown::hideDrop(bool event)
{
if (event)
distributeActionEvent();
mPopup->setVisible(false);
}
void DropDown::mousePressed(gcn::MouseEvent& mouseEvent)
{
// If we have a mouse press on the widget.
if (mouseEvent.getButton() == gcn::MouseEvent::LEFT
&& !mDroppedDown && mouseEvent.getSource() == this)
{
mPushed = true;
dropDown();
}
else
{
mPushed = false;
foldUp();
hideDrop();
}
}
void DropDown::mouseReleased(gcn::MouseEvent &mouseEvent)
{
if (mIsDragged)
mPushed = false;
const int button = mouseEvent.getButton();
const int x = mouseEvent.getX();
const int y = mouseEvent.getY();
// Released outside of widget. Can happen when we have modal
// input focus.
if ((0 > y || y >= mDimension.height || x < 0 || x >= mDimension.width)
&& button == gcn::MouseEvent::LEFT)
{
if (mIsDragged)
foldUp();
}
else if (button == gcn::MouseEvent::LEFT)
{
mPushed = false;
}
mIsDragged = false;
}
void DropDown::mouseDragged(gcn::MouseEvent &mouseEvent)
{
mIsDragged = true;
mouseEvent.consume();
}
void DropDown::mouseWheelMovedUp(gcn::MouseEvent& mouseEvent)
{
setSelected(getSelected() - 1);
mouseEvent.consume();
}
void DropDown::mouseWheelMovedDown(gcn::MouseEvent& mouseEvent)
{
setSelected(getSelected() + 1);
mouseEvent.consume();
}
void DropDown::setSelectedString(const std::string &str)
{
gcn::ListModel *const listModel = mPopup->getListModel();
if (!listModel)
return;
for (int f = 0; f < listModel->getNumberOfElements(); f ++)
{
if (listModel->getElementAt(f) == str)
{
setSelected(f);
break;
}
}
}
std::string DropDown::getSelectedString() const
{
gcn::ListModel *const listModel = mPopup->getListModel();
if (!listModel)
return "";
return listModel->getElementAt(getSelected());
}
void DropDown::adjustHeight()
{
setHeight(getFont()->getHeight() + 2 * mPadding);
}
void DropDown::dropDown()
{
if (!mDroppedDown)
{
if (!mParent)
return;
mDroppedDown = true;
mFoldedUpHeight = getHeight();
adjustHeight();
int x = 0;
int y = 0;
getAbsolutePosition(x, y);
const int frame = mParent->getFrameSize();
const int pad = mPopup->getPadding();
const int pad2 = pad * 2;
// here width should be adjusted on some other parameters
mPopup->setWidth(mDimension.width - pad2 + 8);
mPopup->show(x - mPadding - frame - 1, y + mDimension.height);
mPopup->requestMoveToTop();
mPopup->requestFocus();
}
}
void DropDown::foldUp()
{
if (mDroppedDown)
{
mDroppedDown = false;
adjustHeight();
}
}
int DropDown::getSelected() const
{
return mPopup->getSelected();
}
void DropDown::setSelected(int selected)
{
if (selected >= 0)
mPopup->setSelected(selected);
}
void DropDown::setListModel(gcn::ListModel *const listModel)
{
mPopup->setListModel(listModel);
if (mPopup->getSelected() < 0)
mPopup->setSelected(0);
adjustHeight();
}
gcn::ListModel *DropDown::getListModel()
{
return mPopup->getListModel();
}
void DropDown::action(const gcn::ActionEvent &actionEvent A_UNUSED)
{
foldUp();
distributeActionEvent();
}
gcn::Rectangle DropDown::getChildrenArea()
{
if (mDroppedDown)
{
// Calculate the children area (with the one pixel border in mind)
return gcn::Rectangle(1, mFoldedUpHeight + 1,
mDimension.width - 2, mDimension.height - mFoldedUpHeight - 2);
}
return gcn::Rectangle();
}
void DropDown::valueChanged(const gcn::SelectionEvent& event A_UNUSED)
{
}
void DropDown::updateSelection()
{
mDroppedDown = false;
mPushed = false;
distributeActionEvent();
distributeValueChangedEvent();
}
void DropDown::addSelectionListener(SelectionListener* selectionListener)
{
mSelectionListeners.push_back(selectionListener);
}
void DropDown::removeSelectionListener(SelectionListener* listener)
{
mSelectionListeners.remove(listener);
}
void DropDown::distributeValueChangedEvent()
{
for (SelectionListenerIterator iter = mSelectionListeners.begin();
iter != mSelectionListeners.end();
++iter)
{
gcn::SelectionEvent event(this);
(*iter)->valueChanged(event);
}
}