chain binding, first result, WIP

This commit is contained in:
Dima Granetchi
2014-11-19 01:33:34 +02:00
parent 9fec9bdf25
commit 96e0362d34
3 changed files with 125 additions and 23 deletions
+97 -18
View File
@@ -1,4 +1,5 @@
package bindx; package bindx;
import haxe.macro.Context;
#if macro #if macro
@@ -8,6 +9,7 @@ import haxe.macro.Expr;
import haxe.macro.Type; import haxe.macro.Type;
using Lambda; using Lambda;
using haxe.macro.Tools;
typedef FieldExpr = { typedef FieldExpr = {
var field:ClassField; var field:ClassField;
@@ -53,35 +55,112 @@ class BindExt {
} }
prevField = field; prevField = field;
} }
fields.iter(function (it) trace(printer.printExpr(it.e) + " -> " + it.field.name + (it.params != null ? '(${printer.printExprs(it.params, ",")})' : "") + " bind:" + it.bindable)); for (it in fields)
var i = 0; trace(printer.printExpr(it.e) + " -> " + it.field.name + (it.params != null ? '(${printer.printExprs(it.params, ",")})' : "") + " bind:" + it.bindable);
var i = fields.length - 1;
var res = []; var res = [];
for (field in fields) { var unbindRes = [];
var listenerName = 'listener$i'; res.push(macro var listener0 = $ { listener } );
if (field.bindable) { var zeroListener = null;
var listener = switch (field.field.kind) { while (--i >= 0) {
case FVar(_, _): var field = fields[i];
macro function (from, to) { var next = fields[i + 1];
$listener($a{[from, to]}); var listenerName = 'listener${i+1}';
} var listenerNameExpr = macro $i { listenerName };
case FMethod(_): var nextListenerName = i == fields.length - 2 ? 'listener0' : 'listener${i+2}';
macro function() { var nextListenerNameExpr = macro $i { nextListenerName };
var value = 'value${i+1}';
var valueExpr = macro $i { value };
var fieldName = next.field.name;
var e = next.e;
var fieldListenerBody = [];
var fieldListener:Expr;
if (field.bindable) zeroListener = { f:field, l:listenerNameExpr };
if (next.bindable) zeroListener = { f:next, l:nextListenerNameExpr };
if (field.bindable) {
var type = Context.typeof(field.e).toComplexType();
res.push(macro var $value:Null<$type> = null );
var bind = BindMacros.bindingSignalProvider.getClassFieldBindExpr(macro n, next.field, nextListenerNameExpr );
var unbind = BindMacros.bindingSignalProvider.getClassFieldUnbindExpr(valueExpr, next.field, nextListenerNameExpr );
fieldListenerBody.push(macro
if (n != null) {
$ { bind }
$nextListenerNameExpr($a { next.params != null ? [] : [macro n.$fieldName, macro n.$fieldName] } );
} }
);
if (field.params != null) {
fieldListenerBody.unshift(macro $valueExpr = n );
fieldListenerBody.unshift(macro var n = $e );
fieldListenerBody.unshift(macro if ($valueExpr != null) $unbind );
fieldListener = macro function $listenerName () {
$b { fieldListenerBody };
};
} }
res.push(macro var $listenerName = $listener); else {
var expr = BindMacros.bindingSignalProvider.getClassFieldBindExpr(field.e, field.field, listener = macro $i{listenerName}); fieldListenerBody.unshift(macro $valueExpr = n );
trace(printer.printExpr(expr)); fieldListenerBody.unshift(macro if ($valueExpr != null) $unbind );
fieldListener = macro function $listenerName (o, n) {
$b { fieldListenerBody };
};
}
unbindRes.push(macro if ($valueExpr != null) $unbind);
unbindRes.push(macro $valueExpr = null );
//trace(printer.printExpr(unbind));
} }
else { else {
trace(printer.printExpr(field.e) + " . " + field.field.name); trace(printer.printExpr(field.e) + " . " + field.field.name);
/*if (field.params != null) {
fieldListenerBody.unshift(macro var n = $e );
fieldListener = macro function $listenerName () {
$b { fieldListenerBody };
};
}
else {
fieldListener = macro function $listenerName (o, n) {
$b { fieldListenerBody };
};
}*/
}
if (fieldListener != null) {
res.push(fieldListener);
} }
i++;
} }
return macro {}; if (zeroListener == null) {
Context.error("Chain is not bindable ", expr.pos);
}
var bind = BindMacros.bindingSignalProvider.getClassFieldBindExpr(zeroListener.f.e, zeroListener.f.field, zeroListener.l );
var unbind = BindMacros.bindingSignalProvider.getClassFieldUnbindExpr(zeroListener.f.e, zeroListener.f.field, zeroListener.l );
res.push(macro try { $bind; } catch (e:Dynamic) {
trace("Warning: Can't initialize chain binding " + Std.string(e));
});
unbindRes.push(macro try { $unbind; } catch (e:Dynamic) {
trace("Warning: Unbind chain problem " + Std.string(e));
});
if (zeroListener.f.params != null) {
res.push(macro ${zeroListener.l}());
}
else {
var fieldName = zeroListener.f.field.name;
res.push(macro try { $ { zeroListener.l } (null, $ { zeroListener.f.e } .$fieldName ); } catch (e:Dynamic) {
trace("Warning: Can't initialize chain binding " + Std.string(e));
});
}
res.push(macro function unbind() { $b { unbindRes } } );
var res = macro $b { res };
trace(printer.printExpr(res));
return res;
} }
#end #end
+25 -2
View File
@@ -20,14 +20,36 @@ class ChainBindTest extends BuddySuite {
before({ before({
b = new BindableChain(); b = new BindableChain();
b.c = new BindableChain(); b.c = new BindableChain();
b.c.c = new BindableChain();
callNum = 0; callNum = 0;
}); });
it("BindExt.chain should bind chain changes", { it("BindExt.chain should bind chain changes (1 gap)", {
b.c.f("tada").d = "a"; b.c.f("tada").d = "a";
BindExt.chain(b.c.f("tada").d, function (_, _) callNum++); var unbind = BindExt.chain(b.c.f("tada").d, function (_, t:String) {
callNum++;
t.should.be(b.c.f("tada").d);
});
callNum.should.be(1);
b.c.f("tada").d = "b"; b.c.f("tada").d = "b";
callNum.should.be(2);
unbind();
b.c.f("tada").d = "c";
callNum.should.be(2);
});
it("BindExt.chain should bind chain changes (double gap)", {
b.c.c.d = "a";
BindExt.chain(b.c.c.d, function (_, _) callNum++);
callNum.should.be(1);
BindExt.chain(b.c.c.c.d, function (_, _) callNum++ );
callNum.should.be(1); callNum.should.be(1);
}); });
@@ -43,6 +65,7 @@ class BindableChain implements bindx.IBindable {
public var nd:String; public var nd:String;
public var c:BindableChain; public var c:BindableChain;
@:bindable @:bindable
+1 -1
View File
@@ -15,7 +15,7 @@ class Tests extends BuddySuite {
new MetaTest(), new MetaTest(),
new SignalTest(), new SignalTest(),
new TestProperty(), new TestProperty(),
// new ChainBindTest(), new ChainBindTest(),
], reporter); ], reporter);
runner.run(); runner.run();