GetFileName31W should call unicode version of FD31_AllocPrivate.
[wine] / dlls / commdlg / fontdlg16.c
1 /*
2  * COMMDLG - Font Dialog
3  *
4  * Copyright 1994 Martin Ayotte
5  * Copyright 1996 Albrecht Kleine
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with this library; if not, write to the Free Software
19  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
20  */
21
22 #include <ctype.h>
23 #include <stdlib.h>
24 #include <stdarg.h>
25 #include <stdio.h>
26 #include <string.h>
27 #include "windef.h"
28 #include "winbase.h"
29 #include "wingdi.h"
30 #include "winuser.h"
31 #include "winnls.h"
32 #include "wine/winbase16.h"
33 #include "wine/winuser16.h"
34 #include "commdlg.h"
35 #include "dlgs.h"
36 #include "wine/debug.h"
37 #include "cderr.h"
38
39 WINE_DEFAULT_DEBUG_CHANNEL(commdlg);
40
41 #include "cdlg.h"
42 #include "cdlg16.h"
43
44 static void FONT_LogFont16To32W( const LPLOGFONT16 font16, LPLOGFONTW font32 )
45 {
46     font32->lfHeight = font16->lfHeight;
47     font32->lfWidth = font16->lfWidth;
48     font32->lfEscapement = font16->lfEscapement;
49     font32->lfOrientation = font16->lfOrientation;
50     font32->lfWeight = font16->lfWeight;
51     font32->lfItalic = font16->lfItalic;
52     font32->lfUnderline = font16->lfUnderline;
53     font32->lfStrikeOut = font16->lfStrikeOut;
54     font32->lfCharSet = font16->lfCharSet;
55     font32->lfOutPrecision = font16->lfOutPrecision;
56     font32->lfClipPrecision = font16->lfClipPrecision;
57     font32->lfQuality = font16->lfQuality;
58     font32->lfPitchAndFamily = font16->lfPitchAndFamily;
59     MultiByteToWideChar(CP_ACP, 0, font16->lfFaceName,
60                         LF_FACESIZE, font32->lfFaceName, LF_FACESIZE);
61 };
62
63 static void FONT_Metrics16To32W( const TEXTMETRIC16 *pm16,
64                                  NEWTEXTMETRICEXW *pnm32w)
65 {
66     ZeroMemory( pnm32w, sizeof(NEWTEXTMETRICEXW));
67     /* NOTE: only the fields used by AddFontStyle() are filled in */
68     pnm32w->ntmTm.tmHeight = pm16->tmHeight;
69     pnm32w->ntmTm.tmExternalLeading = pm16->tmExternalLeading;
70 };
71
72 static void CFn_CHOOSEFONT16to32W(LPCHOOSEFONT16 chf16, LPCHOOSEFONTW chf32w)
73 {
74   int len;
75   if(chf16->lpTemplateName)
76   {
77     len = MultiByteToWideChar(CP_ACP, 0, (LPBYTE)chf16->lpTemplateName, -1, NULL, 0);
78     chf32w->lpTemplateName = HeapAlloc(GetProcessHeap(), 0,len*sizeof(WCHAR));
79     MultiByteToWideChar(CP_ACP, 0, (LPSTR)MapSL(chf16->lpTemplateName),
80                         -1, (LPWSTR)chf32w->lpTemplateName, len);
81   }
82   if(chf16->lpszStyle)
83   {
84     len = MultiByteToWideChar(CP_ACP, 0, (LPBYTE)chf16->lpszStyle, -1, NULL, 0);
85     chf32w->lpszStyle = HeapAlloc(GetProcessHeap(), 0, len*sizeof(WCHAR));
86     MultiByteToWideChar(CP_ACP, 0, (LPSTR)MapSL(chf16->lpTemplateName),
87                         -1, chf32w->lpszStyle, len);
88   }
89   chf32w->lStructSize=sizeof(CHOOSEFONTW);
90   chf32w->hwndOwner=HWND_32(chf16->hwndOwner);
91   chf32w->hDC=HDC_32(chf16->hDC);
92   chf32w->iPointSize=chf16->iPointSize;
93   chf32w->Flags=chf16->Flags;
94   chf32w->rgbColors=chf16->rgbColors;
95   chf32w->lCustData=chf16->lCustData;
96   chf32w->lpfnHook=NULL;
97   chf32w->hInstance=HINSTANCE_32(chf16->hInstance);
98   chf32w->lpszStyle=MapSL(chf16->lpszStyle);
99   chf32w->nFontType=chf16->nFontType;
100   chf32w->nSizeMax=chf16->nSizeMax;
101   chf32w->nSizeMin=chf16->nSizeMin;
102   FONT_LogFont16To32W(MapSL(chf16->lpLogFont), chf32w->lpLogFont);
103 };
104
105 /***********************************************************************
106  *                          CFn_HookCallChk                 [internal]
107  */
108 static BOOL CFn_HookCallChk(LPCHOOSEFONT16 lpcf)
109 {
110  if (lpcf)
111   if(lpcf->Flags & CF_ENABLEHOOK)
112    if (lpcf->lpfnHook)
113     return TRUE;
114  return FALSE;
115 }
116
117 /***********************************************************************
118  *                FontFamilyEnumProc                     (COMMDLG.19)
119  */
120 INT16 WINAPI FontFamilyEnumProc16( SEGPTR logfont, SEGPTR metrics,
121                                    UINT16 nFontType, LPARAM lParam )
122 {
123   HWND hwnd=HWND_32(LOWORD(lParam));
124   HWND hDlg=GetParent(hwnd);
125   LPCHOOSEFONT16 lpcf=(LPCHOOSEFONT16)GetWindowLongA(hDlg, DWL_USER);
126   LOGFONT16 *lplf = MapSL( logfont );
127   TEXTMETRIC16 *lpmtrx16 = MapSL(metrics);
128   ENUMLOGFONTEXW elf32w;
129   NEWTEXTMETRICEXW nmtrx32w;
130   FONT_LogFont16To32W(lplf, &(elf32w.elfLogFont));
131   FONT_Metrics16To32W(lpmtrx16, &nmtrx32w);
132   return AddFontFamily(&elf32w, &nmtrx32w, nFontType,
133           (LPCHOOSEFONTW)lpcf->lpTemplateName, hwnd,NULL);
134 }
135
136 /***********************************************************************
137  *                 FontStyleEnumProc                     (COMMDLG.18)
138  */
139 INT16 WINAPI FontStyleEnumProc16( SEGPTR logfont, SEGPTR metrics,
140                                   UINT16 nFontType, LPARAM lParam )
141 {
142   HWND hcmb2=HWND_32(LOWORD(lParam));
143   HWND hcmb3=HWND_32(HIWORD(lParam));
144   HWND hDlg=GetParent(hcmb3);
145   LPCHOOSEFONT16 lpcf=(LPCHOOSEFONT16)GetWindowLongA(hDlg, DWL_USER);
146   LOGFONT16 *lplf = MapSL(logfont);
147   TEXTMETRIC16 *lpmtrx16 = MapSL(metrics);
148   ENUMLOGFONTEXW elf32w;
149   NEWTEXTMETRICEXW nmtrx32w;
150   FONT_LogFont16To32W(lplf, &(elf32w.elfLogFont));
151   FONT_Metrics16To32W(lpmtrx16, &nmtrx32w);
152   return AddFontStyle(&elf32w, &nmtrx32w, nFontType,
153           (LPCHOOSEFONTW)lpcf->lpTemplateName, hcmb2, hcmb3, hDlg, TRUE);
154 }
155
156 /***********************************************************************
157  *                        ChooseFont   (COMMDLG.15)
158  */
159 BOOL16 WINAPI ChooseFont16(LPCHOOSEFONT16 lpChFont)
160 {
161     HINSTANCE16 hInst;
162     HANDLE16 hDlgTmpl16 = 0, hResource16 = 0;
163     HGLOBAL16 hGlobal16 = 0;
164     BOOL16 bRet = FALSE;
165     LPCVOID template;
166     FARPROC16 ptr;
167     CHOOSEFONTW cf32w;
168     LOGFONTW lf32w;
169     LOGFONT16 *font16;
170     SEGPTR lpTemplateName;
171
172     cf32w.lpLogFont=&lf32w;
173     CFn_CHOOSEFONT16to32W(lpChFont, &cf32w);
174
175     TRACE("ChooseFont\n");
176     if (!lpChFont) return FALSE;
177
178     if (TRACE_ON(commdlg))
179         _dump_cf_flags(lpChFont->Flags);
180
181     if (lpChFont->Flags & CF_ENABLETEMPLATEHANDLE)
182     {
183         if (!(template = LockResource16( lpChFont->hInstance )))
184         {
185             COMDLG32_SetCommDlgExtendedError(CDERR_LOADRESFAILURE);
186             return FALSE;
187         }
188     }
189     else if (lpChFont->Flags & CF_ENABLETEMPLATE)
190     {
191         HANDLE16 hResInfo;
192         if (!(hResInfo = FindResource16( lpChFont->hInstance,
193                                          MapSL(lpChFont->lpTemplateName),
194                                          (LPSTR)RT_DIALOG)))
195         {
196             COMDLG32_SetCommDlgExtendedError(CDERR_FINDRESFAILURE);
197             return FALSE;
198         }
199         if (!(hDlgTmpl16 = LoadResource16( lpChFont->hInstance, hResInfo )) ||
200             !(template = LockResource16( hDlgTmpl16 )))
201         {
202             COMDLG32_SetCommDlgExtendedError(CDERR_LOADRESFAILURE);
203             return FALSE;
204         }
205     }
206     else
207     {
208         HRSRC hResInfo;
209         HGLOBAL hDlgTmpl32;
210         LPCVOID template32;
211         DWORD size;
212         if (!(hResInfo = FindResourceA(COMDLG32_hInstance, "CHOOSE_FONT", (LPSTR)RT_DIALOG)))
213         {
214             COMDLG32_SetCommDlgExtendedError(CDERR_FINDRESFAILURE);
215             return FALSE;
216         }
217         if (!(hDlgTmpl32 = LoadResource(COMDLG32_hInstance, hResInfo)) ||
218             !(template32 = LockResource(hDlgTmpl32)))
219         {
220             COMDLG32_SetCommDlgExtendedError(CDERR_LOADRESFAILURE);
221             return FALSE;
222         }
223         size = SizeofResource(COMDLG32_hInstance, hResInfo);
224         hGlobal16 = GlobalAlloc16(0, size);
225         if (!hGlobal16)
226         {
227             COMDLG32_SetCommDlgExtendedError(CDERR_MEMALLOCFAILURE);
228             ERR("alloc failure for %ld bytes\n", size);
229             return FALSE;
230         }
231         template = GlobalLock16(hGlobal16);
232         if (!template)
233         {
234             COMDLG32_SetCommDlgExtendedError(CDERR_MEMLOCKFAILURE);
235             ERR("global lock failure for %x handle\n", hGlobal16);
236             GlobalFree16(hGlobal16);
237             return FALSE;
238         }
239         ConvertDialog32To16((LPVOID)template32, size, (LPVOID)template);
240         hDlgTmpl16 = hGlobal16;
241
242     }
243
244     /* lpTemplateName is not used in the dialog */
245     lpTemplateName=lpChFont->lpTemplateName;
246     lpChFont->lpTemplateName=(SEGPTR)&cf32w;
247
248     ptr = GetProcAddress16(GetModuleHandle16("COMMDLG"), (LPCSTR) 16);
249     hInst = GetWindowLongPtrA(HWND_32(lpChFont->hwndOwner), GWLP_HINSTANCE);
250     bRet = DialogBoxIndirectParam16(hInst, hDlgTmpl16, lpChFont->hwndOwner,
251                      (DLGPROC16) ptr, (DWORD)lpChFont);
252     if (hResource16) FreeResource16(hDlgTmpl16);
253     if (hGlobal16)
254     {
255         GlobalUnlock16(hGlobal16);
256         GlobalFree16(hGlobal16);
257     }
258     lpChFont->lpTemplateName=lpTemplateName;
259
260     lpChFont->iPointSize = cf32w.iPointSize;
261     lpChFont->Flags = cf32w.Flags;
262     lpChFont->rgbColors = cf32w.rgbColors;
263     lpChFont->lCustData = cf32w.lCustData;
264     lpChFont->nFontType = cf32w.nFontType;
265
266     font16 = MapSL(lpChFont->lpLogFont);
267     font16->lfHeight = cf32w.lpLogFont->lfHeight;
268     font16->lfWidth = cf32w.lpLogFont->lfWidth;
269     font16->lfEscapement = cf32w.lpLogFont->lfEscapement;
270     font16->lfOrientation = cf32w.lpLogFont->lfOrientation;
271     font16->lfWeight = cf32w.lpLogFont->lfWeight;
272     font16->lfItalic = cf32w.lpLogFont->lfItalic;
273     font16->lfUnderline = cf32w.lpLogFont->lfUnderline;
274     font16->lfStrikeOut = cf32w.lpLogFont->lfStrikeOut;
275     font16->lfCharSet = cf32w.lpLogFont->lfCharSet;
276     font16->lfOutPrecision = cf32w.lpLogFont->lfOutPrecision;
277     font16->lfClipPrecision = cf32w.lpLogFont->lfClipPrecision;
278     font16->lfQuality = cf32w.lpLogFont->lfQuality;
279     font16->lfPitchAndFamily = cf32w.lpLogFont->lfPitchAndFamily;
280     WideCharToMultiByte(CP_ACP, 0, cf32w.lpLogFont->lfFaceName,
281                           LF_FACESIZE, font16->lfFaceName, LF_FACESIZE, 0, 0);
282
283     HeapFree(GetProcessHeap(), 0, (LPBYTE)cf32w.lpTemplateName);
284     HeapFree(GetProcessHeap(), 0, cf32w.lpszStyle);
285
286     return bRet;
287 }
288
289 /***********************************************************************
290  *           FormatCharDlgProc   (COMMDLG.16)
291              FIXME: 1. some strings are "hardcoded", but it's better load from sysres
292                     2. some CF_.. flags are not supported
293                     3. some TType extensions
294  */
295 BOOL16 CALLBACK FormatCharDlgProc16(HWND16 hDlg16, UINT16 message,
296                                    WPARAM16 wParam, LPARAM lParam)
297 {
298   HWND hDlg = HWND_32(hDlg16);
299   LPCHOOSEFONT16 lpcf;
300   BOOL16 res=0;
301   if (message!=WM_INITDIALOG)
302   {
303    lpcf=(LPCHOOSEFONT16)GetWindowLongA(hDlg, DWL_USER);
304    if (!lpcf && message != WM_MEASUREITEM)
305       return FALSE;
306    if (CFn_HookCallChk(lpcf))
307      res=CallWindowProc16((WNDPROC16)lpcf->lpfnHook,hDlg16,message,wParam,lParam);
308    if (res)
309     return res;
310   }
311   else
312   {
313     lpcf=(LPCHOOSEFONT16)lParam;
314     if (!CFn_WMInitDialog(hDlg, wParam, lParam, (LPCHOOSEFONTW)lpcf->lpTemplateName))
315     {
316       TRACE("CFn_WMInitDialog returned FALSE\n");
317       return FALSE;
318     }
319     if (CFn_HookCallChk(lpcf))
320       return CallWindowProc16((WNDPROC16)lpcf->lpfnHook,hDlg16,WM_INITDIALOG,wParam,lParam);
321   }
322   switch (message)
323     {
324     case WM_MEASUREITEM:
325         {
326             MEASUREITEMSTRUCT16* mis16 = MapSL(lParam);
327             MEASUREITEMSTRUCT mis;
328             mis.CtlType    = mis16->CtlType;
329             mis.CtlID      = mis16->CtlID;
330             mis.itemID     = mis16->itemID;
331             mis.itemWidth  = mis16->itemWidth;
332             mis.itemHeight = mis16->itemHeight;
333             mis.itemData   = mis16->itemData;
334             res = CFn_WMMeasureItem(hDlg, wParam, (LPARAM)&mis);
335             mis16->itemWidth  = (UINT16)mis.itemWidth;
336             mis16->itemHeight = (UINT16)mis.itemHeight;
337         }
338         break;
339     case WM_DRAWITEM:
340         {
341             DRAWITEMSTRUCT16* dis16 = MapSL(lParam);
342             DRAWITEMSTRUCT dis;
343             dis.CtlType       = dis16->CtlType;
344             dis.CtlID         = dis16->CtlID;
345             dis.itemID        = dis16->itemID;
346             dis.itemAction    = dis16->itemAction;
347             dis.itemState     = dis16->itemState;
348             dis.hwndItem      = HWND_32(dis16->hwndItem);
349             dis.hDC           = HDC_32(dis16->hDC);
350             dis.itemData      = dis16->itemData;
351             dis.rcItem.left   = dis16->rcItem.left;
352             dis.rcItem.top    = dis16->rcItem.top;
353             dis.rcItem.right  = dis16->rcItem.right;
354             dis.rcItem.bottom = dis16->rcItem.bottom;
355             res = CFn_WMDrawItem(hDlg, wParam, (LPARAM)&dis);
356         }
357         break;
358     case WM_COMMAND:
359         res=CFn_WMCommand(hDlg, MAKEWPARAM( wParam, HIWORD(lParam) ), LOWORD(lParam),
360                           (LPCHOOSEFONTW)lpcf->lpTemplateName);
361         break;
362     case WM_DESTROY:
363         return TRUE;
364     case WM_CHOOSEFONT_GETLOGFONT:
365         TRACE("WM_CHOOSEFONT_GETLOGFONT lParam=%08lX\n", lParam);
366         FIXME("current logfont back to caller\n");
367         break;
368     case WM_PAINT:
369         res= CFn_WMPaint(hDlg, wParam, lParam, (LPCHOOSEFONTW)lpcf->lpTemplateName);
370         break;
371     }
372   return res;
373 }