299 lines
8.5 KiB
Vue
299 lines
8.5 KiB
Vue
<script>
|
|
import _ from 'underscore';
|
|
import { mapActions, mapGetters } from 'vuex';
|
|
import { GlButton, GlTooltipDirective, GlTooltip, GlLoadingIcon } from '@gitlab/ui';
|
|
import { polyfillSticky } from '~/lib/utils/sticky';
|
|
import ClipboardButton from '~/vue_shared/components/clipboard_button.vue';
|
|
import Icon from '~/vue_shared/components/icon.vue';
|
|
import FileIcon from '~/vue_shared/components/file_icon.vue';
|
|
import { truncateSha } from '~/lib/utils/text_utility';
|
|
import { __, s__, sprintf } from '~/locale';
|
|
import { diffViewerModes } from '~/ide/constants';
|
|
import EditButton from './edit_button.vue';
|
|
import DiffStats from './diff_stats.vue';
|
|
import { scrollToElement } from '~/lib/utils/common_utils';
|
|
|
|
export default {
|
|
components: {
|
|
GlTooltip,
|
|
GlLoadingIcon,
|
|
GlButton,
|
|
ClipboardButton,
|
|
EditButton,
|
|
Icon,
|
|
FileIcon,
|
|
DiffStats,
|
|
},
|
|
directives: {
|
|
GlTooltip: GlTooltipDirective,
|
|
},
|
|
props: {
|
|
discussionPath: {
|
|
type: String,
|
|
required: false,
|
|
default: '',
|
|
},
|
|
diffFile: {
|
|
type: Object,
|
|
required: true,
|
|
},
|
|
collapsible: {
|
|
type: Boolean,
|
|
required: false,
|
|
default: false,
|
|
},
|
|
addMergeRequestButtons: {
|
|
type: Boolean,
|
|
required: false,
|
|
default: false,
|
|
},
|
|
expanded: {
|
|
type: Boolean,
|
|
required: false,
|
|
default: true,
|
|
},
|
|
canCurrentUserFork: {
|
|
type: Boolean,
|
|
required: true,
|
|
},
|
|
},
|
|
computed: {
|
|
...mapGetters('diffs', ['diffHasExpandedDiscussions', 'diffHasDiscussions']),
|
|
diffContentIDSelector() {
|
|
return `#diff-content-${this.diffFile.file_hash}`;
|
|
},
|
|
|
|
titleLink() {
|
|
if (this.diffFile.submodule) {
|
|
return this.diffFile.submodule_tree_url || this.diffFile.submodule_link;
|
|
}
|
|
|
|
if (!this.discussionPath) {
|
|
return this.diffContentIDSelector;
|
|
}
|
|
|
|
return this.discussionPath;
|
|
},
|
|
filePath() {
|
|
if (this.diffFile.submodule) {
|
|
return `${this.diffFile.file_path} @ ${truncateSha(this.diffFile.blob.id)}`;
|
|
}
|
|
|
|
if (this.diffFile.deleted_file) {
|
|
return sprintf(__('%{filePath} deleted'), { filePath: this.diffFile.file_path }, false);
|
|
}
|
|
|
|
return this.diffFile.file_path;
|
|
},
|
|
isUsingLfs() {
|
|
return this.diffFile.stored_externally && this.diffFile.external_storage === 'lfs';
|
|
},
|
|
collapseIcon() {
|
|
return this.expanded ? 'chevron-down' : 'chevron-right';
|
|
},
|
|
viewFileButtonText() {
|
|
const truncatedContentSha = _.escape(truncateSha(this.diffFile.content_sha));
|
|
return sprintf(
|
|
s__('MergeRequests|View file @ %{commitId}'),
|
|
{ commitId: truncatedContentSha },
|
|
false,
|
|
);
|
|
},
|
|
viewReplacedFileButtonText() {
|
|
const truncatedBaseSha = _.escape(truncateSha(this.diffFile.diff_refs.base_sha));
|
|
return sprintf(
|
|
s__('MergeRequests|View replaced file @ %{commitId}'),
|
|
{
|
|
commitId: `<span class="commit-sha">${truncatedBaseSha}</span>`,
|
|
},
|
|
false,
|
|
);
|
|
},
|
|
gfmCopyText() {
|
|
return `\`${this.diffFile.file_path}\``;
|
|
},
|
|
isFileRenamed() {
|
|
return this.diffFile.renamed_file;
|
|
},
|
|
isModeChanged() {
|
|
return this.diffFile.viewer.name === diffViewerModes.mode_changed;
|
|
},
|
|
expandDiffToFullFileTitle() {
|
|
if (this.diffFile.isShowingFullFile) {
|
|
return s__('MRDiff|Show changes only');
|
|
}
|
|
return s__('MRDiff|Show full file');
|
|
},
|
|
},
|
|
mounted() {
|
|
polyfillSticky(this.$refs.header);
|
|
},
|
|
methods: {
|
|
...mapActions('diffs', [
|
|
'toggleFileDiscussions',
|
|
'toggleFileDiscussionWrappers',
|
|
'toggleFullDiff',
|
|
]),
|
|
handleToggleFile() {
|
|
this.$emit('toggleFile');
|
|
},
|
|
showForkMessage() {
|
|
this.$emit('showForkMessage');
|
|
},
|
|
handleFileNameClick(e) {
|
|
const isLinkToOtherPage =
|
|
this.diffFile.submodule_tree_url || this.diffFile.submodule_link || this.discussionPath;
|
|
|
|
if (!isLinkToOtherPage) {
|
|
e.preventDefault();
|
|
const selector = this.diffContentIDSelector;
|
|
scrollToElement(document.querySelector(selector));
|
|
window.location.hash = selector;
|
|
}
|
|
},
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<div
|
|
ref="header"
|
|
class="js-file-title file-title file-title-flex-parent"
|
|
@click.self="handleToggleFile"
|
|
>
|
|
<div class="file-header-content">
|
|
<icon
|
|
v-if="collapsible"
|
|
ref="collapseIcon"
|
|
:name="collapseIcon"
|
|
:size="16"
|
|
aria-hidden="true"
|
|
class="diff-toggle-caret append-right-5"
|
|
@click.stop="handleToggleFile"
|
|
/>
|
|
<a
|
|
v-once
|
|
id="diffFile.file_path"
|
|
ref="titleWrapper"
|
|
class="append-right-4"
|
|
:href="titleLink"
|
|
@click="handleFileNameClick"
|
|
>
|
|
<file-icon
|
|
:file-name="filePath"
|
|
:size="18"
|
|
aria-hidden="true"
|
|
css-classes="append-right-5"
|
|
/>
|
|
<span v-if="isFileRenamed">
|
|
<strong
|
|
v-gl-tooltip
|
|
:title="diffFile.old_path"
|
|
class="file-title-name"
|
|
v-html="diffFile.old_path_html"
|
|
></strong>
|
|
→
|
|
<strong
|
|
v-gl-tooltip
|
|
:title="diffFile.new_path"
|
|
class="file-title-name"
|
|
v-html="diffFile.new_path_html"
|
|
></strong>
|
|
</span>
|
|
|
|
<strong v-else v-gl-tooltip :title="filePath" class="file-title-name" data-container="body">
|
|
{{ filePath }}
|
|
</strong>
|
|
</a>
|
|
|
|
<clipboard-button
|
|
:title="__('Copy file path')"
|
|
:text="diffFile.file_path"
|
|
:gfm="gfmCopyText"
|
|
css-class="btn-default btn-transparent btn-clipboard"
|
|
/>
|
|
|
|
<small v-if="isModeChanged" ref="fileMode" class="mr-1">
|
|
{{ diffFile.a_mode }} → {{ diffFile.b_mode }}
|
|
</small>
|
|
|
|
<span v-if="isUsingLfs" class="label label-lfs append-right-5"> {{ __('LFS') }} </span>
|
|
</div>
|
|
|
|
<div
|
|
v-if="!diffFile.submodule && addMergeRequestButtons"
|
|
class="file-actions d-none d-sm-block"
|
|
>
|
|
<diff-stats :added-lines="diffFile.added_lines" :removed-lines="diffFile.removed_lines" />
|
|
<div class="btn-group" role="group">
|
|
<template v-if="diffFile.blob && diffFile.blob.readable_text">
|
|
<span v-gl-tooltip.hover :title="s__('MergeRequests|Toggle comments for this file')">
|
|
<gl-button
|
|
ref="toggleDiscussionsButton"
|
|
:disabled="!diffHasDiscussions(diffFile)"
|
|
:class="{ active: diffHasExpandedDiscussions(diffFile) }"
|
|
class="js-btn-vue-toggle-comments btn"
|
|
data-qa-selector="toggle_comments_button"
|
|
type="button"
|
|
@click="toggleFileDiscussionWrappers(diffFile)"
|
|
>
|
|
<icon name="comment" />
|
|
</gl-button>
|
|
</span>
|
|
|
|
<edit-button
|
|
v-if="!diffFile.deleted_file"
|
|
:can-current-user-fork="canCurrentUserFork"
|
|
:edit-path="diffFile.edit_path"
|
|
:can-modify-blob="diffFile.can_modify_blob"
|
|
@showForkMessage="showForkMessage"
|
|
/>
|
|
</template>
|
|
|
|
<a
|
|
v-if="diffFile.replaced_view_path"
|
|
ref="replacedFileButton"
|
|
:href="diffFile.replaced_view_path"
|
|
class="btn view-file"
|
|
v-html="viewReplacedFileButtonText"
|
|
>
|
|
</a>
|
|
<gl-button
|
|
v-if="!diffFile.is_fully_expanded"
|
|
ref="expandDiffToFullFileButton"
|
|
v-gl-tooltip.hover
|
|
:title="expandDiffToFullFileTitle"
|
|
class="expand-file"
|
|
@click="toggleFullDiff(diffFile.file_path)"
|
|
>
|
|
<gl-loading-icon v-if="diffFile.isLoadingFullFile" color="dark" inline />
|
|
<icon v-else-if="diffFile.isShowingFullFile" name="doc-changes" />
|
|
<icon v-else name="doc-expand" />
|
|
</gl-button>
|
|
<gl-button
|
|
ref="viewButton"
|
|
v-gl-tooltip.hover
|
|
:href="diffFile.view_path"
|
|
target="blank"
|
|
class="view-file"
|
|
:title="viewFileButtonText"
|
|
>
|
|
<icon name="doc-text" />
|
|
</gl-button>
|
|
|
|
<a
|
|
v-if="diffFile.external_url"
|
|
ref="externalLink"
|
|
v-gl-tooltip.hover
|
|
:href="diffFile.external_url"
|
|
:title="`View on ${diffFile.formatted_external_url}`"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
class="btn btn-file-option"
|
|
>
|
|
<icon name="external-link" />
|
|
</a>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|