Quest Finder

Find quests selected group members have yet to complete. Requires browser versions newer than ~June 2024

// ==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