You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1847 lines
56 KiB
1847 lines
56 KiB
#include "pch.h"
|
|
#pragma hdrstop
|
|
|
|
#include "ownerdlg.h"
|
|
#include "ownerlst.h"
|
|
#include "resource.h"
|
|
#include "uihelp.h"
|
|
#include "uiutils.h"
|
|
|
|
//
|
|
// Private message used to indicate that the owner list thread
|
|
// has completed it's work.
|
|
//
|
|
const UINT PWM_OWNERLIST_COMPLETE = WM_USER + 1;
|
|
//
|
|
// Mask bits indicating what operations are allowed for a given
|
|
// file selection set in the listview.
|
|
//
|
|
const DWORD ACTION_NONE = 0x00000000;
|
|
const DWORD ACTION_TAKEOWNERSHIP = 0x00000001;
|
|
const DWORD ACTION_MOVE = 0x00000002;
|
|
const DWORD ACTION_DELETE = 0x00000004;
|
|
const DWORD ACTION_ANY = 0x00000007;
|
|
|
|
const static DWORD rgFileOwnerDialogHelpIDs[] =
|
|
{
|
|
IDC_CMB_OWNERDLG_OWNERS, IDH_CMB_OWNERDLG_OWNERS,
|
|
IDC_LV_OWNERDLG, IDH_LV_OWNERDLG,
|
|
IDC_BTN_OWNERDLG_DELETE, IDH_BTN_OWNERDLG_DELETE,
|
|
IDC_BTN_OWNERDLG_MOVETO, IDH_BTN_OWNERDLG_MOVETO,
|
|
IDC_BTN_OWNERDLG_TAKE, IDH_BTN_OWNERDLG_TAKE,
|
|
IDC_BTN_OWNERDLG_BROWSE, IDH_BTN_OWNERDLG_BROWSE,
|
|
IDC_EDIT_OWNERDLG_MOVETO, IDH_EDIT_OWNERDLG_MOVETO,
|
|
IDC_CBX_OWNERDLG_EXCLUDEDIRS, IDH_CBX_OWNERDLG_EXCLUDEDIRS,
|
|
IDC_CBX_OWNERDLG_EXCLUDEFILES, IDH_CBX_OWNERDLG_EXCLUDEFILES,
|
|
0,0
|
|
};
|
|
|
|
|
|
CFileOwnerDialog::CFileOwnerDialog(HINSTANCE hInstance,
|
|
HWND hwndParent,
|
|
LPCTSTR pszVolumeRoot,
|
|
const CArray<IDiskQuotaUser *>& rgpOwners
|
|
) : m_hInstance(hInstance),
|
|
m_hwndParent(hwndParent),
|
|
m_hwndDlg(NULL),
|
|
m_hwndLV(NULL),
|
|
m_hwndOwnerCombo(NULL),
|
|
m_hwndEditMoveTo(NULL),
|
|
m_iLastColSorted(-1),
|
|
m_bSortAscending(true),
|
|
m_bAbort(false),
|
|
m_hOwnerListThread(NULL),
|
|
m_rgpOwners(rgpOwners),
|
|
m_strVolumeRoot(pszVolumeRoot)
|
|
{
|
|
}
|
|
|
|
CFileOwnerDialog::~CFileOwnerDialog(
|
|
void
|
|
)
|
|
{
|
|
if (NULL != m_hOwnerListThread)
|
|
{
|
|
CloseHandle(m_hOwnerListThread);
|
|
}
|
|
}
|
|
|
|
|
|
INT_PTR
|
|
CFileOwnerDialog::Run(
|
|
void
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_HIGH, TEXT("CFileOwnerDialog::Run")));
|
|
return DialogBoxParam(m_hInstance,
|
|
MAKEINTRESOURCE(IDD_OWNERSANDFILES),
|
|
m_hwndParent,
|
|
DlgProc,
|
|
(LPARAM)this);
|
|
}
|
|
|
|
|
|
INT_PTR CALLBACK
|
|
CFileOwnerDialog::DlgProc(
|
|
HWND hwnd,
|
|
UINT uMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam
|
|
)
|
|
{
|
|
//
|
|
// Retrieve the dialog object's ptr from the window's userdata.
|
|
// Place there in response to WM_INITDIALOG.
|
|
//
|
|
CFileOwnerDialog *pThis = (CFileOwnerDialog *)GetWindowLongPtr(hwnd, DWLP_USER);
|
|
|
|
try
|
|
{
|
|
switch(uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
//
|
|
// Store "this" ptr in window's userdata.
|
|
//
|
|
SetWindowLongPtr(hwnd, DWLP_USER, (INT_PTR)lParam);
|
|
pThis = (CFileOwnerDialog *)lParam;
|
|
//
|
|
// Save the HWND in our object. We'll need it later.
|
|
//
|
|
pThis->m_hwndDlg = hwnd;
|
|
return pThis->OnInitDialog(hwnd);
|
|
|
|
case WM_DESTROY:
|
|
return pThis->OnDestroy(hwnd);
|
|
|
|
case WM_COMMAND:
|
|
return pThis->OnCommand(hwnd, wParam, lParam);
|
|
|
|
case WM_NOTIFY:
|
|
return pThis->OnNotify(hwnd, wParam, lParam);
|
|
|
|
case WM_CONTEXTMENU:
|
|
return pThis->OnContextMenu((HWND)wParam, LOWORD(lParam), HIWORD(lParam));
|
|
break;
|
|
|
|
case PWM_OWNERLIST_COMPLETE:
|
|
pThis->OnOwnerListComplete();
|
|
break;
|
|
|
|
case WM_SETCURSOR:
|
|
return pThis->OnSetCursor(hwnd);
|
|
break;
|
|
|
|
case WM_HELP:
|
|
WinHelp((HWND)((LPHELPINFO) lParam)->hItemHandle, STR_DSKQUOUI_HELPFILE,
|
|
HELP_WM_HELP, (DWORD_PTR)(LPTSTR) rgFileOwnerDialogHelpIDs);
|
|
return TRUE;
|
|
}
|
|
}
|
|
catch(CAllocException& me)
|
|
{
|
|
//
|
|
// Announce any out-of-memory errors associated with running the dlg.
|
|
//
|
|
DiskQuotaMsgBox(GetDesktopWindow(),
|
|
IDS_OUTOFMEMORY,
|
|
IDS_TITLE_DISK_QUOTA,
|
|
MB_ICONERROR | MB_OK);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
INT_PTR
|
|
CFileOwnerDialog::OnInitDialog(
|
|
HWND hwnd
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_HIGH, TEXT("CFileOwnerDialog::OnInitDialog")));
|
|
|
|
//
|
|
// Save HWNDs of controls we'll need later.
|
|
//
|
|
m_hwndLV = GetDlgItem(hwnd, IDC_LV_OWNERDLG);
|
|
m_hwndOwnerCombo = GetDlgItem(hwnd, IDC_CMB_OWNERDLG_OWNERS);
|
|
m_hwndEditMoveTo = GetDlgItem(hwnd, IDC_EDIT_OWNERDLG_MOVETO);
|
|
//
|
|
// We want these controls to be disabled until the owner list
|
|
// has been populated.
|
|
//
|
|
EnableWindow(m_hwndLV, FALSE);
|
|
EnableWindow(m_hwndOwnerCombo, FALSE);
|
|
EnableWindow(m_hwndEditMoveTo, FALSE);
|
|
EnableWindow(GetDlgItem(hwnd, IDC_BTN_OWNERDLG_BROWSE), FALSE);
|
|
EnableWindow(GetDlgItem(hwnd, IDC_CBX_OWNERDLG_EXCLUDEDIRS), FALSE);
|
|
EnableWindow(GetDlgItem(hwnd, IDC_CBX_OWNERDLG_EXCLUDEFILES), FALSE);
|
|
//
|
|
// Build the list of owners and filenames on the volume.
|
|
// This can take a while depending on how many owners
|
|
// are in m_rgpOwners, the size of the volume and how many
|
|
// files each owner owns. First clear the owner list in case Run()
|
|
// is being called multiple times on the same dialog object.
|
|
// Note that we build this list on a background thread so that
|
|
// we don't block the UI while it's building.
|
|
//
|
|
m_OwnerList.Clear();
|
|
DWORD idThread;
|
|
m_hOwnerListThread = CreateThread(NULL,
|
|
0,
|
|
OwnerListThreadProc,
|
|
this,
|
|
0,
|
|
&idThread);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
INT_PTR
|
|
CFileOwnerDialog::OnSetCursor(
|
|
HWND hwnd
|
|
)
|
|
{
|
|
if (m_hOwnerListThread && (WAIT_TIMEOUT == WaitForSingleObject(m_hOwnerListThread, 0)))
|
|
{
|
|
SetCursor(LoadCursor(NULL, IDC_WAIT));
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
INT_PTR
|
|
CFileOwnerDialog::OnDestroy(
|
|
HWND hwnd
|
|
)
|
|
{
|
|
m_bAbort = true;
|
|
if (NULL != m_hOwnerListThread)
|
|
{
|
|
WaitForSingleObject(m_hOwnerListThread, INFINITE);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
DWORD
|
|
CFileOwnerDialog::OwnerListThreadProc( // [static]
|
|
LPVOID pvParam
|
|
)
|
|
{
|
|
CFileOwnerDialog *pThis = (CFileOwnerDialog *)pvParam;
|
|
|
|
pThis->BuildFileOwnerList(pThis->m_strVolumeRoot,
|
|
pThis->m_rgpOwners,
|
|
&(pThis->m_OwnerList));
|
|
|
|
PostMessage(pThis->m_hwndDlg, PWM_OWNERLIST_COMPLETE, 0, 0);
|
|
return 0;
|
|
}
|
|
|
|
//
|
|
// Called in response to PWM_OWNERLIST_COMPLETE which is posted
|
|
// to the dialog when the OwnerListThreadProc has completed
|
|
// it's processing.
|
|
//
|
|
void
|
|
CFileOwnerDialog::OnOwnerListComplete(
|
|
void
|
|
)
|
|
{
|
|
//
|
|
// Set the message in the top of the dialog.
|
|
//
|
|
CString s(m_hInstance, IDS_FMT_OWNERDLG_HEADER, m_OwnerList.OwnerCount());
|
|
SetWindowText(GetDlgItem(m_hwndDlg, IDC_TXT_OWNERDLG_HEADER), s);
|
|
//
|
|
// Populate the listview and owner combo.
|
|
//
|
|
InitializeList(m_OwnerList, m_hwndLV);
|
|
InitializeOwnerCombo(m_OwnerList, m_hwndOwnerCombo);
|
|
//
|
|
// Now we can enable the controls we disabled in OnInitDialog().
|
|
//
|
|
EnableWindow(m_hwndLV, TRUE);
|
|
EnableWindow(m_hwndOwnerCombo, TRUE);
|
|
EnableWindow(m_hwndEditMoveTo, TRUE);
|
|
EnableWindow(GetDlgItem(m_hwndDlg, IDC_BTN_OWNERDLG_BROWSE), TRUE);
|
|
EnableWindow(GetDlgItem(m_hwndDlg, IDC_CBX_OWNERDLG_EXCLUDEDIRS), TRUE);
|
|
EnableWindow(GetDlgItem(m_hwndDlg, IDC_CBX_OWNERDLG_EXCLUDEFILES), TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
INT_PTR
|
|
CFileOwnerDialog::OnCommand(
|
|
HWND hwnd,
|
|
WPARAM wParam,
|
|
LPARAM lParam
|
|
)
|
|
{
|
|
BOOL bResult = TRUE; // Assume not handled.
|
|
WORD wID = LOWORD(wParam);
|
|
WORD wNotifyCode = HIWORD(wParam);
|
|
HWND hCtl = (HWND)lParam;
|
|
|
|
switch(wID)
|
|
{
|
|
case IDCANCEL:
|
|
EndDialog(hwnd, 0);
|
|
bResult = FALSE;
|
|
break;
|
|
|
|
case IDC_CMB_OWNERDLG_OWNERS:
|
|
if (CBN_SELCHANGE == wNotifyCode)
|
|
{
|
|
int iOwner = ComboBox_GetCurSel(m_hwndOwnerCombo);
|
|
if (1 < m_OwnerList.OwnerCount())
|
|
{
|
|
//
|
|
// Owner list contains more than one owner. The combo
|
|
// contains a leading "All owners" entry.
|
|
//
|
|
iOwner--;
|
|
}
|
|
|
|
DBGASSERT((-1 <= iOwner));
|
|
CAutoSetRedraw autoredraw(m_hwndLV, false);
|
|
//
|
|
// Only show "owner" column if user has selected "all owners" combo item.
|
|
//
|
|
CreateListColumns(m_hwndLV, -1 == iOwner);
|
|
FillListView(m_OwnerList, m_hwndLV, iOwner);
|
|
}
|
|
bResult = FALSE;
|
|
break;
|
|
|
|
case IDC_BTN_OWNERDLG_BROWSE:
|
|
{
|
|
CString s;
|
|
if (BrowseForFolder(hwnd, &s))
|
|
SetWindowText(m_hwndEditMoveTo, s);
|
|
break;
|
|
}
|
|
|
|
case IDC_BTN_OWNERDLG_DELETE:
|
|
DeleteSelectedFiles(m_hwndLV);
|
|
bResult = FALSE;
|
|
break;
|
|
|
|
case IDC_BTN_OWNERDLG_MOVETO:
|
|
{
|
|
CPath strDest;
|
|
CPath strRoot;
|
|
int cchEdit = Edit_GetTextLength(m_hwndEditMoveTo);
|
|
Edit_GetText(m_hwndEditMoveTo,
|
|
strDest.GetBuffer(cchEdit + 1),
|
|
cchEdit + 1);
|
|
strDest.ReleaseBuffer();
|
|
strDest.Trim();
|
|
strDest.GetRoot(&strRoot);
|
|
|
|
HRESULT hr = IsSameVolume(strRoot, m_strVolumeRoot);
|
|
if (S_OK == hr)
|
|
{
|
|
//
|
|
// Don't let operator move files to a folder
|
|
// on the same volume.
|
|
//
|
|
DiskQuotaMsgBox(m_hwndDlg,
|
|
IDS_ERROR_MOVETO_SAMEVOL,
|
|
IDS_TITLE_DISK_QUOTA,
|
|
MB_ICONINFORMATION | MB_OK);
|
|
|
|
SetWindowText(m_hwndEditMoveTo, strDest);
|
|
SetFocus(m_hwndEditMoveTo);
|
|
}
|
|
else if (S_FALSE == hr)
|
|
{
|
|
//
|
|
// Eveything looks OK. Try to move the files.
|
|
//
|
|
MoveSelectedFiles(m_hwndLV, strDest);
|
|
}
|
|
else
|
|
{
|
|
DBGERROR((TEXT("TakeOwnershipOfSelectedFiles failed with hr = 0x%08X"), hr));
|
|
}
|
|
bResult = FALSE;
|
|
break;
|
|
}
|
|
|
|
case IDC_BTN_OWNERDLG_TAKE:
|
|
{
|
|
HRESULT hr = TakeOwnershipOfSelectedFiles(m_hwndLV);
|
|
if (FAILED(hr))
|
|
{
|
|
DBGERROR((TEXT("TakeOwnershipOfSelectedFiles failed with hr = 0x%08X"), hr));
|
|
}
|
|
break;
|
|
}
|
|
|
|
case IDC_EDIT_OWNERDLG_MOVETO:
|
|
if (EN_UPDATE == wNotifyCode)
|
|
{
|
|
//
|
|
// Disable the "Move" button if the destination edit field
|
|
// is blank.
|
|
//
|
|
HWND hwnd = GetDlgItem(m_hwndDlg, IDC_BTN_OWNERDLG_MOVETO);
|
|
bool bEnable = ShouldEnableControl(IDC_BTN_OWNERDLG_MOVETO);
|
|
if (bEnable != boolify(IsWindowEnabled(hwnd)))
|
|
{
|
|
EnableWindow(hwnd, bEnable);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case IDC_CBX_OWNERDLG_EXCLUDEFILES:
|
|
case IDC_CBX_OWNERDLG_EXCLUDEDIRS:
|
|
if (BN_CLICKED == wNotifyCode)
|
|
{
|
|
//
|
|
// The allowable states for these two checkboxes are:
|
|
//
|
|
// Excl Files Excl Dirs
|
|
// --------------- ----------------
|
|
// Checked Unchecked
|
|
// Unchecked Checked
|
|
// Unchecked Unchecked
|
|
//
|
|
// It makes no sense to have both checkboxes checked.
|
|
// This would cause the list to be empty and might
|
|
// generate user confusion.
|
|
//
|
|
if (IsDlgButtonChecked(m_hwndDlg, wID))
|
|
{
|
|
UINT idOther = IDC_CBX_OWNERDLG_EXCLUDEFILES;
|
|
if (IDC_CBX_OWNERDLG_EXCLUDEFILES == wID)
|
|
{
|
|
idOther = IDC_CBX_OWNERDLG_EXCLUDEDIRS;
|
|
}
|
|
CheckDlgButton(m_hwndDlg, idOther, BST_UNCHECKED);
|
|
}
|
|
FillListView(m_OwnerList, m_hwndLV, ComboBox_GetCurSel(m_hwndOwnerCombo) - 1);
|
|
}
|
|
break;
|
|
}
|
|
return bResult;
|
|
}
|
|
|
|
|
|
INT_PTR
|
|
CFileOwnerDialog::OnContextMenu(
|
|
HWND hwndItem,
|
|
int xPos,
|
|
int yPos
|
|
)
|
|
{
|
|
int idCtl = GetDlgCtrlID(hwndItem);
|
|
WinHelp(hwndItem,
|
|
UseWindowsHelp(idCtl) ? NULL : STR_DSKQUOUI_HELPFILE,
|
|
HELP_CONTEXTMENU,
|
|
(DWORD_PTR)((LPTSTR)rgFileOwnerDialogHelpIDs));
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
//
|
|
// Determine what actions are allowed for the current selection.
|
|
//
|
|
DWORD
|
|
CFileOwnerDialog::GetAllowedActions(
|
|
HWND hwndLV
|
|
)
|
|
{
|
|
CArray<COwnerListItemHandle> rgItemHandles;
|
|
|
|
BuildListOfSelectedFiles(hwndLV, NULL, &rgItemHandles);
|
|
if (0 != rgItemHandles.Count())
|
|
{
|
|
const int cHandles = rgItemHandles.Count();
|
|
for(int i = 0; i < cHandles; i++)
|
|
{
|
|
COwnerListItemHandle handle = rgItemHandles[i];
|
|
int iOwner = handle.OwnerIndex();
|
|
int iFile = handle.FileIndex();
|
|
if (m_OwnerList.IsFileDirectory(iOwner, iFile))
|
|
{
|
|
//
|
|
// If any directory exists in the selection,
|
|
// "take ownership" is the only allowed action.
|
|
//
|
|
return ACTION_TAKEOWNERSHIP;
|
|
}
|
|
}
|
|
}
|
|
return ACTION_ANY;
|
|
}
|
|
|
|
|
|
//
|
|
// Determine if one of the move/delete/take buttons should be enabled
|
|
// or disabled.
|
|
//
|
|
bool
|
|
CFileOwnerDialog::ShouldEnableControl(
|
|
UINT idCtl
|
|
)
|
|
{
|
|
bool bEnable = true;
|
|
int cLVSel = ListView_GetSelectedCount(m_hwndLV);
|
|
DWORD actions = GetAllowedActions(m_hwndLV);
|
|
|
|
switch(idCtl)
|
|
{
|
|
case IDC_BTN_OWNERDLG_DELETE:
|
|
bEnable = (0 != (ACTION_DELETE & actions)) && (0 < cLVSel);
|
|
break;
|
|
|
|
case IDC_BTN_OWNERDLG_TAKE:
|
|
bEnable = (0 != (ACTION_TAKEOWNERSHIP & actions)) && (0 < cLVSel);
|
|
break;
|
|
|
|
case IDC_BTN_OWNERDLG_MOVETO:
|
|
bEnable = (0 != (ACTION_MOVE & actions));
|
|
if (bEnable && 0 < cLVSel)
|
|
{
|
|
CPath s;
|
|
int cch = Edit_GetTextLength(m_hwndEditMoveTo);
|
|
Edit_GetText(m_hwndEditMoveTo, s.GetBuffer(cch + 1), cch + 1);
|
|
s.ReleaseBuffer();
|
|
s.Trim();
|
|
bEnable = 0 < s.Length();
|
|
}
|
|
break;
|
|
|
|
case IDC_BTN_OWNERDLG_BROWSE:
|
|
case IDC_EDIT_OWNERDLG_MOVETO:
|
|
bEnable = (0 != (ACTION_MOVE & actions));
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
return bEnable;
|
|
}
|
|
|
|
|
|
INT_PTR
|
|
CFileOwnerDialog::OnNotify(
|
|
HWND hwnd,
|
|
WPARAM wParam,
|
|
LPARAM lParam
|
|
)
|
|
{
|
|
BOOL bResult = TRUE;
|
|
LPNMHDR pnm = (LPNMHDR)lParam;
|
|
|
|
switch(pnm->code)
|
|
{
|
|
case LVN_GETDISPINFO:
|
|
OnLVN_GetDispInfo((LV_DISPINFO *)lParam);
|
|
break;
|
|
|
|
case LVN_COLUMNCLICK:
|
|
OnLVN_ColumnClick((NM_LISTVIEW *)lParam);
|
|
break;
|
|
|
|
case LVN_ITEMCHANGED:
|
|
OnLVN_ItemChanged((NM_LISTVIEW *)lParam);
|
|
break;
|
|
|
|
case LVN_KEYDOWN:
|
|
OnLVN_KeyDown((NMLVKEYDOWN *)lParam);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
|
|
void
|
|
CFileOwnerDialog::OnLVN_GetDispInfo(
|
|
LV_DISPINFO *plvdi
|
|
)
|
|
{
|
|
static CPath strPath;
|
|
static CString strOwner;
|
|
|
|
COwnerListItemHandle hItem(plvdi->item.lParam);
|
|
int iOwner = hItem.OwnerIndex();
|
|
int iFile = hItem.FileIndex();
|
|
|
|
if (LVIF_TEXT & plvdi->item.mask)
|
|
{
|
|
switch(plvdi->item.iSubItem)
|
|
{
|
|
case iLVSUBITEM_FILE:
|
|
{
|
|
CPath s;
|
|
m_OwnerList.GetFileName(iOwner, iFile, &s);
|
|
if (m_OwnerList.IsFileDirectory(iOwner, iFile))
|
|
{
|
|
strPath.Format(m_hInstance, IDS_FMT_OWNERDLG_FOLDERNAME, s.Cstr());
|
|
}
|
|
else
|
|
{
|
|
strPath = s;
|
|
}
|
|
plvdi->item.pszText = (LPTSTR)strPath.Cstr();
|
|
}
|
|
break;
|
|
|
|
case iLVSUBITEM_FOLDER:
|
|
m_OwnerList.GetFolderName(iOwner, iFile, &strPath);
|
|
plvdi->item.pszText = (LPTSTR)strPath.Cstr();
|
|
break;
|
|
|
|
case iLVSUBITEM_OWNER:
|
|
m_OwnerList.GetOwnerName(iOwner, &strOwner);
|
|
plvdi->item.pszText = (LPTSTR)strOwner.Cstr();
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (LVIF_IMAGE & plvdi->item.mask)
|
|
{
|
|
//
|
|
// Not displaying any images. This is just a placeholder.
|
|
// Should be optimized out by compiler.
|
|
//
|
|
}
|
|
}
|
|
|
|
|
|
int CALLBACK
|
|
CFileOwnerDialog::CompareLVItems(
|
|
LPARAM lParam1,
|
|
LPARAM lParam2,
|
|
LPARAM lParamSort
|
|
)
|
|
{
|
|
CFileOwnerDialog *pdlg = reinterpret_cast<CFileOwnerDialog *>(lParamSort);
|
|
int diff = 0;
|
|
try
|
|
{
|
|
COwnerListItemHandle h1(lParam1);
|
|
COwnerListItemHandle h2(lParam2);
|
|
int iOwner1 = h1.OwnerIndex();
|
|
int iOwner2 = h2.OwnerIndex();
|
|
int iFile1 = h1.FileIndex();
|
|
int iFile2 = h2.FileIndex();
|
|
static CPath s1, s2;
|
|
|
|
//
|
|
// This array controls the comparison column IDs used when
|
|
// values for the selected column are equal. These should
|
|
// remain in order of the iLVSUBITEM_xxxxx enumeration with
|
|
// respect to the first element in each row.
|
|
//
|
|
static const int rgColComp[3][3] = {
|
|
{ iLVSUBITEM_FILE, iLVSUBITEM_FOLDER, iLVSUBITEM_OWNER },
|
|
{ iLVSUBITEM_FOLDER, iLVSUBITEM_FILE, iLVSUBITEM_OWNER },
|
|
{ iLVSUBITEM_OWNER, iLVSUBITEM_FILE, iLVSUBITEM_FOLDER }
|
|
};
|
|
int iCompare = 0;
|
|
while(0 == diff && iCompare < ARRAYSIZE(rgColComp))
|
|
{
|
|
switch(rgColComp[pdlg->m_iLastColSorted][iCompare++])
|
|
{
|
|
case iLVSUBITEM_FILE:
|
|
pdlg->m_OwnerList.GetFileName(iOwner1, iFile1, &s1);
|
|
pdlg->m_OwnerList.GetFileName(iOwner2, iFile2, &s2);
|
|
break;
|
|
|
|
case iLVSUBITEM_FOLDER:
|
|
pdlg->m_OwnerList.GetFolderName(iOwner1, iFile1, &s1);
|
|
pdlg->m_OwnerList.GetFolderName(iOwner2, iFile2, &s2);
|
|
break;
|
|
|
|
case iLVSUBITEM_OWNER:
|
|
//
|
|
// Can use CPath (s1 and s2) in place of CString arg since
|
|
// CPath is derived from CString.
|
|
//
|
|
pdlg->m_OwnerList.GetOwnerName(iOwner1, &s1);
|
|
pdlg->m_OwnerList.GetOwnerName(iOwner2, &s2);
|
|
break;
|
|
|
|
default:
|
|
//
|
|
// If you hit this, you need to update this function
|
|
// to handle the new column you've added to the listview.
|
|
//
|
|
DBGASSERT((false));
|
|
break;
|
|
}
|
|
diff = s1.Compare(s2);
|
|
}
|
|
//
|
|
// Don't need contents of static strings between function invocations.
|
|
// The strings are static to avoid repeated construction/destruction.
|
|
// It's only a minor optimization.
|
|
//
|
|
s1.Empty();
|
|
s2.Empty();
|
|
}
|
|
catch(CAllocException& e)
|
|
{
|
|
//
|
|
// Do nothing. Just return diff "as is".
|
|
// Don't want to throw an exception back into comctl32.
|
|
//
|
|
}
|
|
return pdlg->m_bSortAscending ? diff : -1 * diff;
|
|
}
|
|
|
|
|
|
void
|
|
CFileOwnerDialog::OnLVN_ColumnClick(
|
|
NM_LISTVIEW *pnmlv
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_LOW, TEXT("CFileOwnerDialog::OnLVN_ColumnClick")));
|
|
|
|
if (m_iLastColSorted != pnmlv->iSubItem)
|
|
{
|
|
m_bSortAscending = true;
|
|
m_iLastColSorted = pnmlv->iSubItem;
|
|
}
|
|
else
|
|
{
|
|
m_bSortAscending = !m_bSortAscending;
|
|
}
|
|
|
|
ListView_SortItems(m_hwndLV, CompareLVItems, LPARAM(this));
|
|
}
|
|
|
|
|
|
//
|
|
// Called whenever a listview item has changed state.
|
|
// I'm using this to update the "enabledness" of the
|
|
// dialog buttons. If there's nothing selected in the listview,
|
|
// the move/delete/take buttons are disabled.
|
|
//
|
|
void
|
|
CFileOwnerDialog::OnLVN_ItemChanged(
|
|
NM_LISTVIEW *pnmlv
|
|
)
|
|
{
|
|
static const int rgCtls[] = { IDC_BTN_OWNERDLG_DELETE,
|
|
IDC_BTN_OWNERDLG_TAKE,
|
|
IDC_BTN_OWNERDLG_MOVETO,
|
|
IDC_BTN_OWNERDLG_BROWSE,
|
|
IDC_EDIT_OWNERDLG_MOVETO};
|
|
|
|
//
|
|
// LVN_ITEMCHANGED is sent multiple times when you move the
|
|
// highlight bar in a listview.
|
|
// Only run this code when the "focused" state bit is set
|
|
// for the "new state". This should be the last call in
|
|
// the series.
|
|
//
|
|
if (LVIS_FOCUSED & pnmlv->uNewState)
|
|
{
|
|
for (int i = 0; i < ARRAYSIZE(rgCtls); i++)
|
|
{
|
|
HWND hwnd = GetDlgItem(m_hwndDlg, rgCtls[i]);
|
|
bool bEnable = ShouldEnableControl(rgCtls[i]);
|
|
if (bEnable != boolify(IsWindowEnabled(hwnd)))
|
|
{
|
|
EnableWindow(hwnd, bEnable);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
CFileOwnerDialog::OnLVN_KeyDown(
|
|
NMLVKEYDOWN *plvkd
|
|
)
|
|
{
|
|
if (VK_DELETE == plvkd->wVKey)
|
|
{
|
|
DeleteSelectedFiles(m_hwndLV);
|
|
FocusOnSomethingInListview(m_hwndLV);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void
|
|
CFileOwnerDialog::FocusOnSomethingInListview(
|
|
HWND hwndLV
|
|
)
|
|
{
|
|
//
|
|
// Focus on something.
|
|
//
|
|
int iFocus = ListView_GetNextItem(hwndLV, -1, LVNI_FOCUSED);
|
|
if (-1 == iFocus)
|
|
iFocus = 0;
|
|
|
|
ListView_SetItemState(hwndLV, iFocus, LVIS_FOCUSED | LVIS_SELECTED,
|
|
LVIS_FOCUSED | LVIS_SELECTED);
|
|
}
|
|
|
|
|
|
//
|
|
// Creates the listview columns and populates the listview
|
|
// with filenames.
|
|
//
|
|
void
|
|
CFileOwnerDialog::InitializeList(
|
|
const COwnerList& fol, // file & owner list
|
|
HWND hwndList
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_MID, TEXT("CFileOwnerDialog::InitializeList")));
|
|
|
|
CreateListColumns(hwndList, 1 < m_OwnerList.OwnerCount());
|
|
FillListView(fol, hwndList);
|
|
ListView_SetExtendedListViewStyle(hwndList, LVS_EX_FULLROWSELECT);
|
|
}
|
|
|
|
|
|
void
|
|
CFileOwnerDialog::CreateListColumns(
|
|
HWND hwndList,
|
|
bool bShowOwner // Default is true.
|
|
)
|
|
{
|
|
//
|
|
// Clear out the listview and header.
|
|
//
|
|
ListView_DeleteAllItems(hwndList);
|
|
HWND hwndHeader = ListView_GetHeader(hwndList);
|
|
if (NULL != hwndHeader)
|
|
{
|
|
while(0 < Header_GetItemCount(hwndHeader))
|
|
ListView_DeleteColumn(hwndList, 0);
|
|
}
|
|
|
|
//
|
|
// Create the header titles.
|
|
//
|
|
CString strFile(m_hInstance, IDS_OWNERDLG_HDR_FILE);
|
|
CString strFolder(m_hInstance, IDS_OWNERDLG_HDR_FOLDER);
|
|
CString strOwner(m_hInstance, IDS_OWNERDLG_HDR_OWNER);
|
|
|
|
//
|
|
// FEATURE: Should probably allow for vertical scroll bar also.
|
|
//
|
|
RECT rcList;
|
|
GetClientRect(hwndList, &rcList);
|
|
int cxCol = (rcList.right - rcList.left) / (bShowOwner ? 3 : 2);
|
|
|
|
#define LVCOLMASK (LVCF_FMT | LVCF_TEXT | LVCF_WIDTH | LVCF_SUBITEM)
|
|
|
|
LV_COLUMN rgCols[] = {
|
|
{ LVCOLMASK, LVCFMT_LEFT, cxCol, strFile, 0, iLVSUBITEM_FILE },
|
|
{ LVCOLMASK, LVCFMT_LEFT, cxCol, strFolder, 0, iLVSUBITEM_FOLDER },
|
|
{ LVCOLMASK, LVCFMT_LEFT, cxCol, strOwner, 0, iLVSUBITEM_OWNER }
|
|
};
|
|
//
|
|
// Add the columns to the listview.
|
|
//
|
|
int cCols = bShowOwner ? ARRAYSIZE(rgCols) : ARRAYSIZE(rgCols) - 1;
|
|
for (INT i = 0; i < cCols; i++)
|
|
{
|
|
if (-1 == ListView_InsertColumn(hwndList, i, &rgCols[i]))
|
|
{
|
|
DBGERROR((TEXT("CFileOwnerDialog::CreateListColumns failed to add column %d"), i));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
CFileOwnerDialog::FillListView(
|
|
const COwnerList& fol, // file & owner list
|
|
HWND hwndList,
|
|
int iOwner // default is -1 (all owners)
|
|
)
|
|
{
|
|
ListView_DeleteAllItems(hwndList);
|
|
|
|
LV_ITEM item;
|
|
item.iSubItem = 0;
|
|
item.mask = LVIF_TEXT | LVIF_STATE | LVIF_IMAGE | LVIF_PARAM;
|
|
item.state = 0;
|
|
item.stateMask = 0;
|
|
item.pszText = LPSTR_TEXTCALLBACK;
|
|
item.iImage = I_IMAGECALLBACK;
|
|
|
|
int iFirst = iOwner;
|
|
int iLast = iOwner;
|
|
if (-1 == iOwner)
|
|
{
|
|
iFirst = 0;
|
|
iLast = fol.OwnerCount() - 1;
|
|
}
|
|
int iItem = 0;
|
|
const bool bExclFiles = IsDlgButtonChecked(m_hwndDlg, IDC_CBX_OWNERDLG_EXCLUDEFILES);
|
|
const bool bExclDirs = IsDlgButtonChecked(m_hwndDlg, IDC_CBX_OWNERDLG_EXCLUDEDIRS);
|
|
//
|
|
// WARNING: Reusing formal arg iOwner. It's safe to do, but you
|
|
// should be aware that I'm doing it.
|
|
//
|
|
for (iOwner = iFirst; iOwner <= iLast; iOwner++)
|
|
{
|
|
int cFiles = fol.FileCount(iOwner, true);
|
|
for (int iFile = 0; iFile < cFiles; iFile++)
|
|
{
|
|
bool bDirectory = fol.IsFileDirectory(iOwner, iFile);
|
|
bool bFile = !bDirectory;
|
|
|
|
if ((bDirectory && !bExclDirs) || (bFile && !bExclFiles))
|
|
{
|
|
if (!fol.IsFileDeleted(iOwner, iFile))
|
|
{
|
|
item.lParam = COwnerListItemHandle(iOwner, iFile);
|
|
item.iItem = iItem++;
|
|
if (-1 == ListView_InsertItem(hwndList, &item))
|
|
DBGERROR((TEXT("Error adding LV item for owner %d, file %d"), iOwner, iFile));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
CFileOwnerDialog::InitializeOwnerCombo(
|
|
const COwnerList& fol, // file & owner list
|
|
HWND hwndCombo
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_MID, TEXT("CFileOwnerDialog::InitializeList")));
|
|
|
|
int iSelected = ComboBox_GetCurSel(hwndCombo);
|
|
ComboBox_ResetContent(hwndCombo);
|
|
|
|
CString s, s2;
|
|
int cOwners = fol.OwnerCount();
|
|
if (1 < cOwners)
|
|
{
|
|
//
|
|
// Add "all owners" entry.
|
|
//
|
|
s.Format(m_hInstance, IDS_FMT_ALLOWNERS, fol.FileCount());
|
|
ComboBox_InsertString(hwndCombo, -1, s);
|
|
}
|
|
|
|
for (int iOwner = 0; iOwner < cOwners; iOwner++)
|
|
{
|
|
fol.GetOwnerName(iOwner, &s2);
|
|
s.Format(m_hInstance, IDS_FMT_OWNER, s2.Cstr(), fol.FileCount(iOwner));
|
|
ComboBox_InsertString(hwndCombo, -1, s);
|
|
}
|
|
|
|
ComboBox_SetCurSel(hwndCombo, CB_ERR != iSelected ? iSelected : 0);
|
|
|
|
//
|
|
// Set the max height of the owner combo
|
|
//
|
|
RECT rcCombo;
|
|
GetClientRect(m_hwndOwnerCombo, &rcCombo);
|
|
SetWindowPos(m_hwndOwnerCombo,
|
|
NULL,
|
|
0, 0,
|
|
rcCombo.right - rcCombo.left,
|
|
200,
|
|
SWP_NOMOVE | SWP_NOZORDER | SWP_NOREDRAW | SWP_NOACTIVATE);
|
|
}
|
|
|
|
|
|
//
|
|
// Determine if two volume root strings refer to the same volume.
|
|
// With volume mount points, "C:\" and "D:\DriveC" could refer to the
|
|
// same physical volume. To differentiate we need to examine the unique
|
|
// volume name GUID strings.
|
|
//
|
|
HRESULT
|
|
CFileOwnerDialog::IsSameVolume(
|
|
LPCTSTR pszRoot1,
|
|
LPCTSTR pszRoot2
|
|
)
|
|
{
|
|
TCHAR szVolGUID1[MAX_PATH];
|
|
TCHAR szTemp[MAX_PATH];
|
|
HRESULT hr = S_FALSE;
|
|
|
|
//
|
|
// GetVolumeNameForVolumeMountPoint requires trailing backslash on paths.
|
|
//
|
|
lstrcpyn(szTemp, pszRoot1, ARRAYSIZE(szTemp));
|
|
if (!PathAddBackslash(szTemp))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER);
|
|
}
|
|
else
|
|
{
|
|
if (GetVolumeNameForVolumeMountPoint(szTemp, szVolGUID1, ARRAYSIZE(szVolGUID1)))
|
|
{
|
|
TCHAR szVolGUID2[MAX_PATH];
|
|
lstrcpyn(szTemp, pszRoot2, ARRAYSIZE(szTemp));
|
|
if (!PathAddBackslash(szTemp))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER);
|
|
}
|
|
else
|
|
{
|
|
if (GetVolumeNameForVolumeMountPoint(szTemp, szVolGUID2, ARRAYSIZE(szVolGUID2)))
|
|
{
|
|
if (0 == lstrcmpi(szVolGUID1, szVolGUID2))
|
|
hr = S_OK;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Let the user browse for a folder.
|
|
// The selected folder path is returned in *pstrFolder.
|
|
//
|
|
bool
|
|
CFileOwnerDialog::BrowseForFolder(
|
|
HWND hwndParent,
|
|
CString *pstrFolder
|
|
)
|
|
{
|
|
bool bResult = false;
|
|
BROWSEINFO bi;
|
|
ZeroMemory(&bi, sizeof(bi));
|
|
|
|
CString strTitle(m_hInstance, IDS_BROWSEFORFOLDER);
|
|
|
|
bi.hwndOwner = hwndParent;
|
|
bi.pidlRoot = NULL; // Start at desktop.
|
|
bi.pszDisplayName = NULL;
|
|
bi.lpszTitle = strTitle.Cstr();
|
|
//
|
|
// FEATURE: Setting the BIF_EDITBOX flag causes SHBrowseForFolder to invoke
|
|
// autocomplete through SHAutoComplete (in shlwapi). SHAutoComplete
|
|
// loads browseui.dll to implement the autocomplete feature. The bad
|
|
// part is that SHAutoComplete also unloads browseui.dll before it
|
|
// returns, resulting in calls to the unloaded WndProc. I've notified
|
|
// ReinerF about this. Turning off the BIF_EDITBOX bit prevents
|
|
// autocomplete from being used and thus prevents the problem.
|
|
// I want the edit box. Turn it back on once they fix this bug.
|
|
//
|
|
// brianau [1/30/97]
|
|
//
|
|
bi.ulFlags = BIF_RETURNONLYFSDIRS; // | BIF_EDITBOX;
|
|
bi.lpfn = BrowseForFolderCallback;
|
|
bi.lParam = (LPARAM)pstrFolder;
|
|
bi.iImage = 0;
|
|
|
|
bResult = boolify(SHBrowseForFolder(&bi));
|
|
return bResult;
|
|
}
|
|
|
|
|
|
//
|
|
// Callback called by SHBrowseForFolder. Writes selected folder path
|
|
// to CString object who's pointer is passed in lpData arg.
|
|
//
|
|
int
|
|
CFileOwnerDialog::BrowseForFolderCallback(
|
|
HWND hwnd,
|
|
UINT uMsg,
|
|
LPARAM lParam,
|
|
LPARAM lpData
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_MID, TEXT("CFileOwnerDialog::BrowseForFolderCallback")));
|
|
CString *pstrFolder = (CString *)lpData;
|
|
|
|
if (BFFM_SELCHANGED == uMsg)
|
|
{
|
|
SHGetPathFromIDList((LPCITEMIDLIST)lParam, pstrFolder->GetBuffer(MAX_PATH));
|
|
pstrFolder->ReleaseBuffer();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
//
|
|
// Builds a double-nul terminated list of file paths from the listview
|
|
// along with an array of "item handle" objects that acts as a cross-
|
|
// reference between the list items, items in the listview and items
|
|
// in the file owner list. Each handle contains an owner index and
|
|
// file index into the file owner list. Each handle is also the value
|
|
// stored as the lParam in the listview items.
|
|
// Both pList and prgItemHandles arguments are optional. Although,
|
|
// calling with neither non-null is sort of useless.
|
|
//
|
|
void
|
|
CFileOwnerDialog::BuildListOfSelectedFiles(
|
|
HWND hwndLV,
|
|
DblNulTermList *pList,
|
|
CArray<COwnerListItemHandle> *prgItemHandles
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_MID, TEXT("CFileOwnerDialog::BuildListOfSelectedFiles")));
|
|
int iItem = -1;
|
|
CPath strPath;
|
|
LV_ITEM item;
|
|
|
|
if (NULL != prgItemHandles)
|
|
prgItemHandles->Clear();
|
|
while(-1 != (iItem = ListView_GetNextItem(hwndLV, iItem, LVNI_SELECTED)))
|
|
{
|
|
item.iSubItem = 0;
|
|
item.iItem = iItem;
|
|
item.mask = LVIF_PARAM;
|
|
if (-1 != ListView_GetItem(hwndLV, &item))
|
|
{
|
|
COwnerListItemHandle hItem(item.lParam);
|
|
m_OwnerList.GetFileFullPath(hItem.OwnerIndex(),
|
|
hItem.FileIndex(),
|
|
&strPath);
|
|
if (pList)
|
|
pList->AddString(strPath);
|
|
if (prgItemHandles)
|
|
prgItemHandles->Append(hItem);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Given an item "handle", find it's entry in the listview.
|
|
//
|
|
int
|
|
CFileOwnerDialog::FindItemFromHandle(
|
|
HWND hwndLV,
|
|
const COwnerListItemHandle& handle
|
|
)
|
|
{
|
|
LV_FINDINFO lvfi;
|
|
lvfi.flags = LVFI_PARAM;
|
|
lvfi.lParam = handle;
|
|
return ListView_FindItem(hwndLV, -1, &lvfi);
|
|
}
|
|
|
|
|
|
//
|
|
// Scans an array of item handles and removes all corresponding
|
|
// items from the listview.
|
|
//
|
|
void
|
|
CFileOwnerDialog::RemoveListViewItems(
|
|
HWND hwndLV,
|
|
const CArray<COwnerListItemHandle>& rgItemHandles
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_MID, TEXT("CFileOwnerDialog::RemoveListViewItems")));
|
|
LV_ITEM item;
|
|
CPath strPath;
|
|
|
|
CAutoSetRedraw autoredraw(hwndLV, false);
|
|
int cHandles = rgItemHandles.Count();
|
|
for (int iHandle = 0; iHandle < cHandles; iHandle++)
|
|
{
|
|
COwnerListItemHandle handle = rgItemHandles[iHandle];
|
|
int iItem = FindItemFromHandle(hwndLV, handle);
|
|
if (-1 != iItem)
|
|
{
|
|
int iOwner = handle.OwnerIndex();
|
|
int iFile = handle.FileIndex();
|
|
m_OwnerList.GetFileFullPath(iOwner, iFile, &strPath);
|
|
|
|
if ((DWORD)-1 == GetFileAttributes(strPath))
|
|
{
|
|
//
|
|
// File doesn't exist any more.
|
|
// Delete from the listview.
|
|
// Mark it as "deleted" in the ownerlist container.
|
|
//
|
|
ListView_DeleteItem(hwndLV, iItem);
|
|
m_OwnerList.MarkFileDeleted(iOwner, iFile);
|
|
DBGPRINT((DM_VIEW, DL_LOW, TEXT("Removed item %d \"%s\""),
|
|
iItem, strPath.Cstr()));
|
|
}
|
|
}
|
|
}
|
|
//
|
|
// Refresh the owner combo to update the file counts.
|
|
//
|
|
InitializeOwnerCombo(m_OwnerList, m_hwndOwnerCombo);
|
|
}
|
|
|
|
|
|
//
|
|
// Delete the files selected in the listview.
|
|
// Files deleted are removed from the listview.
|
|
//
|
|
void
|
|
CFileOwnerDialog::DeleteSelectedFiles(
|
|
HWND hwndLV
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_MID, TEXT("CFileOwnerDialog::DeleteSelectedFiles")));
|
|
DblNulTermList list(1024); // 1024 is the buffer growth size in chars.
|
|
CArray<COwnerListItemHandle> rgItemHandles;
|
|
|
|
BuildListOfSelectedFiles(hwndLV, &list, &rgItemHandles);
|
|
if (0 < list.Count())
|
|
{
|
|
SHFILEOPSTRUCT fo;
|
|
fo.hwnd = m_hwndDlg;
|
|
fo.wFunc = FO_DELETE;
|
|
fo.pFrom = list;
|
|
fo.pTo = NULL;
|
|
fo.fFlags = 0;
|
|
|
|
if (0 != SHFileOperation(&fo))
|
|
{
|
|
DBGERROR((TEXT("SHFileOperation [FO_DELETE] failed")));
|
|
}
|
|
//
|
|
// Remove listview items if their files were really deleted.
|
|
//
|
|
RemoveListViewItems(hwndLV, rgItemHandles);
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
// Move the selected files to a new location.
|
|
// Moved files are removed from the listview.
|
|
//
|
|
void
|
|
CFileOwnerDialog::MoveSelectedFiles(
|
|
HWND hwndLV,
|
|
LPCTSTR pszDest
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_MID, TEXT("CFileOwnerDialog::DeleteSelectedFiles")));
|
|
DblNulTermList list(1024); // 1024 is the buffer growth size in chars.
|
|
CArray<COwnerListItemHandle> rgItemHandles;
|
|
|
|
BuildListOfSelectedFiles(hwndLV, &list, &rgItemHandles);
|
|
if (0 < list.Count())
|
|
{
|
|
CPath strDest(pszDest);
|
|
if (1 == list.Count())
|
|
{
|
|
//
|
|
// If we have only a single file we MUST create a fully-qualified
|
|
// path to the destination file. Oddities in the shell's move/copy
|
|
// engine won't let us pass merely a destination folder in the
|
|
// case where that folder doesn't exist. If we give the full path
|
|
// including filename we'll get the "folder doesn't exist, create
|
|
// now?" messagebox as we would expect. If we're moving multiple
|
|
// files the shell accepts a single directory path.
|
|
//
|
|
LPCTSTR psz;
|
|
DblNulTermListIter iter(list);
|
|
if (iter.Next(&psz))
|
|
{
|
|
CPath strSrc(psz); // Copy the source
|
|
CPath strFile;
|
|
strSrc.GetFileSpec(&strFile);// Extract the filename.
|
|
strDest.Append(strFile); // Append to the dest path.
|
|
}
|
|
}
|
|
|
|
SHFILEOPSTRUCT fo;
|
|
fo.hwnd = m_hwndDlg;
|
|
fo.wFunc = FO_MOVE;
|
|
fo.pFrom = list;
|
|
fo.pTo = strDest;
|
|
fo.fFlags = FOF_RENAMEONCOLLISION;
|
|
|
|
if (0 != SHFileOperation(&fo))
|
|
{
|
|
DBGERROR((TEXT("SHFileOperation [FO_MOVE] failed")));
|
|
}
|
|
//
|
|
// Remove listview items if their file was really deleted.
|
|
//
|
|
RemoveListViewItems(hwndLV, rgItemHandles);
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
// Get the SID to use for taking ownership of files.
|
|
// First try to get the first group SID with the SE_GROUP_OWNER attribute.
|
|
// If none found, use the operator's account SID. The SID is in a
|
|
// dynamic buffer attached to the ptrSid autoptr argument.
|
|
//
|
|
HRESULT
|
|
CFileOwnerDialog::GetOwnershipSid(
|
|
array_autoptr<BYTE> *ptrSid
|
|
)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
DWORD dwErr = 0;
|
|
|
|
//
|
|
// Get the token handle. First try the thread token then the process
|
|
// token. If these fail we return early. No sense in continuing
|
|
// on if we can't get a user token.
|
|
//
|
|
CWin32Handle hToken;
|
|
if (!OpenThreadToken(GetCurrentThread(),
|
|
TOKEN_READ,
|
|
TRUE,
|
|
hToken.HandlePtr()))
|
|
{
|
|
if (ERROR_NO_TOKEN == GetLastError())
|
|
{
|
|
if (!OpenProcessToken(GetCurrentProcess(),
|
|
TOKEN_READ,
|
|
hToken.HandlePtr()))
|
|
{
|
|
dwErr = GetLastError();
|
|
DBGERROR((TEXT("Error %d opening process token"), dwErr));
|
|
return HRESULT_FROM_WIN32(dwErr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
dwErr = GetLastError();
|
|
DBGERROR((TEXT("Error %d opening thread token"), dwErr));
|
|
return HRESULT_FROM_WIN32(dwErr);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Get the required size of the group token information buffer.
|
|
//
|
|
array_autoptr<BYTE> ptrTokenInfo;
|
|
DWORD cbTokenInfo = 0;
|
|
|
|
if (!GetTokenInformation(hToken,
|
|
TokenGroups,
|
|
NULL,
|
|
cbTokenInfo,
|
|
&cbTokenInfo))
|
|
{
|
|
dwErr = GetLastError();
|
|
if (ERROR_INSUFFICIENT_BUFFER == dwErr)
|
|
{
|
|
ptrTokenInfo = new BYTE[cbTokenInfo];
|
|
}
|
|
else
|
|
{
|
|
dwErr = GetLastError();
|
|
DBGERROR((TEXT("Error %d getting TokenGroups info [for size]"), dwErr));
|
|
hr = HRESULT_FROM_WIN32(dwErr);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Get the group token information.
|
|
//
|
|
if (NULL != ptrTokenInfo.get())
|
|
{
|
|
if (!GetTokenInformation(hToken,
|
|
TokenGroups,
|
|
ptrTokenInfo.get(),
|
|
cbTokenInfo,
|
|
&cbTokenInfo))
|
|
{
|
|
dwErr = GetLastError();
|
|
DBGERROR((TEXT("Error %d getting TokenGroups info"), dwErr));
|
|
hr = HRESULT_FROM_WIN32(dwErr);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Extract the first SID with the GROUP_OWNER bit set.
|
|
//
|
|
TOKEN_GROUPS *ptg = (TOKEN_GROUPS *)ptrTokenInfo.get();
|
|
DBGASSERT((NULL != ptg));
|
|
for (DWORD i = 0; i < ptg->GroupCount; i++)
|
|
{
|
|
SID_AND_ATTRIBUTES *psa = (SID_AND_ATTRIBUTES *)&ptg->Groups[i];
|
|
DBGASSERT((NULL != psa));
|
|
if (SE_GROUP_OWNER & psa->Attributes)
|
|
{
|
|
int cbSid = GetLengthSid(psa->Sid);
|
|
*ptrSid = new BYTE[cbSid];
|
|
CopySid(cbSid, ptrSid->get(), psa->Sid);
|
|
hr = NOERROR;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
//
|
|
// Didn't find a SID from the group information.
|
|
// Use the operator's SID.
|
|
//
|
|
cbTokenInfo = 0;
|
|
if (!GetTokenInformation(hToken,
|
|
TokenUser,
|
|
NULL,
|
|
cbTokenInfo,
|
|
&cbTokenInfo))
|
|
{
|
|
dwErr = GetLastError();
|
|
if (ERROR_INSUFFICIENT_BUFFER == dwErr)
|
|
{
|
|
ptrTokenInfo = new BYTE[cbTokenInfo];
|
|
}
|
|
else
|
|
{
|
|
DBGERROR((TEXT("Error %d getting TokenUser info [for size]"), dwErr));
|
|
hr = HRESULT_FROM_WIN32(dwErr);
|
|
}
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
//
|
|
// Get the user token information.
|
|
//
|
|
if (!GetTokenInformation(hToken,
|
|
TokenUser,
|
|
ptrTokenInfo.get(),
|
|
cbTokenInfo,
|
|
&cbTokenInfo))
|
|
{
|
|
dwErr = GetLastError();
|
|
DBGERROR((TEXT("Error %d getting TokenUser info"), dwErr));
|
|
hr = HRESULT_FROM_WIN32(dwErr);
|
|
}
|
|
else
|
|
{
|
|
SID_AND_ATTRIBUTES *psa = (SID_AND_ATTRIBUTES *)ptrTokenInfo.get();
|
|
DBGASSERT((NULL != psa));
|
|
int cbSid = GetLengthSid(psa->Sid);
|
|
*ptrSid = new BYTE[cbSid];
|
|
CopySid(cbSid, ptrSid->get(), psa->Sid);
|
|
hr = NOERROR;
|
|
}
|
|
}
|
|
}
|
|
if (SUCCEEDED(hr) && NULL != ptrSid->get() && !IsValidSid(ptrSid->get()))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_SID);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
//
|
|
// Transfers ownership of selected files in the listview to the
|
|
// currently logged-on user.
|
|
//
|
|
HRESULT
|
|
CFileOwnerDialog::TakeOwnershipOfSelectedFiles(
|
|
HWND hwndLV
|
|
)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
DWORD dwErr = 0;
|
|
CArray<COwnerListItemHandle> rgItemHandles;
|
|
|
|
BuildListOfSelectedFiles(hwndLV, NULL, &rgItemHandles);
|
|
if (0 == rgItemHandles.Count())
|
|
return S_OK;
|
|
|
|
array_autoptr<BYTE> ptrSid;
|
|
hr = GetOwnershipSid(&ptrSid);
|
|
if (FAILED(hr))
|
|
return hr;
|
|
|
|
CPath strFile;
|
|
int cHandles = rgItemHandles.Count();
|
|
for (int i = 0; i < cHandles; i++)
|
|
{
|
|
COwnerListItemHandle handle = rgItemHandles[i];
|
|
int iItem = FindItemFromHandle(hwndLV, handle);
|
|
if (-1 != iItem)
|
|
{
|
|
SECURITY_DESCRIPTOR sd;
|
|
if (InitializeSecurityDescriptor(&sd, SECURITY_DESCRIPTOR_REVISION))
|
|
{
|
|
int iOwner = handle.OwnerIndex();
|
|
int iFile = handle.FileIndex();
|
|
m_OwnerList.GetFileFullPath(iOwner, iFile, &strFile);
|
|
if (SetSecurityDescriptorOwner(&sd, ptrSid.get(), FALSE))
|
|
{
|
|
if (SetFileSecurity(strFile, OWNER_SECURITY_INFORMATION, &sd))
|
|
{
|
|
ListView_DeleteItem(hwndLV, iItem);
|
|
m_OwnerList.MarkFileDeleted(iOwner, iFile);
|
|
}
|
|
else
|
|
{
|
|
dwErr = GetLastError();
|
|
DBGERROR((TEXT("Error %d setting new owner for \"%s\""),
|
|
dwErr, strFile.Cstr()));
|
|
hr = HRESULT_FROM_WIN32(dwErr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
dwErr = GetLastError();
|
|
DBGERROR((TEXT("Error %d setting security descriptor owner"), dwErr));
|
|
hr = HRESULT_FROM_WIN32(dwErr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
dwErr = GetLastError();
|
|
DBGERROR((TEXT("Error %d initing security descriptor"), GetLastError()));
|
|
hr = HRESULT_FROM_WIN32(dwErr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DBGERROR((TEXT("Can't find listview item for owner %d, file %d"),
|
|
handle.OwnerIndex(), handle.FileIndex()));
|
|
}
|
|
}
|
|
//
|
|
// Refresh the owner combo with the new file counts.
|
|
//
|
|
InitializeOwnerCombo(m_OwnerList, m_hwndOwnerCombo);
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// The original code for listing files owned by a user was
|
|
// contributed by MarkZ. I made some minor modifications
|
|
// to fit it into the diskquota project and make it more
|
|
// exception safe.
|
|
//
|
|
inline VOID *
|
|
Add2Ptr(VOID *pv, ULONG cb)
|
|
{
|
|
return((BYTE *) pv + cb);
|
|
}
|
|
|
|
inline ULONG
|
|
QuadAlign( ULONG Value )
|
|
{
|
|
return (Value + 7) & ~7;
|
|
}
|
|
|
|
|
|
//
|
|
// Add files owned by a particular user on a particular volume.
|
|
//
|
|
HRESULT
|
|
CFileOwnerDialog::AddFilesToOwnerList(
|
|
LPCTSTR pszVolumeRoot,
|
|
HANDLE hVolumeRoot,
|
|
IDiskQuotaUser *pOwner,
|
|
COwnerList *pOwnerList
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_MID, TEXT("CFileOwnerDialog::AddFilesToOwnerList")));
|
|
DBGASSERT((NULL != hVolumeRoot));
|
|
DBGASSERT((NULL != pOwner));
|
|
DBGASSERT((NULL != pOwnerList));
|
|
|
|
struct
|
|
{
|
|
ULONG Restart;
|
|
BYTE Sid[MAX_SID_LEN];
|
|
}FsCtlInput;
|
|
|
|
NTSTATUS status = ERROR_SUCCESS;
|
|
|
|
if (m_bAbort)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
//
|
|
// Get owner's SID.
|
|
//
|
|
HRESULT hr = pOwner->GetSid(FsCtlInput.Sid, sizeof(FsCtlInput.Sid));
|
|
if (FAILED(hr))
|
|
{
|
|
DBGERROR((TEXT("IDiskQuotaUser::GetSid failed with hr = 0x%08X"), hr));
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Add the owner to the owner-file list.
|
|
//
|
|
int iOwner = pOwnerList->AddOwner(pOwner);
|
|
|
|
IO_STATUS_BLOCK iosb;
|
|
FsCtlInput.Restart = 1;
|
|
BYTE Output[1024];
|
|
bool bPathIsRemote = false;
|
|
FILE_FS_DEVICE_INFORMATION DeviceInfo;
|
|
|
|
//
|
|
// Determine if the volume is a remote device. This will affect
|
|
// our handling of the paths returned by NtQueryInformationFile.
|
|
//
|
|
status = NtQueryVolumeInformationFile(
|
|
hVolumeRoot,
|
|
&iosb,
|
|
&DeviceInfo,
|
|
sizeof(DeviceInfo),
|
|
FileFsDeviceInformation);
|
|
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
bPathIsRemote = (FILE_REMOTE_DEVICE == DeviceInfo.Characteristics);
|
|
}
|
|
|
|
while (!m_bAbort)
|
|
{
|
|
status = NtFsControlFile(hVolumeRoot,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&iosb,
|
|
FSCTL_FIND_FILES_BY_SID,
|
|
&FsCtlInput,
|
|
sizeof(FsCtlInput),
|
|
Output,
|
|
sizeof(Output));
|
|
|
|
FsCtlInput.Restart = 0;
|
|
|
|
if (!NT_SUCCESS(status) && STATUS_BUFFER_OVERFLOW != status)
|
|
{
|
|
DBGERROR((TEXT("NtFsControlFile failed with status 0x%08X"), status));
|
|
return HRESULT_FROM_NT(status);
|
|
}
|
|
|
|
if (0 == iosb.Information)
|
|
{
|
|
//
|
|
// No more data.
|
|
//
|
|
break;
|
|
}
|
|
|
|
PFILE_NAME_INFORMATION pFileNameInfo = (PFILE_NAME_INFORMATION)Output;
|
|
|
|
while (!m_bAbort && ((PBYTE)pFileNameInfo < Output + iosb.Information))
|
|
{
|
|
ULONG Length = sizeof(FILE_NAME_INFORMATION) - sizeof(WCHAR) +
|
|
pFileNameInfo->FileNameLength;
|
|
|
|
CNtHandle hChild;
|
|
WCHAR szChild[MAX_PATH];
|
|
ULONG cbWrite = min(pFileNameInfo->FileNameLength, sizeof(szChild));
|
|
|
|
RtlMoveMemory(szChild, pFileNameInfo->FileName, cbWrite);
|
|
|
|
szChild[cbWrite / sizeof(WCHAR)] = L'\0';
|
|
status = OpenNtObject(szChild,
|
|
hVolumeRoot,
|
|
FILE_SYNCHRONOUS_IO_NONALERT,
|
|
FILE_READ_ATTRIBUTES,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
FILE_OPEN,
|
|
hChild.HandlePtr());
|
|
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
DBGERROR((TEXT("Unable to open file \"%s\". Status = 0x%08X"),
|
|
szChild, status));
|
|
}
|
|
else if (!m_bAbort)
|
|
{
|
|
//
|
|
// Directory entries get a slightly different treatment so
|
|
// we need to know if an entry is a directory or not.
|
|
//
|
|
bool bIsDirectory = false;
|
|
IO_STATUS_BLOCK iosb2;
|
|
FILE_BASIC_INFORMATION fbi;
|
|
status = NtQueryInformationFile(hChild,
|
|
&iosb2,
|
|
&fbi,
|
|
sizeof(fbi),
|
|
FileBasicInformation);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
DBGERROR((TEXT("NtQueryInformationFile failed with status 0x%08X for \"%s\""),
|
|
status, szChild));
|
|
}
|
|
else if (0 != (FILE_ATTRIBUTE_DIRECTORY & fbi.FileAttributes))
|
|
{
|
|
bIsDirectory = true;
|
|
}
|
|
|
|
//
|
|
// Get the file's name (full path).
|
|
//
|
|
WCHAR szFile[MAX_PATH + 10];
|
|
status = NtQueryInformationFile(hChild,
|
|
&iosb2,
|
|
szFile,
|
|
sizeof(szFile),
|
|
FileNameInformation);
|
|
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
DBGERROR((TEXT("NtQueryInformation file failed with status 0x%08X for \"%s\""),
|
|
status, szChild));
|
|
}
|
|
else if (!m_bAbort)
|
|
{
|
|
PFILE_NAME_INFORMATION pfn = (PFILE_NAME_INFORMATION)szFile;
|
|
pfn->FileName[pfn->FileNameLength / sizeof(WCHAR)] = L'\0';
|
|
CPath path;
|
|
|
|
//
|
|
// If the path is remote, NtQueryInformationFile returns
|
|
// a string like this:
|
|
//
|
|
// \server\share\dir1\dir2\file.ext
|
|
//
|
|
// If the path is local, NtQueryInformationFile returns
|
|
// a string like this:
|
|
//
|
|
// \dir1\dir2\file.ext
|
|
//
|
|
// For remote paths we merely prepend a '\' to create a
|
|
// valid UNC path. For local paths we prepend the local
|
|
// drive specification.
|
|
//
|
|
if (bPathIsRemote)
|
|
{
|
|
path = L"\\";
|
|
path += CString(pfn->FileName);
|
|
}
|
|
else
|
|
{
|
|
path = pszVolumeRoot;
|
|
path.Append(pfn->FileName);
|
|
}
|
|
DBGPRINT((DM_VIEW, DL_LOW, TEXT("Adding \"%s\""), path.Cstr()));
|
|
pOwnerList->AddFile(iOwner, path, bIsDirectory);
|
|
}
|
|
}
|
|
hChild.Close();
|
|
|
|
pFileNameInfo =
|
|
(PFILE_NAME_INFORMATION) Add2Ptr(pFileNameInfo, QuadAlign(Length));
|
|
}
|
|
}
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//
|
|
// Build a list of files owned by a set of users on a particular volume.
|
|
// pszVolumeRoot is the volume root directory (i.e. "C:\").
|
|
// rgpOwners is an array of user object pointers, one for each owner.
|
|
// pOwnerList is the container where the resulting filenames are placed.
|
|
// Calls AddFilesToOwnerList() for each owner in rgpOwners.
|
|
//
|
|
HRESULT
|
|
CFileOwnerDialog::BuildFileOwnerList(
|
|
LPCTSTR pszVolumeRoot,
|
|
const CArray<IDiskQuotaUser *>& rgpOwners,
|
|
COwnerList *pOwnerList
|
|
)
|
|
{
|
|
DBGTRACE((DM_VIEW, DL_MID, TEXT("CFileOwnerDialog::BuildFileOwnerList")));
|
|
HRESULT hr = NOERROR;
|
|
CNtHandle hVolumeRoot;
|
|
NTSTATUS status = OpenNtObject(pszVolumeRoot,
|
|
NULL,
|
|
FILE_SYNCHRONOUS_IO_NONALERT,
|
|
FILE_READ_ATTRIBUTES,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
FILE_OPEN,
|
|
hVolumeRoot.HandlePtr());
|
|
|
|
if (!NT_SUCCESS(status))
|
|
return HRESULT_FROM_NT(status);
|
|
|
|
int cOwners = rgpOwners.Count();
|
|
for (int i = 0; i < cOwners && !m_bAbort; i++)
|
|
{
|
|
hr = AddFilesToOwnerList(pszVolumeRoot, hVolumeRoot, rgpOwners[i], pOwnerList);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
//
|
|
// MarkZ had this function in his original implementation so I just
|
|
// kept it. I did need to fix a bug in the original code. He was
|
|
// calling RtlFreeHeap() on str.Buffer for all cases. This is was
|
|
// not applicable in the RtlInitUnicodeString() case where the
|
|
// unicode string is merely bound to the pszFile argument.
|
|
//
|
|
NTSTATUS
|
|
CFileOwnerDialog::OpenNtObject (
|
|
LPCWSTR pszFile,
|
|
HANDLE RelatedObject,
|
|
ULONG CreateOptions,
|
|
ULONG DesiredAccess,
|
|
ULONG ShareAccess,
|
|
ULONG CreateDisposition,
|
|
HANDLE *ph)
|
|
{
|
|
NTSTATUS status;
|
|
OBJECT_ATTRIBUTES oa;
|
|
UNICODE_STRING str;
|
|
IO_STATUS_BLOCK isb;
|
|
bool bFreeString = false;
|
|
|
|
if (NULL == RelatedObject)
|
|
{
|
|
RtlDosPathNameToNtPathName_U(pszFile, &str, NULL, NULL);
|
|
bFreeString = true;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// This just attaches pszFile to the rtl string.
|
|
// We don't free it.
|
|
//
|
|
RtlInitUnicodeString(&str, pszFile);
|
|
}
|
|
|
|
InitializeObjectAttributes(&oa,
|
|
&str,
|
|
OBJ_CASE_INSENSITIVE,
|
|
RelatedObject,
|
|
NULL);
|
|
|
|
status = NtCreateFile(ph,
|
|
DesiredAccess | SYNCHRONIZE,
|
|
&oa,
|
|
&isb,
|
|
NULL, // pallocationsize (none!)
|
|
FILE_ATTRIBUTE_NORMAL,
|
|
ShareAccess,
|
|
CreateDisposition,
|
|
CreateOptions,
|
|
NULL, // EA buffer (none!)
|
|
0);
|
|
|
|
if (bFreeString)
|
|
RtlFreeHeap(RtlProcessHeap(), 0, str.Buffer);
|
|
return(status);
|
|
}
|