// ==UserScript==
// @name Greasyfork - Add notes to the script
// @name:zh-CN Greasyfork - 为脚本添加备注
// @name:zh-TW Greasyfork - 為腳本新增備註
// @namespace https://gf.qytechs.cn/zh-CN/users/193133-pana
// @homepage https://www.sailboatweb.com
// @version 1.1.0
// @description Add a note for scripts to help identify and search
// @description:zh-CN 为脚本添加备注功能,以帮助识别和搜索
// @description:zh-TW 為腳本新增備註功能,以幫助識別和搜尋
// @author pana
// @license GNU General Public License v3.0 or later
// @include http*://*gf.qytechs.cn/*
// @include http*://*sleazyfork.org/*
// @grant GM_getValue
// @grant GM_setValue
// ==/UserScript==
(function() {
'use strict';
const LANG = {
'EN': {
'title': 'Note',
'add_button_text': 'Add note',
'add_button_title': 'Add notes to the script',
'modify_button_text': 'Modify note',
'modify_button_title': 'Modify notes for the script',
'input_placeholder': '(Enter a note, delete it when blanked; press Enter to save)',
'save_button_text': 'Save',
'clear_button_text': 'Clear',
'cancel_button_text': 'Cancel',
'search_placeholder': 'Search notes'
},
'ZH_CN': {
'title': '备注',
'add_button_text': '添加备注',
'add_button_title': '为脚本添加备注',
'modify_button_text': '修改备注',
'modify_button_title': '为脚本修改备注',
'input_placeholder': '(请输入备注,置空时删除;按下Enter键保存)',
'save_button_text': '保存',
'clear_button_text': '清除',
'cancel_button_text': '取消',
'search_placeholder': '搜索备注'
},
'ZH_TW': {
'title': '備註',
'add_button_text': '新增備註',
'add_button_title': '為腳本新增備註',
'modify_button_text': '修改備註',
'modify_button_title': '為腳本修改備註',
'input_placeholder': '(請輸入備註,置空時刪除;按下Enter鍵儲存)',
'save_button_text': '儲存',
'clear_button_text': '清除',
'cancel_button_text': '取消',
'search_placeholder': '搜尋備註'
}
};
const ICON = {
'DOWN_ARROW': 'url()',
'UP_ARROW': 'url()',
};
const STYLE_VALUE = `
.my_greasyfork_note_btn {
margin-left: 10px;
}
.list_show, .show_separator {
display: block !important;
}
#presentation_div_for_user {
display: flex;
position: fixed;
background-color: rgba(0, 0, 0, .5);
top: 0;
bottom: 0;
left: 0;
right: 0;
z-index: 1;
align-items: center;
justify-content: center;
}
.dialog_div_for_user {
position: relative;
width: 400px;
background-color: #fff;
border: 0 solid #000;
border-radius: 12px;
display: flex;
flex-direction: column;
}
.user_title_span_for_user {
min-height: 48px;
text-align: center;
border: 1px solid #efefef;
color: #003399;
font-weight: bold;
background-color: rgba(0, 0, 0, 0);
border-top-left-radius: 12px;
border-top-right-radius: 12px;
}
.tag_input_for_user {
min-height: 32px;
margin: 5px;
border: 1px solid #cc6666;
padding-left: 5px;
}
.button_for_user {
min-height: 48px;
cursor: pointer;
border: 1px solid #efefef;
background-color: rgba(0, 0, 0, 0);
}
.cancel_button_for_user {
border-bottom-left-radius: 12px;
border-bottom-right-radius: 12px;
}
#searchFrame {
position: relative;
margin-left: 15px;
}
#myInputSearch {
width: 175px;
height: 25px;
border: 1px solid #999;
border-radius: 3px;
padding: 0 3px;
position: relative;
}
#dropDowns {
width: 15px;
height: 15px;
background-repeat: no-repeat;
background-size: 12px auto;
position: absolute;
top: 8px;
right: 2px;
}
.ins_down_arrow {
background-image: ${ICON.DOWN_ARROW};
}
.ins_up_arrow {
background-image: ${ICON.UP_ARROW};
}
#tagsList {
width: 180px;
height: 220px;
overflow-y: scroll;
text-align: left;
border: 1px solid #999;
display: none;
position: absolute;
top: 27px;
background-color: #fff;
z-index: 1;
}
.ins_list_item {
cursor: pointer;
color: #000;
padding-left: 5px;
}
.ins_highlight {
background-color: #6699cc;
}
.ins_hide {
display: none;
}
.ins_tag_span {
margin-left: 20px;
color: #336699;
}
.my_note_btn_hide {
display: none;
}
ol.script-list li:hover .my_greasyfork_note_btn {
display: inline !important;
}
.citrus-gfork-tag {
margin-left: 0px;
}
tr:hover .my_greasyfork_note_btn {
display: inline !important;
}
`;
class Greasyfork_Note {
constructor(config, lang, show_list = []) {
this.config = config;
this.lang = lang;
this.showList = show_list;
}
createNoteBtn(script_id, callback, class_name = "my_greasyfork_note_btn") {
let btn = document.createElement('a');
btn.className = class_name;
btn.target = '_self';
btn.href = 'javascript:;';
if (this.judgeScripts(script_id)) {
btn.textContent = this.lang.modify_button_text;
btn.title = this.lang.modify_button_title;
} else {
btn.textContent = this.lang.add_button_text;
btn.title = this.lang.add_button_title;
}
btn.addEventListener('click', () => {
document.body.appendChild(this.createNoteFrame(script_id, () => {
if (this.judgeScripts(script_id)) {
btn.textContent = this.lang.modify_button_text;
btn.title = this.lang.modify_button_title;
} else {
btn.textContent = this.lang.add_button_text;
btn.title = this.lang.add_button_title;
}
if (typeof(callback) == 'function') {
callback();
}
}));
});
return btn;
}
judgeScripts(script_id) {
if (this.getScriptIndex(script_id) == -1) {
return false;
}
return true;
}
getScriptIndex(script_id) {
for (let i in this.config.scripts_array) {
if (script_id == this.config.scripts_array[i].id) {
return i;
}
}
return -1;
}
getScriptTag(script_id) {
if (this.judgeScripts(script_id)) {
return this.config.scripts_array[this.getScriptIndex(script_id)].tag;
}
return '';
}
getScriptFormatTag(script_id) {
if (this.judgeScripts(script_id)) {
return '[' + this.getScriptTag(script_id) + ']';
}
return '';
}
writeScripts(script_id, tag_value) {
if (this.judgeScripts(script_id)) {
let index = this.getScriptIndex(script_id);
if (tag_value) {
this.config.scripts_array[index].tag = tag_value;
} else {
this.config.scripts_array.splice(index, 1);
}
} else {
if (tag_value) {
let temp_scripts_obj = {
'id': script_id,
'tag': tag_value
};
this.config.scripts_array.push(temp_scripts_obj);
}
}
GM_setValue('greasyfork_config', this.config);
}
removeNoteFrame(frame_id = 'presentation_div_for_user') {
let temp_ele = document.getElementById(frame_id);
temp_ele.parentNode.removeChild(temp_ele);
}
createNoteFrame(script_id, callback) {
let that = this;
let presentation_div = document.createElement('div');
presentation_div.id = 'presentation_div_for_user';
presentation_div.addEventListener('click', function (event) {
if (event.target === this) {
that.removeNoteFrame();
}
});
let dialog_div = document.createElement('div');
dialog_div.className = 'dialog_div_for_user';
let user_title_p = document.createElement('button');
user_title_p.className = 'user_title_span_for_user';
user_title_p.textContent = "ID: " + script_id;
let tag_input = document.createElement('input');
tag_input.className = 'tag_input_for_user';
tag_input.type = 'text';
tag_input.placeholder = this.lang.input_placeholder;
if (this.judgeScripts(script_id)) {
tag_input.value = this.config.scripts_array[this.getScriptIndex(script_id)].tag;
} else {
tag_input.value = '';
}
tag_input.addEventListener('keyup', (e) => {
if (e.keyCode === 13) {
this.writeScripts(script_id, tag_input.value);
this.resetSearchFrame();
if (typeof(callback) == 'function') {
callback();
}
this.removeNoteFrame();
}
});
setTimeout(function() {
try {
tag_input.focus();
tag_input.select();
} catch(e) {
console.error(e);
}
}, 200);
let save_button = document.createElement('button');
save_button.className = 'button_for_user';
save_button.type = 'button';
save_button.innerText = this.lang.save_button_text;
save_button.addEventListener('click', () => {
this.writeScripts(script_id, tag_input.value);
this.resetSearchFrame();
if (typeof(callback) == 'function') {
callback();
}
this.removeNoteFrame();
});
let clear_button = document.createElement('button');
clear_button.className = 'button_for_user';
clear_button.type = 'button';
clear_button.innerText = this.lang.clear_button_text;
clear_button.addEventListener('click', () => {
this.writeScripts(script_id, '');
this.resetSearchFrame();
if (typeof(callback) == 'function') {
callback();
}
this.removeNoteFrame();
});
let cancel_button = document.createElement('button');
cancel_button.className = 'button_for_user cancel_button_for_user';
cancel_button.type = 'button';
cancel_button.innerText = this.lang.cancel_button_text;
cancel_button.addEventListener('click', () => {
this.removeNoteFrame();
});
dialog_div.appendChild(user_title_p);
dialog_div.appendChild(tag_input);
dialog_div.appendChild(save_button);
dialog_div.appendChild(clear_button);
dialog_div.appendChild(cancel_button);
presentation_div.appendChild(dialog_div);
return presentation_div;
}
resetSearchFrame() {
let tags_list = document.getElementById('tagsList');
if (tags_list) {
tags_list.innerHTML = "";
this.config.scripts_array.forEach((item, index) => {
tags_list.appendChild(this.createListDiv(index, item));
});
}
}
cretaeSearchFrame() {
let search_frame = document.createElement('div');
search_frame.id = 'searchFrame';
let search_input = document.createElement('input');
search_input.id = 'myInputSearch';
search_input.type = 'text';
search_input.placeholder = this.lang.search_placeholder;
search_input.value = "";
search_input.addEventListener('focusin', () => {
document.getElementById('tagsList').classList.add('list_show');
let arrow = document.getElementById('dropDowns');
arrow.classList.remove('ins_down_arrow');
arrow.classList.add('ins_up_arrow');
this.searchEvent(search_input);
});
search_frame.appendChild(search_input);
let dropdowns = document.createElement('div');
dropdowns.id = 'dropDowns';
dropdowns.className = 'ins_down_arrow';
dropdowns.addEventListener('click', function() {
let tags_list = document.getElementById('tagsList');
if (tags_list.classList.contains('list_show')) {
tags_list.classList.remove('list_show');
} else {
tags_list.classList.add('list_show');
}
if (this.classList.contains('ins_up_arrow')) {
this.classList.remove('ins_up_arrow');
} else {
this.classList.add('ins_up_arrow');
}
if (this.classList.contains('ins_down_arrow')) {
this.classList.remove('ins_down_arrow');
} else {
this.classList.add('ins_down_arrow');
}
});
search_frame.appendChild(dropdowns);
let tags_list = document.createElement('div');
tags_list.id = 'tagsList';
for (let i = 0; i < this.config.scripts_array.length; i ++) {
tags_list.appendChild(this.createListDiv(i, this.config.scripts_array[i]));
}
search_frame.appendChild(tags_list);
document.body.onclick = function(e){
e = e || window.event;
let target = e.target || e.srcElement;
if(target !== document.getElementById('dropDowns') && target !== document.getElementById('tagsList') && target !== document.getElementById('myInputSearch')){
document.getElementById('tagsList').classList.remove('list_show');
let arrow = document.getElementById('dropDowns');
arrow.classList.remove('ins_up_arrow');
arrow.classList.add('ins_down_arrow');
}
};
return search_frame;
}
createListDiv(id_number, scripts_obj) {
let list_div = document.createElement('div');
list_div.id = 'tags_' + id_number;
list_div.className = 'ins_list_item';
list_div.textContent = scripts_obj.tag;
list_div.addEventListener('mouseenter', function() {
for (let ele of document.querySelectorAll('#tagsList div')) {
ele.classList.remove('ins_highlight');
}
this.classList.add('ins_highlight');
});
list_div.addEventListener('click', function() {
location.pathname = location.pathname.replace(/^(\/[\w-]+)\/?.*/i, "$1" + "/scripts/" + scripts_obj.id);
});
return list_div;
}
searchEvent(input_dom) {
let list_arr = [];
for (let ele of document.querySelectorAll('#tagsList div')) {
let arr_obj = {
'eleContainer': ele.textContent,
'ele': ele
};
list_arr.push(arr_obj);
}
let current_index = 0;
input_dom.addEventListener('keyup', (event) => {
document.getElementById('tagsList').classList.add('list_show');
let arrow = document.getElementById('dropDowns');
arrow.classList.remove('ins_down_arrow');
arrow.classList.add('ins_up_arrow');
let search_val;
switch (event.keyCode) {
case 38:
case 40:
case 37:
case 39:
event.returnValue = false;
break;
case 13:
this.showList[current_index].click();
break;
default:
search_val = input_dom.value;
this.showList = [];
list_arr.forEach((item) => {
if (item.eleContainer.indexOf(search_val) !== -1) {
item.ele.classList.remove('ins_hide');
this.showList.push(item.ele);
} else {
item.ele.classList.add('ins_hide');
}
});
current_index = 0;
break;
}
this.showList.forEach(function(item, index) {
if (index === current_index) {
item.classList.add('ins_highlight');
document.getElementById('tagsList').scrollTop = item.offsetTop;
} else {
item.classList.remove('ins_highlight');
}
});
let list_height = 22 * this.showList.length;
if (list_height < 220) {
document.getElementById('tagsList').style.height = list_height + 'px';
} else {
document.getElementById('tagsList').style.height = '220px';
}
});
input_dom.addEventListener('keydown', (event) => {
if (event.keyCode === 38) {
current_index --;
if (current_index < 0) {
current_index = 0;
}
} else if (event.keyCode === 40) {
current_index ++;
if (current_index >= this.showList.length) {
current_index = this.showList.length - 1;
}
}
this.showList.forEach(function(item, index) {
if (index === current_index) {
item.classList.add('ins_highlight');
document.getElementById('tagsList').scrollTop = item.offsetTop;
} else {
item.classList.remove('ins_highlight');
}
});
});
}
createNoteSpan(script_id, an_class_name = "") {
let note_span = document.createElement('span');
note_span.className = 'ins_tag_span';
if (an_class_name) {
note_span.classList.add(an_class_name);
}
note_span.textContent = this.getScriptFormatTag(script_id);
return note_span;
}
}
function init(greasyfork_config) {
let pathname = location.pathname;
let style_dom = document.createElement('style');
style_dom.type = 'text/css';
style_dom.innerHTML = STYLE_VALUE;
document.body.appendChild(style_dom);
let lang_str = document.documentElement.lang;
let lang_value;
switch (lang_str) {
case 'zh':
case 'zh-cn':
case 'zh-CN':
lang_value = LANG.ZH_CN;
break;
case 'zh-hk':
case 'zh-HK':
case 'zh-tw':
case 'zh-TW':
lang_value = LANG.ZH_TW;
break;
case 'en':
default:
lang_value = LANG.EN;
break;
}
let note_obj = new Greasyfork_Note(greasyfork_config, lang_value);
let search_li = document.createElement('li');
search_li.appendChild(note_obj.cretaeSearchFrame());
document.querySelector('#site-nav nav').insertAdjacentElement('afterbegin', search_li);
if (/^\/[\w-]+\/scripts\/\d+-/i.test(pathname)) {
let script_id = /^\/[\w-]+\/scripts\/(\d+)-/i.exec(pathname)[1];
if (document.getElementById('script-feedback-suggestion')) {
document.getElementById('script-feedback-suggestion').appendChild(note_obj.createNoteBtn(script_id, function() {
if (document.querySelector('#script-info h2 > span')) {
let span_dom = document.querySelector('#script-info h2 > span');
if (note_obj.judgeScripts(script_id)) {
span_dom.textContent = note_obj.getScriptFormatTag(script_id);
} else {
document.querySelector('#script-info h2').removeChild(span_dom);
}
} else {
if (note_obj.judgeScripts(script_id)) {
document.querySelector('#script-info h2').appendChild(note_obj.createNoteSpan(script_id));
}
}
}));
}
if (note_obj.judgeScripts(script_id)) {
document.querySelector('#script-info h2').appendChild(note_obj.createNoteSpan(script_id));
}
} else if (/^\/[\w-]+\/scripts/i.test(pathname) || /^\/[\w-]+\/users\/\d+/i.test(pathname)) {
let browse_list = document.querySelectorAll('ol.script-list li');
for (let ele of browse_list) {
let script_id = ele.getAttribute('data-script-id');
if (script_id) {
if (ele.querySelector('dd.script-list-author span')) {
ele.querySelector('dd.script-list-author span').appendChild(note_obj.createNoteBtn(script_id, function() {
if (ele.querySelector('h2 > .ins_tag_span')) {
let span_dom = ele.querySelector('h2 .ins_tag_span');
if (note_obj.judgeScripts(script_id)) {
span_dom.textContent = note_obj.getScriptFormatTag(script_id);
} else {
ele.querySelector('h2').removeChild(span_dom);
}
} else {
if (note_obj.judgeScripts(script_id)) {
ele.querySelector('.name-description-separator').after(note_obj.createNoteSpan(script_id));
}
}
}, 'my_greasyfork_note_btn my_note_btn_hide'));
}
if (note_obj.judgeScripts(script_id)) {
ele.querySelector('.name-description-separator').after(note_obj.createNoteSpan(script_id));
}
}
}
document.querySelectorAll('#script-table tbody tr').forEach(item => {
let script_title = item.querySelector('.thetitle a');
if (script_title) {
let script_id = script_title.href.match(/\d+$/) && script_title.href.match(/\d+$/)[0];
note_obj.judgeScripts(script_id) && script_title.after(note_obj.createNoteSpan(script_id, 'citrus-gfork-tag'));
let p_dom = item.querySelector('.theauthor') || item.querySelector('.ins_tag_span') || script_title;
script_id && p_dom.after(note_obj.createNoteBtn(script_id, () => {
let tag_dom = item.querySelector('.ins_tag_span');
if (tag_dom) {
if (note_obj.judgeScripts(script_id)) {
tag_dom.textContent = note_obj.getScriptFormatTag(script_id);
} else {
tag_dom.remove();
}
} else {
note_obj.judgeScripts(script_id) && script_title.after(note_obj.createNoteSpan(script_id, 'citrus-gfork-tag'));
}
}, 'my_greasyfork_note_btn my_note_btn_hide'));
}
});
}
}
Promise.all([GM_getValue('greasyfork_config')]).then(function(data) {
let greasyfork_config = {
scripts_array: []
};
if (data[0] !== undefined) {
greasyfork_config = data[0];
}
init(greasyfork_config);
}).catch(function(e) {
console.error('Script error.');
console.error(e);
});
})();