mirror of
https://github.com/luanti-org/luanti.git
synced 2025-06-27 16:36:03 +00:00
Formspec: allow lists to change size and existence while the formspec is open (#9700)
Fixes #9640.
This commit is contained in:
parent
241bf44260
commit
4fb6b6afa7
3 changed files with 36 additions and 41 deletions
|
@ -47,7 +47,8 @@ GUIInventoryList::GUIInventoryList(gui::IGUIEnvironment *env,
|
|||
m_fs_menu(fs_menu),
|
||||
m_options(options),
|
||||
m_font(font),
|
||||
m_hovered_i(-1)
|
||||
m_hovered_i(-1),
|
||||
m_already_warned(false)
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -58,20 +59,27 @@ void GUIInventoryList::draw()
|
|||
|
||||
Inventory *inv = m_invmgr->getInventory(m_inventoryloc);
|
||||
if (!inv) {
|
||||
warningstream << "GUIInventoryList::draw(): "
|
||||
<< "The inventory location "
|
||||
<< "\"" << m_inventoryloc.dump() << "\" doesn't exist anymore"
|
||||
<< std::endl;
|
||||
if (!m_already_warned) {
|
||||
warningstream << "GUIInventoryList::draw(): "
|
||||
<< "The inventory location "
|
||||
<< "\"" << m_inventoryloc.dump() << "\" doesn't exist"
|
||||
<< std::endl;
|
||||
m_already_warned = true;
|
||||
}
|
||||
return;
|
||||
}
|
||||
InventoryList *ilist = inv->getList(m_listname);
|
||||
if (!ilist) {
|
||||
warningstream << "GUIInventoryList::draw(): "
|
||||
<< "The inventory list \"" << m_listname << "\" @ \""
|
||||
<< m_inventoryloc.dump() << "\" doesn't exist anymore"
|
||||
<< std::endl;
|
||||
if (!m_already_warned) {
|
||||
warningstream << "GUIInventoryList::draw(): "
|
||||
<< "The inventory list \"" << m_listname << "\" @ \""
|
||||
<< m_inventoryloc.dump() << "\" doesn't exist"
|
||||
<< std::endl;
|
||||
m_already_warned = true;
|
||||
}
|
||||
return;
|
||||
}
|
||||
m_already_warned = false;
|
||||
|
||||
video::IVideoDriver *driver = Environment->getVideoDriver();
|
||||
Client *client = m_fs_menu->getClient();
|
||||
|
@ -80,9 +88,11 @@ void GUIInventoryList::draw()
|
|||
core::rect<s32> imgrect(0, 0, m_slot_size.X, m_slot_size.Y);
|
||||
v2s32 base_pos = AbsoluteRect.UpperLeftCorner;
|
||||
|
||||
const s32 list_size = (s32)ilist->getSize();
|
||||
|
||||
for (s32 i = 0; i < m_geom.X * m_geom.Y; i++) {
|
||||
s32 item_i = i + m_start_item_i;
|
||||
if (item_i >= (s32)ilist->getSize())
|
||||
if (item_i >= list_size)
|
||||
break;
|
||||
|
||||
v2s32 p((i % m_geom.X) * m_slot_spacing.X,
|
||||
|
@ -192,10 +202,19 @@ bool GUIInventoryList::OnEvent(const SEvent &event)
|
|||
|
||||
s32 GUIInventoryList::getItemIndexAtPos(v2s32 p) const
|
||||
{
|
||||
// no item if no gui element at pointer
|
||||
if (!IsVisible || AbsoluteClippingRect.getArea() <= 0 ||
|
||||
!AbsoluteClippingRect.isPointInside(p))
|
||||
return -1;
|
||||
|
||||
// there can not be an item if the inventory or the inventorylist does not exist
|
||||
Inventory *inv = m_invmgr->getInventory(m_inventoryloc);
|
||||
if (!inv)
|
||||
return -1;
|
||||
InventoryList *ilist = inv->getList(m_listname);
|
||||
if (!ilist)
|
||||
return -1;
|
||||
|
||||
core::rect<s32> imgrect(0, 0, m_slot_size.X, m_slot_size.Y);
|
||||
v2s32 base_pos = AbsoluteRect.UpperLeftCorner;
|
||||
|
||||
|
@ -210,7 +229,8 @@ s32 GUIInventoryList::getItemIndexAtPos(v2s32 p) const
|
|||
|
||||
rect.clipAgainst(AbsoluteClippingRect);
|
||||
|
||||
if (rect.getArea() > 0 && rect.isPointInside(p))
|
||||
if (rect.getArea() > 0 && rect.isPointInside(p) &&
|
||||
i + m_start_item_i < (s32)ilist->getSize())
|
||||
return i + m_start_item_i;
|
||||
|
||||
return -1;
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue