forked from M3-Academy/challenge-algorithms-v2.0
272 lines
7.6 KiB
JavaScript
272 lines
7.6 KiB
JavaScript
|
'use strict';
|
||
|
|
||
|
Object.defineProperty(exports, '__esModule', {
|
||
|
value: true
|
||
|
});
|
||
|
exports.joinAlignedDiffsNoExpand = exports.joinAlignedDiffsExpand = void 0;
|
||
|
var _cleanupSemantic = require('./cleanupSemantic');
|
||
|
/**
|
||
|
* Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved.
|
||
|
*
|
||
|
* This source code is licensed under the MIT license found in the
|
||
|
* LICENSE file in the root directory of this source tree.
|
||
|
*/
|
||
|
|
||
|
const formatTrailingSpaces = (line, trailingSpaceFormatter) =>
|
||
|
line.replace(/\s+$/, match => trailingSpaceFormatter(match));
|
||
|
const printDiffLine = (
|
||
|
line,
|
||
|
isFirstOrLast,
|
||
|
color,
|
||
|
indicator,
|
||
|
trailingSpaceFormatter,
|
||
|
emptyFirstOrLastLinePlaceholder
|
||
|
) =>
|
||
|
line.length !== 0
|
||
|
? color(
|
||
|
`${indicator} ${formatTrailingSpaces(line, trailingSpaceFormatter)}`
|
||
|
)
|
||
|
: indicator !== ' '
|
||
|
? color(indicator)
|
||
|
: isFirstOrLast && emptyFirstOrLastLinePlaceholder.length !== 0
|
||
|
? color(`${indicator} ${emptyFirstOrLastLinePlaceholder}`)
|
||
|
: '';
|
||
|
const printDeleteLine = (
|
||
|
line,
|
||
|
isFirstOrLast,
|
||
|
{
|
||
|
aColor,
|
||
|
aIndicator,
|
||
|
changeLineTrailingSpaceColor,
|
||
|
emptyFirstOrLastLinePlaceholder
|
||
|
}
|
||
|
) =>
|
||
|
printDiffLine(
|
||
|
line,
|
||
|
isFirstOrLast,
|
||
|
aColor,
|
||
|
aIndicator,
|
||
|
changeLineTrailingSpaceColor,
|
||
|
emptyFirstOrLastLinePlaceholder
|
||
|
);
|
||
|
const printInsertLine = (
|
||
|
line,
|
||
|
isFirstOrLast,
|
||
|
{
|
||
|
bColor,
|
||
|
bIndicator,
|
||
|
changeLineTrailingSpaceColor,
|
||
|
emptyFirstOrLastLinePlaceholder
|
||
|
}
|
||
|
) =>
|
||
|
printDiffLine(
|
||
|
line,
|
||
|
isFirstOrLast,
|
||
|
bColor,
|
||
|
bIndicator,
|
||
|
changeLineTrailingSpaceColor,
|
||
|
emptyFirstOrLastLinePlaceholder
|
||
|
);
|
||
|
const printCommonLine = (
|
||
|
line,
|
||
|
isFirstOrLast,
|
||
|
{
|
||
|
commonColor,
|
||
|
commonIndicator,
|
||
|
commonLineTrailingSpaceColor,
|
||
|
emptyFirstOrLastLinePlaceholder
|
||
|
}
|
||
|
) =>
|
||
|
printDiffLine(
|
||
|
line,
|
||
|
isFirstOrLast,
|
||
|
commonColor,
|
||
|
commonIndicator,
|
||
|
commonLineTrailingSpaceColor,
|
||
|
emptyFirstOrLastLinePlaceholder
|
||
|
);
|
||
|
|
||
|
// In GNU diff format, indexes are one-based instead of zero-based.
|
||
|
const createPatchMark = (aStart, aEnd, bStart, bEnd, {patchColor}) =>
|
||
|
patchColor(
|
||
|
`@@ -${aStart + 1},${aEnd - aStart} +${bStart + 1},${bEnd - bStart} @@`
|
||
|
);
|
||
|
|
||
|
// jest --no-expand
|
||
|
//
|
||
|
// Given array of aligned strings with inverse highlight formatting,
|
||
|
// return joined lines with diff formatting (and patch marks, if needed).
|
||
|
const joinAlignedDiffsNoExpand = (diffs, options) => {
|
||
|
const iLength = diffs.length;
|
||
|
const nContextLines = options.contextLines;
|
||
|
const nContextLines2 = nContextLines + nContextLines;
|
||
|
|
||
|
// First pass: count output lines and see if it has patches.
|
||
|
let jLength = iLength;
|
||
|
let hasExcessAtStartOrEnd = false;
|
||
|
let nExcessesBetweenChanges = 0;
|
||
|
let i = 0;
|
||
|
while (i !== iLength) {
|
||
|
const iStart = i;
|
||
|
while (i !== iLength && diffs[i][0] === _cleanupSemantic.DIFF_EQUAL) {
|
||
|
i += 1;
|
||
|
}
|
||
|
if (iStart !== i) {
|
||
|
if (iStart === 0) {
|
||
|
// at start
|
||
|
if (i > nContextLines) {
|
||
|
jLength -= i - nContextLines; // subtract excess common lines
|
||
|
hasExcessAtStartOrEnd = true;
|
||
|
}
|
||
|
} else if (i === iLength) {
|
||
|
// at end
|
||
|
const n = i - iStart;
|
||
|
if (n > nContextLines) {
|
||
|
jLength -= n - nContextLines; // subtract excess common lines
|
||
|
hasExcessAtStartOrEnd = true;
|
||
|
}
|
||
|
} else {
|
||
|
// between changes
|
||
|
const n = i - iStart;
|
||
|
if (n > nContextLines2) {
|
||
|
jLength -= n - nContextLines2; // subtract excess common lines
|
||
|
nExcessesBetweenChanges += 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
while (i !== iLength && diffs[i][0] !== _cleanupSemantic.DIFF_EQUAL) {
|
||
|
i += 1;
|
||
|
}
|
||
|
}
|
||
|
const hasPatch = nExcessesBetweenChanges !== 0 || hasExcessAtStartOrEnd;
|
||
|
if (nExcessesBetweenChanges !== 0) {
|
||
|
jLength += nExcessesBetweenChanges + 1; // add patch lines
|
||
|
} else if (hasExcessAtStartOrEnd) {
|
||
|
jLength += 1; // add patch line
|
||
|
}
|
||
|
|
||
|
const jLast = jLength - 1;
|
||
|
const lines = [];
|
||
|
let jPatchMark = 0; // index of placeholder line for current patch mark
|
||
|
if (hasPatch) {
|
||
|
lines.push(''); // placeholder line for first patch mark
|
||
|
}
|
||
|
|
||
|
// Indexes of expected or received lines in current patch:
|
||
|
let aStart = 0;
|
||
|
let bStart = 0;
|
||
|
let aEnd = 0;
|
||
|
let bEnd = 0;
|
||
|
const pushCommonLine = line => {
|
||
|
const j = lines.length;
|
||
|
lines.push(printCommonLine(line, j === 0 || j === jLast, options));
|
||
|
aEnd += 1;
|
||
|
bEnd += 1;
|
||
|
};
|
||
|
const pushDeleteLine = line => {
|
||
|
const j = lines.length;
|
||
|
lines.push(printDeleteLine(line, j === 0 || j === jLast, options));
|
||
|
aEnd += 1;
|
||
|
};
|
||
|
const pushInsertLine = line => {
|
||
|
const j = lines.length;
|
||
|
lines.push(printInsertLine(line, j === 0 || j === jLast, options));
|
||
|
bEnd += 1;
|
||
|
};
|
||
|
|
||
|
// Second pass: push lines with diff formatting (and patch marks, if needed).
|
||
|
i = 0;
|
||
|
while (i !== iLength) {
|
||
|
let iStart = i;
|
||
|
while (i !== iLength && diffs[i][0] === _cleanupSemantic.DIFF_EQUAL) {
|
||
|
i += 1;
|
||
|
}
|
||
|
if (iStart !== i) {
|
||
|
if (iStart === 0) {
|
||
|
// at beginning
|
||
|
if (i > nContextLines) {
|
||
|
iStart = i - nContextLines;
|
||
|
aStart = iStart;
|
||
|
bStart = iStart;
|
||
|
aEnd = aStart;
|
||
|
bEnd = bStart;
|
||
|
}
|
||
|
for (let iCommon = iStart; iCommon !== i; iCommon += 1) {
|
||
|
pushCommonLine(diffs[iCommon][1]);
|
||
|
}
|
||
|
} else if (i === iLength) {
|
||
|
// at end
|
||
|
const iEnd = i - iStart > nContextLines ? iStart + nContextLines : i;
|
||
|
for (let iCommon = iStart; iCommon !== iEnd; iCommon += 1) {
|
||
|
pushCommonLine(diffs[iCommon][1]);
|
||
|
}
|
||
|
} else {
|
||
|
// between changes
|
||
|
const nCommon = i - iStart;
|
||
|
if (nCommon > nContextLines2) {
|
||
|
const iEnd = iStart + nContextLines;
|
||
|
for (let iCommon = iStart; iCommon !== iEnd; iCommon += 1) {
|
||
|
pushCommonLine(diffs[iCommon][1]);
|
||
|
}
|
||
|
lines[jPatchMark] = createPatchMark(
|
||
|
aStart,
|
||
|
aEnd,
|
||
|
bStart,
|
||
|
bEnd,
|
||
|
options
|
||
|
);
|
||
|
jPatchMark = lines.length;
|
||
|
lines.push(''); // placeholder line for next patch mark
|
||
|
|
||
|
const nOmit = nCommon - nContextLines2;
|
||
|
aStart = aEnd + nOmit;
|
||
|
bStart = bEnd + nOmit;
|
||
|
aEnd = aStart;
|
||
|
bEnd = bStart;
|
||
|
for (let iCommon = i - nContextLines; iCommon !== i; iCommon += 1) {
|
||
|
pushCommonLine(diffs[iCommon][1]);
|
||
|
}
|
||
|
} else {
|
||
|
for (let iCommon = iStart; iCommon !== i; iCommon += 1) {
|
||
|
pushCommonLine(diffs[iCommon][1]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
while (i !== iLength && diffs[i][0] === _cleanupSemantic.DIFF_DELETE) {
|
||
|
pushDeleteLine(diffs[i][1]);
|
||
|
i += 1;
|
||
|
}
|
||
|
while (i !== iLength && diffs[i][0] === _cleanupSemantic.DIFF_INSERT) {
|
||
|
pushInsertLine(diffs[i][1]);
|
||
|
i += 1;
|
||
|
}
|
||
|
}
|
||
|
if (hasPatch) {
|
||
|
lines[jPatchMark] = createPatchMark(aStart, aEnd, bStart, bEnd, options);
|
||
|
}
|
||
|
return lines.join('\n');
|
||
|
};
|
||
|
|
||
|
// jest --expand
|
||
|
//
|
||
|
// Given array of aligned strings with inverse highlight formatting,
|
||
|
// return joined lines with diff formatting.
|
||
|
exports.joinAlignedDiffsNoExpand = joinAlignedDiffsNoExpand;
|
||
|
const joinAlignedDiffsExpand = (diffs, options) =>
|
||
|
diffs
|
||
|
.map((diff, i, diffs) => {
|
||
|
const line = diff[1];
|
||
|
const isFirstOrLast = i === 0 || i === diffs.length - 1;
|
||
|
switch (diff[0]) {
|
||
|
case _cleanupSemantic.DIFF_DELETE:
|
||
|
return printDeleteLine(line, isFirstOrLast, options);
|
||
|
case _cleanupSemantic.DIFF_INSERT:
|
||
|
return printInsertLine(line, isFirstOrLast, options);
|
||
|
default:
|
||
|
return printCommonLine(line, isFirstOrLast, options);
|
||
|
}
|
||
|
})
|
||
|
.join('\n');
|
||
|
exports.joinAlignedDiffsExpand = joinAlignedDiffsExpand;
|