- up: ðEmacs
- tags. ðLisp ðProgLang
Emacs LispãŸãšã
Emacsã§äœ¿ãããŠããLispæ¹èš. ELISP(elisp)ãšçç¥ãããããšãå€ã.
- æ«å°Ÿååž°ã®æé©åãããªã
- åçã¹ã³ãŒã
- é¢æ°åèšèªã§ã¯ãªã.
Emacs Lisp: ææ³
å€æ°ãšå®æ°
Emacs Lisp㯠åçã¹ã³ãŒã ãæ¡çšããŠããã©ãã§ãå€ãåç §ãšå€æŽãã§ãã.
setq ã§å€æ°ã宣èšãã. defconst ãšããã·ã³ã¿ãã¯ã¹ããããããã¯å¯èªæ§ã ãã®ãã®ã§å®éã¯å€æŽã§ãããšã.
ref. Constant Variables (GNU Emacs Lisp Reference Manual)
Listæäœ
list
listãæ§ç¯ãã.
-
ã¢ãã¹ãããã£(â)ãšlistã§ã®æåã®éã
ã¢ãã¹ãããã£ã§ãªã¹ããã€ãããšäžã®å€æ°ã¯è©äŸ¡ãããªãã, listãã€ãããšè©äŸ¡ããã.
(setq a "1") (setq b "2") (setq ab '(a b)) ;=> (a b) (setq ab2 (list a b)) ;=> ("1" "2")
append
ãªã¹ããçµå.
(setq test-list
(append test-list '("baz")))
;=> ("foo" "bar" "baz")
ïŒã€ã®ãªã¹ããçµåããæ°ãããªã¹ããè¿ãã®ã§ãããå€æ°ã«bindãã.
add-to-list
ãªã¹ãã®å é ã«èŠçŽ ãè¿œå . ãã§ã«èŠçŽ ããããšãã¯çœ®ãæãã.
ãã®ããappendããããããã°ãããããã®ã§èšå®ã¯ãã£ã¡ããã.
äžæ¹ã²ãšã€ã®èŠçŽ ãããªã¹ãã«è¿œå ã§ããªã.
ã€ã³ã¯ãªã¡ã³ã
1+ 1-ãšããèšæ³ããã. ãŸã incf, decf ãšããé¢æ°ããã.
(setq test-value 0)
(setq test-value (+ test-value 1))
(setq test-value (1+ test-value))
(incf test-value)
(setq test-value (1- test-value))
(decf test-value)
ã¬ãŒãåŠç
æ¡ä»¶ãã¿ãããªãå Žåã¯, åŠçãéäžã§äžæããã. return ã è¿ãã®ã¯çºæ³ãæç¶ãç.
if, when ãå©çšãã.
é 延è©äŸ¡
èµ·åæã«å¿ ãèªã¿èŸŒãå¿ èŠãªãé¢æ°ãªã autoload ã䜿ã - ããããŒãã¡ã¢
ããã ãš, Emacs èµ·åæã« require ããã.
(require 'js2-mode)
(add-to-list 'auto-mode-alist '("\\.js$" . js2-mode))
ããã ãš, .js ãéãããšãã« require ããã.
(autoload 'js2-mode "js2-mode" nil t)
(add-to-list 'auto-mode-alist '("\\.js$" . js2-mode))
Scope
lexical scope
Emacs ã®æšæºã¯ãã€ãããã¯ã¹ã³ãŒã.
ããã, Emacs24 ããã¯, ãã¡ã€ã«ã®å é ã«ä»¥äžãæžããš, setq ã§å®£èšããå€æ°ã¯æ¬åœã® Lexical Scope ã«ãªã.
-*- lexical-binding: t -*-
é«éé¢æ°ãåŒæ°ã«ããããšãããšã, M-x eval-buffer ããããšè©äŸ¡ã§ãã.
å±æå€æ°
let, let*, letrec ãå©çšã§ãã.
é¢æ°åãã©ãã€ã ã§ç¶æ ããã€ä»çµã¿.
let*㯠haskell ã® do èšæ³ã«äŒŒãŠãã.
ããŒã«ã«é¢æ°ã®å®çŸ©
let + lambda ãå©çšãã, å€æ°ã«ç¡åé¢æ°ã bind ãããããšã§å®çŸãã.
(let ((p (lambda (a) (message a))))
(funcall p "hoge"))
letrec ãå©çšããæ¹ãæ£åŒã?
letrec ã® rec 㯠ååž°ã®ããš. let ã¯ååž°é¢æ°ãå®çŸ©ã§ããªãã, letrec ã¯ã§ãã.
macro ãå©çš
- cl-flet é¢æ°åå®çŸ©
- cl-labels ããŒã«ã«é¢æ°
- cl-letf é¢æ°ã®äžæ眮ãæã
- noflet
泚æ: flet, lables, letf 㯠obsolite ããã
Elisp: æåå
ã¿ã€ã ã¹ã¿ã³ã
ãã€ãæ©ããã€.
weeklyã®ãã©ãŒããã
(format-time-string "%Y-w%W")
(defun my/create-weekly-org-file (path)
(expand-file-name (format "%s.org" (format-time-string "%Y-w%W")) path))
Emacs API
䞻㫠Emacs ãæäœããããã®é¢æ°ããŸãšãã
ãã¡ã€ã«æäœ
- concat: æååçµå.
- file-truename: ã·ã³ããªãã¯ãªã³ã¯ã絶察ãã¹ã«ãã.
ãã€ã³ã/ã«ãŒãœã«ç§»å
ãããã¡ã®ç¹å¥ãªäœçœ®ã瀺ãæ°å€.
- (point): ã«ã¬ã³ããããã¡ã®ãã€ã³ãã®ååŸ.
- (point-min)/(point-max): ã«ã¬ã³ããããã¡ã®æå°æ倧/ãã€ã³ãååŸ.
- (goto-char p): ãã€ã³ãäœçœ®ãžã«ãŒãœã«ç§»å.
- (goto-line n): Nè¡ç®ã«ç§»å
- (beginning-of-line)/(end-of-line): è¡é /è¡æ«ã«ç§»å.
ã«ãŒãœã«ã®äœçœ®ååŸ
things-at-point
- Emacs Lisp: Using thing-at-point
- ã«ãŒãœã«äœçœ®ã®æ å ±ã§éãŒã. - æ¥ã , ãšãã¯èªã.
ç·šé
- (insert s): æååæ¿å ¥.
ã³ãŒãã£ã³ã°èŠçŽ
ã©ããæé»ã®èŠçŽã®ãããªãã®ãããããã«æããŠãªããªã.
èŠçŽ? ã®ãããªãã®ãæžããããŠããããšã«ãã.
- GNU Emacs Lisp ãªãã¡ã¬ã³ã¹ããã¥ã¢ã«: B. ãã³ããšæ £ç¿
- kyotolisp#1 LT3 çŸãã Lisp ã®æžãæ¹ (1)
ãã¡ã€ã«åœ¢åŒ
Yasnippet ã«ãã.
;;; filename --- description
;; Header ....
;;; Code:
(require 'foo)
(defgroup hogegroup nil
"Hoge in Emacs"
:prefix "hoge:"
:group 'hoge)
(defcustom hoge:xxx nil
"Hoge valuable"
:group 'hoge
:type 'string)
; ...
(defvar hoge:foo nil)
; ...
(defun hoge:reset ()
)
; ...
;;;###autoload
(defun hoge:hoge-start ()
"public functions"
)
; ...
(provide 'hoge)
;;; filename ends here
;; filename --- desc
;; ãããæ å ±
èšè¿°æ¹æ³ããããã«ãŸãšãŸã£ãŠãã.
GNU Emacs Lisp ãªãã¡ã¬ã³ã¹ããã¥ã¢ã«: B. ãã³ããšæ £ç¿
;;; Code:
ã³ãŒããããããæžãå§ãã.;;; Code:ãã€ãã
requiere
äŸåãã elisp ãããã°, ããã«æžã.(ç®ç«ã€ããã«)
defgroup
defcustom
autoload
ãŠãŒã¶ã«å ¬éããé¢æ°ã¯, ãã¡ã€ã«ã®æ«å°Ÿã«æžãæºãã.
æžãå§ãã«ã¯ä»¥äžã®å®£èšãæžã.
;;;###autoload
provide
ãŠãŒã¶ããã¿ããããã®å®£èš.
åœåèŠçŽ
Lisp ç³»èšèªã¯
- å°æå.
- åèªãšåèªã®é㯠- ãããã.
indent
github ã®ããŒãžãäžçªè©³ãã.
以äžã§æãã
- indent-region (C-M-\)
- lisp-indent-line (tab key)
- indent-sexp (C-M-q)
以äžã®è±æèšäºããã®æç².
Top-level functions
ãããã¬ãã«ã®é¢æ°ã¯ 1 åç®ããéå§.
Closing parentheses
ã«ãã³ã¯ãŸãšããŠéãã. ãŸãšããŠéããªãã®ã¯ C ç³»ã®èšèªã«æ £ã芪ããã 人ã®ããããšã .
;;; bad
(defun f (x)
(when (< (g x) 3)
(h x 2)
)
)
;;; good
(defun f (x)
(when (< (g x) 3)
(h x 2)))
Amount of indentation
indent ã®ã¹ããŒã¹ã¯ 2 ã€ããã.
;;; bad
(defun f (x)
(when (< (g x) 3)
(h x 2)))
;;; good
(defun f (x)
(when (< (g x) 3)
(h x 2)))
Comments
ã·ã³ã°ã«ã»ãã³ãã³ã¯, ã³ãŒãã«é¢ãã泚æ㧠ã³ãŒããšåã©ã€ã³ã«æžã.
(if (< (g x) 2) ; is it sufficiently small?
(top-level x) ; if so, abandon everything
(h y)) ; otherwise try again
2 ã€ã®ã»ãã³ãã³ã¯, æ°è¡ã®ã³ãŒãã«ãããã³ã¡ã³ã.
(when (< (g x) 2)
;; reinitialize and abandon everything
(setf *level-number* 0)
(top-level x))
3 ã€ã®ã³ã¡ã³ãã¯é¢æ°ã®èª¬ææã«å©çš.
;;; Compute the amount of space between symbols
;;; as a list of floating point values.
(defun compute-spaces (symbols)
(mapcar #'compute-single-space symbols (cdr symbols)))
Indenting special forms
ã¹ãã·ã£ã«ãã©ãŒã ã¯ãããã決ãŸã£ã indent ã®ã«ãŒã«ããã.
-
Indenting the if special form
3 ã€ã® subexpressons ããšã.
(if (= (f x) 4) (top-level x) (g x))
-
Indenting the when and unless special forms
ã¯ããã®ã©ã€ã³ã¯ æ¡ä»¶å€å®ã«ãããã®ã§, ã¯ããã®ã©ã€ã³ã«æžã. 2 çªç®ããã®ã©ã€ã³ã¯, æ¡ä»¶å€å®ã©ã€ã³ãã 2 〠indent ãäžããŠæžã.
(when (= (f x) 4) (setf *level-number* 0) (unless *do-not-reinitialize* (reinitialize-global-information x) (reinitialize-local-information)) (top-level x))
-
Indenting the let and let* special forms
ã¯ããã®ã©ã€ã³ã¯å€æ°ã®åæåã§, æ®ãã®éšåãã¹ãã·ã£ã«ãã©ãŒã ã«åœãã. å€æ°ã®åæåã¯, ã¯ããã®ã«æžã. æ®ãã®ã©ã€ã³ã¯ 2indent äžããŠæžã.
(let* ((symbols (mapcar #'compute-symbol l)) (spaces (mapcar #'compute-space symbols (cdr symbols)))) (when (verify-spacing symbols spaces) (make-spacing permanent spaces)))
-
Indenting the do and do* special forms
loop ã®éå§æ¡ä»¶, çµäºæ¡ä»¶ã¯åãæãã.æ®ãã® body 㯠2indent äžãã.
(do ((i 1 (1+ i)) (j (length l) (/ j 2))) ((= j 0) i) (iterate i j) (when (= (f x) 4) (setf *level-number* 0) (top-level x)))
ãããã°/ ãã¹ã
helpful
helpfulãšããããã±ãŒãžãã€ãããš, é¢æ°ãå€æ°ã®helpã䟿å©ã«ãªã.
https://github.com/Wilfred/helpful
helpful-at-pointã§ãã®é¢æ°ã®helpãã¿ãã.
print debug
# This is useful for printing values
(messageg "Hello (%s)" foo)
# but doesn't work so well for data structures. For that, use
(prin1 list-foo)
trace-function
é¢æ°ã®ãã¬ãŒã¹ãåºã.
(defun f (x) (+ x 3))
(defun g (x) (+ (f x) 7))
- M-x trace-function 㧠f ãéžæ.
- M-: (g 3) C-x C-e
trace-output buffer ã«åºåçµæãã§ã.untrace-all ã§è§£é€.
ååž°é¢æ°ã®ç¢ºèªã«äŸ¿å©.
(defun fact (n)
(if (= n 0) 1
(* n (fact (1- n)))))
(fact 3)
;; 1 -> (fact 3)
;; | 2 -> (fact 2)
;; | | 3 -> (fact 1)
;; | | | 4 -> (fact 0)
;; | | | 4 <- fact: 1
;; | | 3 <- fact: 1
;; | 2 <- fact: 2
;; 1 <- fact: 6
ã¢ãŒãäœæ
minor-mode
define-minor-mode ãå©çšããŠäœæãã.
ãªãã·ã§ã³
- :lighter â the name, a string, to show in the modeline
- :keymap â the modeâs keymap
- :global â specifies if the minor mode is global (default nil)
Easy-Mmode
Emacs ã« default ã§å ¥ã£ãŠãã.
Bookmarks
ç°¡åãªäŸã«ãã説æ.
- How to Make an Emacs Minor Mode « null program
- keyboard shortcuts - How to create keybindings for a custom minor mode in Emacs - Stack Overflow
ðEmacs Batch Mode
Emacs LispãScriptãšããŠå®è¡ããæ¹æ³.
Basics
âbatchãªãã·ã§ã³ã§emacsãå®è¡ãã.
emacs -batch -l ~/.emacs.editor --eval="(require 'foo)" \
--eval="(require 'bar)" \
--eval="(some-function $*)"
- -batch: batchã¢ãŒãã§Emacsãèµ·å.
- -l: ããŒããããã¡ã€ã«ãæå®
- âeval: è©äŸ¡.
- -f: é¢æ°ãå®è¡
é¢æ°ã«åŒæ°ãæž¡ã
âeval=â(some-func $*)â ã®ããã«, bashã®åŒæ°ãeval=()ã§å ãã§æž¡ã. (-fã§ã¯ãªã).
References
Emacs Lisp Tips
â *scratch* ãããã¡ãå©çšãã
ãªããæ°è»œã«è©äŸ¡ããŠã¿ãããšãã¯, defaultã§äœæããã scratch ãããã¡ãã€ãããšäŸ¿å©.
References
- éåŒãEmacs Lisp - atwikiïŒã¢ãããŠã£ãïŒ
- ããããåŠçã Emacs Lisp ã§æžãå Žå - Life is very short
- Emacs Lisp - Wikipedia
- Emacs-Lispå ¥é 2021