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.
- checksums.yaml +4 -4
- data/app/controllers/concerns/view_component/preview_actions.rb +5 -1
- data/app/controllers/view_components_system_test_controller.rb +24 -1
- data/app/helpers/preview_helper.rb +22 -4
- data/app/views/view_components/_preview_source.html.erb +2 -2
- data/docs/CHANGELOG.md +807 -1
- data/lib/rails/generators/abstract_generator.rb +9 -1
- data/lib/rails/generators/component/component_generator.rb +2 -1
- data/lib/rails/generators/component/templates/component.rb.tt +3 -2
- data/lib/rails/generators/erb/component_generator.rb +1 -1
- data/lib/rails/generators/locale/component_generator.rb +3 -3
- data/lib/rails/generators/preview/templates/component_preview.rb.tt +2 -0
- data/lib/rails/generators/rspec/component_generator.rb +15 -3
- data/lib/rails/generators/rspec/templates/component_spec.rb.tt +1 -1
- data/lib/rails/generators/stimulus/component_generator.rb +8 -3
- data/lib/rails/generators/stimulus/templates/component_controller.ts.tt +9 -0
- data/lib/rails/generators/test_unit/templates/component_test.rb.tt +1 -1
- data/lib/view_component/base.rb +169 -164
- data/lib/view_component/capture_compatibility.rb +44 -0
- data/lib/view_component/collection.rb +20 -8
- data/lib/view_component/compiler.rb +166 -207
- data/lib/view_component/config.rb +63 -14
- data/lib/view_component/deprecation.rb +1 -1
- data/lib/view_component/docs_builder_component.html.erb +5 -1
- data/lib/view_component/docs_builder_component.rb +28 -9
- data/lib/view_component/engine.rb +58 -28
- data/lib/view_component/errors.rb +240 -0
- data/lib/view_component/inline_template.rb +55 -0
- data/lib/view_component/instrumentation.rb +10 -2
- data/lib/view_component/preview.rb +7 -8
- data/lib/view_component/rails/tasks/view_component.rake +11 -2
- data/lib/view_component/slot.rb +119 -1
- data/lib/view_component/slotable.rb +394 -94
- data/lib/view_component/slotable_default.rb +20 -0
- data/lib/view_component/system_test_helpers.rb +5 -5
- data/lib/view_component/template.rb +134 -0
- data/lib/view_component/test_helpers.rb +138 -59
- data/lib/view_component/translatable.rb +45 -26
- data/lib/view_component/use_helpers.rb +42 -0
- data/lib/view_component/version.rb +4 -3
- data/lib/view_component/with_content_helper.rb +3 -8
- data/lib/view_component.rb +3 -12
- metadata +277 -38
- data/lib/view_component/content_areas.rb +0 -56
- data/lib/view_component/polymorphic_slots.rb +0 -103
- data/lib/view_component/preview_template_error.rb +0 -6
- data/lib/view_component/slot_v2.rb +0 -98
- data/lib/view_component/slotable_v2.rb +0 -391
- data/lib/view_component/template_error.rb +0 -9
data/lib/view_component/slot.rb
CHANGED
@@ -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
|
-
|
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 :
|
14
|
-
self.
|
19
|
+
class_attribute :registered_slots
|
20
|
+
self.registered_slots = {}
|
15
21
|
end
|
16
22
|
|
17
23
|
class_methods do
|
18
|
-
|
19
|
-
#
|
20
|
-
#
|
21
|
-
#
|
22
|
-
#
|
23
|
-
#
|
24
|
-
# )
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
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
|
-
|
37
|
-
|
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
|
-
#
|
42
|
-
|
43
|
-
|
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
|
-
|
100
|
+
define_method :"with_#{slot_name}_content" do |content|
|
101
|
+
send(setter_method_name) { content.to_s }
|
53
102
|
|
54
|
-
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
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
|
-
|
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
|
-
|
72
|
-
|
187
|
+
register_slot(slot_name, collection: true, callable: callable)
|
188
|
+
end
|
189
|
+
end
|
73
190
|
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
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.
|
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
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
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
|
-
|
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
|
-
|
115
|
-
|
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
|
-
|
118
|
-
|
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
|
-
|
122
|
-
|
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
|
-
|
125
|
-
|
365
|
+
if @__vc_set_slots[slot_name]
|
366
|
+
return @__vc_set_slots[slot_name]
|
367
|
+
end
|
126
368
|
|
127
369
|
if slot[:collection]
|
128
|
-
|
129
|
-
|
130
|
-
|
131
|
-
|
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
|
-
|
135
|
-
|
421
|
+
if slot_definition[:collection]
|
422
|
+
@__vc_set_slots[slot_name] ||= []
|
423
|
+
@__vc_set_slots[slot_name].push(slot)
|
136
424
|
else
|
137
|
-
|
138
|
-
|
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
|
-
|
142
|
-
|
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
|