@@ -38,14 +38,25 @@ crate type Suggestion = (Vec<(Span, String)>, String, Applicability);
38
38
/// similarly named label and whether or not it is reachable.
39
39
crate type LabelSuggestion = ( Ident , bool ) ;
40
40
41
+ crate enum SuggestionTarget {
42
+ /// The target has a similar name as the name used by the programmer (probably a typo)
43
+ SimilarlyNamed ,
44
+ /// The target is the only valid item that can be used in the corresponding context
45
+ SingleItem ,
46
+ }
47
+
41
48
crate struct TypoSuggestion {
42
49
pub candidate : Symbol ,
43
50
pub res : Res ,
51
+ pub target : SuggestionTarget ,
44
52
}
45
53
46
54
impl TypoSuggestion {
47
- crate fn from_res ( candidate : Symbol , res : Res ) -> TypoSuggestion {
48
- TypoSuggestion { candidate, res }
55
+ crate fn typo_from_res ( candidate : Symbol , res : Res ) -> TypoSuggestion {
56
+ Self { candidate, res, target : SuggestionTarget :: SimilarlyNamed }
57
+ }
58
+ crate fn single_item_from_res ( candidate : Symbol , res : Res ) -> TypoSuggestion {
59
+ Self { candidate, res, target : SuggestionTarget :: SingleItem }
49
60
}
50
61
}
51
62
@@ -80,7 +91,7 @@ impl<'a> Resolver<'a> {
80
91
if let Some ( binding) = resolution. borrow ( ) . binding {
81
92
let res = binding. res ( ) ;
82
93
if filter_fn ( res) {
83
- names. push ( TypoSuggestion :: from_res ( key. ident . name , res) ) ;
94
+ names. push ( TypoSuggestion :: typo_from_res ( key. ident . name , res) ) ;
84
95
}
85
96
}
86
97
}
@@ -623,7 +634,7 @@ impl<'a> Resolver<'a> {
623
634
. get ( & expn_id)
624
635
. into_iter ( )
625
636
. flatten ( )
626
- . map ( |ident| TypoSuggestion :: from_res ( ident. name , res) ) ,
637
+ . map ( |ident| TypoSuggestion :: typo_from_res ( ident. name , res) ) ,
627
638
) ;
628
639
}
629
640
}
@@ -642,7 +653,7 @@ impl<'a> Resolver<'a> {
642
653
suggestions. extend (
643
654
ext. helper_attrs
644
655
. iter ( )
645
- . map ( |name| TypoSuggestion :: from_res ( * name, res) ) ,
656
+ . map ( |name| TypoSuggestion :: typo_from_res ( * name, res) ) ,
646
657
) ;
647
658
}
648
659
}
@@ -652,8 +663,10 @@ impl<'a> Resolver<'a> {
652
663
if let MacroRulesScope :: Binding ( macro_rules_binding) = macro_rules_scope. get ( ) {
653
664
let res = macro_rules_binding. binding . res ( ) ;
654
665
if filter_fn ( res) {
655
- suggestions
656
- . push ( TypoSuggestion :: from_res ( macro_rules_binding. ident . name , res) )
666
+ suggestions. push ( TypoSuggestion :: typo_from_res (
667
+ macro_rules_binding. ident . name ,
668
+ res,
669
+ ) )
657
670
}
658
671
}
659
672
}
@@ -671,15 +684,15 @@ impl<'a> Resolver<'a> {
671
684
suggestions. extend (
672
685
this. registered_attrs
673
686
. iter ( )
674
- . map ( |ident| TypoSuggestion :: from_res ( ident. name , res) ) ,
687
+ . map ( |ident| TypoSuggestion :: typo_from_res ( ident. name , res) ) ,
675
688
) ;
676
689
}
677
690
}
678
691
Scope :: MacroUsePrelude => {
679
692
suggestions. extend ( this. macro_use_prelude . iter ( ) . filter_map (
680
693
|( name, binding) | {
681
694
let res = binding. res ( ) ;
682
- filter_fn ( res) . then_some ( TypoSuggestion :: from_res ( * name, res) )
695
+ filter_fn ( res) . then_some ( TypoSuggestion :: typo_from_res ( * name, res) )
683
696
} ,
684
697
) ) ;
685
698
}
@@ -689,22 +702,22 @@ impl<'a> Resolver<'a> {
689
702
suggestions. extend (
690
703
BUILTIN_ATTRIBUTES
691
704
. iter ( )
692
- . map ( |( name, ..) | TypoSuggestion :: from_res ( * name, res) ) ,
705
+ . map ( |( name, ..) | TypoSuggestion :: typo_from_res ( * name, res) ) ,
693
706
) ;
694
707
}
695
708
}
696
709
Scope :: ExternPrelude => {
697
710
suggestions. extend ( this. extern_prelude . iter ( ) . filter_map ( |( ident, _) | {
698
711
let res = Res :: Def ( DefKind :: Mod , DefId :: local ( CRATE_DEF_INDEX ) ) ;
699
- filter_fn ( res) . then_some ( TypoSuggestion :: from_res ( ident. name , res) )
712
+ filter_fn ( res) . then_some ( TypoSuggestion :: typo_from_res ( ident. name , res) )
700
713
} ) ) ;
701
714
}
702
715
Scope :: ToolPrelude => {
703
716
let res = Res :: NonMacroAttr ( NonMacroAttrKind :: Tool ) ;
704
717
suggestions. extend (
705
718
this. registered_tools
706
719
. iter ( )
707
- . map ( |ident| TypoSuggestion :: from_res ( ident. name , res) ) ,
720
+ . map ( |ident| TypoSuggestion :: typo_from_res ( ident. name , res) ) ,
708
721
) ;
709
722
}
710
723
Scope :: StdLibPrelude => {
@@ -721,7 +734,7 @@ impl<'a> Resolver<'a> {
721
734
Scope :: BuiltinTypes => {
722
735
suggestions. extend ( PrimTy :: ALL . iter ( ) . filter_map ( |prim_ty| {
723
736
let res = Res :: PrimTy ( * prim_ty) ;
724
- filter_fn ( res) . then_some ( TypoSuggestion :: from_res ( prim_ty. name ( ) , res) )
737
+ filter_fn ( res) . then_some ( TypoSuggestion :: typo_from_res ( prim_ty. name ( ) , res) )
725
738
} ) )
726
739
}
727
740
}
@@ -993,20 +1006,31 @@ impl<'a> Resolver<'a> {
993
1006
// | ^
994
1007
return false ;
995
1008
}
1009
+ let prefix = match suggestion. target {
1010
+ SuggestionTarget :: SimilarlyNamed => "similarly named " ,
1011
+ SuggestionTarget :: SingleItem => "" ,
1012
+ } ;
1013
+
996
1014
err. span_label (
997
1015
self . session . source_map ( ) . guess_head_span ( def_span) ,
998
1016
& format ! (
999
- "similarly named {} `{}` defined here" ,
1017
+ "{}{} `{}` defined here" ,
1018
+ prefix,
1000
1019
suggestion. res. descr( ) ,
1001
1020
suggestion. candidate. as_str( ) ,
1002
1021
) ,
1003
1022
) ;
1004
1023
}
1005
- let msg = format ! (
1006
- "{} {} with a similar name exists" ,
1007
- suggestion. res. article( ) ,
1008
- suggestion. res. descr( )
1009
- ) ;
1024
+ let msg = match suggestion. target {
1025
+ SuggestionTarget :: SimilarlyNamed => format ! (
1026
+ "{} {} with a similar name exists" ,
1027
+ suggestion. res. article( ) ,
1028
+ suggestion. res. descr( )
1029
+ ) ,
1030
+ SuggestionTarget :: SingleItem => {
1031
+ format ! ( "maybe you meant this {}" , suggestion. res. descr( ) )
1032
+ }
1033
+ } ;
1010
1034
err. span_suggestion (
1011
1035
span,
1012
1036
& msg,
0 commit comments