Package: emacs;
Reported by: Jim Paris <jim <at> jtan.com>
Date: Thu, 29 Jul 2010 20:32:02 UTC
Severity: normal
Found in versions 23.2, 23.4
Done: Chong Yidong <cyd <at> gnu.org>
Bug is archived. No further changes may be made.
View this message in rfc822 format
From: Jim Paris <jim <at> jtan.com> To: Stefan Monnier <monnier <at> iro.umontreal.ca>, Johan Bockgård <bojohan <at> gnu.org> Cc: 6758 <at> debbugs.gnu.org Subject: bug#6758: 23.2; xterm.el: please provide an option to not discard input in terminal-init-xterm Date: Mon, 2 Aug 2010 16:59:59 -0400
Stefan Monnier wrote: > > I can also imagine a more complete fix that would involve not flushing > > the input buffer, and interpreting the Xterm responses in a more > > asynchronous fashion: > > - Don't discard input > > - Send the \e[>0c probe > > - Allow input into the buffer as usual, but for the next few seconds, > > interpret \e[>0;251;0c responses and perform > > (xterm-turn-on-modify-other-keys) etc. as necessary. > > Time-dependent processing is considered brittle (especially when > working remotely via something like SSH), so we prefer to avoid it. > Especially since coding it right could prove pretty darn tricky. Forget the time-dependent part of what I said, let's just send the query and handle the response whenever it happens to come in. See the below patch. This fixes everything -- it still detects both modifyOtherKeys and the background color, but doesn't require flushing input or rely on any timeouts at all, unlike the existing code. And there's nothing tricky. What do you think? > OTOH, I thought we had agreed that we need to add a configuration > variable xterm-turn-on-modify-other-keys which could be set to t or nil > to force the modifyOtherKeys feature of Xterm ON or OFF without first > checking whether the current xterm indeed supports it or not. > Its default value would be `auto', which would first check for support > and then turn it ON if applicable. This may still be useful for people who don't want any queries sent to their terminal. -jim --- xterm.el-orig 2010-04-03 18:26:04.000000000 -0400 +++ xterm.el 2010-08-02 16:51:45.000000000 -0400 @@ -440,6 +440,65 @@ ;; List of terminals for which modify-other-keys has been turned on. (defvar xterm-modify-other-keys-terminal-list nil) +(defun xterm-osc-translate (event) + "Read and handle a Operating System Controls response" + (let* ((str "") + (chr nil) + (recompute-faces nil)) + + ;; The reply should be of the form: \e ] 11 ; rgb: NUMBER1 / NUMBER2 / NUMBER3 \e \\ + (while (not (equal (setq chr (read-char)) ?\\)) + (setq str (concat str (string chr)))) + + (when (string-match "rgb:\\([a-f0-9]+\\)/\\([a-f0-9]+\\)/\\([a-f0-9]+\\)" str) + (setq recompute-faces + (xterm-maybe-set-dark-background-mode + (string-to-number (match-string 1 str) 16) + (string-to-number (match-string 2 str) 16) + (string-to-number (match-string 3 str) 16)))) + + (when recompute-faces + (tty-set-up-initial-frame-faces)) + + "")) + +(defun xterm-secondary-da-translate (event) + "Read and handle a Secondary Device Attributes response" + (let* ((str "") + (chr nil) + version) + + ;; The reply should be of the form: \e [ > NUMBER1 ; NUMBER2 ; NUMBER3 c + (while (not (equal (setq chr (read-char)) ?c)) + (setq str (concat str (string chr)))) + + (when (string-match "0;\\([0-9]+\\);0" str) + (setq version (string-to-number + (substring str (match-beginning 1) (match-end 1)))) + ;; NUMBER2 is the xterm version number, look for something + ;; greater than 216, the version when modifyOtherKeys was + ;; introduced. + (when (>= version 216) + ;; Make sure that the modifyOtherKeys state is restored when + ;; suspending, resuming and exiting. + (add-hook 'suspend-hook 'xterm-turn-off-modify-other-keys) + (add-hook 'suspend-resume-hook 'xterm-turn-on-modify-other-keys) + (add-hook 'kill-emacs-hook 'xterm-remove-modify-other-keys) + (add-hook 'delete-terminal-functions 'xterm-remove-modify-other-keys) + ;; Add the selected frame to the list of frames that + ;; need to deal with modify-other-keys. + (push (frame-terminal (selected-frame)) + xterm-modify-other-keys-terminal-list) + (xterm-turn-on-modify-other-keys)) + + ;; xterm version 235 supports reporting the background + ;; color, maybe earlier versions do too... + (when (>= version 235) + (define-key input-decode-map "\e]11;" 'xterm-osc-translate) + (send-string-to-terminal "\e]11;?\e\\"))) + + "")) + (defun terminal-init-xterm () "Terminal initialization function for xterm." ;; rxvt terminals sometimes set the TERM variable to "xterm", but @@ -469,71 +528,12 @@ ;; C-. C-, etc. ;; To do that we need to find out if the current terminal supports ;; modifyOtherKeys. At this time only xterm does. - (let ((coding-system-for-read 'binary) - (chr nil) - (str nil) - (recompute-faces nil) - version) - ;; Pending input can be mistakenly returned by the calls to - ;; read-event below. Discard it. - (discard-input) - ;; Try to find out the type of terminal by sending a "Secondary - ;; Device Attributes (DA)" query. - (send-string-to-terminal "\e[>0c") - - ;; The reply should be of the form: \e [ > NUMBER1 ; NUMBER2 ; NUMBER3 c - ;; If the timeout is completely removed for read-event, this - ;; might hang for terminals that pretend to be xterm, but don't - ;; respond to this escape sequence. RMS' opinion was to remove - ;; it completely. That might be right, but let's first try to - ;; see if by using a longer timeout we get rid of most issues. - (when (equal (read-event nil nil 2) ?\e) - (when (equal (read-event nil nil 2) ?\[) - (while (not (equal (setq chr (read-event nil nil 2)) ?c)) - (setq str (concat str (string chr)))) - (when (string-match ">0;\\([0-9]+\\);0" str) - (setq version (string-to-number - (substring str (match-beginning 1) (match-end 1)))) - ;; xterm version 242 supports reporting the background - ;; color, maybe earlier versions do too... - (when (>= version 242) - (send-string-to-terminal "\e]11;?\e\\") - (when (equal (read-event nil nil 2) ?\e) - (when (equal (read-event nil nil 2) ?\]) - (setq str "") - (while (not (equal (setq chr (read-event nil nil 2)) ?\\)) - (setq str (concat str (string chr)))) - (when (string-match "11;rgb:\\([a-f0-9]+\\)/\\([a-f0-9]+\\)/\\([a-f0-9]+\\)" str) - (setq recompute-faces - (xterm-maybe-set-dark-background-mode - (string-to-number (match-string 1 str) 16) - (string-to-number (match-string 2 str) 16) - (string-to-number (match-string 3 str) 16))))))) - ;; NUMBER2 is the xterm version number, look for something - ;; greater than 216, the version when modifyOtherKeys was - ;; introduced. - (when (>= version 216) - ;; Make sure that the modifyOtherKeys state is restored when - ;; suspending, resuming and exiting. - (add-hook 'suspend-hook 'xterm-turn-off-modify-other-keys) - (add-hook 'suspend-resume-hook 'xterm-turn-on-modify-other-keys) - (add-hook 'kill-emacs-hook 'xterm-remove-modify-other-keys) - (add-hook 'delete-terminal-functions 'xterm-remove-modify-other-keys) - ;; Add the selected frame to the list of frames that - ;; need to deal with modify-other-keys. - (push (frame-terminal (selected-frame)) - xterm-modify-other-keys-terminal-list) - (xterm-turn-on-modify-other-keys)) - - ;; Recompute faces here in case the background mode was - ;; set to dark. We used to call - ;; `tty-set-up-initial-frame-faces' only once, but that - ;; caused the light background faces to be computed - ;; incorrectly. See: - ;; http://permalink.gmane.org/gmane.emacs.devel/119627 - (when recompute-faces - (tty-set-up-initial-frame-faces)))))) + ;; Try to find out the type of terminal by sending a "Secondary + ;; Device Attributes (DA)" query. + (define-key input-decode-map "\e[>" 'xterm-secondary-da-translate) + (send-string-to-terminal "\e[>0c") + (run-hooks 'terminal-init-xterm-hook)) ;; Set up colors, for those versions of xterm that support it.
GNU bug tracking system
Copyright (C) 1999 Darren O. Benham,
1997,2003 nCipher Corporation Ltd,
1994-97 Ian Jackson.