mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 14:56:37 +00:00 
			
		
		
		
	equipping items works, but only if you have more than one of the item that you're equipping
This commit is contained in:
		
							parent
							
								
									765881a61d
								
							
						
					
					
						commit
						71de10cb7e
					
				
					 8 changed files with 145 additions and 11 deletions
				
			
		|  | @ -81,6 +81,8 @@ void ContainerBase::onSelectedItemImpl(MyGUI::Widget* _sender, int count) | |||
|     mSelectedItem->attachToWidget(mDragAndDrop->mDragAndDropWidget); | ||||
| 
 | ||||
|     MWWorld::Ptr object = *mSelectedItem->getUserData<MWWorld::Ptr>(); | ||||
|     _unequipItem(object); | ||||
| 
 | ||||
|     int originalCount = object.getRefData().getCount(); | ||||
|     object.getRefData().setCount(count); | ||||
|     mDragAndDrop->mStore.add(object); | ||||
|  | @ -90,6 +92,7 @@ void ContainerBase::onSelectedItemImpl(MyGUI::Widget* _sender, int count) | |||
|     MWBase::Environment::get().getSoundManager()->playSound (sound, 1.0, 1.0); | ||||
| 
 | ||||
|     mDragAndDrop->mDraggedWidget = mSelectedItem; | ||||
|     static_cast<MyGUI::ImageBox*>(mSelectedItem)->setImageTexture(""); // remove the background texture (not visible during drag)
 | ||||
|     static_cast<MyGUI::TextBox*>(mSelectedItem->getChildAt(0)->getChildAt(0))->setCaption( | ||||
|         getCountString((*mDragAndDrop->mStore.begin()).getRefData().getCount())); | ||||
| 
 | ||||
|  | @ -181,9 +184,46 @@ void ContainerBase::drawItems() | |||
| 
 | ||||
|     /// \todo performance improvement: don't create/destroy all the widgets everytime the container window changes size, only reposition them
 | ||||
| 
 | ||||
|     std::vector< std::pair<MWWorld::Ptr, ItemState> > items; | ||||
| 
 | ||||
|     std::vector<MWWorld::Ptr> equippedItems = getEquippedItems(); | ||||
| 
 | ||||
|     // filter out the equipped items of categories we don't want
 | ||||
|     std::vector<MWWorld::Ptr> unwantedItems = equippedItems; | ||||
|     for (MWWorld::ContainerStoreIterator iter (containerStore.begin(categories)); iter!=containerStore.end(); ++iter) | ||||
|     { | ||||
|         std::vector<MWWorld::Ptr>::iterator found = std::find(unwantedItems.begin(), unwantedItems.end(), *iter); | ||||
|         if (found != unwantedItems.end()) | ||||
|         { | ||||
|             unwantedItems.erase(found); | ||||
|         } | ||||
|     } | ||||
|     // now erase everything that's still in unwantedItems.
 | ||||
|     for (std::vector<MWWorld::Ptr>::iterator it=unwantedItems.begin(); | ||||
|         it != unwantedItems.end(); ++it) | ||||
|     { | ||||
|         equippedItems.erase(std::find(unwantedItems.begin(), unwantedItems.end(), *it)); | ||||
|     } | ||||
|     // and add the items that are left (= have the correct category)
 | ||||
|     for (std::vector<MWWorld::Ptr>::const_iterator it=equippedItems.begin(); | ||||
|         it != equippedItems.end(); ++it) | ||||
|     { | ||||
|         items.push_back( std::make_pair(*it, ItemState_Equipped) ); | ||||
|     } | ||||
| 
 | ||||
|     // now add the regular items
 | ||||
|     for (MWWorld::ContainerStoreIterator iter (containerStore.begin(categories)); iter!=containerStore.end(); ++iter) | ||||
|     { | ||||
|         /// \todo sorting
 | ||||
|         if (std::find(equippedItems.begin(), equippedItems.end(), *iter) == equippedItems.end()) | ||||
|             items.push_back( std::make_pair(*iter, ItemState_Normal) ); | ||||
|     } | ||||
| 
 | ||||
|     for (std::vector< std::pair<MWWorld::Ptr, ItemState> >::const_iterator it=items.begin(); | ||||
|         it != items.end(); ++it) | ||||
|     { | ||||
|         index++; | ||||
|         const MWWorld::Ptr* iter = &((*it).first); | ||||
|         if(iter->getRefData().getCount() > 0 && !(onlyMagic && MWWorld::Class::get(*iter).getEnchantment(*iter) == "" && iter->getTypeName() != typeid(ESM::Potion).name())) | ||||
|         { | ||||
|             std::string path = std::string("icons\\"); | ||||
|  | @ -194,7 +234,22 @@ void ContainerBase::drawItems() | |||
|             MyGUI::ImageBox* backgroundWidget = mContainerWidget->createWidget<ImageBox>("ImageBox", MyGUI::IntCoord(x, y, 42, 42), MyGUI::Align::Default); | ||||
|             backgroundWidget->setUserString("ToolTipType", "ItemPtr"); | ||||
|             backgroundWidget->setUserData(*iter); | ||||
|             backgroundWidget->setImageTexture( isMagic ? "textures\\menu_icon_magic.dds" : ""); | ||||
| 
 | ||||
|             std::string backgroundTex = "textures\\menu_icon"; | ||||
|             if (isMagic) | ||||
|                 backgroundTex += "_magic"; | ||||
|             if (it->second == ItemState_Normal) | ||||
|             { | ||||
|                 if (!isMagic) | ||||
|                     backgroundTex = ""; | ||||
|             } | ||||
|             else if (it->second == ItemState_Equipped) | ||||
|             { | ||||
|                 backgroundTex += "_equip"; | ||||
|             } | ||||
|             backgroundTex += ".dds"; | ||||
| 
 | ||||
|             backgroundWidget->setImageTexture(backgroundTex); | ||||
|             backgroundWidget->setProperty("ImageCoord", "0 0 42 42"); | ||||
|             backgroundWidget->eventMouseButtonClick += MyGUI::newDelegate(this, &ContainerBase::onSelectedItem); | ||||
|             backgroundWidget->eventMouseWheel += MyGUI::newDelegate(this, &ContainerBase::onMouseWheel); | ||||
|  |  | |||
|  | @ -63,6 +63,15 @@ namespace MWGui | |||
|             Filter_Misc = 0x05 | ||||
|         }; | ||||
| 
 | ||||
|         enum ItemState | ||||
|         { | ||||
|             ItemState_Normal = 0x01, | ||||
|             ItemState_Equipped = 0x02, | ||||
| 
 | ||||
|             // unimplemented
 | ||||
|             ItemState_Barter = 0x03 | ||||
|         }; | ||||
| 
 | ||||
|         void setWidgets(MyGUI::Widget* containerWidget, MyGUI::ScrollView* itemView); ///< only call once
 | ||||
| 
 | ||||
|         void openContainer(MWWorld::Ptr container); | ||||
|  | @ -87,7 +96,10 @@ namespace MWGui | |||
| 
 | ||||
|         std::string getCountString(const int count); | ||||
| 
 | ||||
|         // to be reimplemented by InventoryWindow
 | ||||
|         virtual bool isInventory() { return false; } | ||||
|         virtual std::vector<MWWorld::Ptr> getEquippedItems() { return std::vector<MWWorld::Ptr>(); } | ||||
|         virtual void _unequipItem(MWWorld::Ptr item) { ; } | ||||
| 
 | ||||
|         void drawItems(); | ||||
|     }; | ||||
|  |  | |||
|  | @ -151,6 +151,8 @@ namespace MWGui | |||
|                 // can't be equipped, try to use instead
 | ||||
|                 boost::shared_ptr<MWWorld::Action> action = MWWorld::Class::get(ptr).use(ptr); | ||||
| 
 | ||||
|                 std::cout << "Item can't be equipped" << std::endl; | ||||
| 
 | ||||
|                 // execute action and script
 | ||||
|                 MWBase::Environment::get().getWorld()->executeActionScript(ptr, action); | ||||
| 
 | ||||
|  | @ -172,19 +174,31 @@ namespace MWGui | |||
|             { | ||||
|                 // put back in inventory
 | ||||
|                 MWWorld::InventoryStore& invStore = static_cast<MWWorld::InventoryStore&>(MWWorld::Class::get(mContainer).getContainerStore(mContainer)); | ||||
|                 invStore.add(ptr); | ||||
| 
 | ||||
|                 // get a ContainerStoreIterator to the item we just re-added into the inventory
 | ||||
|                 MWWorld::ContainerStoreIterator it = invStore.begin(); | ||||
|                 MWWorld::ContainerStoreIterator nextIt = ++it; | ||||
|                 while (nextIt != invStore.end()) | ||||
|                 MWWorld::ContainerStoreIterator it = invStore.add(ptr); | ||||
| 
 | ||||
|                 // retrieve iterator to the item we just re-added (if stacking didn't happen).
 | ||||
|                 // if stacking happened, the iterator was already returned by the add() call
 | ||||
|                 /// \todo this does not work!
 | ||||
|                 if (it == invStore.end()) | ||||
|                 { | ||||
|                     ++it; | ||||
|                     ++nextIt; | ||||
|                     std::cout << "stacking didn't happen" << std::endl; | ||||
|                     for (MWWorld::ContainerStoreIterator it2 = invStore.begin(); | ||||
|                         it2 != invStore.end(); ++it2) | ||||
|                     { | ||||
|                         if (*it2 == ptr) | ||||
|                         { | ||||
|                             std::cout << "found iterator" << std::endl; | ||||
|                             it = it2; | ||||
|                             return; | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
| 
 | ||||
|                 // equip the item in the first available slot
 | ||||
|                 invStore.equip(slots.first.front(), it); | ||||
| 
 | ||||
|                 std::cout << "Equipped item in slot " << slots.first.front() << std::endl; | ||||
|             } | ||||
| 
 | ||||
|             drawItems(); | ||||
|  | @ -197,4 +211,42 @@ namespace MWGui | |||
|         } | ||||
|     } | ||||
| 
 | ||||
|     std::vector<MWWorld::Ptr> InventoryWindow::getEquippedItems() | ||||
|     { | ||||
|         MWWorld::InventoryStore& invStore = static_cast<MWWorld::InventoryStore&>(MWWorld::Class::get(mContainer).getContainerStore(mContainer)); | ||||
| 
 | ||||
|         std::vector<MWWorld::Ptr> items; | ||||
| 
 | ||||
|         for (int slot=0; slot < MWWorld::InventoryStore::Slots; ++slot) | ||||
|         { | ||||
|             MWWorld::ContainerStoreIterator it = invStore.getSlot(slot); | ||||
|             if (it != invStore.end()) | ||||
|             { | ||||
|                 std::cout << "slot " << slot << " is equipped" << std::endl; | ||||
|                 items.push_back(*it); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 std::cout << "slot " << slot << " is empty " << std::endl; | ||||
|             } | ||||
|              | ||||
|         } | ||||
| 
 | ||||
|         return items; | ||||
|     } | ||||
| 
 | ||||
|     void InventoryWindow::_unequipItem(MWWorld::Ptr item) | ||||
|     { | ||||
|         MWWorld::InventoryStore& invStore = static_cast<MWWorld::InventoryStore&>(MWWorld::Class::get(mContainer).getContainerStore(mContainer)); | ||||
| 
 | ||||
|         for (int slot=0; slot < MWWorld::InventoryStore::Slots; ++slot) | ||||
|         { | ||||
|             MWWorld::ContainerStoreIterator it = invStore.getSlot(slot); | ||||
|             if (it != invStore.end() && *it == item) | ||||
|             { | ||||
|                 invStore._freeSlot(slot); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -51,6 +51,8 @@ namespace MWGui | |||
|             void onPinToggled(); | ||||
| 
 | ||||
|             virtual bool isInventory() { return true; } | ||||
|             virtual std::vector<MWWorld::Ptr> getEquippedItems(); | ||||
|             virtual void _unequipItem(MWWorld::Ptr item); | ||||
|     }; | ||||
| } | ||||
| #endif // Inventory_H
 | ||||
|  |  | |||
|  | @ -58,7 +58,7 @@ bool MWWorld::ContainerStore::stacks(const Ptr& ptr1, const Ptr& ptr2) | |||
|     return false; | ||||
| } | ||||
| 
 | ||||
| void MWWorld::ContainerStore::add (const Ptr& ptr) | ||||
| MWWorld::ContainerStoreIterator MWWorld::ContainerStore::add (const Ptr& ptr) | ||||
| { | ||||
|     int type = getType(ptr); | ||||
| 
 | ||||
|  | @ -71,12 +71,13 @@ void MWWorld::ContainerStore::add (const Ptr& ptr) | |||
|             iter->getRefData().setCount( iter->getRefData().getCount() + ptr.getRefData().getCount() ); | ||||
| 
 | ||||
|             flagAsModified(); | ||||
|             return; | ||||
|             return iter; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // if we got here, this means no stacking
 | ||||
|     addImpl(ptr); | ||||
|     return end(); | ||||
| } | ||||
| 
 | ||||
| void MWWorld::ContainerStore::addImpl (const Ptr& ptr) | ||||
|  |  | |||
|  | @ -66,13 +66,15 @@ namespace MWWorld | |||
| 
 | ||||
|             ContainerStoreIterator end(); | ||||
| 
 | ||||
|             void add (const Ptr& ptr); | ||||
|             ContainerStoreIterator add (const Ptr& ptr); | ||||
|             ///< Add the item pointed to by \a ptr to this container. (Stacks automatically if needed)
 | ||||
|             ///
 | ||||
|             /// \note The item pointed to is not required to exist beyond this function call.
 | ||||
|             ///
 | ||||
|             /// \attention Do not add items to an existing stack by increasing the count instead of
 | ||||
|             /// calling this function!
 | ||||
|             ///
 | ||||
|             /// @return if stacking happened, return iterator to the item that was stacked against, otherwise end() iterator
 | ||||
| 
 | ||||
|         protected: | ||||
|             void addImpl (const Ptr& ptr); | ||||
|  |  | |||
|  | @ -97,6 +97,13 @@ void MWWorld::InventoryStore::equip (int slot, const ContainerStoreIterator& ite | |||
|     flagAsModified(); | ||||
| } | ||||
| 
 | ||||
| void MWWorld::InventoryStore::_freeSlot(int slot) | ||||
| { | ||||
|     mSlots[slot] = end(); | ||||
| 
 | ||||
|     flagAsModified(); | ||||
| } | ||||
| 
 | ||||
| MWWorld::ContainerStoreIterator MWWorld::InventoryStore::getSlot (int slot) | ||||
| { | ||||
|     if (slot<0 || slot>=static_cast<int> (mSlots.size())) | ||||
|  |  | |||
|  | @ -60,6 +60,9 @@ namespace MWWorld | |||
|             void equip (int slot, const ContainerStoreIterator& iterator); | ||||
|             ///< \note \a iteartor can be an end-iterator
 | ||||
| 
 | ||||
|             void _freeSlot(int slot); | ||||
|             ///< this method is dangerous, as it doesn't do re-stacking items - you probably want to use equip()
 | ||||
| 
 | ||||
|             ContainerStoreIterator getSlot (int slot); | ||||
| 
 | ||||
|             void autoEquip (const MWMechanics::NpcStats& stats); | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue