2 * Unit test of the Program Manager DDE Interfaces
4 * Copyright 2009 Mikey Alexander
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
21 /* DDE Program Manager Tests
22 * - Covers basic CreateGroup, ShowGroup, DeleteGroup, AddItem, and DeleteItem
24 * - Todo: Handle CommonGroupFlag
25 * Better AddItem Tests (Lots of parameters to test)
26 * Tests for Invalid Characters in Names / Invalid Parameters
30 #include <wine/test.h>
37 /* Timeout on DdeClientTransaction Call */
38 #define MS_TIMEOUT_VAL 1000
39 /* # of times to poll for window creation */
40 #define PDDE_POLL_NUM 150
41 /* time to sleep between polls */
42 #define PDDE_POLL_TIME 300
45 #define DDE_TEST_MISC 0x00010000
46 #define DDE_TEST_CREATEGROUP 0x00020000
47 #define DDE_TEST_DELETEGROUP 0x00030000
48 #define DDE_TEST_SHOWGROUP 0x00040000
49 #define DDE_TEST_ADDITEM 0x00050000
50 #define DDE_TEST_DELETEITEM 0x00060000
51 #define DDE_TEST_COMPOUND 0x00070000
52 #define DDE_TEST_CALLMASK 0x00ff0000
54 #define DDE_TEST_NUMMASK 0x0000ffff
56 static HRESULT (WINAPI
*pSHGetLocalizedName
)(LPCWSTR
, LPWSTR
, UINT
, int *);
57 static BOOL (WINAPI
*pSHGetSpecialFolderPathA
)(HWND
, LPSTR
, int, BOOL
);
58 static BOOL (WINAPI
*pReadCabinetState
)(CABINETSTATE
*, int);
60 static void init_function_pointers(void)
64 hmod
= GetModuleHandleA("shell32.dll");
65 pSHGetLocalizedName
= (void*)GetProcAddress(hmod
, "SHGetLocalizedName");
66 pSHGetSpecialFolderPathA
= (void*)GetProcAddress(hmod
, "SHGetSpecialFolderPathA");
67 pReadCabinetState
= (void*)GetProcAddress(hmod
, "ReadCabinetState");
68 if (!pReadCabinetState
)
69 pReadCabinetState
= (void*)GetProcAddress(hmod
, (LPSTR
)651);
72 static BOOL
use_common(void)
75 static BOOL (WINAPI
*pIsNTAdmin
)(DWORD
, LPDWORD
);
77 /* IsNTAdmin() is available on all platforms. */
78 hmod
= LoadLibraryA("advpack.dll");
79 pIsNTAdmin
= (void*)GetProcAddress(hmod
, "IsNTAdmin");
81 if (!pIsNTAdmin(0, NULL
))
83 /* We are definitely not an administrator */
89 /* If we end up here we are on NT4+ as Win9x and WinMe don't have the
90 * notion of administrators (as we need it).
93 /* As of Vista we should always use the users directory. Tests with the
94 * real Administrator account on Windows 7 proved this.
96 * FIXME: We need a better way of identifying Vista+ as currently this check
97 * also covers Wine and we don't know yet which behavior we want to follow.
99 if (pSHGetLocalizedName
)
105 static BOOL
full_title(void)
109 memset(&cs
, 0, sizeof(cs
));
110 if (pReadCabinetState
)
112 pReadCabinetState(&cs
, sizeof(cs
));
119 win_skip("ReadCabinetState is not available, reading registry directly\n");
120 RegOpenKeyA(HKEY_CURRENT_USER
, "Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\CabinetState", &key
);
122 RegQueryValueExA(key
, "Settings", NULL
, NULL
, (LPBYTE
)&cs
, &size
);
126 return (cs
.fFullPathTitle
== -1);
129 static char ProgramsDir
[MAX_PATH
];
131 static char Group1Title
[MAX_PATH
] = "Group1";
132 static char Group2Title
[MAX_PATH
] = "Group2";
133 static char Group3Title
[MAX_PATH
] = "Group3";
134 static char StartupTitle
[MAX_PATH
] = "Startup";
136 static void init_strings(void)
138 char startup
[MAX_PATH
];
139 char commonprograms
[MAX_PATH
];
140 char programs
[MAX_PATH
];
142 if (pSHGetSpecialFolderPathA
)
144 pSHGetSpecialFolderPathA(NULL
, programs
, CSIDL_PROGRAMS
, FALSE
);
145 pSHGetSpecialFolderPathA(NULL
, commonprograms
, CSIDL_COMMON_PROGRAMS
, FALSE
);
146 pSHGetSpecialFolderPathA(NULL
, startup
, CSIDL_STARTUP
, FALSE
);
153 /* Older Win9x and NT4 */
155 RegOpenKeyA(HKEY_CURRENT_USER
, "Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Shell Folders", &key
);
156 size
= sizeof(programs
);
157 RegQueryValueExA(key
, "Programs", NULL
, NULL
, (LPBYTE
)&programs
, &size
);
158 size
= sizeof(startup
);
159 RegQueryValueExA(key
, "Startup", NULL
, NULL
, (LPBYTE
)&startup
, &size
);
162 RegOpenKeyA(HKEY_LOCAL_MACHINE
, "Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Shell Folders", &key
);
163 size
= sizeof(commonprograms
);
164 RegQueryValueExA(key
, "Common Programs", NULL
, NULL
, (LPBYTE
)&commonprograms
, &size
);
168 /* ProgramsDir on Vista+ is always the users one (CSIDL_PROGRAMS). Before Vista
169 * it depends on whether the user is an administrator (CSIDL_COMMON_PROGRAMS) or
170 * not (CSIDL_PROGRAMS).
173 lstrcpyA(ProgramsDir
, commonprograms
);
175 lstrcpyA(ProgramsDir
, programs
);
179 lstrcpyA(Group1Title
, ProgramsDir
);
180 lstrcatA(Group1Title
, "\\Group1");
181 lstrcpyA(Group2Title
, ProgramsDir
);
182 lstrcatA(Group2Title
, "\\Group2");
183 lstrcpyA(Group3Title
, ProgramsDir
);
184 lstrcatA(Group3Title
, "\\Group3");
186 lstrcpyA(StartupTitle
, startup
);
190 /* Vista has the nice habit of displaying the full path in English
191 * and the short one localized. CSIDL_STARTUP on Vista gives us the
192 * English version so we have to 'translate' this one.
194 * MSDN claims it should be used for files not folders but this one
195 * suits our purposes just fine.
197 if (pSHGetLocalizedName
)
199 WCHAR startupW
[MAX_PATH
];
200 WCHAR module
[MAX_PATH
];
201 WCHAR module_expanded
[MAX_PATH
];
202 WCHAR localized
[MAX_PATH
];
206 MultiByteToWideChar(CP_ACP
, 0, startup
, -1, startupW
, sizeof(startupW
)/sizeof(WCHAR
));
207 hr
= pSHGetLocalizedName(startupW
, module
, MAX_PATH
, &id
);
208 todo_wine
ok(hr
== S_OK
, "got 0x%08x\n", hr
);
209 /* check to be removed when SHGetLocalizedName is implemented */
212 ExpandEnvironmentStringsW(module
, module_expanded
, MAX_PATH
);
213 LoadStringW(GetModuleHandleW(module_expanded
), id
, localized
, MAX_PATH
);
215 WideCharToMultiByte(CP_ACP
, 0, localized
, -1, StartupTitle
, sizeof(StartupTitle
), NULL
, NULL
);
218 lstrcpyA(StartupTitle
, (strrchr(startup
, '\\') + 1));
222 lstrcpyA(StartupTitle
, (strrchr(startup
, '\\') + 1));
227 static HDDEDATA CALLBACK
DdeCallback(UINT type
, UINT format
, HCONV hConv
, HSZ hsz1
, HSZ hsz2
,
228 HDDEDATA hDDEData
, ULONG_PTR data1
, ULONG_PTR data2
)
230 trace("Callback: type=%i, format=%i\n", type
, format
);
235 * Encoded String for Error Messages so that inner failures can determine
236 * what test is failing. Format is: [Code:TestNum]
238 static const char * GetStringFromTestParams(int testParams
)
241 static char testParamString
[64];
244 testNum
= testParams
& DDE_TEST_NUMMASK
;
245 switch (testParams
& DDE_TEST_CALLMASK
)
251 case DDE_TEST_CREATEGROUP
:
254 case DDE_TEST_DELETEGROUP
:
257 case DDE_TEST_SHOWGROUP
:
260 case DDE_TEST_ADDITEM
:
263 case DDE_TEST_DELETEITEM
:
266 case DDE_TEST_COMPOUND
:
271 sprintf(testParamString
, " [%s:%i]", callId
, testNum
);
272 return testParamString
;
275 /* Transfer DMLERR's into text readable strings for Error Messages */
276 #define DMLERR_TO_STR(x) case x: return#x;
277 static const char * GetStringFromError(UINT err
)
281 DMLERR_TO_STR(DMLERR_NO_ERROR
);
282 DMLERR_TO_STR(DMLERR_ADVACKTIMEOUT
);
283 DMLERR_TO_STR(DMLERR_BUSY
);
284 DMLERR_TO_STR(DMLERR_DATAACKTIMEOUT
);
285 DMLERR_TO_STR(DMLERR_DLL_NOT_INITIALIZED
);
286 DMLERR_TO_STR(DMLERR_DLL_USAGE
);
287 DMLERR_TO_STR(DMLERR_EXECACKTIMEOUT
);
288 DMLERR_TO_STR(DMLERR_INVALIDPARAMETER
);
289 DMLERR_TO_STR(DMLERR_LOW_MEMORY
);
290 DMLERR_TO_STR(DMLERR_MEMORY_ERROR
);
291 DMLERR_TO_STR(DMLERR_NOTPROCESSED
);
292 DMLERR_TO_STR(DMLERR_NO_CONV_ESTABLISHED
);
293 DMLERR_TO_STR(DMLERR_POKEACKTIMEOUT
);
294 DMLERR_TO_STR(DMLERR_POSTMSG_FAILED
);
295 DMLERR_TO_STR(DMLERR_REENTRANCY
);
296 DMLERR_TO_STR(DMLERR_SERVER_DIED
);
297 DMLERR_TO_STR(DMLERR_SYS_ERROR
);
298 DMLERR_TO_STR(DMLERR_UNADVACKTIMEOUT
);
299 DMLERR_TO_STR(DMLERR_UNFOUND_QUEUE_ID
);
301 return "Unknown DML Error";
305 /* Helper Function to Transfer DdeGetLastError into a String */
306 static const char * GetDdeLastErrorStr(DWORD instance
)
308 UINT err
= DdeGetLastError(instance
);
310 return GetStringFromError(err
);
313 /* Execute a Dde Command and return the error & result */
314 /* Note: Progman DDE always returns a pointer to 0x00000001 on a successful result */
315 static void DdeExecuteCommand(DWORD instance
, HCONV hConv
, const char *strCmd
, HDDEDATA
*hData
, UINT
*err
, int testParams
)
319 command
= DdeCreateDataHandle(instance
, (LPBYTE
) strCmd
, strlen(strCmd
)+1, 0, 0L, 0, 0);
320 ok (command
!= NULL
, "DdeCreateDataHandle Error %s.%s\n",
321 GetDdeLastErrorStr(instance
), GetStringFromTestParams(testParams
));
322 *hData
= DdeClientTransaction((void *) command
,
331 /* hData is technically a pointer, but for Program Manager,
332 * it is NULL (error) or 1 (success)
333 * TODO: Check other versions of Windows to verify 1 is returned.
334 * While it is unlikely that anyone is actually testing that the result is 1
335 * if all versions of windows return 1, Wine should also.
339 *err
= DdeGetLastError(instance
);
343 *err
= DMLERR_NO_ERROR
;
346 ok(*hData
== (HDDEDATA
) 1, "Expected HDDEDATA Handle == 1, actually %p.%s\n",
347 *hData
, GetStringFromTestParams(testParams
));
350 DdeFreeDataHandle(command
);
354 * Check if Window is onscreen with the appropriate name.
356 * Windows are not created synchronously. So we do not know
357 * when and if the window will be created/shown on screen.
358 * This function implements a polling mechanism to determine
360 * A more complicated method would be to use SetWindowsHookEx.
361 * Since polling worked fine in my testing, no reason to implement
362 * the other. Comments about other methods of determining when
363 * window creation happened were not encouraging (not including
366 static void CheckWindowCreated(const char *winName
, int closeWindow
, int testParams
)
371 /* Poll for Window Creation */
372 for (i
= 0; window
== NULL
&& i
< PDDE_POLL_NUM
; i
++)
374 Sleep(PDDE_POLL_TIME
);
375 window
= FindWindowA(NULL
, winName
);
377 ok (window
!= NULL
, "Window \"%s\" was not created in %i seconds - assumed failure.%s\n",
378 winName
, PDDE_POLL_NUM
*PDDE_POLL_TIME
/1000, GetStringFromTestParams(testParams
));
380 /* Close Window as desired. */
381 if (window
!= NULL
&& closeWindow
)
383 SendMessageA(window
, WM_SYSCOMMAND
, SC_CLOSE
, 0);
387 /* Check for Existence (or non-existence) of a file or group
388 * When testing for existence of a group, groupName is not needed
390 static void CheckFileExistsInProgramGroups(const char *nameToCheck
, int shouldExist
, int isGroup
,
391 const char *groupName
, int testParams
)
397 lstrcpyA(path
, ProgramsDir
);
399 len
= strlen(path
) + strlen(nameToCheck
)+1;
400 if (groupName
!= NULL
)
402 len
+= strlen(groupName
)+1;
404 ok (len
<= MAX_PATH
, "Path Too Long.%s\n", GetStringFromTestParams(testParams
));
407 if (groupName
!= NULL
)
410 strcat(path
, groupName
);
413 strcat(path
, nameToCheck
);
414 attributes
= GetFileAttributes(path
);
417 ok (attributes
== INVALID_FILE_ATTRIBUTES
, "File exists and shouldn't %s.%s\n",
418 path
, GetStringFromTestParams(testParams
));
420 if (attributes
== INVALID_FILE_ATTRIBUTES
)
422 ok (FALSE
, "Created File %s doesn't exist.%s\n", path
, GetStringFromTestParams(testParams
));
423 } else if (isGroup
) {
424 ok (attributes
& FILE_ATTRIBUTE_DIRECTORY
, "%s is not a folder (attr=%x).%s\n",
425 path
, attributes
, GetStringFromTestParams(testParams
));
427 ok (attributes
& FILE_ATTRIBUTE_ARCHIVE
, "Created File %s has wrong attributes (%x).%s\n",
428 path
, attributes
, GetStringFromTestParams(testParams
));
434 /* Create Group Test.
435 * command and expected_result.
436 * if expected_result is DMLERR_NO_ERROR, test
437 * 1. group was created
440 static void CreateGroupTest(DWORD instance
, HCONV hConv
, const char *command
, UINT expected_result
,
441 const char *groupName
, const char *windowTitle
, int testParams
)
446 /* Execute Command & Check Result */
447 DdeExecuteCommand(instance
, hConv
, command
, &hData
, &error
, testParams
);
450 ok (expected_result
== error
, "CreateGroup %s: Expected Error %s, received %s.%s\n",
451 groupName
, GetStringFromError(expected_result
), GetStringFromError(error
),
452 GetStringFromTestParams(testParams
));
456 if (error
== DMLERR_NO_ERROR
)
459 /* Check if Group Now Exists */
460 CheckFileExistsInProgramGroups(groupName
, TRUE
, TRUE
, NULL
, testParams
);
461 /* Check if Window is Open (polling) */
462 CheckWindowCreated(windowTitle
, TRUE
, testParams
);
467 * DDE command, expected_result, and the group name to check for existence
468 * if expected_result is DMLERR_NO_ERROR, test
471 static void ShowGroupTest(DWORD instance
, HCONV hConv
, const char *command
, UINT expected_result
,
472 const char *groupName
, const char *windowTitle
, int closeAfterShowing
, int testParams
)
477 DdeExecuteCommand(instance
, hConv
, command
, &hData
, &error
, testParams
);
478 /* todo_wine... Is expected to fail, wine stubbed functions DO fail */
479 /* TODO REMOVE THIS CODE!!! */
480 if (expected_result
== DMLERR_NOTPROCESSED
)
482 ok (expected_result
== error
, "ShowGroup %s: Expected Error %s, received %s.%s\n",
483 groupName
, GetStringFromError(expected_result
), GetStringFromError(error
),
484 GetStringFromTestParams(testParams
));
488 ok (expected_result
== error
, "ShowGroup %s: Expected Error %s, received %s.%s\n",
489 groupName
, GetStringFromError(expected_result
), GetStringFromError(error
),
490 GetStringFromTestParams(testParams
));
494 if (error
== DMLERR_NO_ERROR
)
496 /* Check if Window is Open (polling) */
497 CheckWindowCreated(windowTitle
, closeAfterShowing
, testParams
);
501 /* Delete Group Test.
502 * DDE command, expected_result, and the group name to check for existence
503 * if expected_result is DMLERR_NO_ERROR, test
504 * 1. group does not exist
506 static void DeleteGroupTest(DWORD instance
, HCONV hConv
, const char *command
, UINT expected_result
,
507 const char *groupName
, int testParams
)
512 DdeExecuteCommand(instance
, hConv
, command
, &hData
, &error
, testParams
);
515 ok (expected_result
== error
, "DeleteGroup %s: Expected Error %s, received %s.%s\n",
516 groupName
, GetStringFromError(expected_result
), GetStringFromError(error
),
517 GetStringFromTestParams(testParams
));
520 if (error
== DMLERR_NO_ERROR
)
522 /* Check that Group does not exist */
523 CheckFileExistsInProgramGroups(groupName
, FALSE
, TRUE
, NULL
, testParams
);
528 * DDE command, expected result, and group and file name where it should exist.
529 * checks to make sure error code matches expected error code
530 * checks to make sure item exists if successful
532 static void AddItemTest(DWORD instance
, HCONV hConv
, const char *command
, UINT expected_result
,
533 const char *fileName
, const char *groupName
, int testParams
)
538 DdeExecuteCommand(instance
, hConv
, command
, &hData
, &error
, testParams
);
541 ok (expected_result
== error
, "AddItem %s: Expected Error %s, received %s.%s\n",
542 fileName
, GetStringFromError(expected_result
), GetStringFromError(error
),
543 GetStringFromTestParams(testParams
));
546 if (error
== DMLERR_NO_ERROR
)
548 /* Check that File exists */
549 CheckFileExistsInProgramGroups(fileName
, TRUE
, FALSE
, groupName
, testParams
);
554 * DDE command, expected result, and group and file name where it should exist.
555 * checks to make sure error code matches expected error code
556 * checks to make sure item does not exist if successful
558 static void DeleteItemTest(DWORD instance
, HCONV hConv
, const char *command
, UINT expected_result
,
559 const char *fileName
, const char *groupName
, int testParams
)
564 DdeExecuteCommand(instance
, hConv
, command
, &hData
, &error
, testParams
);
567 ok (expected_result
== error
, "DeleteItem %s: Expected Error %s, received %s.%s\n",
568 fileName
, GetStringFromError(expected_result
), GetStringFromError(error
),
569 GetStringFromTestParams(testParams
));
572 if (error
== DMLERR_NO_ERROR
)
574 /* Check that File does not exist */
575 CheckFileExistsInProgramGroups(fileName
, FALSE
, FALSE
, groupName
, testParams
);
579 /* Compound Command Test.
580 * not really generic, assumes command of the form:
581 * [CreateGroup ...][AddItem ...][AddItem ...]
582 * All samples I've seen using Compound were of this form (CreateGroup,
583 * AddItems) so this covers minimum expected functionality.
585 static void CompoundCommandTest(DWORD instance
, HCONV hConv
, const char *command
, UINT expected_result
,
586 const char *groupName
, const char *windowTitle
, const char *fileName1
,
587 const char *fileName2
, int testParams
)
592 DdeExecuteCommand(instance
, hConv
, command
, &hData
, &error
, testParams
);
595 ok (expected_result
== error
, "Compound String %s: Expected Error %s, received %s.%s\n",
596 command
, GetStringFromError(expected_result
), GetStringFromError(error
),
597 GetStringFromTestParams(testParams
));
600 if (error
== DMLERR_NO_ERROR
)
602 /* Check that File exists */
603 CheckFileExistsInProgramGroups(groupName
, TRUE
, TRUE
, NULL
, testParams
);
604 CheckWindowCreated(windowTitle
, FALSE
, testParams
);
605 CheckFileExistsInProgramGroups(fileName1
, TRUE
, FALSE
, groupName
, testParams
);
606 CheckFileExistsInProgramGroups(fileName2
, TRUE
, FALSE
, groupName
, testParams
);
610 static void CreateAddItemText(char *itemtext
, const char *cmdline
, const char *name
)
612 lstrcpyA(itemtext
, "[AddItem(");
613 lstrcatA(itemtext
, cmdline
);
614 lstrcatA(itemtext
, ",");
615 lstrcatA(itemtext
, name
);
616 lstrcatA(itemtext
, ")]");
619 /* 1st set of tests */
620 static int DdeTestProgman(DWORD instance
, HCONV hConv
)
625 char temppath
[MAX_PATH
];
626 char f1g1
[MAX_PATH
], f2g1
[MAX_PATH
], f3g1
[MAX_PATH
], f1g3
[MAX_PATH
], f2g3
[MAX_PATH
];
627 char itemtext
[MAX_PATH
+ 20];
628 char comptext
[2 * (MAX_PATH
+ 20) + 21];
631 /* Invalid Command */
632 DdeExecuteCommand(instance
, hConv
, "[InvalidCommand()]", &hData
, &error
, DDE_TEST_MISC
|testnum
++);
633 ok (error
== DMLERR_NOTPROCESSED
, "InvalidCommand(), expected error %s, received %s.\n",
634 GetStringFromError(DMLERR_NOTPROCESSED
), GetStringFromError(error
));
636 /* On Vista+ the files have to exist when adding a link */
637 GetTempPathA(MAX_PATH
, temppath
);
638 GetTempFileNameA(temppath
, "dde", 0, f1g1
);
639 GetTempFileNameA(temppath
, "dde", 0, f2g1
);
640 GetTempFileNameA(temppath
, "dde", 0, f3g1
);
641 GetTempFileNameA(temppath
, "dde", 0, f1g3
);
642 GetTempFileNameA(temppath
, "dde", 0, f2g3
);
644 /* CreateGroup Tests (including AddItem, DeleteItem) */
645 CreateGroupTest(instance
, hConv
, "[CreateGroup(Group1)]", DMLERR_NO_ERROR
, "Group1", Group1Title
, DDE_TEST_CREATEGROUP
|testnum
++);
646 CreateAddItemText(itemtext
, f1g1
, "f1g1Name");
647 AddItemTest(instance
, hConv
, itemtext
, DMLERR_NO_ERROR
, "f1g1Name.lnk", "Group1", DDE_TEST_ADDITEM
|testnum
++);
648 CreateAddItemText(itemtext
, f2g1
, "f2g1Name");
649 AddItemTest(instance
, hConv
, itemtext
, DMLERR_NO_ERROR
, "f2g1Name.lnk", "Group1", DDE_TEST_ADDITEM
|testnum
++);
650 DeleteItemTest(instance
, hConv
, "[DeleteItem(f2g1Name)]", DMLERR_NO_ERROR
, "f2g1Name.lnk", "Group1", DDE_TEST_DELETEITEM
|testnum
++);
651 CreateAddItemText(itemtext
, f3g1
, "f3g1Name");
652 AddItemTest(instance
, hConv
, itemtext
, DMLERR_NO_ERROR
, "f3g1Name.lnk", "Group1", DDE_TEST_ADDITEM
|testnum
++);
653 CreateGroupTest(instance
, hConv
, "[CreateGroup(Group2)]", DMLERR_NO_ERROR
, "Group2", Group2Title
, DDE_TEST_CREATEGROUP
|testnum
++);
654 /* Create Group that already exists - same instance */
655 CreateGroupTest(instance
, hConv
, "[CreateGroup(Group1)]", DMLERR_NO_ERROR
, "Group1", Group1Title
, DDE_TEST_CREATEGROUP
|testnum
++);
657 /* ShowGroup Tests */
658 ShowGroupTest(instance
, hConv
, "[ShowGroup(Group1)]", DMLERR_NOTPROCESSED
, "Group1", Group1Title
, TRUE
, DDE_TEST_SHOWGROUP
|testnum
++);
659 DeleteItemTest(instance
, hConv
, "[DeleteItem(f3g1Name)]", DMLERR_NO_ERROR
, "f3g1Name.lnk", "Group1", DDE_TEST_DELETEITEM
|testnum
++);
660 ShowGroupTest(instance
, hConv
, "[ShowGroup(Startup,0)]", DMLERR_NO_ERROR
, "Startup", StartupTitle
, TRUE
, DDE_TEST_SHOWGROUP
|testnum
++);
661 ShowGroupTest(instance
, hConv
, "[ShowGroup(Group1,0)]", DMLERR_NO_ERROR
, "Group1", Group1Title
, FALSE
, DDE_TEST_SHOWGROUP
|testnum
++);
663 /* DeleteGroup Test - Note that Window is Open for this test */
664 DeleteGroupTest(instance
, hConv
, "[DeleteGroup(Group1)]", DMLERR_NO_ERROR
, "Group1", DDE_TEST_DELETEGROUP
|testnum
++);
666 /* Compound Execute String Command */
667 lstrcpyA(comptext
, "[CreateGroup(Group3)]");
668 CreateAddItemText(itemtext
, f1g3
, "f1g3Name");
669 lstrcatA(comptext
, itemtext
);
670 CreateAddItemText(itemtext
, f2g3
, "f2g3Name");
671 lstrcatA(comptext
, itemtext
);
672 CompoundCommandTest(instance
, hConv
, comptext
, DMLERR_NO_ERROR
, "Group3", Group3Title
, "f1g3Name.lnk", "f2g3Name.lnk", DDE_TEST_COMPOUND
|testnum
++);
674 DeleteGroupTest(instance
, hConv
, "[DeleteGroup(Group3)]", DMLERR_NO_ERROR
, "Group3", DDE_TEST_DELETEGROUP
|testnum
++);
676 /* Full Parameters of Add Item */
677 /* AddItem(CmdLine[,Name[,IconPath[,IconIndex[,xPos,yPos[,DefDir[,HotKey[,fMinimize[fSeparateSpace]]]]]]]) */
688 /* 2nd set of tests - 2nd connection */
689 static void DdeTestProgman2(DWORD instance
, HCONV hConv
, int testnum
)
691 /* Create Group that already exists on a separate connection */
692 CreateGroupTest(instance
, hConv
, "[CreateGroup(Group2)]", DMLERR_NO_ERROR
, "Group2", Group2Title
, DDE_TEST_CREATEGROUP
|testnum
++);
693 DeleteGroupTest(instance
, hConv
, "[DeleteGroup(Group2)]", DMLERR_NO_ERROR
, "Group2", DDE_TEST_DELETEGROUP
|testnum
++);
696 START_TEST(progman_dde
)
704 init_function_pointers();
707 /* Initialize DDE Instance */
708 err
= DdeInitialize(&instance
, DdeCallback
, APPCMD_CLIENTONLY
, 0);
709 ok (err
== DMLERR_NO_ERROR
, "DdeInitialize Error %s\n", GetStringFromError(err
));
711 /* Create Connection */
712 hszProgman
= DdeCreateStringHandle(instance
, "PROGMAN", CP_WINANSI
);
713 ok (hszProgman
!= NULL
, "DdeCreateStringHandle Error %s\n", GetDdeLastErrorStr(instance
));
714 hConv
= DdeConnect(instance
, hszProgman
, hszProgman
, NULL
);
715 ok (DdeFreeStringHandle(instance
, hszProgman
), "DdeFreeStringHandle failure\n");
716 /* Seeing failures on early versions of Windows Connecting to progman, exit if connection fails */
719 ok (DdeUninitialize(instance
), "DdeUninitialize failed\n");
724 testnum
= DdeTestProgman(instance
, hConv
);
727 ok (DdeDisconnect(hConv
), "DdeDisonnect Error %s\n", GetDdeLastErrorStr(instance
));
728 ok (DdeUninitialize(instance
), "DdeUninitialize failed\n");
730 /* 2nd Instance (Followup Tests) */
731 /* Initialize DDE Instance */
733 err
= DdeInitialize(&instance
, DdeCallback
, APPCMD_CLIENTONLY
, 0);
734 ok (err
== DMLERR_NO_ERROR
, "DdeInitialize Error %s\n", GetStringFromError(err
));
736 /* Create Connection */
737 hszProgman
= DdeCreateStringHandle(instance
, "PROGMAN", CP_WINANSI
);
738 ok (hszProgman
!= NULL
, "DdeCreateStringHandle Error %s\n", GetDdeLastErrorStr(instance
));
739 hConv
= DdeConnect(instance
, hszProgman
, hszProgman
, NULL
);
740 ok (hConv
!= NULL
, "DdeConnect Error %s\n", GetDdeLastErrorStr(instance
));
741 ok (DdeFreeStringHandle(instance
, hszProgman
), "DdeFreeStringHandle failure\n");
744 DdeTestProgman2(instance
, hConv
, testnum
);
747 ok (DdeDisconnect(hConv
), "DdeDisonnect Error %s\n", GetDdeLastErrorStr(instance
));
748 ok (DdeUninitialize(instance
), "DdeUninitialize failed\n");