Windows2003-3790/admin/snapin/dsadmin/queryui.cpp
2020-09-30 16:53:55 +02:00

2066 lines
55 KiB
C++

//+----------------------------------------------------------------------------
//
// DS Administration MMC snapin.
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1999
//
// File: queryui.cpp
//
//--------------------------------------------------------------------------
#include "stdafx.h"
#include "resource.h"
#include "queryui.h"
#include "dssnap.h"
#include "uiutil.h"
#include <cmnquery.h> // DSFind
#include <dsquery.h> // DSFind
#include <dsclient.h> // BrowseForContainer
#include <dsqueryp.h> // COLUMNINFO and QueryParamsAddQueryString, QueryParamsAlloc helpers
#include <cmnquryp.h> // CQFF_ISNEVERLISTED
#include <lmaccess.h> // UF_ACCOUNTDISABLE and UF_DONT_EXPIRE_PASSWD
#include <ntldap.h> // LDAP_MATCHING_RULE_BIT_AND_W
#include "ldaputil.h" // LdapEscape().
#define DSQF_LAST_LOGON_QUERY 0x00000001
#define DSQF_NON_EXPIRING_PWD_QUERY 0x00000004
//
// Used to set the maximum text length on fields in the new query dialog
//
#define MAX_QUERY_NAME_LENGTH 259
#define MAX_QUERY_DESC_LENGTH 1024
typedef struct
{
UINT nDisplayStringID;
PWSTR pszFormatString;
} QUERYSTRINGS, * PQUERYSTRINGS;
QUERYSTRINGS g_pQueryStrings[] = {
{ IDS_STARTSWITH, L"(%s=%s*)" },
{ IDS_ENDSWITH, L"(%s=*%s)" },
{ IDS_ISEXACTLY, L"(%s=%s)" },
{ IDS_ISNOT, L"(!%s=%s)" },
{ IDS_PRESENT, L"(%s=%s*)" }, // NOTE: the second string needs to be NULL here
{ IDS_NOTPRESENT, L"(!%s=%s*)" }, // NOTE: the second string needs to be NULL here
{ 0, NULL }
};
static const CString g_szUserAccountCtrlQuery = L"(userAccountControl:" + CString(LDAP_MATCHING_RULE_BIT_AND_W) + L":=%u)";
/*-----------------------------------------------------------------------------
/ QueryParamsAlloc
/ ----------------
/ Construct a block we can pass to the DS query handler which contains
/ all the parameters for the query.
/
/ In:
/ ppDsQueryParams -> receives the parameter block
/ pQuery -> LDAP query string to be used
/ iColumns = number of columns
/ pColumnInfo -> column info structure to use
/
/ Out:
/ HRESULT
/----------------------------------------------------------------------------*/
HRESULT QueryParamsAlloc(LPDSQUERYPARAMS* ppDsQueryParams, LPWSTR pQuery, LONG iColumns, LPCOLUMNINFO aColumnInfo)
{
HRESULT hr = S_OK;
LPDSQUERYPARAMS pDsQueryParams = NULL;
size_t cbStruct;
LONG i;
ASSERT(!*ppDsQueryParams);
TRACE(L"QueryParamsAlloc");
AFX_MANAGE_STATE(AfxGetStaticModuleState());
if ( !pQuery || !iColumns || !ppDsQueryParams )
{
return E_INVALIDARG;
}
//
// Compute the size of the structure we need to be using
//
cbStruct = sizeof(DSQUERYPARAMS) + (sizeof(DSCOLUMN)*iColumns);
cbStruct += (wcslen(pQuery) + 1) * sizeof(WCHAR);
for (i = 0; i < iColumns; i++)
{
if (aColumnInfo[i].pPropertyName)
{
cbStruct += (wcslen(aColumnInfo[i].pPropertyName) + 1) * sizeof(WCHAR);
}
}
pDsQueryParams = (LPDSQUERYPARAMS)CoTaskMemAlloc(cbStruct);
if (!pDsQueryParams)
{
return E_OUTOFMEMORY;
}
//
// Structure allocated so lets fill it with data
//
pDsQueryParams->cbStruct = static_cast<DWORD>(cbStruct);
pDsQueryParams->dwFlags = 0;
pDsQueryParams->hInstance = _Module.m_hInst;
pDsQueryParams->iColumns = iColumns;
pDsQueryParams->dwReserved = 0;
cbStruct = sizeof(DSQUERYPARAMS) + (sizeof(DSCOLUMN)*iColumns);
pDsQueryParams->offsetQuery = static_cast<LONG>(cbStruct);
CopyMemory(&(((LPBYTE)pDsQueryParams)[cbStruct]), pQuery, (wcslen(pQuery) + 1) * sizeof(WCHAR));
cbStruct += (wcslen(pQuery) + 1) * sizeof(WCHAR);
for ( i = 0 ; i < iColumns ; i++ )
{
pDsQueryParams->aColumns[i].dwFlags = 0;
pDsQueryParams->aColumns[i].fmt = aColumnInfo[i].fmt;
pDsQueryParams->aColumns[i].cx = aColumnInfo[i].cx;
pDsQueryParams->aColumns[i].idsName = aColumnInfo[i].idsName;
pDsQueryParams->aColumns[i].dwReserved = 0;
if ( aColumnInfo[i].pPropertyName )
{
pDsQueryParams->aColumns[i].offsetProperty = static_cast<LONG>(cbStruct);
CopyMemory(&(((LPBYTE)pDsQueryParams)[cbStruct]), aColumnInfo[i].pPropertyName, (wcslen(aColumnInfo[i].pPropertyName) + 1) * sizeof(WCHAR));
cbStruct += (wcslen(aColumnInfo[i].pPropertyName) + 1) * sizeof(WCHAR);
}
else
{
pDsQueryParams->aColumns[i].offsetProperty = aColumnInfo[i].iPropertyIndex;
}
}
*ppDsQueryParams = pDsQueryParams;
return hr;
}
/*-----------------------------------------------------------------------------
/ QueryParamsAddQueryString
/ -------------------------
/ Given an existing DS query block appened the given LDAP query string into
/ it. We assume that the query block has been allocated by IMalloc (or CoTaskMemAlloc).
/
/ In:
/ ppDsQueryParams -> receives the parameter block
/ pQuery -> LDAP query string to be appended
/
/ Out:
/ HRESULT
/----------------------------------------------------------------------------*/
HRESULT QueryParamsAddQueryString(LPDSQUERYPARAMS* ppDsQueryParams, LPWSTR pQuery)
{
HRESULT hr = S_OK;
LPWSTR pOriginalQuery = NULL;
LPDSQUERYPARAMS pDsQuery = *ppDsQueryParams;
size_t cbQuery;
LONG i;
LPVOID pv;
//NTRAID#NTBUG9-567482-2002/03/10-jmessec Assertion not backed up by code (well, sorta is below in checking pDsQuery, but not obvious)
ASSERT(*ppDsQueryParams);
TRACE(_T("QueryParamsAddQueryString"));
if ( pQuery )
{
if (!pDsQuery)
{
return E_INVALIDARG;
}
// Work out the size of the bits we are adding, take a copy of the
// query string and finally re-alloc the query block (which may cause it
// to move).
cbQuery = ((wcslen(pQuery) + 1) * sizeof(WCHAR));
TRACE(_T("DSQUERYPARAMS being resized by %d bytes"));
i = static_cast<LONG>((wcslen((LPWSTR)ByteOffset(pDsQuery, pDsQuery->offsetQuery)) + 1) * sizeof(WCHAR));
pOriginalQuery = (WCHAR*) new BYTE[i];
if (!pOriginalQuery)
{
return E_OUTOFMEMORY;
}
lstrcpyW(pOriginalQuery, (LPWSTR)ByteOffset(pDsQuery, pDsQuery->offsetQuery));
pv = CoTaskMemRealloc(*ppDsQueryParams, pDsQuery->cbStruct+cbQuery);
if ( pv == NULL )
{
delete[] pOriginalQuery;
pOriginalQuery = 0;
return E_OUTOFMEMORY;
}
*ppDsQueryParams = (LPDSQUERYPARAMS) pv;
pDsQuery = *ppDsQueryParams; // if may have moved
// Now move everything above the query string up, and fix all the
// offsets that reference those items (probably the property table),
// finally adjust the size to reflect the change
MoveMemory(ByteOffset(pDsQuery, pDsQuery->offsetQuery+cbQuery),
ByteOffset(pDsQuery, pDsQuery->offsetQuery),
(pDsQuery->cbStruct - pDsQuery->offsetQuery));
for ( i = 0 ; i < pDsQuery->iColumns ; i++ )
{
if ( pDsQuery->aColumns[i].offsetProperty > pDsQuery->offsetQuery )
{
pDsQuery->aColumns[i].offsetProperty += static_cast<LONG>(cbQuery);
}
}
//NTRAID#NTBUG9-572009-2002/03/10-jmessec It appears that your struct is one sizeof(WCHAR) too large now, since
//you allocated enough extra for a whole other string, including terminating NULL,
//but you are appending the strings, thus eating a NULL; could this fowl up later
//struct size calculations and offsets?
wcscpy((LPWSTR)ByteOffset(pDsQuery, pDsQuery->offsetQuery), pOriginalQuery);
wcscat((LPWSTR)ByteOffset(pDsQuery, pDsQuery->offsetQuery), pQuery);
pDsQuery->cbStruct += static_cast<DWORD>(cbQuery);
delete[] pOriginalQuery;
pOriginalQuery = 0;
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
// AddQueryUnitWithModifier
HRESULT AddQueryUnitWithModifier(UINT nModifierStringID,
PCWSTR pszAttrName,
PCWSTR pszValue,
CString& szFilter)
{
HRESULT hr = S_OK;
ASSERT(pszAttrName != NULL);
if (pszAttrName == NULL)
{
return E_INVALIDARG;
}
// Escape special characters used in LDAP from the value the
// user entered.
wstring escapedValue;
LdapEscape(
pszValue != NULL ? pszValue : L"",
escapedValue);
CString szNewFilter;
PQUERYSTRINGS pQueryStrings = g_pQueryStrings;
PWSTR pszFormatString = NULL;
while (pQueryStrings->nDisplayStringID != 0)
{
if (nModifierStringID == pQueryStrings->nDisplayStringID)
{
pszFormatString = pQueryStrings->pszFormatString;
break;
}
pQueryStrings++;
}
if (pszFormatString != NULL)
{
szNewFilter.Format(pszFormatString, pszAttrName, escapedValue.c_str());
szFilter += szNewFilter;
}
else
{
hr = E_INVALIDARG;
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
// CQueryPageBase
BEGIN_MESSAGE_MAP(CQueryPageBase, CHelpDialog)
END_MESSAGE_MAP()
///////////////////////////////////////////////////////////////////////////////
// CStdQueryPage
#define FILTER_PREFIX_USER L"(objectCategory=person)(objectClass=user)"
#define FILTER_PREFIX_COMPUTER L"(objectCategory=computer)"
#define FILTER_PREFIX_GROUP L"(objectCategory=group)"
#define ATTR_COL_NAME L"name"
#define ATTR_COL_DESC L"description"
COLUMNINFO UserColumn[] =
{
{ 0, 40, IDS_QUERY_COL_NAME, 0, ATTR_COL_NAME },
{ 0, 40, IDS_QUERY_COL_DESC, 0, ATTR_COL_DESC }
};
int cUserColumns = 2;
BEGIN_MESSAGE_MAP(CStdQueryPage, CQueryPageBase)
ON_CBN_SELCHANGE(IDC_NAME_COMBO, OnNameComboChange)
ON_CBN_SELCHANGE(IDC_DESCRIPTION_COMBO, OnDescriptionComboChange)
END_MESSAGE_MAP()
void CStdQueryPage::DoContextHelp(HWND hWndControl)
{
if (hWndControl)
{
::WinHelp(hWndControl,
DSADMIN_CONTEXT_HELP_FILE,
HELP_WM_HELP,
(DWORD_PTR)(LPTSTR)g_aHelpIDs_IDD_QUERY_STD_PAGE);
}
}
BOOL CStdQueryPage::OnInitDialog()
{
CHelpDialog::OnInitDialog();
PQUERYSTRINGS pQueryStrings = g_pQueryStrings;
ASSERT(pQueryStrings != NULL);
//
// Fill in the combo boxes
//
while (pQueryStrings->nDisplayStringID != 0)
{
CString szComboString;
VERIFY(szComboString.LoadString(pQueryStrings->nDisplayStringID));
//
// Fill in the Name combo
//
LRESULT lRes = SendDlgItemMessage(IDC_NAME_COMBO, CB_ADDSTRING, 0, (LPARAM)(PCWSTR)szComboString);
if (lRes != CB_ERR)
{
lRes = SendDlgItemMessage(IDC_NAME_COMBO, CB_SETITEMDATA, (WPARAM)lRes, (LPARAM)pQueryStrings->nDisplayStringID);
ASSERT(lRes != CB_ERR);
}
//
// Fill in the Description combo
//
lRes = SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_ADDSTRING, 0, (LPARAM)(PCWSTR)szComboString);
if (lRes != CB_ERR)
{
lRes = SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_SETITEMDATA, (WPARAM)lRes, (LPARAM)pQueryStrings->nDisplayStringID);
ASSERT(lRes != CB_ERR);
}
pQueryStrings++;
}
//
// Insert an empty so that there is a way to undo changes
//
LRESULT lBlankName = SendDlgItemMessage(IDC_NAME_COMBO, CB_ADDSTRING, 0, (LPARAM)L"");
if (lBlankName != CB_ERR)
{
SendDlgItemMessage(IDC_NAME_COMBO, CB_SETITEMDATA, (WPARAM)lBlankName, (LPARAM)0);
}
//
// Insert an empty so that there is a way to undo changes
//
LRESULT lBlankDesc = SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_ADDSTRING, 0, (LPARAM)L"");
if (lBlankDesc != CB_ERR)
{
SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_SETITEMDATA, (WPARAM)lBlankDesc, (LPARAM)0);
}
//
// Force the UI to enable and disable controls related to the combo boxes
//
OnNameComboChange();
OnDescriptionComboChange();
return FALSE;
}
void CStdQueryPage::OnNameComboChange()
{
LRESULT lRes = SendDlgItemMessage(IDC_NAME_COMBO, CB_GETCURSEL, 0, 0);
if (lRes != CB_ERR)
{
LRESULT lData = SendDlgItemMessage(IDC_NAME_COMBO, CB_GETITEMDATA, lRes, 0);
if (lData != CB_ERR)
{
if (lData == IDS_PRESENT || lData == IDS_NOTPRESENT || lData == 0)
{
GetDlgItem(IDC_NAME_EDIT)->EnableWindow(FALSE);
SetDlgItemText(IDC_NAME_EDIT, L"");
}
else
{
GetDlgItem(IDC_NAME_EDIT)->EnableWindow(TRUE);
}
}
}
else
{
GetDlgItem(IDC_NAME_EDIT)->EnableWindow(FALSE);
}
}
void CStdQueryPage::OnDescriptionComboChange()
{
LRESULT lRes = SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_GETCURSEL, 0, 0);
if (lRes != CB_ERR)
{
LRESULT lData = SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_GETITEMDATA, lRes, 0);
if (lData != CB_ERR)
{
if (lData == IDS_PRESENT || lData == IDS_NOTPRESENT || lData == 0)
{
GetDlgItem(IDC_DESCRIPTION_EDIT)->EnableWindow(FALSE);
SetDlgItemText(IDC_DESCRIPTION_EDIT, L"");
}
else
{
GetDlgItem(IDC_DESCRIPTION_EDIT)->EnableWindow(TRUE);
}
}
}
else
{
GetDlgItem(IDC_DESCRIPTION_EDIT)->EnableWindow(FALSE);
}
}
void CStdQueryPage::Init()
{
//
// Clear all controls
//
SetDlgItemText(IDC_NAME_EDIT, L"");
SetDlgItemText(IDC_DESCRIPTION_EDIT, L"");
//
// Reselect the blank string in the combo boxes
//
LRESULT lRes = SendDlgItemMessage(IDC_NAME_COMBO, CB_FINDSTRINGEXACT, (WPARAM)-1, (LPARAM)L"");
if (lRes != CB_ERR)
{
SendDlgItemMessage(IDC_NAME_COMBO, CB_SETCURSEL, lRes, 0);
}
lRes = SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_FINDSTRINGEXACT, (WPARAM)-1, (LPARAM)L"");
if (lRes != CB_ERR)
{
SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_SETCURSEL, lRes, 0);
}
}
HRESULT CStdQueryPage::GetQueryParams(LPDSQUERYPARAMS* ppDsQueryParams)
{
HRESULT hr = S_OK;
//
// Build the filter string here
//
CString szFilter;
CString szName;
CString szDescription;
GetDlgItemText(IDC_NAME_EDIT, szName);
GetDlgItemText(IDC_DESCRIPTION_EDIT, szDescription);
//
// Get the selection of the modifier combo
//
LRESULT lSel = SendDlgItemMessage(IDC_NAME_COMBO, CB_GETCURSEL, 0, 0);
if (lSel != CB_ERR)
{
//
// Retrieve the associated string ID
//
LRESULT lData = SendDlgItemMessage(IDC_NAME_COMBO, CB_GETITEMDATA, lSel, 0);
if (lData != CB_ERR)
{
if (!szName.IsEmpty() || lData == IDS_PRESENT || lData == IDS_NOTPRESENT)
{
AddQueryUnitWithModifier(static_cast<UINT>(lData),
ATTR_COL_NAME,
szName,
szFilter);
}
}
}
//
// Get the selection of the modifier combo
//
lSel = SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_GETCURSEL, 0, 0);
if (lSel != CB_ERR)
{
//
// Retrieve the associated string ID
//
LRESULT lData = SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_GETITEMDATA, lSel, 0);
if (lData != CB_ERR)
{
if (!szDescription.IsEmpty() || lData == IDS_PRESENT || lData == IDS_NOTPRESENT)
{
AddQueryUnitWithModifier(static_cast<UINT>(lData),
ATTR_COL_DESC,
szDescription,
szFilter);
}
}
}
if (!szFilter.IsEmpty())
{
szFilter = m_szFilterPrefix + szFilter;
hr = BuildQueryParams(ppDsQueryParams, (LPWSTR)(LPCWSTR)szFilter);
}
return hr;
}
HRESULT CStdQueryPage::BuildQueryParams(LPDSQUERYPARAMS* ppDsQueryParams, LPWSTR pQuery)
{
ASSERT(pQuery);
if(*ppDsQueryParams)
{
return QueryParamsAddQueryString(ppDsQueryParams, pQuery);
}
return QueryParamsAlloc(ppDsQueryParams, pQuery, cUserColumns, UserColumn);
}
HRESULT CStdQueryPage::Persist(IPersistQuery* pPersistQuery, BOOL fRead)
{
HRESULT hr = S_OK;
if (pPersistQuery == NULL)
{
ASSERT(FALSE);
return E_INVALIDARG;
}
if (fRead)
{
//
// Read the Name combo value
//
int iData = 0;
hr = pPersistQuery->ReadInt(m_szFilterPrefix, L"NameCombo", &iData);
if (FAILED(hr))
{
TRACE(_T("Failed to read int \"NameCombo\" from stream: 0x%x\n"), hr);
ASSERT(FALSE);
return hr;
}
//
// Select the appropriate list box item
//
SelectComboAssociatedWithData(IDC_NAME_COMBO, iData);
if (iData != 0 && iData != IDS_PRESENT && iData != IDS_NOTPRESENT)
{
//
// Read the name edit value
//
WCHAR szBuf[MAX_PATH] = {0};
hr = pPersistQuery->ReadString(m_szFilterPrefix, L"NameEdit", szBuf, MAX_PATH);
if (FAILED(hr))
{
TRACE(_T("Failed to read string \"NameEdit\" from stream: 0x%x\n"), hr);
ASSERT(FALSE);
return hr;
}
if (szBuf != NULL)
{
SetDlgItemText(IDC_NAME_EDIT, szBuf);
}
}
else
{
GetDlgItem(IDC_NAME_EDIT)->EnableWindow(FALSE);
}
//
// Read the Description combo value
//
iData = 0;
hr = pPersistQuery->ReadInt(m_szFilterPrefix, L"DescCombo", &iData);
if (FAILED(hr))
{
TRACE(_T("Failed to read int \"DescCombo\" from stream: 0x%x\n"), hr);
ASSERT(FALSE);
return hr;
}
//
// Select the appropriate list box item
//
SelectComboAssociatedWithData(IDC_DESCRIPTION_COMBO, iData);
if (iData != 0 && iData != IDS_PRESENT && iData != IDS_NOTPRESENT)
{
//
// Read the name edit value
//
WCHAR szBuf[MAX_PATH] = {0};
hr = pPersistQuery->ReadString(m_szFilterPrefix, L"DescEdit", szBuf, MAX_PATH);
if (FAILED(hr))
{
TRACE(_T("Failed to read string \"DescEdit\" from stream: 0x%x\n"), hr);
ASSERT(FALSE);
return hr;
}
if (szBuf != NULL)
{
SetDlgItemText(IDC_DESCRIPTION_EDIT, szBuf);
}
}
else
{
GetDlgItem(IDC_DESCRIPTION_EDIT)->EnableWindow(FALSE);
}
OnNameComboChange();
OnDescriptionComboChange();
}
else // write
{
//
// Write out the name info
//
LRESULT lSel = SendDlgItemMessage(IDC_NAME_COMBO, CB_GETCURSEL, 0, 0);
if (lSel != CB_ERR)
{
//
// Retrieve the associated string ID
//
LRESULT lData = SendDlgItemMessage(IDC_NAME_COMBO, CB_GETITEMDATA, lSel, 0);
if (lData != CB_ERR)
{
hr = pPersistQuery->WriteInt(m_szFilterPrefix, L"NameCombo", static_cast<int>(lData));
if (FAILED(hr))
{
ASSERT(FALSE);
return hr;
}
if (lData != 0 && lData != IDS_PRESENT && lData != IDS_NOTPRESENT)
{
CString szName;
GetDlgItemText(IDC_NAME_EDIT, szName);
hr = pPersistQuery->WriteString(m_szFilterPrefix, L"NameEdit", szName);
if (FAILED(hr))
{
ASSERT(FALSE);
return hr;
}
}
}
}
else
{
//
// If there hasn't been a selection, write in the empty string value
//
hr = pPersistQuery->WriteInt(m_szFilterPrefix, L"NameCombo", 0);
}
//
// Write out the description info
//
lSel = SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_GETCURSEL, 0, 0);
if (lSel != CB_ERR)
{
//
// Retrieve the associated string ID
//
LRESULT lData = SendDlgItemMessage(IDC_DESCRIPTION_COMBO, CB_GETITEMDATA, lSel, 0);
if (lData != CB_ERR)
{
hr = pPersistQuery->WriteInt(m_szFilterPrefix, L"DescCombo", static_cast<int>(lData));
if (FAILED(hr))
{
ASSERT(FALSE);
return hr;
}
if (lData != 0 && lData != IDS_PRESENT && lData != IDS_NOTPRESENT)
{
CString szDescription;
GetDlgItemText(IDC_DESCRIPTION_EDIT, szDescription);
hr = pPersistQuery->WriteString(m_szFilterPrefix, L"DescEdit", szDescription);
if (FAILED(hr))
{
ASSERT(FALSE);
return hr;
}
}
}
}
else
{
//
// If there hasn't been a selection, write in the empty string value
//
hr = pPersistQuery->WriteInt(m_szFilterPrefix, L"DescCombo", 0);
}
}
return hr;
}
void CStdQueryPage::SelectComboAssociatedWithData(UINT nCtrlID, LRESULT lData)
{
//
// Selects the item with the associated data in a combo box
//
LRESULT lRes = SendDlgItemMessage(nCtrlID, CB_GETCOUNT, 0, 0);
if (lRes != CB_ERR)
{
for (int idx = 0; idx < static_cast<int>(lRes); idx++)
{
LRESULT lRetData = SendDlgItemMessage(nCtrlID, CB_GETITEMDATA, (WPARAM)idx, 0);
if (lRetData != CB_ERR)
{
if (lRetData == lData)
{
SendDlgItemMessage(nCtrlID, CB_SETCURSEL, (WPARAM)idx, 0);
break;
}
}
}
}
}
///////////////////////////////////////////////////////////////////////////////
// CUserComputerQueryPage
BEGIN_MESSAGE_MAP(CUserComputerQueryPage, CStdQueryPage)
END_MESSAGE_MAP()
BOOL CUserComputerQueryPage::OnInitDialog()
{
return CStdQueryPage::OnInitDialog();
}
void CUserComputerQueryPage::DoContextHelp(HWND hWndControl)
{
if (hWndControl)
{
::WinHelp(hWndControl,
DSADMIN_CONTEXT_HELP_FILE,
HELP_WM_HELP,
(DWORD_PTR)(LPTSTR)g_aHelpIDs_IDD_QUERY_USER_PAGE);
}
}
void CUserComputerQueryPage::Init()
{
//
// Clear all controls
//
CStdQueryPage::Init();
}
HRESULT CUserComputerQueryPage::GetQueryParams(LPDSQUERYPARAMS* ppDsQueryParams)
{
HRESULT hr = S_OK;
//
// Build the filter string here
//
hr = CStdQueryPage::GetQueryParams(ppDsQueryParams);
CString szFilter;
BOOL bDisabledAccounts = FALSE;
//
// Get disabled accounts check
//
LRESULT lRes = SendDlgItemMessage(IDC_DISABLED_ACCOUNTS_CHECK, BM_GETCHECK, 0, 0);
if (lRes == BST_CHECKED)
{
bDisabledAccounts = TRUE;
}
if (bDisabledAccounts)
{
szFilter.Format(g_szUserAccountCtrlQuery, UF_ACCOUNTDISABLE);
szFilter = m_szFilterPrefix + szFilter;
hr = BuildQueryParams(ppDsQueryParams, (LPWSTR)(LPCWSTR)szFilter);
}
return hr;
}
HRESULT CUserComputerQueryPage::Persist(IPersistQuery* pPersistQuery, BOOL fRead)
{
HRESULT hr = CStdQueryPage::Persist(pPersistQuery, fRead);
if (FAILED(hr))
{
return hr;
}
if (pPersistQuery == NULL)
{
ASSERT(FALSE);
return E_INVALIDARG;
}
if (fRead)
{
//
// Read disabled accounts flag
//
int iData = 0;
hr = pPersistQuery->ReadInt(m_szFilterPrefix, L"DisableCheck", &iData);
if (FAILED(hr))
{
TRACE(_T("Failed to read int \"DisableCheck\" from stream: 0x%x\n"), hr);
ASSERT(FALSE);
return hr;
}
SendDlgItemMessage(IDC_DISABLED_ACCOUNTS_CHECK, BM_SETCHECK, (iData > 0) ? BST_CHECKED : BST_UNCHECKED, 0);
}
else
{
//
// Write disabled accounts flag
//
LRESULT lRes = SendDlgItemMessage(IDC_DISABLED_ACCOUNTS_CHECK, BM_GETCHECK, 0, 0);
if (lRes != -1)
{
int iRes = (lRes == BST_CHECKED) ? 1 : 0;
hr = pPersistQuery->WriteInt(m_szFilterPrefix, L"DisableCheck", iRes);
if (FAILED(hr))
{
ASSERT(FALSE);
return hr;
}
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
// CUserQueryPage
BEGIN_MESSAGE_MAP(CUserQueryPage, CStdQueryPage)
END_MESSAGE_MAP()
BOOL CUserQueryPage::OnInitDialog()
{
return CUserComputerQueryPage::OnInitDialog();
}
void CUserQueryPage::DoContextHelp(HWND hWndControl)
{
if (hWndControl)
{
::WinHelp(hWndControl,
DSADMIN_CONTEXT_HELP_FILE,
HELP_WM_HELP,
(DWORD_PTR)(LPTSTR)g_aHelpIDs_IDD_QUERY_USER_PAGE);
}
}
void CUserQueryPage::Init()
{
//
// Clear all controls
//
CUserComputerQueryPage::Init();
}
HRESULT CUserQueryPage::GetQueryParams(LPDSQUERYPARAMS* ppDsQueryParams)
{
HRESULT hr = S_OK;
//
// Build the filter string here
//
hr = CUserComputerQueryPage::GetQueryParams(ppDsQueryParams);
CString szFilter;
BOOL bNonExpPwds = FALSE;
BOOL bLastLogon = FALSE;
DWORD dwLastLogonData = 0;
//
// Get non expiring password check
//
LRESULT lRes = SendDlgItemMessage(IDC_NON_EXPIRING_PWD_CHECK, BM_GETCHECK, 0, 0);
if (lRes == BST_CHECKED)
{
bNonExpPwds = TRUE;
}
//
// Get stale acccounts check
//
lRes = SendDlgItemMessage(IDC_LASTLOGON_COMBO, CB_GETCURSEL, 0, 0);
if (lRes == CB_ERR)
{
lRes = m_lLogonSelection;
}
if (lRes != CB_ERR)
{
LRESULT lTextLen = SendDlgItemMessage(IDC_LASTLOGON_COMBO, CB_GETLBTEXTLEN, (WPARAM)lRes, 0);
if (lTextLen != CB_ERR)
{
if (lTextLen > 0)
{
bLastLogon = TRUE;
WCHAR* pszData = new WCHAR[lTextLen + 1];
if (pszData != NULL)
{
LRESULT lData = SendDlgItemMessage(IDC_LASTLOGON_COMBO, CB_GETLBTEXT, (WPARAM)lRes, (LPARAM)pszData);
if (lData != CB_ERR)
{
dwLastLogonData = static_cast<DWORD>(_wtol(pszData));
// NTRAID#NTBUG9-449871-2001/09/20-lucios
// Adding the appropriate filter.
LARGE_INTEGER li;
GetCurrentTimeStampMinusInterval(dwLastLogonData, &li);
CString szTimeStamp;
litow(li, szTimeStamp);
CString szTempFilter;
szTempFilter.Format
(
L"%s(lastLogonTimestamp<=%s)",
szTempFilter,
szTimeStamp
);
szFilter = szTempFilter + szFilter;
}
delete[] pszData;
pszData = NULL;
}
}
}
}
if (bNonExpPwds || bLastLogon)
{
if (bNonExpPwds)
{
szFilter.Format(g_szUserAccountCtrlQuery, UF_DONT_EXPIRE_PASSWD);
}
szFilter = m_szFilterPrefix + szFilter;
hr = BuildQueryParams(ppDsQueryParams, (LPWSTR)(LPCWSTR)szFilter);
if (SUCCEEDED(hr))
{
if (bLastLogon)
{
(*ppDsQueryParams)->dwFlags |= DSQF_LAST_LOGON_QUERY;
(*ppDsQueryParams)->dwReserved = dwLastLogonData;
}
}
}
return hr;
}
HRESULT CUserQueryPage::Persist(IPersistQuery* pPersistQuery, BOOL fRead)
{
HRESULT hr = CUserComputerQueryPage::Persist(pPersistQuery, fRead);
if (FAILED(hr))
{
return hr;
}
if (pPersistQuery == NULL)
{
ASSERT(FALSE);
return E_INVALIDARG;
}
if (fRead)
{
//
// Read non expiring pwds flag
//
int iData = 0;
hr = pPersistQuery->ReadInt(m_szFilterPrefix, L"NonExpPwdCheck", &iData);
if (FAILED(hr))
{
TRACE(_T("Failed to read int \"NonExpPwdCheck\" from stream: 0x%x\n"), hr);
ASSERT(FALSE);
return hr;
}
SendDlgItemMessage(IDC_NON_EXPIRING_PWD_CHECK, BM_SETCHECK, (iData > 0) ? BST_CHECKED : BST_UNCHECKED, 0);
iData = 0;
hr = pPersistQuery->ReadInt(m_szFilterPrefix, L"LastLogonCombo", &iData);
if (FAILED(hr))
{
TRACE(_T("Failed to read int \"LastLogonCombo\" from stream: 0x%x\n"), hr);
ASSERT(FALSE);
return hr;
}
SendDlgItemMessage(IDC_LASTLOGON_COMBO, CB_SETCURSEL, (WPARAM)iData, 0);
m_lLogonSelection = iData;
}
else
{
//
// Write non expiring pwd flag
//
LRESULT lRes = SendDlgItemMessage(IDC_NON_EXPIRING_PWD_CHECK, BM_GETCHECK, 0, 0);
if (lRes != -1)
{
int iRes = (lRes == BST_CHECKED) ? 1 : 0;
hr = pPersistQuery->WriteInt(m_szFilterPrefix, L"NonExpPwdCheck", iRes);
if (FAILED(hr))
{
ASSERT(FALSE);
return hr;
}
}
//
// Write last logon combo index
//
lRes = SendDlgItemMessage(IDC_LASTLOGON_COMBO, CB_GETCURSEL, 0, 0);
if (lRes == CB_ERR)
{
if (m_lLogonSelection != -1)
{
lRes = m_lLogonSelection;
}
else
{
lRes = 0;
}
}
hr = pPersistQuery->WriteInt(m_szFilterPrefix, L"LastLogonCombo", static_cast<int>(lRes));
if (FAILED(hr))
{
ASSERT(FALSE);
return hr;
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
// CQueryFormBase
HRESULT PageProc(LPCQPAGE pPage, HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam);
INT_PTR CALLBACK DlgProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam);
HRESULT GetQueryParams(HWND hWnd, LPDSQUERYPARAMS* ppDsQueryParams);
STDMETHODIMP CQueryFormBase::Initialize(HKEY)
{
// This method is called to initialize the query form object, it is called before
// any pages are added. hkForm should be ignored, in the future however it
// will be a way to persist form state.
HRESULT hr = S_OK;
return hr;
}
STDMETHODIMP CQueryFormBase::AddForms(LPCQADDFORMSPROC pAddFormsProc, LPARAM lParam)
{
CQFORM cqf;
AFX_MANAGE_STATE(AfxGetStaticModuleState());
// This method is called to allow the form handler to register its query form(s),
// each form is identifiered by a CLSID and registered via the pAddFormProc. Here
// we are going to register a test form.
// When registering a form which is only applicable to a specific task, eg. Find a Domain
// object, it is advised that the form be marked as hidden (CQFF_ISNEVERLISTED) which
// will cause it not to appear in the form picker control. Then when the
// client wants to use this form, they specify the form identifier and ask for the
// picker control to be hidden.
if ( !pAddFormsProc )
{
return E_INVALIDARG;
}
cqf.cbStruct = sizeof(cqf);
cqf.dwFlags = CQFF_NOGLOBALPAGES;
cqf.clsid = CLSID_DSAdminQueryUIForm;
cqf.hIcon = NULL;
CString title;
title.LoadString(IDS_QUERY_TITLE_SAVEDQUERYFORM);
cqf.pszTitle = (LPCTSTR)title;
return pAddFormsProc(lParam, &cqf);
}
STDMETHODIMP CQueryFormBase::AddPages(LPCQADDPAGESPROC pAddPagesProc, LPARAM lParam)
{
HRESULT hr = S_OK;
CQPAGE cqp;
AFX_MANAGE_STATE(AfxGetStaticModuleState());
CThemeContextActivator activator;
// AddPages is called after AddForms, it allows us to add the pages for the
// forms we have registered. Each page is presented on a seperate tab within
// the dialog. A form is a dialog with a DlgProc and a PageProc.
//
// When registering a page the entire structure passed to the callback is copied,
// the amount of data to be copied is defined by the cbStruct field, therefore
// a page implementation can grow this structure to store extra information. When
// the page dialog is constructed via CreateDialog the CQPAGE strucuture is passed
// as the create param.
if ( !pAddPagesProc )
return E_INVALIDARG;
cqp.cbStruct = sizeof(cqp);
cqp.dwFlags = 0x0;
cqp.pPageProc = PageProc;
cqp.hInstance = _Module.GetModuleInstance();
cqp.pDlgProc = DlgProc;
//
// Add the user page
//
cqp.idPageName = IDS_QUERY_TITLE_USERPAGE;
cqp.idPageTemplate = IDD_QUERY_USER_PAGE;
cqp.lParam = (LPARAM)new CUserQueryPage(FILTER_PREFIX_USER);
hr = pAddPagesProc(lParam, CLSID_DSAdminQueryUIForm, &cqp);
//
// Add the computer page (this is just a std page)
//
cqp.idPageName = IDS_QUERY_TITLE_COMPUTER_PAGE;
cqp.idPageTemplate = IDD_QUERY_COMPUTER_PAGE;
cqp.lParam = (LPARAM)new CUserComputerQueryPage(IDD_QUERY_COMPUTER_PAGE, FILTER_PREFIX_COMPUTER);
hr = pAddPagesProc(lParam, CLSID_DSAdminQueryUIForm, &cqp);
//
// Add the group page (this is just a std page)
//
cqp.idPageName = IDS_QUERY_TITLE_GROUP_PAGE;
cqp.idPageTemplate = IDD_QUERY_STD_PAGE;
cqp.lParam = (LPARAM)new CStdQueryPage(IDD_QUERY_STD_PAGE, FILTER_PREFIX_GROUP);
hr = pAddPagesProc(lParam, CLSID_DSAdminQueryUIForm, &cqp);
//
// Add more pages here if needed
//
return hr;
}
#define ExitGracefully(hr, result, text) \
{ hr = result; goto exit_gracefully; }
#define StringByteSizeW(sz) ((sz) ? ((lstrlenW(sz)+1)*sizeof(WCHAR)):0)
//NTRAID#NTBUG9-572010-2002/03/10-jmessec Unsafe wrapper for CopyMemory; what if string is longer than destination buffer?
#define StringByteCopyW(pDest, iOffset, sz) \
{ CopyMemory(&(((LPBYTE)pDest)[iOffset]), sz, StringByteSizeW(sz)); }
STDAPI ClassListAlloc(LPDSQUERYCLASSLIST* ppDsQueryClassList, LPWSTR* aClassNames, INT cClassNames)
{
HRESULT hres;
DWORD cbStruct, offset;
LPDSQUERYCLASSLIST pDsQueryClassList = NULL;
INT i;
USES_CONVERSION;
if ( !ppDsQueryClassList || !aClassNames || !cClassNames )
ExitGracefully(hres, E_FAIL, "Bad parameters (no class list etc)");
// Walk the list of classes working out the size of the structure
// we are going to generate, this consists of the array of
// classes.
cbStruct = sizeof(DSQUERYCLASSLIST)+(cClassNames*sizeof(DWORD));
offset = cbStruct;
for ( i = 0 ; i < cClassNames ; i++ )
{
ASSERT(aClassNames[i]);
cbStruct += StringByteSizeW(aClassNames[i]);
}
// Allocate the structure using the task allocator, then fill
// it in copying all the strings into the data blob.
pDsQueryClassList = (LPDSQUERYCLASSLIST)CoTaskMemAlloc(cbStruct);
ASSERT(pDsQueryClassList);
if ( !pDsQueryClassList )
ExitGracefully(hres, E_OUTOFMEMORY, "Failed to allocate class list structure");
pDsQueryClassList->cbStruct = cbStruct;
pDsQueryClassList->cClasses = cClassNames;
for ( i = 0 ; i < cClassNames ; i++ )
{
pDsQueryClassList->offsetClass[i] = offset;
StringByteCopyW(pDsQueryClassList, offset, aClassNames[i]);
offset += StringByteSizeW(aClassNames[i]);
}
hres = S_OK;
exit_gracefully:
ASSERT(pDsQueryClassList);
if (ppDsQueryClassList)
*ppDsQueryClassList = pDsQueryClassList;
return hres;
}
/*---------------------------------------------------------------------------*/
// The PageProc is used to perform general house keeping and communicate between
// the frame and the page.
//
// All un-handled, or unknown reasons should result in an E_NOIMPL response
// from the proc.
//
// In:
// pPage -> CQPAGE structure (copied from the original passed to pAddPagesProc)
// hwnd = handle of the dialog for the page
// uMsg, wParam, lParam = message parameters for this event
//
// Out:
// HRESULT
//
// uMsg reasons:
// ------------
// CQPM_INIIIALIZE
// CQPM_RELEASE
// These are issued as a result of the page being declared or freed, they
// allow the caller to AddRef, Release or perform basic initialization
// of the form object.
//
// CQPM_ENABLE
// Enable is when the query form needs to enable or disable the controls
// on its page. wParam contains TRUE/FALSE indicating the state that
// is required.
//
// CQPM_GETPARAMETERS
// To collect the parameters for the query each page on the active form
// receives this event. lParam is an LPVOID* which is set to point to the
// parameter block to pass to the handler, if the pointer is non-NULL
// on entry the form needs to appened its query information to it. The
// parameter block is handler specific.
//
// Returning S_FALSE from this event causes the query to be canceled.
//
// CQPM_CLEARFORM
// When the page window is created for the first time, or the user clicks
// the clear search the page receives a CQPM_CLEARFORM notification, at
// which point it needs to clear out the edit controls it has and
// return to a default state.
//
// CQPM_PERSIST:
// When loading of saving a query, each page is called with an IPersistQuery
// interface which allows them to read or write the configuration information
// to save or restore their state. lParam is a pointer to the IPersistQuery object,
// and wParam is TRUE/FALSE indicating read or write accordingly.
HRESULT PageProc(LPCQPAGE pQueryPage, HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
HRESULT hr = S_OK;
AFX_MANAGE_STATE(AfxGetStaticModuleState());
CQueryPageBase* pDialog = (CQueryPageBase*)pQueryPage->lParam;
ASSERT(pDialog);
switch ( uMsg )
{
// Initialize so AddRef the object we are associated with so that
// we don't get unloaded.
case CQPM_INITIALIZE:
break;
case DSQPM_GETCLASSLIST:
{
LPWSTR classes[]={L"user",L"computer",L"group"};
hr = ClassListAlloc((LPDSQUERYCLASSLIST*)lParam, classes, sizeof(classes)/sizeof(*classes));
break;
}
case DSQPM_HELPTOPICS:
{
HWND hwndFrame = (HWND)lParam;
HtmlHelp(hwndFrame, TEXT("omc.chm"), HH_HELP_FINDER, 0);
break;
}
// Changed from qform sample to detach the hwnd, and delete the CDialog
// ensure correct destruction etc.
case CQPM_RELEASE:
pDialog->Detach();
SetWindowLongPtr(hwnd, DWLP_USER, (LPARAM)0);
delete pDialog;
break;
// Enable so fix the state of our two controls within the window.
case CQPM_ENABLE:
SetFocus(GetDlgItem(hwnd, IDC_NAME_COMBO));
break;
// Fill out the parameter structure to return to the caller, this is
// handler specific. In our case we constructure a query of the CN
// and objectClass properties, and we show a columns displaying both
// of these. For further information about the DSQUERYPARAMs structure
// see dsquery.h
case CQPM_GETPARAMETERS:
hr = pDialog->GetQueryParams((LPDSQUERYPARAMS*)lParam);
break;
// Clear form, therefore set the window text for these two controls
// to zero.
case CQPM_CLEARFORM:
hr = pDialog->ClearForm();
break;
// persistance is not currently supported by this form.
case CQPM_PERSIST:
{
BOOL fRead = (BOOL)wParam;
IPersistQuery* pPersistQuery = (IPersistQuery*)lParam;
if ( !pPersistQuery )
{
return E_INVALIDARG;
}
hr = pDialog->Persist(pPersistQuery, fRead);
break;
}
default:
hr = E_NOTIMPL;
break;
}
return hr;
}
/*---------------------------------------------------------------------------*/
// The DlgProc is a standard Win32 dialog proc associated with the form
// window.
INT_PTR CALLBACK DlgProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
LPCQPAGE pQueryPage;
CQueryPageBase* pDialog;
AFX_MANAGE_STATE(AfxGetStaticModuleState());
if ( uMsg == WM_INITDIALOG )
{
// changed from qForm sample to save CDialog pointer
// in the DWL_USER field of the dialog box instance.
pQueryPage = (LPCQPAGE)lParam;
pDialog = (CQueryPageBase*)pQueryPage->lParam;
pDialog->Attach(hwnd);
SetWindowLongPtr(hwnd, DWLP_USER, (LPARAM)pDialog);
return pDialog->OnInitDialog();
}
else
{
// CDialog pointer is stored in DWL_USER
// dialog structure, note however that in some cases this will
// be NULL as it is set on WM_INITDIALOG.
pDialog = (CQueryPageBase*)GetWindowLongPtr(hwnd, DWLP_USER);
}
if(!pDialog)
{
return FALSE;
}
else
{
return AfxCallWndProc(pDialog, hwnd, uMsg, wParam, lParam);
}
}
///////////////////////////////////////////////////////////////////////
// CQueryDialog
CQueryDialog::CQueryDialog(CSavedQueryNode* pQueryNode,
CFavoritesNode* pFavNode,
CDSComponentData* pComponentData,
BOOL bNewQuery,
BOOL bImportQuery)
: CHelpDialog(IDD_CREATE_NEW_QUERY)
{
m_bInit = FALSE;
m_bNewQuery = bNewQuery;
m_bImportQuery = bImportQuery;
m_pComponentData = pComponentData;
m_pQueryNode = pQueryNode;
m_pFavNode = pFavNode;
m_szName = pQueryNode->GetName();
m_szOriginalName = pQueryNode->GetName();
m_szDescription = pQueryNode->GetDesc();
m_szQueryRoot = pQueryNode->GetRootPath();
m_szQueryFilter = pQueryNode->GetQueryString();
m_bMultiLevel = !pQueryNode->IsOneLevel();
m_bLastLogonFilter = pQueryNode->IsFilterLastLogon();
m_dwLastLogonData = pQueryNode->GetLastLogonDays();
m_pPersistQueryImpl = pQueryNode->GetQueryPersist();
if (m_pPersistQueryImpl != NULL)
{
m_pPersistQueryImpl->AddRef();
}
else
{
//
// Create the IPersistQuery object
//
CComObject<CDSAdminPersistQueryFilterImpl>::CreateInstance(&m_pPersistQueryImpl);
ASSERT(m_pPersistQueryImpl != NULL);
//
// created with zero refcount,need to AddRef() to one
//
m_pPersistQueryImpl->AddRef();
}
}
CQueryDialog::~CQueryDialog()
{
if (m_pPersistQueryImpl != NULL)
{
//
// go to refcount of zero, to destroy object
//
m_pPersistQueryImpl->Release();
}
}
BEGIN_MESSAGE_MAP(CQueryDialog, CHelpDialog)
ON_BN_CLICKED(IDC_BROWSE_BUTTON, OnBrowse)
ON_BN_CLICKED(IDC_EDIT_BUTTON, OnEditQuery)
ON_BN_CLICKED(IDC_MULTI_LEVEL_CHECK, OnMultiLevelChange)
ON_EN_CHANGE(IDC_NAME_EDIT, OnNameChange)
ON_EN_CHANGE(IDC_DESCRIPTION_EDIT, OnDescriptionChange)
ON_NOTIFY_EX(TTN_NEEDTEXT, 0, OnNeedToolTipText)
END_MESSAGE_MAP()
void CQueryDialog::DoContextHelp(HWND hWndControl)
{
if (hWndControl)
{
::WinHelp(hWndControl,
DSADMIN_CONTEXT_HELP_FILE,
HELP_WM_HELP,
(DWORD_PTR)(LPTSTR)g_aHelpIDs_IDD_CREATE_NEW_QUERY);
}
}
BOOL CQueryDialog::OnInitDialog()
{
CHelpDialog::OnInitDialog();
if (m_pQueryNode == NULL)
{
ASSERT(FALSE);
EndDialog(IDCANCEL);
}
//
// Change the title for editing queries
//
if (!m_bNewQuery)
{
CString szTitle;
VERIFY(szTitle.LoadString(IDS_SAVED_QUERIES_EDIT_TITLE));
SetWindowText(szTitle);
}
//
// Initialize the controls with data
//
SetDlgItemText(IDC_NAME_EDIT, m_szName);
SendDlgItemMessage(IDC_NAME_EDIT, EM_SETLIMITTEXT, MAX_QUERY_NAME_LENGTH, 0);
SetDlgItemText(IDC_DESCRIPTION_EDIT, m_szDescription);
SendDlgItemMessage(IDC_DESCRIPTION_EDIT, EM_SETLIMITTEXT, MAX_QUERY_DESC_LENGTH, 0);
SendDlgItemMessage(IDC_MULTI_LEVEL_CHECK, BM_SETCHECK, (m_bMultiLevel) ? BST_CHECKED : BST_UNCHECKED, 0);
SetQueryFilterDisplay();
EnableToolTips(TRUE);
SetQueryRoot(m_szQueryRoot);
SetDirty();
m_bInit = TRUE;
return TRUE;
}
void CQueryDialog::SetDirty(BOOL bDirty)
{
if (m_bInit || m_bImportQuery)
{
m_szName.TrimLeft();
m_szName.TrimRight();
if (m_szName.IsEmpty() ||
m_szQueryRoot.IsEmpty() ||
m_szQueryFilter.IsEmpty())
{
m_bDirty = FALSE;
}
else
{
m_bDirty = bDirty;
}
GetDlgItem(IDOK)->EnableWindow(m_bDirty);
}
}
void CQueryDialog::OnOK()
{
CThemeContextActivator activator;
if (m_bDirty)
{
if (m_pQueryNode != NULL)
{
GetDlgItemText(IDC_NAME_EDIT, m_szName);
GetDlgItemText(IDC_DESCRIPTION_EDIT, m_szDescription);
LRESULT lRes = SendDlgItemMessage(IDC_MULTI_LEVEL_CHECK, BM_GETCHECK, 0, 0);
if (lRes == BST_CHECKED)
{
m_bMultiLevel = TRUE;
}
else
{
m_bMultiLevel = FALSE;
}
//
// Trim white space
//
m_szName.TrimLeft();
m_szName.TrimRight();
if (wcscmp(m_szOriginalName, m_szName) != 0 || m_bImportQuery)
{
CUINode* pDupNode = NULL;
if (!m_pFavNode->IsUniqueName(m_szName, &pDupNode))
{
CString szFormatMsg;
VERIFY(szFormatMsg.LoadString(IDS_ERRMSG_NOT_UNIQUE_QUERY_NAME));
CString szErrMsg;
szErrMsg.Format(szFormatMsg, m_szName);
CString szTitle;
VERIFY(szTitle.LoadString(IDS_DSSNAPINNAME));
MessageBox(szErrMsg, szTitle, MB_OK | MB_ICONSTOP);
//
// Set the focus to the name field and select all the text
//
GetDlgItem(IDC_NAME_EDIT)->SetFocus();
SendDlgItemMessage(IDC_NAME_EDIT, EM_SETSEL, 0, -1);
return;
}
}
if (m_bLastLogonFilter)
{
m_pQueryNode->SetLastLogonQuery(m_dwLastLogonData);
}
else
{
m_pQueryNode->SetLastLogonQuery(static_cast<DWORD>(-1));
}
m_pQueryNode->SetQueryString(m_szQueryFilter);
m_pQueryNode->SetName(m_szName);
m_pQueryNode->SetDesc(m_szDescription);
m_pQueryNode->SetRootPath(m_szQueryRoot);
m_pQueryNode->SetOneLevel((m_bMultiLevel == BST_CHECKED) ? FALSE : TRUE);
m_pQueryNode->SetQueryPersist(m_pPersistQueryImpl);
}
}
CHelpDialog::OnOK();
}
BOOL CQueryDialog::OnNeedToolTipText(UINT, NMHDR* pTTTStruct, LRESULT* /*ignored*/)
{
BOOL bRes = FALSE;
TOOLTIPTEXT* pTTText = reinterpret_cast<TOOLTIPTEXT*>(pTTTStruct);
if (pTTText != NULL)
{
if (pTTText->uFlags & TTF_IDISHWND)
{
UINT nCtrlID = ::GetDlgCtrlID((HWND)pTTText->hdr.idFrom);
if (nCtrlID == IDC_ROOT_EDIT)
{
pTTText->lpszText = (LPWSTR)(LPCWSTR)m_szQueryRoot;
bRes = TRUE;
}
}
}
return bRes;
}
void CQueryDialog::OnEditQuery()
{
CWaitCursor wait;
CLIPFORMAT cfDsQueryParams = (CLIPFORMAT)::RegisterClipboardFormat(CFSTR_DSQUERYPARAMS);
//
// create a query object
//
HRESULT hr;
CComPtr<ICommonQuery> spCommonQuery;
hr = ::CoCreateInstance(CLSID_CommonQuery, NULL, CLSCTX_INPROC_SERVER,
IID_ICommonQuery, (PVOID *)&spCommonQuery);
if (FAILED(hr))
{
ReportMessageEx(GetSafeHwnd(), IDS_ERRMSG_NO_DSQUERYUI);
return;
}
//
// setup structs to make the query
//
DSQUERYINITPARAMS dqip;
OPENQUERYWINDOW oqw;
ZeroMemory(&dqip, sizeof(DSQUERYINITPARAMS));
ZeroMemory(&oqw, sizeof(OPENQUERYWINDOW));
dqip.cbStruct = sizeof(dqip);
dqip.dwFlags = DSQPF_NOSAVE | DSQPF_SHOWHIDDENOBJECTS |
DSQPF_ENABLEADMINFEATURES;
dqip.pDefaultScope = NULL;
CString szServerName = m_pComponentData->GetBasePathsInfo()->GetServerName();
if (!szServerName.IsEmpty())
{
dqip.dwFlags |= DSQPF_HASCREDENTIALS;
dqip.pServer = (PWSTR)(PCWSTR)szServerName;
}
oqw.cbStruct = sizeof(oqw);
oqw.dwFlags = OQWF_OKCANCEL | OQWF_DEFAULTFORM | OQWF_SHOWOPTIONAL | /*OQWF_REMOVEFORMS |*/
OQWF_REMOVESCOPES | OQWF_SAVEQUERYONOK | OQWF_HIDEMENUS | OQWF_HIDESEARCHUI;
if (!m_pPersistQueryImpl->IsEmpty())
{
oqw.dwFlags |= OQWF_LOADQUERY;
}
oqw.clsidHandler = CLSID_DsQuery;
oqw.pHandlerParameters = &dqip;
oqw.clsidDefaultForm = CLSID_DSAdminQueryUIForm;
//
// set the IPersistQuery pointer (smart pointer)
//
CComPtr<IPersistQuery> spIPersistQuery;
hr = m_pPersistQueryImpl->QueryInterface(IID_IPersistQuery, (void**)&spIPersistQuery);
if (FAILED(hr))
{
int iRes = ReportMessageEx(GetSafeHwnd(), IDS_ERRMSG_NO_PERSIST_QUERYUI, MB_OKCANCEL | MB_ICONINFORMATION);
if (iRes == IDCANCEL)
{
return;
}
}
//
// now smart pointer has refcount=1 for it lifetime
//
oqw.pPersistQuery = spIPersistQuery;
//
// Get the HWND of the current dialog
//
HWND hWnd = GetSafeHwnd();
//
// make the call to get the query displayed
//
CComPtr<IDataObject> spQueryResultDataObject;
hr = spCommonQuery->OpenQueryWindow(hWnd, &oqw, &spQueryResultDataObject);
if (SUCCEEDED(hr) && spQueryResultDataObject != NULL)
{
//
// retrieve the query string from the data object
//
FORMATETC fmte = {cfDsQueryParams, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL};
STGMEDIUM medium = {TYMED_NULL, NULL, NULL};
hr = spQueryResultDataObject->GetData(&fmte, &medium);
if (SUCCEEDED(hr)) // we have data
{
//
// get the query string
//
LPDSQUERYPARAMS pDsQueryParams = (LPDSQUERYPARAMS)medium.hGlobal;
LPWSTR pwszFilter = (LPWSTR)ByteOffset(pDsQueryParams, pDsQueryParams->offsetQuery);
CString szTempFilter = pwszFilter;
//
// Check to see if we received a "special" query string
//
if (pDsQueryParams->dwFlags & DSQF_LAST_LOGON_QUERY)
{
m_bLastLogonFilter = TRUE;
m_dwLastLogonData = pDsQueryParams->dwReserved;
LARGE_INTEGER li;
GetCurrentTimeStampMinusInterval(m_dwLastLogonData, &li);
CString szTimeStamp;
litow(li, szTimeStamp);
szTempFilter.Format(L"%s(lastLogonTimestamp<=%s)", szTempFilter, szTimeStamp);
}
else
{
m_bLastLogonFilter = FALSE;
m_dwLastLogonData = 0;
}
::ReleaseStgMedium(&medium);
// REVIEW_MARCOC: this is a hack waiting for Diz to fix it...
// the query string should be a well formed expression. Period
// the query string is in the form (<foo>)(<bar>)...
// if more of one token, need to wrap as (& (<foo>)(<bar>)...)
PWSTR pChar = (LPWSTR)(LPCWSTR)szTempFilter;
int nLeftPar = 0;
while (*pChar != NULL)
{
if (*pChar == TEXT('('))
{
nLeftPar++;
if (nLeftPar > 1)
break;
}
pChar++;
}
if (nLeftPar > 1)
{
m_szQueryFilter.Format(_T("(&%s)"), (LPCWSTR)szTempFilter);
}
else
{
m_szQueryFilter = szTempFilter;
}
SetDirty();
}
else
{
//
// The user removed all query data from DSQUERYUI
//
//
// Remove filter data
//
m_szQueryFilter = L"";
m_bLastLogonFilter = FALSE;
m_dwLastLogonData = 0;
SetDirty();
}
}
SetQueryFilterDisplay();
return;
}
void CQueryDialog::SetQueryFilterDisplay()
{
if (m_bLastLogonFilter)
{
CString szTemp;
szTemp.LoadString(IDS_HIDE_LASTLOGON_QUERY);
SetDlgItemText(IDC_QUERY_STRING_EDIT, szTemp);
}
else
{
SetDlgItemText(IDC_QUERY_STRING_EDIT, m_szQueryFilter);
}
}
int SavedQueriesBrowseCallback(HWND, UINT uMsg, LPARAM lParam, LPARAM /*lpData*/)
{
int ret = 0;
switch (uMsg)
{
case DSBM_HELP:
{
TRACE(L"Browse Callback: msg is DSBM_HELP.\n");
LPHELPINFO pHelp = (LPHELPINFO) lParam;
TRACE(_T("CtrlId = %d, ContextId = 0x%x\n"),
pHelp->iCtrlId, pHelp->dwContextId);
if (!pHelp ||
pHelp->iCtrlId != DSBID_CONTAINERLIST)
{
ret = 0; // not handled
break;
}
::WinHelp((HWND)pHelp->hItemHandle,
DSADMIN_CONTEXT_HELP_FILE,
HELP_WM_HELP,
(DWORD_PTR)(LPTSTR)g_aHelpIDs_IDD_BROWSE_CONTAINER);
ret = 1;
}
break;
default:
ret = 0;
break;
}
return ret;
}
void CQueryDialog::OnBrowse()
{
DWORD result;
CString szBrowseTitle;
VERIFY(szBrowseTitle.LoadString(IDS_QUERY_BROWSE_TITLE));
CString szBrowseCaption;
VERIFY(szBrowseCaption.LoadString(IDS_QUERY_BROWSE_CAPTION));
WCHAR szPath[2 * MAX_PATH+1];
//
// Get the root of the console
CString szDNC = m_pComponentData->GetBasePathsInfo()->GetDefaultRootNamingContext();
CString szRootPath;
m_pComponentData->GetBasePathsInfo()->ComposeADsIPath(szRootPath, szDNC);
DSBROWSEINFO dsbi;
::ZeroMemory( &dsbi, sizeof(dsbi) );
dsbi.hwndOwner = GetSafeHwnd();
dsbi.cbStruct = sizeof (DSBROWSEINFO);
dsbi.pszCaption = (LPWSTR)((LPCWSTR)szBrowseTitle);
dsbi.pszTitle = (LPWSTR)((LPCWSTR)szBrowseCaption);
dsbi.pszRoot = szRootPath;
dsbi.pszPath = szPath;
dsbi.cchPath = ((2 * MAX_PATH + 1) / sizeof(WCHAR));
dsbi.dwFlags = DSBI_INCLUDEHIDDEN | DSBI_RETURN_FORMAT;
dsbi.pfnCallback = SavedQueriesBrowseCallback;
dsbi.lParam = 0;
dsbi.dwReturnFormat = ADS_FORMAT_X500;
result = DsBrowseForContainer( &dsbi );
if ( result == IDOK )
{
//
// returns -1, 0, IDOK or IDCANCEL
// get path from BROWSEINFO struct, put in text edit field
//
TRACE(_T("returned from DS Browse successfully with:\n %s\n"),
dsbi.pszPath);
CPathCracker pathCracker;
HRESULT hr = pathCracker.Set(CComBSTR(dsbi.pszPath), ADS_SETTYPE_FULL);
if (SUCCEEDED(hr))
{
hr = pathCracker.SetDisplayType(ADS_DISPLAY_FULL);
if (SUCCEEDED(hr))
{
CComBSTR bstrDN;
hr = pathCracker.Retrieve(ADS_FORMAT_X500_DN, &bstrDN);
if (SUCCEEDED(hr))
{
SetQueryRoot(bstrDN);
SetDirty();
}
}
}
}
}
void CQueryDialog::SetQueryRoot(PCWSTR pszPath)
{
m_szQueryRoot = pszPath;
CPathCracker pathCracker;
HRESULT hr = pathCracker.Set(CComBSTR(m_szQueryRoot), ADS_SETTYPE_DN);
if (SUCCEEDED(hr))
{
hr = pathCracker.SetDisplayType(ADS_DISPLAY_VALUE_ONLY);
if (SUCCEEDED(hr))
{
CComBSTR bstrDisplayPath;
hr = pathCracker.GetElement(0, &bstrDisplayPath);
if (SUCCEEDED(hr))
{
CString szDisplayString;
szDisplayString.Format(L"...\\%s", bstrDisplayPath);
SetDlgItemText(IDC_ROOT_EDIT, szDisplayString);
}
else
{
SetDlgItemText(IDC_ROOT_EDIT, m_szQueryRoot);
}
}
else
{
SetDlgItemText(IDC_ROOT_EDIT, m_szQueryRoot);
}
}
else
{
SetDlgItemText(IDC_ROOT_EDIT, m_szQueryRoot);
}
}
void CQueryDialog::OnMultiLevelChange()
{
SetDirty();
}
void CQueryDialog::OnNameChange()
{
GetDlgItemText(IDC_NAME_EDIT, m_szName);
if (m_szName.IsEmpty())
{
SetDirty(FALSE);
}
else
{
SetDirty();
}
}
void CQueryDialog::OnDescriptionChange()
{
SetDirty();
}
///////////////////////////////////////////////////////////////////////////
// CFavoritesNodePropertyPage
BEGIN_MESSAGE_MAP(CFavoritesNodePropertyPage, CHelpPropertyPage)
ON_EN_CHANGE(IDC_DESCRIPTION_EDIT, OnDescriptionChange)
ON_WM_DESTROY()
END_MESSAGE_MAP()
void CFavoritesNodePropertyPage::DoContextHelp(HWND hWndControl)
{
if (hWndControl)
{
::WinHelp(hWndControl,
DSADMIN_CONTEXT_HELP_FILE,
HELP_WM_HELP,
(DWORD_PTR)(LPTSTR)g_aHelpIDs_IDD_FAVORITES_PROPERTY_PAGE);
}
}
BOOL CFavoritesNodePropertyPage::OnInitDialog()
{
CHelpPropertyPage::OnInitDialog();
m_szOldDescription = m_pFavNode->GetDesc();
SetDlgItemText(IDC_CN, m_pFavNode->GetName());
SetDlgItemText(IDC_DESCRIPTION_EDIT, m_szOldDescription);
return FALSE;
}
void CFavoritesNodePropertyPage::OnDescriptionChange()
{
CString szNewDescription;
GetDlgItemText(IDC_DESCRIPTION_EDIT, szNewDescription);
if (szNewDescription == m_szOldDescription)
{
SetModified(FALSE);
}
else
{
SetModified(TRUE);
}
}
BOOL CFavoritesNodePropertyPage::OnApply()
{
BOOL bRet = TRUE;
CString szNewDescription;
GetDlgItemText(IDC_DESCRIPTION_EDIT, szNewDescription);
if (szNewDescription == m_szOldDescription)
{
return TRUE;
}
else
{
m_pFavNode->SetDesc(szNewDescription);
if (m_lNotifyHandle != NULL && m_pDataObject != NULL)
{
MMCPropertyChangeNotify(m_lNotifyHandle, (LPARAM)m_pDataObject);
}
}
m_szOldDescription = szNewDescription;
return bRet;
}
void CFavoritesNodePropertyPage::OnDestroy()
{
m_pComponentData->SheetUnlockCookie(m_pFavNode);
}