;;; agda2-mode.el --- Major mode for Agda2 ;;; Commentary: ;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; Dependency ;;; Code: (defvar agda2-version "2.2.2" "The version of the Agda mode. Note that, by default, the same version of the underlying Haskell library is used (see `agda2-ghci-options').") (require 'cl) ; haskell-indent requires it anyway. (set (make-local-variable 'lisp-indent-function) 'common-lisp-indent-function) (require 'comint) (require 'pp) (require 'eri) (require 'annotation) (require 'agda-input) (require 'agda2-highlight) (require 'agda2-abbrevs) (require 'haskell-indent) (require 'haskell-ghci) ;; due to a bug in haskell-mode-2.1 (setq haskell-ghci-mode-map (copy-keymap comint-mode-map)) ;; Load filladapt, if it is installed. (condition-case nil (require 'filladapt) (error nil)) (unless (fboundp 'overlays-in) (load "overlay")) ; for Xemacs (unless (fboundp 'propertize) ; for Xemacs 21.4 (defun propertize (string &rest properties) "Return a copy of STRING with text properties added. First argument is the string to copy. Remaining arguments form a sequence of PROPERTY VALUE pairs for text properties to add to the result." (let ((str (copy-sequence string))) (add-text-properties 0 (length str) properties str) str))) (unless (fboundp 'run-mode-hooks) (fset 'run-mode-hooks 'run-hooks)) ; For Emacs versions < 21. ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; Programming utilities (defmacro agda2-protect (form &optional default) "Expands to (condition-case nil FORM (error DEFAULT))." `(condition-case nil ,form (error ,default))) (put 'agda2-protect 'lisp-indent-function 0) (defmacro agda2-let (varbind funcbind &rest body) "Expands to (let* VARBIND (labels FUNCBIND BODY...))." `(let* ,varbind (labels ,funcbind ,@body))) (put 'agda2-let 'lisp-indent-function 2) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; User options (defgroup agda2 nil "Major mode for interactively developing Agda programs." :group 'languages) (defcustom agda2-include-dirs '(".") "The directories Agda uses to search for files. The directory names should be relative to the root of the current project." :type '(repeat directory) :group 'agda2) (defcustom agda2-ghci-options (list (concat "-package Agda-" agda2-version)) "Options set in GHCi before loading `agda2-toplevel-module'. Note that only dynamic options can be set using this variable." :type '(repeat string) :group 'agda2) (defcustom agda2-toplevel-module "Agda.Interaction.GhciTop" "The name of the Agda2 toplevel module." :type 'string :group 'agda2) (defcustom agda2-mode-hook '(agda2-fix-ghci-for-windows) "Hooks for `agda2-mode'." :type 'hook :group 'agda2) (defcustom agda2-indentation 'eri "*The kind of indentation used in `agda2-mode'." :type '(choice (const :tag "Haskell" haskell) (const :tag "Extended relative" eri) (const :tag "None" nil)) :group 'agda2) (defcustom agda2-information-window-max-height 0.35 "*The maximum height of the information window. A multiple of the frame height." :type 'number :group 'agda2) (defcustom agda2-fontset-name (unless (eq window-system 'mac) "fontset-agda2") "Default font to use in the selected frame when activating the Agda2 mode. This is only used if it's non-nil and Emacs is not running in a terminal. It is also ignored in Emacs 23 and up, where the improved font handling makes it unnecessary. Note that this setting (if non-nil) affects non-Agda buffers as well, and that you have to restart Emacs if you want settings to this variable to take effect." :type '(choice (string :tag "Fontset name") (const :tag "Do not change the font" nil)) :group 'agda2) (defcustom agda2-fontset-spec-of-fontset-agda2 "-*-fixed-Medium-r-Normal-*-18-*-*-*-c-*-fontset-agda2, ascii:-Misc-Fixed-Medium-R-Normal--18-120-100-100-C-90-ISO8859-1, latin-iso8859-2:-*-Fixed-*-r-*-*-18-*-*-*-c-*-iso8859-2, latin-iso8859-3:-*-Fixed-*-r-*-*-18-*-*-*-c-*-iso8859-3, latin-iso8859-4:-*-Fixed-*-r-*-*-18-*-*-*-c-*-iso8859-4, cyrillic-iso8859-5:-*-Fixed-*-r-*-*-18-*-*-*-c-*-iso8859-5, greek-iso8859-7:-*-Fixed-*-r-*-*-18-*-*-*-c-*-iso8859-7, latin-iso8859-9:-*-Fixed-*-r-*-*-18-*-*-*-c-*-iso8859-9, mule-unicode-0100-24ff:-Misc-Fixed-Medium-R-Normal--18-120-100-100-C-90-ISO10646-1, mule-unicode-2500-33ff:-Misc-Fixed-Medium-R-Normal--18-120-100-100-C-90-ISO10646-1, mule-unicode-e000-ffff:-Misc-Fixed-Medium-R-Normal--18-120-100-100-C-90-ISO10646-1, japanese-jisx0208:-Misc-Fixed-Medium-R-Normal-ja-18-*-*-*-C-*-JISX0208.1990-0, japanese-jisx0212:-Misc-Fixed-Medium-R-Normal-ja-18-*-*-*-C-*-JISX0212.1990-0, thai-tis620:-Misc-Fixed-Medium-R-Normal--24-240-72-72-C-120-TIS620.2529-1, lao:-Misc-Fixed-Medium-R-Normal--24-240-72-72-C-120-MuleLao-1, tibetan:-TibMdXA-fixed-medium-r-normal--16-160-72-72-m-160-MuleTibetan-0, tibetan-1-column:-TibMdXA-fixed-medium-r-normal--16-160-72-72-m-80-MuleTibetan-1, korean-ksc5601:-Daewoo-Mincho-Medium-R-Normal--16-120-100-100-C-160-KSC5601.1987-0, chinese-gb2312:-ISAS-Fangsong ti-Medium-R-Normal--16-160-72-72-c-160-GB2312.1980-0, chinese-cns11643-1:-HKU-Fixed-Medium-R-Normal--16-160-72-72-C-160-CNS11643.1992.1-0, chinese-big5-1:-ETen-Fixed-Medium-R-Normal--16-150-75-75-C-160-Big5.ETen-0, chinese-big5-2:-ETen-Fixed-Medium-R-Normal--16-150-75-75-C-160-Big5.ETen-0" "Specification of the \"fontset-agda2\" fontset. The \"fontset-agda2\" is the standard setting for `agda2-fontset-name'. If `agda2-fontset-name' is nil, or Emacs is run in a terminal, then \"fontset-agda2\" is not created. Note that the text \"fontset-agda2\" has to be part of the string (in a certain way; see the default setting) in order for the agda2 fontset to be created properly. Note also that the default setting may not work unless suitable fonts are installed on your system. Refer to the README file accompanying the Agda distribution for details. Note finally that you have to restart Emacs if you want settings to this variable to take effect." :group 'agda2 :type 'string) (if (and agda2-fontset-name window-system) (create-fontset-from-fontset-spec agda2-fontset-spec-of-fontset-agda2 t t)) (defun agda2-fix-ghci-for-windows () (if (string-match "windows" system-configuration) (setq haskell-ghci-program-name "ghc" haskell-ghci-program-args '("--interactive")))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; Global and buffer-local vars, initialization (defvar agda2-mode-syntax-table (let ((tbl (make-syntax-table))) ;; Set the syntax of every char to "w" except for those whose default ;; syntax in `standard-syntax-table' is `paren' or `whitespace'. (map-char-table (lambda (keys val) ;; `keys' here can be a normal char, a generic char ;; (Emacs<23), or a char range (Emacs>=23). (unless (memq (car val) (eval-when-compile (mapcar 'car (list (string-to-syntax "(") (string-to-syntax ")") (string-to-syntax " "))))) (modify-syntax-entry keys "w" tbl))) (standard-syntax-table)) ;; Then override the remaining special cases. (dolist (cs '((?{ . "(}1n") (?} . "){4n") (?- . "w 123b") (?\n . "> b") (?. . ".") (?\; . ".") (?_ . ".") (?! . "."))) (modify-syntax-entry (car cs) (cdr cs) tbl)) tbl) "Syntax table used by the Agda 2 mode: {} | Comment characters, matching parentheses. - | Comment character, word constituent. \n | Comment ender. .;_! | Punctuation. Remaining characters inherit their syntax classes from the standard syntax table if that table treats them as matching parentheses or whitespace. Otherwise they are treated as word constituents.") (defconst agda2-command-table `( (agda2-load "\C-c\C-l" (global) "Load") (agda2-load "\C-c\C-x\C-l") (agda2-compile "\C-c\C-x\C-c" (global) "Compile") (agda2-text-state "\C-c\C-x\C-d" (global) "Deactivate Agda") (agda2-quit "\C-c\C-x\C-q" (global) "Quit") (agda2-restart "\C-c\C-x\C-r" (global) "Restart") (agda2-display-implicit-arguments "\C-c\C-x\C-h" (global) "Toggle display of hidden arguments") (agda2-highlight-reload-or-clear "\C-c\C-x\C-s" (global) "Reload syntax highlighting information") (agda2-show-constraints ,(kbd "C-c C-=") (global) "Show constraints") (agda2-solveAll ,(kbd "C-c C-s") (global) "Solve constraints") (agda2-show-goals ,(kbd "C-c C-?") (global) "Show goals") (agda2-next-goal "\C-c\C-f" (global) "Next goal") ; Forward. (agda2-previous-goal "\C-c\C-b" (global) "Previous goal") ; Back. (agda2-give ,(kbd "C-c C-SPC") (local) "Give") (agda2-refine "\C-c\C-r" (local) "Refine") (agda2-make-case "\C-c\C-c" (local) "Case") (agda2-goal-type "\C-c\C-t" (local) "Goal type") (agda2-show-context "\C-c\C-e" (local) "Context (environment)") (agda2-infer-type-maybe-toplevel "\C-c\C-d" (local global) "Infer (deduce) type") (agda2-goal-and-context ,(kbd "C-c C-,") (local) "Goal type and context") (agda2-goal-and-context-and-inferred ,(kbd "C-c C-.") (local) "Goal type, context and inferred type") (agda2-compute-normalised-maybe-toplevel "\C-c\C-n" (local global) "Evaluate term to normal form") (agda2-indent ,(kbd "TAB")) (agda2-indent-reverse [S-iso-lefttab]) (agda2-indent-reverse [S-lefttab]) (agda2-indent-reverse [S-tab]) (agda2-goto-definition-mouse [mouse-2]) (agda2-goto-definition-keyboard "\M-.") (agda2-go-back "\M-*") ) "Table of commands, used to build keymaps and menus. Each element has the form (CMD KEY &optional NAME GOAL-NAME) Where NAME is the name to use in the main Agda2 menu and GOAL-NAME is for the Agda goal menu.") (defvar agda2-mode-map (let ((map (make-sparse-keymap "Agda mode"))) (define-key map [menu-bar Agda2] (cons "Agda2" (make-sparse-keymap "Agda2"))) (define-key map [down-mouse-3] 'agda2-popup-menu-3) (dolist (d (reverse agda2-command-table)) (destructuring-bind (f &optional keys kinds desc) d (if keys (define-key map keys f)) (if (member 'global kinds) (define-key map (vector 'menu-bar 'Agda2 (intern desc)) (cons desc f))))) map) "Keymap for `agda2-mode'.") (defvar agda2-goal-map (let ((map (make-sparse-keymap "Agda goal"))) (dolist (d (reverse agda2-command-table)) (destructuring-bind (f &optional keys kinds desc) d (if (member 'local kinds) (define-key map (vector (intern desc)) (cons desc f))))) map) "Keymap for agda2 goal menu.") (defvar agda2-buffer nil "Agda subprocess buffer. Set in `agda2-restart'.") (defvar agda2-process nil "Agda subprocess. Set in `agda2-restart'.") ;; Some buffer locals (defvar agda2-buffer-external-status "" "External status of an `agda2-mode' buffer (dictated by the Haskell side).") (make-variable-buffer-local 'agda2-buffer-external-status) (defconst agda2-help-address "" "Address accepting submissions of bug reports and questions.") ;; Annotation for a goal ;; {! .... !} ;; ---------- overlay: agda2-gn num, face highlight, after-string num, ;; modification-hooks (agda2-protect-goal-markers) ;; - text-props: category agda2-delim1 ;; - text-props: category agda2-delim2 ;; - text-props: category agda2-delim3 ;; - text-props: category agda2-delim4 ;; ;; Char categories for {! ... !} (defvar agda2-open-brace "{") (defvar agda2-close-brace " }") (setplist 'agda2-delim1 `(display ,agda2-open-brace)) (setplist 'agda2-delim2 `(display ,agda2-open-brace rear-nonsticky t agda2-delim2 t)) (setplist 'agda2-delim3 `(display ,agda2-close-brace agda2-delim3 t)) (setplist 'agda2-delim4 `(display ,agda2-close-brace rear-nonsticky t)) ;; Note that strings used with the display property are compared by ;; reference. If the agda2-*-brace definitions were inlined, then ;; goals would be displayed as "{{ }}n" instead of "{ }n". ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; agda2-mode ;;;###autoload (add-to-list 'auto-mode-alist '("\\.l?agda\\'" . agda2-mode)) ;;;###autoload (modify-coding-system-alist 'file "\\.l?agda\\'" 'utf-8) ;;;###autoload (define-derived-mode agda2-mode nil "Agda2" "Major mode for agda2 files. Note that when this mode is activated the default font of the current frame is changed to the fontset `agda2-fontset-name'. The reason is that Agda programs often use mathematical symbols and other Unicode characters, so we try to provide a suitable default font setting, which can display many of the characters encountered. If you prefer to use your own settings, set `agda2-fontset-name' to nil. Special commands: \\{agda2-mode-map}" (setq local-abbrev-table agda2-mode-abbrev-table indent-tabs-mode nil mode-line-process '((:eval (unless (eq 0 (length agda2-buffer-external-status)) (concat ":" agda2-buffer-external-status))))) (let ((l '(max-specpdl-size 2600 max-lisp-eval-depth 2800))) (while l (set (make-local-variable (pop l)) (pop l)))) (if (and window-system agda2-fontset-name ;; Emacs-23 uses a revamped font engine which should make ;; agda2-fontset-name unnecessary in most cases. And if it turns out ;; to be necessary, we should probably use face-remapping-alist ;; rather than set-frame-font so the special font only applies to ;; Agda buffers, and so it applies in all frames where Agda ;; buffers are displayed. (not (boundp 'face-remapping-alist))) (condition-case nil (set-frame-font agda2-fontset-name) (error (error "Unable to change the font; change agda2-fontset-name or tweak agda2-fontset-spec-fontset-agda2")))) (agda2-indent-setup) (agda2-highlight-setup) (agda2-highlight-reload) (agda2-comments-and-paragraphs-setup) (force-mode-line-update) (set-input-method "Agda")) (defun agda2-restart () "Kill and restart the *ghci* buffer and load `agda2-toplevel-module'." (interactive) (save-excursion (let ((agda2-bufname "*ghci*") (ignore-dot-ghci "-ignore-dot-ghci")) (agda2-protect (kill-buffer agda2-bufname)) ;; Make sure that the user's .ghci is not read. ;; Users can override this by adding ;; "-read-dot-ghci" to ;; `haskell-ghci-program-args'. (unless (equal (car-safe haskell-ghci-program-args) ignore-dot-ghci) (set (make-local-variable 'haskell-ghci-program-args) (cons ignore-dot-ghci haskell-ghci-program-args))) (haskell-ghci-start-process nil) (setq agda2-process haskell-ghci-process agda2-buffer haskell-ghci-process-buffer mode-name "Agda2 GHCi") (set-buffer-file-coding-system 'utf-8) (set-buffer-process-coding-system 'utf-8 'utf-8) (rename-buffer agda2-bufname))) (apply 'agda2-go ":set" agda2-ghci-options) (agda2-go ":mod +" agda2-toplevel-module) (agda2-text-state)) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; Communicating with Agda2 (defun agda2-go (&rest args) "Send the list ARGS of strings to ghci, then wait for output and execute responses, if any" (interactive) (unless (eq 'run (agda2-process-status)) ;; Try restarting automatically, but only once, in case there is ;; some major problem. (agda2-restart) (unless (eq 'run (agda2-process-status)) (error "Problem encountered. The *ghci* buffer can perhaps explain why."))) (save-excursion (haskell-ghci-go (apply 'concat (agda2-intersperse " " args)) nil)) ;;(display-buffer agda2-buffer 'not-tihs-window) (let (response) (with-current-buffer haskell-ghci-process-buffer (haskell-ghci-wait-for-output) ;; Note that the following code may be prone to race conditions ;; (make-temp-file returns a filename, not an open file). This is ;; not likely to be a problem, though. (let ((tempfile (make-temp-file "agda2-mode"))) (unwind-protect (progn (comint-write-output tempfile) (with-temp-buffer (insert-file-contents tempfile) (setq response (buffer-substring-no-properties (point-min) (point-max))))) (delete-file tempfile)))) (agda2-respond response))) (defun agda2-goal-cmd (cmd &optional want ask &rest args) "When in a goal, send CMD, goal num and range, and strings ARGS to agda2. WANT is an optional prompt. When ASK is non-nil, use minibuffer." (multiple-value-bind (o g) (agda2-goal-at (point)) (unless g (error "For this command, please place the cursor in a goal")) (let ((txt (buffer-substring-no-properties (+ (overlay-start o) 2) (- (overlay-end o) 2)))) (if (not want) (setq txt "") (when (or ask (string-match "\\`\\s *\\'" txt)) (setq txt (read-string (concat want ": ") txt)))) (apply 'agda2-go cmd (format "%d" g) (agda2-goal-Range o) (agda2-string-quote txt) args)))) ;; Note that the following function is a security risk, since it ;; evaluates code without first inspecting it. The code (supposedly) ;; comes from the Agda backend, but there could be bugs in the backend ;; which can be exploited by an attacker which manages to trick ;; someone into type-checking compromised Agda code. (defun agda2-respond (response) "Execute 'agda2_mode_code' within RESPONSE string." (while (string-match "agda2_mode_code" response) (setq response (substring response (match-end 0))) (let ((inhibit-read-only t)) (eval (read response))))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; User commands and response processing (defun agda2-load () "Load current buffer." (interactive) (agda2-go "cmd_load" (agda2-string-quote (buffer-file-name)) (agda2-list-quote agda2-include-dirs) )) (defun agda2-compile () "Compile the current module." (interactive) (agda2-go "cmd_compile" (agda2-string-quote (buffer-file-name)) (agda2-list-quote agda2-include-dirs) )) (defun agda2-load-action (gs) "Annotate new goals GS in current buffer." (agda2-annotate gs)) (defun agda2-give() "Give to the goal at point the expression in it" (interactive) (agda2-goal-cmd "cmd_give" "expression to give")) (defun agda2-give-action (old-g paren) "Update the goal OLD-G with the expression in it." (agda2-update old-g paren)) (defun agda2-refine () "Refine the goal at point by the expression in it." (interactive) (agda2-goal-cmd "cmd_refine" "expression to refine")) (defun agda2-make-case () "Refine the pattern var given in the goal. Assumes that = {!!} is on one line." (interactive) (agda2-goal-cmd "cmd_make_case" "partten var to case")) (defun agda2-make-case-action (newcls) "Replace the line at point with new clauses NEWCLS and reload." (agda2-forget-all-goals);; we reload later anyway. (let* ((p0 (point)) ;; (p1 (goto-char (agda2-decl-beginning))) (p1 (goto-char (+ (current-indentation) (line-beginning-position)))) (indent (current-column)) cl) (goto-char p0) (re-search-forward "!}" (line-end-position) 'noerr) (delete-region p1 (point)) (while (setq cl (pop newcls)) (insert cl) (if newcls (insert "\n" (make-string indent ? )))) (goto-char p1)) (agda2-load)) (defun agda2-status-action (status) "Display the string STATUS in the current buffer's mode line. \(precondition: the current buffer has to use the Agda mode as the major mode)." (setq agda2-buffer-external-status status)) (defun agda2-info-action (name text) "Insert TEXT into the Agda info buffer, display it, and display NAME in the buffer's mode line." (interactive) (with-current-buffer (get-buffer-create "*Agda2 information*") (erase-buffer) (insert text) (set-syntax-table agda2-mode-syntax-table) (set-input-method "Agda") (goto-char (point-min)) (put-text-property 0 (length name) 'face '(:weight bold) name) (setq mode-line-buffer-identification name) (save-selected-window (pop-to-buffer (current-buffer) 'not-this-window 'norecord) (shrink-window (- (window-height) (min (truncate (* (frame-height) agda2-information-window-max-height)) (max window-min-height (1+ (count-lines (point-min) (point-max)))))))))) (defun agda2-show-goals() "Show all goals." (interactive) (agda2-go "cmd_metas")) (defun agda2-show-constraints() "Show constraints." (interactive) (agda2-go "cmd_constraints")) (defun agda2-text-state () "UNDER CONSTRUCTION" (interactive) (dolist (o (overlays-in (point-min) (point-max))) (delete-overlay o)) (agda2-go "cmd_reset") (let ((inhibit-read-only t)) (annotation-preserve-mod-p-and-undo (set-text-properties (point-min) (point-max) '())) (force-mode-line-update))) (defun agda2-next-goal () "Go to the next goal, if any." (interactive) (agda2-mv-goal 'next-single-property-change 'agda2-delim2 1 (point-min))) (defun agda2-previous-goal () "Go to the previous goal, if any." (interactive) (agda2-mv-goal 'previous-single-property-change 'agda2-delim3 0 (point-max))) (defun agda2-mv-goal (change delim adjust wrapped) (agda2-let () ((go (p) (while (and (setq p (funcall change p 'category)) (not (eq (get-text-property p 'category) delim)))) (if p (goto-char (+ adjust p))))) (or (go (point)) (go wrapped) (message "No goals in the buffer")))) (defun agda2-quit () "Quit and clean up after agda2." (interactive) (agda2-protect (progn (kill-buffer agda2-buffer) (kill-buffer (current-buffer))))) (defmacro agda2-maybe-normalised (name comment cmd prompt) "This macro constructs a function NAME which runs CMD. COMMENT is used to build the function's comment. The function NAME takes a prefix argument which tells whether it should normalise types or not when running CMD (through `agda2-goal-cmd'; PROMPT, if non-nil, is used as the goal command prompt)." (let ((eval (make-symbol "eval"))) `(defun ,name (&optional not-normalise) ,(concat comment ". With a prefix argument the result is not explicitly normalised.") (interactive "P") (let ((,eval (if not-normalise "Instantiated" "Normalised"))) (agda2-goal-cmd (concat ,cmd " Agda.Interaction.BasicOps." ,eval) ,prompt))))) (defmacro agda2-maybe-normalised-toplevel (name comment cmd prompt) "This macro constructs a function NAME which runs CMD. COMMENT is used to build the function's comments. The function NAME takes a prefix argument which tells whether it should normalise types or not when running CMD (through `agda2-go'; the string PROMPT is used as the goal command prompt)." (let ((eval (make-symbol "eval"))) `(defun ,name (not-normalise expr) ,(concat comment ". With a prefix argument the result is not explicitly normalised.") (interactive ,(concat "P\nM" prompt ": ")) (let ((,eval (if not-normalise "Instantiated" "Normalised"))) (agda2-go (concat ,cmd " Agda.Interaction.BasicOps." ,eval " " (agda2-string-quote expr))))))) (agda2-maybe-normalised agda2-goal-type "Show the type of the goal at point" "cmd_goal_type" nil) (agda2-maybe-normalised agda2-infer-type "Infer the type of the goal at point" "cmd_infer" "expression to type") (agda2-maybe-normalised-toplevel agda2-infer-type-toplevel "Infers the type of the given expression. The scope used for the expression is that of the last point inside the current top-level module" "cmd_infer_toplevel" "Expression") (defun agda2-infer-type-maybe-toplevel () "Infers the type of the given expression. Either uses the scope of the current goal or, if point is not in a goal, the top-level scope." (interactive) (call-interactively (if (agda2-goal-at (point)) 'agda2-infer-type 'agda2-infer-type-toplevel))) (agda2-maybe-normalised agda2-goal-and-context "Shows the type of the goal at point and the currect context" "cmd_goal_type_context" nil) (agda2-maybe-normalised agda2-goal-and-context-and-inferred "Shows the context, the goal and the given expression's inferred type" "cmd_goal_type_context_infer" "expression to type") (agda2-maybe-normalised agda2-show-context "Show the context of the goal at point" "cmd_context" nil) (defun agda2-solveAll () "Solve all goals that are internally already instantiated." (interactive) (agda2-go "cmd_solveAll" )) (defun agda2-solveAll-action (iss) (save-excursion (while iss (let* ((g (pop iss)) (txt (pop iss))) (agda2-replace-goal g txt) (agda2-goto-goal g) (agda2-give))))) (defun agda2-compute-normalised (&optional arg) "Compute the normal form of the expression in the goal at point. With a prefix argument \"abstract\" is ignored during the computation." (interactive "P") (let ((cmd (concat "cmd_compute" (if arg " True" " False")))) (agda2-goal-cmd cmd "expression to normalise"))) (defun agda2-compute-normalised-toplevel (expr &optional arg) "Computes the normal form of the given expression. The scope used for the expression is that of the last point inside the current top-level module. With a prefix argument \"abstract\" is ignored during the computation." (interactive "MExpression: \nP") (let ((cmd (concat "cmd_compute_toplevel" (if arg " True" " False") " "))) (agda2-go (concat cmd (agda2-string-quote expr))))) (defun agda2-compute-normalised-maybe-toplevel () "Computes the normal form of the given expression, using the scope of the current goal or, if point is not in a goal, the top-level scope. With a prefix argument \"abstract\" is ignored during the computation." (interactive) (if (agda2-goal-at (point)) (call-interactively 'agda2-compute-normalised) (call-interactively 'agda2-compute-normalised-toplevel))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; (defun agda2-annotate (goals) "Annotates the goals in the current buffer with text properties. The goal numbers should be given by GOALS (in the order they appear in the buffer)." (agda2-forget-all-goals) (agda2-let (stk top) ((delims() (re-search-forward "[?]\\|[{][-!]\\|[-!][}]\\|--" nil t)) (is-lone-questionmark () (save-excursion (save-match-data (backward-char 3) (looking-at "\\(.[{(]\\|.\\s \\)[?]\\(\\s \\|[)};]\\|$\\)")))) (make(p) (agda2-make-goal p (point) (pop goals))) (err() (error "Unbalanced \{- , -\} , \{\! , \!\}"))) (save-excursion (goto-char (point-min)) (while (and goals (delims)) (labels ((c (s) (equal s (match-string 0)))) (cond ((and (c "--") (not stk)) (end-of-line)) ((c "{-") (push nil stk)) ((c "{!") (push (- (point) 2) stk)) ((c "-}") (unless (and stk (not (pop stk))) (err))) ((c "!}") (if (and stk (setq top (pop stk))) (or stk (make top)) (err))) ((c "?") (progn (when (and (not stk) (is-lone-questionmark)) (delete-char -1) (insert "{!!}") (make (- (point) 4))))))))))) (defun agda2-make-goal (p q n) "Make a goal with number N at

{!...!}. Assume the region is clean." (annotation-preserve-mod-p-and-undo (flet ((atp (x ps) (add-text-properties x (1+ x) ps))) (atp p '(category agda2-delim1)) (atp (1+ p) '(category agda2-delim2)) (atp (- q 2) '(category agda2-delim3)) (atp (1- q) '(category agda2-delim4))) (let ((o (make-overlay p q nil t nil))) (overlay-put o 'modification-hooks '(agda2-protect-goal-markers)) (overlay-put o 'agda2-gn n) (overlay-put o 'face 'highlight) (overlay-put o 'after-string (propertize (format "%s" n) 'face 'highlight))))) (defun agda2-protect-goal-markers (ol action beg end &optional length) "Ensures that the goal markers cannot be tampered with. Except if `inhibit-read-only' is non-nil or /all/ of the goal is modified." (unless inhibit-read-only (if action ;; This is the after-change hook. nil ;; This is the before-change hook. (cond ((and (<= beg (overlay-start ol)) (>= end (overlay-end ol))) ;; The user is trying to remove the whole goal: ;; manually evaporate the overlay and add an undo-log entry so ;; it gets re-added if needed. (when (listp buffer-undo-list) (push (list 'apply 0 (overlay-start ol) (overlay-end ol) 'move-overlay ol (overlay-start ol) (overlay-end ol)) buffer-undo-list)) (delete-overlay ol)) ((or (< beg (+ (overlay-start ol) 2)) (> end (- (overlay-end ol) 2))) (signal 'text-read-only nil)))))) (defun agda2-update (old-g new-txt) "Update the goal OLD-G. If NEW-TXT is a string, then the goal is replaced by the string, and otherwise the text inside the goal is retained (parenthesised if NEW-TXT is `'paren'). Removes the goal braces, but does not remove the goal overlay or text properties." (multiple-value-bind (p q) (agda2-range-of-goal old-g) (save-excursion (cond ((stringp new-txt) (agda2-replace-goal old-g new-txt)) ((equal new-txt 'paren) (goto-char (- q 2)) (insert ")") (goto-char (+ p 2)) (insert "("))) (multiple-value-bind (p q) (agda2-range-of-goal old-g) (delete-region (- q 2) q) (delete-region p (+ p 2)))))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; Misc (defun agda2-process-status () "Status of `agda2-buffer', or \"no process\"." (agda2-protect (process-status agda2-process) "no process")) (defun agda2-intersperse (sep xs) (interactive) (let(ys)(while xs (push (pop xs) ys)(push sep ys))(pop ys)(nreverse ys))) (defun agda2-goal-Range (o) "Range of goal overlay O." (interactive) (format "(Range [Interval %s %s])" (agda2-mkPos (+ (overlay-start o) 2)) (agda2-mkPos (- (overlay-end o) 2)))) (defun agda2-mkPos (&optional p) "Position value of P or point." (interactive) (save-excursion (if p (goto-char p)) (format "(Pn \"%s\" %d %d %d)" (buffer-file-name) (point) (count-lines (point-min) (point)) (1+ (current-column))))) (defun agda2-char-quote (c) "Convert character C to the notation used in Haskell strings. The non-ASCII characters are actually rendered as \"\\xNNNN\\&\", i.e. followed by a \"null character\", to avoid problems if they are followed by digits. ASCII characters (code points < 128) are converted to singleton strings." (if (< c 128) (list c) ;; FIXME: Why return a list rather than a string? --Stef (append (format "\\x%x\\&" (encode-char c 'ucs)) nil))) (defun agda2-string-quote (s) "Convert string S into a string representing it in Haskell syntax. Escape newlines, double quotes, etc.. in the string S, add surrounding double quotes, and convert non-ASCII characters to the \\xNNNN notation used in Haskell strings." (let ((pp-escape-newlines t)) (mapconcat 'agda2-char-quote (pp-to-string s) ""))) (defun agda2-list-quote (strings) "Convert a list of STRINGS into a string representing it in Haskell syntax." (concat "[" (mapconcat 'agda2-string-quote strings ", ") "]")) (defun agda2-goal-at(pos) "Return (goal overlay, goal number) at POS, or nil." (let ((os (and pos (overlays-at pos))) o g) (while (and os (not(setq g (overlay-get (setq o (pop os)) 'agda2-gn))))) (if g (list o g)))) (defun agda2-goal-overlay (g) "Returns the overlay of goal number G, if any." (car (remove nil (mapcar (lambda (o) (if (equal (overlay-get o 'agda2-gn) g) o)) (overlays-in (point-min) (point-max)))))) (defun agda2-range-of-goal (g) "The range of goal G." (let ((o (agda2-goal-overlay g))) (if o (list (overlay-start o) (overlay-end o))))) (defun agda2-goto-goal (g) (let ((p (+ 2 (car (agda2-range-of-goal g))))) (if p (goto-char p)))) (defun agda2-replace-goal (g newtxt) "Replace the content of goal G with NEWTXT." (interactive) (save-excursion (multiple-value-bind (p q) (agda2-range-of-goal g) (setq p (+ p 2) q (- q 2)) (let ((indent (and (goto-char p) (current-column)))) (delete-region p q) (insert newtxt) (while (re-search-backward "^" p t) (insert-char ? indent) (backward-char (1+ indent))))))) (defun agda2-forget-all-goals () "Remove all goal annotations. \(Including some text properties which might be used by other \(minor) modes.)" (annotation-preserve-mod-p-and-undo (remove-text-properties (point-min) (point-max) '(category nil agda2-delim2 nil agda2-delim3 nil display nil rear-nonsticky nil))) (let ((p (point-min))) (while (< (setq p (next-single-char-property-change p 'agda2-gn)) (point-max)) (delete-overlay (car (agda2-goal-at p)))))) (defun agda2-decl-beginning () "Find the beginning point of the declaration containing the point. To do: dealing with semicolon separated decls." (interactive) (save-excursion (let* ((pEnd (point)) (pDef (progn (goto-char (point-min)) (re-search-forward "\\s *" pEnd t))) (cDef (current-column))) (while (re-search-forward "where\\(\\s +\\)\\S \\|^\\(\\s *\\)\\S " pEnd t) (if (match-end 1) (setq pDef (goto-char (match-end 1)) cDef (current-column)) (goto-char (match-end 2)) (if (>= cDef (current-column)) (setq pDef (point) cDef (current-column)))) (forward-char)) (goto-char pDef) (if (equal (current-word) "mutual") (or (match-end 2) (match-end 1)) pDef)))) (defun agda2-beginning-of-decl () (interactive) (goto-char (agda2-decl-beginning))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; Indentation (defun agda2-indent () "This is what happens when TAB is pressed. Depends on the setting of `agda2-indentation'." (interactive) (cond ((eq agda2-indentation 'haskell) (haskell-indent-cycle)) ((eq agda2-indentation 'eri) (eri-indent)))) (defun agda2-indent-reverse () "This is what happens when S-TAB is pressed. Depends on the setting of `agda2-indentation'." (interactive) (cond ((eq agda2-indentation 'eri) (eri-indent-reverse)))) (defun agda2-indent-setup () "Set up and start the indentation subsystem. Depends on the setting of `agda2-indentation'." (interactive) (cond ((eq agda2-indentation 'haskell) (labels ((setl (var val) (set (make-local-variable var) val))) (setl 'indent-line-function 'haskell-indent-cycle) (setl 'haskell-indent-off-side-keywords-re "\\<\\(do\\|let\\|of\\|where\\|sig\\|struct\\)\\>[ \t]*")) (local-set-key "\177" 'backward-delete-char-untabify) (set (make-local-variable 'haskell-literate) (if (string-match "\\.lagda$" (buffer-file-name)) 'latex)) (setq haskell-indent-mode t) (run-hooks 'haskell-indent-hook)))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; Comments and paragraphs (defun agda2-comments-and-paragraphs-setup nil "Set up comment and paragraph handling for Agda mode." ;; Syntax table setup for comments is done elsewhere. ;; Enable highlighting of comments via Font Lock mode (which uses ;; the syntax table). (set (make-local-variable 'font-lock-defaults) '(nil nil nil nil nil)) ;; Empty lines (all white space according to Emacs) delimit ;; paragraphs. (set (make-local-variable 'paragraph-start) "\\s-*$") (set (make-local-variable 'paragraph-separate) paragraph-start) ;; Support for adding/removing comments. (set (make-local-variable 'comment-padding) " ") (set (make-local-variable 'comment-start) "--") ;; Support for proper filling of text in comments (requires that ;; Filladapt is activated). (when (featurep 'filladapt) (add-to-list (make-local-variable 'filladapt-token-table) '("--" agda2-comment)) (add-to-list (make-local-variable 'filladapt-token-match-table) '(agda2-comment agda2-comment) t) (add-to-list (make-local-variable 'filladapt-token-conversion-table) '(agda2-comment . exact)))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; Go to definition site (defun agda2-goto-definition-keyboard (&optional other-window) "Go to the definition site of the name under point (if any). If this function is invoked with a prefix argument then another window is used to display the given position." (interactive "P") (annotation-goto-indirect (point) other-window)) (defun agda2-goto-definition-mouse (ev prefix) "Go to the definition site of the name clicked on, if any. Otherwise, yank (see `mouse-yank-at-click')." (interactive "e\nP") (let ((pos (posn-point (event-end ev)))) (if (annotation-goto-possible pos) (annotation-goto-indirect pos) ;; FIXME: Shouldn't we use something like ;; (call-interactively (key-binding ev))? --Stef (mouse-yank-at-click ev prefix)))) (defun agda2-go-back nil "Go back to the previous position in which `agda2-goto-definition-keyboard' or `agda2-goto-definition-mouse' was invoked." (interactive) (annotation-go-back)) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; Implicit arguments (defun agda2-display-implicit-arguments (&optional arg) "Toggle display of implicit arguments. With prefix argument, turn on display of implicit arguments if the argument is a positive number, otherwise turn it off." (interactive "P") (cond ((eq arg nil) (agda2-go "toggleImplicitArgs")) ((and (numberp arg) (> arg 0)) (agda2-go "showImplicitArgs" "True")) (t (agda2-go "showImplicitArgs" "False")))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;; (defun agda2-popup-menu-3 (ev) "If in a goal, popup the goal menu and call chosen command." (interactive "e") (let (choice) (save-excursion (and (agda2-goal-at (goto-char (posn-point (event-end ev)))) (setq choice (x-popup-menu ev agda2-goal-map)) (call-interactively (lookup-key agda2-goal-map (apply 'vector choice))))))) (provide 'agda2-mode) ;;; agda2-mode.el ends here