889 lines
25 KiB
C++
889 lines
25 KiB
C++
//+---------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation 1996-2001.
|
|
//
|
|
// File: AMember.cpp
|
|
//
|
|
// Contents: implementation of CAttrMember
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
#include "stdafx.h"
|
|
#include "wsecmgr.h"
|
|
#include "snapmgr.h"
|
|
#include "resource.h"
|
|
#include "chklist.h"
|
|
#include "util.h"
|
|
#include "getuser.h"
|
|
#include "snapmgr.h"
|
|
#include "resource.h"
|
|
#include "AMember.h"
|
|
#include "wrapper.h"
|
|
|
|
|
|
#ifdef _DEBUG
|
|
#define new DEBUG_NEW
|
|
#undef THIS_FILE
|
|
static char THIS_FILE[] = __FILE__;
|
|
#endif
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CAttrMember property page
|
|
|
|
IMPLEMENT_DYNCREATE(CAttrMember, CSelfDeletingPropertyPage)
|
|
|
|
CAttrMember::CAttrMember() : CSelfDeletingPropertyPage(CAttrMember::IDD)
|
|
{
|
|
//{{AFX_DATA_INIT(CAttrMember)
|
|
m_fDefineInDatabase = FALSE;
|
|
m_strHeader = _T("");
|
|
//}}AFX_DATA_INIT
|
|
|
|
m_psp.pszTemplate = MAKEINTRESOURCE(IDD_ATTR_GROUP);
|
|
m_psp.dwFlags |= PSP_PREMATURE;
|
|
m_pMergeList = NULL;
|
|
m_fProcessing = FALSE;
|
|
m_fInitialized = FALSE;
|
|
m_bNoMembers = FALSE;
|
|
m_bDirty=false;
|
|
m_fOriginalDefineInDatabase=FALSE;
|
|
m_bAlias=FALSE;
|
|
m_dwType=0;
|
|
CAttribute::m_nDialogs++;
|
|
}
|
|
|
|
CAttrMember::~CAttrMember()
|
|
{
|
|
if ( m_pMergeList ) {
|
|
SceFreeMemory(m_pMergeList, SCE_STRUCT_NAME_STATUS_LIST);
|
|
m_pMergeList = NULL;
|
|
}
|
|
m_pData->Release();
|
|
CAttribute::m_nDialogs--;
|
|
}
|
|
|
|
void CAttrMember::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CSelfDeletingPropertyPage::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CAttrMember)
|
|
DDX_Check(pDX, IDC_DEFINE_GROUP, m_fDefineInDatabase);
|
|
DDX_Text(pDX, IDC_HEADER, m_strHeader);
|
|
DDX_Control(pDX, IDC_NO_MEMBERS,m_eNoMembers);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
|
|
BEGIN_MESSAGE_MAP(CAttrMember, CSelfDeletingPropertyPage)
|
|
//{{AFX_MSG_MAP(CAttrMember)
|
|
ON_BN_CLICKED(IDC_DEFINE_GROUP, OnDefineInDatabase)
|
|
ON_BN_CLICKED(IDC_ADD, OnAdd)
|
|
//}}AFX_MSG_MAP
|
|
ON_NOTIFY(CLN_CLICK, IDC_MEMBERS, OnClickMembers)
|
|
ON_MESSAGE(WM_HELP, OnHelp)
|
|
END_MESSAGE_MAP()
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CAttrMember message handlers
|
|
|
|
void CAttrMember::OnDefineInDatabase()
|
|
{
|
|
if (m_fProcessing)
|
|
return;
|
|
|
|
m_fProcessing = TRUE;
|
|
|
|
//
|
|
// for some strange reason the DDX isn't getting this BOOL set, so just do it
|
|
// here which is basically the same thing
|
|
//
|
|
m_fDefineInDatabase = ( ((CButton *)GetDlgItem(IDC_DEFINE_GROUP))->GetCheck() == 1 );
|
|
|
|
if (m_fDefineInDatabase) {
|
|
(GetDlgItem(IDC_ADD))->EnableWindow(TRUE);
|
|
//
|
|
// use non CWnd calls for the checklist control
|
|
//
|
|
::SendMessage(::GetDlgItem(this->m_hWnd, IDC_MEMBERS), WM_ENABLE, (WPARAM) TRUE, (LPARAM) 0);
|
|
} else {
|
|
(GetDlgItem(IDC_ADD))->EnableWindow(FALSE);
|
|
//
|
|
// use non CWnd calls for the checklist control
|
|
//
|
|
::SendMessage(::GetDlgItem(this->m_hWnd, IDC_MEMBERS), WM_ENABLE, (WPARAM) FALSE, (LPARAM) 0);
|
|
}
|
|
|
|
SetModified(TRUE);
|
|
|
|
//
|
|
// Tell our siblings the m_fDefineInDatabase has changed
|
|
//
|
|
if (m_pAttrMember) {
|
|
m_pAttrMember->SetDefineInDatabase(m_fDefineInDatabase);
|
|
}
|
|
|
|
m_fProcessing = FALSE;
|
|
}
|
|
|
|
|
|
void CAttrMember::OnAdd()
|
|
{
|
|
CGetUser gu;
|
|
HRESULT hr=S_OK;
|
|
DWORD nFlag;
|
|
BOOL fModify=FALSE; //Raid #497350, yanggao, 11/20/2001, make sure a new item is added this time.
|
|
|
|
if ( GROUP_MEMBERS == m_dwType )
|
|
nFlag = SCE_SHOW_USERS | SCE_SHOW_DOMAINGROUPS;
|
|
else {
|
|
|
|
nFlag = SCE_SHOW_GROUPS | SCE_SHOW_ALIASES; // NT5 DS, nested groups
|
|
}
|
|
|
|
nFlag |= SCE_SHOW_SCOPE_ALL | SCE_SHOW_DIFF_MODE_OFF_DC;
|
|
if (gu.Create(GetSafeHwnd(),nFlag)) {
|
|
|
|
PSCE_NAME_STATUS_LIST pList, pLast=NULL;
|
|
LRESULT iItem;
|
|
bool bFound;
|
|
|
|
PSCE_NAME_LIST pName = gu.GetUsers();
|
|
CWnd *pCheckList;
|
|
pCheckList = GetDlgItem(IDC_MEMBERS);
|
|
|
|
if (pName && m_bNoMembers) {
|
|
m_bNoMembers = FALSE;
|
|
m_eNoMembers.ShowWindow(SW_HIDE);
|
|
pCheckList->ShowWindow(SW_SHOW);
|
|
}
|
|
while(pName) {
|
|
if ( pName->Name ) {
|
|
// Is the new name in m_pMerged yet?
|
|
pList = m_pMergeList;
|
|
pLast = NULL;
|
|
iItem = 0;
|
|
|
|
bFound = false;
|
|
while(pList) {
|
|
// If so, then make sure its "Template" box is checked
|
|
if (lstrcmp(pList->Name,pName->Name) == 0) {
|
|
if (!(pCheckList->SendMessage(CLM_GETSTATE,MAKELONG(iItem,1)) & CLST_CHECKED)) {
|
|
m_bDirty = true;
|
|
pCheckList->SendMessage(CLM_SETSTATE,MAKELONG(iItem,1),CLST_CHECKED);
|
|
fModify = true;
|
|
}
|
|
bFound = true;
|
|
break;
|
|
}
|
|
pLast = pList;
|
|
pList = pList->Next;
|
|
iItem++;
|
|
}
|
|
|
|
// Otherwise add it both to m_pMerged and to the CheckList
|
|
if (!bFound) {
|
|
|
|
PSCE_NAME_STATUS_LIST pNewNode;
|
|
|
|
pNewNode = (PSCE_NAME_STATUS_LIST)LocalAlloc(0,sizeof(SCE_NAME_STATUS_LIST));
|
|
if ( pNewNode ) {
|
|
|
|
pNewNode->Name = (LPTSTR)LocalAlloc(0, (lstrlen(pName->Name)+1)*sizeof(TCHAR));
|
|
if ( pNewNode->Name ) {
|
|
// This may not be a safe usage. pNewNode->Name is PWSTR, using WCHAR instead of TCHAR,
|
|
// and lstrlen dose not handle not null-terminated pointer. Consider fix
|
|
lstrcpy(pNewNode->Name, pName->Name);
|
|
pNewNode->Next = NULL;
|
|
pNewNode->Status = MERGED_TEMPLATE;
|
|
} else {
|
|
LocalFree(pNewNode);
|
|
pNewNode = NULL;
|
|
}
|
|
}
|
|
if ( pNewNode ) {
|
|
if ( pLast )
|
|
pLast->Next = pNewNode;
|
|
else
|
|
m_pMergeList = pNewNode;
|
|
pLast = pNewNode;
|
|
|
|
iItem = pCheckList->SendMessage(CLM_ADDITEM,(WPARAM)pLast->Name,(LPARAM)pLast->Name);
|
|
pCheckList->SendMessage(CLM_SETSTATE,MAKELONG(iItem,1),CLST_CHECKED);
|
|
pCheckList->SendMessage(CLM_SETSTATE,MAKELONG(iItem,2),CLST_DISABLED);
|
|
m_bDirty = true;
|
|
fModify = true;
|
|
} else {
|
|
hr = E_OUTOFMEMORY;
|
|
ASSERT(FALSE);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
pName = pName->Next;
|
|
}
|
|
}
|
|
|
|
if (fModify) {
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
if ( FAILED(hr) ) {
|
|
CString str;
|
|
str.LoadString(IDS_CANT_ADD_MEMBER);
|
|
AfxMessageBox(str);
|
|
}
|
|
|
|
}
|
|
|
|
/*---------------------------------------------------------------------
|
|
|
|
Method: OnInitDialog
|
|
|
|
Synopsis: Initialize the check list with members/memberof data
|
|
|
|
Arguments: None
|
|
|
|
Returns: TRUE = initialized successfully
|
|
|
|
----------------------------------------------------------------------*/
|
|
BOOL CAttrMember::OnInitDialog()
|
|
{
|
|
CSelfDeletingPropertyPage::OnInitDialog();
|
|
|
|
|
|
PSCE_NAME_STATUS_LIST pItem;
|
|
HWND hCheckList;
|
|
LRESULT nItem;
|
|
PSCE_GROUP_MEMBERSHIP pgmTemplate;
|
|
PSCE_GROUP_MEMBERSHIP pgmInspect;
|
|
PSCE_NAME_LIST pnlTemplate=NULL;
|
|
PSCE_NAME_LIST pnlInspect=NULL;
|
|
PEDITTEMPLATE pet = NULL;
|
|
CString str;
|
|
|
|
UpdateData(TRUE);
|
|
if (GROUP_MEMBER_OF == m_dwType) {
|
|
str.LoadString(IDS_NO_MEMBER_OF);
|
|
} else {
|
|
str.LoadString(IDS_NO_MEMBERS);
|
|
}
|
|
m_eNoMembers.SetWindowText(str);
|
|
|
|
pgmInspect = (PSCE_GROUP_MEMBERSHIP) m_pData->GetID(); // last inspection saved in ID field
|
|
|
|
if ( NULL == pgmInspect ) { // last inspection can't be NULL
|
|
return TRUE;
|
|
}
|
|
|
|
m_fOriginalDefineInDatabase = m_fDefineInDatabase = FALSE;
|
|
|
|
//
|
|
// Try to find the base group in the template
|
|
//
|
|
pet = m_pSnapin->GetTemplate(GT_COMPUTER_TEMPLATE);
|
|
if ( NULL == pet ) {
|
|
return TRUE;
|
|
}
|
|
|
|
for (pgmTemplate=pet->pTemplate->pGroupMembership;
|
|
pgmTemplate!=NULL;
|
|
pgmTemplate=pgmTemplate->Next) {
|
|
|
|
if ( _wcsicmp(pgmTemplate->GroupName, pgmInspect->GroupName) == 0 ) {
|
|
//
|
|
// If the group is in the template that means it is defined.... duh
|
|
//
|
|
m_fOriginalDefineInDatabase = m_fDefineInDatabase = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// find the name lists to display
|
|
//
|
|
if ( pgmTemplate ) {
|
|
|
|
if (GROUP_MEMBER_OF == m_dwType) {
|
|
pnlTemplate = pgmTemplate->pMemberOf;
|
|
} else {
|
|
pnlTemplate = pgmTemplate->pMembers;
|
|
}
|
|
}
|
|
|
|
if ((LONG_PTR)ULongToPtr(SCE_NO_VALUE) != (LONG_PTR) pgmInspect &&
|
|
pgmInspect ) {
|
|
|
|
if (GROUP_MEMBER_OF == m_dwType) {
|
|
pnlInspect = pgmInspect->pMemberOf;
|
|
} else {
|
|
pnlInspect = pgmInspect->pMembers;
|
|
}
|
|
}
|
|
|
|
m_pMergeList = MergeNameStatusList(pnlTemplate, pnlInspect);
|
|
|
|
pItem = m_pMergeList;
|
|
hCheckList = ::GetDlgItem(m_hWnd,IDC_MEMBERS);
|
|
|
|
::SendMessage(hCheckList, CLM_RESETCONTENT, 0, 0 );
|
|
::SendMessage(hCheckList,CLM_SETCOLUMNWIDTH,0,60);
|
|
|
|
if (!pItem) {
|
|
m_bNoMembers = TRUE;
|
|
m_eNoMembers.ShowWindow(SW_SHOW);
|
|
m_eNoMembers.EnableWindow(FALSE); //Raid #469732, Yanggao
|
|
::ShowWindow(hCheckList,SW_HIDE);
|
|
}
|
|
|
|
while(pItem) {
|
|
//
|
|
// Store the name of the item in the item data so we can retrieve it later
|
|
//
|
|
nItem = ::SendMessage(hCheckList,CLM_ADDITEM,(WPARAM) pItem->Name,(LPARAM) pItem->Name);
|
|
::SendMessage(hCheckList,CLM_SETSTATE,MAKELONG(nItem,1),
|
|
((pItem->Status & MERGED_TEMPLATE) ? CLST_CHECKED : CLST_UNCHECKED));
|
|
::SendMessage(hCheckList,CLM_SETSTATE,MAKELONG(nItem,2),
|
|
((pItem->Status & MERGED_INSPECT) ? CLST_CHECKDISABLED : CLST_DISABLED));
|
|
pItem = pItem->Next;
|
|
}
|
|
|
|
if ( GROUP_MEMBER_OF == m_dwType ) {
|
|
|
|
m_bAlias = TRUE;
|
|
|
|
} else {
|
|
m_bAlias = FALSE;
|
|
}
|
|
|
|
CWnd *cwnd = GetDlgItem(IDC_ADD);
|
|
if ( cwnd ) {
|
|
cwnd->EnableWindow(!m_bAlias);
|
|
}
|
|
|
|
CButton *pButton = (CButton *) GetDlgItem(IDC_DEFINE_GROUP);
|
|
if (pButton) {
|
|
pButton->SetCheck(m_fDefineInDatabase);
|
|
}
|
|
|
|
OnDefineInDatabase();
|
|
|
|
SetModified(FALSE);
|
|
|
|
m_fInitialized = TRUE;
|
|
|
|
return TRUE; // return TRUE unless you set the focus to a control
|
|
// EXCEPTION: OCX Property Pages should return FALSE
|
|
}
|
|
|
|
|
|
/*---------------------------------------------------------------------
|
|
|
|
Method: SetDefineInDatabase
|
|
|
|
Synopsis: Sets the m_fDefineInDatabase member var, and UI accorsingly
|
|
|
|
Arguments: fDefineInDatabase
|
|
|
|
Returns: None
|
|
|
|
----------------------------------------------------------------------*/
|
|
void CAttrMember::SetDefineInDatabase(BOOL fDefineInDatabase)
|
|
{
|
|
if (!m_fInitialized)
|
|
return;
|
|
|
|
if (m_fProcessing)
|
|
return;
|
|
|
|
m_fDefineInDatabase = fDefineInDatabase;
|
|
|
|
CButton *pButton = (CButton *) GetDlgItem(IDC_DEFINE_GROUP);
|
|
if (pButton) {
|
|
pButton->SetCheck(fDefineInDatabase);
|
|
}
|
|
|
|
OnDefineInDatabase();
|
|
}
|
|
|
|
|
|
/*---------------------------------------------------------------------
|
|
|
|
Method: SetSibling
|
|
|
|
Synopsis: Sets the pointer to the Sibling class
|
|
|
|
Arguments: pAttrMember
|
|
|
|
Returns: None
|
|
|
|
----------------------------------------------------------------------*/
|
|
void CAttrMember::SetSibling(CAttrMember *pAttrMember)
|
|
{
|
|
m_pAttrMember = pAttrMember;
|
|
}
|
|
|
|
|
|
/*---------------------------------------------------------------------
|
|
|
|
Method: Initialize
|
|
|
|
Synopsis: Initialize member data
|
|
|
|
Arguments: pData - the CResult data record
|
|
|
|
Returns: None
|
|
|
|
----------------------------------------------------------------------*/
|
|
void CAttrMember::Initialize(CResult * pData)
|
|
{
|
|
m_pData = pData;
|
|
pData->AddRef();
|
|
|
|
m_bDirty=false;
|
|
}
|
|
|
|
|
|
/*---------------------------------------------------------------------
|
|
|
|
Method: GetGroupInTemplate
|
|
|
|
Synopsis: Returns a pointer to the SCE_GROUP_MEMBERSHIP structure
|
|
that is being changed within the template
|
|
|
|
Arguments: None
|
|
|
|
Returns: Pointer the group being modified.
|
|
|
|
----------------------------------------------------------------------*/
|
|
PSCE_GROUP_MEMBERSHIP CAttrMember::GetGroupInTemplate()
|
|
{
|
|
PSCE_GROUP_MEMBERSHIP pgmTemplate;
|
|
PSCE_GROUP_MEMBERSHIP pgmInspect;
|
|
PEDITTEMPLATE pet;
|
|
|
|
pgmInspect = (PSCE_GROUP_MEMBERSHIP) m_pData->GetID(); // last inspection saved in ID field
|
|
|
|
if ( NULL == pgmInspect ) { // last inspection can't be NULL
|
|
return NULL;
|
|
}
|
|
|
|
pet = m_pSnapin->GetTemplate(GT_COMPUTER_TEMPLATE);
|
|
if ( NULL == pet ) {
|
|
return NULL;
|
|
}
|
|
|
|
for (pgmTemplate=pet->pTemplate->pGroupMembership;
|
|
pgmTemplate!=NULL;
|
|
pgmTemplate=pgmTemplate->Next) {
|
|
|
|
if ( _wcsicmp(pgmTemplate->GroupName, pgmInspect->GroupName) == 0 ) {
|
|
return pgmTemplate;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/*---------------------------------------------------------------------
|
|
|
|
Method: SetMemberType
|
|
|
|
Synopsis: Initialize page data depending on the type
|
|
|
|
Arguments: nType - the page type for members of memberof
|
|
|
|
Returns: None
|
|
|
|
----------------------------------------------------------------------*/
|
|
void CAttrMember::SetMemberType(DWORD nType)
|
|
{
|
|
m_dwType = nType;
|
|
if (GROUP_MEMBERS == nType) {
|
|
m_strHeader.LoadString(IDS_GROUP_MEMBERS_HEADER);
|
|
m_strPageTitle.LoadString(IDS_GROUP_MEMBERS_PAGE_TITLE);
|
|
} else {
|
|
m_strHeader.LoadString(IDS_GROUP_MEMBER_OF_HEADER);
|
|
m_strPageTitle.LoadString(IDS_GROUP_MEMBER_OF_PAGE_TITLE);
|
|
}
|
|
m_psp.dwFlags |= PSP_USETITLE;
|
|
m_psp.pszTitle = (LPCTSTR) m_strPageTitle;
|
|
|
|
}
|
|
|
|
void CAttrMember::SetSnapin(CSnapin * pSnapin)
|
|
{
|
|
m_pSnapin = pSnapin;
|
|
}
|
|
|
|
void CAttrMember::OnCancel()
|
|
{
|
|
if ( m_pMergeList ) {
|
|
SceFreeMemory(m_pMergeList,SCE_STRUCT_NAME_STATUS_LIST);
|
|
}
|
|
m_pMergeList = NULL;
|
|
}
|
|
|
|
|
|
void CAttrMember::OnClickMembers(NMHDR *pNM, LRESULT *pResult)
|
|
{
|
|
SetModified(TRUE);
|
|
//
|
|
// If no items are checked then show the no members edit box instead
|
|
//
|
|
CWnd *pCheckList;
|
|
int iItem;
|
|
int nItem;
|
|
PNM_CHECKLIST pNMCheckList;
|
|
|
|
pNMCheckList = (PNM_CHECKLIST) pNM;
|
|
if (pNMCheckList->dwState & CLST_CHECKED) {
|
|
//
|
|
// They checked something, so obviously something is checked
|
|
//
|
|
return;
|
|
}
|
|
|
|
pCheckList = GetDlgItem(IDC_MEMBERS);
|
|
nItem = (int) pCheckList->SendMessage(CLM_GETITEMCOUNT);
|
|
for(iItem=0;iItem<nItem;iItem++) {
|
|
if ((pCheckList->SendMessage(CLM_GETSTATE,MAKELONG(iItem,1)) & CLST_CHECKED) ||
|
|
(pCheckList->SendMessage(CLM_GETSTATE,MAKELONG(iItem,2)) & CLST_CHECKED)) {
|
|
//
|
|
// Something's checked, so abort
|
|
//
|
|
return;
|
|
}
|
|
}
|
|
//
|
|
// Nothing checked. Swap to the no members edit box
|
|
//
|
|
m_bNoMembers = TRUE;
|
|
m_eNoMembers.ShowWindow(SW_SHOW);
|
|
pCheckList->ShowWindow(SW_HIDE);
|
|
}
|
|
|
|
|
|
|
|
BOOL CAttrMember::OnApply()
|
|
{
|
|
int iItem;
|
|
PEDITTEMPLATE pet=NULL;
|
|
PSCE_PROFILE_INFO pspi=NULL;
|
|
PSCE_NAME_STATUS_LIST pIndex;
|
|
CWnd *pCheckList;
|
|
PSCE_NAME_LIST pTemplate=NULL;
|
|
PSCE_NAME_LIST pInspect=NULL;
|
|
PSCE_NAME_LIST pDeleteNameList = NULL;
|
|
PSCE_GROUP_MEMBERSHIP pSetting = NULL;
|
|
PSCE_GROUP_MEMBERSHIP pBaseGroup = NULL;
|
|
PSCE_GROUP_MEMBERSHIP pFindGroup = NULL;
|
|
PSCE_GROUP_MEMBERSHIP pModifiedGroup = NULL;
|
|
|
|
LPCTSTR szGroupName = (LPCTSTR) (m_pData->GetAttr());
|
|
|
|
pCheckList = GetDlgItem(IDC_MEMBERS);
|
|
pIndex = m_pMergeList;
|
|
iItem = 0;
|
|
HRESULT hr=S_OK;
|
|
|
|
//
|
|
// if the fDefineInDatabase has changed it is definitely dirty
|
|
//
|
|
m_bDirty = ( m_bDirty || (m_fOriginalDefineInDatabase != m_fDefineInDatabase) );
|
|
|
|
//
|
|
// only create the name list if the group is going to be defined in the database
|
|
//
|
|
if (m_fDefineInDatabase) {
|
|
while(pIndex) {
|
|
|
|
if (pCheckList->SendMessage(CLM_GETSTATE,MAKELONG(iItem,1)) & CLST_CHECKED) {
|
|
|
|
if ( !(pIndex->Status & MERGED_TEMPLATE) ) {
|
|
m_bDirty = true;
|
|
}
|
|
|
|
if ( SCESTATUS_SUCCESS != SceAddToNameList(&pTemplate, pIndex->Name, lstrlen(pIndex->Name))) {
|
|
hr = E_FAIL;
|
|
break;
|
|
}
|
|
} else if ( pIndex->Status & MERGED_TEMPLATE ) {
|
|
m_bDirty = true;
|
|
}
|
|
|
|
pIndex = pIndex->Next;
|
|
iItem++;
|
|
}
|
|
}
|
|
|
|
if ( SUCCEEDED(hr) && m_bDirty) {
|
|
|
|
pBaseGroup = GetGroupInTemplate();
|
|
|
|
//
|
|
// Need to add the group to the template
|
|
//
|
|
if ( (!pBaseGroup || (LONG_PTR)pBaseGroup == (LONG_PTR)ULongToPtr(SCE_NO_VALUE)) &&
|
|
m_fDefineInDatabase) {
|
|
|
|
pBaseGroup = (PSCE_GROUP_MEMBERSHIP) LocalAlloc(LMEM_ZEROINIT,sizeof(SCE_GROUP_MEMBERSHIP));
|
|
|
|
if ( pBaseGroup && szGroupName ) {
|
|
pBaseGroup->GroupName = (PWSTR)LocalAlloc(0, (lstrlen(szGroupName)+1)*sizeof(TCHAR));
|
|
|
|
if ( pBaseGroup->GroupName ) {
|
|
// This may not be safe usage. Using WCHAR instead of TCHAR. Consider fix.
|
|
lstrcpy(pBaseGroup->GroupName,szGroupName);
|
|
//
|
|
// link the new structure to the pGroupMembership list
|
|
//
|
|
pet = m_pSnapin->GetTemplate(GT_COMPUTER_TEMPLATE,AREA_GROUP_MEMBERSHIP);
|
|
if (pet) {
|
|
pspi = pet->pTemplate;
|
|
} else {
|
|
pspi = NULL;
|
|
}
|
|
|
|
if ( pspi ) {
|
|
pBaseGroup->Next = pspi->pGroupMembership;
|
|
pspi->pGroupMembership = pBaseGroup;
|
|
|
|
pBaseGroup->pMembers = NULL;
|
|
pBaseGroup->pMemberOf = NULL;
|
|
|
|
} else {
|
|
//
|
|
// error
|
|
ASSERT(FALSE);
|
|
LocalFree(pBaseGroup->GroupName);
|
|
hr = E_FAIL;
|
|
}
|
|
} else {
|
|
//
|
|
// no memory
|
|
//
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
} else {
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
|
|
if ( FAILED(hr) && pBaseGroup ) {
|
|
LocalFree(pBaseGroup);
|
|
pBaseGroup = NULL;
|
|
}
|
|
|
|
pModifiedGroup = pBaseGroup;
|
|
|
|
//
|
|
// Need to remove the group from the template
|
|
//
|
|
} else if (pBaseGroup && !m_fDefineInDatabase) {
|
|
|
|
CString szGroupName;
|
|
szGroupName = pBaseGroup->GroupName;
|
|
pBaseGroup = NULL;
|
|
DeleteGroup(szGroupName);
|
|
//
|
|
// An existing group was modified
|
|
//
|
|
} else {
|
|
pModifiedGroup = pBaseGroup;
|
|
}
|
|
|
|
//
|
|
// get group address to change the status field in the last inspection
|
|
//
|
|
pSetting = (PSCE_GROUP_MEMBERSHIP)(m_pData->GetID());
|
|
|
|
int status;
|
|
if (GROUP_MEMBERS == m_dwType) {
|
|
|
|
if (pModifiedGroup != NULL) {
|
|
pDeleteNameList = pModifiedGroup->pMembers;
|
|
pModifiedGroup->pMembers = pTemplate;
|
|
}
|
|
|
|
if (NULL != pSetting ) {
|
|
if ( !((pSetting->Status & SCE_GROUP_STATUS_NOT_ANALYZED) ||
|
|
(pSetting->Status & SCE_GROUP_STATUS_ERROR_ANALYZED))) {
|
|
|
|
//
|
|
// set good, not configured, or mismatch
|
|
//
|
|
pSetting->Status &= ~SCE_GROUP_STATUS_NC_MEMBERS;
|
|
pSetting->Status &= ~SCE_GROUP_STATUS_MEMBERS_MISMATCH;
|
|
if (pModifiedGroup == NULL) {
|
|
pSetting->Status |= SCE_GROUP_STATUS_NC_MEMBERS;
|
|
} else if ( !SceCompareNameList(pTemplate, pSetting->pMembers) ) {
|
|
pSetting->Status |= SCE_GROUP_STATUS_MEMBERS_MISMATCH;
|
|
}
|
|
}
|
|
|
|
} else {
|
|
//
|
|
// else should NEVER occur
|
|
//
|
|
status = SCE_GROUP_STATUS_MEMBERS_MISMATCH;
|
|
}
|
|
|
|
} else {
|
|
//
|
|
// memberof
|
|
//
|
|
|
|
if (pModifiedGroup != NULL) {
|
|
pDeleteNameList = pModifiedGroup->pMemberOf;
|
|
pModifiedGroup->pMemberOf = pTemplate;
|
|
}
|
|
|
|
if ( pSetting ) {
|
|
if ( !((pSetting->Status & SCE_GROUP_STATUS_NOT_ANALYZED) ||
|
|
(pSetting->Status & SCE_GROUP_STATUS_ERROR_ANALYZED))) {
|
|
|
|
//
|
|
// set good, not configured, or mismatch
|
|
//
|
|
pSetting->Status &= ~SCE_GROUP_STATUS_NC_MEMBEROF;
|
|
pSetting->Status &= ~SCE_GROUP_STATUS_MEMBEROF_MISMATCH;
|
|
if (pModifiedGroup == NULL) {
|
|
pSetting->Status |= SCE_GROUP_STATUS_NC_MEMBEROF;
|
|
} else if ( !SceCompareNameList(pTemplate, pSetting->pMemberOf) ) {
|
|
pSetting->Status |= SCE_GROUP_STATUS_MEMBEROF_MISMATCH;
|
|
}
|
|
}
|
|
|
|
} else {
|
|
// else should NEVER occur
|
|
status = SCE_GROUP_STATUS_MEMBEROF_MISMATCH;
|
|
}
|
|
}
|
|
pTemplate = NULL;
|
|
|
|
SceFreeMemory(pDeleteNameList,SCE_STRUCT_NAME_LIST);
|
|
|
|
if ( pSetting ) {
|
|
status = pSetting->Status;
|
|
}
|
|
|
|
//
|
|
// update current record
|
|
//
|
|
// status
|
|
m_pData->SetStatus(GetGroupStatus(status, STATUS_GROUP_RECORD));
|
|
// members status
|
|
m_pData->SetBase(GetGroupStatus(status, STATUS_GROUP_MEMBERS));
|
|
// memberof status
|
|
m_pData->SetSetting(GetGroupStatus(status, STATUS_GROUP_MEMBEROF));
|
|
m_pData->Update(m_pSnapin);
|
|
//
|
|
// update the dirty flag in the template
|
|
//
|
|
pet = m_pSnapin->GetTemplate(GT_COMPUTER_TEMPLATE,AREA_GROUP_MEMBERSHIP);
|
|
if (pet) {
|
|
pet->SetDirty(AREA_GROUP_MEMBERSHIP);
|
|
}
|
|
|
|
} // failed
|
|
|
|
SceFreeMemory(pTemplate,SCE_STRUCT_NAME_LIST);
|
|
|
|
if ( FAILED(hr) ) {
|
|
CString str;
|
|
str.LoadString(IDS_SAVE_FAILED);
|
|
AfxMessageBox(str);
|
|
} else {
|
|
CancelToClose();
|
|
SetModified(TRUE);
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL CAttrMember::OnHelp(WPARAM wParam, LPARAM lParam)
|
|
{
|
|
const LPHELPINFO pHelpInfo = (LPHELPINFO)lParam;
|
|
if (pHelpInfo && pHelpInfo->iContextType == HELPINFO_WINDOW)
|
|
{
|
|
this->DoContextHelp ((HWND) pHelpInfo->hItemHandle);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void CAttrMember::DoContextHelp (HWND hWndControl)
|
|
{
|
|
// Display context help for a control
|
|
if ( !::WinHelp (
|
|
hWndControl,
|
|
GetSeceditHelpFilename(),
|
|
HELP_WM_HELP,
|
|
(DWORD_PTR) a214HelpIDs) )
|
|
{
|
|
|
|
}
|
|
}
|
|
|
|
|
|
void CAttrMember::DeleteGroup(const CString &szGroupName)
|
|
{
|
|
CSingleLock cSL(&m_CS, FALSE);
|
|
cSL.Lock();
|
|
|
|
PEDITTEMPLATE pet = NULL;
|
|
PSCE_PROFILE_INFO pspi = NULL;
|
|
PSCE_GROUP_MEMBERSHIP pFindGroup = NULL;
|
|
PSCE_GROUP_MEMBERSHIP pDeleteGroup = NULL;
|
|
|
|
pet = m_pSnapin->GetTemplate(GT_COMPUTER_TEMPLATE,AREA_GROUP_MEMBERSHIP);
|
|
if (pet) {
|
|
pspi = pet->pTemplate;
|
|
} else {
|
|
pspi = NULL;
|
|
}
|
|
|
|
if ( pspi ) {
|
|
|
|
//
|
|
// find the group in the template and remove it
|
|
//
|
|
pFindGroup = pspi->pGroupMembership;
|
|
|
|
if (pFindGroup == NULL)
|
|
return;
|
|
|
|
if (pFindGroup->GroupName == szGroupName) {
|
|
|
|
pspi->pGroupMembership = pFindGroup->Next;
|
|
pDeleteGroup = pFindGroup;
|
|
|
|
} else {
|
|
|
|
while (pFindGroup->Next && (pFindGroup->Next->GroupName != szGroupName)) {
|
|
pFindGroup = pFindGroup->Next;
|
|
}
|
|
|
|
if (pFindGroup->Next) {
|
|
|
|
pDeleteGroup = pFindGroup->Next;
|
|
pFindGroup->Next = pDeleteGroup->Next;
|
|
}
|
|
}
|
|
|
|
if (pDeleteGroup) {
|
|
LocalFree(pDeleteGroup->GroupName);
|
|
SceFreeMemory(pDeleteGroup->pMembers,SCE_STRUCT_NAME_LIST);
|
|
SceFreeMemory(pDeleteGroup->pMemberOf,SCE_STRUCT_NAME_LIST);
|
|
LocalFree(pDeleteGroup);
|
|
}
|
|
|
|
} else {
|
|
//
|
|
// error
|
|
//
|
|
ASSERT(FALSE);
|
|
}
|
|
}
|