chain binding, first result, WIP
This commit is contained in:
+99
-20
@@ -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 } );
|
||||||
|
var zeroListener = null;
|
||||||
|
while (--i >= 0) {
|
||||||
|
var field = fields[i];
|
||||||
|
var next = fields[i + 1];
|
||||||
|
var listenerName = 'listener${i+1}';
|
||||||
|
var listenerNameExpr = macro $i { listenerName };
|
||||||
|
var nextListenerName = i == fields.length - 2 ? 'listener0' : 'listener${i+2}';
|
||||||
|
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) {
|
if (field.bindable) {
|
||||||
var listener = switch (field.field.kind) {
|
var type = Context.typeof(field.e).toComplexType();
|
||||||
case FVar(_, _):
|
res.push(macro var $value:Null<$type> = null );
|
||||||
macro function (from, to) {
|
var bind = BindMacros.bindingSignalProvider.getClassFieldBindExpr(macro n, next.field, nextListenerNameExpr );
|
||||||
$listener($a{[from, to]});
|
var unbind = BindMacros.bindingSignalProvider.getClassFieldUnbindExpr(valueExpr, next.field, nextListenerNameExpr );
|
||||||
}
|
fieldListenerBody.push(macro
|
||||||
case FMethod(_):
|
if (n != null) {
|
||||||
macro function() {
|
$ { 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
@@ -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
@@ -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();
|
||||||
|
|||||||
Reference in New Issue
Block a user