module MakeWatchable module Watcher extend ActiveSupport::Concern included do has_many :watchings, :class_name => "MakeWatchable::Watching", :as => :watcher end module ClassMethods def watcher? true end end # Watch a +watchable+. # Raises an +AlreadyWatchingError+ if the watcher already is watching the watchable. # Raises an +InvalidWatchableError+ if the watchable is not a valid watchable. def watch!(watchable) check_watchable(watchable) if watches?(watchable) raise Exceptions::AlreadyWatchingError.new end Watching.create(:watchable => watchable, :watcher => self) true end # Watch a +watchable+, but don't raise an error if the watcher is already watching the # watchable. Instead simply return false then and ignore the watch request. # Raises an +InvalidWatchableError+ if the watchable is not a valid watchable. def watch(watchable) begin watch!(watchable) success = true rescue Exceptions::AlreadyWatchingError success = false end success end # Unwatch a +watchable+. # Raises an +NotWatchingError if the watcher is not watching the watchable. # Raises an +InvalidWatchableError+ if the watchable is not a valid watchable. def unwatch!(watchable) check_watchable(watchable) watching = fetch_watching(watchable) raise Exceptions::NotWatchingError unless watching watching.destroy true end # Unwatch a +watchable+, but don't raise an error if the watcher is not watching # the watchable. Instead returns false. # Raises an +InvalidWatchableError+ if the watchable is not a valid watchable. def unwatch(watchable) begin unwatch!(watchable) success = true rescue Exceptions::NotWatchingError success = false end success end # Check if the watcher watches a watchable. def watches?(watchable) fetch_watching(watchable) ? true : false end private def fetch_watching(watchable) watchings.where({ :watchable_type => watchable.class.to_s, :watchable_id => watchable.id }).try(:first) end def check_watchable(watchable) raise Exceptions::InvalidWatchableError unless watchable.class.watchable? end end end