musicbrainz-userscripts/set-recording-comments.user.js
Erik Levin 8bf2b04b6d
recording-comments: Run when tport (or other) query parameters are present
Fixes #563.

Co-authored-by: David Kellner <52860029+kellnerd@users.noreply.github.com>
2024-06-18 08:18:35 +02:00

224 lines
8.4 KiB
JavaScript

// ==UserScript==
// @name MusicBrainz: Set recording comments for a release
// @description Batch set recording comments from a Release page.
// @version 2024.6.18.1
// @author Michael Wiencek
// @license X11
// @namespace 790382e7-8714-47a7-bfbd-528d0caa2333
// @downloadURL https://raw.githubusercontent.com/murdos/musicbrainz-userscripts/master/set-recording-comments.user.js
// @updateURL https://raw.githubusercontent.com/murdos/musicbrainz-userscripts/master/set-recording-comments.user.js
// @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/release\/[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}(\/disc\/\d+|(\?.+?)?$)/
// @grant none
// @run-at document-idle
// ==/UserScript==
// ==License==
// Copyright (C) 2014 Michael Wiencek
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
//
// Except as contained in this notice, the name(s) of the above copyright
// holders shall not be used in advertising or otherwise to promote the sale,
// use or other dealings in this Software without prior written
// authorization.
// ==/License==
setTimeout(function () {
const scr = document.createElement('script');
scr.textContent = `$(${setRecordingComments});`;
document.body.appendChild(scr);
}, 1000);
function setRecordingComments() {
let $tracks;
let $inputs = $();
let EDIT_RECORDING_EDIT = 72;
$('head').append(
$('<style></style>').text(
'input.recording-comment { background: inherit; border: 1px #999 solid; width: 32em; margin-left: 0.5em; }'
)
);
const delay = setInterval(function () {
$tracks = $('.medium tbody tr[id]');
if ($tracks.length) {
clearInterval(delay);
} else {
return;
}
$tracks.each(function () {
let $td = $(this).children('td:not(.pos):not(.video):not(.rating):not(.treleases)').has('a[href^=\\/recording\\/]'),
node = $td.children('td > .mp, td > .name-variation, td > a[href^=\\/recording\\/]').filter(':first'),
$input = $('<input />').addClass('recording-comment').insertAfter(node);
let link = $("a[href^='/recording/']", $td).first().attr('href');
let mbid = link.match(MBID_REGEX)[0];
$input.data('mbid', mbid);
if (!editing) {
$input.hide();
}
$inputs = $inputs.add($input);
});
let release = location.pathname.match(MBID_REGEX)[0];
$.get(`/ws/2/release/${release}?inc=recordings&fmt=json`, function (data) {
let recordings = Array.from(data.media)
.map(medium => medium.tracks)
.flat()
.map(track => track.recording);
$inputs.each(function () {
let mbid = $(this).data('mbid');
let recording = recordings.find(recording => recording.id === mbid);
let comment = recording ? recording.disambiguation : '';
$(this).val(comment).data('old', comment);
});
});
}, 1000);
if (!location.pathname.match(/^\/release\/[a-f\d]{8}-[a-f\d]{4}-[a-f\d]{4}-[a-f\d]{4}-[a-f\d]{12}/)) {
return;
}
const MBID_REGEX = /[a-f\d]{8}-[a-f\d]{4}-[a-f\d]{4}-[a-f\d]{4}-[a-f\d]{12}/;
let editing = false,
activeRequest = null;
$('body').on('input.rc', '.recording-comment', function () {
$(this).css('border-color', this.value === $(this).data('old') ? '#999' : 'red');
});
let $container = $('<div></div>').insertAfter('h2.tracklist');
$('<button>Edit recording comments</button>')
.addClass('styled-button')
.on('click', function () {
editing = !editing;
$('#set-recording-comments').add($inputs).toggle(editing);
$(this).text(`${editing ? 'Hide' : 'Edit'} recording comments`);
if (editing) {
$('#all-recording-comments').focus();
}
})
.appendTo($container);
$container.append(
'\
<table id="set-recording-comments">\
<tr>\
<td><label for="all-recording-comments">Set all visible comments to:</label></td>\
<td><input type="text" id="all-recording-comments" style="width: 32em;"></td>\
</tr>\
<tr>\
<td><label for="recording-comments-edit-note">Edit note:</label></td>\
<td><textarea id="recording-comments-edit-note" style="width: 32em;" rows="5"></textarea></td>\
</tr>\
<tr>\
<td colspan="2" class="auto-editor">\
<label>\
<input id="make-recording-comments-votable" type="checkbox">\
Make all edits votable.\
</label>\
</td>\
</tr>\
<tr>\
<td colspan="2">\
<button id="submit-recording-comments" class="styled-button">Submit changes (visible and marked red)</button>\
</td>\
</tr>\
</table>'
);
$('#set-recording-comments').hide();
$('#all-recording-comments').on('input', function () {
$inputs.filter(':visible').val(this.value).trigger('input.rc');
});
const $submitButton = $('#submit-recording-comments').on('click', function () {
if (activeRequest) {
activeRequest.abort();
activeRequest = null;
$submitButton.text('Submit changes (marked red)');
$inputs.prop('disabled', false).trigger('input.rc');
return;
}
$submitButton.text('Submitting...click to cancel!');
$inputs.prop('disabled', true);
let editData = [],
deferred = $.Deferred();
$.each($tracks, function (i, track) {
if ($(track).filter(':visible').length > 0) {
let $input = $inputs.eq(i),
comment = $input.val();
if (comment === $input.data('old')) {
$input.prop('disabled', false);
return;
}
deferred
.done(function () {
$input.data('old', comment).trigger('input.rc').prop('disabled', false);
})
.fail(function () {
$input.css('border-color', 'red').prop('disabled', false);
});
let link = track.querySelector("td a[href^='/recording/']"),
mbid = link.href.match(MBID_REGEX)[0];
editData.push({ edit_type: EDIT_RECORDING_EDIT, to_edit: mbid, comment: comment });
}
});
if (editData.length === 0) {
$inputs.prop('disabled', false);
$submitButton.prop('disabled', false).text('Submit changes (marked red)');
} else {
let editNote = $('#recording-comments-edit-note').val();
let makeVotable = document.getElementById('make-recording-comments-votable').checked;
activeRequest = $.ajax({
type: 'POST',
url: '/ws/js/edit/create',
dataType: 'json',
data: JSON.stringify({ edits: editData, editNote: editNote, makeVotable: makeVotable }),
contentType: 'application/json; charset=utf-8',
})
.always(function () {
$submitButton.prop('disabled', false).text('Submit changes (marked red)');
})
.done(function () {
deferred.resolve();
})
.fail(function () {
deferred.reject();
});
}
});
}