mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-11-02 20:06:06 +01:00 
			
		
		
		
	Auto merge pull requests when all checks succeeded via WebUI (#19648)
Add WebUI part of Auto merge feature close #19621 Co-authored-by: wxiaoguang <wxiaoguang@gmail.com> Co-authored-by: delvh <dev.lh@web.de>
This commit is contained in:
		@@ -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,39 +28,72 @@
 | 
			
		||||
          </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">
 | 
			
		||||
                {{ msd.textDoMerge }}
 | 
			
		||||
              <!-- 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>
 | 
			
		||||
                <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>
 | 
			
		||||
    </template>
 | 
			
		||||
 | 
			
		||||
      <!-- 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>
 | 
			
		||||
 | 
			
		||||
@@ -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>
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user