user.js 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704
  1. // Base64 functions
  2. var Base64={_keyStr:"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=",encode:function(r){var t,e,o,a,h,n,c,d="",C=0;for(r=Base64._utf8_encode(r);C<r.length;)a=(t=r.charCodeAt(C++))>>2,h=(3&t)<<4|(e=r.charCodeAt(C++))>>4,n=(15&e)<<2|(o=r.charCodeAt(C++))>>6,c=63&o,isNaN(e)?n=c=64:isNaN(o)&&(c=64),d=d+this._keyStr.charAt(a)+this._keyStr.charAt(h)+this._keyStr.charAt(n)+this._keyStr.charAt(c);return d},decode:function(r){var t,e,o,a,h,n,c="",d=0;for(r=r.replace(/[^A-Za-z0-9\+\/\=]/g,"");d<r.length;)t=this._keyStr.indexOf(r.charAt(d++))<<2|(a=this._keyStr.indexOf(r.charAt(d++)))>>4,e=(15&a)<<4|(h=this._keyStr.indexOf(r.charAt(d++)))>>2,o=(3&h)<<6|(n=this._keyStr.indexOf(r.charAt(d++))),c+=String.fromCharCode(t),64!=h&&(c+=String.fromCharCode(e)),64!=n&&(c+=String.fromCharCode(o));return c=Base64._utf8_decode(c)},_utf8_encode:function(r){r=r.replace(/\r\n/g,"\n");for(var t="",e=0;e<r.length;e++){var o=r.charCodeAt(e);o<128?t+=String.fromCharCode(o):o>127&&o<2048?(t+=String.fromCharCode(o>>6|192),t+=String.fromCharCode(63&o|128)):(t+=String.fromCharCode(o>>12|224),t+=String.fromCharCode(o>>6&63|128),t+=String.fromCharCode(63&o|128))}return t},_utf8_decode:function(r){for(var t="",e=0,o=c1=c2=0;e<r.length;)(o=r.charCodeAt(e))<128?(t+=String.fromCharCode(o),e++):o>191&&o<224?(c2=r.charCodeAt(e+1),t+=String.fromCharCode((31&o)<<6|63&c2),e+=2):(c2=r.charCodeAt(e+1),c3=r.charCodeAt(e+2),t+=String.fromCharCode((15&o)<<12|(63&c2)<<6|63&c3),e+=3);return t}};
  3. $(document).ready(function() {
  4. // Spam score slider
  5. var spam_slider = $('#spam_score')[0];
  6. if (typeof spam_slider !== 'undefined') {
  7. noUiSlider.create(spam_slider, {
  8. start: user_spam_score,
  9. connect: [true, true, true],
  10. range: {
  11. 'min': [0], //stepsize is 50.000
  12. '50%': [10],
  13. '70%': [20, 5],
  14. '80%': [50, 10],
  15. '90%': [100, 100],
  16. '95%': [1000, 1000],
  17. 'max': [5000]
  18. },
  19. });
  20. var connect = spam_slider.querySelectorAll('.noUi-connect');
  21. var classes = ['c-1-color', 'c-2-color', 'c-3-color'];
  22. for (var i = 0; i < connect.length; i++) {
  23. connect[i].classList.add(classes[i]);
  24. }
  25. spam_slider.noUiSlider.on('update', function (values, handle) {
  26. $('.spam-ham-score').text('< ' + Math.round(values[0] * 10) / 10);
  27. $('.spam-spam-score').text(Math.round(values[0] * 10) / 10 + ' - ' + Math.round(values[1] * 10) / 10);
  28. $('.spam-reject-score').text('> ' + Math.round(values[1] * 10) / 10);
  29. $('#spam_score_value').val((Math.round(values[0] * 10) / 10) + ',' + (Math.round(values[1] * 10) / 10));
  30. });
  31. }
  32. // syncjobLogModal
  33. $('#syncjobLogModal').on('show.bs.modal', function(e) {
  34. var syncjob_id = $(e.relatedTarget).data('syncjob-id');
  35. $.ajax({
  36. url: '/inc/ajax/syncjob_logs.php',
  37. data: { id: syncjob_id },
  38. dataType: 'text',
  39. success: function(data){
  40. $(e.currentTarget).find('#logText').text(data);
  41. },
  42. error: function(xhr, status, error) {
  43. $(e.currentTarget).find('#logText').text(xhr.responseText);
  44. }
  45. });
  46. });
  47. $(".arrow-toggle").on('click', function(e) { e.preventDefault(); $(this).find('.arrow').toggleClass("animation"); });
  48. $("#pushover_delete").click(function() { return confirm(lang.delete_ays); });
  49. });
  50. jQuery(function($){
  51. // http://stackoverflow.com/questions/24816/escaping-html-strings-with-jquery
  52. var entityMap = {
  53. '&': '&amp;',
  54. '<': '&lt;',
  55. '>': '&gt;',
  56. '"': '&quot;',
  57. "'": '&#39;',
  58. '/': '&#x2F;',
  59. '`': '&#x60;',
  60. '=': '&#x3D;'
  61. };
  62. function escapeHtml(string) {
  63. return String(string).replace(/[&<>"'`=\/]/g, function (s) {
  64. return entityMap[s];
  65. });
  66. }
  67. // http://stackoverflow.com/questions/46155/validate-email-address-in-javascript
  68. function validateEmail(email) {
  69. var re = /^(([^<>()[\]\\.,;:\s@\"]+(\.[^<>()[\]\\.,;:\s@\"]+)*)|(\".+\"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/;
  70. return re.test(email);
  71. }
  72. function unix_time_format(tm) {
  73. var date = new Date(tm ? tm * 1000 : 0);
  74. return date.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});
  75. }
  76. acl_data = JSON.parse(acl);
  77. $('.clear-last-logins').on('click', function () {if (confirm(lang.delete_ays)) {last_logins('reset');}})
  78. $(".login-history").on('click', function(e) {e.preventDefault(); last_logins('get', $(this).data('days'));$(this).parent().find('li a').removeClass('active');$(this).children(':first-child').addClass('active')});
  79. function last_logins(action, days = 7) {
  80. if (action == 'get') {
  81. $('#spinner-last-login').removeClass('d-none');
  82. $.ajax({
  83. dataType: 'json',
  84. url: '/api/v1/get/last-login/' + encodeURIComponent(mailcow_cc_username) + '/' + days,
  85. jsonp: false,
  86. error: function () {
  87. console.log('error reading last logins');
  88. },
  89. success: function (data) {
  90. $('.last-sasl-login').html('');
  91. if (data.sasl) {
  92. $('.last-sasl-login').append('<ul class="list-group">');
  93. $.each(data.sasl, function (i, item) {
  94. var datetime = new Date(item.datetime.replace(/-/g, "/"));
  95. var local_datetime = datetime.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});
  96. var service = '<div class="badge bg-secondary">' + item.service.toUpperCase() + '</div>';
  97. var app_password = item.app_password ? ' <a href="/edit/app-passwd/' + item.app_password + '"><i class="bi bi-key-fill"></i><span class="ms-1">' + escapeHtml(item.app_password_name || "App") + '</span></a>' : '';
  98. var real_rip = item.real_rip.startsWith("Web") ? item.real_rip : '<a href="https://bgp.tools/prefix/' + item.real_rip + '" target="_blank">' + item.real_rip + "</a>";
  99. var ip_location = item.location ? ' <span class="flag-icon flag-icon-' + item.location.toLowerCase() + '"></span>' : '';
  100. var ip_data = real_rip + ip_location + app_password;
  101. $(".last-sasl-login").append(`
  102. <li class="list-group-item d-flex justify-content-between align-items-start">
  103. <div class="ms-2 me-auto d-flex flex-column">
  104. <div class="fw-bold">` + ip_location + real_rip + `</div>
  105. <small class="fst-italic mt-2">` + service + ` ` + local_datetime + `</small>` + app_password + `
  106. </div>
  107. </li>
  108. `);
  109. })
  110. $('.last-sasl-login').append('</ul>');
  111. }
  112. $('#spinner-last-login').addClass('d-none');
  113. }
  114. })
  115. } else if (action == 'reset') {
  116. $.ajax({
  117. dataType: 'json',
  118. url: '/api/v1/get/reset-last-login/' + encodeURIComponent(mailcow_cc_username),
  119. jsonp: false,
  120. error: function () {
  121. console.log('cannot reset last logins');
  122. },
  123. success: function (data) {
  124. last_logins('get');
  125. }
  126. })
  127. }
  128. }
  129. function createSortableDate(td, cellData, date_string = false) {
  130. if (date_string)
  131. var date = new Date(cellData);
  132. else
  133. var date = new Date(cellData ? cellData * 1000 : 0);
  134. var timestamp = date.getTime();
  135. $(td).attr({
  136. "data-order": timestamp,
  137. "data-sort": timestamp
  138. });
  139. $(td).html(date.toLocaleDateString(LOCALE, DATETIME_FORMAT));
  140. }
  141. function draw_tla_table() {
  142. // just recalc width if instance already exists
  143. if ($.fn.DataTable.isDataTable('#tla_table') ) {
  144. $('#tla_table').DataTable().columns.adjust().responsive.recalc();
  145. return;
  146. }
  147. $('#tla_table').DataTable({
  148. responsive: true,
  149. processing: true,
  150. serverSide: false,
  151. stateSave: true,
  152. pageLength: pagination_size,
  153. dom: "<'row'<'col-sm-12 col-md-6'f><'col-sm-12 col-md-6'l>>" +
  154. "tr" +
  155. "<'row'<'col-sm-12 col-md-5'i><'col-sm-12 col-md-7'p>>",
  156. language: lang_datatables,
  157. order: [[4, 'desc']],
  158. ajax: {
  159. type: "GET",
  160. url: "/api/v1/get/time_limited_aliases",
  161. dataSrc: function(data){
  162. $.each(data, function (i, item) {
  163. if (acl_data.spam_alias === 1) {
  164. item.action = '<div class="btn-group">' +
  165. '<a href="#" data-action="delete_selected" data-id="single-tla" data-api-url="delete/time_limited_alias" data-item="' + encodeURIComponent(item.address) + '" class="btn btn-xs btn-danger"><i class="bi bi-trash"></i> ' + lang.remove + '</a>' +
  166. '</div>';
  167. item.chkbox = '<input type="checkbox" class="form-check-input" data-id="tla" name="multi_select" value="' + encodeURIComponent(item.address) + '" />';
  168. item.address = escapeHtml(item.address);
  169. item.validity = {
  170. value: item.validity,
  171. permanent: item.permanent
  172. };
  173. }
  174. else {
  175. item.chkbox = '<input type="checkbox" class="form-check-input" disabled />';
  176. item.action = '<span>-</span>';
  177. }
  178. });
  179. return data;
  180. }
  181. },
  182. columns: [
  183. {
  184. // placeholder, so checkbox will not block child row toggle
  185. title: '',
  186. data: null,
  187. searchable: false,
  188. orderable: false,
  189. defaultContent: ''
  190. },
  191. {
  192. title: '',
  193. data: 'chkbox',
  194. searchable: false,
  195. orderable: false,
  196. defaultContent: ''
  197. },
  198. {
  199. title: lang.alias,
  200. data: 'address',
  201. defaultContent: ''
  202. },
  203. {
  204. title: lang.description,
  205. data: 'description',
  206. defaultContent: '',
  207. render: function (data, type) {
  208. return escapeHtml(data);
  209. }
  210. },
  211. {
  212. title: lang.alias_valid_until,
  213. data: 'validity',
  214. defaultContent: '',
  215. render: function (data, type) {
  216. var date = new Date(data.value ? data.value * 1000 : 0);
  217. switch (type) {
  218. case "sort":
  219. if (data.permanent) {
  220. return 0;
  221. }
  222. return date.getTime();
  223. default:
  224. if (data.permanent) {
  225. return lang.forever;
  226. }
  227. return date.toLocaleDateString(LOCALE, DATETIME_FORMAT);
  228. }
  229. },
  230. },
  231. {
  232. title: lang.created_on,
  233. data: 'created',
  234. defaultContent: '',
  235. createdCell: function(td, cellData) {
  236. createSortableDate(td, cellData, true)
  237. }
  238. },
  239. {
  240. title: lang.action,
  241. data: 'action',
  242. className: 'dt-sm-head-hidden dt-text-right',
  243. defaultContent: ''
  244. }
  245. ]
  246. });
  247. }
  248. function draw_sync_job_table() {
  249. // just recalc width if instance already exists
  250. if ($.fn.DataTable.isDataTable('#sync_job_table') ) {
  251. $('#sync_job_table').DataTable().columns.adjust().responsive.recalc();
  252. return;
  253. }
  254. $('#sync_job_table').DataTable({
  255. responsive: true,
  256. processing: true,
  257. serverSide: false,
  258. stateSave: true,
  259. pageLength: pagination_size,
  260. dom: "<'row'<'col-sm-12 col-md-6'f><'col-sm-12 col-md-6'l>>" +
  261. "tr" +
  262. "<'row'<'col-sm-12 col-md-5'i><'col-sm-12 col-md-7'p>>",
  263. language: lang_datatables,
  264. ajax: {
  265. type: "GET",
  266. url: '/api/v1/get/syncjobs/' + encodeURIComponent(mailcow_cc_username) + '/no_log',
  267. dataSrc: function(data){
  268. $.each(data, function (i, item) {
  269. item.user1 = escapeHtml(item.user1);
  270. item.log = '<a href="#syncjobLogModal" data-bs-toggle="modal" data-syncjob-id="' + item.id + '">' + lang.open_logs + '</a>'
  271. if (!item.exclude > 0) {
  272. item.exclude = '-';
  273. } else {
  274. item.exclude = '<code>' + escapeHtml(item.exclude) + '</code>';
  275. }
  276. item.server_w_port = escapeHtml(item.user1 + '@' + item.host1 + ':' + item.port1);
  277. if (acl_data.syncjobs === 1) {
  278. item.action = '<div class="btn-group">' +
  279. '<a href="/edit/syncjob/' + item.id + '" class="btn btn-xs btn-xs-half btn-secondary"><i class="bi bi-pencil-fill"></i> ' + lang.edit + '</a>' +
  280. '<a href="#" data-action="delete_selected" data-id="single-syncjob" data-api-url="delete/syncjob" data-item="' + item.id + '" class="btn btn-xs btn-xs-half btn-danger"><i class="bi bi-trash"></i> ' + lang.remove + '</a>' +
  281. '</div>';
  282. item.chkbox = '<input type="checkbox" class="form-check-input" data-id="syncjob" name="multi_select" value="' + item.id + '" />';
  283. }
  284. else {
  285. item.action = '<span>-</span>';
  286. item.chkbox = '<input type="checkbox" class="form-check-input" disabled />';
  287. }
  288. if (item.is_running == 1) {
  289. item.is_running = '<span id="active-script" class="badge fs-6 bg-success">' + lang.running + '</span>';
  290. } else {
  291. item.is_running = '<span id="inactive-script" class="badge fs-6 bg-warning">' + lang.waiting + '</span>';
  292. }
  293. if (!item.last_run > 0) {
  294. item.last_run = lang.waiting;
  295. }
  296. if (item.success == null) {
  297. item.success = '-';
  298. item.exit_status = '';
  299. } else {
  300. item.success = '<i class="text-' + (item.success == 1 ? 'success' : 'danger') + ' bi bi-' + (item.success == 1 ? 'check-lg' : 'x-lg') + '"></i>';
  301. }
  302. if (lang['syncjob_'+item.exit_status]) {
  303. item.exit_status = lang['syncjob_'+item.exit_status];
  304. } else if (item.success != '-') {
  305. item.exit_status = lang.syncjob_check_log;
  306. }
  307. item.exit_status = item.success + ' ' + item.exit_status;
  308. });
  309. return data;
  310. }
  311. },
  312. columns: [
  313. {
  314. // placeholder, so checkbox will not block child row toggle
  315. title: '',
  316. data: null,
  317. searchable: false,
  318. orderable: false,
  319. defaultContent: '',
  320. responsivePriority: 1
  321. },
  322. {
  323. title: '',
  324. data: 'chkbox',
  325. searchable: false,
  326. orderable: false,
  327. defaultContent: '',
  328. responsivePriority: 2
  329. },
  330. {
  331. title: 'ID',
  332. data: 'id',
  333. defaultContent: '',
  334. responsivePriority: 3
  335. },
  336. {
  337. title: 'Server',
  338. data: 'server_w_port',
  339. defaultContent: ''
  340. },
  341. {
  342. title: lang.username,
  343. data: 'user1',
  344. defaultContent: '',
  345. responsivePriority: 3
  346. },
  347. {
  348. title: lang.last_run,
  349. data: 'last_run',
  350. defaultContent: ''
  351. },
  352. {
  353. title: lang.syncjob_last_run_result,
  354. data: 'exit_status',
  355. defaultContent: ''
  356. },
  357. {
  358. title: 'Log',
  359. data: 'log',
  360. defaultContent: ''
  361. },
  362. {
  363. title: lang.active,
  364. data: 'active',
  365. defaultContent: '',
  366. render: function (data, type) {
  367. return 1==data?'<i class="bi bi-check-lg"></i>':0==data&&'<i class="bi bi-x-lg"></i>'
  368. }
  369. },
  370. {
  371. title: lang.status,
  372. data: 'is_running',
  373. defaultContent: '',
  374. responsivePriority: 5
  375. },
  376. {
  377. title: lang.encryption,
  378. data: 'enc1',
  379. defaultContent: ''
  380. },
  381. {
  382. title: lang.excludes,
  383. data: 'exclude',
  384. defaultContent: ''
  385. },
  386. {
  387. title: lang.interval + " (min)",
  388. data: 'mins_interval',
  389. defaultContent: ''
  390. },
  391. {
  392. title: lang.action,
  393. data: 'action',
  394. className: 'dt-sm-head-hidden dt-text-right',
  395. defaultContent: '',
  396. responsivePriority: 5
  397. }
  398. ]
  399. });
  400. }
  401. function draw_app_passwd_table() {
  402. // just recalc width if instance already exists
  403. if ($.fn.DataTable.isDataTable('#app_passwd_table') ) {
  404. $('#app_passwd_table').DataTable().columns.adjust().responsive.recalc();
  405. return;
  406. }
  407. $('#app_passwd_table').DataTable({
  408. responsive: true,
  409. processing: true,
  410. serverSide: false,
  411. stateSave: true,
  412. pageLength: pagination_size,
  413. dom: "<'row'<'col-sm-12 col-md-6'f><'col-sm-12 col-md-6'l>>" +
  414. "tr" +
  415. "<'row'<'col-sm-12 col-md-5'i><'col-sm-12 col-md-7'p>>",
  416. language: lang_datatables,
  417. ajax: {
  418. type: "GET",
  419. url: '/api/v1/get/app-passwd/all',
  420. dataSrc: function(data){
  421. $.each(data, function (i, item) {
  422. item.name = escapeHtml(item.name)
  423. item.protocols = []
  424. if (item.imap_access == 1) { item.protocols.push("<code>IMAP</code>"); }
  425. if (item.smtp_access == 1) { item.protocols.push("<code>SMTP</code>"); }
  426. if (item.eas_access == 1) { item.protocols.push("<code>EAS/ActiveSync</code>"); }
  427. if (item.dav_access == 1) { item.protocols.push("<code>DAV</code>"); }
  428. if (item.pop3_access == 1) { item.protocols.push("<code>POP3</code>"); }
  429. if (item.sieve_access == 1) { item.protocols.push("<code>Sieve</code>"); }
  430. item.protocols = item.protocols.join(" ")
  431. if (acl_data.app_passwds === 1) {
  432. item.action = '<div class="btn-group">' +
  433. '<a href="/edit/app-passwd/' + item.id + '" class="btn btn-xs btn-xs-half btn-secondary"><i class="bi bi-pencil-fill"></i> ' + lang.edit + '</a>' +
  434. '<a href="#" data-action="delete_selected" data-id="single-apppasswd" data-api-url="delete/app-passwd" data-item="' + item.id + '" class="btn btn-xs btn-xs-half btn-danger"><i class="bi bi-trash"></i> ' + lang.remove + '</a>' +
  435. '</div>';
  436. item.chkbox = '<input type="checkbox" class="form-check-input" data-id="apppasswd" name="multi_select" value="' + item.id + '" />';
  437. }
  438. else {
  439. item.action = '<span>-</span>';
  440. item.chkbox = '<input type="checkbox" class="form-check-input" disabled />';
  441. }
  442. });
  443. return data;
  444. }
  445. },
  446. columns: [
  447. {
  448. // placeholder, so checkbox will not block child row toggle
  449. title: '',
  450. data: null,
  451. searchable: false,
  452. orderable: false,
  453. defaultContent: ''
  454. },
  455. {
  456. title: '',
  457. data: 'chkbox',
  458. searchable: false,
  459. orderable: false,
  460. defaultContent: ''
  461. },
  462. {
  463. title: 'ID',
  464. data: 'id',
  465. defaultContent: ''
  466. },
  467. {
  468. title: lang.app_name,
  469. data: 'name',
  470. defaultContent: ''
  471. },
  472. {
  473. title: lang.allowed_protocols,
  474. data: 'protocols',
  475. defaultContent: ''
  476. },
  477. {
  478. title: lang.active,
  479. data: 'active',
  480. defaultContent: '',
  481. render: function (data, type) {
  482. return 1==data?'<i class="bi bi-check-lg"></i>':0==data&&'<i class="bi bi-x-lg"></i>'
  483. }
  484. },
  485. {
  486. title: lang.action,
  487. data: 'action',
  488. className: 'dt-sm-head-hidden dt-text-right',
  489. defaultContent: ''
  490. }
  491. ]
  492. });
  493. }
  494. function draw_wl_policy_mailbox_table() {
  495. // just recalc width if instance already exists
  496. if ($.fn.DataTable.isDataTable('#wl_policy_mailbox_table') ) {
  497. $('#wl_policy_mailbox_table').DataTable().columns.adjust().responsive.recalc();
  498. return;
  499. }
  500. $('#wl_policy_mailbox_table').DataTable({
  501. responsive: true,
  502. processing: true,
  503. serverSide: false,
  504. stateSave: true,
  505. pageLength: pagination_size,
  506. dom: "<'row'<'col-sm-12 col-md-6'f><'col-sm-12 col-md-6'l>>" +
  507. "tr" +
  508. "<'row'<'col-sm-12 col-md-5'i><'col-sm-12 col-md-7'p>>",
  509. language: lang_datatables,
  510. ajax: {
  511. type: "GET",
  512. url: '/api/v1/get/policy_wl_mailbox',
  513. dataSrc: function(data){
  514. $.each(data, function (i, item) {
  515. if (validateEmail(item.object)) {
  516. item.chkbox = '<input type="checkbox" class="form-check-input" data-id="policy_wl_mailbox" name="multi_select" value="' + item.prefid + '" />';
  517. }
  518. else {
  519. item.chkbox = '<input type="checkbox" class="form-check-input" disabled title="' + lang.spamfilter_table_domain_policy + '" />';
  520. }
  521. if (acl_data.spam_policy === 0) {
  522. item.chkbox = '<input type="checkbox" class="form-check-input" disabled />';
  523. }
  524. });
  525. return data;
  526. }
  527. },
  528. columns: [
  529. {
  530. // placeholder, so checkbox will not block child row toggle
  531. title: '',
  532. data: null,
  533. searchable: false,
  534. orderable: false,
  535. defaultContent: ''
  536. },
  537. {
  538. title: '',
  539. data: 'chkbox',
  540. searchable: false,
  541. orderable: false,
  542. defaultContent: ''
  543. },
  544. {
  545. title: 'ID',
  546. data: 'prefid',
  547. defaultContent: ''
  548. },
  549. {
  550. title: lang.spamfilter_table_rule,
  551. data: 'value',
  552. defaultContent: ''
  553. },
  554. {
  555. title:'Scope',
  556. data: 'object',
  557. defaultContent: ''
  558. }
  559. ]
  560. });
  561. }
  562. function draw_bl_policy_mailbox_table() {
  563. // just recalc width if instance already exists
  564. if ($.fn.DataTable.isDataTable('#bl_policy_mailbox_table') ) {
  565. $('#bl_policy_mailbox_table').DataTable().columns.adjust().responsive.recalc();
  566. return;
  567. }
  568. $('#bl_policy_mailbox_table').DataTable({
  569. responsive: true,
  570. processing: true,
  571. serverSide: false,
  572. stateSave: true,
  573. pageLength: pagination_size,
  574. dom: "<'row'<'col-sm-12 col-md-6'f><'col-sm-12 col-md-6'l>>" +
  575. "tr" +
  576. "<'row'<'col-sm-12 col-md-5'i><'col-sm-12 col-md-7'p>>",
  577. language: lang_datatables,
  578. ajax: {
  579. type: "GET",
  580. url: '/api/v1/get/policy_bl_mailbox',
  581. dataSrc: function(data){
  582. $.each(data, function (i, item) {
  583. if (validateEmail(item.object)) {
  584. item.chkbox = '<input type="checkbox" class="form-check-input" data-id="policy_bl_mailbox" name="multi_select" value="' + item.prefid + '" />';
  585. }
  586. else {
  587. item.chkbox = '<input type="checkbox" class="form-check-input" disabled tooltip="' + lang.spamfilter_table_domain_policy + '" />';
  588. }
  589. if (acl_data.spam_policy === 0) {
  590. item.chkbox = '<input type="checkbox" class="form-check-input" disabled />';
  591. }
  592. });
  593. return data;
  594. }
  595. },
  596. columns: [
  597. {
  598. // placeholder, so checkbox will not block child row toggle
  599. title: '',
  600. data: null,
  601. searchable: false,
  602. orderable: false,
  603. defaultContent: ''
  604. },
  605. {
  606. title: '',
  607. data: 'chkbox',
  608. searchable: false,
  609. orderable: false,
  610. defaultContent: ''
  611. },
  612. {
  613. title: 'ID',
  614. data: 'prefid',
  615. defaultContent: ''
  616. },
  617. {
  618. title: lang.spamfilter_table_rule,
  619. data: 'value',
  620. defaultContent: ''
  621. },
  622. {
  623. title:'Scope',
  624. data: 'object',
  625. defaultContent: ''
  626. }
  627. ]
  628. });
  629. }
  630. // FIDO2 friendly name modal
  631. $('#fido2ChangeFn').on('show.bs.modal', function (e) {
  632. rename_link = $(e.relatedTarget)
  633. if (rename_link != null) {
  634. $('#fido2_cid').val(rename_link.data('cid'));
  635. $('#fido2_subject_desc').text(Base64.decode(rename_link.data('subject')));
  636. }
  637. })
  638. // Sieve data modal
  639. $('#userFilterModal').on('show.bs.modal', function(e) {
  640. $('#user_sieve_filter').text(lang.loading);
  641. $.ajax({
  642. dataType: 'json',
  643. url: '/api/v1/get/active-user-sieve/' + encodeURIComponent(mailcow_cc_username),
  644. jsonp: false,
  645. error: function () {
  646. console.log('Cannot get active sieve script');
  647. },
  648. complete: function (data) {
  649. if (data.responseText == '{}') {
  650. $('#user_sieve_filter').text(lang.no_active_filter);
  651. } else {
  652. $('#user_sieve_filter').text(JSON.parse(data.responseText));
  653. }
  654. }
  655. })
  656. });
  657. $('#userFilterModal').on('hidden.bs.modal', function () {
  658. $('#user_sieve_filter').text(lang.loading);
  659. });
  660. // detect element visibility changes
  661. function onVisible(element, callback) {
  662. $(document).ready(function() {
  663. element_object = document.querySelector(element);
  664. if (element_object === null) return;
  665. new IntersectionObserver((entries, observer) => {
  666. entries.forEach(entry => {
  667. if(entry.intersectionRatio > 0) {
  668. callback(element_object);
  669. }
  670. });
  671. }).observe(element_object);
  672. });
  673. }
  674. // Load only if the tab is visible
  675. onVisible("[id^=tla_table]", () => draw_tla_table());
  676. onVisible("[id^=bl_policy_mailbox_table]", () => draw_bl_policy_mailbox_table());
  677. onVisible("[id^=wl_policy_mailbox_table]", () => draw_wl_policy_mailbox_table());
  678. onVisible("[id^=sync_job_table]", () => draw_sync_job_table());
  679. onVisible("[id^=app_passwd_table]", () => draw_app_passwd_table());
  680. onVisible("[id^=recent-logins]", () => last_logins('get'));
  681. });