您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Enhance Jira Features: Copy Commit Message
// ==UserScript== // @name Jira Enhance // @namespace https://github.com/codeshareman/useful-scripts.git // @version 2025-05-11 // @description Enhance Jira Features: Copy Commit Message // @author zz_captain_zz // @match // TODO: match your jira issue link // @include // TODO: match your jira issue link // @icon  // @grant GM.setClipboard // @grant GM.addStyle // @run-at document-end // @license MIT // ==/UserScript== const DELAY_MILLI_SECONDS = 1000; const style = ` #jira-enhance-copy-button { position: relative; color: var(--jira-issue-status-default-color); background-color: var(--jira-issue-status-default-bgcolor); height: 24px; display: flex; align-items: center; justify-content: center; padding: 3px 12px; margin-top: 10px; border-radius: 3.01px; cursor: pointer; transition: background-color 0.3s ease; } #jira-enhance-copy-button:hover { background-color: var(--jira-issue-status-hover-default-bgcolor); } #jira-enhance-copy-button-message { position: absolute; padding: 4px 8px; border-radius: 3px; bottom: -100%; left: 50%; transform: translateX(-50%); background-color: #e6e6e6; font-size: 12px; color: var(--jira-issue-status-default-color); z-index: 1000; color: var(--jira-issue-status-default-color);; opacity: 0; transition: opacity 0.3s ease; } #jira-enhance-copy-button-message.active { opacity: 1; } #jira-enhance-copy-button-message::before { content: ""; position: absolute; top: -5px; left: 50%; width: 0; height: 0; border-left: 5px solid transparent; border-right: 5px solid transparent; border-bottom: 5px solid#e6e6e6; transform: translateX(-50%); } `; GM.addStyle(style); (function () { "use strict"; function runScript( config = { summaryId: "#summary-val", issueKeyId: "#key-val", toolbarPrimaryId: "#stalker > div > div.command-bar > div > div > div > div.aui-toolbar2-primary", } ) { try { // Your code here... const $summary = document.querySelector(config.summaryId); const $issue_key = document.querySelector(config.issueKeyId); const $toolbar_primary = document.querySelector(config.toolbarPrimaryId); const metadata = { issue_key: $issue_key.textContent, summary: $summary.textContent, }; /** * Git Enhance */ const git = { generateCommitMsg(issueKey, summary, options = { prefix: "fix:" }) { const { prefix } = options; const commitMsg = `${prefix} [${issueKey}] ${summary}`; return commitMsg; }, }; /** * Jira Enhance */ const jira = { getIssueInfo(metadata) { return { issue_key: metadata.issue_key, summary: metadata.summary, commit_msg: git.generateCommitMsg( metadata.issue_key, metadata.summary ), }; }, logIssueInfo(info) { group("=== Jira Enhance Message ==="); Object.entries(info).forEach(([key, value]) => { log("info")(key, value); }); groupEnd(); }, }; const issueInfo = jira.getIssueInfo(metadata); // 添加复制按钮 appendCopyButton($toolbar_primary, issueInfo.commit_msg); // 打印issue信息 jira.logIssueInfo(issueInfo); } catch (error) { log("error")(error); } } window.addEventListener("load", () => { runScript(); }); })(); // DOM function appendCopyButton(target, copyText = "") { if (!target) { throw new Error("target is required"); } const $button = createToolbarButton({ onClick: () => { GM.setClipboard(JSON.stringify(copyText)); const $message = $button.querySelector( "#jira-enhance-copy-button-message" ); $message.classList.add("active"); setTimeout(() => { $message.classList.remove("active"); }, DELAY_MILLI_SECONDS); log("info")("Copy to clipboard"); }, }); appendToolbarButton(target, $button); } function createToolbarButton(buttonProperties) { const defaultProperties = { id: "jira-enhance-copy-button", icon: "fa-solid fa-clipboard", text: "复制提交信息", onClick: () => {}, }; const { id, icon, text, onClick } = { ...defaultProperties, ...buttonProperties, }; const $button = document.createElement("div"); const $message = document.createElement("div"); $button.id = id; $button.innerHTML = `<i class="${icon}"></i> ${text}`; $button.addEventListener("click", onClick); $message.id = `${id}-message`; $message.innerHTML = `Copied`; $button.appendChild($message); return $button; } function appendToolbarButton(target, button) { if (!target) { throw new Error("target is required"); } if (!button) { throw new Error("button is required"); } const isCopyButtonExist = target.querySelector(`#${button.id}`); if (isCopyButtonExist) return; target.appendChild(button); } /** * Common Utils */ function debounce( fn, options = { delay: DELAY_MILLI_SECONDS, immediate: false } ) { const { delay, immediate } = options; let timer = null; return function (...args) { if (timer) { clearTimeout(timer); } if (immediate && !timer) { fn.apply(this, args); timer = setTimeout(() => { timer = null; }, delay); } else { timer = setTimeout(() => { fn.apply(this, args); timer = null; }, delay); } }; } function throttle( fn, options = { delay: DELAY_MILLI_SECONDS, immediate: false } ) { const { delay, immediate } = options; let timer = null; let isFirst = true; return function (...args) { if (timer) return; if (immediate && isFirst) { fn.apply(this, args); isFirst = false; } timer = setTimeout(() => { fn.apply(this, args); timer = null; }, delay); }; } /** * Console Utils */ function log(type) { const typeColor = { info: "#0000FF", warn: "#FFA500", error: "#FF0000", }; return function (...args) { const color = typeColor[type] || "#000"; const style = `background-color: ${color}; color: #fff;`; console.error(`%c[${type}]`, style, ...args); }; } function group(...args) { const style = `background-color:#cf1fdc; color: #fff;`; console.group.call(console, `%c${args[0]}`, style, ...args.slice(1)); } function groupEnd(...args) { console.groupEnd.call(console, ...args); }