您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Sets GitHub PR tab icon (favicon) to author's avatar
- // ==UserScript==
- // @name GitHub: PR author avatar as tab icon
- // @namespace https://github.com/rybak
- // @version 6
- // @description Sets GitHub PR tab icon (favicon) to author's avatar
- // @author Andrei Rybak
- // @homepageURL https://github.com/rybak/github-pr-avatars-tab-icons
- // @license MIT
- // @match https://github.com/*/pull/*
- // @icon https://github.githubassets.com/favicons/favicon-dark.png
- // @grant none
- // ==/UserScript==
- /*
- * Copyright (c) 2023 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.
- */
- /*
- * Doesn't work very good, because GitHub's tab icons aren't static.
- * Results of GitHub Actions builds are reflected in the icons dynamically.
- */
- (function() {
- 'use strict';
- const LOG_PREFIX = '[PR avatars]';
- function error(...toLog) {
- console.error(LOG_PREFIX, ...toLog);
- }
- function log(...toLog) {
- console.log(LOG_PREFIX, ...toLog);
- }
- /*
- * Extracts the parts 'owner_repo' and 'pull_number' out of the current page's URL.
- */
- function getPullRequestUrlParts() {
- /*
- * URL might be just a PR link:
- * https://github.com/rybak/atlassian-tweaks/pull/10
- * or a subpage of a PR:
- * https://github.com/rybak/atlassian-tweaks/pull/10/commits
- * Result will be
- * 'rybak/atlassian-tweaks' and '10'
- */
- const m = document.location.pathname.match("^/(.*)/pull/(\\d+)");
- if (!m) {
- error("Cannot extract owner_repo and pull_number for REST API URL from", document.location.pathname);
- return null;
- }
- return {
- 'owner_repo': m[1],
- 'pull_number': m[2]
- };
- }
- /*
- * Generates a URL for accessing the data about the pull request via GitHub's REST API.
- * https://docs.github.com/en/rest/pulls/pulls?apiVersion=2022-11-28#get-a-pull-request
- */
- function getRestApiPullRequestUrl() {
- const parts = getPullRequestUrlParts();
- if (!parts) {
- return null;
- }
- return `https://api.github.com/repos/${parts.owner_repo}/pulls/${parts.pull_number}`;
- }
- /*
- * Replaces favicon with the PR author's avatar.
- */
- async function setFavicon() {
- const prUrl = getRestApiPullRequestUrl();
- if (!prUrl) {
- return;
- }
- try {
- const response = await fetch(prUrl);
- const json = await response.json();
- const avatarUrl = json.user.avatar_url;
- if (!avatarUrl) {
- error("Cannot find the avatar URL", json);
- return;
- }
- const faviconNodes = document.querySelectorAll('link[rel="icon"], link[rel="alternate icon"]');
- if (!faviconNodes || faviconNodes.length == 0) {
- error("Cannot find favicon elements.");
- return;
- }
- log("New URL", avatarUrl);
- faviconNodes.forEach(node => {
- log("Replacing old URL =", node.href);
- node.href = avatarUrl;
- });
- } catch (e) {
- error(`Cannot load ${prUrl}. Got error`, e);
- }
- }
- setFavicon();
- })();