Skip to content

Latest commit

 

History

History
 
 

view

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 
 
 
 
 
 
 
 
 
 
 

$mol_view

The base class for all visual components. It provides the infrastructure for reactive lazy rendering, handling exceptions. By default it finds or creates a div without child node changing and additional attributes, fields and event handler creation. You can customize it by inheritance or properties overriding at instantiating.

Properties

dom_name()' : string

Returns name of the DOM-element creating for component, if the element with appropriate id is not presented at DOM yet.

dom_name_space() = 'http://www.w3.org/1999/xhtml'

Returns namespaceURI for the DOM element.

sub() : Array< $mol_view | Node | string | number | boolean > = null

Returns list of child components/elements/primitives. If the list have not been set (by default), then the content of the DOM-element would not be changed in way, it's helpful for manual operating with DOM.

context( next? : $ ) : $ Some rendering context. Parent node injects context to all rendered child components.

minimal_height() = 0

Returns minimum possible height of the component. It's set by hand with constant or some expression.This property is used for lazy rendering.

dom_node() : Element

Returns DOM-element, to which the component is bounded to. At first the method tries to find the element by its id at DOM and only if it would have not been found - the method would create and remember a new one.

dom_tree() : Element

Same as dom_node, but its guarantee, that the content, attributes and properties of the DOM-element should be in actual state.

attr() : { [ key : string ] : string | number | boolean }

Returns the dictionary of the DOM-attributes, which values would be set while rendering. Passing null or false as the value to the attribute would lead to removing the attribute. Passing true is an equivalent to passing its name as value. undefined is just ignored.

field() : { [ key : string ] : any }

Returns dictionary of fields, which is necessary to set to the DOM-element after rendering.

style() : { [ key : string ] : string | number }

Returns dictionary of styles. Numbers will be converted to string with "px" suffix.

event() : { [ key : string ] : ( event : Event )=> void }

Returns dictionary of event handlers. The event handlers are bind to the DOM-element one time, when the value is set to dom_node property. This handlers are synchronous and can be cancelled by preventDefault().

focused( next? : boolean ) : boolean

Determines, whether the component is focused or not at this time. If any inserted component would be focused, then its parent component would be focused also.

plugins() : Array< $mol_view > = null

Array of plugins. Plugin is a component which can be supplemented with the logic of the current components.

For example, list component with keyboard navigation (used $mol_nav plugin):

<= Options $mol_list
    plugins / 
        <= Nav $mol_nav
            keys_y <= options /
    rows <= options /

*.view.tree

view.tree - is a declarative language of describing components, based on tree format. One file can have multiple component definitions, but better to put every component in a separate file, except in very trivial cases. To create a new component in view.tree file you must inherit it from any existing one or $mol_view. Name of the component should begin with $ and be unique globally accordance with principles presented on MAM. For example, let's declare the component $my_button extended from $mol_view:

$my_button $mol_view

It translates to (every *.view.tree code would be translated to *.view.tree.ts):

namespace $ { export class $my_button extends $mol_view {} }

When inheriting, it is possible to declare additional properties or overload existing ones (but the property type must match). For example, lets overload a uri property with "https://example.org" string, and sub - with array of one string "Click me!", besides, lets declare a new property target with "_top" value by default (default value is necessary when declaring a new property):

$my_example $mol_link
	uri \https://example.org
	sub /
		\Click me!
	target \_top
namespace $ { export class $my_example extends $mol_link {

	uri() { return "https://example.org" }

	sub() { return [ "Click me!" ] }

	target() { return "_top" }
} }

Note: For better readability, a single child node in a tree file is often written on a single line. You can expand all nodes in the previous example:

$my_example
	$mol_link
		uri
			\https://example.org
		sub
			/
				\Click me!
		target
			\_top

Node where name starts with $ - name of component. Child nodes beginning with node / - list. You can set type of list, e.g /number, /$mol_view for better type checking. Text after \ - raw data which can contain entirely any data until the end of the line. Node @ marks string for extraction to separate *.locale=en.json file and used for i18n translation, e.g @ \Values example. Numbers, booleans values and null is being wrote as it is, without any prefixes: Nodes after - are ignored, you can use them for commenting and temporary disable subtree.

$my_values $mol_view
	title @ \Values example
	sub /
		0
		1.1
		true
		false
		null
		\I can contain any character! \("o")/
		- I
			am
				remark...
namespace $ { export class $my_values extends $mol_view {

	title() {
		return this.$.$mol_locale.text( '$my_values_title' )
	}

	sub() {
		return [ 0 , 1.1 , true , false , <any> null , "I can contain any character! \\(\"o\")/" ]
	}

} }

Dictionary (correspondence keys to their values) could be declared through a node * (you can use ^ to inherit pairs from superclass). For example, set DOM-element's attribute values:

$my_number $mol_view
	dom_name \input
	attr *
		^
		type \number
		- attribute values must be a strings
		min \0
		max \20
namespace $ { export class $my_number extends $mol_view {

	dom_name() { return "input" }

	attr() {
		return { ...super.attr() ,
			"type" : "number" ,
			"min" : "0" ,
			"max" : "20" ,
		}
	}
} }

To set a value for a DOM element's fields:

$my_scroll $mol_view
	field *
		^
		scrollTop 0
namespace $ { export class $my_scroll extends $mol_view {

	field() {
		return { ...super.field() ,
			"scrollTop" : 0 ,
		}
	}
} }

To set styles:

$my_rotate $mol_view
	style *
		^
		transform \rotate( 180deg )
namespace $ { export class $my_rotate extends $mol_view {

	style() {
		return { ...super.style() ,
			"transform" : "rotate( 180deg )" ,
		}
	}
} }

As a value, we could cast not only constants, but also the contents of other properties through <= one-way binding. For example, let's declare two text properties hint and text and then use them for the field dictionary and sub list:

$my_hint $mol_view
	hint \Default hint
	text \Default text
	field *
		^
		title <= hint -
	sub /
		<= text -
namespace $ { export class $my_hint extends $mol_view {

	hint() { return "Default hint" }

	text() { return "Default text" }

	field() {
		return { ...super.field() ,
			"title" : this.hint() ,
		}
	}

	sub() {
		return [ this.text() ]
	}
} }

It's often convenient to combine declaring a property and using it. The following example is exactly the same as the previous one:

$my_hint $mol_view
	field *
		^
		title <= hint \Default hint 
	sub /
		<= text \Default text

Reactions on DOM-events are required for two-way binding. For example, lets point out, that objects of click event is necessary to put in remove property, which we declare right here and set it a default value null:

$my_remover $mol_view
	event *
		^
		click? <=> remove? null 
	sub /
		\Remove
namespace $ { export class $my_remover extends $mol_view {

	@ $mol_mem
	remove( next? : any ) {
		return ( next !== undefinded ) ? next : null as any
	}

	event() {
		return { ...super.event() ,
			"click" : ( next? : any )=> this.remove( next ) ,
		}
	}

	sub() {
		return [ "Remove" ]
	}
} }

You can declare an instance of another class as a value directly. The following example declares a List property with the value of instance $mol_list_demo_tree and then places it in a list of sub child components:

$my_app $mol_view
	List $mol_list_demo_tree
	sub /
		<= List -
namespace $ { export class $my_app extends $mol_view {

	@ $mol_mem
	List() {
		const obj = new $mol_list_demo_tree
		return obj
	}

	sub() {
		return [ this.List() ]
	}

} }

Properties of a nested component can be overloaded, below overloaded title and content properties of $mol_label component:

$my_name $mol_view
	sub /
		<= Info $mol_label
			title \Name
			content \Jin
namespace $ { export class $my_name extends $mol_view {

	@ $mol_mem
	Info() {
		const obj = new $mol_label
		obj.title = () => "Name"
		obj.content = () => "Jin"
		return obj
	}

	sub() {
		return [ this.Info() ]
	}
} }

Properties of parent and child components can be linked. In the following example, we declare a reactive name property and tell the Input child component to use the name property as its own value property, we also tell the Output child component that we want the name property to output inside that. The Input and Output components are linked through the name parent property, and changing the value in the Input will also update the Output:

$my_greeter $mol_view
	sub /
		<= Input $mol_string
			hint \Name
			value? <=> name? \
		<= Output $mol_view
			sub /
				<= name? \
namespace $ { export class $my_greeter extends $mol_view {

	@ $mol_mem
	name( next? : any ) {
		return ( next !== undefined ) ? next : ""
	}

	@ $mol_mem
	Input() {
		const obj = new $mol_string
		obj.hint = () => "Name"
		obj.value = ( next? : any ) => this.name( next )
		return obj
	}

	@ $mol_mem
	Output() {
		const obj = new $mol_view
		obj.sub = () => [ this.name() ]
		return obj
	}

	sub() {
		return [ this.Input() , this.Output() ]
	}

} }

=> - Right-side binding. It declares alias for property of subcomponent in declared component.

$my_app $mol_scroll
	sub /
		<= Page $mol_page
			Title => Page_title -
			head /
				<= Back $mol_button_minor
					title \Back
				<= Page_title -
namespace $ {
	export class $my_app extends $mol_scroll {
		
		// sub / <= Page
		sub() {
			return [
				this.Page()
			] as readonly any[]
		}
		
		// Back $mol_button_minor title \Back
		@ $mol_mem
		Back() {
			const obj = new this.$.$mol_button_minor()
			
			obj.title = () => "Back"
			
			return obj
		}
		
		// Page_title
		Page_title() {
			return this.Page().Title()
		}
		
		// Page $mol_page
		// 	Title => Page_title
		// 	head /
		// 		<= Back
		// 		<= Page_title
		@ $mol_mem
		Page() {
			const obj = new this.$.$mol_page()
			
			obj.head = () => [
				this.Back(),
				this.Page_title()
			] as readonly any[]
			
			return obj
		}
	}
}

There are certain properties that return different values depending on the key. A typical example of is a list of strings. Each row is a separate component, accessed by a unique key. The list of such properties has a * after the name:

$my_tasks $mol_list
	sub <= task_rows /
	Task_row* $mol_view
		sub /
			<= task_title* <= task_title_default \
namespace $ {
	export class $my_tasks extends $mol_list {
		
		// sub <= task_rows
		sub() { return this.task_rows() }
		
		// Task_row* $mol_view sub / <= task_title*
		@ $mol_mem_key
		Task_row(id: any) {
			const obj = new this.$.$mol_view()
			
			obj.sub = () => [
				this.task_title(id)
			] as readonly any[]
			
			return obj
		}
		
		// task_rows /
		task_rows() { return [] as readonly any[] }
		
		// task_title_default \
		task_title_default() { return "" }
		
		// task_title* <= task_title_default
		task_title(id: any) { return this.task_title_default() }
	}	
}

In above example we declared the property Task_row, which takes on input some ID-key and returns an unique instance of $mol_view for every key. Task_row has overloaded property sub that outputs appropriate task_title for every Task_row (task_title returns content of property task_title_default), which is equal to empty string initially. Further, by overloading any of these properties, we can change any aspect of the component's behavior. You can override task_rows in a subclass to generate rows of your choice. For example":

task_rows() {
	const rows = [] as $mol_view[]
	for( let i = 0 ; i < 10 ; ++ i ) rows.push( this.Task_row( i ) )
	return rows
}

task_title(id: any) {
   return `Title - ${id}`
}

Note: There is old way howto to use property with id. Instead of * you can write !id. E.g. instead of task_title* you can use task_title!key. You might find such usage in old examples, tutorials or old projects.

All special chars

  • - - remarks, ignored by code generation
  • $ - component name prefix, e.g $mol_button
  • / - array, optionally you can set type of array, e.g sub /number
  • * - dictionary (string keys, any values)
  • ^ - return value of the same property from super class
  • \ - raw string, e.g. message \Hello
  • @ - localized string, e.g. message @ \Hello world
  • <= - provides read-only property from owner to sub-componen
  • => - provides read-only property from sub-componen to owner
  • <=> - fully replace sub component property by owner's one
  • property + * or ! - property takes ID as first argument, e.g. Task_row* $mol_view
  • property + ? - property can be changed by providing an additional optional argument, e.g. value <=> name? \

view.ts

In addition to declarative description of component, next to it could be created a file of the same name with view.ts extension, where a behavior could be described. Using a special construction, it could be inherited from realization obtained of view.tree and it would be overloaded automatically by heir:

For example we have following description into ./my/hello/hello.view.tree:

$my_hello $mol_view
	sub /
		<= Input $mol_string
			hint \Name
			value? <=> name? \
		<= message \

Here we have declared 2 properties: name to get the value from Input and message to output the value (we will override this property below). It will be translated into following file ./my/hello/-view.tree/hello.view.tree.ts:

namespace $ {
	export class $my_hello extends $mol_view {
		
		// sub /
		// 	<= Input
		// 	<= message
		sub() {
			return [
				this.Input(),
				this.message()
			] as readonly any[]
		}
		
		// name? \
		@ $mol_mem
		name(val?: any) {
			if ( val !== undefined ) return val as never
			return ""
		}
		
		// Input $mol_string
		// 	hint \Name
		// 	value? <=> name?
		@ $mol_mem
		Input() {
			const obj = new this.$.$mol_string()
			
			obj.hint = () => "Name"
			obj.value = (val?: any) => this.name(val)
			
			return obj
		}
		
		// message \
		message() { return "" }
	}
}

Now let's override the message method, which will use the name property in ./my/hello/hello.view.ts behaivour file. message will depend on the name property entered by the user:

namespace $.$$ {
	export class $my_hello extends $.$my_hello {
		
		message() {
			const name = this.name()
			return name && `Hello, ${name}!`
		}
		
	}
}

IDE Support

Articles