// ==UserScript== // @name GitHub Label Color Picker // @version 1.0.7 // @description A userscript that adds a color picker to the label color input // @license MIT // @author Rob Garrison // @contributor darkred // @namespace https://github.com/Mottie // @include https://github.com/* // @run-at document-idle // @grant GM_addStyle // @grant GM_getValue // @grant GM_setValue // @grant GM_registerMenuCommand // @require https://greasyfork.org/scripts/23181-colorpicker/code/colorPicker.js?version=147862 // @icon https://github.githubassets.com/pinned-octocat.svg // @supportURL https://github.com/Mottie/GitHub-userscripts/issues // @downloadURL none // ==/UserScript== /* global jsColorPicker */ (() => { "use strict"; // GitHub-Dark changes "text-black" to #c0c0c0 GM_addStyle(` div.cp-app { margin:0; z-index:10; } .js-new-label-color-icon { pointer-events:none; } .js-new-label-color-icon.text-black { color:#000 !important; } `); function addPicker() { if ($(".js-new-label-color")) { jsColorPicker(".js-new-label-color-input", { customBG: "#222", noAlpha: true, renderCallback: function(colors) { let input = this && this.input; if (input) { updateSwatch(input, colors); } } }); } } function updateSwatch(input, colors) { let background = "#" + colors.HEX; input.value = background; let textColor = calcContrast(colors.HEX); // Update color swatch next to input let swatch = $(".js-new-label-color", input.closest("dd")); updateIcon(swatch, textColor); updateColors(swatch, background, textColor); // Update label preview swatch = $( ".js-label-preview .IssueLabel--big", input.closest(".table-list-item") ); updateColors(swatch, background, textColor); } function updateIcon(swatch, textColor) { let icon = $(".octicon", swatch); // !important set on these GitHub primer color definitions icon.classList.remove("text-white", "text-black"); icon.classList.add("text-" + textColor); } function updateColors(el, background, color) { el.style.backgroundColor = background; el.style.color = color; } /* replace colorPicker storage */ window.ColorPicker.docCookies = (key, val) => { if (typeof val === "undefined") { return GM_getValue(key); } GM_setValue(key, val); }; /* colorPickerMemosNoAlpha *MUST* follow this format "'rgba(83,25,231,1)','rgba(86,66,66,1)','rgba(22,20,223,1)'" */ function convertColorsToRgba(values) { let result = []; // see http://stackoverflow.com/a/26196012/145346 values .replace(/['"]/g, "") .split(/\s*,(?![^()]*(?:\([^()]*\))?\))\s*/g) .forEach(val => { let rgb = hexToRgb(val); if (rgb) { result.push(`'rgba(${rgb.r},${rgb.g},${rgb.b},1)'`); } else if (rgb === null && val.indexOf("rgba(") > -1) { // allow adding rgba() definitions result.push(`'${val}'`); } }); return result.join(","); } // Modified code from http://stackoverflow.com/a/5624139/145346 function hexToRgb(hex) { let result, // Expand shorthand form (e.g. "03F") to full form (e.g. "0033FF") shorthandRegex = /^#?([a-f\d])([a-f\d])([a-f\d])$/i; hex = hex.replace(shorthandRegex, (m, r, g, b) => { return r + r + g + g + b + b; }); result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex); return result ? { r: parseInt(result[1], 16), g: parseInt(result[2], 16), b: parseInt(result[3], 16) } : null; } // Calculate contrasting text color for the given background color // https://24ways.org/2010/calculating-color-contrast/ function calcContrast(hex) { const r = parseInt(hex.substr(0, 2), 16), g = parseInt(hex.substr(2, 2), 16), b = parseInt(hex.substr(4, 2), 16), yiq = ((r * 299) + (g * 587) + (b * 114)) / 1000; return yiq >= 128 ? "black" : "white"; } function $(selector, el) { return (el || document).querySelector(selector); } // Add GM options GM_registerMenuCommand( "Set label ColorPicker swatches (8 HEX or RGBA Max)", () => { const colors = GM_getValue("colorPickerMemosNoAlpha", "#000000,#ffffff"), val = prompt("Set label default colors (8 max):", colors); if (val !== null && typeof val === "string") { GM_setValue("colorPickerMemosNoAlpha", convertColorsToRgba(val)); } } ); document.body.addEventListener("click", event => { // initialize if "Edit" or "New label" button clicked // because "Save changes" updates the entire item if ( event.target && event.target.matches(".js-edit-label, .js-details-target") ) { addPicker(); } }); addPicker(); })();