转载:atl组件和控件编程 wtl,atl

<script language="javascript" type="text/javascript"> </script>
<script type="text/javascript"> </script> <script type="text/javascript" src="http://blog.csdn.net/WebResource.axd?d=VT-DPrLTYP31jDF3TQ1B-w2&t=632963535947587500"></script> <script type="text/javascript"> // </script> <script type="text/javascript"> //=0xDC00 && c<0xE000) continue; if (c>=0xD800 && c<0xDC00) { if (i>=wide.length) continue; s= wide.charCodeAt(i++); if (s<0xDC00 || c>=0xDE00) continue; c= ((c-0xD800)<<10)+(s-0xDC00)+0x10000; } // output value if (c<0x80) enc += String.fromCharCode(c); else if (c<0x800) enc += String.fromCharCode(0xC0+(c>>6),0x80+(c&0x3F)); else if (c<0x10000) enc += String.fromCharCode(0xE0+(c>>12),0x80+(c>>6&0x3F),0x80+(c&0x3F)); else enc += String.fromCharCode(0xF0+(c>>18),0x80+(c>>12&0x3F),0x80+(c>>6&0x3F),0x80+(c&0x3F)); } return enc; } var hexchars = "0123456789ABCDEF"; function toHex(n) { return hexchars.charAt(n>>4)+hexchars.charAt(n & 0xF); } var okURIchars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_-"; // Mimics function encodeURIComponent in IE 5.5+, Netscape 6+, and Mozilla function encodeURIComponentNew(s) { var s = utf8(s); var c; var enc = ""; for (var i= 0; i -1) { func = new Function(functionPrefix + " " + ev); } else { func = new Function("event", functionPrefix + " " + ev); } eval("control." + eventType + " = func;"); } function Anthem_GetXMLHttpRequest() { if (window.XMLHttpRequest) { return new XMLHttpRequest(); } else { if (window.Anthem_XMLHttpRequestProgID) { return new ActiveXObject(window.Anthem_XMLHttpRequestProgID); } else { var progIDs = ["Msxml2.XMLHTTP.5.0", "Msxml2.XMLHTTP.4.0", "MSXML2.XMLHTTP.3.0", "MSXML2.XMLHTTP", "Microsoft.XMLHTTP"]; for (var i = 0; i < progIDs.length; ++i) { var progID = progIDs[i]; try { var x = new ActiveXObject(progID); window.Anthem_XMLHttpRequestProgID = progID; return x; } catch (e) { } } } } return null; } function Anthem_CallBack(url, target, id, method, args, clientCallBack, clientCallBackArg, includeControlValuesWithCallBack, updatePageAfterCallBack) { if (window.Anthem_PreCallBack) { var preCallBackResult = Anthem_PreCallBack(); if (!(typeof preCallBackResult == "undefined" || preCallBackResult)) { if (window.Anthem_CallBackCancelled) { Anthem_CallBackCancelled(); } return null; } } var x = Anthem_GetXMLHttpRequest(); var result = null; if (!x) { result = { "value": null, "error": "NOXMLHTTP" }; Anthem_DebugError(result.error); if (window.Anthem_Error) { Anthem_Error(result); } if (clientCallBack) { clientCallBack(result, clientCallBackArg); } return result; } x.open("POST", url ? url : Anthem_DefaultURL, clientCallBack ? true : false); x.setRequestHeader("Content-Type", "application/x-www-form-urlencoded; charset=utf-8"); x.setRequestHeader("Accept-Encoding", "gzip, deflate"); if (clientCallBack) { x.onreadystatechange = function() { if (x.readyState != 4) { return; } Anthem_DebugResponseText(x.responseText); result = Anthem_GetResult(x); if (result.error) { Anthem_DebugError(result.error); if (window.Anthem_Error) { Anthem_Error(result); } } if (updatePageAfterCallBack) { Anthem_UpdatePage(result); } Anthem_EvalClientSideScript(result); clientCallBack(result, clientCallBackArg); x = null; if (window.Anthem_PostCallBack) { Anthem_PostCallBack(); } } } var encodedData = ""; if (target == "Page") { encodedData += "&Anthem_PageMethod=" + method; } else if (target == "MasterPage") { encodedData += "&Anthem_MasterPageMethod=" + method; } else if (target == "Control") { encodedData += "&Anthem_ControlID=" + id.split(":").join("_"); encodedData += "&Anthem_ControlMethod=" + method; } if (args) { for (var argsIndex = 0; argsIndex < args.length; ++argsIndex) { if (args[argsIndex] instanceof Array) { for (var i = 0; i < args[argsIndex].length; ++i) { encodedData += "&Anthem_CallBackArgument" + argsIndex + "=" + Anthem_Encode(args[argsIndex][i]); } } else { encodedData += "&Anthem_CallBackArgument" + argsIndex + "=" + Anthem_Encode(args[argsIndex]); } } } if (updatePageAfterCallBack) { encodedData += "&Anthem_UpdatePage=true"; } if (includeControlValuesWithCallBack) { var form = document.getElementById(Anthem_FormID); if (form != null) { for (var elementIndex = 0; elementIndex < form.length; ++elementIndex) { var element = form.elements[elementIndex]; if (element.name) { var elementValue = null; if (element.nodeName.toUpperCase() == "INPUT") { var inputType = element.getAttribute("type").toUpperCase(); if (inputType == "TEXT" || inputType == "PASSWORD" || inputType == "HIDDEN") { elementValue = element.value; } else if (inputType == "CHECKBOX" || inputType == "RADIO") { if (element.checked) { elementValue = element.value; } } } else if (element.nodeName.toUpperCase() == "SELECT") { if (element.multiple) { elementValue = []; for (var i = 0; i < element.length; ++i) { if (element.options[i].selected) { elementValue.push(element.options[i].value); } } } else if (element.length == 0) { elementValue = null; } else { elementValue = element.value; } } else if (element.nodeName.toUpperCase() == "TEXTAREA") { elementValue = element.value; } if (elementValue instanceof Array) { for (var i = 0; i < elementValue.length; ++i) { encodedData += "&" + element.name + "=" + Anthem_Encode(elementValue[i]); } } else if (elementValue != null) { encodedData += "&" + element.name + "=" + Anthem_Encode(elementValue); } } } // ASP.NET 1.1 won't fire any events if neither of the following // two parameters are not in the request so make sure they're // always in the request. if (typeof form.__VIEWSTATE == "undefined") { encodedData += "&__VIEWSTATE="; } if (typeof form.__EVENTTARGET == "undefined") { encodedData += "&__EVENTTARGET="; } } } if (encodedData.length > 0) { encodedData = encodedData.substring(1); } Anthem_DebugRequestText(encodedData.split("&").join("/n&")); x.send(encodedData); if (!clientCallBack) { Anthem_DebugResponseText(x.responseText); result = Anthem_GetResult(x); if (result.error) { Anthem_DebugError(result.error); if (window.Anthem_Error) { Anthem_Error(result); } } if (updatePageAfterCallBack) { Anthem_UpdatePage(result); } Anthem_EvalClientSideScript(result); if (window.Anthem_PostCallBack) { Anthem_PostCallBack(); } } return result; } function Anthem_GetResult(x) { var result = { "value": null, "error": null }; var responseText = x.responseText; try { result = eval("(" + responseText + ")"); } catch (e) { if (responseText.length == 0) { result.error = "NORESPONSE"; } else { result.error = "BADRESPONSE"; result.responseText = responseText; } } return result; } function Anthem_SetHiddenInputValue(form, name, value) { var input = null; if (form[name]) { input = form[name]; } else { input = document.createElement("input"); input.setAttribute("name", name); input.setAttribute("type", "hidden"); } input.setAttribute("value", value); var parentElement = input.parentElement ? input.parentElement : input.parentNode; if (parentElement == null) { form.appendChild(input); form[name] = input; } } function Anthem_RemoveHiddenInput(form, name) { var input = form[name]; if (input != null && typeof(input) != "undefined") { var parentElement = input.parentElement ? input.parentElement : input.parentNode; if (parentElement != null) { form[name] = null; parentElement.removeChild(input); } } } function Anthem_FireEvent(eventTarget, eventArgument, clientCallBack, clientCallBackArg, includeControlValuesWithCallBack, updatePageAfterCallBack) { var form = document.getElementById(Anthem_FormID); Anthem_SetHiddenInputValue(form, "__EVENTTARGET", eventTarget); Anthem_SetHiddenInputValue(form, "__EVENTARGUMENT", eventArgument); Anthem_CallBack(null, null, null, null, null, clientCallBack, clientCallBackArg, includeControlValuesWithCallBack, updatePageAfterCallBack); form.__EVENTTARGET.value = ""; form.__EVENTARGUMENT.value = ""; } function Anthem_UpdatePage(result) { var form = document.getElementById(Anthem_FormID); if (result.viewState) { Anthem_SetHiddenInputValue(form, "__VIEWSTATE", result.viewState); } if (result.viewStateEncrypted) { Anthem_SetHiddenInputValue(form, "__VIEWSTATEENCRYPTED", result.viewStateEncrypted); } if (result.eventValidation) { Anthem_SetHiddenInputValue(form, "__EVENTVALIDATION", result.eventValidation); } if (result.controls) { for (var controlID in result.controls) { var containerID = "Anthem_" + controlID.split("$").join("_") + "__"; var control = document.getElementById(containerID); if (control) { control.innerHTML = result.controls[controlID]; if (result.controls[controlID] == "") { control.style.display = "none"; } else { control.style.display = ""; } } } } if (result.pagescript) { Anthem_LoadPageScript(result, 0); } } // Load each script in order and wait for each one to load before proceeding function Anthem_LoadPageScript(result, index) { if (index < result.pagescript.length) { try { var script = document.createElement('script'); script.type = 'text/javascript'; if (result.pagescript[index].indexOf('src=') == 0) { script.src = result.pagescript[index].substring(4); } else { if (script.canHaveChildren ) { script.appendChild(document.createTextNode(result.pagescript[index])); } else { script.text = result.pagescript[index]; } } var heads = document.getElementsByTagName('head'); if (heads != null && typeof(heads) != "undefined" && heads.length > 0) { var head = heads[0]; // The order that scripts appear is important since later scripts can // redefine a function. Therefore it is important to add every script // to the page and in the same order that they were added on the server. // On the other hand, if we just keep adding scripts the DOM will grow // unnecessarily. This code scans the
element block and removes // previous instances of the identical script. var found = false; for (var child = 0; child < head.childNodes.length; child++) { var control = head.childNodes[child]; if (control.tagName.toUpperCase() == "SCRIPT") { if (script.src.length > 0) { if (script.src == control.src) { found = true; break; } } else if (script.innerHTML.length > 0) { if (script.innerHTML == control.innerHTML) { found = true; break; } } } } if (found) { head.removeChild(control); } // Now we append the new script and move on to the next script. // Note that this is a recursive function. It stops when the // index grows larger than the number of scripts. document.getElementsByTagName('head')[0].appendChild(script); if (typeof script.readyState != "undefined") { script.onreadystatechange = function() { if (script.readyState != "complete" && script.readyState != "loaded") { return; } else { Anthem_LoadPageScript(result, index + 1); } } } else { Anthem_LoadPageScript(result, index + 1); } } } catch (e) { Anthem_DebugError("Error adding page script to head. " + e.name + ": " + e.message); } } } function Anthem_EvalClientSideScript(result) { if (result.script) { for (var i = 0; i < result.script.length; ++i) { try { eval(result.script[i]); } catch (e) { alert("Error evaluating client-side script!/n/nScript: " + result.script[i] + "/n/nException: " + e); } } } } function Anthem_DebugRequestText(text) { } function Anthem_DebugResponseText(text) { } function Anthem_DebugError(text) { } //Fix for bug #1429412, "Reponse callback returns previous response after file push". //see http://sourceforge.net/tracker/index.php?func=detail&aid=1429412&group_id=151897&atid=782464 function Anthem_Clear__EVENTTARGET() { var form = document.getElementById(Anthem_FormID); Anthem_SetHiddenInputValue(form, "__EVENTTARGET", ""); } function Anthem_InvokePageMethod(methodName, args, clientCallBack, clientCallBackArg) { Anthem_Clear__EVENTTARGET(); // fix for bug #1429412 return Anthem_CallBack(null, "Page", null, methodName, args, clientCallBack, clientCallBackArg, true, true); } function Anthem_InvokeMasterPageMethod(methodName, args, clientCallBack, clientCallBackArg) { Anthem_Clear__EVENTTARGET(); // fix for bug #1429412 return Anthem_CallBack(null, "MasterPage", null, methodName, args, clientCallBack, clientCallBackArg, true, true); } function Anthem_InvokeControlMethod(id, methodName, args, clientCallBack, clientCallBackArg) { Anthem_Clear__EVENTTARGET(); // fix for bug #1429412 return Anthem_CallBack(null, "Control", id, methodName, args, clientCallBack, clientCallBackArg, true, true); } function Anthem_PreProcessCallBack( control, e, eventTarget, causesValidation, validationGroup, imageUrlDuringCallBack, textDuringCallBack, enabledDuringCallBack, preCallBackFunction, callBackCancelledFunction, preProcessOut ) { preProcessOut.Enabled = !control.disabled; var preCallBackResult = true; if (preCallBackFunction) { preCallBackResult = preCallBackFunction(control); } if (typeof(preCallBackResult) == "undefined" || preCallBackResult) { var valid = true; if (causesValidation && typeof(Page_ClientValidate) == "function") { valid = Page_ClientValidate(validationGroup); } if (typeof(WebForm_OnSubmit) == "function") { valid = WebForm_OnSubmit(); } if (valid) { var inputType = control.getAttribute("type"); inputType = (inputType == null) ? '' : inputType.toUpperCase(); if (inputType == "IMAGE" && e != null) { var form = document.getElementById(Anthem_FormID); if (e.offsetX) { Anthem_SetHiddenInputValue(form, eventTarget + ".x", e.offsetX); Anthem_SetHiddenInputValue(form, eventTarget + ".y", e.offsetY); } else { Anthem_SetHiddenInputValue(form, eventTarget + ".x", e.clientX - control.offsetLeft + 1); Anthem_SetHiddenInputValue(form, eventTarget + ".y", e.clientY - control.offsetTop + 1); } } if (imageUrlDuringCallBack || textDuringCallBack) { if (control.nodeName.toUpperCase() == "INPUT") { if (inputType == "CHECKBOX" || inputType == "RADIO" || inputType == "TEXT") { preProcessOut.OriginalText = GetLabelText(control.id); SetLabelText(control.id, textDuringCallBack); } else if (inputType == "IMAGE") { if (imageUrlDuringCallBack) { preProcessOut.OriginalText = control.src; control.src = imageUrlDuringCallBack; } else { preProcessOut.ParentElement = control.parentElement ? control.parentElement : control.parentNode; if (preProcessOut.ParentElement) { preProcessOut.OriginalText = preProcessOut.ParentElement.innerHTML; preProcessOut.ParentElement.innerHTML = textDuringCallBack; } } } else if (inputType == "SUBMIT") { preProcessOut.OriginalText = control.value; control.value = textDuringCallBack; } } else if (control.nodeName.toUpperCase() == "SELECT") { preProcessOut.OriginalText = GetLabelText(control.id); SetLabelText(control.id, textDuringCallBack); } else { preProcessOut.OriginalText = control.innerHTML; control.innerHTML = textDuringCallBack; } } control.disabled = (typeof enabledDuringCallBack == "undefined") ? false : !enabledDuringCallBack; return true; } else { return false; } } else { if (callBackCancelledFunction) { callBackCancelledFunction(control); } return false; } } function Anthem_PreProcessCallBackOut() { // Fields this.ParentElement = null; this.OriginalText = ''; this.Enabled = true; } function Anthem_PostProcessCallBack( result, control, eventTarget, clientCallBack, clientCallBackArg, imageUrlDuringCallBack, textDuringCallBack, postCallBackFunction, preProcessOut ) { if (postCallBackFunction) { postCallBackFunction(control); } control.disabled = !preProcessOut.Enabled; var inputType = control.getAttribute("type"); inputType = (inputType == null) ? '' : inputType.toUpperCase(); if (inputType == "IMAGE") { var form = document.getElementById(Anthem_FormID); Anthem_RemoveHiddenInput(form, eventTarget + ".x"); Anthem_RemoveHiddenInput(form, eventTarget + ".y"); } if (imageUrlDuringCallBack || textDuringCallBack) { if (control.nodeName.toUpperCase() == "INPUT") { if (inputType == "CHECKBOX" || inputType == "RADIO" || inputType == "TEXT") { SetLabelText(control.id, preProcessOut.OriginalText); } else if (inputType == "IMAGE") { if (imageUrlDuringCallBack) { control.src = preProcessOut.OriginalText; } else { preProcessOut.ParentElement.innerHTML = preProcessOut.OriginalText; } } else if (inputType == "SUBMIT") { control.value = preProcessOut.OriginalText; } } else if (control.nodeName.toUpperCase() == "SELECT") { SetLabelText(control.id, preProcessOut.OriginalText); } else { control.innerHTML = preProcessOut.OriginalText; } } if (clientCallBack) { clientCallBack(result, clientCallBackArg); } } function Anthem_FireCallBackEvent( control, e, eventTarget, eventArgument, causesValidation, validationGroup, imageUrlDuringCallBack, textDuringCallBack, enabledDuringCallBack, preCallBackFunction, postCallBackFunction, callBackCancelledFunction, includeControlValuesWithCallBack, updatePageAfterCallBack ) { var preProcessOut = new Anthem_PreProcessCallBackOut(); var preProcessResult = Anthem_PreProcessCallBack( control, e, eventTarget, causesValidation, validationGroup, imageUrlDuringCallBack, textDuringCallBack, enabledDuringCallBack, preCallBackFunction, callBackCancelledFunction, preProcessOut ); if (preProcessResult) { Anthem_FireEvent( eventTarget, eventArgument, function(result) { Anthem_PostProcessCallBack( result, control, eventTarget, null, null, imageUrlDuringCallBack, textDuringCallBack, postCallBackFunction, preProcessOut ); }, null, includeControlValuesWithCallBack, updatePageAfterCallBack ); } } function AnthemListControl_OnClick( e, causesValidation, validationGroup, textDuringCallBack, enabledDuringCallBack, preCallBackFunction, postCallBackFunction, callBackCancelledFunction, includeControlValuesWithCallBack, updatePageAfterCallBack ) { var target = e.target || e.srcElement; if (target.nodeName.toUpperCase() == "LABEL" && target.htmlFor != '') return; var eventTarget = target.id.split("_").join("$"); Anthem_FireCallBackEvent( target, e, eventTarget, '', causesValidation, validationGroup, '', textDuringCallBack, enabledDuringCallBack, preCallBackFunction, postCallBackFunction, callBackCancelledFunction, true, true ); } function GetLabelText(id) { var labels = document.getElementsByTagName('label'); for (var i = 0; i < labels.length; i++) { if (labels[i].htmlFor == id) { return labels[i].innerHTML; } } return null; } function SetLabelText(id, text) { var labels = document.getElementsByTagName('label'); for (var i = 0; i < labels.length; i++) { if (labels[i].htmlFor == id) { labels[i].innerHTML = text; return; } } } //]]> </script> <script type="text/javascript" src="http://blog.csdn.net/WebResource.axd?d=1wkz58Z1rsBL-eFHJPld5ERhyO-4qXtUzHp7Q1KwrOE1&t=632963535947587500"></script> <script type="text/javascript"> </script>

COM&&ATL

COM空间

  CSDN |  社区 |  技术中心 |  BLOG首页 |  我的首页 |  个人档案 |  联系作者 |  聚合  |   |  搜索 |  登录
  40篇原创: 0篇翻译: 12篇转载: 13737次点击: 24个评论: 0个Trackbacks

公告

wangweixing2000,陕西人,2003年毕业于沈阳工业学院(现名:沈阳理工大学)信息分院(计算机科学与技术系)。 Locations of visitors to this page <script type="text/javascript"> function cantload() { img = document.getElementById("clustrMapsImg"); img.onerror = null; img.src = "http://clustrmaps.com/images/clustrmaps-back-soon.jpg"; document.getElementById("clustrMapsLink").href = "http://clustrmaps.com"; } img = document.getElementById("clustrMapsImg"); img.onerror = cantload; </script>

文章

收藏

相册

csdn友情连接

推荐经典网站

我的网站

存档

最近评论



<script language="javascript" src="http://tag.csdn.net/urltag.aspx" type="text/javascript"></script> 作者tag: atl组件和控件编程  wtl,atl CSDN 推荐tag: .net  c++  vc++  visual c++  win32  win32 api  wtl  atl  模板  安装 

<script type="text/javascript">function StorePage(){d=document;t=d.selection?(d.selection.type!='None'?d.selection.createRange().text:''):(d.getSelection?d.getSelection():'');void(keyit=window.open('http://www.365key.com/storeit.aspx?t='+escape(d.title)+'&u='+escape(d.location.href)+'&c='+escape(t),'keyit','scrollbars=no,width=475,height=575,left=75,top=20,status=no,resizable=yes'));keyit.focus();}</script>

WTL开发者指南

注:本文是袁晓辉根据Eamon O’ TuathailWTL Developer’s Guide(www.clipcode.biz ) 翻译而来,发布在www.farproc.com 和 blog.csdn.net。作者翻译本文仅仅是为自己和其他WTL爱好者学习之用,请勿用于商业用途。你可以转载本文,但必须保证本文的完整性,并保留该声明!

 

1 WTL概述

目标

该章的目标为:

Ÿ       WTL在整个VC++开发中所处的位置

Ÿ       WTL的目标

Ÿ       WTL和其他用户界面(UI)开发技术的对比

Ÿ       介绍WTL的安装

Ÿ       介绍WTL安装了什么

Ÿ       列举可用的开发资源

Ÿ       对后面各章的介绍

 

概述

以前Visual C++开发者往往选择MFC,因为它提供的功能涵盖了非常广的编程技术领域而模板库所涉及的领域相对狭窄。

最近模板库涉足的领域和质量都有很大发展,因而越来越多的人选择了模板库。

ATL设计的目标是提供快速紧凑的COM组件,ISO C++STL提供集合(collections),VC++ OLE DB Data Consumer and Provider模板提供数据库支持。与之相对应,MFC提供了一个C++类库,这个类库包含对COM创建、集合类(collection class)、数据库和用户界面的普遍支持。

大部分的高级开发人员更喜欢较新的模板开发方式。MFC很庞大、臃肿、对线程不友好,并且,基本上已经过时了!模板的方法既快速(当正确设计时)、有弹性而且涵盖了所有最新的技术,拿来应对新的开发需要是再好不过的了。

到目前为止,让模板狂热分子头疼的主要问题在于如何创建图形用户界面(GUI)。ATL确实提供了一个轻量级的Win32/64 windowing 包装类,但是它并没有涵盖所有的UI需求。ATL开发者为了琐碎的用户界面处理不得不重操MFC UI编程或者用VB给自己的ATL COM做前端,这两种方法都不能完全令人满意。

进入正题,开始说窗口模板库(Windows Template Library)。WTL是一个居于Win32/64 API之上的一个包装类和方便易用的增强模板的高级集合,它对图形用户界面提供了广泛的支持。WTL不但保持着模板库的真正血统,小巧,快速,不具侵入性而且包括了最新的UI理念,非常好地支持多线程。你可以单独使用WTL或和ATLSTLVC++ Data Templates、第三方类库,以及你自己的类库一起使用以便利用他们各自的优点。WTL通常不依赖外部的DLLs,如果你用WTL开发程序的话,你提供给最终用户的可以只有一个EXE文件。

 

WTL的目标

WTL之于用户界面就像ATL之于COMSTL之于collection。和它的堂兄一样,WTL需要一段时间来学习,但是一旦掌握了,就没有比它更好的开发高级程序的途径了。从本质上说,WTL完成了三个重大任务:

    提供了一个应用程序框架(framwork

    集合了UI功能

    包装了windows控件、系统对话框和GDI对象

让我们逐个来看。

 

提供了一个应用程序框架

 

WTL提供了一个轻量级的广泛的应用程序框架,这个框架为基于它的程序提供了非常有价值的工具。它的目标既不是像MFC框架那样难以理解也不像手工写WinMain那样原始。

WTL包含了一系列的C++模板和一个WTL AppWizardAppWizard向程序开发者提出几个问题,然后据此生产VC++工程和程序源代码,源代码里多数是WTL模板类的实例或继承类。一般来说,开发者仅仅需要改变AppWizard生产的代码,而那些很少需要改动的“样板”式的代码都被WTL模板封装了。

WTL中有一个管理整个模块(一个DLLEXE)的类,这个类在AppWizard生成的代码中实例化,在自动生成的WinMain中被初始化和终结。一个WTL程序也可以作为COM server,支持可编程的应用程序。WTL提供了消息处理机制,支持消息过滤和空闲(on-idle)功能,WM_SETTINGCHANGE引发的改变通知也可以被截获。

MFC提供了一个“文档-框架-视图”结构,WTL也提供框架(比如一个包括菜单条、工具条和状态条的顶层窗口)和视图,但是根本不支持文档。MFC的那套方法往往妨碍高级的开发者,它的文档是基于二进制数据的序列化(serialize),在当今互联网的世界里,基于XML/XSL的文档格式越来越流行,对于弹性化存储机制的需求也越来越多(可能根本就不是存储在本地磁盘上的),文档可能需要运用大量的远程机制,比如WebDAVFTP等,进行存储。WTL没有提供任何关于数据或存储格式的功能和机制,程序开发者可以自己编写代码,比如利用高效的Win32 API WriteFileGather/ReadFileScatterXML解析器来根据需要做这些工作。

WTL支持基于对话框的程序,单文档界面(SDI)和多文档界面(MDI)。对于SDI它同样支持多线程SDI,每个SDI窗口一个线程,多数高级的应用程序很有可能用到这种构架(比如,模仿Word2000的方法)。WTL不支持多线程MDI构架,即不支持一个MID子窗口一个线程,但这是可以理解的,比如这样可以避免线程和MDI父子窗口之间交互的许多问题。

 

集合了UI功能

WTL提供了一组实现现代的用户界面所必不可少的特性。

WTL提供了一个框架窗口用于管理一个包含了命令条(增强的菜单条)、工具条的rebar,一个状态条,一个或多个视图(views)。视图可以基于一个普通的空窗口RechEidtListViewTreeView之类的控件。如果使用了多个视图,WTL的分割功能就派上用场了,视图也支持滚动,状态条可以是一个pane或多个panes

WTL也支持传统的菜单条和工具条,但是它的主菜单表现概念是基于命令条的。一个命令条可以容纳菜单和工具条图标,可以放在一个rebar里显示并四处移动。如果一个菜单项目和一个工具条按钮拥有同样的命令ID的话,菜单项目旁边会自动显示对应的图标。WTL支持可停泊或浮动的命令条,WTL的命令条和Internet Explorer中的很像但和Word2000Visual Studio的(更令人向往)不一样,WTL不支持可以隐藏不常用的项目的菜单(比如Word2000的)。动态数据交换的概念允许数据在可见的UI控件和C++数据成员之间双向传递,和MFCDDX很相近。可以用ATL生成对话框,包括那些支持ActiveX控件和窗口容器,利用Visual C++Resource View可以用来在对话框资源上布局标准控件、通用控件和ActiveX控件,利用ATL Window Message Handler Wizard来映射这些控件的消息。

WTL提供了用来管理属性薄、属性页创建和向导创建的模板创建。

打印机设备场景、打印预览、devmode处理和打印作业管理功能可以用来实现文档打印。

使用特殊的设备场景、文件管理器和增强metafile信息类可以支持增强的metafile

 

包装了windows控件、系统对话框和GDI对象

ATL提供了对普通窗口的控制但是没有提供对基于不同窗口类的窗口的特殊支持,开发者不得不根据需要手工编写SnedMessage调用。对于和EditTreeView的通讯他们不得不使用ATLCWindow并且调用Win32SendMessageEM_LIMITTEXTVM_SETITEM还得担心消息的参数是否正确(SendMessage没有类型检查,全部为WPARAMLPARAM)。当程序接收到它们时也是“裸数据”- wParamlParam被送到了消息队列,必须通过代码来转换数据为合适的类型,这时还得保证转换对于每个消息是正确的

一些ATL开发者都使用从ATLCON例程中提取出的代码,这些代码提供了对Windows UI 元素的一些封装封装。WTL现在以及进化出了一整套的广泛的封装类,这些类实现了对所有标准和通用窗口控件、系统对话框和所有GDI对象以及其他的封装。WTL中无论是对editbuttonlistboxtreeviewlistview等还是对通用文件对话框、颜色对话框、字体对话框等还是对DCpenbrushregionfont等都有对应的包装类。

另外,WTL还提供了一整套的message crackers,进入窗口的消息被映射到指定的处理函数。使用ATL时,消息处理函数收到的是lParamwParam,而使用WTLmessage cracker,消息处理函数的参数都是具体消息的具体类型。比如:WM_LBUTTONDOWN消息有一个CPoint类型的参数(CPointWTLWin32 POINT结构的包装-WTL包装了每一个通用的结构)。注意,ATL Windows Message Handler Wizard 使用ATL的原始消息映射,非WTLmessage crackersVC++下一版可能对向导支持会好一些。

许多的开发者选择MFC仅仅是为了使用CStringWTL自己也提供了对CString的实现,并且是几乎和MFC一样的实现。WTLCString支持ASCIIMBCSUNICODE和自动化的BSTR,支持这些类型间的相互转换,支持字符串连接、截断、和比较,支持printf一样的格式化处理,支持弹性的内存管理。

 

WTLATLWin32/64 API之间的关系

WTL是基于ATLWin3264API的,ATL是基于Win32/64 API的,当谈到它们之间的关系时,我们应该从源代码和二进制两个视角来讨论。

源代码

Win32/64 API是成千上百的C函数的集合,涵盖了相当大的范围,包括我们感兴趣的COMwindowing

ATLC++模板的集合,几乎就是为COM编程量身定做的,但是也包括广泛的底层windowing功能的支持。

WTLC++模板的集合,它的关注点在高层的windowing功能,WTL不依赖于COM,但是可以和COM一起使用。只要你不在WTL中选择支持“Com Server”,CoInitialize就不会被调用。

WTL使用ATLwindowing服务,所以要使用WTL,你必须访问ATL模板。

ATL调用Win32/64 APIWTL调用ATL模板和Win32/64 API,你的程序代码调用WTLATLWin32/64 API

当你运行了WTL AppWizard,他给你生成一些源文件,这些源文件将成为你程序代码的一部分,这些生成的代码调用WTL,构建程序框架。

WTL支持Windows 2000Windows MeWindows NT4Windows 98Windows95, OSR 2和“经典”windows 95WTL并不是只支持它们 “最小公分母”,而是提供#define(比如 _RICHEDIT_VER, _WIN32_IE, _WIN32_WINNT and WINVER)来决定使用那些Win32/64的特性。WTL不会自动判别被安装到的操作系统的版本或IE的版本(比如,它不使用LoadLibrary /GetProceAddress GetVersionEx APIs来判别)。程序开发者必须指定#define宏,而且在编译期间已经假定程序在正确版本的客户机上可以正常运行,否则不能。

 

举个例子,菜单中的“>>”符号(v型符号),当一个命令条因为窗口大小改变而被截短时如果不使用v型符号,超出窗口边界的那部分命令条就别直接“剪掉”了。

V型符号是一个包含两个大于号的标志,当被按下时,就弹出一个菜单菜单上包含由于窗口大小改变而被从命令条尾部截去的那些项目。如果包含了IE5v型符号就被支持,反之不支持。

WTL程序中,如果_WIN32_IE 别定义为0x0500或更高(比如,在stdafxh里)那么v型符号在必要时就会显示。

 

二进制

ATLWTL是作为一组包含C++模板的头文件的形式发布的。ATL中有少数的类可以作为一个独立的DLL来使用,但是多数的程序员也把它们的源代码包含到自己的工程里面去,对于WTL程序来说,这也是被推荐的。WTL没有可以作为独立的DLL的东西,所以WTLC++类(并非全部)将被包含进每个程序。当你编译WTL程序,你最终会得到一个EXEDLL,这就是你唯一需要交付最终用户的程序,不依赖外部库,当然除去操作系统自带的DLL,比如kernel32.dll,但是这些库基本上都存在(如果不存在,windows本身也许就无法正常工作了)。一个例外:如果你需要使用浮点数,你不得不使用C语言运行库(CRT)。

 

WTL-ATLMFCVBJavaDHTML等的抉择

……(省略废话数百字)

 

为什么选择WTL

WTL不像MFC那样笨重,最终的应用程序也比较小巧。对于高级程序,WTL给你使用API的机会,但是比纯粹的API编程节省时间。

Win32 API可以直接使用,但是有时候它们看起来相当独立,很难决定如何将它们结合起来使用。

WTL提供了更多的可以完成同样操作的功能。

由于被调用的方法是在编译期而不是运行期决定的,这就是它在效率和代码规模方面占有优势,模板库正在变得越来越流行。

 

安装WTL

WTL的发展很快,它的发行机制也在时常变化。

未来的发行模式

很可能下一版VC++发布时将会整合进WTL,因此,它可能会和ATLSTLData Templates一起自动安装。

现在的安装机制

当前,WTL是作为Microsoft Platform SDK January 2000 或更高)的一部分发布的,需要手动分三步安装,未来可能会自动完成。

第一步-取得文件

可以从MSDN光盘的第三张或http://msdn.microsoft.com得到安装文件。你可以选择全部安装这个SDK或仅仅安装包含Windows Template Library的部分。(译者注:你也以在www.farproc.com下载WTL7.1安装包

 

注意:WTL7.1安装的后面步骤与作者原文有出入,所以下面关于安装的就没有安装原文翻译

 

第二步-WTL Appwizard

安装后打开WTL的安装目录(比如c:/Program Files/Microsoft Platform SDK)进入对应的AppWizXX目录,其中XX为你的VC的版本号,比如你用的是VC6.0的话就进入AppWiz60,你会看到一个setup.js文件,点击运行它就可以为你自动安装WTLAppWizard了。如果你无法运行这个文件或想手工安装,就复制这个目录下的AtlApp60.awx到“……/Microsoft Visual Studio/Common/MSDev98/Template”目录下也可以的。

然后打开vc,在new对话框中就可以看到“ATL/WTL AppWizard”选项了。

 

 

第三步-更新路径

第三步是设置路径以便让编译器找到WTL头文件。WTL是一个包含C++头文件(没有.cpp文件)的模板库,当编译时,编译器需要找到这些文件,这些文件在WTL安装目录/Src/WTL/include下,你可以在你的源文件里包含这个全路径(不推荐,但也可以),或把这个路径添加到VC的“Include Files”里,如下图

 

WTL安装了什么

WTL安装后,会有一个或若干个AppWizXX目录,这是WTL AppWizard文件的所在地;一个Include目录为WTL的头文件(也就是源文件)所在;还有一个samples,一些WTL程序的例子。

                                                       <待续>

接下篇:http://blog.csdn.net/uoyevoli/archive/2005/05/30/384357.aspx



Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=381808

 

Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=1542424


<script src="http://localhost:82/PromoteIcon.aspx?id=1542424" type="text/javascript"></script> [ 收藏到我的网摘]   星发表于 2007年03月27日 10:12:00
<script type="text/javascript">function hide(){showComment();}</script>
相关文章:



<script type="text/javascript">document.write(" ");</script>
没有评论。
<script language="javascript" type="text/javascript"> ad_width=468; ad_height=60; adcss=2; unionuser=19; ad_type='j'; count=5; </script> <script type="text/javascript" src="http://tagegg.csdn.net/showads.js" language="javascript"></script> <script src="http://tagegg.csdn.net/b.aspx?action=displayad&unionuser=19&unionurl=http%3A%2F%2Fblog.csdn.net%2Fwangweixing2000%2Farchive%2F2007%2F03%2F27%2F1542424.aspx&adcss=2&ad_type=j&width=468&height=60&ad_color=&ad_color_border=&count=5" language="JavaScript1.1" type="text/javascript"></script> <script type="text/javascript" src="http://blog.csdn.net/js/showgm.js" language="javascript"></script>

发表评论

 
大名:
网址:
评论 
   
<script src="http://www.csdn.net/common/counter.js" type="text/javascript"></script> width="0" scrolling="no" height="0" frameborder="no" src="http://www.csdn.net/ggmm/dd333.htm" border="0" name="myframe" id="myframe"> <script type="text/javascript"> </script> <script type="text/javascript"> </script> <script language="javascript" type="text/javascript"> </script>  
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值