// ==UserScript==
// @name Quest Finder
// @namespace jasper.groupironmen.questfinder
// @match https://groupiron.men/*
// @grant GM_addStyle
// @run-at document-idle
// @version 2.0
// @author JasperV
// @description Find quests selected group members have yet to complete. Requires browser versions newer than ~June 2024
// @license MIT
// ==/UserScript==
// Possible Improvements:
// - turn all snake_case into camelCase (at least be consistent bro)
// - show result count
// - on hover over a quest, show each player's status
// - refactor code, especially into different files
// - checkbox for filtering out quests that player doesn't have required skills for
// - check skill requirements
// - check if completed required quests
// unique identifier used for css classes
const id = "questfinder"
//#region ENUMs
const QuestStatus = {
NOT_STARTED: 'Not Started',
IN_PROGRESS: 'In Progress',
COMPLETED: 'Completed'
}
/**
* Represents the difficulty levels of quests
* @typedef {Object} QuestDifficulty
* @property {number} rank sorting order
* @property {string} name The name of the difficulty level
* @property {string} icon The URL to the icon. Both used to get the difficulty from the site's quest list,
* and to show the icon again in the quest finder list.
*/
const QuestDifficulty = {
NOVICE: {
rank: 1,
name: "Novice",
icon: "/icons/3399-0.png"
},
INTERMEDIATE: {
rank: 2,
name: "Intermediate",
icon: "/icons/3400-0.png"
},
EXPERIENCED: {
rank: 3,
name: "Experienced",
icon: "/icons/3402-0.png"
},
MASTER: {
rank: 4,
name: "Master",
icon: "/icons/3403-0.png"
},
GRANDMASTER: {
rank: 5,
name: "Grandmaster",
icon: "/icons/3404-0.png"
}
}
const SortOption = {
ALPHABETICAL: 'Alphabetical',
DIFFICULTY: 'Difficulty'
}
//#endregion
//#region Models
class Player
{
constructor(name, panelElement)
{
this.name = name // unique identifier
this.panelElement = panelElement // reference to the player's panel element in the DOM. used for data-gathering
// keep track of the player's quests by status for easy filtering
this.quests = {
[QuestStatus.NOT_STARTED]: new Set(),
[QuestStatus.IN_PROGRESS]: new Set(),
[QuestStatus.COMPLETED]: new Set()
}
}
}
/**
* Represents a quest with its details and player statuses.
* @typedef {Object} Quest
* @property {string} name unique identifier: name of the quest
* @property {string} wiki_url The URL to the quest's runescape.wiki page
* @property {string, QuestStatus} playerStatuses key: player name, value: quest status for this quest
*/
const Quest = {
name: "",
wiki_url: "",
difficulty: QuestDifficulty.NOVICE,
}
//#endregion
/**
* Represents the filters the user has currently selected in the UI,
* which will be applied to the quest list
*/
const Filters = {
playersNotStarted: [], // players that must not have started a quest for it to be included
playersCompleted: [], // players that must have completed a quest for it to be included
sort: SortOption.ALPHABETICAL,
unfinished: false,
countInProgressAsCompleted: false,
removePlayer(player)
{
this.playersNotStarted = this.playersNotStarted.filter(p => p !== player)
this.playersCompleted = this.playersCompleted.filter(p => p !== player)
}
}
//#region quest management
const QuestManager = {
// key = quest name, value = Quest object
quests: new Map(),
/**
* @param {string} name
* @returns {Quest | null} the quest with the given name, or null if not found
*/
GetQuest(name)
{
return this.quests.get(name) || null
},
/**
* Register a new quest.
* First use GetQuest() to ensure doesnt already exist
* @param {string} name
* @param {string} wiki_url
* @param {QuestDifficulty} difficulty
* @returns {Quest} the newly created quest
*/
AddQuest(name, wiki_url, difficulty)
{
const quest = Object.create(Quest)
quest.name = name
quest.wiki_url = wiki_url
quest.difficulty = difficulty
quest.statuses = {}
this.quests.set(name, quest)
return quest
}
}
/**
* Get all quests that match the given criteria
* @param {Player[]} notStartedPlayers players that must not have started the quest
* @param {Player[]} completedPlayers players that must have completed the quest
* @param {boolean} countInProgressAsCompleted if true, in-progress quests are counted as finished
* @returns {Set<Quest>} a set of matching quests
*/
function getMatchingQuests(notStartedPlayers, completedPlayers, countInProgressAsCompleted)
{
if(completedPlayers.length === 0 && notStartedPlayers.length === 0)
return new Set()
const notStartedQuests = (player) =>
countInProgressAsCompleted
? player.quests[QuestStatus.NOT_STARTED]
: player.quests[QuestStatus.NOT_STARTED].union(player.quests[QuestStatus.IN_PROGRESS])
let result = null
for(const player of notStartedPlayers)
{
const playerNotStartedQuests = notStartedQuests(player)
if(result === null)
result = new Set(playerNotStartedQuests)
else
result = result.intersection(playerNotStartedQuests)
}
for(const player of completedPlayers)
{
const playerCompletedQuests = player.quests[QuestStatus.COMPLETED]
if(result === null)
result = new Set(playerCompletedQuests)
else
result = result.intersection(playerCompletedQuests)
}
return result
}
//#endregion
//#region DOM data-gathering
/**
* Set the completion status of each quest for a given player in the QuestManager.
* From the player's panel element in the DOM.
* @param {Player} player
*/
function UpdateQuestStatusesForPlayer(player)
{
const playerPanelElement = player.panelElement
// get the currently active tab so we can restore it later
const activeTab = playerPanelElement.querySelector('.player-panel__tab-active')
let openOldActiveTabAfter = (activeTab != null)
// if no tab was open, the quests tab must be closed after instead
let closeTabAfter = (activeTab === null)
// reset player's quest data
for(const status of Object.values(QuestStatus))
player.quests[status] = new Set()
// open Quests tab in UI (it's dynamically loaded by the site, so it must be clicked for us to see the data)
const questsButton = playerPanelElement.querySelector('button[data-component="player-quests"]')
if(questsButton != activeTab)
questsButton.click()
// quest tab is already active, so it can stay open
else
openOldActiveTabAfter = false
// get all quests from list
// and store in Player.quests the completion status of each
const questList = playerPanelElement.querySelector('.player-quests__list')
const questLinkElements = questList.getElementsByTagName('a')
for(const questLinkElement of questLinkElements)
{
const questElement = questLinkElement.querySelector('.player-quests__quest')
const questName = questElement.textContent.trim() // .replace(/\s+/g, ' ')
// the site gives different CSS classes
// depending on the status of the player's quest (completed, in progress or completed)
let questStatus
if(questElement.classList.contains('player-quests__not-started'))
questStatus = QuestStatus.NOT_STARTED
else if(questElement.classList.contains('player-quests__in-progress'))
questStatus = QuestStatus.IN_PROGRESS
else if(questElement.classList.contains('player-quests__finished'))
questStatus = QuestStatus.COMPLETED
// create a quest object with info about the quest
// or if one already exists for this quest, get that instead
let quest = QuestManager.GetQuest(questName)
if(!quest)
{
const wiki_url = questLinkElement.getAttribute('href')
const difficultyIcon = questElement.querySelector('.player-quests__difficulty-icon').getAttribute('src')
const difficulty = Object.values(QuestDifficulty).find(difficulty => difficulty.icon === difficultyIcon) || QuestDifficulty.NOVICE
quest = QuestManager.AddQuest(questName, wiki_url, difficulty)
}
// add quest to player's quest list
player.quests[questStatus].add(quest)
}
// restore old tab
if(openOldActiveTabAfter)
activeTab.click()
if(closeTabAfter)
questsButton.click()
}
/**
* get all players in the group from the site's DOM
* and create a Player object for them
* @returns {Player[]}
*/
function getPlayers()
{
const panels = document.querySelectorAll('player-panel')
const players = []
panels.forEach(panelElement => {
const name = panelElement.getAttribute('player-name')
const player = new Player(name, panelElement)
players.push(player)
})
return players
}
//#endregion
//#region Update Quest List
/**
* Updates the quest list UI based on the currently selected filters and players
* @param {Element} quests_container the element which the list will be added to
*/
function updateQuestList(quests_container)
{
const questsSet = getMatchingQuests(Filters.playersNotStarted, Filters.playersCompleted, Filters.countInProgressAsCompleted)
// helper to strip leading "A ", "An ", "The " for sorting (case-insensitive)
function stripLeadingArticle(name)
{
return name.replace(/^(a |an |the )/i, '').trim()
}
// sort alphabetically
let sortedQuestsArray = [...questsSet].sort((a, b) =>
stripLeadingArticle(a.name).localeCompare(stripLeadingArticle(b.name))
)
// apply other sort if selected
if(Filters.sort === SortOption.DIFFICULTY)
sortedQuestsArray = sortedQuestsArray.sort((a, b) => a.difficulty.rank - b.difficulty.rank)
populateQuestList(sortedQuestsArray, quests_container)
}
/**
* Creates a list of quests using the provided set and appends it to the container in the DOM
* @param {Quest[]} quests quests to put in container
* @param {HTMLElement} container container to place quests in
*/
function populateQuestList(quests, container)
{
// clear quest list
while(container.firstChild)
container.removeChild(container.firstChild)
// show informative text when no quests match
if(quests.length === 0)
{
const message = document.createElement("div")
if(Filters.playersNotStarted.length === 0 && Filters.playersCompleted.length === 0)
message.textContent = "Select players to compare"
else
message.textContent = "No matching quests found"
container.appendChild(message)
}
// create a quest entry in the list for each given quest
for(const quest of quests)
{
const link = document.createElement("a")
link.target = "_blank"
link.href = quest.wiki_url
const element = document.createElement("div")
element.classList.add(`${id}__quest__item`)//, `${id}__quest__${quest.status.toLowerCase()}`)
link.appendChild(element)
// Difficulty icon
const difficultyImage = document.createElement("img")
difficultyImage.classList.add(`${id}__quest__difficulty-image`)
difficultyImage.src = quest.difficulty.icon
difficultyImage.alt = quest.difficulty.name
difficultyImage.title = quest.difficulty.name
element.appendChild(difficultyImage)
const nameSpan = document.createElement("span")
nameSpan.textContent = quest.name
element.appendChild(nameSpan)
container.appendChild(link)
}
}
//#endregion
//#region UI Panel Creation
/**
* @param {Player[]} players
*/
function createUIPanel(players)
{
const container = document.createElement("div")
container.id = id
container.classList.add(`${id}__window`, `${id}-dark-background`, `${id}-border`)
// these elements are added to body later, but created here so they can be referenced
const body = document.createElement("div")
const quests_container = document.createElement("div") // container for list of matching quests
// header
const header = createHeader(players, body, quests_container)
container.appendChild(header)
body.classList.add(`${id}__body`)
container.appendChild(body)
// title
const playersTitle = document.createElement("h4")
playersTitle.textContent = "Select players to compare"
body.appendChild(playersTitle)
// checkboxes to toggle players
const players_container = createPlayerSelection(players, quests_container)
body.appendChild(players_container)
// title
const questsTitle = document.createElement("h4")
questsTitle.textContent = "Matching quests"
body.appendChild(questsTitle)
// sort by
const sortContainer = createSortSelect(quests_container)
body.appendChild(sortContainer)
// exclude in-progress quests
const excludeCheckbox = createExcludeInProgressCheckbox(quests_container)
body.appendChild(excludeCheckbox)
// list of matching quests
quests_container.classList.add(`${id}__quest-list`)
body.appendChild(quests_container)
document.body.appendChild(container)
updateQuestList(quests_container)
makeDraggable(container, header)
}
/**
* Creates the header for the quest finder UI
* @param {Player[]} players
* @param {Element} body body to hide when the minimise button is clicked
* @param {Element} quests_container container for the list of matching quests
*/
function createHeader(players, body, quests_container)
{
const header = document.createElement("div")
header.classList.add(`${id}__header`)
const title = document.createElement("h1")
title.classList.add(`${id}__header__title`)
title.textContent = "Quest Finder"
header.appendChild(title)
// manual refresh button
const refreshButton = document.createElement("button")
refreshButton.classList.add(`${id}__header__button`, 'men-button')
refreshButton.textContent = "↻"
refreshButton.title = "Update quest data"
refreshButton.addEventListener("click", () =>
{
players.forEach(player => {
UpdateQuestStatusesForPlayer(player)
})
updateQuestList(quests_container)
})
header.appendChild(refreshButton)
// minimise button
const minimiseButton = document.createElement("button")
minimiseButton.classList.add(`${id}__header__button`, 'men-button')
minimiseButton.textContent = "−"
minimiseButton.addEventListener("click", () =>
{
body.classList.toggle("hidden")
if(body.classList.contains("hidden"))
{
minimiseButton.textContent = "+"
refreshButton.classList.add("hidden")
}
else
{
minimiseButton.textContent = "−"
refreshButton.classList.remove("hidden")
}
})
header.appendChild(minimiseButton)
return header
}
/**
* Creates a container for player selection checkboxes
* @param {Player[]} players
* @param {Element} quests_container container element which the quest list will be added to
*/
function createPlayerSelection(players, quests_container)
{
const players_container = document.createElement("div")
players_container.classList.add(`${id}__player-list`)
for (let i = 0; i < players.length; i++)
{
const player = players[i]
const playerContainer = document.createElement("div")
playerContainer.classList.add(`${id}__player`)
// checkbox
const checkbox = document.createElement("input")
checkbox.type = "checkbox"
checkbox.id = `${id}__player-checkbox__${i}`
checkbox.value = player.name
checkbox.checked = false
playerContainer.appendChild(checkbox)
// label
const label = document.createElement("label")
label.textContent = player.name
label.htmlFor = checkbox.id
playerContainer.appendChild(label)
// dropdown with {finished, not started}
const filterSelect = document.createElement("select")
filterSelect.classList.add("hidden")
playerContainer.appendChild(filterSelect)
const notStartedOption = document.createElement("option")
notStartedOption.value = "not_started"
notStartedOption.textContent = "Not Started"
filterSelect.appendChild(notStartedOption)
const finishedOption = document.createElement("option")
finishedOption.value = "finished"
finishedOption.textContent = "Finished"
filterSelect.appendChild(finishedOption)
// on dropdown changed, set filter & update the quest list
filterSelect.addEventListener("change", () =>
{
Filters.removePlayer(player)
if(checkbox.checked)
{
if(filterSelect.value === "finished")
Filters.playersCompleted.push(player)
else
Filters.playersNotStarted.push(player)
}
else
Filters.removePlayer(player)
updateQuestList(quests_container)
})
// on checkbox change, toggle status checkbox visibility
checkbox.addEventListener("change", () =>
{
if(!checkbox.checked)
filterSelect.classList.add("hidden")
else
filterSelect.classList.remove("hidden")
// trigger filterSelect change event,
// to let it handle setting the filter and updating the quest list
filterSelect.dispatchEvent(new Event("change"))
})
players_container.appendChild(playerContainer)
}
return players_container
}
function createSortSelect(quests_container)
{
const sortContainer = document.createElement("div")
sortContainer.classList.add(`${id}__sort`)
const sortLabel = document.createElement("label")
sortLabel.htmlFor = `${id}__sort-select`
sortLabel.textContent = "Sort by:"
sortContainer.appendChild(sortLabel)
const sortSelect = document.createElement("select")
sortSelect.id = `${id}__sort-select`
sortSelect.classList.add(`${id}__sort-select`)
sortContainer.appendChild(sortSelect)
// add options based on SortOption enum
for(const [key, value] of Object.entries(SortOption))
{
const optionElement = document.createElement("option")
optionElement.value = key
optionElement.textContent = value
sortSelect.appendChild(optionElement)
}
// update the filter and automatically refresh the quest list when the user changes selection
sortSelect.addEventListener("change", () =>
{
Filters.sort = SortOption[sortSelect.value]
updateQuestList(quests_container)
})
return sortContainer
}
/**
* Creates a checkbox to exclude in-progress quests from the quest list
* @param {Element} quests_container
* @returns {Element}
*/
function createExcludeInProgressCheckbox(quests_container)
{
const container = document.createElement("div")
container.classList.add(`${id}__exclude-in-progress`)
const checkbox = document.createElement("input")
checkbox.type = "checkbox"
checkbox.id = `${id}__exclude-in-progress__checkbox`
checkbox.checked = false
checkbox.addEventListener("change", () =>
{
Filters.countInProgressAsCompleted = checkbox.checked
updateQuestList(quests_container)
})
container.appendChild(checkbox)
const label = document.createElement("label")
label.textContent = "Consider In-Progress quests as Finished"
label.htmlFor = checkbox.id
container.appendChild(label)
return container
}
/**
* Makes an element draggable
* @param {Element} draggedElement The element to move when dragging
* @param {Element} handle The element that the user can mousedown on to drag the element
*/
function makeDraggable(draggedElement, handle)
{
let isDragging = false
let offsetX = 0
let offsetY = 0
handle.addEventListener('mousedown', (e) =>
{
isDragging = true
offsetX = e.clientX - draggedElement.getBoundingClientRect().left
offsetY = e.clientY - draggedElement.getBoundingClientRect().top
document.body.style.userSelect = 'none' // prevent text selection while dragging
})
document.addEventListener('mousemove', (e) =>
{
if(isDragging)
draggedElement.style.inset = `${e.clientY - offsetY}px auto auto ${e.clientX - offsetX}px`;
})
document.addEventListener('mouseup', () =>
{
isDragging = false
document.body.style.userSelect = '' // re-enable text selection
})
}
function loadCSS()
{
GM_addStyle(`
.hidden {
display: none !important;
}
.${id}-dark-background {
background: url() !important;
}
.${id}-light-background {
background-color: #424343;
/* background: url() !important; */
}
.${id}-border {
border-image: url() 32 32/32px/4px round;
}
#${id} h4 {
margin-top: 8px;
margin-bottom: 4px;
color: var(--orange);
font-size: 20px;
font-weight: 400;
}
.${id}__window {
position: fixed;
z-index: 9999;
bottom: 16px;
right: 16px;
width: 320px;
padding: 8px;
}
.${id}__header {
display: flex;
align-items: center;
}
.${id}__header__title {
font-size: 1em;
flex-grow: 1;
margin: 0;
}
.${id}__header__button {
margin-left: 8px;
}
.${id}__player-list {
display: flex;
flex-direction: column;
}
.${id}__player {
padding-top: 4px;
padding-bottom: 4px;
display: flex;
justify-content: space-between;
}
.${id}__sort-select {
margin-left: 4px;
margin-bottom: 8px;
}
.${id}__exclude-in-progress {
margin-bottom: 8px;
}
.${id}__quest-list {
overflow: hidden;
overflow-y: scroll;
max-height: 50vh;
}
.${id}__quest__item:hover {
background: rgba(255,255,255,.1);
}
.${id}__quest__difficulty-image {
margin-right: 4px;
}
`)
}
//#endregion
//#region Initialisation
let QUEST_FINDER_INITIALISED = false
function init()
{
if(QUEST_FINDER_INITIALISED) return
QUEST_FINDER_INITIALISED = true
const players = getPlayers()
players.forEach(player => {
UpdateQuestStatusesForPlayer(player)
})
createUIPanel(players)
loadCSS()
}
// wait until site is finished loading before initialising
const loadingScreen = document.getElementsByTagName("loading-screen")[0]
const observer = new MutationObserver(() =>
{
// loading is done when the <loading-screen> gets style="display:none"
if(loadingScreen.style.display === 'none')
{
// make sure it's the https://groupiron.men/group that's loaded and not another
if(window.location.href.startsWith("https://groupiron.men/group"))
{
init()
observer.disconnect()
}
}
})
if(loadingScreen.style.display === 'none')
init()
else
observer.observe(loadingScreen, { attributes: true })
//#endregion