rust/crates/ra_assists/src/handlers/raw_string.rs

532 lines
12 KiB
Rust
Raw Normal View History

use std::borrow::Cow;
2019-10-27 03:32:39 -05:00
use ra_syntax::{
ast::{self, HasQuotes, HasStringValue},
2020-02-27 10:19:53 -06:00
AstToken,
2019-10-27 03:32:39 -05:00
SyntaxKind::{RAW_STRING, STRING},
2020-04-24 16:40:41 -05:00
TextSize,
2019-10-27 03:32:39 -05:00
};
2019-09-20 10:55:59 -05:00
2020-06-28 17:36:05 -05:00
use crate::{AssistContext, AssistId, AssistKind, Assists};
2019-09-20 10:55:59 -05:00
2019-10-27 04:22:53 -05:00
// Assist: make_raw_string
//
// Adds `r#` to a plain string literal.
//
// ```
// fn main() {
// "Hello,<|> World!";
// }
// ```
// ->
// ```
// fn main() {
// r#"Hello, World!"#;
// }
// ```
pub(crate) fn make_raw_string(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let token = ctx.find_token_at_offset(STRING).and_then(ast::String::cast)?;
let value = token.value()?;
let target = token.syntax().text_range();
2020-06-28 17:36:05 -05:00
acc.add(
2020-07-02 16:48:35 -05:00
AssistId("make_raw_string", AssistKind::RefactorRewrite),
2020-06-28 17:36:05 -05:00
"Rewrite as raw string",
target,
|edit| {
let max_hash_streak = count_hashes(&value);
let hashes = "#".repeat(max_hash_streak + 1);
if matches!(value, Cow::Borrowed(_)) {
// Avoid replacing the whole string to better position the cursor.
edit.insert(token.syntax().text_range().start(), format!("r{}", hashes));
edit.insert(token.syntax().text_range().end(), format!("{}", hashes));
} else {
edit.replace(
token.syntax().text_range(),
format!("r{}\"{}\"{}", hashes, value, hashes),
);
2020-06-28 17:36:05 -05:00
}
},
)
2019-09-20 10:55:59 -05:00
}
2019-10-27 04:22:53 -05:00
// Assist: make_usual_string
//
// Turns a raw string into a plain string.
//
// ```
// fn main() {
// r#"Hello,<|> "World!""#;
// }
// ```
// ->
// ```
// fn main() {
// "Hello, \"World!\"";
// }
// ```
pub(crate) fn make_usual_string(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let token = ctx.find_token_at_offset(RAW_STRING).and_then(ast::RawString::cast)?;
let value = token.value()?;
let target = token.syntax().text_range();
2020-06-28 17:36:05 -05:00
acc.add(
2020-07-02 16:48:35 -05:00
AssistId("make_usual_string", AssistKind::RefactorRewrite),
2020-06-28 17:36:05 -05:00
"Rewrite as regular string",
target,
|edit| {
// parse inside string to escape `"`
let escaped = value.escape_default().to_string();
if let Some(offsets) = token.quote_offsets() {
if token.text()[offsets.contents - token.syntax().text_range().start()] == escaped {
edit.replace(offsets.quotes.0, "\"");
edit.replace(offsets.quotes.1, "\"");
return;
}
}
2020-06-28 17:36:05 -05:00
edit.replace(token.syntax().text_range(), format!("\"{}\"", escaped));
},
)
2019-09-20 10:55:59 -05:00
}
2019-10-27 04:22:53 -05:00
// Assist: add_hash
//
// Adds a hash to a raw string literal.
//
// ```
// fn main() {
// r#"Hello,<|> World!"#;
// }
// ```
// ->
// ```
// fn main() {
// r##"Hello, World!"##;
// }
// ```
pub(crate) fn add_hash(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let token = ctx.find_token_at_offset(RAW_STRING)?;
let target = token.text_range();
2020-07-02 16:48:35 -05:00
acc.add(AssistId("add_hash", AssistKind::Refactor), "Add # to raw string", target, |edit| {
2020-04-24 16:40:41 -05:00
edit.insert(token.text_range().start() + TextSize::of('r'), "#");
2019-10-27 03:32:39 -05:00
edit.insert(token.text_range().end(), "#");
})
2019-09-20 10:55:59 -05:00
}
2019-10-27 04:22:53 -05:00
// Assist: remove_hash
//
// Removes a hash from a raw string literal.
//
// ```
// fn main() {
// r#"Hello,<|> World!"#;
// }
// ```
// ->
// ```
// fn main() {
// r"Hello, World!";
// }
// ```
pub(crate) fn remove_hash(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let token = ctx.find_token_at_offset(RAW_STRING)?;
2019-09-20 10:55:59 -05:00
let text = token.text().as_str();
if text.starts_with("r\"") {
// no hash to remove
return None;
}
let target = token.text_range();
2020-06-28 17:36:05 -05:00
acc.add(
2020-07-02 16:48:35 -05:00
AssistId("remove_hash", AssistKind::RefactorRewrite),
2020-06-28 17:36:05 -05:00
"Remove hash from raw string",
target,
|edit| {
let result = &text[2..text.len() - 1];
let result = if result.starts_with('\"') {
// FIXME: this logic is wrong, not only the last has has to handled specially
// no more hash, escape
let internal_str = &result[1..result.len() - 1];
format!("\"{}\"", internal_str.escape_default().to_string())
} else {
result.to_owned()
};
edit.replace(token.text_range(), format!("r{}", result));
},
)
2019-09-20 10:55:59 -05:00
}
2019-10-27 03:32:39 -05:00
fn count_hashes(s: &str) -> usize {
let mut max_hash_streak = 0usize;
for idx in s.match_indices("\"#").map(|(i, _)| i) {
let (_, sub) = s.split_at(idx + 1);
let nb_hash = sub.chars().take_while(|c| *c == '#').count();
if nb_hash > max_hash_streak {
max_hash_streak = nb_hash;
}
}
max_hash_streak
}
2019-09-20 10:55:59 -05:00
#[cfg(test)]
mod test {
use super::*;
2020-05-06 03:16:55 -05:00
use crate::tests::{check_assist, check_assist_not_applicable, check_assist_target};
2019-09-20 10:55:59 -05:00
#[test]
fn make_raw_string_target() {
check_assist_target(
make_raw_string,
r#"
fn f() {
let s = <|>"random\nstring";
2019-09-20 10:55:59 -05:00
}
"#,
r#""random\nstring""#,
2019-09-20 10:55:59 -05:00
);
}
#[test]
fn make_raw_string_works() {
check_assist(
make_raw_string,
r#"
2020-06-23 17:30:34 -05:00
fn f() {
let s = <|>"random\nstring";
}
"#,
r##"
2020-06-23 17:30:34 -05:00
fn f() {
let s = r#"random
string"#;
2020-06-23 17:30:34 -05:00
}
"##,
2019-09-20 10:55:59 -05:00
)
}
2019-10-27 03:32:39 -05:00
#[test]
fn make_raw_string_works_inside_macros() {
check_assist(
make_raw_string,
r#"
fn f() {
format!(<|>"x = {}", 92)
}
"#,
r##"
fn f() {
format!(r#"x = {}"#, 92)
2019-10-27 03:32:39 -05:00
}
"##,
)
}
2019-09-20 10:55:59 -05:00
#[test]
fn make_raw_string_hashes_inside_works() {
2019-09-20 10:55:59 -05:00
check_assist(
make_raw_string,
r###"
2020-06-23 17:30:34 -05:00
fn f() {
let s = <|>"#random##\nstring";
}
"###,
r####"
2020-06-23 17:30:34 -05:00
fn f() {
let s = r#"#random##
string"#;
2020-06-23 17:30:34 -05:00
}
"####,
2019-09-20 10:55:59 -05:00
)
}
#[test]
fn make_raw_string_closing_hashes_inside_works() {
check_assist(
make_raw_string,
r###"
2020-06-23 17:30:34 -05:00
fn f() {
let s = <|>"#random\"##\nstring";
}
"###,
r####"
2020-06-23 17:30:34 -05:00
fn f() {
let s = r###"#random"##
string"###;
2020-06-23 17:30:34 -05:00
}
"####,
)
}
2019-09-20 10:55:59 -05:00
#[test]
fn make_raw_string_nothing_to_unescape_works() {
check_assist(
2019-09-20 10:55:59 -05:00
make_raw_string,
r#"
fn f() {
let s = <|>"random string";
2019-09-20 10:55:59 -05:00
}
"#,
r##"
fn f() {
let s = r#"random string"#;
}
"##,
)
2019-09-20 10:55:59 -05:00
}
#[test]
fn make_raw_string_not_works_on_partial_string() {
check_assist_not_applicable(
make_raw_string,
r#"
fn f() {
let s = "foo<|>
}
"#,
)
}
#[test]
fn make_usual_string_not_works_on_partial_string() {
check_assist_not_applicable(
make_usual_string,
r#"
fn main() {
let s = r#"bar<|>
}
"#,
)
}
2019-09-20 10:55:59 -05:00
#[test]
fn add_hash_target() {
check_assist_target(
add_hash,
r#"
fn f() {
let s = <|>r"random string";
}
"#,
r#"r"random string""#,
);
}
#[test]
fn add_hash_works() {
check_assist(
add_hash,
r#"
fn f() {
let s = <|>r"random string";
}
"#,
r##"
fn f() {
let s = r#"random string"#;
2019-09-20 10:55:59 -05:00
}
"##,
)
}
#[test]
fn add_more_hash_works() {
check_assist(
add_hash,
r##"
fn f() {
let s = <|>r#"random"string"#;
}
"##,
r###"
fn f() {
let s = r##"random"string"##;
2019-09-20 10:55:59 -05:00
}
"###,
)
}
#[test]
fn add_hash_not_works() {
check_assist_not_applicable(
add_hash,
r#"
fn f() {
let s = <|>"random string";
}
"#,
);
}
#[test]
fn remove_hash_target() {
check_assist_target(
remove_hash,
r##"
fn f() {
let s = <|>r#"random string"#;
}
"##,
r##"r#"random string"#"##,
);
}
#[test]
fn remove_hash_works() {
check_assist(
remove_hash,
r##"
fn f() {
let s = <|>r#"random string"#;
}
"##,
r#"
fn f() {
let s = r"random string";
2019-09-20 10:55:59 -05:00
}
"#,
)
}
#[test]
fn remove_hash_with_quote_works() {
check_assist(
remove_hash,
r##"
fn f() {
let s = <|>r#"random"str"ing"#;
}
"##,
r#"
fn f() {
let s = r"random\"str\"ing";
2019-09-20 10:55:59 -05:00
}
"#,
)
}
#[test]
fn remove_more_hash_works() {
check_assist(
remove_hash,
r###"
fn f() {
let s = <|>r##"random string"##;
}
"###,
r##"
fn f() {
let s = r#"random string"#;
2019-09-20 10:55:59 -05:00
}
"##,
)
}
#[test]
fn remove_hash_not_works() {
check_assist_not_applicable(
remove_hash,
r#"
fn f() {
let s = <|>"random string";
}
"#,
);
}
#[test]
fn remove_hash_no_hash_not_works() {
check_assist_not_applicable(
remove_hash,
r#"
fn f() {
let s = <|>r"random string";
}
"#,
);
}
#[test]
fn make_usual_string_target() {
check_assist_target(
make_usual_string,
r##"
fn f() {
let s = <|>r#"random string"#;
}
"##,
r##"r#"random string"#"##,
);
}
#[test]
fn make_usual_string_works() {
check_assist(
make_usual_string,
r##"
fn f() {
let s = <|>r#"random string"#;
}
"##,
r#"
fn f() {
let s = "random string";
2019-09-20 10:55:59 -05:00
}
"#,
)
}
#[test]
fn make_usual_string_with_quote_works() {
check_assist(
make_usual_string,
r##"
fn f() {
let s = <|>r#"random"str"ing"#;
}
"##,
r#"
fn f() {
let s = "random\"str\"ing";
2019-09-20 10:55:59 -05:00
}
"#,
)
}
#[test]
fn make_usual_string_more_hash_works() {
check_assist(
make_usual_string,
r###"
fn f() {
let s = <|>r##"random string"##;
}
"###,
r##"
fn f() {
let s = "random string";
2019-09-20 10:55:59 -05:00
}
"##,
)
}
#[test]
fn make_usual_string_not_works() {
check_assist_not_applicable(
make_usual_string,
r#"
fn f() {
let s = <|>"random string";
}
"#,
);
}
#[test]
fn count_hashes_test() {
assert_eq!(0, count_hashes("abc"));
assert_eq!(0, count_hashes("###"));
assert_eq!(1, count_hashes("\"#abc"));
assert_eq!(0, count_hashes("#abc"));
assert_eq!(2, count_hashes("#ab\"##c"));
assert_eq!(4, count_hashes("#ab\"##\"####c"));
}
2019-09-20 10:55:59 -05:00
}