Last active
February 26, 2024 08:00
-
-
Save whispy/bf644aaa7d4c8759e637074e57209658 to your computer and use it in GitHub Desktop.
This script adds a new sticky header to Basecamp projects and todo lists on scroll.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// ==UserScript== | |
// @name Basecamp - Projects Sticky Header | |
// @namespace http://tampermonkey.net/ | |
// @version 0.13 | |
// @description This script adds a new sticky header to Basecamp projects and todo lists on scroll. Never forget where you are again! | |
// @updateURL https://gist.github.com/whispy/bf644aaa7d4c8759e637074e57209658/raw/basecamp-projects-sticky-header.user.js | |
// @downloadURL https://gist.github.com/whispy/bf644aaa7d4c8759e637074e57209658/raw/basecamp-projects-sticky-header.user.js | |
// @author Dan Berkowitz | |
// @match https://3.basecamp.com/* | |
// @grant GM_addStyle | |
// @require https://npmcdn.com/headroom.js/dist/headroom.min.js | |
// ==/UserScript== | |
GM_addStyle( | |
'#project-header-clone { pointer-events: none; transition: transform 120ms ease-in-out; font-size:0.9em; position:fixed; top:0; left:0; right:0; transform:translateY(19%); z-index:-1;}' | |
); | |
GM_addStyle( | |
'#project-header-clone:not(.headroom--top).sticky-pinned {transform:translateY(153%);}' | |
); | |
GM_addStyle( | |
'#project-header-clone.project-header-mobile__padding-top-bottom, #project-header-clone.project-guide__header { display:none; }' | |
); | |
GM_addStyle ( | |
'.sticky-pinned h1 > span.options-menu { background-color:rgba(246,242,239,1); padding:.5em; box-shadow: 0 1px 3px 0 rgba(0,0,0,0.4); border-radius: 0 0 0.6rem 0.6rem;}' | |
); | |
GM_addStyle ( | |
'#project-header-clone button.recording-breadcrumb__jump, #project-header-clone span.recording-breadcrumb__jump-menu {display:none;}' | |
); | |
GM_addStyle ( | |
'.nav__main { background-color: rgba(246,242,239,1); max-width:100%; width:100%;' | |
); | |
GM_addStyle ( | |
'#project-header-clone .crumb-wrapper { pointer-events: all; display:inline-block; background-color:rgba(246,242,239,1); padding:.5em; box-shadow: 0 1px 3px 0 rgba(0,0,0,0.4); border-radius: 0 0 0.6rem 0.6rem;}' | |
); | |
function scrollToTop(scrollDuration) { | |
var cosParameter = window.scrollY / 2, | |
scrollCount = 0, | |
oldTimestamp = performance.now(); | |
function step (newTimestamp) { | |
scrollCount += Math.PI / (scrollDuration / (newTimestamp - oldTimestamp)); | |
if (scrollCount >= Math.PI) window.scrollTo(0, 0); | |
if (window.scrollY === 0) return; | |
window.scrollTo(0, Math.round(cosParameter + cosParameter * Math.cos(scrollCount))); | |
oldTimestamp = newTimestamp; | |
window.requestAnimationFrame(step); | |
} | |
window.requestAnimationFrame(step); | |
} | |
function cloneHeader() { | |
var projectHeader = document.querySelector("div.recording-breadcrumb__title"); | |
var mainNav = document.querySelector("ul.nav__main"); | |
var clone = projectHeader.cloneNode(true); //Clone the projectHeader DOM node and all child elements | |
clone.id = "project-header-clone"; //Add an ID tag to the cloned node so I can target it individually. | |
mainNav.appendChild(clone); //Append the cloned node as a child of mainNav | |
// The breadcrumbs don't have a wrapper, so we need to add one to style them. We can also enhance the breadcrumbs by getting the task name from the window title and appending it | |
var modified_HTML = "<div class='crumb-wrapper'><a class='scrollTop' style='cursor:pointer;'>⬆</>" + clone.innerHTML + '<div class="recording-breadcrumb__separator" style="display:inline;" aria-hidden="true"> › </div>' + document.title + "</div>"; //Copy the breadcrumb nodes from inside the cloned element, get the page title (which is the task name), and then wrap them in a div | |
clone.innerHTML = modified_HTML; //Replace the cloned element's inner HTML with the newly wrapped nodes | |
// Add ability to scroll to top of page. | |
let scrollTopButton = document.querySelector('.scrollTop'); | |
scrollTopButton.addEventListener('click', function(){ | |
scrollToTop(200); | |
}); | |
} | |
function makeSticky(x) { | |
headroomJS = new Headroom(x, { | |
"offset" : 200, | |
"tolerance" : 3, | |
"classes" : { | |
"initial" : "sticky-top", | |
"pinned" : "sticky-pinned", | |
} | |
}); | |
headroomJS.init(); //Initialize Headroom.JS | |
} | |
document.addEventListener("turbolinks:load", function() { // Hooks into Turbolinks' page load event (https://github.com/turbolinks/turbolinks#full-list-of-events) | |
var cloneExists = document.querySelector("#project-header-clone"); | |
if ( cloneExists === null ) { // If element does not exist | |
cloneHeader(); // Clone and append the element | |
cloneExists = document.querySelector("#project-header-clone"); // After clone and append, redefine cloneExists variable in the local scope | |
} | |
else { | |
cloneExists.remove(); // Remove the cloned header due to it persisting for one page too long | |
cloneHeader(); // Clone and append the element | |
var cloneExists = document.querySelector("#project-header-clone"); //Target the newly cloned and appended element | |
} | |
makeSticky(cloneExists); | |
}); | |
window.onload = function() { // Despite Turbolinks' docs saying `turbolinks:load` fires on initial load, it doesn't appear to. This takes care of that. | |
cloneHeader(); // Clone and append the element | |
var cloneExists = document.querySelector("#project-header-clone"); | |
makeSticky(cloneExists); | |
}; | |
0.13
- Allow clicking on items that occupy the same horizontal space as the breadcrumbs.
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
0.12