Skip to content

Instantly share code, notes, and snippets.

@splintor
Last active May 8, 2022 08:38
Show Gist options
  • Save splintor/adccd10959dd83f76741ee7536396a96 to your computer and use it in GitHub Desktop.
Save splintor/adccd10959dd83f76741ee7536396a96 to your computer and use it in GitHub Desktop.
GitHub - Continue in SSO
// ==UserScript==
// @name JIRA Meetinator
// @namespace http://shmulikf.wix.com/
// @version 0.15
// @description Improve team meeting experience in Jira: Enable randomly selecting a team member, set time limit for each team member, celebrate completion.
// @author Shmulik Flint
// @match https://jira.wixpress.com/secure/RapidBoard.jspa*
// @icon https://jira.wixpress.com/s/-8bjowq/813009/g345zt/_/images/fav-jsw.png
// @downloadURL https://gist.githubusercontent.com/splintor/f10c20926335324c6ff8e007439ed64c/raw/JIRA-Meetinator.user.js
// @updateURL https://gist.githubusercontent.com/splintor/f10c20926335324c6ff8e007439ed64c/raw/JIRA-Meetinator.user.js
// @require https://cdn.jsdelivr.net/npm/canvas-confetti@1.5.1/dist/confetti.browser.min.js
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_addStyle
// ==/UserScript==
/* globals confetti */
// TODO: Add sounds: When time is about to end. When time ends, when all members finished
(function() {
'use strict'
const memberHeadingSelector = '.ghx-heading'
const closeMemberSelector = '.ghx-swimlane.ghx-closed'
const openMemberSelector = '.ghx-swimlane:not(.ghx-closed)'
const getMemberHeadings = () => document.querySelectorAll(memberHeadingSelector)
const getMembersProgress = () => document.querySelectorAll(memberHeadingSelector + ' progress')
const getOpenMember = () => document.querySelector(openMemberSelector)
const getContainer = () => document.querySelector('#ghx-pool')
const getMeetinator = () => document.querySelector('#Meetinator')
const markMemberAsVisited = (memberHeading) => memberHeading.classList.add('visitedMember')
const markMemberAsNotVisited = (memberHeading) => memberHeading.classList.remove('visitedMember')
const getMemberName = (memberHeading) => memberHeading.querySelector('.js-expander + span').innerText
function getClickedMembers() {
const clickedMembersJSON = GM_getValue('clicked_members', '{}')
try {
return JSON.parse(clickedMembersJSON)
} catch (e) {
return {}
}
}
let clickedMembers = getClickedMembers()
function updateClickedMembers(updates) {
clickedMembers = { ...clickedMembers, ...updates }
console.log('clickedMembers', clickedMembers)
GM_setValue('clicked_members', JSON.stringify(clickedMembers))
}
function resetClickedMembers() {
GM_setValue('clicked_members', '{}')
clickedMembers = {}
}
const wasConfettiDisplayed = () => GM_getValue('confetti_displayed', '') === 'true'
const setConfettiDisplayed = (value) => GM_setValue('confetti_displayed', String(value))
let timerTime = Number(GM_getValue('timer_time', '300'))
function getMembers() {
const members = [...getMemberHeadings()]
const relevantMembers = members.filter(m => !clickedMembers[getMemberName(m)])
return { members, relevantMembers }
}
function onClick(e) {
if (e.target.type === 'submit') {
const { relevantMembers } = getMembers()
const memberToClick = relevantMembers[Math.floor(Math.random() * relevantMembers.length)]
updateClickedMembers({ [getMemberName(memberToClick)]: new Date().toISOString() })
markMemberAsVisited(memberToClick)
updateButtons()
memberToClick.click()
} else if (e.target.type === 'reset') {
resetClickedMembers()
setConfettiDisplayed(false)
const { members } = getMembers()
members.forEach(markMemberAsNotVisited)
updateButtons()
}
}
function buildUI() {
const ui = document.createElement('div')
ui.id = 'Meetinator'
ui.innerHTML =
'<div class="logo" tooltip="abc">' +
'<img width="50px" src=""/>' +
'<span>Meetinator™</span>' +
'<div class="logoTooltip">' +
'<span>Created by <a href="https://wix.slack.com/team/UD5USBS5R">Shmulik Flint</a> ' +
'<a href="https://gist.github.com/splintor/f10c20926335324c6ff8e007439ed64c" rel="noreferrer"><img width="14px" src="https://github.githubassets.com/favicon.ico"/></a>' +
'<div class="logoAttribution"><a href="https://www.flaticon.com/free-icons/joker">Joker logo source</a></div>' +
'</span>' +
'</div>' +
'</div><div class="buttons">' +
'<button type="submit"></button>' +
'<button type="reset" style="margin-left: 10px">Reset</button>' +
'</div>'
getContainer().appendChild(ui)
ui.addEventListener('click', onClick)
GM_addStyle('#Meetinator { position: absolute; top: 0; margin-top: 90px; margin-left: 400px; z-index: 1000; }')
GM_addStyle('.logo { display: flex; align-items: center; justify-content: center; margin-bottom: 5px; cursor: default; }')
GM_addStyle('.logo > span { margin-inline-start: 8px; font-size: 22px; text-shadow: 2px 2px 5px lightseagreen; }')
GM_addStyle('.logo img { margin-block-start: -4px; }')
GM_addStyle('.buttons { display: flex; }')
GM_addStyle('.logoTooltip { display: none; }')
GM_addStyle('.logo:hover .logoTooltip { display: block; position: absolute; top: 40px; left: 110px; background: lightyellow; border: 1px solid gray; padding: 7px; min-width: 181px; }')
GM_addStyle('.logoAttribution { font-size: 11px; margin-block-start: 4px; }')
GM_addStyle('.memberStatus { display: none; }')
GM_addStyle(openMemberSelector + ' ~ #Meetinator { display: none !important; }')
GM_addStyle(memberHeadingSelector + ':not(.visitedMember) > .js-expander + span { font-weight: bold; }')
GM_addStyle(openMemberSelector + ' .visitedMember > .js-expander + span { background-color: gold; }')
GM_addStyle(openMemberSelector + ' .visitedMember .memberStatus { display: inline; }')
GM_addStyle('.unopenLink { display: none; }')
GM_addStyle('.unopenLink a { margin-inline-start: 7px; font-size: 12px; color: darkviolet; opacity: .4; }')
GM_addStyle(closeMemberSelector + ':hover .visitedMember .unopenLink { display: inline; }')
GM_addStyle('.ghx-swimlane .ghx-heading .memberStatus.editMode { display: none; }')
GM_addStyle('.ghx-swimlane .ghx-heading .memberStatus:not(.editMode) ~ .timerSettings { display: none; }')
GM_addStyle('#Meetinator button { border-radius: 6px; padding: 10px 18px; outline: 0; text-align: center; box-shadow: rgba(0, 0, 0, 0.1) 0 2px 4px; }')
GM_addStyle('#Meetinator button :hover { box-shadow: rgba(253, 76, 0, 0.5) 0 3px 8px; }');
GM_addStyle('progress { vertical-align: middle; border: none; -webkit-appearance: none; -moz-appearance: none; appearance: none; }')
GM_addStyle('progress[data-expired="true"]::-webkit-progress-value { background-color: red; }')
GM_addStyle('progress[data-expired="true"]::-moz-progress-bar { background-color: red; }')
GM_addStyle('.timeCounter { min-width: 40px; display: inline-block; font-weight: bold; }')
GM_addStyle('.timeTracker { cursor: pointer; }')
GM_addStyle('.timerSettings { background-color: cornsilk; padding: 5px; }')
GM_addStyle('.timerSettings input { width: 70px; text-align: end; }')
GM_addStyle('.timerSettings button { margin-inline: 5px; }')
GM_addStyle('.timerSettings a { color: black; }')
const { members, relevantMembers } = getMembers()
members.forEach(markMemberAsVisited)
relevantMembers.forEach(markMemberAsNotVisited)
updateButtons()
addMembersStatus()
}
function updateButtons() {
const ui = document.querySelector('#Meetinator')
const submitButton = ui.querySelector('button[type="submit"]')
const resetButton = submitButton.nextSibling
const { members, relevantMembers } = getMembers()
if (relevantMembers.length === members.length) {
submitButton.innerText = 'Open a random member'
resetButton.style.display = 'none'
} else {
submitButton.innerText = 'Open next random member (' + relevantMembers.length + ' left)'
resetButton.style.display = 'block'
}
submitButton.disabled = relevantMembers.length === 0
}
function addMembersStatus() {
getMemberHeadings().forEach(member => {
const memberName = getMemberName(member)
member.insertAdjacentHTML('beforeend',
'<span>' +
'<span class="memberStatus">' +
'<span class="timeTracker">' +
'<span>Time left: <span class="timeCounter" id="' + memberName + 'Timer"></span></span>' +
'<progress id="' + memberName+ 'Progress" max="' + timerTime + '" value="0"></progress>' +
'</span>' +
'</span>' +
'<span class="unopenLink">' +
'<a href="#">Mark as not opened</a>' +
'</span>' +
'<span class="timerSettings"><form>' +
'Timer duration in seconds: <input type="number">' +
'<button type="submit">Save</button>' +
'<a href="#">X</a>' +
'</form></span>' +
'</span>')
member.querySelector('.unopenLink a').addEventListener('click', e => {
e.preventDefault()
e.stopPropagation()
updateClickedMembers({ [getMemberName(member)]: undefined })
markMemberAsNotVisited(member)
updateButtons()
setConfettiDisplayed(false)
})
const memberStatus = member.querySelector('.memberStatus')
const timeInput = member.querySelector('input')
const button = member.querySelector('.timerSettings button')
member.querySelector('.timeTracker').addEventListener('click', e => {
e.preventDefault()
memberStatus.classList.add('editMode')
timeInput.value = timerTime
})
timeInput.addEventListener('input', e => {
button.disabled = !timeInput.value
})
button.addEventListener('click', e => {
e.preventDefault()
timerTime = Number(timeInput.value)
GM_setValue('timer_time', timeInput.value)
getMembersProgress().forEach(progress => { progress.max = timerTime })
memberStatus.classList.remove('editMode')
})
member.querySelector('.timerSettings a').addEventListener('click', e => {
e.preventDefault()
memberStatus.classList.remove('editMode')
})
})
}
function getTimerString(seconds) {
if (seconds <= 0) {
return '0:00'
}
const m = Math.floor(seconds / 60)
const s = seconds % 60
return m + ':' + String(s).padStart(2, '0')
}
function updateMembersStatus() {
const { members } = getMembers()
members.forEach(member => {
const memberName = getMemberName(member)
const memberStartTime = clickedMembers[memberName]
if (!memberStartTime) {
return
}
const elapsedSeconds = Math.floor((new Date() - new Date(memberStartTime)) / 1000)
const progress = document.getElementById(memberName + 'Progress')
progress.value = elapsedSeconds
progress.setAttribute('data-expired', elapsedSeconds > timerTime)
document.getElementById(memberName + 'Timer').innerText = getTimerString(timerTime - elapsedSeconds)
})
}
function showConfetti() {
const confettiParticleXount = 400
const fireConfetti = (particleRatio, opts) => confetti({
origin: { y: 0.7 },
particleCount: Math.floor(confettiParticleXount * particleRatio),
...opts
})
fireConfetti(0.25, {
spread: 26,
startVelocity: 55,
})
fireConfetti(0.2, {
spread: 60,
})
fireConfetti(0.35, {
spread: 100,
decay: 0.91,
scalar: 0.8
})
fireConfetti(0.1, {
spread: 120,
startVelocity: 25,
decay: 0.92,
scalar: 1.2
})
fireConfetti(0.1, {
spread: 120,
startVelocity: 45,
})
}
function showConfettiOnCompletion() {
if (wasConfettiDisplayed() || getOpenMember() || getMembers().relevantMembers.length > 0) {
return
}
setConfettiDisplayed(true)
showConfetti()
}
function renderMeetinator() {
if (getContainer()) {
if (!getMeetinator()) {
buildUI()
}
updateMembersStatus()
showConfettiOnCompletion()
}
}
setInterval(renderMeetinator, 500)
})()
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment