socialforge/lib/trustie/gitlab/inline_diff.rb

107 lines
3.4 KiB
Ruby
Raw Normal View History

2016-11-26 17:47:39 +08:00
module Trustie
module Gitlab
class InlineDiff
class << self
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
START = "#!idiff-start!#"
FINISH = "#!idiff-finish!#"
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
def processing(diff_arr)
# indexes = _indexes_of_changed_lines diff_arr
#
# indexes.each do |index|
# first_line = diff_arr[index+1]
# second_line = diff_arr[index+2]
#
# # Skip inline diff if empty line was replaced with content
# next if first_line == "-\n"
#
# first_token = find_first_token(first_line, second_line)
# apply_first_token(diff_arr, index, first_token)
#
# last_token = find_last_token(first_line, second_line, first_token)
# apply_last_token(diff_arr, index, last_token)
# end
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
diff_arr
end
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
def apply_first_token(diff_arr, index, first_token)
start = first_token + START
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
if first_token.empty?
# In case if we remove string of spaces in commit
diff_arr[index+1].sub!("-", "-" => "-#{START}")
diff_arr[index+2].sub!("+", "+" => "+#{START}")
else
diff_arr[index+1].sub!(first_token, first_token => start)
diff_arr[index+2].sub!(first_token, first_token => start)
end
2016-11-26 17:06:17 +08:00
end
2016-11-26 17:47:39 +08:00
def apply_last_token(diff_arr, index, last_token)
# This is tricky: escape backslashes so that `sub` doesn't interpret them
# as backreferences. Regexp.escape does NOT do the right thing.
replace_token = FINISH + last_token.gsub(/\\/, '\&\&')
diff_arr[index+1].sub!(/#{Regexp.escape(last_token)}$/, replace_token)
diff_arr[index+2].sub!(/#{Regexp.escape(last_token)}$/, replace_token)
end
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
def find_first_token(first_line, second_line)
max_length = [first_line.size, second_line.size].max
first_the_same_symbols = 0
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
(0..max_length + 1).each do |i|
first_the_same_symbols = i - 1
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
if first_line[i] != second_line[i] && i > 0
break
end
2016-11-26 17:06:17 +08:00
end
2016-11-26 17:47:39 +08:00
first_line[0..first_the_same_symbols][1..-1]
end
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
def find_last_token(first_line, second_line, first_token)
max_length = [first_line.size, second_line.size].max
last_the_same_symbols = 0
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
(1..max_length + 1).each do |i|
last_the_same_symbols = -i
shortest_line = second_line.size > first_line.size ? first_line : second_line
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
if (first_line[-i] != second_line[-i]) || "#{first_token}#{START}".size == shortest_line[1..-i].size
break
end
2016-11-26 17:06:17 +08:00
end
2016-11-26 17:47:39 +08:00
last_the_same_symbols += 1
first_line[last_the_same_symbols..-1]
2016-11-26 17:06:17 +08:00
end
2016-11-26 17:47:39 +08:00
def _indexes_of_changed_lines(diff_arr)
chain_of_first_symbols = ""
diff_arr.each_with_index do |line, i|
chain_of_first_symbols += line[0]
end
chain_of_first_symbols.gsub!(/[^\-\+]/, "#")
2016-11-26 17:06:17 +08:00
2016-11-26 17:47:39 +08:00
offset = 0
indexes = []
while index = chain_of_first_symbols.index("#-+#", offset)
indexes << index
offset = index + 1
end
indexes
2016-11-26 17:06:17 +08:00
end
2016-11-26 17:47:39 +08:00
def replace_markers(line)
line.gsub!(START, "<span class='idiff'>")
line.gsub!(FINISH, "</span>")
line
2016-11-26 17:06:17 +08:00
end
end
end
end
end