diff --git a/crates/collab_ui/src/contact_list.rs b/crates/collab_ui/src/contact_list.rs index 9091e2687b..a4a0b5d18e 100644 --- a/crates/collab_ui/src/contact_list.rs +++ b/crates/collab_ui/src/contact_list.rs @@ -1,4 +1,4 @@ -use std::sync::Arc; +use std::{mem, sync::Arc}; use crate::contacts_popover; use call::ActiveCall; @@ -334,8 +334,14 @@ impl ContactList { } else if !self.entries.is_empty() { self.selection = Some(0); } - cx.notify(); self.list_state.reset(self.entries.len()); + if let Some(ix) = self.selection { + self.list_state.scroll_to(ListOffset { + item_ix: ix, + offset_in_item: 0., + }); + } + cx.notify(); } fn select_prev(&mut self, _: &SelectPrev, cx: &mut ViewContext) { @@ -346,8 +352,14 @@ impl ContactList { self.selection = None; } } - cx.notify(); self.list_state.reset(self.entries.len()); + if let Some(ix) = self.selection { + self.list_state.scroll_to(ListOffset { + item_ix: ix, + offset_in_item: 0., + }); + } + cx.notify(); } fn confirm(&mut self, _: &Confirm, cx: &mut ViewContext) { @@ -404,7 +416,7 @@ impl ContactList { let executor = cx.background().clone(); let prev_selected_entry = self.selection.and_then(|ix| self.entries.get(ix).cloned()); - self.entries.clear(); + let old_entries = mem::take(&mut self.entries); if let Some(room) = ActiveCall::global(cx).read(cx).room() { let room = room.read(cx); @@ -653,7 +665,47 @@ impl ContactList { } } + let old_scroll_top = self.list_state.logical_scroll_top(); self.list_state.reset(self.entries.len()); + + // Attempt to maintain the same scroll position. + if let Some(old_top_entry) = old_entries.get(old_scroll_top.item_ix) { + let new_scroll_top = self + .entries + .iter() + .position(|entry| entry == old_top_entry) + .map(|item_ix| ListOffset { + item_ix, + offset_in_item: old_scroll_top.offset_in_item, + }) + .or_else(|| { + let entry_after_old_top = old_entries.get(old_scroll_top.item_ix + 1)?; + let item_ix = self + .entries + .iter() + .position(|entry| entry == entry_after_old_top)?; + Some(ListOffset { + item_ix, + offset_in_item: 0., + }) + }) + .or_else(|| { + let entry_before_old_top = + old_entries.get(old_scroll_top.item_ix.saturating_sub(1))?; + let item_ix = self + .entries + .iter() + .position(|entry| entry == entry_before_old_top)?; + Some(ListOffset { + item_ix, + offset_in_item: 0., + }) + }); + + self.list_state + .scroll_to(new_scroll_top.unwrap_or(old_scroll_top)); + } + cx.notify(); } diff --git a/crates/gpui/src/elements/list.rs b/crates/gpui/src/elements/list.rs index a1b4ef0c79..f1b747d647 100644 --- a/crates/gpui/src/elements/list.rs +++ b/crates/gpui/src/elements/list.rs @@ -38,8 +38,8 @@ struct StateInner { #[derive(Clone, Copy, Debug, Default, PartialEq)] pub struct ListOffset { - item_ix: usize, - offset_in_item: f32, + pub item_ix: usize, + pub offset_in_item: f32, } #[derive(Clone)] @@ -112,18 +112,7 @@ impl Element for List { let mut new_items = SumTree::new(); let mut rendered_items = VecDeque::new(); let mut rendered_height = 0.; - let mut scroll_top = state - .logical_scroll_top - .unwrap_or_else(|| match state.orientation { - Orientation::Top => ListOffset { - item_ix: 0, - offset_in_item: 0., - }, - Orientation::Bottom => ListOffset { - item_ix: state.items.summary().count, - offset_in_item: 0., - }, - }); + let mut scroll_top = state.logical_scroll_top(); // Render items after the scroll top, including those in the trailing overdraw. let mut cursor = old_items.cursor::(); @@ -421,6 +410,20 @@ impl ListState { ) { self.0.borrow_mut().scroll_handler = Some(Box::new(handler)) } + + pub fn logical_scroll_top(&self) -> ListOffset { + self.0.borrow().logical_scroll_top() + } + + pub fn scroll_to(&self, mut scroll_top: ListOffset) { + let state = &mut *self.0.borrow_mut(); + let item_count = state.items.summary().count; + if scroll_top.item_ix >= item_count { + scroll_top.item_ix = item_count; + scroll_top.offset_in_item = 0.; + } + state.logical_scroll_top = Some(scroll_top); + } } impl StateInner { @@ -514,6 +517,20 @@ impl StateInner { cx.notify(); } + fn logical_scroll_top(&self) -> ListOffset { + self.logical_scroll_top + .unwrap_or_else(|| match self.orientation { + Orientation::Top => ListOffset { + item_ix: 0, + offset_in_item: 0., + }, + Orientation::Bottom => ListOffset { + item_ix: self.items.summary().count, + offset_in_item: 0., + }, + }) + } + fn scroll_top(&self, logical_scroll_top: &ListOffset) -> f32 { let mut cursor = self.items.cursor::(); cursor.seek(&Count(logical_scroll_top.item_ix), Bias::Right, &());