Instantly share code, notes, and snippets.
Created
April 24, 2024 13:08
-
Star
0
(0)
You must be signed in to star a gist -
Fork
0
(0)
You must be signed in to fork a gist
-
Save relipse/37ae4243314500cfee7d82e4280cb986 to your computer and use it in GitHub Desktop.
This file contains hidden or 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 Bible Study | |
// @namespace https://esv.literalword.com/ | |
// @version 2024-03-27 | |
// @description try to take over the world! | |
// @author TearsForNations | |
// @match https://*.literalword.com/* | |
// @require https://unpkg.com/[email protected]/chroma.js | |
// @require http://ajax.googleapis.com/ajax/libs/jquery/1.7.2/jquery.min.js | |
// @grant none | |
// ==/UserScript== | |
// Function to set a cookie | |
let divWidth = 200; | |
let divHeight = 100; | |
setCookie = function(cname, cvalue, exdays = 30) { | |
var d = new Date(); | |
d.setTime(d.getTime() + (exdays*24*60*60*1000)); | |
var expires = "expires="+ d.toUTCString(); | |
document.cookie = cname + "=" + cvalue + ";" + expires + ";path=/"; | |
} | |
// Function to get a cookie | |
getCookie = function(cname) { | |
var name = cname + "="; | |
var decodedCookie = decodeURIComponent(document.cookie); | |
var ca = decodedCookie.split(';'); | |
for(var i = 0; i <ca.length; i++) { | |
var c = ca[i]; | |
while (c.charAt(0) == ' ') { | |
c = c.substring(1); | |
} | |
if (c.indexOf(name) == 0) { | |
return c.substring(name.length, c.length); | |
} | |
} | |
return ""; | |
} | |
let isDivTransformed = false; | |
let BibleStudyNavWidthSmall = '25px'; | |
transformOrUntransform = function(divElement, defaultSize = null, defaultHeight = null){ | |
defaultSize = defaultSize || divWidth; | |
defaultHeight = defaultHeight || divHeight; | |
let isDivTransformed = divElement.style.width == BibleStudyNavWidthSmall; | |
if (!isDivTransformed) { | |
console.log('Making div smaller (transformed)'); | |
$('.BibleStudyNav .options').hide(); | |
$('.BibleStudyNav').removeClass('open').addClass('closed'); | |
// Add transformation | |
divElement.style.width = BibleStudyNavWidthSmall; | |
divElement.style.height = BibleStudyNavWidthSmall; | |
divElement.style.opacity = '0.3'; | |
// Update state | |
isDivTransformed = true; | |
} else { | |
console.log('Div is transformed, reverting to original'); | |
$('.BibleStudyNav .options').show(); | |
$('.BibleStudyNav').removeClass('closed').addClass('open'); | |
// Revert to original size and position | |
divElement.style.width = defaultSize + 'px'; | |
divElement.style.height = defaultHeight + 'px'; | |
divElement.style.opacity = '1'; | |
// Update state | |
isDivTransformed = false; | |
} | |
} | |
function resizeOnDblClick(divElement) { | |
// Define the default style | |
let defaultSize = divWidth // getComputedStyle(divElement).width; | |
let defaultHeight = divHeight; | |
let defaultPosition = getComputedStyle(divElement).left; | |
// Add event listener for double click | |
divElement.addEventListener('dblclick', function() { | |
// Check if div is in normal state | |
transformOrUntransform(divElement, defaultSize, defaultHeight); | |
}); | |
} | |
function addStyles(styles){ | |
$('<style>').text(styles).appendTo(document.head); | |
} | |
function generateRandomColor() { | |
let color = '#'; | |
for (let i = 0; i < 6; i++) { | |
const random = Math.random() * 16 | 0; | |
color += (i === 0 ? random & 0x7 : random).toString(16); | |
} | |
return color.toUpperCase(); | |
} | |
function intToHex(c) { | |
const hex = c.toString(16); | |
return hex.length == 1 ? "0" + hex : hex; | |
} | |
function padHex(c, shouldDarken) { | |
let value = Math.max(0, Math.min(255, c)); | |
if(shouldDarken) { | |
value = Math.floor(value * 0.7); | |
} else { | |
value = Math.floor(value + (255 - value) * 0.3); | |
} | |
return intToHex(value); | |
} | |
function generateColorScheme(shouldFitDarkScheme) { | |
const baseColor = generateRandomColor(); | |
const baseRed = parseInt(baseColor.substring(1,3), 16); | |
const baseGreen = parseInt(baseColor.substring(3,5), 16); | |
const baseBlue = parseInt(baseColor.substring(5,7), 16); | |
const colorOne = "#" + intToHex(255 - baseRed) + intToHex(255 - baseGreen) + intToHex(255 - baseBlue); | |
const colorTwo = "#" + padHex(baseRed, shouldFitDarkScheme) + padHex(baseGreen, shouldFitDarkScheme) + padHex(baseBlue, shouldFitDarkScheme); | |
const colorThree = "#" + padHex(baseRed, !shouldFitDarkScheme) + padHex(baseGreen, !shouldFitDarkScheme) + padHex(baseBlue, !shouldFitDarkScheme); | |
return [baseColor, colorOne, colorTwo, colorThree]; | |
} | |
function getRandomColor1(){ | |
const randomColor = Math.floor(Math.random()*16777215).toString(16); | |
return randomColor; | |
} | |
function getRandomColor2() { | |
let color = ""; | |
for (let i = 0; i < 3; i++) { | |
// Generate a random number between 0 and 255, | |
// but limit it from 0 to 175 (or 'AF' in hexadecimal) to ensure color is not too light and visible on white background | |
let part = Math.floor(Math.random() * 176).toString(16); | |
while (part.length < 2) { // Adding padding if number is single digit | |
part = "0" + part; | |
} | |
color += part; | |
} | |
return color; | |
} | |
function getRandomColor3(excludeColor) { | |
let color = "#"; | |
for (let i = 0; i < 3; i++) { | |
// Generate a random number between 0 and 255 inclusive, | |
// but limit it from 0 to 175 inclusive (or 'AF' in hexadecimal) to ensure the color is not too light and visible on white background | |
let part; | |
do { | |
part = Math.floor(Math.random() * 176).toString(16); | |
} while (excludeColor[i * 2 + 1] === part[0] || excludeColor[i * 2 + 1] === part[1]) | |
while (part.length < 2) { // Adding padding if number is single digit | |
part = "0" + part; | |
} | |
color += part; | |
} | |
return color; | |
} | |
function objectToString(obj) { | |
let result = ''; | |
for (let key in obj) { | |
if (obj.hasOwnProperty(key)) { | |
result += key + ': ' + obj[key] + "\n"; | |
} | |
} | |
// Remove the trailing ', ' for clean formatting | |
result = result.substring(0, result.length - 1); | |
return result; | |
} | |
GENERATE_BIBLE_STUDY = function(numPeople = 2, numVersesToSwitchAt = 5, mode = null, peopleColors = null){ | |
if (mode === null){ | |
mode = $('.modeDarkLight').val(); | |
} | |
if (peopleColors === true){ | |
setCookie("peopleColors", ""); | |
} | |
let o = bibleStudyOptionsSave(); | |
if (peopleColors === null && o.peopleColors && numPeople > 1 && numVersesToSwitchAt > 1){ | |
peopleColors = o.peopleColors; | |
if (!Array.isArray(peopleColors)){ | |
peopleColors = []; | |
} | |
}else{ | |
peopleColors = []; | |
} | |
if (Array.isArray(peopleColors) && peopleColors.length === 0){ | |
let lastColor = '#FFFFFF'; | |
let fourColors = generateColorScheme(mode === 'dark'); | |
console.log(fourColors); | |
for (let i = 0; i < numPeople; ++i){ | |
let c = fourColors[i]; | |
if (!c){ | |
c = getRandomColor(lastColor); | |
} | |
let rc = c; | |
peopleColors.push(rc); | |
lastColor = rc; | |
} | |
setCookie('peopleColors', JSON.stringify(peopleColors)); | |
} | |
let eachPersonReads = {}; | |
let verseCounter = 0; | |
let p = 0; | |
let color; | |
if (peopleColors[p]){ | |
color = peopleColors[p]; | |
}else{ | |
color = ''; | |
} | |
$('span.bV').each(function(){ | |
let person = "Person " + (p+1); | |
let $span = $(this); | |
let $ch = $span.find('.bCh'); | |
let $vs = $span.find('.bVN'); | |
if ($ch.length > 0){ | |
verseCounter = 1; | |
}else{ | |
verseCounter++; | |
} | |
if (typeof(eachPersonReads[person]) === "undefined"){ | |
eachPersonReads[person] = 1; | |
}else{ | |
eachPersonReads[person] = parseInt(eachPersonReads[person], 10) + 1; | |
} | |
$span.css('color', color).attr('title', person); | |
let doSwitch = verseCounter % numVersesToSwitchAt; | |
if (doSwitch === 0){ | |
p++; | |
if (p >= numPeople){ | |
p = 0; | |
} | |
if (color !== ''){ | |
color = peopleColors[p]; | |
} | |
} | |
}); | |
$('.BibleStudyNav .status').html(objectToString(eachPersonReads)); | |
} | |
function addDivWithScroll() { | |
// Create a new div element | |
var newDiv = document.createElement("div"); | |
// Apply styles to make it scroll with page | |
newDiv.style.position = "fixed"; | |
newDiv.style.top = "0"; // adjust position as per requirement | |
newDiv.style.left = "0"; // adjust position as per requirement | |
newDiv.style.width = divWidth + "px"; // adjust size as per requirement | |
newDiv.style.height = divHeight + "px"; // adjust size as per requirement | |
newDiv.style.backgroundColor = "#ddd"; // adjust color as per requirement | |
newDiv.style.overflow = "auto"; | |
// Add some content to the new div | |
newDiv.innerHTML = ` | |
<div class="transformDiv"> | |
<img class="bibleToggleOpenClose" title="Bible Study Helper: Toggle Open/Close" onclick="transformOrUntransform($(this).closest('div.BibleStudyNav').get(0))" style="width: 25px" src="" /> | |
</div> | |
<div class="options"> | |
<h3 title="Bible Study Nav by Jim K"><b>Bible Study Helper</b></h3> | |
<label>People | |
<select class="numPeople" onchange="setCookie('numPeople', this.value); GENERATE_BIBLE_STUDY(this.value, $(this).closest('div').find('.numVerses').val())"> | |
<option>2</option> | |
<option>3</option> | |
<option>4</option> | |
<option>5</option> | |
</select></label> | |
<label>Verses | |
<select class="numVerses" onchange="setCookie('numVerses', this.value); GENERATE_BIBLE_STUDY($(this).closest('div').find('.numPeople').val(), this.value)"> | |
<option>1</option> | |
<option>2</option> | |
<option>3</option> | |
<option>4</option> | |
<option selected>5</option> | |
<option>6</option> | |
<option>7</option> | |
<option>8</option> | |
<option>9</option> | |
<option>10</option> | |
<option>11</option> | |
<option>12</option> | |
<option>13</option> | |
<option>14</option> | |
<option>15</option> | |
</select> | |
</label> | |
<div class="status"></div> | |
<button type="button" onclick="GENERATE_BIBLE_STUDY($(this).closest('div').find('.numPeople').val(), $(this).closest('div').find('.numVerses').val(), null, true);">New Colors</button> | |
<button type="button" onclick="GENERATE_BIBLE_STUDY(0, 1);">No Colors</button> | |
<div> | |
<label>Mode | |
<select class="modeDarkLight"> | |
<option>light</option> | |
<option>dark</option> | |
</select></label> | |
</div> | |
</div> | |
`; | |
newDiv.className = 'BibleStudyNav'; | |
// Add the new div to the body of the current webpage | |
document.body.appendChild(newDiv); | |
$(newDiv).addClass('open'); | |
return newDiv; | |
} | |
function loadScript(url, callback) { | |
// Create a new script element | |
var script = document.createElement("script"); | |
script.type = "text/javascript"; | |
// If the browser supports the `onload` event (all browsers but IE), | |
// use it, else use `onreadystatechange` (for IE) | |
if (script.readyState) { | |
script.onreadystatechange = function() { | |
if (script.readyState === "loaded" || script.readyState === "complete") { | |
script.onreadystatechange = null; | |
// If a callback function was provided, execute it | |
if (callback) callback(); | |
} | |
}; | |
} else { | |
script.onload = function() { | |
// If a callback function was provided, execute it | |
if (callback) callback(); | |
}; | |
} | |
// Set the source of the script to the provided URL | |
script.src = url; | |
// Add the script to the `head` of the document | |
document.getElementsByTagName("head")[0].appendChild(script); | |
} | |
function getRandomColor4(excludeColor) { | |
let newColor; | |
if (typeof(chroma) === "undefined"){ | |
console.log('chroma not really defined'); | |
return getRandomColor3(excludeColor); | |
} | |
do { | |
newColor = chroma.random(); | |
} while(chroma.deltaE(newColor, excludeColor) < 80); | |
return newColor.hex(); | |
} | |
function getRandomColor(excludeColor) { | |
// Convert the excludeColor to Lab color space | |
var excludeColorLab = chroma(excludeColor).lab(); | |
var newColor; | |
// Generate colors until one is found that is different enough from excludeColor and dark enough | |
do { | |
newColor = chroma.random(); | |
// Convert the new color to Lab color space | |
var newColorLab = newColor.lab(); | |
// Get the Euclidean distance in Lab color space between the new color and excludeColor | |
var colorDistance = Math.sqrt(Math.pow(newColorLab[0] - excludeColorLab[0], 2) + | |
Math.pow(newColorLab[1] - excludeColorLab[1], 2) + | |
Math.pow(newColorLab[2] - excludeColorLab[2], 2)); | |
// Get the lightness of the new color | |
var lightness = newColorLab[0]; | |
} | |
// Keep going until we get a color that is both sufficiently different from excludeColor (colorDistance > 80) | |
// and also dark enough to contrast with a light background (lightness < 70) | |
while (colorDistance < 80 || lightness > 70); | |
return newColor.hex(); | |
} | |
getFromCookieOrDefaultAndSetCookie = function(key){ | |
let val = getCookie(key); | |
if (val !== ''){ | |
$('.'+key).val(val); | |
}else{ | |
val = $('.'+key).val(); | |
setCookie(key, val); | |
} | |
return val; | |
} | |
bibleStudyOptionsSave = function(){ | |
let numPeople = getFromCookieOrDefaultAndSetCookie('numPeople'); | |
let numVerses = getFromCookieOrDefaultAndSetCookie('numVerses'); | |
let mode = getFromCookieOrDefaultAndSetCookie('modeDarkLight'); | |
let peopleColors = getCookie('peopleColors'); | |
if (peopleColors !== ''){ | |
peopleColors = JSON.parse(peopleColors); | |
} | |
console.log("From Cookies: ", "numPeople", numPeople, "numVerses", numVerses, "mode", mode, "peopleColors", peopleColors); | |
return {numPeople: numPeople, numVerses: numVerses, mode: mode, peopleColors: peopleColors}; | |
} | |
$(function(){ | |
let div = addDivWithScroll(); | |
addStyles(` | |
.BibleStudyNav { | |
background: rgba(255,255,255,0.5); /* make the background 50% transparent white */ | |
box-shadow: 0px 0px 10px 5px rgba(0,0,0,0.5); /* add a box shadow */ | |
padding-left: 3px; | |
color: black; | |
} | |
.BibleStudyNav .transformDiv{ | |
float: left; | |
} | |
.BibleStudyNav.closed, .BibleStudyNav.open { | |
transition: width 0.5s, height 0.5s; | |
} | |
.BibleStudyNav.closed, .BibleStudyNav.closed .transformDiv{ | |
overflow: hidden !important; /* Hide scrollbars */ | |
} | |
.BibleStudyNav img.bibleToggleOpenClose { | |
cursor: pointer; | |
} | |
`); | |
$('.modeDarkLight').change(function(){ | |
let mode = $(this).val(); | |
switch(mode){ | |
case 'dark': | |
$('body').css('background-color', 'black').css('color', 'white'); | |
$('body').css('color-scheme', 'dark'); | |
$('.bMeatWrapper').css('background-color', '#5C4033'); | |
break; | |
case 'light': | |
$('body').css('background-color','').css('color',''); | |
$('body').css('color-scheme', ''); | |
$('.bMeatWrapper').css('background-color', ''); | |
break; | |
} | |
setCookie('modeDarkLight', mode); | |
}); | |
let o = bibleStudyOptionsSave(); | |
if (o.numPeople !== '' && o.numVerses !== ''){ | |
GENERATE_BIBLE_STUDY(o.numPeople, o.numVerses); | |
}else{ | |
GENERATE_BIBLE_STUDY(); | |
} | |
$('.modeDarkLight').change(); | |
//Taken out because not needed with big bible image //resizeOnDblClick(div); | |
transformOrUntransform(div); | |
}); | |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment