Commit f082bce2 authored by Justin Boyson's avatar Justin Boyson Committed by Phil Hughes

Jdb/refactor inline diff table row

parent 2df757d7
<script> <script>
/* eslint-disable vue/no-v-html */ /* eslint-disable vue/no-v-html */
import { mapActions, mapGetters, mapState } from 'vuex'; import { mapActions, mapGetters, mapState } from 'vuex';
import { GlTooltipDirective } from '@gitlab/ui'; import { GlTooltipDirective, GlIcon } from '@gitlab/ui';
import DiffTableCell from './diff_table_cell.vue';
import { import {
MATCH_LINE_TYPE, MATCH_LINE_TYPE,
NEW_LINE_TYPE, NEW_LINE_TYPE,
...@@ -11,11 +10,19 @@ import { ...@@ -11,11 +10,19 @@ import {
CONTEXT_LINE_CLASS_NAME, CONTEXT_LINE_CLASS_NAME,
LINE_POSITION_LEFT, LINE_POSITION_LEFT,
LINE_POSITION_RIGHT, LINE_POSITION_RIGHT,
LINE_HOVER_CLASS_NAME,
OLD_NO_NEW_LINE_TYPE,
NEW_NO_NEW_LINE_TYPE,
EMPTY_CELL_TYPE,
} from '../constants'; } from '../constants';
import { __ } from '~/locale';
import { getParameterByName, parseBoolean } from '~/lib/utils/common_utils';
import DiffGutterAvatars from './diff_gutter_avatars.vue';
export default { export default {
components: { components: {
DiffTableCell, DiffGutterAvatars,
GlIcon,
}, },
directives: { directives: {
GlTooltip: GlTooltipDirective, GlTooltip: GlTooltipDirective,
...@@ -50,6 +57,7 @@ export default { ...@@ -50,6 +57,7 @@ export default {
}; };
}, },
computed: { computed: {
...mapGetters(['isLoggedIn']),
...mapGetters('diffs', ['fileLineCoverage']), ...mapGetters('diffs', ['fileLineCoverage']),
...mapState({ ...mapState({
isHighlighted(state) { isHighlighted(state) {
...@@ -79,6 +87,70 @@ export default { ...@@ -79,6 +87,70 @@ export default {
coverageState() { coverageState() {
return this.fileLineCoverage(this.filePath, this.line.new_line); return this.fileLineCoverage(this.filePath, this.line.new_line);
}, },
isMetaLine() {
const { type } = this.line;
return (
type === OLD_NO_NEW_LINE_TYPE || type === NEW_NO_NEW_LINE_TYPE || type === EMPTY_CELL_TYPE
);
},
classNameMapCell() {
const { type } = this.line;
return [
type,
{
hll: this.isHighlighted,
[LINE_HOVER_CLASS_NAME]:
this.isLoggedIn && this.isHover && !this.isContextLine && !this.isMetaLine,
},
];
},
addCommentTooltip() {
const brokenSymlinks = this.line.commentsDisabled;
let tooltip = __('Add a comment to this line');
if (brokenSymlinks) {
if (brokenSymlinks.wasSymbolic || brokenSymlinks.isSymbolic) {
tooltip = __(
'Commenting on symbolic links that replace or are replaced by files is currently not supported.',
);
} else if (brokenSymlinks.wasReal || brokenSymlinks.isReal) {
tooltip = __(
'Commenting on files that replace or are replaced by symbolic links is currently not supported.',
);
}
}
return tooltip;
},
shouldRenderCommentButton() {
if (this.isLoggedIn) {
const isDiffHead = parseBoolean(getParameterByName('diff_head'));
return !isDiffHead || gon.features?.mergeRefHeadComments;
}
return false;
},
shouldShowCommentButton() {
return this.isHover && !this.isContextLine && !this.isMetaLine && !this.hasDiscussions;
},
hasDiscussions() {
return this.line.discussions && this.line.discussions.length > 0;
},
lineHref() {
return `#${this.line.line_code || ''}`;
},
lineCode() {
return (
this.line.line_code ||
(this.line.left && this.line.left.line_code) ||
(this.line.right && this.line.right.line_code)
);
},
shouldShowAvatarsOnGutter() {
return this.hasDiscussions;
},
}, },
created() { created() {
this.newLineType = NEW_LINE_TYPE; this.newLineType = NEW_LINE_TYPE;
...@@ -90,12 +162,20 @@ export default { ...@@ -90,12 +162,20 @@ export default {
this.scrollToLineIfNeededInline(this.line); this.scrollToLineIfNeededInline(this.line);
}, },
methods: { methods: {
...mapActions('diffs', ['scrollToLineIfNeededInline']), ...mapActions('diffs', [
'scrollToLineIfNeededInline',
'showCommentForm',
'setHighlightedRow',
'toggleLineDiscussions',
]),
handleMouseMove(e) { handleMouseMove(e) {
// To show the comment icon on the gutter we need to know if we hover the line. // To show the comment icon on the gutter we need to know if we hover the line.
// Current table structure doesn't allow us to do this with CSS in both of the diff view types // Current table structure doesn't allow us to do this with CSS in both of the diff view types
this.isHover = e.type === 'mouseover'; this.isHover = e.type === 'mouseover';
}, },
handleCommentButton() {
this.showCommentForm({ lineCode: this.line.line_code, fileHash: this.fileHash });
},
}, },
}; };
</script> </script>
...@@ -109,25 +189,52 @@ export default { ...@@ -109,25 +189,52 @@ export default {
@mouseover="handleMouseMove" @mouseover="handleMouseMove"
@mouseout="handleMouseMove" @mouseout="handleMouseMove"
> >
<diff-table-cell <td ref="oldTd" class="diff-line-num old_line" :class="classNameMapCell">
:file-hash="fileHash" <span
:line="line" v-if="shouldRenderCommentButton"
:line-type="oldLineType" ref="addNoteTooltip"
:is-bottom="isBottom" v-gl-tooltip
:is-hover="isHover" class="add-diff-note tooltip-wrapper"
:show-comment-button="true" :title="addCommentTooltip"
:is-highlighted="isHighlighted" >
class="diff-line-num old_line" <button
/> v-show="shouldShowCommentButton"
<diff-table-cell ref="addDiffNoteButton"
:file-hash="fileHash" type="button"
:line="line" class="add-diff-note note-button js-add-diff-note-button qa-diff-comment"
:line-type="newLineType" :disabled="line.commentsDisabled"
:is-bottom="isBottom" @click="handleCommentButton"
:is-hover="isHover" >
:is-highlighted="isHighlighted" <gl-icon :size="12" name="comment" />
class="diff-line-num new_line qa-new-diff-line" </button>
/> </span>
<a
v-if="line.old_line"
ref="lineNumberRefOld"
:data-linenumber="line.old_line"
:href="lineHref"
@click="setHighlightedRow(lineCode)"
>
</a>
<diff-gutter-avatars
v-if="shouldShowAvatarsOnGutter"
:discussions="line.discussions"
:discussions-expanded="line.discussionsExpanded"
@toggleLineDiscussions="
toggleLineDiscussions({ lineCode, fileHash, expanded: !line.discussionsExpanded })
"
/>
</td>
<td ref="newTd" class="diff-line-num new_line qa-new-diff-line" :class="classNameMapCell">
<a
v-if="line.new_line"
ref="lineNumberRefNew"
:data-linenumber="line.new_line"
:href="lineHref"
@click="setHighlightedRow(lineCode)"
>
</a>
</td>
<td <td
v-gl-tooltip.hover v-gl-tooltip.hover
:title="coverageState.text" :title="coverageState.text"
......
---
title: Jdb/refactor inline diff table row
merge_request: 40906
author:
type: performance
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import { TEST_HOST } from 'helpers/test_constants';
import { createStore } from '~/mr_notes/stores'; import { createStore } from '~/mr_notes/stores';
import InlineDiffTableRow from '~/diffs/components/inline_diff_table_row.vue'; import InlineDiffTableRow from '~/diffs/components/inline_diff_table_row.vue';
import DiffGutterAvatars from '~/diffs/components/diff_gutter_avatars.vue';
import diffFileMockData from '../mock_data/diff_file'; import diffFileMockData from '../mock_data/diff_file';
import discussionsMockData from '../mock_data/diff_discussions';
const TEST_USER_ID = 'abc123';
const TEST_USER = { id: TEST_USER_ID };
describe('InlineDiffTableRow', () => { describe('InlineDiffTableRow', () => {
let wrapper; let wrapper;
let vm; let store;
const thisLine = diffFileMockData.highlighted_diff_lines[0]; const thisLine = diffFileMockData.highlighted_diff_lines[0];
beforeEach(() => { const createComponent = (props = {}, propsStore = store) => {
wrapper = shallowMount(InlineDiffTableRow, { wrapper = shallowMount(InlineDiffTableRow, {
store: createStore(), store: propsStore,
propsData: { propsData: {
line: thisLine, line: thisLine,
fileHash: diffFileMockData.file_hash, fileHash: diffFileMockData.file_hash,
filePath: diffFileMockData.file_path, filePath: diffFileMockData.file_path,
contextLinesPath: 'contextLinesPath', contextLinesPath: 'contextLinesPath',
isHighlighted: false, isHighlighted: false,
...props,
}, },
}); });
vm = wrapper.vm; };
const setWindowLocation = value => {
Object.defineProperty(window, 'location', {
writable: true,
value,
});
};
beforeEach(() => {
store = createStore();
store.state.notes.userData = TEST_USER;
});
afterEach(() => {
wrapper.destroy();
}); });
it('does not add hll class to line content when line does not match highlighted row', done => { it('does not add hll class to line content when line does not match highlighted row', () => {
vm.$nextTick() createComponent();
.then(() => { expect(wrapper.find('.line_content').classes('hll')).toBe(false);
expect(wrapper.find('.line_content').classes('hll')).toBe(false);
})
.then(done)
.catch(done.fail);
}); });
it('adds hll class to lineContent when line is the highlighted row', done => { it('adds hll class to lineContent when line is the highlighted row', () => {
vm.$nextTick() store.state.diffs.highlightedRow = thisLine.line_code;
.then(() => { createComponent({}, store);
vm.$store.state.diffs.highlightedRow = thisLine.line_code; expect(wrapper.find('.line_content').classes('hll')).toBe(true);
return vm.$nextTick();
})
.then(() => {
expect(wrapper.find('.line_content').classes('hll')).toBe(true);
})
.then(done)
.catch(done.fail);
}); });
it('adds hll class to lineContent when line is part of a multiline comment', () => { it('adds hll class to lineContent when line is part of a multiline comment', () => {
wrapper.setProps({ isCommented: true }); createComponent({ isCommented: true });
return vm.$nextTick().then(() => { expect(wrapper.find('.line_content').classes('hll')).toBe(true);
expect(wrapper.find('.line_content').classes('hll')).toBe(true);
});
}); });
describe('sets coverage title and class', () => { describe('sets coverage title and class', () => {
it('for lines with coverage', done => { it('for lines with coverage', () => {
vm.$nextTick() const name = diffFileMockData.file_path;
.then(() => { const line = thisLine.new_line;
const name = diffFileMockData.file_path;
const line = thisLine.new_line; store.state.diffs.coverageFiles = { files: { [name]: { [line]: 5 } } };
createComponent({}, store);
vm.$store.state.diffs.coverageFiles = { files: { [name]: { [line]: 5 } } }; const coverage = wrapper.find('.line-coverage');
return vm.$nextTick(); expect(coverage.attributes('title')).toContain('Test coverage: 5 hits');
}) expect(coverage.classes('coverage')).toBe(true);
.then(() => { });
const coverage = wrapper.find('.line-coverage');
it('for lines without coverage', () => {
expect(coverage.attributes('title')).toContain('Test coverage: 5 hits'); const name = diffFileMockData.file_path;
expect(coverage.classes('coverage')).toBe(true); const line = thisLine.new_line;
})
.then(done) store.state.diffs.coverageFiles = { files: { [name]: { [line]: 0 } } };
.catch(done.fail); createComponent({}, store);
const coverage = wrapper.find('.line-coverage');
expect(coverage.attributes('title')).toContain('No test coverage');
expect(coverage.classes('no-coverage')).toBe(true);
});
it('for unknown lines', () => {
store.state.diffs.coverageFiles = {};
createComponent({}, store);
const coverage = wrapper.find('.line-coverage');
expect(coverage.attributes('title')).toBeUndefined();
expect(coverage.classes('coverage')).toBe(false);
expect(coverage.classes('no-coverage')).toBe(false);
});
});
describe('Table Cells', () => {
const findNewTd = () => wrapper.find({ ref: 'newTd' });
const findOldTd = () => wrapper.find({ ref: 'oldTd' });
describe('td', () => {
it('highlights when isHighlighted true', () => {
store.state.diffs.highlightedRow = thisLine.line_code;
createComponent({}, store);
expect(findNewTd().classes()).toContain('hll');
expect(findOldTd().classes()).toContain('hll');
});
it('does not highlight when isHighlighted false', () => {
createComponent();
expect(findNewTd().classes()).not.toContain('hll');
expect(findOldTd().classes()).not.toContain('hll');
});
});
describe('comment button', () => {
const findNoteButton = () => wrapper.find({ ref: 'addDiffNoteButton' });
it.each`
userData | query | mergeRefHeadComments | expectation
${TEST_USER} | ${'diff_head=false'} | ${false} | ${true}
${TEST_USER} | ${'diff_head=true'} | ${true} | ${true}
${TEST_USER} | ${'diff_head=true'} | ${false} | ${false}
${null} | ${''} | ${true} | ${false}
`(
'exists is $expectation - with userData ($userData) query ($query)',
({ userData, query, mergeRefHeadComments, expectation }) => {
store.state.notes.userData = userData;
gon.features = { mergeRefHeadComments };
setWindowLocation({ href: `${TEST_HOST}?${query}` });
createComponent({}, store);
expect(findNoteButton().exists()).toBe(expectation);
},
);
it.each`
isHover | line | expectation
${true} | ${{ ...thisLine, discussions: [] }} | ${true}
${false} | ${{ ...thisLine, discussions: [] }} | ${false}
${true} | ${{ ...thisLine, type: 'context', discussions: [] }} | ${false}
${true} | ${{ ...thisLine, type: 'old-nonewline', discussions: [] }} | ${false}
${true} | ${{ ...thisLine, discussions: [{}] }} | ${false}
`('visible is $expectation - line ($line)', ({ isHover, line, expectation }) => {
createComponent({ line });
wrapper.setData({ isHover });
return wrapper.vm.$nextTick().then(() => {
expect(findNoteButton().isVisible()).toBe(expectation);
});
});
it.each`
disabled | commentsDisabled
${'disabled'} | ${true}
${undefined} | ${false}
`(
'has attribute disabled=$disabled when the outer component has prop commentsDisabled=$commentsDisabled',
({ disabled, commentsDisabled }) => {
createComponent({
line: { ...thisLine, commentsDisabled },
});
wrapper.setData({ isHover: true });
return wrapper.vm.$nextTick().then(() => {
expect(findNoteButton().attributes('disabled')).toBe(disabled);
});
},
);
const symlinkishFileTooltip =
'Commenting on symbolic links that replace or are replaced by files is currently not supported.';
const realishFileTooltip =
'Commenting on files that replace or are replaced by symbolic links is currently not supported.';
const otherFileTooltip = 'Add a comment to this line';
const findTooltip = () => wrapper.find({ ref: 'addNoteTooltip' });
it.each`
tooltip | commentsDisabled
${symlinkishFileTooltip} | ${{ wasSymbolic: true }}
${symlinkishFileTooltip} | ${{ isSymbolic: true }}
${realishFileTooltip} | ${{ wasReal: true }}
${realishFileTooltip} | ${{ isReal: true }}
${otherFileTooltip} | ${false}
`(
'has the correct tooltip when commentsDisabled=$commentsDisabled',
({ tooltip, commentsDisabled }) => {
createComponent({
line: { ...thisLine, commentsDisabled },
});
wrapper.setData({ isHover: true });
return wrapper.vm.$nextTick().then(() => {
expect(findTooltip().attributes('title')).toBe(tooltip);
});
},
);
}); });
it('for lines without coverage', done => { describe('line number', () => {
vm.$nextTick() const findLineNumberOld = () => wrapper.find({ ref: 'lineNumberRefOld' });
.then(() => { const findLineNumberNew = () => wrapper.find({ ref: 'lineNumberRefNew' });
const name = diffFileMockData.file_path;
const line = thisLine.new_line; it('renders line numbers in correct cells', () => {
createComponent();
expect(findLineNumberOld().exists()).toBe(false);
expect(findLineNumberNew().exists()).toBe(true);
});
vm.$store.state.diffs.coverageFiles = { files: { [name]: { [line]: 0 } } }; describe('with lineNumber prop', () => {
const TEST_LINE_CODE = 'LC_42';
const TEST_LINE_NUMBER = 1;
return vm.$nextTick(); describe.each`
}) lineProps | findLineNumber | expectedHref | expectedClickArg
.then(() => { ${{ line_code: TEST_LINE_CODE, old_line: TEST_LINE_NUMBER }} | ${findLineNumberOld} | ${`#${TEST_LINE_CODE}`} | ${TEST_LINE_CODE}
const coverage = wrapper.find('.line-coverage'); ${{ line_code: undefined, old_line: TEST_LINE_NUMBER }} | ${findLineNumberOld} | ${'#'} | ${undefined}
${{ line_code: undefined, left: { line_code: TEST_LINE_CODE }, old_line: TEST_LINE_NUMBER }} | ${findLineNumberOld} | ${'#'} | ${TEST_LINE_CODE}
${{ line_code: undefined, right: { line_code: TEST_LINE_CODE }, new_line: TEST_LINE_NUMBER }} | ${findLineNumberNew} | ${'#'} | ${TEST_LINE_CODE}
`(
'with line ($lineProps)',
({ lineProps, findLineNumber, expectedHref, expectedClickArg }) => {
beforeEach(() => {
jest.spyOn(store, 'dispatch').mockImplementation();
createComponent({
line: { ...thisLine, ...lineProps },
});
});
expect(coverage.attributes('title')).toContain('No test coverage'); it('renders', () => {
expect(coverage.classes('no-coverage')).toBe(true); expect(findLineNumber().exists()).toBe(true);
}) expect(findLineNumber().attributes()).toEqual({
.then(done) href: expectedHref,
.catch(done.fail); 'data-linenumber': TEST_LINE_NUMBER.toString(),
});
});
it('on click, dispatches setHighlightedRow', () => {
expect(store.dispatch).toHaveBeenCalledTimes(1);
findLineNumber().trigger('click');
expect(store.dispatch).toHaveBeenCalledWith(
'diffs/setHighlightedRow',
expectedClickArg,
);
expect(store.dispatch).toHaveBeenCalledTimes(2);
});
},
);
});
}); });
it('for unknown lines', done => { describe('diff-gutter-avatars', () => {
vm.$nextTick() const TEST_LINE_CODE = 'LC_42';
.then(() => { const TEST_FILE_HASH = diffFileMockData.file_hash;
vm.$store.state.diffs.coverageFiles = {}; const findAvatars = () => wrapper.find(DiffGutterAvatars);
let line;
return vm.$nextTick();
}) beforeEach(() => {
.then(() => { jest.spyOn(store, 'dispatch').mockImplementation();
const coverage = wrapper.find('.line-coverage');
line = {
expect(coverage.attributes('title')).toBeUndefined(); line_code: TEST_LINE_CODE,
expect(coverage.classes('coverage')).toBe(false); type: 'new',
expect(coverage.classes('no-coverage')).toBe(false); old_line: null,
}) new_line: 1,
.then(done) discussions: [{ ...discussionsMockData }],
.catch(done.fail); discussionsExpanded: true,
text: '+<span id="LC1" class="line" lang="plaintext"> - Bad dates</span>\n',
rich_text: '+<span id="LC1" class="line" lang="plaintext"> - Bad dates</span>\n',
meta_data: null,
};
});
describe('with showCommentButton', () => {
it('renders if line has discussions', () => {
createComponent({ line });
expect(findAvatars().props()).toEqual({
discussions: line.discussions,
discussionsExpanded: line.discussionsExpanded,
});
});
it('does notrender if line has no discussions', () => {
line.discussions = [];
createComponent({ line });
expect(findAvatars().exists()).toEqual(false);
});
it('toggles line discussion', () => {
createComponent({ line });
expect(store.dispatch).toHaveBeenCalledTimes(1);
findAvatars().vm.$emit('toggleLineDiscussions');
expect(store.dispatch).toHaveBeenCalledWith('diffs/toggleLineDiscussions', {
lineCode: TEST_LINE_CODE,
fileHash: TEST_FILE_HASH,
expanded: !line.discussionsExpanded,
});
});
});
}); });
}); });
}); });
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment