/indra/newview/llwearableitemslist.cpp

https://bitbucket.org/lindenlab/viewer-beta/ · C++ · 1060 lines · 765 code · 182 blank · 113 comment · 145 complexity · 949dcde6c929dde8f4470431c6cb2e68 MD5 · raw file

  1. /**
  2. * @file llwearableitemslist.cpp
  3. * @brief A flat list of wearable items.
  4. *
  5. * $LicenseInfo:firstyear=2010&license=viewerlgpl$
  6. * Second Life Viewer Source Code
  7. * Copyright (C) 2010, Linden Research, Inc.
  8. *
  9. * This library is free software; you can redistribute it and/or
  10. * modify it under the terms of the GNU Lesser General Public
  11. * License as published by the Free Software Foundation;
  12. * version 2.1 of the License only.
  13. *
  14. * This library is distributed in the hope that it will be useful,
  15. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  17. * Lesser General Public License for more details.
  18. *
  19. * You should have received a copy of the GNU Lesser General Public
  20. * License along with this library; if not, write to the Free Software
  21. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  22. *
  23. * Linden Research, Inc., 945 Battery Street, San Francisco, CA 94111 USA
  24. * $/LicenseInfo$
  25. */
  26. #include "llviewerprecompiledheaders.h"
  27. #include "llwearableitemslist.h"
  28. #include "lliconctrl.h"
  29. #include "llmenugl.h" // for LLContextMenu
  30. #include "llagentwearables.h"
  31. #include "llappearancemgr.h"
  32. #include "llinventoryfunctions.h"
  33. #include "lltransutil.h"
  34. #include "llviewerattachmenu.h"
  35. #include "llvoavatarself.h"
  36. class LLFindOutfitItems : public LLInventoryCollectFunctor
  37. {
  38. public:
  39. LLFindOutfitItems() {}
  40. virtual ~LLFindOutfitItems() {}
  41. virtual bool operator()(LLInventoryCategory* cat,
  42. LLInventoryItem* item);
  43. };
  44. bool LLFindOutfitItems::operator()(LLInventoryCategory* cat,
  45. LLInventoryItem* item)
  46. {
  47. if(item)
  48. {
  49. if((item->getType() == LLAssetType::AT_CLOTHING)
  50. || (item->getType() == LLAssetType::AT_BODYPART)
  51. || (item->getType() == LLAssetType::AT_OBJECT))
  52. {
  53. return TRUE;
  54. }
  55. }
  56. return FALSE;
  57. }
  58. //////////////////////////////////////////////////////////////////////////
  59. //////////////////////////////////////////////////////////////////////////
  60. //////////////////////////////////////////////////////////////////////////
  61. void LLPanelWearableListItem::onMouseEnter(S32 x, S32 y, MASK mask)
  62. {
  63. LLPanelInventoryListItemBase::onMouseEnter(x, y, mask);
  64. setWidgetsVisible(true);
  65. reshapeWidgets();
  66. }
  67. void LLPanelWearableListItem::onMouseLeave(S32 x, S32 y, MASK mask)
  68. {
  69. LLPanelInventoryListItemBase::onMouseLeave(x, y, mask);
  70. setWidgetsVisible(false);
  71. reshapeWidgets();
  72. }
  73. LLPanelWearableListItem::LLPanelWearableListItem(LLViewerInventoryItem* item, const LLPanelWearableListItem::Params& params)
  74. : LLPanelInventoryListItemBase(item, params)
  75. {
  76. }
  77. //////////////////////////////////////////////////////////////////////////
  78. //////////////////////////////////////////////////////////////////////////
  79. //////////////////////////////////////////////////////////////////////////
  80. // static
  81. LLPanelWearableOutfitItem* LLPanelWearableOutfitItem::create(LLViewerInventoryItem* item,
  82. bool worn_indication_enabled)
  83. {
  84. LLPanelWearableOutfitItem* list_item = NULL;
  85. if (item)
  86. {
  87. const LLPanelInventoryListItemBase::Params& params = LLUICtrlFactory::getDefaultParams<LLPanelInventoryListItemBase>();
  88. list_item = new LLPanelWearableOutfitItem(item, worn_indication_enabled, params);
  89. list_item->initFromParams(params);
  90. list_item->postBuild();
  91. }
  92. return list_item;
  93. }
  94. LLPanelWearableOutfitItem::LLPanelWearableOutfitItem(LLViewerInventoryItem* item,
  95. bool worn_indication_enabled,
  96. const LLPanelWearableOutfitItem::Params& params)
  97. : LLPanelInventoryListItemBase(item, params)
  98. , mWornIndicationEnabled(worn_indication_enabled)
  99. {
  100. }
  101. // virtual
  102. void LLPanelWearableOutfitItem::updateItem(const std::string& name,
  103. EItemState item_state)
  104. {
  105. std::string search_label = name;
  106. // Updating item's worn status depending on whether it is linked in COF or not.
  107. // We don't use get_is_item_worn() here because this update is triggered by
  108. // an inventory observer upon link in COF beind added or removed so actual
  109. // worn status of a linked item may still remain unchanged.
  110. if (mWornIndicationEnabled && LLAppearanceMgr::instance().isLinkInCOF(mInventoryItemUUID))
  111. {
  112. search_label += LLTrans::getString("worn");
  113. item_state = IS_WORN;
  114. }
  115. LLPanelInventoryListItemBase::updateItem(search_label, item_state);
  116. }
  117. //////////////////////////////////////////////////////////////////////////
  118. //////////////////////////////////////////////////////////////////////////
  119. //////////////////////////////////////////////////////////////////////////
  120. static LLWidgetNameRegistry::StaticRegistrar sRegisterPanelClothingListItem(&typeid(LLPanelClothingListItem::Params), "clothing_list_item");
  121. LLPanelClothingListItem::Params::Params()
  122. : up_btn("up_btn"),
  123. down_btn("down_btn"),
  124. edit_btn("edit_btn"),
  125. lock_panel("lock_panel"),
  126. edit_panel("edit_panel"),
  127. lock_icon("lock_icon")
  128. {}
  129. // static
  130. LLPanelClothingListItem* LLPanelClothingListItem::create(LLViewerInventoryItem* item)
  131. {
  132. LLPanelClothingListItem* list_item = NULL;
  133. if(item)
  134. {
  135. const LLPanelClothingListItem::Params& params = LLUICtrlFactory::getDefaultParams<LLPanelClothingListItem>();
  136. list_item = new LLPanelClothingListItem(item, params);
  137. list_item->initFromParams(params);
  138. list_item->postBuild();
  139. }
  140. return list_item;
  141. }
  142. LLPanelClothingListItem::LLPanelClothingListItem(LLViewerInventoryItem* item, const LLPanelClothingListItem::Params& params)
  143. : LLPanelDeletableWearableListItem(item, params)
  144. {
  145. LLButton::Params button_params = params.up_btn;
  146. applyXUILayout(button_params, this);
  147. addChild(LLUICtrlFactory::create<LLButton>(button_params));
  148. button_params = params.down_btn;
  149. applyXUILayout(button_params, this);
  150. addChild(LLUICtrlFactory::create<LLButton>(button_params));
  151. LLPanel::Params panel_params = params.lock_panel;
  152. applyXUILayout(panel_params, this);
  153. LLPanel* lock_panelp = LLUICtrlFactory::create<LLPanel>(panel_params);
  154. addChild(lock_panelp);
  155. panel_params = params.edit_panel;
  156. applyXUILayout(panel_params, this);
  157. LLPanel* edit_panelp = LLUICtrlFactory::create<LLPanel>(panel_params);
  158. addChild(edit_panelp);
  159. if (lock_panelp)
  160. {
  161. LLIconCtrl::Params icon_params = params.lock_icon;
  162. applyXUILayout(icon_params, this);
  163. lock_panelp->addChild(LLUICtrlFactory::create<LLIconCtrl>(icon_params));
  164. }
  165. if (edit_panelp)
  166. {
  167. button_params = params.edit_btn;
  168. applyXUILayout(button_params, this);
  169. edit_panelp->addChild(LLUICtrlFactory::create<LLButton>(button_params));
  170. }
  171. setSeparatorVisible(false);
  172. }
  173. LLPanelClothingListItem::~LLPanelClothingListItem()
  174. {
  175. }
  176. BOOL LLPanelClothingListItem::postBuild()
  177. {
  178. LLPanelDeletableWearableListItem::postBuild();
  179. addWidgetToRightSide("btn_move_up");
  180. addWidgetToRightSide("btn_move_down");
  181. addWidgetToRightSide("btn_lock");
  182. addWidgetToRightSide("btn_edit_panel");
  183. setWidgetsVisible(false);
  184. reshapeWidgets();
  185. return TRUE;
  186. }
  187. //////////////////////////////////////////////////////////////////////////
  188. //////////////////////////////////////////////////////////////////////////
  189. //////////////////////////////////////////////////////////////////////////
  190. static LLWidgetNameRegistry::StaticRegistrar sRegisterPanelBodyPartsListItem(&typeid(LLPanelBodyPartsListItem::Params), "bodyparts_list_item");
  191. LLPanelBodyPartsListItem::Params::Params()
  192. : edit_btn("edit_btn"),
  193. edit_panel("edit_panel"),
  194. lock_panel("lock_panel"),
  195. lock_icon("lock_icon")
  196. {}
  197. // static
  198. LLPanelBodyPartsListItem* LLPanelBodyPartsListItem::create(LLViewerInventoryItem* item)
  199. {
  200. LLPanelBodyPartsListItem* list_item = NULL;
  201. if(item)
  202. {
  203. const Params& params = LLUICtrlFactory::getDefaultParams<LLPanelBodyPartsListItem>();
  204. list_item = new LLPanelBodyPartsListItem(item, params);
  205. list_item->initFromParams(params);
  206. list_item->postBuild();
  207. }
  208. return list_item;
  209. }
  210. LLPanelBodyPartsListItem::LLPanelBodyPartsListItem(LLViewerInventoryItem* item, const LLPanelBodyPartsListItem::Params& params)
  211. : LLPanelWearableListItem(item, params)
  212. {
  213. LLPanel::Params panel_params = params.edit_panel;
  214. applyXUILayout(panel_params, this);
  215. LLPanel* edit_panelp = LLUICtrlFactory::create<LLPanel>(panel_params);
  216. addChild(edit_panelp);
  217. panel_params = params.lock_panel;
  218. applyXUILayout(panel_params, this);
  219. LLPanel* lock_panelp = LLUICtrlFactory::create<LLPanel>(panel_params);
  220. addChild(lock_panelp);
  221. if (edit_panelp)
  222. {
  223. LLButton::Params btn_params = params.edit_btn;
  224. applyXUILayout(btn_params, this);
  225. edit_panelp->addChild(LLUICtrlFactory::create<LLButton>(btn_params));
  226. }
  227. if (lock_panelp)
  228. {
  229. LLIconCtrl::Params icon_params = params.lock_icon;
  230. applyXUILayout(icon_params, this);
  231. lock_panelp->addChild(LLUICtrlFactory::create<LLIconCtrl>(icon_params));
  232. }
  233. setSeparatorVisible(true);
  234. }
  235. LLPanelBodyPartsListItem::~LLPanelBodyPartsListItem()
  236. {
  237. }
  238. BOOL LLPanelBodyPartsListItem::postBuild()
  239. {
  240. LLPanelInventoryListItemBase::postBuild();
  241. addWidgetToRightSide("btn_lock");
  242. addWidgetToRightSide("btn_edit_panel");
  243. setWidgetsVisible(false);
  244. reshapeWidgets();
  245. return TRUE;
  246. }
  247. static LLWidgetNameRegistry::StaticRegistrar sRegisterPanelDeletableWearableListItem(&typeid(LLPanelDeletableWearableListItem::Params), "deletable_wearable_list_item");
  248. LLPanelDeletableWearableListItem::Params::Params()
  249. : delete_btn("delete_btn")
  250. {}
  251. // static
  252. LLPanelDeletableWearableListItem* LLPanelDeletableWearableListItem::create(LLViewerInventoryItem* item)
  253. {
  254. LLPanelDeletableWearableListItem* list_item = NULL;
  255. if(item)
  256. {
  257. const Params& params = LLUICtrlFactory::getDefaultParams<LLPanelDeletableWearableListItem>();
  258. list_item = new LLPanelDeletableWearableListItem(item, params);
  259. list_item->initFromParams(params);
  260. list_item->postBuild();
  261. }
  262. return list_item;
  263. }
  264. LLPanelDeletableWearableListItem::LLPanelDeletableWearableListItem(LLViewerInventoryItem* item, const LLPanelDeletableWearableListItem::Params& params)
  265. : LLPanelWearableListItem(item, params)
  266. {
  267. LLButton::Params button_params = params.delete_btn;
  268. applyXUILayout(button_params, this);
  269. addChild(LLUICtrlFactory::create<LLButton>(button_params));
  270. setSeparatorVisible(true);
  271. }
  272. BOOL LLPanelDeletableWearableListItem::postBuild()
  273. {
  274. LLPanelWearableListItem::postBuild();
  275. addWidgetToLeftSide("btn_delete");
  276. LLButton* delete_btn = getChild<LLButton>("btn_delete");
  277. // Reserve space for 'delete' button event if it is invisible.
  278. setLeftWidgetsWidth(delete_btn->getRect().mRight);
  279. setWidgetsVisible(false);
  280. reshapeWidgets();
  281. return TRUE;
  282. }
  283. // static
  284. LLPanelAttachmentListItem* LLPanelAttachmentListItem::create(LLViewerInventoryItem* item)
  285. {
  286. LLPanelAttachmentListItem* list_item = NULL;
  287. if(item)
  288. {
  289. const Params& params = LLUICtrlFactory::getDefaultParams<LLPanelDeletableWearableListItem>();
  290. list_item = new LLPanelAttachmentListItem(item, params);
  291. list_item->initFromParams(params);
  292. list_item->postBuild();
  293. }
  294. return list_item;
  295. }
  296. void LLPanelAttachmentListItem::updateItem(const std::string& name,
  297. EItemState item_state)
  298. {
  299. std::string title_joint = name;
  300. LLViewerInventoryItem* inv_item = getItem();
  301. if (inv_item && isAgentAvatarValid() && gAgentAvatarp->isWearingAttachment(inv_item->getLinkedUUID()))
  302. {
  303. std::string joint = LLTrans::getString(gAgentAvatarp->getAttachedPointName(inv_item->getLinkedUUID()));
  304. title_joint = title_joint + " (" + joint + ")";
  305. }
  306. LLPanelInventoryListItemBase::updateItem(title_joint, item_state);
  307. }
  308. //////////////////////////////////////////////////////////////////////////
  309. //////////////////////////////////////////////////////////////////////////
  310. //////////////////////////////////////////////////////////////////////////
  311. static LLWidgetNameRegistry::StaticRegistrar sRegisterPanelDummyClothingListItem(&typeid(LLPanelDummyClothingListItem::Params), "dummy_clothing_list_item");
  312. LLPanelDummyClothingListItem::Params::Params()
  313. : add_panel("add_panel"),
  314. add_btn("add_btn")
  315. {}
  316. LLPanelDummyClothingListItem* LLPanelDummyClothingListItem::create(LLWearableType::EType w_type)
  317. {
  318. const Params& params = LLUICtrlFactory::getDefaultParams<LLPanelDummyClothingListItem>();
  319. LLPanelDummyClothingListItem* list_item = new LLPanelDummyClothingListItem(w_type, params);
  320. list_item->initFromParams(params);
  321. list_item->postBuild();
  322. return list_item;
  323. }
  324. BOOL LLPanelDummyClothingListItem::postBuild()
  325. {
  326. addWidgetToRightSide("btn_add_panel");
  327. setIconImage(LLInventoryIcon::getIcon(LLAssetType::AT_CLOTHING, LLInventoryType::IT_NONE, mWearableType, FALSE));
  328. updateItem(wearableTypeToString(mWearableType));
  329. // Make it look loke clothing item - reserve space for 'delete' button
  330. setLeftWidgetsWidth(getChildView("item_icon")->getRect().mLeft);
  331. setWidgetsVisible(false);
  332. reshapeWidgets();
  333. return TRUE;
  334. }
  335. LLWearableType::EType LLPanelDummyClothingListItem::getWearableType() const
  336. {
  337. return mWearableType;
  338. }
  339. LLPanelDummyClothingListItem::LLPanelDummyClothingListItem(LLWearableType::EType w_type, const LLPanelDummyClothingListItem::Params& params)
  340. : LLPanelWearableListItem(NULL, params),
  341. mWearableType(w_type)
  342. {
  343. LLPanel::Params panel_params(params.add_panel);
  344. applyXUILayout(panel_params, this);
  345. LLPanel* add_panelp = LLUICtrlFactory::create<LLPanel>(panel_params);
  346. addChild(add_panelp);
  347. if (add_panelp)
  348. {
  349. LLButton::Params button_params(params.add_btn);
  350. applyXUILayout(button_params, this);
  351. add_panelp->addChild(LLUICtrlFactory::create<LLButton>(button_params));
  352. }
  353. setSeparatorVisible(true);
  354. }
  355. typedef std::map<LLWearableType::EType, std::string> clothing_to_string_map_t;
  356. clothing_to_string_map_t init_clothing_string_map()
  357. {
  358. clothing_to_string_map_t w_map;
  359. w_map.insert(std::make_pair(LLWearableType::WT_SHIRT, "shirt_not_worn"));
  360. w_map.insert(std::make_pair(LLWearableType::WT_PANTS, "pants_not_worn"));
  361. w_map.insert(std::make_pair(LLWearableType::WT_SHOES, "shoes_not_worn"));
  362. w_map.insert(std::make_pair(LLWearableType::WT_SOCKS, "socks_not_worn"));
  363. w_map.insert(std::make_pair(LLWearableType::WT_JACKET, "jacket_not_worn"));
  364. w_map.insert(std::make_pair(LLWearableType::WT_GLOVES, "gloves_not_worn"));
  365. w_map.insert(std::make_pair(LLWearableType::WT_UNDERSHIRT, "undershirt_not_worn"));
  366. w_map.insert(std::make_pair(LLWearableType::WT_UNDERPANTS, "underpants_not_worn"));
  367. w_map.insert(std::make_pair(LLWearableType::WT_SKIRT, "skirt_not_worn"));
  368. w_map.insert(std::make_pair(LLWearableType::WT_ALPHA, "alpha_not_worn"));
  369. w_map.insert(std::make_pair(LLWearableType::WT_TATTOO, "tattoo_not_worn"));
  370. w_map.insert(std::make_pair(LLWearableType::WT_PHYSICS, "physics_not_worn"));
  371. return w_map;
  372. }
  373. std::string LLPanelDummyClothingListItem::wearableTypeToString(LLWearableType::EType w_type)
  374. {
  375. static const clothing_to_string_map_t w_map = init_clothing_string_map();
  376. static const std::string invalid_str = LLTrans::getString("invalid_not_worn");
  377. std::string type_str = invalid_str;
  378. clothing_to_string_map_t::const_iterator it = w_map.find(w_type);
  379. if(w_map.end() != it)
  380. {
  381. type_str = LLTrans::getString(it->second);
  382. }
  383. return type_str;
  384. }
  385. //////////////////////////////////////////////////////////////////////////
  386. //////////////////////////////////////////////////////////////////////////
  387. //////////////////////////////////////////////////////////////////////////
  388. LLWearableItemTypeNameComparator::LLWearableTypeOrder::LLWearableTypeOrder(LLWearableItemTypeNameComparator::ETypeListOrder order_priority, bool sort_asset_by_name, bool sort_wearable_by_name):
  389. mOrderPriority(order_priority),
  390. mSortAssetTypeByName(sort_asset_by_name),
  391. mSortWearableTypeByName(sort_wearable_by_name)
  392. {
  393. }
  394. LLWearableItemTypeNameComparator::LLWearableItemTypeNameComparator()
  395. {
  396. // By default the sort order conforms the order by spec of MY OUTFITS items list:
  397. // 1. CLOTHING - sorted by name
  398. // 2. OBJECT - sorted by type
  399. // 3. BODYPART - sorted by name
  400. mWearableOrder[LLAssetType::AT_CLOTHING] = LLWearableTypeOrder(ORDER_RANK_1, false, false);
  401. mWearableOrder[LLAssetType::AT_OBJECT] = LLWearableTypeOrder(ORDER_RANK_2, true, true);
  402. mWearableOrder[LLAssetType::AT_BODYPART] = LLWearableTypeOrder(ORDER_RANK_3, false, true);
  403. }
  404. void LLWearableItemTypeNameComparator::setOrder(LLAssetType::EType items_of_type, LLWearableItemTypeNameComparator::ETypeListOrder order_priority, bool sort_asset_items_by_name, bool sort_wearable_items_by_name)
  405. {
  406. mWearableOrder[items_of_type] = LLWearableTypeOrder(order_priority, sort_asset_items_by_name, sort_wearable_items_by_name);
  407. }
  408. /*virtual*/
  409. bool LLWearableItemNameComparator::doCompare(const LLPanelInventoryListItemBase* wearable_item1, const LLPanelInventoryListItemBase* wearable_item2) const
  410. {
  411. std::string name1 = wearable_item1->getItemName();
  412. std::string name2 = wearable_item2->getItemName();
  413. LLStringUtil::toUpper(name1);
  414. LLStringUtil::toUpper(name2);
  415. return name1 < name2;
  416. }
  417. /*virtual*/
  418. bool LLWearableItemTypeNameComparator::doCompare(const LLPanelInventoryListItemBase* wearable_item1, const LLPanelInventoryListItemBase* wearable_item2) const
  419. {
  420. const LLAssetType::EType item_type1 = wearable_item1->getType();
  421. const LLAssetType::EType item_type2 = wearable_item2->getType();
  422. LLWearableItemTypeNameComparator::ETypeListOrder item_type_order1 = getTypeListOrder(item_type1);
  423. LLWearableItemTypeNameComparator::ETypeListOrder item_type_order2 = getTypeListOrder(item_type2);
  424. if (item_type_order1 != item_type_order2)
  425. {
  426. // If items are of different asset types we can compare them
  427. // by types order in the list.
  428. return item_type_order1 < item_type_order2;
  429. }
  430. if (sortAssetTypeByName(item_type1))
  431. {
  432. // If both items are of the same asset type except AT_CLOTHING and AT_BODYPART
  433. // we can compare them by name.
  434. return LLWearableItemNameComparator::doCompare(wearable_item1, wearable_item2);
  435. }
  436. const LLWearableType::EType item_wearable_type1 = wearable_item1->getWearableType();
  437. const LLWearableType::EType item_wearable_type2 = wearable_item2->getWearableType();
  438. if (item_wearable_type1 != item_wearable_type2)
  439. // If items are of different LLWearableType::EType types they are compared
  440. // by LLWearableType::EType. types order determined in LLWearableType::EType.
  441. {
  442. // If items are of different LLWearableType::EType types they are compared
  443. // by LLWearableType::EType. types order determined in LLWearableType::EType.
  444. return item_wearable_type1 < item_wearable_type2;
  445. }
  446. else
  447. {
  448. // If both items are of the same clothing type they are compared
  449. // by description and place in reverse order (i.e. outer layer item
  450. // on top) OR by name
  451. if(sortWearableTypeByName(item_type1))
  452. {
  453. return LLWearableItemNameComparator::doCompare(wearable_item1, wearable_item2);
  454. }
  455. return wearable_item1->getDescription() > wearable_item2->getDescription();
  456. }
  457. }
  458. LLWearableItemTypeNameComparator::ETypeListOrder LLWearableItemTypeNameComparator::getTypeListOrder(LLAssetType::EType item_type) const
  459. {
  460. wearable_type_order_map_t::const_iterator const_it = mWearableOrder.find(item_type);
  461. if(const_it == mWearableOrder.end())
  462. {
  463. llwarns<<"Absent information about order rang of items of "<<LLAssetType::getDesc(item_type)<<" type"<<llendl;
  464. return ORDER_RANK_UNKNOWN;
  465. }
  466. return const_it->second.mOrderPriority;
  467. }
  468. bool LLWearableItemTypeNameComparator::sortAssetTypeByName(LLAssetType::EType item_type) const
  469. {
  470. wearable_type_order_map_t::const_iterator const_it = mWearableOrder.find(item_type);
  471. if(const_it == mWearableOrder.end())
  472. {
  473. llwarns<<"Absent information about sorting items of "<<LLAssetType::getDesc(item_type)<<" type"<<llendl;
  474. return true;
  475. }
  476. return const_it->second.mSortAssetTypeByName;
  477. }
  478. bool LLWearableItemTypeNameComparator::sortWearableTypeByName(LLAssetType::EType item_type) const
  479. {
  480. wearable_type_order_map_t::const_iterator const_it = mWearableOrder.find(item_type);
  481. if(const_it == mWearableOrder.end())
  482. {
  483. llwarns<<"Absent information about sorting items of "<<LLAssetType::getDesc(item_type)<<" type"<<llendl;
  484. return true;
  485. }
  486. return const_it->second.mSortWearableTypeByName;
  487. }
  488. /*virtual*/
  489. bool LLWearableItemCreationDateComparator::doCompare(const LLPanelInventoryListItemBase* item1, const LLPanelInventoryListItemBase* item2) const
  490. {
  491. time_t date1 = item1->getCreationDate();
  492. time_t date2 = item2->getCreationDate();
  493. if (date1 == date2)
  494. {
  495. return LLWearableItemNameComparator::doCompare(item1, item2);
  496. }
  497. return date1 > date2;
  498. }
  499. //////////////////////////////////////////////////////////////////////////
  500. //////////////////////////////////////////////////////////////////////////
  501. //////////////////////////////////////////////////////////////////////////
  502. static LLWearableItemTypeNameComparator WEARABLE_TYPE_NAME_COMPARATOR;
  503. static const LLWearableItemTypeNameComparator WEARABLE_TYPE_LAYER_COMPARATOR;
  504. static const LLWearableItemNameComparator WEARABLE_NAME_COMPARATOR;
  505. static const LLWearableItemCreationDateComparator WEARABLE_CREATION_DATE_COMPARATOR;
  506. static const LLDefaultChildRegistry::Register<LLWearableItemsList> r("wearable_items_list");
  507. LLWearableItemsList::Params::Params()
  508. : standalone("standalone", true)
  509. , worn_indication_enabled("worn_indication_enabled", true)
  510. {}
  511. LLWearableItemsList::LLWearableItemsList(const LLWearableItemsList::Params& p)
  512. : LLInventoryItemsList(p)
  513. {
  514. setSortOrder(E_SORT_BY_TYPE_LAYER, false);
  515. mIsStandalone = p.standalone;
  516. if (mIsStandalone)
  517. {
  518. // Use built-in context menu.
  519. setRightMouseDownCallback(boost::bind(&LLWearableItemsList::onRightClick, this, _2, _3));
  520. }
  521. mWornIndicationEnabled = p.worn_indication_enabled;
  522. setNoItemsCommentText(LLTrans::getString("LoadingData"));
  523. }
  524. // virtual
  525. LLWearableItemsList::~LLWearableItemsList()
  526. {}
  527. // virtual
  528. void LLWearableItemsList::addNewItem(LLViewerInventoryItem* item, bool rearrange /*= true*/)
  529. {
  530. if (!item)
  531. {
  532. llwarns << "No inventory item. Couldn't create flat list item." << llendl;
  533. llassert(item != NULL);
  534. }
  535. LLPanelWearableOutfitItem *list_item = LLPanelWearableOutfitItem::create(item, mWornIndicationEnabled);
  536. if (!list_item)
  537. return;
  538. bool is_item_added = addItem(list_item, item->getUUID(), ADD_BOTTOM, rearrange);
  539. if (!is_item_added)
  540. {
  541. llwarns << "Couldn't add flat list item." << llendl;
  542. llassert(is_item_added);
  543. }
  544. }
  545. void LLWearableItemsList::updateList(const LLUUID& category_id)
  546. {
  547. LLInventoryModel::cat_array_t cat_array;
  548. LLInventoryModel::item_array_t item_array;
  549. LLFindOutfitItems collector = LLFindOutfitItems();
  550. // collectDescendentsIf takes non-const reference:
  551. gInventory.collectDescendentsIf(
  552. category_id,
  553. cat_array,
  554. item_array,
  555. LLInventoryModel::EXCLUDE_TRASH,
  556. collector);
  557. if(item_array.empty() && gInventory.isCategoryComplete(category_id))
  558. {
  559. setNoItemsCommentText(LLTrans::getString("EmptyOutfitText"));
  560. }
  561. refreshList(item_array);
  562. }
  563. void LLWearableItemsList::updateChangedItems(const uuid_vec_t& changed_items_uuids)
  564. {
  565. // nothing to update
  566. if (changed_items_uuids.empty()) return;
  567. typedef std::vector<LLPanel*> item_panel_list_t;
  568. item_panel_list_t items;
  569. getItems(items);
  570. for (item_panel_list_t::iterator items_iter = items.begin();
  571. items_iter != items.end();
  572. ++items_iter)
  573. {
  574. LLPanelInventoryListItemBase* item = dynamic_cast<LLPanelInventoryListItemBase*>(*items_iter);
  575. if (!item) continue;
  576. LLViewerInventoryItem* inv_item = item->getItem();
  577. if (!inv_item) continue;
  578. LLUUID linked_uuid = inv_item->getLinkedUUID();
  579. for (uuid_vec_t::const_iterator iter = changed_items_uuids.begin();
  580. iter != changed_items_uuids.end();
  581. ++iter)
  582. {
  583. if (linked_uuid == *iter)
  584. {
  585. item->setNeedsRefresh(true);
  586. break;
  587. }
  588. }
  589. }
  590. }
  591. void LLWearableItemsList::onRightClick(S32 x, S32 y)
  592. {
  593. uuid_vec_t selected_uuids;
  594. getSelectedUUIDs(selected_uuids);
  595. if (selected_uuids.empty())
  596. {
  597. return;
  598. }
  599. ContextMenu::instance().show(this, selected_uuids, x, y);
  600. }
  601. void LLWearableItemsList::setSortOrder(ESortOrder sort_order, bool sort_now)
  602. {
  603. switch (sort_order)
  604. {
  605. case E_SORT_BY_MOST_RECENT:
  606. setComparator(&WEARABLE_CREATION_DATE_COMPARATOR);
  607. break;
  608. case E_SORT_BY_NAME:
  609. setComparator(&WEARABLE_NAME_COMPARATOR);
  610. break;
  611. case E_SORT_BY_TYPE_LAYER:
  612. setComparator(&WEARABLE_TYPE_LAYER_COMPARATOR);
  613. break;
  614. case E_SORT_BY_TYPE_NAME:
  615. {
  616. WEARABLE_TYPE_NAME_COMPARATOR.setOrder(LLAssetType::AT_CLOTHING, LLWearableItemTypeNameComparator::ORDER_RANK_1, false, true);
  617. setComparator(&WEARABLE_TYPE_NAME_COMPARATOR);
  618. break;
  619. }
  620. // No "default:" to raise compiler warning
  621. // if we're not handling something
  622. }
  623. mSortOrder = sort_order;
  624. if (sort_now)
  625. {
  626. sort();
  627. }
  628. }
  629. //////////////////////////////////////////////////////////////////////////
  630. /// ContextMenu
  631. //////////////////////////////////////////////////////////////////////////
  632. LLWearableItemsList::ContextMenu::ContextMenu()
  633. : mParent(NULL)
  634. {
  635. }
  636. void LLWearableItemsList::ContextMenu::show(LLView* spawning_view, const uuid_vec_t& uuids, S32 x, S32 y)
  637. {
  638. mParent = dynamic_cast<LLWearableItemsList*>(spawning_view);
  639. LLListContextMenu::show(spawning_view, uuids, x, y);
  640. mParent = NULL; // to avoid dereferencing an invalid pointer
  641. }
  642. // virtual
  643. LLContextMenu* LLWearableItemsList::ContextMenu::createMenu()
  644. {
  645. LLUICtrl::CommitCallbackRegistry::ScopedRegistrar registrar;
  646. const uuid_vec_t& ids = mUUIDs; // selected items IDs
  647. LLUUID selected_id = ids.front(); // ID of the first selected item
  648. functor_t take_off = boost::bind(&LLAppearanceMgr::removeItemFromAvatar, LLAppearanceMgr::getInstance(), _1);
  649. // Register handlers common for all wearable types.
  650. registrar.add("Wearable.Wear", boost::bind(wear_multiple, ids, true));
  651. registrar.add("Wearable.Add", boost::bind(wear_multiple, ids, false));
  652. registrar.add("Wearable.Edit", boost::bind(handleMultiple, LLAgentWearables::editWearable, ids));
  653. registrar.add("Wearable.CreateNew", boost::bind(createNewWearable, selected_id));
  654. registrar.add("Wearable.ShowOriginal", boost::bind(show_item_original, selected_id));
  655. registrar.add("Wearable.TakeOffDetach", boost::bind(handleMultiple, take_off, ids));
  656. // Register handlers for clothing.
  657. registrar.add("Clothing.TakeOff", boost::bind(handleMultiple, take_off, ids));
  658. // Register handlers for body parts.
  659. // Register handlers for attachments.
  660. registrar.add("Attachment.Detach", boost::bind(handleMultiple, take_off, ids));
  661. registrar.add("Attachment.Profile", boost::bind(show_item_profile, selected_id));
  662. registrar.add("Object.Attach", boost::bind(LLViewerAttachMenu::attachObjects, ids, _2));
  663. // Create the menu.
  664. LLContextMenu* menu = createFromFile("menu_wearable_list_item.xml");
  665. // Determine which items should be visible/enabled.
  666. updateItemsVisibility(menu);
  667. // Update labels for the items requiring that.
  668. updateItemsLabels(menu);
  669. return menu;
  670. }
  671. void LLWearableItemsList::ContextMenu::updateItemsVisibility(LLContextMenu* menu)
  672. {
  673. if (!menu)
  674. {
  675. llwarns << "Invalid menu" << llendl;
  676. return;
  677. }
  678. const uuid_vec_t& ids = mUUIDs; // selected items IDs
  679. U32 mask = 0; // mask of selected items' types
  680. U32 n_items = ids.size(); // number of selected items
  681. U32 n_worn = 0; // number of worn items among the selected ones
  682. U32 n_already_worn = 0; // number of items worn of same type as selected items
  683. U32 n_links = 0; // number of links among the selected items
  684. U32 n_editable = 0; // number of editable items among the selected ones
  685. bool can_be_worn = true;
  686. for (uuid_vec_t::const_iterator it = ids.begin(); it != ids.end(); ++it)
  687. {
  688. LLUUID id = *it;
  689. LLViewerInventoryItem* item = gInventory.getItem(id);
  690. if (!item)
  691. {
  692. llwarns << "Invalid item" << llendl;
  693. // *NOTE: the logic below may not work in this case
  694. continue;
  695. }
  696. updateMask(mask, item->getType());
  697. const LLWearableType::EType wearable_type = item->getWearableType();
  698. const bool is_link = item->getIsLinkType();
  699. const bool is_worn = get_is_item_worn(id);
  700. const bool is_editable = gAgentWearables.isWearableModifiable(id);
  701. const bool is_already_worn = gAgentWearables.selfHasWearable(wearable_type);
  702. if (is_worn)
  703. {
  704. ++n_worn;
  705. }
  706. if (is_editable)
  707. {
  708. ++n_editable;
  709. }
  710. if (is_link)
  711. {
  712. ++n_links;
  713. }
  714. if (is_already_worn)
  715. {
  716. ++n_already_worn;
  717. }
  718. if (can_be_worn)
  719. {
  720. can_be_worn = get_can_item_be_worn(item->getLinkedUUID());
  721. }
  722. } // for
  723. bool standalone = mParent ? mParent->isStandalone() : false;
  724. bool wear_add_visible = mask & (MASK_CLOTHING|MASK_ATTACHMENT) && n_worn == 0 && can_be_worn && (n_already_worn != 0 || mask & MASK_ATTACHMENT);
  725. // *TODO: eliminate multiple traversals over the menu items
  726. setMenuItemVisible(menu, "wear_wear", n_already_worn == 0 && n_worn == 0 && can_be_worn);
  727. setMenuItemEnabled(menu, "wear_wear", n_already_worn == 0 && n_worn == 0);
  728. setMenuItemVisible(menu, "wear_add", wear_add_visible);
  729. setMenuItemEnabled(menu, "wear_add", canAddWearables(ids));
  730. setMenuItemVisible(menu, "wear_replace", n_worn == 0 && n_already_worn != 0 && can_be_worn);
  731. //visible only when one item selected and this item is worn
  732. setMenuItemVisible(menu, "edit", !standalone && mask & (MASK_CLOTHING|MASK_BODYPART) && n_worn == n_items && n_worn == 1);
  733. setMenuItemEnabled(menu, "edit", n_editable == 1 && n_worn == 1 && n_items == 1);
  734. setMenuItemVisible(menu, "create_new", mask & (MASK_CLOTHING|MASK_BODYPART) && n_items == 1);
  735. setMenuItemEnabled(menu, "create_new", canAddWearables(ids));
  736. setMenuItemVisible(menu, "show_original", !standalone);
  737. setMenuItemEnabled(menu, "show_original", n_items == 1 && n_links == n_items);
  738. setMenuItemVisible(menu, "take_off", mask == MASK_CLOTHING && n_worn == n_items);
  739. setMenuItemVisible(menu, "detach", mask == MASK_ATTACHMENT && n_worn == n_items);
  740. setMenuItemVisible(menu, "take_off_or_detach", mask == (MASK_ATTACHMENT|MASK_CLOTHING));
  741. setMenuItemEnabled(menu, "take_off_or_detach", n_worn == n_items);
  742. setMenuItemVisible(menu, "object_profile", !standalone);
  743. setMenuItemEnabled(menu, "object_profile", n_items == 1);
  744. setMenuItemVisible(menu, "--no options--", FALSE);
  745. setMenuItemEnabled(menu, "--no options--", FALSE);
  746. // Populate or hide the "Attach to..." / "Attach to HUD..." submenus.
  747. if (mask == MASK_ATTACHMENT && n_worn == 0)
  748. {
  749. LLViewerAttachMenu::populateMenus("wearable_attach_to", "wearable_attach_to_hud");
  750. }
  751. else
  752. {
  753. setMenuItemVisible(menu, "wearable_attach_to", false);
  754. setMenuItemVisible(menu, "wearable_attach_to_hud", false);
  755. }
  756. if (mask & MASK_UNKNOWN)
  757. {
  758. llwarns << "Non-wearable items passed." << llendl;
  759. }
  760. U32 num_visible_items = 0;
  761. for (U32 menu_item_index = 0; menu_item_index < menu->getItemCount(); ++menu_item_index)
  762. {
  763. const LLMenuItemGL* menu_item = menu->getItem(menu_item_index);
  764. if (menu_item && menu_item->getVisible())
  765. {
  766. num_visible_items++;
  767. }
  768. }
  769. if (num_visible_items == 0)
  770. {
  771. setMenuItemVisible(menu, "--no options--", TRUE);
  772. }
  773. }
  774. void LLWearableItemsList::ContextMenu::updateItemsLabels(LLContextMenu* menu)
  775. {
  776. llassert(menu);
  777. if (!menu) return;
  778. // Set proper label for the "Create new <WEARABLE_TYPE>" menu item.
  779. LLViewerInventoryItem* item = gInventory.getLinkedItem(mUUIDs.back());
  780. if (!item || !item->isWearableType()) return;
  781. LLWearableType::EType w_type = item->getWearableType();
  782. std::string new_label = LLTrans::getString("create_new_" + LLWearableType::getTypeName(w_type));
  783. LLMenuItemGL* menu_item = menu->getChild<LLMenuItemGL>("create_new");
  784. menu_item->setLabel(new_label);
  785. }
  786. // We need this method to convert non-zero BOOL values to exactly 1 (TRUE).
  787. // Otherwise code relying on a BOOL value being TRUE may fail
  788. // (I experienced a weird assert in LLView::drawChildren() because of that.
  789. // static
  790. void LLWearableItemsList::ContextMenu::setMenuItemVisible(LLContextMenu* menu, const std::string& name, bool val)
  791. {
  792. menu->setItemVisible(name, val);
  793. }
  794. // static
  795. void LLWearableItemsList::ContextMenu::setMenuItemEnabled(LLContextMenu* menu, const std::string& name, bool val)
  796. {
  797. menu->setItemEnabled(name, val);
  798. }
  799. // static
  800. void LLWearableItemsList::ContextMenu::updateMask(U32& mask, LLAssetType::EType at)
  801. {
  802. if (at == LLAssetType::AT_CLOTHING)
  803. {
  804. mask |= MASK_CLOTHING;
  805. }
  806. else if (at == LLAssetType::AT_BODYPART)
  807. {
  808. mask |= MASK_BODYPART;
  809. }
  810. else if (at == LLAssetType::AT_OBJECT)
  811. {
  812. mask |= MASK_ATTACHMENT;
  813. }
  814. else
  815. {
  816. mask |= MASK_UNKNOWN;
  817. }
  818. }
  819. // static
  820. void LLWearableItemsList::ContextMenu::createNewWearable(const LLUUID& item_id)
  821. {
  822. LLViewerInventoryItem* item = gInventory.getLinkedItem(item_id);
  823. if (!item || !item->isWearableType()) return;
  824. LLAgentWearables::createWearable(item->getWearableType(), true);
  825. }
  826. // Returns true if all the given objects and clothes can be added.
  827. // static
  828. bool LLWearableItemsList::ContextMenu::canAddWearables(const uuid_vec_t& item_ids)
  829. {
  830. // TODO: investigate wearables may not be loaded at this point EXT-8231
  831. U32 n_objects = 0;
  832. boost::unordered_map<LLWearableType::EType, U32> clothes_by_type;
  833. // Count given clothes (by wearable type) and objects.
  834. for (uuid_vec_t::const_iterator it = item_ids.begin(); it != item_ids.end(); ++it)
  835. {
  836. LLViewerInventoryItem* item = gInventory.getItem(*it);
  837. if (!item)
  838. {
  839. return false;
  840. }
  841. if (item->getType() == LLAssetType::AT_OBJECT)
  842. {
  843. ++n_objects;
  844. }
  845. else if (item->getType() == LLAssetType::AT_CLOTHING)
  846. {
  847. ++clothes_by_type[item->getWearableType()];
  848. }
  849. else
  850. {
  851. llwarns << "Unexpected wearable type" << llendl;
  852. return false;
  853. }
  854. }
  855. // Check whether we can add all the objects.
  856. if (!isAgentAvatarValid() || !gAgentAvatarp->canAttachMoreObjects(n_objects))
  857. {
  858. return false;
  859. }
  860. // Check whether we can add all the clothes.
  861. boost::unordered_map<LLWearableType::EType, U32>::const_iterator m_it;
  862. for (m_it = clothes_by_type.begin(); m_it != clothes_by_type.end(); ++m_it)
  863. {
  864. LLWearableType::EType w_type = m_it->first;
  865. U32 n_clothes = m_it->second;
  866. U32 wearable_count = gAgentWearables.getWearableCount(w_type);
  867. if ((wearable_count > 0) && !LLWearableType::getAllowMultiwear(w_type))
  868. {
  869. return false;
  870. }
  871. if ((wearable_count + n_clothes) > LLAgentWearables::MAX_CLOTHING_PER_TYPE)
  872. {
  873. return false;
  874. }
  875. }
  876. return true;
  877. }
  878. // EOF