]> git.eshelyaron.com Git - emacs.git/commitdiff
Major mode for editing files of the Interactive Data Language
authorCarsten Dominik <dominik@science.uva.nl>
Mon, 20 Dec 1999 11:10:02 +0000 (11:10 +0000)
committerCarsten Dominik <dominik@science.uva.nl>
Mon, 20 Dec 1999 11:10:02 +0000 (11:10 +0000)
lisp/progmodes/idlwave.el [new file with mode: 0644]

diff --git a/lisp/progmodes/idlwave.el b/lisp/progmodes/idlwave.el
new file mode 100644 (file)
index 0000000..960d84f
--- /dev/null
@@ -0,0 +1,5751 @@
+;;; idlwave.el --- IDL and WAVE CL editing mode for GNU Emacs
+;; Copyright (c) 1994-1997 Chris Chase
+;; Copyright (c) 1999 Carsten Dominik
+;; Copyright (c) 1999 Free Software Foundation
+
+;; Author: Chris Chase <chase@att.com>
+;; Maintainer: Carsten Dominik <dominik@strw.leidenuniv.nl>
+;; Version: 3.11
+;; Date: $Date: 1999/12/16 10:42:46 $
+;; Keywords: languages
+
+;; This file is part of the GNU Emacs.
+
+;; GNU Emacs 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.
+
+;; GNU Emacs 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 GNU Emacs; see the file COPYING.  If not, write to the
+;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+;; Boston, MA 02111-1307, USA.
+
+;;; Commentary:
+
+;; In distant past, based on pascal.el.  Though bears little
+;; resemblance to that now.
+;;
+;; Incorporates many ideas, such as abbrevs, action routines, and
+;; continuation line indenting, from wave.el.
+;; wave.el original written by Lubos Pochman, Precision Visuals, Boulder.
+;;
+;; See the mode description ("C-h m" in idlwave-mode or "C-h f idlwave-mode")
+;; for features, key bindings, and info.
+;; Also, Info format documentation is available with `M-x idlwave-info'
+;;
+;;
+;; INSTALLATION
+;; ============
+;;
+;; Follow the instructions in the INSTALL file of the distribution.
+;; In short, put this file on your load path and add the following
+;; lines to your .emacs file:
+;;
+;; (autoload 'idlwave-mode "idlwave" "IDLWAVE Mode" t)
+;; (autoload 'idlwave-shell "idlwave-shell" "IDLWAVE Shell" t)
+;; (setq auto-mode-alist (cons '("\\.pro\\'" . idlwave-mode) auto-mode-alist))
+;;
+;;
+;; SOURCE
+;; ======
+;;
+;; The newest version of this file is available from the maintainers
+;; Webpage.
+;;
+;;   http://www.strw.leidenuniv.el/~dominik/Tools/idlwave
+;;
+;; DOCUMENTATION
+;; =============
+;;
+;; IDLWAVE is documented online in info format.
+;; A printable version of the documentation is available from the
+;; maintainers webpage (see under SOURCE)
+;;
+;; 
+;; ACKNOWLEDGMENTS
+;; ===============
+;;
+;;  Thanks to the following people for their contributions and comments:
+;;
+;;    Ulrik Dickow <dickow@nbi.dk>
+;;    Eric E. Dors <edors@lanl.gov>
+;;    Stein Vidar H. Haugan <s.v.h.haugan@astro.uio.no>
+;;    David Huenemoerder <dph@space.mit.edu>
+;;    Kevin Ivory <Kevin.Ivory@linmpi.mpg.de>
+;;    Xuyong Liu <liu@stsci.edu>
+;;    Simon Marshall <Simon.Marshall@esrin.esa.it>
+;;    Laurent Mugnier <mugnier@onera.fr>
+;;    Lubos Pochman <lubos@rsinc.com>
+;;    Patrick M. Ryan <pat@jaameri.gsfc.nasa.gov>
+;;    Marty Ryba <ryba@ll.mit.edu>
+;;    Phil Williams <williams@irc.chmcc.org>
+;;    J.D. Smith <jdsmith@astrosun.tn.cornell.edu>
+;;    Phil Sterne <sterne@dublin.llnl.gov>
+;;
+;; CUSTOMIZATION:
+;; =============
+;;
+;; IDLWAVE has customize support - so if you want to learn about the
+;;  variables which control the behavior of the mode, use
+;; `M-x idlwave-customize'.
+;;
+;; You can set your own preferred values with Customize, or with Lisp
+;; code in .emacs.  For an example of what to put into .emacs, check
+;; the TexInfo documentation.
+;;
+;; KNOWN PROBLEMS:
+;; ==============
+;;
+;;   Moving the point backwards in conjunction with abbrev expansion
+;;   does not work as I would like it, but this is a problem with
+;;   emacs abbrev expansion done by the self-insert-command.  It ends
+;;   up inserting the character that expanded the abbrev after moving
+;;   point backward, e.g., "\cl" expanded with a space becomes
+;;   "LONG( )" with point before the close paren.  This is solved by
+;;   using a temporary function in `post-command-hook' - not pretty, 
+;;   but it works.<
+;;
+;;   Tabs and spaces are treated equally as whitespace when filling a
+;;   comment paragraph.  To accomplish this, tabs are permanently
+;;   replaced by spaces in the text surrounding the paragraph, which
+;;   may be an undesirable side-effect.  Replacing tabs with spaces is
+;;   limited to comments only and occurs only when a comment
+;;   paragraph is filled via `idlwave-fill-paragraph'.
+;;
+;;   "&" is ignored when parsing statements.
+;;   Avoid muti-statement lines (using "&") on block begin and end
+;;   lines.  Multi-statement lines can mess up the formatting, for
+;;   example, multiple end statements on a line: endif & endif.
+;;   Using "&" outside of block begin/end lines should be okay.
+;;
+;;   It is possible that the parser which decides what to complete has
+;;   problems with pointer dereferencing statements.  I don't use
+;;   pointers often enough to find out - please report any problems.
+;;
+;;   Completion of keywords for SETPROPERTY and GETPROPERTY assumes that
+;;   all INIT keywords are allowed in these methods as well.  In some
+;;   cases, there are exceptions to this rule and IDLWAVE will offer
+;;   a few illegal keyword parameters.
+;;
+;;   Completion and Routine Info do not know about inheritance.  Thus,
+;;   Keywords inherited from superclasses are not displayed and cannot
+;;   completed.
+;;
+;;   When forcing completion of method keywords, the initial
+;;   query for a method has multiple entries for some methods.  Would
+;;   be too difficult to fix this hardly used problem.
+;;
+\f
+;;; Code:
+
+(eval-when-compile (require 'cl))
+
+(eval-and-compile
+  ;; Kludge to allow `defcustom' for Emacs 19.
+  (condition-case () (require 'custom) (error nil))
+  (if (and (featurep 'custom) (fboundp 'custom-declare-variable))
+      nil ;; We've got what we needed
+    ;; We have the old or no custom-library, hack around it!
+    (defmacro defgroup (&rest args) nil)
+    (defmacro defcustom (var value doc &rest args) 
+      (` (defvar (, var) (, value) (, doc))))))
+
+(defgroup idlwave nil
+  "Major mode for editing IDL/WAVE CL .pro files"
+  :tag "IDLWAVE"
+  :link '(url-link :tag "Home Page" 
+                  "http://strw.leidenuniv.nl/~dominik/Tools/idlwave")
+  :link '(emacs-commentary-link :tag "Commentary in idlwave-shell.el"
+                               "idlwave-shell.el")
+  :link '(emacs-commentary-link :tag "Commentary in idlwave.el" "idlwave.el")
+  :link '(custom-manual "(idlwave)Top")
+  :prefix "idlwave"
+  :group 'languages)
+
+;;; Variables for indentation behavior ---------------------------------------
+
+(defgroup idlwave-code-formatting nil
+  "Indentation and formatting options for IDLWAVE mode."
+  :group 'idlwave)
+
+(defcustom idlwave-main-block-indent 0
+  "*Extra indentation for the main block of code.
+That is the block between the FUNCTION/PRO statement and the END
+statement for that program unit."
+  :group 'idlwave-code-formatting
+  :type 'integer)
+
+(defcustom idlwave-block-indent 4
+  "*Extra indentation applied to block lines.
+If you change this, you probably also want to change `idlwave-end-offset'."
+  :group 'idlwave-code-formatting
+  :type 'integer)
+
+(defcustom idlwave-end-offset -4
+  "*Extra indentation applied to block END lines.
+A value equal to negative `idlwave-block-indent' will make END lines
+line up with the block BEGIN lines."
+  :group 'idlwave-code-formatting
+  :type 'integer)
+
+(defcustom idlwave-continuation-indent 2
+  "*Extra indentation applied to continuation lines.
+This extra offset applies to the first of a set of continuation lines.
+The following lines receive the same indentation as the first.
+Also, the value of this variable applies to continuation lines inside
+parenthesis.  When the current line contains an open unmatched ([{,
+the next line is indented to that parenthesis plus the value of this variable."
+  :group 'idlwave-code-formatting
+  :type 'integer)
+
+(defcustom idlwave-hanging-indent t
+  "*If set non-nil then comment paragraphs are indented under the
+hanging indent given by `idlwave-hang-indent-regexp' match in the first line
+of the paragraph."
+  :group 'idlwave-code-formatting
+  :type 'boolean)
+
+(defcustom idlwave-hang-indent-regexp "- "
+  "*Regular expression matching the position of the hanging indent
+in the first line of a comment paragraph. The size of the indent
+extends to the end of the match for the regular expression."
+  :group 'idlwave-code-formatting
+  :type 'regexp)
+
+(defcustom idlwave-use-last-hang-indent nil
+  "*If non-nil then use last match on line for `idlwave-indent-regexp'."
+  :group 'idlwave-code-formatting
+  :type 'boolean)
+
+(defcustom idlwave-fill-comment-line-only t
+  "*If non-nil then auto fill will only operate on comment lines."
+  :group 'idlwave-code-formatting
+  :type 'boolean)
+
+(defcustom idlwave-auto-fill-split-string t
+  "*If non-nil then auto fill will split strings with the IDL `+' operator.
+When the line end falls within a string, string concatenation with the 
+'+' operator will be used to distribute a long string over lines.  
+If nil and a string is split then a terminal beep and warning are issued.
+
+This variable is ignored when `idlwave-fill-comment-line-only' is
+non-nil, since in this case code is not auto-filled."
+  :group 'idlwave-code-formatting
+  :type 'boolean)
+
+(defcustom idlwave-split-line-string t
+  "*If non-nil then `idlwave-split-line' will split strings with `+'.
+When the splitting point of a line falls inside a string, split the string
+using the `+' string concatenation operator.  If nil and a string is
+split then a terminal beep and warning are issued."
+  :group 'idlwave-code-formatting
+  :type 'boolean)
+
+(defcustom idlwave-no-change-comment ";;;"
+  "*The indentation of a comment that starts with this regular
+expression will not be changed. Note that the indentation of a comment
+at the beginning of a line is never changed."
+  :group 'idlwave-code-formatting
+  :type 'string)
+
+(defcustom idlwave-begin-line-comment nil
+  "*A comment anchored at the beginning of line.
+A comment matching this regular expression will not have its
+indentation changed.  If nil the default is \"^;\", i.e., any line
+beginning with a \";\".  Expressions for comments at the beginning of
+the line should begin with \"^\"."
+  :group 'idlwave-code-formatting
+  :type '(choice (const :tag "Any line beginning with `;'" nil)
+                'regexp))
+
+(defcustom idlwave-code-comment ";;[^;]"
+  "*A comment that starts with this regular expression on a line by
+itself is indented as if it is a part of IDL code.  As a result if
+the comment is not preceded by whitespace it is unchanged."
+  :group 'idlwave-code-formatting
+  :type 'regexp)
+
+;; Comments not matching any of the above will be indented as a
+;; right-margin comment, i.e., to a minimum of `comment-column'.
+
+
+;;; Routine Info and Completion ---------------------------------------
+
+(defgroup idlwave-routine-info-and-completion nil
+  "Routine info and name/keyword completion options for IDLWAVE mode."
+  :group 'idlwave)
+
+(defcustom idlwave-scan-all-buffers-for-routine-info t
+  "*Non-nil means, scan all buffers for IDL programs when updating info.
+`idlwave-update-routine-info' scans buffers of the current Emacs session
+for routine definitions.  When this variable is nil, it only parses the
+current buffer.  When non-nil, all buffers are searched.
+A prefix to \\[idlwave-update-routine-info] toggles the meaning of this
+variable for the duration of the command."
+  :group 'idlwave-routine-info-and-completion
+  :type 'boolean)
+
+(defcustom idlwave-query-shell-for-routine-info t
+  "*Non-nil means query the shell for info about compiled routines.
+Querying the shell is useful to get information about compiled modules,
+and it is turned on by default.  However, when you have a complete library
+scan, this is not necessary."
+  :group 'idlwave-routine-info-and-completion
+  :type 'boolean)
+
+(defcustom idlwave-library-path nil
+  "Library path for Windows and MacOS.  Not needed under Unix.
+When selecting the directories to scan for IDL library routine info,
+IDLWAVE can under UNIX query the shell for the exact search path.
+However, under Windows and MacOS, the IDLWAVE shell does not work.  In this
+case, this variable specifies the path where IDLWAVE can find library files.
+The shell will only be asked when this variable is nil.
+The value is a list of directories.  A directory preceeded by a `+' will
+be search recursively."
+  :group 'idlwave-routine-info-and-completion
+  :type '(repeat (directory)))
+
+(defcustom idlwave-libinfo-file nil
+  "*File for routine information of the IDL library.
+When this points to a file, the file will be loaded when IDLWAVE first
+accesses routine info (or does completion).
+When you scan the library with `idlwave-create-libinfo-file', this file
+will be used to store the result."
+  :group 'idlwave-routine-info-and-completion
+  :type 'file)
+
+(eval-and-compile
+  (defconst idlwave-tmp
+    '(choice :tag "by applying the function"
+      (const upcase)
+      (const downcase)
+      (const capitalize)
+      (const preserve)
+      (symbol :tag "Other"))))
+
+
+(defcustom idlwave-completion-case '((routine . upcase)
+                                    (keyword . upcase)
+                                    (class   . preserve)
+                                    (method  . preserve))
+  "Association list setting the case of completed words.
+
+This variable determines the case (UPPER/lower/Capitalized...) of
+words inserted into the buffer by completion.  The preferred case can
+be specified separately for routine names, keywords, classes and
+methods. 
+This alist should therefore have entries for `routine' (normal
+functions and procedures, i.e. non-methods), `keyword', `class', and
+`method'.  Plausible values are
+
+upcase      upcase whole word, like `BOX_CURSOR'
+downcase    downcase whole word, like `read_ppm'
+capitalize  capitalize each part, like `Widget_Control'
+preserve    preserve case as is, like `IDLgrView'
+
+The value can also be any Emacs Lisp function which transforms the
+case of characters in a string.
+
+A value of `preserve' means that the case of the completed word is
+identical to the way it was written in the definition statement of the
+routine.  This was implemented to allow for mixed-case completion, in
+particular of object classes and methods.
+If a completable word is defined in multiple locations, the meaning of
+`preserve' is not unique since the different definitions might be
+cased differently.  Therefore IDLWAVE always takes the case of the
+*first* definition it encounters during routine info collection and
+uses the case derived from it consistently.
+
+Note that a lowercase-only string in the buffer will always be completed in
+lower case (but see the variable `idlwave-completion-force-default-case').
+
+After changing this variable, you need to either restart Emacs or press
+`C-u C-c C-i' to update the internal lists."
+  :group 'idlwave-routine-info-and-completion
+  :type `(repeat
+         (cons (symbol :tag "Derive completion case for")
+               ,idlwave-tmp)))
+
+(defcustom idlwave-completion-force-default-case nil
+  "*Non-nil means, completion will always honor `idlwave-completion-case'.
+When nil, only the completion of a mixed case or upper case string
+will honor the default settings in `idlwave-completion-case', while
+the completion of lower case strings will be completed entirely in
+lower case."
+  :group 'idlwave-routine-info-and-completion
+  :type 'boolean)
+
+(defcustom idlwave-complete-empty-string-as-lower-case nil
+  "*Non-nil means, the empty string is considered downcase for completion.
+The case of what is already in the buffer determines the case of completions.
+When this variable is non-nil, the empty string is considered to be downcase.
+Completing on the empty string then offers downcase versions of the possible
+completions."
+  :group 'idlwave-routine-info-and-completion
+  :type 'boolean)
+
+(defvar idlwave-default-completion-case-is-down nil
+  "Obsolete variable.  See `idlwave-complete-empty-string-as-lower-case' and
+`idlwave-completion-case'.")
+
+(defcustom idlwave-buffer-case-takes-precedence nil
+  "*Non-nil means, the case of tokens in buffers dominates over system stuff.
+To make this possible, we need to re-case everything each time we update
+the routine info from the buffers.  This is slow.
+The default is to consider the case given in the system and library files
+first which makes updating much faster."
+  :group 'idlwave-routine-info-and-completion
+  :type 'boolean)
+
+(defcustom idlwave-completion-show-classes 1
+  "*Number of classes to show when completing object methods and keywords.
+When completing methods or keywords for an object with unknown class,
+the *Completions* buffer will show the legal classes for each completion
+like this:
+
+MyMethod <Class1,Class2,Class3>
+
+The value of this variable may be nil to inhibit display, or an integer to
+indicate the maximum number of classes to display.
+
+On XEmacs, a full list of classes will also be placed into a `help-echo'
+property on the competion items, so that the list of classes for the current
+item is displayed in the echo area.  If the value of this variable is a
+negative integer, the `help-echo' property will be suppressed."
+  :group 'idlwave-routine-info-and-completion
+  :type '(choice (const :tag "Don't show" nil)
+                (integer :tag "Number of classes shown" 1)))
+
+(defcustom idlwave-completion-fontify-classes t
+  "*Non-nil means, fontify the classes in completions buffer.
+This makes it easier to distinguish the completion items from the extra
+class info listed.  See `idlwave-completion-show-classes'."
+  :group 'idlwave-routine-info-and-completion
+  :type 'boolean)
+
+(defcustom idlwave-query-class '((method-default . nil)
+                                (keyword-default . nil))
+  "Association list governing specification of object classes for completion.
+
+When IDLWAVE is trying to complete items which belong to the object
+oriented part of IDL, it usually cannot determine the class of a given
+object from context.  In order to provide the user with a correct list
+of methods or keywords, it would have to determine the appropriate
+class.  IDLWAVE has two ways to deal with this problem.
+
+1. One possibility is to combine the items of all available
+   classes for the purpose of completion.  So when completing a
+   method, all methods of all classes are available, and when
+   completing a keyword, all keywords allowed for this method in any
+   class will be possible completions.  This behavior is very much
+   like normal completion and is therefore the default.  It works much 
+   better than one might think - only for the INIT, GETPROPERTY and
+   SETPROPERTY the keyword lists become uncomfortably long.
+   See also `idlwave-completion-show-classes'.
+
+2. The second possibility is to ask the user on each occasion.  To
+   make this less interruptive, IDLWAVE can store the class as a text
+   property on the object operator `->'.  For a given object in the
+   source code, class selection will then be needed only once
+   - for example to complete the method.  Keywords to the method can
+   then be completed directly, because the class is already known.
+   You will have to turn on the storage of the selected class
+   explicitly with the variable `idlwave-store-inquired-class'.
+
+This variable allows to configure IDLWAVE's behavior during
+completion.  Its value is an alist, which should contain at least two
+elements: (method-default . VALUE) and (keyword-default . VALUE),
+where VALUE is either t or nil.  These specify if the class should be
+determined during method and keyword completion, respectively.
+
+The alist may have additional entries specifying exceptions from the 
+keyword completion rule for specific methods, like INIT or
+GETPROPERTY.  In order to turn on class specification for the INIT
+method, add an entry (\"INIT\" . t).  The method name must be ALL-CAPS."
+  :group 'idlwave-routine-info-and-completion
+  :type '(list
+         (cons (const method-default)
+               (boolean :tag "Determine class when completing METHODS    "))
+         (cons (const keyword-default)
+               (boolean :tag "Determine class when completing KEYWORDS   "))
+         (repeat
+          :tag "Exceptions to defaults"
+          :inline t
+          (cons (string  :tag "MODULE" :value "")
+                (boolean :tag "Determine class for this method")))))
+
+(defcustom idlwave-store-inquired-class nil
+  "*Non-nil means, store class of a method call as text property on `->'.
+IDLWAVE sometimes has to ask the user for the class associated with a
+particular object method call.  This happens during the commands
+`idlwave-routine-info' and `idlwave-complete', depending upon the
+value of the variable `idlwave-query-class'.
+
+When you specify a class, this information can be stored as a text
+property on the `->' arrow in the source code, so that during the same 
+editing session, IDLWAVE will not have to ask again.  When this
+variable is non-nil, IDLWAVE will store and reuse the class information.
+The class stored can be checked and removed with `\\[idlwave-routine-info]'
+on the arrow.
+
+The default of this variable is nil, since the result of commands then
+is more predictable.  However, if you know what you are doing, it can
+be nice to turn this on.
+
+An arrow which knows the class will be highlighted with
+`idlwave-class-arrow-face'.  The command \\[idlwave-routine-info]
+displays (with prefix arg: deletes) the class stored on the arrow
+at point."
+  :group 'idlwave-routine-info-and-completion
+  :type 'boolean)
+
+(defcustom idlwave-class-arrow-face 'bold
+  "*Face to highlight object operator arrows `->' which carry a class property.
+When IDLWAVE stores a class name as text property on an object arrow
+(see variable `idlwave-store-inquired-class', it highlights the arrow
+with this font in order to remind the user that this arrow is special."
+  :group 'idlwave-routine-info-and-completion
+  :type 'symbol)
+
+(defcustom idlwave-resize-routine-help-window t
+  "*Non-nil means, resize the Routine-info *Help* window to fit the content."
+  :group 'idlwave-routine-info-and-completion
+  :type 'boolean)
+
+(defcustom idlwave-keyword-completion-adds-equal t
+  "*Non-nil means, completion automatically adds `=' after completed keywords."
+  :group 'idlwave-routine-info
+  :type 'boolean)
+
+(defcustom idlwave-function-completion-adds-paren t
+  "*Non-nil means, completion automatically adds `(' after completed function.
+Nil means, don't add anything.
+A value of `2' means, also add the closing parenthesis and position cursor
+between the two."
+  :group 'idlwave-routine-info
+  :type '(choice (const :tag "Nothing" nil)
+                (const :tag "(" t)
+                (const :tag "()" 2)))
+
+(defcustom idlwave-completion-restore-window-configuration t
+  "*Non-nil means, try to restore the window configuration after completion.
+When completion is not unique, Emacs displays a list of completions.
+This messes up your window configuration.  With this variable set, IDLWAVE
+restores the old configuration after successful completion."
+  :group 'idlwave-routine-info-and-completion
+  :type 'boolean)
+
+;;; Variables for abbrev and action behavior -----------------------------
+
+(defgroup idlwave-abbrev-and-indent-action nil
+  "IDLWAVE performs actions when expanding abbreviations or indenting lines.
+The variables in this group govern this."
+  :group 'idlwave)
+
+(defcustom idlwave-do-actions nil
+  "*Non-nil means performs actions when indenting.
+The actions that can be performed are listed in `idlwave-indent-action-table'."
+  :group 'idlwave-abbrev-and-indent-action
+  :type 'boolean)
+
+(defcustom idlwave-abbrev-start-char "\\"
+  "*A single character string used to start abbreviations in abbrev mode.
+Possible characters to chose from: ~`\%
+or even '?'.  '.' is not a good choice because it can make structure
+field names act like abbrevs in certain circumstances.
+
+Changes to this in `idlwave-mode-hook' will have no effect.  Instead a user
+must set it directly using `setq' in the .emacs file before idlwave.el
+is loaded."
+  :group 'idlwave-abbrev-and-indent-action
+  :type 'string)
+
+(defcustom idlwave-surround-by-blank nil
+  "*Non-nil means, enable `idlwave-surround'.
+If non-nil, `=',`<',`>',`&',`,' are surrounded with spaces by
+`idlwave-surround'.
+See help for `idlwave-indent-action-table' for symbols using `idlwave-surround'.
+
+Also see the default key bindings for keys using `idlwave-surround'.
+Keys are bound and made into actions calling `idlwave-surround' with
+`idlwave-action-and-binding'.
+See help for `idlwave-action-and-binding' for examples.
+
+Also see help for `idlwave-surround'."
+  :group 'idlwave-abbrev-and-indent-action
+  :type 'boolean)
+
+(defcustom idlwave-pad-keyword t
+  "*Non-nil means pad '=' for keywords like assignments.
+Whenever `idlwave-surround' is non-nil then this affects how '=' is padded
+for keywords.  If non-nil it is padded the same as for assignments.
+If nil then spaces are removed."
+  :group 'idlwave-abbrev-and-indent-action
+  :type 'boolean)
+
+(defcustom idlwave-show-block t
+  "*Non-nil means point blinks to block beginning for `idlwave-show-begin'."
+  :group 'idlwave-abbrev-and-indent-action
+  :type 'boolean)
+
+(defcustom idlwave-expand-generic-end nil
+  "*Non-nil means expand generic END to ENDIF/ENDELSE/ENDWHILE etc."
+  :group 'idlwave-abbrev-and-indent-action
+  :type 'boolean)
+
+(defcustom idlwave-abbrev-move t
+  "*Non-nil means the abbrev hook can move point.
+Set to nil by `idlwave-expand-region-abbrevs'. To see the abbrev
+definitions, use the command `list-abbrevs', for abbrevs that move
+point. Moving point is useful, for example, to place point between
+parentheses of expanded functions.
+
+See `idlwave-check-abbrev'."
+  :group 'idlwave-abbrev-and-indent-action
+  :type 'boolean)
+
+(defcustom idlwave-abbrev-change-case nil
+  "*Non-nil means all abbrevs will be forced to either upper or lower case.
+If the value t, all expanded abbrevs will be upper case.
+If the value is 'down then abbrevs will be forced to lower case.
+If nil, the case will not change.
+If `idlwave-reserved-word-upcase' is non-nil, reserved words will always be
+upper case, regardless of this variable."
+  :group 'idlwave-abbrev-and-indent-action
+  :type 'boolean)
+
+(defcustom idlwave-reserved-word-upcase nil
+  "*Non-nil means, reserved words will be made upper case via abbrev expansion.
+If nil case of reserved words is controlled by `idlwave-abbrev-change-case'.
+Has effect only if in abbrev-mode."
+  :group 'idlwave-abbrev-and-indent-action
+  :type 'boolean)
+
+;;; Action/Expand Tables.
+;;
+;; The average user may have difficulty modifying this directly.  It
+;; can be modified/set in idlwave-mode-hook, but it is easier to use
+;; idlwave-action-and-binding. See help for idlwave-action-and-binding for
+;; examples of how to add an action.
+;;
+;; The action table is used by `idlwave-indent-line' whereas both the
+;; action and expand tables are used by `idlwave-indent-and-action'.  In
+;; general, the expand table is only used when a line is explicitly
+;; indented.  Whereas, in addition to being used when the expand table
+;; is used, the action table is used when a line is indirectly
+;; indented via line splitting, auto-filling or a new line creation.
+;;
+;; Example actions:
+;;
+;;  Capitalize system vars
+;;   (idlwave-action-and-binding idlwave-sysvar '(capitalize-word 1) t)
+;;
+;;  Capitalize procedure name
+;;   (idlwave-action-and-binding "\\<\\(pro\\|function\\)\\>[ \t]*\\<"
+;;                           '(capitalize-word 1) t)
+;;
+;;  Capitalize common block name
+;;   (idlwave-action-and-binding "\\<common\\>[ \t]+\\<"
+;;                           '(capitalize-word 1) t)
+;;  Capitalize label
+;;   (idlwave-action-and-binding (concat "^[ \t]*" idlwave-label)
+;;                           '(capitalize-word -1) t)
+
+(defvar idlwave-indent-action-table nil
+  "*Associated array containing action lists of search string (car),
+and function as a cdr. This table is used by `idlwave-indent-line'.
+See documentation for `idlwave-do-action' for a complete description of
+the action lists.
+
+Additions to the table are made with `idlwave-action-and-binding' when a
+binding is not requested.
+See help on `idlwave-action-and-binding' for examples.")
+
+(defvar idlwave-indent-expand-table nil
+  "*Associated array containing action lists of search string (car),
+and function as a cdr. The table is used by the
+`idlwave-indent-and-action' function. See documentation for
+`idlwave-do-action' for a complete description of the action lists.
+
+Additions to the table are made with `idlwave-action-and-binding' when a
+binding is requested.
+See help on `idlwave-action-and-binding' for examples.")
+
+;;; Documentation header and history keyword ---------------------------------
+
+(defgroup idlwave-documentation nil
+  "Options for documenting IDLWAVE files."
+  :group 'idlwave)
+
+;; FIXME: make defcustom?
+(defvar idlwave-file-header
+  (list nil
+        ";+
+; NAME:
+;
+;
+;
+; PURPOSE:
+;
+;
+;
+; CATEGORY:
+;
+;
+;
+; CALLING SEQUENCE:
+;
+;
+;
+; INPUTS:
+;
+;
+;
+; OPTIONAL INPUTS:
+;
+;
+;
+; KEYWORD PARAMETERS:
+;
+;
+;
+; OUTPUTS:
+;
+;
+;
+; OPTIONAL OUTPUTS:
+;
+;
+;
+; COMMON BLOCKS:
+;
+;
+;
+; SIDE EFFECTS:
+;
+;
+;
+; RESTRICTIONS:
+;
+;
+;
+; PROCEDURE:
+;
+;
+;
+; EXAMPLE:
+;
+;
+;
+; MODIFICATION HISTORY:
+;
+;-
+")
+  "*A list (PATHNAME STRING) specifying the doc-header template to use for
+summarizing a file. If PATHNAME is non-nil then this file will be included.
+Otherwise STRING is used. If NIL, the file summary will be omitted.
+For example you might set PATHNAME to the path for the
+lib_template.pro file included in the IDL distribution.")
+
+(defcustom idlwave-timestamp-hook 'idlwave-default-insert-timestamp
+  "*The hook function used to update the timestamp of a function."
+  :group 'idlwave-documentation
+  :type 'function)
+
+(defcustom idlwave-doc-modifications-keyword "HISTORY"
+  "*The modifications keyword to use with the log documentation commands.
+A ':' is added to the keyword end.
+Inserted by doc-header and used to position logs by doc-modification.
+If nil it will not be inserted."
+  :group 'idlwave-documentation
+  :type 'string)
+
+(defcustom idlwave-doclib-start "^;+\\+"
+  "*Regexp matching the start of a document library header."
+  :group 'idlwave-documentation
+  :type 'regexp)
+
+(defcustom idlwave-doclib-end "^;+-"
+  "*Regexp matching the end of a document library header."
+  :group 'idlwave-documentation
+  :type 'regexp)
+
+;;; External Programs -------------------------------------------------------
+
+(defgroup idlwave-external-programs nil
+  "Miscellaneous options for IDLWAVE mode."
+  :group 'idlwave)
+
+;; WARNING: The following variable has recently been moved from
+;; idlwave-shell.el to this file.  I hope this does not break
+;; anything.
+
+(defcustom idlwave-shell-explicit-file-name "idl"
+  "*If non-nil, is the command to run IDL.
+Should be an absolute file path or path relative to the current environment
+execution search path."
+  :group 'idlwave-external-programs
+  :type 'string)
+
+;; FIXME: Document a case when is this needed.
+(defcustom idlwave-shell-command-line-options nil
+  "*A list of command line options for calling the IDL program."
+  :type '(repeat (string :value ""))
+  :group 'idlwave-external-programs)
+
+(defcustom idlwave-help-application "idlhelp"
+  "*The external application providing reference help for programming."
+  :group 'idlwave-external-programs
+  :type 'string)
+
+;;; Miscellaneous variables -------------------------------------------------
+
+(defgroup idlwave-misc nil
+  "Miscellaneous options for IDLWAVE mode."
+  :group 'idlwave)
+
+(defcustom idlwave-startup-message t
+  "*Non-nil displays a startup message when `idlwave-mode' is first called."
+  :group 'idlwave-misc
+  :type 'boolean)
+
+(defcustom idlwave-default-font-lock-items 
+  '(pros-and-functions batch-files idl-keywords label goto
+                      common-blocks class-arrows)
+  "Items which should be fontified on the default fontification level 2.
+IDLWAVE defines 3 levels of fontification.  Level 1 is very little, level 3
+is everything and level 2 is specified by this list.
+This variable must be set before IDLWAVE gets loaded.  It is
+a list of symbols, the following symbols are allowed.
+
+pros-and-functions   Procedure and Function definitions
+batch-files          Batch Files
+idl-keywords         IDL Keywords
+label                Statement Labels
+goto                 Goto Statements
+common-blocks        Common Blocks
+keyword-parameters   Keyword Parameters in routine definitions and calls
+system-variables     System Variables
+fixme                FIXME: Warning in comments (on XEmacs only v. 21.0 and up)
+class-arrows         Object Arrows with class property"
+  :group 'idlwave-misc
+  :type '(set
+         :inline t :greedy t
+         (const :tag "Procedure and Function definitions" pros-and-functions)
+         (const :tag "Batch Files"                        batch-files)
+         (const :tag "IDL Keywords (reserved words)"      idl-keywords)
+         (const :tag "Statement Labels"                   label)
+         (const :tag "Goto Statements"                    goto)
+         (const :tag "Common Blocks"                      common-blocks)
+         (const :tag "Keyword Parameters"                 keyword-parameters)
+         (const :tag "System Variables"                   system-variables)
+         (const :tag "FIXME: Warning"                     fixme)
+         (const :tag "Object Arrows with class property " class-arrows)))
+
+(defcustom idlwave-mode-hook nil
+  "Normal hook.  Executed when a buffer is put into `idlwave-mode'."
+  :group 'idlwave-misc
+  :type 'hook)
+
+(defcustom idlwave-load-hook nil
+  "Normal hook.  Executed when idlwave.el is loaded."
+  :group 'idlwave-misc
+  :type 'hook)
+
+;;;
+;;; End customization variables section
+;;;
+
+;;; Non customization variables
+
+;;; font-lock mode - Additions by Phil Williams, Ulrik Dickow and
+;;; Simon Marshall <simon@gnu.ai.mit.edu>
+;;; and Carsten Dominik...
+
+(defconst idlwave-font-lock-keywords-1 nil
+  "Subdued level highlighting for IDLWAVE mode.")
+
+(defconst idlwave-font-lock-keywords-2 nil
+  "Medium level highlighting for IDLWAVE mode.")
+
+(defconst idlwave-font-lock-keywords-3 nil
+  "Gaudy level highlighting for IDLWAVE mode.")
+
+(let* ((oldp (or (string-match "Lucid" emacs-version)
+                (not (boundp 'emacs-minor-version))
+                (and (<= emacs-major-version 19) 
+                     (<= emacs-minor-version 29))))
+
+       ;; The following are the reserved words in IDL.  Maybe we should
+       ;; highlight some more stuff as well?       
+       (idl-keywords
+;      '("and" "or" "xor" "not"
+;        "eq" "ge" "gt" "le" "lt" "ne" 
+;        "for" "do" "endfor"
+;        "if" "then" "endif" "else" "endelse" 
+;        "case" "of" "endcase"
+;        "begin" "end"
+;        "repeat" "until" "endrep"
+;        "while" "endwhile" 
+;        "goto" "return"
+;         "inherits" "mod" "on_error" "on_ioerror")  ;; on_error is not reserved
+       (concat "\\<\\("
+               "and\\|begin\\|case\\|do\\|e\\(lse\\|nd\\(case\\|else\\|"
+               "for\\|if\\|rep\\|while\\)?\\|q\\)\\|for\\|g\\(oto\\|[et]\\)"
+               "\\|i\\(f\\|nherits\\)\\|l[et]\\|mod\\|n\\(e\\|ot\\)\\|"
+               "o\\(n_ioerror\\|[fr]\\)\\|re\\(peat\\|turn\\)\\|then\\|"
+               "until\\|while\\|xor"
+               "\\)\\>"))
+
+       ;; Procedure declarations.  Fontify keyword plus procedure name.
+       ;; Function  declarations.  Fontify keyword plus function  name.
+       (pros-and-functions
+       '("\\<\\(function\\|pro\\)\\>[ \t]+\\(\\sw+\\(::\\sw+\\)?\\)"
+         (1 font-lock-keyword-face)
+         (2 font-lock-function-name-face nil t)))
+
+       ;; Common blocks
+       (common-blocks
+       '("\\<\\(common\\)\\>[ \t]*\\(\\sw+\\)?[ \t]*,?"
+         (1 font-lock-keyword-face)              ; "common"
+         (2 font-lock-reference-face nil t)      ; block name
+         (font-lock-match-c++-style-declaration-item-and-skip-to-next
+          ;; Start with point after block name and comma
+          (goto-char (match-end 0))  ; needed for XEmacs, could be nil 
+          nil
+          (1 font-lock-variable-name-face)       ; variable names
+          )))
+
+       ;; Batch files
+       (batch-files
+       '("^[ \t]*\\(@[^ \t\n]+\\)" (1 font-lock-string-face)))
+
+       ;; FIXME warning.
+       (fixme
+       '("\\<FIXME:" (0 font-lock-warning-face t)))
+
+       ;; Labels
+       (label
+       '("^[ \t]*\\([a-zA-Z]\\sw*:\\)" (1 font-lock-reference-face)))
+
+       ;; The goto statement and its label
+       (goto
+       '("\\(goto\\)[ \t]*,[ \t]*\\([a-zA-Z]\\sw*\\)"
+         (1 font-lock-keyword-face)
+         (2 font-lock-reference-face)))
+
+       ;; Named parameters, like /xlog or ,xrange=[]
+       ;; This is anchored to the comma preceeding the keyword.
+       ;; With continuation lines, works only during whole buffer fontification.
+       (keyword-parameters
+       '("[(,][ \t]*\\(\\$[ \t]*\n[ \t]*\\)?\\(/[a-zA-Z_]\\sw*\\|[a-zA-Z_]\\sw*[ \t]*=\\)"
+         (2 font-lock-reference-face)))
+
+       ;; System variables stars with a bang.
+       (system-variables
+       '("\\(![a-zA-Z_]+\\(\\.\\sw+\\)?\\)"
+         (1 font-lock-variable-name-face)))
+
+       ;; Special and unusual operators (not used because too noisy)
+       (special-operators
+       '("[<>#]" (0 font-lock-keyword-face)))
+
+       ;; All operators (not used because too noisy)
+       (all-operators
+       '("[-*^#+<>/]" (0 font-lock-keyword-face)))
+       
+       ;; Arrows with text property `idlwave-class'
+       (class-arrows
+       (list 'idlwave-match-class-arrows 
+             (list 0 (if (featurep 'xemacs) 
+                         idlwave-class-arrow-face
+                       'idlwave-class-arrow-face))))
+
+       )
+
+  ;; The following lines are just a dummy to make the compiler shut up
+  ;; about variables bound but not used.
+  (setq oldp oldp
+       idl-keywords idl-keywords
+       pros-and-functions pros-and-functions
+       common-blocks common-blocks
+       batch-files batch-files
+       fixme fixme
+       label label
+       goto goto
+       keyword-parameters keyword-parameters
+       system-variables system-variables
+       special-operators special-operators
+       all-operators all-operators
+       class-arrows class-arrows)
+
+  (setq idlwave-font-lock-keywords-1
+       (list pros-and-functions
+             batch-files
+             ))
+
+  (setq idlwave-font-lock-keywords-2
+       (mapcar 'symbol-value idlwave-default-font-lock-items))
+
+  (setq idlwave-font-lock-keywords-3 
+       (list pros-and-functions
+             batch-files
+             idl-keywords
+             label goto
+             common-blocks
+             keyword-parameters
+             system-variables
+             class-arrows
+             ))
+  )
+
+(defun idlwave-match-class-arrows (limit)
+  ;; Match an object arrow with class property
+  (and idlwave-store-inquired-class
+       (re-search-forward "->" limit 'limit)
+       (get-text-property (match-beginning 0) 'idlwave-class)))
+
+(defvar idlwave-font-lock-keywords idlwave-font-lock-keywords-2
+  "Default expressions to highlight in IDLWAVE mode.")
+
+(defvar idlwave-font-lock-defaults
+  '((idlwave-font-lock-keywords
+     idlwave-font-lock-keywords-1 
+     idlwave-font-lock-keywords-2
+     idlwave-font-lock-keywords-3)
+    nil t 
+    ((?$ . "w") (?_ . "w") (?. . "w")) 
+    beginning-of-line))
+
+(put 'idlwave-mode 'font-lock-defaults 
+     idlwave-font-lock-defaults) ; XEmacs
+
+(defconst idlwave-comment-line-start-skip "^[ \t]*;"
+  "Regexp to match the start of a full-line comment.
+That is the _beginning_ of a line containing a comment delimiter `;' preceded
+only by whitespace.")
+
+(defconst idlwave-begin-block-reg "\\<\\(pro\\|function\\|begin\\|case\\)\\>"
+  "Regular expression to find the beginning of a block. The case does
+not matter. The search skips matches in comments.")
+
+(defconst idlwave-begin-unit-reg "\\<\\(pro\\|function\\)\\>\\|\\`"
+  "Regular expression to find the beginning of a unit. The case does
+not matter.")
+
+(defconst idlwave-end-unit-reg "\\<\\(pro\\|function\\)\\>\\|\\'"
+  "Regular expression to find the line that indicates the end of unit.
+This line is the end of buffer or the start of another unit. The case does
+not matter. The search skips matches in comments.")
+
+(defconst idlwave-continue-line-reg "\\<\\$"
+  "Regular expression to match a continued line.")
+
+(defconst idlwave-end-block-reg
+  "\\<end\\(\\|case\\|else\\|for\\|if\\|rep\\|while\\)\\>"
+  "Regular expression to find the end of a block. The case does
+not matter. The search skips matches found in comments.")
+
+(defconst idlwave-block-matches
+  '(("pro"      . "end")
+    ("function" . "end")
+    ("case"     . "endcase")
+    ("else"     . "endelse")
+    ("for"      . "endfor")
+    ("then"     . "endif")
+    ("repeat"   . "endrep")
+    ("while"    . "endwhile"))
+  "Matches between statements and the corresponding END variant.
+The cars are the reserved words starting a block.  If the block really
+begins with BEGIN, the cars are the reserved words before the begin
+which can be used to identify the block type.
+This is used to check for the correct END type, to close blocks and
+to expand generic end statements to their detailed form.")
+
+(defconst idlwave-block-match-regexp
+  "\\<\\(else\\|for\\|then\\|repeat\\|while\\)\\>"
+"Regular expression matching reserved words which can stand before
+blocks starting with a BEGIN statement.  The matches must have associations
+`idlwave-block-matches'")
+
+(defconst idlwave-identifier "[a-zA-Z][a-zA-Z0-9$_]*"
+  "Regular expression matching an IDL identifier.")
+
+(defconst idlwave-sysvar (concat "!" idlwave-identifier)
+  "Regular expression matching IDL system variables.")
+
+(defconst idlwave-variable (concat idlwave-identifier "\\|" idlwave-sysvar)
+  "Regular expression matching IDL variable names.")
+
+(defconst idlwave-label (concat idlwave-identifier ":")
+  "Regular expression matching IDL labels.")
+
+(defconst idlwave-statement-match
+  (list
+   ;; "endif else" is the the only possible "end" that can be
+   ;; followed by a statement on the same line.
+   '(endelse . ("end\\(\\|if\\)\\s +else" "end\\(\\|if\\)\\s +else"))
+   ;; all other "end"s can not be followed by a statement.
+   (cons 'end (list idlwave-end-block-reg nil))
+   '(if . ("if\\>" "then"))
+   '(for . ("for\\>" "do"))
+   '(begin . ("begin\\>" nil))
+   '(pdef . ("pro\\>\\|function\\>" nil))
+   '(while . ("while\\>" "do"))
+   '(repeat . ("repeat\\>" "repeat"))
+   '(goto . ("goto\\>" nil))
+   '(case . ("case\\>" nil))
+   (cons 'call (list (concat idlwave-identifier "\\(\\s *$\\|\\s *,\\)") nil))
+   '(assign . ("[^=\n]*=" nil)))
+  
+  "Associated list of statement matching regular expressions.
+Each regular expression matches the start of an IDL statement.  The
+first element of each association is a symbol giving the statement
+type.  The associated value is a list.  The first element of this list
+is a regular expression matching the start of an IDL statement for
+identifying the statement type.  The second element of this list is a
+regular expression for finding a substatement for the type.  The
+substatement starts after the end of the found match modulo
+whitespace.  If it is nil then the statement has no substatement.  The
+list order matters since matching an assignment statement exactly is
+not possible without parsing.  Thus assignment statement become just
+the leftover unidentified statements containing and equal sign. "  )
+
+(defvar idlwave-fill-function 'auto-fill-function
+  "IDL mode auto fill function.")
+
+(defvar idlwave-comment-indent-function 'comment-indent-function
+  "IDL mode comment indent function.")
+
+;; Note that this is documented in the v18 manuals as being a string
+;; of length one rather than a single character.
+;; The code in this file accepts either format for compatibility.
+(defvar idlwave-comment-indent-char ?\ 
+  "Character to be inserted for IDL comment indentation.
+Normally a space.")
+
+(defconst idlwave-continuation-char ?$
+  "Character which is inserted as a last character on previous line by
+   \\[idlwave-split-line] to begin a continuation line.  Normally $.")
+
+(defconst idlwave-mode-version " 3.11")
+
+(defmacro idlwave-keyword-abbrev (&rest args)
+  "Creates a function for abbrev hooks to call `idlwave-check-abbrev' with args."
+  (` (quote (lambda ()
+              (, (append '(idlwave-check-abbrev) args))))))
+
+;; If I take the time I can replace idlwave-keyword-abbrev with
+;; idlwave-code-abbrev and remove the quoted abbrev check from
+;; idlwave-check-abbrev.  Then, e.g, (idlwave-keyword-abbrev 0 t) becomes
+;; (idlwave-code-abbrev idlwave-check-abbrev 0 t).  In fact I should change
+;; the name of idlwave-check-abbrev to something like idlwave-modify-abbrev.
+
+(defmacro idlwave-code-abbrev (&rest args)
+  "Creates a function for abbrev hooks that ensures abbrevs are not quoted.
+Specifically, if the abbrev is in a comment or string it is unexpanded.
+Otherwise ARGS forms a list that is evaluated."
+  (` (quote (lambda ()
+             (, (prin1-to-string args))  ;; Puts the code in the doc string
+              (if (idlwave-quoted) (progn (unexpand-abbrev) nil)
+                (, (append args)))))))
+
+(defvar idlwave-mode-map (make-sparse-keymap)
+  "Keymap used in IDL mode.")
+
+(defvar idlwave-mode-syntax-table (make-syntax-table)
+  "Syntax table in use in `idlwave-mode' buffers.")
+
+(modify-syntax-entry ?+   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?-   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?*   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?/   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?^   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?#   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?=   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?%   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?<   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?>   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?\'  "\"" idlwave-mode-syntax-table)
+(modify-syntax-entry ?\"  "\"" idlwave-mode-syntax-table)
+(modify-syntax-entry ?\\  "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?_   "_"  idlwave-mode-syntax-table)
+(modify-syntax-entry ?{   "(}" idlwave-mode-syntax-table)
+(modify-syntax-entry ?}   "){" idlwave-mode-syntax-table)
+(modify-syntax-entry ?$   "_"  idlwave-mode-syntax-table)
+(modify-syntax-entry ?.   "."  idlwave-mode-syntax-table)
+(modify-syntax-entry ?\;  "<"  idlwave-mode-syntax-table)
+(modify-syntax-entry ?\n  ">"  idlwave-mode-syntax-table)
+(modify-syntax-entry ?\f  ">"  idlwave-mode-syntax-table)
+
+(defvar idlwave-find-symbol-syntax-table
+  (copy-syntax-table idlwave-mode-syntax-table)
+  "Syntax table that treats symbol characters as word characters.")
+
+(modify-syntax-entry ?$   "w"  idlwave-find-symbol-syntax-table)
+(modify-syntax-entry ?_   "w"  idlwave-find-symbol-syntax-table)
+
+(defun idlwave-action-and-binding (key cmd &optional select)
+  "KEY and CMD are made into a key binding and an indent action.
+KEY is a string - same as for the `define-key' function.  CMD is a
+function of no arguments or a list to be evaluated.  CMD is bound to
+KEY in `idlwave-mode-map' by defining an anonymous function calling
+`self-insert-command' followed by CMD.  If KEY contains more than one
+character a binding will only be set if SELECT is 'both.
+
+(KEY . CMD\ is also placed in the `idlwave-indent-expand-table',
+replacing any previous value for KEY.  If a binding is not set then it
+will instead be placed in `idlwave-indent-action-table'.
+
+If the optional argument SELECT is nil then an action and binding are
+created.  If SELECT is 'noaction, then a binding is always set and no
+action is created.  If SELECT is 'both then an action and binding
+will both be created even if KEY contains more than one character.
+Otherwise, if SELECT is non-nil then only an action is created.
+
+Some examples:
+No spaces before and 1 after a comma
+   (idlwave-action-and-binding \",\"  '(idlwave-surround 0 1))
+A minimum of 1 space before and after `=' (see `idlwave-expand-equal').
+   (idlwave-action-and-binding \"=\"  '(idlwave-expand-equal -1 -1))
+Capitalize system variables - action only
+   (idlwave-action-and-binding idlwave-sysvar '(capitalize-word 1) t)"
+  (if (not (equal select 'noaction))
+      ;; Add action
+      (let* ((table (if select 'idlwave-indent-action-table
+                      'idlwave-indent-expand-table))
+             (cell (assoc key (eval table))))
+        (if cell
+            ;; Replace action command
+            (setcdr cell cmd)
+          ;; New action
+          (set table (append (eval table) (list (cons key cmd)))))))
+  ;; Make key binding for action
+  (if (or (and (null select) (= (length key) 1))
+          (equal select 'noaction)
+          (equal select 'both))
+      (define-key idlwave-mode-map key
+        (append '(lambda ()
+                            (interactive)
+                            (self-insert-command 1))
+                (list (if (listp cmd)
+                          cmd
+                        (list cmd)))))))
+
+(fset 'idlwave-debug-map (make-sparse-keymap))
+
+(define-key idlwave-mode-map "'"        'idlwave-show-matching-quote)
+(define-key idlwave-mode-map "\""       'idlwave-show-matching-quote)
+(define-key idlwave-mode-map "\C-c;"    'idlwave-toggle-comment-region)
+(define-key idlwave-mode-map "\C-\M-a"  'idlwave-beginning-of-subprogram)
+(define-key idlwave-mode-map "\C-\M-e"  'idlwave-end-of-subprogram)
+(define-key idlwave-mode-map "\C-c{"    'idlwave-beginning-of-block)
+(define-key idlwave-mode-map "\C-c}"    'idlwave-end-of-block)
+(define-key idlwave-mode-map "\C-c]"    'idlwave-close-block)
+(define-key idlwave-mode-map "\M-\C-h"  'idlwave-mark-subprogram)
+(define-key idlwave-mode-map "\M-\C-n"  'idlwave-forward-block)
+(define-key idlwave-mode-map "\M-\C-p"  'idlwave-backward-block)
+(define-key idlwave-mode-map "\M-\C-d"  'idlwave-down-block)
+(define-key idlwave-mode-map "\M-\C-u"  'idlwave-backward-up-block)
+(define-key idlwave-mode-map "\M-\r"    'idlwave-split-line)
+(define-key idlwave-mode-map "\M-\C-q"  'idlwave-indent-subprogram)
+(define-key idlwave-mode-map "\C-c\C-p" 'idlwave-previous-statement)
+(define-key idlwave-mode-map "\C-c\C-n" 'idlwave-next-statement)
+;; (define-key idlwave-mode-map "\r"       'idlwave-newline)
+;; (define-key idlwave-mode-map "\t"       'idlwave-indent-line)
+(define-key idlwave-mode-map "\C-c\C-a" 'idlwave-auto-fill-mode)
+(define-key idlwave-mode-map "\M-q"     'idlwave-fill-paragraph)
+(define-key idlwave-mode-map "\M-s"     'idlwave-edit-in-idlde)
+(define-key idlwave-mode-map "\C-c\C-h" 'idlwave-doc-header)
+(define-key idlwave-mode-map "\C-c\C-m" 'idlwave-doc-modification)
+(define-key idlwave-mode-map "\C-c\C-c" 'idlwave-case)
+(define-key idlwave-mode-map "\C-c\C-d" 'idlwave-debug-map)
+(define-key idlwave-mode-map "\C-c\C-d\C-c" 'idlwave-shell-save-and-run)
+(define-key idlwave-mode-map "\C-c\C-f" 'idlwave-for)
+;;  (define-key idlwave-mode-map "\C-c\C-f" 'idlwave-function)
+;;  (define-key idlwave-mode-map "\C-c\C-p" 'idlwave-procedure)
+(define-key idlwave-mode-map "\C-c\C-r" 'idlwave-repeat)
+(define-key idlwave-mode-map "\C-c\C-w" 'idlwave-while)
+(define-key idlwave-mode-map "\C-c\C-s" 'idlwave-shell)
+(define-key idlwave-mode-map "\C-c\C-l" 'idlwave-shell-recenter-shell-window)
+(autoload 'idlwave-shell-send-command "idlwave-shell")
+(autoload 'idlwave-shell-recenter-shell-window "idlwave-shell"
+  "Run `idlwave-shell' and switch back to current window" t)
+(autoload 'idlwave-shell-save-and-run "idlwave-shell"
+  "Save and run buffer under the shell." t)
+(define-key idlwave-mode-map "\C-c\C-v"   'idlwave-find-module)
+(define-key idlwave-mode-map "\C-c?"      'idlwave-routine-info)
+(define-key idlwave-mode-map "\M-?"       'idlwave-routine-info-from-idlhelp)
+(define-key idlwave-mode-map [(meta tab)] 'idlwave-complete)
+(define-key idlwave-mode-map "\C-c\C-i"   'idlwave-update-routine-info)
+(define-key idlwave-mode-map "\C-c="      'idlwave-resolve)
+
+;; Set action and key bindings.
+;; See description of the function `idlwave-action-and-binding'.
+;; Automatically add spaces for the following characters
+(idlwave-action-and-binding "&"  '(idlwave-surround -1 -1))
+(idlwave-action-and-binding "<"  '(idlwave-surround -1 -1))
+(idlwave-action-and-binding ">"  '(idlwave-surround -1 -1 '(?-)))
+(idlwave-action-and-binding ","  '(idlwave-surround 0 -1))
+;; Automatically add spaces to equal sign if not keyword
+(idlwave-action-and-binding "="  '(idlwave-expand-equal -1 -1))
+
+;;;
+;;; Abbrev Section
+;;;
+;;; When expanding abbrevs and the abbrev hook moves backward, an extra
+;;; space is inserted (this is the space typed by the user to expanded
+;;; the abbrev).
+;;;
+
+(condition-case nil
+    (modify-syntax-entry (string-to-char idlwave-abbrev-start-char) 
+                        "w" idlwave-mode-syntax-table)
+  (error nil))
+
+(defvar idlwave-mode-abbrev-table nil
+  "Abbreviation table used for IDLWAVE mode")
+(define-abbrev-table 'idlwave-mode-abbrev-table ())
+(let ((abbrevs-changed nil)          ;; mask the current value to avoid save
+      (tb idlwave-mode-abbrev-table)
+      (c idlwave-abbrev-start-char))
+  ;;
+  ;; Templates
+  ;;
+  (define-abbrev tb (concat c "c")   "" (idlwave-code-abbrev idlwave-case))
+  (define-abbrev tb (concat c "f")   "" (idlwave-code-abbrev idlwave-for))
+  (define-abbrev tb (concat c "fu")  "" (idlwave-code-abbrev idlwave-function))
+  (define-abbrev tb (concat c "pr")  "" (idlwave-code-abbrev idlwave-procedure))
+  (define-abbrev tb (concat c "r")   "" (idlwave-code-abbrev idlwave-repeat))
+  (define-abbrev tb (concat c "w")   "" (idlwave-code-abbrev idlwave-while))
+  (define-abbrev tb (concat c "i")   "" (idlwave-code-abbrev idlwave-if))
+  (define-abbrev tb (concat c "elif") "" (idlwave-code-abbrev idlwave-elif))
+  ;;
+  ;; Keywords, system functions, conversion routines
+  ;;
+  (define-abbrev tb (concat c "b")  "begin"        (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb (concat c "co") "common"       (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb (concat c "cb") "byte()"       (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "cx") "fix()"        (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "cl") "long()"       (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "cf") "float()"      (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "cs") "string()"     (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "cc") "complex()"    (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "cd") "double()"     (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "e")  "else"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb (concat c "ec") "endcase"      'idlwave-show-begin)
+  (define-abbrev tb (concat c "ee") "endelse"      'idlwave-show-begin)
+  (define-abbrev tb (concat c "ef") "endfor"       'idlwave-show-begin)
+  (define-abbrev tb (concat c "ei") "endif else if" 'idlwave-show-begin)
+  (define-abbrev tb (concat c "el") "endif else"   'idlwave-show-begin)
+  (define-abbrev tb (concat c "en") "endif"        'idlwave-show-begin)
+  (define-abbrev tb (concat c "er") "endrep"       'idlwave-show-begin)
+  (define-abbrev tb (concat c "ew") "endwhile"     'idlwave-show-begin)
+  (define-abbrev tb (concat c "g")  "goto,"        (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb (concat c "h")  "help,"        (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "k")  "keyword_set()" (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "n")  "n_elements()" (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "on") "on_error,"    (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "oi") "on_ioerror,"  (idlwave-keyword-abbrev 0 1))
+  (define-abbrev tb (concat c "ow") "openw,"       (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "or") "openr,"       (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "ou") "openu,"       (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "p")  "print,"       (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "pt") "plot,"        (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "re") "read,"        (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "rf") "readf,"       (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "ru") "readu,"       (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "rt") "return"       (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "sc") "strcompress()" (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "sn") "strlen()"     (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "sl") "strlowcase()" (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "su") "strupcase()"  (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "sm") "strmid()"     (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "sp") "strpos()"     (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "st") "strput()"     (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "sr") "strtrim()"    (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "t")  "then"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb (concat c "u")  "until"        (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb (concat c "wu") "writeu,"      (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "ine") "if n_elements() eq 0 then"
+    (idlwave-keyword-abbrev 11))
+  (define-abbrev tb (concat c "inn") "if n_elements() ne 0 then"
+    (idlwave-keyword-abbrev 11))
+  (define-abbrev tb (concat c "np") "n_params()"   (idlwave-keyword-abbrev 0))
+  (define-abbrev tb (concat c "s")  "size()"       (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "wi") "widget_info()" (idlwave-keyword-abbrev 1))
+  (define-abbrev tb (concat c "wc") "widget_control," (idlwave-keyword-abbrev 0))
+  
+  ;; This section is reserved words only. (From IDL user manual)
+  ;;
+  (define-abbrev tb "and"        "and"        (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "begin"      "begin"      (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "case"       "case"       (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "common"     "common"     (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "do"         "do"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "else"       "else"       (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "end"        "end"        'idlwave-show-begin-check)
+  (define-abbrev tb "endcase"    "endcase"    'idlwave-show-begin-check)
+  (define-abbrev tb "endelse"    "endelse"    'idlwave-show-begin-check)
+  (define-abbrev tb "endfor"     "endfor"     'idlwave-show-begin-check)
+  (define-abbrev tb "endif"      "endif"      'idlwave-show-begin-check)
+  (define-abbrev tb "endrep"     "endrep"     'idlwave-show-begin-check)
+  (define-abbrev tb "endwhi"     "endwhi"     'idlwave-show-begin-check)
+  (define-abbrev tb "endwhile"   "endwhile"   'idlwave-show-begin-check)
+  (define-abbrev tb "eq"         "eq"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "for"        "for"        (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "function"   "function"   (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "ge"         "ge"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "goto"       "goto"       (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "gt"         "gt"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "if"         "if"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "le"         "le"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "lt"         "lt"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "mod"        "mod"        (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "ne"         "ne"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "not"        "not"        (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "of"         "of"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "on_ioerror" "on_ioerror" (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "or"         "or"         (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "pro"        "pro"        (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "repeat"     "repeat"     (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "then"       "then"       (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "until"      "until"      (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "while"      "while"      (idlwave-keyword-abbrev 0 t))
+  (define-abbrev tb "xor"        "xor"        (idlwave-keyword-abbrev 0 t)))
+
+(defvar imenu-create-index-function)
+(defvar extract-index-name-function)
+(defvar prev-index-position-function)
+(defvar imenu-extract-index-name-function)
+(defvar imenu-prev-index-position-function)
+;; defined later - so just make the compiler shut up
+(defvar idlwave-mode-menu)  
+(defvar idlwave-mode-debug-menu)
+
+;;;###autoload
+(defun idlwave-mode ()
+  "Major mode for editing IDL and WAVE CL .pro files.
+
+The main features of this mode are
+
+1. Indentation and Formatting
+   --------------------------
+   Like other Emacs programming modes, C-j inserts a newline and indents.
+   TAB is used for explicit indentation of the current line.
+
+   To start a continuation line, use \\[idlwave-split-line].  This function can also
+   be used in the middle of a line to split the line at that point.
+   When used inside a long constant string, the string is split at
+   that point with the `+' concatenation operator.
+
+   Comments are indented as follows:
+
+   `;;;' Indentation remains unchanged.
+   `;;'  Indent like the surrounding code
+   `;'   Indent to a minimum column.
+
+   The indentation of comments starting in column 0 is never changed.
+
+   Use \\[idlwave-fill-paragraph] to refill a paragraph inside a comment.  The indentation
+   of the second line of the paragraph relative to the first will be
+   retained.  Use \\[idlwave-auto-fill-mode] to toggle auto-fill mode for these comments.
+   When the variable `idlwave-fill-comment-line-only' is nil, code
+   can also be auto-filled and auto-indented (not recommended).
+
+   To convert pre-existing IDL code to your formatting style, mark the
+   entire buffer with \\[mark-whole-buffer] and execute \\[idlwave-expand-region-abbrevs].
+   Then mark the entire buffer again followed by \\[indent-region] (`indent-region').
+
+2. Routine Info
+   ------------
+   IDLWAVE displays information about the calling sequence and the accepted
+   keyword parameters of a procedure or function with \\[idlwave-routine-info].
+   \\[idlwave-find-module] jumps to the source file of a module.
+   These commands know about system routines, all routines in idlwave-mode
+   buffers and (when the idlwave-shell is active) about all modules
+   currently compiled under this shell.  Use \\[idlwave-update-routine-info] to update this
+   information, which is also used for completion (see next item).
+
+3. Completion
+   ----------
+   \\[idlwave-complete] completes the names of procedures, functions and
+   keyword parameters.  It is context sensitive and figures out what
+   is expected at point (procedure/function/keyword).  Lower case
+   strings are completed in lower case, other strings in mixed or
+   upper case.
+
+4. Code Templates and Abbreviations
+   --------------------------------
+   Many Abbreviations are predefined to expand to code fragments and templates.
+   The abbreviations start generally with a `\\`.  Some examples
+
+   \\pr        PROCEDURE template
+   \\fu        FUNCTION template
+   \\c         CASE statement template
+   \\f         FOR loop template
+   \\r         REPEAT Loop template
+   \\w         WHILE loop template
+   \\i         IF statement template
+   \\elif      IF-ELSE statement template
+   \\b         BEGIN
+   
+   For a full list, use \\[idlwave-list-abbrevs].  Some templates also have
+   direct keybindings - see the list of keybindings below.
+
+   \\[idlwave-doc-header] inserts a documentation header at the beginning of the
+   current program unit (pro, function or main).  Change log entries
+   can be added to the current program unit with \\[idlwave-doc-modification].
+
+5. Automatic Case Conversion
+   -------------------------
+   The case of reserved words and some abbrevs is controlled by
+   `idlwave-reserved-word-upcase' and `idlwave-abbrev-change-case'.
+
+6. Automatic END completion
+   ------------------------
+   If the variable `idlwave-expand-generic-end' is non-nil, each END typed
+   will be converted to the specific version, like ENDIF, ENDFOR, etc.
+
+7. Hooks
+   -----
+   Loading idlwave.el runs `idlwave-load-hook'.
+   Turning on `idlwave-mode' runs `idlwave-mode-hook'.
+
+8. Documentation and Customization
+   -------------------------------
+   Info documentation for this package is available.  Use \\[idlwave-info]
+   to display (complain to your sysadmin if that does not work).
+   For Postscript and HTML versions of the documentation, check IDLWAVE's
+   homepage at `http://www.strw.leidenuniv.nl/~dominik/Tools/idlwave'.
+   IDLWAVE has customize support - see the group `idlwave'.
+
+9. Keybindings
+   -----------
+   Here is a list of all keybindings of this mode.
+   If some of the key bindings below show with ??, use \\[describe-key]
+   followed by the key sequence to see what the key sequence does.
+
+\\{idlwave-mode-map}"
+
+  (interactive)
+  (kill-all-local-variables)
+  
+  (if idlwave-startup-message
+      (message "Emacs IDLWAVE mode version %s." idlwave-mode-version))
+  (setq idlwave-startup-message nil)
+  
+  (setq local-abbrev-table idlwave-mode-abbrev-table)
+  (set-syntax-table idlwave-mode-syntax-table)
+  
+  (set (make-local-variable 'indent-line-function) 'idlwave-indent-and-action)
+  
+  (make-local-variable idlwave-comment-indent-function)
+  (set idlwave-comment-indent-function 'idlwave-comment-hook)
+  
+  (set (make-local-variable 'comment-start-skip) ";+[ \t]*")
+  (set (make-local-variable 'comment-start) ";")
+  (set (make-local-variable 'require-final-newline) t)
+  (set (make-local-variable 'abbrev-all-caps) t)
+  (set (make-local-variable 'indent-tabs-mode) nil)
+  (set (make-local-variable 'completion-ignore-case) t)
+  
+  (use-local-map idlwave-mode-map)
+
+  (when (featurep 'easymenu)
+    (easy-menu-add idlwave-mode-menu idlwave-mode-map)
+    (easy-menu-add idlwave-mode-debug-menu idlwave-mode-map))
+
+  (setq mode-name "IDLWAVE")
+  (setq major-mode 'idlwave-mode)
+  (setq abbrev-mode t)
+  
+  (set (make-local-variable idlwave-fill-function) 'idlwave-auto-fill)
+  (setq comment-end "")
+  (set (make-local-variable 'comment-multi-line) nil)
+  (set (make-local-variable 'paragraph-separate) "[ \t\f]*$\\|[ \t]*;+[ \t]*$")
+  (set (make-local-variable 'paragraph-start) "[ \t\f]\\|[ \t]*;+[ \t]")
+  (set (make-local-variable 'paragraph-ignore-fill-prefix) nil)
+  (set (make-local-variable 'parse-sexp-ignore-comments) nil)
+  
+  ;; Set tag table list to use IDLTAGS as file name.
+  (if (boundp 'tag-table-alist)
+      (add-to-list 'tag-table-alist '("\\.pro$" . "IDLTAGS")))
+  
+  ;; Font-lock additions - originally Phil Williams, then Ulrik Dickow
+  ;; Following line is for Emacs - XEmacs uses the corresponding porperty
+  ;; on the `idlwave-mode' symbol.
+  (set (make-local-variable 'font-lock-defaults) idlwave-font-lock-defaults)
+
+  ;; Imenu setup
+  (set (make-local-variable 'imenu-create-index-function)
+       'imenu-default-create-index-function)
+  (set (make-local-variable 'imenu-extract-index-name-function)
+       'idlwave-unit-name)
+  (set (make-local-variable 'imenu-prev-index-position-function)
+       'idlwave-prev-index-position)
+
+  ;; Make a local post-command-hook and add our hook to it
+  (make-local-hook 'post-command-hook)
+  (add-hook 'post-command-hook 'idlwave-command-hook nil t)
+
+  ;; Run the mode hook
+  (run-hooks 'idlwave-mode-hook))
+
+;;
+;;  Done with start up and initialization code.
+;;  The remaining routines are the code formatting functions.
+;;
+
+(defun idlwave-push-mark (&rest rest)
+  "Push mark for compatibility with Emacs 18/19."
+  (if (fboundp 'iconify-frame)
+      (apply 'push-mark rest)
+    (push-mark)))
+
+(defun idlwave-hard-tab ()
+  "Inserts TAB in buffer in current position."
+  (interactive)
+  (insert "\t"))
+
+;;; This stuff is experimental
+
+(defvar idlwave-command-hook nil
+  "If non-nil, a list that can be evaluated using `eval'.
+It is evaluated in the lisp function `idlwave-command-hook' which is
+placed in `post-command-hook'.")
+
+(defun idlwave-command-hook ()
+  "Command run after every command.
+Evaluates a non-nil value of the *variable* `idlwave-command-hook' and
+sets the variable to zero afterwards."
+  (and idlwave-command-hook
+       (listp idlwave-command-hook)
+       (condition-case nil
+          (eval idlwave-command-hook)
+        (error nil)))
+  (setq idlwave-command-hook nil))
+
+;;; End experiment
+
+;; It would be better to use expand.el for better abbrev handling and
+;; versatility.
+
+(defun idlwave-check-abbrev (arg &optional reserved)
+  "Reverses abbrev expansion if in comment or string.
+Argument ARG is the number of characters to move point
+backward if `idlwave-abbrev-move' is non-nil.
+If optional argument RESERVED is non-nil then the expansion
+consists of reserved words, which will be capitalized if
+`idlwave-reserved-word-upcase' is non-nil.
+Otherwise, the abbrev will be capitalized if `idlwave-abbrev-change-case'
+is non-nil, unless its value is \`down in which case the abbrev will be
+made into all lowercase.
+Returns non-nil if abbrev is left expanded."
+  (if (idlwave-quoted)
+      (progn (unexpand-abbrev)
+             nil)
+    (if (and reserved idlwave-reserved-word-upcase)
+        (upcase-region last-abbrev-location (point))
+      (cond
+       ((equal idlwave-abbrev-change-case 'down)
+        (downcase-region last-abbrev-location (point)))
+       (idlwave-abbrev-change-case
+        (upcase-region last-abbrev-location (point)))))
+    (if (and idlwave-abbrev-move (> arg 0))
+        (if (boundp 'post-command-hook)
+            (setq idlwave-command-hook (list 'backward-char (1+ arg)))
+          (backward-char arg)))
+    t))
+
+(defun idlwave-in-comment ()
+  "Returns t if point is inside a comment, nil otherwise."
+  (save-excursion
+    (let ((here (point)))
+      (and (idlwave-goto-comment) (> here (point))))))
+
+(defun idlwave-goto-comment ()
+  "Move to start of comment delimiter on current line.
+Moves to end of line if there is no comment delimiter.
+Ignores comment delimiters in strings.
+Returns point if comment found and nil otherwise."
+  (let ((eos (progn (end-of-line) (point)))
+        (data (match-data))
+        found)
+    ;; Look for first comment delimiter not in a string
+    (beginning-of-line)
+    (setq found (search-forward comment-start eos 'lim))
+    (while (and found (idlwave-in-quote))
+      (setq found (search-forward comment-start eos 'lim)))
+    (store-match-data data)
+    (and found (not (idlwave-in-quote))
+         (progn
+           (backward-char 1)
+           (point)))))
+
+(defun idlwave-show-matching-quote ()
+  "Insert quote and show matching quote if this is end of a string."
+  (interactive)
+  (let ((bq (idlwave-in-quote))
+        (inq last-command-char))
+    (if (and bq (not (idlwave-in-comment)))
+        (let ((delim (char-after bq)))
+          (insert inq)
+          (if (eq inq delim)
+              (save-excursion
+                (goto-char bq)
+                (sit-for 1))))
+      ;; Not the end of a string
+      (insert inq))))
+
+(defun idlwave-show-begin-check ()
+  "Ensure that the previous word was a token before `idlwave-show-begin'.
+An END token must be preceded by whitespace."
+  (if
+      (save-excursion
+        (backward-word 1)
+        (backward-char 1)
+        (looking-at "[ \t\n\f]"))
+      (idlwave-show-begin)))
+
+(defun idlwave-show-begin ()
+  "Finds the start of current block and blinks to it for a second.
+Also checks if the correct end statement has been used."
+  ;; All end statements are reserved words
+  (let* ((pos (point))
+        end end1)
+    (when (and (idlwave-check-abbrev 0 t)
+              idlwave-show-block)
+      (save-excursion
+       ;; Move inside current block
+       (setq end (buffer-substring 
+                  (save-excursion (skip-chars-backward "a-zA-Z")
+                                  (point))
+                  (point)))
+       (idlwave-beginning-of-statement)
+       (idlwave-block-jump-out -1 'nomark)
+       (when (setq end1 (cdr (idlwave-block-master)))
+         (cond
+          ((null end1)) ; no-opeartion
+          ((string= (downcase end) (downcase end1))
+           (sit-for 1))
+          ((string= (downcase end) "end")
+           ;; A generic end
+           (if idlwave-expand-generic-end
+               (save-excursion
+                 (goto-char pos)
+                 (backward-char 3)
+                 (insert (if (string= end "END") (upcase end1) end1))
+                 (delete-char 3)))
+           (sit-for 1))
+          (t
+           (beep)
+           (message "Warning: Shouldn't this be \"%s\" instead of \"%s\"?" 
+                    end1 end)
+           (sit-for 1))))))))
+
+(defun idlwave-block-master ()
+  (let ((case-fold-search t))
+    (save-excursion
+      (cond
+       ((looking-at "pro\\|case\\|function\\>")
+       (assoc (downcase (match-string 0)) idlwave-block-matches))
+       ((looking-at "begin\\>")
+       (let ((limit (save-excursion 
+                      (idlwave-beginning-of-statement) 
+                      (point))))
+         (cond
+          ((re-search-backward idlwave-block-match-regexp limit t)
+           (assoc (downcase (match-string 1))
+                  idlwave-block-matches))
+          ;;((re-search-backward ":[ \t]*\\=" limit t)
+          ;; ;; seems to be a case thing
+          ;; '("begin" . "end"))
+          (t
+           ;; Just a nromal block
+           '("begin" . "end")))))
+       (t nil)))))
+
+(defun idlwave-close-block ()
+  "Terminate the current block with the correct END statement."
+  (interactive)
+
+  ;; Start new line if we are not in a new line
+  (unless (save-excursion
+           (skip-chars-backward " \t")
+           (bolp))
+    (let ((idlwave-show-block nil))
+      (newline-and-indent)))
+
+  ;; Check which end is needed and insert it.
+  (let ((case-fold-search t) end)
+    (save-excursion
+      (idlwave-beginning-of-statement)
+      (idlwave-block-jump-out -1 'nomark)
+      (if (setq end (idlwave-block-master))
+         (setq end (cdr end))
+       (error "Cannot close block")))
+    (insert end)
+    (idlwave-newline)))
+
+(defun idlwave-surround (&optional before after escape-chars)
+  "Surround the character before point with blanks.
+Optional arguments BEFORE and AFTER affect the behavior before and
+after the previous character. See description of `idlwave-make-space'.
+
+The function does nothing if any of the following conditions is true:
+- `idlwave-surround-by-blank' is nil
+- the character before point is inside a string or comment
+
+When the character 2 positions before point is a member of
+ESCAPE-CHARS, BEFORE is forced to nil."
+
+  (if (and idlwave-surround-by-blank
+          (not (idlwave-quoted)))
+      (progn
+       (if (memq (char-after (- (point) 2)) escape-chars)
+           (setq before nil))
+        (backward-char 1)
+        (save-restriction
+          (let ((here (point)))
+            (skip-chars-backward " \t")
+            (if (bolp)
+                ;; avoid clobbering indent
+                (progn
+                  (move-to-column (idlwave-calculate-indent))
+                  (if (<= (point) here)
+                      (narrow-to-region (point) here))
+                  (goto-char here)))
+            (idlwave-make-space before))
+          (skip-chars-forward " \t"))
+        (forward-char 1)
+        (idlwave-make-space after)
+        ;; Check to see if the line should auto wrap
+        (if (and (equal (char-after (1- (point))) ? )
+                 (> (current-column) fill-column))
+            (funcall auto-fill-function)))))
+
+(defun idlwave-make-space (n)
+  "Make space at point.
+The space affected is all the spaces and tabs around point.
+If n is non-nil then point is left abs(n) spaces from the beginning of
+the contiguous space.
+The amount of space at point is determined by N.
+If the value of N is:
+nil   - do nothing.
+c > 0 - exactly c spaces.
+c < 0 - a minimum of -c spaces, i.e., do not change if there are
+        already -c spaces.
+0     - no spaces."
+  (if (integerp n)
+      (let
+          ((start-col (progn (skip-chars-backward " \t") (current-column)))
+           (left (point))
+           (end-col (progn (skip-chars-forward " \t") (current-column))))
+        (delete-horizontal-space)
+        (cond
+         ((> n 0)
+          (idlwave-indent-to (+ start-col n))
+          (goto-char (+ left n)))
+         ((< n 0)
+          (idlwave-indent-to end-col (- n))
+          (goto-char (- left n)))
+         ;; n = 0, done
+         ))))
+
+(defun idlwave-newline ()
+  "Inserts a newline and indents the current and previous line."
+  (interactive)
+  ;;
+  ;; Handle unterminated single and double quotes
+  ;; If not in a comment and in a string then insertion of a newline
+  ;; will mean unbalanced quotes.
+  ;;
+  (if (and (not (idlwave-in-comment)) (idlwave-in-quote))
+      (progn (beep)
+             (message "Warning: unbalanced quotes?")))
+  (newline)
+  ;;
+  ;; The current line is being split, the cursor should be at the
+  ;; beginning of the new line skipping the leading indentation.
+  ;;
+  ;; The reason we insert the new line before indenting is that the
+  ;; indenting could be confused by keywords (e.g. END) on the line
+  ;; after the split point.  This prevents us from just using
+  ;; `indent-for-tab-command' followed by `newline-and-indent'.
+  ;;
+  (beginning-of-line 0)
+  (idlwave-indent-line)
+  (forward-line)
+  (idlwave-indent-line))
+
+;;
+;;  Use global variable 'comment-column' to set parallel comment
+;;
+;; Modeled on lisp.el
+;; Emacs Lisp and IDL (Wave CL) have identical comment syntax
+(defun idlwave-comment-hook ()
+  "Compute indent for the beginning of the IDL comment delimiter."
+  (if (or (looking-at idlwave-no-change-comment)
+          (if idlwave-begin-line-comment
+              (looking-at idlwave-begin-line-comment)
+           (looking-at "^;")))
+      (current-column)
+    (if (looking-at idlwave-code-comment)
+        (if (save-excursion (skip-chars-backward " \t") (bolp))
+            ;; On line by itself, indent as code
+            (let ((tem (idlwave-calculate-indent)))
+              (if (listp tem) (car tem) tem))
+          ;; after code - do not change
+          (current-column))
+      (skip-chars-backward " \t")
+      (max (if (bolp) 0 (1+ (current-column)))
+           comment-column))))
+
+(defun idlwave-split-line ()
+  "Continue line by breaking line at point and indent the lines.
+For a code line insert continuation marker. If the line is a line comment
+then the new line will contain a comment with the same indentation.
+Splits strings with the IDL operator `+' if `idlwave-split-line-string' is
+non-nil."
+  (interactive)
+  (let (beg)
+    (if (not (idlwave-in-comment))
+        ;; For code line add continuation.
+        ;; Check if splitting a string.
+        (progn
+          (if (setq beg (idlwave-in-quote))
+              (if idlwave-split-line-string
+                  ;; Split the string.
+                  (progn (insert (setq beg (char-after beg)) " + "
+                                 idlwave-continuation-char beg)
+                         (backward-char 1))
+                ;; Do not split the string.
+                (beep)
+                (message "Warning: continuation inside string!!")
+                (insert " " idlwave-continuation-char))
+            ;; Not splitting a string.
+            (insert " " idlwave-continuation-char))
+          (newline-and-indent))
+      (indent-new-comment-line))
+    ;; Indent previous line
+    (setq beg (- (point-max) (point)))
+    (forward-line -1)
+    (idlwave-indent-line)
+    (goto-char (- (point-max) beg))
+    ;; Reindent new line
+    (idlwave-indent-line)))
+
+(defun idlwave-beginning-of-subprogram ()
+  "Moves point to the beginning of the current program unit."
+  (interactive)
+  (idlwave-find-key idlwave-begin-unit-reg -1))
+
+(defun idlwave-end-of-subprogram ()
+  "Moves point to the start of the next program unit."
+  (interactive)
+  (idlwave-end-of-statement)
+  (idlwave-find-key idlwave-end-unit-reg 1))
+
+(defun idlwave-mark-statement ()
+  "Mark current IDL statement."
+  (interactive)
+  (idlwave-end-of-statement)
+  (let ((end (point)))
+    (idlwave-beginning-of-statement)
+    (idlwave-push-mark end nil t)))
+
+(defun idlwave-mark-block ()
+  "Mark containing block."
+  (interactive)
+  (idlwave-end-of-statement)
+  (idlwave-backward-up-block -1)
+  (idlwave-end-of-statement)
+  (let ((end (point)))
+    (idlwave-backward-block)
+    (idlwave-beginning-of-statement)
+    (idlwave-push-mark end nil t)))
+
+
+(defun idlwave-mark-subprogram ()
+  "Put mark at beginning of program, point at end.
+The marks are pushed."
+  (interactive)
+  (idlwave-end-of-statement)
+  (idlwave-beginning-of-subprogram)
+  (let ((beg (point)))
+    (idlwave-forward-block)
+    (idlwave-push-mark beg nil t))
+  (exchange-point-and-mark))
+
+(defun idlwave-backward-up-block (&optional arg)
+  "Move to beginning of enclosing block if prefix ARG >= 0.
+If prefix ARG < 0 then move forward to enclosing block end."
+  (interactive "p")
+  (idlwave-block-jump-out (- arg) 'nomark))
+
+(defun idlwave-beginning-of-block ()
+  "Go to the beginning of the current block."
+  (interactive)
+  (idlwave-block-jump-out -1 'nomark)
+  (forward-word 1))
+
+(defun idlwave-end-of-block ()
+  "Go to the beginning of the current block."
+  (interactive)
+  (idlwave-block-jump-out 1 'nomark)
+  (backward-word 1))
+
+(defun idlwave-forward-block ()
+  "Move across next nested block."
+  (interactive)
+  (if (idlwave-down-block 1)
+      (idlwave-block-jump-out 1 'nomark)))
+
+(defun idlwave-backward-block ()
+  "Move backward across previous nested block."
+  (interactive)
+  (if (idlwave-down-block -1)
+      (idlwave-block-jump-out -1 'nomark)))
+
+(defun idlwave-down-block (&optional arg)
+  "Go down a block.
+With ARG: ARG >= 0 go forwards, ARG < 0 go backwards.
+Returns non-nil if successfull."
+  (interactive "p")
+  (let (status)
+    (if (< arg 0)
+        ;; Backward
+        (let ((eos (save-excursion
+                     (idlwave-block-jump-out -1 'nomark)
+                     (point))))
+          (if (setq status (idlwave-find-key 
+                           idlwave-end-block-reg -1 'nomark eos))
+              (idlwave-beginning-of-statement)
+            (message "No nested block before beginning of containing block.")))
+      ;; Forward
+      (let ((eos (save-excursion
+                   (idlwave-block-jump-out 1 'nomark)
+                   (point))))
+        (if (setq status (idlwave-find-key 
+                         idlwave-begin-block-reg 1 'nomark eos))
+            (idlwave-end-of-statement)
+          (message "No nested block before end of containing block."))))
+    status))
+
+(defun idlwave-mark-doclib ()
+  "Put point at beginning of doc library header, mark at end.
+The marks are pushed."
+  (interactive)
+  (let (beg
+        (here (point)))
+    (goto-char (point-max))
+    (if (re-search-backward idlwave-doclib-start nil t)
+        (progn 
+         (setq beg (progn (beginning-of-line) (point)))
+         (if (re-search-forward idlwave-doclib-end nil t)
+             (progn
+               (forward-line 1)
+               (idlwave-push-mark beg nil t)
+               (message "Could not find end of doc library header.")))
+         (message "Could not find doc library header start.")
+         (goto-char here)))))
+
+(defvar idlwave-shell-prompt-pattern)
+(defun idlwave-beginning-of-statement ()
+  "Move to beginning of the current statement.
+Skips back past statement continuations.
+Point is placed at the beginning of the line whether or not this is an
+actual statement."
+  (interactive)
+  (cond
+   ((eq major-mode 'idlwave-shell-mode)
+    (if (re-search-backward idlwave-shell-prompt-pattern nil t)
+       (goto-char (match-end 0))))
+   (t  
+    (if (save-excursion (forward-line -1) (idlwave-is-continuation-line))
+       (idlwave-previous-statement)
+      (beginning-of-line)))))
+
+(defun idlwave-previous-statement ()
+  "Moves point to beginning of the previous statement.
+Returns t if the current line before moving is the beginning of
+the first non-comment statement in the file, and nil otherwise."
+  (interactive)
+  (let (first-statement)
+    (if (not (= (forward-line -1) 0))
+        ;; first line in file
+        t
+      ;; skip blank lines, label lines, include lines and line comments
+      (while (and
+              ;; The current statement is the first statement until we
+              ;; reach another statement.
+              (setq first-statement
+                    (or
+                     (looking-at idlwave-comment-line-start-skip)
+                     (looking-at "[ \t]*$")
+                     (looking-at (concat "[ \t]*" idlwave-label "[ \t]*$"))
+                     (looking-at "^@")))
+              (= (forward-line -1) 0)))
+      ;; skip continuation lines
+      (while (and
+              (save-excursion
+                (forward-line -1)
+                (idlwave-is-continuation-line))
+              (= (forward-line -1) 0)))
+      first-statement)))
+
+;; FIXME: end-of-statement does not work correctly when comment lines
+;; are inside the statement.  It does work correctly for line-end
+;; comments, though.
+(defun idlwave-end-of-statement ()
+  "Moves point to the end of the current IDL statement.
+If not in a statement just moves to end of line. Returns position."
+  (interactive)
+  (while (and (idlwave-is-continuation-line)
+              (= (forward-line 1) 0)))
+  (end-of-line)
+  (point))
+
+(defun idlwave-next-statement ()
+  "Moves point to beginning of the next IDL statement.
+ Returns t if that statement is the last
+ non-comment IDL statement in the file, and nil otherwise."
+  (interactive)
+  (let (last-statement)
+    (idlwave-end-of-statement)
+    ;; skip blank lines, label lines, include lines and line comments
+    (while (and (= (forward-line 1) 0)
+                ;; The current statement is the last statement until
+                ;; we reach a new statement.
+                (setq last-statement
+                      (or
+                       (looking-at idlwave-comment-line-start-skip)
+                       (looking-at "[ \t]*$")
+                       (looking-at (concat "[ \t]*" idlwave-label "[ \t]*$"))
+                       (looking-at "^@")))))
+    last-statement))
+
+(defun idlwave-skip-label ()
+  "Skip label or case statement element.
+Returns position after label.
+If there is no label point is not moved and nil is returned."
+  ;; Just look for the first non quoted colon and check to see if it
+  ;; is inside a sexp.  If is not in a sexp it must be part of a label
+  ;; or case statement element.
+  (let ((start (point))
+        (end (idlwave-find-key ":" 1 'nomark
+                              (save-excursion
+                                (idlwave-end-of-statement) (point)))))
+    (if (and end
+             (= (nth 0 (parse-partial-sexp start end)) 0))
+        (progn
+          (forward-char)
+          (point))
+      (goto-char start)
+      nil)))
+
+(defun idlwave-start-of-substatement (&optional pre)
+  "Move to start of next IDL substatement after point.
+Uses the type of the current IDL statement to determine if the next
+statement is on a new line or is a subpart of the current statement.
+Returns point at start of substatement modulo whitespace.
+If optional argument is non-nil move to beginning of current
+substatement. "
+  (let ((orig (point))
+        (eos (idlwave-end-of-statement))
+        (ifnest 0)
+        st nst last)
+    (idlwave-beginning-of-statement)
+    (idlwave-skip-label)
+    (setq last (point))
+    ;; Continue looking for substatements until we are past orig
+    (while (and (<= (point) orig) (not (eobp)))
+      (setq last (point))
+      (setq nst (nth 1 (cdr (setq st (car (idlwave-statement-type))))))
+      (if (equal (car st) 'if) (setq ifnest (1+ ifnest)))
+      (cond ((and nst
+                  (idlwave-find-key nst 1 'nomark eos))
+             (goto-char (match-end 0)))
+            ((and (> ifnest 0) (idlwave-find-key "\\<else\\>" 1 'nomark eos))
+             (setq ifnest (1- ifnest))
+             (goto-char (match-end 0)))
+            (t (setq ifnest 0)
+               (idlwave-next-statement))))
+    (if pre (goto-char last))
+    (point)))
+
+(defun idlwave-statement-type ()
+  "Return the type of the current IDL statement.
+Uses `idlwave-statement-match' to return a cons of (type . point) with
+point the ending position where the type was determined. Type is the
+association from `idlwave-statement-match', i.e. the cons cell from the
+list not just the type symbol. Returns nil if not an identifiable
+statement."
+  (save-excursion
+    ;; Skip whitespace within a statement which is spaces, tabs, continuations
+    (while (looking-at "[ \t]*\\<\\$")
+      (forward-line 1))
+    (skip-chars-forward " \t")
+    (let ((st idlwave-statement-match)
+          (case-fold-search t))
+      (while (and (not (looking-at (nth 0 (cdr (car st)))))
+                  (setq st (cdr st))))
+      (if st
+          (append st (match-end 0))))))
+
+(defun idlwave-expand-equal (&optional before after)
+  "Pad '=' with spaces.
+Two cases: Assignment statement, and keyword assignment.
+The case is determined using `idlwave-start-of-substatement' and
+`idlwave-statement-type'.
+The equal sign will be surrounded by BEFORE and AFTER blanks.
+If `idlwave-pad-keyword' is non-nil then keyword
+assignment is treated just like assignment statements.  Otherwise,
+spaces are removed for keyword assignment.
+Limits in for loops are treated as keyword assignment.
+See `idlwave-surround'. "
+  ;; Even though idlwave-surround checks `idlwave-surround-by-blank' this
+  ;; check saves the time of finding the statement type.
+  (if idlwave-surround-by-blank
+      (let ((st (save-excursion
+                  (idlwave-start-of-substatement t)
+                  (idlwave-statement-type))))
+        (if (or
+             (and (equal (car (car st)) 'assign)
+                  (equal (cdr st) (point)))
+             idlwave-pad-keyword)
+            ;; An assignment statement
+            (idlwave-surround before after)
+          (idlwave-surround 0 0)))))
+
+(defun idlwave-indent-and-action ()
+  "Call `idlwave-indent-line' and do expand actions."
+  (interactive)
+  (idlwave-indent-line t)
+  )
+
+(defun idlwave-indent-line (&optional expand)
+  "Indents current IDL line as code or as a comment.
+The actions in `idlwave-indent-action-table' are performed.
+If the optional argument EXPAND is non-nil then the actions in
+`idlwave-indent-expand-table' are performed."
+  (interactive)
+  ;; Move point out of left margin.
+  (if (save-excursion
+        (skip-chars-backward " \t")
+        (bolp))
+      (skip-chars-forward " \t"))
+  (let ((mloc (point-marker)))
+    (save-excursion
+      (beginning-of-line)
+      (if (looking-at idlwave-comment-line-start-skip)
+          ;; Indentation for a line comment
+          (progn
+            (skip-chars-forward " \t")
+            (idlwave-indent-left-margin (idlwave-comment-hook)))
+        ;;
+        ;; Code Line
+        ;;
+        ;; Before indenting, run action routines.
+        ;;
+        (if (and expand idlwave-do-actions)
+            (mapcar 'idlwave-do-action idlwave-indent-expand-table))
+        ;;
+        (if idlwave-do-actions
+            (mapcar 'idlwave-do-action idlwave-indent-action-table))
+        ;;
+        ;; No longer expand abbrevs on the line.  The user can do this
+        ;; manually using expand-region-abbrevs.
+        ;;
+        ;; Indent for code line
+        ;;
+        (beginning-of-line)
+        (if (or
+             ;; a label line
+             (looking-at (concat "^" idlwave-label "[ \t]*$"))
+             ;; a batch command
+             (looking-at "^[ \t]*@"))
+            ;; leave flush left
+            nil
+          ;; indent the line
+          (idlwave-indent-left-margin (idlwave-calculate-indent)))
+        ;; Adjust parallel comment
+        (end-of-line)
+        (if (idlwave-in-comment)
+            (indent-for-comment))))
+    (goto-char mloc)
+    ;; Get rid of marker
+    (set-marker mloc nil)
+    ))
+
+(defun idlwave-do-action (action)
+  "Perform an action repeatedly on a line.
+ACTION is a list (REG . FUNC).  REG is a regular expression.  FUNC is
+either a function name to be called with `funcall' or a list to be
+evaluated with `eval'.  The action performed by FUNC should leave point
+after the match for REG - otherwise an infinite loop may be entered."
+  (let ((action-key (car action))
+        (action-routine (cdr action)))
+    (beginning-of-line)
+    (while (idlwave-look-at action-key)
+      (if (listp action-routine)
+          (eval action-routine)
+        (funcall action-routine)))))
+
+(defun idlwave-indent-to (col &optional min)
+  "Indent from point with spaces until column COL.
+Inserts space before markers at point."
+  (if (not min) (setq min 0))
+  (insert-before-markers
+   (make-string (max min (- col (current-column))) ? )))
+
+(defun idlwave-indent-left-margin (col)
+  "Indent the current line to column COL.
+Indents such that first non-whitespace character is at column COL
+Inserts spaces before markers at point."
+  (save-excursion
+    (beginning-of-line)
+    (delete-horizontal-space)
+    (idlwave-indent-to col)))
+
+(defun idlwave-indent-subprogram ()
+  "Indents program unit which contains point."
+  (interactive)
+  (save-excursion
+    (idlwave-end-of-statement)
+    (idlwave-beginning-of-subprogram)
+    (let ((beg (point)))
+      (idlwave-forward-block)
+      (message "Indenting subprogram...")
+      (indent-region beg (point) nil))
+    (message "Indenting subprogram...done.")))
+
+(defun idlwave-calculate-indent ()
+  "Return appropriate indentation for current line as IDL code."
+  (save-excursion
+    (beginning-of-line)
+    (cond
+     ;; Check for beginning of unit - main (beginning of buffer), pro, or
+     ;; function
+     ((idlwave-look-at idlwave-begin-unit-reg)
+      0)
+     ;; Check for continuation line
+     ((save-excursion
+        (and (= (forward-line -1) 0)
+             (idlwave-is-continuation-line)))
+      (idlwave-calculate-cont-indent))
+     ;; calculate indent based on previous and current statements
+     (t (let ((the-indent
+               ;; calculate indent based on previous statement
+               (save-excursion
+                 (cond
+                  ((idlwave-previous-statement)
+                   0)
+                  ;; Main block
+                  ((idlwave-look-at idlwave-begin-unit-reg t)
+                   (+ (idlwave-current-statement-indent)
+                     idlwave-main-block-indent))
+                  ;; Begin block
+                  ((idlwave-look-at idlwave-begin-block-reg t)
+                   (+ (idlwave-current-statement-indent)
+                     idlwave-block-indent))
+                  ((idlwave-look-at idlwave-end-block-reg t)
+                   (- (idlwave-current-statement-indent)
+                     idlwave-end-offset
+                     idlwave-block-indent))
+                  ((idlwave-current-statement-indent))))))
+          ;; adjust the indentation based on the current statement
+          (cond
+           ;; End block
+           ((idlwave-look-at idlwave-end-block-reg t)
+            (+ the-indent idlwave-end-offset))
+           (the-indent)))))))
+
+;;
+;; Parenthesses balacing/indent
+;;
+
+(defun idlwave-calculate-cont-indent ()
+  "Calculates the IDL continuation indent column from the previous statement.
+Note that here previous statement means the beginning of the current
+statement if this statement is a continuation of the previous line.
+Intervening comments or comments within the previous statement can
+screw things up if the comments contain parentheses characters."
+  (save-excursion
+    (let* (open
+           (case-fold-search t)
+           (end-reg (progn (beginning-of-line) (point)))
+           (close-exp (progn (skip-chars-forward " \t") (looking-at "\\s)")))
+           (beg-reg (progn (idlwave-previous-statement) (point))))
+      ;;
+      ;; If PRO or FUNCTION declaration indent after name, and first comma.
+      ;;
+      (if (idlwave-look-at "\\<\\(pro\\|function\\)\\>")
+          (progn
+            (forward-sexp 1)
+            (if (looking-at "[ \t]*,[ \t]*")
+                (goto-char (match-end 0)))
+            (current-column))
+        ;;
+        ;; Not a PRO or FUNCTION
+        ;;
+        ;; Look for innermost unmatched open paren
+        ;;
+        (if (setq open (car (cdr (parse-partial-sexp beg-reg end-reg))))
+            ;; Found innermost open paren.
+            (progn
+              (goto-char open)
+             ;; Line up with next word unless this is a closing paren.
+              (cond
+               ;; This is a closed paren - line up under open paren.
+               (close-exp
+                (current-column))
+               ;; Empty - just add regular indent. Take into account
+               ;; the forward-char
+               ((progn
+                  ;; Skip paren
+                  (forward-char 1)
+                  (looking-at "[ \t$]*$"))
+                (+ (current-column) idlwave-continuation-indent -1))
+               ;; Line up with first word
+               ((progn
+                  (skip-chars-forward " \t")
+                  (current-column)))))
+          ;; No unmatched open paren. Just a simple continuation.
+          (goto-char beg-reg)
+          (+ (idlwave-current-indent)
+             ;; Make adjustments based on current line
+             (cond
+              ;; Else statement
+              ((progn
+                 (goto-char end-reg)
+                 (skip-chars-forward " \t")
+                 (looking-at "else"))
+               0)
+              ;; Ordinary continuation
+              (idlwave-continuation-indent))))))))
+
+(defun idlwave-find-key (key-reg &optional dir nomark limit)
+  "Move in direction of the optional second argument DIR to the
+next keyword not contained in a comment or string and occurring before
+optional fourth argument LIMIT. DIR defaults to forward direction.  If
+DIR is negative the search is backwards, otherwise, it is
+forward. LIMIT defaults to the beginning or end of the buffer
+according to the direction of the search. The keyword is given by the
+regular expression argument KEY-REG.  The search is case insensitive.
+Returns position if successful and nil otherwise.  If found
+`push-mark' is executed unless the optional third argument NOMARK is
+non-nil. If found, the point is left at the keyword beginning."
+  (or dir (setq dir 0))
+  (or limit (setq limit (cond ((>= dir 0) (point-max)) ((point-min)))))
+  (let (found
+        (old-syntax-table (syntax-table))
+        (case-fold-search t))
+    (unwind-protect
+       (save-excursion
+         (set-syntax-table idlwave-find-symbol-syntax-table)
+         (if (>= dir 0)
+             (while (and (setq found (and
+                                      (re-search-forward key-reg limit t)
+                                      (match-beginning 0)))
+                         (idlwave-quoted)
+                         (not (eobp))))
+           (while (and (setq found (and
+                                    (re-search-backward key-reg limit t)
+                                    (match-beginning 0)))
+                       (idlwave-quoted)
+                       (not (bobp))))))
+      (set-syntax-table old-syntax-table))
+    (if found (progn
+                (if (not nomark) (push-mark))
+                (goto-char found)))))
+
+(defun idlwave-block-jump-out (&optional dir nomark)
+  "When optional argument DIR is non-negative, move forward to end of
+current block using the `idlwave-begin-block-reg' and `idlwave-end-block-reg'
+regular expressions. When DIR is negative, move backwards to block beginning.
+Recursively calls itself to skip over nested blocks. DIR defaults to
+forward. Calls `push-mark' unless the optional argument NOMARK is
+non-nil. Movement is limited by the start of program units because of
+possibility of unbalanced blocks."
+  (interactive "P")
+  (or dir (setq dir 0))
+  (let* ((here (point))
+         (case-fold-search t)
+         (limit (if (>= dir 0) (point-max) (point-min)))
+         (block-limit (if (>= dir 0) 
+                         idlwave-begin-block-reg
+                       idlwave-end-block-reg))
+         found
+         (block-reg (concat idlwave-begin-block-reg "\\|"
+                           idlwave-end-block-reg))
+         (unit-limit (or (save-excursion
+                          (if (< dir 0)
+                              (idlwave-find-key
+                               idlwave-begin-unit-reg dir t limit)
+                            (end-of-line)
+                            (idlwave-find-key 
+                             idlwave-end-unit-reg dir t limit)))
+                        limit)))
+    (if (>= dir 0) (end-of-line)) ;Make sure we are in current block
+    (if (setq found (idlwave-find-key  block-reg dir t unit-limit))
+        (while (and found (looking-at block-limit))
+          (if (>= dir 0) (forward-word 1))
+          (idlwave-block-jump-out dir t)
+          (setq found (idlwave-find-key block-reg dir t unit-limit))))
+    (if (not nomark) (push-mark here))
+    (if (not found) (goto-char unit-limit)
+      (if (>= dir 0) (forward-word 1)))))
+
+(defun idlwave-current-statement-indent ()
+  "Return indentation of the current statement.
+If in a statement, moves to beginning of statement before finding indent."
+  (idlwave-beginning-of-statement)
+  (idlwave-current-indent))
+
+(defun idlwave-current-indent ()
+  "Return the column of the indentation of the current line.
+Skips any whitespace. Returns 0 if the end-of-line follows the whitespace."
+  (save-excursion
+    (beginning-of-line)
+    (skip-chars-forward " \t")
+    ;; if we are at the end of blank line return 0
+    (cond ((eolp) 0)
+          ((current-column)))))
+
+(defun idlwave-is-continuation-line ()
+  "Tests if current line is continuation line."
+  (save-excursion
+    (idlwave-look-at "\\<\\$")))
+
+(defun idlwave-is-comment-line ()
+  (save-excursion
+    (beginning-of-line 1)
+    (looking-at "[ \t]*;")))
+
+(defun idlwave-look-at (regexp &optional cont beg)
+  "Searches current line from current point for the regular expression
+REGEXP. If optional argument CONT is non-nil, searches to the end of
+the current statement. If optional arg BEG is non-nil, search starts
+from the beginning of the current statement. Ignores matches that end
+in a comment or inside a string expression. Returns point if
+successful, nil otherwise.  This function produces unexpected results
+if REGEXP contains quotes or a comment delimiter. The search is case
+insensitive.  If successful leaves point after the match, otherwise,
+does not move point."
+  (let ((here (point))
+        (old-syntax-table (syntax-table))
+        (case-fold-search t)
+        eos
+        found)
+    (unwind-protect
+       (progn
+         (set-syntax-table idlwave-find-symbol-syntax-table)
+         (setq eos
+               (if cont
+                   (save-excursion (idlwave-end-of-statement) (point))
+                 (save-excursion (end-of-line) (point))))
+         (if beg (idlwave-beginning-of-statement))
+         (while (and (setq found (re-search-forward regexp eos t))
+                     (idlwave-quoted))))
+      (set-syntax-table old-syntax-table))
+    (if (not found) (goto-char here))
+    found))
+
+(defun idlwave-fill-paragraph (&optional nohang)
+  "Fills paragraphs in comments.
+A paragraph is made up of all contiguous lines having the same comment
+leader (the leading whitespace before the comment delimiter and the
+comment delimiter).  In addition, paragraphs are separated by blank
+line comments. The indentation is given by the hanging indent of the
+first line, otherwise by the minimum indentation of the lines after
+the first line. The indentation of the first line does not change.
+Does not effect code lines. Does not fill comments on the same line
+with code.  The hanging indent is given by the end of the first match
+matching `idlwave-hang-indent-regexp' on the paragraph's first line . If the
+optional argument NOHANG is non-nil then the hanging indent is
+ignored."
+  (interactive "P")
+  ;; check if this is a line comment
+  (if (save-excursion
+        (beginning-of-line)
+        (skip-chars-forward " \t")
+        (looking-at comment-start))
+      (let
+          ((indent 999)
+           pre here diff fill-prefix-reg bcl first-indent
+           hang start end)
+        ;; Change tabs to spaces in the surrounding paragraph.
+        ;; The surrounding paragraph will be the largest containing block of
+        ;; contiguous line comments. Thus, we may be changing tabs in
+        ;; a much larger area than is needed, but this is the easiest
+        ;; brute force way to do it.
+        ;;
+        ;; This has the undesirable side effect of replacing the tabs
+        ;; permanently without the user's request or knowledge.
+        (save-excursion
+          (backward-paragraph)
+          (setq start (point)))
+        (save-excursion
+          (forward-paragraph)
+          (setq end (point)))
+        (untabify start end)
+        ;;
+        (setq here (point))
+        (beginning-of-line)
+        (setq bcl (point))
+        (re-search-forward
+         (concat "^[ \t]*" comment-start "+")
+         (save-excursion (end-of-line) (point))
+         t)
+        ;; Get the comment leader on the line and its length
+        (setq pre (current-column))
+        ;; the comment leader is the indentation plus exactly the
+        ;; number of consecutive ";".
+        (setq fill-prefix-reg
+              (concat
+               (setq fill-prefix
+                     (regexp-quote
+                      (buffer-substring (save-excursion
+                                          (beginning-of-line) (point))
+                                        (point))))
+               "[^;]"))
+       
+        ;; Mark the beginning and end of the paragraph
+        (goto-char bcl)
+        (while (and (looking-at fill-prefix-reg)
+                    (not (looking-at paragraph-separate))
+                    (not (bobp)))
+          (forward-line -1))
+        ;; Move to first line of paragraph
+        (if (/= (point) bcl)
+            (forward-line 1))
+        (setq start (point))
+        (goto-char bcl)
+        (while (and (looking-at fill-prefix-reg)
+                    (not (looking-at paragraph-separate))
+                    (not (eobp)))
+          (forward-line 1))
+        (beginning-of-line)
+        (if (or (not (looking-at fill-prefix-reg))
+                (looking-at paragraph-separate))
+            (forward-line -1))
+        (end-of-line)
+        ;; if at end of buffer add a newline (need this because
+        ;; fill-region needs END to be at the beginning of line after
+        ;; the paragraph or it will add a line).
+        (if (eobp)
+            (progn (insert ?\n) (backward-char 1)))
+        ;; Set END to the beginning of line after the paragraph
+        ;; END is calculated as distance from end of buffer
+        (setq end (- (point-max) (point) 1))
+        ;;
+        ;; Calculate the indentation for the paragraph.
+        ;;
+        ;; In the following while statements, after one iteration
+        ;; point will be at the beginning of a line in which case
+        ;; the while will not be executed for the
+        ;; the first paragraph line and thus will not affect the
+        ;; indentation.
+        ;;
+        ;; First check to see if indentation is based on hanging indent.
+        (if (and (not nohang) idlwave-hanging-indent
+                 (setq hang
+                       (save-excursion
+                         (goto-char start)
+                         (idlwave-calc-hanging-indent))))
+            ;; Adjust lines of paragraph by inserting spaces so that
+            ;; each line's indent is at least as great as the hanging
+            ;; indent. This is needed for fill-paragraph to work with
+            ;; a fill-prefix.
+            (progn
+              (setq indent hang)
+              (beginning-of-line)
+              (while (> (point) start)
+                (re-search-forward comment-start-skip
+                                   (save-excursion (end-of-line) (point))
+                                   t)
+                (if (> (setq diff (- indent (current-column))) 0)
+                    (progn
+                      (if (>= here (point))
+                          ;; adjust the original location for the
+                          ;; inserted text.
+                          (setq here (+ here diff)))
+                      (insert (make-string diff ? ))))
+                (forward-line -1))
+              )
+         
+          ;; No hang. Instead find minimum indentation of paragraph
+          ;; after first line.
+          ;; For the following while statement, since START is at the
+          ;; beginning of line and END is at the the end of line
+          ;; point is greater than START at least once (which would
+          ;; be the case for a single line paragraph).
+          (while (> (point) start)
+            (beginning-of-line)
+            (setq indent
+                  (min indent
+                       (progn
+                         (re-search-forward
+                          comment-start-skip
+                          (save-excursion (end-of-line) (point))
+                          t)
+                         (current-column))))
+            (forward-line -1))
+          )
+        (setq fill-prefix (concat fill-prefix
+                                  (make-string (- indent pre)
+                                               ? )))
+        ;; first-line indent
+        (setq first-indent
+              (max
+               (progn
+                 (re-search-forward
+                  comment-start-skip
+                  (save-excursion (end-of-line) (point))
+                  t)
+                 (current-column))
+               indent))
+       
+        ;; try to keep point at its original place
+        (goto-char here)
+
+        ;; In place of the more modern fill-region-as-paragraph, a hack
+        ;; to keep whitespace untouched on the first line within the
+        ;; indent length and to preserve any indent on the first line
+        ;; (first indent).
+        (save-excursion
+          (setq diff
+                (buffer-substring start (+ start first-indent -1)))
+          (subst-char-in-region start (+ start first-indent -1) ?  ?~ nil)
+          (fill-region-as-paragraph
+           start
+           (- (point-max) end)
+           (current-justification)
+           nil)
+          (delete-region start (+ start first-indent -1))
+          (goto-char start)
+          (insert diff))
+        ;; When we want the point at the beginning of the comment
+        ;; body fill-region will put it at the beginning of the line.
+        (if (bolp) (skip-chars-forward (concat " \t" comment-start)))
+        (setq fill-prefix nil))))
+
+(defun idlwave-calc-hanging-indent ()
+  "Calculate the position of the hanging indent for the comment
+paragraph.  The hanging indent position is given by the first match
+with the `idlwave-hang-indent-regexp'.  If `idlwave-use-last-hang-indent' is
+non-nil then use last occurrence matching `idlwave-hang-indent-regexp' on
+the line.
+If not found returns nil."
+  (if idlwave-use-last-hang-indent
+      (save-excursion
+        (end-of-line)
+        (if (re-search-backward
+             idlwave-hang-indent-regexp
+             (save-excursion (beginning-of-line) (point))
+             t)
+            (+ (current-column) (length idlwave-hang-indent-regexp))))
+    (save-excursion
+      (beginning-of-line)
+      (if (re-search-forward
+           idlwave-hang-indent-regexp
+           (save-excursion (end-of-line) (point))
+           t)
+          (current-column)))))
+
+(defun idlwave-auto-fill ()
+  "Called to break lines in auto fill mode.
+Only fills comment lines if `idlwave-fill-comment-line-only' is non-nil.
+Places a continuation character at the end of the line
+if not in a comment.  Splits strings with IDL concatenation operator
+`+' if `idlwave-auto-fill-split-string is non-nil."
+  (if (<= (current-column) fill-column)
+      nil                             ; do not to fill
+    (if (or (not idlwave-fill-comment-line-only)
+           (save-excursion
+             ;; Check for comment line
+             (beginning-of-line)
+             (looking-at idlwave-comment-line-start-skip)))
+       (let (beg)
+         (idlwave-indent-line)
+         ;; Prevent actions do-auto-fill which calls indent-line-function.
+         (let (idlwave-do-actions
+               (paragraph-start ".")
+               (paragraph-separate "."))
+           (do-auto-fill))
+         (save-excursion
+           (end-of-line 0)
+           ;; Indent the split line
+           (idlwave-indent-line)
+           )
+         (if (save-excursion
+               (beginning-of-line)
+               (looking-at idlwave-comment-line-start-skip))
+             ;; A continued line comment
+             ;; We treat continued line comments as part of a comment
+             ;; paragraph. So we check for a hanging indent.
+             (if idlwave-hanging-indent
+                 (let ((here (- (point-max) (point)))
+                       (indent
+                        (save-excursion
+                          (forward-line -1)
+                          (idlwave-calc-hanging-indent))))
+                   (if indent
+                       (progn
+                         ;; Remove whitespace between comment delimiter and
+                         ;; text, insert spaces for appropriate indentation.
+                         (beginning-of-line)
+                         (re-search-forward
+                          comment-start-skip
+                          (save-excursion (end-of-line) (point)) t)
+                         (delete-horizontal-space)
+                         (idlwave-indent-to indent)
+                         (goto-char (- (point-max) here)))
+                     )))
+           ;; Split code or comment?
+           (if (save-excursion
+                 (end-of-line 0)
+                 (idlwave-in-comment))
+               ;; Splitting a non-line comment.
+               ;; Insert the comment delimiter from split line
+               (progn
+                 (save-excursion
+                   (beginning-of-line)
+                   (skip-chars-forward " \t")
+                   ;; Insert blank to keep off beginning of line
+                   (insert " "
+                           (save-excursion
+                             (forward-line -1)
+                             (buffer-substring (idlwave-goto-comment)
+                                               (progn
+                                                 (skip-chars-forward "; ")
+                                                 (point))))))
+                 (idlwave-indent-line))
+             ;; Split code line - add continuation character
+             (save-excursion
+               (end-of-line 0)
+               ;; Check to see if we split a string
+               (if (and (setq beg (idlwave-in-quote))
+                        idlwave-auto-fill-split-string)
+                   ;; Split the string and concatenate.
+                   ;; The first extra space is for the space
+                   ;; the line was split. That space was removed.
+                   (insert " " (char-after beg) " +"))
+               (insert " $"))
+             (if beg
+                 (if idlwave-auto-fill-split-string
+                     ;; Make the second part of continued string
+                     (save-excursion
+                       (beginning-of-line)
+                       (skip-chars-forward " \t")
+                       (insert (char-after beg)))
+                   ;; Warning
+                   (beep)
+                   (message "Warning: continuation inside a string.")))
+             ;; Although do-auto-fill (via indent-new-comment-line) calls
+             ;; idlwave-indent-line for the new line, re-indent again
+             ;; because of the addition of the continuation character.
+             (idlwave-indent-line))
+           )))))
+
+(defun idlwave-auto-fill-mode (arg)
+  "Toggle auto-fill mode for IDL mode.
+With arg, turn auto-fill mode on if arg is positive.
+In auto-fill mode, inserting a space at a column beyond `fill-column'
+automatically breaks the line at a previous space."
+  (interactive "P")
+  (prog1 (set idlwave-fill-function
+              (if (if (null arg)
+                      (not (symbol-value idlwave-fill-function))
+                    (> (prefix-numeric-value arg) 0))
+                  'idlwave-auto-fill
+                nil))
+    ;; update mode-line
+    (set-buffer-modified-p (buffer-modified-p))))
+
+(defun idlwave-doc-header (&optional nomark )
+  "Insert a documentation header at the beginning of the unit.
+Inserts the value of the variable idlwave-file-header. Sets mark before
+moving to do insertion unless the optional prefix argument NOMARK
+is non-nil."
+  (interactive "P")
+  (or nomark (push-mark))
+  ;; make sure we catch the current line if it begins the unit
+  (end-of-line)
+  (idlwave-beginning-of-subprogram)
+  (beginning-of-line)
+  ;; skip function or procedure line
+  (if (idlwave-look-at "\\<\\(pro\\|function\\)\\>")
+      (progn
+        (idlwave-end-of-statement)
+        (if (> (forward-line 1) 0) (insert "\n"))))
+  (if idlwave-file-header
+      (cond ((car idlwave-file-header)
+             (insert-file (car idlwave-file-header)))
+            ((stringp (car (cdr idlwave-file-header)))
+             (insert (car (cdr idlwave-file-header)))))))
+
+
+(defun idlwave-default-insert-timestamp ()
+  "Default timestamp insertion function"
+  (insert (current-time-string))
+  (insert ", " (user-full-name))
+  (insert " <" (user-login-name) "@" (system-name) ">")
+  ;; Remove extra spaces from line
+  (idlwave-fill-paragraph)
+  ;; Insert a blank line comment to separate from the date entry -
+  ;; will keep the entry from flowing onto date line if re-filled.
+  (insert "\n;\n;\t\t"))
+
+(defun idlwave-doc-modification ()
+  "Insert a brief modification log at the beginning of the current program.
+Looks for an occurrence of the value of user variable
+`idlwave-doc-modifications-keyword' if non-nil. Inserts time and user name
+and places the point for the user to add a log. Before moving, saves
+location on mark ring so that the user can return to previous point."
+  (interactive)
+  (push-mark)
+  ;; make sure we catch the current line if it begins the unit
+  (end-of-line)
+  (idlwave-beginning-of-subprogram)
+  (let ((pro (idlwave-look-at "\\<\\(function\\|pro\\)\\>"))
+        (case-fold-search nil))
+    (if (re-search-forward
+         (concat idlwave-doc-modifications-keyword ":")
+         ;; set search limit at next unit beginning
+         (save-excursion (idlwave-end-of-subprogram) (point))
+         t)
+        (end-of-line)
+      ;; keyword not present, insert keyword
+      (if pro (idlwave-next-statement))  ; skip past pro or function statement
+      (beginning-of-line)
+      (insert "\n" comment-start "\n")
+      (forward-line -2)
+      (insert comment-start " " idlwave-doc-modifications-keyword ":")))
+  (idlwave-newline)
+  (beginning-of-line)
+  (insert ";\n;\t")
+  (run-hooks 'idlwave-timestamp-hook))
+
+;;; CJC 3/16/93
+;;; Interface to expand-region-abbrevs which did not work when the
+;;; abbrev hook associated with an abbrev moves point backwards
+;;; after abbrev expansion, e.g., as with the abbrev '.n'.
+;;; The original would enter an infinite loop in attempting to expand
+;;; .n (it would continually expand and unexpand the abbrev without expanding
+;;; because the point would keep going back to the beginning of the
+;;; abbrev instead of to the end of the abbrev). We now keep the
+;;; abbrev hook from moving backwards.
+;;;
+(defun idlwave-expand-region-abbrevs (start end)
+  "Expand each abbrev occurrence in the region.
+Calling from a program, arguments are START END."
+  (interactive "r")
+  (save-excursion
+    (goto-char (min start end))
+    (let ((idlwave-show-block nil)          ;Do not blink
+          (idlwave-abbrev-move nil))        ;Do not move
+      (expand-region-abbrevs start end 'noquery))))
+
+(defun idlwave-quoted ()
+  "Returns t if point is in a comment or quoted string.
+nil otherwise."
+  (or (idlwave-in-comment) (idlwave-in-quote)))
+
+(defun idlwave-in-quote ()
+  "Returns location of the opening quote
+if point is in a IDL string constant, nil otherwise.
+Ignores comment delimiters on the current line.
+Properly handles nested quotation marks and octal
+constants - a double quote followed by an octal digit."
+;;; Treat an octal inside an apostrophe to be a normal string. Treat a
+;;; double quote followed by an octal digit to be an octal constant
+;;; rather than a string. Therefore, there is no terminating double
+;;; quote.
+  (save-excursion
+    ;; Because single and double quotes can quote each other we must
+    ;; search for the string start from the beginning of line.
+    (let* ((start (point))
+           (eol (progn (end-of-line) (point)))
+           (bq (progn (beginning-of-line) (point)))
+           (endq (point))
+           (data (match-data))
+           delim
+           found)
+      (while  (< endq start)
+       ;; Find string start
+       ;; Don't find an octal constant beginning with a double quote
+       (if (re-search-forward "\"[^0-7]\\|'\\|\"$" eol 'lim)
+           ;; Find the string end.
+           ;; In IDL, two consecutive delimiters after the start of a
+           ;; string act as an
+           ;; escape for the delimiter in the string.
+           ;; Two consecutive delimiters alone (i.e., not after the
+           ;; start of a string) is the the null string.
+           (progn
+             ;; Move to position after quote
+             (goto-char (1+ (match-beginning 0)))
+             (setq bq (1- (point)))
+             ;; Get the string delimiter
+             (setq delim (char-to-string (preceding-char)))
+             ;; Check for null string
+             (if (looking-at delim)
+                 (progn (setq endq (point)) (forward-char 1))
+               ;; Look for next unpaired delimiter
+               (setq found (search-forward delim eol 'lim))
+               (while (looking-at delim)
+                 (forward-char 1)
+                 (setq found (search-forward delim eol 'lim)))
+               (if found
+                   (setq endq (- (point) 1))
+                 (setq endq (point)))
+               ))
+         (progn (setq bq (point)) (setq endq (point)))))
+      (store-match-data data)
+      ;; return string beginning position or nil
+      (if (> start bq) bq))))
+
+;; Statement templates
+
+;; Replace these with a general template function, something like
+;; expand.el (I think there was also something with a name similar to
+;; dmacro.el)
+
+(defun idlwave-template (s1 s2 &optional prompt noindent)
+  "Build a template with optional prompt expression.
+
+Opens a line if point is not followed by a newline modulo intervening
+whitespace.  S1 and S2 are strings.  S1 is inserted at point followed
+by S2.  Point is inserted between S1 and S2.  If optional argument
+PROMPT is a string then it is displayed as a message in the
+minibuffer.  The PROMPT serves as a reminder to the user of an
+expression to enter.
+
+The lines containing S1 and S2 are reindented using `indent-region'
+unless the optional second argument NOINDENT is non-nil."
+  (let ((beg (save-excursion (beginning-of-line) (point)))
+        end)
+    (if (not (looking-at "\\s-*\n"))
+        (open-line 1))
+    (insert s1)
+    (save-excursion
+      (insert s2)
+      (setq end (point)))
+    (if (not noindent)
+        (indent-region beg end nil))
+    (if (stringp prompt)
+        (message prompt))))
+
+(defun idlwave-elif ()
+  "Build skeleton IDL if-else block."
+  (interactive)
+  (idlwave-template "if" 
+                   " then begin\n\nendif else begin\n\nendelse"
+                   "Condition expression"))
+
+(defun idlwave-case ()
+  "Build skeleton IDL case statement."
+  (interactive)
+  (idlwave-template "case" " of\n\nendcase" "Selector expression"))
+
+(defun idlwave-for ()
+  "Build skeleton for loop statment."
+  (interactive)
+  (idlwave-template "for" " do begin\n\nendfor" "Loop expression"))
+
+(defun idlwave-if ()
+  "Build skeleton for loop statment."
+  (interactive)
+  (idlwave-template "if" " then begin\n\nendif" "Scalar logical expression"))
+
+(defun idlwave-procedure ()
+  (interactive)
+  (idlwave-template "pro" "\n\nreturn\nend" "Procedure name"))
+
+(defun idlwave-function ()
+  (interactive)
+  (idlwave-template "function" "\n\nreturn\nend" "Function name"))
+
+(defun idlwave-repeat ()
+  (interactive)
+  (idlwave-template "repeat begin\n\nendrep until" "" "Exit condition"))
+
+(defun idlwave-while ()
+  (interactive)
+  (idlwave-template "while" " do begin\n\nendwhile" "Entry condition"))
+
+(defun idlwave-split-string (string &optional pattern)
+  "Return a list of substrings of STRING which are separated by PATTERN.
+If PATTERN is omitted, it defaults to \"[ \\f\\t\\n\\r\\v]+\"."
+  (or pattern
+      (setq pattern "[ \f\t\n\r\v]+"))
+  (let (parts (start 0))
+    (while (string-match pattern string start)
+      (setq parts (cons (substring string start (match-beginning 0)) parts)
+           start (match-end 0)))
+    (nreverse (cons (substring string start) parts))))
+
+(defun idlwave-replace-string (string replace_string replace_with)
+  (let* ((start 0)
+        (last (length string))
+        (ret_string "")
+        end)
+    (while (setq end (string-match replace_string string start))
+      (setq ret_string
+           (concat ret_string (substring string start end) replace_with))
+      (setq start (match-end 0)))
+    (setq ret_string (concat ret_string (substring string start last)))))
+
+(defun idlwave-get-buffer-visiting (file)
+  ;; Return the buffer currently visiting FILE
+  (cond
+   ((boundp 'find-file-compare-truenames) ; XEmacs
+    (let ((find-file-compare-truenames t))
+      (get-file-buffer file)))
+   ((fboundp 'find-buffer-visiting)       ; Emacs
+    (find-buffer-visiting file))
+   (t (error "This should not happen (idlwave-get-buffer-visiting)"))))
+
+(defun idlwave-find-file-noselect (file)
+  ;; Return a buffer visiting file.
+  (or (idlwave-get-buffer-visiting file)
+      (find-file-noselect file)))
+
+(defvar idlwave-scanned-lib-directories)
+(defun idlwave-find-lib-file-noselet (file)
+  ;; Find FILE on the scanned lib path and return a buffer visiting it
+  (let* ((dirs idlwave-scanned-lib-directories)
+        dir efile)
+    (catch 'exit
+      (while (setq dir (pop dirs))
+       (if (file-regular-p
+            (setq efile (expand-file-name file dir)))
+           (throw 'exit (idlwave-find-file-noselect efile)))))))
+
+(defun idlwave-make-tags ()
+  "Creates the IDL tags file IDLTAGS in the current directory from
+the list of directories specified in the minibuffer. Directories may be
+for example: . /usr/local/rsi/idl/lib. All the subdirectories of the
+specified top directories are searched if the directory name is prefixed
+by @. Specify @ directories with care, it may take a long, long time if
+you specify /."
+  (interactive)
+  (let (directory directories cmd append status numdirs dir getsubdirs
+                 buffer save_buffer files numfiles item errbuf)
+    
+    ;;
+    ;; Read list of directories
+    (setq directory (read-string "Tag Directories: " "."))
+    (setq directories (idlwave-split-string directory "[ \t]+"))
+    ;;
+    ;; Set etags command, vars
+    (setq cmd "etags --output=IDLTAGS --language=none --regex='/[
+\\t]*[pP][Rr][Oo][ \\t]+\\([^ \\t,]+\\)/' --regex='/[
+\\t]*[Ff][Uu][Nn][Cc][Tt][Ii][Oo][Nn][ \\t]+\\([^ \\t,]+\\)/' ")
+    (setq append " ")
+    (setq status 0)
+    ;;
+    ;; For each directory
+    (setq numdirs 0)
+    (setq dir (nth numdirs directories))
+    (while (and dir)
+      ;;
+      ;; Find the subdirectories
+      (if (string-match "^[@]\\(.+\\)$" dir)
+         (setq getsubdirs t) (setq getsubdirs nil))
+      (if (and getsubdirs) (setq dir (substring dir 1 (length dir))))
+      (setq dir (expand-file-name dir))
+      (if (file-directory-p dir)
+         (progn
+           (if (and getsubdirs)
+               (progn
+                 (setq buffer (get-buffer-create "*idltags*"))
+                 (call-process "sh" nil buffer nil "-c"
+                               (concat "find " dir " -type d -print"))
+                 (setq save_buffer (current-buffer))
+                 (set-buffer buffer)
+                 (setq files (idlwave-split-string
+                              (idlwave-replace-string
+                               (buffer-substring 1 (point-max))
+                               "\n" "/*.pro ")
+                              "[ \t]+"))
+                 (set-buffer save_buffer)
+                 (kill-buffer buffer))
+             (setq files (list (concat dir "/*.pro"))))
+           ;;
+           ;; For each subdirectory
+           (setq numfiles 0)
+           (setq item (nth numfiles files))
+           (while (and item)
+             ;;
+             ;; Call etags
+             (if (not (string-match "^[ \\t]*$" item))
+                 (progn
+                   (message (concat "Tagging " item "..."))
+                   (setq errbuf (get-buffer-create "*idltags-error*"))
+                   (setq status (+ status
+                                   (call-process "sh" nil errbuf nil "-c"
+                                                 (concat cmd append item))))
+                   ;;
+                   ;; Append additional tags
+                   (setq append " --append ")
+                   (setq numfiles (1+ numfiles))
+                   (setq item (nth numfiles files)))
+               (progn
+                 (setq numfiles (1+ numfiles))
+                 (setq item (nth numfiles files))
+                 )))
+           
+           (setq numdirs (1+ numdirs))
+           (setq dir (nth numdirs directories)))
+       (progn
+         (setq numdirs (1+ numdirs))
+         (setq dir (nth numdirs directories)))))
+    
+    (setq errbuf (get-buffer-create "*idltags-error*"))
+    (if (= status 0)
+       (kill-buffer errbuf))
+    (message "")
+    ))
+
+(defun idlwave-toggle-comment-region (beg end &optional n)
+  "Comment the lines in the region if the first non-blank line is
+commented, and conversely, uncomment region. If optional prefix arg
+N is non-nil, then for N positive, add N comment delimiters or for N
+negative, remove N comment delimiters.
+Uses `comment-region' which does not place comment delimiters on
+blank lines."
+  (interactive "r\nP")
+  (if n
+      (comment-region beg end (prefix-numeric-value n))
+    (save-excursion
+      (goto-char beg)
+      (beginning-of-line)
+      ;; skip blank lines
+      (skip-chars-forward " \t\n")
+      (if (looking-at (concat "[ \t]*\\(" comment-start "+\\)"))
+          (comment-region beg end
+                          (- (length (buffer-substring
+                                      (match-beginning 1)
+                                      (match-end 1)))))
+        (comment-region beg end)))))
+
+
+;; ----------------------------------------------------------------------------
+;; ----------------------------------------------------------------------------
+;; ----------------------------------------------------------------------------
+;; ----------------------------------------------------------------------------
+;;
+;; Completion and Routine Info
+;;
+
+;; String "intern" functions
+
+;; For the completion and routine info function, we want to normalize
+;; the case of procedure names etc.  We do this by "interning" these
+;; string is a hand-crafted way.  Hashes are used to map the downcase
+;; version of the strings to the cased versions.  Since these cased
+;; versions are really lisp objects, we can use `eq' to search, which
+;; is a large performance boost.
+;; All new strings need to be "sinterned".  We do this as early as
+;; possible after getting these strings from completion or buffer
+;; substrings.  So most of the code can simply assume to deal with
+;; "sinterned" strings.  The only exception is that the functions
+;; which scan whole buffers for routine information do not intern the
+;; grabbed strings.  This is only done afterwards.  Therefore in these 
+;; functions it is *not* save to assume the strings can be compared
+;; with `eq' and be fed into the routine assq functions.
+
+;; Here we define the hashing functions.
+
+;; The variables which hold the hashes.
+(defvar idlwave-sint-routines '(nil))
+(defvar idlwave-sint-keywords '(nil))
+(defvar idlwave-sint-methods  '(nil))
+(defvar idlwave-sint-classes  '(nil))
+(defvar idlwave-sint-files    '(nil))
+
+(defun idlwave-reset-sintern (&optional what)
+  "Reset all sintern hashes."
+  ;; Make sure the hash functions are accessible.
+  (if (or (not (fboundp 'gethash))
+         (not (fboundp 'puthash)))
+      (progn 
+       (require 'cl)
+       (or (fboundp 'puthash)
+           (defalias 'puthash 'cl-puthash))))
+  (let ((entries '((idlwave-sint-routines 1000 10)
+                  (idlwave-sint-keywords 1000 10)
+                  (idlwave-sint-methods   100 10)
+                  (idlwave-sint-classes    10 10))))
+
+    ;; Make sure these are lists
+    (loop for entry in entries
+      for var = (car entry)
+      do (if (not (consp (symbol-value var))) (set var (list nil))))
+
+    (when (or (eq what t) (eq what 'syslib)
+             (null (cdr idlwave-sint-routines)))
+      ;; Reset the system & library hash
+      (loop for entry in entries
+       for var = (car entry) for size = (nth 1 entry)
+       do (setcdr (symbol-value var) 
+                  (make-hash-table ':size size ':test 'equal)))
+      (setq idlwave-sint-files nil))
+
+    (when (or (eq what t) (eq what 'bufsh)
+             (null (car idlwave-sint-routines)))
+      ;; Reset the buffer & shell hash
+      (loop for entry in entries
+       for var = (car entry) for size = (nth 1 entry)
+       do (setcar (symbol-value var) 
+                  (make-hash-table ':size size ':test 'equal))))))
+
+(defun idlwave-sintern-routine-or-method (name &optional class set)
+  (if class
+      (idlwave-sintern-method name set)
+    (idlwave-sintern-routine name set)))
+
+(defun idlwave-sintern (stype &rest args)
+  (apply (intern (concat "idlwave-sintern-" (symbol-name stype))) args))
+
+;;(defmacro idlwave-sintern (type var)
+;;  `(cond ((not (stringp name)) name)
+;;      ((gethash (downcase name) (cdr ,var)))
+;;      ((gethash (downcase name) (car ,var)))
+;;      (set (idlwave-sintern-set name ,type ,var set))
+;;      (name)))
+
+(defun idlwave-sintern-routine (name &optional set)
+  (cond ((not (stringp name)) name)
+       ((gethash (downcase name) (cdr idlwave-sint-routines)))
+       ((gethash (downcase name) (car idlwave-sint-routines)))
+       (set (idlwave-sintern-set name 'routine idlwave-sint-routines set))
+       (name)))
+(defun idlwave-sintern-keyword (name &optional set)
+  (cond ((not (stringp name)) name)
+       ((gethash (downcase name) (cdr idlwave-sint-keywords)))
+       ((gethash (downcase name) (car idlwave-sint-keywords)))
+       (set (idlwave-sintern-set name 'keyword idlwave-sint-keywords set))
+       (name)))
+(defun idlwave-sintern-method (name &optional set)
+  (cond ((not (stringp name)) name)
+       ((gethash (downcase name) (cdr idlwave-sint-methods)))
+       ((gethash (downcase name) (car idlwave-sint-methods)))
+       (set (idlwave-sintern-set name 'method idlwave-sint-methods set))
+       (name)))
+(defun idlwave-sintern-class (name &optional set)
+  (cond ((not (stringp name)) name)
+       ((gethash (downcase name) (cdr idlwave-sint-classes)))
+       ((gethash (downcase name) (car idlwave-sint-classes)))
+       (set (idlwave-sintern-set name 'class idlwave-sint-classes set))
+       (name)))
+
+(defun idlwave-sintern-file (name &optional set)
+  (car (or (member name idlwave-sint-files)
+          (setq idlwave-sint-files (cons name idlwave-sint-files)))))
+
+(defun idlwave-sintern-set (name type tables set)
+  (let* ((func (or (cdr (assq type idlwave-completion-case))
+                  'identity))
+        (iname (funcall (if (eq func 'preserve) 'identity func) name))
+        (table (if (eq set 'sys) (cdr tables) (car tables))))
+    (puthash (downcase name) iname table)
+    iname))
+
+(defun idlwave-sintern-rinfo-list (list &optional set)
+  "Sintern all strings in the rinfo LIST.  With optional parameter SET:
+also set new patterns.  Probably this will always have to be t."
+  (let (entry name type class kwds res source call olh new)
+    (while list
+      (setq entry (car list)
+           list (cdr list)
+           name (car entry)
+           type (nth 1 entry)
+           class (nth 2 entry)
+           source (nth 3 entry)
+           call (nth 4 entry)
+           kwds (nth 5 entry)
+           olh (nth 6 entry))
+      (setq kwds (mapcar (lambda (x)
+                          (list (idlwave-sintern-keyword (car x) set)))
+                        kwds))
+      (if class
+         (progn
+           (if (symbolp class) (setq class (symbol-name class)))
+           (setq class (idlwave-sintern-class class set))
+           (setq name (idlwave-sintern-method name set)))
+       (setq name (idlwave-sintern-routine name set)))
+      (if (stringp (cdr source))
+         (setcdr source (idlwave-sintern-file (cdr source) t)))
+      (setq new (if olh
+                   (list name type class source call kwds olh)
+                 (list name type class source call kwds)))
+      (setq res (cons new res)))
+    (nreverse res)))
+
+;;---------------------------------------------------------------------------
+
+
+;; The variables which hold the information
+(defvar idlwave-builtin-routines nil
+  "Holds the routine-info obtained by scanning buffers.")
+(defvar idlwave-buffer-routines nil
+  "Holds the routine-info obtained by scanning buffers.")
+(defvar idlwave-compiled-routines nil
+  "Holds the procedure routine-info obtained by asking the shell.")
+(defvar idlwave-library-routines nil
+  "Holds the procedure routine-info from the library scan.")
+(defvar idlwave-scanned-lib-directories nil
+  "The directories scanned to get libinfo.")
+(defvar idlwave-routines nil
+  "Holds the combinded procedure routine-info.")
+(defvar idlwave-class-alist nil
+  "Holds the class names known to IDLWAVE.")
+(defvar idlwave-class-history nil
+  "The history of classes selected with the minibuffer.")
+(defvar idlwave-force-class-query nil)
+(defvar idlwave-before-completion-wconf nil
+  "The window configuration just before the completion buffer was displayed.")
+
+;;
+;; The code to get routine info from different sources.
+
+(defvar idlwave-builtin-routines)
+(defun idlwave-routines ()
+  "Provide a list of IDL routines.
+This routine loads the builtin routines on the first call.  Later it
+only returns the value of the variable."
+  (or idlwave-routines
+      (progn
+       (idlwave-update-routine-info)
+       ;; return the current value
+       idlwave-routines)))
+
+(defun idlwave-update-routine-info (&optional arg)
+  "Update the internal routine-info lists.
+These lists are used by `idlwave-routine-info' (\\[idlwave-routine-info])
+and by `idlwave-complete' (\\[idlwave-complete]) to provide information
+about individual routines.
+
+The information can come from 4 sources:
+1. IDL programs in the current editing session
+2. Compiled modules in an IDL shell running as Emacs subprocess
+3. A list which covers the IDL system routines.
+4. A list which covers the prescanned library files.
+
+Scans all IDLWAVE-mode buffers of the current editing session (see
+`idlwave-scan-all-buffers-for-routine-info').
+When an IDL shell is running, this command also queries the IDL program
+for currently compiled routines.
+
+With prefix ARG, also reload the system and library lists.
+With two prefix ARG's, also rescans the library tree."
+  (interactive "P")
+  (if (equal arg '(16))
+      (idlwave-create-libinfo-file t)
+    (let* ((reload (or arg
+                      idlwave-buffer-case-takes-precedence
+                      (null idlwave-builtin-routines))))
+      
+      (setq idlwave-buffer-routines nil
+           idlwave-compiled-routines nil)
+      ;; Reset the appropriate hashes
+      (idlwave-reset-sintern (cond (reload t)
+                                  ((null idlwave-builtin-routines) t)
+                                  (t 'bufsh)))
+      
+      (if idlwave-buffer-case-takes-precedence
+         ;; We can safely scan the buffer stuff first
+         (progn
+           (idlwave-update-buffer-routine-info)
+           (and reload (idlwave-load-system-rinfo)))
+       ;; We first do the system info, and then the buffers
+       (and reload (idlwave-load-system-rinfo))
+       (idlwave-update-buffer-routine-info))
+
+      ;; Let's see if there is a shell
+      (let* ((shell-is-running (and (fboundp 'idlwave-shell-is-running)
+                                   (idlwave-shell-is-running)))
+            (ask-shell (and shell-is-running
+                            idlwave-query-shell-for-routine-info)))
+      
+       (if (or (not ask-shell)
+               (not (interactive-p)))
+           ;; 1. If we are not going to ask the shell, we need to do the
+           ;;    concatenation now.
+           ;; 2. When this function is called non-interactively, it means
+           ;;    that someone needs routine info *now*.  The shell update
+           ;;    causes the concatenation *delayed*, so not in time for
+           ;;    the current command.  Therefore, we do a concatenation
+           ;;    now, even though the shell might do it again.
+           (idlwave-concatenate-rinfo-lists))
+      
+       (when ask-shell
+         ;; Ask the shell about the routines it knows.
+         (message "Querying the shell")
+         (idlwave-shell-update-routine-info))))))
+
+(defun idlwave-load-system-rinfo ()
+  ;; Load and case-treat the system and lib info files.
+  (load "idlwave-rinfo" t)
+  (message "Normalizing idlwave-builtin-routines...")
+  (setq idlwave-builtin-routines
+       (idlwave-sintern-rinfo-list idlwave-builtin-routines 'sys))
+  (message "Normalizing idlwave-builtin-routines...done")
+  (setq idlwave-routines idlwave-builtin-routines)
+  (when (and (stringp idlwave-libinfo-file)
+            (file-regular-p idlwave-libinfo-file))
+    (condition-case nil
+       (progn
+         (load-file idlwave-libinfo-file)
+         (message "Normalizing idlwave-library-routines...")
+         (setq idlwave-library-routines (idlwave-sintern-rinfo-list
+                                         idlwave-library-routines 'sys))
+         (message "Normalizing idlwave-library-routines...done"))
+      (error nil))))
+
+(defun idlwave-update-buffer-routine-info ()
+  (let (res)
+    (if idlwave-scan-all-buffers-for-routine-info
+       (progn
+         ;; Scan all buffers, current buffer last
+         (message "Scanning all buffers...")
+         (setq res (idlwave-get-routine-info-from-buffers 
+                    (reverse (buffer-list)))))
+      ;; Just scan this buffer
+      (if (eq major-mode 'idlwave-mode)
+         (progn
+           (message "Scanning current buffer...")
+           (setq res (idlwave-get-routine-info-from-buffers
+                      (list (current-buffer)))))))
+    ;; Put the result into the correct variable
+    (setq idlwave-buffer-routines 
+         (idlwave-sintern-rinfo-list res t))))
+
+(defun idlwave-concatenate-rinfo-lists ()
+  "Put the different sources for routine information together."
+  ;; The sequence here is important because earlier definitions shadow 
+  ;; later ones.  We assume that if things in the buffers are newer
+  ;; then in the shell of the system, it is meant to be different.
+  ;; FIXME: should the builtin stuff be before the library?
+  ;;        This is how IDL searches, the user may also have
+  ;;        functions overloading system stuff, and then the lib
+  ;;        should be first.  Difficult to find a general solution.
+  ;; FIXME: can't we use nconc here in some way, to save memory?
+  ;;        This is possible for buffer abd shell stuff, but these are 
+  ;;        small anyway, and so it is not so critical.
+  (setq idlwave-routines (append idlwave-buffer-routines
+                                idlwave-compiled-routines
+                                idlwave-library-routines
+                                idlwave-builtin-routines))
+  (setq idlwave-class-alist nil)
+  (let (class)
+    (loop for x in idlwave-routines do
+      (when (and (setq class (nth 2 x))
+                (not (assq class idlwave-class-alist)))
+       (push (list class) idlwave-class-alist))))
+  ;; Give a message with information about the number of routines we have.
+  (message 
+   "Routine info updated:  buffer(%d)  compiled(%d)  library(%d)  system(%d)"
+   (length idlwave-buffer-routines)
+   (length idlwave-compiled-routines)
+   (length idlwave-library-routines)
+   (length idlwave-builtin-routines)))
+
+;;----- Scanning buffers -------------------
+
+(defun idlwave-get-routine-info-from-buffers (buffers)
+  "Call `idlwave-get-buffer-routine-info' on idlwave-mode buffers in BUFFERS."
+  (let (buf routine-lists res)
+    (save-excursion
+      (while (setq buf (pop buffers))
+       (set-buffer buf)
+       (if (eq major-mode 'idlwave-mode)
+           ;; yes, this buffer has the right mode.
+           (progn (setq res (condition-case nil
+                                (idlwave-get-buffer-routine-info)
+                              (error nil)))
+                  (push res routine-lists)))))
+    ;; Concatenate the individual lists and return the result
+    (apply 'nconc routine-lists)))
+
+(defun idlwave-get-buffer-routine-info ()
+  "Scan the current buffer for routine info.  Return (PRO-LIST FUNC-LIST)."
+  (let* ((case-fold-search t)
+        routine-list string entry)
+    (save-excursion
+      (save-restriction
+       (widen)
+       (goto-char (point-min))
+       (while (re-search-forward 
+               "^[ \t]*\\<\\(pro\\|function\\)\\>" nil t)
+         (setq string (buffer-substring 
+                       (match-beginning 0)
+                       (progn 
+                         (idlwave-end-of-statement)
+                         (point))))
+         (setq entry (idlwave-parse-definition string))
+         (push entry routine-list))))
+    routine-list))
+
+(defun idlwave-parse-definition (string)
+  "Parse a module definition."
+  (let ((case-fold-search t)
+       start name args type keywords class)
+    ;; Remove comments
+    (while (string-match ";.*" string)
+      (setq string (replace-match "" t t string)))
+    ;; Remove the continuation line stuff
+    (while (string-match "\\([^a-zA-Z0-9$_]\\)\\$[ \t]*\n" string)
+      (setq string (replace-match "\\1 " t nil string)))
+    ;; Match the name and type.
+    (when (string-match
+          "\\<\\(pro\\|function\\)\\>\\s-+\\(\\([a-zA-Z0-9$_]+\\)::\\)?\\([a-zA-Z0-9$_]+\\)" string)
+      (setq start (match-end 0))
+      (setq type (downcase (match-string 1 string)))
+      (if (match-beginning 3)
+         (setq class (match-string 3 string)))
+      (setq name (match-string 4 string)))
+    ;; Match normal args and keyword args
+    (while (string-match
+           ",\\s-*\\([a-zA-Z][a-zA-Z0-9$_]*\\|_extra\\)\\s-*\\(=\\)?"
+           string start)
+      (setq start (match-end 0))
+      (if (match-beginning 2)
+         (push (match-string 1 string) keywords)
+       (push (match-string 1 string) args)))
+    ;; Normalize and sort.
+    (setq args (nreverse args))
+    (setq keywords (sort keywords (lambda (a b) 
+                                   (string< (downcase a) (downcase b)))))
+    ;; Make and return the entry
+    ;; We don't know which argument are optional, so this information
+    ;; will not be contained in the calling sequence.
+    (list name
+         (if (equal type "pro") 'pro 'fun)
+         class
+         (cond ((not (boundp 'idlwave-scanning-lib))
+                (cons 'buffer (buffer-file-name)))
+               ((string= (downcase 
+                          (file-name-sans-extension
+                           (file-name-nondirectory (buffer-file-name))))
+                         (downcase name))
+                (list 'lib))
+               (t (cons 'lib (file-name-nondirectory (buffer-file-name)))))
+         (concat 
+          (if (string= type "function") "Result = " "")
+          (if class "Obj ->[%s::]" "")
+          "%s"
+          (if args
+              (concat
+               (if (string= type "function") "(" ", ")
+               (mapconcat 'identity args ", ")
+               (if (string= type "function") ")" ""))))
+         (if keywords
+             (mapcar 'list keywords)
+           nil))))
+
+;;----- Scanning the library -------------------
+
+(defun idlwave-create-libinfo-file (&optional arg)
+  "Scan all files on selected dirs of IDL search path for routine information.
+A widget checklist will allow you to choose the directories.
+Write the result as a file `idlwave-libinfo-file'.  When this file exists,
+will be automatically loaded to give routine information about library
+routines.
+With ARG, just rescan the same directories as last time - so no widget
+will pop up."
+  (interactive "P")
+  ;; Make sure the file is loaded if it exists.
+  (if (and (stringp idlwave-libinfo-file)
+          (file-regular-p idlwave-libinfo-file))
+      (condition-case nil
+         (load-file idlwave-libinfo-file)
+       (error nil)))
+  ;; Make sure the file name makes sense
+  (unless (and (stringp idlwave-libinfo-file)
+              (file-accessible-directory-p
+               (file-name-directory idlwave-libinfo-file))
+              (not (string= "" (file-name-nondirectory 
+                                idlwave-libinfo-file))))
+    (error "`idlwave-libinfo-file' does not point to file in accessible directory."))
+  
+  (cond
+   ((and arg idlwave-scanned-lib-directories)
+    ;; Rescan the known directories
+    (idlwave-scan-lib-files idlwave-scanned-lib-directories))
+   (idlwave-library-path
+    ;; Get the directories from that variable
+    (idlwave-display-libinfo-widget
+     (idlwave-expand-path idlwave-library-path)
+     idlwave-scanned-lib-directories))
+   (t
+    ;; Ask the shell for the path and run the widget
+    (message "Asking the shell for IDL path...")
+    (idlwave-shell-send-command 
+     "__pa=expand_path(!path,/array)&for i=0,n_elements(__pa)-1 do print,'PATH:',__pa[i]"
+     '(idlwave-libinfo-command-hook nil)
+     'hide))))
+
+(defun idlwave-libinfo-command-hook (&optional arg)
+  ;; Command hook used by `idlwave-create-libinfo-file'.
+  (if arg
+      ;; Scan immediately
+      (idlwave-scan-lib-files idlwave-scanned-lib-directories)
+    ;; Display the widget
+    (idlwave-display-libinfo-widget (idlwave-shell-path-filter)
+                                   idlwave-scanned-lib-directories)))
+
+(defvar idlwave-shell-command-output)
+(defun idlwave-shell-path-filter ()
+  ;; Convert the output of the path query into a list of directories
+  (let ((path-string idlwave-shell-command-output)
+       (case-fold-search t)
+       (start 0)
+       dirs)
+    (while (string-match "^PATH:[ \t]*\\(.*\\)\n" path-string start)
+      (push (match-string 1 path-string) dirs)
+      (setq start (match-end 0)))
+    (nreverse dirs)))
+
+(defconst idlwave-libinfo-widget-help-string 
+  "This is the front-end to the creation of IDLWAVE library routine info.
+Please select below the directories on IDL's search path from which you
+would like to extract routine information, which will be stored in the file
+
+             %s
+
+If this is not the correct file, first set variable `idlwave-libinfo-file'.
+Then call this command again.
+After selecting the directories, choose [Scan & Save] to scan the library
+directories and save the routine info.
+\n")
+
+(defvar idlwave-widget)
+(defvar widget-keymap)
+(defun idlwave-display-libinfo-widget (dirs selected-dirs)
+  "Create the widget to select IDL search path directories for scanning."
+  (interactive)
+  (require 'widget)
+  (require 'wid-edit)
+  (unless dirs
+      (error "Don't know IDL's search path"))
+
+  ;; Allow only those directories to be selected which are in the path.
+  (setq selected-dirs (delq nil (mapcar (lambda (x)
+                                         (if (member x dirs) x nil))
+                                       selected-dirs)))
+  (kill-buffer (get-buffer-create "*IDLWAVE Widget*"))
+  (switch-to-buffer (get-buffer-create "*IDLWAVE Widget*"))
+  (kill-all-local-variables)
+  (make-local-variable 'idlwave-widget)
+  (widget-insert (format idlwave-libinfo-widget-help-string
+                        idlwave-libinfo-file))
+  
+  (widget-create 'push-button
+                :notify 'idlwave-widget-scan-lib-files
+                :help-echo "testing"
+                "Scan & Save")
+  (widget-insert "  ")
+  (widget-create 'push-button
+                :notify (lambda (&rest ignore)
+                          (kill-buffer (current-buffer)))
+                "Quit")
+  (widget-insert "  ")
+  (widget-create 'push-button
+                :notify 'idlwave-delete-libinfo-file
+                "Delete File")
+  (widget-insert "  ")
+  (widget-create 'push-button
+                :notify '(lambda (&rest ignore)
+                           (idlwave-display-libinfo-widget
+                            (widget-get idlwave-widget :path-dirs)
+                            (widget-get idlwave-widget :path-dirs)))
+                "Select All")
+  (widget-insert "  ")
+  (widget-create 'push-button
+                :notify '(lambda (&rest ignore)
+                           (idlwave-display-libinfo-widget
+                            (widget-get idlwave-widget :path-dirs)
+                            nil))
+                "Deselect All")
+  (widget-insert "\n\n")
+
+  (widget-insert "Select Directories\n")
+  
+  (setq idlwave-widget
+       (apply 'widget-create
+              'checklist
+              :value selected-dirs
+              :greedy t
+              :tag "List of directories"
+              (mapcar (lambda (x) (list 'item x)) dirs)))
+  (widget-put idlwave-widget :path-dirs dirs)
+  (widget-insert "\n")
+  (use-local-map widget-keymap)
+  (widget-setup)
+  (goto-char (point-min))
+  (delete-other-windows))
+  
+(defun idlwave-delete-libinfo-file (&rest ignore)
+  (if (yes-or-no-p
+       (format "Delete file %s " idlwave-libinfo-file))
+      (progn
+       (delete-file idlwave-libinfo-file)
+       (message "%s has been deleted" idlwave-libinfo-file))))
+
+(defun idlwave-widget-scan-lib-files (&rest ignore)
+  ;; Call `idlwave-scan-lib-files' with data taken from the widget.
+  (let* ((widget idlwave-widget)
+        (selected-dirs (widget-value widget)))
+    (idlwave-scan-lib-files selected-dirs)))
+
+(defvar font-lock-mode)
+(defun idlwave-scan-lib-files (selected-dirs)
+  ;; Scan the files in SELECTED-DIRS and store the info in a file
+  (let* ((idlwave-scanning-lib t)
+        (idlwave-completion-case nil)
+        dirs dir files file)
+    (setq idlwave-library-routines nil)
+    (setq idlwave-scanned-lib-directories selected-dirs)
+    (save-excursion
+      (set-buffer (get-buffer-create "*idlwave-scan.pro*"))
+      (idlwave-mode)
+      (setq dirs (reverse selected-dirs))
+      (while (setq dir (pop dirs))
+       (when (file-directory-p dir)
+         (setq files (directory-files dir 'full "\\.[pP][rR][oO]\\'"))
+         (while (setq file (pop files))
+           (when (file-regular-p file)
+             (if (not (file-readable-p file))
+                 (message "Skipping %s (no read permission)" file)
+               (message "Scanning %s..." file)
+               (erase-buffer)
+               (insert-file-contents file 'visit)
+               (setq idlwave-library-routines
+                     (append (idlwave-get-routine-info-from-buffers
+                              (list (current-buffer)))
+                             idlwave-library-routines)))
+             )))))
+    (kill-buffer "*idlwave-scan.pro*")
+    (kill-buffer (get-buffer-create "*IDLWAVE Widget*"))
+    (let ((font-lock-maximum-size 0))
+      (find-file idlwave-libinfo-file))
+    (if (and (boundp 'font-lock-mode)
+            font-lock-mode)
+       (font-lock-mode 0))
+    (erase-buffer)
+    (insert ";; IDLWAVE libinfo file\n")
+    (insert (format ";; Created %s\n\n" (current-time-string)))
+
+    ;; Define the variable which contains a list of all scanned directories
+    (insert "\n(setq idlwave-scanned-lib-directories\n    '(")
+    (mapcar (lambda (x)
+             (insert (format "\n      \"%s\"" x)))
+           selected-dirs)
+    (insert "))\n")
+    ;; Define the routine info list
+    (insert "\n(setq idlwave-library-routines\n    '(")
+    (mapcar (lambda (x)
+             (insert "\n    ")
+             (insert (with-output-to-string (prin1 x))))
+           idlwave-library-routines)
+    (insert (format "))\n\n;;; %s ends here\n"
+                   (file-name-nondirectory idlwave-libinfo-file)))
+    (goto-char (point-min))
+    ;; Save the buffer
+    (save-buffer 0)
+    (kill-buffer (current-buffer)))
+  (message "Info for %d routines saved in %s"
+          (length idlwave-library-routines)
+          idlwave-libinfo-file)
+  (sit-for 2)
+  (idlwave-update-routine-info t))
+
+(defun idlwave-expand-path (path &optional default-dir)
+  ;; Expand parts of path starting with '+' recursively into directory list.
+  ;; Relative recursive path elements are expanded relative to DEFAULT-DIR.
+  (message "Expanding path...")
+  (let (path1 dir recursive)
+    (while (setq dir (pop path))
+      (if (setq recursive (string= (substring dir 0 1) "+"))
+         (setq dir (substring dir 1)))
+      (if (and recursive
+              (not (file-name-absolute-p dir)))
+         (setq dir (expand-file-name dir default-dir)))
+      (if recursive
+         ;; Expand recursively
+         (setq path1 (append (idlwave-recursive-directory-list dir) path1))
+       ;; Keep unchanged
+       (push dir path1)))
+    (message "Expanding path...done")
+    (nreverse path1)))
+
+(defun idlwave-recursive-directory-list (dir)
+  ;; Return a list of all directories below DIR, including DIR itself
+  (let ((path (list dir)) path1 file files)
+    (while (setq dir (pop path))
+      (when (file-directory-p dir)
+       (setq files (nreverse (directory-files dir t "[^.]")))
+       (while (setq file (pop files))
+         (if (file-directory-p file) 
+             (push (file-name-as-directory file) path)))
+       (push dir path1)))
+    path1))
+
+;;----- Asking the shell -------------------
+
+;; First, here is the idl program which can be used to query IDL for
+;; defined routines. 
+(defconst idlwave-routine-info.pro
+  "
+function idlwave_make_info_entry,name,func=func,separator=sep
+  ;; See if it's an object method
+  func    = keyword_set(func) 
+  methsep = strpos(name,'::')
+  meth    = methsep ne -1
+  
+  ;; Get routine info
+  pars   = routine_info(name,/parameters,functions=func)
+  source = routine_info(name,/source,functions=func)
+  nargs  = pars.num_args
+  nkw    = pars.num_kw_args
+  if nargs gt 0 then args = pars.args
+  if nkw   gt 0 then kwargs = pars.kw_args
+  
+  ;; Trim the class, and make the name
+  if meth then begin 
+      class = strmid(name,0,methsep)
+      name  = strmid(name,methsep+2,strlen(name)-1)
+      if nargs gt 0 then begin 
+          ;; remove the self argument
+          wh = where(args ne 'SELF',nargs)
+          if nargs gt 0 then args = args(wh)
+      endif
+  endif else begin
+      ;; No class, just a normal routine.
+      class = \"\"
+  endelse
+   
+  ;; Calling sequence
+  cs = \"\"
+  if func then cs = 'Result = '
+  if meth then cs = cs + 'Obj -> [' + '%s' + '::]'
+  cs = cs + '%s'
+  if func then cs = cs + '(' else if nargs gt 0 then cs = cs + ', '
+  if nargs gt 0 then begin
+      for j=0,nargs-1 do begin
+          cs = cs + args(j)
+          if j lt nargs-1 then cs = cs + ', '
+      endfor
+  end
+  if func then cs = cs + ')'
+  ;; Keyword arguments
+  kwstring = ''
+  if nkw gt 0 then begin
+      for j=0,nkw-1 do begin
+          kwstring = kwstring + ' ' + kwargs(j)
+      endfor
+  endif
+  
+  ret=(['IDLWAVE-PRO','IDLWAVE-FUN', $
+        'IDLWAVE-PRO','IDLWAVE-FUN'])(func+2*meth)
+  
+  return,ret + ': ' + name + sep + class + sep + source(0).path  $
+    + sep + cs + sep + kwstring
+end
+
+pro idlwave_routine_info
+  sep = '<@>'
+  print,'>>>BEGIN OF IDLWAVE ROUTINE INFO (\"' + sep + '\" IS THE SEPARATOR)'
+  all = routine_info()
+  for i=0,n_elements(all)-1 do $
+    print,idlwave_make_info_entry(all(i),separator=sep)
+  all = routine_info(/functions)
+  for i=0,n_elements(all)-1 do $
+    print,idlwave_make_info_entry(all(i),/func,separator=sep)
+  print,'>>>END OF IDLWAVE ROUTINE INFO'
+end
+" 
+  "The idl program to get the routine info stuff.
+The output of this program is parsed by `idlwave-shell-routine-info-filter'.")
+
+(defun idlwave-shell-routine-info-filter ()
+  "Function which parses the special output from idlwave_routine_info.pro."
+  (let ((text idlwave-shell-command-output)
+       (start 0)
+       sep sep-re file type spec specs name cs key keys class)
+    ;; Initialize variables
+    (setq idlwave-compiled-routines nil)
+    ;; Cut out the correct part of the output.
+    (if (string-match
+        "^>>>BEGIN OF IDLWAVE ROUTINE INFO (\"\\(.+\\)\" IS THE SEPARATOR.*"
+        text)
+       (setq sep (match-string 1 text)
+             sep-re (concat (regexp-quote sep) " *")
+             text (substring text (match-end 0)))
+      (error "Routine Info error: No match for BEGIN line"))
+    (if (string-match "^>>>END OF IDLWAVE ROUTINE INFO.*" text)
+       (setq text (substring text 0 (match-beginning 0)))
+      (error "Routine Info error: No match for END line"))
+    ;; Match the output lines
+    (while (string-match "^IDLWAVE-\\(PRO\\|FUN\\): \\(.*\\)" text start)
+      (setq start (match-end 0))
+      (setq type (match-string 1 text)
+           spec (match-string 2 text)
+           specs (idlwave-split-string spec sep-re)
+           name (nth 0 specs)
+           class (if (equal (nth 1 specs) "") nil (nth 1 specs))
+           file (nth 2 specs)
+           cs (nth 3 specs)
+           key (nth 4 specs)
+           keys (if (and (stringp key)
+                         (not (string-match "\\` *\\'" key)))
+                    (mapcar 'list 
+                            (delete "" (idlwave-split-string key " +")))))
+      (setq name (idlwave-sintern-routine-or-method name class t)
+           class (idlwave-sintern-class class t)
+           keys (mapcar (lambda (x) 
+                          (list (idlwave-sintern-keyword (car x) t))) keys))
+      ;; Make sure we use the same string object for the same file
+      (setq file (idlwave-sintern-file file t))
+      ;; FIXME: What should I do with routines from the temp file???
+      ;;        Maybe just leave it in - there is a chance that the
+      ;;        routine is still in there.
+      ;;      (if (equal file idlwave-shell-temp-pro-file)
+      ;;         (setq file nil))
+
+      ;; In the following ignore routines already defined in buffers,
+      ;; assuming that if the buffer stuff differs, it is a "new"
+      ;; version. 
+      ;; We could do the same for the library to avoid duplicates -
+      ;; but I think frequently a user might have several versions of
+      ;; the same function in different programs, and in this case the 
+      ;; compiled one will be the best guess of all version.
+      ;; Therefore, we leave duplicates of library routines in.
+
+      (cond ((string= name "$MAIN$"))    ; ignore this one
+           ((and (string= type "PRO")
+                 ;; FIXME: is it OK to make the buffer routines dominate?
+                 (not (idlwave-rinfo-assq name 'pro class 
+                                          idlwave-buffer-routines))
+                 ;; FIXME: is it OK to make the library routines dominate?
+                 ;;(not (idlwave-rinfo-assq name 'pro class 
+                 ;;                       idlwave-library-routines))
+                 )
+            (push (list name 'pro class (cons 'compiled file) cs keys)
+                  idlwave-compiled-routines))
+           ((and (string= type "FUN")
+                 ;; FIXME: is it OK to make the buffer routines dominate?
+                 (not (idlwave-rinfo-assq name 'fun class 
+                                          idlwave-buffer-routines))
+                 ;; FIXME: is it OK to make the library routines dominate?
+                 ;; (not (idlwave-rinfo-assq name 'fun class 
+                 ;;                       idlwave-library-routines))
+                 )
+            (push (list name 'fun class (cons 'compiled file) cs keys)
+                  idlwave-compiled-routines)))))
+  ;; Reverse the definitions so that they are alphabetically sorted.
+  (setq idlwave-compiled-routines
+       (nreverse idlwave-compiled-routines)))
+
+(defvar idlwave-shell-temp-pro-file)
+(defun idlwave-shell-update-routine-info ()
+  "Query the shell for routine_info of compiled modules and update the lists."
+  ;; Save and compile the procedure
+  (save-excursion
+    (set-buffer (idlwave-find-file-noselect
+                idlwave-shell-temp-pro-file))
+    (erase-buffer)
+    (insert idlwave-routine-info.pro)
+    (save-buffer 0))
+  (idlwave-shell-send-command (concat ".run " idlwave-shell-temp-pro-file)
+                             nil 'hide)
+
+  ;; Execute the procedure and analyze the output
+  (idlwave-shell-send-command "idlwave_routine_info"
+                             '(progn
+                                (idlwave-shell-routine-info-filter)
+                                (idlwave-concatenate-rinfo-lists))
+                             'hide))
+
+;; ---------------------------------------------------------------------------
+;;
+;; Completion and displaying routine calling sequences
+
+(defun idlwave-complete (&optional arg module class)
+  "Complete a function, procedure or keyword name at point.
+This function is smart and figures out what can be legally completed
+at this point.
+- At the beginning of a statement it completes procedure names.
+- In the middle of a statement it completes function names.
+- after a `(' or `,' in the argument list of a function or procedure,
+  it completes a keyword of the relevant function or procedure.
+- In the first arg of `OBJ_NEW', it completes a class name.
+
+When several completions are possible, a list will be displayed in the 
+*Completions* buffer.  If this list is too long to fit into the
+window, scrolling can be achieved by repeatedly pressing \\[idlwave-complete].
+
+The function also knows about object methods.  When it needs a class
+name, the action depends upon `idlwave-query-class', which see.  You
+can force IDLWAVE to ask you for a class name with a \\[universal-argument] prefix
+argument to this command.
+
+See also the variables `idlwave-keyword-completion-adds-equal' and
+`idlwave-function-completion-adds-paren'.
+
+The optional ARG can be used to specify the completion type in order
+to override IDLWAVE's idea of what should be completed at point.
+Possible values are:
+
+0  <=>  query for the completion type
+1  <=>  'procedure
+2  <=>  'procedure-keyword
+3  <=>  'function
+4  <=>  'function-keyword
+5  <=>  'procedure-method
+6  <=>  'procedure-method-keyword
+7  <=>  'function-method
+8  <=>  'function-method-keyword
+9  <=>  'class
+
+For Lisp programmers only:
+When we force a keyword, optional argument MODULE can contain the module name.
+When we force a method or a method keyword, CLASS can specify the class."
+  (interactive "P")
+  (idlwave-routines)
+  (let* ((where-list
+         (if (and arg
+                  (or (integerp arg)
+                      (symbolp arg)))
+             (idlwave-make-force-complete-where-list arg module class)
+           (idlwave-where)))
+        (what (nth 2 where-list))
+        (idlwave-force-class-query (equal arg '(4)))
+        cwin)
+
+    (if (and module (string-match "::" module))
+       (setq class (substring module 0 (match-beginning 0))
+             module (substring module (match-end 0))))
+
+    (cond
+
+     ((and (null arg)
+          (eq (car-safe last-command) 'idlwave-display-completion-list)
+          (setq cwin (get-buffer-window "*Completions*")))
+      (setq this-command last-command)
+      (idlwave-scroll-completions))
+
+     ((null what)
+      (error "Nothing to complete here"))
+
+     ((eq what 'class)
+      (idlwave-complete-class))
+
+     ((eq what 'procedure)
+      ;; Complete a procedure name
+      (let* ((class-selector (idlwave-determine-class (nth 3 where-list) 'pro))
+            (isa (concat "procedure" (if class-selector "-method" "")))
+            (type-selector 'pro))
+       (idlwave-complete-in-buffer
+        'procedure (if class-selector 'method 'routine)
+        (idlwave-routines) 'idlwave-selector
+        (format "Select a %s name%s"
+                isa
+                (if class-selector
+                    (format " (class is %s)" class-selector)
+                  ""))
+        isa
+        'idlwave-attach-method-classes)))
+
+     ((eq what 'function)
+      ;; Complete a function name
+      (let* ((class-selector (idlwave-determine-class (nth 3 where-list) 'fun))
+            (isa (concat "function" (if class-selector "-method" "")))
+            (type-selector 'fun))
+       (idlwave-complete-in-buffer
+        'function (if class-selector 'method 'routine)
+        (idlwave-routines) 'idlwave-selector
+        (format "Select a %s name%s"
+                isa
+                (if class-selector
+                    (format " (class is %s)" class-selector)
+                  ""))
+        isa
+        'idlwave-attach-method-classes)))
+
+     ((eq what 'procedure-keyword)
+      ;; Complete a procedure keyword
+      (let* ((where (nth 3 where-list))
+            (name  (car where))
+            (method-selector name)
+            (type-selector 'pro)
+            (class (idlwave-determine-class where 'pro))
+            (class-selector class)
+            (isa (format "procedure%s-keyword" (if class "-method" "")))
+            (entry (idlwave-rinfo-assq
+                    name 'pro class (idlwave-routines)))
+            (list (nth 5 entry)))
+       (unless (or entry (eq class t))
+         (error "Nothing known about procedure %s"
+                (idlwave-make-full-name class name)))
+       (setq list (idlwave-fix-keywords name 'pro class list))
+       (unless list (error (format "No keywords available for procedure %s"
+                                   (idlwave-make-full-name class name))))
+       (idlwave-complete-in-buffer
+        'keyword 'keyword list nil
+        (format "Select keyword for procedure %s%s"
+                (idlwave-make-full-name class name)
+                (if (member '("_EXTRA") list) " (note _EXTRA)" ""))
+        isa
+        'idlwave-attach-keyword-classes)))
+
+     ((eq what 'function-keyword)
+      ;; Complete a function keyword
+      (let* ((where (nth 3 where-list))
+            (name  (car where))
+            (method-selector name)
+            (type-selector 'fun)
+            (class (idlwave-determine-class where 'fun))
+            (class-selector class)
+            (isa (format "function%s-keyword" (if class "-method" "")))
+            (entry (idlwave-rinfo-assq
+                    name 'fun class (idlwave-routines)))
+            (list (nth 5 entry)))
+       (unless (or entry (eq class t))
+         (error "Nothing known about function %s"
+                (idlwave-make-full-name class name)))
+       (setq list (idlwave-fix-keywords name 'fun class list))
+       (unless list (error (format "No keywords available for function %s"
+                                   (idlwave-make-full-name class name))))
+       (idlwave-complete-in-buffer
+        'keyword 'keyword list nil
+        (format "Select keyword for function %s%s"
+                (idlwave-make-full-name class name)
+                (if (member '("_EXTRA") list) " (note _EXTRA)" ""))
+        isa
+        'idlwave-attach-keyword-classes)))
+                
+     (t (error "This should not happen (idlwave-complete)")))))
+
+(defun idlwave-make-force-complete-where-list (what &optional module class)
+  ;; Return an artificial WHERE specification to force the completion
+  ;; routine to complete a specific item independent of context.
+  ;; WHAT is the prefix arg of `idlwave-complete', see there for details.
+  ;; MODULE and CLASS can be used to specify the routine name and class.
+  ;; The class name will also be found in MODULE if that is like "class::mod".
+  (let* ((what-list '(("procedure") ("procedure-keyword")
+                     ("function") ("function-keyword")
+                     ("procedure-method") ("procedure-method-keyword")
+                     ("function-method") ("function-method-keyword")
+                     ("class")))
+        (module (idlwave-sintern-routine-or-method module class))
+        (class (idlwave-sintern-class class))
+        (what (cond 
+               ((equal what 0)
+                (setq what
+                      (intern (completing-read 
+                               "Complete what? " what-list nil t))))
+               ((integerp what)
+                (setq what (intern (car (nth (1- what) what-list)))))
+               ((and what
+                     (symbolp what)
+                     (assoc (symbol-name what) what-list))
+                what)
+               (t (error "Illegal WHAT"))))
+        (nil-list '(nil nil nil nil))
+        (class-list (list nil nil (or class t) nil)))
+
+    (cond
+
+     ((eq what 'procedure)
+      (list nil-list nil-list 'procedure nil-list nil))
+
+     ((eq what 'procedure-keyword)
+      (let* ((class-selector nil)
+            (type-selector 'pro)
+            (pro (or module
+                     (idlwave-completing-read 
+                      "Procedure: " (idlwave-routines) 'idlwave-selector))))
+       (setq pro (idlwave-sintern-routine pro))
+       (list nil-list nil-list 'procedure-keyword
+             (list pro nil nil nil) nil)))
+
+     ((eq what 'function)
+      (list nil-list nil-list 'function nil-list nil))
+
+     ((eq what 'function-keyword)
+      (let* ((class-selector nil)
+            (type-selector 'fun)
+            (func (or module
+                      (idlwave-completing-read 
+                       "Function: " (idlwave-routines) 'idlwave-selector))))
+       (setq func (idlwave-sintern-routine func))
+       (list nil-list nil-list 'function-keyword
+             (list func nil nil nil) nil)))
+
+     ((eq what 'procedure-method)
+      (list nil-list nil-list 'procedure class-list nil))
+
+     ((eq what 'procedure-method-keyword)
+      (let* ((class (idlwave-determine-class class-list 'pro))
+            (class-selector class)
+            (type-selector 'pro)
+            (pro (or module
+                     (idlwave-completing-read
+                      (format "Procedure in %s class: " class-selector)
+                      (idlwave-routines) 'idlwave-selector))))
+       (setq pro (idlwave-sintern-method pro))
+       (list nil-list nil-list 'procedure-keyword
+             (list pro nil class nil) nil)))
+
+     ((eq what 'function-method)
+      (list nil-list nil-list 'function class-list nil))
+
+     ((eq what 'function-method-keyword)
+      (let* ((class (idlwave-determine-class class-list 'fun))
+            (class-selector class)
+            (type-selector 'fun)
+            (func (or module
+                      (idlwave-completing-read
+                       (format "Function in %s class: " class-selector)
+                       (idlwave-routines) 'idlwave-selector))))
+       (setq func (idlwave-sintern-method func))
+       (list nil-list nil-list 'function-keyword
+             (list func nil class nil) nil)))
+
+     ((eq what 'class)
+      (list nil-list nil-list 'class nil-list nil))
+     
+     (t (error "Illegal value for WHAT")))))
+
+(defun idlwave-completing-read (&rest args)
+  ;; Completing read, case insensitive
+  (let ((old-value (default-value 'completion-ignore-case)))
+    (unwind-protect
+       (progn
+         (setq-default completion-ignore-case t)
+         (apply 'completing-read args))
+      (setq-default completion-ignore-case old-value))))
+
+(defun idlwave-make-full-name (class name)
+  ;; Make a fully qualified module name including the class name
+  (concat (if class (format "%s::" class) "") name))
+
+(defun idlwave-rinfo-assq (name type class list)
+  ;; Works like assq, but also checks type and class
+  (catch 'exit
+    (let (match)
+      (while (setq match (assq name list))
+       (and (or (eq type t)
+                (eq (nth 1 match) type))
+            (eq (nth 2 match) class)
+            (throw 'exit match))
+       (setq list (cdr (memq match list)))))))
+
+(defun idlwave-all-assq (key list)
+  "Return a list of all associations of Key in LIST."
+  (let (rtn elt)
+    (while (setq elt (assq key list))
+      (push elt rtn)
+      (setq list (cdr (memq elt list))))
+    (nreverse rtn)))
+
+(defun idlwave-all-method-classes (method &optional type)
+  "Return all classes which have a method METHOD.  TYPE is 'fun or 'pro.
+When TYPE is not specified, both procedures and functions will be considered."
+  (if (null method)
+      (mapcar 'car idlwave-class-alist)
+    (let (rtn)
+      (mapcar (lambda (x)
+               (and (nth 2 x)
+                    (or (not type)
+                        (eq type (nth 1 x)))
+                    (push (nth 2 x) rtn)))
+             (idlwave-all-assq method (idlwave-routines)))
+      (idlwave-uniquify rtn))))
+
+(defun idlwave-all-method-keyword-classes (method keyword &optional type)
+  "Return all classes which have a method METHOD with keyword KEYWORD.
+TYPE is 'fun or 'pro.
+When TYPE is not specified, both procedures and functions will be considered."
+  (if (or (null method)
+         (null keyword))
+      nil
+    (let (rtn)
+      (mapcar (lambda (x)
+               (and (nth 2 x)
+                    (or (not type)
+                        (eq type (nth 1 x)))
+                    (assoc keyword (nth 5 x))
+                    (push (nth 2 x) rtn)))
+             (idlwave-all-assq method (idlwave-routines)))
+      (idlwave-uniquify rtn))))
+
+(defun idlwave-determine-class (info type)
+  ;; Determine the class of a routine call.  INFO is the structure returned
+  ;; `idlwave-what-function' or `idlwave-what-procedure'.
+  ;; The third element in this structure is the class.  When nil, we return nil.
+  ;; When t, try to get the class from text properties at the arrow,
+  ;; otherwise prompt the user for a class name.  Also stores the selected
+  ;; class as a text property at the arrow.
+  ;; TYPE is 'fun or 'pro.
+  (let* ((class (nth 2 info))
+        (apos (nth 3 info))
+        (nassoc (assoc (if (stringp (car info))
+                           (upcase (car info))
+                         (car info))
+                       idlwave-query-class))
+        (dassoc (assq (if (car info) 'keyword-default 'method-default)
+                      idlwave-query-class))
+        (query (cond (nassoc (cdr nassoc))
+                     (dassoc (cdr dassoc))
+                     (t t)))
+        (arrow (and apos (string= (buffer-substring apos (+ 2 apos)) "->")))
+        (force-query idlwave-force-class-query)
+        store class-alist)
+    (cond
+     ((null class) nil)
+     ((eq t class)
+      ;; There is an object which would like to know its class
+      (if (and arrow (get-text-property apos 'idlwave-class)
+              idlwave-store-inquired-class
+              (not force-query))
+         (setq class (get-text-property apos 'idlwave-class)
+               class (idlwave-sintern-class class)))
+      (when (and (eq class t)
+                (or force-query query))
+       (setq class-alist 
+             (mapcar 'list (idlwave-all-method-classes (car info) type)))
+       (setq class
+             (idlwave-sintern-class
+              (cond
+               ((and (= (length class-alist) 0) (not force-query))
+                (error "No classes available with method %s" (car info)))
+               ((and (= (length class-alist) 1) (not force-query))
+                (car (car class-alist)))
+               (t 
+                (setq store idlwave-store-inquired-class)
+                (idlwave-completing-read 
+                 (format "Class%s: " (if (stringp (car info))
+                                         (format " for %s method %s"
+                                                 type (car info))
+                                       ""))
+                 class-alist nil nil nil 'idlwave-class-history))))))
+      (when (and class (not (eq t class)))
+       ;; We have a real class here
+       (when (and store arrow)
+         (put-text-property apos (+ apos 2) 'idlwave-class class)
+         (put-text-property apos (+ apos 2) 'face idlwave-class-arrow-face))
+       (setf (nth 2 info) class))
+      ;; Return the class
+      class)
+     ;; Default as fallback
+     (t class))))
+
+(defvar type-selector)
+(defvar class-selector)
+(defvar method-selector)
+(defun idlwave-selector (a)
+  (and (eq (nth 1 a) type-selector)
+       (or (and (nth 2 a) (eq class-selector t))
+          (eq (nth 2 a) class-selector))))
+
+(defun idlwave-where ()
+  "Find out where we are.
+The return value is a list with the following stuff:
+(PRO-LIST FUNC-LIST COMPLETE-WHAT CW-LIST LAST-CHAR)
+
+PRO-LIST       (PRO POINT CLASS ARROW)
+FUNC-LIST      (FUNC POINT CLASS ARROW)
+COMPLETE-WHAT  a symbol indicating what kind of completion makes sense here
+CW-LIST        Like PRO-LIST, for what can be copmpleted here.
+LAST-CHAR      last relevant character before point (non-white non-comment,
+               not part of current identifier or leading slash).
+
+In the lists, we have these meanings:
+PRO:    Procedure name
+FUNC:   Function name
+POINT:  Where is this
+CLASS:  What class has the routine (nil=no, t=is method, but class unknown)
+ARROW:  Where is the arrow?"
+  (idlwave-routines)
+  (let* ((bos (save-excursion (idlwave-beginning-of-statement) (point)))
+        (func-entry (idlwave-what-function bos))
+         (func (car func-entry))
+         (func-class (nth 1 func-entry))
+         (func-arrow (nth 2 func-entry))
+        (func-point (or (nth 3 func-entry) 0))
+        (func-level (or (nth 4 func-entry) 0))
+        (pro-entry (idlwave-what-procedure bos))
+        (pro (car pro-entry))
+         (pro-class (nth 1 pro-entry))
+         (pro-arrow (nth 2 pro-entry))
+        (pro-point (or (nth 3 pro-entry) 0))
+        (last-char (idlwave-last-valid-char))
+         (case-fold-search t)
+        cw cw-mod cw-arrow cw-class cw-point)
+    (if (< func-point pro-point) (setq func nil))
+    (cond
+     ((string-match 
+       "\\`[ \t]*\\([a-zA-Z][a-zA-Z0-9$_]*\\)?\\'" 
+       (buffer-substring (if (> pro-point 0) pro-point bos) (point)))
+      (setq cw 'procedure cw-class pro-class cw-point pro-point
+           cw-arrow pro-arrow))
+     ((string-match "\\`[ \t]*\\(pro\\|function\\)\\>"
+                   (buffer-substring bos (point)))
+      nil)
+     ((string-match "OBJ_NEW([ \t]*'\\([a-zA-Z][a-zA-Z0-9$_]*\\)?\\'"
+                   (buffer-substring bos (point)))
+      (setq cw 'class))                    
+     ((and func 
+          (> func-point pro-point)
+          (= func-level 1)
+          (memq last-char '(?\( ?,)))
+      (setq cw 'function-keyword cw-mod func cw-point func-point
+           cw-class func-class cw-arrow func-arrow))
+     ((and pro (eq last-char ?,))
+      (setq cw 'procedure-keyword cw-mod pro cw-point pro-point
+           cw-class pro-class cw-arrow pro-arrow))
+;     ((member last-char '(?\' ?\) ?\] ?!))
+;      ;; after these chars, a function makes no sense
+;      ;; FIXME: I am sure there can be more in this list
+;      ;; FIXME: Do we want to do this at all?
+;      nil)
+     ;; Everywhere else we try a function.
+     (t
+      (setq cw 'function)
+      (save-excursion
+       (if (re-search-backward "->[ \t]*\\(\\([$a-zA-Z0-9_]+\\)::\\)?[$a-zA-Z0-9_]*\\=" bos t)
+           (setq cw-arrow (match-beginning 0)
+                 cw-class (if (match-end 2)
+                               (idlwave-sintern-class (match-string 2))
+                              t))))))
+    (list (list pro pro-point pro-class pro-arrow)
+          (list func func-point func-class func-arrow)
+          cw
+         (list cw-mod cw-point cw-class cw-arrow)
+         last-char)))
+
+(defun idlwave-this-word (&optional class)
+  ;; Grab the word around point.  CLASS is for the `skip-chars=...' functions
+  (setq class (or class "a-zA-Z0-9$_"))
+  (save-excursion
+    (buffer-substring-no-properties
+     (progn (skip-chars-backward class) (point))
+     (progn (skip-chars-forward  class) (point)))))
+
+(defvar idlwave-find-symbol-syntax-table)
+(defun idlwave-what-function (&optional bound)
+  ;; Find out if point is within the argument list of a function.
+  ;; The return value is ("function-name" (point) level).
+  ;; Level is 1 on the to level parenthesis, higher further down.
+
+  ;; If the optional BOUND is an integer, bound backwards directed
+  ;;    searches to this point.
+
+  (catch 'exit
+    (let (pos 
+         func-point
+         (old-syntax (syntax-table))
+         (cnt 0)
+         func arrow-start class)
+      (unwind-protect
+         (save-restriction
+           (save-excursion
+             (set-syntax-table idlwave-find-symbol-syntax-table)
+             (narrow-to-region (max 1 (or bound 0)) (point-max))
+             ;; move back out of the current parenthesis
+             (while (condition-case nil
+                        (progn (up-list -1) t)
+                      (error nil))
+               (setq pos (point))
+               (incf cnt)
+               (when (and (= (following-char) ?\()
+                          (re-search-backward 
+                           "\\(::\\|\\<\\)\\([a-zA-Z][a-zA-Z0-9$_]*\\)\\="
+                                              bound t))
+                 (setq func (match-string 2)
+                       func-point (goto-char (match-beginning 2))
+                       pos func-point)
+                 (if (re-search-backward 
+                      "->[ \t]*\\(\\([a-zA-Z][a-zA-Z0-9$_]*\\)::\\)?\\=" bound t)
+                     (setq arrow-start (match-beginning 0)
+                           class (or (match-string 2) t)))
+                 (throw 
+                  'exit 
+                  (list
+                   (idlwave-sintern-routine-or-method func class)
+                   (idlwave-sintern-class class)
+                   arrow-start func-point cnt)))
+               (goto-char pos))
+             (throw 'exit nil)))
+       (set-syntax-table old-syntax)))))
+
+(defun idlwave-what-procedure (&optional bound)
+  ;; Find out if point is within the argument list of a procedure.
+  ;; The return value is ("procedure-name" class arrow-pos (point)).
+
+  ;; If the optional BOUND is an integer, bound backwards directed
+  ;;    searches to this point.
+  (let ((pos (point)) pro-point
+       pro class arrow-start string)
+    (save-excursion 
+      (idlwave-beginning-of-statement)
+      (setq string (buffer-substring (point) pos))
+      (if (string-match 
+          "\\`[ \t]*\\([a-zA-Z][a-zA-Z0-9$_]*\\)[ \t]*\\(,\\|\\'\\)" string)
+         (setq pro (match-string 1 string)
+               pro-point (+ (point) (match-beginning 1)))
+       (if (and (idlwave-skip-object)
+                (setq string (buffer-substring (point) pos))
+                (string-match 
+                 "\\`[ \t]*\\(->\\)[ \t]*\\(\\([a-zA-Z][a-zA-Z0-9$_]*\\)::\\)?\\([a-zA-Z][a-zA-Z0-9$_]*\\)?[ \t]*\\(,\\|\\'\\)" string))
+           (setq pro (if (match-beginning 4)
+                         (match-string 4 string))
+                 pro-point (if (match-beginning 4)
+                               (+ (point) (match-beginning 4))
+                       pos)
+                 arrow-start (+ (point) (match-beginning 1))
+                 class (or (match-string 3 string) t)))))
+    (list (idlwave-sintern-routine-or-method pro class)
+         (idlwave-sintern-class class)
+         arrow-start
+         pro-point)))
+
+(defun idlwave-skip-object ()
+  ;; If there is an object at point, move over it and return t.
+  (let ((pos (point)))
+    (if (catch 'exit
+         (save-excursion
+           (skip-chars-forward "       ")  ; white space
+           (skip-chars-forward "*")        ; de-reference
+           (cond
+            ((looking-at idlwave-identifier)
+             (goto-char (match-end 0)))
+            ((eq (following-char) ?\()
+             nil)
+            (t (throw 'exit nil)))
+           (catch 'endwhile
+             (while t
+               (cond ((eq (following-char) ?.)
+                      (forward-char 1)
+                      (if (not (looking-at idlwave-identifier))
+                          (throw 'exit nil))
+                      (goto-char (match-end 0)))
+                     ((memq (following-char) '(?\( ?\[))
+                      (condition-case nil
+                          (forward-list 1)
+                        (error (throw 'exit nil))))
+                     (t (throw 'endwhile t)))))
+           (if (looking-at "[ \t]*->")
+               (throw 'exit (setq pos (match-beginning 0)))
+             (throw 'exit nil))))
+       (goto-char pos)
+      nil)))
+  
+
+(defun idlwave-last-valid-char ()
+  "Return the last character before point which is not white or a comment
+and also not part of the current identifier.  Since we do this in
+order to identify places where keywords are, we consider the initial
+`/' of a keyword as part of the identifier.
+This function is not general, can only be used for completion stuff."
+  (catch 'exit
+    (save-excursion
+      ;; skip the current identifier
+      (skip-chars-backward "a-zA-Z0-9_$")
+      ;; also skip a leading slash which might be belong to the keyword
+      (if (eq (preceding-char) ?/)
+         (backward-char 1))
+      ;; FIXME: does not check if this is a valid identifier
+      (while t
+       (skip-chars-backward " \t")
+       (cond
+        ((memq (preceding-char) '(?\; ?\$)) (throw 'exit nil))
+        ((eq (preceding-char) ?\n)
+         (beginning-of-line 0)
+         (if (looking-at "\\([^;]\\)*\\$[ \t]*\\(;.*\\)?\n")
+             ;; continuation line
+             (goto-char (match-end 1))
+           (throw 'exit nil)))
+        (t (throw 'exit (preceding-char))))))))
+
+(defvar idlwave-complete-after-success-form nil
+  "A form to evaluate after successful completion.")
+(defvar idlwave-complete-after-success-form-force nil
+  "A form to evaluate after completion selection in *Completions* buffer.")
+(defconst idlwave-completion-mark (make-marker)
+  "A mark pointing to the beginning of the completion string.")
+
+(defun idlwave-complete-in-buffer (type stype list selector prompt isa
+                                       &optional prepare-display-function)
+  "Perform TYPE completion of word before point against LIST.
+SELECTOR is the PREDICATE argument for the completion function.
+Show PROMPT in echo area.  TYPE is one of 'function, 'procedure or 'keyword."
+  (let* ((completion-ignore-case t)
+        beg (end (point)) slash part spart completion all-completions
+        dpart dcompletion)
+
+    (unless list
+      (error (concat prompt ": No completions available")))
+
+    ;; What is already in the buffer?
+    (save-excursion
+      (skip-chars-backward "a-zA-Z0-9_$")
+      (setq slash (eq (preceding-char) ?/)
+           beg (point)
+           idlwave-complete-after-success-form
+           (list 'idlwave-after-successful-completion
+                 (list 'quote type) slash beg)
+           idlwave-complete-after-success-form-force
+           (list 'idlwave-after-successful-completion
+                 (list 'quote type) slash (list 'quote 'force))))
+
+    ;; Try a completion
+    (setq part (buffer-substring beg end)
+         dpart (downcase part)
+         spart (idlwave-sintern stype part)
+         completion (try-completion part list selector)
+         dcompletion (if (stringp completion) (downcase completion)))
+    (cond
+     ((null completion)
+      ;; nothing available.
+      (error "Can't find %s completion for \"%s\"" isa part))
+     ((and (not (equal dpart dcompletion))
+          (not (eq t completion)))
+      ;; We can add something
+      (delete-region beg end)
+      (if (and (string= part dpart)
+              (or (not (string= part ""))
+                  idlwave-complete-empty-string-as-lower-case)
+              (not idlwave-completion-force-default-case))
+         (insert dcompletion)
+       (insert completion))
+      (if (eq t (try-completion completion list selector))
+         ;; Now this is a unique match
+         (idlwave-after-successful-completion type slash beg))
+      t)
+     ((or (eq completion t)
+         (and (equal dpart dcompletion)
+              (= 1 (length (setq all-completions
+                                 (idlwave-uniquify
+                                  (all-completions part list selector)))))))
+      ;; This is already complete
+      (idlwave-after-successful-completion type slash beg)
+      (message "%s is already the complete %s" part isa)
+      nil)
+     (t
+      ;; We cannot add something - offer a list.
+      (message "Making completion list...")
+      (let* ((list all-completions)
+            (complete (memq spart all-completions))
+            (completion-highlight-first-word-only t) ; XEmacs
+            (completion-fixup-function               ; Emacs
+             (lambda () (and (eq (preceding-char) ?>)
+                             (re-search-backward " <" beg t)))))
+       (setq list (sort list (lambda (a b)
+                               (string< (downcase a) (downcase b)))))
+       (if prepare-display-function
+           (setq list (funcall prepare-display-function list)))
+       (if (and (string= part dpart)
+                (or (not (string= part ""))
+                    idlwave-complete-empty-string-as-lower-case)
+                (not idlwave-completion-force-default-case))
+           (setq list (mapcar (lambda (x)
+                                (if (listp x) 
+                                    (setcar x (downcase (car x)))
+                                  (setq x (downcase x)))
+                                x)
+                              list)))
+       (idlwave-display-completion-list list prompt beg complete))
+      t))))
+
+(defun idlwave-complete-class ()
+  "Complete a class at point."
+  (interactive)
+  ;; Call `idlwave-routines' to make sure the class list will be available
+  (idlwave-routines)
+  ;; Now do the completion
+  (idlwave-complete-in-buffer 'class 'class idlwave-class-alist nil 
+                             "Select a class" "class"))
+
+
+(defun idlwave-attach-classes (list is-kwd show-classes)
+  ;; attach the proper class list to a LIST of completion items.
+  ;; IS-KWD, when non-nil, shows its keywords - otherwise its methods
+  ;; SHOW-CLASSES is the value of `idlwave-completion-show-classes'.
+  (catch 'exit
+    (if (or (null show-classes)         ; don't wnat to see classes
+           (null class-selector)       ; not a method call
+           (stringp class-selector))   ; the class is already known
+       ;; In these cases, we do not have to do anything
+       (throw 'exit list))
+    
+    ;; The property and dots stuff currently only make sense with XEmacs
+    ;; because Emacs drops text properties when filling the *Completions*
+    ;; buffer.
+    (let* ((do-prop (and (featurep 'xemacs) (>= show-classes 0)))
+          (do-buf (not (= show-classes 0)))
+          (do-dots (featurep 'xemacs))
+          (max (abs show-classes))
+          (lmax (if do-dots (apply 'max (mapcar 'length list))))
+         classes nclasses class-info space)
+      (mapcar 
+       (lambda (x)
+        ;; get the classes
+        (setq classes
+              (if is-kwd 
+                  (idlwave-all-method-keyword-classes
+                   method-selector x type-selector)
+                (idlwave-all-method-classes x type-selector)))
+        (setq nclasses (length classes))
+        ;; Make the separator between item and class-info
+        (if do-dots
+            (setq space (concat " " (make-string (- lmax (length x)) ?.)))
+          (setq space " "))
+        (if  do-buf
+            ;; We do want info in the buffer
+            (if (<= nclasses max)
+                (setq class-info (concat
+                                  space
+                                  "<" (mapconcat 'identity classes ",") ">"))
+              (setq class-info (format "%s<%d classes>" space nclasses)))
+          (setq class-info nil))
+        (when do-prop
+          ;; We do want properties
+          (setq x (copy-sequence x))
+          (put-text-property 0 (length x)
+                             'help-echo (mapconcat 'identity classes " ")
+                             x))
+        (if class-info
+            (list x class-info)
+          x))
+       list))))
+
+(defun idlwave-attach-method-classes (list)
+  ;; Call idlwave-attach-classes with method parameters
+  (idlwave-attach-classes list nil idlwave-completion-show-classes))
+(defun idlwave-attach-keyword-classes (list)
+  ;; Call idlwave-attach-classes with keyword parameters
+  (idlwave-attach-classes list t idlwave-completion-show-classes))
+
+;;----------------------------------------------------------------------
+;;----------------------------------------------------------------------
+;;----------------------------------------------------------------------
+;;----------------------------------------------------------------------
+;;----------------------------------------------------------------------
+
+(defun idlwave-scroll-completions (&optional message)
+  "Scroll the completion window on this frame."
+  (let ((cwin (get-buffer-window "*Completions*" 'visible))
+       (win (selected-window)))
+    (unwind-protect
+       (progn
+         (select-window cwin)
+         (condition-case nil
+             (scroll-up)
+           (error (if (and (listp last-command)
+                           (nth 2 last-command))
+                      (progn
+                        (select-window win)
+                        (eval idlwave-complete-after-success-form))
+                    (set-window-start cwin (point-min)))))
+         (and message (message message)))
+      (select-window win))))
+
+(defun idlwave-display-completion-list (list &optional message beg complete)
+  "Display the completions in LIST in the completions buffer and echo MESSAGE."
+  (unless (and (get-buffer-window "*Completions*")
+              (idlwave-local-value 'idlwave-completion-p "*Completions*"))
+    (move-marker idlwave-completion-mark beg)
+    (setq idlwave-before-completion-wconf (current-window-configuration)))
+
+  (if (featurep 'xemacs)
+      (idlwave-display-completion-list-xemacs list)
+    (idlwave-display-completion-list-emacs list))
+
+  ;; Store a special value in `this-command'.  When `idlwave-complete'
+  ;; finds this in `last-command', it will scroll the *Completions* buffer.
+  (setq this-command (list 'idlwave-display-completion-list message complete))
+
+  ;; Mark the completions buffer as created by cib
+  (idlwave-set-local 'idlwave-completion-p t "*Completions*")
+
+  ;; Fontify the classes
+  (if (and idlwave-completion-fontify-classes
+           (consp (car list)))
+      (idlwave-completion-fontify-classes))
+
+  ;; Display the message
+  (message (or message "Making completion list...done")))
+
+(defun idlwave-choose (function &rest args)
+  "Call FUNCTION as a completion chooser and pass ARGS to it."
+  (let ((completion-ignore-case t))        ; install correct value
+    (apply function args))
+  (eval idlwave-complete-after-success-form-force))
+
+(defun idlwave-restore-wconf-after-completion ()
+  "Restore the old (before completion) window configuration."
+  (and idlwave-completion-restore-window-configuration
+       idlwave-before-completion-wconf
+       (set-window-configuration idlwave-before-completion-wconf)))
+
+(defun idlwave-set-local (var value &optional buffer)
+  "Set the buffer-local value of VAR in BUFFER to VALUE."
+  (save-excursion
+    (set-buffer (or buffer (current-buffer)))
+    (set (make-local-variable var) value)))
+
+(defun idlwave-local-value (var &optional buffer)
+  "Return the value of VAR in BUFFER, but only if VAR is local to BUFFER."
+  (save-excursion
+    (set-buffer (or buffer (current-buffer)))
+    (and (local-variable-p var (current-buffer))
+        (symbol-value var))))
+
+;; In XEmacs, we can use :activate-callback directly
+
+(defun idlwave-display-completion-list-xemacs (list)
+  (with-output-to-temp-buffer "*Completions*"
+    (display-completion-list list :activate-callback 
+                            'idlwave-default-choose-completion)))
+
+(defun idlwave-default-choose-completion (&rest args)
+  "Execute `default-choose-completion' and then restore the win-conf."
+  (apply 'idlwave-choose 'default-choose-completion args))
+
+;; In Emacs we have to replace the keymap in the *Completions* buffer
+;; in order to install our wrappers.
+
+(defvar idlwave-completion-map nil
+  "Keymap for completion-list-mode with idlwave-complete.")
+
+(defun idlwave-display-completion-list-emacs (list)
+  "Display completion list and install the choose wrappers."
+  (with-output-to-temp-buffer "*Completions*"
+    (display-completion-list list))
+  (save-excursion
+    (set-buffer "*Completions*")
+    (use-local-map
+     (or idlwave-completion-map
+        (setq idlwave-completion-map
+              (idlwave-make-modified-completion-map (current-local-map)))))))
+  
+(defun idlwave-make-modified-completion-map (old-map)
+  "Replace `choose-completion' and `mouse-choose-completion' in OLD-MAP."
+  (let ((new-map (copy-keymap old-map)))
+    (substitute-key-definition 
+     'choose-completion 'idlwave-choose-completion new-map)
+    (substitute-key-definition
+     'mouse-choose-completion 'idlwave-mouse-choose-completion new-map)
+    new-map))
+
+(defun idlwave-choose-completion (&rest args)
+  "Choose the completion that point is in or next to."
+  (interactive)
+  (apply 'idlwave-choose 'choose-completion args))
+
+(defun idlwave-mouse-choose-completion (&rest args)
+  "Click on an alternative in the `*Completions*' buffer to choose it."
+  (interactive "e")
+  (apply 'idlwave-choose 'mouse-choose-completion args))
+
+;;----------------------------------------------------------------------
+;;----------------------------------------------------------------------
+
+(defun idlwave-completion-fontify-classes ()
+  "Goto the *Completions* buffer and fontify the class info."
+  (when (featurep 'font-lock)
+    (save-excursion
+      (set-buffer "*Completions*")
+      (save-excursion
+       (goto-char (point-min))
+       (while (re-search-forward "\\.*<[^>]+>" nil t)
+         (put-text-property (match-beginning 0) (match-end 0)
+                            'face 'font-lock-string-face))))))
+
+(defun idlwave-uniquify (list)
+  (let (nlist)
+    (loop for x in list do
+      (add-to-list 'nlist x))
+    nlist))
+
+(defun idlwave-after-successful-completion (type slash &optional verify)
+  "Add `=' or `(' after successful completion of keyword and function.
+Restore the pre-completion window configuration if possible."
+  (cond
+   ((eq type 'procedure)
+    nil)
+   ((eq type 'function)
+    (cond
+     ((equal idlwave-function-completion-adds-paren nil) nil)
+     ((or (equal idlwave-function-completion-adds-paren t)
+         (equal idlwave-function-completion-adds-paren 1))
+      (insert "("))
+     ((equal idlwave-function-completion-adds-paren 2)
+      (insert "()")
+      (backward-char 1))
+     (t nil)))
+   ((eq type 'keyword)
+    (if (and idlwave-keyword-completion-adds-equal
+            (not slash))
+       (progn (insert "=") t)
+      nil)))
+
+  ;; Restore the pre-completion window configuration if this is safe.
+  
+  (if (or (eq verify 'force)                                    ; force 
+         (and 
+          (get-buffer-window "*Completions*")                  ; visible
+          (idlwave-local-value 'idlwave-completion-p 
+                               "*Completions*")                ; cib-buffer
+          (eq (marker-buffer idlwave-completion-mark)
+              (current-buffer))                                ; buffer OK
+          (equal (marker-position idlwave-completion-mark)
+                 verify)))                                     ; pos OK
+      (idlwave-restore-wconf-after-completion))
+  (move-marker idlwave-completion-mark nil)
+  (setq idlwave-before-completion-wconf nil))
+
+(defun idlwave-routine-info-from-idlhelp (&optional arg)
+  "Make IDLHELP display the online documentation about the routine at point.
+Sends the command `? MODULE' to the IDLWAVE-Shell.  Shell must be running,
+it does not autostart for this task."
+  (interactive "P")
+  (idlwave-routine-info arg 'external))
+
+(defun idlwave-routine-info (&optional arg external)
+  "Display a routines calling sequence and list of keywords.
+When point is on the name a function or procedure, or in the argument
+list of a function or procedure, this command displays a help buffer
+with the information.  When called with prefix arg, enforce class
+query.
+
+When point is on an object operator `->', display the class stored in
+this arrow, if any (see `idlwave-store-inquired-class').  With a
+prefix arg, the class property is cleared out."
+
+  (interactive "P")
+  (idlwave-routines)
+  (if (string-match "->" (buffer-substring
+                         (max (point-min) (1- (point)))
+                         (min (+ 2 (point)) (point-max))))
+      ;; Cursor is on an arrow
+      (if (get-text-property (point) 'idlwave-class)
+         ;; arrow has class property
+         (if arg
+             ;; Remove property
+             (save-excursion
+               (backward-char 1)
+               (when (looking-at ".?\\(->\\)")
+                 (remove-text-properties (match-beginning 1) (match-end 1)
+                                         '(idlwave-class nil face nil))
+                 (message "Class property removed from arrow")))
+           ;; Echo class property
+           (message "Arrow has text property identifying object to be class %s"
+                    (get-text-property (point) 'idlwave-class)))
+       ;; No property found
+       (message "Arrow has no class text property"))
+
+    ;; Not on an arrow...
+    (let* ((idlwave-query-class nil)
+          (idlwave-force-class-query (equal arg '(4)))
+          (module (idlwave-what-module)))
+      (cond ((car module)
+            (if external
+                (apply 'idlwave-search-online-help module)
+              (apply 'idlwave-display-calling-sequence module)))
+           (t
+            (error "Don't know which calling sequence to show."))))))
+
+(defun idlwave-search-online-help (name &optional type class olh)
+  "Tell IDL to lookup CLASS::NAME with type TYPE in the online help.
+If TYPE and CLASS are both nil, just look up NAME in the default help file."
+  ;; If only the IDLHELP application was better designed, so that
+  ;; we could make it open the right thing right away.  As things are,
+  ;; we need to pipe the stuff through the help search engine, and we
+  ;; cannot enter a space.
+  (let* (extra book full string cmd)
+
+    ;; Try to find a clue for the right help book
+    (if (and type (not olh))
+       (setq olh (or (nth 6 (idlwave-rinfo-assq
+                             name type class idlwave-builtin-routines))
+                     (nth 6 (idlwave-rinfo-assq 
+                             name type class idlwave-routines)))))
+
+    ;; Sometimes the book is given as a symbol - make it a string
+    (if (and olh (symbolp olh)) (setq olh (symbol-name olh)))
+    (setq book (or olh "idl"))   ; We need a default
+    ;; Add the FULL_PATH keyword if appropriate
+    (if (and (file-name-absolute-p book)
+            (file-exists-p book))
+       (setq full ",/FULL_PATH")
+      (setq full ""))
+
+    ;; We would like to add "Method" or so, but stupid IDL online help
+    ;; command treats a space as a separator and interprets the next thing as
+    ;; the book name.
+    ;; (setq extra (cond ((eq type 'kwd) " keyword")
+    ;;               (class          " method")
+    ;;               ((eq type 'pro) " procedure")
+    ;;               ((eq type 'fun) " function")
+    ;;               (t "")))
+    (setq extra "")
+
+    ;; Methods are subitems of classes, the separator is a single `:'
+    (if (and name class (not (eq type 'kwd)))
+       (setq name (concat class ":" name)))
+    ;; FIXME:  We used to use book, but in idl5.3, all help is in idl.hlp
+    (setq string (concat name extra)
+         cmd (format "ONLINE_HELP,'%s',BOOK='%s'%s" string "idl" full))
+;        cmd (format "ONLINE_HELP,'%s',BOOK='%s'%s" string book full))
+    (message "Sending to IDL: %s" cmd) (sit-for 2)
+    (idlwave-shell-send-command cmd)))
+
+(defun idlwave-resolve (&optional arg)
+  "Call RESOLVE on the module name at point.
+Like `idlwave-routine-info', this looks for a routine call at point.
+After confirmation in the minibuffer, it will use the shell to issue
+a RESOLVE call for this routine, to attempt to make it defined and its
+routine info available for IDLWAVE.  If the routine is a method call,
+both `class__method' and `class__define' will be tried.
+With ARG, enforce query for the class of object methods."
+  (interactive "P")
+  (let* ((idlwave-query-class nil)
+        (idlwave-force-class-query (equal arg '(4)))
+        (module (idlwave-what-module))
+        (name (idlwave-make-full-name (nth 2 module) (car module)))
+        (type (if (eq (nth 1 module) 'pro) "pro" "function"))
+        (resolve (read-string "Resolve: " (format "%s %s" type name)))
+        (kwd "")
+        class)
+    (if (string-match "\\(pro\\|function\\)[ \t]+\\(\\(.*\\)::\\)?\\(.*\\)"
+                     resolve)
+       (setq type (match-string 1 resolve)
+             class (if (match-beginning 2) 
+                       (match-string 3 resolve)
+                     nil)
+             name (match-string 4 resolve)))
+    (if (string= (downcase type) "function")
+       (setq kwd ",/is_function"))
+
+    (cond
+     ((null class)
+      (idlwave-shell-send-command 
+       (format "resolve_routine,'%s'%s" (downcase name) kwd)
+       'idlwave-update-routine-info
+       nil t))
+     (t
+      (idlwave-shell-send-command 
+       (format "resolve_routine,'%s__define'%s" (downcase class) kwd)
+       (list 'idlwave-shell-send-command 
+            (format "resolve_routine,'%s__%s'%s" 
+                    (downcase class) (downcase name) kwd)
+            '(idlwave-update-routine-info)
+            nil t))))))
+
+(defun idlwave-find-module (&optional arg)
+  "Find the source code of an IDL module.
+Works for modules for which IDLWAVE has routine info available.
+The function offers as default the module name `idlwave-routine-info' would
+use.  With ARG force class query for object methods."
+  (interactive "P")
+  (let* ((idlwave-query-class nil)
+        (idlwave-force-class-query (equal arg '(4)))
+        (module (idlwave-what-module))
+        (default (concat (idlwave-make-full-name (nth 2 module) (car module))
+                         (if (eq (nth 1 module) 'pro) "<p>" "<f>")))
+        (list 
+         (delq nil
+               (mapcar (lambda (x) 
+                         (if (eq 'system (car-safe (nth 3 x)))
+                             ;; Take out system routines with no source.
+                             nil
+                           (cons
+                            (concat (idlwave-make-full-name (nth 2 x) (car x))
+                                    (if (eq (nth 1 x) 'pro) "<p>" "<f>"))
+                            (cdr x))))
+                       (idlwave-routines))))
+        (name (idlwave-completing-read
+               (format "Module (Default %s): " 
+                       (if default default "none"))
+               list))
+        type class)
+    (if (string-match "\\`\\s-*\\'" name)
+       ;; Nothing, use the default.
+       (setq name default))
+    (if (string-match "<[fp]>" name)
+       (setq type (substring name -2 -1)
+             name (substring name 0 -3)))
+    (if (string-match "\\(.*\\)::\\(.*\\)" name)
+       (setq class (match-string 1 name)
+             name (match-string 2 name)))
+    (setq name (idlwave-sintern-routine-or-method name class)
+         class (idlwave-sintern-class class)
+         type (cond ((equal type "f") 'fun)
+                    ((equal type "p") 'pro)
+                    (t t)))
+    (idlwave-do-find-module name type class)))
+
+(defun idlwave-do-find-module (name type class)
+  (let ((name1 (idlwave-make-full-name class name))
+       source buf1 entry
+       (buf (current-buffer))
+       (pos (point)))
+    (setq entry (idlwave-rinfo-assq name type class (idlwave-routines))
+         source (nth 3 entry))
+    (cond
+     ((or (null name) (equal name ""))
+      (error "Abort"))
+     ((null entry)
+      (error "Nothing known about a module %s" name1))
+     ((eq (car source) 'system)
+      (error "Source code for system routine %s is not available." 
+            name1))
+     ((equal (cdr source) "")
+      (error "Source code for routine %s is not available."
+            name1))
+     ((memq (car source) '(buffer lib compiled))
+      (setq buf1 
+           (if (eq (car source) 'lib)
+               (idlwave-find-lib-file-noselet 
+                (or (cdr source)
+                    (format "%s.pro" (downcase name))))
+             (idlwave-find-file-noselect (cdr source))))
+      (pop-to-buffer buf1)
+      (goto-char 1)
+      (let ((case-fold-search t))
+       (if (re-search-forward
+            (concat "^[ \t]*\\<"
+                    (cond ((equal type "f") "function")
+                          ((equal type "p") "pro")
+                          (t "\\(pro\\|function\\)"))
+                    "\\>[ \t]+" 
+                    (regexp-quote (downcase name1))
+                    "[^a-zA-Z0-9_$]")
+            nil t)
+           (goto-char (match-beginning 0))
+         (pop-to-buffer buf)
+         (goto-char pos)
+         (error "Could not find routine %s" name1)))))))
+
+(defun idlwave-what-module ()
+  "Return a default module for stuff near point.
+Used by `idlwave-routine-info' and `idlwave-find-module'."
+  (idlwave-routines)
+  (let* ((where (idlwave-where))
+        (cw (nth 2 where))
+        (pro (car (nth 0 where)))
+        (func (car (nth 1 where)))
+        (this-word (idlwave-this-word "a-zA-Z0-9$_"))
+        (next-char (save-excursion (skip-chars-forward "a-zA-Z0-9$_")
+                                   (following-char)))
+        )
+    (cond
+     ((and (eq cw 'procedure)
+          (not (equal this-word "")))
+      (setq this-word (idlwave-sintern-routine-or-method 
+                      this-word (nth 2 (nth 3 where))))
+      (list this-word 'pro
+           (idlwave-determine-class 
+            (cons this-word (cdr (nth 3 where)))
+            'pro)))
+     ((and (eq cw 'function) 
+          (not (equal this-word ""))
+          (eq next-char ?\())              ; exclude arrays, vars.
+      (setq this-word (idlwave-sintern-routine-or-method 
+                      this-word (nth 2 (nth 3 where))))
+      (list this-word 'fun
+           (idlwave-determine-class
+            (cons this-word (cdr (nth 3 where)))
+            'fun)))
+     (func
+      (list func 'fun (idlwave-determine-class (nth 1 where) 'fun)))
+     (pro
+      (list pro 'pro (idlwave-determine-class (nth 0 where) 'pro)))
+     (t nil))))
+
+(defun idlwave-fix-keywords (name type class keywords)
+  ;; This fixes the list of keywords.
+  (let ((case-fold-search t)
+       name1 type1)
+
+    ;; If this is the OBJ_NEW function, try to figure out the class and use
+    ;; the keywords from the corresponding INIT method.
+    (if (and (equal name "OBJ_NEW")
+            (eq major-mode 'idlwave-mode))
+       (let* ((bos (save-excursion (idlwave-beginning-of-statement) (point)))
+              (string (buffer-substring bos (point)))
+              (case-fold-search t)
+              class)
+         (and (string-match "obj_new([^'\"]*['\"]\\([a-zA-Z0-9_]+\\)"
+                            string)
+              (setq class (idlwave-sintern-class (match-string 1 string)))
+              (setq keywords 
+                    (append keywords 
+                            (nth 5 (idlwave-rinfo-assq
+                                    (idlwave-sintern-method "INIT")
+                                    'fun
+                                    class
+                                    (idlwave-routines))))))))
+
+    ;; If the class is `t', combine all keywords of all methods NAME
+    (when (eq class t)
+      (loop for x in (idlwave-routines) do
+       (and (nth 2 x)                         ; non-nil class
+            (or (and (eq (nth 1 x) type)      ; default type
+                     (eq (car x) name))       ; default name
+                (and (eq (nth 1 x) type1)     ; backup type
+                     (eq (car x) name1)))     ; backup name
+            (mapcar (lambda (k) (add-to-list 'keywords k))
+                    (nth 5 x))))
+      (setq keywords (idlwave-uniquify keywords)))
+    ;; Return the final list
+    keywords))
+
+(defvar idlwave-rinfo-map (make-sparse-keymap))
+(define-key idlwave-rinfo-map 
+  (if (featurep 'xemacs) [button2] [mouse-2])
+  'idlwave-mouse-active-rinfo)
+(define-key idlwave-rinfo-map 
+  (if (featurep 'xemacs) [button3] [mouse-3])
+  'idlwave-mouse-active-rinfo-right)
+(defvar idlwave-popup-source)
+
+(defun idlwave-display-calling-sequence (name type class)
+  ;; Display the calling sequence of module NAME, type TYPE in class CLASS.
+  (let* ((entry (idlwave-rinfo-assq
+                name type class (idlwave-routines)))
+        (name (or (car entry) name))
+        (class (or (nth 2 entry) class))
+        (source (nth 3 entry))
+        ;;(system (eq (car source) 'system))
+        (calling-seq (nth 4 entry))
+        (keywords (nth 5 entry))
+        (olh (nth 6 entry))
+        (help-echo3  "                                     Button3: IDL Online Help")
+        (help-echo23 "Button2: Pop to source and back.     Button3: IDL Online Help")
+        (col 0)
+        (data (list name type class (current-buffer) olh))
+        (km-prop (if (featurep 'xemacs) 'keymap 'local-map))
+        beg props win)
+    (setq keywords (idlwave-fix-keywords name type class keywords))
+    (cond
+     ((null entry)
+      (error "No %s %s known" type name))
+     ((or (null name) (equal name ""))
+      (error "No function or procedure call at point."))
+     ((null calling-seq)
+      (error "Calling sequence of %s %s is not available" type name))
+     (t
+      (save-excursion
+       (set-buffer (get-buffer-create "*Help*"))
+       (setq buffer-read-only nil)
+       (erase-buffer)
+       (set (make-local-variable 'idlwave-popup-source) nil)
+       (setq props (list 'mouse-face 'highlight
+                         km-prop idlwave-rinfo-map
+                         'help-echo help-echo23
+                         'data (cons 'usage data)))
+       (insert "Usage:    ")
+       (setq beg (point))
+       (insert (if class
+                   (format calling-seq class name)
+                 (format calling-seq name))
+               "\n")
+       (add-text-properties beg (point) props)
+
+       (insert "Keywords:")
+       (if (null keywords)
+           (insert " No keywords accepted.")
+         (setq col 9)
+         (mapcar
+          (lambda (x)
+            (if (>= (+ col 1 (length (car x))) 
+                    (window-width))
+                (progn
+                  (insert "\n         ")
+                  (setq col 9)))
+            (insert " ")
+            (setq beg (point)
+                  props (list 'mouse-face 'highlight
+                              km-prop idlwave-rinfo-map
+                              'data (cons 'keyword data)
+                              'help-echo help-echo3
+                              'keyword (car x)))
+            (insert (car x))
+            (add-text-properties beg (point) props)
+            (setq col (+ col 1 (length (car x)))))
+          keywords))
+       (insert "\n")
+
+       (insert "Origin:   ")
+       (setq beg (point)
+             props (list 'mouse-face 'highlight
+                         km-prop idlwave-rinfo-map
+                         'help-echo help-echo23
+                         'data (cons 'origin data)))
+       (cond
+        ((eq (car source) 'system)
+         (insert "system routine"))
+        ((equal source '(lib))
+         (insert (format "library file %s.pro" (downcase name))))
+        ((eq (car source) 'lib)
+         (insert "library file ")
+         (insert (cdr source)))
+        ((eq (car source) 'buffer)
+         (insert "buffer visiting ")
+         (insert (abbreviate-file-name (cdr source))))
+        ((eq (car source) 'compiled)
+         (insert "compiled from ")
+         (insert (cdr source))))
+       (add-text-properties beg (point) props)
+       (setq buffer-read-only t))
+      (display-buffer "*Help*")
+      (if (and (setq win (get-buffer-window "*Help*"))
+              idlwave-resize-routine-help-window)
+         (progn
+           (let ((ww (selected-window)))
+             (unwind-protect
+                 (progn
+                   (select-window win)
+                   (enlarge-window (- (/ (frame-height) 2) 
+                                      (window-height)))
+                   (shrink-window-if-larger-than-buffer))
+               (select-window ww)))))))))
+
+(defun idlwave-mouse-active-rinfo-right (ev)
+  (interactive "e")
+  (idlwave-mouse-active-rinfo ev 'right))
+
+(defun idlwave-mouse-active-rinfo (ev &optional right)
+  (interactive "e")
+  (mouse-set-point ev)
+  (let (data id name type class buf keyword olh bufwin)
+    (setq data (get-text-property (point) 'data)
+         keyword (get-text-property (point) 'keyword)
+         id (car data)
+         name (nth 1 data)
+         type (nth 2 data)
+         class (nth 3 data)
+         buf (nth 4 data)
+         olh (nth 5 data)
+         bufwin (get-buffer-window buf t))
+    (cond ((or (eq id 'usage) (eq id 'origin))
+          (if right
+              (idlwave-search-online-help name type class)
+            (setq idlwave-popup-source (not idlwave-popup-source))
+            (if idlwave-popup-source
+                (condition-case err
+                    (idlwave-do-find-module name type class)
+                  (error
+                   (setq idlwave-popup-source nil)
+                   (if (window-live-p bufwin) (select-window bufwin))
+                   (error (nth 1 err))))
+              (if bufwin
+                  (select-window bufwin)
+                (pop-to-buffer buf)))))
+         ((eq id 'keyword)
+          (if right
+              (idlwave-search-online-help keyword 'kwd class olh)
+            (error "Button2 not active for keywords"))))))
+
+;; ----------------------------------------------------------------------------
+;;
+;; Additions for use with imenu.el and func-menu.el
+;; (pop-up a list of IDL units in the current file).
+;;
+
+(defun idlwave-prev-index-position ()
+  "Search for the previous procedure or function.
+Return nil if not found.  For use with imenu.el."
+  (save-match-data
+    (cond
+     ((idlwave-find-key "\\<\\(pro\\|function\\)\\>" -1 'nomark))
+     ;;   ((idlwave-find-key idlwave-begin-unit-reg 1 'nomark)
+     (t nil))))
+
+(defun idlwave-unit-name ()
+  "Return the unit name.
+Assumes that point is at the beginning of the unit as found by
+`idlwave-prev-index-position'."
+  (forward-sexp 2)
+  (forward-sexp -1)
+  (let ((begin (point)))
+    (re-search-forward "[a-zA-Z][a-zA-Z0-9$_]+\\(::[a-zA-Z][a-zA-Z0-9$_]+\\)?")
+    (if (fboundp 'buffer-substring-no-properties)
+        (buffer-substring-no-properties begin (point))
+      (buffer-substring begin (point)))))
+
+(defun idlwave-function-menu ()
+  "Use `imenu' or `function-menu' to jump to a procedure or function."
+  (interactive)
+  (if (string-match "XEmacs" emacs-version)
+      (progn
+       (require 'func-menu)
+       (function-menu))
+    (require 'imenu)
+    (imenu (imenu-choose-buffer-index))))
+
+;; Here we kack func-menu.el in order to support this new mode.
+;; The latest versions of func-menu.el already have this stuff in, so
+;; we hack only if it is not already there.
+(when (fboundp 'eval-after-load)
+  (eval-after-load "func-menu"
+    '(progn
+       (or (assq 'idlwave-mode fume-function-name-regexp-alist)
+          (not (boundp 'fume-function-name-regexp-idl))      ; avoid problems
+          (setq fume-function-name-regexp-alist
+                (cons '(idlwave-mode . fume-function-name-regexp-idl)
+                      fume-function-name-regexp-alist)))
+       (or (assq 'idlwave-mode fume-find-function-name-method-alist)
+          (not (fboundp 'fume-find-next-idl-function-name))  ; avoid problems
+          (setq fume-find-function-name-method-alist
+                (cons '(idlwave-mode . fume-find-next-idl-function-name)
+                      fume-find-function-name-method-alist))))))
+
+(defun idlwave-edit-in-idlde ()
+  "Edit the current file in IDL Development environment."
+  (interactive)
+  (start-process "idldeclient" nil
+                idlwave-shell-explicit-file-name "-c" "-e"
+                 (buffer-file-name) "&"))
+                
+(defun idlwave-launch-idlhelp ()
+  "Start the IDLhelp application."
+  (interactive)
+  (start-process "idlhelp" nil idlwave-help-application))
+;; Menus - using easymenu.el
+(defvar idlwave-mode-menu-def
+  `("IDLWAVE"
+    ["PRO/FUNC menu" idlwave-function-menu t]
+    ("Motion"
+     ["Subprogram Start" idlwave-beginning-of-subprogram t]
+     ["Subprogram End" idlwave-end-of-subprogram t]
+     ["Block Start" idlwave-beginning-of-block t]
+     ["Block End" idlwave-end-of-block t]
+     ["Up Block" idlwave-backward-up-block t]
+     ["Down Block" idlwave-down-block t]
+     ["Skip Block Backward" idlwave-backward-block t]
+     ["Skip Block Forward" idlwave-forward-block t])
+    ("Mark"
+     ["Subprogram" idlwave-mark-subprogram t]
+     ["Block" idlwave-mark-block t]
+     ["Header" idlwave-mark-doclib t])
+    ("Format"
+     ["Indent Subprogram" idlwave-indent-subprogram t]
+     ["(Un)Comment Region" idlwave-toggle-comment-region "C-c ;"]
+     ["Continue/Split line" idlwave-split-line t]
+     "--"
+     ["Toggle Auto Fill" idlwave-auto-fill-mode :style toggle
+      :selected (symbol-value idlwave-fill-function)])
+    ("Templates"
+     ["Procedure" idlwave-procedure t]
+     ["Function" idlwave-function t]
+     ["Doc Header" idlwave-doc-header t]
+     ["Log" idlwave-doc-modification t]
+     "--"
+     ["Case" idlwave-case t]
+     ["For" idlwave-for t]
+     ["Repeat" idlwave-repeat t]
+     ["While" idlwave-while t]
+     "--"
+     ["Close Block" idlwave-close-block t])
+    ("Completion / RInfo"
+     ["Complete" idlwave-complete t]
+     ("Complete Special"
+      ["1 Procedure Name" (idlwave-complete 'procedure) t]
+      ["2 Procedure Keyword" (idlwave-complete 'procedure-keyword) t]
+      "--"
+      ["3 Function Name" (idlwave-complete 'function) t]
+      ["4 Function Keyword" (idlwave-complete 'function-keyword) t]
+      "--"
+      ["5 Procedure Method Name" (idlwave-complete 'procedure-method) t]
+      ["6 Procedure Method Keyword" (idlwave-complete 'procedure-method-keyword) t]
+      "--"
+      ["7 Function Method Name" (idlwave-complete 'function-method) t]
+      ["8 Function Method Keyword" (idlwave-complete 'function-method-keyword) t]
+      "--"
+      ["9 Class Name"  idlwave-complete-class t])
+     "--"
+     ["Show Routine Info" idlwave-routine-info t]
+     ["Show Routine Doc with IDLHELP" idlwave-routine-info-from-idlhelp t]
+     "--"
+     ["Find Routine Source" idlwave-find-module t]
+     "--"
+     ["Update Routine Info" idlwave-update-routine-info t]
+     "--"
+     "IDL Library Routine Info"
+     ["Select Library Directories" idlwave-create-libinfo-file t]
+     ["Scan Directories" (idlwave-update-routine-info '(16)) idlwave-scanned-lib-directories])
+     "--"
+    ("External"
+     ["Generate IDL tags" idlwave-make-tags t]
+     ["Start IDL shell" idlwave-shell t]
+     ["Edit file in IDLDE" idlwave-edit-in-idlde t]
+     ["Launch IDL Help" idlwave-launch-idlhelp t])
+    "--"
+    ("Customize"
+     ["Browse IDLWAVE Group" idlwave-customize t]
+     "--"
+     ["Build Full Customize Menu" idlwave-create-customize-menu 
+      (fboundp 'customize-menu-create)])
+    ("Documentation"
+     ["Describe Mode" describe-mode t]
+     ["Abbreviation List" idlwave-list-abbrevs t]
+     "--"
+     ["Commentary in idlwave.el" idlwave-show-commentary t]
+     ["Commentary in idlwave-shell.el" idlwave-shell-show-commentary t]
+     "--"
+     ["Info" idlwave-info t]
+     "--"
+     ["Launch IDL Help" idlwave-launch-idlhelp t])))
+
+(defvar idlwave-mode-debug-menu-def
+  '("Debug"
+    ["Start IDL shell" idlwave-shell t]
+    ["Save and .RUN buffer" idlwave-shell-save-and-run
+     (and (boundp 'idlwave-shell-automatic-start) 
+         idlwave-shell-automatic-start)]))
+
+(if (or (featurep 'easymenu) (load "easymenu" t))
+    (progn
+      (easy-menu-define idlwave-mode-menu idlwave-mode-map 
+                       "IDL and WAVE CL editing menu" 
+                       idlwave-mode-menu-def)
+      (easy-menu-define idlwave-mode-debug-menu idlwave-mode-map 
+                       "IDL and WAVE CL editing menu" 
+                       idlwave-mode-debug-menu-def)))
+
+(defun idlwave-customize ()
+  "Call the customize function with idlwave as argument."
+  (interactive)
+  ;; Try to load the code for the shell, so that we can customize it 
+  ;; as well.
+  (or (featurep 'idlwave-shell)
+      (load "idlwave-shell" t))
+  (customize-browse 'idlwave))
+
+(defun idlwave-create-customize-menu ()
+  "Create a full customization menu for IDLWAVE, insert it into the menu."
+  (interactive)
+  (if (fboundp 'customize-menu-create)
+      (progn
+       ;; Try to load the code for the shell, so that we can customize it 
+       ;; as well.
+       (or (featurep 'idlwave-shell)
+           (load "idlwave-shell" t))
+       (easy-menu-change 
+        '("IDLWAVE") "Customize"
+        `(["Browse IDLWAVE group" idlwave-customize t]
+          "--"
+          ,(customize-menu-create 'idlwave)
+          ["Set" Custom-set t]
+          ["Save" Custom-save t]
+          ["Reset to Current" Custom-reset-current t]
+          ["Reset to Saved" Custom-reset-saved t]
+          ["Reset to Standard Settings" Custom-reset-standard t]))
+       (message "\"IDLWAVE\"-menu now contains full customization menu"))
+    (error "Cannot expand menu (outdated version of cus-edit.el)")))
+
+(defun idlwave-show-commentary ()
+  "Use the finder to view the file documentation from `idlwave.el'."
+  (interactive)
+  (require 'finder)
+  (finder-commentary "idlwave.el"))
+
+(defun idlwave-shell-show-commentary ()
+  "Use the finder to view the file documentation from `idlwave-shell.el'."
+  (interactive)
+  (require 'finder)
+  (finder-commentary "idlwave-shell.el"))
+
+(defun idlwave-info ()
+  "Read documentation for IDLWAVE in the info system."
+  (interactive)
+  (require 'info)
+  (Info-goto-node "(idlwave)"))
+
+(defun idlwave-list-abbrevs (arg)
+  "Show the code abbreviations define in IDLWAVE mode.
+This lists all abbrevs where the replacement text differs from the input text.
+These are the ones the users want to learn to speed up their writing.
+
+The function does *not* list abbrevs which replace a word with itself
+to call a hook.  These hooks are used to change the case of words or
+to blink the matching `begin', and the user does not need to know them.
+
+With arg, list all abbrevs with the corresponding hook.
+
+This function was written since `list-abbrevs' looks terrible for IDLWAVE mode."
+
+  (interactive "P")
+  (let ((table (symbol-value 'idlwave-mode-abbrev-table))
+       abbrevs
+       str rpl func fmt (len-str 0) (len-rpl 0))
+    (mapatoms 
+     (lambda (sym)
+       (if (symbol-value sym)
+          (progn
+            (setq str (symbol-name sym)
+                  rpl (symbol-value sym)
+                  func (symbol-function sym))
+            (if arg
+                (setq func (prin1-to-string func))
+              (if (and (listp func) (stringp (nth 2 func)))
+                  (setq rpl (concat "EVAL: " (nth 2 func))
+                        func "")
+                (setq func "")))
+            (if (or arg (not (string= rpl str)))
+                (progn
+                  (setq len-str (max len-str (length str)))
+                  (setq len-rpl (max len-rpl (length rpl)))
+                  (setq abbrevs (cons (list str rpl func) abbrevs)))))))
+     table)
+    ;; sort the list
+    (setq abbrevs (sort abbrevs (lambda (a b) (string< (car a) (car b)))))
+    ;; Make the format
+    (setq fmt (format "%%-%ds   %%-%ds   %%s\n" len-str len-rpl))
+    (with-output-to-temp-buffer "*Help*"
+      (if arg
+         (progn
+           (princ "Abbreviations and Actions in IDLWAVE-Mode\n") 
+           (princ "=========================================\n\n")
+           (princ (format fmt "KEY" "REPLACE" "HOOK"))
+           (princ (format fmt "---" "-------" "----")))
+       (princ "Code Abbreviations and Templates in IDLWAVE-Mode\n")
+       (princ "================================================\n\n")
+       (princ (format fmt "KEY" "ACTION" ""))
+       (princ (format fmt "---" "------" "")))
+      (mapcar
+       (lambda (list)
+        (setq str (car list)
+              rpl (nth 1 list)
+              func (nth 2 list))
+        (princ (format fmt str rpl func)))
+       abbrevs)))
+  ;; Make sure each abbreviation uses only one display line
+  (save-excursion
+    (set-buffer "*Help*")
+    (setq truncate-lines t)))
+
+(run-hooks 'idlwave-load-hook)
+
+(provide 'idlwave)
+
+;;; idlwave.el ends here
+