| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | /********************************************************************** | 
					
						
							|  |  |  | *  | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | **********************************************************************/ | 
					
						
							|  |  |  | ((typeof define)[0]=='u'?function(f){module.exports=f(require)}:define) | 
					
						
							|  |  |  | (function(require){ var module={} // make module AMD/node compatible...
 | 
					
						
							|  |  |  | /*********************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // XXX
 | 
					
						
							|  |  |  | //var object = require('lib/object')
 | 
					
						
							|  |  |  | var object = require('ig-object') | 
					
						
							|  |  |  | var types = require('ig-types') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-05 18:21:18 +03:00
										 |  |  | var pwpath = require('./path') | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | var parser = require('./parser') | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | var filters = require('./filters/base') | 
					
						
							|  |  |  | var markdown = require('./filters/markdown') | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //---------------------------------------------------------------------
 | 
					
						
							|  |  |  | // Page...
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var relProxy =  | 
					
						
							|  |  |  | function(name){ | 
					
						
							|  |  |  | 	var func = function(path='.', ...args){ | 
					
						
							|  |  |  | 		return this.store[name]( | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 			pwpath.relative(this.path, path),  | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 			...args) }  | 
					
						
							|  |  |  | 	Object.defineProperty(func, 'name', {value: name}) | 
					
						
							|  |  |  | 	return func }  | 
					
						
							|  |  |  | var relMatchProxy =  | 
					
						
							|  |  |  | function(name){ | 
					
						
							|  |  |  | 	var func = function(path='.', strict=this.strict){ | 
					
						
							|  |  |  | 		if(path === true || path === false){ | 
					
						
							|  |  |  | 			strict = path | 
					
						
							|  |  |  | 			path = '.' } | 
					
						
							|  |  |  | 		return this.store[name]( | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 			pwpath.relative(this.path, path),  | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 			strict) }  | 
					
						
							|  |  |  | 	Object.defineProperty(func, 'name', {value: name}) | 
					
						
							|  |  |  | 	return func } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | // XXX PATH_VARS
 | 
					
						
							|  |  |  | // XXX FUNC need to handle functions in store...
 | 
					
						
							|  |  |  | var BasePage = | 
					
						
							|  |  |  | module.BasePage =  | 
					
						
							|  |  |  | object.Constructor('BasePage', { | 
					
						
							|  |  |  | 	// root page used to clone new instances via the .clone(..) method...
 | 
					
						
							|  |  |  | 	//root: undefined,
 | 
					
						
							| 
									
										
										
										
											2022-08-11 23:56:43 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// a base page to be used as a base for cloning if root is of a 
 | 
					
						
							|  |  |  | 	// different "class"...
 | 
					
						
							| 
									
										
										
										
											2022-08-12 01:38:32 +03:00
										 |  |  | 	//__clone_proto__: undefined,
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | 	// NOTE: this can be inherited...
 | 
					
						
							|  |  |  | 	//store: undefined,
 | 
					
						
							|  |  |  | 	//__store: undefined,
 | 
					
						
							|  |  |  | 	get store(){ | 
					
						
							|  |  |  | 		return this.__store  | 
					
						
							|  |  |  | 			?? (this.root ?? {}).__store }, | 
					
						
							|  |  |  | 	set store(value){ | 
					
						
							|  |  |  | 		this.__store = value }, | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	// Path variables...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX PATH_VARS should these be here???
 | 
					
						
							|  |  |  | 	// 		other places path variables can be resolved:
 | 
					
						
							|  |  |  | 	// 			- navigation (below)
 | 
					
						
							|  |  |  | 	// 			- macro expansion...
 | 
					
						
							|  |  |  | 	// XXX EXPERIMENTAL...
 | 
					
						
							|  |  |  | 	path_vars: { | 
					
						
							|  |  |  | 		NOW: function(){ | 
					
						
							|  |  |  | 			return Date.now() }, | 
					
						
							|  |  |  | 		PATH: function(){ | 
					
						
							|  |  |  | 			return this.path }, | 
					
						
							|  |  |  | 		NAME: function(){ | 
					
						
							|  |  |  | 			return this.name }, | 
					
						
							|  |  |  | 		DIR: function(){ | 
					
						
							|  |  |  | 			return this.dir }, | 
					
						
							|  |  |  | 		//TITLE: function(){
 | 
					
						
							|  |  |  | 		//	return this.title },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/*/ XXX this needs: | 
					
						
							|  |  |  | 		// 		- macro context...
 | 
					
						
							|  |  |  | 		// 		- sort order...
 | 
					
						
							|  |  |  | 		INDEX: function(context){ | 
					
						
							|  |  |  | 			return context.index }, | 
					
						
							|  |  |  | 		//*/
 | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	resolvePathVars: function(path, context={}){ | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		return Object.entries(this.path_vars) | 
					
						
							|  |  |  | 			.reduce(function(res, [key, func]){ | 
					
						
							|  |  |  | 				return res | 
					
						
							|  |  |  | 					.replace( | 
					
						
							|  |  |  | 						new RegExp('(\\${'+key+'}|\\$'+key+')', 'g'),  | 
					
						
							|  |  |  | 						func.call(that, context)) | 
					
						
							|  |  |  | 			}, path) }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// page location...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: path variables are resolved relative to the page BEFORE 
 | 
					
						
							|  |  |  | 	// 		navigation...
 | 
					
						
							|  |  |  | 	// NOTE: the actual work is done by the .onNavigate(..) method...
 | 
					
						
							|  |  |  | 	__location: undefined, | 
					
						
							|  |  |  | 	get location(){ | 
					
						
							|  |  |  | 		return this.__location ?? '/' }, | 
					
						
							|  |  |  | 	set location(path){ | 
					
						
							|  |  |  | 		// trigger the event...
 | 
					
						
							|  |  |  | 		this.onNavigate(path) }, | 
					
						
							|  |  |  | 	// referrer -- a previous page location...
 | 
					
						
							|  |  |  | 	referrer: undefined, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// events...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX revise naming...
 | 
					
						
							|  |  |  | 	// XXX should this be able to prevent navigation???
 | 
					
						
							|  |  |  | 	onBeforeNavigate: types.event.Event('beforeNavigate'), | 
					
						
							|  |  |  | 	onNavigate: types.event.Event('navigate', | 
					
						
							| 
									
										
										
										
											2022-09-01 00:04:13 +03:00
										 |  |  | 		function(handle, location){ | 
					
						
							|  |  |  | 			this.onBeforeNavigate(location) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 			this.referrer = this.location | 
					
						
							| 
									
										
										
										
											2022-09-01 00:04:13 +03:00
										 |  |  | 			var {path, args} = pwpath.splitArgs(location) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 			var cur = this.__location =  | 
					
						
							|  |  |  | 				this.resolvePathVars( | 
					
						
							| 
									
										
										
										
											2022-09-01 00:04:13 +03:00
										 |  |  | 					// NOTE: this is done instead of simply assigning 
 | 
					
						
							|  |  |  | 					// 		location as-is to normalize the paths and 
 | 
					
						
							|  |  |  | 					// 		arguments...
 | 
					
						
							|  |  |  | 					pwpath.joinArgs( | 
					
						
							|  |  |  | 						pwpath.relative( | 
					
						
							|  |  |  | 							this.path,  | 
					
						
							|  |  |  | 							path), | 
					
						
							|  |  |  | 						pwpath.obj2args(args))) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 			// trigger handlers...
 | 
					
						
							|  |  |  | 			handle() }), | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// .path is a proxy to .location
 | 
					
						
							|  |  |  | 	// XXX do we need this???
 | 
					
						
							|  |  |  | 	get path(){ | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		return pwpath.splitArgs(this.location).path }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	set path(value){ | 
					
						
							|  |  |  | 		this.location = value }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 	get args(){ | 
					
						
							|  |  |  | 		return pwpath.splitArgs(this.location).args }, | 
					
						
							|  |  |  | 	set args(args){ | 
					
						
							|  |  |  | 		this.location = this.path +':'+ pwpath.obj2args(args) }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-02 14:58:45 +03:00
										 |  |  | 	// NOTE: these are mostly here as helpers to be accessed via page 
 | 
					
						
							|  |  |  | 	// 		actions...
 | 
					
						
							|  |  |  | 	// XXX should these be here or in Page???
 | 
					
						
							|  |  |  | 	// XXX should this call .match(..) or .resolve(..)???
 | 
					
						
							|  |  |  | 	get resolved(){ | 
					
						
							|  |  |  | 		return this.resolve() }, | 
					
						
							|  |  |  | 	get rootpath(){ | 
					
						
							|  |  |  | 		return this.root ?  | 
					
						
							|  |  |  | 			this.root.path  | 
					
						
							|  |  |  | 			: this.path }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// XXX should these be writable???
 | 
					
						
							|  |  |  | 	get name(){ | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		return pwpath.basename(this.path) }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	//set name(value){ },
 | 
					
						
							|  |  |  | 	get dir(){ | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		return pwpath.dirname(this.path) }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	//set dir(value){ },
 | 
					
						
							|  |  |  | 	get isPattern(){ | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		return this.path.includes('*') }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	// store interface...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX we are only doing modifiers here...
 | 
					
						
							|  |  |  | 	// 		...these ar mainly used to disable writing in .ro(..)
 | 
					
						
							|  |  |  | 	__update__: function(data){ | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		return this.store.update(this.path, data) }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	__delete__: function(path='.'){ | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		return this.store.delete(pwpath.relative(this.path, path)) }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// page data...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	strict: undefined, | 
					
						
							| 
									
										
										
										
											2022-08-30 10:58:04 +03:00
										 |  |  | 	energetic: undefined, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	get data(){ return (async function(){ | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		// NOTE: we need to make sure each page gets the chance to handle 
 | 
					
						
							| 
									
										
										
										
											2022-08-29 18:22:35 +03:00
										 |  |  | 		// 		its context (i.e. bind action to page)....
 | 
					
						
							|  |  |  | 		if(this.isPattern  | 
					
						
							|  |  |  | 				// XXX ENERGETIC...
 | 
					
						
							| 
									
										
										
										
											2022-08-30 10:58:04 +03:00
										 |  |  | 				&& !(this.energetic  | 
					
						
							|  |  |  | 					|| await this.store.isEnergetic(this.path))){ | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 			return this | 
					
						
							|  |  |  | 				.map(function(page){ | 
					
						
							|  |  |  | 					return page.data }) } | 
					
						
							|  |  |  | 		// single page...
 | 
					
						
							| 
									
										
										
										
											2022-08-30 10:58:04 +03:00
										 |  |  | 		// XXX ENERGETIC...
 | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		var res = await this.store.get(this.path, !!this.strict, !!this.energetic) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		return typeof(res) == 'function' ? | 
					
						
							|  |  |  | 			res.bind(this) | 
					
						
							|  |  |  | 			: res }).call(this) }, | 
					
						
							|  |  |  | 	set data(value){ | 
					
						
							|  |  |  | 		this.__update__(value) }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// metadata...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: in the general case this is the same as .data but in also allows
 | 
					
						
							|  |  |  | 	// 		storing of data (metadata) for pattern paths...
 | 
					
						
							|  |  |  | 	get metadata(){ | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		return this.store.metadata(this.path) }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	set metadata(value){ | 
					
						
							|  |  |  | 		this.__update__(value) }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 	get type(){ return async function(){ | 
					
						
							|  |  |  | 		return this.store.isStore(this.path) ? | 
					
						
							|  |  |  | 				'store' | 
					
						
							|  |  |  | 			: typeof(await this.data) == 'function' ?  | 
					
						
							|  |  |  | 				'action'  | 
					
						
							|  |  |  | 			: 'page' }.bind(this)() }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	// number of matching pages...
 | 
					
						
							|  |  |  | 	// NOTE: this can be both sync and async...
 | 
					
						
							|  |  |  | 	get length(){ | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		var p = this.resolve(this.path) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		return p instanceof Array ? | 
					
						
							|  |  |  | 				p.length | 
					
						
							|  |  |  | 			: p instanceof Promise ? | 
					
						
							|  |  |  | 				p.then(function(res){ | 
					
						
							|  |  |  | 					return res instanceof Array ? | 
					
						
							|  |  |  | 						res.length | 
					
						
							|  |  |  | 						: 1 }) | 
					
						
							|  |  |  | 			: 1 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// relative proxies to store...
 | 
					
						
							|  |  |  | 	exists: relProxy('exists'),  | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 	//* XXX MATCH
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	match: relMatchProxy('match'),  | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 	/*/ | 
					
						
							|  |  |  | 	match: async function(path='.', strict=false){ | 
					
						
							|  |  |  | 		if(path === true || path === false){ | 
					
						
							|  |  |  | 			strict = path | 
					
						
							|  |  |  | 			path = '.' } | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		path = pwpath.relative(this.path, path) | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 		var res = await this.store.match(path, strict)  | 
					
						
							|  |  |  | 		return res.length == 0 ? | 
					
						
							|  |  |  | 			// XXX are we going outside of match semantics here???
 | 
					
						
							|  |  |  | 			this.store.find(path)  | 
					
						
							|  |  |  | 			: res }, | 
					
						
							|  |  |  | 	//*/
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	resolve: relMatchProxy('resolve'), | 
					
						
							|  |  |  | 	delete: function(path='.'){ | 
					
						
							| 
									
										
										
										
											2022-08-15 17:32:07 +03:00
										 |  |  | 		this.__delete__(path)  | 
					
						
							|  |  |  | 		return this }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// XXX should these be implemented here or proxy to .store???
 | 
					
						
							|  |  |  | 	copy: async function(to){ | 
					
						
							|  |  |  | 		this.get(to).data = await this.data | 
					
						
							|  |  |  | 		this.path = to | 
					
						
							|  |  |  | 		return this }, | 
					
						
							|  |  |  | 	move: async function(to){ | 
					
						
							|  |  |  | 		var from = this.path | 
					
						
							|  |  |  | 		await this.copy(to) | 
					
						
							|  |  |  | 		this.delete(from) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		return this }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Find current path (non-strict)
 | 
					
						
							|  |  |  | 	// 	.find()
 | 
					
						
							|  |  |  | 	// 	.find(false)
 | 
					
						
							|  |  |  | 	// 	.find('.')
 | 
					
						
							|  |  |  | 	// 	.find('.', false)
 | 
					
						
							|  |  |  | 	// 		-> path
 | 
					
						
							|  |  |  | 	// 		-> undefined
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Find current path in strict/non-strict mode...
 | 
					
						
							|  |  |  | 	// 	.find(true)
 | 
					
						
							|  |  |  | 	// 	.find(false)
 | 
					
						
							|  |  |  | 	// 		-> path
 | 
					
						
							|  |  |  | 	// 		-> undefined
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Find path relative to current page (strict/non-strict)
 | 
					
						
							|  |  |  | 	// 	.find(<path>[, <strict>])
 | 
					
						
							|  |  |  | 	// 		-> path
 | 
					
						
							|  |  |  | 	// 		-> undefined
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	find: function(path='.', strict=false){ | 
					
						
							|  |  |  | 		if(path === true || path === false){ | 
					
						
							|  |  |  | 			strict = path | 
					
						
							|  |  |  | 			path = '.' } | 
					
						
							|  |  |  | 		return this.store.find( | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 			pwpath.relative(this.path, path), strict) }, | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	.get(<path>[, <data>])
 | 
					
						
							|  |  |  | 	// 	.get(<path>, <strict>[, <data>])
 | 
					
						
							|  |  |  | 	// 		-> <page>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	get: function(path, strict, data={}){ | 
					
						
							|  |  |  | 		if(strict instanceof Object){ | 
					
						
							|  |  |  | 			data = strict | 
					
						
							|  |  |  | 			strict = undefined } | 
					
						
							|  |  |  | 		return this.clone({ | 
					
						
							|  |  |  | 				location: path,  | 
					
						
							|  |  |  | 				...data, | 
					
						
							|  |  |  | 				referrer: data.referrer  | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 					//?? this.path,
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 					?? this.referrer, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				strict, | 
					
						
							|  |  |  | 			}) }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// XXX should this be an iterator???
 | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 	each: async function*(path){ | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		// NOTE: we are trying to avoid resolving non-pattern paths unless 
 | 
					
						
							|  |  |  | 		// 		we really have to...
 | 
					
						
							|  |  |  | 		path = path ? | 
					
						
							|  |  |  | 			pwpath.relative(this.path, path) | 
					
						
							|  |  |  | 			: this.path | 
					
						
							| 
									
										
										
										
											2022-08-29 18:22:35 +03:00
										 |  |  | 		var paths = path.includes('*')  | 
					
						
							|  |  |  | 				// XXX ENERGETIC...
 | 
					
						
							| 
									
										
										
										
											2022-08-30 10:58:04 +03:00
										 |  |  | 				&& !(this.energetic | 
					
						
							|  |  |  | 					|| await this.store.isEnergetic(path)) ? | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 			this.resolve(path) | 
					
						
							|  |  |  | 			: path | 
					
						
							|  |  |  | 		paths = paths instanceof Array ?  | 
					
						
							|  |  |  | 				paths  | 
					
						
							|  |  |  | 			: paths instanceof Promise ? | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 				await paths | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 			: [paths] | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 		/*/ XXX MATCH | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 		paths = paths.length == 0 ? | 
					
						
							|  |  |  | 			[await this.find(path)] | 
					
						
							|  |  |  | 			: paths | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 		//*/
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 		for(var path of paths){ | 
					
						
							| 
									
										
										
										
											2022-08-11 10:39:55 +03:00
										 |  |  | 			yield this.get('/'+ path) } }, | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 	[Symbol.asyncIterator]: async function*(){ | 
					
						
							|  |  |  | 		yield* this.each() }, | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	map: async function(func){ | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		return this.each().map(func) }, | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 	filter: async function(func){ | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		return this.each().filter(func) }, | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 	reduce: async function(func, dfl){ | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		return this.each().reduce(func, dfl) }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// sorting...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX should this be page-level (current) store level???
 | 
					
						
							|  |  |  | 	// XXX when this is async, should this return a promise????
 | 
					
						
							|  |  |  | 	sort: async function(cmp){ | 
					
						
							|  |  |  | 		// not sorting single pages...
 | 
					
						
							|  |  |  | 		if(this.length <= 1){ | 
					
						
							|  |  |  | 			return this } | 
					
						
							|  |  |  | 		// sort...
 | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 		this.metadata =  | 
					
						
							|  |  |  | 			{ order: await this.each() | 
					
						
							|  |  |  | 				.sort(...arguments) | 
					
						
							|  |  |  | 				.map(function(p){ | 
					
						
							|  |  |  | 					return p.path }) } | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		return this }, | 
					
						
							|  |  |  | 	reverse: async function(){ | 
					
						
							|  |  |  | 		// not sorting single pages...
 | 
					
						
							|  |  |  | 		if(this.length <= 1){ | 
					
						
							|  |  |  | 			return this } | 
					
						
							|  |  |  | 		this.metadata = { order: (await this.match()).reverse() } | 
					
						
							|  |  |  | 		return this }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Clone a page optionally asigning data into it...
 | 
					
						
							|  |  |  | 	// 	.clone()
 | 
					
						
							|  |  |  | 	// 	.clone({ .. }[, <clone-history>])
 | 
					
						
							|  |  |  | 	// 		-> <page>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Fully clone a page optionally asigning data into it...
 | 
					
						
							|  |  |  | 	// 	.clone(true[, <clone-history>])
 | 
					
						
							|  |  |  | 	// 	.clone(true, { .. }[, <clone-history>])
 | 
					
						
							|  |  |  | 	// 		-> <page>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// Normal cloning will inherit all the "clones" from the original 
 | 
					
						
							|  |  |  | 	// page overloading .location and .referrer
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: <clone-history> by default is false unless fully cloning
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	clone: function(data={}, history=false){ | 
					
						
							|  |  |  | 		var [data, ...args] = [...arguments] | 
					
						
							|  |  |  | 		var full = data === true | 
					
						
							|  |  |  | 		history =  | 
					
						
							|  |  |  | 			typeof(args[args.length-1]) == 'boolean' ?  | 
					
						
							|  |  |  | 				args.pop()  | 
					
						
							|  |  |  | 				: full | 
					
						
							|  |  |  | 		data = full ?  | 
					
						
							|  |  |  | 			args[0] ?? {}  | 
					
						
							|  |  |  | 			: data | 
					
						
							| 
									
										
										
										
											2022-08-12 01:38:32 +03:00
										 |  |  | 		var src = this.__clone_proto__  | 
					
						
							|  |  |  | 			?? (this.root || {}).__clone_proto__ | 
					
						
							|  |  |  | 			?? this.root | 
					
						
							|  |  |  | 			?? this | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		return Object.assign( | 
					
						
							|  |  |  | 			full ? | 
					
						
							|  |  |  | 				// full copy...
 | 
					
						
							| 
									
										
										
										
											2022-08-12 01:38:32 +03:00
										 |  |  | 				// XXX src or this???
 | 
					
						
							|  |  |  | 				//this.constructor(this.path, this.referrer, this.store)
 | 
					
						
							|  |  |  | 				src.constructor(this.path, this.referrer, this.store) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				// NOTE: this will restrict all the clones to the first 
 | 
					
						
							|  |  |  | 				// 		generation maintaining the original (.root) page as 
 | 
					
						
							|  |  |  | 				// 		the common root...
 | 
					
						
							|  |  |  | 				// 		this will make all the non-shadowed attrs set on the
 | 
					
						
							|  |  |  | 				// 		root visible to all sub-pages.
 | 
					
						
							| 
									
										
										
										
											2022-08-12 01:38:32 +03:00
										 |  |  | 				: Object.create(src), | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 			// XXX
 | 
					
						
							|  |  |  | 			//{...this},
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 			{ | 
					
						
							|  |  |  | 				root: this.root ?? this, | 
					
						
							|  |  |  | 				location: this.location,  | 
					
						
							|  |  |  | 				referrer: this.referrer, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			data) }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Create a read-only page...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: all pages that are created via a read-only page are also 
 | 
					
						
							|  |  |  | 	// 		read-only.
 | 
					
						
							|  |  |  | 	// XXX EXPERIMENTAL...
 | 
					
						
							|  |  |  | 	ro: function(data={}){ | 
					
						
							|  |  |  | 		return Object.assign({ | 
					
						
							|  |  |  | 			__proto__: this, | 
					
						
							|  |  |  | 			__update__: function(){ return this }, | 
					
						
							|  |  |  | 			__delete__: function(){ return this }, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		data) }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Create a virtual page at current path...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// Virtual pages do not affect store data in any way but behave like 
 | 
					
						
							|  |  |  | 	// normal pages.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: .get(..) / .clone(..) will return normal non-virtual pages
 | 
					
						
							|  |  |  | 	// 		unless the target path is the same as the virtual page .path...
 | 
					
						
							|  |  |  | 	// NOTE: changing .path/.location is not supported.
 | 
					
						
							|  |  |  | 	// XXX EXPERIMENTAL...
 | 
					
						
							|  |  |  | 	virtual: function(data={}){ | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		return { | 
					
						
							|  |  |  | 			__proto__: this, | 
					
						
							|  |  |  | 			// make the location read-only...
 | 
					
						
							|  |  |  | 			get location(){ | 
					
						
							|  |  |  | 				// NOTE: since we are not providing this as a basis for 
 | 
					
						
							|  |  |  | 				// 		inheritance we do not need to properly access 
 | 
					
						
							|  |  |  | 				// 		the parent prop...
 | 
					
						
							|  |  |  | 				// 		...otherwise use:
 | 
					
						
							|  |  |  | 				// 			object.parentProperty(..)
 | 
					
						
							|  |  |  | 				return this.__proto__.location }, | 
					
						
							|  |  |  | 			__update__: function(data){  | 
					
						
							|  |  |  | 				Object.assign(this.data, data) | 
					
						
							|  |  |  | 				return this }, | 
					
						
							|  |  |  | 			__delete__: function(){ return this }, | 
					
						
							|  |  |  | 			// NOTE: we need to proxy .clone(..) back to parent so as to 
 | 
					
						
							|  |  |  | 			// 		avoid overloading .data in the children too...
 | 
					
						
							|  |  |  | 			// NOTE: we are also keeping all first level queries resolving 
 | 
					
						
							|  |  |  | 			// 		to current path also virtual...
 | 
					
						
							|  |  |  | 			clone: function(...args){ | 
					
						
							|  |  |  | 				var res = that.clone(...args)  | 
					
						
							|  |  |  | 				return res.path == this.path ? | 
					
						
							|  |  |  | 					that.virtual(this.data) | 
					
						
							|  |  |  | 					: res }, | 
					
						
							|  |  |  | 			data: Object.assign( | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					ctime: Date.now(), | 
					
						
							|  |  |  | 					mtime: Date.now(), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				data), | 
					
						
							|  |  |  | 		} }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// XXX should this be update or assign???
 | 
					
						
							|  |  |  | 	// XXX how should this work on multiple pages...
 | 
					
						
							|  |  |  | 	// 		...right now this will write what-ever is given, even if it
 | 
					
						
							|  |  |  | 	// 		will never be explicitly be accessible...
 | 
					
						
							|  |  |  | 	// XXX sync/async???
 | 
					
						
							|  |  |  | 	update: function(...data){ | 
					
						
							|  |  |  | 		return Object.assign(this, ...data) }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | 	// XXX should this take an options/dict argument????
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	__init__: function(path, referrer, store){ | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | 		if(referrer && typeof(referrer) != 'string'){ | 
					
						
							|  |  |  | 			store = referrer | 
					
						
							|  |  |  | 			referrer = undefined } | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		// NOTE: this will allow inheriting .store from the prototype
 | 
					
						
							|  |  |  | 		if(store){ | 
					
						
							|  |  |  | 			this.store = store } | 
					
						
							|  |  |  | 		this.location = path | 
					
						
							|  |  |  | 		this.referrer = referrer }, | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // pepper in event functionality...
 | 
					
						
							|  |  |  | types.event.EventMixin(BasePage.prototype) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | //---------------------------------------------------------------------
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | // XXX should these be something more generic like Object.assign(..) ???
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | // XXX do we need anything else like .doc, attrs???
 | 
					
						
							|  |  |  | var Macro = | 
					
						
							|  |  |  | module.Macro =  | 
					
						
							|  |  |  | function(spec, func){ | 
					
						
							|  |  |  | 	var args = [...arguments] | 
					
						
							|  |  |  | 	// function...
 | 
					
						
							|  |  |  | 	func = args.pop() | 
					
						
							|  |  |  | 	// arg sepc...
 | 
					
						
							|  |  |  | 	;(args.length > 0 && args[args.length-1] instanceof Array) | 
					
						
							|  |  |  | 		&& (func.arg_spec = args.pop()) | 
					
						
							|  |  |  | 	// XXX do we need anything else like .doc, attrs???
 | 
					
						
							|  |  |  | 	return func } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // XXX PATH_VARS need to handle path variables...
 | 
					
						
							|  |  |  | // XXX filters (and macros?) should be features for simpler plugin handlng (???)
 | 
					
						
							|  |  |  | // XXX STUB filters...
 | 
					
						
							| 
									
										
										
										
											2022-08-04 19:47:08 +03:00
										 |  |  | // XXX rename to pWikiPage???
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | var Page = | 
					
						
							|  |  |  | module.Page =  | 
					
						
							|  |  |  | object.Constructor('Page', BasePage, { | 
					
						
							|  |  |  | 	__parser__: parser.parser, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-27 16:30:23 +03:00
										 |  |  | 	NESTING_DEPTH_LIMIT: 20, | 
					
						
							| 
									
										
										
										
											2022-08-27 16:17:25 +03:00
										 |  |  | 	NESTING_RECURSION_TEST_THRESHOLD: 50, | 
					
						
							| 
									
										
										
										
											2022-08-17 11:46:50 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	// Filter that will isolate the page/include/.. from parent filters...
 | 
					
						
							|  |  |  | 	ISOLATED_FILTERS: 'isolated', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// list of macros that will get raw text of their content...
 | 
					
						
							|  |  |  | 	QUOTING_MACROS: ['quote'], | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// templates used to render a page via .text
 | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 	PAGE_TEMPLATE: '_view', | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// NOTE: comment this out to make the system fail when nothing is 
 | 
					
						
							|  |  |  | 	// 		resolved, not even the System/NotFound page...
 | 
					
						
							|  |  |  | 	// NOTE: we can't use any of the page actions here (like @source(./path)) 
 | 
					
						
							|  |  |  | 	// 		as if we reach this it's likely all the bootstrap is either also
 | 
					
						
							|  |  |  | 	// 		not present or broken.
 | 
					
						
							|  |  |  | 	// NOTE: to force the system to fail set this to undefined.
 | 
					
						
							| 
									
										
										
										
											2022-08-05 02:47:31 +03:00
										 |  |  | 	NOT_FOUND_ERROR: 'NotFoundError', | 
					
						
							|  |  |  | 	RECURSION_ERROR: 'RecursionError', | 
					
						
							| 
									
										
										
										
											2022-08-18 01:23:49 +03:00
										 |  |  | 	NOT_FOUND_TEMPLATE_ERROR: 'NotFoundTemplateError', | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 	QUOTE_ACTION_PAGE: 'QuoteActionPage', | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-21 18:10:39 +03:00
										 |  |  | 	// Format:
 | 
					
						
							|  |  |  | 	// 	{
 | 
					
						
							|  |  |  | 	// 		<path>: Set([<path>, ...]),
 | 
					
						
							|  |  |  | 	// 	}
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-22 13:24:26 +03:00
										 |  |  | 	// NOTE: this is stored in .root...
 | 
					
						
							| 
									
										
										
										
											2022-08-22 12:43:49 +03:00
										 |  |  | 	//__dependencies: undefined,
 | 
					
						
							|  |  |  | 	get dependencies(){ | 
					
						
							|  |  |  | 		return (this.root ?? this).__dependencies ?? {} }, | 
					
						
							|  |  |  | 	set dependencies(value){ | 
					
						
							|  |  |  | 		((this.root ?? this).__dependencies) = value }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-21 18:10:39 +03:00
										 |  |  | 	// NOTE: for this to populate .text must be done at least once...
 | 
					
						
							|  |  |  | 	get depends(){ | 
					
						
							|  |  |  | 		return (this.dependencies ?? {})[this.path] }, | 
					
						
							|  |  |  | 	set depends(value){ | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 		if(value == null){ | 
					
						
							|  |  |  | 			delete (this.dependencies ?? {})[this.path] | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			;(this.dependencies = this.dependencies ?? {})[this.path] = value } }, | 
					
						
							| 
									
										
										
										
											2022-08-21 18:10:39 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-16 09:35:07 +03:00
										 |  |  | 	// The page that started the current render...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// This is set by .text and maintained by .clone(..).
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: for manual rendering (.parse(..), ... etc.) this has to be 
 | 
					
						
							|  |  |  | 	// 		setup manually.
 | 
					
						
							|  |  |  | 	render_root: undefined, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-02 14:58:45 +03:00
										 |  |  | 	get renderer(){ | 
					
						
							|  |  |  | 		return (this.render_root || {}).path }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	<filter>(<source>)
 | 
					
						
							|  |  |  | 	// 		-> <result>
 | 
					
						
							|  |  |  | 	// 		-> undefined
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX might be a good idea to fix filter order...
 | 
					
						
							|  |  |  | 	filters: { | 
					
						
							|  |  |  | 		// placeholders...
 | 
					
						
							|  |  |  | 		nofilters: function(){}, | 
					
						
							|  |  |  | 		isolated: function(){}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// XXX TESTING...
 | 
					
						
							|  |  |  | 		dummy: function(){}, | 
					
						
							|  |  |  | 		test: function(source){ | 
					
						
							|  |  |  | 			return source  | 
					
						
							|  |  |  | 				.replace(/test/g, 'TEST') }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-14 02:08:27 +03:00
										 |  |  | 		'quote-tags': function(source){ | 
					
						
							|  |  |  | 			return source | 
					
						
							|  |  |  | 				.replace(/&/g, '&') | 
					
						
							|  |  |  | 				.replace(/</g, '<') | 
					
						
							|  |  |  | 				.replace(/>/g, '>') }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-09 11:15:37 +03:00
										 |  |  | 		// XXX one way to do this in a stable manner is to wrap the source 
 | 
					
						
							|  |  |  | 		// 		in something like <span wikiwords=yes> .. </span> and only 
 | 
					
						
							|  |  |  | 		// 		process those removing the wrapper in dom...
 | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | 		// 		...not sure how to handle -wikiword filter calls -- now 
 | 
					
						
							|  |  |  | 		// 		this is entirely handled by the parser without calling this...
 | 
					
						
							| 
									
										
										
										
											2022-08-08 23:04:42 +03:00
										 |  |  | 		wikiword: function(){},  | 
					
						
							|  |  |  | 		'quote-wikiword': function(){}, | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		markdown: markdown.markdown, | 
					
						
							|  |  |  | 		'quote-markdown': markdown.quoteMarkdown, | 
					
						
							| 
									
										
										
										
											2022-08-04 19:47:08 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		text: function(source){ | 
					
						
							|  |  |  | 			return `<pre>${source}</pre>` }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	<macro>(<args>, <body>, <state>){ .. }
 | 
					
						
							|  |  |  | 	// 		-> undefined
 | 
					
						
							|  |  |  | 	// 		-> <text>
 | 
					
						
							|  |  |  | 	// 		-> <array>
 | 
					
						
							|  |  |  | 	// 		-> <iterator>
 | 
					
						
							|  |  |  | 	// 		-> <func>(<state>)
 | 
					
						
							|  |  |  | 	// 			-> ...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX ASYNC make these support async page getters...
 | 
					
						
							|  |  |  | 	macros: { | 
					
						
							| 
									
										
										
										
											2022-08-31 18:49:59 +03:00
										 |  |  | 		//
 | 
					
						
							|  |  |  | 		//	@(<name>[ <default>][ local])
 | 
					
						
							|  |  |  | 		//	@(name=<name>[ default=<default>][ local])
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		//
 | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		//	@arg(<name>[ <default>][ local])
 | 
					
						
							|  |  |  | 		//	@arg(name=<name>[ default=<value>][ local])
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							| 
									
										
										
										
											2022-08-31 18:49:59 +03:00
										 |  |  | 		//	<arg <name>[ <default>][ local]/>
 | 
					
						
							|  |  |  | 		//	<arg name=<name>[ default=<value>][ local]/>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		arg: Macro( | 
					
						
							|  |  |  | 			['name', 'default', ['local']], | 
					
						
							|  |  |  | 			function(args){ | 
					
						
							|  |  |  | 				return this.args[args.name]  | 
					
						
							|  |  |  | 					|| (!args.local  | 
					
						
							|  |  |  | 						&& this.root | 
					
						
							|  |  |  | 						&& this.root.args[args.name]) | 
					
						
							|  |  |  | 					|| args.default }), | 
					
						
							| 
									
										
										
										
											2022-08-31 18:49:59 +03:00
										 |  |  | 		// alias to @arg(..)...
 | 
					
						
							|  |  |  | 		'': Macro(  | 
					
						
							|  |  |  | 			['name', 'default', ['local']], | 
					
						
							|  |  |  | 			function(args){ | 
					
						
							|  |  |  | 				return this.macros.arg.call(this, args) }), | 
					
						
							| 
									
										
										
										
											2022-08-31 15:46:51 +03:00
										 |  |  | 		//
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		// 	@filter(<filter-spec>)
 | 
					
						
							|  |  |  | 		// 	<filter <filter-spec>/>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	<filter <filter-spec>>
 | 
					
						
							|  |  |  | 		// 		...
 | 
					
						
							|  |  |  | 		// 	</filter>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	<filter-spec> ::=
 | 
					
						
							|  |  |  | 		// 		<filter> <filter-spec>
 | 
					
						
							|  |  |  | 		// 		| -<filter> <filter-spec>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		filter: function(args, body, state, expand=true){ | 
					
						
							|  |  |  | 			var that = this | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-14 02:08:27 +03:00
										 |  |  | 			var outer = state.filters =  | 
					
						
							|  |  |  | 				state.filters ?? [] | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 			var local = Object.keys(args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// trigger quote-filter...
 | 
					
						
							|  |  |  | 			var quote = local | 
					
						
							|  |  |  | 				.map(function(filter){ | 
					
						
							|  |  |  | 					return (that.filters[filter] ?? {})['quote'] ?? [] }) | 
					
						
							|  |  |  | 				.flat() | 
					
						
							|  |  |  | 			quote.length > 0 | 
					
						
							|  |  |  | 				&& this.macros['quote-filter'] | 
					
						
							|  |  |  | 					.call(this, Object.fromEntries(Object.entries(quote)), null, state) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// local filters...
 | 
					
						
							|  |  |  | 			if(body){ | 
					
						
							|  |  |  | 				// expand the body...
 | 
					
						
							|  |  |  | 				var ast = expand ? | 
					
						
							|  |  |  | 						this.__parser__.expand(this, body, state) | 
					
						
							|  |  |  | 					: body instanceof Array ? | 
					
						
							|  |  |  | 						body | 
					
						
							|  |  |  | 					// NOTE: wrapping the body in an array effectively 
 | 
					
						
							|  |  |  | 					// 		escapes it from parsing...
 | 
					
						
							|  |  |  | 					: [body] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				return async function(state){ | 
					
						
							| 
									
										
										
										
											2022-08-14 02:08:27 +03:00
										 |  |  | 					// XXX can we lose stuff from state this way???
 | 
					
						
							|  |  |  | 					// 		...at this stage it should more or less be static -- check!
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 					var res =  | 
					
						
							| 
									
										
										
										
											2022-08-14 03:00:06 +03:00
										 |  |  | 						await this.__parser__.parse(this, ast, { | 
					
						
							| 
									
										
										
										
											2022-08-14 02:08:27 +03:00
										 |  |  | 							...state, | 
					
						
							|  |  |  | 							filters: local.includes(this.ISOLATED_FILTERS) ? | 
					
						
							|  |  |  | 								local | 
					
						
							|  |  |  | 								: [...outer, ...local], | 
					
						
							|  |  |  | 						}) | 
					
						
							|  |  |  | 					return {data: res} } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// global filters...
 | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				state.filters = [...outer, ...local] } }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	@include(<path>)
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	@include(<path> isolated recursive=<text>)
 | 
					
						
							|  |  |  | 		// 	@include(src=<path> isolated recursive=<text>)
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	<include src=<path> .. >
 | 
					
						
							|  |  |  | 		// 		<text>
 | 
					
						
							|  |  |  | 		// 	</include>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							| 
									
										
										
										
											2022-08-17 11:46:50 +03:00
										 |  |  | 		// NOTE: there can be two ways of recursion in pWiki:
 | 
					
						
							|  |  |  | 		// 			- flat recursion
 | 
					
						
							|  |  |  | 		// 				/A -> /A -> /A -> ..
 | 
					
						
							|  |  |  | 		// 			- nested recursion 
 | 
					
						
							|  |  |  | 		// 				/A -> /A/A -> /A/A/A -> ..
 | 
					
						
							|  |  |  | 		// 		Both can be either direct (type I) or indirect (type II).
 | 
					
						
							|  |  |  | 		// 		The former is trivial to check for while the later is 
 | 
					
						
							|  |  |  | 		// 		not quite so, as we can have different contexts at 
 | 
					
						
							|  |  |  | 		// 		different paths that would lead to different resulting 
 | 
					
						
							|  |  |  | 		// 		renders.
 | 
					
						
							|  |  |  | 		// 		At the moment nested recursion is checked in a fast but 
 | 
					
						
							|  |  |  | 		// 		not 100% correct manner focusing on path depth and ignoring
 | 
					
						
							|  |  |  | 		// 		the context, this potentially can lead to false positives.
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		include: Macro( | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 			['src', 'recursive', 'join', ['strict', 'nonstrict', 'isolated']], | 
					
						
							| 
									
										
										
										
											2022-08-18 10:50:14 +03:00
										 |  |  | 			async function*(args, body, state, key='included', handler){ | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				var macro = 'include' | 
					
						
							|  |  |  | 				if(typeof(args) == 'string'){ | 
					
						
							| 
									
										
										
										
											2022-08-05 02:47:31 +03:00
										 |  |  | 					var [macro, args, body, state, key, handler] = arguments  | 
					
						
							|  |  |  | 					key = key ?? 'included' } | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 				var base = this.get(this.path.split(/\*/).shift()) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				var src = args.src | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 					&& await base.parse(args.src, state) | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 				if(!src){ | 
					
						
							|  |  |  | 					return } | 
					
						
							| 
									
										
										
										
											2022-08-17 15:45:30 +03:00
										 |  |  | 				var recursive = args.recursive ?? body | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				var isolated = args.isolated  | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 				var strict = args.strict | 
					
						
							|  |  |  | 					&& !args.nonstrict | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 				var join = args.join  | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 					&& await base.parse(args.join, state) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-21 18:10:39 +03:00
										 |  |  | 				var depends = state.depends =  | 
					
						
							|  |  |  | 					state.depends  | 
					
						
							|  |  |  | 						?? new Set() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				handler = handler  | 
					
						
							| 
									
										
										
										
											2022-08-27 16:17:25 +03:00
										 |  |  | 					?? async function(src, state){ | 
					
						
							| 
									
										
										
										
											2022-08-14 02:08:27 +03:00
										 |  |  | 						return isolated ? | 
					
						
							|  |  |  | 							{data: await this.get(src) | 
					
						
							| 
									
										
										
										
											2022-08-21 18:10:39 +03:00
										 |  |  | 								.parse({ | 
					
						
							|  |  |  | 									seen: state.seen,  | 
					
						
							|  |  |  | 									depends, | 
					
						
							|  |  |  | 								})} | 
					
						
							| 
									
										
										
										
											2022-08-14 02:08:27 +03:00
										 |  |  | 							: this.get(src) | 
					
						
							|  |  |  | 								.parse(state) } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-18 10:50:14 +03:00
										 |  |  | 				var first = true | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 				for await (var page of this.get(src).asPages(strict)){ | 
					
						
							| 
									
										
										
										
											2022-08-18 20:34:13 +03:00
										 |  |  | 					if(join && !first){ | 
					
						
							| 
									
										
										
										
											2022-08-18 10:50:14 +03:00
										 |  |  | 						yield join } | 
					
						
							|  |  |  | 					first = false | 
					
						
							| 
									
										
										
										
											2022-08-18 10:54:27 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-18 10:50:14 +03:00
										 |  |  | 					var full = page.path | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// handle recursion...
 | 
					
						
							|  |  |  | 					var parent_seen = 'seen' in state | 
					
						
							|  |  |  | 					var seen = state.seen =  | 
					
						
							|  |  |  | 						new Set(state.seen ?? []) | 
					
						
							|  |  |  | 					// recursion detected...
 | 
					
						
							|  |  |  | 					if(seen.has(full) | 
					
						
							|  |  |  | 							// nesting path recursion...
 | 
					
						
							| 
									
										
										
										
											2022-08-27 16:17:25 +03:00
										 |  |  | 							|| (full.length % (this.NESTING_RECURSION_TEST_THRESHOLD || 50) == 0 | 
					
						
							| 
									
										
										
										
											2022-08-27 16:30:23 +03:00
										 |  |  | 								&& (pwpath.split(full).length > 3 | 
					
						
							|  |  |  | 									&& new Set([ | 
					
						
							|  |  |  | 											await page.find(), | 
					
						
							|  |  |  | 											await page.get('..').find(), | 
					
						
							|  |  |  | 											await page.get('../..').find(), | 
					
						
							|  |  |  | 										]).size == 1 | 
					
						
							|  |  |  | 									// XXX HACK???
 | 
					
						
							|  |  |  | 									|| pwpath.split(full).length > (this.NESTING_DEPTH_LIMIT || 20)))){ | 
					
						
							| 
									
										
										
										
											2022-08-18 10:50:14 +03:00
										 |  |  | 						if(recursive == null){ | 
					
						
							| 
									
										
										
										
											2022-08-27 16:17:25 +03:00
										 |  |  | 							console.warn( | 
					
						
							|  |  |  | 								`@${key}(..): ${ | 
					
						
							|  |  |  | 									seen.has(full) ? | 
					
						
							|  |  |  | 										'direct' | 
					
						
							|  |  |  | 										: 'depth-limit' | 
					
						
							| 
									
										
										
										
											2022-08-28 10:33:47 +03:00
										 |  |  | 								} recursion detected:`, full, seen)
 | 
					
						
							| 
									
										
										
										
											2022-08-27 16:17:25 +03:00
										 |  |  | 							yield page.get(page.RECURSION_ERROR).parse()  | 
					
						
							| 
									
										
										
										
											2022-08-18 10:50:14 +03:00
										 |  |  | 							continue } | 
					
						
							|  |  |  | 						// have the 'recursive' arg...
 | 
					
						
							|  |  |  | 						yield base.parse(recursive, state)  | 
					
						
							|  |  |  | 						continue } | 
					
						
							|  |  |  | 					seen.add(full) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// load the included page...
 | 
					
						
							| 
									
										
										
										
											2022-08-27 16:17:25 +03:00
										 |  |  | 					var res = await handler.call(page, full, state) | 
					
						
							| 
									
										
										
										
											2022-08-21 18:10:39 +03:00
										 |  |  | 					depends.add(full) | 
					
						
							| 
									
										
										
										
											2022-08-18 10:50:14 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					// NOTE: we only track recursion down and not sideways...
 | 
					
						
							|  |  |  | 					seen.delete(full) | 
					
						
							|  |  |  | 					if(!parent_seen){ | 
					
						
							|  |  |  | 						delete state.seen } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					yield res } }), | 
					
						
							| 
									
										
										
										
											2022-08-05 02:47:31 +03:00
										 |  |  | 		// NOTE: the main difference between this and @include is that 
 | 
					
						
							|  |  |  | 		// 		this renders the src in the context of current page while 
 | 
					
						
							|  |  |  | 		// 		include is rendered in the context of its page but with
 | 
					
						
							|  |  |  | 		// 		the same state...
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		source: Macro( | 
					
						
							|  |  |  | 			['src'], | 
					
						
							| 
									
										
										
										
											2022-08-18 10:50:14 +03:00
										 |  |  | 			async function*(args, body, state){ | 
					
						
							|  |  |  | 				yield* this.macros.include.call(this,  | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 					'source', | 
					
						
							|  |  |  | 					args, body, state, 'sources',  | 
					
						
							| 
									
										
										
										
											2022-08-27 16:17:25 +03:00
										 |  |  | 					async function(src, state){ | 
					
						
							| 
									
										
										
										
											2022-08-17 11:46:50 +03:00
										 |  |  | 						return this.parse(this.get(src).raw, state) }) }), | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	@quote(<src>)
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	<quote src=<src>[ filter="<filter> ..."]/>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	<quote text=" .. "[ filter="<filter> ..."]/>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	<quote[ filter="<filter> ..."]>
 | 
					
						
							|  |  |  | 		// 		..
 | 
					
						
							|  |  |  | 		// 	</quote>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// NOTE: src ant text arguments are mutually exclusive, src takes 
 | 
					
						
							|  |  |  | 		// 		priority.
 | 
					
						
							|  |  |  | 		// NOTE: the filter argument has the same semantics as the filter 
 | 
					
						
							|  |  |  | 		// 		macro with one exception, when used in quote, the body is 
 | 
					
						
							|  |  |  | 		// 		not expanded...
 | 
					
						
							| 
									
										
										
										
											2022-08-13 17:02:32 +03:00
										 |  |  | 		// NOTE: the filter argument uses the same filters as @filter(..)
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		//
 | 
					
						
							|  |  |  | 		// XXX need a way to escape macros -- i.e. include </quote> in a quoted text...
 | 
					
						
							|  |  |  | 		quote: Macro( | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 			['src', 'filter', 'text', 'join', ['expandactions']], | 
					
						
							| 
									
										
										
										
											2022-08-18 20:34:13 +03:00
										 |  |  | 			async function*(args, body, state){ | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				var src = args.src //|| args[0]
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 				var base = this.get(this.path.split(/\*/).shift()) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				var text = args.text  | 
					
						
							|  |  |  | 					?? body  | 
					
						
							|  |  |  | 					?? [] | 
					
						
							|  |  |  | 				// parse arg values...
 | 
					
						
							|  |  |  | 				src = src ?  | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 					await base.parse(src, state) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 					: src | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 				var expandactions = args.expandactions | 
					
						
							| 
									
										
										
										
											2022-08-18 20:34:13 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-21 18:10:39 +03:00
										 |  |  | 				var depends = state.depends =  | 
					
						
							|  |  |  | 					state.depends  | 
					
						
							|  |  |  | 						?? new Set() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-18 20:34:13 +03:00
										 |  |  | 				var pages = src ? | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 						(!expandactions  | 
					
						
							|  |  |  | 								&& await this.get(src).type == 'action' ? | 
					
						
							|  |  |  | 							base.get(this.QUOTE_ACTION_PAGE) | 
					
						
							|  |  |  | 							: this.get(src).asPages()) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 					: text instanceof Array ? | 
					
						
							| 
									
										
										
										
											2022-08-18 20:34:13 +03:00
										 |  |  | 						[text.join('')] | 
					
						
							|  |  |  | 					: typeof(text) == 'string' ? | 
					
						
							|  |  |  | 						[text] | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 					: text | 
					
						
							|  |  |  | 				// empty...
 | 
					
						
							| 
									
										
										
										
											2022-08-18 20:34:13 +03:00
										 |  |  | 				if(!pages){ | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 					return } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-18 20:34:13 +03:00
										 |  |  | 				var join = args.join  | 
					
						
							|  |  |  | 					&& await base.parse(args.join, state) | 
					
						
							|  |  |  | 				var first = true | 
					
						
							|  |  |  | 				for await (var page of pages){ | 
					
						
							|  |  |  | 					if(join && !first){ | 
					
						
							|  |  |  | 						yield join } | 
					
						
							|  |  |  | 					first = false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					text = typeof(page) == 'string' ? | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 							page | 
					
						
							|  |  |  | 						: (!expandactions  | 
					
						
							|  |  |  | 								&& await page.type == 'action') ? | 
					
						
							|  |  |  | 							base.get(this.QUOTE_ACTION_PAGE).raw | 
					
						
							| 
									
										
										
										
											2022-08-18 20:34:13 +03:00
										 |  |  | 						: await page.raw | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-21 18:10:39 +03:00
										 |  |  | 					page.path | 
					
						
							|  |  |  | 						&& depends.add(page.path) | 
					
						
							| 
									
										
										
										
											2022-08-18 20:34:13 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					var filters =  | 
					
						
							|  |  |  | 						args.filter  | 
					
						
							|  |  |  | 							&& args.filter | 
					
						
							|  |  |  | 								.trim() | 
					
						
							|  |  |  | 								.split(/\s+/g) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// NOTE: we are delaying .quote_filters handling here to 
 | 
					
						
							|  |  |  | 					// 		make their semantics the same as general filters...
 | 
					
						
							|  |  |  | 					// 		...and since we are internally calling .filter(..)
 | 
					
						
							|  |  |  | 					// 		macro we need to dance around it's architecture too...
 | 
					
						
							|  |  |  | 					// NOTE: since the body of quote(..) only has filters applied 
 | 
					
						
							|  |  |  | 					// 		to it doing the first stage of .filter(..) as late 
 | 
					
						
							|  |  |  | 					// 		as the second stage here will have no ill effect...
 | 
					
						
							|  |  |  | 					// NOTE: this uses the same filters as @filter(..)
 | 
					
						
							|  |  |  | 					// NOTE: the function wrapper here isolates text in 
 | 
					
						
							|  |  |  | 					// 		a closure per function...
 | 
					
						
							|  |  |  | 					yield (function(text){ | 
					
						
							|  |  |  | 						return async function(state){ | 
					
						
							|  |  |  | 							// add global quote-filters...
 | 
					
						
							|  |  |  | 							filters = | 
					
						
							|  |  |  | 								(state.quote_filters  | 
					
						
							|  |  |  | 										&& !(filters ?? []).includes(this.ISOLATED_FILTERS)) ? | 
					
						
							|  |  |  | 									[...state.quote_filters, ...(filters ?? [])] | 
					
						
							|  |  |  | 									: filters | 
					
						
							|  |  |  | 							return filters ? | 
					
						
							|  |  |  | 								await this.__parser__.callMacro( | 
					
						
							|  |  |  | 										this, 'filter', filters, text, state, false) | 
					
						
							|  |  |  | 									.call(this, state) | 
					
						
							|  |  |  | 								: text } })(text) } }), | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		// very similar to @filter(..) but will affect @quote(..) filters...
 | 
					
						
							|  |  |  | 		'quote-filter': function(args, body, state){ | 
					
						
							|  |  |  | 			var filters = state.quote_filters =  | 
					
						
							|  |  |  | 				state.quote_filters ?? [] | 
					
						
							|  |  |  | 			filters.splice(filters.length, 0, ...Object.keys(args)) }, | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		//	<slot name=<name>/>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		//	<slot name=<name> text=<text>/>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		//	<slot name=<name>>
 | 
					
						
							|  |  |  | 		//		...
 | 
					
						
							|  |  |  | 		//	</slot>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		//	Force show a slot...
 | 
					
						
							|  |  |  | 		//	<slot shown ... />
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		//	Force hide a slot...
 | 
					
						
							|  |  |  | 		//	<slot hidden ... />
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							| 
									
										
										
										
											2022-08-18 01:23:49 +03:00
										 |  |  | 		//	Insert previous slot content...
 | 
					
						
							|  |  |  | 		//	<content/>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		//
 | 
					
						
							|  |  |  | 		// NOTE: by default only the first slot with <name> is visible, 
 | 
					
						
							| 
									
										
										
										
											2022-08-17 14:10:25 +03:00
										 |  |  | 		// 		all other slots with <name> will replace its content, unless
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		// 		explicit shown/hidden arguments are given.
 | 
					
						
							|  |  |  | 		// NOTE: hidden has precedence over shown if both are given.
 | 
					
						
							| 
									
										
										
										
											2022-08-19 13:13:17 +03:00
										 |  |  | 		// NOTE: slots are handled in order of occurrence of opening tags 
 | 
					
						
							|  |  |  | 		// 		in text and not by hierarchy, i.e. the later slot overrides
 | 
					
						
							|  |  |  | 		// 		the former and the most nested overrides the parent.
 | 
					
						
							|  |  |  | 		// 		This also works for cases where slots override slots they 
 | 
					
						
							|  |  |  | 		// 		are contained in, this will not lead to recursion.
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		//
 | 
					
						
							| 
									
										
										
										
											2022-08-08 12:29:02 +03:00
										 |  |  | 		// XXX revise the use of hidden/shown use mechanic and if it's 
 | 
					
						
							|  |  |  | 		// 		needed...
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		slot: Macro( | 
					
						
							|  |  |  | 			['name', 'text', ['shown', 'hidden']], | 
					
						
							|  |  |  | 			async function(args, body, state){ | 
					
						
							|  |  |  | 				var name = args.name | 
					
						
							|  |  |  | 				var text = args.text  | 
					
						
							|  |  |  | 					?? body  | 
					
						
							|  |  |  | 					// NOTE: this can't be undefined for .expand(..) to work 
 | 
					
						
							|  |  |  | 					// 		correctly...
 | 
					
						
							|  |  |  | 					?? [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				var slots = state.slots =  | 
					
						
							|  |  |  | 					state.slots  | 
					
						
							|  |  |  | 						?? {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// parse arg values...
 | 
					
						
							|  |  |  | 				name = name ? | 
					
						
							|  |  |  | 					await this.parse(name, state) | 
					
						
							|  |  |  | 					: name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				//var hidden = name in slots
 | 
					
						
							|  |  |  | 				// XXX EXPERIMENTAL
 | 
					
						
							|  |  |  | 				var hidden =  | 
					
						
							|  |  |  | 					// 'hidden' has priority... 
 | 
					
						
							|  |  |  | 					args.hidden | 
					
						
							|  |  |  | 						// explicitly show... ()
 | 
					
						
							|  |  |  | 						|| (args.shown ? | 
					
						
							|  |  |  | 							false | 
					
						
							|  |  |  | 							// show first instance...
 | 
					
						
							|  |  |  | 							: name in slots) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 				// set slot value...
 | 
					
						
							| 
									
										
										
										
											2022-08-19 13:13:17 +03:00
										 |  |  | 				var stack = [] | 
					
						
							|  |  |  | 				slots[name] | 
					
						
							|  |  |  | 					&& stack.push(slots[name]) | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 				delete slots[name] | 
					
						
							|  |  |  | 				var slot = await this.__parser__.expand(this, text, state) | 
					
						
							| 
									
										
										
										
											2022-08-19 13:13:17 +03:00
										 |  |  | 				slots[name] | 
					
						
							|  |  |  | 					&& stack.unshift(slot) | 
					
						
							| 
									
										
										
										
											2022-08-18 01:23:49 +03:00
										 |  |  | 				slot = slots[name] =  | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 					slots[name]  | 
					
						
							|  |  |  | 						?? slot | 
					
						
							| 
									
										
										
										
											2022-08-18 11:14:31 +03:00
										 |  |  | 				// handle <content/>...
 | 
					
						
							| 
									
										
										
										
											2022-08-19 13:13:17 +03:00
										 |  |  | 				for(prev of stack){ | 
					
						
							|  |  |  | 					// get the first <content/>
 | 
					
						
							|  |  |  | 					for(var i in slot){ | 
					
						
							|  |  |  | 						if(typeof(slot[i]) != 'string' | 
					
						
							|  |  |  | 								&& slot[i].name == 'content'){ | 
					
						
							|  |  |  | 							break }  | 
					
						
							|  |  |  | 						i = null } | 
					
						
							|  |  |  | 					i != null | 
					
						
							|  |  |  | 						&& slot.splice(i, 1,  | 
					
						
							|  |  |  | 							...prev | 
					
						
							|  |  |  | 								// remove nested slot handlers...
 | 
					
						
							|  |  |  | 								.filter(function(e){ | 
					
						
							|  |  |  | 									return typeof(e) != 'function' | 
					
						
							|  |  |  | 											|| e.slot != name }) ) } | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				return hidden ? | 
					
						
							|  |  |  | 					'' | 
					
						
							| 
									
										
										
										
											2022-08-19 13:13:17 +03:00
										 |  |  | 					: Object.assign( | 
					
						
							|  |  |  | 						function(state){ | 
					
						
							|  |  |  | 							return state.slots[name] }, | 
					
						
							|  |  |  | 						{slot: name}) }),  | 
					
						
							| 
									
										
										
										
											2022-08-18 01:23:49 +03:00
										 |  |  | 		'content': ['slot'], | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// 	
 | 
					
						
							|  |  |  | 		// 	<macro src=<url>> .. </macro>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	<macro name=<name> src=<url> sort=<sort-spec>> .. </macro>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	<macro ...> ... </macro>
 | 
					
						
							|  |  |  | 		// 	<macro ... text=<text>/>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 	<macro ... else=<text>> ... </macro>
 | 
					
						
							|  |  |  | 		// 	<macro ...>
 | 
					
						
							|  |  |  | 		// 		...
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 		<join>
 | 
					
						
							|  |  |  | 		// 			...
 | 
					
						
							|  |  |  | 		// 		</join>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// 		<else>
 | 
					
						
							|  |  |  | 		// 			...
 | 
					
						
							|  |  |  | 		// 		</else>
 | 
					
						
							|  |  |  | 		// 	</macro>
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// NOTE: if both strict and nonstrict are given the later takes 
 | 
					
						
							|  |  |  | 		// 		precedence.
 | 
					
						
							| 
									
										
										
										
											2022-08-18 10:50:14 +03:00
										 |  |  | 		//
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		// XXX SORT sorting not implemented yet....
 | 
					
						
							|  |  |  | 		macro: Macro( | 
					
						
							|  |  |  | 			['name', 'src', 'sort', 'text', 'join', 'else', ['strict', 'nonstrict']], | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 			async function*(args, body, state){ | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				var that = this | 
					
						
							|  |  |  | 				var name = args.name //?? args[0]
 | 
					
						
							|  |  |  | 				var src = args.src | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 				var base = this.get(this.path.split(/\*/).shift()) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				var sort = (args.sort ?? '') | 
					
						
							|  |  |  | 					.split(/\s+/g) | 
					
						
							|  |  |  | 					.filter(function(e){  | 
					
						
							|  |  |  | 						return e != '' }) | 
					
						
							|  |  |  | 				// NOTE: args.text will need parsing...
 | 
					
						
							|  |  |  | 				var text = args.text  | 
					
						
							|  |  |  | 					?? body  | 
					
						
							|  |  |  | 					?? [] | 
					
						
							|  |  |  | 				text = typeof(text) == 'string' ? | 
					
						
							|  |  |  | 					[...this.__parser__.group(this, text+'</macro>', 'macro')] | 
					
						
							|  |  |  | 					: text | 
					
						
							|  |  |  | 				var strict = args.strict | 
					
						
							|  |  |  | 					&& !args.nonstrict | 
					
						
							| 
									
										
										
										
											2022-08-21 01:28:29 +03:00
										 |  |  | 				var join | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-21 18:10:39 +03:00
										 |  |  | 				var depends = state.depends =  | 
					
						
							|  |  |  | 					state.depends  | 
					
						
							|  |  |  | 						?? new Set() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 				var _getBlock = function(name){ | 
					
						
							|  |  |  | 					var block = args[name] ? | 
					
						
							|  |  |  | 						[{ | 
					
						
							|  |  |  | 							args: {}, | 
					
						
							|  |  |  | 							body: args[name], | 
					
						
							|  |  |  | 						}] | 
					
						
							|  |  |  | 						: (text ?? []) | 
					
						
							|  |  |  | 							.filter(function(e){  | 
					
						
							|  |  |  | 								return typeof(e) != 'string'  | 
					
						
							|  |  |  | 									&& e.name == name }) | 
					
						
							|  |  |  | 					if(block.length == 0){ | 
					
						
							|  |  |  | 						return } | 
					
						
							|  |  |  | 					// NOTE: when multiple blocks are present the 
 | 
					
						
							|  |  |  | 					// 		last one is used...
 | 
					
						
							|  |  |  | 					block = block.pop() | 
					
						
							|  |  |  | 					block =  | 
					
						
							|  |  |  | 						block.args.text  | 
					
						
							|  |  |  | 							?? block.body | 
					
						
							|  |  |  | 					return block } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if(name){ | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 					name = await base.parse(name, state) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 					// define new named macro...
 | 
					
						
							| 
									
										
										
										
											2022-08-21 01:28:29 +03:00
										 |  |  | 					if(text.length != 0){ | 
					
						
							|  |  |  | 						// NOTE: we do not need to worry about saving 
 | 
					
						
							|  |  |  | 						// 		stateful text here because it is only 
 | 
					
						
							|  |  |  | 						// 		grouped and not expanded...
 | 
					
						
							|  |  |  | 						;(state.macros = state.macros ?? {})[name] = [text, _getBlock('join')] | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 					// use existing macro...
 | 
					
						
							|  |  |  | 					} else if(state.macros  | 
					
						
							|  |  |  | 							&& name in state.macros){ | 
					
						
							| 
									
										
										
										
											2022-08-21 01:28:29 +03:00
										 |  |  | 						[text, join] = state.macros[name] } } | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				if(src){ | 
					
						
							| 
									
										
										
										
											2022-08-21 01:28:29 +03:00
										 |  |  | 					join = _getBlock('join')  | 
					
						
							|  |  |  | 						?? join  | 
					
						
							| 
									
										
										
										
											2022-08-19 19:59:58 +03:00
										 |  |  | 					join = join | 
					
						
							|  |  |  | 						&& await base.parse(join, state) | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-25 11:28:01 +03:00
										 |  |  | 					var match = this.get(await base.parse(src, state)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 					// NOTE: thie does not introduce a dependency on each 
 | 
					
						
							|  |  |  | 					// 		of the iterated pages, that is handled by the 
 | 
					
						
							|  |  |  | 					// 		respective include/source/.. macros, this however
 | 
					
						
							|  |  |  | 					// 		only depends on page count...
 | 
					
						
							| 
									
										
										
										
											2022-08-25 00:40:57 +03:00
										 |  |  | 					depends.add(match.path) | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 					// expand matches...
 | 
					
						
							|  |  |  | 					var first = true | 
					
						
							| 
									
										
										
										
											2022-08-25 00:40:57 +03:00
										 |  |  | 					for await(var page of match.asPages(strict)){ | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 						if(join && !first){ | 
					
						
							|  |  |  | 							yield join } | 
					
						
							|  |  |  | 						first = false  | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 						yield this.__parser__.expand(page, text, state) } | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 					// else...
 | 
					
						
							|  |  |  | 					if(first | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 							&& (text || args['else'])){ | 
					
						
							|  |  |  | 						var else_block = _getBlock('else') | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 						if(else_block){ | 
					
						
							|  |  |  | 							yield this.__parser__.expand(this, else_block, state) } } } }), | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// nesting rules...
 | 
					
						
							|  |  |  | 		'else': ['macro'], | 
					
						
							|  |  |  | 		'join': ['macro'], | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-02 14:58:45 +03:00
										 |  |  | 	// direct actions...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// These are evaluated directly without the need to go through the 
 | 
					
						
							|  |  |  | 	// whole page acquisition process...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: these can not be overloaded. 
 | 
					
						
							|  |  |  | 	// 		(XXX should this be so?)
 | 
					
						
							|  |  |  | 	actions: new Set([ | 
					
						
							|  |  |  | 		'location', | 
					
						
							|  |  |  | 		'referrer', | 
					
						
							|  |  |  | 		'path', | 
					
						
							|  |  |  | 		'name', | 
					
						
							|  |  |  | 		'dir', | 
					
						
							|  |  |  | 		'resolved', | 
					
						
							|  |  |  | 		'rootpath', | 
					
						
							|  |  |  | 		'renderer', | 
					
						
							|  |  |  | 		'length', | 
					
						
							|  |  |  | 		'type', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		//'ctime',
 | 
					
						
							|  |  |  | 		//'mtime',
 | 
					
						
							|  |  |  | 	]), | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	// events...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: textUpdate event will not get triggered if text is updated 
 | 
					
						
							|  |  |  | 	// 		directly via .data or .__update__(..)
 | 
					
						
							|  |  |  | 	/*/ XXX EVENTS do we need this??? | 
					
						
							|  |  |  | 	onTextUpdate: types.event.Event('textUpdate', | 
					
						
							|  |  |  | 		function(handle, text){ | 
					
						
							|  |  |  | 			this.__update__({text}) }), | 
					
						
							|  |  |  | 	// XXX EVENTS not sure where to trigger this???
 | 
					
						
							|  |  |  | 	// 		...on .parse(..) is a bit too granular, something like .text??
 | 
					
						
							|  |  |  | 	// XXX not triggered yet...
 | 
					
						
							|  |  |  | 	//onParse: types.event.Event('parse'),
 | 
					
						
							|  |  |  | 	//*/
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// page parser...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	parse: async function(text, state){ | 
					
						
							|  |  |  | 		var that = this | 
					
						
							| 
									
										
										
										
											2022-08-17 11:46:50 +03:00
										 |  |  | 		text = await text | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		// .parser(<state>)
 | 
					
						
							|  |  |  | 		if(arguments.length == 1  | 
					
						
							|  |  |  | 				&& text instanceof Object | 
					
						
							|  |  |  | 				&& !(text instanceof Array)){ | 
					
						
							|  |  |  | 			state = text | 
					
						
							|  |  |  | 			text = null } | 
					
						
							|  |  |  | 		state = state ?? {} | 
					
						
							| 
									
										
										
										
											2022-08-08 23:04:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 		return this.__parser__.parse(this, text, state) }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// true if page has an array value but is not a pattern page...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX the split into pattern and array pages feels a bit overcomplicated...
 | 
					
						
							|  |  |  | 	// 		...can we merge the two and simplify things???
 | 
					
						
							|  |  |  | 	// XXX EXPERIMENTAL
 | 
					
						
							|  |  |  | 	get isArray(){ return (async function(){ | 
					
						
							|  |  |  | 		return !this.isPattern  | 
					
						
							|  |  |  | 			// NOTE: we can't only use .data here as it can be a function 
 | 
					
						
							|  |  |  | 			// 		that will return an array...
 | 
					
						
							|  |  |  | 			&& await this.raw instanceof Array }).call(this) }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// raw page text...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: writing to .raw is the same as writing to .text...
 | 
					
						
							|  |  |  | 	// NOTE: when matching multiple pages this will return a list...
 | 
					
						
							| 
									
										
										
										
											2022-08-28 16:34:46 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX revise how we handle .strict mode...
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	get raw(){ return (async function(){ | 
					
						
							| 
									
										
										
										
											2022-09-02 14:58:45 +03:00
										 |  |  | 		// direct actions...
 | 
					
						
							|  |  |  | 		if(this.actions  | 
					
						
							|  |  |  | 				&& this.actions.has(this.name)){ | 
					
						
							|  |  |  | 			var name = this.name | 
					
						
							|  |  |  | 			var page = this.get('..') | 
					
						
							|  |  |  | 			var res = (this.isPattern  | 
					
						
							|  |  |  | 					&& !this.energetic) ? | 
					
						
							|  |  |  | 				page | 
					
						
							|  |  |  | 					.each() | 
					
						
							|  |  |  | 					.map(function(page){ | 
					
						
							|  |  |  | 						var res = page[name]  | 
					
						
							|  |  |  | 						return typeof(res) == 'function' ? | 
					
						
							|  |  |  | 							res.call(page) | 
					
						
							|  |  |  | 							: res }) | 
					
						
							|  |  |  | 				: page[this.name]  | 
					
						
							|  |  |  | 			return typeof(res) == 'function' ? | 
					
						
							|  |  |  | 				res.call(page)	 | 
					
						
							|  |  |  | 				: res } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		var data = await this.data | 
					
						
							|  |  |  | 		// no data...
 | 
					
						
							|  |  |  | 		// NOTE: if we hit this it means that nothing was resolved, 
 | 
					
						
							|  |  |  | 		// 		not even the System/NotFound page, i.e. something 
 | 
					
						
							|  |  |  | 		// 		went really wrong...
 | 
					
						
							| 
									
										
										
										
											2022-08-28 16:34:46 +03:00
										 |  |  | 		// NOTE: in .strict mode this will explicitly fail and not try 
 | 
					
						
							|  |  |  | 		// 		to recover...
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		if(data == null){ | 
					
						
							| 
									
										
										
										
											2022-08-28 16:34:46 +03:00
										 |  |  | 			if(!this.strict  | 
					
						
							|  |  |  | 					&& this.NOT_FOUND_ERROR){ | 
					
						
							| 
									
										
										
										
											2022-08-17 11:46:50 +03:00
										 |  |  | 				var msg = this.get(this.NOT_FOUND_ERROR) | 
					
						
							| 
									
										
										
										
											2022-08-05 02:47:31 +03:00
										 |  |  | 				if(await msg.match()){ | 
					
						
							|  |  |  | 					return msg.raw } } | 
					
						
							|  |  |  | 			// last resort...
 | 
					
						
							|  |  |  | 			throw new Error('NOT FOUND ERROR: '+ this.path) } | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 		// get the data...
 | 
					
						
							|  |  |  | 		return ( | 
					
						
							|  |  |  | 			// action...
 | 
					
						
							|  |  |  | 			typeof(data) == 'function' ? | 
					
						
							|  |  |  | 				data() | 
					
						
							|  |  |  | 			// multiple matches...
 | 
					
						
							|  |  |  | 			: data instanceof Array ? | 
					
						
							| 
									
										
										
										
											2022-08-19 16:02:45 +03:00
										 |  |  | 				Promise.all(data | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 					.map(function(d){ | 
					
						
							|  |  |  | 						return typeof(d) == 'function'? | 
					
						
							|  |  |  | 							d() | 
					
						
							|  |  |  | 							: d.text }) | 
					
						
							| 
									
										
										
										
											2022-08-19 16:02:45 +03:00
										 |  |  | 					.flat()) | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  |    			: data.text )}).call(this) }, | 
					
						
							|  |  |  | 	set raw(value){ | 
					
						
							|  |  |  | 		this.__update__({text: value}) }, | 
					
						
							|  |  |  | 		//this.onTextUpdate(value) },
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 	// iterate matches or content list as pages...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 	// 	.asPages()
 | 
					
						
							|  |  |  | 	// 	.asPages(<path>[, <options>])
 | 
					
						
							|  |  |  | 	// 	.asPages(<strict>[, <options>])
 | 
					
						
							|  |  |  | 	// 	.asPages(<path>, <strict>[, <options>])
 | 
					
						
							|  |  |  | 	// 	.asPages(<options>)
 | 
					
						
							|  |  |  | 	// 		-> <iter>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: this will get .raw for non-pattern pages this it can trigger 
 | 
					
						
							|  |  |  | 	// 		actions...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 	// XXX revise name...
 | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 	asPages: async function*(path='.', strict=false, noexpandactions=false){ | 
					
						
							|  |  |  | 		// options...
 | 
					
						
							|  |  |  | 		var args = [...arguments] | 
					
						
							|  |  |  | 		var opts = typeof(args.at(-1)) == 'object' ? | 
					
						
							|  |  |  | 			args.pop() | 
					
						
							|  |  |  | 			: {} | 
					
						
							|  |  |  | 		var {path, strict, noexpandactions} = { | 
					
						
							|  |  |  | 			...opts, | 
					
						
							|  |  |  | 			path: typeof(args[0]) == 'string' ? | 
					
						
							|  |  |  | 				args.shift() | 
					
						
							|  |  |  | 				: '.', | 
					
						
							|  |  |  | 			strict: args.shift() ?? false, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 		var page = this.get(path, strict) | 
					
						
							|  |  |  | 		// handle lists in pages (actions, ... etc.)...
 | 
					
						
							|  |  |  | 		if(!page.isPattern){ | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 			if(noexpandactions  | 
					
						
							|  |  |  | 					&& await page.type == 'action'){ | 
					
						
							|  |  |  | 				//yield this.get(this.QUOTE_ACTION_PAGE)
 | 
					
						
							|  |  |  | 				yield page  | 
					
						
							|  |  |  | 				return } | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 			var raw = await page.raw | 
					
						
							| 
									
										
										
										
											2022-08-21 20:45:29 +03:00
										 |  |  | 			if(raw == null){ | 
					
						
							|  |  |  | 				return } | 
					
						
							| 
									
										
										
										
											2022-08-19 19:40:28 +03:00
										 |  |  | 			yield* raw instanceof Array ? | 
					
						
							|  |  |  | 				raw | 
					
						
							|  |  |  | 					.map(function(p){ | 
					
						
							|  |  |  | 						return page.virtual({text: p}) }) | 
					
						
							|  |  |  | 				: [page]  | 
					
						
							|  |  |  | 		// each...
 | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			yield* page } }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	// expanded page text...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-05 18:21:18 +03:00
										 |  |  | 	// NOTE: this uses .PAGE_TEMPLATE to render the page.
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	// NOTE: writing to .raw is the same as writing to .text...
 | 
					
						
							| 
									
										
										
										
											2022-08-25 11:28:01 +03:00
										 |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-28 16:34:46 +03:00
										 |  |  | 	// XXX revise how we handle strict mode...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-25 11:28:01 +03:00
										 |  |  | 	// NOTE: .__debug_last_render_state is mainly exposed for introspection 
 | 
					
						
							|  |  |  | 	// 		and debugging, set comment it out to disable...
 | 
					
						
							|  |  |  | 	//__debug_last_render_state: undefined,
 | 
					
						
							| 
									
										
										
										
											2022-08-05 02:47:31 +03:00
										 |  |  | 	get text(){ return (async function(){ | 
					
						
							| 
									
										
										
										
											2022-08-28 16:34:46 +03:00
										 |  |  | 		// strict mode -- break on non-existing pages...
 | 
					
						
							|  |  |  | 		if(this.strict  | 
					
						
							|  |  |  | 				&& !await this.resolve(true)){ | 
					
						
							|  |  |  | 			throw new Error('NOT FOUND ERROR: '+ this.path) } | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 		var path = pwpath.split(this.path) | 
					
						
							|  |  |  | 		path.at(-1)[0] == '_' | 
					
						
							| 
									
										
										
										
											2022-08-05 18:21:18 +03:00
										 |  |  | 			|| path.push(this.PAGE_TEMPLATE) | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		var tpl = pwpath.join(path) | 
					
						
							|  |  |  | 		var tpl_name = path.pop() | 
					
						
							|  |  |  | 		path = pwpath.join(path) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// get the template relative to the top most pattern...
 | 
					
						
							|  |  |  | 		tpl = await this.get(tpl).find(true) | 
					
						
							|  |  |  | 		if(!tpl){ | 
					
						
							| 
									
										
										
										
											2022-08-18 01:23:49 +03:00
										 |  |  | 			console.warn('UNKNOWN RENDER TEMPLATE: '+ tpl_name)  | 
					
						
							|  |  |  | 			return this.get(this.NOT_FOUND_TEMPLATE_ERROR).parse() } | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// render template in context of page...
 | 
					
						
							| 
									
										
										
										
											2022-08-21 18:10:39 +03:00
										 |  |  | 		var depends = this.depends = new Set([tpl]) | 
					
						
							|  |  |  | 		var state = {depends} | 
					
						
							| 
									
										
										
										
											2022-08-25 11:28:01 +03:00
										 |  |  | 		// this is here for debugging and introspection...
 | 
					
						
							|  |  |  | 		'__debug_last_render_state' in this | 
					
						
							|  |  |  | 			&& (this.__debug_last_render_state = state) | 
					
						
							| 
									
										
										
										
											2022-08-16 11:00:23 +03:00
										 |  |  | 		var data = { render_root: this } | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 		return this.get(path, data) | 
					
						
							| 
									
										
										
										
											2022-08-24 23:13:24 +03:00
										 |  |  | 			.parse( | 
					
						
							|  |  |  | 				this.get('/'+tpl, data).raw,  | 
					
						
							|  |  |  | 				state) }).call(this) }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 	set text(value){ | 
					
						
							|  |  |  | 		this.__update__({text: value}) }, | 
					
						
							|  |  |  | 		//this.onTextUpdate(value) },
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-16 09:35:07 +03:00
										 |  |  | 	// pass on .render_root to clones...
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 	clone: function(data={}, ...args){ | 
					
						
							|  |  |  | 		this.render_root | 
					
						
							|  |  |  | 			&& (data = {render_root: this.render_root, ...data}) | 
					
						
							|  |  |  | 		return object.parentCall(Page.prototype.clone, this, data, ...args) }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | //---------------------------------------------------------------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-25 11:28:01 +03:00
										 |  |  | var getCachedProp =  | 
					
						
							|  |  |  | function(obj, name){ | 
					
						
							| 
									
										
										
										
											2022-08-24 23:47:42 +03:00
										 |  |  | 	var that = obj | 
					
						
							|  |  |  | 	var value = obj.cache ? | 
					
						
							|  |  |  | 		obj.cache[name] | 
					
						
							|  |  |  | 		: object.parentProperty(CachedPage.prototype, name).get.call(obj) | 
					
						
							|  |  |  | 	value instanceof Promise | 
					
						
							|  |  |  | 		&& value.then(function(value){ | 
					
						
							|  |  |  | 			that.cache = {[name]: value} }) | 
					
						
							|  |  |  | 	return value } | 
					
						
							| 
									
										
										
										
											2022-08-25 11:28:01 +03:00
										 |  |  | var setCachedProp =  | 
					
						
							|  |  |  | function(obj, name, value){ | 
					
						
							| 
									
										
										
										
											2022-08-24 23:47:42 +03:00
										 |  |  | 	return object.parentProperty(CachedPage.prototype, name).set.call(obj, value) } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | // XXX this is good enough on the front-side to think about making the 
 | 
					
						
							|  |  |  | // 		cache persistent with a very large timeout (if set at all), but
 | 
					
						
							|  |  |  | // 		we are not tracking changes on the store-side...
 | 
					
						
							|  |  |  | var CachedPage = | 
					
						
							|  |  |  | module.CachedPage = | 
					
						
							|  |  |  | object.Constructor('CachedPage', Page, { | 
					
						
							| 
									
										
										
										
											2022-09-01 00:04:13 +03:00
										 |  |  | 	// Sets what to use for cache id...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// Can be:
 | 
					
						
							|  |  |  | 	// 		'location' (default)
 | 
					
						
							|  |  |  | 	// 		'path'
 | 
					
						
							|  |  |  | 	cache_id: 'location', | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 	// NOTE: set this to null/undefined/0 to disable...
 | 
					
						
							|  |  |  | 	cache_timeout: '20m', | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 12:31:19 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// keep all the cache in one place -- .root
 | 
					
						
							|  |  |  | 	//__cachestore: undefined,
 | 
					
						
							|  |  |  | 	get cachestore(){ | 
					
						
							|  |  |  | 		return (this.root ?? this).__cachestore }, | 
					
						
							|  |  |  | 	set cachestore(value){ | 
					
						
							|  |  |  | 		;(this.root ?? this).__cachestore = value }, | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 	get cache(){ | 
					
						
							| 
									
										
										
										
											2022-09-01 00:04:13 +03:00
										 |  |  | 		this.checkCache(this[this.cache_id]) | 
					
						
							|  |  |  | 		return ((this.cachestore ?? {})[this[this.cache_id]] ?? {}).value }, | 
					
						
							| 
									
										
										
										
											2022-08-22 19:23:06 +03:00
										 |  |  | 	// XXX check * paths for matches...
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 	set cache(value){ | 
					
						
							|  |  |  | 		if(this.cachestore === false  | 
					
						
							|  |  |  | 				|| this.cache == value){ | 
					
						
							|  |  |  | 			return } | 
					
						
							| 
									
										
										
										
											2022-09-01 00:04:13 +03:00
										 |  |  | 		var id = this[this.cache_id ?? 'location'] | 
					
						
							| 
									
										
										
										
											2022-08-22 19:23:06 +03:00
										 |  |  | 		// clear...
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 		if(value == null){ | 
					
						
							| 
									
										
										
										
											2022-09-01 00:04:13 +03:00
										 |  |  | 			delete (this.cachestore ?? {})[id] | 
					
						
							| 
									
										
										
										
											2022-08-22 19:23:06 +03:00
										 |  |  | 		// set...
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2022-09-01 00:04:13 +03:00
										 |  |  | 			var prev = ((this.cachestore = this.cachestore ?? {})[id] ?? {}).value ?? {} | 
					
						
							|  |  |  | 			;(this.cachestore = this.cachestore ?? {})[id] = { | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 				created: Date.now(), | 
					
						
							|  |  |  | 				// XXX
 | 
					
						
							|  |  |  | 				valid: undefined, | 
					
						
							| 
									
										
										
										
											2022-08-22 19:23:06 +03:00
										 |  |  | 				value: { | 
					
						
							|  |  |  | 					...prev,  | 
					
						
							|  |  |  | 					...value, | 
					
						
							|  |  |  | 				}, | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 			} }  | 
					
						
							|  |  |  | 		// clear depended pages from cache...
 | 
					
						
							|  |  |  | 		for(var [key, deps] of Object.entries(this.dependencies)){ | 
					
						
							| 
									
										
										
										
											2022-08-22 19:23:06 +03:00
										 |  |  | 			// XXX also check pattern paths...
 | 
					
						
							|  |  |  | 			// 		...the problem here is that it's getting probabilistic, 
 | 
					
						
							|  |  |  | 			// 		i.e. if we match * as a single path item then we might 
 | 
					
						
							|  |  |  | 			// 		miss creating a subtree (ex: /tree), while matching 
 | 
					
						
							|  |  |  | 			// 		/* to anything will give us lots of false positives...
 | 
					
						
							| 
									
										
										
										
											2022-09-01 00:04:13 +03:00
										 |  |  | 			if(key != id && deps.has(id)){ | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 				delete this.cachestore[key] } } }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 13:24:26 +03:00
										 |  |  | 	checkCache: function(...paths){ | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 		if(!this.cache_timeout || !this.cachestore){ | 
					
						
							|  |  |  | 			return this } | 
					
						
							|  |  |  | 		paths = paths.length == 0 ? | 
					
						
							|  |  |  | 			Object.keys(this.cachestore) | 
					
						
							|  |  |  | 			: paths | 
					
						
							|  |  |  | 		for(var path of paths){ | 
					
						
							|  |  |  | 			var {created, valid, value} = this.cachestore[path] ?? {} | 
					
						
							|  |  |  | 			if(value){ | 
					
						
							|  |  |  | 				var now = Date.now() | 
					
						
							|  |  |  | 				valid = valid  | 
					
						
							|  |  |  | 					?? Date.str2ms(this.cache_timeout) | 
					
						
							|  |  |  | 				// drop cache...
 | 
					
						
							|  |  |  | 				if(now > created + valid){ | 
					
						
							|  |  |  | 					//console.log('CACHE: DROP:', this.path)
 | 
					
						
							|  |  |  | 					delete this.cachestore[path] } } } | 
					
						
							|  |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2022-08-22 19:23:06 +03:00
										 |  |  | 	clearCache: function(...paths){ | 
					
						
							|  |  |  | 		if(this.cachestore){ | 
					
						
							|  |  |  | 			if(arguments.length == 0){ | 
					
						
							|  |  |  | 				this.cachestore = null  | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				for(var path of paths){ | 
					
						
							|  |  |  | 					delete this.cachestore[path] } } } | 
					
						
							|  |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 13:24:26 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 	__update__: function(){ | 
					
						
							|  |  |  | 		this.cache = null | 
					
						
							|  |  |  | 		return object.parentCall(CachedPage.prototype.__update__, this, ...arguments) }, | 
					
						
							|  |  |  | 	__delete__: function(){ | 
					
						
							|  |  |  | 		this.cache = null | 
					
						
							|  |  |  | 		return object.parentCall(CachedPage.prototype.__delete__, this, ...arguments) }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-24 23:47:42 +03:00
										 |  |  | 	/* XXX do we need to cache .raw??? | 
					
						
							|  |  |  | 	//		...yes this makes things marginally faster but essentially 
 | 
					
						
							|  |  |  | 	//		copies the db into memory...
 | 
					
						
							|  |  |  | 	get raw(){ | 
					
						
							|  |  |  | 		return getCachedProp(this, 'raw') }, | 
					
						
							|  |  |  | 	set raw(value){ | 
					
						
							|  |  |  | 		return setCachedProp(this, 'raw', value) }, | 
					
						
							|  |  |  | 	//*/
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-24 23:47:42 +03:00
										 |  |  | 	get text(){ | 
					
						
							|  |  |  | 		return getCachedProp(this, 'text') }, | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 	set text(value){ | 
					
						
							| 
									
										
										
										
											2022-08-24 23:47:42 +03:00
										 |  |  | 		return setCachedProp(this, 'text', value) }, | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-06 11:09:25 +03:00
										 |  |  | //---------------------------------------------------------------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | var wikiword = require('./dom/wikiword') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var pWikiPageElement = | 
					
						
							|  |  |  | module.pWikiPageElement =  | 
					
						
							| 
									
										
										
										
											2022-08-24 23:13:24 +03:00
										 |  |  | // XXX CACHE...
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | object.Constructor('pWikiPageElement', CachedPage, { | 
					
						
							| 
									
										
										
										
											2022-08-24 23:13:24 +03:00
										 |  |  | /*/ | 
					
						
							|  |  |  | object.Constructor('pWikiPageElement', Page, { | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | //*/
 | 
					
						
							| 
									
										
										
										
											2022-08-06 11:09:25 +03:00
										 |  |  | 	dom: undefined, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-11 10:39:55 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | 	domFilters: { | 
					
						
							|  |  |  | 		// XXX see Page.filters.wikiword for notes...
 | 
					
						
							|  |  |  | 		wikiword: wikiword.wikiWordText, | 
					
						
							| 
									
										
										
										
											2022-08-08 12:29:02 +03:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-24 23:13:24 +03:00
										 |  |  | 	// XXX CACHE
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 	__clone_constructor__: CachedPage, | 
					
						
							| 
									
										
										
										
											2022-08-24 23:13:24 +03:00
										 |  |  | 	/*/ | 
					
						
							|  |  |  | 	__clone_constructor__: Page, | 
					
						
							|  |  |  | 	//*/
 | 
					
						
							| 
									
										
										
										
											2022-08-12 01:38:32 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	__clone_proto: undefined, | 
					
						
							|  |  |  | 	get __clone_proto__(){ | 
					
						
							|  |  |  | 		return (this.__clone_proto = this.__clone_proto  | 
					
						
							|  |  |  | 			?? this.__clone_constructor__('/', '/', this.store)) }, | 
					
						
							|  |  |  | 	set __clone_proto__(value){ | 
					
						
							|  |  |  | 		this.__clone_proto = value }, | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2022-09-02 14:58:45 +03:00
										 |  |  | 	actions: new Set([ | 
					
						
							|  |  |  | 		...CachedPage.prototype.actions, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		'title', | 
					
						
							|  |  |  | 		'hash' | 
					
						
							|  |  |  | 	]), | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-11 10:39:55 +03:00
										 |  |  | 	// NOTE: setting location will reset .hash set it directly via either
 | 
					
						
							|  |  |  | 	// 		one of:
 | 
					
						
							|  |  |  | 	// 			.location = [path, hash]
 | 
					
						
							|  |  |  | 	// 			.location = 'path#hash'
 | 
					
						
							|  |  |  | 	hash: undefined, | 
					
						
							|  |  |  | 	// NOTE: getting .location will not return the hash, so as not to force
 | 
					
						
							|  |  |  | 	// 		the user to parse it out each time.
 | 
					
						
							|  |  |  | 	get location(){ | 
					
						
							|  |  |  | 		return object.parentProperty(pWikiPageElement.prototype, 'location') | 
					
						
							|  |  |  | 			.get.call(this) }, | 
					
						
							|  |  |  | 	set location(value){ | 
					
						
							|  |  |  | 		var [value, hash] =  | 
					
						
							|  |  |  | 			// .location = [path, hash]
 | 
					
						
							|  |  |  | 			value instanceof Array ? | 
					
						
							|  |  |  | 				value | 
					
						
							|  |  |  | 			// .location = '<path>#<hash>'
 | 
					
						
							|  |  |  | 			: value.includes('#') ? | 
					
						
							|  |  |  | 				value.split('#') | 
					
						
							|  |  |  | 			// no hash is given...
 | 
					
						
							|  |  |  | 			: [value, undefined] | 
					
						
							|  |  |  | 		this.hash = hash | 
					
						
							|  |  |  | 		object.parentProperty(pWikiPageElement.prototype, 'location') | 
					
						
							|  |  |  | 			.set.call(this, value) }, | 
					
						
							| 
									
										
										
										
											2022-08-08 23:04:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-11 10:39:55 +03:00
										 |  |  | 	// XXX this is not persistent, is this what we want???
 | 
					
						
							| 
									
										
										
										
											2022-09-02 14:58:45 +03:00
										 |  |  | 	// XXX should this default to .path or to .name???
 | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | 	get title(){ | 
					
						
							|  |  |  | 		return this.dom.getAttribute('title') | 
					
						
							|  |  |  | 			|| (this.dom.querySelector('h1') || {}).innerText | 
					
						
							|  |  |  | 			|| this.path }, | 
					
						
							|  |  |  | 	set title(value){ | 
					
						
							|  |  |  | 		this.dom.setAttribute('title', value) }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-08 12:29:02 +03:00
										 |  |  | 	// events...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | 	__pWikiLoadedDOMEvent: new Event('pwikiloaded'), | 
					
						
							|  |  |  | 	onLoad: types.event.Event('onLoad', function(){ | 
					
						
							|  |  |  | 		this.dom.dispatchEvent(this.__pWikiLoadedDOMEvent) }), | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 	// XXX CACHE...
 | 
					
						
							|  |  |  | 	__last_refresh_path: undefined, | 
					
						
							|  |  |  | 	refresh: async function(full=false){ | 
					
						
							|  |  |  | 		// drop cache if re-refreshing or when full refresh requested...
 | 
					
						
							|  |  |  | 		// XXX CACHE...
 | 
					
						
							|  |  |  | 		;(full | 
					
						
							|  |  |  | 				|| this.__last_refresh_path == this.path)	 | 
					
						
							|  |  |  | 			&& this.cache  | 
					
						
							|  |  |  | 			&& (this.cache = null) | 
					
						
							|  |  |  | 		this.__last_refresh_path = this.path | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | 		var dom = this.dom | 
					
						
							|  |  |  | 		dom.innerHTML = await this.text  | 
					
						
							|  |  |  | 		for(var filter of Object.values(this.domFilters)){ | 
					
						
							|  |  |  | 			filter | 
					
						
							|  |  |  | 				&& filter.call(this, dom) } | 
					
						
							|  |  |  | 		this.onLoad() | 
					
						
							|  |  |  | 		return this }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-11 10:39:55 +03:00
										 |  |  | 	// handle dom as first argument...
 | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | 	__init__: function(dom, ...args){ | 
					
						
							| 
									
										
										
										
											2022-08-11 23:56:43 +03:00
										 |  |  | 		if(typeof(Element) != 'undefined'){ | 
					
						
							|  |  |  | 			if(dom instanceof Element){ | 
					
						
							|  |  |  | 				this.dom = dom | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				args.unshift(dom) } } | 
					
						
							| 
									
										
										
										
											2022-08-10 11:42:31 +03:00
										 |  |  | 		return object.parentCall(pWikiPageElement.prototype.__init__, this, ...args) }, | 
					
						
							| 
									
										
										
										
											2022-08-06 11:09:25 +03:00
										 |  |  | }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | //---------------------------------------------------------------------
 | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | // System pages/actions...
 | 
					
						
							| 
									
										
										
										
											2022-08-22 00:30:14 +03:00
										 |  |  | // XXX move this to a more appropriate module...
 | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | var System =  | 
					
						
							|  |  |  | module.System = { | 
					
						
							|  |  |  | 	// base templates...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-05 18:21:18 +03:00
										 |  |  | 	// These are used to control how a page is rendered. 
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// pWiki has to have a template appended to any path, if one is not 
 | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 	// given then "_view" is used internally.
 | 
					
						
							| 
									
										
										
										
											2022-08-05 18:21:18 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	// A template is rendered in the context of the parent page, e.g. 
 | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 	// for /path/to/page, the actual rendered template is /path/to/page/_view
 | 
					
						
							| 
									
										
										
										
											2022-08-05 18:21:18 +03:00
										 |  |  | 	// and it is rendered from /path/to/page.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// A template is any page named starting with an underscore ("_") 
 | 
					
						
							|  |  |  | 	// thus it is not recommended to use underscores to start page names.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// The actual default template is controlled via <page>.PAGE_TEMPLATE
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// Example:
 | 
					
						
							|  |  |  | 	// 		_list: {
 | 
					
						
							|  |  |  | 	//			text: '<macro src="." join="\n">- @source(.)</macro>' },
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 	// XXX all of these should support pattern pages...
 | 
					
						
							| 
									
										
										
										
											2022-08-04 19:47:08 +03:00
										 |  |  | 	_text: { | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 		text: '@include(. isolated join="@source(file-separator)")' }, | 
					
						
							| 
									
										
										
										
											2022-08-28 16:34:46 +03:00
										 |  |  | 	// XXX /rootpath here is not relative -- makes reuse harder...
 | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 	_view: { | 
					
						
							| 
									
										
										
										
											2022-08-17 14:10:25 +03:00
										 |  |  | 		text: object.doc`
 | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 			<slot name="header"> | 
					
						
							| 
									
										
										
										
											2022-08-30 10:58:04 +03:00
										 |  |  | 				<a href="#/list">☰</a> | 
					
						
							|  |  |  | 				@source(./path/!)  | 
					
						
							|  |  |  | 				<a href="#@source(./path/!)/_edit">(edit)</a> | 
					
						
							| 
									
										
										
										
											2022-08-22 10:07:22 +03:00
										 |  |  | 			</slot> | 
					
						
							| 
									
										
										
										
											2022-08-17 14:10:25 +03:00
										 |  |  | 			<hr> | 
					
						
							|  |  |  | 			<slot name="content"></slot> | 
					
						
							|  |  |  | 			<hr> | 
					
						
							|  |  |  | 			<slot name="footer"></slot> | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-29 18:22:35 +03:00
										 |  |  | 			<!-- fill slots defaults --> | 
					
						
							| 
									
										
										
										
											2022-08-17 15:45:30 +03:00
										 |  |  | 			<slot name="content" hidden> | 
					
						
							|  |  |  | 				@include(. join="@source(file-separator)" recursive="") | 
					
						
							| 
									
										
										
										
											2022-08-17 14:10:25 +03:00
										 |  |  | 			</slot>` }, | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 	// XXX add join...
 | 
					
						
							| 
									
										
										
										
											2022-08-05 12:46:00 +03:00
										 |  |  | 	_raw: { | 
					
						
							|  |  |  | 		text: '@quote(.)' }, | 
					
						
							| 
									
										
										
										
											2022-08-05 02:47:31 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-14 02:08:27 +03:00
										 |  |  | 	// XXX not sure if this is the right way to go...
 | 
					
						
							|  |  |  | 	_code: { | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 		text:  | 
					
						
							|  |  |  | 			'<macro src="." join="@source(file-separator)">' | 
					
						
							|  |  |  | 				+'<pre wikiwords="no"><quote filter="quote-tags" src="."/></pre>'  | 
					
						
							|  |  |  | 			+'</macro>'}, | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 	/* XXX can we reuse _view here??? | 
					
						
							|  |  |  | 	_edit: { | 
					
						
							|  |  |  | 		text:  | 
					
						
							|  |  |  | 			'@include(PageTemplate)' | 
					
						
							|  |  |  | 			+'<slot name="header">@source(./path)</slot>' | 
					
						
							|  |  |  | 			+'<slot name="content">' | 
					
						
							|  |  |  | 				+'<macro src="." join="@source(file-separator)">' | 
					
						
							|  |  |  | 					+'<pre class="editor" ' | 
					
						
							|  |  |  | 							+'wikiwords="no" ' | 
					
						
							|  |  |  | 							+'contenteditable ' | 
					
						
							|  |  |  | 							+'oninput="saveContent(\'@source(./path)\', this.innerText)">' | 
					
						
							|  |  |  | 						+'<quote filter="quote-tags" src="."/>' | 
					
						
							|  |  |  | 					+'</pre>'  | 
					
						
							|  |  |  | 				+'</macro>' | 
					
						
							|  |  |  | 			+'</slot>'}, | 
					
						
							|  |  |  | 	/*/ | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 	_edit: { | 
					
						
							| 
									
										
										
										
											2022-08-14 03:00:06 +03:00
										 |  |  | 		text:  | 
					
						
							| 
									
										
										
										
											2022-08-17 14:10:25 +03:00
										 |  |  | 			'@source(./path)' | 
					
						
							|  |  |  | 			+'<hr>' | 
					
						
							|  |  |  | 			+'<macro src="." join="@source(file-separator)">' | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 				+'<pre class="editor" ' | 
					
						
							|  |  |  | 						+'wikiwords="no" ' | 
					
						
							|  |  |  | 						+'contenteditable ' | 
					
						
							|  |  |  | 						+'oninput="saveContent(\'@source(./path)\', this.innerText)">' | 
					
						
							|  |  |  | 					+'<quote filter="quote-tags" src="."/>' | 
					
						
							|  |  |  | 				+'</pre>'  | 
					
						
							|  |  |  | 			+'</macro>'}, | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 	//*/
 | 
					
						
							|  |  |  | 	edit: { | 
					
						
							| 
									
										
										
										
											2022-08-22 00:30:14 +03:00
										 |  |  | 		text: object.doc`
 | 
					
						
							|  |  |  | 			@include(_view) | 
					
						
							|  |  |  | 			<slot name="header">@source(../path)</slot> | 
					
						
							|  |  |  | 			<slot name="content"> | 
					
						
							|  |  |  | 				<macro src=".." join="@source(file-separator)"> | 
					
						
							|  |  |  | 					<pre class="editor" | 
					
						
							|  |  |  | 							wikiwords="no" | 
					
						
							|  |  |  | 							contenteditable | 
					
						
							|  |  |  | 							oninput="saveContent(\'@source(./path)\', this.innerText)" | 
					
						
							|  |  |  | 					><quote filter="quote-tags" src="."/></pre>  | 
					
						
							|  |  |  | 				</macro> | 
					
						
							|  |  |  | 			</slot>`}, | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-17 15:45:30 +03:00
										 |  |  | 	// XXX this does not yet work...
 | 
					
						
							|  |  |  | 	// XXX "_test" breaks differently than "test"
 | 
					
						
							|  |  |  | 	//_test: {
 | 
					
						
							|  |  |  | 	test: { | 
					
						
							|  |  |  | 		text: object.doc`
 | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 			@source(_view) | 
					
						
							| 
									
										
										
										
											2022-08-17 15:45:30 +03:00
										 |  |  | 			<slot name="header">HEADER</slot> | 
					
						
							|  |  |  | 			<slot name="content">CONTENT</slot> | 
					
						
							|  |  |  | 			<slot name="footer">FOOTER</slot> `}, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// XXX debug...
 | 
					
						
							|  |  |  | 	_path: {text: '@source(./path join=" ")'}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list: { | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 		text: object.doc`
 | 
					
						
							|  |  |  | 			<slot name="header"> | 
					
						
							| 
									
										
										
										
											2022-08-30 10:58:04 +03:00
										 |  |  | 				<a href="#/list">☰</a> | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 				<a href="#@source(../../path)/list">⇑</a> | 
					
						
							|  |  |  | 				@source(../path) | 
					
						
							|  |  |  | 			</slot> | 
					
						
							|  |  |  | 			<macro src="../*" join="@source(line-separator)"> | 
					
						
							| 
									
										
										
										
											2022-08-29 18:22:35 +03:00
										 |  |  | 				<a href="#@source(./path)">@source(./name)</a> | 
					
						
							|  |  |  | 				<sup> | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 					<macro src="./isAction"> | 
					
						
							|  |  |  | 						a | 
					
						
							|  |  |  | 						<else> | 
					
						
							|  |  |  | 							<macro src="./isStore">s</macro> | 
					
						
							|  |  |  | 						</else> | 
					
						
							|  |  |  | 					</macro> | 
					
						
							| 
									
										
										
										
											2022-08-29 18:22:35 +03:00
										 |  |  | 				</sup> | 
					
						
							| 
									
										
										
										
											2022-09-02 14:58:45 +03:00
										 |  |  | 				(<a href="#@source(./path)/list">@include(./*/length/!)</a>) | 
					
						
							| 
									
										
										
										
											2022-08-29 18:22:35 +03:00
										 |  |  | 				  | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 				<a href="#@source(./path)/delete">×</a> | 
					
						
							|  |  |  | 			</macro>` }, | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 	// XXX this is really slow...
 | 
					
						
							|  |  |  | 	tree: { | 
					
						
							|  |  |  | 		text: object.doc`
 | 
					
						
							|  |  |  | 			<macro src="../*"> | 
					
						
							| 
									
										
										
										
											2022-09-01 00:04:13 +03:00
										 |  |  | 				<div> | 
					
						
							|  |  |  | 					<div class="item"> | 
					
						
							|  |  |  | 						<a href="#@source(./path)">@source(./name)</a> | 
					
						
							|  |  |  | 						<a class="show-on-hover" href="#@source(./path)/info">🛈</a> | 
					
						
							|  |  |  | 						<a class="show-on-hover" href="#@source(./path)/delete">×</a> | 
					
						
							|  |  |  | 					</div> | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 					<div style="padding-left: 30px"> | 
					
						
							|  |  |  | 						@source(./tree) | 
					
						
							|  |  |  | 					</div> | 
					
						
							|  |  |  | 				</div> | 
					
						
							|  |  |  | 			</macro>` }, | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 	all: { | 
					
						
							| 
									
										
										
										
											2022-08-19 20:31:37 +03:00
										 |  |  | 		text: `@include(../**/path join="@source(line-separator)")`}, | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 	info: { | 
					
						
							|  |  |  | 		text: object.doc`
 | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 			Path: @source(../path)  | 
					
						
							|  |  |  | 				(<a href="#@source(../path)/edit">edit</a>)<br> | 
					
						
							|  |  |  | 			Resolved path: @source(../resolved)  | 
					
						
							|  |  |  | 				(<a href="#@source(../resolved)/edit">edit</a>)<br> | 
					
						
							|  |  |  | 			Referrer: @source(../referrer) | 
					
						
							|  |  |  | 				(<a href="#@source(../referrer)/edit">edit</a>)<br> | 
					
						
							|  |  |  | 			Renderer: @source(../renderer) | 
					
						
							|  |  |  | 				(<a href="#@source(../renderer)/edit">edit</a>)<br> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			type: @source(../type)<br> | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 			ctime: @source(../ctime)<br> | 
					
						
							|  |  |  | 			mtime: @source(../mtime)<br> | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 			<hr> | 
					
						
							|  |  |  | 			<pre wikiwords="no"><quote filter="quote-tags" src=".."/></pre> ` }, | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-26 18:16:07 +03:00
										 |  |  | 	// XXX need to also be able to list things about each store...
 | 
					
						
							|  |  |  | 	stores: function(){ | 
					
						
							|  |  |  | 		return Object.keys(this.store.substores ?? {}) }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-14 23:54:02 +03:00
										 |  |  | 	// page parts...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	'line-separator': { text: '<br>' }, | 
					
						
							|  |  |  | 	'file-separator': { text: '<hr>' }, | 
					
						
							| 
									
										
										
										
											2022-08-05 18:21:18 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// base system pages...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: these are last resort pages, preferably overloaded in /Templates.
 | 
					
						
							| 
									
										
										
										
											2022-08-26 18:16:07 +03:00
										 |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-05 02:47:31 +03:00
										 |  |  | 	RecursionError: { | 
					
						
							| 
									
										
										
										
											2022-08-27 16:17:25 +03:00
										 |  |  | 		text: 'RECURSION ERROR: @source(../path)' }, | 
					
						
							| 
									
										
										
										
											2022-08-05 02:47:31 +03:00
										 |  |  | 	NotFoundError: {  | 
					
						
							| 
									
										
										
										
											2022-08-27 16:17:25 +03:00
										 |  |  | 		text: 'NOT FOUND ERROR: @source(./path)' }, | 
					
						
							| 
									
										
										
										
											2022-08-18 01:23:49 +03:00
										 |  |  | 	NotFoundTemplateError: { | 
					
						
							| 
									
										
										
										
											2022-08-27 16:17:25 +03:00
										 |  |  | 		text: 'NOT FOUND TEMPLATE ERROR: @source(../path)' }, | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 	DeletingPage: { | 
					
						
							|  |  |  | 		text: 'Deleting: @source(../path)' }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 	PageTemplate: { | 
					
						
							|  |  |  | 		text: object.doc`
 | 
					
						
							|  |  |  | 			<slot name="header">@source(./path)/_edit</slot> | 
					
						
							|  |  |  | 			<hr> | 
					
						
							|  |  |  | 			<slot name="content"></slot> | 
					
						
							|  |  |  | 			<hr> | 
					
						
							|  |  |  | 			<slot name="footer"></slot> ` }, | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 	QuoteActionPage: { | 
					
						
							|  |  |  | 		text: '[ native code ]' }, | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-05 18:21:18 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// page actions...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | 	// metadata...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2022-08-21 20:45:29 +03:00
										 |  |  | 	ctime: async function(){ | 
					
						
							| 
									
										
										
										
											2022-08-22 00:30:14 +03:00
										 |  |  | 		var date = (await this.get('..').data).ctime  | 
					
						
							|  |  |  | 		return date ? | 
					
						
							|  |  |  | 			(new Date(date)).getTimeStamp() | 
					
						
							|  |  |  | 			: date }, | 
					
						
							| 
									
										
										
										
											2022-08-21 20:45:29 +03:00
										 |  |  | 	mtime: async function(){ | 
					
						
							| 
									
										
										
										
											2022-08-22 00:30:14 +03:00
										 |  |  | 		var date = (await this.get('..').data).mtime  | 
					
						
							|  |  |  | 		return date ? | 
					
						
							|  |  |  | 			(new Date(date)).getTimeStamp() | 
					
						
							|  |  |  | 			: date }, | 
					
						
							| 
									
										
										
										
											2022-08-21 20:45:29 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-30 10:58:04 +03:00
										 |  |  | 	// XXX ENERGETIC -- a better name???
 | 
					
						
							|  |  |  | 	// XXX test this with pages...
 | 
					
						
							|  |  |  | 	'!': Object.assign( | 
					
						
							|  |  |  | 		async function(){ | 
					
						
							|  |  |  | 			return this.get('..', {energetic: true}).raw }, | 
					
						
							|  |  |  | 		{energetic: true}), | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 00:30:14 +03:00
										 |  |  | 	// XXX EXPERIMENTAL -- page types...
 | 
					
						
							| 
									
										
										
										
											2022-08-21 20:45:29 +03:00
										 |  |  | 	isAction: async function(){ | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 		return await this.get('..').type == 'action' ? | 
					
						
							|  |  |  | 			'action' | 
					
						
							|  |  |  | 			: undefined }, | 
					
						
							|  |  |  | 	isStore: async function(){ | 
					
						
							|  |  |  | 		return await this.get('..').type == 'store' ? | 
					
						
							|  |  |  | 			'store' | 
					
						
							| 
									
										
										
										
											2022-08-21 20:45:29 +03:00
										 |  |  | 			: undefined }, | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// utils...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX System/subpaths
 | 
					
						
							|  |  |  | 	// XXX
 | 
					
						
							|  |  |  | 	links: function(){ | 
					
						
							|  |  |  | 		// XXX
 | 
					
						
							|  |  |  | 		return '' }, | 
					
						
							|  |  |  | 	// XXX links to pages...
 | 
					
						
							| 
									
										
										
										
											2022-08-20 09:37:05 +03:00
										 |  |  | 	LinksTo: function(){ | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | 		return (this.get('..').data || {}).to ?? [] }, | 
					
						
							|  |  |  | 	// XXX pages linking to us...
 | 
					
						
							| 
									
										
										
										
											2022-08-20 09:37:05 +03:00
										 |  |  | 	LinksFrom: function(){ | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | 		return (this.get('..').data || {})['from'] ?? [] }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// actions...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	delete: function(){ | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 		var target = this.get('..') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-21 23:07:47 +03:00
										 |  |  | 		console.log('DELETE:', target.path) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-15 14:29:45 +03:00
										 |  |  | 		target.delete() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// redirect...
 | 
					
						
							|  |  |  | 		this.render_root | 
					
						
							|  |  |  | 			&& (this.render_root.location = this.referrer) | 
					
						
							|  |  |  | 		// show info about the delete operation...
 | 
					
						
							| 
									
										
										
										
											2022-08-22 19:23:06 +03:00
										 |  |  | 		return target.get('DeletingPage/_text').text }, | 
					
						
							| 
									
										
										
										
											2022-08-20 09:37:05 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-22 19:23:06 +03:00
										 |  |  | 	// XXX copy/move/...
 | 
					
						
							|  |  |  | 	// 		...need arguments
 | 
					
						
							| 
									
										
										
										
											2022-08-21 12:31:04 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | 	// XXX System/back
 | 
					
						
							|  |  |  | 	// XXX System/forward
 | 
					
						
							|  |  |  | 	// XXX System/sort
 | 
					
						
							|  |  |  | 	// XXX System/reverse
 | 
					
						
							| 
									
										
										
										
											2022-08-17 23:38:24 +03:00
										 |  |  | 	 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-20 09:37:05 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var Test = | 
					
						
							|  |  |  | module.Test = { | 
					
						
							|  |  |  | 	'list/action': function(){ | 
					
						
							|  |  |  | 		return [...'abcdef'] }, | 
					
						
							|  |  |  | 	'list/statuc': { | 
					
						
							|  |  |  | 		text: [...'abcdef'] }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	slots: { | 
					
						
							| 
									
										
										
										
											2022-08-18 01:23:49 +03:00
										 |  |  | 		text: object.doc`
 | 
					
						
							|  |  |  | 			Sequential: | 
					
						
							|  |  |  | 			<slot name="sequential">unfilled</slot> | 
					
						
							|  |  |  | 			<slot name="sequential">filled</slot> | 
					
						
							|  |  |  | 			<slot name="sequential">refilled</slot>  | 
					
						
							|  |  |  | 			<br><br> | 
					
						
							|  |  |  | 			Nested: | 
					
						
							|  |  |  | 			<slot name="nested"> | 
					
						
							|  |  |  | 				unfilled | 
					
						
							|  |  |  | 				<slot name="nested"> | 
					
						
							|  |  |  | 					filled | 
					
						
							|  |  |  | 					<slot name="nested"> | 
					
						
							|  |  |  | 						refilled | 
					
						
							|  |  |  | 					</slot> | 
					
						
							|  |  |  | 				</slot> | 
					
						
							|  |  |  | 			</slot>  | 
					
						
							|  |  |  | 			<br><br> | 
					
						
							|  |  |  | 			Content: A B C: | 
					
						
							|  |  |  | 			<slot name="slot-content">A</slot> | 
					
						
							|  |  |  | 			<slot name="slot-content"><content/> B</slot> | 
					
						
							|  |  |  | 			<slot name="slot-conten"><content/> C</slot> | 
					
						
							|  |  |  | 			<br><br> | 
					
						
							| 
									
										
										
										
											2022-08-18 20:34:13 +03:00
										 |  |  | 			Nested content: A B C: | 
					
						
							|  |  |  | 			<slot name="nested-slot-content"> | 
					
						
							|  |  |  | 				A | 
					
						
							|  |  |  | 				<slot name="nested-slot-content"> | 
					
						
							|  |  |  | 					<content/> B | 
					
						
							|  |  |  | 					<slot name="nested-slot-content"> | 
					
						
							|  |  |  | 						<content/> C | 
					
						
							|  |  |  | 					</slot> | 
					
						
							|  |  |  | 				</slot> | 
					
						
							| 
									
										
										
										
											2022-08-19 13:13:17 +03:00
										 |  |  | 			</slot> | 
					
						
							|  |  |  | 			<br><br> | 
					
						
							|  |  |  | 			Mixed content: X A B C Z: | 
					
						
							|  |  |  | 			<slot name="mixed-slot-content"> | 
					
						
							|  |  |  | 				X | 
					
						
							|  |  |  | 			</slot> | 
					
						
							|  |  |  | 			<slot name="mixed-slot-content"> | 
					
						
							|  |  |  | 				<content/> A | 
					
						
							|  |  |  | 				<slot name="mixed-slot-content"> | 
					
						
							| 
									
										
										
										
											2022-08-18 01:23:49 +03:00
										 |  |  | 					<content/> B | 
					
						
							|  |  |  | 				</slot> | 
					
						
							| 
									
										
										
										
											2022-08-19 13:13:17 +03:00
										 |  |  | 				<slot name="mixed-slot-content"> | 
					
						
							|  |  |  | 					<content/> C | 
					
						
							|  |  |  | 				</slot> | 
					
						
							|  |  |  | 			</slot>  | 
					
						
							|  |  |  | 			<slot name="mixed-slot-content"> | 
					
						
							|  |  |  | 				<content/> Z | 
					
						
							| 
									
										
										
										
											2022-08-18 01:23:49 +03:00
										 |  |  | 			</slot> ` }, | 
					
						
							| 
									
										
										
										
											2022-08-21 01:28:29 +03:00
										 |  |  | 	macros: { | 
					
						
							|  |  |  | 		text: object.doc`
 | 
					
						
							|  |  |  | 			<macro name="list" join="<br>"> | 
					
						
							|  |  |  | 				- @include(./path) | 
					
						
							|  |  |  | 			</macro> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			<macro name="list" src="/Test/*"/> | 
					
						
							|  |  |  | 			<br><br> | 
					
						
							|  |  |  | 			<macro name="list" src="/Test/*" join=",<br>"/> | 
					
						
							|  |  |  | 		`},
 | 
					
						
							| 
									
										
										
										
											2022-08-04 14:29:25 +03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-14 03:00:06 +03:00
										 |  |  | var Settings = | 
					
						
							|  |  |  | module.Settings = { | 
					
						
							|  |  |  | 	Export: { | 
					
						
							|  |  |  | 		text: '<button onclick="exportData()">Export</button>' }, | 
					
						
							|  |  |  | 	// XXX
 | 
					
						
							| 
									
										
										
										
											2022-08-14 03:10:24 +03:00
										 |  |  | 	Config: { | 
					
						
							| 
									
										
										
										
											2022-08-14 03:00:06 +03:00
										 |  |  | 		text: '{}' }, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-06 11:09:25 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 11:00:21 +03:00
										 |  |  | /********************************************************************** | 
					
						
							|  |  |  | * vim:set ts=4 sw=4 nowrap :                        */ return module }) |