GitHub RepoNotesHelper

Have you forgotten what the repository you starred is for? This is a script that can add notes to the repository you starred, and your notes can be uploaded and downloaded to GitHub Gist

目前為 2025-04-21 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         GitHub RepoNotesHelper
// @name:zh-CN   GitHub 星标仓库备注助手
// @description  Have you forgotten what the repository you starred is for? This is a script that can add notes to the repository you starred, and your notes can be uploaded and downloaded to GitHub Gist
// @description:zh-CN 曾经 Star 过的仓库是不是忘记了它们的用途? 这是一个可以给你收藏过的仓库添加备注的的脚本,并且你备注的信息可以上传下载到 GitHub Gist
// @author       malagebidi,人民的勤务员 <[email protected]>
// @namespace    https://github.com/ChinaGodMan/UserScripts
// @supportURL   https://github.com/ChinaGodMan/UserScripts/issues
// @homepageURL  https://github.com/ChinaGodMan/UserScripts
// @license      MIT
// @match        https://github.com/*
// @icon              
// @compatible   chrome
// @compatible   firefox
// @compatible   edge
// @compatible   opera
// @compatible   safari
// @compatible   kiwi
// @compatible   qq
// @compatible   via
// @compatible   brave
// @version      2025.04.21.1652
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_deleteValue
// @grant        GM_addStyle
// @grant              GM_xmlhttpRequest
// @grant              GM_registerMenuCommand
// @created      2025-04-21 16:52:52
// @modified     2025-04-21 16:52:52
// ==/UserScript==
/**
 * File: github-starred-repo-note.user.js
 * Project: UserScripts
 * File Created: 2025/04/21,Monday 16:52:52
 * Author: malagebidi<https://greasyfork.org/zh-CN/users/314803>,人民的勤务员@ChinaGodMan ([email protected])
 * -----
 * Last Modified: 2025/04/21,Monday 18:32:01
 * Modified By: 人民的勤务员@ChinaGodMan ([email protected])
 * -----
 * License: MIT License
 * Copyright © 2024 - 2025 ChinaGodMan,Inc
 */

(async function () {
    'use strict'
    // --- Configuration ---
    var GITHUB_PAT_TOKEN = GM_getValue('GITHUB_PAT_TOKEN', '')
    const userLang =
        (navigator.languages && navigator.languages[0]) ||
        navigator.language ||
        'en'
    const translations = {
        en: {
            NOTE_PLACEHOLDER: 'Enter your note...',
            ADD_BUTTON_TEXT: 'Add Note',
            EDIT_BUTTON_TEXT: 'Edit Note',
            SAVE_BUTTON_TEXT: 'Save',
            CANCEL_BUTTON_TEXT: 'Cancel',
            DELETE_BUTTON_TEXT: 'Delete',
            DOWNLOAD_BUTTON_TEXT: 'Download notes',
            UPLOAD_BUTTON_TEXT: 'Upload notes',
            NO_TOKEN_ALERT_TEXT: 'Please enter your GitHub token',
            DOWNLOAD_GIST_SUCCESS: 'Your cloud backup notes have been restored, please refresh the current page. ',
            DOWNLOAD_GIST_FAILURE: 'Your cloud backup notes failed to be restored, please check whether the GitHub Token is correct. ',
            UPLOAD_GIST_SUCCESS: 'Your cloud backup notes have been saved',
            UPLOAD_GIST_FAILURE: 'Your cloud backup notes failed to be saved, please check whether the GitHub Token is correct. ',
            DELETE_CONFIRM: 'Are you sure you want to delete the note for \"{repoFullName}\"?'

        },
        'zh-CN,zh,zh-SG': {
            NOTE_PLACEHOLDER: '输入备注...',
            ADD_BUTTON_TEXT: '备注',
            EDIT_BUTTON_TEXT: '编辑备注',
            SAVE_BUTTON_TEXT: '保存',
            CANCEL_BUTTON_TEXT: '取消',
            DELETE_BUTTON_TEXT: '删除',
            DOWNLOAD_BUTTON_TEXT: '下载备注',
            UPLOAD_BUTTON_TEXT: '上传备注',
            NO_TOKEN_ALERT_TEXT: '请先输入GitHub Token',
            DOWNLOAD_GIST_SUCCESS: '你的云备份笔记已被恢复,请刷新当前页面.',
            DOWNLOAD_GIST_FAILURE: '你的云备份笔记恢复失败,请检查GitHub Token是否正确.',
            UPLOAD_GIST_SUCCESS: '你的云备份笔记已被保存',
            UPLOAD_GIST_FAILURE: '你的云备份笔记保存失败,请检查GitHub Token是否正确.',
            DELETE_CONFIRM: '你确定要删除\"{repoFullName}\"仓库的备注嘛?'

        },
        'zh-TW,zh-HK,zh-MO': {
            NOTE_PLACEHOLDER: '輸入您的筆記...',
            ADD_BUTTON_TEXT: '添加筆記',
            EDIT_BUTTON_TEXT: '編輯筆記',
            SAVE_BUTTON_TEXT: '保存',
            CANCEL_BUTTON_TEXT: '取消',
            DELETE_BUTTON_TEXT: '刪除'
        },
        vi: {
            NOTE_PLACEHOLDER: 'Nhập ghi chú của bạn...',
            ADD_BUTTON_TEXT: 'Thêm ghi chú',
            EDIT_BUTTON_TEXT: 'Chỉnh sửa ghi chú',
            SAVE_BUTTON_TEXT: 'Lưu',
            CANCEL_BUTTON_TEXT: 'Hủy bỏ',
            DELETE_BUTTON_TEXT: 'Xóa'
        },
        ja: {
            NOTE_PLACEHOLDER: 'メモを入力してください...',
            ADD_BUTTON_TEXT: 'メモを追加',
            EDIT_BUTTON_TEXT: 'メモを編集',
            SAVE_BUTTON_TEXT: '保存',
            CANCEL_BUTTON_TEXT: 'キャンセル',
            DELETE_BUTTON_TEXT: '削除'
        },
        ko: {
            NOTE_PLACEHOLDER: '메모를 입력하세요...',
            ADD_BUTTON_TEXT: '메모 추가',
            EDIT_BUTTON_TEXT: '메모 편집',
            SAVE_BUTTON_TEXT: '저장',
            CANCEL_BUTTON_TEXT: '취소',
            DELETE_BUTTON_TEXT: '삭제'
        }

    }
    const getTranslations = (lang) => {
        for (const key in translations) {
            if (key === lang || key.split(',').includes(lang)) {
                return translations[key]
            }
        }
        return translations['en']
    }
    const translate = new Proxy(
        function (key) {
            const lang = userLang
            const strings = getTranslations(lang)
            return strings[key] || translations['en'][key]
        },
        {
            get(target, prop) {
                const lang = userLang
                const strings = getTranslations(lang)
                return strings[prop] || translations['en'][prop]
            }
        }
    )

    // --- Styles ---
    GM_addStyle(`
        .ghsn-container {
            padding-right: var(--base-size-24, 24px) !important;
            color: var(--fgColor-muted, var(--color-fg-muted)) !important;
            width: 74.99999997%;
        }
        .ghsn-display {
            font-style: italic;
            border: var(--borderWidth-thin) solid var(--borderColor-default, var(--color-border-default, #d2dff0));
            border-radius: 100px;
            padding: 2.5px 5px;
            white-space: nowrap;
            overflow: hidden;
            text-overflow: ellipsis;
            display: block;
            max-width: fit-content;
        }
        .ghsn-textarea {
            width: 100%;
            min-height: 60px;
            margin-bottom: 5px;
            padding: 5px;
            border: 1px solid var(--color-border-default);
            border-radius: 3px;
            background-color: var(--color-canvas-default);
            color: var(--color-fg-default);
            box-sizing: border-box;
        }
        .ghsn-buttons button {
            margin-right: 5px;
            padding: 3px 8px;
            font-size: 0.9em;
            cursor: pointer;
            border-radius: 4px;
            border: 1px solid var(--color-border-muted);
        }
        .ghsn-buttons button.ghsn-save {
            background-color: var(--color-btn-primary-bg);
            color: var(--color-btn-primary-text);
            border-color: var(--color-btn-primary-border);
        }
        .ghsn-buttons button.ghsn-delete {
            background-color: var(--color-btn-danger-bg);
            color: var(--color-btn-danger-text);
            border-color: var(--color-btn-danger-border);
        }
        .ghsn-buttons button.ghsn-cancel {
            background-color: var(--color-btn-bg);
            color: var(--color-btn-text);
        }
        .ghsn-buttons button:hover {
            filter: brightness(1.1);
        }
        .ghsn-hidden {
            display: none !important;
        }
        .ghsn-note-btn {
            margin-left: 16px;
            color: var(--fgColor-muted);
            cursor: pointer;
            text-decoration: none;
        }
        .ghsn-note-btn:hover {
            color: var(--fgColor-accent) !important;
            -webkit-text-decoration: none;
            text-decoration: none;
        }
        .ghsn-note-btn svg {
            margin-right: 4px;
        }
    `)

    /* ------------------------------- GITHUB GIST ------------------------------ */
    const GistManager = {
        githubToken: null,
        description: null,

        init: function (token, description) {
            this.githubToken = token
            this.description = description || 'Default Gist Description'
        },
        updateToken: function (newToken) {
            this.githubToken = newToken
        },
        // 查找 Gist 根据描述
        findGistByDescription: function (callback) {
            const url = 'https://api.github.com/gists'
            GM_xmlhttpRequest({
                method: 'GET',
                url: url,
                headers: {
                    'Authorization': `token ${this.githubToken}`,
                    'Content-Type': 'application/json'
                },
                onload: function (response) {
                    if (response.status === 200) {
                        const gists = JSON.parse(response.responseText)
                        for (let gist of gists) {
                            if (gist.description === GistManager.description) {
                                console.log('找到匹配的 Gist:', gist.html_url)
                                return callback(gist.id)
                            }
                        }
                        callback(null)
                    } else {
                        console.error('获取 Gist 列表失败:', response.responseText)
                        callback(null)
                    }
                }
            })
        },

        // 上传或更新 Gist
        uploadToGist: function (filename, content) {

            this.findGistByDescription((gistId) => {
                if (gistId) {
                    this.updateGist(gistId, filename, content)
                } else {
                    this.createGist(filename, content)
                }
            })
        },

        // 创建新的 Gist
        createGist: function (filename, content) {
            const url = 'https://api.github.com/gists'
            const data = {
                'description': this.description,
                'public': false,
                'files': {
                    [filename]: {
                        'content': content
                    }
                }
            }
            GM_xmlhttpRequest({
                method: 'POST',
                url: url,
                headers: {
                    'Authorization': `token ${this.githubToken}`,
                    'Content-Type': 'application/json'
                },
                data: JSON.stringify(data),
                onload: function (response) {
                    if (response.status === 201) {
                        const responseData = JSON.parse(response.responseText)
                        console.log('Gist 创建成功:', responseData.html_url)
                    } else {
                        console.error('Gist 创建失败:', response.responseText)
                    }
                }
            })
        },
        downloadGistAsJson: function (filename) {
            this.findGistByDescription((gistId) => {
                if (!gistId) {
                    console.error('未找到匹配的 Gist')
                    return
                }

                const url = `https://api.github.com/gists/${gistId}`
                GM_xmlhttpRequest({
                    method: 'GET',
                    url: url,
                    headers: {
                        'Authorization': `token ${this.githubToken}`,
                        'Content-Type': 'application/json'
                    },
                    onload: function (response) {
                        if (response.status === 200) {
                            const gistData = JSON.parse(response.responseText)
                            const fileContent = gistData.files[filename].content
                            const parsedJson = JSON.parse(fileContent)
                            GM_setValue('starred_notes', parsedJson)
                            alert(translate.DOWNLOAD_GIST_SUCCESS)
                        } else {
                            alert(translate.DOWNLOAD_GIST_FAILURE)
                            console.error('下载 Gist 失败:', response.responseText)
                        }
                    }
                })
            })
        },

        // 更新已有的 Gist
        updateGist: function (gistId, filename, content) {
            const url = `https://api.github.com/gists/${gistId}`
            const data = {
                'files': {
                    [filename]: {
                        'content': content
                    }
                }
            }
            GM_xmlhttpRequest({
                method: 'PATCH',
                url: url,
                headers: {
                    'Authorization': `token ${this.githubToken}`,
                    'Content-Type': 'application/json'
                },
                data: JSON.stringify(data),
                onload: function (response) {
                    if (response.status === 200) {
                        alert(translate.UPLOAD_GIST_SUCCESS)
                    } else {
                        alert(translate.UPLOAD_GIST_FAILURE)
                        console.error('Gist 更新失败:', response.responseText)
                    }
                }
            })
        }
    }
    const firstLanuch = GM_getValue('firstLanuch', true)//!首次启动脚本,提醒输入GitHub Token用于上传下载Gist
    if (firstLanuch && !GITHUB_PAT_TOKEN) {
        const userInput = prompt(translate.NO_TOKEN_ALERT_TEXT)
        if (userInput) {
            GM_setValue('GITHUB_PAT_TOKEN', userInput)
            GITHUB_PAT_TOKEN = userInput
        }
        GM_setValue('firstLanuch', false)
    }
    const description = 'github_starred_repo_note'
    GistManager.init(GITHUB_PAT_TOKEN, description)
    const filename = 'github_starred_repo_note.json'
    GM_registerMenuCommand(translate.UPLOAD_BUTTON_TEXT, uploadToGist)
    GM_registerMenuCommand(translate.DOWNLOAD_BUTTON_TEXT, downloadGistAsJson)
    if (Object.keys(GM_getValue('starred_notes', {})).length === 0 && GITHUB_PAT_TOKEN) {
        downloadGistAsJson()
    }
    function uploadToGist() {
        const panelData = GM_getValue('starred_notes', {})
        if (!getGitHubToken()) {
            return
        }
        const jsonData = JSON.stringify(panelData, null, 2)
        GistManager.uploadToGist(filename, jsonData)
    }
    function downloadGistAsJson() {
        if (!getGitHubToken()) {
            return
        }
        GistManager.downloadGistAsJson(filename)
    }
    function getGitHubToken() {
        if (GITHUB_PAT_TOKEN) {
            return true
        }
        const userInput = prompt(translate.NO_TOKEN_ALERT_TEXT)
        if (userInput) {
            GM_setValue('GITHUB_PAT_TOKEN', userInput)
            GITHUB_PAT_TOKEN = userInput
            GistManager.updateToken(GITHUB_PAT_TOKEN)
            return true
        }
        return false
    }
    /* ---------------------------------- json ---------------------------------- */
    // 写
    function saveStarredNote(storageKey, newNote) {
        let starredNotes = GM_getValue('starred_notes', '{}')
        starredNotes = JSON.parse(starredNotes)
        starredNotes[storageKey] = newNote
        GM_setValue('starred_notes', JSON.stringify(starredNotes))
    }
    //读
    function getStarredNote(storageKey) {
        let starredNotes = GM_getValue('starred_notes', '{}')
        starredNotes = JSON.parse(starredNotes)
        return starredNotes[storageKey] || ''
    }
    //删
    function deleteStarredNote(storageKey) {
        let starredNotes = GM_getValue('starred_notes', '{}')
        starredNotes = JSON.parse(starredNotes)

        if (storageKey in starredNotes) {
            delete starredNotes[storageKey]
            GM_setValue('starred_notes', JSON.stringify(starredNotes))
        }
    }
    // --- Core Logic ---
    // Get repo unique identifier (owner/repo)
    function getRepoFullName(repoElement) {
        const link = repoElement.querySelector('div[itemprop="name codeRepository"] > a, h3 > a, h2 > a')
        if (link && link.pathname) {
            return link.pathname.substring(1).replace(/\/$/, '')
        }
        const starForm = repoElement.querySelector('form[action^="/stars/"]')
        if (starForm && starForm.action) {
            const match = starForm.action.match(/\/stars\/([^/]+\/[^/]+)\/star/)
            if (match && match[1]) {
                return match[1]
            }
        }
        console.warn('RepoNotes: Could not find repo name for element:', repoElement)
        return null
    }
    // Create note button with icon
    function createNoteButton(isEdit = false) {
        const button = document.createElement('a')
        button.className = 'ghsn-note-btn'
        button.href = 'javascript:void(0);' // 使用 void(0) 避免页面跳转
        // SVG icon (pencil)
        const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg')
        svg.setAttribute('aria-hidden', 'true')
        svg.setAttribute('height', '16')
        svg.setAttribute('width', '16')
        svg.setAttribute('viewBox', '0 0 16 16')
        svg.setAttribute('fill', 'currentColor')
        svg.setAttribute('class', 'octicon octicon-star')
        const path = document.createElementNS('http://www.w3.org/2000/svg', 'path')
        // Pencil icon path data
        path.setAttribute('d', 'M11.013 1.427a1.75 1.75 0 0 1 2.474 0l1.086 1.086a1.75 1.75 0 0 1 0 2.474l-8.61 8.61c-.21.21-.47.364-.756.445l-3.251.93a.75.75 0 0 1-.927-.928l.929-3.25c.081-.286.235-.547.445-.758l8.61-8.61Zm.176 4.823L9.75 4.81l-6.286 6.287a.253.253 0 0 0-.064.108l-.558 1.953 1.953-.558a.253.253 0 0 0 .108-.064Zm1.238-3.763a.25.25 0 0 0-.354 0L10.811 3.75l1.439 1.44 1.263-1.263a.25.25 0 0 0 0-.354Z')
        svg.appendChild(path)
        button.appendChild(svg)
        const textNode = document.createTextNode(isEdit ? translate.EDIT_BUTTON_TEXT : translate.ADD_BUTTON_TEXT)
        button.appendChild(textNode)
        button.updateText = function (isEditing) {
            textNode.textContent = isEditing ? translate.EDIT_BUTTON_TEXT : translate.ADD_BUTTON_TEXT
        }
        return button
    }
    // Add note UI for a single repository
    async function addNoteUI(repoElement) {
        if (repoElement.querySelector('.ghsn-container')) {
            // console.log('RepoNotes: UI already exists for this repo element. Skipping.');
            return
        }
        const existingButton = repoElement.querySelector('.ghsn-star-row .ghsn-note-btn')
        if (existingButton) {
            // console.log('RepoNotes: Button already exists in star row. Skipping.');
            return
        }
        const repoFullName = getRepoFullName(repoElement)
        if (!repoFullName) {
            // console.warn('RepoNotes: Could not get repo full name. Skipping element:', repoElement);
            return
        }
        const storageKey = `ghsn_${repoFullName}`
        let currentNote = getStarredNote(storageKey)
        const starLink = repoElement.querySelector('a[href$="/stargazers"]')
        if (!starLink) {
            // console.warn(`RepoNotes: Could not find star link for repo: ${repoFullName}. Skipping.`);
            return
        }
        let starRow = starLink.parentNode
        if (!starRow.classList.contains('d-flex') && !starRow.classList.contains('float-right')) {
            const potentialRow = starLink.closest('span, div.d-inline-block, div.color-fg-muted')
            if (potentialRow) {
                starRow = potentialRow
            }
        }
        starRow.classList.add('ghsn-star-row')
        const noteButton = createNoteButton(!!currentNote) // !!currentNote 将其转为布尔值
        const container = document.createElement('div')
        container.className = 'ghsn-container'
        if (!currentNote) {
            container.classList.add('ghsn-hidden')
        }
        const displaySpan = document.createElement('span')
        displaySpan.className = 'ghsn-display'
        displaySpan.textContent = currentNote
        if (!currentNote) {
            displaySpan.classList.add('ghsn-hidden')
        }
        const noteTextarea = document.createElement('textarea')
        noteTextarea.className = 'ghsn-textarea ghsn-hidden'
        noteTextarea.placeholder = translate.NOTE_PLACEHOLDER
        const buttonsDiv = document.createElement('div')
        buttonsDiv.className = 'ghsn-buttons ghsn-hidden'
        const saveButton = document.createElement('button')
        saveButton.textContent = translate.SAVE_BUTTON_TEXT
        saveButton.className = 'ghsn-save'
        const cancelButton = document.createElement('button')
        cancelButton.textContent = translate.CANCEL_BUTTON_TEXT
        cancelButton.className = 'ghsn-cancel'
        const deleteButton = document.createElement('button')
        deleteButton.textContent = translate.DELETE_BUTTON_TEXT
        deleteButton.className = 'ghsn-delete'
        noteButton.addEventListener('click', (e) => {
            e.preventDefault()
            const isEditing = !noteTextarea.classList.contains('ghsn-hidden')
            if (!isEditing) {
                noteTextarea.value = currentNote
                displaySpan.classList.add('ghsn-hidden')
                noteTextarea.classList.remove('ghsn-hidden')
                buttonsDiv.classList.remove('ghsn-hidden')
                if (currentNote) {
                    deleteButton.classList.remove('ghsn-hidden')
                } else {
                    deleteButton.classList.add('ghsn-hidden')
                }
                container.classList.remove('ghsn-hidden')
                noteTextarea.focus()
            } else {
                cancelButton.click()
            }
        })
        cancelButton.addEventListener('click', () => {
            noteTextarea.classList.add('ghsn-hidden')
            buttonsDiv.classList.add('ghsn-hidden')
            if (currentNote) {
                displaySpan.textContent = currentNote
                displaySpan.classList.remove('ghsn-hidden')
                container.classList.remove('ghsn-hidden')
            } else {
                container.classList.add('ghsn-hidden')
            }
        })
        saveButton.addEventListener('click', async () => {
            const newNote = noteTextarea.value.trim()
            saveStarredNote(storageKey, newNote)
            currentNote = newNote
            noteButton.updateText(!!newNote)
            if (newNote) {
                displaySpan.textContent = newNote
                displaySpan.classList.remove('ghsn-hidden')
                container.classList.remove('ghsn-hidden')
            } else {
                displaySpan.classList.add('ghsn-hidden')
                container.classList.add('ghsn-hidden')
                deleteStarredNote(storageKey)

            }
            noteTextarea.classList.add('ghsn-hidden')
            buttonsDiv.classList.add('ghsn-hidden')
        })
        deleteButton.addEventListener('click', async () => {
            const showInfo = translate.DELETE_CONFIRM.replace('{repoFullName}', repoFullName)
            //${repoFullName}
            if (window.confirm(`${showInfo}`)) {
                deleteStarredNote(storageKey)
                currentNote = ''
                noteButton.updateText(false)
                displaySpan.classList.add('ghsn-hidden')
                noteTextarea.classList.add('ghsn-hidden')
                buttonsDiv.classList.add('ghsn-hidden')
                container.classList.add('ghsn-hidden')
            }
        })
        buttonsDiv.appendChild(deleteButton)
        buttonsDiv.appendChild(saveButton)
        buttonsDiv.appendChild(cancelButton)
        container.appendChild(displaySpan)
        container.appendChild(noteTextarea)
        container.appendChild(buttonsDiv)
        // 修改这里:将按钮作为starRow的最后一个元素
        starRow.appendChild(noteButton)
        const description = repoElement.querySelector('p.color-fg-muted')
        const topics = repoElement.querySelector('.topic-tag-list')
        const insertAfterElement = topics || description || repoElement.querySelector('h3, h2')
        if (insertAfterElement && insertAfterElement.parentNode) {
            insertAfterElement.parentNode.insertBefore(container, insertAfterElement.nextSibling)
        } else {
            repoElement.appendChild(container)
            console.warn(`RepoNotes: Could not find ideal insertion point for note container in repo: ${repoFullName}. Appending to end.`)
        }
    }
    // --- Process all repositories on the page ---
    function processRepositories() {
        const repoSelector = 'div.col-12.d-block.width-full.py-4.border-bottom.color-border-muted, article.Box-row'
        const repoElements = document.querySelectorAll(repoSelector)
        // console.log(`RepoNotes: Found ${repoElements.length} repository elements.`);
        if (repoElements.length === 0) {
            // console.log("RepoNotes: No repository elements found with selector:", repoSelector);
            const fallbackSelector = 'li[data-view-component="true"].Box-row'
            const fallbackElements = document.querySelectorAll(fallbackSelector)
            fallbackElements.forEach(addNoteUI)
        } else {
            repoElements.forEach(addNoteUI)
        }
    }

    // --- Observe DOM changes (handle dynamic loading like infinite scroll) ---
    let observer = null

    function setupObserver() {
        if (observer) {
            observer.disconnect()
        }

        const targetNode = document.getElementById('user-repositories-list') || document.querySelector('main') || document.body

        if (!targetNode) {
            console.error('RepoNotes: Could not find target node for MutationObserver.')
            return
        }
        // console.log('RepoNotes: Setting up MutationObserver on target:', targetNode);

        observer = new MutationObserver(mutations => {
            // console.log('RepoNotes: MutationObserver detected changes.');
            let needsProcessing = false
            mutations.forEach(mutation => {
                mutation.addedNodes.forEach(node => {
                    if (node.nodeType === 1) {
                        const repoSelector = 'div.col-12.d-block.width-full.py-4.border-bottom.color-border-muted, article.Box-row, li[data-view-component="true"].Box-row'
                        if (node.matches(repoSelector)) {
                            // console.log('RepoNotes: Added node matches repo selector:', node);
                            addNoteUI(node)
                            needsProcessing = true
                        } else {
                            const nestedRepos = node.querySelectorAll(repoSelector)
                            if (nestedRepos.length > 0) {
                                // console.log(`RepoNotes: Found ${nestedRepos.length} nested repos in added node:`, node);
                                nestedRepos.forEach(addNoteUI)
                                needsProcessing = true
                            }
                        }
                    }
                })
            })
        })

        observer.observe(targetNode, {
            childList: true,
            subtree: true
        })
    }

    // --- Startup and Navigation Handling ---

    function initializeOrReinitialize() {
        if (window.location.search.includes('tab=stars') || document.querySelector('div.col-12.d-block.width-full.py-4') || document.querySelector('article.Box-row')) {
            // console.log('RepoNotes: Running processRepositories.');
            processRepositories()
            // console.log('RepoNotes: Setting up observer.');
            setupObserver()
        } else {
            // console.log('RepoNotes: Not on a relevant page, skipping processing and observer setup.');
            if (observer) {
                observer.disconnect()
                // console.log('RepoNotes: Disconnected observer.');
            }
        }
    }

    document.addEventListener('turbo:load', () => {
        // console.log('RepoNotes: turbo:load event detected.');
        initializeOrReinitialize()
    })

    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', initializeOrReinitialize)
    } else {
        initializeOrReinitialize()
    }

})()