2 * MIME OLE International interface
4 * Copyright 2008 Huw Davies for CodeWeavers
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #define NONAMELESSUNION
36 #include "wine/list.h"
37 #include "wine/unicode.h"
38 #include "wine/debug.h"
40 #include "inetcomm_private.h"
42 WINE_DEFAULT_DEBUG_CHANNEL(inetcomm
);
52 IMimeInternational IMimeInternational_iface
;
57 LONG next_charset_handle
;
58 HCHARSET default_charset
;
61 static inline internat_impl
*impl_from_IMimeInternational(IMimeInternational
*iface
)
63 return CONTAINING_RECORD(iface
, internat_impl
, IMimeInternational_iface
);
66 static inline HRESULT
get_mlang(IMultiLanguage
**ml
)
68 return CoCreateInstance(&CLSID_CMultiLanguage
, NULL
, CLSCTX_INPROC_SERVER
| CLSCTX_INPROC_HANDLER
,
69 &IID_IMultiLanguage
, (void **)ml
);
72 static HRESULT WINAPI
MimeInternat_QueryInterface( IMimeInternational
*iface
, REFIID riid
, LPVOID
*ppobj
)
74 if (IsEqualGUID(riid
, &IID_IUnknown
) ||
75 IsEqualGUID(riid
, &IID_IMimeInternational
))
77 IMimeInternational_AddRef( iface
);
82 FIXME("interface %s not implemented\n", debugstr_guid(riid
));
86 static ULONG WINAPI
MimeInternat_AddRef( IMimeInternational
*iface
)
88 internat_impl
*This
= impl_from_IMimeInternational( iface
);
89 return InterlockedIncrement(&This
->refs
);
92 static ULONG WINAPI
MimeInternat_Release( IMimeInternational
*iface
)
94 internat_impl
*This
= impl_from_IMimeInternational( iface
);
97 refs
= InterlockedDecrement(&This
->refs
);
100 charset_entry
*charset
, *cursor2
;
102 LIST_FOR_EACH_ENTRY_SAFE(charset
, cursor2
, &This
->charsets
, charset_entry
, entry
)
104 list_remove(&charset
->entry
);
105 HeapFree(GetProcessHeap(), 0, charset
);
107 This
->cs
.DebugInfo
->Spare
[0] = 0;
108 DeleteCriticalSection(&This
->cs
);
109 HeapFree(GetProcessHeap(), 0, This
);
115 static HRESULT WINAPI
MimeInternat_SetDefaultCharset(IMimeInternational
*iface
, HCHARSET hCharset
)
117 internat_impl
*This
= impl_from_IMimeInternational( iface
);
119 TRACE("(%p)->(%p)\n", iface
, hCharset
);
121 if(hCharset
== NULL
) return E_INVALIDARG
;
122 /* FIXME check hCharset is valid */
124 InterlockedExchangePointer(&This
->default_charset
, hCharset
);
129 static HRESULT WINAPI
MimeInternat_GetDefaultCharset(IMimeInternational
*iface
, LPHCHARSET phCharset
)
131 internat_impl
*This
= impl_from_IMimeInternational( iface
);
134 TRACE("(%p)->(%p)\n", iface
, phCharset
);
136 if(This
->default_charset
== NULL
)
139 hr
= IMimeInternational_GetCodePageCharset(iface
, GetACP(), CHARSET_BODY
, &hcs
);
141 InterlockedCompareExchangePointer(&This
->default_charset
, hcs
, NULL
);
143 *phCharset
= This
->default_charset
;
148 static HRESULT
mlang_getcodepageinfo(UINT cp
, MIMECPINFO
*mlang_cp_info
)
157 hr
= IMultiLanguage_GetCodePageInfo(ml
, cp
, mlang_cp_info
);
158 IMultiLanguage_Release(ml
);
163 static HRESULT WINAPI
MimeInternat_GetCodePageCharset(IMimeInternational
*iface
, CODEPAGEID cpiCodePage
,
164 CHARSETTYPE ctCsetType
,
165 LPHCHARSET phCharset
)
168 MIMECPINFO mlang_cp_info
;
170 TRACE("(%p)->(%d, %d, %p)\n", iface
, cpiCodePage
, ctCsetType
, phCharset
);
174 hr
= mlang_getcodepageinfo(cpiCodePage
, &mlang_cp_info
);
177 const WCHAR
*charset_nameW
= NULL
;
184 charset_nameW
= mlang_cp_info
.wszBodyCharset
;
187 charset_nameW
= mlang_cp_info
.wszHeaderCharset
;
190 charset_nameW
= mlang_cp_info
.wszWebCharset
;
193 return MIME_E_INVALID_CHARSET_TYPE
;
195 len
= WideCharToMultiByte(CP_ACP
, 0, charset_nameW
, -1, NULL
, 0, NULL
, NULL
);
196 charset_name
= HeapAlloc(GetProcessHeap(), 0, len
);
197 WideCharToMultiByte(CP_ACP
, 0, charset_nameW
, -1, charset_name
, len
, NULL
, NULL
);
198 hr
= IMimeInternational_FindCharset(iface
, charset_name
, phCharset
);
199 HeapFree(GetProcessHeap(), 0, charset_name
);
204 static HRESULT
mlang_getcsetinfo(const char *charset
, MIMECSETINFO
*mlang_info
)
206 DWORD len
= MultiByteToWideChar(CP_ACP
, 0, charset
, -1, NULL
, 0);
207 BSTR bstr
= SysAllocStringLen(NULL
, len
- 1);
211 MultiByteToWideChar(CP_ACP
, 0, charset
, -1, bstr
, len
);
217 hr
= IMultiLanguage_GetCharsetInfo(ml
, bstr
, mlang_info
);
218 IMultiLanguage_Release(ml
);
221 if(FAILED(hr
)) hr
= MIME_E_NOT_FOUND
;
225 static HCHARSET
add_charset(struct list
*list
, MIMECSETINFO
*mlang_info
, HCHARSET handle
)
227 charset_entry
*charset
= HeapAlloc(GetProcessHeap(), 0, sizeof(*charset
));
229 WideCharToMultiByte(CP_ACP
, 0, mlang_info
->wszCharset
, -1,
230 charset
->cs_info
.szName
, sizeof(charset
->cs_info
.szName
), NULL
, NULL
);
231 charset
->cs_info
.cpiWindows
= mlang_info
->uiCodePage
;
232 charset
->cs_info
.cpiInternet
= mlang_info
->uiInternetEncoding
;
233 charset
->cs_info
.hCharset
= handle
;
234 charset
->cs_info
.dwReserved1
= 0;
235 list_add_head(list
, &charset
->entry
);
237 return charset
->cs_info
.hCharset
;
240 static HRESULT WINAPI
MimeInternat_FindCharset(IMimeInternational
*iface
, LPCSTR pszCharset
,
241 LPHCHARSET phCharset
)
243 internat_impl
*This
= impl_from_IMimeInternational( iface
);
244 HRESULT hr
= MIME_E_NOT_FOUND
;
245 charset_entry
*charset
;
247 TRACE("(%p)->(%s, %p)\n", iface
, debugstr_a(pszCharset
), phCharset
);
251 EnterCriticalSection(&This
->cs
);
253 LIST_FOR_EACH_ENTRY(charset
, &This
->charsets
, charset_entry
, entry
)
255 if(!lstrcmpiA(charset
->cs_info
.szName
, pszCharset
))
257 *phCharset
= charset
->cs_info
.hCharset
;
263 if(hr
== MIME_E_NOT_FOUND
)
265 MIMECSETINFO mlang_info
;
267 LeaveCriticalSection(&This
->cs
);
268 hr
= mlang_getcsetinfo(pszCharset
, &mlang_info
);
269 EnterCriticalSection(&This
->cs
);
272 *phCharset
= add_charset(&This
->charsets
, &mlang_info
,
273 UlongToHandle(InterlockedIncrement(&This
->next_charset_handle
)));
276 LeaveCriticalSection(&This
->cs
);
280 static HRESULT WINAPI
MimeInternat_GetCharsetInfo(IMimeInternational
*iface
, HCHARSET hCharset
,
281 LPINETCSETINFO pCsetInfo
)
283 internat_impl
*This
= impl_from_IMimeInternational( iface
);
284 HRESULT hr
= MIME_E_INVALID_HANDLE
;
285 charset_entry
*charset
;
287 TRACE("(%p)->(%p, %p)\n", iface
, hCharset
, pCsetInfo
);
289 EnterCriticalSection(&This
->cs
);
291 LIST_FOR_EACH_ENTRY(charset
, &This
->charsets
, charset_entry
, entry
)
293 if(charset
->cs_info
.hCharset
== hCharset
)
295 *pCsetInfo
= charset
->cs_info
;
301 LeaveCriticalSection(&This
->cs
);
306 static HRESULT WINAPI
MimeInternat_GetCodePageInfo(IMimeInternational
*iface
, CODEPAGEID cpiCodePage
,
307 LPCODEPAGEINFO pCodePageInfo
)
313 static HRESULT WINAPI
MimeInternat_CanConvertCodePages(IMimeInternational
*iface
, CODEPAGEID cpiSource
,
319 TRACE("(%p)->(%d, %d)\n", iface
, cpiSource
, cpiDest
);
321 /* Could call mlang.IsConvertINetStringAvailable() to avoid the COM overhead if need be. */
326 hr
= IMultiLanguage_IsConvertible(ml
, cpiSource
, cpiDest
);
327 IMultiLanguage_Release(ml
);
333 static HRESULT WINAPI
MimeInternat_DecodeHeader(IMimeInternational
*iface
, HCHARSET hCharset
,
335 LPPROPVARIANT pDecoded
,
336 LPRFC1522INFO pRfc1522Info
)
342 static HRESULT WINAPI
MimeInternat_EncodeHeader(IMimeInternational
*iface
, HCHARSET hCharset
,
345 LPRFC1522INFO pRfc1522Info
)
351 static HRESULT WINAPI
MimeInternat_ConvertBuffer(IMimeInternational
*iface
, CODEPAGEID cpiSource
,
352 CODEPAGEID cpiDest
, LPBLOB pIn
, LPBLOB pOut
,
358 TRACE("(%p)->(%d, %d, %p, %p, %p)\n", iface
, cpiSource
, cpiDest
, pIn
, pOut
, pcbRead
);
363 /* Could call mlang.ConvertINetString() to avoid the COM overhead if need be. */
369 UINT in_size
= pIn
->cbSize
, out_size
;
371 hr
= IMultiLanguage_ConvertString(ml
, &mode
, cpiSource
, cpiDest
, pIn
->pBlobData
, &in_size
,
373 if(hr
== S_OK
) /* S_FALSE means the conversion could not be performed */
375 pOut
->pBlobData
= CoTaskMemAlloc(out_size
);
381 in_size
= pIn
->cbSize
;
382 hr
= IMultiLanguage_ConvertString(ml
, &mode
, cpiSource
, cpiDest
, pIn
->pBlobData
, &in_size
,
383 pOut
->pBlobData
, &out_size
);
388 pOut
->cbSize
= out_size
;
391 CoTaskMemFree(pOut
->pBlobData
);
394 IMultiLanguage_Release(ml
);
400 static HRESULT WINAPI
MimeInternat_ConvertString(IMimeInternational
*iface
, CODEPAGEID cpiSource
,
401 CODEPAGEID cpiDest
, LPPROPVARIANT pIn
,
408 TRACE("(%p)->(%d, %d, %p %p)\n", iface
, cpiSource
, cpiDest
, pIn
, pOut
);
413 if(cpiSource
== CP_UNICODE
) cpiSource
= GetACP();
414 src_len
= strlen(pIn
->u
.pszVal
);
417 cpiSource
= CP_UNICODE
;
418 src_len
= strlenW(pIn
->u
.pwszVal
) * sizeof(WCHAR
);
428 UINT in_size
= src_len
, out_size
;
430 hr
= IMultiLanguage_ConvertString(ml
, &mode
, cpiSource
, cpiDest
, (BYTE
*)pIn
->u
.pszVal
, &in_size
,
432 if(hr
== S_OK
) /* S_FALSE means the conversion could not be performed */
434 out_size
+= (cpiDest
== CP_UNICODE
) ? sizeof(WCHAR
) : sizeof(char);
436 pOut
->u
.pszVal
= CoTaskMemAlloc(out_size
);
443 hr
= IMultiLanguage_ConvertString(ml
, &mode
, cpiSource
, cpiDest
, (BYTE
*)pIn
->u
.pszVal
, &in_size
,
444 (BYTE
*)pOut
->u
.pszVal
, &out_size
);
448 if(cpiDest
== CP_UNICODE
)
450 pOut
->u
.pwszVal
[out_size
/ sizeof(WCHAR
)] = 0;
451 pOut
->vt
= VT_LPWSTR
;
455 pOut
->u
.pszVal
[out_size
] = '\0';
460 CoTaskMemFree(pOut
->u
.pszVal
);
463 IMultiLanguage_Release(ml
);
468 static HRESULT WINAPI
MimeInternat_MLANG_ConvertInetReset(IMimeInternational
*iface
)
474 static HRESULT WINAPI
MimeInternat_MLANG_ConvertInetString(IMimeInternational
*iface
, CODEPAGEID cpiSource
,
485 static HRESULT WINAPI
MimeInternat_Rfc1522Decode(IMimeInternational
*iface
, LPCSTR pszValue
,
494 static HRESULT WINAPI
MimeInternat_Rfc1522Encode(IMimeInternational
*iface
, LPCSTR pszValue
,
502 static IMimeInternationalVtbl mime_internat_vtbl
=
504 MimeInternat_QueryInterface
,
506 MimeInternat_Release
,
507 MimeInternat_SetDefaultCharset
,
508 MimeInternat_GetDefaultCharset
,
509 MimeInternat_GetCodePageCharset
,
510 MimeInternat_FindCharset
,
511 MimeInternat_GetCharsetInfo
,
512 MimeInternat_GetCodePageInfo
,
513 MimeInternat_CanConvertCodePages
,
514 MimeInternat_DecodeHeader
,
515 MimeInternat_EncodeHeader
,
516 MimeInternat_ConvertBuffer
,
517 MimeInternat_ConvertString
,
518 MimeInternat_MLANG_ConvertInetReset
,
519 MimeInternat_MLANG_ConvertInetString
,
520 MimeInternat_Rfc1522Decode
,
521 MimeInternat_Rfc1522Encode
524 static internat_impl
*global_internat
;
526 HRESULT
MimeInternational_Construct(IMimeInternational
**internat
)
528 global_internat
= HeapAlloc(GetProcessHeap(), 0, sizeof(*global_internat
));
529 global_internat
->IMimeInternational_iface
.lpVtbl
= &mime_internat_vtbl
;
530 global_internat
->refs
= 0;
531 InitializeCriticalSection(&global_internat
->cs
);
532 global_internat
->cs
.DebugInfo
->Spare
[0] = (DWORD_PTR
)(__FILE__
": global_internat.cs");
534 list_init(&global_internat
->charsets
);
535 global_internat
->next_charset_handle
= 0;
536 global_internat
->default_charset
= NULL
;
538 *internat
= &global_internat
->IMimeInternational_iface
;
540 IMimeInternational_AddRef(*internat
);
544 HRESULT WINAPI
MimeOleGetInternat(IMimeInternational
**internat
)
546 TRACE("(%p)\n", internat
);
548 *internat
= &global_internat
->IMimeInternational_iface
;
549 IMimeInternational_AddRef(*internat
);
553 HRESULT WINAPI
MimeOleFindCharset(LPCSTR name
, LPHCHARSET charset
)
555 IMimeInternational
*internat
;
558 TRACE("(%s, %p)\n", debugstr_a(name
), charset
);
560 hr
= MimeOleGetInternat(&internat
);
563 hr
= IMimeInternational_FindCharset(internat
, name
, charset
);
564 IMimeInternational_Release(internat
);
569 HRESULT WINAPI
MimeOleGetCharsetInfo(HCHARSET hCharset
, LPINETCSETINFO pCsetInfo
)
571 IMimeInternational
*internat
;
574 TRACE("(%p, %p)\n", hCharset
, pCsetInfo
);
576 hr
= MimeOleGetInternat(&internat
);
579 hr
= IMimeInternational_GetCharsetInfo(internat
, hCharset
, pCsetInfo
);
580 IMimeInternational_Release(internat
);
585 HRESULT WINAPI
MimeOleGetDefaultCharset(LPHCHARSET charset
)
587 IMimeInternational
*internat
;
590 TRACE("(%p)\n", charset
);
592 hr
= MimeOleGetInternat(&internat
);
595 hr
= IMimeInternational_GetDefaultCharset(internat
, charset
);
596 IMimeInternational_Release(internat
);