# frozen_string_literal: true require "dry/core/equalizer" require "dry/core/constants" module Hanami class View # Evaluation context for templates (including layouts and partials) and # provides a place to encapsulate view-specific behaviour alongside a # template and its locals. # # @abstract Subclass this and provide your own methods adding view-specific # behavior. You should not override `#initialize` # # @see https://dry-rb.org/gems/dry-view/templates/ # @see https://dry-rb.org/gems/dry-view/scopes/ # # @api public # @since 2.1.0 class Scope # @api private CONVENIENCE_METHODS = %i[format context locals].freeze include Dry::Equalizer(:_name, :_locals, :_rendering) # Returns the scope's name. # # @return [Symbol] # # @api public # @since 2.1.0 attr_reader :_name # Returns the scope's locals # # @overload _locals # Returns the locals. # @overload locals # A convenience alias for `#_locals.` Is available unless there is a local named `locals` # # @return [Hash[] # # @api public # @since 2.1.0 attr_reader :_locals # Returns the current rendering. # # @return [Rendering] # # @api private # @since 2.1.0 attr_reader :_rendering # Returns a new Scope instance. # # @param name [Symbol, nil] scope name # @param locals [Hash] template locals # @param rendering [Rendering] the current rendering # # @return [Scope] # # @api public # @since 2.1.0 def initialize( name: nil, locals: Dry::Core::Constants::EMPTY_HASH, rendering: RenderingMissing.new ) @_name = name @_locals = locals @_rendering = rendering end # @overload render(partial_name, **locals, &block) # Renders a partial using the scope. # # @param partial_name [Symbol, String] partial name # @param locals [Hash] partial locals # @yieldreturn [String] string content to include where the partial calls `yield` # # @overload render(**locals, &block) # Renders a partial (named after the scope's own name) using the scope. # # @param locals[Hash] partial locals # @yieldreturn [String] string content to include where the partial calls `yield` # # @return [String] the rendered partial output # # @api public # @since 2.1.0 def render(partial_name = nil, **locals, &block) partial_name ||= _name unless partial_name raise ArgumentError, "+partial_name+ must be provided for unnamed scopes" end if partial_name.is_a?(Class) partial_name = _inflector.underscore(_inflector.demodulize(partial_name.to_s)) end _rendering.partial(partial_name, _render_scope(**locals), &block) end # Builds a new scope using a scope class matching the provided name. # # @param name [Symbol, Class] scope name (or class) # @param locals [Hash] scope locals # # @return [Scope] # # @api public # @since 2.1.0 def scope(name = nil, **locals) _rendering.scope(name, locals) end # Returns the template format for the current render environment. # # @overload _format # Returns the format. # @overload format # A convenience alias for `#_format.` Is available unless there is a # local named `format` # # @return [Symbol] format # # @api public # @since 2.1.0 def _format _rendering.format end # Returns the context object for the current render environment. # # @overload _context # Returns the context. # @overload context # A convenience alias for `#_context`. Is available unless there is a # local named `context`. # # @return [Context] context # # @api public # @since 2.1.0 def _context _rendering.context end private # Handles missing methods, according to the following rules: # # 1. If there is a local with a name matching the method, it returns the local. # 2. If the `context` responds to the method, then it will be sent the method and all its # arguments. def method_missing(name, *args, &block) if _locals.key?(name) _locals[name] elsif _context.respond_to?(name) _context.public_send(name, *args, &block) elsif CONVENIENCE_METHODS.include?(name) __send__(:"_#{name}", *args, &block) else super end end ruby2_keywords(:method_missing) if respond_to?(:ruby2_keywords, true) def respond_to_missing?(name, include_private = false) _locals.key?(name) || _rendering.context.respond_to?(name) || CONVENIENCE_METHODS.include?(name) || super end def _render_scope(**locals) if locals.none? self else self.class.new( # FIXME: what about `name`? locals: locals, rendering: _rendering ) end end def _inflector _rendering.inflector end end end end