unofficial mirror of emacs-devel@gnu.org 
 help / color / mirror / code / Atom feed
From: Stefan Monnier <monnier@iro.umontreal.ca>
To: "Lennart Borgman \(gmail\)" <lennart.borgman@gmail.com>
Cc: emacs-pretest-bug@gnu.org
Subject: Re: 23.0.60; Crash on w32 related to invisible text
Date: Sun, 23 Mar 2008 20:30:25 -0400	[thread overview]
Message-ID: <jwvtzix6k87.fsf-monnier+emacsbugreports@gnu.org> (raw)
In-Reply-To: <47E6F26B.6000507@gmail.com> (Lennart Borgman's message of "Mon,  24 Mar 2008 01:14:35 +0100")

>>> I get crashes in org-mode when I do things in a region with invisible
>>> text. I have not made any simple test case, but from my testing I can see
>>> this:
>> 
>>> - There is something strange with the display in org-mode an items content
>>> is hidden. You can position point after the three dots ... If you do that
>>> and check (current-column) you get 0.
>> 
>>> - In code if you in org-mode do
>> 
>>> (while (my-invisible-p (point))
>>> (forward-line 1))
>> 
>>> until point is on a visible char and there insert an overlay with an
>>> after-string it crashes.
>> 
>> Please make the "insert an overlay with an
>> after-string" more precise, e.g. actual Elisp code.

> It looks similar to this, I have not tested if propertize is involved:

>   (setq ovl-str (concat "before"
>                       (propertize "txt" 'face 'highlight-popup)
>                       "after"))
>   (overlay-put ovl 'after-string ovl-str)
>   (overlay-put ovl 'display "")

> See also the attached file. The crash happens in tabkey2-overlay-message if
> you let it take the path avoided right under the comment:

>       ;; Fix-me: Emacs bug workaround
>       (if nil ;;(tabkey2-invisible-p (1- (point)))


>> Also, does it happen with any Org-mode buffer, or is the
>> contents/setup important?

> It does not seem to be dependent on the buffers content.

>> Finally, please provide a backtrace.

> Sorry, it is compiled with debug info, but there is no backtrace.

You need to run it under a debugger, of course, in order to get
a backtrace.  Please read etc/DEBUG for more info about it.

I'm sorry, but you presume I'm much more intelligent than I am.
I do not understand what tabkey2.el has to do with it all and how to
create an Org-mode buffer that will trigger the crash.

So, please give us a recipe starting from `emacs -Q' with *every single
step* necessary to trigger the crash.  Including a small sample
Org-mode file.  It's always safer to assume we're complete idiots.


        Stefan


>> Stefan

> PS: The attached file implements a way to handle Tab for completion.

> ;;; tabkey2.el --- Use second tab key pressed for what you want
> ;;
> ;; Author: Lennart Borgman (lennart O borgman A gmail O com)
> ;; Created: 2008-03-15T14:40:28+0100 Sat
> (defconst tabkey2:version "1.17")
> ;; Last-Updated: 2008-03-21T20:21:34+0100 Fri
> ;; URL: http://www.emacswiki.org/cgi-bin/wiki/tabkey2.el
> ;; Keywords:
> ;; Compatibility:
> ;;
> ;; Features that might be required by this library:
> ;;
> ;;   `appmenu', `cl', `ourcomments-util', `popcmp'.
> ;;
> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;
> ;;; Commentary:
> ;;
> ;; The tab key is in Emacs often used for indentation.  However if you
> ;; press the tab key a second time and Emacs tries to do indentation
> ;; again, then usually nothing exciting will happen.  Then why not use
> ;; second tab key in a row for something else?
> ;;
> ;; Commonly used completion functions in Emacs is often bound to
> ;; something corresponding to Alt-Tab.  Unfortunately this is unusable
> ;; if you have a window manager that have an apetite for it (like that
> ;; on MS Windows for example, and several on GNU/Linux).
> ;;
> ;; Then using the second tab key press for completion might be a good
> ;; choice and perhaps also easy to remember.
> ;;
> ;; This little library tries to make it easy to do use the second tab
> ;; press for completion.  Or you can see this library as a swizz army
> ;; knife for the tab key ;-)
> ;;
> ;; See `tabkey2-mode' for more information.
> ;;
> ;;
> ;; This is a generalization of an idea Sebastien Rocca Serra once
> ;; presented on Emacs Wiki and called "Smart Tab".  (It seems like
> ;; many others have also been using Tab for completion in one way or
> ;; another for years.)
> ;;
> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;
> ;;; Change log:
> ;;
> ;; Version 1.04:
> ;; - Add overlay to display state after first tab.
> ;;
> ;; Version 1.05:
> ;; - Fix remove overlay problem.
> ;;
> ;; Version 1.06:
> ;; - Add completion function choice.
> ;; - Add support for popcmp popup completion.
> ;;
> ;; Version 1.07:
> ;; - Add informational message after first tab.
> ;;
> ;; Version 1.08:
> ;; - Give better informational message after first tab.
> ;;
> ;; Version 1.09:
> ;; - Put flyspell first.
> ;;
> ;; Version 1.09:
> ;; - Give the overlay higher priority.
> ;;
> ;; Version 1.10:
> ;; - Correct tabkey2-completion-functions.
> ;; - Add double-tab for modes where tab can not be typed again.
> ;; - Use better condition for when completion can be done, so that it
> ;;   can be done later while still on the same line.
> ;; - Add a better message handling for the "Tab completion state".
> ;; - Add C-g break out of the "Tab completion state".
> ;; - Add faces for highlight.
> ;; - Make it work in custom mode buffers.
> ;; - Fix documentation for `tabkey2-first'
> ;;
> ;; Version 1.11:
> ;; - Don't call chosen completion function directly.  Instead make it
> ;;   default for current buffer.
> ;;
> ;; Version 1.12:
> ;; - Simplify code.
> ;; - Add help to C-f1 during "Tab completion state".
> ;; - Fix documentation basics.
> ;; - Add customization of state message and line marking.
> ;; - Fix handling of double-Tab modes.
> ;; - Make user interaction better.
> ;; - Handle read-only in custom buffers better.
> ;; - Add more flexible check for if completion function is active.
> ;; - Support predictive mode.
> ;; - Reorder and simplify.
> ;;
> ;; Version 1.13:
> ;; - Add org-mode to the double-tab gang.
> ;; - Make it possible to use double-tab in normal buffers.
> ;; - Add cycling through completion functions to S-tab.
> ;;
> ;; Version 1.14:
> ;; - Fix bug in handling of read-only.
> ;; - Show completion binding in help message.
> ;; - Add binding to make current choice buffer local when cycling.
> ;;
> ;; Version 1.15:
> ;; - Fix problem at buffer end.
> ;; - Add S-tab to enter completion state without indentation.
> ;; - Add backtab bindings too for this.
> ;; - Remove double-tab, S-tab is better.
> ;; - Add list of modes that uses more tabs.
> ;; - Add list of modes that uses tab only for completion.
> ;; - Move first overlay when indentation changes.
> ;; - Make mark at line beginning 1 char long.
> ;;
> ;; Version 1.16:
> ;; - Don't call tab function when alternate key is pressed.
> ;;
> ;; Version 1.17:
> ;; - Let alternate key cycle completion functions instead of complete.
> ;; - Bind backtab.
> ;; - Fix bug when only one completion funciton was available.
> ;; - Fix bug when alt key and major without fix indent.
> ;;
> ;; Version 1.18:
> ;; - Add popup style messages.
> ;; - Add delay to first message.
> ;; - Use different face for indicator on line and message.
> ;; - Use different face for echo area and popup messages.
> ;; - Add anything to completion functions.
> ;; - Put help funciton on f1.
> ;;
> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;
> ;;; Known bugs
> ;;
> ;; - Problems with comint shell.
> ;; - Does not check visibility.
> ;;
> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;
> ;; This program is free software; you can redistribute it and/or
> ;; modify it under the terms of the GNU General Public License as
> ;; published by the Free Software Foundation; either version 2, or
> ;; (at your option) any later version.
> ;;
> ;; This program is distributed in the hope that it will be useful,
> ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
> ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
> ;; General Public License for more details.
> ;;
> ;; You should have received a copy of the GNU General Public License
> ;; along with this program; see the file COPYING.  If not, write to
> ;; the Free Software Foundation, Inc., 51 Franklin Street, Fifth
> ;; Floor, Boston, MA 02110-1301, USA.
> ;;
> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;
> ;;; Code:

> (require 'popcmp nil t)
> (require 'appmenu nil t)

> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;;; Custom

> (defgroup tabkey2 nil
>   "Customization of second tab key press."
>   :group 'nxhtml
>   :group 'convenience)

> (defface tabkey2-highlight-line
>   '((t :inherit highlight))
>   "Face for marker on current line."
>   :group 'tabkey2)

> (defface tabkey2-highlight-message
>   '((t :inherit tabkey2-highlight-line))
>   "Face for messages in echo area."
>   :group 'tabkey2)

> (defface tabkey2-highlight-popup
>   '((default :box t :inherit tabkey2-highlight-message)
>     (((class color) (background light)) :foreground "black")
>     (((class color) (background dark)) :foreground "yellow"))
>   "Face for popup messages."
>   :group 'tabkey2)

> (defcustom tabkey2-show-message 1
>   "Show message after Tab if non-nil."
>   :type '(radio
>           (const :tag "Always" t)
>           (const :tag "First time on row only" 1)
>           (const :tag "Never" nil))
>   :group 'tabkey2)

> (defcustom tabkey2-show-mark-on-active-line t
>   "Show mark on active line if non-nil.
> This mark is shown during 'Tab completion state'."
>   :type 'boolean
>   :group 'tabkey2)

> ;;(setq tabkey2-delay-first-message nil)
> (defcustom tabkey2-delay-first-message 1.0
>   "Number of seconds to delay message after Tab.
> If nil do not delay."
>   :type '(choice (const :tag "Don't delay" nil)
>                  (float :tag "Seconds to delay"))
>   :group 'tabkey2)

> ;; (setq tabkey2-message-style 'popup)
> ;; (setq tabkey2-message-style 'echo-area)
> (defcustom tabkey2-message-style 'popup
>   "How to show messages."
>   :type '(choice (const :tag "Popup" 'popup)
>                  (const :tag "Echo area" 'echo-area))
>   :group 'tabkey2)

> (defcustom tabkey2-in-minibuffer nil
>   "If non-nil use command `tabkey2-mode' also in minibuffer."
>   :type 'boolean
>   :group 'tabkey2)

> (defcustom tabkey2-in-appmenu t
>   "Show a completion menu in command `appmenu-mode' if t."
>   :type 'boolean
>   :set (lambda (sym val)
>          (set-default sym val)
>          (when (featurep 'appmenu)
>            (if val
>                (appmenu-add 'tabkey2 nil t "Completion" 'tabkey2-appmenu)
>              (appmenu-remove 'tabkey2))))
>   :group 'tabkey2)

> (defcustom tabkey2-completion-functions
>   '(
>     ("Spell check word" flyspell-correct-word-before-point)
>     ("Predictive word" complete-word-at-point predictive-mode)
>     ("XML completion" nxml-complete)
>     ("Complete Emacs symbol" lisp-complete-symbol)
>     ("Widget complete" widget-complete)
>     ("Comint Dynamic Complete" comint-dynamic-complete)
>     ("PHP completion" php-complete-function)
>     ("Tags completion" complete-symbol)
>     ("Dynamic word expansion" dabbrev-expand)
>     ("Ispell complete word" ispell-complete-word)
>     ("Anything" anything '(fboundp 'anything))
>     )
>   "List of completion functions.
> These are used by `tabkey2-choose-completion-function' in the
> order they are found here.

> The first 'active' entry in this list is normally used during the
> 'Tab completion state'.  An entry in the list is considered
> active if:

> -  The symbol is bound to a function

> and

> - it has a key binding at point,

>   or

>   the elisp expression evaluates to non-nil.
>   If it is a single symbol then its variable value is used,
>   otherwise the elisp form is evaled."
>   :type '(repeat (list string (choice (command :tag "Currently know command")
>                                       (symbol  :tag "Command not known yet"))
>                        (sexp :tag "Elisp, if evaluate to non-nil then active")))
>   :group 'tabkey2)

> (defcustom tabkey2-alternate-key [f8]
>   "Alternate key, enter state, show/cycle completion functions."
>   :set (lambda (sym val)
>          (when (boundp 'tabkey2-mode-emul-map)
>            (define-key tabkey2-mode-emul-map
>              tabkey2-alternate-key nil)
>            (define-key tabkey2-completion-state-emul-map
>              tabkey2-alternate-key nil)
>            (define-key tabkey2-mode-emul-map val 'tabkey2-first)
>            (define-key tabkey2-completion-state-emul-map val 'tabkey2-cycle-through-completion-functions)
>            (when tabkey2-completion-state-mode
>              (tabkey2-completion-state-mode -1)
>              (tabkey2-completion-state-mode 1)))
>          (set-default sym val))
>   :type 'key-sequence
>   :group 'tabkey2)

> (defcustom tabkey2-modes-that-uses-more-tabs
>   '(python-mode
>     haskell-mode
>     makefile-mode
>     org-mode
>     Custom-mode
>     ;; other
>     cmd-mode
>     )
>   "In those modes use must use S-Tab to start completion state.
> In those modes pressing Tab several types may make sense so you
> can not go into´'Tab completion state' just because one Tab has
> been pressed.  Instead you use S-Tab to go into that state.
> After that Tab does completion.

> You can do use S-Tab in other modes too if you want too."
>   :type '(repeat command)
>   :group 'tabkey2)

> (defcustom tabkey2-modes-that-just-completes
>   ;; Fix-me: Sometimes S-tab does not work here, why?
>   '(shell-mode)
>   "Tab is only used for completion in these modes.
> Therefore `tabkey2-first' just calls the function on Tab."
>   :type '(repeat command)
>   :group 'tabkey2)

> ;;(setq tabkey2-use-popup-menus nil)
> (defcustom tabkey2-use-popup-menus (when (featurep 'popcmp) t)
>   "Use pop menus if available."
>   :type 'boolean
>   :group 'tabkey2)

> (defvar tabkey2-preferred nil
>   "Preferred function for second tab key press.")
> (make-variable-buffer-local 'tabkey2-preferred)
> (put 'tabkey2-preferred 'permanent-local t)

> (defvar tabkey2-fallback nil
>   "Fallback function for second tab key press.")
> (make-variable-buffer-local 'tabkey2-fallback)
> (put 'tabkey2-fallback 'permanent-local t)


> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;;; State

> (defvar tabkey2-overlay nil
>   "Show when tab key 2 action is to be done.")
> (defvar tabkey2-keymap-overlay nil
>   "Hold the keymap for tab key 2.")
> (defvar tabkey2-beg-mark nil)
> (defvar tabkey2-end-mark nil)

> ;; (setq tabkey2-keymap-overlay nil)
> (defconst tabkey2-completion-state-emul-map
>   (let ((map (make-sparse-keymap)))
>     ;;(define-key map [(control shift tab)] 'tabkey2-choose-completion-function)
>     ;;(define-key map [(control backtab)]   'tabkey2-choose-completion-function)
>     (define-key map [(control ?c) tab]    'tabkey2-make-current-sticky)

>     (define-key map tabkey2-alternate-key 'tabkey2-cycle-through-completion-functions)
>     (define-key map [backtab]             'tabkey2-cycle-through-completion-functions)
>     (define-key map [(shift tab)]         'tabkey2-cycle-through-completion-functions)

>     (define-key map [(control f1)]        'tabkey2-completion-function-help)
>     (define-key map [(meta f1)]           'tabkey2-show-completion-functions)
>     (define-key map [f1]                  'tabkey2-completion-state-help)

>     (define-key map [(control ?g)]        'tabkey2-completion-state-off)
>     (define-key map [tab]                 'tabkey2-complete)
>     map)
>   "This keymap is for `tabkey2-keymap-overlay'.")

> (defun tabkey2-completion-state-p ()
>   "Return t if Tab completion state should continue.
> Otherwise return nil."
>   (and (or (eobp)
>            (eq (key-binding [tab]) 'tabkey2-complete))
>        (= (line-beginning-position)
>           (overlay-start tabkey2-keymap-overlay))))

> (defvar tabkey2-current-tab-info nil
>   "Saved information message for Tab completion state.")
> (defvar tabkey2-current-tab-function nil
>   "Tab completion state current completion function.")

> (defun tabkey2-read-only-p ()
>   "Return non-nil if buffer seems to be read-only at point."
>   (or buffer-read-only
>       (get-char-property (min (1+ (point)) (point-max)) 'read-only)
>       (let ((remap (command-remapping 'self-insert-command (point))))
>         (memq remap '(Custom-no-edit)))))

> ;;;; Minor mode active after first tab

> (defun tabkey2-move-overlays ()
>   ;; The marking overlay
>   (let* ((beg (let ((inhibit-field-text-motion t))
>                 (line-beginning-position)))
>          (ind (current-indentation))
>          (end (+ beg 1)) ;(if (> ind 0) ind 1)))
>          (inhibit-read-only t))
>     ;;(message "beg=%s, end=%s" beg end) (sit-for 2)
>     (unless tabkey2-overlay
>       (setq tabkey2-overlay (make-overlay beg end)))
>     ;; Fix-me: gets some strange errors, try avoid moving:
>     (unless (and (eq (current-buffer) (overlay-buffer tabkey2-overlay))
>                  (= beg (overlay-start tabkey2-overlay))
>                  (= end (overlay-end   tabkey2-overlay)))
>       (move-overlay tabkey2-overlay beg end (current-buffer)))
>     ;; Give it a high priority, it is very temporary
>     (overlay-put tabkey2-overlay 'priority 1000)
>     (if tabkey2-show-mark-on-active-line
>         (progn
>           (overlay-put tabkey2-overlay 'face 'tabkey2-highlight-line)
>           (overlay-put tabkey2-overlay 'help-echo
>                        "This highlight shows that Tab completion state is on"))
>       (overlay-put tabkey2-overlay 'face nil)
>       (overlay-put tabkey2-overlay 'help-echo nil)))
>   ;; The keymap overlay
>   (let ((beg (line-beginning-position))
>         (end (line-end-position)))
>     (setq tabkey2-beg-mark (copy-marker beg nil))
>     (setq tabkey2-end-mark (copy-marker (1+ end) t)))
>   (assert (or (eobp)
>               (= tabkey2-beg-mark
>                  (save-excursion
>                    (goto-char (1- tabkey2-end-mark))
>                    (line-beginning-position)))))
>   (unless tabkey2-keymap-overlay
>     (setq tabkey2-keymap-overlay (make-overlay 1 1)))
>   (overlay-put tabkey2-keymap-overlay 'priority 1000)
>   (overlay-put tabkey2-keymap-overlay 'keymap
>                tabkey2-completion-state-emul-map)
>   (move-overlay tabkey2-keymap-overlay tabkey2-beg-mark tabkey2-end-mark (current-buffer)))

> (defvar tabkey2-message-countdown nil)

> (defvar tabkey2-completion-state-mode nil)
> (defun tabkey2-completion-state-mode (arg)
>   "Tab completion state minor mode.
> This pseudo-minor mode holds the 'Tab completion state'.  When this
> minor mode is on completion key bindings are available.

> With ARG a positive number turn on, otherwise turn off this minor
> mode.

> See `tabkey2-first' for more information."
>   (unless (assoc 'tabkey2-completion-state-mode minor-mode-alist)
>     (setq minor-mode-alist (cons '(tabkey2-completion-state-mode " Tab2")
>                                  minor-mode-alist)))
>   (setq tabkey2-completion-state-mode (when (and (numberp arg)
>                                                  (> arg 0))
>                                         t))
>   (if tabkey2-completion-state-mode
>       (progn
>         ;; Move overlays
>         (tabkey2-move-overlays)
>         ;; Set up for post-command-hook
>         (add-hook 'post-command-hook 'tabkey2-post-command)
>         (setq tabkey2-message-countdown tabkey2-show-message))
>     (let ((inhibit-read-only t))
>       (when tabkey2-keymap-overlay
>         (delete-overlay tabkey2-keymap-overlay))
>       (when tabkey2-overlay
>         (delete-overlay tabkey2-overlay)))
>     (remove-hook 'post-command-hook 'tabkey2-post-command)
>     (message "")))

> (defun tabkey2-completion-state-off ()
>   "Quit Tab completion state."
>   (interactive)
>   (tabkey2-completion-state-mode -1)
>   (message "Quit"))

> (defvar tabkey2-is-cycling nil)

> (defun tabkey2-post-command ()
>   "Turn off Tab completion state if not feasable any more.
> This is run in `post-command-hook' after each command."
>   (condition-case err
>       (save-match-data
>         (tabkey2-cancel-delayed-message)
>         ;;(message "was first=%s" tabkey2-was-first-command) (sit-for 2)
>         (setq tabkey2-delay-next-message tabkey2-was-first-command)
>         (setq tabkey2-was-first-command nil)
>         (unless (or (eq this-command 'tabkey2)
>                     (= 0 (length (this-command-keys-vector)))
>                     (active-minibuffer-window))
>           (if (not (tabkey2-completion-state-p))
>               (tabkey2-completion-state-mode -1)
>             (tabkey2-move-overlays)
>             (when (eq tabkey2-is-cycling 'tentative) (setq tabkey2-is-cycling nil))
>             (when (and tabkey2-message-countdown
>                        (not tabkey2-is-cycling))
>               (when (integerp tabkey2-message-countdown)
>                 (setq tabkey2-message-countdown (1- tabkey2-message-countdown))
>                 (when (> 0 tabkey2-message-countdown)
>                   (setq tabkey2-message-countdown nil))
>                 (when tabkey2-message-countdown
>                   (unless (active-minibuffer-window)
>                     (if (current-message)
>                         (tabkey2-message "%s - %s"
>                                  tabkey2-current-tab-info
>                                  (current-message))
>                       (tabkey2-show-message))))))
>             (when tabkey2-is-cycling (setq tabkey2-is-cycling 'tentative))))
>         )
>     (error (message "tabkey2: %s" (error-message-string err)))))

> (defun tabkey2-show-message ()
>   (tabkey2-message "%s" tabkey2-current-tab-info))

> (defvar tabkey2-delay-next-message nil)

> (defun tabkey2-echo-area-message (txt)
>   (message "%s" (propertize txt 'face 'tabkey2-highlight-message)))

> (defun tabkey2-deliver-message (txt)
>   (case tabkey2-message-style
>     (popup (tabkey2-overlay-message txt))
>     (t (tabkey2-echo-area-message txt))))

> (defun tabkey2-timer-deliver-message (txt)
>   (condition-case err
>       (tabkey2-deliver-message txt)
>     (error (message "tabkey2-timer-deliver-message: %s"
>                     (error-message-string err)))))

> (defvar tabkey2-delayed-timer nil)

> (defun tabkey2-cancel-delayed-message ()
>   (when tabkey2-delayed-timer
>     (cancel-timer tabkey2-delayed-timer)
>     (setq tabkey2-delayed-timer)))

> (defun tabkey2-delayed-message (txt)
>   (tabkey2-cancel-delayed-message)
>   (if (and tabkey2-delay-next-message
>            tabkey2-delay-first-message)
>       (setq tabkey2-delayed-timer
>             (run-with-idle-timer
>              tabkey2-delay-first-message nil
>              'tabkey2-timer-deliver-message txt))
>     (tabkey2-deliver-message txt))
>   (setq tabkey2-delay-next-message nil))

> (defun tabkey2-message (format-string &rest args)
>   (let ((txt (apply 'format format-string args)))
>     (tabkey2-delayed-message txt)))

> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;;; Completion function selection etc

> (defun tabkey2-is-active (fun chk)
>   "Return t FUN is active.
> Return t if CHK is a symbol with non-nil value or a form that
> evals to non-nil.

> Otherwise return t if FUN has a key binding at point."
>   (or (if (symbolp chk)
>           (when (boundp chk) (symbol-value chk))
>         (eval chk))
>       (let ((keys (tabkey2-symbol-keys fun))
>             kb-bound)
>         (dolist (key keys)
>           (unless (memq (car (append key nil))
>                         '(menu-bar))
>             (setq kb-bound t)))
>         kb-bound)))

> (defun tabkey2-is-active-p (fun)
>   "Return FUN is active.
> Look it up in `tabkey2-completion-functions' to find out what to
> check and return the value from `tabkey2-is-active'."
>   (let ((chk (catch 'chk
>                (dolist (rec tabkey2-completion-functions)
>                  (when (eq fun (nth 1 rec))
>                    (throw 'chk (nth 2 rec)))))))
>     (tabkey2-is-active fun chk)))

> (defun tabkey2-get-active-completion-functions ()
>   "Get a list of active completion functions.
> Consider only those in `tabkey2-completion-functions'."
>   (delq nil
>         (mapcar (lambda (rec)
>                   (let ((fun (nth 1 rec))
>                         (chk (nth 2 rec)))
>                     (when (tabkey2-is-active fun chk) rec)))
>                 tabkey2-completion-functions)))

> (defvar tabkey2-chosen-completion-function nil)
> (make-variable-buffer-local 'tabkey2-chosen-completion-function)
> (put 'tabkey2-chosen-completion-function 'permanent-local t)


> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;;; Handling of Tab key

> (defun tabkey2-first (prefix)
>   "Do something else after first Tab.
> The Tab key is easy to type on your keyboard.  Then why not use
> it for completion, something that is very useful?  Shells usually
> use Tab for completion so many are used to it.  This was the idea
> of Smart Tabs and this is a generalization of that idea.

> However in Emacs the Tab key is usually used for indentation.
> The idea here is that if indentation has been done once since you
> entered a line, then as long as point is on the same line further
> Tab keys might as well do completion.

> So you kind of do Tab-Tab for first completion and then just Tab
> for further completions on the same line.

> This function is bound to the Tab key when minor mode command
> `tabkey2-mode' is on.  It works like this:

> 1. The first time Tab is pressed do whatever Tab would have done
>    if minor mode command `tabkey2-mode' was off.

>    Then enter a new temporary 'Tab completion state' for just the
>    next command.  Show this by a highlight on the indentation and
>    a marker \"Tab2\" in the mode line.

>    However if either - the minibuffer is active and
>    `tabkey2-in-minibuffer' is nil - `major-mode' is in
>    `tabkey2-modes-that-uses-more-tabs' then do not enter this
>    temporary 'Tab completion state' unless the alternate keyboard
>    key `tabkey2-alternate-key' was used.

>    For major modes where it make sense to press Tab several times
>    S-Tab \(or backtab) can often be used to enter 'Tab completion
>    state'.


> 2. As long as point is on the same line do completion when Tab is
>    pressed again.  Show that this state is active with a
>    highlighting at the line beginning, a marker on the mode
>    line (Tab2) and a message in the echo area which tells what
>    kind of completion will be done.

>    When deciding what kind of completion to do look in the table
>    below and do whatever it found first that is not nil:

>    - `tabkey2-preferred'
>    - `tabkey2-completion-functions'
>    - `tabkey2-fallback'

> 3. Of course, there must be some way for you to easily determine
>    what kind of completion because there are many in Emacs. If
>    you do not turn it off this function show that to you.  And if
>    you turn it off you can still display it, see the key bindings
>    below.

>    If this function is used with a PREFIX argument then it just
>    shows what Tab will do.

>    If the default kind of completion is not what you want then
>    you can choose completion function from any of the candidates
>    in `tabkey2-completion-functions'.  During the 'Tab completion
>    state' the following extra key bindings are available:

> \\{tabkey2-completion-state-emul-map}

> Of course, some languages does not have a fixed indent as is
> assumed above. You can put major modes for those in
> `tabkey2-modes-that-just-completes'.

> Some major modes uses tab for something else already. Those are
> in `tabkey2-modes-that-uses-more-tabs'.  There is an alternate
> key, `tabkey2-alternate-key' if you want to do completion
> there. Note that this key does not do completion. It enters 'Tab
> completion state' in which you have access to the keys above for
> completion etc.

> -----
> NOTE: This uses `emulation-mode-map-alists' and it supposes that
> nothing else is bound to Tab there."
>   (interactive "P")
>   (if (and tabkey2-keymap-overlay
>            (eq (overlay-buffer tabkey2-keymap-overlay) (current-buffer))
>            (>= (point) (overlay-start tabkey2-keymap-overlay))
>            (<= (point) (overlay-end   tabkey2-keymap-overlay)))
>       ;; We should maybe not be here, but the keymap does not work at
>       ;; the end of the buffer so we call the second tab function from
>       ;; here:
>       (if (memq 'shift (event-modifiers last-input-event))
>           (call-interactively 'tabkey2-cycle-through-completion-functions)
>         (call-interactively 'tabkey2-complete prefix))
>     (let* ((emma-without-tabkey2
>             ;; Remove keymaps from tabkey2 in this copy:
>             (delq 'tabkey2--emul-keymap-alist
>                   (copy-sequence emulation-mode-map-alists)))
>            (just-complete (memq major-mode tabkey2-modes-that-just-completes))
>            (is-alt-key (equal tabkey2-alternate-key (this-command-keys-vector)))
>            (what (if is-alt-key
>                      'cycle
>                    (if just-complete
>                        'complete
>                      (if (or (unless tabkey2-in-minibuffer
>                                (active-minibuffer-window))
>                              (use-region-p)
>                              (memq major-mode tabkey2-modes-that-uses-more-tabs))
>                          'indent
>                        'indent-complete
>                        ))))
>            (to-do-1 (if (eq what 'cycle)
>                         'tabkey2-cycle-through-completion-functions
>                       (unless (or
>                                ;; Skip action on tab if shift tab,
>                                ;; backtab, alternate key or a mode in
>                                ;; the "just complete" list
>                                (memq 'shift (event-modifiers last-input-event))
>                                (equal [backtab] (this-command-keys-vector))
>                                (eq what 'complete))
>                         (let ((emulation-mode-map-alists emma-without-tabkey2))
>                           (key-binding [?\t] t)))))
>            (to-do-2 (unless (memq what '(cycle indent))
>                       (let ((emulation-mode-map-alists emma-without-tabkey2))
>                         (or (when (and tabkey2-chosen-completion-function
>                                        (tabkey2-is-active-p
>                                         tabkey2-chosen-completion-function))
>                               tabkey2-chosen-completion-function)
>                             tabkey2-preferred
>                             (tabkey2-first-active-from-completion-functions)
>                             tabkey2-fallback)))))
>       (if prefix
>           (if (memq 'shift (event-modifiers last-input-event))
>               (message "(TabKey2) Shift tab: turn on 'Tab completion state'")
>             (message "(TabKey2) First tab: %s, next: maybe %s"
>                      to-do-1 to-do-2))
>         (when to-do-1
>           (setq tabkey2-was-first-command t)
>           (let ((last-command to-do-1)
>                 mumamo-multi-major-mode)
>             (call-interactively to-do-1)))
>         (unless (tabkey2-read-only-p)
>           (when to-do-2
>             (tabkey2-make-message-and-set-fun to-do-2)
>             (tabkey2-completion-state-mode 1)
>             (when just-complete (call-interactively 'tabkey2-complete))
>             ))))))

> (defvar tabkey2-was-first-command nil)

> (defun tabkey2-complete (prefix)
>   "Call current completion function.
> If used with a PREFIX argument then just show what Tab will do."
>   (interactive "P")
>   (if prefix
>       (message "(TabKey2) Tab: %s" tabkey2-current-tab-function)
>     (call-interactively tabkey2-current-tab-function)))

> ;; Use emulation mode map for first Tab key
> (defconst tabkey2-mode-emul-map
>   (let ((map (make-sparse-keymap)))
>     (define-key map [tab] 'tabkey2-first)
>     (define-key map [backtab] 'tabkey2-first)
>     (define-key map tabkey2-alternate-key 'tabkey2-first)
>     map)
>   "This keymap just binds tab all the time.")

> (defconst tabkey2--emul-keymap-alist nil)

> (define-minor-mode tabkey2-mode
>   "More fun with Tab key number two (completion etc).
> This minor mode binds Tab in a way that let you do completion
> with Tab in all buffers \(where it is possible).  See
> `tabkey2-first' for more information."
>   :keymap nil
>   :global t
>   :group 'tabkey2
>   (if tabkey2-mode
>       (progn
>         ;; Update emul here if keymap have changed
>         (setq tabkey2--emul-keymap-alist
>               (list (cons 'tabkey2-mode
>                           tabkey2-mode-emul-map)))
>         (add-to-list 'emulation-mode-map-alists 'tabkey2--emul-keymap-alist))
>     (tabkey2-completion-state-mode -1)
>     (setq emulation-mode-map-alists (delq 'tabkey2--emul-keymap-alist
>                                           emulation-mode-map-alists))))

> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;;; User interaction

> (defun tabkey2-show-completion-state-help ()
>   "Help for 'Tab completion state'.
> During this state the keymap below is active. This state stops as
> soon as you leave the current row.

> \\{tabkey2-completion-state-emul-map}
> See function `tabkey2-mode' for more information.

> If you want to use Emacs normal help function then press F1
> again.")

> (defun tabkey2-completion-state-help ()
>   "Show help for 'Tab completion state'."
>   (interactive)
>   (tabkey2-show-message)
>   (let ((invoked-by-f1 (equal (this-command-keys-vector) [f1])))
>     (if (and (eq last-command 'tabkey2-completion-state-help)
>              invoked-by-f1)
>         (progn
>           (tabkey2-completion-state-mode -1)
>           (isearch-unread 'f1))
>       (describe-function 'tabkey2-show-completion-state-help))))

> (defun tabkey2-completion-function-help ()
>   "Show help for current completion function."
>   (interactive)
>   (describe-function tabkey2-current-tab-function))

> (defun tabkey2-get-key-binding (fun)
>   "Get key binding for FUN during 'Tab completion state'."
>   (let* ((remapped (command-remapping fun))
>          (key (where-is-internal fun
>                                  tabkey2-completion-state-emul-map
>                                  t
>                                  nil
>                                  remapped)))
>     key))

> (defun tabkey2-make-message-and-set-fun (comp-fun)
>   "Set current completion function to COMP-FUN.
> Build message but don't show it."
>   (setq tabkey2-current-tab-function comp-fun)
>   (let* ((chs-fun 'tabkey2-cycle-through-completion-functions)
>          (key (tabkey2-get-key-binding chs-fun))
>          (active-funs (tabkey2-get-active-completion-functions))
>          what)
>     (dolist (rec tabkey2-completion-functions)
>       (let ((fun (nth 1 rec))
>             (txt (nth 0 rec)))
>         (when (eq fun comp-fun) (setq what txt))))
>     (let ((info (concat (format "Tab: %s" what)
>                         (if (cdr (tabkey2-get-active-completion-functions))
>                             (format ", other %s, help F1" (key-description key))
>                           ""))))
>       (setq tabkey2-current-tab-info info))))

> (defun tabkey2-get-active-string (bnd fun buf)
>   "Get string to show for state.
> BND: means active
> FUN: function
> BUF: buffer"
>   (if bnd
>       (if (with-current-buffer buf (tabkey2-read-only-p))
>           (propertize "active, but read-only" 'face '( :foreground "red"))
>         (propertize "active" 'face '( :foreground "green")))
>     (if (fboundp fun)
>         (propertize "not active" 'face '( :foreground "red"))
>       (propertize "not defined" 'face '( :foreground "gray")))))

> (defun tabkey2-show-completion-functions ()
>   "Show what currently may be used for completion."
>   (interactive)
>   (let ((orig-buf (current-buffer))
>         (orig-mn  mode-name)
>         (active-mark (concat " " (propertize " <= " 'face '( :background "yellow"))))
>         (act-found nil)
>         (chosen-fun tabkey2-chosen-completion-function)
>         what
>         chosen)
>     (when chosen-fun
>       (dolist (rec tabkey2-completion-functions)
>         (let ((fun (nth 1 rec))
>               (txt (nth 0 rec)))
>           (when (eq fun chosen-fun) (setq what txt))))
>       (setq chosen (list what chosen-fun)))
>     (with-output-to-temp-buffer (help-buffer)
>       (help-setup-xref (list #'tabkey2-show-completion-functions) (interactive-p))
>       (with-current-buffer (help-buffer)
>         (insert "The completion functions available for 'Tab completion' in buffer\n'"
>                 (buffer-name orig-buf)
>                 "' at point with mode " orig-mn " are shown below.\n"
>                 "The first active function is used.\n\n")
>         (if (not chosen)
>             (insert "  None is chosen")
>           (let* ((txt (nth 0 chosen))
>                  (fun (nth 1 chosen))
>                  (chk (nth 2 chosen))
>                  (bnd (with-current-buffer orig-buf
>                         (tabkey2-is-active fun chk)))
>                  (act (tabkey2-get-active-string bnd fun orig-buf)))
>             (insert (format "  Chosen currently is %s (%s): %s"
>                             txt fun act))
>             (when bnd (insert active-mark) (setq act-found t))))
>         (insert "\n\n")
>         (if (not tabkey2-preferred)
>             (insert "  None is preferred")
>           (let* ((txt (nth 0 tabkey2-preferred))
>                  (fun (nth 1 tabkey2-preferred))
>                  (chk (nth 2 chosen))
>                  (bnd (with-current-buffer orig-buf
>                         (tabkey2-is-active fun chk)))
>                  (act (tabkey2-get-active-string bnd fun orig-buf)))
>             (insert (format "  Preferred is %s (`%s')': %s"
>                             txt fun act))
>             (when bnd (insert active-mark) (setq act-found t))))
>         (insert "\n\n")
>         (dolist (comp-fun tabkey2-completion-functions)
>           (let* ((txt (nth 0 comp-fun))
>                  (fun (nth 1 comp-fun))
>                  (chk (nth 2 comp-fun))
>                  (bnd (with-current-buffer orig-buf
>                         (tabkey2-is-active fun chk)))
>                  (act (tabkey2-get-active-string bnd fun orig-buf)))
>             (insert
>              (format
>               "  %s (`%s'): %s"
>               txt fun act))
>             (when (and (not act-found) bnd)
>               (insert active-mark) (setq act-found t))
>             (insert "\n")))
>         (insert "\n")
>         (if (not tabkey2-fallback)
>             (insert "  There is no fallback")
>           (let* ((txt (nth 0 tabkey2-fallback))
>                  (fun (nth 1 tabkey2-fallback))
>                  (chk (nth 2 tabkey2-fallback))
>                  (bnd (with-current-buffer orig-buf
>                         (tabkey2-is-active fun chk)))
>                  (act (tabkey2-get-active-string bnd fun orig-buf)))
>             (insert (format "  Fallback is %s (`%s'): %s"
>                             txt fun act))
>             (when (and (not act-found) bnd) (insert active-mark) (setq act-found t))))
>         (insert "\n\nSee function `tabkey2-mode' for more information.")
>         (print-help-return-message)))))

> (defun tabkey2-first-active-from-completion-functions ()
>   "Return first active completion function.
> Look in `tabkey2-completion-functions' for the first function
> that has an active key binding."
>   (catch 'active-fun
>     (dolist (rec tabkey2-completion-functions)
>       (let ((fun (nth 1 rec))
>             (chk (nth 2 rec)))
>         (when (tabkey2-is-active fun chk)
>           (throw 'active-fun fun))))))

> (defvar tabkey2-completing-read 'completing-read)

> (defun tabkey2-symbol-keys (comp-fun)
>   "Get a list of all key bindings for COMP-FUN."
>   (let* ((remapped (command-remapping comp-fun)))
>     (where-is-internal comp-fun
>                        nil ;;overriding-local-map
>                        nil nil remapped)))

> (defun tabkey2-set-fun (fun)
>   "Use function FUN for Tab in 'Tab completion state'."
>   (setq tabkey2-chosen-completion-function fun)
>   (unless fun
>     (setq fun (tabkey2-first-active-from-completion-functions)))
>   (tabkey2-make-message-and-set-fun fun)
>   (when (tabkey2-completion-state-p)
>     (message "%s" tabkey2-current-tab-info)))

> (defun tabkey2-appmenu ()
>   "Make a menu for minor mode command `appmenu-mode'."
>   (unless (tabkey2-read-only-p)
>     (let* ((cf-r (reverse (tabkey2-get-active-completion-functions)))
>            (tit "Complete")
>            (map (make-sparse-keymap tit)))
>       (define-key map [tabkey2-usage]
>         (list 'menu-item "Show available completion functions"
>               'tabkey2-show-completion-functions))
>       (define-key map [tabkey2-divider-1] (list 'menu-item "--"))
>       (let ((set-map (make-sparse-keymap "Set completion")))
>         (define-key map [tabkey2-choose]
>           (list 'menu-item "Set Tab completion for buffer" set-map))
>         (dolist (cf-rec cf-r)
>           (let ((dsc (nth 0 cf-rec))
>                 (fun (nth 1 cf-rec)))
>             (define-key set-map
>               (vector (intern (format "tabkey2-set-%s" fun)))
>               (list 'menu-item dsc
>                     `(lambda ()
>                        (interactive)
>                        (tabkey2-set-fun ',fun))
>                     :button
>                     `(:radio
>                       . (eq ',fun tabkey2-chosen-completion-function))))))
>         (define-key set-map [tabkey2-set-div] (list 'menu-item "--"))
>         (define-key set-map [tabkey2-set-default]
>           (list 'menu-item "Default Tab completion"
>                 (lambda ()
>                   (interactive)
>                   (tabkey2-set-fun nil))
>                 :button
>                 '(:radio . (null tabkey2-chosen-completion-function))))
>         (define-key set-map [tabkey2-set-header-div] (list 'menu-item "--"))
>         (define-key set-map [tabkey2-set-header]
>           (list 'menu-item "Set Tab completion for buffer"))
>         )
>       (define-key map [tabkey2-divider] (list 'menu-item "--"))
>       (dolist (cf-rec cf-r)
>         (let ((dsc (nth 0 cf-rec))
>               (fun (nth 1 cf-rec)))
>           (define-key map
>             (vector (intern (format "tabkey2-call-%s" fun)))
>             (list 'menu-item dsc fun
>                   :button
>                   `(:toggle
>                     . (eq ',fun tabkey2-chosen-completion-function))
>                   ))))
>       map)))

> (defun tabkey2-completion-menu-popup ()
>   "Pop up a menu with completion alternatives."
>   (interactive)
>   (let ((menu (tabkey2-appmenu)))
>     (popup-menu-at-point menu)))

> (defun tabkey2-add-to-appmenu ()
>   "Add a menu to function `appmenu-mode'."
>   (appmenu-add 'tabkey2 nil t "Completion" 'tabkey2-appmenu))

> (defun tabkey2-choose-completion-function ()
>   "Set current completion function.
> Let user choose completion function from those in
> `tabkey2-completion-functions' that have some key binding at
> point.

> Let the chosen completion function be the default for subsequent
> completions in the current buffer."
>   ;; Fix-me: adjust to mumamo.
>   (interactive)
>   (save-match-data
>     (if (and (featurep 'popcmp)
>              tabkey2-use-popup-menus)
>         (tabkey2-completion-menu-popup)
>       (when (eq 'completing-read tabkey2-completing-read) (isearch-unread 'tab))
>       (let* ((cf-r (reverse (tabkey2-get-active-completion-functions)))
>              (cf (cons '("- Use default Tab completion" nil) cf-r))
>              (hist (mapcar (lambda (rec)
>                              (car rec))
>                            cf))
>              (tit (funcall tabkey2-completing-read "Set current completion function: " cf
>                            nil ;; predicate
>                            t   ;; require-match
>                            nil ;; initial-input
>                            'hist ;; hist
>                            ))
>              (fun-rec (assoc-string tit cf))
>              (fun (cadr fun-rec)))
>         (setq tabkey2-chosen-completion-function fun)
>         (unless fun
>           (setq fun (tabkey2-first-active-from-completion-functions)))
>         (tabkey2-make-message-and-set-fun fun)
>         (when (tabkey2-completion-state-p)
>           (tabkey2-show-message))))))

> (defun tabkey2-make-current-sticky ()
>   "Make current Tab completion function sticky.
> Set the current Tab completion function at point as default for
> the current buffer."
>   (interactive)
>   (let ((set-it
>          (y-or-n-p
>           (format
>            "Make %s default for Tab completion in current buffer? "
>            tabkey2-current-tab-function))))
>     (when set-it
>       (setq tabkey2-chosen-completion-function
>             tabkey2-current-tab-function))
>     (unless set-it
>       (when (local-variable-p 'tabkey2-chosen-completion-function)
>         (when (y-or-n-p "Use default Tab completion selection in buffer? ")
>           (setq set-it t))
>         (kill-local-variable 'tabkey2-chosen-completion-function)))
>     (when (tabkey2-completion-state-p)
>       (tabkey2-message "%s%s" tabkey2-current-tab-info
>                        (if set-it " - Done" "")))))

> (defun tabkey2-cycle-through-completion-functions (prefix)
>   "Cycle through completion functions or show current.
> If PREFIX is given just show what shift tab will do."
>   (interactive "P")
>   (save-match-data
>     (if prefix
>         (message "(TabKey2) Shift tab: show/cycle completion function")
>       (when (or tabkey2-message-countdown
>                 tabkey2-is-cycling)
>         ;; Message is shown currently so change
>         (let* ((active (mapcar (lambda (rec)
>                                  (nth 1 rec))
>                                (tabkey2-get-active-completion-functions)))
>                (first (car active))
>                next)
>           (when tabkey2-current-tab-function
>             (while (and active (not next))
>               (when (eq (car active) tabkey2-current-tab-function)
>                 (setq next (cadr active)))
>               (setq active (cdr active))))
>           (unless next (setq next first))
>           (tabkey2-make-message-and-set-fun next)))
>       (tabkey2-show-message)
>       (setq tabkey2-is-cycling t))))


> ;; (defun tabkey2-tooltip (txt)
> ;;   (let* ((params tooltip-frame-parameters)
> ;;          (coord (car (point-to-coord (point))))
> ;;          (left (car coord))
> ;;          (top  (cadr coord))
> ;;          tooltip-frame-parameters
> ;;          )
> ;;     ;; Fix-me: how do you get char height??
> ;;     (setq top (+ top 50))
> ;;     (setq params (tooltip-set-param params 'left left))
> ;;     (setq params (tooltip-set-param params 'top top))
> ;;     (setq params (tooltip-set-param params 'top top))
> ;;     (setq tooltip-frame-parameters params)
> ;;     (tooltip-hide)
> ;;     (tooltip-show txt nil)))

> (defvar tabkey2-overlay-message nil)

> (defun tabkey2-overlay-message (txt)
>   (if (not txt)
>       (delete-overlay tabkey2-overlay-message)
>     (let ((ovl tabkey2-overlay-message)
>           (column (current-column))
>           (txt-len (length txt))
>           (here (point))
>           beg end
>           (before "")
>           (after "")
>           ovl-str too-much
>           (is-eob (eobp))
>           (direction 1))
>       (unless ovl (setq ovl (make-overlay 0 0)))

>       (when is-eob
>         (setq direction -1))
>       ;;(forward-line 0)
>       (forward-line direction)
>       (when (>= (point) (window-end))
>         ;; Go back inside window to avoid aggressive scrolling:
>         (forward-line -1)
>         (scroll-up 1)
>         (forward-line 1))
>       ;; Fix-me: Emacs bug workaround
>       (if (tabkey2-invisible-p (1- (point)))
>           (progn
>             (goto-char here)
>             (tabkey2-echo-area-message txt))
>         (when (tabkey2-invisible-p (point))
>           (while (tabkey2-invisible-p (point))
>             (forward-line direction)))
>         (setq beg (line-beginning-position))
>         (setq end (line-end-position))

>         ;; string before
>         (move-to-column column)
>         (setq before (buffer-substring beg (point)))
>         (when (< (current-column) column)
>           (setq before
>                 (concat before
>                         (make-string (- column (current-column)) ? ))))
>         (setq too-much (- (+ txt-len (length before))
>                           (window-width)))
>         (when (> too-much 0)
>           (setq before (substring before 0 (- too-much))))

>         (unless (> too-much 0)
>           (move-to-column (+ txt-len (length before)))
>           (setq after (buffer-substring (point) end)))

> ;;;       (when is-eob
> ;;;         ;; fix-me
> ;;;         (setq before (concat "\n" before))
> ;;;         (setq beg (point-max-marker))
> ;;;         (setq end (point-max-marker))
> ;;;         (set-marker-insertion-type beg t)
> ;;;         (set-marker-insertion-type end t)
> ;;;         )

>         ;;(setq before (propertize before 'face 'secondary-selection))
>         ;;(setq after (propertize after 'face 'secondary-selection))
>         (setq ovl-str (concat before
>                               (propertize txt 'face 'tabkey2-highlight-popup)
>                               after))

>         (overlay-put ovl 'after-string ovl-str)
>         (overlay-put ovl 'display "")
>         (overlay-put ovl 'window (selected-window))
>         (move-overlay ovl beg end)
>         ;;(message "after move ovl, beg=%s,%s end=%s,%s" beg (tabkey2-invisible-p beg) end (tabkey2-invisible-p end) )(sit-for 2)

>         (setq tabkey2-overlay-message ovl)
>         (add-hook 'pre-command-hook 'tabkey2-remove-overlay-message)
>         (goto-char here)
>         ;;(message "leaving tabkey2-overlay-message")(sit-for 2)
>         ))))

> (defun tabkey2-remove-overlay-message ()
>   (condition-case err
>       (tabkey2-overlay-message nil)
>     (error (message "tabkey2-remove-overlay-message: %s" (error-message-string err))))
>   (remove-hook 'pre-command-hook 'tabkey2-remove-overlay-message))

> (defun tabkey2-invisible-p (pos)
>   "Return non-nil if the character after POS is currently invisible."
>   (let ((prop
>          (get-char-property pos 'invisible)))
>     (if (eq buffer-invisibility-spec t)
>         prop
>       (if (listp prop)
>           (catch 'invis
>             (dolist (p prop)
>               (when (or (memq p buffer-invisibility-spec)
>                         (assq p buffer-invisibility-spec))
>                 (throw 'invis t))))
>         (or (memq prop buffer-invisibility-spec)
>             (assq prop buffer-invisibility-spec))))))

> (provide 'tabkey2)
> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;; tabkey2.el ends here




  reply	other threads:[~2008-03-24  0:30 UTC|newest]

Thread overview: 5+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2008-03-23 20:43 23.0.60; Crash on w32 related to invisible text Lennart Borgman (gmail)
2008-03-23 23:27 ` Stefan Monnier
2008-03-24  0:14   ` Lennart Borgman (gmail)
2008-03-24  0:30     ` Stefan Monnier [this message]
2008-03-24  1:29       ` Lennart Borgman (gmail)

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

  List information: https://www.gnu.org/software/emacs/

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=jwvtzix6k87.fsf-monnier+emacsbugreports@gnu.org \
    --to=monnier@iro.umontreal.ca \
    --cc=emacs-pretest-bug@gnu.org \
    --cc=lennart.borgman@gmail.com \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
Code repositories for project(s) associated with this public inbox

	https://git.savannah.gnu.org/cgit/emacs.git

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).