const SpaccDotWeb = ((args) => { ////////////////////////////////////////////// let windowObject, documentObject; let Lib = {}; let isBuildingApp = false; let __scriptname; const platformIsNode = (typeof module === 'object' && typeof module.exports === 'object'); const platformIsBrowser = (typeof window !== 'undefined' && typeof window.document !== 'undefined'); const __filename__ = (typeof __filename !== 'undefined' ? __filename : ''); const JsdomOptions = { /*resources: "usable",*/ runScripts: /*"dangerously"*/"outside-only" }; if (platformIsNode) { Lib.fs = require('fs'); Lib.crypto = require('crypto'); Lib.childProcess = require('child_process'); Lib.jsdom = require('jsdom'); __scriptname = __filename__.split('/').slice(-1)[0]; windowObject = new Lib.jsdom.JSDOM('', JsdomOptions).window; }; if (platformIsBrowser) { windowObject = window; }; documentObject = windowObject.document; const SpaccDotWeb = ((args) => { ////////////////////////////////////////////// let SpaccDotWeb = {}; if (platformIsNode) { SpaccDotWeb.AppBuildStandalone = (opts) => { // TODO: build result of dom after JS, to make base page usable without JS isBuildingApp = true; opts ||= {}; opts.Page ||= 'index.html'; opts.Modules && !opts.Modules.includes('Main') && (opts.Modules = [...opts.Modules, 'Main']); Lib.fs.mkdirSync(`${__dirname}/Build/App-${opts.Page}`, { recursive: true }); let htmlIndex = Lib.fs.readFileSync(opts.Page, 'utf8'); windowObject = new Lib.jsdom.JSDOM(htmlIndex, JsdomOptions).window; documentObject = windowObject.document; DomSetup(opts.Modules); Lib.fs.writeFileSync(`${__dirname}/Build/App-${opts.Page}/Full.html`, `${documentObject.documentElement.outerHTML}`); isBuildingApp = false; }; SpaccDotWeb.LibBuild = () => { Lib.fs.mkdirSync(`${__dirname}/Build/Assets.tmp`, { recursive: true }); let uptodate = true; const compiledPath = `${__dirname}/Build/SpaccDotWeb.js`; const minifiedPath = `${__dirname}/Build/SpaccDotWeb.min.js`; const hashPath = `${__dirname}/Build/SpaccDotWeb.js.hash`; const hashOld = (Lib.fs.existsSync(hashPath) && Lib.fs.readFileSync(hashPath, 'utf8')); const hashNew = Lib.crypto.createHash('sha256').update(Lib.fs.readFileSync(__filename__, 'utf8')).digest('hex'); if (!Lib.fs.existsSync(compiledPath) || !Lib.fs.existsSync(minifiedPath) || !(hashOld === hashNew)) { uptodate = false; Lib.fs.writeFileSync(hashPath, hashNew); Lib.fs.writeFileSync(compiledPath, Lib.childProcess.execSync(`cat "${__filename__}" | npx babel -f "${__scriptname}"`)); Lib.fs.writeFileSync(minifiedPath, Lib.childProcess.execSync(`cat "${compiledPath}" | npx uglifyjs`)); }; uptodate && console.log('Library is up-to-date.'); return { compiledText: Lib.fs.readFileSync(compiledPath, 'utf8'), minified: Lib.fs.readFileSync(minifiedPath, 'utf8') }; }; }; SpaccDotWeb.AppInit = () => { try { DomSetup(); return SpaccDotWeb; } catch(err) { console.log(err) }; }; //SpaccDotWeb.Make = () => {}; SpaccDotWeb.Create = (tag, attrs) => { let elem = documentObject.createElement(tag); for (const key in attrs) { elem[key] = attrs[key]; }; return elem; }; SpaccDotWeb.Select = (query) => { let elem = documentObject.querySelector(query); elem && (elem.Insert = elem.appendChild); return elem; }; SpaccDotWeb.RequireScript = (src) => { if (platformIsBrowser) { SpaccDotWeb.Select('body').Insert(SpaccDotWeb.Create('script', { src: src })); //} else if (platformIsNode) { // require(src); } }; // TODO: make Meta element optional without breaking things const AppMetaGet = () => { const elem = SpaccDotWeb.Select('script[module="Meta"]'); if (elem) { if (['application/json', 'text/json'].includes(elem.getAttribute('type'))) { return JSON.parse(elem.innerHTML); } else { return eval(elem.innerHTML); }; }; }; const DomMakeBase = (Modules) => { const meta = AppMetaGet(); if (meta) { const htmlFrags = { Title: (meta.Name ? `