emacs-matlab/matlab.el

6288 lines
238 KiB
EmacsLisp
Raw Normal View History

;;; matlab.el --- major mode for MATLAB(R) dot-m files
2005-12-01 20:08:16 +01:00
;; Author: Matt Wette <mwette@alumni.caltech.edu>,
;; Eric M. Ludlam <eludlam@mathworks.com>
;; Maintainer: Eric M. Ludlam <eludlam@mathworks.com>
;; Created: 04 Jan 91
;; Keywords: MATLAB(R)
2005-12-01 20:08:16 +01:00
;; Version:
(defconst matlab-mode-version "3.3.6"
"Current version of MATLAB(R) mode.")
2005-12-01 20:08:16 +01:00
;;
;; Copyright (C) 2004-2010, 2014, 2016-2017 The Mathworks, Inc
2005-12-01 20:08:16 +01:00
;; Copyright (C) 1997-2004 Eric M. Ludlam: The MathWorks, Inc
;; Copyright (C) 1991-1997 Matthew R. Wette
;;
;; This program is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation; either version 2, or (at your option)
;; any later version.
;;
;; This program is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;;
;; You should have received a copy of the GNU General Public License
;; along with GNU Emacs; see the file COPYING. If not, write to
;; the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
;;
;;; Commentary:
;;
;; This major mode for GNU Emacs provides support for editing MATLAB(R) dot-m
2005-12-01 20:08:16 +01:00
;; files. It automatically indents for block structures (including nested
;; functions), line continuations (e.g., ...), and comments.
;;
;; Additional features include auto-fill including auto-additions of
;; ellipsis for commands, and even strings. Block/end construct
;; highlighting as you edit. Primitive code-verification and
;; identification. Templates and other code editing functions.
;; Advanced symbol completion. Code highlighting via font-lock.
;; There are many navigation commands that let you move across blocks
;; of code at different levels.
;;
;; Lastly, there is support for running MATLAB(R) in an Emacs buffer,
2005-12-01 20:08:16 +01:00
;; with full shell history and debugger support (when used with the db
;; commands.) The shell can be used as an online help while editing
;; code, providing help on functions, variables, or running arbitrary
;; blocks of code from the buffer you are editing.
;;; Code:
(require 'easymenu)
(require 'tempo)
(require 'derived)
;; compatibility
(eval-and-compile
(if (string-match "X[Ee]macs" emacs-version)
(progn
(defalias 'matlab-make-overlay 'make-extent)
(defalias 'matlab-overlay-put 'set-extent-property)
(defalias 'matlab-overlay-get 'extent-property)
(defalias 'matlab-delete-overlay 'delete-extent)
(defalias 'matlab-overlay-start 'extent-start-position)
(defalias 'matlab-overlay-end 'extent-end-position)
(defalias 'matlab-previous-overlay-change 'previous-extent-change)
(defalias 'matlab-next-overlay-change 'next-extent-change)
(defalias 'matlab-overlays-at
(lambda (pos) (when (fboundp 'extent-list) (extent-list nil pos pos))))
(defalias 'matlab-cancel-timer 'delete-itimer)
(defun matlab-run-with-idle-timer (secs repeat function &rest args)
(condition-case nil
(apply 'start-itimer
"matlab" function secs
(if repeat secs nil) t
t (car args)))
2005-12-01 20:08:16 +01:00
(error
;; If the above doesn't work, then try this old version of
;; start itimer.
(when (fboundp 'start-itimer)
(start-itimer "matlab" function secs (if repeat secs nil)))))
)
;; Else GNU Emacs
(defalias 'matlab-make-overlay 'make-overlay)
(defalias 'matlab-overlay-put 'overlay-put)
(defalias 'matlab-overlay-get 'overlay-get)
(defalias 'matlab-delete-overlay 'delete-overlay)
(defalias 'matlab-overlay-start 'overlay-start)
(defalias 'matlab-overlay-end 'overlay-end)
(defalias 'matlab-previous-overlay-change 'previous-overlay-change)
(defalias 'matlab-next-overlay-change 'next-overlay-change)
(defalias 'matlab-overlays-at 'overlays-at)
(defalias 'matlab-cancel-timer 'cancel-timer)
(defalias 'matlab-run-with-idle-timer 'run-with-idle-timer)
))
2005-12-01 20:08:16 +01:00
;;; Helper aliases to suppress compiler warnings ===============================
(eval-and-compile
;; `set-face-underline-p' is an obsolete function (as of 24.3); use `set-face-underline' instead.
(cond ((fboundp 'set-face-underlined)
(defalias 'matlab-set-face-underline 'set-face-underlined))
(t
(defalias 'matlab-set-face-underline 'set-face-underline-p)))
;; `set-face-bold-p' is an obsolete function (as of 24.4); use `set-face-bold' instead.
(cond ((fboundp 'set-face-bold)
(defalias 'matlab-set-face-bold 'set-face-bold))
(t
(defalias 'matlab-set-face-bold 'set-face-bold-p)))
;; `default-fill-column' is an obsolete variable (as of 23.2); use `fill-column' instead.
(cond ((boundp 'fill-column)
(defvaralias 'matlab-fill-column 'fill-column))
(t
(defvaralias 'matlab-fill-column 'default-fill-column)))
;; `interactive-p' is an obsolete function (as of 23.2); use `called-interactively-p' instead.
(defun matlab-called-interactively-p-helper ()
(called-interactively-p 'interactive))
(cond ((fboundp 'called-interactively-p)
(defalias 'matlab-called-interactively-p 'matlab-called-interactively-p-helper))
(t
(defalias 'matlab-called-interactively-p 'interactive-p)))
;; `toggle-read-only' is an obsolete function (as of 24.3); use `read-only-mode' instead.
;; (matlab-read-only-mode -1) ==> make writable
;; (matlab-read-only-mode 1) ==> make read-only
(cond ((fboundp 'read-only-mode)
(defalias 'matlab-read-only-mode 'read-only-mode))
(t
(defalias 'matlab-read-only-mode 'toggle-read-only)))
(cond ((fboundp 'point-at-bol)
(defalias 'matlab-point-at-bol 'point-at-bol)
(defalias 'matlab-point-at-eol 'point-at-eol))
;; Emacs 20.4
((fboundp 'line-beginning-position)
(defalias 'matlab-point-at-bol 'line-beginning-position)
(defalias 'matlab-point-at-eol 'line-end-position))
(t
(defmacro matlab-point-at-bol ()
(save-excursion (beginning-of-line) (point)))
(defmacro matlab-point-at-eol ()
(save-excursion (end-of-line) (point)))))
)
2005-12-01 20:08:16 +01:00
(defmacro matlab-run-in-matlab-mode-only (&rest body)
"Execute BODY only if the active buffer is a MATLAB(R) M-file buffer."
2005-12-01 20:08:16 +01:00
`(if (eq major-mode 'matlab-mode)
(progn
,@body)
(error "This command works only in a MATLAB M-file buffer")))
;;; User-changeable variables =================================================
;; Variables which the user can change
(defgroup matlab nil
"MATLAB(R) mode."
2005-12-01 20:08:16 +01:00
:prefix "matlab-"
:group 'languages)
(defcustom matlab-indent-level 4
"*The basic indentation amount in `matlab-mode'."
:group 'matlab
:type 'integer)
(defcustom matlab-cont-level 4
"*Basic indentation after continuation if no other methods are found."
:group 'matlab
:type 'integer)
(defcustom matlab-cont-requires-ellipsis t
"*Specify if ellipses are required at the end of a line for continuation.
Future versions of Matlab may not require ellipses ... , so a heuristic
determining if there is to be continuation is used instead."
:group 'matlab
:type 'integer)
(defcustom matlab-case-level '(2 . 2)
"*How far to indent case/otherwise statements in a switch.
This can be an integer, which is the distance to indent the CASE and
OTHERWISE commands, and how far to indent commands appearing in CASE
and OTHERWISE blocks. It can also be a cons cell which is of form
(CASEINDENT . COMMANDINDENT)
where CASEINDENT is the indentation of the CASE and OTHERWISE
statements, and COMMANDINDENT is the indentation of commands appearing
after the CASE or OTHERWISE command.
Note: Currently a bug exists if:
CASEINDENT+COMMANDINDENT != `matlab-indent-level'
so if you customize these variables, follow the above rule, and you
should be ok."
:group 'matlab
:type 'sexp)
(defcustom matlab-align-to-paren t
"*Whether continuation lines should be aligned to the opening parenthesis.
When non-nil, continuation lines are aligned to the opening parenthesis if the
opening is not followed by only spaces and ellipses. When nil, continued lines
are simply indented by `matlab-cont-level'."
:group 'matlab
:type 'boolean
)
(defcustom matlab-indent-function-body 'guess
"*If non-nil, indent body of function.
If the global value is nil, do not indent function bodies.
If the global value is t, always indent function bodies.
If the global value is 'guess, then the local value will be set to
either nil or t when the MATLAB mode is started in a buffer based on the
file's current indentation.
If the global value is 'MathWorks-Standard, then the local value is not
changed, and functions are indented based on `matlab-functions-have-end'."
2005-12-01 20:08:16 +01:00
:group 'matlab
:type '(choice (const :tag "Always" t)
(const :tag "Never" nil)
(const :tag "Guess" 'guess)
(const :tag "MathWorks Standard"
'MathWorks-Standard))
)
2005-12-01 20:08:16 +01:00
(make-variable-buffer-local 'matlab-indent-function-body)
(defcustom matlab-functions-have-end nil
"*If non-nil, functions-have-end minor mode is on by default."
:group 'matlab
:type 'boolean)
(make-variable-buffer-local 'matlab-functions-have-end)
(defun matlab-toggle-functions-have-end ()
(interactive)
(matlab-toggle-functions-have-end-minor-mode))
;; The following minor mode is on if and only if the above variable is true;
(easy-mmode-define-minor-mode matlab-functions-have-end-minor-mode
"Toggle functions-have-end minor mode, indicating function/end pairing."
2005-12-01 20:08:16 +01:00
nil
(condition-case nil ;; avoid parse error on xemacs
(eval (read "#(\" function...end\" 0 15 (face (font-lock-keyword-face) fontified t))"))
(error " function...end"))
nil ; empty mode-map
;; body of matlab-functions-have-end-minor-mode
(if matlab-functions-have-end-minor-mode
(setq matlab-functions-have-end t)
(setq matlab-functions-have-end nil)
)
2005-12-01 20:08:16 +01:00
)
(defvar matlab-defun-regex) ;; Quiet compiler warning (is defined below)
(defun matlab-do-functions-have-end-p ()
"Non-nil if the first function in the current buffer terminates with end."
(save-excursion
(goto-char (point-min))
(if (re-search-forward matlab-defun-regex nil t)
(let ((matlab-functions-have-end t))
(beginning-of-line)
(condition-case nil
(progn (matlab-forward-sexp) t)
(error nil))
)
nil
)
))
(defun matlab-toggle-functions-have-end-minor-mode ()
(matlab-functions-have-end-minor-mode)
(if (and matlab-functions-have-end-minor-mode (not (eq major-mode 'matlab-mode)))
(progn
(matlab-functions-have-end-minor-mode -1)
(error "functions-have-end minor mode is only for MATLAB Major mode")))
(setq matlab-functions-have-end matlab-functions-have-end-minor-mode))
(defun matlab-indent-function-body-p ()
"Non-nil if functions bodies are indented.
See `matlab-indent-function-body' variable."
(if (eq matlab-indent-function-body 'MathWorks-Standard)
;; Dec '09
;; The MathWorks standard is the same as if functions have end.
matlab-functions-have-end
;; Else, just return the variable.
matlab-indent-function-body))
2005-12-01 20:08:16 +01:00
(defcustom matlab-indent-past-arg1-functions
"[sg]et\\(_param\\)?\\|waitfor"
"*Regex describing functions whose first arg is special.
This specialness means that all following parameters which appear on
continued lines should appear indented to line up with the second
argument, not the first argument."
:group 'matlab
:type 'string)
(defcustom matlab-arg1-max-indent-length 15
"*The maximum length to indent when indenting past arg1.
If arg1 is exceptionally long, then only this number of characters
will be indented beyond the open paren starting the parameter list.")
(defcustom matlab-maximum-indents '(;; = is a convenience. Don't go too far
(?= . (10 . 4))
;; Fns should provide hard limits
(?\( . 50)
;; Matrix/Cell arrays
(?\[ . 20)
(?\{ . 20))
"Alist of maximum indentations when lining up code.
Each element is of the form (CHAR . INDENT) where char is a character
the indent engine is using, and INDENT is the maximum indentation
allowed. Indent could be of the form (MAXIMUM . INDENT), where
MAXIMUM is the maximum allowed calculated indent, and INDENT is the
amount to use if MAXIMUM is reached."
:group 'matlab
:type '(repeat (cons (character :tag "Open List Character")
(sexp :tag "Number (max) or cons (max indent)"))))
(defcustom matlab-handle-simulink t
"*If true, add in a few simulink customizations.
This variable's state is mostly useful when set at load time when
simulink font lock keywords can be removed. This will handle
additional cases as the need arises."
2005-12-01 20:08:16 +01:00
:group 'matlab
:type 'boolean)
(defcustom matlab-auto-fill t
"*If true, set variable `auto-fill-function' to our function at startup."
:group 'matlab
:type 'boolean)
(defcustom matlab-fill-fudge 10
"Number of characters around `fill-column' we can fudge filling.
Basically, there are places that are very convenient to fill at, but
might not be the closest fill spot, or occur after `fill-column'.
If they occur within this fudge factor, we will use them.
Also, if none of the above occur, and we find a symbol to break at,
but an open paren (group) starts or ends within this fudge factor,
move there to boost the amount of fill leverage we can get."
:group 'matlab
:type 'integer)
(defcustom matlab-fill-fudge-hard-maximum 79
"The longest line allowed when auto-filling code.
This overcomes situations where the `fill-column' plus the
`matlab-fill-fudge' is greater than some hard desired limit."
:group 'matlab
:type 'integer)
(defcustom matlab-elipsis-string "..."
"Text used to perform continuation on code lines.
This is used to generate and identify continuation lines.")
(defcustom matlab-fill-code t
"*If true, `auto-fill-mode' causes code lines to be automatically continued."
:group 'matlab
:type 'boolean)
(defcustom matlab-fill-count-ellipsis-flag t
"*Non-nil means to count the ellipsis when auto filling.
This effectively shortens the `fill-column' by the length of
`matlab-elipsis-string'.")
(defcustom matlab-fill-strings-flag t
"*Non-nil means that when auto-fill is on, strings are broken across lines.
If `matlab-fill-count-ellipsis-flag' is non nil, this shortens the
`fill-column' by the length of `matlab-elipsis-string'.")
(defcustom matlab-comment-column 40
"*The goal comment column in `matlab-mode' buffers."
:group 'matlab
:type 'integer)
(defcustom matlab-comment-anti-indent 0
"*Amount of anti-indentation to use for comments in relation to code."
:group 'matlab
:type 'integer)
(defcustom matlab-comment-line-s "% "
"*String to start comment on line by itself."
:group 'matlab
:type 'string)
(defcustom matlab-comment-on-line-s "% "
"*String to start comment on line with code."
:group 'matlab
:type 'string)
(defcustom matlab-comment-region-s "% $$$ "
"*String inserted by \\[matlab-comment-region] at start of each line in \
region."
:group 'matlab
:type 'string)
(defcustom matlab-verify-on-save-flag t
"*Non-nil means to verify M whenever we save a file."
:group 'matlab
:type 'boolean)
(defcustom matlab-mode-verify-fix-functions
'(matlab-mode-vf-functionname matlab-mode-vf-classname)
2005-12-01 20:08:16 +01:00
"List of function symbols which perform a verification and fix to M code.
Each function gets no arguments, and returns nothing. They can move
point, but it will be restored for them."
:group 'matlab
:type '(repeat (choice :tag "Function: "
'(matlab-mode-vf-functionname
matlab-mode-vf-classname
2005-12-01 20:08:16 +01:00
matlab-mode-vf-block-matches-forward
matlab-mode-vf-block-matches-backward
matlab-mode-vf-quiesce-buffer
))))
(defcustom matlab-block-verify-max-buffer-size 50000
"*Largest buffer size allowed for block verification during save."
:group 'matlab
:type 'integer)
;; It is time to disable this.
(defcustom matlab-vers-on-startup nil
2005-12-01 20:08:16 +01:00
"*If non-nil, show the version number on startup."
:group 'matlab
:type 'boolean)
(defcustom matlab-highlight-block-match-flag t
"*Non-nil means to highlight the matching if/end/whatever.
The highlighting only occurs when the cursor is on a block start or end
keyword."
:group 'matlab
:type 'boolean)
(defcustom matlab-show-periodic-code-details-flag nil
"*Non-nil means to show code details in the minibuffer.
This will only work if `matlab-highlight-block-match-flag' is non-nil."
:group 'matlab
:type 'boolean)
(defcustom matlab-use-eei t
"*Use Emacs Link for save-and-go and run-region."
:group 'matlab
:type 'boolean)
(defcustom matlab-mode-hook nil
"*List of functions to call on entry to MATLAB mode."
:group 'matlab
:type 'hook)
(defcustom matlab-completion-technique 'complete
"*How the `matlab-complete-symbol' interfaces with the user.
Valid values are:
'increment - which means that new strings are tried with each
successive call until all methods are exhausted.
(Similar to `hippie-expand'.)
'complete - Which means that if there is no single completion, then
all possibilities are displayed in a completion buffer."
:group 'matlab
:type '(radio (const :tag "Incremental completion (hippie-expand)."
increment)
(const :tag "Show completion buffer."
complete)))
(defcustom matlab-show-mlint-warnings nil
"*If non-nil, show mlint warnings."
:group 'matlab
:type 'boolean)
(make-variable-buffer-local 'matlab-show-mlint-warnings)
(defcustom matlab-highlight-cross-function-variables nil
"*If non-nil, highlight cross-function variables."
:group 'matlab
:type 'boolean)
(make-variable-buffer-local 'matlab-highlight-cross-function-variables)
(defcustom matlab-return-add-semicolon nil
"*If non nil, check to see a semicolon is needed when RET is pressed."
:group 'matlab
:type 'boolean)
(make-variable-buffer-local 'matlab-return-add-semicolon)
2016-02-10 11:27:23 +01:00
(defcustom matlab-change-current-directory nil
"*If non nil, make file's directory the current directory when
evaluating it."
:group 'matlab
:type 'boolean)
(make-variable-buffer-local 'matlab-change-current-directory)
2005-12-01 20:08:16 +01:00
;; Load in the region we use for highlighting stuff.
(if (and (featurep 'custom) (fboundp 'custom-declare-variable))
(let ((l-region-face (if (facep 'region) 'region 'zmacs-region)))
;; If we have custom, we can make our own special face like this
(defface matlab-region-face
(list
(list t
(list :background (face-background l-region-face)
:foreground (face-foreground l-region-face))))
"*Face used to highlight a matlab region."
:group 'matlab))
;; If we do not, then we can fake it by copying 'region.
(cond ((facep 'region)
(copy-face 'region 'matlab-region-face))
(t
(copy-face 'zmacs-region 'matlab-region-face))))
(defvar matlab-unterminated-string-face 'matlab-unterminated-string-face
"Self reference for unterminated string face.")
(defvar matlab-simulink-keyword-face 'matlab-simulink-keyword-face
"Self reference for simulink keywords.")
(defvar matlab-nested-function-keyword-face 'matlab-nested-function-keyword-face
"Self reference for nested function/end keywords.")
(defvar matlab-cross-function-variable-face 'matlab-cross-function-variable-face
"Self reference for cross-function variables.")
(defvar matlab-cellbreak-face 'matlab-cellbreak-face
"Self reference for cellbreaks.")
2005-12-01 20:08:16 +01:00
(defun matlab-font-lock-adjustments ()
"Make adjustments for font lock.
If font lock is not loaded, lay in wait."
(if (and (featurep 'custom) (fboundp 'custom-declare-variable))
2005-12-01 20:08:16 +01:00
(progn
(defface matlab-unterminated-string-face
(list
(list t
(list :background (face-background font-lock-string-face)
:foreground (face-foreground font-lock-string-face)
:underline t)))
"*Face used to highlight unterminated strings."
:group 'matlab)
(defface matlab-simulink-keyword-face
(list
(list t
(list :background (face-background font-lock-type-face)
:foreground (face-foreground font-lock-type-face)
:underline t)))
"*Face used to highlight simulink specific functions."
:group 'matlab)
(defface matlab-nested-function-keyword-face
(list
(list t
(list :slant 'italic)))
"*Face to use for cross-function variables.")
(defface matlab-cross-function-variable-face
(list
(list t
(list :weight 'bold
:slant 'italic)))
"*Face to use for cross-function variables."
:group 'matlab)
(defface matlab-cellbreak-face
(list
(list t
(list :background (face-background font-lock-comment-face)
:foreground (face-foreground font-lock-comment-face)
:overline t
:bold t)))
"*Face to use for cellbreak %% lines.")
2005-12-01 20:08:16 +01:00
)
2005-12-01 20:08:16 +01:00
;; Now, lets make the unterminated string face
(cond ((facep 'font-lock-string-face)
(copy-face 'font-lock-string-face
'matlab-unterminated-string-face))
(t
(make-face 'matlab-unterminated-string-face)))
(matlab-set-face-underline 'matlab-unterminated-string-face t)
2005-12-01 20:08:16 +01:00
;; Now make some simulink faces
(cond ((facep 'font-lock-type-face)
(copy-face 'font-lock-type-face 'matlab-simulink-keyword-face))
(t
(make-face 'matlab-simulink-keyword-face)))
(matlab-set-face-underline 'matlab-simulink-keyword-face t)
2005-12-01 20:08:16 +01:00
;; Now make some nested function/end keyword faces
(cond ((facep 'font-lock-type-face)
(copy-face 'font-lock-type-face 'matlab-nested-function-keyword-face))
(t
(make-face 'matlab-nested-function-keyword-face)))
2005-12-01 20:08:16 +01:00
;; Now make some cross-function variable faces
(cond ((facep 'font-lock-type-face)
(copy-face 'font-lock-type-face 'matlab-cross-function-variable-face))
(t
(make-face 'matlab-cross-function-variable-face)))
(matlab-set-face-bold 'matlab-cross-function-variable-face t)
;; Now make some cellbreak variable faces
(cond ((facep 'font-comment-face)
(copy-face 'font-lock-comment-face 'matlab-cellbreak-face))
(t
(make-face 'matlab-cellbreak-face)))
(matlab-set-face-bold 'matlab-cellbreak-face t)
(condition-case nil
(set-face-attribute 'matlab-cellbreak-face nil :overline t)
(error nil))
2005-12-01 20:08:16 +01:00
)
(remove-hook 'font-lock-mode-hook 'matlab-font-lock-adjustments))
;; Make the adjustments for font lock after it's loaded.
;; I found that eval-after-load was unreliable.
(if (featurep 'font-lock)
(matlab-font-lock-adjustments)
(add-hook 'font-lock-mode-hook 'matlab-font-lock-adjustments))
;;; MATLAB mode variables =====================================================
(defvar matlab-tempo-tags nil
"List of templates used in MATLAB mode.")
;; syntax table
(defvar matlab-mode-syntax-table
(let ((st (make-syntax-table (standard-syntax-table))))
(modify-syntax-entry ?_ "_" st)
(modify-syntax-entry ?% "<" st)
(modify-syntax-entry ?\n ">" st)
(modify-syntax-entry ?\\ "." st)
(modify-syntax-entry ?\t " " st)
(modify-syntax-entry ?+ "." st)
(modify-syntax-entry ?- "." st)
(modify-syntax-entry ?* "." st)
(modify-syntax-entry ?' "." st)
(modify-syntax-entry ?/ "." st)
(modify-syntax-entry ?= "." st)
(modify-syntax-entry ?< "." st)
(modify-syntax-entry ?> "." st)
(modify-syntax-entry ?& "." st)
(modify-syntax-entry ?| "." st)
st)
"The syntax table used in `matlab-mode' buffers.")
(defvar matlab-mode-special-syntax-table
(let ((st (copy-syntax-table matlab-mode-syntax-table)))
;; Make _ a part of words so we can skip them better
(modify-syntax-entry ?_ "w" st)
st)
"The syntax table used when navigating blocks.")
;; abbrev table
(defvar matlab-mode-abbrev-table nil
"The abbrev table used in `matlab-mode' buffers.")
(define-abbrev-table 'matlab-mode-abbrev-table ())
;;; Keybindings ===============================================================
(defvar matlab-help-map
(let ((km (make-sparse-keymap)))
(define-key km "r" 'matlab-shell-run-command)
(define-key km "f" 'matlab-shell-describe-command)
(define-key km "a" 'matlab-shell-apropos)
(define-key km "v" 'matlab-shell-describe-variable)
(define-key km "t" 'matlab-shell-topic-browser)
km)
"The help key map for `matlab-mode' and `matlab-shell-mode'.")
(defvar matlab-insert-map
(let ((km (make-sparse-keymap)))
(define-key km "c" 'matlab-insert-next-case)
(define-key km "e" 'matlab-insert-end-block)
(define-key km "i" 'tempo-template-matlab-if)
(define-key km "I" 'tempo-template-matlab-if-else)
(define-key km "f" 'tempo-template-matlab-for)
(define-key km "s" 'tempo-template-matlab-switch)
(define-key km "t" 'tempo-template-matlab-try)
(define-key km "w" 'tempo-template-matlab-while)
(define-key km "F" 'tempo-template-matlab-function)
(define-key km "'" 'matlab-stringify-region)
;; Not really inserts, but auto coding stuff
(define-key km "\C-s" 'matlab-ispell-strings)
(define-key km "\C-c" 'matlab-ispell-comments)
km)
"Keymap used for inserting simple texts based on context.")
;; mode map
(defvar matlab-mode-map
(let ((km (make-sparse-keymap)))
(define-key km [return] 'matlab-return)
(define-key km "%" 'matlab-electric-comment)
(define-key km "\C-c;" 'matlab-comment-region)
(define-key km "\C-c:" 'matlab-uncomment-region)
(define-key km [(control c) return] 'matlab-comment-return)
(define-key km [(control c) (control c)] matlab-insert-map)
(define-key km [(control c) (control f)] 'matlab-fill-comment-line)
(define-key km [(control c) (control j)] 'matlab-justify-line)
(define-key km [(control c) (control q)] 'matlab-fill-region)
(define-key km [(control c) (control s)] 'matlab-shell-save-and-go)
(define-key km [(control c) (control r)] 'matlab-shell-run-region)
(define-key km [(meta control return)] 'matlab-shell-run-cell)
2005-12-01 20:08:16 +01:00
(define-key km [(control c) (control t)] 'matlab-show-line-info)
(define-key km [(control c) ?. ] 'matlab-find-file-on-path)
(define-key km [(control h) (control m)] matlab-help-map)
(define-key km [(control j)] 'matlab-linefeed)
(define-key km "\M-\r" 'newline)
(define-key km [(meta \;)] 'matlab-comment)
(define-key km [(meta q)] 'matlab-fill-paragraph)
(define-key km [(meta a)] 'matlab-beginning-of-command)
(define-key km [(meta e)] 'matlab-end-of-command)
(define-key km [(meta j)] 'matlab-comment-line-break-function)
(define-key km [(meta s)] 'matlab-show-matlab-shell-buffer)
(define-key km "\M-\t" 'matlab-complete-symbol)
(define-key km [(meta control f)] 'matlab-forward-sexp)
(define-key km [(meta control b)] 'matlab-backward-sexp)
(define-key km [(meta control q)] 'matlab-indent-sexp)
(define-key km [(meta control a)] 'matlab-beginning-of-defun)
(define-key km [(meta control e)] 'matlab-end-of-defun)
(if (string-match "XEmacs" emacs-version)
(define-key km [(control meta button1)] 'matlab-find-file-click)
(define-key km [(control meta mouse-2)] 'matlab-find-file-click))
(substitute-key-definition 'comment-region 'matlab-comment-region
km) ; global-map ;torkel
km)
"The keymap used in `matlab-mode'.")
;;; Font locking keywords =====================================================
(defvar matlab-string-char-regexp "\\(?9:['\"]\\)"
"Regexp used to find a character that starts a char vector or string.
This remembers what chart was matched via index 9.")
(defvar matlab-string-start-regexp "\\(^\\|[^]})a-zA-Z0-9_.'\"]\\)"
2005-12-01 20:08:16 +01:00
"Regexp used to represent the character before the string char '.
The ' character has restrictions on what starts a string which is needed
when attempting to understand the current context.")
;; To quote a quote, put two in a row, thus we need an anchored
;; first quote. In addition, we don't want to color strings in comments.
(defvar matlab-string-content-regexp "[^\\9\n]*\\(?:\\9\\9[^\\9\n]*\\)*"
"Regexp that matches the content of a char vector or string.
Stops matching when end of line, or end of that char vector or string is met.
depends on `matlab-string-char-regexp' being at the beginning of a compound expression.")
(defvar matlab-string-end-regexp (concat matlab-string-content-regexp "\\9")
"Regexp used to represent the character pattern for ending a char vector or string.
This depends on `matlab-string-char-regexp' being used earlier to specify what
\9 represents (ie - either ' or \").")
(defun matlab-match-string-end-for-string ()
"If the last searched regexp included `matlab-string-char-regexp', then find the end of that string."
(let ((strchar (match-string 9)))
(re-search-forward (concat "[^" strchar "\n]*\\(?:" strchar strchar "[^" strchar "\n]*\\)*" strchar)
nil t)))
2005-12-01 20:08:16 +01:00
(defun matlab-font-lock-string-match-normal (limit)
"When font locking strings, call this function for normal strings.
Argument LIMIT is the maximum distance to scan."
(matlab-font-lock-string-match-here
(concat matlab-string-start-regexp
"\\(?2:" matlab-string-char-regexp matlab-string-end-regexp "\\)"
"\\([^\\9]\\|$\\)")
2005-12-01 20:08:16 +01:00
limit))
(defun matlab-font-lock-string-match-unterminated (limit)
"When font locking strings, call this function for normal strings.
Argument LIMIT is the maximum distance to scan."
(matlab-font-lock-string-match-here
(concat matlab-string-start-regexp "\\(?2:" matlab-string-char-regexp matlab-string-content-regexp "\\)$")
2005-12-01 20:08:16 +01:00
limit))
(defun matlab-font-lock-string-match-here (regex limit)
"When font-locking strings, call this function to determine a match.
Argument REGEX is the expression to scan for. Match 2 must be the string.
Argument LIMIT is the maximum distance to scan."
(let (e)
(while (and (re-search-forward regex limit t)
(progn
;; This gets us out of a comment after the string.
(setq e (match-end 2))
(goto-char (match-beginning 2))
(prog1
(or (matlab-cursor-in-comment)
(if (bolp) nil
(save-excursion
(forward-char -1)
(matlab-cursor-in-string))))
(goto-char e))))
(setq e nil))
(if (not e)
nil
(goto-char e)
t)))
(defun matlab-font-lock-comment-match (limit)
"When font-locking comments, call this function to determine a match.
Argument LIMIT is the maximum distance to scan."
(let (e)
(while (and (re-search-forward "\\(%[^%\n]*\\)" limit t)
(progn
(setq e (match-end 1))
(member (get-text-property (match-beginning 0) 'face)
'(font-lock-string-face
matlab-unterminated-string-face))))
(setq e nil))
(if (not e)
nil
(goto-char e)
t)))
(defun matlab-find-unreachable-code (limit)
"Find code that is if'd out with if(0) or if(false), and mark it as a comment.
The if(0) and else/end construct should be highlighted differently.
Argument LIMIT is the maximum distance to search."
(if (and (< (point) limit)
(re-search-forward
"\\<\\(if\\>\\s-*(?\\s-*\\(0\\|false\\)\\s-*)?$\\)"
limit t))
(let ((b1 (match-beginning 1))
(e1 (match-end 1))
(b2 nil) (e2 nil)
(b3 nil) (e3 nil))
(goto-char b1)
(condition-case nil
(progn
;; Go forward over the matlab sexp. Include scanning
;; for ELSE since parts of the ELSE block are not
;; `commented out'.
(matlab-forward-sexp t)
(forward-word -1)
;; Is there an ELSE in this block?
(if (looking-at (matlab-block-mid-re))
(progn
(setq b3 (match-beginning 0)
e3 (match-end 0))
;; Now find the REAL end.
(matlab-forward-sexp)
(forward-word -1)))
;; End of block stuff
(if (looking-at (matlab-block-end-re))
(progn
(setq b2 (match-beginning 0)
e2 (match-end 0))
;; make sure something exists...
(if (not b3) (setq b3 b2 e3 e2)))
(error "Eh?"))
;; Ok, build up some match data.
(set-match-data
(list b1 e2 ;the real deal.
b1 e1 ;if (0)
b2 e2 ;end
b3 e3 ;else (if applicable.)
b1 e3)) ;body commented out.
t)
(error nil)))))
(defun matlab-font-lock-nested-function-keyword-match (limit)
"Find next nested function/end keyword for font-lock.
Argument LIMIT is the maximum distance to search."
; Because of the way overlays are setup, the cursor will be sitting
; on either a "function" or "end" keyword.
(catch 'result
(let ((pos (point))
overlays)
(while (< pos limit)
(setq overlays (matlab-overlays-at pos))
(while overlays
(let ((overlay (car overlays)))
(when (matlab-overlay-get overlay 'nested-function)
(when (= pos (matlab-overlay-start overlay))
(goto-char pos)
;; The following line presumably returns true.
(throw 'result (re-search-forward "function" (+ pos 8) t)))
(let ((end-of-overlay (- (matlab-overlay-end overlay) 3)))
(when (<= pos end-of-overlay)
(goto-char end-of-overlay)
(throw 'result
(re-search-forward "end" (+ end-of-overlay 3) t))))))
2005-12-01 20:08:16 +01:00
(setq overlays (cdr overlays)))
(setq pos (matlab-next-overlay-change pos)))
nil ;; no matches, stop
)))
(defun matlab-font-lock-cross-function-variables-match (limit)
"Find next cross-function variable for font-lock.
Argument LIMIT is the maximum distance to search."
(catch 'result
(let ((pos (point))
overlays variables)
(while (< pos limit)
(let ((overlays (matlab-overlays-at pos)))
(while overlays
(let ((overlay (car overlays)))
(setq variables (matlab-overlay-get
overlay 'cross-function-variables))
(if variables
(progn
(goto-char pos)
(setq pos (min limit (matlab-overlay-end overlay)))
(if (re-search-forward variables pos t)
(progn
(throw 'result t))))))
(setq overlays (cdr overlays))))
2005-12-01 20:08:16 +01:00
(setq pos (matlab-next-overlay-change pos)))
nil ;; no matches, stop
)))
(defun matlab-find-block-comments (limit)
"Find code that is commented out with %{ until %}.
Argument LIMIT is the maximum distance to search."
(if (and (< (point) limit)
(re-search-forward "%{" limit t))
(let ((b1 (match-beginning 0))
(e1 (match-end 0))
(b2 nil) (e2 nil)
(b3 nil) (e3 nil))
(goto-char b1)
(forward-char -1)
(when (not (matlab-cursor-in-comment))
(setq b2 (re-search-forward "%}" limit t))
(when b2
(setq b2 (match-beginning 0)
e2 (match-end 0))
(set-match-data
(list b1 e2 ; full match
b1 e2 ; the full comment
b1 e1 ; the block start
b2 e2 ; the block end
))
t
)))))
(defcustom matlab-keyword-list '("global" "persistent" "for" "parfor" "while"
"spmd" "if" "elseif" "else"
2005-12-01 20:08:16 +01:00
"endfunction" "return" "break" "continue"
"switch" "case" "otherwise" "try"
"catch" "tic" "toc"
;; MCOS keywords
2008-10-17 21:09:19 +02:00
"classdef" "properties" "methods" "enumeration"
)
2005-12-01 20:08:16 +01:00
"List of keywords for MATLAB used in highlighting.
Customizing this variable is only useful if `regexp-opt' is available."
:group 'matlab
:type '(repeat (string :tag "Keyword: ")))
(defcustom matlab-handle-graphics-list '("figure" "axes" "axis" "line"
"surface" "patch" "text" "light"
"image" "set" "get" "uicontrol"
"uimenu" "uitoolbar"
"uitoggletool" "uipushtool"
"uicontext" "uicontextmenu"
"setfont" "setcolor")
"List of handle graphics functions used in highlighting.
Customizing this variable is only useful if `regexp-opt' is available."
:group 'matlab
:type '(repeat (string :tag "HG Keyword: ")))
(defcustom matlab-debug-list '("dbstop" "dbclear" "dbcont" "dbdown" "dbmex"
"dbstack" "dbstatus" "dbstep" "dbtype" "dbup"
"dbquit")
"List of debug commands used in highlighting.
Customizing this variable is only useful if `regexp-opt' is available."
:group 'matlab
:type '(repeat (string :tag "Debug Keyword: ")))
;; font-lock keywords
(defvar matlab-font-lock-keywords
(list
;; String quote chars are also used as transpose, but only if directly
;; after characters, numbers, underscores, or closing delimiters.
'(matlab-font-lock-string-match-normal 2 font-lock-string-face)
;; A string with no termination is not currently highlighted.
;; This will show that the string needs some attention.
'(matlab-font-lock-string-match-unterminated
2 matlab-unterminated-string-face)
;; Comments must occur after the string, that way we can check to see
;; if the comment start char has occurred inside our string. (EL)
'(matlab-font-lock-comment-match 1 font-lock-comment-face)
;; Various pragmas should be in different colors.
;; I think pragmas are always lower case?
'("%#\\([a-z]+\\)" (1 'bold prepend))
;; General keywords
(list
(if (fboundp 'regexp-opt)
(concat "\\<\\(" (regexp-opt matlab-keyword-list) "\\)\\>")
;; Original hard-coded value for pre Emacs 20.1
"\\<\\(break\\|ca\\(se\\|tch\\)\\|e\\(lse\\(\\|if\\)\\|ndfunction\\)\
\\|\\(par\\)?for\\|spmd\\|global\\|if\\|otherwise\\|return\\|switch\\|try\\|while\\|tic\\|toc\\)\\>")
2005-12-01 20:08:16 +01:00
'(0 font-lock-keyword-face))
;; The end keyword is only a keyword when not used as an array
;; dereferencing part.
'("\\(^\\|[;,]\\)[ \t]*\\(end\\)\\b"
2 (if (matlab-valid-end-construct-p) font-lock-keyword-face nil))
;; How about unreachable code? MUsT BE AFTER KEYWORDS in order to
;; get double-highlighting.
'(matlab-find-unreachable-code
(1 'underline prepend) ;if part
(2 'underline prepend) ;end part
(3 'underline prepend) ;else part (if applicable)
(4 font-lock-comment-face prepend) ;commented out part.
)
;; block comments need to be commented out too!
'(matlab-find-block-comments
(1 font-lock-comment-face prepend) ; commented out
(2 'underline prepend)
(3 'underline prepend) ;the comment parts
)
;; Cell mode breaks get special treatment
'("^\\s-*\\(%%[^\n]*\n\\)" (1 matlab-cellbreak-face append))
2005-12-01 20:08:16 +01:00
;; Highlight cross function variables
'(matlab-font-lock-cross-function-variables-match
(1 matlab-cross-function-variable-face prepend))
;; Highlight nested function/end keywords
'(matlab-font-lock-nested-function-keyword-match
(0 matlab-nested-function-keyword-face prepend))
;; The global keyword defines some variables. Mark them.
'("^\\s-*global\\s-+"
("\\(\\w+\\)\\(\\s-*=[^,; \t\n]+\\|[, \t;]+\\|$\\)"
nil nil (1 font-lock-variable-name-face)))
2005-12-01 20:08:16 +01:00
;; Handle graphics stuff
(list
(if (fboundp 'regexp-opt)
(concat "\\<\\(" (regexp-opt matlab-handle-graphics-list) "\\)\\>")
;; The original regular expression for pre Emacs 20.1
"\\<\\(ax\\(es\\|is\\)\\|figure\\|get\\|image\\|li\\(ght\\|ne\\)\\|\
patch\\|s\\(et\\(\\|color\\|font\\)\\|urface\\)\\|text\\|\
ui\\(cont\\(ext\\(\\|menu\\)\\|rol\\)\\|menu\\|\
\\(toggle\\|push\\)tool\\|toolbar\\)\\)\\>")
'(0 font-lock-type-face))
)
"Expressions to highlight in MATLAB mode.")
(defconst matlab-function-arguments
"\\(([^)]*)\\)?\\s-*\\([,;\n%]\\|$\\)")
2005-12-01 20:08:16 +01:00
(defvar matlab-gaudy-font-lock-keywords
(append
matlab-font-lock-keywords
(list
;; defining a function, a (possibly empty) list of assigned variables,
;; function name, and an optional (possibly empty) list of input variables
(list (concat "^\\s-*\\(function\\)\\>[ \t\n.]*"
"\\(\\[[^]]*\\]\\|\\sw+\\)[ \t\n.]*"
"=[ \t\n.]*\\(\\sw+\\)[ \t\n.]*"
matlab-function-arguments)
'(1 font-lock-keyword-face append)
'(2 font-lock-variable-name-face append)
'(3 font-lock-function-name-face append))
;; defining a function, a function name, and an optional (possibly
;; empty) list of input variables
(list (concat "^\\s-*\\(function\\)[ \t\n.]+"
"\\(\\sw+\\)[ \t\n.]*"
matlab-function-arguments)
'(1 font-lock-keyword-face append)
'(2 font-lock-function-name-face append))
;; Anchor on the function keyword, highlight params
(list (concat "^\\s-*function\\>[ \t\n.]*"
"\\(\\(\\[[^]]*\\]\\|\\sw+\\)[ \t\n.]*=[ \t\n.]*\\)?"
"\\sw+\\s-*(")
'("\\s-*\\(\\sw+\\)\\s-*[,)]" nil nil
2005-12-01 20:08:16 +01:00
(1 font-lock-variable-name-face)))
;; I like variables for FOR loops
'("\\<\\(for\\|parfor\\)\\s-+\\(\\sw+\\)\\s-*=\\s-*\
2005-12-01 20:08:16 +01:00
\\(\\([^\n,;%(]+\\|([^\n%)]+)\\)+\\)"
(1 font-lock-keyword-face)
(2 font-lock-variable-name-face append)
(3 font-lock-reference-face append))
;; Items after a switch statements are cool
'("\\<\\(case\\|switch\\)\\s-+\\({[^}\n]+}\\|[^,%\n]+\\)"
(1 font-lock-keyword-face) (2 font-lock-reference-face))
;; How about a few matlab constants such as pi, infinity, and sqrt(-1)?
;; The ^>> is in case I use this in an interactive mode someday
'("\\<\\(eps\\|pi\\|inf\\|Inf\\|NaN\\|nan\\|ans\\|i\\|j\\|^>>\\)\\>"
1 font-lock-reference-face)
'("\\<[0-9]\\.?\\(i\\|j\\)\\>" 1 font-lock-reference-face)
;; Define these as variables since this is about as close
;; as matlab gets to variables
(list (concat "\\<" matlab-indent-past-arg1-functions "\\s-*")
'("(\\s-*\\(\\w+\\)\\s-*\\(,\\|)\\)" nil nil
2005-12-01 20:08:16 +01:00
(1 font-lock-variable-name-face)))
))
"Expressions to highlight in MATLAB mode.")
(defvar matlab-really-gaudy-font-lock-keywords
(append
matlab-gaudy-font-lock-keywords
(list
;; Since it's a math language, how bout dem symbols?
'("\\([<>~]=?\\|\\.[/*^']\\|==\\|\\<xor\\>\\|[-!^&|*+\\/~:]\\)"
1 font-lock-type-face)
;; How about references in the HELP text.
(list (concat "^" matlab-comment-line-s "\\s-*"
"\\(\\([A-Z]+\\s-*=\\s-+\\|\\[[^]]+]\\s-*=\\s-+\\|\\)"
"\\([A-Z][0-9A-Z]+\\)\\(([^)\n]+)\\| \\)\\)")
'(1 font-lock-reference-face prepend))
(list (concat "^" matlab-comment-line-s "\\s-*"
"See also\\s-+")
'("\\([A-Z][A-Z0-9]+\\)\\([,.]\\| and\\|$\\) *" nil nil
2005-12-01 20:08:16 +01:00
(1 font-lock-reference-face prepend)))
(list (concat "^" matlab-comment-line-s "\\s-*"
"\\(\\$" "Revision" "[^\n$]+\\$\\)")
'(1 font-lock-reference-face prepend))
;; continuation ellipsis.
'("[^.]\\(\\.\\.\\.+\\)\\([^\n]*\\)" (1 'underline)
(2 font-lock-comment-face))
;; How about debugging statements?
;;'("\\<\\(db\\sw+\\)\\>" 1 'bold)
(list
(if (fboundp 'regexp-opt)
(concat "\\<\\(" (regexp-opt matlab-debug-list) "\\)\\>")
;; pre-regexp-opt days.
"\\<\\(db\\(c\\(lear\\|ont\\)\\|down\\|mex\\|quit\\|\
st\\(a\\(ck\\|tus\\)\\|ep\\|op\\)\\|type\\|up\\)\\)\\>")
'(0 'bold)))
(if matlab-handle-simulink
;; Simulink functions, but only if the user wants it.
(list (list (concat "\\<\\(\\([sg]et_param\\|sim\\([gs]et\\)?\\|"
"\\(mld\\|ss\\)[A-Z]\\w+\\)\\|"
"\\(new\\|open\\|close\\|save\\|find\\)_system\\|"
"\\(add\\|delete\\|replace\\)_\\(block\\|line\\)\\|"
"simulink\\|bd\\(root\\|close\\)"
"\\)\\>")
1 matlab-simulink-keyword-face))
nil))
"Expressions to highlight in MATLAB mode.")
(defvar matlab-shell-font-lock-keywords
(list
;; How about Errors?
'("^\\(Error in\\|Syntax error in\\)\\s-+==>\\s-+\\(.+\\)$"
(1 font-lock-comment-face) (2 font-lock-string-face))
;; and line numbers
'("^\\(\\(On \\)?line [0-9]+\\)" 1 font-lock-comment-face)
2005-12-01 20:08:16 +01:00
;; User beep things
'("\\(\\?\\?\\?[^\n]+\\)" 1 font-lock-comment-face)
;; Useful user commands, but not useful programming constructs
'("\\<\\(demo\\|whatsnew\\|info\\|subscribe\\|help\\|doc\\|lookfor\\|what\
\\|whos?\\|cd\\|clear\\|load\\|save\\|helpdesk\\|helpwin\\)\\>"
1 font-lock-keyword-face)
;; Various notices
'(" M A T L A B " 0 'underline)
'("All Rights Reserved" 0 'italic)
'("\\((c)\\s-+Copyright[^\n]+\\)" 1 font-lock-comment-face)
'("\\(Version\\)\\s-+\\([^\n]+\\)"
(1 font-lock-function-name-face) (2 font-lock-variable-name-face))
)
"Additional keywords used by MATLAB when reporting errors in interactive\
mode.")
;; Imenu support.
2005-12-01 20:08:16 +01:00
(defvar matlab-imenu-generic-expression
'((nil "^\\s-*function\\>[ \t\n.]*\\(\\(\\[[^]]*\\]\\|\\sw+\\)[ \t\n.]*\
< =\[ \t\n.]*\\)?\\([a-zA-Z0-9_]+\\)" 3))
2005-12-01 20:08:16 +01:00
"Expressions which find function headings in MATLAB M files.")
;;; MATLAB mode entry point ==================================================
;;;###autoload
(add-to-list 'auto-mode-alist '("\\.m$" . matlab-mode))
(declare-function mlint-minor-mode "mlint.el")
(declare-function mlint-buffer "mlint.el")
(declare-function mlint-clear-warnings "mlint.el")
(declare-function mlint-clear-cross-function-variable-highlighting "mlint.el")
2005-12-01 20:08:16 +01:00
;;;###autoload
(defun matlab-mode ()
"MATLAB(R) mode is a major mode for editing MATLAB dot-m files.
2005-12-01 20:08:16 +01:00
\\<matlab-mode-map>
Convenient editing commands are:
\\[matlab-comment-region] - Comment/Uncomment out a region of code.
\\[matlab-fill-comment-line] - Fill the current comment line.
\\[matlab-fill-region] - Fill code and comments in region.
\\[matlab-fill-paragraph] - Refill the current command or comment.
\\[matlab-complete-symbol] - Symbol completion of matlab symbols\
based on the local syntax.
\\[matlab-indent-sexp] - Indent syntactic block of code.
2005-12-01 20:08:16 +01:00
Convenient navigation commands are:
\\[matlab-beginning-of-command] - Move to the beginning of a command.
\\[matlab-end-of-command] - Move to the end of a command.
\\[matlab-beginning-of-defun] - Move to the beginning of a function.
\\[matlab-end-of-defun] - Move do the end of a function.
\\[matlab-forward-sexp] - Move forward over a syntactic block of code.
\\[matlab-backward-sexp] - Move backwards over a syntactic block of code.
Convenient template insertion commands:
\\[tempo-template-matlab-function] - Insert a function definition.
\\[tempo-template-matlab-if] - Insert an IF END block.
\\[tempo-template-matlab-for] - Insert a FOR END block.
\\[tempo-template-matlab-switch] - Insert a SWITCH END statement.
\\[matlab-insert-next-case] - Insert the next CASE condition in a SWITCH.
\\[matlab-insert-end-block] - Insert a matched END statement. With \
optional ARG, reindent.
\\[matlab-stringify-region] - Convert plain text in region to a string \
2005-12-01 20:08:16 +01:00
with correctly quoted chars.
Variables:
`matlab-indent-level' Level to indent blocks.
`matlab-cont-level' Level to indent continuation lines.
`matlab-cont-requires-ellipsis' Does your MATLAB support implied elipsis.
`matlab-case-level' Level to unindent case statements.
`matlab-indent-past-arg1-functions'
Regexp of functions to indent past the first
argument on continuation lines.
`matlab-maximum-indents' List of maximum indents during lineups.
`matlab-comment-column' Goal column for on-line comments.
`fill-column' Column used in auto-fill.
`matlab-indent-function-body' If non-nil, indents body of MATLAB functions.
`matlab-functions-have-end' If non-nil, MATLAB functions terminate with end.
`matlab-return-function' Customize RET handling with this function.
`matlab-auto-fill' Non-nil, do auto-fill at startup.
`matlab-fill-code' Non-nil, auto-fill code.
`matlab-fill-strings' Non-nil, auto-fill strings.
`matlab-verify-on-save-flag' Non-nil, enable code checks on save.
`matlab-highlight-block-match-flag'
Enable matching block begin/end keywords.
`matlab-vers-on-startup' If t, show version on start-up.
`matlab-handle-simulink' If t, enable simulink keyword highlighting.
All Key Bindings:
\\{matlab-mode-map}"
(interactive)
(kill-all-local-variables)
(use-local-map matlab-mode-map)
(setq major-mode 'matlab-mode)
(setq mode-name "MATLAB")
(if (boundp 'whitespace-modes)
(add-to-list 'whitespace-modes 'matlab-mode))
(setq local-abbrev-table matlab-mode-abbrev-table)
(set-syntax-table matlab-mode-syntax-table)
(setq indent-tabs-mode nil)
(make-local-variable 'indent-line-function)
(setq indent-line-function 'matlab-indent-line)
(make-local-variable 'paragraph-start)
(setq paragraph-start (concat "^$\\|" page-delimiter))
(make-local-variable 'paragraph-separate)
(setq paragraph-separate paragraph-start)
(make-local-variable 'paragraph-ignore-fill-prefix)
(setq paragraph-ignore-fill-prefix t)
(make-local-variable 'comment-start-skip)
(setq comment-start-skip "%\\s-+")
(make-local-variable 'comment-start)
(setq comment-start "%")
(make-local-variable 'page-delimiter)
(setq page-delimiter "^\\(\f\\|%%\\(\\s-\\|\n\\)\\)")
2005-12-01 20:08:16 +01:00
(make-local-variable 'comment-column)
(setq comment-column matlab-comment-column)
(make-local-variable 'comment-indent-function)
(setq comment-indent-function 'matlab-comment-indent)
(make-local-variable 'add-log-current-defun-function)
(setq add-log-current-defun-function 'matlab-current-defun)
(make-local-variable 'fill-column)
(setq fill-column matlab-fill-column)
2005-12-01 20:08:16 +01:00
(make-local-variable 'auto-fill-function)
(if matlab-auto-fill (setq auto-fill-function 'matlab-auto-fill))
;; Emacs 20 supports this variable. This lets users turn auto-fill
;; on and off and still get the right fill function.
(make-local-variable 'normal-auto-fill-function)
(setq normal-auto-fill-function 'matlab-auto-fill)
(make-local-variable 'fill-prefix)
(make-local-variable 'imenu-generic-expression)
(setq imenu-generic-expression matlab-imenu-generic-expression)
;; Save hook for verifying src. This lets us change the name of
;; the function in `write-file' and have the change be saved.
;; It also lets us fix mistakes before a `save-and-go'.
(make-local-variable 'write-contents-hooks)
(add-hook 'write-contents-hooks 'matlab-mode-verify-fix-file-fn)
;; Tempo tags
(make-local-variable 'tempo-local-tags)
(setq tempo-local-tags (append matlab-tempo-tags tempo-local-tags))
;; give each file it's own parameter history
(make-local-variable 'matlab-shell-save-and-go-history)
(make-local-variable 'font-lock-defaults)
(setq font-lock-defaults '((matlab-font-lock-keywords
matlab-gaudy-font-lock-keywords
matlab-really-gaudy-font-lock-keywords
)
t ; do not do string/comment highlighting
nil ; keywords are case sensitive.
;; This puts _ as a word constituent,
;; simplifying our keywords significantly
((?_ . "w"))))
(matlab-enable-block-highlighting 1)
(if window-system (matlab-frame-init))
;; If first function is terminated with an end statement, then functions have
;; ends.
(if (matlab-do-functions-have-end-p)
(matlab-functions-have-end-minor-mode 1)
(matlab-functions-have-end-minor-mode -1)
)
;; When matlab-indent-function-body is set to 'MathWorks-Standard,
;; - we indent all functions that terminate with an end statement
;; - old style functions (those without end statements) are not
;; indented.
;; It is desired that all code be terminate with an end statement.
;;
;; When matlab-indent-function-body is set to 'guess,
;; - look at the first line of code and if indented, keep indentation
;; otherwise use MathWorks-Standard
;;
(cond
((eq matlab-indent-function-body 'MathWorks-Standard)
)
((eq matlab-indent-function-body 'guess)
(save-excursion
(goto-char (point-max))
(if (re-search-backward matlab-defun-regex nil t)
(let ((beg (point))
end ; filled in later
(cc (current-column))
)
(setq end (if matlab-functions-have-end
(progn (forward-line 0) (point))
(point-max)))
(goto-char beg)
(catch 'done
(while (progn (forward-line 1) (< (point) end))
(if (looking-at "\\s-*\\(%\\|$\\)")
nil ; go on to next line
(looking-at "\\s-*")
(goto-char (match-end 0))
(setq matlab-indent-function-body (> (current-column) cc))
(throw 'done nil))))
)
(setq matlab-indent-function-body 'MathWorks-Standard)
))
)
(t)
)
2005-12-01 20:08:16 +01:00
(if (or (featurep 'mlint)
matlab-show-mlint-warnings
matlab-highlight-cross-function-variables)
;; Some users may not feel like getting all the extra stuff
;; needed for mlint working. Do this only if we can get
;; mlint loaded ok.
(condition-case nil
(mlint-minor-mode
(if (or matlab-show-mlint-warnings matlab-highlight-cross-function-variables)
1
0))
;; If there is an error loading the stuff, don't
;; continue.
(error nil)))
(save-excursion
(goto-char (point-min))
(run-hooks 'matlab-mode-hook))
2005-12-01 20:08:16 +01:00
(if matlab-vers-on-startup (matlab-show-version)))
;;; Utilities =================================================================
(defun matlab-show-version ()
"Show the version number in the minibuffer."
(interactive)
(message "matlab-mode, version %s" matlab-mode-version))
(defun matlab-find-prev-line ()
"Recurse backwards until a code line is found."
(if (= -1 (forward-line -1)) nil
(if (or (matlab-ltype-empty)
(matlab-ltype-comm-ignore))
(matlab-find-prev-line) t)))
(defun matlab-prev-line ()
"Go to the previous line of code. Return nil if not found."
(interactive)
(let ((old-point (point)))
(if (matlab-find-prev-line) t (goto-char old-point) nil)))
(defun matlab-uniquafy-list (lst)
"Return a list that is a subset of LST where all elements are unique."
(let ((nlst nil))
(while lst
(if (and (car lst) (not (member (car lst) nlst)))
(setq nlst (cons (car lst) nlst)))
(setq lst (cdr lst)))
(nreverse nlst)))
; Aki Vehtari <Aki.Vehtari@hut.fi> recommends this: (19.29 required)
;(require 'backquote)
;(defmacro matlab-navigation-syntax (&rest body)
; "Evaluate BODY with the matlab-mode-special-syntax-table"
; '(let ((oldsyntax (syntax-table)))
; (unwind-protect
; (progn
; (set-syntax-table matlab-mode-special-syntax-table)
; ,@body)
; (set-syntax-table oldsyntax))))
(defmacro matlab-navigation-syntax (&rest forms)
"Set the current environment for syntax-navigation and execute FORMS."
(list 'let '((oldsyntax (syntax-table))
(case-fold-search nil))
(list 'unwind-protect
(list 'progn
'(set-syntax-table matlab-mode-special-syntax-table)
(cons 'progn forms))
'(set-syntax-table oldsyntax))))
(put 'matlab-navigation-syntax 'lisp-indent-function 0)
(add-hook 'edebug-setup-hook
(lambda ()
(def-edebug-spec matlab-navigation-syntax def-body)))
(defun matlab-up-list (count &optional restrict)
"Move forwards or backwards up a list by COUNT.
Optional argument RESTRICT is where to restrict the search."
;; MATLAB syntax table has no disabling strings or comments.
(let ((dir (if (> 0 count) -1 +1))
(origin (point))
(ms nil))
;; Make count positive
(setq count (* count dir))
(if (= dir -1)
(while (/= count 0)
;; Search till we find an unstrung paren object.
(setq ms (re-search-backward "\\s(\\|\\s)" restrict t))
(while (and (save-match-data (matlab-cursor-in-string-or-comment))
(setq ms (re-search-backward "\\s(\\|\\s)" restrict t))))
(if (not ms)
(progn
(goto-char origin)
(error "Scan Error: List mismatch")))
2005-12-01 20:08:16 +01:00
;; View it's match.
(let ((s (match-string 0)))
(if (string-match "\\s(" s)
(setq count (1- count))
(setq count (1+ count)))))
(error "Not implemented"))
ms))
(defun matlab-valid-end-construct-p ()
"Return non-nil if the end after point terminates a block.
Return nil if it is being used to dereference an array."
(let ((p (point))
(err1 t))
(condition-case nil
(save-restriction
;; Restrict navigation only to the current command line
(save-excursion
(matlab-beginning-of-command)
(narrow-to-region (point)
(save-excursion
(goto-char p)
(matlab-point-at-eol))))
;; This used to add some sort of protection, but I don't know what
;; the condition was, or why the simple case doesn't handle it.
;;
;; The above replacement fixes a case where a continuation in an array
;; befuddles the identifier.
;; (progn ;;(matlab-end-of-command (point))
;; (end-of-line)
;; (if (> p (point))
;; (progn
;; (setq err1 nil)
;; (error)))
;; (point))))
2005-12-01 20:08:16 +01:00
(save-excursion
;; beginning of param list
(matlab-up-list -1)
;; backup over the parens. If that fails
(condition-case nil
(progn
(forward-sexp 1)
;; If we get here, the END is inside parens, which is not a
;; valid location for the END keyword. As such it is being
;; used to dereference array parameters
nil)
;; This error means that we have an unterminated paren
;; block, so this end is currently invalid.
(error nil))))
;; an error means the list navigation failed, which also means we are
;; at the top-level
(error err1))))
;;; Regexps for MATLAB language ===============================================
;; "-pre" means "partial regular expression"
;; "-if" and "-no-if" means "[no] Indent Function"
(defconst matlab-defun-regex "^\\(\\s-*function\\|classdef\\)[ \t.[]"
2005-12-01 20:08:16 +01:00
"Regular expression defining the beginning of a MATLAB function.")
2008-10-17 21:09:19 +02:00
(defconst matlab-mcos-regexp "\\|classdef\\|properties\\|methods\\|enumeration"
"Keywords which mark the beginning of mcos blocks.")
(defcustom matlab-block-indent-tic-toc-flag nil
"*Non-nil means that tic,toc should indent like a if,end block.
This variable should be set before loading matlab.el"
:group 'matlab
:type 'boolean)
(defconst matlab-block-beg-pre-if
(if matlab-block-indent-tic-toc-flag
(concat "function\\|parfor\\|spmd\\|for\\|while\\|if\\|switch\\|try\\|tic"
matlab-mcos-regexp)
(concat "function\\|parfor\\|spmd\\|for\\|while\\|if\\|switch\\|try"
matlab-mcos-regexp))
2005-12-01 20:08:16 +01:00
"Keywords which mark the beginning of an indented block.
Includes function.")
(defconst matlab-block-beg-pre-no-if
(if matlab-block-indent-tic-toc-flag
(concat "parfor\\|for\\|spmd\\|while\\|if\\|switch\\|try\\|tic"
matlab-mcos-regexp)
(concat "parfor\\|for\\|spmd\\|while\\|if\\|switch\\|try"
matlab-mcos-regexp))
2005-12-01 20:08:16 +01:00
"Keywords which mark the beginning of an indented block.
Excludes function.")
(defun matlab-block-beg-pre ()
"Partial regular expression to recognize MATLAB block-begin keywords."
(if matlab-functions-have-end
matlab-block-beg-pre-if
matlab-block-beg-pre-no-if))
(defconst matlab-block-mid-pre
"elseif\\|else\\|catch"
"Partial regular expression to recognize MATLAB mid-block keywords.")
(defconst matlab-block-end-pre-if
(if matlab-block-indent-tic-toc-flag
"end\\(function\\)?\\|function\\|\\(\\sw+\\s-*\\((.*)\\)?\\s-*=\\s-*\\)?toc"
"end\\(function\\)?\\|function")
2005-12-01 20:08:16 +01:00
"Partial regular expression to recognize MATLAB block-end keywords.")
(defconst matlab-block-end-pre-no-if
(if matlab-block-indent-tic-toc-flag
"end\\|\\(\\sw+\\s-*\\((.*)\\)?\\s-*=\\s-*\\)?toc"
"end")
2005-12-01 20:08:16 +01:00
"Partial regular expression to recognize MATLAB block-end keywords.")
(defun matlab-block-end-pre ()
"Partial regular expression to recognize MATLAB block-end keywords."
(if matlab-functions-have-end
matlab-block-end-pre-if
matlab-block-end-pre-no-if))
;; Not used.
;;(defconst matlab-other-pre
;; "function\\|return"
;; "Partial regular express to recognize MATLAB non-block keywords.")
(defconst matlab-endless-blocks
"case\\|otherwise"
"Keywords which initialize new blocks, but don't have explicit ends.
Thus, they are endless. A new case or otherwise will end a previous
endless block, and end will end this block, plus any outside normal
2005-12-01 20:08:16 +01:00
blocks.")
(defun matlab-block-re ()
"Regular expression for keywords which begin MATLAB blocks."
(concat "\\(^\\|[;,]\\)\\s-*\\("
(matlab-block-beg-pre) "\\|"
matlab-block-mid-pre "\\|"
(matlab-block-end-pre) "\\|"
matlab-endless-blocks "\\)\\b"))
2005-12-01 20:08:16 +01:00
(defun matlab-block-scan-re ()
"Expression used to scan over matching pairs of begin/ends."
(concat "\\(^\\|[;,]\\)\\s-*\\("
(matlab-block-beg-pre) "\\|"
(matlab-block-end-pre) "\\)\\b"))
(defun matlab-block-beg-re ()
"Expression used to find the beginning of a block."
(concat "\\(" (matlab-block-beg-pre) "\\)"))
(defun matlab-block-mid-re ()
"Expression used to find block center parts (like else)."
(concat "\\(" matlab-block-mid-pre "\\)"))
(defun matlab-block-end-re ()
"Expression used to end a block. Usually just `end'."
(concat "\\(" (matlab-block-end-pre) "\\)"))
(defun matlab-block-end-no-function-re ()
"Expression representing and end if functions are excluded."
(concat "\\<\\(" matlab-block-end-pre-no-if "\\)\\>"))
(defun matlab-endless-blocks-re ()
"Expression of block starters that do not have associated ends."
(concat "\\(" matlab-endless-blocks "\\)"))
(defun matlab-match-function-re ()
"Expression to match a function start line.
There are no reliable numeric matches in this expression.
Know that `match-end' of 0 is the end of the function name."
2005-12-01 20:08:16 +01:00
;; old function was too unstable.
;;"\\(^function\\s-+\\)\\([^=\n]+=[ \t\n.]*\\)?\\(\\sw+\\)"
(concat "\\(^\\s-*function\\b[ \t\n.]*\\)\\(\\(\\[[^]]*\\]\\|\\sw+\\)"
"[ \t\n.]*=[ \t\n.]*\\|\\(\\)\\)\\(\\sw+\\)"))
(defun matlab-match-classdef-re ()
"Expression to match a classdef start line.
The class name is match 2."
"\\(^\\s-*classdef\\b[ \t\n]*\\)\\(\\sw+\\)\\(\\s-*<\\)?")
2005-12-01 20:08:16 +01:00
(defconst matlab-cline-start-skip "[ \t]*%[ \t]*"
"*The regular expression for skipping comment start.")
;;; Lists for matlab keywords =================================================
(defvar matlab-keywords-solo
'("break" "case" "else" "elseif" "end" "for" "parfor" "function" "if" "tic" "toc"
"otherwise" "profile" "switch" "while" "try" "catch" "spmd")
2005-12-01 20:08:16 +01:00
"Keywords that appear on a line by themselves.")
(defvar matlab-keywords-return
'("acos" "acosh" "acot" "acoth" "acsch" "asech" "asin" "asinh"
"atan" "atan2" "atanh" "cos" "cosh" "coth" "csc" "csch" "exp"
"log" "log10" "log2" "sec" "sech" "sin" "sinh" "tanh"
"abs" "sign" "sqrt" )
"List of MATLAB keywords that have return arguments.
This list still needs lots of help.")
(defvar matlab-keywords-boolean
'("all" "any" "exist" "isempty" "isequal" "ishold" "isfinite" "isglobal"
"isinf" "isletter" "islogical" "isnan" "isprime" "isreal" "isspace"
"logical" "isa")
"List of keywords that are typically used as boolean expressions.")
(defvar matlab-core-properties
'("ButtonDownFcn" "Children" "Clipping" "CreateFcn" "DeleteFcn"
"BusyAction" "HandleVisibility" "HitTest" "Interruptible"
"Parent" "Selected" "SelectionHighlight" "Tag" "Type"
"UIContextMenu" "UserData" "Visible")
"List of properties belonging to all HG objects.")
(defvar matlab-property-lists
'(("root" .
("CallbackObject" "Language" "CurrentFigure" "Diary" "DiaryFile"
"Echo" "ErrorMessage" "Format" "FormatSpacing" "PointerLocation"
"MonitorPositions"
"PointerWindow" "Profile" "ProfileFile" "ProfileCount"
"ProfileInterval" "RecursionLimit" "ScreenDepth" "ScreenSize"
"ShowHiddenHandles" "TerminalHideGraphCommand" "TerminalOneWindow"
"TerminalDimensions" "TerminalProtocol" "TerminalShowGraphCommand"
"Units" "AutomaticFileUpdates" ))
("axes" .
("AmbientLightColor" "Box" "CameraPosition" "CameraPositionMode"
"CameraTarget" "CameraTargetMode" "CameraUpVector"
"CameraUpVectorMode" "CameraViewAngle" "CameraViewAngleMode" "CLim"
"CLimMode" "Color" "CurrentPoint" "ColorOrder" "DataAspectRatio"
"DataAspectRatioMode" "DrawMode" "FontAngle" "FontName" "FontSize"
"FontUnits" "FontWeight" "GridLineStyle" "Layer" "LineStyleOrder"
"LineWidth" "NextPlot" "PlotBoxAspectRatio" "PlotBoxAspectRatioMode"
"Projection" "Position" "TickLength" "TickDir" "TickDirMode" "Title"
"Units" "View" "XColor" "XDir" "XGrid" "XLabel" "XAxisLocation" "XLim"
"XLimMode" "XScale" "XTick" "XTickLabel" "XTickLabelMode" "XTickMode"
"YColor" "YDir" "YGrid" "YLabel" "YAxisLocation" "YLim" "YLimMode"
"YScale" "YTick" "YTickLabel" "YTickLabelMode" "YTickMode" "ZColor"
"ZDir" "ZGrid" "ZLabel" "ZLim" "ZLimMode" "ZScale" "ZTick"
"ZTickLabel" "ZTickLabelMode" "ZTickMode"))
("figure" .
("BackingStore" "CloseRequestFcn" "Color" "Colormap"
"CurrentAxes" "CurrentCharacter" "CurrentObject" "CurrentPoint"
"Dithermap" "DithermapMode" "FixedColors" "IntegerHandle"
"InvertHardcopy" "KeyPressFcn" "MenuBar" "MinColormap" "Name"
"NextPlot" "NumberTitle" "PaperUnits" "PaperOrientation"
"PaperPosition" "PaperPositionMode" "PaperSize" "PaperType"
"Pointer" "PointerShapeCData" "PointerShapeHotSpot" "Position"
"Renderer" "RendererMode" "Resize" "ResizeFcn" "SelectionType"
"ShareColors" "Units" "WindowButtonDownFcn"
"WindowButtonMotionFcn" "WindowButtonUpFcn" "WindowStyle"))
("image" . ("CData" "CDataMapping" "EraseMode" "XData" "YData"))
("light" . ("Position" "Color" "Style"))
("line" .
("Color" "EraseMode" "LineStyle" "LineWidth" "Marker" "LineSmoothing"
"MarkerSize" "MarkerEdgeColor" "MarkerFaceColor" "XData" "YData"
"ZData"))
("patch" .
("CData" "CDataMapping" "FaceVertexCData" "EdgeColor" "EraseMode"
"FaceColor" "Faces" "LineStyle" "LineWidth" "Marker" "LineSmoothing"
"MarkerEdgeColor" "MarkerFaceColor" "MarkerSize" "Vertices"
"XData" "YData" "ZData" "FaceLighting" "EdgeLighting"
"BackFaceLighting" "AmbientStrength" "DiffuseStrength"
"SpecularStrength" "SpecularExponent" "SpecularColorReflectance"
"VertexNormals" "NormalMode"))
("surface" .
("CData" "CDataMapping" "EdgeColor" "EraseMode" "FaceColor"
"LineStyle" "LineWidth" "Marker" "MarkerEdgeColor" "LineSmoothing"
"MarkerFaceColor" "MarkerSize" "MeshStyle" "XData" "YData"
"ZData" "FaceLighting" "EdgeLighting" "BackFaceLighting"
"AmbientStrength" "DiffuseStrength" "SpecularStrength"
"SpecularExponent" "SpecularColorReflectance" "VertexNormals"
"NormalMode"))
("text\\|title\\|xlabel\\|ylabel\\|zlabel" .
("Color" "EraseMode" "Editing" "Extent" "FontAngle" "FontName"
"FontSize" "FontUnits" "FontWeight" "HorizontalAlignment"
"BackgroundColor" "EdgeColor" "Margin"
"Position" "Rotation" "String" "Units" "Interpreter"
"VerticalAlignment"))
("uicontextmenu" . ("Callback"))
("uicontrol" .
("BackgroundColor" "Callback" "CData" "Enable" "Extent"
"FontAngle" "FontName" "FontSize" "FontUnits" "FontWeight"
"ForegroundColor" "HorizontalAlignment" "ListboxTop" "Max" "Min"
"Position" "String" "Style" "SliderStep" "TooltipString" "Units"
"Value"))
("uimenu" .
("Accelerator" "Callback" "Checked" "Enable" "ForegroundColor"
"Label" "Position" "Separator"))
;; Flesh this out more later.
("uipushtool\\|uitoggletool\\|uitoolbar" .
("Cdata" "Callback" "Separator" "Visible"))
)
"List of property lists on a per object type basis.")
(defvar matlab-unknown-type-commands
"[gs]et\\|findobj\\|waitfor"
"Expression for commands that have unknown types.")
(defun matlab-all-known-properties ()
"Return a list of all properties."
(let ((lst matlab-core-properties)
(tl matlab-property-lists))
(while tl
(setq lst (append lst (cdr (car tl)))
tl (cdr tl)))
(matlab-uniquafy-list lst)))
(defvar matlab-all-known-properties (matlab-all-known-properties)
"List of all the known properties.")
(defmacro matlab-property-function ()
"Regexp of all builtin functions that take property lists."
'(let ((r matlab-unknown-type-commands)
(tl matlab-property-lists))
(while tl
(setq r (concat r "\\|" (car (car tl)))
tl (cdr tl)))
r))
;;; Navigation ===============================================================
(defvar matlab-scan-on-screen-only nil
"When this is set to non-nil, then forward/backward sexp stops off screen.
This is so the block highlighter doesn't gobble up lots of time when
a block is not terminated.")
(defun matlab-backward-sexp (&optional autoend noerror)
"Go backwards one balanced set of MATLAB expressions.
If optional AUTOEND, then pretend we are at an end.
If optional NOERROR, then we return t on success, and nil on failure.
This assumes that expressions do not cross \"function\" at the left margin."
(interactive "P")
(matlab-navigation-syntax
(if (and (not autoend)
(save-excursion (backward-word 1)
(or (not
(and (looking-at
(matlab-block-end-no-function-re))
(matlab-valid-end-construct-p)))
(matlab-cursor-in-string-or-comment))))
;; Go backwards one simple expression
(forward-sexp -1)
;; otherwise go backwards recursively across balanced expressions
;; backup over our end
(if (not autoend) (forward-word -1))
(let ((done nil) (start (point)) (returnme t) (bound nil))
(when (search-backward "\nfunction" nil t)
(if (progn (forward-char 9) (looking-at "\\b"))
(setq bound (- (point) 8)))
(goto-char start))
(while (and (not done)
(or (not matlab-scan-on-screen-only)
(pos-visible-in-window-p)))
(if (re-search-backward (matlab-block-scan-re) bound t)
(progn
(goto-char (match-beginning 2))
(if (looking-at (matlab-block-end-no-function-re))
(if (or (matlab-cursor-in-string-or-comment)
(not (matlab-valid-end-construct-p)))
nil
;; we must skip the expression and keep searching
(forward-word 1)
(matlab-backward-sexp))
(if (not (matlab-cursor-in-string-or-comment))
(setq done t))))
(goto-char start)
(if noerror
(setq returnme nil)
(error "Unstarted END construct"))))
returnme))))
2005-12-01 20:08:16 +01:00
(defun matlab-forward-sexp (&optional includeelse)
"Go forward one balanced set of MATLAB expressions.
Optional argument INCLUDEELSE will stop on ELSE if it matches the starting IF."
(interactive "P")
(let (p) ;; go to here if no error.
(save-excursion ;; don't go anywhere if there is an error
(matlab-navigation-syntax
;; skip over preceding whitespace
2005-12-01 20:08:16 +01:00
(skip-chars-forward " \t\n;")
(if (or (not (looking-at (concat "\\("
(matlab-block-beg-pre)
"\\|"
(matlab-block-mid-re)
"\\)\\>")))
(matlab-cursor-in-string-or-comment))
;; Go forwards one simple expression
(forward-sexp 1)
;; otherwise go forwards recursively across balanced expressions
(forward-word 1)
(let ((done nil) (s nil)
(expr-scan (if includeelse
(matlab-block-re)
(matlab-block-scan-re)))
(expr-look (matlab-block-beg-pre)))
(while (and (not done)
(setq s (re-search-forward expr-scan nil t))
(or (not matlab-scan-on-screen-only)
(pos-visible-in-window-p)))
(goto-char (match-beginning 2))
(if (looking-at expr-look)
(if (matlab-cursor-in-string-or-comment)
(forward-word 1)
;; we must skip the expression and keep searching
;; NEVER EVER call with value of INCLUDEELSE
(matlab-forward-sexp))
(forward-word 1)
(if (and (not (matlab-cursor-in-string-or-comment))
(matlab-valid-end-construct-p))
(setq done t))))
(if (not s)
(error "Unterminated block"))))
(setq p (point)))) ;; really go here
(goto-char p)))
(defun matlab-indent-sexp ()
"Indent the syntactic block starting at point."
(interactive)
(indent-region (point) (save-excursion (matlab-forward-sexp) (point)) nil))
(defun matlab-beginning-of-enclosing-defun ()
"Move cursor to beginning of enclosing function.
If `matlab-functions-have-end', skip over functions with end."
(catch 'done
(let ((start (point))
(beg nil))
(while (re-search-backward matlab-defun-regex nil t)
(setq beg (point))
(condition-case nil
(progn
(matlab-forward-sexp)
(if (> (point) start) (throw 'done beg)))
(error (throw 'done beg)))
(goto-char beg)))
nil))
(defun matlab-beginning-of-defun ()
"Go to the beginning of the current function."
(interactive)
(if matlab-functions-have-end
(goto-char (or (matlab-beginning-of-enclosing-defun) (point-min)))
(or (re-search-backward matlab-defun-regex nil t)
(goto-char (point-min)))))
(defun matlab-end-of-defun ()
"Go to the end of the current function."
(interactive)
(or (progn
(if (looking-at matlab-defun-regex) (goto-char (match-end 0)))
(if (re-search-forward matlab-defun-regex nil t)
(progn (forward-line -1)
t)))
(goto-char (point-max))))
(defun matlab-current-defun ()
"Return the name of the current function."
(save-excursion
(matlab-beginning-of-defun)
(if (looking-at (matlab-match-function-re))
(progn
(goto-char (match-end 0))
(current-word)))))
(defun matlab-beginning-of-command ()
"Go to the beginning of an M command.
Travels across continuations."
(interactive)
(beginning-of-line)
(let ((p nil)
;; This restriction is a wild guess where to end reverse
;; searching for array continuations. The reason is that
;; matlab up list is very slow, and most people would never
;; put a blank line in a matrix. Either way, it's worth the
;; trade off to speed this up for large files.
;; This list of keywords is NOT meant to be comprehensive.
(r (save-excursion
(re-search-backward
"^\\s-*\\(%\\|if\\|else\\(if\\)\\|while\\|\\(par\\)?for\\|$\\)\\>"
2005-12-01 20:08:16 +01:00
nil t))))
(while (and (or (save-excursion (and (matlab-prev-line)
(matlab-lattr-cont)))
(matlab-ltype-continued-comm)
(setq p (matlab-lattr-array-cont r)))
(save-excursion (beginning-of-line) (not (bobp))))
(if p (goto-char p) (matlab-prev-line))
(setq p nil))
(back-to-indentation)))
(defun matlab-end-of-command (&optional beginning)
"Go to the end of an M command.
Optional BEGINNING is where the command starts from."
(interactive)
(while (and (or (matlab-lattr-cont)
(save-excursion
(forward-line 1)
(or (matlab-ltype-continued-comm)
(matlab-lattr-array-cont beginning))))
;; This hack is a short circuit. If a user did not
;; correctly end a matrix, this will short-circuit
;; as soon as something that would never appear in a matrix
2005-12-01 20:08:16 +01:00
;; becomes visible.
(not (save-excursion
(beginning-of-line)
(looking-at (matlab-block-scan-re))))
;; If we hit the end of the buffer unexpectedly, this test
;; will fail and we'll avoid looping forever. (E.g., this
;; is triggered if a continuation line is the last one in
;; the buffer, and the line lacks the final newline.)
(zerop (forward-line 1))))
(end-of-line))
;;; Line types and attributes =================================================
(defun matlab-ltype-empty () ; blank line
"Return t if current line is empty."
(save-excursion
(beginning-of-line)
(looking-at "^[ \t]*$")))
(defun matlab-ltype-comm () ; comment line
"Return t if current line is a MATLAB comment line.
Return the symbol 'cellstart if it is a double %%.
Return the symbol 'blockcomm if it is a block comment start."
2005-12-01 20:08:16 +01:00
(save-excursion
(beginning-of-line)
(cond ((looking-at "[ \t]*%\\([^%]\\|$\\)")
t)
((looking-at "[ \t]*%%")
'cellstart)
((matlab-ltype-block-comm)
'blockcomm)
(t nil))))
2005-12-01 20:08:16 +01:00
(defun matlab-ltype-comm-ignore () ; comment out a region line
"Return t if current line is a MATLAB comment region line."
(save-excursion
(beginning-of-line)
(looking-at (concat "[ \t]*" matlab-comment-region-s))))
(defun matlab-ltype-help-comm ()
"Return t if the current line is part of the MATLAB help comment."
(save-excursion
(if (not (matlab-ltype-comm))
nil
(while (and (matlab-ltype-comm) (not (bobp))
(matlab-prev-line))
(beginning-of-line))
(matlab-ltype-function-definition))))
(defun matlab-ltype-block-comm ()
"Return t if we are in a block comment."
(save-excursion
(if (looking-at "%{")
t
(when (re-search-backward "\\%\\([{}]\\)" nil t)
(let ((ms (match-string 1)))
(if (string= ms "{") t nil))))))
2005-12-01 20:08:16 +01:00
(defun matlab-ltype-endfunction-comm ()
"Return t if the current line is an ENDFUNCTION style comment."
(save-excursion
(if (not (matlab-ltype-comm))
nil
(beginning-of-line)
(if (looking-at "^[ \t]*%[ \t]*endfunction")
t
(while (and (or (matlab-ltype-comm)
(matlab-ltype-empty))
(not (eobp)))
(forward-line 1))
(and (matlab-ltype-function-definition)
(not (save-excursion (matlab-beginning-of-enclosing-defun))))
))))
2005-12-01 20:08:16 +01:00
(defun matlab-ltype-continued-comm ()
"Return column of previous line's comment start, or nil."
(save-excursion
(beginning-of-line)
(let ((commtype (matlab-ltype-comm)))
(if (or (eq commtype 'cellstart) ;; Cells are not continuations from previous comments.
(null commtype)
(bobp))
nil
;; We use forward-line and not matlab-prev-line because
;; we want blank lines to terminate this indentation method.
(forward-line -1)
(let ((col (matlab-lattr-comm)))
(if col
(progn
(goto-char col)
(current-column))
nil))))))
2005-12-01 20:08:16 +01:00
(defun matlab-ltype-function-definition ()
"Return t if the current line is a function definition."
(save-excursion
(beginning-of-line)
(looking-at matlab-defun-regex)))
(defun matlab-ltype-code () ; line of code
"Return t if current line is a MATLAB code line."
(and (not (matlab-ltype-empty)) (not (matlab-ltype-comm))))
(defun matlab-lattr-comm () ; line has comment
"Return t if current line contain a comment."
(save-excursion (matlab-comment-on-line)))
(defun matlab-lattr-implied-continuation ()
"Return non-nil if this line has implied continuation on the next.
This is only useful for new versions of MATLAB where ... is optional."
(when (not (matlab-lattr-comm))
(let ((imp nil))
(save-excursion
(end-of-line)
(skip-chars-backward " \t")
;; Test for operator incompleteness.
2005-12-01 20:08:16 +01:00
(setq imp
(/= (point)
;; Careful, - means range in this expression.
(progn (skip-chars-backward "-+=/*.^&~<>")
(point))))
(if (not imp)
;; Test for argument list incompleteness
(condition-case nil
(progn
(end-of-line)
(matlab-up-list -1)
(setq imp (looking-at "(")))
(error nil)))
)
imp)))
(defun matlab-lattr-cont () ; line has continuation
"Return non-nil if current line ends in ... and optional comment.
If `matlab-cont-requires-ellipsis' is nil, then we need to apply
a heuristic to determine if this line would use continuation
based on what it ends with."
(save-excursion
(beginning-of-line)
(or
;; Here, if the line ends in ..., then it is what we are supposed to do.
(and (re-search-forward "[^ \t.][ \t]*\\.\\.+[ \t]*\\(%.*\\)?$"
(matlab-point-at-eol) t)
(progn (goto-char (match-beginning 0))
(not (matlab-cursor-in-comment))))
;; If the line doesn't end in ..., but we have optional ..., then
;; use this annoying heuristic.
(and (null matlab-cont-requires-ellipsis)
(matlab-lattr-implied-continuation))
)))
(defun matlab-lattr-array-cont (&optional restrict)
"Return non-nil if current line is in an array.
If the entirety of the array is on this line, return nil.
Optional option RESTRICT is the distance to restrict the search."
2005-12-01 20:08:16 +01:00
(condition-case nil
(save-excursion
(beginning-of-line)
(matlab-up-list -1 restrict)
(and (looking-at "[[{]") (point)))
(error nil)))
(defun matlab-lattr-array-end ()
"Return non-nil if the current line closes an array.
by close, the first character is the end of an array."
(save-excursion
(back-to-indentation)
(and (looking-at "[]}]") (matlab-lattr-array-cont))))
(defun matlab-lattr-block-cont (&optional eol)
"Return a number representing the number of unterminated block constructs.
This is any block, such as if or for, that doesn't have an END on this line.
Optional EOL indicates a virtual end of line."
(let ((v 0))
(save-excursion
(beginning-of-line)
(save-restriction
(narrow-to-region (point) (or eol (matlab-point-at-eol)))
(matlab-navigation-syntax
(while (re-search-forward (concat "\\<" (matlab-block-beg-re) "\\>")
nil t)
(if (matlab-cursor-in-string-or-comment)
;; Do nothing
nil
;; Increment counter, move to end.
(setq v (1+ v))
(let ((p (point)))
(forward-word -1)
(condition-case nil
(progn
(matlab-forward-sexp)
(setq v (1- v)))
(error (goto-char p))))))
v)))))
(defun matlab-lattr-middle-block-cont ()
"Return the number of middle block continuations.
This should be 1 or nil, and only true if the line starts with one of these
special items."
(save-excursion
(back-to-indentation)
(if (looking-at (concat (matlab-block-mid-re) "\\>"))
(if (and (re-search-forward (matlab-block-end-pre)
(matlab-point-at-eol)
t)
(matlab-valid-end-construct-p))
;; If there is an END, we still need to return non-nil,
;; but the number value is a net of 0.
0
1)
nil)))
(defun matlab-lattr-endless-block-cont ()
"Return the number of middle block continuations.
This should be 1 or nil, and only true if the line starts with one of these
special items."
(save-excursion
(back-to-indentation)
(if (looking-at (concat (matlab-endless-blocks-re) "\\>"))
1
nil)))
(defun matlab-lattr-block-close (&optional start)
"Return the number of closing block constructs.
Argument START is where to start searching from."
2005-12-01 20:08:16 +01:00
(let ((v 0))
(save-excursion
(when start (goto-char start))
2005-12-01 20:08:16 +01:00
(save-restriction
(narrow-to-region (save-excursion
(matlab-beginning-of-command)
(point))
(matlab-point-at-eol))
(goto-char (point-max))
(while (and (re-search-backward (concat "\\<" (matlab-block-end-re) "\\>")
nil t)
(not (matlab-cursor-in-string-or-comment))
2005-12-01 20:08:16 +01:00
(matlab-valid-end-construct-p))
(setq v (1+ v))
(let ((startmove (match-end 0))
(nomove (point)))
(condition-case nil
(progn
(matlab-backward-sexp t)
(setq v (1- v)))
(error (goto-char nomove)))
))
;; If we can't scoot back, do a cheat-test to see if there
;; is a matching else or elseif.
(goto-char (point-min))
(back-to-indentation)
(if (looking-at (matlab-block-mid-re))
(setq v (1- v)))
;; Return nil, or a number
(if (<= v 0) nil v)))))
2005-12-01 20:08:16 +01:00
(defun matlab-lattr-local-end ()
"Return t if this line begins with an end construct."
(save-excursion
(back-to-indentation)
(and (looking-at (concat "\\<" (matlab-block-end-re) "\\>"))
(matlab-valid-end-construct-p))))
(defun matlab-lattr-semantics (&optional prefix)
"Return the semantics of the current position.
Values are nil 'solo, 'value, and 'boolean. Boolean is a subset of
value. nil means there is no semantic content (ie, string or comment.)
If optional PREFIX, then return 'solo if that is the only thing on the
line."
(cond ;((matlab-cursor-in-string-or-comment)
;nil)
((or (matlab-ltype-empty)
(and prefix (save-excursion
(beginning-of-line)
(looking-at (concat "\\s-*" prefix "\\s-*$")))))
'solo)
((save-excursion
(matlab-beginning-of-command)
(looking-at "\\s-*\\(if\\|elseif\\|while\\)\\>"))
'boolean)
((save-excursion
(matlab-beginning-of-command)
(looking-at (concat "\\s-*\\(" (matlab-property-function)
"\\)\\>")))
'property)
(t
'value)))
(defun matlab-function-called-at-point ()
"Return a string representing the function called nearby point."
(save-excursion
(beginning-of-line)
(cond ((looking-at "\\s-*\\([a-zA-Z]\\w+\\)[^=][^=]")
(match-string 1))
((and (re-search-forward "=" (matlab-point-at-eol) t)
(looking-at "\\s-*\\([a-zA-Z]\\w+\\)\\s-*[^=]"))
(match-string 1))
(t nil))))
(defun matlab-cursor-in-string-or-comment ()
"Return t if the cursor is in a valid MATLAB comment or string."
;; comment and string depend on each other. Here is one test
;; that does both.
(save-restriction
(narrow-to-region (matlab-point-at-bol) (matlab-point-at-eol))
(let ((p (1+ (point)))
(returnme nil)
(sregex (concat matlab-string-start-regexp matlab-string-char-regexp)))
2005-12-01 20:08:16 +01:00
(save-excursion
(goto-char (point-min))
(while (and (re-search-forward
(concat "'\\|%\\|" (regexp-quote matlab-elipsis-string))
nil t)
(<= (point) p))
(if (or (= ?% (preceding-char))
(= ?. (preceding-char)))
;; Here we are in a comment for the rest of it.
(progn
(goto-char p)
(setq returnme t))
;; Here, we could be a string start, or transpose...
(if (or (= (current-column) 1)
(save-excursion (forward-char -2)
(looking-at sregex)))
;; a valid string start, find the end
(let ((f (matlab-match-string-end-for-string)))
2005-12-01 20:08:16 +01:00
(if f
(setq returnme (> (point) p))
(setq returnme t)))
;; Ooops, a transpose, keep going.
))))
returnme)))
(defun matlab-cursor-in-comment ()
"Return t if the cursor is in a valid MATLAB comment."
(save-match-data
(save-restriction
(narrow-to-region (matlab-point-at-bol) (matlab-point-at-eol))
(save-excursion
(let ((prev-match nil))
(while (and (re-search-backward
(concat "%\\|" (regexp-quote matlab-elipsis-string) "+")
nil t)
(not (matlab-cursor-in-string)))
(setq prev-match (point)))
(if (and prev-match (matlab-cursor-in-string))
(goto-char prev-match))
(and (looking-at (concat "%\\|"
(regexp-quote matlab-elipsis-string)))
(not (matlab-cursor-in-string))))))))
(defun matlab-cursor-in-string (&optional incomplete)
"Return t if the cursor is in a valid MATLAB string.
If the optional argument INCOMPLETE is non-nil, then return t if we
are in what could be a an incomplete string."
(let ((m (match-data))
(returnme nil))
(save-restriction
(narrow-to-region (matlab-point-at-bol) (matlab-point-at-eol))
(let ((p (1+ (point)))
(sregex (concat matlab-string-start-regexp matlab-string-char-regexp))
2005-12-01 20:08:16 +01:00
(instring nil))
(save-excursion
;; Comment hunters need strings to not call the comment
;; identifiers. Thus, this routines must be savvy of comments
;; without recursing to them.
(goto-char (point-min))
(while (or (and instring (looking-at "'"))
(and (re-search-forward
(concat "'\\|%\\|"
(regexp-quote matlab-elipsis-string))
nil t)
(<= (point) p)
;; Short circuit to fix this.
(progn (setq instring nil) t)))
;; The next line emulates re-search-forward
2005-12-01 20:08:16 +01:00
(if instring (goto-char (match-end 0)))
(if (or (= ?% (preceding-char))
(= ?. (preceding-char)))
;; Here we are in a comment for the rest of it.
;; thus returnme is a force-false.
(goto-char p)
;; Here, we could be in a string start, or transpose...
(if (or (= (current-column) 1)
instring
(save-excursion (forward-char -2)
(looking-at sregex)))
;; a valid string start, find the end
(let ((f (matlab-match-string-end-for-string)))
2005-12-01 20:08:16 +01:00
(if (and (not f) incomplete)
(setq returnme t)
(setq returnme (> (point) p))
(setq instring t)))
;; Ooops, a transpose, keep going.
))))))
(set-match-data m)
returnme))
2005-12-01 20:08:16 +01:00
(defun matlab-comment-on-line ()
"Place the cursor on the beginning of a valid comment on this line.
If there isn't one, then return nil, point otherwise."
(interactive)
(let ((eol (matlab-point-at-eol))
(p (point))
(signal-error-on-buffer-boundary nil))
(beginning-of-line)
(while (and (re-search-forward "%" eol t)
(save-excursion (forward-char -1) (matlab-cursor-in-string t))))
(if (not (bolp)) (forward-char -1))
(if (and (looking-at "%") (not (matlab-cursor-in-string t)))
(point)
(goto-char p)
nil)))
;;; Indent functions ==========================================================
(defun matlab-indent-line ()
"Indent a line in `matlab-mode'."
(interactive)
(let ((i (matlab-calc-indent))
(c (current-column)))
(save-excursion
(back-to-indentation)
(if (= i (current-column))
nil
(beginning-of-line)
(delete-horizontal-space)
(indent-to i))
;; If line contains a comment, format it.
(if () (if (matlab-lattr-comm) (matlab-comment))))
(if (<= c i) (move-to-column i))))
(defun matlab-calc-indent ()
"Return the appropriate indentation for this line as an integer."
(interactive)
;; The first step is to find the current indentation.
;; This is defined to be zero if all previous lines are empty.
(let* ((ci (save-excursion (if (not (matlab-prev-line))
0
(matlab-next-line-indentation))))
(sem (matlab-calculate-indentation ci)))
;; simplistic
(nth 1 sem)))
(defconst matlab-functions-have-end-should-be-true
"This end closes a function definition.\nDo you want functions to have ends? "
"Prompt the user about whether to change matlab-functions-have-end")
(defun matlab-calculate-indentation (current-indentation)
"Calculate out the indentation of the current line.
Return a list of descriptions for this line. Return format is:
'(TYPE DEPTHNUMBER)
where TYPE is one of (comment, code, function, blockstart, blockmid,
blockendless, blockend) DEPTHNUMBER is how many characters to indent
this line.
Argument CURRENT-INDENTATION is what the previous line thinks
this line's indentation should be. See `matlab-next-line-indentation'."
(matlab-navigation-syntax
(matlab-calculate-indentation-1 current-indentation)))
(defun matlab-calculate-indentation-1 (current-indentation)
"Do the indentation work of `matlab-calculate-indentation'.
Argument CURRENT-INDENTATION is what the previous line recommends for indentation."
2005-12-01 20:08:16 +01:00
(let ((ci current-indentation)
(tmp nil))
(cond
;; COMMENTS
((matlab-ltype-comm)
(cond
;; HELP COMMENT and COMMENT REGION
((or (matlab-ltype-help-comm)
(matlab-ltype-comm-ignore))
(list 'comment-help
(save-excursion
(matlab-beginning-of-defun)
(current-indentation))))
;; COMMENT Continued From Previous Line
((setq tmp (matlab-ltype-continued-comm))
(list 'comment tmp))
;; END FUNCTION COMMENT
((matlab-ltype-endfunction-comm)
(list 'comment-endfunction 0))
(t
(list 'comment (+ ci matlab-comment-anti-indent)))))
;; FUNCTION DEFINITION
((matlab-ltype-function-definition)
(if matlab-functions-have-end
;; A function line has intrinsic indentation iff function bodies are
;; not indented and the function line is nested within another function.
(if (and (not (matlab-indent-function-body-p))
2005-12-01 20:08:16 +01:00
(save-excursion
(beginning-of-line)
(matlab-beginning-of-enclosing-defun)))
(setq ci (+ ci matlab-indent-level))
;; If no intrinsic indentation, do not change from ci.
)
;; If functions are not nested, functions go to left margin.
(setq ci 0))
(list 'function ci))
;; END keyword
((matlab-lattr-local-end)
(let ((end-of-function
(let ((matlab-functions-have-end t))
(save-excursion
(beginning-of-line)
(matlab-backward-sexp t) ;; may throw "unstarted block" error
(matlab-ltype-function-definition)))))
(if end-of-function
(if (or matlab-functions-have-end
(if (yes-or-no-p matlab-functions-have-end-should-be-true)
;; TODO - ask user to reindent the fcn now?
2005-12-01 20:08:16 +01:00
(setq matlab-functions-have-end t)
(error "Unmatched end")))
(if (matlab-indent-function-body-p)
2005-12-01 20:08:16 +01:00
(setq ci (- ci matlab-indent-level))))
;; Next, see if this line starts with an end, and whether the
;; end is matched, and whether the line is blank up to the match.
;; If so, return the indentation of the match.
(catch 'indent
(save-excursion
(when (progn (beginning-of-line)
(and (looking-at "[ \t]*end\\b")
(matlab-backward-sexp t t)))
(let ((match (point)))
(beginning-of-line)
(looking-at "[ \t]*")
(when (= match (match-end 0))
(let ((match-col-end
(save-excursion
(goto-char match)
(current-column)))
(match-col-beginning
(save-excursion
(goto-char (match-beginning 0))
(current-column)))
)
(setq ci (- match-col-end match-col-beginning)))
2005-12-01 20:08:16 +01:00
(throw 'indent nil)))))
;; End of special case for end and match after "^[ \t]*".
(setq ci (+ ci
(* (1- (matlab-lattr-block-cont (point)))
matlab-indent-level))))))
(list 'blockend ci))
;; ELSE/CATCH keywords
((matlab-lattr-middle-block-cont)
(let ((m (match-string 1)))
(list 'blockmid
(condition-case nil
(save-excursion
(beginning-of-line)
(matlab-backward-sexp t)
(if (matlab-ltype-function-definition) (error ""))
(current-column))
(error (error "Unmatched %s" m))))))
;; CASE/OTHERWISE keywords
((matlab-lattr-endless-block-cont)
(list 'blockendless
(condition-case nil
(save-excursion
(beginning-of-line)
(matlab-backward-sexp t)
(if (not (looking-at "switch\\>")) (error ""))
(+ (current-column)
(if (listp matlab-case-level)
(car matlab-case-level)
matlab-case-level)))
(error (error "Unmatched case/otherwise part")))))
;; End of a MATRIX
((matlab-lattr-array-end)
(list 'array-end (save-excursion
(back-to-indentation)
(matlab-up-list -1)
(let* ((fc (following-char))
(mi (assoc fc matlab-maximum-indents))
(max (if mi (if (listp (cdr mi))
(car (cdr mi)) (cdr mi))
nil))
(ind (if mi (if (listp (cdr mi))
(cdr (cdr mi)) (cdr mi))
nil)))
;; apply the maximum limits.
(if (and ind (> (- (current-column) ci) max))
(1- ind) ; decor
(current-column))))))
;; Code lines
((save-excursion
(beginning-of-line)
(back-to-indentation)
(= (point) (progn (matlab-beginning-of-command) (point))))
;; This means we are at the beginning of a command structure.
;; Always match up against the previous line.
(list 'code ci))
;; Lines continued from previous statements.
(t
(list (if (matlab-ltype-empty) 'empty
(if (matlab-lattr-array-cont) 'array-cont 'code))
(condition-case nil
;; Line up with opening paren/brace/bracket
(let ((boc (save-excursion
(matlab-beginning-of-command)
(point))))
(save-excursion
(beginning-of-line)
(matlab-up-list -1)
(if (> boc (point)) (error nil))
;; Ok, it MIGHT be that we are in a program
;; statement, and this particular command is an HG
;; statement that would look better if the
;; following lines lined up AFTER the first
;; argument. Lets look.
(let ((parendepth (current-column)))
(cond ((and (= (following-char) ?\( )
(save-excursion
(matlab-navigation-syntax
(forward-word -1)
(looking-at
matlab-indent-past-arg1-functions)))
(let ((start-paren (point)))
(while
(and
(re-search-forward
"," (matlab-point-at-eol) t)
(save-excursion
(matlab-up-list -1)
(> (point) start-paren))))
(if (and
(= (preceding-char) ?,)
;; Don't bother if we hit the EOL.
(not (looking-at
"\\s-*\\(\\.\\.\\.\\|$\\|)\\)")))
t
(move-to-column parendepth)
nil)))
(skip-chars-forward " \t")
(if (> (- (current-column) parendepth)
matlab-arg1-max-indent-length)
(+ parendepth matlab-arg1-max-indent-length)
(current-column)))
(t
(let* ((fc (following-char))
(mi (assoc fc matlab-maximum-indents))
(max (if mi
(if (listp (cdr mi))
(car (cdr mi)) (cdr mi))
nil))
(ind (if mi
(if (listp (cdr mi))
(cdr (cdr mi)) (cdr mi))
nil)))
(forward-char 1)
(skip-chars-forward " \t")
;; If we are at the end of a line and
;; this open paren is there, then we
;; DONT want to indent to it. Use the
;; standard indent.
(if (or (not matlab-align-to-paren)
(looking-at "\\.\\.\\.\\|$"))
2005-12-01 20:08:16 +01:00
;; This could happen in another set
;; of matrices. Find a current
2005-12-01 20:08:16 +01:00
;; indentation based on the
;; previous line.
(let ((cci (current-indentation)))
(+ cci matlab-cont-level))
;; apply the maximum limits.
(if (and ind (> (- (current-column) ci) max))
(+ ci ind)
(current-column)))))))))
(error
;; Line up to an equals sign.
(save-excursion
(matlab-beginning-of-command)
(while (and (re-search-forward "=" (matlab-point-at-eol) t)
(matlab-cursor-in-string-or-comment)))
(if (/= (preceding-char) ?=)
(+ ci matlab-cont-level)
(skip-chars-forward " \t")
(let ((cc (current-column))
(mi (assoc ?= matlab-maximum-indents)))
(if (looking-at "\\.\\.\\.\\|$")
;; In this case, the user obviously wants the
;; indentation to be somewhere else.
(+ ci (cdr (cdr mi)))
;; If the indent delta is greater than the max,
;; use the max + current
2005-12-01 20:08:16 +01:00
(if (and mi (> (- cc ci) (if (listp (cdr mi))
(car (cdr mi))
(cdr mi))))
(setq cc (+ ci (if (listp (cdr mi))
(cdr (cdr mi))
(cdr mi)))))
cc))))))))
)))
(defun matlab-next-line-indentation ()
"Calculate the indentation for lines following this command line.
Assume that the following line does not contribute its own indentation
\(as it does in the case of nested functions in the following situations):
o function---positive indentation when not indenting function bodies.
o end---negative indentation except when the 'end' matches a function and
not indenting function bodies.
See `matlab-calculate-indentation'."
(matlab-navigation-syntax
(let ((startpnt (point-at-eol)))
(save-excursion
(matlab-beginning-of-command)
(let ((cc (or (matlab-lattr-block-close startpnt) 0))
(end (matlab-lattr-local-end))
(bc (matlab-lattr-block-cont startpnt))
(mc (matlab-lattr-middle-block-cont))
(ec (matlab-lattr-endless-block-cont))
(hc (and (matlab-indent-function-body-p) (matlab-ltype-help-comm)))
(rc (and (/= 0 matlab-comment-anti-indent)
(matlab-ltype-comm)
(not (matlab-ltype-help-comm))
(not (matlab-ltype-continued-comm))
(not (matlab-ltype-endfunction-comm))))
(ci (current-indentation)))
;; When the current point is on a line with a function, the value of bc will
;; reflect the function in a block count iff if matlab-functions-have-end is
;; true. However, if matlab-indent-function-body-p is false, there should be
;; no actual indentation, so bc needs to be decremented by 1. Similarly, if
;; on a line with an end that closes a function, bc needs to be decremented
;; by 1 if matlab-functions-have-end is true and matlab-indent-function-body-p
;; is false. However, just to be safe, indentation is not allowed to go
;; negative. Thus:
(if matlab-functions-have-end
(if (and
(not (matlab-indent-function-body-p))
(or (matlab-ltype-function-definition)
(and (matlab-lattr-local-end)
(save-excursion
(matlab-backward-sexp t)
(looking-at "function\\b")))))
(if (> bc 0)
(setq bc (1- bc))
(if (>= ci matlab-indent-level)
(setq bc -1))))
(if (and (matlab-indent-function-body-p) (matlab-ltype-function-definition))
(setq bc (1+ bc))))
;; Remove 1 from the close count if there is an END on the beginning
;; of this line, since in that case, the unindent has already happened.
(when end (setq cc (1- cc)))
;; Calculate the suggested indentation.
(+ ci
(* matlab-indent-level bc)
(* matlab-indent-level (or mc 0))
(* matlab-indent-level (- cc))
(* (if (listp matlab-case-level)
(cdr matlab-case-level) matlab-case-level)
(or ec 0))
(if hc matlab-indent-level 0)
(if rc (- 0 matlab-comment-anti-indent) 0)
))))))
2005-12-01 20:08:16 +01:00
;;; The return key ============================================================
(defcustom matlab-return-function 'matlab-indent-end-before-ret
"Function to handle return key.
Must be one of:
'matlab-plain-ret
'matlab-indent-after-ret
'matlab-indent-end-before-ret
'matlab-indent-before-ret"
:group 'matlab
:type '(choice (function-item matlab-plain-ret)
(function-item matlab-indent-after-ret)
(function-item matlab-indent-end-before-ret)
(function-item matlab-indent-before-ret)))
(defun matlab-return ()
"Handle carriage return in `matlab-mode'."
(interactive)
(matlab-semicolon-on-return)
(funcall matlab-return-function))
(defun matlab-plain-ret ()
"Vanilla new line."
(interactive)
(newline))
2005-12-01 20:08:16 +01:00
(defun matlab-indent-after-ret ()
"Indent after new line."
(interactive)
(newline)
(matlab-indent-line))
(defun matlab-indent-end-before-ret ()
"Indent line if block end, start new line, and indent again."
(interactive)
(if (save-excursion
(beginning-of-line)
(looking-at (concat "^\\s-*\\(" (matlab-block-end-re)
"\\|" (matlab-block-mid-re)
"\\|" (matlab-endless-blocks-re)
"\\|function\\)")))
(matlab-indent-line))
(newline)
(matlab-indent-line))
(defvar matlab-quiesce-nosemi-regexp) ;; quiet compiler warning (var is defined below)
2005-12-01 20:08:16 +01:00
(defun matlab-semicolon-on-return ()
"If needed, add a semicolon at point automatically."
(if matlab-return-add-semicolon
(if (and (not (matlab-ltype-empty))
(not (save-excursion
(skip-chars-backward " \t;" (matlab-point-at-bol))
(looking-at "\\s-*;")))
(save-excursion
(let ((p (point)))
(matlab-end-of-command (point))
(eq p (point))))
(save-excursion
(matlab-beginning-of-command)
;; Note: Compile warning below, but defined later.
2005-12-01 20:08:16 +01:00
(not (looking-at matlab-quiesce-nosemi-regexp))))
(insert ";"))
))
(defun matlab-indent-before-ret ()
"Indent line, start new line, and indent again."
(interactive)
(matlab-indent-line)
(newline)
(matlab-indent-line))
(defun matlab-linefeed ()
"Handle line feed in `matlab-mode'.
Has effect of `matlab-return' with (not matlab-indent-before-return)."
(interactive)
(matlab-indent-line)
(newline)
(matlab-indent-line))
(defun matlab-comment-return ()
"Handle carriage return for MATLAB comment line."
(interactive)
(cond
((matlab-ltype-comm)
(matlab-set-comm-fill-prefix) (newline) (insert fill-prefix)
(matlab-reset-fill-prefix) (matlab-indent-line))
((matlab-lattr-comm)
(newline) (indent-to comment-column)
(insert matlab-comment-on-line-s))
(t
(newline) (matlab-comment) (matlab-indent-line))))
(defun matlab-comm-from-prev ()
"If the previous line is a comment-line then set up a comment on this line."
(save-excursion
;; If the previous line is a comment-line then set the fill prefix from
;; the previous line and fill this line.
(if (and (= 0 (forward-line -1)) (matlab-ltype-comm))
(progn
(matlab-set-comm-fill-prefix)
(forward-line 1) (beginning-of-line)
(delete-horizontal-space)
(if (looking-at "%") (delete-char 1))
(delete-horizontal-space)
(insert fill-prefix)
(matlab-reset-fill-prefix)))))
(defun matlab-electric-comment (arg)
"Indent line and insert comment character.
Argument ARG specifies how many %s to insert."
(interactive "P")
(self-insert-command (or arg 1))
(when (matlab-ltype-comm)
(matlab-indent-line)
;; The above seems to put the cursor on the %, not after it.
(skip-chars-forward "%")))
2005-12-01 20:08:16 +01:00
;;; Comment management========================================================
(defun matlab-comment ()
"Add a comment to the current line."
(interactive)
(cond ((matlab-ltype-empty) ; empty line
(matlab-comm-from-prev)
(if (matlab-lattr-comm)
(skip-chars-forward " \t%")
(insert matlab-comment-line-s)
(matlab-indent-line)))
((matlab-ltype-comm) ; comment line
(matlab-comm-from-prev)
(skip-chars-forward " \t%"))
((matlab-lattr-comm) ; code line w/ comment
(beginning-of-line)
(re-search-forward "[^%]\\(%\\)[ \t]")
(goto-char (match-beginning 1))
2005-12-01 20:08:16 +01:00
(if (> (current-column) comment-column) (delete-horizontal-space))
(if (< (current-column) comment-column) (indent-to comment-column))
;; Now see if the current line is too long to fit. Can we back indent?
(let ((eol-col (- (point-at-eol) (point-at-bol))))
(when (> eol-col fill-column)
(delete-horizontal-space)
(indent-to (- comment-column (- eol-col fill-column)))))
(skip-chars-forward "% \t"))
2005-12-01 20:08:16 +01:00
(t ; code line w/o comment
(end-of-line)
(re-search-backward "[^ \t\n^]" 0 t)
(forward-char)
(delete-horizontal-space)
(if (< (current-column) comment-column)
(indent-to comment-column)
(insert " "))
(insert matlab-comment-on-line-s))))
(defun matlab-comment-line-break-function (&optional soft)
"Break the current line, and if in a comment, continue it.
Optional argument SOFT indicates that the newline is soft, and not hard."
(interactive)
(if (not (matlab-cursor-in-comment))
(matlab-return)
;; Will the below fn work in old emacsen?
(if soft (insert-and-inherit ?\n) (newline 1))
(insert "% ")
(matlab-indent-line)
(end-of-line)))
(defun matlab-comment-indent ()
"Indent a comment line in `matlab-mode'."
(matlab-calc-indent))
(defun matlab-comment-region (beg-region end-region arg)
"Comments every line in the region.
Puts `matlab-comment-region-s' at the beginning of every line in the region.
BEG-REGION and END-REGION are arguments which specify the region boundaries.
With non-nil ARG, uncomments the region."
(interactive "*r\nP")
(let ((end-region-mark (make-marker)) (save-point (point-marker)))
(set-marker end-region-mark end-region)
(goto-char beg-region)
(beginning-of-line)
(if (not arg) ;comment the region
(progn (insert matlab-comment-region-s)
(while (and (= (forward-line 1) 0)
(< (point) end-region-mark))
(insert matlab-comment-region-s)))
(let ((com (regexp-quote matlab-comment-region-s))) ;uncomment the region
(if (looking-at com)
(delete-region (point) (match-end 0)))
(while (and (= (forward-line 1) 0)
(< (point) end-region-mark))
(if (looking-at com)
(delete-region (point) (match-end 0))))))
(goto-char save-point)
(set-marker end-region-mark nil)
(set-marker save-point nil)))
(defun matlab-uncomment-region (beg end)
"Uncomment the current region if it is commented out.
Argument BEG and END indicate the region to uncomment."
(interactive "*r")
(matlab-comment-region beg end t))
;;; Filling ===================================================================
(defun matlab-set-comm-fill-prefix ()
"Set the `fill-prefix' for the current (comment) line."
(interactive)
(if (matlab-lattr-comm)
(setq fill-prefix
(save-excursion
(beginning-of-line)
(let ((e (matlab-point-at-eol))
(pf nil))
(while (and (re-search-forward "%+[ \t]*\\($$$ \\)?" e t)
2005-12-01 20:08:16 +01:00
(matlab-cursor-in-string)))
(setq pf (match-string 0))
(concat (make-string (- (current-column) (length pf)) ? )
pf))))))
(defun matlab-set-comm-fill-prefix-post-code ()
"Set the `fill-prefix' for the current post-code comment line."
(interactive)
(matlab-set-comm-fill-prefix))
(defun matlab-reset-fill-prefix ()
"Reset the `fill-prefix'."
(setq fill-prefix nil))
(defun matlab-find-convenient-line-break ()
"For the current line, position the cursor where we want to break the line.
Basically, spaces are best, then operators. Always less than `fill-column'
unless we decide we can fudge the numbers. Return nil if this line should
not be broken. This function will ONLY work on code."
;; First of all, if this is a continuation, then the user is
;; requesting that we don't mess with his stuff.
(if (matlab-lattr-cont)
nil
(save-restriction
(narrow-to-region (matlab-point-at-bol) (matlab-point-at-eol))
;; get ourselves onto the fill-column.
(move-to-column fill-column)
(let ((pos nil)
(orig (point)))
(or
;; Next, if we have a trailing comment, use that.
(progn (setq pos (or (matlab-lattr-comm) (matlab-point-at-bol)))
(goto-char pos)
(if (and (> (current-column) (- fill-column matlab-fill-fudge))
(< (current-column) (+ fill-column matlab-fill-fudge)))
t
(goto-char orig)
nil))
;; Now, lets find the nearest space (after or before fill column)
(let* ((after (save-excursion
(re-search-forward "[ \t]" nil t)))
(before (save-excursion
(re-search-backward "[ \t]" nil t)))
(afterd (- (or after (matlab-point-at-eol)) (point)))
(befored (- (point) (or before (matlab-point-at-bol)))))
;; Here, if "before" is actually the beginning of our
;; indentation, then this is most obviously a bad place to
2005-12-01 20:08:16 +01:00
;; break our lines.
(if before
(save-excursion
(goto-char before)
(if (<= (point) (save-excursion
(back-to-indentation)
(point)))
(setq before nil))))
(cond ((and after
(< afterd matlab-fill-fudge)
(< afterd befored))
(goto-char after)
t)
((and before
(< befored matlab-fill-fudge)
(< befored afterd))
(goto-char before)
t)
(t (goto-char orig)
nil)))
;; Now, lets find the nearest backwards
(progn
(re-search-backward "\\(\\s-\\|\\s.\\)+" nil t)
(while (and (looking-at "\\^\\|\\.\\|'")
(re-search-backward "\\(\\s-\\|\\s.\\)+" nil t)))
(if (or (not (looking-at "\\(\\s-\\|\\s.\\)+"))
(<= (point) (save-excursion
(back-to-indentation)
(point))))
(progn
;; We failed in our mission to find anything, or fell
;; of the edge of the earth. If we are out of
;; bounds, lets try again.
(goto-char orig)
(if (re-search-backward "\\s.+" nil t)
t
nil))
;; Ok, we have a good location to break. Check for column
;; and ref against nearest list ending to predict a possibly
;; better break point.
(forward-char 1)
(let ((okpos (current-column))
(startlst (save-excursion
(condition-case nil
(matlab-up-list -1)
(error nil))
(if (save-excursion
(forward-char -1)
(looking-at "\\w"))
(forward-word -1))
(current-column)))
(endlst (save-excursion
(condition-case nil
(matlab-up-list 1)
(error nil))
(current-column))))
;; When evaluating list fudge factors, breaking on the
2005-12-01 20:08:16 +01:00
;; edge of a list, or at the beginning of a function
;; call can be more valuable than breaking on a symbol
;; of a mid-sized list. As such, allow double-fudge
;; for lists.
(cond
;; First, pick the end of a list.
((and (< endlst matlab-fill-fudge-hard-maximum)
(<= endlst (+ fill-column matlab-fill-fudge))
(or (<= (* matlab-fill-fudge 2) (- endlst okpos))
(<= endlst fill-column))
(save-excursion
(move-to-column endlst)
(not (looking-at "\\^"))))
(move-to-column endlst)
t)
;; Else, back up over this list and poke around
((>= (* 2 matlab-fill-fudge) (- okpos startlst))
(move-to-column startlst)
t)
;; Oh well, just do this symbol.
(t (move-to-column okpos)
t)))))
;; Well, this just sucks
(progn (goto-char orig)
nil))))))
(defun matlab-auto-fill ()
"Do auto filling.
Set variable `auto-fill-function' to this symbol to enable MATLAB style auto
filling which will automatically insert `...' and the end of a line."
(interactive)
(let ((fill-prefix fill-prefix) ;; safe way of modifying fill-prefix.
(fill-column (- fill-column
(if matlab-fill-count-ellipsis-flag
(save-excursion
(move-to-column fill-column)
(if (not (bobp))
(forward-char -1))
(if (matlab-cursor-in-string 'incomplete)
4 3))
0))))
(if (> (current-column) fill-column)
(cond
((matlab-ltype-comm-ignore)
nil)
((or (matlab-ltype-comm)
(and (save-excursion (move-to-column fill-column)
(matlab-cursor-in-comment))
(matlab-lattr-comm)))
;; If the whole line is a comment, do this.
(matlab-set-comm-fill-prefix) (do-auto-fill)
(matlab-reset-fill-prefix))
((and (matlab-ltype-code)
(not (matlab-lattr-cont))
matlab-fill-code)
;; If we are on a code line, we ellipsify before we fill.
(let ((m (make-marker)))
(move-marker m (point))
(set-marker-insertion-type m t)
(if (not (matlab-find-convenient-line-break))
nil
(if (not (save-excursion
(forward-char -1)
(matlab-cursor-in-string 'incomplete)))
(progn
(delete-horizontal-space)
(insert " " matlab-elipsis-string "\n")
(matlab-indent-line))
(if matlab-fill-strings-flag
(let ((pos (point))
(pos2 nil))
(while (and (re-search-backward "'" (point-at-bol) t)
(progn (forward-char -1)
(looking-at "''"))))
(setq pos2 (point))
;; Check if there is already an opening bracket or if string is continued
(if (or (looking-at "\\[")
(save-excursion (skip-chars-backward " \t")
(forward-char -1)
(looking-at "\\["))
(progn
(beginning-of-line)
(skip-chars-backward (concat " \t\n" matlab-elipsis-string))
(if (> (point) (point-min))
(progn
(forward-char -1)
(looking-at (concat "'\\s-*" matlab-elipsis-string))))))
(goto-char pos)
(goto-char pos2)
(forward-char 1)
(insert "[")
(goto-char pos)
(forward-char 1))
;(delete-horizontal-space)
(skip-chars-forward " \t")
(insert "' " matlab-elipsis-string "\n")
(matlab-indent-line)
(insert "'")
;; Re scan forward for the end of the string. Add an end bracket
;; if there isn't one already. Also add an apostrophe if necessary.
(if (not (looking-at "'\\s-*]"))
(save-excursion
(if (not (re-search-forward "[^']'\\([^']\\|$\\)" (line-end-position) t))
2005-12-01 20:08:16 +01:00
(progn
(end-of-line)
(insert "']")
(move-marker m (- (point) 2)))
(re-search-backward "'")
(cond ((looking-at "'\\s-*]")
nil ; already in an array.
)
((or (looking-at "'\\s-*$") (looking-at "'\\s-*[^]]"))
;; in a string, add an array end.
(forward-char 1)
(insert "]"))
((looking-at "'\\s-*\\.\\.\\.")
;; Already extended to next line ... leave it alone.
nil)
))))
))))
(goto-char m)))
))))
2005-12-01 20:08:16 +01:00
(defun matlab-join-comment-lines ()
"Join current comment line to the next comment line."
;; New w/ V2.0: This used to join the previous line, but I could find
;; no editors that had a "join" that did that. I modified join to have
;; a behaviour I thought more inline with other editors.
(interactive)
(end-of-line)
(if (looking-at "\n[ \t]*%")
(replace-match " " t t nil)
(error "No following comment to join with")))
(defun matlab-fill-region (beg-region end-region &optional justify-flag)
"Fill the region between BEG-REGION and END-REGION.
Non-nil JUSTIFY-FLAG means justify comment lines as well."
(interactive "*r\nP")
(let ((end-reg-mk (make-marker)))
(set-marker end-reg-mk end-region)
(goto-char beg-region)
(beginning-of-line)
(while (< (point) end-reg-mk)
;; This function must also leave the point at the end of the
;; justified line.
(matlab-fill-paragraph justify-flag)
(forward-line 1)
(beginning-of-line))))
(defun matlab-fill-comment-line (&optional justify)
"Fill the current comment line.
With optional argument, JUSTIFY the comment as well."
(interactive)
(if (not (matlab-comment-on-line))
(error "No comment to fill"))
(beginning-of-line)
;; First, find the beginning of this comment...
(while (and (looking-at matlab-cline-start-skip)
(not (bobp)))
(forward-line -1)
(beginning-of-line))
(if (not (looking-at matlab-cline-start-skip))
(forward-line 1))
;; Now scan to the end of this comment so we have our outer bounds,
;; and narrow to that region.
(save-restriction
(narrow-to-region (point)
(save-excursion
(while (and (looking-at matlab-cline-start-skip)
(not (save-excursion (end-of-line) (eobp))))
(forward-line 1)
(beginning-of-line))
(if (not (looking-at matlab-cline-start-skip))
(forward-line -1))
(end-of-line)
(point)))
;; Find the fill prefix...
(matlab-comment-on-line)
(looking-at "%[ \t]*")
(let ((fill-prefix (concat (make-string (current-column) ? )
(match-string 0))))
(fill-region (point-min) (point-max) justify))))
(defun matlab-justify-line ()
"Delete space on end of line and justify."
(interactive)
(save-excursion
(end-of-line)
(delete-horizontal-space)
(justify-current-line)))
(defun matlab-fill-paragraph (arg)
"When in a comment, fill the current paragraph.
Paragraphs are always assumed to be in a comment.
ARG is passed to `fill-paragraph' and will justify the text."
(interactive "P")
(cond ((or (matlab-ltype-comm)
(and (matlab-cursor-in-comment)
(not (matlab-lattr-cont))))
;; We are in a comment, lets fill the paragraph with some
;; nice regular expressions.
;; Cell start/end markers of %% also separate paragraphs
(let ((paragraph-separate "%%\\|%[a-zA-Z]\\|%[ \t]*$\\|[ \t]*$")
2005-12-01 20:08:16 +01:00
(paragraph-start "%[a-zA-Z]\\|%[ \t]*$\\|[ \t]*$")
(paragraph-ignore-fill-prefix nil)
(start (save-excursion (matlab-beginning-of-command)
(if (looking-at "%%")
(progn (end-of-line)
(forward-char 1)))
2005-12-01 20:08:16 +01:00
(point)))
(end (save-excursion (matlab-end-of-command)
(point)))
(fill-prefix nil))
(matlab-set-comm-fill-prefix)
(save-restriction
;; Ben North fixed to handle comment at the end of
;; a buffer.
(narrow-to-region start (min (point-max) (+ end 1)))
(fill-paragraph arg))))
((matlab-ltype-code)
;; Ok, lets get the outer bounds of this command, then
;; completely refill it using the smart line breaking code.
(save-restriction
(narrow-to-region (save-excursion
(matlab-beginning-of-command)
(beginning-of-line)
(point))
(save-excursion
(matlab-end-of-command)
(point)))
;; Remove all line breaks
(goto-char (point-min))
(while (and (re-search-forward "$" nil t)
(not (eobp)))
(delete-horizontal-space)
;; Blow away continuation marks
(if (matlab-lattr-cont)
(progn
(goto-char (match-beginning 0))
(forward-char 1)
(delete-region (point) (matlab-point-at-eol))))
;; Zap the CR
(if (not (eobp)) (delete-char 1))
;; Clean up whitespace
(delete-horizontal-space)
;; Clean up trailing comments
(if (and (looking-at "% *")
(matlab-cursor-in-comment))
(progn
(delete-char 1)
(delete-horizontal-space)))
(insert " "))
;; Now fill till we are done
(goto-char (point-max))
(while (or (> (current-column) (+ fill-column matlab-fill-fudge))
(> (current-column) matlab-fill-fudge-hard-maximum))
(if (= (point)
(progn
(matlab-auto-fill)
(point)))
(error "Fill algorithm failed!"))
2005-12-01 20:08:16 +01:00
(if arg (save-excursion
(forward-line -1)
(matlab-justify-line))))
(if arg (save-excursion
(forward-line -1)
(matlab-justify-line)))))
(t
(message "Paragraph Fill not supported in this context."))))
;;; Semantic text insertion and management ====================================
(defun matlab-find-recent-variable-list (prefix)
"Return a list of most recent variables starting with PREFIX as a string.
Reverse searches for the following are done first:
1) Assignment
2) if|for|while|switch <var>
3) global variables
4) function arguments.
All elements are saved in a list, which is then uniquified.
2005-12-01 20:08:16 +01:00
If NEXT is non-nil, then the next element from the saved list is used.
If the list is empty, then searches continue backwards through the code."
(matlab-navigation-syntax
(let* ((bounds (save-excursion
(if (re-search-backward "^\\s-*function\\>" nil t)
(match-beginning 0) (point-min))))
(syms
(append
(save-excursion
(let ((lst nil))
(while (and
(re-search-backward
(concat "^\\s-*\\(" prefix "\\w+\\)\\s-*=")
bounds t)
(< (length lst) 10))
(setq lst (cons (match-string 1) lst)))
(nreverse lst)))
(save-excursion
(let ((lst nil))
(while (and (re-search-backward
(concat "\\<\\(" matlab-block-beg-pre-no-if
"\\)\\s-+(?\\s-*\\(" prefix
"\\w+\\)\\>")
bounds t)
(< (length lst) 10))
(setq lst (cons (match-string 2) lst)))
(nreverse lst)))
(save-excursion
(if (re-search-backward "^\\s-*global\\s-+" bounds t)
(let ((lst nil) m e)
(goto-char (match-end 0))
(while (looking-at "\\(\\w+\\)\\([ \t]+\\|$\\)")
(setq m (match-string 1)
e (match-end 0))
(if (equal 0 (string-match prefix m))
(setq lst (cons m lst)))
(goto-char e))
(nreverse lst))))
(save-excursion
(if (and (re-search-backward "^\\s-*function\\>" bounds t)
(re-search-forward "\\<\\(\\w+\\)("
(matlab-point-at-eol) t))
(let ((lst nil) m e)
(while (looking-at "\\(\\w+\\)\\s-*[,)]\\s-*")
(setq m (match-string 1)
e (match-end 0))
(if (equal 0 (string-match prefix m))
(setq lst (cons m lst)))
(goto-char e))
(nreverse lst))))))
(fl nil))
(while syms
(if (car syms) (setq fl (cons (car syms) fl)))
(setq syms (cdr syms)))
(matlab-uniquafy-list (nreverse fl)))))
(defvar matlab-most-recent-variable-list nil
"Maintained by `matlab-find-recent-variable'.")
(defun matlab-find-recent-variable (prefix &optional next)
"Return the most recently used variable starting with PREFIX as a string.
See `matlab-find-recent-variable-list' for details.
In NEXT is non-nil, than continue through the list of elements."
(if next
(let ((next (car matlab-most-recent-variable-list)))
(setq matlab-most-recent-variable-list
(cdr matlab-most-recent-variable-list))
next)
(let ((syms (matlab-find-recent-variable-list prefix))
(first nil))
(if (eq matlab-completion-technique 'complete)
syms
(setq first (car syms))
(setq matlab-most-recent-variable-list (cdr syms))
first))))
(defun matlab-find-user-functions-list (prefix)
"Return a list of user defined functions that match PREFIX."
(matlab-navigation-syntax
(let ((syms
(append
(save-excursion
(goto-char (point-min))
(let ((lst nil))
(while (re-search-forward "^\\s-*function\\>" nil t)
(if (re-search-forward
(concat "\\(" prefix "\\w+\\)\\s-*\\($\\|(\\)")
(matlab-point-at-eol) t)
(setq lst (cons (match-string 1) lst))))
(nreverse lst)))
(let ((lst nil)
(files (directory-files
default-directory nil
(concat "^" prefix
"[a-zA-Z][a-zA-Z0-9_]+\\.m$"))))
(while files
(setq lst (cons (progn (string-match "\\.m" (car files))
(substring (car files) 0
(match-beginning 0)))
lst)
files (cdr files)))
lst)))
(fl nil))
(while syms
(if (car syms) (setq fl (cons (car syms) fl)))
(setq syms (cdr syms)))
(matlab-uniquafy-list (nreverse fl)))))
(defvar matlab-user-function-list nil
"Maintained by `matlab-find-user-functions'.")
(defun matlab-find-user-functions (prefix &optional next)
"Return a user function that match PREFIX and return it.
If optional argument NEXT is non-nil, then return the next found
object."
(if next
(let ((next (car matlab-user-function-list)))
(setq matlab-user-function-list (cdr matlab-user-function-list))
next)
(let ((syms (matlab-find-user-functions-list prefix))
(first nil))
(if (eq matlab-completion-technique 'complete)
syms
(setq first (car syms))
(setq matlab-user-function-list (cdr syms))
first))))
(defvar matlab-generic-list-placeholder nil
"Maintained by `matlab-generic-list-expand'.
2005-12-01 20:08:16 +01:00
Holds sub-lists of symbols left to be expanded.")
(defun matlab-generic-list-expand (list prefix &optional next)
"Return an element from LIST that start with PREFIX.
If optional NEXT argument is non nil, then the next element in the
list is used. nil is returned if there are not matches."
(if next
(let ((next (car matlab-generic-list-placeholder)))
(setq matlab-generic-list-placeholder
(cdr matlab-generic-list-placeholder))
next)
(let ((re (concat "^" (regexp-quote prefix)))
(first nil)
(fl nil))
(while list
(if (string-match re (car list))
(setq fl (cons (car list) fl)))
(setq list (cdr list)))
(setq fl (nreverse fl))
(if (eq matlab-completion-technique 'complete)
fl
(setq first (car fl))
(setq matlab-generic-list-placeholder (cdr fl))
first))))
(defun matlab-solo-completions (prefix &optional next)
"Return PREFIX matching elements for solo symbols.
If NEXT then the next patch from the list is used."
(matlab-generic-list-expand matlab-keywords-solo prefix next))
(defun matlab-value-completions (prefix &optional next)
"Return PREFIX matching elements for value symbols.
If NEXT then the next patch from the list is used."
(matlab-generic-list-expand matlab-keywords-return prefix next))
(defun matlab-boolean-completions (prefix &optional next)
"Return PREFIX matching elements for boolean symbols.
If NEXT then the next patch from the list is used."
(matlab-generic-list-expand matlab-keywords-boolean prefix next))
2005-12-01 20:08:16 +01:00
(defun matlab-property-completions (prefix &optional next)
"Return PREFIX matching elements for property names in strings.
If NEXT then the next property from the list is used."
(let ((f (matlab-function-called-at-point))
(lst matlab-property-lists)
(foundlst nil)
(expandto nil))
;; Look for this function. If it is a known function then we
;; can now use a subset of available properties!
(while (and lst (not foundlst))
(if (string= (car (car lst)) f)
(setq foundlst (cdr (car lst))))
(setq lst (cdr lst)))
(if foundlst
(setq foundlst (append foundlst matlab-core-properties))
(setq foundlst matlab-all-known-properties))
(setq expandto (matlab-generic-list-expand foundlst prefix next))
;; This looks to see if we have a singular completion. If so,
;; then return it, and also append the "'" to the end.
(cond ((and (listp expandto) (= (length expandto) 1))
(setq expandto (list (concat (car expandto) "'"))))
((stringp expandto)
(setq expandto (concat expandto "'"))))
expandto))
(defvar matlab-last-prefix nil
"Maintained by `matlab-complete-symbol'.
The prefix used for the first completion command.")
(defvar matlab-last-semantic nil
"Maintained by `matlab-complete-symbol'.
The last type of semantic used while completing things.")
(defvar matlab-completion-search-state nil
"List of searching things we will be doing.")
(defun matlab-complete-symbol (&optional arg)
"Complete a partially typed symbol in a MATLAB mode buffer.
If the previously entered command was also `matlab-complete-symbol'
then undo the last completion, and find a new one.
The types of symbols tried are based on the semantics of the current
cursor position. There are two types of symbols. For example, if the
cursor is in an if statement, boolean style functions and symbols are
tried first. If the line is blank, then flow control, or high level
functions are tried first.
The completion technique is controlled with `matlab-completion-technique'
It defaults to incremental completion described above. If a
completion list is preferred, then change this to 'complete. If you
just want a completion list once, then use the universal argument ARG
to change it temporarily."
(interactive "P")
(matlab-navigation-syntax
(let* ((prefix (if (and (not (eq last-command 'matlab-complete-symbol))
(member (preceding-char) '(? ?\t ?\n ?, ?\( ?\[ ?\')))
""
(buffer-substring-no-properties
(save-excursion (forward-word -1) (point))
(point))))
(sem (matlab-lattr-semantics prefix))
(matlab-completion-technique
(if arg (cond ((eq matlab-completion-technique 'complete)
'increment)
(t 'complete))
matlab-completion-technique)))
(if (not (eq last-command 'matlab-complete-symbol))
(setq matlab-last-prefix prefix
matlab-last-semantic sem
matlab-completion-search-state
(cond ((eq sem 'solo)
'(matlab-solo-completions
matlab-find-user-functions
matlab-find-recent-variable))
((eq sem 'boolean)
'(matlab-find-recent-variable
matlab-boolean-completions
matlab-find-user-functions
matlab-value-completions))
((eq sem 'value)
'(matlab-find-recent-variable
matlab-find-user-functions
matlab-value-completions
matlab-boolean-completions))
((eq sem 'property)
'(matlab-property-completions
matlab-find-user-functions
matlab-find-recent-variable
matlab-value-completions))
(t '(matlab-find-recent-variable
matlab-find-user-functions
matlab-value-completions
matlab-boolean-completions)))))
(cond
((eq matlab-completion-technique 'increment)
(let ((r nil) (donext (eq last-command 'matlab-complete-symbol)))
(while (and (not r) matlab-completion-search-state)
(message "Expand with %S" (car matlab-completion-search-state))
(setq r (funcall (car matlab-completion-search-state)
matlab-last-prefix donext))
(if (not r) (setq matlab-completion-search-state
(cdr matlab-completion-search-state)
donext nil)))
(delete-region (point) (progn (forward-char (- (length prefix)))
(point)))
(if r
(insert r)
(insert matlab-last-prefix)
(message "No completions."))))
((eq matlab-completion-technique 'complete)
(let ((allsyms (apply 'append
(mapcar (lambda (f) (funcall f prefix))
matlab-completion-search-state))))
(cond ((null allsyms)
(message "No completions.")
(ding))
((= (length allsyms) 1)
(delete-region (point) (progn
(forward-char (- (length prefix)))
(point)))
(insert (car allsyms)))
((= (length allsyms) 0)
(message "No completions."))
(t
(let* ((al (mapcar (lambda (a) (list a)) allsyms))
(c (try-completion prefix al)))
;; This completion stuff lets us expand as much as is
;; available to us. When the completion is the prefix
;; then we want to display all the strings we've
;; encountered.
(if (and (stringp c) (not (string= prefix c)))
(progn
(delete-region
(point)
(progn (forward-char (- (length prefix)))
(point)))
(insert c))
;; `display-completion-list' does all the complex
;; ui work for us.
(with-output-to-temp-buffer "*Completions*"
(display-completion-list
(matlab-uniquafy-list allsyms)))))))))))))
(defun matlab-insert-end-block (&optional reindent)
"Insert and END block based on the current syntax.
Optional argument REINDENT indicates if the specified block should be re-indented."
(interactive "P")
(if (not (matlab-ltype-empty)) (progn (end-of-line) (insert "\n")))
(let ((valid t) (begin nil))
(save-excursion
(condition-case nil
(progn
(matlab-backward-sexp t)
(setq begin (point)
valid (buffer-substring-no-properties
(point) (save-excursion
(re-search-forward "[\n,;.]" nil t)
(point)))))
(error (setq valid nil))))
(if (not valid)
(error "No block to end")
(insert "end")
(if (stringp valid) (insert " % " valid))
(matlab-indent-line)
(if reindent (indent-region begin (point) nil)))))
(tempo-define-template
"matlab-for"
'("for " p "=" p "," > n>
r> &
"end" > %)
"for"
"Insert a MATLAB for statement"
'matlab-tempo-tags
)
(tempo-define-template
"matlab-while"
'("while (" p ")," > n>
r> &
"end" > %)
"while"
"Insert a MATLAB while statement"
'matlab-tempo-tags
)
(tempo-define-template
"matlab-if"
'("if " p > n
r>
"end" > n)
"if"
"Insert a MATLAB if statement"
'matlab-tempo-tags
)
(tempo-define-template
"matlab-if-else"
'("if " p > n
r>
"else" > n
"end" > n)
"if"
"Insert a MATLAB if statement"
'matlab-tempo-tags
)
(tempo-define-template
"matlab-try"
'("try " > n
r>
"catch" > n
p > n
"end" > n)
"try"
"Insert a MATLAB try catch statement"
'matlab-tempo-tags
)
(tempo-define-template
"matlab-switch"
'("switch " p > n
"otherwise" > n
r>
"end" > n)
"switch"
"Insert a MATLAB switch statement with region in the otherwise clause."
'matlab-tempo-tags)
(defun matlab-insert-next-case ()
"Insert a case statement inside this switch statement."
(interactive)
;; First, make sure we are where we think we are.
(let ((valid t))
(save-excursion
(condition-case nil
(progn
(matlab-backward-sexp t)
(setq valid (looking-at "switch")))
(error (setq valid nil))))
(if (not valid)
(error "Not in a switch statement")))
(if (not (matlab-ltype-empty)) (progn (end-of-line) (insert "\n")))
(indent-to 0)
(insert "case ")
(matlab-indent-line))
(tempo-define-template
"matlab-function"
'("function "
(P "output argument(s): " output t)
;; Insert brackets only if there is more than one output argument
(if (string-match "," (tempo-lookup-named 'output))
'(l "[" (s output) "]")
'(l (s output)))
;; Insert equal sign only if there is output argument(s)
(if (= 0 (length (tempo-lookup-named 'output))) nil
" = ")
;; The name of a function, as defined in the first line, should
;; be the same as the name of the file without .m extension
(if (= 1 (count-lines 1 (point)))
(tempo-save-named
'fname
(file-name-nondirectory (file-name-sans-extension
(buffer-file-name))))
'(l (P "function name: " fname t)))
(tempo-lookup-named 'fname)
"(" (P "input argument(s): ") ")" n
"% " (upcase (tempo-lookup-named 'fname)) " - " (P "H1 line: ") n
"% " p n
(if matlab-functions-have-end
'(l "end" n)))
"function"
"Insert a MATLAB function statement"
'matlab-tempo-tags
)
(defun matlab-stringify-region (begin end)
"Put MATLAB 's around region, and quote all quotes in the string.
Stringification allows you to type in normal MATLAB code, mark it, and
then turn it into a MATLAB string that will output exactly what's in
the region. BEGIN and END mark the region to be stringified."
(interactive "r")
(save-excursion
(goto-char begin)
(if (re-search-forward "\n" end t)
(error
"You may only stringify regions that encompass less than one line"))
(let ((m (make-marker)))
(move-marker m end)
(goto-char begin)
(insert "'")
(while (re-search-forward "'" m t)
(insert "'"))
(goto-char m)
(insert "'"))))
(defun matlab-ispell-strings-region (begin end)
"Spell check valid strings in region with Ispell.
Argument BEGIN and END mark the region boundary."
(interactive "r")
(require 'ispell)
(save-excursion
(goto-char begin)
;; Here we use the font lock function for finding strings.
;; Its cheap, fast, and accurate.
(while (and (matlab-font-lock-string-match-normal end)
(ispell-region (match-beginning 2) (match-end 2))))))
(defun matlab-ispell-strings ()
"Spell check valid strings in the current buffer with Ispell.
Calls `matlab-ispell-strings-region'"
(interactive)
(matlab-ispell-strings-region (point-min) (point-max)))
(defun matlab-ispell-comments (&optional arg)
"Spell check comments in the current buffer with Ispell.
Optional ARG means to only check the current comment."
(interactive "P")
(let ((beg (point-min))
(end (point-max)))
(if (and arg (matlab-ltype-comm))
(setq beg (save-excursion (matlab-beginning-of-command) (point))
end (save-excursion (matlab-end-of-command) (point))))
(save-excursion
(goto-char beg)
(beginning-of-line)
(while (and (matlab-font-lock-comment-match end)
(ispell-region (match-beginning 1) (match-end 1)))))))
(defun matlab-generate-latex ()
"Convert a MATLAB M file into a Latex document for printing.
Author: Uwe Brauer oub@eucmos.sim.ucm.es
Created: 14 Feb 2002"
(interactive "*")
(save-restriction
(save-excursion
(goto-char (point-min))
(insert "\\documentclass[12pt]{report}\n
\\usepackage{listings}
\\lstloadlanguages{Matlab}
\\lstset{language=Matlab,keywordstyle=\\bfseries,labelstep=1,escapechar=\\#}
\\begin{document}
\\begin{lstlisting}{}")
(newline)
(goto-char (point-max))
(insert "\n\\end{lstlisting}\n\\end{document}")
(widen)))
(font-lock-mode nil)
(LaTeX-mode)
(font-lock-mode nil))
;;; Block highlighting ========================================================
(defvar matlab-block-highlighter-timer nil
"The timer representing the block highlighter.")
(defun matlab-enable-block-highlighting (&optional arg)
"Start or stop the block highlighter.
Optional ARG is 1 to force enable, and -1 to disable.
If ARG is nil, then highlighting is toggled."
(interactive "P")
(if (not (fboundp 'matlab-run-with-idle-timer))
(setq matlab-highlight-block-match-flag nil))
;; Only do it if it's enabled.
(if (not matlab-highlight-block-match-flag)
nil
;; Use post command idle hook as a local hook to dissuade too much
;; cpu time while doing other things.
;;(make-local-hook 'post-command-hook)
(if (not arg)
(setq arg
(if (member 'matlab-start-block-highlight-timer
post-command-hook)
-1 1)))
(if (> arg 0)
(add-hook 'post-command-hook 'matlab-start-block-highlight-timer nil :local)
(remove-hook 'post-command-hook 'matlab-start-block-highlight-timer :local))))
2005-12-01 20:08:16 +01:00
(defvar matlab-block-highlight-overlay nil
"The last highlighted overlay.")
(make-variable-buffer-local 'matlab-block-highlight-overlay)
(defvar matlab-block-highlight-timer nil
"Last started timer.")
(make-variable-buffer-local 'matlab-block-highlight-timer)
(defun matlab-start-block-highlight-timer ()
"Set up a one-shot timer if we are in MATLAB mode."
(if (eq major-mode 'matlab-mode)
(progn
(if matlab-block-highlight-overlay
(unwind-protect
(matlab-delete-overlay matlab-block-highlight-overlay)
(setq matlab-block-highlight-overlay nil)))
(if matlab-block-highlight-timer
(unwind-protect
(matlab-cancel-timer matlab-block-highlight-timer)
(setq matlab-block-highlight-timer nil)))
(setq matlab-block-highlight-timer
(matlab-run-with-idle-timer
1 nil 'matlab-highlight-block-match
(current-buffer))))))
2005-12-01 20:08:16 +01:00
(defun matlab-highlight-block-match (&optional buff-when-launched)
"Highlight a matching block if available.
BUFF-WHEN-LAUNCHED is the buffer that was active when the timer was set."
(setq matlab-block-highlight-timer nil)
(if (null buff-when-launched)
;; We were passed a null. This indicates an old version of XEmacs
;; so just turn the feature off
(setq matlab-highlight-block-match-flag nil)
;; Only do neat stuff in the same buffer as the one we were
;; initialized from.
(when (and buff-when-launched
(eq buff-when-launched (current-buffer)))
(let ((inhibit-quit nil) ;turn on G-g
(matlab-scan-on-screen-only t))
(if matlab-show-periodic-code-details-flag
(matlab-show-line-info))
(if (not (matlab-cursor-in-string-or-comment))
(save-excursion
(if (or (bolp)
(looking-at "\\s-")
(save-excursion (forward-char -1) (looking-at "\\s-")))
nil
(forward-word -1))
(if (and (looking-at (concat (matlab-block-beg-re) "\\>"))
(not (looking-at "function")))
(progn
;; We scan forward...
(matlab-forward-sexp)
(backward-word 1)
(if (not (looking-at matlab-block-end-pre-if))
nil ;(message "Unterminated block, or end off screen.")
(setq matlab-block-highlight-overlay
(matlab-make-overlay (point)
(progn (forward-word 1)
(point))
(current-buffer)))
(matlab-overlay-put matlab-block-highlight-overlay
'face 'matlab-region-face)))
(if (and (looking-at (concat (matlab-block-end-pre) "\\>"))
(not (looking-at "function"))
(matlab-valid-end-construct-p))
(progn
;; We scan backward
(forward-word 1)
(condition-case nil
(progn
(matlab-backward-sexp)
(if (not (looking-at (matlab-block-beg-re)))
nil ;(message "Unstarted block at cursor.")
(setq matlab-block-highlight-overlay
(matlab-make-overlay (point)
(progn (forward-word 1)
(point))
(current-buffer)))
(matlab-overlay-put matlab-block-highlight-overlay
'face 'matlab-region-face)))
(error (message "Unstarted block at cursor."))))
;; do nothing
))))))))
;;; M Block Folding with hideshow =============================================
(defun matlab-hideshow-forward-sexp-func (arg)
"Move forward one sexp for hideshow.
Argument ARG specifies the number of blocks to move forward."
(beginning-of-line)
(matlab-forward-sexp arg)
)
(defun matlab-hideshow-adjust-beg-func (arg)
"Adjust the beginning of a hideshow block.
Argument ARG to make it happy."
(end-of-line)
(point)
)
;; Use this to enable hideshow in MATLAB.
;; It has not been tested by me enough.
;; REMOVE PUSHNEW FROM THIS LINE
;;(pushnew (list 'matlab-mode
2005-12-01 20:08:16 +01:00
;; (matlab-block-beg-pre)
;; (matlab-block-end-pre)
;; "%"
;; 'matlab-hideshow-forward-sexp-func
;; 'matlab-hideshow-adjust-beg-func
;; )
;; hs-special-modes-alist :test 'equal)
;;; M Code verification & Auto-fix ============================================
(defun matlab-mode-verify-fix-file-fn ()
"Verify the current buffer from `write-contents-hooks'."
(if matlab-verify-on-save-flag
(matlab-mode-verify-fix-file (> (point-max)
matlab-block-verify-max-buffer-size)))
;; Always return nil.
nil)
(defun matlab-mode-verify-fix-file (&optional fast)
"Verify the current buffer satisfies all M things that might be useful.
We will merely loop across a list of verifiers/fixers in
`matlab-mode-verify-fix-functions'.
If optional FAST is non-nil, do not perform usually lengthy checks."
(interactive)
(let ((p (point))
(l matlab-mode-verify-fix-functions))
(while l
(funcall (car l) fast)
(setq l (cdr l)))
(goto-char p))
(if (matlab-called-interactively-p)
2005-12-01 20:08:16 +01:00
(message "Done.")))
(defun matlab-toggle-show-mlint-warnings ()
"Toggle `matlab-show-mlint-warnings'."
(interactive)
(setq matlab-show-mlint-warnings (not matlab-show-mlint-warnings))
(if matlab-highlight-cross-function-variables
(if matlab-show-mlint-warnings
(mlint-buffer) ; became true, recompute mlint info
(mlint-clear-warnings))) ; became false, just remove highlighting
;; change mlint mode altogether
(mlint-minor-mode
(if (or matlab-highlight-cross-function-variables
matlab-show-mlint-warnings)
1 -1)))
2005-12-01 20:08:16 +01:00
(defun matlab-toggle-highlight-cross-function-variables ()
"Toggle `matlab-highlight-cross-function-variables'."
(interactive)
(setq matlab-highlight-cross-function-variables
(not matlab-highlight-cross-function-variables))
(if matlab-show-mlint-warnings
(if matlab-highlight-cross-function-variables
(mlint-buffer) ; became true, recompute mlint info
; became false, just remove highlighting ...
(mlint-clear-cross-function-variable-highlighting)))
(mlint-minor-mode
(if (or matlab-highlight-cross-function-variables
matlab-show-mlint-warnings)
1 -1))) ; change mlint mode altogether
2005-12-01 20:08:16 +01:00
;;
;; Add more auto verify/fix functions here!
;;
(defun matlab-mode-vf-functionname (&optional fast)
"Verify/Fix the function name of this file.
Optional argument FAST is ignored."
(matlab-navigation-syntax
(goto-char (point-min))
(while (and (or (matlab-ltype-empty) (matlab-ltype-comm))
(/= (matlab-point-at-eol) (point-max)))
(forward-line 1))
(let ((func nil)
(bn (file-name-sans-extension
(file-name-nondirectory (buffer-file-name)))))
(if (looking-at (matlab-match-function-re))
;; The expression above creates too many numeric matches
;; to apply a known one to our function. We cheat by knowing that
;; match-end 0 is at the end of the function name. We can then go
;; backwards, and get the extents we need. Navigation syntax
;; lets us know that backward-word really covers the word.
(let ((end (match-end 0))
(begin (progn (goto-char (match-end 0))
(forward-word -1)
(point))))
(setq func (buffer-substring-no-properties begin end))
2005-12-01 20:08:16 +01:00
(if (not (string= func bn))
(if (not (matlab-mode-highlight-ask
begin end
"Function and file names are different. Fix?"))
nil
(goto-char begin)
(delete-region begin end)
(insert bn))))))))
(defun matlab-mode-vf-classname (&optional fast)
"Verify/Fix the class name of this file.
Optional argument FAST is ignored."
(matlab-navigation-syntax
(goto-char (point-min))
;; Skip over whitespace.
(while (and (or (matlab-ltype-empty) (matlab-ltype-comm))
(/= (matlab-point-at-eol) (point-max)))
(forward-line 1))
(let ((class nil)
(bn (file-name-sans-extension
(file-name-nondirectory (buffer-file-name)))))
(if (looking-at (matlab-match-classdef-re))
;; The name of this class is match 2.
(let ((end (match-end 2))
(begin (match-beginning 2)))
(setq class (buffer-substring-no-properties begin end))
(if (not (string= class bn))
(if (not (matlab-mode-highlight-ask
begin end
"Class name and file names are different. Fix?"))
nil
(goto-char begin)
(delete-region begin end)
(insert bn))))))))
2005-12-01 20:08:16 +01:00
(defun matlab-mode-vf-block-matches-forward (&optional fast)
"Verify/Fix unterminated (or un-ended) blocks.
This only checks block regions like if/end.
Optional argument FAST causes this check to be skipped."
(goto-char (point-min))
(let ((go t)
(expr (concat "\\<\\(" (matlab-block-beg-pre) "\\)\\>")))
(matlab-navigation-syntax
(while (and (not fast) go (re-search-forward expr nil t))
(forward-word -1) ;back over the special word
(let ((s (point)))
(condition-case nil
(if (and (not (matlab-cursor-in-string-or-comment))
(not (looking-at "function")))
(progn
(matlab-forward-sexp)
(forward-word -1)
(if (not (looking-at
(concat matlab-block-end-pre-no-if "\\>")))
(setq go nil)))
(forward-word 1))
(error (setq go nil)))
(if (and (not go) (goto-char s)
(not (matlab-mode-highlight-ask
(point) (save-excursion (forward-word 1) (point))
"Unterminated block. Continue anyway?")))
(error "Unterminated Block found!")))
(message "Block-check: %d%%" (/ (/ (* 100 (point)) (point-max)) 2))))))
2005-12-01 20:08:16 +01:00
(defun matlab-mode-vf-block-matches-backward (&optional fast)
"Verify/fix unstarted (or dangling end) blocks.
Optional argument FAST causes this check to be skipped."
(goto-char (point-max))
(let ((go t) (expr (concat "\\<\\(" (matlab-block-end-no-function-re)
"\\)\\>")))
(matlab-navigation-syntax
(while (and (not fast) go (re-search-backward expr nil t))
(forward-word 1)
(let ((s (point)))
(condition-case nil
(if (and (not (matlab-cursor-in-string-or-comment))
(matlab-valid-end-construct-p))
(matlab-backward-sexp)
(backward-word 1))
(error (setq go nil)))
(if (and (not go) (goto-char s)
(not (matlab-mode-highlight-ask
(point) (save-excursion (backward-word 1) (point))
"Unstarted block. Continue anyway?")))
(error "Unstarted Block found!")))
(message "Block-check: %d%%"
(+ (/ (/ (* 100 (- (point-max) (point))) (point-max)) 2) 50))))))
;;; Utility for verify/fix actions if you need to highlight
;; a section of the buffer for the user's approval.
(defun matlab-mode-highlight-ask (begin end prompt)
"Highlight from BEGIN to END while asking PROMPT as a yes-no question."
(let ((mo (matlab-make-overlay begin end (current-buffer)))
(ans nil))
(condition-case nil
(progn
(matlab-overlay-put mo 'face 'matlab-region-face)
(setq ans (y-or-n-p prompt))
(matlab-delete-overlay mo))
(quit (matlab-delete-overlay mo) (error "Quit")))
ans))
;;; Quiesce an M file to remove accidental display of ANS during a run.
;; Useful if you have random outputs and you don't know where they are from,
;; or before compiling to standalone where some functions now have outputs
;; that did not have outputs earlier.
;;
;; You probably don't want this as a default verify function
(defvar matlab-quiesce-nosemi-regexp "\\s-*\\(function\\|parfor\\|for\\|spmd\\|while\\|try\\|catch\\|\
2005-12-01 20:08:16 +01:00
switch\\|otherwise\\|case\\|break\\|if\\|else\\|end\\|return\\|disp\\|\
$\\|%\\)"
"Regular expression used to detect if a semicolon is needed at the end of a line.")
(defun matlab-mode-vf-quiesce-buffer (&optional fast)
"Find all commands that do not end in ;, and add one.
This has the effect of removing any extraneous output that may not be
desired. Optional argument FAST is not used."
(interactive)
(save-excursion
(push-mark)
(goto-char (point-min))
(let ((msgpos 0) (dir .2))
(while (not (save-excursion (end-of-line) (eobp)))
(message (aref [ "Scanning o...." "Scanning .o..." "Scanning ..o.."
"Scanning ...o." "Scanning ....o" ] (floor msgpos)))
(setq msgpos (+ msgpos dir))
(if (or (> msgpos 5) (< msgpos 0)) (setq dir (- dir)
msgpos (+ (* 2 dir) msgpos)))
(matlab-end-of-command (point))
(if (matlab-cursor-in-comment)
(progn
(matlab-comment-on-line)
(skip-chars-backward " \t")))
(if (and (not (= (preceding-char) ?\;))
(not (matlab-cursor-in-string t))
(not (save-excursion
(beginning-of-line)
(looking-at matlab-quiesce-nosemi-regexp))))
(let ((p (point)))
(skip-chars-backward " \t")
(if (/= p (point))
(progn
(delete-region p (point))
(forward-line -1))
(if (matlab-mode-highlight-ask (point) (+ 1 (point))
"Add Semi colon here? ")
(insert ";")))))
(forward-line 1))))
(message "Scanning .... done"))
2005-12-01 20:08:16 +01:00
;;; V19 stuff =================================================================
;; XXX is the eei code still in use?
(defvar matlab-eei-process) ;; quiet compiler warning. XXX where is matlab-eei-process
(defun matlab-with-emacs-link ()
"Return non-nil if Emacs Link is running and user wants to use it."
(and (featurep 'matlab-eei) ;; XXX where is matlab-eei
matlab-use-eei
matlab-eei-process))
2005-12-01 20:08:16 +01:00
(defvar matlab-mode-menu-keymap nil
"Keymap used in MATLAB mode to provide a menu.")
(defun matlab-frame-init ()
"Initialize Emacs menu system."
2005-12-01 20:08:16 +01:00
(interactive)
;; make a menu keymap
(easy-menu-define
matlab-mode-menu
matlab-mode-map
"MATLAB menu"
'("MATLAB"
["Start MATLAB" matlab-shell
:active (not (or (matlab-with-emacs-link) (matlab-shell-active-p)))
:visible (not (matlab-shell-active-p)) ]
["Switch to MATLAB" matlab-shell
:active (and (not (matlab-with-emacs-link)) (matlab-shell-active-p))
:visible (matlab-shell-active-p) ]
2005-12-01 20:08:16 +01:00
["Save and go" matlab-shell-save-and-go t]
["Run Region" matlab-shell-run-region t]
["Run Cell" matlab-shell-run-cell t]
2005-12-01 20:08:16 +01:00
["Version" matlab-show-version t]
"----"
["Find M file" matlab-find-file-on-path t]
["Show M-Lint Warnings" matlab-toggle-show-mlint-warnings
:active (and (locate-library "mlint") (fboundp 'mlint-minor-mode))
:style toggle :selected matlab-show-mlint-warnings
]
("Auto Fix"
["Verify/Fix source" matlab-mode-verify-fix-file t]
["Spell check strings" matlab-ispell-strings t]
["Spell check comments" matlab-ispell-comments t]
["Quiesce source" matlab-mode-vf-quiesce-buffer t]
)
("Navigate"
["Beginning of Command" matlab-beginning-of-command t]
["End of Command" matlab-end-of-command t]
["Forward Block" matlab-forward-sexp t]
["Backward Block" matlab-backward-sexp t]
["Beginning of Function" matlab-beginning-of-defun t]
["End of Function" matlab-end-of-defun t])
("Format"
["Justify Line" matlab-justify-line t]
["Fill Region" matlab-fill-region t]
["Fill Comment Paragraph" matlab-fill-paragraph
(save-excursion (matlab-comment-on-line))]
["Join Comment" matlab-join-comment-lines
(save-excursion (matlab-comment-on-line))]
["Comment Region" matlab-comment-region t]
["Uncomment Region" matlab-uncomment-region t]
["Indent Syntactic Block" matlab-indent-sexp])
2005-12-01 20:08:16 +01:00
("Insert"
["Complete Symbol" matlab-complete-symbol t]
["Comment" matlab-comment t]
["if end" tempo-template-matlab-if t]
["if else end" tempo-template-matlab-if-else t]
["for end" tempo-template-matlab-for t]
["switch otherwise end" tempo-template-matlab-switch t]
["Next case" matlab-insert-next-case t]
["try catch end" tempo-template-matlab-try t]
["while end" tempo-template-matlab-while t]
["End of block" matlab-insert-end-block t]
["Function" tempo-template-matlab-function t]
["Stringify Region" matlab-stringify-region t]
)
("Customize"
; ["Auto Fill Counts Elipsis"
; (lambda () (setq matlab-fill-count-ellipsis-flag
; (not matlab-fill-count-ellipsis-flag)))
; :style toggle :selected 'matlab-fill-count-ellipsis-flag]
["Indent Function Body"
(setq matlab-indent-function-body (not (matlab-indent-function-body-p)))
2005-12-01 20:08:16 +01:00
:style toggle :selected matlab-indent-function-body]
["Functions Have end"
matlab-toggle-functions-have-end
:style toggle :selected matlab-functions-have-end]
["Verify File on Save"
(setq matlab-verify-on-save-flag (not matlab-verify-on-save-flag))
:style toggle :selected matlab-verify-on-save-flag]
["Auto Fill does Code"
(setq matlab-fill-code (not matlab-fill-code))
:style toggle :selected matlab-fill-code ]
["Periodic Code Details"
(setq matlab-show-periodic-code-details-flag
(not matlab-show-periodic-code-details-flag))
:style toggle :selected matlab-show-periodic-code-details-flag ]
["Highlight Matching Blocks"
(matlab-enable-block-highlighting)
:style toggle :selected (member 'matlab-start-block-highlight-timer
post-command-hook) ]
["Highlight Cross-Function Variables"
matlab-toggle-highlight-cross-function-variables
:active (locate-library "mlint")
:style toggle :selected matlab-highlight-cross-function-variables
]
["Add Needed Semicolon on RET"
(setq matlab-return-add-semicolon (not matlab-return-add-semicolon))
:style toggle :selected matlab-return-add-semicolon
]
["Customize" (customize-group 'matlab)
(and (featurep 'custom) (fboundp 'custom-declare-variable))
]
)
"----"
["Run M Command" matlab-shell-run-command (matlab-shell-active-p)]
["Describe Command" matlab-shell-describe-command (matlab-shell-active-p)]
["Describe Variable" matlab-shell-describe-variable (matlab-shell-active-p)]
["Command Apropos" matlab-shell-apropos (matlab-shell-active-p)]
["Topic Browser" matlab-shell-topic-browser (matlab-shell-active-p)]
))
(easy-menu-add matlab-mode-menu matlab-mode-map))
;;; MATLAB shell =============================================================
(defgroup matlab-shell nil
"MATLAB shell mode."
:prefix "matlab-shell-"
:group 'matlab)
(defcustom matlab-shell-command "matlab"
"*The name of the command to be run which will start the MATLAB process."
:group 'matlab-shell
:type 'string)
(defcustom matlab-shell-command-switches '("-nodesktop")
"*Command line parameters run with `matlab-shell-command'.
Command switches are a list of strings. Each entry is one switch."
2005-12-01 20:08:16 +01:00
:group 'matlab-shell
:type '(list :tag "Switch: "))
(defcustom matlab-shell-echoes t
"*If `matlab-shell-command' echoes input."
:group 'matlab-shell
:type 'boolean)
(defvar matlab-shell-running-matlab-version nil
"The version of MATLAB running in the current `matlab-shell' buffer.")
(defvar matlab-shell-running-matlab-release nil
"The release of MATLAB running in the current `matlab-shell' buffer.")
(defvar matlab-shell-use-emacs-toolbox
;; matlab may not be on path. (Name change, explicit load, etc)
(let* ((mlfile (locate-library "matlab"))
(dir (expand-file-name "toolbox/emacsinit.m"
(file-name-directory (or mlfile "")))))
(and mlfile (file-exists-p dir)))
"Add the `matlab-shell' MATLAB toolbox to the MATLAB path on startup.")
(defcustom matlab-shell-emacsclient-command "emacsclient -n"
"*The command to use as an external editor for MATLAB.
Using emacsclient allows the currently running Emacs to also be the
external editor for MATLAB. Setting this to the empty string
will disable use emacsclient as the external editor."
:group 'matlab-shell
:type 'integer)
(defvar matlab-custom-startup-command nil
"Custom matlab command to be run at startup")
(defcustom matlab-shell-history-file "~/.matlab/%s/history.m"
2005-12-01 20:08:16 +01:00
"*Location of the history file.
A %s is replaced with the MATLAB version release number, such as R12.
2005-12-01 20:08:16 +01:00
This file is read to initialize the comint input ring.")
(defcustom matlab-shell-input-ring-size 32
"*Number of history elements to keep."
:group 'matlab-shell
:type 'integer)
(defcustom matlab-shell-enable-gud-flag t
"*Non-nil means to use GUD mode when running the MATLAB shell."
:group 'matlab-shell
:type 'boolean)
(defcustom matlab-shell-mode-hook nil
"*List of functions to call on entry to MATLAB shell mode."
:group 'matlab-shell
:type 'hook)
(defcustom matlab-shell-tab-use-company t
"*Use `company' (complete anything) for TAB completions in
MATLAB shell when `company' is installed. Note, when you type to
narrow completions, you may find the responses slow and if so,
you can try turning this off."
:group 'matlab-shell
:type 'boolean)
(defvar matlab-shell-tab-company-available nil
"If we have `company' (completion anything) use it to show
`matlab-shell' completions when `matlab-shell-tab-use-company' is t.")
(when (require 'company nil 'noerror)
(setq matlab-shell-tab-company-available t))
(defcustom matlab-shell-ask-MATLAB-for-completions t
2005-12-01 20:08:16 +01:00
"When Non-nil, ask MATLAB for a completion list.
When nil, complete against file names."
2005-12-01 20:08:16 +01:00
:group 'matlab-shell
:type 'boolean)
(defvar matlab-shell-buffer-name "MATLAB"
"Name used to create `matlab-shell' mode buffers.
This name will have *'s surrounding it.")
(declare-function comint-check-proc "comint.el" (buffer))
2005-12-01 20:08:16 +01:00
(defun matlab-shell-active-p ()
"Return t if the MATLAB shell is active."
(let ((msbn (get-buffer (concat "*" matlab-shell-buffer-name "*"))))
(if msbn
(with-current-buffer msbn
(if (comint-check-proc (current-buffer))
(current-buffer))))))
2005-12-01 20:08:16 +01:00
(defvar matlab-shell-mode-map ()
"Keymap used in `matlab-shell-mode'.")
(defvar matlab-shell-font-lock-keywords-1
(append matlab-font-lock-keywords matlab-shell-font-lock-keywords)
"Keyword symbol used for font-lock mode.")
(defvar matlab-shell-font-lock-keywords-2
(append matlab-shell-font-lock-keywords-1 matlab-gaudy-font-lock-keywords)
"Keyword symbol used for gaudy font-lock symbols.")
(defvar matlab-shell-font-lock-keywords-3
(append matlab-shell-font-lock-keywords-2
matlab-really-gaudy-font-lock-keywords)
"Keyword symbol used for really gaudy font-lock symbols.")
(defvar matlab-prompt-seen nil
"Track visibility of MATLAB prompt in MATLAB Shell.")
(eval-and-compile (require 'gud) (require 'comint) (require 'shell))
2005-12-01 20:08:16 +01:00
;;;###autoload
(defun matlab-shell ()
"Create a buffer with MATLAB running as a subprocess.
MATLAB shell cannot work on the MS Windows platform because MATLAB is not
a console application."
(interactive)
;; MATLAB shell does not work by default on the Windows platform. Only
;; permit it's operation when the shell command string is different from
;; the default value. (True when the engine program is running.)
(if (and (or (eq window-system 'pc) (eq window-system 'w32))
(string= matlab-shell-command "matlab"))
(error "MATLAB cannot be run as a inferior process. \
Try C-h f matlab-shell RET"))
(require 'shell)
(require 'gud)
;; Make sure this is safe to use gud to debug MATLAB
(when (and matlab-shell-enable-gud-flag (not (fboundp 'gud-def)))
(message "Your emacs is missing `gud-def' which means matlab-shell won't work correctly.")
2005-12-01 20:08:16 +01:00
(setq matlab-shell-enable-gud-flag nil))
(switch-to-buffer (concat "*" matlab-shell-buffer-name "*"))
(if (matlab-shell-active-p)
nil
;; Clean up crufty state
(kill-all-local-variables)
;; Build keymap here in case someone never uses comint mode
(if matlab-shell-mode-map
()
(setq matlab-shell-mode-map
(let ((km (make-sparse-keymap 'matlab-shell-mode-map)))
(if (fboundp 'set-keymap-parent)
(set-keymap-parent km comint-mode-map)
;; 19.31 doesn't have set-keymap-parent
(setq km (nconc km comint-mode-map)))
(substitute-key-definition 'next-error 'matlab-shell-last-error
km global-map)
(define-key km [(control h) (control m)]
matlab-help-map)
(define-key km "\C-c." 'matlab-find-file-on-path)
(if (not matlab-shell-tab-company-available)
(progn
(define-key km "\C-i" 'matlab-shell-c-tab)
(define-key km (kbd "TAB") 'matlab-shell-c-tab)) ;; classic emacs tab complete
;; else have both company (popup) tab complete and classic tab complete
(define-key km (kbd "TAB") 'matlab-shell-tab)
(define-key km "\C-i" 'matlab-shell-tab)
(define-key km (kbd "<C-tab>") 'matlab-shell-c-tab))
2005-12-01 20:08:16 +01:00
(define-key km [(control up)]
'comint-previous-matching-input-from-input)
(define-key km [(control down)]
'comint-next-matching-input-from-input)
(define-key km [up]
'matlab-shell-previous-matching-input-from-input)
2005-12-01 20:08:16 +01:00
(define-key km [down]
'matlab-shell-next-matching-input-from-input)
2005-12-01 20:08:16 +01:00
(define-key km [(control return)] 'comint-kill-input)
(define-key km "\C-?"
'matlab-shell-delete-backwards-no-prompt)
(define-key km [(backspace)]
'matlab-shell-delete-backwards-no-prompt)
km)))
(switch-to-buffer
;; Thx David Chappaz for reminding me about this patch.
(let* ((windowid (frame-parameter (selected-frame) 'outer-window-id))
(newvar (concat "WINDOWID=" windowid))
(process-environment (cons newvar process-environment)))
(apply 'make-comint matlab-shell-buffer-name matlab-shell-command
nil matlab-shell-command-switches)))
2005-12-01 20:08:16 +01:00
(setq shell-dirtrackp t)
(comint-mode)
(when matlab-shell-enable-gud-flag
(gud-mode)
(make-local-variable 'matlab-prompt-seen)
(setq matlab-prompt-seen nil)
(make-local-variable 'gud-marker-filter)
(setq gud-marker-filter 'gud-matlab-marker-filter)
(make-local-variable 'gud-find-file)
(setq gud-find-file 'gud-matlab-find-file)
(set-process-filter (get-buffer-process (current-buffer))
'gud-filter)
(set-process-sentinel (get-buffer-process (current-buffer))
'gud-sentinel)
(gud-set-buffer))
2005-12-01 20:08:16 +01:00
;; Comint and GUD both try to set the mode. Now reset it to
;; matlab mode.
(matlab-shell-mode)))
(defcustom matlab-shell-logo
(if (fboundp 'locate-data-file)
;; Starting from XEmacs 20.4 use locate-data-file
(locate-data-file "matlab.xpm")
(expand-file-name "matlab.xpm" data-directory))
"*The MATLAB logo file."
:group 'matlab-shell
:type '(choice (const :tag "None" nil)
(file :tag "File" "")))
2005-12-01 20:08:16 +01:00
(defun matlab-shell-hack-logo (str)
"Replace the text logo with a real logo.
STR is passed from the commint filter."
(when (string-match "< M A T L A B >" str)
(save-excursion
(when (re-search-backward "^[ \t]+< M A T L A B (R) >" (point-min) t)
(delete-region (match-beginning 0) (match-end 0))
(insert (make-string 16 ? ))
(when (fboundp 'set-extent-begin-glyph)
(set-extent-begin-glyph (when (fboundp 'make-extent) (make-extent (point) (point)))
(when (fboundp 'make-glyph) (make-glyph matlab-shell-logo))))))
;; Remove this function from `comint-output-filter-functions'
(remove-hook 'comint-output-filter-functions
'matlab-shell-hack-logo))
)
(defvar matlab-shell-last-error-anchor) ;; Quiet compiler warning
(defun matlab-shell-version-scrape (str)
"Scrape the MATLAB Version from the MATLAB startup text.
Argument STR is the string to examine for version information."
(when (string-match "\\(Version\\)\\s-+\\([.0-9]+\\)\\s-+(\\(R[.0-9]+[ab]?\\))" str)
;; Extract the release number
(setq matlab-shell-running-matlab-version
(match-string 2 str)
matlab-shell-running-matlab-release
(match-string 3 str))
;; Now get our history loaded
(setq comint-input-ring-file-name
(format matlab-shell-history-file matlab-shell-running-matlab-release))
(if (fboundp 'comint-read-input-ring)
(comint-read-input-ring t))
;; Remove the scrape from our list of things to do.
(remove-hook 'comint-output-filter-functions
'matlab-shell-version-scrape)))
2005-12-01 20:08:16 +01:00
(defun matlab-shell-mode ()
"Run MATLAB as a subprocess in an Emacs buffer.
This mode will allow standard Emacs shell commands/completion to occur
with MATLAB running as an inferior process. Additionally, this shell
mode is integrated with `matlab-mode', a major mode for editing M
code.
> From an M file buffer:
\\<matlab-mode-map>
\\[matlab-shell-save-and-go] - Save the current M file, and run it in a \
MATLAB shell.
> From Shell mode:
\\<matlab-shell-mode-map>
\\[matlab-shell-last-error] - find location of last MATLAB runtime error \
in the offending M file.
> From an M file, or from Shell mode:
\\<matlab-mode-map>
\\[matlab-shell-run-command] - Run COMMAND and show result in a popup buffer.
\\[matlab-shell-describe-variable] - Show variable contents in a popup buffer.
\\[matlab-shell-describe-command] - Show online documentation for a command \
in a popup buffer.
\\[matlab-shell-apropos] - Show output from LOOKFOR command in a popup buffer.
\\[matlab-shell-topic-browser] - Topic browser using HELP.
> Keymap:
\\{matlab-mode-map}"
(setq major-mode 'matlab-shell-mode
mode-name "M-Shell"
comint-prompt-regexp "^\\(K\\|EDU\\)?>> *"
comint-delimiter-argument-list (list [ 59 ]) ; semi colon
comint-dynamic-complete-functions '(comint-replace-by-expanded-history)
comint-process-echoes matlab-shell-echoes
)
;; matlab-shell variable setup
(make-local-variable 'matlab-shell-last-error-anchor)
(setq matlab-shell-last-error-anchor nil)
;; Shell Setup
2005-12-01 20:08:16 +01:00
(require 'shell)
(if (fboundp 'shell-directory-tracker)
(add-hook 'comint-input-filter-functions 'shell-directory-tracker nil t)) ;; patch Eli Merriam
2005-12-01 20:08:16 +01:00
;; Add a spiffy logo if we are running XEmacs
(if (and (string-match "XEmacs" emacs-version)
(stringp matlab-shell-logo)
(file-readable-p matlab-shell-logo))
(add-hook 'comint-output-filter-functions 'matlab-shell-hack-logo))
;; Add a version scraping logo identification filter.
(add-hook 'comint-output-filter-functions 'matlab-shell-version-scrape)
2005-12-01 20:08:16 +01:00
;; Add pseudo html-renderer
(add-hook 'comint-output-filter-functions 'matlab-shell-render-html-anchor nil t)
;(add-hook 'comint-output-filter-functions 'matlab-shell-render-html-txt-format nil t)
;(add-hook 'comint-output-filter-functions 'matlab-shell-render-errors-as-anchor nil t)
;; Scroll to bottom after running cell/region
(add-hook 'comint-output-filter-functions 'comint-postoutput-scroll-to-bottom)
2005-12-01 20:08:16 +01:00
(make-local-variable 'comment-start)
(setq comment-start "%")
(use-local-map matlab-shell-mode-map)
(set-syntax-table matlab-mode-syntax-table)
(make-local-variable 'font-lock-defaults)
(setq font-lock-defaults '((matlab-shell-font-lock-keywords-1
matlab-shell-font-lock-keywords-2
matlab-shell-font-lock-keywords-3)
t nil ((?_ . "w"))))
(set (make-local-variable 'comint-input-ring-size)
matlab-shell-input-ring-size)
(set (make-local-variable 'comint-input-ring-file-name)
(format matlab-shell-history-file "R12"))
2005-12-01 20:08:16 +01:00
(if (fboundp 'comint-read-input-ring)
(comint-read-input-ring t))
(make-local-variable 'gud-marker-acc)
(easy-menu-define
matlab-shell-menu
matlab-shell-mode-map
"MATLAB shell menu"
'("MATLAB"
["Goto last error" matlab-shell-last-error t]
"----"
["Stop On Errors" matlab-shell-dbstop-error t]
["Don't Stop On Errors" matlab-shell-dbclear-error t]
"----"
["Run Command" matlab-shell-run-command t]
["Describe Variable" matlab-shell-describe-variable t]
["Describe Command" matlab-shell-describe-command t]
["Lookfor Command" matlab-shell-apropos t]
["Topic Browser" matlab-shell-topic-browser t]
"----"
["Complete command" matlab-shell-c-tab t]
["Popup complete command" matlab-shell-tab :visible matlab-shell-tab-company-available]
"----"
2005-12-01 20:08:16 +01:00
["Demos" matlab-shell-demos t]
["Close Current Figure" matlab-shell-close-current-figure t]
["Close Figures" matlab-shell-close-figures t]
"----"
["Customize" (customize-group 'matlab-shell)
(and (featurep 'custom) (fboundp 'custom-declare-variable))
]
["Exit" matlab-shell-exit t]))
(easy-menu-add matlab-shell-menu matlab-shell-mode-map)
2005-12-01 20:08:16 +01:00
(if matlab-shell-enable-gud-flag
(progn
(gud-def gud-break "dbstop at %l in %f" "\C-b" "Set breakpoint at current line.")
(gud-def gud-remove "dbclear at %l in %f" "\C-d" "Remove breakpoint at current line")
(gud-def gud-step "dbstep in;\ndbhotlink(1)" "\C-s" "Step one source line, possibly into a function.")
(gud-def gud-next "dbstep %p;\ndbhotlink(1)" "\C-n" "Step over one source line.")
(gud-def gud-cont "dbcont;\ndbhotlink(1)" "\C-r" "Continue with display.")
2005-12-01 20:08:16 +01:00
(gud-def gud-finish "dbquit" "\C-f" "Finish executing current function.")
;; (gud-def gud-up "dbup %p;\ndbhotlink()" "<" "Up N stack frames (numeric arg).")
;; (gud-def gud-down "dbdown %p;\ndbhotlink()" ">" "Down N stack frames (numeric arg).")
(gud-def gud-up "dbup;\n[~,a___]=dbstack;\ndbhotlink(a___)" "<" "Up N stack frames (numeric arg).")
(gud-def gud-down "dbdown;\n[~,a___]=dbstack;\ndbhotlink(a___)" ">" "Down N stack frames (numeric arg).")
2005-12-01 20:08:16 +01:00
(gud-def gud-print "%e" "\C-p" "Evaluate M expression at point.")
(if (fboundp 'gud-make-debug-menu)
(gud-make-debug-menu))
(if (fboundp 'gud-overload-functions)
(gud-overload-functions
'((gud-massage-args . gud-matlab-massage-args)
(gud-marker-filter . gud-matlab-marker-filter)
(gud-find-file . gud-matlab-find-file))))
;; XEmacs doesn't seem to have this concept already. Oh well.
(setq gud-marker-acc nil)
;; XEmacs has problems w/ this variable. Set it here.
(set-marker comint-last-output-start (point-max))
))
(run-hooks 'matlab-shell-mode-hook)
(matlab-show-version)
)
(defvar gud-matlab-marker-regexp-prefix "error:\\|opentoline"
2005-12-01 20:08:16 +01:00
"A prefix to scan for to know if output might be scarfed later.")
(defvar matlab-shell-html-map
(let ((km (make-sparse-keymap)))
(if (string-match "XEmacs" emacs-version)
(define-key km [button2] 'matlab-shell-html-click)
(define-key km [mouse-2] 'matlab-shell-html-click))
(define-key km [return] 'matlab-shell-html-go)
km)
"Keymap used on overlays that represent errors.")
2005-12-01 20:08:16 +01:00
;; ANCHORS
(defvar matlab-anchor-beg "<a href=\"\\(\\(?:matlab:\\)?[^\"]+\\)\">"
"Beginning of html anchor.")
(defvar matlab-anchor-end "</a>"
"End of html anchor.")
2005-12-01 20:08:16 +01:00
(defun matlab-shell-render-html-anchor (str)
"Render html anchors inserted into the MATLAB shell buffer.
Argument STR is the text for the anchor."
(if (string-match matlab-anchor-end str)
(save-excursion
(while (re-search-backward matlab-anchor-beg
;; Arbitrary back-buffer. We don't
;; usually get text in such huge chunks
(max (point-min) (- (point-max) 8192))
t)
2005-12-01 20:08:16 +01:00
(let* ((anchor-beg-start (match-beginning 0))
(anchor-beg-finish (match-end 0))
(anchor-text (match-string 1))
(anchor-end-finish (search-forward matlab-anchor-end))
(anchor-end-start (match-beginning 0))
(o (matlab-make-overlay anchor-beg-finish anchor-end-start)))
(matlab-overlay-put o 'mouse-face 'highlight)
(matlab-overlay-put o 'face 'underline)
(matlab-overlay-put o 'matlab-url anchor-text)
(matlab-overlay-put o 'keymap matlab-shell-html-map)
(matlab-overlay-put o 'help-echo anchor-text)
2005-12-01 20:08:16 +01:00
(delete-region anchor-end-start anchor-end-finish)
(delete-region anchor-beg-start anchor-beg-finish)
))))
)
;; TEXT FORMATTING
(defvar matlab-txt-format-beg "<\\(strong\\|u\\)>"
"Beginning of html text formatting signal in HTML.")
(defvar matlab-txt-format-end "</%s>"
"End of some html text formatter.
Includes a %s to match the kind of text format start regexp.")
(defun matlab-shell-render-html-txt-format (str)
"Render html text format inserted into the MATLAB shell buffer.
Argument STR is the text for the text formatter."
(if (string-match "</\\w+>" str)
(save-excursion
(while (re-search-backward matlab-txt-format-beg
;; Arbitrary back-buffer. We don't
;; usually get text in such huge chunks
(max (point-min) (- (point-max) 8192))
t)
(let* ((txt-format-beg-start (match-beginning 0))
(txt-format-beg-finish (match-end 0))
(txt-format-text (match-string 1))
(txt-format-end-finish
;; The finish combines the text from the start to get an
;; exact match.
(search-forward (format matlab-txt-format-end txt-format-text)))
(txt-format-end-start (match-beginning 0))
(o (matlab-make-overlay txt-format-beg-finish txt-format-end-start)))
(cond ((string= txt-format-text "strong")
(upcase-region txt-format-beg-finish txt-format-end-start)
(matlab-overlay-put o 'face 'bold))
((string= txt-format-text "u")
(matlab-overlay-put o 'face 'underline))
(t
;; If we don't match, delete the overlay instead.
(matlab-delete-overlay o)
(setq o nil)
))
(when o
(delete-region txt-format-end-start txt-format-end-finish)
(delete-region txt-format-beg-start txt-format-beg-finish))
))))
)
;; The regular expression covers the following form:
;; Errors: Error in ==> <function name>
;; On line # ==> <command_name>
;; Errors: Error using ==> <function name> at <#>
;; Syntax: Syntax error in ==> <filename>
;; On line # ==> <sample-text>
;; Warning: In <filename> at line # <stuff>
;;
;; New Error Formats:
;; Errors: Error in <function name> (line <#>)
(defvar gud-matlab-error-regexp
(if nil
;; Newer MATLAB's use this. Debug and merge with below.
(concat "\\(Error \\(?:in\\|using\\)\\|Syntax error in\\) "
"\\([-@.a-zA-Z_0-9/\\\\:]+\\)[\n ]*(line "
"\\([0-9]+\\)) ?")
;; else
(concat "\\(Error \\(?:in\\|using\\) ==>\\|Syntax error in ==>\\|In\\) "
"\\([-@.a-zA-Z_0-9/ \\\\:]+\\)\\(?:>[^ ]+\\)?.*[\n ]\\(?:On\\|at\\)\\(?: line\\)? "
"\\([0-9]+\\) ?")
) ;; end if
"Regular expression finding where an error occurred.")
(defvar matlab-shell-last-error-anchor nil
"Last point where an error anchor was set.")
(defvar matlab-shell-last-anchor-as-frame nil
;; NOTE: this isn't being used yet.
"The last error anchor saved, represented as a debugger frame.")
(defun matlab-shell-render-errors-as-anchor (str)
"Detect non-url errors, and treat them as if they were url anchors.
Argument STR is the text that might have errors in it."
(save-excursion
;; We have found an error stack to investigate.
(let ((first nil)
(overlaystack nil))
(while (re-search-backward gud-matlab-error-regexp
(if matlab-shell-last-error-anchor
(min matlab-shell-last-error-anchor (point))
(point))
t)
(let* ((err-start (match-beginning 0))
(err-end (match-end 0))
(err-text (match-string 0))
(err-file (match-string 2))
(err-line (match-string 3))
(o (matlab-make-overlay err-start err-end))
(url (concat "opentoline('" err-file "'," err-line ",0)"))
)
(matlab-overlay-put o 'mouse-face 'highlight)
(matlab-overlay-put o 'face 'underline)
;; The url will recycle opentoline code.
(matlab-overlay-put o 'matlab-url url)
(matlab-overlay-put o 'keymap matlab-shell-html-map)
(matlab-overlay-put o 'help-echo (concat "Jump to error at " err-file "."))
(setq first url)
(push o overlaystack)
;; Save as a frame
(setq matlab-shell-last-anchor-as-frame
(cons err-file err-line))
))
;; Keep track of the very first error in this error stack.
;; It will represent the "place to go" for "go-to-last-error".
(dolist (O overlaystack)
(matlab-overlay-put O 'first-in-error-stack first))
;; Once we've found something, don't scan it again.
(setq matlab-shell-last-error-anchor (point-marker)))))
(defvar gud-matlab-marker-regexp-1 "^K>>"
"Regular expression for finding a file line-number.")
(defvar gud-matlab-marker-regexp-2
(concat "^> In \\(" matlab-anchor-beg
"\\|\\)\\([-.a-zA-Z0-9_>/@]+\\) \\((\\w+) \\|\\)at line \\([0-9]+\\)[ \n]+")
"Regular expression for finding a file line-number.
Please note: The leading > character represents the current stack frame, so if
there are several frames, this makes sure we pick the right one to popup.")
2005-12-01 20:08:16 +01:00
(defun gud-matlab-massage-args (file args)
"Argument message for starting matlab file.
2005-12-01 20:08:16 +01:00
I don't think I have to do anything, but I'm not sure.
FILE is ignored, and ARGS is returned."
args)
(defun gud-matlab-marker-filter (string)
"Filters STRING for the Unified Debugger based on MATLAB output."
2005-12-01 20:08:16 +01:00
(if matlab-prompt-seen
nil
(when (string-match ">> " string)
(if matlab-shell-use-emacs-toolbox
;; Use our local toolbox directory.
(process-send-string
(get-buffer-process gud-comint-buffer)
(format "addpath('%s','-begin'); rehash; emacsinit('%s');\n"
(expand-file-name "toolbox"
(file-name-directory
(locate-library "matlab")))
matlab-shell-emacsclient-command))
;; Setup is misconfigured - we need emacsinit because it tells us how to debug
(error "unable to initialize matlab, emacsinit.m and other files missing"))
(if matlab-custom-startup-command
(process-send-string
(get-buffer-process gud-comint-buffer)
(concat matlab-custom-startup-command "\n")))
;; Mark that we've seen at least one prompt.
(setq matlab-prompt-seen t)
))
2005-12-01 20:08:16 +01:00
(let ((garbage (concat "\\(" (regexp-quote "\C-g") "\\|"
(regexp-quote "\033[H0") "\\|"
(regexp-quote "\033[H\033[2J") "\\|"
(regexp-quote "\033H\033[2J") "\\)")))
(while (string-match garbage string)
(if (= (aref string (match-beginning 0)) ?\C-g)
(beep t))
(setq string (replace-match "" t t string))))
2005-12-01 20:08:16 +01:00
(setq gud-marker-acc (concat gud-marker-acc string))
(let ((output "") (frame nil))
(when (not frame)
(when (string-match gud-matlab-marker-regexp-1 gud-marker-acc)
(when (not frame)
;; If there is a debug prompt, and no frame currently set,
;; go find one.
(let ((url gud-marker-acc)
ef el)
(cond
((string-match "^error:\\(.*\\),\\([0-9]+\\),\\([0-9]+\\)$" url)
(setq ef (substring url (match-beginning 1) (match-end 1))
el (substring url (match-beginning 2) (match-end 2)))
)
((string-match "opentoline('\\([^']+\\)',\\([0-9]+\\),\\([0-9]+\\))" url)
(setq ef (substring url (match-beginning 1) (match-end 1))
el (substring url (match-beginning 2) (match-end 2)))
)
;; If we have the prompt, but no match (as above),
;; perhaps it is already dumped out into the buffer. In
;; that case, look back through the buffer.
)
(when ef
(setq frame (cons ef (string-to-number el)))))))
)
;; This if makes sure that the entirety of an error output is brought in
;; so that matlab-shell-mode doesn't try to display a file that only partially
;; exists in the buffer. Thus, if MATLAB output:
;; error: /home/me/my/mo/mello.m,10,12
;; All of that is in the buffer, and it goes to mello.m, not just
;; the first half of that file name.
;; The below used to match against the prompt, not \n, but then text that
;; had error: in it for some other reason wouldn't display at all.
(if (and matlab-prompt-seen ;; Don't collect during boot
(not frame) ;; don't collect debug stuff
(let ((start (string-match gud-matlab-marker-regexp-prefix gud-marker-acc)))
(and start
(not (string-match "\n" gud-marker-acc start))
;;(not (string-match "^K?>>\\|\\?\\?\\?\\s-Error while evaluating" gud-marker-acc start))
)))
2005-12-01 20:08:16 +01:00
;; We could be collecting something. Wait for a while.
nil
;; Finish off this part of the output. None of our special stuff
;; ends with a \n, so display those as they show up...
(while (string-match "^[^\n]*\n" gud-marker-acc)
(setq output (concat output (substring gud-marker-acc 0 (match-end 0)))
gud-marker-acc (substring gud-marker-acc (match-end 0))))
2005-12-01 20:08:16 +01:00
(setq output (concat output gud-marker-acc)
gud-marker-acc "")
2005-12-01 20:08:16 +01:00
;; Check our output for a prompt, and existence of a frame.
;; If t his is true, throw out the debug arrow stuff.
(if (and (string-match "^>> $" output)
gud-last-last-frame)
(progn
(setq overlay-arrow-position nil
gud-last-last-frame nil
gud-overlay-arrow-position nil)
2005-12-01 20:08:16 +01:00
(sit-for 0)
)))
(if frame (setq gud-last-frame frame))
;;(message "[%s] [%s]" output gud-marker-acc)
output))
(defun gud-matlab-find-file (f)
"Find file F when debugging frames in MATLAB."
(save-excursion
(let* ((realfname (if (string-match "\\.\\(p\\)$" f)
(progn
(aset f (match-beginning 1) ?m)
f)
f))
(buf (find-file-noselect realfname)))
(set-buffer buf)
(if (fboundp 'gud-make-debug-menu)
(gud-make-debug-menu))
buf)))
(defun matlab-shell-next-matching-input-from-input (n)
"Get the Nth next matching input from for the command line."
(interactive "p")
(matlab-shell-previous-matching-input-from-input (- n)))
(defun matlab-shell-previous-matching-input-from-input (n)
"Get the Nth previous matching input from for the command line."
(interactive "p")
(end-of-line) ;; patch: Mark Histed
(if (comint-after-pmark-p)
(if (memq last-command '(matlab-shell-previous-matching-input-from-input
matlab-shell-next-matching-input-from-input))
;; This hack keeps the cycling working well.
(let ((last-command 'comint-previous-matching-input-from-input))
(comint-next-matching-input-from-input (- n)))
;; first time.
(comint-next-matching-input-from-input (- n)))
;; If somewhere else, just move around.
(forward-line (- n))))
2005-12-01 20:08:16 +01:00
(defun matlab-shell-delete-backwards-no-prompt (&optional arg)
"Delete one char backwards without destroying the matlab prompt.
Optional argument ARG describes the number of chars to delete."
(interactive "P")
(let ((promptend (save-excursion
(beginning-of-line)
(if (looking-at "K?>> ")
(match-end 0)
(point))))
(numchars (if (integerp arg) (- arg) -1)))
(if (<= promptend (+ (point) numchars))
(delete-char numchars)
(error "Beginning of line"))))
(defun matlab-shell-completion-list (str)
"Get a list of completions from MATLAB.
STR is a command substring to complete."
(let* ((msbn (matlab-shell-buffer-barf-not-running))
(cmd (concat "emacsdocomplete('" str "')"))
(comint-scroll-show-maximum-output nil)
output
(replacement-text "")
(cmd-text-to-replace "")
(completions nil))
(with-current-buffer msbn
(if (not (matlab-on-prompt-p))
(error "MATLAB shell must be non-busy to do that"))
(setq output (matlab-shell-collect-command-output cmd))
(if (not (string-match "emacs_completions_output =" output))
(error "Internal error, '%s' returned unexpected output, %s" cmd output))
2005-12-01 20:08:16 +01:00
(setq output (substring output (match-end 0)))
(when (string-match "^'\\([^']+\\)' --> '\\([^']*\\)'" output)
;; 'CMD_TEXT_TO_REPLACE' --> 'REPLACEMENT_TEXT'
;; 'OPTION1'
;; 'OPTION2'
;; ...
;; Note, the CMD_TEXT_TO_REPLACE line is only present when there needs
;; to be replacement, e.g. imagine a command that takes glob patterns
;; >> mycmd foo*ba<TAB>
;; 'foo*-bar' --> 'foo-and-bar'
;; '-or-goo'
;; '-or-too'
;; which completes to either 'foo-and-bar-or-goo' OR 'foo-and-bar-or-too'.
;; If there is only one completion that needs replacement, don't have options:
;; >> mycmd foo*ba*-too<TAB>
;; 'foo*ba*-too' --> 'foo-and-bar-or-too'
;; The replacement line is not present when the completion just appends to the
;; command str, e.g.
;; >> mycmd foo-and-bar<TAB>
;; '-or-goo'
;; '-or-too'
(setq cmd-text-to-replace (match-string 1 output))
(setq replacement-text (match-string 2 output))
(setq output (substring output (match-end 0))))
;; Parse the output string.
(while (string-match "'" output)
;; Hack off the preceding quote
(setq output (substring output (match-end 0)))
(string-match "'" output)
;; we are making a completion list, so that is a list of lists.
(setq completions (cons (list (substring output 0 (match-beginning 0)))
completions)
output (substring output (match-end 0))))
;; Return them
(list (cons 'cmd-text-to-replace cmd-text-to-replace)
(cons 'replacement-text replacement-text)
(cons 'completions (nreverse completions)))
)))
(defun matlab-shell-which-fcn (fcn)
"Get the location of FCN's M file.
Returns an alist: ( LOCATION . BUILTINFLAG )
LOCATION is a string indicating where it is, and BUILTINFLAG is
non-nil if FCN is a builtin."
(save-excursion
(let* ((msbn (matlab-shell-buffer-barf-not-running))
(cmd (concat "which " fcn))
(comint-scroll-show-maximum-output nil)
output
builtin
)
(set-buffer msbn)
(if (not (matlab-on-prompt-p))
(error "MATLAB shell must be non-busy to do that"))
(setq output (matlab-shell-collect-command-output cmd))
;; BUILT-IN
(cond
((string-match "built-in (\\([^)]+\\))" output)
(cons (concat (substring output (match-beginning 1) (match-end 1))
".m")
t))
;; Error
((string-match "not found" output)
nil)
;; JUST AN M FILE
(t
(string-match "$" output)
(cons (substring output 0 (match-beginning 0)) nil))))))
2005-12-01 20:08:16 +01:00
(defun matlab-shell-matlabroot ()
"Get the location of this shell's root.
Returns a string path to the root of the executing MATLAB."
(save-excursion
(let* ((msbn (matlab-shell-buffer-barf-not-running))
(cmd "disp(matlabroot)")
(comint-scroll-show-maximum-output nil)
output
builtin
)
(set-buffer msbn)
(if (and (boundp 'matlab-shell-matlabroot-run)
matlab-shell-matlabroot-run)
matlab-shell-matlabroot-run
;; If we haven't cached it, calculate it now.
(if (not (matlab-on-prompt-p))
(error "MATLAB shell must be non-busy to do that"))
(setq output (matlab-shell-collect-command-output cmd))
(string-match "$" output)
(substring output 0 (match-beginning 0))))))
(defvar matlab-shell-window-exists-for-display-completion-flag nil
"Non-nil means there was an 'other-window' available when `display-completion-list' is called.")
(defun matlab-shell-get-completion-limit-pos (last-cmd completions)
"Used by `matlab-shell-tab' to get the starting location within last-cmd
of the common substring used in matching the completions, i.e.
(substring last-cmd limit-pos (length last-cmd))
is the common starting substring of each completion in completions."
(let ((limit-pos (length last-cmd)))
(when completions
(let* ((completion (car (car completions)))
(i (length completion))
(chomp-num-chars nil))
(while (> i 0)
(let ((part (substring completion 0 i)))
(if (string-suffix-p part last-cmd)
(progn
(setq chomp-num-chars i)
(setq i 0))
(setq i (- i 1)))))
(if chomp-num-chars
(setq limit-pos (- (length last-cmd) chomp-num-chars)))))
limit-pos))
(defun matlab-shell-get-completion-info ()
"Helper function for `matlab-shell-tab' and `company-matlab-shell' to get completion
information for item at point"
(if (or (not (= (point) (point-max)))
(not (matlab-on-prompt-p)))
nil ;; no completions. We can only complete when typing a command.
(let ((inhibit-field-text-motion t)
(last-cmd nil)
(last-cmd-start-point nil)
(common-substr nil)
(limit-pos nil)
(completions nil)
(common-substr-start-pt nil)
(common-substr-end-pt nil)
(did-completion nil))
;; Load last-cmd which is the command we are completing on.
;; We need to avoid altering point because when we ask for completions, we send
;; emacsdocomplete(last-cmd-quoted) to the MATLAB command window and then grab the results
;; and erase them.
(save-excursion
(goto-char (point-max))
(beginning-of-line)
(re-search-forward comint-prompt-regexp)
(setq last-cmd-start-point (point))
;; save the old (last) command
(setq last-cmd (buffer-substring (point) (matlab-point-at-eol))))
;; Get the list of completions.
;; When obtaining completions, we can't use save-excursion because we are
;; manipulating the text in the *MATLAB* window at the point and this
;; move point-marker which causes save-excursion to move to the wrong
;; location. We need to do this before we manipulate the text in the
;; *MATLAB* buffer because `matlab-shell-completion-list' sends
;; emacsdocompletion('statement') to matlab and matlab produces output in
;; the *MATLAB* buffer, then `matlab-shell-completion-list' removes the
;; output from the *MATLAB* buffer.
(let ((last-cmd-quoted last-cmd))
;; Load last-cmd-quoted which the expression typed (e.g. "!mv file.").
;; Note, this has single quotes doubled up so we can ask
;; MATLAB for completions on last-cmd-quoted.
(while (string-match "[^']\\('\\)\\($\\|[^']\\)" last-cmd-quoted)
(setq last-cmd-quoted (replace-match "''" t t last-cmd-quoted 1)))
(let* ((completion-list (matlab-shell-completion-list last-cmd-quoted))
(cmd-text-to-replace (cdr (assoc 'cmd-text-to-replace completion-list))))
(setq completions (cdr (assoc 'completions completion-list)))
(when cmd-text-to-replace
;; need to alter the command to replace replacement-text with the common substring
(let ((replacement-text (cdr (assoc 'replacement-text completion-list)))
(last-cmd-start-len (- (length last-cmd) (length cmd-text-to-replace))))
;; Replace the text typed in the *MATLAB* and update last-cmd
(goto-char (+ last-cmd-start-point last-cmd-start-len))
(delete-region (point) (matlab-point-at-eol))
(insert replacement-text)
(setq last-cmd (concat (substring last-cmd 0 last-cmd-start-len) replacement-text))
(if (not completions)
(setq did-completion t))
))))
;; Consider
;; >> ! touch foo.ext1 foo.ext2
;; >> ! mv foo.<TAB>
;; 'completions' will contain (("foo.ext1") ("foo.ext2")) and
;; common-substr will be "foo." which is used in displaying the
;; completions. The limit-pos in this case will be 5 and
;; last-cmd "! mv foo."
(setq limit-pos (matlab-shell-get-completion-limit-pos last-cmd completions))
(setq common-substr (substring last-cmd limit-pos))
;; Mark the subfield of the completion result so we can say no completions
;; if there aren't any otherwise we need to remove it.
(save-excursion
(goto-char (point-max))
(beginning-of-line)
(re-search-forward comint-prompt-regexp)
(setq common-substr-start-pt (+ (point) limit-pos))
(setq common-substr-end-pt (matlab-point-at-eol))
(if (and (eq (length completions) 1)
(string-equal (buffer-substring-no-properties
common-substr-start-pt common-substr-end-pt)
(car (car completions))))
(setq completions nil))) ;; force display of "No completions"
;; Result
(list (cons 'last-cmd last-cmd)
(cons 'common-substr common-substr)
(cons 'limit-pos limit-pos)
(cons 'completions completions)
(cons 'common-substr-start-pt common-substr-start-pt)
(cons 'common-substr-end-pt common-substr-end-pt)
(cons 'did-completion did-completion)
))))
(defun matlab-display-completion-list (completions common-substring)
;; In emacs 24.4 the common-substring is no longer needed
(let ((args (if (or (< emacs-major-version 24)
(and (= emacs-major-version 24) (< emacs-minor-version 4)))
(list completions common-substring)
(list completions))))
(apply 'display-completion-list args)))
(defun matlab-shell-c-tab ()
"Send [TAB] to the currently running matlab process and retrieve completion
to show using classing emacs tab completion."
(interactive)
(let ((matlab-shell-tab-company-available nil))
(matlab-shell-tab)))
;; matlab-shell-tab,
;; This sends the command text at the prompt to emacsdocomplete.m which returns a list
;; of possible completions. To do this we use comint to 'type' emacsdocomplete(command)
;; in the *MATLAB* buffer and then we extract the result. The emacsdocomplete returns
;; a list of possible completions of the command where each completion starts with
;; zero or more characters of the command "suffix". For example, give a directory
;; containing files, foo.ext1 and foo.ext2,
;; >> ! mv foo.<TAB>
;; will call emacsdocomplete('! mv foo.') which returns (("foo.ext1") ("foo.ext2"))
;; where we have four characters "foo." matching the suffix of command. There can
;; be zero suffix match as in
;; >> !ls /usr/
;; which on Linux returns a long list of items e.g. (("bin", "include", ...))
;;
;;
;; Test cases (using R2016b):
;;
;; >> h=figure;
;; >> h.Num<TAB> Should show Number
;; >> h.Number<TAB> Should show Number and NumberTitle
;; >> h.Num<TAB> Should do same. The extra white space shouldn't mess things up.
;;
;; >> h.NumberTitle<TAB> Should display message "No completions"
;; >> set(h,'<TAB> Should display a long list
;; type P<TAB> Should narrow to Parent, Position, ...
;;
;; >> !touch file.ext Assuming no other fil* names in current directory.
;; >> !mv file.<TAB> Should complete to file.ext
;; >> !mv file.ext<TAB> Should do nothing
;;
;; >> !/usr/b<TAB> Should complete to /usr/bin
;; >> !/usr/bin/cpp-4.<TAB> Should complete to something like /usr/bin/cpp-4.9
;;
;; >> ls /usr/include/byteswap.<TAB> Some file with a '.' should complete correctly
;;
;; >> !touch foo.ext1 foo.ext2
;; >> ! mv foo.<TAB> Should give options foo.ext1 foo.ext2
;; >> ! mv foo.ext1<TAB> Should say no completions
;; >> ! mv foo.ext1 <TAB> Should say no completions (note the space before the TAB)
;;
;; >> vdp
;; >> get_param('vdp','Pos<TAB> Should show several completions
2005-12-01 20:08:16 +01:00
(defun matlab-shell-tab ()
"Send [TAB] to the currently running matlab process and retrieve completion."
(interactive)
(if (and matlab-shell-tab-company-available matlab-shell-tab-use-company)
;; We don't add to company-backends because we bind TAB to matlab-shell-tab
;; which means completions must be explicitly requested. The default
;; company-complete tries to complete as you type which doesn't work
;; so well because it can take MATLAB a bit to compute completions.
(call-interactively 'company-matlab-shell)
(if (not matlab-shell-ask-MATLAB-for-completions)
(call-interactively 'comint-dynamic-complete-filename)
(let* ((inhibit-field-text-motion t)
(completion-info (matlab-shell-get-completion-info))
(last-cmd (cdr (assoc 'last-cmd completion-info)))
(common-substr (cdr (assoc 'common-substr completion-info)))
(limit-pos (cdr (assoc 'limit-pos completion-info)))
(completions (cdr (assoc 'completions completion-info)))
(common-substr-start-pt (cdr (assoc 'common-substr-start-pt completion-info)))
(common-substr-end-pt (cdr (assoc 'common-substr-end-pt completion-info)))
(did-completion (cdr (assoc 'did-completion completion-info))))
(when (not did-completion)
;; Whack the old command 'substring' that is starting part of the
;; completions so we can insert it back later
(delete-region common-substr-start-pt common-substr-end-pt)
(goto-char (point-max))
;; Process the completions
(if (eq (length completions) 1)
;; If there is only one, then there is an obvious thing to do.
(progn
(insert (car (car completions)))
;; kill completions buffer if still visible
(matlab-shell-tab-hide-completions))
;; else handle multiple completions
(let ((try nil))
(setq try (try-completion common-substr completions))
;; Insert in a good completion.
(cond ((or (eq try nil) (eq try t)
(and (stringp try)
(string= try common-substr)))
(insert common-substr)
;; Before displaying the completions buffer, check to see if
;; the completions window is already displayed, or if there is
;; a next window to display. This determines how to remove the
;; completions later.
(if (get-buffer-window "*Completions*")
nil ;; Recycle old value of the display flag.
;; Else, reset this variable.
(setq matlab-shell-window-exists-for-display-completion-flag
;; Else, it isn't displayed, save an action.
(if (eq (next-window) (selected-window))
;; If there is no other window, the post action is
;; to delete.
'delete
;; If there is a window to display, the post
;; action is to bury.
'bury)))
(with-output-to-temp-buffer "*Completions*"
(matlab-display-completion-list (mapcar 'car completions)
common-substr)))
((stringp try)
(insert try)
(matlab-shell-tab-hide-completions))
(t
(insert common-substr))))
))))))
2005-12-01 20:08:16 +01:00
(defun matlab-shell-tab-hide-completions ()
"Hide any completion windows for `matlab-shell-tab'."
(cond ((eq matlab-shell-window-exists-for-display-completion-flag 'delete)
(when (get-buffer "*Completions*")
(delete-windows-on "*Completions*")))
((eq matlab-shell-window-exists-for-display-completion-flag 'bury)
(let ((orig (selected-window))
(bw nil))
(while (setq bw (get-buffer-window "*Completions*"))
(select-window bw)
(bury-buffer))
(select-window orig)))
)
;; Reset state.
(setq matlab-shell-window-exists-for-display-completion-flag nil))
2005-12-01 20:08:16 +01:00
;;; MATLAB mode Shell commands ================================================
(defun matlab-show-matlab-shell-buffer ()
"Switch to the buffer containing the matlab process."
(interactive)
(let ((msbn (concat "*" matlab-shell-buffer-name "*")))
(if (get-buffer msbn)
(switch-to-buffer-other-window msbn)
(message "There is not an active MATLAB process."))))
(defvar matlab-shell-save-and-go-history '("()")
"Keep track of parameters passed to the MATLAB shell.")
(defun matlab-shell-add-to-input-history (string)
"Add STRING to the input-ring and run `comint-input-filter-functions' on it.
Similar to `comint-send-input'."
(if (and (funcall comint-input-filter string)
(or (null comint-input-ignoredups)
(not (ring-p comint-input-ring))
(ring-empty-p comint-input-ring)
(not (string-equal (ring-ref comint-input-ring 0) string))))
(ring-insert comint-input-ring string))
(run-hook-with-args 'comint-input-filter-functions
(concat string "\n"))
(if (boundp 'comint-save-input-ring-index);only bound in GNU emacs
(setq comint-save-input-ring-index comint-input-ring-index))
(setq comint-input-ring-index nil))
(defun matlab-shell-save-and-go ()
"Save this M file, and evaluate it in a MATLAB shell."
(interactive)
(if (not (eq major-mode 'matlab-mode))
(error "Save and go is only useful in a MATLAB buffer!"))
(if (not (buffer-file-name (current-buffer)))
(call-interactively 'write-file))
(let ((fn-name (file-name-sans-extension
(file-name-nondirectory (buffer-file-name))))
(msbn (concat "*" matlab-shell-buffer-name "*"))
2016-02-10 11:27:23 +01:00
(dir (file-name-directory buffer-file-name))
(change-cd matlab-change-current-directory)
2005-12-01 20:08:16 +01:00
(param ""))
(save-buffer)
;; Do we need parameters?
(if (save-excursion
(goto-char (point-min))
(end-of-line)
(forward-sexp -1)
(looking-at "([a-zA-Z]"))
(setq param (read-string "Parameters: "
(car matlab-shell-save-and-go-history)
'matlab-shell-save-and-go-history)))
(if (matlab-with-emacs-link)
;; Execute the current file in MATLAB
(when (fboundp 'matlab-eei-run) (matlab-eei-run)) ;; XXX where is matlab-eei-run?
2005-12-01 20:08:16 +01:00
;; No buffer? Make it!
(if (not (get-buffer msbn)) (matlab-shell))
;; Ok, now fun the function in the matlab shell
(if (get-buffer-window msbn t)
(select-window (get-buffer-window msbn t))
(switch-to-buffer (concat "*" matlab-shell-buffer-name "*")))
2016-02-10 11:27:23 +01:00
;; change current directory?
(if change-cd
(let ((cmd (progn
(mapc
(lambda (e)
(while (string-match (car e) dir)
(setq dir (replace-match
(format "', char(%s), '" (cdr e)) t t dir))))
'(("ô" . "244")
("é" . "233")
("è" . "232")
("à" . "224")))
dir)))
(matlab-shell-send-string (concat "cd(['" cmd "'])\n"))))
2005-12-01 20:08:16 +01:00
(let ((cmd (concat fn-name " " param)))
(matlab-shell-add-to-input-history cmd)
(matlab-shell-send-string (concat cmd "\n"))))))
(defun matlab-shell-run-region (beg end &optional noshow)
2005-12-01 20:08:16 +01:00
"Run region from BEG to END and display result in MATLAB shell.
If NOSHOW is non-nil, replace newlines with commas to suppress output.
2005-12-01 20:08:16 +01:00
This command requires an active MATLAB shell."
(interactive "r")
(if (> beg end) (let (mid) (setq mid beg beg end end mid)))
(let ((command
(let ((str (concat (buffer-substring beg end) "\n")))
;; Remove comments
(with-temp-buffer
(insert str)
(goto-char (point-min))
(while (search-forward "%" nil t)
(when (not (matlab-cursor-in-string))
(delete-region (1- (point)) (matlab-point-at-eol))))
(setq str (buffer-substring-no-properties (point-min) (point-max))))
(while (string-match "\n\\s-*\n" str)
(setq str (concat (substring str 0 (match-beginning 0))
"\n"
(substring str (match-end 0)))))
(when noshow
;; Remove continuations
(while (string-match
(concat "\\s-*"
(regexp-quote matlab-elipsis-string)
"\\s-*\n")
str)
(setq str (replace-match " " t t str)))
(while (string-match "\n" str)
(setq str (replace-match ", " t t str)))
(setq str (concat str "\n")))
str))
2005-12-01 20:08:16 +01:00
(msbn nil)
(lastcmd)
(inhibit-field-text-motion t))
(if (matlab-with-emacs-link)
;; Run the region w/ Emacs Link
(when (fboundp 'matlab-eei-eval-region) (matlab-eei-eval-region beg end))
2005-12-01 20:08:16 +01:00
(save-excursion
(setq msbn (matlab-shell-buffer-barf-not-running))
(set-buffer msbn)
(if (not (matlab-on-prompt-p))
(error "MATLAB shell must be non-busy to do that"))
;; Save the old command
(beginning-of-line)
(re-search-forward comint-prompt-regexp)
(setq lastcmd (buffer-substring (point) (matlab-point-at-eol)))
(delete-region (point) (matlab-point-at-eol))
;; We are done error checking, run the command.
(matlab-shell-send-string command)
(insert lastcmd))
(set-buffer msbn)
(goto-char (point-max))
(display-buffer msbn nil "visible"))
2005-12-01 20:08:16 +01:00
))
(defun matlab-shell-run-cell ()
"Run the cell the cursor is in."
(interactive)
(let ((start (save-excursion (forward-page -1)
(if (looking-at "function")
(error "You are not in a cell. Try `matlab-shell-save-and-go' instead"))
(when (matlab-ltype-comm)
;; Skip over starting comment from the current cell.
(matlab-end-of-command 1)
(end-of-line)
(forward-char 1))
(point)))
(end (save-excursion (forward-page 1)
(when (matlab-ltype-comm)
(beginning-of-line)
(forward-char -1))
(point))))
(matlab-shell-run-region start end t)))
2005-12-01 20:08:16 +01:00
(defun matlab-shell-run-region-or-line ()
"Run region from BEG to END and display result in MATLAB shell.
pIf region is not active run the current line.
2005-12-01 20:08:16 +01:00
This command requires an active MATLAB shell."
(interactive)
(if (and transient-mark-mode mark-active)
2005-12-01 20:08:16 +01:00
(matlab-shell-run-region (mark) (point))
(matlab-shell-run-region (matlab-point-at-bol) (matlab-point-at-eol))))
2005-12-01 20:08:16 +01:00
;;; MATLAB Shell Commands =====================================================
(defun matlab-read-word-at-point ()
"Get the word closest to point, but do not change position.
Has a preference for looking backward when not directly on a symbol.
Snatched and hacked from dired-x.el"
(let ((word-chars "a-zA-Z0-9_")
(bol (matlab-point-at-bol))
(eol (matlab-point-at-eol))
start)
(save-excursion
;; First see if just past a word.
(if (looking-at (concat "[" word-chars "]"))
nil
(skip-chars-backward (concat "^" word-chars "{}()\[\]") bol)
(if (not (bobp)) (backward-char 1)))
(if (numberp (string-match (concat "[" word-chars "]")
(char-to-string (following-char))))
(progn
(skip-chars-backward word-chars bol)
(setq start (point))
(skip-chars-forward word-chars eol))
(setq start (point))) ; If no found, return empty string
(buffer-substring start (point)))))
(defun matlab-read-line-at-point ()
"Get the line under point, if command line."
(if (eq major-mode 'matlab-shell-mode)
(save-excursion
(let ((inhibit-field-text-motion t))
(beginning-of-line)
(if (not (looking-at (concat comint-prompt-regexp)))
""
(search-forward-regexp comint-prompt-regexp)
(buffer-substring (point) (matlab-point-at-eol)))))
(save-excursion
;; In matlab buffer, find all the text for a command.
;; so back over until there is no more continuation.
(while (save-excursion (forward-line -1) (matlab-lattr-cont))
(forward-line -1))
;; Go forward till there is no continuation
(beginning-of-line)
(let ((start (point)))
(while (matlab-lattr-cont) (forward-line 1))
(end-of-line)
(buffer-substring start (point))))))
(defun matlab-non-empty-lines-in-string (str)
"Return number of non-empty lines in STR."
(let ((count 0)
(start 0))
(while (string-match "^.+$" str start)
(setq count (1+ count)
start (match-end 0)))
count))
(defun matlab-output-to-temp-buffer (buffer output)
"Print output to temp buffer, or a message if empty string.
BUFFER is the buffer to output to, and OUTPUT is the text to insert."
(let ((lines-found (matlab-non-empty-lines-in-string output)))
(cond ((= lines-found 0)
(message "(MATLAB command completed with no output)"))
((= lines-found 1)
(string-match "^.+$" output)
(message (substring output (match-beginning 0)(match-end 0))))
(t (with-output-to-temp-buffer buffer (princ output))
(with-current-buffer buffer
2005-12-01 20:08:16 +01:00
(matlab-shell-help-mode))))))
(defun matlab-shell-run-command (command)
"Run COMMAND and display result in a buffer.
This command requires an active MATLAB shell."
(interactive (list (read-from-minibuffer
"MATLAB command line: "
(cons (matlab-read-line-at-point) 0))))
(let ((doc (matlab-shell-collect-command-output command)))
(matlab-output-to-temp-buffer "*MATLAB Help*" doc)))
(defun matlab-shell-describe-variable (variable)
"Get the contents of VARIABLE and display them in a buffer.
This uses the WHOS (MATLAB 5) command to find viable commands.
This command requires an active MATLAB shell."
(interactive (list (read-from-minibuffer
"MATLAB variable: "
(cons (matlab-read-word-at-point) 0))))
(let ((doc (matlab-shell-collect-command-output (concat "whos " variable))))
(matlab-output-to-temp-buffer "*MATLAB Help*" doc)))
(defun matlab-shell-describe-command (command)
"Describe COMMAND textually by fetching it's doc from the MATLAB shell.
This uses the lookfor command to find viable commands.
This command requires an active MATLAB shell."
(interactive
(let ((fn (matlab-function-called-at-point))
val)
(setq val (read-string (if fn
(format "Describe function (default %s): " fn)
"Describe function: ")))
(if (string= val "") (list fn) (list val))))
(let ((doc (matlab-shell-collect-command-output (concat "help " command))))
(matlab-output-to-temp-buffer "*MATLAB Help*" doc)))
(defun matlab-shell-apropos (matlabregex)
"Look for any active commands in MATLAB matching MATLABREGEX.
This uses the lookfor command to find viable commands."
(interactive (list (read-from-minibuffer
"MATLAB command subexpression: "
(cons (matlab-read-word-at-point) 0))))
(let ((ap (matlab-shell-collect-command-output
(concat "lookfor " matlabregex))))
(matlab-output-to-temp-buffer "*MATLAB Apropos*" ap)))
2005-12-01 20:08:16 +01:00
(defun matlab-on-prompt-p ()
"Return t if we MATLAB can accept input."
(save-excursion
(let ((inhibit-field-text-motion t))
(goto-char (point-max))
(beginning-of-line)
(looking-at comint-prompt-regexp))))
(defun matlab-on-empty-prompt-p ()
"Return t if we MATLAB is on an empty prompt."
(save-excursion
(let ((inhibit-field-text-motion t))
(goto-char (point-max))
(beginning-of-line)
(looking-at (concat comint-prompt-regexp "\\s-*$")))))
(defun matlab-shell-buffer-barf-not-running ()
"Return a running MATLAB buffer iff it is currently active."
(or (matlab-shell-active-p)
(error "You need to run the command `matlab-shell' to do that!")))
(defun matlab-shell-collect-command-output (command)
"If there is a MATLAB shell, run the MATLAB COMMAND and return it's output.
It's output is returned as a string with no face properties. The text output
of the command is removed from the MATLAB buffer so there will be no
indication that it ran."
(let ((msbn (matlab-shell-buffer-barf-not-running)))
;; We are unable to use save-excursion to save point position because we are
;; manipulating the *MATLAB* buffer by erasing the current text typed at the
;; MATLAB prompt (where point is) and then we send command to MATLAB and
;; grab the result. After this we erase the output from command and then
;; restore the current text at the MATLAB prompt and move to start-point.
;; Note, save-excursion works by tracking `point-marker' and when you manipulate
;; the text at point, `point-marker' moves causing save-excursion to move
;; the point in to a location we don't want. See:
;; http://emacs.stackexchange.com/questions/7574/why-save-excursion-doesnt-save-point-position
;; Ideally there would be some way to prevent the *MATLAB* buffer from refreshing
;; as we are interacting with it, but I couldn't figure out a way to do that.
(with-current-buffer msbn
(save-window-excursion
(let ((pos nil)
(str nil)
(lastcmd)
(inhibit-field-text-motion t)
(start-point (point)))
(if (not (matlab-on-prompt-p))
(error "MATLAB shell must be non-busy to do that"))
;; Save the old command
(goto-char (point-max))
(beginning-of-line)
(re-search-forward comint-prompt-regexp)
;; Backup if there are extra spaces. To see why, try tab completion on command with
;; leading spaces, e.g.
;; >> h=figure;
;; >> h.Num<TAB>
(re-search-backward ">")
(forward-char 2)
(setq lastcmd (buffer-substring (point) (matlab-point-at-eol)))
(delete-region (point) (matlab-point-at-eol))
;; We are done error checking, run the command.
(setq pos (point))
;; Note, comint-simple-send in emacs 24.4 appends a newline and code below assumes
;; one prompt indicates command completed, so don't append a newline.
(comint-simple-send (get-buffer-process (current-buffer)) command)
;; Wait for the command to finish, by looking for new prompt.
(goto-char (point-max))
;; Turn on C-g by using wiht-local-quit. This is needed to prevent message:
;; "Blocking call to accept-process-output with quit inhibited!! [115 times]"
;; when using `company-matlab-shell' for TAB completions.
(with-local-quit
(while (or (>= (+ pos (string-width command)) (point)) (not (matlab-on-empty-prompt-p)))
(accept-process-output (get-buffer-process (current-buffer)))
(goto-char (point-max))))
;; Get result of command into str
(goto-char pos)
(setq str (buffer-substring-no-properties (save-excursion
(goto-char pos)
(beginning-of-line)
(forward-line 1)
(point))
(save-excursion
(goto-char (point-max))
(beginning-of-line)
(point))))
;; delete the result of command
(delete-region pos (point-max))
;; restore contents of buffer so it looks like nothing happened.
(insert lastcmd)
(goto-char start-point)
;; return result 'string' from executing MATLAB command
str)))))
2005-12-01 20:08:16 +01:00
(defun matlab-shell-send-string (string)
"Send STRING to the currently running matlab process."
(if (not matlab-shell-echoes)
(let ((proc (get-buffer-process (current-buffer))))
(goto-char (point-max))
(insert string)
(set-marker (process-mark proc) (point))))
(comint-send-string (get-buffer-process (current-buffer)) string))
(defun matlab-url-at (p)
"Return the matlab-url overlay at P, or nil."
(let ((url nil) (o (matlab-overlays-at p)))
(while (and o (not url))
(setq url (matlab-overlay-get (car o) 'matlab-url)
o (cdr o)))
url))
(defun matlab-url-stack-top-at (p)
"Return the matlab-url overlay at P, or nil."
(let ((url nil) (o (matlab-overlays-at p)))
(while (and o (not url))
(setq url (or (matlab-overlay-get (car o) 'first-in-error-stack)
(matlab-overlay-get (car o) 'matlab-url))
o (cdr o)))
url))
(defun matlab-shell-previous-matlab-url (&optional stacktop)
"Find a previous occurrence of an overlay with a MATLAB URL.
If STACKTOP is non-nil, then also get the top of some stack, which didn't
show up in reverse order."
2005-12-01 20:08:16 +01:00
(save-excursion
(let ((url nil) (o nil) (p (point)))
(while (and (not url)
(setq p (matlab-previous-overlay-change p))
(not (eq p (point-min))))
(setq url
(if stacktop
(matlab-url-stack-top-at p)
(matlab-url-at p))))
2005-12-01 20:08:16 +01:00
url)))
(defun matlab-find-other-window-file-line-column (ef el ec &optional debug)
"Find file EF in other window and to go line EL and 1-basec column EC.
If DEBUG is non-nil, then setup GUD debugging features."
(cond ((file-exists-p ef)
nil);; keep ef the same
((file-exists-p (concat ef ".m"))
(setq ef (concat ef ".m"))) ;; Displayed w/out .m?
((string-match ">" ef)
(setq ef (concat (substring ef 0 (match-beginning 0)) ".m")))
)
2005-12-01 20:08:16 +01:00
(find-file-other-window ef)
(with-no-warnings
(goto-line (string-to-number el)))
2005-12-01 20:08:16 +01:00
(when debug
(setq gud-last-frame (cons (buffer-file-name) (string-to-number el)))
2005-12-01 20:08:16 +01:00
(gud-display-frame))
(setq ec (string-to-number ec))
2005-12-01 20:08:16 +01:00
(if (> ec 0) (forward-char (1- ec))))
(defun matlab-find-other-window-via-url (url &optional debug)
"Find other window using matlab URL and optionally set DEBUG cursor."
(cond ((string-match "^error:\\(.*\\),\\([0-9]+\\),\\([0-9]+\\)$" url)
(let ((ef (substring url (match-beginning 1) (match-end 1)))
(el (substring url (match-beginning 2) (match-end 2)))
(ec (substring url (match-beginning 3) (match-end 3))))
(matlab-find-other-window-file-line-column ef el ec debug)))
((string-match "opentoline('\\([^']+\\)',\\([0-9]+\\),\\([0-9]+\\))" url)
(let ((ef (substring url (match-beginning 1) (match-end 1)))
(el (substring url (match-beginning 2) (match-end 2)))
(ec (substring url (match-beginning 3) (match-end 3))))
(matlab-find-other-window-file-line-column ef el ec debug)))
((string-match "^matlab:*\\(.*\\)$" url)
2005-12-01 20:08:16 +01:00
(process-send-string
(get-buffer-process gud-comint-buffer)
(concat (substring url (match-beginning 1) (match-end 1)) "\n")))))
(defun matlab-shell-last-error ()
"In the MATLAB interactive buffer, find the last MATLAB error, and go there.
To reference old errors, put the cursor just after the error text."
(interactive)
(catch 'done
(let ((url (matlab-shell-previous-matlab-url t)))
2005-12-01 20:08:16 +01:00
(if url
(progn (matlab-find-other-window-via-url url) (throw 'done nil))
(save-excursion
(end-of-line) ;; In case we are before the line number 1998/06/05 16:54sk
2005-12-01 20:08:16 +01:00
(if (not (re-search-backward gud-matlab-error-regexp nil t))
(error "No errors found!"))
(let ((ef (buffer-substring-no-properties
(match-beginning 2) (match-end 2)))
(el (buffer-substring-no-properties
(match-beginning 3) (match-end 3))))
(matlab-find-other-window-file-line-column ef el "0")))))))
2005-12-01 20:08:16 +01:00
(defun matlab-shell-html-click (e)
"Go to the error at the location of event E."
(interactive "e")
(mouse-set-point e)
(matlab-shell-html-go))
(defun matlab-shell-html-go ()
"Go to the error at the location `point'."
(interactive)
2005-12-01 20:08:16 +01:00
(let ((url (matlab-url-at (point))))
(if url (matlab-find-other-window-via-url url))))
(defun matlab-shell-dbstop-error ()
"Stop on errors."
(interactive)
(comint-send-string (get-buffer-process (current-buffer))
"dbstop if error\n"))
(defun matlab-shell-dbclear-error ()
"Don't stop on errors."
(interactive)
(comint-send-string (get-buffer-process (current-buffer))
"dbclear if error\n"))
(defun matlab-shell-demos ()
"MATLAB demos."
(interactive)
(comint-send-string (get-buffer-process (current-buffer)) "demo\n"))
(defun matlab-shell-close-figures ()
"Close any open figures."
(interactive)
(comint-send-string (get-buffer-process (current-buffer)) "close all\n"))
(defun matlab-shell-close-current-figure ()
"Close current figure."
(interactive)
(comint-send-string (get-buffer-process (current-buffer)) "delete(gcf)\n"))
(defun matlab-shell-exit ()
"Exit MATLAB shell."
(interactive)
(comint-send-string (get-buffer-process (current-buffer)) "exit\n")
(kill-buffer nil))
;;; matlab-shell based Topic Browser and Help =================================
(defcustom matlab-shell-topic-mode-hook nil
"*MATLAB shell topic hook."
:group 'matlab-shell
:type 'hook)
(defvar matlab-shell-topic-current-topic nil
"The currently viewed topic in a MATLAB shell topic buffer.")
(defun matlab-shell-topic-browser ()
"Create a topic browser by querying an active MATLAB shell using HELP.
Maintain state in our topic browser buffer."
(interactive)
;; Reset topic browser if it doesn't exist.
(if (not (get-buffer "*MATLAB Topic*"))
(setq matlab-shell-topic-current-topic nil))
(let ((b (get-buffer-create "*MATLAB Topic*")))
(switch-to-buffer b)
(if (string= matlab-shell-topic-current-topic "")
nil
(matlab-shell-topic-mode)
(matlab-shell-topic-browser-create-contents ""))))
(defvar matlab-shell-topic-mouse-face-keywords
'(;; These are subtopic fields...
("^\\(\\w+/\\w+\\)[ \t]+-" 1 font-lock-reference-face)
;; These are functions...
("^[ \t]+\\(\\w+\\)[ \t]+-" 1 font-lock-function-name-face)
;; Here is a See Also line...
("[ \t]+See also "
("\\(\\w+\\)\\([,.]\\| and\\|$\\) *" nil nil (1 font-lock-reference-face))))
2005-12-01 20:08:16 +01:00
"These are keywords we also want to put mouse-faces on.")
(defvar matlab-shell-topic-font-lock-keywords
(append matlab-shell-topic-mouse-face-keywords
'(("^[^:\n]+:$" 0 font-lock-keyword-face)
;; These are subheadings...
("^[ \t]+\\([^.\n]+[a-zA-Z.]\\)$" 1 'underline)
))
"Keywords useful for highlighting a MATLAB TOPIC buffer.")
(defvar matlab-shell-help-font-lock-keywords
(append matlab-shell-topic-mouse-face-keywords
'(;; Function call examples
("[ \t]\\([A-Z]+\\)\\s-*=\\s-*\\([A-Z]+[0-9]*\\)("
(1 font-lock-variable-name-face)
(2 font-lock-function-name-face))
("[ \t]\\([A-Z]+[0-9]*\\)("
(1 font-lock-function-name-face))
;; Parameters: Not very accurate, unfortunately.
("[ \t]\\([A-Z]+[0-9]*\\)("
("'?\\(\\w+\\)'?\\([,)]\\) *" nil nil
2005-12-01 20:08:16 +01:00
(1 font-lock-variable-name-face))
)
;; Reference uppercase words
("\\<\\([A-Z]+[0-9]*\\)\\>" 1 font-lock-reference-face)))
"Keywords for regular help buffers.")
;; View-major-mode is an emacs20 thing. This gives us a small compatibility
;; layer.
(eval-and-compile
(if (not (fboundp 'view-major-mode)) (defalias 'view-major-mode 'view-mode)))
(defvar matlab-shell-help-mode-menu) ;; Quiet compiler warning (var is defined below)
2005-12-01 20:08:16 +01:00
(define-derived-mode matlab-shell-help-mode
view-major-mode "M-Help"
"Major mode for viewing MATLAB help text.
Entry to this mode runs the normal hook `matlab-shell-help-mode-hook'.
Commands:
\\{matlab-shell-help-mode-map}"
(make-local-variable 'font-lock-defaults)
(setq font-lock-defaults '((matlab-shell-help-font-lock-keywords)
t nil ((?_ . "w"))))
;; This makes sure that we really enter font lock since
;; kill-all-local-variables is not used by old view-mode.
(and (boundp 'global-font-lock-mode) global-font-lock-mode
(not font-lock-mode) (font-lock-mode 1))
(easy-menu-add matlab-shell-help-mode-menu matlab-shell-help-mode-map)
(matlab-shell-topic-mouse-highlight-subtopics)
)
(define-key matlab-shell-help-mode-map [return] 'matlab-shell-topic-choose)
(define-key matlab-shell-help-mode-map "t" 'matlab-shell-topic-browser)
(define-key matlab-shell-help-mode-map "q" 'bury-buffer)
(define-key matlab-shell-help-mode-map
[(control h) (control m)] matlab-help-map)
(if (string-match "XEmacs" emacs-version)
(define-key matlab-shell-help-mode-map [button2] 'matlab-shell-topic-click)
(define-key matlab-shell-help-mode-map [mouse-2] 'matlab-shell-topic-click))
(defvar mode-motion-hook) ;; quiet compiler warning (used in XEmacs)
2005-12-01 20:08:16 +01:00
(easy-menu-define
matlab-shell-help-mode-menu matlab-shell-help-mode-map
"MATLAB shell topic menu"
'("MATLAB Help"
["Describe This Command" matlab-shell-topic-choose t]
"----"
["Describe Command" matlab-shell-describe-command t]
["Describe Variable" matlab-shell-describe-variable t]
["Command Apropos" matlab-shell-apropos t]
["Topic Browser" matlab-shell-topic-browser t]
"----"
["Exit" bury-buffer t]))
(defvar matlab-shell-topic-mode-menu) ;; forward declaration to quite compiler warning
2005-12-01 20:08:16 +01:00
(define-derived-mode matlab-shell-topic-mode
matlab-shell-help-mode "M-Topic"
"Major mode for browsing MATLAB HELP topics.
The output of the MATLAB command HELP with no parameters creates a listing
of known help topics at a given installation. This mode parses that listing
and allows selecting a topic and getting more help for it.
Entry to this mode runs the normal hook `matlab-shell-topic-mode-hook'.
Commands:
\\{matlab-shell-topic-mode-map}"
(setq font-lock-defaults '((matlab-shell-topic-font-lock-keywords)
t t ((?_ . "w"))))
(if (string-match "XEmacs" emacs-version)
(setq mode-motion-hook 'matlab-shell-topic-highlight-line))
(easy-menu-add matlab-shell-topic-mode-menu matlab-shell-topic-mode-map)
)
(easy-menu-define
matlab-shell-topic-mode-menu matlab-shell-topic-mode-map
"MATLAB shell topic menu"
'("MATLAB Topic"
["Select This Topic" matlab-shell-topic-choose t]
["Top Level Topics" matlab-shell-topic-browser t]
"----"
["Exit" bury-buffer t]))
(defun matlab-shell-topic-browser-create-contents (subtopic)
"Fill in a topic browser with the output from SUBTOPIC."
(matlab-read-only-mode -1)
2005-12-01 20:08:16 +01:00
(erase-buffer)
(insert (matlab-shell-collect-command-output (concat "help " subtopic)))
(goto-char (point-min))
(forward-line 1)
(delete-region (point-min) (point))
(setq matlab-shell-topic-current-topic subtopic)
(if (not (string-match "XEmacs" emacs-version))
(matlab-shell-topic-mouse-highlight-subtopics))
(matlab-read-only-mode 1)
2005-12-01 20:08:16 +01:00
)
(defun matlab-shell-topic-click (e)
"Click on an item in a MATLAB topic buffer we want more information on.
Must be bound to event E."
(interactive "e")
(mouse-set-point e)
(matlab-shell-topic-choose))
(defun matlab-shell-topic-choose ()
"Choose the topic to expand on that is under the cursor.
This can fill the topic buffer with new information. If the topic is a
command, use `matlab-shell-describe-command' instead of changing the topic
buffer."
(interactive)
(let ((topic nil) (fun nil) (p (point)))
(save-excursion
(beginning-of-line)
(if (looking-at "^\\w+/\\(\\w+\\)[ \t]+-")
(setq topic (match-string 1))
(if (looking-at "^[ \t]+\\(\\(\\w\\|_\\)+\\)[ \t]+-")
(setq fun (match-string 1))
(if (and (not (looking-at "^[ \t]+See also"))
(not (save-excursion (forward-char -2)
(looking-at ",$"))))
(error "You did not click on a subtopic, function or reference")
(goto-char p)
(forward-word -1)
(if (not (looking-at "\\(\\(\\w\\|_\\)+\\)\\([.,]\\| and\\|\n\\)"))
(error "You must click on a reference")
(setq topic (match-string 1)))))))
(message "Opening item %s..." (or topic fun))
(if topic
(matlab-shell-topic-browser-create-contents (downcase topic))
(matlab-shell-describe-command fun))
))
(defun matlab-shell-topic-mouse-highlight-subtopics ()
"Put a `mouse-face' on all clickable targets in this buffer."
(save-excursion
(let ((el matlab-shell-topic-mouse-face-keywords))
(while el
(goto-char (point-min))
(while (re-search-forward (car (car el)) nil t)
(let ((cd (car (cdr (car el)))))
(if (numberp cd)
(put-text-property (match-beginning cd) (match-end cd)
'mouse-face 'highlight)
(while (re-search-forward (car cd) nil t)
(put-text-property (match-beginning (car (nth 3 cd)))
(match-end (car (nth 3 cd)))
'mouse-face 'highlight)))))
(setq el (cdr el))))))
(defvar mouse-grabbed-buffer) ;; Suppress compilation warning in Emacs (an XEmacs only variable)
(defvar mode-motion-extent) ;; Suppress compilation warning in Emacs (an XEmacs only variable)
2005-12-01 20:08:16 +01:00
(defun matlab-shell-topic-highlight-line (event)
"A value of `mode-motion-hook' which will highlight topics under the mouse.
EVENT is the user mouse event."
;; XEMACS only function
(let* ((buffer (when (fboundp 'event-buffer) (event-buffer event)))
(point (and buffer (when (fboundp 'event-point) (event-point event)))))
2005-12-01 20:08:16 +01:00
(if (and buffer (not (eq buffer mouse-grabbed-buffer)))
(save-excursion
(save-window-excursion
(set-buffer buffer)
(when (fboundp 'mode-motion-ensure-extent-ok) (mode-motion-ensure-extent-ok event))
2005-12-01 20:08:16 +01:00
(if (not point)
(when (fboundp 'detach-extent) (detach-extent mode-motion-extent))
2005-12-01 20:08:16 +01:00
(goto-char point)
(end-of-line)
(setq point (point))
(beginning-of-line)
(if (or (looking-at "^\\w+/\\(\\w+\\)[ \t]+-")
(looking-at "^[ \t]+\\(\\(\\w\\|_\\)+\\)[ \t]+-"))
(when (fboundp 'set-extent-endpoints)
(set-extent-endpoints mode-motion-extent (point) point))
(when (fboundp 'detach-extent) (detach-extent mode-motion-extent)))))))))
2005-12-01 20:08:16 +01:00
;;; M File path stuff =========================================================
(defun matlab-mode-determine-mfile-path ()
"Create the path in `matlab-mode-install-path'."
(let ((path (file-name-directory matlab-shell-command)))
;; if we don't have a path, find the MATLAB executable on our path.
(if (not path)
(let ((pl exec-path))
(while (and pl (not path))
(if (and (file-exists-p (concat (car pl) "/" matlab-shell-command))
(not (car (file-attributes (concat (car pl) "/"
matlab-shell-command)))))
(setq path (car pl)))
(setq pl (cdr pl)))))
(if (not path)
nil
;; When we find the path, we need to massage it to identify where
;; the M files are that we need for our completion lists.
(if (string-match "/bin$" path)
(setq path (substring path 0 (match-beginning 0))))
;; Everything stems from toolbox (I think)
(setq path (concat path "/toolbox/")))
path))
(defcustom matlab-mode-install-path (list (matlab-mode-determine-mfile-path))
"Base path pointing to the locations of all the m files used by matlab.
All directories under each element of `matlab-mode-install-path' are
checked, so only top level toolbox directories need be added.
Paths should be added in the order in which they should be searched."
:group 'matlab-shell
:type '(repeat (string :tag "Path: ")))
(defun matlab-find-file-under-path (path filename)
"Return the pathname or nil of PATH under FILENAME."
(if (file-exists-p (concat path filename))
(concat path filename)
(let ((dirs (if (file-directory-p path)
;; Not checking as a directory first fails on XEmacs
;; Stelios Kyriacou <kyriacou@cbmv.jhu.edu>
(directory-files path t nil t)))
(found nil))
(while (and dirs (not found))
(if (and (car (file-attributes (car dirs)))
;; require directory readable
(file-readable-p (car dirs))
;; don't redo our path names
(not (string-match "/\\.\\.?$" (car dirs)))
;; don't find files in object directories.
(not (string-match "@" (car dirs))))
(setq found
(matlab-find-file-under-path (concat (car dirs) "/")
filename)))
(setq dirs (cdr dirs)))
found)))
(defun matlab-find-file-on-path (filename)
"Find FILENAME on the current MATLAB path.
The MATLAB path is determined by `matlab-mode-install-path' and the
current directory. You must add user-installed paths into
`matlab-mode-install-path' if you would like to have them included."
(interactive
(list
(let ((default (matlab-read-word-at-point)))
(if default
(let ((s (read-string (concat "File (default " default "): "))))
(if (string= s "") default s))
(read-string "File: ")))))
(if (string= filename "")
(error "You must specify an M file"))
(if (not (string-match "\\.m$" filename))
(setq filename (concat filename ".m")))
(let ((fname nil)
(dirs matlab-mode-install-path))
(if (file-exists-p (concat default-directory filename))
(setq fname (concat default-directory filename)))
(while (and (not fname) dirs)
(if (stringp (car dirs))
(progn
(message "Searching for %s in %s" filename (car dirs))
(setq fname (matlab-find-file-under-path (car dirs) filename))))
(setq dirs (cdr dirs)))
(if fname (find-file fname)
(error "File %s not found on any known paths. \
Check `matlab-mode-install-path'" filename))))
(defun matlab-find-file-click (e)
"Find the file clicked on with event E on the current path."
(interactive "e")
(mouse-set-point e)
(let ((f (matlab-read-word-at-point)))
(if (not f) (error "To find an M file, click on a word"))
(matlab-find-file-on-path f)))
;;; matlab-mode debugging =====================================================
(defun matlab-show-line-info ()
"Display type and attributes of current line. Used in debugging."
(interactive)
(let ((msg "line-info:")
(indent (matlab-calculate-indentation (current-indentation)))
(nexti (matlab-next-line-indentation)))
(setq msg (concat msg
" Line type: " (symbol-name (car indent))
" This Line: " (int-to-string (nth 1 indent))
" Next Line: " (int-to-string nexti)))
(if (matlab-lattr-cont)
(setq msg (concat msg " w/cont")))
(if (matlab-lattr-comm)
(setq msg (concat msg " w/comm")))
(message msg)))
(provide 'matlab)
;;; matlab.el ends here
;; LocalWords: el Wette mwette caltech edu Ludlam eludlam defconst online el
;; LocalWords: easymenu Ee progn defalias fboundp itimer defun boundp Wette el
;; LocalWords: defvaralias bol eol defmacro defcustom CASEINDENT COMMANDINDENT
;; LocalWords: sexp mmode xemacs fontified setq regex sg Fns Alist elipsis vf
;; LocalWords: functionname vers minibuffer eei featurep facep zmacs defface
;; LocalWords: cellbreak cellbreaks overline keymap stringify ispell torkel el
;; LocalWords: prog bolp if'd cdr uicontext setcolor pragmas tch lse mwette
;; LocalWords: ndfunction MUs caltech edu Ludlam eludlam defconst online Wette
;; LocalWords: easymenu Ee progn defalias fboundp itimer defun boundp mwette
;; LocalWords: defvaralias bol eol defmacro defcustom CASEINDENT COMMANDINDENT
;; LocalWords: sexp mmode xemacs fontified setq regex sg Fns Alist elipsis vf
;; LocalWords: functionname vers minibuffer eei featurep facep zmacs defface
;; LocalWords: cellbreak cellbreaks overline keymap stringify ispell torkel
;; LocalWords: prog bolp if'd cdr uicontext setcolor pragmas tch lse caltech
;; LocalWords: ndfunction MUs ght urface rol sw edu Ludlam eludlam defconst
;; LocalWords: online easymenu Ee progn defalias fboundp itimer defun boundp
;; LocalWords: defvaralias bol eol defmacro defcustom CASEINDENT COMMANDINDENT
;; LocalWords: sexp mmode xemacs fontified setq regex sg Fns Alist elipsis vf
;; LocalWords: functionname vers minibuffer eei featurep facep zmacs defface
;; LocalWords: cellbreak cellbreaks overline keymap stringify ispell torkel
;; LocalWords: prog bolp if'd cdr uicontext setcolor pragmas tch lse Wette RET
;; LocalWords: ndfunction MUs ght urface rol sw mwette caltech edu Ludlam lear
;; LocalWords: eludlam defconst online easymenu Ee progn defalias fboundp ont
;; LocalWords: itimer defun boundp defvaralias bol eol defmacro defcustom tus
;; LocalWords: CASEINDENT COMMANDINDENT sexp mmode xemacs fontified setq regex
;; LocalWords: sg Fns Alist elipsis vf functionname vers minibuffer eei ep mld
;; LocalWords: featurep facep zmacs defface cellbreak cellbreaks overline
;; LocalWords: keymap stringify ispell torkel prog bolp if'd cdr uicontext
;; LocalWords: setcolor pragmas tch lse ndfunction MUs ght urface rol sw dem
;; LocalWords: Imenu imenu alist reindent unindent fn prev ltype uniquafy lst
;; LocalWords: nlst nreverse Aki Vehtari backquote oldsyntax edebug parens
;; LocalWords: cline CLim XColor XDir XLabel XAxis XScale YColor YDir YAxis
;; LocalWords: YScale YTick ZColor ZDir ZGrid ZLabel ZScale ZTick Dithermap tl
;; LocalWords: autoend noerror returnme Unstarted includeelse lattr bobp zerop
;; LocalWords: cellstart blockcomm eobp commtype startmove nomove sregex Ooops
;; LocalWords: instring calc ci sem DEPTHNUMBER blockstart blockmid
;; LocalWords: blockendless blockend tmp unstarted listp fc boc parendepth
;; LocalWords: DONT cci startpnt bc ec hc rc funcall nosemi emacsen uncomments
;; LocalWords: afterd befored okpos startlst endlst ellipsify foundlst
;; LocalWords: expandto stringp donext allsyms mapcar fname nondirectory
;; LocalWords: upcase Uwe Brauer oub eucmos ucm documentclass usepackage
;; LocalWords: lstloadlanguages lstset keywordstyle bfseries labelstep
;; LocalWords: escapechar lstlisting hideshow func PUSHNEW pushnew hs bn un
;; LocalWords: msgpos aref nodesktop mlfile emacsinit emacsclient comint gud
;; LocalWords: msbn subprocess pc nconc kbd Thx Chappaz windowid dirtrackp xpm
;; LocalWords: commint postoutput ndbhotlink formatter overlaystack dolist ef
;; LocalWords: mello realfname aset buf noselect pmark memq promptend numchars
;; LocalWords: integerp emacsdocomplete mycmd ba FCN's BUILTINFLAG substr
;; LocalWords: emacsdocompletion subfield usr fil byteswap bw ignoredups mapc
;; LocalWords: noshow lastcmd dired numberp princ matlabregex stackexchange
;; LocalWords: doesnt stacktop basec sk downcase mfile dirs Stelios Kyriacou
;; LocalWords: kyriacou cbmv jhu nexti Keybindings ui mcos XGrid YGrid YLabel
;; LocalWords: ie ret proc acc misconfigured vdp subexpression