metamaps--metamaps/app/models/map.rb
2017-11-25 11:23:47 -08:00

169 lines
5.1 KiB
Ruby

# frozen_string_literal: true
class Map < ApplicationRecord
ATTRS_TO_WATCH = %w(name desc permission).freeze
belongs_to :user
belongs_to :source, class_name: :Map
belongs_to :updated_by, class_name: 'User'
has_many :topicmappings, -> { Mapping.topicmapping },
class_name: :Mapping, dependent: :destroy
has_many :synapsemappings, -> { Mapping.synapsemapping },
class_name: :Mapping, dependent: :destroy
has_many :topics, through: :topicmappings, source: :mappable, source_type: 'Topic'
has_many :synapses, through: :synapsemappings, source: :mappable, source_type: 'Synapse'
has_many :messages, as: :resource, dependent: :destroy
has_many :stars, dependent: :destroy
has_many :follows, as: :followed, dependent: :destroy
has_many :followers, through: :follows, source: :user
has_many :access_requests, dependent: :destroy
has_many :user_maps, dependent: :destroy
has_many :collaborators, through: :user_maps, source: :user
has_many :webhooks, as: :hookable
has_many :events, -> { includes :user }, as: :eventable, dependent: :destroy
# This method associates the attribute ":image" with a file attachment
has_attached_file :screenshot,
styles: {
thumb: ['220x220#', :png]
},
default_url: 'https://s3.amazonaws.com/metamaps-assets/site/missing-map-square.png'
validates :name, presence: true
validates :arranged, inclusion: { in: [true, false] }
validates :permission, presence: true
validates :permission, inclusion: { in: Perm::ISSIONS.map(&:to_s) }
# Validate the attached image is image/jpg, image/png, etc
validates_attachment_content_type :screenshot, content_type: %r{\Aimage/.*\Z}
after_create :after_created
after_update :after_updated
after_save :update_deferring_topics_and_synapses, if: :permission_changed?
before_destroy :before_destroyed
delegate :count, to: :topics, prefix: :topic # same as `def topic_count; topics.count; end`
delegate :count, to: :synapses, prefix: :synapse
delegate :count, to: :contributors, prefix: :contributor
delegate :count, to: :stars, prefix: :star
delegate :name, to: :user, prefix: true
def mappings
topicmappings.or(synapsemappings)
end
def contributors
User.where(id: mappings.map(&:user_id).uniq)
end
def editors
User.where(id: user_id).or(User.where(id: collaborators))
end
def user_image
user.image.url(:thirtytwo)
end
def collaborator_ids
collaborators.map(&:id)
end
def screenshot_url
screenshot.url(:thumb)
end
def created_at_str
created_at.strftime('%m/%d/%Y')
end
def updated_at_str
updated_at.strftime('%m/%d/%Y')
end
def starred_by_user?(user)
user&.stars&.where(map: self)&.exists? || false # return false, not nil
end
def as_json(_options = {})
json = super(
methods: %i(user_name user_image star_count topic_count synapse_count
contributor_count collaborator_ids screenshot_url),
except: %i(screenshot_content_type screenshot_file_size screenshot_file_name
screenshot_updated_at)
)
json[:created_at_clean] = created_at_str
json[:updated_at_clean] = updated_at_str
json
end
# user param helps determine what records are visible
def contains(user)
{
map: self,
topics: Pundit.policy_scope(user, topics).to_a,
synapses: Pundit.policy_scope(user, synapses).to_a,
mappings: Pundit.policy_scope(user, mappings).to_a,
mappers: contributors,
collaborators: editors,
messages: messages.sort_by(&:created_at),
stars: stars,
requests: access_requests
}
end
def add_new_collaborators(user_ids)
users = User.where(id: user_ids)
added = users.map do |new_user|
next nil if editors.include?(new_user)
UserMap.create(user_id: new_user.id, map_id: id)
new_user.id
end
added.compact
end
def remove_old_collaborators(user_ids)
removed = editors.map(&:id).map do |old_user_id|
next nil if user_ids.include?(old_user_id)
user_maps.where(user_id: old_user_id).find_each(&:destroy)
access_requests.where(user_id: old_user_id).find_each(&:destroy)
old_user_id
end
removed.compact
end
def update_deferring_topics_and_synapses
Topic.where(defer_to_map_id: id).update(permission: permission)
Synapse.where(defer_to_map_id: id).update(permission: permission)
end
protected
def after_created
FollowService.follow(self, user, 'created')
# notify users following the map creator
end
def after_updated
return unless ATTRS_TO_WATCH.any? { |k| changed_attributes.key?(k) }
ActionCable.server.broadcast 'map_' + id.to_s, type: 'mapUpdated'
end
def after_updated_async
if ATTRS_TO_WATCH.any? { |k| changed_attributes.key?(k) }
FollowService.follow(self, updated_by, 'contributed')
# NotificationService.notify_followers(self, 'map_updated', changed_attributes)
# or better yet publish an event
end
end
handle_asynchronously :after_updated_async
def before_destroyed
Map.where(source_id: id).find_each do |forked_map|
forked_map.update(source_id: nil)
end
end
end