事件分发器

这个分发器是之前做slg的丹神写的,后面再稍微修改了存在的bug(有注释部分是修改过的)。

Dispatcher.lua

local Dispatcher = {}
Dispatcher.obj = {}

function Dispatcher:addEventListener(eventName, class, callBackFunc)
    if self.obj[eventName] then
        local t = self.obj[eventName]
        t[class] = callBackFunc
    else
        local t = {}
        t[class] = callBackFunc
        self.obj[eventName] = t
    end
end


function Dispatcher:removeEventListener(eventName, class)
    local t = self.obj[eventName]
    if t then
        t[class] = nil
        local haveAnyChild = false
        for _, _ in pairs(t) do
            haveAnyChild = true
            break
        end
        if not haveAnyChild then
            self.obj[eventName] = nil
        end
    end

    if(self.temps ~= nil) then
        for _,temp in pairs(self.temps) do
            if(temp.eventName == eventName) then
                local keys = nil
                for func,class_ in pairs(temp) do
                    if(class_ == class) then
                        keys = keys or {}
                        keys[func] = true
                    end
                end
                if(keys ~= nil) then
                    for func,_ in pairs(keys) do
                        temp[func] = nil
                    end
                end
            end
        end
    end
end

function Dispatcher:removeAllEventListener()
    for k, v in pairs(self.obj) do
        self.obj[k] = nil
    end
end

function Dispatcher:dispatchEvent(event)
    -- 遍历事件的时候产生的问题:
    --   某个函数的回调中增加或移除该类型的事件监听,造成表格混乱,遍历不到或多次遍历的问题
    --      为了解决上面的问题,引入一个临时表,但又引入了新的问题:
    --          在回调中移除临时表后面的回调函数的时候,临时表中的回调没有删除,这样继续引入新的表格控制
    local eventName = event["name"]
    local t = self.obj[eventName]
    if t then
        local helper = {}             -- 为了解决某个函数的回调中增加或移除该类型的事件监听,造成表格混乱,遍历不到或多次遍历的问题
        self.temps = self.temps or {} -- 解决上面遍历时移除事件监听的问题
        self.temps[helper] = {}
        local temp = self.temps[helper]
        temp.eventName = eventName
        for class, func in pairs(t) do
            temp[func] = class
            table.insert(helper, func)
        end
        repeat
            local func = table.remove(helper, 1)
            if(not func) then break end

            if(temp[func] ~= nil) then
                func(event)
            end
        until false
        self.temps[helper] = nil
    end
end

return Dispatcher

用法:
新增事件监听
Dispatcher:addEventListener(Event.UI_WINDOW_CLOSE, self, handler(self, self.onWindowClose))
移除事件监听
Dispatcher:removeEventListener(Event.UI_WINDOW_CLOSE, self)

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • ¥开启¥ 【iAPP实现进入界面执行逐一显】 〖2017-08-25 15:22:14〗 《//首先开一个线程,因...
    小菜c阅读 11,856评论 0 17
  • Lua 5.1 参考手册 by Roberto Ierusalimschy, Luiz Henrique de F...
    苏黎九歌阅读 14,750评论 0 38
  •   JavaScript 与 HTML 之间的交互是通过事件实现的。   事件,就是文档或浏览器窗口中发生的一些特...
    霜天晓阅读 8,883评论 1 11
  • 事件是一种异步编程的实现方式,本质上是程序各个组成部分之间的通信。DOM支持大量的事件,本节介绍DOM的事件编程。...
    许先生__阅读 4,509评论 0 3
  • 周清自从萌生了网上开店的想法后,就一直再考虑,并把这个想法告诉了老板,老板也觉得很不错,并把这事交给了周清去处理...
    478339124d9c阅读 1,306评论 0 0

友情链接更多精彩内容