selection.rs

  1use crate::{Anchor, BufferSnapshot, TextDimension};
  2use std::cmp::Ordering;
  3use std::ops::Range;
  4
  5#[derive(Copy, Clone, Debug, PartialEq)]
  6pub enum SelectionGoal {
  7    None,
  8    HorizontalPosition(f32),
  9    HorizontalRange { start: f32, end: f32 },
 10    WrappedHorizontalPosition((u32, f32)),
 11}
 12
 13#[derive(Clone, Debug, PartialEq)]
 14pub struct Selection<T> {
 15    pub id: usize,
 16    pub start: T,
 17    pub end: T,
 18    pub reversed: bool,
 19    pub goal: SelectionGoal,
 20}
 21
 22impl Default for SelectionGoal {
 23    fn default() -> Self {
 24        Self::None
 25    }
 26}
 27
 28impl<T: Clone> Selection<T> {
 29    pub fn head(&self) -> T {
 30        if self.reversed {
 31            self.start.clone()
 32        } else {
 33            self.end.clone()
 34        }
 35    }
 36
 37    pub fn tail(&self) -> T {
 38        if self.reversed {
 39            self.end.clone()
 40        } else {
 41            self.start.clone()
 42        }
 43    }
 44
 45    pub fn map<F, S>(&self, f: F) -> Selection<S>
 46    where
 47        F: Fn(T) -> S,
 48    {
 49        Selection::<S> {
 50            id: self.id,
 51            start: f(self.start.clone()),
 52            end: f(self.end.clone()),
 53            reversed: self.reversed,
 54            goal: self.goal,
 55        }
 56    }
 57
 58    pub fn collapse_to(&mut self, point: T, new_goal: SelectionGoal) {
 59        self.start = point.clone();
 60        self.end = point;
 61        self.goal = new_goal;
 62        self.reversed = false;
 63    }
 64}
 65
 66impl<T: Copy + Ord> Selection<T> {
 67    pub fn is_empty(&self) -> bool {
 68        self.start == self.end
 69    }
 70
 71    pub fn set_head(&mut self, head: T, new_goal: SelectionGoal) {
 72        if head.cmp(&self.tail()) < Ordering::Equal {
 73            if !self.reversed {
 74                self.end = self.start;
 75                self.reversed = true;
 76            }
 77            self.start = head;
 78        } else {
 79            if self.reversed {
 80                self.start = self.end;
 81                self.reversed = false;
 82            }
 83            self.end = head;
 84        }
 85        self.goal = new_goal;
 86    }
 87
 88    pub fn range(&self) -> Range<T> {
 89        self.start..self.end
 90    }
 91}
 92
 93impl Selection<usize> {
 94    #[cfg(feature = "test-support")]
 95    pub fn from_offset(offset: usize) -> Self {
 96        Selection {
 97            id: 0,
 98            start: offset,
 99            end: offset,
100            goal: SelectionGoal::None,
101            reversed: false,
102        }
103    }
104
105    pub fn equals(&self, offset_range: &Range<usize>) -> bool {
106        self.start == offset_range.start && self.end == offset_range.end
107    }
108}
109
110impl Selection<Anchor> {
111    pub fn resolve<'a, D: 'a + TextDimension>(
112        &'a self,
113        snapshot: &'a BufferSnapshot,
114    ) -> Selection<D> {
115        Selection {
116            id: self.id,
117            start: snapshot.summary_for_anchor(&self.start),
118            end: snapshot.summary_for_anchor(&self.end),
119            reversed: self.reversed,
120            goal: self.goal,
121        }
122    }
123}