打算写一个DirectUI库,在写其中底层窗口的回调构造的时候遇到一个问题。

Invoker是一个模板,因为closure的关系,它必须保存一个类对象的指针,和回调函数的地址。而函数调用的时候,就可以用一个通用的接口invoke就行了。

接口的声明如下:

class IOperation
{
public:
virtual ~IOperation() {};
virtual void Invoke(IUIWnd* wnd, const EventArg& arg) = 0;
};

  回调函数调用的时候,调用Invoke(...),传入(谁)那个窗口调用的,和调用的可能需要的数据。因为回调函数可能有不同的参数,为了方便,我把所有的回调都封装成参数相同的函数。数据都被封装到EventArg结构里面,这里我们不讨论EventArg。

Invoker类继承了IOperation这个接口,声明如下:  

template<class TObj>
class Invoker : public IOperation
{
public:
typedef void (TObj::*TFunc)(IUIWnd*, const EventArg&);
struct closure
{
TObj* _Obj;
TFunc _Func;
}; Invoker(TObj* obj, TFunc func)
{
_closure._Obj = obj;
_closure._Func = func;
} Invoker(const Invoker& ci)
{
_closure._Obj = ci._closure._Obj;
_closure._Func = ci._closure._Func;
} bool Equel(TObj* obj, TFunc func)
{
if (_closure._Obj == obj && _closure._Func == func)
{
return true;
}
return false;
} virtual void Invoke(IUIWnd* wnd, const EventArg& arg)
{
(_closure._Obj->*_closure._Func)(wnd, arg);
} private:
closure _closure;
};

  里面有个closure,这个就不多讲了,博客园某博主翻译过一篇,写得很好:点击这里查看。感谢博主@Jans的翻译。

  从invoker类可以很明了的看到,怎么去构造一个Invoker(再重复一遍,回调函数是一个类的成员函数,因此,在类外调用的时候,需要一个类成员的对象(或对象指针),和成员函数的地址。)

  假设现在有一个button,有一个操作是要相应按下的操作。

virtual IOperation* SetPressFunc( IOperation* oper ) = 0;

  我的思路是,先构造一个Invoker<TObj> invoker(obj, func), 然后把invoker的指针赋给button中响应按下操作的IOperation指针。

  假设有个button对象:btnOK, 于是就用 btnOK->SetPressFunc(invoker); 的确工作了。还算不错。

  但是有个小问题:btnOK必须是已经初始化完成了的,也就是说,已经 new 出来了。说实话,我可不想让库的使用者自己去调用 new Button()。竟然库是我写的,我必须用库管理好这些窗口(控件)。我会在程序初始化,读窗口配置文件的时候,为程序员用户,new好这些窗口;在程序结束的时候,释放掉。

  然后这也引出了一个问题:窗口都是库来初始化的,程序员用户想让按钮A被点击的时候,相应自己写的回调函数。作为用户,我都不知道A的对象在哪,叫什么名字,这怎么把对调绑定到这个按钮上?真这样了,用户肯定要骂我。程序员骂人都是很透彻的,代码里面fuck不少,你懂的。我需要写一个让你定义的窗口指针,关联到你想关联的窗口对象上。

  现在整理一下,如果我是用户,我当前所有的资源。

  1、界面配置文件是我用UI编辑器写的,每个窗口(控件)都有一个唯一(准确地说,是父窗口下唯一的)标识吧,比方说,枚举ID(MFC就是这么干的),或者名字。

  2、类对象,和要回调成员函数的地址。用于生成一个invoker;

  OK,就这么多,够了。

  我写了一个关联的函数 InitControl:

template<class TWnd, class TObj>
void InitControl(TObj* obj, TWnd*& out, const std::wstring& id)
{
InitControl(obj, out, obj->getMainWnd(), id);
}

  又是模板 = =%。。好吧,其实这个模板函数里面调用的InitControl,还是个模板。这些程序员用户不用管,我现在只是说明原理。如果写完了,开源之后,你们再去研究代码。

  别骂我,你想想啊,你的一个类,肯定是一个界面吧,一个程序可能有各种界面吧。我不能为每个类对象去写一个InitControl吧。嘿嘿。要付出代价的。

  InitControl参数说明:第一个,类对象指针;第二个,需要关联的窗口指针;第三个参数,要关联的窗口的id,字符串表示哈。

  比方说,我CUIXXX类下面,有一个button,声明是IUIButton* btnTest;界面配置文件里面,我想关联的对象的id是btnTest;

  我关联的时候,就 InitControl (this, btnTest, L"btnTest"); 然后btnTest就跟我想关联的按钮关联上了。

  现在我想让它顺便把按下的响应函数也加上去。好吧,我要改InitControl了。

template<class TWnd, class TObj>
void InitControl(TObj* obj, TWnd*& out, const std::wstring& id,IOperation* oper)
{
InitControl(obj, out, obj->getMainWnd(), id);
//....
}

  很显然,这样是不行的。

  IOperation只是记录了一个类对象和类成员函数的地址,以及调用所执行的操作。

  一个窗口有鼠标左键单击事件,可能有鼠标右键单击事件,以及其它事件:例如,editbox可能有TextChange事件,Combobox有ItemChange事件,就单个IOperation是无法记录这么多事件的,因此,我们需要引入一个中间的传递机制。

  把问题留在这里。下一篇我们再讲。

=====>THE END<=====

关于UI回调Invoker的实现(一)的更多相关文章

  1. 关于UI回调Invoker的实现(二)

    上篇我说到,光有一个IOperation*的指针,是无法记录这么多事件的.由于无法确定要把回调绑定到哪个事件上,因此,我们需要引入一个中间的传递机制. 没有看到前面的请先查阅上一篇 关于UI回调Inv ...

  2. 一个强大的UI node 抽象

    基于cocos2d -x的一个强大的 界面对象的基类 ---@type uinode ui 对象的抽象 --@usage -- 界面打开的执行流程 -- 带*的是可选重写的函数,不带*的为必须实现的 ...

  3. [Android]使用MVP解决技术债务(翻译)

    以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/5892671.html 使用MVP解决技术债务 原文:https ...

  4. iOS开发项目之MVC与MVVM

    MVC MVC,Model-View-Controller,我们从这个古老而经典的设计模式入手.采用 MVC 这个架构的最大的优点在于其概念简单,易于理解,几乎任何一个程序员都会有所了解,几乎每一所计 ...

  5. 猿题库 iOS 客户端架构设计

    原文: http://mp.weixin.qq.com/s?__biz=MjM5NTIyNTUyMQ==&mid=444322139&idx=1&sn=c7bef4d439f4 ...

  6. 猿题库 iOS 客户端架构设计-唐巧

    序 猿题库是一个拥有数千万用户的创业公司,从20013年题库项目起步到2015年,团队保持了极高的生产效率,使我们的产品完成了五个大版本和数十个小版本的高速迭代. 在如此快速的开发过程中,如何保证代码 ...

  7. [iOS] 响应式编程开发-ReactiveCocoa(二)

    RAC实现图片下载功能 在实现异步RAC下载图片的过程中,需要注意以下几点: • 通过 NSURLConnection 对象的 +(RACSignal *)rac_sendAsynchronousRe ...

  8. 用好lua+unity,让性能飞起来——luajit集成篇/平台相关篇

    luajit集成篇 大家都知道luajit比原生lua快,快在jit这三个字上. 但实际情况是,luajit的行为十分复杂.尤其jit并不是一个简单的把代码翻译成机器码的机制,背后有很多会影响性能的因 ...

  9. Android ANR的产生与分析

      ANR即Application Not Responding应用无响应,一般在ANR的时候会弹出一个应用无响应对话框.也许有些开发者在使用某些手机开发中不在弹出应用无响应弹出框,特别是国产手机An ...

随机推荐

  1. Vue前端数据采集 埋点 追踪用户系列行为

    什么是埋点?  综合    vue埋点 埋点分析,是网站分析的一种常用的数据采集方法.数据埋点分为初级.中级.高级三种方式.数据埋点是一种良好的私有化部署数据采集方式. 埋点技术如何采集数据,有何优缺 ...

  2. ElasticSearch关联查找

    ElasticSearch是一个基于Lucene的开源搜索引擎,支持全文检索,提供restful接口.在ES中,提供了类似于MongoDB的面向文档存储服务,这种面向文档的存储非常灵活,但是文档与文档 ...

  3. [javascript]—jQuery解析本地 XML 文档

    Create a jQuery object using an XML string and obtain the value of the title node. <!doctype html ...

  4. jmeter结果分析(图形报表和聚合报告)

    采用Jmeter测试工具对web系统作的负载测试,得出的响应报表,数据比较难懂,现作一具体说明.以下是在一次具体负载测试中得出的具体数值,测试线程设置情况为:线程数:200,等待时间(ramp-up) ...

  5. 总结day2 ---- while循环的简单使用, 格式化输出.运算符.以及编码的应用

    内容提要 一 : while 循环 while 的基本语句操作 如何终止循环 二 :格式化输出 三 :运算符号 四 :编码初识别 一 : while 循环 1  >>>>whi ...

  6. Visual Studio性能计数器,负载测试结果分析- Part III

    对于一个多用户的应用程序,性能是非常重要的.性能不仅是执行的速度,它包括负载和并发方面.Visual Studio是可以用于性能测试的工具之一.Visual Studio Test版或Visual S ...

  7. pygame学习_part1_pygame写程序前的准备工作

    import pygame from pygame import * pygame.init() #准备pygame,不准备无法使用功能 pygame.display.set_mode((x,y坐标) ...

  8. Java代码度量分析工具:Designite简介

    前言 在Java面向对象课程的学习过程中,我们需要使用度量工具来分析自己程序的代码结构.此类的度量工具有许多,或以插件形式存在于各个IDE中,或以.jar包的形式供用户使用.在这里,笔者向大家简单的介 ...

  9. windows下几个方便的右键菜单

    一直在用,拷来拷去麻烦,贴这里方便自己方便大家 PSHere.reg 右键菜单打开powershell Windows Registry Editor Version 5.00 [HKEY_CLASS ...

  10. OS X获取process.env.NODE_ENV出错

    原来项目是其它小组在维护,现在我们需要维护部分功能,把项目带到OS X上运行发现 webpack.config.js获取process.env.NODE_ENV变量出错 解决: 根据电脑操作系统平台类 ...