blob: 0ac39f00a146a22157fecdf38a05e3f71d028521 [file] [log] [blame]
Dave Borowitz8cdc76b2018-03-26 10:04:27 -04001/**
2 * @license
3 * Copyright (C) 2016 The Android Open Source Project
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 */
Ben Rohlfs32b83822020-08-14 22:08:37 +020017import {GestureEventListeners} from '@polymer/polymer/lib/mixins/gesture-event-listeners';
18import {LegacyElementMixin} from '@polymer/polymer/lib/legacy/legacy-element-mixin';
19import {PolymerElement} from '@polymer/polymer/polymer-element';
20import {
21 GrDiffLine,
22 GrDiffLineType,
23 FILE,
24 Highlights,
25} from '../gr-diff/gr-diff-line';
26import {
27 GrDiffGroup,
28 GrDiffGroupType,
29 hideInContextControl,
30} from '../gr-diff/gr-diff-group';
31import {CancelablePromise, util} from '../../../scripts/util';
32import {customElement, property} from '@polymer/decorators';
33import {DiffContent} from '../../../types/common';
34import {DiffSide} from '../gr-diff/gr-diff-utils';
Wyatt Allen7f2bd972016-06-27 12:19:21 -070035
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +010036const WHOLE_FILE = -1;
Wyatt Allen7f2bd972016-06-27 12:19:21 -070037
Ben Rohlfs32b83822020-08-14 22:08:37 +020038interface State {
39 lineNums: {
40 left: number;
41 right: number;
42 };
43 chunkIndex: number;
44}
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +010045
Ben Rohlfs32b83822020-08-14 22:08:37 +020046interface ChunkEnd {
47 offset: number;
48 keyLocation: boolean;
49}
50
Ben Rohlfs4fa7c532020-08-24 18:16:13 +020051export interface KeyLocations {
Ben Rohlfs32b83822020-08-14 22:08:37 +020052 left: {[key: string]: boolean};
53 right: {[key: string]: boolean};
54}
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +010055
56/**
57 * The maximum size for an addition or removal chunk before it is broken down
58 * into a series of chunks that are this size at most.
59 *
60 * Note: The value of 120 is chosen so that it is larger than the default
61 * _asyncThreshold of 64, but feel free to tune this constant to your
62 * performance needs.
63 */
64const MAX_GROUP_SIZE = 120;
65
66/**
67 * Converts the API's `DiffContent`s to `GrDiffGroup`s for rendering.
68 *
69 * Glossary:
70 * - "chunk": A single `DiffContent` as returned by the API.
71 * - "group": A single `GrDiffGroup` as used for rendering.
72 * - "common" chunk/group: A chunk/group that should be considered unchanged
73 * for diffing purposes. This can mean its either actually unchanged, or it
74 * has only whitespace changes.
75 * - "key location": A line number and side of the diff that should not be
76 * collapsed e.g. because a comment is attached to it, or because it was
77 * provided in the URL and thus should be visible
78 * - "uncollapsible" chunk/group: A chunk/group that is either not "common",
79 * or cannot be collapsed because it contains a key location
80 *
81 * Here a a number of tasks this processor performs:
82 * - splitting large chunks to allow more granular async rendering
83 * - adding a group for the "File" pseudo line that file-level comments can
84 * be attached to
85 * - replacing common parts of the diff that are outside the user's
86 * context setting and do not have comments with a group representing the
87 * "expand context" widget. This may require splitting a chunk/group so
88 * that the part that is within the context or has comments is shown, while
89 * the rest is not.
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +010090 */
Ben Rohlfs32b83822020-08-14 22:08:37 +020091@customElement('gr-diff-processor')
92export class GrDiffProcessor extends GestureEventListeners(
93 LegacyElementMixin(PolymerElement)
94) {
95 @property({type: Number})
96 context = 3;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +010097
Ben Rohlfs32b83822020-08-14 22:08:37 +020098 @property({type: Array, notify: true})
99 groups: GrDiffGroup[] = [];
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100100
Ben Rohlfs32b83822020-08-14 22:08:37 +0200101 @property({type: Object})
102 keyLocations: KeyLocations = {left: {}, right: {}};
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100103
Ben Rohlfs32b83822020-08-14 22:08:37 +0200104 @property({type: Number})
105 _asyncThreshold = 64;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100106
Ben Rohlfs32b83822020-08-14 22:08:37 +0200107 @property({type: Number})
108 _nextStepHandle: number | null = null;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100109
Ben Rohlfs32b83822020-08-14 22:08:37 +0200110 @property({type: Object})
111 _processPromise: CancelablePromise<void> | null = null;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100112
Ben Rohlfs32b83822020-08-14 22:08:37 +0200113 @property({type: Boolean})
114 _isScrolling?: boolean;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100115
116 /** @override */
117 attached() {
118 super.attached();
119 this.listen(window, 'scroll', '_handleWindowScroll');
120 }
121
122 /** @override */
123 detached() {
124 super.detached();
125 this.cancel();
126 this.unlisten(window, 'scroll', '_handleWindowScroll');
127 }
128
129 _handleWindowScroll() {
130 this._isScrolling = true;
Ben Rohlfs32b83822020-08-14 22:08:37 +0200131 this.debounce(
132 'resetIsScrolling',
133 () => {
134 this._isScrolling = false;
135 },
136 50
137 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100138 }
Renan Oliveiraa132fae2019-09-19 14:38:22 +0000139
Wyatt Allen32b03fc2016-08-05 15:56:33 -0700140 /**
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100141 * Asynchronously process the diff chunks into groups. As it processes, it
142 * will splice groups into the `groups` property of the component.
Wyatt Allen32b03fc2016-08-05 15:56:33 -0700143 *
Ben Rohlfs32b83822020-08-14 22:08:37 +0200144 * @return A promise that resolves with an
145 * array of GrDiffGroups when the diff is completely processed.
Wyatt Allen32b03fc2016-08-05 15:56:33 -0700146 */
Ben Rohlfs32b83822020-08-14 22:08:37 +0200147 process(chunks: DiffContent[], isBinary: boolean) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100148 // Cancel any still running process() calls, because they append to the
149 // same groups field.
150 this.cancel();
Wyatt Allen32b03fc2016-08-05 15:56:33 -0700151
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100152 this.groups = [];
153 this.push('groups', this._makeFileComments());
Wyatt Allen7f2bd972016-06-27 12:19:21 -0700154
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100155 // If it's a binary diff, we won't be rendering hunks of text differences
156 // so finish processing.
Ben Rohlfs32b83822020-08-14 22:08:37 +0200157 if (isBinary) {
158 return Promise.resolve();
159 }
Wyatt Allen7f2bd972016-06-27 12:19:21 -0700160
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100161 this._processPromise = util.makeCancelable(
Ben Rohlfs32b83822020-08-14 22:08:37 +0200162 new Promise(resolve => {
163 const state = {
164 lineNums: {left: 0, right: 0},
165 chunkIndex: 0,
166 };
Wyatt Allen7f2bd972016-06-27 12:19:21 -0700167
Ben Rohlfs32b83822020-08-14 22:08:37 +0200168 chunks = this._splitLargeChunks(chunks);
169 chunks = this._splitCommonChunksWithKeyLocations(chunks);
Wyatt Allen7f2bd972016-06-27 12:19:21 -0700170
Ben Rohlfs32b83822020-08-14 22:08:37 +0200171 let currentBatch = 0;
172 const nextStep = () => {
173 if (this._isScrolling) {
174 this._nextStepHandle = this.async(nextStep, 100);
175 return;
176 }
177 // If we are done, resolve the promise.
178 if (state.chunkIndex >= chunks.length) {
179 resolve();
180 this._nextStepHandle = null;
181 return;
182 }
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100183
Ben Rohlfs32b83822020-08-14 22:08:37 +0200184 // Process the next chunk and incorporate the result.
185 const stateUpdate = this._processNext(state, chunks);
186 for (const group of stateUpdate.groups) {
187 this.push('groups', group);
188 currentBatch += group.lines.length;
189 }
190 state.lineNums.left += stateUpdate.lineDelta.left;
191 state.lineNums.right += stateUpdate.lineDelta.right;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100192
Ben Rohlfs32b83822020-08-14 22:08:37 +0200193 // Increment the index and recurse.
194 state.chunkIndex = stateUpdate.newChunkIndex;
195 if (currentBatch >= this._asyncThreshold) {
196 currentBatch = 0;
197 this._nextStepHandle = this.async(nextStep, 1);
198 } else {
199 nextStep.call(this);
200 }
201 };
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100202
Ben Rohlfs32b83822020-08-14 22:08:37 +0200203 nextStep.call(this);
204 })
205 );
206 return this._processPromise.finally(() => {
207 this._processPromise = null;
208 });
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100209 }
210
211 /**
212 * Cancel any jobs that are running.
213 */
214 cancel() {
Ben Rohlfs32b83822020-08-14 22:08:37 +0200215 if (this._nextStepHandle !== null) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100216 this.cancelAsync(this._nextStepHandle);
217 this._nextStepHandle = null;
Dmitrii Filippov3fd2b102019-11-15 16:16:46 +0100218 }
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100219 if (this._processPromise) {
220 this._processPromise.cancel();
Dmitrii Filippov3fd2b102019-11-15 16:16:46 +0100221 }
222 }
223
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100224 /**
225 * Process the next uncollapsible chunk, or the next collapsible chunks.
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100226 */
Ben Rohlfs32b83822020-08-14 22:08:37 +0200227 _processNext(state: State, chunks: DiffContent[]) {
228 const firstUncollapsibleChunkIndex = this._firstUncollapsibleChunkIndex(
229 chunks,
230 state.chunkIndex
231 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100232 if (firstUncollapsibleChunkIndex === state.chunkIndex) {
233 const chunk = chunks[state.chunkIndex];
234 return {
235 lineDelta: {
236 left: this._linesLeft(chunk).length,
237 right: this._linesRight(chunk).length,
238 },
Ben Rohlfs32b83822020-08-14 22:08:37 +0200239 groups: [
240 this._chunkToGroup(
241 chunk,
242 state.lineNums.left + 1,
243 state.lineNums.right + 1
244 ),
245 ],
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100246 newChunkIndex: state.chunkIndex + 1,
247 };
248 }
249
250 return this._processCollapsibleChunks(
Ben Rohlfs32b83822020-08-14 22:08:37 +0200251 state,
252 chunks,
253 firstUncollapsibleChunkIndex
254 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100255 }
256
Ben Rohlfs32b83822020-08-14 22:08:37 +0200257 _linesLeft(chunk: DiffContent) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100258 return chunk.ab || chunk.a || [];
259 }
260
Ben Rohlfs32b83822020-08-14 22:08:37 +0200261 _linesRight(chunk: DiffContent) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100262 return chunk.ab || chunk.b || [];
263 }
264
Ben Rohlfs32b83822020-08-14 22:08:37 +0200265 _firstUncollapsibleChunkIndex(chunks: DiffContent[], offset: number) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100266 let chunkIndex = offset;
Ben Rohlfs32b83822020-08-14 22:08:37 +0200267 while (
268 chunkIndex < chunks.length &&
269 this._isCollapsibleChunk(chunks[chunkIndex])
270 ) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100271 chunkIndex++;
272 }
273 return chunkIndex;
274 }
275
Ben Rohlfs32b83822020-08-14 22:08:37 +0200276 _isCollapsibleChunk(chunk: DiffContent) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100277 return (chunk.ab || chunk.common) && !chunk.keyLocation;
278 }
279
280 /**
281 * Process a stretch of collapsible chunks.
282 *
283 * Outputs up to three groups:
Ben Rohlfs32b83822020-08-14 22:08:37 +0200284 * 1) Visible context before the hidden common code, unless it's the
285 * very beginning of the file.
286 * 2) Context hidden behind a context bar, unless empty.
287 * 3) Visible context after the hidden common code, unless it's the very
288 * end of the file.
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100289 */
290 _processCollapsibleChunks(
Ben Rohlfs32b83822020-08-14 22:08:37 +0200291 state: State,
292 chunks: DiffContent[],
293 firstUncollapsibleChunkIndex: number
294 ) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100295 const collapsibleChunks = chunks.slice(
Ben Rohlfs32b83822020-08-14 22:08:37 +0200296 state.chunkIndex,
297 firstUncollapsibleChunkIndex
298 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100299 const lineCount = collapsibleChunks.reduce(
Ben Rohlfs32b83822020-08-14 22:08:37 +0200300 (sum, chunk) => sum + this._commonChunkLength(chunk),
301 0
302 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100303
304 let groups = this._chunksToGroups(
Ben Rohlfs32b83822020-08-14 22:08:37 +0200305 collapsibleChunks,
306 state.lineNums.left + 1,
307 state.lineNums.right + 1
308 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100309
310 if (this.context !== WHOLE_FILE) {
311 const hiddenStart = state.chunkIndex === 0 ? 0 : this.context;
Ben Rohlfs32b83822020-08-14 22:08:37 +0200312 const hiddenEnd =
313 lineCount -
314 (firstUncollapsibleChunkIndex === chunks.length ? 0 : this.context);
315 groups = hideInContextControl(groups, hiddenStart, hiddenEnd);
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100316 }
317
318 return {
319 lineDelta: {
320 left: lineCount,
321 right: lineCount,
322 },
323 groups,
324 newChunkIndex: firstUncollapsibleChunkIndex,
325 };
326 }
327
Ben Rohlfs32b83822020-08-14 22:08:37 +0200328 _commonChunkLength(chunk: DiffContent) {
329 console.assert(!!chunk.ab || !!chunk.common);
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100330 console.assert(
Ben Rohlfs32b83822020-08-14 22:08:37 +0200331 !chunk.a || (!!chunk.b && chunk.a.length === chunk.b.length),
332 'common chunk needs same number of a and b lines: ',
333 chunk
334 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100335 return this._linesLeft(chunk).length;
336 }
337
Ben Rohlfs32b83822020-08-14 22:08:37 +0200338 _chunksToGroups(
339 chunks: DiffContent[],
340 offsetLeft: number,
341 offsetRight: number
342 ): GrDiffGroup[] {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100343 return chunks.map(chunk => {
344 const group = this._chunkToGroup(chunk, offsetLeft, offsetRight);
345 const chunkLength = this._commonChunkLength(chunk);
346 offsetLeft += chunkLength;
347 offsetRight += chunkLength;
348 return group;
349 });
350 }
351
Ben Rohlfs32b83822020-08-14 22:08:37 +0200352 _chunkToGroup(
353 chunk: DiffContent,
354 offsetLeft: number,
355 offsetRight: number
356 ): GrDiffGroup {
Ben Rohlfs5e2d1e72020-08-03 19:33:31 +0200357 const type = chunk.ab ? GrDiffGroupType.BOTH : GrDiffGroupType.DELTA;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100358 const lines = this._linesFromChunk(chunk, offsetLeft, offsetRight);
359 const group = new GrDiffGroup(type, lines);
Ben Rohlfs32b83822020-08-14 22:08:37 +0200360 group.keyLocation = !!chunk.keyLocation;
361 group.dueToRebase = !!chunk.due_to_rebase;
362 group.ignoredWhitespaceOnly = !!chunk.common;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100363 return group;
364 }
365
Ben Rohlfs32b83822020-08-14 22:08:37 +0200366 _linesFromChunk(chunk: DiffContent, offsetLeft: number, offsetRight: number) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100367 if (chunk.ab) {
Ben Rohlfs32b83822020-08-14 22:08:37 +0200368 return chunk.ab.map((row, i) =>
369 this._lineFromRow(GrDiffLineType.BOTH, offsetLeft, offsetRight, row, i)
370 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100371 }
Ben Rohlfs32b83822020-08-14 22:08:37 +0200372 let lines: GrDiffLine[] = [];
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100373 if (chunk.a) {
374 // Avoiding a.push(...b) because that causes callstack overflows for
375 // large b, which can occur when large files are added removed.
Ben Rohlfs32b83822020-08-14 22:08:37 +0200376 lines = lines.concat(
377 this._linesFromRows(
378 GrDiffLineType.REMOVE,
379 chunk.a,
380 offsetLeft,
381 chunk.edit_a
382 )
383 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100384 }
385 if (chunk.b) {
386 // Avoiding a.push(...b) because that causes callstack overflows for
387 // large b, which can occur when large files are added removed.
Ben Rohlfs32b83822020-08-14 22:08:37 +0200388 lines = lines.concat(
389 this._linesFromRows(
390 GrDiffLineType.ADD,
391 chunk.b,
392 offsetRight,
393 chunk.edit_b
394 )
395 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100396 }
397 return lines;
398 }
399
Ben Rohlfs32b83822020-08-14 22:08:37 +0200400 _linesFromRows(
401 lineType: GrDiffLineType,
402 rows: string[],
403 offset: number,
404 intralineInfos?: number[][]
405 ): GrDiffLine[] {
406 const grDiffHighlights = intralineInfos
407 ? this._convertIntralineInfos(rows, intralineInfos)
408 : undefined;
409 return rows.map((row, i) =>
410 this._lineFromRow(lineType, offset, offset, row, i, grDiffHighlights)
411 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100412 }
413
Ben Rohlfs32b83822020-08-14 22:08:37 +0200414 _lineFromRow(
415 type: GrDiffLineType,
416 offsetLeft: number,
417 offsetRight: number,
418 row: string,
419 i: number,
420 highlights: Highlights[] = []
421 ): GrDiffLine {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100422 const line = new GrDiffLine(type);
423 line.text = row;
Ben Rohlfs5e2d1e72020-08-03 19:33:31 +0200424 if (type !== GrDiffLineType.ADD) line.beforeNumber = offsetLeft + i;
425 if (type !== GrDiffLineType.REMOVE) line.afterNumber = offsetRight + i;
Ben Rohlfs32b83822020-08-14 22:08:37 +0200426 if (highlights) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100427 line.hasIntralineInfo = true;
Ben Rohlfs32b83822020-08-14 22:08:37 +0200428 line.highlights = highlights.filter(hl => hl.contentIndex === i);
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100429 } else {
430 line.hasIntralineInfo = false;
431 }
432 return line;
433 }
434
435 _makeFileComments() {
Ben Rohlfs5e2d1e72020-08-03 19:33:31 +0200436 const line = new GrDiffLine(GrDiffLineType.BOTH);
437 line.beforeNumber = FILE;
438 line.afterNumber = FILE;
439 return new GrDiffGroup(GrDiffGroupType.BOTH, [line]);
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100440 }
441
442 /**
443 * Split chunks into smaller chunks of the same kind.
444 *
445 * This is done to prevent doing too much work on the main thread in one
446 * uninterrupted rendering step, which would make the browser unresponsive.
447 *
448 * Note that in the case of unmodified chunks, we only split chunks if the
449 * context is set to file (because otherwise they are split up further down
450 * the processing into the visible and hidden context), and only split it
451 * into 2 chunks, one max sized one and the rest (for reasons that are
452 * unclear to me).
453 *
Ben Rohlfs32b83822020-08-14 22:08:37 +0200454 * @param chunks Chunks as returned from the server
455 * @return Finer grained chunks.
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100456 */
Ben Rohlfs32b83822020-08-14 22:08:37 +0200457 _splitLargeChunks(chunks: DiffContent[]): DiffContent[] {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100458 const newChunks = [];
459
460 for (const chunk of chunks) {
461 if (!chunk.ab) {
462 for (const subChunk of this._breakdownChunk(chunk)) {
463 newChunks.push(subChunk);
464 }
465 continue;
466 }
467
468 // If the context is set to "whole file", then break down the shared
469 // chunks so they can be rendered incrementally. Note: this is not
470 // enabled for any other context preference because manipulating the
471 // chunks in this way violates assumptions by the context grouper logic.
472 if (this.context === -1 && chunk.ab.length > MAX_GROUP_SIZE * 2) {
473 // Split large shared chunks in two, where the first is the maximum
474 // group size.
475 newChunks.push({ab: chunk.ab.slice(0, MAX_GROUP_SIZE)});
476 newChunks.push({ab: chunk.ab.slice(MAX_GROUP_SIZE)});
477 } else {
478 newChunks.push(chunk);
479 }
480 }
481 return newChunks;
482 }
483
484 /**
485 * In order to show key locations, such as comments, out of the bounds of
486 * the selected context, treat them as separate chunks within the model so
487 * that the content (and context surrounding it) renders correctly.
488 *
Ben Rohlfs32b83822020-08-14 22:08:37 +0200489 * @param chunks DiffContents as returned from server.
490 * @return Finer grained DiffContents.
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100491 */
Ben Rohlfs32b83822020-08-14 22:08:37 +0200492 _splitCommonChunksWithKeyLocations(chunks: DiffContent[]): DiffContent[] {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100493 const result = [];
494 let leftLineNum = 1;
495 let rightLineNum = 1;
496
497 for (const chunk of chunks) {
498 // If it isn't a common chunk, append it as-is and update line numbers.
499 if (!chunk.ab && !chunk.common) {
500 if (chunk.a) {
501 leftLineNum += chunk.a.length;
502 }
503 if (chunk.b) {
504 rightLineNum += chunk.b.length;
505 }
506 result.push(chunk);
507 continue;
508 }
509
Ben Rohlfs32b83822020-08-14 22:08:37 +0200510 if (chunk.common && chunk.a!.length !== chunk.b!.length) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100511 throw new Error(
Ben Rohlfs32b83822020-08-14 22:08:37 +0200512 'DiffContent with common=true must always have equal length'
513 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100514 }
515 const numLines = this._commonChunkLength(chunk);
516 const chunkEnds = this._findChunkEndsAtKeyLocations(
Ben Rohlfs32b83822020-08-14 22:08:37 +0200517 numLines,
518 leftLineNum,
519 rightLineNum
520 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100521 leftLineNum += numLines;
522 rightLineNum += numLines;
523
524 if (chunk.ab) {
Ben Rohlfs32b83822020-08-14 22:08:37 +0200525 result.push(
526 ...this._splitAtChunkEnds(chunk.ab, chunkEnds).map(
527 ({lines, keyLocation}) => {
Tao Zhou4cd35cb2020-07-22 11:28:22 +0200528 return {
529 ...chunk,
530 ab: lines,
531 keyLocation,
532 };
Ben Rohlfs32b83822020-08-14 22:08:37 +0200533 }
534 )
535 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100536 } else if (chunk.common) {
Ben Rohlfs32b83822020-08-14 22:08:37 +0200537 const aChunks = this._splitAtChunkEnds(chunk.a!, chunkEnds);
538 const bChunks = this._splitAtChunkEnds(chunk.b!, chunkEnds);
539 result.push(
540 ...aChunks.map(({lines, keyLocation}, i) => {
541 return {
542 ...chunk,
543 a: lines,
544 b: bChunks[i].lines,
545 keyLocation,
546 };
547 })
548 );
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100549 }
550 }
551
552 return result;
553 }
554
555 /**
Ben Rohlfs32b83822020-08-14 22:08:37 +0200556 * @return Offsets of the new chunk ends, including whether it's a key
557 * location.
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100558 */
Ben Rohlfs32b83822020-08-14 22:08:37 +0200559 _findChunkEndsAtKeyLocations(
560 numLines: number,
561 leftOffset: number,
562 rightOffset: number
563 ): ChunkEnd[] {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100564 const result = [];
565 let lastChunkEnd = 0;
Ben Rohlfs32b83822020-08-14 22:08:37 +0200566 for (let i = 0; i < numLines; i++) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100567 // If this line should not be collapsed.
Ben Rohlfs32b83822020-08-14 22:08:37 +0200568 if (
569 this.keyLocations[DiffSide.LEFT][leftOffset + i] ||
570 this.keyLocations[DiffSide.RIGHT][rightOffset + i]
571 ) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100572 // If any lines have been accumulated into the chunk leading up to
573 // this non-collapse line, then add them as a chunk and start a new
574 // one.
575 if (i > lastChunkEnd) {
576 result.push({offset: i, keyLocation: false});
577 lastChunkEnd = i;
578 }
579
580 // Add the non-collapse line as its own chunk.
581 result.push({offset: i + 1, keyLocation: true});
582 }
583 }
584
585 if (numLines > lastChunkEnd) {
586 result.push({offset: numLines, keyLocation: false});
587 }
588
589 return result;
590 }
591
Ben Rohlfs32b83822020-08-14 22:08:37 +0200592 _splitAtChunkEnds(lines: string[], chunkEnds: ChunkEnd[]) {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100593 const result = [];
594 let lastChunkEndOffset = 0;
595 for (const {offset, keyLocation} of chunkEnds) {
Ben Rohlfs32b83822020-08-14 22:08:37 +0200596 result.push({
597 lines: lines.slice(lastChunkEndOffset, offset),
598 keyLocation,
599 });
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100600 lastChunkEndOffset = offset;
601 }
602 return result;
603 }
604
605 /**
606 * Converts `IntralineInfo`s return by the API to `GrLineHighlights` used
607 * for rendering.
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100608 */
Ben Rohlfs32b83822020-08-14 22:08:37 +0200609 _convertIntralineInfos(
610 rows: string[],
611 intralineInfos: number[][]
612 ): Highlights[] {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100613 let rowIndex = 0;
614 let idx = 0;
615 const normalized = [];
616 for (const [skipLength, markLength] of intralineInfos) {
617 let line = rows[rowIndex] + '\n';
618 let j = 0;
619 while (j < skipLength) {
620 if (idx === line.length) {
621 idx = 0;
622 line = rows[++rowIndex] + '\n';
623 continue;
624 }
625 idx++;
626 j++;
627 }
Ben Rohlfs32b83822020-08-14 22:08:37 +0200628 let lineHighlight: Highlights = {
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100629 contentIndex: rowIndex,
630 startIndex: idx,
631 };
632
633 j = 0;
634 while (line && j < markLength) {
635 if (idx === line.length) {
636 idx = 0;
637 line = rows[++rowIndex] + '\n';
638 normalized.push(lineHighlight);
639 lineHighlight = {
640 contentIndex: rowIndex,
641 startIndex: idx,
642 };
643 continue;
644 }
645 idx++;
646 j++;
647 }
648 lineHighlight.endIndex = idx;
649 normalized.push(lineHighlight);
650 }
651 return normalized;
652 }
653
654 /**
655 * If a group is an addition or a removal, break it down into smaller groups
656 * of that type using the MAX_GROUP_SIZE. If the group is a shared chunk
657 * or a delta it is returned as the single element of the result array.
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100658 */
Ben Rohlfs32b83822020-08-14 22:08:37 +0200659 _breakdownChunk(chunk: DiffContent): DiffContent[] {
660 let key: 'a' | 'b' | 'ab' | null = null;
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100661 if (chunk.a && !chunk.b) {
662 key = 'a';
663 } else if (chunk.b && !chunk.a) {
664 key = 'b';
665 } else if (chunk.ab) {
666 key = 'ab';
667 }
668
Ben Rohlfs32b83822020-08-14 22:08:37 +0200669 if (!key) {
670 return [chunk];
671 }
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100672
Ben Rohlfs32b83822020-08-14 22:08:37 +0200673 return this._breakdown(chunk[key]!, MAX_GROUP_SIZE).map(subChunkLines => {
674 const subChunk: DiffContent = {};
675 subChunk[key!] = subChunkLines;
676 if (chunk.due_to_rebase) {
677 subChunk.due_to_rebase = true;
678 }
679 return subChunk;
680 });
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100681 }
682
683 /**
684 * Given an array and a size, return an array of arrays where no inner array
685 * is larger than that size, preserving the original order.
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100686 */
Ben Rohlfs32b83822020-08-14 22:08:37 +0200687 _breakdown<T>(array: T[], size: number): T[][] {
688 if (!array.length) {
689 return [];
690 }
691 if (array.length < size) {
692 return [array];
693 }
Dmitrii Filippovdaf0ec92020-03-17 11:27:28 +0100694
695 const head = array.slice(0, array.length - size);
696 const tail = array.slice(array.length - size);
697
698 return this._breakdown(head, size).concat([tail]);
699 }
700}
701
Ben Rohlfs32b83822020-08-14 22:08:37 +0200702declare global {
703 interface HTMLElementTagNameMap {
704 'gr-diff-processor': GrDiffProcessor;
705 }
706}