view_component 2.83.0 → 3.21.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (49) hide show
  1. checksums.yaml +4 -4
  2. data/app/controllers/concerns/view_component/preview_actions.rb +5 -1
  3. data/app/controllers/view_components_system_test_controller.rb +24 -1
  4. data/app/helpers/preview_helper.rb +22 -4
  5. data/app/views/view_components/_preview_source.html.erb +2 -2
  6. data/docs/CHANGELOG.md +807 -1
  7. data/lib/rails/generators/abstract_generator.rb +9 -1
  8. data/lib/rails/generators/component/component_generator.rb +2 -1
  9. data/lib/rails/generators/component/templates/component.rb.tt +3 -2
  10. data/lib/rails/generators/erb/component_generator.rb +1 -1
  11. data/lib/rails/generators/locale/component_generator.rb +3 -3
  12. data/lib/rails/generators/preview/templates/component_preview.rb.tt +2 -0
  13. data/lib/rails/generators/rspec/component_generator.rb +15 -3
  14. data/lib/rails/generators/rspec/templates/component_spec.rb.tt +1 -1
  15. data/lib/rails/generators/stimulus/component_generator.rb +8 -3
  16. data/lib/rails/generators/stimulus/templates/component_controller.ts.tt +9 -0
  17. data/lib/rails/generators/test_unit/templates/component_test.rb.tt +1 -1
  18. data/lib/view_component/base.rb +169 -164
  19. data/lib/view_component/capture_compatibility.rb +44 -0
  20. data/lib/view_component/collection.rb +20 -8
  21. data/lib/view_component/compiler.rb +166 -207
  22. data/lib/view_component/config.rb +63 -14
  23. data/lib/view_component/deprecation.rb +1 -1
  24. data/lib/view_component/docs_builder_component.html.erb +5 -1
  25. data/lib/view_component/docs_builder_component.rb +28 -9
  26. data/lib/view_component/engine.rb +58 -28
  27. data/lib/view_component/errors.rb +240 -0
  28. data/lib/view_component/inline_template.rb +55 -0
  29. data/lib/view_component/instrumentation.rb +10 -2
  30. data/lib/view_component/preview.rb +7 -8
  31. data/lib/view_component/rails/tasks/view_component.rake +11 -2
  32. data/lib/view_component/slot.rb +119 -1
  33. data/lib/view_component/slotable.rb +394 -94
  34. data/lib/view_component/slotable_default.rb +20 -0
  35. data/lib/view_component/system_test_helpers.rb +5 -5
  36. data/lib/view_component/template.rb +134 -0
  37. data/lib/view_component/test_helpers.rb +138 -59
  38. data/lib/view_component/translatable.rb +45 -26
  39. data/lib/view_component/use_helpers.rb +42 -0
  40. data/lib/view_component/version.rb +4 -3
  41. data/lib/view_component/with_content_helper.rb +3 -8
  42. data/lib/view_component.rb +3 -12
  43. metadata +277 -38
  44. data/lib/view_component/content_areas.rb +0 -56
  45. data/lib/view_component/polymorphic_slots.rb +0 -103
  46. data/lib/view_component/preview_template_error.rb +0 -6
  47. data/lib/view_component/slot_v2.rb +0 -98
  48. data/lib/view_component/slotable_v2.rb +0 -391
  49. data/lib/view_component/template_error.rb +0 -9
@@ -1,7 +1,125 @@
1
1
  # frozen_string_literal: true
2
2
 
3
+ require "view_component/with_content_helper"
4
+
3
5
  module ViewComponent
4
6
  class Slot
5
- attr_accessor :content
7
+ include ViewComponent::WithContentHelper
8
+
9
+ attr_writer :__vc_component_instance, :__vc_content_block, :__vc_content
10
+
11
+ def initialize(parent)
12
+ @parent = parent
13
+ end
14
+
15
+ def content?
16
+ return true if defined?(@__vc_content) && @__vc_content.present?
17
+ return true if defined?(@__vc_content_set_by_with_content) && @__vc_content_set_by_with_content.present?
18
+ return true if defined?(@__vc_content_block) && @__vc_content_block.present?
19
+ return false if !__vc_component_instance?
20
+
21
+ @__vc_component_instance.content?
22
+ end
23
+
24
+ def with_content(args)
25
+ if __vc_component_instance?
26
+ @__vc_component_instance.with_content(args)
27
+ else
28
+ super
29
+ end
30
+ end
31
+
32
+ # Used to render the slot content in the template
33
+ #
34
+ # There's currently 3 different values that may be set, that we can render.
35
+ #
36
+ # If the slot renderable is a component, the string class name of a
37
+ # component, or a function that returns a component, we render that
38
+ # component instance, returning the string.
39
+ #
40
+ # If the slot renderable is a function and returns a string, it's
41
+ # set as `@__vc_content` and is returned directly.
42
+ #
43
+ # If there is no slot renderable, we evaluate the block passed to
44
+ # the slot and return it.
45
+ def to_s
46
+ return @content if defined?(@content)
47
+
48
+ view_context = @parent.send(:view_context)
49
+
50
+ if defined?(@__vc_content_block) && defined?(@__vc_content_set_by_with_content)
51
+ raise DuplicateSlotContentError.new(self.class.name)
52
+ end
53
+
54
+ @content =
55
+ if __vc_component_instance?
56
+ @__vc_component_instance.__vc_original_view_context = @parent.__vc_original_view_context
57
+
58
+ if defined?(@__vc_content_block)
59
+ # render_in is faster than `parent.render`
60
+ @__vc_component_instance.render_in(view_context) do |*args|
61
+ return @__vc_content_block.call(*args) if @__vc_content_block&.source_location.nil?
62
+
63
+ block_context = @__vc_content_block.binding.receiver
64
+
65
+ if block_context.class < ActionView::Base
66
+ block_context.capture(*args, &@__vc_content_block)
67
+ else
68
+ @__vc_content_block.call(*args)
69
+ end
70
+ end
71
+ else
72
+ @__vc_component_instance.render_in(view_context)
73
+ end
74
+ elsif defined?(@__vc_content)
75
+ @__vc_content
76
+ elsif defined?(@__vc_content_block)
77
+ view_context.capture(&@__vc_content_block)
78
+ elsif defined?(@__vc_content_set_by_with_content)
79
+ @__vc_content_set_by_with_content
80
+ end
81
+
82
+ @content = @content.to_s
83
+ end
84
+
85
+ # Allow access to public component methods via the wrapper
86
+ #
87
+ # for example
88
+ #
89
+ # calling `header.name` (where `header` is a slot) will call `name`
90
+ # on the `HeaderComponent` instance.
91
+ #
92
+ # Where the component may look like:
93
+ #
94
+ # class MyComponent < ViewComponent::Base
95
+ # has_one :header, HeaderComponent
96
+ #
97
+ # class HeaderComponent < ViewComponent::Base
98
+ # def name
99
+ # @name
100
+ # end
101
+ # end
102
+ # end
103
+ #
104
+ def method_missing(symbol, *args, &block)
105
+ @__vc_component_instance.public_send(symbol, *args, &block)
106
+ end
107
+ ruby2_keywords(:method_missing) if respond_to?(:ruby2_keywords, true)
108
+
109
+ def html_safe?
110
+ # :nocov:
111
+ to_s.html_safe?
112
+ # :nocov:
113
+ end
114
+
115
+ def respond_to_missing?(symbol, include_all = false)
116
+ __vc_component_instance? && @__vc_component_instance.respond_to?(symbol, include_all)
117
+ end
118
+
119
+ private
120
+
121
+ def __vc_component_instance?
122
+ defined?(@__vc_component_instance)
123
+ end
6
124
  end
7
125
  end
@@ -1,145 +1,445 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  require "active_support/concern"
4
-
4
+ require "active_support/inflector/inflections"
5
5
  require "view_component/slot"
6
6
 
7
7
  module ViewComponent
8
8
  module Slotable
9
9
  extend ActiveSupport::Concern
10
10
 
11
+ RESERVED_NAMES = {
12
+ singular: %i[content render].freeze,
13
+ plural: %i[contents renders].freeze
14
+ }.freeze
15
+
16
+ # Setup component slot state
11
17
  included do
12
18
  # Hash of registered Slots
13
- class_attribute :slots
14
- self.slots = {}
19
+ class_attribute :registered_slots
20
+ self.registered_slots = {}
15
21
  end
16
22
 
17
23
  class_methods do
18
- # support initializing slots as:
19
- #
20
- # with_slot(
21
- # :header,
22
- # collection: true|false,
23
- # class_name: "Header" # class name string, used to instantiate Slot
24
- # )
25
- def with_slot(*slot_names, collection: false, class_name: nil)
26
- ViewComponent::Deprecation.deprecation_warning(
27
- "`with_slot`", "use the new slots API (https://viewcomponent.org/guide/slots.html) instead"
28
- )
29
-
30
- slot_names.each do |slot_name|
31
- # Ensure slot_name isn't already declared
32
- if slots.key?(slot_name)
33
- raise ArgumentError.new("#{slot_name} slot declared multiple times")
24
+ ##
25
+ # Registers a sub-component
26
+ #
27
+ # = Example
28
+ #
29
+ # renders_one :header -> (classes:) do
30
+ # HeaderComponent.new(classes: classes)
31
+ # end
32
+ #
33
+ # # OR
34
+ #
35
+ # renders_one :header, HeaderComponent
36
+ #
37
+ # where `HeaderComponent` is defined as:
38
+ #
39
+ # class HeaderComponent < ViewComponent::Base
40
+ # def initialize(classes:)
41
+ # @classes = classes
42
+ # end
43
+ # end
44
+ #
45
+ # and has the following template:
46
+ #
47
+ # <header class="<%= @classes %>">
48
+ # <%= content %>
49
+ # </header>
50
+ #
51
+ # = Rendering sub-component content
52
+ #
53
+ # The component's sidecar template can access the sub-component by calling a
54
+ # helper method with the same name as the sub-component.
55
+ #
56
+ # <h1>
57
+ # <%= header do %>
58
+ # My header title
59
+ # <% end %>
60
+ # </h1>
61
+ #
62
+ # = Setting sub-component content
63
+ #
64
+ # Consumers of the component can render a sub-component by calling a
65
+ # helper method with the same name as the slot prefixed with `with_`.
66
+ #
67
+ # <%= render_inline(MyComponent.new) do |component| %>
68
+ # <% component.with_header(classes: "Foo") do %>
69
+ # <p>Bar</p>
70
+ # <% end %>
71
+ # <% end %>
72
+ #
73
+ # Additionally, content can be set by calling `with_SLOT_NAME_content`
74
+ # on the component instance.
75
+ #
76
+ # <%= render_inline(MyComponent.new.with_header_content("Foo")) %>
77
+ def renders_one(slot_name, callable = nil)
78
+ validate_singular_slot_name(slot_name)
79
+
80
+ if callable.is_a?(Hash) && callable.key?(:types)
81
+ register_polymorphic_slot(slot_name, callable[:types], collection: false)
82
+ else
83
+ validate_plural_slot_name(ActiveSupport::Inflector.pluralize(slot_name).to_sym)
84
+
85
+ setter_method_name = :"with_#{slot_name}"
86
+
87
+ define_method setter_method_name do |*args, &block|
88
+ set_slot(slot_name, nil, *args, &block)
34
89
  end
90
+ ruby2_keywords(setter_method_name) if respond_to?(:ruby2_keywords, true)
35
91
 
36
- # Ensure slot name isn't :content
37
- if slot_name == :content
38
- raise ArgumentError.new ":content is a reserved slot name. Please use another name, such as ':body'"
92
+ self::GeneratedSlotMethods.define_method slot_name do
93
+ get_slot(slot_name)
39
94
  end
40
95
 
41
- # Set the name of the method used to access the Slot(s)
42
- accessor_name =
43
- if collection
44
- # If Slot is a collection, set the accessor
45
- # name to the pluralized form of the slot name
46
- # For example: :tab => :tabs
47
- ActiveSupport::Inflector.pluralize(slot_name)
48
- else
49
- slot_name
50
- end
96
+ self::GeneratedSlotMethods.define_method :"#{slot_name}?" do
97
+ get_slot(slot_name).present?
98
+ end
51
99
 
52
- instance_variable_name = "@#{accessor_name}"
100
+ define_method :"with_#{slot_name}_content" do |content|
101
+ send(setter_method_name) { content.to_s }
53
102
 
54
- # If the slot is a collection, define an accesor that defaults to an empty array
55
- if collection
56
- class_eval <<-RUBY, __FILE__, __LINE__ + 1
57
- def #{accessor_name}
58
- content unless content_evaluated? # ensure content is loaded so slots will be defined
59
- #{instance_variable_name} ||= []
60
- end
61
- RUBY
62
- else
63
- class_eval <<-RUBY, __FILE__, __LINE__ + 1
64
- def #{accessor_name}
65
- content unless content_evaluated? # ensure content is loaded so slots will be defined
66
- #{instance_variable_name} if defined?(#{instance_variable_name})
103
+ self
104
+ end
105
+
106
+ register_slot(slot_name, collection: false, callable: callable)
107
+ end
108
+ end
109
+
110
+ ##
111
+ # Registers a collection sub-component
112
+ #
113
+ # = Example
114
+ #
115
+ # renders_many :items, -> (name:) { ItemComponent.new(name: name }
116
+ #
117
+ # # OR
118
+ #
119
+ # renders_many :items, ItemComponent
120
+ #
121
+ # = Rendering sub-components
122
+ #
123
+ # The component's sidecar template can access the slot by calling a
124
+ # helper method with the same name as the slot.
125
+ #
126
+ # <h1>
127
+ # <% items.each do |item| %>
128
+ # <%= item %>
129
+ # <% end %>
130
+ # </h1>
131
+ #
132
+ # = Setting sub-component content
133
+ #
134
+ # Consumers of the component can set the content of a slot by calling a
135
+ # helper method with the same name as the slot prefixed with `with_`. The
136
+ # method can be called multiple times to append to the slot.
137
+ #
138
+ # <%= render_inline(MyComponent.new) do |component| %>
139
+ # <% component.with_item(name: "Foo") do %>
140
+ # <p>One</p>
141
+ # <% end %>
142
+ #
143
+ # <% component.with_item(name: "Bar") do %>
144
+ # <p>two</p>
145
+ # <% end %>
146
+ # <% end %>
147
+ def renders_many(slot_name, callable = nil)
148
+ validate_plural_slot_name(slot_name)
149
+
150
+ if callable.is_a?(Hash) && callable.key?(:types)
151
+ register_polymorphic_slot(slot_name, callable[:types], collection: true)
152
+ else
153
+ singular_name = ActiveSupport::Inflector.singularize(slot_name)
154
+ validate_singular_slot_name(ActiveSupport::Inflector.singularize(slot_name).to_sym)
155
+
156
+ setter_method_name = :"with_#{singular_name}"
157
+
158
+ define_method setter_method_name do |*args, &block|
159
+ set_slot(slot_name, nil, *args, &block)
160
+ end
161
+ ruby2_keywords(setter_method_name) if respond_to?(:ruby2_keywords, true)
162
+
163
+ define_method :"with_#{singular_name}_content" do |content|
164
+ send(setter_method_name) { content.to_s }
165
+
166
+ self
167
+ end
168
+
169
+ define_method :"with_#{slot_name}" do |collection_args = nil, &block|
170
+ collection_args.map do |args|
171
+ if args.respond_to?(:to_hash)
172
+ set_slot(slot_name, nil, **args, &block)
173
+ else
174
+ set_slot(slot_name, nil, *args, &block)
67
175
  end
68
- RUBY
176
+ end
177
+ end
178
+
179
+ self::GeneratedSlotMethods.define_method slot_name do
180
+ get_slot(slot_name)
181
+ end
182
+
183
+ self::GeneratedSlotMethods.define_method :"#{slot_name}?" do
184
+ get_slot(slot_name).present?
69
185
  end
70
186
 
71
- # Default class_name to ViewComponent::Slot
72
- class_name = "ViewComponent::Slot" unless class_name.present?
187
+ register_slot(slot_name, collection: true, callable: callable)
188
+ end
189
+ end
73
190
 
74
- # Register the slot on the component
75
- slots[slot_name] = {
76
- class_name: class_name,
77
- instance_variable_name: instance_variable_name,
78
- collection: collection
79
- }
191
+ def slot_type(slot_name)
192
+ registered_slot = registered_slots[slot_name]
193
+ if registered_slot
194
+ registered_slot[:collection] ? :collection : :single
195
+ else
196
+ plural_slot_name = ActiveSupport::Inflector.pluralize(slot_name).to_sym
197
+ plural_registered_slot = registered_slots[plural_slot_name]
198
+ plural_registered_slot&.fetch(:collection) ? :collection_item : nil
80
199
  end
81
200
  end
82
201
 
83
202
  def inherited(child)
84
203
  # Clone slot configuration into child class
85
204
  # see #test_slots_pollution
86
- child.slots = slots.clone
205
+ child.registered_slots = registered_slots.clone
206
+
207
+ # Add a module for slot methods, allowing them to be overriden by the component class
208
+ # see #test_slot_name_can_be_overriden
209
+ unless child.const_defined?(:GeneratedSlotMethods, false)
210
+ generated_slot_methods = Module.new
211
+ child.const_set(:GeneratedSlotMethods, generated_slot_methods)
212
+ child.include generated_slot_methods
213
+ end
87
214
 
88
215
  super
89
216
  end
90
- end
91
217
 
92
- # Build a Slot instance on a component,
93
- # exposing it for use inside the
94
- # component template.
95
- #
96
- # slot: Name of Slot, in symbol form
97
- # **args: Arguments to be passed to Slot initializer
98
- #
99
- # For example:
100
- # <%= render(SlotsComponent.new) do |component| %>
101
- # <% component.slot(:footer, class_names: "footer-class") do %>
102
- # <p>This is my footer!</p>
103
- # <% end %>
104
- # <% end %>
105
- #
106
- def slot(slot_name, **args, &block)
107
- # Raise ArgumentError if `slot` doesn't exist
108
- unless slots.key?(slot_name)
109
- raise ArgumentError.new "Unknown slot '#{slot_name}' - expected one of '#{slots.keys}'"
218
+ def register_polymorphic_slot(slot_name, types, collection:)
219
+ self::GeneratedSlotMethods.define_method(slot_name) do
220
+ get_slot(slot_name)
221
+ end
222
+
223
+ self::GeneratedSlotMethods.define_method(:"#{slot_name}?") do
224
+ get_slot(slot_name).present?
225
+ end
226
+
227
+ renderable_hash = types.each_with_object({}) do |(poly_type, poly_attributes_or_callable), memo|
228
+ if poly_attributes_or_callable.is_a?(Hash)
229
+ poly_callable = poly_attributes_or_callable[:renders]
230
+ poly_slot_name = poly_attributes_or_callable[:as]
231
+ else
232
+ poly_callable = poly_attributes_or_callable
233
+ poly_slot_name = nil
234
+ end
235
+
236
+ poly_slot_name ||=
237
+ if collection
238
+ "#{ActiveSupport::Inflector.singularize(slot_name)}_#{poly_type}"
239
+ else
240
+ "#{slot_name}_#{poly_type}"
241
+ end
242
+
243
+ memo[poly_type] = define_slot(
244
+ poly_slot_name, collection: collection, callable: poly_callable
245
+ )
246
+
247
+ setter_method_name = :"with_#{poly_slot_name}"
248
+
249
+ if instance_methods.include?(setter_method_name)
250
+ raise AlreadyDefinedPolymorphicSlotSetterError.new(setter_method_name, poly_slot_name)
251
+ end
252
+
253
+ define_method(setter_method_name) do |*args, &block|
254
+ set_polymorphic_slot(slot_name, poly_type, *args, &block)
255
+ end
256
+ ruby2_keywords(setter_method_name) if respond_to?(:ruby2_keywords, true)
257
+
258
+ define_method :"with_#{poly_slot_name}_content" do |content|
259
+ send(setter_method_name) { content.to_s }
260
+
261
+ self
262
+ end
263
+ end
264
+
265
+ registered_slots[slot_name] = {
266
+ collection: collection,
267
+ renderable_hash: renderable_hash
268
+ }
269
+ end
270
+
271
+ # Called by the compiler, as instance methods are not defined when slots are first registered
272
+ def register_default_slots
273
+ registered_slots.each do |slot_name, config|
274
+ config[:default_method] = instance_methods.find { |method_name| method_name == :"default_#{slot_name}" }
275
+
276
+ registered_slots[slot_name] = config
277
+ end
278
+ end
279
+
280
+ private
281
+
282
+ def register_slot(slot_name, **kwargs)
283
+ registered_slots[slot_name] = define_slot(slot_name, **kwargs)
284
+ end
285
+
286
+ def define_slot(slot_name, collection:, callable:)
287
+ slot = {collection: collection}
288
+ return slot unless callable
289
+
290
+ # If callable responds to `render_in`, we set it on the slot as a renderable
291
+ if callable.respond_to?(:method_defined?) && callable.method_defined?(:render_in)
292
+ slot[:renderable] = callable
293
+ elsif callable.is_a?(String)
294
+ # If callable is a string, we assume it's referencing an internal class
295
+ slot[:renderable_class_name] = callable
296
+ elsif callable.respond_to?(:call)
297
+ # If slot doesn't respond to `render_in`, we assume it's a proc,
298
+ # define a method, and save a reference to it to call when setting
299
+ method_name = :"_call_#{slot_name}"
300
+ define_method method_name, &callable
301
+ slot[:renderable_function] = instance_method(method_name)
302
+ else
303
+ raise(InvalidSlotDefinitionError)
304
+ end
305
+
306
+ slot
307
+ end
308
+
309
+ def validate_plural_slot_name(slot_name)
310
+ if RESERVED_NAMES[:plural].include?(slot_name.to_sym)
311
+ raise ReservedPluralSlotNameError.new(name, slot_name)
312
+ end
313
+
314
+ raise_if_slot_name_uncountable(slot_name)
315
+ raise_if_slot_conflicts_with_call(slot_name)
316
+ raise_if_slot_ends_with_question_mark(slot_name)
317
+ raise_if_slot_registered(slot_name)
318
+ end
319
+
320
+ def validate_singular_slot_name(slot_name)
321
+ if slot_name.to_sym == :content
322
+ raise ContentSlotNameError.new(name)
323
+ end
324
+
325
+ if RESERVED_NAMES[:singular].include?(slot_name.to_sym)
326
+ raise ReservedSingularSlotNameError.new(name, slot_name)
327
+ end
328
+
329
+ raise_if_slot_conflicts_with_call(slot_name)
330
+ raise_if_slot_ends_with_question_mark(slot_name)
331
+ raise_if_slot_registered(slot_name)
110
332
  end
111
333
 
112
- slot = slots[slot_name]
334
+ def raise_if_slot_registered(slot_name)
335
+ if registered_slots.key?(slot_name)
336
+ # TODO remove? This breaks overriding slots when slots are inherited
337
+ raise RedefinedSlotError.new(name, slot_name)
338
+ end
339
+ end
113
340
 
114
- # The class name of the Slot, such as Header
115
- slot_class = self.class.const_get(slot[:class_name])
341
+ def raise_if_slot_ends_with_question_mark(slot_name)
342
+ raise SlotPredicateNameError.new(name, slot_name) if slot_name.to_s.end_with?("?")
343
+ end
116
344
 
117
- unless slot_class <= ViewComponent::Slot
118
- raise ArgumentError.new "#{slot[:class_name]} must inherit from ViewComponent::Slot"
345
+ def raise_if_slot_conflicts_with_call(slot_name)
346
+ if slot_name.start_with?("call_")
347
+ raise InvalidSlotNameError, "Slot cannot start with 'call_'. Please rename #{slot_name}"
348
+ end
119
349
  end
120
350
 
121
- # Instantiate Slot class, accommodating Slots that don't accept arguments
122
- slot_instance = args.present? ? slot_class.new(**args) : slot_class.new
351
+ def raise_if_slot_name_uncountable(slot_name)
352
+ slot_name = slot_name.to_s
353
+ if slot_name.pluralize == slot_name.singularize
354
+ raise UncountableSlotNameError.new(name, slot_name)
355
+ end
356
+ end
357
+ end
358
+
359
+ def get_slot(slot_name)
360
+ content unless content_evaluated? # ensure content is loaded so slots will be defined
361
+
362
+ slot = self.class.registered_slots[slot_name]
363
+ @__vc_set_slots ||= {}
123
364
 
124
- # Capture block and assign to slot_instance#content
125
- slot_instance.content = view_context.capture(&block).to_s.strip.html_safe if block
365
+ if @__vc_set_slots[slot_name]
366
+ return @__vc_set_slots[slot_name]
367
+ end
126
368
 
127
369
  if slot[:collection]
128
- # Initialize instance variable as an empty array
129
- # if slot is a collection and has yet to be initialized
130
- unless instance_variable_defined?(slot[:instance_variable_name])
131
- instance_variable_set(slot[:instance_variable_name], [])
370
+ []
371
+ end
372
+ end
373
+
374
+ def set_slot(slot_name, slot_definition = nil, *args, &block)
375
+ slot_definition ||= self.class.registered_slots[slot_name]
376
+ slot = Slot.new(self)
377
+
378
+ # Passing the block to the sub-component wrapper like this has two
379
+ # benefits:
380
+ #
381
+ # 1. If this is a `content_area` style sub-component, we will render the
382
+ # block via the `slot`
383
+ #
384
+ # 2. Since we have to pass block content to components when calling
385
+ # `render`, evaluating the block here would require us to call
386
+ # `view_context.capture` twice, which is slower
387
+ slot.__vc_content_block = block if block
388
+
389
+ # If class
390
+ if slot_definition[:renderable]
391
+ slot.__vc_component_instance = slot_definition[:renderable].new(*args)
392
+ # If class name as a string
393
+ elsif slot_definition[:renderable_class_name]
394
+ slot.__vc_component_instance =
395
+ self.class.const_get(slot_definition[:renderable_class_name]).new(*args)
396
+ # If passed a lambda
397
+ elsif slot_definition[:renderable_function]
398
+ # Use `bind(self)` to ensure lambda is executed in the context of the
399
+ # current component. This is necessary to allow the lambda to access helper
400
+ # methods like `content_tag` as well as parent component state.
401
+ renderable_function = slot_definition[:renderable_function].bind(self)
402
+ renderable_value =
403
+ if block
404
+ renderable_function.call(*args) do |*rargs|
405
+ view_context.capture(*rargs, &block)
406
+ end
407
+ else
408
+ renderable_function.call(*args)
409
+ end
410
+
411
+ # Function calls can return components, so if it's a component handle it specially
412
+ if renderable_value.respond_to?(:render_in)
413
+ slot.__vc_component_instance = renderable_value
414
+ else
415
+ slot.__vc_content = renderable_value
132
416
  end
417
+ end
418
+
419
+ @__vc_set_slots ||= {}
133
420
 
134
- # Append Slot instance to collection accessor Array
135
- instance_variable_get(slot[:instance_variable_name]) << slot_instance
421
+ if slot_definition[:collection]
422
+ @__vc_set_slots[slot_name] ||= []
423
+ @__vc_set_slots[slot_name].push(slot)
136
424
  else
137
- # Assign the Slot instance to the slot accessor
138
- instance_variable_set(slot[:instance_variable_name], slot_instance)
425
+ @__vc_set_slots[slot_name] = slot
426
+ end
427
+
428
+ slot
429
+ end
430
+ ruby2_keywords(:set_slot) if respond_to?(:ruby2_keywords, true)
431
+
432
+ def set_polymorphic_slot(slot_name, poly_type = nil, *args, &block)
433
+ slot_definition = self.class.registered_slots[slot_name]
434
+
435
+ if !slot_definition[:collection] && (defined?(@__vc_set_slots) && @__vc_set_slots[slot_name])
436
+ raise ContentAlreadySetForPolymorphicSlotError.new(slot_name)
139
437
  end
140
438
 
141
- # Return nil, as this method shouldn't output anything to the view itself.
142
- nil
439
+ poly_def = slot_definition[:renderable_hash][poly_type]
440
+
441
+ set_slot(slot_name, poly_def, *args, &block)
143
442
  end
443
+ ruby2_keywords(:set_polymorphic_slot) if respond_to?(:ruby2_keywords, true)
144
444
  end
145
445
  end
@@ -0,0 +1,20 @@
1
+ module ViewComponent
2
+ module SlotableDefault
3
+ def get_slot(slot_name)
4
+ @__vc_set_slots ||= {}
5
+
6
+ return super unless !@__vc_set_slots[slot_name] && (default_method = registered_slots[slot_name][:default_method])
7
+
8
+ renderable_value = send(default_method)
9
+ slot = Slot.new(self)
10
+
11
+ if renderable_value.respond_to?(:render_in)
12
+ slot.__vc_component_instance = renderable_value
13
+ else
14
+ slot.__vc_content = renderable_value
15
+ end
16
+
17
+ slot
18
+ end
19
+ end
20
+ end