Auto merge of #11994 - eddyb:struct-literal-field-shorthand, r=nrc
Implement field shorthands in struct literal expressions. Implements #37340 in a straight-forward way: `Foo { x, y: f() }` parses as `Foo { x: x, y: f() }`. Because of the added `is_shorthand` to `ast::Field`, this is `[syntax-breaking]` (cc @Manishearth). * [x] Mark the fields as being a shorthand (the exact same way we do it in patterns), for pretty-printing. * [x] Gate the shorthand syntax with `#![feature(field_init_shorthand)]`. * [x] Don't parse numeric field as identifiers. * [x] Arbitrary field order tests.
This commit is contained in:
commit
bc283c9487
@ -543,6 +543,7 @@ impl<'a> LoweringContext<'a> {
|
||||
name: respan(f.ident.span, f.ident.node.name),
|
||||
expr: self.lower_expr(&f.expr),
|
||||
span: f.span,
|
||||
is_shorthand: f.is_shorthand,
|
||||
}
|
||||
}
|
||||
|
||||
@ -1682,6 +1683,7 @@ impl<'a> LoweringContext<'a> {
|
||||
},
|
||||
span: span,
|
||||
expr: expr,
|
||||
is_shorthand: false,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -817,6 +817,7 @@ pub struct Field {
|
||||
pub name: Spanned<Name>,
|
||||
pub expr: P<Expr>,
|
||||
pub span: Span,
|
||||
pub is_shorthand: bool,
|
||||
}
|
||||
|
||||
#[derive(Clone, PartialEq, Eq, RustcEncodable, RustcDecodable, Hash, Debug, Copy)]
|
||||
|
@ -1229,8 +1229,10 @@ impl<'a> State<'a> {
|
||||
&fields[..],
|
||||
|s, field| {
|
||||
s.ibox(indent_unit)?;
|
||||
s.print_name(field.name.node)?;
|
||||
s.word_space(":")?;
|
||||
if !field.is_shorthand {
|
||||
s.print_name(field.name.node)?;
|
||||
s.word_space(":")?;
|
||||
}
|
||||
s.print_expr(&field.expr)?;
|
||||
s.end()
|
||||
},
|
||||
|
@ -900,6 +900,7 @@ pub struct Field {
|
||||
pub ident: SpannedIdent,
|
||||
pub expr: P<Expr>,
|
||||
pub span: Span,
|
||||
pub is_shorthand: bool,
|
||||
}
|
||||
|
||||
pub type SpannedIdent = Spanned<Ident>;
|
||||
|
@ -713,7 +713,7 @@ impl<'a> AstBuilder for ExtCtxt<'a> {
|
||||
self.expr(b.span, ast::ExprKind::Block(b))
|
||||
}
|
||||
fn field_imm(&self, span: Span, name: Ident, e: P<ast::Expr>) -> ast::Field {
|
||||
ast::Field { ident: respan(span, name), expr: e, span: span }
|
||||
ast::Field { ident: respan(span, name), expr: e, span: span, is_shorthand: false }
|
||||
}
|
||||
fn expr_struct(&self, span: Span, path: ast::Path, fields: Vec<ast::Field>) -> P<ast::Expr> {
|
||||
self.expr(span, ast::ExprKind::Struct(path, fields, None))
|
||||
|
@ -306,6 +306,9 @@ declare_features! (
|
||||
|
||||
// Allows attributes on lifetime/type formal parameters in generics (RFC 1327)
|
||||
(active, generic_param_attrs, "1.11.0", Some(34761)),
|
||||
|
||||
// Allows field shorthands (`x` meaning `x: x`) in struct literal expressions.
|
||||
(active, field_init_shorthand, "1.14.0", Some(37340)),
|
||||
);
|
||||
|
||||
declare_features! (
|
||||
@ -1087,6 +1090,14 @@ impl<'a> Visitor for PostExpansionVisitor<'a> {
|
||||
ast::ExprKind::InPlace(..) => {
|
||||
gate_feature_post!(&self, placement_in_syntax, e.span, EXPLAIN_PLACEMENT_IN);
|
||||
}
|
||||
ast::ExprKind::Struct(_, ref fields, _) => {
|
||||
for field in fields {
|
||||
if field.is_shorthand {
|
||||
gate_feature_post!(&self, field_init_shorthand, field.span,
|
||||
"struct field shorthands are unstable");
|
||||
}
|
||||
}
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
visit::walk_expr(self, e);
|
||||
|
@ -823,11 +823,12 @@ pub fn noop_fold_struct_field<T: Folder>(f: StructField, fld: &mut T) -> StructF
|
||||
}
|
||||
}
|
||||
|
||||
pub fn noop_fold_field<T: Folder>(Field {ident, expr, span}: Field, folder: &mut T) -> Field {
|
||||
pub fn noop_fold_field<T: Folder>(f: Field, folder: &mut T) -> Field {
|
||||
Field {
|
||||
ident: respan(ident.span, folder.fold_ident(ident.node)),
|
||||
expr: folder.fold_expr(expr),
|
||||
span: folder.new_span(span)
|
||||
ident: respan(f.ident.span, folder.fold_ident(f.ident.node)),
|
||||
expr: folder.fold_expr(f.expr),
|
||||
span: folder.new_span(f.span),
|
||||
is_shorthand: f.is_shorthand,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -2007,17 +2007,30 @@ impl<'a> Parser<'a> {
|
||||
}
|
||||
}
|
||||
|
||||
/// Parse ident COLON expr
|
||||
/// Parse ident (COLON expr)?
|
||||
pub fn parse_field(&mut self) -> PResult<'a, Field> {
|
||||
let lo = self.span.lo;
|
||||
let i = self.parse_field_name()?;
|
||||
let hi = self.prev_span.hi;
|
||||
self.expect(&token::Colon)?;
|
||||
let e = self.parse_expr()?;
|
||||
let hi;
|
||||
|
||||
// Check if a colon exists one ahead. This means we're parsing a fieldname.
|
||||
let (fieldname, expr, is_shorthand) = if self.look_ahead(1, |t| t == &token::Colon) {
|
||||
let fieldname = self.parse_field_name()?;
|
||||
self.bump();
|
||||
hi = self.prev_span.hi;
|
||||
(fieldname, self.parse_expr()?, false)
|
||||
} else {
|
||||
let fieldname = self.parse_ident()?;
|
||||
hi = self.prev_span.hi;
|
||||
|
||||
// Mimic `x: x` for the `x` field shorthand.
|
||||
let path = ast::Path::from_ident(mk_sp(lo, hi), fieldname);
|
||||
(fieldname, self.mk_expr(lo, hi, ExprKind::Path(None, path), ThinVec::new()), true)
|
||||
};
|
||||
Ok(ast::Field {
|
||||
ident: spanned(lo, hi, i),
|
||||
span: mk_sp(lo, e.span.hi),
|
||||
expr: e,
|
||||
ident: spanned(lo, hi, fieldname),
|
||||
span: mk_sp(lo, expr.span.hi),
|
||||
expr: expr,
|
||||
is_shorthand: is_shorthand,
|
||||
})
|
||||
}
|
||||
|
||||
|
@ -1893,8 +1893,10 @@ impl<'a> State<'a> {
|
||||
&fields[..],
|
||||
|s, field| {
|
||||
try!(s.ibox(INDENT_UNIT));
|
||||
try!(s.print_ident(field.ident.node));
|
||||
try!(s.word_space(":"));
|
||||
if !field.is_shorthand {
|
||||
try!(s.print_ident(field.ident.node));
|
||||
try!(s.word_space(":"));
|
||||
}
|
||||
try!(s.print_expr(&field.expr));
|
||||
s.end()
|
||||
},
|
||||
|
24
src/test/compile-fail/feature-gate-field-init-shorthand.rs
Normal file
24
src/test/compile-fail/feature-gate-field-init-shorthand.rs
Normal file
@ -0,0 +1,24 @@
|
||||
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
|
||||
// file at the top-level directory of this distribution and at
|
||||
// http://rust-lang.org/COPYRIGHT.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
||||
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||
// option. This file may not be copied, modified, or distributed
|
||||
// except according to those terms.
|
||||
|
||||
struct Foo {
|
||||
x: i32,
|
||||
y: bool,
|
||||
z: i32
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let (x, y, z) = (1, true, 2);
|
||||
let _ = Foo {
|
||||
x, //~ ERROR struct field shorthands are unstable
|
||||
y: y,
|
||||
z //~ ERROR struct field shorthands are unstable
|
||||
};
|
||||
}
|
24
src/test/compile-fail/struct-fields-shorthand-unresolved.rs
Normal file
24
src/test/compile-fail/struct-fields-shorthand-unresolved.rs
Normal file
@ -0,0 +1,24 @@
|
||||
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
|
||||
// file at the top-level directory of this distribution and at
|
||||
// http://rust-lang.org/COPYRIGHT.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
||||
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||
// option. This file may not be copied, modified, or distributed
|
||||
// except according to those terms.
|
||||
|
||||
#![feature(field_init_shorthand)]
|
||||
|
||||
struct Foo {
|
||||
x: i32,
|
||||
y: i32
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let x = 0;
|
||||
let foo = Foo {
|
||||
x,
|
||||
y //~ ERROR unresolved name `y`
|
||||
};
|
||||
}
|
24
src/test/compile-fail/struct-fields-shorthand.rs
Normal file
24
src/test/compile-fail/struct-fields-shorthand.rs
Normal file
@ -0,0 +1,24 @@
|
||||
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
|
||||
// file at the top-level directory of this distribution and at
|
||||
// http://rust-lang.org/COPYRIGHT.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
||||
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||
// option. This file may not be copied, modified, or distributed
|
||||
// except according to those terms.
|
||||
|
||||
#![feature(field_init_shorthand)]
|
||||
|
||||
struct Foo {
|
||||
x: i32,
|
||||
y: i32
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let (x, y, z) = (0, 1, 2);
|
||||
let foo = Foo {
|
||||
x, y, z //~ ERROR struct `Foo` has no field named `z`
|
||||
};
|
||||
}
|
||||
|
@ -18,5 +18,5 @@ fn removed_with() {
|
||||
|
||||
let a = S { foo: (), bar: () };
|
||||
let b = S { foo: (), with a };
|
||||
//~^ ERROR expected `:`, found `a`
|
||||
//~^ ERROR expected one of `,` or `}`, found `a`
|
||||
}
|
||||
|
19
src/test/parse-fail/struct-field-numeric-shorthand.rs
Normal file
19
src/test/parse-fail/struct-field-numeric-shorthand.rs
Normal file
@ -0,0 +1,19 @@
|
||||
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
|
||||
// file at the top-level directory of this distribution and at
|
||||
// http://rust-lang.org/COPYRIGHT.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
||||
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||
// option. This file may not be copied, modified, or distributed
|
||||
// except according to those terms.
|
||||
|
||||
// compile-flags: -Z parse-only
|
||||
|
||||
#![feature(field_init_shorthand)]
|
||||
|
||||
struct Rgb(u8, u8, u8);
|
||||
|
||||
fn main() {
|
||||
let _ = Rgb { 0, 1, 2 }; //~ ERROR expected identifier, found `0`
|
||||
}
|
37
src/test/run-pass/struct-field-shorthand.rs
Normal file
37
src/test/run-pass/struct-field-shorthand.rs
Normal file
@ -0,0 +1,37 @@
|
||||
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
|
||||
// file at the top-level directory of this distribution and at
|
||||
// http://rust-lang.org/COPYRIGHT.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
||||
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||
// option. This file may not be copied, modified, or distributed
|
||||
// except according to those terms.
|
||||
|
||||
#![feature(field_init_shorthand)]
|
||||
|
||||
struct Foo {
|
||||
x: i32,
|
||||
y: bool,
|
||||
z: i32
|
||||
}
|
||||
|
||||
struct Bar {
|
||||
x: i32
|
||||
}
|
||||
|
||||
pub fn main() {
|
||||
let (x, y, z) = (1, true, 2);
|
||||
let a = Foo { x, y: y, z };
|
||||
assert_eq!(a.x, x);
|
||||
assert_eq!(a.y, y);
|
||||
assert_eq!(a.z, z);
|
||||
|
||||
let b = Bar { x, };
|
||||
assert_eq!(b.x, x);
|
||||
|
||||
let c = Foo { z, y, x };
|
||||
assert_eq!(c.x, x);
|
||||
assert_eq!(c.y, y);
|
||||
assert_eq!(c.z, z);
|
||||
}
|
Loading…
Reference in New Issue
Block a user