Source code of Windows XP (NT5)
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.

971 lines
26 KiB

  1. //+---------------------------------------------------------------------------
  2. //
  3. // Microsoft Windows
  4. // Copyright (C) Microsoft Corporation 1996-2001.
  5. //
  6. // File: ANumber.cpp
  7. //
  8. // Contents: Implementation of CAttrNumber
  9. //
  10. //----------------------------------------------------------------------------
  11. #include "stdafx.h"
  12. #include "wsecmgr.h"
  13. #include "snapmgr.h"
  14. #include "util.h"
  15. #include "ANumber.h"
  16. #include "DDWarn.h"
  17. #ifdef _DEBUG
  18. #define new DEBUG_NEW
  19. #undef THIS_FILE
  20. static char THIS_FILE[] = __FILE__;
  21. #endif
  22. /*--------------------------------------------------------------------------------------
  23. The constant values below are used to retrieve the string ID of the description for
  24. a range of values.
  25. struct {
  26. int iMin - >= The value must be greater than or equal to this value.
  27. int iMax - <= The value must be less than or equale to this value.
  28. WORD uResource - The resource id of the item.
  29. WORD uMask - Flag that describes which memebers are valid. The Resource
  30. ID must always be valid. If a flag is not set for the
  31. coorisponding item, then the point is not checked against
  32. the value.
  33. RDIF_MIN - The [iMin] member is valid.
  34. RDIF_MAX - [iMax] member is valid.
  35. RDIF_END - This is the end of the array. The last
  36. item in all declarations must set this
  37. flag.
  38. --------------------------------------------------------------------------------------*/
  39. //
  40. // Minimum password description for number attribute.
  41. //
  42. RANGEDESCRIPTION g_rdMinPassword[] =
  43. {
  44. { 0, 0, IDS_CHANGE_IMMEDIATELY, RDIF_MIN | RDIF_MAX },
  45. { 1, 0, IDS_PASSWORD_CHANGE, RDIF_MIN | RDIF_END }
  46. };
  47. //
  48. // Maximum password description for number attribute.
  49. //
  50. RANGEDESCRIPTION g_rdMaxPassword[] =
  51. {
  52. { 1, 999, IDS_PASSWORD_EXPIRE, RDIF_MIN | RDIF_MAX },
  53. { 0, 0, IDS_PASSWORD_FOREVER, RDIF_MIN | RDIF_END},
  54. };
  55. //
  56. // Password len descriptions.
  57. //
  58. RANGEDESCRIPTION g_rdPasswordLen[] =
  59. {
  60. {0, 0, IDS_PERMIT_BLANK, RDIF_MIN | RDIF_MAX },
  61. {1, 0, IDS_PASSWORD_LEN, RDIF_MIN | RDIF_END }
  62. };
  63. //
  64. // Password histroy description.
  65. //
  66. RANGEDESCRIPTION g_rdPasswordHistory[] =
  67. {
  68. {0, 0, IDS_NO_HISTORY, RDIF_MIN | RDIF_MAX },
  69. {1, 0, IDS_PASSWORD_REMEMBER, RDIF_MIN | RDIF_END }
  70. };
  71. //
  72. // Password lockout descriptions
  73. //
  74. RANGEDESCRIPTION g_rdLockoutAccount[] =
  75. {
  76. {0, 0, IDS_NO_LOCKOUT, RDIF_MIN | RDIF_MAX },
  77. {1, 0, IDS_LOCKOUT_AFTER, RDIF_MIN | RDIF_END }
  78. };
  79. //
  80. // Lockout duration description.
  81. //
  82. RANGEDESCRIPTION g_rdLockoutFor[] =
  83. {
  84. {1, 0, IDS_DURATION, RDIF_MIN },
  85. {0, 0, IDS_LOCKOUT_FOREVER, RDIF_MAX | RDIF_END}
  86. };
  87. RANGEDESCRIPTION g_rdAutoDisconnect[] =
  88. {
  89. { 1, 0, IDS_RNH_AUTODISCONNECT_STATIC, RDIF_MIN },
  90. { 0, 0, IDS_RNH_AUTODISCONNECT_SPECIAL, RDIF_MAX | RDIF_END}
  91. };
  92. RANGEDESCRIPTION g_rdPasswordWarnings[] =
  93. {
  94. { 0, 0, IDS_RNH_PASSWORD_WARNINGS_SPECIAL, RDIF_MIN | RDIF_MAX},
  95. { 1, 0, IDS_RNH_PASSWORD_WARNINGS_STATIC, RDIF_MIN | RDIF_END}
  96. };
  97. RANGEDESCRIPTION g_rdCachedLogons[] =
  98. {
  99. { 0, 0, IDS_RNH_CACHED_LOGONS_SPECIAL, RDIF_MIN | RDIF_MAX},
  100. { 1, 0, IDS_RNH_CACHED_LOGONS_STATIC, RDIF_MIN | RDIF_END}
  101. };
  102. /*--------------------------------------------------------------------------------------
  103. Method: GetRangeDescription
  104. Synopsis: This function was specifically created for SCE. Call this function if the
  105. Item ID to, and current range value to retrieve the corrisponding string.
  106. Arguments: [uType] - [in] ID of the point you want the description for.
  107. [i] - [in] The point you want the description for.
  108. [pstrRet] - [out] The return value.
  109. Returns: ERROR_SUCCESS - The operation was successfull.
  110. ERROR_INVALID_DATA - The id may not be supported or [pstrRet] is NULL.
  111. Other Win32 errors if resource loading was not successful.
  112. --------------------------------------------------------------------------------------*/
  113. DWORD
  114. GetRangeDescription(
  115. IN UINT uType,
  116. IN int i,
  117. OUT CString *pstrRet
  118. )
  119. {
  120. switch(uType){
  121. case IDS_LOCK_DURATION:
  122. uType = GetRangeDescription(g_rdLockoutFor, i);
  123. break;
  124. case IDS_MAX_PAS_AGE:
  125. uType = GetRangeDescription(g_rdMaxPassword, i);
  126. break;
  127. case IDS_LOCK_COUNT:
  128. uType = GetRangeDescription(g_rdLockoutAccount, i);
  129. break;
  130. case IDS_MIN_PAS_AGE:
  131. uType = GetRangeDescription(g_rdMinPassword, i);
  132. break;
  133. case IDS_MIN_PAS_LEN:
  134. uType = GetRangeDescription(g_rdPasswordLen, i);
  135. break;
  136. case IDS_PAS_UNIQUENESS:
  137. uType = GetRangeDescription(g_rdPasswordHistory, i);
  138. break;
  139. case IDS_LOCK_RESET_COUNT:
  140. uType = 0;
  141. break;
  142. default:
  143. uType = 0;
  144. }
  145. if(uType && pstrRet){
  146. //
  147. // Try to load the resource string.
  148. //
  149. __try {
  150. if( pstrRet->LoadString(uType) ){
  151. return ERROR_SUCCESS;
  152. }
  153. } __except(EXCEPTION_CONTINUE_EXECUTION) {
  154. return (DWORD)E_POINTER;
  155. }
  156. return GetLastError();
  157. }
  158. return ERROR_INVALID_DATA;
  159. }
  160. /*--------------------------------------------------------------------------------------
  161. Method: GetRangeDescription
  162. Synopsis: This function works directly with the RANGEDESCRIPTION structure. Tests
  163. to see which string resource ID to return. This is determined by testing [i]
  164. with the [iMin] and [iMax] value of a RANGEDESCRIPTION structure. RDIF_MIN
  165. or/and RDIF_MAX must be set in the [uMask] member for this function to
  166. perform any comparisons
  167. Arguments: [pDesc] - [in] An array of RANGEDESCRIPTIONS, the last member of this
  168. array must set RDIF_END flag in the [uMask] member.
  169. [i] - [in] The point to test.
  170. Returns: A String resource ID if successfull. Otherwise 0.
  171. --------------------------------------------------------------------------------------*/
  172. UINT
  173. GetRangeDescription(
  174. RANGEDESCRIPTION *pDesc,
  175. int i
  176. )
  177. {
  178. RANGEDESCRIPTION *pCur = pDesc;
  179. if(!pDesc){
  180. return 0;
  181. }
  182. //
  183. // The uMask member of the description tells us wich members
  184. // of the structure is valid.
  185. //
  186. while( 1 ){
  187. if( (pCur->uMask & RDIF_MIN) ) {
  188. //
  189. // Test the minimum.
  190. //
  191. if(i >= pCur->iMin){
  192. if(pCur->uMask & RDIF_MAX){
  193. //
  194. // Test the maximum.
  195. //
  196. if( i <= pCur->iMax) {
  197. return pCur->uResource;
  198. }
  199. } else {
  200. return pCur->uResource;
  201. }
  202. }
  203. } else if(pCur->uMask & RDIF_MAX) {
  204. //
  205. // Test only the maximum.
  206. //
  207. if(i <= pCur->iMax){
  208. return pCur->uResource;
  209. }
  210. }
  211. if(pCur->uMask & RDIF_END){
  212. //
  213. // This is the last element of the array, so end the loop.
  214. //
  215. break;
  216. }
  217. pCur++;
  218. }
  219. return 0;
  220. }
  221. /////////////////////////////////////////////////////////////////////////////
  222. // CAttrNumber dialog
  223. CAttrNumber::CAttrNumber(UINT nTemplateID)
  224. : CAttribute(nTemplateID ? nTemplateID : IDD),
  225. m_cMinutes(0),
  226. m_nLow(0),
  227. m_nHigh(999),
  228. m_nSave(0),
  229. m_pRDescription(NULL)
  230. {
  231. //{{AFX_DATA_INIT(CAttrNumber)
  232. m_strUnits = _T("");
  233. m_strSetting = _T("");
  234. m_strBase = _T("");
  235. m_strTemplateTitle = _T("");
  236. m_strLastInspectTitle = _T("");
  237. //}}AFX_DATA_INIT
  238. m_pHelpIDs = (DWORD_PTR)a168HelpIDs;
  239. m_uTemplateResID = IDD;
  240. }
  241. void CAttrNumber::DoDataExchange(CDataExchange* pDX)
  242. {
  243. CAttribute::DoDataExchange(pDX);
  244. //{{AFX_DATA_MAP(CAttrNumber)
  245. DDX_Control(pDX, IDC_SPIN, m_SpinValue);
  246. DDX_Text(pDX, IDC_UNITS, m_strUnits);
  247. DDX_Text(pDX, IDC_CURRENT, m_strSetting);
  248. DDX_Text(pDX, IDC_NEW, m_strBase);
  249. DDX_Text(pDX, IDC_TEMPLATE_TITLE, m_strTemplateTitle);
  250. DDX_Text(pDX, IDC_LI_TITLE, m_strLastInspectTitle);
  251. DDX_Text(pDX, IDC_RANGEERROR,m_strError);
  252. //}}AFX_DATA_MAP
  253. }
  254. BEGIN_MESSAGE_MAP(CAttrNumber, CAttribute)
  255. //{{AFX_MSG_MAP(CAttrNumber)
  256. ON_NOTIFY(UDN_DELTAPOS, IDC_SPIN, OnDeltaposSpin)
  257. ON_EN_KILLFOCUS(IDC_NEW, OnKillFocusNew)
  258. ON_BN_CLICKED(IDC_CONFIGURE, OnConfigure)
  259. ON_EN_UPDATE(IDC_NEW, OnUpdateNew)
  260. //}}AFX_MSG_MAP
  261. END_MESSAGE_MAP()
  262. /////////////////////////////////////////////////////////////////////////////
  263. // CAttrNumber message handlers
  264. void CAttrNumber::OnDeltaposSpin( NMHDR* pNMHDR, LRESULT* pResult )
  265. {
  266. NM_UPDOWN FAR *pnmud = (NM_UPDOWN FAR *)pNMHDR;
  267. if ( pnmud ) {
  268. //
  269. // get current value
  270. //
  271. long lVal = CurrentEditValue();
  272. if (SCE_FOREVER_VALUE == lVal) {
  273. if (pnmud->iDelta > 0) {
  274. if (m_cMinutes & DW_VALUE_OFF) {
  275. lVal = SCE_KERBEROS_OFF_VALUE;
  276. } else {
  277. lVal = m_nHigh;
  278. }
  279. } else {
  280. lVal = m_nLow;
  281. }
  282. } else if (SCE_KERBEROS_OFF_VALUE == lVal) {
  283. if (pnmud->iDelta < 0) {
  284. if (m_cMinutes & DW_VALUE_FOREVER) {
  285. lVal = SCE_FOREVER_VALUE;
  286. } else {
  287. lVal = m_nLow;
  288. }
  289. } else {
  290. lVal = m_nHigh;
  291. }
  292. } else {
  293. lVal -= (LONG)(m_iAccRate*pnmud->iDelta);
  294. if ( lVal > m_nHigh ) {
  295. // if it is overflow, go back to low
  296. if ( m_cMinutes & DW_VALUE_OFF ) {
  297. lVal = SCE_KERBEROS_OFF_VALUE;
  298. } else if (m_cMinutes & DW_VALUE_FOREVER) {
  299. lVal = SCE_FOREVER_VALUE;
  300. } else {
  301. lVal = m_nLow;
  302. }
  303. } else if ( (lVal < m_nLow) &&
  304. ((lVal != SCE_KERBEROS_OFF_VALUE) || !(m_cMinutes & DW_VALUE_OFF)) &&
  305. ((lVal != SCE_FOREVER_VALUE) || !(m_cMinutes & DW_VALUE_FOREVER))) {
  306. // if it is underflow, go back to high
  307. if ( (m_cMinutes & DW_VALUE_FOREVER) && (lVal != SCE_FOREVER_VALUE)) {
  308. lVal = SCE_FOREVER_VALUE;
  309. } else if ((m_cMinutes & DW_VALUE_OFF) && (lVal != SCE_KERBEROS_OFF_VALUE)) {
  310. lVal = SCE_KERBEROS_OFF_VALUE;
  311. } else {
  312. lVal = m_nHigh;
  313. }
  314. }
  315. if ( 0 == lVal && (m_cMinutes & DW_VALUE_NOZERO) ) {
  316. // zero is not allowed
  317. if ( m_nLow > 0 ) {
  318. lVal = m_nLow;
  319. } else {
  320. lVal = 1;
  321. }
  322. }
  323. }
  324. SetValueToEdit(lVal);
  325. }
  326. *pResult = 0;
  327. }
  328. void CAttrNumber::OnKillFocusNew()
  329. {
  330. LONG lVal = CurrentEditValue();
  331. SetValueToEdit(lVal);
  332. }
  333. void CAttrNumber::SetValueToEdit(LONG lVal)
  334. {
  335. CString strNew;
  336. if ( m_iStaticId )
  337. m_strTemplateTitle.LoadString(m_iStaticId);
  338. else
  339. m_strTemplateTitle = _T("");
  340. if ( 0 == lVal ) {
  341. strNew.Format(TEXT("%d"),lVal);
  342. if ( m_cMinutes & DW_VALUE_NEVER &&
  343. m_iNeverId > 0 ) {
  344. // change to never
  345. m_strTemplateTitle.LoadString(m_iNeverId);
  346. }
  347. } else if ( SCE_FOREVER_VALUE == lVal ) {
  348. strNew.LoadString(IDS_FOREVER);
  349. if ( m_iNeverId )
  350. m_strTemplateTitle.LoadString(m_iNeverId);
  351. } else if (SCE_KERBEROS_OFF_VALUE == lVal) {
  352. strNew.LoadString(IDS_OFF);
  353. if ( m_iNeverId ) {
  354. m_strTemplateTitle.LoadString(m_iNeverId);
  355. }
  356. } else {
  357. strNew.Format(TEXT("%d"),lVal);
  358. }
  359. m_nSave = lVal;
  360. SetDlgItemText(IDC_NEW,strNew);
  361. SetDlgItemText(IDC_TEMPLATE_TITLE,m_strTemplateTitle);
  362. SetModified(TRUE);
  363. }
  364. LONG CAttrNumber::CurrentEditValue()
  365. {
  366. UINT uiVal = 0;
  367. LONG lVal = 0;
  368. BOOL bTrans = FALSE;
  369. uiVal = GetDlgItemInt(IDC_NEW,&bTrans,FALSE);
  370. lVal = uiVal;
  371. if (!bTrans ) {
  372. // if ( 0 == lVal && !bTrans ) {
  373. // errored, overflow, or nonnumeric
  374. CString str;
  375. if(m_cMinutes & DW_VALUE_FOREVER){
  376. str.LoadString(IDS_FOREVER);
  377. if(str == m_strBase){
  378. return SCE_FOREVER_VALUE;
  379. }
  380. }
  381. lVal = _ttol((LPCTSTR)m_strBase);
  382. if ( lVal == 0 ) {
  383. //
  384. // nonnumeric
  385. //
  386. lVal = (LONG) m_nSave;
  387. return lVal;
  388. }
  389. }
  390. if ( m_iAccRate > 1 && lVal > 0 ) {
  391. //
  392. // for log max size, make it multiples of m_iAccRate
  393. //
  394. int nCount = lVal % m_iAccRate;
  395. if ( nCount > 0 ) {
  396. lVal = ((LONG)(lVal/m_iAccRate))*m_iAccRate;
  397. }
  398. }
  399. if ( lVal > m_nHigh ) {
  400. // if it is overflow, go back to low
  401. if ( m_cMinutes & DW_VALUE_FOREVER ) {
  402. lVal = SCE_FOREVER_VALUE;
  403. } else if (m_cMinutes & DW_VALUE_OFF) {
  404. lVal = SCE_KERBEROS_OFF_VALUE;
  405. } else {
  406. // Leave alone and let the OnKillActive catch it
  407. }
  408. }
  409. if ( (lVal < m_nLow) &&
  410. (lVal != SCE_KERBEROS_OFF_VALUE) &&
  411. (lVal != SCE_FOREVER_VALUE) ) {
  412. // if it is underflow, go back to high
  413. if (m_cMinutes & DW_VALUE_OFF) {
  414. lVal = SCE_KERBEROS_OFF_VALUE;
  415. } else if ( m_cMinutes & DW_VALUE_FOREVER) {
  416. lVal = SCE_FOREVER_VALUE;
  417. } else {
  418. // Leave alone and let the OnKillActive catch it
  419. }
  420. }
  421. if ( 0 == lVal && (m_cMinutes & DW_VALUE_NOZERO) ) {
  422. // zero is not allowed
  423. if ( m_nLow > 0 ) {
  424. lVal = m_nLow;
  425. } else {
  426. lVal = 1;
  427. }
  428. }
  429. return lVal;
  430. }
  431. void CAttrNumber::OnConfigure()
  432. {
  433. CWnd *cwnd;
  434. CAttribute::OnConfigure();
  435. //
  436. // Enable disable OK button depending on the state of the other controls.
  437. //
  438. cwnd = GetDlgItem(IDOK);
  439. if(cwnd){
  440. if(!m_bConfigure){
  441. cwnd->EnableWindow(TRUE);
  442. } else {
  443. OnUpdateNew();
  444. }
  445. }
  446. }
  447. BOOL CAttrNumber::OnInitDialog()
  448. {
  449. CAttribute::OnInitDialog();
  450. UpdateData(TRUE);
  451. /*
  452. if (m_bMinutes) {
  453. m_SpinValue.SetRange(-1,UD_MAXVAL-1);
  454. } else {
  455. m_SpinValue.SetRange(0,UD_MAXVAL);
  456. }
  457. */
  458. AddUserControl(IDC_NEW);
  459. AddUserControl(IDC_SPIN);
  460. OnConfigure();
  461. return TRUE; // return TRUE unless you set the focus to a control
  462. // EXCEPTION: OCX Property Pages should return FALSE
  463. }
  464. BOOL CAttrNumber::OnApply()
  465. {
  466. if ( !m_bReadOnly )
  467. {
  468. BOOL bUpdateAll = FALSE;
  469. DWORD dw = 0;
  470. CString strForever,strOff;
  471. int status = 0;
  472. UpdateData(TRUE);
  473. if (!m_bConfigure)
  474. dw = SCE_NO_VALUE;
  475. else
  476. dw = CurrentEditValue();
  477. bUpdateAll = FALSE;
  478. CEditTemplate *pet = m_pSnapin->GetTemplate(GT_COMPUTER_TEMPLATE,AREA_SECURITY_POLICY);
  479. //
  480. // Check the numbers dependencies if a dependency fails then the dialog
  481. // will return ERROR_MORE_DATA and we can display more information for the user
  482. // to see.
  483. //
  484. if(DDWarn.CheckDependencies( dw ) == ERROR_MORE_DATA )
  485. {
  486. //
  487. // If the user presses cancel then we won't let them set this item's information
  488. // until they have set the configurion for the other items.
  489. //
  490. CThemeContextActivator activator;
  491. if( DDWarn.DoModal() != IDOK)
  492. return FALSE;
  493. //
  494. // The user pressed Auto set so we can set the other items. They are automatically set
  495. // to the correct values.
  496. //
  497. for(int i = 0; i < DDWarn.GetFailedCount(); i++)
  498. {
  499. PDEPENDENCYFAILED pItem = DDWarn.GetFailedInfo(i);
  500. if(pItem && pItem->pResult )
  501. {
  502. pItem->pResult->SetBase( pItem->dwSuggested );
  503. status = m_pSnapin->SetAnalysisInfo(
  504. pItem->pResult->GetID(),
  505. pItem->dwSuggested,
  506. pItem->pResult);
  507. pItem->pResult->SetStatus( status );
  508. pItem->pResult->Update(m_pSnapin, FALSE);
  509. }
  510. }
  511. }
  512. //
  513. // Update the items security profile.
  514. // and redraw.
  515. //
  516. m_pData->SetBase((LONG_PTR)ULongToPtr(dw));
  517. status = m_pSnapin->SetAnalysisInfo(m_pData->GetID(),(LONG_PTR)ULongToPtr(dw), m_pData);
  518. m_pData->SetStatus(status);
  519. m_pData->Update(m_pSnapin, FALSE);
  520. }
  521. return CAttribute::OnApply();
  522. }
  523. void CAttrNumber::Initialize(CResult * pResult)
  524. {
  525. LONG_PTR dw=0;
  526. CAttribute::Initialize(pResult);
  527. DDWarn.InitializeDependencies(m_pSnapin,pResult);
  528. m_strUnits = pResult->GetUnits();
  529. m_cMinutes = DW_VALUE_NOZERO;
  530. m_nLow = 0;
  531. m_nHigh = 999;
  532. m_nSave = 0;
  533. m_iNeverId = 0;
  534. m_iAccRate = 1;
  535. m_iStaticId = 0;
  536. CEditTemplate *pTemplate = pResult->GetBaseProfile();
  537. switch (pResult->GetID()) {
  538. // below no zero value
  539. case IDS_LOCK_DURATION:
  540. m_cMinutes = DW_VALUE_FOREVER | DW_VALUE_NOZERO;
  541. m_nHigh = 99999;
  542. m_iStaticId = IDS_DURATION;
  543. m_iNeverId = IDS_LOCKOUT_FOREVER;
  544. m_pRDescription = g_rdLockoutAccount;
  545. break;
  546. case IDS_MAX_PAS_AGE:
  547. m_cMinutes = DW_VALUE_FOREVER | DW_VALUE_NOZERO;
  548. m_iStaticId = IDS_PASSWORD_EXPIRE;
  549. m_iNeverId = IDS_PASSWORD_FOREVER;
  550. break;
  551. // below zero value means differently
  552. case IDS_LOCK_COUNT:
  553. m_cMinutes = DW_VALUE_NEVER;
  554. m_iNeverId = IDS_NO_LOCKOUT;
  555. m_iStaticId = IDS_LOCKOUT_AFTER;
  556. break;
  557. case IDS_MIN_PAS_AGE:
  558. m_cMinutes = DW_VALUE_NEVER;
  559. m_iNeverId = IDS_CHANGE_IMMEDIATELY;
  560. m_iStaticId = IDS_PASSWORD_CHANGE;
  561. m_nHigh = 998;
  562. m_pRDescription = g_rdMinPassword;
  563. break;
  564. case IDS_MIN_PAS_LEN:
  565. m_cMinutes = DW_VALUE_NEVER;
  566. m_nHigh = 14;
  567. m_iNeverId = IDS_PERMIT_BLANK;
  568. m_iStaticId = IDS_PASSWORD_LEN;
  569. m_pRDescription = g_rdPasswordLen;
  570. break;
  571. case IDS_PAS_UNIQUENESS:
  572. m_cMinutes = DW_VALUE_NEVER;
  573. m_nHigh = 24;
  574. m_iNeverId = IDS_NO_HISTORY;
  575. m_iStaticId = IDS_PASSWORD_REMEMBER;
  576. break;
  577. // below there is no zero values
  578. case IDS_LOCK_RESET_COUNT:
  579. m_nLow = 1;
  580. m_nHigh = 99999;
  581. m_iStaticId = IDS_LOCK_RESET_COUNT;
  582. break;
  583. case IDS_SYS_LOG_MAX:
  584. case IDS_SEC_LOG_MAX:
  585. case IDS_APP_LOG_MAX:
  586. m_nLow = 64;
  587. m_nHigh = 4194240;
  588. m_iAccRate = 64;
  589. // no static text
  590. break;
  591. case IDS_SYS_LOG_DAYS:
  592. case IDS_SEC_LOG_DAYS:
  593. case IDS_APP_LOG_DAYS:
  594. m_nHigh = 365;
  595. m_nLow = 1;
  596. m_iStaticId = IDS_OVERWRITE_EVENT;
  597. break;
  598. case IDS_KERBEROS_MAX_AGE:
  599. m_cMinutes = DW_VALUE_FOREVER | DW_VALUE_NOZERO;
  600. m_nHigh = 99999;
  601. m_iStaticId = IDS_TICKET_EXPIRE;
  602. m_iNeverId = IDS_TICKET_FOREVER;
  603. break;
  604. case IDS_KERBEROS_RENEWAL:
  605. m_cMinutes = DW_VALUE_FOREVER | DW_VALUE_NOZERO; // | DW_VALUE_OFF;
  606. m_nHigh = 99999;
  607. m_iStaticId = IDS_TICKET_RENEWAL_EXPIRE;
  608. m_iNeverId = IDS_TICKET_RENEWAL_FOREVER;
  609. break;
  610. case IDS_KERBEROS_MAX_SERVICE:
  611. m_nLow = 10;
  612. m_cMinutes = DW_VALUE_FOREVER | DW_VALUE_NOZERO;
  613. m_iStaticId = IDS_TICKET_EXPIRE;
  614. m_iNeverId = IDS_TICKET_FOREVER;
  615. m_nHigh = 99999;
  616. break;
  617. case IDS_KERBEROS_MAX_CLOCK:
  618. m_cMinutes = DW_VALUE_FOREVER;
  619. m_nHigh = 99999;
  620. m_iStaticId = IDS_MAX_TOLERANCE;
  621. m_iNeverId = IDS_NO_MAX_TOLERANCE;
  622. break;
  623. }
  624. if ((m_cMinutes & DW_VALUE_NOZERO) && (0 == m_nLow)) {
  625. m_nLow = 1;
  626. }
  627. m_strTemplateTitle = _T("");
  628. m_strLastInspectTitle = _T("");
  629. m_strBase.Empty();
  630. m_strSetting.Empty();
  631. dw = pResult->GetBase();
  632. if ((LONG_PTR)ULongToPtr(SCE_NO_VALUE) == dw)
  633. {
  634. m_bConfigure = FALSE;
  635. }
  636. else
  637. {
  638. m_bConfigure = TRUE;
  639. SetInitialValue(PtrToUlong((PVOID)dw));
  640. }
  641. pResult->GetDisplayName( NULL, m_strSetting, 2 );
  642. dw = pResult->GetSetting();
  643. if ((LONG_PTR)ULongToPtr(SCE_NO_VALUE) != dw) {
  644. if ((LONG_PTR)ULongToPtr(SCE_FOREVER_VALUE) == dw) {
  645. if ( (m_cMinutes & DW_VALUE_FOREVER) &&
  646. m_iNeverId > 0 ) {
  647. m_strLastInspectTitle.LoadString(m_iNeverId);
  648. }
  649. } else {
  650. if ( 0 == dw && (m_cMinutes & DW_VALUE_NEVER) &&
  651. m_iNeverId > 0 ) {
  652. // zero means different values
  653. m_strLastInspectTitle.LoadString(m_iNeverId);
  654. } else if ( m_iStaticId > 0 ) {
  655. m_strLastInspectTitle.LoadString(m_iStaticId);
  656. }
  657. }
  658. }
  659. }
  660. void CAttrNumber::SetInitialValue(DWORD_PTR dw)
  661. {
  662. //
  663. // Don't overwrite an already set value.
  664. //
  665. if (!m_strBase.IsEmpty())
  666. {
  667. return;
  668. }
  669. if (SCE_FOREVER_VALUE == dw) {
  670. m_strBase.LoadString(IDS_FOREVER);
  671. if ( (m_cMinutes & DW_VALUE_FOREVER) &&
  672. m_iNeverId > 0 ) {
  673. m_strTemplateTitle.LoadString(m_iNeverId);
  674. }
  675. m_nSave = SCE_FOREVER_VALUE;
  676. } else if (SCE_KERBEROS_OFF_VALUE == dw) {
  677. m_strBase.LoadString(IDS_OFF);
  678. if ( (m_cMinutes & DW_VALUE_OFF) &&
  679. m_iNeverId > 0 ) {
  680. m_strTemplateTitle.LoadString(m_iNeverId);
  681. }
  682. m_nSave = SCE_KERBEROS_OFF_VALUE;
  683. }
  684. else
  685. {
  686. m_nSave = dw;
  687. if ( 0 == dw && (m_cMinutes & DW_VALUE_NEVER) &&
  688. m_iNeverId > 0 ) {
  689. // zero means different values
  690. m_strTemplateTitle.LoadString(m_iNeverId);
  691. } else if ( m_iStaticId > 0 ) {
  692. m_strTemplateTitle.LoadString(m_iStaticId);
  693. }
  694. m_strBase.Format(TEXT("%d"),dw);
  695. }
  696. }
  697. void CAttrNumber::OnUpdateNew()
  698. {
  699. DWORD dwRes = 0;
  700. UpdateData(TRUE);
  701. CString sNum;
  702. CEdit *pEdit = (CEdit *)GetDlgItem(IDC_NEW);
  703. CWnd *pOK = GetDlgItem(IDOK);
  704. DWORD dwValue = _ttoi(m_strBase);
  705. //
  706. // Don't do anything if the string is equal to predefined strings.
  707. //
  708. sNum.LoadString(IDS_FOREVER);
  709. if (m_strBase.IsEmpty()) {
  710. if (pOK) {
  711. pOK->EnableWindow(FALSE);
  712. }
  713. } else if (m_strBase == sNum) {
  714. if (pOK) {
  715. pOK->EnableWindow(TRUE);
  716. }
  717. } else {
  718. if ((long)dwValue < m_nLow) {
  719. //
  720. // Disable the OK button.
  721. //
  722. if ( pOK ) {
  723. pOK->EnableWindow(FALSE);
  724. }
  725. if (pEdit) {
  726. //
  727. // We will only force a select if edit text length >=
  728. // minimum text length
  729. //
  730. sNum.Format(TEXT("%d"), m_nLow);
  731. dwValue = m_nLow;
  732. if (sNum.GetLength() < m_strBase.GetLength()) {
  733. pEdit->SetSel(0, -1);
  734. }
  735. }
  736. } else if ( (long)dwValue > m_nHigh ) {
  737. if (pOK) {
  738. pOK->EnableWindow(TRUE);
  739. }
  740. if (pEdit) {
  741. if (m_cMinutes & DW_VALUE_FOREVER) {
  742. sNum.LoadString(IDS_FOREVER);
  743. dwValue = 0;
  744. } else {
  745. sNum.Format(TEXT("%d"), m_nHigh);
  746. dwValue = m_nHigh;
  747. }
  748. m_strBase = sNum;
  749. UpdateData(FALSE);
  750. pEdit->SetSel(0, -1);
  751. }
  752. } else {
  753. //
  754. // Enable the OK button.
  755. //
  756. if (pOK) {
  757. pOK->EnableWindow(TRUE);
  758. }
  759. }
  760. }
  761. //
  762. // Load the description for this string.
  763. //
  764. if ((dwValue <= 0) && (m_iNeverId != 0)) {
  765. m_strTemplateTitle.LoadString(m_iNeverId);
  766. } else {
  767. m_strTemplateTitle.LoadString(m_iStaticId);
  768. }
  769. GetDlgItem(IDC_TEMPLATE_TITLE)->SetWindowText(m_strTemplateTitle);
  770. SetModified(TRUE); //Raid #404145
  771. }
  772. BOOL CAttrNumber::OnKillActive()
  773. {
  774. UINT uiVal = 0;
  775. LONG lVal = 0;
  776. BOOL bTrans = FALSE;
  777. CString strRange;
  778. int lMin = m_nLow;
  779. UpdateData(TRUE);
  780. if (m_cMinutes & DW_VALUE_NOZERO &&
  781. !(m_cMinutes & DW_VALUE_FOREVER) &&
  782. lMin == 0) {
  783. lMin = 1;
  784. }
  785. CString strFormat;
  786. strFormat.LoadString(IDS_RANGE);
  787. strRange.Format(strFormat,lMin,m_nHigh);
  788. uiVal = GetDlgItemInt(IDC_NEW,&bTrans,TRUE);
  789. lVal = uiVal;
  790. if ( !bTrans ) {
  791. CString str;
  792. if (m_cMinutes & DW_VALUE_FOREVER) {
  793. str.LoadString(IDS_FOREVER);
  794. if (str == m_strBase) {
  795. return TRUE;;
  796. }
  797. }
  798. lVal = _ttol((LPCTSTR)m_strBase);
  799. if ( lVal == 0 )
  800. {
  801. // nonnumeric
  802. lVal = (LONG) m_nSave;
  803. m_strError = strRange;
  804. UpdateData(FALSE);
  805. return FALSE;
  806. }
  807. }
  808. if ( m_iAccRate > 1 && lVal > 0 ) {
  809. // for log max size, make it multiples of m_iAccRate
  810. int nCount = lVal % m_iAccRate;
  811. if ( nCount > 0 ) {
  812. lVal = ((LONG)(lVal/m_iAccRate))*m_iAccRate;
  813. }
  814. }
  815. if ( lVal > m_nHigh ) {
  816. m_strError = strRange;
  817. UpdateData(FALSE);
  818. return FALSE;
  819. }
  820. if ( (lVal < m_nLow) &&
  821. (lVal != SCE_KERBEROS_OFF_VALUE) &&
  822. (lVal != SCE_FOREVER_VALUE) ) {
  823. // if it is underflow, go back to high
  824. if (m_cMinutes & DW_VALUE_OFF) {
  825. lVal = SCE_KERBEROS_OFF_VALUE;
  826. } else if ( m_cMinutes & DW_VALUE_FOREVER) {
  827. lVal = SCE_FOREVER_VALUE;
  828. } else {
  829. // Leave alone and let the OnKillActive catch it
  830. }
  831. }
  832. if ( (lVal < m_nLow) &&
  833. (lVal != SCE_KERBEROS_OFF_VALUE) &&
  834. (lVal != SCE_FOREVER_VALUE) ) {
  835. // if it is underflow, go back to high
  836. m_strError = strRange;
  837. UpdateData(FALSE);
  838. return FALSE;
  839. }
  840. if ( 0 == lVal && (m_cMinutes & DW_VALUE_NOZERO) ) {
  841. // zero is not allowed
  842. m_strError = strRange;
  843. UpdateData(FALSE);
  844. return FALSE;
  845. }
  846. return CAttribute::OnKillActive();
  847. }