Fixed #31523 -- Removed jQuery dependency from actions.js.

This commit is contained in:
Jon Dufresne 2020-06-24 11:13:51 +02:00 committed by Carlton Gibson
parent 074844e947
commit 30e59705fc
7 changed files with 178 additions and 158 deletions

View File

@ -211,6 +211,10 @@ p img, h1 img, h2 img, h3 img, h4 img, td img {
white-space: nowrap; white-space: nowrap;
} }
.hidden {
display: none;
}
/* TABLES */ /* TABLES */
table { table {

View File

@ -300,7 +300,6 @@
#changelist .actions span.question { #changelist .actions span.question {
font-size: 13px; font-size: 13px;
margin: 0 0.5em; margin: 0 0.5em;
display: none;
} }
#changelist .actions:last-child { #changelist .actions:last-child {

View File

@ -22,10 +22,6 @@ form .form-row p {
padding-left: 0; padding-left: 0;
} }
.hidden {
display: none;
}
/* FORM LABELS */ /* FORM LABELS */
label { label {

View File

@ -1,154 +1,170 @@
/*global gettext, interpolate, ngettext*/ /*global gettext, interpolate, ngettext*/
'use strict'; 'use strict';
{ {
const $ = django.jQuery; function show(selector) {
let lastChecked; document.querySelectorAll(selector).forEach(function(el) {
el.classList.remove('hidden');
});
}
$.fn.actions = function(opts) { function hide(selector) {
const options = $.extend({}, $.fn.actions.defaults, opts); document.querySelectorAll(selector).forEach(function(el) {
const actionCheckboxes = $(this); el.classList.add('hidden');
let list_editable_changed = false;
const showQuestion = function() {
$(options.acrossClears).hide();
$(options.acrossQuestions).show();
$(options.allContainer).hide();
},
showClear = function() {
$(options.acrossClears).show();
$(options.acrossQuestions).hide();
$(options.actionContainer).toggleClass(options.selectedClass);
$(options.allContainer).show();
$(options.counterContainer).hide();
},
reset = function() {
$(options.acrossClears).hide();
$(options.acrossQuestions).hide();
$(options.allContainer).hide();
$(options.counterContainer).show();
},
clearAcross = function() {
reset();
$(options.acrossInput).val(0);
$(options.actionContainer).removeClass(options.selectedClass);
},
checker = function(checked) {
if (checked) {
showQuestion();
} else {
reset();
}
$(actionCheckboxes).prop("checked", checked)
.parent().parent().toggleClass(options.selectedClass, checked);
},
updateCounter = function() {
const sel = $(actionCheckboxes).filter(":checked").length;
// data-actions-icnt is defined in the generated HTML
// and contains the total amount of objects in the queryset
const actions_icnt = $('.action-counter').data('actionsIcnt');
$(options.counterContainer).html(interpolate(
ngettext('%(sel)s of %(cnt)s selected', '%(sel)s of %(cnt)s selected', sel), {
sel: sel,
cnt: actions_icnt
}, true));
$(options.allToggle).prop("checked", function() {
let value;
if (sel === actionCheckboxes.length) {
value = true;
showQuestion();
} else {
value = false;
clearAcross();
}
return value;
});
};
// Show counter by default
$(options.counterContainer).show();
// Check state of checkboxes and reinit state if needed
$(this).filter(":checked").each(function(i) {
$(this).parent().parent().toggleClass(options.selectedClass);
updateCounter();
if ($(options.acrossInput).val() === 1) {
showClear();
}
}); });
$(options.allToggle).show().on('click', function() { }
checker($(this).prop("checked"));
updateCounter(); function showQuestion(options) {
hide(options.acrossClears);
show(options.acrossQuestions);
hide(options.allContainer);
}
function showClear(options) {
show(options.acrossClears);
hide(options.acrossQuestions);
document.querySelector(options.actionContainer).classList.remove(options.selectedClass);
show(options.allContainer);
hide(options.counterContainer);
}
function reset(options) {
hide(options.acrossClears);
hide(options.acrossQuestions);
hide(options.allContainer);
show(options.counterContainer);
}
function clearAcross(options) {
reset(options);
document.querySelector(options.acrossInput).value = 0;
document.querySelector(options.actionContainer).classList.remove(options.selectedClass);
}
function checker(actionCheckboxes, options, checked) {
if (checked) {
showQuestion(options);
} else {
reset(options);
}
actionCheckboxes.forEach(function(el) {
el.checked = checked;
el.closest('tr').classList.toggle(options.selectedClass, checked);
}); });
$("a", options.acrossQuestions).on('click', function(event) { }
event.preventDefault();
$(options.acrossInput).val(1); function updateCounter(actionCheckboxes, options) {
showClear(); const sel = Array.from(actionCheckboxes).filter(function(el) {
}); return el.checked;
$("a", options.acrossClears).on('click', function(event) { }).length;
event.preventDefault(); const counter = document.querySelector(options.counterContainer);
$(options.allToggle).prop("checked", false); // data-actions-icnt is defined in the generated HTML
clearAcross(); // and contains the total amount of objects in the queryset
checker(0); const actions_icnt = Number(counter.dataset.actionsIcnt);
updateCounter(); counter.textContent = interpolate(
}); ngettext('%(sel)s of %(cnt)s selected', '%(sel)s of %(cnt)s selected', sel), {
lastChecked = null; sel: sel,
$(actionCheckboxes).on('click', function(event) { cnt: actions_icnt
if (!event) { event = window.event; } }, true);
const target = event.target ? event.target : event.srcElement; const allToggle = document.getElementById(options.allToggleId);
if (lastChecked && $.data(lastChecked) !== $.data(target) && event.shiftKey === true) { allToggle.checked = sel === actionCheckboxes.length;
let inrange = false; if (allToggle.checked) {
$(lastChecked).prop("checked", target.checked) showQuestion(options);
.parent().parent().toggleClass(options.selectedClass, target.checked); } else {
$(actionCheckboxes).each(function() { clearAcross(options);
if ($.data(this) === $.data(lastChecked) || $.data(this) === $.data(target)) { }
inrange = (inrange) ? false : true; }
}
if (inrange) { const defaults = {
$(this).prop("checked", target.checked)
.parent().parent().toggleClass(options.selectedClass, target.checked);
}
});
}
$(target).parent().parent().toggleClass(options.selectedClass, target.checked);
lastChecked = target;
updateCounter();
});
$('form#changelist-form table#result_list tr').on('change', 'td:gt(0) :input', function() {
list_editable_changed = true;
});
$('form#changelist-form button[name="index"]').on('click', function(event) {
if (list_editable_changed) {
return confirm(gettext("You have unsaved changes on individual editable fields. If you run an action, your unsaved changes will be lost."));
}
});
$('form#changelist-form input[name="_save"]').on('click', function(event) {
let action_changed = false;
$('select option:selected', options.actionContainer).each(function() {
if ($(this).val()) {
action_changed = true;
}
});
if (action_changed) {
if (list_editable_changed) {
return confirm(gettext("You have selected an action, but you havent saved your changes to individual fields yet. Please click OK to save. Youll need to re-run the action."));
} else {
return confirm(gettext("You have selected an action, and you havent made any changes on individual fields. Youre probably looking for the Go button rather than the Save button."));
}
}
});
};
/* Setup plugin defaults */
$.fn.actions.defaults = {
actionContainer: "div.actions", actionContainer: "div.actions",
counterContainer: "span.action-counter", counterContainer: "span.action-counter",
allContainer: "div.actions span.all", allContainer: "div.actions span.all",
acrossInput: "div.actions input.select-across", acrossInput: "div.actions input.select-across",
acrossQuestions: "div.actions span.question", acrossQuestions: "div.actions span.question",
acrossClears: "div.actions span.clear", acrossClears: "div.actions span.clear",
allToggle: "#action-toggle", allToggleId: "action-toggle",
selectedClass: "selected" selectedClass: "selected"
}; };
$(document).ready(function() {
const $actionsEls = $('tr input.action-select'); window.Actions = function(actionCheckboxes, options) {
if ($actionsEls.length > 0) { options = Object.assign({}, defaults, options);
$actionsEls.actions(); let list_editable_changed = false;
document.getElementById(options.allToggleId).addEventListener('click', function(event) {
checker(actionCheckboxes, options, this.checked);
updateCounter(actionCheckboxes, options);
});
document.querySelectorAll(options.acrossQuestions + " a").forEach(function(el) {
el.addEventListener('click', function(event) {
event.preventDefault();
const acrossInput = document.querySelector(options.acrossInput);
acrossInput.value = 1;
showClear(options);
});
});
document.querySelectorAll(options.acrossClears + " a").forEach(function(el) {
el.addEventListener('click', function(event) {
event.preventDefault();
document.getElementById(options.allToggleId).checked = false;
clearAcross(options);
checker(actionCheckboxes, options, false);
updateCounter(actionCheckboxes, options);
});
});
Array.from(document.getElementById('result_list').tBodies).forEach(function(el) {
el.addEventListener('change', function(event) {
const target = event.target;
if (target.classList.contains('action-select')) {
target.closest('tr').classList.toggle(options.selectedClass, target.checked);
updateCounter(actionCheckboxes, options);
} else {
list_editable_changed = true;
}
});
});
document.querySelector('#changelist-form button[name=index]').addEventListener('click', function() {
if (list_editable_changed) {
const confirmed = confirm(gettext("You have unsaved changes on individual editable fields. If you run an action, your unsaved changes will be lost."));
if (!confirmed) {
event.preventDefault();
}
}
});
const el = document.querySelector('#changelist-form input[name=_save]');
// The button does not exist if no fields are editable.
if (el) {
el.addEventListener('click', function(event) {
if (document.querySelector('[name=action]').value) {
const text = list_editable_changed
? gettext("You have selected an action, but you havent saved your changes to individual fields yet. Please click OK to save. Youll need to re-run the action.")
: gettext("You have selected an action, and you havent made any changes on individual fields. Youre probably looking for the Go button rather than the Save button.");
if (!confirm(text)) {
event.preventDefault();
}
}
});
}
};
// Call function fn when the DOM is loaded and ready. If it is already
// loaded, call the function now.
// http://youmightnotneedjquery.com/#ready
function ready(fn) {
if (document.readyState !== 'loading') {
fn();
} else {
document.addEventListener('DOMContentLoaded', fn);
}
}
ready(function() {
const actionsEls = document.querySelectorAll('tr input.action-select');
if (actionsEls.length > 0) {
Actions(actionsEls);
} }
}); });
} }

View File

@ -11,11 +11,11 @@
{% if actions_selection_counter %} {% if actions_selection_counter %}
<span class="action-counter" data-actions-icnt="{{ cl.result_list|length }}">{{ selection_note }}</span> <span class="action-counter" data-actions-icnt="{{ cl.result_list|length }}">{{ selection_note }}</span>
{% if cl.result_count != cl.result_list|length %} {% if cl.result_count != cl.result_list|length %}
<span class="all">{{ selection_note_all }}</span> <span class="all hidden">{{ selection_note_all }}</span>
<span class="question"> <span class="question hidden">
<a href="#" title="{% translate "Click here to select the objects across all pages" %}">{% blocktranslate with cl.result_count as total_count %}Select all {{ total_count }} {{ module_name }}{% endblocktranslate %}</a> <a href="#" title="{% translate "Click here to select the objects across all pages" %}">{% blocktranslate with cl.result_count as total_count %}Select all {{ total_count }} {{ module_name }}{% endblocktranslate %}</a>
</span> </span>
<span class="clear"><a href="#">{% translate "Clear selection" %}</a></span> <span class="clear hidden"><a href="#">{% translate "Clear selection" %}</a></span>
{% endif %} {% endif %}
{% endif %} {% endif %}
{% endblock %} {% endblock %}

View File

@ -11,7 +11,7 @@ QUnit.module('admin.actions', {
const $ = django.jQuery; const $ = django.jQuery;
$('#qunit-fixture').append($('#result-table').text()); $('#qunit-fixture').append($('#result-table').text());
$('tr input.action-select').actions(); Actions(document.querySelectorAll('tr input.action-select'));
} }
}); });

View File

@ -11,18 +11,23 @@
<div id="qunit-fixture"> <div id="qunit-fixture">
</div> </div>
<script type="text/html" id="result-table"> <script type="text/html" id="result-table">
<table id="result_list"> <form id="changelist-form">
<tr> <button type="submit" class="button" name="index" value="0">Go</button>
<th> <span class="action-counter" data-actions-icnt="100"></span>
<input type="checkbox" id="action-toggle"> <table id="result_list">
</th> <tr>
</tr> <th>
<tr> <input type="checkbox" id="action-toggle">
<td class="action-checkbox"> </th>
<input class="action-select" type="checkbox" value="618"> </tr>
</td> <tr>
</tr> <td class="action-checkbox">
</table> <input class="action-select" type="checkbox" value="618">
</td>
</tr>
</table>
<input type="submit" name="_save" value="Save">
</form>
</script> </script>
<script type="text/html" id="tabular-formset"> <script type="text/html" id="tabular-formset">
<input id="id_first-TOTAL_FORMS" value="1"> <input id="id_first-TOTAL_FORMS" value="1">