Skip to content

Commit bc4414e

Browse files
authored
Merge pull request rust-lang#3101 from nrc/pair-newline
Simplify multi-lining binop exprs
2 parents 075aa90 + 7be173e commit bc4414e

File tree

8 files changed

+70
-68
lines changed

8 files changed

+70
-68
lines changed

src/expr.rs

+5-4
Original file line numberDiff line numberDiff line change
@@ -898,10 +898,11 @@ impl<'a> ControlFlow<'a> {
898898
let trial = self.rewrite_single_line(&pat_expr_string, context, shape.width);
899899

900900
if let Some(cond_str) = trial {
901-
if cond_str.len() <= context
902-
.config
903-
.width_heuristics()
904-
.single_line_if_else_max_width
901+
if cond_str.len()
902+
<= context
903+
.config
904+
.width_heuristics()
905+
.single_line_if_else_max_width
905906
{
906907
return Some((cond_str, 0));
907908
}

src/lists.rs

+6-4
Original file line numberDiff line numberDiff line change
@@ -146,10 +146,12 @@ impl ListItem {
146146
}
147147

148148
pub fn is_different_group(&self) -> bool {
149-
self.inner_as_ref().contains('\n') || self.pre_comment.is_some() || self
150-
.post_comment
151-
.as_ref()
152-
.map_or(false, |s| s.contains('\n'))
149+
self.inner_as_ref().contains('\n')
150+
|| self.pre_comment.is_some()
151+
|| self
152+
.post_comment
153+
.as_ref()
154+
.map_or(false, |s| s.contains('\n'))
153155
}
154156

155157
pub fn is_multiline(&self) -> bool {

src/macros.rs

+18-16
Original file line numberDiff line numberDiff line change
@@ -1168,22 +1168,24 @@ fn indent_macro_snippet(
11681168
.min()?;
11691169

11701170
Some(
1171-
first_line + "\n" + &trimmed_lines
1172-
.iter()
1173-
.map(
1174-
|&(trimmed, ref line, prefix_space_width)| match prefix_space_width {
1175-
_ if !trimmed => line.to_owned(),
1176-
Some(original_indent_width) => {
1177-
let new_indent_width = indent.width() + original_indent_width
1178-
.saturating_sub(min_prefix_space_width);
1179-
let new_indent = Indent::from_width(context.config, new_indent_width);
1180-
format!("{}{}", new_indent.to_string(context.config), line)
1181-
}
1182-
None => String::new(),
1183-
},
1184-
)
1185-
.collect::<Vec<_>>()
1186-
.join("\n"),
1171+
first_line
1172+
+ "\n"
1173+
+ &trimmed_lines
1174+
.iter()
1175+
.map(
1176+
|&(trimmed, ref line, prefix_space_width)| match prefix_space_width {
1177+
_ if !trimmed => line.to_owned(),
1178+
Some(original_indent_width) => {
1179+
let new_indent_width = indent.width()
1180+
+ original_indent_width.saturating_sub(min_prefix_space_width);
1181+
let new_indent = Indent::from_width(context.config, new_indent_width);
1182+
format!("{}{}", new_indent.to_string(context.config), line)
1183+
}
1184+
None => String::new(),
1185+
},
1186+
)
1187+
.collect::<Vec<_>>()
1188+
.join("\n"),
11871189
)
11881190
}
11891191

src/pairs.rs

+20-35
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ pub(crate) fn rewrite_all_pairs(
4343
context: &RewriteContext,
4444
) -> Option<String> {
4545
// First we try formatting on one line.
46-
if let Some(list) = expr.flatten(context, false) {
46+
if let Some(list) = expr.flatten(false) {
4747
if let Some(r) = rewrite_pairs_one_line(&list, shape, context) {
4848
return Some(r);
4949
}
@@ -53,7 +53,7 @@ pub(crate) fn rewrite_all_pairs(
5353
// to only flatten pairs with the same operator, that way we don't
5454
// necessarily need one line per sub-expression, but we don't do anything
5555
// too funny wrt precedence.
56-
expr.flatten(context, true)
56+
expr.flatten(true)
5757
.and_then(|list| rewrite_pairs_multiline(list, shape, context))
5858
}
5959

@@ -83,33 +83,22 @@ fn rewrite_pairs_one_line<T: Rewrite>(
8383
result.push(' ');
8484
}
8585

86+
let prefix_len = result.len();
8687
let last = list.list.last().unwrap();
8788
let cur_shape = base_shape.offset_left(last_line_width(&result))?;
88-
let rewrite = last.rewrite(context, cur_shape)?;
89-
result.push_str(&rewrite);
89+
let last_rewrite = last.rewrite(context, cur_shape)?;
90+
result.push_str(&last_rewrite);
9091

9192
if first_line_width(&result) > shape.width {
9293
return None;
9394
}
9495

95-
// Check the last expression in the list. We let this expression go over
96-
// multiple lines, but we check that if this is necessary, then we can't
97-
// do better using multi-line formatting.
98-
if !is_single_line(&result) {
99-
let multiline_shape = shape.offset_left(list.separators.last().unwrap().len() + 1)?;
100-
let multiline_list: PairList<T> = PairList {
101-
list: vec![last],
102-
separators: vec![],
103-
separator_place: list.separator_place,
104-
};
105-
// Format as if we were multi-line.
106-
if let Some(rewrite) = rewrite_pairs_multiline(multiline_list, multiline_shape, context) {
107-
// Also, don't let expressions surrounded by parens go multi-line,
108-
// this looks really bad.
109-
if rewrite.starts_with('(') || is_single_line(&rewrite) {
110-
return None;
111-
}
112-
}
96+
// Check the last expression in the list. We sometimes let this expression
97+
// go over multiple lines, but we check for some ugly conditions.
98+
if !(is_single_line(&result) || last_rewrite.starts_with('{'))
99+
&& (last_rewrite.starts_with('(') || prefix_len > context.config.tab_spaces())
100+
{
101+
return None;
113102
}
114103

115104
wrap_str(result, context.config.max_width(), shape)
@@ -215,11 +204,12 @@ where
215204
// If the length of the lhs is equal to or shorter than the tab width or
216205
// the rhs looks like block expression, we put the rhs on the same
217206
// line with the lhs even if the rhs is multi-lined.
218-
let allow_same_line = lhs_result.len() <= tab_spaces || rhs_result
219-
.lines()
220-
.next()
221-
.map(|first_line| first_line.ends_with('{'))
222-
.unwrap_or(false);
207+
let allow_same_line = lhs_result.len() <= tab_spaces
208+
|| rhs_result
209+
.lines()
210+
.next()
211+
.map(|first_line| first_line.ends_with('{'))
212+
.unwrap_or(false);
223213
if !rhs_result.contains('\n') || allow_same_line {
224214
let one_line_width = last_line_width(&lhs_result)
225215
+ pp.infix.len()
@@ -272,19 +262,18 @@ trait FlattenPair: Rewrite + Sized {
272262
// operator into the list. E.g,, if the source is `a * b + c`, if `_same_op`
273263
// is true, we make `[(a * b), c]` if `_same_op` is false, we make
274264
// `[a, b, c]`
275-
fn flatten(&self, _context: &RewriteContext, _same_op: bool) -> Option<PairList<Self>> {
265+
fn flatten(&self, _same_op: bool) -> Option<PairList<Self>> {
276266
None
277267
}
278268
}
279269

280270
struct PairList<'a, 'b, T: Rewrite + 'b> {
281271
list: Vec<&'b T>,
282272
separators: Vec<&'a str>,
283-
separator_place: SeparatorPlace,
284273
}
285274

286275
impl FlattenPair for ast::Expr {
287-
fn flatten(&self, context: &RewriteContext, same_op: bool) -> Option<PairList<ast::Expr>> {
276+
fn flatten(&self, same_op: bool) -> Option<PairList<ast::Expr>> {
288277
let top_op = match self.node {
289278
ast::ExprKind::Binary(op, _, _) => op.node,
290279
_ => return None,
@@ -320,11 +309,7 @@ impl FlattenPair for ast::Expr {
320309
}
321310

322311
assert_eq!(list.len() - 1, separators.len());
323-
Some(PairList {
324-
list,
325-
separators,
326-
separator_place: context.config.binop_separator(),
327-
})
312+
Some(PairList { list, separators })
328313
}
329314
}
330315

src/string.rs

+6-5
Original file line numberDiff line numberDiff line change
@@ -291,11 +291,12 @@ fn break_string(max_chars: usize, trim_end: bool, line_end: &str, input: &[&str]
291291
return break_at(max_chars - 1);
292292
}
293293
if let Some(url_index_end) = detect_url(input, max_chars) {
294-
let index_plus_ws = url_index_end + input[url_index_end..]
295-
.iter()
296-
.skip(1)
297-
.position(|grapheme| not_whitespace_except_line_feed(grapheme))
298-
.unwrap_or(0);
294+
let index_plus_ws = url_index_end
295+
+ input[url_index_end..]
296+
.iter()
297+
.skip(1)
298+
.position(|grapheme| not_whitespace_except_line_feed(grapheme))
299+
.unwrap_or(0);
299300
return if trim_end {
300301
SnippetState::LineEnd(
301302
input[..=url_index_end].join("").to_string(),

src/utils.rs

+5-4
Original file line numberDiff line numberDiff line change
@@ -326,10 +326,11 @@ pub fn mk_sp(lo: BytePos, hi: BytePos) -> Span {
326326
// Return true if the given span does not intersect with file lines.
327327
macro_rules! out_of_file_lines_range {
328328
($self:ident, $span:expr) => {
329-
!$self.config.file_lines().is_all() && !$self
330-
.config
331-
.file_lines()
332-
.intersects(&$self.source_map.lookup_line_range($span))
329+
!$self.config.file_lines().is_all()
330+
&& !$self
331+
.config
332+
.file_lines()
333+
.intersects(&$self.source_map.lookup_line_range($span))
333334
};
334335
}
335336

tests/source/chains.rs

+5
Original file line numberDiff line numberDiff line change
@@ -259,3 +259,8 @@ fn issue_2773() {
259259
None
260260
});
261261
}
262+
263+
fn issue_3034() {
264+
disallowed_headers.iter().any(|header| *header == name) ||
265+
disallowed_header_prefixes.iter().any(|prefix| name.starts_with(prefix))
266+
}

tests/target/chains.rs

+5
Original file line numberDiff line numberDiff line change
@@ -299,3 +299,8 @@ fn issue_2773() {
299299
None
300300
});
301301
}
302+
303+
fn issue_3034() {
304+
disallowed_headers.iter().any(|header| *header == name)
305+
|| disallowed_header_prefixes.iter().any(|prefix| name.starts_with(prefix))
306+
}

0 commit comments

Comments
 (0)