module Shoulda
  module Matchers
    module Doublespeak
      # @private
      class Double
        attr_reader :calls

        def initialize(world, klass, method_name, implementation)
          @world = world
          @klass = klass
          @method_name = method_name
          @implementation = implementation
          @activated = false
          @calls = []

          if world.doubles_activated?
            activate
          end
        end

        def activated?
          @activated
        end

        def to_return(value = nil, &block)
          if block
            implementation.returns(&block)
          else
            implementation.returns(value)
          end
        end

        def activate
          unless @activated
            store_original_method
            replace_method_with_double
            @activated = true
          end
        end

        def deactivate
          if @activated
            restore_original_method
            @activated = false
          end
        end

        def record_call(call)
          calls << call
        end

        def call_original_method(call)
          unbound_method = world.original_method_for(klass, call.method_name)

          if unbound_method
            unbound_method.bind(call.object).call(*call.args, &call.block)
          end
        end

        protected

        attr_reader :world, :klass, :method_name, :implementation,
          :original_method

        def store_original_method
          world.store_original_method_for(klass, method_name)
        end

        def replace_method_with_double
          double = self
          implementation = @implementation
          _method_name = method_name

          if klass.instance_methods(false).include?(method_name)
            klass.__send__(:remove_method, method_name)
          end

          klass.__send__(:define_method, method_name) do |*args, &block|
            call = MethodCall.new(
              double: double,
              object: self,
              method_name: _method_name,
              args: args,
              block: block,
              caller: caller
            )
            implementation.call(call)
          end
        end

        def restore_original_method
          original_method = world.original_method_for(klass, method_name)

          klass.__send__(:remove_method, method_name)

          klass.__send__(:define_method, method_name) do |*args, &block|
            original_method.bind(self).call(*args, &block)
          end
        end
      end
    end
  end
end
