@@ -61,15 +61,18 @@ type ParserOptions = {
61
61
// like undefined functions and variables. If `failOnWarn` is set to true,
62
62
// warnings will still cause the parser to raise an error. Defaults to false.
63
63
quiet: boolean ;
64
+
64
65
// An optional string representing the origin of the GLSL, for debugging and
65
66
// error messages. For example, "main.js". If the parser raises an error, the
66
67
// grammarSource shows up in the error.source field. If you format the error
67
68
// (see the errors section), the grammarSource shows up in the formatted error
68
69
// string. Defaults to undefined.
69
70
grammarSource: string ;
71
+
70
72
// If true, sets location information on each AST node, in the form of
71
73
// { column: number, line: number, offset: number }. Defaults to false.
72
74
includeLocation: boolean ;
75
+
73
76
// If true, causes the parser to raise an error instead of log a warning.
74
77
// The parser does limited type checking, and things like undeclared variables
75
78
// are treated as warnings. Defaults to false.
@@ -427,19 +430,86 @@ visitors.
427
430
428
431
### Utility Functions
429
432
430
- Rename all the variables in a program:
433
+ #### Rename variables / identifiers in a program
434
+
435
+ You can rename bindings (aka variables), functions, and types (aka structs) with ` renameBindings ` , ` renameFunctions ` , and ` renameTypes ` respectively.
436
+
437
+ The signature for these methods:
438
+
439
+ ` ` ` ts
440
+ const renameBindings = (
441
+ // The scope to rename the bindings in. ast.scopes[0] is the global scope.
442
+ // Passing this ast.scopes[0] renames all global variables
443
+ bindings : ScopeIndex ,
444
+
445
+ // The rename function. This is called once per scope entry with the original
446
+ // name in the scope, to generate the renamed variable.
447
+ mangle : (name : string ) => string
448
+ ): ScopeIndex
449
+ ` ` `
450
+
451
+ These scope renaming functions, ` renameBindings ` , ` renameFunctions ` , and ` renameTypes ` , do two things:
452
+ 1. Each function *mutates* the AST to rename identifiers in place.
453
+ 2. They *return* an *immutable* new ScopeIndex where the scope references
454
+ themselves are renamed.
455
+
456
+ If you want your ast.scopes array to stay in sync with your AST, you need to
457
+ re-assign it to the output of the functions! Examples:
431
458
432
459
` ` ` typescript
433
460
import { renameBindings , renameFunctions , renameTypes } from ' @shaderfrog/glsl-parser/utils' ;
434
461
435
- // ... parse an ast...
462
+ // Suffix top level variables with _x, and update the scope
463
+ ast .scopes [0 ].bindings = renameBindings (ast .scopes [0 ].bindings , (name ) => ` ${name }_x ` );
436
464
437
- // Suffix top level variables with _x
438
- renameBindings (ast .scopes [0 ], (name , node ) => ` ${name }_x ` );
439
465
// Suffix function names with _x
440
- renameFunctions (ast .scopes [0 ], (name , node ) => ` ${name }_x ` );
466
+ ast .scopes [0 ].functions = renameFunctions (ast .scopes [0 ].functions , (name ) => ` ${name }_x ` );
467
+
441
468
// Suffix struct names and usages (including constructors) with _x
442
- renameTypes (ast .scopes [0 ], (name , node ) => ` ${name }_x ` );
469
+ ast .scopes [0 ].types = renameTypes (ast .scopes [0 ].types , (name ) => ` ${name }_x ` );
470
+ ` ` `
471
+
472
+ There are also functions to rename only one variable/identifier in a given
473
+ scope. Use these if you know specifically which variable you want to rename.
474
+
475
+ ` ` ` typescript
476
+ import { renameBinding , renameFunction , renameType } from ' @shaderfrog/glsl-parser/utils' ;
477
+
478
+ // Replace all instances of "oldVar" with "newVar" in the global scope, and
479
+ // creates a new global scope entry named "newVar"
480
+ ast .scopes [0 ].bindings .newVar = renameBinding (
481
+ ast .scopes [0 ].bindings .oldVar ,
482
+ ' newVar' ,
483
+ );
484
+ // You need to manually delete the old scope entry if you want the scope to stay
485
+ // in sync with your program AST
486
+ delete ast .scopes [0 ].bindings .oldVar ;
487
+
488
+ // Rename a specific function
489
+ ast .scopes [0 ].functions .newFn = renameFunction (
490
+ ast .scopes [0 ].functions .oldFn ,
491
+ ' newFn' ,
492
+ );
493
+ delete ast .scopes [0 ].functions .oldFn ;
494
+
495
+ // Rename a specific type/struct
496
+ ast .scopes [0 ].functions .newType = renametype (
497
+ ast .scopes [0 ].functions .oldType ,
498
+ ' newType' ,
499
+ );
500
+ delete ast .scopes [0 ].functions .oldType ;
501
+ ` ` `
502
+
503
+ #### Debugging utility functions
504
+
505
+ The parser also exports a debugging function, useful for logging information
506
+ about the AST.
507
+
508
+ ` ` ` ts
509
+ import { debugScopes } from ' @shaderfrog/glsl-parser/parser/utils' ;
510
+
511
+ // Print a condensed representation of the AST scopes to the console
512
+ debugScopes (ast );
443
513
` ` `
444
514
445
515
## What are "parsing" and "preprocessing"?
0 commit comments