<?xml version="1.0" encoding="utf-8"?>
<?xml-stylesheet type="text/xsl" href="../assets/xml/rss.xsl" media="all"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:atom="http://www.w3.org/2005/Atom"><channel><title>Linux Sin Humo (Publicaciones sobre Lua)</title><link>https://sergiobelkin.com/</link><description></description><atom:link href="https://sergiobelkin.com/categories/lua.xml" rel="self" type="application/rss+xml"></atom:link><language>es</language><copyright>Contents © 2026 &lt;a href="mailto:sebelk@gmail.com"&gt;sebelk&lt;/a&gt; 
&lt;a rel="license" href="https://creativecommons.org/licenses/by-nc-sa/4.0/"&gt;
&lt;img alt="Creative Commons License BY-NC-SA"
style="border-width:0; margin-bottom:12px;"
src="https://i.creativecommons.org/l/by-nc-sa/4.0/88x31.png"&gt;&lt;/a&gt;
</copyright><lastBuildDate>Thu, 28 May 2026 23:26:44 GMT</lastBuildDate><generator>Nikola (getnikola.com)</generator><docs>http://blogs.law.harvard.edu/tech/rss</docs><item><title>¿Cómo convertir tu configuración de init.vim a init.lua?</title><link>https://sergiobelkin.com/posts/como-convertir-tu-configuracion-de-initvim-a-initlua/</link><dc:creator>sebelk</dc:creator><description>&lt;h3 id="requisitos"&gt;Requisitos&lt;/h3&gt;
&lt;ol&gt;
&lt;li&gt;&lt;strong&gt;Curiosidad&lt;/strong&gt;. Es la condición más importante para leer este post. Si la tenés, con algo de esfuerzo sumado a algo de tiempo podrás aprender Vim, Neovim, etc.&lt;/li&gt;
&lt;li&gt;Conocimientos básicos o intermedios de Vim y haber leído el &lt;a href="https://sergiobelkin.com/posts/una-introduccion-pragmatica-al-uso-de-plugins-en-neovim/"&gt;post anterior&lt;/a&gt;.&lt;/li&gt;
&lt;li&gt;Conocimientos básicos de &lt;a href="https://github.com/junegunn/vim-plug"&gt;vim-plug&lt;/a&gt;.&lt;/li&gt;
&lt;/ol&gt;
&lt;h3 id="repaso-de-conceptos-importantes"&gt;Repaso de conceptos importantes&lt;/h3&gt;
&lt;p&gt;En el &lt;a href="https://sergiobelkin.com/posts/una-introduccion-pragmatica-al-uso-de-plugins-en-neovim/"&gt;post anterior&lt;/a&gt; vimos cómo un usuario de Vim puede comenzar a usar Neovim. En el presente artículo haremos la transición de &lt;strong&gt;Vim script&lt;/strong&gt; a &lt;strong&gt;Lua&lt;/strong&gt;. Ok, booting the post 😄 ...&lt;/p&gt;
&lt;p&gt;&lt;img alt="De Vim script a Lua" class="img-izquierda-responsiva" src="https://sergiobelkin.com/images/bridge_with_Vim_script_Lua_web.webp"&gt;&lt;/p&gt;
&lt;p&gt;Recordemos que Vim usa &lt;strong&gt;Vim script&lt;/strong&gt; (aka VimL) como lenguaje de scripting nativo. Es un lenguaje específico que permite personalizar y extender su funcionalidad. Existe en dos formas: el Vim script tradicional (también llamado "legacy script") y Vim9script (introducido en Vim 9.0), que ofrece mejor rendimiento y una sintaxis más moderna. Sin embargo, &lt;em&gt;Vim9script ni siquiera se usa de manera predeterminada en Vim 9.&lt;/em&gt;&lt;/p&gt;
&lt;p&gt;&lt;strong&gt;Lua&lt;/strong&gt;, en cambio, es un lenguaje de programación ligero, de alto nivel y multiparadigma, diseñado principalmente para ser incorporado en aplicaciones. Se destaca por su simplicidad, eficiencia y portabilidad. &lt;strong&gt;Neovim&lt;/strong&gt; incorporó Lua como lenguaje de scripting de primera clase, lo que ha ganado popularidad debido a su mejor rendimiento y facilidad de uso comparado con Vim script tradicional.&lt;/p&gt;
&lt;p&gt;Neovim usa una implementación de Lua llamada &lt;strong&gt;LuaJIT&lt;/strong&gt;, que compila código Lua a código máquina nativo durante la ejecución. Es un detalle técnico relevante, pero no la razón principal para preferir Lua sobre Vim script en la configuración del editor — en ese terreno el cuello de botella suele estar en el startup time y en el diseño de los plugins, no en la velocidad del lenguaje.&lt;/p&gt;
&lt;p&gt;&lt;strong&gt;Diferencia de idiosincrasia:&lt;/strong&gt;&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;Vim script:&lt;/strong&gt; scripting lineal, global y directo. Bueno para pequeños ajustes, pero difícil de escalar.&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Lua (Neovim):&lt;/strong&gt; lenguaje estructurado, orientado a modularizar, reusar y escalar. Favorece el uso de variables locales, tablas y funciones que mejoran el mantenimiento de la configuración.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;&lt;strong&gt;Por qué importa esto en la práctica:&lt;/strong&gt;&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;La &lt;strong&gt;API de Neovim en Lua&lt;/strong&gt; es estructurada y predecible (tablas, namespaces como &lt;code&gt;vim.api.*&lt;/code&gt;, &lt;code&gt;vim.opt&lt;/code&gt;), donde Vim script ofrecía variables globales y comandos planos.&lt;/li&gt;
&lt;li&gt;El &lt;strong&gt;ecosistema de plugins modernos asume Lua como lenguaje nativo&lt;/strong&gt; — los gestores actuales (&lt;code&gt;lazy.nvim&lt;/code&gt;, &lt;code&gt;packer.nvim&lt;/code&gt;), los runtime helpers, los plugins de LSP y UI conviven mejor con configs en Lua.&lt;/li&gt;
&lt;li&gt;La &lt;strong&gt;modularización con &lt;code&gt;require&lt;/code&gt;&lt;/strong&gt; permite dividir la configuración en archivos lógicos con scoping limpio, algo que Vim script soporta de forma rudimentaria.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;LuaJIT existe y es rápido, pero esa rapidez rara vez es lo que el sysadmin nota al usar Neovim — lo que nota es que su configuración escala mejor y se integra con el ecosistema sin fricciones.&lt;/p&gt;
&lt;h3 id="3-conceptos-esenciales-lo-minimo-que-siempre-debes-recordar"&gt;3 conceptos esenciales — lo mínimo que siempre debés recordar&lt;/h3&gt;
&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Concepto&lt;/th&gt;
&lt;th&gt;Por qué es esencial&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;&lt;strong&gt;Tablas (&lt;code&gt;table&lt;/code&gt;)&lt;/strong&gt;&lt;/td&gt;
&lt;td&gt;&lt;strong&gt;Las tablas son la única estructura de datos compuesta en Lua&lt;/strong&gt;. Opciones (&lt;code&gt;vim.o&lt;/code&gt;, &lt;code&gt;vim.opt&lt;/code&gt;), autocmds, comandos, mappings y configuraciones de plugins.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;strong&gt;Variables locales (&lt;code&gt;local&lt;/code&gt;)&lt;/strong&gt;&lt;/td&gt;
&lt;td&gt;Permiten organizar el código, evitar errores y mejorar el rendimiento. Imprescindible para no contaminar el entorno global.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;strong&gt;API de Neovim (&lt;code&gt;vim.api.*&lt;/code&gt;)&lt;/strong&gt;&lt;/td&gt;
&lt;td&gt;Es la manera oficial y flexible de interactuar con Neovim: crear comandos, autocmds, mappings y manipular el editor desde Lua.&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;
&lt;p&gt;&lt;strong&gt;Si entendés estos 3 conceptos, podés crear y mantener cualquier configuración básica y muchas avanzadas.&lt;/strong&gt;&lt;/p&gt;
&lt;h4 id="que-es-una-tabla-y-que-no-en-la-configuracion-de-neovim-lua"&gt;¿Qué es una tabla y qué no en la configuración de Neovim (Lua)?&lt;/h4&gt;
&lt;p&gt;&lt;strong&gt;Tablas 🆗:&lt;/strong&gt;&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;vim.g&lt;/li&gt;
&lt;li&gt;vim.opt / vim.o&lt;/li&gt;
&lt;li&gt;vim.api&lt;/li&gt;
&lt;li&gt;Configuraciones de plugins&lt;/li&gt;
&lt;li&gt;Listas/patrones&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;&lt;strong&gt;No son tablas ⚠:&lt;/strong&gt;&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;Funciones&lt;/li&gt;
&lt;li&gt;Comandos ejecutados&lt;/li&gt;
&lt;li&gt;Strings, números, booleanos&lt;/li&gt;
&lt;/ul&gt;
&lt;h3 id="el-podio-de-las-tablas-globales-mas-importantes-en-neovim"&gt;El podio de las tablas globales más importantes en Neovim&lt;/h3&gt;
&lt;p&gt;Estas son las tres tablas globales más importantes del entorno Lua en Neovim:&lt;/p&gt;
&lt;h4 id="vim"&gt;🏅 vim&lt;/h4&gt;
&lt;p&gt;Es la tabla principal y núcleo del entorno Lua en Neovim. Contiene todo lo necesario para interactuar con el editor:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;Subtablas como &lt;code&gt;vim.api&lt;/code&gt;, &lt;code&gt;vim.cmd&lt;/code&gt;, &lt;code&gt;vim.opt&lt;/code&gt;, &lt;code&gt;vim.g&lt;/code&gt;, &lt;code&gt;vim.fn&lt;/code&gt;, etc.&lt;/li&gt;
&lt;li&gt;Siempre disponible sin necesidad de importar.&lt;/li&gt;
&lt;/ul&gt;
&lt;h4 id="_g"&gt;🥈 _G&lt;/h4&gt;
&lt;p&gt;Es la tabla global del lenguaje Lua. Desde aquí podés definir funciones o variables globales accesibles desde cualquier parte de tu configuración:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;_G.mi_funcion = function() ... end&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;También contiene &lt;code&gt;vim&lt;/code&gt; como una de sus claves.&lt;/li&gt;
&lt;/ul&gt;
&lt;h4 id="package-de-lua"&gt;🥉 package (de Lua)&lt;/h4&gt;
&lt;p&gt;Maneja los módulos cargados por &lt;code&gt;require&lt;/code&gt;. Es crucial para modularizar tu configuración:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;package.loaded&lt;/code&gt; permite ver y recargar módulos.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;&lt;strong&gt;Nota&lt;/strong&gt;&lt;/p&gt;
&lt;p&gt;Tablas como &lt;code&gt;vim.api&lt;/code&gt;, &lt;code&gt;vim.opt&lt;/code&gt;, &lt;code&gt;vim.g&lt;/code&gt; son importantes, pero &lt;strong&gt;no son globales de nivel superior&lt;/strong&gt;: viven dentro de &lt;code&gt;vim&lt;/code&gt;. Lo mismo aplica para &lt;code&gt;math&lt;/code&gt;, &lt;code&gt;string&lt;/code&gt;, &lt;code&gt;table&lt;/code&gt;, etc., que viven en &lt;code&gt;_G&lt;/code&gt;.&lt;/p&gt;
&lt;p&gt;→ Estas 3 (&lt;code&gt;vim&lt;/code&gt;, &lt;code&gt;_G&lt;/code&gt;, &lt;code&gt;package&lt;/code&gt;) forman la base del entorno ejecutable de cualquier configuración Neovim moderna en Lua.&lt;/p&gt;
&lt;h3 id="subtablas-mas-importantes"&gt;Subtablas más importantes&lt;/h3&gt;
&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Tabla&lt;/th&gt;
&lt;th&gt;Contiene&lt;/th&gt;
&lt;th&gt;Ejemplo equivalente en Vim script&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;vim.o&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Opciones globales&lt;/td&gt;
&lt;td&gt;&lt;code&gt;set tabstop=4&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;vim.opt&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Acceso a opciones con métodos (&lt;code&gt;:append&lt;/code&gt;, &lt;code&gt;:remove&lt;/code&gt;, &lt;code&gt;:prepend&lt;/code&gt;); requerido para listas&lt;/td&gt;
&lt;td&gt;&lt;code&gt;set nf+=alpha&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;vim.cmd&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Comandos que se ejecutan en modo ex&lt;/td&gt;
&lt;td&gt;&lt;code&gt;colorscheme PaperColor&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;vim.g&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Variables globales&lt;/td&gt;
&lt;td&gt;&lt;code&gt;let g:loaded_perl_provider = 0&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;vim.keymap.set&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Asignación de atajos de teclado a una función o combinación de teclas&lt;/td&gt;
&lt;td&gt;&lt;code&gt;nnoremap ñ :wq&amp;lt;CR&amp;gt;&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;vim.api.nvim_create_user_command&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Creación de comandos personalizados&lt;/td&gt;
&lt;td&gt;&lt;code&gt;command Nbuild execute 'w | Silent nikola build'&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;
&lt;h3 id="diccionario-cruzado-vim-script-lua"&gt;Diccionario cruzado Vim script / Lua&lt;/h3&gt;
&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Concepto&lt;/th&gt;
&lt;th&gt;Vim script&lt;/th&gt;
&lt;th&gt;Lua&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;Variable global&lt;/td&gt;
&lt;td&gt;&lt;code&gt;let g:mi_var = 10&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;&lt;code&gt;vim.g.mi_var = 10&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Variable local&lt;/td&gt;
&lt;td&gt;&lt;code&gt;let l:var = 10&lt;/code&gt; (function-local), &lt;code&gt;s:&lt;/code&gt;, &lt;code&gt;b:&lt;/code&gt;, &lt;code&gt;w:&lt;/code&gt; por contexto&lt;/td&gt;
&lt;td&gt;&lt;code&gt;local var = 10&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Lista&lt;/td&gt;
&lt;td&gt;&lt;code&gt;let lista = [1,2,3]&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;&lt;code&gt;lista = {1, 2, 3}&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Diccionario&lt;/td&gt;
&lt;td&gt;&lt;code&gt;let dict = {'a':1}&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;&lt;code&gt;dict = {a = 1}&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Función&lt;/td&gt;
&lt;td&gt;&lt;code&gt;function! Hola() ... endfunction&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;&lt;code&gt;function hola() ... end&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Función anónima&lt;/td&gt;
&lt;td&gt;&lt;code&gt;{a, b -&amp;gt; a + b}&lt;/code&gt; (lambda, desde Vim 8.0)&lt;/td&gt;
&lt;td&gt;&lt;code&gt;function() ... end&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Autocomando&lt;/td&gt;
&lt;td&gt;&lt;code&gt;autocmd BufWrite ...&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;&lt;code&gt;vim.api.nvim_create_autocmd&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Mapping&lt;/td&gt;
&lt;td&gt;&lt;code&gt;nnoremap&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;&lt;code&gt;vim.keymap.set&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Comando&lt;/td&gt;
&lt;td&gt;&lt;code&gt;command ...&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;&lt;code&gt;vim.api.nvim_create_user_command&lt;/code&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;
&lt;h3 id="ejemplo-de-migracion-vim-script-lua"&gt;Ejemplo de migración Vim script → Lua&lt;/h3&gt;
&lt;p&gt;Bueno, con la teoría suficiente, ✋✋ a la obra 😉. Supongamos que tenemos esta configuración del archivo &lt;code&gt;init.vim&lt;/code&gt;:&lt;/p&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;&lt;span class="k"&gt;set&lt;/span&gt; &lt;span class="nb"&gt;tabstop&lt;/span&gt;&lt;span class="p"&gt;=&lt;/span&gt;&lt;span class="m"&gt;4&lt;/span&gt;
&lt;span class="k"&gt;set&lt;/span&gt; &lt;span class="nb"&gt;nf&lt;/span&gt;&lt;span class="p"&gt;+=&lt;/span&gt;alpha

&lt;span class="nb"&gt;nnoremap&lt;/span&gt; ñ :&lt;span class="k"&gt;wq&lt;/span&gt;&lt;span class="p"&gt;&amp;lt;&lt;/span&gt;CR&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;

command &lt;span class="p"&gt;-&lt;/span&gt;nargs&lt;span class="p"&gt;=+&lt;/span&gt; Silent execute &lt;span class="s1"&gt;'silent !&amp;lt;args&amp;gt;'&lt;/span&gt; &lt;span class="p"&gt;|&lt;/span&gt; &lt;span class="k"&gt;redraw&lt;/span&gt;&lt;span class="p"&gt;!&lt;/span&gt;
command Nbuild execute &lt;span class="s1"&gt;'w | Silent nikola build'&lt;/span&gt;

&lt;span class="k"&gt;filetype&lt;/span&gt; plugin &lt;span class="k"&gt;on&lt;/span&gt;
runtime plugins.&lt;span class="k"&gt;vim&lt;/span&gt;
&lt;span class="k"&gt;colorscheme&lt;/span&gt; PaperColor
&lt;/pre&gt;&lt;/div&gt;

&lt;p&gt;Es un archivo pequeño, pero que sirve de punto de partida para moverse a una configuración en Lua.&lt;/p&gt;
&lt;h4 id="crear-el-entorno-para-neovim"&gt;Crear el entorno para Neovim&lt;/h4&gt;
&lt;p&gt;&lt;img alt="El sistema modular de Neovim" class="img-izquierda-responsiva" src="https://sergiobelkin.com/images/ElsistemamodulardeNeovim.webp"&gt;&lt;/p&gt;
&lt;p&gt;Como vimos en el post anterior, mi recomendación es comenzar con un archivo &lt;code&gt;init.vim&lt;/code&gt;. Una vez que ya estamos usando Neovim podemos realizar la migración a &lt;code&gt;init.lua&lt;/code&gt;. Sin embargo, tenemos que crear los directorios para que Neovim guarde su configuración. Lo único que dejaremos por ahora en Vim script es la configuración de plugins con &lt;strong&gt;vim-plug&lt;/strong&gt;. Por lo tanto ejecutamos:&lt;/p&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;mkdir&lt;span class="w"&gt; &lt;/span&gt;-p&lt;span class="w"&gt; &lt;/span&gt;~/.config/nvim/lua/utils
mkdir&lt;span class="w"&gt; &lt;/span&gt;-p&lt;span class="w"&gt; &lt;/span&gt;~/.local/share/nvim/site/autoload
&lt;span class="c1"&gt;# Suponiendo que usabas un archivo para gestionar plugins en Vim&lt;/span&gt;
mv&lt;span class="w"&gt; &lt;/span&gt;~/.vim/autoload/plug.vim&lt;span class="w"&gt; &lt;/span&gt;~/.local/share/nvim/site/autoload
&lt;/pre&gt;&lt;/div&gt;

&lt;ul&gt;
&lt;li&gt;El directorio &lt;code&gt;~/.config/nvim&lt;/code&gt; es obligatorio.&lt;/li&gt;
&lt;li&gt;El directorio &lt;code&gt;~/.config/nvim/lua&lt;/code&gt; &lt;em&gt;no&lt;/em&gt; es obligatorio, pero es muy útil para modularizar la configuración de Neovim.&lt;/li&gt;
&lt;li&gt;Los nombres dados de los archivos y directorios dentro de &lt;code&gt;~/.config/nvim/lua&lt;/code&gt; son totalmente arbitrarios.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;En Neovim tendremos una estructura más modular. Si bien podemos tener todo en &lt;code&gt;init.lua&lt;/code&gt;, vamos a aprovecharnos de esta configuración que es mucho más flexible.&lt;/p&gt;
&lt;p&gt;&lt;strong&gt;Archivo ~/.config/nvim/init.lua&lt;/strong&gt;&lt;/p&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;&lt;span class="c1"&gt;-- Cargamos los plugins primero — algunas configuraciones&lt;/span&gt;
&lt;span class="c1"&gt;-- (como `colorscheme`) dependen de tenerlos disponibles en el runtimepath&lt;/span&gt;
&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;cmd&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'source ~/.config/nvim/plugins.vim'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;

&lt;span class="c1"&gt;-- Cargamos nuestros módulos (el require ejecuta el archivo;&lt;/span&gt;
&lt;span class="c1"&gt;-- el efecto viene del side-effect, no del valor capturado)&lt;/span&gt;
&lt;span class="nb"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'utils.commands'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="nb"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'utils.extracommands'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="nb"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'keymaps'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="nb"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'opciones'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;/pre&gt;&lt;/div&gt;

&lt;p&gt;&lt;strong&gt;Archivo ~/.config/nvim/lua/keymaps.lua&lt;/strong&gt;&lt;/p&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;&lt;span class="c1"&gt;-- Alias&lt;/span&gt;
&lt;span class="kd"&gt;local&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;map&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;keymap&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;set&lt;/span&gt;
&lt;span class="c1"&gt;-- maps&lt;/span&gt;
&lt;span class="nf"&gt;map&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'n'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s1"&gt;'ñ'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s1"&gt;':wq&amp;lt;CR&amp;gt;'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;/pre&gt;&lt;/div&gt;

&lt;blockquote&gt;
&lt;p&gt;&lt;strong&gt;Nota sobre &lt;code&gt;vim.keymap.set&lt;/code&gt;&lt;/strong&gt;: usa &lt;code&gt;remap = false&lt;/code&gt; por default, así que &lt;code&gt;map('n', 'ñ', ':wq&amp;lt;CR&amp;gt;')&lt;/code&gt; ya es equivalente a &lt;code&gt;nnoremap&lt;/code&gt; (no a &lt;code&gt;nmap&lt;/code&gt;). El "nore" que en VimL diferenciaba &lt;code&gt;nnoremap&lt;/code&gt; de &lt;code&gt;nmap&lt;/code&gt; está implícito en el default de la API Lua.&lt;/p&gt;
&lt;/blockquote&gt;
&lt;p&gt;&lt;strong&gt;Archivo ~/.config/nvim/lua/opciones.lua&lt;/strong&gt;&lt;/p&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;&lt;span class="c1"&gt;-- Alias&lt;/span&gt;
&lt;span class="kd"&gt;local&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;vo&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;o&lt;/span&gt;
&lt;span class="kd"&gt;local&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;vop&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;opt&lt;/span&gt;

&lt;span class="c1"&gt;-- Opciones&lt;/span&gt;
&lt;span class="nv"&gt;vo&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;tabstop&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="mi"&gt;4&lt;/span&gt;
&lt;span class="nv"&gt;vop&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;nf&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="nf"&gt;append&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'alpha'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;/pre&gt;&lt;/div&gt;

&lt;p&gt;&lt;strong&gt;Archivo ~/.config/nvim/lua/utils/commands.lua&lt;/strong&gt;&lt;/p&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;&lt;span class="c1"&gt;-- Creamos una tabla vacía para guardar funciones&lt;/span&gt;

&lt;span class="kd"&gt;local&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;M&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{}&lt;/span&gt;

&lt;span class="kr"&gt;function&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nc"&gt;M&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;create_command&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nv"&gt;name&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;command_or_function&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;opts&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="w"&gt;  &lt;/span&gt;&lt;span class="nb"&gt;pcall&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;api&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;nvim_del_user_command&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;name&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="w"&gt;  &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;api&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;nvim_create_user_command&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nv"&gt;name&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;command_or_function&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;opts&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="ow"&gt;or&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{})&lt;/span&gt;
&lt;span class="kr"&gt;end&lt;/span&gt;

&lt;span class="c1"&gt;-- Alias&lt;/span&gt;
&lt;span class="kd"&gt;local&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;com&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;M&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;create_command&lt;/span&gt;

&lt;span class="nf"&gt;com&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'Silent'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kr"&gt;function&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nv"&gt;opts&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="w"&gt;  &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;cmd&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'silent !'&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;..&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nb"&gt;table.concat&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nv"&gt;opts&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;fargs&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s1"&gt;' '&lt;/span&gt;&lt;span class="p"&gt;))&lt;/span&gt;
&lt;span class="w"&gt;  &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;cmd&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'redraw!'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="kr"&gt;end&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;nargs&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s1"&gt;'+'&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;})&lt;/span&gt;

&lt;span class="nf"&gt;com&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'Nbuild'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kr"&gt;function&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;
&lt;span class="w"&gt;  &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;cmd&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'w'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="w"&gt;  &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;cmd&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'Silent nikola build'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="kr"&gt;end&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{})&lt;/span&gt;


&lt;span class="kr"&gt;return&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;M&lt;/span&gt;
&lt;/pre&gt;&lt;/div&gt;

&lt;p&gt;&lt;strong&gt;Archivo ~/.config/nvim/lua/utils/extracommands.lua&lt;/strong&gt;&lt;/p&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;&lt;span class="c1"&gt;-- Alias&lt;/span&gt;
&lt;span class="kd"&gt;local&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;vcmd&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;cmd&lt;/span&gt;

&lt;span class="c1"&gt;-- Extra commands&lt;/span&gt;
&lt;span class="nf"&gt;vcmd&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'filetype plugin on'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="c1"&gt;-- Esta configuración es innecesaria porque ya está en init.lua&lt;/span&gt;
&lt;span class="c1"&gt;-- vcmd('runtime plugins.vim')&lt;/span&gt;
&lt;span class="nf"&gt;vcmd&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'colorscheme PaperColor'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;/pre&gt;&lt;/div&gt;

&lt;p&gt;&lt;strong&gt;Archivo ~/.config/nvim/plugins.vim&lt;/strong&gt;&lt;/p&gt;
&lt;p&gt;Este es el único archivo que dejamos en Vim script (con código Lua embebido):&lt;/p&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;&lt;span class="c"&gt;"Plugins (Plug)&lt;/span&gt;
&lt;span class="k"&gt;call&lt;/span&gt; plug#begin&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'~/.vim/plugged'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;

Plug &lt;span class="s1"&gt;'itchyny/lightline.vim'&lt;/span&gt;
Plug &lt;span class="s1"&gt;'nyngwang/NeoTerm.lua'&lt;/span&gt;
Plug &lt;span class="s1"&gt;'NLKNguyen/papercolor-theme'&lt;/span&gt;

&lt;span class="k"&gt;call&lt;/span&gt; plug#&lt;span class="k"&gt;end&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;

&lt;span class="k"&gt;lua&lt;/span&gt; &lt;span class="p"&gt;&amp;lt;&amp;lt;&lt;/span&gt; EOF                                                       
require&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s2"&gt;"neo-term"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;.setup {
  exclude_filetypes &lt;span class="p"&gt;=&lt;/span&gt; { &lt;span class="s2"&gt;"oil"&lt;/span&gt; }
}                          
EOF

&lt;span class="nb"&gt;nnoremap&lt;/span&gt; &lt;span class="p"&gt;&amp;lt;&lt;/span&gt;F4&lt;span class="p"&gt;&amp;gt;&lt;/span&gt; :NeoTermToggle&lt;span class="p"&gt;&amp;lt;&lt;/span&gt;CR&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;
tnoremap &lt;span class="p"&gt;&amp;lt;&lt;/span&gt;F4&lt;span class="p"&gt;&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;&amp;lt;&lt;/span&gt;C&lt;span class="p"&gt;-&lt;/span&gt;\&lt;span class="p"&gt;&amp;gt;&amp;lt;&lt;/span&gt;C&lt;span class="p"&gt;-&lt;/span&gt;&lt;span class="k"&gt;n&lt;/span&gt;&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;:NeoTermToggle&lt;span class="p"&gt;&amp;lt;&lt;/span&gt;CR&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;
tnoremap &lt;span class="p"&gt;&amp;lt;&lt;/span&gt;Esc&lt;span class="p"&gt;&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;&amp;lt;&lt;/span&gt;C&lt;span class="p"&gt;-&lt;/span&gt;\&lt;span class="p"&gt;&amp;gt;&amp;lt;&lt;/span&gt;C&lt;span class="p"&gt;-&lt;/span&gt;&lt;span class="k"&gt;n&lt;/span&gt;&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;:NeoTermEnterNormal&lt;span class="p"&gt;&amp;lt;&lt;/span&gt;CR&lt;span class="p"&gt;&amp;gt;&lt;/span&gt;
&lt;/pre&gt;&lt;/div&gt;

&lt;p&gt;Y listo...&lt;/p&gt;
&lt;h4 id="explicacion"&gt;Explicación&lt;/h4&gt;
&lt;p&gt;&lt;strong&gt;¿Por qué definimos alias en los archivos de módulos?&lt;/strong&gt;&lt;/p&gt;
&lt;p&gt;Los alias son básicamente variables. Los definimos porque este es un ejemplo de tipo "esqueleto" de configuración. Cuando comiences a agregar más opciones, atajos de teclado y comandos personalizados, no querrás tipear tanto 😉.&lt;/p&gt;
&lt;p&gt;&lt;strong&gt;¿Para qué agregamos la función &lt;code&gt;create_command&lt;/code&gt;?&lt;/strong&gt;&lt;/p&gt;
&lt;p&gt;En Lua cuando usás la API de Neovim, si querés redefinir una función de un comando personalizado, tenés que borrarlo y volver a definirlo, si no da error. Esto es para evitar sobrescrituras accidentales. Por eso creamos la función &lt;code&gt;create_command&lt;/code&gt; para que cuando redefinamos un comando lo hagamos de manera explícita.&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;Usa &lt;code&gt;pcall&lt;/code&gt; para borrar el comando personalizado y si no existe no da error. &lt;/li&gt;
&lt;li&gt;Vuelve a recrearlo con la nueva definición.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;El ejemplo de &lt;code&gt;Nbuild&lt;/code&gt; deja ver diferencias más grandes entre VimL y Lua. Las analizamos en cinco dimensiones:&lt;/p&gt;
&lt;h5 id="1-brevedad-vs-mantenibilidad"&gt;1. &lt;strong&gt;Brevedad vs mantenibilidad&lt;/strong&gt;&lt;/h5&gt;
&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;VimL:&lt;/strong&gt;&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;&lt;code&gt;vim
  command -nargs=+ Silent execute 'silent !&amp;lt;args&amp;gt;' | redraw!
  command Nbuild execute 'w | Silent nikola build'&lt;/code&gt;&lt;/p&gt;
&lt;p&gt;Es breve, directo y funciona. Pero:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;Es poco &lt;strong&gt;expresivo&lt;/strong&gt;&lt;/li&gt;
&lt;li&gt;No permite &lt;strong&gt;manejar errores&lt;/strong&gt; fácilmente&lt;/li&gt;
&lt;li&gt;
&lt;p&gt;No escala bien con lógica compleja&lt;/p&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;p&gt;&lt;strong&gt;Lua:&lt;/strong&gt;&lt;/p&gt;
&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;&lt;code&gt;lua
  com('Silent', function(opts)
    vim.cmd('silent !' .. table.concat(opts.fargs, ' '))
    vim.cmd('redraw!')
  end, { nargs = '+' })&lt;/code&gt;&lt;/p&gt;
&lt;p&gt;Es más verboso, sí, pero:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;La función puede ser más compleja si se necesita&lt;/li&gt;
&lt;li&gt;Es &lt;strong&gt;programable&lt;/strong&gt; y &lt;strong&gt;reutilizable&lt;/strong&gt;&lt;/li&gt;
&lt;li&gt;Puede integrarse con condicionales, bucles, o lógica externa&lt;/li&gt;
&lt;/ul&gt;
&lt;h5 id="2-control-de-errores-y-depuracion"&gt;2. &lt;strong&gt;Control de errores y depuración&lt;/strong&gt;&lt;/h5&gt;
&lt;ul&gt;
&lt;li&gt;En VimL: si un comando falla, tenés poco margen para capturar y manejar el error.&lt;/li&gt;
&lt;li&gt;En Lua: podés usar &lt;code&gt;pcall&lt;/code&gt;, &lt;code&gt;vim.notify&lt;/code&gt;, &lt;code&gt;vim.fn.systemlist&lt;/code&gt;, y &lt;strong&gt;construir mensajes personalizados o fallback&lt;/strong&gt; si algo sale mal.&lt;/li&gt;
&lt;/ul&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;&lt;span class="kd"&gt;local&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;ok&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;_&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nb"&gt;pcall&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="kr"&gt;function&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;
&lt;span class="w"&gt;  &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;cmd&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'silent !nikola build'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="kr"&gt;end&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="kr"&gt;if&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="ow"&gt;not&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;ok&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kr"&gt;then&lt;/span&gt;
&lt;span class="w"&gt;  &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;notify&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'Error al ejecutar nikola build'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;vim&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;log&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;levels&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="py"&gt;ERROR&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="kr"&gt;end&lt;/span&gt;
&lt;/pre&gt;&lt;/div&gt;

&lt;h5 id="3-modularizacion"&gt;3. &lt;strong&gt;Modularización&lt;/strong&gt;&lt;/h5&gt;
&lt;ul&gt;
&lt;li&gt;En Lua podés agrupar comandos en módulos, con &lt;code&gt;require(...)&lt;/code&gt;.&lt;/li&gt;
&lt;li&gt;En VimL tenés que recurrir a &lt;code&gt;runtime&lt;/code&gt;, &lt;code&gt;source&lt;/code&gt;, y no hay namespaces claros.&lt;/li&gt;
&lt;/ul&gt;
&lt;div class="code"&gt;&lt;pre class="code literal-block"&gt;&lt;span class="kd"&gt;local&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nv"&gt;commands&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="nb"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'utils.commands'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="nv"&gt;commands&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;create_command&lt;/span&gt;&lt;span class="p"&gt;(...)&lt;/span&gt;
&lt;/pre&gt;&lt;/div&gt;

&lt;p&gt;Esto &lt;strong&gt;facilita dividir la configuración en archivos lógicos&lt;/strong&gt;, más fácil de leer, mantener y compartir.&lt;/p&gt;
&lt;h5 id="4-compatibilidad-futura-y-ecosistema-neovim"&gt;4. &lt;strong&gt;Compatibilidad futura y ecosistema Neovim&lt;/strong&gt;&lt;/h5&gt;
&lt;ul&gt;
&lt;li&gt;Neovim ha declarado que Lua es su &lt;strong&gt;lenguaje de configuración nativo moderno&lt;/strong&gt;.&lt;/li&gt;
&lt;li&gt;La mayoría de los &lt;strong&gt;plugins modernos están escritos en Lua&lt;/strong&gt;.&lt;/li&gt;
&lt;li&gt;VimL sigue siendo soportado, pero está &lt;strong&gt;siendo superado funcionalmente&lt;/strong&gt;.&lt;/li&gt;
&lt;/ul&gt;
&lt;h5 id="5-legibilidad-a-largo-plazo"&gt;5. &lt;strong&gt;Legibilidad a largo plazo&lt;/strong&gt;&lt;/h5&gt;
&lt;ul&gt;
&lt;li&gt;
&lt;p&gt;Aunque VimL sea más corto, en un entorno de colaboración o con configuraciones grandes, Lua:&lt;/p&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;p&gt;Es más &lt;strong&gt;explícito&lt;/strong&gt;.&lt;/p&gt;
&lt;/li&gt;
&lt;li&gt;Permite &lt;strong&gt;documentar&lt;/strong&gt;.&lt;/li&gt;
&lt;li&gt;La API de Neovim en Lua permite &lt;strong&gt;un uso más explícito y predecible de tipos de datos&lt;/strong&gt;.&lt;/li&gt;
&lt;li&gt;Es más &lt;strong&gt;fácil de extender&lt;/strong&gt; con lógica externa.&lt;/li&gt;
&lt;/ul&gt;
&lt;h3 id="conclusion"&gt;Conclusión&lt;/h3&gt;
&lt;p&gt;Si sabés migrar opciones, mappings, comandos y cargar plugins → podés convertir el 80% o más de tu &lt;code&gt;init.vim&lt;/code&gt; (equivalente al viejo &lt;code&gt;.vimrc&lt;/code&gt;) a &lt;code&gt;init.lua&lt;/code&gt;.&lt;/p&gt;
&lt;p&gt;Migrar de Vim script a Lua no es una mejora cosmética: cambia las posibilidades de la configuración. Lo que en Vim script son comandos breves, en Lua son funciones programables — más verbosas, sí, pero abiertas a control de errores con &lt;code&gt;pcall&lt;/code&gt;, modularización con &lt;code&gt;require&lt;/code&gt; y un ecosistema de plugins que ya asume Lua como lenguaje nativo.&lt;/p&gt;
&lt;p&gt;La estructura mostrada acá es una entre muchas. El esqueleto &lt;code&gt;init.lua&lt;/code&gt; + módulos en &lt;code&gt;lua/&lt;/code&gt; se sostiene porque escala: cada archivo crece sin contaminar al resto, y la configuración deja de ser una pared de líneas para volverse un conjunto de piezas que podés rearmar cuando lo necesites.&lt;/p&gt;
&lt;p&gt;Lo que no migramos en este post es la configuración de plugins, que dejamos en &lt;code&gt;plugins.vim&lt;/code&gt;. Reemplazar &lt;strong&gt;vim-plug&lt;/strong&gt; por un gestor en Lua como &lt;strong&gt;lazy.nvim&lt;/strong&gt; es el siguiente paso natural — y tema de &lt;a href="https://sergiobelkin.com/posts/como-migrar-de-vim-plug-a-lazynvim/"&gt;este otro post&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;Al cierre del post dejé un resumen de 10 conceptos clave para tener a mano cuando empieces a extender tu config.&lt;/p&gt;
&lt;h3 id="apendice-los-10-conceptos-fundamentales-de-lua-resumen"&gt;Apéndice: Los 10 conceptos fundamentales de Lua (resumen)&lt;/h3&gt;
&lt;p&gt;Resumen de los 10 conceptos clave de Lua en Neovim. Los 3 marcados como &lt;em&gt;(esencial)&lt;/em&gt; son los que vimos al principio; el resto extiende esa base:&lt;/p&gt;
&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Concepto&lt;/th&gt;
&lt;th&gt;Descripción breve&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;&lt;strong&gt;Tablas (&lt;code&gt;table&lt;/code&gt;)&lt;/strong&gt; &lt;em&gt;(esencial)&lt;/em&gt;&lt;/td&gt;
&lt;td&gt;Estructuras clave-valor o listas. Base de toda la configuración.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;strong&gt;Variables locales (&lt;code&gt;local&lt;/code&gt;)&lt;/strong&gt; &lt;em&gt;(esencial)&lt;/em&gt;&lt;/td&gt;
&lt;td&gt;Para mantener el código limpio y evitar contaminación global.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;strong&gt;Funciones anónimas&lt;/strong&gt;&lt;/td&gt;
&lt;td&gt;Permiten definir acciones sin nombre, usadas en mappings, autocmds y comandos.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;strong&gt;API de Neovim (&lt;code&gt;vim.api.*&lt;/code&gt;)&lt;/strong&gt; &lt;em&gt;(esencial)&lt;/em&gt;&lt;/td&gt;
&lt;td&gt;Puente entre Lua y Neovim: crear comandos, mappings, autocmds.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;vim.opt&lt;/code&gt; vs &lt;code&gt;vim.o&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;&lt;code&gt;vim.o&lt;/code&gt; da acceso directo al valor; &lt;code&gt;vim.opt&lt;/code&gt; lo envuelve en un objeto type-aware con métodos (necesario para listas).&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;require&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Para cargar módulos externos y dividir la configuración.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;strong&gt;Paréntesis en llamadas&lt;/strong&gt;&lt;/td&gt;
&lt;td&gt;Obligatorio usar &lt;code&gt;()&lt;/code&gt; al llamar funciones.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;strong&gt;Referencias&lt;/strong&gt;&lt;/td&gt;
&lt;td&gt;Asignar una tabla a otra variable sin copiarla (ejemplo: &lt;code&gt;local vg = vim.g&lt;/code&gt;).&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;_G (global)&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Tabla especial que contiene las variables y funciones globales accesibles en toda la configuración.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;package.loaded&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Permite ver o manipular qué módulos están cargados, útil para recargar configuraciones dinámicamente.&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;
&lt;h3 id="fuentes-y-mas-recursos"&gt;Fuentes y más Recursos&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;a href="https://neovim.io/doc/user/lua.html"&gt;Lua en Neovim (&lt;code&gt;:help lua&lt;/code&gt;)&lt;/a&gt; — referencia completa de la API Lua de Neovim.&lt;/li&gt;
&lt;li&gt;&lt;a href="https://neovim.io/doc/user/lua-guide.html"&gt;Guía de Lua para Neovim (&lt;code&gt;:help lua-guide&lt;/code&gt;)&lt;/a&gt; — survival kit oficial para configurar Neovim en Lua.&lt;/li&gt;
&lt;li&gt;&lt;a href="https://www.lua.org/manual/5.1/"&gt;Manual de referencia de Lua 5.1&lt;/a&gt; — definición oficial del lenguaje (LuaJIT, que usa Neovim, está basado en esta versión).&lt;/li&gt;
&lt;li&gt;&lt;a href="https://neovim.io/doc/lua-resources/"&gt;Recursos de Lua recomendados por Neovim&lt;/a&gt; — curaduría oficial de tutoriales y libros.&lt;/li&gt;
&lt;li&gt;&lt;a href="https://dev.to/tolomeo/a-pragmatic-approach-to-migrating-from-vscode-to-neovim-4fae"&gt;A pragmatic approach to migrating from VSCode to Neovim&lt;/a&gt; — otra mirada sobre migración hacia Neovim+Lua, desde VSCode.&lt;/li&gt;
&lt;li&gt;&lt;a href="https://www.kushcreates.com/blogs/neovim-is-better-but-why-developers-arent-switching-to-it"&gt;NeoVim Is Better, But Why Developers Aren't Switching To It?&lt;/a&gt; — perspectiva crítica sobre la adopción de Neovim.&lt;/li&gt;
&lt;/ul&gt;</description><category>Lua</category><category>Neovim</category><category>vim</category><guid>https://sergiobelkin.com/posts/como-convertir-tu-configuracion-de-initvim-a-initlua/</guid><pubDate>Thu, 15 May 2025 19:01:18 GMT</pubDate></item></channel></rss>