169 lines
5.1 KiB
Ruby
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
|