1use crate::{
2 display_map::ToDisplayPoint, link_go_to_definition::hide_link_definition,
3 movement::surrounding_word, Anchor, Autoscroll, Editor, Event, ExcerptId, MultiBuffer,
4 MultiBufferSnapshot, NavigationData, ToPoint as _, FORMAT_TIMEOUT,
5};
6use anyhow::{anyhow, Result};
7use futures::FutureExt;
8use gpui::{
9 elements::*, geometry::vector::vec2f, AppContext, Entity, ModelHandle, MutableAppContext,
10 RenderContext, Subscription, Task, View, ViewContext, ViewHandle,
11};
12use language::{Bias, Buffer, File as _, OffsetRangeExt, SelectionGoal};
13use project::{File, FormatTrigger, Project, ProjectEntryId, ProjectPath};
14use rope::point::Point;
15use rpc::proto::{self, update_view};
16use settings::Settings;
17use smallvec::SmallVec;
18use std::{
19 borrow::Cow,
20 cmp::{self, Ordering},
21 fmt::Write,
22 ops::Range,
23 path::{Path, PathBuf},
24};
25use text::Selection;
26use util::TryFutureExt;
27use workspace::{
28 searchable::{Direction, SearchEvent, SearchableItem, SearchableItemHandle},
29 FollowableItem, Item, ItemEvent, ItemHandle, ItemNavHistory, ProjectItem, StatusItemView,
30 ToolbarItemLocation,
31};
32
33pub const MAX_TAB_TITLE_LEN: usize = 24;
34
35impl FollowableItem for Editor {
36 fn from_state_proto(
37 pane: ViewHandle<workspace::Pane>,
38 project: ModelHandle<Project>,
39 state: &mut Option<proto::view::Variant>,
40 cx: &mut MutableAppContext,
41 ) -> Option<Task<Result<ViewHandle<Self>>>> {
42 let state = if matches!(state, Some(proto::view::Variant::Editor(_))) {
43 if let Some(proto::view::Variant::Editor(state)) = state.take() {
44 state
45 } else {
46 unreachable!()
47 }
48 } else {
49 return None;
50 };
51
52 let buffer = project.update(cx, |project, cx| {
53 project.open_buffer_by_id(state.buffer_id, cx)
54 });
55 Some(cx.spawn(|mut cx| async move {
56 let buffer = buffer.await?;
57 let editor = pane
58 .read_with(&cx, |pane, cx| {
59 pane.items_of_type::<Self>().find(|editor| {
60 editor.read(cx).buffer.read(cx).as_singleton().as_ref() == Some(&buffer)
61 })
62 })
63 .unwrap_or_else(|| {
64 pane.update(&mut cx, |_, cx| {
65 cx.add_view(|cx| Editor::for_buffer(buffer, Some(project), cx))
66 })
67 });
68 editor.update(&mut cx, |editor, cx| {
69 let excerpt_id;
70 let buffer_id;
71 {
72 let buffer = editor.buffer.read(cx).read(cx);
73 let singleton = buffer.as_singleton().unwrap();
74 excerpt_id = singleton.0.clone();
75 buffer_id = singleton.1;
76 }
77 let selections = state
78 .selections
79 .into_iter()
80 .map(|selection| {
81 deserialize_selection(&excerpt_id, buffer_id, selection)
82 .ok_or_else(|| anyhow!("invalid selection"))
83 })
84 .collect::<Result<Vec<_>>>()?;
85 if !selections.is_empty() {
86 editor.set_selections_from_remote(selections, cx);
87 }
88
89 if let Some(anchor) = state.scroll_top_anchor {
90 editor.set_scroll_top_anchor(
91 Anchor {
92 buffer_id: Some(state.buffer_id as usize),
93 excerpt_id,
94 text_anchor: language::proto::deserialize_anchor(anchor)
95 .ok_or_else(|| anyhow!("invalid scroll top"))?,
96 },
97 vec2f(state.scroll_x, state.scroll_y),
98 cx,
99 );
100 }
101
102 Ok::<_, anyhow::Error>(())
103 })?;
104 Ok(editor)
105 }))
106 }
107
108 fn set_leader_replica_id(
109 &mut self,
110 leader_replica_id: Option<u16>,
111 cx: &mut ViewContext<Self>,
112 ) {
113 self.leader_replica_id = leader_replica_id;
114 if self.leader_replica_id.is_some() {
115 self.buffer.update(cx, |buffer, cx| {
116 buffer.remove_active_selections(cx);
117 });
118 } else {
119 self.buffer.update(cx, |buffer, cx| {
120 if self.focused {
121 buffer.set_active_selections(
122 &self.selections.disjoint_anchors(),
123 self.selections.line_mode,
124 cx,
125 );
126 }
127 });
128 }
129 cx.notify();
130 }
131
132 fn to_state_proto(&self, cx: &AppContext) -> Option<proto::view::Variant> {
133 let buffer_id = self.buffer.read(cx).as_singleton()?.read(cx).remote_id();
134 Some(proto::view::Variant::Editor(proto::view::Editor {
135 buffer_id,
136 scroll_top_anchor: Some(language::proto::serialize_anchor(
137 &self.scroll_top_anchor.text_anchor,
138 )),
139 scroll_x: self.scroll_position.x(),
140 scroll_y: self.scroll_position.y(),
141 selections: self
142 .selections
143 .disjoint_anchors()
144 .iter()
145 .map(serialize_selection)
146 .collect(),
147 }))
148 }
149
150 fn add_event_to_update_proto(
151 &self,
152 event: &Self::Event,
153 update: &mut Option<proto::update_view::Variant>,
154 _: &AppContext,
155 ) -> bool {
156 let update =
157 update.get_or_insert_with(|| proto::update_view::Variant::Editor(Default::default()));
158
159 match update {
160 proto::update_view::Variant::Editor(update) => match event {
161 Event::ScrollPositionChanged { .. } => {
162 update.scroll_top_anchor = Some(language::proto::serialize_anchor(
163 &self.scroll_top_anchor.text_anchor,
164 ));
165 update.scroll_x = self.scroll_position.x();
166 update.scroll_y = self.scroll_position.y();
167 true
168 }
169 Event::SelectionsChanged { .. } => {
170 update.selections = self
171 .selections
172 .disjoint_anchors()
173 .iter()
174 .chain(self.selections.pending_anchor().as_ref())
175 .map(serialize_selection)
176 .collect();
177 true
178 }
179 _ => false,
180 },
181 }
182 }
183
184 fn apply_update_proto(
185 &mut self,
186 message: update_view::Variant,
187 cx: &mut ViewContext<Self>,
188 ) -> Result<()> {
189 match message {
190 update_view::Variant::Editor(message) => {
191 let buffer = self.buffer.read(cx);
192 let buffer = buffer.read(cx);
193 let (excerpt_id, buffer_id, _) = buffer.as_singleton().unwrap();
194 let excerpt_id = excerpt_id.clone();
195 drop(buffer);
196
197 let selections = message
198 .selections
199 .into_iter()
200 .filter_map(|selection| {
201 deserialize_selection(&excerpt_id, buffer_id, selection)
202 })
203 .collect::<Vec<_>>();
204
205 if !selections.is_empty() {
206 self.set_selections_from_remote(selections, cx);
207 self.request_autoscroll_remotely(Autoscroll::Newest, cx);
208 } else if let Some(anchor) = message.scroll_top_anchor {
209 self.set_scroll_top_anchor(
210 Anchor {
211 buffer_id: Some(buffer_id),
212 excerpt_id,
213 text_anchor: language::proto::deserialize_anchor(anchor)
214 .ok_or_else(|| anyhow!("invalid scroll top"))?,
215 },
216 vec2f(message.scroll_x, message.scroll_y),
217 cx,
218 );
219 }
220 }
221 }
222 Ok(())
223 }
224
225 fn should_unfollow_on_event(event: &Self::Event, _: &AppContext) -> bool {
226 match event {
227 Event::Edited => true,
228 Event::SelectionsChanged { local } => *local,
229 Event::ScrollPositionChanged { local } => *local,
230 _ => false,
231 }
232 }
233}
234
235fn serialize_selection(selection: &Selection<Anchor>) -> proto::Selection {
236 proto::Selection {
237 id: selection.id as u64,
238 start: Some(language::proto::serialize_anchor(
239 &selection.start.text_anchor,
240 )),
241 end: Some(language::proto::serialize_anchor(
242 &selection.end.text_anchor,
243 )),
244 reversed: selection.reversed,
245 }
246}
247
248fn deserialize_selection(
249 excerpt_id: &ExcerptId,
250 buffer_id: usize,
251 selection: proto::Selection,
252) -> Option<Selection<Anchor>> {
253 Some(Selection {
254 id: selection.id as usize,
255 start: Anchor {
256 buffer_id: Some(buffer_id),
257 excerpt_id: excerpt_id.clone(),
258 text_anchor: language::proto::deserialize_anchor(selection.start?)?,
259 },
260 end: Anchor {
261 buffer_id: Some(buffer_id),
262 excerpt_id: excerpt_id.clone(),
263 text_anchor: language::proto::deserialize_anchor(selection.end?)?,
264 },
265 reversed: selection.reversed,
266 goal: SelectionGoal::None,
267 })
268}
269
270impl Item for Editor {
271 fn navigate(&mut self, data: Box<dyn std::any::Any>, cx: &mut ViewContext<Self>) -> bool {
272 if let Ok(data) = data.downcast::<NavigationData>() {
273 let newest_selection = self.selections.newest::<Point>(cx);
274 let buffer = self.buffer.read(cx).read(cx);
275 let offset = if buffer.can_resolve(&data.cursor_anchor) {
276 data.cursor_anchor.to_point(&buffer)
277 } else {
278 buffer.clip_point(data.cursor_position, Bias::Left)
279 };
280
281 let scroll_top_anchor = if buffer.can_resolve(&data.scroll_top_anchor) {
282 data.scroll_top_anchor
283 } else {
284 buffer.anchor_before(
285 buffer.clip_point(Point::new(data.scroll_top_row, 0), Bias::Left),
286 )
287 };
288
289 drop(buffer);
290
291 if newest_selection.head() == offset {
292 false
293 } else {
294 let nav_history = self.nav_history.take();
295 self.scroll_position = data.scroll_position;
296 self.scroll_top_anchor = scroll_top_anchor;
297 self.change_selections(Some(Autoscroll::Fit), cx, |s| {
298 s.select_ranges([offset..offset])
299 });
300 self.nav_history = nav_history;
301 true
302 }
303 } else {
304 false
305 }
306 }
307
308 fn tab_description<'a>(&'a self, detail: usize, cx: &'a AppContext) -> Option<Cow<'a, str>> {
309 match path_for_buffer(&self.buffer, detail, true, cx)? {
310 Cow::Borrowed(path) => Some(path.to_string_lossy()),
311 Cow::Owned(path) => Some(path.to_string_lossy().to_string().into()),
312 }
313 }
314
315 fn tab_content(
316 &self,
317 detail: Option<usize>,
318 style: &theme::Tab,
319 cx: &AppContext,
320 ) -> ElementBox {
321 Flex::row()
322 .with_child(
323 Label::new(self.title(cx).into(), style.label.clone())
324 .aligned()
325 .boxed(),
326 )
327 .with_children(detail.and_then(|detail| {
328 let path = path_for_buffer(&self.buffer, detail, false, cx)?;
329 let description = path.to_string_lossy();
330 Some(
331 Label::new(
332 if description.len() > MAX_TAB_TITLE_LEN {
333 description[..MAX_TAB_TITLE_LEN].to_string() + "…"
334 } else {
335 description.into()
336 },
337 style.description.text.clone(),
338 )
339 .contained()
340 .with_style(style.description.container)
341 .aligned()
342 .boxed(),
343 )
344 }))
345 .boxed()
346 }
347
348 fn project_path(&self, cx: &AppContext) -> Option<ProjectPath> {
349 let buffer = self.buffer.read(cx).as_singleton()?;
350 let file = buffer.read(cx).file();
351 File::from_dyn(file).map(|file| ProjectPath {
352 worktree_id: file.worktree_id(cx),
353 path: file.path().clone(),
354 })
355 }
356
357 fn project_entry_ids(&self, cx: &AppContext) -> SmallVec<[ProjectEntryId; 3]> {
358 self.buffer
359 .read(cx)
360 .files(cx)
361 .into_iter()
362 .filter_map(|file| File::from_dyn(Some(file))?.project_entry_id(cx))
363 .collect()
364 }
365
366 fn is_singleton(&self, cx: &AppContext) -> bool {
367 self.buffer.read(cx).is_singleton()
368 }
369
370 fn clone_on_split(&self, cx: &mut ViewContext<Self>) -> Option<Self>
371 where
372 Self: Sized,
373 {
374 Some(self.clone(cx))
375 }
376
377 fn set_nav_history(&mut self, history: ItemNavHistory, _: &mut ViewContext<Self>) {
378 self.nav_history = Some(history);
379 }
380
381 fn deactivated(&mut self, cx: &mut ViewContext<Self>) {
382 let selection = self.selections.newest_anchor();
383 self.push_to_nav_history(selection.head(), None, cx);
384 }
385
386 fn workspace_deactivated(&mut self, cx: &mut ViewContext<Self>) {
387 hide_link_definition(self, cx);
388 self.link_go_to_definition_state.last_mouse_location = None;
389 }
390
391 fn is_dirty(&self, cx: &AppContext) -> bool {
392 self.buffer().read(cx).read(cx).is_dirty()
393 }
394
395 fn has_conflict(&self, cx: &AppContext) -> bool {
396 self.buffer().read(cx).read(cx).has_conflict()
397 }
398
399 fn can_save(&self, cx: &AppContext) -> bool {
400 !self.buffer().read(cx).is_singleton() || self.project_path(cx).is_some()
401 }
402
403 fn save(
404 &mut self,
405 project: ModelHandle<Project>,
406 cx: &mut ViewContext<Self>,
407 ) -> Task<Result<()>> {
408 self.report_event("save editor", cx);
409
410 let buffer = self.buffer().clone();
411 let buffers = buffer.read(cx).all_buffers();
412 let mut timeout = cx.background().timer(FORMAT_TIMEOUT).fuse();
413 let format = project.update(cx, |project, cx| {
414 project.format(buffers, true, FormatTrigger::Save, cx)
415 });
416 cx.spawn(|_, mut cx| async move {
417 let transaction = futures::select_biased! {
418 _ = timeout => {
419 log::warn!("timed out waiting for formatting");
420 None
421 }
422 transaction = format.log_err().fuse() => transaction,
423 };
424
425 buffer
426 .update(&mut cx, |buffer, cx| {
427 if let Some(transaction) = transaction {
428 if !buffer.is_singleton() {
429 buffer.push_transaction(&transaction.0);
430 }
431 }
432
433 buffer.save(cx)
434 })
435 .await?;
436 Ok(())
437 })
438 }
439
440 fn save_as(
441 &mut self,
442 project: ModelHandle<Project>,
443 abs_path: PathBuf,
444 cx: &mut ViewContext<Self>,
445 ) -> Task<Result<()>> {
446 let buffer = self
447 .buffer()
448 .read(cx)
449 .as_singleton()
450 .expect("cannot call save_as on an excerpt list");
451
452 project.update(cx, |project, cx| {
453 project.save_buffer_as(buffer, abs_path, cx)
454 })
455 }
456
457 fn reload(
458 &mut self,
459 project: ModelHandle<Project>,
460 cx: &mut ViewContext<Self>,
461 ) -> Task<Result<()>> {
462 let buffer = self.buffer().clone();
463 let buffers = self.buffer.read(cx).all_buffers();
464 let reload_buffers =
465 project.update(cx, |project, cx| project.reload_buffers(buffers, true, cx));
466 cx.spawn(|this, mut cx| async move {
467 let transaction = reload_buffers.log_err().await;
468 this.update(&mut cx, |editor, cx| {
469 editor.request_autoscroll(Autoscroll::Fit, cx)
470 });
471 buffer.update(&mut cx, |buffer, _| {
472 if let Some(transaction) = transaction {
473 if !buffer.is_singleton() {
474 buffer.push_transaction(&transaction.0);
475 }
476 }
477 });
478 Ok(())
479 })
480 }
481
482 fn git_diff_recalc(
483 &mut self,
484 _project: ModelHandle<Project>,
485 cx: &mut ViewContext<Self>,
486 ) -> Task<Result<()>> {
487 self.buffer().update(cx, |multibuffer, cx| {
488 multibuffer.git_diff_recalc(cx);
489 });
490 Task::ready(Ok(()))
491 }
492
493 fn to_item_events(event: &Self::Event) -> Vec<workspace::ItemEvent> {
494 let mut result = Vec::new();
495 match event {
496 Event::Closed => result.push(ItemEvent::CloseItem),
497 Event::Saved | Event::TitleChanged => {
498 result.push(ItemEvent::UpdateTab);
499 result.push(ItemEvent::UpdateBreadcrumbs);
500 }
501 Event::Reparsed => {
502 result.push(ItemEvent::UpdateBreadcrumbs);
503 }
504 Event::SelectionsChanged { local } if *local => {
505 result.push(ItemEvent::UpdateBreadcrumbs);
506 }
507 Event::DirtyChanged => {
508 result.push(ItemEvent::UpdateTab);
509 }
510 Event::BufferEdited => {
511 result.push(ItemEvent::Edit);
512 result.push(ItemEvent::UpdateBreadcrumbs);
513 }
514 _ => {}
515 }
516 result
517 }
518
519 fn as_searchable(&self, handle: &ViewHandle<Self>) -> Option<Box<dyn SearchableItemHandle>> {
520 Some(Box::new(handle.clone()))
521 }
522
523 fn breadcrumb_location(&self) -> ToolbarItemLocation {
524 ToolbarItemLocation::PrimaryLeft { flex: None }
525 }
526
527 fn breadcrumbs(&self, theme: &theme::Theme, cx: &AppContext) -> Option<Vec<ElementBox>> {
528 let cursor = self.selections.newest_anchor().head();
529 let multibuffer = &self.buffer().read(cx);
530 let (buffer_id, symbols) =
531 multibuffer.symbols_containing(cursor, Some(&theme.editor.syntax), cx)?;
532 let buffer = multibuffer.buffer(buffer_id)?;
533
534 let buffer = buffer.read(cx);
535 let filename = if let Some(file) = buffer.file() {
536 if file.path().file_name().is_none()
537 || self
538 .project
539 .as_ref()
540 .map(|project| project.read(cx).visible_worktrees(cx).count() > 1)
541 .unwrap_or_default()
542 {
543 file.full_path(cx).to_string_lossy().to_string()
544 } else {
545 file.path().to_string_lossy().to_string()
546 }
547 } else {
548 "untitled".to_string()
549 };
550
551 let mut breadcrumbs = vec![Label::new(filename, theme.breadcrumbs.text.clone()).boxed()];
552 breadcrumbs.extend(symbols.into_iter().map(|symbol| {
553 Text::new(symbol.text, theme.breadcrumbs.text.clone())
554 .with_highlights(symbol.highlight_ranges)
555 .boxed()
556 }));
557 Some(breadcrumbs)
558 }
559}
560
561impl ProjectItem for Editor {
562 type Item = Buffer;
563
564 fn for_project_item(
565 project: ModelHandle<Project>,
566 buffer: ModelHandle<Buffer>,
567 cx: &mut ViewContext<Self>,
568 ) -> Self {
569 Self::for_buffer(buffer, Some(project), cx)
570 }
571}
572
573enum BufferSearchHighlights {}
574impl SearchableItem for Editor {
575 type Match = Range<Anchor>;
576
577 fn to_search_event(event: &Self::Event) -> Option<SearchEvent> {
578 match event {
579 Event::BufferEdited => Some(SearchEvent::MatchesInvalidated),
580 Event::SelectionsChanged { .. } => Some(SearchEvent::ActiveMatchChanged),
581 _ => None,
582 }
583 }
584
585 fn clear_matches(&mut self, cx: &mut ViewContext<Self>) {
586 self.clear_background_highlights::<BufferSearchHighlights>(cx);
587 }
588
589 fn update_matches(&mut self, matches: Vec<Range<Anchor>>, cx: &mut ViewContext<Self>) {
590 self.highlight_background::<BufferSearchHighlights>(
591 matches,
592 |theme| theme.search.match_background,
593 cx,
594 );
595 }
596
597 fn query_suggestion(&mut self, cx: &mut ViewContext<Self>) -> String {
598 let display_map = self.snapshot(cx).display_snapshot;
599 let selection = self.selections.newest::<usize>(cx);
600 if selection.start == selection.end {
601 let point = selection.start.to_display_point(&display_map);
602 let range = surrounding_word(&display_map, point);
603 let range = range.start.to_offset(&display_map, Bias::Left)
604 ..range.end.to_offset(&display_map, Bias::Right);
605 let text: String = display_map.buffer_snapshot.text_for_range(range).collect();
606 if text.trim().is_empty() {
607 String::new()
608 } else {
609 text
610 }
611 } else {
612 display_map
613 .buffer_snapshot
614 .text_for_range(selection.start..selection.end)
615 .collect()
616 }
617 }
618
619 fn activate_match(
620 &mut self,
621 index: usize,
622 matches: Vec<Range<Anchor>>,
623 cx: &mut ViewContext<Self>,
624 ) {
625 self.unfold_ranges([matches[index].clone()], false, cx);
626 self.change_selections(Some(Autoscroll::Fit), cx, |s| {
627 s.select_ranges([matches[index].clone()])
628 });
629 }
630
631 fn match_index_for_direction(
632 &mut self,
633 matches: &Vec<Range<Anchor>>,
634 mut current_index: usize,
635 direction: Direction,
636 cx: &mut ViewContext<Self>,
637 ) -> usize {
638 let buffer = self.buffer().read(cx).snapshot(cx);
639 let cursor = self.selections.newest_anchor().head();
640 if matches[current_index].start.cmp(&cursor, &buffer).is_gt() {
641 if direction == Direction::Prev {
642 if current_index == 0 {
643 current_index = matches.len() - 1;
644 } else {
645 current_index -= 1;
646 }
647 }
648 } else if matches[current_index].end.cmp(&cursor, &buffer).is_lt() {
649 if direction == Direction::Next {
650 current_index = 0;
651 }
652 } else if direction == Direction::Prev {
653 if current_index == 0 {
654 current_index = matches.len() - 1;
655 } else {
656 current_index -= 1;
657 }
658 } else if direction == Direction::Next {
659 if current_index == matches.len() - 1 {
660 current_index = 0
661 } else {
662 current_index += 1;
663 }
664 };
665 current_index
666 }
667
668 fn find_matches(
669 &mut self,
670 query: project::search::SearchQuery,
671 cx: &mut ViewContext<Self>,
672 ) -> Task<Vec<Range<Anchor>>> {
673 let buffer = self.buffer().read(cx).snapshot(cx);
674 cx.background().spawn(async move {
675 let mut ranges = Vec::new();
676 if let Some((_, _, excerpt_buffer)) = buffer.as_singleton() {
677 ranges.extend(
678 query
679 .search(excerpt_buffer.as_rope())
680 .await
681 .into_iter()
682 .map(|range| {
683 buffer.anchor_after(range.start)..buffer.anchor_before(range.end)
684 }),
685 );
686 } else {
687 for excerpt in buffer.excerpt_boundaries_in_range(0..buffer.len()) {
688 let excerpt_range = excerpt.range.context.to_offset(&excerpt.buffer);
689 let rope = excerpt.buffer.as_rope().slice(excerpt_range.clone());
690 ranges.extend(query.search(&rope).await.into_iter().map(|range| {
691 let start = excerpt
692 .buffer
693 .anchor_after(excerpt_range.start + range.start);
694 let end = excerpt
695 .buffer
696 .anchor_before(excerpt_range.start + range.end);
697 buffer.anchor_in_excerpt(excerpt.id.clone(), start)
698 ..buffer.anchor_in_excerpt(excerpt.id.clone(), end)
699 }));
700 }
701 }
702 ranges
703 })
704 }
705
706 fn active_match_index(
707 &mut self,
708 matches: Vec<Range<Anchor>>,
709 cx: &mut ViewContext<Self>,
710 ) -> Option<usize> {
711 active_match_index(
712 &matches,
713 &self.selections.newest_anchor().head(),
714 &self.buffer().read(cx).snapshot(cx),
715 )
716 }
717}
718
719pub fn active_match_index(
720 ranges: &[Range<Anchor>],
721 cursor: &Anchor,
722 buffer: &MultiBufferSnapshot,
723) -> Option<usize> {
724 if ranges.is_empty() {
725 None
726 } else {
727 match ranges.binary_search_by(|probe| {
728 if probe.end.cmp(cursor, &*buffer).is_lt() {
729 Ordering::Less
730 } else if probe.start.cmp(cursor, &*buffer).is_gt() {
731 Ordering::Greater
732 } else {
733 Ordering::Equal
734 }
735 }) {
736 Ok(i) | Err(i) => Some(cmp::min(i, ranges.len() - 1)),
737 }
738 }
739}
740
741pub struct CursorPosition {
742 position: Option<Point>,
743 selected_count: usize,
744 _observe_active_editor: Option<Subscription>,
745}
746
747impl Default for CursorPosition {
748 fn default() -> Self {
749 Self::new()
750 }
751}
752
753impl CursorPosition {
754 pub fn new() -> Self {
755 Self {
756 position: None,
757 selected_count: 0,
758 _observe_active_editor: None,
759 }
760 }
761
762 fn update_position(&mut self, editor: ViewHandle<Editor>, cx: &mut ViewContext<Self>) {
763 let editor = editor.read(cx);
764 let buffer = editor.buffer().read(cx).snapshot(cx);
765
766 self.selected_count = 0;
767 let mut last_selection: Option<Selection<usize>> = None;
768 for selection in editor.selections.all::<usize>(cx) {
769 self.selected_count += selection.end - selection.start;
770 if last_selection
771 .as_ref()
772 .map_or(true, |last_selection| selection.id > last_selection.id)
773 {
774 last_selection = Some(selection);
775 }
776 }
777 self.position = last_selection.map(|s| s.head().to_point(&buffer));
778
779 cx.notify();
780 }
781}
782
783impl Entity for CursorPosition {
784 type Event = ();
785}
786
787impl View for CursorPosition {
788 fn ui_name() -> &'static str {
789 "CursorPosition"
790 }
791
792 fn render(&mut self, cx: &mut RenderContext<Self>) -> ElementBox {
793 if let Some(position) = self.position {
794 let theme = &cx.global::<Settings>().theme.workspace.status_bar;
795 let mut text = format!("{},{}", position.row + 1, position.column + 1);
796 if self.selected_count > 0 {
797 write!(text, " ({} selected)", self.selected_count).unwrap();
798 }
799 Label::new(text, theme.cursor_position.clone()).boxed()
800 } else {
801 Empty::new().boxed()
802 }
803 }
804}
805
806impl StatusItemView for CursorPosition {
807 fn set_active_pane_item(
808 &mut self,
809 active_pane_item: Option<&dyn ItemHandle>,
810 cx: &mut ViewContext<Self>,
811 ) {
812 if let Some(editor) = active_pane_item.and_then(|item| item.downcast::<Editor>()) {
813 self._observe_active_editor = Some(cx.observe(&editor, Self::update_position));
814 self.update_position(editor, cx);
815 } else {
816 self.position = None;
817 self._observe_active_editor = None;
818 }
819
820 cx.notify();
821 }
822}
823
824fn path_for_buffer<'a>(
825 buffer: &ModelHandle<MultiBuffer>,
826 mut height: usize,
827 include_filename: bool,
828 cx: &'a AppContext,
829) -> Option<Cow<'a, Path>> {
830 let file = buffer.read(cx).as_singleton()?.read(cx).file()?;
831 // Ensure we always render at least the filename.
832 height += 1;
833
834 let mut prefix = file.path().as_ref();
835 while height > 0 {
836 if let Some(parent) = prefix.parent() {
837 prefix = parent;
838 height -= 1;
839 } else {
840 break;
841 }
842 }
843
844 // Here we could have just always used `full_path`, but that is very
845 // allocation-heavy and so we try to use a `Cow<Path>` if we haven't
846 // traversed all the way up to the worktree's root.
847 if height > 0 {
848 let full_path = file.full_path(cx);
849 if include_filename {
850 Some(full_path.into())
851 } else {
852 Some(full_path.parent().unwrap().to_path_buf().into())
853 }
854 } else {
855 let mut path = file.path().strip_prefix(prefix).unwrap();
856 if !include_filename {
857 path = path.parent().unwrap();
858 }
859 Some(path.into())
860 }
861}