Passed
Push — master ( 8addf5...3df5b4 )
by EMP
01:20
created

main.js ➔ reloadAccount   B

Complexity

Conditions 6

Size

Total Lines 55
Code Lines 36

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
eloc 36
dl 0
loc 55
rs 8.0826
c 0
b 0
f 0
cc 6

How to fix   Long Method   

Long Method

Small methods make your code easier to understand, in particular if combined with a good name. Besides, if your method is small, finding a good name is usually much easier.

For example, if you find yourself adding comments to a method's body, this is usually a good sign to extract the commented part to a new method, and use the comment as a starting point when coming up with a good name for this new method.

Commonly applied refactorings include:

1
"use strict";
2
3
sodium.ready.then(function() {
1 ignored issue
show
Bug introduced by
The variable sodium seems to be never declared. If this is a global, consider adding a /** global: sodium */ comment.

This checks looks for references to variables that have not been declared. This is most likey a typographical error or a variable has been renamed.

To learn more about declaring variables in Javascript, see the MDN.

Loading history...
4
5
const ae = new AllEars(function(ok) {
1 ignored issue
show
Bug introduced by
The variable AllEars seems to be never declared. If this is a global, consider adding a /** global: AllEars */ comment.

This checks looks for references to variables that have not been declared. This is most likey a typographical error or a variable has been renamed.

To learn more about declaring variables in Javascript, see the MDN.

Loading history...
6
	if (ok) {
7
		document.getElementById("txt_skey").style.background = "#466";
8
		document.getElementById("txt_skey").maxLength = "64";
9
	} else {
10
		console.log("Failed to load All-Ears");
11
	}
12
});
13
14
function TabState(cur, max, btnDele, btnUpdt) {
15
	this.cur = cur;
16
	this.max = max;
17
	this.btnDele = btnDele;
18
	this.btnUpdt = btnUpdt;
19
}
20
21
const tabs = [
22
	new TabState(0, 0, false, true), // Inbox
23
	new TabState(0, 0, false, true), // Outbx
24
	new TabState(0, 2, true, false), // Write
25
	new TabState(0, 0, false, false), // Notes
26
	new TabState(0, 3, false, true) // Admin
27
];
28
29
let tab = 0;
30
const TAB_INBOX = 0;
31
const TAB_OUTBX = 1;
0 ignored issues
show
Unused Code introduced by
The constant TAB_OUTBX seems to be never used. Consider removing it.
Loading history...
32
const TAB_WRITE = 2;
33
const TAB_NOTES = 3;
0 ignored issues
show
Unused Code introduced by
The constant TAB_NOTES seems to be never used. Consider removing it.
Loading history...
34
const TAB_ADMIN = 4;
35
36
// Helper functions
37
function getCountryName(countryCode) {
38
	const opts = document.getElementById("gatekeeper_country");
39
40
	for (let i = 0; i < opts.length; i++) {
41
		if (opts[i].value === countryCode) {
42
			return opts[i].textContent;
43
		}
44
	}
45
46
	return "Unknown countrycode: " + countryCode;
47
}
48
49
function getCountryFlag(countryCode) {
50
	return sodium.to_string(new Uint8Array([
1 ignored issue
show
Bug introduced by
The variable sodium seems to be never declared. If this is a global, consider adding a /** global: sodium */ comment.

This checks looks for references to variables that have not been declared. This is most likey a typographical error or a variable has been renamed.

To learn more about declaring variables in Javascript, see the MDN.

Loading history...
51
		240, 159, 135, 166 + countryCode.codePointAt(0) - 65,
52
		240, 159, 135, 166 + countryCode.codePointAt(1) - 65
53
	]));
54
}
55
56
function getMsgId(num) {
57
	let i;
58
	if (ae.GetExtMsgHeaders(num).toLowerCase().slice(0, 11) === "message-id:") {
59
		i = 0;
60
	} else {
61
		i = ae.GetExtMsgHeaders(num).toLowerCase().indexOf("\nmessage-id:");
62
		if (i < 1) return "ERR";
63
		i++;
64
	}
65
66
	const x = ae.GetExtMsgHeaders(num).slice(i + 11).trim();
67
	if (x[0] !== "<") return "ERR2";
68
	return x.slice(1, x.indexOf(">"));
69
}
70
71
function displayMsg(isInt, num) {
72
	document.getElementById("midright").scroll(0, 0);
73
74
	const ts = isInt? ae.GetIntMsgTime(num) : ae.GetExtMsgTime(num);
75
76
	document.getElementById("btn_reply").disabled = false;
77
	document.getElementById("btn_reply").onclick = function() {
78
		document.getElementById("write_recv").value = isInt? ae.GetIntMsgFrom(num) : ae.GetExtMsgFrom(num);
79
		document.getElementById("write_subj").value = "Re: " + (isInt ? ae.GetIntMsgTitle(num) : ae.GetExtMsgTitle(num));
80
		document.getElementById("write_rply").textContent = (isInt? "" : getMsgId(num));
81
		document.getElementById("btn_write").click();
82
		document.getElementById("div_write_1").hidden = false;
83
		document.getElementById("div_write_2").hidden = true;
84
		document.getElementById("write_body").focus();
85
		for (const opt of document.getElementById("write_from").options) {
86
			if (opt.value === (isInt ? ae.GetIntMsgTo(num) : ae.GetExtMsgTo(num))) {
87
				opt.selected = true;
88
			}
89
		}
90
	};
91
92
	document.getElementById("msg").hidden = false;
93
	document.getElementById("msg").getElementsByTagName("h1")[0].textContent = isInt ? ae.GetIntMsgTitle(num) : ae.GetExtMsgTitle(num);
94
	document.getElementById("msg").getElementsByTagName("pre")[0].textContent = isInt ? ae.GetIntMsgBody(num) : ae.GetExtMsgBody(num);
95
96
	document.getElementById("readmsg_to").textContent = isInt ? ae.GetIntMsgTo(num) : ae.GetExtMsgTo(num);
97
	document.getElementById("readmsg_date").children[0].textContent = new Date(ts * 1000).toISOString().slice(0, 19).replace("T", " ");
98
99
	if (!isInt) {
100
		document.getElementById("readmsg_ip").hidden = false;
101
		document.getElementById("readmsg_country").hidden = false;
102
		document.getElementById("readmsg_tls").hidden = false;
103
		document.getElementById("readmsg_greet").hidden = false;
104
		document.getElementById("readmsg_timing").hidden = false;
105
		document.getElementById("readmsg_envfrom").hidden = false;
106
107
		const cc = ae.GetExtMsgCountry(num);
108
109
		document.getElementById("readmsg_ip").children[0].textContent = ae.GetExtMsgIp(num);
110
		document.getElementById("readmsg_country").textContent = getCountryFlag(cc) + " " + getCountryName(cc);
111
		document.getElementById("readmsg_tls").children[0].textContent = ae.GetExtMsgTLS(num);
112
		document.getElementById("readmsg_greet").children[0].textContent = ae.GetExtMsgGreet(num);
113
		document.getElementById("readmsg_envfrom").textContent = ae.GetExtMsgFrom(num);
114
115
		let flagText = "";
116
		if (!ae.GetExtMsgFlagVPad(num)) flagText += "<abbr title=\"Invalid padding\">PAD</abbr> ";
117
		if (!ae.GetExtMsgFlagVSig(num)) flagText += "<abbr title=\"Invalid signature\">SIG</abbr> ";
118
		if (!ae.GetExtMsgFlagPExt(num)) flagText += "<abbr title=\"The sender did not use the Extended (ESMTP) protocol\">SMTP</abbr> ";
119
		if (!ae.GetExtMsgFlagQuit(num)) flagText += "<abbr title=\"The sender did not issue the required QUIT command\">QUIT</abbr> ";
120
		if (ae.GetExtMsgFlagRare(num)) flagText += "<abbr title=\"The sender issued unusual command(s)\">RARE</abbr> ";
121
		if (ae.GetExtMsgFlagFail(num)) flagText += "<abbr title=\"The sender issued invalid command(s)\">FAIL</abbr> ";
122
		if (ae.GetExtMsgFlagPErr(num)) flagText += "<abbr title=\"The sender violated the protocol\">PROT</abbr> ";
123
		document.getElementById("readmsg_flags").children[0].innerHTML = flagText.trim();
124
	} else {
125
		document.getElementById("readmsg_ip").hidden = true;
126
		document.getElementById("readmsg_country").hidden = true;
127
		document.getElementById("readmsg_tls").hidden = true;
128
		document.getElementById("readmsg_greet").hidden = true;
129
		document.getElementById("readmsg_timing").hidden = true;
130
		document.getElementById("readmsg_envfrom").hidden = true;
131
132
		let symbol = "<span title=\"Invalid level\">&#x26a0;</span>";
133
		if (ae.GetIntMsgFrom(num) === "system") {if (ae.GetIntMsgLevel(num) === 3) symbol = "<span title=\"System\">&#x1f162;</span>";} // S (System)
134
		else if (ae.GetIntMsgLevel(num) === 0) symbol = "<span title=\"Level 0 User\">&#x1f10c;</span>"; // 0
135
		else if (ae.GetIntMsgLevel(num) === 1) symbol = "<span title=\"Level 1 User\">&#x278a;</span>"; // 1
136
		else if (ae.GetIntMsgLevel(num) === 2) symbol = "<span title=\"Level 2 User\">&#x278b;</span>"; // 2
137
		else if (ae.GetIntMsgLevel(num) === 3) symbol = "<span title=\"Administrator\">&#x1f150;</span>"; // A (Admin)
138
		document.getElementById("readmsg_from").innerHTML = symbol + " " + ae.GetIntMsgFrom(num);
139
140
		let flagText = "";
141
		if (!ae.GetIntMsgFlagVPad(num)) flagText += "<abbr title=\"Invalid padding\">PAD</abbr> ";
142
		if (!ae.GetIntMsgFlagVSig(num)) flagText += "<abbr title=\"Invalid signature\">SIG</abbr> ";
143
		document.getElementById("readmsg_flags").children[0].innerHTML = flagText.trim();
144
	}
145
}
146
147
// Interface
148
function addMsg(isInt, i) {
149
	const inbox = document.getElementById("tbl_inbox");
150
	const sent = document.getElementById("tbl_outbx");
151
152
	const isSent = false; //TODO
153
	const table = isSent ? sent : inbox;
154
155
	const row = table.insertRow(-1);
156
	const cellTime = row.insertCell(-1);
157
	const cellSubj = row.insertCell(-1);
158
	const cellSnd1 = row.insertCell(-1);
159
	const cellSnd2 = row.insertCell(-1);
160
161
	const ts = isInt? ae.GetIntMsgTime(i) : ae.GetExtMsgTime(i);
162
	cellTime.setAttribute("data-ts", ts);
163
	cellTime.textContent = new Date(ts * 1000).toISOString().slice(0, 10);
164
165
	cellSubj.textContent = isInt? ae.GetIntMsgTitle(i) : ae.GetExtMsgTitle(i);
166
167
	if (isInt) {
168
		cellSnd1.textContent = ae.GetIntMsgFrom(i);
169
		cellSnd1.className = (ae.GetIntMsgFrom(i).length === 16) ? "mono" : "";
170
	} else {
171
		const from1 = ae.GetExtMsgFrom(i);
172
		const from2 = from1.substring(from1.indexOf("@") + 1);
173
		const cc = ae.GetExtMsgCountry(i);
174
175
		cellSnd1.textContent = from1.substring(0, from1.indexOf("@"));
176
177
		const flag = document.createElement("abbr");
178
		flag.textContent = getCountryFlag(cc);
179
		flag.title = getCountryName(cc);
180
		cellSnd2.appendChild(flag);
181
182
		const fromText = document.createElement("span");
183
		fromText.textContent = " " + from2;
184
		cellSnd2.appendChild(fromText);
185
	}
186
187
//	divDel.innerHTML = "<input class=\"delMsg\" type=\"checkbox\" data-id=\"" + ae.GetIntMsgIdHex(i) + "\">";
188
189
	row.onclick = function() {
190
		displayMsg(isInt, i);
191
	};
192
/*
193
	cellDel.children[0].onchange = function() {
194
		if (!divDel.children[0].checked) {
195
			const checkboxes = elmt.getElementsByTagName("input");
196
			let checked = false;
197
198
			for (let j = 0; j < checkboxes.length; j++) {
199
				if (checkboxes[j].checked) {
200
					checked = true;
201
					break;
202
				}
203
			}
204
205
			if (!checked) {
206
				document.getElementById(isSent ? "btn_sentdel" : "btn_msgdel").hidden = true;
207
				return;
208
			}
209
		}
210
211
		document.getElementById(isSent? "btn_sentdel" : "btn_msgdel").hidden = false;
212
	};
213
*/
214
}
215
216
function getRowsPerPage() {
217
	const tbl = document.getElementById("tbl_inbox");
218
	tbl.innerHTML = "";
219
	const row = tbl.insertRow(-1);
220
	const cell = row.insertCell(-1);
221
	cell.textContent = "0";
222
223
	const rowsPerPage = Math.floor(getComputedStyle(document.getElementById("div_inbox")).height.replace("px", "") / getComputedStyle(document.querySelector("#tbl_inbox > tbody > tr:first-child")).height.replace("px", "")) - 1; // -1 allows space for 'load more'
224
	tbl.innerHTML = "";
225
	return rowsPerPage;
226
}
227
228
function addMessages() {
229
	const rowsPerPage = getRowsPerPage();
230
	let skipMsgs = rowsPerPage * tabs[TAB_INBOX].cur;
231
232
	const maxExt = ae.GetExtMsgCount();
233
	const maxInt = ae.GetIntMsgCount();
234
235
	tabs[TAB_INBOX].max = Math.floor((maxExt + maxInt) / rowsPerPage);
236
237
	let numExt = 0;
238
	let numInt = 0;
239
	let numAdd = 0;
240
241
	while (numAdd < rowsPerPage) {
242
		const tsInt = (numInt < maxInt) ? ae.GetIntMsgTime(numInt) : -1;
243
		const tsExt = (numExt < maxExt) ? ae.GetExtMsgTime(numExt) : -1;
244
		if (tsInt === -1 && tsExt === -1) break;
245
246
		if (tsInt !== -1 && (tsExt === -1 || tsInt > tsExt)) {
247
			if (skipMsgs > 0) skipMsgs--; else {addMsg(true, numInt); numAdd++;}
248
			numInt++;
249
		} else if (tsExt !== -1) {
250
			if (skipMsgs > 0) skipMsgs--; else {addMsg(false, numExt); numAdd++;}
251
			numExt++;
252
		}
253
	}
254
255
	if (ae.GetReadyMsgKilos() < ae.GetTotalMsgKilos()) {
256
		const inbox = document.getElementById("tbl_inbox");
257
		const row = inbox.insertRow(-1);
258
		const cell = row.insertCell(-1);
259
		cell.textContent = "Load more (" + (ae.GetTotalMsgKilos() - ae.GetReadyMsgKilos()) + " KiB left)";
260
261
		row.onclick = function() {
262
			this.onclick = "";
263
264
			ae.Message_Browse(false, function(successBrowse) {
265
				document.getElementById("tbl_inbox").style.opacity = 1;
266
267
				if (successBrowse) {
268
					addMessages();
269
					if (tabs[tab].cur < tabs[tab].max) document.getElementById("btn_rght").disabled = false;
270
				}
271
			});
272
		};
273
	}
274
}
275
276
function updateAddressCounts() {
277
	document.getElementById("limit_normal").textContent = (ae.GetAddressCountNormal() + "/" + ae.GetAddressLimitNormal(ae.GetUserLevel())).padStart(ae.GetAddressLimitNormal(ae.GetUserLevel()) > 9 ? 5 : 1);
278
	document.getElementById("limit_shield").textContent = (ae.GetAddressCountShield() + "/" + ae.GetAddressLimitShield(ae.GetUserLevel())).padStart(ae.GetAddressLimitShield(ae.GetUserLevel()) > 9 ? 5 : 1);
279
	document.getElementById("limit_total").textContent = ((ae.GetAddressCountNormal() + ae.GetAddressCountShield()) + "/" + ae.GetAddrPerUser()).padStart(5);
280
}
281
282
function reloadAccount() {
283
	// Limits
284
	const tblLimits = document.getElementById("tbl_limits");
285
	for (let i = 0; i < 4; i++) {
286
		tblLimits.rows[i].cells[1].children[0].value = ae.GetStorageLimit(i) + 1;
287
		tblLimits.rows[i].cells[2].children[0].value = ae.GetAddressLimitNormal(i);
288
		tblLimits.rows[i].cells[3].children[0].value = ae.GetAddressLimitShield(i);
289
	}
290
291
	// Accounts
292
	const tblAccs = document.getElementById("tbd_accs");
293
294
	// All: Our account
295
	let row = tblAccs.insertRow(-1);
296
	let cell;
297
	cell = row.insertCell(-1); cell.textContent = ae.GetUserPkHex();
298
	cell = row.insertCell(-1); cell.textContent = Math.round(ae.GetTotalMsgKilos() / 1024);
299
	cell = row.insertCell(-1); cell.textContent = ae.GetAddressCountNormal();
300
	cell = row.insertCell(-1); cell.textContent = ae.GetAddressCountShield();
301
	cell = row.insertCell(-1); cell.textContent = ae.GetUserLevel();
302
	cell = row.insertCell(-1); cell.innerHTML = "<button type=\"button\" autocomplete=\"off\" disabled=\"disabled\">+</button>";
303
	cell = row.insertCell(-1); cell.innerHTML = "<button id=\"btn_downme\" type=\"button\" autocomplete=\"off\" disabled=\"disabled\">&minus;</button>";
304
	cell = row.insertCell(-1); cell.innerHTML = "<button id=\"btn_killme\" type=\"button\" autocomplete=\"off\" disabled=\"disabled\">X</button>";
305
306
	// Admin: Other accounts
307
	if (ae.IsUserAdmin()) {
308
		for (let i = 0; i < ae.Admin_GetUserCount(); i++) {
309
			row = tblAccs.insertRow(-1);
310
			cell = row.insertCell(-1); cell.textContent = ae.Admin_GetUserPkHex(i);
311
			cell = row.insertCell(-1); cell.textContent = ae.Admin_GetUserSpace(i);
312
			cell = row.insertCell(-1); cell.textContent = ae.Admin_GetUserNAddr(i);
313
			cell = row.insertCell(-1); cell.textContent = ae.Admin_GetUserSAddr(i);
314
			cell = row.insertCell(-1); cell.textContent = ae.Admin_GetUserLevel(i);
315
			cell = row.insertCell(-1); cell.innerHTML = "<button type=\"button\" autocomplete=\"off\">+</button>";
316
			cell = row.insertCell(-1); cell.innerHTML = "<button type=\"button\" autocomplete=\"off\">&minus;</button>";
317
			cell = row.insertCell(-1); cell.innerHTML = "<button type=\"button\" autocomplete=\"off\">X</button>";
318
		}
319
	}
320
321
	// Contacts
322
	for (let i = 0; i < ae.GetContactCount(); i++) {
323
		addContact(
324
			ae.GetContactMail(i),
325
			ae.GetContactName(i),
326
			ae.GetContactNote(i)
327
		);
328
	}
329
330
	// Addresses
331
	for (let i = 0; i < ae.GetAddressCount(); i++) {
332
		addAddress(i);
333
	}
334
335
	updateAddressCounts();
336
}
337
338 View Code Duplication
function deleteAddress(addr) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated in your project.
Loading history...
339
	let btns = document.getElementById("tbl_addrs").getElementsByTagName("button");
340
	for (let i = 0; i < btns.length; i++) btns[i].disabled = true;
341
342
	let addressToDelete = -1;
343
344
	for (let i = 0; i < ae.GetAddressCount(); i++) {
345
		if (addr === ae.GetAddress(i)) {
346
			addressToDelete = i;
347
			break;
348
		}
349
	}
350
351
	if (addressToDelete === -1) return;
352
353
	ae.Address_Delete(addressToDelete, function(success) {
354
		if (success) {
355
			document.getElementById("tbl_addrs").deleteRow(addressToDelete);
356
			document.getElementById("write_from").remove(addressToDelete);
357
			updateAddressCounts();
358
359
			if (ae.GetAddressCountNormal() < ae.GetAddressLimitNormal(ae.GetUserLevel())) document.getElementById("btn_address_create_normal").disabled = false;
360
			if (ae.GetAddressCountShield() < ae.GetAddressLimitShield(ae.GetUserLevel())) document.getElementById("btn_address_create_shield").disabled = false;
361
362
			ae.Private_Update(function(success2) {
363
				if (!success2) console.log("Failed to update the Private field");
364
365
				btns = document.getElementById("tbl_addrs").getElementsByTagName("button");
366
				for (let i = 0; i < btns.length; i++) btns[i].disabled = false;
367
			});
368
		} else {
369
			console.log("Failed to delete address");
370
371
			btns = document.getElementById("tbl_addrs").getElementsByTagName("button");
372
			for (let i = 0; i < btns.length; i++) btns[i].disabled = false;
373
		}
374
	});
375
}
376
377 View Code Duplication
function shieldMix(addr) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated in your project.
Loading history...
378
	let newAddr = "";
379
380
	for (let i = 0; i < 16; i++) {
381
		switch (addr.charAt(i)) {
382
			case '1':
383
				newAddr += "1iIlL".charAt(Math.floor(Math.random() * 5));
384
				break;
385
			case '0':
386
				newAddr += "0oO".charAt(Math.floor(Math.random() * 3));
387
				break;
388
			case 'w':
389
				newAddr += "VvWw".charAt(Math.floor(Math.random() * 4));
390
				break;
391
			default:
392
				newAddr += (Math.random() > 0.5) ? addr.charAt(i) : addr.charAt(i).toUpperCase();
393
		}
394
	}
395
396
	return newAddr;
397
}
398
399
function addAddress(num) {
400
	const addrTable = document.getElementById("tbl_addrs");
401
	const row = addrTable.insertRow(-1);
402
	const cellAddr = row.insertCell(-1);
403
	const cellChk1 = row.insertCell(-1);
404
	const cellChk2 = row.insertCell(-1);
405
	const cellChk3 = row.insertCell(-1);
406
	const cellBtnD = row.insertCell(-1);
407
408
	cellAddr.textContent = ae.GetAddress(num);
409
	cellAddr.onclick = function() {
410
		if (cellAddr.textContent.length === 16)
411
			navigator.clipboard.writeText(shieldMix(cellAddr.textContent) + "@" + ae.GetDomainEml());
1 ignored issue
show
Bug introduced by
The variable navigator seems to be never declared. If this is a global, consider adding a /** global: navigator */ comment.

This checks looks for references to variables that have not been declared. This is most likey a typographical error or a variable has been renamed.

To learn more about declaring variables in Javascript, see the MDN.

Loading history...
412
		else
413
			navigator.clipboard.writeText(cellAddr.textContent + "@" + ae.GetDomainEml());
414
	};
415
416
	cellChk1.innerHTML = ae.GetAddressAccExt(num) ? "<input type=\"checkbox\" checked=\"checked\">" : "<input type=\"checkbox\">";
417
	cellChk2.innerHTML = ae.GetAddressAccInt(num) ? "<input type=\"checkbox\" checked=\"checked\">" : "<input type=\"checkbox\">";
418
	cellChk3.innerHTML = ae.GetAddressUse_Gk(num) ? "<input type=\"checkbox\" checked=\"checked\">" : "<input type=\"checkbox\">";
419
420
	cellBtnD.innerHTML = "<button type=\"button\">X</button>";
421
	cellBtnD.onclick = function() {deleteAddress(cellAddr.textContent);};
422
423
	const opt = document.createElement("option");
424
	opt.value = cellAddr.textContent;
425
	opt.textContent = cellAddr.textContent + "@" + ae.GetDomainEml();
426
	document.getElementById("write_from").appendChild(opt);
427
}
428
429
document.getElementById("btn_dele").onclick = function() {
430
	this.blur();
431
432
	if (tab === TAB_WRITE) {
433
		tabs[tab].cur = 0;
434
		updateTab();
435
436
		document.getElementById("write_recv").value = "";
437
		document.getElementById("write_subj").value = "";
438
		document.getElementById("write_body").value = "";
439
440
		document.getElementById("write_recv").focus();
441
	}
442
};
443
444
document.getElementById("btn_updt").onclick = function() {
445
	const btn = this;
446
	btn.disabled = true;
447
	btn.blur();
448
449
	if (tab === TAB_INBOX) {
450
		document.getElementById("tbl_inbox").style.opacity = 0.5;
451
452
		ae.Message_Browse(true, function(successBrowse) {
453
			document.getElementById("tbl_inbox").style.opacity = 1;
454
455
			if (successBrowse) {
456
				addMessages();
457
				btn.disabled = false;
458
			} else {
459
				console.log("Failed to refresh");
460
				btn.disabled = false;
461
			}
462
		});
463
	}
464
};
465
466
function addContact(mail, name, note) {
467
	const tbl = document.getElementById("tbl_ctact");
468
	const row = tbl.insertRow(-1);
469
	const cellMail = row.insertCell(-1);
470
	const cellName = row.insertCell(-1);
471
	const cellNote = row.insertCell(-1);
472
	const cellBtnD = row.insertCell(-1);
473
474
	cellMail.textContent = mail;
475
	cellName.textContent = name;
476
	cellNote.textContent = note;
477
	cellBtnD.innerHTML = "<button type=\"button\">X</button>";
478
479
	cellMail.contentEditable = true;
480
	cellName.contentEditable = true;
481
	cellNote.contentEditable = true;
482
483
	cellBtnD.onclick = function() {row.remove();};
484
}
485
486
document.getElementById("btn_newcontact").onclick = function() {
487
	addContact("", "", "");
488
};
489
490
document.getElementById("btn_savecontacts").onclick = function() {
491
	while (ae.GetContactCount() > 0) {
492
		ae.DeleteContact(0);
493
	}
494
495
	for (const row of document.getElementById("tbl_ctact").rows) {
496
		ae.AddContact(row.cells[0].textContent, row.cells[1].textContent, row.cells[2].textContent);
497
	}
498
499
	const btn = this;
500
	btn.disabled = true;
501
502
	ae.Private_Update(function(success) {
503
		btn.disabled = false;
504
505
		if (!success) {
506
			console.log("Failed contacts update");
507
		}
508
	});
509
};
510
511
function updateTab() {
512
	switch (tab) {
513
		case TAB_INBOX:
514
			addMessages();
515
		break;
516
517
		case TAB_WRITE:
518
			switch (tabs[tab].cur) {
519
				case 0: // Write
520
					document.getElementById("div_write_1").hidden = false;
521
					document.getElementById("div_write_2").hidden = true;
522
					document.getElementById("write_body").focus();
523
				break;
524
525
				case 1: // Verify
526
					ae.Address_Lookup(document.getElementById("write_recv").value, function(pk) {
527
						if (pk) {
528
							document.getElementById("div_write_1").hidden = true;
529
							document.getElementById("div_write_2").hidden = false;
530
531
							document.getElementById("write2_from").textContent = document.getElementById("write_from").value + "@" + ae.GetDomainEml();
532
							document.getElementById("write2_recv").textContent = document.getElementById("write_recv").value;
533
							document.getElementById("write2_pkey").textContent = sodium.to_hex(pk);
1 ignored issue
show
Bug introduced by
The variable sodium seems to be never declared. If this is a global, consider adding a /** global: sodium */ comment.

This checks looks for references to variables that have not been declared. This is most likey a typographical error or a variable has been renamed.

To learn more about declaring variables in Javascript, see the MDN.

Loading history...
534
535
							document.getElementById("write2_subj").textContent = document.getElementById("write_subj").value;
536
							document.getElementById("write2_rply").textContent = document.getElementById("write_rply").textContent;
537
							document.getElementById("write2_body").textContent = document.getElementById("write_body").value;
538
						} else {
539
							console.log("Failed lookup");
540
						}
541
					});
542
				break;
543
544
				case 2: // Send
545
					ae.Message_Create(
546
						document.getElementById("write_subj").value,
547
						document.getElementById("write_body").value,
548
						document.getElementById("write_from").value,
549
						document.getElementById("write_recv").value,
550
						document.getElementById("write_rply").textContent,
551
						(document.getElementById("write2_recv").textContent.indexOf("@") > 0) ? null : sodium.from_hex(document.getElementById("write2_pkey").textContent),
1 ignored issue
show
Bug introduced by
The variable sodium seems to be never declared. If this is a global, consider adding a /** global: sodium */ comment.

This checks looks for references to variables that have not been declared. This is most likey a typographical error or a variable has been renamed.

To learn more about declaring variables in Javascript, see the MDN.

Loading history...
552
						function(success) {
553
							if (success) {
554
								console.log("Sent ok");
555
							} else {
556
								console.log("Failed sending");
557
							}
558
						}
559
					);
560
				break;
561
			}
562
		break;
563
564
		case TAB_ADMIN:
565
			for (let i = 0; i <= tabs[tab].max; i++) {
566
				document.getElementById("div_admin").children[i].hidden = (i !== tabs[tab].cur);
567
			}
568
		break;
569
	}
570
571
	document.getElementById("btn_left").disabled = (tabs[tab].cur === 0);
572
	document.getElementById("btn_rght").disabled = (tabs[tab].cur === tabs[tab].max);
573
}
574
575
document.getElementById("btn_left").onclick = function() {
576
	tabs[tab].cur--;
577
	if (tabs[tab].cur === 0) this.disabled = true;
578
	if (tabs[tab].cur < tabs[tab].max) document.getElementById("btn_rght").disabled = false;
579
	updateTab();
580
	this.blur();
581
};
582
583
document.getElementById("btn_rght").onclick = function() {
584
	tabs[tab].cur++;
585
	if (tabs[tab].cur === tabs[tab].max) this.disabled = true;
586
	document.getElementById("btn_left").disabled = false;
587
	updateTab();
588
	this.blur();
589
};
590
591
const buttons = document.querySelector("#main1 > .top").getElementsByTagName("button");
592
for (let i = 0; i < buttons.length; i++) {
593
	buttons[i].onclick = function() {
594
		tab = i;
595
596
		for (let j = 0; j < buttons.length; j++) {
597
			document.querySelector("#main1 > .mid").children[j].hidden = (tab !== j);
598
			buttons[j].disabled = (tab === j);
599
		}
600
601
		document.getElementById("btn_left").disabled = (tabs[tab].cur === 0);
0 ignored issues
show
Bug introduced by
The variable tab is changed as part of the for loop for example by i on line 594. Only the value of the last iteration will be visible in this function if it is called after the loop.
Loading history...
602
		document.getElementById("btn_rght").disabled = (tabs[tab].cur === tabs[tab].max);
603
		document.getElementById("btn_dele").disabled = !tabs[tab].btnDele;
604
		document.getElementById("btn_updt").disabled = !tabs[tab].btnUpdt;
605
	};
606
}
607
608
function addressCreate(addr) {
609
	const btnN = document.getElementById("btn_address_create_normal");
610
	const btnS = document.getElementById("btn_address_create_shield");
611
	btnN.disabled = true;
612
	btnS.disabled = true;
613
614
	ae.Address_Create(addr, function(success1) {
615
		if (success1) {
616
			ae.Private_Update(function(success2) {
617
				addAddress(ae.GetAddressCount() - 1);
618
				if (addr !== "SHIELD") document.getElementById("txt_address_create_normal").value = "";
619
				updateAddressCounts();
620
621
				if (!success2) console.log("Failed to update the Private field");
622
623
				if (ae.GetAddressCountNormal() < ae.GetAddressLimitNormal(ae.GetUserLevel())) btnN.disabled = false;
624
				if (ae.GetAddressCountShield() < ae.GetAddressLimitShield(ae.GetUserLevel())) btnS.disabled = false;
625
			});
626
		} else {
627
			console.log("Failed to add address");
628
629
			if (ae.GetAddressCountNormal() < ae.GetAddressLimitNormal(ae.GetUserLevel())) btnN.disabled = false;
630
			if (ae.GetAddressCountShield() < ae.GetAddressLimitShield(ae.GetUserLevel())) btnS.disabled = false;
631
		}
632
	});
633
}
634
635
document.getElementById("btn_address_create_normal").onclick = function() {
636
	if (ae.GetAddressCountNormal() >= ae.GetAddressLimitNormal(ae.GetUserLevel())) return;
637
638
	const txtNewAddr = document.getElementById("txt_address_create_normal");
639
	if (!txtNewAddr.reportValidity()) return;
640
641
	addressCreate(txtNewAddr.value);
642
};
643
644
document.getElementById("btn_address_create_shield").onclick = function() {
645
	if (ae.GetAddressCountShield() >= ae.GetAddressLimitShield(ae.GetUserLevel())) return;
646
647
	addressCreate("SHIELD");
648
};
649
650
document.getElementById("chk_downme").onclick = function() {document.getElementById("btn_downme").disabled = !this.checked;};
651
document.getElementById("chk_killme").onclick = function() {document.getElementById("btn_killme").disabled = !this.checked;};
652
653
document.getElementById("txt_skey").onkeyup = function(event) {
654
	if (event.key === "Enter") {
655
		event.preventDefault();
656
		document.getElementById("btn_enter").click();
657
	}
658
};
659
660
document.getElementById("btn_enter").onclick = function() {
661
	const txtSkey = document.getElementById("txt_skey");
662
	if (!txtSkey.reportValidity()) return;
663
664
	const btn = this;
665
	btn.disabled = true;
666
	document.getElementById("txt_skey").style.background = "#233";
667
668
	ae.SetKeys(txtSkey.value, function(successSetKeys) {
669
		if (successSetKeys) {
670
			ae.Account_Browse(0, function(successBrowse) {
671
				if (successBrowse) {
672
					txtSkey.value = "";
673
674
					reloadAccount();
675
					document.getElementById("div_begin").hidden = true;
676
					document.getElementById("div_main").style.display = "grid";
677
678
					document.getElementById("btn_updt").click();
679
				} else {
680
					console.log("Failed to enter");
681
					btn.disabled = false;
682
					document.getElementById("txt_skey").style.background = "#466";
683
					txtSkey.focus();
684
				}
685
			});
686
		} else {
687
			console.log("Invalid format for key");
688
			btn.disabled = false;
689
			document.getElementById("txt_skey").style.background = "#466";
690
			txtSkey.focus();
691
		}
692
	});
693
};
694
695
});
696