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
 89impl<T: Copy> Selection<T> {
 90    pub fn range(&self) -> Range<T> {
 91        self.start..self.end
 92    }
 93}
 94
 95impl Selection<usize> {
 96    #[cfg(feature = "test-support")]
 97    pub fn from_offset(offset: usize) -> Self {
 98        Selection {
 99            id: 0,
100            start: offset,
101            end: offset,
102            goal: SelectionGoal::None,
103            reversed: false,
104        }
105    }
106
107    pub fn equals(&self, offset_range: &Range<usize>) -> bool {
108        self.start == offset_range.start && self.end == offset_range.end
109    }
110}
111
112impl Selection<Anchor> {
113    pub fn resolve<'a, D: 'a + TextDimension>(
114        &'a self,
115        snapshot: &'a BufferSnapshot,
116    ) -> Selection<D> {
117        Selection {
118            id: self.id,
119            start: snapshot.summary_for_anchor(&self.start),
120            end: snapshot.summary_for_anchor(&self.end),
121            reversed: self.reversed,
122            goal: self.goal,
123        }
124    }
125}