# frozen_string_literal: true

module Vedeu

  # Provides a single interface to all registered renderers.
  #
  module Renderers

    extend Enumerable
    extend self
    extend Vedeu::Repositories::Storage

    # Instructs each renderer registered with Vedeu to clear their
    # content.
    #
    # @example
    #   Vedeu.clear
    #   Vedeu.trigger(:_clear_)
    #   Vedeu.renderers.clear
    #
    # @return [Array<void>]
    def clear
      storage.map do |renderer|
        log('Clearing', renderer)

        threaded(renderer) { renderer.clear }
      end.each(&:join) if Vedeu.ready?

      ''
    end

    # Provides access to the list of renderers.
    #
    # @example
    #   Vedeu.renderers
    #
    # @return [Module]
    def renderers
      self
    end

    # Instructs each renderer registered with Vedeu to render the
    # output as content.
    #
    # @example
    #   Vedeu.renderers.render(output)
    #
    # @param output [void]
    # @return [Array<void>]
    def render(output)
      storage.map do |renderer|
        log('Rendering', renderer)

        threaded(renderer) { renderer.render(output) }
      end.each(&:join) if Vedeu.ready?

      output
    end

    private

    # @return [Set]
    def in_memory
      if Vedeu.config.renderers.any?
        Set.new(Vedeu.config.renderers)

      else
        Set.new([Vedeu::Renderers::Terminal.new])

      end
    end

    # @param message [String]
    # @param renderer [Class]
    # @return [void]
    def log(message, renderer)
      Vedeu.log(type:    :render,
                message: "#{message} via #{renderer.class.name}")
    end

    # @return [Mutex]
    def mutex
      @mutex ||= Mutex.new
    end

    # @param block [Proc]
    # @return [void]
    def threaded(renderer, &block)
      Thread.new(renderer) do
        mutex.synchronize do
          toggle_cursor do
            yield
          end
        end
      end
    end

    # @param block [Proc]
    # @return [void]
    def toggle_cursor(&block)
      Vedeu.trigger(:_hide_cursor_)

      yield

      Vedeu.trigger(:_show_cursor_)
    end

  end # Renderers

  # @example
  #   Vedeu.renderers
  #
  # @!method renderers
  #   @see Vedeu::Renderers#renderers
  def_delegators Vedeu::Renderers,
                 :renderers

end # Vedeu

require 'vedeu/renderers/options'
require 'vedeu/renderers/file'
require 'vedeu/renderers/escape'
require 'vedeu/renderers/html'
require 'vedeu/renderers/json'
require 'vedeu/renderers/terminal'
require 'vedeu/renderers/text'