mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
c81af6ae72
We sometimes say "✂️ newline after `private`" in a code review (e.g. https://github.com/rails/rails/pull/18546#discussion_r23188776, https://github.com/rails/rails/pull/34832#discussion_r244847195). Now `Layout/EmptyLinesAroundAccessModifier` cop have new enforced style `EnforcedStyle: only_before` (https://github.com/rubocop-hq/rubocop/pull/7059). That cop and enforced style will reduce the our code review cost.
106 lines
2.8 KiB
Ruby
106 lines
2.8 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module ActionView
|
|
# This class defines the interface for a renderer. Each class that
|
|
# subclasses +AbstractRenderer+ is used by the base +Renderer+ class to
|
|
# render a specific type of object.
|
|
#
|
|
# The base +Renderer+ class uses its +render+ method to delegate to the
|
|
# renderers. These currently consist of
|
|
#
|
|
# PartialRenderer - Used for rendering partials
|
|
# TemplateRenderer - Used for rendering other types of templates
|
|
# StreamingTemplateRenderer - Used for streaming
|
|
#
|
|
# Whenever the +render+ method is called on the base +Renderer+ class, a new
|
|
# renderer object of the correct type is created, and the +render+ method on
|
|
# that new object is called in turn. This abstracts the setup and rendering
|
|
# into a separate classes for partials and templates.
|
|
class AbstractRenderer #:nodoc:
|
|
delegate :template_exists?, :any_templates?, :formats, to: :@lookup_context
|
|
|
|
def initialize(lookup_context)
|
|
@lookup_context = lookup_context
|
|
end
|
|
|
|
def render
|
|
raise NotImplementedError
|
|
end
|
|
|
|
class RenderedCollection # :nodoc:
|
|
def self.empty(format)
|
|
EmptyCollection.new format
|
|
end
|
|
|
|
attr_reader :rendered_templates
|
|
|
|
def initialize(rendered_templates, spacer)
|
|
@rendered_templates = rendered_templates
|
|
@spacer = spacer
|
|
end
|
|
|
|
def body
|
|
@rendered_templates.map(&:body).join(@spacer.body).html_safe
|
|
end
|
|
|
|
def format
|
|
rendered_templates.first.format
|
|
end
|
|
|
|
class EmptyCollection
|
|
attr_reader :format
|
|
|
|
def initialize(format)
|
|
@format = format
|
|
end
|
|
|
|
def body; nil; end
|
|
end
|
|
end
|
|
|
|
class RenderedTemplate # :nodoc:
|
|
attr_reader :body, :template
|
|
|
|
def initialize(body, template)
|
|
@body = body
|
|
@template = template
|
|
end
|
|
|
|
def format
|
|
template.format
|
|
end
|
|
|
|
EMPTY_SPACER = Struct.new(:body).new
|
|
end
|
|
|
|
private
|
|
def extract_details(options) # :doc:
|
|
@lookup_context.registered_details.each_with_object({}) do |key, details|
|
|
value = options[key]
|
|
|
|
details[key] = Array(value) if value
|
|
end
|
|
end
|
|
|
|
def instrument(name, **options) # :doc:
|
|
ActiveSupport::Notifications.instrument("render_#{name}.action_view", options) do |payload|
|
|
yield payload
|
|
end
|
|
end
|
|
|
|
def prepend_formats(formats) # :doc:
|
|
formats = Array(formats)
|
|
return if formats.empty? || @lookup_context.html_fallback_for_js
|
|
|
|
@lookup_context.formats = formats | @lookup_context.formats
|
|
end
|
|
|
|
def build_rendered_template(content, template)
|
|
RenderedTemplate.new content, template
|
|
end
|
|
|
|
def build_rendered_collection(templates, spacer)
|
|
RenderedCollection.new templates, spacer
|
|
end
|
|
end
|
|
end
|