Yet Another Various Artists Helper
// ==UserScript== // @name What.CD: YAVAH // @namespace hateradio))) // @author hateradio // @version 7.2 // @description Yet Another Various Artists Helper // @icon  // @include /https://redacted\.sh/(torrents\.php(\?|\?page=\d+&)id=\d+(&torrentid=\d+)?(#comments)?|upload\.php(\?requestid=\d+)?|requests\.php*)/ // @include /https://orpheus\.network/(torrents\.php(\?|\?page=\d+&)id=\d+(&torrentid=\d+)?(#comments)?|upload\.php(\?requestid=\d+)?|requests\.php*)/ // @include /https://notwhat\.cd/(torrents\.php(\?|\?page=\d+&)id=\d+(&torrentid=\d+)?(#comments)?|upload\.php(\?requestid=\d+)?|requests\.php*)/ // #updated 26 Nov 2024 // #since 18 Jun 2010 // ==/UserScript== (() => { if (!Element.prototype.matches) Element.prototype.matches = Element.prototype.msMatchesSelector || Element.prototype.webkitMatchesSelector if (!Element.prototype.closest) { Element.prototype.closest = function (s) { let el = this if (!document.documentElement.contains(el)) return null do { if (el.matches(s)) return el el = el.parentElement || el.parentNode } while (el !== null && el.nodeType === 1) return null } } const _ = { css: text => { if (!this.style) { this.style = document.createElement('style') this.style.type = 'text/css' document.body.appendChild(this.style) } this.style.appendChild(document.createTextNode(`${text}\n`)) }, js: func => { const script = document.createElement('script') script.type = 'application/javascript' script.textContent = `;(${func})();` document.body.appendChild(script) document.body.removeChild(script) }, debounce: (func, wait) => { let timeout return function (...args) { const run = () => { timeout = null func.apply(this, args) } clearTimeout(timeout) timeout = setTimeout(run, wait) } }, on: (element, type, selector, listener) => { element.addEventListener(type, event => { const found = event.target.closest(selector) if (found) listener.call(found, event) }, false) } } class YavaMenu { constructor() { this.sibling = document.querySelector('.box_addartists, #artist_tr') this.setup() YavaMenu.check = document.getElementById('yavah_semi') } get types() { return ['Main', 'Guest', 'Remixer', 'Composer', 'Conductor', 'DJ / Compiler', 'Producer'] } setup() { if (!this.sibling) return const box = document.createElement('div') box.id = 'YAVAH' _.on(box, 'click', 'a', this.toggle) this.boxSetup(box) box.querySelector('a').click() this.box = box } boxSetup(box) { box.className = 'box' box.innerHTML = ` <div class="head"> <strong><abbr title="Yet Another Various Artists Helper">YAVAH</abbr></strong> </div> <div style="padding: 3px 6px 6px"> ${this.generateInputs()} </div>` this.sibling.parentElement.insertBefore(box, this.sibling) } toggle(e) { e.preventDefault() const tog = this.parentElement.nextElementSibling.classList.toggle('hidden') this.innerHTML = `<code>${tog ? '+' : '-'}</code> ${this.dataset.type}` } generateInputs() { // let yavahtog = this.nextElementSibling.classList.toggle('hidden'); // this.firstElementChild.innerHTML = '<code>' + (yavahtog ? '+' : '-') + '</code> ' + this.firstElementChild.dataset.type const boxes = this.types.map(type => { return `<p><a href="#" data-type="${type}"><code>+</code> ${type}</a></p><textarea class="noWhutBB yavahtext hidden"></textarea>` }).join('') return ` <p>Enter artists, one per line.</p> <p> <input type="checkbox" id="yavah_semi"> <label for="yavah_semi">Split semi-colons</label> </p> ${boxes} <p>Review the changes below.</p>` } addEvent(cb) { return this.box && !this.box.addEventListener('input', _.debounce(cb, 250), false) } } class YavaMenuTr extends YavaMenu { boxSetup(box) { box.innerHTML = this.generateInputs() const tr = document.createElement('tr') tr.innerHTML = '<td class="label">YAVAH</td><td></td>' tr.lastElementChild.appendChild(box) this.sibling.parentElement.insertBefore(tr, this.sibling) } } class Yavah { constructor() { this.selector = 'input[name="aliasname[]"]:last-of-type, input[name="artists[]"]:last-of-type' this.add = document.querySelector('.box_addartists a, #artistfields a.brackets') this.inputs = document.querySelector('#AddArtists, #artistfields') if (!this.inputs) return _.css('#YAVAH p a { display:block } .yavahtext{width: 90%; height: 6em}') this.stored = this.inputs.innerHTML this.event = this.event.bind(this) } regex() { if (YavaMenu.check.checked) return /[^\r\n;]+/g return /[^\r\n]+/g } /** * * @param {HTMLTextAreaElement} textarea * @param {number} index Index of HTMLOptionElement within HTMLSelectElement to set (Main, Guest, Composer, etc.) */ fill(textarea, index) { const lines = textarea.value.match(this.REGEX) || [] const unique = new Set(lines.map(_ => _.trim())) unique.delete('') unique.forEach(name => { this.inputs.querySelector(this.selector).value = name this.inputs.querySelector('select:last-of-type').value = index + 1 this.add.click() }) } event() { // Reset the artist box this.inputs.innerHTML = this.stored _.js(() => window.ArtistFieldCount = -1000) const textareas = document.querySelectorAll('#YAVAH textarea') this.REGEX = this.regex() Array.from(textareas).forEach(this.fill, this) } static main() { const yava = new Yavah() const menu = /(?:requests\.php|upload\.php)/.test(document.location.pathname) ? new YavaMenuTr : new YavaMenu menu.addEvent(yava.event) if (document.location.hash === '#yavah-test') { new YavaTest(yava) } } } class YavaTest { constructor(yavah) { this.y = yavah this.indices = '1-1-1-2-2-2-3-3-3-4-4-4-5-5-5-6-6-6-7-7-7-1' this.values = 'A1;A1.1-A2-A3-B1;B1.1-B2-B3-C1;C1.1-C2-C3-D1;D1.1-D2-D3-E1;E1.1-E2-E3-F1;F1.1-F2-F3-G1;G1.1-G2-G3-' this.indicesSemi = '1-1-1-1-2-2-2-2-3-3-3-3-4-4-4-4-5-5-5-5-6-6-6-6-7-7-7-7-1' this.valuesSemi = this.values.replaceAll(';', '-') this.testInit() this.testRegular() this.testSemi() } testInit() { console.log('YAVAH TEST!') // asume if no inputs, then there is nothing to do if (!this.y.inputs) return // fill data const textareas = Array.from(document.querySelectorAll('.yavahtext')) textareas.forEach((t, i) => { const ch = String.fromCharCode(65 + i); t.value = `${ch}1;${ch}1.1\n${ch}2\n${ch}3` }) // go for it! this.y.event() } selects(indices) { // assert dropdowns console.group('YAVAH DROPDOWN TEST') const selects = document.querySelectorAll('#AddArtists select, #artistfields select') if (selects.length === indices.split('-').length) console.debug('[Passed] Valid number of dropdowns') else console.warn('Invalid number of dropdowns') if (Array.from(selects).map(_ => _.value).join('-') === indices) console.debug('[Passed] Valid values for dropdowns') else console.warn('Invalid values for dropdowns') console.groupEnd() } inputs(values) { // assert inputs console.group('YAVAH INPUT TEST') const inputs = document.querySelectorAll('input[name="aliasname[]"], input[name="artists[]"]') if (inputs.length === values.split('-').length) console.debug('[Passed] Valid number of inputs') else console.warn('Invalid number of inputs') if (Array.from(inputs).map(_ => _.value).join('-') === values) console.debug('[Passed] Valid values for inputs') else console.warn('Invalid values for inputs') console.groupEnd() } testRegular() { console.info('Starting Regular Test') this.selects(this.indices) this.inputs(this.values) } testSemi() { console.info('Starting Semicolon Test') YavaMenu.check.checked = true this.y.event() this.selects(this.indicesSemi) this.inputs(this.valuesSemi) } } Yavah.main() })()