Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

3923 lines
120 KiB

  1. //---------------------------------------------------------------------------
  2. //
  3. //---------------------------------------------------------------------------
  4. #include "grpconv.h"
  5. #include "util.h"
  6. #include "rcids.h"
  7. #include "group.h"
  8. #include "gcinst.h"
  9. #include <port32.h>
  10. #include <regstr.h>
  11. #define INITGUID
  12. #include <initguid.h>
  13. #pragma data_seg(DATASEG_READONLY)
  14. #include <coguid.h>
  15. #include <oleguid.h>
  16. #pragma data_seg()
  17. #ifdef DEBUG
  18. extern UINT GC_TRACE;
  19. #endif
  20. //---------------------------------------------------------------------------
  21. // Exported.
  22. const TCHAR c_szMapGroups[] = TEXT("MapGroups");
  23. #ifndef WINNT
  24. const TCHAR c_szDelGroups[] = TEXT("DelGroups");
  25. #endif
  26. //---------------------------------------------------------------------------
  27. // Global to this file only;
  28. static const TCHAR c_szGrpConv[] = TEXT("Grpconv");
  29. static const TCHAR c_szLastModDateTime[] = TEXT("LastModDateTime");
  30. static const TCHAR c_szRegistry[] = TEXT("Registry");
  31. static const TCHAR c_szDefaultUser[] = TEXT("DefaultUser");
  32. static const TCHAR c_szGrpConvData[] = TEXT("compat.csv");
  33. static const TCHAR c_szProgmanStartup[] = TEXT("Software\\Microsoft\\Windows NT\\CurrentVersion\\Program Manager\\Settings\\Startup");
  34. static const TCHAR c_szDotPif[] = TEXT(".pif");
  35. // New group stuff
  36. HDSA hdsaPMItems; // current group
  37. HDSA g_hdsaAppList;
  38. HKEY hkeyGroups = NULL;
  39. BOOL g_fDoingCommonGroups = FALSE;
  40. #pragma pack(1)
  41. typedef struct tagRECTS
  42. {
  43. short left;
  44. short top;
  45. short right;
  46. short bottom;
  47. } RECTS;
  48. typedef struct
  49. {
  50. LPTSTR lpszDesc;
  51. LPTSTR lpszCL;
  52. LPTSTR lpszWD;
  53. LPTSTR lpszIconPath;
  54. WORD wiIcon;
  55. WORD wHotKey;
  56. int nShowCmd;
  57. #ifdef WINNT
  58. BOOL bSepVdm;
  59. #endif
  60. } PMITEM, *PPMITEM, *LPPMITEM;
  61. // Old Progman stuff.
  62. #define GROUP_MAGIC 0x43434D50L // 'PMCC'
  63. #define GROUP_UNICODE 0x43554D50L // 'PMUC'
  64. /*
  65. * Win 3.1 .GRP file formats (ITEMDEF for items, GROUPDEF for groups)
  66. */
  67. typedef struct
  68. {
  69. POINTS pt;
  70. WORD iIcon;
  71. WORD cbHeader;
  72. WORD cbANDPlane;
  73. WORD cbXORPlane;
  74. WORD pHeader;
  75. WORD pANDPlane;
  76. WORD pXORPlane;
  77. WORD pName;
  78. WORD pCommand;
  79. WORD pIconPath;
  80. } ITEMDEF, *PITEMDEF, *LPITEMDEF;
  81. typedef struct
  82. {
  83. DWORD dwMagic;
  84. WORD wCheckSum;
  85. WORD cbGroup;
  86. WORD nCmdShow;
  87. RECTS rcNormal;
  88. POINTS ptMin;
  89. WORD pName;
  90. WORD cxIcon;
  91. WORD cyIcon;
  92. WORD wIconFormat;
  93. WORD wReserved;
  94. WORD cItems;
  95. } GROUPDEF, *PGROUPDEF, *LPGROUPDEF;
  96. typedef struct
  97. {
  98. WORD wID;
  99. WORD wItem;
  100. WORD cb;
  101. } PMTAG, *PPMTAG, *LPPMTAG;
  102. // Thank God the tag stuff never really caught on.
  103. #define TAG_MAGIC GROUP_MAGIC
  104. #define ID_MAINTAIN 0x8000
  105. #define ID_MAGIC 0x8000
  106. #define ID_WRITERVERSION 0x8001
  107. #define ID_APPLICATIONDIR 0x8101
  108. #define ID_HOTKEY 0x8102
  109. #define ID_MINIMIZE 0x8103
  110. #ifdef WINNT
  111. #define ID_NEWVDM 0x8104
  112. #endif
  113. #define ID_LASTTAG 0xFFFF
  114. /*
  115. * NT 3.1 Ansi .GRP File format structures
  116. */
  117. typedef struct tagGROUPDEF_A {
  118. DWORD dwMagic; /* magical bytes 'PMCC' */
  119. WORD wCheckSum; /* adjust this for zero sum of file */
  120. WORD cbGroup; /* length of group segment */
  121. RECT rcNormal; /* rectangle of normal window */
  122. POINT ptMin; /* point of icon */
  123. WORD nCmdShow; /* min, max, or normal state */
  124. WORD pName; /* name of group */
  125. /* these four change interpretation */
  126. WORD cxIcon; /* width of icons */
  127. WORD cyIcon; /* hieght of icons */
  128. WORD wIconFormat; /* planes and BPP in icons */
  129. WORD wReserved; /* This word is no longer used. */
  130. WORD cItems; /* number of items in group */
  131. WORD rgiItems[1]; /* array of ITEMDEF offsets */
  132. } NT_GROUPDEF_A, *PNT_GROUPDEF_A;
  133. typedef NT_GROUPDEF_A *LPNT_GROUPDEF_A;
  134. typedef struct tagITEMDEF_A {
  135. POINT pt; /* location of item icon in group */
  136. WORD idIcon; /* id of item icon */
  137. WORD wIconVer; /* icon version */
  138. WORD cbIconRes; /* size of icon resource */
  139. WORD indexIcon; /* index of item icon */
  140. WORD dummy2; /* - not used anymore */
  141. WORD pIconRes; /* offset of icon resource */
  142. WORD dummy3; /* - not used anymore */
  143. WORD pName; /* offset of name string */
  144. WORD pCommand; /* offset of command string */
  145. WORD pIconPath; /* offset of icon path */
  146. } NT_ITEMDEF_A, *PNT_ITEMDEF_A;
  147. typedef NT_ITEMDEF_A *LPNT_ITEMDEF_A;
  148. /*
  149. * NT 3.1a Unicode .GRP File format structures
  150. */
  151. typedef struct tagGROUPDEF {
  152. DWORD dwMagic; /* magical bytes 'PMCC' */
  153. DWORD cbGroup; /* length of group segment */
  154. RECT rcNormal; /* rectangle of normal window */
  155. POINT ptMin; /* point of icon */
  156. WORD wCheckSum; /* adjust this for zero sum of file */
  157. WORD nCmdShow; /* min, max, or normal state */
  158. DWORD pName; /* name of group */
  159. /* these four change interpretation */
  160. WORD cxIcon; /* width of icons */
  161. WORD cyIcon; /* hieght of icons */
  162. WORD wIconFormat; /* planes and BPP in icons */
  163. WORD wReserved; /* This word is no longer used. */
  164. WORD cItems; /* number of items in group */
  165. WORD Reserved1;
  166. DWORD Reserved2;
  167. DWORD rgiItems[1]; /* array of ITEMDEF offsets */
  168. } NT_GROUPDEF, *PNT_GROUPDEF;
  169. typedef NT_GROUPDEF *LPNT_GROUPDEF;
  170. typedef struct tagITEMDEF {
  171. POINT pt; /* location of item icon in group */
  172. WORD iIcon; /* id of item icon */
  173. WORD wIconVer; /* icon version */
  174. WORD cbIconRes; /* size of icon resource */
  175. WORD wIconIndex; /* index of the item icon (not the same as the id) */
  176. DWORD pIconRes; /* offset of icon resource */
  177. DWORD pName; /* offset of name string */
  178. DWORD pCommand; /* offset of command string */
  179. DWORD pIconPath; /* offset of icon path */
  180. } NT_ITEMDEF, *PNT_ITEMDEF;
  181. typedef NT_ITEMDEF *LPNT_ITEMDEF;
  182. typedef struct _tag
  183. {
  184. WORD wID; // tag identifier
  185. WORD dummy1; // need this for alignment!
  186. int wItem; // (unde the covers 32 bit point!)item the tag belongs to
  187. WORD cb; // size of record, including id and count
  188. WORD dummy2; // need this for alignment!
  189. BYTE rgb[1];
  190. } NT_PMTAG, * LPNT_PMTAG;
  191. /* the pointers in the above structures are short pointers relative to the
  192. * beginning of the segments. This macro converts the short pointer into
  193. * a long pointer including the proper segment/selector value. It assumes
  194. * that its argument is an lvalue somewhere in a group segment, for example,
  195. * PTR(lpgd->pName) returns a pointer to the group name, but k=lpgd->pName;
  196. * PTR(k) is obviously wrong as it will use either SS or DS for its segment,
  197. * depending on the storage class of k.
  198. */
  199. #define PTR(base, offset) (LPBYTE)((PBYTE)base + offset)
  200. /* PTR2 is used for those cases where a variable already contains an offset
  201. * (The "case that doesn't work", above)
  202. */
  203. #define PTR2(lp,offset) ((LPBYTE)MAKELONG(offset,HIWORD(lp)))
  204. /* this macro is used to retrieve the i-th item in the group segment. Note
  205. * that this pointer will NOT be NULL for an unused slot.
  206. */
  207. #define ITEM(lpgd,i) ((LPNT_ITEMDEF)PTR(lpgd, lpgd->rgiItems[i]))
  208. /* Keeping things starting on aligned boundaries allows faster access on
  209. * most platforms.
  210. */
  211. #define MyDwordAlign(size) (((size) + 3) & ~3)
  212. #pragma pack()
  213. #define CFree(a) if(a) Free(a)
  214. //---------------------------------------------------------------------------
  215. #define Stream_Write(ps, pv, cb) SUCCEEDED((ps)->lpVtbl->Write(ps, pv, cb, NULL))
  216. #define Stream_Close(ps) (void)(ps)->lpVtbl->Release(ps)
  217. #define VOF_BAD 0
  218. #define VOF_WIN31 1
  219. #define VOF_WINNT 2
  220. int ConvertToUnicodeGroup(LPNT_GROUPDEF_A lpGroupORI, LPHANDLE lphNewGroup);
  221. //---------------------------------------------------------------------------
  222. // Init the group stuff
  223. BOOL ItemList_Create(LPCTSTR lpszGroup)
  224. {
  225. if (!hdsaPMItems)
  226. hdsaPMItems = DSA_Create(SIZEOF(PMITEM), 16);
  227. if (hdsaPMItems)
  228. return TRUE;
  229. DebugMsg(DM_ERROR, TEXT("cg.gi: Unable to init."));
  230. return FALSE;
  231. }
  232. //---------------------------------------------------------------------------
  233. // Tidyup.
  234. void ItemList_Destroy(void)
  235. {
  236. int i;
  237. int cItems;
  238. LPPMITEM lppmitem;
  239. // Clean up the items.
  240. cItems = DSA_GetItemCount(hdsaPMItems);
  241. for(i=0; i < cItems; i++)
  242. {
  243. lppmitem = DSA_GetItemPtr(hdsaPMItems, 0);
  244. // Nuke the strings.
  245. CFree(lppmitem->lpszDesc);
  246. CFree(lppmitem->lpszCL);
  247. CFree(lppmitem->lpszWD);
  248. CFree(lppmitem->lpszIconPath);
  249. // Nuke the structure.
  250. DSA_DeleteItem(hdsaPMItems, 0);
  251. }
  252. DSA_Destroy(hdsaPMItems);
  253. hdsaPMItems = NULL;
  254. }
  255. //---------------------------------------------------------------------------
  256. // Returns TRUE if the file smells like an old PM group, the title of the
  257. // group is returned in lpszTitle which must be at least 32 chars big.
  258. // REVIEW - Is it worth checking the checksum?
  259. UINT Group_ValidOldFormat(LPCTSTR lpszOldGroup, LPTSTR lpszTitle)
  260. {
  261. #ifdef UNICODE
  262. HANDLE fh;
  263. DWORD dwBytesRead;
  264. #else
  265. HFILE fh;
  266. #endif
  267. UINT nCode;
  268. GROUPDEF grpdef;
  269. // Find and open the group file.
  270. #ifdef UNICODE
  271. fh = CreateFile(
  272. lpszOldGroup,
  273. GENERIC_READ,
  274. FILE_SHARE_READ,
  275. NULL,
  276. OPEN_EXISTING,
  277. 0,
  278. NULL
  279. );
  280. if (fh != INVALID_HANDLE_VALUE)
  281. #else
  282. fh = _lopen(lpszOldGroup, OF_READ | OF_SHARE_DENY_NONE);
  283. if (fh != HFILE_ERROR)
  284. #endif
  285. {
  286. // Get the definition.
  287. #ifdef UNICODE
  288. ReadFile(fh, &grpdef, SIZEOF(grpdef), &dwBytesRead, NULL);
  289. #else
  290. _lread(fh, &grpdef, SIZEOF(grpdef));
  291. #endif
  292. // Does it have the right magic bytes?.
  293. switch( grpdef.dwMagic )
  294. {
  295. case GROUP_UNICODE:
  296. {
  297. NT_GROUPDEF nt_grpdef;
  298. #ifdef UNICODE
  299. SetFilePointer(fh, 0, NULL, FILE_BEGIN);
  300. ReadFile(fh, &nt_grpdef, SIZEOF(nt_grpdef), &dwBytesRead, NULL);
  301. #else
  302. _llseek(fh, 0, 0); // Back to the start
  303. _lread(fh, &nt_grpdef, SIZEOF(nt_grpdef));
  304. #endif
  305. // Yep, Get it's size..
  306. // Is it at least as big as the header says it is?
  307. #ifdef UNICODE
  308. if ( nt_grpdef.cbGroup <= (DWORD)SetFilePointer(fh, 0L, NULL, FILE_END))
  309. #else
  310. if ( nt_grpdef.cbGroup <= (DWORD)_llseek(fh, 0L, 2))
  311. #endif
  312. {
  313. WCHAR wchGroupName[MAXGROUPNAMELEN+1];
  314. // Yep, probably valid.
  315. // Get its title.
  316. #ifdef UNICODE
  317. SetFilePointer(fh, nt_grpdef.pName, 0, FILE_BEGIN);
  318. ReadFile(fh, wchGroupName, SIZEOF(wchGroupName), &dwBytesRead, NULL);
  319. lstrcpy(lpszTitle, wchGroupName);
  320. #else
  321. _llseek(fh, nt_grpdef.pName, 0);
  322. _lread(fh,wchGroupName, SIZEOF(wchGroupName));
  323. WideCharToMultiByte (CP_ACP, 0, wchGroupName, -1,
  324. lpszTitle, MAXGROUPNAMELEN+1, NULL, NULL);
  325. #endif
  326. nCode = VOF_WINNT;
  327. }
  328. else
  329. {
  330. // No. Too small.
  331. DebugMsg(DM_TRACE, TEXT("gc.gvof: File has invalid size."));
  332. nCode = VOF_BAD;
  333. }
  334. }
  335. break;
  336. case GROUP_MAGIC:
  337. {
  338. CHAR chGroupName[MAXGROUPNAMELEN+1];
  339. // Yep, Get it's size..
  340. // Is it at least as big as the header says it is?
  341. #ifdef UNICODE
  342. if (grpdef.cbGroup <= (WORD) SetFilePointer(fh, 0L, NULL, FILE_END))
  343. #else
  344. if (grpdef.cbGroup <= (WORD) _llseek(fh, 0L, 2))
  345. #endif
  346. {
  347. // Check to make sure there is a name embedded in the
  348. // .grp file. If not, just use the filename
  349. if (grpdef.pName==0)
  350. {
  351. LPTSTR lpszFile, lpszExt, lpszDest = lpszTitle;
  352. lpszFile = PathFindFileName( lpszOldGroup );
  353. lpszExt = PathFindExtension( lpszOldGroup );
  354. for( ;
  355. lpszFile && lpszExt && (lpszFile != lpszExt);
  356. *lpszDest++ = *lpszFile++
  357. );
  358. *lpszDest = TEXT('\0');
  359. }
  360. else
  361. {
  362. // Yep, probably valid.
  363. // Get it's title.
  364. #ifdef UNICODE
  365. SetFilePointer(fh, grpdef.pName, NULL, FILE_BEGIN);
  366. ReadFile(fh, chGroupName, MAXGROUPNAMELEN+1, &dwBytesRead, NULL);
  367. MultiByteToWideChar(
  368. CP_ACP,
  369. MB_PRECOMPOSED,
  370. chGroupName,
  371. -1,
  372. lpszTitle,
  373. MAXGROUPNAMELEN+1
  374. ) ;
  375. #else
  376. _llseek(fh, grpdef.pName, 0);
  377. _lread(fh, lpszTitle, MAXGROUPNAMELEN+1);
  378. #endif
  379. }
  380. nCode = VOF_WIN31;
  381. }
  382. else
  383. {
  384. // No. Too small.
  385. DebugMsg(DM_TRACE, TEXT("gc.gvof: File has invalid size."));
  386. nCode = VOF_BAD;
  387. }
  388. break;
  389. }
  390. default:
  391. // No, the magic bytes are wrong.
  392. DebugMsg(DM_TRACE, TEXT("gc.gvof: File has invalid magic bytes."));
  393. nCode = VOF_BAD;
  394. break;
  395. }
  396. #ifdef UNICODE
  397. CloseHandle(fh);
  398. #else
  399. _lclose(fh);
  400. #endif
  401. }
  402. else
  403. {
  404. // No. Can't even read the file.
  405. DebugMsg(DM_TRACE, TEXT("gc.gvof: File is unreadble."));
  406. nCode = VOF_BAD;
  407. }
  408. return nCode;
  409. }
  410. BOOL _IsValidFileNameChar(TBYTE ch, UINT flags)
  411. {
  412. switch (ch) {
  413. case TEXT('\\'): // path separator
  414. return flags & PRICF_ALLOWSLASH;
  415. case TEXT(';'): // terminator
  416. case TEXT(','): // terminator
  417. case TEXT('|'): // pipe
  418. case TEXT('>'): // redir
  419. case TEXT('<'): // redir
  420. case TEXT('"'): // quote
  421. case TEXT('?'): // wc we only do wilds here because they're
  422. case TEXT('*'): // wc legal for qualifypath
  423. case TEXT(':'): // drive colon
  424. case TEXT('/'): // path sep
  425. return FALSE;
  426. }
  427. // Can not be a control char...
  428. return ch >= TEXT(' ');
  429. }
  430. void PathRemoveIllegalChars(LPTSTR pszPath, int iGroupName, UINT flags)
  431. {
  432. LPTSTR pszT = pszPath + iGroupName;
  433. // Map all of the strange characters out of the name for both LFn and not
  434. // machines
  435. while (*pszT)
  436. {
  437. if (!_IsValidFileNameChar(*pszT, flags))
  438. *pszT = TEXT('_'); // Don't Allow invalid chars in names
  439. pszT = CharNext(pszT);
  440. }
  441. }
  442. //---------------------------------------------------------------------------
  443. // We want certain groups to end up in a new location eg Games is now
  444. // Applications\Games.
  445. void MapGroupTitle(LPCTSTR lpszOld, LPTSTR lpszNew, UINT cchNew)
  446. {
  447. // Is there a mapping?
  448. if (!Reg_GetString(g_hkeyGrpConv, c_szMapGroups, lpszOld, lpszNew, cchNew*sizeof(TCHAR)))
  449. {
  450. // Nope, just use the given name.
  451. lstrcpyn(lpszNew, lpszOld, cchNew);
  452. }
  453. DebugMsg(DM_TRACE, TEXT("gc.mgr: From %s to %s"), lpszOld, lpszNew);
  454. }
  455. #ifndef WINNT
  456. BOOL Group_DeleteIfRequired(LPCTSTR lpszOldGrpTitle, LPCTSTR lpszOldGrpFile)
  457. {
  458. BOOL fRet;
  459. HKEY hkeyNew;
  460. TCHAR szIniFile[MAX_PATH], szFile[MAX_PATH];
  461. if (Reg_GetString(g_hkeyGrpConv, c_szDelGroups, lpszOldGrpTitle, NULL, 0))
  462. {
  463. Win32DeleteFile(lpszOldGrpFile);
  464. DebugMsg(DM_TRACE, TEXT("gc.mgr: old group %s has been deleted"), lpszOldGrpTitle);
  465. // Remove old Group entry from registry..
  466. //
  467. if (RegOpenKey(g_hkeyGrpConv, c_szGroups, &hkeyNew) == ERROR_SUCCESS)
  468. {
  469. if (RegDeleteValue(hkeyNew, lpszOldGrpFile) == ERROR_SUCCESS)
  470. {
  471. fRet = TRUE;
  472. }
  473. RegCloseKey(hkeyNew);
  474. }
  475. // Remove old Group entry from progman.ini
  476. if (FindProgmanIni(szIniFile))
  477. {
  478. UINT uSize;
  479. LPTSTR pSection, pKey;
  480. for (uSize = 1024; uSize < 1024 * 8; uSize += 1024)
  481. {
  482. pSection = (PSTR)LocalAlloc(LPTR, uSize);
  483. if (!pSection)
  484. break;
  485. if ((UINT)GetPrivateProfileString(c_szGroups, NULL, c_szNULL, pSection, uSize / sizeof(pSection[0]), szIniFile) < uSize - 5)
  486. break;
  487. LocalFree((HLOCAL)pSection);
  488. pSection = NULL;
  489. fRet = FALSE;
  490. }
  491. if (pSection)
  492. {
  493. for (pKey = pSection; *pKey; pKey += lstrlen(pKey) + 1)
  494. {
  495. GetPrivateProfileString(c_szGroups, pKey, c_szNULL, szFile, ARRAYSIZE(szFile), szIniFile);
  496. if (lstrcmpi(lpszOldGrpFile, szFile) == 0)
  497. {
  498. WritePrivateProfileString(c_szGroups, pKey, NULL, szIniFile);
  499. break;
  500. }
  501. }
  502. LocalFree((HLOCAL)pSection);
  503. pSection = NULL;
  504. }
  505. }
  506. }
  507. else
  508. {
  509. fRet = FALSE;
  510. }
  511. return fRet;
  512. }
  513. #endif // !WINNT
  514. #undef PathRemoveExtension
  515. //---------------------------------------------------------------------------
  516. void PathRemoveExtension(LPTSTR pszPath)
  517. {
  518. LPTSTR pExt = PathFindExtension(pszPath);
  519. if (*pExt)
  520. {
  521. Assert(*pExt == TEXT('.'));
  522. *pExt = 0; // null out the "."
  523. }
  524. }
  525. //---------------------------------------------------------------------------
  526. // Given a path to an old group, create and return a path to where the new
  527. // group will be.
  528. BOOL Group_GenerateNewGroupPath(HWND hwnd, LPCTSTR lpszOldGrpTitle,
  529. LPTSTR lpszNewGrpPath, LPCTSTR pszOldGrpPath)
  530. {
  531. int iLen;
  532. TCHAR szGrpTitle[MAX_PATH];
  533. TCHAR szOldGrpTitle[32];
  534. // Get the location for all the special shell folders.
  535. if (g_fDoingCommonGroups)
  536. SHGetSpecialFolderPath(hwnd, lpszNewGrpPath, CSIDL_COMMON_PROGRAMS, TRUE);
  537. else
  538. SHGetSpecialFolderPath(hwnd, lpszNewGrpPath, CSIDL_PROGRAMS, TRUE);
  539. if (IsLFNDrive(lpszNewGrpPath))
  540. {
  541. // Fix it a bit.
  542. lstrcpyn(szOldGrpTitle, lpszOldGrpTitle, ARRAYSIZE(szOldGrpTitle));
  543. PathRemoveIllegalChars(szOldGrpTitle, 0, PRICF_NORMAL);
  544. // Munge the names so that things move to the new locations.
  545. MapGroupTitle(szOldGrpTitle, szGrpTitle, ARRAYSIZE(szGrpTitle));
  546. // Stick on the new group name.
  547. PathAddBackslash(lpszNewGrpPath);
  548. iLen = lstrlen(lpszNewGrpPath);
  549. // NB Don't use PathAppend() - very bad if there's a colons in the title.
  550. lstrcpyn(lpszNewGrpPath+iLen, szGrpTitle, MAX_PATH-iLen);
  551. PathRemoveIllegalChars(lpszNewGrpPath, iLen, PRICF_ALLOWSLASH);
  552. }
  553. else
  554. {
  555. // Just use the old group file name - this will make sure the group
  556. // names remain unique.
  557. PathAppend(lpszNewGrpPath, PathFindFileName(pszOldGrpPath));
  558. PathRemoveExtension(lpszNewGrpPath);
  559. }
  560. if (!PathFileExists(lpszNewGrpPath))
  561. {
  562. // Folder doesn't exist.
  563. // return Win32CreateDirectory(lpszNewGrpPath, NULL);
  564. return (SHCreateDirectory(hwnd, lpszNewGrpPath) == 0);
  565. }
  566. // Folder already exists.
  567. return TRUE;
  568. }
  569. //---------------------------------------------------------------------------
  570. // Returns true if the offsets given in the item def are valid-ish.
  571. BOOL CheckItemDef(LPITEMDEF lpitemdef, WORD cbGroup)
  572. {
  573. if (lpitemdef->pHeader < cbGroup && lpitemdef->pANDPlane < cbGroup &&
  574. lpitemdef->pXORPlane < cbGroup && lpitemdef->pName < cbGroup &&
  575. lpitemdef->pCommand < cbGroup && lpitemdef->pIconPath < cbGroup &&
  576. lpitemdef->pHeader && lpitemdef->pXORPlane && lpitemdef->pCommand)
  577. return TRUE;
  578. else
  579. {
  580. return FALSE;
  581. }
  582. }
  583. //---------------------------------------------------------------------------
  584. // Returns true if the offsets given in the item def are valid-ish.
  585. BOOL CheckItemDefNT(LPNT_ITEMDEF lpitemdef, DWORD cbGroup)
  586. {
  587. if (lpitemdef->pName < cbGroup &&
  588. lpitemdef->pCommand < cbGroup &&
  589. lpitemdef->pIconPath < cbGroup &&
  590. lpitemdef->pCommand)
  591. return TRUE;
  592. else
  593. {
  594. return FALSE;
  595. }
  596. }
  597. //---------------------------------------------------------------------------
  598. // Read the tags info from the given file handle from the given offset.
  599. #ifdef UNICODE
  600. void HandleTags(HANDLE fh, WORD oTags)
  601. #else
  602. void HandleTags(int fh, WORD oTags)
  603. #endif
  604. {
  605. LONG cbGroupReal;
  606. PMTAG pmtag;
  607. BOOL fTags = TRUE;
  608. TCHAR szText[MAX_PATH];
  609. BOOL fFirstTag = FALSE;
  610. LPPMITEM lppmitem;
  611. WORD wHotKey;
  612. #ifdef UNICODE
  613. DWORD dwBytesRead;
  614. #endif
  615. DebugMsg(DM_TRACE, TEXT("cg.ht: Reading tags."));
  616. #ifdef UNICODE
  617. cbGroupReal = SetFilePointer(fh, 0, NULL, FILE_END);
  618. #else
  619. cbGroupReal = (WORD) _llseek(fh, 0L, 2);
  620. #endif
  621. if (cbGroupReal <= (LONG) oTags)
  622. {
  623. // No tags in this file.
  624. return;
  625. }
  626. // Get to the tags section.
  627. #ifdef UNICODE
  628. SetFilePointer(fh, oTags, NULL, FILE_BEGIN);
  629. #else
  630. _llseek(fh, oTags, 0);
  631. #endif
  632. while (fTags)
  633. {
  634. #ifdef UNICODE
  635. if (!ReadFile(fh, &pmtag, SIZEOF(pmtag), &dwBytesRead, NULL) || dwBytesRead == 0) {
  636. fTags = FALSE;
  637. break;
  638. }
  639. #else
  640. fTags = _lread(fh, &pmtag, SIZEOF(pmtag));
  641. #endif
  642. switch (pmtag.wID)
  643. {
  644. case ID_MAGIC:
  645. {
  646. // DebugMsg(DM_TRACE, "gc.ht: First tag found.");
  647. fFirstTag = TRUE;
  648. #ifdef UNICODE
  649. SetFilePointer(fh, pmtag.cb - SIZEOF(PMTAG), NULL, FILE_CURRENT);
  650. #else
  651. _llseek(fh, pmtag.cb - SIZEOF(PMTAG), 1);
  652. #endif
  653. break;
  654. }
  655. case ID_LASTTAG:
  656. {
  657. // DebugMsg(DM_TRACE, "gc.ht: Last tag found.");
  658. fTags = FALSE;
  659. break;
  660. }
  661. case ID_APPLICATIONDIR:
  662. {
  663. // DebugMsg(DM_TRACE, "gc.ht: App dir %s found for %d.", (LPSTR) szText, pmtag.wItem);
  664. fgets(szText, ARRAYSIZE(szText), fh);
  665. lppmitem = DSA_GetItemPtr(hdsaPMItems, pmtag.wItem);
  666. if (lppmitem)
  667. {
  668. Str_SetPtr(&lppmitem->lpszCL, szText);
  669. }
  670. #ifdef DEBUG
  671. else
  672. {
  673. DebugMsg(DM_ERROR, TEXT("gc.ht: Item is invalid."));
  674. }
  675. #endif
  676. break;
  677. }
  678. case ID_HOTKEY:
  679. {
  680. // DebugMsg(DM_TRACE, "gc.ht: Hotkey found for %d.", pmtag.wItem);
  681. #ifdef UNICODE
  682. ReadFile(fh, &wHotKey, SIZEOF(wHotKey), &dwBytesRead, NULL);
  683. #else
  684. _lread(fh, &wHotKey, SIZEOF(wHotKey));
  685. #endif
  686. lppmitem = DSA_GetItemPtr(hdsaPMItems, pmtag.wItem);
  687. if (lppmitem)
  688. {
  689. lppmitem->wHotKey = wHotKey;
  690. }
  691. #ifdef DEBUG
  692. else
  693. {
  694. DebugMsg(DM_ERROR, TEXT("gc.ht: Item is invalid."));
  695. }
  696. #endif
  697. break;
  698. }
  699. case ID_MINIMIZE:
  700. {
  701. // DebugMsg(DM_TRACE, "gc.ht: Minimise flag found for %d.", pmtag.wItem);
  702. lppmitem = DSA_GetItemPtr(hdsaPMItems, pmtag.wItem);
  703. if (lppmitem)
  704. {
  705. lppmitem->nShowCmd = SW_SHOWMINNOACTIVE;
  706. }
  707. #ifdef DEBUG
  708. else
  709. {
  710. DebugMsg(DM_ERROR, TEXT("gc.ht: Item is invalid."));
  711. }
  712. #endif
  713. // Skip to the next tag.
  714. #ifdef UNICODE
  715. SetFilePointer(fh, pmtag.cb - SIZEOF(PMTAG), NULL, FILE_CURRENT);
  716. #else
  717. _llseek(fh, pmtag.cb - SIZEOF(PMTAG), 1);
  718. #endif
  719. break;
  720. }
  721. #ifdef WINNT
  722. case ID_NEWVDM:
  723. {
  724. // DebugMsg(DM_TRACE, "gc.ht: Separate VDM flag found for %d.", pmtag.wItem );
  725. lppmitem = DSA_GetItemPtr(hdsaPMItems, pmtag.wItem);
  726. if (lppmitem)
  727. {
  728. lppmitem->bSepVdm = TRUE;
  729. }
  730. #ifdef DEBUG
  731. else
  732. {
  733. DebugMsg(DM_ERROR, TEXT("gc.ht: Item is invalid."));
  734. }
  735. #endif
  736. // Skip to the next tag.
  737. #ifdef UNICODE
  738. SetFilePointer(fh, pmtag.cb - SIZEOF(PMTAG), NULL, FILE_CURRENT);
  739. #else
  740. _llseek(fh, pmtag.cb - SIZEOF(PMTAG), 1);
  741. #endif
  742. break;
  743. }
  744. #endif
  745. default:
  746. {
  747. // We've found something we don't understand but we haven't
  748. // found the first tag yet - probably a bust file.
  749. if (!fFirstTag)
  750. {
  751. DebugMsg(DM_TRACE, TEXT("gc.ht: No initial tag found - tags section is corrupt."));
  752. fTags = FALSE;
  753. }
  754. else
  755. {
  756. // Some unknown tag.
  757. if (pmtag.cb < SIZEOF(PMTAG))
  758. {
  759. // Can't continue!
  760. DebugMsg(DM_TRACE, TEXT("gc.ht: Tag has invalid size - ignoring remaining tags."));
  761. fTags = FALSE;
  762. }
  763. else
  764. {
  765. // Just ignore its data and continue.
  766. #ifdef UNICODE
  767. SetFilePointer(fh, pmtag.cb - SIZEOF(PMTAG), NULL, FILE_CURRENT);
  768. #else
  769. _llseek(fh, pmtag.cb - SIZEOF(PMTAG), 1);
  770. #endif
  771. }
  772. }
  773. break;
  774. }
  775. }
  776. }
  777. }
  778. //---------------------------------------------------------------------------
  779. // Read the tags info from the given file handle from the given offset.
  780. #ifdef UNICODE
  781. void HandleTagsNT(HANDLE fh, DWORD oTags)
  782. #else
  783. void HandleTagsNT(int fh, DWORD oTags)
  784. #endif
  785. {
  786. DWORD cbGroupReal;
  787. DWORD dwPosition;
  788. NT_PMTAG pmtag;
  789. BOOL fTags = TRUE;
  790. WCHAR wszTemp[MAX_PATH];
  791. TCHAR szText[MAX_PATH];
  792. BOOL fFirstTag = FALSE;
  793. LPPMITEM lppmitem;
  794. WORD wHotKey;
  795. #ifdef UNICODE
  796. DWORD dwBytesRead;
  797. #endif
  798. DebugMsg(DM_TRACE, TEXT("cg.ht: Reading tags."));
  799. #ifdef UNICODE
  800. cbGroupReal = SetFilePointer(fh, 0, NULL, FILE_END);
  801. #else
  802. cbGroupReal = _llseek(fh, 0L, 2);
  803. #endif
  804. if (cbGroupReal <= oTags)
  805. {
  806. // No tags in this file.
  807. return;
  808. }
  809. // Get to the tags section.
  810. dwPosition = oTags;
  811. while (fTags)
  812. {
  813. #ifdef UNICODE
  814. SetFilePointer(fh, dwPosition, NULL, FILE_BEGIN);
  815. if (!ReadFile(fh, &pmtag, SIZEOF(pmtag), &dwBytesRead, NULL) || dwBytesRead == 0) {
  816. fTags = FALSE;
  817. break;
  818. }
  819. #else
  820. _llseek(fh,dwPosition,0);
  821. fTags = _lread(fh, &pmtag, SIZEOF(pmtag));
  822. #endif
  823. switch (pmtag.wID)
  824. {
  825. case ID_MAGIC:
  826. {
  827. // DebugMsg(DM_TRACE, "gc.ht: First tag found.");
  828. fFirstTag = TRUE;
  829. dwPosition += pmtag.cb;
  830. break;
  831. }
  832. case ID_LASTTAG:
  833. {
  834. // DebugMsg(DM_TRACE, "gc.ht: Last tag found.");
  835. fTags = FALSE;
  836. break;
  837. }
  838. case ID_APPLICATIONDIR:
  839. {
  840. #ifdef UNICODE
  841. SetFilePointer(fh, dwPosition+FIELD_OFFSET(NT_PMTAG,rgb[0]), NULL, FILE_BEGIN);
  842. ReadFile(fh, wszTemp, SIZEOF(wszTemp), &dwBytesRead, NULL);
  843. lstrcpy(szText, wszTemp);
  844. #else
  845. _llseek(fh,dwPosition+FIELD_OFFSET(NT_PMTAG,rgb[0]),0);
  846. _lread(fh,wszTemp,SIZEOF(wszTemp));
  847. WideCharToMultiByte (CP_ACP, 0, wszTemp, -1,
  848. szText, ARRAYSIZE(szText), NULL, NULL);
  849. #endif
  850. // DebugMsg(DM_TRACE, "gc.ht: App dir %s found for %d.", (LPSTR) szText, pmtag.wItem);
  851. lppmitem = DSA_GetItemPtr(hdsaPMItems, pmtag.wItem);
  852. if (lppmitem)
  853. {
  854. Str_SetPtr(&lppmitem->lpszCL, szText);
  855. }
  856. #ifdef DEBUG
  857. else
  858. {
  859. DebugMsg(DM_ERROR, TEXT("gc.ht: Item is invalid."));
  860. }
  861. #endif
  862. dwPosition += pmtag.cb;
  863. break;
  864. }
  865. case ID_HOTKEY:
  866. {
  867. // DebugMsg(DM_TRACE, "gc.ht: Hotkey found for %d.", pmtag.wItem);
  868. #ifdef UNICODE
  869. ReadFile(fh, &wHotKey, SIZEOF(wHotKey), &dwBytesRead, NULL);
  870. #else
  871. _lread(fh, &wHotKey, SIZEOF(wHotKey));
  872. #endif
  873. lppmitem = DSA_GetItemPtr(hdsaPMItems, pmtag.wItem);
  874. if (lppmitem)
  875. {
  876. lppmitem->wHotKey = wHotKey;
  877. }
  878. #ifdef DEBUG
  879. else
  880. {
  881. DebugMsg(DM_ERROR, TEXT("gc.ht: Item is invalid."));
  882. }
  883. #endif
  884. dwPosition += pmtag.cb;
  885. break;
  886. }
  887. case ID_MINIMIZE:
  888. {
  889. // DebugMsg(DM_TRACE, "gc.ht: Minimise flag found for %d.", pmtag.wItem);
  890. lppmitem = DSA_GetItemPtr(hdsaPMItems, pmtag.wItem);
  891. if (lppmitem)
  892. {
  893. lppmitem->nShowCmd = SW_SHOWMINNOACTIVE;
  894. }
  895. #ifdef DEBUG
  896. else
  897. {
  898. DebugMsg(DM_ERROR, TEXT("gc.ht: Item is invalid."));
  899. }
  900. #endif
  901. // Skip to the next tag.
  902. dwPosition += pmtag.cb;
  903. break;
  904. }
  905. #ifdef WINNT
  906. case ID_NEWVDM:
  907. {
  908. // DebugMsg(DM_TRACE, "gc.ht: Separate VDM flag found for %d.", pmtag.wItem );
  909. lppmitem = DSA_GetItemPtr(hdsaPMItems, pmtag.wItem);
  910. if (lppmitem)
  911. {
  912. lppmitem->bSepVdm = TRUE;
  913. }
  914. #ifdef DEBUG
  915. else
  916. {
  917. DebugMsg(DM_ERROR, TEXT("gc.ht: Item is invalid."));
  918. }
  919. #endif
  920. // Skip to the next tag.
  921. dwPosition += pmtag.cb;
  922. break;
  923. }
  924. #endif
  925. default:
  926. {
  927. // We've found something we don't understand but we haven't
  928. // found the first tag yet - probably a bust file.
  929. if (!fFirstTag)
  930. {
  931. DebugMsg(DM_TRACE, TEXT("gc.ht: No initial tag found - tags section is corrupt."));
  932. fTags = FALSE;
  933. }
  934. else
  935. {
  936. // Some unknown tag.
  937. if (pmtag.cb < SIZEOF(PMTAG))
  938. {
  939. // Can't continue!
  940. DebugMsg(DM_TRACE, TEXT("gc.ht: Tag has invalid size - ignoring remaining tags."));
  941. fTags = FALSE;
  942. }
  943. else
  944. {
  945. // Just ignore its data and continue.
  946. dwPosition += pmtag.cb;
  947. }
  948. }
  949. break;
  950. }
  951. }
  952. }
  953. }
  954. //---------------------------------------------------------------------------
  955. void DeleteBustedItems(void)
  956. {
  957. int i, cItems;
  958. LPPMITEM ppmitem;
  959. cItems = DSA_GetItemCount(hdsaPMItems);
  960. for (i=0; i<cItems; i++)
  961. {
  962. ppmitem = DSA_GetItemPtr(hdsaPMItems, i);
  963. // Is the item broken?
  964. if (!ppmitem->lpszDesc || !(*ppmitem->lpszDesc))
  965. {
  966. // Yep, delete it.
  967. DSA_DeleteItem(hdsaPMItems, i);
  968. cItems--;
  969. i--;
  970. }
  971. }
  972. }
  973. //---------------------------------------------------------------------------
  974. void ShortenDescriptions(void)
  975. {
  976. int i, cItems;
  977. LPPMITEM ppmitem;
  978. cItems = DSA_GetItemCount(hdsaPMItems);
  979. for (i=0; i<cItems; i++)
  980. {
  981. ppmitem = DSA_GetItemPtr(hdsaPMItems, i);
  982. // Shorten the descriptions
  983. lstrcpyn(ppmitem->lpszDesc, ppmitem->lpszDesc, 9);
  984. }
  985. }
  986. //---------------------------------------------------------------------------
  987. // Kinda like PathFindFileName() but handles things like c:\foo\ differently
  988. // to match progmans code.
  989. LPTSTR WINAPI _PathFindFileName(LPCTSTR pPath)
  990. {
  991. LPCTSTR pT;
  992. for (pT = pPath; *pPath; pPath = CharNext(pPath)) {
  993. if ((pPath[0] == TEXT('\\') || pPath[0] == TEXT(':')) && (pPath[1] != TEXT('\\')))
  994. pT = pPath + 1;
  995. }
  996. return (LPTSTR)pT; // const -> non const
  997. }
  998. //---------------------------------------------------------------------------
  999. // Take a 3.1 format WD and exe and convert them to the new style.
  1000. // NB Old style was WD+exename and exepath - new style is exepath+exename and
  1001. // WD.
  1002. void MungePaths(void)
  1003. {
  1004. LPTSTR lpszFileName; // Ptr to filename part (plus params).
  1005. LPTSTR lpszParams; // Ptr to first char of params.
  1006. TCHAR szCL[MAX_PATH];
  1007. TCHAR szWD[MAX_PATH];
  1008. int i, cItems;
  1009. LPPMITEM lppmitem;
  1010. cItems = DSA_GetItemCount(hdsaPMItems);
  1011. for (i=0; i<cItems; i++)
  1012. {
  1013. szCL[0] = TEXT('\0');
  1014. szWD[0] = TEXT('\0');
  1015. lppmitem = DSA_GetItemPtr(hdsaPMItems, i);
  1016. // Get the current command line.
  1017. Str_GetPtr(lppmitem->lpszCL, szCL, ARRAYSIZE(szCL));
  1018. // Get the current working dir.
  1019. Str_GetPtr(lppmitem->lpszWD, szWD, ARRAYSIZE(szWD));
  1020. #ifdef OLDWAY
  1021. // Find the filename part...
  1022. // Params will confuse PFFN.
  1023. lpszParams = PathGetArgs(szWD);
  1024. if (*lpszParams)
  1025. {
  1026. // Chop them off.
  1027. // NB Previous char is a space by definition.
  1028. *(lpszParams-1) = TEXT('\0');
  1029. lpszFileName = _PathFindFileName(szWD);
  1030. // Put them back
  1031. *(lpszParams-1) = TEXT(' ');
  1032. }
  1033. else
  1034. {
  1035. // No params.
  1036. lpszFileName = PathFindFileName(szWD);
  1037. }
  1038. // Copy this onto the exe path.
  1039. lstrcat((LPTSTR) szCL, lpszFileName);
  1040. // Remove it from the end of the WD.
  1041. *lpszFileName = TEXT('\0');
  1042. // For anything but things like c:\ remove the last slash.
  1043. if (!PathIsRoot(szWD))
  1044. {
  1045. *(lpszFileName-1) = TEXT('\0');
  1046. }
  1047. #else
  1048. lpszFileName = szWD;
  1049. if (*lpszFileName == TEXT('"'))
  1050. {
  1051. while (lpszFileName)
  1052. {
  1053. lpszFileName = StrChr(lpszFileName+1,TEXT('"'));
  1054. if (!lpszFileName)
  1055. {
  1056. //
  1057. // The directory is not in quotes and since the command
  1058. // path starts with a quote, there is no working directory.
  1059. //
  1060. lpszFileName = szWD;
  1061. break;
  1062. }
  1063. if (*(lpszFileName+1) == TEXT('\\'))
  1064. {
  1065. //
  1066. // The working directory is in quotes.
  1067. //
  1068. lpszFileName++;
  1069. break;
  1070. }
  1071. }
  1072. }
  1073. else
  1074. {
  1075. //
  1076. // if there's a working directory, it is not in quotes
  1077. // Copy up until the last \ preceding any quote, space, or the end
  1078. //
  1079. LPTSTR lpEnd = lpszFileName;
  1080. while (*lpszFileName && *lpszFileName != TEXT('"') && *lpszFileName != TEXT(' '))
  1081. {
  1082. if ((*lpszFileName == TEXT('\\') || *lpszFileName == TEXT(':')) && *(lpszFileName+1) != TEXT('\\'))
  1083. lpEnd = lpszFileName;
  1084. lpszFileName = CharNext(lpszFileName);
  1085. }
  1086. lpszFileName = lpEnd;
  1087. }
  1088. //
  1089. // If the split is at the beginning,
  1090. // then there is no working dir
  1091. //
  1092. if (lpszFileName == szWD)
  1093. {
  1094. lstrcat(szCL, szWD);
  1095. szWD[0] = TEXT('\0');
  1096. }
  1097. else
  1098. {
  1099. lstrcat(szCL, lpszFileName+1);
  1100. *(lpszFileName+1) = TEXT('\0'); // Split it.
  1101. //
  1102. // Remove quotes from the working dir NOW.
  1103. //
  1104. if (szWD[0] == TEXT('"')) {
  1105. LPTSTR lpTemp;
  1106. for (lpTemp = szWD+1; *lpTemp && *lpTemp != TEXT('"'); lpTemp++)
  1107. *(lpTemp-1) = *lpTemp;
  1108. if (*lpTemp == TEXT('"')) {
  1109. *(lpTemp-1) = TEXT('\0');
  1110. }
  1111. }
  1112. // For anything but things like c:\ remove the last slash.
  1113. if (!PathIsRoot(szWD))
  1114. {
  1115. *lpszFileName = TEXT('\0');
  1116. }
  1117. }
  1118. #endif
  1119. // Replace the data.
  1120. Str_SetPtr(&lppmitem->lpszCL, szCL);
  1121. Str_SetPtr(&lppmitem->lpszWD, szWD);
  1122. // DebugMsg(DM_TRACE, "gc.mp: Exe %s, WD %s", (LPSTR)szCL, (LPSTR)szWD);
  1123. }
  1124. }
  1125. //---------------------------------------------------------------------------
  1126. // Set all the fields of the given pmitem to clear;
  1127. void PMItem_Clear(LPPMITEM lppmitem)
  1128. {
  1129. lppmitem->lpszDesc = NULL;
  1130. lppmitem->lpszCL = NULL;
  1131. lppmitem->lpszWD = NULL;
  1132. lppmitem->lpszIconPath = NULL;
  1133. lppmitem->wiIcon = 0;
  1134. lppmitem->wHotKey = 0;
  1135. lppmitem->nShowCmd = SW_SHOWNORMAL;
  1136. #ifdef WINNT
  1137. lppmitem->bSepVdm = FALSE;
  1138. #endif
  1139. }
  1140. //---------------------------------------------------------------------------
  1141. // Read the item data from the file and add it to the list.
  1142. // Returns TRUE if everything went perfectly.
  1143. #ifdef UNICODE
  1144. BOOL GetAllItemData(HANDLE fh, WORD cItems, WORD cbGroup, LPTSTR lpszOldGrpTitle, LPTSTR lpszNewGrpPath)
  1145. #else
  1146. BOOL GetAllItemData(HFILE fh, WORD cItems, WORD cbGroup, LPTSTR lpszOldGrpTitle, LPTSTR lpszNewGrpPath)
  1147. #endif
  1148. {
  1149. UINT cbItemArray;
  1150. WORD *rgItems;
  1151. UINT i, iItem;
  1152. TCHAR szDesc[CCHSZNORMAL];
  1153. TCHAR szCL[CCHSZNORMAL];
  1154. TCHAR szIconPath[CCHSZNORMAL];
  1155. ITEMDEF itemdef;
  1156. BOOL fOK = TRUE;
  1157. UINT cbRead;
  1158. PMITEM pmitem;
  1159. #ifdef UNICODE
  1160. DWORD dwBytesRead;
  1161. #endif
  1162. // Read in the old item table...
  1163. iItem = 0;
  1164. cbItemArray = cItems * SIZEOF(*rgItems);
  1165. rgItems = (WORD *)LocalAlloc(LPTR, cbItemArray);
  1166. if (!rgItems)
  1167. {
  1168. DebugMsg(DM_ERROR, TEXT("gc.gcnfo: Out of memory."));
  1169. return FALSE;
  1170. }
  1171. #ifdef UNICODE
  1172. SetFilePointer(fh, SIZEOF(GROUPDEF), NULL, FILE_BEGIN);
  1173. ReadFile(fh, rgItems, cbItemArray, &dwBytesRead, NULL);
  1174. #else
  1175. _llseek(fh, SIZEOF(GROUPDEF), 0);
  1176. _lread(fh, rgItems, cbItemArray);
  1177. #endif
  1178. // Show progress in two stages, first reading then writing.
  1179. Group_SetProgressNameAndRange(lpszNewGrpPath, (cItems*2)-1);
  1180. // Read in the items.
  1181. // NB Don't just skip busted items since the tag data contains
  1182. // indices to items and that includes busted ones. Just use
  1183. // an empty description to indicate that the link is invalid.
  1184. for (i=0; i<cItems; i++)
  1185. {
  1186. Group_SetProgress(i);
  1187. szDesc[0] = TEXT('\0');
  1188. szCL[0] = TEXT('\0');
  1189. szIconPath[0] = TEXT('\0');
  1190. itemdef.iIcon = 0;
  1191. if (rgItems[i] == 0)
  1192. {
  1193. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file has empty item definition - skipping."));
  1194. goto AddItem;
  1195. }
  1196. if (rgItems[i] > cbGroup)
  1197. {
  1198. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file busted (item entry in invalid part of file) - skipping item."));
  1199. fOK = FALSE;
  1200. goto AddItem;
  1201. }
  1202. #ifdef UNICODE
  1203. SetFilePointer(fh, rgItems[i], NULL, FILE_BEGIN);
  1204. ReadFile(fh, &itemdef, SIZEOF(itemdef), &cbRead, NULL);
  1205. #else
  1206. _llseek(fh, rgItems[i], 0);
  1207. cbRead = _lread(fh, &itemdef, SIZEOF(itemdef));
  1208. #endif
  1209. if (cbRead != SIZEOF(itemdef))
  1210. {
  1211. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file busted (invalid definition) - skipping item %d."), i);
  1212. fOK = FALSE;
  1213. goto AddItem;
  1214. }
  1215. if (!CheckItemDef(&itemdef, cbGroup))
  1216. {
  1217. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file busted (invalid item field) - skipping item %d."), i);
  1218. fOK = FALSE;
  1219. goto AddItem;
  1220. }
  1221. #ifdef UNICODE
  1222. SetFilePointer(fh, itemdef.pName, NULL, FILE_BEGIN);
  1223. #else
  1224. _llseek(fh, itemdef.pName, 0);
  1225. #endif
  1226. fgets(szDesc, SIZEOF(szDesc), fh);
  1227. if (!*szDesc)
  1228. {
  1229. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file busted (empty name) - skipping item %d."), i);
  1230. fOK = FALSE;
  1231. goto AddItem;
  1232. }
  1233. #ifdef UNICODE
  1234. SetFilePointer(fh, itemdef.pCommand, NULL, FILE_BEGIN);
  1235. #else
  1236. _llseek(fh, itemdef.pCommand, 0);
  1237. #endif
  1238. fgets(szCL, SIZEOF(szCL), fh);
  1239. // We hit this case with links to c:\ (rare, very rare).
  1240. #if 0
  1241. if (!*szCL)
  1242. {
  1243. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file busted (empty command line) - skipping item %d."), i);
  1244. // We use a null description to signal a problem with this item.
  1245. szDesc[0] = TEXT('\0');
  1246. fOK = FALSE;
  1247. goto AddItem;
  1248. }
  1249. #endif
  1250. if (itemdef.pIconPath!=0xFFFF)
  1251. {
  1252. #ifdef UNICODE
  1253. SetFilePointer(fh, itemdef.pIconPath, NULL, FILE_BEGIN);
  1254. #else
  1255. _llseek(fh, itemdef.pIconPath, 0);
  1256. #endif
  1257. fgets(szIconPath, SIZEOF(szIconPath), fh);
  1258. }
  1259. else
  1260. {
  1261. szIconPath[ 0 ] = TEXT('\0');
  1262. }
  1263. if (!*szIconPath)
  1264. {
  1265. // NB Do nothing. Empty icon paths are legal - associated apps where the associated
  1266. // app is missing will have an empty icon path.
  1267. }
  1268. // NB Forget about the icon data.
  1269. // DebugMsg(DM_TRACE, "gc.gcnfo: Found item %s.", (LPSTR) szDesc);
  1270. // Store away the data....
  1271. // NB We load the old commands line into the working dir field because
  1272. // only the leaf is the command, the rest is the WD. Once we've been
  1273. // through the tags section we can sort out the mess.
  1274. AddItem:
  1275. PMItem_Clear(&pmitem);
  1276. #ifdef DEBUG
  1277. DebugMsg(GC_TRACE, TEXT("gc.gaid: Desc %s"), (LPTSTR) szDesc);
  1278. DebugMsg(GC_TRACE, TEXT(" WD: %s"), (LPTSTR) szCL);
  1279. DebugMsg(GC_TRACE, TEXT(" IP: %s(%d)"), (LPTSTR) szIconPath, itemdef.iIcon);
  1280. #endif
  1281. // Don't store anything for items with invalid descriptions.
  1282. if (*szDesc)
  1283. {
  1284. // Remove illegal chars.
  1285. PathRemoveIllegalChars(szDesc, 0, PRICF_NORMAL);
  1286. Str_SetPtr(&pmitem.lpszDesc, szDesc);
  1287. Str_SetPtr(&pmitem.lpszWD, szCL);
  1288. Str_SetPtr(&pmitem.lpszIconPath, szIconPath);
  1289. pmitem.wiIcon = itemdef.iIcon;
  1290. }
  1291. DSA_InsertItem(hdsaPMItems, iItem, &pmitem);
  1292. iItem++;
  1293. }
  1294. LocalFree((HLOCAL)rgItems);
  1295. return fOK;
  1296. }
  1297. //-----------------------------------------------------------------------------
  1298. // Functions to try to find out which icon was appropriate given the NT icon
  1299. // identifier number (the identifier for the RT_ICON resource only).
  1300. //-----------------------------------------------------------------------------
  1301. typedef struct _enumstruct {
  1302. UINT iIndex;
  1303. BOOL fFound;
  1304. WORD wIconRTIconID;
  1305. } ENUMSTRUCT, *LPENUMSTRUCT;
  1306. BOOL EnumIconFunc(
  1307. HMODULE hMod,
  1308. LPCTSTR lpType,
  1309. LPTSTR lpName,
  1310. LPARAM lParam
  1311. ) {
  1312. HANDLE h;
  1313. PBYTE p;
  1314. int id;
  1315. LPENUMSTRUCT lpes = (LPENUMSTRUCT)lParam;
  1316. if (!lpName)
  1317. return TRUE;
  1318. h = FindResource(hMod, lpName, lpType);
  1319. if (!h)
  1320. return TRUE;
  1321. h = LoadResource(hMod, h);
  1322. p = LockResource(h);
  1323. id = LookupIconIdFromDirectory(p, TRUE);
  1324. UnlockResource(h);
  1325. FreeResource(h);
  1326. if (id == lpes->wIconRTIconID)
  1327. {
  1328. lpes->fFound = TRUE;
  1329. return FALSE;
  1330. }
  1331. lpes->iIndex++;
  1332. return TRUE;
  1333. }
  1334. WORD FindAppropriateIcon( LPTSTR lpszFileName, WORD wIconRTIconID )
  1335. {
  1336. HINSTANCE hInst;
  1337. TCHAR szExe[MAX_PATH];
  1338. WORD wIcon = wIconRTIconID;
  1339. ENUMSTRUCT es;
  1340. int olderror;
  1341. hInst = FindExecutable(lpszFileName,NULL,szExe);
  1342. if ( hInst <= (HINSTANCE)HINSTANCE_ERROR )
  1343. {
  1344. return 0;
  1345. }
  1346. olderror = SetErrorMode(SEM_FAILCRITICALERRORS);
  1347. hInst = LoadLibraryEx(szExe,NULL, DONT_RESOLVE_DLL_REFERENCES);
  1348. SetErrorMode(olderror);
  1349. if ( hInst <= (HINSTANCE)HINSTANCE_ERROR )
  1350. {
  1351. return 0;
  1352. }
  1353. es.iIndex = 0;
  1354. es.fFound = FALSE;
  1355. es.wIconRTIconID = wIconRTIconID;
  1356. EnumResourceNames( hInst, RT_GROUP_ICON, EnumIconFunc, (LPARAM)&es );
  1357. FreeLibrary( hInst );
  1358. if (es.fFound)
  1359. {
  1360. return (WORD)es.iIndex;
  1361. }
  1362. else
  1363. {
  1364. return 0;
  1365. }
  1366. }
  1367. //---------------------------------------------------------------------------
  1368. // Read the item data from the file and add it to the list.
  1369. // Returns TRUE if everything went perfectly.
  1370. #ifdef UNICODE
  1371. BOOL GetAllItemDataNT(HANDLE fh, WORD cItems, DWORD cbGroup, LPTSTR lpszOldGrpTitle, LPTSTR lpszNewGrpPath)
  1372. #else
  1373. BOOL GetAllItemDataNT(HFILE fh, WORD cItems, DWORD cbGroup, LPTSTR lpszOldGrpTitle, LPTSTR lpszNewGrpPath)
  1374. #endif
  1375. {
  1376. UINT cbItemArray;
  1377. DWORD *rgItems;
  1378. UINT i, iItem;
  1379. WCHAR wszTemp[CCHSZNORMAL];
  1380. TCHAR szDesc[CCHSZNORMAL];
  1381. TCHAR szCL[CCHSZNORMAL];
  1382. TCHAR szIconPath[CCHSZNORMAL];
  1383. NT_ITEMDEF itemdef;
  1384. BOOL fOK = TRUE;
  1385. #ifdef UNICODE
  1386. DWORD cbRead;
  1387. #else
  1388. UINT cbRead;
  1389. #endif
  1390. PMITEM pmitem;
  1391. // Read in the old item table...
  1392. iItem = 0;
  1393. cbItemArray = cItems * SIZEOF(*rgItems);
  1394. rgItems = (DWORD *)LocalAlloc(LPTR, cbItemArray);
  1395. if (!rgItems)
  1396. {
  1397. DebugMsg(DM_ERROR, TEXT("gc.gcnfo: Out of memory."));
  1398. return FALSE;
  1399. }
  1400. #ifdef UNICODE
  1401. SetFilePointer(fh, FIELD_OFFSET(NT_GROUPDEF,rgiItems[0]), NULL, FILE_BEGIN);
  1402. ReadFile(fh, rgItems, cbItemArray, &cbRead, NULL);
  1403. #else
  1404. _llseek(fh, FIELD_OFFSET(NT_GROUPDEF,rgiItems[0]), 0);
  1405. _lread(fh, rgItems, cbItemArray);
  1406. #endif
  1407. // Show progress in two stages, first reading then writing.
  1408. Group_SetProgressNameAndRange(lpszNewGrpPath, (cItems*2)-1);
  1409. // Read in the items.
  1410. // NB Don't just skip busted items since the tag data contains
  1411. // indices to items and that includes busted ones. Just use
  1412. // an empty description to indicate that the link is invalid.
  1413. for (i=0; i<cItems; i++)
  1414. {
  1415. Group_SetProgress(i);
  1416. szDesc[0] = TEXT('\0');
  1417. szCL[0] = TEXT('\0');
  1418. szIconPath[0] = TEXT('\0');
  1419. itemdef.iIcon = 0;
  1420. if (rgItems[i] == 0)
  1421. {
  1422. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file has empty item definition - skipping."));
  1423. goto AddItem;
  1424. }
  1425. if (rgItems[i] > cbGroup)
  1426. {
  1427. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file busted (item entry in invalid part of file) - skipping item."));
  1428. fOK = FALSE;
  1429. goto AddItem;
  1430. }
  1431. #ifdef UNICODE
  1432. SetFilePointer(fh, rgItems[i], NULL, FILE_BEGIN);
  1433. ReadFile(fh, &itemdef, SIZEOF(itemdef), &cbRead, NULL);
  1434. #else
  1435. _llseek(fh, rgItems[i], 0);
  1436. cbRead = _lread(fh, &itemdef, SIZEOF(itemdef));
  1437. #endif
  1438. if (cbRead != SIZEOF(itemdef))
  1439. {
  1440. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file busted (invalid definition) - skipping item %d."), i);
  1441. fOK = FALSE;
  1442. goto AddItem;
  1443. }
  1444. if (!CheckItemDefNT(&itemdef, cbGroup))
  1445. {
  1446. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file busted (invalid item field) - skipping item %d."), i);
  1447. fOK = FALSE;
  1448. goto AddItem;
  1449. }
  1450. #ifdef UNICODE
  1451. SetFilePointer(fh, itemdef.pName, NULL, FILE_BEGIN);
  1452. ReadFile(fh, wszTemp, SIZEOF(wszTemp), &cbRead, NULL);
  1453. #else
  1454. _llseek(fh, itemdef.pName, 0);
  1455. _lread(fh, wszTemp, SIZEOF(wszTemp)); // There will be a NUL somewhere
  1456. #endif
  1457. if (!*wszTemp)
  1458. {
  1459. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file busted (empty name) - skipping item %d."), i);
  1460. fOK = FALSE;
  1461. goto AddItem;
  1462. }
  1463. #ifdef UNICODE
  1464. lstrcpy(szDesc, wszTemp);
  1465. #else
  1466. WideCharToMultiByte (CP_ACP, 0, wszTemp, -1,
  1467. szDesc, ARRAYSIZE(szDesc), NULL, NULL);
  1468. #endif
  1469. #ifdef UNICODE
  1470. SetFilePointer(fh, itemdef.pCommand, NULL, FILE_BEGIN);
  1471. ReadFile(fh, &wszTemp, SIZEOF(wszTemp), &cbRead, NULL);
  1472. #else
  1473. _llseek(fh, itemdef.pCommand, 0);
  1474. _lread(fh, wszTemp, SIZEOF(wszTemp));
  1475. #endif
  1476. if (!*wszTemp)
  1477. {
  1478. DebugMsg(DM_TRACE, TEXT("gc.gcnfo: Old group file busted (empty command line) - skipping item %d."), i);
  1479. // We use a null description to signal a problem with this item.
  1480. szDesc[0] = TEXT('\0');
  1481. fOK = FALSE;
  1482. goto AddItem;
  1483. }
  1484. #ifdef UNICODE
  1485. lstrcpy(szCL, wszTemp);
  1486. #else
  1487. WideCharToMultiByte (CP_ACP, 0, wszTemp, -1,
  1488. szCL, ARRAYSIZE(szCL), NULL, NULL);
  1489. #endif
  1490. #ifdef UNICODE
  1491. SetFilePointer(fh, itemdef.pIconPath, NULL, FILE_BEGIN);
  1492. ReadFile(fh, wszTemp, SIZEOF(wszTemp), &cbRead, NULL);
  1493. #else
  1494. _llseek(fh, itemdef.pIconPath, 0);
  1495. _lread(fh, wszTemp, SIZEOF(wszTemp));
  1496. #endif
  1497. if (!*wszTemp)
  1498. {
  1499. // NB Do nothing. Empty icon paths are legal - associated apps where the associated
  1500. // app is missing will have an empty icon path.
  1501. }
  1502. #ifdef UNICODE
  1503. lstrcpy(szIconPath, wszTemp);
  1504. #else
  1505. WideCharToMultiByte (CP_ACP, 0, wszTemp, -1,
  1506. szIconPath, ARRAYSIZE(szIconPath), NULL, NULL);
  1507. #endif
  1508. // NB Forget about the icon data.
  1509. // DebugMsg(DM_TRACE, "gc.gcnfo: Found item %s.", (LPSTR) szDesc);
  1510. // Store away the data....
  1511. // NB We load the old commands line into the working dir field because
  1512. // only the leaf is the command, the rest is the WD. Once we've been
  1513. // through the tags section we can sort out the mess.
  1514. AddItem:
  1515. PMItem_Clear(&pmitem);
  1516. #ifdef DEBUG
  1517. DebugMsg(GC_TRACE, TEXT("gc.gaid: Desc %s"), (LPTSTR) szDesc);
  1518. DebugMsg(GC_TRACE, TEXT(" WD: %s"), (LPTSTR) szCL);
  1519. DebugMsg(GC_TRACE, TEXT(" IP: %s(%d)"), (LPTSTR) szIconPath, itemdef.iIcon);
  1520. #endif
  1521. // Don't store anything for items with invalid descriptions.
  1522. if (*szDesc)
  1523. {
  1524. WORD wIconIndex;
  1525. // Remove illegal chars.
  1526. PathRemoveIllegalChars(szDesc, 0, PRICF_NORMAL);
  1527. Str_SetPtr(&pmitem.lpszDesc, szDesc);
  1528. Str_SetPtr(&pmitem.lpszWD, szCL);
  1529. Str_SetPtr(&pmitem.lpszIconPath, szIconPath);
  1530. wIconIndex = itemdef.wIconIndex;
  1531. if ( wIconIndex == 0 )
  1532. {
  1533. WORD wIcon;
  1534. HICON hIcon;
  1535. if ( *szIconPath == TEXT('\0') )
  1536. {
  1537. FindExecutable(szCL,NULL,szIconPath);
  1538. }
  1539. if ( *szIconPath != TEXT('\0') )
  1540. {
  1541. wIconIndex = FindAppropriateIcon( szIconPath, itemdef.iIcon);
  1542. }
  1543. }
  1544. pmitem.wiIcon = wIconIndex;
  1545. }
  1546. DSA_InsertItem(hdsaPMItems, iItem, &pmitem);
  1547. iItem++;
  1548. }
  1549. LocalFree((HLOCAL)rgItems);
  1550. return fOK;
  1551. }
  1552. //---------------------------------------------------------------------------
  1553. // Create the links in the given dest dir.
  1554. void CreateLinks(LPCTSTR lpszNewGrpPath, BOOL fStartup, INT cItemsStart)
  1555. {
  1556. int i, cItems;
  1557. TCHAR szLinkName[MAX_PATH];
  1558. TCHAR szBuffer[MAX_PATH];
  1559. // we make this 3*MAX_PATH so that DARWIN and LOGO3 callers can pass their extra information
  1560. TCHAR szExpBuff[3*MAX_PATH];
  1561. WCHAR wszPath[MAX_PATH];
  1562. LPTSTR lpszArgs;
  1563. LPCTSTR dirs[2];
  1564. IShellLink *psl;
  1565. LPTSTR pszExt;
  1566. if (SUCCEEDED(ICoCreateInstance(&CLSID_ShellLink, &IID_IShellLink, &psl))) {
  1567. IPersistFile *ppf;
  1568. psl->lpVtbl->QueryInterface(psl, &IID_IPersistFile, &ppf);
  1569. cItems = DSA_GetItemCount(hdsaPMItems);
  1570. for (i = 0; i < cItems; i++) {
  1571. LPPMITEM lppmitem = DSA_GetItemPtr(hdsaPMItems, i);
  1572. // We show the progress in 2 halves.
  1573. Group_SetProgress(cItemsStart+(i*cItemsStart/cItems));
  1574. // command line and args.
  1575. // if this command line points to net drives we should add
  1576. // the UNC mapping to the link
  1577. Str_GetPtr(lppmitem->lpszCL, szBuffer, ARRAYSIZE(szBuffer));
  1578. // Spaces at the begining of the CL will confuse us.
  1579. PathRemoveBlanks(szBuffer);
  1580. lpszArgs = PathGetArgs(szBuffer);
  1581. if (*lpszArgs)
  1582. *(lpszArgs-1) = TEXT('\0');
  1583. // NB Special case, remove all links to Progman[.exe] from the
  1584. // Startup Group. A lot of people put it there to give it a hotkey.
  1585. // We want to be able to delete it regardless of its name ie we
  1586. // can't just use setup.ini to do the work.
  1587. if (fStartup)
  1588. {
  1589. if ((lstrcmpi(c_szProgmanExe, PathFindFileName(szBuffer)) == 0) ||
  1590. (lstrcmpi(c_szProgman, PathFindFileName(szBuffer)) == 0))
  1591. continue;
  1592. }
  1593. psl->lpVtbl->SetArguments(psl, lpszArgs);
  1594. //
  1595. // Remove quotes from the command file name NOW.
  1596. //
  1597. if (szBuffer[0] == TEXT('"')) {
  1598. LPTSTR lpTemp;
  1599. for (lpTemp = szBuffer+1; *lpTemp && *lpTemp != TEXT('"'); lpTemp++)
  1600. *(lpTemp-1) = *lpTemp;
  1601. if (*lpTemp == TEXT('"')) {
  1602. *(lpTemp-1) = TEXT('\0');
  1603. }
  1604. }
  1605. // working directory
  1606. // NB Progman assumed an empty WD meant use the windows
  1607. // directory but we want to change this so to be
  1608. // backwards compatable we'll fill in missing WD's here.
  1609. if (!lppmitem->lpszWD || !*lppmitem->lpszWD)
  1610. {
  1611. // NB For links to pif's we don't fill in a default WD
  1612. // so we'll pick it up from pif itself. This fixes a
  1613. // problem upgrading some Compaq Deskpro's.
  1614. pszExt = PathFindExtension(szBuffer);
  1615. if (lstrcmpi(pszExt, c_szDotPif) == 0)
  1616. {
  1617. psl->lpVtbl->SetWorkingDirectory(psl, c_szNULL);
  1618. }
  1619. else
  1620. {
  1621. #ifdef WINNT
  1622. // Avoid setting to %windir%, under NT we want to change to the users home directory.
  1623. psl->lpVtbl->SetWorkingDirectory( psl, TEXT("%HOMEDRIVE%%HOMEPATH%") );
  1624. #else
  1625. // Not a pif. Set the WD to be that of the windows dir.
  1626. psl->lpVtbl->SetWorkingDirectory(psl, TEXT("%windir%"));
  1627. #endif
  1628. }
  1629. }
  1630. else
  1631. {
  1632. psl->lpVtbl->SetWorkingDirectory(psl, lppmitem->lpszWD);
  1633. }
  1634. // icon location
  1635. // REVIEW, do we want to unqualify the icon path if possible? also,
  1636. // if the icon path is the same as the command line we don't need it
  1637. if (lppmitem->wiIcon != 0 || lstrcmpi(lppmitem->lpszIconPath, szBuffer) != 0)
  1638. {
  1639. // Remove args.
  1640. lpszArgs = PathGetArgs(lppmitem->lpszIconPath);
  1641. if (*lpszArgs)
  1642. *(lpszArgs-1) = TEXT('\0');
  1643. psl->lpVtbl->SetIconLocation(psl, lppmitem->lpszIconPath, lppmitem->wiIcon);
  1644. }
  1645. else
  1646. {
  1647. psl->lpVtbl->SetIconLocation(psl, NULL, 0);
  1648. }
  1649. // hotkey
  1650. psl->lpVtbl->SetHotkey(psl, lppmitem->wHotKey);
  1651. // show command
  1652. psl->lpVtbl->SetShowCmd(psl, lppmitem->nShowCmd);
  1653. // Description. Currently pifmgr is the only guy
  1654. // that cares about the description and they use
  1655. // it to overide the default pif description.
  1656. psl->lpVtbl->SetDescription(psl, lppmitem->lpszDesc);
  1657. //
  1658. // NOTE it is very important to set filename *last*
  1659. // because if this is a group item to another link
  1660. // (either .lnk or .pif) we want the link properties
  1661. // to override the ones we just set.
  1662. //
  1663. // qualify path to subject (szBuffer)
  1664. dirs[0] = lppmitem->lpszWD;
  1665. dirs[1] = NULL;
  1666. // Try expanding szBuffer
  1667. ExpandEnvironmentStrings( szBuffer, szExpBuff, MAX_PATH );
  1668. szExpBuff[ MAX_PATH-1 ] = TEXT('\0');
  1669. if (!PathResolve(szExpBuff, dirs, PRF_TRYPROGRAMEXTENSIONS))
  1670. {
  1671. // Just assume the expanded thing was a-ok...
  1672. ExpandEnvironmentStrings(szBuffer, szExpBuff, MAX_PATH);
  1673. szExpBuff[ MAX_PATH-1 ] = TEXT('\0');
  1674. }
  1675. // all we need to call is setpath, it takes care of creating the
  1676. // pidl for us.
  1677. psl->lpVtbl->SetPath( psl, szBuffer );
  1678. #ifdef WINNT
  1679. {
  1680. IShellLinkDataList* psldl;
  1681. if (SUCCEEDED(psl->lpVtbl->QueryInterface(psl, &IID_IShellLinkDataList, (LPVOID)&psldl)))
  1682. {
  1683. DWORD dwFlags;
  1684. if (SUCCEEDED(psldl->lpVtbl->GetFlags(psldl, &dwFlags)))
  1685. {
  1686. if (lppmitem->bSepVdm)
  1687. dwFlags |= SLDF_RUN_IN_SEPARATE;
  1688. else
  1689. dwFlags &= (~SLDF_RUN_IN_SEPARATE);
  1690. psldl->lpVtbl->SetFlags(psldl, dwFlags);
  1691. }
  1692. psldl->lpVtbl->Release(psldl);
  1693. }
  1694. }
  1695. #endif
  1696. // over write the link if it already exists
  1697. PathCombine(szLinkName, lpszNewGrpPath, lppmitem->lpszDesc);
  1698. lstrcat(szLinkName, TEXT(".lnk"));
  1699. PathQualify(szLinkName);
  1700. // OLE string.
  1701. StrToOleStrN(wszPath, ARRAYSIZE(wszPath), szLinkName, -1);
  1702. ppf->lpVtbl->Save(ppf, wszPath, TRUE);
  1703. }
  1704. ppf->lpVtbl->Release(ppf);
  1705. psl->lpVtbl->Release(psl);
  1706. }
  1707. }
  1708. //----------------------------------------------------------------------------
  1709. // Returns TRUE if the specified group title is that of the startup group.
  1710. BOOL StartupCmp(LPTSTR szGrp)
  1711. {
  1712. static TCHAR szOldStartupGrp[MAX_PATH];
  1713. TCHAR szNewStartupPath[MAX_PATH];
  1714. if (!*szOldStartupGrp)
  1715. {
  1716. // Was it over-ridden in progman ini?
  1717. GetPrivateProfileString(c_szSettings, c_szStartup, c_szNULL, szOldStartupGrp,
  1718. ARRAYSIZE(szOldStartupGrp), c_szProgmanIni);
  1719. if (!*szOldStartupGrp)
  1720. {
  1721. LONG lResult;
  1722. DWORD cbSize;
  1723. // No, try reading it from the NT registry
  1724. cbSize = MAX_PATH;
  1725. lResult = RegQueryValue(HKEY_CURRENT_USER, c_szProgmanStartup, szOldStartupGrp, &cbSize );
  1726. // Potential porblem with Kana Start
  1727. if ( lResult != ERROR_SUCCESS )
  1728. {
  1729. // No, use the default name.
  1730. LoadString(g_hinst, IDS_STARTUP, szOldStartupGrp, ARRAYSIZE(szOldStartupGrp));
  1731. }
  1732. }
  1733. if (*szOldStartupGrp)
  1734. {
  1735. // Yes, use the over-riding name by updating the registry.
  1736. SHGetSpecialFolderPath(NULL, szNewStartupPath, CSIDL_PROGRAMS, FALSE);
  1737. PathAddBackslash(szNewStartupPath);
  1738. lstrcat(szNewStartupPath, szOldStartupGrp);
  1739. DebugMsg(DM_TRACE, TEXT("gc.sc: Non-default Startup path is %s."), szNewStartupPath);
  1740. Reg_SetString(HKEY_CURRENT_USER, REGSTR_PATH_EXPLORER_SHELLFOLDERS, c_szStartup, szNewStartupPath);
  1741. }
  1742. }
  1743. // Does it match?
  1744. if (*szOldStartupGrp && (lstrcmpi(szGrp, szOldStartupGrp) == 0))
  1745. return TRUE;
  1746. else
  1747. return FALSE;
  1748. }
  1749. //---------------------------------------------------------------------------
  1750. BOOL CALLBACK IsDescUnique(LPCTSTR lpsz, UINT n)
  1751. {
  1752. int i, cItems;
  1753. LPPMITEM pitem;
  1754. // DebugMsg(DM_TRACE, "gc.idu: Checking uniqueness of %s.", lpsz);
  1755. cItems = DSA_GetItemCount(hdsaPMItems);
  1756. for (i=0; i<cItems; i++)
  1757. {
  1758. // N is our guy, skip it.
  1759. if ((UINT)i == n)
  1760. continue;
  1761. pitem = DSA_GetItemPtr(hdsaPMItems, i);
  1762. Assert(pitem);
  1763. if (pitem->lpszDesc && *pitem->lpszDesc && (lstrcmpi(pitem->lpszDesc, lpsz) == 0))
  1764. {
  1765. // DebugMsg(DM_TRACE, "gc.idu: Not Unique.");
  1766. return FALSE;
  1767. }
  1768. }
  1769. // Yep. can't find it, must be unique.
  1770. // DebugMsg(DM_TRACE, "gc.idu: Unique.");
  1771. return TRUE;
  1772. }
  1773. //---------------------------------------------------------------------------
  1774. // If there are two or more items with the same link name then change them so
  1775. // that they are unique.
  1776. void ResolveDuplicates(LPCTSTR pszNewGrpPath)
  1777. {
  1778. LPPMITEM pitem;
  1779. int i, cItems;
  1780. TCHAR szNew[MAX_PATH];
  1781. BOOL fLFN;
  1782. int cchSpace;
  1783. DebugMsg(DM_TRACE, TEXT("gc.rd: Fixing dups..."));
  1784. // How much room is there for adding the #xx stuff?
  1785. cchSpace = (ARRAYSIZE(szNew)-lstrlen(pszNewGrpPath))-2;
  1786. if (cchSpace > 0)
  1787. {
  1788. // LFN's or no?
  1789. fLFN = IsLFNDrive(pszNewGrpPath);
  1790. if (!fLFN && cchSpace > 8)
  1791. cchSpace = 8;
  1792. // Fix dups
  1793. cItems = DSA_GetItemCount(hdsaPMItems);
  1794. for (i=0; i<(cItems-1); i++)
  1795. {
  1796. pitem = DSA_GetItemPtr(hdsaPMItems, i);
  1797. Assert(pitem);
  1798. YetAnotherMakeUniqueName(szNew, cchSpace, pitem->lpszDesc, IsDescUnique, i, fLFN);
  1799. // Did we get a new name?
  1800. if (lstrcmp(szNew, pitem->lpszDesc) != 0)
  1801. {
  1802. // Yep.
  1803. DebugMsg(DM_TRACE, TEXT("gc.rd: %s to %s"), pitem->lpszDesc, szNew);
  1804. Str_SetPtr(&pitem->lpszDesc, szNew);
  1805. }
  1806. }
  1807. }
  1808. DebugMsg(DM_TRACE, TEXT("gc.rd: Done."));
  1809. }
  1810. //---------------------------------------------------------------------------
  1811. typedef struct
  1812. {
  1813. LPTSTR pszName;
  1814. LPTSTR pszPath;
  1815. LPTSTR pszModule;
  1816. LPTSTR pszVer;
  1817. } ALITEM;
  1818. typedef ALITEM *PALITEM;
  1819. //---------------------------------------------------------------------------
  1820. // Record the total list of apps in a DSA.
  1821. void AppList_WriteFile(void)
  1822. {
  1823. int i, cItems;
  1824. PALITEM palitem;
  1825. TCHAR szBetaID[MAX_PATH];
  1826. TCHAR szLine[4*MAX_PATH];
  1827. HANDLE hFile;
  1828. DWORD cbWritten;
  1829. Assert(g_hdsaAppList);
  1830. cItems = DSA_GetItemCount(g_hdsaAppList);
  1831. if (cItems)
  1832. {
  1833. // Get the beta ID.
  1834. szBetaID[0] = TEXT('\0');
  1835. Reg_GetString(HKEY_LOCAL_MACHINE, c_szRegistry, c_szDefaultUser, szBetaID, SIZEOF(szBetaID));
  1836. // Ick - Hard coded file name and in the current dir!
  1837. hFile = CreateFile(c_szGrpConvData, GENERIC_WRITE, FILE_SHARE_READ, NULL,
  1838. CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
  1839. if (hFile != INVALID_HANDLE_VALUE)
  1840. {
  1841. for (i=0; i < cItems; i++)
  1842. {
  1843. palitem = DSA_GetItemPtr(g_hdsaAppList, i);
  1844. wsprintf(szLine, TEXT("%s,\"%s\",\"%s\",\"%s\",\"%s\",,,\r\n"), szBetaID, palitem->pszName,
  1845. palitem->pszPath, palitem->pszModule, palitem->pszVer);
  1846. DebugMsg(DM_TRACE,TEXT("gc.al_wf: %s"), szLine);
  1847. WriteFile(hFile, szLine, lstrlen(szLine)*SIZEOF(TCHAR), &cbWritten, NULL);
  1848. }
  1849. CloseHandle(hFile);
  1850. }
  1851. else
  1852. {
  1853. DebugMsg(DM_ERROR, TEXT("gc.al_wf: Can't write file."));
  1854. }
  1855. }
  1856. else
  1857. {
  1858. DebugMsg(DM_TRACE, TEXT("gc.al_wf: Empty app list. Nothing to write."));
  1859. }
  1860. }
  1861. //---------------------------------------------------------------------------
  1862. //#define DSA_AppendItem(hdsa, pitem) DSA_InsertItem(hdsa, 0x7fff, pitem)
  1863. //---------------------------------------------------------------------------
  1864. static TCHAR const c_szTranslation[] = TEXT("\\VarFileInfo\\Translation");
  1865. static TCHAR const c_szStringFileInfo[] = TEXT("\\StringFileInfo\\");
  1866. static TCHAR const c_szEngLangCharSet[] = TEXT("040904e4");
  1867. static TCHAR const c_szSlash[] = TEXT("\\");
  1868. static TCHAR const c_szInternalName[] = TEXT("InternalName");
  1869. static TCHAR const c_szProductVersion[] = TEXT("ProductVersion");
  1870. //----------------------------------------------------------------------------
  1871. // Semi-decent wrappers around the not very good ver apis.
  1872. BOOL Ver_GetDefaultCharSet(const PVOID pBuf, LPTSTR pszLangCharSet, int cbLangCharSet)
  1873. {
  1874. LPWORD pTransTable;
  1875. DWORD cb;
  1876. Assert(pszLangCharSet);
  1877. Assert(cbLangCharSet > 8);
  1878. if (VerQueryValue(pBuf, (LPTSTR)c_szTranslation, &pTransTable, &cb))
  1879. {
  1880. wsprintf(pszLangCharSet, TEXT("%04X%04X"), *pTransTable, *(pTransTable+1));
  1881. return TRUE;
  1882. }
  1883. return FALSE;
  1884. }
  1885. //----------------------------------------------------------------------------
  1886. // Semi-decent wrappers around the not very good ver apis.
  1887. BOOL Ver_GetStringFileInfo(PVOID pBuf, LPCTSTR pszLangCharSet,
  1888. LPCTSTR pszStringName, LPTSTR pszValue, int cbValue)
  1889. {
  1890. TCHAR szSubBlock[MAX_PATH];
  1891. LPTSTR pszBuf;
  1892. DWORD cbBuf;
  1893. lstrcpy(szSubBlock, c_szStringFileInfo);
  1894. lstrcat(szSubBlock, pszLangCharSet);
  1895. lstrcat(szSubBlock, c_szSlash);
  1896. lstrcat(szSubBlock, pszStringName);
  1897. if (VerQueryValue(pBuf, szSubBlock, &pszBuf, &cbBuf))
  1898. {
  1899. lstrcpyn(pszValue, pszBuf, cbValue);
  1900. return TRUE;
  1901. }
  1902. return FALSE;
  1903. }
  1904. //---------------------------------------------------------------------------
  1905. void GetVersionInfo(LPTSTR pszPath, LPTSTR pszModule, int cbModule, LPTSTR pszVer, int cbVer)
  1906. {
  1907. DWORD cbBuf;
  1908. LPVOID pBuf;
  1909. TCHAR szCharSet[MAX_PATH];
  1910. DWORD dwWasteOfAnAuto;
  1911. Assert(pszModule);
  1912. Assert(pszVer);
  1913. pszModule[0] = TEXT('\0');
  1914. pszVer[0] = TEXT('\0');
  1915. cbBuf = GetFileVersionInfoSize(pszPath, &dwWasteOfAnAuto);
  1916. if (cbBuf)
  1917. {
  1918. pBuf = SHAlloc(cbBuf);
  1919. if (pBuf)
  1920. {
  1921. if (GetFileVersionInfo(pszPath, 0, cbBuf, pBuf))
  1922. {
  1923. // Try the default language from the translation tables.
  1924. if (Ver_GetDefaultCharSet(pBuf, szCharSet, ARRAYSIZE(szCharSet)))
  1925. {
  1926. Ver_GetStringFileInfo(pBuf, szCharSet, c_szInternalName, pszModule, cbModule);
  1927. Ver_GetStringFileInfo(pBuf, szCharSet, c_szProductVersion, pszVer, cbVer);
  1928. }
  1929. else
  1930. {
  1931. // Try the same language as us.
  1932. LoadString(g_hinst, IDS_DEFLANGCHARSET, szCharSet, ARRAYSIZE(szCharSet));
  1933. Ver_GetStringFileInfo(pBuf, szCharSet, c_szInternalName, pszModule, cbModule);
  1934. Ver_GetStringFileInfo(pBuf, szCharSet, c_szProductVersion, pszVer, cbVer);
  1935. }
  1936. // Last chance - try English.
  1937. if (!*pszModule)
  1938. Ver_GetStringFileInfo(pBuf, c_szEngLangCharSet, c_szInternalName, pszModule, cbModule);
  1939. if (!*pszVer)
  1940. Ver_GetStringFileInfo(pBuf, c_szEngLangCharSet, c_szProductVersion, pszVer, cbVer);
  1941. }
  1942. else
  1943. {
  1944. DebugMsg(DM_TRACE, TEXT("gc.gvi: Can't get version info."));
  1945. }
  1946. SHFree(pBuf);
  1947. }
  1948. else
  1949. {
  1950. DebugMsg(DM_TRACE, TEXT("gc.gvi: Can't allocate version info buffer."));
  1951. }
  1952. }
  1953. else
  1954. {
  1955. DebugMsg(DM_TRACE, TEXT("gc.gvi: No version info."));
  1956. }
  1957. }
  1958. //---------------------------------------------------------------------------
  1959. // Record the total list of apps in a DSA.
  1960. BOOL AppList_Create(void)
  1961. {
  1962. Assert(!g_hdsaAppList);
  1963. g_hdsaAppList = DSA_Create(SIZEOF(ALITEM), 0);
  1964. if (g_hdsaAppList)
  1965. {
  1966. return TRUE;
  1967. }
  1968. else
  1969. {
  1970. DebugMsg(DM_ERROR, TEXT("gc.al_c: Can't create app list."));
  1971. return FALSE;
  1972. }
  1973. }
  1974. //---------------------------------------------------------------------------
  1975. // Record the total list of apps in a DSA.
  1976. void AppList_Destroy(void)
  1977. {
  1978. int i, cItems;
  1979. PALITEM palitem;
  1980. Assert(g_hdsaAppList);
  1981. cItems = DSA_GetItemCount(g_hdsaAppList);
  1982. for (i=0; i < cItems; i++)
  1983. {
  1984. palitem = DSA_GetItemPtr(g_hdsaAppList, i);
  1985. if (palitem->pszName)
  1986. SHFree(palitem->pszName);
  1987. if (palitem->pszPath)
  1988. SHFree(palitem->pszPath);
  1989. if (palitem->pszModule)
  1990. SHFree(palitem->pszModule);
  1991. if (palitem->pszVer)
  1992. SHFree(palitem->pszVer);
  1993. }
  1994. DSA_Destroy(g_hdsaAppList);
  1995. g_hdsaAppList = NULL;
  1996. }
  1997. //---------------------------------------------------------------------------
  1998. // Record the total list of apps in a DSA.
  1999. void AppList_Append(void)
  2000. {
  2001. int i, cItems;
  2002. // char szName[MAX_PATH];
  2003. // char szPath[MAX_PATH];
  2004. TCHAR szModule[MAX_PATH];
  2005. TCHAR szVer[MAX_PATH];
  2006. TCHAR szCL[MAX_PATH];
  2007. LPTSTR lpszArgs;
  2008. LPCTSTR dirs[2];
  2009. ALITEM alitem;
  2010. Assert(g_hdsaAppList);
  2011. cItems = DSA_GetItemCount(hdsaPMItems);
  2012. for (i = 0; i < cItems; i++)
  2013. {
  2014. LPPMITEM lppmitem = DSA_GetItemPtr(hdsaPMItems, i);
  2015. // We show the progress in 2 halves.
  2016. Group_SetProgress(cItems+i);
  2017. // Command line and args.
  2018. Str_GetPtr(lppmitem->lpszCL, szCL, ARRAYSIZE(szCL));
  2019. lpszArgs = PathGetArgs(szCL);
  2020. if (*lpszArgs)
  2021. *(lpszArgs-1) = TEXT('\0');
  2022. dirs[0] = lppmitem->lpszWD;
  2023. dirs[1] = NULL;
  2024. PathResolve(szCL, dirs, PRF_TRYPROGRAMEXTENSIONS);
  2025. // Version info.
  2026. GetVersionInfo(szCL, szModule, ARRAYSIZE(szModule), szVer, ARRAYSIZE(szVer));
  2027. alitem.pszName = NULL;
  2028. alitem.pszPath = NULL;
  2029. alitem.pszModule = NULL;
  2030. alitem.pszVer = NULL;
  2031. Str_SetPtr(&alitem.pszName, lppmitem->lpszDesc);
  2032. Str_SetPtr(&alitem.pszPath, szCL);
  2033. Str_SetPtr(&alitem.pszModule, szModule);
  2034. Str_SetPtr(&alitem.pszVer, szVer);
  2035. DSA_AppendItem(g_hdsaAppList, &alitem);
  2036. }
  2037. DebugMsg(DM_TRACE, TEXT("gc.al_a: %d items"), DSA_GetItemCount(g_hdsaAppList));
  2038. }
  2039. //---------------------------------------------------------------------------
  2040. // Reads an old format Progman Group files and creates a directory containing
  2041. // links that matches the group file.
  2042. BOOL Group_CreateNewFromOld(HWND hwnd, LPCTSTR lpszOldGrpPath, UINT options)
  2043. {
  2044. GROUPDEF grpdef;
  2045. #ifdef UNICODE
  2046. HANDLE fh;
  2047. DWORD dwBytesRead;
  2048. #else
  2049. HFILE fh;
  2050. #endif
  2051. TCHAR szNewGrpPath[MAX_PATH];
  2052. TCHAR szOldGrpTitle[MAXGROUPNAMELEN + 1];
  2053. // LPSTR lpszExt;
  2054. BOOL fStatus = FALSE;
  2055. SHELLEXECUTEINFO sei;
  2056. BOOL fStartup = FALSE;
  2057. if (!ItemList_Create(lpszOldGrpPath))
  2058. return FALSE;
  2059. #ifdef UNICODE
  2060. fh = CreateFile(
  2061. lpszOldGrpPath,
  2062. GENERIC_READ,
  2063. FILE_SHARE_READ,
  2064. NULL,
  2065. OPEN_EXISTING,
  2066. 0,
  2067. NULL
  2068. );
  2069. if (fh == INVALID_HANDLE_VALUE) {
  2070. #else
  2071. fh = _lopen(lpszOldGrpPath, OF_READ | OF_SHARE_DENY_NONE);
  2072. if (fh == HFILE_ERROR) {
  2073. #endif
  2074. DebugMsg(DM_ERROR, TEXT("gc.gcnfo: Unable to open group."));
  2075. goto ProcExit2;
  2076. }
  2077. #ifdef UNICODE
  2078. if ((!ReadFile(fh, &grpdef, SIZEOF(grpdef), &dwBytesRead, NULL)) ||
  2079. (dwBytesRead != SIZEOF(grpdef))) {
  2080. #else
  2081. if (_lread(fh, &grpdef, SIZEOF(grpdef)) != SIZEOF(grpdef)) {
  2082. #endif
  2083. DebugMsg(DM_ERROR, TEXT("gc.gcnfo: header too small."));
  2084. goto ProcExit;
  2085. }
  2086. if (grpdef.cItems > 50) {
  2087. // NB This isn;t fatal so carry on.
  2088. DebugMsg(DM_ERROR, TEXT("gc.gcnfo: Too many items."));
  2089. }
  2090. // Check to make sure there is a name embedded in the
  2091. // .grp file. If not, just use the filename
  2092. if (grpdef.pName==0) {
  2093. LPTSTR lpszFile, lpszExt, lpszDest = szOldGrpTitle;
  2094. lpszFile = PathFindFileName( lpszOldGrpPath );
  2095. lpszExt = PathFindExtension( lpszOldGrpPath );
  2096. for( ;
  2097. lpszFile && lpszExt && (lpszFile != lpszExt);
  2098. *lpszDest++ = *lpszFile++
  2099. );
  2100. *lpszDest = TEXT('\0');
  2101. } else {
  2102. #ifdef UNICODE
  2103. CHAR szAnsiTitle[ MAXGROUPNAMELEN + 1 ];
  2104. SetFilePointer(fh, grpdef.pName, NULL, FILE_BEGIN);
  2105. ReadFile(fh, szAnsiTitle, SIZEOF(szAnsiTitle), &dwBytesRead, NULL);
  2106. MultiByteToWideChar( CP_ACP, 0, szAnsiTitle, -1, szOldGrpTitle, ARRAYSIZE(szOldGrpTitle) );
  2107. #else
  2108. _llseek(fh, grpdef.pName, 0);
  2109. _lread(fh, szOldGrpTitle, SIZEOF(szOldGrpTitle));
  2110. #endif
  2111. }
  2112. // Get the destination dir, use the title from the old group...
  2113. // Special case the startup group.
  2114. if (StartupCmp(szOldGrpTitle)) {
  2115. fStartup = TRUE;
  2116. if (g_fDoingCommonGroups) {
  2117. SHGetSpecialFolderPath(hwnd, szNewGrpPath, CSIDL_COMMON_STARTUP, TRUE);
  2118. } else {
  2119. SHGetSpecialFolderPath(hwnd, szNewGrpPath, CSIDL_STARTUP, TRUE);
  2120. }
  2121. } else {
  2122. if (!Group_GenerateNewGroupPath(hwnd, szOldGrpTitle, szNewGrpPath, lpszOldGrpPath)) {
  2123. DebugMsg(DM_ERROR, TEXT("gc.gcnfo; Unable to create destination directory."));
  2124. goto ProcExit;
  2125. }
  2126. }
  2127. // PathQualify(szNewGrpPath);
  2128. // ResolveDuplicateGroupNames(szNewGrpPath);
  2129. // Go through every item in the old group and make it a link...
  2130. if (!GetAllItemData(fh, grpdef.cItems, grpdef.cbGroup, szOldGrpTitle, szNewGrpPath)) {
  2131. if (options & GC_REPORTERROR)
  2132. MyMessageBox(hwnd, IDS_APPTITLE, IDS_BADOLDGROUP, NULL, MB_OK | MB_ICONEXCLAMATION);
  2133. }
  2134. // Deal with the tags section.
  2135. HandleTags(fh, grpdef.cbGroup);
  2136. // Now we've dealt with the tags we don't need to keep track of
  2137. // busted items so delete them now. From here on we always have
  2138. // valid items.
  2139. DeleteBustedItems();
  2140. // Shorten descs on non-lfn drives.
  2141. if (!IsLFNDrive(szNewGrpPath))
  2142. ShortenDescriptions();
  2143. // Fixup the paths/WD stuff.
  2144. MungePaths();
  2145. // Fix dups.
  2146. ResolveDuplicates(szNewGrpPath);
  2147. // Do we just want a list of the apps or create some links?
  2148. if (options & GC_BUILDLIST)
  2149. AppList_Append();
  2150. else
  2151. CreateLinks(szNewGrpPath, fStartup, grpdef.cItems);
  2152. // Get the cabinet to show the new group.
  2153. if (options & GC_OPENGROUP)
  2154. {
  2155. sei.cbSize = SIZEOF(sei);
  2156. sei.fMask = 0;
  2157. sei.hwnd = hwnd;
  2158. sei.lpVerb = NULL;
  2159. sei.lpFile = szNewGrpPath;
  2160. sei.lpParameters = NULL;
  2161. sei.lpDirectory = NULL;
  2162. sei.lpClass = NULL;
  2163. sei.nShow = SW_SHOWNORMAL;
  2164. sei.hInstApp = g_hinst;
  2165. // ShellExecute(hwnd, NULL, szNewGrpPath, NULL, NULL, SW_SHOWNORMAL);
  2166. ShellExecuteEx(&sei);
  2167. }
  2168. // Everything went OK.
  2169. fStatus = TRUE;
  2170. ProcExit:
  2171. #ifdef UNICODE
  2172. CloseHandle(fh);
  2173. #else
  2174. _lclose(fh);
  2175. #endif
  2176. #ifndef WINNT
  2177. // we only need to call Group_DeleteIfRequired
  2178. // when we are on a Japanese language machine (win95J or win98J). We
  2179. // should have a runtime check for Japanese here.
  2180. // Delete old group file when it is specified in special
  2181. // registry entry. Bug#7259-win95d
  2182. //
  2183. if (fStatus == TRUE)
  2184. {
  2185. // delete it only if the conversion was successful.
  2186. Group_DeleteIfRequired(szOldGrpTitle,lpszOldGrpPath);
  2187. }
  2188. #endif // !WINNT
  2189. ProcExit2:
  2190. ItemList_Destroy();
  2191. return fStatus;
  2192. }
  2193. //---------------------------------------------------------------------------
  2194. // Reads an NT format Progman Group files and creates a directory containing
  2195. // links that matches the group file.
  2196. BOOL Group_CreateNewFromOldNT(HWND hwnd, LPCTSTR lpszOldGrpPath, UINT options)
  2197. {
  2198. NT_GROUPDEF grpdef;
  2199. #ifdef UNICODE
  2200. HANDLE fh;
  2201. DWORD dwBytesRead;
  2202. #else
  2203. HFILE fh;
  2204. #endif
  2205. TCHAR szNewGrpPath[MAX_PATH];
  2206. WCHAR szOldGrpTitleUnicode[MAXGROUPNAMELEN + 1];
  2207. TCHAR szOldGrpTitle[MAXGROUPNAMELEN + 1];
  2208. // LPSTR lpszExt;
  2209. BOOL fStatus = FALSE;
  2210. SHELLEXECUTEINFO sei;
  2211. BOOL fStartup = FALSE;
  2212. if (!ItemList_Create(lpszOldGrpPath))
  2213. return FALSE;
  2214. #ifdef UNICODE
  2215. fh = CreateFile(
  2216. lpszOldGrpPath,
  2217. GENERIC_READ,
  2218. FILE_SHARE_READ,
  2219. NULL,
  2220. OPEN_EXISTING,
  2221. 0,
  2222. NULL
  2223. );
  2224. if (fh == INVALID_HANDLE_VALUE) {
  2225. #else
  2226. fh = _lopen(lpszOldGrpPath, OF_READ | OF_SHARE_DENY_NONE);
  2227. if (fh == HFILE_ERROR) {
  2228. #endif
  2229. DebugMsg(DM_ERROR, TEXT("gc.gcnfont: Unable to open group."));
  2230. goto ProcExit2;
  2231. }
  2232. #ifdef UNICODE
  2233. if (!ReadFile(fh, &grpdef, SIZEOF(grpdef), &dwBytesRead, NULL) ||
  2234. dwBytesRead != SIZEOF(grpdef)) {
  2235. #else
  2236. if (_lread(fh, &grpdef, SIZEOF(grpdef)) != SIZEOF(grpdef)) {
  2237. #endif
  2238. DebugMsg(DM_ERROR, TEXT("gc.gcnfont: header too small."));
  2239. goto ProcExit;
  2240. }
  2241. if (grpdef.cItems > 50) {
  2242. // NB This isn;t fatal so carry on.
  2243. DebugMsg(DM_ERROR, TEXT("gc.gcnfont: Too many items."));
  2244. }
  2245. #ifdef UNICODE
  2246. SetFilePointer(fh, grpdef.pName, NULL, FILE_BEGIN);
  2247. ReadFile(fh, szOldGrpTitleUnicode, SIZEOF(szOldGrpTitleUnicode), &dwBytesRead, NULL);
  2248. #else
  2249. _llseek(fh, grpdef.pName, 0);
  2250. _lread(fh, szOldGrpTitleUnicode, SIZEOF(szOldGrpTitleUnicode));
  2251. #endif
  2252. #ifdef UNICODE
  2253. lstrcpy(szOldGrpTitle, szOldGrpTitleUnicode);
  2254. #else
  2255. WideCharToMultiByte (CP_ACP, 0, szOldGrpTitleUnicode, -1,
  2256. szOldGrpTitle, MAXGROUPNAMELEN+1, NULL, NULL);
  2257. #endif
  2258. // Get the destination dir, use the title from the old group.
  2259. // REVIEW UNDONE - until we get long filenames we'll use the old
  2260. // groups' filename as the basis for the new group instead of it's
  2261. // title.
  2262. // Special case the startup group.
  2263. if (StartupCmp(szOldGrpTitle)) {
  2264. if (g_fDoingCommonGroups) {
  2265. SHGetSpecialFolderPath(hwnd, szNewGrpPath, CSIDL_COMMON_STARTUP, TRUE);
  2266. } else {
  2267. SHGetSpecialFolderPath(hwnd, szNewGrpPath, CSIDL_STARTUP, TRUE);
  2268. }
  2269. } else {
  2270. if (!Group_GenerateNewGroupPath(hwnd, szOldGrpTitle, szNewGrpPath, lpszOldGrpPath)) {
  2271. DebugMsg(DM_ERROR, TEXT("gc.gcnfo; Unable to create destination directory."));
  2272. goto ProcExit;
  2273. }
  2274. }
  2275. // Go through every item in the old group and make it a link...
  2276. if (!GetAllItemDataNT(fh, grpdef.cItems, grpdef.cbGroup, szOldGrpTitle, szNewGrpPath)) {
  2277. if (options & GC_REPORTERROR)
  2278. MyMessageBox(hwnd, IDS_APPTITLE, IDS_BADOLDGROUP, NULL, MB_OK | MB_ICONEXCLAMATION);
  2279. }
  2280. // Deal with the tags section.
  2281. HandleTagsNT(fh, grpdef.cbGroup);
  2282. // Now we've dealt with the tags we don't need to keep track of
  2283. // busted items so delete them now. From here on we always have
  2284. // valid items.
  2285. DeleteBustedItems();
  2286. // Shorten descs on non-lfn drives.
  2287. if (!IsLFNDrive(szNewGrpPath))
  2288. ShortenDescriptions();
  2289. // Fixup the paths/WD stuff.
  2290. MungePaths();
  2291. // Fix dups.
  2292. ResolveDuplicates(szNewGrpPath);
  2293. // Do we just want a list of the apps or create some links?
  2294. if (options & GC_BUILDLIST)
  2295. AppList_Append();
  2296. else
  2297. CreateLinks(szNewGrpPath, fStartup, grpdef.cItems);
  2298. // Get the cabinet to show the new group.
  2299. if (options & GC_OPENGROUP)
  2300. {
  2301. sei.cbSize = SIZEOF(sei);
  2302. sei.fMask = 0;
  2303. sei.hwnd = hwnd;
  2304. sei.lpVerb = NULL;
  2305. sei.lpFile = szNewGrpPath;
  2306. sei.lpParameters = NULL;
  2307. sei.lpDirectory = NULL;
  2308. sei.lpClass = NULL;
  2309. sei.nShow = SW_SHOWNORMAL;
  2310. sei.hInstApp = g_hinst;
  2311. // ShellExecute(hwnd, NULL, szNewGrpPath, NULL, NULL, SW_SHOWNORMAL);
  2312. ShellExecuteEx(&sei);
  2313. }
  2314. // Everything went OK.
  2315. fStatus = TRUE;
  2316. ProcExit:
  2317. #ifdef UNICODE
  2318. CloseHandle(fh);
  2319. #else
  2320. _lclose(fh);
  2321. #endif
  2322. ProcExit2:
  2323. ItemList_Destroy();
  2324. return fStatus;
  2325. }
  2326. //---------------------------------------------------------------------------
  2327. // Record the last write date/time of the given group in the ini file.
  2328. void Group_WriteLastModDateTime(LPCTSTR lpszGroupFile,DWORD dwLowDateTime)
  2329. {
  2330. Reg_SetStruct(g_hkeyGrpConv, c_szGroups, lpszGroupFile, &dwLowDateTime, SIZEOF(dwLowDateTime));
  2331. }
  2332. //---------------------------------------------------------------------------
  2333. // Read the last write date/time of the given group from the ini file.
  2334. DWORD Group_ReadLastModDateTime(LPCTSTR lpszGroupFile)
  2335. {
  2336. DWORD dwDateTime = 0;
  2337. Reg_GetStruct(g_hkeyGrpConv, c_szGroups, lpszGroupFile, &dwDateTime, SIZEOF(dwDateTime));
  2338. return dwDateTime;
  2339. }
  2340. //---------------------------------------------------------------------------
  2341. // Convert the given group to the new format.
  2342. // Returns FALSE if something goes wrong.
  2343. // Returns true if the given group got converted or the user cancelled.
  2344. BOOL Group_Convert(HWND hwnd, LPCTSTR lpszOldGrpFile, UINT options)
  2345. {
  2346. TCHAR szGroupTitle[MAXGROUPNAMELEN + 1]; // PM Groups had a max title len of 30.
  2347. BOOL fStatus;
  2348. WIN32_FIND_DATA fd;
  2349. HANDLE hff;
  2350. UINT nCode;
  2351. UINT iErrorId;
  2352. Log(TEXT("Grp: %s"), lpszOldGrpFile);
  2353. DebugMsg(DM_TRACE, TEXT("gc.gc: Converting group %s"), (LPTSTR) lpszOldGrpFile);
  2354. // Does the group exist?
  2355. if (PathFileExists(lpszOldGrpFile))
  2356. {
  2357. // Group exists - is it valid?
  2358. nCode = Group_ValidOldFormat(lpszOldGrpFile, szGroupTitle);
  2359. switch( nCode )
  2360. {
  2361. case VOF_WINNT:
  2362. case VOF_WIN31:
  2363. // Yes - ask for confirmation.
  2364. if (!(options & GC_PROMPTBEFORECONVERT) ||
  2365. MyMessageBox(hwnd, IDS_APPTITLE, IDS_OKTOCONVERT, szGroupTitle, MB_YESNO) == IDYES)
  2366. {
  2367. // Everything went OK?
  2368. if ( nCode == VOF_WIN31 )
  2369. {
  2370. fStatus = Group_CreateNewFromOld(hwnd,lpszOldGrpFile,
  2371. options);
  2372. }
  2373. else
  2374. {
  2375. fStatus = Group_CreateNewFromOldNT(hwnd,lpszOldGrpFile,
  2376. options);
  2377. }
  2378. if ( fStatus )
  2379. {
  2380. iErrorId = 0;
  2381. }
  2382. else
  2383. {
  2384. // Nope - FU. Warn and exit.
  2385. iErrorId = IDS_CONVERTERROR;
  2386. }
  2387. }
  2388. else
  2389. {
  2390. // User cancelled...
  2391. iErrorId = 0;
  2392. }
  2393. break;
  2394. default:
  2395. case VOF_BAD:
  2396. {
  2397. // Nope, File is invalid.
  2398. // Warn user.
  2399. iErrorId = IDS_NOTGROUPFILE;
  2400. }
  2401. break;
  2402. }
  2403. }
  2404. else
  2405. {
  2406. // Nope, File doesn't even exist.
  2407. iErrorId = IDS_MISSINGFILE;
  2408. }
  2409. if ( iErrorId != 0 )
  2410. {
  2411. if (options & GC_REPORTERROR)
  2412. {
  2413. MyMessageBox(hwnd, IDS_APPTITLE, iErrorId,
  2414. lpszOldGrpFile, MB_OK|MB_ICONEXCLAMATION);
  2415. }
  2416. Log(TEXT("Grp: %s done."), lpszOldGrpFile);
  2417. return FALSE;
  2418. }
  2419. else
  2420. {
  2421. DebugMsg(DM_TRACE, TEXT("gc.gc: Done."));
  2422. Log(TEXT("Grp: %s done."), lpszOldGrpFile);
  2423. return TRUE;
  2424. }
  2425. }
  2426. //---------------------------------------------------------------------------
  2427. // Checks the date/time stamp of the given group against the one in
  2428. // grpconv.ini
  2429. BOOL GroupHasBeenModified(LPCTSTR lpszGroupFile)
  2430. {
  2431. WIN32_FIND_DATA fd;
  2432. HANDLE hff;
  2433. BOOL fModified;
  2434. hff = FindFirstFile(lpszGroupFile, &fd);
  2435. if (hff != INVALID_HANDLE_VALUE)
  2436. {
  2437. if (Group_ReadLastModDateTime(lpszGroupFile) != fd.ftLastWriteTime.dwLowDateTime)
  2438. {
  2439. DebugMsg(DM_TRACE, TEXT("cg.ghbm: Group %s has been modified."), (LPTSTR)lpszGroupFile);
  2440. fModified = TRUE;
  2441. }
  2442. else
  2443. {
  2444. DebugMsg(DM_TRACE, TEXT("cg.ghbm: Group %s has not been modified."), (LPTSTR)lpszGroupFile);
  2445. fModified = FALSE;
  2446. }
  2447. FindClose(hff);
  2448. return fModified;
  2449. }
  2450. else
  2451. {
  2452. // Hmm, file doesn't exist, pretend it's up to date.
  2453. return TRUE;
  2454. }
  2455. }
  2456. //---------------------------------------------------------------------------
  2457. // Converts a group file from its NT registry into a real file on disk. Since
  2458. // the disk format for NT 1.0 files never existed and collided in its usage
  2459. // the GROUP_MAGIC file type, we will convert it from the registry, directly
  2460. // into a GROUP_UNICODE format file. In this way we will always be able to
  2461. // distiguish the NT group files from the Win 3.1 group files.
  2462. BOOL MakeGroupFile( LPTSTR lpFileName, LPTSTR lpGroupName)
  2463. {
  2464. LONG lResult;
  2465. DWORD cbSize;
  2466. HGLOBAL hBuffer;
  2467. HGLOBAL hNew;
  2468. LPBYTE lpBuffer;
  2469. BOOL fOk;
  2470. HANDLE hFile;
  2471. HKEY hkey;
  2472. DWORD cbWrote;
  2473. fOk = FALSE;
  2474. lResult = RegOpenKeyEx(hkeyGroups, lpGroupName, 0,
  2475. KEY_READ, &hkey );
  2476. if ( lResult != ERROR_SUCCESS )
  2477. {
  2478. return FALSE;
  2479. }
  2480. lResult = RegQueryValueEx( hkey, NULL, NULL, NULL, NULL, &cbSize);
  2481. if ( lResult != ERROR_SUCCESS )
  2482. {
  2483. goto CleanupKey;
  2484. }
  2485. hBuffer = GlobalAlloc(GMEM_MOVEABLE,cbSize);
  2486. if ( hBuffer == NULL )
  2487. {
  2488. goto CleanupKey;
  2489. }
  2490. lpBuffer = (LPBYTE)GlobalLock(hBuffer);
  2491. if ( lpBuffer == NULL )
  2492. {
  2493. goto CleanupMem;
  2494. }
  2495. lResult = RegQueryValueEx( hkey, NULL, NULL, NULL,
  2496. lpBuffer, &cbSize );
  2497. if ( lResult != ERROR_SUCCESS )
  2498. {
  2499. goto Cleanup;
  2500. }
  2501. if ( *(DWORD *)lpBuffer == GROUP_MAGIC )
  2502. {
  2503. HGLOBAL hNew;
  2504. cbSize = ConvertToUnicodeGroup( (LPNT_GROUPDEF_A)lpBuffer, &hNew );
  2505. GlobalUnlock( hBuffer );
  2506. GlobalFree( hBuffer );
  2507. hBuffer = hNew;
  2508. lpBuffer = GlobalLock( hBuffer );
  2509. if ( lpBuffer == NULL )
  2510. {
  2511. goto CleanupMem;
  2512. }
  2513. }
  2514. hFile = CreateFile(lpFileName,GENERIC_WRITE,0,NULL,
  2515. CREATE_ALWAYS,FILE_ATTRIBUTE_NORMAL,NULL);
  2516. if (hFile != INVALID_HANDLE_VALUE)
  2517. {
  2518. fOk = WriteFile(hFile,lpBuffer,cbSize,&cbWrote,NULL);
  2519. CloseHandle(hFile);
  2520. }
  2521. Cleanup:
  2522. GlobalUnlock(hBuffer);
  2523. CleanupMem:
  2524. GlobalFree(hBuffer);
  2525. CleanupKey:
  2526. RegCloseKey( hkey );
  2527. return fOk;
  2528. }
  2529. #define BIG_STEP 1024
  2530. //----------------------------------------------------------------------------
  2531. // Enumerate all the groups or just all the modified groups.
  2532. int Group_Enum(PFNGRPCALLBACK pfncb, BOOL fProgress,
  2533. BOOL fModifiedOnly)
  2534. {
  2535. TCHAR szIniFile[MAX_PATH], szFile[MAX_PATH];
  2536. FILETIME ft;
  2537. UINT uSize;
  2538. LPTSTR pSection, pKey;
  2539. int cGroups = 0;
  2540. HANDLE hFile;
  2541. WIN32_FIND_DATA fd;
  2542. if (!FindProgmanIni(szIniFile))
  2543. return 0;
  2544. for (uSize = BIG_STEP; uSize < BIG_STEP * 8; uSize += BIG_STEP)
  2545. {
  2546. pSection = (LPTSTR)LocalAlloc(LPTR, uSize);
  2547. if (!pSection)
  2548. return 0;
  2549. if ((UINT)GetPrivateProfileString(c_szGroups, NULL, c_szNULL, pSection, uSize / sizeof(pSection[0]), szIniFile) < uSize - 5)
  2550. break;
  2551. LocalFree((HLOCAL)pSection);
  2552. pSection = NULL;
  2553. }
  2554. if (!pSection)
  2555. return 0;
  2556. if (fProgress)
  2557. Group_CreateProgressDlg();
  2558. for (pKey = pSection; *pKey; pKey += lstrlen(pKey) + 1)
  2559. {
  2560. GetPrivateProfileString(c_szGroups, pKey, c_szNULL, szFile, ARRAYSIZE(szFile), szIniFile);
  2561. if (szFile[0])
  2562. {
  2563. if (!fModifiedOnly || GroupHasBeenModified(szFile))
  2564. {
  2565. (*pfncb)(szFile);
  2566. cGroups++;
  2567. hFile = FindFirstFile (szFile, &fd);
  2568. if (hFile != INVALID_HANDLE_VALUE) {
  2569. FindClose (hFile);
  2570. Group_WriteLastModDateTime(szFile, fd.ftLastWriteTime.dwLowDateTime);
  2571. }
  2572. }
  2573. }
  2574. }
  2575. // Cabinet uses the date/time of progman.ini as a hint to speed things up
  2576. // so set it here so we won't run automatically again.
  2577. GetSystemTimeAsFileTime(&ft);
  2578. Group_WriteLastModDateTime(szIniFile,ft.dwLowDateTime);
  2579. LocalFree((HLOCAL)pSection);
  2580. if (fProgress)
  2581. Group_DestroyProgressDlg();
  2582. return cGroups;
  2583. }
  2584. //----------------------------------------------------------------------------
  2585. // Enumerate all the NT groups or just all the modified groups.
  2586. int Group_EnumNT(PFNGRPCALLBACK pfncb, BOOL fProgress,
  2587. BOOL fModifiedOnly, HKEY hKeyRoot, LPCTSTR lpKey)
  2588. {
  2589. LONG lResult;
  2590. DWORD dwSubKey = 0;
  2591. TCHAR szGroupName[MAXGROUPNAMELEN+1];
  2592. TCHAR szFileName[MAX_PATH];
  2593. TCHAR szTempFileDir[MAX_PATH];
  2594. TCHAR szTempFileName[MAX_PATH];
  2595. DWORD cchGroupNameLen;
  2596. FILETIME ft;
  2597. BOOL fOk;
  2598. BOOL fDialog = FALSE;
  2599. BOOL fProcess;
  2600. int cGroups = 0;
  2601. //
  2602. // Look for groups in the registry
  2603. //
  2604. lResult = RegOpenKeyEx(hKeyRoot, lpKey, 0,
  2605. KEY_READ, &hkeyGroups );
  2606. if ( lResult != ERROR_SUCCESS )
  2607. {
  2608. return 0;
  2609. }
  2610. while ( TRUE )
  2611. {
  2612. cchGroupNameLen = ARRAYSIZE(szGroupName);
  2613. lResult = RegEnumKeyEx( hkeyGroups, dwSubKey, szGroupName,
  2614. &cchGroupNameLen, NULL, NULL, NULL, &ft );
  2615. szGroupName[MAXGROUPNAMELEN] = TEXT('\0');
  2616. if ( lResult == ERROR_NO_MORE_ITEMS )
  2617. {
  2618. break;
  2619. }
  2620. if ( lResult == ERROR_SUCCESS )
  2621. {
  2622. GetWindowsDirectory(szFileName, ARRAYSIZE(szFileName));
  2623. // Save this dir for use by GetTempFileName below
  2624. lstrcpy(szTempFileDir, szFileName);
  2625. #ifdef WINNT
  2626. GetEnvironmentVariable(TEXT("USERPROFILE"), szTempFileDir, MAX_PATH);
  2627. #endif
  2628. lstrcat(szFileName,TEXT("\\"));
  2629. lstrcat(szFileName,szGroupName);
  2630. lstrcat(szFileName,TEXT(".grp"));
  2631. //
  2632. // If the key has been modified since we last processed it,
  2633. // then time to process it again.
  2634. //
  2635. fProcess = FALSE;
  2636. if (fModifiedOnly)
  2637. {
  2638. if ( Group_ReadLastModDateTime(szFileName) != ft.dwLowDateTime )
  2639. {
  2640. fProcess = TRUE;
  2641. }
  2642. }
  2643. else
  2644. {
  2645. fProcess = TRUE;
  2646. }
  2647. if (fProcess)
  2648. {
  2649. if (GetTempFileName(szTempFileDir,TEXT("grp"),0,szTempFileName) != 0)
  2650. {
  2651. fOk = MakeGroupFile(szTempFileName,szGroupName);
  2652. if ( fOk )
  2653. {
  2654. if (fProgress && !fDialog)
  2655. {
  2656. Group_CreateProgressDlg();
  2657. fDialog = TRUE;
  2658. }
  2659. (*pfncb)(szTempFileName);
  2660. DeleteFile(szTempFileName);
  2661. Group_WriteLastModDateTime(szFileName,ft.dwLowDateTime);
  2662. cGroups++;
  2663. }
  2664. }
  2665. }
  2666. }
  2667. dwSubKey++;
  2668. }
  2669. RegCloseKey( hkeyGroups );
  2670. hkeyGroups = NULL;
  2671. if (fProgress && fDialog)
  2672. Group_DestroyProgressDlg();
  2673. return cGroups;
  2674. }
  2675. //---------------------------------------------------------------------------
  2676. // Find the progman ini from before an upgrade.
  2677. BOOL FindOldProgmanIni(LPTSTR pszPath)
  2678. {
  2679. if (Reg_GetString(HKEY_LOCAL_MACHINE, REGSTR_PATH_SETUP, REGSTR_VAL_OLDWINDIR, pszPath, MAX_PATH*SIZEOF(TCHAR)))
  2680. {
  2681. PathAppend(pszPath, c_szProgmanIni);
  2682. if (PathFileExists(pszPath))
  2683. {
  2684. return TRUE;
  2685. }
  2686. DebugMsg(DM_ERROR, TEXT("Can't find old progman.ini"));
  2687. return FALSE;
  2688. }
  2689. return FALSE;
  2690. }
  2691. //----------------------------------------------------------------------------
  2692. // Enumerate all the old groups.
  2693. void Group_EnumOldGroups(PFNGRPCALLBACK pfncb, BOOL fProgress)
  2694. {
  2695. TCHAR szIniFile[MAX_PATH], szFile[MAX_PATH];
  2696. UINT uSize;
  2697. LPTSTR pSection, pKey;
  2698. if (!FindOldProgmanIni(szIniFile))
  2699. return;
  2700. for (uSize = BIG_STEP; uSize < BIG_STEP * 8; uSize += BIG_STEP)
  2701. {
  2702. pSection = (LPTSTR)LocalAlloc(LPTR, uSize);
  2703. if (!pSection)
  2704. return;
  2705. if ((UINT)GetPrivateProfileString(c_szGroups, NULL, c_szNULL, pSection, uSize / sizeof(pSection[0]), szIniFile) < uSize - 5)
  2706. break;
  2707. LocalFree((HLOCAL)pSection);
  2708. pSection = NULL;
  2709. }
  2710. if (!pSection)
  2711. return;
  2712. if (fProgress)
  2713. Group_CreateProgressDlg();
  2714. for (pKey = pSection; *pKey; pKey += lstrlen(pKey) + 1)
  2715. {
  2716. GetPrivateProfileString(c_szGroups, pKey, c_szNULL, szFile, ARRAYSIZE(szFile), szIniFile);
  2717. if (szFile[0])
  2718. {
  2719. (*pfncb)(szFile);
  2720. }
  2721. }
  2722. if (fProgress)
  2723. Group_DestroyProgressDlg();
  2724. LocalFree((HLOCAL)pSection);
  2725. }
  2726. //----------------------------------------------------------------------------
  2727. // Given a pidl for a link, extract the appropriate info and append it to
  2728. // the app list.
  2729. void AppList_AppendCurrentItem(LPITEMIDLIST pidlFolder, LPSHELLFOLDER psf,
  2730. LPITEMIDLIST pidlItem, IShellLink *psl, IPersistFile *ppf)
  2731. {
  2732. STRRET str;
  2733. WCHAR wszPath[MAX_PATH];
  2734. TCHAR szName[MAX_PATH];
  2735. TCHAR sz[MAX_PATH];
  2736. TCHAR szPath[MAX_PATH];
  2737. TCHAR szModule[MAX_PATH];
  2738. TCHAR szVer[MAX_PATH];
  2739. ALITEM alitem;
  2740. if (SUCCEEDED(psf->lpVtbl->GetDisplayNameOf(psf, pidlItem, SHGDN_NORMAL, &str)))
  2741. {
  2742. // Get the name.
  2743. StrRetToStrN(szName, ARRAYSIZE(szName), &str, pidlItem);
  2744. DebugMsg(DM_TRACE, TEXT("c.gi_gi: Link %s"), szName);
  2745. // Get the path from the link...
  2746. SHGetPathFromIDList(pidlFolder, sz);
  2747. PathAppend(sz, szName);
  2748. lstrcat(sz, TEXT(".lnk"));
  2749. StrToOleStrN(wszPath, ARRAYSIZE(wszPath), sz, -1);
  2750. ppf->lpVtbl->Load(ppf, wszPath, 0);
  2751. // Copy all the data.
  2752. szPath[0] = TEXT('\0');
  2753. if (SUCCEEDED(psl->lpVtbl->GetPath(psl, szPath, ARRAYSIZE(szPath), NULL, SLGP_SHORTPATH)))
  2754. {
  2755. // Valid CL?
  2756. if (szPath[0])
  2757. {
  2758. GetVersionInfo(szPath, szModule, ARRAYSIZE(szModule), szVer, sizeof(szVer));
  2759. alitem.pszName = NULL;
  2760. alitem.pszPath = NULL;
  2761. alitem.pszModule = NULL;
  2762. alitem.pszVer = NULL;
  2763. Str_SetPtr(&alitem.pszName, szName);
  2764. Str_SetPtr(&alitem.pszPath, szPath);
  2765. Str_SetPtr(&alitem.pszModule, szModule);
  2766. Str_SetPtr(&alitem.pszVer, szVer);
  2767. DSA_AppendItem(g_hdsaAppList, &alitem);
  2768. }
  2769. }
  2770. }
  2771. }
  2772. //----------------------------------------------------------------------------
  2773. HRESULT AppList_ShellFolderEnum(LPITEMIDLIST pidlFolder, LPSHELLFOLDER psf)
  2774. {
  2775. HRESULT hres;
  2776. LPENUMIDLIST penum;
  2777. IShellLink *psl;
  2778. LPITEMIDLIST pidlItem;
  2779. UINT celt;
  2780. IPersistFile *ppf;
  2781. DWORD dwAttribs;
  2782. LPSHELLFOLDER psfItem;
  2783. LPITEMIDLIST pidlPath;
  2784. DebugMsg(DM_TRACE, TEXT("gc.al_sfe: Enum..."));
  2785. hres = psf->lpVtbl->EnumObjects(psf, (HWND)NULL, SHCONTF_FOLDERS | SHCONTF_NONFOLDERS, &penum);
  2786. if (SUCCEEDED(hres))
  2787. {
  2788. hres = ICoCreateInstance(&CLSID_ShellLink, &IID_IShellLink, &psl);
  2789. if (SUCCEEDED(hres))
  2790. {
  2791. psl->lpVtbl->QueryInterface(psl, &IID_IPersistFile, &ppf);
  2792. while ((penum->lpVtbl->Next(penum, 1, &pidlItem, &celt) == NOERROR) && (celt == 1))
  2793. {
  2794. dwAttribs = SFGAO_LINK|SFGAO_FOLDER;
  2795. if (SUCCEEDED(psf->lpVtbl->GetAttributesOf(psf, 1, &pidlItem, &dwAttribs)))
  2796. {
  2797. // Is it a folder
  2798. if (dwAttribs & SFGAO_FOLDER)
  2799. {
  2800. // Recurse.
  2801. DebugMsg(DM_TRACE, TEXT("al_sfe: Folder."));
  2802. hres = psf->lpVtbl->BindToObject(psf, pidlItem, NULL, &IID_IShellFolder, &psfItem);
  2803. if (SUCCEEDED(hres))
  2804. {
  2805. pidlPath = ILCombine(pidlFolder, pidlItem);
  2806. if (pidlPath)
  2807. {
  2808. AppList_ShellFolderEnum(pidlPath, psfItem);
  2809. psfItem->lpVtbl->Release(psfItem);
  2810. ILFree(pidlPath);
  2811. }
  2812. }
  2813. }
  2814. else if (dwAttribs & SFGAO_LINK)
  2815. {
  2816. // Regular link, add it to the list.
  2817. DebugMsg(DM_TRACE, TEXT("al_sfe: Link."));
  2818. AppList_AppendCurrentItem(pidlFolder, psf, pidlItem, psl, ppf);
  2819. }
  2820. }
  2821. SHFree(pidlItem);
  2822. }
  2823. ppf->lpVtbl->Release(ppf);
  2824. psl->lpVtbl->Release(psl);
  2825. }
  2826. penum->lpVtbl->Release(penum);
  2827. }
  2828. return hres;
  2829. }
  2830. //----------------------------------------------------------------------------
  2831. void Applist_SpecialFolderEnum(int nFolder)
  2832. {
  2833. HRESULT hres;
  2834. LPITEMIDLIST pidlGroup;
  2835. LPSHELLFOLDER psf, psfDesktop;
  2836. TCHAR sz[MAX_PATH];
  2837. // Get the group info.
  2838. if (SHGetSpecialFolderPath(NULL, sz, nFolder, FALSE))
  2839. {
  2840. pidlGroup = ILCreateFromPath(sz);
  2841. if (pidlGroup)
  2842. {
  2843. if (SUCCEEDED(ICoCreateInstance(&CLSID_ShellDesktop, &IID_IShellFolder, &psfDesktop)))
  2844. {
  2845. hres = psfDesktop->lpVtbl->BindToObject(psfDesktop, pidlGroup, NULL, &IID_IShellFolder, &psf);
  2846. if (SUCCEEDED(hres))
  2847. {
  2848. hres = AppList_ShellFolderEnum(pidlGroup, psf);
  2849. psf->lpVtbl->Release(psf);
  2850. }
  2851. psfDesktop->lpVtbl->Release(psfDesktop);
  2852. }
  2853. else
  2854. {
  2855. DebugMsg(DM_ERROR, TEXT("OneTree: failed to bind to Desktop root"));
  2856. }
  2857. ILFree(pidlGroup);
  2858. }
  2859. else
  2860. {
  2861. DebugMsg(DM_ERROR, TEXT("gc.al_acs: Can't create IDList for path.."));
  2862. }
  2863. }
  2864. else
  2865. {
  2866. DebugMsg(DM_ERROR, TEXT("gc.al_acs: Can't find programs folder."));
  2867. }
  2868. }
  2869. BOOL StartMenuIsProgramsParent(void)
  2870. {
  2871. LPITEMIDLIST pidlStart, pidlProgs;
  2872. BOOL fParent = FALSE;
  2873. if (SHGetSpecialFolderLocation(NULL, CSIDL_STARTMENU, &pidlStart))
  2874. {
  2875. if (SHGetSpecialFolderLocation(NULL, CSIDL_PROGRAMS, &pidlProgs))
  2876. {
  2877. if (ILIsParent(pidlStart, pidlProgs, FALSE))
  2878. fParent = TRUE;
  2879. ILFree(pidlProgs);
  2880. }
  2881. ILFree(pidlStart);
  2882. }
  2883. return fParent;
  2884. }
  2885. //---------------------------------------------------------------------------
  2886. // Return the links in a group.
  2887. void AppList_AddCurrentStuff(void)
  2888. {
  2889. DebugMsg(DM_TRACE, TEXT("gc.al_acs: Enumerating everything..."));
  2890. DebugMsg(DM_TRACE, TEXT("gc.al_acs: Enumerating StartMenu..."));
  2891. Applist_SpecialFolderEnum(CSIDL_STARTMENU);
  2892. if (!StartMenuIsProgramsParent())
  2893. {
  2894. DebugMsg(DM_TRACE, TEXT("gc.al_acs: Enumerating Programs..."));
  2895. Applist_SpecialFolderEnum(CSIDL_PROGRAMS);
  2896. }
  2897. }
  2898. // On NT we plan on converting NT formated group files into folders and links
  2899. // therefore we need the ability of supporting all of the NT group file formats
  2900. /*--------------------------------------------------------------------------*/
  2901. /* */
  2902. /* SIZEOFGroup() - */
  2903. /* */
  2904. /*--------------------------------------------------------------------------*/
  2905. DWORD SizeofGroup(LPNT_GROUPDEF lpgd)
  2906. {
  2907. LPNT_PMTAG lptag;
  2908. DWORD cbSeg;
  2909. DWORD cb;
  2910. cbSeg = (DWORD)GlobalSize(lpgd);
  2911. // The following needs to be verified
  2912. lptag = (LPNT_PMTAG)((LPSTR)lpgd+lpgd->cbGroup);
  2913. if ((DWORD)((PCHAR)lptag - (PCHAR)lpgd +MyDwordAlign(SIZEOF(NT_PMTAG))-MyDwordAlign(SIZEOF(lptag->rgb))+4) <= cbSeg
  2914. && lptag->wID == ID_MAGIC
  2915. && lptag->wItem == (int)0xFFFF
  2916. && lptag->cb == (WORD)(MyDwordAlign(SIZEOF(NT_PMTAG))-MyDwordAlign(SIZEOF(lptag->rgb)) + 4)
  2917. && *(PLONG)lptag->rgb == TAG_MAGIC)
  2918. {
  2919. while ((cb = (DWORD)((PCHAR)lptag - (PCHAR)lpgd + MyDwordAlign(SIZEOF(NT_PMTAG))-MyDwordAlign(SIZEOF(lptag->rgb)))) <= cbSeg)
  2920. {
  2921. if (lptag->wID == ID_LASTTAG)
  2922. return cb;
  2923. (LPSTR)lptag += lptag->cb;
  2924. }
  2925. }
  2926. return lpgd->cbGroup;
  2927. }
  2928. /*--------------------------------------------------------------------------*/
  2929. /* */
  2930. /* FindTag() - */
  2931. /* */
  2932. /*--------------------------------------------------------------------------*/
  2933. LPNT_PMTAG FindTag(LPNT_GROUPDEF lpgd, int item, WORD id)
  2934. {
  2935. LPNT_PMTAG lptag;
  2936. DWORD cbSeg;
  2937. DWORD cb;
  2938. cbSeg = (DWORD)GlobalSize(lpgd);
  2939. lptag = (LPNT_PMTAG)((LPSTR)lpgd+lpgd->cbGroup);
  2940. if ((PCHAR)lptag - (PCHAR)lpgd + MyDwordAlign(SIZEOF(NT_PMTAG))-MyDwordAlign(SIZEOF(lptag->rgb)) + 4 <= cbSeg
  2941. && lptag->wID == ID_MAGIC
  2942. && lptag->wItem == (int)0xFFFF
  2943. && lptag->cb == (WORD)(MyDwordAlign(SIZEOF(NT_PMTAG))-MyDwordAlign(SIZEOF(lptag->rgb)) +4)
  2944. && *(LONG *)lptag->rgb == TAG_MAGIC) {
  2945. while ((cb = (DWORD)((PCHAR)lptag - (PCHAR)lpgd + MyDwordAlign(SIZEOF(NT_PMTAG))-MyDwordAlign(SIZEOF(lptag->rgb)))) <= cbSeg)
  2946. {
  2947. if ((item == lptag->wItem)
  2948. && (id == 0 || id == lptag->wID)) {
  2949. return lptag;
  2950. }
  2951. if (lptag->wID == ID_LASTTAG)
  2952. return NULL;
  2953. (LPSTR)lptag += lptag->cb;
  2954. }
  2955. }
  2956. return NULL;
  2957. }
  2958. /*--------------------------------------------------------------------------*/
  2959. /* */
  2960. /* DeleteTag() - */
  2961. /* */
  2962. /* in: */
  2963. /* hGroup group handle, can be discardable (alwayws shrink object) */
  2964. /* */
  2965. /*--------------------------------------------------------------------------*/
  2966. VOID DeleteTag(HANDLE hGroup, int item, WORD id)
  2967. {
  2968. LPNT_PMTAG lptag;
  2969. LPWSTR lp1, lp2;
  2970. LPWSTR lpend;
  2971. LPNT_GROUPDEF lpgd;
  2972. lpgd = (LPNT_GROUPDEF) GlobalLock(hGroup);
  2973. lptag = FindTag(lpgd,item,id);
  2974. if (lptag == NULL) {
  2975. GlobalUnlock(hGroup);
  2976. return;
  2977. }
  2978. lp1 = (LPWSTR)lptag;
  2979. lp2 = (LPWSTR)((LPSTR)lptag + lptag->cb);
  2980. lpend = (LPWSTR)((LPSTR)lpgd + SizeofGroup(lpgd));
  2981. while (lp2 < lpend) {
  2982. *lp1++ = *lp2++;
  2983. }
  2984. /* always reallocing smaller
  2985. */
  2986. GlobalUnlock(hGroup);
  2987. GlobalReAlloc(hGroup, (DWORD)((LPSTR)lp1 - (LPSTR)lpgd), 0);
  2988. return;
  2989. }
  2990. /*--------------------------------------------------------------------------*/
  2991. /* */
  2992. /* AddTag() - */
  2993. /* */
  2994. /* in: */
  2995. /* h group handle, must not be discardable! */
  2996. /* */
  2997. /* returns: */
  2998. /* 0 failure */
  2999. /* 1 success */
  3000. /*--------------------------------------------------------------------------*/
  3001. INT AddTag(HANDLE h, int item, WORD id, LPWSTR lpbuf, UINT cb)
  3002. {
  3003. LPNT_PMTAG lptag;
  3004. WORD fAddFirst;
  3005. LPNT_GROUPDEF lpgd;
  3006. int cbNew;
  3007. int cbMyLen;
  3008. LPNT_GROUPDEF lpgdOld;
  3009. if (!cb && lpbuf) {
  3010. cb = SIZEOF(WCHAR)*(lstrlenW(lpbuf) + 1);
  3011. }
  3012. cbMyLen = MyDwordAlign(cb);
  3013. if (!lpbuf) {
  3014. cb = 0;
  3015. cbMyLen = 0;
  3016. }
  3017. /*
  3018. * Remove the old version of the tag, if any.
  3019. */
  3020. DeleteTag(h, item, id);
  3021. lpgd = (LPNT_GROUPDEF)GlobalLock(h);
  3022. lptag = FindTag(lpgd, (int)0xFFFF, (WORD)ID_LASTTAG);
  3023. if (!lptag) {
  3024. /*
  3025. * In this case, there are no tags at all, and we have to add
  3026. * the first tag, the interesting tag, and the last tag
  3027. */
  3028. cbNew = 3 * (MyDwordAlign(SIZEOF(NT_PMTAG)) - MyDwordAlign(SIZEOF(lptag->rgb))) + 4 + cbMyLen;
  3029. fAddFirst = TRUE;
  3030. lptag = (LPNT_PMTAG)((LPSTR)lpgd + lpgd->cbGroup);
  3031. } else {
  3032. /*
  3033. * In this case, only the interesting tag needs to be added
  3034. * but we count in the last because the delta is from lptag
  3035. */
  3036. cbNew = 2 * (MyDwordAlign(SIZEOF(NT_PMTAG)) - MyDwordAlign(SIZEOF(lptag->rgb))) + cbMyLen;
  3037. fAddFirst = FALSE;
  3038. }
  3039. /*
  3040. * check for 64K limit
  3041. */
  3042. if ((DWORD_PTR)lptag + cbNew < (DWORD_PTR)lptag) {
  3043. return 0;
  3044. }
  3045. cbNew += (DWORD)((PCHAR)lptag -(PCHAR)lpgd);
  3046. lpgdOld = lpgd;
  3047. GlobalUnlock(h);
  3048. if (!GlobalReAlloc(h, (DWORD)cbNew, GMEM_MOVEABLE)) {
  3049. return 0;
  3050. }
  3051. lpgd = (LPNT_GROUPDEF)GlobalLock(h);
  3052. lptag = (LPNT_PMTAG)((LPSTR)lpgd + ((LPSTR)lptag - (LPSTR)lpgdOld));
  3053. if (fAddFirst) {
  3054. /*
  3055. * Add the first tag
  3056. */
  3057. lptag->wID = ID_MAGIC;
  3058. lptag->wItem = (int)0xFFFF;
  3059. *(LONG *)lptag->rgb = TAG_MAGIC;
  3060. lptag->cb = (WORD)(MyDwordAlign(SIZEOF(NT_PMTAG)) - MyDwordAlign(SIZEOF(lptag->rgb)) + 4);
  3061. (LPSTR)lptag += lptag->cb;
  3062. }
  3063. /*
  3064. * Add the tag
  3065. */
  3066. lptag->wID = id;
  3067. lptag->wItem = item;
  3068. lptag->cb = (WORD)(MyDwordAlign(SIZEOF(NT_PMTAG)) - MyDwordAlign(SIZEOF(lptag->rgb)) + cbMyLen);
  3069. if (lpbuf) {
  3070. memmove(lptag->rgb, lpbuf, (WORD)cb);
  3071. }
  3072. (LPSTR)lptag += lptag->cb;
  3073. /*
  3074. * Add the end tag
  3075. */
  3076. lptag->wID = ID_LASTTAG;
  3077. lptag->wItem = (int)0xFFFF;
  3078. lptag->cb = 0;
  3079. GlobalUnlock(h);
  3080. return 1;
  3081. }
  3082. /*--------------------------------------------------------------------------*/
  3083. /* */
  3084. /* CreateNewGroupFromAnsiGroup() - */
  3085. /* */
  3086. /* This function creates a new, empty group. */
  3087. /* */
  3088. /*--------------------------------------------------------------------------*/
  3089. HANDLE CreateNewGroupFromAnsiGroup(LPNT_GROUPDEF_A lpGroupORI)
  3090. {
  3091. HANDLE hT;
  3092. LPNT_GROUPDEF lpgd;
  3093. int i;
  3094. int cb;
  3095. int cItems; // number of items in 16bit group
  3096. LPSTR pGroupName; // 32bit group name
  3097. LPWSTR pGroupNameUNI = NULL; // 32bit UNICODE group name
  3098. UINT wGroupNameLen; // length of pGroupName DWORD aligned.
  3099. INT cchWideChar = 0; //character count of resultant unicode string
  3100. INT cchMultiByte = 0;
  3101. pGroupName = (LPSTR)PTR(lpGroupORI, lpGroupORI->pName);
  3102. //
  3103. // convert pGroupName to unicode here
  3104. //
  3105. cchMultiByte=MultiByteToWideChar(CP_ACP,MB_PRECOMPOSED,pGroupName,
  3106. -1,pGroupNameUNI,cchWideChar) ;
  3107. pGroupNameUNI = LocalAlloc(LPTR,(++cchMultiByte)*SIZEOF(WCHAR)) ;
  3108. if (NULL == pGroupNameUNI)
  3109. goto Exit;
  3110. MultiByteToWideChar(CP_ACP,MB_PRECOMPOSED,pGroupName,
  3111. -1,pGroupNameUNI,cchMultiByte) ;
  3112. wGroupNameLen = MyDwordAlign(SIZEOF(WCHAR)*(lstrlenW(pGroupNameUNI) + 1));
  3113. cItems = lpGroupORI->cItems;
  3114. cb = SIZEOF(NT_GROUPDEF) + (cItems * SIZEOF(DWORD)) + wGroupNameLen;
  3115. //
  3116. // In CreateNewGroup before GlobalAlloc.
  3117. //
  3118. hT = GlobalAlloc(GHND, (DWORD)cb);
  3119. if (!hT) {
  3120. LocalFree((HLOCAL)pGroupNameUNI);
  3121. goto Exit;
  3122. }
  3123. lpgd = (LPNT_GROUPDEF)GlobalLock(hT);
  3124. //
  3125. // use the NT 1.0 group settings for what we can.
  3126. //
  3127. lpgd->nCmdShow = lpGroupORI->nCmdShow;
  3128. lpgd->wIconFormat = lpGroupORI->wIconFormat;
  3129. lpgd->cxIcon = lpGroupORI->cxIcon;
  3130. lpgd->cyIcon = lpGroupORI->cyIcon;
  3131. lpgd->ptMin.x = (INT)lpGroupORI->ptMin.x;
  3132. lpgd->ptMin.y = (INT)lpGroupORI->ptMin.y;
  3133. CopyRect(&(lpgd->rcNormal),&(lpGroupORI->rcNormal));
  3134. lpgd->dwMagic = GROUP_UNICODE;
  3135. lpgd->cbGroup = (DWORD)cb;
  3136. lpgd->pName = SIZEOF(NT_GROUPDEF) + cItems * SIZEOF(DWORD);
  3137. lpgd->Reserved1 = (WORD)-1;
  3138. lpgd->Reserved2 = (DWORD)-1;
  3139. lpgd->cItems = (WORD)cItems;
  3140. for (i = 0; i < cItems; i++) {
  3141. lpgd->rgiItems[i] = 0;
  3142. }
  3143. lstrcpyW((LPWSTR)((LPBYTE)lpgd + SIZEOF(NT_GROUPDEF) + cItems * SIZEOF(DWORD)),
  3144. pGroupNameUNI); // lhb tracks
  3145. LocalFree((HLOCAL)pGroupNameUNI);
  3146. GlobalUnlock(hT);
  3147. return(hT);
  3148. Exit:
  3149. return NULL;
  3150. }
  3151. /*--------------------------------------------------------------------------*/
  3152. /* */
  3153. /* AddThing() - */
  3154. /* */
  3155. /* in: */
  3156. /* hGroup group handle, must not be discardable */
  3157. /* lpStuff pointer to data or NULL to init data to zero */
  3158. /* cbStuff count of item (may be 0) if lpStuff is a string */
  3159. /* */
  3160. /* Adds an object to the group segment and returns its offset. Will */
  3161. /* reallocate the segment if necessary. */
  3162. /* */
  3163. /* Handle passed in must not be discardable */
  3164. /* */
  3165. /* returns: */
  3166. /* 0 failure */
  3167. /* > 0 offset to thing in the segment */
  3168. /* */
  3169. /*--------------------------------------------------------------------------*/
  3170. DWORD AddThing(HANDLE hGroup, LPWSTR lpStuff, DWORD cbStuff)
  3171. {
  3172. DWORD cb;
  3173. LPNT_GROUPDEF lpgd;
  3174. DWORD offset;
  3175. LPWSTR lpT;
  3176. DWORD cbStuffSize;
  3177. DWORD cbGroupSize;
  3178. DWORD myOffset;
  3179. if (cbStuff == 0xFFFFFFFF) {
  3180. return 0xFFFFFFFF;
  3181. }
  3182. if (!cbStuff) {
  3183. cbStuff = SIZEOF(WCHAR)*(DWORD)(1 + lstrlenW(lpStuff));
  3184. }
  3185. cbStuffSize = MyDwordAlign((int)cbStuff);
  3186. lpgd = (LPNT_GROUPDEF)GlobalLock(hGroup);
  3187. cb = SizeofGroup(lpgd);
  3188. cbGroupSize = MyDwordAlign((int)cb);
  3189. offset = lpgd->cbGroup;
  3190. myOffset = (DWORD)MyDwordAlign((int)offset);
  3191. GlobalUnlock(hGroup);
  3192. if (!GlobalReAlloc(hGroup,(DWORD)(cbGroupSize + cbStuffSize), GMEM_MOVEABLE))
  3193. return 0;
  3194. lpgd = (LPNT_GROUPDEF)GlobalLock(hGroup);
  3195. /*
  3196. * Slide the tags up
  3197. */
  3198. memmove((LPSTR)lpgd + myOffset + cbStuffSize, (LPSTR)lpgd + myOffset,
  3199. (cbGroupSize - myOffset));
  3200. lpgd->cbGroup += cbStuffSize;
  3201. lpT = (LPWSTR)((LPSTR)lpgd + myOffset);
  3202. if (lpStuff) {
  3203. memcpy(lpT, lpStuff, cbStuff);
  3204. } else {
  3205. /*
  3206. * Zero it
  3207. */
  3208. while (cbStuffSize--) {
  3209. *((LPBYTE)lpT)++ = 0;
  3210. }
  3211. }
  3212. GlobalUnlock(hGroup);
  3213. return myOffset;
  3214. }
  3215. DWORD AddThing_A(HANDLE hGroup, LPSTR lpStuff, WORD cbStuff)
  3216. {
  3217. LPWSTR lpStuffUNI = NULL;
  3218. BOOL bAlloc = FALSE;
  3219. DWORD cb;
  3220. if (cbStuff == 0xFFFF) {
  3221. return 0xFFFF;
  3222. }
  3223. if (!cbStuff) {
  3224. INT cchMultiByte;
  3225. INT cchWideChar = 0;
  3226. bAlloc = TRUE;
  3227. cchMultiByte=MultiByteToWideChar(CP_ACP,MB_PRECOMPOSED,lpStuff,
  3228. -1,lpStuffUNI,cchWideChar) ;
  3229. lpStuffUNI = LocalAlloc(LPTR,(++cchMultiByte)*SIZEOF(WCHAR)) ;
  3230. if (lpStuffUNI)
  3231. {
  3232. MultiByteToWideChar(CP_ACP,MB_PRECOMPOSED,lpStuff,
  3233. -1,lpStuffUNI,cchMultiByte) ;
  3234. cbStuff = (WORD)SIZEOF(WCHAR)*(1 + lstrlenW(lpStuffUNI)); // lhb tracks
  3235. }
  3236. } else {
  3237. lpStuffUNI = (LPWSTR)lpStuff;
  3238. }
  3239. if (lpStuffUNI)
  3240. {
  3241. cb = AddThing(hGroup, lpStuffUNI, cbStuff);
  3242. if (bAlloc)
  3243. LocalFree(lpStuffUNI);
  3244. }
  3245. else
  3246. {
  3247. cb = 0;
  3248. }
  3249. return(cb);
  3250. }
  3251. /*--------------------------------------------------------------------------*/
  3252. /* */
  3253. /* ConvertToUnicodeGroup() - */
  3254. /* */
  3255. /* returns the size of the new unicode group. */
  3256. /* */
  3257. /*--------------------------------------------------------------------------*/
  3258. int ConvertToUnicodeGroup(LPNT_GROUPDEF_A lpGroupORI, LPHANDLE lphNewGroup)
  3259. {
  3260. HANDLE hNewGroup;
  3261. LPNT_GROUPDEF lpgd;
  3262. LPNT_ITEMDEF lpid;
  3263. LPBYTE lpid_A;
  3264. LPNT_PMTAG lptag_A;
  3265. LPSTR lpTagValue;
  3266. WORD wTagId;
  3267. LPSTR lpT;
  3268. DWORD offset;
  3269. int cb;
  3270. int i;
  3271. INT cchMultiByte;
  3272. INT cchWideChar;
  3273. LPWSTR lpTagValueUNI;
  3274. BOOL bAlloc = FALSE;
  3275. hNewGroup = CreateNewGroupFromAnsiGroup(lpGroupORI);
  3276. if (!hNewGroup) {
  3277. return(0);
  3278. }
  3279. //
  3280. // Add all items to the new formatted group.
  3281. //
  3282. for (i = 0; i < (int)lpGroupORI->cItems; i++) {
  3283. //
  3284. // Get the pointer to the 16bit item
  3285. //
  3286. lpid_A = (LPBYTE)ITEM(lpGroupORI, i);
  3287. if (lpGroupORI->rgiItems[i]) {
  3288. //
  3289. // Create the item.
  3290. //
  3291. offset = AddThing(hNewGroup, NULL, SIZEOF(NT_ITEMDEF));
  3292. if (!offset) {
  3293. DebugMsg(DM_ERROR, TEXT("gc.ctug: AddThing NT_ITEMDEF failed"));
  3294. goto QuitThis;
  3295. }
  3296. lpgd = (LPNT_GROUPDEF)GlobalLock(hNewGroup);
  3297. lpgd->rgiItems[i] = offset;
  3298. lpid = ITEM(lpgd, i);
  3299. //
  3300. // Set the item's position.
  3301. //
  3302. lpid->pt.x = ((LPNT_ITEMDEF_A)lpid_A)->pt.x;
  3303. lpid->pt.y = ((LPNT_ITEMDEF_A)lpid_A)->pt.y;
  3304. //
  3305. // Add the item's Name.
  3306. //
  3307. GlobalUnlock(hNewGroup);
  3308. lpT = (LPSTR)PTR(lpGroupORI,((LPNT_ITEMDEF_A)lpid_A)->pName);
  3309. offset = AddThing_A(hNewGroup, lpT, 0);
  3310. if (!offset) {
  3311. DebugMsg(DM_ERROR, TEXT("gc.ctug: AddThing pName failed"));
  3312. goto PuntCreation;
  3313. }
  3314. lpgd = (LPNT_GROUPDEF)GlobalLock(hNewGroup);
  3315. lpid = ITEM(lpgd, i);
  3316. lpid->pName = offset;
  3317. //
  3318. // Add the item's Command line.
  3319. //
  3320. GlobalUnlock(hNewGroup);
  3321. lpT = (LPSTR)PTR(lpGroupORI, ((LPNT_ITEMDEF_A)lpid_A)->pCommand);
  3322. offset = AddThing_A(hNewGroup, lpT, 0);
  3323. if (!offset) {
  3324. DebugMsg(DM_ERROR, TEXT("gc.ctug: AddThing pCommand failed"));
  3325. goto PuntCreation;
  3326. }
  3327. lpgd = (LPNT_GROUPDEF)GlobalLock(hNewGroup);
  3328. lpid = ITEM(lpgd, i);
  3329. lpid->pCommand = offset;
  3330. //
  3331. // Add the item's Icon path.
  3332. //
  3333. GlobalUnlock(hNewGroup);
  3334. lpT = (LPSTR)PTR(lpGroupORI, ((LPNT_ITEMDEF_A)lpid_A)->pIconPath);
  3335. offset = AddThing_A(hNewGroup, lpT, 0);
  3336. if (!offset) {
  3337. DebugMsg(DM_ERROR, TEXT("gc.ctug: AddThing pIconPath failed"));
  3338. goto PuntCreation;
  3339. }
  3340. lpgd = (LPNT_GROUPDEF)GlobalLock(hNewGroup);
  3341. lpid = ITEM(lpgd, i);
  3342. lpid->pIconPath = offset;
  3343. //
  3344. // Get the item's icon resource using the Icon path and the icon index.
  3345. // And add the item's Icon resource.
  3346. //
  3347. lpid->iIcon = ((LPNT_ITEMDEF_A)lpid_A)->idIcon;
  3348. lpid->cbIconRes = ((LPNT_ITEMDEF_A)lpid_A)->cbIconRes;
  3349. lpid->wIconVer = ((LPNT_ITEMDEF_A)lpid_A)->wIconVer;
  3350. GlobalUnlock(hNewGroup);
  3351. lpT = (LPBYTE)PTR(lpGroupORI, ((LPNT_ITEMDEF_A)lpid_A)->pIconRes);
  3352. offset = AddThing_A(hNewGroup, (LPSTR)lpT, lpid->cbIconRes);
  3353. if (!offset) {
  3354. DebugMsg(DM_ERROR, TEXT("gc.ctug: AddThing pIconRes failed"));
  3355. goto PuntCreation;
  3356. }
  3357. lpgd = (LPNT_GROUPDEF)GlobalLock(hNewGroup);
  3358. lpid = ITEM(lpgd, i);
  3359. lpid->pIconRes = offset;
  3360. GlobalUnlock(hNewGroup);
  3361. }
  3362. }
  3363. /*
  3364. * Copy all the tags to the new group format.
  3365. */
  3366. lptag_A = (LPNT_PMTAG)((LPSTR)lpGroupORI + lpGroupORI->cbGroup); // lhb tracks
  3367. if (lptag_A->wID == ID_MAGIC &&
  3368. lptag_A->wItem == (int)0xFFFF &&
  3369. *(LONG *)lptag_A->rgb == TAG_MAGIC) {
  3370. //
  3371. // This is the first tag id, goto start of item tags.
  3372. //
  3373. (LPBYTE)lptag_A += lptag_A->cb;
  3374. while (lptag_A->wID != ID_LASTTAG) {
  3375. wTagId = lptag_A->wID;
  3376. cb = lptag_A->cb - (3 * SIZEOF(DWORD)); // cb - sizeof tag
  3377. if (wTagId == ID_MINIMIZE) {
  3378. lpTagValueUNI = NULL;
  3379. }
  3380. else {
  3381. lpTagValue = lptag_A->rgb ;
  3382. if (wTagId != ID_HOTKEY) {
  3383. bAlloc = TRUE;
  3384. cchWideChar = 0;
  3385. cchMultiByte=MultiByteToWideChar(CP_ACP,
  3386. MB_PRECOMPOSED,lpTagValue,
  3387. -1,NULL,cchWideChar) ;
  3388. lpTagValueUNI = LocalAlloc(LPTR,(++cchMultiByte)*SIZEOF(WCHAR)) ;
  3389. MultiByteToWideChar(CP_ACP,MB_PRECOMPOSED,lpTagValue,
  3390. -1,lpTagValueUNI,cchMultiByte) ;
  3391. cb = SIZEOF(WCHAR)*(lstrlenW(lpTagValueUNI) + 1); // lhb tracks
  3392. }
  3393. else {
  3394. lpTagValueUNI = (LPWSTR)lpTagValue;
  3395. }
  3396. }
  3397. if (! AddTag( hNewGroup,
  3398. lptag_A->wItem, // wItem
  3399. wTagId, // wID
  3400. lpTagValueUNI, // rgb : tag value
  3401. cb
  3402. )) {
  3403. DebugMsg(DM_ERROR, TEXT("gc.ctug: AddTag failed"));
  3404. }
  3405. if (bAlloc && lpTagValueUNI) {
  3406. LocalFree(lpTagValueUNI);
  3407. bAlloc = FALSE;
  3408. }
  3409. (LPBYTE)lptag_A += lptag_A->cb ; // go to next tag
  3410. }
  3411. }
  3412. lpgd = GlobalLock(hNewGroup);
  3413. cb = SizeofGroup(lpgd);
  3414. GlobalUnlock(hNewGroup);
  3415. *lphNewGroup = hNewGroup;
  3416. return(cb);
  3417. PuntCreation:
  3418. QuitThis:
  3419. if (hNewGroup) {
  3420. GlobalFree(hNewGroup);
  3421. }
  3422. return(0);
  3423. }