From fef372f8f730d614759fbfaaddb4b425b4a3f292 Mon Sep 17 00:00:00 2001 From: andrei Date: Fri, 10 Dec 2021 10:31:40 +0200 Subject: [PATCH 1/5] inventory_selector: add contents right after item --- src/inventory_ui.cpp | 22 +++++----------------- 1 file changed, 5 insertions(+), 17 deletions(-) diff --git a/src/inventory_ui.cpp b/src/inventory_ui.cpp index 45824c7d97f2f..ad4bb969caec2 100644 --- a/src/inventory_ui.cpp +++ b/src/inventory_ui.cpp @@ -1466,6 +1466,11 @@ void inventory_selector::add_items( inventory_column &target_column, } add_entry( target_column, std::move( locations ), nat_category ); + for( item *it_elem : elem ) { + item_location parent = locator( it_elem ); + add_contained_items( parent, target_column, custom_category, get_topmost_parent( nullptr, parent, + preset ) ); + } } } @@ -1525,11 +1530,6 @@ void inventory_selector::add_character_items( Character &character ) return item_location( character, it ); }, restack_items( ( *elem ).begin(), ( *elem ).end(), preset.get_checking_components() ), &item_category_ITEMS_WORN.obj() ); - for( item &it_elem : *elem ) { - item_location parent( character, &it_elem ); - add_contained_items( parent, own_inv_column, &item_category_ITEMS_WORN.obj(), - get_topmost_parent( nullptr, parent, preset ) ); - } } // this is a little trick; we want the default behavior for contained items to be in own_inv_column // and this function iterates over all the entries after we added them to the inventory selector @@ -1549,12 +1549,6 @@ void inventory_selector::add_map_items( const tripoint &target ) add_items( map_column, [ &target ]( item * it ) { return item_location( map_cursor( target ), it ); }, restack_items( items.begin(), items.end(), preset.get_checking_components() ), custom_cat ); - - for( item &it_elem : items ) { - item_location parent( map_cursor( target ), &it_elem ); - add_contained_items( parent, map_column, custom_cat, get_topmost_parent( nullptr, parent, - preset ) ); - } } } @@ -1577,12 +1571,6 @@ void inventory_selector::add_vehicle_items( const tripoint &target ) add_items( map_column, [ veh, part ]( item * it ) { return item_location( vehicle_cursor( *veh, part ), it ); }, restack_items( items.begin(), items.end(), check_components ), custom_cat ); - - for( item &it_elem : items ) { - item_location parent( vehicle_cursor( *veh, part ), &it_elem ); - add_contained_items( parent, map_column, custom_cat, get_topmost_parent( nullptr, parent, - preset ) ); - } } void inventory_selector::add_nearby_items( int radius ) From 5a8660dddedc59598291034ac49a85dca3b18ac5 Mon Sep 17 00:00:00 2001 From: andrei Date: Mon, 22 Nov 2021 09:21:54 +0200 Subject: [PATCH 2/5] inventory_column: improve sorting properly sort entries alphabetically sort entries below their parents --- src/inventory_ui.cpp | 105 +++++++++++++++++++++++++++++++++++-------- src/inventory_ui.h | 7 ++- 2 files changed, 92 insertions(+), 20 deletions(-) diff --git a/src/inventory_ui.cpp b/src/inventory_ui.cpp index ad4bb969caec2..00c2facf963da 100644 --- a/src/inventory_ui.cpp +++ b/src/inventory_ui.cpp @@ -28,6 +28,7 @@ #include "point.h" #include "ret_val.h" #include "sdltiles.h" +#include "localized_comparator.h" #include "string_formatter.h" #include "string_input_popup.h" #include "trade_ui.h" @@ -69,6 +70,43 @@ item *get_topmost_parent( item *topmost, item_location loc, return preset.is_shown( loc ) ? topmost != nullptr ? topmost : loc.get_item() : nullptr; } +bool is_child( inventory_entry const &parent, inventory_entry const &child ) +{ + return std::any_of( parent.locations.begin(), parent.locations.end(), + [&child]( item_location const & loc ) { + return loc.eventually_contains( child.any_item() ); + } ); +} + +using parent_path_t = std::vector; +parent_path_t path_to_top( inventory_entry const &e ) +{ + item_location it = e.any_item(); + parent_path_t path{ it }; + while( it.has_parent() ) { + it = it.parent_item(); + path.emplace_back( it ); + } + return path; +} + +using common_depth_t = std::pair; +common_depth_t common_depth( inventory_entry const &lhs, inventory_entry const &rhs ) +{ + parent_path_t const path_lhs = path_to_top( lhs ); + parent_path_t const path_rhs = path_to_top( rhs ); + parent_path_t::size_type const common_depth = std::min( path_lhs.size(), path_rhs.size() ); + item_location p_lhs = path_lhs[path_lhs.size() - common_depth]; + item_location p_rhs = path_rhs[path_rhs.size() - common_depth]; + // parent of both entries must be lowest common ancestor + while( p_lhs.has_parent() and p_lhs.parent_item() != p_rhs.parent_item() ) { + p_lhs = p_lhs.parent_item(); + p_rhs = p_rhs.parent_item(); + } + + return std::make_pair( p_lhs, p_rhs ); +} + } // namespace /** The maximum distance from the screen edge, to snap a window to it */ @@ -236,7 +274,7 @@ nc_color inventory_entry::get_invlet_color() const void inventory_entry::update_cache() { - cached_name = any_item()->tname( 1 ); + cached_name = any_item()->tname( 1, false ); } const item_category *inventory_entry::get_category_ptr() const @@ -288,7 +326,13 @@ inventory_selector_preset::inventory_selector_preset() bool inventory_selector_preset::sort_compare( const inventory_entry &lhs, const inventory_entry &rhs ) const { - return lhs.cached_name.compare( rhs.cached_name ) < 0; // Simple alphabetic order + if( lhs.cached_name == rhs.cached_name ) { + std::string const ltname = lhs.any_item()->tname(); + std::string const rtname = rhs.any_item()->tname(); + return localized_compare( ltname, rtname ); + } + + return localized_compare( lhs.cached_name, rhs.cached_name ); } bool inventory_selector_preset::cat_sort_compare( const inventory_entry &lhs, @@ -931,6 +975,30 @@ void inventory_column::move_entries_to( inventory_column &dest ) clear(); } +bool inventory_column::sort_compare( inventory_entry const &lhs, inventory_entry const &rhs ) +{ + if( lhs.is_selectable() != rhs.is_selectable() ) { + return lhs.is_selectable(); // Disabled items always go last + } + Character &player_character = get_player_character(); + // Place favorite items and items with an assigned inventory letter first, + // since the player cared enough to assign them + const bool left_has_invlet = + player_character.inv->assigned_invlet.count( lhs.any_item()->invlet ) != 0; + const bool right_has_invlet = + player_character.inv->assigned_invlet.count( rhs.any_item()->invlet ) != 0; + if( left_has_invlet != right_has_invlet ) { + return left_has_invlet; + } + const bool left_fav = lhs.any_item()->is_favorite; + const bool right_fav = rhs.any_item()->is_favorite; + if( left_fav != right_fav ) { + return left_fav; + } + + return preset.sort_compare( lhs, rhs ); +} + void inventory_column::prepare_paging( const std::string &filter ) { if( paging_is_valid ) { @@ -976,23 +1044,24 @@ void inventory_column::prepare_paging( const std::string &filter ) } if( ordered_categories.count( from->get_category_ptr()->get_id().c_str() ) == 0 ) { std::stable_sort( from, to, [ this ]( const inventory_entry & lhs, const inventory_entry & rhs ) { - if( lhs.is_selectable() != rhs.is_selectable() ) { - return lhs.is_selectable(); // Disabled items always go last - } - Character &player_character = get_player_character(); - // Place favorite items and items with an assigned inventory letter first, - // since the player cared enough to assign them - const bool left_has_invlet = player_character.inv->assigned_invlet.count( lhs.any_item()->invlet ); - const bool right_has_invlet = player_character.inv->assigned_invlet.count( rhs.any_item()->invlet ); - if( left_has_invlet != right_has_invlet ) { - return left_has_invlet; - } - const bool left_fav = lhs.any_item()->is_favorite; - const bool right_fav = rhs.any_item()->is_favorite; - if( left_fav != right_fav ) { - return left_fav; + if( indent_entries() ) { + // place children below all parents + bool const rhs_is_child = is_child( lhs, rhs ); + if( rhs_is_child ) { + return true; + } + bool const lhs_is_child = is_child( rhs, lhs ); + if( lhs_is_child ) { + return false; + } + // otherwise sort the entries at their common level + common_depth_t const common_level = common_depth( lhs, rhs ); + inventory_entry const ep_lhs( { common_level.first } ); + inventory_entry const ep_rhs( { common_level.second } ); + return sort_compare( ep_lhs, ep_rhs ); } - return preset.sort_compare( lhs, rhs ); + + return sort_compare( lhs, rhs ); } ); } from = to; diff --git a/src/inventory_ui.h b/src/inventory_ui.h index 468dfc821848f..ee440a3b16896 100644 --- a/src/inventory_ui.h +++ b/src/inventory_ui.h @@ -85,8 +85,9 @@ class inventory_entry locations( locations ), chosen_count( chosen_count ), custom_category( custom_category ), - enabled( enabled ) - {} + enabled( enabled ) { + update_cache(); + } bool operator==( const inventory_entry &other ) const; bool operator!=( const inventory_entry &other ) const { @@ -431,6 +432,8 @@ class inventory_column size_t page_of( size_t index ) const; size_t page_of( const inventory_entry &entry ) const; + bool sort_compare( inventory_entry const &lhs, inventory_entry const &rhs ); + /** * Indentation of the entry. * @param entry The entry to check From 1ad189f930a49c7146bb39349d716394222e6883 Mon Sep 17 00:00:00 2001 From: andrei Date: Thu, 9 Dec 2021 08:10:34 +0200 Subject: [PATCH 3/5] inventory_sel_preset: prettier sort_compare Co-authored-by: jbytheway <52664+jbytheway@users.noreply.github.com> --- src/inventory_ui.cpp | 11 ++++------- 1 file changed, 4 insertions(+), 7 deletions(-) diff --git a/src/inventory_ui.cpp b/src/inventory_ui.cpp index 00c2facf963da..4559ba9e2f326 100644 --- a/src/inventory_ui.cpp +++ b/src/inventory_ui.cpp @@ -326,13 +326,10 @@ inventory_selector_preset::inventory_selector_preset() bool inventory_selector_preset::sort_compare( const inventory_entry &lhs, const inventory_entry &rhs ) const { - if( lhs.cached_name == rhs.cached_name ) { - std::string const ltname = lhs.any_item()->tname(); - std::string const rtname = rhs.any_item()->tname(); - return localized_compare( ltname, rtname ); - } - - return localized_compare( lhs.cached_name, rhs.cached_name ); + auto const sort_key = []( inventory_entry const & e ) { + return std::make_tuple( e.cached_name, e.any_item()->tname() ); + }; + return localized_compare( sort_key( lhs ), sort_key( rhs ) ); } bool inventory_selector_preset::cat_sort_compare( const inventory_entry &lhs, From 2c666c6375dea7f39c9be37682cddd46d6becff8 Mon Sep 17 00:00:00 2001 From: andrei Date: Thu, 9 Dec 2021 00:49:34 +0200 Subject: [PATCH 4/5] inventory_selector: nuke oder_by_parent send it back whence it came --- src/inventory_ui.cpp | 69 +------------------------------------------- 1 file changed, 1 insertion(+), 68 deletions(-) diff --git a/src/inventory_ui.cpp b/src/inventory_ui.cpp index 4559ba9e2f326..77424db5fbd13 100644 --- a/src/inventory_ui.cpp +++ b/src/inventory_ui.cpp @@ -128,7 +128,7 @@ static const int normal_column_gap = 8; static const double min_ratio_to_center = 0.85; /** These categories should keep their original order and can't be re-sorted by inventory presets */ -static const std::set ordered_categories = {{ "ITEMS_WORN" }}; +static const std::set ordered_categories = {}; struct navigation_mode_data { navigation_mode next_mode; @@ -845,72 +845,6 @@ void inventory_column::on_change( const inventory_entry &/* entry */ ) // stub } -void inventory_column::order_by_parent() -{ - std::unordered_map original_order; - original_order.reserve( entries.size() ); - for( size_t idx = 0; idx < entries.size(); ++idx ) { - if( entries[idx].is_item() ) { - for( const item_location &loc : entries[idx].locations ) { - original_order.emplace( reinterpret_cast( &*loc ), idx ); - } - } else { - original_order.emplace( reinterpret_cast( &entries[idx] ), idx ); - } - } - - struct entry_info { - inventory_entry entry; - std::vector recursive_order; - - entry_info( inventory_entry &&moved_entry, - const std::unordered_map &original_order ) - : entry( std::move( moved_entry ) ) { - if( entry.is_item() ) { - item_location loc = entry.any_item(); - while( true ) { - const std::uintptr_t uintptr = reinterpret_cast( &*loc ); - const auto it = original_order.find( uintptr ); - if( it != original_order.end() ) { - recursive_order.emplace_back( it->second ); - } - if( loc.has_parent() ) { - loc = loc.parent_item(); - } else { - break; - } - } - std::reverse( recursive_order.begin(), recursive_order.end() ); - } else { - const std::uintptr_t uintptr = reinterpret_cast( &moved_entry ); - const auto it = original_order.find( uintptr ); - if( it != original_order.end() ) { - recursive_order.emplace_back( it->second ); - } - } - } - - // NOLINTNEXTLINE(google-explicit-constructor) - operator inventory_entry &&() && { // *NOPAD* - return std::move( entry ); - } - - bool operator<( const entry_info &rhs ) const { - return recursive_order < rhs.recursive_order; - } - }; - - std::vector sorted_entries; - sorted_entries.reserve( entries.size() ); - for( inventory_entry &entry : entries ) { - sorted_entries.emplace_back( std::move( entry ), original_order ); - } - std::stable_sort( sorted_entries.begin(), sorted_entries.end() ); - - entries.assign( std::make_move_iterator( sorted_entries.begin() ), - std::make_move_iterator( sorted_entries.end() ) ); -} - void inventory_column::add_entry( const inventory_entry &entry ) { if( std::find( entries.begin(), entries.end(), entry ) != entries.end() ) { @@ -2450,7 +2384,6 @@ void inventory_selector::toggle_categorize_contained() /*custom_category=*/custom_category, /*chosen_count=*/entry->chosen_count, entry->topmost_parent ); } - own_gear_column.order_by_parent(); own_inv_column.clear(); } if( !selected.empty() ) { From 508f34dc7ac72a57f7d86b4a6248bbf653eaa7dd Mon Sep 17 00:00:00 2001 From: andrei Date: Sat, 11 Dec 2021 20:50:58 +0200 Subject: [PATCH 5/5] inventory_entry: record generation number and use it to completely define sort order --- src/inventory_ui.cpp | 8 +++++--- src/inventory_ui.h | 6 +++++- 2 files changed, 10 insertions(+), 4 deletions(-) diff --git a/src/inventory_ui.cpp b/src/inventory_ui.cpp index 77424db5fbd13..8be311b0b0aeb 100644 --- a/src/inventory_ui.cpp +++ b/src/inventory_ui.cpp @@ -327,7 +327,7 @@ bool inventory_selector_preset::sort_compare( const inventory_entry &lhs, const inventory_entry &rhs ) const { auto const sort_key = []( inventory_entry const & e ) { - return std::make_tuple( e.cached_name, e.any_item()->tname() ); + return std::make_tuple( e.cached_name, e.any_item()->tname(), e.generation ); }; return localized_compare( sort_key( lhs ), sort_key( rhs ) ); } @@ -881,6 +881,7 @@ void inventory_column::add_entry( const inventory_entry &entry ) locations.insert( locations.end(), entry.locations.begin(), entry.locations.end() ); inventory_entry nentry( locations, entry.get_category_ptr() ); nentry.topmost_parent = entry_with_loc->topmost_parent; + nentry.generation = entry_with_loc->generation; entries.erase( entry_with_loc ); add_entry( nentry ); } @@ -987,8 +988,8 @@ void inventory_column::prepare_paging( const std::string &filter ) } // otherwise sort the entries at their common level common_depth_t const common_level = common_depth( lhs, rhs ); - inventory_entry const ep_lhs( { common_level.first } ); - inventory_entry const ep_rhs( { common_level.second } ); + inventory_entry const ep_lhs( { common_level.first }, nullptr, true, 0, lhs.generation ); + inventory_entry const ep_rhs( { common_level.second }, nullptr, true, 0, rhs.generation ); return sort_compare( ep_lhs, ep_rhs ); } @@ -1425,6 +1426,7 @@ void inventory_selector::add_entry( inventory_column &target_column, entry.collapsed = locations.front()->is_collapsed(); entry.topmost_parent = topmost_parent; + entry.generation = entry_generation_number++; target_column.add_entry( entry ); shared_ptr_fast current_ui = ui.lock(); diff --git a/src/inventory_ui.h b/src/inventory_ui.h index ee440a3b16896..81ef664108a71 100644 --- a/src/inventory_ui.h +++ b/src/inventory_ui.h @@ -81,9 +81,11 @@ class inventory_entry explicit inventory_entry( const std::vector &locations, const item_category *custom_category = nullptr, bool enabled = true, - const size_t chosen_count = 0 ) : + const size_t chosen_count = 0, + size_t generation_number = 0 ) : locations( locations ), chosen_count( chosen_count ), + generation( generation_number ), custom_category( custom_category ), enabled( enabled ) { update_cache(); @@ -148,6 +150,7 @@ class inventory_entry bool collapsed = false; // topmost visible parent, used for visibility checks item *topmost_parent = nullptr; + size_t generation = 0; private: const item_category *custom_category = nullptr; @@ -793,6 +796,7 @@ class inventory_selector bool display_stats = true; bool use_invlet = true; selector_invlet_type invlet_type_ = SELECTOR_INVLET_DEFAULT; + size_t entry_generation_number = 0; public: std::string action_bound_to_key( char key ) const;