Pracuję nad trybem Emacsa, który pozwala kontrolować Emacsa za pomocą rozpoznawania mowy. Jednym z problemów, na jakie natknąłem się, jest to, że sposób, w jaki Emacs obsługuje cofanie, nie pasuje do tego, jak można oczekiwać od sterowania głosowego.
Gdy użytkownik wypowiada kilka słów, a następnie zatrzymuje się, nazywa się to „wypowiedzią”. Wypowiedź może składać się z wielu poleceń Emacsa do wykonania. Często zdarza się, że program rozpoznający nieprawidłowo rozpoznaje jedno lub więcej poleceń w wypowiedzi. W tym momencie chcę móc powiedzieć „cofnąć” i pozwolić Emacsowi cofnąć wszystkie działania wykonane przez wypowiedź, a nie tylko ostatnie działanie w wypowiedzi. Innymi słowy, chcę, aby Emacs traktował wypowiedź jako pojedyncze polecenie w zakresie cofania, nawet jeśli wypowiedź składa się z wielu poleceń. Chciałbym też wrócić do tego, co było przed wypowiedzią, zauważyłem, że normalne cofanie Emacsa tego nie robi.
Skonfigurowałem Emacsa, aby na początku i na końcu każdej wypowiedzi pojawiały się oddzwaniania, więc mogę wykryć sytuację, muszę tylko dowiedzieć się, co zrobić Emacs. Idealnie nazwałbym coś takiego, (undo-start-collapsing)
a wtedy (undo-stop-collapsing)
wszystko zrobione pomiędzy nimi byłoby magicznie zwinięte w jeden rekord.
Przeszukałem trochę dokumentacji i znalazłem undo-boundary
, ale jest to odwrotność tego, czego chcę - muszę zwinąć wszystkie działania w wypowiedzi w jeden rekord cofania, a nie dzielić je. Mogę używać undo-boundary
między wypowiedziami, aby upewnić się, że wstawienia są uważane za osobne (Emacs domyślnie uważa, że kolejne operacje wstawiania są jedną operacją do pewnego limitu), ale to wszystko.
Inne komplikacje:
- Mój demon rozpoznawania mowy wysyła niektóre polecenia do Emacsa, symulując naciśnięcia klawiszy X11, i przesyła je przez,
emacsclient -e
więc jeśli powiedzą, że(undo-collapse &rest ACTIONS)
nie ma centralnego miejsca, w którym mógłbym zawinąć. - Używam
undo-tree
, nie jestem pewien, czy to komplikuje sprawę. Idealnie byłoby, gdyby rozwiązanie działałoundo-tree
i normalne zachowanie cofania Emacsa. - Co jeśli jednym z poleceń w wypowiedzi jest „cofnij” lub „powtórz”? Myślę, że mógłbym zmienić logikę wywołania zwrotnego, aby zawsze wysyłać je do Emacsa jako odrębne wypowiedzi, aby uprościć sprawę, to powinno być obsługiwane tak samo, jak w przypadku korzystania z klawiatury.
- Cel rozciągnięcia: Wypowiedź może zawierać polecenie, które przełącza aktualnie aktywne okno lub bufor. W takim przypadku dobrze jest powiedzieć „cofnij” raz osobno w każdym buforze, nie muszę być tak fantazyjny. Ale wszystkie polecenia w jednym buforze powinny być nadal pogrupowane, więc jeśli powiem „do-x do-y do-z przełącznik-bufor do-a do-b do-c”, to x, y, z powinno być cofnięte rekord w oryginalnym buforze, a a, b, c powinny być jednym rekordem w przełączonym buforze.
Czy jest na to łatwy sposób? AFAICT nie ma nic wbudowanego, ale Emacs jest ogromny i głęboki ...
Aktualizacja: Skończyłem używać rozwiązania jhc poniżej z małym dodatkowym kodem. W globalnym before-change-hook
sprawdzam, czy zmieniany bufor znajduje się na globalnej liście buforów zmodyfikowanych w tej wypowiedzi, jeśli nie, to trafia na listę i undo-collapse-begin
jest wywoływany. Na koniec wypowiedzi iteruję wszystkie bufory na liście i wywołuję undo-collapse-end
. Kod poniżej (md - dodany przed nazwami funkcji do celów przestrzeni nazw):
(defvar md-utterance-changed-buffers nil)
(defvar-local md-collapse-undo-marker nil)
(defun md-undo-collapse-begin (marker)
"Mark the beginning of a collapsible undo block.
This must be followed with a call to undo-collapse-end with a marker
eq to this one.
Taken from jch's stackoverflow answer here:
http://emacs.stackexchange.com/a/7560/2301
"
(push marker buffer-undo-list))
(defun md-undo-collapse-end (marker)
"Collapse undo history until a matching marker.
Taken from jch's stackoverflow answer here:
http://emacs.stackexchange.com/a/7560/2301"
(cond
((eq (car buffer-undo-list) marker)
(setq buffer-undo-list (cdr buffer-undo-list)))
(t
(let ((l buffer-undo-list))
(while (not (eq (cadr l) marker))
(cond
((null (cdr l))
(error "md-undo-collapse-end with no matching marker"))
((eq (cadr l) nil)
(setf (cdr l) (cddr l)))
(t (setq l (cdr l)))))
;; remove the marker
(setf (cdr l) (cddr l))))))
(defmacro md-with-undo-collapse (&rest body)
"Execute body, then collapse any resulting undo boundaries.
Taken from jch's stackoverflow answer here:
http://emacs.stackexchange.com/a/7560/2301"
(declare (indent 0))
(let ((marker (list 'apply 'identity nil)) ; build a fresh list
(buffer-var (make-symbol "buffer")))
`(let ((,buffer-var (current-buffer)))
(unwind-protect
(progn
(md-undo-collapse-begin ',marker)
,@body)
(with-current-buffer ,buffer-var
(md-undo-collapse-end ',marker))))))
(defun md-check-undo-before-change (beg end)
"When a modification is detected, we push the current buffer
onto a list of buffers modified this utterance."
(unless (or
;; undo itself causes buffer modifications, we
;; don't want to trigger on those
undo-in-progress
;; we only collapse utterances, not general actions
(not md-in-utterance)
;; ignore undo disabled buffers
(eq buffer-undo-list t)
;; ignore read only buffers
buffer-read-only
;; ignore buffers we already marked
(memq (current-buffer) md-utterance-changed-buffers)
;; ignore buffers that have been killed
(not (buffer-name)))
(push (current-buffer) md-utterance-changed-buffers)
(setq md-collapse-undo-marker (list 'apply 'identity nil))
(undo-boundary)
(md-undo-collapse-begin md-collapse-undo-marker)))
(defun md-pre-utterance-undo-setup ()
(setq md-utterance-changed-buffers nil)
(setq md-collapse-undo-marker nil))
(defun md-post-utterance-collapse-undo ()
(unwind-protect
(dolist (i md-utterance-changed-buffers)
;; killed buffers have a name of nil, no point
;; in undoing those
(when (buffer-name i)
(with-current-buffer i
(condition-case nil
(md-undo-collapse-end md-collapse-undo-marker)
(error (message "Couldn't undo in buffer %S" i))))))
(setq md-utterance-changed-buffers nil)
(setq md-collapse-undo-marker nil)))
(defun md-force-collapse-undo ()
"Forces undo history to collapse, we invoke when the user is
trying to do an undo command so the undo itself is not collapsed."
(when (memq (current-buffer) md-utterance-changed-buffers)
(md-undo-collapse-end md-collapse-undo-marker)
(setq md-utterance-changed-buffers (delq (current-buffer) md-utterance-changed-buffers))))
(defun md-resume-collapse-after-undo ()
"After the 'undo' part of the utterance has passed, we still want to
collapse anything that comes after."
(when md-in-utterance
(md-check-undo-before-change nil nil)))
(defun md-enable-utterance-undo ()
(setq md-utterance-changed-buffers nil)
(when (featurep 'undo-tree)
(advice-add #'md-force-collapse-undo :before #'undo-tree-undo)
(advice-add #'md-resume-collapse-after-undo :after #'undo-tree-undo)
(advice-add #'md-force-collapse-undo :before #'undo-tree-redo)
(advice-add #'md-resume-collapse-after-undo :after #'undo-tree-redo))
(advice-add #'md-force-collapse-undo :before #'undo)
(advice-add #'md-resume-collapse-after-undo :after #'undo)
(add-hook 'before-change-functions #'md-check-undo-before-change)
(add-hook 'md-start-utterance-hooks #'md-pre-utterance-undo-setup)
(add-hook 'md-end-utterance-hooks #'md-post-utterance-collapse-undo))
(defun md-disable-utterance-undo ()
;;(md-force-collapse-undo)
(when (featurep 'undo-tree)
(advice-remove #'md-force-collapse-undo :before #'undo-tree-undo)
(advice-remove #'md-resume-collapse-after-undo :after #'undo-tree-undo)
(advice-remove #'md-force-collapse-undo :before #'undo-tree-redo)
(advice-remove #'md-resume-collapse-after-undo :after #'undo-tree-redo))
(advice-remove #'md-force-collapse-undo :before #'undo)
(advice-remove #'md-resume-collapse-after-undo :after #'undo)
(remove-hook 'before-change-functions #'md-check-undo-before-change)
(remove-hook 'md-start-utterance-hooks #'md-pre-utterance-undo-setup)
(remove-hook 'md-end-utterance-hooks #'md-post-utterance-collapse-undo))
(md-enable-utterance-undo)
;; (md-disable-utterance-undo)
buffer-undo-list
znacznika - być może wpis formularza(apply FUN-NAME . ARGS)
? Następnie, aby cofnąć wypowiedź, którą wielokrotnie wołasz,undo
aż znajdziesz następny znacznik. Ale podejrzewam, że są tutaj różnego rodzaju komplikacje. :)