c++ - 如何将 CString 转换为 VT_ARRAY|VT_UI1

标签 c++ mfc atl

我继承了一个使用 CppWebBrowser 控件的旧应用程序。
该应用程序一直对 GET 请求使用::Navigate 方法,但现在我需要开始将它用于 POST;但是,现有代码不适用于 POST。我对使用这些数据类型的了解有限,但我怀疑它与 CString 到 VT_ARRAY|VT_UI1 的转换有关。

void WebPostData(TCppWebBrowser *CppWebBrowser, CString sURL, CString sPostData)
{
    BSTR bstrHeaders = NULL;
    TVariant vFlags = { 0 }, vTargetFrameName = { 0 }, vPostData = { 0 }, vHeaders = { 0 };
    LPSAFEARRAY psa;
    LPCTSTR cszPostData = sPostData;
    UINT cElems = lstrlen(cszPostData);
    LPSTR pPostData;
    LPVARIANT pvPostData;

        bstrHeaders = SysAllocString(L"Content-Type: application/x-www-form-urlencodedrn");
        if (!bstrHeaders){
            Application->MessageBox("Could not allocate bstrHeaders", "Warning", MB_OK | MB_ICONWARNING);
            return;
        }

        V_VT(&vHeaders) = VT_BSTR;
        V_BSTR(&vHeaders) = bstrHeaders;

        pvPostData = vPostData;

        if (pvPostData){
            VariantInit(pvPostData);

            psa = SafeArrayCreateVector(VT_UI1, 0, cElems);
            if (!psa){
                return;
            }

            SafeArrayAccessData(psa, (LPVOID*)&pPostData);
            memcpy(pPostData, cszPostData, cElems);
            SafeArrayUnaccessData(psa);

            V_VT(pvPostData) = VT_ARRAY | VT_UI1;
            V_ARRAY(pvPostData) = psa;
        }

        CppWebBrowser->Navigate((TVariant)sURL, &vFlags, &vTargetFrameName, &vPostData, &vHeaders);
}

最佳答案

您显示的代码是从这个 C++Builder 示例中复制的:

How to post data using TCppWebBrowser in C++Builder

CString 是一个MFC类,但是C++Builder不支持MFC。原始代码使用 System::String 代替,它是一个 C++Builder 类。那么,您是否尝试将此代码转换为 Visual Studio?还是您真的在使用 C++Builder?你没说。

我假设您使用的是 C++Builder,否则您必须重新编写 Visual Studio 类的代码。

因此,让我们首先将 CString 改回 String。现在,根据您使用的 C++Builder 版本,StringANSI (C++Builder 3-2007) 或 UTF-16(C++Builder 2009+)。在后一种情况下有一个 _DELPHI_STRING_UNICODE 预编译器定义。

application/x-www-form-urlencoded 类型通常不会作为 UTF-16 发送,并且无法保证 Web 服务器能够处理它。所以我建议您将 sPostData 数据转换为 UTF-8,然后您可以创建一个 UTF-8 字节的 VT_UI1 数组:

#ifndef _DELPHI_STRING_UNICODE
inline bool NeedsUTF8(String s)
{
    // String is 1-indexed
    for (int i = 1; i <= s.Length(); ++i)
    {
        int ch = s[i];
        if (ch > 0x7F) return true;
    }
    return false;
}
#endif

void WebPostData(TCppWebBrowser *CppWebBrowser, String sURL, String sPostData)
{
    TVariant vFlags, vTargetFrameName, vPostData, vHeaders;

    // note: UTF8String was added in C++Builder 6,
    // use AnsiString in C++Builder 3-5 instead
    UTF8String utf8;

    #ifdef _DELPHI_STRING_UNICODE
    // C++Builder 2009+, convert from UTF-16 to UTF-8 directly
    utf8 = sPostData;
    #else
    // C++Builder 3-2007, check if ASCII or ANSI
    if (NeedsUTF8(sPostData))
    {
        // ANSI, so convert from ANSI to UTF-16 to UTF-8,
        // but you have to do the last step manually
        WideString wPostData(sPostData);
        int len = WideCharToMultiByte(CP_UTF8, wPostData.c_bstr(), wPostData.Length(), NULL, 0, NULL, NULL);
        utf8.SetLength(len);
        WideCharToMultiByte(CP_UTF8, wPostData.c_bstr(), wPostData.Length(), utf8.c_str(), len, NULL, NULL);
    }
    else
    {
        // ASCII, compatible with UTF-8 as-is
        utf8 = sPostData;
    }
    #endif

    LPSAFEARRAY psa = SafeArrayCreateVector(VT_UI1, 0, utf8.Length());
    if (!psa)
        return;

    // TVariant takes ownership of a safearray
    vPostData = psa;

    // TVariant takes ownership of a copy of a WideString's BSTR
    vHeaders = WideString("Content-Type: application/x-www-form-urlencoded;charset=utf-8\r\n");

    LPVOID pPostData;
    if (FAILED(SafeArrayAccessData(psa, &pPostData)))
        return;

    memcpy(pPostData, utf8.c_str(), utf8.Length());
    SafeArrayUnaccessData(psa);

    CppWebBrowser->Navigate((TVariant)sURL, &vFlags, &vTargetFrameName, &vPostData, &vHeaders);
}

关于c++ - 如何将 CString 转换为 VT_ARRAY|VT_UI1,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30515774/

相关文章:

c++ - 使用 Write 成员将 CString 写入 CFile 时每个字符后为空

c++ - 在 DHTML/AJAX 页面的情况下,如何使用浏览器帮助对象 (BHO) 获取完整的 HTML 正文?

c++ - COM 错误 0x80004003(无效指针)访问 MS Outlook 联系人

C++ 哈希表 - 如何解决 unordered_map 与自定义数据类型作为键的冲突?

c++ - C++ 中具有 std::vector 数据成员的 constexpr 成员函数

c++ - 具有一个默认参数的构造函数

c++ - 表达式 : string iterator not dereferencable while using boost regex

c++ - 使用函数指针调用 C++ 函数

c++ - 如何更新无模式对话框

c++ - 非平凡对象的 ATL 集合