Skip to content

Latest commit

 

History

History
112 lines (92 loc) · 2.38 KB

README.md

File metadata and controls

112 lines (92 loc) · 2.38 KB

Update

This cookbook article shows a more native haxe way of creating a fluent interface, the repository remains as an archive of advanced macro usage.

Fluent

This is a library for creating fluent interfaces over existing classes. Uses @:genericBuild.

Quick start

import fluent.Fluent;

class Main
{
    public static function main()
    {
        var foo = new Fluent<Foo>();
        var returnedValue = foo
            .hello() //Hello
            .bar()
                .hi('You') //Hi You!
            .end()
            .bar().end()
            .listBars() //A bar, A bar
            .returner()
        ;
        //returnedValue == 4
    }
}

class Foo
{
    private var bars:Array<Bar> = new Array();

    public function new() {}

    public function listBars()
    {
        trace(bars.join(', '));
    }

    public function hello()
    {
        trace('Hello');
    }

    public function returner()
    {
        return 2 + 2;
    }

    @Fluent
    public function bar()
    {
        var bar = new Bar();
        bars.push(bar);

        return bar;
    }
}

class Bar
{
    public function new() {}

    public function hi(who:String)
    {
        trace('Hi $who!');
    }

    public function toString()
    {
        return 'A bar';
    }
}

Flags

There are three flags you can use

  • -D fluent_debug - will dump out bodies of generated classes
  • -D fluent_dynamic - allows extending your API, of course is costly and further hinders any type-completion you might have
  • -D fluent_dynamic_warning - will warn you if you didn't implement resolve method in underlying class

Limitations

If you are using fluent_dynamic, you cannot access class variables in resolve function. So:

Invalid - will cause infinite recursion, because extensions will change into this.resolve('extensions') in wrapper class

private var extensions:Map<String, Dynamic> = new Map();

public function resolve(method:String)
{
    for(key in extensions.keys()) {
        if(method == key) //do something
    }
}

Valid:

private var extensions:Map<String, Dynamic> = new Map();

public function resolve(method:String):Dynamic
{
    return findExtension();
}

private function findExtension(method:String)
{
    for(key in extensions.keys()) {
        if(method == key) //do something
    }
}