这次说的是剪贴板格式 vs. 上下文菜单&属性页之间的关系。这种关系并非相声和水利的关系。
为了更好地证明SEF的强大威力,同时为了更方便自己在网上down东西,决定写一个non-rooted NSE:网络硬盘,把@163.com的邮箱变成一个虚拟硬盘。网上已经有类似的东西,叫做GmailDrive,可惜一来不公开源码,二来上传大文件的时候太费劲。
WebDrive是植根在desktop上的non-rooted NSE,在经过10分钟的工作之后,建议了一个小框架,可以在桌面生成一个小图标,并且有了基本的树型结构,当然是虚拟的数据,没有从网易直接拉HTML。
但是发现,桌面上的图标的右键菜单(上下文菜单)里面居然没有我的自定义菜单和属性菜单项。好了,下断点大法。
结果是发现两个接口的Initialize函数都被调用过,但是下一步的接口再没被调用。为什么呢?
观察Debugger的output窗口,发现在Initialize()之后有个com exception。看地址来自webdisk.dll。找到对应的函数之后,发现是在IDataObjectImpl的GetData方法中。在这里,代码会检查static_cast<T*>(this)->OnGetData()的返回值,如果FAILED(),就抛出一个COM异常。
下断点在这里,发现tymed居然是TYMED_NULL,而不是任何我们已经定义的格式。这是为什么?!
好了,看看别人是怎么处理的。 :)
写一个简单的console程序,取得My Documents的接口,调它的IDataObject::GetData,跟进去看看。。。。。。。
原来如此:Explorer会调用non-rooted NSE的这个接口,查询是否有数据的时候,用TYMED_NULL当探路石。这个时候,只要返回E_INVALIDARG就可以。Explorer并不真的需要这些数据。原来是自己的代码错误检查太严了。
好了,知道了原理,把代码修改一下,如果是non-rooted模式,并且检查点是在根节点的话,就放行TYMED_NULL。