Commit 762862cb authored by Phil Hughes's avatar Phil Hughes

Merge branch '31236-auto-update-merge-widget' into 'master'

Auto update mr widget when new commits are pushed

Closes #31236

See merge request gitlab-org/gitlab!19725
parents f1936f5a 7627e673
...@@ -3,8 +3,9 @@ import _ from 'underscore'; ...@@ -3,8 +3,9 @@ import _ from 'underscore';
import successSvg from 'icons/_icon_status_success.svg'; import successSvg from 'icons/_icon_status_success.svg';
import warningSvg from 'icons/_icon_status_warning.svg'; import warningSvg from 'icons/_icon_status_warning.svg';
import simplePoll from '~/lib/utils/simple_poll'; import simplePoll from '~/lib/utils/simple_poll';
import { __ } from '~/locale'; import { __, sprintf } from '~/locale';
import readyToMergeMixin from 'ee_else_ce/vue_merge_request_widget/mixins/ready_to_merge'; import readyToMergeMixin from 'ee_else_ce/vue_merge_request_widget/mixins/ready_to_merge';
import { GlIcon } from '@gitlab/ui';
import MergeRequest from '../../../merge_request'; import MergeRequest from '../../../merge_request';
import { refreshUserMergeRequestCounts } from '~/commons/nav/user_merge_requests'; import { refreshUserMergeRequestCounts } from '~/commons/nav/user_merge_requests';
import Flash from '../../../flash'; import Flash from '../../../flash';
...@@ -24,6 +25,7 @@ export default { ...@@ -24,6 +25,7 @@ export default {
CommitsHeader, CommitsHeader,
CommitEdit, CommitEdit,
CommitMessageDropdown, CommitMessageDropdown,
GlIcon,
}, },
mixins: [readyToMergeMixin], mixins: [readyToMergeMixin],
props: { props: {
...@@ -111,6 +113,18 @@ export default { ...@@ -111,6 +113,18 @@ export default {
shouldShowMergeEdit() { shouldShowMergeEdit() {
return !this.mr.ffOnlyEnabled; return !this.mr.ffOnlyEnabled;
}, },
shaMismatchLink() {
const href = this.mr.mergeRequestDiffsPath;
return sprintf(
__('New changes were added. %{linkStart}Reload the page to review them%{linkEnd}'),
{
linkStart: `<a href="${href}">`,
linkEnd: '</a>',
},
false,
);
},
}, },
methods: { methods: {
updateMergeCommitMessage(includeDescription) { updateMergeCommitMessage(includeDescription) {
...@@ -123,7 +137,7 @@ export default { ...@@ -123,7 +137,7 @@ export default {
} }
const options = { const options = {
sha: this.mr.sha, sha: this.mr.latestSHA || this.mr.sha,
commit_message: this.commitMessage, commit_message: this.commitMessage,
auto_merge_strategy: useAutoMerge ? this.mr.preferredAutoMergeStrategy : undefined, auto_merge_strategy: useAutoMerge ? this.mr.preferredAutoMergeStrategy : undefined,
should_remove_source_branch: this.removeSourceBranch === true, should_remove_source_branch: this.removeSourceBranch === true,
...@@ -314,6 +328,10 @@ export default { ...@@ -314,6 +328,10 @@ export default {
</template> </template>
</div> </div>
</div> </div>
<div v-if="mr.isSHAMismatch" class="d-flex align-items-center mt-2 js-sha-mismatch">
<gl-icon name="warning-solid" class="text-warning mr-1" />
<span class="text-warning" v-html="shaMismatchLink"></span>
</div>
</div> </div>
</div> </div>
<template v-if="shouldShowMergeControls"> <template v-if="shouldShowMergeControls">
......
...@@ -25,7 +25,6 @@ import NothingToMergeState from './components/states/nothing_to_merge.vue'; ...@@ -25,7 +25,6 @@ import NothingToMergeState from './components/states/nothing_to_merge.vue';
import MissingBranchState from './components/states/mr_widget_missing_branch.vue'; import MissingBranchState from './components/states/mr_widget_missing_branch.vue';
import NotAllowedState from './components/states/mr_widget_not_allowed.vue'; import NotAllowedState from './components/states/mr_widget_not_allowed.vue';
import ReadyToMergeState from './components/states/ready_to_merge.vue'; import ReadyToMergeState from './components/states/ready_to_merge.vue';
import ShaMismatchState from './components/states/sha_mismatch.vue';
import UnresolvedDiscussionsState from './components/states/unresolved_discussions.vue'; import UnresolvedDiscussionsState from './components/states/unresolved_discussions.vue';
import PipelineBlockedState from './components/states/mr_widget_pipeline_blocked.vue'; import PipelineBlockedState from './components/states/mr_widget_pipeline_blocked.vue';
import PipelineFailedState from './components/states/pipeline_failed.vue'; import PipelineFailedState from './components/states/pipeline_failed.vue';
...@@ -63,7 +62,7 @@ export default { ...@@ -63,7 +62,7 @@ export default {
'mr-widget-not-allowed': NotAllowedState, 'mr-widget-not-allowed': NotAllowedState,
'mr-widget-missing-branch': MissingBranchState, 'mr-widget-missing-branch': MissingBranchState,
'mr-widget-ready-to-merge': ReadyToMergeState, 'mr-widget-ready-to-merge': ReadyToMergeState,
'sha-mismatch': ShaMismatchState, 'sha-mismatch': ReadyToMergeState,
'mr-widget-checking': CheckingState, 'mr-widget-checking': CheckingState,
'mr-widget-unresolved-discussions': UnresolvedDiscussionsState, 'mr-widget-unresolved-discussions': UnresolvedDiscussionsState,
'mr-widget-pipeline-blocked': PipelineBlockedState, 'mr-widget-pipeline-blocked': PipelineBlockedState,
......
...@@ -48,6 +48,7 @@ export default class MergeRequestStore { ...@@ -48,6 +48,7 @@ export default class MergeRequestStore {
this.commits = data.commits_without_merge_commits || []; this.commits = data.commits_without_merge_commits || [];
this.squashCommitMessage = data.default_squash_commit_message; this.squashCommitMessage = data.default_squash_commit_message;
this.rebaseInProgress = data.rebase_in_progress; this.rebaseInProgress = data.rebase_in_progress;
this.mergeRequestDiffsPath = data.diffs_path;
if (data.issues_links) { if (data.issues_links) {
const links = data.issues_links; const links = data.issues_links;
...@@ -81,6 +82,7 @@ export default class MergeRequestStore { ...@@ -81,6 +82,7 @@ export default class MergeRequestStore {
this.isOpen = data.state === 'opened'; this.isOpen = data.state === 'opened';
this.hasMergeableDiscussionsState = data.mergeable_discussions_state === false; this.hasMergeableDiscussionsState = data.mergeable_discussions_state === false;
this.isSHAMismatch = this.sha !== data.diff_head_sha; this.isSHAMismatch = this.sha !== data.diff_head_sha;
this.latestSHA = data.diff_head_sha;
this.canBeMerged = data.can_be_merged || false; this.canBeMerged = data.can_be_merged || false;
this.isMergeAllowed = data.mergeable || false; this.isMergeAllowed = data.mergeable || false;
this.mergeOngoing = data.merge_ongoing; this.mergeOngoing = data.merge_ongoing;
......
...@@ -70,6 +70,10 @@ class MergeRequestPollCachedWidgetEntity < IssuableEntity ...@@ -70,6 +70,10 @@ class MergeRequestPollCachedWidgetEntity < IssuableEntity
presenter(merge_request).source_branch_with_namespace_link presenter(merge_request).source_branch_with_namespace_link
end end
expose :diffs_path do |merge_request|
diffs_project_merge_request_path(merge_request.project, merge_request)
end
private private
delegate :current_user, to: :request delegate :current_user, to: :request
......
---
title: Allow merge without refresh when new commits are pushed
merge_request: 19725
author:
type: changed
...@@ -11461,6 +11461,9 @@ msgstr "" ...@@ -11461,6 +11461,9 @@ msgstr ""
msgid "New branch unavailable" msgid "New branch unavailable"
msgstr "" msgstr ""
msgid "New changes were added. %{linkStart}Reload the page to review them%{linkEnd}"
msgstr ""
msgid "New deploy key" msgid "New deploy key"
msgstr "" msgstr ""
......
...@@ -26,6 +26,7 @@ ...@@ -26,6 +26,7 @@
"has_conflicts": { "type": "boolean" }, "has_conflicts": { "type": "boolean" },
"can_be_merged": { "type": "boolean" }, "can_be_merged": { "type": "boolean" },
"remove_source_branch": { "type": ["boolean", "null"] }, "remove_source_branch": { "type": ["boolean", "null"] },
"diffs_path": { "type": "string" },
"source_branch_exists": { "type": "boolean" }, "source_branch_exists": { "type": "boolean" },
"branch_missing": { "type": "boolean" }, "branch_missing": { "type": "boolean" },
"commits_without_merge_commits": { "type": "array" }, "commits_without_merge_commits": { "type": "array" },
......
...@@ -938,4 +938,31 @@ describe('ReadyToMerge', () => { ...@@ -938,4 +938,31 @@ describe('ReadyToMerge', () => {
expect(customVm.$el.querySelector('.js-modify-commit-message-button')).toBeNull(); expect(customVm.$el.querySelector('.js-modify-commit-message-button')).toBeNull();
}); });
}); });
describe('with a mismatched SHA', () => {
const findMismatchShaBlock = () => vm.$el.querySelector('.js-sha-mismatch');
beforeEach(() => {
vm = createComponent({
mr: {
isSHAMismatch: true,
mergeRequestDiffsPath: '/merge_requests/1/diffs',
},
});
});
it('displays a warning message', () => {
expect(findMismatchShaBlock()).toExist();
});
it('warns the user to refresh to review', () => {
expect(findMismatchShaBlock().textContent.trim()).toBe(
'New changes were added. Reload the page to review them',
);
});
it('displays link to the diffs tab', () => {
expect(findMismatchShaBlock().querySelector('a').href).toContain(vm.mr.mergeRequestDiffsPath);
});
});
}); });
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