wine/programs/explorer/explorer.c
Zhiyi Zhang c03409e9d7 shell32: Implement SHOpenFolderAndSelectItems().
Wine-Bug: https://bugs.winehq.org/show_bug.cgi?id=39987
Signed-off-by: Zhiyi Zhang <zzhang@codeweavers.com>
2022-07-15 20:01:05 +02:00

935 lines
30 KiB
C

/*
* explorer.exe
*
* Copyright 2004 CodeWeavers, Mike Hearn
* Copyright 2005,2006 CodeWeavers, Aric Stewart
* Copyright 2011 Jay Yang
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
*/
#define COBJMACROS
#include "wine/debug.h"
#include "wine/heap.h"
#include "explorer_private.h"
#include "resource.h"
#include <initguid.h>
#include <windows.h>
#include <shellapi.h>
#include <shobjidl.h>
#include <shlobj.h>
#include <shlwapi.h>
#include <commoncontrols.h>
#include <commctrl.h>
WINE_DEFAULT_DEBUG_CHANNEL(explorer);
#define EXPLORER_INFO_INDEX 0
#define NAV_TOOLBAR_HEIGHT 30
#define PATHBOX_HEIGHT 24
static int nav_toolbar_height;
static int pathbox_height;
#define DEFAULT_WIDTH 640
#define DEFAULT_HEIGHT 480
static int default_width;
static int default_height;
static const WCHAR EXPLORER_CLASS[] = {'E','x','p','l','o','r','e','r','W','C','l','a','s','s',0};
static const WCHAR PATH_BOX_NAME[] = {'\0'};
HINSTANCE explorer_hInstance;
typedef struct parametersTAG {
BOOL explorer_mode;
WCHAR root[MAX_PATH];
WCHAR selection[MAX_PATH];
} parameters_struct;
typedef struct
{
IExplorerBrowser *browser;
HWND main_window,path_box;
INT rebar_height;
LPITEMIDLIST pidl;
IImageList *icon_list;
DWORD advise_cookie;
IShellWindows *sw;
LONG sw_cookie;
} explorer_info;
enum
{
BACK_BUTTON,FORWARD_BUTTON,UP_BUTTON
};
static void variant_from_pidl(VARIANT *var, const ITEMIDLIST *pidl)
{
V_VT(var) = VT_ARRAY | VT_UI1;
V_ARRAY(var) = SafeArrayCreateVector(VT_UI1, 0, ILGetSize(pidl));
memcpy(V_ARRAY(var)->pvData, pidl, ILGetSize(pidl));
}
typedef struct
{
IExplorerBrowserEvents IExplorerBrowserEvents_iface;
explorer_info* info;
LONG ref;
} IExplorerBrowserEventsImpl;
static IExplorerBrowserEventsImpl *impl_from_IExplorerBrowserEvents(IExplorerBrowserEvents *iface)
{
return CONTAINING_RECORD(iface, IExplorerBrowserEventsImpl, IExplorerBrowserEvents_iface);
}
static HRESULT WINAPI IExplorerBrowserEventsImpl_fnQueryInterface(IExplorerBrowserEvents *iface, REFIID riid, void **ppvObject)
{
return E_NOINTERFACE;
}
static ULONG WINAPI IExplorerBrowserEventsImpl_fnAddRef(IExplorerBrowserEvents *iface)
{
IExplorerBrowserEventsImpl *This = impl_from_IExplorerBrowserEvents(iface);
return InterlockedIncrement(&This->ref);
}
static ULONG WINAPI IExplorerBrowserEventsImpl_fnRelease(IExplorerBrowserEvents *iface)
{
IExplorerBrowserEventsImpl *This = impl_from_IExplorerBrowserEvents(iface);
ULONG ref = InterlockedDecrement(&This->ref);
if(!ref)
HeapFree(GetProcessHeap(),0,This);
return ref;
}
static BOOL create_combobox_item(IShellFolder *folder, LPCITEMIDLIST child_pidl, IImageList *icon_list, COMBOBOXEXITEMW *item)
{
STRRET strret;
HRESULT hres;
PIDLIST_ABSOLUTE parent_pidl, pidl;
SHFILEINFOW info;
IImageList *list;
strret.uType=STRRET_WSTR;
hres = IShellFolder_GetDisplayNameOf( folder, child_pidl, SHGDN_FORADDRESSBAR, &strret );
if(SUCCEEDED(hres))
hres = StrRetToStrW(&strret, child_pidl, &item->pszText);
if(FAILED(hres))
{
WINE_WARN("Could not get name for pidl\n");
return FALSE;
}
item->mask &= ~CBEIF_IMAGE;
hres = SHGetIDListFromObject( (IUnknown *)folder, &parent_pidl );
if (FAILED(hres)) return FALSE;
pidl = ILCombine( parent_pidl, child_pidl );
if (pidl)
{
list = (IImageList *)SHGetFileInfoW( (WCHAR *)pidl, 0, &info, sizeof(info),
SHGFI_PIDL | SHGFI_SMALLICON | SHGFI_SYSICONINDEX );
if (list)
{
IImageList_Release( list );
item->iImage = info.iIcon;
item->mask |= CBEIF_IMAGE;
}
ILFree( pidl );
}
ILFree( parent_pidl );
return TRUE;
}
static void update_path_box(explorer_info *info)
{
COMBOBOXEXITEMW item;
COMBOBOXEXITEMW main_item;
IShellFolder *desktop;
IPersistFolder2 *persist;
LPITEMIDLIST desktop_pidl;
IEnumIDList *ids;
SendMessageW(info->path_box,CB_RESETCONTENT,0,0);
SHGetDesktopFolder(&desktop);
IShellFolder_QueryInterface(desktop,&IID_IPersistFolder2,(void**)&persist);
IPersistFolder2_GetCurFolder(persist,&desktop_pidl);
IPersistFolder2_Release(persist);
persist = NULL;
/*Add Desktop*/
item.iItem = -1;
item.mask = CBEIF_TEXT | CBEIF_INDENT | CBEIF_LPARAM;
item.iIndent = 0;
create_combobox_item(desktop,desktop_pidl,info->icon_list,&item);
item.lParam = (LPARAM)desktop_pidl;
SendMessageW(info->path_box,CBEM_INSERTITEMW,0,(LPARAM)&item);
if(ILIsEqual(info->pidl,desktop_pidl))
main_item = item;
else
CoTaskMemFree(item.pszText);
/*Add all direct subfolders of Desktop*/
if(SUCCEEDED(IShellFolder_EnumObjects(desktop,NULL,SHCONTF_FOLDERS,&ids))
&& ids!=NULL)
{
LPITEMIDLIST curr_pidl=NULL;
HRESULT hres;
item.iIndent = 1;
while(1)
{
ILFree(curr_pidl);
curr_pidl=NULL;
hres = IEnumIDList_Next(ids,1,&curr_pidl,NULL);
if(FAILED(hres) || hres == S_FALSE)
break;
if(!create_combobox_item(desktop,curr_pidl,info->icon_list,&item))
WINE_WARN("Could not create a combobox item\n");
else
{
LPITEMIDLIST full_pidl = ILCombine(desktop_pidl,curr_pidl);
item.lParam = (LPARAM)full_pidl;
SendMessageW(info->path_box,CBEM_INSERTITEMW,0,(LPARAM)&item);
if(ILIsEqual(full_pidl,info->pidl))
main_item = item;
else if(ILIsParent(full_pidl,info->pidl,FALSE))
{
/*add all parents of the pidl passed in*/
LPITEMIDLIST next_pidl = ILFindChild(full_pidl,info->pidl);
IShellFolder *curr_folder = NULL, *temp;
hres = IShellFolder_BindToObject(desktop,curr_pidl,NULL,
&IID_IShellFolder,
(void**)&curr_folder);
if(FAILED(hres))
WINE_WARN("Could not get an IShellFolder\n");
while(!ILIsEmpty(next_pidl))
{
LPITEMIDLIST first = ILCloneFirst(next_pidl);
CoTaskMemFree(item.pszText);
if(!create_combobox_item(curr_folder,first,
info->icon_list,&item))
{
WINE_WARN("Could not create a combobox item\n");
break;
}
++item.iIndent;
full_pidl = ILCombine(full_pidl,first);
item.lParam = (LPARAM)full_pidl;
SendMessageW(info->path_box,CBEM_INSERTITEMW,0,(LPARAM)&item);
temp=NULL;
hres = IShellFolder_BindToObject(curr_folder,first,NULL,
&IID_IShellFolder,
(void**)&temp);
if(FAILED(hres))
{
WINE_WARN("Could not get an IShellFolder\n");
break;
}
IShellFolder_Release(curr_folder);
curr_folder = temp;
ILFree(first);
next_pidl = ILGetNext(next_pidl);
}
memcpy(&main_item,&item,sizeof(item));
if(curr_folder)
IShellFolder_Release(curr_folder);
item.iIndent = 1;
}
else
CoTaskMemFree(item.pszText);
}
}
ILFree(curr_pidl);
IEnumIDList_Release(ids);
}
else
WINE_WARN("Could not enumerate the desktop\n");
SendMessageW(info->path_box,CBEM_SETITEMW,0,(LPARAM)&main_item);
CoTaskMemFree(main_item.pszText);
}
static HRESULT WINAPI IExplorerBrowserEventsImpl_fnOnNavigationComplete(IExplorerBrowserEvents *iface, PCIDLIST_ABSOLUTE pidl)
{
IExplorerBrowserEventsImpl *This = impl_from_IExplorerBrowserEvents(iface);
IShellFolder *parent;
PCUITEMID_CHILD child_pidl;
HRESULT hres;
STRRET strret;
WCHAR *name;
if (This->info->sw)
{
VARIANT var;
variant_from_pidl(&var, pidl);
IShellWindows_OnNavigate(This->info->sw, This->info->sw_cookie, &var);
VariantClear(&var);
}
ILFree(This->info->pidl);
This->info->pidl = ILClone(pidl);
update_path_box(This->info);
hres = SHBindToParent(pidl, &IID_IShellFolder, (void **)&parent, &child_pidl);
if (SUCCEEDED(hres))
{
hres = IShellFolder_GetDisplayNameOf(parent, child_pidl, SHGDN_FORADDRESSBAR, &strret);
if (SUCCEEDED(hres))
hres = StrRetToStrW(&strret, child_pidl, &name);
if (SUCCEEDED(hres))
{
SetWindowTextW(This->info->main_window, name);
CoTaskMemFree(name);
}
IShellFolder_Release(parent);
}
return hres;
}
static HRESULT WINAPI IExplorerBrowserEventsImpl_fnOnNavigationFailed(IExplorerBrowserEvents *iface, PCIDLIST_ABSOLUTE pidl)
{
return S_OK;
}
static HRESULT WINAPI IExplorerBrowserEventsImpl_fnOnNavigationPending(IExplorerBrowserEvents *iface, PCIDLIST_ABSOLUTE pidl)
{
return S_OK;
}
static HRESULT WINAPI IExplorerBrowserEventsImpl_fnOnViewCreated(IExplorerBrowserEvents *iface, IShellView *psv)
{
return S_OK;
}
static IExplorerBrowserEventsVtbl vt_IExplorerBrowserEvents =
{
IExplorerBrowserEventsImpl_fnQueryInterface,
IExplorerBrowserEventsImpl_fnAddRef,
IExplorerBrowserEventsImpl_fnRelease,
IExplorerBrowserEventsImpl_fnOnNavigationPending,
IExplorerBrowserEventsImpl_fnOnViewCreated,
IExplorerBrowserEventsImpl_fnOnNavigationComplete,
IExplorerBrowserEventsImpl_fnOnNavigationFailed
};
static IExplorerBrowserEvents *make_explorer_events(explorer_info *info)
{
IExplorerBrowserEventsImpl *ret
= HeapAlloc(GetProcessHeap(), 0, sizeof(IExplorerBrowserEventsImpl));
ret->IExplorerBrowserEvents_iface.lpVtbl = &vt_IExplorerBrowserEvents;
ret->info = info;
ret->ref = 1;
SHGetImageList(SHIL_SMALL,&IID_IImageList,(void**)&(ret->info->icon_list));
SendMessageW(info->path_box,CBEM_SETIMAGELIST,0,(LPARAM)ret->info->icon_list);
return &ret->IExplorerBrowserEvents_iface;
}
static IShellFolder *get_starting_shell_folder(WCHAR *path)
{
IShellFolder* desktop,*folder;
LPITEMIDLIST root_pidl;
HRESULT hres;
SHGetDesktopFolder(&desktop);
if (!path)
return desktop;
hres = IShellFolder_ParseDisplayName(desktop, NULL, NULL, path, NULL, &root_pidl, NULL);
if(FAILED(hres))
{
return desktop;
}
hres = IShellFolder_BindToObject(desktop,root_pidl,NULL,
&IID_IShellFolder,
(void**)&folder);
ILFree(root_pidl);
if(FAILED(hres))
{
return desktop;
}
IShellFolder_Release(desktop);
return folder;
}
static void make_explorer_window(parameters_struct *params)
{
RECT rect;
HWND rebar,nav_toolbar;
FOLDERSETTINGS fs;
IExplorerBrowserEvents *events;
explorer_info *info;
HRESULT hres;
WCHAR explorer_title[100];
WCHAR pathbox_label[50];
TBADDBITMAP bitmap_info;
TBBUTTON nav_buttons[3];
int hist_offset,view_offset;
REBARBANDINFOW band_info;
VARIANT var, empty_var;
IShellFolder *folder;
IDispatch *dispatch;
WCHAR *path = NULL;
IShellWindows *sw;
ITEMIDLIST *pidl;
UINT dpix, dpiy;
DWORD size;
LONG hwnd;
HDC hdc;
MSG msg;
CoCreateInstance(&CLSID_ShellWindows, NULL, CLSCTX_LOCAL_SERVER,
&IID_IShellWindows, (void **)&sw);
if (params->root[0])
{
size = GetFullPathNameW(params->root, 0, NULL, NULL);
path = malloc( size * sizeof(WCHAR) );
GetFullPathNameW(params->root, size, path, NULL);
}
if (sw && path)
{
if (!(pidl = ILCreateFromPathW(path)))
{
ERR("Failed to create PIDL for %s.\n", debugstr_w(path));
IShellWindows_Release(sw);
free(path);
return;
}
variant_from_pidl(&var, pidl);
V_VT(&empty_var) = VT_EMPTY;
hres = IShellWindows_FindWindowSW(sw, &var, &empty_var, SWC_EXPLORER, &hwnd, 0, &dispatch);
VariantClear(&var);
ILFree(pidl);
if (hres == S_OK)
{
TRACE("Found window %#lx already browsing path %s.\n", hwnd, debugstr_w(path));
SetForegroundWindow((HWND)(LONG_PTR)hwnd);
IShellWindows_Release(sw);
free(path);
return;
}
}
memset(nav_buttons,0,sizeof(nav_buttons));
LoadStringW(explorer_hInstance,IDS_EXPLORER_TITLE,explorer_title, ARRAY_SIZE( explorer_title ));
LoadStringW(explorer_hInstance,IDS_PATHBOX_LABEL,pathbox_label, ARRAY_SIZE( pathbox_label ));
hdc = GetDC(0);
dpix = GetDeviceCaps(hdc, LOGPIXELSX);
dpiy = GetDeviceCaps(hdc, LOGPIXELSY);
ReleaseDC(0, hdc);
nav_toolbar_height = MulDiv(NAV_TOOLBAR_HEIGHT, dpiy, USER_DEFAULT_SCREEN_DPI);
pathbox_height = MulDiv(PATHBOX_HEIGHT, dpiy, USER_DEFAULT_SCREEN_DPI);
default_width = MulDiv(DEFAULT_WIDTH, dpix, USER_DEFAULT_SCREEN_DPI);
default_height = MulDiv(DEFAULT_HEIGHT, dpiy, USER_DEFAULT_SCREEN_DPI);
info = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(explorer_info));
if(!info)
{
WINE_ERR("Could not allocate an explorer_info struct\n");
IShellWindows_Release(sw);
free(path);
return;
}
hres = CoCreateInstance(&CLSID_ExplorerBrowser,NULL,CLSCTX_INPROC_SERVER,
&IID_IExplorerBrowser,(LPVOID*)&info->browser);
if(FAILED(hres))
{
WINE_ERR("Could not obtain an instance of IExplorerBrowser\n");
HeapFree(GetProcessHeap(),0,info);
IShellWindows_Release(sw);
free(path);
return;
}
info->rebar_height=0;
info->main_window
= CreateWindowW(EXPLORER_CLASS,explorer_title,WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT,CW_USEDEFAULT,default_width,
default_height,NULL,NULL,explorer_hInstance,NULL);
if (sw)
{
IShellWindows_Register(sw, NULL, (LONG_PTR)info->main_window, SWC_EXPLORER, &info->sw_cookie);
info->sw = sw;
}
fs.ViewMode = FVM_DETAILS;
fs.fFlags = FWF_AUTOARRANGE;
SetRect(&rect, 0, 0, default_width, default_height);
IExplorerBrowser_Initialize(info->browser,info->main_window,&rect,&fs);
IExplorerBrowser_SetOptions(info->browser,EBO_SHOWFRAMES);
SetWindowLongPtrW(info->main_window,EXPLORER_INFO_INDEX,(LONG_PTR)info);
/*setup navbar*/
rebar = CreateWindowExW(WS_EX_TOOLWINDOW,REBARCLASSNAMEW,NULL,
WS_CHILD|WS_VISIBLE|RBS_VARHEIGHT|CCS_TOP|CCS_NODIVIDER,
0,0,0,0,info->main_window,NULL,explorer_hInstance,NULL);
nav_toolbar
= CreateWindowExW(TBSTYLE_EX_MIXEDBUTTONS,TOOLBARCLASSNAMEW,NULL,
WS_CHILD|WS_VISIBLE|TBSTYLE_FLAT,0,0,0,0,rebar,NULL,
explorer_hInstance,NULL);
bitmap_info.hInst = HINST_COMMCTRL;
bitmap_info.nID = IDB_HIST_LARGE_COLOR;
hist_offset= SendMessageW(nav_toolbar,TB_ADDBITMAP,0,(LPARAM)&bitmap_info);
bitmap_info.nID = IDB_VIEW_LARGE_COLOR;
view_offset= SendMessageW(nav_toolbar,TB_ADDBITMAP,0,(LPARAM)&bitmap_info);
nav_buttons[0].iBitmap=hist_offset+HIST_BACK;
nav_buttons[0].idCommand=BACK_BUTTON;
nav_buttons[0].fsState=TBSTATE_ENABLED;
nav_buttons[0].fsStyle=BTNS_BUTTON|BTNS_AUTOSIZE;
nav_buttons[1].iBitmap=hist_offset+HIST_FORWARD;
nav_buttons[1].idCommand=FORWARD_BUTTON;
nav_buttons[1].fsState=TBSTATE_ENABLED;
nav_buttons[1].fsStyle=BTNS_BUTTON|BTNS_AUTOSIZE;
nav_buttons[2].iBitmap=view_offset+VIEW_PARENTFOLDER;
nav_buttons[2].idCommand=UP_BUTTON;
nav_buttons[2].fsState=TBSTATE_ENABLED;
nav_buttons[2].fsStyle=BTNS_BUTTON|BTNS_AUTOSIZE;
SendMessageW(nav_toolbar,TB_BUTTONSTRUCTSIZE,sizeof(TBBUTTON),0);
SendMessageW(nav_toolbar,TB_ADDBUTTONSW,ARRAY_SIZE( nav_buttons ),(LPARAM)nav_buttons);
band_info.cbSize = sizeof(band_info);
band_info.fMask = RBBIM_STYLE|RBBIM_CHILD|RBBIM_CHILDSIZE|RBBIM_SIZE;
band_info.hwndChild = nav_toolbar;
band_info.fStyle=RBBS_GRIPPERALWAYS|RBBS_CHILDEDGE;
band_info.cyChild=nav_toolbar_height;
band_info.cx=0;
band_info.cyMinChild=nav_toolbar_height;
band_info.cxMinChild=0;
SendMessageW(rebar,RB_INSERTBANDW,-1,(LPARAM)&band_info);
info->path_box = CreateWindowW(WC_COMBOBOXEXW,PATH_BOX_NAME,
WS_CHILD | WS_VISIBLE | CBS_DROPDOWN,
0,0,default_width,pathbox_height,rebar,NULL,
explorer_hInstance,NULL);
GetWindowRect(info->path_box, &rect);
band_info.cyChild = rect.bottom - rect.top;
band_info.cx=0;
band_info.cyMinChild = rect.bottom - rect.top;
band_info.cxMinChild=0;
band_info.fMask|=RBBIM_TEXT;
band_info.lpText=pathbox_label;
band_info.fStyle|=RBBS_BREAK;
band_info.hwndChild=info->path_box;
SendMessageW(rebar,RB_INSERTBANDW,-1,(LPARAM)&band_info);
events = make_explorer_events(info);
IExplorerBrowser_Advise(info->browser,events,&info->advise_cookie);
folder = get_starting_shell_folder(path);
IExplorerBrowser_BrowseToObject(info->browser, (IUnknown *)folder, SBSP_ABSOLUTE);
IShellFolder_Release(folder);
free(path);
ShowWindow(info->main_window,SW_SHOWDEFAULT);
UpdateWindow(info->main_window);
IExplorerBrowserEvents_Release(events);
while (GetMessageW(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessageW(&msg);
}
}
static void update_window_size(explorer_info *info, int height, int width)
{
RECT new_rect;
new_rect.left = 0;
new_rect.top = info->rebar_height;
new_rect.right = width;
new_rect.bottom = height;
IExplorerBrowser_SetRect(info->browser,NULL,new_rect);
}
static void do_exit(int code)
{
OleUninitialize();
ExitProcess(code);
}
static LRESULT explorer_on_end_edit(explorer_info *info,NMCBEENDEDITW *edit_info)
{
LPITEMIDLIST pidl = NULL;
WINE_TRACE("iWhy=%x\n",edit_info->iWhy);
switch(edit_info->iWhy)
{
case CBENF_DROPDOWN:
if(edit_info->iNewSelection!=CB_ERR)
pidl = (LPITEMIDLIST)SendMessageW(edit_info->hdr.hwndFrom,
CB_GETITEMDATA,
edit_info->iNewSelection,0);
break;
case CBENF_RETURN:
{
WCHAR path[MAX_PATH];
HWND edit_ctrl = (HWND)SendMessageW(edit_info->hdr.hwndFrom,
CBEM_GETEDITCONTROL,0,0);
*((WORD*)path)=MAX_PATH;
SendMessageW(edit_ctrl,EM_GETLINE,0,(LPARAM)path);
pidl = ILCreateFromPathW(path);
break;
}
case CBENF_ESCAPE:
/*make sure the that the path box resets*/
update_path_box(info);
return 0;
default:
return 0;
}
if(pidl)
IExplorerBrowser_BrowseToIDList(info->browser,pidl,SBSP_ABSOLUTE);
if(edit_info->iWhy==CBENF_RETURN)
ILFree(pidl);
return 0;
}
static LRESULT update_rebar_size(explorer_info* info,NMRBAUTOSIZE *size_info)
{
RECT new_rect;
RECT window_rect;
info->rebar_height = size_info->rcTarget.bottom-size_info->rcTarget.top;
GetWindowRect(info->main_window,&window_rect);
new_rect.left = 0;
new_rect.top = info->rebar_height;
new_rect.right = window_rect.right-window_rect.left;
new_rect.bottom = window_rect.bottom-window_rect.top;
IExplorerBrowser_SetRect(info->browser,NULL,new_rect);
return 0;
}
static LRESULT explorer_on_notify(explorer_info* info,NMHDR* notification)
{
WINE_TRACE("code=%i\n",notification->code);
switch(notification->code)
{
case CBEN_BEGINEDIT:
{
WCHAR path[MAX_PATH];
HWND edit_ctrl = (HWND)SendMessageW(notification->hwndFrom,
CBEM_GETEDITCONTROL,0,0);
SHGetPathFromIDListW(info->pidl,path);
SetWindowTextW(edit_ctrl,path);
break;
}
case CBEN_ENDEDITA:
{
NMCBEENDEDITA *edit_info_a = (NMCBEENDEDITA*)notification;
NMCBEENDEDITW edit_info_w;
edit_info_w.hdr = edit_info_a->hdr;
edit_info_w.fChanged = edit_info_a->fChanged;
edit_info_w.iNewSelection = edit_info_a->iNewSelection;
MultiByteToWideChar(CP_ACP,0,edit_info_a->szText,-1,
edit_info_w.szText,CBEMAXSTRLEN);
edit_info_w.iWhy = edit_info_a->iWhy;
return explorer_on_end_edit(info,&edit_info_w);
}
case CBEN_ENDEDITW:
return explorer_on_end_edit(info,(NMCBEENDEDITW*)notification);
case CBEN_DELETEITEM:
{
NMCOMBOBOXEXW *entry = (NMCOMBOBOXEXW*)notification;
if(entry->ceItem.lParam)
ILFree((LPITEMIDLIST)entry->ceItem.lParam);
break;
}
case RBN_AUTOSIZE:
return update_rebar_size(info,(NMRBAUTOSIZE*)notification);
default:
break;
}
return 0;
}
static BOOL handle_copydata(const explorer_info *info, const COPYDATASTRUCT *cds)
{
static const unsigned int magic = 0xe32ee32e;
unsigned int i, flags, count;
const ITEMIDLIST *child;
unsigned char *ptr;
IShellView *sv;
SVSIF sv_flags;
TRACE("\n");
/* For SHOpenFolderAndSelectItems() */
if (cds->dwData != magic)
return FALSE;
ptr = cds->lpData;
memcpy(&count, ptr, sizeof(count));
ptr += sizeof(count);
memcpy(&flags, ptr, sizeof(flags));
ptr += sizeof(flags);
sv_flags = flags & OFASI_EDIT ? SVSI_EDIT : SVSI_SELECT;
IExplorerBrowser_GetCurrentView(info->browser, &IID_IShellView, (void **)&sv);
for (i = 0; i < count; ++i)
{
child = (const ITEMIDLIST *)ptr;
if (i == 0)
IShellView_SelectItem(sv, child, sv_flags | SVSI_ENSUREVISIBLE | SVSI_FOCUSED | SVSI_DESELECTOTHERS);
else
IShellView_SelectItem(sv, child, sv_flags);
ptr += ILGetSize(child);
}
IShellView_Release(sv);
return TRUE;
}
static LRESULT CALLBACK explorer_wnd_proc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
explorer_info *info
= (explorer_info*)GetWindowLongPtrW(hwnd,EXPLORER_INFO_INDEX);
IExplorerBrowser *browser = NULL;
WINE_TRACE("(hwnd=%p,uMsg=%u,wParam=%Ix,lParam=%Ix)\n",hwnd,uMsg,wParam,lParam);
if(info)
browser = info->browser;
switch(uMsg)
{
case WM_DESTROY:
if(info->sw)
{
IShellWindows_Revoke(info->sw, info->sw_cookie);
IShellWindows_Release(info->sw);
}
IExplorerBrowser_Unadvise(browser,info->advise_cookie);
IExplorerBrowser_Destroy(browser);
IExplorerBrowser_Release(browser);
ILFree(info->pidl);
IImageList_Release(info->icon_list);
HeapFree(GetProcessHeap(),0,info);
SetWindowLongPtrW(hwnd,EXPLORER_INFO_INDEX,0);
PostQuitMessage(0);
break;
case WM_QUIT:
do_exit(wParam);
case WM_NOTIFY:
return explorer_on_notify(info,(NMHDR*)lParam);
case WM_COMMAND:
if(HIWORD(wParam)==BN_CLICKED)
{
switch(LOWORD(wParam))
{
case BACK_BUTTON:
IExplorerBrowser_BrowseToObject(browser,NULL,SBSP_NAVIGATEBACK);
break;
case FORWARD_BUTTON:
IExplorerBrowser_BrowseToObject(browser,NULL,SBSP_NAVIGATEFORWARD);
break;
case UP_BUTTON:
IExplorerBrowser_BrowseToObject(browser,NULL,SBSP_PARENT);
break;
}
}
break;
case WM_SIZE:
update_window_size(info,HIWORD(lParam),LOWORD(lParam));
break;
case WM_COPYDATA:
return handle_copydata(info, (const COPYDATASTRUCT *)lParam);
default:
return DefWindowProcW(hwnd,uMsg,wParam,lParam);
}
return 0;
}
static void register_explorer_window_class(void)
{
WNDCLASSEXW window_class;
window_class.cbSize = sizeof(WNDCLASSEXW);
window_class.style = 0;
window_class.cbClsExtra = 0;
window_class.cbWndExtra = sizeof(LONG_PTR);
window_class.lpfnWndProc = explorer_wnd_proc;
window_class.hInstance = explorer_hInstance;
window_class.hIcon = NULL;
window_class.hCursor = NULL;
window_class.hbrBackground = (HBRUSH)COLOR_BACKGROUND;
window_class.lpszMenuName = NULL;
window_class.lpszClassName = EXPLORER_CLASS;
window_class.hIconSm = NULL;
RegisterClassExW(&window_class);
}
static WCHAR *copy_path_string(WCHAR *target, WCHAR *source)
{
INT i = 0;
while (iswspace(*source)) source++;
if (*source == '\"')
{
source ++;
while (*source && *source != '\"') target[i++] = *source++;
target[i] = 0;
if (*source) source++;
}
else
{
while (*source && *source != ',') target[i++] = *source++;
target[i] = 0;
}
PathRemoveBackslashW(target);
return source;
}
static void copy_path_root(LPWSTR root, LPWSTR path)
{
LPWSTR p,p2;
INT i = 0;
p = path;
while (*p!=0)
p++;
while (*p!='\\' && p > path)
p--;
if (p == path)
return;
p2 = path;
while (p2 != p)
{
root[i] = *p2;
i++;
p2++;
}
root[i] = 0;
}
/*
* Command Line parameters are:
* [/n] Opens in single-paned view for each selected items. This is default
* [/e,] Uses Windows Explorer View
* [/cd,object] Specifies the root level of the view
* [/root,object] Specifies the root level of the view
* [/select,object] parent folder is opened and specified object is selected
*/
static void parse_command_line(LPWSTR commandline,parameters_struct *parameters)
{
static const WCHAR arg_n[] = {'/','n'};
static const WCHAR arg_e[] = {'/','e',','};
static const WCHAR arg_cd[] = {'/','c','d',','};
static const WCHAR arg_root[] = {'/','r','o','o','t',','};
static const WCHAR arg_select[] = {'/','s','e','l','e','c','t',','};
static const WCHAR arg_desktop[] = {'/','d','e','s','k','t','o','p'};
static const WCHAR arg_desktop_quotes[] = {'"','/','d','e','s','k','t','o','p'};
LPWSTR p = commandline;
while (*p)
{
while (iswspace(*p)) p++;
if (wcsncmp(p, arg_n, ARRAY_SIZE( arg_n ))==0)
{
parameters->explorer_mode = FALSE;
p += ARRAY_SIZE( arg_n );
}
else if (wcsncmp(p, arg_e, ARRAY_SIZE( arg_e ))==0)
{
parameters->explorer_mode = TRUE;
p += ARRAY_SIZE( arg_e );
}
else if (wcsncmp(p, arg_cd, ARRAY_SIZE( arg_cd ))==0)
{
p += ARRAY_SIZE( arg_cd );
p = copy_path_string(parameters->root,p);
}
else if (wcsncmp(p, arg_root, ARRAY_SIZE( arg_root ))==0)
{
p += ARRAY_SIZE( arg_root );
p = copy_path_string(parameters->root,p);
}
else if (wcsncmp(p, arg_select, ARRAY_SIZE( arg_select ))==0)
{
p += ARRAY_SIZE( arg_select );
p = copy_path_string(parameters->selection,p);
if (!parameters->root[0])
copy_path_root(parameters->root,
parameters->selection);
}
else if (wcsncmp(p, arg_desktop, ARRAY_SIZE( arg_desktop ))==0)
{
p += ARRAY_SIZE( arg_desktop );
manage_desktop( p ); /* the rest of the command line is handled by desktop mode */
}
/* workaround for Worms Armageddon that hardcodes a /desktop option with quotes */
else if (wcsncmp(p, arg_desktop_quotes, ARRAY_SIZE( arg_desktop_quotes ))==0)
{
p += ARRAY_SIZE( arg_desktop_quotes );
manage_desktop( p ); /* the rest of the command line is handled by desktop mode */
}
else
{
/* left over command line is generally the path to be opened */
copy_path_string(parameters->root,p);
break;
}
}
}
int WINAPI wWinMain(HINSTANCE hinstance,
HINSTANCE previnstance,
LPWSTR cmdline,
int cmdshow)
{
parameters_struct parameters;
HRESULT hres;
INITCOMMONCONTROLSEX init_info;
memset(&parameters,0,sizeof(parameters));
explorer_hInstance = hinstance;
parse_command_line(cmdline,&parameters);
hres = OleInitialize(NULL);
if(FAILED(hres))
{
WINE_ERR("Could not initialize COM\n");
ExitProcess(EXIT_FAILURE);
}
if(parameters.root[0] && !PathIsDirectoryW(parameters.root))
if(ShellExecuteW(NULL,NULL,parameters.root,NULL,NULL,SW_SHOWDEFAULT) > (HINSTANCE)32)
ExitProcess(EXIT_SUCCESS);
init_info.dwSize = sizeof(INITCOMMONCONTROLSEX);
init_info.dwICC = ICC_USEREX_CLASSES | ICC_BAR_CLASSES | ICC_COOL_CLASSES;
if(!InitCommonControlsEx(&init_info))
{
WINE_ERR("Could not initialize Comctl\n");
ExitProcess(EXIT_FAILURE);
}
register_explorer_window_class();
make_explorer_window(&parameters);
return 0;
}