
Release Notes: - N/A --------- Co-authored-by: Antonio Scandurra <me@as-cii.com> Co-authored-by: Max Brunsfeld <maxbrunsfeld@gmail.com> Co-authored-by: Antonio <antonio@zed.dev> Co-authored-by: Max <max@zed.dev>
2694 lines
102 KiB
Rust
2694 lines
102 KiB
Rust
use super::{
|
|
wrap_map::{self, WrapEdit, WrapPoint, WrapSnapshot},
|
|
Highlights,
|
|
};
|
|
use crate::{EditorStyle, GutterDimensions};
|
|
use collections::{Bound, HashMap, HashSet};
|
|
use gpui::{AnyElement, EntityId, Pixels, WindowContext};
|
|
use language::{Chunk, Patch, Point};
|
|
use multi_buffer::{
|
|
Anchor, ExcerptId, ExcerptInfo, MultiBufferRow, MultiBufferSnapshot, ToPoint as _,
|
|
};
|
|
use parking_lot::Mutex;
|
|
use std::{
|
|
cell::RefCell,
|
|
cmp::{self, Ordering},
|
|
fmt::Debug,
|
|
ops::{Deref, DerefMut, Range, RangeBounds},
|
|
sync::{
|
|
atomic::{AtomicUsize, Ordering::SeqCst},
|
|
Arc,
|
|
},
|
|
};
|
|
use sum_tree::{Bias, SumTree, Summary, TreeMap};
|
|
use text::Edit;
|
|
use ui::ElementId;
|
|
|
|
const NEWLINES: &[u8] = &[b'\n'; u8::MAX as usize];
|
|
const BULLETS: &str = "********************************************************************************************************************************";
|
|
|
|
/// Tracks custom blocks such as diagnostics that should be displayed within buffer.
|
|
///
|
|
/// See the [`display_map` module documentation](crate::display_map) for more information.
|
|
pub struct BlockMap {
|
|
next_block_id: AtomicUsize,
|
|
wrap_snapshot: RefCell<WrapSnapshot>,
|
|
custom_blocks: Vec<Arc<CustomBlock>>,
|
|
custom_blocks_by_id: TreeMap<CustomBlockId, Arc<CustomBlock>>,
|
|
transforms: RefCell<SumTree<Transform>>,
|
|
show_excerpt_controls: bool,
|
|
buffer_header_height: u32,
|
|
excerpt_header_height: u32,
|
|
excerpt_footer_height: u32,
|
|
}
|
|
|
|
pub struct BlockMapReader<'a> {
|
|
blocks: &'a Vec<Arc<CustomBlock>>,
|
|
pub snapshot: BlockSnapshot,
|
|
}
|
|
|
|
pub struct BlockMapWriter<'a>(&'a mut BlockMap);
|
|
|
|
#[derive(Clone)]
|
|
pub struct BlockSnapshot {
|
|
wrap_snapshot: WrapSnapshot,
|
|
transforms: SumTree<Transform>,
|
|
custom_blocks_by_id: TreeMap<CustomBlockId, Arc<CustomBlock>>,
|
|
pub(super) buffer_header_height: u32,
|
|
pub(super) excerpt_header_height: u32,
|
|
pub(super) excerpt_footer_height: u32,
|
|
}
|
|
|
|
#[derive(Clone, Copy, Debug, Default, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
|
pub struct CustomBlockId(usize);
|
|
|
|
impl From<CustomBlockId> for ElementId {
|
|
fn from(val: CustomBlockId) -> Self {
|
|
ElementId::Integer(val.0)
|
|
}
|
|
}
|
|
|
|
#[derive(Copy, Clone, Debug, Default, Eq, Ord, PartialOrd, PartialEq)]
|
|
pub struct BlockPoint(pub Point);
|
|
|
|
#[derive(Copy, Clone, Debug, Default, Eq, Ord, PartialOrd, PartialEq)]
|
|
pub struct BlockRow(pub(super) u32);
|
|
|
|
#[derive(Copy, Clone, Debug, Default, Eq, Ord, PartialOrd, PartialEq)]
|
|
struct WrapRow(u32);
|
|
|
|
pub type RenderBlock = Box<dyn Send + FnMut(&mut BlockContext) -> AnyElement>;
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
|
pub enum BlockPlacement<T> {
|
|
Above(T),
|
|
Below(T),
|
|
Replace(Range<T>),
|
|
}
|
|
|
|
impl<T> BlockPlacement<T> {
|
|
fn start(&self) -> &T {
|
|
match self {
|
|
BlockPlacement::Above(position) => position,
|
|
BlockPlacement::Below(position) => position,
|
|
BlockPlacement::Replace(range) => &range.start,
|
|
}
|
|
}
|
|
|
|
fn end(&self) -> &T {
|
|
match self {
|
|
BlockPlacement::Above(position) => position,
|
|
BlockPlacement::Below(position) => position,
|
|
BlockPlacement::Replace(range) => &range.end,
|
|
}
|
|
}
|
|
|
|
pub fn as_ref(&self) -> BlockPlacement<&T> {
|
|
match self {
|
|
BlockPlacement::Above(position) => BlockPlacement::Above(position),
|
|
BlockPlacement::Below(position) => BlockPlacement::Below(position),
|
|
BlockPlacement::Replace(range) => BlockPlacement::Replace(&range.start..&range.end),
|
|
}
|
|
}
|
|
|
|
pub fn map<R>(self, mut f: impl FnMut(T) -> R) -> BlockPlacement<R> {
|
|
match self {
|
|
BlockPlacement::Above(position) => BlockPlacement::Above(f(position)),
|
|
BlockPlacement::Below(position) => BlockPlacement::Below(f(position)),
|
|
BlockPlacement::Replace(range) => BlockPlacement::Replace(f(range.start)..f(range.end)),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl BlockPlacement<Anchor> {
|
|
fn cmp(&self, other: &Self, buffer: &MultiBufferSnapshot) -> Ordering {
|
|
match (self, other) {
|
|
(BlockPlacement::Above(anchor_a), BlockPlacement::Above(anchor_b))
|
|
| (BlockPlacement::Below(anchor_a), BlockPlacement::Below(anchor_b)) => {
|
|
anchor_a.cmp(anchor_b, buffer)
|
|
}
|
|
(BlockPlacement::Above(anchor_a), BlockPlacement::Below(anchor_b)) => {
|
|
anchor_a.cmp(anchor_b, buffer).then(Ordering::Less)
|
|
}
|
|
(BlockPlacement::Below(anchor_a), BlockPlacement::Above(anchor_b)) => {
|
|
anchor_a.cmp(anchor_b, buffer).then(Ordering::Greater)
|
|
}
|
|
(BlockPlacement::Above(anchor), BlockPlacement::Replace(range)) => {
|
|
anchor.cmp(&range.start, buffer).then(Ordering::Less)
|
|
}
|
|
(BlockPlacement::Replace(range), BlockPlacement::Above(anchor)) => {
|
|
range.start.cmp(anchor, buffer).then(Ordering::Greater)
|
|
}
|
|
(BlockPlacement::Below(anchor), BlockPlacement::Replace(range)) => {
|
|
anchor.cmp(&range.start, buffer).then(Ordering::Greater)
|
|
}
|
|
(BlockPlacement::Replace(range), BlockPlacement::Below(anchor)) => {
|
|
range.start.cmp(anchor, buffer).then(Ordering::Less)
|
|
}
|
|
(BlockPlacement::Replace(range_a), BlockPlacement::Replace(range_b)) => range_a
|
|
.start
|
|
.cmp(&range_b.start, buffer)
|
|
.then_with(|| range_b.end.cmp(&range_a.end, buffer)),
|
|
}
|
|
}
|
|
|
|
fn to_wrap_row(&self, wrap_snapshot: &WrapSnapshot) -> Option<BlockPlacement<WrapRow>> {
|
|
let buffer_snapshot = wrap_snapshot.buffer_snapshot();
|
|
match self {
|
|
BlockPlacement::Above(position) => {
|
|
let mut position = position.to_point(buffer_snapshot);
|
|
position.column = 0;
|
|
let wrap_row = WrapRow(wrap_snapshot.make_wrap_point(position, Bias::Left).row());
|
|
Some(BlockPlacement::Above(wrap_row))
|
|
}
|
|
BlockPlacement::Below(position) => {
|
|
let mut position = position.to_point(buffer_snapshot);
|
|
position.column = buffer_snapshot.line_len(MultiBufferRow(position.row));
|
|
let wrap_row = WrapRow(wrap_snapshot.make_wrap_point(position, Bias::Left).row());
|
|
Some(BlockPlacement::Below(wrap_row))
|
|
}
|
|
BlockPlacement::Replace(range) => {
|
|
let mut start = range.start.to_point(buffer_snapshot);
|
|
let mut end = range.end.to_point(buffer_snapshot);
|
|
if start == end {
|
|
None
|
|
} else {
|
|
start.column = 0;
|
|
let start_wrap_row =
|
|
WrapRow(wrap_snapshot.make_wrap_point(start, Bias::Left).row());
|
|
end.column = buffer_snapshot.line_len(MultiBufferRow(end.row));
|
|
let end_wrap_row =
|
|
WrapRow(wrap_snapshot.make_wrap_point(end, Bias::Left).row());
|
|
Some(BlockPlacement::Replace(start_wrap_row..end_wrap_row))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Ord for BlockPlacement<WrapRow> {
|
|
fn cmp(&self, other: &Self) -> Ordering {
|
|
match (self, other) {
|
|
(BlockPlacement::Above(row_a), BlockPlacement::Above(row_b))
|
|
| (BlockPlacement::Below(row_a), BlockPlacement::Below(row_b)) => row_a.cmp(row_b),
|
|
(BlockPlacement::Above(row_a), BlockPlacement::Below(row_b)) => {
|
|
row_a.cmp(row_b).then(Ordering::Less)
|
|
}
|
|
(BlockPlacement::Below(row_a), BlockPlacement::Above(row_b)) => {
|
|
row_a.cmp(row_b).then(Ordering::Greater)
|
|
}
|
|
(BlockPlacement::Above(row), BlockPlacement::Replace(range)) => {
|
|
row.cmp(&range.start).then(Ordering::Less)
|
|
}
|
|
(BlockPlacement::Replace(range), BlockPlacement::Above(row)) => {
|
|
range.start.cmp(row).then(Ordering::Greater)
|
|
}
|
|
(BlockPlacement::Below(row), BlockPlacement::Replace(range)) => {
|
|
row.cmp(&range.start).then(Ordering::Greater)
|
|
}
|
|
(BlockPlacement::Replace(range), BlockPlacement::Below(row)) => {
|
|
range.start.cmp(row).then(Ordering::Less)
|
|
}
|
|
(BlockPlacement::Replace(range_a), BlockPlacement::Replace(range_b)) => range_a
|
|
.start
|
|
.cmp(&range_b.start)
|
|
.then_with(|| range_b.end.cmp(&range_a.end)),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl PartialOrd for BlockPlacement<WrapRow> {
|
|
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
|
|
Some(self.cmp(other))
|
|
}
|
|
}
|
|
|
|
pub struct CustomBlock {
|
|
id: CustomBlockId,
|
|
placement: BlockPlacement<Anchor>,
|
|
height: u32,
|
|
style: BlockStyle,
|
|
render: Arc<Mutex<RenderBlock>>,
|
|
priority: usize,
|
|
}
|
|
|
|
pub struct BlockProperties<P> {
|
|
pub placement: BlockPlacement<P>,
|
|
pub height: u32,
|
|
pub style: BlockStyle,
|
|
pub render: RenderBlock,
|
|
pub priority: usize,
|
|
}
|
|
|
|
impl<P: Debug> Debug for BlockProperties<P> {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
f.debug_struct("BlockProperties")
|
|
.field("placement", &self.placement)
|
|
.field("height", &self.height)
|
|
.field("style", &self.style)
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd)]
|
|
pub enum BlockStyle {
|
|
Fixed,
|
|
Flex,
|
|
Sticky,
|
|
}
|
|
|
|
pub struct BlockContext<'a, 'b> {
|
|
pub context: &'b mut WindowContext<'a>,
|
|
pub anchor_x: Pixels,
|
|
pub max_width: Pixels,
|
|
pub gutter_dimensions: &'b GutterDimensions,
|
|
pub em_width: Pixels,
|
|
pub line_height: Pixels,
|
|
pub block_id: BlockId,
|
|
pub selected: bool,
|
|
pub editor_style: &'b EditorStyle,
|
|
}
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Ord, PartialOrd, Hash)]
|
|
pub enum BlockId {
|
|
ExcerptBoundary(Option<ExcerptId>),
|
|
Custom(CustomBlockId),
|
|
}
|
|
|
|
impl From<BlockId> for ElementId {
|
|
fn from(value: BlockId) -> Self {
|
|
match value {
|
|
BlockId::Custom(CustomBlockId(id)) => ("Block", id).into(),
|
|
BlockId::ExcerptBoundary(next_excerpt) => match next_excerpt {
|
|
Some(id) => ("ExcerptBoundary", EntityId::from(id)).into(),
|
|
None => "LastExcerptBoundary".into(),
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
impl std::fmt::Display for BlockId {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match self {
|
|
Self::Custom(id) => write!(f, "Block({id:?})"),
|
|
Self::ExcerptBoundary(id) => write!(f, "ExcerptHeader({id:?})"),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug)]
|
|
struct Transform {
|
|
summary: TransformSummary,
|
|
block: Option<Block>,
|
|
}
|
|
|
|
#[allow(clippy::large_enum_variant)]
|
|
#[derive(Clone)]
|
|
pub enum Block {
|
|
Custom(Arc<CustomBlock>),
|
|
ExcerptBoundary {
|
|
prev_excerpt: Option<ExcerptInfo>,
|
|
next_excerpt: Option<ExcerptInfo>,
|
|
height: u32,
|
|
starts_new_buffer: bool,
|
|
show_excerpt_controls: bool,
|
|
},
|
|
}
|
|
|
|
impl Block {
|
|
pub fn id(&self) -> BlockId {
|
|
match self {
|
|
Block::Custom(block) => BlockId::Custom(block.id),
|
|
Block::ExcerptBoundary { next_excerpt, .. } => {
|
|
BlockId::ExcerptBoundary(next_excerpt.as_ref().map(|info| info.id))
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn height(&self) -> u32 {
|
|
match self {
|
|
Block::Custom(block) => block.height,
|
|
Block::ExcerptBoundary { height, .. } => *height,
|
|
}
|
|
}
|
|
|
|
pub fn style(&self) -> BlockStyle {
|
|
match self {
|
|
Block::Custom(block) => block.style,
|
|
Block::ExcerptBoundary { .. } => BlockStyle::Sticky,
|
|
}
|
|
}
|
|
|
|
fn place_above(&self) -> bool {
|
|
match self {
|
|
Block::Custom(block) => matches!(block.placement, BlockPlacement::Above(_)),
|
|
Block::ExcerptBoundary { next_excerpt, .. } => next_excerpt.is_some(),
|
|
}
|
|
}
|
|
|
|
fn place_below(&self) -> bool {
|
|
match self {
|
|
Block::Custom(block) => matches!(block.placement, BlockPlacement::Below(_)),
|
|
Block::ExcerptBoundary { next_excerpt, .. } => next_excerpt.is_none(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Debug for Block {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match self {
|
|
Self::Custom(block) => f.debug_struct("Custom").field("block", block).finish(),
|
|
Self::ExcerptBoundary {
|
|
starts_new_buffer,
|
|
next_excerpt,
|
|
prev_excerpt,
|
|
..
|
|
} => f
|
|
.debug_struct("ExcerptBoundary")
|
|
.field("prev_excerpt", &prev_excerpt)
|
|
.field("next_excerpt", &next_excerpt)
|
|
.field("starts_new_buffer", &starts_new_buffer)
|
|
.finish(),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug, Default)]
|
|
struct TransformSummary {
|
|
input_rows: u32,
|
|
output_rows: u32,
|
|
longest_row: u32,
|
|
longest_row_chars: u32,
|
|
}
|
|
|
|
pub struct BlockChunks<'a> {
|
|
transforms: sum_tree::Cursor<'a, Transform, (BlockRow, WrapRow)>,
|
|
input_chunks: wrap_map::WrapChunks<'a>,
|
|
input_chunk: Chunk<'a>,
|
|
output_row: u32,
|
|
max_output_row: u32,
|
|
masked: bool,
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
pub struct BlockBufferRows<'a> {
|
|
transforms: sum_tree::Cursor<'a, Transform, (BlockRow, WrapRow)>,
|
|
input_buffer_rows: wrap_map::WrapBufferRows<'a>,
|
|
output_row: BlockRow,
|
|
started: bool,
|
|
}
|
|
|
|
impl BlockMap {
|
|
pub fn new(
|
|
wrap_snapshot: WrapSnapshot,
|
|
show_excerpt_controls: bool,
|
|
buffer_header_height: u32,
|
|
excerpt_header_height: u32,
|
|
excerpt_footer_height: u32,
|
|
) -> Self {
|
|
let row_count = wrap_snapshot.max_point().row() + 1;
|
|
let mut transforms = SumTree::default();
|
|
push_isomorphic(&mut transforms, row_count, &wrap_snapshot);
|
|
let map = Self {
|
|
next_block_id: AtomicUsize::new(0),
|
|
custom_blocks: Vec::new(),
|
|
custom_blocks_by_id: TreeMap::default(),
|
|
transforms: RefCell::new(transforms),
|
|
wrap_snapshot: RefCell::new(wrap_snapshot.clone()),
|
|
show_excerpt_controls,
|
|
buffer_header_height,
|
|
excerpt_header_height,
|
|
excerpt_footer_height,
|
|
};
|
|
map.sync(
|
|
&wrap_snapshot,
|
|
Patch::new(vec![Edit {
|
|
old: 0..row_count,
|
|
new: 0..row_count,
|
|
}]),
|
|
);
|
|
map
|
|
}
|
|
|
|
pub fn read(&self, wrap_snapshot: WrapSnapshot, edits: Patch<u32>) -> BlockMapReader {
|
|
self.sync(&wrap_snapshot, edits);
|
|
*self.wrap_snapshot.borrow_mut() = wrap_snapshot.clone();
|
|
BlockMapReader {
|
|
blocks: &self.custom_blocks,
|
|
snapshot: BlockSnapshot {
|
|
wrap_snapshot,
|
|
transforms: self.transforms.borrow().clone(),
|
|
custom_blocks_by_id: self.custom_blocks_by_id.clone(),
|
|
buffer_header_height: self.buffer_header_height,
|
|
excerpt_header_height: self.excerpt_header_height,
|
|
excerpt_footer_height: self.excerpt_footer_height,
|
|
},
|
|
}
|
|
}
|
|
|
|
pub fn write(&mut self, wrap_snapshot: WrapSnapshot, edits: Patch<u32>) -> BlockMapWriter {
|
|
self.sync(&wrap_snapshot, edits);
|
|
*self.wrap_snapshot.borrow_mut() = wrap_snapshot;
|
|
BlockMapWriter(self)
|
|
}
|
|
|
|
fn sync(&self, wrap_snapshot: &WrapSnapshot, mut edits: Patch<u32>) {
|
|
let buffer = wrap_snapshot.buffer_snapshot();
|
|
|
|
// Handle changing the last excerpt if it is empty.
|
|
if buffer.trailing_excerpt_update_count()
|
|
!= self
|
|
.wrap_snapshot
|
|
.borrow()
|
|
.buffer_snapshot()
|
|
.trailing_excerpt_update_count()
|
|
{
|
|
let max_point = wrap_snapshot.max_point();
|
|
let edit_start = wrap_snapshot.prev_row_boundary(max_point);
|
|
let edit_end = max_point.row() + 1;
|
|
edits = edits.compose([WrapEdit {
|
|
old: edit_start..edit_end,
|
|
new: edit_start..edit_end,
|
|
}]);
|
|
}
|
|
|
|
let edits = edits.into_inner();
|
|
if edits.is_empty() {
|
|
return;
|
|
}
|
|
|
|
let mut transforms = self.transforms.borrow_mut();
|
|
let mut new_transforms = SumTree::default();
|
|
let mut cursor = transforms.cursor::<WrapRow>(&());
|
|
let mut last_block_ix = 0;
|
|
let mut blocks_in_edit = Vec::new();
|
|
let mut edits = edits.into_iter().peekable();
|
|
|
|
while let Some(edit) = edits.next() {
|
|
let mut old_start = WrapRow(edit.old.start);
|
|
let mut new_start = WrapRow(edit.new.start);
|
|
|
|
// Preserve transforms that:
|
|
// * strictly precedes this edit
|
|
// * isomorphic or replace transforms that end *at* the start of the edit
|
|
// * below blocks that end at the start of the edit
|
|
new_transforms.append(cursor.slice(&old_start, Bias::Left, &()), &());
|
|
if let Some(transform) = cursor.item() {
|
|
if transform.summary.input_rows > 0 && cursor.end(&()) == old_start {
|
|
// Preserve the transform (push and next)
|
|
new_transforms.push(transform.clone(), &());
|
|
cursor.next(&());
|
|
|
|
// Preserve below blocks at end of edit
|
|
while let Some(transform) = cursor.item() {
|
|
if transform.block.as_ref().map_or(false, |b| b.place_below()) {
|
|
new_transforms.push(transform.clone(), &());
|
|
cursor.next(&());
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Ensure the edit starts at a transform boundary.
|
|
// If the edit starts within an isomorphic transform, preserve its prefix
|
|
// If the edit lands within a replacement block, expand the edit to include the start of the replaced input range
|
|
let mut preserved_blocks_above_edit = false;
|
|
let transform = cursor.item().unwrap();
|
|
let transform_rows_before_edit = old_start.0 - cursor.start().0;
|
|
if transform_rows_before_edit > 0 {
|
|
if transform.block.is_none() {
|
|
// Preserve any portion of the old isomorphic transform that precedes this edit.
|
|
push_isomorphic(
|
|
&mut new_transforms,
|
|
transform_rows_before_edit,
|
|
wrap_snapshot,
|
|
);
|
|
} else {
|
|
// We landed within a block that replaces some lines, so we
|
|
// extend the edit to start at the beginning of the
|
|
// replacement.
|
|
debug_assert!(transform.summary.input_rows > 0);
|
|
old_start.0 -= transform_rows_before_edit;
|
|
new_start.0 -= transform_rows_before_edit;
|
|
// The blocks *above* it are already in the new transforms, so
|
|
// we don't need to re-insert them when querying blocks.
|
|
preserved_blocks_above_edit = true;
|
|
}
|
|
}
|
|
|
|
// Decide where the edit ends
|
|
// * It should end at a transform boundary
|
|
// * Coalesce edits that intersect the same transform
|
|
let mut old_end = WrapRow(edit.old.end);
|
|
let mut new_end = WrapRow(edit.new.end);
|
|
loop {
|
|
// Seek to the transform starting at or after the end of the edit
|
|
cursor.seek(&old_end, Bias::Left, &());
|
|
cursor.next(&());
|
|
|
|
// Extend edit to the end of the discarded transform so it is reconstructed in full
|
|
let transform_rows_after_edit = cursor.start().0 - old_end.0;
|
|
old_end.0 += transform_rows_after_edit;
|
|
new_end.0 += transform_rows_after_edit;
|
|
|
|
// Combine this edit with any subsequent edits that intersect the same transform.
|
|
while let Some(next_edit) = edits.peek() {
|
|
if next_edit.old.start <= cursor.start().0 {
|
|
old_end = WrapRow(next_edit.old.end);
|
|
new_end = WrapRow(next_edit.new.end);
|
|
cursor.seek(&old_end, Bias::Left, &());
|
|
cursor.next(&());
|
|
edits.next();
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if *cursor.start() == old_end {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Discard below blocks at the end of the edit. They'll be reconstructed.
|
|
while let Some(transform) = cursor.item() {
|
|
if transform.block.as_ref().map_or(false, |b| b.place_below()) {
|
|
cursor.next(&());
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Find the blocks within this edited region.
|
|
let new_buffer_start =
|
|
wrap_snapshot.to_point(WrapPoint::new(new_start.0, 0), Bias::Left);
|
|
let start_bound = Bound::Included(new_buffer_start);
|
|
let start_block_ix =
|
|
match self.custom_blocks[last_block_ix..].binary_search_by(|probe| {
|
|
probe
|
|
.start()
|
|
.to_point(buffer)
|
|
.cmp(&new_buffer_start)
|
|
// Move left until we find the index of the first block starting within this edit
|
|
.then(Ordering::Greater)
|
|
}) {
|
|
Ok(ix) | Err(ix) => last_block_ix + ix,
|
|
};
|
|
|
|
let end_bound;
|
|
let end_block_ix = if new_end.0 > wrap_snapshot.max_point().row() {
|
|
end_bound = Bound::Unbounded;
|
|
self.custom_blocks.len()
|
|
} else {
|
|
let new_buffer_end =
|
|
wrap_snapshot.to_point(WrapPoint::new(new_end.0, 0), Bias::Left);
|
|
end_bound = Bound::Excluded(new_buffer_end);
|
|
match self.custom_blocks[start_block_ix..].binary_search_by(|probe| {
|
|
probe
|
|
.start()
|
|
.to_point(buffer)
|
|
.cmp(&new_buffer_end)
|
|
.then(Ordering::Greater)
|
|
}) {
|
|
Ok(ix) | Err(ix) => start_block_ix + ix,
|
|
}
|
|
};
|
|
last_block_ix = end_block_ix;
|
|
|
|
debug_assert!(blocks_in_edit.is_empty());
|
|
|
|
blocks_in_edit.extend(
|
|
self.custom_blocks[start_block_ix..end_block_ix]
|
|
.iter()
|
|
.filter_map(|block| {
|
|
Some((
|
|
block.placement.to_wrap_row(wrap_snapshot)?,
|
|
Block::Custom(block.clone()),
|
|
))
|
|
}),
|
|
);
|
|
|
|
if buffer.show_headers() {
|
|
blocks_in_edit.extend(BlockMap::header_and_footer_blocks(
|
|
self.show_excerpt_controls,
|
|
self.excerpt_footer_height,
|
|
self.buffer_header_height,
|
|
self.excerpt_header_height,
|
|
buffer,
|
|
(start_bound, end_bound),
|
|
wrap_snapshot,
|
|
));
|
|
}
|
|
|
|
BlockMap::sort_blocks(&mut blocks_in_edit);
|
|
|
|
// For each of these blocks, insert a new isomorphic transform preceding the block,
|
|
// and then insert the block itself.
|
|
for (block_placement, block) in blocks_in_edit.drain(..) {
|
|
if preserved_blocks_above_edit
|
|
&& block_placement == BlockPlacement::Above(new_start)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
let mut summary = TransformSummary {
|
|
input_rows: 0,
|
|
output_rows: block.height(),
|
|
longest_row: 0,
|
|
longest_row_chars: 0,
|
|
};
|
|
|
|
let rows_before_block;
|
|
match block_placement {
|
|
BlockPlacement::Above(position) => {
|
|
rows_before_block = position.0 - new_transforms.summary().input_rows;
|
|
}
|
|
BlockPlacement::Below(position) => {
|
|
rows_before_block = (position.0 + 1) - new_transforms.summary().input_rows;
|
|
}
|
|
BlockPlacement::Replace(range) => {
|
|
rows_before_block = range.start.0 - new_transforms.summary().input_rows;
|
|
summary.input_rows = range.end.0 - range.start.0 + 1;
|
|
}
|
|
}
|
|
|
|
push_isomorphic(&mut new_transforms, rows_before_block, wrap_snapshot);
|
|
new_transforms.push(
|
|
Transform {
|
|
summary,
|
|
block: Some(block),
|
|
},
|
|
&(),
|
|
);
|
|
}
|
|
|
|
// Insert an isomorphic transform after the final block.
|
|
let rows_after_last_block = new_end
|
|
.0
|
|
.saturating_sub(new_transforms.summary().input_rows);
|
|
push_isomorphic(&mut new_transforms, rows_after_last_block, wrap_snapshot);
|
|
}
|
|
|
|
new_transforms.append(cursor.suffix(&()), &());
|
|
debug_assert_eq!(
|
|
new_transforms.summary().input_rows,
|
|
wrap_snapshot.max_point().row() + 1
|
|
);
|
|
|
|
drop(cursor);
|
|
*transforms = new_transforms;
|
|
}
|
|
|
|
pub fn replace_blocks(&mut self, mut renderers: HashMap<CustomBlockId, RenderBlock>) {
|
|
for block in &mut self.custom_blocks {
|
|
if let Some(render) = renderers.remove(&block.id) {
|
|
*block.render.lock() = render;
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn show_excerpt_controls(&self) -> bool {
|
|
self.show_excerpt_controls
|
|
}
|
|
|
|
fn header_and_footer_blocks<'a, 'b: 'a, 'c: 'a + 'b, R, T>(
|
|
show_excerpt_controls: bool,
|
|
excerpt_footer_height: u32,
|
|
buffer_header_height: u32,
|
|
excerpt_header_height: u32,
|
|
buffer: &'b multi_buffer::MultiBufferSnapshot,
|
|
range: R,
|
|
wrap_snapshot: &'c WrapSnapshot,
|
|
) -> impl Iterator<Item = (BlockPlacement<WrapRow>, Block)> + 'b
|
|
where
|
|
R: RangeBounds<T>,
|
|
T: multi_buffer::ToOffset,
|
|
{
|
|
buffer
|
|
.excerpt_boundaries_in_range(range)
|
|
.filter_map(move |excerpt_boundary| {
|
|
let wrap_row;
|
|
if excerpt_boundary.next.is_some() {
|
|
wrap_row = wrap_snapshot
|
|
.make_wrap_point(Point::new(excerpt_boundary.row.0, 0), Bias::Left)
|
|
.row();
|
|
} else {
|
|
wrap_row = wrap_snapshot
|
|
.make_wrap_point(
|
|
Point::new(
|
|
excerpt_boundary.row.0,
|
|
buffer.line_len(excerpt_boundary.row),
|
|
),
|
|
Bias::Left,
|
|
)
|
|
.row();
|
|
}
|
|
|
|
let starts_new_buffer = match (&excerpt_boundary.prev, &excerpt_boundary.next) {
|
|
(_, None) => false,
|
|
(None, Some(_)) => true,
|
|
(Some(prev), Some(next)) => prev.buffer_id != next.buffer_id,
|
|
};
|
|
|
|
let mut height = 0;
|
|
if excerpt_boundary.prev.is_some() {
|
|
if show_excerpt_controls {
|
|
height += excerpt_footer_height;
|
|
}
|
|
}
|
|
if excerpt_boundary.next.is_some() {
|
|
if starts_new_buffer {
|
|
height += buffer_header_height;
|
|
if show_excerpt_controls {
|
|
height += excerpt_header_height;
|
|
}
|
|
} else {
|
|
height += excerpt_header_height;
|
|
}
|
|
}
|
|
|
|
if height == 0 {
|
|
return None;
|
|
}
|
|
|
|
Some((
|
|
if excerpt_boundary.next.is_some() {
|
|
BlockPlacement::Above(WrapRow(wrap_row))
|
|
} else {
|
|
BlockPlacement::Below(WrapRow(wrap_row))
|
|
},
|
|
Block::ExcerptBoundary {
|
|
prev_excerpt: excerpt_boundary.prev,
|
|
next_excerpt: excerpt_boundary.next,
|
|
height,
|
|
starts_new_buffer,
|
|
show_excerpt_controls,
|
|
},
|
|
))
|
|
})
|
|
}
|
|
|
|
fn sort_blocks(blocks: &mut Vec<(BlockPlacement<WrapRow>, Block)>) {
|
|
blocks.sort_unstable_by(|(placement_a, block_a), (placement_b, block_b)| {
|
|
placement_a
|
|
.cmp(&placement_b)
|
|
.then_with(|| match (block_a, block_b) {
|
|
(
|
|
Block::ExcerptBoundary {
|
|
next_excerpt: next_excerpt_a,
|
|
..
|
|
},
|
|
Block::ExcerptBoundary {
|
|
next_excerpt: next_excerpt_b,
|
|
..
|
|
},
|
|
) => next_excerpt_a
|
|
.as_ref()
|
|
.map(|excerpt| excerpt.id)
|
|
.cmp(&next_excerpt_b.as_ref().map(|excerpt| excerpt.id)),
|
|
(Block::ExcerptBoundary { next_excerpt, .. }, Block::Custom(_)) => {
|
|
if next_excerpt.is_some() {
|
|
Ordering::Less
|
|
} else {
|
|
Ordering::Greater
|
|
}
|
|
}
|
|
(Block::Custom(_), Block::ExcerptBoundary { next_excerpt, .. }) => {
|
|
if next_excerpt.is_some() {
|
|
Ordering::Greater
|
|
} else {
|
|
Ordering::Less
|
|
}
|
|
}
|
|
(Block::Custom(block_a), Block::Custom(block_b)) => block_a
|
|
.priority
|
|
.cmp(&block_b.priority)
|
|
.then_with(|| block_a.id.cmp(&block_b.id)),
|
|
})
|
|
});
|
|
blocks.dedup_by(|(right, _), (left, _)| match (left, right) {
|
|
(BlockPlacement::Replace(range), BlockPlacement::Above(row)) => {
|
|
range.start < *row && range.end >= *row
|
|
}
|
|
(BlockPlacement::Replace(range), BlockPlacement::Below(row)) => {
|
|
range.start <= *row && range.end > *row
|
|
}
|
|
(BlockPlacement::Replace(range_a), BlockPlacement::Replace(range_b)) => {
|
|
if range_a.end >= range_b.start && range_a.start <= range_b.end {
|
|
range_a.end = range_a.end.max(range_b.end);
|
|
true
|
|
} else {
|
|
false
|
|
}
|
|
}
|
|
_ => false,
|
|
});
|
|
}
|
|
}
|
|
|
|
fn push_isomorphic(tree: &mut SumTree<Transform>, rows: u32, wrap_snapshot: &WrapSnapshot) {
|
|
if rows == 0 {
|
|
return;
|
|
}
|
|
|
|
let wrap_row_start = tree.summary().input_rows;
|
|
let wrap_row_end = wrap_row_start + rows;
|
|
let wrap_summary = wrap_snapshot.text_summary_for_range(wrap_row_start..wrap_row_end);
|
|
let summary = TransformSummary {
|
|
input_rows: rows,
|
|
output_rows: rows,
|
|
longest_row: wrap_summary.longest_row,
|
|
longest_row_chars: wrap_summary.longest_row_chars,
|
|
};
|
|
let mut merged = false;
|
|
tree.update_last(
|
|
|last_transform| {
|
|
if last_transform.block.is_none() {
|
|
last_transform.summary.add_summary(&summary, &());
|
|
merged = true;
|
|
}
|
|
},
|
|
&(),
|
|
);
|
|
if !merged {
|
|
tree.push(
|
|
Transform {
|
|
summary,
|
|
block: None,
|
|
},
|
|
&(),
|
|
);
|
|
}
|
|
}
|
|
|
|
impl BlockPoint {
|
|
pub fn new(row: u32, column: u32) -> Self {
|
|
Self(Point::new(row, column))
|
|
}
|
|
}
|
|
|
|
impl Deref for BlockPoint {
|
|
type Target = Point;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.0
|
|
}
|
|
}
|
|
|
|
impl std::ops::DerefMut for BlockPoint {
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
&mut self.0
|
|
}
|
|
}
|
|
|
|
impl<'a> Deref for BlockMapReader<'a> {
|
|
type Target = BlockSnapshot;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.snapshot
|
|
}
|
|
}
|
|
|
|
impl<'a> DerefMut for BlockMapReader<'a> {
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
&mut self.snapshot
|
|
}
|
|
}
|
|
|
|
impl<'a> BlockMapReader<'a> {
|
|
pub fn row_for_block(&self, block_id: CustomBlockId) -> Option<BlockRow> {
|
|
let block = self.blocks.iter().find(|block| block.id == block_id)?;
|
|
let buffer_row = block
|
|
.start()
|
|
.to_point(self.wrap_snapshot.buffer_snapshot())
|
|
.row;
|
|
let wrap_row = self
|
|
.wrap_snapshot
|
|
.make_wrap_point(Point::new(buffer_row, 0), Bias::Left)
|
|
.row();
|
|
let start_wrap_row = WrapRow(
|
|
self.wrap_snapshot
|
|
.prev_row_boundary(WrapPoint::new(wrap_row, 0)),
|
|
);
|
|
let end_wrap_row = WrapRow(
|
|
self.wrap_snapshot
|
|
.next_row_boundary(WrapPoint::new(wrap_row, 0))
|
|
.unwrap_or(self.wrap_snapshot.max_point().row() + 1),
|
|
);
|
|
|
|
let mut cursor = self.transforms.cursor::<(WrapRow, BlockRow)>(&());
|
|
cursor.seek(&start_wrap_row, Bias::Left, &());
|
|
while let Some(transform) = cursor.item() {
|
|
if cursor.start().0 > end_wrap_row {
|
|
break;
|
|
}
|
|
|
|
if let Some(BlockId::Custom(id)) = transform.block.as_ref().map(|block| block.id()) {
|
|
if id == block_id {
|
|
return Some(cursor.start().1);
|
|
}
|
|
}
|
|
cursor.next(&());
|
|
}
|
|
|
|
None
|
|
}
|
|
}
|
|
|
|
impl<'a> BlockMapWriter<'a> {
|
|
pub fn insert(
|
|
&mut self,
|
|
blocks: impl IntoIterator<Item = BlockProperties<Anchor>>,
|
|
) -> Vec<CustomBlockId> {
|
|
let blocks = blocks.into_iter();
|
|
let mut ids = Vec::with_capacity(blocks.size_hint().1.unwrap_or(0));
|
|
let mut edits = Patch::default();
|
|
let wrap_snapshot = &*self.0.wrap_snapshot.borrow();
|
|
let buffer = wrap_snapshot.buffer_snapshot();
|
|
|
|
let mut previous_wrap_row_range: Option<Range<u32>> = None;
|
|
for block in blocks {
|
|
if let BlockPlacement::Replace(_) = &block.placement {
|
|
debug_assert!(block.height > 0);
|
|
}
|
|
|
|
let id = CustomBlockId(self.0.next_block_id.fetch_add(1, SeqCst));
|
|
ids.push(id);
|
|
|
|
let start = block.placement.start().to_point(buffer);
|
|
let end = block.placement.end().to_point(buffer);
|
|
let start_wrap_row = wrap_snapshot.make_wrap_point(start, Bias::Left).row();
|
|
let end_wrap_row = wrap_snapshot.make_wrap_point(end, Bias::Left).row();
|
|
|
|
let (start_row, end_row) = {
|
|
previous_wrap_row_range.take_if(|range| {
|
|
!range.contains(&start_wrap_row) || !range.contains(&end_wrap_row)
|
|
});
|
|
let range = previous_wrap_row_range.get_or_insert_with(|| {
|
|
let start_row =
|
|
wrap_snapshot.prev_row_boundary(WrapPoint::new(start_wrap_row, 0));
|
|
let end_row = wrap_snapshot
|
|
.next_row_boundary(WrapPoint::new(end_wrap_row, 0))
|
|
.unwrap_or(wrap_snapshot.max_point().row() + 1);
|
|
start_row..end_row
|
|
});
|
|
(range.start, range.end)
|
|
};
|
|
let block_ix = match self
|
|
.0
|
|
.custom_blocks
|
|
.binary_search_by(|probe| probe.placement.cmp(&block.placement, buffer))
|
|
{
|
|
Ok(ix) | Err(ix) => ix,
|
|
};
|
|
let new_block = Arc::new(CustomBlock {
|
|
id,
|
|
placement: block.placement,
|
|
height: block.height,
|
|
render: Arc::new(Mutex::new(block.render)),
|
|
style: block.style,
|
|
priority: block.priority,
|
|
});
|
|
self.0.custom_blocks.insert(block_ix, new_block.clone());
|
|
self.0.custom_blocks_by_id.insert(id, new_block);
|
|
|
|
edits = edits.compose([Edit {
|
|
old: start_row..end_row,
|
|
new: start_row..end_row,
|
|
}]);
|
|
}
|
|
|
|
self.0.sync(wrap_snapshot, edits);
|
|
ids
|
|
}
|
|
|
|
pub fn resize(&mut self, mut heights: HashMap<CustomBlockId, u32>) {
|
|
let wrap_snapshot = &*self.0.wrap_snapshot.borrow();
|
|
let buffer = wrap_snapshot.buffer_snapshot();
|
|
let mut edits = Patch::default();
|
|
let mut last_block_buffer_row = None;
|
|
|
|
for block in &mut self.0.custom_blocks {
|
|
if let Some(new_height) = heights.remove(&block.id) {
|
|
if let BlockPlacement::Replace(_) = &block.placement {
|
|
debug_assert!(new_height > 0);
|
|
}
|
|
|
|
if block.height != new_height {
|
|
let new_block = CustomBlock {
|
|
id: block.id,
|
|
placement: block.placement.clone(),
|
|
height: new_height,
|
|
style: block.style,
|
|
render: block.render.clone(),
|
|
priority: block.priority,
|
|
};
|
|
let new_block = Arc::new(new_block);
|
|
*block = new_block.clone();
|
|
self.0.custom_blocks_by_id.insert(block.id, new_block);
|
|
|
|
let start_row = block.placement.start().to_point(buffer).row;
|
|
let end_row = block.placement.end().to_point(buffer).row;
|
|
if last_block_buffer_row != Some(end_row) {
|
|
last_block_buffer_row = Some(end_row);
|
|
let start_wrap_row = wrap_snapshot
|
|
.make_wrap_point(Point::new(start_row, 0), Bias::Left)
|
|
.row();
|
|
let end_wrap_row = wrap_snapshot
|
|
.make_wrap_point(Point::new(end_row, 0), Bias::Left)
|
|
.row();
|
|
let start =
|
|
wrap_snapshot.prev_row_boundary(WrapPoint::new(start_wrap_row, 0));
|
|
let end = wrap_snapshot
|
|
.next_row_boundary(WrapPoint::new(end_wrap_row, 0))
|
|
.unwrap_or(wrap_snapshot.max_point().row() + 1);
|
|
edits.push(Edit {
|
|
old: start..end,
|
|
new: start..end,
|
|
})
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
self.0.sync(wrap_snapshot, edits);
|
|
}
|
|
|
|
pub fn remove(&mut self, block_ids: HashSet<CustomBlockId>) {
|
|
let wrap_snapshot = &*self.0.wrap_snapshot.borrow();
|
|
let buffer = wrap_snapshot.buffer_snapshot();
|
|
let mut edits = Patch::default();
|
|
let mut last_block_buffer_row = None;
|
|
let mut previous_wrap_row_range: Option<Range<u32>> = None;
|
|
self.0.custom_blocks.retain(|block| {
|
|
if block_ids.contains(&block.id) {
|
|
let start = block.placement.start().to_point(buffer);
|
|
let end = block.placement.end().to_point(buffer);
|
|
if last_block_buffer_row != Some(end.row) {
|
|
last_block_buffer_row = Some(end.row);
|
|
let start_wrap_row = wrap_snapshot.make_wrap_point(start, Bias::Left).row();
|
|
let end_wrap_row = wrap_snapshot.make_wrap_point(end, Bias::Left).row();
|
|
let (start_row, end_row) = {
|
|
previous_wrap_row_range.take_if(|range| {
|
|
!range.contains(&start_wrap_row) || !range.contains(&end_wrap_row)
|
|
});
|
|
let range = previous_wrap_row_range.get_or_insert_with(|| {
|
|
let start_row =
|
|
wrap_snapshot.prev_row_boundary(WrapPoint::new(start_wrap_row, 0));
|
|
let end_row = wrap_snapshot
|
|
.next_row_boundary(WrapPoint::new(end_wrap_row, 0))
|
|
.unwrap_or(wrap_snapshot.max_point().row() + 1);
|
|
start_row..end_row
|
|
});
|
|
(range.start, range.end)
|
|
};
|
|
|
|
edits.push(Edit {
|
|
old: start_row..end_row,
|
|
new: start_row..end_row,
|
|
})
|
|
}
|
|
false
|
|
} else {
|
|
true
|
|
}
|
|
});
|
|
self.0
|
|
.custom_blocks_by_id
|
|
.retain(|id, _| !block_ids.contains(id));
|
|
self.0.sync(wrap_snapshot, edits);
|
|
}
|
|
}
|
|
|
|
impl BlockSnapshot {
|
|
#[cfg(test)]
|
|
pub fn text(&self) -> String {
|
|
self.chunks(
|
|
0..self.transforms.summary().output_rows,
|
|
false,
|
|
false,
|
|
Highlights::default(),
|
|
)
|
|
.map(|chunk| chunk.text)
|
|
.collect()
|
|
}
|
|
|
|
pub(crate) fn chunks<'a>(
|
|
&'a self,
|
|
rows: Range<u32>,
|
|
language_aware: bool,
|
|
masked: bool,
|
|
highlights: Highlights<'a>,
|
|
) -> BlockChunks<'a> {
|
|
let max_output_row = cmp::min(rows.end, self.transforms.summary().output_rows);
|
|
|
|
let mut cursor = self.transforms.cursor::<(BlockRow, WrapRow)>(&());
|
|
cursor.seek(&BlockRow(rows.start), Bias::Right, &());
|
|
let transform_output_start = cursor.start().0 .0;
|
|
let transform_input_start = cursor.start().1 .0;
|
|
|
|
let mut input_start = transform_input_start;
|
|
let mut input_end = transform_input_start;
|
|
if let Some(transform) = cursor.item() {
|
|
if transform.block.is_none() {
|
|
input_start += rows.start - transform_output_start;
|
|
input_end += cmp::min(
|
|
rows.end - transform_output_start,
|
|
transform.summary.input_rows,
|
|
);
|
|
}
|
|
}
|
|
|
|
BlockChunks {
|
|
input_chunks: self.wrap_snapshot.chunks(
|
|
input_start..input_end,
|
|
language_aware,
|
|
highlights,
|
|
),
|
|
input_chunk: Default::default(),
|
|
transforms: cursor,
|
|
output_row: rows.start,
|
|
max_output_row,
|
|
masked,
|
|
}
|
|
}
|
|
|
|
pub(super) fn buffer_rows(&self, start_row: BlockRow) -> BlockBufferRows {
|
|
let mut cursor = self.transforms.cursor::<(BlockRow, WrapRow)>(&());
|
|
cursor.seek(&start_row, Bias::Right, &());
|
|
let (output_start, input_start) = cursor.start();
|
|
let overshoot = if cursor
|
|
.item()
|
|
.map_or(false, |transform| transform.block.is_none())
|
|
{
|
|
start_row.0 - output_start.0
|
|
} else {
|
|
0
|
|
};
|
|
let input_start_row = input_start.0 + overshoot;
|
|
BlockBufferRows {
|
|
transforms: cursor,
|
|
input_buffer_rows: self.wrap_snapshot.buffer_rows(input_start_row),
|
|
output_row: start_row,
|
|
started: false,
|
|
}
|
|
}
|
|
|
|
pub fn blocks_in_range(&self, rows: Range<u32>) -> impl Iterator<Item = (u32, &Block)> {
|
|
let mut cursor = self.transforms.cursor::<BlockRow>(&());
|
|
cursor.seek(&BlockRow(rows.start), Bias::Left, &());
|
|
while cursor.start().0 < rows.start && cursor.end(&()).0 <= rows.start {
|
|
cursor.next(&());
|
|
}
|
|
|
|
std::iter::from_fn(move || {
|
|
while let Some(transform) = cursor.item() {
|
|
let start_row = cursor.start().0;
|
|
if start_row > rows.end
|
|
|| (start_row == rows.end
|
|
&& transform
|
|
.block
|
|
.as_ref()
|
|
.map_or(false, |block| block.height() > 0))
|
|
{
|
|
break;
|
|
}
|
|
if let Some(block) = &transform.block {
|
|
cursor.next(&());
|
|
return Some((start_row, block));
|
|
} else {
|
|
cursor.next(&());
|
|
}
|
|
}
|
|
None
|
|
})
|
|
}
|
|
|
|
pub fn block_for_id(&self, block_id: BlockId) -> Option<Block> {
|
|
let buffer = self.wrap_snapshot.buffer_snapshot();
|
|
|
|
match block_id {
|
|
BlockId::Custom(custom_block_id) => {
|
|
let custom_block = self.custom_blocks_by_id.get(&custom_block_id)?;
|
|
Some(Block::Custom(custom_block.clone()))
|
|
}
|
|
BlockId::ExcerptBoundary(next_excerpt_id) => {
|
|
let wrap_point;
|
|
if let Some(next_excerpt_id) = next_excerpt_id {
|
|
let excerpt_range = buffer.range_for_excerpt::<Point>(next_excerpt_id)?;
|
|
wrap_point = self
|
|
.wrap_snapshot
|
|
.make_wrap_point(excerpt_range.start, Bias::Left);
|
|
} else {
|
|
wrap_point = self
|
|
.wrap_snapshot
|
|
.make_wrap_point(buffer.max_point(), Bias::Left);
|
|
}
|
|
|
|
let mut cursor = self.transforms.cursor::<(WrapRow, BlockRow)>(&());
|
|
cursor.seek(&WrapRow(wrap_point.row()), Bias::Left, &());
|
|
while let Some(transform) = cursor.item() {
|
|
if let Some(block) = transform.block.as_ref() {
|
|
if block.id() == block_id {
|
|
return Some(block.clone());
|
|
}
|
|
} else if cursor.start().0 > WrapRow(wrap_point.row()) {
|
|
break;
|
|
}
|
|
|
|
cursor.next(&());
|
|
}
|
|
|
|
None
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn max_point(&self) -> BlockPoint {
|
|
let row = self.transforms.summary().output_rows.saturating_sub(1);
|
|
BlockPoint::new(row, self.line_len(BlockRow(row)))
|
|
}
|
|
|
|
pub fn longest_row(&self) -> u32 {
|
|
self.transforms.summary().longest_row
|
|
}
|
|
|
|
pub(super) fn line_len(&self, row: BlockRow) -> u32 {
|
|
let mut cursor = self.transforms.cursor::<(BlockRow, WrapRow)>(&());
|
|
cursor.seek(&BlockRow(row.0), Bias::Right, &());
|
|
if let Some(transform) = cursor.item() {
|
|
let (output_start, input_start) = cursor.start();
|
|
let overshoot = row.0 - output_start.0;
|
|
if transform.block.is_some() {
|
|
0
|
|
} else {
|
|
self.wrap_snapshot.line_len(input_start.0 + overshoot)
|
|
}
|
|
} else if row.0 == 0 {
|
|
0
|
|
} else {
|
|
panic!("row out of range");
|
|
}
|
|
}
|
|
|
|
pub(super) fn is_block_line(&self, row: BlockRow) -> bool {
|
|
let mut cursor = self.transforms.cursor::<(BlockRow, WrapRow)>(&());
|
|
cursor.seek(&row, Bias::Right, &());
|
|
cursor.item().map_or(false, |t| t.block.is_some())
|
|
}
|
|
|
|
pub fn clip_point(&self, point: BlockPoint, bias: Bias) -> BlockPoint {
|
|
let mut cursor = self.transforms.cursor::<(BlockRow, WrapRow)>(&());
|
|
cursor.seek(&BlockRow(point.row), Bias::Right, &());
|
|
|
|
let max_input_row = WrapRow(self.transforms.summary().input_rows);
|
|
let mut search_left =
|
|
(bias == Bias::Left && cursor.start().1 .0 > 0) || cursor.end(&()).1 == max_input_row;
|
|
let mut reversed = false;
|
|
|
|
loop {
|
|
if let Some(transform) = cursor.item() {
|
|
let (output_start_row, input_start_row) = cursor.start();
|
|
let (output_end_row, input_end_row) = cursor.end(&());
|
|
let output_start = Point::new(output_start_row.0, 0);
|
|
let input_start = Point::new(input_start_row.0, 0);
|
|
let input_end = Point::new(input_end_row.0, 0);
|
|
|
|
match transform.block.as_ref() {
|
|
Some(Block::Custom(block))
|
|
if matches!(block.placement, BlockPlacement::Replace(_)) =>
|
|
{
|
|
if ((bias == Bias::Left || search_left) && output_start <= point.0)
|
|
|| (!search_left && output_start >= point.0)
|
|
{
|
|
return BlockPoint(output_start);
|
|
}
|
|
}
|
|
None => {
|
|
let input_point = if point.row >= output_end_row.0 {
|
|
let line_len = self.wrap_snapshot.line_len(input_end_row.0 - 1);
|
|
self.wrap_snapshot
|
|
.clip_point(WrapPoint::new(input_end_row.0 - 1, line_len), bias)
|
|
} else {
|
|
let output_overshoot = point.0.saturating_sub(output_start);
|
|
self.wrap_snapshot
|
|
.clip_point(WrapPoint(input_start + output_overshoot), bias)
|
|
};
|
|
|
|
if (input_start..input_end).contains(&input_point.0) {
|
|
let input_overshoot = input_point.0.saturating_sub(input_start);
|
|
return BlockPoint(output_start + input_overshoot);
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
|
|
if search_left {
|
|
cursor.prev(&());
|
|
} else {
|
|
cursor.next(&());
|
|
}
|
|
} else if reversed {
|
|
return self.max_point();
|
|
} else {
|
|
reversed = true;
|
|
search_left = !search_left;
|
|
cursor.seek(&BlockRow(point.row), Bias::Right, &());
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn to_block_point(&self, wrap_point: WrapPoint) -> BlockPoint {
|
|
let mut cursor = self.transforms.cursor::<(WrapRow, BlockRow)>(&());
|
|
cursor.seek(&WrapRow(wrap_point.row()), Bias::Right, &());
|
|
if let Some(transform) = cursor.item() {
|
|
if transform.block.is_some() {
|
|
BlockPoint::new(cursor.start().1 .0, 0)
|
|
} else {
|
|
let (input_start_row, output_start_row) = cursor.start();
|
|
let input_start = Point::new(input_start_row.0, 0);
|
|
let output_start = Point::new(output_start_row.0, 0);
|
|
let input_overshoot = wrap_point.0 - input_start;
|
|
BlockPoint(output_start + input_overshoot)
|
|
}
|
|
} else {
|
|
self.max_point()
|
|
}
|
|
}
|
|
|
|
pub fn to_wrap_point(&self, block_point: BlockPoint, bias: Bias) -> WrapPoint {
|
|
let mut cursor = self.transforms.cursor::<(BlockRow, WrapRow)>(&());
|
|
cursor.seek(&BlockRow(block_point.row), Bias::Right, &());
|
|
if let Some(transform) = cursor.item() {
|
|
match transform.block.as_ref() {
|
|
Some(block) => {
|
|
if block.place_below() {
|
|
let wrap_row = cursor.start().1 .0 - 1;
|
|
WrapPoint::new(wrap_row, self.wrap_snapshot.line_len(wrap_row))
|
|
} else if block.place_above() {
|
|
WrapPoint::new(cursor.start().1 .0, 0)
|
|
} else if bias == Bias::Left {
|
|
WrapPoint::new(cursor.start().1 .0, 0)
|
|
} else {
|
|
let wrap_row = cursor.end(&()).1 .0 - 1;
|
|
WrapPoint::new(wrap_row, self.wrap_snapshot.line_len(wrap_row))
|
|
}
|
|
}
|
|
None => {
|
|
let overshoot = block_point.row - cursor.start().0 .0;
|
|
let wrap_row = cursor.start().1 .0 + overshoot;
|
|
WrapPoint::new(wrap_row, block_point.column)
|
|
}
|
|
}
|
|
} else {
|
|
self.wrap_snapshot.max_point()
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a> BlockChunks<'a> {
|
|
/// Go to the next transform
|
|
fn advance(&mut self) {
|
|
self.transforms.next(&());
|
|
while let Some(transform) = self.transforms.item() {
|
|
if transform
|
|
.block
|
|
.as_ref()
|
|
.map_or(false, |block| block.height() == 0)
|
|
{
|
|
self.transforms.next(&());
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if self
|
|
.transforms
|
|
.item()
|
|
.map_or(false, |transform| transform.block.is_none())
|
|
{
|
|
let start_input_row = self.transforms.start().1 .0;
|
|
let start_output_row = self.transforms.start().0 .0;
|
|
if start_output_row < self.max_output_row {
|
|
let end_input_row = cmp::min(
|
|
self.transforms.end(&()).1 .0,
|
|
start_input_row + (self.max_output_row - start_output_row),
|
|
);
|
|
self.input_chunks.seek(start_input_row..end_input_row);
|
|
}
|
|
self.input_chunk = Chunk::default();
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a> Iterator for BlockChunks<'a> {
|
|
type Item = Chunk<'a>;
|
|
|
|
fn next(&mut self) -> Option<Self::Item> {
|
|
if self.output_row >= self.max_output_row {
|
|
return None;
|
|
}
|
|
|
|
let transform = self.transforms.item()?;
|
|
if transform.block.is_some() {
|
|
let block_start = self.transforms.start().0 .0;
|
|
let mut block_end = self.transforms.end(&()).0 .0;
|
|
self.advance();
|
|
if self.transforms.item().is_none() {
|
|
block_end -= 1;
|
|
}
|
|
|
|
let start_in_block = self.output_row - block_start;
|
|
let end_in_block = cmp::min(self.max_output_row, block_end) - block_start;
|
|
let line_count = end_in_block - start_in_block;
|
|
self.output_row += line_count;
|
|
|
|
return Some(Chunk {
|
|
text: unsafe { std::str::from_utf8_unchecked(&NEWLINES[..line_count as usize]) },
|
|
..Default::default()
|
|
});
|
|
}
|
|
|
|
if self.input_chunk.text.is_empty() {
|
|
if let Some(input_chunk) = self.input_chunks.next() {
|
|
self.input_chunk = input_chunk;
|
|
} else {
|
|
if self.output_row < self.max_output_row {
|
|
self.output_row += 1;
|
|
self.advance();
|
|
if self.transforms.item().is_some() {
|
|
return Some(Chunk {
|
|
text: "\n",
|
|
..Default::default()
|
|
});
|
|
}
|
|
}
|
|
return None;
|
|
}
|
|
}
|
|
|
|
let transform_end = self.transforms.end(&()).0 .0;
|
|
let (prefix_rows, prefix_bytes) =
|
|
offset_for_row(self.input_chunk.text, transform_end - self.output_row);
|
|
self.output_row += prefix_rows;
|
|
|
|
let (mut prefix, suffix) = self.input_chunk.text.split_at(prefix_bytes);
|
|
self.input_chunk.text = suffix;
|
|
if self.output_row == transform_end {
|
|
self.advance();
|
|
}
|
|
|
|
if self.masked {
|
|
// Not great for multibyte text because to keep cursor math correct we
|
|
// need to have the same number of bytes in the input as output.
|
|
let chars = prefix.chars().count();
|
|
let bullet_len = chars;
|
|
prefix = &BULLETS[..bullet_len];
|
|
}
|
|
|
|
Some(Chunk {
|
|
text: prefix,
|
|
..self.input_chunk.clone()
|
|
})
|
|
}
|
|
}
|
|
|
|
impl<'a> Iterator for BlockBufferRows<'a> {
|
|
type Item = Option<BlockRow>;
|
|
|
|
fn next(&mut self) -> Option<Self::Item> {
|
|
if self.started {
|
|
self.output_row.0 += 1;
|
|
} else {
|
|
self.started = true;
|
|
}
|
|
|
|
if self.output_row.0 >= self.transforms.end(&()).0 .0 {
|
|
self.transforms.next(&());
|
|
while let Some(transform) = self.transforms.item() {
|
|
if transform
|
|
.block
|
|
.as_ref()
|
|
.map_or(false, |block| block.height() == 0)
|
|
{
|
|
self.transforms.next(&());
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if self.transforms.item()?.block.is_none() {
|
|
self.input_buffer_rows.seek(self.transforms.start().1 .0);
|
|
}
|
|
}
|
|
|
|
let transform = self.transforms.item()?;
|
|
if transform.block.is_some() {
|
|
Some(None)
|
|
} else {
|
|
Some(self.input_buffer_rows.next().unwrap().map(BlockRow))
|
|
}
|
|
}
|
|
}
|
|
|
|
impl sum_tree::Item for Transform {
|
|
type Summary = TransformSummary;
|
|
|
|
fn summary(&self, _cx: &()) -> Self::Summary {
|
|
self.summary.clone()
|
|
}
|
|
}
|
|
|
|
impl sum_tree::Summary for TransformSummary {
|
|
type Context = ();
|
|
|
|
fn zero(_cx: &()) -> Self {
|
|
Default::default()
|
|
}
|
|
|
|
fn add_summary(&mut self, summary: &Self, _: &()) {
|
|
if summary.longest_row_chars > self.longest_row_chars {
|
|
self.longest_row = self.output_rows + summary.longest_row;
|
|
self.longest_row_chars = summary.longest_row_chars;
|
|
}
|
|
self.input_rows += summary.input_rows;
|
|
self.output_rows += summary.output_rows;
|
|
}
|
|
}
|
|
|
|
impl<'a> sum_tree::Dimension<'a, TransformSummary> for WrapRow {
|
|
fn zero(_cx: &()) -> Self {
|
|
Default::default()
|
|
}
|
|
|
|
fn add_summary(&mut self, summary: &'a TransformSummary, _: &()) {
|
|
self.0 += summary.input_rows;
|
|
}
|
|
}
|
|
|
|
impl<'a> sum_tree::Dimension<'a, TransformSummary> for BlockRow {
|
|
fn zero(_cx: &()) -> Self {
|
|
Default::default()
|
|
}
|
|
|
|
fn add_summary(&mut self, summary: &'a TransformSummary, _: &()) {
|
|
self.0 += summary.output_rows;
|
|
}
|
|
}
|
|
|
|
impl<'a> Deref for BlockContext<'a, '_> {
|
|
type Target = WindowContext<'a>;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
self.context
|
|
}
|
|
}
|
|
|
|
impl DerefMut for BlockContext<'_, '_> {
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
self.context
|
|
}
|
|
}
|
|
|
|
impl CustomBlock {
|
|
pub fn render(&self, cx: &mut BlockContext) -> AnyElement {
|
|
self.render.lock()(cx)
|
|
}
|
|
|
|
pub fn start(&self) -> Anchor {
|
|
*self.placement.start()
|
|
}
|
|
|
|
pub fn end(&self) -> Anchor {
|
|
*self.placement.end()
|
|
}
|
|
|
|
pub fn style(&self) -> BlockStyle {
|
|
self.style
|
|
}
|
|
}
|
|
|
|
impl Debug for CustomBlock {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
f.debug_struct("Block")
|
|
.field("id", &self.id)
|
|
.field("placement", &self.placement)
|
|
.field("height", &self.height)
|
|
.field("style", &self.style)
|
|
.field("priority", &self.priority)
|
|
.finish_non_exhaustive()
|
|
}
|
|
}
|
|
|
|
// Count the number of bytes prior to a target point. If the string doesn't contain the target
|
|
// point, return its total extent. Otherwise return the target point itself.
|
|
fn offset_for_row(s: &str, target: u32) -> (u32, usize) {
|
|
let mut row = 0;
|
|
let mut offset = 0;
|
|
for (ix, line) in s.split('\n').enumerate() {
|
|
if ix > 0 {
|
|
row += 1;
|
|
offset += 1;
|
|
}
|
|
if row >= target {
|
|
break;
|
|
}
|
|
offset += line.len();
|
|
}
|
|
(row, offset)
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use crate::display_map::{
|
|
fold_map::FoldMap, inlay_map::InlayMap, tab_map::TabMap, wrap_map::WrapMap,
|
|
};
|
|
use gpui::{div, font, px, AppContext, Context as _, Element};
|
|
use language::{Buffer, Capability};
|
|
use multi_buffer::{ExcerptRange, MultiBuffer};
|
|
use rand::prelude::*;
|
|
use settings::SettingsStore;
|
|
use std::env;
|
|
use util::RandomCharIter;
|
|
|
|
#[gpui::test]
|
|
fn test_offset_for_row() {
|
|
assert_eq!(offset_for_row("", 0), (0, 0));
|
|
assert_eq!(offset_for_row("", 1), (0, 0));
|
|
assert_eq!(offset_for_row("abcd", 0), (0, 0));
|
|
assert_eq!(offset_for_row("abcd", 1), (0, 4));
|
|
assert_eq!(offset_for_row("\n", 0), (0, 0));
|
|
assert_eq!(offset_for_row("\n", 1), (1, 1));
|
|
assert_eq!(offset_for_row("abc\ndef\nghi", 0), (0, 0));
|
|
assert_eq!(offset_for_row("abc\ndef\nghi", 1), (1, 4));
|
|
assert_eq!(offset_for_row("abc\ndef\nghi", 2), (2, 8));
|
|
assert_eq!(offset_for_row("abc\ndef\nghi", 3), (2, 11));
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_basic_blocks(cx: &mut gpui::TestAppContext) {
|
|
cx.update(init_test);
|
|
|
|
let text = "aaa\nbbb\nccc\nddd";
|
|
|
|
let buffer = cx.update(|cx| MultiBuffer::build_simple(text, cx));
|
|
let buffer_snapshot = cx.update(|cx| buffer.read(cx).snapshot(cx));
|
|
let subscription = buffer.update(cx, |buffer, _| buffer.subscribe());
|
|
let (mut inlay_map, inlay_snapshot) = InlayMap::new(buffer_snapshot.clone());
|
|
let (mut fold_map, fold_snapshot) = FoldMap::new(inlay_snapshot);
|
|
let (mut tab_map, tab_snapshot) = TabMap::new(fold_snapshot, 1.try_into().unwrap());
|
|
let (wrap_map, wraps_snapshot) =
|
|
cx.update(|cx| WrapMap::new(tab_snapshot, font("Helvetica"), px(14.0), None, cx));
|
|
let mut block_map = BlockMap::new(wraps_snapshot.clone(), true, 1, 1, 1);
|
|
|
|
let mut writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
let block_ids = writer.insert(vec![
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Above(buffer_snapshot.anchor_after(Point::new(1, 0))),
|
|
height: 1,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Above(buffer_snapshot.anchor_after(Point::new(1, 2))),
|
|
height: 2,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Below(buffer_snapshot.anchor_after(Point::new(3, 3))),
|
|
height: 3,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
]);
|
|
|
|
let snapshot = block_map.read(wraps_snapshot, Default::default());
|
|
assert_eq!(snapshot.text(), "aaa\n\n\n\nbbb\nccc\nddd\n\n\n");
|
|
|
|
let blocks = snapshot
|
|
.blocks_in_range(0..8)
|
|
.map(|(start_row, block)| {
|
|
let block = block.as_custom().unwrap();
|
|
(start_row..start_row + block.height, block.id)
|
|
})
|
|
.collect::<Vec<_>>();
|
|
|
|
// When multiple blocks are on the same line, the newer blocks appear first.
|
|
assert_eq!(
|
|
blocks,
|
|
&[
|
|
(1..2, block_ids[0]),
|
|
(2..4, block_ids[1]),
|
|
(7..10, block_ids[2]),
|
|
]
|
|
);
|
|
|
|
assert_eq!(
|
|
snapshot.to_block_point(WrapPoint::new(0, 3)),
|
|
BlockPoint::new(0, 3)
|
|
);
|
|
assert_eq!(
|
|
snapshot.to_block_point(WrapPoint::new(1, 0)),
|
|
BlockPoint::new(4, 0)
|
|
);
|
|
assert_eq!(
|
|
snapshot.to_block_point(WrapPoint::new(3, 3)),
|
|
BlockPoint::new(6, 3)
|
|
);
|
|
|
|
assert_eq!(
|
|
snapshot.to_wrap_point(BlockPoint::new(0, 3), Bias::Left),
|
|
WrapPoint::new(0, 3)
|
|
);
|
|
assert_eq!(
|
|
snapshot.to_wrap_point(BlockPoint::new(1, 0), Bias::Left),
|
|
WrapPoint::new(1, 0)
|
|
);
|
|
assert_eq!(
|
|
snapshot.to_wrap_point(BlockPoint::new(3, 0), Bias::Left),
|
|
WrapPoint::new(1, 0)
|
|
);
|
|
assert_eq!(
|
|
snapshot.to_wrap_point(BlockPoint::new(7, 0), Bias::Left),
|
|
WrapPoint::new(3, 3)
|
|
);
|
|
|
|
assert_eq!(
|
|
snapshot.clip_point(BlockPoint::new(1, 0), Bias::Left),
|
|
BlockPoint::new(0, 3)
|
|
);
|
|
assert_eq!(
|
|
snapshot.clip_point(BlockPoint::new(1, 0), Bias::Right),
|
|
BlockPoint::new(4, 0)
|
|
);
|
|
assert_eq!(
|
|
snapshot.clip_point(BlockPoint::new(1, 1), Bias::Left),
|
|
BlockPoint::new(0, 3)
|
|
);
|
|
assert_eq!(
|
|
snapshot.clip_point(BlockPoint::new(1, 1), Bias::Right),
|
|
BlockPoint::new(4, 0)
|
|
);
|
|
assert_eq!(
|
|
snapshot.clip_point(BlockPoint::new(4, 0), Bias::Left),
|
|
BlockPoint::new(4, 0)
|
|
);
|
|
assert_eq!(
|
|
snapshot.clip_point(BlockPoint::new(4, 0), Bias::Right),
|
|
BlockPoint::new(4, 0)
|
|
);
|
|
assert_eq!(
|
|
snapshot.clip_point(BlockPoint::new(6, 3), Bias::Left),
|
|
BlockPoint::new(6, 3)
|
|
);
|
|
assert_eq!(
|
|
snapshot.clip_point(BlockPoint::new(6, 3), Bias::Right),
|
|
BlockPoint::new(6, 3)
|
|
);
|
|
assert_eq!(
|
|
snapshot.clip_point(BlockPoint::new(7, 0), Bias::Left),
|
|
BlockPoint::new(6, 3)
|
|
);
|
|
assert_eq!(
|
|
snapshot.clip_point(BlockPoint::new(7, 0), Bias::Right),
|
|
BlockPoint::new(6, 3)
|
|
);
|
|
|
|
assert_eq!(
|
|
snapshot
|
|
.buffer_rows(BlockRow(0))
|
|
.map(|row| row.map(|r| r.0))
|
|
.collect::<Vec<_>>(),
|
|
&[
|
|
Some(0),
|
|
None,
|
|
None,
|
|
None,
|
|
Some(1),
|
|
Some(2),
|
|
Some(3),
|
|
None,
|
|
None,
|
|
None
|
|
]
|
|
);
|
|
|
|
// Insert a line break, separating two block decorations into separate lines.
|
|
let buffer_snapshot = buffer.update(cx, |buffer, cx| {
|
|
buffer.edit([(Point::new(1, 1)..Point::new(1, 1), "!!!\n")], None, cx);
|
|
buffer.snapshot(cx)
|
|
});
|
|
|
|
let (inlay_snapshot, inlay_edits) =
|
|
inlay_map.sync(buffer_snapshot, subscription.consume().into_inner());
|
|
let (fold_snapshot, fold_edits) = fold_map.read(inlay_snapshot, inlay_edits);
|
|
let (tab_snapshot, tab_edits) =
|
|
tab_map.sync(fold_snapshot, fold_edits, 4.try_into().unwrap());
|
|
let (wraps_snapshot, wrap_edits) = wrap_map.update(cx, |wrap_map, cx| {
|
|
wrap_map.sync(tab_snapshot, tab_edits, cx)
|
|
});
|
|
let snapshot = block_map.read(wraps_snapshot, wrap_edits);
|
|
assert_eq!(snapshot.text(), "aaa\n\nb!!!\n\n\nbb\nccc\nddd\n\n\n");
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_multibuffer_headers_and_footers(cx: &mut AppContext) {
|
|
init_test(cx);
|
|
|
|
let buffer1 = cx.new_model(|cx| Buffer::local("Buffer 1", cx));
|
|
let buffer2 = cx.new_model(|cx| Buffer::local("Buffer 2", cx));
|
|
let buffer3 = cx.new_model(|cx| Buffer::local("Buffer 3", cx));
|
|
|
|
let mut excerpt_ids = Vec::new();
|
|
let multi_buffer = cx.new_model(|cx| {
|
|
let mut multi_buffer = MultiBuffer::new(Capability::ReadWrite);
|
|
excerpt_ids.extend(multi_buffer.push_excerpts(
|
|
buffer1.clone(),
|
|
[ExcerptRange {
|
|
context: 0..buffer1.read(cx).len(),
|
|
primary: None,
|
|
}],
|
|
cx,
|
|
));
|
|
excerpt_ids.extend(multi_buffer.push_excerpts(
|
|
buffer2.clone(),
|
|
[ExcerptRange {
|
|
context: 0..buffer2.read(cx).len(),
|
|
primary: None,
|
|
}],
|
|
cx,
|
|
));
|
|
excerpt_ids.extend(multi_buffer.push_excerpts(
|
|
buffer3.clone(),
|
|
[ExcerptRange {
|
|
context: 0..buffer3.read(cx).len(),
|
|
primary: None,
|
|
}],
|
|
cx,
|
|
));
|
|
|
|
multi_buffer
|
|
});
|
|
|
|
let font = font("Helvetica");
|
|
let font_size = px(14.);
|
|
let font_id = cx.text_system().resolve_font(&font);
|
|
let mut wrap_width = px(0.);
|
|
for c in "Buff".chars() {
|
|
wrap_width += cx
|
|
.text_system()
|
|
.advance(font_id, font_size, c)
|
|
.unwrap()
|
|
.width;
|
|
}
|
|
|
|
let multi_buffer_snapshot = multi_buffer.read(cx).snapshot(cx);
|
|
let (_, inlay_snapshot) = InlayMap::new(multi_buffer_snapshot.clone());
|
|
let (_, fold_snapshot) = FoldMap::new(inlay_snapshot);
|
|
let (_, tab_snapshot) = TabMap::new(fold_snapshot, 4.try_into().unwrap());
|
|
let (_, wraps_snapshot) = WrapMap::new(tab_snapshot, font, font_size, Some(wrap_width), cx);
|
|
|
|
let block_map = BlockMap::new(wraps_snapshot.clone(), true, 1, 1, 1);
|
|
let snapshot = block_map.read(wraps_snapshot, Default::default());
|
|
|
|
// Each excerpt has a header above and footer below. Excerpts are also *separated* by a newline.
|
|
assert_eq!(
|
|
snapshot.text(),
|
|
"\n\nBuff\ner 1\n\n\n\nBuff\ner 2\n\n\n\nBuff\ner 3\n"
|
|
);
|
|
|
|
let blocks: Vec<_> = snapshot
|
|
.blocks_in_range(0..u32::MAX)
|
|
.map(|(row, block)| (row..row + block.height(), block.id()))
|
|
.collect();
|
|
assert_eq!(
|
|
blocks,
|
|
vec![
|
|
(0..2, BlockId::ExcerptBoundary(Some(excerpt_ids[0]))), // path, header
|
|
(4..7, BlockId::ExcerptBoundary(Some(excerpt_ids[1]))), // footer, path, header
|
|
(9..12, BlockId::ExcerptBoundary(Some(excerpt_ids[2]))), // footer, path, header
|
|
(14..15, BlockId::ExcerptBoundary(None)), // footer
|
|
]
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_replace_with_heights(cx: &mut gpui::TestAppContext) {
|
|
cx.update(init_test);
|
|
|
|
let text = "aaa\nbbb\nccc\nddd";
|
|
|
|
let buffer = cx.update(|cx| MultiBuffer::build_simple(text, cx));
|
|
let buffer_snapshot = cx.update(|cx| buffer.read(cx).snapshot(cx));
|
|
let _subscription = buffer.update(cx, |buffer, _| buffer.subscribe());
|
|
let (_inlay_map, inlay_snapshot) = InlayMap::new(buffer_snapshot.clone());
|
|
let (_fold_map, fold_snapshot) = FoldMap::new(inlay_snapshot);
|
|
let (_tab_map, tab_snapshot) = TabMap::new(fold_snapshot, 1.try_into().unwrap());
|
|
let (_wrap_map, wraps_snapshot) =
|
|
cx.update(|cx| WrapMap::new(tab_snapshot, font("Helvetica"), px(14.0), None, cx));
|
|
let mut block_map = BlockMap::new(wraps_snapshot.clone(), false, 1, 1, 0);
|
|
|
|
let mut writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
let block_ids = writer.insert(vec![
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Above(buffer_snapshot.anchor_after(Point::new(1, 0))),
|
|
height: 1,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Above(buffer_snapshot.anchor_after(Point::new(1, 2))),
|
|
height: 2,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Below(buffer_snapshot.anchor_after(Point::new(3, 3))),
|
|
height: 3,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
]);
|
|
|
|
{
|
|
let snapshot = block_map.read(wraps_snapshot.clone(), Default::default());
|
|
assert_eq!(snapshot.text(), "aaa\n\n\n\nbbb\nccc\nddd\n\n\n");
|
|
|
|
let mut block_map_writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
|
|
let mut new_heights = HashMap::default();
|
|
new_heights.insert(block_ids[0], 2);
|
|
block_map_writer.resize(new_heights);
|
|
let snapshot = block_map.read(wraps_snapshot.clone(), Default::default());
|
|
assert_eq!(snapshot.text(), "aaa\n\n\n\n\nbbb\nccc\nddd\n\n\n");
|
|
}
|
|
|
|
{
|
|
let mut block_map_writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
|
|
let mut new_heights = HashMap::default();
|
|
new_heights.insert(block_ids[0], 1);
|
|
block_map_writer.resize(new_heights);
|
|
|
|
let snapshot = block_map.read(wraps_snapshot.clone(), Default::default());
|
|
assert_eq!(snapshot.text(), "aaa\n\n\n\nbbb\nccc\nddd\n\n\n");
|
|
}
|
|
|
|
{
|
|
let mut block_map_writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
|
|
let mut new_heights = HashMap::default();
|
|
new_heights.insert(block_ids[0], 0);
|
|
block_map_writer.resize(new_heights);
|
|
|
|
let snapshot = block_map.read(wraps_snapshot.clone(), Default::default());
|
|
assert_eq!(snapshot.text(), "aaa\n\n\nbbb\nccc\nddd\n\n\n");
|
|
}
|
|
|
|
{
|
|
let mut block_map_writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
|
|
let mut new_heights = HashMap::default();
|
|
new_heights.insert(block_ids[0], 3);
|
|
block_map_writer.resize(new_heights);
|
|
|
|
let snapshot = block_map.read(wraps_snapshot.clone(), Default::default());
|
|
assert_eq!(snapshot.text(), "aaa\n\n\n\n\n\nbbb\nccc\nddd\n\n\n");
|
|
}
|
|
|
|
{
|
|
let mut block_map_writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
|
|
let mut new_heights = HashMap::default();
|
|
new_heights.insert(block_ids[0], 3);
|
|
block_map_writer.resize(new_heights);
|
|
|
|
let snapshot = block_map.read(wraps_snapshot.clone(), Default::default());
|
|
// Same height as before, should remain the same
|
|
assert_eq!(snapshot.text(), "aaa\n\n\n\n\n\nbbb\nccc\nddd\n\n\n");
|
|
}
|
|
}
|
|
|
|
#[cfg(target_os = "macos")]
|
|
#[gpui::test]
|
|
fn test_blocks_on_wrapped_lines(cx: &mut gpui::TestAppContext) {
|
|
cx.update(init_test);
|
|
|
|
let _font_id = cx.text_system().font_id(&font("Helvetica")).unwrap();
|
|
|
|
let text = "one two three\nfour five six\nseven eight";
|
|
|
|
let buffer = cx.update(|cx| MultiBuffer::build_simple(text, cx));
|
|
let buffer_snapshot = cx.update(|cx| buffer.read(cx).snapshot(cx));
|
|
let (_, inlay_snapshot) = InlayMap::new(buffer_snapshot.clone());
|
|
let (_, fold_snapshot) = FoldMap::new(inlay_snapshot);
|
|
let (_, tab_snapshot) = TabMap::new(fold_snapshot, 4.try_into().unwrap());
|
|
let (_, wraps_snapshot) = cx.update(|cx| {
|
|
WrapMap::new(tab_snapshot, font("Helvetica"), px(14.0), Some(px(60.)), cx)
|
|
});
|
|
let mut block_map = BlockMap::new(wraps_snapshot.clone(), true, 1, 1, 0);
|
|
|
|
let mut writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
writer.insert(vec![
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Above(buffer_snapshot.anchor_after(Point::new(1, 12))),
|
|
render: Box::new(|_| div().into_any()),
|
|
height: 1,
|
|
priority: 0,
|
|
},
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Below(buffer_snapshot.anchor_after(Point::new(1, 1))),
|
|
render: Box::new(|_| div().into_any()),
|
|
height: 1,
|
|
priority: 0,
|
|
},
|
|
]);
|
|
|
|
// Blocks with an 'above' disposition go above their corresponding buffer line.
|
|
// Blocks with a 'below' disposition go below their corresponding buffer line.
|
|
let snapshot = block_map.read(wraps_snapshot, Default::default());
|
|
assert_eq!(
|
|
snapshot.text(),
|
|
"one two \nthree\n\nfour five \nsix\n\nseven \neight"
|
|
);
|
|
}
|
|
|
|
#[gpui::test]
|
|
fn test_replace_lines(cx: &mut gpui::TestAppContext) {
|
|
cx.update(init_test);
|
|
|
|
let text = "line1\nline2\nline3\nline4\nline5";
|
|
|
|
let buffer = cx.update(|cx| MultiBuffer::build_simple(text, cx));
|
|
let buffer_subscription = buffer.update(cx, |buffer, _cx| buffer.subscribe());
|
|
let buffer_snapshot = cx.update(|cx| buffer.read(cx).snapshot(cx));
|
|
let (mut inlay_map, inlay_snapshot) = InlayMap::new(buffer_snapshot.clone());
|
|
let (mut fold_map, fold_snapshot) = FoldMap::new(inlay_snapshot);
|
|
let tab_size = 1.try_into().unwrap();
|
|
let (mut tab_map, tab_snapshot) = TabMap::new(fold_snapshot, tab_size);
|
|
let (wrap_map, wraps_snapshot) =
|
|
cx.update(|cx| WrapMap::new(tab_snapshot, font("Helvetica"), px(14.0), None, cx));
|
|
let mut block_map = BlockMap::new(wraps_snapshot.clone(), false, 1, 1, 0);
|
|
|
|
let mut writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
writer.insert(vec![BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Replace(
|
|
buffer_snapshot.anchor_after(Point::new(1, 3))
|
|
..buffer_snapshot.anchor_before(Point::new(3, 1)),
|
|
),
|
|
height: 4,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
}]);
|
|
|
|
let blocks_snapshot = block_map.read(wraps_snapshot, Default::default());
|
|
assert_eq!(blocks_snapshot.text(), "line1\n\n\n\n\nline5");
|
|
|
|
let buffer_snapshot = buffer.update(cx, |buffer, cx| {
|
|
buffer.edit([(Point::new(2, 0)..Point::new(3, 0), "")], None, cx);
|
|
buffer.snapshot(cx)
|
|
});
|
|
let (inlay_snapshot, inlay_edits) = inlay_map.sync(
|
|
buffer_snapshot.clone(),
|
|
buffer_subscription.consume().into_inner(),
|
|
);
|
|
let (fold_snapshot, fold_edits) = fold_map.read(inlay_snapshot, inlay_edits);
|
|
let (tab_snapshot, tab_edits) = tab_map.sync(fold_snapshot, fold_edits, tab_size);
|
|
let (wraps_snapshot, wrap_edits) = wrap_map.update(cx, |wrap_map, cx| {
|
|
wrap_map.sync(tab_snapshot, tab_edits, cx)
|
|
});
|
|
let blocks_snapshot = block_map.read(wraps_snapshot.clone(), wrap_edits);
|
|
assert_eq!(blocks_snapshot.text(), "line1\n\n\n\n\nline5");
|
|
|
|
let buffer_snapshot = buffer.update(cx, |buffer, cx| {
|
|
buffer.edit(
|
|
[(
|
|
Point::new(1, 5)..Point::new(1, 5),
|
|
"\nline 6\nline7\nline 8\nline 9",
|
|
)],
|
|
None,
|
|
cx,
|
|
);
|
|
buffer.snapshot(cx)
|
|
});
|
|
let (inlay_snapshot, inlay_edits) = inlay_map.sync(
|
|
buffer_snapshot.clone(),
|
|
buffer_subscription.consume().into_inner(),
|
|
);
|
|
let (fold_snapshot, fold_edits) = fold_map.read(inlay_snapshot, inlay_edits);
|
|
let (tab_snapshot, tab_edits) = tab_map.sync(fold_snapshot, fold_edits, tab_size);
|
|
let (wraps_snapshot, wrap_edits) = wrap_map.update(cx, |wrap_map, cx| {
|
|
wrap_map.sync(tab_snapshot, tab_edits, cx)
|
|
});
|
|
let blocks_snapshot = block_map.read(wraps_snapshot.clone(), wrap_edits);
|
|
assert_eq!(blocks_snapshot.text(), "line1\n\n\n\n\nline5");
|
|
|
|
// Ensure blocks inserted above the start or below the end of the replaced region are shown.
|
|
let mut writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
writer.insert(vec![
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Above(buffer_snapshot.anchor_after(Point::new(1, 3))),
|
|
height: 1,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Below(buffer_snapshot.anchor_after(Point::new(6, 2))),
|
|
height: 1,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
]);
|
|
let blocks_snapshot = block_map.read(wraps_snapshot.clone(), Default::default());
|
|
assert_eq!(blocks_snapshot.text(), "line1\n\n\n\n\n\n\nline5");
|
|
|
|
// Ensure blocks inserted *inside* replaced region are hidden.
|
|
let mut writer = block_map.write(wraps_snapshot.clone(), Default::default());
|
|
writer.insert(vec![
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Below(buffer_snapshot.anchor_after(Point::new(1, 3))),
|
|
height: 1,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Above(buffer_snapshot.anchor_after(Point::new(2, 1))),
|
|
height: 1,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement: BlockPlacement::Above(buffer_snapshot.anchor_after(Point::new(6, 1))),
|
|
height: 1,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
},
|
|
]);
|
|
let blocks_snapshot = block_map.read(wraps_snapshot, Default::default());
|
|
assert_eq!(blocks_snapshot.text(), "line1\n\n\n\n\n\n\nline5");
|
|
}
|
|
|
|
#[gpui::test(iterations = 100)]
|
|
fn test_random_blocks(cx: &mut gpui::TestAppContext, mut rng: StdRng) {
|
|
cx.update(init_test);
|
|
|
|
let operations = env::var("OPERATIONS")
|
|
.map(|i| i.parse().expect("invalid `OPERATIONS` variable"))
|
|
.unwrap_or(10);
|
|
|
|
let wrap_width = if rng.gen_bool(0.2) {
|
|
None
|
|
} else {
|
|
Some(px(rng.gen_range(0.0..=100.0)))
|
|
};
|
|
let tab_size = 1.try_into().unwrap();
|
|
let font_size = px(14.0);
|
|
let buffer_start_header_height = rng.gen_range(1..=5);
|
|
let excerpt_header_height = rng.gen_range(1..=5);
|
|
let excerpt_footer_height = rng.gen_range(1..=5);
|
|
|
|
log::info!("Wrap width: {:?}", wrap_width);
|
|
log::info!("Excerpt Header Height: {:?}", excerpt_header_height);
|
|
log::info!("Excerpt Footer Height: {:?}", excerpt_footer_height);
|
|
let is_singleton = rng.gen();
|
|
let buffer = if is_singleton {
|
|
let len = rng.gen_range(0..10);
|
|
let text = RandomCharIter::new(&mut rng).take(len).collect::<String>();
|
|
log::info!("initial singleton buffer text: {:?}", text);
|
|
cx.update(|cx| MultiBuffer::build_simple(&text, cx))
|
|
} else {
|
|
cx.update(|cx| {
|
|
let multibuffer = MultiBuffer::build_random(&mut rng, cx);
|
|
log::info!(
|
|
"initial multi-buffer text: {:?}",
|
|
multibuffer.read(cx).read(cx).text()
|
|
);
|
|
multibuffer
|
|
})
|
|
};
|
|
|
|
let mut buffer_snapshot = cx.update(|cx| buffer.read(cx).snapshot(cx));
|
|
let (mut inlay_map, inlay_snapshot) = InlayMap::new(buffer_snapshot.clone());
|
|
let (mut fold_map, fold_snapshot) = FoldMap::new(inlay_snapshot);
|
|
let (mut tab_map, tab_snapshot) = TabMap::new(fold_snapshot, 4.try_into().unwrap());
|
|
let (wrap_map, wraps_snapshot) = cx
|
|
.update(|cx| WrapMap::new(tab_snapshot, font("Helvetica"), font_size, wrap_width, cx));
|
|
let mut block_map = BlockMap::new(
|
|
wraps_snapshot,
|
|
true,
|
|
buffer_start_header_height,
|
|
excerpt_header_height,
|
|
excerpt_footer_height,
|
|
);
|
|
|
|
for _ in 0..operations {
|
|
let mut buffer_edits = Vec::new();
|
|
match rng.gen_range(0..=100) {
|
|
0..=19 => {
|
|
let wrap_width = if rng.gen_bool(0.2) {
|
|
None
|
|
} else {
|
|
Some(px(rng.gen_range(0.0..=100.0)))
|
|
};
|
|
log::info!("Setting wrap width to {:?}", wrap_width);
|
|
wrap_map.update(cx, |map, cx| map.set_wrap_width(wrap_width, cx));
|
|
}
|
|
20..=39 => {
|
|
let block_count = rng.gen_range(1..=5);
|
|
let block_properties = (0..block_count)
|
|
.map(|_| {
|
|
let buffer = cx.update(|cx| buffer.read(cx).read(cx).clone());
|
|
let offset =
|
|
buffer.clip_offset(rng.gen_range(0..=buffer.len()), Bias::Left);
|
|
let mut min_height = 0;
|
|
let placement = match rng.gen_range(0..3) {
|
|
0 => {
|
|
min_height = 1;
|
|
let start = buffer.anchor_after(offset);
|
|
let end = buffer.anchor_after(buffer.clip_offset(
|
|
rng.gen_range(offset..=buffer.len()),
|
|
Bias::Left,
|
|
));
|
|
BlockPlacement::Replace(start..end)
|
|
}
|
|
1 => BlockPlacement::Above(buffer.anchor_after(offset)),
|
|
_ => BlockPlacement::Below(buffer.anchor_after(offset)),
|
|
};
|
|
|
|
let height = rng.gen_range(min_height..5);
|
|
log::info!(
|
|
"inserting block {:?} with height {}",
|
|
placement.as_ref().map(|p| p.to_point(&buffer)),
|
|
height
|
|
);
|
|
BlockProperties {
|
|
style: BlockStyle::Fixed,
|
|
placement,
|
|
height,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
}
|
|
})
|
|
.collect::<Vec<_>>();
|
|
|
|
let (inlay_snapshot, inlay_edits) =
|
|
inlay_map.sync(buffer_snapshot.clone(), vec![]);
|
|
let (fold_snapshot, fold_edits) = fold_map.read(inlay_snapshot, inlay_edits);
|
|
let (tab_snapshot, tab_edits) =
|
|
tab_map.sync(fold_snapshot, fold_edits, tab_size);
|
|
let (wraps_snapshot, wrap_edits) = wrap_map.update(cx, |wrap_map, cx| {
|
|
wrap_map.sync(tab_snapshot, tab_edits, cx)
|
|
});
|
|
let mut block_map = block_map.write(wraps_snapshot, wrap_edits);
|
|
block_map.insert(block_properties.iter().map(|props| BlockProperties {
|
|
placement: props.placement.clone(),
|
|
height: props.height,
|
|
style: props.style,
|
|
render: Box::new(|_| div().into_any()),
|
|
priority: 0,
|
|
}));
|
|
}
|
|
40..=59 if !block_map.custom_blocks.is_empty() => {
|
|
let block_count = rng.gen_range(1..=4.min(block_map.custom_blocks.len()));
|
|
let block_ids_to_remove = block_map
|
|
.custom_blocks
|
|
.choose_multiple(&mut rng, block_count)
|
|
.map(|block| block.id)
|
|
.collect::<HashSet<_>>();
|
|
|
|
let (inlay_snapshot, inlay_edits) =
|
|
inlay_map.sync(buffer_snapshot.clone(), vec![]);
|
|
let (fold_snapshot, fold_edits) = fold_map.read(inlay_snapshot, inlay_edits);
|
|
let (tab_snapshot, tab_edits) =
|
|
tab_map.sync(fold_snapshot, fold_edits, tab_size);
|
|
let (wraps_snapshot, wrap_edits) = wrap_map.update(cx, |wrap_map, cx| {
|
|
wrap_map.sync(tab_snapshot, tab_edits, cx)
|
|
});
|
|
let mut block_map = block_map.write(wraps_snapshot, wrap_edits);
|
|
block_map.remove(block_ids_to_remove);
|
|
}
|
|
_ => {
|
|
buffer.update(cx, |buffer, cx| {
|
|
let mutation_count = rng.gen_range(1..=5);
|
|
let subscription = buffer.subscribe();
|
|
buffer.randomly_mutate(&mut rng, mutation_count, cx);
|
|
buffer_snapshot = buffer.snapshot(cx);
|
|
buffer_edits.extend(subscription.consume());
|
|
log::info!("buffer text: {:?}", buffer_snapshot.text());
|
|
});
|
|
}
|
|
}
|
|
|
|
let (inlay_snapshot, inlay_edits) =
|
|
inlay_map.sync(buffer_snapshot.clone(), buffer_edits);
|
|
let (fold_snapshot, fold_edits) = fold_map.read(inlay_snapshot, inlay_edits);
|
|
let (tab_snapshot, tab_edits) = tab_map.sync(fold_snapshot, fold_edits, tab_size);
|
|
let (wraps_snapshot, wrap_edits) = wrap_map.update(cx, |wrap_map, cx| {
|
|
wrap_map.sync(tab_snapshot, tab_edits, cx)
|
|
});
|
|
let blocks_snapshot = block_map.read(wraps_snapshot.clone(), wrap_edits);
|
|
assert_eq!(
|
|
blocks_snapshot.transforms.summary().input_rows,
|
|
wraps_snapshot.max_point().row() + 1
|
|
);
|
|
log::info!("wrapped text: {:?}", wraps_snapshot.text());
|
|
log::info!("blocks text: {:?}", blocks_snapshot.text());
|
|
|
|
let mut expected_blocks = Vec::new();
|
|
expected_blocks.extend(block_map.custom_blocks.iter().filter_map(|block| {
|
|
Some((
|
|
block.placement.to_wrap_row(&wraps_snapshot)?,
|
|
Block::Custom(block.clone()),
|
|
))
|
|
}));
|
|
|
|
// Note that this needs to be synced with the related section in BlockMap::sync
|
|
expected_blocks.extend(BlockMap::header_and_footer_blocks(
|
|
true,
|
|
excerpt_footer_height,
|
|
buffer_start_header_height,
|
|
excerpt_header_height,
|
|
&buffer_snapshot,
|
|
0..,
|
|
&wraps_snapshot,
|
|
));
|
|
|
|
BlockMap::sort_blocks(&mut expected_blocks);
|
|
|
|
for (placement, block) in &expected_blocks {
|
|
log::info!(
|
|
"Block {:?} placement: {:?} Height: {:?}",
|
|
block.id(),
|
|
placement,
|
|
block.height()
|
|
);
|
|
}
|
|
|
|
let mut sorted_blocks_iter = expected_blocks.into_iter().peekable();
|
|
|
|
let input_buffer_rows = buffer_snapshot
|
|
.buffer_rows(MultiBufferRow(0))
|
|
.collect::<Vec<_>>();
|
|
let mut expected_buffer_rows = Vec::new();
|
|
let mut expected_text = String::new();
|
|
let mut expected_block_positions = Vec::new();
|
|
let input_text = wraps_snapshot.text();
|
|
|
|
// Loop over the input lines, creating (N - 1) empty lines for
|
|
// blocks of height N.
|
|
//
|
|
// It's important to note that output *starts* as one empty line,
|
|
// so we special case row 0 to assume a leading '\n'.
|
|
//
|
|
// Linehood is the birthright of strings.
|
|
let mut input_text_lines = input_text.split('\n').enumerate().peekable();
|
|
let mut block_row = 0;
|
|
while let Some((wrap_row, input_line)) = input_text_lines.next() {
|
|
let wrap_row = wrap_row as u32;
|
|
|
|
// Create empty lines for the above block
|
|
while let Some((placement, block)) = sorted_blocks_iter.peek() {
|
|
if placement.start().0 == wrap_row && block.place_above() {
|
|
let (_, block) = sorted_blocks_iter.next().unwrap();
|
|
expected_block_positions.push((block_row, block.id()));
|
|
if block.height() > 0 {
|
|
let text = "\n".repeat((block.height() - 1) as usize);
|
|
if block_row > 0 {
|
|
expected_text.push('\n')
|
|
}
|
|
expected_text.push_str(&text);
|
|
for _ in 0..block.height() {
|
|
expected_buffer_rows.push(None);
|
|
}
|
|
block_row += block.height();
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Skip lines within replace blocks, then create empty lines for the replace block's height
|
|
let mut is_in_replace_block = false;
|
|
if let Some((BlockPlacement::Replace(replace_range), block)) =
|
|
sorted_blocks_iter.peek()
|
|
{
|
|
if wrap_row >= replace_range.start.0 {
|
|
is_in_replace_block = true;
|
|
if wrap_row == replace_range.end.0 {
|
|
expected_block_positions.push((block_row, block.id()));
|
|
if block.height() > 0 {
|
|
let text = "\n".repeat((block.height() - 1) as usize);
|
|
if block_row > 0 {
|
|
expected_text.push('\n');
|
|
}
|
|
expected_text.push_str(&text);
|
|
for _ in 0..block.height() {
|
|
expected_buffer_rows.push(None);
|
|
}
|
|
block_row += block.height();
|
|
}
|
|
|
|
sorted_blocks_iter.next();
|
|
}
|
|
}
|
|
}
|
|
|
|
if !is_in_replace_block {
|
|
let buffer_row = input_buffer_rows[wraps_snapshot
|
|
.to_point(WrapPoint::new(wrap_row, 0), Bias::Left)
|
|
.row as usize];
|
|
|
|
let soft_wrapped = wraps_snapshot
|
|
.to_tab_point(WrapPoint::new(wrap_row, 0))
|
|
.column()
|
|
> 0;
|
|
expected_buffer_rows.push(if soft_wrapped { None } else { buffer_row });
|
|
if block_row > 0 {
|
|
expected_text.push('\n');
|
|
}
|
|
expected_text.push_str(input_line);
|
|
block_row += 1;
|
|
}
|
|
|
|
while let Some((placement, block)) = sorted_blocks_iter.peek() {
|
|
if placement.end().0 == wrap_row && block.place_below() {
|
|
let (_, block) = sorted_blocks_iter.next().unwrap();
|
|
expected_block_positions.push((block_row, block.id()));
|
|
if block.height() > 0 {
|
|
let text = "\n".repeat((block.height() - 1) as usize);
|
|
if block_row > 0 {
|
|
expected_text.push('\n')
|
|
}
|
|
expected_text.push_str(&text);
|
|
for _ in 0..block.height() {
|
|
expected_buffer_rows.push(None);
|
|
}
|
|
block_row += block.height();
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
let expected_lines = expected_text.split('\n').collect::<Vec<_>>();
|
|
let expected_row_count = expected_lines.len();
|
|
|
|
assert_eq!(
|
|
blocks_snapshot.max_point().row + 1,
|
|
expected_row_count as u32
|
|
);
|
|
|
|
log::info!("expected text: {:?}", expected_text);
|
|
|
|
for start_row in 0..expected_row_count {
|
|
let end_row = rng.gen_range(start_row + 1..=expected_row_count);
|
|
let mut expected_text = expected_lines[start_row..end_row].join("\n");
|
|
if end_row < expected_row_count {
|
|
expected_text.push('\n');
|
|
}
|
|
|
|
let actual_text = blocks_snapshot
|
|
.chunks(
|
|
start_row as u32..end_row as u32,
|
|
false,
|
|
false,
|
|
Highlights::default(),
|
|
)
|
|
.map(|chunk| chunk.text)
|
|
.collect::<String>();
|
|
assert_eq!(
|
|
actual_text,
|
|
expected_text,
|
|
"incorrect text starting row row range {:?}",
|
|
start_row..end_row
|
|
);
|
|
assert_eq!(
|
|
blocks_snapshot
|
|
.buffer_rows(BlockRow(start_row as u32))
|
|
.map(|row| row.map(|r| r.0))
|
|
.collect::<Vec<_>>(),
|
|
&expected_buffer_rows[start_row..]
|
|
);
|
|
}
|
|
|
|
assert_eq!(
|
|
blocks_snapshot
|
|
.blocks_in_range(0..(expected_row_count as u32))
|
|
.map(|(row, block)| (row, block.id()))
|
|
.collect::<Vec<_>>(),
|
|
expected_block_positions,
|
|
"invalid blocks_in_range({:?})",
|
|
0..expected_row_count
|
|
);
|
|
|
|
for (_, expected_block) in
|
|
blocks_snapshot.blocks_in_range(0..(expected_row_count as u32))
|
|
{
|
|
let actual_block = blocks_snapshot.block_for_id(expected_block.id());
|
|
assert_eq!(
|
|
actual_block.map(|block| block.id()),
|
|
Some(expected_block.id())
|
|
);
|
|
}
|
|
|
|
for (block_row, block_id) in expected_block_positions {
|
|
if let BlockId::Custom(block_id) = block_id {
|
|
assert_eq!(
|
|
blocks_snapshot.row_for_block(block_id),
|
|
Some(BlockRow(block_row))
|
|
);
|
|
}
|
|
}
|
|
|
|
let mut expected_longest_rows = Vec::new();
|
|
let mut longest_line_len = -1_isize;
|
|
for (row, line) in expected_lines.iter().enumerate() {
|
|
let row = row as u32;
|
|
|
|
assert_eq!(
|
|
blocks_snapshot.line_len(BlockRow(row)),
|
|
line.len() as u32,
|
|
"invalid line len for row {}",
|
|
row
|
|
);
|
|
|
|
let line_char_count = line.chars().count() as isize;
|
|
match line_char_count.cmp(&longest_line_len) {
|
|
Ordering::Less => {}
|
|
Ordering::Equal => expected_longest_rows.push(row),
|
|
Ordering::Greater => {
|
|
longest_line_len = line_char_count;
|
|
expected_longest_rows.clear();
|
|
expected_longest_rows.push(row);
|
|
}
|
|
}
|
|
}
|
|
|
|
let longest_row = blocks_snapshot.longest_row();
|
|
assert!(
|
|
expected_longest_rows.contains(&longest_row),
|
|
"incorrect longest row {}. expected {:?} with length {}",
|
|
longest_row,
|
|
expected_longest_rows,
|
|
longest_line_len,
|
|
);
|
|
|
|
// Ensure that conversion between block points and wrap points is stable.
|
|
for row in 0..=blocks_snapshot.wrap_snapshot.max_point().row() {
|
|
let wrap_point = WrapPoint::new(row, 0);
|
|
let block_point = blocks_snapshot.to_block_point(wrap_point);
|
|
let left_wrap_point = blocks_snapshot.to_wrap_point(block_point, Bias::Left);
|
|
let right_wrap_point = blocks_snapshot.to_wrap_point(block_point, Bias::Right);
|
|
assert_eq!(blocks_snapshot.to_block_point(left_wrap_point), block_point);
|
|
assert_eq!(
|
|
blocks_snapshot.to_block_point(right_wrap_point),
|
|
block_point
|
|
);
|
|
}
|
|
|
|
let mut block_point = BlockPoint::new(0, 0);
|
|
for c in expected_text.chars() {
|
|
let left_point = blocks_snapshot.clip_point(block_point, Bias::Left);
|
|
let left_buffer_point = blocks_snapshot.to_point(left_point, Bias::Left);
|
|
assert_eq!(
|
|
blocks_snapshot
|
|
.to_block_point(blocks_snapshot.to_wrap_point(left_point, Bias::Left)),
|
|
left_point,
|
|
"block point: {:?}, wrap point: {:?}",
|
|
block_point,
|
|
blocks_snapshot.to_wrap_point(left_point, Bias::Left)
|
|
);
|
|
assert_eq!(
|
|
left_buffer_point,
|
|
buffer_snapshot.clip_point(left_buffer_point, Bias::Right),
|
|
"{:?} is not valid in buffer coordinates",
|
|
left_point
|
|
);
|
|
|
|
let right_point = blocks_snapshot.clip_point(block_point, Bias::Right);
|
|
let right_buffer_point = blocks_snapshot.to_point(right_point, Bias::Right);
|
|
assert_eq!(
|
|
blocks_snapshot
|
|
.to_block_point(blocks_snapshot.to_wrap_point(right_point, Bias::Right)),
|
|
right_point,
|
|
"block point: {:?}, wrap point: {:?}",
|
|
block_point,
|
|
blocks_snapshot.to_wrap_point(right_point, Bias::Right)
|
|
);
|
|
assert_eq!(
|
|
right_buffer_point,
|
|
buffer_snapshot.clip_point(right_buffer_point, Bias::Left),
|
|
"{:?} is not valid in buffer coordinates",
|
|
right_point
|
|
);
|
|
|
|
if c == '\n' {
|
|
block_point.0 += Point::new(1, 0);
|
|
} else {
|
|
block_point.column += c.len_utf8() as u32;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn init_test(cx: &mut gpui::AppContext) {
|
|
let settings = SettingsStore::test(cx);
|
|
cx.set_global(settings);
|
|
theme::init(theme::LoadThemes::JustBase, cx);
|
|
assets::Assets.load_test_fonts(cx);
|
|
}
|
|
|
|
impl Block {
|
|
fn as_custom(&self) -> Option<&CustomBlock> {
|
|
match self {
|
|
Block::Custom(block) => Some(block),
|
|
Block::ExcerptBoundary { .. } => None,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl BlockSnapshot {
|
|
fn to_point(&self, point: BlockPoint, bias: Bias) -> Point {
|
|
self.wrap_snapshot
|
|
.to_point(self.to_wrap_point(point, bias), bias)
|
|
}
|
|
}
|
|
}
|