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/debug.h"
39 #include "inetcomm_private.h"
41 WINE_DEFAULT_DEBUG_CHANNEL(inetcomm);
51 const IMimeInternationalVtbl *lpVtbl;
56 LONG next_charset_handle;
57 HCHARSET default_charset;
60 static inline internat *impl_from_IMimeInternational( IMimeInternational *iface )
62 return (internat *)((char*)iface - FIELD_OFFSET(internat, lpVtbl));
65 static inline HRESULT get_mlang(IMultiLanguage **ml)
67 return CoCreateInstance(&CLSID_CMultiLanguage, NULL, CLSCTX_INPROC_SERVER | CLSCTX_INPROC_HANDLER,
68 &IID_IMultiLanguage, (void **)ml);
71 static HRESULT WINAPI MimeInternat_QueryInterface( IMimeInternational *iface, REFIID riid, LPVOID *ppobj )
73 if (IsEqualGUID(riid, &IID_IUnknown) ||
74 IsEqualGUID(riid, &IID_IMimeInternational))
76 IMimeInternational_AddRef( iface );
81 FIXME("interface %s not implemented\n", debugstr_guid(riid));
85 static ULONG WINAPI MimeInternat_AddRef( IMimeInternational *iface )
87 internat *This = impl_from_IMimeInternational( iface );
88 return InterlockedIncrement(&This->refs);
91 static ULONG WINAPI MimeInternat_Release( IMimeInternational *iface )
93 internat *This = impl_from_IMimeInternational( iface );
96 refs = InterlockedDecrement(&This->refs);
99 charset_entry *charset, *cursor2;
101 LIST_FOR_EACH_ENTRY_SAFE(charset, cursor2, &This->charsets, charset_entry, entry)
103 list_remove(&charset->entry);
104 HeapFree(GetProcessHeap(), 0, charset);
106 HeapFree(GetProcessHeap(), 0, This);
112 static HRESULT WINAPI MimeInternat_SetDefaultCharset(IMimeInternational *iface, HCHARSET hCharset)
118 static HRESULT WINAPI MimeInternat_GetDefaultCharset(IMimeInternational *iface, LPHCHARSET phCharset)
120 internat *This = impl_from_IMimeInternational( iface );
123 TRACE("(%p)->(%p)\n", iface, phCharset);
125 if(This->default_charset == NULL)
128 hr = IMimeInternational_GetCodePageCharset(iface, GetACP(), CHARSET_BODY, &hcs);
130 InterlockedCompareExchangePointer(&This->default_charset, hcs, NULL);
132 *phCharset = This->default_charset;
137 static HRESULT mlang_getcodepageinfo(UINT cp, MIMECPINFO *mlang_cp_info)
146 hr = IMultiLanguage_GetCodePageInfo(ml, cp, mlang_cp_info);
147 IMultiLanguage_Release(ml);
152 static HRESULT WINAPI MimeInternat_GetCodePageCharset(IMimeInternational *iface, CODEPAGEID cpiCodePage,
153 CHARSETTYPE ctCsetType,
154 LPHCHARSET phCharset)
157 MIMECPINFO mlang_cp_info;
159 TRACE("(%p)->(%d, %d, %p)\n", iface, cpiCodePage, ctCsetType, phCharset);
163 if(ctCsetType < CHARSET_BODY || ctCsetType > CHARSET_WEB)
164 return MIME_E_INVALID_CHARSET_TYPE;
166 hr = mlang_getcodepageinfo(cpiCodePage, &mlang_cp_info);
169 const WCHAR *charset_nameW = NULL;
176 charset_nameW = mlang_cp_info.wszBodyCharset;
179 charset_nameW = mlang_cp_info.wszHeaderCharset;
182 charset_nameW = mlang_cp_info.wszWebCharset;
185 len = WideCharToMultiByte(CP_ACP, 0, charset_nameW, -1, NULL, 0, NULL, NULL);
186 charset_name = HeapAlloc(GetProcessHeap(), 0, len);
187 WideCharToMultiByte(CP_ACP, 0, charset_nameW, -1, charset_name, len, NULL, NULL);
188 hr = IMimeInternational_FindCharset(iface, charset_name, phCharset);
189 HeapFree(GetProcessHeap(), 0, charset_name);
194 static HRESULT mlang_getcsetinfo(const char *charset, MIMECSETINFO *mlang_info)
196 DWORD len = MultiByteToWideChar(CP_ACP, 0, charset, -1, NULL, 0);
197 BSTR bstr = SysAllocStringLen(NULL, len - 1);
201 MultiByteToWideChar(CP_ACP, 0, charset, -1, bstr, len);
207 hr = IMultiLanguage_GetCharsetInfo(ml, bstr, mlang_info);
208 IMultiLanguage_Release(ml);
211 if(FAILED(hr)) hr = MIME_E_NOT_FOUND;
215 static HCHARSET add_charset(struct list *list, MIMECSETINFO *mlang_info, HCHARSET handle)
217 charset_entry *charset = HeapAlloc(GetProcessHeap(), 0, sizeof(*charset));
219 WideCharToMultiByte(CP_ACP, 0, mlang_info->wszCharset, -1,
220 charset->cs_info.szName, sizeof(charset->cs_info.szName), NULL, NULL);
221 charset->cs_info.cpiWindows = mlang_info->uiCodePage;
222 charset->cs_info.cpiInternet = mlang_info->uiInternetEncoding;
223 charset->cs_info.hCharset = handle;
224 charset->cs_info.dwReserved1 = 0;
225 list_add_head(list, &charset->entry);
227 return charset->cs_info.hCharset;
230 static HRESULT WINAPI MimeInternat_FindCharset(IMimeInternational *iface, LPCSTR pszCharset,
231 LPHCHARSET phCharset)
233 internat *This = impl_from_IMimeInternational( iface );
234 HRESULT hr = MIME_E_NOT_FOUND;
235 charset_entry *charset;
237 TRACE("(%p)->(%s, %p)\n", iface, debugstr_a(pszCharset), phCharset);
241 EnterCriticalSection(&This->cs);
243 LIST_FOR_EACH_ENTRY(charset, &This->charsets, charset_entry, entry)
245 if(!strcmp(charset->cs_info.szName, pszCharset))
247 *phCharset = charset->cs_info.hCharset;
253 if(hr == MIME_E_NOT_FOUND)
255 MIMECSETINFO mlang_info;
257 LeaveCriticalSection(&This->cs);
258 hr = mlang_getcsetinfo(pszCharset, &mlang_info);
259 EnterCriticalSection(&This->cs);
262 *phCharset = add_charset(&This->charsets, &mlang_info,
263 (HCHARSET)InterlockedIncrement(&This->next_charset_handle));
266 LeaveCriticalSection(&This->cs);
270 static HRESULT WINAPI MimeInternat_GetCharsetInfo(IMimeInternational *iface, HCHARSET hCharset,
271 LPINETCSETINFO pCsetInfo)
273 internat *This = impl_from_IMimeInternational( iface );
274 HRESULT hr = MIME_E_INVALID_HANDLE;
275 charset_entry *charset;
277 TRACE("(%p)->(%p, %p)\n", iface, hCharset, pCsetInfo);
279 EnterCriticalSection(&This->cs);
281 LIST_FOR_EACH_ENTRY(charset, &This->charsets, charset_entry, entry)
283 if(charset->cs_info.hCharset == hCharset)
285 *pCsetInfo = charset->cs_info;
291 LeaveCriticalSection(&This->cs);
296 static HRESULT WINAPI MimeInternat_GetCodePageInfo(IMimeInternational *iface, CODEPAGEID cpiCodePage,
297 LPCODEPAGEINFO pCodePageInfo)
303 static HRESULT WINAPI MimeInternat_DecodeHeader(IMimeInternational *iface, HCHARSET hCharset,
305 LPPROPVARIANT pDecoded,
306 LPRFC1522INFO pRfc1522Info)
312 static HRESULT WINAPI MimeInternat_EncodeHeader(IMimeInternational *iface, HCHARSET hCharset,
315 LPRFC1522INFO pRfc1522Info)
321 static HRESULT WINAPI MimeInternat_ConvertBuffer(IMimeInternational *iface, CODEPAGEID cpiSource,
331 static HRESULT WINAPI MimeInternat_ConvertString(IMimeInternational *iface, CODEPAGEID cpiSource,
340 static HRESULT WINAPI MimeInternat_MLANG_ConvertInetReset(IMimeInternational *iface)
346 static HRESULT WINAPI MimeInternat_MLANG_ConvertInetString(IMimeInternational *iface, CODEPAGEID cpiSource,
357 static HRESULT WINAPI MimeInternat_Rfc1522Decode(IMimeInternational *iface, LPCSTR pszValue,
366 static HRESULT WINAPI MimeInternat_Rfc1522Encode(IMimeInternational *iface, LPCSTR pszValue,
374 static IMimeInternationalVtbl mime_internat_vtbl =
376 MimeInternat_QueryInterface,
378 MimeInternat_Release,
379 MimeInternat_SetDefaultCharset,
380 MimeInternat_GetDefaultCharset,
381 MimeInternat_GetCodePageCharset,
382 MimeInternat_FindCharset,
383 MimeInternat_GetCharsetInfo,
384 MimeInternat_GetCodePageInfo,
385 MimeInternat_DecodeHeader,
386 MimeInternat_EncodeHeader,
387 MimeInternat_ConvertBuffer,
388 MimeInternat_ConvertString,
389 MimeInternat_MLANG_ConvertInetReset,
390 MimeInternat_MLANG_ConvertInetString,
391 MimeInternat_Rfc1522Decode,
392 MimeInternat_Rfc1522Encode
395 static internat *global_internat;
397 HRESULT MimeInternational_Construct(IMimeInternational **internat)
399 global_internat = HeapAlloc(GetProcessHeap(), 0, sizeof(*global_internat));
400 global_internat->lpVtbl = &mime_internat_vtbl;
401 global_internat->refs = 0;
402 InitializeCriticalSection(&global_internat->cs);
404 list_init(&global_internat->charsets);
405 global_internat->next_charset_handle = 0;
406 global_internat->default_charset = NULL;
408 *internat = (IMimeInternational*)&global_internat->lpVtbl;
410 IMimeInternational_AddRef(*internat);
414 HRESULT WINAPI MimeOleGetInternat(IMimeInternational **internat)
416 TRACE("(%p)\n", internat);
418 *internat = (IMimeInternational *)&global_internat->lpVtbl;
419 IMimeInternational_AddRef(*internat);