7

似乎 kendo 的 unobtrusive-javascript 样式事件调用this在我的方法上下文中中断。

假设我有一个对象Foo,实例化为bar = new Foo()

function Foo(){};

Foo.prototype.name = "Herring";
Foo.prototype.doSomething = function(e) {
   alert(this.name);
};

bar = new Foo();

并使用 data-click 例如附加事件

<a data-role="button" data-click="bar.doSomething">Click Me</a>

对象上下文bar被替换(不知道为什么,因为我们有方便的element容器。)所以this.name是未定义的。

我已经var self = this;在对象构造函数中尝试过旧的,但它不起作用,有谁知道解决这个问题的最佳方法是什么?

更新:哈克解决方法

因为我真的不想失去将模块包装为类的好处,所以我创建了事件调用函数包装器,然后调用适当对象上的方法。

例如,将标记连接到包装函数。

<a data-role="button" data-click="doSomething">Click Me</a>

包装函数只调用object.method。

function doSomething(e){ bar.doSomething(e) };  

现在,这达到了预期的结果,但它非常可怕,从标记调用的每个事件都必须具有像上面那样的代理函数。所以想象一下你有 300 个事件的场景......你会立即明白为什么这很可怕。

如果没有其他解决方案,我非常希望有。我将发布此解决方法作为答案,但就我而言,它远非理想。

脚注

老实说,这似乎是剑道的主要架构缺陷,因为这种从标记中调用事件的方法是“剑道方式”。显然它不能被修补,因为可能已经有相当多的代码this作为对 html 元素的引用来处理。

能够覆盖它,或者能够通过可以传递调用的通用处理程序来路由这些事件调用,本质上是通用代理函数,是可以处理这种情况的可能方式。它也可以是kendo.对象上的简单可配置值。

理论解决方案

如果可行,我将发布后续操作,理论上可以在通用代理上引发事件,并让它调用适当范围的函数。

假设我们使用事件属性来调用代理,然后创建一个单独的属性来传达对象/方法调用。例如。

<a data-role="button" data-click="prox" data-prox="o.eventHandler">Click Me</a>

代理函数将从prox属性数据集中提取:

方法 - 使用 eval

不是因为我是邪恶的,而是必须的。

// sitting in global namespace 
function prox(e){
    var p = e.sender.element.data['prox'];
    // make sure our delegate is a function.
    if("function" == eval("typeof "+p)) { 
        eval(p + "(e)");
    }
}

显然我想要一个更好的方法来做到这一点,但至少它是干燥的。

(我一会儿会做一个非评估方法......)

开始评估...

让我们使用窗口上下文来定位对象/方法。

function prox(e) {
   var p = e.sender.element.data['prox'];
   if(p.indexOf(".") == -1){
      // global function : where *this* is window.
      // check you've got the function if not ditch it.
      if("function" == typeof window[p]) window[p](e);
   } else {
      // object/method (one level deep only.)
      var s = p.split(".");
      var o = s[0], m = s[1];
      // check the object/method is a function before executing it.
      if("function" == typeof window[o][p]) window[o][p](e);
   }
}

当然对于全局(窗口)范围的函数, this 作为元素可能更有用,但在这种情况下,你可以选择,我会省略

4

4 回答 4

0

正在使用的版本。

// dynamic proxy for retaining object context on methods called by
// data- attributes in Kendo.
// 
// e.g.
//
//     data-click="o.method" 
//
// Would lose context with `o` - context would be set in the same
// way as JQuery handlers, which is an inconvenience.
// 
// Alternatively we use the prox method
//
//     data-click="prox"
// 
// We'd then set `data-prox` on the same element, to the
// object.method pair.
//
//     data-prox="o.method"
//
// This is read by prox, transformed into a method call, type
// checked and executed if it's a valid method.
//
// A `data-prox` value in any form other than `object.method` will
// be ignored, for example, `object.child.method` will fail. If
// you're doing that sort of thing, feel free to hack it.
// 
// There's a backup eval() to locate the object if window doesn't
// own it. It should be possible to remove it under most
// circumstances, it's here for compatability with
// JSFiddle. (JSBin works without it.)
function prox(e) {
    var p = this.element.data().prox;
    if(p.indexOf(".") > -1){
        var s = p.split("."); if(s.length > 2) return; 
        var o = s[0], m = s[1];
        if("object" == typeof window[o]) {
            o = window[o];
        } 
        if("function" == typeof o[m]) o[m](e);
        // comment this out in production:
        l( "prox called " + s[0] + "::" + s[1] );
    }
}

function l(s) { console.log(s); }

注意事项

如果您在同一个元素上有多个处理程序,则不prox()适合,例如,如果您有data-init,data-show等,prox则无法区分,并且会失败。

我可能会更新这个,特别是如果这对我来说成为一个普遍的用例。

于 2012-12-31T09:11:35.667 回答
0

我暂时尝试了第三种方法,使用非通用技术,它的工作原理是这样的。

伪代码:

MyObject {

   method : function(e) {
      if (this instanceof MyObject) {
        // Do something with this
      } else {
        myInstance.method(e); // otherwise re-call the method to set this properly.
      }
   }
}
myInstance = new MyObject();

不如 prox 方法灵活,但适合我的用例,并且至少不需要单独的函数代理远离我们要使用的方法。我们可以通过预先进行类型检查和重新调用来使其更简洁。

例如

MyObject = {
   method : function(e) {
      if (! this instanceof MyObject) myInstance.method(e); // re-call

      // Method body... 

   }
}
myInstance = new MyObject();

这也意味着我的标记中不需要自定义data-属性。

注意:此方法对于将具有多个实例的对象是有问题的,但是,我应用的对象是单个实例。

如果您有需要特定于实例的处理程序(这是我提出这个问题的主要原因),则该prox方法比这更合适,这只是一种执行每个事件代理功能的更简洁的方法。

于 2013-01-05T08:38:26.393 回答
0

您可以使用 jQuery 代理 ( http://api.jquery.com/jQuery.proxy/ )。

function Foo(){};

Foo.prototype.name = "Herring";
Foo.prototype.doSomething = function(e) {
   alert(this.name);
};

bar = new Foo();

$("btn").click($.proxy(bar.doSomething), bar);

或用于内部使用

$("btn").click($.proxy(this.doSomething), this);
于 2013-03-19T11:53:07.747 回答
0

我使用 JS Proxy Polyfill 开发了一种代理方法,该方法通过 custon html data-* 属性中的参数简化了调用自定义逻辑。

包括https://raw.githubusercontent.com/GoogleChrome/proxy-polyfill/master/proxy.js

function makeGridTemplateEventProxy(o) {
    return new Proxy(o, {
        get(target, eventName) {
            return function (options) {
                return templateEventProxy(options, eventName);
            }
        }
    });
}

templateEventProxy: function (options, attribute) {
    if (!options.sender.element.attr('data-proxy-' + attribute)) {
        throw new Error('Cannot find attribute data-proxy-' + attribute + ' on ' + options.sender.name + ' widget');
    }
    var proxyParams = JSON.parse(options.sender.element.attr('data-proxy-' + attribute));

    method = $("#" + proxyParams.id).data(proxyParams.widget).element.data(proxyParams.method);

    if (method && typeof method == 'function') {
        return $.proxy(method, this)(options);
    }

    return null;

}


var eventproxy = makeGridTemplateEventProxy({});

例如上传组件

<input type=file ... 
       data-success="eventproxy.customsuccesshandler"
       data-proxy-customsuccesshandler='{widget:"kendoGrid",method:"<myJqueryDataDefinedMethod>",id:"<gridId>"}'
       ....
/>

用您的参数替换myJqueryDataDefinedMethodgridId

如您所见,您可以在 data-success 中定义具有动态名称的事件代理

data-success="eventproxy.CUSTOMKEY"

并在定义自定义属性之后

data-proxy-CUSTOMKEY

data-proxy-CUSTOMKEY 包含可用于实现自定义逻辑的参数(JSON 编码),

我建议自定义逻辑,它可以通过 $.data 检索存储在 kendo 小部件网格上的 JS 方法

$("#" + proxyParams.id).data(proxyParams.widget).element.data(proxyParams.method)

例如,您可以将方法绑定到网格

$('#my-grid-id').data("kendoGrid").element.data('methodName',function(e){ 
    // my implementation 
});
于 2017-06-21T12:40:14.117 回答