== About
This plugin provides an easy way to give your ActiveRecord models clean urlnames, for
use anywhere you don't want to use a plain old integer id.
To use it, first either copy over the migration in db/00x_add_urlnames_table.rb
and migrate it to it, or import the SQL in db/urlnames_table.sql.
This plugin also requires at least Rails 1.1 (currently edge. See http://wiki.rubyonrails.com/rails/pages/EdgeRails)
For updates see http://gabriel.gironda.org, and for licensing see LICENSE.
Version: 0.6
----
== Simple usage:
=== Setting the url name
The usage that will work for 80% of cases is this:
class Article < ActiveRecord::Base
acts_as_urlnameable :title
end
This will format the value of the +title+ attribute of the record to one suitable for
use in a URL and save it. Further changes to +title+ will not overwrite the url name.
The value can then be accessed with +object.urlname+. If the url name already exists,
an error will be added to the +urlname+ attribute.
To allow overwrites of the url name, call +acts_as_urlname+ with the following option:
class Article < ActiveRecord::Base
acts_as_urlnameable :title, :overwrite => true
end
=== Validation
The url name will now be overwritten if the value of +title+ changes. To change the
default validation message you can use the :message option:
class Article < ActiveRecord::Base
acts_as_urlnameable :title, :overwrite => true, :message => 'is unavailable.'
end
=== Finding records
Records can be located via the +find_by_urlname+ method, like so:
Article.find_by_urlname('some_urlname')
----
== Advanced usage:
This section is for those of us who like to mess with ActiveRecord a little more than
usual.
=== Setting the url name
You can set objects to be known by any past url names they may have had, even if the
current one is different. The usage in this case would be:
class Article < ActiveRecord::Base
acts_as_urlnameable :title, :mode => :multiple
end
The instance method +past_urlnames+ will then return all url names other than the current
one, while +all_urlnames+ will return all url names the object has ever been known by,
including the current url name. The :overwrite option has no effect on objects
with multiple urlnames for obvious reasons.
You can also define your own urlnameify method if you would prefer not to use the
default url name formatting.
class Article < ActiveRecord::Base
acts_as_urlnameable :title, :mode => :multiple
protected
def urlnameify(text)
text.to_s.downcase + '_permalink'
end
end
=== Validation
Acts as Urlnameable gives you five options for validation:
* Skip any kind of validation (usually undesirable)
* Validating against the base class (the default)
* Validating against the child class (for use with single table inheritance)
* Validating against the parent object (for associations)
* Define your own validate_urlname method
==== Skipping validation
To skip validation, use the plugin like so:
class Article < ActiveRecord::Base
acts_as_urlnameable :title, :mode => :multiple, :validate => false
end
==== Validating against the base class or child class
If you're using single table inheritance in ActiveRecord, and your models look like this for example:
class Article < ActiveRecord::Base
acts_as_urlnameable :title
end
class SpecialArticle < Article
end
class Draft < Article
end
Then by default, a Draft can not have the same url name as a SpecialArticle. If you'd like to change
this behaviour, then use the :sti_class option with :validate
class Article < ActiveRecord::Base
acts_as_urlnameable :title, :validate => :sti_class
end
Now Draft and SpecialArticle can have the same url name set. SpecialArticle.find_by_urlname
will find only the SpecialArticle with the matching url name, and Draft.find_by_urlname will
find only the Draft with the url name given.
Article.find_by_urlname will find the first Article with the given url name, whereas Article.find_all_by_urlname
will find both the Draft and SpecialArticle objects.
==== Validating against the parent object
Here's an example model setup:
class Person < ActiveRecord::Base
has_many :articles
end
class Article < ActiveRecord::Base
acts_as_urlnameable :title
belongs_to :person
end
If you would like to validate the url name for an article against the parent object, pass the name
of the belongs_to association as the option for :validate. Example:
class Article < ActiveRecord::Base
acts_as_urlnameable :title, :validate => :person
belongs_to :person
end
Now for the validation to work properly, you must instantiate the new article through the association.
This is best explained like this:
# Find two different people
bob = Person.find_by_first_name('bob')
joe = Person.find_by_first_name('joe')
# Have Bob write a new article
bobs_article = bob.articles.build(:title => "My first article", :body => "This is my first article")
bobs_article.save # passes validation and saves ok
# Have Joe write a new article
joes_article = joe.articles.build(:title => "My first article", :body => "This is my first article")
joes_article.save # passes validation and saves ok
# Find Bob's first article
bob.articles.find_by_urlname('my_first_article')
# Find Joe's first article
joe.articles.find_by_urlname('my_first_article')
Both articles validate because the validation is scoped against the owner. Bob and Joe can both have
articles known as 'my_first_article', and the finder method will find only their own articles.
Article.find_all_by_urlname('my_first_article') will find both articles.
==== Custom validation
You can also define your own validate_urlname method in the class for custom validation, like so:
class Person < ActiveRecord::Base
acts_as_urlnameable :first_name
protected
def validate_urlname
if Person.find_by_urlname(attr_to_urlname)
errors.add(:urlname, 'is not available.')
elsif attr_to_urlname == 'bob'
errors.add_to_base('You might be Bob, and I hate Bob.')
end
end
end