2005-08-15 19:17:32 +00:00
|
|
|
class Page < ActiveRecord::Base
|
|
|
|
belongs_to :web
|
|
|
|
has_many :revisions, :order => 'id'
|
2005-09-11 16:49:08 +00:00
|
|
|
has_many :wiki_references, :order => 'referenced_name'
|
2005-08-15 19:17:32 +00:00
|
|
|
has_one :current_revision, :class_name => 'Revision', :order => 'id DESC'
|
2005-09-09 20:40:35 +00:00
|
|
|
|
2005-09-10 06:12:57 +00:00
|
|
|
def revise(content, time, author, renderer)
|
2005-08-15 19:17:32 +00:00
|
|
|
revisions_size = new_record? ? 0 : revisions.size
|
|
|
|
if (revisions_size > 0) and content == current_revision.content
|
|
|
|
raise Instiki::ValidationError.new(
|
|
|
|
"You have tried to save page '#{name}' without changing its content")
|
|
|
|
end
|
|
|
|
|
|
|
|
author = Author.new(author.to_s) unless author.is_a?(Author)
|
|
|
|
|
|
|
|
# Try to render content to make sure that markup engine can take it,
|
2005-09-10 06:12:57 +00:00
|
|
|
renderer.revision = Revision.new(
|
2005-09-11 16:49:08 +00:00
|
|
|
:page => self, :content => content, :author => author, :revised_at => time)
|
2005-09-11 05:44:34 +00:00
|
|
|
renderer.display_content
|
2005-08-15 19:17:32 +00:00
|
|
|
|
|
|
|
# A user may change a page, look at it and make some more changes - several times.
|
|
|
|
# Not to record every such iteration as a new revision, if the previous revision was done
|
|
|
|
# by the same author, not more than 30 minutes ago, then update the last revision instead of
|
|
|
|
# creating a new one
|
|
|
|
if (revisions_size > 0) && continous_revision?(time, author)
|
|
|
|
current_revision.update_attributes(:content => content, :revised_at => time)
|
|
|
|
else
|
2005-09-11 16:49:08 +00:00
|
|
|
revisions.create(:content => content, :author => author, :revised_at => time)
|
2005-08-15 19:17:32 +00:00
|
|
|
end
|
|
|
|
save
|
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2005-09-10 06:12:57 +00:00
|
|
|
def rollback(revision_number, time, author_ip, renderer)
|
2005-08-15 19:17:32 +00:00
|
|
|
roll_back_revision = self.revisions[revision_number]
|
|
|
|
if roll_back_revision.nil?
|
|
|
|
raise Instiki::ValidationError.new("Revision #{revision_number} not found")
|
|
|
|
end
|
2005-09-10 06:12:57 +00:00
|
|
|
author = Author.new(roll_back_revision.author.name, author_ip)
|
|
|
|
revise(roll_back_revision.content, time, author, renderer)
|
2005-08-15 19:17:32 +00:00
|
|
|
end
|
2005-09-10 06:12:57 +00:00
|
|
|
|
2005-08-15 19:17:32 +00:00
|
|
|
def revisions?
|
|
|
|
revisions.size > 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def previous_revision(revision)
|
|
|
|
revision_index = revisions.each_with_index do |rev, index|
|
|
|
|
if rev.id == revision.id
|
|
|
|
break index
|
|
|
|
else
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
if revision_index.nil? or revision_index == 0
|
|
|
|
nil
|
|
|
|
else
|
|
|
|
revisions[revision_index - 1]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def authors
|
|
|
|
revisions.collect { |rev| rev.author }
|
|
|
|
end
|
|
|
|
|
|
|
|
def references
|
|
|
|
web.select.pages_that_reference(name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def linked_from
|
|
|
|
web.select.pages_that_link_to(name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def included_from
|
|
|
|
web.select.pages_that_include(name)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Returns the original wiki-word name as separate words, so "MyPage" becomes "My Page".
|
|
|
|
def plain_name
|
|
|
|
web.brackets_only? ? name : WikiWords.separate(name)
|
|
|
|
end
|
|
|
|
|
|
|
|
LOCKING_PERIOD = 30.minutes
|
|
|
|
|
|
|
|
def lock(time, locked_by)
|
|
|
|
update_attributes(:locked_at => time, :locked_by => locked_by)
|
|
|
|
end
|
|
|
|
|
|
|
|
def lock_duration(time)
|
|
|
|
((time - locked_at) / 60).to_i unless locked_at.nil?
|
|
|
|
end
|
|
|
|
|
|
|
|
def unlock
|
|
|
|
update_attribute(:locked_at, nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
def locked?(comparison_time)
|
|
|
|
locked_at + LOCKING_PERIOD > comparison_time unless locked_at.nil?
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def continous_revision?(time, author)
|
|
|
|
(current_revision.author == author) && (revised_at + 30.minutes > time)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Forward method calls to the current revision, so the page responds to all revision calls
|
|
|
|
def method_missing(method_id, *args, &block)
|
|
|
|
method_name = method_id.to_s
|
|
|
|
# Perform a hand-off to AR::Base#method_missing
|
|
|
|
if @attributes.include?(method_name) or md = /(=|\?|_before_type_cast)$/.match(method_name)
|
|
|
|
super(method_id, *args, &block)
|
|
|
|
else
|
|
|
|
current_revision.send(method_id)
|
|
|
|
end
|
|
|
|
end
|
2005-08-09 02:20:28 +00:00
|
|
|
end
|