mirror of
https://github.com/fadden/ciderpress.git
synced 2024-11-27 08:49:20 +00:00
d60a6dbcfa
Includes "printf format" annotation on debug log function.
590 lines
19 KiB
C++
590 lines
19 KiB
C++
/*
|
|
* CiderPress
|
|
* Copyright (C) 2007 by faddenSoft, LLC. All Rights Reserved.
|
|
* See the file LICENSE for distribution terms.
|
|
*/
|
|
/*
|
|
* Support for file selection dialog.
|
|
*
|
|
* What we want to do is get the full set of selected files out of the dialog.
|
|
* However, Windows does not provide a way to do that. There are various
|
|
* slightly awkward approaches:
|
|
*
|
|
* - Get the IShellView pointer via the IShellBrowser pointer via the
|
|
* undocumented WM_GETISHELLBROWSER message. You can then play with the
|
|
* shellview window, select items with SelectItem(), and get an object
|
|
* representing the selected items with another call. What form the
|
|
* object takes is frustratingly unspecified.
|
|
* http://www.codeproject.com/dialog/select_all_button.asp
|
|
* - Get the CListView so you can play with its members. This is done by
|
|
* GetParent --> GetItem(lst2) --> GetItem(1), which is probably
|
|
* somewhat fragile, but seems to work on 98 through XP. Standard LVN
|
|
* stuff applies.
|
|
* http://www.codeguru.com/mfc/comments/10216.shtml
|
|
* - Using a window hook (not an OFN hook), get the class name and strcmp()
|
|
* it for the class we're looking for ("syslistview32"). Once we have
|
|
* it, proceed as above.
|
|
* http://www.codeproject.com/dialog/customize_dialog.asp
|
|
*
|
|
* Care must be taken with the ListView because it doesn't contain file names,
|
|
* but rather display names (potentially without filename extensions). The
|
|
* next big assumptive leap is that the ItemData pointer is a shell PIDL that
|
|
* can be used directly.
|
|
*
|
|
* The PIDL stored is, of course, relative to the IShellFolder currently being
|
|
* displayed. There's no easy way to get that, but if we just go ahead and
|
|
* call SHGetPathFromIDList anyway we get the right file name with the wrong
|
|
* path (it appears to be the desktop folder). We can strip the path off and
|
|
* prepend the value from the undocumented GetFolderPath() call (which just
|
|
* issues a CDM_GETFOLDERPATH message).
|
|
*
|
|
* To make matters even more interesting, it is necessary to provide a "hook"
|
|
* function to prevent double-clicking from closing the dialog while side-
|
|
* stepping our additions. Of course, adding an OFN hook renders all of the
|
|
* existing message map and initialization stuff inoperable. You can stuff
|
|
* things through various "user data" pointers and end up with access to your
|
|
* object, and if you cram the hook procedure's hDlg into the object's m_hWnd
|
|
* you can treat "this" like a window instead of passing HWNDs around.
|
|
*/
|
|
#include "stdafx.h"
|
|
#include "SelectFilesDialog.h"
|
|
#include "PathName.h"
|
|
#include "Util.h"
|
|
#include <dlgs.h>
|
|
|
|
|
|
/*
|
|
* Our CFileDialog "hook" function.
|
|
*
|
|
* "hdlg" is a handle to the child dialog box. Use the GetParent() function
|
|
* to get the handle of the dialog box window.
|
|
*
|
|
* uiMsg identifies the message being received. If it's WM_INITDIALOG, then
|
|
* lParam points to the OPENFILENAME structure.
|
|
*
|
|
* Do not call EndDialog from here. Instead, PostMessage a WM_COMMAND with
|
|
* IDABORT. (Looks like you can EndDialog on the parent and have it work, at
|
|
* least for IDCANCEL.)
|
|
*
|
|
* Return zero to enable standard processing, nonzero to claim ownership of
|
|
* the message.
|
|
*/
|
|
/*static*/ UINT CALLBACK
|
|
SelectFilesDialog::OFNHookProc(HWND hDlg, UINT uiMsg, WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
OPENFILENAME* pOfn;
|
|
SelectFilesDialog* pSFD = NULL;
|
|
pOfn = (OPENFILENAME*) GetWindowLong(hDlg, GWL_USERDATA);
|
|
if (pOfn != NULL) {
|
|
pSFD = (SelectFilesDialog*) pOfn->lCustData;
|
|
/* allow our "this" pointer to play with the window */
|
|
/* [does not seem to cause double-frees on cleanup] */
|
|
if (pSFD->m_hWnd == NULL)
|
|
pSFD->m_hWnd = hDlg;
|
|
}
|
|
|
|
switch (uiMsg) {
|
|
case WM_INITDIALOG:
|
|
LOGI("WM_INITDIALOG, OFN=0x%08lx", lParam);
|
|
SetWindowLong(hDlg, GWL_USERDATA, lParam);
|
|
break;
|
|
case WM_NOTIFY: // 0x4e
|
|
ASSERT(pSFD != NULL);
|
|
return pSFD->HandleNotify(hDlg, (LPOFNOTIFY)lParam);
|
|
case WM_COMMAND:
|
|
ASSERT(pSFD != NULL);
|
|
return pSFD->HandleCommand(hDlg, wParam, lParam);
|
|
case WM_SIZE:
|
|
ASSERT(pSFD != NULL);
|
|
return pSFD->HandleSize(hDlg, wParam, LOWORD(lParam), HIWORD(lParam));
|
|
case WM_HELP:
|
|
ASSERT(pSFD != NULL);
|
|
return pSFD->HandleHelp(hDlg, (LPHELPINFO) lParam);
|
|
default:
|
|
//LOGI("OFNHookProc: hDlg=0x%08lx uiMsg=0x%08lx "
|
|
// "wParam=0x%08lx lParam=0x%08lx",
|
|
// hDlg, uiMsg, wParam, lParam);
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Handle WM_NOTIFY messages.
|
|
*
|
|
* You can indicate displeasure with the CDN_* messages by using SetWindowLong
|
|
* to alter the DWL_MSGRESULT value.
|
|
*/
|
|
UINT
|
|
SelectFilesDialog::HandleNotify(HWND hDlg, LPOFNOTIFY pofn)
|
|
{
|
|
// int count;
|
|
|
|
switch (pofn->hdr.code) {
|
|
case CDN_INITDONE:
|
|
MyOnInitDone();
|
|
return 1;
|
|
case CDN_SELCHANGE:
|
|
LOGI(" CDN_SELCHANGE");
|
|
MyOnFileNameChange(/*&count*/);
|
|
//ClearFileName();
|
|
return 1;
|
|
case CDN_FOLDERCHANGE:
|
|
LOGI(" CDN_FOLDERCHANGE");
|
|
break;
|
|
case CDN_SHAREVIOLATION:
|
|
LOGI(" CDN_SHAREVIOLATION");
|
|
break;
|
|
case CDN_HELP:
|
|
LOGI(" CDN_HELP!");
|
|
break;
|
|
case CDN_FILEOK:
|
|
LOGI(" CDN_FILEOK");
|
|
/* act like they hit the Accept button */
|
|
// MyOnFileNameChange(&count);
|
|
//ClearFileName();
|
|
// if (count != 0) {
|
|
// LOGI("Count = %d, accepting CDN_FILEOK", count);
|
|
// MyOnAccept();
|
|
// } else {
|
|
// OPENFILENAME* pOfn;
|
|
// pOfn = (OPENFILENAME*) GetWindowLong(hDlg, GWL_USERDATA);
|
|
// LOGI("Count=0, name='%ls'", pOfn->lpstrFile);
|
|
// }
|
|
PrepEndDialog();
|
|
/* must do this every time, or it fails in funky ways */
|
|
SetWindowLong(hDlg, DWL_MSGRESULT, 1);
|
|
return 1;
|
|
case CDN_TYPECHANGE:
|
|
LOGI(" CDN_TYPECHANGE");
|
|
break;
|
|
case CDN_INCLUDEITEM:
|
|
LOGI(" CDN_INCLUDEITEM");
|
|
default:
|
|
LOGI(" HandleNotify, code=%d, pOfn=0x%p", pofn->hdr.code, pofn);
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Handle WM_COMMAND messages.
|
|
*/
|
|
UINT
|
|
SelectFilesDialog::HandleCommand(HWND hDlg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LOGI(" HandleCommand wParam=%d lParam=0x%08lx", wParam, lParam);
|
|
|
|
if ((int) wParam == fAcceptButtonID) {
|
|
MyOnAccept();
|
|
return 1;
|
|
} else if (wParam == IDCANCEL) {
|
|
MyOnCancel();
|
|
return 1;
|
|
} else {
|
|
return MyOnCommand(wParam, lParam);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Handle WM_SIZE.
|
|
*/
|
|
UINT
|
|
SelectFilesDialog::HandleSize(HWND hDlg, UINT nType, int cx, int cy)
|
|
{
|
|
//LOGI("Dialog: old size %d,%d (ready=%d)",
|
|
// fLastWinSize.Width(), fLastWinSize.Height(), fReady);
|
|
//LOGI("Dialog: new size %d,%d", cx, cy);
|
|
|
|
// we get called once before we have a chance to initialize
|
|
if (!fReady)
|
|
return 0;
|
|
|
|
int deltaX, deltaY;
|
|
deltaX = cx - fLastWinSize.Width();
|
|
deltaY = cy - fLastWinSize.Height();
|
|
//LOGI("Delta is %d,%d", deltaX, deltaY);
|
|
|
|
ShiftControls(deltaX, 0 /*deltaY*/);
|
|
|
|
GetParent()->GetWindowRect(&fLastWinSize);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* User hit F1 or applied the '?' button to something. Our heritage is
|
|
* dubious, so use global functions to access the help file.
|
|
*/
|
|
UINT
|
|
SelectFilesDialog::HandleHelp(HWND hDlg, LPHELPINFO lpHelpInfo)
|
|
{
|
|
CWnd* pWndMain = ::AfxGetMainWnd();
|
|
CWinApp* pAppMain = ::AfxGetApp();
|
|
DWORD context = lpHelpInfo->iCtrlId;
|
|
BOOL result;
|
|
|
|
//LOGI("Handling help with context %ld", context);
|
|
result = ::WinHelp(pWndMain->m_hWnd, pAppMain->m_pszHelpFilePath,
|
|
HELP_CONTEXTPOPUP, context);
|
|
//LOGI("SFD WinHelp returned %d", result);
|
|
return TRUE; // yes, we handled it
|
|
}
|
|
|
|
|
|
/*
|
|
* When the CFileDialog finishes doing its thing, we "fix" stuff a bit.
|
|
* We can't really do this earlier, because we'd be destroying windows that
|
|
* the parent dialog wants to move.
|
|
*
|
|
* We need to shift everything up by the difference between the IDOK button
|
|
* and our "accept" button.
|
|
*/
|
|
void
|
|
SelectFilesDialog::MyOnInitDone(void)
|
|
{
|
|
LOGI("OnInitDone!");
|
|
CWnd* pParent = GetParent();
|
|
CWnd* pWnd;
|
|
CRect okRect, cancelRect, acceptRect;
|
|
int vertDiff;
|
|
|
|
ASSERT(pParent != NULL);
|
|
pWnd = GetDlgItem(fAcceptButtonID);
|
|
ASSERT(pWnd != NULL);
|
|
pWnd->GetWindowRect(&acceptRect);
|
|
|
|
pWnd = pParent->GetDlgItem(IDOK);
|
|
ASSERT(pWnd != NULL);
|
|
pWnd->GetWindowRect(&okRect);
|
|
pWnd = pParent->GetDlgItem(IDCANCEL);
|
|
ASSERT(pWnd != NULL);
|
|
pWnd->GetWindowRect(&cancelRect);
|
|
|
|
vertDiff = acceptRect.top - okRect.top;
|
|
LOGI("vertDiff = %d (horizDiff=%d)", vertDiff,
|
|
acceptRect.left - okRect.left);
|
|
|
|
ShiftControls(0, -vertDiff);
|
|
|
|
// DestroyItem(pParent, stc3); // "File name"
|
|
// DestroyItem(pParent, edt1); // (file name edit)
|
|
DestroyItem(pParent, stc2); // "Files of type"
|
|
DestroyItem(pParent, cmb1); // (file type combo)
|
|
DestroyItem(pParent, IDOK); // "Open"/"Save"
|
|
DestroyItem(pParent, IDCANCEL); // "Cancel"
|
|
DestroyItem(this, stc32); // our placeholder
|
|
|
|
pParent->GetWindowRect(&fLastWinSize);
|
|
fLastWinSize.bottom -= vertDiff;
|
|
pParent->MoveWindow(&fLastWinSize);
|
|
|
|
// let sub-classes initialize the data fields
|
|
MyDataExchange(false);
|
|
|
|
fReady = true;
|
|
}
|
|
|
|
/*
|
|
* Shift the controls when the window size changes. This is a bit tricky
|
|
* because the CFileDialog is also moving the controls, though it doesn't
|
|
* move them in quite the way we want.
|
|
*/
|
|
void
|
|
SelectFilesDialog::ShiftControls(int deltaX, int deltaY)
|
|
{
|
|
if (deltaX == 0 && deltaY == 0) {
|
|
LOGI("SFD OnSize: no meaningful change");
|
|
return;
|
|
} else {
|
|
LOGI("ShiftControls x=%d y=%d", deltaX, deltaY);
|
|
}
|
|
MoveControl(this, fAcceptButtonID, deltaX, deltaY, false);
|
|
MoveControl(this, IDCANCEL, deltaX, deltaY, false);
|
|
//StretchControl(this, IDC_FVIEW_EDITBOX, deltaX, deltaY);
|
|
|
|
// erase & redraw
|
|
Invalidate(true);
|
|
}
|
|
|
|
|
|
/*
|
|
* Get the list view control out of the common file dialog.
|
|
*
|
|
* Returns "NULL" if it can't find it.
|
|
*/
|
|
CWnd*
|
|
SelectFilesDialog::GetListCtrl(void)
|
|
{
|
|
CWnd* pItem;
|
|
CWnd* pList;
|
|
|
|
/* our dialog is a child; get our parent, then grab the shellview */
|
|
pItem = GetParent()->GetDlgItem(lst2);
|
|
ASSERT(pItem != NULL);
|
|
if (pItem == NULL)
|
|
return NULL;
|
|
|
|
/* pull the listview out of the shellview */
|
|
pList = pItem->GetDlgItem( 1);
|
|
ASSERT(pList != NULL);
|
|
|
|
return pList;
|
|
}
|
|
|
|
/*
|
|
* When the selection changes, update our dialog.
|
|
*/
|
|
void
|
|
SelectFilesDialog::MyOnFileNameChange(void)
|
|
{
|
|
//LOGI("OnFileNameChange");
|
|
|
|
CListCtrl* pList;
|
|
|
|
pList = (CListCtrl*) GetListCtrl();
|
|
if (pList == NULL) {
|
|
LOGI("GLITCH: could not get list control");
|
|
return;
|
|
}
|
|
ASSERT(pList != NULL);
|
|
|
|
//LOGI("Selected count=%d", pList->GetSelectedCount());
|
|
//*pCount = pList->GetSelectedCount();
|
|
|
|
//CWnd* pItem;
|
|
//pItem = GetDlgItem(IDC_SELECT_ACCEPT);
|
|
//ASSERT(pItem != NULL);
|
|
//pItem->EnableWindow(*pCount != 0);
|
|
}
|
|
|
|
/*
|
|
* The user hit the "Accept" button. Package up the file selection.
|
|
*/
|
|
void
|
|
SelectFilesDialog::MyOnAccept(void)
|
|
{
|
|
//LOGI("OnAccept!");
|
|
PrepEndDialog();
|
|
}
|
|
|
|
/*
|
|
* Either the user hit the "Accept" button or the OFN dialog has indicated
|
|
* that it wants to close.
|
|
*
|
|
* Returns "true" if all went well, "false" if it failed (e.g. because the
|
|
* user hasn't selected any files).
|
|
*/
|
|
bool
|
|
SelectFilesDialog::PrepEndDialog(void)
|
|
{
|
|
CListCtrl* pList;
|
|
int nextSpot = 0;
|
|
|
|
// let sub-classes copy data out
|
|
if (!MyDataExchange(true)) {
|
|
LOGI("MyDataExchange failed!");
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
* Start with the set of stuff that the window wants to tell us about.
|
|
* Run through the list, converting all '\0' to '\\'. Later on we
|
|
* convert them back.
|
|
*
|
|
* nFileOffset will be zero if they click on "accept" instead of hitting
|
|
* return in the edit box or double-clicking on files. This can make it
|
|
* tricky to have names in the text field and files selected, because
|
|
* clicking rather than hitting "enter" will yield different results.
|
|
*
|
|
* The OFN dialog does add names to the box as they are selected, which
|
|
* makes it awkward to do anything reasonable.
|
|
*
|
|
* Fortunately I believe the world is divided into "typers" and
|
|
* "clickers", and so long as their paths don't cross we're fine.
|
|
*/
|
|
LOGI("PrepEndDialog: got max=%d off=%d", m_ofn.nMaxFile, m_ofn.nFileOffset);
|
|
if (m_ofn.nFileOffset != 0) {
|
|
WCHAR* buf = m_ofn.lpstrFile;
|
|
buf += m_ofn.nFileOffset;
|
|
while (*buf != '\0') {
|
|
if (buf > m_ofn.lpstrFile)
|
|
*(buf-1) = '\\';
|
|
LOGI(" File '%ls'", buf);
|
|
buf += wcslen(buf) +1;
|
|
}
|
|
//Sleep(1000);
|
|
nextSpot = (buf - m_ofn.lpstrFile) -1;
|
|
ASSERT(*(m_ofn.lpstrFile + nextSpot) == '\0');
|
|
ASSERT(*(m_ofn.lpstrFile + nextSpot+1) == '\0');
|
|
|
|
/* stick a '\' on the very end, so we get double-null action later */
|
|
*(m_ofn.lpstrFile + nextSpot) = '\\';
|
|
}
|
|
LOGI("Last offset was %d", nextSpot);
|
|
|
|
#if 0
|
|
/* make it clear that they're only getting one */
|
|
/* (bad case: click on some files and hit "Accept") */
|
|
if (nextSpot == 0) {
|
|
CWnd* pEditWnd;
|
|
CString editText;
|
|
pEditWnd = GetParent()->GetDlgItem(edt1);
|
|
pEditWnd->GetWindowText(editText);
|
|
if (!editText.IsEmpty()) {
|
|
pEditWnd->SetWindowText("");
|
|
return false;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Now merge in the selected files.
|
|
*/
|
|
pList = (CListCtrl*) GetListCtrl();
|
|
if (pList == NULL) {
|
|
LOGI("GLITCH: could not get list control");
|
|
return false;
|
|
}
|
|
ASSERT(pList != NULL);
|
|
|
|
CString fileNames;
|
|
|
|
int count = pList->GetSelectedCount();
|
|
if (count == 0) {
|
|
if (nextSpot == 0) {
|
|
MessageBox(L"Please select one or more files and directories.",
|
|
m_ofn.lpstrTitle, MB_OK | MB_ICONWARNING);
|
|
/* make it clear that we're ignoring the names they typed */
|
|
ClearFileName();
|
|
return false;
|
|
}
|
|
|
|
/* nothing but typed-in names */
|
|
fileNames = m_ofn.lpstrFile;
|
|
fFileNameOffset = m_ofn.nFileOffset;
|
|
} else {
|
|
bool compare;
|
|
if (nextSpot == 0) {
|
|
fileNames = GetFolderPath();
|
|
/* add a trailing '\', which gets stomped to '\0' later on */
|
|
if (fileNames.Right(1) != L"\\")
|
|
fileNames += L"\\";
|
|
fFileNameOffset = fileNames.GetLength();
|
|
compare = false;
|
|
} else {
|
|
fileNames = m_ofn.lpstrFile;
|
|
ASSERT(fileNames.Right(1) == L"\\");
|
|
fFileNameOffset = m_ofn.nFileOffset;
|
|
compare = true;
|
|
}
|
|
ASSERT(fFileNameOffset > 0);
|
|
|
|
POSITION posn;
|
|
posn = pList->GetFirstSelectedItemPosition();
|
|
if (posn == NULL) {
|
|
/* shouldn't happen -- Accept button should be dimmed */
|
|
ASSERT(false);
|
|
return false;
|
|
}
|
|
while (posn != NULL) {
|
|
/* do this every time, because "fileNames" can be reallocated */
|
|
const WCHAR* tailStr = fileNames;
|
|
tailStr += fFileNameOffset-1;
|
|
|
|
int num = pList->GetNextSelectedItem(posn); // posn is updated
|
|
|
|
/* here we make a big assumption: that GetItemData returns a PIDL */
|
|
LPITEMIDLIST pidl;
|
|
WCHAR buf[MAX_PATH];
|
|
pidl = (LPITEMIDLIST) pList->GetItemData(num);
|
|
if (SHGetPathFromIDList(pidl, buf)) {
|
|
/* it's a relative PIDL but SHGetPathFromIDList wants a full
|
|
one, so it returns CWD + filename... strip bogus path off */
|
|
CString compareName(L"\\");
|
|
PathName path(buf);
|
|
compareName += path.GetFileName();
|
|
compareName += L"\\";
|
|
//LOGI(" Checking name='%ls'", compareName);
|
|
|
|
if (compare && Stristr(tailStr, compareName) != NULL) {
|
|
LOGI(" Matched '%ls', not adding", (LPCWSTR) compareName);
|
|
} else {
|
|
if (compare) {
|
|
LOGI(" No match on '%ls', adding", (LPCWSTR) compareName);
|
|
} else {
|
|
LOGI(" Found '%ls', adding", (LPCWSTR) compareName);
|
|
}
|
|
fileNames += path.GetFileName();
|
|
fileNames += L"\\";
|
|
}
|
|
} else {
|
|
/* expected, for things like "Control Panels" or "My Network" */
|
|
LOGI(" No path for '%ls'",
|
|
(LPCWSTR) pList->GetItemText(num, 0));
|
|
}
|
|
}
|
|
|
|
if (fileNames.GetLength() >= (int)m_ofn.nMaxFile) {
|
|
LOGW("GLITCH: excessively long file name list");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
LOGI("Final result: names at %d, len=%d, str='%ls'",
|
|
fFileNameOffset, wcslen(fileNames), (LPCWSTR) fileNames);
|
|
|
|
/*
|
|
* Null-terminate with extreme prejudice. Every filename should be
|
|
* separated with a null, and the last filename should be followed by
|
|
* two. Every component was followed by '\\', and the last one
|
|
* naturally has a null, so we should be in great shape after this.
|
|
*
|
|
* Could probably have done it entirely with CString, but I'm not sure
|
|
* how CStrings react to buffers with multiple null-terminated
|
|
* sub-strings.
|
|
*/
|
|
ASSERT(fFileNames != m_ofn.lpstrFile);
|
|
delete[] fFileNames;
|
|
fFileNames = wcsdup(fileNames);
|
|
WCHAR* cp = fFileNames;
|
|
cp += fFileNameOffset-1;
|
|
while (*cp != '\0') {
|
|
if (*cp == '\\')
|
|
*cp = '\0';
|
|
cp++;
|
|
}
|
|
//fileNames.ReleaseBuffer(-1);
|
|
|
|
fExitStatus = IDOK;
|
|
CDialog* pDialog = (CDialog*) GetParent();
|
|
pDialog->EndDialog(IDCANCEL);
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* User hit our cancel button.
|
|
*/
|
|
void
|
|
SelectFilesDialog::MyOnCancel(void)
|
|
{
|
|
fExitStatus = IDCANCEL;
|
|
CDialog* pDialog = (CDialog*) GetParent();
|
|
pDialog->EndDialog(IDCANCEL);
|
|
}
|
|
|
|
/*
|
|
* Clear the filename field.
|
|
*/
|
|
void
|
|
SelectFilesDialog::ClearFileName(void)
|
|
{
|
|
CWnd* pWnd = GetParent()->GetDlgItem(edt1);
|
|
if (pWnd != NULL)
|
|
pWnd->SetWindowText(L"");
|
|
}
|