5

我有两个模型TicketTicketComment, TicketComment 是 Ticket 的孩子。

票证.rb

class Ticket < ActiveRecord::Base
  has_many :ticket_comments, :dependent => :destroy, :order => 'created_at DESC'

  # allow the ticket comments to be created from within a ticket form
  accepts_nested_attributes_for :ticket_comments, :reject_if => proc { |attributes| attributes['comment'].blank? }
end

ticket_comment.rb

class TicketComment < ActiveRecord::Base
  belongs_to :ticket

  validates_presence_of :comment
end

我想做的是模仿 Trac 中的功能,如果用户对工单进行更改和/或添加评论,则会向分配给工单的人员发送一封电子邮件。

我想使用 after_update 或 after_save 回调,以便在发送电子邮件之前知道信息已全部保存。

如何检测模型的更改(ticket.changes)以及是否创建了新评论(ticket.comments)并在一封电子邮件中发送此更新(x 更改为 y,用户添加了评论“文本”)回调方法?

4

1 回答 1

4

您可以使用ActiveRecord::Dirty模块,它允许您跟踪未保存的更改。

例如

t1 = Ticket.first
t1.some_attribute = some_new_value
t1.changed? => true
t1.some_attribute_changed? => true
t1.some_attribute_was => old_value 

因此,在 before_create 的 before_update 中,您应该使用那些(您只能在保存之前检查!)。

收集所有这些方法的一个非常好的地方是在Observer 类TicketObserver 中,因此您可以将“观察者”代码与实际模型分开。

例如

class TicketObserver < ActiveRecord::Observer
  def before_update
    .. do some checking here ..
  end
end

要启用观察者类,您需要将其添加到您的environment.rb

config.active_record.observers = :ticket_observer

这应该让你开始:)

与链接评论有关的问题。如果你这样做:

new_comment = ticket.ticket_comments.build
new_comment.new_record? => true
ticket.comments.changed => true

所以这正是你所需要的。这对你不起作用吗?再次注意:当然,您需要在保存之前检查这一点:)

我想您必须收集在 before_create 或 before_update 中更改的数据,并在 after_update/create 中实际发送邮件(因为您确定它成功了)。

显然还不清楚。我会让它更明确一点。我会推荐使用 TicketObserver 类。但是如果你想使用回调,它会是这样的:

class Ticked

  before_save :check_state
  after_save :send_mail_if_needed

  def check_state
    @logmsg=""
    if ticket_comments.changed
      # find the comment
      ticket_comments.each do |c| 
        @logmsg << "comment changed" if c.changed?
        @logmsg << "comment added" if c.new_record? 
      end
    end
  end

end
def send_mail_if_needed
  if @logmsg.size > 0
    ..send mail..
  end
end
于 2010-05-11T23:46:25.360 回答