|
1 |
| -// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT |
| 1 | +// Copyright 2012-2015 The Rust Project Developers. See the COPYRIGHT |
2 | 2 | // file at the top-level directory of this distribution and at
|
3 | 3 | // http://rust-lang.org/COPYRIGHT.
|
4 | 4 | //
|
@@ -108,6 +108,7 @@ use lint;
|
108 | 108 | use util::common::{block_query, indenter, loop_query};
|
109 | 109 | use util::ppaux::{self, Repr};
|
110 | 110 | use util::nodemap::{DefIdMap, FnvHashMap, NodeMap};
|
| 111 | +use util::lev_distance::lev_distance; |
111 | 112 |
|
112 | 113 | use std::cell::{Cell, Ref, RefCell};
|
113 | 114 | use std::mem::replace;
|
@@ -3071,11 +3072,43 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
|
3071 | 3072 | actual)
|
3072 | 3073 | },
|
3073 | 3074 | expr_t, None);
|
| 3075 | + if let Some(t) = ty::ty_to_def_id(expr_t) { |
| 3076 | + suggest_field_names(t, field, tcx, vec![]); |
| 3077 | + } |
3074 | 3078 | }
|
3075 | 3079 |
|
3076 | 3080 | fcx.write_error(expr.id);
|
3077 | 3081 | }
|
3078 | 3082 |
|
| 3083 | + // displays hints about the closest matches in field names |
| 3084 | + fn suggest_field_names<'tcx>(id : DefId, |
| 3085 | + field : &ast::SpannedIdent, |
| 3086 | + tcx : &ty::ctxt<'tcx>, |
| 3087 | + skip : Vec<&str>) { |
| 3088 | + let ident = token::get_ident(field.node); |
| 3089 | + let name = ident.get(); |
| 3090 | + // only find fits with at least one matching letter |
| 3091 | + let mut best_dist = name.len(); |
| 3092 | + let mut best = None; |
| 3093 | + let fields = ty::lookup_struct_fields(tcx, id); |
| 3094 | + for elem in fields.iter() { |
| 3095 | + let n = elem.name.as_str(); |
| 3096 | + // ignore already set fields |
| 3097 | + if skip.iter().any(|&x| x == n) { |
| 3098 | + continue; |
| 3099 | + } |
| 3100 | + let dist = lev_distance(n, name); |
| 3101 | + if dist < best_dist { |
| 3102 | + best = Some(n); |
| 3103 | + best_dist = dist; |
| 3104 | + } |
| 3105 | + } |
| 3106 | + if let Some(n) = best { |
| 3107 | + tcx.sess.span_help(field.span, |
| 3108 | + format!("did you mean `{}`?", n).as_slice()); |
| 3109 | + } |
| 3110 | + } |
| 3111 | + |
3079 | 3112 | // Check tuple index expressions
|
3080 | 3113 | fn check_tup_field(fcx: &FnCtxt,
|
3081 | 3114 | expr: &ast::Expr,
|
@@ -3183,6 +3216,13 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
|
3183 | 3216 | },
|
3184 | 3217 | struct_ty,
|
3185 | 3218 | None);
|
| 3219 | + // prevent all specified fields from being suggested |
| 3220 | + let skip_fields = ast_fields.iter().map(|ref x| x.ident.node.name.as_str()); |
| 3221 | + let actual_id = match enum_id_opt { |
| 3222 | + Some(_) => class_id, |
| 3223 | + None => ty::ty_to_def_id(struct_ty).unwrap() |
| 3224 | + }; |
| 3225 | + suggest_field_names(actual_id, &field.ident, tcx, skip_fields.collect()); |
3186 | 3226 | error_happened = true;
|
3187 | 3227 | }
|
3188 | 3228 | Some((_, true)) => {
|
|
0 commit comments