Comment valider la présence d'un champ ou d'un autre mais pas les deux et au moins un?Valider la présence d'un champ ou d'un autre (XOR)
64
A
Répondre
111
Votre code ne fonctionnera que si vous ajoutez conditionals aux validations de numéricité comme ceci:
class Transaction < ActiveRecord::Base
validates_presence_of :date
validates_presence_of :name
validates_numericality_of :charge, allow_nil: true
validates_numericality_of :payment, allow_nil: true
validate :charge_xor_payment
private
def charge_xor_payment
unless charge.blank?^payment.blank?
errors.add(:base, "Specify a charge or a payment, not both")
end
end
end
9
Exemple de rails 3.
class Transaction < ActiveRecord::Base
validates_presence_of :date
validates_presence_of :name
validates_numericality_of :charge, :unless => proc{|obj| obj.charge.blank?}
validates_numericality_of :payment, :unless => proc{|obj| obj.payment.blank?}
validate :charge_xor_payment
private
def charge_xor_payment
if !(charge.blank?^payment.blank?)
errors[:base] << "Specify a charge or a payment, not both"
end
end
end
2
validate :father_or_mother
#Father nom ou mère nom est obligatoire
def father_or_mother
if father_last_name == "Last Name" or father_last_name.blank?
errors.add(:father_last_name, "cant blank")
errors.add(:mother_last_name, "cant blank")
end
end
Essayez ci-dessus exemple simple.
7
class Transaction < ActiveRecord::Base
validates_presence_of :date
validates_presence_of :name
validates_numericality_of :charge, allow_nil: true
validates_numericality_of :payment, allow_nil: true
validate :charge_xor_payment
private
def charge_xor_payment
if [charge, payment].compact.count != 1
errors.add(:base, "Specify a charge or a payment, not both")
end
end
end
Vous pouvez même le faire avec 3 ou plusieurs valeurs:
if [month_day, week_day, hour].compact.count != 1
31
Je pense que cela est plus idiomatiques dans Rails 3+:
par exemple: Pour valider que l'un des user_name
ou email
est présent:
validates :user_name, presence: true, unless: ->(user){user.email.present?}
validates :email, presence: true, unless: ->(user){user.user_name.present?}
+18
Cela ne gère pas les critères "pas les deux" –
0
J'ai mis ma réponse à cette question ci-dessous. Dans cet exemple :description
et :keywords
sont des champs dont l'un de ce pas être vide:
validate :some_was_present
belongs_to :seo_customable, polymorphic: true
def some_was_present
desc = description.blank?
errors.add(desc ? :description : :keywords, t('errors.messages.blank')) if desc && keywords.blank?
end
Ceci est appelé réponse parfaite. Merci @Semanticart –
La fonctionnalité fonctionne très bien. Mais, je ne pouvais pas obtenir les erreurs montrées sur la page de formulaire. Sauf si je fais quelque chose comme '= @ invoice.errors [: base] [0] 'sur mon _form.slim. Aucune suggestion? –