2006年11月22日

下载地址:http://www.cppblog.com/Files/yxm/wtldoc.rar

posted @ 2006-11-22 15:18 yxm 阅读(2607) | 评论 (9)编辑 收藏

2006年10月26日

详细内容请看微软官方说明:http://support.microsoft.com/kb/320216/zh-cn

这里贴上我的解决方法(vs2005环境):

设置工程属性->配置属性-> 清单工具->输入和输出->嵌入清单,选择[否],确定后重新生成即可。


posted @ 2006-10-26 17:11 yxm 阅读(1208) | 评论 (2)编辑 收藏

2006年10月23日

一、指出ActiveX控件对于脚本与数据初始化是安全的

1.  将下行添加到组件的继承列表的末尾:

public  IObjectSafetyImpl < CPolyCtl, INTERFACESAFE_FOR_UNTRUSTED_CALLER  |  INTERFACESAFE_FOR_UNTRUSTED_DATA >

2.  将下行添加到COM 映射:

COM_INTERFACE_ENTRY(IObjectSafety)

二、电子签名
可使用 MAKECERT 和 CERT2SPC 实用工具生成测试 Software Publisher Certificate。请注意,此测试 SPC 对软件发布无效,但可用于测试代码签名

1.  创建证书文件:

makecert  /sv  "私钥文件.pvk"  /"CN=公司的名字"  "证书文件名.cer" //注意保留扩展名

2.  转换cer格式为spc格式:

cert2spc   dream.cer   dream.spc   

3.  给cab文件签名

signtool signwizard

4.  在web页上嵌套入签名的CAB文件

<BODY>
<OBJECT ID="PolyCtl" 
CLASSID
="CLSID:4CBBC676-507F-11D0-B98B-000000000000"
CODEBASE
="http://example.sq-soft.com/mydir/myctl.cab#version=1,0,0,1">
</OBJECT>

</BODY>

CODEBASE 也可以直接指向 OCX 或 DLL 文件:


三、创建CAB文件(摘录自MSDN)

1.  创建INF文件

; Sample INF file for POLYGON.DLL
[version] 
; version signature (same 
for both NT and Win95) do not remove
signature
="$CHICAGO$"
AdvancedINF
=2.0  

[Add.Code]
polygon.dll
=polygon.dll
atl.dll
=atl.dll

; needed DLL
[atl.dll]
file
-win32-x86=thiscab
FileVersion
=2,00,0,7024
DestDir
=11
RegisterServer
=yes 

[polygon.dll]
file
-win32-x86=thiscab
clsid
={4CBBC676-507F-11D0-B98B-000000000000} 
FileVersion
=1,0,0,1 
RegisterServer
=yes  
; end of INF file

其中“thiscab”是一个关键字,表示 CAB 包含此 DLL,也可通过指定一条绝对或相对路径,从一个 HTTP 位置下载所需的 DLL,例如:

file-win32-x86=http://example.microsoft.com/mydir/NEEDED.DLL 

关键字“file-win32-x86”将平台标识为 x86 特定的。

可通过在 Windows 资源管理器中的文件上单击鼠标右键来获取文件的版本号。从显示的列表中选择“属性”,然后在显示的对话框中选择“版本”选项卡。有时需要在文件版本中额外插入一个 0。例如,在对话框中 ATL.DLL 的版本号显示为 2.00.7024。而在 INF 文件中,这变为 2, 00, 0, 7024。

“DestDir”是将要加载文件的目录所在的位置:11 指定系统目录为 WINDOWS/SYSTEM 或 WINNT/SYSTEM32;10 指定 Windows 目录、WINDOWS 或 WINNT。如果未指定任何 DestDir(典型情况),则代码安装在固定的 OCCACHE 目录中。

“clsid”是将要安装的控件的 CLSID。

2.  创建CAB文件

创建了 INF 文件后,运行 CABARC 实用工具以创建 CAB 文件。应在包含源文件的目录下运行 CABARC。在命令行上,按源文件出现在 INF 中的顺序排列它们,并将 INF 文件放在最后。例如:

CABARC -6144  n  MYCTL.CAB NEEDED1.DLL NEEDED2.DLL MYCTL.OCX MYCTL.INF

应在包含源文件(INF、OCX 和 DLL 文件)的目录下运行 CABARC。将在 CAB 文件中存档的文件应该按它们在 INF 文件中列出的同一顺序在命令行中列出。在上面的示例中,INF 文件应该将 NEEDED1.DLL 列在第一位,接下来是 NEEDED2.DLL,然后是 MYCTL.OCX。

-s 选项在压缩文件中保留用于代码签名的空间。n 命令指定希望创建 CAB 文件。
 

posted @ 2006-10-23 17:28 yxm 阅读(1067) | 评论 (0)编辑 收藏

2006年10月20日

今天在实现一个COM的事件接收器的时,客户端经过ATL事件映射后,运行时返回:E_INVALIDARG,调试跟踪也没有发现参数不对。折腾了好久后,发现客户端的事件函数的调用约定应为__stdcall,应该与所有的COM接口函数一样使用__stdcall调用约定。犯了一个愚蠢的错误。

posted @ 2006-10-20 11:52 yxm 阅读(1041) | 评论 (0)编辑 收藏

2006年10月17日

今天在聚合一个COM组件的时候,一开始使用智能指针创建时报错:“组件没有注册类别”,但检查注册确实已经注册了,而在使用原始指针时报错:“类别不支持聚合”。后来经过仔细调试发现是由于在聚合组件FinalConstruct中的CoCreateInstance参数有错。在使用智能指针时报的错误真是让人不知道是怎么回事。

posted @ 2006-10-17 17:37 yxm 阅读(918) | 评论 (0)编辑 收藏

在新增ATL简单对象的时候,新对象的名称不能与工程同名,否则会有此问题。

posted @ 2006-10-17 15:25 yxm 阅读(660) | 评论 (0)编辑 收藏

2006年10月13日

方法一:使用MFC映射宏,从CCmdTartget派生子类EventSink

1.  .h文件中申明接口、分派ID映射

DECLARE_DISPATCH_MAP()
DECLARE_INTERFACE_MAP()

2.   .cpp文件实现接口、分派ID映射

// 映射分派ID
BEGIN_DISPATCH_MAP(cThisClass, CBaseClass)
     DISP_FUNCTION_ID(CThisClass, 
            " ComFunc " ,
             DISP_ID, 
              LocalFuncName, 
              VARIANT_RETURN_TYPE, 
              VARIANT_PARAMS_TYPE)
END_DISPATCH_MAP()
      
// 映射事件接口
BEGIN_INTERFACE_MAP(CThisClass,CBaseClass)    
     INTERFACE_PART(CThisClass, DIID_IEventXX, Dispatch) 
// 这里Dispatch为CCmdTart类的一个结构,该宏填充这个结构
END_INTERFACE_MAP()

3.  #import "...\组件名.tlb" no_namespace named_guids no_namespace

4. 初始化COM库、Dispatch结构并建立连接

      CoInitialize(NULL);
EnableAutomation();    //同上面的宏一起实现了IDispatch接口
    AfxConnectionAdvise(ComObject,  DIID__IEventXX,  GetIDispatch(FALSE),  FALSE, &m_dwCookie);

5.  实现映射函数(LocalFuncName)

6.  卸载COM库、取消连接

AfxConnectionUnadvise()   
      CoUnInitialize();


方法二:  使用ATL映射宏,继承IDsipEventImpl

1.   #import "...\组件名.tlb" no_namespace named_guids

2.  实现事件接收类CEventSink

      a.  使用ATL宏定义函数信息结构      

_ATL_FUNCTION_INFO FuncInfo  =
{
     CC_STDCALL,  
// 调用方式
     VT_EMPTY,        // 返回类型
      1 ,                           // 参数个数
    {VT_I4}                 // 参数类型表
}
;

      b.   ATL宏映射事件函数           

BEGIN_SINK_MAP(CEventSink)
       SINK_ENTRY_INFO(
1 , DIID_I, DISP_ID, LocalFunc,  & FuncINfo)
       
END_SINK_MAP()

 3.  实例化组件对象和EventSink对象,用EventSink对象或AtlAdvise函数建立连接: EventSink->Advise(...) 或EventSink->DispEventAdvise(...)  或 AtlAdvise(...)

 4.  取消连接,销毁对象

     #ifndef __EVENTSINK_H
     
#define __EVENTSINK_H
     
     
namespace 
     
{
     
static const int DISPID_SHOW = 1;
     
static const int DISPID_SHOW2 = 2;
     
     _ATL_FUNC_INFO OnShowInfo 
=
     
{
      CC_STDCALL, 
//calling conv
      VT_EMPTY,//return value
      0 ,//number of arguments
      NULL//argumnent types
     }
;
     _ATL_FUNC_INFO OnShow2Info 
=
     
{
      CC_STDCALL, 
//calling conv
      VT_EMPTY,//return value
      1 ,//number of arguments
      { VT_I2}//argumnent types
     }
;
     
     }

     
     
class CEventSink :  public IDispEventImpl<1, CEventSink, &DIID__IAtlTest2Events, &LIBID_TEST2Lib>
     
{
     
public
       CEventSink()
{};
       
virtual ~CEventSink(){};
     
       
void __stdcall OnShow(); //没有参数的情况
       void __stdcall OnShow2(short);//有参数的情况
     
       
//必须要用SINK_ENTRY_INFO,必须指定参数信息
       BEGIN_SINK_MAP(CEventSink)
        SINK_ENTRY_INFO(
1,DIID_ISimpleCOMEvents,DISPID_SHOW,OnShow,&OnShowInfo)
        SINK_ENTRY_INFO(
1,DIID_ISimpleCOMEvents,DISPID_SHOW2,OnShow2,&OnShow2Info)
       END_SINK_MAP()
     }
;
     
     
#endif

方法三:自定义EventSink,从IDispatch派生,实现所有的虚方法,在Invoke函数中根据DISP_ID处理回调 (代码略)

posted @ 2006-10-13 11:44 yxm 阅读(1330) | 评论 (0)编辑 收藏

仅列出标题