繁体   English   中英

JavaScript自定义事件监听器

[英]JavaScript custom Event Listener

我想知道是否有人可以帮助我了解如何准确地创建不同的自定义事件侦听器。

我没有事件的具体案例,但我想大致了解它是如何完成的,因此可以将其应用于需要的地方。

我想做的是,以防某些人可能需要知道的是:

var position = 0;

for(var i = 0; i < 10; i++)
{
    position++;
    if((position + 1) % 4 == 0)
    {
        // do some functions
    }
}
var evt = document.createEvent("Event");
evt.initEvent("myEvent",true,true);

// custom param
evt.foo = "bar";

//register
document.addEventListener("myEvent",myEventHandler,false);

//invoke
document.dispatchEvent(evt);

这是在本地做更多事情的方法,可以准确地定位听众和发布者: http : //www.kaizou.org/2010/03/generating-custom-javascript-events/

实现自定义事件并不难。 您可以通过多种方式实现它。 最近我正在这样做:

/***************************************************************
*
*   Observable
*
***************************************************************/
var Observable;
(Observable = function() {
}).prototype = {
    listen: function(type, method, scope, context) {
        var listeners, handlers;
        if (!(listeners = this.listeners)) {
            listeners = this.listeners = {};
        }
        if (!(handlers = listeners[type])){
            handlers = listeners[type] = [];
        }
        scope = (scope ? scope : window);
        handlers.push({
            method: method,
            scope: scope,
            context: (context ? context : scope)
        });
    },
    fireEvent: function(type, data, context) {
        var listeners, handlers, i, n, handler, scope;
        if (!(listeners = this.listeners)) {
            return;
        }
        if (!(handlers = listeners[type])){
            return;
        }
        for (i = 0, n = handlers.length; i < n; i++){
            handler = handlers[i];
            if (typeof(context)!=="undefined" && context !== handler.context) continue;
            if (handler.method.call(
                handler.scope, this, type, data
            )===false) {
                return false;
            }
        }
        return true;
    }
};

只需将Observable的原型与该构造函数的原型混合,即可由任何构造函数需要的对象都可以重用和应用Observable对象。

要开始收听,您必须将自己注册到可观察对象,如下所示:

var obs = new Observable();
obs.listen("myEvent", function(observable, eventType, data){
    //handle myEvent
});

或者,如果您的侦听器是对象的方法,则如下所示:

obs.listen("myEvent", listener.handler, listener);

其中,侦听器是对象的实例,它实现方法“处理程序”。

Observable对象现在可以在发生任何想与侦听器通信的事件时调用其fireEvent方法:

this.fireEvent("myEvent", data);

数据是一些使听众感兴趣的数据。 无论您放置在这里什么都取决于您-您最清楚自定义事件的构成。

fireEvent方法只是遍历为“ myEvent”注册的所有侦听器,并调用已注册的函数。 如果函数返回false,则表示该事件已取消,并且可观察对象将不会调用其他侦听器。 结果,整个fireEvent方法也将返回fasle,因此可观察对象知道现在通知其侦听器的任何操作都应该回滚。

也许这种解决方案并不适合每个人,但是我从这段相对简单的代码中受益匪浅。

从这里:

https://developer.mozilla.org/zh-CN/docs/Web/Guide/Events/Creating_and_triggering_events

// create the event
var evt = document.createEvent('Event');
// define that the event name is `build`
evt.initEvent('build', true, true);

// elem is any element
elem.dispatchEvent(evt);


// later on.. binding to that event
// we'll bind to the document for the event delegation style. 
document.addEventListener('build', function(e){
   // e.target matches the elem from above
}, false);

这是一个非常简单的(TypeScript / Babelish)实现:

const simpleEvent = <T extends Function>(context = null) => {
    let cbs: T[] = [];
    return {
        addListener: (cb: T) => { cbs.push(cb); },
        removeListener: (cb: T) => { let i = cbs.indexOf(cb); cbs.splice(i, Math.max(i, 0)); },
        trigger: (<T> (((...args) => cbs.forEach(cb => cb.apply(context, args))) as any))
    };
};

您可以这样使用它:

let onMyEvent = simpleEvent();
let listener = (test) => { console.log("triggered", test); };
onMyEvent.addListener(listener);
onMyEvent.trigger("hello");
onMyEvent.removeListener(listener);

或像这样的课程

class Example {
    public onMyEvent = simpleEvent(this);
}

如果您想要纯JavaScript,则可以使用TypeScript Playground进行编译。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM