/* * The ManaPlus Client * Copyright (C) 2007-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/itemshortcutcontainer.h" #include "client.h" #include "dragdrop.h" #include "inventory.h" #include "item.h" #include "itemshortcut.h" #include "spellshortcut.h" #include "being/playerinfo.h" #include "input/inputmanager.h" #include "gui/viewport.h" #include "gui/fonts/font.h" #include "gui/popups/itempopup.h" #include "gui/popups/spellpopup.h" #include "gui/windows/inventorywindow.h" #include "gui/windows/skilldialog.h" #include "gui/widgets/skillinfo.h" #include "resources/image.h" #include "resources/skillconsts.h" #include "utils/delete2.h" #include "debug.h" ItemShortcutContainer::ItemShortcutContainer(Widget2 *const widget, const unsigned number) : ShortcutContainer(widget), mItemClicked(false), mNumber(number), mItemPopup(new ItemPopup), mSpellPopup(new SpellPopup), mEquipedColor(getThemeColor(Theme::ITEM_EQUIPPED)), mEquipedColor2(getThemeColor(Theme::ITEM_EQUIPPED_OUTLINE)), mUnEquipedColor(getThemeColor(Theme::ITEM_NOT_EQUIPPED)), mUnEquipedColor2(getThemeColor(Theme::ITEM_NOT_EQUIPPED_OUTLINE)) { mItemPopup->postInit(); mSpellPopup->postInit(); addMouseListener(this); addWidgetListener(this); mForegroundColor2 = getThemeColor(Theme::TEXT_OUTLINE); mBackgroundImg = Theme::getImageFromThemeXml( "item_shortcut_background.xml", "background.xml"); if (itemShortcut[mNumber]) mMaxItems = itemShortcut[mNumber]->getItemCount(); else mMaxItems = 0; if (mBackgroundImg) { mBackgroundImg->setAlpha(client->getGuiAlpha()); mBoxHeight = mBackgroundImg->getHeight(); mBoxWidth = mBackgroundImg->getWidth(); } else { mBoxHeight = 1; mBoxWidth = 1; } mForegroundColor = getThemeColor(Theme::TEXT); } ItemShortcutContainer::~ItemShortcutContainer() { if (mBackgroundImg) { mBackgroundImg->decRef(); mBackgroundImg = nullptr; } delete2(mItemPopup); delete2(mSpellPopup); } void ItemShortcutContainer::setWidget2(const Widget2 *const widget) { Widget2::setWidget2(widget); mEquipedColor = getThemeColor(Theme::ITEM_EQUIPPED); mEquipedColor2 = getThemeColor(Theme::ITEM_EQUIPPED_OUTLINE); mUnEquipedColor = getThemeColor(Theme::ITEM_NOT_EQUIPPED); mUnEquipedColor2 = getThemeColor(Theme::ITEM_NOT_EQUIPPED_OUTLINE); mForegroundColor = getThemeColor(Theme::TEXT); mForegroundColor2 = getThemeColor(Theme::TEXT_OUTLINE); } void ItemShortcutContainer::draw(Graphics *graphics) { BLOCK_START("ItemShortcutContainer::draw") const ItemShortcut *const selShortcut = itemShortcut[mNumber]; if (!selShortcut) { BLOCK_END("ItemShortcutContainer::draw") return; } if (client->getGuiAlpha() != mAlpha) { if (mBackgroundImg) mBackgroundImg->setAlpha(mAlpha); mAlpha = client->getGuiAlpha(); } Font *const font = getFont(); drawBackground(graphics); const Inventory *const inv = PlayerInfo::getInventory(); if (!inv) { BLOCK_END("ItemShortcutContainer::draw") return; } // +++ for future usage need reorder drawing images before text or back for (unsigned i = 0; i < mMaxItems; i++) { const int itemX = (i % mGridWidth) * mBoxWidth; const int itemY = (i / mGridWidth) * mBoxHeight; // Draw item keyboard shortcut. const std::string key = inputManager.getKeyValueString( Input::KEY_SHORTCUT_1 + i); graphics->setColorAll(mForegroundColor, mForegroundColor); font->drawString(graphics, key, itemX + 2, itemY + 2); const int itemId = selShortcut->getItem(i); const unsigned char itemColor = selShortcut->getItemColor(i); if (itemId < 0) continue; // this is item if (itemId < SPELL_MIN_ID) { const Item *const item = inv->findItem(itemId, itemColor); if (item) { // Draw item icon. Image *const image = item->getImage(); if (image) { std::string caption; if (item->getQuantity() > 1) caption = toString(item->getQuantity()); else if (item->isEquipped()) caption = "Eq."; image->setAlpha(1.0F); graphics->drawImage(image, itemX, itemY); if (item->isEquipped()) { graphics->setColorAll(mEquipedColor, mEquipedColor2); } else { graphics->setColorAll(mUnEquipedColor, mUnEquipedColor2); } font->drawString(graphics, caption, itemX + (mBoxWidth - font->getWidth(caption)) / 2, itemY + mBoxHeight - 14); } } } else if (itemId < SKILL_MIN_ID && spellManager) { // this is magic shortcut const TextCommand *const spell = spellManager ->getSpellByItem(itemId); if (spell) { if (!spell->isEmpty()) { Image *const image = spell->getImage(); if (image) { image->setAlpha(1.0F); graphics->drawImage(image, itemX, itemY); } } font->drawString(graphics, spell->getSymbol(), itemX + 2, itemY + mBoxHeight / 2); } } else if (skillDialog) { const SkillInfo *const skill = skillDialog->getSkill( itemId - SKILL_MIN_ID); if (skill) { Image *const image = skill->data->icon; if (image) { image->setAlpha(1.0F); graphics->drawImage(image, itemX, itemY); } font->drawString(graphics, skill->data->shortName, itemX + 2, itemY + mBoxHeight / 2); } } } BLOCK_END("ItemShortcutContainer::draw") } void ItemShortcutContainer::mouseDragged(MouseEvent &event) { ItemShortcut *const selShortcut = itemShortcut[mNumber]; if (!selShortcut) return; if (event.getButton() == MouseButton::LEFT) { if (dragDrop.isEmpty() && mItemClicked) { mItemClicked = false; const int index = getIndexFromGrid(event.getX(), event.getY()); if (index == -1) return; const int itemId = selShortcut->getItem(index); const unsigned char itemColor = selShortcut->getItemColor(index); if (itemId < 0) return; event.consume(); if (itemId < SPELL_MIN_ID) { // items if (!PlayerInfo::getInventory()) return; const Item *const item = PlayerInfo::getInventory()->findItem( itemId, itemColor); if (item) { selShortcut->removeItem(index); dragDrop.dragItem(item, DRAGDROP_SOURCE_SHORTCUTS, index); } else { dragDrop.clear(); } } else if (itemId < SKILL_MIN_ID) { // spells/commands if (!spellManager) { dragDrop.clear(); return; } const TextCommand *const spell = spellManager->getSpellByItem( itemId); if (spell) { selShortcut->removeItem(index); dragDrop.dragCommand(spell, DRAGDROP_SOURCE_SHORTCUTS, index); dragDrop.setItem(itemId); } else { dragDrop.clear(); } } else { // skills if (!skillDialog) { dragDrop.clear(); return; } const SkillInfo *const skill = skillDialog->getSkillByItem(itemId); if (skill) { selShortcut->removeItem(index); dragDrop.dragSkill(skill, DRAGDROP_SOURCE_SHORTCUTS, index); dragDrop.setItem(itemId); } else { dragDrop.clear(); } } } } } void ItemShortcutContainer::mousePressed(MouseEvent &event) { ItemShortcut *const selShortcut = itemShortcut[mNumber]; if (!selShortcut) return; const int index = getIndexFromGrid(event.getX(), event.getY()); if (index == -1) return; if (event.getButton() == MouseButton::LEFT) { event.consume(); // Stores the selected item if theirs one. if (selShortcut->isItemSelected() && inventoryWindow && (inventoryWindow->isWindowVisible() || selShortcut->getSelectedItem() >= SPELL_MIN_ID)) { selShortcut->setItem(index); selShortcut->setItemSelected(-1); if (spellShortcut) spellShortcut->setItemSelected(-1); inventoryWindow->unselectItem(); } else if (selShortcut->getItem(index)) { mItemClicked = true; } } else if (event.getButton() == MouseButton::RIGHT) { event.consume(); if (viewport && selShortcut) { viewport->showItemPopup(selShortcut->getItem(index), selShortcut->getItemColor(index)); } } } void ItemShortcutContainer::mouseReleased(MouseEvent &event) { ItemShortcut *const selShortcut = itemShortcut[mNumber]; if (!selShortcut) return; if (event.getButton() == MouseButton::LEFT) { if (selShortcut->isItemSelected()) selShortcut->setItemSelected(-1); const int index = getIndexFromGrid(event.getX(), event.getY()); if (index == -1) return; if (dragDrop.isEmpty()) { if (selShortcut->getItem(index) && mItemClicked) selShortcut->useItem(index); } else { if (dragDrop.getSource() == DRAGDROP_SOURCE_SHORTCUTS) { const int oldIndex = dragDrop.getTag(); selShortcut->setItem(oldIndex, dragDrop.getItem(), dragDrop.getItemColor()); selShortcut->swap(oldIndex, index); } else { selShortcut->setItem(index, dragDrop.getItem(), dragDrop.getItemColor()); } dragDrop.clear(); dragDrop.deselect(); } mItemClicked = false; } } void ItemShortcutContainer::mouseMoved(MouseEvent &event) { const ItemShortcut *const selShortcut = itemShortcut[mNumber]; if (!selShortcut) return; const int index = getIndexFromGrid(event.getX(), event.getY()); if (index == -1) return; const int itemId = selShortcut->getItem(index); const unsigned char itemColor = selShortcut->getItemColor(index); if (itemId < 0) return; if (itemId < SPELL_MIN_ID) { mSpellPopup->setVisible(false); Inventory *const inv = PlayerInfo::getInventory(); if (!inv) return; const Item *const item = inv->findItem(itemId, itemColor); if (item && viewport) { mItemPopup->setItem(item); mItemPopup->position(viewport->getMouseX(), viewport->getMouseY()); } else { mItemPopup->setVisible(false); } } else if (itemId < SKILL_MIN_ID && spellManager) { mItemPopup->setVisible(false); const TextCommand *const spell = spellManager->getSpellByItem(itemId); if (spell && viewport) { mSpellPopup->setItem(spell); mSpellPopup->view(viewport->getMouseX(), viewport->getMouseY()); } else { mSpellPopup->setVisible(false); } } else if (skillDialog) { mItemPopup->setVisible(false); } } // Hide ItemTooltip void ItemShortcutContainer::mouseExited(MouseEvent &event A_UNUSED) { if (mItemPopup) mItemPopup->setVisible(false); if (mSpellPopup) mSpellPopup->setVisible(false); } void ItemShortcutContainer::widgetHidden(const Event &event A_UNUSED) { if (mItemPopup) mItemPopup->setVisible(false); if (mSpellPopup) mSpellPopup->setVisible(false); }