diff options
author | mitchell <70453897+orbitalquark@users.noreply.github.com> | 2021-04-11 22:51:59 -0400 |
---|---|---|
committer | mitchell <70453897+orbitalquark@users.noreply.github.com> | 2021-04-11 22:51:59 -0400 |
commit | 1e693f06a6556b87ed4f56a3635a0c10640b1f92 (patch) | |
tree | 80c2fd9581119cd02fb54f558dfbe3634f231c8c /modules/textadept | |
parent | fe3d84c16252a65c117e26b873668ec967e96b4d (diff) |
Save/restore view state when undoing/redoing full-buffer changes.
For example external code formatting commands that replace buffer contents.
Diffstat (limited to 'modules/textadept')
-rw-r--r-- | modules/textadept/history.lua | 21 |
1 files changed, 18 insertions, 3 deletions
diff --git a/modules/textadept/history.lua b/modules/textadept/history.lua index da9f5885..05081ddd 100644 --- a/modules/textadept/history.lua +++ b/modules/textadept/history.lua @@ -33,20 +33,35 @@ local view_history = setmetatable({}, { end }) +local restore_position, first_visible_line = false, nil +-- Restore position after a full-buffer undo/redo operation, e.g. after replacing buffer contents +-- with a formatting command and then performing an undo. +events.connect(events.UPDATE_UI, function(updated) + if not restore_position or updated & buffer.UPDATE_SELECTION == 0 then return end + restore_position = false + M.back() + view.first_visible_line, first_visible_line = first_visible_line, nil +end) + -- Listens for text insertion and deletion events and records their locations. events.connect(events.MODIFIED, function(position, mod, text, length) local buffer = buffer -- Only interested in text insertion or deletion. if mod & buffer.MOD_INSERTTEXT > 0 then - if length == buffer.length then return end -- ignore file loading + if length == buffer.length then + if mod & buffer.MULTILINEUNDOREDO > 0 then restore_position = true end + return -- ignore file loading or replacing buffer contents + end position = position + length elseif mod & buffer.MOD_DELETETEXT > 0 then if buffer.length == 0 then return end -- ignore replacing buffer contents + elseif mod & (buffer.PERFORMED_UNDO | buffer.PERFORMED_REDO) > 0 and + (mod & buffer.MOD_BEFOREDELETE > 0) and length == buffer.length then + first_visible_line = view.first_visible_line -- save for potential undo before it's lost else return end - -- Ignore undo/redo. - if mod & (buffer.PERFORMED_UNDO | buffer.PERFORMED_REDO) > 0 then return end + if mod & (buffer.PERFORMED_UNDO | buffer.PERFORMED_REDO) > 0 then return end -- ignore undo/redo M.record(nil, buffer:line_from_position(position), buffer.column[position]) end) |