2 * Unit test suite for comdlg32 API functions: file dialogs
4 * Copyright 2007 Google (Lei Zhang)
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
23 #include <wine/test.h>
32 static int resizesupported
= TRUE
;
34 static void toolbarcheck( HWND hDlg
)
36 /* test toolbar properties */
43 for( ctrl
= GetWindow( hDlg
, GW_CHILD
);
44 ctrl
; ctrl
= GetWindow( ctrl
, GW_HWNDNEXT
)) {
45 GetClassName( ctrl
, classname
, 10);
47 if( !strcmp( classname
, "Toolbar")) break;
49 ok( ctrl
!= NULL
, "could not get the toolbar control\n");
50 ret
= SendMessage( ctrl
, TB_ADDSTRING
, 0, (LPARAM
)"winetestwinetest\0\0");
51 ok( ret
== 0, "addstring returned %d (expected 0)\n", ret
);
52 maxtextrows
= SendMessage( ctrl
, TB_GETTEXTROWS
, 0, 0);
53 ok( maxtextrows
== 0 || broken(maxtextrows
== 1), /* Win2k and below */
54 "Get(Max)TextRows returned %d (expected 0)\n", maxtextrows
);
58 static UINT_PTR CALLBACK
OFNHookProc( HWND hDlg
, UINT msg
, WPARAM wParam
, LPARAM lParam
)
64 nmh
= (LPNMHDR
) lParam
;
65 if( nmh
->code
== CDN_INITDONE
)
67 PostMessage( GetParent(hDlg
), WM_COMMAND
, IDCANCEL
, FALSE
);
68 } else if (nmh
->code
== CDN_FOLDERCHANGE
)
73 memset(buf
, 0x66, sizeof(buf
));
74 ret
= SendMessage( GetParent(hDlg
), CDM_GETFOLDERIDLIST
, 5, (LPARAM
)buf
);
75 ok(ret
> 0, "CMD_GETFOLDERIDLIST not implemented\n");
77 ok(buf
[0] == 0x66 && buf
[1] == 0x66, "CMD_GETFOLDERIDLIST: The buffer was touched on failure\n");
78 toolbarcheck( GetParent(hDlg
));
86 static void test_DialogCancel(void)
90 char szFileName
[MAX_PATH
] = "";
91 char szInitialDir
[MAX_PATH
];
93 GetWindowsDirectory(szInitialDir
, MAX_PATH
);
95 ZeroMemory(&ofn
, sizeof(ofn
));
97 ofn
.lStructSize
= sizeof(ofn
);
99 ofn
.lpstrFilter
= "Text Files (*.txt)\0*.txt\0All Files (*.*)\0*.*\0";
100 ofn
.lpstrFile
= szFileName
;
101 ofn
.nMaxFile
= MAX_PATH
;
102 ofn
.Flags
= OFN_EXPLORER
| OFN_FILEMUSTEXIST
| OFN_HIDEREADONLY
| OFN_ENABLEHOOK
;
103 ofn
.lpstrDefExt
= "txt";
104 ofn
.lpfnHook
= OFNHookProc
;
105 ofn
.lpstrInitialDir
= szInitialDir
;
108 ok(CDERR_INITIALIZATION
== CommDlgExtendedError(),
109 "expected CDERR_INITIALIZATION, got %d\n", CommDlgExtendedError());
111 result
= GetOpenFileNameA(&ofn
);
112 ok(0 == result
, "expected 0, got %d\n", result
);
113 ok(0 == CommDlgExtendedError(), "expected 0, got %d\n",
114 CommDlgExtendedError());
117 ok(CDERR_INITIALIZATION
== CommDlgExtendedError(),
118 "expected CDERR_INITIALIZATION, got %d\n", CommDlgExtendedError());
120 result
= GetSaveFileNameA(&ofn
);
121 ok(0 == result
, "expected 0, got %d\n", result
);
122 ok(0 == CommDlgExtendedError(), "expected 0, got %d\n",
123 CommDlgExtendedError());
126 ok(CDERR_INITIALIZATION
== CommDlgExtendedError(),
127 "expected CDERR_INITIALIZATION, got %d\n", CommDlgExtendedError());
129 /* Before passing the ofn to Unicode functions, remove the ANSI strings */
130 ofn
.lpstrFilter
= NULL
;
131 ofn
.lpstrInitialDir
= NULL
;
132 ofn
.lpstrDefExt
= NULL
;
135 ok(CDERR_INITIALIZATION
== CommDlgExtendedError(),
136 "expected CDERR_INITIALIZATION, got %d\n", CommDlgExtendedError());
138 SetLastError(0xdeadbeef);
139 result
= GetOpenFileNameW((LPOPENFILENAMEW
) &ofn
);
140 if (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED
)
141 win_skip("GetOpenFileNameW is not implemented\n");
144 ok(0 == result
, "expected 0, got %d\n", result
);
145 ok(0 == CommDlgExtendedError() ||
146 broken(CDERR_INITIALIZATION
== CommDlgExtendedError()), /* win9x */
147 "expected 0, got %d\n", CommDlgExtendedError());
150 SetLastError(0xdeadbeef);
151 result
= GetSaveFileNameW((LPOPENFILENAMEW
) &ofn
);
152 if (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED
)
153 win_skip("GetSaveFileNameW is not implemented\n");
156 ok(0 == result
, "expected 0, got %d\n", result
);
157 ok(0 == CommDlgExtendedError() ||
158 broken(CDERR_INITIALIZATION
== CommDlgExtendedError()), /* win9x */
159 "expected 0, got %d\n", CommDlgExtendedError());
163 static UINT_PTR CALLBACK
create_view_window2_hook(HWND dlg
, UINT msg
, WPARAM wParam
, LPARAM lParam
)
165 if (msg
== WM_NOTIFY
)
167 if (((LPNMHDR
)lParam
)->code
== CDN_FOLDERCHANGE
)
169 IShellBrowser
*shell_browser
= (IShellBrowser
*)SendMessage(GetParent(dlg
), WM_USER
+ 7 /* WM_GETISHELLBROWSER */, 0, 0);
170 IShellView
*shell_view
= NULL
;
171 IShellView2
*shell_view2
= NULL
;
172 SV2CVW2_PARAMS view_params
;
173 FOLDERSETTINGS folder_settings
;
175 RECT rect
= {0, 0, 0, 0};
177 hr
= IShellBrowser_QueryActiveShellView(shell_browser
, &shell_view
);
178 ok(SUCCEEDED(hr
), "QueryActiveShellView returned %#x\n", hr
);
179 if (FAILED(hr
)) goto cleanup
;
181 hr
= IShellView_QueryInterface(shell_view
, &IID_IShellView2
, (void **)&shell_view2
);
182 if (hr
== E_NOINTERFACE
)
184 win_skip("IShellView2 not supported\n");
187 ok(SUCCEEDED(hr
), "QueryInterface returned %#x\n", hr
);
188 if (FAILED(hr
)) goto cleanup
;
190 hr
= IShellView2_DestroyViewWindow(shell_view2
);
191 ok(SUCCEEDED(hr
), "DestroyViewWindow returned %#x\n", hr
);
193 folder_settings
.ViewMode
= FVM_LIST
;
194 folder_settings
.fFlags
= 0;
196 view_params
.cbSize
= sizeof(view_params
);
197 view_params
.psvPrev
= NULL
;
198 view_params
.pfs
= &folder_settings
;
199 view_params
.psbOwner
= shell_browser
;
200 view_params
.prcView
= &rect
;
201 view_params
.pvid
= NULL
;
202 view_params
.hwndView
= NULL
;
204 hr
= IShellView2_CreateViewWindow2(shell_view2
, &view_params
);
207 win_skip("CreateViewWindow2 is broken on Vista/W2K8\n");
210 ok(SUCCEEDED(hr
), "CreateViewWindow2 returned %#x\n", hr
);
211 if (FAILED(hr
)) goto cleanup
;
213 hr
= IShellView2_GetCurrentInfo(shell_view2
, &folder_settings
);
214 ok(SUCCEEDED(hr
), "GetCurrentInfo returned %#x\n", hr
);
215 ok(folder_settings
.ViewMode
== FVM_LIST
,
216 "view mode is %d, expected FVM_LIST\n",
217 folder_settings
.ViewMode
);
219 hr
= IShellView2_DestroyViewWindow(shell_view2
);
220 ok(SUCCEEDED(hr
), "DestroyViewWindow returned %#x\n", hr
);
222 /* XP and W2K3 need this. On Win9x and W2K the call to DestroyWindow() fails and has
223 * no side effects. NT4 doesn't get here. (FIXME: Vista doesn't get here yet).
225 DestroyWindow(view_params
.hwndView
);
227 view_params
.pvid
= &VID_Details
;
228 hr
= IShellView2_CreateViewWindow2(shell_view2
, &view_params
);
229 ok(SUCCEEDED(hr
), "CreateViewWindow2 returned %#x\n", hr
);
230 if (FAILED(hr
)) goto cleanup
;
232 hr
= IShellView2_GetCurrentInfo(shell_view2
, &folder_settings
);
233 ok(SUCCEEDED(hr
), "GetCurrentInfo returned %#x\n", hr
);
234 ok(folder_settings
.ViewMode
== FVM_DETAILS
||
235 broken(folder_settings
.ViewMode
== FVM_LIST
), /* Win9x */
236 "view mode is %d, expected FVM_DETAILS\n",
237 folder_settings
.ViewMode
);
240 if (shell_view2
) IShellView2_Release(shell_view2
);
241 if (shell_view
) IShellView_Release(shell_view
);
242 PostMessage(GetParent(dlg
), WM_COMMAND
, IDCANCEL
, 0);
248 static LONG_PTR WINAPI
template_hook(HWND dlg
, UINT msg
, WPARAM wParam
, LPARAM lParam
)
250 if (msg
== WM_INITDIALOG
)
255 ok(p
!=NULL
, "Failed to get parent of template\n");
256 cb
= GetDlgItem(p
,0x470);
257 ok(cb
!=NULL
, "Failed to get filter combobox\n");
258 sel
= SendMessage(cb
, CB_GETCURSEL
, 0, 0);
259 ok (sel
!= -1, "Failed to get selection from filter listbox\n");
261 if (msg
== WM_NOTIFY
)
263 if (((LPNMHDR
)lParam
)->code
== CDN_FOLDERCHANGE
)
264 PostMessage(GetParent(dlg
), WM_COMMAND
, IDCANCEL
, 0);
269 static void test_create_view_window2(void)
271 OPENFILENAMEA ofn
= {0};
272 char filename
[1024] = {0};
275 ofn
.lStructSize
= sizeof(ofn
);
276 ofn
.lpstrFile
= filename
;
278 ofn
.lpfnHook
= create_view_window2_hook
;
279 ofn
.Flags
= OFN_ENABLEHOOK
| OFN_EXPLORER
;
280 ret
= GetOpenFileNameA(&ofn
);
281 ok(!ret
, "GetOpenFileNameA returned %#x\n", ret
);
282 ret
= CommDlgExtendedError();
283 ok(!ret
, "CommDlgExtendedError returned %#x\n", ret
);
286 static void test_create_view_template(void)
288 OPENFILENAMEA ofn
= {0};
289 char filename
[1024] = {0};
292 ofn
.lStructSize
= sizeof(ofn
);
293 ofn
.lpstrFile
= filename
;
295 ofn
.lpfnHook
= (LPOFNHOOKPROC
)template_hook
;
296 ofn
.Flags
= OFN_ENABLEHOOK
| OFN_EXPLORER
| OFN_ENABLETEMPLATE
;
297 ofn
.hInstance
= GetModuleHandleA(NULL
);
298 ofn
.lpTemplateName
= "template1";
299 ofn
.lpstrFilter
="text\0*.txt\0All\0*\0\0";
300 ret
= GetOpenFileNameA(&ofn
);
301 ok(!ret
, "GetOpenFileNameA returned %#x\n", ret
);
302 ret
= CommDlgExtendedError();
303 ok(!ret
, "CommDlgExtendedError returned %#x\n", ret
);
306 /* test cases for resizing of the file dialog */
309 int resize_folderchange
;/* change in CDN_FOLDERCHANGE handler */
310 int resize_timer1
; /* change in first WM_TIMER handler */
311 int resize_check
; /* expected change (in second WM_TIMER handler) */
312 BOOL todo
; /* mark that test todo_wine */
313 BOOL testcontrols
; /* test resizing and moving of the controls */
314 } resize_testcases
[] = {
315 { 0 , 10, 10, 20,FALSE
,FALSE
}, /* 0 */
316 { 0 ,-10,-10,-20,FALSE
,FALSE
},
317 { OFN_ENABLESIZING
, 0, 0, 0,FALSE
,FALSE
},
318 { OFN_ENABLESIZING
, 0,-10, 0,FALSE
,FALSE
},
319 { OFN_ENABLESIZING
, 0, 10, 10,FALSE
, TRUE
},
320 { OFN_ENABLESIZING
,-10, 0, 10,FALSE
,FALSE
}, /* 5 */
321 { OFN_ENABLESIZING
, 10, 0, 10,FALSE
,FALSE
},
322 { OFN_ENABLESIZING
, 0, 10, 20,FALSE
,FALSE
},
327 static LONG_PTR WINAPI
resize_template_hook(HWND dlg
, UINT msg
, WPARAM wParam
, LPARAM lParam
)
329 static RECT initrc
, rc
;
330 static int index
, count
;
331 static int gotSWP_bottom
, gotShowWindow
;
332 HWND parent
= GetParent( dlg
);
334 #define MAXNRCTRLS 30
335 static RECT ctrlrcs
[MAXNRCTRLS
];
336 static int ctrlids
[MAXNRCTRLS
];
337 static HWND ctrls
[MAXNRCTRLS
];
346 index
= ((OPENFILENAME
*)lParam
)->lCustData
;
348 gotSWP_bottom
= gotShowWindow
= 0;
350 style
= GetWindowLong( parent
, GWL_STYLE
);
351 if( resize_testcases
[index
].flags
& OFN_ENABLESIZING
)
352 if( !(style
& WS_SIZEBOX
)) {
353 win_skip( "OFN_ENABLESIZING flag not supported.\n");
354 resizesupported
= FALSE
;
355 PostMessage( parent
, WM_COMMAND
, IDCANCEL
, 0);
357 ok( style
& WS_SIZEBOX
,
358 "testid %d: dialog should have a WS_SIZEBOX style.\n", index
);
360 ok( !(style
& WS_SIZEBOX
),
361 "testid %d: dialog should not have a WS_SIZEBOX style.\n", index
);
366 if(( (LPNMHDR
)lParam
)->code
== CDN_FOLDERCHANGE
){
367 GetWindowRect( parent
, &initrc
);
368 if( (resize
= resize_testcases
[index
].resize_folderchange
)){
369 MoveWindow( parent
, initrc
.left
,initrc
.top
, initrc
.right
- initrc
.left
+ resize
,
370 initrc
.bottom
- initrc
.top
+ resize
, TRUE
);
372 SetTimer( dlg
, 0, 100, 0);
379 /* store the control rectangles */
380 if( resize_testcases
[index
].testcontrols
) {
383 for( i
= 0, ctrl
= GetWindow( parent
, GW_CHILD
);
384 i
< MAXNRCTRLS
&& ctrl
;
385 i
++, ctrl
= GetWindow( ctrl
, GW_HWNDNEXT
)) {
386 ctrlids
[i
] = GetDlgCtrlID( ctrl
);
387 GetWindowRect( ctrl
, &ctrlrcs
[i
]);
388 MapWindowPoints( NULL
, parent
, (LPPOINT
) &ctrlrcs
[i
], 2);
393 if( (resize
= resize_testcases
[index
].resize_timer1
)){
394 GetWindowRect( parent
, &rc
);
395 MoveWindow( parent
, rc
.left
,rc
.top
, rc
.right
- rc
.left
+ resize
,
396 rc
.bottom
- rc
.top
+ resize
, TRUE
);
398 } else if( count
== 1){
399 resize
= resize_testcases
[index
].resize_check
;
400 GetWindowRect( parent
, &rc
);
401 if( resize_testcases
[index
].todo
){
403 ok( resize
== rc
.right
- rc
.left
- initrc
.right
+ initrc
.left
,
404 "testid %d size-x change %d expected %d\n", index
,
405 rc
.right
- rc
.left
- initrc
.right
+ initrc
.left
, resize
);
406 ok( resize
== rc
.bottom
- rc
.top
- initrc
.bottom
+ initrc
.top
,
407 "testid %d size-y change %d expected %d\n", index
,
408 rc
.bottom
- rc
.top
- initrc
.bottom
+ initrc
.top
, resize
);
411 ok( resize
== rc
.right
- rc
.left
- initrc
.right
+ initrc
.left
,
412 "testid %d size-x change %d expected %d\n", index
,
413 rc
.right
- rc
.left
- initrc
.right
+ initrc
.left
, resize
);
414 ok( resize
== rc
.bottom
- rc
.top
- initrc
.bottom
+ initrc
.top
,
415 "testid %d size-y change %d expected %d\n", index
,
416 rc
.bottom
- rc
.top
- initrc
.bottom
+ initrc
.top
, resize
);
418 if( resize_testcases
[index
].testcontrols
) {
421 for( i
= 0; i
< nrctrls
; i
++) {
422 GetWindowRect( ctrls
[i
], &rc
);
423 MapWindowPoints( NULL
, parent
, (LPPOINT
) &rc
, 2);
426 /* test if RECT R1, moved and sized result in R2 */
427 #define TESTRECTS( R1, R2, Mx, My, Sx, Sy) \
428 ((R1).left + (Mx) ==(R2).left \
429 &&(R1).top + (My) ==(R2).top \
430 &&(R1).right + (Mx) + (Sx) == (R2).right \
431 &&(R1).bottom + (My) + (Sy) ==(R2).bottom)
433 /* sized horizontal and moved vertical */
436 ok( TESTRECTS( ctrlrcs
[i
], rc
, 0, 10, 10, 0) ||
437 broken(TESTRECTS( ctrlrcs
[i
], rc
, 0, 10, 0, 0)),/*win98*/
438 "control id %03x should have sized horizontally and moved vertically, before %d,%d-%d,%d after %d,%d-%d,%d\n",
439 ctrlids
[i
], ctrlrcs
[i
].left
, ctrlrcs
[i
].top
,
440 ctrlrcs
[i
].right
, ctrlrcs
[i
].bottom
,
441 rc
.left
, rc
.top
, rc
.right
, rc
.bottom
);
443 /* sized horizontal and vertical */
445 ok( TESTRECTS( ctrlrcs
[i
], rc
, 0, 0, 10, 10),
446 "control id %03x should have sized horizontally and vertically, before %d,%d-%d,%d after %d,%d-%d,%d\n",
447 ctrlids
[i
], ctrlrcs
[i
].left
, ctrlrcs
[i
].top
,
448 ctrlrcs
[i
].right
, ctrlrcs
[i
].bottom
,
449 rc
.left
, rc
.top
, rc
.right
, rc
.bottom
);
451 /* moved horizontal and vertical */
454 ok( TESTRECTS( ctrlrcs
[i
], rc
, 10, 10, 0, 0) ||
455 broken(TESTRECTS( ctrlrcs
[i
], rc
, 0, 10, 0, 0)),/*win98*/
456 "control id %03x should have moved horizontally and vertically, before %d,%d-%d,%d after %d,%d-%d,%d\n",
457 ctrlids
[i
], ctrlrcs
[i
].left
, ctrlrcs
[i
].top
,
458 ctrlrcs
[i
].right
, ctrlrcs
[i
].bottom
,
459 rc
.left
, rc
.top
, rc
.right
, rc
.bottom
);
461 /* moved vertically */
465 ok( TESTRECTS( ctrlrcs
[i
], rc
, 0, 10, 0, 0),
466 "control id %03x should have moved vertically, before %d,%d-%d,%d after %d,%d-%d,%d\n",
467 ctrlids
[i
], ctrlrcs
[i
].left
, ctrlrcs
[i
].top
,
468 ctrlrcs
[i
].right
, ctrlrcs
[i
].bottom
,
469 rc
.left
, rc
.top
, rc
.right
, rc
.bottom
);
471 /* resized horizontal */
472 case cmb2
: /* aka IDC_LOOKIN */
473 ok( TESTRECTS( ctrlrcs
[i
], rc
, 0, 0, 10, 0)||
474 TESTRECTS( ctrlrcs
[i
], rc
, 0, 0, 0, 0), /* Vista and higher */
475 "control id %03x should have resized horizontally, before %d,%d-%d,%d after %d,%d-%d,%d\n",
476 ctrlids
[i
], ctrlrcs
[i
].left
, ctrlrcs
[i
].top
,
477 ctrlrcs
[i
].right
, ctrlrcs
[i
].bottom
,
478 rc
.left
, rc
.top
, rc
.right
, rc
.bottom
);
480 /* non moving non sizing controls */
482 ok( TESTRECTS( rc
, ctrlrcs
[i
], 0, 0, 0, 0),
483 "control id %03x was moved/resized, before %d,%d-%d,%d after %d,%d-%d,%d\n",
484 ctrlids
[i
], ctrlrcs
[i
].left
, ctrlrcs
[i
].top
,
485 ctrlrcs
[i
].right
, ctrlrcs
[i
].bottom
,
486 rc
.left
, rc
.top
, rc
.right
, rc
.bottom
);
488 /* todo_wine: non moving non sizing controls */
491 ok( TESTRECTS( rc
, ctrlrcs
[i
], 0, 0, 0, 0),
492 "control id %03x was moved/resized, before %d,%d-%d,%d after %d,%d-%d,%d\n",
493 ctrlids
[i
], ctrlrcs
[i
].left
, ctrlrcs
[i
].top
,
494 ctrlrcs
[i
].right
, ctrlrcs
[i
].bottom
,
495 rc
.left
, rc
.top
, rc
.right
, rc
.bottom
);
497 /* don't test: id is not unique */
504 trace("untested control id %03x before %d,%d-%d,%d after %d,%d-%d,%d\n",
505 ctrlids
[i
], ctrlrcs
[i
].left
, ctrlrcs
[i
].top
,
506 ctrlrcs
[i
].right
, ctrlrcs
[i
].bottom
,
507 rc
.left
, rc
.top
, rc
.right
, rc
.bottom
);
514 PostMessage( parent
, WM_COMMAND
, IDCANCEL
, 0);
519 case WM_WINDOWPOSCHANGING
:
521 WINDOWPOS
*pwp
= (WINDOWPOS
*)lParam
;
522 if( !index
&& pwp
->hwndInsertAfter
== HWND_BOTTOM
){
524 ok( gotShowWindow
== 0, "The WM_WINDOWPOSCHANGING message came after a WM_SHOWWINDOW message\n");
532 ok( gotSWP_bottom
== 1, "No WM_WINDOWPOSCHANGING message came before a WM_SHOWWINDOW message\n");
540 static void test_resize(void)
542 OPENFILENAME ofn
= { sizeof(OPENFILENAME
)};
543 char filename
[1024] = {0};
547 ofn
.lpstrFile
= filename
;
549 ofn
.lpfnHook
= (LPOFNHOOKPROC
) resize_template_hook
;
550 ofn
.hInstance
= GetModuleHandle(NULL
);
551 ofn
.lpTemplateName
= "template_sz";
552 for( i
= 0; resize_testcases
[i
].flags
!= 0xffffffff; i
++) {
554 ofn
.Flags
= resize_testcases
[i
].flags
|
555 OFN_ENABLEHOOK
| OFN_EXPLORER
| OFN_ENABLETEMPLATE
| OFN_SHOWHELP
;
556 ret
= GetOpenFileName(&ofn
);
557 ok(!ret
, "GetOpenFileName returned %#x\n", ret
);
558 ret
= CommDlgExtendedError();
559 ok(!ret
, "CommDlgExtendedError returned %#x\n", ret
);
563 /* test cases for control message IDOK */
564 /* Show case for bug #19079 */
566 int retval
; /* return code of the message handler */
567 BOOL setmsgresult
; /* set the result in the DWLP_MSGRESULT */
568 BOOL usemsgokstr
; /* use the FILEOKSTRING message instead of WM_NOTIFY:CDN_FILEOK */
569 BOOL do_subclass
; /* subclass the dialog hook procedure */
570 BOOL expclose
; /* is the dialog expected to close ? */
571 BOOL actclose
; /* has the dialog actually closed ? */
572 } ok_wndproc_testcase
;
574 static ok_wndproc_testcase ok_testcases
[] = {
575 { 0, FALSE
, FALSE
, FALSE
, TRUE
},
576 { 0, TRUE
, FALSE
, FALSE
, TRUE
},
577 { 0, FALSE
, FALSE
, TRUE
, TRUE
},
578 { 0, TRUE
, FALSE
, TRUE
, TRUE
},
579 { 1, FALSE
, FALSE
, FALSE
, TRUE
},
580 { 1, TRUE
, FALSE
, FALSE
, FALSE
},
581 { 1, FALSE
, FALSE
, TRUE
, FALSE
},
582 { 1, TRUE
, FALSE
, TRUE
, FALSE
},
583 /* FILEOKSTRING tests */
584 { 1, TRUE
, TRUE
, FALSE
, FALSE
},
585 { 1, FALSE
, TRUE
, TRUE
, FALSE
},
590 /* test_ok_wndproc can be used as hook procedure or a subclass
591 * window proc for the file dialog */
592 static LONG_PTR WINAPI
test_ok_wndproc(HWND dlg
, UINT msg
, WPARAM wParam
, LPARAM lParam
)
594 HWND parent
= GetParent( dlg
);
595 static ok_wndproc_testcase
*testcase
= NULL
;
596 static UINT msgFILEOKSTRING
;
597 if (msg
== WM_INITDIALOG
)
599 testcase
= (ok_wndproc_testcase
*)((OPENFILENAME
*)lParam
)->lCustData
;
600 testcase
->actclose
= TRUE
;
601 msgFILEOKSTRING
= RegisterWindowMessageA( FILEOKSTRING
);
603 if( msg
== WM_NOTIFY
) {
604 if(((LPNMHDR
)lParam
)->code
== CDN_FOLDERCHANGE
) {
605 SetTimer( dlg
, 0, 100, 0);
606 PostMessage( parent
, WM_COMMAND
, IDOK
, 0);
608 } else if(((LPNMHDR
)lParam
)->code
== CDN_FILEOK
) {
609 if( testcase
->usemsgokstr
)
611 if( testcase
->setmsgresult
)
612 SetWindowLongPtrA( dlg
, DWLP_MSGRESULT
, testcase
->retval
);
613 return testcase
->retval
;
616 if( msg
== msgFILEOKSTRING
) {
617 if( !testcase
->usemsgokstr
)
619 if( testcase
->setmsgresult
)
620 SetWindowLongPtrA( dlg
, DWLP_MSGRESULT
, testcase
->retval
);
621 return testcase
->retval
;
623 if( msg
== WM_TIMER
) {
624 /* the dialog did not close automatically */
625 testcase
->actclose
= FALSE
;
627 PostMessage( parent
, WM_COMMAND
, IDCANCEL
, 0);
630 if( testcase
&& testcase
->do_subclass
)
631 return DefWindowProc( dlg
, msg
, wParam
, lParam
);
635 static LONG_PTR WINAPI
ok_template_hook(HWND dlg
, UINT msg
, WPARAM wParam
, LPARAM lParam
)
637 if (msg
== WM_SETFONT
)
638 SetWindowLongPtrA( dlg
, GWLP_WNDPROC
, (LONG_PTR
) test_ok_wndproc
);
642 static void test_ok(void)
644 OPENFILENAME ofn
= { sizeof(OPENFILENAME
)};
645 char filename
[1024] = {0};
646 char tmpfilename
[ MAX_PATH
];
647 char curdir
[MAX_PATH
];
651 ok(GetCurrentDirectoryA(sizeof(curdir
), curdir
) != 0, "Failed to get current dir err %d\n", GetLastError());
652 if (!GetTempFileNameA(".", "txt", 0, tmpfilename
)) {
653 skip("Failed to create a temporary file name\n");
656 ofn
.lpstrFile
= filename
;
658 ofn
.hInstance
= GetModuleHandle(NULL
);
659 ofn
.lpTemplateName
= "template1";
660 ofn
.Flags
= OFN_ENABLEHOOK
| OFN_EXPLORER
| OFN_ENABLETEMPLATE
;
661 for( i
= 0; ok_testcases
[i
].retval
!= -1; i
++) {
662 strcpy( filename
, tmpfilename
);
663 ofn
.lCustData
= (LPARAM
)(ok_testcases
+ i
);
664 ofn
.lpfnHook
= ok_testcases
[i
].do_subclass
665 ? (LPOFNHOOKPROC
) ok_template_hook
666 : (LPOFNHOOKPROC
) test_ok_wndproc
;
667 ret
= GetOpenFileNameA(&ofn
);
668 ok( ok_testcases
[i
].expclose
== ok_testcases
[i
].actclose
,
669 "testid %d: Open File dialog should %shave closed.\n", i
,
670 ok_testcases
[i
].expclose
? "" : "NOT ");
671 ok(ret
== ok_testcases
[i
].expclose
, "testid %d: GetOpenFileName returned %#x\n", i
, ret
);
672 ret
= CommDlgExtendedError();
673 ok(!ret
, "CommDlgExtendedError returned %#x\n", ret
);
674 ok(SetCurrentDirectoryA(curdir
), "Failed to restore current dir err %d\n", GetLastError());
676 ret
= DeleteFileA( tmpfilename
);
677 ok( ret
, "Failed to delete temporary file %s err %d\n", tmpfilename
, GetLastError());
680 /* test arranging with a custom template */
682 int x
, y
; /* left, top coordinates */
683 int cx
, cy
; /* width and height */
686 int nrcontrols
; /* 0: no controls, 1: just the stc32 control 2: with button */
691 } arrange_tests
[] = {
692 /* do not change the first two cases: used to get the uncustomized sizes */
693 { 0, {0},{0},{0},0 },
694 { 0, {0},{0},{0}, OFN_SHOWHELP
},
695 /* two tests with just a subdialog, no controls */
696 { 0, {0, 0, 316, 76},{0},{0},0 },
697 { 0, {0, 0, 100, 76},{0},{0}, OFN_SHOWHELP
},
698 /* now with a control with id stc32 */
699 { 1, {0, 0, 316, 76} ,{0, 0, 204, 76,},{0},0 }, /* bug #17748*/
700 { 1, {0, 0, 316, 76} ,{0, 0, 204, 76,},{0}, OFN_SHOWHELP
}, /* bug #17748*/
701 /* tests with size of the stc32 control higher or wider then the standard dialog */
702 { 1, {0, 0, 316, 170} ,{0, 0, 204, 170,},{0},0 },
703 { 1, {0, 0, 316, 165} ,{0, 0, 411, 165,},{0}, OFN_SHOWHELP
},
704 /* move the stc32 control around */
705 { 1, {0, 0, 300, 100} ,{73, 17, 50, 50,},{0},0 },
707 { 2, {0, 0, 280, 100} ,{0, 0, 50, 50,},{300,20,30,30},0 },
708 /* enable resizing should make the dialog bigger */
709 { 0, {0},{0},{0}, OFN_SHOWHELP
|OFN_ENABLESIZING
},
714 static LONG_PTR WINAPI
template_hook_arrange(HWND dlgChild
, UINT msg
, WPARAM wParam
, LPARAM lParam
)
716 static int index
, fixhelp
;
717 static posz posz0
[2];
718 static RECT clrcParent
, clrcChild
, rcStc32
;
719 static HWND hwndStc32
;
722 dlgParent
= GetParent( dlgChild
);
723 if (msg
== WM_INITDIALOG
) {
724 index
= ((OPENFILENAME
*)lParam
)->lCustData
;
725 /* get the positions before rearrangement */
726 GetClientRect( dlgParent
, &clrcParent
);
727 GetClientRect( dlgChild
, &clrcChild
);
728 hwndStc32
= GetDlgItem( dlgChild
, stc32
);
729 if( hwndStc32
) GetWindowRect( hwndStc32
, &rcStc32
);
731 if (msg
== WM_NOTIFY
&& ((LPNMHDR
)lParam
)->code
== CDN_FOLDERCHANGE
) {
734 GetWindowRect( dlgParent
, &wrcParent
);
735 /* the fist two "tests" just save the dialogs position, with and without
738 posz0
[0].x
= wrcParent
.left
;
739 posz0
[0].y
= wrcParent
.top
;
740 posz0
[0].cx
= wrcParent
.right
- wrcParent
.left
;
741 posz0
[0].cy
= wrcParent
.bottom
- wrcParent
.top
;
742 } else if( index
== 1) {
743 posz0
[1].x
= wrcParent
.left
;
744 posz0
[1].y
= wrcParent
.top
;
745 posz0
[1].cx
= wrcParent
.right
- wrcParent
.left
;
746 posz0
[1].cy
= wrcParent
.bottom
- wrcParent
.top
;
747 fixhelp
= posz0
[1].cy
- posz0
[0].cy
;
751 int expectx
, expecty
;
754 withhelp
= (arrange_tests
[index
].ofnflags
& OFN_SHOWHELP
) != 0;
755 GetWindowRect( dlgParent
, &wrcParent
);
757 /* case with no custom subitem with stc32:
758 * default to all custom controls below the standard */
759 expecty
= posz0
[withhelp
].cy
+ clrcChild
.bottom
;
760 expectx
= posz0
[withhelp
].cx
;
762 /* special case: there is a control with id stc32 */
763 /* expected height */
764 expecty
= posz0
[withhelp
].cy
;
765 if( rcStc32
.bottom
- rcStc32
.top
> clrcParent
.bottom
) {
766 expecty
+= clrcChild
.bottom
- clrcParent
.bottom
;
767 if( !withhelp
) expecty
+= fixhelp
;
770 expecty
+= clrcChild
.bottom
- ( rcStc32
.bottom
- rcStc32
.top
) ;
772 expectx
= posz0
[withhelp
].cx
;
773 if( rcStc32
.right
- rcStc32
.left
> clrcParent
.right
) {
774 expectx
+= clrcChild
.right
- clrcParent
.right
;
777 expectx
+= clrcChild
.right
- ( rcStc32
.right
- rcStc32
.left
) ;
779 style
= GetWindowLong( dlgParent
, GWL_STYLE
);
780 if( !(style
& WS_SIZEBOX
)) {
781 /* without the OFN_ENABLESIZING flag */
782 ok( wrcParent
.bottom
- wrcParent
.top
== expecty
,
783 "Wrong height of dialog %d, expected %d\n",
784 wrcParent
.bottom
- wrcParent
.top
, expecty
);
785 ok( wrcParent
.right
- wrcParent
.left
== expectx
,
786 "Wrong width of dialog %d, expected %d\n",
787 wrcParent
.right
- wrcParent
.left
, expectx
);
789 /* with the OFN_ENABLESIZING flag */
790 ok( wrcParent
.bottom
- wrcParent
.top
> expecty
,
791 "Wrong height of dialog %d, expected more than %d\n",
792 wrcParent
.bottom
- wrcParent
.top
, expecty
);
793 ok( wrcParent
.right
- wrcParent
.left
> expectx
,
794 "Wrong width of dialog %d, expected more than %d\n",
795 wrcParent
.right
- wrcParent
.left
, expectx
);
799 PostMessage( dlgParent
, WM_COMMAND
, IDCANCEL
, 0);
804 static void test_arrange(void)
806 OPENFILENAMEA ofn
= {0};
807 char filename
[1024] = {0};
812 DLGTEMPLATE
*template;
813 DLGITEMTEMPLATE
*itemtemplateStc32
, *itemtemplateBtn
;
816 /* load subdialog template into memory */
817 hRes
= FindResource( GetModuleHandle(NULL
), "template_stc32", (LPSTR
)RT_DIALOG
);
818 hDlgTmpl
= LoadResource( GetModuleHandle(NULL
), hRes
);
819 /* get pointers to the structures for the dialog and the controls */
820 pv
= LockResource( hDlgTmpl
);
821 template = (DLGTEMPLATE
*)pv
;
822 if( template->x
!= 11111) {
823 win_skip("could not find the dialog template\n");
826 /* skip dialog template, menu, class and title */
827 pv
+= sizeof(DLGTEMPLATE
);
828 pv
+= 3 * sizeof(WORD
);
833 /* align on 32 bit boundaries */
834 pv
= (LPBYTE
)(((UINT_PTR
)pv
+ 3 ) & ~3);
835 itemtemplateStc32
= (DLGITEMTEMPLATE
*)pv
;
836 if( itemtemplateStc32
->x
!= 22222) {
837 win_skip("could not find the first item template\n");
840 /* skip itemtemplate, class, title and creation data */
841 pv
+= sizeof(DLGITEMTEMPLATE
);
842 pv
+= 4 * sizeof(WORD
);
843 /* align on 32 bit boundaries */
844 pv
= (LPBYTE
)(((UINT_PTR
)pv
+ 3 ) & ~3);
845 itemtemplateBtn
= (DLGITEMTEMPLATE
*)pv
;
846 if( itemtemplateBtn
->x
!= 12345) {
847 win_skip("could not find the second item template\n");
851 ofn
.lStructSize
= sizeof(ofn
);
852 ofn
.lpstrFile
= filename
;
854 ofn
.lpfnHook
= (LPOFNHOOKPROC
)template_hook_arrange
;
855 ofn
.hInstance
= hDlgTmpl
;
856 ofn
.lpstrFilter
="text\0*.txt\0All\0*\0\0";
857 for( i
= 0; arrange_tests
[i
].nrcontrols
!= -1; i
++) {
859 ofn
.Flags
= OFN_ENABLEHOOK
| OFN_EXPLORER
| OFN_ENABLETEMPLATEHANDLE
| OFN_HIDEREADONLY
|
860 arrange_tests
[i
].ofnflags
;
861 template->cdit
= arrange_tests
[i
].nrcontrols
;
862 template->x
= arrange_tests
[i
].poszDlg
.x
;
863 template->y
= arrange_tests
[i
].poszDlg
.y
;
864 template->cx
= arrange_tests
[i
].poszDlg
.cx
;
865 template->cy
= arrange_tests
[i
].poszDlg
.cy
;
866 itemtemplateStc32
->x
= arrange_tests
[i
].poszStc32
.x
;
867 itemtemplateStc32
->y
= arrange_tests
[i
].poszStc32
.y
;
868 itemtemplateStc32
->cx
= arrange_tests
[i
].poszStc32
.cx
;
869 itemtemplateStc32
->cy
= arrange_tests
[i
].poszStc32
.cy
;
870 itemtemplateBtn
->x
= arrange_tests
[i
].poszBtn
.x
;
871 itemtemplateBtn
->y
= arrange_tests
[i
].poszBtn
.y
;
872 itemtemplateBtn
->cx
= arrange_tests
[i
].poszBtn
.cx
;
873 itemtemplateBtn
->cy
= arrange_tests
[i
].poszBtn
.cy
;
874 ret
= GetOpenFileNameA(&ofn
);
875 ok(!ret
, "GetOpenFileNameA returned %#x\n", ret
);
876 ret
= CommDlgExtendedError();
877 ok(!ret
, "CommDlgExtendedError returned %#x\n", ret
);
881 static CHAR SYSDIR
[MAX_PATH
];
883 static UINT_PTR CALLBACK
path_hook_proc( HWND hDlg
, UINT msg
, WPARAM wParam
, LPARAM lParam
)
887 if( msg
== WM_NOTIFY
)
889 nmh
= (LPNMHDR
) lParam
;
890 if( nmh
->code
== CDN_INITDONE
)
892 PostMessage( GetParent(hDlg
), WM_COMMAND
, IDCANCEL
, FALSE
);
894 else if ( nmh
->code
== CDN_FOLDERCHANGE
)
899 memset(buf
, 0x66, sizeof(buf
));
900 ret
= SendMessageA( GetParent(hDlg
), CDM_GETFOLDERPATH
, sizeof(buf
), (LPARAM
)buf
);
901 ok(!lstrcmpiA(SYSDIR
, buf
), "Expected '%s', got '%s'\n", SYSDIR
, buf
);
902 ok(lstrlenA(SYSDIR
) + 1 == ret
, "Expected %d, got %d\n", lstrlenA(SYSDIR
) + 1, ret
);
909 static void test_getfolderpath(void)
913 char szFileName
[MAX_PATH
] = "";
914 char szInitialDir
[MAX_PATH
];
916 /* We need to pick a different directory as the other tests because of new
917 * Windows 7 behavior.
919 GetSystemDirectory(szInitialDir
, MAX_PATH
);
920 lstrcpyA(SYSDIR
, szInitialDir
);
922 ZeroMemory(&ofn
, sizeof(ofn
));
924 ofn
.lStructSize
= sizeof(ofn
);
925 ofn
.hwndOwner
= NULL
;
926 ofn
.lpstrFilter
= "Text Files (*.txt)\0*.txt\0All Files (*.*)\0*.*\0";
927 ofn
.lpstrFile
= szFileName
;
928 ofn
.nMaxFile
= MAX_PATH
;
929 ofn
.Flags
= OFN_EXPLORER
| OFN_FILEMUSTEXIST
| OFN_HIDEREADONLY
| OFN_ENABLEHOOK
;
930 ofn
.lpstrDefExt
= "txt";
931 ofn
.lpfnHook
= path_hook_proc
;
932 ofn
.lpstrInitialDir
= szInitialDir
;
934 result
= GetOpenFileNameA(&ofn
);
935 ok(0 == result
, "expected 0, got %d\n", result
);
936 ok(0 == CommDlgExtendedError(), "expected 0, got %d\n",
937 CommDlgExtendedError());
939 result
= GetSaveFileNameA(&ofn
);
940 ok(0 == result
, "expected 0, got %d\n", result
);
941 ok(0 == CommDlgExtendedError(), "expected 0, got %d\n",
942 CommDlgExtendedError());
945 static void test_resizable2(void)
947 OPENFILENAMEA ofn
= {0};
948 char filename
[1024] = "pls press Enter if sizable, Esc otherwise";
951 /* interactive because there is no hook function */
952 if( !winetest_interactive
) {
953 skip( "some interactive resizable dialog tests (set WINETEST_INTERACTIVE=1)\n");
956 ofn
.lStructSize
= sizeof(ofn
);
957 ofn
.lpstrFile
= filename
;
960 ofn
.hInstance
= GetModuleHandleA(NULL
);
961 ofn
.lpTemplateName
= "template1";
962 ofn
.Flags
= OFN_EXPLORER
;
964 ret
= GetOpenFileNameA(&ofn
);
965 ok( ret
== ISSIZABLE
, "File Dialog should have been sizable\n");
966 ret
= CommDlgExtendedError();
967 ok(!ret
, "CommDlgExtendedError returned %#x\n", ret
);
968 ofn
.Flags
= OFN_EXPLORER
| OFN_ENABLETEMPLATE
;
969 ret
= GetOpenFileNameA(&ofn
);
970 ok( ret
!= ISSIZABLE
, "File Dialog should NOT have been sizable\n");
971 ret
= CommDlgExtendedError();
972 ok(!ret
, "CommDlgExtendedError returned %#x\n", ret
);
973 ofn
.Flags
= OFN_EXPLORER
| OFN_ENABLETEMPLATEHANDLE
;
974 ofn
.hInstance
= LoadResource( GetModuleHandle(NULL
), FindResource( GetModuleHandle(NULL
), "template1", (LPSTR
)RT_DIALOG
));
975 ofn
.lpTemplateName
= NULL
;
976 ret
= GetOpenFileNameA(&ofn
);
977 ok( ret
!= ISSIZABLE
, "File Dialog should NOT have been sizable\n");
978 ret
= CommDlgExtendedError();
979 ok(!ret
, "CommDlgExtendedError returned %#x\n", ret
);
980 ofn
.Flags
= OFN_EXPLORER
| OFN_ENABLEHOOK
;
981 ret
= GetOpenFileNameA(&ofn
);
982 ok( ret
!= ISSIZABLE
, "File Dialog should NOT have been sizable\n");
983 ret
= CommDlgExtendedError();
984 ok(!ret
, "CommDlgExtendedError returned %#x\n", ret
);
988 static void test_mru(void)
990 ok_wndproc_testcase testcase
= {0};
991 OPENFILENAME ofn
= {sizeof(OPENFILENAME
)};
992 const char *test_dir_name
= "C:\\mru_test";
993 const char *test_file_name
= "test.txt";
994 const char *test_full_path
= "C:\\mru_test\\test.txt";
995 char filename_buf
[MAX_PATH
];
998 ofn
.lpstrFile
= filename_buf
;
999 ofn
.nMaxFile
= sizeof(filename_buf
);
1000 ofn
.lpTemplateName
= "template1";
1001 ofn
.hInstance
= GetModuleHandle(NULL
);
1002 ofn
.Flags
= OFN_ENABLEHOOK
| OFN_EXPLORER
| OFN_ENABLETEMPLATE
| OFN_NOCHANGEDIR
;
1003 ofn
.lCustData
= (LPARAM
)&testcase
;
1004 ofn
.lpfnHook
= (LPOFNHOOKPROC
)test_ok_wndproc
;
1006 SetLastError(0xdeadbeef);
1007 ret
= CreateDirectoryA(test_dir_name
, NULL
);
1008 ok(ret
== TRUE
, "CreateDirectoryA should have succeeded: %d\n", GetLastError());
1010 /* "teach" comdlg32 about this directory */
1011 strcpy(filename_buf
, test_full_path
);
1012 SetLastError(0xdeadbeef);
1013 ret
= GetOpenFileNameA(&ofn
);
1014 ok(ret
, "GetOpenFileNameA should have succeeded: %d\n", GetLastError());
1015 ret
= CommDlgExtendedError();
1016 ok(!ret
, "CommDlgExtendedError returned %x\n", ret
);
1017 ok(testcase
.actclose
, "Open File dialog should have closed.\n");
1018 ok(!strcmp(ofn
.lpstrFile
, test_full_path
), "Expected to get %s, got %s\n", test_full_path
, ofn
.lpstrFile
);
1020 /* get a filename without a full path. it should return the file in
1021 * test_dir_name, not in the CWD */
1022 strcpy(filename_buf
, test_file_name
);
1023 SetLastError(0xdeadbeef);
1024 ret
= GetOpenFileNameA(&ofn
);
1025 ok(ret
, "GetOpenFileNameA should have succeeded: %d\n", GetLastError());
1026 ret
= CommDlgExtendedError();
1027 ok(!ret
, "CommDlgExtendedError returned %x\n", ret
);
1028 ok(testcase
.actclose
, "Open File dialog should have closed.\n");
1029 if(strcmp(ofn
.lpstrFile
, test_full_path
) != 0)
1030 win_skip("Platform doesn't save MRU data\n");
1032 SetLastError(0xdeadbeef);
1033 ret
= RemoveDirectoryA(test_dir_name
);
1034 ok(ret
== TRUE
, "RemoveDirectoryA should have succeeded: %d\n", GetLastError());
1039 test_DialogCancel();
1040 test_create_view_window2();
1041 test_create_view_template();
1045 test_getfolderpath();
1047 if( resizesupported
) test_resizable2();