// ==UserScript== // @name GitHub Dark Script // @version 2.0.2 // @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 /render\.githubusercontent\.com/ // @include /raw\.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://greasyfork.org/scripts/15563-jscolor/code/jscolor.js?version=106439 // @downloadURL none // ==/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', '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', '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 = ` `, monospaceIcon = ` `, fileIcon = ` `, $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) { 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); 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', reset ? defaults : data); updatePanel(); if (debug) { console.log((reset ? 'Resetting' : 'Saving') + ' current values', data); } } // 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', 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') // 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', 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 += ``; } // circle-question-mark icon icon = ` `; // Settings panel markup make({ el : 'div', appendTo: 'body', attr : { id: 'ghd-settings' }, html : `