2013-03-16 20:25:38 +00:00
|
|
|
/*
|
|
|
|
* PROJECT: ReactOS Software Control Panel
|
|
|
|
* FILE: dll/cpl/appwiz/createlink.c
|
|
|
|
* PURPOSE: ReactOS Software Control Panel
|
|
|
|
* PROGRAMMER: Gero Kuehn (reactos.filter@gkware.com)
|
|
|
|
* Dmitry Chapyshev (lentind@yandex.ru)
|
|
|
|
* Johannes Anderwald
|
2018-07-08 14:40:14 +00:00
|
|
|
* Katayama Hirofumi MZ (katayama.hirofumi.mz@gmail.com)
|
2007-10-08 15:56:31 +00:00
|
|
|
* UPDATE HISTORY:
|
|
|
|
* 06-17-2004 Created
|
2007-10-03 17:50:43 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "appwiz.h"
|
2020-02-09 05:42:26 +00:00
|
|
|
#include <commctrl.h>
|
2020-01-28 14:31:09 +00:00
|
|
|
#include <shellapi.h>
|
2018-07-08 14:40:14 +00:00
|
|
|
#include <strsafe.h>
|
2023-07-22 03:39:51 +00:00
|
|
|
#include <shlwapi_undoc.h> // for PathFindOnPathExW
|
2024-02-29 23:42:59 +00:00
|
|
|
#include "../../win32/shell32/shresdef.h" // for IDS_NEWITEMFORMAT and IDS_LNK_FILE
|
2014-01-19 10:17:54 +00:00
|
|
|
|
2009-09-14 22:55:44 +00:00
|
|
|
BOOL
|
|
|
|
IsShortcut(HKEY hKey)
|
|
|
|
{
|
|
|
|
WCHAR Value[10];
|
|
|
|
DWORD Size;
|
|
|
|
DWORD Type;
|
|
|
|
|
|
|
|
Size = sizeof(Value);
|
|
|
|
if (RegQueryValueExW(hKey, L"IsShortcut", NULL, &Type, (LPBYTE)Value, &Size) != ERROR_SUCCESS)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (Type != REG_SZ)
|
|
|
|
return FALSE;
|
|
|
|
|
2024-05-11 16:23:44 +00:00
|
|
|
return (_wcsicmp(Value, L"yes") == 0);
|
2009-09-14 22:55:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOL
|
|
|
|
IsExtensionAShortcut(LPWSTR lpExtension)
|
|
|
|
{
|
|
|
|
HKEY hKey;
|
|
|
|
WCHAR Buffer[100];
|
|
|
|
DWORD Size;
|
|
|
|
DWORD Type;
|
|
|
|
|
|
|
|
if (RegOpenKeyExW(HKEY_CLASSES_ROOT, lpExtension, 0, KEY_READ, &hKey) != ERROR_SUCCESS)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (IsShortcut(hKey))
|
|
|
|
{
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
Size = sizeof(Buffer);
|
|
|
|
if (RegQueryValueEx(hKey, NULL, NULL, &Type, (LPBYTE)Buffer, &Size) != ERROR_SUCCESS || Type != REG_SZ)
|
|
|
|
{
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
|
|
|
|
if (RegOpenKeyExW(HKEY_CLASSES_ROOT, Buffer, 0, KEY_READ, &hKey) != ERROR_SUCCESS)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (IsShortcut(hKey))
|
|
|
|
{
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2007-10-19 23:21:45 +00:00
|
|
|
BOOL
|
2007-10-04 23:51:10 +00:00
|
|
|
CreateShortcut(PCREATE_LINK_CONTEXT pContext)
|
|
|
|
{
|
2024-02-29 23:42:59 +00:00
|
|
|
IShellLinkW *pShellLink;
|
2007-10-04 23:51:10 +00:00
|
|
|
IPersistFile *pPersistFile;
|
|
|
|
HRESULT hr;
|
2009-09-14 22:55:44 +00:00
|
|
|
|
2024-02-29 23:42:59 +00:00
|
|
|
hr = CoCreateInstance(&CLSID_ShellLink, NULL, CLSCTX_ALL, &IID_IShellLinkW, (void**)&pShellLink);
|
|
|
|
if (FAILED(hr))
|
|
|
|
return FALSE;
|
2009-09-14 22:55:44 +00:00
|
|
|
|
2024-02-29 23:42:59 +00:00
|
|
|
hr = IUnknown_QueryInterface(pShellLink, &IID_IPersistFile, (void**)&pPersistFile);
|
|
|
|
if (FAILED(hr))
|
2024-02-28 06:22:44 +00:00
|
|
|
{
|
2024-02-29 23:42:59 +00:00
|
|
|
IUnknown_Release(pShellLink);
|
|
|
|
return FALSE;
|
2024-02-28 06:22:44 +00:00
|
|
|
}
|
2009-09-14 22:55:44 +00:00
|
|
|
|
2024-02-29 23:42:59 +00:00
|
|
|
if (IsExtensionAShortcut(PathFindExtensionW(pContext->szTarget)))
|
|
|
|
{
|
2009-09-14 22:55:44 +00:00
|
|
|
hr = pPersistFile->lpVtbl->Load(pPersistFile, (LPCOLESTR)pContext->szTarget, STGM_READ);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-02-29 23:42:59 +00:00
|
|
|
if (pContext->pidlTarget)
|
|
|
|
pShellLink->lpVtbl->SetIDList(pShellLink, pContext->pidlTarget);
|
|
|
|
else
|
|
|
|
pShellLink->lpVtbl->SetPath(pShellLink, pContext->szTarget);
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2024-02-29 23:42:59 +00:00
|
|
|
if (pContext->szArguments[0])
|
|
|
|
pShellLink->lpVtbl->SetArguments(pShellLink, pContext->szArguments);
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2024-02-29 23:42:59 +00:00
|
|
|
if (pContext->szDescription[0])
|
|
|
|
pShellLink->lpVtbl->SetDescription(pShellLink, pContext->szDescription);
|
2007-10-04 23:51:10 +00:00
|
|
|
}
|
|
|
|
|
2024-02-29 23:42:59 +00:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
hr = pPersistFile->lpVtbl->Save(pPersistFile, pContext->szLinkName, TRUE);
|
|
|
|
|
2018-07-08 14:40:14 +00:00
|
|
|
IUnknown_Release(pPersistFile);
|
|
|
|
IUnknown_Release(pShellLink);
|
2024-02-29 23:42:59 +00:00
|
|
|
return SUCCEEDED(hr);
|
2007-10-04 23:51:10 +00:00
|
|
|
}
|
|
|
|
|
2018-07-08 14:40:14 +00:00
|
|
|
BOOL
|
|
|
|
CreateInternetShortcut(PCREATE_LINK_CONTEXT pContext)
|
|
|
|
{
|
|
|
|
IUniformResourceLocatorW *pURL = NULL;
|
|
|
|
IPersistFile *pPersistFile = NULL;
|
|
|
|
HRESULT hr;
|
|
|
|
WCHAR szPath[MAX_PATH];
|
|
|
|
GetFullPathNameW(pContext->szLinkName, _countof(szPath), szPath, NULL);
|
|
|
|
|
|
|
|
hr = CoCreateInstance(&CLSID_InternetShortcut, NULL, CLSCTX_ALL,
|
|
|
|
&IID_IUniformResourceLocatorW, (void **)&pURL);
|
|
|
|
if (FAILED(hr))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
hr = IUnknown_QueryInterface(pURL, &IID_IPersistFile, (void **)&pPersistFile);
|
|
|
|
if (FAILED(hr))
|
|
|
|
{
|
|
|
|
IUnknown_Release(pURL);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
pURL->lpVtbl->SetURL(pURL, pContext->szTarget, 0);
|
|
|
|
|
|
|
|
hr = pPersistFile->lpVtbl->Save(pPersistFile, szPath, TRUE);
|
|
|
|
|
|
|
|
IUnknown_Release(pPersistFile);
|
|
|
|
IUnknown_Release(pURL);
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2018-07-08 14:40:14 +00:00
|
|
|
return SUCCEEDED(hr);
|
|
|
|
}
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2018-07-08 14:40:14 +00:00
|
|
|
BOOL IsInternetLocation(LPCWSTR pszLocation)
|
|
|
|
{
|
|
|
|
return (PathIsURLW(pszLocation) || wcsstr(pszLocation, L"www.") == pszLocation);
|
|
|
|
}
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2020-01-28 14:31:09 +00:00
|
|
|
/* Remove all invalid characters from the name */
|
|
|
|
void
|
|
|
|
DoConvertNameForFileSystem(LPWSTR szName)
|
|
|
|
{
|
|
|
|
LPWSTR pch1, pch2;
|
|
|
|
for (pch1 = pch2 = szName; *pch1; ++pch1)
|
|
|
|
{
|
|
|
|
if (wcschr(L"\\/:*?\"<>|", *pch1) != NULL)
|
|
|
|
{
|
|
|
|
/* *pch1 is an invalid character */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
*pch2 = *pch1;
|
|
|
|
++pch2;
|
|
|
|
}
|
|
|
|
*pch2 = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL
|
|
|
|
DoValidateShortcutName(PCREATE_LINK_CONTEXT pContext)
|
|
|
|
{
|
|
|
|
SIZE_T cch;
|
|
|
|
LPCWSTR pch, pszName = pContext->szDescription;
|
|
|
|
|
|
|
|
if (!pszName || !pszName[0])
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
cch = wcslen(pContext->szOrigin) + wcslen(pszName) + 1;
|
|
|
|
if (cch >= MAX_PATH)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
pch = pszName;
|
|
|
|
for (pch = pszName; *pch; ++pch)
|
|
|
|
{
|
|
|
|
if (wcschr(L"\\/:*?\"<>|", *pch) != NULL)
|
|
|
|
{
|
|
|
|
/* *pch is an invalid character */
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2007-10-03 17:50:43 +00:00
|
|
|
INT_PTR
|
|
|
|
CALLBACK
|
|
|
|
WelcomeDlgProc(HWND hwndDlg,
|
|
|
|
UINT uMsg,
|
|
|
|
WPARAM wParam,
|
|
|
|
LPARAM lParam)
|
|
|
|
{
|
2007-10-04 23:51:10 +00:00
|
|
|
LPPROPSHEETPAGEW ppsp;
|
2024-02-28 06:22:44 +00:00
|
|
|
PCREATE_LINK_CONTEXT pContext = (PCREATE_LINK_CONTEXT)GetWindowLongPtr(hwndDlg, DWLP_USER);
|
2007-10-04 23:51:10 +00:00
|
|
|
LPPSHNOTIFY lppsn;
|
2024-02-28 06:22:44 +00:00
|
|
|
WCHAR szPath[MAX_PATH * 2], szDisplayName[MAX_PATH];
|
2007-10-04 23:51:10 +00:00
|
|
|
WCHAR szDesc[100];
|
2023-07-20 11:17:22 +00:00
|
|
|
WCHAR szTitle[100];
|
2007-10-19 23:21:45 +00:00
|
|
|
BROWSEINFOW brws;
|
2007-10-04 23:51:10 +00:00
|
|
|
LPITEMIDLIST pidllist;
|
2020-01-28 14:31:09 +00:00
|
|
|
SHFILEINFOW FileInfo;
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2007-10-03 17:50:43 +00:00
|
|
|
switch(uMsg)
|
|
|
|
{
|
|
|
|
case WM_INITDIALOG:
|
2007-10-04 23:51:10 +00:00
|
|
|
ppsp = (LPPROPSHEETPAGEW)lParam;
|
|
|
|
pContext = (PCREATE_LINK_CONTEXT) ppsp->lParam;
|
|
|
|
SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pContext);
|
2007-10-03 17:50:43 +00:00
|
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), 0);
|
2024-02-29 23:42:59 +00:00
|
|
|
SHAutoComplete(GetDlgItem(hwndDlg, IDC_LINK_LOCATION), SHACF_DEFAULT);
|
2007-10-03 17:50:43 +00:00
|
|
|
break;
|
|
|
|
case WM_COMMAND:
|
2023-07-22 03:39:51 +00:00
|
|
|
{
|
2024-02-28 06:22:44 +00:00
|
|
|
switch (LOWORD(wParam))
|
2007-10-03 17:50:43 +00:00
|
|
|
{
|
2024-02-29 23:42:59 +00:00
|
|
|
case IDC_LINK_LOCATION:
|
2024-02-28 06:22:44 +00:00
|
|
|
{
|
|
|
|
if (HIWORD(wParam) == EN_CHANGE)
|
2007-10-03 17:50:43 +00:00
|
|
|
{
|
2024-02-28 06:22:44 +00:00
|
|
|
/* The text was changed by user. Invalidate pidlTarget. */
|
|
|
|
if (pContext->pidlTarget)
|
|
|
|
{
|
|
|
|
CoTaskMemFree(pContext->pidlTarget);
|
|
|
|
pContext->pidlTarget = NULL;
|
|
|
|
}
|
|
|
|
|
2024-02-29 23:42:59 +00:00
|
|
|
if (SendDlgItemMessage(hwndDlg, IDC_LINK_LOCATION, WM_GETTEXTLENGTH, 0, 0))
|
2024-02-28 06:22:44 +00:00
|
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_NEXT);
|
|
|
|
else
|
|
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), 0);
|
2007-10-03 17:50:43 +00:00
|
|
|
}
|
2007-10-04 23:51:10 +00:00
|
|
|
break;
|
2024-02-28 06:22:44 +00:00
|
|
|
}
|
2007-10-04 23:51:10 +00:00
|
|
|
case IDC_SHORTCUT_BROWSE:
|
2024-02-28 06:22:44 +00:00
|
|
|
{
|
2023-07-20 11:17:22 +00:00
|
|
|
LoadStringW(hApplet, IDS_BROWSE_FOR_TARGET, szTitle, _countof(szTitle));
|
2007-10-31 17:13:52 +00:00
|
|
|
ZeroMemory(&brws, sizeof(brws));
|
2007-10-04 23:51:10 +00:00
|
|
|
brws.hwndOwner = hwndDlg;
|
|
|
|
brws.pidlRoot = NULL;
|
2024-02-28 06:22:44 +00:00
|
|
|
brws.pszDisplayName = szDisplayName;
|
2023-07-20 11:17:22 +00:00
|
|
|
brws.lpszTitle = szTitle;
|
2024-02-28 06:22:44 +00:00
|
|
|
brws.ulFlags = BIF_BROWSEINCLUDEFILES | BIF_NEWDIALOGSTYLE | BIF_SHAREABLE;
|
2007-10-04 23:51:10 +00:00
|
|
|
brws.lpfn = NULL;
|
2013-05-19 21:21:20 +00:00
|
|
|
pidllist = SHBrowseForFolderW(&brws);
|
2007-10-04 23:51:10 +00:00
|
|
|
if (!pidllist)
|
|
|
|
break;
|
|
|
|
|
2024-02-28 06:22:44 +00:00
|
|
|
StringCchCopyW(pContext->szDescription, _countof(pContext->szDescription),
|
|
|
|
szDisplayName);
|
|
|
|
|
|
|
|
SHGetPathFromIDListW(pidllist, szPath);
|
|
|
|
|
|
|
|
if (PathFileExistsW(szPath) && !PathIsRelativeW(szPath))
|
2024-02-29 23:42:59 +00:00
|
|
|
SetDlgItemTextW(hwndDlg, IDC_LINK_LOCATION, szPath);
|
2023-07-19 23:46:51 +00:00
|
|
|
else
|
2024-02-29 23:42:59 +00:00
|
|
|
SetDlgItemTextW(hwndDlg, IDC_LINK_LOCATION, szDisplayName);
|
2023-07-19 23:46:51 +00:00
|
|
|
|
2024-02-29 23:42:59 +00:00
|
|
|
SendDlgItemMessageW(hwndDlg, IDC_LINK_LOCATION, EM_SETSEL, 0, -1);
|
2024-02-28 06:22:44 +00:00
|
|
|
|
|
|
|
if (pContext->pidlTarget)
|
|
|
|
CoTaskMemFree(pContext->pidlTarget);
|
|
|
|
pContext->pidlTarget = pidllist;
|
2007-10-04 23:51:10 +00:00
|
|
|
break;
|
2024-02-28 06:22:44 +00:00
|
|
|
}
|
2007-10-04 23:51:10 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
|
|
lppsn = (LPPSHNOTIFY) lParam;
|
2020-01-28 17:29:16 +00:00
|
|
|
if (lppsn->hdr.code == PSN_SETACTIVE)
|
|
|
|
{
|
2024-02-29 23:42:59 +00:00
|
|
|
SetDlgItemTextW(hwndDlg, IDC_LINK_LOCATION, pContext->szTarget);
|
2020-01-28 17:29:16 +00:00
|
|
|
}
|
|
|
|
else if (lppsn->hdr.code == PSN_WIZNEXT)
|
2007-10-04 23:51:10 +00:00
|
|
|
{
|
2024-02-29 23:42:59 +00:00
|
|
|
GetDlgItemTextW(hwndDlg, IDC_LINK_LOCATION, pContext->szTarget, _countof(pContext->szTarget));
|
2018-07-08 14:40:14 +00:00
|
|
|
StrTrimW(pContext->szTarget, L" \t");
|
2020-01-28 17:29:16 +00:00
|
|
|
ExpandEnvironmentStringsW(pContext->szTarget, szPath, _countof(szPath));
|
|
|
|
|
2023-07-22 03:39:51 +00:00
|
|
|
if (IsInternetLocation(szPath)) /* The internet location */
|
2007-10-04 23:51:10 +00:00
|
|
|
{
|
2018-07-08 14:40:14 +00:00
|
|
|
WCHAR szName[128];
|
|
|
|
LoadStringW(hApplet, IDS_NEW_INTERNET_SHORTCUT, szName, _countof(szName));
|
|
|
|
StringCchCopyW(pContext->szDescription, _countof(pContext->szDescription), szName);
|
2023-07-22 03:39:51 +00:00
|
|
|
pContext->szArguments[0] = 0;
|
|
|
|
return FALSE;
|
2018-07-08 14:40:14 +00:00
|
|
|
}
|
2023-07-22 03:39:51 +00:00
|
|
|
|
2024-02-28 06:22:44 +00:00
|
|
|
if (pContext->pidlTarget) /* The result of SHBrowseForFolderW */
|
|
|
|
{
|
|
|
|
SHGetPathFromIDListW(pContext->pidlTarget, pContext->szTarget);
|
|
|
|
pContext->szArguments[0] = 0;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Otherwise, the target is a command line or pathname */
|
|
|
|
|
2023-07-22 03:39:51 +00:00
|
|
|
/* Split and build args */
|
|
|
|
LPWSTR pszArgs = PathGetArgsW(szPath);
|
|
|
|
if (pszArgs && pszArgs > szPath)
|
2018-07-08 14:40:14 +00:00
|
|
|
{
|
2023-07-22 03:39:51 +00:00
|
|
|
PathRemoveArgsW(szPath);
|
|
|
|
StringCchCopyW(pContext->szArguments, _countof(pContext->szArguments), pszArgs);
|
2007-10-04 23:51:10 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-07-22 03:39:51 +00:00
|
|
|
pContext->szArguments[0] = 0;
|
|
|
|
}
|
2020-01-28 14:31:09 +00:00
|
|
|
|
2023-07-22 03:39:51 +00:00
|
|
|
/* Find the file */
|
2024-02-28 06:22:44 +00:00
|
|
|
if (!PathFindOnPathExW(szPath, NULL, WHICH_DEFAULT) &&
|
|
|
|
!PathFileExistsW(szPath))
|
2023-07-22 03:39:51 +00:00
|
|
|
{
|
|
|
|
/* Not found */
|
2024-02-29 23:42:59 +00:00
|
|
|
SendDlgItemMessageW(hwndDlg, IDC_LINK_LOCATION, EM_SETSEL, 0, -1);
|
2020-01-28 14:31:09 +00:00
|
|
|
|
2018-07-08 14:40:14 +00:00
|
|
|
LoadStringW(hApplet, IDS_CREATE_SHORTCUT, szDesc, _countof(szDesc));
|
|
|
|
LoadStringW(hApplet, IDS_ERROR_NOT_FOUND, szPath, _countof(szPath));
|
2023-07-22 03:39:51 +00:00
|
|
|
|
|
|
|
WCHAR szError[MAX_PATH + 100];
|
2018-07-08 14:40:14 +00:00
|
|
|
StringCchPrintfW(szError, _countof(szError), szPath, pContext->szTarget);
|
|
|
|
MessageBoxW(hwndDlg, szError, szDesc, MB_ICONERROR);
|
2020-01-28 14:31:09 +00:00
|
|
|
|
2023-07-22 03:39:51 +00:00
|
|
|
/* Prevent the wizard to go next */
|
2020-01-28 14:31:09 +00:00
|
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
|
|
|
|
return TRUE;
|
2007-10-04 23:51:10 +00:00
|
|
|
}
|
2023-07-22 03:39:51 +00:00
|
|
|
|
|
|
|
/* Rebuild target */
|
2024-02-28 06:22:44 +00:00
|
|
|
if (PathIsRelativeW(szPath))
|
|
|
|
GetFullPathNameW(szPath, _countof(pContext->szTarget), pContext->szTarget, NULL);
|
|
|
|
else
|
|
|
|
StringCchCopyW(pContext->szTarget, _countof(pContext->szTarget), szPath);
|
2023-07-22 03:39:51 +00:00
|
|
|
|
|
|
|
/* Get display name */
|
2024-02-28 06:22:44 +00:00
|
|
|
FileInfo.szDisplayName[0] = UNICODE_NULL;
|
|
|
|
if (SHGetFileInfoW(pContext->szTarget, 0, &FileInfo, sizeof(FileInfo), SHGFI_DISPLAYNAME))
|
2023-07-22 03:39:51 +00:00
|
|
|
StringCchCopyW(pContext->szDescription, _countof(pContext->szDescription), FileInfo.szDisplayName);
|
|
|
|
|
2024-02-28 06:22:44 +00:00
|
|
|
break;
|
2007-10-04 23:51:10 +00:00
|
|
|
}
|
2020-01-28 17:29:16 +00:00
|
|
|
else if (lppsn->hdr.code == PSN_RESET && !lppsn->lParam)
|
2020-01-27 13:58:58 +00:00
|
|
|
{
|
|
|
|
/* The user has clicked [Cancel] */
|
|
|
|
DeleteFileW(pContext->szOldFile);
|
|
|
|
SHChangeNotify(SHCNE_DELETE, SHCNF_PATHW, pContext->szOldFile, NULL);
|
|
|
|
}
|
2007-10-04 23:51:10 +00:00
|
|
|
break;
|
2023-07-22 03:39:51 +00:00
|
|
|
}
|
2007-10-03 17:50:43 +00:00
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
INT_PTR
|
|
|
|
CALLBACK
|
|
|
|
FinishDlgProc(HWND hwndDlg,
|
|
|
|
UINT uMsg,
|
|
|
|
WPARAM wParam,
|
|
|
|
LPARAM lParam)
|
|
|
|
{
|
2007-10-04 23:51:10 +00:00
|
|
|
LPPROPSHEETPAGEW ppsp;
|
|
|
|
PCREATE_LINK_CONTEXT pContext;
|
|
|
|
LPPSHNOTIFY lppsn;
|
2024-02-29 23:42:59 +00:00
|
|
|
WCHAR szText[MAX_PATH], szPath[MAX_PATH];
|
2020-01-28 14:31:09 +00:00
|
|
|
WCHAR szMessage[128];
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2007-10-03 17:50:43 +00:00
|
|
|
switch(uMsg)
|
|
|
|
{
|
|
|
|
case WM_INITDIALOG:
|
2007-10-04 23:51:10 +00:00
|
|
|
ppsp = (LPPROPSHEETPAGEW)lParam;
|
|
|
|
pContext = (PCREATE_LINK_CONTEXT) ppsp->lParam;
|
|
|
|
SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pContext);
|
2007-10-03 17:50:43 +00:00
|
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_FINISH);
|
|
|
|
break;
|
2007-10-04 23:51:10 +00:00
|
|
|
case WM_COMMAND:
|
|
|
|
switch(HIWORD(wParam))
|
|
|
|
{
|
|
|
|
case EN_CHANGE:
|
2008-08-06 18:18:43 +00:00
|
|
|
if (SendDlgItemMessage(hwndDlg, IDC_SHORTCUT_NAME, WM_GETTEXTLENGTH, 0, 0))
|
2007-10-04 23:51:10 +00:00
|
|
|
{
|
2020-01-28 14:31:09 +00:00
|
|
|
GetDlgItemTextW(hwndDlg, IDC_SHORTCUT_NAME, szText, _countof(szText));
|
|
|
|
StrTrimW(szText, L" \t");
|
|
|
|
if (szText[0])
|
|
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK | PSWIZB_FINISH);
|
|
|
|
else
|
|
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK);
|
2007-10-04 23:51:10 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
PropSheet_SetWizButtons(GetParent(hwndDlg), PSWIZB_BACK);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
|
|
lppsn = (LPPSHNOTIFY) lParam;
|
|
|
|
pContext = (PCREATE_LINK_CONTEXT) GetWindowLongPtr(hwndDlg, DWLP_USER);
|
2020-01-28 17:29:16 +00:00
|
|
|
if (lppsn->hdr.code == PSN_SETACTIVE)
|
|
|
|
{
|
2024-02-29 23:42:59 +00:00
|
|
|
/* Remove invalid characters */
|
2020-01-28 17:29:16 +00:00
|
|
|
DoConvertNameForFileSystem(pContext->szDescription);
|
2024-02-29 23:42:59 +00:00
|
|
|
PathCleanupSpec(NULL, pContext->szDescription);
|
|
|
|
|
|
|
|
/* Is it empty? (rare case) */
|
|
|
|
if (!pContext->szDescription[0])
|
|
|
|
{
|
|
|
|
HMODULE hShell32 = GetModuleHandleW(L"shell32.dll");
|
|
|
|
LoadStringW(hShell32, IDS_NEWITEMFORMAT, szText, _countof(szText));
|
|
|
|
LoadStringW(hShell32, IDS_LNK_FILE, szMessage, _countof(szMessage));
|
|
|
|
StringCchPrintfW(pContext->szDescription, _countof(pContext->szDescription),
|
|
|
|
szText, szMessage);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Build a path from szOldFile */
|
|
|
|
StringCchCopyW(szText, _countof(szText), pContext->szOldFile);
|
|
|
|
PathRemoveFileSpecW(szText);
|
|
|
|
|
|
|
|
/* Rename duplicate if necessary */
|
|
|
|
PathAddExtensionW(pContext->szDescription,
|
|
|
|
(IsInternetLocation(pContext->szTarget) ? L".url" : L".lnk"));
|
|
|
|
PathYetAnotherMakeUniqueName(szPath, szText, NULL, pContext->szDescription);
|
|
|
|
StringCchCopyW(pContext->szDescription, _countof(pContext->szDescription),
|
|
|
|
PathFindFileNameW(szPath));
|
|
|
|
PathRemoveExtensionW(pContext->szDescription);
|
|
|
|
|
2020-01-28 17:29:16 +00:00
|
|
|
SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_NAME, pContext->szDescription);
|
2020-04-30 06:02:33 +00:00
|
|
|
SendDlgItemMessageW(hwndDlg, IDC_SHORTCUT_NAME, EM_SETSEL, 0, -1);
|
|
|
|
SetFocus(GetDlgItem(hwndDlg, IDC_SHORTCUT_NAME));
|
2020-01-28 17:29:16 +00:00
|
|
|
}
|
|
|
|
else if (lppsn->hdr.code == PSN_WIZFINISH)
|
2007-10-04 23:51:10 +00:00
|
|
|
{
|
2020-01-28 14:31:09 +00:00
|
|
|
GetDlgItemTextW(hwndDlg, IDC_SHORTCUT_NAME, pContext->szDescription, _countof(pContext->szDescription));
|
2018-07-08 14:40:14 +00:00
|
|
|
StrTrimW(pContext->szDescription, L" \t");
|
|
|
|
|
2020-01-28 14:31:09 +00:00
|
|
|
if (!DoValidateShortcutName(pContext))
|
|
|
|
{
|
|
|
|
SendDlgItemMessageW(hwndDlg, IDC_SHORTCUT_NAME, EM_SETSEL, 0, -1);
|
|
|
|
|
|
|
|
LoadStringW(hApplet, IDS_INVALID_NAME, szMessage, _countof(szMessage));
|
|
|
|
MessageBoxW(hwndDlg, szMessage, NULL, MB_ICONERROR);
|
|
|
|
|
|
|
|
/* prevent the wizard to go next */
|
|
|
|
SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, -1);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2018-07-08 14:40:14 +00:00
|
|
|
/* if old shortcut file exists, then delete it now */
|
2020-01-27 13:58:58 +00:00
|
|
|
DeleteFileW(pContext->szOldFile);
|
|
|
|
SHChangeNotify(SHCNE_DELETE, SHCNF_PATHW, pContext->szOldFile, NULL);
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2018-07-08 14:40:14 +00:00
|
|
|
if (IsInternetLocation(pContext->szTarget))
|
|
|
|
{
|
|
|
|
/* internet */
|
|
|
|
StringCchCopyW(pContext->szLinkName, _countof(pContext->szLinkName),
|
|
|
|
pContext->szOrigin);
|
|
|
|
PathAppendW(pContext->szLinkName, pContext->szDescription);
|
|
|
|
|
|
|
|
/* change extension if any */
|
2023-07-22 03:39:51 +00:00
|
|
|
PathRemoveExtensionW(pContext->szLinkName);
|
2024-02-29 23:42:59 +00:00
|
|
|
PathAddExtensionW(pContext->szLinkName, L".url");
|
2018-07-08 14:40:14 +00:00
|
|
|
|
|
|
|
if (!CreateInternetShortcut(pContext))
|
|
|
|
{
|
|
|
|
LoadStringW(hApplet, IDS_CANTMAKEINETSHORTCUT, szMessage, _countof(szMessage));
|
|
|
|
MessageBoxW(hwndDlg, szMessage, NULL, MB_ICONERROR);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* file */
|
|
|
|
StringCchCopyW(pContext->szLinkName, _countof(pContext->szLinkName),
|
|
|
|
pContext->szOrigin);
|
|
|
|
PathAppendW(pContext->szLinkName, pContext->szDescription);
|
|
|
|
|
|
|
|
/* change extension if any */
|
2023-07-22 03:39:51 +00:00
|
|
|
PathRemoveExtensionW(pContext->szLinkName);
|
2024-02-29 23:42:59 +00:00
|
|
|
PathAddExtensionW(pContext->szLinkName, L".lnk");
|
2018-07-08 14:40:14 +00:00
|
|
|
|
|
|
|
if (!CreateShortcut(pContext))
|
|
|
|
{
|
|
|
|
WCHAR szMessage[128];
|
|
|
|
LoadStringW(hApplet, IDS_CANTMAKESHORTCUT, szMessage, _countof(szMessage));
|
|
|
|
MessageBoxW(hwndDlg, szMessage, NULL, MB_ICONERROR);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-01-28 17:29:16 +00:00
|
|
|
else if (lppsn->hdr.code == PSN_RESET && !lppsn->lParam)
|
2020-01-27 13:58:58 +00:00
|
|
|
{
|
|
|
|
/* The user has clicked [Cancel] */
|
|
|
|
DeleteFileW(pContext->szOldFile);
|
|
|
|
SHChangeNotify(SHCNE_DELETE, SHCNF_PATHW, pContext->szOldFile, NULL);
|
|
|
|
}
|
2018-07-08 14:40:14 +00:00
|
|
|
break;
|
2007-10-03 17:50:43 +00:00
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2018-12-14 11:06:57 +00:00
|
|
|
static int CALLBACK
|
|
|
|
PropSheetProc(HWND hwndDlg, UINT uMsg, LPARAM lParam)
|
|
|
|
{
|
|
|
|
// NOTE: This callback is needed to set large icon correctly.
|
|
|
|
HICON hIcon;
|
|
|
|
switch (uMsg)
|
|
|
|
{
|
|
|
|
case PSCB_INITIALIZED:
|
|
|
|
{
|
|
|
|
hIcon = LoadIconW(hApplet, MAKEINTRESOURCEW(IDI_APPINETICO));
|
|
|
|
SendMessageW(hwndDlg, WM_SETICON, ICON_BIG, (LPARAM)hIcon);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-10-03 17:50:43 +00:00
|
|
|
LONG CALLBACK
|
2020-02-09 05:42:26 +00:00
|
|
|
ShowCreateShortcutWizard(HWND hwndCPl, LPCWSTR szPath)
|
2007-10-03 17:50:43 +00:00
|
|
|
{
|
|
|
|
PROPSHEETHEADERW psh;
|
|
|
|
HPROPSHEETPAGE ahpsp[2];
|
|
|
|
PROPSHEETPAGE psp;
|
|
|
|
UINT nPages = 0;
|
2007-10-04 23:51:10 +00:00
|
|
|
UINT nLength;
|
2018-12-28 20:55:19 +00:00
|
|
|
PCREATE_LINK_CONTEXT pContext;
|
2018-12-31 07:42:53 +00:00
|
|
|
WCHAR szMessage[128];
|
2020-01-27 13:58:58 +00:00
|
|
|
LPWSTR pch;
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2018-12-28 20:55:19 +00:00
|
|
|
pContext = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*pContext));
|
2007-10-04 23:51:10 +00:00
|
|
|
if (!pContext)
|
|
|
|
{
|
2018-12-28 20:55:19 +00:00
|
|
|
/* no memory */
|
2018-12-31 07:42:53 +00:00
|
|
|
LoadStringW(hApplet, IDS_NO_MEMORY, szMessage, _countof(szMessage));
|
|
|
|
MessageBoxW(hwndCPl, szMessage, NULL, MB_ICONERROR);
|
2018-12-28 20:55:19 +00:00
|
|
|
return FALSE;
|
2007-10-04 23:51:10 +00:00
|
|
|
}
|
2018-07-08 14:40:14 +00:00
|
|
|
|
2007-10-04 23:51:10 +00:00
|
|
|
nLength = wcslen(szPath);
|
|
|
|
if (!nLength)
|
|
|
|
{
|
2011-10-02 12:22:39 +00:00
|
|
|
HeapFree(GetProcessHeap(), 0, pContext);
|
|
|
|
|
2007-10-04 23:51:10 +00:00
|
|
|
/* no directory given */
|
2018-12-31 07:42:53 +00:00
|
|
|
LoadStringW(hApplet, IDS_NO_DIRECTORY, szMessage, _countof(szMessage));
|
|
|
|
MessageBoxW(hwndCPl, szMessage, NULL, MB_ICONERROR);
|
2007-10-04 23:51:10 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
2010-07-16 12:52:37 +00:00
|
|
|
|
2020-01-27 13:58:58 +00:00
|
|
|
if (!PathFileExistsW(szPath))
|
2010-07-16 12:52:37 +00:00
|
|
|
{
|
2011-10-02 12:22:39 +00:00
|
|
|
HeapFree(GetProcessHeap(), 0, pContext);
|
|
|
|
|
2010-07-16 12:52:37 +00:00
|
|
|
/* invalid path */
|
2018-12-31 07:42:53 +00:00
|
|
|
LoadStringW(hApplet, IDS_INVALID_PATH, szMessage, _countof(szMessage));
|
|
|
|
MessageBoxW(hwndCPl, szMessage, NULL, MB_ICONERROR);
|
2010-07-16 12:52:37 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
2007-10-04 23:51:10 +00:00
|
|
|
|
2018-07-08 14:40:14 +00:00
|
|
|
/* build the pContext->szOrigin and pContext->szOldFile */
|
2020-01-27 13:58:58 +00:00
|
|
|
if (PathIsDirectoryW(szPath))
|
2007-10-04 23:51:10 +00:00
|
|
|
{
|
2020-01-27 13:58:58 +00:00
|
|
|
StringCchCopyW(pContext->szOrigin, _countof(pContext->szOrigin), szPath);
|
|
|
|
pContext->szOldFile[0] = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
StringCchCopyW(pContext->szOrigin, _countof(pContext->szOrigin), szPath);
|
2018-07-08 14:40:14 +00:00
|
|
|
pch = PathFindFileNameW(pContext->szOrigin);
|
2020-01-27 13:58:58 +00:00
|
|
|
if (pch && *pch)
|
|
|
|
*pch = 0;
|
|
|
|
|
|
|
|
StringCchCopyW(pContext->szOldFile, _countof(pContext->szOldFile), szPath);
|
|
|
|
|
|
|
|
pch = PathFindFileNameW(szPath);
|
2018-07-08 14:40:14 +00:00
|
|
|
if (pch && *pch)
|
2019-04-24 01:44:44 +00:00
|
|
|
{
|
|
|
|
/* build szDescription */
|
|
|
|
StringCchCopyW(pContext->szDescription, _countof(pContext->szDescription), pch);
|
|
|
|
*pch = 0;
|
|
|
|
|
|
|
|
pch = PathFindExtensionW(pContext->szDescription);
|
2018-07-08 14:40:14 +00:00
|
|
|
*pch = 0;
|
2019-04-24 01:44:44 +00:00
|
|
|
}
|
2007-10-04 23:51:10 +00:00
|
|
|
}
|
2018-07-08 14:40:14 +00:00
|
|
|
PathAddBackslashW(pContext->szOrigin);
|
2007-10-03 17:50:43 +00:00
|
|
|
|
|
|
|
/* Create the Welcome page */
|
|
|
|
psp.dwSize = sizeof(PROPSHEETPAGE);
|
|
|
|
psp.dwFlags = PSP_DEFAULT | PSP_HIDEHEADER;
|
|
|
|
psp.hInstance = hApplet;
|
|
|
|
psp.pfnDlgProc = WelcomeDlgProc;
|
2013-12-02 22:40:06 +00:00
|
|
|
psp.pszTemplate = MAKEINTRESOURCEW(IDD_SHORTCUT_LOCATION);
|
2007-10-04 23:51:10 +00:00
|
|
|
psp.lParam = (LPARAM)pContext;
|
2007-10-03 17:50:43 +00:00
|
|
|
ahpsp[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
|
|
|
|
/* Create the Finish page */
|
|
|
|
psp.dwFlags = PSP_DEFAULT | PSP_HIDEHEADER;
|
|
|
|
psp.pfnDlgProc = FinishDlgProc;
|
2013-12-02 22:40:06 +00:00
|
|
|
psp.pszTemplate = MAKEINTRESOURCEW(IDD_SHORTCUT_FINISH);
|
2007-10-03 17:50:43 +00:00
|
|
|
ahpsp[nPages++] = CreatePropertySheetPage(&psp);
|
|
|
|
|
|
|
|
/* Create the property sheet */
|
|
|
|
psh.dwSize = sizeof(PROPSHEETHEADER);
|
2018-12-14 11:06:57 +00:00
|
|
|
psh.dwFlags = PSH_WIZARD97 | PSH_WATERMARK | PSH_USEICONID | PSH_USECALLBACK;
|
2007-10-03 17:50:43 +00:00
|
|
|
psh.hInstance = hApplet;
|
2018-12-14 11:06:57 +00:00
|
|
|
psh.pszIcon = MAKEINTRESOURCEW(IDI_APPINETICO);
|
2007-10-03 17:50:43 +00:00
|
|
|
psh.hwndParent = NULL;
|
|
|
|
psh.nPages = nPages;
|
|
|
|
psh.nStartPage = 0;
|
|
|
|
psh.phpage = ahpsp;
|
2015-05-16 08:59:21 +00:00
|
|
|
psh.pszbmWatermark = MAKEINTRESOURCEW(IDB_SHORTCUT);
|
2018-12-14 11:06:57 +00:00
|
|
|
psh.pfnCallback = PropSheetProc;
|
2007-10-03 17:50:43 +00:00
|
|
|
|
|
|
|
/* Display the wizard */
|
|
|
|
PropertySheet(&psh);
|
2024-02-28 06:22:44 +00:00
|
|
|
|
|
|
|
CoTaskMemFree(pContext->pidlTarget);
|
2007-10-04 23:51:10 +00:00
|
|
|
HeapFree(GetProcessHeap(), 0, pContext);
|
2007-10-03 17:50:43 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
2007-10-04 23:51:10 +00:00
|
|
|
|
|
|
|
LONG
|
|
|
|
CALLBACK
|
|
|
|
NewLinkHereW(HWND hwndCPl, UINT uMsg, LPARAM lParam1, LPARAM lParam2)
|
|
|
|
{
|
2020-02-09 05:42:26 +00:00
|
|
|
InitCommonControls();
|
2024-03-22 11:58:24 +00:00
|
|
|
return ShowCreateShortcutWizard(hwndCPl, (LPCWSTR)lParam1);
|
2007-10-04 23:51:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LONG
|
|
|
|
CALLBACK
|
|
|
|
NewLinkHereA(HWND hwndCPl, UINT uMsg, LPARAM lParam1, LPARAM lParam2)
|
|
|
|
{
|
|
|
|
WCHAR szFile[MAX_PATH];
|
|
|
|
|
2024-03-22 11:58:24 +00:00
|
|
|
if (MultiByteToWideChar(CP_ACP, 0, (LPCSTR)lParam1, -1, szFile, _countof(szFile)))
|
|
|
|
return NewLinkHereW(hwndCPl, uMsg, (LPARAM)szFile, lParam2);
|
2007-10-04 23:51:10 +00:00
|
|
|
return -1;
|
|
|
|
}
|