您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Adds convenience links in PRs of Bitbucket v7.6.+
// ==UserScript== // @name Bitbucket: commit links in diff tab of PRs // @namespace https://github.com/rybak/atlassian-tweaks // @version 16 // @license MIT // @description Adds convenience links in PRs of Bitbucket v7.6.+ // @author Andrei Rybak // @include https://*bitbucket*/*/repos/*/pull-requests/* // @match https://bitbucket.example.com/*/repos/*/pull-requests/* // @icon https://bitbucket.org/favicon.ico // @homepageURL https://github.com/rybak/atlassian-tweaks // @grant none // ==/UserScript== /* * Copyright (c) 2021-2025 Andrei Rybak * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ (function() { 'use strict'; const LOG_PREFIX = "[PR commit links]"; function warn(...toLog) { console.warn(LOG_PREFIX, ...toLog); } function log(...toLog) { console.log(LOG_PREFIX, ...toLog); } const ABBREV_LEN = 8; // abbreviate commit hashes to this number of characters const BLOCK_ID = 'RybakCommitLinkDiv'; const URL_ID = 'RybakCommitLinkA'; const TOOLTIP_BLOCK_ID = 'RybakCommitMessageDiv'; const TOOLTIP_MSG_ID = 'RybakCommitMessagePre'; const parsePath = /[/](projects|users)[/]([^/]*)[/]repos[/]([^/]*)[/].*[/]commits[/]([0-9a-f]+)/ function createTooltip(message) { $('#' + BLOCK_ID).hover((e) => { $('#' + TOOLTIP_BLOCK_ID).remove(); // delete previous tooltip const tooltipHtml = $('<div id="' + TOOLTIP_BLOCK_ID + '" class="Tooltip sc-jnlKLf ghcsui sc-bZQynM WXFrO sc-EHOje cffcMV"' + 'style="z-index:800; opacity: 1; position: fixed; top: 0px; left: 0px;' + // tweaked original element.style 'max-width: 600px; width: auto;' + // override of .ghcsui for better fitting of text 'background-color: rgb(23, 43, 77); border-radius: 3px; box-sizing: border-box; color: rgb(255, 255, 255); font-size: 12px; ' + // from .WXFrO 'line-height: 1.3; padding: 2px 6px; overflow-wrap: break-word;' + // from .WXFrO 'pointer-events: none;' + // from .cffcMV '">' + '<pre id="' + TOOLTIP_MSG_ID + '" class="commit-message-tooltip" style="' + 'white-space: pre-wrap; word-break: break-word;' + // from .commit-message-tooltip '"></pre>' + '</div>'); $($('.atlaskit-portal-container')[0]).append(tooltipHtml); $('#' + TOOLTIP_MSG_ID).text(message); // text added early to calculate height correctly const width = $('#' + TOOLTIP_BLOCK_ID).outerWidth(); const height = $('#' + TOOLTIP_BLOCK_ID).height(); const block = $('#' + BLOCK_ID); const blockOffset = block.offset(); var x = blockOffset.left; var y = blockOffset.top + block.height() + 8; // 8 is from CSS rule ".changes-scope-actions > *" const maxX = $(window).width() + window.pageXOffset; const maxY = $(window).height() + window.pageYOffset; if (x + width > maxX) { x = Math.max(maxX - width, 0); } if (y + height > maxY) { y = Math.max(maxY - height, 0); } $('#' + TOOLTIP_BLOCK_ID).css({left: x, top: y}).show(); }, (e) => { $('#' + TOOLTIP_BLOCK_ID).hide(); }); } function ensureCommitLink(label) { const matching = document.location.pathname.match(parsePath); if (!matching) { log(label, "No commit in the URL: " + document.location.pathname); return; } const origin = document.location.origin; const hash = document.location.hash; // add hash in case the user clicked to a different file const projectOrUser = matching[1]; const project = matching[2]; const repository = matching[3]; /* * TODO: keep track of `commit` and avoid recreating everything if * `commit` hasn't changed. */ const commit = matching[4]; log(label, "Parsed " + project + "/" + repository + "/" + commit); const url = origin + '/' + projectOrUser + '/' + project + '/repos/' + repository + '/commits/' + commit + document.location.hash; const linkText = commit.substring(0, ABBREV_LEN); log(label, "Link: " + url); log(label, "Text: " + linkText); const prevBlock = $('#' + BLOCK_ID); if (prevBlock.length) { log(label, "Updating the link..."); } else { const searchCodeButton = document.querySelector('#main .changes-scope-actions [data-testid="search-action-button-tooltip--container"] button'); const container = document.createElement('div'); container.id = BLOCK_ID; container.classList.add(searchCodeButton.classList[0]); const link = document.createElement('a'); link.id = URL_ID; container.appendChild(link); document.querySelector('.changes-scope-actions').append(container); log(label, "Creating the link..."); } $('#' + URL_ID) .attr('href', url) .text(linkText); const restApiUrl = document.location.origin + "/rest/api/1.0/" + projectOrUser + "/" + project + '/repos/' + repository + '/commits/' + commit; log(label, "Ajax...: " + restApiUrl); $.ajax({ // https://docs.atlassian.com/bitbucket-server/rest/7.6.0/bitbucket-rest.html#idp224 url: (restApiUrl) }).then(data => { log(label, "Ajax response received"); if (!document.getElementById(BLOCK_ID)) { warn(label, `Something happened to #${BLOCK_ID}. Re-creating...`); ensureCommitLink("[smth happened]"); } createTooltip(data.message); }); log(label, "Done"); } $(document).ready(function() { ensureCommitLink("[document.ready]"); window.onpopstate = function(event) { ensureCommitLink("[onpopstate]"); }; }); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址