mirror of
https://github.com/reactos/reactos.git
synced 2025-08-05 01:25:41 +00:00
[RSHELL]
* Implement skeleton shell dde parser. [SHELL32] * Copy over the skeleton from rshell, and fill in the handlers for ViewFolder and ExploreFolder, with rudimentary implementations. CORE-7552 svn path=/branches/shell-experiments/; revision=64395
This commit is contained in:
parent
42040abadc
commit
6aaa167a00
2 changed files with 544 additions and 78 deletions
|
@ -21,9 +21,21 @@
|
|||
#include "precomp.h"
|
||||
#include <ddeml.h>
|
||||
#include <strsafe.h>
|
||||
#include <shlwapi_undoc.h>
|
||||
|
||||
WINE_DEFAULT_DEBUG_CHANNEL(shelldde);
|
||||
|
||||
typedef DWORD(CALLBACK * pfnCommandHandler)(PWSTR strCommand, PWSTR strPath, LPITEMIDLIST pidl, INT unkS);
|
||||
|
||||
struct DDECommandHandler
|
||||
{
|
||||
WCHAR Command[32];
|
||||
pfnCommandHandler Handler;
|
||||
};
|
||||
|
||||
extern DDECommandHandler HandlerList [];
|
||||
extern const int HandlerListLength;
|
||||
|
||||
/* DDE Instance ID */
|
||||
static DWORD dwDDEInst;
|
||||
|
||||
|
@ -34,6 +46,8 @@ static HSZ hszShell;
|
|||
static HSZ hszAppProperties;
|
||||
static HSZ hszFolders;
|
||||
|
||||
static BOOL bInitialized;
|
||||
|
||||
static BOOL Dde_OnConnect(HSZ hszTopic, HSZ hszService)
|
||||
{
|
||||
WCHAR szTopic[MAX_PATH];
|
||||
|
@ -54,7 +68,7 @@ static void Dde_OnConnectConfirm(HCONV hconv, HSZ hszTopic, HSZ hszService)
|
|||
|
||||
DdeQueryStringW(dwDDEInst, hszTopic, szTopic, _countof(szTopic), CP_WINUNICODE);
|
||||
DdeQueryStringW(dwDDEInst, hszService, szService, _countof(szService), CP_WINUNICODE);
|
||||
|
||||
|
||||
DbgPrint("Dde_OnConnectConfirm: hconv=%p, topic=%S, service=%S\n", hconv, szTopic, szService);
|
||||
}
|
||||
|
||||
|
@ -84,6 +98,32 @@ static HDDEDATA Dde_OnRequest(UINT uFmt, HCONV hconv, HSZ hszTopic, HSZ hszItem)
|
|||
return NULL;
|
||||
}
|
||||
|
||||
static LPITEMIDLIST _ILReadFromSharedMemory(PCWSTR strField)
|
||||
{
|
||||
LPITEMIDLIST ret = NULL;
|
||||
|
||||
// Ensure it really is an IDLIST-formatted parameter
|
||||
// Format for IDLIST params: ":pid:shared"
|
||||
if (*strField != L':')
|
||||
return NULL;
|
||||
|
||||
HANDLE hData = (HANDLE) StrToIntW(strField + 1);
|
||||
PWSTR strSecond = StrChrW(strField + 1, L':');
|
||||
|
||||
if (strSecond)
|
||||
{
|
||||
int pid = StrToIntW(strSecond + 1);
|
||||
void* pvShared = SHLockShared(hData, pid);
|
||||
if (pvShared)
|
||||
{
|
||||
ret = ILClone((LPCITEMIDLIST) pvShared);
|
||||
SHUnlockShared(pvShared);
|
||||
SHFreeShared(hData, pid);
|
||||
}
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
static DWORD Dde_OnExecute(HCONV hconv, HSZ hszTopic, HDDEDATA hdata)
|
||||
{
|
||||
WCHAR szTopic[MAX_PATH];
|
||||
|
@ -92,7 +132,7 @@ static DWORD Dde_OnExecute(HCONV hconv, HSZ hszTopic, HDDEDATA hdata)
|
|||
|
||||
DdeQueryStringW(dwDDEInst, hszTopic, szTopic, _countof(szTopic), CP_WINUNICODE);
|
||||
|
||||
pszCommand = (WCHAR*)DdeAccessData(hdata, NULL);
|
||||
pszCommand = (WCHAR*) DdeAccessData(hdata, NULL);
|
||||
if (!pszCommand)
|
||||
return DDE_FNOTPROCESSED;
|
||||
|
||||
|
@ -103,15 +143,139 @@ static DWORD Dde_OnExecute(HCONV hconv, HSZ hszTopic, HDDEDATA hdata)
|
|||
DbgPrint("Dde_OnExecute: hconv=%p, topic=%S, command=%S\n", hconv, szTopic, pszCommand);
|
||||
|
||||
/*
|
||||
|
||||
[ViewFolder("%l", %I, %S)] -- Open a folder in standard mode
|
||||
[ExploreFolder("%l", %I, %S)] -- Open a folder in "explore" mode (file tree is shown to the left by default)
|
||||
[FindFolder("%l", %I)] -- Open a folder in "find" mode (search panel is shown to the left by default)
|
||||
[ShellFile("%1","%1",%S)] -- Execute the contents of the specified .SCF file
|
||||
|
||||
|
||||
Approximate grammar (Upper names define rules, <lower> names define terminals, single-quotes are literals):
|
||||
|
||||
Rules
|
||||
Command = ('[' Function ']') | Function
|
||||
Function = <identifier> '(' Parameters ')'
|
||||
Parameters = (<quoted-string> (',' <idlist> (',' <number>)?)?)?
|
||||
|
||||
Terminals
|
||||
<identifier> = [a-zA-Z]+
|
||||
<quoted-string> = \"([^\"]|\\.)\"
|
||||
<idlist> = \:[0-9]+\:[0-9]+
|
||||
<number> = [0-9]+
|
||||
*/
|
||||
|
||||
return DDE_FACK;
|
||||
WCHAR Command[MAX_PATH] = L"";
|
||||
WCHAR Path[MAX_PATH] = L"";
|
||||
LPITEMIDLIST IdList = NULL;
|
||||
INT UnknownParameter = 0;
|
||||
|
||||
// Simplified parsing (assumes the command will not be TOO broken):
|
||||
|
||||
PWSTR cmd = szCommand;
|
||||
// 1. if starts with [, skip first char
|
||||
if (*cmd == L'[')
|
||||
cmd++;
|
||||
|
||||
if (*cmd == L']')
|
||||
{
|
||||
ERR("Empty command. Nothing to run.\n");
|
||||
return DDE_FNOTPROCESSED;
|
||||
}
|
||||
|
||||
// Read until first (, and take text before ( as command name
|
||||
{
|
||||
PWSTR cmdEnd = StrChrW(cmd, L'(');
|
||||
|
||||
if (!cmdEnd)
|
||||
{
|
||||
ERR("Could not find '('. Invalid command.\n");
|
||||
return DDE_FNOTPROCESSED;
|
||||
}
|
||||
|
||||
*cmdEnd = 0;
|
||||
|
||||
StringCchCopy(Command, _countof(Command), cmd);
|
||||
|
||||
cmd = cmdEnd + 1;
|
||||
}
|
||||
|
||||
// Read first param after (, expecting quoted string
|
||||
if (*cmd != L')')
|
||||
{
|
||||
// Copy unescaped string
|
||||
PWSTR dst = Path;
|
||||
BOOL isQuote = FALSE;
|
||||
|
||||
PWSTR arg = cmd;
|
||||
|
||||
while (*arg && (isQuote || *arg != L','))
|
||||
{
|
||||
if (*arg == L'"')
|
||||
{
|
||||
isQuote = !isQuote;
|
||||
if (isQuote && arg != cmd) // do not copy the " at the beginning of the string
|
||||
{
|
||||
*(dst++) = L'"';
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
*(dst++) = *arg;
|
||||
}
|
||||
|
||||
arg++;
|
||||
}
|
||||
|
||||
cmd = arg + 1;
|
||||
|
||||
while (*cmd == L' ')
|
||||
cmd++;
|
||||
}
|
||||
|
||||
// Read second param, expecting an idlist in shared memory
|
||||
if (*cmd != L')')
|
||||
{
|
||||
if (*cmd != ':')
|
||||
{
|
||||
ERR("Expected ':'. Invalid command.\n");
|
||||
return DDE_FNOTPROCESSED;
|
||||
}
|
||||
|
||||
PWSTR idlistEnd = StrChrW(cmd, L',');
|
||||
|
||||
if (!idlistEnd)
|
||||
idlistEnd = StrChrW(cmd, L')');
|
||||
|
||||
if (!idlistEnd)
|
||||
{
|
||||
ERR("Expected ',' or ')'. Invalid command.\n");
|
||||
return DDE_FNOTPROCESSED;
|
||||
}
|
||||
|
||||
IdList = _ILReadFromSharedMemory(cmd);
|
||||
|
||||
cmd = idlistEnd + 1;
|
||||
}
|
||||
|
||||
// Read third param, expecting an integer
|
||||
if (*cmd != L')')
|
||||
{
|
||||
UnknownParameter = StrToIntW(cmd);
|
||||
}
|
||||
|
||||
DbgPrint("Parse end: cmd=%S, S=%d, pidl=%p, path=%S\n", Command, UnknownParameter, IdList, Path);
|
||||
|
||||
// Find handler in list
|
||||
for (int i = 0; i < HandlerListLength; i++)
|
||||
{
|
||||
DDECommandHandler & handler = HandlerList[i];
|
||||
if (StrCmpW(handler.Command, Command) == 0)
|
||||
{
|
||||
return handler.Handler(Command, Path, IdList, UnknownParameter);
|
||||
}
|
||||
}
|
||||
|
||||
// No handler found
|
||||
ERR("Unknown command %S\n", Command);
|
||||
return DDE_FNOTPROCESSED;
|
||||
}
|
||||
|
||||
static void Dde_OnDisconnect(HCONV hconv)
|
||||
|
@ -132,26 +296,26 @@ static HDDEDATA CALLBACK DdeCallback(
|
|||
switch (uType)
|
||||
{
|
||||
case XTYP_CONNECT:
|
||||
return (HDDEDATA)(DWORD_PTR)Dde_OnConnect(hsz1, hsz2);
|
||||
return (HDDEDATA) (DWORD_PTR) Dde_OnConnect(hsz1, hsz2);
|
||||
case XTYP_CONNECT_CONFIRM:
|
||||
Dde_OnConnectConfirm(hconv, hsz1, hsz2);
|
||||
return NULL;
|
||||
case XTYP_WILDCONNECT:
|
||||
return (HDDEDATA)(DWORD_PTR)Dde_OnWildConnect(hsz1, hsz2);
|
||||
return (HDDEDATA) (DWORD_PTR) Dde_OnWildConnect(hsz1, hsz2);
|
||||
case XTYP_REQUEST:
|
||||
return Dde_OnRequest(uFmt, hconv, hsz1, hsz2);
|
||||
case XTYP_EXECUTE:
|
||||
return (HDDEDATA)(DWORD_PTR)Dde_OnExecute(hconv, hsz1, hdata);
|
||||
return (HDDEDATA) (DWORD_PTR) Dde_OnExecute(hconv, hsz1, hdata);
|
||||
case XTYP_DISCONNECT:
|
||||
Dde_OnDisconnect(hconv);
|
||||
return NULL;
|
||||
case XTYP_REGISTER:
|
||||
return NULL;
|
||||
default:
|
||||
DbgPrint("DdeCallback: unknown uType=%d\n", uType);
|
||||
case XTYP_REGISTER:
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
|
||||
/*************************************************************************
|
||||
* ShellDDEInit (SHELL32.@)
|
||||
*
|
||||
|
@ -168,7 +332,7 @@ EXTERN_C void WINAPI ShellDDEInit(BOOL bInit)
|
|||
{
|
||||
DbgPrint("ShellDDEInit bInit = %s\n", bInit ? "TRUE" : "FALSE");
|
||||
|
||||
if (bInit)
|
||||
if (bInit && !bInitialized)
|
||||
{
|
||||
DdeInitializeW(&dwDDEInst, DdeCallback, CBF_FAIL_ADVISES | CBF_FAIL_POKES, 0);
|
||||
|
||||
|
@ -178,19 +342,67 @@ EXTERN_C void WINAPI ShellDDEInit(BOOL bInit)
|
|||
hszAppProperties = DdeCreateStringHandleW(dwDDEInst, L"AppProperties", CP_WINUNICODE);
|
||||
hszFolders = DdeCreateStringHandleW(dwDDEInst, L"Folders", CP_WINUNICODE);
|
||||
|
||||
DdeNameService(dwDDEInst, hszFolders, 0, DNS_FILTEROFF);
|
||||
if (hszProgmanTopic && hszProgmanService &&
|
||||
hszShell && hszAppProperties && hszFolders &&
|
||||
DdeNameService(dwDDEInst, hszFolders, 0, DNS_REGISTER) &&
|
||||
DdeNameService(dwDDEInst, hszProgmanService, 0, DNS_REGISTER) &&
|
||||
DdeNameService(dwDDEInst, hszShell, 0, DNS_REGISTER))
|
||||
{
|
||||
bInitialized = TRUE;
|
||||
}
|
||||
}
|
||||
else
|
||||
else if (!bInit && bInitialized)
|
||||
{
|
||||
/* unregister all services */
|
||||
DdeNameService(dwDDEInst, 0, 0, DNS_UNREGISTER);
|
||||
|
||||
DdeFreeStringHandle(dwDDEInst, hszFolders);
|
||||
DdeFreeStringHandle(dwDDEInst, hszAppProperties);
|
||||
DdeFreeStringHandle(dwDDEInst, hszShell);
|
||||
DdeFreeStringHandle(dwDDEInst, hszProgmanService);
|
||||
DdeFreeStringHandle(dwDDEInst, hszProgmanTopic);
|
||||
if (hszFolders)
|
||||
DdeFreeStringHandle(dwDDEInst, hszFolders);
|
||||
if (hszAppProperties)
|
||||
DdeFreeStringHandle(dwDDEInst, hszAppProperties);
|
||||
if (hszShell)
|
||||
DdeFreeStringHandle(dwDDEInst, hszShell);
|
||||
if (hszProgmanService)
|
||||
DdeFreeStringHandle(dwDDEInst, hszProgmanService);
|
||||
if (hszProgmanTopic)
|
||||
DdeFreeStringHandle(dwDDEInst, hszProgmanTopic);
|
||||
|
||||
DdeUninitialize(dwDDEInst);
|
||||
|
||||
bInitialized = FALSE;
|
||||
}
|
||||
}
|
||||
|
||||
static DWORD CALLBACK DDE_OnViewFolder(PWSTR strCommand, PWSTR strPath, LPITEMIDLIST pidl, INT unkS)
|
||||
{
|
||||
UNIMPLEMENTED;
|
||||
return DDE_FACK;
|
||||
}
|
||||
|
||||
static DWORD CALLBACK DDW_OnExploreFolder(PWSTR strCommand, PWSTR strPath, LPITEMIDLIST pidl, INT unkS)
|
||||
{
|
||||
UNIMPLEMENTED;
|
||||
return DDE_FACK;
|
||||
}
|
||||
|
||||
static DWORD CALLBACK DDE_OnFindFolder(PWSTR strCommand, PWSTR strPath, LPITEMIDLIST pidl, INT unkS)
|
||||
{
|
||||
UNIMPLEMENTED;
|
||||
return DDE_FNOTPROCESSED;
|
||||
}
|
||||
|
||||
static DWORD CALLBACK DDE_OnShellFile(PWSTR strCommand, PWSTR strPath, LPITEMIDLIST pidl, INT unkS)
|
||||
{
|
||||
UNIMPLEMENTED;
|
||||
return DDE_FNOTPROCESSED;
|
||||
}
|
||||
|
||||
DDECommandHandler HandlerList [] = {
|
||||
|
||||
{ L"ViewFolder", DDE_OnViewFolder },
|
||||
{ L"ExploreFolder", DDW_OnExploreFolder },
|
||||
{ L"FindFolder", DDE_OnFindFolder },
|
||||
{ L"ShellFile", DDE_OnShellFile }
|
||||
};
|
||||
|
||||
const int HandlerListLength = _countof(HandlerList);
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue