Fix timestamps not sorting in datatables
Timestamps retrieved from the API were always converted to a browser
local format. The format specified for moment.js added in
5160eff294
did not work because of this.
Additionally, the format specified used `dd` which looks for two letter
days, such as "Mo", "Tu", "We", etc. Furthermore, `mm` is used for
minutes, not months.
Because the locale formatted datetime can vary a lot, it is not easy to
get this into moment.js to enable the sorting of datetimes in the
datatables. In other words, there is no conversion from an
`Intl.DateTimeFormat` specifier string to moment.js. Adding many
`$.fn.dataTable.moment(format);` with different `format`s is not useful.
I have fixed this rewriting how the timestamps from the API are added
to the tables. It still uses the locale of the browser, because not
everyone wants to use ISO 8601, but no longer requires moment.js (which
has been removed).
Two data attributes are added to the `td`s of the timestamps:
- `data-order`
- `data-sort`
The values of these are the timestamps as returned by the server, which
are very easily sorted (as they are just UNIX timestamps). Then, when
creating the cell in the table, it will be converted to what the locale
of the browser specified (this has not changed).
This commit is contained in:
parent
55d57c552d
commit
240b2c63f6
7
data/web/js/build/004-moment.min.js
vendored
7
data/web/js/build/004-moment.min.js
vendored
File diff suppressed because one or more lines are too long
@ -1,70 +0,0 @@
|
|||||||
/**
|
|
||||||
* This plug-in for DataTables represents the ultimate option in extensibility
|
|
||||||
* for sorting date / time strings correctly. It uses
|
|
||||||
* [Moment.js](http://momentjs.com) to create automatic type detection and
|
|
||||||
* sorting plug-ins for DataTables based on a given format. This way, DataTables
|
|
||||||
* will automatically detect your temporal information and sort it correctly.
|
|
||||||
*
|
|
||||||
* For usage instructions, please see the DataTables blog
|
|
||||||
* post that [introduces it](//datatables.net/blog/2014-12-18).
|
|
||||||
*
|
|
||||||
* @name Ultimate Date / Time sorting
|
|
||||||
* @summary Sort date and time in any format using Moment.js
|
|
||||||
* @author [Allan Jardine](//datatables.net)
|
|
||||||
* @depends DataTables 1.10+, Moment.js 1.7+
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* $.fn.dataTable.moment( 'HH:mm MMM D, YY' );
|
|
||||||
* $.fn.dataTable.moment( 'dddd, MMMM Do, YYYY' );
|
|
||||||
*
|
|
||||||
* $('#example').DataTable();
|
|
||||||
*/
|
|
||||||
|
|
||||||
(function (factory) {
|
|
||||||
if (typeof define === "function" && define.amd) {
|
|
||||||
define(["jquery", "moment", "datatables.net"], factory);
|
|
||||||
} else {
|
|
||||||
factory(jQuery, moment);
|
|
||||||
}
|
|
||||||
}(function ($, moment) {
|
|
||||||
|
|
||||||
function strip (d) {
|
|
||||||
if ( typeof d === 'string' ) {
|
|
||||||
// Strip HTML tags and newline characters if possible
|
|
||||||
d = d.replace(/(<.*?>)|(\r?\n|\r)/g, '');
|
|
||||||
|
|
||||||
// Strip out surrounding white space
|
|
||||||
d = d.trim();
|
|
||||||
}
|
|
||||||
|
|
||||||
return d;
|
|
||||||
}
|
|
||||||
|
|
||||||
$.fn.dataTable.moment = function ( format, locale, reverseEmpties ) {
|
|
||||||
var types = $.fn.dataTable.ext.type;
|
|
||||||
|
|
||||||
// Add type detection
|
|
||||||
types.detect.unshift( function ( d ) {
|
|
||||||
d = strip(d);
|
|
||||||
|
|
||||||
// Null and empty values are acceptable
|
|
||||||
if ( d === '' || d === null ) {
|
|
||||||
return 'moment-'+format;
|
|
||||||
}
|
|
||||||
|
|
||||||
return moment( d, format, locale, true ).isValid() ?
|
|
||||||
'moment-'+format :
|
|
||||||
null;
|
|
||||||
} );
|
|
||||||
|
|
||||||
// Add sorting method - use an integer for the sorting
|
|
||||||
types.order[ 'moment-'+format+'-pre' ] = function ( d ) {
|
|
||||||
d = strip(d);
|
|
||||||
|
|
||||||
return !moment(d, format, locale, true).isValid() ?
|
|
||||||
(reverseEmpties ? -Infinity : Infinity) :
|
|
||||||
parseInt( moment( d, format, locale, true ).format( 'x' ), 10 );
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
}));
|
|
@ -1,354 +1,353 @@
|
|||||||
$(document).ready(function() {
|
$(document).ready(function() {
|
||||||
// mailcow alert box generator
|
// mailcow alert box generator
|
||||||
window.mailcow_alert_box = function(message, type) {
|
window.mailcow_alert_box = function(message, type) {
|
||||||
msg = $('<span/>').text(message).text();
|
msg = $('<span/>').text(message).text();
|
||||||
if (type == 'danger' || type == 'info') {
|
if (type == 'danger' || type == 'info') {
|
||||||
auto_hide = 0;
|
auto_hide = 0;
|
||||||
$('#' + localStorage.getItem("add_modal")).modal('show');
|
$('#' + localStorage.getItem("add_modal")).modal('show');
|
||||||
localStorage.removeItem("add_modal");
|
localStorage.removeItem("add_modal");
|
||||||
} else {
|
} else {
|
||||||
auto_hide = 5000;
|
auto_hide = 5000;
|
||||||
}
|
}
|
||||||
$.notify({message: msg},{z_index: 20000, delay: auto_hide, type: type,placement: {from: "bottom",align: "right"},animate: {enter: 'animated fadeInUp',exit: 'animated fadeOutDown'}});
|
$.notify({message: msg},{z_index: 20000, delay: auto_hide, type: type,placement: {from: "bottom",align: "right"},animate: {enter: 'animated fadeInUp',exit: 'animated fadeOutDown'}});
|
||||||
}
|
}
|
||||||
|
|
||||||
$(".generate_password").click(function( event ) {
|
$(".generate_password").click(function( event ) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
$('[data-hibp]').trigger('input');
|
$('[data-hibp]').trigger('input');
|
||||||
if (typeof($(this).closest("form").data('pwgen-length')) == "number") {
|
if (typeof($(this).closest("form").data('pwgen-length')) == "number") {
|
||||||
var random_passwd = GPW.pronounceable($(this).closest("form").data('pwgen-length'))
|
var random_passwd = GPW.pronounceable($(this).closest("form").data('pwgen-length'))
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
var random_passwd = GPW.pronounceable(8)
|
var random_passwd = GPW.pronounceable(8)
|
||||||
}
|
}
|
||||||
$(this).closest("form").find('[data-pwgen-field]').attr('type', 'text');
|
$(this).closest("form").find('[data-pwgen-field]').attr('type', 'text');
|
||||||
$(this).closest("form").find('[data-pwgen-field]').val(random_passwd);
|
$(this).closest("form").find('[data-pwgen-field]').val(random_passwd);
|
||||||
});
|
});
|
||||||
function str_rot13(str) {
|
function str_rot13(str) {
|
||||||
return (str + '').replace(/[a-z]/gi, function(s){
|
return (str + '').replace(/[a-z]/gi, function(s){
|
||||||
return String.fromCharCode(s.charCodeAt(0) + (s.toLowerCase() < 'n' ? 13 : -13))
|
return String.fromCharCode(s.charCodeAt(0) + (s.toLowerCase() < 'n' ? 13 : -13))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
$(".rot-enc").html(function(){
|
$(".rot-enc").html(function(){
|
||||||
return str_rot13($(this).html())
|
return str_rot13($(this).html())
|
||||||
});
|
});
|
||||||
// https://stackoverflow.com/questions/4399005/implementing-jquerys-shake-effect-with-animate
|
// https://stackoverflow.com/questions/4399005/implementing-jquerys-shake-effect-with-animate
|
||||||
function shake(div,interval,distance,times) {
|
function shake(div,interval,distance,times) {
|
||||||
if(typeof interval === 'undefined') {
|
if(typeof interval === 'undefined') {
|
||||||
interval = 100;
|
interval = 100;
|
||||||
}
|
}
|
||||||
if(typeof distance === 'undefined') {
|
if(typeof distance === 'undefined') {
|
||||||
distance = 10;
|
distance = 10;
|
||||||
}
|
}
|
||||||
if(typeof times === 'undefined') {
|
if(typeof times === 'undefined') {
|
||||||
times = 4;
|
times = 4;
|
||||||
}
|
}
|
||||||
$(div).css('position','relative');
|
$(div).css('position','relative');
|
||||||
for(var iter=0;iter<(times+1);iter++){
|
for(var iter=0;iter<(times+1);iter++){
|
||||||
$(div).animate({ left: ((iter%2==0 ? distance : distance*-1))}, interval);
|
$(div).animate({ left: ((iter%2==0 ? distance : distance*-1))}, interval);
|
||||||
}
|
}
|
||||||
$(div).animate({ left: 0},interval);
|
$(div).animate({ left: 0},interval);
|
||||||
}
|
}
|
||||||
|
|
||||||
// form cache
|
// form cache
|
||||||
$('[data-cached-form="true"]').formcache({key: $(this).data('id')});
|
$('[data-cached-form="true"]').formcache({key: $(this).data('id')});
|
||||||
|
|
||||||
// tooltips
|
// tooltips
|
||||||
$(function () {
|
$(function () {
|
||||||
$('[data-bs-toggle="tooltip"]').tooltip()
|
$('[data-bs-toggle="tooltip"]').tooltip()
|
||||||
});
|
});
|
||||||
|
|
||||||
// remember last navigation pill
|
// remember last navigation pill
|
||||||
(function () {
|
(function () {
|
||||||
'use strict';
|
'use strict';
|
||||||
// remember desktop tabs
|
// remember desktop tabs
|
||||||
$('button[data-bs-toggle="tab"]').on('click', function (e) {
|
$('button[data-bs-toggle="tab"]').on('click', function (e) {
|
||||||
if ($(this).data('dont-remember') == 1) {
|
if ($(this).data('dont-remember') == 1) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
var id = $(this).parents('[role="tablist"]').attr('id');
|
var id = $(this).parents('[role="tablist"]').attr('id');
|
||||||
var key = 'lastTag';
|
var key = 'lastTag';
|
||||||
if (id) {
|
if (id) {
|
||||||
key += ':' + id;
|
key += ':' + id;
|
||||||
}
|
}
|
||||||
|
|
||||||
var tab_id = $(e.target).attr('data-bs-target').substring(1);
|
var tab_id = $(e.target).attr('data-bs-target').substring(1);
|
||||||
localStorage.setItem(key, tab_id);
|
localStorage.setItem(key, tab_id);
|
||||||
});
|
});
|
||||||
// remember mobile tabs
|
// remember mobile tabs
|
||||||
$('button[data-bs-target^="#collapse-tab-"]').on('click', function (e) {
|
$('button[data-bs-target^="#collapse-tab-"]').on('click', function (e) {
|
||||||
// only remember tab if its being opened
|
// only remember tab if its being opened
|
||||||
if ($(this).hasClass('collapsed')) return false;
|
if ($(this).hasClass('collapsed')) return false;
|
||||||
var tab_id = $(this).closest('div[role="tabpanel"]').attr('id');
|
var tab_id = $(this).closest('div[role="tabpanel"]').attr('id');
|
||||||
|
|
||||||
if ($(this).data('dont-remember') == 1) {
|
if ($(this).data('dont-remember') == 1) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
var id = $(this).parents('[role="tablist"]').attr('id');;
|
var id = $(this).parents('[role="tablist"]').attr('id');;
|
||||||
var key = 'lastTag';
|
var key = 'lastTag';
|
||||||
if (id) {
|
if (id) {
|
||||||
key += ':' + id;
|
key += ':' + id;
|
||||||
}
|
}
|
||||||
|
|
||||||
localStorage.setItem(key, tab_id);
|
localStorage.setItem(key, tab_id);
|
||||||
});
|
});
|
||||||
// open last tab
|
// open last tab
|
||||||
$('[role="tablist"]').each(function (idx, elem) {
|
$('[role="tablist"]').each(function (idx, elem) {
|
||||||
var id = $(elem).attr('id');
|
var id = $(elem).attr('id');
|
||||||
var key = 'lastTag';
|
var key = 'lastTag';
|
||||||
if (id) {
|
if (id) {
|
||||||
key += ':' + id;
|
key += ':' + id;
|
||||||
}
|
}
|
||||||
var lastTab = localStorage.getItem(key);
|
var lastTab = localStorage.getItem(key);
|
||||||
if (lastTab) {
|
if (lastTab) {
|
||||||
$('[data-bs-target="#' + lastTab + '"]').click();
|
$('[data-bs-target="#' + lastTab + '"]').click();
|
||||||
var tab = $('[id^="' + lastTab + '"]');
|
var tab = $('[id^="' + lastTab + '"]');
|
||||||
$(tab).find('.card-body.collapse').collapse('show');
|
$(tab).find('.card-body.collapse').collapse('show');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
})();
|
})();
|
||||||
|
|
||||||
// IE fix to hide scrollbars when table body is empty
|
// IE fix to hide scrollbars when table body is empty
|
||||||
$('tbody').filter(function (index) {
|
$('tbody').filter(function (index) {
|
||||||
return $(this).children().length < 1;
|
return $(this).children().length < 1;
|
||||||
}).remove();
|
}).remove();
|
||||||
|
|
||||||
// selectpicker
|
// selectpicker
|
||||||
$('select').selectpicker({
|
$('select').selectpicker({
|
||||||
'styleBase': 'btn btn-xs-lg',
|
'styleBase': 'btn btn-xs-lg',
|
||||||
'noneSelectedText': lang_footer.nothing_selected
|
'noneSelectedText': lang_footer.nothing_selected
|
||||||
});
|
});
|
||||||
|
|
||||||
// haveibeenpwned and passwd policy
|
// haveibeenpwned and passwd policy
|
||||||
$.ajax({
|
$.ajax({
|
||||||
url: '/api/v1/get/passwordpolicy/html',
|
url: '/api/v1/get/passwordpolicy/html',
|
||||||
type: 'GET',
|
type: 'GET',
|
||||||
success: function(res) {
|
success: function(res) {
|
||||||
$(".hibp-out").after(res);
|
$(".hibp-out").after(res);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
$('[data-hibp]').after('<p class="small haveibeenpwned"><i class="bi bi-shield-fill-exclamation"></i> ' + lang_footer.hibp_check + '</p><span class="hibp-out"></span>');
|
$('[data-hibp]').after('<p class="small haveibeenpwned"><i class="bi bi-shield-fill-exclamation"></i> ' + lang_footer.hibp_check + '</p><span class="hibp-out"></span>');
|
||||||
$('[data-hibp]').on('input', function() {
|
$('[data-hibp]').on('input', function() {
|
||||||
out_field = $(this).next('.haveibeenpwned').next('.hibp-out').text('').attr('class', 'hibp-out');
|
out_field = $(this).next('.haveibeenpwned').next('.hibp-out').text('').attr('class', 'hibp-out');
|
||||||
});
|
});
|
||||||
$('.haveibeenpwned:not(.task-running)').on('click', function() {
|
$('.haveibeenpwned:not(.task-running)').on('click', function() {
|
||||||
var hibp_field = $(this)
|
var hibp_field = $(this)
|
||||||
$(hibp_field).addClass('task-running');
|
$(hibp_field).addClass('task-running');
|
||||||
var hibp_result = $(hibp_field).next('.hibp-out')
|
var hibp_result = $(hibp_field).next('.hibp-out')
|
||||||
var password_field = $(this).prev('[data-hibp]')
|
var password_field = $(this).prev('[data-hibp]')
|
||||||
if ($(password_field).val() == '') {
|
if ($(password_field).val() == '') {
|
||||||
shake(password_field);
|
shake(password_field);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
$(hibp_result).attr('class', 'hibp-out badge fs-5 bg-info');
|
$(hibp_result).attr('class', 'hibp-out badge fs-5 bg-info');
|
||||||
$(hibp_result).text(lang_footer.loading);
|
$(hibp_result).text(lang_footer.loading);
|
||||||
var password_digest = $.sha1($(password_field).val())
|
var password_digest = $.sha1($(password_field).val())
|
||||||
var digest_five = password_digest.substring(0, 5).toUpperCase();
|
var digest_five = password_digest.substring(0, 5).toUpperCase();
|
||||||
var queryURL = "https://api.pwnedpasswords.com/range/" + digest_five;
|
var queryURL = "https://api.pwnedpasswords.com/range/" + digest_five;
|
||||||
var compl_digest = password_digest.substring(5, 41).toUpperCase();
|
var compl_digest = password_digest.substring(5, 41).toUpperCase();
|
||||||
$.ajax({
|
$.ajax({
|
||||||
url: queryURL,
|
url: queryURL,
|
||||||
type: 'GET',
|
type: 'GET',
|
||||||
success: function(res) {
|
success: function(res) {
|
||||||
if (res.search(compl_digest) > -1){
|
if (res.search(compl_digest) > -1){
|
||||||
$(hibp_result).removeClass('badge fs-5 bg-info').addClass('badge fs-5 bg-danger');
|
$(hibp_result).removeClass('badge fs-5 bg-info').addClass('badge fs-5 bg-danger');
|
||||||
$(hibp_result).text(lang_footer.hibp_nok)
|
$(hibp_result).text(lang_footer.hibp_nok)
|
||||||
} else {
|
} else {
|
||||||
$(hibp_result).removeClass('badge fs-5 bg-info').addClass('badge fs-5 bg-success');
|
$(hibp_result).removeClass('badge fs-5 bg-info').addClass('badge fs-5 bg-success');
|
||||||
$(hibp_result).text(lang_footer.hibp_ok)
|
$(hibp_result).text(lang_footer.hibp_ok)
|
||||||
}
|
}
|
||||||
$(hibp_field).removeClass('task-running');
|
$(hibp_field).removeClass('task-running');
|
||||||
},
|
},
|
||||||
error: function(xhr, status, error) {
|
error: function(xhr, status, error) {
|
||||||
$(hibp_result).removeClass('badge fs-5 bg-info').addClass('badge fs-5 bg-warning');
|
$(hibp_result).removeClass('badge fs-5 bg-info').addClass('badge fs-5 bg-warning');
|
||||||
$(hibp_result).text('API error: ' + xhr.responseText)
|
$(hibp_result).text('API error: ' + xhr.responseText)
|
||||||
$(hibp_field).removeClass('task-running');
|
$(hibp_field).removeClass('task-running');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Disable disallowed inputs
|
// Disable disallowed inputs
|
||||||
$('[data-acl="0"]').each(function(event){
|
$('[data-acl="0"]').each(function(event){
|
||||||
if ($(this).is("a")) {
|
if ($(this).is("a")) {
|
||||||
$(this).removeAttr("data-bs-toggle");
|
$(this).removeAttr("data-bs-toggle");
|
||||||
$(this).removeAttr("data-bs-target");
|
$(this).removeAttr("data-bs-target");
|
||||||
$(this).removeAttr("data-action");
|
$(this).removeAttr("data-action");
|
||||||
$(this).click(function(event) {
|
$(this).click(function(event) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if ($(this).is("select")) {
|
if ($(this).is("select")) {
|
||||||
$(this).selectpicker('destroy');
|
$(this).selectpicker('destroy');
|
||||||
$(this).replaceWith(function() {
|
$(this).replaceWith(function() {
|
||||||
return '<label class="control-label"><b>' + this.innerText + '</b></label>';
|
return '<label class="control-label"><b>' + this.innerText + '</b></label>';
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if ($(this).hasClass('btn-group')) {
|
if ($(this).hasClass('btn-group')) {
|
||||||
$(this).find('a').each(function(){
|
$(this).find('a').each(function(){
|
||||||
$(this).removeClass('dropdown-toggle')
|
$(this).removeClass('dropdown-toggle')
|
||||||
.removeAttr('data-bs-toggle')
|
.removeAttr('data-bs-toggle')
|
||||||
.removeAttr('data-bs-target')
|
.removeAttr('data-bs-target')
|
||||||
.removeAttr('data-action')
|
.removeAttr('data-action')
|
||||||
.removeAttr('id')
|
.removeAttr('id')
|
||||||
.attr("disabled", true);
|
.attr("disabled", true);
|
||||||
$(this).click(function(event) {
|
$(this).click(function(event) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
return;
|
return;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
$(this).find('button').each(function() {
|
$(this).find('button').each(function() {
|
||||||
$(this).attr("disabled", true);
|
$(this).attr("disabled", true);
|
||||||
});
|
});
|
||||||
} else if ($(this).hasClass('input-group')) {
|
} else if ($(this).hasClass('input-group')) {
|
||||||
$(this).find('input').each(function() {
|
$(this).find('input').each(function() {
|
||||||
$(this).removeClass('dropdown-toggle')
|
$(this).removeClass('dropdown-toggle')
|
||||||
.removeAttr('data-bs-toggle')
|
.removeAttr('data-bs-toggle')
|
||||||
.attr("disabled", true);
|
.attr("disabled", true);
|
||||||
$(this).click(function(event) {
|
$(this).click(function(event) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
$(this).find('button').each(function() {
|
$(this).find('button').each(function() {
|
||||||
$(this).attr("disabled", true);
|
$(this).attr("disabled", true);
|
||||||
});
|
});
|
||||||
} else if ($(this).hasClass('form-group')) {
|
} else if ($(this).hasClass('form-group')) {
|
||||||
$(this).find('input').each(function() {
|
$(this).find('input').each(function() {
|
||||||
$(this).attr("disabled", true);
|
$(this).attr("disabled", true);
|
||||||
});
|
});
|
||||||
} else if ($(this).hasClass('btn')) {
|
} else if ($(this).hasClass('btn')) {
|
||||||
$(this).attr("disabled", true);
|
$(this).attr("disabled", true);
|
||||||
} else if ($(this).attr('data-provide') == 'slider') {
|
} else if ($(this).attr('data-provide') == 'slider') {
|
||||||
$(this).attr('disabled', true);
|
$(this).attr('disabled', true);
|
||||||
} else if ($(this).is(':checkbox')) {
|
} else if ($(this).is(':checkbox')) {
|
||||||
$(this).attr("disabled", true);
|
$(this).attr("disabled", true);
|
||||||
}
|
}
|
||||||
$(this).data("toggle", "tooltip");
|
$(this).data("toggle", "tooltip");
|
||||||
$(this).attr("title", lang_acl.prohibited);
|
$(this).attr("title", lang_acl.prohibited);
|
||||||
$(this).tooltip();
|
$(this).tooltip();
|
||||||
});
|
});
|
||||||
|
|
||||||
// disable submit after submitting form (not API driven buttons)
|
// disable submit after submitting form (not API driven buttons)
|
||||||
$('form').submit(function() {
|
$('form').submit(function() {
|
||||||
if ($('form button[type="submit"]').data('submitted') == '1') {
|
if ($('form button[type="submit"]').data('submitted') == '1') {
|
||||||
return false;
|
return false;
|
||||||
} else {
|
} else {
|
||||||
$(this).find('button[type="submit"]').first().text(lang_footer.loading);
|
$(this).find('button[type="submit"]').first().text(lang_footer.loading);
|
||||||
$('form button[type="submit"]').attr('data-submitted', '1');
|
$('form button[type="submit"]').attr('data-submitted', '1');
|
||||||
function disableF5(e) { if ((e.which || e.keyCode) == 116 || (e.which || e.keyCode) == 82) e.preventDefault(); };
|
function disableF5(e) { if ((e.which || e.keyCode) == 116 || (e.which || e.keyCode) == 82) e.preventDefault(); };
|
||||||
$(document).on("keydown", disableF5);
|
$(document).on("keydown", disableF5);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
// Textarea line numbers
|
// Textarea line numbers
|
||||||
$(".textarea-code").numberedtextarea({allowTabChar: true});
|
$(".textarea-code").numberedtextarea({allowTabChar: true});
|
||||||
// trigger container restart
|
// trigger container restart
|
||||||
$('#RestartContainer').on('show.bs.modal', function(e) {
|
$('#RestartContainer').on('show.bs.modal', function(e) {
|
||||||
var container = $(e.relatedTarget).data('container');
|
var container = $(e.relatedTarget).data('container');
|
||||||
$('#containerName').text(container);
|
$('#containerName').text(container);
|
||||||
$('#triggerRestartContainer').click(function(){
|
$('#triggerRestartContainer').click(function(){
|
||||||
$(this).prop("disabled",true);
|
$(this).prop("disabled",true);
|
||||||
$(this).html('<div class="spinner-border text-white" role="status"><span class="visually-hidden">Loading...</span></div>');
|
$(this).html('<div class="spinner-border text-white" role="status"><span class="visually-hidden">Loading...</span></div>');
|
||||||
$('#statusTriggerRestartContainer').html(lang_footer.restarting_container);
|
$('#statusTriggerRestartContainer').html(lang_footer.restarting_container);
|
||||||
$.ajax({
|
$.ajax({
|
||||||
method: 'get',
|
method: 'get',
|
||||||
url: '/inc/ajax/container_ctrl.php',
|
url: '/inc/ajax/container_ctrl.php',
|
||||||
timeout: docker_timeout,
|
timeout: docker_timeout,
|
||||||
data: {
|
data: {
|
||||||
'service': container,
|
'service': container,
|
||||||
'action': 'restart'
|
'action': 'restart'
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.always( function (data, status) {
|
.always( function (data, status) {
|
||||||
$('#statusTriggerRestartContainer').append(data);
|
$('#statusTriggerRestartContainer').append(data);
|
||||||
var htmlResponse = $.parseHTML(data)
|
var htmlResponse = $.parseHTML(data)
|
||||||
if ($(htmlResponse).find('span').hasClass('text-success')) {
|
if ($(htmlResponse).find('span').hasClass('text-success')) {
|
||||||
$('#triggerRestartContainer').html('<i class="bi bi-check-lg"></i> ');
|
$('#triggerRestartContainer').html('<i class="bi bi-check-lg"></i> ');
|
||||||
setTimeout(function(){
|
setTimeout(function(){
|
||||||
$('#RestartContainer').modal('toggle');
|
$('#RestartContainer').modal('toggle');
|
||||||
window.location = window.location.href.split("#")[0];
|
window.location = window.location.href.split("#")[0];
|
||||||
}, 1200);
|
}, 1200);
|
||||||
} else {
|
} else {
|
||||||
$('#triggerRestartContainer').html('<i class="bi bi-slash-lg"></i> ');
|
$('#triggerRestartContainer').html('<i class="bi bi-slash-lg"></i> ');
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
|
|
||||||
// Jquery Datatables, enable responsive plugin and date sort plugin
|
// Jquery Datatables, enable responsive plugin
|
||||||
$.extend($.fn.dataTable.defaults, {
|
$.extend($.fn.dataTable.defaults, {
|
||||||
responsive: true
|
responsive: true
|
||||||
});
|
});
|
||||||
$.fn.dataTable.moment('dd:mm:YYYY');
|
|
||||||
|
// tag boxes
|
||||||
// tag boxes
|
$('.tag-box .tag-add').click(function(){
|
||||||
$('.tag-box .tag-add').click(function(){
|
addTag(this);
|
||||||
addTag(this);
|
});
|
||||||
});
|
$(".tag-box .tag-input").keydown(function (e) {
|
||||||
$(".tag-box .tag-input").keydown(function (e) {
|
if (e.which == 13){
|
||||||
if (e.which == 13){
|
e.preventDefault();
|
||||||
e.preventDefault();
|
addTag(this);
|
||||||
addTag(this);
|
}
|
||||||
}
|
});
|
||||||
});
|
|
||||||
|
// Dark Mode Loader
|
||||||
// Dark Mode Loader
|
$('#dark-mode-toggle').click(toggleDarkMode);
|
||||||
$('#dark-mode-toggle').click(toggleDarkMode);
|
if ($('#dark-mode-theme').length) {
|
||||||
if ($('#dark-mode-theme').length) {
|
$('#dark-mode-toggle').prop('checked', true);
|
||||||
$('#dark-mode-toggle').prop('checked', true);
|
if ($('#rspamd_logo').length) $('#rspamd_logo').attr('src', '/img/rspamd_logo_light.png');
|
||||||
if ($('#rspamd_logo').length) $('#rspamd_logo').attr('src', '/img/rspamd_logo_light.png');
|
if ($('#rspamd_logo_sm').length) $('#rspamd_logo_sm').attr('src', '/img/rspamd_logo_light.png');
|
||||||
if ($('#rspamd_logo_sm').length) $('#rspamd_logo_sm').attr('src', '/img/rspamd_logo_light.png');
|
}
|
||||||
}
|
function toggleDarkMode(){
|
||||||
function toggleDarkMode(){
|
if($('#dark-mode-theme').length){
|
||||||
if($('#dark-mode-theme').length){
|
$('#dark-mode-theme').remove();
|
||||||
$('#dark-mode-theme').remove();
|
$('#dark-mode-toggle').prop('checked', false);
|
||||||
$('#dark-mode-toggle').prop('checked', false);
|
if ($('#rspamd_logo').length) $('#rspamd_logo').attr('src', '/img/rspamd_logo_dark.png');
|
||||||
if ($('#rspamd_logo').length) $('#rspamd_logo').attr('src', '/img/rspamd_logo_dark.png');
|
if ($('#rspamd_logo_sm').length) $('#rspamd_logo_sm').attr('src', '/img/rspamd_logo_dark.png');
|
||||||
if ($('#rspamd_logo_sm').length) $('#rspamd_logo_sm').attr('src', '/img/rspamd_logo_dark.png');
|
localStorage.setItem('theme', 'light');
|
||||||
localStorage.setItem('theme', 'light');
|
}else{
|
||||||
}else{
|
$('head').append('<link id="dark-mode-theme" rel="stylesheet" type="text/css" href="/css/themes/mailcow-darkmode.css">');
|
||||||
$('head').append('<link id="dark-mode-theme" rel="stylesheet" type="text/css" href="/css/themes/mailcow-darkmode.css">');
|
$('#dark-mode-toggle').prop('checked', true);
|
||||||
$('#dark-mode-toggle').prop('checked', true);
|
if ($('#rspamd_logo').length) $('#rspamd_logo').attr('src', '/img/rspamd_logo_light.png');
|
||||||
if ($('#rspamd_logo').length) $('#rspamd_logo').attr('src', '/img/rspamd_logo_light.png');
|
if ($('#rspamd_logo_sm').length) $('#rspamd_logo_sm').attr('src', '/img/rspamd_logo_light.png');
|
||||||
if ($('#rspamd_logo_sm').length) $('#rspamd_logo_sm').attr('src', '/img/rspamd_logo_light.png');
|
localStorage.setItem('theme', 'dark');
|
||||||
localStorage.setItem('theme', 'dark');
|
}
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
});
|
|
||||||
|
|
||||||
|
// https://stackoverflow.com/questions/24816/escaping-html-strings-with-jquery
|
||||||
// https://stackoverflow.com/questions/24816/escaping-html-strings-with-jquery
|
function escapeHtml(n){var entityMap={"&":"&","<":"<",">":">",'"':""","'":"'","/":"/","`":"`","=":"="}; return String(n).replace(/[&<>"'`=\/]/g,function(n){return entityMap[n]})}
|
||||||
function escapeHtml(n){var entityMap={"&":"&","<":"<",">":">",'"':""","'":"'","/":"/","`":"`","=":"="}; return String(n).replace(/[&<>"'`=\/]/g,function(n){return entityMap[n]})}
|
function unescapeHtml(t){var n={"&":"&","<":"<",">":">",""":'"',"'":"'","/":"/","`":"`","=":"="};return String(t).replace(/&|<|>|"|'|/|`|=/g,function(t){return n[t]})}
|
||||||
function unescapeHtml(t){var n={"&":"&","<":"<",">":">",""":'"',"'":"'","/":"/","`":"`","=":"="};return String(t).replace(/&|<|>|"|'|/|`|=/g,function(t){return n[t]})}
|
|
||||||
|
function addTag(tagAddElem, tag = null){
|
||||||
function addTag(tagAddElem, tag = null){
|
var tagboxElem = $(tagAddElem).parent();
|
||||||
var tagboxElem = $(tagAddElem).parent();
|
var tagInputElem = $(tagboxElem).find(".tag-input")[0];
|
||||||
var tagInputElem = $(tagboxElem).find(".tag-input")[0];
|
var tagValuesElem = $(tagboxElem).find(".tag-values")[0];
|
||||||
var tagValuesElem = $(tagboxElem).find(".tag-values")[0];
|
|
||||||
|
if (!tag)
|
||||||
if (!tag)
|
tag = $(tagInputElem).val();
|
||||||
tag = $(tagInputElem).val();
|
if (!tag) return;
|
||||||
if (!tag) return;
|
var value_tags = [];
|
||||||
var value_tags = [];
|
try {
|
||||||
try {
|
value_tags = JSON.parse($(tagValuesElem).val());
|
||||||
value_tags = JSON.parse($(tagValuesElem).val());
|
} catch {}
|
||||||
} catch {}
|
if (!Array.isArray(value_tags)) value_tags = [];
|
||||||
if (!Array.isArray(value_tags)) value_tags = [];
|
if (value_tags.includes(tag)) return;
|
||||||
if (value_tags.includes(tag)) return;
|
|
||||||
|
$('<span class="badge bg-primary tag-badge btn-badge"><i class="bi bi-tag-fill"></i> ' + escapeHtml(tag) + '</span>').insertBefore('.tag-input').click(function(){
|
||||||
$('<span class="badge bg-primary tag-badge btn-badge"><i class="bi bi-tag-fill"></i> ' + escapeHtml(tag) + '</span>').insertBefore('.tag-input').click(function(){
|
var del_tag = unescapeHtml($(this).text());
|
||||||
var del_tag = unescapeHtml($(this).text());
|
var del_tags = [];
|
||||||
var del_tags = [];
|
try {
|
||||||
try {
|
del_tags = JSON.parse($(tagValuesElem).val());
|
||||||
del_tags = JSON.parse($(tagValuesElem).val());
|
} catch {}
|
||||||
} catch {}
|
if (Array.isArray(del_tags)){
|
||||||
if (Array.isArray(del_tags)){
|
del_tags.splice(del_tags.indexOf(del_tag), 1);
|
||||||
del_tags.splice(del_tags.indexOf(del_tag), 1);
|
$(tagValuesElem).val(JSON.stringify(del_tags));
|
||||||
$(tagValuesElem).val(JSON.stringify(del_tags));
|
}
|
||||||
}
|
$(this).remove();
|
||||||
$(this).remove();
|
});
|
||||||
});
|
|
||||||
|
value_tags.push(tag);
|
||||||
value_tags.push(tag);
|
$(tagValuesElem).val(JSON.stringify(value_tags));
|
||||||
$(tagValuesElem).val(JSON.stringify(value_tags));
|
$(tagInputElem).val('');
|
||||||
$(tagInputElem).val('');
|
}
|
||||||
}
|
|
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue
Block a user