// ==UserScript== // @name PR Merge Control // @namespace https://userjs.khuong.dev // @version 1.4 // @description Automatically disable regular/squash merge on target branch for GitHub PRs // @icon64  // @author Lam Ngoc Khuong // @updateURL https://raw.githubusercontent.com/lamngockhuong/userjs/main/scripts/github/pr-merge-control.user.js // @downloadURL https://raw.githubusercontent.com/lamngockhuong/userjs/main/scripts/github/pr-merge-control.user.js // @match https://github.com/*/pull/* // @grant none // ==/UserScript== (function () { 'use strict'; const RELEASE_BRANCHES = ['main', 'master']; function getTargetBranch() { const selectors = [ '.base-ref .css-truncate-target', '[data-hovercard-type="repository"] + span .css-truncate-target', '.commit-ref.base-ref .css-truncate-target', 'span.commit-ref.css-truncate.user-select-contain.expandable.base-ref span.css-truncate-target' ]; for (const selector of selectors) { const el = document.querySelector(selector); if (el?.textContent) { return el.textContent.trim(); } } const mergeInfo = document.querySelector('.merge-message .commit-ref'); if (mergeInfo?.textContent) { return mergeInfo.textContent.trim(); } return null; } function isReleaseBranch(branch) { return RELEASE_BRANCHES.includes(branch); } function shouldDisableMethod(methodLabel, isRelease) { const label = methodLabel.toLowerCase(); const isMergeCommit = label.includes('merge commit') || label === 'merge pull request'; const isSquash = label.includes('squash'); const isRebase = label.includes('rebase'); if (isRelease) { // Release branch: only allow merge commit return isSquash || isRebase; } else { // Feature branch: only allow squash return isMergeCommit || isRebase; } } function disableElement(el) { el.style.opacity = '0.35'; el.style.pointerEvents = 'none'; el.style.cursor = 'not-allowed'; el.setAttribute('aria-disabled', 'true'); if (el.tagName === 'BUTTON') { el.disabled = true; } } function enableElement(el) { el.style.opacity = ''; el.style.pointerEvents = ''; el.style.cursor = ''; el.removeAttribute('aria-disabled'); if (el.tagName === 'BUTTON') { el.disabled = false; } } function addDisabledBadge(container) { if (container.querySelector('.pr-merge-disabled-badge')) return; const badge = document.createElement('span'); badge.className = 'pr-merge-disabled-badge'; badge.style.cssText = 'color: #cf222e; font-size: 0.85em; margin-left: 6px;'; badge.textContent = '(disabled)'; container.appendChild(badge); } function removeDisabledBadge(container) { const badge = container.querySelector('.pr-merge-disabled-badge'); if (badge) badge.remove(); } // Disable dropdown menu items function applyMenuRestrictions(menu, targetBranch) { const items = menu.querySelectorAll('li[role="menuitemradio"]'); if (items.length === 0) return; const isRelease = isReleaseBranch(targetBranch); items.forEach((item) => { const labelEl = item.querySelector('[class*="ItemLabel"]'); const label = labelEl?.textContent?.trim(); if (!label) return; enableElement(item); removeDisabledBadge(labelEl); if (shouldDisableMethod(label, isRelease)) { disableElement(item); addDisabledBadge(labelEl); } }); console.log(`[PR Merge Control] Menu restricted for: ${targetBranch}`); } // Disable main merge button function applyMainButtonRestriction(targetBranch) { // Find all buttons with merge-related text const allButtons = document.querySelectorAll('button.flex-1[data-variant="primary"]'); for (const mainButton of allButtons) { const labelEl = mainButton.querySelector('[data-component="text"]'); const label = labelEl?.textContent?.trim(); if (!label) continue; // Check if this is a merge-related button const lowerLabel = label.toLowerCase(); if (!lowerLabel.includes('merge') && !lowerLabel.includes('squash') && !lowerLabel.includes('rebase')) { continue; } const isRelease = isReleaseBranch(targetBranch); enableElement(mainButton); if (shouldDisableMethod(label, isRelease)) { disableElement(mainButton); console.log(`[PR Merge Control] Main button disabled: "${label}"`); } } } function observeDOM() { const targetBranch = getTargetBranch(); const observer = new MutationObserver((mutations) => { const branch = getTargetBranch(); if (!branch) return; for (const mutation of mutations) { for (const node of mutation.addedNodes) { if (node.nodeType !== Node.ELEMENT_NODE) continue; // Check for dropdown menu const menu = node.querySelector?.('ul[role="menu"]') || (node.matches?.('ul[role="menu"]') ? node : null); if (menu && menu.querySelector('li[aria-keyshortcuts="c"]')) { setTimeout(() => applyMenuRestrictions(menu, branch), 10); } // Check for button group (merge button area) - use partial class match for stability if ( node.querySelector?.('[class*="ButtonGroup"]') || node.matches?.('[class*="ButtonGroup"]') || node.closest?.('[class*="ButtonGroup"]') ) { setTimeout(() => applyMainButtonRestriction(branch), 50); } } } // Also check main button on any mutation in merge area applyMainButtonRestriction(branch); }); observer.observe(document.body, { childList: true, subtree: true }); return observer; } // Poll until merge box is found and restriction applied function pollForMergeBox(maxAttempts = 30, interval = 500) { let attempts = 0; let applied = false; const poll = () => { const branch = getTargetBranch(); if (!branch) { if (attempts < maxAttempts) { attempts++; setTimeout(poll, interval); } return; } const mainButton = document.querySelector('button.flex-1[data-variant="primary"]'); if (mainButton && !applied) { applyMainButtonRestriction(branch); applied = true; console.log(`[PR Merge Control] Applied via polling after ${attempts} attempts`); } else if (attempts < maxAttempts) { attempts++; setTimeout(poll, interval); } }; poll(); } function init() { observeDOM(); pollForMergeBox(); console.log('[PR Merge Control] v1.4 Initialized'); } if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', init); } else { init(); } })();