Sha256: eb746eb5c9c584805ca17ac92650d6a82784888d8bef45398c3bc483ba3d0d97

Contents?: true

Size: 1.66 KB

Versions: 1

Compression:

Stored size: 1.66 KB

Contents

module Popolo
  module Sluggable
    extend ActiveSupport::Concern

    included do
      # The field used as the basis of the slug, usually the resource's name.
      field slug_source, type: String
      # A lowercase identifier composed of letters, numbers and dashes.
      field :slug, type: String
      # The field used to sort the document.
      field :sort_name, type: String

      index sort_name: 1

      validates_presence_of slug_source, :slug, :sort_name

      before_validation :set_slug_and_sort_name
    end

    module ClassMethods
      # Finds a resource by its slug or ID.
      #
      # @param [String] slug a slug or ID
      # @return a matching resource
      def find_by_slug_or_id(slug)
        where(slug: slug.to_s.parameterize).first || find(slug)
      end

      # Finds a resource by its slug or raises an error.
      #
      # @param [String] slug a slug
      # @param [Hash] opts conditions
      # @return a matching resource
      # @raises [Mongoid::Errors::DocumentNotFound] if not found
      def find_by_slug(slug, opts = {})
        find_by(opts.merge(slug: slug.parameterize))
      end

      # @private
      def slug_source
        if defined?(super)
          super
        else
          :name
        end
      end
    end

    def to_param
      slug
    end

  private

    # @return [String] the value of the field used as the basis of the slug
    def slug_source
      send(self.class.slug_source)
    end

    # @note Leave it to the content manager to choose a slug in case of conflicts.
    def set_slug_and_sort_name
      self.slug ||= slug_source.parameterize if slug_source
      self.sort_name ||= slug_source
    end
  end
end

Version data entries

1 entries across 1 versions & 1 rubygems

Version Path
popolo-0.0.2 lib/popolo/mixins/sluggable.rb