blob: ebdff481532ec88049d34c78cd13594a9bda3b4a [file] [log] [blame]
/**
* @license
* Copyright 2016 Google LLC
* SPDX-License-Identifier: Apache-2.0
*/
import '../../../test/common-test-setup-karma';
import './gr-diff-selection';
import {GrDiffSelection} from './gr-diff-selection';
import {createDiff} from '../../../test/test-data-generators';
import {DiffInfo, Side} from '../../../api/diff';
import {GrFormattedText} from '../../../elements/shared/gr-formatted-text/gr-formatted-text';
import {fixture, html} from '@open-wc/testing';
import {mouseDown} from '../../../test/test-utils';
const diffTableTemplate = html`
<table id="diffTable" class="side-by-side">
<tr class="diff-row">
<td class="blame" data-line-number="1"></td>
<td class="lineNum left" data-value="1">1</td>
<td class="content">
<div class="contentText" data-side="left">ba ba</div>
<div data-side="left">
<div class="comment-thread">
<div class="gr-formatted-text message">
<span id="output" class="gr-linked-text">This is a comment</span>
</div>
</div>
</div>
</td>
<td class="lineNum right" data-value="1">1</td>
<td class="content">
<div class="contentText" data-side="right">some other text</div>
</td>
</tr>
<tr class="diff-row">
<td class="blame" data-line-number="2"></td>
<td class="lineNum left" data-value="2">2</td>
<td class="content">
<div class="contentText" data-side="left">zin</div>
</td>
<td class="lineNum right" data-value="2">2</td>
<td class="content">
<div class="contentText" data-side="right">more more more</div>
<div data-side="right">
<div class="comment-thread">
<div class="gr-formatted-text message">
<span id="output" class="gr-linked-text"
>This is a comment on the right</span
>
</div>
</div>
</div>
</td>
</tr>
<tr class="diff-row">
<td class="blame" data-line-number="3"></td>
<td class="lineNum left" data-value="3">3</td>
<td class="content">
<div class="contentText" data-side="left">ga ga</div>
<div data-side="left">
<div class="comment-thread">
<div class="gr-formatted-text message">
<span id="output" class="gr-linked-text"
>This is <a>a</a> different comment 💩 unicode is fun</span
>
</div>
</div>
</div>
</td>
<td class="lineNum right" data-value="3">3</td>
</tr>
<tr class="diff-row">
<td class="blame" data-line-number="4"></td>
<td class="lineNum left" data-value="4">4</td>
<td class="content">
<div class="contentText" data-side="left">ga ga</div>
<div data-side="left">
<div class="comment-thread">
<textarea data-side="right">test for textarea copying</textarea>
</div>
</div>
</td>
<td class="lineNum right" data-value="4">4</td>
</tr>
<tr class="not-diff-row">
<td class="other">
<div class="contentText" data-side="right">some other text</div>
</td>
</tr>
</table>
`;
suite('gr-diff-selection', () => {
let element: GrDiffSelection;
let diffTable: HTMLTableElement;
const emulateCopyOn = function (target: HTMLElement | null) {
const fakeEvent = {
target,
preventDefault: sinon.stub(),
composedPath() {
return [target];
},
clipboardData: {
setData: sinon.stub(),
},
};
element.handleCopy(fakeEvent as unknown as ClipboardEvent);
return fakeEvent;
};
setup(async () => {
element = new GrDiffSelection();
diffTable = await fixture<HTMLTableElement>(diffTableTemplate);
const diff: DiffInfo = {
...createDiff(),
content: [
{
a: ['ba ba'],
b: ['some other text'],
},
{
a: ['zin'],
b: ['more more more'],
},
{
a: ['ga ga'],
b: ['some other text'],
},
],
};
element.init(diff, diffTable);
});
test('applies selected-left on left side click', () => {
element.diffTable!.classList.add('selected-right');
const lineNumberEl = diffTable.querySelector<HTMLElement>('.lineNum.left');
if (!lineNumberEl) assert.fail('line number element missing');
mouseDown(lineNumberEl);
assert.isTrue(
element.diffTable!.classList.contains('selected-left'),
'adds selected-left'
);
assert.isFalse(
element.diffTable!.classList.contains('selected-right'),
'removes selected-right'
);
});
test('applies selected-right on right side click', () => {
element.diffTable!.classList.add('selected-left');
const lineNumberEl = diffTable.querySelector<HTMLElement>('.lineNum.right');
if (!lineNumberEl) assert.fail('line number element missing');
mouseDown(lineNumberEl);
assert.isTrue(
element.diffTable!.classList.contains('selected-right'),
'adds selected-right'
);
assert.isFalse(
element.diffTable!.classList.contains('selected-left'),
'removes selected-left'
);
});
test('applies selected-blame on blame click', () => {
element.diffTable!.classList.add('selected-left');
const blameDiv = document.createElement('div');
blameDiv.classList.add('blame');
element.diffTable!.appendChild(blameDiv);
mouseDown(blameDiv);
assert.isTrue(
element.diffTable!.classList.contains('selected-blame'),
'adds selected-right'
);
assert.isFalse(
element.diffTable!.classList.contains('selected-left'),
'removes selected-left'
);
});
test('ignores copy for non-content Element', () => {
const getSelectedTextStub = sinon.stub(element, 'getSelectedText');
emulateCopyOn(diffTable.querySelector('.not-diff-row'));
assert.isFalse(getSelectedTextStub.called);
});
test('asks for text for left side Elements', () => {
const getSelectedTextStub = sinon.stub(element, 'getSelectedText');
emulateCopyOn(diffTable.querySelector('div.contentText'));
assert.deepEqual([Side.LEFT, false], getSelectedTextStub.lastCall.args);
});
test('reacts to copy for content Elements', () => {
const getSelectedTextStub = sinon.stub(element, 'getSelectedText');
emulateCopyOn(diffTable.querySelector('div.contentText'));
assert.isTrue(getSelectedTextStub.called);
});
test('copy event is prevented for content Elements', () => {
const getSelectedTextStub = sinon.stub(element, 'getSelectedText');
getSelectedTextStub.returns('test');
const event = emulateCopyOn(diffTable.querySelector('div.contentText'));
assert.isTrue(event.preventDefault.called);
});
test('inserts text into clipboard on copy', () => {
sinon.stub(element, 'getSelectedText').returns('the text');
const event = emulateCopyOn(diffTable.querySelector('div.contentText'));
assert.deepEqual(
['Text', 'the text'],
event.clipboardData.setData.lastCall.args
);
});
test('setClasses adds given SelectionClass values, removes others', () => {
element.diffTable!.classList.add('selected-right');
element.setClasses(['selected-comment', 'selected-left']);
assert.isTrue(element.diffTable!.classList.contains('selected-comment'));
assert.isTrue(element.diffTable!.classList.contains('selected-left'));
assert.isFalse(element.diffTable!.classList.contains('selected-right'));
assert.isFalse(element.diffTable!.classList.contains('selected-blame'));
element.setClasses(['selected-blame']);
assert.isFalse(element.diffTable!.classList.contains('selected-comment'));
assert.isFalse(element.diffTable!.classList.contains('selected-left'));
assert.isFalse(element.diffTable!.classList.contains('selected-right'));
assert.isTrue(element.diffTable!.classList.contains('selected-blame'));
});
test('setClasses removes before it ads', () => {
element.diffTable!.classList.add('selected-right');
const addStub = sinon.stub(element.diffTable!.classList, 'add');
const removeStub = sinon
.stub(element.diffTable!.classList, 'remove')
.callsFake(() => {
assert.isFalse(addStub.called);
});
element.setClasses(['selected-comment', 'selected-left']);
assert.isTrue(addStub.called);
assert.isTrue(removeStub.called);
});
test('copies content correctly', () => {
element.diffTable!.classList.add('selected-left');
element.diffTable!.classList.remove('selected-right');
const selection = document.getSelection();
if (selection === null) assert.fail('no selection');
selection.removeAllRanges();
const range = document.createRange();
range.setStart(diffTable.querySelector('div.contentText')!.firstChild!, 3);
range.setEnd(
diffTable.querySelectorAll('div.contentText')[4]!.firstChild!,
2
);
selection.addRange(range);
assert.equal(element.getSelectedText(Side.LEFT, false), 'ba\nzin\nga');
});
test('copies comments', () => {
element.diffTable!.classList.add('selected-left');
element.diffTable!.classList.add('selected-comment');
element.diffTable!.classList.remove('selected-right');
const selection = document.getSelection();
if (selection === null) assert.fail('no selection');
selection.removeAllRanges();
const range = document.createRange();
range.setStart(
diffTable.querySelector('.gr-formatted-text *')!.firstChild!,
3
);
range.setEnd(
diffTable.querySelectorAll('.gr-formatted-text *')[2].childNodes[2],
7
);
selection.addRange(range);
assert.equal(
's is a comment\nThis is a differ',
element.getSelectedText(Side.LEFT, true)
);
});
test('respects astral chars in comments', () => {
element.diffTable!.classList.add('selected-left');
element.diffTable!.classList.add('selected-comment');
element.diffTable!.classList.remove('selected-right');
const selection = document.getSelection();
if (selection === null) assert.fail('no selection');
selection.removeAllRanges();
const range = document.createRange();
const nodes = diffTable.querySelectorAll('.gr-formatted-text *');
range.setStart(nodes[2].childNodes[2], 13);
range.setEnd(nodes[2].childNodes[2], 23);
selection.addRange(range);
assert.equal('mment 💩 u', element.getSelectedText(Side.LEFT, true));
});
test('defers to default behavior for textarea', () => {
element.diffTable!.classList.add('selected-left');
element.diffTable!.classList.remove('selected-right');
const selectedTextSpy = sinon.spy(element, 'getSelectedText');
emulateCopyOn(diffTable.querySelector('textarea'));
assert.isFalse(selectedTextSpy.called);
});
test('regression test for 4794', () => {
element.diffTable!.classList.add('selected-right');
element.diffTable!.classList.remove('selected-left');
const selection = document.getSelection();
if (!selection) assert.fail('no selection');
selection.removeAllRanges();
const range = document.createRange();
range.setStart(
diffTable.querySelectorAll('div.contentText')[1]!.firstChild!,
4
);
range.setEnd(
diffTable.querySelectorAll('div.contentText')[1]!.firstChild!,
10
);
selection.addRange(range);
assert.equal(element.getSelectedText(Side.RIGHT, false), ' other');
});
test('copies to end of side (issue 7895)', () => {
element.diffTable!.classList.add('selected-left');
element.diffTable!.classList.remove('selected-right');
const selection = document.getSelection();
if (selection === null) assert.fail('no selection');
selection.removeAllRanges();
const range = document.createRange();
range.setStart(diffTable.querySelector('div.contentText')!.firstChild!, 3);
range.setEnd(
diffTable.querySelectorAll('div.contentText')[4]!.firstChild!,
2
);
selection.addRange(range);
assert.equal(element.getSelectedText(Side.LEFT, false), 'ba\nzin\nga');
});
suite('getTextContentForRange', () => {
let selection: Selection;
let range: Range;
let nodes: NodeListOf<GrFormattedText>;
setup(() => {
element.diffTable!.classList.add('selected-left');
element.diffTable!.classList.add('selected-comment');
element.diffTable!.classList.remove('selected-right');
const s = document.getSelection();
if (s === null) assert.fail('no selection');
selection = s;
selection.removeAllRanges();
range = document.createRange();
nodes = diffTable.querySelectorAll('.gr-formatted-text *');
});
test('multi level element contained in range', () => {
range.setStart(nodes[2].childNodes[0], 1);
range.setEnd(nodes[2].childNodes[2], 7);
selection.addRange(range);
assert.equal(
element.getTextContentForRange(diffTable, selection, range),
'his is a differ'
);
});
test('multi level element as startContainer of range', () => {
range.setStart(nodes[2].childNodes[1], 0);
range.setEnd(nodes[2].childNodes[2], 7);
selection.addRange(range);
assert.equal(
element.getTextContentForRange(diffTable, selection, range),
'a differ'
);
});
test('startContainer === endContainer', () => {
range.setStart(nodes[0].firstChild!, 2);
range.setEnd(nodes[0].firstChild!, 12);
selection.addRange(range);
assert.equal(
element.getTextContentForRange(diffTable, selection, range),
'is is a co'
);
});
});
});