| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306 | var pas = {};var rtl = {  version: 10501,  quiet: false,  debug_load_units: false,  debug_rtti: false,  debug: function(){    if (rtl.quiet || !console || !console.log) return;    console.log(arguments);  },  error: function(s){    rtl.debug('Error: ',s);    throw s;  },  warn: function(s){    rtl.debug('Warn: ',s);  },  checkVersion: function(v){    if (rtl.version != v) throw "expected rtl version "+v+", but found "+rtl.version;  },  hasString: function(s){    return rtl.isString(s) && (s.length>0);  },  isArray: function(a) {    return Array.isArray(a);  },  isFunction: function(f){    return typeof(f)==="function";  },  isModule: function(m){    return rtl.isObject(m) && rtl.hasString(m.$name) && (pas[m.$name]===m);  },  isImplementation: function(m){    return rtl.isObject(m) && rtl.isModule(m.$module) && (m.$module.$impl===m);  },  isNumber: function(n){    return typeof(n)==="number";  },  isObject: function(o){    var s=typeof(o);    return (typeof(o)==="object") && (o!=null);  },  isString: function(s){    return typeof(s)==="string";  },  getNumber: function(n){    return typeof(n)==="number"?n:NaN;  },  getChar: function(c){    return ((typeof(c)==="string") && (c.length===1)) ? c : "";  },  getObject: function(o){    return ((typeof(o)==="object") || (typeof(o)==='function')) ? o : null;  },  isTRecord: function(type){    return (rtl.isObject(type) && type.hasOwnProperty('$new') && (typeof(type.$new)==='function'));  },  isPasClass: function(type){    return (rtl.isObject(type) && type.hasOwnProperty('$classname') && rtl.isObject(type.$module));  },  isPasClassInstance: function(type){    return (rtl.isObject(type) && rtl.isPasClass(type.$class));  },  hexStr: function(n,digits){    return ("000000000000000"+n.toString(16).toUpperCase()).slice(-digits);  },  m_loading: 0,  m_loading_intf: 1,  m_intf_loaded: 2,  m_loading_impl: 3, // loading all used unit  m_initializing: 4, // running initialization  m_initialized: 5,  module: function(module_name, intfuseslist, intfcode, impluseslist, implcode){    if (rtl.debug_load_units) rtl.debug('rtl.module name="'+module_name+'" intfuses='+intfuseslist+' impluses='+impluseslist+' hasimplcode='+rtl.isFunction(implcode));    if (!rtl.hasString(module_name)) rtl.error('invalid module name "'+module_name+'"');    if (!rtl.isArray(intfuseslist)) rtl.error('invalid interface useslist of "'+module_name+'"');    if (!rtl.isFunction(intfcode)) rtl.error('invalid interface code of "'+module_name+'"');    if (!(impluseslist==undefined) && !rtl.isArray(impluseslist)) rtl.error('invalid implementation useslist of "'+module_name+'"');    if (!(implcode==undefined) && !rtl.isFunction(implcode)) rtl.error('invalid implementation code of "'+module_name+'"');    if (pas[module_name])      rtl.error('module "'+module_name+'" is already registered');    var module = pas[module_name] = {      $name: module_name,      $intfuseslist: intfuseslist,      $impluseslist: impluseslist,      $state: rtl.m_loading,      $intfcode: intfcode,      $implcode: implcode,      $impl: null,      $rtti: Object.create(rtl.tSectionRTTI)    };    module.$rtti.$module = module;    if (implcode) module.$impl = {      $module: module,      $rtti: module.$rtti    };  },  exitcode: 0,  run: function(module_name){      function doRun(){      if (!rtl.hasString(module_name)) module_name='program';      if (rtl.debug_load_units) rtl.debug('rtl.run module="'+module_name+'"');      rtl.initRTTI();      var module = pas[module_name];      if (!module) rtl.error('rtl.run module "'+module_name+'" missing');      rtl.loadintf(module);      rtl.loadimpl(module);      if (module_name=='program'){        if (rtl.debug_load_units) rtl.debug('running $main');        var r = pas.program.$main();        if (rtl.isNumber(r)) rtl.exitcode = r;      }    }        if (rtl.showUncaughtExceptions) {      try{        doRun();      } catch(re) {        var errMsg = rtl.hasString(re.$classname) ? re.$classname : '';	    errMsg +=  ((errMsg) ? ': ' : '') + (re.hasOwnProperty('fMessage') ? re.fMessage : re);        alert('Uncaught Exception : '+errMsg);        rtl.exitCode = 216;      }    } else {      doRun();    }    return rtl.exitcode;  },  loadintf: function(module){    if (module.$state>rtl.m_loading_intf) return; // already finished    if (rtl.debug_load_units) rtl.debug('loadintf: "'+module.$name+'"');    if (module.$state===rtl.m_loading_intf)      rtl.error('unit cycle detected "'+module.$name+'"');    module.$state=rtl.m_loading_intf;    // load interfaces of interface useslist    rtl.loaduseslist(module,module.$intfuseslist,rtl.loadintf);    // run interface    if (rtl.debug_load_units) rtl.debug('loadintf: run intf of "'+module.$name+'"');    module.$intfcode(module.$intfuseslist);    // success    module.$state=rtl.m_intf_loaded;    // Note: units only used in implementations are not yet loaded (not even their interfaces)  },  loaduseslist: function(module,useslist,f){    if (useslist==undefined) return;    for (var i in useslist){      var unitname=useslist[i];      if (rtl.debug_load_units) rtl.debug('loaduseslist of "'+module.$name+'" uses="'+unitname+'"');      if (pas[unitname]==undefined)        rtl.error('module "'+module.$name+'" misses "'+unitname+'"');      f(pas[unitname]);    }  },  loadimpl: function(module){    if (module.$state>=rtl.m_loading_impl) return; // already processing    if (module.$state<rtl.m_intf_loaded) rtl.error('loadimpl: interface not loaded of "'+module.$name+'"');    if (rtl.debug_load_units) rtl.debug('loadimpl: load uses of "'+module.$name+'"');    module.$state=rtl.m_loading_impl;    // load interfaces of implementation useslist    rtl.loaduseslist(module,module.$impluseslist,rtl.loadintf);    // load implementation of interfaces useslist    rtl.loaduseslist(module,module.$intfuseslist,rtl.loadimpl);    // load implementation of implementation useslist    rtl.loaduseslist(module,module.$impluseslist,rtl.loadimpl);    // Note: At this point all interfaces used by this unit are loaded. If    //   there are implementation uses cycles some used units might not yet be    //   initialized. This is by design.    // run implementation    if (rtl.debug_load_units) rtl.debug('loadimpl: run impl of "'+module.$name+'"');    if (rtl.isFunction(module.$implcode)) module.$implcode(module.$impluseslist);    // run initialization    if (rtl.debug_load_units) rtl.debug('loadimpl: run init of "'+module.$name+'"');    module.$state=rtl.m_initializing;    if (rtl.isFunction(module.$init)) module.$init();    // unit initialized    module.$state=rtl.m_initialized;  },  createCallback: function(scope, fn){    var cb;    if (typeof(fn)==='string'){      cb = function(){        return scope[fn].apply(scope,arguments);      };    } else {      cb = function(){        return fn.apply(scope,arguments);      };    };    cb.scope = scope;    cb.fn = fn;    return cb;  },  cloneCallback: function(cb){    return rtl.createCallback(cb.scope,cb.fn);  },  eqCallback: function(a,b){    // can be a function or a function wrapper    if (a==b){      return true;    } else {      return (a!=null) && (b!=null) && (a.fn) && (a.scope===b.scope) && (a.fn==b.fn);    }  },  initStruct: function(c,parent,name){    if ((parent.$module) && (parent.$module.$impl===parent)) parent=parent.$module;    c.$parent = parent;    if (rtl.isModule(parent)){      c.$module = parent;      c.$name = name;    } else {      c.$module = parent.$module;      c.$name = parent.$name+'.'+name;    };    return parent;  },  initClass: function(c,parent,name,initfn){    parent[name] = c;    c.$class = c; // Note: o.$class === Object.getPrototypeOf(o)    c.$classname = name;    parent = rtl.initStruct(c,parent,name);    c.$fullname = parent.$name+'.'+name;    // rtti    if (rtl.debug_rtti) rtl.debug('initClass '+c.$fullname);    var t = c.$module.$rtti.$Class(c.$name,{ "class": c });    c.$rtti = t;    if (rtl.isObject(c.$ancestor)) t.ancestor = c.$ancestor.$rtti;    if (!t.ancestor) t.ancestor = null;    // init members    initfn.call(c);  },  createClass: function(parent,name,ancestor,initfn){    // create a normal class,    // ancestor must be null or a normal class,    // the root ancestor can be an external class    var c = null;    if (ancestor != null){      c = Object.create(ancestor);      c.$ancestor = ancestor;      // Note:      // if root is an "object" then c.$ancestor === Object.getPrototypeOf(c)      // if root is a "function" then c.$ancestor === c.__proto__, Object.getPrototypeOf(c) returns the root    } else {      c = {};      c.$create = function(fn,args){        if (args == undefined) args = [];        var o = Object.create(this);        o.$init();        try{          if (typeof(fn)==="string"){            o[fn].apply(o,args);          } else {            fn.apply(o,args);          };          o.AfterConstruction();        } catch($e){          // do not call BeforeDestruction          if (o.Destroy) o.Destroy();          o.$final();          throw $e;        }        return o;      };      c.$destroy = function(fnname){        this.BeforeDestruction();        if (this[fnname]) this[fnname]();        this.$final();      };    };    rtl.initClass(c,parent,name,initfn);  },  createClassExt: function(parent,name,ancestor,newinstancefnname,initfn){    // Create a class using an external ancestor.    // If newinstancefnname is given, use that function to create the new object.    // If exist call BeforeDestruction and AfterConstruction.    var c = Object.create(ancestor);    c.$create = function(fn,args){      if (args == undefined) args = [];      var o = null;      if (newinstancefnname.length>0){        o = this[newinstancefnname](fn,args);      } else {        o = Object.create(this);      }      if (o.$init) o.$init();      try{        if (typeof(fn)==="string"){          o[fn].apply(o,args);        } else {          fn.apply(o,args);        };        if (o.AfterConstruction) o.AfterConstruction();      } catch($e){        // do not call BeforeDestruction        if (o.Destroy) o.Destroy();        if (o.$final) this.$final();        throw $e;      }      return o;    };    c.$destroy = function(fnname){      if (this.BeforeDestruction) this.BeforeDestruction();      if (this[fnname]) this[fnname]();      if (this.$final) this.$final();    };    rtl.initClass(c,parent,name,initfn);  },  createHelper: function(parent,name,ancestor,initfn){    // create a helper,    // ancestor must be null or a helper,    var c = null;    if (ancestor != null){      c = Object.create(ancestor);      c.$ancestor = ancestor;      // c.$ancestor === Object.getPrototypeOf(c)    } else {      c = {};    };    parent[name] = c;    c.$class = c; // Note: o.$class === Object.getPrototypeOf(o)    c.$classname = name;    parent = rtl.initStruct(c,parent,name);    c.$fullname = parent.$name+'.'+name;    // rtti    var t = c.$module.$rtti.$Helper(c.$name,{ "helper": c });    c.$rtti = t;    if (rtl.isObject(ancestor)) t.ancestor = ancestor.$rtti;    if (!t.ancestor) t.ancestor = null;    // init members    initfn.call(c);  },  tObjectDestroy: "Destroy",  free: function(obj,name){    if (obj[name]==null) return null;    obj[name].$destroy(rtl.tObjectDestroy);    obj[name]=null;  },  freeLoc: function(obj){    if (obj==null) return null;    obj.$destroy(rtl.tObjectDestroy);    return null;  },  recNewT: function(parent,name,initfn,full){    // create new record type    var t = {};    if (parent) parent[name] = t;    function hide(prop){      Object.defineProperty(t,prop,{enumerable:false});    }    if (full){      rtl.initStruct(t,parent,name);      t.$record = t;      hide('$record');      hide('$name');      hide('$parent');      hide('$module');    }    initfn.call(t);    if (!t.$new){      t.$new = function(){ return Object.create(this); };    }    t.$clone = function(r){ return this.$new().$assign(r); };    hide('$new');    hide('$clone');    hide('$eq');    hide('$assign');    return t;  },  is: function(instance,type){    return type.isPrototypeOf(instance) || (instance===type);  },  isExt: function(instance,type,mode){    // mode===1 means instance must be a Pascal class instance    // mode===2 means instance must be a Pascal class    // Notes:    // isPrototypeOf and instanceof return false on equal    // isPrototypeOf does not work for Date.isPrototypeOf(new Date())    //   so if isPrototypeOf is false test with instanceof    // instanceof needs a function on right side    if (instance == null) return false; // Note: ==null checks for undefined too    if ((typeof(type) !== 'object') && (typeof(type) !== 'function')) return false;    if (instance === type){      if (mode===1) return false;      if (mode===2) return rtl.isPasClass(instance);      return true;    }    if (type.isPrototypeOf && type.isPrototypeOf(instance)){      if (mode===1) return rtl.isPasClassInstance(instance);      if (mode===2) return rtl.isPasClass(instance);      return true;    }    if ((typeof type == 'function') && (instance instanceof type)) return true;    return false;  },  Exception: null,  EInvalidCast: null,  EAbstractError: null,  ERangeError: null,  EIntOverflow: null,  EPropWriteOnly: null,  raiseE: function(typename){    var t = rtl[typename];    if (t==null){      var mod = pas.SysUtils;      if (!mod) mod = pas.sysutils;      if (mod){        t = mod[typename];        if (!t) t = mod[typename.toLowerCase()];        if (!t) t = mod['Exception'];        if (!t) t = mod['exception'];      }    }    if (t){      if (t.Create){        throw t.$create("Create");      } else if (t.create){        throw t.$create("create");      }    }    if (typename === "EInvalidCast") throw "invalid type cast";    if (typename === "EAbstractError") throw "Abstract method called";    if (typename === "ERangeError") throw "range error";    throw typename;  },  as: function(instance,type){    if((instance === null) || rtl.is(instance,type)) return instance;    rtl.raiseE("EInvalidCast");  },  asExt: function(instance,type,mode){    if((instance === null) || rtl.isExt(instance,type,mode)) return instance;    rtl.raiseE("EInvalidCast");  },  createInterface: function(module, name, guid, fnnames, ancestor, initfn){    //console.log('createInterface name="'+name+'" guid="'+guid+'" names='+fnnames);    var i = ancestor?Object.create(ancestor):{};    module[name] = i;    i.$module = module;    i.$name = name;    i.$fullname = module.$name+'.'+name;    i.$guid = guid;    i.$guidr = null;    i.$names = fnnames?fnnames:[];    if (rtl.isFunction(initfn)){      // rtti      if (rtl.debug_rtti) rtl.debug('createInterface '+i.$fullname);      var t = i.$module.$rtti.$Interface(name,{ "interface": i, module: module });      i.$rtti = t;      if (ancestor) t.ancestor = ancestor.$rtti;      if (!t.ancestor) t.ancestor = null;      initfn.call(i);    }    return i;  },  strToGUIDR: function(s,g){    var p = 0;    function n(l){      var h = s.substr(p,l);      p+=l;      return parseInt(h,16);    }    p+=1; // skip {    g.D1 = n(8);    p+=1; // skip -    g.D2 = n(4);    p+=1; // skip -    g.D3 = n(4);    p+=1; // skip -    if (!g.D4) g.D4=[];    g.D4[0] = n(2);    g.D4[1] = n(2);    p+=1; // skip -    for(var i=2; i<8; i++) g.D4[i] = n(2);    return g;  },  guidrToStr: function(g){    if (g.$intf) return g.$intf.$guid;    var h = rtl.hexStr;    var s='{'+h(g.D1,8)+'-'+h(g.D2,4)+'-'+h(g.D3,4)+'-'+h(g.D4[0],2)+h(g.D4[1],2)+'-';    for (var i=2; i<8; i++) s+=h(g.D4[i],2);    s+='}';    return s;  },  createTGUID: function(guid){    var TGuid = (pas.System)?pas.System.TGuid:pas.system.tguid;    var g = rtl.strToGUIDR(guid,TGuid.$new());    return g;  },  getIntfGUIDR: function(intfTypeOrVar){    if (!intfTypeOrVar) return null;    if (!intfTypeOrVar.$guidr){      var g = rtl.createTGUID(intfTypeOrVar.$guid);      if (!intfTypeOrVar.hasOwnProperty('$guid')) intfTypeOrVar = Object.getPrototypeOf(intfTypeOrVar);      g.$intf = intfTypeOrVar;      intfTypeOrVar.$guidr = g;    }    return intfTypeOrVar.$guidr;  },  addIntf: function (aclass, intf, map){    function jmp(fn){      if (typeof(fn)==="function"){        return function(){ return fn.apply(this.$o,arguments); };      } else {        return function(){ rtl.raiseE('EAbstractError'); };      }    }    if(!map) map = {};    var t = intf;    var item = Object.create(t);    if (!aclass.hasOwnProperty('$intfmaps')) aclass.$intfmaps = {};    aclass.$intfmaps[intf.$guid] = item;    do{      var names = t.$names;      if (!names) break;      for (var i=0; i<names.length; i++){        var intfname = names[i];        var fnname = map[intfname];        if (!fnname) fnname = intfname;        //console.log('addIntf: intftype='+t.$name+' index='+i+' intfname="'+intfname+'" fnname="'+fnname+'" old='+typeof(item[intfname]));        item[intfname] = jmp(aclass[fnname]);      }      t = Object.getPrototypeOf(t);    }while(t!=null);  },  getIntfG: function (obj, guid, query){    if (!obj) return null;    //console.log('getIntfG: obj='+obj.$classname+' guid='+guid+' query='+query);    // search    var maps = obj.$intfmaps;    if (!maps) return null;    var item = maps[guid];    if (!item) return null;    // check delegation    //console.log('getIntfG: obj='+obj.$classname+' guid='+guid+' query='+query+' item='+typeof(item));    if (typeof item === 'function') return item.call(obj); // delegate. Note: COM contains _AddRef    // check cache    var intf = null;    if (obj.$interfaces){      intf = obj.$interfaces[guid];      //console.log('getIntfG: obj='+obj.$classname+' guid='+guid+' cache='+typeof(intf));    }    if (!intf){ // intf can be undefined!      intf = Object.create(item);      intf.$o = obj;      if (!obj.$interfaces) obj.$interfaces = {};      obj.$interfaces[guid] = intf;    }    if (typeof(query)==='object'){      // called by queryIntfT      var o = null;      if (intf.QueryInterface(rtl.getIntfGUIDR(query),          {get:function(){ return o; }, set:function(v){ o=v; }}) === 0){        return o;      } else {        return null;      }    } else if(query===2){      // called by TObject.GetInterfaceByStr      if (intf.$kind === 'com') intf._AddRef();    }    return intf;  },  getIntfT: function(obj,intftype){    return rtl.getIntfG(obj,intftype.$guid);  },  queryIntfT: function(obj,intftype){    return rtl.getIntfG(obj,intftype.$guid,intftype);  },  queryIntfIsT: function(obj,intftype){    var i = rtl.queryIntfG(obj,intftype.$guid);    if (!i) return false;    if (i.$kind === 'com') i._Release();    return true;  },  asIntfT: function (obj,intftype){    var i = rtl.getIntfG(obj,intftype.$guid);    if (i!==null) return i;    rtl.raiseEInvalidCast();  },  intfIsClass: function(intf,classtype){    return (intf!=null) && (rtl.is(intf.$o,classtype));  },  intfAsClass: function(intf,classtype){    if (intf==null) return null;    return rtl.as(intf.$o,classtype);  },  intfToClass: function(intf,classtype){    if ((intf!==null) && rtl.is(intf.$o,classtype)) return intf.$o;    return null;  },  // interface reference counting  intfRefs: { // base object for temporary interface variables    ref: function(id,intf){      // called for temporary interface references needing delayed release      var old = this[id];      //console.log('rtl.intfRefs.ref: id='+id+' old="'+(old?old.$name:'null')+'" intf="'+(intf?intf.$name:'null')+' $o='+(intf?intf.$o:'null'));      if (old){        // called again, e.g. in a loop        delete this[id];        old._Release(); // may fail      }      this[id]=intf;      return intf;    },    free: function(){      //console.log('rtl.intfRefs.free...');      for (var id in this){        if (this.hasOwnProperty(id)){          //console.log('rtl.intfRefs.free: id='+id+' '+this[id].$name+' $o='+this[id].$o.$classname);          this[id]._Release();        }      }    }  },  createIntfRefs: function(){    //console.log('rtl.createIntfRefs');    return Object.create(rtl.intfRefs);  },  setIntfP: function(path,name,value,skipAddRef){    var old = path[name];    //console.log('rtl.setIntfP path='+path+' name='+name+' old="'+(old?old.$name:'null')+'" value="'+(value?value.$name:'null')+'"');    if (old === value) return;    if (old !== null){      path[name]=null;      old._Release();    }    if (value !== null){      if (!skipAddRef) value._AddRef();      path[name]=value;    }  },  setIntfL: function(old,value,skipAddRef){    //console.log('rtl.setIntfL old="'+(old?old.$name:'null')+'" value="'+(value?value.$name:'null')+'"');    if (old !== value){      if (value!==null){        if (!skipAddRef) value._AddRef();      }      if (old!==null){        old._Release();  // Release after AddRef, to avoid double Release if Release creates an exception      }    } else if (skipAddRef){      if (old!==null){        old._Release();  // value has an AddRef      }    }    return value;  },  _AddRef: function(intf){    //if (intf) console.log('rtl._AddRef intf="'+(intf?intf.$name:'null')+'"');    if (intf) intf._AddRef();    return intf;  },  _Release: function(intf){    //if (intf) console.log('rtl._Release intf="'+(intf?intf.$name:'null')+'"');    if (intf) intf._Release();    return intf;  },  checkMethodCall: function(obj,type){    if (rtl.isObject(obj) && rtl.is(obj,type)) return;    rtl.raiseE("EInvalidCast");  },  oc: function(i){    // overflow check integer    if ((Math.floor(i)===i) && (i>=-0x1fffffffffffff) && (i<=0x1fffffffffffff)) return i;    rtl.raiseE('EIntOverflow');  },  rc: function(i,minval,maxval){    // range check integer    if ((Math.floor(i)===i) && (i>=minval) && (i<=maxval)) return i;    rtl.raiseE('ERangeError');  },  rcc: function(c,minval,maxval){    // range check char    if ((typeof(c)==='string') && (c.length===1)){      var i = c.charCodeAt(0);      if ((i>=minval) && (i<=maxval)) return c;    }    rtl.raiseE('ERangeError');  },  rcSetCharAt: function(s,index,c){    // range check setCharAt    if ((typeof(s)!=='string') || (index<0) || (index>=s.length)) rtl.raiseE('ERangeError');    return rtl.setCharAt(s,index,c);  },  rcCharAt: function(s,index){    // range check charAt    if ((typeof(s)!=='string') || (index<0) || (index>=s.length)) rtl.raiseE('ERangeError');    return s.charAt(index);  },  rcArrR: function(arr,index){    // range check read array    if (Array.isArray(arr) && (typeof(index)==='number') && (index>=0) && (index<arr.length)){      if (arguments.length>2){        // arr,index1,index2,...        arr=arr[index];        for (var i=2; i<arguments.length; i++) arr=rtl.rcArrR(arr,arguments[i]);        return arr;      }      return arr[index];    }    rtl.raiseE('ERangeError');  },  rcArrW: function(arr,index,value){    // range check write array    // arr,index1,index2,...,value    for (var i=3; i<arguments.length; i++){      arr=rtl.rcArrR(arr,index);      index=arguments[i-1];      value=arguments[i];    }    if (Array.isArray(arr) && (typeof(index)==='number') && (index>=0) && (index<arr.length)){      return arr[index]=value;    }    rtl.raiseE('ERangeError');  },  length: function(arr){    return (arr == null) ? 0 : arr.length;  },  arraySetLength: function(arr,defaultvalue,newlength){    // multi dim: (arr,defaultvalue,dim1,dim2,...)    if (arr == null) arr = [];    var p = arguments;    function setLength(a,argNo){      var oldlen = a.length;      var newlen = p[argNo];      if (oldlen!==newlength){        a.length = newlength;        if (argNo === p.length-1){          if (rtl.isArray(defaultvalue)){            for (var i=oldlen; i<newlen; i++) a[i]=[]; // nested array          } else if (rtl.isObject(defaultvalue)) {            if (rtl.isTRecord(defaultvalue)){              for (var i=oldlen; i<newlen; i++) a[i]=defaultvalue.$new(); // e.g. record            } else {              for (var i=oldlen; i<newlen; i++) a[i]={}; // e.g. set            }          } else {            for (var i=oldlen; i<newlen; i++) a[i]=defaultvalue;          }        } else {          for (var i=oldlen; i<newlen; i++) a[i]=[]; // nested array        }      }      if (argNo < p.length-1){        // multi argNo        for (var i=0; i<newlen; i++) a[i]=setLength(a[i],argNo+1);      }      return a;    }    return setLength(arr,2);  },  arrayEq: function(a,b){    if (a===null) return b===null;    if (b===null) return false;    if (a.length!==b.length) return false;    for (var i=0; i<a.length; i++) if (a[i]!==b[i]) return false;    return true;  },  arrayClone: function(type,src,srcpos,endpos,dst,dstpos){    // type: 0 for references, "refset" for calling refSet(), a function for new type()    // src must not be null    // This function does not range check.    if(type === 'refSet') {      for (; srcpos<endpos; srcpos++) dst[dstpos++] = rtl.refSet(src[srcpos]); // ref set    } else if (rtl.isTRecord(type)){      for (; srcpos<endpos; srcpos++) dst[dstpos++] = type.$clone(src[srcpos]); // clone record    }  else {      for (; srcpos<endpos; srcpos++) dst[dstpos++] = src[srcpos]; // reference    };  },  arrayConcat: function(type){    // type: see rtl.arrayClone    var a = [];    var l = 0;    for (var i=1; i<arguments.length; i++){      var src = arguments[i];      if (src !== null) l+=src.length;    };    a.length = l;    l=0;    for (var i=1; i<arguments.length; i++){      var src = arguments[i];      if (src === null) continue;      rtl.arrayClone(type,src,0,src.length,a,l);      l+=src.length;    };    return a;  },  arrayConcatN: function(){    var a = null;    for (var i=1; i<arguments.length; i++){      var src = arguments[i];      if (src === null) continue;      if (a===null){        a=src; // Note: concat(a) does not clone      } else {        a=a.concat(src);      }    };    return a;  },  arrayCopy: function(type, srcarray, index, count){    // type: see rtl.arrayClone    // if count is missing, use srcarray.length    if (srcarray === null) return [];    if (index < 0) index = 0;    if (count === undefined) count=srcarray.length;    var end = index+count;    if (end>srcarray.length) end = srcarray.length;    if (index>=end) return [];    if (type===0){      return srcarray.slice(index,end);    } else {      var a = [];      a.length = end-index;      rtl.arrayClone(type,srcarray,index,end,a,0);      return a;    }  },  setCharAt: function(s,index,c){    return s.substr(0,index)+c+s.substr(index+1);  },  getResStr: function(mod,name){    var rs = mod.$resourcestrings[name];    return rs.current?rs.current:rs.org;  },  createSet: function(){    var s = {};    for (var i=0; i<arguments.length; i++){      if (arguments[i]!=null){        s[arguments[i]]=true;      } else {        var first=arguments[i+=1];        var last=arguments[i+=1];        for(var j=first; j<=last; j++) s[j]=true;      }    }    return s;  },  cloneSet: function(s){    var r = {};    for (var key in s) r[key]=true;    return r;  },  refSet: function(s){    Object.defineProperty(s, '$shared', {      enumerable: false,      configurable: true,      writable: true,      value: true    });    return s;  },  includeSet: function(s,enumvalue){    if (s.$shared) s = rtl.cloneSet(s);    s[enumvalue] = true;    return s;  },  excludeSet: function(s,enumvalue){    if (s.$shared) s = rtl.cloneSet(s);    delete s[enumvalue];    return s;  },  diffSet: function(s,t){    var r = {};    for (var key in s) if (!t[key]) r[key]=true;    return r;  },  unionSet: function(s,t){    var r = {};    for (var key in s) r[key]=true;    for (var key in t) r[key]=true;    return r;  },  intersectSet: function(s,t){    var r = {};    for (var key in s) if (t[key]) r[key]=true;    return r;  },  symDiffSet: function(s,t){    var r = {};    for (var key in s) if (!t[key]) r[key]=true;    for (var key in t) if (!s[key]) r[key]=true;    return r;  },  eqSet: function(s,t){    for (var key in s) if (!t[key]) return false;    for (var key in t) if (!s[key]) return false;    return true;  },  neSet: function(s,t){    return !rtl.eqSet(s,t);  },  leSet: function(s,t){    for (var key in s) if (!t[key]) return false;    return true;  },  geSet: function(s,t){    for (var key in t) if (!s[key]) return false;    return true;  },  strSetLength: function(s,newlen){    var oldlen = s.length;    if (oldlen > newlen){      return s.substring(0,newlen);    } else if (s.repeat){      // Note: repeat needs ECMAScript6!      return s+' '.repeat(newlen-oldlen);    } else {       while (oldlen<newlen){         s+=' ';         oldlen++;       };       return s;    }  },  spaceLeft: function(s,width){    var l=s.length;    if (l>=width) return s;    if (s.repeat){      // Note: repeat needs ECMAScript6!      return ' '.repeat(width-l) + s;    } else {      while (l<width){        s=' '+s;        l++;      };    };  },  floatToStr: function(d,w,p){    // input 1-3 arguments: double, width, precision    if (arguments.length>2){      return rtl.spaceLeft(d.toFixed(p),w);    } else {	  // exponent width	  var pad = "";	  var ad = Math.abs(d);	  if (ad<1.0e+10) {		pad='00';	  } else if (ad<1.0e+100) {		pad='0';      }  		  if (arguments.length<2) {	    w=9;		      } else if (w<9) {		w=9;      }		        var p = w-8;      var s=(d>0 ? " " : "" ) + d.toExponential(p);      s=s.replace(/e(.)/,'E$1'+pad);      return rtl.spaceLeft(s,w);    }  },  valEnum: function(s, enumType, setCodeFn){    s = s.toLowerCase();    for (var key in enumType){      if((typeof(key)==='string') && (key.toLowerCase()===s)){        setCodeFn(0);        return enumType[key];      }    }    setCodeFn(1);    return 0;  },  and: function(a,b){    var hi = 0x80000000;    var low = 0x7fffffff;    var h = (a / hi) & (b / hi);    var l = (a & low) & (b & low);    return h*hi + l;  },  or: function(a,b){    var hi = 0x80000000;    var low = 0x7fffffff;    var h = (a / hi) | (b / hi);    var l = (a & low) | (b & low);    return h*hi + l;  },  xor: function(a,b){    var hi = 0x80000000;    var low = 0x7fffffff;    var h = (a / hi) ^ (b / hi);    var l = (a & low) ^ (b & low);    return h*hi + l;  },  initRTTI: function(){    if (rtl.debug_rtti) rtl.debug('initRTTI');    // base types    rtl.tTypeInfo = { name: "tTypeInfo" };    function newBaseTI(name,kind,ancestor){      if (!ancestor) ancestor = rtl.tTypeInfo;      if (rtl.debug_rtti) rtl.debug('initRTTI.newBaseTI "'+name+'" '+kind+' ("'+ancestor.name+'")');      var t = Object.create(ancestor);      t.name = name;      t.kind = kind;      rtl[name] = t;      return t;    };    function newBaseInt(name,minvalue,maxvalue,ordtype){      var t = newBaseTI(name,1 /* tkInteger */,rtl.tTypeInfoInteger);      t.minvalue = minvalue;      t.maxvalue = maxvalue;      t.ordtype = ordtype;      return t;    };    newBaseTI("tTypeInfoInteger",1 /* tkInteger */);    newBaseInt("shortint",-0x80,0x7f,0);    newBaseInt("byte",0,0xff,1);    newBaseInt("smallint",-0x8000,0x7fff,2);    newBaseInt("word",0,0xffff,3);    newBaseInt("longint",-0x80000000,0x7fffffff,4);    newBaseInt("longword",0,0xffffffff,5);    newBaseInt("nativeint",-0x10000000000000,0xfffffffffffff,6);    newBaseInt("nativeuint",0,0xfffffffffffff,7);    newBaseTI("char",2 /* tkChar */);    newBaseTI("string",3 /* tkString */);    newBaseTI("tTypeInfoEnum",4 /* tkEnumeration */,rtl.tTypeInfoInteger);    newBaseTI("tTypeInfoSet",5 /* tkSet */);    newBaseTI("double",6 /* tkDouble */);    newBaseTI("boolean",7 /* tkBool */);    newBaseTI("tTypeInfoProcVar",8 /* tkProcVar */);    newBaseTI("tTypeInfoMethodVar",9 /* tkMethod */,rtl.tTypeInfoProcVar);    newBaseTI("tTypeInfoArray",10 /* tkArray */);    newBaseTI("tTypeInfoDynArray",11 /* tkDynArray */);    newBaseTI("tTypeInfoPointer",15 /* tkPointer */);    var t = newBaseTI("pointer",15 /* tkPointer */,rtl.tTypeInfoPointer);    t.reftype = null;    newBaseTI("jsvalue",16 /* tkJSValue */);    newBaseTI("tTypeInfoRefToProcVar",17 /* tkRefToProcVar */,rtl.tTypeInfoProcVar);    // member kinds    rtl.tTypeMember = {};    function newMember(name,kind){      var m = Object.create(rtl.tTypeMember);      m.name = name;      m.kind = kind;      rtl[name] = m;    };    newMember("tTypeMemberField",1); // tmkField    newMember("tTypeMemberMethod",2); // tmkMethod    newMember("tTypeMemberProperty",3); // tmkProperty    // base object for storing members: a simple object    rtl.tTypeMembers = {};    // tTypeInfoStruct - base object for tTypeInfoClass, tTypeInfoRecord, tTypeInfoInterface    var tis = newBaseTI("tTypeInfoStruct",0);    tis.$addMember = function(name,ancestor,options){      if (rtl.debug_rtti){        if (!rtl.hasString(name) || (name.charAt()==='$')) throw 'invalid member "'+name+'", this="'+this.name+'"';        if (!rtl.is(ancestor,rtl.tTypeMember)) throw 'invalid ancestor "'+ancestor+':'+ancestor.name+'", "'+this.name+'.'+name+'"';        if ((options!=undefined) && (typeof(options)!='object')) throw 'invalid options "'+options+'", "'+this.name+'.'+name+'"';      };      var t = Object.create(ancestor);      t.name = name;      this.members[name] = t;      this.names.push(name);      if (rtl.isObject(options)){        for (var key in options) if (options.hasOwnProperty(key)) t[key] = options[key];      };      return t;    };    tis.addField = function(name,type,options){      var t = this.$addMember(name,rtl.tTypeMemberField,options);      if (rtl.debug_rtti){        if (!rtl.is(type,rtl.tTypeInfo)) throw 'invalid type "'+type+'", "'+this.name+'.'+name+'"';      };      t.typeinfo = type;      this.fields.push(name);      return t;    };    tis.addFields = function(){      var i=0;      while(i<arguments.length){        var name = arguments[i++];        var type = arguments[i++];        if ((i<arguments.length) && (typeof(arguments[i])==='object')){          this.addField(name,type,arguments[i++]);        } else {          this.addField(name,type);        };      };    };    tis.addMethod = function(name,methodkind,params,result,options){      var t = this.$addMember(name,rtl.tTypeMemberMethod,options);      t.methodkind = methodkind;      t.procsig = rtl.newTIProcSig(params);      t.procsig.resulttype = result?result:null;      this.methods.push(name);      return t;    };    tis.addProperty = function(name,flags,result,getter,setter,options){      var t = this.$addMember(name,rtl.tTypeMemberProperty,options);      t.flags = flags;      t.typeinfo = result;      t.getter = getter;      t.setter = setter;      // Note: in options: params, stored, defaultvalue      if (rtl.isArray(t.params)) t.params = rtl.newTIParams(t.params);      this.properties.push(name);      if (!rtl.isString(t.stored)) t.stored = "";      return t;    };    tis.getField = function(index){      return this.members[this.fields[index]];    };    tis.getMethod = function(index){      return this.members[this.methods[index]];    };    tis.getProperty = function(index){      return this.members[this.properties[index]];    };    newBaseTI("tTypeInfoRecord",12 /* tkRecord */,rtl.tTypeInfoStruct);    newBaseTI("tTypeInfoClass",13 /* tkClass */,rtl.tTypeInfoStruct);    newBaseTI("tTypeInfoClassRef",14 /* tkClassRef */);    newBaseTI("tTypeInfoInterface",18 /* tkInterface */,rtl.tTypeInfoStruct);    newBaseTI("tTypeInfoHelper",19 /* tkHelper */,rtl.tTypeInfoStruct);  },  tSectionRTTI: {    $module: null,    $inherited: function(name,ancestor,o){      if (rtl.debug_rtti){        rtl.debug('tSectionRTTI.newTI "'+(this.$module?this.$module.$name:"(no module)")          +'"."'+name+'" ('+ancestor.name+') '+(o?'init':'forward'));      };      var t = this[name];      if (t){        if (!t.$forward) throw 'duplicate type "'+name+'"';        if (!ancestor.isPrototypeOf(t)) throw 'typeinfo ancestor mismatch "'+name+'" ancestor="'+ancestor.name+'" t.name="'+t.name+'"';      } else {        t = Object.create(ancestor);        t.name = name;        t.$module = this.$module;        this[name] = t;      }      if (o){        delete t.$forward;        for (var key in o) if (o.hasOwnProperty(key)) t[key]=o[key];      } else {        t.$forward = true;      }      return t;    },    $Scope: function(name,ancestor,o){      var t=this.$inherited(name,ancestor,o);      t.members = {};      t.names = [];      t.fields = [];      t.methods = [];      t.properties = [];      return t;    },    $TI: function(name,kind,o){ var t=this.$inherited(name,rtl.tTypeInfo,o); t.kind = kind; return t; },    $Int: function(name,o){ return this.$inherited(name,rtl.tTypeInfoInteger,o); },    $Enum: function(name,o){ return this.$inherited(name,rtl.tTypeInfoEnum,o); },    $Set: function(name,o){ return this.$inherited(name,rtl.tTypeInfoSet,o); },    $StaticArray: function(name,o){ return this.$inherited(name,rtl.tTypeInfoArray,o); },    $DynArray: function(name,o){ return this.$inherited(name,rtl.tTypeInfoDynArray,o); },    $ProcVar: function(name,o){ return this.$inherited(name,rtl.tTypeInfoProcVar,o); },    $RefToProcVar: function(name,o){ return this.$inherited(name,rtl.tTypeInfoRefToProcVar,o); },    $MethodVar: function(name,o){ return this.$inherited(name,rtl.tTypeInfoMethodVar,o); },    $Record: function(name,o){ return this.$Scope(name,rtl.tTypeInfoRecord,o); },    $Class: function(name,o){ return this.$Scope(name,rtl.tTypeInfoClass,o); },    $ClassRef: function(name,o){ return this.$inherited(name,rtl.tTypeInfoClassRef,o); },    $Pointer: function(name,o){ return this.$inherited(name,rtl.tTypeInfoPointer,o); },    $Interface: function(name,o){ return this.$Scope(name,rtl.tTypeInfoInterface,o); },    $Helper: function(name,o){ return this.$Scope(name,rtl.tTypeInfoHelper,o); }  },  newTIParam: function(param){    // param is an array, 0=name, 1=type, 2=optional flags    var t = {      name: param[0],      typeinfo: param[1],      flags: (rtl.isNumber(param[2]) ? param[2] : 0)    };    return t;  },  newTIParams: function(list){    // list: optional array of [paramname,typeinfo,optional flags]    var params = [];    if (rtl.isArray(list)){      for (var i=0; i<list.length; i++) params.push(rtl.newTIParam(list[i]));    };    return params;  },  newTIProcSig: function(params,result,flags){    var s = {      params: rtl.newTIParams(params),      resulttype: result,      flags: flags    };    return s;  }}
 |