// ==UserScript==
// @name Reddit - Quick RES user tagging
// @description Quickly tag multiple users with the same tag and open the tag popup using the keyboard
// @author James Skinner <spiralx@gmail.com> (http://github.com/spiralx)
// @namespace http://spiralx.org/
// @version 2.2.0
// @license MIT
// @icon 
// @supportURL https://greasyfork.org/en/scripts/370256-reddit-quick-user-tagging/feedback
// @match *://*.reddit.com/r/*
// @match *://*.reddit.com/user/*
// @grant none
// @run-at document-end
// @require https://unpkg.com/jquery@3/dist/jquery.min.js
// @require https://greasyfork.org/scripts/389748-console-message-v2/code/console-message-v2.js?version=730537
// ==/UserScript==
/* jshint asi: true, esnext: true, laxbreak: true */
/* global jQuery, message */
/**
==== 2.2.0 (2022.07.24) ====
* Change name and description
* Add Q keyboard shortcut to open tag dialog
* Removed references to unused Watcher object
* Tidy up some code
==== 2.1.2 (2022.07.01) ====
* Update icon and add license metadata field
==== 2.1.1 (2022.04.16) ====
* Clicking 'Clear tag' closes tag popup
==== 2.1.0 (2021.11.05) ====
* Fix when ConsoleMessage not available
==== 2.0.5 (2021.06.08) ====
* Update icons
==== 2.0.4 (2021.06.02) ====
* Set width of tag dialog to 800px
==== 2.0.0 (2019.09.03) ====
* Rename script and update version to 2.0.0
==== 1.0.0 (2018.10.09) ====
* Rewrite code base
* Set previous tag on preset tag clicked
==== 0.9.0 (2018.10.09) ====
* Made tag modal wider and text smaller to accomodate more preset tags
* Added clear tag link to the right of the colour drop-down
==== 0.8.0 (2018.07.13) ====
* Changed console.message require to use GreasyFork
==== 0.7.0 (2018.02.13) ====
* Changed the rendering of the tag preview to match how RES does it
* Moved output to all use console.message
==== 0.6.0 (2018.02.11) ====
* Use unpkg.com for jQuery
* Add console.message for logging
* Use localstorage to save tags
* Use new Tag class to store tag info
* Updated ID for text field in tag popup
==== 0.5.1 (2018.02.11) ====
* Update icons to match other Reddit script
==== 0.5.0 (26.08.2017) ====
* Change to simple use of GM_getValue and GM_setValue for storage
==== 0.4.0 (21.08.2017) ====
* Update all other tags correctly when changing a user's tag
* Handle removing all other tags when clearing a user's tag
==== 0.3.0 (13.07.2017) ====
* Checks tag link to see if tag set, always overwrites if not
* Updates other tags for same user on current page
==== 0.2.1 (27.06.2017) ====
* Changed timeout of field set function to 250ms
==== 0.2.0 (31.05.2017) ====
* Updated jQuery to v3.2.1
* Added timeout before overriding tag/colour fields
* Update preview when setting tag/colour
*/
; (($, message) => {
const STYLES = {
func: { color: '#c41', 'font-weight': 'bold' },
attr: { color: '#1a2', 'font-weight': 'bold' },
value: { color: '#05f' },
punc: { 'font-weight': 'bold' },
// comment: { color: 'c1007f' },
// error: { color: '#f4f', 'font-weight': 'bold' },
// link: { color: '#05f', 'text-decoration': 'underline' },
}
// --------------------------------------------------------------------------
if (message) {
message().extend({
tag({ text, colour }) {
return this.text(text, {
fontSize: '0.9em',
padding: '0 4px',
border: 'solid 1px rgb(199, 199, 199)',
borderRadius: 3,
...getStyleForColour(colour)
})
}
})
}
// --------------------------------------------------------------------------
const TAG_STORAGE_KEY = 'resPreviousUserTag'
const TAG_DIALOG_OPEN_TIMEOUT = 200
const BACKGROUND_TO_TEXT_COLOUR_MAP = {
none: 'inherit',
aqua: 'black',
lime: 'black',
pink: 'black',
silver: 'black',
white: 'black',
yellow: 'black'
}
const IGNORE_TAGS = new Set([ 'A', 'BUTTON', 'INPUT', 'TEXTAREA' ])
// --------------------------------------------------------------------
const CLEAR_TAG_LINK = `
<a href="javascript:void(0)"
title="Clear tag">
clear tag
</a>
`
const NO_PREVIEW_TAG = `
<span class="RESUserTag">
<a href="javascript:void 0"
title="Set a tag"
class="RESUserTagImage userTagLink truncateTag"
> </a>
</span>
`
const QUICK_REPEAT_TAG = `
<span class="RESUserTag" style="filter: hue-rotate(90deg) saturate(0.5);">
<a href="javascript:void 0"
title="Repeat previous tag"
class="RESUserTagImage userTagLink truncateTag"
> </a>
</span>
`
// --------------------------------------------------------------------------
function getStyleForColour (colour) {
return {
color: BACKGROUND_TO_TEXT_COLOUR_MAP[ colour ] || 'white',
backgroundColor: colour === 'none'
? 'transparent'
: colour
}
}
// --------------------------------------------------------------------------
const getPreviewForTag = ({ text, colour }) => {
const { color, backgroundColor } = getStyleForColour(colour)
return `
<span class="RESUserTag">
<a href="javascript:void 0"
title="${text}"
class="userTagLink hasTag truncateTag"
style="color: ${color}; background-color: ${backgroundColor};"
>${text}</a>
</span>
`
}
// --------------------------------------------------------------------------
const EMPTY_TAG = Object.freeze({
text: '',
colour: 'none'
})
const isEmptyTag = ({ text, colour } = EMPTY_TAG) => text === '' && colour === 'none'
const areEqualTags = (a, b) => a.text === b.text && a.colour === b.colour
// --------------------------------------------------------------------------
let previousUserTag = EMPTY_TAG
let ctx = null
if (localStorage[ TAG_STORAGE_KEY ]) {
previousUserTag = JSON.parse(localStorage[ TAG_STORAGE_KEY ])
}
if (message) {
let msg = message()
.text('reddit-save-res-tag.init', STYLES.func)
.text(': ', STYLES.punc)
if (!isEmptyTag(previousUserTag)) {
msg = msg
.text('previousUserTag', STYLES.attr)
.text(' = ', STYLES.punc)
.tag(previousUserTag)
} else {
msg = msg.text('No previous tag found')
}
msg.print()
} else if (previousUserTag !== EMPTY_TAG) {
console.log(`previousUserTag = %o`, previousUserTag)
}
// --------------------------------------------------------------------------
class TagContext {
constructor ($thingElem) {
this.$thingElem = $thingElem
this.authorId = $thingElem.data('author-fullname')
this.$dialog = $('.userTagger-dialog-head').closest('.RESHover')
this.user = $('.RESHover .RESHoverTitle .res-icon + span').text() || null
this.$textField = $('#userTaggerText')
this.$colourField = $('#userTaggerColor')
this.$previewElem = $('#userTaggerPreview')
this.$presetTags = $('#userTaggerPresetTags')
}
clearFields () {
this.$textField.val('').focus()
this.$colourField.val('none')
this.$previewElem.html(NO_PREVIEW_TAG)
}
setFields ({ text, colour }) {
this.$textField.val(text)
this.$colourField.val(colour)
this.$previewElem.html(getPreviewForTag({ text, colour }))
}
getTag () {
return {
text: this.$textField.val().trim(),
colour: this.$colourField.val()
}
}
}
// --------------------------------------------------------------------------
function onTagSelected (tag) {
// console.info(`onTagSelected: tag = %o, ctx = %o, prevTag = %o`, tag, ctx, previousUserTag)
if (message) {
message()
.text('onTagSelected', STYLES.func)
.text('(', STYLES.punc)
.text('tag', STYLES.attr)
.text(': ', STYLES.punc)
.tag(tag)
.text('): ', STYLES.punc)
.text('previousUserTag', STYLES.attr)
.text(' = ', STYLES.punc)
.tag(previousUserTag)
.text(', ', STYLES.punc)
.text('ctx', STYLES.attr)
.text(' = ', STYLES.punc)
.object(ctx)
.print()
}
const $tagLinks = $(`.id-${ctx.authorId}`)
.next()
.children(0)
if (isEmptyTag(tag)) {
$tagLinks
.html(' ')
.css('background-color', 'transparent')
.removeClass('hasTag')
.addClass('RESUserTagImage')
} else if (!areEqualTags(tag, previousUserTag)) {
previousUserTag = tag
localStorage[TAG_STORAGE_KEY] = JSON.stringify(previousUserTag)
$tagLinks
.text(tag.text)
.css(getStyleForColour(tag.colour))
.addClass('hasTag')
.removeClass('RESUserTagImage')
}
ctx = null
}
// --------------------------------------------------------------------------
function onTagClicked ($tagLink) {
const $thing = $tagLink.closest('.thing')
// console.info(`a.userTagLink.click`, this, $thing)
setTimeout(() => {
ctx = new TagContext($thing)
ctx.$dialog.width(800)
const authorHasTag = $tagLink.hasClass('hasTag')
if (!authorHasTag && !isEmptyTag(previousUserTag)) {
ctx.setFields(previousUserTag)
}
$(CLEAR_TAG_LINK)
.click(function () {
ctx.clearFields()
$('#userTaggerSave').click()
return false
})
.css({ marginLeft: 'auto' })
.insertAfter(ctx.$colourField)
const $srcField = ctx.$previewElem.parent().next()
ctx.$presetTags
.one('click.resrem', '.userTagLink', function () {
const $clicked = $(this)
const clickedTag = {
text: $clicked.text(),
colour: $clicked.css('background-color')
}
// console.info(`preset .userTagLink.click`, this, clickedTag, ctx)
onTagSelected(clickedTag)
})
.parent()
.insertBefore($srcField)
$srcField.next().css({ float: 'left', width: '50%' })
}, TAG_DIALOG_OPEN_TIMEOUT)
}
// --------------------------------------------------------------------------
$('body')
.on('click.resrem', '.RESUserTag', function () {
const $tagLink = $(this).children('a.userTagLink')
onTagClicked($tagLink)
})
.on('click.resrem', '#userTaggerSave', function () {
if (ctx) {
onTagSelected(ctx.getTag())
}
})
.on('keypress.resrem', event => {
if (ctx || event.key !== 'q' || IGNORE_TAGS.has(event.target.tagName)) {
return
}
$('div.thing.comment.noncollapsed.res-selected .entry.res-selected .RESUserTag').click()
return false
})
// --------------------------------------------------------------------------
/*
<div class="RESHover RESHoverInfoCard RESDialogSmall" style="top: 387.918px; left: 215.767px; width: 350px; display: block; opacity: 1;">
<h3 class="RESHoverTitle" data-hover-element="0">
<div>
<span class="res-icon"></span> <span>Plan-Six</span>
</div>
</h3>
<div class="RESCloseButton">x</div>
<div class="RESHoverBody RESDialogContents" data-hover-element="1">
<form id="userTaggerToolTip">
<div class="fieldPair">
<label class="fieldPair-label" for="userTaggerText">Text</label>
<input class="fieldPair-text" type="text" id="userTaggerText">
</div>
<div class="fieldPair">
<label class="fieldPair-label" for="userTaggerColor">Color</label>
<select id="userTaggerColor">
<option style="color: inherit; background-color: none" value="none">none</option>
<option style="color: black; background-color: aqua" value="aqua">aqua</option>
<option style="color: white; background-color: black" value="black">black</option>
<option style="color: white; background-color: blue" value="blue">blue</option>
<option style="color: white; background-color: cornflowerblue" value="cornflowerblue">cornflowerblue</option>
<option style="color: white; background-color: fuchsia" value="fuchsia">fuchsia</option>
<option style="color: white; background-color: gray" value="gray">gray</option>
<option style="color: white; background-color: green" value="green">green</option>
<option style="color: black; background-color: lime" value="lime">lime</option>
<option style="color: white; background-color: maroon" value="maroon">maroon</option>
<option style="color: white; background-color: navy" value="navy">navy</option>
<option style="color: white; background-color: olive" value="olive">olive</option>
<option style="color: white; background-color: orange" value="orange">orange</option>
<option style="color: white; background-color: orangered" value="orangered">orangered</option>
<option style="color: black; background-color: pink" value="pink">pink</option>
<option style="color: white; background-color: purple" value="purple">purple</option>
<option style="color: white; background-color: red" value="red">red</option>
<option style="color: black; background-color: silver" value="silver">silver</option>
<option style="color: white; background-color: teal" value="teal">teal</option>
<option style="color: black; background-color: white" value="white">white</option>
<option style="color: black; background-color: yellow" value="yellow">yellow</option>
</select>
</div>
<div class="fieldPair">
<label class="fieldPair-label" for="userTaggerPreview">Preview</label>
<span id="userTaggerPreview" style="color: white; background-color: olive;">
<span class="RESUserTag">
<a class="userTagLink hasTag truncateTag" style="background-color: olive; color: white !important;" title="Sexist" href="javascript:void 0">Sexist</a>
</span>
</span>
</div>
<a class="userTagLink hasTag truncateTag" style="background-color: olive; color: white !important;" title="Feminist" href="javascript:void 0">Feminist</a>
<div class="fieldPair res-usertag-ignore">
<label class="fieldPair-label" for="userTaggerIgnore">Ignore</label>
<div id="userTaggerIgnoreContainer" class="toggleButton ">
<span class="toggleThumb"></span>
<div class="toggleLabel res-icon" data-enabled-text="" data-disabled-text=""></div>
<input id="userTaggerIgnore" name="userTaggerIgnore" type="checkbox">
</div>
<a class="gearIcon" href="#res:settings/userTagger/hardIgnore" title="RES Settings > User Tagger > hardIgnore"> configure </a>
</div>
<div class="fieldPair">
<label class="fieldPair-label" for="userTaggerLink">
<span class="userTaggerOpenLink">
<a title="open link" href="javascript:void 0">Source URL</a>
</span>
</label>
<input class="fieldPair-text" type="text" id="userTaggerLink" value="https://www.reddit.com/r/giantbomb/comments/7x251d/all_systems_goku_02/du5fpwr/">
</div>
<div class="fieldPair">
<label class="fieldPair-label" for="userTaggerVotesUp" title="Upvotes you have given this redditor">Upvotes</label>
<input type="number" style="width: 50px;" id="userTaggerVotesUp" value="0">
</div>
<div class="fieldPair">
<label class="fieldPair-label" for="userTaggerVotesDown" title="Downvotes you have given this redditor">Downvotes</label>
<input type="number" style="width: 50px;" id="userTaggerVotesDown" value="0">
</div>
<div class="res-usertagger-footer">
<a href="/r/dashboard#userTaggerContents" target="_blank" rel="noopener noreferer">View tagged users</a>
<input type="submit" id="userTaggerSave" value="✓ save tag">
</div>
</form>
</div>
</div>
*/
})(jQuery, typeof message !== 'undefined' ? message : null)
jQuery.noConflict(true)