您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
曾经 Star 过的仓库是不是忘记了它们的用途? 这是一个可以给你收藏过的仓库添加备注的的脚本,并且你备注的信息可以上传下载到 GitHub Gist
当前为
// ==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() } })()