2012-01-31 23:12:06 +04:00
|
|
|
#include "selection.hh"
|
|
|
|
|
2012-10-08 16:26:57 +04:00
|
|
|
#include "utf8.hh"
|
2014-05-25 23:28:32 +04:00
|
|
|
#include "buffer_utils.hh"
|
2012-10-08 16:26:57 +04:00
|
|
|
|
2012-01-31 23:12:06 +04:00
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2014-03-29 12:55:45 +04:00
|
|
|
void Selection::merge_with(const Selection& range)
|
2012-11-30 21:32:49 +04:00
|
|
|
{
|
2014-01-28 23:05:49 +04:00
|
|
|
m_cursor = range.m_cursor;
|
|
|
|
if (m_anchor < m_cursor)
|
|
|
|
m_anchor = std::min(m_anchor, range.m_anchor);
|
|
|
|
if (m_anchor > m_cursor)
|
|
|
|
m_anchor = std::max(m_anchor, range.m_anchor);
|
2012-11-30 21:32:49 +04:00
|
|
|
}
|
|
|
|
|
2014-05-25 23:28:32 +04:00
|
|
|
SelectionList::SelectionList(Buffer& buffer, Selection s, size_t timestamp)
|
2014-05-13 02:25:15 +04:00
|
|
|
: m_buffer(&buffer), m_selections({ s }), m_timestamp(timestamp)
|
2014-05-14 02:22:54 +04:00
|
|
|
{
|
|
|
|
check_invariant();
|
|
|
|
}
|
2014-05-13 02:25:15 +04:00
|
|
|
|
2014-05-25 23:28:32 +04:00
|
|
|
SelectionList::SelectionList(Buffer& buffer, Selection s)
|
2014-05-13 02:25:15 +04:00
|
|
|
: SelectionList(buffer, s, buffer.timestamp())
|
|
|
|
{}
|
|
|
|
|
2014-05-25 23:28:32 +04:00
|
|
|
SelectionList::SelectionList(Buffer& buffer, std::vector<Selection> s, size_t timestamp)
|
2014-05-13 02:25:15 +04:00
|
|
|
: m_buffer(&buffer), m_selections(std::move(s)), m_timestamp(timestamp)
|
2014-05-14 02:22:54 +04:00
|
|
|
{
|
|
|
|
kak_assert(size() > 0);
|
|
|
|
check_invariant();
|
|
|
|
}
|
2014-05-13 02:25:15 +04:00
|
|
|
|
2014-05-25 23:28:32 +04:00
|
|
|
SelectionList::SelectionList(Buffer& buffer, std::vector<Selection> s)
|
2014-05-13 02:25:15 +04:00
|
|
|
: SelectionList(buffer, std::move(s), buffer.timestamp())
|
|
|
|
{}
|
|
|
|
|
2014-05-26 23:57:10 +04:00
|
|
|
namespace
|
2013-05-02 21:04:59 +04:00
|
|
|
{
|
2014-05-26 23:57:10 +04:00
|
|
|
|
|
|
|
ByteCoord update_insert(ByteCoord coord, ByteCoord begin, ByteCoord end)
|
|
|
|
{
|
|
|
|
if (coord < begin)
|
|
|
|
return coord;
|
|
|
|
if (begin.line == coord.line)
|
|
|
|
coord.column += end.column - begin.column;
|
|
|
|
coord.line += end.line - begin.line;
|
|
|
|
kak_assert(coord.line >= 0 and coord.column >= 0);
|
|
|
|
return coord;
|
|
|
|
}
|
|
|
|
|
|
|
|
ByteCoord update_erase(ByteCoord coord, ByteCoord begin, ByteCoord end)
|
|
|
|
{
|
|
|
|
if (coord < begin)
|
|
|
|
return coord;
|
|
|
|
if (coord <= end)
|
|
|
|
return begin;
|
|
|
|
if (end.line == coord.line)
|
|
|
|
coord.column -= end.column - begin.column;
|
|
|
|
coord.line -= end.line - begin.line;
|
|
|
|
kak_assert(coord.line >= 0 and coord.column >= 0);
|
|
|
|
return coord;
|
2013-05-02 21:04:59 +04:00
|
|
|
}
|
|
|
|
|
2014-06-02 18:13:56 +04:00
|
|
|
static bool compare_selections(const Selection& lhs, const Selection& rhs)
|
2013-05-02 21:04:59 +04:00
|
|
|
{
|
2014-06-02 18:13:56 +04:00
|
|
|
return lhs.min() < rhs.min();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename Iterator, typename OverlapsFunc>
|
|
|
|
Iterator merge_overlapping(Iterator begin, Iterator end, size_t& main, OverlapsFunc overlaps)
|
|
|
|
{
|
|
|
|
kak_assert(std::is_sorted(begin, end, compare_selections));
|
|
|
|
size_t size = end - begin;
|
|
|
|
size_t i = 0;
|
|
|
|
for (size_t j = 1; j < size; ++j)
|
|
|
|
{
|
|
|
|
if (overlaps(begin[i], begin[j]))
|
|
|
|
{
|
|
|
|
begin[i].merge_with(begin[j]);
|
|
|
|
if (i < main)
|
|
|
|
--main;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
++i;
|
|
|
|
if (i != j)
|
|
|
|
begin[i] = std::move(begin[j]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return begin + i + 1;
|
2014-05-26 23:57:10 +04:00
|
|
|
}
|
|
|
|
|
2014-05-29 08:48:40 +04:00
|
|
|
// This tracks position changes for changes that are done
|
|
|
|
// in a forward way (each change takes place at a position)
|
|
|
|
// *after* the previous one.
|
|
|
|
struct PositionChangesTracker
|
|
|
|
{
|
|
|
|
ByteCoord last_pos;
|
|
|
|
ByteCoord pos_change;
|
|
|
|
|
|
|
|
void update(const Buffer::Change& change)
|
|
|
|
{
|
|
|
|
if (change.type == Buffer::Change::Insert)
|
|
|
|
{
|
|
|
|
pos_change.line += change.end.line - change.begin.line;
|
|
|
|
if (change.begin.line != last_pos.line)
|
|
|
|
pos_change.column = 0;
|
|
|
|
pos_change.column += change.end.column - change.begin.column;
|
|
|
|
last_pos = change.end;
|
|
|
|
}
|
|
|
|
else if (change.type == Buffer::Change::Erase)
|
|
|
|
{
|
|
|
|
pos_change.line -= change.end.line - change.begin.line;
|
|
|
|
if (last_pos.line != change.end.line)
|
|
|
|
pos_change.column = 0;
|
|
|
|
pos_change.column -= change.end.column - change.begin.column;
|
|
|
|
last_pos = change.begin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void update(const Buffer& buffer, size_t& timestamp)
|
|
|
|
{
|
|
|
|
for (auto& change : buffer.changes_since(timestamp))
|
|
|
|
update(change);
|
|
|
|
timestamp = buffer.timestamp();
|
|
|
|
}
|
|
|
|
|
|
|
|
ByteCoord get_new_coord(ByteCoord coord)
|
|
|
|
{
|
|
|
|
if (last_pos.line - pos_change.line == coord.line)
|
2014-06-02 18:13:56 +04:00
|
|
|
coord.column = std::max(0_byte, coord.column + pos_change.column);
|
2014-05-29 08:48:40 +04:00
|
|
|
coord.line += pos_change.line;
|
|
|
|
return coord;
|
|
|
|
}
|
|
|
|
|
|
|
|
void update_sel(Selection& sel)
|
|
|
|
{
|
|
|
|
sel.anchor() = get_new_coord(sel.anchor());
|
|
|
|
sel.cursor() = get_new_coord(sel.cursor());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-06-02 05:16:19 +04:00
|
|
|
bool relevant(const Buffer::Change& change, ByteCoord coord)
|
|
|
|
{
|
|
|
|
return change.type == Buffer::Change::Insert ? change.begin <= coord
|
|
|
|
: change.begin < coord;
|
|
|
|
}
|
|
|
|
|
2014-06-02 18:13:56 +04:00
|
|
|
bool forward_sorted(const Buffer::Change& lhs, const Buffer::Change& rhs)
|
|
|
|
{
|
|
|
|
return lhs.begin < rhs.begin;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool backward_sorted(const Buffer::Change& lhs, const Buffer::Change& rhs)
|
|
|
|
{
|
|
|
|
return lhs.begin > rhs.end;
|
|
|
|
}
|
|
|
|
|
2014-06-02 05:16:19 +04:00
|
|
|
void update_forward(memoryview<Buffer::Change> changes, std::vector<Selection>& selections)
|
|
|
|
{
|
|
|
|
PositionChangesTracker changes_tracker;
|
|
|
|
|
|
|
|
auto change_it = changes.begin();
|
|
|
|
auto advance_while_relevant = [&](const ByteCoord& pos) mutable {
|
|
|
|
while (relevant(*change_it, changes_tracker.get_new_coord(pos)) and
|
|
|
|
change_it != changes.end())
|
|
|
|
changes_tracker.update(*change_it++);
|
|
|
|
};
|
|
|
|
|
|
|
|
for (auto& sel : selections)
|
|
|
|
{
|
|
|
|
auto& sel_min = sel.min();
|
|
|
|
auto& sel_max = sel.max();
|
|
|
|
advance_while_relevant(sel_min);
|
|
|
|
sel_min = changes_tracker.get_new_coord(sel_min);
|
|
|
|
|
|
|
|
advance_while_relevant(sel_max);
|
|
|
|
sel_max = changes_tracker.get_new_coord(sel_max);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void update_backward(memoryview<Buffer::Change> changes, std::vector<Selection>& selections)
|
|
|
|
{
|
|
|
|
PositionChangesTracker changes_tracker;
|
|
|
|
|
|
|
|
using ReverseIt = std::reverse_iterator<const Buffer::Change*>;
|
|
|
|
auto change_it = ReverseIt(changes.end());
|
|
|
|
auto change_end = ReverseIt(changes.begin());
|
|
|
|
auto advance_while_relevant = [&](const ByteCoord& pos) mutable {
|
|
|
|
while (change_it != change_end)
|
|
|
|
{
|
|
|
|
auto change = *change_it;
|
|
|
|
change.begin = changes_tracker.get_new_coord(change.begin);
|
|
|
|
change.end = changes_tracker.get_new_coord(change.end);
|
|
|
|
if (not relevant(change, changes_tracker.get_new_coord(pos)))
|
|
|
|
break;
|
|
|
|
changes_tracker.update(change);
|
|
|
|
++change_it;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
for (auto& sel : selections)
|
|
|
|
{
|
|
|
|
auto& sel_min = sel.min();
|
|
|
|
auto& sel_max = sel.max();
|
|
|
|
advance_while_relevant(sel_min);
|
|
|
|
sel_min = changes_tracker.get_new_coord(sel_min);
|
|
|
|
|
|
|
|
advance_while_relevant(sel_max);
|
|
|
|
sel_max = changes_tracker.get_new_coord(sel_max);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-02 21:04:59 +04:00
|
|
|
}
|
|
|
|
|
2014-06-02 18:13:56 +04:00
|
|
|
std::vector<Selection> compute_modified_ranges(Buffer& buffer, size_t timestamp)
|
|
|
|
{
|
|
|
|
std::vector<Selection> ranges;
|
|
|
|
auto changes = buffer.changes_since(timestamp);
|
|
|
|
auto change_it = changes.begin();
|
|
|
|
while (change_it != changes.end())
|
|
|
|
{
|
|
|
|
auto forward_end = std::is_sorted_until(change_it, changes.end(), forward_sorted);
|
|
|
|
auto backward_end = std::is_sorted_until(change_it, changes.end(), backward_sorted);
|
|
|
|
|
|
|
|
size_t prev_size = ranges.size();
|
|
|
|
if (forward_end >= backward_end)
|
|
|
|
{
|
|
|
|
update_forward({ change_it, forward_end }, ranges);
|
|
|
|
|
|
|
|
PositionChangesTracker changes_tracker;
|
|
|
|
for (; change_it != forward_end; ++change_it)
|
|
|
|
{
|
|
|
|
if (change_it->type == Buffer::Change::Insert)
|
|
|
|
ranges.push_back({ change_it->begin, change_it->end });
|
|
|
|
else
|
|
|
|
ranges.push_back({ change_it->begin });
|
|
|
|
changes_tracker.update(*change_it);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
update_backward({ change_it, backward_end }, ranges);
|
|
|
|
|
|
|
|
using ReverseIt = std::reverse_iterator<const Buffer::Change*>;
|
|
|
|
PositionChangesTracker changes_tracker;
|
|
|
|
for (ReverseIt it{backward_end}, end{change_it}; it != end; ++it)
|
|
|
|
{
|
|
|
|
if (it->type == Buffer::Change::Insert)
|
|
|
|
ranges.push_back({ changes_tracker.get_new_coord(it->begin),
|
|
|
|
changes_tracker.get_new_coord(it->end) });
|
|
|
|
else
|
|
|
|
ranges.push_back({ changes_tracker.get_new_coord(it->begin) });
|
|
|
|
changes_tracker.update(*it);
|
|
|
|
}
|
|
|
|
change_it = backward_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
kak_assert(std::is_sorted(ranges.begin(), ranges.begin() + prev_size, compare_selections));
|
|
|
|
kak_assert(std::is_sorted(ranges.begin() + prev_size, ranges.end(), compare_selections));
|
|
|
|
std::inplace_merge(ranges.begin(), ranges.begin() + prev_size, ranges.end(), compare_selections);
|
|
|
|
}
|
|
|
|
for (auto& sel : ranges)
|
|
|
|
{
|
|
|
|
sel.anchor() = buffer.clamp(sel.anchor());
|
|
|
|
sel.cursor() = buffer.clamp(sel.cursor());
|
|
|
|
}
|
|
|
|
|
|
|
|
auto touches = [&](const Selection& lhs, const Selection& rhs) {
|
|
|
|
return buffer.char_next(lhs.max()) >= rhs.min();
|
|
|
|
};
|
|
|
|
size_t dummy = 0;
|
|
|
|
ranges.erase(merge_overlapping(ranges.begin(), ranges.end(), dummy, touches), ranges.end());
|
|
|
|
|
|
|
|
for (auto& sel : ranges)
|
|
|
|
{
|
|
|
|
if (sel.anchor() != sel.cursor())
|
|
|
|
sel.cursor() = buffer.char_prev(sel.cursor());
|
|
|
|
}
|
|
|
|
return ranges;
|
|
|
|
}
|
|
|
|
|
2014-05-13 02:25:15 +04:00
|
|
|
void SelectionList::update()
|
|
|
|
{
|
2014-05-14 03:59:36 +04:00
|
|
|
if (m_timestamp == m_buffer->timestamp())
|
|
|
|
return;
|
|
|
|
|
2014-05-29 08:48:40 +04:00
|
|
|
auto changes = m_buffer->changes_since(m_timestamp);
|
|
|
|
auto change_it = changes.begin();
|
|
|
|
while (change_it != changes.end())
|
2014-05-26 23:57:10 +04:00
|
|
|
{
|
2014-06-02 18:13:56 +04:00
|
|
|
auto forward_end = std::is_sorted_until(change_it, changes.end(), forward_sorted);
|
|
|
|
auto backward_end = std::is_sorted_until(change_it, changes.end(), backward_sorted);
|
2014-05-29 08:48:40 +04:00
|
|
|
|
2014-06-02 05:16:19 +04:00
|
|
|
if (forward_end >= backward_end)
|
|
|
|
{
|
|
|
|
update_forward({ change_it, forward_end }, m_selections);
|
|
|
|
change_it = forward_end;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
update_backward({ change_it, backward_end }, m_selections);
|
|
|
|
change_it = backward_end;
|
2014-05-26 23:57:10 +04:00
|
|
|
}
|
2014-06-02 05:16:19 +04:00
|
|
|
kak_assert(std::is_sorted(m_selections.begin(), m_selections.end(),
|
|
|
|
compare_selections));
|
2014-05-26 23:57:10 +04:00
|
|
|
}
|
2014-05-20 23:33:38 +04:00
|
|
|
for (auto& sel : m_selections)
|
2014-05-13 02:25:15 +04:00
|
|
|
{
|
2014-05-26 23:57:10 +04:00
|
|
|
sel.anchor() = m_buffer->clamp(sel.anchor());
|
|
|
|
sel.cursor() = m_buffer->clamp(sel.cursor());
|
2014-05-13 02:25:15 +04:00
|
|
|
}
|
2014-06-01 18:57:12 +04:00
|
|
|
m_selections.erase(merge_overlapping(begin(), end(), m_main, overlaps), end());
|
2014-05-13 02:25:15 +04:00
|
|
|
check_invariant();
|
2014-05-14 03:59:36 +04:00
|
|
|
|
|
|
|
m_timestamp = m_buffer->timestamp();
|
2014-05-13 02:25:15 +04:00
|
|
|
}
|
|
|
|
|
2013-05-03 15:38:48 +04:00
|
|
|
void SelectionList::check_invariant() const
|
|
|
|
{
|
2014-05-13 02:25:15 +04:00
|
|
|
#ifdef KAK_DEBUG
|
|
|
|
auto& buffer = this->buffer();
|
2013-12-14 18:11:14 +04:00
|
|
|
kak_assert(size() > 0);
|
2013-12-13 03:17:06 +04:00
|
|
|
kak_assert(m_main < size());
|
2014-05-13 02:25:15 +04:00
|
|
|
for (size_t i = 0; i < size(); ++i)
|
|
|
|
{
|
|
|
|
auto& sel = (*this)[i];
|
2014-05-14 03:59:36 +04:00
|
|
|
if (i+1 < size())
|
|
|
|
kak_assert((*this)[i].min() <= (*this)[i+1].min());
|
2014-05-13 02:25:15 +04:00
|
|
|
kak_assert(buffer.is_valid(sel.anchor()));
|
|
|
|
kak_assert(buffer.is_valid(sel.cursor()));
|
|
|
|
kak_assert(not buffer.is_end(sel.anchor()));
|
|
|
|
kak_assert(not buffer.is_end(sel.cursor()));
|
2014-05-14 22:21:19 +04:00
|
|
|
kak_assert(utf8::is_character_start(buffer.byte_at(sel.anchor())));
|
|
|
|
kak_assert(utf8::is_character_start(buffer.byte_at(sel.cursor())));
|
2014-05-13 02:25:15 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void SelectionList::sort_and_merge_overlapping()
|
|
|
|
{
|
|
|
|
if (size() == 1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto& main = this->main();
|
|
|
|
const auto main_begin = main.min();
|
|
|
|
m_main = std::count_if(begin(), end(), [&](const Selection& sel) {
|
|
|
|
auto begin = sel.min();
|
|
|
|
if (begin == main_begin)
|
|
|
|
return &sel < &main;
|
|
|
|
else
|
|
|
|
return begin < main_begin;
|
|
|
|
});
|
|
|
|
std::stable_sort(begin(), end(), compare_selections);
|
2014-06-01 18:57:12 +04:00
|
|
|
m_selections.erase(merge_overlapping(begin(), end(), m_main, overlaps), end());
|
2013-05-03 15:38:48 +04:00
|
|
|
}
|
2014-05-14 23:56:27 +04:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
|
|
|
inline void _avoid_eol(const Buffer& buffer, ByteCoord& coord)
|
|
|
|
{
|
|
|
|
const auto column = coord.column;
|
|
|
|
const auto& line = buffer[coord.line];
|
|
|
|
if (column != 0 and column == line.length() - 1)
|
|
|
|
coord.column = line.byte_count_to(line.char_length() - 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline void _avoid_eol(const Buffer& buffer, Selection& sel)
|
|
|
|
{
|
|
|
|
_avoid_eol(buffer, sel.anchor());
|
|
|
|
_avoid_eol(buffer, sel.cursor());
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void SelectionList::avoid_eol()
|
|
|
|
{
|
|
|
|
update();
|
|
|
|
for (auto& sel : m_selections)
|
|
|
|
_avoid_eol(buffer(), sel);
|
|
|
|
}
|
2013-05-03 15:38:48 +04:00
|
|
|
|
2014-05-25 23:28:32 +04:00
|
|
|
BufferIterator prepare_insert(Buffer& buffer, const Selection& sel, InsertMode mode)
|
|
|
|
{
|
|
|
|
switch (mode)
|
|
|
|
{
|
|
|
|
case InsertMode::Insert:
|
|
|
|
return buffer.iterator_at(sel.min());
|
2014-05-26 01:59:29 +04:00
|
|
|
case InsertMode::InsertCursor:
|
|
|
|
return buffer.iterator_at(sel.cursor());
|
2014-05-25 23:28:32 +04:00
|
|
|
case InsertMode::Replace:
|
|
|
|
return erase(buffer, sel);
|
|
|
|
case InsertMode::Append:
|
|
|
|
{
|
|
|
|
// special case for end of lines, append to current line instead
|
|
|
|
auto pos = buffer.iterator_at(sel.max());
|
|
|
|
return *pos == '\n' ? pos : utf8::next(pos);
|
|
|
|
}
|
|
|
|
case InsertMode::InsertAtLineBegin:
|
|
|
|
return buffer.iterator_at(sel.min().line);
|
|
|
|
case InsertMode::AppendAtLineEnd:
|
|
|
|
return buffer.iterator_at({sel.max().line, buffer[sel.max().line].length() - 1});
|
|
|
|
case InsertMode::InsertAtNextLineBegin:
|
|
|
|
return buffer.iterator_at(sel.max().line+1);
|
|
|
|
case InsertMode::OpenLineBelow:
|
|
|
|
return buffer.insert(buffer.iterator_at(sel.max().line + 1), "\n");
|
|
|
|
case InsertMode::OpenLineAbove:
|
|
|
|
return buffer.insert(buffer.iterator_at(sel.min().line), "\n");
|
|
|
|
}
|
|
|
|
kak_assert(false);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
void SelectionList::insert(memoryview<String> strings, InsertMode mode)
|
|
|
|
{
|
|
|
|
if (strings.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
update();
|
2014-05-26 23:57:10 +04:00
|
|
|
PositionChangesTracker changes_tracker;
|
|
|
|
for (size_t index = 0; index < m_selections.size(); ++index)
|
2014-05-25 23:28:32 +04:00
|
|
|
{
|
|
|
|
auto& sel = m_selections[index];
|
2014-05-26 23:57:10 +04:00
|
|
|
|
|
|
|
changes_tracker.update_sel(sel);
|
|
|
|
kak_assert(m_buffer->is_valid(sel.anchor()));
|
|
|
|
kak_assert(m_buffer->is_valid(sel.cursor()));
|
|
|
|
|
2014-05-25 23:28:32 +04:00
|
|
|
auto pos = prepare_insert(*m_buffer, sel, mode);
|
2014-05-26 23:57:10 +04:00
|
|
|
changes_tracker.update(*m_buffer, m_timestamp);
|
|
|
|
|
2014-05-25 23:28:32 +04:00
|
|
|
const String& str = strings[std::min(index, strings.size()-1)];
|
|
|
|
pos = m_buffer->insert(pos, str);
|
2014-05-26 23:57:10 +04:00
|
|
|
|
|
|
|
auto& change = m_buffer->changes_since(m_timestamp).back();
|
|
|
|
changes_tracker.update(change);
|
|
|
|
m_timestamp = m_buffer->timestamp();
|
|
|
|
|
2014-05-25 23:28:32 +04:00
|
|
|
if (mode == InsertMode::Replace)
|
|
|
|
{
|
2014-05-26 23:57:10 +04:00
|
|
|
sel.anchor() = change.begin;
|
|
|
|
sel.cursor() = m_buffer->char_prev(change.end);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sel.anchor() = m_buffer->clamp(update_insert(sel.anchor(), change.begin, change.end));
|
|
|
|
sel.cursor() = m_buffer->clamp(update_insert(sel.cursor(), change.begin, change.end));
|
2014-05-25 23:28:32 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
check_invariant();
|
|
|
|
m_buffer->check_invariant();
|
|
|
|
}
|
|
|
|
|
2014-05-26 23:57:10 +04:00
|
|
|
void SelectionList::erase()
|
|
|
|
{
|
|
|
|
update();
|
|
|
|
PositionChangesTracker changes_tracker;
|
|
|
|
for (auto& sel : m_selections)
|
|
|
|
{
|
|
|
|
changes_tracker.update_sel(sel);
|
|
|
|
auto pos = Kakoune::erase(*m_buffer, sel);
|
|
|
|
sel.anchor() = sel.cursor() = m_buffer->clamp(pos.coord());
|
|
|
|
changes_tracker.update(*m_buffer, m_timestamp);
|
|
|
|
}
|
|
|
|
m_buffer->check_invariant();
|
|
|
|
}
|
|
|
|
|
2012-01-31 23:12:06 +04:00
|
|
|
}
|