dprint-core

Rust crate to help build a code formatter.

Api

Use:

rust let print_items = ...; // parsed out IR (see example below) let result = dprint::print(print_items, PrintOptions { indent_width: 4, max_width: 10, // Set this to true while testing. It runs additional validation on the // strings to ensure the print items are being parsed out correctly. is_testing: false, use_tabs: false, newline_kind: "\n", });

Or do the steps individually:

```rust // Send the print items to be transformed into write items. let writeitems = dprint::getwriteitems(printitems, GetWriteItemsOptions { indentwidth: 4, maxwidth: 10, is_testing: false, });

// Write out the write items to a string. let result = dprint::printwriteitems(writeitems, PrintWriteItemsOptions { usetabs: false, newlinekind: "\n", indentwidth: 4, }) ```

It is useful to only do the first step of getting the "write items" when using this library in WebAssembly. That allows for avoiding to copy/encode over the strings from JavaScript to Rust and backā€”all the strings can remain in JS. An example of this can be seen in the @dprint/rust-printer package.

Example

This reimplements the example from overview.md, but in rust.

Given the following AST nodes:

```rust enum Node<'a> { ArrayLiteralExpression(&'a ArrayLiteralExpression), ArrayElement(&'a ArrayElement), }

[derive(Clone)]

struct Position { /// Line number in the original source code. pub linenumber: u32, /// Column number in the original source code. pub columnnumber: u32, }

[derive(Clone)]

struct ArrayLiteralExpression { pub position: Position, pub elements: Vec, }

[derive(Clone)]

struct ArrayElement { pub position: Position, pub text: String, } ```

With the following expected outputs (when max line width configured in printer is 10):

```ts // input [a , b , c ] // output [a, b, c]

// input [four, four, four] // output (since it exceeds the line width of 10) [ four, four, four ]

// input [ four] // output (since first element was placed on a different line) [ four ] ```

Here's some example IR generation:

```rust use dprint::*;

pub fn format(expr: &ArrayLiteralExpression) -> String { // parse out the print items from the AST let printitems = parsenode(Node::ArrayLiteralExpression(expr));

// print them
dprint::print(print_items, PrintOptions {
    indent_width: 4,
    max_width: 10,
    use_tabs: false,
    newline_kind: "\n",
})

}

// node parsing functions

fn parse_node(node: Node) -> PrintItems { // in a real implementation this function would deal with surrounding comments

match node {
    Node::ArrayLiteralExpression(expr) => parse_array_literal_expression(&expr),
    Node::ArrayElement(array_element) => parse_array_element(&array_element),
}

}

fn parsearrayliteralexpression(expr: &ArrayLiteralExpression) -> PrintItems { let mut items = PrintItems::new(); let startinfo = Info::new("start"); let endinfo = Info::new("end"); let ismultiplelines = createismultiplelinesresolver( expr.position.clone(), expr.elements.iter().map(|e| e.position.clone()).collect(), startinfo, end_info );

items.push_info(start_info);

items.push_str("[");
items.extend(parser_helpers::if_true(
    "arrayStartNewLine",
    is_multiple_lines.clone(),
    Signal::NewLine.into()
));

let parsed_elements = parse_elements(&expr.elements, &is_multiple_lines).into_rc_path();
items.push_condition(Condition::new("indentIfMultipleLines", ConditionProperties {
    condition: Box::new(is_multiple_lines.clone()),
    true_path: Some(parser_helpers::with_indent(parsed_elements.clone().into())),
    false_path: Some(parsed_elements.into()),
}).into());

items.extend(parser_helpers::if_true(
    "arrayEndNewLine",
    is_multiple_lines,
    Signal::NewLine.into()
));
items.push_str("]");

items.push_info(end_info);

return items;

fn parse_elements(
    elements: &Vec<ArrayElement>,
    is_multiple_lines: &(impl Fn(&mut ConditionResolverContext) -> Option<bool> + Clone + 'static)
) -> PrintItems {
    let mut items = PrintItems::new();
    let elements_len = elements.len();

    for (i, elem) in elements.iter().enumerate() {
        items.extend(parse_node(Node::ArrayElement(elem)));

        if i < elements_len - 1 {
            items.push_str(",");
            items.extend(parser_helpers::if_true_or(
                "afterCommaSeparator",
                is_multiple_lines.clone(),
                Signal::NewLine.into(),
                Signal::SpaceOrNewLine.into()
            ));
        }
    }

    items
}

}

fn parsearrayelement(element: &ArrayElement) -> PrintItems { (&element.text).into() }

// helper functions

fn createismultiplelinesresolver( parentposition: Position, childpositions: Vec, startinfo: Info, endinfo: Info ) -> impl Fn(&mut ConditionResolverContext) -> Option + Clone + 'static { // todo: this could be more efficient only only use references and avoid the clones // I'm too lazy to update this sample, but it should help you get the idea. return move |conditioncontext: &mut ConditionResolverContext| { // no items, so format on the same line if childpositions.len() == 0 { return Some(false); } // first child is on a different line than the start of the parent // so format all the children as multi-line if parentposition.linenumber < childpositions[0].linenumber { return Some(true); }

    // check if it spans multiple lines, and if it does then make it multi-line
    condition_resolvers::is_multiple_lines(condition_context, &start_info, &end_info)
};

} ```