@@ -38,6 +38,7 @@ M.dirty = false
38
38
--- @field name string display name and name used for plugin config files
39
39
--- @field uri string
40
40
--- @field dir string
41
+ --- @field dep ? boolean True if this plugin is only in the spec as a dependency
41
42
--- @field enabled ? boolean | (fun (): boolean )
42
43
--- @field opt ? boolean
43
44
--- @field dependencies ? string[]
@@ -82,7 +83,8 @@ function Spec.load(modname, modpath)
82
83
end
83
84
84
85
--- @param plugin LazyPlugin
85
- function Spec :add (plugin )
86
+ --- @param is_dep ? boolean
87
+ function Spec :add (plugin , is_dep )
86
88
local pkg = plugin [1 ]
87
89
if type (pkg ) ~= " string" then
88
90
Util .error (" Invalid plugin spec " .. vim .inspect (plugin ))
@@ -108,27 +110,30 @@ function Spec:add(plugin)
108
110
plugin .name = slash and name :sub (# name - slash + 2 ) or pkg :gsub (" %W+" , " _" )
109
111
end
110
112
113
+ plugin .dep = is_dep
114
+
111
115
M .process_local (plugin )
112
116
local other = self .plugins [plugin .name ]
113
- self .plugins [plugin .name ] = other and vim . tbl_extend ( " force " , self . plugins [ plugin . name ] , plugin ) or plugin
117
+ self .plugins [plugin .name ] = other and M . merge ( other , plugin ) or plugin
114
118
return self .plugins [plugin .name ]
115
119
end
116
120
117
121
--- @param spec LazySpec
118
122
--- @param results ? string[]
119
- function Spec :normalize (spec , results )
123
+ --- @param is_dep ? boolean
124
+ function Spec :normalize (spec , results , is_dep )
120
125
results = results or {}
121
126
if type (spec ) == " string" then
122
- table.insert (results , self :add ({ spec }).name )
127
+ table.insert (results , self :add ({ spec }, is_dep ).name )
123
128
elseif # spec > 1 or Util .is_list (spec ) then
124
129
--- @cast spec LazySpec[]
125
130
for _ , s in ipairs (spec ) do
126
- self :normalize (s , results )
131
+ self :normalize (s , results , is_dep )
127
132
end
128
133
elseif spec .enabled == nil or spec .enabled == true or (type (spec .enabled ) == " function" and spec .enabled ()) then
129
134
--- @cast spec LazyPlugin
130
- local plugin = self :add (spec )
131
- plugin .dependencies = plugin .dependencies and self :normalize (plugin .dependencies , {}) or nil
135
+ local plugin = self :add (spec , is_dep )
136
+ plugin .dependencies = plugin .dependencies and self :normalize (plugin .dependencies , {}, true ) or nil
132
137
table.insert (results , plugin .name )
133
138
end
134
139
return results
@@ -152,22 +157,55 @@ function Spec.revive(spec)
152
157
return spec
153
158
end
154
159
155
- function M .update_state (check_clean )
160
+ --- @param old LazyPlugin
161
+ --- @param new LazyPlugin
162
+ --- @return LazyPlugin
163
+ function M .merge (old , new )
164
+ local is_dep = old .dep and new .dep
165
+
166
+ local Handler = require (" lazy.core.handler" )
167
+ --- @diagnostic disable-next-line : no-unknown
168
+ for k , v in pairs (new ) do
169
+ if k == " dep" then
170
+ elseif old [k ] ~= nil and old [k ] ~= v then
171
+ if Handler .handlers [k ] then
172
+ local values = type (v ) == " string" and { v } or v
173
+ vim .list_extend (values , type (old [k ]) == " string" and { old [k ] } or old [k ])
174
+ --- @diagnostic disable-next-line : no-unknown
175
+ old [k ] = values
176
+ else
177
+ error (" Merging plugins is not supported for key `" .. k .. " `" )
178
+ end
179
+ else
180
+ --- @diagnostic disable-next-line : no-unknown
181
+ old [k ] = v
182
+ end
183
+ end
184
+ old .dep = is_dep
185
+ return old
186
+ end
187
+
188
+ --- @param opts ? { clean : boolean , installed : boolean , plugins ?: LazyPlugin[] }
189
+ function M .update_state (opts )
190
+ opts = opts or {}
191
+
156
192
--- @type table< " opt" | " start" , table<string,FileType>>
157
193
local installed = { opt = {}, start = {} }
158
- for opt , packs in pairs (installed ) do
159
- Util .ls (Config .options .packpath .. " /" .. opt , function (_ , name , type )
160
- if type == " directory" or type == " link" then
161
- packs [name ] = type
162
- end
163
- end )
194
+ if opts .installed ~= false then
195
+ for opt , packs in pairs (installed ) do
196
+ Util .ls (Config .options .packpath .. " /" .. opt , function (_ , name , type )
197
+ if type == " directory" or type == " link" then
198
+ packs [name ] = type
199
+ end
200
+ end )
201
+ end
164
202
end
165
203
166
- for _ , plugin in pairs (Config .plugins ) do
204
+ for _ , plugin in pairs (opts . plugins or Config .plugins ) do
167
205
plugin ._ = plugin ._ or {}
168
206
plugin [1 ] = plugin [" 1" ] or plugin [1 ]
169
207
if plugin .opt == nil then
170
- plugin .opt = Config .options .opt
208
+ plugin .opt = plugin . dep or Config .options .opt
171
209
end
172
210
local opt = plugin .opt and " opt" or " start"
173
211
plugin .dir = Config .options .packpath .. " /" .. opt .. " /" .. plugin .name
@@ -179,7 +217,7 @@ function M.update_state(check_clean)
179
217
end
180
218
end
181
219
182
- if check_clean then
220
+ if opts . clean then
183
221
Config .to_clean = {}
184
222
for opt , packs in pairs (installed ) do
185
223
for pack in pairs (packs ) do
@@ -244,7 +282,7 @@ function M.load()
244
282
for _ , spec in ipairs (specs ) do
245
283
for _ , plugin in pairs (spec .plugins ) do
246
284
local other = Config .plugins [plugin .name ]
247
- Config .plugins [plugin .name ] = other and vim . tbl_extend ( " force " , other , plugin ) or plugin
285
+ Config .plugins [plugin .name ] = other and M . merge ( other , plugin ) or plugin
248
286
end
249
287
end
250
288
0 commit comments