744 lines
22 KiB
C++
744 lines
22 KiB
C++
|
/*
|
||
|
* contmenu.cpp - Context menu implementation for URL class.
|
||
|
*/
|
||
|
|
||
|
|
||
|
/* Headers
|
||
|
**********/
|
||
|
|
||
|
#include "project.hpp"
|
||
|
#pragma hdrstop
|
||
|
|
||
|
#include <mapi.h>
|
||
|
|
||
|
#include "resource.h"
|
||
|
|
||
|
#ifdef UNIX
|
||
|
#include <tchar.h>
|
||
|
#include <inetreg.h>
|
||
|
static HRESULT UnixReadNews(HWND hwndParent, PCSTR pszCmdLine);
|
||
|
#endif
|
||
|
|
||
|
/* Types
|
||
|
********/
|
||
|
|
||
|
/* MAPISendMail() typedef */
|
||
|
|
||
|
typedef ULONG (FAR PASCAL *MAPISENDMAILPROC)(LHANDLE lhSession, ULONG ulUIParam, lpMapiMessageA lpMessage, FLAGS flFlags, ULONG ulReserved);
|
||
|
|
||
|
/* RunDLL32 DLL entry point typedef */
|
||
|
|
||
|
typedef void (WINAPI *RUNDLL32PROC)(HWND hwndParent, HINSTANCE hinst, PSTR pszCmdLine, int nShowCmd);
|
||
|
|
||
|
|
||
|
/* Module Constants
|
||
|
*******************/
|
||
|
|
||
|
#pragma data_seg(DATA_SEG_READ_ONLY)
|
||
|
|
||
|
// case-insensitive
|
||
|
|
||
|
PRIVATE_DATA const char s_cszFileProtocolPrefix[] = "file:";
|
||
|
PRIVATE_DATA const char s_cszMailToProtocolPrefix[] = "mailto:";
|
||
|
PRIVATE_DATA const char s_cszRLoginProtocolPrefix[] = "rlogin:";
|
||
|
PRIVATE_DATA const char s_cszTelnetProtocolPrefix[] = "telnet:";
|
||
|
PRIVATE_DATA const char s_cszTN3270ProtocolPrefix[] = "tn3270:";
|
||
|
|
||
|
PRIVATE_DATA const char s_cszNewsDLL[] = "mcm.dll";
|
||
|
#ifndef UNIX
|
||
|
PRIVATE_DATA const char s_cszTelnetApp[] = "telnet.exe";
|
||
|
#else
|
||
|
PRIVATE_DATA const char s_cszTelnetApp[] = "telnet";
|
||
|
#endif
|
||
|
|
||
|
PRIVATE_DATA const char s_cszMAPISection[] = "Mail";
|
||
|
PRIVATE_DATA const char s_cszMAPIKey[] = "CMCDLLName32";
|
||
|
|
||
|
PRIVATE_DATA const char s_cszMAPISendMail[] = "MAPISendMail";
|
||
|
PRIVATE_DATA const char s_cszNewsProtocolHandler[] = "NewsProtocolHandler";
|
||
|
|
||
|
#pragma data_seg()
|
||
|
|
||
|
|
||
|
/***************************** Exported Functions ****************************/
|
||
|
|
||
|
|
||
|
#pragma warning(disable:4100) /* "unreferenced formal parameter" warning */
|
||
|
|
||
|
extern "C" void WINAPI OpenURL(HWND hwndParent, HINSTANCE hinst,
|
||
|
PSTR pszCmdLine, int nShowCmd)
|
||
|
{
|
||
|
HRESULT hr;
|
||
|
InternetShortcut intshcut;
|
||
|
int nResult;
|
||
|
|
||
|
DebugEntry(OpenURL);
|
||
|
|
||
|
ASSERT(IS_VALID_HANDLE(hwndParent, WND));
|
||
|
ASSERT(IS_VALID_HANDLE(hinst, INSTANCE));
|
||
|
ASSERT(IS_VALID_STRING_PTR(pszCmdLine, STR));
|
||
|
ASSERT(IsValidShowCmd(nShowCmd));
|
||
|
|
||
|
// Assume the entire command line is an Internet Shortcut file path.
|
||
|
|
||
|
TrimWhiteSpace(pszCmdLine);
|
||
|
|
||
|
TRACE_OUT(("OpenURL(): Trying to open Internet Shortcut %s.",
|
||
|
pszCmdLine));
|
||
|
|
||
|
hr = intshcut.LoadFromFile(pszCmdLine, TRUE);
|
||
|
|
||
|
if (hr == S_OK)
|
||
|
{
|
||
|
URLINVOKECOMMANDINFO urlici;
|
||
|
|
||
|
urlici.dwcbSize = sizeof(urlici);
|
||
|
urlici.hwndParent = hwndParent;
|
||
|
urlici.pcszVerb = NULL;
|
||
|
urlici.dwFlags = (IURL_INVOKECOMMAND_FL_ALLOW_UI |
|
||
|
IURL_INVOKECOMMAND_FL_USE_DEFAULT_VERB);
|
||
|
|
||
|
hr = intshcut.InvokeCommand(&urlici);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_LOADFROMFILE_FAILED),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult, pszCmdLine)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
DebugExitVOID(OpenURL);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
extern "C" void WINAPI FileProtocolHandler(HWND hwndParent, HINSTANCE hinst,
|
||
|
PSTR pszCmdLine, int nShowCmd)
|
||
|
{
|
||
|
char szDefaultVerb[MAX_PATH_LEN];
|
||
|
PCSTR pcszVerb;
|
||
|
HINSTANCE hinstExec;
|
||
|
int nResult;
|
||
|
PARSEDURL pu;
|
||
|
|
||
|
DebugEntry(FileProtocolHandler);
|
||
|
|
||
|
ASSERT(IS_VALID_HANDLE(hwndParent, WND));
|
||
|
ASSERT(IS_VALID_HANDLE(hinst, INSTANCE));
|
||
|
ASSERT(IS_VALID_STRING_PTR(pszCmdLine, STR));
|
||
|
ASSERT(IsValidShowCmd(nShowCmd));
|
||
|
|
||
|
// Assume the entire command line is a file: URL.
|
||
|
|
||
|
TrimWhiteSpace(pszCmdLine);
|
||
|
|
||
|
pu.cbSize = sizeof(pu);
|
||
|
if (S_OK == ParseURL(pszCmdLine, &pu) &&
|
||
|
URL_SCHEME_FILE == pu.nScheme)
|
||
|
{
|
||
|
pszCmdLine = (LPSTR)pu.pszSuffix;
|
||
|
}
|
||
|
|
||
|
// Get default verb if available.
|
||
|
|
||
|
if (GetPathDefaultVerb(pszCmdLine, szDefaultVerb, sizeof(szDefaultVerb)))
|
||
|
pcszVerb = szDefaultVerb;
|
||
|
else
|
||
|
pcszVerb = NULL;
|
||
|
|
||
|
TRACE_OUT(("FileProtocolHandler(): Invoking %s verb on %s.",
|
||
|
pcszVerb ? pcszVerb : "open",
|
||
|
pszCmdLine));
|
||
|
|
||
|
hinstExec = ShellExecute(hwndParent, pcszVerb, pszCmdLine, NULL, NULL,
|
||
|
nShowCmd);
|
||
|
|
||
|
if (hinstExec > (HINSTANCE)32)
|
||
|
TRACE_OUT(("FileProtocolHandler(): ShellExecute() %s verb on %s succeeded.",
|
||
|
pcszVerb ? pcszVerb : "open",
|
||
|
pszCmdLine));
|
||
|
else
|
||
|
{
|
||
|
WARNING_OUT(("FileProtocolHandler(): ShellExecute() %s verb on %s failed, returning %lu.",
|
||
|
pcszVerb ? pcszVerb : "open",
|
||
|
pszCmdLine,
|
||
|
hinstExec));
|
||
|
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_SHELLEXECUTE_FAILED),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult, pszCmdLine)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
DebugExitVOID(FileProtocolHandler);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
extern "C" void WINAPI MailToProtocolHandler(HWND hwndParent, HINSTANCE hinst,
|
||
|
PSTR pszCmdLine, int nShowCmd)
|
||
|
{
|
||
|
int nResult;
|
||
|
char szMAPIDLL[MAX_PATH_LEN];
|
||
|
|
||
|
DebugEntry(MailToProtocolHandler);
|
||
|
|
||
|
ASSERT(IS_VALID_HANDLE(hwndParent, WND));
|
||
|
ASSERT(IS_VALID_HANDLE(hinst, INSTANCE));
|
||
|
ASSERT(IS_VALID_STRING_PTR(pszCmdLine, STR));
|
||
|
ASSERT(IsValidShowCmd(nShowCmd));
|
||
|
|
||
|
if (GetProfileString(s_cszMAPISection, s_cszMAPIKey, EMPTY_STRING,
|
||
|
szMAPIDLL, sizeof(szMAPIDLL)) > 0)
|
||
|
{
|
||
|
HINSTANCE hinstMAPI;
|
||
|
|
||
|
TRACE_OUT(("MailToProtocolHandler(): MAPI provider DLL is %s.",
|
||
|
szMAPIDLL));
|
||
|
|
||
|
hinstMAPI = LoadLibrary(szMAPIDLL);
|
||
|
|
||
|
if (hinstMAPI)
|
||
|
{
|
||
|
MAPISENDMAILPROC MAPISendMailProc;
|
||
|
|
||
|
MAPISendMailProc = (MAPISENDMAILPROC)GetProcAddress(
|
||
|
hinstMAPI,
|
||
|
s_cszMAPISendMail);
|
||
|
|
||
|
if (MAPISendMailProc)
|
||
|
{
|
||
|
MapiRecipDescA mapito;
|
||
|
MapiMessage mapimsg;
|
||
|
ULONG ulResult;
|
||
|
|
||
|
// Assume the entire command line is a mailto: URL.
|
||
|
|
||
|
TrimWhiteSpace(pszCmdLine);
|
||
|
|
||
|
// Skip over any url: prefix.
|
||
|
|
||
|
if (! lstrnicmp(pszCmdLine, g_cszURLPrefix, g_ucbURLPrefixLen))
|
||
|
pszCmdLine += g_ucbURLPrefixLen;
|
||
|
|
||
|
// Skip over any mailto: prefix.
|
||
|
|
||
|
// (- 1) for null terminator.
|
||
|
|
||
|
if (! lstrnicmp(pszCmdLine, s_cszMailToProtocolPrefix,
|
||
|
SIZECHARS(s_cszMailToProtocolPrefix) - 1))
|
||
|
pszCmdLine += SIZECHARS(s_cszMailToProtocolPrefix) - 1;
|
||
|
|
||
|
ZeroMemory(&mapito, sizeof(mapito));
|
||
|
|
||
|
mapito.ulRecipClass = MAPI_TO;
|
||
|
mapito.lpszName = pszCmdLine;
|
||
|
|
||
|
ZeroMemory(&mapimsg, sizeof(mapimsg));
|
||
|
|
||
|
mapimsg.nRecipCount = 1;
|
||
|
mapimsg.lpRecips = &mapito;
|
||
|
|
||
|
TRACE_OUT(("MailToProtocolHandler(): Trying to send mail to %s.",
|
||
|
mapito.lpszName));
|
||
|
|
||
|
ulResult = (*MAPISendMailProc)(NULL, 0, &mapimsg,
|
||
|
(MAPI_LOGON_UI | MAPI_DIALOG), 0);
|
||
|
|
||
|
if (ulResult == SUCCESS_SUCCESS)
|
||
|
TRACE_OUT(("MailToProtocolHandler(): MAPISendMail() to %s succeeded.",
|
||
|
pszCmdLine));
|
||
|
else
|
||
|
{
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_MAPI_MAPISENDMAIL_FAILED),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_MAPI_GETPROCADDRESS_FAILED),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult,
|
||
|
szMAPIDLL, s_cszMAPISendMail)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
EVAL(FreeLibrary(hinstMAPI));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_MAPI_LOADLIBRARY_FAILED),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult, szMAPIDLL)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_NO_MAPI_PROVIDER),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
DebugExitVOID(MailToProtocolHandler);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
extern "C" void WINAPI NewsProtocolHandler(HWND hwndParent, HINSTANCE hinst,
|
||
|
PSTR pszCmdLine, int nShowCmd)
|
||
|
{
|
||
|
int nResult;
|
||
|
HINSTANCE hinstNews;
|
||
|
|
||
|
DebugEntry(NewsProtocolHandler);
|
||
|
|
||
|
ASSERT(IS_VALID_HANDLE(hwndParent, WND));
|
||
|
ASSERT(IS_VALID_HANDLE(hinst, INSTANCE));
|
||
|
ASSERT(IS_VALID_STRING_PTR(pszCmdLine, STR));
|
||
|
ASSERT(IsValidShowCmd(nShowCmd));
|
||
|
|
||
|
#ifndef UNIX
|
||
|
// Assume the entire command line is a news: URL.
|
||
|
|
||
|
TrimWhiteSpace(pszCmdLine);
|
||
|
|
||
|
// Skip over any url: prefix.
|
||
|
|
||
|
if (! lstrnicmp(pszCmdLine, g_cszURLPrefix, g_ucbURLPrefixLen))
|
||
|
pszCmdLine += g_ucbURLPrefixLen;
|
||
|
|
||
|
hinstNews = LoadLibrary(s_cszNewsDLL);
|
||
|
|
||
|
if (hinstNews)
|
||
|
{
|
||
|
RUNDLL32PROC RealNewsProtocolHandler;
|
||
|
|
||
|
RealNewsProtocolHandler = (RUNDLL32PROC)GetProcAddress(hinstNews,
|
||
|
s_cszNewsProtocolHandler);
|
||
|
|
||
|
if (RealNewsProtocolHandler)
|
||
|
{
|
||
|
TRACE_OUT(("NewsProtocolHandler(): Trying to open %s.",
|
||
|
pszCmdLine));
|
||
|
|
||
|
(*RealNewsProtocolHandler)(hwndParent, hinst, pszCmdLine, nShowCmd);
|
||
|
|
||
|
TRACE_OUT(("NewsProtocolHandler(): Returned from NewsProtocolHandler()."));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_NEWS_GETPROCADDRESS_FAILED),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
EVAL(FreeLibrary(hinstNews));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_NEWS_LOADLIBRARY_FAILED),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
}
|
||
|
#else
|
||
|
UnixReadNews(hwndParent, pszCmdLine);
|
||
|
#endif
|
||
|
|
||
|
DebugExitVOID(NewsProtocolHandler);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
#ifndef ISSPACE
|
||
|
#define ISSPACE(ch) (((ch) == 32) || ((unsigned)((ch) - 9)) <= 13 - 9)
|
||
|
#endif
|
||
|
#ifndef ISQUOTE
|
||
|
#define ISQUOTE(ch) ((ch) == '\"' || (ch) == '\'')
|
||
|
#endif
|
||
|
|
||
|
extern "C" void WINAPI TelnetProtocolHandler(HWND hwndParent, HINSTANCE hinst,
|
||
|
PSTR pszCmdLine, int nShowCmd)
|
||
|
{
|
||
|
HRESULT hr;
|
||
|
int nResult;
|
||
|
char *p;
|
||
|
char *pDest;
|
||
|
BOOL fRemove;
|
||
|
|
||
|
#ifdef UNIX
|
||
|
char szApp[MAX_PATH];
|
||
|
char szArgs[MAX_PATH];
|
||
|
|
||
|
#ifndef ANSI_SHELL32_ON_UNIX
|
||
|
char szCmdLine[MAX_PATH];
|
||
|
|
||
|
memset(szCmdLine, 0, MAX_PATH);
|
||
|
|
||
|
SHUnicodeToAnsi((LPWSTR)pszCmdLine, szCmdLine, MAX_PATH);
|
||
|
pszCmdLine = szCmdLine;
|
||
|
#endif /* ANSI_SHELL32_ON_UNIX */
|
||
|
#endif
|
||
|
|
||
|
|
||
|
DebugEntry(TelnetProtocolHandler);
|
||
|
|
||
|
ASSERT(IS_VALID_HANDLE(hwndParent, WND));
|
||
|
ASSERT(IS_VALID_HANDLE(hinst, INSTANCE));
|
||
|
ASSERT(IS_VALID_STRING_PTR(pszCmdLine, STR));
|
||
|
ASSERT(IsValidShowCmd(nShowCmd));
|
||
|
|
||
|
// Assume the entire command line is a telnet URL.
|
||
|
|
||
|
TrimWhiteSpace(pszCmdLine);
|
||
|
|
||
|
// Skip over any url: prefix.
|
||
|
|
||
|
if (! lstrnicmp(pszCmdLine, g_cszURLPrefix, g_ucbURLPrefixLen))
|
||
|
pszCmdLine += g_ucbURLPrefixLen;
|
||
|
|
||
|
// Skip over any telnet:, rlogin:, or tn3270: prefix.
|
||
|
|
||
|
// (- 1) for null terminator.
|
||
|
|
||
|
if (! lstrnicmp(pszCmdLine, s_cszTelnetProtocolPrefix,
|
||
|
SIZECHARS(s_cszTelnetProtocolPrefix) - 1))
|
||
|
pszCmdLine += SIZECHARS(s_cszTelnetProtocolPrefix) - 1;
|
||
|
else if (! lstrnicmp(pszCmdLine, s_cszRLoginProtocolPrefix,
|
||
|
SIZECHARS(s_cszRLoginProtocolPrefix) - 1))
|
||
|
pszCmdLine += SIZECHARS(s_cszRLoginProtocolPrefix) - 1;
|
||
|
else if (! lstrnicmp(pszCmdLine, s_cszTN3270ProtocolPrefix,
|
||
|
SIZECHARS(s_cszTN3270ProtocolPrefix) - 1))
|
||
|
pszCmdLine += SIZECHARS(s_cszTN3270ProtocolPrefix) - 1;
|
||
|
|
||
|
// Remove leading and trailing slashes.
|
||
|
|
||
|
TrimSlashes(pszCmdLine);
|
||
|
|
||
|
// Skip user name if given
|
||
|
|
||
|
p = StrChr(pszCmdLine, '@');
|
||
|
|
||
|
if (p)
|
||
|
pszCmdLine = p + 1;
|
||
|
|
||
|
// Eliminate double quotes...should be no need for these
|
||
|
// unless trouble is afoot.
|
||
|
for (pDest = p = pszCmdLine; *p; p++)
|
||
|
{
|
||
|
if (!ISQUOTE(*p))
|
||
|
{
|
||
|
*pDest = *p;
|
||
|
pDest++;
|
||
|
}
|
||
|
}
|
||
|
*pDest = '\0';
|
||
|
|
||
|
// For security reasons, strip the filename cmdline option
|
||
|
if (pszCmdLine)
|
||
|
{
|
||
|
for (p = pszCmdLine; *p; p++)
|
||
|
{
|
||
|
// Be careful and don't nuke servernames that start with -f.
|
||
|
// Since hostnames can't start with a dash, ensure previous char is
|
||
|
// whitespace, or we're at the beginning.
|
||
|
//
|
||
|
// Also, -a sends credentials over the wire, so strip it, too.
|
||
|
if ((*p == '/' || *p == '-') &&
|
||
|
(*(p+1) == 'f' || *(p+1) == 'F' || *(p+1) == 'a' || *(p+1) == 'A'))
|
||
|
{
|
||
|
fRemove = TRUE;
|
||
|
if (!((p == pszCmdLine || ISSPACE(*(p-1)) || ISQUOTE(*(p-1)) )))
|
||
|
{
|
||
|
char *pPortChar = p-1;
|
||
|
// Doesn't meet easy criteria, but it might be harder to
|
||
|
// detect, such as site:-ffilename. In this case, consider
|
||
|
// the -f piece unsafe if everything between -f and a colon
|
||
|
// to the left is a digit (no digits will also be unsafe).
|
||
|
// If anything else is hit first, then consider it to
|
||
|
// be part of the hostname. Walking to the beginning
|
||
|
// be considered safe (e.g. "80-ffilename" would be considered
|
||
|
// the hostname).
|
||
|
while (pPortChar >= pszCmdLine && *pPortChar != ':')
|
||
|
{
|
||
|
if (*pPortChar < '0' || *pPortChar > '9')
|
||
|
{
|
||
|
fRemove = FALSE;
|
||
|
break;
|
||
|
}
|
||
|
pPortChar--;
|
||
|
}
|
||
|
if (pPortChar < pszCmdLine)
|
||
|
fRemove = FALSE;
|
||
|
}
|
||
|
|
||
|
if (!fRemove)
|
||
|
continue;
|
||
|
|
||
|
BOOL fQuotedFilename = FALSE;
|
||
|
LPSTR pStart = p;
|
||
|
|
||
|
// move past -f
|
||
|
p+=2;
|
||
|
|
||
|
// Skip over whitespace and filename following -f option
|
||
|
if (*(p-1) == 'f' || *(p-1) == 'F')
|
||
|
{
|
||
|
while (*p && ISSPACE(*p))
|
||
|
p++;
|
||
|
|
||
|
// but wait, it may be a long filename surrounded by quotes
|
||
|
if (ISQUOTE(*p))
|
||
|
{
|
||
|
fQuotedFilename = TRUE;
|
||
|
p++;
|
||
|
}
|
||
|
|
||
|
// Loop until null OR whitespace if not quoted pathname OR quote if a quoted pathname
|
||
|
while (!((*p == '\0') ||
|
||
|
(ISSPACE(*p) && !fQuotedFilename) ||
|
||
|
(ISQUOTE(*p) && fQuotedFilename)))
|
||
|
p++;
|
||
|
}
|
||
|
|
||
|
// phase out the -a and -f options, but keep going to search the rest of the string
|
||
|
memmove((VOID *)pStart, (VOID *)p, strlen(p)+1);
|
||
|
p = pStart-1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// If a port has been specified, turn ':' into space, which will make the
|
||
|
// port become the second command line argument.
|
||
|
|
||
|
p = StrChr(pszCmdLine, ':');
|
||
|
|
||
|
if (p)
|
||
|
*p = ' ';
|
||
|
|
||
|
TRACE_OUT(("TelnetProtocolHandler(): Trying telnet to %s.",
|
||
|
pszCmdLine));
|
||
|
|
||
|
#ifndef UNIX
|
||
|
hr = MyExecute(s_cszTelnetApp, pszCmdLine, 0);
|
||
|
#else
|
||
|
// On UNIX, telnet is not GUI application, so we need to create
|
||
|
// separate xterm for it if we want to seet in in the separate window.
|
||
|
strcpy(szApp, "xterm");
|
||
|
wnsprintf(szArgs, sizeof(szArgs), "-e %s %s", s_cszTelnetApp, pszCmdLine);
|
||
|
hr = MyExecute(szApp, szArgs, 0);
|
||
|
#endif
|
||
|
|
||
|
switch (hr)
|
||
|
{
|
||
|
case S_OK:
|
||
|
break;
|
||
|
|
||
|
case E_FILE_NOT_FOUND:
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_TELNET_APP_NOT_FOUND),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult,
|
||
|
s_cszTelnetApp)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case E_OUTOFMEMORY:
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_OPEN_INTSHCUT_OUT_OF_MEMORY),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
ASSERT(hr == E_FAIL);
|
||
|
if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
||
|
MAKEINTRESOURCE(IDS_TELNET_EXEC_FAILED),
|
||
|
(MB_OK | MB_ICONEXCLAMATION), &nResult,
|
||
|
s_cszTelnetApp)) {
|
||
|
ASSERT(nResult == IDOK);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
DebugExitVOID(TelnetProtocolHandler);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
#ifdef UNIX
|
||
|
#define OE_NEWS_COMMAND_KEY TEXT("Software\\Clients\\News\\Outlook Express\\shell\\open\\command")
|
||
|
#define OE_URL_COMMAND_NAME TEXT("URLCommand")
|
||
|
#define IE_HOME_ENVIRONMENT TEXT("MWDEV")
|
||
|
|
||
|
HRESULT UnixLaunchOENews(PCSTR pszCmdLine)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
TCHAR *tszCommand = NULL;
|
||
|
TCHAR tszIEHome[MAX_PATH];
|
||
|
DWORD cchIEHome;
|
||
|
DWORD cchCommand;
|
||
|
DWORD dwDisposition;
|
||
|
TCHAR *pchPos;
|
||
|
BOOL bMailed;
|
||
|
STARTUPINFO stInfo;
|
||
|
HKEY hkey = NULL;
|
||
|
int i;
|
||
|
#ifndef ANSI_SHELL32_ON_UNIX
|
||
|
char szCmdLine[MAX_PATH];
|
||
|
|
||
|
memset(szCmdLine, 0, MAX_PATH);
|
||
|
|
||
|
SHUnicodeToAnsi((LPWSTR)pszCmdLine, szCmdLine, MAX_PATH);
|
||
|
pszCmdLine = szCmdLine;
|
||
|
#endif /* ANSI_SHELL32_ON_UNIX */
|
||
|
|
||
|
cchIEHome = GetEnvironmentVariable(IE_HOME_ENVIRONMENT, tszIEHome, MAX_PATH);
|
||
|
if (cchIEHome)
|
||
|
{
|
||
|
_tcscat(tszIEHome, TEXT("/bin"));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return E_FAIL;
|
||
|
}
|
||
|
|
||
|
hr = RegCreateKeyEx(HKEY_LOCAL_MACHINE, OE_NEWS_COMMAND_KEY, 0, NULL, 0, KEY_READ, NULL, &hkey, &dwDisposition);
|
||
|
if (hr != ERROR_SUCCESS)
|
||
|
{
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
|
||
|
hr = RegQueryValueEx(hkey, OE_URL_COMMAND_NAME, NULL, NULL, (LPBYTE)NULL, &cchCommand);
|
||
|
if (hr != ERROR_SUCCESS)
|
||
|
{
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
cchCommand += _tcslen(tszIEHome) + _tcslen(pszCmdLine) + 1;
|
||
|
tszCommand = (TCHAR*)malloc((cchCommand)*sizeof(TCHAR));
|
||
|
|
||
|
_tcscpy(tszCommand, tszIEHome);
|
||
|
_tcscat(tszCommand, TEXT("/"));
|
||
|
dwDisposition = _tcslen(tszCommand);
|
||
|
|
||
|
hr = RegQueryValueEx(hkey, OE_URL_COMMAND_NAME, NULL, NULL, (LPBYTE)(&tszCommand[dwDisposition]), &cchCommand);
|
||
|
if (hr != ERROR_SUCCESS)
|
||
|
{
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
|
||
|
_tcscat(tszCommand, pszCmdLine);
|
||
|
|
||
|
|
||
|
memset(&stInfo, 0, sizeof(stInfo));
|
||
|
stInfo.cb = sizeof(stInfo);
|
||
|
bMailed = CreateProcess(NULL, tszCommand, NULL, NULL, FALSE, DETACHED_PROCESS, NULL, NULL, &stInfo, NULL);
|
||
|
|
||
|
Cleanup:
|
||
|
if ( hkey != NULL )
|
||
|
RegCloseKey(hkey);
|
||
|
|
||
|
if (tszCommand)
|
||
|
free(tszCommand);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT UnixReadNews(HWND hWndParent, PCSTR pszCmdLine)
|
||
|
{
|
||
|
HRESULT hr = S_FALSE;
|
||
|
|
||
|
CHAR szCommand[MAX_PATH];
|
||
|
CHAR szExpandedCommand[MAX_PATH];
|
||
|
UINT nCommandSize;
|
||
|
int i;
|
||
|
HKEY hkey;
|
||
|
DWORD dw;
|
||
|
BOOL bMailed;
|
||
|
STARTUPINFOA stInfo;
|
||
|
|
||
|
DWORD dwType;
|
||
|
DWORD dwSize = sizeof(DWORD);
|
||
|
DWORD dwUseOENews;
|
||
|
|
||
|
hr = SHGetValue(IE_USE_OE_NEWS_HKEY, IE_USE_OE_NEWS_KEY, IE_USE_OE_NEWS_VALUE,
|
||
|
&dwType, (void*)&dwUseOENews, &dwSize);
|
||
|
if ((hr) && (dwType != REG_DWORD))
|
||
|
{
|
||
|
// The default value for mail is FALSE
|
||
|
dwUseOENews = FALSE;
|
||
|
}
|
||
|
if (dwUseOENews)
|
||
|
{
|
||
|
return UnixLaunchOENews(pszCmdLine);
|
||
|
}
|
||
|
|
||
|
hr = RegCreateKeyEx(HKEY_CURRENT_USER, REGSTR_PATH_NEWSCLIENTS, 0, NULL, 0, KEY_READ, NULL, &hkey, &dw);
|
||
|
if (hr != ERROR_SUCCESS)
|
||
|
goto Cleanup;
|
||
|
dw = MAX_PATH;
|
||
|
hr = RegQueryValueEx(hkey, REGSTR_PATH_CURRENT, NULL, NULL, (LPBYTE)szCommand, &dw);
|
||
|
if (hr != ERROR_SUCCESS)
|
||
|
{
|
||
|
RegCloseKey(hkey);
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
|
||
|
if (strlen(szCommand) == 0)
|
||
|
{
|
||
|
MessageBox(hWndParent,
|
||
|
TEXT("There is currently no default association for \"news:\" links in UNIX Internet Explorer. If you have a favorite news reader that supports news specifications on the command line (id and group name), you can define an association for \"news:\" links for this program in the View->Internet Options, Programs Tab.\n"),
|
||
|
TEXT("Microsoft Internet Explorer"),
|
||
|
MB_OK|MB_ICONSTOP);
|
||
|
return S_FALSE;
|
||
|
}
|
||
|
|
||
|
dw = ExpandEnvironmentStringsA(szCommand, szExpandedCommand, MAX_PATH);
|
||
|
if (!dw)
|
||
|
{
|
||
|
strcpy(szExpandedCommand, szCommand);
|
||
|
}
|
||
|
|
||
|
strcpy(szCommand, szExpandedCommand);
|
||
|
for (i = lstrlen(szCommand)-1; i >= 0; i--)
|
||
|
if (szCommand[i] == '/')
|
||
|
{
|
||
|
szCommand[i] = '\0';
|
||
|
break;
|
||
|
}
|
||
|
strcat(szCommand, " ");
|
||
|
strcat(szCommand, (pszCmdLine + strlen("news:")));
|
||
|
|
||
|
memset(&stInfo, 0, sizeof(stInfo));
|
||
|
stInfo.cb = sizeof(stInfo);
|
||
|
bMailed = CreateProcessA(szExpandedCommand, szCommand, NULL, NULL, FALSE, DETACHED_PROCESS, NULL, NULL, &stInfo, NULL);
|
||
|
|
||
|
Cleanup:
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#pragma warning(default:4100) /* "unreferenced formal parameter" warning */
|
||
|
|