// ==UserScript==
// @name Branding? What branding? Logo picker for twitter.com
// @namespace Itsnotlupus Industries
// @match https://twitter.com/*
// @version 1.3
// @author Itsnotlupus
// @license MIT
// @description Adds a drop down by the site logo to let you pick logo alternatives.
// @defaulticon https://abs.twimg.com/favicons/twitter.2.ico
// @require https://greatest.deepsurf.us/scripts/468394-itsnotlupus-tiny-utilities/code/utils.js
// @run-at document-start
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_addValueChangeListener
// ==/UserScript==
/* jshint esversion:11 */
let logo = GM_getValue("logo", {});
GM_addValueChangeListener("logo", () => {
logo = GM_getValue("logo");
document.body.classList.toggle('logoChanged');
});
const LOGOS = [
{
// new weird X logo
html: `<svg viewBox="0 0 24 24" aria-hidden="true" class="twitter-x"><g><path d=" M14.258 10.152L23.176 0h-2.113l-7.747 8.813L7.133 0H0l9.352 13.328L0 23.973h2.113l8.176-9.309 6.531 9.309h7.133zm-2.895 3.293l-.949-1.328L2.875 1.56h3.246l6.086 8.523.945 1.328 7.91 11.078h-3.246zm0 0"></path></g></svg>`,
favicon: "https://abs.twimg.com/favicons/twitter.3.ico"
},
{
// recent twitter bird logo
html: `<svg viewBox="0 0 24 24" aria-hidden="true" class="twitter-bird"><g><path d="M23.643 4.937c-.835.37-1.732.62-2.675.733.962-.576 1.7-1.49 2.048-2.578-.9.534-1.897.922-2.958 1.13-.85-.904-2.06-1.47-3.4-1.47-2.572 0-4.658 2.086-4.658 4.66 0 .364.042.718.12 1.06-3.873-.195-7.304-2.05-9.602-4.868-.4.69-.63 1.49-.63 2.342 0 1.616.823 3.043 2.072 3.878-.764-.025-1.482-.234-2.11-.583v.06c0 2.257 1.605 4.14 3.737 4.568-.392.106-.803.162-1.227.162-.3 0-.593-.028-.877-.082.593 1.85 2.313 3.198 4.352 3.234-1.595 1.25-3.604 1.995-5.786 1.995-.376 0-.747-.022-1.112-.065 2.062 1.323 4.51 2.093 7.14 2.093 8.57 0 13.255-7.098 13.255-13.254 0-.2-.005-.402-.014-.602.91-.658 1.7-1.477 2.323-2.41z"></path></g></svg>`,
favicon: "https://abs.twimg.com/favicons/twitter.2.ico"
},
{ // CSP prevents us from linking to 3rd party servers for old images, so we inline them instead.
html: `<img class="twitter-classic" src="">`,
favicon: ""
},
{
html: `<img class="twitter-classic" src="">`,
favicon: ""
}
// moar later. probably.
];
document.head.prepend(crel('style', { textContent: `
header[role="banner"] h1[role="heading"] {
flex-direction: row;
}
header[role="banner"] h1[role="heading"]:hover .logo-dropdown-arrow {
opacity: 1;
}
body.logo-dropdown-open .logo-dropdown-arrow {
opacity: 1;
}
.logo-dropdown-arrow {
opacity: 0;
transition: all 250ms;
width: 20px;
height: 20px;
line-height: 22px;
margin-right: -20px;
text-align: center;
color: var(--twitter-icon-color);
border-radius: 9px;
background: var(--twitter-bg-color);
}
.logo-dropdown-arrow:hover {
background: var(--icon-hover-bg);
}
.logo-dropdown-backdrop {
position: fixed;
inset: 0;
background: rgba(0,0,0,0);
}
.logo-dropdown {
position: fixed;
width: 3rem;
background: var(--twitter-bg-color);
padding: 0.5em;
border-radius: 5px;
box-shadow: var(--dropdown-box-1) 0px 0px 15px, var(--dropdown-box-2) 0px 0px 3px 1px;
}
.logo-dropdown-item {
cursor: pointer;
height: 2rem;
margin-top: 0.5em;
margin-bottom: 0.5em;
padding: 8px;
transition: all 250ms;
border-radius: 999px;
}
.logo-dropdown-item:hover {
background: var(--icon-hover-bg);
}
/* custom CSS for each logo */
.twitter-x {
height: 2rem;
-ms-flex-positive: 1;
-webkit-box-flex: 1;
-webkit-flex-grow: 1;
flex-grow: 1;
color: var(--twitter-icon-color);
-moz-user-select: none;
-ms-user-select: none;
-webkit-user-select: none;
user-select: none;
vertical-align: text-bottom;
position: relative;
max-width: 100%;
fill: currentcolor;
display: inline-block;
}
.twitter-bird {
height: 2rem;
color: rgba(29,155,240,1.00) !important;
vertical-align: text-bottom;
position: relative;
max-width: 100%;
fill: currentcolor;
-moz-user-select: none;
-ms-user-select: none;
-webkit-user-select: none;
user-select: none;
display: inline-block;
}
.twitter-classic {
max-width: 2rem;
vertical-align: text-bottom;
position: relative;
-moz-user-select: none;
-ms-user-select: none;
-webkit-user-select: none;
user-select: none;
display: inline-block;
}
`}));
// 1. early run to replace placeholder logo on app start.
untilDOM("#placeholder").then(e=> {
// tweak the loading logo right quick.
if (logo.html) {
const classes = e.firstChild.getAttribute("class");
e.innerHTML = logo.html;
e.firstChild.setAttribute('class', e.firstChild.getAttribute('class') + ' ' + classes);
e.firstChild.setAttribute('style', "max-width: initial; height: initial");
}
});
// 2. watch for lightmode/darkmode changes and adjust (cheaply.)
(async()=> {
function getBgColor() {
try {
return getComputedStyle(document.body).backgroundColor;
} catch {
return document.body?.style.backgroundColor;
}
}
let prevBgColor = undefined;
while (true) {
const bgColor = getBgColor();
if (bgColor != prevBgColor) {
prevBgColor = bgColor;
const isDarkMode = bgColor.replace(/[rgb()]+/g,'').split(', ').reduce((v,a)=>+a+v, 0) < 255;
document.body.style.setProperty("--twitter-bg-color", bgColor);
document.body.style.setProperty("--twitter-icon-color", isDarkMode ? "rgba(214,217,219,1.00)" : "rgba(36,46,54,1.00)");
document.body.style.setProperty("--icon-hover-bg", isDarkMode ? "rgba(239, 243, 244, 0.1)" : "rgba(15, 20, 25, 0.1)");
document.body.style.setProperty("--dropdown-bg-color", isDarkMode ? "#111" : "#fff");
document.body.style.setProperty("--dropdown-box-1", isDarkMode ? "rgba(255, 255, 255, 0.2)" : "rgba(101, 119, 134, 0.2)")
document.body.style.setProperty("--dropdown-box-2", isDarkMode ? "rgba(255, 255, 255, 0.15)" : "rgba(101, 119, 134, 0.15)")
}
await sleep();
}
})();
// 3. inject our logo picker dropdown, potentially several times.
(async()=> {
while (true) {
const heading = await untilDOM(`header[role="banner"] h1[role="heading"]`);
heading.append(crel('a', {
className: "logo-dropdown-arrow",
textContent: '▾',
onclick() { openLogoDropDown(); }
}));
// await applyLogo();
// wait until our dropdown gets wiped, and reapply our tweaks
await untilDOM(()=>!$`.logo-dropdown-arrow`);
}
})();
// 3½. dropdown and logo selection logic
function openLogoDropDown() {
const { bottom, left } = $`.logo-dropdown-arrow`.getBoundingClientRect();
const backdrop = crel('div', {
className: "logo-dropdown-backdrop",
onclick() {
backdrop.remove();
dropdown.remove();
document.body.classList.remove('logo-dropdown-open');
}
});
const dropdown = crel('div', {
className: "logo-dropdown",
style: `top: ${bottom}px; left: ${left}px;`
}, ...LOGOS.map(l => crel('div', {
className: 'logo-dropdown-item',
innerHTML: l.html,
onclick() {
logo = l;
GM_setValue("logo", logo);
// applyLogo();
backdrop.click();
}
})));
document.body.append(backdrop, dropdown);
document.body.classList.add('logo-dropdown-open');
}
// 4. wait until the placeholder logo is out of the way, then replace the favicon and all logos aggressively.
(async function applyLogo() {
await untilDOM(()=>!$`#placeholder`);
observeDOM(async() => {
if (logo.favicon) {
const shortcutLink = $`link[rel="shortcut icon"]`;
if (shortcutLink && shortcutLink.href !== logo.favicon) {
shortcutLink.href = logo.favicon;
}
}
if (logo.html) {
$$`path[d="M14.258 10.152L23.176 0h-2.113l-7.747 8.813L7.133 0H0l9.352 13.328L0 23.973h2.113l8.176-9.309 6.531 9.309h7.133zm-2.895 3.293l-.949-1.328L2.875 1.56h3.246l6.086 8.523.945 1.328 7.91 11.078h-3.246zm0 0"]`.forEach(path=> {
const svg = path.closest`svg`;
const classes = svg.getAttribute("class");
const logoElt = crel('div', { innerHTML: logo.html}).firstChild;
svg.replaceWith(logoElt);
logoElt.setAttribute('class', logoElt.getAttribute('class') + ' ' + classes);
logoElt.setAttribute('style', "max-width: initial"); //; height: initial");
});
const logoParent = $`header[role="banner"] h1[role="heading"] a[href="/home"] > div`;
if (logoParent && logoParent.innerHTML !== logo.html) {
logoParent.innerHTML = logo.html;
}
}
});
})();