add a menu

Ignore-this: 6ead244dfba763748526761ade907ac4

darcs-hash:20120203103140-c41ad-ce51c697f42bdfbcd0212b18581ac1f1e2a0c36b
This commit is contained in:
Jeremie Dimino 2012-02-03 11:31:40 +01:00
parent 21ebe44ff0
commit 2f218dd339
1 changed files with 111 additions and 22 deletions

View File

@ -5,6 +5,38 @@
;; ;;
;; This file is a part of utop. ;; This file is a part of utop.
(require 'easymenu)
;; +-----------------------------------------------------------------+
;; | License |
;; +-----------------------------------------------------------------+
(defconst utop-license "BSD3"
"Copyright (c) 2011, Jeremie Dimino <jeremie@dimino.org>
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
* Neither the name of Jeremie Dimino nor the names of his
contributors may be used to endorse or promote products derived
from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE AUTHOR AND CONTRIBUTORS BE LIABLE FOR ANY
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.")
;; +-----------------------------------------------------------------+ ;; +-----------------------------------------------------------------+
;; | Customizable variables | ;; | Customizable variables |
;; +-----------------------------------------------------------------+ ;; +-----------------------------------------------------------------+
@ -73,7 +105,20 @@ This hook is only run if exiting actually kills the buffer."
(defvar utop-process nil (defvar utop-process nil
"The Lisp-object for the utop sub-process") "The Lisp-object for the utop sub-process")
(defvar utop-mode-map nil (defvar utop-mode-map
(let ((map (make-sparse-keymap)))
(define-key map [return] 'utop-send-input)
(define-key map [(control ?m)] 'utop-send-input)
(define-key map [(control ?j)] 'utop-send-input)
(define-key map [home] 'utop-bol)
(define-key map [(control ?a)] 'utop-bol)
(define-key map [(meta ?p)] 'utop-history-goto-prev)
(define-key map [(meta ?n)] 'utop-history-goto-next)
(define-key map [tab] 'utop-complete)
(define-key map [(control ?c) (control ?c)] 'utop-interrupt)
(define-key map [(control ?c) (control ?i)] 'utop-interrupt)
(define-key map [(control ?c) (control ?k)] 'utop-kill)
map)
"The utop local keymap.") "The utop local keymap.")
(defvar utop-prompt-min 0 (defvar utop-prompt-min 0
@ -366,6 +411,16 @@ sub-process."
(setq lines (cdr lines))) (setq lines (cdr lines)))
(process-send-string utop-process "end:\n")))))))) (process-send-string utop-process "end:\n"))))))))
(defun utop-end-phrase-and-send-input ()
"End the current phrase and send it to ocaml."
(interactive)
(with-current-buffer utop-buffer-name
(when (eq utop-state 'edit)
(goto-char (point-max))
(when (= utop-prompt-max (point-max)) (insert "()"))
(insert ";;")
(utop-send-input))))
;; +-----------------------------------------------------------------+ ;; +-----------------------------------------------------------------+
;; | Completion | ;; | Completion |
;; +-----------------------------------------------------------------+ ;; +-----------------------------------------------------------------+
@ -477,12 +532,15 @@ To automatically do that just add these lines to your .emacs:
(autoload 'utop-tuareg-setup \"utop\" \"Toplevel for OCaml\" t) (autoload 'utop-tuareg-setup \"utop\" \"Toplevel for OCaml\" t)
(add-hook 'tuareg-mode-hook 'utop-tuareg-setup)" (add-hook 'tuareg-mode-hook 'utop-tuareg-setup)"
(interactive) (interactive)
;; Redefine tuareg functions
(defun tuareg-eval-phrase () (interactive) (utop-eval-phrase)) (defun tuareg-eval-phrase () (interactive) (utop-eval-phrase))
(defun tuareg-eval-region (start end) (interactive "r") (utop-eval-region start end)) (defun tuareg-eval-region (start end) (interactive "r") (utop-eval-region start end))
(defun tuareg-eval-buffer () (interactive) (utop-eval-buffer)) (defun tuareg-eval-buffer () (interactive) (utop-eval-buffer))
(defun tuareg-interrupt-caml () (interactive) (utop-interrupt)) (defun tuareg-interrupt-caml () (interactive) (utop-interrupt))
(defun tuareg-kill-caml () (interactive) (utop-kill)) (defun tuareg-kill-caml () (interactive) (utop-kill))
(defun tuareg-run-caml () (interactive) (utop)) (defun tuareg-run-caml () (interactive) (utop))
;; Redefine this variable so menu will work
(setq tuareg-interactive-buffer-name utop-buffer-name)
nil) nil)
;; +-----------------------------------------------------------------+ ;; +-----------------------------------------------------------------+
@ -537,6 +595,38 @@ To automatically do that just add these lines to your .emacs:
;; Make the whole buffer read-only ;; Make the whole buffer read-only
(add-text-properties (point-min) (point-max) utop-non-editable-properties))))))))) (add-text-properties (point-min) (point-max) utop-non-editable-properties)))))))))
;; +-----------------------------------------------------------------+
;; | Menu |
;; +-----------------------------------------------------------------+
(defun utop-is-running ()
(let ((buf (get-buffer utop-buffer-name)))
(when buf
(with-current-buffer buf
(and utop-process (eq (process-status utop-process) 'run))))))
(defun utop-about ()
(interactive)
(describe-variable 'utop-license))
(defun utop-help ()
(interactive)
(describe-function 'utop))
(easy-menu-define
utop-menu utop-mode-map
"utop menu."
'("utop"
["Start OCaml" utop t]
["Interrupt OCaml" utop-interrupt :active (utop-is-running)]
["Kill OCaml" utop-kill :active (utop-is-running)]
["Evaluate Phrase" utop-end-phrase-and-send-input :active (and (utop-is-running) (eq utop-state 'edit))]
"---"
["Customize utop" (customize-group 'utop) t]
"---"
["About" utop-about t]
["Help" utop-help t]))
;; +-----------------------------------------------------------------+ ;; +-----------------------------------------------------------------+
;; | The mode | ;; | The mode |
;; +-----------------------------------------------------------------+ ;; +-----------------------------------------------------------------+
@ -571,12 +661,9 @@ To automatically do that just add these lines to your .emacs:
(utop-start)) (utop-start))
(defun utop-mode () (defun utop-mode ()
"Caml Emacs-Lisp Toplevel. "Set the buffer mode to utop."
\\{utop-mode-map}"
;; Local variables ;; Local variables
(make-local-variable 'utop-mode-map)
(make-local-variable 'utop-process) (make-local-variable 'utop-process)
(make-local-variable 'utop-prompt-min) (make-local-variable 'utop-prompt-min)
(make-local-variable 'utop-prompt-max) (make-local-variable 'utop-prompt-max)
@ -595,26 +682,12 @@ To automatically do that just add these lines to your .emacs:
(setq major-mode 'utop-mode) (setq major-mode 'utop-mode)
(setq mode-name "utop") (setq mode-name "utop")
;; Create and use the local keymap utop-mode-map ;; Use the utop keymap
(setq utop-mode-map (make-sparse-keymap))
(use-local-map utop-mode-map) (use-local-map utop-mode-map)
;; Set the hook to call before changing the buffer ;; Set the hook to call before changing the buffer
(add-hook 'before-change-functions 'utop-before-change nil t) (add-hook 'before-change-functions 'utop-before-change nil t)
;; Define keys
(define-key utop-mode-map [return] 'utop-send-input)
(define-key utop-mode-map [(control ?m)] 'utop-send-input)
(define-key utop-mode-map [(control ?j)] 'utop-send-input)
(define-key utop-mode-map [home] 'utop-bol)
(define-key utop-mode-map [(control ?a)] 'utop-bol)
(define-key utop-mode-map [(meta ?p)] 'utop-history-goto-prev)
(define-key utop-mode-map [(meta ?n)] 'utop-history-goto-next)
(define-key utop-mode-map [tab] 'utop-complete)
(define-key utop-mode-map [(control ?c) (control ?c)] 'utop-interrupt)
(define-key utop-mode-map [(control ?c) (control ?i)] 'utop-interrupt)
(define-key utop-mode-map [(control ?c) (control ?k)] 'utop-kill)
;; Register the exit hook ;; Register the exit hook
(add-hook 'kill-buffer-hook (lambda () (run-hooks 'utop-exit-hook)) t t) (add-hook 'kill-buffer-hook (lambda () (run-hooks 'utop-exit-hook)) t t)
@ -622,7 +695,10 @@ To automatically do that just add these lines to your .emacs:
(utop-start) (utop-start)
;; Call hooks ;; Call hooks
(run-mode-hooks 'utop-mode-hook)) (run-mode-hooks 'utop-mode-hook)
;; Add the menu
(easy-menu-add utop-menu))
;; +-----------------------------------------------------------------+ ;; +-----------------------------------------------------------------+
;; | Starting utop | ;; | Starting utop |
@ -630,7 +706,20 @@ To automatically do that just add these lines to your .emacs:
;;;###autoload ;;;###autoload
(defun utop () (defun utop ()
"Start utop." "A universal toplevel for OCaml.
url: https://forge.ocamlcore.org/projects/utop/
utop is a enhanced toplevel for OCaml with many features,
including context sensitive completion.
This is the emacs frontend for utop. You can use the utop buffer
as a standard OCaml toplevel.
To complete an identifier, simply press TAB.
Special keys for utop:
\\{utop-mode-map}"
(interactive) (interactive)
;; Create the utop buffer if it does not exists, otherwise just ;; Create the utop buffer if it does not exists, otherwise just
;; retreive it ;; retreive it