- // ==UserScript==
- // @name GitHub Dark Script
- // @version 2.0.15
- // @description GitHub Dark in userscript form, with a settings panel
- // @namespace https://github.com/StylishThemes
- // @include /^https?://((gist|guides|help|raw|status|developer)\.)?github\.com((?!generated_pages\/preview).)*$/
- // @include https://*.githubusercontent.com/*
- // @grant GM_addStyle
- // @grant GM_getValue
- // @grant GM_setValue
- // @grant GM_info
- // @grant GM_xmlhttpRequest
- // @grant GM_registerMenuCommand
- // @connect githubusercontent.com
- // @connect raw.githubusercontent.com
- // @run-at document-start
- // @require https://greatest.deepsurf.us/scripts/15563-jscolor/code/jscolor.js?version=106439
- // ==/UserScript==
- /* global GM_addStyle, GM_getValue, GM_setValue, GM_info, GM_xmlhttpRequest, GM_registerMenuCommand, jscolor */
- /* jshint esnext:true */
- (() => {
- 'use strict';
-
- const version = GM_info.script.version,
-
- // delay until package.json allowed to load
- delay = 8.64e7, // 24 hours in milliseconds
-
- // Keyboard shortcut to open ghd panel (only a two key combo coded)
- keyboardOpen = 'g+0',
- keyboardToggle = 'g+-',
- // keyboard shortcut delay from first to second letter
- keyboardDelay = 1000,
-
- // base urls to fetch style and package.json
- root = 'https://raw.githubusercontent.com/StylishThemes/GitHub-Dark/master/',
-
- defaults = {
- attach : 'scroll',
- color : '#4183C4',
- enable : true,
- font : 'Menlo',
- image : 'url("")',
- tab : 4,
- theme : 'Twilight',
- type : 'tiled',
- wrap : false,
-
- // toggle buttons
- enableCodeWrap : true,
- enableMonospace : true,
- // diff toggle + accordion mode
- modeDiffToggle : '1',
-
- // internal variables
- date : 0,
- version : 0,
- rawCss : '',
- themeCss : '',
- processedCss : '',
- last : {}
- },
-
- // extract style & theme name
- regex = /\/\*! [^\*]+ \*\//,
- // "themes/" prefix not included here
- themes = {
- 'Ambiance' : 'ambiance.min.css',
- 'Chaos' : 'chaos.min.css',
- 'Clouds Midnight' : 'clouds-midnight.min.css',
- 'Cobalt' : 'cobalt.min.css',
- 'GitHub Dark' : 'github-dark.min.css',
- 'Idle Fingers' : 'idle-fingers.min.css',
- 'Kr Theme' : 'kr-theme.min.css',
- 'Merbivore' : 'merbivore.min.css',
- 'Merbivore Soft' : 'merbivore-soft.min.css',
- 'Mono Industrial' : 'mono-industrial.min.css',
- 'Mono Industrial Clear' : 'mono-industrial-clear.min.css',
- 'Monokai' : 'monokai.min.css',
- 'Monokai Spacegray Eighties' : 'monokai-spacegray-eighties.min.css',
- 'Obsidian' : 'obsidian.min.css',
- 'Pastel on Dark' : 'pastel-on-dark.min.css',
- 'Solarized Dark' : 'solarized-dark.min.css',
- 'Terminal' : 'terminal.min.css',
- 'Tomorrow Night' : 'tomorrow-night.min.css',
- 'Tomorrow Night Blue' : 'tomorrow-night-blue.min.css',
- 'Tomorrow Night Bright' : 'tomorrow-night-bright.min.css',
- 'Tomorrow Night Eigthies' : 'tomorrow-night-eighties.min.css',
- 'Twilight' : 'twilight.min.css',
- 'Vibrant Ink' : 'vibrant-ink.min.css'
- },
-
- type = {
- tiled : `
- background-repeat: repeat !important;
- background-size: auto !important;
- background-position: left top !important;
- `,
- fit : `
- background-repeat: no-repeat !important;
- background-size: cover !important;
- background-position: center top !important;
- `
- },
-
- wrapCss = {
- wrapped : `
- white-space: pre-wrap !important;
- word-break: break-all !important;
- overflow-wrap: break-word !important;
- display: block !important;
- `,
- unwrap : `
- white-space: pre !important;
- word-break: normal !important;
- display: block !important;
- `
- },
-
- // https://github.com/StylishThemes/GitHub-code-wrap/blob/master/github-code-wrap.css
- wrapCodeCss = `
- /* GitHub: Enable wrapping of long code lines */
- .blob-code-inner,
- .markdown-body pre > code,
- .markdown-body .highlight > pre { ${wrapCss.wrapped} }
- td.blob-code-inner {
- display: table-cell !important;
- }
- `,
-
- wrapIcon = `
- <svg xmlns="http://www.w3.org/2000/svg" width="768" height="768" viewBox="0 0 768 768">
- <path d="M544.5 352.5q52.5 0 90 37.5t37.5 90-37.5 90-90 37.5H480V672l-96-96 96-96v64.5h72q25.5 0 45-19.5t19.5-45-19.5-45-45-19.5H127.5v-63h417zm96-192v63h-513v-63h513zm-513 447v-63h192v63h-192z"/>
- </svg>
- `,
-
- monospaceIcon = `
- <svg class="octicon" xmlns="http://www.w3.org/2000/svg" width="18" height="18" viewBox="0 0 32 32">
- <path d="M5.91 7.31v8.41c0 .66.05 1.09.14 1.31.09.21.23.37.41.48.18.11.52.16 1.02.16v.41H2.41v-.41c.5 0 .86-.05 1.03-.14.16-.11.3-.27.41-.5.11-.23.16-.66.16-1.3V11.7c0-1.14-.04-1.87-.11-2.2-.04-.26-.13-.42-.24-.53-.11-.1-.27-.14-.46-.14-.21 0-.48.05-.77.18l-.18-.41 3.14-1.28h.52v-.01zm-.95-5.46c.32 0 .59.11.82.34.23.23.34.5.34.82 0 .32-.11.59-.34.82-.23.22-.51.34-.82.34-.32 0-.59-.11-.82-.34s-.36-.5-.36-.82c0-.32.11-.59.34-.82.24-.23.51-.34.84-.34zm19.636 19.006h-3.39v-1.64h5.39v9.8h3.43v1.66h-9.18v-1.66h3.77v-8.16h-.02zm.7-6.44c.21 0 .43.04.63.13.18.09.36.2.5.34s.25.3.34.5c.07.18.13.39.13.61 0 .22-.04.41-.13.61s-.19.36-.34.5-.3.25-.5.32c-.2.09-.39.13-.62.13-.21 0-.43-.04-.61-.12-.19-.07-.35-.19-.5-.34-.14-.14-.25-.3-.34-.5-.07-.2-.13-.39-.13-.61s.04-.43.13-.61c.07-.18.2-.36.34-.5s.31-.25.5-.34c.17-.09.39-.12.6-.12zM2 30L27.82 2H30L4.14 30H2z"/>
- </svg>
- `,
-
- fileIcon = `
- <svg class="octicon" xmlns="http://www.w3.org/2000/svg" width="10" height="6.5" viewBox="0 0 10 6.5">
- <path d="M0 1.497L1.504 0l3.49 3.76L8.505.016 10 1.52 4.988 6.51 0 1.496z"/>
- </svg>
- `,
-
- $style = make({
- el: 'style',
- cl4ss: 'ghd-style'
- });
-
- let timer, picker, // jscolor picker
- isInitialized = 'pending',
- // prevent mutationObserver from going nuts
- isUpdating = false,
- // set when css code to test is pasted into the settings panel
- testing = false,
- //
- debug = GM_getValue('debug', false),
- data = {};
-
- function updatePanel() {
- if (!isInitialized || !$('#ghd-settings-inner')) { return; }
- // prevent multiple change events from processing
- isUpdating = true;
-
- let temp, el,
- body = $('body'),
- panel = $('#ghd-settings-inner');
-
- $('.ghd-attach', panel).value = data.attach || defaults.attach;
- $('.ghd-font', panel).value = data.font || defaults.font;
- $('.ghd-image', panel).value = data.image || defaults.image;
- $('.ghd-tab', panel).value = data.tab || defaults.tab;
- $('.ghd-theme', panel).value = data.theme || defaults.theme;
- $('.ghd-type', panel).value = data.type || defaults.type;
-
- $('.ghd-enable', panel).checked = isBool('enable');
- $('.ghd-wrap', panel).checked = isBool('wrap');
-
- $('.ghd-codewrap-checkbox', panel).checked = isBool('enableCodeWrap');
- $('.ghd-monospace-checkbox', panel).checked = isBool('enableMonospace');
-
- el = $('.ghd-diff-select', panel);
- temp = '' + (data.modeDiffToggle || defaults.modeDiffToggle);
- el.value = temp;
- toggleClass(el, 'enabled', temp !== '0');
-
- // update version tooltip
- $('.ghd-versions', panel).setAttribute('aria-label', getVersionTooltip());
-
- temp = data.color || defaults.color;
- $('.ghd-color').value = temp;
- // update swatch color & color picker value
- $('#ghd-swatch').style.backgroundColor = temp;
-
- if (picker) {
- picker.fromString(temp);
- }
- $style.disabled = !data.enable;
-
- toggleClass(body, 'ghd-disabled', !data.enable);
- toggleClass(body, 'nowrap', !data.wrap);
-
- if (data.enableCodeWrap !== data.lastCW ||
- data.enableMonospace !== data.lastMS ||
- data.modeDiffToggle !== data.lastDT) {
-
- data.lastCW = data.enableCodeWrap;
- data.lastMS = data.enableMonospace;
- data.lastDT = data.modeDiffToggle;
- updateToggles();
- }
-
- isUpdating = false;
- }
-
- function getStoredValues(init) {
- data = GM_getValue('data', defaults);
- try {
- data = JSON.parse(data);
- if (!Object.keys(data).length || ({}).toString.call(data) !== "[object Object]") {
- throw new Error();
- }
- } catch(err) { // compat
- data = GM_getValue('data', defaults);
- }
- if (debug) {
- if (init) {
- console.log('GitHub-Dark Script initializing!');
- }
- console.log('Retrieved stored values', data);
- }
- }
-
- function setStoredValues(reset) {
- data.processedCss = $style.textContent;
- GM_setValue('data', JSON.stringify(reset ? defaults : data));
- updatePanel();
- if (debug) {
- console.log((reset ? 'Resetting' : 'Saving') + ' current values', data);
- }
- }
-
- // modified from http://stackoverflow.com/a/5624139/145346
- function hexToRgb(hex) {
- let result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
- return result ? [
- parseInt(result[1], 16),
- parseInt(result[2], 16),
- parseInt(result[3], 16)
- ].join(", ") : "";
- }
-
- // convert version "1.2.3" into "001002003" for easier comparison
- function convertVersion(val) {
- let index,
- parts = val ? val.split('.') : '',
- str = '',
- len = parts.length;
- for (index = 0; index < len; index++) {
- str += ('000' + parts[index]).slice(-3);
- }
- if (debug) {
- console.log(`Converted version "${val}" to "${str}" for easy comparison`);
- }
- return val ? str : val;
- }
-
- function checkVersion() {
- if (debug) {
- console.log('Fetching package.json');
- }
- GM_xmlhttpRequest({
- method : 'GET',
- url : root + 'package.json',
- onload : response => {
- let pkg = JSON.parse(response.responseText);
-
- // save last loaded date, so package.json is only loaded once a day
- data.date = new Date().getTime();
-
- let ver = convertVersion(pkg.version);
- // if new available, load it & parse
- if (ver > data.version) {
- if (data.version !== 0 && debug) {
- console.log('Updating from ${data.version} to ${ver}');
- }
- data.version = ver;
- fetchAndApplyStyle();
- } else {
- addSavedStyle();
- }
- // save new date/version
- GM_setValue('data', JSON.stringify(data));
- }
- });
- }
-
- function fetchAndApplyStyle() {
- if (debug) {
- console.log(`Fetching ${root}github-dark.css`);
- }
- GM_xmlhttpRequest({
- method : 'GET',
- url : root + 'github-dark.css',
- onload : response => {
- data.rawCss = response.responseText;
- processStyle();
- }
- });
- }
-
- // load syntax highlighting theme
- function fetchAndApplyTheme() {
- if (!data.enable) {
- if (debug) {
- console.log('Disabled: stop theme processing');
- }
- return;
- }
- if (data.lastTheme === data.theme && data.themeCss !== '') {
- return applyTheme();
- }
- let name = data.theme || 'Twilight',
- themeUrl = root + 'themes/' + themes[name];
- if (debug) {
- console.log(`Fetching ${name} theme`, themeUrl);
- }
- GM_xmlhttpRequest({
- method : 'GET',
- url : themeUrl,
- onload : response => {
- let theme = response.responseText;
- if (response.status === 200 && theme) {
- data.themeCss = theme;
- data.lastTheme = name;
- applyTheme();
- } else {
- throw Error(`Failed to load theme file: "${theme}"`);
- }
- }
- });
- }
-
- function applyTheme() {
- if (debug) {
- console.log('Adding syntax theme "' + (data.themeCss || '').match(regex) + '" to css');
- }
- let css = data.processedCss || '';
- css = css.replace('/*[[syntax-theme]]*/', data.themeCss || '');
- applyStyle(css);
- setStoredValues();
- isUpdating = false;
- }
-
- function processStyle() {
- let url = /^url/.test(data.image || '') ? data.image :
- (data.image === 'none' ? 'none' : 'url("' + data.image + '")');
- if (!data.enable) {
- if (debug) {
- console.log('Disabled: stop processing');
- }
- return;
- }
- if (debug) {
- console.log('Processing set styles');
- }
-
- let processed = (data.rawCss || '')
- // remove moz-document wrapper
- .replace(/@-moz-document regexp\((.*)\) \{(\n|\r)+/, '')
- // replace background image; if no 'url' at start, then use 'none'
- .replace(/\/\*\[\[bg-choice\]\]\*\/ url\(.*\)/, url)
- // Add tiled or fit window size css
- .replace('/*[[bg-options]]*/', type[data.type || 'tiled'])
- // set scroll or fixed background image
- .replace('/*[[bg-attachment]]*/ fixed', data.attach || 'scroll')
- // replace base-color
- .replace(/\/\*\[\[base-color\]\]\*\/ #\w{3,6}/g, data.color || '#4183C4')
- // replace base-color-rgb
- .replace(/\/\*\[\[base-color-rgb\]\]\*\//g, hexToRgb(data.color || '#4183c4'))
- // add font choice
- .replace('/*[[font-choice]]*/', data.font || 'Menlo')
- // add tab size
- .replace(/\/\*\[\[tab-size\]\]\*\/ \d+/g, data.tab || 4)
- // code wrap css
- .replace('/*[[code-wrap]]*/', data.wrap ? wrapCodeCss : '')
- // remove default syntax
- .replace(/\s+\/\* grunt build - remove to end of file(.*(\n|\r))+\}$/m, '');
-
- // see https://github.com/StylishThemes/GitHub-Dark/issues/275
- if (/firefox/i.test(navigator.userAgent)) {
- processed = processed
- // line in github-dark.css = "select, input:not(.btn-link), textarea"
- .replace('select, input:not(.btn-link)', 'input { color:#eee !important; } select')
- .replace(/input\[type=\"checkbox\"\][\s\S]+?}/gm, '');
- }
- data.processedCss = processed;
- fetchAndApplyTheme();
- }
-
- function applyStyle(css) {
- if (debug) {
- console.log('Applying style', '"' + (css || '').match(regex) + '"');
- }
- $style.textContent = css || '';
- }
-
- function addSavedStyle() {
- if (debug) {
- console.log('Adding previously saved style');
- }
- // apply already processed css to prevent FOUC
- $style.textContent = data.processedCss;
- }
-
- function updateStyle() {
- isUpdating = true;
-
- if (debug) {
- console.log('Updating user settings');
- }
-
- let body = $('body'),
- panel = $('#ghd-settings-inner');
-
- data.attach = $('.ghd-attach', panel).value;
- // get hex value directly
- data.color = picker.toHEXString();
- data.enable = $('.ghd-enable', panel).checked;
- data.font = $('.ghd-font', panel).value;
- data.image = $('.ghd-image', panel).value;
- data.tab = $('.ghd-tab', panel).value;
- data.theme = $('.ghd-theme', panel).value;
- data.type = $('.ghd-type', panel).value;
- data.wrap = $('.ghd-wrap', panel).checked;
-
- data.enableCodeWrap = $('.ghd-codewrap-checkbox', panel).checked;
- data.enableMonospace = $('.ghd-monospace-checkbox', panel).checked;
-
- data.modeDiffToggle = $('.ghd-diff-select', panel).value;
-
- $style.disabled = !data.enable;
-
- toggleClass(body, 'ghd-disabled', !data.enable);
- toggleClass(body, 'nowrap', !data.wrap);
-
- if (testing) {
- processStyle();
- testing = false;
- } else {
- fetchAndApplyStyle();
- }
- isUpdating = false;
- }
-
- // user can force GitHub-dark update
- function forceUpdate(css) {
- if (css) {
- // add raw css directly for style testing
- data.rawCss = css;
- processStyle();
- } else {
- // clear saved date
- data.version = 0;
- data.themeCss = '';
- GM_setValue('data', JSON.stringify(data));
- closePanel('forced');
- }
- }
-
- function getVersionTooltip() {
- let indx,
- ver = [],
- // convert stored css version from "001014049" into "1.14.49" for tooltip
- parts = String(data.version).match(/\d{3}/g),
- len = parts && parts.length || 0;
- for (indx = 0; indx < len; indx++) {
- ver.push(parseInt(parts[indx], 10));
- }
- return `Script v${version}\nCSS ${(ver.length ? 'v' + ver.join('.') : 'unknown')}`;
- }
-
- function buildSettings() {
- if (debug) {
- console.log('Adding settings panel & GitHub Dark link to profile dropdown');
- }
- // Script-specific CSS
- GM_addStyle(`
- #ghd-menu:hover { cursor:pointer }
- #ghd-settings { position:fixed; z-index:65535; top:0; bottom:0; left:0; right:0; opacity:0; visibility:hidden; }
- #ghd-settings.in { opacity:1; visibility:visible; background:rgba(0,0,0,.5); }
- #ghd-settings-inner { position:fixed; left:50%; top:50%; transform:translate(-50%,-50%); width:25rem; box-shadow:0 .5rem 1rem #111; color:#c0c0c0 }
- #ghd-settings label { margin-left:.5rem; position:relative; top:-1px }
- #ghd-settings-close { height:1rem; width:1rem; fill:#666; float:right; cursor:pointer }
- #ghd-settings-close:hover { fill:#ccc }
- #ghd-settings .ghd-right { float:right; padding:5px; }
- #ghd-settings p { line-height:22px; }
- #ghd-settings .form-select, #ghd-settings input[type="text"] { height:30px; min-height:30px; }
- #ghd-swatch { width:25px; height:22px; display:inline-block; margin:3px 10px; border-radius:4px; }
- #ghd-settings input, #ghd-settings select { border:#555 1px solid; }
- #ghd-settings .ghd-attach, #ghd-settings .ghd-diff-select { padding-right:25px; }
- #ghd-settings input[type="checkbox"] { margin-top:3px; width: 16px !important; height: 16px !important; border-radius: 3px !important; }
- #ghd-settings .boxed-group-inner { padding:0; }
- #ghd-settings .ghd-footer { padding:10px; border-top:#555 solid 1px; }
- #ghd-settings .ghd-settings-wrapper { max-height:60vh; overflow-y:auto; padding:4px 10px; }
- #ghd-settings .ghd-tab { width:5em; }
- #ghd-settings .octicon { vertical-align:text-bottom !important; }
- #ghd-settings .ghd-paste-area { position:absolute; bottom:50px; top:37px; left:2px; right:2px; width:396px !important; height:-moz-calc(100% - 85px);
- resize:none !important; border-style:solid; z-index:0; }
-
- /* code wrap toggle: https://gist.github.com/silverwind/6c1701f56e62204cc42b
- icons next to a pre */
- .ghd-wrap-toggle { position:absolute; right:1.4em; margin-top:.2em; -moz-user-select:none; -webkit-user-select:none; cursor:pointer; z-index:20; }
- /* file & diff code tables */
- .ghd-wrap-table .blob-code-inner { white-space:pre-wrap !important; word-break:break-all !important; }
- .ghd-unwrap-table .blob-code-inner { white-space:pre !important; word-break:normal !important; }
- .ghd-wrap-toggle > *, .ghd-monospace > *, .ghd-file-toggle > * { pointer-events:none; vertical-align:middle !important; }
- /* icons inside a wrapper immediatly around a pre */
- .highlight > .ghd-wrap-toggle { right:.5em; top:.5em; margin-top:0; }
- /* icons for non-syntax highlighted code blocks; see https://github.com/gjtorikian/html-proofer/blob/master/README.md */
- .markdown-body:not(.comment-body) .ghd-wrap-toggle:not(:first-child) { right:3.4em; }
- .ghd-wrap-toggle svg { height:1.25em; width:1.25em; fill:rgba(110,110,110,.4); }
- /* wrap disabled (red) */
- .ghd-wrap-toggle.unwrap:hover svg, .ghd-wrap-toggle:hover svg { fill:#8b0000; }
- /* wrap enabled (green) */
- body:not(.nowrap) .ghd-wrap-toggle:not(.unwrap):hover svg, .ghd-wrap-toggle.wrapped:hover svg { fill:#006400; }
- .blob-wrapper, .markdown-body pre, .markdown-body .highlight { position:relative; }
- /* monospace font toggle */
- .ghd-monospace-font { font-family:"${data.font}", Menlo, Inconsolata, "Droid Mono", monospace !important; font-size:1em !important; }
- /* file collapsed icon */
- .ghd-file-collapsed > :not(.file-header) { display:none !important; }
- .ghd-file-collapsed .ghd-file-toggle svg { -webkit-transform:rotate(90deg); transform:rotate(90deg); }
- `);
-
- let indx, theme, icon,
- opts = '',
- ver = getVersionTooltip(),
- names = Object.keys(themes),
- len = names.length;
- for (indx = 0; indx < len; indx++) {
- theme = names[indx];
- opts += `<option value="${theme}">${theme}</option>`;
- }
-
- // circle-question-mark icon
- icon = `
- <svg class="octicon" xmlns="http://www.w3.org/2000/svg" height="16" width="14" viewBox="0 0 16 14">
- <path d="M6 10h2v2H6V10z m4-3.5c0 2.14-2 2.5-2 2.5H6c0-0.55 0.45-1 1-1h0.5c0.28 0 0.5-0.22 0.5-0.5v-1c0-0.28-0.22-0.5-0.5-0.5h-1c-0.28 0-0.5 0.22-0.5 0.5v0.5H4c0-1.5 1.5-3 3-3s3 1 3 2.5zM7 2.3c3.14 0 5.7 2.56 5.7 5.7S10.14 13.7 7 13.7 1.3 11.14 1.3 8s2.56-5.7 5.7-5.7m0-1.3C3.14 1 0 4.14 0 8s3.14 7 7 7 7-3.14 7-7S10.86 1 7 1z" />
- </svg>
- `;
-
- // Settings panel markup
- make({
- el : 'div',
- appendTo: 'body',
- attr : { id: 'ghd-settings' },
- html : `
- <div id="ghd-settings-inner" class="boxed-group">
- <h3>GitHub-Dark Settings
- <a href="https://github.com/StylishThemes/GitHub-Dark-Script/wiki" class="tooltipped tooltipped-e" aria-label="See documentation">${icon}</a>
- <svg id="ghd-settings-close" xmlns="http://www.w3.org/2000/svg" width="768" height="768" viewBox="160 160 608 608"><path d="M686.2 286.8L507.7 465.3l178.5 178.5-45 45-178.5-178.5-178.5 178.5-45-45 178.5-178.5-178.5-178.5 45-45 178.5 178.5 178.5-178.5z"/></svg>
- </h3>
- <div class="boxed-group-inner">
- <form>
- <div class="ghd-settings-wrapper">
- <p class="ghd-checkbox">
- <label>Enable GitHub-Dark<input class="ghd-enable ghd-right" type="checkbox"></label>
- </p>
- <p>
- <label>Color:</label>
- <input class="ghd-color ghd-right" type="text" value="#4183C4">
- <span id="ghd-swatch" class="ghd-right"></span>
- </p>
- <h4>Background</h4>
- <p>
- <label>Image:</label>
- <input class="ghd-image ghd-right" type="text">
- <a href="https://github.com/StylishThemes/GitHub-Dark/wiki/Image" class="tooltipped tooltipped-e" aria-label="Click to learn about GitHub\'s Content Security Policy and how to add a custom image">${icon}</a>
- </p>
- <p>
- <label>Image type:</label>
- <select class="ghd-type ghd-right form-select">
- <option value="tiled">Tiled</option>
- <option value="fit">Fit window</option>
- </select>
- </p>
- <p>
- <label>Image attachment:</label>
- <select class="ghd-attach ghd-right form-select">
- <option value="scroll">Scroll</option>
- <option value="fixed">Fixed</option>
- </select>
- </p>
- <h4>Code</h4>
- <p><label>Theme:</label> <select class="ghd-theme ghd-right form-select">${opts}</select></p>
- <p>
- <label>Font Name:</label> <input class="ghd-font ghd-right" type="text">
- <a href="http://www.cssfontstack.com/" class="tooltipped tooltipped-e" aria-label="Add a system installed (monospaced) font name; this script will not load external fonts!">${icon}</a>
- </p>
- <p>
- <label>Tab Size:</label> <input class="ghd-tab ghd-right" type="text">
- </p>
- <p class="ghd-checkbox">
- <label>Wrap<input class="ghd-wrap ghd-right" type="checkbox"></label>
- </p>
- <h4>Toggles</h4>
- <p class="ghd-checkbox">
- <label>Code Wrap<input class="ghd-codewrap-checkbox ghd-right" type="checkbox"></label>
- </p>
- <p class="ghd-checkbox">
- <label>Comment Monospace Font<input class="ghd-monospace-checkbox ghd-right" type="checkbox"></label>
- </p>
- <p class="ghd-range">
- <label>Diff File Collapse</label>
- <select class="ghd-diff-select ghd-right form-select">
- <option value="0">Disabled</option>
- <option value="1">Enabled</option>
- <option value="2">Accordion</option>
- </select>
- </p>
- </div>
- <div class="ghd-footer">
- <div class="BtnGroup">
- <a href="#" class="ghd-update btn btn-sm tooltipped tooltipped-n tooltipped-multiline" aria-label="Update style if the newest release is not loading; the page will reload!">Force Update Style</a>
- <a href="#" class="ghd-textarea-toggle btn btn-sm tooltipped tooltipped-n" aria-label="Paste CSS update">
- <svg xmlns="http://www.w3.org/2000/svg" width="8" height="8" viewBox="0 0 16 16" fill="#eee">
- <path d="M15 11 1 11 8 3z"/>
- </svg>
- </a>
- <div class="ghd-paste-area-content" aria-hidden="true" style="display:none">
- <textarea class="ghd-paste-area" placeholder="Paste GitHub-Dark Style here!"></textarea>
- </div>
- </div>
- <a href="#" class="ghd-reset btn btn-sm btn-danger tooltipped tooltipped-n" aria-label="Reset to defaults; there is no undo!">Reset All Settings</a>
- <span class="ghd-versions ghd-right tooltipped tooltipped-n" aria-label="${ver}">
- <svg class="ghd-info" xmlns="http://www.w3.org/2000/svg" width="16px" height="16px" viewBox="0 0 24 24">
- <path fill="#444" d="M12,9c0.82,0,1.5-0.68,1.5-1.5S12.82,6,12,6s-1.5,0.68-1.5,1.5S11.18,9,12,9z M12,1.5 C6.211,1.5,1.5,6.211,1.5,12S6.211,22.5,12,22.5S22.5,17.789,22.5,12S17.789,1.5,12,1.5z M12,19.5c-4.148,0-7.5-3.352-7.5-7.5 S7.852,4.5,12,4.5s7.5,3.352,7.5,7.5S16.148,19.5,12,19.5z M13.5,12c0-0.75-0.75-1.5-1.5-1.5s-0.75,0-1.5,0S9,11.25,9,12h1.5 c0,0,0,3.75,0,4.5S11.25,18,12,18s0.75,0,1.5,0s1.5-0.75,1.5-1.5h-1.5C13.5,16.5,13.5,12.75,13.5,12z"/>
- </svg>
- </span>
- </div>
- </form>
- </div>
- </div>
- `
- });
-
- updateToggles();
- }
-
- // Add code wrap toggle
- function buildCodeWrap() {
- // mutation events happen quick, so we still add an update flag
- isUpdating = true;
- let icon = make({
- el : 'div',
- cl4ss : 'ghd-wrap-toggle tooltipped tooltipped-w',
- attr : { 'aria-label' : 'Toggle code wrap' },
- html : wrapIcon
- });
- $$('.blob-wrapper').forEach(el => {
- el.insertBefore(icon.cloneNode(true), el.childNodes[0]);
- });
- $$('.markdown-body pre').forEach(el => {
- el.parentNode.insertBefore(icon.cloneNode(true), el);
- });
- isUpdating = false;
- }
-
- // Add monospace font toggle
- function addMonospaceToggle() {
- isUpdating = true;
- let button = make({
- el : 'button',
- cl4ss : 'ghd-monospace toolbar-item tooltipped tooltipped-n',
- attr : {
- 'type' : 'button',
- 'aria-label' : 'Toggle monospaced font',
- 'tabindex' : '-1'
- },
- html : monospaceIcon
- });
- $$('.toolbar-commenting').forEach(el => {
- if (!$('.ghd-monospace', el)) {
- // prepend
- el.insertBefore(button.cloneNode(true), el.childNodes[0]);
- }
- });
- isUpdating = false;
- }
-
- // Add file diffs toggle
- function addFileToggle() {
- isUpdating = true;
- let firstButton,
- button = make({
- el : 'button',
- cl4ss : 'ghd-file-toggle btn btn-sm tooltipped tooltipped-n',
- attr : {
- 'type' : 'button',
- 'aria-label' : 'Click to Expand or Collapse file',
- 'tabindex' : '-1'
- },
- html : fileIcon
- });
- $$('#files .file-actions').forEach(el => {
- if (!$('.ghd-file-toggle', el)) {
- el.appendChild(button.cloneNode(true));
- }
- });
- firstButton = $('.ghd-file-toggle');
- // accordion mode = start with all but first entry collapsed
- if (firstButton && data.modeDiffToggle === '2') {
- toggleFile({
- target: firstButton
- }, true);
- }
- isUpdating = false;
- }
-
- // Add toggle buttons after page updates
- function updateToggles() {
- if (data.enableCodeWrap) {
- buildCodeWrap();
- } else {
- removeAll('.ghd-wrap-toggle');
- toggleClass($$('.ghd-file-collapsed'), 'ghd-file-collapsed', false);
- }
- if (data.enableMonospace) {
- addMonospaceToggle();
- } else {
- removeAll('.ghd-monospace');
- toggleClass($$('.ghd-monospace-font'), 'ghd-monospace-font', false);
- }
- if (data.modeDiffToggle !== '0') {
- addFileToggle();
- } else {
- removeAll('.ghd-file-toggle');
- toggleClass($$('.ghd-file-collapsed'), 'ghd-file-collapsed', false);
- }
- }
-
- function makeRow(vals, str) {
- return make({
- el : 'tr',
- cl4ss : 'ghd-shortcut',
- html : `<td class="keys"><kbd>${vals[0]}</kbd> <kbd>${vals[1]}</kbd></td><td>${str}</td>`
- });
- }
-
- // add keyboard shortcut to help menu (press "?")
- function buildShortcut() {
- let row,
- el = $('.keyboard-mappings tbody');
- if (el && !$('.ghd-shortcut')) {
- row = makeRow(keyboardOpen.split('+'), 'GitHub-Dark: open settings');
- el.appendChild(row);
- row = makeRow(keyboardToggle.split('+'), 'GitHub-Dark: toggle style');
- el.appendChild(row);
- }
- }
-
- function toggleCodeWrap(el) {
- let css,
- overallWrap = data.wrap,
- code = next(el, '.highlight, .diff-table, code, pre'),
- tmp = code ? $('code', code) : '';
- if (tmp) {
- // find code element
- code = tmp;
- }
- if (!code) {
- if (debug) {
- console.log('Code wrap icon associated code not found', el);
- }
- return;
- }
- // code with line numbers
- if (code.nodeName === 'TABLE') {
- if (code.className.indexOf('ghd-') < 0) {
- css = !overallWrap;
- } else {
- css = code.classList.contains('ghd-unwrap-table');
- }
- toggleClass(code, 'ghd-wrap-table', css);
- toggleClass(code, 'ghd-unwrap-table', !css);
- toggleClass(el, 'wrapped', css);
- toggleClass(el, 'unwrap', !css);
- } else {
- css = code.getAttribute('style') || '';
- if (css === '') {
- css = wrapCss[overallWrap ? 'unwrap' : 'wrapped'];
- } else {
- css = wrapCss[css === wrapCss.wrapped ? 'unwrap' : 'wrapped'];
- }
- code.setAttribute('style', css);
- toggleClass(el, 'wrapped', css === wrapCss.wrapped);
- toggleClass(el, 'unwrap', css === wrapCss.unwrap);
- }
- }
-
- function toggleMonospace(el) {
- let tmp = closest(el, '.previewable-comment-form'),
- // single comment
- textarea = $('.comment-form-textarea', tmp);
- if (textarea) {
- toggleClass(textarea, 'ghd-monospace-font');
- textarea.focus();
- tmp = textarea.classList.contains('ghd-monospace-font');
- toggleClass(el, 'ghd-icon-active', tmp);
- }
- }
-
- function toggleSibs(target, state) {
- let el,
- isCollapsed = state || target.classList.contains('ghd-file-collapsed'),
- toggles = document.querySelectorAll('.file'),
- indx = toggles.length;
- while (indx--) {
- el = toggles[indx];
- if (el !== target) {
- el.classList[isCollapsed ? 'add' : 'remove']('ghd-file-collapsed');
- }
- }
- }
-
- function toggleFile(event, init) {
- isUpdating = true;
- let el = closest(event.target, '.file');
- if (data.modeDiffToggle === '2') {
- if (!init) {
- el.classList.toggle('ghd-file-collapsed');
- }
- toggleSibs(el, true);
- } else {
- el.classList.toggle('ghd-file-collapsed');
- // shift+click toggle all files!
- if (event.shiftKey) {
- toggleSibs(el);
- }
- }
- isUpdating = false;
- }
-
- function bindEvents() {
- let el, menu, lastKey,
- panel = $('#ghd-settings-inner'),
- swatch = $('#ghd-swatch', panel);
-
- // finish initialization
- $('#ghd-settings-inner .ghd-enable').checked = data.enable;
- toggleClass($('body'), 'ghd-disabled', !data.enable);
-
- // Create our menu entry
- menu = make({
- el : 'a',
- cl4ss : 'dropdown-item',
- html : 'GitHub Dark Settings',
- attr : { id : 'ghd-menu' }
- });
-
- el = $$('.header .dropdown-item[href="/settings/profile"], .header .dropdown-item[data-ga-click*="go to profile"]');
- // get last found item - gists only have the "go to profile" item; GitHub has both
- el = el[el.length - 1];
- if (el) {
- // insert after
- el.parentNode.insertBefore(menu, el.nextSibling);
- on($('#ghd-menu'), 'click', () => {
- openPanel();
- });
- }
-
- on(document, 'keypress keydown', event => {
- clearTimeout(timer);
- // use "g+o" to open up ghd options panel
- let openKeys = keyboardOpen.split('+'),
- toggleKeys = keyboardToggle.split('+'),
- key = String.fromCharCode(event.which).toLowerCase(),
- panelVisible = $('#ghd-settings').classList.contains('in');
-
- // press escape to close the panel
- if (event.which === 27 && panelVisible) {
- closePanel();
- return;
- }
- // use event.which from keypress for shortcuts
- // prevent opening panel while typing "go" in comments
- if (event.type === 'keydown' || /(input|textarea)/i.test(document.activeElement.nodeName)) {
- return;
- }
- if (lastKey === openKeys[0] && key === openKeys[1]) {
- if (panelVisible) {
- closePanel();
- } else {
- openPanel();
- }
- }
- if (lastKey === toggleKeys[0] && key === toggleKeys[1]) {
- toggleStyle();
- }
- lastKey = key;
- timer = setTimeout(() => {
- lastKey = null;
- }, keyboardDelay);
-
- // add shortcut to help menu
- if (key === '?') {
- // table doesn't exist until user presses "?"
- setTimeout(() => {
- buildShortcut();
- }, 300);
- }
- });
-
- // add ghd-settings panel bindings
- on($$('#ghd-settings, #ghd-settings-close'), 'click keyup', event => {
- // press escape to close settings
- if (event.type === 'keyup' && event.which !== 27) {
- return;
- }
- closePanel();
- });
-
- on(panel, 'click', event => {
- event.stopPropagation();
- });
-
- on($('.ghd-reset', panel), 'click', event => {
- event.preventDefault();
- isUpdating = true;
- // pass true to reset values
- setStoredValues(true);
- // add reset values back to data
- getStoredValues();
- // add reset values to panel
- updatePanel();
- // update style
- updateStyle();
- });
-
- on($$('input[type="text"]', panel), 'focus', function() {
- // select all text when focused
- this.select();
- });
-
- on($$('select, input', panel), 'change', () => {
- if (!isUpdating) {
- updateStyle();
- }
- });
-
- on($('.ghd-update', panel), 'click', event => {
- event.preventDefault();
- forceUpdate();
- });
-
- on($('.ghd-textarea-toggle', panel), 'click', function(event) {
- event.preventDefault();
- let hidden, el;
- this.classList.remove('selected');
- el = next(this, '.ghd-paste-area-content');
- if (el) {
- hidden = el.style.display === 'none';
- el.style.display = hidden ? '' : 'none';
- if (el.style.display !== 'none') {
- el.classList.add('selected');
- el = $('textarea', el);
- el.focus();
- el.select();
- }
- }
- return false;
- });
-
- on($('.ghd-paste-area-content', panel), 'paste', event => {
- let toggle = $('.ghd-textarea-toggle', panel),
- textarea = event.target;
- setTimeout(() => {
- textarea.parentNode.style.display = 'none';
- toggle.classList.remove('selected');
- testing = true;
- forceUpdate(textarea.value);
- }, 200);
- });
-
- // Toggles
- on($('body'), 'click', event => {
- let target = event.target;
- if (data.enableCodeWrap && target.classList.contains('ghd-wrap-toggle')) {
- // **** CODE WRAP TOGGLE ****
- event.stopPropagation();
- toggleCodeWrap(target);
- } else if (data.enableMonospace && target.classList.contains('ghd-monospace')) {
- // **** MONOSPACE FONT TOGGLE ****
- event.stopPropagation();
- toggleMonospace(target);
- return false;
- } else if (data.modeDiffToggle !== '0' && target.classList.contains('ghd-file-toggle')) {
- // **** CODE DIFF COLLAPSE TOGGLE ****
- event.stopPropagation();
- toggleFile(event);
- }
- });
-
- // style color picker
- picker = new jscolor($('.ghd-color', panel));
- picker.zIndex = 65536;
- picker.hash = true;
- picker.backgroundColor = '#333';
- picker.padding = 0;
- picker.borderWidth = 0;
- picker.borderColor = '#444';
- picker.onFineChange = () => {
- swatch.style.backgroundColor = '#' + picker;
- };
- }
-
- function openPanel() {
- $('.modal-backdrop').click();
- updatePanel();
- $('#ghd-settings').classList.add('in');
- }
-
- function closePanel(flag) {
- $('#ghd-settings').classList.remove('in');
- picker.hide();
-
- if (flag === 'forced') {
- // forced update; partial re-initialization
- init();
- } else {
- // apply changes when the panel is closed
- updateStyle();
- }
- }
-
- function toggleStyle() {
- let isEnabled = !data.enable;
- data.enable = isEnabled;
- $('#ghd-settings-inner .ghd-enable').checked = isEnabled;
- // add processedCss back into style (emptied when disabled)
- if (isEnabled) {
- // data.processedCss is empty when ghd is disabled on init
- if (!data.processedCss) {
- processStyle();
- } else {
- addSavedStyle();
- }
- }
- $style.disabled = !isEnabled;
- }
-
- function init() {
- if (!document.head) return;
-
- document.head.appendChild($style);
- getStoredValues(true);
-
- $style.disabled = !data.enable;
- data.lastTheme = data.theme;
- data.lastCW = data.enableCodeWrap;
- data.lastMS = data.enableMonospace;
- data.lastDT = data.modeDiffToggle;
-
- // only load package.json once a day, or after a forced update
- if ((new Date().getTime() > data.date + delay) || data.version === 0) {
- // get package.json from GitHub-Dark & compare versions
- // load new script if a newer one is available
- checkVersion();
- } else {
- addSavedStyle();
- }
- isInitialized = false;
- }
-
- // add style at document-start
- init();
-
- on(document, 'DOMContentLoaded', () => {
- if (isInitialized === 'pending') {
- // init after DOM loaded on .atom pages
- init();
- }
- // add panel even if you're not logged in - open panel using keyboard shortcut
- // just not on githubusercontent pages (no settings panel needed)
- if (window.location.host.indexOf('githubusercontent.com') < 0) {
- buildSettings();
- // add event binding on document ready
- bindEvents();
-
- $$('#js-repo-pjax-container, #js-pjax-container, .js-contribution-activity').forEach(target => {
- new MutationObserver(mutations => {
- mutations.forEach(mutation => {
- // preform checks before adding code wrap to minimize function calls
- if (!(isUpdating || $$('.ghd-wrap-toggle').length) &&
- mutation.target === target) {
- updateToggles();
- }
- });
- }).observe(target, {
- childList: true,
- subtree: true
- });
- });
- }
-
- isInitialized = true;
- });
-
- /* utility functions */
- function isBool(name) {
- let val = data[name];
- return typeof val === 'boolean' ? val : defaults[name];
- }
- function $(str, el) {
- return (el || document).querySelector(str);
- }
- function $$(str, el) {
- return Array.from((el || document).querySelectorAll(str));
- }
- function next(el, selector) {
- while ((el = el.nextElementSibling)) {
- if (el && el.matches(selector)) {
- return el;
- }
- }
- return null;
- }
- function closest(el, selector) {
- while (el && el.nodeName !== 'BODY' && !el.matches(selector)) {
- el = el.parentNode;
- }
- return el && el.matches(selector) ? el : [];
- }
- function make(obj) {
- let key,
- el = document.createElement(obj.el);
- if (obj.cl4ss) { el.className = obj.cl4ss; }
- if (obj.html) { el.innerHTML = obj.html; }
- if (obj.attr) {
- for (key in obj.attr) {
- if (obj.attr.hasOwnProperty(key)) {
- el.setAttribute(key, obj.attr[key]);
- }
- }
- }
- if (obj.appendTo) {
- $(obj.appendTo).appendChild(el);
- }
- return el;
- }
- function removeAll(selector) {
- $$(selector).forEach(el => {
- el.parentNode.removeChild(el);
- });
- }
- function on(els, name, callback) {
- els = Array.isArray(els) ? els : [els];
- let events = name.split(/\s+/);
- els.forEach(el => {
- events.forEach(ev => {
- el.addEventListener(ev, callback);
- });
- });
- }
- function toggleClass(els, cl4ss, flag) {
- els = Array.isArray(els) ? els : [els];
- els.forEach(el => {
- if (el) {
- if (typeof flag === 'undefined') {
- flag = !el.classList.contains(cl4ss);
- }
- if (flag) {
- el.classList.add(cl4ss);
- } else {
- el.classList.remove(cl4ss);
- }
- }
- });
- }
-
- // Add GM options
- GM_registerMenuCommand("GitHub Dark Script debug logging", () => {
- let val = prompt('Toggle GitHub Dark Script debug log (true/false):', !debug);
- if (val) {
- debug = /^t/.test(val);
- GM_setValue('debug', debug);
- }
- });
-
- })();