2007-01-22 14:43:50 +01:00
|
|
|
module ActiveRecord
|
|
|
|
module Associations
|
|
|
|
class BelongsToPolymorphicAssociation < AssociationProxy #:nodoc:
|
|
|
|
def replace(record)
|
|
|
|
if record.nil?
|
|
|
|
@target = @owner[@reflection.primary_key_name] = @owner[@reflection.options[:foreign_type]] = nil
|
|
|
|
else
|
|
|
|
@target = (AssociationProxy === record ? record.target : record)
|
|
|
|
|
2009-08-04 17:16:03 +02:00
|
|
|
@owner[@reflection.primary_key_name] = record_id(record)
|
2008-06-02 08:35:38 +02:00
|
|
|
@owner[@reflection.options[:foreign_type]] = record.class.base_class.name.to_s
|
2007-01-22 14:43:50 +01:00
|
|
|
|
|
|
|
@updated = true
|
|
|
|
end
|
|
|
|
|
2010-05-25 19:45:45 +02:00
|
|
|
set_inverse_instance(record, @owner)
|
2007-01-22 14:43:50 +01:00
|
|
|
loaded
|
|
|
|
record
|
|
|
|
end
|
|
|
|
|
|
|
|
def updated?
|
|
|
|
@updated
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
2010-05-25 19:45:45 +02:00
|
|
|
# NOTE - for now, we're only supporting inverse setting from belongs_to back onto
|
|
|
|
# has_one associations.
|
|
|
|
def we_can_set_the_inverse_on_this?(record)
|
|
|
|
if @reflection.has_inverse?
|
|
|
|
inverse_association = @reflection.polymorphic_inverse_of(record.class)
|
|
|
|
inverse_association && inverse_association.macro == :has_one
|
2007-01-22 14:43:50 +01:00
|
|
|
else
|
2010-05-25 19:45:45 +02:00
|
|
|
false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def set_inverse_instance(record, instance)
|
|
|
|
return if record.nil? || !we_can_set_the_inverse_on_this?(record)
|
|
|
|
inverse_relationship = @reflection.polymorphic_inverse_of(record.class)
|
|
|
|
unless inverse_relationship.nil?
|
|
|
|
record.send(:"set_#{inverse_relationship.name}_target", instance)
|
2007-01-22 14:43:50 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-05-25 19:45:45 +02:00
|
|
|
def find_target
|
|
|
|
return nil if association_class.nil?
|
|
|
|
|
|
|
|
target =
|
|
|
|
if @reflection.options[:conditions]
|
|
|
|
association_class.find(
|
|
|
|
@owner[@reflection.primary_key_name],
|
|
|
|
:select => @reflection.options[:select],
|
|
|
|
:conditions => conditions,
|
|
|
|
:include => @reflection.options[:include]
|
|
|
|
)
|
|
|
|
else
|
|
|
|
association_class.find(@owner[@reflection.primary_key_name], :select => @reflection.options[:select], :include => @reflection.options[:include])
|
|
|
|
end
|
|
|
|
set_inverse_instance(target, @owner)
|
|
|
|
target
|
|
|
|
end
|
|
|
|
|
2007-01-22 14:43:50 +01:00
|
|
|
def foreign_key_present
|
|
|
|
!@owner[@reflection.primary_key_name].nil?
|
|
|
|
end
|
|
|
|
|
2009-08-04 17:16:03 +02:00
|
|
|
def record_id(record)
|
|
|
|
record.send(@reflection.options[:primary_key] || :id)
|
|
|
|
end
|
|
|
|
|
2007-01-22 14:43:50 +01:00
|
|
|
def association_class
|
|
|
|
@owner[@reflection.options[:foreign_type]] ? @owner[@reflection.options[:foreign_type]].constantize : nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|