engine/app/models/locomotive/content_entry.rb

147 lines
4.0 KiB
Ruby

module Locomotive
class ContentEntry
include Locomotive::Mongoid::Document
## extensions ##
include ::CustomFields::Target
include Extensions::Shared::Seo
## fields ##
field :_slug
field :_label_field_name
field :_position, :type => Integer, :default => 0
field :_visible, :type => Boolean, :default => true
## validations ##
validates :_slug, :presence => true, :uniqueness => { :scope => :content_type_id }
## associations ##
belongs_to :site
belongs_to :content_type, :class_name => 'Locomotive::ContentType', :inverse_of => :entries
## callbacks ##
before_validation :set_slug
before_save :set_visibility
before_save :set_label_field_name
before_create :add_to_list_bottom
after_create :send_notifications
## named scopes ##
scope :visible, :where => { :_visible => true }
scope :latest_updated, :order_by => :updated_at.desc, :limit => Locomotive.config.ui.latest_entries_nb
## methods ##
alias :visible? :_visible?
alias :_permalink :_slug
alias :_permalink= :_slug=
def _label(type = nil)
if self._label_field_name
self.send(self._label_field_name.to_sym)
else
self.send((type || self.content_type).label_field_name.to_sym)
end
end
def visible?
self._visible || self._visible.nil?
end
def next
next_or_previous :gt
end
def previous
next_or_previous :lt
end
def self.find_by_permalink(permalink)
self.where(:_slug => permalink).first
end
def self.sort_entries!(ids)
list = self.any_in(:_id => ids.map { |id| BSON::ObjectId.from_string(id.to_s) }).to_a
ids.each_with_index do |id, position|
if entry = list.detect { |e| e._id.to_s == id.to_s }
entry.update_attributes :_position => position
end
end
end
def to_liquid
Locomotive::Liquid::Drops::ContentEntry.new(self)
end
def to_presenter(options = {})
Locomotive::ContentEntryPresenter.new(self, options)
end
def as_json(options = {})
self.to_presenter.as_json
end
protected
def next_or_previous(matcher = :gt)
order_by = self.content_type.order_by_definition
criterion = attribute.send(matcher)
self.class.where(criterion => self.send(attribute)).order_by([order_by]).limit(1).first
end
# Sets the slug of the instance by using the value of the highlighted field
# (if available). If a sibling content instance has the same permalink then a
# unique one will be generated
def set_slug
self._slug = self._label.dup if self._slug.blank? && self._label.present?
if self._slug.present?
self._slug.permalink!
self._slug = self.next_unique_slug if self.slug_already_taken?
end
end
# Return the next available unique slug as a string
def next_unique_slug
slug = self._slug.gsub(/-\d*$/, '')
last_slug = self.class.where(:_id.ne => self._id, :_slug => /^#{slug}-?\d*?$/i).order_by(:_slug).last._slug
next_number = last_slug.scan(/-(\d)$/).flatten.first.to_i + 1
[slug, next_number].join('-')
end
def slug_already_taken?
self.class.where(:_id.ne => self._id, :_slug => self._slug).any?
end
def set_visibility
[:visible, :active].each do |meth|
if self.respond_to?(meth)
self._visible = self.send(meth)
return
end
end
end
def set_label_field_name
self._label_field_name = self.content_type.label_field_name
end
def add_to_list_bottom
self._position = self.class.max(:_position).to_i + 1
end
def send_notifications
return if !self.content_type.public_submission_enabled? || self.content_type.public_submission_accounts.blank?
self.content_type.site.accounts.each do |account|
next unless self.content_type.public_submission_accounts.include?(account._id.to_s)
Locomotive::Notifications.new_content_entry(account, self).deliver
end
end
end
end