538 lines
14 KiB
C
538 lines
14 KiB
C
#include "local.h"
|
|
|
|
#include "resource.h"
|
|
|
|
#include <mluisupp.h>
|
|
|
|
#ifdef _HSFOLDER
|
|
#define LODWORD(_qw) (DWORD)(_qw)
|
|
|
|
// Invoke Command verb strings
|
|
const CHAR c_szOpen[] = "open";
|
|
const CHAR c_szDelcache[] = "delete";
|
|
const CHAR c_szProperties[] = "properties";
|
|
const CHAR c_szCopy[] = "copy";
|
|
|
|
|
|
void FileTimeToDateTimeStringInternal(FILETIME UNALIGNED *ulpft, LPTSTR pszText, int cchText, BOOL fUsePerceivedTime)
|
|
{
|
|
FILETIME ft;
|
|
FILETIME aft;
|
|
LPFILETIME lpft;
|
|
|
|
aft = *ulpft;
|
|
lpft = &aft;
|
|
|
|
if (!fUsePerceivedTime && (FILETIMEtoInt64(*lpft) != FT_NTFS_UNKNOWNGMT))
|
|
FileTimeToLocalFileTime(lpft, &ft);
|
|
else
|
|
ft = *lpft;
|
|
|
|
if (FILETIMEtoInt64(ft) == FT_NTFS_UNKNOWNGMT ||
|
|
FILETIMEtoInt64(ft) == FT_FAT_UNKNOWNLOCAL)
|
|
{
|
|
static TCHAR szNone[40] = {0};
|
|
if (szNone[0] == 0)
|
|
MLLoadString(IDS_HSFNONE, szNone, ARRAYSIZE(szNone));
|
|
|
|
StrCpyN(pszText, szNone, cchText);
|
|
}
|
|
else
|
|
{
|
|
TCHAR szTempStr[MAX_PATH];
|
|
LPTSTR pszTempStr = szTempStr;
|
|
SYSTEMTIME st;
|
|
|
|
FileTimeToSystemTime(&ft, &st);
|
|
|
|
if (GetDateFormat(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &st, NULL, szTempStr, ARRAYSIZE(szTempStr)) > 0)
|
|
{
|
|
int iLen = lstrlen(szTempStr);
|
|
ASSERT(TEXT('\0') == szTempStr[iLen]); // Make sure multi-byte isn't biting us.
|
|
pszTempStr = (LPTSTR)(pszTempStr + iLen);
|
|
*pszTempStr++ = ' ';
|
|
GetTimeFormat(LOCALE_USER_DEFAULT, TIME_NOSECONDS, &st, NULL, pszTempStr, ARRAYSIZE(szTempStr)-(iLen+1));
|
|
StrCpyN(pszText, szTempStr, cchText);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
HMENU LoadPopupMenu(UINT id, UINT uSubOffset)
|
|
{
|
|
HMENU hmParent, hmPopup;
|
|
|
|
HINSTANCE hinst = MLLoadShellLangResources();
|
|
hmParent = LoadMenu_PrivateNoMungeW(hinst, MAKEINTRESOURCEW(id));
|
|
if (!hmParent)
|
|
{
|
|
long error = GetLastError();
|
|
return NULL;
|
|
}
|
|
|
|
hmPopup = GetSubMenu(hmParent, uSubOffset);
|
|
RemoveMenu(hmParent, uSubOffset, MF_BYPOSITION);
|
|
DestroyMenu(hmParent);
|
|
|
|
MLFreeLibrary(hinst);
|
|
|
|
return hmPopup;
|
|
}
|
|
|
|
UINT MergePopupMenu(HMENU *phMenu, UINT idResource, UINT uSubOffset, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast)
|
|
{
|
|
HMENU hmMerge;
|
|
|
|
if (*phMenu == NULL)
|
|
{
|
|
*phMenu = CreatePopupMenu();
|
|
if (*phMenu == NULL)
|
|
return 0;
|
|
|
|
indexMenu = 0; // at the bottom
|
|
}
|
|
|
|
hmMerge = LoadPopupMenu(idResource, uSubOffset);
|
|
if (!hmMerge)
|
|
return 0;
|
|
|
|
idCmdLast = Shell_MergeMenus(*phMenu, hmMerge, indexMenu, idCmdFirst, idCmdLast, MM_ADDSEPARATOR);
|
|
|
|
DestroyMenu(hmMerge);
|
|
return idCmdLast;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Helper Fuctions for item.cpp and folder.cpp
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// copy and flatten the CACHE_ENTRY_INFO data
|
|
|
|
void _CopyCEI(UNALIGNED INTERNET_CACHE_ENTRY_INFO *pdst, LPINTERNET_CACHE_ENTRY_INFO psrc, DWORD dwBuffSize)
|
|
{
|
|
// This assumes how urlcache does allocation
|
|
memcpy(pdst, psrc, dwBuffSize);
|
|
|
|
// convert pointers to offsets
|
|
pdst->lpszSourceUrlName = (LPTSTR) PtrDifference(psrc->lpszSourceUrlName, psrc);
|
|
pdst->lpszLocalFileName = (LPTSTR) PtrDifference(psrc->lpszLocalFileName, psrc);
|
|
pdst->lpszFileExtension = (LPTSTR) PtrDifference(psrc->lpszFileExtension, psrc);
|
|
pdst->lpHeaderInfo = psrc->lpHeaderInfo ? (TCHAR*) PtrDifference(psrc->lpHeaderInfo, psrc) : NULL;
|
|
}
|
|
|
|
INT g_fProfilesEnabled = -1;
|
|
|
|
BOOL IsProfilesEnabled();
|
|
|
|
BOOL _FilterUserName(LPINTERNET_CACHE_ENTRY_INFO pcei, LPCTSTR pszCachePrefix, LPTSTR pszUserName)
|
|
{
|
|
TCHAR szTemp[MAX_URL_STRING];
|
|
LPCTSTR pszTemp = szTemp;
|
|
|
|
// chrisfra 3/27/97, more constant crapola. this all needs to be fixed.
|
|
TCHAR szPrefix[80];
|
|
BOOL fRet = 0;
|
|
|
|
if (g_fProfilesEnabled==-1)
|
|
{
|
|
g_fProfilesEnabled = IsProfilesEnabled();
|
|
}
|
|
|
|
if (g_fProfilesEnabled)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
StrCpyN(szTemp, pcei->lpszSourceUrlName, ARRAYSIZE(szTemp));
|
|
StrCpyN(szPrefix, pszCachePrefix, ARRAYSIZE(szPrefix));
|
|
StrCatBuff(szPrefix, pszUserName, ARRAYSIZE(szPrefix));
|
|
|
|
// find the '@' character if it exists
|
|
pszTemp = StrChr(pszTemp, TEXT('@'));
|
|
|
|
if ( (pszTemp) && (*pszTemp == TEXT('@')) )
|
|
{
|
|
fRet = (StrCmpNI(szTemp, szPrefix, lstrlen(szPrefix)) == 0);
|
|
}
|
|
else
|
|
{
|
|
fRet = (*pszUserName == TEXT('\0'));
|
|
}
|
|
|
|
return fRet;
|
|
}
|
|
|
|
|
|
BOOL _FilterPrefix(LPINTERNET_CACHE_ENTRY_INFO pcei, LPCTSTR pszCachePrefix)
|
|
{
|
|
#define MAX_PREFIX (80)
|
|
TCHAR szTemp[MAX_URL_STRING];
|
|
LPCTSTR pszStripped;
|
|
BOOL fRet = 0;
|
|
|
|
StrCpyN(szTemp, pcei->lpszSourceUrlName, ARRAYSIZE(szTemp));
|
|
pszStripped = _StripContainerUrlUrl(szTemp);
|
|
|
|
if (pszStripped && pszStripped-szTemp < MAX_PREFIX)
|
|
{
|
|
fRet = (StrCmpNI(szTemp, pszCachePrefix, ((int) (pszStripped-szTemp))/sizeof(TCHAR)) == 0);
|
|
}
|
|
return fRet;
|
|
}
|
|
|
|
LPCTSTR _StripContainerUrlUrl(LPCTSTR pszHistoryUrl)
|
|
{
|
|
// NOTE: for our purposes, we don't want a history URL if we can't detect our
|
|
// prefix, so we return NULL.
|
|
|
|
LPCTSTR pszTemp = pszHistoryUrl;
|
|
LPCTSTR pszCPrefix;
|
|
LPCTSTR pszReturn = NULL;
|
|
|
|
// Check for "Visited: "
|
|
pszCPrefix = c_szHistPrefix;
|
|
while (*pszTemp == *pszCPrefix && *pszTemp != TEXT('\0'))
|
|
{
|
|
pszTemp = CharNext(pszTemp);
|
|
pszCPrefix = CharNext(pszCPrefix);
|
|
}
|
|
|
|
if (*pszCPrefix == TEXT('\0'))
|
|
{
|
|
// Found "Visited: "
|
|
pszReturn = pszTemp;
|
|
}
|
|
else if (pszTemp == (LPTSTR) pszHistoryUrl)
|
|
{
|
|
// Check for ":YYYYMMDDYYYYMMDD: "
|
|
pszCPrefix = TEXT(":nnnnnnnnnnnnnnnn: ");
|
|
while (*pszTemp != TEXT('\0'))
|
|
{
|
|
if (*pszCPrefix == TEXT('n'))
|
|
{
|
|
if (*pszTemp < TEXT('0') || *pszTemp > TEXT('9')) break;
|
|
}
|
|
else if (*pszCPrefix != *pszTemp) break;
|
|
pszTemp = CharNext(pszTemp);
|
|
pszCPrefix = CharNext(pszCPrefix);
|
|
}
|
|
}
|
|
return (*pszCPrefix == TEXT('\0')) ? pszTemp : pszReturn;
|
|
}
|
|
|
|
LPCTSTR _StripHistoryUrlToUrl(LPCTSTR pszHistoryUrl)
|
|
{
|
|
LPCTSTR pszTemp = pszHistoryUrl;
|
|
|
|
if (!pszHistoryUrl)
|
|
return NULL;
|
|
|
|
pszTemp = StrChr(pszHistoryUrl, TEXT('@'));
|
|
if (pszTemp && *pszTemp)
|
|
return CharNext(pszTemp);
|
|
|
|
pszTemp = StrChr(pszHistoryUrl, TEXT(' '));
|
|
if (pszTemp && *pszTemp)
|
|
return CharNext(pszTemp);
|
|
else
|
|
return NULL; // error, the URL passed in wasn't a history url
|
|
}
|
|
|
|
// assumes this is a real URL and not a "history" url
|
|
void _GetURLHostFromUrl_NoStrip(LPCTSTR lpszUrl, LPTSTR szHost, DWORD dwHostSize, LPCTSTR pszLocalHost)
|
|
{
|
|
DWORD cch = dwHostSize;
|
|
if (S_OK != UrlGetPart(lpszUrl, szHost, &cch, URL_PART_HOSTNAME, 0)
|
|
|| !*szHost)
|
|
{
|
|
// default to the local host name.
|
|
StrCpyN(szHost, pszLocalHost, dwHostSize);
|
|
}
|
|
}
|
|
|
|
void _GetURLHost(LPINTERNET_CACHE_ENTRY_INFO pcei, LPTSTR szHost, DWORD dwHostSize, LPCTSTR pszLocalHost)
|
|
{
|
|
TCHAR szSourceUrl[MAX_URL_STRING];
|
|
|
|
ASSERT(ARRAYSIZE(szSourceUrl) > lstrlen(pcei->lpszSourceUrlName))
|
|
StrCpyN(szSourceUrl, pcei->lpszSourceUrlName, ARRAYSIZE(szSourceUrl));
|
|
|
|
_GetURLHostFromUrl(szSourceUrl, szHost, dwHostSize, pszLocalHost);
|
|
}
|
|
|
|
LPHEIPIDL _IsValid_HEIPIDL(LPCITEMIDLIST pidl)
|
|
{
|
|
LPHEIPIDL phei = (LPHEIPIDL)pidl;
|
|
|
|
if (phei && ((phei->cb > sizeof(HEIPIDL)) && (phei->usSign == (USHORT)HEIPIDL_SIGN)) &&
|
|
(phei->usUrl == 0 || phei->usUrl < phei->cb) &&
|
|
(phei->usTitle == 0 || (phei->usTitle + sizeof(WCHAR)) <= phei->cb))
|
|
{
|
|
return phei;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
LPBASEPIDL _IsValid_IDPIDL(LPCITEMIDLIST pidl)
|
|
{
|
|
LPBASEPIDL pcei = (LPBASEPIDL)pidl;
|
|
|
|
if (pcei && VALID_IDSIGN(pcei->usSign) && pcei->cb > 0)
|
|
{
|
|
return pcei;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
LPCTSTR _FindURLFileName(LPCTSTR pszURL)
|
|
{
|
|
LPCTSTR psz, pszRet = pszURL; // need to set to pszURL in case no '/'
|
|
LPCTSTR pszNextToLast = NULL;
|
|
|
|
for (psz = pszURL; *psz; psz = CharNext(psz))
|
|
{
|
|
if ((*psz == TEXT('\\') || *psz == TEXT('/')))
|
|
{
|
|
pszNextToLast = pszRet;
|
|
pszRet = CharNext(psz);
|
|
}
|
|
}
|
|
return *pszRet ? pszRet : pszNextToLast;
|
|
}
|
|
|
|
int _LaunchApp(HWND hwnd, LPCTSTR pszPath)
|
|
{
|
|
SHELLEXECUTEINFO ei = { 0 };
|
|
|
|
ei.cbSize = sizeof(SHELLEXECUTEINFO);
|
|
ei.hwnd = hwnd;
|
|
ei.lpFile = pszPath;
|
|
ei.nShow = SW_SHOWNORMAL;
|
|
|
|
return ShellExecuteEx(&ei);
|
|
}
|
|
|
|
|
|
int _LaunchAppForPidl(HWND hwnd, LPITEMIDLIST pidl)
|
|
{
|
|
SHELLEXECUTEINFO ei = { 0 };
|
|
|
|
ei.cbSize = sizeof(SHELLEXECUTEINFO);
|
|
ei.fMask = SEE_MASK_IDLIST;
|
|
ei.hwnd = hwnd;
|
|
ei.lpIDList = pidl;
|
|
ei.nShow = SW_SHOWNORMAL;
|
|
|
|
return ShellExecuteEx(&ei);
|
|
}
|
|
|
|
void _GenerateEvent(LONG lEventId, LPCITEMIDLIST pidlFolder, LPCITEMIDLIST pidlIn, LPCITEMIDLIST pidlNewIn)
|
|
{
|
|
LPITEMIDLIST pidl;
|
|
if (pidlIn)
|
|
{
|
|
pidl = ILCombine(pidlFolder, pidlIn);
|
|
}
|
|
else
|
|
{
|
|
pidl = ILClone(pidlFolder);
|
|
}
|
|
if (pidl)
|
|
{
|
|
if (pidlNewIn)
|
|
{
|
|
LPITEMIDLIST pidlNew = ILCombine(pidlFolder, pidlNewIn);
|
|
if (pidlNew)
|
|
{
|
|
SHChangeNotify(lEventId, SHCNF_IDLIST, pidl, pidlNew);
|
|
ILFree(pidlNew);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SHChangeNotify(lEventId, SHCNF_IDLIST, pidl, NULL);
|
|
}
|
|
ILFree(pidl);
|
|
}
|
|
}
|
|
|
|
const struct {
|
|
LPCSTR pszVerb;
|
|
UINT idCmd;
|
|
} rgcmds[] = {
|
|
{ c_szOpen, RSVIDM_OPEN },
|
|
{ c_szCopy, RSVIDM_COPY },
|
|
{ c_szDelcache, RSVIDM_DELCACHE },
|
|
{ c_szProperties, RSVIDM_PROPERTIES }
|
|
};
|
|
|
|
int _GetCmdID(LPCSTR pszCmd)
|
|
{
|
|
if (HIWORD(pszCmd))
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < ARRAYSIZE(rgcmds); i++)
|
|
{
|
|
if (StrCmpIA(rgcmds[i].pszVerb, pszCmd) == 0)
|
|
{
|
|
return rgcmds[i].idCmd;
|
|
}
|
|
}
|
|
|
|
return -1; // unknown
|
|
}
|
|
return (int)LOWORD(pszCmd);
|
|
}
|
|
|
|
HRESULT _CreatePropSheet(HWND hwnd, LPCITEMIDLIST pidl, int iDlg, DLGPROC pfnDlgProc, LPCTSTR pszTitle)
|
|
{
|
|
PROPSHEETPAGE psp = { 0 };
|
|
PROPSHEETHEADER psh = { 0 } ;
|
|
|
|
// initialize propsheet page.
|
|
psp.dwSize = sizeof(PROPSHEETPAGE);
|
|
psp.dwFlags = 0;
|
|
psp.hInstance = MLGetHinst();
|
|
psp.DUMMYUNION_MEMBER(pszTemplate) = MAKEINTRESOURCE(iDlg);
|
|
psp.DUMMYUNION2_MEMBER(pszIcon) = NULL;
|
|
psp.pfnDlgProc = pfnDlgProc;
|
|
psp.pszTitle = NULL;
|
|
psp.lParam = (LPARAM)pidl; // send it the cache entry struct
|
|
|
|
// initialize propsheet header.
|
|
psh.dwSize = sizeof(PROPSHEETHEADER);
|
|
psh.dwFlags = PSH_PROPSHEETPAGE | PSH_NOAPPLYNOW | PSH_PROPTITLE;
|
|
psh.hwndParent = hwnd;
|
|
psh.pszCaption = pszTitle;
|
|
psh.nPages = 1;
|
|
psh.DUMMYUNION2_MEMBER(nStartPage) = 0;
|
|
psh.DUMMYUNION3_MEMBER(ppsp) = (LPCPROPSHEETPAGE)&psp;
|
|
|
|
// invoke the property sheet
|
|
PropertySheet(&psh);
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
INT_PTR CALLBACK HistoryConfirmDeleteDlgProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
switch(message) {
|
|
|
|
case WM_INITDIALOG:
|
|
SetDlgItemText(hDlg, IDD_TEXT4, (LPCTSTR)lParam);
|
|
break;
|
|
|
|
case WM_DESTROY:
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
switch(LOWORD(wParam))
|
|
{
|
|
case IDYES:
|
|
case IDNO:
|
|
case IDCANCEL:
|
|
EndDialog(hDlg, wParam);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
// This function restores the Unicode characters from file system URLs
|
|
//
|
|
// If the URL isn't a file URL, it is copied directly to pszBuf. Otherwise, any
|
|
// UTF8-escaped parts of the URL are converted into Unicode, and the result is
|
|
// stored in pszBuf. This should be the same as the string we received in
|
|
// History in the first place
|
|
//
|
|
// The return value is always pszBuf.
|
|
// The input and output buffers may be the same.
|
|
|
|
|
|
LPCTSTR ConditionallyDecodeUTF8(LPCTSTR pszUrl, LPTSTR pszBuf, DWORD cchBuf)
|
|
{
|
|
BOOL fDecoded = FALSE;
|
|
|
|
if (PathIsFilePath(pszUrl))
|
|
{
|
|
TCHAR szDisplayUrl[MAX_URL_STRING];
|
|
DWORD cchDisplayUrl = ARRAYSIZE(szDisplayUrl);
|
|
DWORD cchBuf2 = cchBuf; // we may need the old cchBuf later
|
|
|
|
// After PrepareUrlForDisplayUTF8, we have a fully unescaped URL. If we
|
|
// ShellExec this, then Shell will unescape it again, so we need to re-escape
|
|
// it to preserve any real %dd sequences that might be in the string.
|
|
|
|
if (SUCCEEDED(PrepareURLForDisplayUTF8(pszUrl, szDisplayUrl, &cchDisplayUrl, TRUE)) &&
|
|
SUCCEEDED(UrlCanonicalize(szDisplayUrl, pszBuf, &cchBuf2, URL_ESCAPE_UNSAFE | URL_ESCAPE_PERCENT)))
|
|
{
|
|
fDecoded = TRUE;
|
|
}
|
|
}
|
|
|
|
if (!fDecoded && (pszUrl != pszBuf))
|
|
{
|
|
StrCpyN(pszBuf, pszUrl, cchBuf);
|
|
}
|
|
|
|
return pszBuf;
|
|
}
|
|
|
|
//
|
|
// These routines make a string into a legal filename by replacing
|
|
// all invalid characters with spaces.
|
|
//
|
|
// The list of invalid characters was obtained from the NT error
|
|
// message you get when you try to rename a file to an invalid name.
|
|
//
|
|
|
|
#ifndef UNICODE
|
|
#error The MakeLegalFilename code only works when it's part of a UNICODE build
|
|
#endif
|
|
|
|
//
|
|
// This function takes a string and makes it into a
|
|
// valid filename (by calling PathCleanupSpec).
|
|
//
|
|
// The PathCleanupSpec function wants to know what
|
|
// directory the file will live in. But it's going
|
|
// on the clipboard, so we don't know. We just
|
|
// guess the desktop.
|
|
//
|
|
// It only uses this path to decide if the filesystem
|
|
// supports long filenames or not, and to check for
|
|
// MAX_PATH overflow.
|
|
//
|
|
void MakeLegalFilenameW(LPWSTR pszFilename)
|
|
{
|
|
WCHAR szDesktopPath[MAX_PATH];
|
|
|
|
GetWindowsDirectoryW(szDesktopPath,MAX_PATH);
|
|
PathCleanupSpec(szDesktopPath,pszFilename);
|
|
|
|
}
|
|
|
|
//
|
|
// ANSI wrapper for above function
|
|
//
|
|
void MakeLegalFilenameA(LPSTR pszFilename)
|
|
{
|
|
WCHAR szFilenameW[MAX_PATH];
|
|
|
|
SHAnsiToUnicode(pszFilename, szFilenameW, MAX_PATH);
|
|
|
|
MakeLegalFilenameW(szFilenameW);
|
|
|
|
SHUnicodeToAnsi(szFilenameW, pszFilename, MAX_PATH);
|
|
|
|
}
|
|
|
|
#endif // _HSFOLDER
|