153 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			153 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 
								 | 
							
								import NodeFunction from '../core/NodeFunction.js';
							 | 
						||
| 
								 | 
							
								import NodeFunctionInput from '../core/NodeFunctionInput.js';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								const declarationRegexp = /^\s*(highp|mediump|lowp)?\s*([a-z_0-9]+)\s*([a-z_0-9]+)?\s*\(([\s\S]*?)\)/i;
							 | 
						||
| 
								 | 
							
								const propertiesRegexp = /[a-z_0-9]+/ig;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								const pragmaMain = '#pragma main';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								const parse = ( source ) => {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									source = source.trim();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									const pragmaMainIndex = source.indexOf( pragmaMain );
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									const mainCode = pragmaMainIndex !== - 1 ? source.slice( pragmaMainIndex + pragmaMain.length ) : source;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									const declaration = mainCode.match( declarationRegexp );
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if ( declaration !== null && declaration.length === 5 ) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										// tokenizer
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										const inputsCode = declaration[ 4 ];
							 | 
						||
| 
								 | 
							
										const propsMatches = [];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										let nameMatch = null;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										while ( ( nameMatch = propertiesRegexp.exec( inputsCode ) ) !== null ) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											propsMatches.push( nameMatch );
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										// parser
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										const inputs = [];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										let i = 0;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										while ( i < propsMatches.length ) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											const isConst = propsMatches[ i ][ 0 ] === 'const';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											if ( isConst === true ) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												i ++;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											let qualifier = propsMatches[ i ][ 0 ];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											if ( qualifier === 'in' || qualifier === 'out' || qualifier === 'inout' ) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												i ++;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											} else {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												qualifier = '';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											const type = propsMatches[ i ++ ][ 0 ];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											let count = Number.parseInt( propsMatches[ i ][ 0 ] );
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											if ( Number.isNaN( count ) === false ) i ++;
							 | 
						||
| 
								 | 
							
											else count = null;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											const name = propsMatches[ i ++ ][ 0 ];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											inputs.push( new NodeFunctionInput( type, name, count, qualifier, isConst ) );
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										//
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										const blockCode = mainCode.substring( declaration[ 0 ].length );
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										const name = declaration[ 3 ] !== undefined ? declaration[ 3 ] : '';
							 | 
						||
| 
								 | 
							
										const type = declaration[ 2 ];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										const presicion = declaration[ 1 ] !== undefined ? declaration[ 1 ] : '';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										const headerCode = pragmaMainIndex !== - 1 ? source.slice( 0, pragmaMainIndex ) : '';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										return {
							 | 
						||
| 
								 | 
							
											type,
							 | 
						||
| 
								 | 
							
											inputs,
							 | 
						||
| 
								 | 
							
											name,
							 | 
						||
| 
								 | 
							
											presicion,
							 | 
						||
| 
								 | 
							
											inputsCode,
							 | 
						||
| 
								 | 
							
											blockCode,
							 | 
						||
| 
								 | 
							
											headerCode
							 | 
						||
| 
								 | 
							
										};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									} else {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										throw new Error( 'FunctionNode: Function is not a GLSL code.' );
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class GLSLNodeFunction extends NodeFunction {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									constructor( source ) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										const { type, inputs, name, presicion, inputsCode, blockCode, headerCode } = parse( source );
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										super( type, inputs, name, presicion );
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										this.inputsCode = inputsCode;
							 | 
						||
| 
								 | 
							
										this.blockCode = blockCode;
							 | 
						||
| 
								 | 
							
										this.headerCode = headerCode;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									getCode( name = this.name ) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										let code;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										const blockCode = this.blockCode;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										if ( blockCode !== '' ) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											const { type, inputsCode, headerCode, presicion } = this;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											let declarationCode = `${ type } ${ name } ( ${ inputsCode.trim() } )`;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											if ( presicion !== '' ) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												declarationCode = `${ presicion } ${ declarationCode }`;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											code = headerCode + declarationCode + blockCode;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										} else {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											// interface function
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											code = '';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										return code;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								export default GLSLNodeFunction;
							 |