Skip to content

Commit

Permalink
add comments and use fixed find_only_token_in_range
Browse files Browse the repository at this point in the history
  • Loading branch information
davidszotten committed Jul 10, 2023
1 parent c015187 commit 476a978
Showing 1 changed file with 41 additions and 29 deletions.
70 changes: 41 additions & 29 deletions crates/ruff_python_formatter/src/comments/placement.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
use std::cmp::Ordering;

use ruff_text_size::{TextRange, TextSize};
use ruff_text_size::TextRange;
use rustpython_parser::ast;
use rustpython_parser::ast::{Expr, ExprIfExp, ExprSlice, Ranged};

Expand Down Expand Up @@ -1247,17 +1247,19 @@ fn handle_comprehension_comment<'a>(
comment: DecoratedComment<'a>,
locator: &Locator,
) -> CommentPlacement<'a> {
fn find_if_token(locator: &Locator, start: TextSize, end: TextSize) -> Option<TextSize> {
let mut tokens =
SimpleTokenizer::new(locator.contents(), TextRange::new(start, end)).skip_trivia();
tokens.next().map(|t| t.start())
}

let AnyNodeRef::Comprehension(comprehension) = comment.enclosing_node() else {
return CommentPlacement::Default(comment);
};
let is_own_line = comment.line_position().is_own_line();

// Comments between the `for` and target
// ```python
// [
// a
// for # attache as dangling on the comprehension
// b in c
// ]
// ```
if comment.slice().end() < comprehension.target.range().start() {
return if is_own_line {
// own line comments are correctly assigned as leading the target
Expand All @@ -1268,20 +1270,24 @@ fn handle_comprehension_comment<'a>(
};
}

let mut tokens = SimpleTokenizer::new(
locator.contents(),
let in_token = find_only_token_in_range(
TextRange::new(
comprehension.target.range().end(),
comprehension.iter.range().start(),
),
)
.skip_trivia();
let Some(in_token) = tokens.next() else {
// Should always have an `in`
debug_assert!(false);
return CommentPlacement::Default(comment);
};
locator,
TokenKind::In,
);

// Comments between the target and the `in`
// ```python
// [
// a for b
// # attach as dangling on the target
// # (to be rendered as leading on the "in")
// in c
// ]
// ```
if comment.slice().start() < in_token.start() {
// attach as dangling comments on the target
// (to be rendered as leading on the "in")
Expand All @@ -1293,14 +1299,19 @@ fn handle_comprehension_comment<'a>(
};
}

// Comments between the `in` and the iter
// ```python
// [
// a for b
// in # attach as dangling on the iter
// c
// ]
// ```
if comment.slice().start() < comprehension.iter.range().start() {
// attach as dangling comments on the iter

return if is_own_line {
// after the `in` and own line: leading on the iter
CommentPlacement::leading((&comprehension.iter).into(), comment)
CommentPlacement::Default(comment)
} else {
// after the `in` but same line, turn into trailin on the `in` token
// after the `in` but same line, turn into trailing on the `in` token
CommentPlacement::dangling((&comprehension.iter).into(), comment)
};
}
Expand All @@ -1323,17 +1334,18 @@ fn handle_comprehension_comment<'a>(
// # above g
// g # g
// ]
let Some(if_token_start) = find_if_token(locator, last_end, if_node.range().end()) else {
// there should always be an `if` here
debug_assert!(false);
return CommentPlacement::Default(comment);
};

let if_token = find_only_token_in_range(
TextRange::new(last_end, if_node.range().start()),
locator,
TokenKind::If,
);
if is_own_line {
if last_end < comment.slice().start() && comment.slice().start() < if_token_start {
if last_end < comment.slice().start() && comment.slice().start() < if_token.start() {
return CommentPlacement::dangling((if_node).into(), comment);
}
} else {
if if_token_start < comment.slice().start()
if if_token.start() < comment.slice().start()
&& comment.slice().start() < if_node.range().start()
{
return CommentPlacement::dangling((if_node).into(), comment);
Expand All @@ -1342,7 +1354,7 @@ fn handle_comprehension_comment<'a>(
last_end = if_node.range().end();
}

return CommentPlacement::Default(comment);
CommentPlacement::Default(comment)
}

/// Returns `true` if `right` is `Some` and `left` and `right` are referentially equal.
Expand Down

0 comments on commit 476a978

Please sign in to comment.