windows-nt/Source/XPSP1/NT/printscan/ui/wiaacmgr/mintrans.cpp
2020-09-26 16:20:57 +08:00

551 lines
18 KiB
C++

/*******************************************************************************
*
* (C) COPYRIGHT MICROSOFT CORPORATION, 1998-2000
*
* TITLE: MINTRANS.CPP
*
* VERSION: 1.0
*
* AUTHOR: ShaunIv
*
* DATE: 12/6/1999
*
* DESCRIPTION:
*
*******************************************************************************/
#include "precomp.h"
#pragma hdrstop
#include <initguid.h>
#include <wiaregst.h>
#include <shlguid.h>
#include "shellext.h"
#include "shlobj.h"
#include "resource.h" // resource ids
#include "itranhlp.h"
#include "mintrans.h"
#include "comctrlp.h"
#include "shlwapip.h"
#include "acqmgrcw.h"
namespace
{
//
// Define constants for dwords stored in the registry
#define ACTION_RUNAPP 0
#define ACTION_AUTOSAVE 1
#define ACTION_NOTHING 2
#define ACTION_MAX 2
static const TCHAR c_szConnectionSettings[] = TEXT("OnConnect\\%ls");
struct CMinimalTransferSettings
{
DWORD dwAction;
BOOL bDeleteImages;
CSimpleString strFolderPath;
CComPtr<IWiaTransferHelper> pXfer;
BOOL bSaveInDatedDir;
};
#ifndef REGSTR_VALUE_USEDATE
#define REGSTR_VALUE_USEDATE TEXT("UseDate")
#endif
/*******************************************************************************
ConstructDatedFolderPath
Concatenate the date to an existing folder name
*******************************************************************************/
static
CSimpleString
ConstructDatedFolderPath(
const CSimpleString &strOriginal
)
{
CSimpleString strPath = strOriginal;
//
// Get the current date and format it as a string
//
SYSTEMTIME SystemTime;
TCHAR szDate[MAX_PATH] = TEXT("");
GetLocalTime( &SystemTime );
GetDateFormat( LOCALE_USER_DEFAULT, 0, &SystemTime, CSimpleString(IDS_DATEFORMAT,g_hInstance), szDate, ARRAYSIZE(szDate) );
//
// Make sure there is a trailing backslash
//
if (!strPath.MatchLastCharacter( TEXT('\\')))
{
strPath += CSimpleString(TEXT("\\"));
}
//
// Append the date
//
strPath += szDate;
return strPath;
}
/////////////////////////////////////////////////////////////////////////////
// CPersistCallback and helpers
/*******************************************************************************
CheckAndCreateFolder
Make sure the target path exists or can be created. Failing that, prompt the
user for a folder.
*******************************************************************************/
void
CheckAndCreateFolder (CSimpleString &strFolderPath)
{
// Convert to a full path name. If strFolderPath is not a full path,
// we want it to be a subfolder of My Pictures
TCHAR szFullPath[MAX_PATH] = TEXT("");
SHGetFolderPath (NULL, CSIDL_MYPICTURES, NULL, 0, szFullPath);
LPTSTR szUnused;
BOOL bPrompt = false;
if (*szFullPath)
{
SetCurrentDirectory (szFullPath);
}
GetFullPathName (strFolderPath, ARRAYSIZE(szFullPath), szFullPath, &szUnused);
strFolderPath = szFullPath;
// make sure the folder exists
DWORD dw = GetFileAttributes(strFolderPath);
if (dw == 0xffffffff)
{
bPrompt = !CAcquisitionManagerControllerWindow::RecursiveCreateDirectory( strFolderPath );
}
else if (!(dw & FILE_ATTRIBUTE_DIRECTORY))
{
bPrompt = TRUE;
}
// Ask the user for a valid folder
if (bPrompt)
{
BROWSEINFO bi;
TCHAR szPath[MAX_PATH] = TEXT("\0");
LPITEMIDLIST pidl;
TCHAR szTitle[200];
LoadString (g_hInstance,
IDS_MINTRANS_FOLDERPATH_CAPTION,
szTitle,
200);
ZeroMemory (&bi, sizeof(bi));
bi.hwndOwner = NULL;
bi.lpszTitle = szTitle;
bi.ulFlags = BIF_RETURNONLYFSDIRS | BIF_USENEWUI;
pidl = SHBrowseForFolder (&bi);
if (pidl)
{
SHGetPathFromIDList (pidl, szPath);
}
strFolderPath = szPath;
}
}
/*******************************************************************************
GetSaveSettings
Find out what the user configured us to do with the images
*******************************************************************************/
void
GetSaveSettings (CMinimalTransferSettings &settings, BSTR bstrDeviceId)
{
CSimpleReg regSettings(HKEY_CURRENT_USER,
REGSTR_PATH_USER_SETTINGS,
true,
KEY_READ);
// Default to My Pictures/no delete if registry settings not there
TCHAR szMyPictures[MAX_PATH];
SHGetFolderPath (NULL, CSIDL_MYPICTURES, NULL, 0, szMyPictures);
settings.bDeleteImages = 0;
settings.strFolderPath = const_cast<LPCTSTR>(szMyPictures);
settings.dwAction = ACTION_RUNAPP;
settings.bSaveInDatedDir = FALSE;
// BUGBUG: Should we prompt the user if the registry path
// isn't set?
if (regSettings.OK())
{
CSimpleString strSubKey;
strSubKey.Format (c_szConnectionSettings, bstrDeviceId);
CSimpleReg regActions (regSettings, strSubKey, true, KEY_READ);
settings.bDeleteImages = regActions.Query (REGSTR_VALUE_AUTODELETE, 0);
settings.strFolderPath = regActions.Query (REGSTR_VALUE_SAVEFOLDER,
CSimpleString(szMyPictures));
settings.dwAction = regActions.Query (REGSTR_VALUE_CONNECTACT,
ACTION_AUTOSAVE);
settings.bSaveInDatedDir = (regActions.Query(REGSTR_VALUE_USEDATE,0) != 0);
if (settings.bSaveInDatedDir)
{
settings.strFolderPath = ConstructDatedFolderPath( settings.strFolderPath );
}
}
}
// For the short term, have an array of format/extension pairs
struct MYFMTS
{
const GUID *pFmt;
LPCWSTR pszExt;
} FMTS [] =
{
{&WiaImgFmt_BMP, L".bmp"},
{&WiaImgFmt_JPEG, L".jpg"},
{&WiaImgFmt_FLASHPIX, L".fpx"},
{&WiaImgFmt_TIFF, L".tif"},
{NULL, L""}
};
/*******************************************************************************
GetDropTarget
Get an IDropTarget interface for the given folder
*******************************************************************************/
HRESULT
GetDropTarget (IShellFolder *pDesktop, LPCTSTR szPath, IDropTarget **ppDrop)
{
HRESULT hr;
LPITEMIDLIST pidl;
CSimpleStringWide strPath = CSimpleStringConvert::WideString (CSimpleString(szPath));
CComPtr<IShellFolder> psf;
hr = pDesktop->ParseDisplayName(NULL,
NULL,
const_cast<LPWSTR>(static_cast<LPCWSTR>(strPath)),
NULL,
&pidl,
NULL);
if (SUCCEEDED(hr))
{
hr = pDesktop->BindToObject(const_cast<LPCITEMIDLIST>(pidl),
NULL,
IID_IShellFolder,
reinterpret_cast<LPVOID*>(&psf));
if (SUCCEEDED(hr))
{
hr = psf->CreateViewObject (NULL,
IID_IDropTarget,
reinterpret_cast<LPVOID*>(ppDrop));
}
}
return hr;
}
/*******************************************************************************
FreePidl
Called when the array of pidls is destroyed, to free the pidls
*******************************************************************************/
INT
FreePidl (LPITEMIDLIST pidl, IMalloc *pMalloc)
{
pMalloc->Free (pidl);
return 1;
}
HRESULT
SaveItemsFromFolder (IShellFolder *pRoot, CSimpleString &strPath, BOOL bDelete)
{
CComPtr<IEnumIDList> pEnum;
LPITEMIDLIST pidl;
HRESULT hr = S_FALSE;
CComPtr<IMalloc> pMalloc;
if (SUCCEEDED(SHGetMalloc (&pMalloc)))
{
CComPtr<IShellFolder> pDesktop;
if (SUCCEEDED(SHGetDesktopFolder (&pDesktop)))
{
// enum the non-folder objects first
if (SUCCEEDED(pRoot->EnumObjects (NULL,
SHCONTF_FOLDERS | SHCONTF_NONFOLDERS ,
&pEnum)))
{
HDPA dpaItems;
dpaItems = DPA_Create(10);
while (NOERROR == pEnum->Next(1, &pidl, NULL))
{
DPA_AppendPtr (dpaItems, pidl);
}
//
// Now create the array of pidls and get the IDataObject
//
INT nSize = DPA_GetPtrCount (dpaItems);
if (nSize > 0)
{
LPITEMIDLIST *aidl = new LPITEMIDLIST[nSize];
if (aidl)
{
CComPtr<IDataObject> pdo;
for (INT i=0;i<nSize;i++)
{
aidl[i] = reinterpret_cast<LPITEMIDLIST>(DPA_FastGetPtr(dpaItems, i));
}
hr = pRoot->GetUIObjectOf (NULL,
nSize,
const_cast<LPCITEMIDLIST*>(aidl),
IID_IDataObject,
NULL,
reinterpret_cast<LPVOID*>(&pdo));
if (SUCCEEDED(hr))
{
CComPtr<IDropTarget> pDrop;
CComQIPtr<IAsyncOperation, &IID_IAsyncOperation> pasync(pdo);
if (pasync.p)
{
pasync->SetAsyncMode(FALSE);
}
CheckAndCreateFolder (strPath);
if (strPath.Length())
{
//
// Get an IDropTarget for the destination folder
// and do the drag/drop
//
hr = GetDropTarget (pDesktop,
strPath,
&pDrop);
}
else
{
hr = S_FALSE;
}
if (S_OK == hr)
{
DWORD dwKeyState;
if (bDelete)
{
// the "move" keys
dwKeyState = MK_SHIFT | MK_LBUTTON;
}
else
{ // the copy keys
dwKeyState = MK_CONTROL|MK_LBUTTON;
}
hr = SHSimulateDrop (pDrop,
pdo,
dwKeyState,
NULL,
NULL);
}
}
}
else
{
hr = E_OUTOFMEMORY;
}
}
else
{
hr = S_FALSE; // no images to download
}
DPA_DestroyCallback (dpaItems,
reinterpret_cast<PFNDPAENUMCALLBACK>(FreePidl),
reinterpret_cast<LPVOID>(pMalloc.p));
}
}
}
return hr;
}
/*******************************************************************************
SaveItems
This function uses IShellFolder and IDataObject interfaces to simulate
a drag/drop operation from the WIA virtual folder for the given device.
*******************************************************************************/
#define STR_WIASHEXT TEXT("wiashext.dll")
static
HRESULT
SaveItems (BSTR strDeviceId, CMinimalTransferSettings &settings)
{
WIA_PUSH_FUNCTION((TEXT("SaveItems( %ws, ... )"), strDeviceId ));
CComPtr<IShellFolder>pRoot;
HRESULT hr = SHGetDesktopFolder (&pRoot);
if (SUCCEEDED(hr))
{
//
// Get the system directory, which is where wiashext.dll lives
//
TCHAR szShellExtensionPath[MAX_PATH] = {0};
if (GetSystemDirectory( szShellExtensionPath, ARRAYSIZE(szShellExtensionPath)))
{
//
// Make sure the path variable is long enough to hold this path
//
if (lstrlen(szShellExtensionPath) + lstrlen(STR_WIASHEXT) + lstrlen(TEXT("\\")) < ARRAYSIZE(szShellExtensionPath))
{
//
// Concatenate the backslash and module name to the system path
//
lstrcat( szShellExtensionPath, TEXT("\\") );
lstrcat( szShellExtensionPath, STR_WIASHEXT );
//
// Load the DLL
//
HINSTANCE hInstanceShellExt = LoadLibrary(szShellExtensionPath);
if (hInstanceShellExt)
{
//
// Get the pidl making function
//
WIAMAKEFULLPIDLFORDEVICE pfnMakeFullPidlForDevice = reinterpret_cast<WIAMAKEFULLPIDLFORDEVICE>(GetProcAddress(hInstanceShellExt, "MakeFullPidlForDevice"));
if (pfnMakeFullPidlForDevice)
{
//
// Get the pidl
//
LPITEMIDLIST pidlDevice = NULL;
hr = pfnMakeFullPidlForDevice( strDeviceId, &pidlDevice );
if (SUCCEEDED(hr))
{
//
// Bind to the folder for this device
//
CComPtr<IShellFolder> pDevice;
hr = pRoot->BindToObject (const_cast<LPCITEMIDLIST> (pidlDevice), NULL, IID_IShellFolder, reinterpret_cast<LPVOID*>(&pDevice));
if (SUCCEEDED(hr))
{
hr = SaveItemsFromFolder (pDevice, settings.strFolderPath, settings.bDeleteImages);
if (S_OK == hr && settings.bDeleteImages)
{
//
// DoDeleteAllItems will pop up a dialog to confirm the delete.
//
DoDeleteAllItems (strDeviceId, NULL);
}
}
else
{
WIA_PRINTHRESULT((hr,TEXT("BindToObject failed!")));
}
CComPtr<IMalloc> pMalloc;
if (SUCCEEDED(SHGetMalloc(&pMalloc)) && pMalloc)
{
pMalloc->Free(pidlDevice);
}
}
else
{
WIA_PRINTHRESULT((hr,TEXT("MakeFullPidlForDevice failed!")));
}
}
else
{
hr = HRESULT_FROM_WIN32(GetLastError());
WIA_PRINTHRESULT((hr,TEXT("GetProcAddress for MakeFullPidlForDevice failed!")));
}
FreeLibrary(hInstanceShellExt);
}
else
{
hr = HRESULT_FROM_WIN32(GetLastError());
WIA_PRINTHRESULT((hr,TEXT("Unable to load wiashext.dll!")));
}
}
else
{
hr = E_FAIL;
WIA_PRINTHRESULT((hr,TEXT("Buffer size was too small")));
}
}
else
{
hr = HRESULT_FROM_WIN32(GetLastError());
WIA_PRINTHRESULT((hr,TEXT("Unable to get system folder!")));
}
}
else
{
WIA_PRINTHRESULT((hr,TEXT("SHGetDesktopFolder failed!")));
}
return hr;
}
} // End namespace MinimalTransfer
LRESULT
MinimalTransferThreadProc (BSTR bstrDeviceId)
{
if (bstrDeviceId)
{
CMinimalTransferSettings settings;
HRESULT hr = CoInitialize(NULL);
if (SUCCEEDED(hr))
{
GetSaveSettings (settings, bstrDeviceId);
// Bail if the default action is donothing or if the user cancelled
// the browse for folder
if (settings.dwAction == ACTION_AUTOSAVE)
{
hr = SaveItems (bstrDeviceId, settings);
// Show the folder the user saved to
if (NOERROR == hr)
{
SHELLEXECUTEINFO sei;
ZeroMemory (&sei, sizeof(sei));
sei.cbSize = sizeof(sei);
sei.lpDirectory = settings.strFolderPath;
sei.nShow = SW_SHOW;
ShellExecuteEx (&sei);
}
else if (FAILED(hr))
{
WIA_PRINTHRESULT((hr,TEXT("SaveItems failed!")));
// we can rely on SaveItems reporting errors to the user
}
}
CoUninitialize();
}
#ifndef DBG_GENERATE_PRETEND_EVENT
WIA_TRACE((TEXT("Module::m_nLockCnt: %d"),_Module.m_nLockCnt));
_Module.Unlock();
#endif
SysFreeString(bstrDeviceId);
}
return 0;
}