require "diff-lcs"
require "hyp_diff"
module Scrivito
class Diff; class << self
def for(mode, field_type, old, new)
case field_type
when "widget"
ListComparison.for(mode, old || [], new || [])
when "html"
for_html(mode, old, new)
when "text"
for_string(mode, old, new)
when "string"
for_string(mode, old, new)
end
end
private
def for_string(mode, old, new)
for_html(mode, ERB::Util.h(old), ERB::Util.h(new))
end
def for_html(mode, old, new)
return if !old
return if !new
return unless %w[added deleted diff].include?(mode)
diff = HypDiff.compare(old, new, hyp_diff_options(mode))
[StringTagging.tag_as_html(diff)]
end
def hyp_diff_options(mode)
render_insertion_proc = proc { |html| "#{html}" }
render_deletion_proc = proc { |html| "#{html}" }
render_nothing_proc = proc {}
case mode
when 'added'
{
render_insertion: render_insertion_proc,
render_deletion: render_nothing_proc,
markup_from: 'after'
}
when 'deleted'
{
render_insertion: render_nothing_proc,
render_deletion: render_deletion_proc,
markup_from: 'before'
}
when 'diff'
{
render_insertion: render_insertion_proc,
render_deletion: render_deletion_proc
}
end
end
class ListComparison
def self.for(mode, old, new)
new(mode).compare(old, new).result
end
def initialize(mode)
@mode = mode
@list = []
@modifications = []
end
def compare(old, new)
::Diff::LCS.sdiff(old, new).map do |change|
case change.action
when "-" then
note_deletion(change)
when "+" then
note_insertion(change)
when "=" then
note_equal(change)
when "!" then
note_deletion(change)
note_insertion(change)
else
raise "unexpected change.action #{change.action}"
end
end
self
end
def result
[@list, @modifications]
end
private
def note_equal(change)
@modifications << nil
if @mode == "deleted"
@list << change.old_element
else
@list << change.new_element
end
end
def note_deletion(change)
unless @mode == "added"
@modifications << "deleted"
@list << change.old_element
end
end
def note_insertion(change)
unless @mode == "deleted"
@modifications << "new"
@list << change.new_element
end
end
end
end; end
end