您需要先安装一款用户样式管理器扩展(如 Stylus)后才能安装此样式。
您需要先安装一款用户样式管理器扩展(如 Stylus)后才能安装此样式。
您需要先安装一款用户样式管理器扩展(如 Stylus)后才能安装此样式。
您需要先安装一款用户样式管理器扩展后才能安装此样式。
您需要先安装一款用户样式管理器扩展后才能安装此样式。
您需要先安装一款用户样式管理器扩展后才能安装此样式。
(我已经安装了用户样式管理器,让我安装!)
// ==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()
})()