/* * The ManaPlus Client * Copyright (C) 2004-2009 The Mana World Development Team * Copyright (C) 2009-2010 The Mana Developers * Copyright (C) 2011-2020 The ManaPlus Developers * Copyright (C) 2020-2023 The ManaVerse 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 . */ #include "gui/windows/buydialog.h" #include "actormanager.h" #include "configuration.h" #include "being/being.h" #include "being/playerinfo.h" #include "enums/gui/layouttype.h" #include "gui/windows/setupwindow.h" #ifdef TMWA_SUPPORT #include "gui/windows/tradewindow.h" #endif // TMWA_SUPPORT #include "gui/models/shopitems.h" #include "gui/models/sortlistmodelbuy.h" #include "gui/widgets/button.h" #include "gui/widgets/containerplacer.h" #include "gui/widgets/createwidget.h" #include "gui/widgets/dropdown.h" #include "gui/widgets/inttextfield.h" #include "gui/widgets/label.h" #include "gui/widgets/layout.h" #include "gui/widgets/scrollarea.h" #include "gui/widgets/shoplistbox.h" #include "gui/widgets/slider.h" #include "net/adminhandler.h" #include "net/buysellhandler.h" #include "net/cashshophandler.h" #include "net/markethandler.h" #include "net/net.h" #include "net/vendinghandler.h" #include "net/npchandler.h" #include "resources/iteminfo.h" #include "resources/db/unitsdb.h" #include "utils/delete2.h" #include "utils/foreach.h" #include #include "debug.h" namespace { class SortItemPriceFunctor final { public: A_DEFAULT_COPY(SortItemPriceFunctor) bool operator() (const ShopItem *const item1, const ShopItem *const item2) const { if ((item1 == nullptr) || (item2 == nullptr)) return false; const int price1 = item1->getPrice(); const int price2 = item2->getPrice(); if (price1 == price2) return item1->getDisplayName() < item2->getDisplayName(); return price1 < price2; } } itemPriceBuySorter; class SortItemNameFunctor final { public: A_DEFAULT_COPY(SortItemNameFunctor) bool operator() (const ShopItem *const item1, const ShopItem *const item2) const { if ((item1 == nullptr) || (item2 == nullptr)) return false; const std::string &name1 = item1->getDisplayName(); const std::string &name2 = item2->getDisplayName(); if (name1 == name2) return item1->getPrice() < item2->getPrice(); return name1 < name2; } } itemNameBuySorter; class SortItemIdFunctor final { public: A_DEFAULT_COPY(SortItemIdFunctor) bool operator() (const ShopItem *const item1, const ShopItem *const item2) const { if ((item1 == nullptr) || (item2 == nullptr)) return false; const int id1 = item1->getId(); const int id2 = item2->getId(); if (id1 == id2) return item1->getPrice() < item2->getPrice(); return id1 < id2; } } itemIdBuySorter; class SortItemWeightFunctor final { public: A_DEFAULT_COPY(SortItemWeightFunctor) bool operator() (const ShopItem *const item1, const ShopItem *const item2) const { if ((item1 == nullptr) || (item2 == nullptr)) return false; const int weight1 = item1->getInfo().getWeight(); const int weight2 = item2->getInfo().getWeight(); if (weight1 == weight2) return item1->getPrice() < item2->getPrice(); return weight1 < weight2; } } itemWeightBuySorter; class SortItemWeightTimesAmountFunctor final { public: A_DEFAULT_COPY(SortItemWeightTimesAmountFunctor) bool operator() (const ShopItem *const item1, const ShopItem *const item2) const { if ((item1 == nullptr) || (item2 == nullptr)) return false; const int weight1 = item1->getInfo().getWeight() * item1->getQuantity(); const int weight2 = item2->getInfo().getWeight() * item2->getQuantity(); if (weight1 == weight2) return item1->getPrice() < item2->getPrice(); return weight1 < weight2; } } itemWeightTimesAmountBuySorter; class SortItemAmountFunctor final { public: A_DEFAULT_COPY(SortItemAmountFunctor) bool operator() (const ShopItem *const item1, const ShopItem *const item2) const { if ((item1 == nullptr) || (item2 == nullptr)) return false; const int amount1 = item1->getQuantity(); const int amount2 = item2->getQuantity(); if (amount1 == amount2) return item1->getPrice() < item2->getPrice(); return amount1 < amount2; } } itemAmountBuySorter; class SortItemTypeFunctor final { public: A_DEFAULT_COPY(SortItemTypeFunctor) bool operator() (const ShopItem *const item1, const ShopItem *const item2) const { if ((item1 == nullptr) || (item2 == nullptr)) return false; const ItemDbTypeT type1 = item1->getInfo().getType(); const ItemDbTypeT type2 = item2->getInfo().getType(); if (type1 == type2) return item1->getPrice() < item2->getPrice(); return type1 < type2; } } itemTypeBuySorter; } // namespace BuyDialog::DialogList BuyDialog::instances; BuyDialog::BuyDialog() : // TRANSLATORS: buy dialog name Window(_("Create items"), Modal_false, nullptr, "buy.xml"), ActionListener(), SelectionListener(), mSortModel(new SortListModelBuy), mSortDropDown(new DropDown(this, mSortModel, false, Modal_false, this, "sort")), mFilterTextField(new TextField(this, "", LoseFocusOnTab_true, this, "namefilter", true)), mFilterLabel(nullptr), mNick(), mCurrency(), mNpcId(fromInt(Items, BeingId)), mMoney(0), mAmountItems(0), mMaxItems(0), mAdvanced(false) { init(); } BuyDialog::BuyDialog(const BeingId npcId, const std::string ¤cy) : // TRANSLATORS: buy dialog name Window(_("Buy"), Modal_false, nullptr, "buy.xml"), ActionListener(), SelectionListener(), mSortModel(nullptr), mSortDropDown(nullptr), mFilterTextField(new TextField(this, "", LoseFocusOnTab_true, this, "namefilter", true)), mFilterLabel(nullptr), mNick(), mCurrency(currency), mNpcId(npcId), mMoney(0), mAmountItems(0), mMaxItems(0), mAdvanced(Net::getNetworkType() != ServerType::TMWATHENA) { init(); } #ifdef TMWA_SUPPORT BuyDialog::BuyDialog(const std::string &nick, const std::string ¤cy) : // TRANSLATORS: buy dialog name Window(_("Buy"), Modal_false, nullptr, "buy.xml"), ActionListener(), SelectionListener(), mSortModel(new SortListModelBuy), mSortDropDown(new DropDown(this, mSortModel, false, Modal_false, this, "sort")), mFilterTextField(new TextField(this, "", LoseFocusOnTab_true, this, "namefilter", true)), mFilterLabel(nullptr), mNick(nick), mCurrency(currency), mNpcId(fromInt(Nick, BeingId)), mMoney(0), mAmountItems(0), mMaxItems(0), mAdvanced(false) { init(); } #endif // TMWA_SUPPORT BuyDialog::BuyDialog(const Being *const being, const std::string ¤cy) : // TRANSLATORS: buy dialog name Window(_("Buy"), Modal_false, nullptr, "buy.xml"), ActionListener(), SelectionListener(), mSortModel(new SortListModelBuy), mSortDropDown(new DropDown(this, mSortModel, false, Modal_false, this, "sort")), mFilterTextField(new TextField(this, "", LoseFocusOnTab_true, this, "namefilter", true)), mFilterLabel(nullptr), mNick(being != nullptr ? being->getName() : std::string()), mCurrency(currency), mNpcId(fromInt(Vending, BeingId)), mMoney(0), mAmountItems(0), mMaxItems(0), mAdvanced(true) { init(); } void BuyDialog::init() { setWindowName("Buy"); setResizable(true); setCloseButton(true); setStickyButtonLock(true); setMinWidth(260); setMinHeight(220); setDefaultSize(260, 230, ImagePosition::CENTER, 0, 0); #ifdef TMWA_SUPPORT // reset advance flag for personal shops if (mAdvanced && mNpcId == fromInt(Nick, BeingId)) { mAdvanced = false; } #endif // TMWA_SUPPORT if (setupWindow != nullptr) setupWindow->registerWindowForReset(this); mShopItems = new ShopItems(false, mCurrency); CREATEWIDGETV(mShopItemList, ShopListBox, this, mShopItems, mShopItems, ShopListBoxType::Unknown); mScrollArea = new ScrollArea(this, mShopItemList, fromBool(getOptionBool("showbackground", false), Opaque), "buy_background.xml"); mScrollArea->setHorizontalScrollPolicy(ScrollArea::SHOW_NEVER); mSlider = new Slider(this, 1.0, 1.0); mQuantityLabel = new Label(this, strprintf( "%d / %d", mAmountItems, mMaxItems)); mQuantityLabel->setAlignment(Graphics::CENTER); mMoneyLabel = new Label(this, strprintf( // TRANSLATORS: buy dialog label _("Price: %s / Total: %s"), "", "")); mAmountField = new IntTextField(this, 1, 1, 123, Enable_true, 0); mAmountField->setActionEventId("amount"); mAmountField->addActionListener(this); mAmountField->setSendAlwaysEvents(true); mAmountField->setEnabled(false); // TRANSLATORS: buy dialog label mAmountLabel = new Label(this, _("Amount:")); mAmountLabel->adjustSize(); // TRANSLATORS: This is a narrow symbol used to denote 'increasing'. // You may change this symbol if your language uses another. mIncreaseButton = new Button(this, _("+"), "inc", BUTTON_SKIN, this); // TRANSLATORS: This is a narrow symbol used to denote 'decreasing'. // You may change this symbol if your language uses another. mDecreaseButton = new Button(this, _("-"), "dec", BUTTON_SKIN, this); mBuyButton = new Button(this, mNpcId == fromInt(Items, BeingId) // TRANSLATORS: buy dialog button ? _("Create") : (mAdvanced ? _("Add") : _("Buy")), "buy", BUTTON_SKIN, this); if (mAdvanced) { mConfirmButton = new Button(this, // TRANSLATORS: buy dialog button _("Buy"), "confirm", BUTTON_SKIN, this); mConfirmButton->setEnabled(false); } else { mConfirmButton = nullptr; } // TRANSLATORS: buy dialog button mQuitButton = new Button(this, _("Quit"), "quit", BUTTON_SKIN, this); // TRANSLATORS: buy dialog button mAddMaxButton = new Button(this, _("Max"), "max", BUTTON_SKIN, this); mDecreaseButton->adjustSize(); mDecreaseButton->setWidth(mIncreaseButton->getWidth()); mIncreaseButton->setEnabled(false); mDecreaseButton->setEnabled(false); mBuyButton->setEnabled(false); mSlider->setEnabled(false); mSlider->setActionEventId("slider"); mSlider->addActionListener(this); mShopItemList->setDistributeMousePressed(false); mShopItemList->setActionEventId("buy"); mShopItemList->addActionListener(this); mShopItemList->addSelectionListener(this); mFilterTextField->setWidth(100); ContainerPlacer placer = getPlacer(0, 0); placer(0, 0, mScrollArea, 9, 5).setPadding(3); placer(0, 5, mDecreaseButton, 1, 1); placer(1, 5, mSlider, 4, 1); placer(5, 5, mIncreaseButton, 1, 1); placer(6, 5, mQuantityLabel, 2, 1); placer(8, 5, mAddMaxButton, 1, 1); placer(0, 6, mAmountLabel, 2, 1); placer(2, 6, mAmountField, 2, 1); placer(0, 7, mMoneyLabel, 8, 1); if (mSortDropDown != nullptr) { placer(0, 8, mSortDropDown, 2, 1); } else { // TRANSLATORS: buy dialog label mFilterLabel = new Label(this, _("Filter:")); mFilterLabel->adjustSize(); placer(0, 8, mFilterLabel, 2, 1); } placer(2, 8, mFilterTextField, 2, 1); if (mAdvanced) { placer(6, 8, mBuyButton, 1, 1); placer(7, 8, mConfirmButton, 1, 1); } else { placer(7, 8, mBuyButton, 1, 1); } placer(8, 8, mQuitButton, 1, 1); Layout &layout = getLayout(); layout.setRowHeight(0, LayoutType::SET); center(); loadWindowState(); enableVisibleSound(true); instances.push_back(this); setVisible(Visible_true); if (mSortDropDown != nullptr) mSortDropDown->setSelected(config.getIntValue("buySortOrder")); } BuyDialog::~BuyDialog() { delete2(mShopItems) delete2(mSortModel) instances.remove(this); if (buySellHandler != nullptr) buySellHandler->cleanDialogReference(this); } void BuyDialog::setMoney(const int amount) { mMoney = amount; mShopItemList->setPlayersMoney(amount); updateButtonsAndLabels(); } void BuyDialog::reset() { mShopItems->clear(); mShopItemList->adjustSize(); // Reset previous selected items to prevent failing asserts mShopItemList->setSelected(-1); mSlider->setValue(0); setMoney(0); } ShopItem *BuyDialog::addItem(const int id, const ItemTypeT type, const ItemColor color, const int amount, const int price) { ShopItem *const item = mShopItems->addItem(id, type, color, amount, price); mShopItemList->adjustSize(); return item; } void BuyDialog::sort() { if ((mSortDropDown != nullptr) && (mShopItems != nullptr)) { STD_VECTOR &items = mShopItems->items(); switch (mSortDropDown->getSelected()) { case 1: std::sort(items.begin(), items.end(), itemPriceBuySorter); break; case 2: std::sort(items.begin(), items.end(), itemNameBuySorter); break; case 3: std::sort(items.begin(), items.end(), itemIdBuySorter); break; case 4: std::sort(items.begin(), items.end(), itemWeightBuySorter); break; case 5: std::sort(items.begin(), items.end(), itemWeightTimesAmountBuySorter); break; case 6: std::sort(items.begin(), items.end(), itemAmountBuySorter); break; case 7: std::sort(items.begin(), items.end(), itemTypeBuySorter); break; case 0: default: break; } } } void BuyDialog::close() { switch (toInt(mNpcId, int)) { #ifdef TMWA_SUPPORT case Nick: #endif // TMWA_SUPPORT case Items: break; case Market: marketHandler->close(); break; case Cash: cashShopHandler->close(); break; case Vending: vendingHandler->close(); break; default: buySellHandler->close(); break; } Window::close(); } void BuyDialog::action(const ActionEvent &event) { const std::string &eventId = event.getId(); if (eventId == "quit") { close(); return; } else if (eventId == "sort") { sort(); if (mSortDropDown != nullptr) config.setValue("buySortOrder", mSortDropDown->getSelected()); return; } else if (eventId == "namefilter") { applyNameFilter(mFilterTextField->getText()); } const int selectedItem = mShopItemList->getSelected(); // The following actions require a valid selection if (selectedItem < 0 || selectedItem >= mShopItems->getNumberOfElements()) return; if (eventId == "slider") { mAmountItems = CAST_S32(mSlider->getValue()); mAmountField->setValue(mAmountItems); updateButtonsAndLabels(); } else if (eventId == "inc" && mAmountItems < mMaxItems) { mAmountItems++; mSlider->setValue(mAmountItems); mAmountField->setValue(mAmountItems); updateButtonsAndLabels(); } else if (eventId == "dec" && mAmountItems > 1) { mAmountItems--; mSlider->setValue(mAmountItems); mAmountField->setValue(mAmountItems); updateButtonsAndLabels(); } else if (eventId == "max") { mAmountItems = mMaxItems; mSlider->setValue(mAmountItems); mAmountField->setValue(mAmountItems); updateButtonsAndLabels(); } else if (eventId == "amount") { mAmountItems = mAmountField->getValue(); mSlider->setValue(mAmountItems); updateButtonsAndLabels(); } else if (eventId == "buy" && mAmountItems > 0 && mAmountItems <= mMaxItems) { ShopItem *const item = mShopItems->at(selectedItem); if (item == nullptr) return; if (mNpcId == fromInt(Items, BeingId)) { adminHandler->createItems(item->getId(), item->getColor(), mAmountItems); } #ifdef TMWA_SUPPORT else if (mNpcId == fromInt(Nick, BeingId)) { if (tradeWindow != nullptr) { buySellHandler->sendBuyRequest(mNick, item, mAmountItems); tradeWindow->addAutoMoney(mNick, item->getPrice() * mAmountItems); } } #endif // TMWA_SUPPORT else if (mNpcId == fromInt(Vending, BeingId)) { item->increaseUsedQuantity(mAmountItems); item->update(); if (mConfirmButton != nullptr) mConfirmButton->setEnabled(true); } #ifdef TMWA_SUPPORT else if (mNpcId != fromInt(Nick, BeingId)) #else else #endif // TMWA_SUPPORT { if (mAdvanced) { item->increaseUsedQuantity(mAmountItems); item->update(); if (mConfirmButton != nullptr) mConfirmButton->setEnabled(true); } else if (mNpcId == fromInt(Market, BeingId)) { marketHandler->buyItem(item->getId(), item->getType(), item->getColor(), mAmountItems); item->increaseQuantity(-mAmountItems); item->update(); } else if (mNpcId == fromInt(Cash, BeingId)) { cashShopHandler->buyItem(item->getPrice(), item->getId(), item->getColor(), mAmountItems); } else { npcHandler->buyItem(mNpcId, item->getId(), item->getColor(), mAmountItems); } updateSlider(selectedItem); } } else if (eventId == "confirm") { STD_VECTOR &items = mShopItems->allItems(); if (mNpcId == fromInt(Market, BeingId)) { marketHandler->buyItems(items); } else if (mNpcId == fromInt(Vending, BeingId)) { const Being *const being = actorManager->findBeingByName( mNick, ActorType::Player); if (being != nullptr) { vendingHandler->buyItems(being, items); } } else if (mNpcId == fromInt(Cash, BeingId)) { cashShopHandler->buyItems(0, items); } else { npcHandler->buyItems(items); } close(); } } void BuyDialog::updateSlider(const int selectedItem) { // Update money and adjust the max number of items // that can be bought mMaxItems -= mAmountItems; const ShopItem *const item = mShopItems->at(selectedItem); if (item != nullptr) setMoney(mMoney - mAmountItems * item->getPrice()); else setMoney(mMoney); // Reset selection mAmountItems = 1; mSlider->setScale(1, mMaxItems); mSlider->setValue(1); } void BuyDialog::valueChanged(const SelectionEvent &event A_UNUSED) { // Reset amount of items and update labels mAmountItems = 1; mSlider->setValue(1); updateButtonsAndLabels(); mSlider->setScale(1, mMaxItems); mAmountField->setRange(1, mMaxItems); mAmountField->setValue(1); } void BuyDialog::updateButtonsAndLabels() { const int selectedItem = mShopItemList->getSelected(); int price = 0; if (selectedItem > -1) { const ShopItem *const item = mShopItems->at(selectedItem); if (item != nullptr) { const int itemPrice = item->getPrice(); // Calculate how many the player can afford if (mNpcId == fromInt(Items, BeingId)) // /createitems "shop" mMaxItems = 100; else if (itemPrice != 0) mMaxItems = mMoney / itemPrice; else mMaxItems = 1; // Calculate how many the player can carry using namespace PlayerInfo; // trick for line-length. const int itemWeight = item->getInfo().getWeight(); // Should be inside if, but line-length... const int myTotalWeight = getAttribute(Attributes::TOTAL_WEIGHT); const int myMaxWeight = getAttribute(Attributes::MAX_WEIGHT); if (itemWeight != 0) { const int myFreeWeight = myMaxWeight - myTotalWeight; const int canCarry = myFreeWeight / itemWeight; mMaxItems = std::min(mMaxItems, canCarry); } if (mNpcId == fromInt(Market, BeingId)) { if (mMaxItems > item->getQuantity()) mMaxItems = item->getQuantity(); } else if (item->getQuantity() > 0 && mMaxItems > item->getQuantity()) { mMaxItems = item->getQuantity(); } if (mAmountItems > mMaxItems) mAmountItems = mMaxItems; price = mAmountItems * itemPrice; } } else { mMaxItems = 0; mAmountItems = 0; } mIncreaseButton->setEnabled(mAmountItems < mMaxItems); mDecreaseButton->setEnabled(mAmountItems > 1); mBuyButton->setEnabled(mAmountItems > 0); mSlider->setEnabled(mMaxItems > 1); mAmountField->setEnabled(mAmountItems > 0); mQuantityLabel->setCaption(strprintf("%d / %d", mAmountItems, mMaxItems)); // TRANSLATORS: buy dialog label mMoneyLabel->setCaption(strprintf(_("Price: %s / Total: %s"), UnitsDb::formatCurrency(mCurrency, price).c_str(), UnitsDb::formatCurrency(mCurrency, mMoney - price).c_str())); } void BuyDialog::setVisible(Visible visible) { Window::setVisible(visible); if (visible == Visible_true && (mShopItemList != nullptr)) mShopItemList->requestFocus(); else scheduleDelete(); } void BuyDialog::closeAll() { FOR_EACH (DialogList::const_iterator, it, instances) { if (*it != nullptr) (*it)->close(); } } void BuyDialog::applyNameFilter(const std::string &filter) { STD_VECTOR &items = mShopItems->allItems(); std::string filterStr = filter; toLower(filterStr); FOR_EACH (STD_VECTOR::iterator, it, items) { ShopItem *const item = *it; if (item == nullptr) continue; std::string name = item->getName(); toLower(name); if (name.find(filterStr) != std::string::npos) item->setVisible(true); else item->setVisible(false); } mShopItems->updateList(); }