/*++ Copyright (c) 1994-1998 Microsoft Corporation Module Name : security.cpp Abstract: FTP Security Property Page Author: Ronald Meijer (ronaldm) Project: Internet Services Manager Revision History: --*/ // // Include Files // #include "stdafx.h" #include "fscfg.h" #include "accessdl.h" #include "security.h" #ifdef _DEBUG #define new DEBUG_NEW #undef THIS_FILE static char THIS_FILE[] = __FILE__; #endif // // Needed for granted/denied icons // #include "..\comprop\resource.h" IMPLEMENT_DYNCREATE(CFtpSecurityPage, CInetPropertyPage) CFtpSecurityPage::CFtpSecurityPage( IN CInetPropertySheet * pSheet ) /*++ Routine Description: Constructor Arguments: CInetPropertySheet * pSheet : Sheet object Return Value: N/A --*/ : CInetPropertyPage( CFtpSecurityPage::IDD, pSheet, USE_DEFAULT_CAPTION, TRUE // Enable enhanced fonts ), m_ListBoxRes( IDB_ACCESS, m_list_IpAddresses.nBitmaps ), m_oblAccessList(), m_list_IpAddresses(TRUE), m_fIpDirty(FALSE), m_fOldDefaultGranted(TRUE), m_fDefaultGranted(TRUE) // By default, we grant access { #if 0 // Keep class wizard happy //{{AFX_DATA_INIT(CFtpSecurityPage) m_nGrantedDenied = 0; //}}AFX_DATA_INIT #endif // 0 m_list_IpAddresses.AttachResources(&m_ListBoxRes); } CFtpSecurityPage::~CFtpSecurityPage() /*++ Routine Description: Destructor Arguments: N/A Return Value: N/A --*/ { // // The access list will clean itself up // } void CFtpSecurityPage::DoDataExchange( IN CDataExchange * pDX ) /*++ Routine Description: Initialise/Store control data Arguments: CDataExchange * pDX - DDX/DDV control structure Return Value: None --*/ { CInetPropertyPage::DoDataExchange(pDX); //{{AFX_DATA_MAP(CFtpSecurityPage) DDX_Control(pDX, IDC_BUTTON_REMOVE, m_button_Remove); DDX_Control(pDX, IDC_BUTTON_EDIT, m_button_Edit); DDX_Control(pDX, IDC_BUTTON_ADD, m_button_Add); DDX_Control(pDX, IDC_ICON_GRANTED, m_icon_Granted); DDX_Control(pDX, IDC_ICON_DENIED, m_icon_Denied); DDX_Control(pDX, IDC_RADIO_GRANTED, m_radio_Granted); DDX_Radio(pDX, IDC_RADIO_GRANTED, m_nGrantedDenied); //}}AFX_DATA_MAP // // Private DDX/DDV Routines // DDX_Control(pDX, IDC_RADIO_DENIED, m_radio_Denied); DDX_Control(pDX, IDC_LIST_IP_ADDRESSES, m_list_IpAddresses); } // // Message Map // BEGIN_MESSAGE_MAP(CFtpSecurityPage, CInetPropertyPage) //{{AFX_MSG_MAP(CFtpSecurityPage) ON_BN_CLICKED(IDC_BUTTON_ADD, OnButtonAdd) ON_BN_CLICKED(IDC_BUTTON_EDIT, OnButtonEdit) ON_BN_CLICKED(IDC_BUTTON_REMOVE, OnButtonRemove) ON_BN_CLICKED(IDC_RADIO_GRANTED, OnRadioGranted) ON_BN_CLICKED(IDC_RADIO_DENIED, OnRadioDenied) ON_LBN_DBLCLK(IDC_LIST_IP_ADDRESSES, OnDblclkListIpAddresses) ON_LBN_ERRSPACE(IDC_LIST_IP_ADDRESSES, OnErrspaceListIpAddresses) ON_LBN_SELCHANGE(IDC_LIST_IP_ADDRESSES, OnSelchangeListIpAddresses) ON_WM_VKEYTOITEM() //}}AFX_MSG_MAP END_MESSAGE_MAP() /* virtual */ HRESULT CFtpSecurityPage::FetchLoadedValues() /*++ Routine Description: Move configuration data from sheet to dialog controls Arguments: None Return Value: HRESULT --*/ { m_nGrantedDenied = 0; // // Build the IPL list // CError err(BuildIplOblistFromBlob( ((CFtpSheet *)GetSheet())->GetDirectoryProperties().m_ipl, m_oblAccessList, m_fDefaultGranted )); err.MessageBoxOnFailure(); m_nGrantedDenied = m_fDefaultGranted ? DEFAULT_GRANTED : DEFAULT_DENIED; m_fOldDefaultGranted = m_fDefaultGranted; return S_OK; } /* virtual */ HRESULT CFtpSecurityPage::SaveInfo() /*++ Routine Description: Save the information on this property page Arguments: None Return Value: Error return code --*/ { ASSERT(IsDirty()); TRACEEOLID("Saving FTP security page now..."); BOOL fIplDirty = m_fIpDirty || (m_fOldDefaultGranted != m_fDefaultGranted); // // Use m_ notation because the message crackers require it. // CBlob m_ipl; if (fIplDirty) { BuildIplBlob(m_oblAccessList, m_fDefaultGranted, m_ipl); } CError err; BeginWaitCursor(); BEGIN_META_DIR_WRITE(CFtpSheet) if (fIplDirty) { STORE_DIR_DATA_ON_SHEET(m_ipl); } END_META_DIR_WRITE(err) EndWaitCursor(); if (err.Succeeded()) { m_fIpDirty = FALSE; m_fOldDefaultGranted = m_fDefaultGranted; } return err; } // // Message Handlers // // <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< void CFtpSecurityPage::OnButtonAdd() /*++ Routine Description: 'Add' button handler Arguments: None Return Value: None --*/ { if (ShowPropertiesDialog(TRUE) == IDOK) { SetControlStates(); m_fIpDirty = TRUE; OnItemChanged(); } } void CFtpSecurityPage::OnButtonEdit() /*++ Routine Description: 'Edit' button handler Arguments: None Return Value: None --*/ { if (ShowPropertiesDialog(FALSE) == IDOK) { SetControlStates(); m_fIpDirty = TRUE; OnItemChanged(); } } void CFtpSecurityPage::OnButtonRemove() /*++ Routine Description: 'Remove' button handler Arguments: None Return Value: None --*/ { int nSel = 0; int nCurSel = m_list_IpAddresses.GetCurSel(); while (m_list_IpAddresses.GetNextSelectedItem(&nSel)) { m_oblAccessList.RemoveIndex(nSel); m_list_IpAddresses.DeleteString(nSel); } m_fIpDirty = TRUE; OnItemChanged(); if (nCurSel > 0) { --nCurSel; } m_list_IpAddresses.SetCurSel(nCurSel); if (!SetControlStates()) { m_button_Add.SetFocus(); } } BOOL CFtpSecurityPage::SetControlStates() /*++ Routine Description: Set the enabled status of the controls depending on the current state of the dialog Arguments: None Return Value: TRUE if at least one item is currently selected in the listbox --*/ { BOOL fSomeSelection = m_list_IpAddresses.GetSelCount() > 0; m_button_Edit.EnableWindow(m_list_IpAddresses.GetSelCount() == 1); m_button_Remove.EnableWindow(m_list_IpAddresses.GetSelCount() > 0); return fSomeSelection; } void CFtpSecurityPage::FillListBox( IN CIPAccessDescriptor * pSelection OPTIONAL ) /*++ Routine Description: Populate the listbox with the access list entries Arguments: CIPAccessDescriptor * pSelection : Item to be selected or NULL. Return Value: None --*/ { CObListIter obli(m_oblAccessList); const CIPAccessDescriptor * pAccess; m_list_IpAddresses.SetRedraw(FALSE); m_list_IpAddresses.ResetContent(); int cItems = 0 ; int nSel = LB_ERR, nItem; for ( /**/ ; pAccess = (CIPAccessDescriptor *)obli.Next(); ++cItems) { // // We only list those not adhering to the default // if (pAccess->HasAccess() != m_fDefaultGranted) { nItem = m_list_IpAddresses.AddItem(pAccess); if (pAccess == pSelection) { // // Found item to be selected // nSel = nItem; } } } m_list_IpAddresses.SetCurSel(nSel); m_list_IpAddresses.SetRedraw(TRUE); } DWORD CFtpSecurityPage::SortAccessList() /*++ Routine Description: Sorting the access list by grant denied and ip address FillListBox() should be called after this because the listbox will no longer reflect the true status of the list of directories. Arguments: None Return Value: Error return code --*/ { BeginWaitCursor(); DWORD dw = m_oblAccessList.Sort( (CObjectPlus::PCOBJPLUS_ORDER_FUNC)&CIPAccessDescriptor::OrderByAddress); EndWaitCursor(); return dw; } INT_PTR CFtpSecurityPage::ShowPropertiesDialog( IN BOOL fAdd ) /*++ Routine Description: Bring up the dialog used for add or edit. Return the value returned by the dialog Arguments: BOOL fAdd : If TRUE, create new item. Otherwise, edit existing item Return Value: Dialog return code (IDOK/IDCANCEL) --*/ { // // Bring up the dialog // CIPAccessDescriptor * pAccess = NULL; int nCurSel = LB_ERR; if (!fAdd) { // // Edit existing entry -- there better be only one... // pAccess = m_list_IpAddresses.GetSelectedItem(); ASSERT(pAccess != NULL); if (pAccess == NULL) { // // Double click? // return IDCANCEL; } } CIPAccessDlg dlgAccess( m_fDefaultGranted, pAccess, &m_oblAccessList, this ); INT_PTR nReturn = dlgAccess.DoModal(); if (nReturn == IDOK) { CError err; ASSERT(pAccess != NULL); if (pAccess == NULL) { err = ERROR_NOT_ENOUGH_MEMORY; } else { try { if (fAdd) { m_oblAccessList.AddTail(pAccess); } SortAccessList(); FillListBox(pAccess); } catch(CMemoryException * e) { err = ERROR_NOT_ENOUGH_MEMORY; e->Delete(); } } err.MessageBoxOnFailure(); } return nReturn; } void CFtpSecurityPage::OnDblclkListIpAddresses() /*++ Routine Description: Map listbox double click to the edit button Arguments: None Return Value: None --*/ { OnButtonEdit(); } void CFtpSecurityPage::OnErrspaceListIpAddresses() /*++ Routine Description: Handle error condition in the ip address listbox Arguments: None Return Value: None --*/ { SetControlStates(); } void CFtpSecurityPage::OnSelchangeListIpAddresses() /*++ Routine Description: Handle change in the selection of the listbox Arguments: None Return Value: None --*/ { SetControlStates(); } void CFtpSecurityPage::OnItemChanged() /*++ Routine Description: All EN_CHANGE messages map to this function Arguments: None Return Value: None --*/ { SetModified(TRUE); } BOOL CFtpSecurityPage::OnInitDialog() /*++ Routine Description: WM_INITDIALOG handler. Initialize the dialog. Arguments: None. Return Value: TRUE if focus is to be set automatically, FALSE if the focus is already set. --*/ { CInetPropertyPage::OnInitDialog(); m_icon_Granted.SetIcon(::AfxGetApp()->LoadIcon(IDI_GRANTED)); m_icon_Denied.SetIcon(::AfxGetApp()->LoadIcon(IDI_DENIED)); m_list_IpAddresses.Initialize(); m_list_IpAddresses.EnableWindow(HasIPAccessCheck()); m_button_Add.EnableWindow(HasIPAccessCheck()); GetDlgItem(IDC_RADIO_GRANTED)->EnableWindow(HasIPAccessCheck()); GetDlgItem(IDC_RADIO_DENIED)->EnableWindow(HasIPAccessCheck()); GetDlgItem(IDC_ICON_GRANTED)->EnableWindow(HasIPAccessCheck()); GetDlgItem(IDC_ICON_DENIED)->EnableWindow(HasIPAccessCheck()); FillListBox(); SetControlStates(); return TRUE; } void CFtpSecurityPage::OnRadioGranted() /*++ Routine Description: Granted by default has been selected. Refill the listbox with items that have been explicitly denied. Although we can only have a deny list or a grant list, we keep both of them around until it comes time to saving the information. Arguments: None Return Value: None --*/ { if (!m_fDefaultGranted) { m_fDefaultGranted = TRUE; FillListBox(); OnItemChanged(); SetControlStates(); } } void CFtpSecurityPage::OnRadioDenied() /*++ Routine Description: As above, but reverse granted and denied Arguments: None Return Value: None --*/ { if (m_fDefaultGranted) { m_fDefaultGranted = FALSE; FillListBox(); OnItemChanged(); SetControlStates(); } } int CFtpSecurityPage::OnVKeyToItem( IN UINT nKey, IN CListBox * pListBox, IN UINT nIndex ) /*++ Routine Description: Map insert and delete keys for the listbox Arguments: UINT nKey : Key pressed CListBox * pListBox : Listbox UINT nIndex : Index selected Return Value: -2 if fully handled, -1 if partially handled, 0+ if not handled at all --*/ { switch(nKey) { case VK_DELETE: OnButtonRemove(); break; case VK_INSERT: OnButtonAdd(); break; default: // // Not completely handled by this function, let // windows handle the remaining default action. // return -1; } // // No further action is neccesary. // return -2; }