class Mocha::ClassMethod

def define_new_method

def define_new_method
  stubbee.__metaclass__.class_eval(%{
    def #{method}(*args, &block)
      mocha.method_missing(:#{method}, *args, &block)
    end
  }, __FILE__, __LINE__)
end

def hide_original_method

def hide_original_method
  if method_exists?(method)
    begin
      @original_method = stubbee._method(method)
      @original_visibility = :public
      if stubbee.__metaclass__.protected_instance_methods.include?(method)
        @original_visibility = :protected
      elsif stubbee.__metaclass__.private_instance_methods.include?(method)
        @original_visibility = :private
      end
      if @original_method && @original_method.owner == stubbee.__metaclass__
        stubbee.__metaclass__.send(:remove_method, method)
      end
    rescue NameError
      # deal with nasties like ActiveRecord::Associations::AssociationProxy
    end
  end
end

def initialize(stubbee, method)

def initialize(stubbee, method)
  @stubbee = stubbee
  @original_method, @original_visibility = nil, nil
  @method = RUBY_VERSION < '1.9' ? method.to_s : method.to_sym
end

def matches?(other)

def matches?(other)
  return false unless (other.class == self.class)
  (stubbee.object_id == other.stubbee.object_id) and (method == other.method)
end

def method_exists?(method)

def method_exists?(method)
  symbol = method.to_sym
  __metaclass__ = stubbee.__metaclass__
  __metaclass__.public_method_defined?(symbol) || __metaclass__.protected_method_defined?(symbol) || __metaclass__.private_method_defined?(symbol)
end

def mock

def mock
  stubbee.mocha
end

def remove_new_method

def remove_new_method
  stubbee.__metaclass__.send(:remove_method, method)
end

def reset_mocha

def reset_mocha
  stubbee.reset_mocha
end

def restore_original_method

def restore_original_method
  if @original_method && @original_method.owner == stubbee.__metaclass__
    if RUBY_VERSION < '1.9'
      original_method = @original_method
      stubbee.__metaclass__.send(:define_method, method) do |*args, &block|
        original_method.call(*args, &block)
      end
    else
      stubbee.__metaclass__.send(:define_method, method, @original_method)
    end
  end
  if @original_visibility
    Module.instance_method(@original_visibility).bind(stubbee.__metaclass__).call(method)
  end
end

def stub

def stub
  hide_original_method
  define_new_method
end

def to_s

def to_s
  "#{stubbee}.#{method}"
end

def unstub

def unstub
  remove_new_method
  restore_original_method
  mock.unstub(method.to_sym)
  unless mock.any_expectations?
    reset_mocha
  end
end