# frozen_string_literal: true require "tsort" require "dry/core/equalizer" module Hanami class View # @api private # @since 2.1.0 class Exposures include Dry::Equalizer(:exposures) include TSort # @api private # @since 2.1.0 attr_reader :exposures # @api private # @since 2.1.0 def initialize(exposures = {}) @exposures = exposures end # @api private # @since 2.1.0 def key?(name) exposures.key?(name) end # @api private # @since 2.1.0 def [](name) exposures[name] end # @api private # @since 2.1.0 def each(&block) exposures.each(&block) end # @api private # @since 2.1.0 def add(name, proc = nil, **options) exposures[name] = Exposure.new(name, proc, **options) end # @api private # @since 2.1.0 def import(name, exposure) exposures[name] = exposure.dup end # @api private # @since 2.1.0 def bind(obj) bound_exposures = exposures.each_with_object({}) { |(name, exposure), memo| memo[name] = exposure.bind(obj) } self.class.new(bound_exposures) end # @api private # @since 2.1.0 def call(input) # Avoid performance cost of tsorting when we don't need it names = if exposures.values.any?(&:dependencies?) # TODO: this sholud be cachable at time of `#add` tsort else exposures.keys end names .each_with_object({}) { |name, memo| next unless (exposure = self[name]) value = exposure.(input, memo) value = yield(value, exposure) if block_given? memo[name] = value } .tap { |hsh| names.each do |key| hsh.delete(key) if self[key].private? end } end private def tsort_each_node(&block) exposures.each_key(&block) end def tsort_each_child(name, &block) self[name].dependency_names.each(&block) if exposures.key?(name) end end end end