(continued)

 |
Web Devs: Moonlight as a Game Developer and Win Cool Prizes by Accepting the RIA Run Challenge
Now, your mission--should you choose to accept: Take your shot at gaming stardom if you think you might have what it takes to build a cool RIA game and you could win an Xbox 360 or other fabulous prizes. Hurry! You only have until May 15, 2008 to enter.
»
Article: Leveraging Your Flash Development with Silverlight
You're not giving up Flash any time soon (and we don't blame you.) But if you could get your Flash application working in Silverlight, why wouldn't you? We show you the tools and techniques required to have your rockin' Flash application rolled for Silverlight. Learn more here.
»
Article: What Does it Take to Build the Best RIA?
With the proliferation of Rich Interactive Application (RIA) platform choices out there, you no longer have to take a one-size-fits-all approach to developing your next RIA application. Knowing the strengths (and weaknesses) of each platform can help you to decide the best RIA for your next application.
»
|
 |

Environment: Developed in (but not restricted to) VC++ .NET
I recently needed to support the asynchronous version of the send() operation of the XMLHTTPRequest object that's included in Microsoft's XML 4.0 library. To achieve this, I chose to leverage the "onreadystatechange" property to signal when the send() operation was completed and data was ready to be received.
IXMLHTTPRequest provides the "onreadystatechange" property to allow the caller to monitor and react to the state changes of an asynchronous send() invocation. The documentation for this property states that "onreadystatechange" is "not readily accessible" in C++. However, it turns out to be rather simple to implement using ATL templates.
Architecture
To receive onreadystatechange events, I created a class called CXMLHTTPEvent that implements the IDispEventSimpleImpl template. IDispEventSimpleImpl provides the minimum framework to capture events returned by XMLHTTPRequest.
class CXMLHTTPEvent : public IDispEventSimpleImpl< 1,
CXMLHTTPEvent, &__uuidof(MSXML::XMLDOMDocumentEvents)>
I then added a SINK_ENTRY_INFO to the SINK_MAP to capture the XMLHTTPRequest events.
BEGIN_SINK_MAP(CXMLHTTPEvent)
SINK_ENTRY_INFO( 1, __uuidof(MSXML::
XMLDOMDocumentEvents),
0, OnReadyStateChange,
&OnEventInfo)
END_SINK_MAP()
When an event is captured, I then notify the main process of the state change via a Windows message or function callback.
void __stdcall CXMLHTTPEvent::OnReadyStateChange ( )
{
ATLTRACE(L"CXMLHTTPEvent: ReadyStateChange = %i \n",
m_spRequest->readyState);
if (m_pCallBack)
m_pCallBack->OnReadyStateChange(m_spRequest->GetreadyState());
if (m_hwndPostWindow)
::PostMessage(m_hwndPostWindow, WM_XMLHTTP_READYSTATE_CHANGE,
0, MAKELPARAM(m_spRequest->GetreadyState(), 0));
}
Using CXMLHTTPEvent
After creating the XMLHTTPRequest object, create an instance of the CXMLHTTPEvent prior to invoking XMLHTTPRequest operations. This will connect XMLHTTPRequest with how you'd like your application to be notified about the state changes.
void CXMLHTTPCallBackDlg::OnBnClickedButton1()
{
m_spXMLHTTPRequest.CreateInstance(L"Msxml2.XMLHTTP.4.0");
...
m_pXMLHTTPEvent = new CXMLHTTPEvent(m_spXMLHTTPRequest,
GetSafeHwnd());
...
m_pXMLHTTPEvent = new CXMLHTTPEvent(m_spXMLHTTPRequest,
NULL, this);
..
m_spXMLHTTPRequest->open("GET", m_sURL.AllocSysString(),
VARIANT_TRUE);
m_spXMLHTTPRequest->send();
}
If you are using the message pump to get notifications, add a message handler and map entry for the WM_XMLHTTP_READYSTATE_CHANGE message. This message will be sent by CXMLHTTPEvent when XMLHTTPRequest signals a ready state change.
BEGIN_MESSAGE_MAP(CXMLHTTPCallBackDlg, CDialog)
...
ON_MESSAGE(WM_XMLHTTP_READYSTATE_CHANGE,
OnReadyStateChange2)
...
END_MESSAGE_MAP()
When WM_XMLHTTP_READYSTATE_CHANGE is sent, the lParam will contain the ReadyState value.
LRESULT CXMLHTTPCallBackDlg::OnReadyStateChange2(WPARAM wParam,
LPARAM lParam)
{
UINT nState = (UINT)lParam;
...
return 0;
}
If you're using the callback method, make sure the target class for events inherits from the CXMLHTTPEventCallBack class and implements the OnReadyStateChange(long lReadyState) method.
class CXMLHTTPCallBackDlg : public CDialog, CXMLHTTPEventCallBack
...
void CXMLHTTPCallBackDlg::OnReadyStateChange(long lReadyState)
{
...
}
Downloads
Download demo project - 23 Kb
Download source - 27 Kb