1 /* 2 * Unit test suite for comdlg32 API functions: file dialogs 3 * 4 * Copyright 2007 Google (Lei Zhang) 5 * 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. 10 * 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. 15 * 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 19 * 20 */ 21 22 #include <windows.h> 23 #include <wine/test.h> 24 25 #include "shlguid.h" 26 #define COBJMACROS 27 #include "shobjidl.h" 28 #include "commdlg.h" 29 #include "cderr.h" 30 #include "dlgs.h" 31 32 #ifdef __REACTOS__ 33 #include <reactos/undocuser.h> 34 #endif 35 36 /* ##### */ 37 38 static BOOL resizesupported = TRUE; 39 40 static void toolbarcheck( HWND hDlg) 41 { 42 /* test toolbar properties */ 43 /* bug #10532 */ 44 int maxtextrows; 45 HWND ctrl; 46 DWORD ret; 47 char classname[20]; 48 49 for( ctrl = GetWindow( hDlg, GW_CHILD); 50 ctrl ; ctrl = GetWindow( ctrl, GW_HWNDNEXT)) { 51 GetClassNameA( ctrl, classname, 10); 52 classname[7] = '\0'; 53 if( !strcmp( classname, "Toolbar")) break; 54 } 55 ok( ctrl != NULL, "could not get the toolbar control\n"); 56 ret = SendMessageA( ctrl, TB_ADDSTRINGA, 0, (LPARAM)"winetestwinetest\0\0"); 57 ok( ret == 0, "addstring returned %d (expected 0)\n", ret); 58 maxtextrows = SendMessageA( ctrl, TB_GETTEXTROWS, 0, 0); 59 ok( maxtextrows == 0 || broken(maxtextrows == 1), /* Win2k and below */ 60 "Get(Max)TextRows returned %d (expected 0)\n", maxtextrows); 61 } 62 63 64 static UINT_PTR CALLBACK OFNHookProc( HWND hDlg, UINT msg, WPARAM wParam, LPARAM lParam) 65 { 66 LPNMHDR nmh; 67 68 if( msg == WM_NOTIFY) 69 { 70 nmh = (LPNMHDR) lParam; 71 if( nmh->code == CDN_INITDONE) 72 { 73 PostMessageA( GetParent(hDlg), WM_COMMAND, IDCANCEL, FALSE); 74 } else if (nmh->code == CDN_FOLDERCHANGE ) 75 { 76 char buf[1024]; 77 int ret; 78 79 memset(buf, 0x66, sizeof(buf)); 80 ret = SendMessageA( GetParent(hDlg), CDM_GETFOLDERIDLIST, 5, (LPARAM)buf); 81 ok(ret > 0, "CMD_GETFOLDERIDLIST not implemented\n"); 82 if (ret > 5) 83 ok(buf[0] == 0x66 && buf[1] == 0x66, "CMD_GETFOLDERIDLIST: The buffer was touched on failure\n"); 84 toolbarcheck( GetParent(hDlg)); 85 } 86 } 87 88 return 0; 89 } 90 91 /* bug 6829 */ 92 static void test_DialogCancel(void) 93 { 94 OPENFILENAMEA ofn; 95 BOOL result; 96 char szFileName[MAX_PATH] = ""; 97 char szInitialDir[MAX_PATH]; 98 99 GetWindowsDirectoryA(szInitialDir, MAX_PATH); 100 101 ZeroMemory(&ofn, sizeof(ofn)); 102 103 ofn.lStructSize = OPENFILENAME_SIZE_VERSION_400A; 104 ofn.hwndOwner = NULL; 105 ofn.lpstrFilter = "Text Files (*.txt)\0*.txt\0All Files (*.*)\0*.*\0"; 106 ofn.lpstrFile = szFileName; 107 ofn.nMaxFile = MAX_PATH; 108 ofn.Flags = OFN_EXPLORER | OFN_FILEMUSTEXIST | OFN_HIDEREADONLY | OFN_ENABLEHOOK; 109 ofn.lpstrDefExt = "txt"; 110 ofn.lpfnHook = OFNHookProc; 111 ofn.lpstrInitialDir = szInitialDir; 112 113 PrintDlgA(NULL); 114 ok(CDERR_INITIALIZATION == CommDlgExtendedError(), 115 "expected CDERR_INITIALIZATION, got %d\n", CommDlgExtendedError()); 116 117 result = GetOpenFileNameA(&ofn); 118 ok(FALSE == result, "expected FALSE, got %d\n", result); 119 ok(0 == CommDlgExtendedError(), "expected 0, got %d\n", 120 CommDlgExtendedError()); 121 122 PrintDlgA(NULL); 123 ok(CDERR_INITIALIZATION == CommDlgExtendedError(), 124 "expected CDERR_INITIALIZATION, got %d\n", CommDlgExtendedError()); 125 126 result = GetSaveFileNameA(&ofn); 127 ok(FALSE == result, "expected FALSE, got %d\n", result); 128 ok(0 == CommDlgExtendedError(), "expected 0, got %d\n", 129 CommDlgExtendedError()); 130 131 PrintDlgA(NULL); 132 ok(CDERR_INITIALIZATION == CommDlgExtendedError(), 133 "expected CDERR_INITIALIZATION, got %d\n", CommDlgExtendedError()); 134 135 /* Before passing the ofn to Unicode functions, remove the ANSI strings */ 136 ofn.lpstrFilter = NULL; 137 ofn.lpstrInitialDir = NULL; 138 ofn.lpstrDefExt = NULL; 139 140 PrintDlgA(NULL); 141 ok(CDERR_INITIALIZATION == CommDlgExtendedError(), 142 "expected CDERR_INITIALIZATION, got %d\n", CommDlgExtendedError()); 143 144 SetLastError(0xdeadbeef); 145 result = GetOpenFileNameW((LPOPENFILENAMEW) &ofn); 146 if (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED) 147 win_skip("GetOpenFileNameW is not implemented\n"); 148 else 149 { 150 ok(FALSE == result, "expected FALSE, got %d\n", result); 151 ok(0 == CommDlgExtendedError(), "expected 0, got %d\n", CommDlgExtendedError()); 152 } 153 154 SetLastError(0xdeadbeef); 155 result = GetSaveFileNameW((LPOPENFILENAMEW) &ofn); 156 if (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED) 157 win_skip("GetSaveFileNameW is not implemented\n"); 158 else 159 { 160 ok(FALSE == result, "expected FALSE, got %d\n", result); 161 ok(0 == CommDlgExtendedError(), "expected 0, got %d\n", CommDlgExtendedError()); 162 } 163 } 164 165 static UINT_PTR CALLBACK create_view_window2_hook(HWND dlg, UINT msg, WPARAM wParam, LPARAM lParam) 166 { 167 if (msg == WM_NOTIFY) 168 { 169 if (((LPNMHDR)lParam)->code == CDN_FOLDERCHANGE) 170 { 171 IShellBrowser *shell_browser = (IShellBrowser *)SendMessageA(GetParent(dlg), WM_USER + 7 /* WM_GETISHELLBROWSER */, 0, 0); 172 IShellView *shell_view = NULL; 173 IShellView2 *shell_view2 = NULL; 174 SV2CVW2_PARAMS view_params; 175 FOLDERSETTINGS folder_settings; 176 HRESULT hr; 177 RECT rect = {0, 0, 0, 0}; 178 179 hr = IShellBrowser_QueryActiveShellView(shell_browser, &shell_view); 180 ok(SUCCEEDED(hr), "QueryActiveShellView returned %#x\n", hr); 181 if (FAILED(hr)) goto cleanup; 182 183 hr = IShellView_QueryInterface(shell_view, &IID_IShellView2, (void **)&shell_view2); 184 if (hr == E_NOINTERFACE) 185 { 186 win_skip("IShellView2 not supported\n"); 187 goto cleanup; 188 } 189 ok(SUCCEEDED(hr), "QueryInterface returned %#x\n", hr); 190 if (FAILED(hr)) goto cleanup; 191 192 hr = IShellView2_DestroyViewWindow(shell_view2); 193 ok(SUCCEEDED(hr), "DestroyViewWindow returned %#x\n", hr); 194 195 folder_settings.ViewMode = FVM_LIST; 196 folder_settings.fFlags = 0; 197 198 view_params.cbSize = sizeof(view_params); 199 view_params.psvPrev = NULL; 200 view_params.pfs = &folder_settings; 201 view_params.psbOwner = shell_browser; 202 view_params.prcView = ▭ 203 view_params.pvid = NULL; 204 view_params.hwndView = NULL; 205 206 hr = IShellView2_CreateViewWindow2(shell_view2, &view_params); 207 if (hr == E_FAIL) 208 { 209 win_skip("CreateViewWindow2 is broken on Vista/W2K8\n"); 210 goto cleanup; 211 } 212 ok(SUCCEEDED(hr), "CreateViewWindow2 returned %#x\n", hr); 213 if (FAILED(hr)) goto cleanup; 214 215 hr = IShellView2_GetCurrentInfo(shell_view2, &folder_settings); 216 ok(SUCCEEDED(hr), "GetCurrentInfo returned %#x\n", hr); 217 ok(folder_settings.ViewMode == FVM_LIST, 218 "view mode is %d, expected FVM_LIST\n", 219 folder_settings.ViewMode); 220 221 hr = IShellView2_DestroyViewWindow(shell_view2); 222 ok(SUCCEEDED(hr), "DestroyViewWindow returned %#x\n", hr); 223 224 /* XP and W2K3 need this. On W2K the call to DestroyWindow() fails and has 225 * no side effects. NT4 doesn't get here. (FIXME: Vista doesn't get here yet). 226 */ 227 DestroyWindow(view_params.hwndView); 228 229 view_params.pvid = &VID_Details; 230 hr = IShellView2_CreateViewWindow2(shell_view2, &view_params); 231 ok(SUCCEEDED(hr), "CreateViewWindow2 returned %#x\n", hr); 232 if (FAILED(hr)) goto cleanup; 233 234 hr = IShellView2_GetCurrentInfo(shell_view2, &folder_settings); 235 ok(SUCCEEDED(hr), "GetCurrentInfo returned %#x\n", hr); 236 ok(folder_settings.ViewMode == FVM_DETAILS || broken(folder_settings.ViewMode == FVM_LIST), /* nt4 */ 237 "view mode is %d, expected FVM_DETAILS\n", 238 folder_settings.ViewMode); 239 240 cleanup: 241 if (shell_view2) IShellView2_Release(shell_view2); 242 if (shell_view) IShellView_Release(shell_view); 243 PostMessageA(GetParent(dlg), WM_COMMAND, IDCANCEL, 0); 244 } 245 } 246 return 0; 247 } 248 249 static UINT_PTR WINAPI template_hook(HWND dlg, UINT msg, WPARAM wParam, LPARAM lParam) 250 { 251 if (msg == WM_INITDIALOG) 252 { 253 HWND p,cb; 254 INT sel; 255 p = GetParent(dlg); 256 ok(p!=NULL, "Failed to get parent of template\n"); 257 cb = GetDlgItem(p,0x470); 258 ok(cb!=NULL, "Failed to get filter combobox\n"); 259 sel = SendMessageA(cb, CB_GETCURSEL, 0, 0); 260 ok (sel != -1, "Failed to get selection from filter listbox\n"); 261 } 262 if (msg == WM_NOTIFY) 263 { 264 if (((LPNMHDR)lParam)->code == CDN_FOLDERCHANGE) 265 PostMessageA(GetParent(dlg), WM_COMMAND, IDCANCEL, 0); 266 } 267 return 0; 268 } 269 270 static void test_create_view_window2(void) 271 { 272 OPENFILENAMEA ofn = {0}; 273 char filename[1024] = {0}; 274 DWORD ret; 275 276 ofn.lStructSize = OPENFILENAME_SIZE_VERSION_400A; 277 ofn.lpstrFile = filename; 278 ofn.nMaxFile = 1024; 279 ofn.lpfnHook = create_view_window2_hook; 280 ofn.Flags = OFN_ENABLEHOOK | OFN_EXPLORER; 281 ret = GetOpenFileNameA(&ofn); 282 ok(!ret, "GetOpenFileNameA returned %#x\n", ret); 283 ret = CommDlgExtendedError(); 284 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 285 } 286 287 static void test_create_view_template(void) 288 { 289 OPENFILENAMEA ofn = {0}; 290 char filename[1024] = {0}; 291 DWORD ret; 292 293 ofn.lStructSize = OPENFILENAME_SIZE_VERSION_400A; 294 ofn.lpstrFile = filename; 295 ofn.nMaxFile = 1024; 296 ofn.lpfnHook = template_hook; 297 ofn.Flags = OFN_ENABLEHOOK | OFN_EXPLORER| OFN_ENABLETEMPLATE; 298 ofn.hInstance = GetModuleHandleA(NULL); 299 ofn.lpTemplateName = "template1"; 300 ofn.lpstrFilter="text\0*.txt\0All\0*\0\0"; 301 ret = GetOpenFileNameA(&ofn); 302 ok(!ret, "GetOpenFileNameA returned %#x\n", ret); 303 ret = CommDlgExtendedError(); 304 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 305 } 306 307 /* test cases for resizing of the file dialog */ 308 static const struct { 309 DWORD flags; 310 int resize_folderchange;/* change in CDN_FOLDERCHANGE handler */ 311 int resize_timer1; /* change in first WM_TIMER handler */ 312 int resize_check; /* expected change (in second WM_TIMER handler) */ 313 BOOL todo; /* mark that test todo_wine */ 314 BOOL testcontrols; /* test resizing and moving of the controls */ 315 } resize_testcases[] = { 316 { 0 , 10, 10, 20,FALSE,FALSE}, /* 0 */ 317 { 0 ,-10,-10,-20,FALSE,FALSE}, 318 { OFN_ENABLESIZING , 0, 0, 0,FALSE,FALSE}, 319 { OFN_ENABLESIZING , 0,-10, 0,FALSE,FALSE}, 320 { OFN_ENABLESIZING , 0, 10, 10,FALSE, TRUE}, 321 { OFN_ENABLESIZING ,-10, 0, 10,FALSE,FALSE}, /* 5 */ 322 { OFN_ENABLESIZING , 10, 0, 10,FALSE,FALSE}, 323 { OFN_ENABLESIZING , 0, 10, 20,FALSE,FALSE}, 324 /* mark the end */ 325 { 0xffffffff } 326 }; 327 328 static UINT_PTR WINAPI resize_template_hook(HWND dlg, UINT msg, WPARAM wParam, LPARAM lParam) 329 { 330 static RECT initrc, rc; 331 static int index, count; 332 static BOOL gotSWP_bottom, gotShowWindow; 333 HWND parent = GetParent( dlg); 334 int resize; 335 #define MAXNRCTRLS 30 336 static RECT ctrlrcs[MAXNRCTRLS]; 337 static int ctrlids[MAXNRCTRLS]; 338 static HWND ctrls[MAXNRCTRLS]; 339 static int nrctrls; 340 341 switch( msg) 342 { 343 case WM_INITDIALOG: 344 { 345 DWORD style; 346 347 index = ((OPENFILENAMEA*)lParam)->lCustData; 348 count = 0; 349 gotSWP_bottom = gotShowWindow = FALSE; 350 /* test style */ 351 style = GetWindowLongA( parent, GWL_STYLE); 352 if( resize_testcases[index].flags & OFN_ENABLESIZING) 353 if( !(style & WS_SIZEBOX)) { 354 win_skip( "OFN_ENABLESIZING flag not supported.\n"); 355 resizesupported = FALSE; 356 PostMessageA( parent, WM_COMMAND, IDCANCEL, 0); 357 } else 358 ok( style & WS_SIZEBOX, 359 "testid %d: dialog should have a WS_SIZEBOX style.\n", index); 360 else 361 ok( !(style & WS_SIZEBOX), 362 "testid %d: dialog should not have a WS_SIZEBOX style.\n", index); 363 break; 364 } 365 case WM_NOTIFY: 366 { 367 if(( (LPNMHDR)lParam)->code == CDN_FOLDERCHANGE){ 368 GetWindowRect( parent, &initrc); 369 if( (resize = resize_testcases[index].resize_folderchange)){ 370 MoveWindow( parent, initrc.left,initrc.top, initrc.right - initrc.left + resize, 371 initrc.bottom - initrc.top + resize, TRUE); 372 } 373 SetTimer( dlg, 0, 100, 0); 374 } 375 break; 376 } 377 case WM_TIMER: 378 { 379 if( count == 0){ 380 /* store the control rectangles */ 381 if( resize_testcases[index].testcontrols) { 382 HWND ctrl; 383 int i; 384 for( i = 0, ctrl = GetWindow( parent, GW_CHILD); 385 i < MAXNRCTRLS && ctrl; 386 i++, ctrl = GetWindow( ctrl, GW_HWNDNEXT)) { 387 ctrlids[i] = GetDlgCtrlID( ctrl); 388 GetWindowRect( ctrl, &ctrlrcs[i]); 389 MapWindowPoints( NULL, parent, (LPPOINT) &ctrlrcs[i], 2); 390 ctrls[i] = ctrl; 391 } 392 nrctrls = i; 393 } 394 if( (resize = resize_testcases[index].resize_timer1)){ 395 GetWindowRect( parent, &rc); 396 MoveWindow( parent, rc.left,rc.top, rc.right - rc.left + resize, 397 rc.bottom - rc.top + resize, TRUE); 398 } 399 } else if( count == 1){ 400 resize = resize_testcases[index].resize_check; 401 GetWindowRect( parent, &rc); 402 todo_wine_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); 409 } 410 if( resize_testcases[index].testcontrols) { 411 int i; 412 RECT rc; 413 for( i = 0; i < nrctrls; i++) { 414 GetWindowRect( ctrls[i], &rc); 415 MapWindowPoints( NULL, parent, (LPPOINT) &rc, 2); 416 switch( ctrlids[i]){ 417 418 /* test if RECT R1, moved and sized result in R2 */ 419 #define TESTRECTS( R1, R2, Mx, My, Sx, Sy) \ 420 ((R1).left + (Mx) ==(R2).left \ 421 &&(R1).top + (My) ==(R2).top \ 422 &&(R1).right + (Mx) + (Sx) == (R2).right \ 423 &&(R1).bottom + (My) + (Sy) ==(R2).bottom) 424 425 /* sized horizontal and moved vertical */ 426 case cmb1: 427 case edt1: 428 ok( TESTRECTS( ctrlrcs[i], rc, 0, 10, 10, 0), 429 "control id %03x should have sized horizontally and moved vertically, before %s after %s\n", 430 ctrlids[i], wine_dbgstr_rect( &ctrlrcs[i] ), 431 wine_dbgstr_rect( &rc )); 432 break; 433 /* sized horizontal and vertical */ 434 case lst2: 435 ok( TESTRECTS( ctrlrcs[i], rc, 0, 0, 10, 10), 436 "control id %03x should have sized horizontally and vertically, before %s after %s\n", 437 ctrlids[i], wine_dbgstr_rect( &ctrlrcs[i] ), 438 wine_dbgstr_rect( &rc )); 439 break; 440 /* moved horizontal and vertical */ 441 case IDCANCEL: 442 case pshHelp: 443 ok( TESTRECTS( ctrlrcs[i], rc, 10, 10, 0, 0), 444 "control id %03x should have moved horizontally and vertically, before %s after %s\n", 445 ctrlids[i], wine_dbgstr_rect( &ctrlrcs[i] ), 446 wine_dbgstr_rect( &rc )); 447 break; 448 /* moved vertically */ 449 case chx1: 450 case stc2: 451 case stc3: 452 ok( TESTRECTS( ctrlrcs[i], rc, 0, 10, 0, 0), 453 "control id %03x should have moved vertically, before %s after %s\n", 454 ctrlids[i], wine_dbgstr_rect( &ctrlrcs[i] ), 455 wine_dbgstr_rect( &rc )); 456 break; 457 /* resized horizontal */ 458 case cmb2: /* aka IDC_LOOKIN */ 459 ok( TESTRECTS( ctrlrcs[i], rc, 0, 0, 10, 0)|| 460 TESTRECTS( ctrlrcs[i], rc, 0, 0, 0, 0), /* Vista and higher */ 461 "control id %03x should have resized horizontally, before %s after %s\n", 462 ctrlids[i], wine_dbgstr_rect( &ctrlrcs[i] ), 463 wine_dbgstr_rect( &rc )); 464 break; 465 /* non moving non sizing controls */ 466 case stc4: 467 ok( TESTRECTS( rc, ctrlrcs[i], 0, 0, 0, 0), 468 "control id %03x was moved/resized, before %s after %s\n", 469 ctrlids[i], wine_dbgstr_rect( &ctrlrcs[i] ), 470 wine_dbgstr_rect( &rc )); 471 break; 472 /* todo_wine: non moving non sizing controls */ 473 case lst1: 474 todo_wine 475 ok( TESTRECTS( rc, ctrlrcs[i], 0, 0, 0, 0), 476 "control id %03x was moved/resized, before %s after %s\n", 477 ctrlids[i], wine_dbgstr_rect( &ctrlrcs[i] ), 478 wine_dbgstr_rect( &rc )); 479 break; 480 /* don't test: id is not unique */ 481 case IDOK: 482 case stc1: 483 case 0: 484 case -1: 485 break; 486 default: 487 trace("untested control id %03x before %s after %s\n", 488 ctrlids[i], wine_dbgstr_rect( &ctrlrcs[i] ), 489 wine_dbgstr_rect( &rc )); 490 #undef TESTRECTS 491 #undef MAXNRCTRLS 492 } 493 } 494 } 495 KillTimer( dlg, 0); 496 PostMessageA( parent, WM_COMMAND, IDCANCEL, 0); 497 } 498 count++; 499 } 500 break; 501 case WM_WINDOWPOSCHANGING: 502 { 503 WINDOWPOS *pwp = (WINDOWPOS *)lParam; 504 if( !index && pwp->hwndInsertAfter == HWND_BOTTOM){ 505 gotSWP_bottom = TRUE; 506 ok(!gotShowWindow, "The WM_WINDOWPOSCHANGING message came after a WM_SHOWWINDOW message\n"); 507 } 508 } 509 break; 510 case WM_SHOWWINDOW: 511 { 512 if( !index){ 513 gotShowWindow = TRUE; 514 ok(gotSWP_bottom, "No WM_WINDOWPOSCHANGING message came before a WM_SHOWWINDOW message\n"); 515 } 516 } 517 break; 518 } 519 return 0; 520 } 521 522 static void test_resize(void) 523 { 524 OPENFILENAMEA ofn = { OPENFILENAME_SIZE_VERSION_400A }; 525 char filename[1024] = {0}; 526 DWORD ret; 527 int i; 528 529 ofn.lpstrFile = filename; 530 ofn.nMaxFile = 1024; 531 ofn.lpfnHook = resize_template_hook; 532 ofn.hInstance = GetModuleHandleA(NULL); 533 ofn.lpTemplateName = "template_sz"; 534 for( i = 0; resize_testcases[i].flags != 0xffffffff; i++) { 535 ofn.lCustData = i; 536 ofn.Flags = resize_testcases[i].flags | 537 OFN_ENABLEHOOK | OFN_EXPLORER| OFN_ENABLETEMPLATE | OFN_SHOWHELP ; 538 ret = GetOpenFileNameA(&ofn); 539 ok(!ret, "GetOpenFileName returned %#x\n", ret); 540 ret = CommDlgExtendedError(); 541 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 542 } 543 } 544 545 /* test cases for control message IDOK */ 546 /* Show case for bug #19079 */ 547 typedef struct { 548 int retval; /* return code of the message handler */ 549 BOOL setmsgresult; /* set the result in the DWLP_MSGRESULT */ 550 BOOL usemsgokstr; /* use the FILEOKSTRING message instead of WM_NOTIFY:CDN_FILEOK */ 551 BOOL do_subclass; /* subclass the dialog hook procedure */ 552 BOOL expclose; /* is the dialog expected to close ? */ 553 BOOL actclose; /* has the dialog actually closed ? */ 554 } ok_wndproc_testcase; 555 556 static ok_wndproc_testcase ok_testcases[] = { 557 { 0, FALSE, FALSE, FALSE, TRUE}, 558 { 0, TRUE, FALSE, FALSE, TRUE}, 559 { 0, FALSE, FALSE, TRUE, TRUE}, 560 { 0, TRUE, FALSE, TRUE, TRUE}, 561 { 1, FALSE, FALSE, FALSE, TRUE}, 562 { 1, TRUE, FALSE, FALSE, FALSE}, 563 { 1, FALSE, FALSE, TRUE, FALSE}, 564 { 1, TRUE, FALSE, TRUE, FALSE}, 565 /* FILEOKSTRING tests */ 566 { 1, TRUE, TRUE, FALSE, FALSE}, 567 { 1, FALSE, TRUE, TRUE, FALSE}, 568 /* mark the end */ 569 { -1 } 570 }; 571 572 /* test_ok_wndproc can be used as hook procedure or a subclass 573 * window proc for the file dialog */ 574 static UINT_PTR WINAPI test_ok_wndproc(HWND dlg, UINT msg, WPARAM wParam, LPARAM lParam) 575 { 576 HWND parent = GetParent( dlg); 577 static ok_wndproc_testcase *testcase = NULL; 578 static UINT msgFILEOKSTRING; 579 if (msg == WM_INITDIALOG) 580 { 581 testcase = (ok_wndproc_testcase*)((OPENFILENAMEA*)lParam)->lCustData; 582 testcase->actclose = TRUE; 583 msgFILEOKSTRING = RegisterWindowMessageA( FILEOKSTRINGA); 584 } 585 if( msg == WM_NOTIFY) { 586 if(((LPNMHDR)lParam)->code == CDN_FOLDERCHANGE) { 587 SetTimer( dlg, 0, 100, 0); 588 PostMessageA( parent, WM_COMMAND, IDOK, 0); 589 return FALSE; 590 } else if(((LPNMHDR)lParam)->code == CDN_FILEOK) { 591 if( testcase->usemsgokstr) 592 return FALSE; 593 if( testcase->setmsgresult) 594 SetWindowLongPtrA( dlg, DWLP_MSGRESULT, testcase->retval); 595 return testcase->retval; 596 } 597 } 598 if( msg == msgFILEOKSTRING) { 599 if( !testcase->usemsgokstr) 600 return FALSE; 601 if( testcase->setmsgresult) 602 SetWindowLongPtrA( dlg, DWLP_MSGRESULT, testcase->retval); 603 return testcase->retval; 604 } 605 if( msg == WM_TIMER) { 606 /* the dialog did not close automatically */ 607 testcase->actclose = FALSE; 608 KillTimer( dlg, 0); 609 PostMessageA( parent, WM_COMMAND, IDCANCEL, 0); 610 return FALSE; 611 } 612 if( testcase && testcase->do_subclass) 613 return DefWindowProcA( dlg, msg, wParam, lParam); 614 return FALSE; 615 } 616 617 static UINT_PTR WINAPI ok_template_hook(HWND dlg, UINT msg, WPARAM wParam, LPARAM lParam) 618 { 619 if (msg == WM_SETFONT) 620 SetWindowLongPtrA( dlg, GWLP_WNDPROC, (LONG_PTR) test_ok_wndproc); 621 return FALSE; 622 } 623 624 static void test_ok(void) 625 { 626 OPENFILENAMEA ofn = { OPENFILENAME_SIZE_VERSION_400A }; 627 char filename[1024] = {0}; 628 char tmpfilename[ MAX_PATH]; 629 char curdir[MAX_PATH]; 630 int i; 631 DWORD ret; 632 BOOL cdret; 633 634 cdret = GetCurrentDirectoryA(sizeof(curdir), curdir); 635 ok(cdret, "Failed to get current dir err %d\n", GetLastError()); 636 if (!GetTempFileNameA(".", "txt", 0, tmpfilename)) { 637 skip("Failed to create a temporary file name\n"); 638 return; 639 } 640 ofn.lpstrFile = filename; 641 ofn.nMaxFile = 1024; 642 ofn.hInstance = GetModuleHandleA(NULL); 643 ofn.lpTemplateName = "template1"; 644 ofn.Flags = OFN_ENABLEHOOK | OFN_EXPLORER| OFN_ENABLETEMPLATE ; 645 for( i = 0; ok_testcases[i].retval != -1; i++) { 646 strcpy( filename, tmpfilename); 647 ofn.lCustData = (LPARAM)(ok_testcases + i); 648 ofn.lpfnHook = ok_testcases[i].do_subclass ? ok_template_hook : test_ok_wndproc; 649 ret = GetOpenFileNameA(&ofn); 650 ok( ok_testcases[i].expclose == ok_testcases[i].actclose, 651 "testid %d: Open File dialog should %shave closed.\n", i, 652 ok_testcases[i].expclose ? "" : "NOT "); 653 ok(ret == ok_testcases[i].expclose, "testid %d: GetOpenFileName returned %#x\n", i, ret); 654 ret = CommDlgExtendedError(); 655 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 656 cdret = SetCurrentDirectoryA(curdir); 657 ok(cdret, "Failed to restore current dir err %d\n", GetLastError()); 658 } 659 ret = DeleteFileA( tmpfilename); 660 ok( ret, "Failed to delete temporary file %s err %d\n", tmpfilename, GetLastError()); 661 } 662 663 /* test arranging with a custom template */ 664 typedef struct { 665 int x, y; /* left, top coordinates */ 666 int cx, cy; /* width and height */ 667 } posz; 668 static struct { 669 int nrcontrols; /* 0: no controls, 1: just the stc32 control 2: with button */ 670 posz poszDlg; 671 posz poszStc32; 672 posz poszBtn; 673 DWORD ofnflags; 674 } arrange_tests[] = { 675 /* do not change the first two cases: used to get the uncustomized sizes */ 676 { 0, {0},{0},{0},0 }, 677 { 0, {0},{0},{0}, OFN_SHOWHELP}, 678 /* two tests with just a subdialog, no controls */ 679 { 0, {0, 0, 316, 76},{0},{0},0 }, 680 { 0, {0, 0, 100, 76},{0},{0}, OFN_SHOWHELP}, 681 /* now with a control with id stc32 */ 682 { 1, {0, 0, 316, 76} ,{0, 0, 204, 76,},{0},0 }, /* bug #17748*/ 683 { 1, {0, 0, 316, 76} ,{0, 0, 204, 76,},{0}, OFN_SHOWHELP}, /* bug #17748*/ 684 /* tests with size of the stc32 control higher or wider then the standard dialog */ 685 { 1, {0, 0, 316, 170} ,{0, 0, 204, 170,},{0},0 }, 686 { 1, {0, 0, 316, 165} ,{0, 0, 411, 165,},{0}, OFN_SHOWHELP }, 687 /* move the stc32 control around */ 688 { 1, {0, 0, 300, 100} ,{73, 17, 50, 50,},{0},0 }, 689 /* add control */ 690 { 2, {0, 0, 280, 100} ,{0, 0, 50, 50,},{300,20,30,30},0 }, 691 /* enable resizing should make the dialog bigger */ 692 { 0, {0},{0},{0}, OFN_SHOWHELP|OFN_ENABLESIZING}, 693 /* mark the end */ 694 { -1 } 695 }; 696 697 static UINT_PTR WINAPI template_hook_arrange(HWND dlgChild, UINT msg, WPARAM wParam, LPARAM lParam) 698 { 699 static int index, fixhelp; 700 static posz posz0[2]; 701 static RECT clrcParent, clrcChild, rcStc32; 702 static HWND hwndStc32; 703 HWND dlgParent; 704 705 dlgParent = GetParent( dlgChild); 706 if (msg == WM_INITDIALOG) { 707 index = ((OPENFILENAMEA*)lParam)->lCustData; 708 /* get the positions before rearrangement */ 709 GetClientRect( dlgParent, &clrcParent); 710 GetClientRect( dlgChild, &clrcChild); 711 hwndStc32 = GetDlgItem( dlgChild, stc32); 712 if( hwndStc32) GetWindowRect( hwndStc32, &rcStc32); 713 } 714 if (msg == WM_NOTIFY && ((LPNMHDR)lParam)->code == CDN_FOLDERCHANGE) { 715 RECT wrcParent; 716 717 GetWindowRect( dlgParent, &wrcParent); 718 /* the fist two "tests" just save the dialogs position, with and without 719 * help button */ 720 if( index == 0) { 721 posz0[0].x = wrcParent.left; 722 posz0[0].y = wrcParent.top; 723 posz0[0].cx = wrcParent.right - wrcParent.left; 724 posz0[0].cy = wrcParent.bottom - wrcParent.top; 725 } else if( index == 1) { 726 posz0[1].x = wrcParent.left; 727 posz0[1].y = wrcParent.top; 728 posz0[1].cx = wrcParent.right - wrcParent.left; 729 posz0[1].cy = wrcParent.bottom - wrcParent.top; 730 fixhelp = posz0[1].cy - posz0[0].cy; 731 } else { 732 /* the real tests */ 733 int withhelp; 734 int expectx, expecty; 735 DWORD style; 736 737 withhelp = (arrange_tests[index].ofnflags & OFN_SHOWHELP) != 0; 738 GetWindowRect( dlgParent, &wrcParent); 739 if( !hwndStc32) { 740 /* case with no custom subitem with stc32: 741 * default to all custom controls below the standard */ 742 expecty = posz0[withhelp].cy + clrcChild.bottom; 743 expectx = posz0[withhelp].cx; 744 } else { 745 /* special case: there is a control with id stc32 */ 746 /* expected height */ 747 expecty = posz0[withhelp].cy; 748 if( rcStc32.bottom - rcStc32.top + (withhelp ? 0 : fixhelp) > clrcParent.bottom) { 749 expecty += clrcChild.bottom - clrcParent.bottom; 750 if( !withhelp) expecty += fixhelp; 751 } 752 else 753 expecty += clrcChild.bottom - ( rcStc32.bottom - rcStc32.top) ; 754 /* expected width */ 755 expectx = posz0[withhelp].cx; 756 if( rcStc32.right - rcStc32.left > clrcParent.right) { 757 expectx += clrcChild.right - clrcParent.right; 758 } 759 else 760 expectx += clrcChild.right - ( rcStc32.right - rcStc32.left) ; 761 } 762 style = GetWindowLongA( dlgParent, GWL_STYLE); 763 if( !(style & WS_SIZEBOX)) { 764 /* without the OFN_ENABLESIZING flag */ 765 ok( wrcParent.bottom - wrcParent.top == expecty, 766 "Wrong height of dialog %d, expected %d\n", 767 wrcParent.bottom - wrcParent.top, expecty); 768 ok( wrcParent.right - wrcParent.left == expectx, 769 "Wrong width of dialog %d, expected %d\n", 770 wrcParent.right - wrcParent.left, expectx); 771 } else { 772 /* with the OFN_ENABLESIZING flag */ 773 ok( wrcParent.bottom - wrcParent.top > expecty, 774 "Wrong height of dialog %d, expected more than %d\n", 775 wrcParent.bottom - wrcParent.top, expecty); 776 ok( wrcParent.right - wrcParent.left > expectx, 777 "Wrong width of dialog %d, expected more than %d\n", 778 wrcParent.right - wrcParent.left, expectx); 779 } 780 781 } 782 PostMessageA( dlgParent, WM_COMMAND, IDCANCEL, 0); 783 } 784 return 0; 785 } 786 787 static void test_arrange(void) 788 { 789 OPENFILENAMEA ofn = {0}; 790 char filename[1024] = {0}; 791 DWORD ret; 792 HRSRC hRes; 793 HANDLE hDlgTmpl; 794 LPBYTE pv; 795 DLGTEMPLATE *template; 796 DLGITEMTEMPLATE *itemtemplateStc32, *itemtemplateBtn; 797 int i; 798 799 /* load subdialog template into memory */ 800 hRes = FindResourceA( GetModuleHandleA(NULL), "template_stc32", (LPSTR)RT_DIALOG); 801 hDlgTmpl = LoadResource( GetModuleHandleA(NULL), hRes ); 802 /* get pointers to the structures for the dialog and the controls */ 803 pv = LockResource( hDlgTmpl ); 804 template = (DLGTEMPLATE*)pv; 805 if( template->x != 11111) { 806 win_skip("could not find the dialog template\n"); 807 return; 808 } 809 /* skip dialog template, menu, class and title */ 810 pv += sizeof(DLGTEMPLATE); 811 pv += 3 * sizeof(WORD); 812 /* skip font info */ 813 while( *(WORD*)pv) 814 pv += sizeof(WORD); 815 pv += sizeof(WORD); 816 /* align on 32 bit boundaries */ 817 pv = (LPBYTE)(((UINT_PTR)pv + 3 ) & ~3); 818 itemtemplateStc32 = (DLGITEMTEMPLATE*)pv; 819 if( itemtemplateStc32->x != 22222) { 820 win_skip("could not find the first item template\n"); 821 return; 822 } 823 /* skip itemtemplate, class, title and creation data */ 824 pv += sizeof(DLGITEMTEMPLATE); 825 pv += 4 * sizeof(WORD); 826 /* align on 32 bit boundaries */ 827 pv = (LPBYTE)(((UINT_PTR)pv + 3 ) & ~3); 828 itemtemplateBtn = (DLGITEMTEMPLATE*)pv; 829 if( itemtemplateBtn->x != 12345) { 830 win_skip("could not find the second item template\n"); 831 return; 832 } 833 834 ofn.lStructSize = OPENFILENAME_SIZE_VERSION_400A; 835 ofn.lpstrFile = filename; 836 ofn.nMaxFile = 1024; 837 ofn.lpfnHook = template_hook_arrange; 838 ofn.hInstance = hDlgTmpl; 839 ofn.lpstrFilter="text\0*.txt\0All\0*\0\0"; 840 for( i = 0; arrange_tests[i].nrcontrols != -1; i++) { 841 ofn.lCustData = i; 842 ofn.Flags = OFN_ENABLEHOOK | OFN_EXPLORER| OFN_ENABLETEMPLATEHANDLE | OFN_HIDEREADONLY | 843 arrange_tests[i].ofnflags; 844 template->cdit = arrange_tests[i].nrcontrols; 845 template->x = arrange_tests[i].poszDlg.x; 846 template->y = arrange_tests[i].poszDlg.y; 847 template->cx = arrange_tests[i].poszDlg.cx; 848 template->cy = arrange_tests[i].poszDlg.cy; 849 itemtemplateStc32->x = arrange_tests[i].poszStc32.x; 850 itemtemplateStc32->y = arrange_tests[i].poszStc32.y; 851 itemtemplateStc32->cx = arrange_tests[i].poszStc32.cx; 852 itemtemplateStc32->cy = arrange_tests[i].poszStc32.cy; 853 itemtemplateBtn->x = arrange_tests[i].poszBtn.x; 854 itemtemplateBtn->y = arrange_tests[i].poszBtn.y; 855 itemtemplateBtn->cx = arrange_tests[i].poszBtn.cx; 856 itemtemplateBtn->cy = arrange_tests[i].poszBtn.cy; 857 ret = GetOpenFileNameA(&ofn); 858 ok(!ret, "GetOpenFileNameA returned %#x\n", ret); 859 ret = CommDlgExtendedError(); 860 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 861 } 862 } 863 864 static CHAR SYSDIR[MAX_PATH]; 865 866 static UINT_PTR CALLBACK path_hook_proc( HWND hDlg, UINT msg, WPARAM wParam, LPARAM lParam) 867 { 868 LPNMHDR nmh; 869 870 if( msg == WM_NOTIFY) 871 { 872 nmh = (LPNMHDR) lParam; 873 if( nmh->code == CDN_INITDONE) 874 { 875 PostMessageA( GetParent(hDlg), WM_COMMAND, IDCANCEL, FALSE); 876 } 877 else if ( nmh->code == CDN_FOLDERCHANGE) 878 { 879 char buf[1024]; 880 int ret; 881 882 memset(buf, 0x66, sizeof(buf)); 883 ret = SendMessageA( GetParent(hDlg), CDM_GETFOLDERPATH, sizeof(buf), (LPARAM)buf); 884 ok(!lstrcmpiA(SYSDIR, buf), "Expected '%s', got '%s'\n", SYSDIR, buf); 885 ok(lstrlenA(SYSDIR) + 1 == ret, "Expected %d, got %d\n", lstrlenA(SYSDIR) + 1, ret); 886 } 887 } 888 889 return 0; 890 } 891 892 static void test_getfolderpath(void) 893 { 894 OPENFILENAMEA ofn; 895 BOOL result; 896 char szFileName[MAX_PATH] = ""; 897 char szInitialDir[MAX_PATH]; 898 899 /* We need to pick a different directory as the other tests because of new 900 * Windows 7 behavior. 901 */ 902 GetSystemDirectoryA(szInitialDir, MAX_PATH); 903 lstrcpyA(SYSDIR, szInitialDir); 904 905 ZeroMemory(&ofn, sizeof(ofn)); 906 907 ofn.lStructSize = OPENFILENAME_SIZE_VERSION_400A; 908 ofn.hwndOwner = NULL; 909 ofn.lpstrFilter = "Text Files (*.txt)\0*.txt\0All Files (*.*)\0*.*\0"; 910 ofn.lpstrFile = szFileName; 911 ofn.nMaxFile = MAX_PATH; 912 ofn.Flags = OFN_EXPLORER | OFN_FILEMUSTEXIST | OFN_HIDEREADONLY | OFN_ENABLEHOOK; 913 ofn.lpstrDefExt = "txt"; 914 ofn.lpfnHook = path_hook_proc; 915 ofn.lpstrInitialDir = szInitialDir; 916 917 result = GetOpenFileNameA(&ofn); 918 ok(FALSE == result, "expected FALSE, got %d\n", result); 919 ok(0 == CommDlgExtendedError(), "expected 0, got %d\n", 920 CommDlgExtendedError()); 921 922 result = GetSaveFileNameA(&ofn); 923 ok(FALSE == result, "expected FALSE, got %d\n", result); 924 ok(0 == CommDlgExtendedError(), "expected 0, got %d\n", 925 CommDlgExtendedError()); 926 } 927 928 static void test_resizable2(void) 929 { 930 OPENFILENAMEA ofn = {0}; 931 char filename[1024] = "pls press Enter if sizable, Esc otherwise"; 932 DWORD ret; 933 934 /* interactive because there is no hook function */ 935 if( !winetest_interactive) { 936 skip( "some interactive resizable dialog tests (set WINETEST_INTERACTIVE=1)\n"); 937 return; 938 } 939 ofn.lStructSize = OPENFILENAME_SIZE_VERSION_400A; 940 ofn.lpstrFile = filename; 941 ofn.nMaxFile = 1024; 942 ofn.lpfnHook = NULL; 943 ofn.hInstance = GetModuleHandleA(NULL); 944 ofn.lpTemplateName = "template1"; 945 ofn.Flags = OFN_EXPLORER; 946 #define ISSIZABLE TRUE 947 ret = GetOpenFileNameA(&ofn); 948 ok( ret == ISSIZABLE, "File Dialog should have been sizable\n"); 949 ret = CommDlgExtendedError(); 950 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 951 ofn.Flags = OFN_EXPLORER | OFN_ENABLETEMPLATE; 952 ret = GetOpenFileNameA(&ofn); 953 ok( ret != ISSIZABLE, "File Dialog should NOT have been sizable\n"); 954 ret = CommDlgExtendedError(); 955 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 956 ofn.Flags = OFN_EXPLORER | OFN_ENABLETEMPLATEHANDLE; 957 ofn.hInstance = LoadResource( GetModuleHandleA(NULL), FindResourceA( GetModuleHandleA(NULL), "template1", (LPSTR)RT_DIALOG)); 958 ofn.lpTemplateName = NULL; 959 ret = GetOpenFileNameA(&ofn); 960 ok( ret != ISSIZABLE, "File Dialog should NOT have been sizable\n"); 961 ret = CommDlgExtendedError(); 962 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 963 ofn.Flags = OFN_EXPLORER | OFN_ENABLEHOOK; 964 ret = GetOpenFileNameA(&ofn); 965 ok( ret != ISSIZABLE, "File Dialog should NOT have been sizable\n"); 966 ret = CommDlgExtendedError(); 967 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 968 #undef ISSIZABLE 969 } 970 971 static void test_mru(void) 972 { 973 ok_wndproc_testcase testcase = {0}; 974 OPENFILENAMEA ofn = { OPENFILENAME_SIZE_VERSION_400A }; 975 const char *test_dir_name = "C:\\mru_test"; 976 const char *test_file_name = "test.txt"; 977 const char *test_full_path = "C:\\mru_test\\test.txt"; 978 char filename_buf[MAX_PATH]; 979 DWORD ret; 980 981 ofn.lpstrFile = filename_buf; 982 ofn.nMaxFile = sizeof(filename_buf); 983 ofn.lpTemplateName = "template1"; 984 ofn.hInstance = GetModuleHandleA(NULL); 985 ofn.Flags = OFN_ENABLEHOOK | OFN_EXPLORER | OFN_ENABLETEMPLATE | OFN_NOCHANGEDIR; 986 ofn.lCustData = (LPARAM)&testcase; 987 ofn.lpfnHook = test_ok_wndproc; 988 989 SetLastError(0xdeadbeef); 990 ret = CreateDirectoryA(test_dir_name, NULL); 991 ok(ret == TRUE, "CreateDirectoryA should have succeeded: %d\n", GetLastError()); 992 993 /* "teach" comdlg32 about this directory */ 994 strcpy(filename_buf, test_full_path); 995 SetLastError(0xdeadbeef); 996 ret = GetOpenFileNameA(&ofn); 997 ok(ret, "GetOpenFileNameA should have succeeded: %d\n", GetLastError()); 998 ret = CommDlgExtendedError(); 999 ok(!ret, "CommDlgExtendedError returned %x\n", ret); 1000 ok(testcase.actclose, "Open File dialog should have closed.\n"); 1001 ok(!strcmp(ofn.lpstrFile, test_full_path), "Expected to get %s, got %s\n", test_full_path, ofn.lpstrFile); 1002 1003 /* get a filename without a full path. it should return the file in 1004 * test_dir_name, not in the CWD */ 1005 strcpy(filename_buf, test_file_name); 1006 SetLastError(0xdeadbeef); 1007 ret = GetOpenFileNameA(&ofn); 1008 ok(ret, "GetOpenFileNameA should have succeeded: %d\n", GetLastError()); 1009 ret = CommDlgExtendedError(); 1010 ok(!ret, "CommDlgExtendedError returned %x\n", ret); 1011 ok(testcase.actclose, "Open File dialog should have closed.\n"); 1012 if(strcmp(ofn.lpstrFile, test_full_path) != 0) 1013 win_skip("Platform doesn't save MRU data\n"); 1014 1015 SetLastError(0xdeadbeef); 1016 ret = RemoveDirectoryA(test_dir_name); 1017 ok(ret == TRUE, "RemoveDirectoryA should have succeeded: %d\n", GetLastError()); 1018 } 1019 1020 static UINT_PTR WINAPI test_extension_wndproc(HWND dlg, UINT msg, WPARAM wParam, LPARAM lParam) 1021 { 1022 HWND parent = GetParent( dlg); 1023 if( msg == WM_NOTIFY) { 1024 SetTimer( dlg, 0, 1000, 0); 1025 PostMessageA( parent, WM_COMMAND, IDOK, 0); 1026 } 1027 if( msg == WM_TIMER) { 1028 /* the dialog did not close automatically */ 1029 KillTimer( dlg, 0); 1030 PostMessageA( parent, WM_COMMAND, IDCANCEL, 0); 1031 } 1032 return FALSE; 1033 } 1034 1035 #define ARRAY_SIZE(a) (sizeof(a)/sizeof((a)[0])) 1036 1037 static void test_extension_helper(OPENFILENAMEA* ofn, const char *filter, 1038 const char *expected_filename) 1039 { 1040 char *filename_ptr; 1041 DWORD ret; 1042 BOOL boolret; 1043 1044 strcpy(ofn->lpstrFile, "deadbeef"); 1045 ofn->lpstrFilter = filter; 1046 1047 boolret = GetSaveFileNameA(ofn); 1048 ok(boolret, "%s: expected TRUE\n", filter); 1049 1050 ret = CommDlgExtendedError(); 1051 ok(!ret, "%s: CommDlgExtendedError returned %#x\n", filter, ret); 1052 1053 filename_ptr = ofn->lpstrFile + ofn->nFileOffset; 1054 ok(strcmp(filename_ptr, expected_filename) == 0, 1055 "%s: Filename is %s, expected %s\n", filter, filename_ptr, expected_filename); 1056 } 1057 1058 static void test_extension(void) 1059 { 1060 OPENFILENAMEA ofn = { OPENFILENAME_SIZE_VERSION_400A }; 1061 char filename[1024] = {0}; 1062 char curdir[MAX_PATH]; 1063 unsigned int i; 1064 BOOL boolret; 1065 1066 const char *defext_concrete_filters[] = { 1067 "TestFilter (*.abc)\0*.abc\0", 1068 "TestFilter (*.abc;)\0*.abc;\0", 1069 "TestFilter (*.abc;*.def)\0*.abc;*.def\0", 1070 }; 1071 1072 const char *defext_wildcard_filters[] = { 1073 "TestFilter (*.pt*)\0*.pt*\0", 1074 "TestFilter (*.pt*;*.abc)\0*.pt*;*.abc\0", 1075 "TestFilter (*.ab?)\0*.ab?\0", 1076 "TestFilter (*.*)\0*.*\0", 1077 "TestFilter (*sav)\0*sav\0", 1078 NULL /* is a test, not an endmark! */ 1079 }; 1080 1081 boolret = GetCurrentDirectoryA(sizeof(curdir), curdir); 1082 ok(boolret, "Failed to get current dir err %d\n", GetLastError()); 1083 1084 ofn.hwndOwner = NULL; 1085 ofn.lpstrFile = filename; 1086 ofn.nMaxFile = MAX_PATH; 1087 ofn.Flags = OFN_EXPLORER | OFN_ENABLEHOOK; 1088 ofn.lpstrInitialDir = curdir; 1089 ofn.lpfnHook = test_extension_wndproc; 1090 ofn.nFileExtension = 0; 1091 1092 ofn.lpstrDefExt = NULL; 1093 1094 /* Without lpstrDefExt, append no extension */ 1095 test_extension_helper(&ofn, "TestFilter (*.abc) lpstrDefExt=NULL\0*.abc\0", "deadbeef"); 1096 test_extension_helper(&ofn, "TestFilter (*.ab?) lpstrDefExt=NULL\0*.ab?\0", "deadbeef"); 1097 1098 ofn.lpstrDefExt = ""; 1099 1100 /* If lpstrDefExt="" and the filter has a concrete extension, append it */ 1101 test_extension_helper(&ofn, "TestFilter (*.abc) lpstrDefExt=\"\"\0*.abc\0", "deadbeef.abc"); 1102 1103 /* If lpstrDefExt="" and the filter has a wildcard extension, do nothing */ 1104 test_extension_helper(&ofn, "TestFilter (*.ab?) lpstrDefExt=\"\"\0*.ab?\0", "deadbeef"); 1105 1106 ofn.lpstrDefExt = "xyz"; 1107 1108 /* Append concrete extensions from filters */ 1109 for (i = 0; i < ARRAY_SIZE(defext_concrete_filters); i++) { 1110 test_extension_helper(&ofn, defext_concrete_filters[i], "deadbeef.abc"); 1111 } 1112 1113 /* Append nothing from this filter */ 1114 test_extension_helper(&ofn, "TestFilter (*.)\0*.\0", "deadbeef"); 1115 1116 /* Ignore wildcard extensions in filters */ 1117 for (i = 0; i < ARRAY_SIZE(defext_wildcard_filters); i++) { 1118 test_extension_helper(&ofn, defext_wildcard_filters[i], "deadbeef.xyz"); 1119 } 1120 1121 /* Append valid extensions consisting of multiple parts */ 1122 test_extension_helper(&ofn, "TestFilter (*.abc.def)\0*.abc.def\0", "deadbeef.abc.def"); 1123 test_extension_helper(&ofn, "TestFilter (.abc.def)\0.abc.def\0", "deadbeef.abc.def"); 1124 test_extension_helper(&ofn, "TestFilter (*.*.def)\0*.*.def\0", "deadbeef.xyz"); 1125 } 1126 1127 #undef ARRAY_SIZE 1128 1129 1130 static BOOL WINAPI test_null_enum(HWND hwnd, LPARAM lParam) 1131 { 1132 /* Find the textbox and send a filename so IDOK will work. 1133 If the file textbox is empty IDOK will be ignored */ 1134 CHAR className[20]; 1135 if(GetClassNameA(hwnd, className, sizeof(className)) > 0 && !strcmp("Edit",className)) 1136 { 1137 SetWindowTextA(hwnd, "testfile"); 1138 return FALSE; /* break window enumeration */ 1139 } 1140 return TRUE; 1141 } 1142 1143 static UINT_PTR WINAPI test_null_wndproc(HWND dlg, UINT msg, WPARAM wParam, LPARAM lParam) 1144 { 1145 HWND parent = GetParent( dlg); 1146 if( msg == WM_NOTIFY) { 1147 SetTimer( dlg, 0, 100, 0); 1148 SetTimer( dlg, 1, 1000, 0); 1149 EnumChildWindows( parent, test_null_enum, 0); 1150 } 1151 if( msg == WM_TIMER) { 1152 if(!wParam) 1153 PostMessageA( parent, WM_COMMAND, IDOK, 0); 1154 else { 1155 /* the dialog did not close automatically */ 1156 KillTimer( dlg, 0); 1157 PostMessageA( parent, WM_COMMAND, IDCANCEL, 0); 1158 } 1159 } 1160 return FALSE; 1161 } 1162 1163 static void test_null_filename(void) 1164 { 1165 OPENFILENAMEA ofnA = {0}; 1166 OPENFILENAMEW ofnW = {0}; 1167 WCHAR filterW[] = {'t','e','x','t','\0','*','.','t','x','t','\0', 1168 'A','l','l','\0','*','\0','\0'}; 1169 DWORD ret; 1170 1171 ofnA.lStructSize = OPENFILENAME_SIZE_VERSION_400A; 1172 ofnA.lpstrFile = NULL; 1173 ofnA.nMaxFile = 0; 1174 ofnA.nFileOffset = 0xdead; 1175 ofnA.nFileExtension = 0xbeef; 1176 ofnA.lpfnHook = test_null_wndproc; 1177 ofnA.Flags = OFN_ENABLEHOOK | OFN_EXPLORER; 1178 ofnA.hInstance = GetModuleHandleA(NULL); 1179 ofnA.lpstrFilter = "text\0*.txt\0All\0*\0\0"; 1180 ofnA.lpstrDefExt = NULL; 1181 ret = GetOpenFileNameA(&ofnA); 1182 todo_wine ok(ret, "GetOpenFileNameA returned %#x\n", ret); 1183 ret = CommDlgExtendedError(); 1184 todo_wine ok(!ret, "CommDlgExtendedError returned %#x, should be 0\n", ret); 1185 1186 todo_wine ok(ofnA.nFileOffset != 0xdead, "ofnA.nFileOffset is 0xdead\n"); 1187 todo_wine ok(ofnA.nFileExtension != 0xbeef, "ofnA.nFileExtension is 0xbeef\n"); 1188 1189 ofnA.lpstrFile = NULL; 1190 ofnA.nMaxFile = 1024; /* bogus input - lpstrFile = NULL but fake 1024 bytes available */ 1191 ofnA.nFileOffset = 0xdead; 1192 ofnA.nFileExtension = 0xbeef; 1193 ret = GetOpenFileNameA(&ofnA); 1194 ok(ret, "GetOpenFileNameA returned %#x\n", ret); 1195 ret = CommDlgExtendedError(); 1196 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 1197 1198 ok(ofnA.nFileOffset != 0xdead, "ofnA.nFileOffset is 0xdead\n"); 1199 ok(ofnA.nFileExtension == 0, "ofnA.nFileExtension is 0x%x, should be 0\n", ofnA.nFileExtension); 1200 1201 /* unicode tests */ 1202 ofnW.lStructSize = OPENFILENAME_SIZE_VERSION_400W; 1203 ofnW.lpstrFile = NULL; 1204 ofnW.nMaxFile = 0; 1205 ofnW.nFileOffset = 0xdead; 1206 ofnW.nFileExtension = 0xbeef; 1207 ofnW.lpfnHook = test_null_wndproc; 1208 ofnW.Flags = OFN_ENABLEHOOK | OFN_EXPLORER; 1209 ofnW.hInstance = GetModuleHandleW(NULL); 1210 ofnW.lpstrFilter = filterW; 1211 ofnW.lpstrDefExt = NULL; 1212 ret = GetOpenFileNameW(&ofnW); 1213 todo_wine ok(ret, "GetOpenFileNameW returned %#x\n", ret); 1214 ret = CommDlgExtendedError(); 1215 todo_wine ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 1216 1217 todo_wine ok(ofnW.nFileOffset != 0xdead, "ofnW.nFileOffset is 0xdead\n"); 1218 todo_wine ok(ofnW.nFileExtension != 0xbeef, "ofnW.nFileExtension is 0xbeef\n"); 1219 1220 ofnW.lpstrFile = NULL; 1221 ofnW.nMaxFile = 1024; /* bogus input - lpstrFile = NULL but fake 1024 bytes available */ 1222 ofnW.nFileOffset = 0xdead; 1223 ofnW.nFileExtension = 0xbeef; 1224 ret = GetOpenFileNameW(&ofnW); 1225 ok(ret, "GetOpenFileNameA returned %#x\n", ret); 1226 ret = CommDlgExtendedError(); 1227 ok(!ret, "CommDlgExtendedError returned %#x\n", ret); 1228 1229 ok(ofnW.nFileOffset != 0xdead, "ofnW.nFileOffset is 0xdead\n"); 1230 ok(ofnW.nFileExtension == 0, "ofnW.nFileExtension is 0x%x, should be 0\n", ofnW.nFileExtension); 1231 } 1232 1233 static void test_directory_filename(void) 1234 { 1235 OPENFILENAMEA ofnA = {0}; 1236 OPENFILENAMEW ofnW = {0}; 1237 WCHAR filterW[] = {'t','e','x','t','\0','*','.','t','x','t','\0', 1238 'A','l','l','\0','*','\0','\0'}; 1239 char szInitialDir[MAX_PATH] = {0}; 1240 WCHAR szInitialDirW[MAX_PATH] = {0}; 1241 DWORD ret; 1242 1243 GetWindowsDirectoryA(szInitialDir, MAX_PATH); 1244 GetWindowsDirectoryW(szInitialDirW, MAX_PATH); 1245 1246 szInitialDir[strlen(szInitialDir)] = '\\'; 1247 szInitialDirW[lstrlenW(szInitialDirW)] = '\\'; 1248 1249 ofnA.lStructSize = OPENFILENAME_SIZE_VERSION_400A; 1250 ofnA.lpstrFile = szInitialDir; 1251 ofnA.nMaxFile = MAX_PATH; 1252 ofnA.lpfnHook = test_null_wndproc; 1253 ofnA.Flags = OFN_ENABLEHOOK | OFN_EXPLORER; 1254 ofnA.hInstance = GetModuleHandleA(NULL); 1255 ofnA.lpstrFilter = "text\0*.txt\0All\0*\0\0"; 1256 ofnA.lpstrDefExt = NULL; 1257 ret = GetOpenFileNameA(&ofnA); 1258 todo_wine ok(!ret, "GetOpenFileNameA returned %#x\n", ret); 1259 1260 /* unicode tests */ 1261 ofnW.lStructSize = OPENFILENAME_SIZE_VERSION_400W; 1262 ofnW.lpstrFile = szInitialDirW; 1263 ofnW.nMaxFile = MAX_PATH; 1264 ofnW.lpfnHook = test_null_wndproc; 1265 ofnW.Flags = OFN_ENABLEHOOK | OFN_EXPLORER; 1266 ofnW.hInstance = GetModuleHandleW(NULL); 1267 ofnW.lpstrFilter = filterW; 1268 ofnW.lpstrDefExt = NULL; 1269 ret = GetOpenFileNameW(&ofnW); 1270 todo_wine ok(!ret, "GetOpenFileNameW returned %#x\n", ret); 1271 } 1272 1273 static UINT_PTR WINAPI test_ole_init_wndproc(HWND dlg, UINT msg, WPARAM wParam, LPARAM lParam) 1274 { 1275 HRESULT hr; 1276 1277 hr = OleInitialize(NULL); 1278 ok(hr == S_FALSE, "OleInitialize() returned %#x\n", hr); 1279 OleUninitialize(); 1280 1281 if (msg == WM_NOTIFY) 1282 PostMessageA(GetParent(dlg), WM_COMMAND, IDCANCEL, 0); 1283 return FALSE; 1284 } 1285 1286 static LRESULT CALLBACK hook_proc(int code, WPARAM wp, LPARAM lp) 1287 { 1288 static BOOL first_dlg = TRUE; 1289 HRESULT hr; 1290 1291 if (code == HCBT_CREATEWND) 1292 { 1293 CBT_CREATEWNDW *c = (CBT_CREATEWNDW *)lp; 1294 1295 if (c->lpcs->lpszClass == (LPWSTR)WC_DIALOG) 1296 { 1297 /* OleInitialize() creates a window for the main apartment. Since 1298 * Vista OleInitialize() is called before the file dialog is 1299 * created. SimCity 2000 expects that the first window created 1300 * after GetOpenFileA() is a file dialog window. Mark Vista+ 1301 * behavior as broken. */ 1302 hr = OleInitialize(NULL); 1303 ok((first_dlg ? hr == S_OK : hr == S_FALSE) 1304 || broken(first_dlg && hr == S_FALSE), 1305 "OleInitialize() returned %#x (first dialog %#x)\n", hr, first_dlg); 1306 OleUninitialize(); 1307 first_dlg = FALSE; 1308 } 1309 } 1310 1311 return CallNextHookEx(NULL, code, wp, lp); 1312 } 1313 1314 static void test_ole_initialization(void) 1315 { 1316 char file[MAX_PATH] = {0}; 1317 OPENFILENAMEA ofn = {0}; 1318 HRESULT hr; 1319 HHOOK hook; 1320 BOOL ret; 1321 1322 hook = SetWindowsHookExW(WH_CBT, hook_proc, NULL, GetCurrentThreadId()); 1323 1324 ofn.lStructSize = OPENFILENAME_SIZE_VERSION_400A; 1325 ofn.lpstrFile = file; 1326 ofn.nMaxFile = MAX_PATH; 1327 ofn.lpfnHook = test_ole_init_wndproc; 1328 ofn.Flags = OFN_ENABLEHOOK | OFN_EXPLORER; 1329 ofn.hInstance = GetModuleHandleA(NULL); 1330 ret = GetOpenFileNameA(&ofn); 1331 ok(!ret, "GetOpenFileNameA returned %#x\n", ret); 1332 1333 hr = OleInitialize(NULL); 1334 ok(hr == S_OK, "OleInitialize() returned %#x\n", hr); 1335 OleUninitialize(); 1336 1337 UnhookWindowsHookEx(hook); 1338 } 1339 1340 START_TEST(filedlg) 1341 { 1342 test_DialogCancel(); 1343 test_create_view_window2(); 1344 test_create_view_template(); 1345 test_arrange(); 1346 test_resize(); 1347 test_ok(); 1348 test_getfolderpath(); 1349 test_mru(); 1350 if( resizesupported) test_resizable2(); 1351 test_extension(); 1352 test_null_filename(); 1353 test_directory_filename(); 1354 test_ole_initialization(); 1355 } 1356