Class: Compony::Component

Inherits:
Object
  • Object
show all
Defined in:
lib/compony/component.rb

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(parent_comp = nil, index: 0, **comp_opts) ⇒ Component

Returns a new instance of Component.



43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
# File 'lib/compony/component.rb', line 43

def initialize(parent_comp = nil, index: 0, **comp_opts)
  @parent_comp = parent_comp
  @sub_comps = []
  @index = index
  @comp_opts = comp_opts
  @before_render_blocks = NaturalOrdering.new
  @content_blocks = NaturalOrdering.new
  @actions = NaturalOrdering.new
  @skipped_actions = Set.new
  @path_block = proc do |model = nil, *args_for_path_helper, standalone_name: nil, **kwargs_for_path_helper|
    kwargs_for_path_helper.merge!(id: model.id) if model
    next Rails.application.routes.url_helpers.send(
      "#{Compony.path_helper_name(comp_cst, family_cst, standalone_name&.to_sym)}_path",
      *args_for_path_helper,
      **kwargs_for_path_helper
    )
  end

  init_standalone
  init_labelling

  fail "#{inspect} is missing a call to `setup`." unless setup_blocks&.any?

  setup_blocks.each do |setup_block|
    instance_exec(&setup_block)
  end
end

Instance Attribute Details

#comp_optsObject (readonly)



9
10
11
# File 'lib/compony/component.rb', line 9

def comp_opts
  @comp_opts
end

#content_blocksObject (readonly)

needed in RequestContext for nesting



10
11
12
# File 'lib/compony/component.rb', line 10

def content_blocks
  @content_blocks
end

#parent_compObject (readonly)



8
9
10
# File 'lib/compony/component.rb', line 8

def parent_comp
  @parent_comp
end

Class Method Details

.comp_cstObject

Returns the name of the class constant of this component. Do not override.



34
35
36
# File 'lib/compony/component.rb', line 34

def self.comp_cst
  name.demodulize.to_sym
end

.comp_nameObject

Returns the component name



39
40
41
# File 'lib/compony/component.rb', line 39

def self.comp_name
  comp_cst.to_s.underscore
end

.family_cstObject

Returns the name of the module constant (=family) of this component. Do not override.



24
25
26
# File 'lib/compony/component.rb', line 24

def self.family_cst
  module_parent.to_s.demodulize.to_sym
end

.family_nameObject

Returns the family name



29
30
31
# File 'lib/compony/component.rb', line 29

def self.family_name
  family_cst.to_s.underscore
end

.setup(&block) ⇒ Object

DSL method



16
17
18
19
20
21
# File 'lib/compony/component.rb', line 16

def self.setup(&block)
  fail("`setup` expects a block in #{inspect}.") unless block_given?
  self.setup_blocks ||= []
  self.setup_blocks = setup_blocks.dup # This is required to prevent the parent class to see children's setup blocks.
  setup_blocks << block
end

Instance Method Details

#action(action_name, before: nil, &block) ⇒ Object

DSL method Adds or replaces an action (for action buttons) If before: is specified, will insert the action before the named action. When replacing, an element keeps its position unless before: is specified.



232
233
234
# File 'lib/compony/component.rb', line 232

def action(action_name, before: nil, &block)
  @actions.natural_push(action_name, block, before:)
end

#before_render(name = :main, before: nil, &block) ⇒ Object

DSL method Adds or overrides a before_render block. You can use controller.redirect_to to redirect away and halt the before_render/content chain

Parameters:

  • name (Symbol, String) (defaults to: :main)

    The name of the before_render block, defaults to :main

  • before (nil, Symbol, String) (defaults to: nil)

    If nil, the block will be added to the bottom of the before_render chain. Otherwise, pass the name of another block.

  • block (Proc)

    The block that should be run as part of the before_render pipeline. Will run in the component's context.



155
156
157
158
# File 'lib/compony/component.rb', line 155

def before_render(name = :main, before: nil, **, &block)
  fail("`before_render` expects a block in #{inspect}.") unless block_given?
  @before_render_blocks.natural_push(name, block, before:, **)
end

#content(name = :main, before: nil, &block) ⇒ Object

DSL method Adds or overrides a content block.

Parameters:

  • name (Symbol, String) (defaults to: :main)

    The name of the content block, defaults to :main

  • before (nil, Symbol, String) (defaults to: nil)

    If nil, the block will be added to the bottom of the content chain. Otherwise, pass the name of another block.

  • kwargs (Hash)

    If hidden is true, the content will not be rendered by default, allowing you to nest it in another content block.

  • block (Proc)

    The block that should be run as part of the content pipeline. Will run in the component's context. You can use Dyny here.



166
167
168
169
170
171
172
# File 'lib/compony/component.rb', line 166

def content(name = :main, before: nil, **, &block)
  # A block is required here, but if this is an override (e.g. to hide another content block), we can tolerate the missing block.
  if !block_given? && @content_blocks.find { |b| b.name == name }.nil?
    fail("`content` expects a block in #{inspect}.")
  end
  @content_blocks.natural_push(name, block || :missing, before:, **)
end

#idObject

Returns an identifier describing this component. Must be unique among simplings under the same parent_comp. Do not override.



90
91
92
# File 'lib/compony/component.rb', line 90

def id
  "#{family_name}_#{comp_name}_#{@index}"
end

#id_pathObject

Returns the id_path from the root_comp. Do not overwrite.



96
97
98
99
100
101
102
# File 'lib/compony/component.rb', line 96

def id_path
  if root_comp?
    id
  else
    "#{parent_comp.id_path}/#{id}"
  end
end

#id_path_hashObject

Returns a hash for the id_path. Used for params prefixing. Do not overwrite.



106
107
108
# File 'lib/compony/component.rb', line 106

def id_path_hash
  Digest::SHA1.hexdigest(id_path)[..4]
end

#inspectObject



71
72
73
# File 'lib/compony/component.rb', line 71

def inspect
  "#<#{self.class.name}:#{hash}>"
end

#param_name(unprefixed_param_name) ⇒ Object

Given an unprefixed name of a param, adds the id_path hash Do not overwrite.



112
113
114
# File 'lib/compony/component.rb', line 112

def param_name(unprefixed_param_name)
  "#{id_path_hash}_#{unprefixed_param_name}"
end

#path(&block) ⇒ Object

DSL method Overrides how the path to this component should be generated. The block will be given the following args: a model (optional), pos. args for the path helper, the kwarg standalone_name and kwargs for the path helper. The block is expected to return a Rails path. It is not given controller or helpers, instead use: Rails.application.routes.url_helpers. For an example, refer to the initializer of this class, where the default block is defined.



140
141
142
143
144
145
146
147
# File 'lib/compony/component.rb', line 140

def path(*, **, &block)
  if block_given?
    # Assignment via DSL
    @path_block = block
  else
    @path_block.call(*, **)
  end
end

#remove_content(name) ⇒ Object

DSL method Removes a content block. Use this in subclasses if a content block defined in the parent should be removed from the child.

Parameters:

  • name (Symbol, String)

    Name of the content block that should be removed



177
178
179
180
181
182
183
184
185
# File 'lib/compony/component.rb', line 177

def remove_content(name) # rubocop:disable Naming/PredicateMethod
  existing_index = @content_blocks.find_index { |el| el.name == name.to_sym }
  if existing_index.nil?
    return false
  else
    @content_blocks.delete_at(existing_index)
    return true
  end
end

#remove_content!(name) ⇒ Object

DSL method Removes a content block and fails if the content block was not found.

Parameters:

  • name (Symbol, String)

    Name of the content block that should be removed



190
191
192
# File 'lib/compony/component.rb', line 190

def remove_content!(name)
  remove_content(name) || fail("Content block #{name.inspect} not found for removal in #{inspect}.")
end

#render(controller, standalone: false, **locals) ⇒ Object

Renders the component using the controller passsed to it and returns it as a string. Do not overwrite.

Parameters:

  • standalone (Boolean) (defaults to: false)

    pass true iff render is called from render_standalone



197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
# File 'lib/compony/component.rb', line 197

def render(controller, standalone: false, **locals)
  # Call before_render hooks (if any) and backfire instance variables back to the component
  @before_render_blocks.each do |element|
    RequestContext.new(self, controller, locals:).evaluate_with_backfire(&element.payload)
    # Stop if a `before_render` block issued a body (e.g. through redirecting)
    break unless controller.response_body.nil?
  end

  # Render, unless before_render has already issued a body (e.g. through redirecting).
  if controller.response_body.nil?
    fail "#{self.class.inspect} must define `content` or set a response body in `before_render`" if @content_blocks.none?
    return controller.render_to_string(
      type:   :dyny,
      locals: { content_blocks: @content_blocks, standalone:, component: self, render_locals: locals },
      inline: "        if Compony.content_before_root_comp_block && standalone\n          Compony::RequestContext.new(component, controller, helpers: self, locals: render_locals).evaluate(&Compony.content_before_root_comp_block)\n        end\n        content_blocks.reject{ |el| el.hidden }.each do |element|\n          # Instanciate and evaluate a fresh RequestContext in order to use the buffer allocated by the ActionView (needed for `concat` calls)\n          Compony::RequestContext.new(component, controller, helpers: self, locals: render_locals).evaluate(&element.payload)\n        end\n        if Compony.content_after_root_comp_block && standalone\n          Compony::RequestContext.new(component, controller, helpers: self, locals: render_locals).evaluate(&Compony.content_after_root_comp_block)\n        end\n      RUBY\n    )\n  else\n    return nil # Prevent double render errors\n  end\nend\n"

#render_actions(controller, wrapper_class: '', action_class: '') ⇒ Object

Used to render all actions of this component, each button wrapped in a div with the specified class



243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
# File 'lib/compony/component.rb', line 243

def render_actions(controller, wrapper_class: '', action_class: '')
  h = controller.helpers
  h.(:div, class: wrapper_class) do
    button_htmls = @actions.map do |action|
      next if @skipped_actions.include?(action.name)
      Compony.with_button_defaults(feasibility_action: action.name.to_sym) do
        action_button = action.payload.call(controller)
        next unless action_button
        button_html = action_button.render(controller)
        next if button_html.blank?
        h.(:div, button_html, class: action_class)
      end
    end
    next h.safe_join button_htmls
  end
end

#resourceful?Boolean

Is true for resourceful components

Returns:

  • (Boolean)


261
262
263
# File 'lib/compony/component.rb', line 261

def resourceful?
  return false
end

#root_compObject

Returns the current root comp. Do not overwrite.



77
78
79
80
# File 'lib/compony/component.rb', line 77

def root_comp
  return self unless parent_comp
  return parent_comp.root_comp
end

#root_comp?Boolean

Returns whether or not this is the root comp. Do not overwrite.

Returns:

  • (Boolean)


84
85
86
# File 'lib/compony/component.rb', line 84

def root_comp?
  parent_comp.nil?
end

#skip_action(action_name) ⇒ Object

DSL method Marks an action for skip



238
239
240
# File 'lib/compony/component.rb', line 238

def skip_action(action_name)
  @skipped_actions << action_name.to_sym
end

#sub_comp(component_class, **comp_opts) ⇒ Object

Instanciate a component with self as a parent



117
118
119
120
121
# File 'lib/compony/component.rb', line 117

def sub_comp(component_class, **comp_opts)
  sub = component_class.new(self, index: @sub_comps.count, **comp_opts)
  @sub_comps << sub
  return sub
end