# Capítulo 26. Variables de Vimscript y su ámbito Antes de adentrarnos en las funciones de Vimscript, vamos a aprender las diferentes fuentes y ámbitos en los que operan las variables en Vim. ## Variables mutables e inmutables Puedes asignar un valor a una variable en Vim con `let`: ``` let tortitas = "tortitas" ``` Después puedes llamar a la variable en cualquier momento. ``` echo tortitas " devuelve "tortitas" ``` `let` es mutable, esto significa que puedes cambiar el valor de la variable en cualquier momento en el futuro. ``` let tortitas = "tortitas" let tortitas = "no gofres" echo tortitas " devuelve "no gofres" ``` Ten en cuenta que cuando quieres cambiar el valor de una variable, necesitas utilizar `let`. ``` let bebida = "leche" bebida = "zumo de naranja" " lanza un error ``` Puedes definir variables inmutables con `const`. Al ser inmutables, una vez que a la variable se le asigna un valor, ya no puedes reasignarle un valor diferente más adelante. ``` const gofres = "gofres" const gofres = "tortitas" " lanza un error ``` ## Fuentes de variable Hay tres fuentes de las que obtener los valores de variables: variables de entorno, variables de opciones y variables de registro. ### Variables de entorno Vim puede acceder a las variables de entorno de tu terminal. Por ejemplo, si tienes la variable `SHELL` de entorno en tu terminal, puedes acceder a ella desde Vim mediante: ``` echo $SHELL " devuelve el valor de $SHELL. E mi caso devuelve /bin/bash ``` ### Variables de opciones Puedes acceder a las opciones de Vim con `&` (estos son ajustes a los que accedes con `set`). Por ejemplo, para ver qué fondo utiliza Vim, puedes ejecutar: ``` echo &background " devuelve un valor que puede ser "light" o "dark" (claro u oscuro) ``` De manera alternativa, siempre puedes ejecutar `set background?` para ver el valor de la opción `background`. ### Variables de registro Puedes acceder a los registros de Vim (ver el capítulo 08) mediante `@`. Supongamos que el valor "chocolate" está ya guardado en el registro a. Para acceder a el, puedes utilizar `@a`. También puedes actualizarlo con `let`. ``` echo @a " devuelve chocolate let @a .= " donut" echo @a " devuelve "chocolate donut" ``` Ahora cuando pegues el contenido del registro a (`"ap`), este mostrará el contenido que es "chocolate donut". El operador `.=` une dos cadenas. La expresión `let @a .= " donut"` es lo mismo que `let @a = @a . " donut"` ## Ámbito de las variables Hay 9 diferentes ámbitos en los que pueden operar las variables en Vim. Puedes reconocerlos porque les antecede una letra: ``` g: Variable global {nothing} Variable global b: Variable de buffer local w: Variable de ventana local t: Variable de pestaña local s: Variable procedente de Vimscript l: Variable de función local a: Variable de parámetro de función formal v: Variable propia de Vim ``` ### Global variable When you are declaring a "regular" variable: ``` let pancake = "pancake" ``` `pancake` is actually a global variable. When you define a global variable, you can call them from anywhere. Prepending `g:` to a variable also creates a global variable. ``` let g:waffle = "waffle" ``` In this case both `pancake` and `g:waffle` have the same scope. You can call each of them with or without `g:`. ``` echo pancake " devuelve "pancake" echo g:pancake "returns "pancake" echo waffle " devuelve "waffle" echo g:waffle " devuelve "waffle" ``` ### Buffer Variable A variable preceded with `b:` is a buffer variable. A buffer variable is a variable that is local to the current buffer (Ch. 02). If you have multiple buffers open, each buffer will have their own separate list of buffer variables. In buffer 1: ``` const b:donut = "chocolate donut" ``` In buffer 2: ``` const b:donut = "blueberry donut" ``` If you run `echo b:donut` from buffer 1, it will return "chocolate donut". If you run it from buffer 2, it will return "blueberry donut". On the side note, Vim has a *special* buffer variable `b:changedtick` that keeps track of all the changes done to the current buffer. 1. Run `echo b:changedtick` and note the number it returns.. 2. Make changes in Vim. 3. Run `echo b:changedtick` again and note the number it now returns. ### Window Variable A variable preceded with `w:` is a window variable. It exists only in that window. In window 1: ``` const w:donut = "chocolate donut" ``` In window 2: ``` const w:donut = "raspberry donut" ``` On each window, you can call `echo w:donut` to get unique values. ### Tab Variable A variable preceded with `t:` is a tab variable. It exists only in that tab. In tab 1: ``` const t:donut = "chocolate donut" ``` In tab 2: ``` const t:donut = "blackberry donut" ``` On each tab, you can call `echo t:donut` to get unique values. ### Script variable A variable preceded with `s:` is a script variable. These variables can only be accessed from inside that script. If you have an arbitrary file `dozen.vim` and inside it you have: ``` let s:dozen = 12 function Consume() let s:dozen -= 1 echo s:dozen " is left" endfunction ``` Source the file with `:source dozen.vim`. Now call the `Consume` function: ``` :call Consume() " devuelve "11 is left" :call Consume() " devuelve "10 is left" :echo s:dozen " Undefined variable error ``` When you call `Consume`, you see it decrements the `s:dozen` value as expected. When you try to get `s:dozen` value directly, Vim won't find it because you are out of scope. `s:dozen` is only accessible from inside `dozen.vim`. Each time you source the `dozen.vim` file, it resets the `s:dozen` counter. If you are in the middle of decrementing `s:dozen` value and you run `:source dozen.vim`, the counter resets back to 12. This can be a problem for unsuspecting users. To fix this issue, refactor the code: ``` if !exists("s:dozen") let s:dozen = 12 endif function Consume() let s:dozen -= 1 echo s:dozen endfunction ``` Now when you source `dozen.vim` while in the middle of decrementing, Vim reads `!exists("s:dozen")`, finds that it is true, and doesn't reset the value back to 12. ### Function Local And Function Formal Parameter variable Both the function local variable (`l:`) and the function formal variable (`a:`) will be covered in the next chapter. ### Built-in Vim Variables A variable prepended with `v:` is a special built-in Vim variable. You cannot define these variables. You have seen some of them already. - `v:version` tells you what Vim version you are using. - `v:key` contains the current item value when iterating through a dictionary. - `v:val` contains the current item value when running a `map()` or `filter()` operation. - `v:true`, `v:false`, `v:null`, and `v:none` are special data types. There are other variables. For a list of Vim built-in variables, check out `:h vim-variable` or `:h v:`. ## Using Vim Variable Scopes The Smart Way Being able to quickly access environment, option, and register variables give you a broad flexibility to customize your editor and terminal environment. You also learned that Vim has 9 different variable scopes, each existing under a certain constraints. You can take advantage of these unique variable types to decouple your program. You made it this far. You learned about data types, means of combinations, and variable scopes. Only one thing is left: functions.