// ==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://gf.zukizuki.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()
}
})()