|
|
|
@ -1,9 +1,23 @@ |
|
|
|
|
<template> |
|
|
|
|
<!-- |
|
|
|
|
if this component is shown, either the user is admin (can do merge without checks), or they is a writer who has the permission to do merge |
|
|
|
|
if the user is a writer and can't do merge now (canMergeNow==false), then only show the Auto Merge for them |
|
|
|
|
How to test the UI manually: |
|
|
|
|
* Method 1: manually set some variables in pull.tmpl, eg: {{$notAllOverridableChecksOk = true}} {{$canMergeNow = false}} |
|
|
|
|
* Method 2: make a protected branch, then set state=pending/success : |
|
|
|
|
curl -X POST ${root_url}/api/v1/repos/${owner}/${repo}/statuses/${sha} \ |
|
|
|
|
-H "accept: application/json" -H "authorization: Basic $base64_auth" -H "Content-Type: application/json" \ |
|
|
|
|
-d '{"context": "test/context", "description": "description", "state": "${state}", "target_url": "http://localhost"}' |
|
|
|
|
--> |
|
|
|
|
<div> |
|
|
|
|
<!-- eslint-disable --> |
|
|
|
|
<div v-if="mergeForm.hasPendingPullRequestMerge" v-html="mergeForm.hasPendingPullRequestMergeTip" class="ui info message"></div> |
|
|
|
|
|
|
|
|
|
<div class="ui form" v-if="showActionForm"> |
|
|
|
|
<form :action="mergeForm.baseLink+'/merge'" method="post"> |
|
|
|
|
<input type="hidden" name="_csrf" :value="csrfToken"> |
|
|
|
|
<input type="hidden" name="head_commit_id" v-model="mergeForm.pullHeadCommitID"> |
|
|
|
|
<input type="hidden" name="merge_when_checks_succeed" v-model="autoMergeWhenSucceed"> |
|
|
|
|
|
|
|
|
|
<template v-if="!mergeStyleDetail.hideMergeMessageTexts"> |
|
|
|
|
<div class="field"> |
|
|
|
@ -14,40 +28,73 @@ |
|
|
|
|
</div> |
|
|
|
|
</template> |
|
|
|
|
|
|
|
|
|
<button class="ui button" :class="[mergeForm.allOverridableChecksOk?'green':'red']" type="submit" name="do" :value="mergeStyle"> |
|
|
|
|
<button class="ui button" :class="mergeButtonStyleClass" type="submit" name="do" :value="mergeStyle"> |
|
|
|
|
{{ mergeStyleDetail.textDoMerge }} |
|
|
|
|
<template v-if="autoMergeWhenSucceed"> |
|
|
|
|
{{ mergeForm.textAutoMergeButtonWhenSucceed }} |
|
|
|
|
</template> |
|
|
|
|
</button> |
|
|
|
|
|
|
|
|
|
<button class="ui button merge-cancel" @click="toggleActionForm(false)"> |
|
|
|
|
{{ mergeForm.textCancel }} |
|
|
|
|
</button> |
|
|
|
|
|
|
|
|
|
<div class="ui checkbox ml-2" v-if="mergeForm.isPullBranchDeletable"> |
|
|
|
|
<div class="ui checkbox ml-2" v-if="mergeForm.isPullBranchDeletable && !autoMergeWhenSucceed"> |
|
|
|
|
<input name="delete_branch_after_merge" type="checkbox" v-model="deleteBranchAfterMerge" id="delete-branch-after-merge"> |
|
|
|
|
<label for="delete-branch-after-merge">{{ mergeForm.textDeleteBranch }}</label> |
|
|
|
|
</div> |
|
|
|
|
</form> |
|
|
|
|
</div> |
|
|
|
|
|
|
|
|
|
<template v-if="!showActionForm"> |
|
|
|
|
<div class="ui buttons merge-button" :class="[mergeForm.allOverridableChecksOk?'green':'red']" @click="toggleActionForm(true)"> |
|
|
|
|
<div v-if="!showActionForm" class="df"> |
|
|
|
|
<!-- the merge button --> |
|
|
|
|
<div class="ui buttons merge-button" :class="mergeButtonStyleClass" @click="toggleActionForm(true)" > |
|
|
|
|
<button class="ui button"> |
|
|
|
|
<svg-icon name="octicon-git-merge"/> |
|
|
|
|
<span class="button-text">{{ mergeStyleDetail.textDoMerge }}</span> |
|
|
|
|
<span class="button-text"> |
|
|
|
|
{{ mergeStyleDetail.textDoMerge }} |
|
|
|
|
<template v-if="autoMergeWhenSucceed"> |
|
|
|
|
{{ mergeForm.textAutoMergeButtonWhenSucceed }} |
|
|
|
|
</template> |
|
|
|
|
</span> |
|
|
|
|
</button> |
|
|
|
|
<div class="ui dropdown icon button no-text" @click.stop="showMergeStyleMenu = !showMergeStyleMenu" v-if="mergeStyleAllowedCount>1"> |
|
|
|
|
<svg-icon name="octicon-triangle-down" :size="14"/> |
|
|
|
|
<div class="menu" :class="{'show':showMergeStyleMenu}"> |
|
|
|
|
<template v-for="msd in mergeForm.mergeStyles"> |
|
|
|
|
<div class="item" v-if="msd.allowed" :key="msd.name" @click.stop="mergeStyle=msd.name"> |
|
|
|
|
<!-- if can merge now, show one action "merge now", and an action "auto merge when succeed" --> |
|
|
|
|
<div class="item" v-if="msd.allowed && mergeForm.canMergeNow" :key="msd.name" @click.stop="switchMergeStyle(msd.name)"> |
|
|
|
|
<div class="action-text"> |
|
|
|
|
{{ msd.textDoMerge }} |
|
|
|
|
</div> |
|
|
|
|
</template> |
|
|
|
|
<div v-if="!msd.hideAutoMerge" class="auto-merge-small" @click.stop="switchMergeStyle(msd.name, true)"> |
|
|
|
|
<svg-icon name="octicon-clock" :size="14"/> |
|
|
|
|
<div class="auto-merge-tip"> |
|
|
|
|
{{ mergeForm.textAutoMergeWhenSucceed }} |
|
|
|
|
</div> |
|
|
|
|
</div> |
|
|
|
|
</div> |
|
|
|
|
|
|
|
|
|
<!-- if can NOT merge now, only show one action "auto merge when succeed" --> |
|
|
|
|
<div class="item" v-if="msd.allowed && !mergeForm.canMergeNow && !msd.hideAutoMerge" :key="msd.name" @click.stop="switchMergeStyle(msd.name, true)"> |
|
|
|
|
<div class="action-text"> |
|
|
|
|
{{ msd.textDoMerge }} {{ mergeForm.textAutoMergeButtonWhenSucceed }} |
|
|
|
|
</div> |
|
|
|
|
</div> |
|
|
|
|
</template> |
|
|
|
|
</div> |
|
|
|
|
</div> |
|
|
|
|
</div> |
|
|
|
|
|
|
|
|
|
<!-- the cancel auto merge button --> |
|
|
|
|
<form v-if="mergeForm.hasPendingPullRequestMerge" :action="mergeForm.baseLink+'/cancel_auto_merge'" method="post" class="ml-4"> |
|
|
|
|
<input type="hidden" name="_csrf" :value="csrfToken"> |
|
|
|
|
<button class="ui button"> |
|
|
|
|
{{ mergeForm.textAutoMergeCancelSchedule }} |
|
|
|
|
</button> |
|
|
|
|
</form> |
|
|
|
|
</div> |
|
|
|
|
</div> |
|
|
|
|
</template> |
|
|
|
|
|
|
|
|
|
<script> |
|
|
|
@ -68,6 +115,7 @@ export default { |
|
|
|
|
mergeTitleFieldValue: '', |
|
|
|
|
mergeMessageFieldValue: '', |
|
|
|
|
deleteBranchAfterMerge: false, |
|
|
|
|
autoMergeWhenSucceed: false, |
|
|
|
|
|
|
|
|
|
mergeStyle: '', |
|
|
|
|
mergeStyleDetail: { // dummy only, these values will come from one of the mergeForm.mergeStyles |
|
|
|
@ -82,6 +130,13 @@ export default { |
|
|
|
|
showActionForm: false, |
|
|
|
|
}), |
|
|
|
|
|
|
|
|
|
computed: { |
|
|
|
|
mergeButtonStyleClass() { |
|
|
|
|
if (this.mergeForm.allOverridableChecksOk) return 'green'; |
|
|
|
|
return this.autoMergeWhenSucceed ? 'blue' : 'red'; |
|
|
|
|
} |
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
watch: { |
|
|
|
|
mergeStyle(val) { |
|
|
|
|
this.mergeStyleDetail = this.mergeForm.mergeStyles.find((e) => e.name === val); |
|
|
|
@ -90,7 +145,7 @@ export default { |
|
|
|
|
|
|
|
|
|
created() { |
|
|
|
|
this.mergeStyleAllowedCount = this.mergeForm.mergeStyles.reduce((v, msd) => v + (msd.allowed ? 1 : 0), 0); |
|
|
|
|
this.mergeStyle = this.mergeForm.mergeStyles.find((e) => e.allowed)?.name; |
|
|
|
|
this.switchMergeStyle(this.mergeForm.mergeStyles.find((e) => e.allowed)?.name, !this.mergeForm.canMergeNow); |
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
mounted() { |
|
|
|
@ -111,7 +166,11 @@ export default { |
|
|
|
|
this.deleteBranchAfterMerge = this.mergeForm.defaultDeleteBranchAfterMerge; |
|
|
|
|
this.mergeTitleFieldValue = this.mergeStyleDetail.mergeTitleFieldText; |
|
|
|
|
this.mergeMessageFieldValue = this.mergeStyleDetail.mergeMessageFieldText; |
|
|
|
|
} |
|
|
|
|
}, |
|
|
|
|
switchMergeStyle(name, autoMerge = false) { |
|
|
|
|
this.mergeStyle = name; |
|
|
|
|
this.autoMergeWhenSucceed = autoMerge; |
|
|
|
|
}, |
|
|
|
|
}, |
|
|
|
|
}; |
|
|
|
|
</script> |
|
|
|
@ -124,4 +183,59 @@ export default { |
|
|
|
|
.ui.checkbox label { |
|
|
|
|
cursor: pointer; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/* make the dropdown list left-aligned */ |
|
|
|
|
.ui.merge-button { |
|
|
|
|
position: relative; |
|
|
|
|
} |
|
|
|
|
.ui.merge-button .ui.dropdown { |
|
|
|
|
position: static; |
|
|
|
|
} |
|
|
|
|
.ui.merge-button > .ui.dropdown:last-child > .menu:not(.left) { |
|
|
|
|
left: 0; |
|
|
|
|
right: auto; |
|
|
|
|
} |
|
|
|
|
.ui.merge-button .ui.dropdown .menu > .item { |
|
|
|
|
display: flex; |
|
|
|
|
align-items: stretch; |
|
|
|
|
padding: 0 !important; /* polluted by semantic.css: .ui.dropdown .menu > .item { !important } */ |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/* merge style list item */ |
|
|
|
|
.action-text { |
|
|
|
|
padding: 0.8rem; |
|
|
|
|
flex: 1 |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
.auto-merge-small { |
|
|
|
|
width: 40px; |
|
|
|
|
display: flex; |
|
|
|
|
align-items: center; |
|
|
|
|
justify-content: center; |
|
|
|
|
position: relative; |
|
|
|
|
} |
|
|
|
|
.auto-merge-small .auto-merge-tip { |
|
|
|
|
display: none; |
|
|
|
|
left: 38px; |
|
|
|
|
top: -1px; |
|
|
|
|
bottom: -1px; |
|
|
|
|
position: absolute; |
|
|
|
|
align-items: center; |
|
|
|
|
color: var(--color-info-text); |
|
|
|
|
background-color: var(--color-info-bg); |
|
|
|
|
border: 1px solid var(--color-info-border); |
|
|
|
|
border-left: none; |
|
|
|
|
padding-right: 1rem; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
.auto-merge-small:hover { |
|
|
|
|
color: var(--color-info-text); |
|
|
|
|
background-color: var(--color-info-bg); |
|
|
|
|
border: 1px solid var(--color-info-border); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
.auto-merge-small:hover .auto-merge-tip { |
|
|
|
|
display: flex; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
</style> |
|
|
|
|