@@ -784,7 +784,7 @@ export function processTypeDeclaration(declaration: string, isExported = true):
784
784
}
785
785
786
786
/**
787
- * Enhanced function signature extraction with better destructuring and generics support
787
+ * Extract complete function signature
788
788
*/
789
789
export interface FunctionSignature {
790
790
name : string
@@ -814,9 +814,11 @@ export function extractFunctionSignature(declaration: string): FunctionSignature
814
814
. replace ( / ^ f u n c t i o n \s + / , '' )
815
815
. trim ( )
816
816
817
- // Extract generics with improved regex
817
+ // Extract complete generic section including constraints
818
818
const genericsMatch = cleanDeclaration . match ( / < ( [ ^ > ] + ) > (? = \s * \( ) / )
819
- const generics = genericsMatch ? `<${ genericsMatch [ 1 ] } >` : ''
819
+ const generics = genericsMatch
820
+ ? normalizeGenerics ( genericsMatch [ 1 ] )
821
+ : ''
820
822
821
823
// Remove generics for further parsing
822
824
const withoutGenerics = cleanDeclaration . replace ( / < ( [ ^ > ] + ) > (? = \s * \( ) / , '' )
@@ -829,25 +831,42 @@ export function extractFunctionSignature(declaration: string): FunctionSignature
829
831
const paramsMatch = withoutGenerics . match ( / \( ( [ \s \S ] * ?) \) (? = \s * : ) / )
830
832
let params = paramsMatch ? paramsMatch [ 1 ] . trim ( ) : ''
831
833
832
- // Clean up parameters
834
+ // Clean up parameters while preserving generic references
833
835
params = cleanParameters ( params )
834
836
835
- // Extract return type with improved generic handling
837
+ // Extract return type with generic preservation
836
838
const returnTypeMatch = withoutGenerics . match ( / \) \s * : \s * ( [ \s \S ] + ?) (? = \{ | $ ) / )
837
839
let returnType = returnTypeMatch ? returnTypeMatch [ 1 ] . trim ( ) : 'void'
838
840
839
841
// Ensure generic type parameters in return types are preserved
840
- returnType = returnType . replace ( / \s + / g , '' )
842
+ returnType = normalizeType ( returnType )
841
843
842
844
return {
843
845
name,
844
846
params,
845
847
returnType,
846
848
isAsync : declaration . includes ( 'async' ) ,
847
- generics,
849
+ generics : generics ? `< ${ generics } >` : '' ,
848
850
}
849
851
}
850
852
853
+ /**
854
+ * Normalize generic type parameters and constraints
855
+ */
856
+ function normalizeGenerics ( generics : string ) : string {
857
+ return generics
858
+ . split ( ',' )
859
+ . map ( ( param ) => {
860
+ const parts = param . trim ( ) . split ( / \s + e x t e n d s \s + / )
861
+ if ( parts . length === 2 ) {
862
+ const [ typeParam , constraint ] = parts
863
+ return `${ typeParam . trim ( ) } extends ${ normalizeType ( constraint ) } `
864
+ }
865
+ return param . trim ( )
866
+ } )
867
+ . join ( ', ' )
868
+ }
869
+
851
870
/**
852
871
* Process function declaration with fixed generic handling
853
872
*/
@@ -864,19 +883,20 @@ export function processFunctionDeclaration(
864
883
generics,
865
884
} = extractFunctionSignature ( declaration )
866
885
867
- // Track all used types
886
+ // Track all used types including generics
868
887
trackUsedTypes ( `${ generics } ${ params } ${ returnType } ` , usedTypes )
869
888
870
- // Build declaration string ensuring correct placement of generics
889
+ // Build declaration string ensuring proper generic scope
871
890
const parts = [
872
891
isExported ? 'export' : '' ,
873
892
'declare' ,
874
893
isAsync ? 'async' : '' ,
875
894
'function' ,
876
895
name ,
896
+ // Ensure generics are properly placed
877
897
generics ,
878
898
`(${ params } )` ,
879
- `:` ,
899
+ ':' ,
880
900
returnType ,
881
901
';' ,
882
902
]
@@ -886,7 +906,7 @@ export function processFunctionDeclaration(
886
906
. join ( ' ' )
887
907
. replace ( / \s + ( [ < > ( ) , ; : ] ) / g, '$1' )
888
908
. replace ( / ( [ < > ( ) , ; : ] ) \s + / g, '$1 ' )
889
- . replace ( / \s + / g, ' ' )
909
+ . replace ( / \s { 2 , } / g, ' ' )
890
910
. trim ( )
891
911
}
892
912
@@ -898,27 +918,36 @@ export function cleanParameters(params: string): string {
898
918
return ''
899
919
900
920
return params
901
- // Handle destructured parameters
921
+ // Handle destructured parameters while preserving generic references
902
922
. replace ( / \{ ( [ ^ } ] + ) \} : \s * ( [ ^ , ) ] + ) / g, ( _ , props , type ) => {
903
- // Convert destructured parameters to a single options parameter
904
- const typeName = type . trim ( )
923
+ // Preserve the generic type reference
924
+ const typeName = normalizeType ( type . trim ( ) )
905
925
return `options: ${ typeName } `
906
926
} )
907
927
// Normalize spaces around special characters
908
928
. replace ( / \s * ( [ , : ] ) \s * / g, '$1 ' )
909
- // Clean up multiple spaces
910
- . replace ( / \s + / g, ' ' )
911
929
// Add space after commas if missing
912
930
. replace ( / , ( \S ) / g, ', $1' )
913
931
// Normalize optional parameter syntax
914
932
. replace ( / \s * \? \s * : / g, '?: ' )
915
- // Clean up spaces around array/generic brackets
933
+ // Clean up spaces around array/generic brackets while preserving content
916
934
. replace ( / \s * ( [ < [ \] > ] ) \s * / g, '$1' )
917
935
// Final cleanup of any double spaces
918
936
. replace ( / \s { 2 , } / g, ' ' )
919
937
. trim ( )
920
938
}
921
939
940
+ /**
941
+ * Normalize type references while preserving generic parameters
942
+ */
943
+ function normalizeType ( type : string ) : string {
944
+ return type
945
+ . replace ( / \s + / g, ' ' )
946
+ . replace ( / \s * ( [ < > ] ) \s * / g, '$1' )
947
+ . replace ( / \s * , \s * / g, ', ' )
948
+ . trim ( )
949
+ }
950
+
922
951
/**
923
952
* Track used types in function signatures and bodies
924
953
*/
@@ -932,7 +961,7 @@ export function trackUsedTypes(content: string, usedTypes: Set<string>): void {
932
961
if ( type ) {
933
962
// Extract base type and any nested generic types
934
963
const [ baseType , ...genericParams ] = type . split ( / [ < > ] / )
935
- if ( baseType )
964
+ if ( baseType && / ^ [ A - Z ] / . test ( baseType ) )
936
965
usedTypes . add ( baseType )
937
966
938
967
// Process generic parameters
0 commit comments