2005-11-16 12:45:51 +01:00
|
|
|
/*
|
|
|
|
* explorer.exe
|
|
|
|
*
|
2006-02-19 18:50:27 +01:00
|
|
|
* Copyright 2004 CodeWeavers, Mike Hearn
|
2006-01-14 17:05:18 +01:00
|
|
|
* Copyright 2005,2006 CodeWeavers, Aric Stewart
|
2011-06-11 23:42:17 +02:00
|
|
|
* Copyright 2011 Jay Yang
|
2005-11-16 12:45:51 +01:00
|
|
|
*
|
|
|
|
* 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
|
2006-05-18 14:49:52 +02:00
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
2005-11-16 12:45:51 +01:00
|
|
|
*/
|
|
|
|
|
2011-06-11 23:42:17 +02:00
|
|
|
#define COBJMACROS
|
2005-11-16 12:45:51 +01:00
|
|
|
|
2008-04-23 18:04:43 +02:00
|
|
|
#include "wine/unicode.h"
|
2011-06-11 23:42:17 +02:00
|
|
|
#include "wine/debug.h"
|
2006-03-07 11:41:52 +01:00
|
|
|
#include "explorer_private.h"
|
2011-06-11 23:42:17 +02:00
|
|
|
#include "resource.h"
|
|
|
|
|
|
|
|
#include <initguid.h>
|
|
|
|
#include <windows.h>
|
|
|
|
#include <shobjidl.h>
|
|
|
|
#include <shlobj.h>
|
2011-07-04 03:35:55 +02:00
|
|
|
#include <commoncontrols.h>
|
|
|
|
#include <commctrl.h>
|
2011-06-11 23:42:17 +02:00
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(explorer);
|
|
|
|
|
|
|
|
#define EXPLORER_INFO_INDEX 0
|
|
|
|
|
2011-07-04 03:35:55 +02:00
|
|
|
#define NAV_TOOLBAR_HEIGHT 30
|
2011-07-04 03:36:01 +02:00
|
|
|
#define PATHBOX_HEIGHT 24
|
2011-07-04 03:35:55 +02:00
|
|
|
|
2011-06-11 23:42:17 +02:00
|
|
|
#define DEFAULT_WIDTH 640
|
|
|
|
#define DEFAULT_HEIGHT 480
|
|
|
|
|
|
|
|
|
|
|
|
static const WCHAR EXPLORER_CLASS[] = {'W','I','N','E','_','E','X','P','L','O','R','E','R','\0'};
|
2011-07-04 03:36:01 +02:00
|
|
|
static const WCHAR PATH_BOX_NAME[] = {'\0'};
|
2011-06-11 23:42:17 +02:00
|
|
|
|
|
|
|
HINSTANCE explorer_hInstance;
|
2006-02-19 18:50:27 +01:00
|
|
|
|
2005-11-16 12:45:51 +01:00
|
|
|
typedef struct parametersTAG {
|
|
|
|
BOOL explorer_mode;
|
|
|
|
WCHAR root[MAX_PATH];
|
|
|
|
WCHAR selection[MAX_PATH];
|
|
|
|
} parameters_struct;
|
|
|
|
|
2011-06-11 23:42:17 +02:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
IExplorerBrowser *browser;
|
2011-07-04 03:36:01 +02:00
|
|
|
HWND main_window,path_box;
|
2011-07-04 03:35:55 +02:00
|
|
|
INT rebar_height;
|
2011-07-04 03:36:01 +02:00
|
|
|
LPCITEMIDLIST pidl;
|
2011-07-04 03:36:05 +02:00
|
|
|
IImageList *icon_list;
|
2011-06-11 23:42:17 +02:00
|
|
|
} explorer_info;
|
|
|
|
|
2011-07-04 03:35:55 +02:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
BACK_BUTTON,FORWARD_BUTTON,UP_BUTTON
|
|
|
|
};
|
|
|
|
|
2011-07-04 03:36:01 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2011-07-04 03:36:05 +02:00
|
|
|
static BOOL create_combobox_item(IShellFolder *folder, LPCITEMIDLIST pidl, IImageList *icon_list, COMBOBOXEXITEMW *item)
|
|
|
|
{
|
|
|
|
STRRET strret;
|
|
|
|
HRESULT hres;
|
|
|
|
IExtractIconW *extract_icon;
|
|
|
|
UINT reserved;
|
|
|
|
WCHAR icon_file[MAX_PATH];
|
|
|
|
INT icon_index;
|
|
|
|
UINT icon_flags;
|
|
|
|
HICON icon;
|
|
|
|
strret.uType=STRRET_WSTR;
|
|
|
|
hres = IShellFolder_GetDisplayNameOf(folder,pidl,SHGDN_FORADDRESSBAR,&strret);
|
|
|
|
if(FAILED(hres))
|
|
|
|
{
|
|
|
|
WINE_WARN("Could not get name for pidl\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
switch(strret.uType)
|
|
|
|
{
|
|
|
|
case STRRET_WSTR:
|
|
|
|
item->pszText = strret.pOleStr;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
WINE_FIXME("Unimplemented STRRET type:%u\n",strret.uType);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
hres = IShellFolder_GetUIObjectOf(folder,NULL,1,&pidl,&IID_IExtractIconW,
|
|
|
|
&reserved,(void**)&extract_icon);
|
|
|
|
if(SUCCEEDED(hres))
|
|
|
|
{
|
|
|
|
item->mask |= CBEIF_IMAGE;
|
|
|
|
IExtractIconW_GetIconLocation(extract_icon,GIL_FORSHELL,icon_file,
|
|
|
|
sizeof(icon_file)/sizeof(WCHAR),
|
|
|
|
&icon_index,&icon_flags);
|
|
|
|
IExtractIconW_Extract(extract_icon,icon_file,icon_index,NULL,&icon,20);
|
|
|
|
item->iImage = ImageList_AddIcon((HIMAGELIST)icon_list,icon);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
item->mask &= ~CBEIF_IMAGE;
|
|
|
|
WINE_WARN("Could not get an icon for %s\n",wine_dbgstr_w(item->pszText));
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2011-07-04 03:36:01 +02:00
|
|
|
static void update_path_box(explorer_info *info)
|
|
|
|
{
|
2011-07-04 03:36:05 +02:00
|
|
|
COMBOBOXEXITEMW item;
|
|
|
|
COMBOBOXEXITEMW main_item;
|
|
|
|
IShellFolder *desktop;
|
|
|
|
IPersistFolder2 *persist;
|
|
|
|
LPITEMIDLIST desktop_pidl;
|
|
|
|
IEnumIDList *ids;
|
|
|
|
|
|
|
|
ImageList_Remove((HIMAGELIST)info->icon_list,-1);
|
|
|
|
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;
|
|
|
|
HRESULT hres;
|
|
|
|
|
|
|
|
item.iIndent = 1;
|
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
WINE_WARN("Could not enumerate the desktop\n");
|
|
|
|
SendMessageW(info->path_box,CBEM_SETITEMW,0,(LPARAM)&main_item);
|
|
|
|
CoTaskMemFree(main_item.pszText);
|
2011-07-04 03:36:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI IExplorerBrowserEventsImpl_fnOnNavigationComplete(IExplorerBrowserEvents *iface, PCIDLIST_ABSOLUTE pidl)
|
|
|
|
{
|
|
|
|
IExplorerBrowserEventsImpl *This = impl_from_IExplorerBrowserEvents(iface);
|
|
|
|
This->info->pidl = pidl;
|
|
|
|
update_path_box(This->info);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
2011-07-04 03:36:05 +02:00
|
|
|
SHGetImageList(SHIL_SMALL,&IID_IImageList,(void**)&(ret->info->icon_list));
|
|
|
|
SendMessageW(info->path_box,CBEM_SETIMAGELIST,0,(LPARAM)ret->info->icon_list);
|
2011-07-04 03:36:01 +02:00
|
|
|
IExplorerBrowserEvents_AddRef(&ret->IExplorerBrowserEvents_iface);
|
|
|
|
return &ret->IExplorerBrowserEvents_iface;
|
|
|
|
}
|
|
|
|
|
2011-06-11 23:42:17 +02:00
|
|
|
static void make_explorer_window(IShellFolder* startFolder)
|
|
|
|
{
|
|
|
|
RECT explorerRect;
|
2011-07-04 03:35:55 +02:00
|
|
|
HWND rebar,nav_toolbar;
|
2011-06-11 23:42:17 +02:00
|
|
|
FOLDERSETTINGS fs;
|
2011-07-04 03:36:01 +02:00
|
|
|
IExplorerBrowserEvents *events;
|
2011-06-11 23:42:17 +02:00
|
|
|
explorer_info *info;
|
|
|
|
HRESULT hres;
|
2011-07-04 03:36:01 +02:00
|
|
|
DWORD cookie;
|
2011-06-11 23:42:17 +02:00
|
|
|
WCHAR explorer_title[100];
|
2011-07-04 03:36:01 +02:00
|
|
|
WCHAR pathbox_label[50];
|
2011-07-04 03:35:55 +02:00
|
|
|
TBADDBITMAP bitmap_info;
|
|
|
|
TBBUTTON nav_buttons[3];
|
|
|
|
int hist_offset,view_offset;
|
|
|
|
REBARBANDINFOW band_info;
|
|
|
|
memset(nav_buttons,0,sizeof(nav_buttons));
|
2011-06-11 23:42:17 +02:00
|
|
|
LoadStringW(explorer_hInstance,IDS_EXPLORER_TITLE,explorer_title,
|
|
|
|
sizeof(explorer_title)/sizeof(WCHAR));
|
2011-07-04 03:36:01 +02:00
|
|
|
LoadStringW(explorer_hInstance,IDS_PATHBOX_LABEL,pathbox_label,
|
|
|
|
sizeof(pathbox_label)/sizeof(WCHAR));
|
2011-06-11 23:42:17 +02:00
|
|
|
info = HeapAlloc(GetProcessHeap(),0,sizeof(explorer_info));
|
|
|
|
if(!info)
|
|
|
|
{
|
|
|
|
WINE_ERR("Could not allocate a explorer_info struct\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
hres = CoCreateInstance(&CLSID_ExplorerBrowser,NULL,CLSCTX_INPROC_SERVER,
|
|
|
|
&IID_IExplorerBrowser,(LPVOID*)&info->browser);
|
|
|
|
if(!SUCCEEDED(hres))
|
|
|
|
{
|
|
|
|
WINE_ERR("Could not obtain an instance of IExplorerBrowser\n");
|
|
|
|
HeapFree(GetProcessHeap(),0,info);
|
|
|
|
return;
|
|
|
|
}
|
2011-07-04 03:35:55 +02:00
|
|
|
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);
|
2011-06-11 23:42:17 +02:00
|
|
|
|
|
|
|
fs.ViewMode = FVM_DETAILS;
|
|
|
|
fs.fFlags = FWF_AUTOARRANGE;
|
|
|
|
explorerRect.left = 0;
|
|
|
|
explorerRect.top = 0;
|
|
|
|
explorerRect.right = DEFAULT_WIDTH;
|
|
|
|
explorerRect.bottom = DEFAULT_HEIGHT;
|
|
|
|
|
2011-07-04 03:35:55 +02:00
|
|
|
IExplorerBrowser_Initialize(info->browser,info->main_window,&explorerRect,&fs);
|
2011-06-11 23:42:17 +02:00
|
|
|
IExplorerBrowser_SetOptions(info->browser,EBO_SHOWFRAMES);
|
2011-07-05 19:17:45 +02:00
|
|
|
SetWindowLongPtrW(info->main_window,EXPLORER_INFO_INDEX,(LONG_PTR)info);
|
2011-07-04 03:35:55 +02:00
|
|
|
|
|
|
|
/*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,sizeof(nav_buttons)/sizeof(TBBUTTON),(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);
|
2011-07-04 03:36:01 +02:00
|
|
|
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);
|
|
|
|
band_info.cyChild=PATHBOX_HEIGHT;
|
|
|
|
band_info.cx=0;
|
|
|
|
band_info.cyMinChild=PATHBOX_HEIGHT;
|
|
|
|
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,&cookie);
|
2011-06-11 23:42:17 +02:00
|
|
|
IExplorerBrowser_BrowseToObject(info->browser,(IUnknown*)startFolder,
|
|
|
|
SBSP_ABSOLUTE);
|
2011-07-04 03:35:55 +02:00
|
|
|
ShowWindow(info->main_window,SW_SHOWDEFAULT);
|
|
|
|
UpdateWindow(info->main_window);
|
2011-07-04 03:36:01 +02:00
|
|
|
IExplorerBrowserEvents_Release(events);
|
2011-06-11 23:42:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void update_window_size(explorer_info *info, int height, int width)
|
|
|
|
{
|
|
|
|
RECT new_rect;
|
|
|
|
new_rect.left = 0;
|
2011-07-04 03:35:55 +02:00
|
|
|
new_rect.top = info->rebar_height;
|
2011-06-11 23:42:17 +02:00
|
|
|
new_rect.right = width;
|
|
|
|
new_rect.bottom = height;
|
|
|
|
IExplorerBrowser_SetRect(info->browser,NULL,new_rect);
|
|
|
|
}
|
2005-11-16 12:45:51 +01:00
|
|
|
|
2011-06-11 23:42:17 +02:00
|
|
|
static void do_exit(int code)
|
|
|
|
{
|
|
|
|
OleUninitialize();
|
|
|
|
ExitProcess(code);
|
|
|
|
}
|
|
|
|
|
2011-07-04 03:36:01 +02:00
|
|
|
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)
|
|
|
|
{
|
2011-07-04 03:36:05 +02:00
|
|
|
case CBENF_DROPDOWN:
|
|
|
|
if(edit_info->iNewSelection!=CB_ERR)
|
|
|
|
pidl = (LPITEMIDLIST)SendMessageW(edit_info->hdr.hwndFrom,
|
|
|
|
CB_GETITEMDATA,
|
|
|
|
edit_info->iNewSelection,0);
|
|
|
|
break;
|
2011-07-04 03:36:01 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2011-07-04 03:35:55 +02:00
|
|
|
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)
|
|
|
|
{
|
2011-07-04 03:36:05 +02:00
|
|
|
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);
|
|
|
|
}
|
2011-07-04 03:36:01 +02:00
|
|
|
case CBEN_ENDEDITW:
|
|
|
|
return explorer_on_end_edit(info,(NMCBEENDEDITW*)notification);
|
2011-07-04 03:36:05 +02:00
|
|
|
case CBEN_DELETEITEM:
|
|
|
|
{
|
|
|
|
NMCOMBOBOXEXW *entry = (NMCOMBOBOXEXW*)notification;
|
|
|
|
if(entry->ceItem.lParam)
|
|
|
|
ILFree((LPITEMIDLIST)entry->ceItem.lParam);
|
|
|
|
break;
|
|
|
|
}
|
2011-07-04 03:35:55 +02:00
|
|
|
case RBN_AUTOSIZE:
|
|
|
|
return update_rebar_size(info,(NMRBAUTOSIZE*)notification);
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-06-19 14:09:31 +02:00
|
|
|
static LRESULT CALLBACK explorer_wnd_proc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
2011-06-11 23:42:17 +02:00
|
|
|
{
|
|
|
|
explorer_info *info
|
|
|
|
= (explorer_info*)GetWindowLongPtrW(hwnd,EXPLORER_INFO_INDEX);
|
|
|
|
IExplorerBrowser *browser = NULL;
|
|
|
|
|
2011-07-04 03:35:55 +02:00
|
|
|
WINE_TRACE("(hwnd=%p,uMsg=%u,wParam=%lx,lParam=%lx)\n",hwnd,uMsg,wParam,lParam);
|
2011-06-11 23:42:17 +02:00
|
|
|
if(info)
|
|
|
|
browser = info->browser;
|
|
|
|
switch(uMsg)
|
|
|
|
{
|
|
|
|
case WM_DESTROY:
|
|
|
|
IExplorerBrowser_Release(browser);
|
|
|
|
HeapFree(GetProcessHeap(),0,info);
|
|
|
|
PostQuitMessage(0);
|
|
|
|
break;
|
|
|
|
case WM_QUIT:
|
|
|
|
do_exit(wParam);
|
2011-07-04 03:35:55 +02:00
|
|
|
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;
|
2011-06-11 23:42:17 +02:00
|
|
|
case WM_SIZE:
|
|
|
|
update_window_size(info,HIWORD(lParam),LOWORD(lParam));
|
|
|
|
break;
|
|
|
|
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;
|
2011-07-04 03:35:55 +02:00
|
|
|
window_class.hbrBackground = (HBRUSH)COLOR_BACKGROUND;
|
2011-06-11 23:42:17 +02:00
|
|
|
window_class.lpszMenuName = NULL;
|
|
|
|
window_class.lpszClassName = EXPLORER_CLASS;
|
|
|
|
window_class.hIconSm = NULL;
|
|
|
|
RegisterClassExW(&window_class);
|
|
|
|
}
|
|
|
|
|
2011-06-11 23:42:21 +02:00
|
|
|
static IShellFolder* get_starting_shell_folder(parameters_struct* params)
|
|
|
|
{
|
|
|
|
IShellFolder* desktop,*folder;
|
|
|
|
LPITEMIDLIST root_pidl;
|
|
|
|
HRESULT hres;
|
|
|
|
|
|
|
|
SHGetDesktopFolder(&desktop);
|
2011-06-28 08:31:31 +02:00
|
|
|
if (!params->root[0])
|
2011-06-11 23:42:21 +02:00
|
|
|
{
|
|
|
|
return desktop;
|
|
|
|
}
|
|
|
|
hres = IShellFolder_ParseDisplayName(desktop,NULL,NULL,
|
|
|
|
params->root,NULL,
|
|
|
|
&root_pidl,NULL);
|
|
|
|
|
|
|
|
if(FAILED(hres))
|
|
|
|
{
|
|
|
|
return desktop;
|
|
|
|
}
|
|
|
|
hres = IShellFolder_BindToObject(desktop,root_pidl,NULL,
|
|
|
|
&IID_IShellFolder,
|
|
|
|
(void**)&folder);
|
|
|
|
if(FAILED(hres))
|
|
|
|
{
|
|
|
|
return desktop;
|
|
|
|
}
|
|
|
|
IShellFolder_Release(desktop);
|
|
|
|
return folder;
|
|
|
|
}
|
|
|
|
|
2011-06-11 23:42:17 +02:00
|
|
|
static int copy_path_string(LPWSTR target, LPWSTR source)
|
2005-11-16 12:45:51 +01:00
|
|
|
{
|
|
|
|
INT i = 0;
|
|
|
|
|
2008-04-23 18:04:43 +02:00
|
|
|
while (isspaceW(*source)) source++;
|
2005-11-16 12:45:51 +01:00
|
|
|
|
|
|
|
if (*source == '\"')
|
|
|
|
{
|
|
|
|
source ++;
|
2008-04-23 18:04:43 +02:00
|
|
|
while (*source != '\"') target[i++] = *source++;
|
|
|
|
target[i] = 0;
|
2005-11-16 12:45:51 +01:00
|
|
|
source ++;
|
|
|
|
i+=2;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-04-23 18:04:43 +02:00
|
|
|
while (*source && !isspaceW(*source)) target[i++] = *source++;
|
|
|
|
target[i] = 0;
|
2005-11-16 12:45:51 +01:00
|
|
|
}
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-11 23:42:17 +02:00
|
|
|
static void copy_path_root(LPWSTR root, LPWSTR path)
|
2005-11-16 12:45:51 +01:00
|
|
|
{
|
|
|
|
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
|
|
|
|
* [/root,object] Specifies the root level of the view
|
|
|
|
* [/select,object] parent folder is opened and specified object is selected
|
|
|
|
*/
|
2011-06-11 23:42:17 +02:00
|
|
|
static void parse_command_line(LPWSTR commandline,parameters_struct *parameters)
|
2005-11-16 12:45:51 +01:00
|
|
|
{
|
2008-04-23 18:04:43 +02:00
|
|
|
static const WCHAR arg_n[] = {'/','n'};
|
|
|
|
static const WCHAR arg_e[] = {'/','e',','};
|
|
|
|
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'};
|
|
|
|
|
|
|
|
LPWSTR p, p2;
|
|
|
|
|
2005-11-16 12:45:51 +01:00
|
|
|
p2 = commandline;
|
2008-04-23 18:04:43 +02:00
|
|
|
p = strchrW(commandline,'/');
|
2005-11-16 12:45:51 +01:00
|
|
|
while(p)
|
|
|
|
{
|
2008-04-23 18:04:43 +02:00
|
|
|
if (strncmpW(p, arg_n, sizeof(arg_n)/sizeof(WCHAR))==0)
|
2005-11-16 12:45:51 +01:00
|
|
|
{
|
|
|
|
parameters->explorer_mode = FALSE;
|
2008-04-23 18:04:43 +02:00
|
|
|
p += sizeof(arg_n)/sizeof(WCHAR);
|
2005-11-16 12:45:51 +01:00
|
|
|
}
|
2008-04-23 18:04:43 +02:00
|
|
|
else if (strncmpW(p, arg_e, sizeof(arg_e)/sizeof(WCHAR))==0)
|
2005-11-16 12:45:51 +01:00
|
|
|
{
|
|
|
|
parameters->explorer_mode = TRUE;
|
2008-04-23 18:04:43 +02:00
|
|
|
p += sizeof(arg_e)/sizeof(WCHAR);
|
2005-11-16 12:45:51 +01:00
|
|
|
}
|
2008-04-23 18:04:43 +02:00
|
|
|
else if (strncmpW(p, arg_root, sizeof(arg_root)/sizeof(WCHAR))==0)
|
2005-11-16 12:45:51 +01:00
|
|
|
{
|
2008-04-23 18:04:43 +02:00
|
|
|
p += sizeof(arg_root)/sizeof(WCHAR);
|
2011-06-11 23:42:17 +02:00
|
|
|
p+=copy_path_string(parameters->root,p);
|
2005-11-16 12:45:51 +01:00
|
|
|
}
|
2008-04-23 18:04:43 +02:00
|
|
|
else if (strncmpW(p, arg_select, sizeof(arg_select)/sizeof(WCHAR))==0)
|
2005-11-16 12:45:51 +01:00
|
|
|
{
|
2008-04-23 18:04:43 +02:00
|
|
|
p += sizeof(arg_select)/sizeof(WCHAR);
|
2011-06-11 23:42:17 +02:00
|
|
|
p+=copy_path_string(parameters->selection,p);
|
2005-11-16 12:45:51 +01:00
|
|
|
if (!parameters->root[0])
|
2011-06-11 23:42:17 +02:00
|
|
|
copy_path_root(parameters->root,
|
|
|
|
parameters->selection);
|
2005-11-16 12:45:51 +01:00
|
|
|
}
|
2008-04-23 18:04:43 +02:00
|
|
|
else if (strncmpW(p, arg_desktop, sizeof(arg_desktop)/sizeof(WCHAR))==0)
|
2006-03-07 11:41:52 +01:00
|
|
|
{
|
2008-04-23 18:04:43 +02:00
|
|
|
p += sizeof(arg_desktop)/sizeof(WCHAR);
|
|
|
|
manage_desktop( p ); /* the rest of the command line is handled by desktop mode */
|
2006-03-07 11:41:52 +01:00
|
|
|
}
|
2008-04-23 18:04:43 +02:00
|
|
|
else p++;
|
|
|
|
|
2005-11-16 12:45:51 +01:00
|
|
|
p2 = p;
|
2008-04-23 18:04:43 +02:00
|
|
|
p = strchrW(p,'/');
|
2005-11-16 12:45:51 +01:00
|
|
|
}
|
|
|
|
if (p2 && *p2)
|
|
|
|
{
|
|
|
|
/* left over command line is generally the path to be opened */
|
2011-06-11 23:42:17 +02:00
|
|
|
copy_path_string(parameters->root,p2);
|
2005-11-16 12:45:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-04-23 18:04:43 +02:00
|
|
|
int WINAPI wWinMain(HINSTANCE hinstance,
|
|
|
|
HINSTANCE previnstance,
|
|
|
|
LPWSTR cmdline,
|
|
|
|
int cmdshow)
|
2005-11-16 12:45:51 +01:00
|
|
|
{
|
2011-06-11 23:42:17 +02:00
|
|
|
|
2005-11-16 12:45:51 +01:00
|
|
|
parameters_struct parameters;
|
2011-06-11 23:42:17 +02:00
|
|
|
HRESULT hres;
|
|
|
|
MSG msg;
|
|
|
|
IShellFolder *folder;
|
2011-07-04 03:35:55 +02:00
|
|
|
INITCOMMONCONTROLSEX init_info;
|
2005-11-16 12:45:51 +01:00
|
|
|
|
|
|
|
memset(¶meters,0,sizeof(parameters));
|
2011-06-11 23:42:17 +02:00
|
|
|
explorer_hInstance = hinstance;
|
|
|
|
parse_command_line(cmdline,¶meters);
|
|
|
|
hres = OleInitialize(NULL);
|
|
|
|
if(!SUCCEEDED(hres))
|
2006-01-14 17:05:18 +01:00
|
|
|
{
|
2011-06-11 23:42:17 +02:00
|
|
|
WINE_ERR("Could not initialize COM\n");
|
|
|
|
ExitProcess(EXIT_FAILURE);
|
2006-01-14 17:05:18 +01:00
|
|
|
}
|
2011-07-04 03:35:55 +02:00
|
|
|
init_info.dwSize = sizeof(INITCOMMONCONTROLSEX);
|
2011-07-04 03:36:01 +02:00
|
|
|
init_info.dwICC = ICC_USEREX_CLASSES | ICC_BAR_CLASSES | ICC_COOL_CLASSES;
|
2011-07-04 03:35:55 +02:00
|
|
|
if(!InitCommonControlsEx(&init_info))
|
|
|
|
{
|
|
|
|
WINE_ERR("Could not initialize Comctl\n");
|
|
|
|
ExitProcess(EXIT_FAILURE);
|
|
|
|
}
|
2011-06-11 23:42:17 +02:00
|
|
|
register_explorer_window_class();
|
2011-06-11 23:42:21 +02:00
|
|
|
folder = get_starting_shell_folder(¶meters);
|
2011-06-11 23:42:17 +02:00
|
|
|
make_explorer_window(folder);
|
|
|
|
IShellFolder_Release(folder);
|
|
|
|
while(GetMessageW( &msg, NULL, 0, 0 ) != 0)
|
2006-01-14 17:05:18 +01:00
|
|
|
{
|
2011-06-11 23:42:17 +02:00
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessageW(&msg);
|
2006-01-14 17:05:18 +01:00
|
|
|
}
|
2005-11-16 12:45:51 +01:00
|
|
|
return 0;
|
|
|
|
}
|