mberlanda/cheidelacoriera

View on GitHub
app/controllers/dry_crud/render_callbacks.rb

Summary

Maintainability
A
0 mins
Test Coverage
A
100%
module DryCrud

  # Provide +before_render+ callbacks.
  module RenderCallbacks

    extend ActiveSupport::Concern

    included do
      extend ActiveModel::Callbacks
      prepend Prepends
    end

    # Prepended methods for callbacks.
    module Prepends

      # Helper method to run +before_render+ callbacks and render the action.
      # If a callback renders or redirects, the action is not rendered.
      def render(*args, &block)
        options = _normalize_render(*args, &block)
        callback = "render_#{options[:template]}"

        run_callbacks(callback) if respond_to?(:"_#{callback}_callbacks", true)

        super(*args, &block) unless performed?
      end

      private

      # Helper method the run the given block in between the before and after
      # callbacks of the given kinds.
      def with_callbacks(*kinds, &block)
        kinds.reverse.reduce(block) do |a, e|
          -> { run_callbacks(e, &a) }
        end.call
      end

    end

    # Class methods for callbacks.
    module ClassMethods

      # Defines before callbacks for the render actions.
      def define_render_callbacks(*actions)
        args = actions.map { |a| :"render_#{a}" }
        args << { only: :before, terminator: render_callback_terminator }
        define_model_callbacks(*args)
      end

      private

      def render_callback_terminator
        proc do |ctrl, result_lambda|
          terminate = true
          catch(:abort) do
            result_lambda.call if result_lambda.is_a?(Proc)
            terminate = !ctrl.response_body.nil?
          end
          terminate
        end
      end

    end

  end
end