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 HeapFree(GetProcessHeap(), 0, This
);
113 static HRESULT WINAPI
MimeInternat_SetDefaultCharset(IMimeInternational
*iface
, HCHARSET hCharset
)
115 internat_impl
*This
= impl_from_IMimeInternational( iface
);
117 TRACE("(%p)->(%p)\n", iface
, hCharset
);
119 if(hCharset
== NULL
) return E_INVALIDARG
;
120 /* FIXME check hCharset is valid */
122 InterlockedExchangePointer(&This
->default_charset
, hCharset
);
127 static HRESULT WINAPI
MimeInternat_GetDefaultCharset(IMimeInternational
*iface
, LPHCHARSET phCharset
)
129 internat_impl
*This
= impl_from_IMimeInternational( iface
);
132 TRACE("(%p)->(%p)\n", iface
, phCharset
);
134 if(This
->default_charset
== NULL
)
137 hr
= IMimeInternational_GetCodePageCharset(iface
, GetACP(), CHARSET_BODY
, &hcs
);
139 InterlockedCompareExchangePointer(&This
->default_charset
, hcs
, NULL
);
141 *phCharset
= This
->default_charset
;
146 static HRESULT
mlang_getcodepageinfo(UINT cp
, MIMECPINFO
*mlang_cp_info
)
155 hr
= IMultiLanguage_GetCodePageInfo(ml
, cp
, mlang_cp_info
);
156 IMultiLanguage_Release(ml
);
161 static HRESULT WINAPI
MimeInternat_GetCodePageCharset(IMimeInternational
*iface
, CODEPAGEID cpiCodePage
,
162 CHARSETTYPE ctCsetType
,
163 LPHCHARSET phCharset
)
166 MIMECPINFO mlang_cp_info
;
168 TRACE("(%p)->(%d, %d, %p)\n", iface
, cpiCodePage
, ctCsetType
, phCharset
);
172 hr
= mlang_getcodepageinfo(cpiCodePage
, &mlang_cp_info
);
175 const WCHAR
*charset_nameW
= NULL
;
182 charset_nameW
= mlang_cp_info
.wszBodyCharset
;
185 charset_nameW
= mlang_cp_info
.wszHeaderCharset
;
188 charset_nameW
= mlang_cp_info
.wszWebCharset
;
191 return MIME_E_INVALID_CHARSET_TYPE
;
193 len
= WideCharToMultiByte(CP_ACP
, 0, charset_nameW
, -1, NULL
, 0, NULL
, NULL
);
194 charset_name
= HeapAlloc(GetProcessHeap(), 0, len
);
195 WideCharToMultiByte(CP_ACP
, 0, charset_nameW
, -1, charset_name
, len
, NULL
, NULL
);
196 hr
= IMimeInternational_FindCharset(iface
, charset_name
, phCharset
);
197 HeapFree(GetProcessHeap(), 0, charset_name
);
202 static HRESULT
mlang_getcsetinfo(const char *charset
, MIMECSETINFO
*mlang_info
)
204 DWORD len
= MultiByteToWideChar(CP_ACP
, 0, charset
, -1, NULL
, 0);
205 BSTR bstr
= SysAllocStringLen(NULL
, len
- 1);
209 MultiByteToWideChar(CP_ACP
, 0, charset
, -1, bstr
, len
);
215 hr
= IMultiLanguage_GetCharsetInfo(ml
, bstr
, mlang_info
);
216 IMultiLanguage_Release(ml
);
219 if(FAILED(hr
)) hr
= MIME_E_NOT_FOUND
;
223 static HCHARSET
add_charset(struct list
*list
, MIMECSETINFO
*mlang_info
, HCHARSET handle
)
225 charset_entry
*charset
= HeapAlloc(GetProcessHeap(), 0, sizeof(*charset
));
227 WideCharToMultiByte(CP_ACP
, 0, mlang_info
->wszCharset
, -1,
228 charset
->cs_info
.szName
, sizeof(charset
->cs_info
.szName
), NULL
, NULL
);
229 charset
->cs_info
.cpiWindows
= mlang_info
->uiCodePage
;
230 charset
->cs_info
.cpiInternet
= mlang_info
->uiInternetEncoding
;
231 charset
->cs_info
.hCharset
= handle
;
232 charset
->cs_info
.dwReserved1
= 0;
233 list_add_head(list
, &charset
->entry
);
235 return charset
->cs_info
.hCharset
;
238 static HRESULT WINAPI
MimeInternat_FindCharset(IMimeInternational
*iface
, LPCSTR pszCharset
,
239 LPHCHARSET phCharset
)
241 internat_impl
*This
= impl_from_IMimeInternational( iface
);
242 HRESULT hr
= MIME_E_NOT_FOUND
;
243 charset_entry
*charset
;
245 TRACE("(%p)->(%s, %p)\n", iface
, debugstr_a(pszCharset
), phCharset
);
249 EnterCriticalSection(&This
->cs
);
251 LIST_FOR_EACH_ENTRY(charset
, &This
->charsets
, charset_entry
, entry
)
253 if(!lstrcmpiA(charset
->cs_info
.szName
, pszCharset
))
255 *phCharset
= charset
->cs_info
.hCharset
;
261 if(hr
== MIME_E_NOT_FOUND
)
263 MIMECSETINFO mlang_info
;
265 LeaveCriticalSection(&This
->cs
);
266 hr
= mlang_getcsetinfo(pszCharset
, &mlang_info
);
267 EnterCriticalSection(&This
->cs
);
270 *phCharset
= add_charset(&This
->charsets
, &mlang_info
,
271 UlongToHandle(InterlockedIncrement(&This
->next_charset_handle
)));
274 LeaveCriticalSection(&This
->cs
);
278 static HRESULT WINAPI
MimeInternat_GetCharsetInfo(IMimeInternational
*iface
, HCHARSET hCharset
,
279 LPINETCSETINFO pCsetInfo
)
281 internat_impl
*This
= impl_from_IMimeInternational( iface
);
282 HRESULT hr
= MIME_E_INVALID_HANDLE
;
283 charset_entry
*charset
;
285 TRACE("(%p)->(%p, %p)\n", iface
, hCharset
, pCsetInfo
);
287 EnterCriticalSection(&This
->cs
);
289 LIST_FOR_EACH_ENTRY(charset
, &This
->charsets
, charset_entry
, entry
)
291 if(charset
->cs_info
.hCharset
== hCharset
)
293 *pCsetInfo
= charset
->cs_info
;
299 LeaveCriticalSection(&This
->cs
);
304 static HRESULT WINAPI
MimeInternat_GetCodePageInfo(IMimeInternational
*iface
, CODEPAGEID cpiCodePage
,
305 LPCODEPAGEINFO pCodePageInfo
)
311 static HRESULT WINAPI
MimeInternat_CanConvertCodePages(IMimeInternational
*iface
, CODEPAGEID cpiSource
,
317 TRACE("(%p)->(%d, %d)\n", iface
, cpiSource
, cpiDest
);
319 /* Could call mlang.IsConvertINetStringAvailable() to avoid the COM overhead if need be. */
324 hr
= IMultiLanguage_IsConvertible(ml
, cpiSource
, cpiDest
);
325 IMultiLanguage_Release(ml
);
331 static HRESULT WINAPI
MimeInternat_DecodeHeader(IMimeInternational
*iface
, HCHARSET hCharset
,
333 LPPROPVARIANT pDecoded
,
334 LPRFC1522INFO pRfc1522Info
)
340 static HRESULT WINAPI
MimeInternat_EncodeHeader(IMimeInternational
*iface
, HCHARSET hCharset
,
343 LPRFC1522INFO pRfc1522Info
)
349 static HRESULT WINAPI
MimeInternat_ConvertBuffer(IMimeInternational
*iface
, CODEPAGEID cpiSource
,
350 CODEPAGEID cpiDest
, LPBLOB pIn
, LPBLOB pOut
,
356 TRACE("(%p)->(%d, %d, %p, %p, %p)\n", iface
, cpiSource
, cpiDest
, pIn
, pOut
, pcbRead
);
361 /* Could call mlang.ConvertINetString() to avoid the COM overhead if need be. */
367 UINT in_size
= pIn
->cbSize
, out_size
;
369 hr
= IMultiLanguage_ConvertString(ml
, &mode
, cpiSource
, cpiDest
, pIn
->pBlobData
, &in_size
,
371 if(hr
== S_OK
) /* S_FALSE means the conversion could not be performed */
373 pOut
->pBlobData
= CoTaskMemAlloc(out_size
);
379 in_size
= pIn
->cbSize
;
380 hr
= IMultiLanguage_ConvertString(ml
, &mode
, cpiSource
, cpiDest
, pIn
->pBlobData
, &in_size
,
381 pOut
->pBlobData
, &out_size
);
386 pOut
->cbSize
= out_size
;
389 CoTaskMemFree(pOut
->pBlobData
);
392 IMultiLanguage_Release(ml
);
398 static HRESULT WINAPI
MimeInternat_ConvertString(IMimeInternational
*iface
, CODEPAGEID cpiSource
,
399 CODEPAGEID cpiDest
, LPPROPVARIANT pIn
,
406 TRACE("(%p)->(%d, %d, %p %p)\n", iface
, cpiSource
, cpiDest
, pIn
, pOut
);
411 if(cpiSource
== CP_UNICODE
) cpiSource
= GetACP();
412 src_len
= strlen(pIn
->u
.pszVal
);
415 cpiSource
= CP_UNICODE
;
416 src_len
= strlenW(pIn
->u
.pwszVal
) * sizeof(WCHAR
);
426 UINT in_size
= src_len
, out_size
;
428 hr
= IMultiLanguage_ConvertString(ml
, &mode
, cpiSource
, cpiDest
, (BYTE
*)pIn
->u
.pszVal
, &in_size
,
430 if(hr
== S_OK
) /* S_FALSE means the conversion could not be performed */
432 out_size
+= (cpiDest
== CP_UNICODE
) ? sizeof(WCHAR
) : sizeof(char);
434 pOut
->u
.pszVal
= CoTaskMemAlloc(out_size
);
441 hr
= IMultiLanguage_ConvertString(ml
, &mode
, cpiSource
, cpiDest
, (BYTE
*)pIn
->u
.pszVal
, &in_size
,
442 (BYTE
*)pOut
->u
.pszVal
, &out_size
);
446 if(cpiDest
== CP_UNICODE
)
448 pOut
->u
.pwszVal
[out_size
/ sizeof(WCHAR
)] = 0;
449 pOut
->vt
= VT_LPWSTR
;
453 pOut
->u
.pszVal
[out_size
] = '\0';
458 CoTaskMemFree(pOut
->u
.pszVal
);
461 IMultiLanguage_Release(ml
);
466 static HRESULT WINAPI
MimeInternat_MLANG_ConvertInetReset(IMimeInternational
*iface
)
472 static HRESULT WINAPI
MimeInternat_MLANG_ConvertInetString(IMimeInternational
*iface
, CODEPAGEID cpiSource
,
483 static HRESULT WINAPI
MimeInternat_Rfc1522Decode(IMimeInternational
*iface
, LPCSTR pszValue
,
492 static HRESULT WINAPI
MimeInternat_Rfc1522Encode(IMimeInternational
*iface
, LPCSTR pszValue
,
500 static IMimeInternationalVtbl mime_internat_vtbl
=
502 MimeInternat_QueryInterface
,
504 MimeInternat_Release
,
505 MimeInternat_SetDefaultCharset
,
506 MimeInternat_GetDefaultCharset
,
507 MimeInternat_GetCodePageCharset
,
508 MimeInternat_FindCharset
,
509 MimeInternat_GetCharsetInfo
,
510 MimeInternat_GetCodePageInfo
,
511 MimeInternat_CanConvertCodePages
,
512 MimeInternat_DecodeHeader
,
513 MimeInternat_EncodeHeader
,
514 MimeInternat_ConvertBuffer
,
515 MimeInternat_ConvertString
,
516 MimeInternat_MLANG_ConvertInetReset
,
517 MimeInternat_MLANG_ConvertInetString
,
518 MimeInternat_Rfc1522Decode
,
519 MimeInternat_Rfc1522Encode
522 static internat_impl
*global_internat
;
524 HRESULT
MimeInternational_Construct(IMimeInternational
**internat
)
526 global_internat
= HeapAlloc(GetProcessHeap(), 0, sizeof(*global_internat
));
527 global_internat
->IMimeInternational_iface
.lpVtbl
= &mime_internat_vtbl
;
528 global_internat
->refs
= 0;
529 InitializeCriticalSection(&global_internat
->cs
);
531 list_init(&global_internat
->charsets
);
532 global_internat
->next_charset_handle
= 0;
533 global_internat
->default_charset
= NULL
;
535 *internat
= &global_internat
->IMimeInternational_iface
;
537 IMimeInternational_AddRef(*internat
);
541 HRESULT WINAPI
MimeOleGetInternat(IMimeInternational
**internat
)
543 TRACE("(%p)\n", internat
);
545 *internat
= &global_internat
->IMimeInternational_iface
;
546 IMimeInternational_AddRef(*internat
);
550 HRESULT WINAPI
MimeOleFindCharset(LPCSTR name
, LPHCHARSET charset
)
552 IMimeInternational
*internat
;
555 TRACE("(%s, %p)\n", debugstr_a(name
), charset
);
557 hr
= MimeOleGetInternat(&internat
);
560 hr
= IMimeInternational_FindCharset(internat
, name
, charset
);
561 IMimeInternational_Release(internat
);
566 HRESULT WINAPI
MimeOleGetCharsetInfo(HCHARSET hCharset
, LPINETCSETINFO pCsetInfo
)
568 IMimeInternational
*internat
;
571 TRACE("(%p, %p)\n", hCharset
, pCsetInfo
);
573 hr
= MimeOleGetInternat(&internat
);
576 hr
= IMimeInternational_GetCharsetInfo(internat
, hCharset
, pCsetInfo
);
577 IMimeInternational_Release(internat
);
582 HRESULT WINAPI
MimeOleGetDefaultCharset(LPHCHARSET charset
)
584 IMimeInternational
*internat
;
587 TRACE("(%p)\n", charset
);
589 hr
= MimeOleGetInternat(&internat
);
592 hr
= IMimeInternational_GetDefaultCharset(internat
, charset
);
593 IMimeInternational_Release(internat
);