Class: Gitlab::Conflict::File
- Inherits:
-
Object
- Object
- Gitlab::Conflict::File
- Includes:
- Routing, Utils::StrongMemoize, IconsHelper
- Defined in:
- lib/gitlab/conflict/file.rb
Constant Summary collapse
- CONTEXT_LINES =
3
Constants included from IconsHelper
IconsHelper::DEFAULT_ICON_SIZE
Instance Attribute Summary collapse
-
#merge_request ⇒ Object
readonly
Returns the value of attribute merge_request.
-
#raw ⇒ Object
readonly
'raw' holds the Gitlab::Git::Conflict::File that this instance wraps.
Instance Method Summary collapse
- #as_json(opts = {}) ⇒ Object
- #content_path ⇒ Object
- #create_match_line(line) ⇒ Object
-
#find_match_line_header(index) ⇒ Object
Any line beginning with a letter, an underscore, or a dollar can be used in a match line header.
- #highlight_lines! ⇒ Object
-
#initialize(raw, merge_request:) ⇒ File
constructor
A new instance of File.
- #line_code(line) ⇒ Object
- #lines ⇒ Object
- #resolve_lines(resolution) ⇒ Object
- #sections ⇒ Object
-
#update_match_line_text(match_line, line) ⇒ Object
Set the match line's text for the current line.
Methods included from Utils::StrongMemoize
#clear_memoization, #strong_memoize, #strong_memoized?
Methods included from IconsHelper
#audit_icon, #boolean_to_icon, #custom_icon, #external_snippet_icon, #file_type_icon_class, #icon, #loading_icon, #spinner, #sprite_file_icons_path, #sprite_icon, #sprite_icon_path, #visibility_level_icon
Methods included from Routing
add_helpers, includes_helpers, redirect_legacy_paths, url_helpers
Constructor Details
#initialize(raw, merge_request:) ⇒ File
Returns a new instance of File.
19 20 21 22 23 |
# File 'lib/gitlab/conflict/file.rb', line 19 def initialize(raw, merge_request:) @raw = raw @merge_request = merge_request @match_line_headers = {} end |
Instance Attribute Details
#merge_request ⇒ Object (readonly)
Returns the value of attribute merge_request
12 13 14 |
# File 'lib/gitlab/conflict/file.rb', line 12 def merge_request @merge_request end |
#raw ⇒ Object (readonly)
'raw' holds the Gitlab::Git::Conflict::File that this instance wraps
15 16 17 |
# File 'lib/gitlab/conflict/file.rb', line 15 def raw @raw end |
Instance Method Details
#as_json(opts = {}) ⇒ Object
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 |
# File 'lib/gitlab/conflict/file.rb', line 150 def as_json(opts = {}) json_hash = { old_path: their_path, new_path: our_path, blob_icon: file_type_icon_class('file', our_mode, our_path), blob_path: project_blob_path(merge_request.project, ::File.join(merge_request.diff_refs.head_sha, our_path)) } json_hash.tap do |json_hash| if opts[:full_content] json_hash[:content] = content else json_hash[:sections] = sections if type.text? json_hash[:type] = type json_hash[:content_path] = content_path end end end |
#content_path ⇒ Object
169 170 171 172 173 174 |
# File 'lib/gitlab/conflict/file.rb', line 169 def content_path conflict_for_path_project_merge_request_path(merge_request.project, merge_request, old_path: their_path, new_path: our_path) end |
#create_match_line(line) ⇒ Object
116 117 118 |
# File 'lib/gitlab/conflict/file.rb', line 116 def create_match_line(line) Gitlab::Diff::Line.new('', 'match', line.index, line.old_pos, line.new_pos) end |
#find_match_line_header(index) ⇒ Object
Any line beginning with a letter, an underscore, or a dollar can be used in a match line header. Only context sections can contain match lines, as match lines have to exist in both versions of the file.
123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 |
# File 'lib/gitlab/conflict/file.rb', line 123 def find_match_line_header(index) return @match_line_headers[index] if @match_line_headers.key?(index) @match_line_headers[index] = begin if index >= 0 line = lines[index] if line.type.nil? && line.text.match(/\A[A-Za-z$_]/) " #{line.text}" else find_match_line_header(index - 1) end end end end |
#highlight_lines! ⇒ Object
35 36 37 38 39 40 41 42 43 44 45 46 47 |
# File 'lib/gitlab/conflict/file.rb', line 35 def highlight_lines! their_highlight = Gitlab::Highlight.highlight(their_path, their_lines, language: their_language).lines our_highlight = Gitlab::Highlight.highlight(our_path, our_lines, language: our_language).lines lines.each do |line| line.rich_text = if line.type == 'old' their_highlight[line.old_line - 1].try(:html_safe) else our_highlight[line.new_line - 1].try(:html_safe) end end end |
#line_code(line) ⇒ Object
112 113 114 |
# File 'lib/gitlab/conflict/file.rb', line 112 def line_code(line) Gitlab::Git.diff_line_code(our_path, line.new_pos, line.old_pos) end |
#lines ⇒ Object
25 26 27 28 29 |
# File 'lib/gitlab/conflict/file.rb', line 25 def lines return @lines if defined?(@lines) @lines = raw.lines.nil? ? nil : map_raw_lines(raw.lines) end |
#resolve_lines(resolution) ⇒ Object
31 32 33 |
# File 'lib/gitlab/conflict/file.rb', line 31 def resolve_lines(resolution) map_raw_lines(raw.resolve_lines(resolution)) end |
#sections ⇒ Object
49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 |
# File 'lib/gitlab/conflict/file.rb', line 49 def sections return @sections if @sections chunked_lines = lines.chunk { |line| line.type.nil? }.to_a match_line = nil sections_count = chunked_lines.size @sections = chunked_lines.flat_map.with_index do |(no_conflict, lines), i| section = nil # We need to reduce context sections to CONTEXT_LINES. Conflict sections are # always shown in full. if no_conflict conflict_before = i > 0 conflict_after = (sections_count - i) > 1 if conflict_before && conflict_after # Create a gap in a long context section. if lines.length > CONTEXT_LINES * 2 head_lines = lines.first(CONTEXT_LINES) tail_lines = lines.last(CONTEXT_LINES) # Ensure any existing match line has text for all lines up to the last # line of its context. update_match_line_text(match_line, head_lines.last) # Insert a new match line after the created gap. match_line = create_match_line(tail_lines.first) section = [ { conflict: false, lines: head_lines }, { conflict: false, lines: tail_lines.unshift(match_line) } ] end elsif conflict_after tail_lines = lines.last(CONTEXT_LINES) # Create a gap and insert a match line at the start. if lines.length > tail_lines.length match_line = create_match_line(tail_lines.first) tail_lines.unshift(match_line) end lines = tail_lines elsif conflict_before # We're at the end of the file (no conflicts after), so just remove extra # trailing lines. lines = lines.first(CONTEXT_LINES) end end # We want to update the match line's text every time unless we've already # created a gap and its corresponding match line. update_match_line_text(match_line, lines.last) unless section section ||= { conflict: !no_conflict, lines: lines } section[:id] = line_code(lines.first) unless no_conflict section end end |
#update_match_line_text(match_line, line) ⇒ Object
Set the match line's text for the current line. A match line takes its start position and context header (where present) from itself, and its end position from the line passed in.
142 143 144 145 146 147 148 |
# File 'lib/gitlab/conflict/file.rb', line 142 def update_match_line_text(match_line, line) return unless match_line header = find_match_line_header(match_line.index - 1) match_line.text = "@@ -#{match_line.old_pos},#{line.old_pos} +#{match_line.new_pos},#{line.new_pos} @@#{header}" end |