203 lines
4.8 KiB
Vue
203 lines
4.8 KiB
Vue
<script>
|
|
import createFlash from '~/flash';
|
|
import { s__ } from '~/locale';
|
|
import { mapState, mapGetters, mapActions } from 'vuex';
|
|
import Icon from '~/vue_shared/components/icon.vue';
|
|
import DiffGutterAvatars from './diff_gutter_avatars.vue';
|
|
import { LINE_POSITION_RIGHT, UNFOLD_COUNT } from '../constants';
|
|
import * as utils from '../store/utils';
|
|
|
|
export default {
|
|
components: {
|
|
DiffGutterAvatars,
|
|
Icon,
|
|
},
|
|
props: {
|
|
fileHash: {
|
|
type: String,
|
|
required: true,
|
|
},
|
|
contextLinesPath: {
|
|
type: String,
|
|
required: true,
|
|
},
|
|
lineType: {
|
|
type: String,
|
|
required: false,
|
|
default: '',
|
|
},
|
|
lineNumber: {
|
|
type: Number,
|
|
required: false,
|
|
default: 0,
|
|
},
|
|
lineCode: {
|
|
type: String,
|
|
required: false,
|
|
default: '',
|
|
},
|
|
linePosition: {
|
|
type: String,
|
|
required: false,
|
|
default: '',
|
|
},
|
|
metaData: {
|
|
type: Object,
|
|
required: false,
|
|
default: () => ({}),
|
|
},
|
|
showCommentButton: {
|
|
type: Boolean,
|
|
required: false,
|
|
default: false,
|
|
},
|
|
isBottom: {
|
|
type: Boolean,
|
|
required: false,
|
|
default: false,
|
|
},
|
|
isMatchLine: {
|
|
type: Boolean,
|
|
required: false,
|
|
default: false,
|
|
},
|
|
isMetaLine: {
|
|
type: Boolean,
|
|
required: false,
|
|
default: false,
|
|
},
|
|
isContextLine: {
|
|
type: Boolean,
|
|
required: false,
|
|
default: false,
|
|
},
|
|
discussions: {
|
|
type: Array,
|
|
required: false,
|
|
default: () => [],
|
|
},
|
|
},
|
|
computed: {
|
|
...mapState({
|
|
diffViewType: state => state.diffs.diffViewType,
|
|
diffFiles: state => state.diffs.diffFiles,
|
|
}),
|
|
...mapGetters(['isLoggedIn']),
|
|
lineHref() {
|
|
return this.lineCode ? `#${this.lineCode}` : '#';
|
|
},
|
|
shouldShowCommentButton() {
|
|
return (
|
|
this.isLoggedIn &&
|
|
this.showCommentButton &&
|
|
!this.isMatchLine &&
|
|
!this.isContextLine &&
|
|
!this.isMetaLine &&
|
|
!this.hasDiscussions
|
|
);
|
|
},
|
|
hasDiscussions() {
|
|
return this.discussions.length > 0;
|
|
},
|
|
shouldShowAvatarsOnGutter() {
|
|
if (!this.lineType && this.linePosition === LINE_POSITION_RIGHT) {
|
|
return false;
|
|
}
|
|
|
|
return this.hasDiscussions && this.showCommentButton;
|
|
},
|
|
},
|
|
methods: {
|
|
...mapActions('diffs', ['loadMoreLines', 'showCommentForm']),
|
|
handleCommentButton() {
|
|
this.showCommentForm({ lineCode: this.lineCode });
|
|
},
|
|
handleLoadMoreLines() {
|
|
if (this.isRequesting) {
|
|
return;
|
|
}
|
|
|
|
this.isRequesting = true;
|
|
const endpoint = this.contextLinesPath;
|
|
const oldLineNumber = this.metaData.oldPos || 0;
|
|
const newLineNumber = this.metaData.newPos || 0;
|
|
const offset = newLineNumber - oldLineNumber;
|
|
const bottom = this.isBottom;
|
|
const { fileHash } = this;
|
|
const view = this.diffViewType;
|
|
let unfold = true;
|
|
let lineNumber = newLineNumber - 1;
|
|
let since = lineNumber - UNFOLD_COUNT;
|
|
let to = lineNumber;
|
|
|
|
if (bottom) {
|
|
lineNumber = newLineNumber + 1;
|
|
since = lineNumber;
|
|
to = lineNumber + UNFOLD_COUNT;
|
|
} else {
|
|
const diffFile = utils.findDiffFile(this.diffFiles, this.fileHash);
|
|
const indexForInline = utils.findIndexInInlineLines(diffFile.highlightedDiffLines, {
|
|
oldLineNumber,
|
|
newLineNumber,
|
|
});
|
|
const prevLine = diffFile.highlightedDiffLines[indexForInline - 2];
|
|
const prevLineNumber = (prevLine && prevLine.newLine) || 0;
|
|
|
|
if (since <= prevLineNumber + 1) {
|
|
since = prevLineNumber + 1;
|
|
unfold = false;
|
|
}
|
|
}
|
|
|
|
const params = { since, to, bottom, offset, unfold, view };
|
|
const lineNumbers = { oldLineNumber, newLineNumber };
|
|
this.loadMoreLines({ endpoint, params, lineNumbers, fileHash })
|
|
.then(() => {
|
|
this.isRequesting = false;
|
|
})
|
|
.catch(() => {
|
|
createFlash(s__('Diffs|Something went wrong while fetching diff lines.'));
|
|
this.isRequesting = false;
|
|
});
|
|
},
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<div>
|
|
<span
|
|
v-if="isMatchLine"
|
|
class="context-cell"
|
|
role="button"
|
|
@click="handleLoadMoreLines"
|
|
>...</span>
|
|
<template
|
|
v-else
|
|
>
|
|
<button
|
|
v-show="shouldShowCommentButton"
|
|
type="button"
|
|
class="add-diff-note js-add-diff-note-button"
|
|
title="Add a comment to this line"
|
|
@click="handleCommentButton"
|
|
>
|
|
<icon
|
|
:size="12"
|
|
name="comment"
|
|
/>
|
|
</button>
|
|
<a
|
|
v-if="lineNumber"
|
|
:data-linenumber="lineNumber"
|
|
:href="lineHref"
|
|
>
|
|
</a>
|
|
<diff-gutter-avatars
|
|
v-if="shouldShowAvatarsOnGutter"
|
|
:discussions="discussions"
|
|
/>
|
|
</template>
|
|
</div>
|
|
</template>
|