2013-08-02 12:24:27 +00:00
|
|
|
---
|
2024-10-20 21:46:35 +00:00
|
|
|
language: Emacs Lisp
|
2013-08-02 12:24:27 +00:00
|
|
|
contributors:
|
|
|
|
- ["Bastien Guerry", "http://bzg.fr"]
|
|
|
|
translators:
|
|
|
|
- ["Guillermo Vayá", "http://willyfrog.es"]
|
2013-08-12 16:53:40 +00:00
|
|
|
filename: learn-emacs-lisp-es.el
|
2013-08-02 12:24:27 +00:00
|
|
|
---
|
|
|
|
|
|
|
|
```scheme
|
|
|
|
;; Introduccion a Emacs Lisp en 15 minutos (v0.2d)
|
|
|
|
;;
|
|
|
|
;; Autor: Bastien / @bzg2 / http://bzg.fr
|
2013-08-06 08:01:42 +00:00
|
|
|
;; Traducción: Guillermo Vayá / @Driadan / http://willyfrog.es
|
2013-08-02 12:24:27 +00:00
|
|
|
;;
|
|
|
|
;; Antes de nada, lee este texto de Peter Norvig:
|
2013-08-06 08:01:42 +00:00
|
|
|
;; Traducido: http://loro.sourceforge.net/notes/21-dias.html
|
|
|
|
;; Original: http://norvig.com/21-days.html
|
2013-08-02 12:24:27 +00:00
|
|
|
;;
|
|
|
|
;; Ahora instala GNU Emacs 24.3:
|
|
|
|
;;
|
|
|
|
;; Debian: apt-get install emacs
|
|
|
|
;; (o sigue las instrucciones de tu distribución preferida)
|
|
|
|
;; OSX: http://emacsformacosx.com/emacs-builds/Emacs-24.3-universal-10.6.8.dmg
|
|
|
|
;; Windows: http://ftp.gnu.org/gnu/windows/emacs/emacs-24.3-bin-i386.zip
|
|
|
|
;;
|
|
|
|
;; Puedes encontrar información general sobre Emacs en:
|
|
|
|
;; http://www.gnu.org/software/emacs/#Obtaining
|
|
|
|
|
|
|
|
;; Aviso importante:
|
|
|
|
;;
|
|
|
|
;; Seguir este tutorial no provocará daños en tu ordenador a menos que
|
|
|
|
;; te enfades tanto que que acabes tirándolo al suelo. En tal caso
|
|
|
|
;; declino cualquier responsabilidad. ¡A divertirse!
|
|
|
|
|
|
|
|
|
|
|
|
;; "N. del. T.": Algunos términos comunes de la informática se han dejado
|
|
|
|
;; sin traducir ya que es mucho más probable que el lector los conozca en
|
|
|
|
;; su forma en inglés, siendo la versión en español de muy raro uso.
|
|
|
|
;; Además "sexps" se ha decidido traducir por sexpresión.
|
|
|
|
;; Por último, añadir que no se han traducido los ejemplos de código ya que no
|
|
|
|
;; es necesario entender qué dice el string para comprender el funcionamiento
|
|
|
|
;; y podría llevar a error.
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;;
|
|
|
|
;; Inicia Emacs.
|
|
|
|
;;
|
|
|
|
;; Pulsa la tecla `q' para pasar el mensaje de bienvenida.
|
|
|
|
;;
|
|
|
|
;; Mira a la línea gris en la parte inferior de la ventana:
|
|
|
|
;;
|
|
|
|
;; "*scratch*" es el nombre del espacio editable donde estás.
|
|
|
|
;; A este espacio editable se le llama "buffer".
|
|
|
|
;;
|
|
|
|
;; Scratch es el buffer por defecto cuando abres Emacs.
|
|
|
|
;; En Emacs nunca editas ficheros, sino que editas buffers que
|
|
|
|
;; posteriormente pueden grabarse a un fichero.
|
|
|
|
;; can save to a file.
|
|
|
|
;;
|
|
|
|
;; "Lisp interaction" indica el conjunto de ordenes disponibles.
|
|
|
|
;;
|
|
|
|
;; Emacs dispone de un set de comandos disponibles en cualquier buffer
|
|
|
|
;; ("built-ins") y aparte varios conjuntos de ordenes disponibles
|
|
|
|
;; según el modo específico que esté activo. En nuestro caso
|
|
|
|
;; estamos usando `lisp-interaction-mode', el cual incluye las
|
|
|
|
;; ordenes necesarias para evaluar y navegar código Elisp.
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;;
|
|
|
|
;; Un punto y coma comienza un comentario. Pueden ponerse en cualquier
|
|
|
|
;; posicion de la linea.
|
|
|
|
;;
|
|
|
|
;; Los programas en Elisp se componen de expresiones simbólicas
|
|
|
|
;; tambien llamadas "sexps":
|
|
|
|
(+ 2 2)
|
|
|
|
|
|
|
|
;; Esta expresión simbólica se lee tal que "Suma 2 y 2"
|
|
|
|
|
|
|
|
;; Las sexpresiones se rodean por paréntesis, y pueden anidarse:
|
|
|
|
(+ 2 (+ 1 1))
|
|
|
|
|
|
|
|
;; Una expresion simbólica está formada bien por átomos o bien por otras
|
|
|
|
;; expresiones simbólicas. En el ejemplo de arriba, 1 y 2 son átomos,
|
|
|
|
;; mientras que (+ 2 (+ 1 1)) y (+ 1 1) son expresiones simbólicas.
|
|
|
|
|
|
|
|
;; Gracias a `lisp-interaction-mode' puedes evaluar las sexpresiones.
|
|
|
|
;; Coloca el cursor justo despues del paréntesis de cierre y
|
|
|
|
;; mantén pulsada la tecla Control y la j (para abreviar usaremos "C-j").
|
|
|
|
|
|
|
|
(+ 3 (+ 1 2))
|
|
|
|
;; ^ pon aquí el cursor
|
|
|
|
;; `C-j' => 6
|
|
|
|
|
|
|
|
;; `C-j' añade el resultado de la evaluación al buffer.
|
|
|
|
|
|
|
|
;; `C-xC-e' muestra el mismo resultado pero en la linea inferior
|
2024-05-13 10:29:03 +00:00
|
|
|
;; la cual se llama "echo area". Este será el metodo que usaremos
|
2013-08-02 12:24:27 +00:00
|
|
|
;; normalmente para no llenar el buffer con texto inútil.
|
|
|
|
|
|
|
|
;; `setq' guarda un valor en una variable:
|
|
|
|
(setq my-name "Bastien")
|
2024-05-13 10:29:03 +00:00
|
|
|
;; `C-xC-e' => "Bastien" (aparece en el echo area)
|
2013-08-02 12:24:27 +00:00
|
|
|
|
|
|
|
;; `insert' añade "Hello!" en el punto donde esté tu cursor:
|
|
|
|
(insert "Hello!")
|
|
|
|
;; `C-xC-e' => "Hello!"
|
|
|
|
|
|
|
|
;; Aunque hemos usado `insert' con solo un parámetro "Hello!", se
|
|
|
|
;; pueden pasar más. Por ejemplo, en esta otra sexpresión usamos dos:
|
|
|
|
|
|
|
|
(insert "Hello" " world!")
|
|
|
|
;; `C-xC-e' => "Hello world!"
|
|
|
|
|
|
|
|
;; Se pueden usar variables en lugar de strings:
|
|
|
|
(insert "Hello, I am " my-name)
|
|
|
|
;; `C-xC-e' => "Hello, I am Bastien"
|
|
|
|
|
|
|
|
;; Puedes combinar sexpresiones en funciones:
|
|
|
|
(defun hello () (insert "Hello, I am " my-name))
|
|
|
|
;; `C-xC-e' => hello
|
|
|
|
|
|
|
|
;; Evaluemos la funcion:
|
|
|
|
(hello)
|
|
|
|
;; `C-xC-e' => Hello, I am Bastien
|
|
|
|
|
|
|
|
;; Los parentesis vacios en la definicion de una funcion indican
|
|
|
|
;; que no acepta parámetros. En cualquier caso, usar `my-name' siempre
|
|
|
|
;; es aburrido, asi que vamos a hacer que la función accepte un parámetro
|
|
|
|
;; (en este caso el parametro se llama "name"):
|
|
|
|
(defun hello (name) (insert "Hello " name))
|
|
|
|
;; `C-xC-e' => hello
|
|
|
|
|
|
|
|
;; Ahora vamos a llamar a la funcion con el string "you" como valor para
|
|
|
|
;; el único parámetro que posee.
|
|
|
|
(hello "you")
|
|
|
|
;; `C-xC-e' => "Hello you"
|
|
|
|
|
|
|
|
;; ¡Genial!
|
|
|
|
|
|
|
|
;; Descansa un poco y respira.
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;;
|
|
|
|
;; Ahora cambiaremos al nuevo buffer, llamado "*test*", en una nueva ventana.
|
|
|
|
|
|
|
|
(switch-to-buffer-other-window "*test*")
|
|
|
|
;; `C-xC-e'
|
|
|
|
;; => [La pantalla ahora tiene dos ventanas y el cursor está en el buffer *test*]
|
|
|
|
|
|
|
|
;; Mueve el ratón sobre la ventana superior y pulsa el boton izdo. para volver.
|
|
|
|
;; Otra forma es usando `C-xo' (pulsa simultaneamente control y x y luego la o)
|
|
|
|
;; para ir a la otra ventana.
|
|
|
|
|
|
|
|
;; Se pueden combinar varias sexpresiones mediante `progn':
|
|
|
|
(progn
|
|
|
|
(switch-to-buffer-other-window "*test*")
|
|
|
|
(hello "you"))
|
|
|
|
;; `C-xC-e'
|
|
|
|
;; => [De las dos ventanas de la pantalla, el cursor está en la marcada como *test*]
|
|
|
|
|
|
|
|
;; A partir de ahora, si no te importa, dejaremos de decir que pulses `C-xC-e':
|
|
|
|
;; tendrás que hacerlo para ejecutar cada sexpresión que siga.
|
|
|
|
|
|
|
|
;; También tendrás que volver al buffer *scratch* bien con el ratón o con `C-xo'.
|
|
|
|
|
|
|
|
;; En ocasiones será util limpiar el buffer:
|
|
|
|
(progn
|
|
|
|
(switch-to-buffer-other-window "*test*")
|
|
|
|
(erase-buffer)
|
|
|
|
(hello "there"))
|
|
|
|
|
|
|
|
;; O volver a la ventana anterior:
|
|
|
|
(progn
|
|
|
|
(switch-to-buffer-other-window "*test*")
|
|
|
|
(erase-buffer)
|
|
|
|
(hello "you")
|
|
|
|
(other-window 1))
|
|
|
|
|
|
|
|
;; Puedes enlazar un valor a una variable local con `let':
|
|
|
|
(let ((local-name "you"))
|
|
|
|
(switch-to-buffer-other-window "*test*")
|
|
|
|
(erase-buffer)
|
|
|
|
(hello local-name)
|
|
|
|
(other-window 1))
|
|
|
|
|
|
|
|
;; En este caso, no hace falta añadir `progn' ya que `let' permite combinar
|
|
|
|
;; varias sexpresiones.
|
|
|
|
|
|
|
|
;; Vamos a darle formato a un string:
|
|
|
|
(format "Hello %s!\n" "visitor")
|
|
|
|
|
|
|
|
;; Cada %s indica la posicion donde irá un string, el cual será reemplazado
|
|
|
|
;; por "visitor". "\n" es el caracter de nueva línea.
|
|
|
|
|
|
|
|
;; Mejoremos nuestra funcion usando `format':
|
|
|
|
(defun hello (name)
|
|
|
|
(insert (format "Hello %s!\n" name)))
|
|
|
|
|
|
|
|
(hello "you")
|
|
|
|
|
|
|
|
;; Creemos una nueva funcion que utililce `let':
|
|
|
|
(defun greeting (name)
|
|
|
|
(let ((your-name "Bastien"))
|
|
|
|
(insert (format "Hello %s!\n\nI am %s."
|
|
|
|
name ; the argument of the function
|
|
|
|
your-name ; the let-bound variable "Bastien"
|
|
|
|
))))
|
|
|
|
|
|
|
|
;; Y ahora la evaluamos:
|
|
|
|
(greeting "you")
|
|
|
|
|
|
|
|
;; Algunas funciones son interactivas:
|
|
|
|
(read-from-minibuffer "Enter your name: ")
|
|
|
|
|
|
|
|
;; Al evaluar esta función, ésta devuelve lo que hayas introducido.
|
|
|
|
|
|
|
|
;; Ahora hagamos nuestra función `greeting' preguntar por tu nombre:
|
|
|
|
(defun greeting (from-name)
|
|
|
|
(let ((your-name (read-from-minibuffer "Enter your name: ")))
|
|
|
|
(insert (format "Hello!\n\nI am %s and you are %s."
|
|
|
|
from-name ; the argument of the function
|
|
|
|
your-name ; the let-bound var, entered at prompt
|
|
|
|
))))
|
|
|
|
|
|
|
|
(greeting "Bastien")
|
|
|
|
|
|
|
|
;; Y ahora la completamos mostrando el resultado en la otra ventana:
|
|
|
|
(defun greeting (from-name)
|
|
|
|
(let ((your-name (read-from-minibuffer "Enter your name: ")))
|
|
|
|
(switch-to-buffer-other-window "*test*")
|
|
|
|
(erase-buffer)
|
|
|
|
(insert (format "Hello %s!\n\nI am %s." your-name from-name))
|
|
|
|
(other-window 1)))
|
|
|
|
|
|
|
|
;; Probémosla:
|
|
|
|
(greeting "Bastien")
|
|
|
|
|
|
|
|
;; Descansa un poco y respira.
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;;
|
|
|
|
;; Creemos una lista de nombres:
|
|
|
|
(setq list-of-names '("Sarah" "Chloe" "Mathilde"))
|
|
|
|
|
|
|
|
;; Para coger el primer elemento de la lista usaremos `car':
|
|
|
|
(car list-of-names)
|
|
|
|
|
|
|
|
;; Para coger todos menos el primer elemento de la lista
|
|
|
|
;; usaremos `cdr':
|
|
|
|
(cdr list-of-names)
|
|
|
|
|
|
|
|
;; Para añadir un elemento al comienzo de la lista utilizamos `push':
|
|
|
|
(push "Stephanie" list-of-names)
|
|
|
|
|
|
|
|
;; OJO: `car' y `cdr' no modifican la lista, mientras que `push' sí.
|
|
|
|
;; ¡Es una diferencia importante! Algunas funciones no tienen efectos
|
|
|
|
;; colaterales (como `car') mientras que otras sí (como `push').
|
|
|
|
;; "N. del T.": estos efectos colaterales se les llama `side-effects' en
|
|
|
|
;; las distintas variantes de lisp.
|
|
|
|
|
|
|
|
;; Llamemos a `hello' con cada elemento de `list-of-names':
|
|
|
|
(mapcar 'hello list-of-names)
|
|
|
|
|
|
|
|
;; Retocamos `greeting' para que salude a todos los que estén en `list-of-names':
|
|
|
|
(defun greeting ()
|
|
|
|
(switch-to-buffer-other-window "*test*")
|
|
|
|
(erase-buffer)
|
|
|
|
(mapcar 'hello list-of-names)
|
|
|
|
(other-window 1))
|
|
|
|
|
|
|
|
(greeting)
|
|
|
|
|
|
|
|
;; ¿Te acuerdas de la función `hello' definida un poco más arriba?
|
|
|
|
;; Recibía un parámetro: `name'. Así que `mapcar' llama a `hello' con cada
|
|
|
|
;; elemento de `list-of-names' como parámetro de `hello'.
|
|
|
|
|
|
|
|
;; Ahora ordenaremos un poco lo que tenemos en el buffer:
|
|
|
|
|
|
|
|
(defun replace-hello-by-bonjour ()
|
|
|
|
(switch-to-buffer-other-window "*test*")
|
|
|
|
(goto-char (point-min))
|
|
|
|
(while (search-forward "Hello")
|
|
|
|
(replace-match "Bonjour"))
|
|
|
|
(other-window 1))
|
|
|
|
|
|
|
|
;; (goto-char (point-min)) mueve el cursor al principio del buffer.
|
|
|
|
;; (search-forward "Hello") busca un string "Hello".
|
|
|
|
;; (while x y) evalua la/s sexpresion/es y mientras que x devuelva
|
|
|
|
;; alguna cosa.
|
|
|
|
;; En el momento que x devuelva `nil' (es decir nada), sale del
|
|
|
|
;; bucle `while'.
|
|
|
|
|
|
|
|
(replace-hello-by-bonjour)
|
|
|
|
|
|
|
|
;; Observamos que todas las veces que teníamos la palabra "Hello" en el buffer *test*
|
|
|
|
;; han sido reemplazadas por "Bonjour".
|
|
|
|
|
|
|
|
;; Y además, hemos obtenido un error: "Search failed: Hello".
|
|
|
|
;;
|
|
|
|
;; Para evitar este error, hay que decirle a `search-forward' si debería dejar de
|
|
|
|
;; buscar en el buffer en algún momento y si debería fallar sin quejarse cuando
|
|
|
|
;; no encuentra nada.
|
|
|
|
|
|
|
|
;; (search-forward "Hello" nil t) justo hace eso:
|
|
|
|
|
|
|
|
;; El argumento `nil' significa que la busqueda no está ligada a ninguna posición.
|
|
|
|
;; Y el argumento `t' le pide que no diga nada si no encuentra el string.
|
|
|
|
|
|
|
|
;; Usaremos esta sexpresión en la función siguiente, la cual ya
|
|
|
|
;; no muestra ningún error:
|
|
|
|
|
|
|
|
(defun hello-to-bonjour ()
|
|
|
|
(switch-to-buffer-other-window "*test*")
|
|
|
|
(erase-buffer)
|
|
|
|
;; Say hello to names in `list-of-names'
|
|
|
|
(mapcar 'hello list-of-names)
|
|
|
|
(goto-char (point-min))
|
|
|
|
;; Replace "Hello" by "Bonjour"
|
|
|
|
(while (search-forward "Hello" nil t)
|
|
|
|
(replace-match "Bonjour"))
|
|
|
|
(other-window 1))
|
|
|
|
|
|
|
|
(hello-to-bonjour)
|
|
|
|
|
|
|
|
;; Añadamos algo de color a los nombres:
|
|
|
|
|
|
|
|
(defun boldify-names ()
|
|
|
|
(switch-to-buffer-other-window "*test*")
|
|
|
|
(goto-char (point-min))
|
|
|
|
(while (re-search-forward "Bonjour \\(.+\\)!" nil t)
|
|
|
|
(add-text-properties (match-beginning 1)
|
|
|
|
(match-end 1)
|
|
|
|
(list 'face 'bold)))
|
|
|
|
(other-window 1))
|
|
|
|
|
|
|
|
;; Esta función nos presenta `re-search-forward': en vez de
|
|
|
|
;; buscar el string "Bonjour" exacto, se busca por un patrón
|
|
|
|
;; usando una "expresión regular" (lo cual se muestra abreviado
|
|
|
|
;; en el prefijo "re-" del inglés "Regular Expression").
|
|
|
|
|
|
|
|
;; La expresión regular a utilizar es "Bonjour \\(.+\\)!" y se traduce como:
|
|
|
|
;; el string "Bonjour ", seguido de
|
|
|
|
;; un grupo de | representado por \\( ... \\)
|
|
|
|
;; cualquier caracter | representado por .
|
|
|
|
;; al menos una vez | representado por +
|
|
|
|
;; y el string "!".
|
|
|
|
|
|
|
|
;; ¿Preparado? ¡Probemoslo!
|
|
|
|
|
|
|
|
(boldify-names)
|
|
|
|
|
|
|
|
;; `add-text-properties' añade propiedades al texto, como una fuente.
|
|
|
|
|
|
|
|
;; ¡Hale! ¡Ya lo tenemos! ¡Feliz hacking!
|
|
|
|
|
|
|
|
;; Si quieres saber más sobre una función o una variable:
|
|
|
|
;;
|
|
|
|
;; C-h v la-variable RET
|
|
|
|
;; C-h f la-funcion RET
|
|
|
|
;;
|
|
|
|
;; Si quieres leer el manual de Emacs Lisp desde dentro de Emacs:
|
|
|
|
;;
|
|
|
|
;; C-h i m elisp RET
|
|
|
|
;;
|
|
|
|
;; Para leer una introducción en linea de Emacs Lisp:
|
|
|
|
;; https://www.gnu.org/software/emacs/manual/html_node/eintr/index.html
|
|
|
|
|
|
|
|
;; Me gustaría agradecer a las siguientes personas su feedback y sugerencias:
|
|
|
|
;; - Wes Hardaker
|
|
|
|
;; - notbob
|
|
|
|
;; - Kevin Montuori
|
|
|
|
;; - Arne Babenhauserheide
|
|
|
|
;; - Alan Schmitt
|
|
|
|
;; - LinXitoW
|
|
|
|
;; - Aaron Meurer
|
|
|
|
```
|