sbbs/load/funcs.js

563 lines
13 KiB
JavaScript
Raw Normal View History

2020-07-18 00:36:49 +10:00
// Array of page owners
pageowners = [];
// String repeat.
2019-10-03 14:08:48 +10:00
if (!String.prototype.repeat) {
2019-10-15 21:48:16 +11:00
String.prototype.repeat = function(count) {
'use strict';
2022-04-16 15:36:17 +10:00
if (this === null) {
2019-10-15 21:48:16 +11:00
throw new TypeError('can\'t convert ' + this + ' to object');
}
var str = '' + this;
count = +count;
2022-04-16 15:36:17 +10:00
if (count !== count) {
2019-10-15 21:48:16 +11:00
count = 0;
}
if (count < 0) {
2022-04-16 15:36:17 +10:00
throw new RangeError('repeat count must be non-negative: '+count);
2019-10-15 21:48:16 +11:00
}
2022-04-16 15:36:17 +10:00
if (count === Infinity) {
2019-10-15 21:48:16 +11:00
throw new RangeError('repeat count must be less than infinity');
}
count = Math.floor(count);
2022-04-16 15:36:17 +10:00
if (str.length === 0 || count === 0) {
2019-10-15 21:48:16 +11:00
return '';
}
// Ensuring count is a 31-bit integer allows us to heavily optimize the
// main part. But anyway, most current (August 2014) browsers can't handle
// strings 1 << 28 chars or longer, so:
if (str.length * count >= 1 << 28) {
throw new RangeError('repeat count must not overflow maximum string size');
}
var rpt = '';
for (;;) {
2022-04-16 15:36:17 +10:00
if ((count & 1) === 1) {
2019-10-15 21:48:16 +11:00
rpt += str;
}
count >>>= 1;
2022-04-16 15:36:17 +10:00
if (count === 0) {
2019-10-15 21:48:16 +11:00
break;
}
str += str;
}
return rpt;
};
2019-10-03 14:08:48 +10:00
}
/*
// Cant enable this - problem with frame.js, line 451. c.open not a function
// Group By
if (!Array.prototype.groupby) {
Array.prototype.groupby = function(prop) {
return this.reduce(function(groups, item) {
const val = item[prop]
groups[val] = groups[val] || []
groups[val].push(item)
return groups
}, {})
};
}
if (!Array.prototype.min) {
Array.prototype.min = function() {
return this[0];
}
}
if (!Array.prototype.max) {
Array.prototype.max = function() {
return this.reverse()[0];
}
}
if (!Array.prototype.pluck) {
Array.prototype.pluck = function(item) {
var pluck = [];
for(var x in this) {
if (this[x][item])
pluck.push(this[x][item]);
}
return pluck;
}
}
*/
2020-08-12 23:23:26 +10:00
/**
* Convert ANSI into BIN for loading into a Frame
*
* @param ansi
*/
function ans2bin(ansi,frame) {
var x = new Graphic;
x.ANSI = ansi;
var o = 0; // offset into 'bin'
for (var yy = 0; yy < 22; yy++) {
for (var xx = 0; xx < 80; xx++) {
frame.setData(
xx,
yy,
x.BIN.substr(o,1),
x.BIN.substr(o+1,1).charCodeAt(0) || BG_BLACK
);
o = o+2;
}
}
}
2022-12-09 17:19:33 +11:00
/**
* Dynamic Field Processing
*
* @param field field we want to get a value for
* @param length length to fill this field
* @param pad the padding character if field is less than length
* @param context a context value to determine the value from
* @returns {string|*|null}
* @note bbs.atcodes() cannot process modifiers, so this function is a replacement.
*/
function atcode(field,length,pad,context) {
'use strict';
pad = pad ? pad : ' ';
var result = '';
2022-05-13 22:31:47 +10:00
var args = [];
if (field.search(/:/)) {
args = field.split(':');
field = args.shift();
}
//log(LOG_DEBUG,'Field:'+field,'Args:'+JSON.stringify(args));
switch(field) {
// Get the ECHOAREA FTN AREA_TAG
2022-05-03 21:10:09 +10:00
case 'msg_area_areatag':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
result = context.msgbase.cfg.area_tag;
break;
// Get the ECHOAREA Description
2022-05-03 21:10:09 +10:00
case 'msg_area_desc':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
result = context.msgbase.cfg.description;
break;
2022-05-03 21:10:09 +10:00
// Oldest message in msgarea
// Our oldest message, is the first message with a tag from the headers
2022-05-03 21:10:09 +10:00
case 'msg_area_msgoldest_date':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
var x = context.list_tagged[0];
2022-05-03 21:10:09 +10:00
result = x ? x.date : '';
2022-05-03 21:10:09 +10:00
break;
case 'msg_area_msgoldest_page':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
var x = context.list_tagged[0];
2022-05-03 21:10:09 +10:00
return x ? context.getMessagePage(x.number) : null;
2022-05-03 21:10:09 +10:00
// Newest message in msgarea
// Our newest message, is the last message with a tag from the headers
2022-05-03 21:10:09 +10:00
case 'msg_area_msgnewest_date':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
var x = context.list_tagged[context.list_tagged.length-1];
result = x ? x.date : '';
2022-05-03 21:10:09 +10:00
break;
case 'msg_area_msgnewest_page':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
var x = context.list_tagged[context.list_tagged.length-1];
return x ? context.getMessagePage(x.number) : null;
2022-05-03 21:10:09 +10:00
// First unread message
case 'msg_area_msgunread_date':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
var x = context.newMsgs();
result = x.length ? x.shift().date : '';
2022-05-03 21:10:09 +10:00
break;
case 'msg_area_msgunread_page':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
var x = context.newMsgs();
return x.length ? context.getMessagePage(x.shift().number) : null;
2022-05-03 21:10:09 +10:00
// First unread message to me
case 'msg_area_msgotome_date':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
var x = context.newMsgsToMe();
2022-05-03 21:10:09 +10:00
result = x.length > 1 ? x[1].date : '';
2022-05-03 21:10:09 +10:00
break;
case 'msg_area_msgtome_page':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
var x = context.newMsgsToMe();
return x.length > 1 ? context.getMessagePage(x[1].number) : null;
2022-05-03 21:10:09 +10:00
// Count of unread messages
case 'msg_area_new':
if (args.length === 1) {
context = new MsgArea();
context.code = args[0];
}
2022-05-03 21:10:09 +10:00
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
result = ''+context.newMsgs().length;
2022-05-03 21:10:09 +10:00
break;
// Count of unread messages to me
case 'msg_area_newtome':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
result = ''+(context.newMsgsToMe().length > 1 ? context.newMsgsToMe().length-1 : 0);
2022-05-03 21:10:09 +10:00
break;
// Is this message area in my new scan list
case 'msg_area_newscan':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
result = (context.getUserStats().scan_ptr & SCAN_CFG_TOYOU) ? 'YES' : 'NO';
break;
// Is this message area in my new scan list
case 'msg_area_pending':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
result = ''+context.list_untagged.length;
break;
// Get the ECHOAREA Total Number of Messages
case 'msg_area_total':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
result = ''+context.msgbase.total_msgs;
break;
// Get the ECHOAREA Group Name
case 'msg_grp_name':
if (typeof context !== 'object') {
log(LOG_ERROR,'Unable to render ['+field+'], no context provided');
break;
}
result = context.zone_name;
break;
case 'nodeid':
result = getNodeID();
break;
default:
2022-12-09 17:19:33 +11:00
result = (typeof bbs === 'undefined') ? '*'.repeat(Math.abs(length)) : bbs.atcode(field);
}
2022-05-03 21:10:09 +10:00
if ((result === null) || (typeof result === 'undefined'))
result = '';
length = length ? length : result.length;
//log(LOG_DEBUG,' - result length ['+result.length+'] desired ('+length+')');
if (result.length < Math.abs(length))
result = (length < 0) ? padright(result,Math.abs(length),pad) : padleft(result,length,pad);
else if (result.length > Math.abs(length))
result = result.substr(0,Math.abs(length));
log(LOG_DEBUG,'- ATCODE ['+field+'] ('+length+'|"'+pad+'") returns ['+result+']');
return result;
}
2019-10-15 21:48:16 +11:00
/**
* Find a message base by code
*
* @param code
* @returns {number | string|boolean}
* @deprecated Can this move to the msgbases.js?
2019-10-15 21:48:16 +11:00
*/
function findMsgBase(code) {
2019-10-15 21:48:16 +11:00
if (! code)
code = "vtx_data";
2019-10-15 21:48:16 +11:00
for (var s in msg_area.sub) {
var sub = msg_area.sub[s];
writeln('sub:'+sub.code);
2019-10-15 21:48:16 +11:00
2022-04-16 15:36:17 +10:00
if (sub.code.substr(-code.length).toLowerCase() === code)
return sub;
2019-10-15 21:48:16 +11:00
}
return false;
2019-10-03 14:08:48 +10:00
}
/**
* Return an argument from argv, or an error if it doesnt exit
*
* @param key
* @param error
2022-04-16 15:36:17 +10:00
* @param abort
2019-10-03 14:08:48 +10:00
*/
function getArg(key,error,abort) {
2019-10-15 21:48:16 +11:00
index = argv.indexOf(key);
2019-10-03 14:08:48 +10:00
2019-10-15 21:48:16 +11:00
if ((index !== -1) && (! (argv[index+1] === undefined || argv[index+1].match(/^-/)))) {
return argv[index+1];
}
2019-10-03 14:08:48 +10:00
2019-10-15 21:48:16 +11:00
if (abort) {
log(LOG_ERROR,error);
exit(1);
}
2019-10-03 14:08:48 +10:00
}
2020-11-01 21:55:32 +11:00
/**
* Returns the current node number
*
* @returns {string}
*/
function getNodeID() {
var regex = new RegExp('^'+SYSTEM_ZONE+':');
var matches = [];
2020-11-01 21:55:32 +11:00
for each (var addr in system.fido_addr_list)
{
if (regex.test(addr)) {
addr = addr.replace(regex,'');
matches = addr.split('/',2);
break;
}
}
2022-12-09 17:19:33 +11:00
return (matches.length === 0) ? '-' : padright(matches[0],3,'0')+padright(matches[1],3,'0')+padright((typeof bbs === 'undefined') ? 0 : bbs.node_num,2,'0');
2020-11-01 21:55:32 +11:00
}
function getPageOwners() {
// Load the owner configuration into memory
if (! pageowners.length) {
var f = new File(file_cfgname(system.mods_dir,'ansitex/ctrl/videotex.ini'));
if (f.open('r')) {
2020-07-30 23:47:40 +10:00
var logoans = f.iniGetValue('prefix','logoans');
var logovtx = f.iniGetValue('prefix','logovtx');
var users = f.iniGetValue('prefix','user');
//log(LOG_DEBUG,'+ pageOwner: users='+JSON.stringify(users));
2020-07-30 23:47:40 +10:00
pageowners.push({prefix: 0,logoans: logoans,logovtx: logovtx,user:users});
f.iniGetSections('prefix:').forEach(function (prefix) {
var p = parseInt(prefix.substr(7));
2020-07-30 23:47:40 +10:00
var logoans = f.iniGetValue(prefix,'logoans','');
var logovtx = f.iniGetValue(prefix,'logovtx','');
var users = f.iniGetValue(prefix,'user','');
//log(LOG_DEBUG,'+ pageOwner: users='+JSON.stringify(users));
2020-07-30 23:47:40 +10:00
pageowners.push({prefix: p,logoans: logoans,logovtx: logovtx,user:users});
});
}
f.close();
// Sort the pageowners ascending
pageowners.sort(function(a,b) { return (a.prefix < b.prefix) ? 1 : ((b.prefix < a.prefix) ? -1 : 0); });
//log(LOG_DEBUG,'+ pageOwner: pageowners='+JSON.stringify(pageowners));
}
return pageowners;
}
function loadOptions(option) {
var f = new File(file_cfgname(system.mods_dir,'ansitex/ctrl/videotex.ini'));
2019-10-20 22:31:15 +11:00
if (! f.open('r')) {
2019-10-20 22:31:15 +11:00
return undefined;
}
val = f.iniGetObject(option);
2019-10-20 22:31:15 +11:00
f.close();
2019-10-20 22:31:15 +11:00
return val;
}
function msgBaseImport(msgbase,page,text) {
var msgbase = new MsgBase(findMsgBase(msgbase).code);
2019-10-15 21:48:16 +11:00
log(LOG_DEBUG,'Sending ['+page+'] to message base ['+msgbase.cfg.code+']');
var hdr = { to:'All', from:'Videotex', subject:'Frame: '+page };
var body = '';
body += text+"\r\n";
body += "--- " + js.exec_file + " " + '1.0' + "\r\n";
return msgbase.save_msg(hdr, body);
}
// Right Pad a string with char c
function padright(n,width,c) {
c = c || '0';
n = n + '';
return n.length >= width ? n : new Array(width - n.length + 1).join(c) + n;
}
// Left Pad a string with char c
function padleft(n,width,c) {
c = c || '0';
2020-11-01 21:55:32 +11:00
n = n + '';
return n.length >= width ? n : n+new Array(width - n.length + 1).join(c);
2020-11-01 21:55:32 +11:00
}
2019-10-03 14:08:48 +10:00
/**
* Return the frame as a string
*/
function pageStr(page) {
2020-03-26 17:22:46 +11:00
if (page.frame==null)
return null;
2019-10-15 21:48:16 +11:00
if (! page.index)
page.index = 'a';
2019-10-03 14:08:48 +10:00
2020-07-18 00:36:49 +10:00
return page.frame.toString()+page.index;
2019-10-03 14:08:48 +10:00
}
2019-10-22 21:57:25 +11:00
/**
2020-07-18 00:36:49 +10:00
* Read our videotex.ini configuration and determine who owns a page.
* If there is no prefix for the page, it is owned by the system '0'
2019-10-22 21:57:25 +11:00
*
2020-07-18 00:36:49 +10:00
* @param page
* @returns {undefined}
2019-10-22 21:57:25 +11:00
*/
2020-07-18 00:36:49 +10:00
function pageOwner(page) {
var BreakException = {};
var o = null;
2020-07-18 00:36:49 +10:00
try {
getPageOwners().forEach(function(owner) {
var p = owner.prefix.toString();
o = owner;
2020-07-18 00:36:49 +10:00
var re = new RegExp('^' + p, 'g');
if (page.toString().match(re)) {
//log(LOG_DEBUG,'= pageOwner: p='+p+',o: '+o);
throw BreakException;
}
});
2020-07-18 00:36:49 +10:00
} catch (e) {
if (e !== BreakException) throw e;
}
2020-07-18 00:36:49 +10:00
//log(LOG_DEBUG,'+ pageOwner: page='+page+', owner: '+JSON.stringify(o));
2020-07-22 22:37:00 +10:00
return o;
}
2020-07-18 00:36:49 +10:00
/**
* Can the user edit the frame
*
* @param page
* @param user
*/
function pageEditor(page) {
//log(LOG_DEBUG,'+ pageEditor: page='+page+', user #'+user.number);
2019-10-22 21:57:25 +11:00
var BreakException = {};
var pageditor = false;
2020-07-18 00:36:49 +10:00
try {
getPageOwners().forEach(function(owner) {
var p = owner.prefix.toString();
//log(LOG_DEBUG,' - pageEditor: '+JSON.stringify(owner));
frameusers = owner.user ? owner.user.toString().split(',') : [1];
2020-07-18 00:36:49 +10:00
log(LOG_DEBUG,' - pageEditor: p='+p+'('+p.length+') user ['+JSON.stringify(frameusers)+'] - :'+frameusers.indexOf(user.number.toString()));
2020-07-18 00:36:49 +10:00
var re = new RegExp('^' + p, 'g');
if (page.toString().match(re) && (frameusers.indexOf(user.number.toString()) !== -1)) {
pageditor = true;
throw BreakException;
}
});
} catch (e) {
if (e !== BreakException) throw e;
}
2020-07-18 00:36:49 +10:00
2020-07-22 22:37:00 +10:00
log(LOG_DEBUG,'+ pageEditor: page='+page+', editor: '+JSON.stringify(pageditor));
return pageditor;
2020-07-18 00:36:49 +10:00
}
2019-10-22 21:57:25 +11:00
/**
* This function returns a list of zones used by this system.
*/
function zones() {
var z = [];
var ftn = /([0-9]+):([0-9]+)\/([0-9]+)(\.([0-9]+))?/;
for(var g=0;g<system.fido_addr_list.length;g++) {
z.push(parseInt(system.fido_addr_list[g].match(ftn)[1]));
}
return z.sort(function(a,b) {
return (a>b);
});
2019-10-22 21:57:25 +11:00
}
2020-11-01 21:55:32 +11:00
this;