fluent-impl Travis Build Status AppVeyor Build Status Crate

Documentation

A procedural macro that generates chaining methods from non-chaining ones in an impl block.

When applied to an impl block, #[fluent_impl] will scan all methods in the block in search for chain-able methods, and generate chaining methods from them.

Chain-able methods are the ones with &mut self as a first argument, and return nothing. That's it, there are no other restrictions.

Detailed Build Status

fluent_impl remains nightly-only, for now. The feature use_extern_macros is supposedly days away from stabilisation.

Travis

| Linux | OSX | |:----:|:----:| | linux-nightly | osx-nightly | | linux-beta | osx-beta | | linux-stable | osx-stable |

AppVeyor

Failure could be caused by a bug in compiletest-rs (I'm not sure). Feedback from Windows users would be greatly appreciated.

| Windows | x8664 | i686 | |:-------:|:------:|:----:| | nightly | nightly x86</em>64 | nightly i686 | | beta | beta x86<em>64 | beta i686 | | stable | satble x86</em>64 | stable i686 |

Usage

Add fluent-impl to the dependencies in Cargo.toml:

toml [dependencies] fluent-impl = "0.1"

Then add the following to the top of src/lib.rs:

``` rust ignore // Will be stabilized soon

![feature(useexternmacros)]

extern crate fluent_impl;

use fluentimpl::{fluentimpl, fluentimplopts};

```

Examples

If we have a simple struct with a simple impl block:

``` rust

[derive(Default, PartialEq, Debug)]

pub struct Simple { num: i32, }

impl Simple { // ... pub fn add_1(&mut self) { self.num +=1; } } ```

Then we add the macro attribute to the impl block:

``` rust ignore

[fluent_impl]

impl Simple { // ... pub fn add_1(&mut self) { self.num +=1; } } ```

The macro will generate a new impl block with the content:

``` rust ignore

[doc = "Chaining (fluent) methods for [Simple]."]

impl Simple { #[doc = "The chaining (fluent) equivalent of [add_1()].\n\n [add_1]: Simple::add1\n [add_1()]: Simple::add1"] pub fn withadd1(mut self) -> Self { self.add_1(); self } } ```

A full more involved example can be found bellow the Attribute Configuration section.

Attribute Configuration

#[fluent_impl] is configurable with comma-separated options passed to the attribute itself, and options passed to a method-level attribute #[fluent_impl_opts].

#[fluent_impl] Attribute Options

(inblock, non_public, prefix, impl_doc, doc)

impl block-level configuration.

### Example

rust ignore #[fluent_impl(inblock, non_public, prefix="chain_")] impl Simple { // ... }

### Options

#[fluent_impl_opts] Attribute Options

(inblock, non_public, skip, prefix, rename, name, doc)

Options passed to override block-level defaults, or set method-specific configurations.

Unlike #[fluent_impl], this attribute: 1. Applies to methods instead of impl blocks. 2. Can be passed multiple times to the same method if you please.

Example

``` rust ignore

[fluent_impl]

impl Simple { #[fluentimplopts(nonpublic, inblock)] #[fluentimplopts(prefix="chain", rename="added1")] fn add1(&mut self) { // ... } } ```

### Options

#### Inherited

Full Example

``` rust

![feature(useexternmacros)]

extern crate fluent_impl;

pub mod m { use fluentimpl::{fluentimpl, fluentimplopts}; use std::borrow::Borrow; use std::ops::AddAssign;

#[derive(PartialEq, Debug)]
pub struct TCounter(pub u32);

#[derive(PartialEq, Debug)]
pub struct St<A: AddAssign> {
    value: A,
    text: String,
}

#[fluent_impl]
// impl block with generic arguments works
impl<A: AddAssign> St<A> {
    // Constants (or any other items) in impl block are okay
    pub(crate) const C_TC: u32 = 100;

    pub fn new(value: A, text: String) -> Self {
        Self { value, text }
    }

    pub fn get_value(&self) -> &A {
        &self.value
    }

    pub fn get_text(&self) -> &str {
        &self.text
    }

    #[fluent_impl_opts(rename = "added_value")]
    // Destructuring patterns in method arguments are okay
    pub fn add_value(
        &mut self,
        to_be_added: A,
        TCounter(counter): &mut TCounter,
    ) {
        self.value += to_be_added;
        *counter += 1;
    }

    #[fluent_impl_opts(rename = "appended_text")]
    // Generic method arguments are okay
    pub fn append_text<S: Borrow<str>>(&mut self, arg: S) {
        self.text += arg.borrow();
    }

    #[fluent_impl_opts(rename = "appended_text_impl_trait")]
    // Needless to say, impl Trait method arguments are also okay
    pub fn append_text_impl_trait(&mut self, arg: impl Borrow<str>) {
        self.text += arg.borrow();
    }
}

}

fn main() { use m::{St, TCounter}; // ======== let mut tc1 = TCounter(St::::CTC); let mut s1 = St::new(0u32, "".into()); s1.appendtext("simple "); s1.appendtext::<&str>("turbo fish "); s1.appendtextimpltrait("impl trait"); s1.addvalue(5, &mut tc1); asserteq!(s1.gettext(), "simple turbo fish impl trait"); asserteq!(tc1, TCounter(St::::CTC + 1)); // ======== let mut tc2 = TCounter(St::::CTC); let s2 = St::new(0u32, "".into()) .withappendedtext("simple ") .withappendedtext::<&str>("turbo fish ") .withappendedtextimpltrait("impl trait") .withaddedvalue(5, &mut tc2); asserteq!(s2, s1); asserteq!(tc2, tc1); } ```