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 ? `${meta.Name}` : ''), Description: (meta.Description ? `` : ''), Uri: (meta.Uri ? `` : '') }; let [scriptsCode, elementsHtml, scriptsHtml] = ['', '', '']; if (isBuildingApp) { scriptsHtml += ``; scriptsHtml += ``; scriptsHtml += `${SpaccDotWeb.LibBuild().minified}`; for (const elem of documentObject.querySelectorAll('script[module]')) { //if (elem.module === 'Meta' && !['application/json', 'text/json'].includes(elem.type)) { // elem.innerHTML = `(${elem.innerHTML})`; //}; if (elem.getAttribute('module') !== 'SpaccDotWeb' && (!Modules || (Modules && Modules.includes(elem.getAttribute('module'))))) { if (elem.getAttribute('src')) { scriptsHtml += ``; // TODO somehow include this in prerendered DOM? } else { const tmpHash = Lib.crypto.createHash('sha256').update(elem.innerHTML).digest('hex'); const tmpPath = `${__dirname}/Build/Assets.tmp/${tmpHash}.js`; Lib.fs.writeFileSync(tmpPath, elem.innerHTML); const scriptCode = Lib.childProcess.execSync(`cat "${tmpPath}" | npx babel -f "${tmpHash}.js" | npx uglifyjs`); scriptsCode += scriptCode; scriptsHtml += `${Lib.childProcess.execSync(`cat "${tmpPath}" | npx babel -f "${tmpHash}.js" | npx uglifyjs`)}`; }; }; elem.remove(); }; // select and include all remaining actual elements, except parent meta-elements for (const elem of documentObject.querySelectorAll('* > * > *')) { elementsHtml += elem.outerHTML; } }; return { head: ` ${htmlFrags.Title} ${htmlFrags.Description} ${htmlFrags.Uri} `, body: `
${elementsHtml}${scriptsHtml}`, scriptsCode: scriptsCode, }; }; }; const DomSetup = (Modules) => { const doctypeNew = documentObject.implementation.createHTMLDocument().doctype; windowObject.document.doctype ? documentObject.replaceChild(doctypeNew, documentObject.doctype) : documentObject.insertBefore(doctypeNew, documentObject.childNodes[0]); const domBase = DomMakeBase(Modules); if (domBase) { documentObject.write(domBase.head + domBase.body); documentObject.head.innerHTML = domBase.head; documentObject.body.innerHTML = domBase.body; if (isBuildingApp) { windowObject.eval(Lib.fs.readFileSync(__filename__, 'utf-8') + domBase.scriptsCode); }; }; }; return SpaccDotWeb; })(); ///////////////////////////////////////////////////////////////////////// platformIsBrowser && (window.SpaccDotWeb = SpaccDotWeb); platformIsNode && process.argv.length >= 2 && console.log(eval(process.argv.slice(-1)[0])); return SpaccDotWeb; })(); /////////////////////////////////////////////////////////////////////////