three 基础库
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

279 lines
4.6 KiB

2 years ago
import { TempNode } from './TempNode.js';
import { NodeLib } from './NodeLib.js';
const declarationRegexp = /^\s*([a-z_0-9]+)\s+([a-z_0-9]+)\s*\(([\s\S]*?)\)/i,
propertiesRegexp = /[a-z_0-9]+/ig;
class FunctionNode extends TempNode {
constructor( src, includes, extensions, keywords, type ) {
super( type );
this.isMethod = type === undefined;
this.isInterface = false;
this.parse( src, includes, extensions, keywords );
}
getShared( /* builder, output */ ) {
return ! this.isMethod;
}
getType( builder ) {
return builder.getTypeByFormat( this.type );
}
getInputByName( name ) {
let i = this.inputs.length;
while ( i -- ) {
if ( this.inputs[ i ].name === name ) {
return this.inputs[ i ];
}
}
}
getIncludeByName( name ) {
let i = this.includes.length;
while ( i -- ) {
if ( this.includes[ i ].name === name ) {
return this.includes[ i ];
}
}
}
generate( builder, output ) {
let match, offset = 0, src = this.src;
for ( let i = 0; i < this.includes.length; i ++ ) {
builder.include( this.includes[ i ], this );
}
for ( const ext in this.extensions ) {
builder.extensions[ ext ] = true;
}
const matches = [];
while ( match = propertiesRegexp.exec( this.src ) ) matches.push( match );
for ( let i = 0; i < matches.length; i ++ ) {
const match = matches[ i ];
const prop = match[ 0 ],
isGlobal = this.isMethod ? ! this.getInputByName( prop ) : true;
let reference = prop;
if ( this.keywords[ prop ] || ( this.useKeywords && isGlobal && NodeLib.containsKeyword( prop ) ) ) {
let node = this.keywords[ prop ];
if ( ! node ) {
const keyword = NodeLib.getKeywordData( prop );
if ( keyword.cache ) node = builder.keywords[ prop ];
node = node || NodeLib.getKeyword( prop, builder );
if ( keyword.cache ) builder.keywords[ prop ] = node;
}
reference = node.build( builder );
}
if ( prop !== reference ) {
src = src.substring( 0, match.index + offset ) + reference + src.substring( match.index + prop.length + offset );
offset += reference.length - prop.length;
}
if ( this.getIncludeByName( reference ) === undefined && NodeLib.contains( reference ) ) {
builder.include( NodeLib.get( reference ) );
}
}
if ( output === 'source' ) {
return src;
} else if ( this.isMethod ) {
if ( ! this.isInterface ) {
builder.include( this, false, src );
}
return this.name;
} else {
return builder.format( '( ' + src + ' )', this.getType( builder ), output );
}
}
parse( src, includes, extensions, keywords ) {
this.src = src || '';
this.includes = includes || [];
this.extensions = extensions || {};
this.keywords = keywords || {};
if ( this.isMethod ) {
const match = this.src.match( declarationRegexp );
this.inputs = [];
if ( match && match.length == 4 ) {
this.type = match[ 1 ];
this.name = match[ 2 ];
const inputs = match[ 3 ].match( propertiesRegexp );
if ( inputs ) {
let i = 0;
while ( i < inputs.length ) {
let qualifier = inputs[ i ++ ];
let type;
if ( qualifier === 'in' || qualifier === 'out' || qualifier === 'inout' ) {
type = inputs[ i ++ ];
} else {
type = qualifier;
qualifier = '';
}
const name = inputs[ i ++ ];
this.inputs.push( {
name: name,
type: type,
qualifier: qualifier
} );
}
}
this.isInterface = this.src.indexOf( '{' ) === - 1;
} else {
this.type = '';
this.name = '';
}
}
}
copy( source ) {
super.copy( source );
this.isMethod = source.isMethod;
this.useKeywords = source.useKeywords;
this.parse( source.src, source.includes, source.extensions, source.keywords );
if ( source.type !== undefined ) this.type = source.type;
return this;
}
toJSON( meta ) {
let data = this.getJSONNode( meta );
if ( ! data ) {
data = this.createJSONNode( meta );
data.src = this.src;
data.isMethod = this.isMethod;
data.useKeywords = this.useKeywords;
if ( ! this.isMethod ) data.type = this.type;
data.extensions = JSON.parse( JSON.stringify( this.extensions ) );
data.keywords = {};
for ( const keyword in this.keywords ) {
data.keywords[ keyword ] = this.keywords[ keyword ].toJSON( meta ).uuid;
}
if ( this.includes.length ) {
data.includes = [];
for ( let i = 0; i < this.includes.length; i ++ ) {
data.includes.push( this.includes[ i ].toJSON( meta ).uuid );
}
}
}
return data;
}
}
FunctionNode.prototype.nodeType = 'Function';
FunctionNode.prototype.useKeywords = true;
export { FunctionNode };