aboutsummaryrefslogtreecommitdiff
path: root/crates/ra_editor/src/assists/change_visibility.rs
blob: 98c218f329bbeeb0e9e453bd557e6d0e5dca8693 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
use ra_text_edit::TextEditBuilder;
use ra_syntax::{
    SourceFileNode,
    algo::find_leaf_at_offset,
    SyntaxKind::{VISIBILITY, FN_KW, MOD_KW, STRUCT_KW, ENUM_KW, TRAIT_KW, FN_DEF, MODULE, STRUCT_DEF, ENUM_DEF, TRAIT_DEF},
    TextUnit,
};

use crate::assists::LocalEdit;

pub fn change_visibility<'a>(
    file: &'a SourceFileNode,
    offset: TextUnit,
) -> Option<impl FnOnce() -> LocalEdit + 'a> {
    let syntax = file.syntax();

    let keyword = find_leaf_at_offset(syntax, offset).find(|leaf| match leaf.kind() {
        FN_KW | MOD_KW | STRUCT_KW | ENUM_KW | TRAIT_KW => true,
        _ => false,
    })?;
    let parent = keyword.parent()?;
    let def_kws = vec![FN_DEF, MODULE, STRUCT_DEF, ENUM_DEF, TRAIT_DEF];
    let node_start = parent.range().start();
    Some(move || {
        let mut edit = TextEditBuilder::new();

        if !def_kws.iter().any(|&def_kw| def_kw == parent.kind())
            || parent.children().any(|child| child.kind() == VISIBILITY)
        {
            return LocalEdit {
                label: "make pub crate".to_string(),
                edit: edit.finish(),
                cursor_position: Some(offset),
            };
        }

        edit.insert(node_start, "pub(crate) ".to_string());
        LocalEdit {
            label: "make pub crate".to_string(),
            edit: edit.finish(),
            cursor_position: Some(node_start),
        }
    })
}

#[cfg(test)]
mod tests {
    use super::*;
    use crate::test_utils::check_action;

    #[test]
    fn test_change_visibility() {
        check_action(
            "<|>fn foo() {}",
            "<|>pub(crate) fn foo() {}",
            |file, off| change_visibility(file, off).map(|f| f()),
        );
        check_action(
            "f<|>n foo() {}",
            "<|>pub(crate) fn foo() {}",
            |file, off| change_visibility(file, off).map(|f| f()),
        );
        check_action(
            "<|>struct Foo {}",
            "<|>pub(crate) struct Foo {}",
            |file, off| change_visibility(file, off).map(|f| f()),
        );
        check_action("<|>mod foo {}", "<|>pub(crate) mod foo {}", |file, off| {
            change_visibility(file, off).map(|f| f())
        });
        check_action(
            "<|>trait Foo {}",
            "<|>pub(crate) trait Foo {}",
            |file, off| change_visibility(file, off).map(|f| f()),
        );
        check_action("m<|>od {}", "<|>pub(crate) mod {}", |file, off| {
            change_visibility(file, off).map(|f| f())
        });
        check_action(
            "pub(crate) f<|>n foo() {}",
            "pub(crate) f<|>n foo() {}",
            |file, off| change_visibility(file, off).map(|f| f()),
        );
        check_action(
            "unsafe f<|>n foo() {}",
            "<|>pub(crate) unsafe fn foo() {}",
            |file, off| change_visibility(file, off).map(|f| f()),
        );
    }
}