// ==UserScript==
// @name GitHub Custom Global Navigation
// @namespace https://github.com/blakegearin/github-custom-global-navigation
// @version 1.5.1
// @description Customize GitHub's new global navigation
// @author Blake Gearin
// @match *://github.com/*
// @require https://openuserjs.org/src/libs/sizzle/GM_config.js
// @grant GM.getValue
// @grant GM.setValue
// @license MIT
// @icon https://raw.githubusercontent.com/blakegearin/github-custom-global-navigation/main/img/white_logo.svg
// @supportURL https://github.com/blakegearin/github-custom-global-navigation/issues
// ==/UserScript==
/*global GM_config*/
(function () {
'use strict';
const SILENT = 0;
const QUIET = 1;
const INFO = 2;
const DEBUG = 3;
const VERBOSE = 4;
const TRACE = 5;
let CURRENT_LOG_LEVEL = QUIET;
const USERSCRIPT_NAME = 'GitHub Custom Global Navigation';
function log(level, message, variable = -1) {
if (CURRENT_LOG_LEVEL < level) return;
console.log(`${USERSCRIPT_NAME}: ${message}`);
if (variable !== -1) console.log(variable);
}
function logError(message, variable = null) {
console.error(`${USERSCRIPT_NAME}: ${message}`);
if (variable) console.log(variable);
}
log(TRACE, 'Starting');
function updateHeader() {
log(DEBUG, 'updateHeader()');
if (CONFIG.backgroundColor !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.header.self}
{
background-color: ${CONFIG.backgroundColor} !important;
}
`;
}
updateHamburgerButton();
updateLogo();
if (CONFIG.repositoryHeader.import) importRepositoryHeader();
updatePageTitle();
updateSearch();
if (CONFIG.divider.remove) removeDivider();
if (CONFIG.marketplace.add) createMarketplaceLink();
if (CONFIG.explore.add) createExploreLink();
updateLink('issues');
updateLink('pullRequests');
if (CONFIG.marketplace.add) updateLink('marketplace');
if (CONFIG.explore.add) updateLink('explore');
if (CONFIG.flipIssuesPullRequests) flipIssuesPullRequests();
updateCreateNewButton();
updateInboxLink();
if (CONFIG.flipCreateInbox) flipCreateInbox();
updateAvatar();
updateGlobalBar();
updateLocalBar();
updateSidebars();
modifyThenObserve(() => {
document.body.appendChild(HEADER_STYLE);
});
}
function updateHamburgerButton() {
log(DEBUG, 'updateHamburgerButton()');
const configKey = 'hamburgerButton';
const elementConfig = CONFIG.hamburgerButton;
log(DEBUG, 'elementConfig', elementConfig);
const hamburgerButton = HEADER.querySelector(SELECTORS[configKey]);
if (!hamburgerButton) {
logError(`Selector '${SELECTORS[configKey]}' not found`);
return;
}
if (elementConfig.remove) {
HEADER_STYLE.textContent += cssHideElement(SELECTORS[configKey]);
return;
}
}
function updateLogo() {
log(DEBUG, 'updateLogo()');
const configKey = 'logo';
const elementConfig = CONFIG[configKey];
const elementSelector = SELECTORS[configKey];
if (elementConfig.remove) {
HEADER_STYLE.textContent += cssHideElement(elementSelector.topDiv);
}
const logo = HEADER.querySelector(elementSelector.svg);
if (elementConfig.color !== '') {
HEADER_STYLE.textContent += `
${elementSelector.svg} path
{
fill: ${elementConfig.color} !important;
}
`;
}
if (elementConfig.customSvg !== '') {
const oldSvg = logo;
let newSvg;
if (isValidURL(elementConfig.customSvg)) {
newSvg = document.createElement('img');
newSvg.src = elementConfig.customSvg;
} else {
const parser = new DOMParser();
const svgDoc = parser.parseFromString(elementConfig.customSvg, 'image/svg+xml');
newSvg = svgDoc.documentElement;
}
oldSvg.parentNode.replaceChild(newSvg, oldSvg);
}
}
function removePageTitle() {
HEADER_STYLE.textContent += cssHideElement(createId(SELECTORS.pageTitle.id));
}
function updatePageTitle() {
log(DEBUG, 'updatePageTitle()');
const elementConfig = CONFIG.pageTitle;
log(DEBUG, 'elementConfig', elementConfig);
const pageTitle = HEADER.querySelector(SELECTORS.pageTitle.topDiv);
if (!pageTitle) {
logError(`Selector '${SELECTORS.pageTitle.topDiv}' not found`);
return;
}
pageTitle.setAttribute('id', SELECTORS.pageTitle.id);
if (elementConfig.remove) {
removePageTitle();
return;
}
if (elementConfig.color !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.pageTitle.links}
{
color: ${elementConfig.color} !important;
}
`;
}
if (elementConfig.hover.color !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.pageTitle.links}:hover
{
color: ${elementConfig.hover.color} !important;
}
`;
}
if (elementConfig.hover.backgroundColor !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.pageTitle.links}:hover
{
background-color: ${elementConfig.hover.backgroundColor} !important;
}
`;
}
}
function updateSearch() {
log(DEBUG, 'updateSearch()');
const configKey = 'search';
const elementConfig = CONFIG[configKey];
const elementSelector = SELECTORS[configKey];
let topDivSelector = elementSelector.id;
const topDiv = HEADER.querySelector(createId(elementSelector.id)) ||
HEADER.querySelector(elementSelector.topDiv);
if (!topDiv) {
logError(`Selectors '${createId(elementSelector.id)}' and '${elementSelector.topDiv}' not found`);
return;
}
topDiv.setAttribute('id', elementSelector.id);
if (elementConfig.remove) {
HEADER_STYLE.textContent += cssHideElement(createId(elementSelector.id));
return;
}
if (elementConfig.alignLeft) {
const response = cloneAndLeftAlignElement(createId(topDivSelector), topDivSelector);
if (response.length === 0) return;
// Also need to hide button due to it showing up on larger screen widths
HEADER_STYLE.textContent += cssHideElement(`${createId(topDivSelector)} ${elementSelector.input}`);
HEADER_STYLE.textContent += `
${createId(topDivSelector)}
{
flex-grow: 1 !important;
}
`;
const [cloneId, _cloneElement] = response;
topDivSelector = createId(cloneId);
HEADER_STYLE.textContent += `
${topDivSelector}
{
flex: 0 1 auto !important;
justify-content: flex-start !important;
}
`;
}
if (elementConfig.width === 'max') {
log(DEBUG, 'elementSelector', elementSelector);
HEADER_STYLE.textContent += `
@media (min-width: 1012px) {
${elementSelector.input}
{
width: auto !important
}
${SELECTORS.header.leftAligned}
{
flex: 0 1 auto !important;
}
${SELECTORS.header.rightAligned}
{
flex: 1 1 auto !important;
justify-content: space-between !important;
}
${createId(topDivSelector)}
{
display: block !important;
}
${elementSelector.topDiv}-whenRegular
{
max-width: none !important;
}
}
`;
} else if (elementConfig.width !== '') {
HEADER_STYLE.textContent += `
@media (min-width: 1012px)
{
${topDivSelector},
${elementSelector.input}
{
width: ${elementConfig.width} !important
}
}
@media (min-width: 768px)
{
${topDivSelector},
${elementSelector.input}
{
--feed-sidebar: 320px;
}
}
@media (min-width: 1400px)
{
${topDivSelector},
${elementSelector.input}
{
--feed-sidebar: 336px;
}
}
`;
}
if (elementConfig.margin.left !== '') {
HEADER_STYLE.textContent += `
@media (min-width: 1012px)
{
${elementSelector.input}
{
margin-left: ${elementConfig.margin.left} !important
}
}
`;
}
if (elementConfig.margin.right!== '') {
HEADER_STYLE.textContent += `
@media (min-width: 1012px)
{
${elementSelector.input}
{
margin-right: ${elementConfig.margin.right} !important
}
}
`;
}
if (elementConfig.rightButton !== 'command palette') {
const commandPaletteButton = HEADER.querySelector(elementSelector.commandPalette);
if (!commandPaletteButton) {
logError(`Selector '${elementSelector.commandPalette}' not found`);
} else {
HEADER_STYLE.textContent += cssHideElement(elementSelector.commandPalette);
}
}
const placeholderSpan = HEADER.querySelector(elementSelector.placeholderSpan);
if (!placeholderSpan) {
logError(`Selector '${elementSelector.placeholderSpan}' not found`);
return;
}
if (elementConfig.placeholder.text !== '') {
// Without this, the placeholder text is overwritten by the shadow DOM
// You may see the following error in the console:
// qbsearch-input-element.ts:421 Uncaught (in promise) TypeError: Cannot read properties of undefined (reading 'innerHTML')
placeholderSpan.setAttribute('data-target', 'avoidShadowDOM');
placeholderSpan.innerText = elementConfig.placeholder.text;
}
if (elementConfig.placeholder.color !== '') {
HEADER_STYLE.textContent += `
${elementSelector.placeholderSpan}
{
color: ${elementConfig.placeholder.color} !important;
}
`;
}
const searchButton = HEADER.querySelector(elementSelector.button);
if (!searchButton) {
logError(`Selector '${elementSelector.button}' not found`);
return;
}
if (elementConfig.backgroundColor !== '') {
HEADER_STYLE.textContent += `
${elementSelector.button}
{
background-color: ${elementConfig.backgroundColor} !important;
}
`;
}
if (elementConfig.borderColor !== '') {
// There are different buttons at different widths
HEADER_STYLE.textContent += `
${elementSelector.input} button
{
border-color: ${elementConfig.borderColor} !important;
}
`;
}
if (elementConfig.boxShadow !== '') {
HEADER_STYLE.textContent += `
${elementSelector.button}
{
box-shadow: ${elementConfig.boxShadow} !important;
}
`;
}
if (elementConfig.magnifyingGlassIcon.remove) {
HEADER_STYLE.textContent += cssHideElement(elementSelector.magnifyingGlassIcon);
}
if (elementConfig.modal.width !== '') {
HEADER_STYLE.textContent += `
${elementSelector.modal}
{
width: ${elementConfig.modal.width} !important;
}
`;
}
if (elementConfig.rightButton === 'slash key') {
HEADER_STYLE.textContent += `
${elementSelector.placeholderSpan}
{
width: 100% !important;
}
`;
const parser = new DOMParser();
const svgDoc = parser.parseFromString(
'<svg xmlns="http://www.w3.org/2000/svg" width="22" height="20" aria-hidden="true"><path fill="none" stroke="#979A9C" opacity=".4" d="M3.5.5h12c1.7 0 3 1.3 3 3v13c0 1.7-1.3 3-3 3h-12c-1.7 0-3-1.3-3-3v-13c0-1.7 1.3-3 3-3z"></path><path fill="#979A9C" d="M11.8 6L8 15.1h-.9L10.8 6h1z"></path></svg>',
'image/svg+xml',
);
const slashImg = svgDoc.documentElement;
slashImg.alt = 'slash key to search';
const placeholderDiv = HEADER.querySelector(elementSelector.placeholderDiv);
if (!placeholderDiv) {
logError(`Selector '${elementSelector.placeholderDiv}' not found`);
return;
}
HEADER_STYLE.textContent += `
${elementSelector.placeholderDiv}
{
display: flex !important;
}
${elementSelector.button}
{
padding-inline-start: 8px !important;
}
`;
placeholderDiv.appendChild(slashImg);
}
}
function removeDivider() {
log(DEBUG, 'removeDivider()');
HEADER_STYLE.textContent += `
${SELECTORS.header.actionsDiv}::before
{
content: none !important;
}
`;
}
function updateLink(configKey) {
log(DEBUG, 'updateLink()');
const elementConfig = CONFIG[configKey];
const elementSelector = SELECTORS[configKey];
let link;
const tooltipElement = SELECTORS.toolTips[configKey];
if (tooltipElement) {
link = tooltipElement.previousElementSibling;
} else {
log(DEBUG, `Tooltip for '${configKey}' not found`);
const linkId = createId(SELECTORS[configKey].id);
link = HEADER.querySelector(linkId);
if (!link) {
logError(`Selector '${linkId}' not found`);
return;
}
}
let linkSelector = elementSelector.id;
link.setAttribute('id', linkSelector);
if (elementConfig.remove) {
HEADER_STYLE.textContent += cssHideElement(createId(configKey));
return;
}
if (!elementConfig.tooltip && SELECTORS.toolTips[configKey]?.id) {
HEADER_STYLE.textContent += cssHideElement(createId(SELECTORS.toolTips[configKey].id));
}
if (elementConfig.alignLeft) {
const response = cloneAndLeftAlignElement(createId(elementSelector.id), elementSelector.id);
if (response.length === 0) return;
const [cloneId, cloneElement] = response;
elementSelector[CONFIG_NAME] = {
leftAlignedId: cloneId,
};
link = cloneElement;
linkSelector = createId(cloneId);
}
const padding = '7px';
link.style.setProperty('padding-left', padding, 'important');
link.style.setProperty('padding-right', padding, 'important');
let textContent = elementConfig.text.content;
if (elementConfig.icon.remove) {
const svgId = `${configKey}-svg`;
const svg = link.querySelector('svg');
if (!svg) {
logError(`Selector '${configKey} svg' not found`);
return;
}
svg.setAttribute('id', svgId);
HEADER_STYLE.textContent += cssHideElement(createId(svgId));
} else {
link.querySelector('svg').style.setProperty('fill', elementConfig.icon.color);
textContent = UNICODE_NON_BREAKING_SPACE + textContent;
}
modifyThenObserve(() => {
HEADER.querySelector(createId(elementSelector.textContent))?.remove();
});
if (elementConfig.text.content !== '') {
const spanElement = document.createElement('span');
const spanId = `${configKey}-text-content-span`;
spanElement.setAttribute('id', spanId);
if (elementConfig.text.color) {
HEADER_STYLE.textContent += `
${createId(spanId)}
{
color: ${elementConfig.text.color} !important;
}
`;
}
const textNode = document.createTextNode(textContent);
spanElement.appendChild(textNode);
link.appendChild(spanElement);
}
if (!elementConfig.border) {
HEADER_STYLE.textContent += `
${linkSelector}
{
border: none !important;
}
`;
}
if (elementConfig.boxShadow !== '') {
HEADER_STYLE.textContent += `
${linkSelector}
{
box-shadow: ${elementConfig.boxShadow} !important;
}
`;
}
if (elementConfig.hover.backgroundColor !== '') {
HEADER_STYLE.textContent += `
${linkSelector}:hover
{
background-color: ${elementConfig.hover.backgroundColor} !important;
}
`;
}
if (elementConfig.hover.color !== '') {
HEADER_STYLE.textContent += `
${linkSelector} span:hover
{
color: ${elementConfig.hover.color} !important;
}
`;
}
log(DEBUG, `Updates applied to link ${configKey}`, link);
}
function cloneAndFlipElements(firstElementSelector, secondElementSelector, firstElementId, secondElementId) {
log(DEBUG, 'cloneAndFlipElements()');
const firstElement = HEADER.querySelector(firstElementSelector);
if (!firstElement) {
logError(`Selector '${firstElementSelector}' not found`);
return [];
}
const secondElement = HEADER.querySelector(secondElementSelector);
if (!secondElement) {
logError(`Selector '${secondElementSelector}' not found`);
return [];
}
const firstElementClone = firstElement.cloneNode(true);
const secondElementClone = secondElement.cloneNode(true);
const firstElementCloneId = `${firstElementId}-clone`;
const secondElementCloneId = `${secondElementId}-clone`;
firstElementClone.setAttribute('id', firstElementCloneId);
secondElementClone.setAttribute('id', secondElementCloneId);
firstElementClone.style.setProperty('display', 'none');
secondElementClone.style.setProperty('display', 'none');
HEADER_STYLE.textContent = HEADER_STYLE.textContent.replace(
new RegExp(escapeRegExp(firstElementSelector), 'g'),
createId(firstElementCloneId),
);
HEADER_STYLE.textContent = HEADER_STYLE.textContent.replace(
new RegExp(escapeRegExp(secondElementSelector), 'g'),
createId(secondElementCloneId),
);
HEADER_STYLE.textContent += cssHideElement(firstElementSelector);
HEADER_STYLE.textContent += cssHideElement(secondElementSelector);
log(VERBOSE, `#${firstElementCloneId}, #${secondElementCloneId}`);
HEADER_STYLE.textContent += `
#${firstElementCloneId},
#${secondElementCloneId}
{
display: flex !important;
}
`;
if (secondElement.nextElementSibling === null) {
secondElement.parentNode.appendChild(firstElementClone);
} else {
secondElement.parentNode.insertBefore(firstElementClone, secondElement.nextElementSibling);
}
if (firstElement.nextElementSibling === null) {
firstElement.parentNode.appendChild(secondElementClone);
} else {
firstElement.parentNode.insertBefore(secondElementClone, firstElement.nextElementSibling);
}
if (firstElementSelector.includes('clone')) {
firstElement.remove();
}
if (secondElementSelector.includes('clone')) {
secondElement.remove();
}
NEW_ELEMENTS.push(firstElementClone);
NEW_ELEMENTS.push(secondElementClone);
return [firstElementClone, secondElementClone];
}
function flipIssuesPullRequests() {
log(DEBUG, 'flipIssuesPullRequest()');
const issuesId = SELECTORS.issues[CONFIG_NAME]?.leftAlignedId || SELECTORS.issues.id;
log(VERBOSE, 'issuesId', issuesId);
const pullRequestsId = SELECTORS.pullRequests[CONFIG_NAME]?.leftAlignedId || SELECTORS.pullRequests.id;
log(VERBOSE, 'pullRequestsId', pullRequestsId);
cloneAndFlipElements(
createId(issuesId),
createId(pullRequestsId),
`${issuesId}-flip-div`,
`${pullRequestsId}-flip-div`,
);
}
function createOldLink(configKey, svgString) {
const pullRequestsLink = HEADER.querySelector(SELECTORS.pullRequests.link);
if (!pullRequestsLink) {
logError(`Selector '${SELECTORS.pullRequests.link}' not found`);
return;
}
const clonedLink = pullRequestsLink.cloneNode(true);
const linkId = SELECTORS[configKey].id;
clonedLink.setAttribute('id', linkId);
const oldSvg = clonedLink.querySelector('svg');
const parser = new DOMParser();
const svgDoc = parser.parseFromString(svgString, 'image/svg+xml');
const newSvg = svgDoc.documentElement;
oldSvg.parentNode.replaceChild(newSvg, oldSvg);
const ariaId = `tooltip-${configKey}`;
clonedLink.setAttribute('href', `/${configKey}`);
clonedLink.setAttribute('aria-labelledby', ariaId);
clonedLink.removeAttribute('data-analytics-event');
clonedLink.querySelector('span')?.remove();
pullRequestsLink.parentNode.appendChild(clonedLink);
NEW_ELEMENTS.push(clonedLink);
}
function createMarketplaceLink() {
log(DEBUG, 'createMarketplaceLink()');
const svgString = `
<svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-gift">
<path d="M2 2.75A2.75 2.75 0 0 1 4.75 0c.983 0 1.873.42 2.57 1.232.268.318.497.668.68 1.042.183-.375.411-.725.68-1.044C9.376.42 10.266 0 11.25 0a2.75 2.75 0 0 1 2.45 4h.55c.966 0 1.75.784 1.75 1.75v2c0 .698-.409 1.301-1 1.582v4.918A1.75 1.75 0 0 1 13.25 16H2.75A1.75 1.75 0 0 1 1 14.25V9.332C.409 9.05 0 8.448 0 7.75v-2C0 4.784.784 4 1.75 4h.55c-.192-.375-.3-.8-.3-1.25ZM7.25 9.5H2.5v4.75c0 .138.112.25.25.25h4.5Zm1.5 0v5h4.5a.25.25 0 0 0 .25-.25V9.5Zm0-4V8h5.5a.25.25 0 0 0 .25-.25v-2a.25.25 0 0 0-.25-.25Zm-7 0a.25.25 0 0 0-.25.25v2c0 .138.112.25.25.25h5.5V5.5h-5.5Zm3-4a1.25 1.25 0 0 0 0 2.5h2.309c-.233-.818-.542-1.401-.878-1.793-.43-.502-.915-.707-1.431-.707ZM8.941 4h2.309a1.25 1.25 0 0 0 0-2.5c-.516 0-1 .205-1.43.707-.337.392-.646.975-.879 1.793Z"></path>
</svg>
`;
createOldLink('marketplace', svgString);
}
function createExploreLink() {
log(DEBUG, 'createExploreLink()');
const svgString = `
<svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-telescope">
<path d="M14.184 1.143v-.001l1.422 2.464a1.75 1.75 0 0 1-.757 2.451L3.104 11.713a1.75 1.75 0 0 1-2.275-.702l-.447-.775a1.75 1.75 0 0 1 .53-2.32L11.682.573a1.748 1.748 0 0 1 2.502.57Zm-4.709 9.32h-.001l2.644 3.863a.75.75 0 1 1-1.238.848l-1.881-2.75v2.826a.75.75 0 0 1-1.5 0v-2.826l-1.881 2.75a.75.75 0 1 1-1.238-.848l2.049-2.992a.746.746 0 0 1 .293-.253l1.809-.87a.749.749 0 0 1 .944.252ZM9.436 3.92h-.001l-4.97 3.39.942 1.63 5.42-2.61Zm3.091-2.108h.001l-1.85 1.26 1.505 2.605 2.016-.97a.247.247 0 0 0 .13-.151.247.247 0 0 0-.022-.199l-1.422-2.464a.253.253 0 0 0-.161-.119.254.254 0 0 0-.197.038ZM1.756 9.157a.25.25 0 0 0-.075.33l.447.775a.25.25 0 0 0 .325.1l1.598-.769-.83-1.436-1.465 1Z"></path>
</svg>
`;
createOldLink('explore', svgString);
}
function updateCreateNewButton() {
log(DEBUG, 'updateCreateNewButton()');
const configKey = 'create';
const elementSelector = SELECTORS[configKey];
const tooltipElement = SELECTORS.toolTips[configKey];
if (!tooltipElement) {
logError(`Selector '${SELECTORS.toolTips[configKey]}' not found`);
return;
}
let button = HEADER.querySelector(elementSelector.button);
let oldButtonId = null;
if (!button) {
logError(`Selector '${elementSelector.button}' not found`);
oldButtonId = `${elementSelector.button}-old`;
button = HEADER.querySelector(oldButtonId);
if (!button) {
logError(`Selector '${oldButtonId}' not found`);
return;
}
}
const elementConfig = CONFIG[configKey];
if (elementConfig.remove) {
HEADER_STYLE.textContent += cssHideElement(elementSelector.topDiv);
return;
} else if (!elementConfig.tooltip) {
HEADER_STYLE.textContent += cssHideElement(createId(tooltipElement.id));
}
const topDiv = HEADER.querySelector(elementSelector.topDiv);
if (!topDiv) {
logError(`Selector '${elementSelector.topDiv}' not found`);
return;
}
topDiv.setAttribute('id', elementSelector.id);
const buttonLabel = button.querySelector(elementSelector.dropdownIcon);
if (elementConfig.plusIcon.remove) {
HEADER_STYLE.textContent += `
${oldButtonId || elementSelector.button} ${elementSelector.plusIcon}
{
display: none !important
}
`;
} else {
if (elementConfig.plusIcon.color !== '') {
HEADER_STYLE.textContent += `
${elementSelector.plusIcon}
{
color: ${elementConfig.plusIcon.color} !important;
}
`;
}
if (elementConfig.plusIcon.hover.color !== '') {
HEADER_STYLE.textContent += `
${elementSelector.plusIcon.split(' ').join(':hover ')} svg path
{
fill: ${elementConfig.plusIcon.hover.color} !important;
}
`;
}
if (elementConfig.plusIcon.marginRight !== '') {
HEADER_STYLE.textContent += `
${elementSelector.plusIcon}
{
margin-right: ${elementConfig.plusIcon.marginRight} !important;
}
`;
}
}
modifyThenObserve(() => {
HEADER.querySelector(createId(SELECTORS[configKey].textContent))?.remove();
});
if (elementConfig.text.content !== '') {
// Update the text's font properties to match the others
HEADER_STYLE.textContent += `
${elementSelector.button}
{
font-size: var(--text-body-size-medium, 0.875rem) !important;
font-weight: var(--base-text-weight-medium, 500) !important;
}
`;
const spanElement = document.createElement('span');
spanElement.setAttribute('id', elementSelector.textContent);
spanElement.style.setProperty('color', elementConfig.text.color);
spanElement.textContent = elementConfig.text.content;
// New span is inserted between the plus sign and dropdown icon
buttonLabel.parentNode.insertBefore(spanElement, buttonLabel);
}
if (elementConfig.dropdownIcon.remove) {
HEADER_STYLE.textContent += `
${elementSelector.dropdownIcon}
{
display: none !important
}
`;
} else {
HEADER_STYLE.textContent += `
${elementSelector.dropdownIcon}
{
grid-area: initial !important;
}
`;
if (elementConfig.dropdownIcon.color !== '') {
HEADER_STYLE.textContent += `
${elementSelector.dropdownIcon}
{
color: ${elementConfig.dropdownIcon.color} !important;
}
`;
}
if (elementConfig.dropdownIcon.hover.color !== '') {
HEADER_STYLE.textContent += `
${elementSelector.dropdownIcon.split(' ').join(':hover ')} svg path
{
fill: ${elementConfig.dropdownIcon.hover.color} !important;
}
`;
}
}
if (!elementConfig.border) {
HEADER_STYLE.textContent += `
${elementSelector.button}
{
border: none !important;
}
`;
}
if (elementConfig.boxShadow !== '') {
HEADER_STYLE.textContent += `
${elementSelector.button}
{
box-shadow: ${elementConfig.boxShadow} !important;
}
`;
}
if (elementConfig.hoverBackgroundColor !== '') {
HEADER_STYLE.textContent += `
${elementSelector.button}:hover
{
background-color: ${elementConfig.hoverBackgroundColor} !important;
}
`;
}
}
function updateInboxLink() {
log(DEBUG, 'updateInboxLink()');
const configKey = 'notifications';
const elementConfig = CONFIG[configKey];
const elementSelector = SELECTORS[configKey];
const notificationIndicator = HEADER.querySelector(createId(elementSelector.id)) ||
HEADER.querySelector(elementSelector.indicator);
if (!notificationIndicator) {
logError(`Selectors '${createId(elementSelector.id)}' and '${elementSelector.indicator}' not found`);
return;
}
notificationIndicator.setAttribute('id', elementSelector.id);
const inboxLink = notificationIndicator.querySelector('a');
if (!inboxLink) {
logError(`Selector '${elementSelector.indicator} a' not found`);
return;
}
if (elementConfig.remove) {
HEADER_STYLE.textContent += cssHideElement(elementSelector.indicator);
}
if (!elementConfig.tooltip) {
HEADER_STYLE.textContent += cssHideElement(createId(SELECTORS.toolTips.notifications.id));
}
if (elementConfig.dot.remove) {
HEADER_STYLE.textContent += `
${elementSelector.dot}
{
content: none !important;
}
`;
} else {
if (elementConfig.dot.color !== '') {
HEADER_STYLE.textContent += `
${elementSelector.dot}
{
background: ${elementConfig.dot.color} !important;
}
`;
}
if (elementConfig.dot.boxShadowColor !== '') {
HEADER_STYLE.textContent += `
${elementSelector.dot}
{
box-shadow: 0 0 0 calc(var(--base-size-4, 4px)/2) ${elementConfig.dot.boxShadowColor} !important;
}
`;
}
}
if (elementConfig.icon.symbol === 'inbox') {
if (elementConfig.icon.color !== '') {
HEADER_STYLE.textContent += `
${createId(elementSelector.id)} a svg
{
fill: elementConfig.icon.color !important;
}
`;
}
} else {
const inboxSvgId = 'inbox-svg';
const inboxSvg = inboxLink.querySelector('svg');
inboxSvg.setAttribute('id', inboxSvgId);
HEADER_STYLE.textContent += cssHideElement(createId(inboxSvgId));
}
if (elementConfig.icon.symbol === 'bell') {
// Bell icon from https://gist.github.com
const bellSvgId = 'bell-svg';
const bellSvg = `
<svg id=${bellSvgId} style="display: none;" aria-hidden='true' height='16' viewBox='0 0 16 16' version='1.1' width='16' data-view-component='true' class='octicon octicon-bell'>
<path d='M8 16a2 2 0 0 0 1.985-1.75c.017-.137-.097-.25-.235-.25h-3.5c-.138 0-.252.113-.235.25A2 2 0 0 0 8 16ZM3 5a5 5 0 0 1 10 0v2.947c0 .05.015.098.042.139l1.703 2.555A1.519 1.519 0 0 1 13.482 13H2.518a1.516 1.516 0 0 1-1.263-2.36l1.703-2.554A.255.255 0 0 0 3 7.947Zm5-3.5A3.5 3.5 0 0 0 4.5 5v2.947c0 .346-.102.683-.294.97l-1.703 2.556a.017.017 0 0 0-.003.01l.001.006c0 .002.002.004.004.006l.006.004.007.001h10.964l.007-.001.006-.004.004-.006.001-.007a.017.017 0 0 0-.003-.01l-1.703-2.554a1.745 1.745 0 0 1-.294-.97V5A3.5 3.5 0 0 0 8 1.5Z'></path>
</svg>
`;
inboxLink.insertAdjacentHTML('afterbegin', bellSvg);
HEADER_STYLE.textContent += `
${createId(bellSvgId)}
{
display: initial !important;
}
`;
if (elementConfig.icon.color !== '') {
HEADER_STYLE.textContent += `
${createId(bellSvgId)} path
{
fill: ${elementConfig.icon.color} !important;
}
`;
}
}
if (elementConfig.icon.hover.color !== '') {
HEADER_STYLE.textContent += `
${createId(elementSelector.id)} a:hover svg path
{
fill: ${elementConfig.icon.hover.color} !important;
}
`;
}
modifyThenObserve(() => {
HEADER.querySelector(createId(SELECTORS[configKey].textContent))?.remove();
});
if (elementConfig.text.content !== '') {
const padding = '9px';
HEADER_STYLE.textContent += `
${createId(elementSelector.id)} a
{
padding-left: ${padding} !important;
padding-right: ${padding} !important;
width: auto !important;
text-decoration: none !important;
display: flex !important;
}
`;
let textContent = elementConfig.text.content;
if (elementConfig.icon !== '') {
textContent = UNICODE_NON_BREAKING_SPACE + textContent;
}
const spanElement = document.createElement('span');
spanElement.setAttribute('id', elementSelector.textContent);
// Update the text's font properties to match the others
spanElement.style.setProperty('font-size', 'var(--text-body-size-medium, 0.875rem)', 'important');
spanElement.style.setProperty('font-weight', 'var(--base-text-weight-medium, 500)', 'important');
if (elementConfig.text.color) spanElement.style.setProperty('color', elementConfig.text.color);
const textNode = document.createTextNode(textContent);
spanElement.appendChild(textNode);
inboxLink.appendChild(spanElement);
}
if (!elementConfig.border) {
HEADER_STYLE.textContent += `
${createId(elementSelector.id)} a
{
border: none !important;
}
`;
}
if (elementConfig.boxShadow !== '') {
HEADER_STYLE.textContent += `
${createId(elementSelector.id)} a
{
box-shadow: ${elementConfig.boxShadow} !important;
}
`;
}
if (elementConfig.dot.displayOverIcon) {
HEADER_STYLE.textContent += `
${elementSelector.dot}
{
top: 5px !important;
left: 18px !important;
}
`;
}
if (elementConfig.hoverBackgroundColor !== '') {
HEADER_STYLE.textContent += `
${createId(elementSelector.id)} a:hover
{
background-color: ${elementConfig.hoverBackgroundColor} !important;
}
`;
}
log(DEBUG, `Updates applied to link ${configKey}: `, inboxLink);
}
function insertAvatarDropdown() {
log(DEBUG, 'insertAvatarDropdown()');
const elementSelector = SELECTORS.avatar;
const svgSelector = elementSelector.svg;
if (HEADER.querySelector(createId(svgSelector))) {
log(VERBOSE, `Selector ${createId(svgSelector)} not found`);
return;
}
const dropdownSvg = `
<svg id='${svgSelector}' style="display: none;" height="100%" width="100%" fill="#FFFFFF" class="octicon octicon-triangle-down" aria-hidden="true" viewBox="0 0 16 16" version="1.1" data-view-component="true">
<path d="m4.427 7.427 3.396 3.396a.25.25 0 0 0 .354 0l3.396-3.396A.25.25 0 0 0 11.396 7H4.604a.25.25 0 0 0-.177.427Z"></path>
</svg>
`;
const button = HEADER.querySelector(elementSelector.button);
if (!button) {
logError(`Selector '${elementSelector.button}' not found`);
return;
}
const divElement = document.createElement('div');
divElement.insertAdjacentHTML('afterbegin', dropdownSvg);
button.appendChild(divElement);
}
function updateAvatarButton() {
log(DEBUG, 'updateAvatarButton()');
const elementSelector = SELECTORS.sidebars.right;
const backdropSelector = elementSelector.backdrop;
const backdrop = HEADER.querySelector(backdropSelector);
if (!backdrop) {
log(DEBUG, `Selector ${backdropSelector} not found`);
return;
}
const avatarButton = HEADER.querySelector(SELECTORS.avatar.button);
if (!avatarButton) {
log(DEBUG, `Selector ${SELECTORS.avatar.button} not found`);
return;
}
if (backdrop.classList.contains('Overlay--hidden')) {
if (avatarButton.hasAttribute('data-close-dialog-id')) {
const dialogId = avatarButton.getAttribute('data-close-dialog-id');
avatarButton.setAttribute('data-show-dialog-id', dialogId);
avatarButton.removeAttribute('data-close-dialog-id');
}
} else {
if (avatarButton.hasAttribute('data-show-dialog-id')) {
const dialogId = avatarButton.getAttribute('data-show-dialog-id');
avatarButton.setAttribute('data-close-dialog-id', dialogId);
avatarButton.removeAttribute('data-show-dialog-id');
}
}
}
function updateAvatar() {
log(DEBUG, 'updateAvatar()');
const configKey = 'avatar';
const elementConfig = CONFIG[configKey];
const elementSelector = SELECTORS[configKey];
if (elementConfig.remove) {
HEADER_STYLE.textContent += cssHideElement(elementSelector.topDiv);
return;
}
if (elementConfig.size !== '') {
HEADER_STYLE.textContent += `
${elementSelector.img}
{
height: ${elementConfig.size} !important;
width: ${elementConfig.size} !important;
}
`;
}
if (elementConfig.dropdownIcon) {
insertAvatarDropdown();
HEADER_STYLE.textContent += `
${elementSelector.topDiv}
{
background-color: transparent !important;
}
${createId(elementSelector.svg)}
{
display: initial !important;
fill: #FFFFFF;
height: 16px;
width: 16px;
margin-bottom: 1.5px;
}
${elementSelector.button}:hover ${createId(elementSelector.svg)} path
{
fill: #FFFFFFB3 !important;
}
${elementSelector.button}
{
gap: 0px !important;
}
`;
}
}
function flipCreateInbox() {
log(DEBUG, 'flipCreateInbox()');
cloneAndFlipElements(
createId(SELECTORS.create.id),
createId(SELECTORS.notifications.id),
`${SELECTORS.create.id}-flip`,
`${SELECTORS.notifications.id}-flip`,
);
}
function updateGlobalBar() {
log(DEBUG, 'updateGlobalBar()');
const elementConfig = CONFIG.globalBar;
if (elementConfig.boxShadowColor !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.header.globalBar}
{
box-shadow: inset 0 calc(var(--borderWidth-thin, 1px)*-1) ${elementConfig.boxShadowColor} !important;
}
`;
}
if (elementConfig.rightAligned.gap !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.header.rightAligned}
{
gap: ${elementConfig.rightAligned.gap} !important;
}
`;
}
if (elementConfig.leftAligned.gap !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.header.leftAligned}
{
gap: ${elementConfig.leftAligned.gap} !important;
}
`;
}
}
function updateLocalBar() {
log(DEBUG, 'updateLocalBar()');
const elementConfig = CONFIG.localBar;
if (elementConfig.backgroundColor !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.header.localBar.topDiv}
{
background-color: ${elementConfig.backgroundColor} !important;
box-shadow: inset 0 calc(var(--borderWidth-thin, 1px)*-1) var(--color-border-default) !important;
}
`;
}
if (elementConfig.alignCenter) {
HEADER_STYLE.textContent += `
${SELECTORS.header.localBar.underlineNavActions}
{
display: initial !important;
padding-right: 0px !important;
}
${SELECTORS.header.localBar.topDiv} nav
{
max-width: 1280px;
margin-right: auto;
margin-left: auto;
}
@media (min-width: 768px) {
${SELECTORS.header.localBar.topDiv} nav
{
padding-right: var(--base-size-24, 24px) !important;
padding-left: var(--base-size-24, 24px) !important;
}
}
@media (min-width: 1012px) {
${SELECTORS.header.localBar.topDiv} nav
{
padding-right: var(--base-size-32, 32px) !important;
padding-left: var(--base-size-32, 32px) !important;
}
.notification-shelf > div
{
padding-right: var(--base-size-32, 32px) !important;
padding-left: var(--base-size-32, 32px) !important;
max-width: 1280px;
margin-right: auto;
margin-left: auto;
}
}
`;
}
if (elementConfig.boxShadow.consistentColor) {
HEADER_STYLE.textContent += `
.UnderlineNav
{
box-shadow: none !important;
}
`;
}
if (elementConfig.links.color !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.header.localBar.topDiv} a,
${SELECTORS.header.localBar.topDiv} a span
{
color: ${elementConfig.links.color} !important;
}
`;
}
}
function preloadLeftSidebar(elementSelector) {
log(DEBUG, 'preloadLeftSidebar()');
if (!LEFT_SIDEBAR_PRELOADED) return;
const leftModalDialog = HEADER.querySelector(elementSelector.left.modalDialog).remove();
if (!leftModalDialog) {
logError(`Selector '${elementSelector.left.modalDialog}' not found`);
preloadLeftSidebar(elementSelector);
return;
}
window.addEventListener('load', () => {
HEADER.querySelector(`${SELECTORS.hamburgerButton} button`).click();
log(INFO, 'Left sidebar preloaded');
});
LEFT_SIDEBAR_PRELOADED = true;
}
function preloadRightSidebar(elementSelector) {
log(DEBUG, 'preloadRightSidebar()');
if (!RIGHT_SIDEBAR_PRELOADED) return;
const rightModalDialog = HEADER.querySelector(elementSelector.right.modalDialog);
if (!rightModalDialog) {
logError(`Selector '${elementSelector.right.modalDialog}' not found`);
preloadRightSidebar(elementSelector);
return;
}
rightModalDialog.remove();
window.addEventListener('load', () => {
HEADER.querySelector(SELECTORS.avatar.button).click();
log(INFO, 'Right sidebar preloaded');
});
RIGHT_SIDEBAR_PRELOADED = true;
}
function updateSidebars() {
log(DEBUG, 'updateSidebars()');
const configKey = 'sidebars';
const elementConfig = CONFIG[configKey];
const elementSelector = SELECTORS[configKey];
if (elementConfig.backdrop.color !== '') {
HEADER_STYLE.textContent += `
${elementSelector.backdrop}
{
background: ${CONFIG.sidebars.backdrop.color} !important;
}
`;
}
if (elementConfig.left.preload) preloadLeftSidebar(elementSelector);
if (elementConfig.right.floatUnderneath) {
HEADER_STYLE.textContent += `
${elementSelector.right.modalDialog}
{
border-top-right-radius: 0.75rem !important;
border-bottom-right-radius: 0.75rem !important;
bottom: initial !important;
margin-top: 55px;
margin-right: 20px;
}
${elementSelector.right.navParentDiv}
{
margin-bottom: 0px !important;
}
${elementSelector.right.nav}
{
padding-bottom: 0px !important;
}
`;
}
if (elementConfig.right.preload) preloadRightSidebar(elementSelector);
if (elementConfig.right.maxHeight) {
HEADER_STYLE.textContent += `
${elementSelector.right.modalDialog}
{
max-height: ${elementConfig.right.maxHeight} !important;
}
`;
}
if (elementConfig.right.width !== '') {
HEADER_STYLE.textContent += `
${elementSelector.right.modalDialog}.Overlay.Overlay--size-small-portrait
{
--overlay-width: ${elementConfig.right.width};
}
`;
}
}
function importRepositoryHeader() {
log(DEBUG, 'importRepositoryHeader()');
const configKey = 'repositoryHeader';
const repositoryHeader = document.querySelector(SELECTORS[configKey].id);
if (!repositoryHeader) {
// This is expected on pages that aren't repositories
log(DEBUG, `Selector '${SELECTORS[configKey].id}' not found`);
return;
}
const topRepositoryHeaderElement = document.createElement('div');
topRepositoryHeaderElement.style.setProperty('display', 'flex');
topRepositoryHeaderElement.style.setProperty('padding', '0px');
topRepositoryHeaderElement.style.setProperty('box-shadow', 'none');
const elementConfig = CONFIG[configKey];
if (elementConfig.backgroundColor !== '') {
topRepositoryHeaderElement.style.setProperty('background-color', elementConfig.backgroundColor);
}
if (repositoryHeader.hidden) {
log(VERBOSE, `Selector '${SELECTORS[configKey].id}' is hidden`);
if (!HEADER.querySelector(SELECTORS.pageTitle.separator)) {
log(INFO, `Selector '${SELECTORS.pageTitle.separator}' not found, not creating a repository header`);
return;
}
// A repo tab other than Code is being loaded for the first time
const pageTitle = HEADER.querySelector(SELECTORS.pageTitle.topDiv);
if (!pageTitle) {
logError(`Selector '${SELECTORS.pageTitle.topDiv}' not found`);
return;
}
const repositoryHeaderElement = document.createElement('div');
repositoryHeaderElement.setAttribute('id', TEMP_REPOSITORY_HEADER_FLAG);
repositoryHeaderElement.classList.add(REPOSITORY_HEADER_CLASS, 'pt-3', 'mb-2', 'px-md-4');
const clonedPageTitle = pageTitle.cloneNode(true);
repositoryHeaderElement.appendChild(clonedPageTitle);
topRepositoryHeaderElement.appendChild(repositoryHeaderElement);
insertNewGlobalBar(topRepositoryHeaderElement);
} else if (HEADER.querySelector(createId(TEMP_REPOSITORY_HEADER_FLAG))) {
log(VERBOSE, `Selector '${createId(TEMP_REPOSITORY_HEADER_FLAG)}' found`);
// The Code tab is being loaded from another tab which has a temporary header
const tempRepositoryHeader = HEADER.querySelector(createId(TEMP_REPOSITORY_HEADER_FLAG));
NEW_ELEMENTS = NEW_ELEMENTS.filter(element => element !== tempRepositoryHeader);
tempRepositoryHeader.remove();
insertPermanentRepositoryHeader(topRepositoryHeaderElement, repositoryHeader);
} else {
log(
VERBOSE,
`${SELECTORS[configKey].id} is hidden and selector '${createId(TEMP_REPOSITORY_HEADER_FLAG)}' not found`,
);
// The Code tab being loaded for the first time
insertPermanentRepositoryHeader(topRepositoryHeaderElement, repositoryHeader);
}
updateRepositoryHeaderName();
if (elementConfig.backgroundColor !== '') {
HEADER_STYLE.textContent += `
.${REPOSITORY_HEADER_CLASS},
.notification-shelf
{
background-color: ${elementConfig.backgroundColor} !important;
}
`;
}
if (elementConfig.alignCenter) {
HEADER_STYLE.textContent += `
.${REPOSITORY_HEADER_CLASS}
{
max-width: 1280px;
margin-right: auto;
margin-left: auto;
}
.${REPOSITORY_HEADER_CLASS} div
{
padding-left: 0px !important;
padding-right: 0px !important;
}
@media (min-width: 768px) {
.${REPOSITORY_HEADER_CLASS}
{
padding-right: var(--base-size-24, 24px) !important;
padding-left: var(--base-size-24, 24px) !important;
}
}
@media (min-width: 1012px) {
.${REPOSITORY_HEADER_CLASS}
{
padding-right: var(--base-size-32, 32px) !important;
padding-left: var(--base-size-32, 32px) !important;
}
}
`;
}
if (elementConfig.link.color !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.repositoryHeader.links}
{
color: ${elementConfig.link.color} !important;
}
`;
}
if (elementConfig.link.hover.color !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.repositoryHeader.links}:hover
{
color: ${elementConfig.link.hover.color} !important;
}
`;
}
if (elementConfig.link.hover.backgroundColor !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.repositoryHeader.links}:hover
{
background-color: ${elementConfig.link.hover.backgroundColor} !important;
}
`;
}
if (elementConfig.link.hover.textDecoration !== '') {
HEADER_STYLE.textContent += `
${SELECTORS.repositoryHeader.links}:hover
{
text-decoration: ${elementConfig.link.hover.textDecoration} !important;
}
`;
}
HEADER_STYLE.textContent += `
.${REPOSITORY_HEADER_CLASS}
{
flex: auto !important;
}
${SELECTORS.repositoryHeader.details}
{
display: flex;
align-items: center;
}
${SELECTORS.pageTitle.topDiv}
{
flex: 0 1 auto !important;
height: auto !important;
min-width: 0 !important;
}
@media (min-width: 768px)
{
.AppHeader-context .AppHeader-context-compact
{
display: none !important;
}
}
.AppHeader-context .AppHeader-context-full
{
display: inline-flex !important;
width: 100% !important;
min-width: 0 !important;
max-width: 100% !important;
overflow: hidden !important;
}
.AppHeader-context .AppHeader-context-full ul {
display: flex;
flex-direction: row;
}
.AppHeader-context .AppHeader-context-full li:first-child {
flex: 0 100 max-content;
}
.AppHeader-context .AppHeader-context-full li {
display: inline-grid;
grid-auto-flow: column;
align-items: center;
flex: 0 99999 auto;
}
.AppHeader-context .AppHeader-context-full ul, .AppHeader .AppHeader-globalBar .AppHeader-context .AppHeader-context-full li {
list-style: none;
}
.AppHeader-context .AppHeader-context-item {
display: flex;
align-items: center;
min-width: 3ch;
line-height: var(--text-body-lineHeight-medium, 1.4285714286);
text-decoration: none !important;
border-radius: var(--borderRadius-medium, 6px);
padding-inline: var(--control-medium-paddingInline-condensed, 8px);
padding-block: var(--control-medium-paddingBlock, 6px);
}
.AppHeader-context .AppHeader-context-full li:last-child .AppHeader-context-item {
font-weight: var(--base-text-weight-semibold, 600);
}
.AppHeader-context .AppHeader-context-item-separator {
color: var(--fgColor-muted, var(--color-fg-muted));
white-space: nowrap;
}
${SELECTORS.header.globalBar}
{
padding: 16px !important;
}
`;
if (elementConfig.removePageTitle) removePageTitle();
return true;
}
function insertPermanentRepositoryHeader(topRepositoryHeaderElement, repositoryHeader) {
log(DEBUG, 'insertPermanentRepositoryHeader()');
const clonedRepositoryHeader = repositoryHeader.cloneNode(true);
// This is needed to prevent pop-in via Turbo when navigating between tabs on a repo
repositoryHeader.removeAttribute('data-turbo-replace');
clonedRepositoryHeader.removeAttribute('data-turbo-replace');
repositoryHeader.style.setProperty('display', 'none', 'important');
clonedRepositoryHeader.classList.add(REPOSITORY_HEADER_SUCCESS_FLAG, REPOSITORY_HEADER_CLASS);
topRepositoryHeaderElement.appendChild(clonedRepositoryHeader);
insertNewGlobalBar(topRepositoryHeaderElement);
clonedRepositoryHeader.firstElementChild.classList.remove('container-xl', 'px-lg-5');
NEW_ELEMENTS.push(clonedRepositoryHeader);
}
function updateRepositoryHeaderName() {
log(DEBUG, 'updateRepositoryHeaderName()');
const elementConfig = CONFIG.repositoryHeader;
const name = document.querySelector(SELECTORS.repositoryHeader.name);
if (!name) {
// When not in a repo, this is expected
log(DEBUG, `Selector '${SELECTORS.repositoryHeader.name}' not found`);
return;
}
name.style.setProperty('display', 'none', 'important');
const pageTitle = HEADER.querySelector(SELECTORS.pageTitle.topDiv);
if (!pageTitle) {
logError(`Selector '${SELECTORS.pageTitle.topDiv}' not found`);
return;
}
const ownerImg = document.querySelector(SELECTORS.repositoryHeader.ownerImg);
if (!ownerImg) {
log(INFO, `Selector '${SELECTORS.repositoryHeader.ownerImg}' not found`);
return;
}
const clonedPageTitle = pageTitle.cloneNode(true);
clonedPageTitle.style.display = '';
const pageTitleId = `${REPOSITORY_HEADER_CLASS}_pageTitle`;
clonedPageTitle.setAttribute('id', pageTitleId);
clonedPageTitle.querySelector('img')?.remove();
HEADER_STYLE.textContent += `
${createId(pageTitleId)}
{
display: initial !important;
}
`;
clonedPageTitle.querySelectorAll('svg').forEach(svg => svg.remove());
clonedPageTitle.querySelectorAll('a[href$="/stargazers"]').forEach(link => link.remove());
ownerImg.parentNode.insertBefore(clonedPageTitle, ownerImg.nextSibling);
NEW_ELEMENTS.push(clonedPageTitle);
if (elementConfig.avatar.remove) {
ownerImg.remove();
} else if (elementConfig.avatar.customSvg !== '') {
if (isValidURL(elementConfig.avatar.customSvg)) {
ownerImg.src = elementConfig.avatar.customSvg;
} else {
const divElement = document.createElement('div');
divElement.style.setProperty('display', 'flex');
divElement.style.setProperty('align-items', 'center');
divElement.innerHTML = elementConfig.avatar.customSvg;
ownerImg.parentNode.replaceChild(divElement, ownerImg);
}
}
HEADER_STYLE.textContent += cssHideElement(SELECTORS.repositoryHeader.bottomBorder);
}
function cloneAndLeftAlignElement(elementSelector, elementId) {
log(DEBUG, 'cloneAndLeftAlignElement()');
const leftAlignedDiv = HEADER.querySelector(SELECTORS.header.leftAligned);
if (!leftAlignedDiv) {
logError(`Selector '${SELECTORS.header.leftAligned}' not found`);
return [];
}
const element = HEADER.querySelector(elementSelector);
if (!element) {
logError(`Selector '${elementSelector}' not found`);
return [];
}
const elementClone = element.cloneNode(true);
const elementCloneId = `${elementId}-clone`;
elementClone.setAttribute('id', elementCloneId);
elementClone.style.setProperty('display', 'none');
HEADER_STYLE.textContent += cssHideElement(elementSelector);
HEADER_STYLE.textContent += `
${createId(elementCloneId)}
{
display: flex !important;
}
`;
leftAlignedDiv.appendChild(elementClone);
NEW_ELEMENTS.push(elementClone);
return [elementCloneId, elementClone];
}
function insertNewGlobalBar(element) {
log(DEBUG, 'insertNewGlobalBar()');
const elementToInsertAfter = HEADER.querySelector(SELECTORS.header.globalBar);
elementToInsertAfter.parentNode.insertBefore(element, elementToInsertAfter.nextSibling);
}
function createId(string) {
log(TRACE, 'createId()');
if (string.startsWith('#')) return string;
if (string.startsWith('.')) {
logError(`Attempted to create an id from a class: "${string}"`);
return;
}
if (string.startsWith('[')) {
logError(`Attempted to create an id from an attribute selector: "${string}"`);
return;
}
return `#${string}`;
}
function cssHideElement(elementSelector) {
log(TRACE, 'cssHideElement()');
return `
${elementSelector}
{
display: none !important;
}
`;
}
function isValidURL(string) {
log(DEBUG, 'isValidURL()');
const urlPattern = /^(https?:\/\/)?([\w.]+)\.([a-z]{2,6}\.?)(\/[\w.]*)*\/?$/i;
return urlPattern.test(string);
}
function escapeRegExp(string) {
log(DEBUG, 'escapeRegExp()');
return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
}
function compareObjects(firstObject, secondObject, firstName, secondName) {
log(DEBUG, 'compareObjects()');
if (typeof firstObject !== 'object' || typeof secondObject !== 'object') {
return 'Invalid input. Please provide valid objects.';
}
const differences = [];
function findKeyDifferences(obj1, obj2, path = '') {
const keys1 = Object.keys(obj1);
const keys2 = Object.keys(obj2);
keys1.forEach(key => {
const nestedPath = path ? `${path}.${key}` : key;
if (!keys2.includes(key)) {
differences.push(`Found "${nestedPath}" in ${firstName} but not in ${secondName}`);
} else if (typeof obj1[key] === 'object' && typeof obj2[key] === 'object') {
findKeyDifferences(obj1[key], obj2[key], nestedPath);
}
});
keys2.forEach(key => {
const nestedPath = path ? `${path}.${key}` : key;
if (!keys1.includes(key)) {
differences.push(`Found "${nestedPath}" in ${secondName} but not in ${firstName}`);
}
});
}
findKeyDifferences(firstObject, secondObject);
return differences.length > 0 ? differences : [];
}
// eslint-disable-next-line no-unused-vars
function checkConfigConsistency(configs) {
log(DEBUG, 'checkConfigConsistency()');
const lightDarkDifference = compareObjects(
configs.happyMedium.light,
configs.happyMedium.dark,
'Happy Medium Light',
'Happy Medium Dark',
);
if (lightDarkDifference.length > 0) {
logError('lightDarkDifference', lightDarkDifference);
return false;
}
const typeDifference = compareObjects(
configs.happyMedium,
configs.oldSchool,
'Happy Medium',
'Old School',
);
if (typeDifference.length > 0) {
logError('typeDifference', typeDifference);
return false;
}
return true;
}
function updateSelectors() {
log(DEBUG, 'updateSelectors()');
const toolTips = Array.from(HEADER.querySelectorAll('tool-tip'));
SELECTORS.toolTips = {
create: toolTips.find(
tooltip => tooltip.textContent.includes('Create new'),
),
pullRequests: toolTips.find(
tooltip => tooltip.textContent.includes('Pull requests'),
),
issues: toolTips.find(
tooltip => tooltip.textContent.includes('Issues'),
),
notifications: toolTips.find(
tooltip => tooltip.getAttribute('data-target') === 'notification-indicator.tooltip',
),
};
}
function waitForFeaturePreviewButton() {
log(VERBOSE, 'waitForFeaturePreviewButton()');
if (!HEADER) return;
const liElementId = 'custom-global-navigation-menu-item';
if (HEADER.querySelector(createId(liElementId))) return;
const featurePreviewSearch = Array.from(
document.querySelectorAll('[data-position-regular="right"] span'),
)?.find(element => element.textContent === 'Feature preview') || null;
if (featurePreviewSearch) {
const featurePreviewSpan = featurePreviewSearch;
const featurePreviewLabelDiv = featurePreviewSpan.parentNode;
const featurePreviewLi = featurePreviewLabelDiv.parentNode;
const newLiElement = featurePreviewLi.cloneNode(true);
newLiElement.setAttribute('id', liElementId);
newLiElement.onclick = () => {
GMC.open();
if (GMC.get('on_open') === 'close sidebar') HEADER.querySelector(SELECTORS.sidebars.right.closeButton)?.click();
};
const textElement = newLiElement.querySelector('[data-component="ActionList.Item--DividerContainer"]');
textElement.textContent = GMC.get('menu_item_title');
const oldSvg = newLiElement.querySelector('svg');
const menuItemIcon = GMC.get('menu_item_icon');
if (menuItemIcon === 'logo') {
const newSvg = document.createElement('img');
newSvg.setAttribute('height', '16px');
newSvg.setAttribute('width', '16px');
newSvg.src = `https://raw.githubusercontent.com/blakegearin/github-custom-global-navigation/main/img/${THEME}_logo.svg`;
oldSvg.parentNode.replaceChild(newSvg, oldSvg);
} else {
let svgString;
if (menuItemIcon === 'cog') {
svgString = `
<svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-gear">
<path d="M8 0a8.2 8.2 0 0 1 .701.031C9.444.095 9.99.645 10.16 1.29l.288 1.107c.018.066.079.158.212.224.231.114.454.243.668.386.123.082.233.09.299.071l1.103-.303c.644-.176 1.392.021 1.82.63.27.385.506.792.704 1.218.315.675.111 1.422-.364 1.891l-.814.806c-.049.048-.098.147-.088.294.016.257.016.515 0 .772-.01.147.038.246.088.294l.814.806c.475.469.679 1.216.364 1.891a7.977 7.977 0 0 1-.704 1.217c-.428.61-1.176.807-1.82.63l-1.102-.302c-.067-.019-.177-.011-.3.071a5.909 5.909 0 0 1-.668.386c-.133.066-.194.158-.211.224l-.29 1.106c-.168.646-.715 1.196-1.458 1.26a8.006 8.006 0 0 1-1.402 0c-.743-.064-1.289-.614-1.458-1.26l-.289-1.106c-.018-.066-.079-.158-.212-.224a5.738 5.738 0 0 1-.668-.386c-.123-.082-.233-.09-.299-.071l-1.103.303c-.644.176-1.392-.021-1.82-.63a8.12 8.12 0 0 1-.704-1.218c-.315-.675-.111-1.422.363-1.891l.815-.806c.05-.048.098-.147.088-.294a6.214 6.214 0 0 1 0-.772c.01-.147-.038-.246-.088-.294l-.815-.806C.635 6.045.431 5.298.746 4.623a7.92 7.92 0 0 1 .704-1.217c.428-.61 1.176-.807 1.82-.63l1.102.302c.067.019.177.011.3-.071.214-.143.437-.272.668-.386.133-.066.194-.158.211-.224l.29-1.106C6.009.645 6.556.095 7.299.03 7.53.01 7.764 0 8 0Zm-.571 1.525c-.036.003-.108.036-.137.146l-.289 1.105c-.147.561-.549.967-.998 1.189-.173.086-.34.183-.5.29-.417.278-.97.423-1.529.27l-1.103-.303c-.109-.03-.175.016-.195.045-.22.312-.412.644-.573.99-.014.031-.021.11.059.19l.815.806c.411.406.562.957.53 1.456a4.709 4.709 0 0 0 0 .582c.032.499-.119 1.05-.53 1.456l-.815.806c-.081.08-.073.159-.059.19.162.346.353.677.573.989.02.03.085.076.195.046l1.102-.303c.56-.153 1.113-.008 1.53.27.161.107.328.204.501.29.447.222.85.629.997 1.189l.289 1.105c.029.109.101.143.137.146a6.6 6.6 0 0 0 1.142 0c.036-.003.108-.036.137-.146l.289-1.105c.147-.561.549-.967.998-1.189.173-.086.34-.183.5-.29.417-.278.97-.423 1.529-.27l1.103.303c.109.029.175-.016.195-.045.22-.313.411-.644.573-.99.014-.031.021-.11-.059-.19l-.815-.806c-.411-.406-.562-.957-.53-1.456a4.709 4.709 0 0 0 0-.582c-.032-.499.119-1.05.53-1.456l.815-.806c.081-.08.073-.159.059-.19a6.464 6.464 0 0 0-.573-.989c-.02-.03-.085-.076-.195-.046l-1.102.303c-.56.153-1.113.008-1.53-.27a4.44 4.44 0 0 0-.501-.29c-.447-.222-.85-.629-.997-1.189l-.289-1.105c-.029-.11-.101-.143-.137-.146a6.6 6.6 0 0 0-1.142 0ZM11 8a3 3 0 1 1-6 0 3 3 0 0 1 6 0ZM9.5 8a1.5 1.5 0 1 0-3.001.001A1.5 1.5 0 0 0 9.5 8Z"></path>
</svg>
`;
} else if (menuItemIcon === 'compass') {
svgString = `
<svg xmlns="http://www.w3.org/2000/svg" height="1em" viewBox="0 0 512 512">
<!--! Font Awesome Free 6.4.2 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license (Commercial License) Copyright 2023 Fonticons, Inc. -->
<path d="M464 256A208 208 0 1 0 48 256a208 208 0 1 0 416 0zM0 256a256 256 0 1 1 512 0A256 256 0 1 1 0 256zm306.7 69.1L162.4 380.6c-19.4 7.5-38.5-11.6-31-31l55.5-144.3c3.3-8.5 9.9-15.1 18.4-18.4l144.3-55.5c19.4-7.5 38.5 11.6 31 31L325.1 306.7c-3.2 8.5-9.9 15.1-18.4 18.4zM288 256a32 32 0 1 0 -64 0 32 32 0 1 0 64 0z"/>
</svg>
`;
}
const parser = new DOMParser();
const svgDoc = parser.parseFromString(svgString, 'image/svg+xml');
const newSvg = svgDoc.documentElement;
oldSvg.parentNode.replaceChild(newSvg, oldSvg);
}
const parentUl = featurePreviewLi.parentNode;
const settingsLi = document.querySelector('[data-position-regular="right"] a[href="/settings/profile"]').parentNode;
parentUl.insertBefore(newLiElement, settingsLi.nextSibling);
const divider = featurePreviewLi.parentNode.querySelector('.ActionList-sectionDivider');
const newDivider = divider.cloneNode(true);
parentUl.insertBefore(newDivider, settingsLi.nextSibling);
} else {
setTimeout(waitForFeaturePreviewButton, 100);
}
}
function generateCustomConfig() {
log(DEBUG, 'generateCustomConfig()');
const customConfig = {
light: {},
dark: {},
};
function recursivelyGenerateCustomConfig(obj, customObj, themePrefix, parentKey = '') {
for (const key in obj) {
const currentKey = parentKey ? `${parentKey}.${key}` : key;
if (typeof obj[key] === 'object') {
customObj[key] = {};
recursivelyGenerateCustomConfig(obj[key], customObj[key], themePrefix, currentKey);
} else {
const gmcKey = `${themePrefix}_${currentKey.replace(/\./g, '_')}`;
if (gmcKey in GMC.fields) {
customObj[key] = GMC.get(gmcKey);
} else {
logError(`GMC field not found for key: ${gmcKey}`);
return;
}
}
}
}
recursivelyGenerateCustomConfig(configs.happyMedium.light, customConfig.light, 'light');
recursivelyGenerateCustomConfig(configs.happyMedium.dark, customConfig.dark, 'dark');
return customConfig;
}
function setTheme() {
log(DEBUG, 'setTheme()');
const dataColorMode = document.querySelector('html').getAttribute('data-color-mode');
if (dataColorMode === 'auto') {
if (window.matchMedia('(prefers-color-scheme: dark)').matches) {
THEME = 'dark';
}
} else if (dataColorMode === 'dark') {
THEME = 'dark';
} else if (dataColorMode !== 'light') {
logError('Unknown color mode');
}
log(VERBOSE, `THEME: ${THEME}`);
}
function gmcInitialized() {
log(DEBUG, 'gmcInitialized()');
updateLogLevel();
log(QUIET, 'Running');
GMC.css.basic = '';
startObserving();
}
function gmcAddSavedSpan(div) {
log(DEBUG, 'gmcAddSavedSpan()');
const savedDiv = document.createElement('div');
savedDiv.setAttribute('id', 'gmc-saved');
const iconSpan = document.createElement('span');
iconSpan.style = 'margin-right: 4px;';
iconSpan.innerHTML = `
<svg aria-hidden="true" focusable="false" role="img" class="octicon octicon-check-circle-fill" viewBox="0 0 12 12" width="12" height="12" fill="currentColor" style="display: inline-block;user-select: none;vertical-align: text-bottom;">
<path d="M6 0a6 6 0 1 1 0 12A6 6 0 0 1 6 0Zm-.705 8.737L9.63 4.403 8.392 3.166 5.295 6.263l-1.7-1.702L2.356 5.8l2.938 2.938Z"></path>
</svg>
`;
const textSpan = document.createElement('span');
textSpan.innerText = 'Saved';
savedDiv.appendChild(iconSpan);
savedDiv.appendChild(textSpan);
div.insertBefore(savedDiv, div.firstChild);
}
function gmcAddNewIssueButton(div) {
log(DEBUG, 'gmcAddNewIssueButton()');
const small = document.createElement('small');
small.classList.add('left-aligned');
small.setAttribute('title', 'Submit bug or feature request');
const link = document.createElement('a');
link.href = 'https://github.com/blakegearin/github-custom-global-navigation/issues';
link.innerText = 'submit bug or feature request';
small.appendChild(link);
div.insertBefore(small, div.firstChild);
}
function gmcOpened() {
log(DEBUG, 'gmcOpened()');
function updateCheckboxes() {
log(DEBUG, 'updateCheckboxes()');
const checkboxes = document.querySelectorAll('#gmc-frame input[type="checkbox"]');
if (checkboxes.length > 0) {
checkboxes.forEach(checkbox => {
checkbox.classList.add('gmc-checkbox');
});
} else {
setTimeout(updateCheckboxes, 100);
}
}
updateCheckboxes();
const configVars = document.querySelectorAll('.config_var');
configVars.forEach(configVar => {
const label = configVar.querySelector('.field_label');
const input = configVar.querySelector('input');
if (label && input && input.type === 'text') label.style.lineHeight = '33px';
const select = configVar.querySelector('select');
if (label && select) label.style.lineHeight = '33px';
});
modifyThenObserve(() => {
document.querySelector('#gmc-frame .reset_holder').remove();
const buttonHolderSelector = '#gmc-frame_buttons_holder';
const parentDiv = document.querySelector(buttonHolderSelector);
if (!parentDiv) {
logError(`Selector ${buttonHolderSelector} not found`);
return;
}
gmcAddSavedSpan(parentDiv);
gmcAddNewIssueButton(parentDiv);
});
document.querySelector('#gmc').classList.remove('hidden');
}
function gmcRefreshTab() {
location.reload();
}
function gmcRunScript() {
applyCustomizations(true);
}
function updateLogLevel() {
CURRENT_LOG_LEVEL = {
'silent': SILENT,
'quiet': QUIET,
'debug': DEBUG,
'verbose': VERBOSE,
'trace': TRACE,
}[GMC.get('log_level')];
}
function gmcSaved() {
log(DEBUG, 'gmcSaved()');
const gmcSaved = document.getElementById('gmc-saved');
gmcSaved.style.display = 'block';
setTimeout(
() => gmcSaved.style.display = 'none',
2750,
);
updateLogLevel();
switch (GMC.get('on_save')) {
case 'refresh tab':
gmcRefreshTab();
break;
case 'refresh tab and close':
gmcRefreshTab();
GMC.close();
break;
case 'run script':
gmcRunScript();
break;
case 'run script and close':
gmcRunScript();
GMC.close();
break;
}
}
function gmcClosed() {
log(DEBUG, 'gmcClosed()');
switch (GMC.get('on_close')) {
case 'refresh tab':
gmcRefreshTab();
break;
case 'run script':
gmcRunScript();
break;
}
document.querySelector('#gmc').classList.add('hidden');
}
function gmcClearCustom() {
log(DEBUG, 'gmcClearCustom()');
const confirmed = confirm('Are you sure you want to clear your custom configuration? This is irreversible.');
if (confirmed) {
const currentType = GMC.get('type');
GMC.reset();
GMC.save();
GMC.set('type', currentType);
GMC.save();
}
}
function configsToGMC(config, path = []) {
log(DEBUG, 'configsToGMC()');
for (const key in config) {
if (typeof config[key] === 'object' && !Array.isArray(config[key])) {
configsToGMC(config[key], path.concat(key));
} else {
const fieldName = path.concat(key).join('_');
const fieldValue = config[key];
log(VERBOSE, 'fieldName', fieldName);
GMC.set(fieldName, fieldValue);
}
}
}
function gmcApplyCustomHappyMediumConfig() {
log(DEBUG, 'gmcApplyCustomHappyMediumConfig()');
const confirmed = confirm('Are you sure you want to overwrite your custom configuration with Happy Medium? This is irreversible.');
if (confirmed) {
configsToGMC(configs.happyMedium);
GMC.save();
}
}
function gmcApplyCustomOldSchoolConfig() {
log(DEBUG, 'gmcApplyCustomOldSchoolConfig()');
const confirmed = confirm('Are you sure you want to overwrite your custom configuration with Old School? This is irreversible.');
if (confirmed) {
configsToGMC(configs.oldSchool);
GMC.save();
}
}
function gmcBuildStyle() {
log(DEBUG, 'gmcBuildStyle()');
const headerIdPartials = [
'hamburgerButton_remove_var',
'logo_remove_var',
'pageTitle_remove_var',
'search_remove_var',
'divider_remove_var',
'create_remove_var',
'issues_remove_var',
'pullRequests_remove_var',
'marketplace_add_var',
'explore_add_var',
'notifications_remove_var',
'light_avatar_remove_var',
'dark_avatar_remove_var',
'globalBar_boxShadowColor_var',
'localBar_backgroundColor_var',
'sidebars_backdrop_color_var',
'repositoryHeader_import_var',
'flipCreateInbox_var',
'flipIssuesPullRequests_var',
];
const sectionSelectors = headerIdPartials
.map(varName => `#gmc-frame .config_var[id*='${varName}']`)
.join(',\n');
const gmcFrameStyle = document.createElement('style');
gmcFrameStyle.textContent += `
/* Modal */
#gmc
{
display: inline-flex !important;
justify-content: center !important;
align-items: center !important;
position: fixed !important;
top: 0 !important;
left: 0 !important;
width: 100vw !important;
height: 100vh !important;
z-index: 9999;
background: none !important;
pointer-events: none;
}
#gmc.hidden
{
display: none !important;
}
#gmc-frame
{
font-family: -apple-system,BlinkMacSystemFont,"Segoe UI","Noto Sans",Helvetica,Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji";
text-align: left;
inset: initial !important;
border: none !important;
max-height: initial !important;
max-width: initial !important;
opacity: 1 !important;
position: static !important;
z-index: initial !important;
width: 85% !important;
height: 75% !important;
overflow-y: auto !important;
border: none !important;
border-radius: 0.375rem !important;
pointer-events: auto;
}
#gmc-frame_wrapper
{
display: flow-root !important;
padding: 2rem !important;
}
/* Sections */
#gmc-frame #gmc-frame_section_0
{
width: 100%;
border-radius: 6px;
display: table;
}
#gmc-frame #gmc-frame_section_1,
#gmc-frame #gmc-frame_section_2,
#gmc-frame #gmc-frame_section_3,
#gmc-frame #gmc-frame_section_4
{
margin-top: 2rem;
width: 49%;
box-sizing: border-box;
}
#gmc-frame #gmc-frame_section_1
{
border-radius: 6px;
float: left;
}
#gmc-frame #gmc-frame_section_2
{
border-radius: 6px;
float: right;
}
#gmc-frame #gmc-frame_section_3
{
width: 49%;
margin-top: 2rem;
box-sizing: border-box;
border-radius: 6px;
float: left;
}
#gmc-frame #gmc-frame_section_4
{
display: inline-grid;
width: 49%;
margin-top: 2rem;
box-sizing: border-box;
border-radius: 6px;
float: right
}
#gmc-frame #gmc-frame_section_3 .config_var:not(:last-child),
#gmc-frame #gmc-frame_section_4 .config_var:not(:last-child)
{
padding-bottom: 1rem;
}
/* Fields */
#gmc-frame .config_header
{
font-size: 2em;
font-weight: 400;
line-height: 1.25;
padding-bottom: 0.3em;
margin-bottom: 16px;
}
#gmc-frame #gmc-frame_type_var
{
display: inline-flex;
}
#gmc-frame .section_header
{
font-size: 1.5em;
font-weight: 600;
line-height: 1.25;
margin-bottom: 16px;
padding: 1rem 1.5rem;
}
#gmc-frame .section_desc,
#gmc-frame h3
{
background: none;
border: none;
font-size: 1.25em;
margin-bottom: 16px;
font-weight: 600;
line-height: 1.25;
text-align: left;
}
#gmc-frame .config_var
{
padding: 0rem 1.5rem;
margin-bottom: 1rem;
display: flex;
}
${sectionSelectors}
{
display: flow;
padding-top: 1rem;
}
#gmc-frame .config_var[id*='flipCreateInbox_var'],
#gmc-frame .config_var[id*='flipIssuesPullRequests_var']
{
display: flex;
}
#gmc-frame .field_label
{
font-weight: 600;
margin-right: 0.5rem;
}
#gmc-frame .field_label,
#gmc-frame .gmc-label
{
width: 15vw;
}
#gmc-frame .radio_label:not(:last-child)
{
margin-right: 4rem;
}
#gmc-frame .radio_label
{
line-height: 17px;
}
#gmc-frame .gmc-label
{
display: table-caption;
line-height: 17px;
}
#gmc-frame input[type="radio"]
{
appearance: none;
border-style: solid;
cursor: pointer;
height: 1rem;
place-content: center;
position: relative;
width: 1rem;
border-radius: 624rem;
transition: background-color 0s ease 0s, border-color 80ms cubic-bezier(0.33, 1, 0.68, 1) 0s;
margin-right: 0.5rem;
flex: none;
}
#gmc-frame input[type="checkbox"]
{
appearance: none;
border-style: solid;
border-width: 1px;
cursor: pointer;
place-content: center;
position: relative;
height: 17px;
width: 17px;
border-radius: 3px;
transition: background-color 0s ease 0s, border-color 80ms cubic-bezier(0.33, 1, 0.68, 1) 0s;
}
#gmc-frame #gmc-frame_field_type
{
display: flex;
}
#gmc-frame input[type="radio"]:checked
{
border-width: 0.25rem;
}
#gmc-frame input[type="radio"]:checked,
#gmc-frame .gmc-checkbox:checked
{
border-color: #2f81f7;
}
#gmc-frame .gmc-checkbox:checked
{
background-color: #2f81f7;
}
#gmc-frame .gmc-checkbox:checked::before
{
visibility: visible;
transition: visibility 0s linear 0s;
}
#gmc-frame .gmc-checkbox::before,
#gmc-frame .gmc-checkbox:indeterminate::before
{
animation: 80ms cubic-bezier(0.65, 0, 0.35, 1) 80ms 1 normal forwards running checkmarkIn;
}
#gmc-frame .gmc-checkbox::before
{
width: 1rem;
height: 1rem;
visibility: hidden;
content: "";
background-color: #FFFFFF;
clip-path: inset(1rem 0 0 0);
-webkit-mask-image: url("");
-webkit-mask-size: 75%;
-webkit-mask-repeat: no-repeat;
-webkit-mask-position: center center;
display: block;
}
#gmc-frame .gmc-checkbox
{
appearance: none;
border-style: solid;
border-width: 1px;
cursor: pointer;
height: var(--base-size-16,16px);
margin: 0.125rem 0px 0px;
place-content: center;
position: relative;
width: var(--base-size-16,16px);
border-radius: 3px;
transition: background-color 0s ease 0s, border-color 80ms cubic-bezier(0.33, 1, 0.68, 1) 0s;
}
#gmc-frame input
{
color: fieldtext;
letter-spacing: normal;
word-spacing: normal;
text-transform: none;
text-indent: 0px;
text-shadow: none;
display: inline-block;
text-align: start;
appearance: auto;
-webkit-rtl-ordering: logical;
}
#gmc-frame .gmc-checkbox:checked
{
transition: background-color 0s ease 0s, border-color 80ms cubic-bezier(0.32, 0, 0.67, 0) 0ms;
}
#gmc-frame input[type="text"],
#gmc-frame textarea,
#gmc-frame select
{
padding: 5px 12px;
border-radius: 6px;
}
#gmc-frame input[type="text"]:focus,
#gmc-frame textarea:focus,
#gmc-frame select:focus
{
border-color: #2f81f7;
outline: 1px solid #2f81f7;
}
#gmc-frame svg
{
height: 17px;
width: 17px;
margin-left: 0.5rem;
}
#gmc small
{
font-size: x-small;
font-weight: 600;
margin-left: 3px;
}
/* Button bar */
#gmc-frame #gmc-frame_buttons_holder
{
position: fixed;
width: 85%;
text-align: right;
left: 50%;
bottom: 2%;
transform: translate(-50%, 0%);
padding: 1rem;
border-radius: 0.375rem;
display: flex;
align-items: center;
}
#gmc-frame #gmc-frame_buttons_holder .left-aligned
{
order: 1;
margin-right: auto;
}
#gmc-frame #gmc-frame_buttons_holder > *
{
order: 2;
}
#gmc-frame .saveclose_buttons
{
margin-left: 0.5rem;
}
#gmc-frame [type=button],
#gmc-frame .saveclose_buttons
{
position: relative;
display: inline-block;
padding: 5px 16px;
font-size: 14px;
font-weight: 500;
line-height: 20px;
white-space: nowrap;
vertical-align: middle;
cursor: pointer;
-webkit-user-select: none;
user-select: none;
border: 1px solid;
border-radius: 6px;
-webkit-appearance: none;
appearance: none;
font-family: -apple-system,BlinkMacSystemFont,"Segoe UI","Noto Sans",Helvetica,Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji";
}
@keyframes fadeOut
{
from {
opacity: 1;
}
to {
opacity: 0;
}
}
#gmc-saved
{
display: none;
margin-right: 10px;
animation: fadeOut 0.75s ease 2s forwards;
}
`;
if (THEME === 'light') {
gmcFrameStyle.textContent += `
#gmc-frame
{
background-color: #F6F8FA;
color: #1F2328;
box-shadow: 0 0 0 1px #D0D7DE, 0 16px 32px rgba(1,4,9,0.2) !important;
}
#gmc-frame .section_header_holder
{
background-color: #FFFFFF;
border: 1px solid #D0D7DE;
}
#gmc-frame_buttons_holder
{
background-color: #FFFFFF;
box-shadow: 0 0 0 1px #D0D7DE, 0 16px 32px rgba(1,4,9,0.2) !important;
}
#gmc-frame input[type="text"],
#gmc-frame textarea,
#gmc-frame select
{
border: 1px solid #D0D7DE;
}
#gmc-frame select
{
background-color: #F6F8FA;
}
#gmc-frame select:hover
{
background-color: #F3F4F6;
border-color: #1F232826;
}
#gmc-frame input[type="text"],
#gmc-frame textarea
{
background-color: #F6F8FA;
color: #1F2328;
}
#gmc-frame input[type="text"]:focus,
#gmc-frame textarea:focus
{
background-color: #FFFFFF;
}
#gmc-frame [type=button],
#gmc-frame .saveclose_buttons
{
background-color: #f6f8fa;
border-color: #1f232826;
box-shadow: 0 1px 0 rgba(31,35,40,0.04), inset 0 1px 0 rgba(255,255,255,0.25);
color: #24292f;
}
#gmc-frame [type=button]:hover,
#gmc-frame .saveclose_buttons:hover
{
background-color: #f3f4f6;
border-color: #1f232826;
}
#gmc-frame .gmc-checkbox
{
border-color: #6E7781;
}
#gmc-frame input[type="radio"]
{
color: #6E7781;
}
#gmc-frame svg
{
fill: #000000;
}
#gmc-frame .section_header
{
border-bottom: 1px solid #D0D7DE;
}
${sectionSelectors}
{
border-top: 1px solid #D0D7DE;
}
#gmc-frame #gmc-frame_section_3 .config_var:not(:last-child),
#gmc-frame #gmc-frame_section_4 .config_var:not(:last-child)
{
border-bottom: 1px solid #D0D7DE;
}
#gmc-frame #gmc-frame_saveBtn
{
background-color: #1F883D;
border-color: rgba(31, 35, 40, 0.15);
box-shadow: rgba(31, 35, 40, 0.1) 0px 1px 0px;
color: #FFFFFF;
}
#gmc-frame #gmc-frame_saveBtn:hover
{
background-color: rgb(26, 127, 55);
}
#gmc-frame #gmc-frame_section_4
{
border: 1px solid #FF818266;
}
#gmc-frame #gmc-frame_section_4 input
{
background-color: #F6F8FA;
border-color: #1F232826;
box-shadow: 0 1px 0 rgba(31,35,40,0.04), inset 0 1px 0 rgba(255,255,255,0.25);
color: #CF222E;
}
#gmc-frame #gmc-frame_section_4 input:hover
{
background-color: #A40E26;
border-color: #1F232826;
box-shadow: 0 1px 0 rgba(31,35,40,0.04);
color: #ffffff;
}
#gmc-saved
{
color: #1a7f37;
}
#gmc-saved svg path
{
fill: #1a7f37;
}
`;
} else if (THEME === 'dark') {
gmcFrameStyle.textContent += `
#gmc-frame
{
background-color: #161B22;
color: #E6EDF3;
box-shadow: 0 0 0 1px #30363D, 0 16px 32px #010409 !important;
}
#gmc-frame .section_header_holder
{
background-color: #0D1117;
border: 1px solid #30363D;
}
#gmc-frame_buttons_holder
{
background-color: #161B22;
box-shadow: 0 0 0 1px #30363D, 0 16px 32px #010409 !important;
}
#gmc-frame input[type="text"],
#gmc-frame textarea,
#gmc-frame select
{
border: 1px solid #5B626C;
}
#gmc-frame input[type="text"],
#gmc-frame textarea
{
background-color: #010409;
color: #FFFFFF;
}
#gmc-frame [type=button]:hover,
#gmc-frame .saveclose_buttons:hover
{
background-color: #30363d;
border-color: #8b949e;
}
#gmc-frame .gmc-checkbox
{
border-color: #6E7681;
}
#gmc-frame input[type="radio"]
{
color: #6D7681;
}
#gmc-frame input[type="text"]:focus,
textarea:focus
{
background-color: #0D1117;
}
#gmc-frame [type=button],
#gmc-frame .saveclose_buttons
{
color: #c9d1d9;
background-color: #21262d;
border-color: #f0f6fc1a;
}
#gmc-frame svg
{
fill: #E6EDF3;
}
#gmc-frame .section_header
{
border-bottom: 1px solid #30363D;
}
${sectionSelectors}
{
border-top: 1px solid #30363D;
}
#gmc-frame #gmc-frame_section_3 .config_var:not(:last-child),
#gmc-frame #gmc-frame_section_4 .config_var:not(:last-child)
{
padding-bottom: 1rem;
border-bottom: 1px solid #30363D;
}
#gmc-frame #gmc-frame_saveBtn
{
background-color: #238636;
border-color: #F0F6FC1A;
box-shadow: 0 0 transparent;
color: #FFFFFF;
}
#gmc-frame #gmc-frame_saveBtn:hover
{
background-color: #2EA043;
border-color: #F0F6FC1A;
}
#gmc-frame #gmc-frame_section_4
{
border: 1px solid #f8514966;
}
#gmc-frame #gmc-frame_section_4 input
{
background-color: #21262D;
border-color: #F0F6FC1A;
}
#gmc-frame #gmc-frame_section_4 input
{
color: #F85149;
}
#gmc-frame #gmc-frame_section_4 input:hover
{
background-color: #DA3633;
border-color: #F85149;
color: #FFFFFF;
}
#gmc-saved
{
color: #3FB950;
}
#gmc-saved svg path
{
fill: #3FB950;
}
`;
}
document.head.appendChild(gmcFrameStyle);
}
function gmcBuildFrame() {
log(DEBUG, 'gmcBuildFrame()');
const body = document.querySelector('body');
const gmcDiv = document.createElement('div');
gmcDiv.setAttribute('id', 'gmc');
gmcDiv.classList.add('hidden');
body.appendChild(gmcDiv);
const gmcFrameDiv = document.createElement('div');
gmcFrameDiv.setAttribute('id', 'gmc-frame');
gmcDiv.appendChild(gmcFrameDiv);
gmcBuildStyle();
return gmcFrameDiv;
}
function applyCustomizations(refresh = false) {
log(DEBUG, 'applyCustomizations()');
log(DEBUG, 'refresh', refresh);
HEADER = document.querySelector(SELECTORS.header.self);
if (!HEADER) return 'continue';
const featurePreviewButton = document.querySelector(SELECTORS.avatar.button);
if (!featurePreviewButton) {
logError(`Selector ${SELECTORS.avatar.button} not found`);
return 'break';
}
featurePreviewButton.addEventListener('click', waitForFeaturePreviewButton);
CONFIG_NAME = {
'Off': 'off',
'Happy Medium': 'happyMedium',
'Old School': 'oldSchool',
'Custom': 'custom',
}[GMC.get('type')];
log(DEBUG, 'CONFIG_NAME', CONFIG_NAME);
if (CONFIG_NAME === 'off') return 'break';
if (CONFIG_NAME === 'custom') configs.custom = generateCustomConfig();
CONFIG = configs[CONFIG_NAME][THEME];
log(VERBOSE, 'CONFIG', CONFIG);
const headerSuccessFlag = 'customizedHeader';
const foundHeaderSuccessFlag = document.getElementById(headerSuccessFlag);
log(DEBUG, 'foundHeaderSuccessFlag', foundHeaderSuccessFlag);
const configurationApplied = HEADER.classList.contains(CONFIG_NAME);
if (!configurationApplied && (foundHeaderSuccessFlag === null || refresh)) {
updateSelectors();
if (refresh) {
modifyThenObserve(() => {
document.querySelector(createId(SELECTORS.header.style))?.remove();
HEADER_STYLE.textContent = '';
HEADER.classList.remove(OLD_CONFIG_NAME);
NEW_ELEMENTS.forEach((element) => element.remove());
});
}
if (CONFIG_NAME === 'oldSchool') {
HEADER_STYLE.textContent += `
@media (max-width: 767.98px)
{
action-menu
{
display: none !important;
}
}
`;
}
HEADER_UPDATES_COUNT++;
updateHeader();
HEADER.setAttribute('id', headerSuccessFlag);
HEADER.classList.add(CONFIG_NAME);
OLD_CONFIG_NAME = CONFIG_NAME;
log(QUIET, 'Complete');
return 'break';
} else {
if (CONFIG.avatar.dropdownIcon) insertAvatarDropdown();
if (CONFIG.avatar.canCloseSidebar) updateAvatarButton();
if (CONFIG.repositoryHeader.import) {
// When starting in a repository tab like Issues, the proper repository header
// (including Unwatch, Star, and Fork) is not present per GitHub's design.
// If page title is removed, the page will be missing any location context in the header.
// To improve this experience, a temporary repository header is created with the
// page title or breadcrumbs.
// The proper repository header replaces the temporary one when navigating to the Code tab.
if (
!document.querySelector(SELECTORS.repositoryHeader.id)?.hidden &&
(
document.querySelector(createId(TEMP_REPOSITORY_HEADER_FLAG)) ||
!document.querySelector(`.${REPOSITORY_HEADER_SUCCESS_FLAG}`)
)
) {
const updated = importRepositoryHeader();
if (updated) {
HEADER_UPDATES_COUNT++;
log(QUIET, 'Repository header updated');
} else {
IDLE_MUTATION_COUNT++;
}
return 'break';
}
}
}
if (CONFIG.avatar.dropdownIcon) insertAvatarDropdown();
}
function startObserving() {
log(DEBUG, 'startObserving()');
OBSERVER.observe(
document.body,
{
childList: true,
subtree: true,
},
);
}
function modifyThenObserve(callback) {
log(DEBUG, 'modifyThenObserve()');
OBSERVER.disconnect();
callback();
startObserving();
}
function observeAndModify(mutationsList) {
log(VERBOSE, 'observeAndModify()');
if (IDLE_MUTATION_COUNT > MAX_IDLE_MUTATIONS) {
// This is a failsafe to prevent infinite loops
logError('MAX_IDLE_MUTATIONS exceeded');
OBSERVER.disconnect();
return;
} else if (HEADER_UPDATES_COUNT >= MAX_HEADER_UPDATES) {
// This is a failsafe to prevent infinite loops
logError('MAX_HEADER_UPDATES exceeded');
OBSERVER.disconnect();
return;
}
for (const mutation of mutationsList) {
// Use header id to determine if updates have already been applied
if (mutation.type !== 'childList') return;
log(TRACE, 'mutation', mutation);
const outcome = applyCustomizations();
log(DEBUG, 'outcome', outcome);
if (outcome === 'continue') continue;
if (outcome === 'break') break;
}
}
const UNICODE_NON_BREAKING_SPACE = '\u00A0';
const REPOSITORY_HEADER_SUCCESS_FLAG = 'permCustomizedRepositoryHeader';
const TEMP_REPOSITORY_HEADER_FLAG = 'tempCustomizedRepositoryHeader';
const REPOSITORY_HEADER_CLASS = 'customizedRepositoryHeader';
const MAX_IDLE_MUTATIONS = 1000;
const MAX_HEADER_UPDATES = 100;
let CONFIG;
let CONFIG_NAME;
let OLD_CONFIG_NAME;
let HEADER;
let HEADER_STYLE = document.createElement('style');
let THEME = 'light';
let NEW_ELEMENTS = [];
let LEFT_SIDEBAR_PRELOADED = false;
let RIGHT_SIDEBAR_PRELOADED = false;
let IDLE_MUTATION_COUNT = 0;
let HEADER_UPDATES_COUNT = 0;
let SELECTORS = {
header: {
self: 'header.AppHeader',
actionsDiv: '.AppHeader-actions',
globalBar: '.AppHeader-globalBar',
localBar: {
topDiv: '.AppHeader-localBar',
underlineNavActions: '.UnderlineNav-actions',
},
leftAligned: '.AppHeader-globalBar-start',
rightAligned: '.AppHeader-globalBar-end',
style: 'customHeaderStyle',
},
logo: {
topDiv: '.AppHeader-globalBar-start .AppHeader-logo',
svg: '.AppHeader-logo svg',
},
hamburgerButton: '.AppHeader-globalBar-start deferred-side-panel',
pageTitle: {
id: 'custom-page-title',
topDiv: '.AppHeader-context',
links: '.AppHeader-context a',
separator: '.AppHeader-context-item-separator',
},
search: {
id: 'search-div',
topDiv: '.AppHeader-search',
input: '.search-input',
button: '[data-target="qbsearch-input.inputButton"]',
magnifyingGlassIcon: '.AppHeader-search-control label',
commandPalette: '#AppHeader-commandPalette-button',
placeholderSpan: '#qb-input-query',
placeholderDiv: '.AppHeader-search-control .overflow-hidden',
modal: '[data-target="qbsearch-input.queryBuilderContainer"]',
},
create: {
id: 'create-div',
topDiv: 'react-partial-anchor',
button: '#global-create-menu-anchor',
overlay: '#global-create-menu-overlay',
plusIcon: '#global-create-menu-anchor .Button-visual.Button-leadingVisual',
dropdownIcon: '#global-create-menu-anchor .Button-label',
textContent: 'create-text-content-span',
},
issues: {
id: 'issues',
textContent: 'issues-text-content-span',
},
pullRequests: {
id: 'pullRequests',
link: '.AppHeader-globalBar-end .AppHeader-actions a[href="/pulls"]',
textContent: 'pullRequests-text-content-span',
},
marketplace: {
id: 'marketplace',
link: '.AppHeader-globalBar-end .AppHeader-actions a[href="/marketplace"]',
textContent: 'marketplace-text-content-span',
},
explore: {
id: 'explore',
link: '.AppHeader-globalBar-end .AppHeader-actions a[href="/explore"]',
textContent: 'explore-text-content-span',
},
notifications: {
id: 'custom-notifications',
indicator: 'notification-indicator',
dot: '.AppHeader-button.AppHeader-button--hasIndicator::before',
textContent: 'textContent-text-content-spa',
},
avatar: {
topDiv: '.AppHeader-user',
button: '.AppHeader-user button',
img: '.AppHeader-user button img.avatar',
svg: 'avatar-dropdown',
},
repositoryHeader: {
id: '#repository-container-header',
ownerImg: `.${REPOSITORY_HEADER_CLASS} img`,
name: `.${REPOSITORY_HEADER_CLASS} strong`,
links: `.${REPOSITORY_HEADER_CLASS} nav[role="navigation"] a`,
details: '#repository-details-container',
bottomBorder: `.${REPOSITORY_HEADER_CLASS} .border-bottom.mx-xl-5`,
},
sidebars: {
backdrop: 'dialog.Overlay.SidePanel::backdrop',
left: {
modalDialog: '.Overlay--placement-left',
},
right: {
modalDialog: '.AppHeader-user .Overlay--placement-right',
backdrop: '.AppHeader-user .Overlay--placement-right ::backdrop',
closeButton: '.AppHeader-user .Overlay--placement-right .Overlay-closeButton.close-button',
navParentDiv: '.AppHeader-user .Overlay--placement-right div.Overlay-body > div',
nav: '.AppHeader-user .Overlay--placement-right nav',
},
},
};
HEADER_STYLE.setAttribute('id', SELECTORS.header.style);
setTheme();
const oldSchoolColor = '#F0F6FC';
const oldSchoolHoverColor = '#FFFFFFB3';
const oldSchoolHoverBackgroundColor = 'transparent';
let configs = {
happyMedium: {
light: {
backgroundColor: '',
hamburgerButton: {
remove: false,
},
logo: {
remove: false,
color: '',
customSvg: '',
},
pageTitle: {
remove: false,
color: '',
hover: {
backgroundColor: '',
color: '',
},
},
search: {
remove: false,
backgroundColor: '',
borderColor: '',
boxShadow: '',
alignLeft: false,
width: 'max',
margin: {
left: '',
right: '',
},
magnifyingGlassIcon: {
remove: false,
},
placeholder: {
text: '',
color: '',
},
rightButton: 'command palette',
modal: {
width: '',
},
},
divider: {
remove: true,
},
flipCreateInbox: false,
create: {
remove: false,
border: true,
tooltip: false,
boxShadow: '',
hoverBackgroundColor: '',
plusIcon: {
remove: false,
color: '',
marginRight: '0.25rem',
hover: {
color: '',
},
},
text: {
content: 'Create',
color: '',
},
dropdownIcon: {
remove: false,
color: '',
hover: {
color: '',
},
},
},
flipIssuesPullRequests: true,
issues: {
remove: false,
border: true,
tooltip: false,
alignLeft: false,
boxShadow: '',
icon: {
remove: false,
color: '',
},
text: {
content: 'Issues',
color: '',
},
hover: {
backgroundColor: '',
color: '',
},
},
pullRequests: {
remove: false,
border: true,
tooltip: false,
alignLeft: false,
boxShadow: '',
icon: {
remove: false,
color: '',
},
text: {
content: 'Pull requests',
color: '',
},
hover: {
backgroundColor: '',
color: '',
},
},
marketplace: {
add: false,
border: false,
alignLeft: false,
boxShadow: '',
icon: {
remove: false,
color: '',
},
text: {
content: 'Marketplace',
color: '',
},
hover: {
backgroundColor: '',
color: '',
},
},
explore: {
add: false,
border: false,
alignLeft: false,
boxShadow: '',
icon: {
remove: false,
color: '',
},
text: {
content: 'Explore',
color: '',
},
hover: {
backgroundColor: '',
color: '',
},
},
notifications: {
remove: false,
border: true,
tooltip: false,
boxShadow: '',
hoverBackgroundColor: '',
icon: {
symbol: 'bell', // Accepts 'inbox', 'bell', or ''
color: '',
hover: {
color: '',
},
},
text: {
content: 'Inbox',
color: '',
},
dot: {
remove: false,
boxShadowColor: '',
color: '',
displayOverIcon: true,
},
},
avatar: {
remove: false,
size: '',
dropdownIcon: false,
canCloseSidebar: true,
},
globalBar: {
boxShadowColor: '',
leftAligned: {
gap: '',
},
rightAligned: {
gap: '',
},
},
localBar: {
backgroundColor: '#F6F8FA',
alignCenter: false,
boxShadow: {
consistentColor: true,
},
links: {
color: '',
},
},
sidebars: {
backdrop: {
color: 'transparent',
},
left: {
preload: true,
},
right: {
preload: true,
floatUnderneath: false,
width: '',
maxHeight: '',
},
},
repositoryHeader: {
import: true,
alignCenter: false,
removePageTitle: true,
backgroundColor: '#F6F8FA',
avatar: {
remove: false,
customSvg: '',
},
link: {
color: '',
hover: {
backgroundColor: 'transparent',
color: 'var(--color-accent-fg)',
textDecoration: 'underline',
},
},
},
},
dark: {
backgroundColor: '',
hamburgerButton: {
remove: false,
},
logo: {
remove: false,
color: '',
customSvg: '',
},
pageTitle: {
remove: false,
color: '',
hover: {
backgroundColor: '',
color: '',
},
},
search: {
remove: false,
backgroundColor: '',
borderColor: '',
boxShadow: '',
alignLeft: false,
width: 'max',
margin: {
left: '',
right: '',
},
magnifyingGlassIcon: {
remove: false,
},
placeholder: {
text: '',
color: '',
},
rightButton: 'command palette',
modal: {
width: '',
},
},
divider: {
remove: true,
},
flipCreateInbox: false,
create: {
remove: false,
border: true,
tooltip: false,
boxShadow: '',
hoverBackgroundColor: '',
plusIcon: {
remove: false,
color: '',
marginRight: '0.25rem',
hover: {
color: '',
},
},
text: {
content: 'Create',
color: '',
},
dropdownIcon: {
remove: false,
color: '',
hover: {
color: '',
},
},
},
flipIssuesPullRequests: true,
issues: {
remove: false,
border: true,
tooltip: false,
alignLeft: false,
boxShadow: '',
icon: {
remove: false,
color: '',
},
text: {
content: 'Issues',
color: '',
},
hover: {
backgroundColor: '',
color: '',
},
},
pullRequests: {
remove: false,
border: true,
tooltip: false,
alignLeft: false,
boxShadow: '',
icon: {
remove: false,
color: '',
},
text: {
content: 'Pull requests',
color: '',
},
hover: {
backgroundColor: '',
color: '',
},
},
marketplace: {
add: false,
border: false,
alignLeft: false,
boxShadow: '',
icon: {
remove: false,
color: '',
},
text: {
content: 'Marketplace',
color: '',
},
hover: {
backgroundColor: '',
color: '',
},
},
explore: {
add: false,
border: false,
alignLeft: false,
boxShadow: '',
icon: {
remove: false,
color: '',
},
text: {
content: 'Explore',
color: '',
},
hover: {
backgroundColor: '',
color: '',
},
},
notifications: {
remove: false,
border: true,
tooltip: false,
boxShadow: '',
hoverBackgroundColor: '',
icon: {
symbol: 'bell', // Accepts 'inbox', 'bell', or ''
color: '',
hover: {
color: '',
},
},
text: {
content: 'Inbox',
color: '',
},
dot: {
remove: false,
boxShadowColor: '',
color: '',
displayOverIcon: true,
},
},
avatar: {
remove: false,
size: '',
dropdownIcon: false,
canCloseSidebar: true,
},
globalBar: {
boxShadowColor: '',
leftAligned: {
gap: '',
},
rightAligned: {
gap: '',
},
},
localBar: {
backgroundColor: '#02040A',
alignCenter: false,
boxShadow: {
consistentColor: true,
},
links: {
color: '',
},
},
sidebars: {
backdrop: {
color: 'transparent',
},
left: {
preload: true,
},
right: {
preload: true,
floatUnderneath: false,
width: '',
maxHeight: '',
},
},
repositoryHeader: {
import: true,
alignCenter: false,
removePageTitle: true,
backgroundColor: '#02040A',
avatar: {
remove: false,
customSvg: '',
},
link: {
color: '#6AAFF9',
hover: {
backgroundColor: 'transparent',
color: 'var(--color-accent-fg)',
textDecoration: 'underline',
},
},
},
},
},
oldSchool: {
light: {
backgroundColor: '#161C20',
hamburgerButton: {
remove: true,
},
logo: {
remove: false,
color: '#e6edf3',
customSvg: '',
},
pageTitle: {
remove: true,
color: oldSchoolColor,
hover: {
backgroundColor: oldSchoolHoverBackgroundColor,
color: oldSchoolHoverColor,
},
},
search: {
remove: false,
backgroundColor: '#494D54',
borderColor: '#30363d',
boxShadow: 'none',
alignLeft: true,
width: 'calc(var(--feed-sidebar) - 67px)',
margin: {
left: '',
right: '',
},
magnifyingGlassIcon: {
remove: true,
},
placeholder: {
text: 'Search or jump to...',
color: '#B3B3B5',
},
rightButton: 'slash key',
modal: {
width: '450px',
},
},
divider: {
remove: true,
},
flipCreateInbox: true,
create: {
remove: false,
border: false,
tooltip: false,
boxShadow: 'none',
hoverBackgroundColor: oldSchoolHoverBackgroundColor,
plusIcon: {
remove: false,
color: oldSchoolColor,
marginRight: '0px',
hover: {
color: oldSchoolHoverColor,
},
},
text: {
content: '',
color: '',
},
dropdownIcon: {
remove: false,
color: oldSchoolColor,
hover: {
color: oldSchoolHoverColor,
},
},
},
flipIssuesPullRequests: true,
issues: {
remove: false,
border: false,
tooltip: false,
alignLeft: true,
boxShadow: 'none',
icon: {
remove: true,
color: '',
},
text: {
content: 'Issues',
color: oldSchoolColor,
},
hover: {
backgroundColor: oldSchoolHoverBackgroundColor,
color: oldSchoolHoverColor,
},
},
pullRequests: {
remove: false,
border: false,
tooltip: false,
alignLeft: true,
boxShadow: 'none',
icon: {
remove: true,
color: '',
},
text: {
content: 'Pull requests',
color: oldSchoolColor,
},
hover: {
backgroundColor: oldSchoolHoverBackgroundColor,
color: oldSchoolHoverColor,
},
},
marketplace: {
add: true,
border: false,
tooltip: false,
alignLeft: true,
boxShadow: 'none',
icon: {
remove: true,
color: '',
},
text: {
content: 'Marketplace',
color: oldSchoolColor,
},
hover: {
backgroundColor: oldSchoolHoverBackgroundColor,
color: oldSchoolHoverColor,
},
},
explore: {
add: true,
border: false,
tooltip: false,
alignLeft: true,
boxShadow: 'none',
icon: {
remove: true,
color: '',
},
text: {
content: 'Explore',
color: oldSchoolColor,
},
hover: {
backgroundColor: oldSchoolHoverBackgroundColor,
color: oldSchoolHoverColor,
},
},
notifications: {
remove: false,
border: false,
tooltip: false,
boxShadow: 'none',
hoverBackgroundColor: oldSchoolHoverBackgroundColor,
icon: {
symbol: 'bell',
color: oldSchoolColor,
hover: {
color: oldSchoolHoverColor,
},
},
text: {
content: '',
color: '',
},
dot: {
remove: false,
boxShadowColor: '#161C20',
color: '#2f81f7',
displayOverIcon: true,
},
},
avatar: {
remove: false,
size: '24px',
dropdownIcon: true,
canCloseSidebar: true,
},
globalBar: {
boxShadowColor: '#21262D',
leftAligned: {
gap: '0.75rem',
},
rightAligned: {
gap: '2px',
},
},
localBar: {
backgroundColor: '#FAFBFD',
alignCenter: true,
boxShadow: {
consistentColor: true,
},
links: {
color: '',
},
},
sidebars: {
backdrop: {
color: oldSchoolHoverBackgroundColor,
},
left: {
preload: true,
},
right: {
preload: true,
floatUnderneath: true,
width: '',
maxHeight: '50vh',
},
},
repositoryHeader: {
import: true,
alignCenter: true,
removePageTitle: true,
backgroundColor: '#FAFBFD',
avatar: {
remove: false,
customSvg: '<svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-repo mr-1 color-fg-muted"><path d="M2 2.5A2.5 2.5 0 0 1 4.5 0h8.75a.75.75 0 0 1 .75.75v12.5a.75.75 0 0 1-.75.75h-2.5a.75.75 0 0 1 0-1.5h1.75v-2h-8a1 1 0 0 0-.714 1.7.75.75 0 1 1-1.072 1.05A2.495 2.495 0 0 1 2 11.5Zm10.5-1h-8a1 1 0 0 0-1 1v6.708A2.486 2.486 0 0 1 4.5 9h8ZM5 12.25a.25.25 0 0 1 .25-.25h3.5a.25.25 0 0 1 .25.25v3.25a.25.25 0 0 1-.4.2l-1.45-1.087a.249.249 0 0 0-.3 0L5.4 15.7a.25.25 0 0 1-.4-.2Z"></path></svg>',
},
link: {
color: '#2F81F7',
hover: {
backgroundColor: 'transparent',
color: '#0969da',
textDecoration: 'underline',
},
},
},
},
dark: {
backgroundColor: '#161C20',
hamburgerButton: {
remove: true,
},
logo: {
remove: false,
color: '#e6edf3',
customSvg: '',
},
pageTitle: {
remove: true,
color: oldSchoolColor,
hover: {
backgroundColor: oldSchoolHoverBackgroundColor,
color: oldSchoolHoverColor,
},
},
search: {
remove: false,
backgroundColor: '#0E1217',
borderColor: '#30363d',
boxShadow: 'none',
alignLeft: true,
width: 'calc(var(--feed-sidebar) - 67px)',
margin: {
left: '',
right: '',
},
magnifyingGlassIcon: {
remove: true,
},
placeholder: {
text: 'Search or jump to...',
color: '#B3B3B5',
},
rightButton: 'slash key',
modal: {
width: '450px',
},
},
divider: {
remove: true,
},
flipCreateInbox: true,
create: {
remove: false,
border: false,
tooltip: false,
boxShadow: 'none',
hoverBackgroundColor: oldSchoolHoverBackgroundColor,
plusIcon: {
remove: false,
color: oldSchoolColor,
marginRight: '0px',
hover: {
color: oldSchoolHoverColor,
},
},
text: {
content: '',
color: '',
},
dropdownIcon: {
remove: false,
color: oldSchoolColor,
hover: {
color: oldSchoolHoverColor,
},
},
},
flipIssuesPullRequests: true,
issues: {
remove: false,
border: false,
tooltip: false,
alignLeft: true,
boxShadow: 'none',
icon: {
remove: true,
color: '',
},
text: {
content: 'Issues',
color: oldSchoolColor,
},
hover: {
backgroundColor: oldSchoolHoverBackgroundColor,
color: oldSchoolHoverColor,
},
},
pullRequests: {
remove: false,
border: false,
tooltip: false,
alignLeft: true,
boxShadow: 'none',
icon: {
remove: true,
color: '',
},
text: {
content: 'Pull requests',
color: oldSchoolColor,
},
hover: {
backgroundColor: oldSchoolHoverBackgroundColor,
color: oldSchoolHoverColor,
},
},
marketplace: {
add: true,
border: false,
alignLeft: true,
boxShadow: 'none',
icon: {
remove: true,
color: '',
},
text: {
content: 'Marketplace',
color: oldSchoolColor,
},
hover: {
backgroundColor: oldSchoolHoverBackgroundColor,
color: oldSchoolHoverColor,
},
},
explore: {
add: true,
border: false,
alignLeft: true,
boxShadow: 'none',
icon: {
remove: true,
color: '',
},
text: {
content: 'Explore',
color: oldSchoolColor,
},
hover: {
backgroundColor: oldSchoolHoverBackgroundColor,
color: oldSchoolHoverColor,
},
},
notifications: {
remove: false,
border: false,
tooltip: false,
boxShadow: 'none',
hoverBackgroundColor: oldSchoolHoverBackgroundColor,
icon: {
symbol: 'bell',
color: oldSchoolColor,
hover: {
color: oldSchoolHoverColor,
},
},
text: {
content: '',
color: '',
},
dot: {
remove: false,
boxShadowColor: '#161C20',
color: '#2f81f7',
displayOverIcon: true,
},
},
avatar: {
remove: false,
size: '24px',
dropdownIcon: true,
canCloseSidebar: true,
},
globalBar: {
boxShadowColor: '#21262D',
leftAligned: {
gap: '0.75rem',
},
rightAligned: {
gap: '2px',
},
},
localBar: {
backgroundColor: '#0D1117',
alignCenter: true,
boxShadow: {
consistentColor: true,
},
links: {
color: '#e6edf3',
},
},
sidebars: {
backdrop: {
color: oldSchoolHoverBackgroundColor,
},
left: {
preload: true,
},
right: {
preload: true,
floatUnderneath: true,
width: '',
maxHeight: '50vh',
},
},
repositoryHeader: {
import: true,
alignCenter: true,
removePageTitle: true,
backgroundColor: '#0D1116',
avatar: {
remove: false,
customSvg: '<svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon octicon-repo mr-1 color-fg-muted"><path d="M2 2.5A2.5 2.5 0 0 1 4.5 0h8.75a.75.75 0 0 1 .75.75v12.5a.75.75 0 0 1-.75.75h-2.5a.75.75 0 0 1 0-1.5h1.75v-2h-8a1 1 0 0 0-.714 1.7.75.75 0 1 1-1.072 1.05A2.495 2.495 0 0 1 2 11.5Zm10.5-1h-8a1 1 0 0 0-1 1v6.708A2.486 2.486 0 0 1 4.5 9h8ZM5 12.25a.25.25 0 0 1 .25-.25h3.5a.25.25 0 0 1 .25.25v3.25a.25.25 0 0 1-.4.2l-1.45-1.087a.249.249 0 0 0-.3 0L5.4 15.7a.25.25 0 0 1-.4-.2Z"></path></svg>',
},
link: {
color: '#58A6FF',
hover: {
backgroundColor: 'transparent',
color: '#2F81F7',
textDecoration: 'underline',
},
},
},
},
},
};
// For testing purposes
// if (!checkConfigConsistency(configs)) return;
let OBSERVER = new MutationObserver(observeAndModify);
let GMC = new GM_config({
id: 'gmc-frame',
title: `
Custom Global Navigation
<small>
<a
href="https://github.com/blakegearin/github-custom-global-navigation"
target="_blank"
>
source
</a>
</small>
`,
events: {
init: gmcInitialized,
open: gmcOpened,
save: gmcSaved,
close: gmcClosed,
},
frame: gmcBuildFrame(),
fields: {
type: {
section: [
`
Configuration Type
<small>
<a
href="https://github.com/blakegearin/github-custom-global-navigation#configurations"
target="_blank"
>
learn more
</a>
</small>
`,
],
type: 'radio',
options: [
'Off',
'Happy Medium',
'Old School',
'Custom',
],
default: 'Old School',
},
light_backgroundColor: {
label: 'Background color',
section: [
'Custom Light',
],
type: 'text',
default: '',
},
light_hamburgerButton_remove: {
label: '<h3>Hamburger button</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
light_logo_remove: {
label: '<h3>Logo</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
light_logo_color: {
label: 'Color',
type: 'text',
default: '',
},
light_logo_customSvg: {
label: 'Custom SVG (URL or text)',
type: 'textarea',
default: '',
},
light_pageTitle_remove: {
label: '<h3>Page title</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
light_pageTitle_color: {
label: 'Color',
type: 'text',
default: '',
},
light_pageTitle_hover_backgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
light_pageTitle_hover_color: {
label: 'Hover color',
type: 'text',
default: '',
},
light_search_remove: {
label: '<h3>Search</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
light_search_backgroundColor: {
label: 'Background color',
type: 'text',
default: '',
},
light_search_borderColor: {
label: 'Border color',
type: 'text',
default: '',
},
light_search_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
light_search_alignLeft: {
label: 'Left aligned',
type: 'checkbox',
default: false,
},
light_search_width: {
label: 'Width',
type: 'text',
default: '',
},
light_search_margin_left: {
label: 'Margin left',
type: 'text',
default: '',
},
light_search_margin_right: {
label: 'Margin right',
type: 'text',
default: '',
},
light_search_magnifyingGlassIcon_remove: {
label: 'Magnifying glass icon remove',
type: 'checkbox',
default: false,
},
light_search_placeholder_text: {
label: 'Placeholder text',
type: 'text',
default: '',
},
light_search_placeholder_color: {
label: 'Placeholder color',
type: 'text',
default: '',
},
light_search_rightButton: {
label: 'Right button',
type: 'select',
options: [
'none',
'command palette',
'slash key',
],
default: 'command palette',
},
light_search_modal_width: {
label: 'Modal width',
type: 'text',
default: '',
},
light_divider_remove: {
label: '<h3>Divider</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
light_flipCreateInbox: {
label: 'Flip the order of Create and Notifications',
type: 'checkbox',
default: false,
},
light_create_remove: {
label: '<h3>Create button</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
light_create_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
light_create_tooltip: {
label: 'Tooltip',
type: 'checkbox',
default: true,
},
light_create_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
light_create_hoverBackgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
light_create_plusIcon_remove: {
label: 'Plus icon remove',
type: 'checkbox',
default: false,
},
light_create_plusIcon_color: {
label: 'Plus icon color',
type: 'text',
default: '',
},
light_create_plusIcon_marginRight: {
label: 'Plus icon margin right',
type: 'text',
default: '',
},
light_create_plusIcon_hover_color: {
label: 'Plus icon hover color',
type: 'text',
default: '',
},
light_create_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
light_create_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
light_create_dropdownIcon_remove: {
label: 'Dropdown icon remove',
type: 'checkbox',
default: false,
},
light_create_dropdownIcon_color: {
label: 'Dropdown icon color',
type: 'text',
default: '',
},
light_create_dropdownIcon_hover_color: {
label: 'Dropdown icon hover color',
type: 'text',
default: '',
},
light_flipIssuesPullRequests: {
label: 'Flip the order of Issues and Pull requests',
type: 'checkbox',
default: false,
},
light_issues_remove: {
label: '<h3>Issues button</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
light_issues_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
light_issues_tooltip: {
label: 'Tooltip',
type: 'checkbox',
default: true,
},
light_issues_alignLeft: {
label: 'Align left',
type: 'checkbox',
default: false,
},
light_issues_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
light_issues_icon_remove: {
label: 'Icon remove',
type: 'checkbox',
default: false,
},
light_issues_icon_color: {
label: 'Icon color',
type: 'text',
default: '',
},
light_issues_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
light_issues_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
light_issues_hover_backgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
light_issues_hover_color: {
label: 'Hover color',
type: 'text',
default: '',
},
light_pullRequests_remove: {
label: '<h3>Pull requests button</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
light_pullRequests_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
light_pullRequests_tooltip: {
label: 'Tooltip',
type: 'checkbox',
default: true,
},
light_pullRequests_alignLeft: {
label: 'Align left',
type: 'checkbox',
default: false,
},
light_pullRequests_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
light_pullRequests_icon_remove: {
label: 'Icon remove',
type: 'checkbox',
default: false,
},
light_pullRequests_icon_color: {
label: 'Icon color',
type: 'text',
default: '',
},
light_pullRequests_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
light_pullRequests_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
light_pullRequests_hover_backgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
light_pullRequests_hover_color: {
label: 'Hover color',
type: 'text',
default: '',
},
light_marketplace_add: {
label: '<h3>Marketplace</h3><div class="gmc-label">Add</div>',
type: 'checkbox',
default: false,
},
light_marketplace_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
light_marketplace_alignLeft: {
label: 'Align left',
type: 'checkbox',
default: false,
},
light_marketplace_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
light_marketplace_icon_remove: {
label: 'Icon remove',
type: 'checkbox',
default: false,
},
light_marketplace_icon_color: {
label: 'Icon color',
type: 'text',
default: '',
},
light_marketplace_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
light_marketplace_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
light_marketplace_hover_backgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
light_marketplace_hover_color: {
label: 'Hover color',
type: 'text',
default: '',
},
light_explore_add: {
label: '<h3>Explore</h3><div class="gmc-label">Add</div>',
type: 'checkbox',
default: false,
},
light_explore_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
light_explore_alignLeft: {
label: 'Align left',
type: 'checkbox',
default: false,
},
light_explore_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
light_explore_icon_remove: {
label: 'Icon remove',
type: 'checkbox',
default: false,
},
light_explore_icon_color: {
label: 'Icon color',
type: 'text',
default: '',
},
light_explore_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
light_explore_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
light_explore_hover_backgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
light_explore_hover_color: {
label: 'Hover color',
type: 'text',
default: '',
},
light_notifications_remove: {
label: '<h3>Notifications button</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
light_notifications_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
light_notifications_tooltip: {
label: 'Tooltip',
type: 'checkbox',
default: true,
},
light_notifications_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
light_notifications_hoverBackgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
light_notifications_icon_symbol: {
label: 'Icon symbol',
type: 'select',
options: [
'none',
'inbox',
'bell',
],
default: 'inbox',
},
light_notifications_icon_color: {
label: 'Icon color',
type: 'text',
default: '',
},
light_notifications_icon_hover_color: {
label: 'Icon hover color',
type: 'text',
default: '',
},
light_notifications_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
light_notifications_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
light_notifications_dot_remove: {
label: 'Dot remove',
type: 'checkbox',
default: false,
},
light_notifications_dot_boxShadowColor: {
label: 'Dot hover color',
type: 'text',
default: '',
},
light_notifications_dot_color: {
label: 'Dot color',
type: 'text',
default: '',
},
light_notifications_dot_displayOverIcon: {
label: 'Dot display over icon',
type: 'checkbox',
default: false,
},
light_avatar_remove: {
label: '<h3>Avatar</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
light_avatar_size: {
label: 'Size',
type: 'text',
default: '',
},
light_avatar_dropdownIcon: {
label: 'Dropdown icon',
type: 'checkbox',
default: false,
},
light_avatar_canCloseSidebar: {
label: 'Can close sidebar',
type: 'checkbox',
default: false,
},
light_globalBar_boxShadowColor: {
label: '<h3>Global bar</h3><div class="gmc-label">Box shadow color</div>',
type: 'text',
default: '',
},
light_globalBar_leftAligned_gap: {
label: 'Left aligned gap',
type: 'text',
default: '',
},
light_globalBar_rightAligned_gap: {
label: 'Right aligned gap',
type: 'text',
default: '',
},
light_localBar_backgroundColor: {
label: '<h3>Local bar</h3><div class="gmc-label">Background color</div>',
type: 'text',
default: '',
},
light_localBar_alignCenter: {
label: 'Align center',
type: 'checkbox',
default: false,
},
light_localBar_boxShadow_consistentColor: {
label: 'Box shadow consistent color',
type: 'checkbox',
default: false,
},
light_localBar_links_color: {
label: 'Links color',
type: 'text',
default: '',
},
light_sidebars_backdrop_color: {
label: '<h3>Sidebars</h3><div class="gmc-label">Backdrop color</div>',
type: 'text',
default: '',
},
light_sidebars_left_preload: {
label: 'Left preload',
type: 'checkbox',
default: false,
},
light_sidebars_right_preload: {
label: 'Right preload',
type: 'checkbox',
default: false,
},
light_sidebars_right_floatUnderneath: {
label: 'Right float underneath',
type: 'checkbox',
default: false,
},
light_sidebars_right_width: {
label: 'Right width',
type: 'text',
default: '',
},
light_sidebars_right_maxHeight: {
label: 'Right max height',
type: 'text',
default: '',
},
light_repositoryHeader_import: {
label: '<h3>Repository header</h3><div class="gmc-label">Import</div>',
type: 'checkbox',
default: false,
},
light_repositoryHeader_alignCenter: {
label: 'Align center',
type: 'checkbox',
default: false,
},
light_repositoryHeader_removePageTitle: {
label: 'Remove page title',
type: 'checkbox',
default: false,
},
light_repositoryHeader_backgroundColor: {
label: 'Background color',
type: 'text',
default: '',
},
light_repositoryHeader_avatar_remove: {
label: 'Avatar remove',
type: 'checkbox',
default: false,
},
light_repositoryHeader_avatar_customSvg: {
label: 'Custom SVG (URL or text)',
type: 'textarea',
default: '',
},
light_repositoryHeader_link_color: {
label: 'Link color',
type: 'text',
default: '',
},
light_repositoryHeader_link_hover_backgroundColor: {
label: 'Link hover background color',
type: 'text',
default: '',
},
light_repositoryHeader_link_hover_color: {
label: 'Link hover color',
type: 'text',
default: '',
},
light_repositoryHeader_link_hover_textDecoration: {
label: 'Link hover text decoration',
type: 'text',
default: '',
},
dark_backgroundColor: {
label: 'Background color',
section: [
'Custom Dark',
],
type: 'text',
default: '',
},
dark_hamburgerButton_remove: {
label: '<h3>Hamburger button</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
dark_logo_remove: {
label: '<h3>Logo</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
dark_logo_color: {
label: 'Color',
type: 'text',
default: '',
},
dark_logo_customSvg: {
label: 'Custom SVG (URL or text)',
type: 'textarea',
default: '',
},
dark_pageTitle_remove: {
label: '<h3>Page title</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
dark_pageTitle_color: {
label: 'Color',
type: 'text',
default: '',
},
dark_pageTitle_hover_backgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
dark_pageTitle_hover_color: {
label: 'Hover color',
type: 'text',
default: '',
},
dark_search_remove: {
label: '<h3>Search</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
dark_search_backgroundColor: {
label: 'Background color',
type: 'text',
default: '',
},
dark_search_borderColor: {
label: 'Border color',
type: 'text',
default: '',
},
dark_search_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
dark_search_alignLeft: {
label: 'Left aligned',
type: 'checkbox',
default: false,
},
dark_search_width: {
label: 'Width',
type: 'text',
default: '',
},
dark_search_margin_left: {
label: 'Margin left',
type: 'text',
default: '',
},
dark_search_margin_right: {
label: 'Margin right',
type: 'text',
default: '',
},
dark_search_magnifyingGlassIcon_remove: {
label: 'Magnifying glass icon remove',
type: 'checkbox',
default: false,
},
dark_search_placeholder_text: {
label: 'Placeholder text',
type: 'text',
default: '',
},
dark_search_placeholder_color: {
label: 'Placeholder color',
type: 'text',
default: '',
},
dark_search_rightButton: {
label: 'Right button',
type: 'select',
options: [
'none',
'command palette',
'slash key',
],
default: 'command palette',
},
dark_search_modal_width: {
label: 'Modal width',
type: 'text',
default: '',
},
dark_divider_remove: {
label: '<h3>Divider</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
dark_flipCreateInbox: {
label: 'Flip the order of Create and Notifications',
type: 'checkbox',
default: false,
},
dark_create_remove: {
label: '<h3>Create button</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
dark_create_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
dark_create_tooltip: {
label: 'Tooltip',
type: 'checkbox',
default: true,
},
dark_create_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
dark_create_hoverBackgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
dark_create_plusIcon_remove: {
label: 'Plus icon remove',
type: 'checkbox',
default: false,
},
dark_create_plusIcon_color: {
label: 'Plus icon color',
type: 'text',
default: '',
},
dark_create_plusIcon_marginRight: {
label: 'Plus icon margin right',
type: 'text',
default: '',
},
dark_create_plusIcon_hover_color: {
label: 'Plus icon hover color',
type: 'text',
default: '',
},
dark_create_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
dark_create_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
dark_create_dropdownIcon_remove: {
label: 'Dropdown icon remove',
type: 'checkbox',
default: false,
},
dark_create_dropdownIcon_color: {
label: 'Dropdown icon color',
type: 'text',
default: '',
},
dark_create_dropdownIcon_hover_color: {
label: 'Dropdown icon hover color',
type: 'text',
default: '',
},
dark_flipIssuesPullRequests: {
label: 'Flip the order of Issues and Pull requests',
type: 'checkbox',
default: false,
},
dark_issues_remove: {
label: '<h3>Issues button</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
dark_issues_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
dark_issues_tooltip: {
label: 'Tooltip',
type: 'checkbox',
default: true,
},
dark_issues_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
dark_issues_alignLeft: {
label: 'Align left',
type: 'checkbox',
default: false,
},
dark_issues_icon_remove: {
label: 'Icon remove',
type: 'checkbox',
default: false,
},
dark_issues_icon_color: {
label: 'Icon color',
type: 'text',
default: '',
},
dark_issues_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
dark_issues_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
dark_issues_hover_backgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
dark_issues_hover_color: {
label: 'Hover color',
type: 'text',
default: '',
},
dark_pullRequests_remove: {
label: '<h3>Pull requests button</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
dark_pullRequests_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
dark_pullRequests_tooltip: {
label: 'Tooltip',
type: 'checkbox',
default: true,
},
dark_pullRequests_alignLeft: {
label: 'Align left',
type: 'checkbox',
default: false,
},
dark_pullRequests_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
dark_pullRequests_icon_remove: {
label: 'Icon remove',
type: 'checkbox',
default: false,
},
dark_pullRequests_icon_color: {
label: 'Icon color',
type: 'text',
default: '',
},
dark_pullRequests_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
dark_pullRequests_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
dark_pullRequests_hover_backgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
dark_pullRequests_hover_color: {
label: 'Hover color',
type: 'text',
default: '',
},
dark_marketplace_add: {
label: '<h3>Marketplace</h3><div class="gmc-label">Add</div>',
type: 'checkbox',
default: false,
},
dark_marketplace_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
dark_marketplace_alignLeft: {
label: 'Align left',
type: 'checkbox',
default: false,
},
dark_marketplace_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
dark_marketplace_icon_remove: {
label: 'Icon remove',
type: 'checkbox',
default: false,
},
dark_marketplace_icon_color: {
label: 'Icon color',
type: 'text',
default: '',
},
dark_marketplace_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
dark_marketplace_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
dark_marketplace_hover_backgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
dark_marketplace_hover_color: {
label: 'Hover color',
type: 'text',
default: '',
},
dark_explore_add: {
label: '<h3>Explore</h3><div class="gmc-label">Add</div>',
type: 'checkbox',
default: false,
},
dark_explore_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
dark_explore_alignLeft: {
label: 'Align left',
type: 'checkbox',
default: false,
},
dark_explore_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
dark_explore_icon_remove: {
label: 'Icon remove',
type: 'checkbox',
default: false,
},
dark_explore_icon_color: {
label: 'Icon color',
type: 'text',
default: '',
},
dark_explore_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
dark_explore_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
dark_explore_hover_backgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
dark_explore_hover_color: {
label: 'Hover color',
type: 'text',
default: '',
},
dark_notifications_remove: {
label: '<h3>Notifications button</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
dark_notifications_border: {
label: 'Border',
type: 'checkbox',
default: true,
},
dark_notifications_tooltip: {
label: 'Tooltip',
type: 'checkbox',
default: true,
},
dark_notifications_boxShadow: {
label: 'Box shadow',
type: 'text',
default: '',
},
dark_notifications_hoverBackgroundColor: {
label: 'Hover background color',
type: 'text',
default: '',
},
dark_notifications_icon_symbol: {
label: 'Icon symbol',
type: 'select',
options: [
'none',
'inbox',
'bell',
],
default: 'inbox',
},
dark_notifications_icon_color: {
label: 'Icon color',
type: 'text',
default: '',
},
dark_notifications_icon_hover_color: {
label: 'Icon hover color',
type: 'text',
default: '',
},
dark_notifications_text_content: {
label: 'Text content',
type: 'text',
default: '',
},
dark_notifications_text_color: {
label: 'Text color',
type: 'text',
default: '',
},
dark_notifications_dot_remove: {
label: 'Dot remove',
type: 'checkbox',
default: false,
},
dark_notifications_dot_boxShadowColor: {
label: 'Dot hover color',
type: 'text',
default: '',
},
dark_notifications_dot_color: {
label: 'Dot color',
type: 'text',
default: '',
},
dark_notifications_dot_displayOverIcon: {
label: 'Dot display over icon',
type: 'checkbox',
default: false,
},
dark_avatar_remove: {
label: '<h3>Avatar</h3><div class="gmc-label">Remove</div>',
type: 'checkbox',
default: false,
},
dark_avatar_size: {
label: 'Size',
type: 'text',
default: '',
},
dark_avatar_dropdownIcon: {
label: 'Dropdown icon',
type: 'checkbox',
default: false,
},
dark_avatar_canCloseSidebar: {
label: 'Can close sidebar',
type: 'checkbox',
default: false,
},
dark_globalBar_boxShadowColor: {
label: '<h3>Global bar</h3><div class="gmc-label">Box shadow color</div>',
type: 'text',
default: '',
},
dark_globalBar_leftAligned_gap: {
label: 'Left aligned gap',
type: 'text',
default: '',
},
dark_globalBar_rightAligned_gap: {
label: 'Right aligned gap',
type: 'text',
default: '',
},
dark_localBar_backgroundColor: {
label: '<h3>Local bar</h3><div class="gmc-label">Background color</div>',
type: 'text',
default: '',
},
dark_localBar_alignCenter: {
label: 'Align center',
type: 'checkbox',
default: false,
},
dark_localBar_boxShadow_consistentColor: {
label: 'Box shadow consistent color',
type: 'checkbox',
default: false,
},
dark_localBar_links_color: {
label: 'Links color',
type: 'text',
default: '',
},
dark_sidebars_backdrop_color: {
label: '<h3>Sidebars</h3><div class="gmc-label">Backdrop color</div>',
type: 'text',
default: '',
},
dark_sidebars_left_preload: {
label: 'Left preload',
type: 'checkbox',
default: false,
},
dark_sidebars_right_preload: {
label: 'Right preload',
type: 'checkbox',
default: false,
},
dark_sidebars_right_floatUnderneath: {
label: 'Right float underneath',
type: 'checkbox',
default: false,
},
dark_sidebars_right_width: {
label: 'Right width',
type: 'text',
default: '',
},
dark_sidebars_right_maxHeight: {
label: 'Right max height',
type: 'text',
default: '',
},
dark_repositoryHeader_import: {
label: '<h3>Repository header</h3><div class="gmc-label">Import</div>',
type: 'checkbox',
default: false,
},
dark_repositoryHeader_alignCenter: {
label: 'Align enter',
type: 'checkbox',
default: false,
},
dark_repositoryHeader_removePageTitle: {
label: 'Remove page title',
type: 'checkbox',
default: false,
},
dark_repositoryHeader_backgroundColor: {
label: 'Background color',
type: 'text',
default: '',
},
dark_repositoryHeader_avatar_remove: {
label: 'Avatar remove',
type: 'checkbox',
default: false,
},
dark_repositoryHeader_avatar_customSvg: {
label: 'Custom SVG (URL or text)',
type: 'textarea',
default: '',
},
dark_repositoryHeader_link_color: {
label: 'Link color',
type: 'text',
default: '',
},
dark_repositoryHeader_link_hover_backgroundColor: {
label: 'Link hover background color',
type: 'text',
default: '',
},
dark_repositoryHeader_link_hover_color: {
label: 'Link hover color',
type: 'text',
default: '',
},
dark_repositoryHeader_link_hover_textDecoration: {
label: 'Link hover text decoration',
type: 'text',
default: '',
},
on_save: {
label: 'On save',
section: ['Settings'],
type: 'select',
options: [
'do nothing',
'refresh tab',
'refresh tab and close',
'run script',
'run script and close',
],
default: 'do nothing',
},
on_close: {
label: 'On close',
type: 'select',
options: [
'do nothing',
'refresh tab',
'run script',
],
default: 'do nothing',
},
on_open: {
label: 'On open',
type: 'select',
options: [
'do nothing',
'close sidebar',
],
default: 'close sidebar',
},
menu_item_title: {
label: 'Menu item title',
type: 'text',
default: 'Custom global navigation',
},
menu_item_icon: {
label: 'Menu item icon',
type: 'select',
options: [
'logo',
'compass',
'cog',
],
default: 'logo',
},
log_level: {
label: 'Log level',
type: 'select',
options: [
'silent',
'quiet',
'debug',
'verbose',
'trace',
],
default: 'quiet',
},
clear_custom_config: {
label: 'Clear Custom',
section: ['Danger Zone'],
type: 'button',
click: gmcClearCustom,
},
apply_happyMedium_config: {
label: 'Overwrite Custom with Happy Medium',
type: 'button',
click: gmcApplyCustomHappyMediumConfig,
},
apply_oldSchool_config: {
label: 'Overwrite Custom with Old School',
type: 'button',
click: gmcApplyCustomOldSchoolConfig,
},
},
});
})();