Windows2000/private/shell/ext/tweakui/desktop.c
2020-09-30 17:12:32 +02:00

613 lines
13 KiB
C

/*
* desktop - Dialog box property sheet for "desktop customization"
*/
#include "tweakui.h"
#pragma BEGIN_CONST_DATA
const GUID c_clsidBfc = { 0x85BBD920, 0x42A0, 0x1069,
{ 0xA2, 0xE4, 0x08, 0x00, 0x2B, 0x30, 0x30, 0x9D} };
const static DWORD CODESEG rgdwHelp[] = {
IDC_ICONLVTEXT, IDH_GROUP,
IDC_ICONLVTEXT2, IDH_ICONLV,
IDC_ICONLV, IDH_ICONLV,
IDC_CREATENOWTEXT, IDH_GROUP,
IDC_CREATENOW, IDH_CREATENOW,
#if 0
IDC_RESET, IDH_RESET,
#endif
0, 0,
};
#pragma END_CONST_DATA
/*
* cchFriendlyMax should be at least MAX_PATH, because we also use it
* to hold icon file names.
*/
#define cchFriendlyMax 256
/*
* Evil hack! Since the ::{guid} hack doesn't work unless the object
* really exists in the name space, we pull an evil trick and just
* create the idlist that the shell would've made if you had asked for
* it...
*/
typedef struct RIDL {
USHORT cb; /* Must be 20 */
USHORT sig; /* Must be 0x1F */
CLSID clsid; /* Guid goes here */
USHORT zero; /* Must be zero */
} RIDL, *PRIDL;
#define pidlPnsi(pnsi) ((PIDL)&(pnsi)->ridl)
/*
* The only abnormal key is Network Neighborhood, since its presence
* is controlled by a system policy...
* You need to set nsiflNever if something should not have a check
* box next to it because it cannot be added to the namespace.
* You need to set nsiflDir for things that aren't valid namespace
* items but should be created as directory-like objects.
* Briefcase is doubly abnormal, because it doesn't do anything at all!
* So we just exclude him from the enumeration.
*/
typedef BYTE NSIFL; /* Random flags */
#define nsiflNormal 1 /* Is a regular thing */
#define nsiflDir 2 /* Is a directory-like object */
#define nsiflNever 4 /* Not a valid namespace item */
#define nsiflEdited 8 /* The name has been edited */
typedef struct NSI { /* namespace item */
NSIFL nsifl; /* Is this a normal regkey? */
TCH tszClsid[ctchClsid]; /* Class id */
RIDL ridl; /* Regitem idlist */
} NSI, *PNSI;
#define insiPlvi(plvi) ((UINT)(plvi)->lParam)
#define pnsiInsi(insi) (&pddii->pnsi[insi])
#define pnsiPlvi(plvi) pnsiInsi(insiPlvi(plvi))
typedef struct DDII {
Declare_Gxa(NSI, nsi);
HKEY hkNS;
} DDII, *PDDII;
DDII ddii;
#define pddii (&ddii)
/*
* Desktop_GetClsidAttributes
* Return the Attributes registry key for a class id.
*/
#define ctchPathShellFolder 21
DWORD PASCAL
Desktop_GetClsidAttributes(PCTSTR ptszClsid)
{
TCH tsz[ctchPathShellFolder + ctchClsid];
wsprintf(tsz, c_tszPathShellFolder, ptszClsid);
return GetRegDword(hhkCR, tsz, c_tszAttributes, 0);
}
/*
* Desktop_HasSubkey
* Return whether the key has a child key with the specified name.
*/
BOOL PASCAL
Desktop_HasSubkey(HKEY hk, LPCTSTR ptszChild)
{
HKEY hkS;
if (_RegOpenKey(hk, ptszChild, &hkS) == 0) {
RegCloseKey(hkS);
return 1;
} else {
return 0;
}
}
/*
* Desktop_IsNSKey
* Determine whether the specified class is in the desktop namespace
* right now.
*/
#define Desktop_IsNSKey(ptszClsid) Desktop_HasSubkey(pddii->hkNS, ptszClsid)
/*
* Desktop_GetNetHood
* Determine whether the network neighborhood is visible now.
*/
#define Desktop_GetNetHood() GetRestriction(c_tszNoNetHood)
/*
* Desktop_SetNetHood
* Set the new network neighborhood visibility.
*/
#define Desktop_SetNetHood(f) SetRestriction(c_tszNoNetHood, f)
/*
* Desktop_IsHereNow
* Determine the state of the object as it is in the world today.
*/
#define Desktop_IsHereNow(pnsi, ptszClsid) \
((pnsi->nsifl & nsiflNormal) ? Desktop_IsNSKey(ptszClsid) \
: Desktop_GetNetHood())
/*
* Desktop_AddNSKey
* Okay, we've committed ourselves to adding the key to the listview.
* No turning back now!
* We default to the shared desktop, so set the initial state accordingly.
*/
void PASCAL
Desktop_AddNSKey(HWND hwnd, PNSI pnsi, LPCTSTR ptszClsid,
LPCTSTR ptszFriendly, int iImage, NSIFL nsifl)
{
pnsi->nsifl = nsifl;
lstrcpy(pnsi->tszClsid, ptszClsid);
LV_AddItem(hwnd, pddii->cnsi++, ptszFriendly, iImage,
(nsifl & nsiflNever) ? -1 : Desktop_IsHereNow(pnsi, ptszClsid));
}
/*
* Desktop_MakeRidl
* Initialize a RIDL from a GUID display name.
*/
HRESULT PASCAL
Desktop_MakeRidl(PNSI pnsi, LPCTSTR ptszClsid)
{
pnsi->ridl.cb = 20; /* Always */
pnsi->ridl.sig = 0x1F; /* Always */
pnsi->ridl.zero = 0; /* Always */
return Ole_ClsidFromString(ptszClsid, &pnsi->ridl.clsid);
}
/*
* Desktop_CheckNSKey
* Check and possibly add a new namespace key. My Computer is
* excluded because you can't get rid of it. Network Neighborhood
* is here, although it is somewhat weird. We handle the weirdness
* as it arises...
* Briefcase is excluded because it doesn't work.
* All that has been validated so far is that the key exists, it
* has a ShellEx subkey, and it has nonzero attributes.
* We haven't yet validated that it has an icon. We'll notice that
* when we try to build up the listview info.
* And people complain that lisp has too many levels of nesting...
*/
void PASCAL
Desktop_CheckNSKey(HWND hwnd, HKEY hk, LPCTSTR ptszClsid, NSIFL nsifl)
{
PNSI pnsi = Misc_AllocPx(&pddii->gxa);
if (pnsi) {
if (SUCCEEDED(Desktop_MakeRidl(pnsi, ptszClsid)) &&
!IsEqualGUID(&pnsi->ridl.clsid, &c_clsidBfc)) {
SHFILEINFO sfi;
if (SHGetFileInfo((LPCSTR)pidlPnsi(pnsi), 0, &sfi, cbX(sfi),
SHGFI_PIDL | SHGFI_DISPLAYNAME | SHGFI_SYSICONINDEX |
SHGFI_SMALLICON)) {
/*
* Stupid gacky Net Hood hack. Shell won't give me a name
* if I don't have one in the registry.
*/
if (sfi.szDisplayName[0] == 0 && !(nsifl & nsiflNormal)) {
LoadString(hinstCur, IDS_NETHOOD,
sfi.szDisplayName, cA(sfi.szDisplayName));
}
/*
* It must have a name and must have a custom icon.
*/
if (sfi.szDisplayName[0] && sfi.iIcon != 3) {
Desktop_AddNSKey(hwnd, pnsi, ptszClsid, sfi.szDisplayName,
sfi.iIcon, nsifl);
}
} /* couldn't get file info */
} /* not a valid guid */
} /* else no memory to store this entry */
}
/*
* Desktop_AddSpecialNSKey
* Add some special namespace keys, which eluded our enumeration.
*/
void PASCAL
Desktop_AddSpecialNSKey(HWND hwnd, LPCTSTR ptszClsid, NSIFL nsifl)
{
int insi;
HKEY hk;
for (insi = 0; insi < pddii->cnsi; insi++) {
if (lstrcmpi(pddii->pnsi[insi].tszClsid, ptszClsid) == 0) {
goto found;
}
}
hk = hkOpenClsid(ptszClsid);
if (hk) {
Desktop_CheckNSKey(hwnd, hk, ptszClsid, nsifl);
RegCloseKey(hk);
}
found:;
}
/*
* Desktop_EnumClasses
* Locate all the classes that are possible namespace keys.
*/
void PASCAL
Desktop_EnumClasses(HWND hwnd)
{
int ihk;
TCH tsz[ctchClsid];
for (ihk = 0; RegEnumKey(pcdii->hkClsid, ihk, tsz, cA(tsz)) == 0; ihk++) {
/* BUGBUG -- exclude bfc here */
HKEY hk = hkOpenClsid(tsz);
if (hk) {
if (Desktop_GetClsidAttributes(tsz)) {
Desktop_CheckNSKey(hwnd, hk, tsz, nsiflNormal);
}
RegCloseKey(hk);
}
}
Desktop_AddSpecialNSKey(hwnd, c_tszClsidNetHood, nsiflDir);
Desktop_AddSpecialNSKey(hwnd, c_tszClsidCpl,
nsiflNormal | nsiflNever | nsiflDir);
Desktop_AddSpecialNSKey(hwnd, c_tszClsidPrint,
nsiflNormal | nsiflNever | nsiflDir);
Misc_LV_SetCurSel(hwnd, 0); /* Default to top of list */
}
/*
* Desktop_OnInitDialog
* We have much nontrivial work to do. Fill all the list boxes with
* defaults.
*/
BOOL PASCAL
Desktop_OnInitDialog(HWND hwnd)
{
ZeroMemory(pddii, cbX(*pddii));
if (Misc_InitPgxa(&pddii->gxa, cbX(NSI))) {
if (RegCreateKey(pcdii->hkLMExplorer, c_tszDesktopNameSpace,
&pddii->hkNS) == 0) {
Desktop_EnumClasses(hwnd);
}
}
return 1;
}
/*
* Desktop_OnDestroy
* Free the memory we allocated.
* We also destroy the imagelist, because listview gets confused if
* it gets two image lists which are the same.
*/
BOOL PASCAL
Desktop_OnDestroy(HWND hdlg)
{
Misc_FreePgxa(&pddii->gxa);
if (pddii->hkNS) {
RegCloseKey(pddii->hkNS);
}
return 1;
}
#if 0
/*
* Desktop_FactoryReset
* This is scary and un-undoable, so let's do extra confirmation.
*/
void PASCAL
Desktop_FactoryReset(HWND hdlg)
{
if (MessageBoxId(hdlg, IDS_DESKTOPRESETOK,
tszName, MB_YESNO + MB_DEFBUTTON2) == IDYES) {
pcdii->fRunShellInf = 1;
Common_NeedLogoff(hdlg);
PropSheet_Apply(GetParent(hdlg));
}
}
#endif
/*
* Desktop_OnCreateNow
* Somebody asked to create another one...
* Use common dialogs to do the work.
*/
void PASCAL
Desktop_OnCreateNow(HWND hwnd, int iItem)
{
LV_ITEM lvi;
COFN cofn;
lvi.pszText = cofn.tsz;
lvi.cchTextMax = cA(cofn.tsz);
Misc_LV_GetItemInfo(hwnd, &lvi, iItem, LVIF_PARAM | LVIF_TEXT);
InitOpenFileName(GetParent(hwnd), &cofn, IDS_ALLFILES, cofn.tsz);
cofn.ofn.nMaxFile -= ctchClsid + 1; /* Leave room for dot and clsid */
cofn.ofn.Flags |= OFN_NOREADONLYRETURN;
if (GetSaveFileName(&cofn.ofn)) {
PNSI pnsi;
lstrcat(cofn.tsz, c_tszDot);
pnsi = pnsiPlvi(&lvi);
lstrcat(cofn.tsz, pnsi->tszClsid);
if ((pnsi->nsifl & nsiflDir) ||
Desktop_GetClsidAttributes(pnsi->tszClsid) & SFGAO_FOLDER) {
CreateDirectory(cofn.tsz, 0);
} else {
fCreateNil(cofn.tsz);
}
}
}
#if 0
/*
* Desktop_OnCommand
*/
void PASCAL
Desktop_OnCommand(HWND hdlg, int id, UINT codeNotify)
{
switch (id) {
case IDC_RESET:
if (codeNotify == BN_CLICKED) {
Desktop_FactoryReset(hdlg);
}
break;
}
}
#endif
/*
* Desktop_LV_Dirtify
* Mark this item as having been renamed during the property sheet
* page's lifetime.
*/
void PASCAL
Desktop_LV_Dirtify(LPARAM insi)
{
pddii->pnsi[insi].nsifl |= nsiflEdited;
}
/*
* Desktop_LV_GetIcon
* Produce the icon associated with an item. This is called when
* we need to rebuild the icon list after the icon cache has been
* purged.
*/
int PASCAL
Desktop_LV_GetIcon(LPARAM insi)
{
SHFILEINFO sfi;
sfi.iIcon = 0;
SHGetFileInfo((LPCSTR)pidlPnsi(pnsiInsi(insi)), 0, &sfi, cbX(sfi),
SHGFI_PIDL | SHGFI_SYSICONINDEX | SHGFI_SMALLICON);
return sfi.iIcon;
}
/*
* Desktop_OnSelChange
* Disable the "Create as File" button if we are on Net Hood.
*/
void PASCAL
Desktop_OnSelChange(HWND hwnd, int iItem)
{
PNSI pnsi = pnsiInsi(Misc_LV_GetParam(hwnd, iItem));
EnableWindow(GetDlgItem(GetParent(hwnd), IDC_CREATENOW),
pnsi->nsifl & nsiflNormal);
}
/*
* Desktop_OnApply
* Write the changes to the registry.
*/
void PASCAL
Desktop_OnApply(HWND hdlg)
{
HWND hwnd = GetDlgItem(hdlg, IDC_ICONLV);
int cItems = ListView_GetItemCount(hwnd);
BOOL fChanged = 0;
LV_ITEM lvi;
TCH tsz[cchFriendlyMax];
lvi.pszText = tsz;
lvi.cchTextMax = cA(tsz);
for (lvi.iItem = 0; lvi.iItem < cItems; lvi.iItem++) {
PNSI pnsi;
lvi.stateMask = LVIS_STATEIMAGEMASK;
Misc_LV_GetItemInfo(hwnd, &lvi, lvi.iItem,
LVIF_PARAM | LVIF_TEXT | LVIF_STATE);
pnsi = pnsiPlvi(&lvi);
if (Desktop_IsHereNow(pnsi, pnsi->tszClsid) != LV_IsChecked(&lvi)) {
fChanged = 1;
if (pnsi->nsifl & nsiflNormal) {
if (LV_IsChecked(&lvi)) {
HKEY hk;
if (RegCreateKey(pddii->hkNS, pnsi->tszClsid, &hk) == 0) {
RegSetValuePtsz(hk, 0, lvi.pszText);
RegCloseKey(hk);
}
} else {
RegDeleteTree(pddii->hkNS, pnsi->tszClsid);
}
} else { /* Ah, the Net Hood... */
Desktop_SetNetHood(LV_IsChecked(&lvi));
Common_NeedLogoff(hdlg);
if (!LV_IsChecked(&lvi)) {
if (MessageBoxId(hdlg, IDS_NONETHOOD, g_tszName, MB_YESNO)
== IDYES) {
WinHelp(hdlg, c_tszMyHelp, HELP_CONTEXT, IDH_NONETHOOD);
}
}
}
}
/* Not worth cacheing this */
if (pnsi->nsifl & nsiflEdited) {
SetNameOfPidl(psfDesktop, pidlPnsi(pnsi), lvi.pszText);
SHChangeNotify(SHCNE_UPDATEIMAGE, SHCNF_DWORD, (LPVOID)lvi.iImage, 0L);
}
}
if (fChanged) {
SHChangeNotify(SHCNE_ASSOCCHANGED, SHCNF_DWORD, 0L, 0L);
}
}
/*
* Desktop_LV_OnInitContextMenu
* Propagate the status of the Create Now button.
*/
void PASCAL
Desktop_LV_OnInitContextMenu(HWND hwnd, int iItem, HMENU hmenu)
{
Misc_EnableMenuFromHdlgId(hmenu, GetParent(hwnd), IDC_CREATENOW);
}
/*
* Oh yeah, we need this too.
*/
#pragma BEGIN_CONST_DATA
LVV lvvDesktop = {
#if 0
Desktop_OnCommand,
#else
0,
#endif
Desktop_LV_OnInitContextMenu,
Desktop_LV_Dirtify,
Desktop_LV_GetIcon,
Desktop_OnInitDialog,
Desktop_OnApply,
Desktop_OnDestroy,
Desktop_OnSelChange,
1, /* iMenu */
rgdwHelp,
0, /* Double-click action */
lvvflIcons | /* We need icons */
lvvflCanCheck | /* And check boxes */
lvvflCanRename, /* and you can rename by clicking */
{
{ IDC_CREATENOW, Desktop_OnCreateNow },
{ 0, 0 },
}
};
#pragma END_CONST_DATA
/*
* Our window procedure.
*/
BOOL EXPORT
Desktop_DlgProc(HWND hdlg, UINT wm, WPARAM wParam, LPARAM lParam)
{
return LV_DlgProc(&lvvDesktop, hdlg, wm, wParam, lParam);
}