इतिहास को कैसे पूरा करें?


17

मैं एक एमएसीएस मोड पर काम कर रहा हूं जो आपको वाक् पहचान के साथ एमएसीएस को नियंत्रित करने देता है। समस्याओं में से एक मैं भाग गया है कि जिस तरह से Emacs संभालती है वह मेल नहीं खाती है कि आप कैसे आवाज से नियंत्रित करते समय काम करने की उम्मीद करेंगे।

जब उपयोगकर्ता कई शब्द बोलता है और फिर रुक जाता है, तो उसे 'उच्चारण' कहा जाता है। Emacs को निष्पादित करने के लिए एक उच्चारण में कई कमांड शामिल हो सकते हैं। यह अक्सर ऐसा होता है कि पहचानकर्ता गलत तरीके से एक या अधिक आदेशों को पहचानता है। उस बिंदु पर मैं "पूर्ववत" कहने में सक्षम होना चाहता हूं और Emacs को उच्चारण द्वारा किए गए सभी कार्यों को करना है, न कि केवल कथन के भीतर अंतिम क्रिया। दूसरे शब्दों में, मैं चाहता हूं कि Emacs एक आदेश के रूप में एक ही व्यवहार के रूप में जहाँ तक पूर्ववत का संबंध है, तब भी जब एक उच्चारण में कई आदेश होते हैं। मैं यह भी कहना चाहूंगा कि जहां यह उच्चारण करने से पहले था, मैं ठीक उसी ओर जा रहा हूं, मैंने देखा है कि सामान्य Emacs ऐसा नहीं करता है।

मेरे पास प्रत्येक उच्चारण के आरंभ और अंत में कॉलबैक प्राप्त करने के लिए Emacs सेटअप है, इसलिए मैं स्थिति का पता लगा सकता हूं, मुझे केवल यह पता लगाने की आवश्यकता है कि Emacs क्या है। आदर्श रूप से मुझे कुछ पसंद आएगा (undo-start-collapsing)और फिर (undo-stop-collapsing)कुछ भी किया जाएगा और इनबेटवाइड जादुई रूप से एक रिकॉर्ड में ढह जाएगा।

मैंने दस्तावेज़ीकरण के माध्यम से कुछ काम किया और पाया undo-boundary, लेकिन यह इसके विपरीत है जो मैं चाहता हूं - मुझे एक बार में सभी कार्यों को एक पूर्व रिकॉर्ड में ध्वस्त करने की आवश्यकता है, न कि उन्हें विभाजित करना। मैं undo-boundaryयह सुनिश्चित करने के लिए कि आवृत्तियों को अलग माना जाता है, के बीच उपयोग कर सकते हैं (डिफ़ॉल्ट रूप से Emacs लगातार डालने की क्रियाओं को कुछ सीमा तक एक क्रिया मानते हैं), लेकिन यह बात है।

अन्य जटिलताओं:

  • मेरा भाषण मान्यता डेमॉन X11 कीपेस का अनुकरण करके Emacs को कुछ कमांड भेजता है और कुछ के माध्यम से भेजता emacsclient -eहै, अगर वहाँ कहते हैं कि (undo-collapse &rest ACTIONS)कोई केंद्रीय स्थान नहीं है जो मैं लपेट सकता हूं।
  • मैं उपयोग करता हूं undo-tree, यकीन नहीं तो इससे चीजें और जटिल हो जाएंगी। आदर्श रूप में एक समाधान undo-treeEmacs के सामान्य पूर्ववत व्यवहार के साथ काम करेगा ।
  • क्या होगा अगर एक उच्चारण के भीतर कोई कमांड "पूर्ववत करें" या "फिर से करें"? मुझे लगता है कि मैं कॉलबैक लॉजिक को हमेशा चीजों को सरल रखने के लिए अलग-अलग कथनों के रूप में Emacs में भेज सकता हूं, फिर इसे वैसे ही संभाला जाए जैसे कि मैं कीबोर्ड का उपयोग कर रहा था।
  • खिंचाव लक्ष्य: एक उच्चारण में एक कमांड हो सकती है जो वर्तमान में सक्रिय विंडो या बफर को स्विच करती है। इस मामले में प्रत्येक बफर में एक बार अलग से "पूर्ववत" कहने के लिए ठीक है, मुझे यह कल्पना करने की आवश्यकता नहीं है। लेकिन एक एकल बफ़र में सभी कमांड को अभी भी समूहीकृत किया जाना चाहिए, इसलिए यदि मैं कहता हूं "do-x do-y do-z switch-बफर do-a do-b do-c" तो x, y, z एक पूर्ववत होना चाहिए मूल बफर में रिकॉर्ड और ए, बी, सी को बफर में स्विच करने में एक रिकॉर्ड होना चाहिए।

क्या इसे करने का कोई आसान तरीका है? AFAICT में कुछ भी नहीं बनाया गया है, लेकिन Emacs विशाल और गहरा है ...

अद्यतन: मैंने थोड़ा अतिरिक्त कोड के साथ नीचे jhc के समाधान का उपयोग करके समाप्त किया। वैश्विक में before-change-hookमैं जाँच करता हूं कि क्या बफर को बदला जा रहा है, बफ़र्स की एक वैश्विक सूची में इस कथन को संशोधित किया गया है, यदि यह सूची में नहीं जाता है और undo-collapse-beginकहा जाता है। फिर उच्चारण के अंत में मैं सभी बफ़र्स को सूची में बुलाता हूं और कॉल करता हूं undo-collapse-end। नीचे कोड (नाम-नाम उद्देश्यों के लिए फ़ंक्शन नामों से पहले जोड़ा गया):

(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मार्कर के रूप में अपनी खुद की प्रविष्टियाँ डालने में सक्षम हो सकते हैं - शायद फॉर्म की प्रविष्टि (apply FUN-NAME . ARGS)? फिर एक बार फिर से उच्चारण करने के लिए undoजब तक आप अपना अगला मार्कर नहीं ढूंढ लेते । लेकिन मुझे संदेह है कि यहां सभी तरह की जटिलताएं हैं। :)
ग्लूकास

सीमाओं को हटाने से बेहतर दांव लगेगा।
JCH

यदि मैं पूर्ववत् वृक्ष का उपयोग कर रहा हूँ तो बफर-पूर्ववत सूची कार्य में हेरफेर करता है? मैं इसे पूर्व-वृक्ष स्रोत में संदर्भित देखता हूं, इसलिए मैं हां का अनुमान लगा रहा हूं, लेकिन पूरे मोड का अर्थ बनाना एक बड़ा प्रयास होगा।
जोसेफ गार्विन

@JosephGarvin मैं भाषण के साथ Emacs को नियंत्रित करने में रुचि रखता हूं। क्या आपके पास कोई स्रोत उपलब्ध है?
पायथन नॉट

@PythonNut: हाँ :) github.com/jgarvin/mandimus पैकेजिंग अधूरी है ... और कोड भी आंशिक रूप से मेरे joe- आदि रेपो में है: p लेकिन मैं इसे पूरे दिन उपयोग करता हूं और यह काम करता है।
जोसेफ गार्विन

जवाबों:


13

दिलचस्प रूप से पर्याप्त है, ऐसा करने के लिए कोई अंतर्निहित कार्य नहीं प्रतीत होता है।

निम्नलिखित कोड buffer-undo-listएक बंधनेवाला ब्लॉक की शुरुआत में एक अद्वितीय मार्कर डालने और nilएक ब्लॉक के अंत में सभी सीमाओं ( तत्वों) को हटाने के द्वारा काम करता है , फिर मार्कर को हटाता है। यदि कुछ गलत हो जाता है, तो मार्कर (apply identity nil)यह सुनिश्चित करने के लिए है कि यह पूर्ववत सूची में रहता है या नहीं।

आदर्श रूप से, आपको with-undo-collapseमैक्रो का उपयोग करना चाहिए , न कि अंतर्निहित कार्यों का। चूँकि आपने उल्लेख किया है कि आप रैपिंग नहीं कर सकते, इसलिए सुनिश्चित करें कि आप निम्न-स्तरीय फ़ंक्शंस मार्करों को पास करते हैं eq, जो कि हैं ही नहीं equal

यदि लागू कोड बफ़र को स्विच करता है, तो आपको यह सुनिश्चित करना होगा कि undo-collapse-endइसे उसी बफ़र में कहा जाए undo-collapse-begin। उस स्थिति में, प्रारंभिक बफर में केवल पूर्ववत प्रविष्टियां ही ढह जाएंगी।

(defun 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."
  (push marker buffer-undo-list))

(defun undo-collapse-end (marker)
  "Collapse undo history until a matching marker."
  (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 "undo-collapse-end with no matching marker"))
           ((null (cadr l))
            (setf (cdr l) (cddr l)))
           (t (setq l (cdr l)))))
       ;; remove the marker
       (setf (cdr l) (cddr l))))))

 (defmacro with-undo-collapse (&rest body)
  "Execute body, then collapse any resulting undo boundaries."
  (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
              (undo-collapse-begin ',marker)
              ,@body)
         (with-current-buffer ,buffer-var
           (undo-collapse-end ',marker))))))

यहाँ उपयोग का एक उदाहरण है:

(defun test-no-collapse ()
  (interactive)
  (insert "toto")
  (undo-boundary)
  (insert "titi"))

(defun test-collapse ()
  (interactive)
  (with-undo-collapse
    (insert "toto")
    (undo-boundary)
    (insert "titi")))

मैं समझता हूं कि आपका मार्कर एक ताजा सूची क्यों है, लेकिन क्या उन विशिष्ट तत्वों का कारण है?
मालाबार २०'१५

@ मलबारबा ऐसा इसलिए है क्योंकि एक प्रविष्टि (apply identity nil)कुछ भी नहीं करेगी यदि आप primitive-undoउस पर कॉल करते हैं - यह कुछ भी नहीं टूटेगा यदि किसी कारण से इसे सूची में छोड़ दिया जाता है।
jch

मेरे द्वारा जोड़े गए कोड को शामिल करने के लिए मेरे प्रश्न को अपडेट किया। धन्यवाद!
जोसेफ गार्विन

(eq (cadr l) nil)इसके बजाय करने का कोई कारण (null (cadr l))?
विचारमेन .42

@ ideasman42 आपके सुझाव के अनुसार संशोधित किया गया।
jch

3

पूर्ववत मशीनरी में कुछ परिवर्तन "हाल ही में" टूट गए कुछ हैक viper-modeइस तरह के टकराने के लिए उपयोग कर रहा था (जिज्ञासु के लिए, इसका उपयोग निम्नलिखित मामलों में किया जाता है: जब आप ESCप्रविष्टि / प्रतिस्थापन / संस्करण को समाप्त करने के लिए दबाते हैं , तो वाइपर पूरी तरह से ध्वस्त करना चाहता है। एक पूर्ववत चरण में बदलें)।

इसे सफाई से ठीक करने के लिए, हमने एक नया फ़ंक्शन undo-amalgamate-change-group(जो आपके लिए कम या ज्यादा संगत है undo-stop-collapsing) पेश किया और prepare-change-groupशुरुआत को चिह्नित करने के लिए मौजूदा का पुन: उपयोग करता है (यानी आपके लिए कम या ज्यादा संगत है undo-start-collapsing)।

संदर्भ के लिए, यहां एक नया वाइपर कोड दिया गया है:

(viper-deflocalvar viper--undo-change-group-handle nil)
(put 'viper--undo-change-group-handle 'permanent-local t)

(defun viper-adjust-undo ()
  (when viper--undo-change-group-handle
    (undo-amalgamate-change-group
     (prog1 viper--undo-change-group-handle
       (setq viper--undo-change-group-handle nil)))))

(defun viper-set-complex-command-for-undo ()
  (and (listp buffer-undo-list)
       (not viper--undo-change-group-handle)
       (setq viper--undo-change-group-handle
             (prepare-change-group))))

यह नया फ़ंक्शन Emacs-26 में दिखाई देगा, इसलिए यदि आप इसे माध्य समय में उपयोग करना चाहते हैं, तो आप इसकी परिभाषा को कॉपी कर सकते हैं (आवश्यकताएं cl-lib:

(defun undo-amalgamate-change-group (handle)
  "Amalgamate changes in change-group since HANDLE.
Remove all undo boundaries between the state of HANDLE and now.
HANDLE is as returned by `prepare-change-group'."
  (dolist (elt handle)
    (with-current-buffer (car elt)
      (setq elt (cdr elt))
      (when (consp buffer-undo-list)
        (let ((old-car (car-safe elt))
              (old-cdr (cdr-safe elt)))
          (unwind-protect
              (progn
                ;; Temporarily truncate the undo log at ELT.
                (when (consp elt)
                  (setcar elt t) (setcdr elt nil))
                (when
                    (or (null elt)        ;The undo-log was empty.
                        ;; `elt' is still in the log: normal case.
                        (eq elt (last buffer-undo-list))
                        ;; `elt' is not in the log any more, but that's because
                        ;; the log is "all new", so we should remove all
                        ;; boundaries from it.
                        (not (eq (last buffer-undo-list) (last old-cdr))))
                  (cl-callf (lambda (x) (delq nil x))
                      (if (car buffer-undo-list)
                          buffer-undo-list
                        ;; Preserve the undo-boundaries at either ends of the
                        ;; change-groups.
                        (cdr buffer-undo-list)))))
            ;; Reset the modified cons cell ELT to its original content.
            (when (consp elt)
              (setcar elt old-car)
              (setcdr elt old-cdr))))))))

मैंने देखा undo-amalgamate-change-group, और with-undo-collapseइस पृष्ठ पर परिभाषित मैक्रो की तरह इसका उपयोग करने के लिए एक सुविधाजनक तरीका प्रतीत नहीं होता है , क्योंकि atomic-change-groupयह उस तरीके से काम नहीं करता है जिसके साथ समूह को कॉल करने की अनुमति मिलती है undo-amalgamate-change-group
विचार .42

बेशक, आप इसका उपयोग नहीं करते हैं atomic-change-group: आप इसका उपयोग करते हैं prepare-change-group, जो आपके द्वारा किए गए हैंडल को तब वापस कर देता है undo-amalgamate-change-groupजब आप काम कर चुके होते हैं।
स्टीफन

एक मैक्रो नहीं होगा जो इस के साथ काम करता है उपयोगी होगा? (with-undo-amalgamate ...)जो परिवर्तन समूह सामग्री को संभालता है। अन्यथा यह कुछ संचालन को ध्वस्त करने के लिए थोड़ी परेशानी का कारण है।
विचारमान 42

अभी तक यह केवल वाइपर IIRC द्वारा उपयोग किया जाता है और वाइपर इस तरह के मैक्रो का उपयोग करने में सक्षम नहीं होगा क्योंकि दो कॉल अलग-अलग कमांड में होते हैं, इसलिए इसके लिए कोई रोने की आवश्यकता नहीं है। लेकिन इस तरह के मैक्रो को लिखना तुच्छ होगा, ज़ाहिर है।
स्टेफन

1
क्या इस मैक्रो को लिखा और emacs में शामिल किया जा सकता है? जबकि एक अनुभवी डेवलपर के लिए यह तुच्छ है, किसी ऐसे व्यक्ति के लिए जो अपने पूर्ववत इतिहास को ध्वस्त करना चाहता है और यह नहीं जानता कि कहां से शुरू करना है - यह कुछ समय ऑनलाइन गड़बड़ कर रहा है और इस धागे पर ठोकर खा रहा है ... तो यह पता लगाने के लिए कि कौन सा उत्तर सबसे अच्छा है - जब वे पर्याप्त अनुभव करने के लिए सक्षम नहीं हैं। मैंने यहाँ एक उत्तर जोड़ा: emacs.stackexchange.com/a/54412/2418
ideasman42

2

यहाँ एक with-undo-collapseमैक्रो है जो Emacs-26 परिवर्तन-समूह सुविधा का उपयोग करता है।

यह atomic-change-groupएक पंक्ति परिवर्तन के साथ है, जोड़ना undo-amalgamate-change-group

इसके फायदे हैं:

  • यह पूर्ववत डेटा को सीधे हेरफेर करने की आवश्यकता नहीं है।
  • यह सुनिश्चित करता है कि डेटा पूर्ववत नहीं किया गया है।
(defmacro with-undo-collapse (&rest body)
  "Like `progn' but perform BODY with undo collapsed."
  (declare (indent 0) (debug t))
  (let ((handle (make-symbol "--change-group-handle--"))
        (success (make-symbol "--change-group-success--")))
    `(let ((,handle (prepare-change-group))
            ;; Don't truncate any undo data in the middle of this.
            (undo-outer-limit nil)
            (undo-limit most-positive-fixnum)
            (undo-strong-limit most-positive-fixnum)
            (,success nil))
       (unwind-protect
         (progn
           (activate-change-group ,handle)
           (prog1 ,(macroexp-progn body)
             (setq ,success t)))
         (if ,success
           (progn
             (accept-change-group ,handle)
             (undo-amalgamate-change-group ,handle))
           (cancel-change-group ,handle))))))
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.