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.

3050 lines
72 KiB

  1. // File: dlgcall.cpp
  2. #include "precomp.h"
  3. #include "resource.h"
  4. #include "help_ids.h"
  5. #include "dlgcall2.h"
  6. #include "dirutil.h"
  7. #include "mrulist2.h"
  8. #include "ldap.h"
  9. #include "wab.h"
  10. #include "calv.h"
  11. #include "speedial.h"
  12. #include "history.h"
  13. #include "dlgacd.h"
  14. #include "conf.h"
  15. #include "confroom.h"
  16. #include "confutil.h"
  17. #include "confpolicies.h"
  18. #include "cmd.h"
  19. // GUI constants for this dialog
  20. const UINT DX_BORDER = 6; // Standard spacing for Office dialogs
  21. const UINT DY_BORDER = 6; // Standard spacing for Office dialogs
  22. const UINT DY_COMBOBOX = 200; // Height of the dropped list
  23. const UINT DX_DLGCALL_MIN = 460;
  24. const UINT DY_DLGCALL_MIN = 400;
  25. const UINT DX_DLGCALL_DEFAULT = 800;
  26. const UINT DY_DLGCALL_DEFAULT = 600;
  27. void GetDefaultRect(const HWND hwndParent, RECT & rcRect, const int iDefaultWidth, const int iDefaultHeight);
  28. int GetPixelsPerChar(const HWND hwnd);
  29. const static struct _ColumnInfo
  30. {
  31. int iColumnLabelIds;
  32. int iMinColumnChars;
  33. } ciColumnInfo[ MAX_DIR_COLUMNS ] =
  34. {
  35. { IDS_DIR_COLUMN_EMAIL, 28 },
  36. { IDS_DIR_COLUMN_AUDIO, 0 },
  37. { IDS_DIR_COLUMN_VIDEO, 0 },
  38. { IDS_DIR_COLUMN_LAST_NAME, 16 },
  39. { IDS_DIR_COLUMN_FIRST_NAME, 14 },
  40. { IDS_DIR_COLUMN_LOCATION, 16 },
  41. { IDS_DIR_COLUMN_COMMENTS, 22 }
  42. };
  43. ///////////////////////////////////////////////////////////////////////////
  44. // Local Data
  45. static const DWORD _mpIdHelpDlgCall[] =
  46. {
  47. IDM_DLGCALL_DELETE_ILS, IDH_FINDSOMEONE_DELETE_ILS,
  48. IDM_DLGCALL_NEWWINDOW, IDH_FINDSOMEONE_BROWSER,
  49. IDM_DLGCALL_REFRESH, IDH_FINDSOMEONE_REFRESH,
  50. ID_TB_HELP, IDH_FINDSOMEONE_HELP,
  51. IDS_DLGCALL_EDIT_HDR, IDH_PLACECALL_NAME,
  52. IDE_DLGCALL_NAME, IDH_PLACECALL_NAME,
  53. IDC_DLGCALL_COMBO, IDH_PLACECALL_INFOSTORE,
  54. IDS_SECURITY_CHECKBOX, IDH_PLACECALL_SECURITY_CHKBX,
  55. IDL_DLGCALL_LIST, IDH_PLACECALL_LIST,
  56. IDL_DLGCALL_LISTOWNERDATA, IDH_PLACECALL_LIST,
  57. IDL_DLGCALL_ILS_LISTVIEW, IDH_PLACECALL_LIST,
  58. IDOK, IDH_PLACECALL_CALL,
  59. 0, 0
  60. };
  61. int g_cBusy = 0; // non-zero if busy connecting to server
  62. #ifdef nmDlgCallNormal
  63. const int NM_CALLDLG_DEFAULT = nmDlgCallNormal;
  64. const int NM_CALLDLG_NO_ILS_FILTER = nmDlgCallNoFilter;
  65. const int NM_CALLDLG_NO_ILS = nmDlgCallNoIls;
  66. const int NM_CALLDLG_NO_GAL = nmDlgCallNoGal;
  67. const int NM_CALLDLG_NO_WAB = nmDlgCallNoWab;
  68. const int NM_CALLDLG_NO_SPEEDDIAL = nmDlgCallNoSpeedDial;
  69. const int NM_CALLDLG_NO_HISTORY = nmDlgCallNoHistory;
  70. #endif /* old NM_CALL_DLG constants */
  71. // Dir list types:
  72. const UINT DLT_ULS = 0;
  73. const UINT DLT_IN_CALL_LOG = 1;
  74. const UINT DLT_OUT_CALL_LOG = 2;
  75. const UINT DLT_WAB = 3;
  76. struct DLTCOLUMNINFO
  77. {
  78. int nColumns; // number of columns
  79. int nIconColumns; // number of icon-only columns
  80. UINT uStringID; // string ID of first column
  81. LPTSTR pszRVOrder; // reg val to store column order
  82. LPTSTR pszRVWidths; // reg val to store column widths
  83. LPTSTR pszRVSortAscending; // reg val to store sort direction
  84. LPTSTR pszRVSortColumn; // reg val to store sort column
  85. };
  86. const DLTCOLUMNINFO g_rgDLColumnInfo[] =
  87. {
  88. { MAX_DIR_COLUMNS, 2, IDS_DIR_COLUMN_FIRST, // DLT_ULS
  89. &REGVAL_DIR_COLUMN_ORDER, &REGVAL_DIR_COLUMN_WIDTHS,
  90. &REGVAL_DIR_SORT_ASCENDING, &REGVAL_DIR_SORT_COLUMN },
  91. };
  92. CFindSomeone* CFindSomeone::s_pDlgCall= NULL;
  93. ///////////////////////////////////////////////////////////////////////////
  94. int StringCompare( const TCHAR * const psz1, const TCHAR * const psz2 );
  95. void UpdateSecurityCheck(CConfRoom *pConfRoom, HWND hDlg, UINT idCheck)
  96. {
  97. bool userAlterable, secure;
  98. pConfRoom->get_securitySettings( userAlterable, secure );
  99. EnableWindow( GetDlgItem(hDlg, idCheck), userAlterable );
  100. ::CheckDlgButton( hDlg, idCheck, secure? BST_CHECKED: BST_UNCHECKED );
  101. }
  102. /* C D L G C A L L */
  103. /*-------------------------------------------------------------------------
  104. %%Function: CFindSomeone
  105. -------------------------------------------------------------------------*/
  106. CFindSomeone::CFindSomeone(CConfRoom *pConfRoom):
  107. m_pAccel(NULL),
  108. m_dwOptions(0),
  109. m_hwndList(NULL),
  110. m_ilsListView(NULL),
  111. m_hwndCombo(NULL),
  112. m_hwndComboEdit(NULL),
  113. m_hwndEdit(NULL),
  114. m_webView( NULL ),
  115. m_WndOwnerDataListOldWndProc(NULL),
  116. m_fInEdit(FALSE),
  117. m_iSel(0),
  118. m_pMruServer(NULL),
  119. m_pUls(NULL),
  120. m_pSpeedDial(NULL),
  121. #if USE_GAL
  122. m_pGAL(NULL),
  123. #endif //#if USE_GAL
  124. m_pAlv(NULL),
  125. m_pWab(NULL),
  126. m_pHistory(NULL),
  127. m_pszDefault(NULL),
  128. m_pRai(NULL),
  129. m_pConfRoom(pConfRoom),
  130. m_bPlacedCall(false)
  131. {
  132. SetEmptySz(m_szAddress);
  133. SetEmptySz(m_szDirectory);
  134. // BUGBUG georgep: Why are we checking m_dwOptions? It was set to 0 above.
  135. if ((0 == (m_dwOptions & NM_CALLDLG_NO_WAB)) &&
  136. (NULL == CWAB::GetInstance()) )
  137. {
  138. m_pWab = new CWAB();
  139. }
  140. if (0 == (m_dwOptions & NM_CALLDLG_NO_ILS))
  141. {
  142. m_pMruServer = GetMruListServer();
  143. m_pUls = new CLDAP();
  144. }
  145. if (0 == (m_dwOptions & NM_CALLDLG_NO_SPEEDDIAL))
  146. {
  147. m_pSpeedDial = new CSPEEDDIAL();
  148. }
  149. #if USE_GAL
  150. if ((0 == (m_dwOptions & NM_CALLDLG_NO_GAL)) &&
  151. CGAL::FLoadMapiFns() )
  152. {
  153. m_pGAL = new CGAL();
  154. }
  155. #endif // USE_GAL
  156. if (0 == (m_dwOptions & NM_CALLDLG_NO_HISTORY))
  157. {
  158. m_pHistory = new CHISTORY();
  159. }
  160. // Load the small icon list
  161. m_himlIcon = ImageList_Create(DXP_ICON_SMALL, DYP_ICON_SMALL, ILC_MASK, 1, 0);
  162. if (NULL != m_himlIcon)
  163. {
  164. HBITMAP hBmp = ::LoadBitmap(::GetInstanceHandle(), MAKEINTRESOURCE(IDB_ICON_IMAGES));
  165. if (NULL != hBmp)
  166. {
  167. ImageList_AddMasked(m_himlIcon, hBmp, TOOLBAR_MASK_COLOR);
  168. ::DeleteObject(hBmp);
  169. }
  170. }
  171. m_pConfRoom->AddConferenceChangeHandler(this);
  172. }
  173. CFindSomeone::~CFindSomeone(void)
  174. {
  175. // Clear any existing data
  176. if (NULL != m_pAlv)
  177. {
  178. m_pAlv->ClearItems();
  179. }
  180. delete m_pMruServer;
  181. if (NULL != m_pWab)
  182. {
  183. m_pWab->Release();
  184. }
  185. if (NULL != m_pUls)
  186. {
  187. m_pUls->Release();
  188. }
  189. if (NULL != m_pSpeedDial)
  190. {
  191. m_pSpeedDial->Release();
  192. }
  193. if (NULL != m_himlIcon)
  194. {
  195. ImageList_Destroy(m_himlIcon);
  196. }
  197. #if USE_GAL
  198. if (NULL != m_pGAL)
  199. {
  200. m_pGAL->Release();
  201. }
  202. #endif // USE_GAL
  203. if (NULL != m_pHistory)
  204. {
  205. m_pHistory->Release();
  206. }
  207. ClearRai(&m_pRai);
  208. m_pConfRoom->RemoveConferenceChangeHandler(this);
  209. }
  210. //--------------------------------------------------------------------------//
  211. // CFindSomeone::findSomeone. //
  212. //--------------------------------------------------------------------------//
  213. void
  214. CFindSomeone::findSomeone(CConfRoom *pConfRoom)
  215. {
  216. if (NULL == s_pDlgCall)
  217. {
  218. s_pDlgCall = new CFindSomeone(pConfRoom);
  219. s_pDlgCall->doModeless();
  220. }
  221. if (NULL != s_pDlgCall)
  222. {
  223. HWND hwnd = s_pDlgCall->GetWindow();
  224. ::ShowWindow( hwnd, SW_RESTORE );
  225. ::SetForegroundWindow( hwnd );
  226. }
  227. } // End of CFindSomeone::findSomeone.
  228. //--------------------------------------------------------------------------//
  229. // CDlgCall::Destroy. //
  230. //--------------------------------------------------------------------------//
  231. void
  232. CFindSomeone::Destroy()
  233. {
  234. WARNING_OUT(("Entering CFindSomeone::Destroy( )..."));
  235. CFindSomeone * pDlgCall = s_pDlgCall;
  236. s_pDlgCall = NULL;
  237. if( pDlgCall != NULL )
  238. {
  239. if( pDlgCall->m_pUls != NULL )
  240. {
  241. // Stop the CLDAP thread from trying to interact with the FindSomeone
  242. // windows after we've already destroyed them...
  243. pDlgCall->m_pUls->CloseServer();
  244. }
  245. HWND hwnd = pDlgCall->GetWindow();
  246. if (NULL != hwnd)
  247. {
  248. WARNING_OUT(("CFindSomeone::Destroy( ): destroying windows..."));
  249. DestroyWindow(hwnd);
  250. }
  251. pDlgCall->Release();
  252. }
  253. WARNING_OUT(("Exiting CFindSomeone::Destroy( )..."));
  254. } // End of CDlgCall::Destroy.
  255. HWND CFindSomeone::GetHwndList()
  256. {
  257. return ( m_pAlv && m_pAlv->FOwnerData() ) ?
  258. m_hwndOwnerDataList : (m_pAlv == (CALV *) m_pUls)? m_ilsListView: m_hwndList;
  259. }
  260. RAI * CFindSomeone::GetAddrInfo()
  261. {
  262. return DupRai(m_pRai);
  263. }
  264. /* F M S G S P E C I A L */
  265. /*-------------------------------------------------------------------------
  266. %%Function: FMsgSpecial
  267. Return TRUE if this is a special message, not to be passed to IsDialogMessage
  268. -------------------------------------------------------------------------*/
  269. BOOL CFindSomeone::FMsgSpecial(MSG * pMsg)
  270. {
  271. switch (pMsg->message)
  272. {
  273. case WM_KEYDOWN:
  274. case WM_KEYUP:
  275. case WM_CHAR:
  276. if (VK_F5 == pMsg->wParam)
  277. return TRUE;
  278. if (VK_RETURN == pMsg->wParam)
  279. {
  280. if (m_hwndComboEdit == GetFocus())
  281. {
  282. return TRUE;
  283. }
  284. }
  285. break;
  286. default:
  287. break;
  288. }
  289. return FALSE;
  290. }
  291. /* D O M O D A L */
  292. /*-------------------------------------------------------------------------
  293. %%Function: DoModal
  294. -------------------------------------------------------------------------*/
  295. HRESULT
  296. CFindSomeone::doModeless(void)
  297. {
  298. #if 0
  299. HACCEL hAccel = LoadAccelerators(::GetInstanceHandle(), MAKEINTRESOURCE(IDA_DLGCALL));
  300. #endif
  301. HRESULT hr = CreateDlgCall(NULL);
  302. #if 0
  303. if (SUCCEEDED(hr) && (NULL != m_hwnd))
  304. {
  305. while (IsWindow(m_hwnd))
  306. {
  307. MSG msg;
  308. if (!::GetMessage(&msg, NULL, 0, 0))
  309. {
  310. // Don't eat the WM_QUIT
  311. PostQuitMessage(msg.wParam);
  312. break; // out of loop
  313. }
  314. if (FMsgSpecial(&msg) || !IsDialogMessage(m_hwnd, &msg))
  315. {
  316. if (FALSE == ::TranslateAccelerator(m_hwnd, hAccel, &msg))
  317. {
  318. ::TranslateMessage(&msg);
  319. ::DispatchMessage(&msg);
  320. }
  321. }
  322. }
  323. }
  324. #endif
  325. return hr;
  326. }
  327. // Determine the standard font size
  328. VOID CFindSomeone::CalcDyText(void)
  329. {
  330. m_dyText = 14; // default to something reasonable
  331. HWND hwnd = GetWindow();
  332. ASSERT(NULL != hwnd);
  333. HDC hdc = GetDC(hwnd);
  334. if (NULL == hdc)
  335. return;
  336. TEXTMETRIC tm;
  337. if (GetTextMetrics(hdc, &tm))
  338. {
  339. m_dyText = tm.tmHeight;
  340. }
  341. // Leave space for 4 n's (two on each side)
  342. TCHAR sz[MAX_PATH];
  343. const int cSpaceN = 4;
  344. for (int ich = 0; ich < cSpaceN; ich++)
  345. {
  346. sz[ich] = _T('n');
  347. }
  348. if (FLoadString( IDS_DLGCALL_CLOSE, &sz[cSpaceN], CCHMAX(sz)-cSpaceN))
  349. {
  350. HFONT hfont = SelectFont(hdc, ::GetDefaultFont());
  351. RECT rc;
  352. SetRect(&rc, 0, 0, 0, 0);
  353. m_dyButton = 2*DY_BORDER + DrawText(hdc, sz, -1, &rc, DT_CALCRECT);
  354. m_dxButton = rc.right;
  355. hfont = SelectFont(hdc, hfont);
  356. }
  357. ReleaseDC(hwnd, hdc);
  358. }
  359. HWND CreatIconButton(HWND hwndParent, int idTooltip, int idCmd, int idIcon)
  360. {
  361. HWND hwndButton = CreateButton( hwndParent, idTooltip, idCmd );
  362. if (NULL != hwndButton)
  363. {
  364. AddToolTip(hwndParent, hwndButton, idTooltip);
  365. SetWindowLong(hwndButton, GWL_STYLE, BS_PUSHBUTTON|BS_ICON|WS_VISIBLE|WS_CHILD|WS_TABSTOP);
  366. HANDLE hIcon = LoadImage(GetInstanceHandle(), MAKEINTRESOURCE(idIcon),
  367. IMAGE_ICON, 0, 0, 0);
  368. SendMessage(hwndButton, BM_SETIMAGE, IMAGE_ICON, reinterpret_cast<LPARAM>(hIcon));
  369. }
  370. return(hwndButton);
  371. }
  372. /* C R E A T E D L G C A L L */
  373. /*-------------------------------------------------------------------------
  374. %%Function: CreateDlgCall
  375. The order in which these windows is created is very important.
  376. See the Layout function and pay attention to the tab order
  377. -------------------------------------------------------------------------*/
  378. HRESULT CFindSomeone::CreateDlgCall(HWND hwndParent)
  379. {
  380. TCHAR sz[CCHMAXSZ];
  381. HRESULT hr = S_OK;
  382. if (NULL == m_himlIcon)
  383. return E_FAIL; // no pretty icons, don't bother trying to continue
  384. if (!FLoadString(IDS_DLGCALL_TITLE, sz, CCHMAX(sz)))
  385. return E_FAIL;
  386. // Get the window position
  387. PVOID pv;
  388. RECT rcDest; // left, right, width, height
  389. RegEntry re(DLGCALL_MRU_KEY, HKEY_CURRENT_USER);
  390. if (sizeof(rcDest) == re.GetBinary(REGVAL_DLGCALL_POSITION, &pv))
  391. {
  392. CopyMemory(&rcDest, pv, sizeof(rcDest));
  393. }
  394. else
  395. {
  396. GetDefaultRect( hwndParent, rcDest, DX_DLGCALL_DEFAULT, DY_DLGCALL_DEFAULT );
  397. rcDest.right -= rcDest.left;
  398. rcDest.bottom -= rcDest.top;
  399. }
  400. if (!CFrame::Create(
  401. hwndParent,
  402. sz,
  403. WS_OVERLAPPEDWINDOW | WS_CAPTION | WS_SYSMENU | WS_THICKFRAME,
  404. 0,
  405. rcDest.left, rcDest.top, rcDest.right, rcDest.bottom,
  406. GetInstanceHandle(),
  407. LoadIcon(GetInstanceHandle(), MAKEINTRESOURCE(IDI_WEBVIEW))
  408. ))
  409. {
  410. return E_OUTOFMEMORY;
  411. }
  412. HWND hwndThis = GetWindow();
  413. CalcDyText();
  414. //////////////////////////////////
  415. // Static text at the top
  416. HWND hwnd = CreateStaticText(hwndThis, IDS_DLGCALL_HDR);
  417. DWORD dwStyle = WS_CHILD | WS_BORDER | WS_VSCROLL | WS_VISIBLE | WS_TABSTOP |
  418. WS_CLIPCHILDREN | WS_CLIPSIBLINGS | CBS_AUTOHSCROLL;
  419. // Either a complicated combo edit/listbox or a simple drop list
  420. if ((NULL != m_pUls) && m_pUls->FAvailable())
  421. {
  422. dwStyle |= (0 != (nmDlgCallNoServerEdit & m_dwOptions)) ?
  423. CBS_DROPDOWNLIST : CBS_DROPDOWN;
  424. }
  425. else
  426. {
  427. dwStyle |= CBS_DROPDOWNLIST;
  428. }
  429. m_hwndCombo = ::CreateWindowEx(0L, g_cszComboBoxEx, g_cszEmpty, dwStyle,
  430. 0, 0, DX_BORDER, DY_COMBOBOX,
  431. hwndThis, (HMENU) IDC_DLGCALL_COMBO,
  432. ::GetInstanceHandle(), NULL);
  433. if (NULL != m_hwndCombo)
  434. {
  435. // Set the icons
  436. ::SendMessage(m_hwndCombo, CBEM_SETIMAGELIST, 0, (LPARAM) m_himlIcon);
  437. // Set the font:
  438. ::SendMessage(m_hwndCombo, WM_SETFONT, (WPARAM) GetDefaultFont(), 0);
  439. // Limit the text
  440. ::SendMessage(m_hwndCombo, CB_LIMITTEXT, CCHMAXSZ_ADDRESS-1, 0L);
  441. // Set extended user interface
  442. ::SendMessage(m_hwndCombo, CB_SETEXTENDEDUI, 1, 0L);
  443. m_hwndComboEdit = (HWND) SendMessage(m_hwndCombo, CBEM_GETEDITCONTROL, 0, 0);
  444. if (NULL != m_hwndComboEdit)
  445. {
  446. // Add a tooltip to the combo box (actually, the edit control)
  447. AddToolTip(hwndThis, m_hwndComboEdit, IDS_DLGCALL_COMBO_TOOLTIP);
  448. }
  449. else
  450. {
  451. // Add a tooltip to the combo box
  452. AddToolTip(hwndThis, m_hwndCombo, IDS_DLGCALL_COMBO_TOOLTIP);
  453. }
  454. }
  455. CreatIconButton(hwndThis, IDS_TT_DELETE_ILS, IDM_DLGCALL_DELETE_ILS, IDI_DELETE);
  456. CreatIconButton(hwndThis, IDS_TT_NEWWINDOW , IDM_DLGCALL_NEWWINDOW , IDI_NEWHTMLWINDOW);
  457. CreatIconButton(hwndThis, IDS_TT_REFRESH , IDM_DLGCALL_REFRESH , IDI_REFRESH);
  458. CreatIconButton(hwndThis, IDS_TT_HELP , ID_TB_HELP , IDI_HELP);
  459. //////////////////////////////////
  460. // GroupBox
  461. m_hwndFrame = ::CreateWindowEx( 0, TEXT( "button" ), g_cszEmpty,
  462. WS_CHILD | WS_CLIPSIBLINGS | WS_VISIBLE | WS_GROUP | BS_GROUPBOX,
  463. 0, 0, 0, 0,
  464. hwndThis, (HMENU) -1,
  465. ::GetInstanceHandle(), NULL);
  466. //////////////////////////////////
  467. // Edit Control
  468. CreateStaticText(hwndThis, IDS_DLGCALL_EDIT_HDR);
  469. m_hwndEdit = ::CreateWindowEx(WS_EX_CLIENTEDGE, g_cszEdit, g_cszEmpty,
  470. WS_CHILD | WS_BORDER | WS_VISIBLE | WS_TABSTOP | WS_CLIPCHILDREN |
  471. ES_AUTOHSCROLL,
  472. 0, 0, 0, 0,
  473. hwndThis, (HMENU) IDE_DLGCALL_NAME,
  474. ::GetInstanceHandle(), NULL);
  475. if (NULL != m_hwndEdit)
  476. {
  477. // Set the font:
  478. ::SendMessage(m_hwndEdit, WM_SETFONT, (WPARAM) GetDefaultFont(), 0);
  479. // Limit the text
  480. ::SendMessage(m_hwndEdit, EM_LIMITTEXT, CCHMAXSZ_ADDRESS-1, 0L);
  481. // Add a tooltip to the edit control
  482. AddToolTip(hwndThis, m_hwndEdit, IDS_DLGCALL_EDIT_TOOLTIP);
  483. }
  484. ////////////////////////////////
  485. // Main Listbox
  486. m_hwndList = CreateWindowEx(WS_EX_CLIENTEDGE, g_cszListView, g_cszEmpty,
  487. WS_CHILD | WS_CLIPCHILDREN | WS_BORDER | WS_TABSTOP | WS_VISIBLE |
  488. LVS_REPORT | LVS_SINGLESEL | LVS_SHOWSELALWAYS | LVS_SHAREIMAGELISTS |
  489. LVS_SORTASCENDING | LVS_NOSORTHEADER,
  490. 0, 0, 0, 0,
  491. hwndThis, (HMENU) IDL_DLGCALL_LIST,
  492. ::GetInstanceHandle(), NULL);
  493. if (NULL != m_hwndList)
  494. {
  495. ListView_SetExtendedListViewStyle(m_hwndList, LVS_EX_FULLROWSELECT);
  496. ListView_SetImageList(m_hwndList, m_himlIcon, LVSIL_SMALL);
  497. LV_COLUMN lvc;
  498. ClearStruct(&lvc);
  499. lvc.mask = LVCF_TEXT | LVCF_SUBITEM;
  500. lvc.pszText = sz;
  501. if (FLoadString(IDS_NAME, sz, CCHMAX(sz)))
  502. {
  503. lvc.iSubItem = 0;
  504. ListView_InsertColumn(m_hwndList, IDI_DLGCALL_NAME, &lvc);
  505. }
  506. if (FLoadString(IDS_ADDRESS, sz, CCHMAX(sz)))
  507. {
  508. lvc.iSubItem = 1;
  509. ListView_InsertColumn(m_hwndList, IDI_DLGCALL_ADDRESS, &lvc);
  510. }
  511. m_cVisible = ListView_GetCountPerPage(m_hwndList);
  512. }
  513. ////////////////////////////////
  514. // OwnerData Listbox
  515. m_hwndOwnerDataList = CreateWindowEx(WS_EX_CLIENTEDGE, g_cszListView, g_cszEmpty,
  516. WS_CHILD | WS_CLIPCHILDREN | WS_BORDER | WS_TABSTOP | WS_VISIBLE |
  517. LVS_REPORT | LVS_SINGLESEL | LVS_SHOWSELALWAYS | LVS_SHAREIMAGELISTS |
  518. LVS_SORTASCENDING | LVS_NOSORTHEADER | LVS_OWNERDATA,
  519. 0, 0, 0, 0,
  520. hwndThis, (HMENU) IDL_DLGCALL_LISTOWNERDATA,
  521. ::GetInstanceHandle(), NULL);
  522. if (NULL != m_hwndOwnerDataList)
  523. {
  524. ListView_SetExtendedListViewStyle(m_hwndOwnerDataList, LVS_EX_FULLROWSELECT);
  525. ListView_SetImageList(m_hwndOwnerDataList, m_himlIcon, LVSIL_SMALL);
  526. LV_COLUMN lvc;
  527. ClearStruct(&lvc);
  528. lvc.mask = LVCF_TEXT | LVCF_SUBITEM;
  529. lvc.pszText = sz;
  530. if (FLoadString(IDS_NAME, sz, CCHMAX(sz)))
  531. {
  532. lvc.iSubItem = 0;
  533. ListView_InsertColumn(m_hwndOwnerDataList, IDI_DLGCALL_NAME, &lvc);
  534. }
  535. if (FLoadString(IDS_ADDRESS, sz, CCHMAX(sz)))
  536. {
  537. lvc.iSubItem = 1;
  538. ListView_InsertColumn(m_hwndOwnerDataList, IDI_DLGCALL_ADDRESS, &lvc);
  539. }
  540. m_WndOwnerDataListOldWndProc = reinterpret_cast< WNDPROC >(GetWindowLongPtr(m_hwndOwnerDataList, GWLP_WNDPROC));
  541. SetWindowLongPtr(m_hwndOwnerDataList, GWLP_USERDATA, reinterpret_cast< LONG_PTR >( this ));
  542. SetWindowLongPtr(m_hwndOwnerDataList, GWLP_WNDPROC, reinterpret_cast< LONG_PTR >( OwnerDataListWndProc));
  543. }
  544. ////////////////////////////////
  545. // ILS Listbox
  546. m_ilsListView = CreateWindowEx( WS_EX_CLIENTEDGE,
  547. WC_LISTVIEW, // list view class
  548. TEXT( "" ), // no default text
  549. WS_VISIBLE | WS_CHILD | WS_CLIPCHILDREN | WS_VSCROLL | WS_TABSTOP |
  550. LVS_REPORT | LVS_SINGLESEL | LVS_AUTOARRANGE | LVS_SHAREIMAGELISTS | LVS_SHOWSELALWAYS,
  551. 0, 0,
  552. 0, 0,
  553. hwndThis,
  554. (HMENU) IDL_DLGCALL_ILS_LISTVIEW,
  555. ::GetInstanceHandle(),
  556. NULL );
  557. if( m_ilsListView != NULL )
  558. {
  559. InitColumns();
  560. // Associate the image list with the list view
  561. ListView_SetImageList( m_ilsListView, m_himlIcon, LVSIL_SMALL);
  562. // set the style to do drag and drop headers and full row select
  563. ListView_SetExtendedListViewStyle( m_ilsListView, ListView_GetExtendedListViewStyle( m_ilsListView ) |
  564. LVS_EX_HEADERDRAGDROP | LVS_EX_FULLROWSELECT | LVS_EX_SUBITEMIMAGES );
  565. }
  566. TCHAR szWebDir[1024];
  567. CDirectoryManager::get_webDirectoryUrl(szWebDir, ARRAY_ELEMENTS(szWebDir));
  568. m_webView = CreateWindowEx( WS_EX_CLIENTEDGE | WS_EX_CONTROLPARENT,
  569. TEXT( "AtlAxWin" ),
  570. szWebDir,
  571. WS_CHILD | WS_VISIBLE | WS_HSCROLL | WS_VSCROLL | WS_TABSTOP,
  572. 0, 0,
  573. 0, 0,
  574. hwndThis,
  575. (HMENU) IDL_DLGCALL_WEB_VIEW,
  576. ::GetInstanceHandle(),
  577. NULL );
  578. //////////////////////////////////
  579. // Security Control
  580. FLoadString( IDS_SECURITY_CHECKBOX, sz, CCHMAX( sz ) );
  581. HWND securityCheckBox;
  582. securityCheckBox = ::CreateWindow(
  583. TEXT( "button" ),
  584. sz,
  585. WS_CHILD | WS_TABSTOP | WS_VISIBLE | BS_AUTOCHECKBOX | BS_MULTILINE | BS_TOP,
  586. 0, 0, 0, 0,
  587. hwndThis,
  588. (HMENU) IDS_SECURITY_CHECKBOX,
  589. ::GetInstanceHandle(),
  590. NULL );
  591. if( securityCheckBox != NULL )
  592. {
  593. // Set the font:
  594. ::SendMessage(securityCheckBox, WM_SETFONT, (WPARAM) GetDefaultFont(), 0);
  595. UpdateSecurityCheck(m_pConfRoom, hwndThis, IDS_SECURITY_CHECKBOX);
  596. }
  597. CreateButton( hwndThis, IDS_DLGCALL_CALL, IDOK );
  598. CreateButton( hwndThis, IDS_DLGCALL_CLOSE, IDCANCEL );
  599. // Finally, initialize the data in each list
  600. InitAlv();
  601. Layout();
  602. if( m_ilsListView != NULL )
  603. {
  604. LoadColumnInfo();
  605. }
  606. if (m_pConfRoom->FIsConferenceActive())
  607. {
  608. // Simulate a call started to get all the states right
  609. OnCallStarted();
  610. }
  611. ShowWindow(hwndThis, SW_SHOWNORMAL);
  612. ::EnableWindow(GetDlgItem(hwndThis, IDOK), FALSE);
  613. if( m_pAlv )
  614. {
  615. SetFocus(m_hwndEdit);
  616. if (!FEmptySz(m_pszDefault))
  617. {
  618. SetWindowText(m_hwndEdit, m_pszDefault);
  619. }
  620. }
  621. else
  622. {
  623. hr = E_FAIL;
  624. }
  625. return hr;
  626. }
  627. bool
  628. get_textRect
  629. (
  630. const HWND window,
  631. RECT & rect,
  632. const TCHAR * const measureText = NULL
  633. );
  634. //--------------------------------------------------------------------------//
  635. // get_textRect. //
  636. //--------------------------------------------------------------------------//
  637. bool
  638. get_textRect
  639. (
  640. const HWND window,
  641. RECT & rect,
  642. const TCHAR * const measureText
  643. ){
  644. HDC dc = GetDC( window );
  645. bool result = false;
  646. if( dc != NULL )
  647. {
  648. HFONT font = SelectFont( dc, ::GetDefaultFont() );
  649. if( measureText != NULL )
  650. {
  651. DrawText( dc, measureText, -1, &rect, DT_CALCRECT | DT_WORDBREAK );
  652. result = true;
  653. }
  654. else
  655. {
  656. int length;
  657. if( (length = ::SendMessage( window, WM_GETTEXTLENGTH, 0, 0 )) > 0 )
  658. {
  659. TCHAR * text = new TCHAR [ length + 1 ];
  660. if( text != NULL )
  661. {
  662. ::SendMessage( window, WM_GETTEXT, length + 1, (LPARAM) text );
  663. DrawText( dc, text, -1, &rect, DT_CALCRECT | DT_WORDBREAK );
  664. delete [] text;
  665. result = true;
  666. }
  667. }
  668. }
  669. SelectFont( dc, font );
  670. ReleaseDC( window, dc );
  671. }
  672. return( result );
  673. } // End of get_textRect.
  674. //--------------------------------------------------------------------------//
  675. // CFindSomeone::Layout. //
  676. //--------------------------------------------------------------------------//
  677. void
  678. CFindSomeone::Layout(void)
  679. {
  680. // BUGBUG georgep: Hard coded for now
  681. static const int IconButtonWidth = 24;
  682. static const int IconButtonHeight = 24;
  683. static const int idIconButtons[] =
  684. {
  685. IDM_DLGCALL_DELETE_ILS,
  686. IDM_DLGCALL_NEWWINDOW,
  687. IDM_DLGCALL_REFRESH,
  688. ID_TB_HELP,
  689. };
  690. static const int NumIconButtons = ARRAY_ELEMENTS(idIconButtons);
  691. HWND hwnd = GetWindow();
  692. HDWP hdwp = BeginDeferWindowPos(15);
  693. // Get client area (inside margin/border)...
  694. RECT rc;
  695. GetClientRect(hwnd, &rc);
  696. int clientWidth = rc.right - 2*DX_BORDER;
  697. int clientHeight = rc.bottom - (DY_BORDER + DY_BORDER + m_dyButton);
  698. int maxComboLabelWidth = (clientWidth - NumIconButtons*(DX_BORDER + IconButtonWidth)) / 2;
  699. HWND comboLabel = GetDlgItem( hwnd, IDS_DLGCALL_HDR );
  700. RECT comboRect;
  701. RECT comboLabelRect;
  702. GetWindowRect( m_hwndCombo, &comboRect );
  703. SetRect( &comboLabelRect, 0, 0, maxComboLabelWidth, 0 );
  704. get_textRect( comboLabel, comboLabelRect );
  705. int comboHeight = RectHeight( comboRect );
  706. int comboLabelWidth = RectWidth( comboLabelRect );
  707. int comboLabelHeight = RectHeight( comboLabelRect );
  708. int maxHeight = max( comboHeight, comboLabelHeight );
  709. int comboLabelLeft = DX_BORDER;
  710. int comboLabelTop = max( DY_BORDER, (DY_BORDER + ((maxHeight - comboLabelHeight) / 2)) );
  711. hdwp = DeferWindowPos(hdwp,
  712. comboLabel,
  713. NULL,
  714. comboLabelLeft,
  715. comboLabelTop,
  716. comboLabelWidth,
  717. comboLabelHeight,
  718. SWP_NOACTIVATE | SWP_NOZORDER );
  719. int comboLeft = comboLabelLeft + comboLabelWidth + DX_BORDER;
  720. int comboTop = max( DY_BORDER, (DY_BORDER + ((maxHeight - comboHeight) / 2)) );
  721. int comboWidth = clientWidth - (comboLabelLeft + comboLabelWidth
  722. + NumIconButtons*(DX_BORDER + IconButtonWidth) );
  723. hdwp = DeferWindowPos(hdwp,
  724. m_hwndCombo,
  725. NULL,
  726. comboLeft,
  727. comboTop,
  728. comboWidth,
  729. DY_COMBOBOX,
  730. SWP_NOACTIVATE | SWP_NOZORDER );
  731. {
  732. for (int i=0; i<NumIconButtons; ++i)
  733. {
  734. hdwp = DeferWindowPos(hdwp,
  735. GetDlgItem(hwnd, idIconButtons[i]),
  736. NULL,
  737. comboLeft + comboWidth + DX_BORDER + i*(IconButtonWidth+DX_BORDER),
  738. comboTop,
  739. IconButtonWidth,
  740. IconButtonHeight,
  741. SWP_NOACTIVATE | SWP_NOZORDER );
  742. }
  743. }
  744. int frameTop = maxHeight + DY_BORDER;
  745. hdwp = DeferWindowPos(hdwp,
  746. m_hwndFrame,
  747. NULL,
  748. DX_BORDER,
  749. frameTop,
  750. clientWidth,
  751. clientHeight - frameTop,
  752. SWP_NOACTIVATE | SWP_NOZORDER );
  753. // Add a little because groups draw inside their bounds
  754. frameTop += 2 * DY_BORDER;
  755. RECT editLabelRect;
  756. HWND editLabel = GetDlgItem( hwnd, IDS_DLGCALL_EDIT_HDR );
  757. int editLabelWidth = (clientWidth - (DX_BORDER * 6)) / 2;
  758. SetRect( &editLabelRect, 0, 0, editLabelWidth, 0 );
  759. get_textRect( editLabel, editLabelRect );
  760. int editLabelHeight = RectHeight( editLabelRect );
  761. int editLabelLeft = DY_BORDER * 3;
  762. int editLabelTop = frameTop + DY_BORDER;
  763. hdwp = DeferWindowPos(hdwp,
  764. editLabel,
  765. NULL,
  766. editLabelLeft,
  767. editLabelTop,
  768. editLabelWidth,
  769. editLabelHeight,
  770. SWP_NOACTIVATE | SWP_NOZORDER );
  771. int editTop = editLabelTop + editLabelHeight + DY_BORDER;
  772. // Edit Control
  773. hdwp = DeferWindowPos(hdwp,
  774. m_hwndEdit,
  775. NULL,
  776. editLabelLeft,
  777. editTop,
  778. editLabelWidth,
  779. comboHeight,
  780. SWP_NOACTIVATE | SWP_NOZORDER );
  781. int buttonTop = clientHeight - ((DY_BORDER * 3) / 2) - m_dyButton;
  782. int listViewLeft = 3 * DX_BORDER;
  783. int listViewTop = editTop + comboHeight + DY_BORDER;
  784. int listViewWidth = clientWidth - (DX_BORDER * 4);
  785. int listViewHeight = buttonTop - listViewTop - DY_BORDER;
  786. int columns = _GetCurListViewNumColumns();
  787. int columnWidth = listViewWidth / columns;
  788. // ListBox
  789. hdwp = DeferWindowPos(hdwp,
  790. m_hwndList,
  791. NULL,
  792. listViewLeft,
  793. listViewTop,
  794. listViewWidth,
  795. listViewHeight,
  796. SWP_NOACTIVATE | SWP_NOZORDER );
  797. hdwp = DeferWindowPos(hdwp,
  798. m_hwndOwnerDataList,
  799. NULL,
  800. listViewLeft,
  801. listViewTop,
  802. listViewWidth,
  803. listViewHeight,
  804. SWP_NOACTIVATE | SWP_NOZORDER );
  805. hdwp = DeferWindowPos(hdwp,
  806. m_ilsListView,
  807. NULL,
  808. listViewLeft,
  809. listViewTop,
  810. listViewWidth,
  811. listViewHeight,
  812. SWP_NOACTIVATE | SWP_NOZORDER );
  813. HWND hWndList = GetHwndList();
  814. if( hWndList != m_ilsListView )
  815. {
  816. for( int i = 0; i < columns; i++ )
  817. {
  818. ListView_SetColumnWidth( hWndList, i, columnWidth );
  819. }
  820. }
  821. InvalidateRect( ListView_GetHeader(hWndList), NULL, true );
  822. m_cVisible = ListView_GetCountPerPage(hWndList);
  823. {
  824. int securityWidth = editLabelWidth - GetSystemMetrics( SM_CXSMICON );
  825. HWND security = GetDlgItem( hwnd, IDS_SECURITY_CHECKBOX );
  826. RECT securityRect;
  827. SetRect( &securityRect, 0, 0, securityWidth, 0 );
  828. get_textRect( security, securityRect );
  829. // + 2 for the focus rect
  830. int securityHeight = RectHeight( securityRect ) + 2;
  831. int checkHeight = GetSystemMetrics(SM_CYMENUCHECK);
  832. securityHeight = max (securityHeight, checkHeight);
  833. hdwp = DeferWindowPos(hdwp,
  834. security,
  835. NULL,
  836. listViewLeft,
  837. buttonTop,
  838. securityWidth,
  839. securityHeight,
  840. SWP_NOACTIVATE | SWP_NOZORDER );
  841. }
  842. HWND callButton = GetDlgItem( hwnd, IDOK );
  843. hdwp = DeferWindowPos(hdwp,
  844. callButton,
  845. NULL,
  846. clientWidth - DX_BORDER - m_dxButton,
  847. buttonTop,
  848. m_dxButton,
  849. m_dyButton,
  850. SWP_NOACTIVATE | SWP_NOZORDER );
  851. hdwp = DeferWindowPos(hdwp,
  852. m_webView,
  853. NULL,
  854. DX_BORDER,
  855. frameTop,
  856. clientWidth,
  857. clientHeight - frameTop,
  858. SWP_NOACTIVATE | SWP_NOZORDER );
  859. // Center the close button at the bottom of the window
  860. HWND closeButton = GetDlgItem( hwnd, IDCANCEL );
  861. hdwp = DeferWindowPos(hdwp,
  862. closeButton,
  863. NULL,
  864. (clientWidth - m_dxButton) / 2,
  865. clientHeight + DY_BORDER,
  866. m_dxButton,
  867. m_dyButton,
  868. SWP_NOACTIVATE | SWP_NOZORDER );
  869. EndDeferWindowPos(hdwp);
  870. } // End of CFindSomeone::Layout.
  871. static BOOL IsInClient(HWND hwnd, POINT *ppt)
  872. {
  873. RECT rc;
  874. GetClientRect(hwnd, &rc);
  875. MapWindowPoints(hwnd, HWND_DESKTOP, (POINT *) &rc, 2);
  876. return(PtInRect(&rc, *ppt));
  877. }
  878. void CFindSomeone::OnContextMenu(HWND hwnd, HWND hwndContext, UINT xPos, UINT yPos)
  879. {
  880. POINT pt = { xPos, yPos };
  881. if (IsInClient(GetHwndList(), &pt))
  882. {
  883. m_pAlv->OnRClick(pt);
  884. return;
  885. }
  886. if (IsInClient(GetWindow(), &pt))
  887. {
  888. DoHelpWhatsThis(reinterpret_cast<WPARAM>(hwndContext), _mpIdHelpDlgCall);
  889. return;
  890. }
  891. FORWARD_WM_CONTEXTMENU(hwnd, hwndContext, xPos, yPos, CFrame::ProcessMessage);
  892. }
  893. LRESULT CFindSomeone::ProcessMessage(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
  894. {
  895. switch (uMsg)
  896. {
  897. HANDLE_MSG(hwnd, WM_COMMAND , OnCommand);
  898. HANDLE_MSG(hwnd, WM_CONTEXTMENU, OnContextMenu);
  899. case WM_GETMINMAXINFO:
  900. {
  901. LPMINMAXINFO lpminmax = (LPMINMAXINFO) lParam;
  902. lpminmax->ptMinTrackSize.x = DX_DLGCALL_MIN;
  903. lpminmax->ptMinTrackSize.y = DY_DLGCALL_MIN;
  904. return 0;
  905. }
  906. case WM_DISPLAY_MSG:
  907. {
  908. DisplayMsgId(hwnd, (int) wParam);
  909. return 0;
  910. }
  911. case WM_CLOSE:
  912. Destroy();
  913. break;
  914. case WM_CREATE:
  915. {
  916. HACCEL hAccel = LoadAccelerators(_Module.GetModuleInstance(), MAKEINTRESOURCE(IDA_DLGCALL));
  917. if (NULL != hAccel)
  918. {
  919. m_pAccel = new CTranslateAccelTable(hwnd, hAccel);
  920. if (NULL != m_pAccel)
  921. {
  922. AddTranslateAccelerator(m_pAccel);
  923. }
  924. }
  925. // Must be done AFTER adding the accelerator table
  926. AddModelessDlg(hwnd);
  927. break;
  928. }
  929. case WM_DESTROY:
  930. {
  931. if (NULL != m_pAccel)
  932. {
  933. RemoveTranslateAccelerator(m_pAccel);
  934. m_pAccel->Release();
  935. m_pAccel = NULL;
  936. }
  937. RemoveModelessDlg(hwnd);
  938. if (NULL != m_pAlv)
  939. {
  940. m_pAlv->ClearItems();
  941. m_pAlv->ClearHwnd();
  942. }
  943. RegEntry re(DLGCALL_MRU_KEY, HKEY_CURRENT_USER);
  944. // Save the window size/position
  945. WINDOWPLACEMENT wp = { sizeof(wp), };
  946. if (GetWindowPlacement(hwnd, &wp))
  947. {
  948. RECT rc = wp.rcNormalPosition;
  949. rc.right -= rc.left;
  950. rc.bottom -= rc.top;
  951. re.SetValue(REGVAL_DLGCALL_POSITION, &rc, sizeof(rc));
  952. }
  953. // Save the selected view
  954. TCHAR szServer[CCHMAXSZ_SERVER];
  955. if (0 != GetWindowText(m_hwndComboEdit, szServer, CCHMAX(szServer)))
  956. {
  957. TRACE_OUT(("Saving last used directory is [%s]", szServer));
  958. re.SetValue(REGVAL_DLGCALL_DEFDIR, szServer);
  959. }
  960. StoreColumnInfo();
  961. return 0;
  962. }
  963. case WM_HELP:
  964. {
  965. DoHelp(lParam, _mpIdHelpDlgCall);
  966. break;
  967. }
  968. case WM_NOTIFY:
  969. {
  970. switch (wParam)
  971. {
  972. case IDL_DLGCALL_LIST:
  973. case IDL_DLGCALL_LISTOWNERDATA:
  974. case IDL_DLGCALL_ILS_LISTVIEW:
  975. return OnNotifyList(lParam);
  976. case IDC_DLGCALL_COMBO:
  977. OnNotifyCombo(lParam);
  978. break;
  979. default:
  980. break;
  981. }
  982. break;
  983. }
  984. case WM_SETTINGCHANGE:
  985. {
  986. WINDOWPLACEMENT wp;
  987. wp.length = sizeof(wp);
  988. if (!GetWindowPlacement(hwnd, &wp))
  989. break;
  990. switch (wp.showCmd)
  991. {
  992. case SW_NORMAL:
  993. case SW_RESTORE:
  994. {
  995. CalcDyText();
  996. Layout();
  997. break;
  998. }
  999. default:
  1000. break;
  1001. }
  1002. break;
  1003. }
  1004. case WM_SETCURSOR:
  1005. {
  1006. if (0 == g_cBusy)
  1007. break;
  1008. ::SetCursor(LoadCursor(NULL, MAKEINTRESOURCE(IDC_APPSTARTING)));
  1009. return TRUE;
  1010. }
  1011. case WM_SETFOCUS:
  1012. {
  1013. ::SetFocus(m_hwndEdit);
  1014. return 0;
  1015. }
  1016. default:
  1017. break;
  1018. }
  1019. return(CFrame::ProcessMessage(hwnd, uMsg, wParam, lParam));
  1020. }
  1021. /* O N N O T I F Y L I S T */
  1022. /*-------------------------------------------------------------------------
  1023. %%Function: OnNotifyList
  1024. -------------------------------------------------------------------------*/
  1025. LRESULT CFindSomeone::OnNotifyList(LPARAM lParam)
  1026. {
  1027. LPNMHDR pnmh = (LPNMHDR) lParam;
  1028. HWND hwnd = GetWindow();
  1029. switch (pnmh->code)
  1030. {
  1031. case NM_DBLCLK:
  1032. {
  1033. OnCommand(hwnd, IDOK, NULL, 0);
  1034. break;
  1035. }
  1036. case LVN_KEYDOWN:
  1037. {
  1038. LPNMLVKEYDOWN pkdn = (LPNMLVKEYDOWN) lParam;
  1039. switch (pkdn->wVKey)
  1040. {
  1041. case VK_F5:
  1042. OnCommand(hwnd, IDM_DLGCALL_REFRESH, NULL, 0);
  1043. break;
  1044. case VK_DELETE:
  1045. OnCommand(hwnd, IDM_DLGCALL_DELETE, NULL, 0);
  1046. break;
  1047. default:
  1048. break;
  1049. }
  1050. break;
  1051. }
  1052. case LVN_ITEMCHANGED:
  1053. {
  1054. CConfRoom *pConfRoom = GetConfRoom();
  1055. if (m_fInEdit || NULL == pConfRoom || !pConfRoom->IsNewCallAllowed())
  1056. break;
  1057. int iSel = m_pAlv->GetSelection();
  1058. BOOL fEnable = (-1 != iSel);
  1059. EnableWindow(GetDlgItem(hwnd, IDOK), fEnable);
  1060. break;
  1061. }
  1062. case LVN_GETDISPINFO:
  1063. {
  1064. LV_DISPINFO *lpdi = (LV_DISPINFO *)lParam;
  1065. if (lpdi->item.mask & LVIF_TEXT)
  1066. {
  1067. switch( lpdi->item.iSubItem )
  1068. {
  1069. case 0:
  1070. m_pAlv->OnListGetColumn1Data(lpdi->item.iItem, lpdi->item.cchTextMax, lpdi->item.pszText);
  1071. break;
  1072. case 1:
  1073. m_pAlv->OnListGetColumn2Data(lpdi->item.iItem, lpdi->item.cchTextMax, lpdi->item.pszText);
  1074. break;
  1075. case 2:
  1076. m_pAlv->OnListGetColumn3Data(lpdi->item.iItem, lpdi->item.cchTextMax, lpdi->item.pszText);
  1077. break;
  1078. }
  1079. }
  1080. if( lpdi->item.mask & LVIF_IMAGE )
  1081. {
  1082. lpdi->item.iImage = m_pAlv->OnListGetImageForItem( lpdi->item.iItem );
  1083. }
  1084. break;
  1085. }
  1086. case LVN_ODCACHEHINT:
  1087. {
  1088. LPNMLVCACHEHINT lpCacheHint = reinterpret_cast< LPNMLVCACHEHINT > (lParam);
  1089. m_pAlv->OnListCacheHint(lpCacheHint->iFrom, lpCacheHint->iTo);
  1090. break;
  1091. }
  1092. case LVN_ODFINDITEM:
  1093. {
  1094. LPNMLVFINDITEM lpFindItem = reinterpret_cast< LPNMLVFINDITEM > (lParam);
  1095. if (0 == (lpFindItem->lvfi.flags & (LVFI_PARTIAL | LVFI_STRING)))
  1096. break;
  1097. return m_pAlv->OnListFindItem(lpFindItem->lvfi.psz);
  1098. }
  1099. case LVN_COLUMNCLICK:
  1100. {
  1101. if( m_pAlv == m_pUls )
  1102. {
  1103. // The user clicked on one of the column headings - sort by
  1104. // this column.
  1105. TRACE_OUT(("CFindSomeone::OnNotify called (NM_COLUMNCLICK)"));
  1106. NM_LISTVIEW *pNm = (NM_LISTVIEW *)lParam;
  1107. ASSERT(pNm);
  1108. if (pNm->iSubItem == m_iSortColumn)
  1109. {
  1110. m_fSortAscending = !m_fSortAscending;
  1111. }
  1112. else if( pNm->iSubItem != -1 )
  1113. {
  1114. m_fSortAscending = TRUE;
  1115. }
  1116. if( pNm->iSubItem != -1 )
  1117. {
  1118. m_iSortColumn = pNm->iSubItem;
  1119. }
  1120. else
  1121. {
  1122. pNm->iSubItem = m_iSortColumn;
  1123. }
  1124. SendMessage( pNm->hdr.hwndFrom, WM_SETREDRAW, FALSE, 0 );
  1125. ListView_SortItems( pNm->hdr.hwndFrom,
  1126. CompareWrapper,
  1127. (LPARAM) this );
  1128. SendMessage( pNm->hdr.hwndFrom, WM_SETREDRAW, TRUE, 0 );
  1129. }
  1130. break;
  1131. }
  1132. case NM_CUSTOMDRAW:
  1133. {
  1134. return DoCustomDraw((LPNMLVCUSTOMDRAW)lParam);
  1135. }
  1136. default:
  1137. break;
  1138. }
  1139. return 0;
  1140. }
  1141. /* D O C U S T O M D R A W */
  1142. /*-------------------------------------------------------------------------
  1143. %%Function: DoCustomDraw
  1144. -------------------------------------------------------------------------*/
  1145. LRESULT CFindSomeone::DoCustomDraw(LPNMLVCUSTOMDRAW lplvcd)
  1146. {
  1147. switch (lplvcd->nmcd.dwDrawStage)
  1148. {
  1149. case CDDS_PREPAINT:
  1150. {
  1151. return CDRF_NOTIFYITEMDRAW;
  1152. }
  1153. case CDDS_ITEMPREPAINT:
  1154. {
  1155. if (!m_pAlv->IsItemBold(lplvcd->nmcd.dwItemSpec))
  1156. break;
  1157. //get the existing font
  1158. HFONT hFont = (HFONT)SendMessage(m_hwndOwnerDataList, WM_GETFONT, 0, 0);
  1159. if (NULL == hFont)
  1160. break;
  1161. LOGFONT lf;
  1162. if (0 == GetObject(hFont, sizeof(lf), &lf))
  1163. break;
  1164. lf.lfWeight = FW_BOLD;
  1165. hFont = CreateFontIndirect(&lf);
  1166. if (NULL == hFont)
  1167. break;
  1168. SelectObject(lplvcd->nmcd.hdc, hFont);
  1169. return CDRF_NEWFONT | CDRF_NOTIFYPOSTPAINT;
  1170. }
  1171. case CDDS_ITEMPOSTPAINT:
  1172. {
  1173. HFONT hFont = (HFONT)GetStockObject(DEFAULT_GUI_FONT);
  1174. if (NULL != hFont)
  1175. {
  1176. //clean up stuff here
  1177. hFont = (HFONT)SelectObject(lplvcd->nmcd.hdc, hFont);
  1178. DeleteFont(hFont);
  1179. }
  1180. break;
  1181. }
  1182. default:
  1183. break;
  1184. }
  1185. return CDRF_DODEFAULT;
  1186. }
  1187. /* U P D A T E I L S S E R V E R */
  1188. /*-------------------------------------------------------------------------
  1189. %%Function: UpdateIlsServer
  1190. Update the current ILS server based on the text in the combo's edit control.
  1191. -------------------------------------------------------------------------*/
  1192. VOID CFindSomeone::UpdateIlsServer(void)
  1193. {
  1194. TCHAR szServer[CCHMAXSZ_SERVER];
  1195. if (0 == GetWindowText(m_hwndComboEdit, szServer, CCHMAX(szServer)))
  1196. return;
  1197. const TCHAR * const displayName = CDirectoryManager::get_displayName( szServer );
  1198. const TCHAR * const dnsName = CDirectoryManager::get_dnsName( szServer );
  1199. SetWindowText( m_hwndComboEdit, displayName );
  1200. int iSel = ::SendMessage(m_hwndCombo, CB_FINDSTRINGEXACT, (WPARAM) -1, (LPARAM) displayName );
  1201. if ((CB_ERR == iSel) && (NULL != m_pUls) && m_pUls->FAvailable())
  1202. {
  1203. RegEntry rePol(POLICIES_KEY, HKEY_CURRENT_USER);
  1204. //
  1205. // Assume this is an ILS server - and add it to the list ONLY if
  1206. // not prevented by policy.
  1207. //
  1208. if (!rePol.GetNumber(REGVAL_POL_NO_ADDING_NEW_ULS, DEFAULT_POL_NO_ADDING_NEW_ULS))
  1209. {
  1210. if (NULL != m_pMruServer)
  1211. {
  1212. if( m_pMruServer->AddNewEntry( dnsName ) )
  1213. {
  1214. iSel = AddAlvSz( m_pUls, displayName, m_iIlsFirst );
  1215. }
  1216. }
  1217. }
  1218. }
  1219. if (CB_ERR == iSel)
  1220. return;
  1221. if (NULL != m_pAlv)
  1222. {
  1223. m_pAlv->ClearItems();
  1224. }
  1225. if (iSel >= m_iIlsFirst)
  1226. {
  1227. if( NULL != m_pUls )
  1228. {
  1229. m_pAlv = m_pUls;
  1230. if( !CDirectoryManager::isWebDirectory( dnsName ) )
  1231. {
  1232. m_pUls->SetServer( dnsName );
  1233. // We have to send this twice so that we don't autocomplete..
  1234. SendMessage(m_hwndCombo, CB_SETCURSEL, (WPARAM) iSel, 0);
  1235. SendMessage(m_hwndCombo, CB_SETCURSEL, (WPARAM) iSel, 0);
  1236. }
  1237. }
  1238. }
  1239. ShowList(iSel);
  1240. }
  1241. LRESULT CFindSomeone::OnNotifyCombo(LPARAM lParam)
  1242. {
  1243. NMCOMBOBOXEX * pNmcbe = (NMCOMBOBOXEX*) lParam;
  1244. switch(pNmcbe->hdr.code)
  1245. {
  1246. case CBEN_ENDEDIT:
  1247. {
  1248. PNMCBEENDEDIT pnmcbee = (PNMCBEENDEDIT) lParam;
  1249. if (_T('\0') != pnmcbee->szText[0])
  1250. {
  1251. EndComboEdit(pnmcbee->iWhy);
  1252. }
  1253. break;
  1254. }
  1255. // NOTE: the IE 3.0 comctl32.dll sends us the wrong notification
  1256. // when running under NT. We handle it here, but if we are running
  1257. // on NT using the IE 4.0 comctl32.dll, it will be handled above.
  1258. case CBEN_ENDEDITW:
  1259. {
  1260. PNMCBEENDEDITW pnmcbee = (PNMCBEENDEDITW) lParam;
  1261. if (L'\0' != pnmcbee->szText[0])
  1262. {
  1263. EndComboEdit(pnmcbee->iWhy);
  1264. }
  1265. break;
  1266. }
  1267. default:
  1268. break;
  1269. }
  1270. return 0;
  1271. }
  1272. VOID CFindSomeone::EndComboEdit(int iWhy)
  1273. {
  1274. switch (iWhy)
  1275. {
  1276. case CBENF_RETURN:
  1277. {
  1278. if (SendMessage(m_hwndCombo, CB_GETDROPPEDSTATE, 0, 0))
  1279. {
  1280. // remove the dropdown
  1281. SendMessage(m_hwndCombo, CB_SHOWDROPDOWN, 0, 0);
  1282. }
  1283. UpdateIlsServer();
  1284. break;
  1285. }
  1286. case CBENF_KILLFOCUS:
  1287. {
  1288. // remove the dropdown
  1289. SendMessage(m_hwndCombo, CB_SHOWDROPDOWN, 0, 0);
  1290. break;
  1291. }
  1292. case CBENF_DROPDOWN:
  1293. default:
  1294. break;
  1295. }
  1296. }
  1297. static void DoAxMethod(HWND hwnd, LPWSTR szMethod)
  1298. {
  1299. IUnknown *pUnk;
  1300. if (SUCCEEDED(AtlAxGetControl(hwnd, &pUnk)))
  1301. {
  1302. CComPtr<IDispatch> spDispatch = com_cast<IDispatch>(pUnk);
  1303. if (spDispatch)
  1304. {
  1305. DISPID dispid;
  1306. if (SUCCEEDED(spDispatch->GetIDsOfNames(IID_NULL, &szMethod, 1,
  1307. LOCALE_USER_DEFAULT, &dispid)))
  1308. {
  1309. DISPPARAMS dispparamsNoArgs = {NULL, NULL, 0, 0};
  1310. HRESULT hr = spDispatch->Invoke(
  1311. dispid,
  1312. IID_NULL,
  1313. LOCALE_USER_DEFAULT,
  1314. DISPATCH_METHOD,
  1315. &dispparamsNoArgs, NULL, NULL, NULL);
  1316. }
  1317. }
  1318. pUnk->Release();
  1319. }
  1320. }
  1321. static void NewHtmlWindow(HWND hwnd)
  1322. {
  1323. IUnknown *pUnk;
  1324. if (SUCCEEDED(AtlAxGetControl(hwnd, &pUnk)))
  1325. {
  1326. CComPtr<IWebBrowser> spWeb = com_cast<IWebBrowser>(pUnk);
  1327. if (spWeb)
  1328. {
  1329. BSTR bstrLocation;
  1330. if (SUCCEEDED(spWeb->get_LocationURL(&bstrLocation)))
  1331. {
  1332. USES_CONVERSION;
  1333. LaunchRedirWebPage(OLE2T(bstrLocation));
  1334. SysFreeString(bstrLocation);
  1335. }
  1336. }
  1337. pUnk->Release();
  1338. }
  1339. }
  1340. /* O N C O M M A N D */
  1341. /*-------------------------------------------------------------------------
  1342. %%Function: OnCommand
  1343. -------------------------------------------------------------------------*/
  1344. void CFindSomeone::OnCommand(HWND hwnd, int id, HWND hwndCtl, UINT codeNotify)
  1345. {
  1346. ASSERT(NULL != hwnd);
  1347. switch (id)
  1348. {
  1349. case ID_TB_HELP:
  1350. FORWARD_WM_SYSCOMMAND(hwnd, SC_CONTEXTHELP, 0, 0, SendMessage);
  1351. break;
  1352. case IDM_DLGCALL_ENTER:
  1353. if (IsWindowActive(m_hwndCombo))
  1354. {
  1355. UpdateIlsServer();
  1356. }
  1357. else
  1358. {
  1359. HWND hwndFocus = GetFocus();
  1360. TCHAR szClass[20];
  1361. GetClassName(hwndFocus, szClass, ARRAY_ELEMENTS(szClass));
  1362. if (0 == lstrcmpi(szClass, TEXT("button"))
  1363. && BS_PUSHBUTTON == (GetWindowStyle(hwndFocus)&(BS_PUSHBUTTON|BS_CHECKBOX|BS_RADIOBUTTON)))
  1364. {
  1365. // Push the button
  1366. OnCommand(hwnd, GetDlgCtrlID(hwndFocus), hwndFocus, BN_CLICKED);
  1367. }
  1368. else if (IsWindowEnabled(GetDlgItem(hwnd, IDOK)))
  1369. {
  1370. // Try to complete the call
  1371. OnCommand(hwnd, IDOK, NULL, 0);
  1372. }
  1373. else
  1374. {
  1375. MessageBeep(MB_ICONHAND);
  1376. }
  1377. }
  1378. break;
  1379. case IDC_DLGCALL_COMBO:
  1380. {
  1381. switch (codeNotify)
  1382. {
  1383. case CBN_EDITCHANGE:
  1384. {
  1385. OnEditChangeDirectory();
  1386. break;
  1387. }
  1388. case CBN_KILLFOCUS:
  1389. {
  1390. int iSel = ::SendMessage(m_hwndCombo, CB_GETCURSEL, 0, 0);
  1391. if (CB_ERR == iSel)
  1392. {
  1393. UpdateIlsServer();
  1394. }
  1395. else if (iSel != m_iSel)
  1396. {
  1397. ShowList(iSel);
  1398. }
  1399. break;
  1400. }
  1401. case CBN_SELENDOK:
  1402. {
  1403. int iSel = ::SendMessage(m_hwndCombo, CB_GETCURSEL, 0, 0);
  1404. // We have to send this so that we don't autocomplete..
  1405. SendMessage(m_hwndCombo, CB_SETCURSEL, (WPARAM) iSel, 0);
  1406. ShowList(iSel);
  1407. break;
  1408. }
  1409. default:
  1410. break;
  1411. } /* switch (HIWORD(wParam)) */
  1412. break;
  1413. }
  1414. case IDM_DLGCALL_CALL:
  1415. case IDOK:
  1416. {
  1417. CConfRoom *pConfRoom = GetConfRoom();
  1418. if (NULL == pConfRoom || !pConfRoom->IsNewCallAllowed())
  1419. {
  1420. WARNING_OUT(("Meeting settings prevent outgoing calls"));
  1421. break;
  1422. }
  1423. if (SUCCEEDED(HrGetSelection()))
  1424. {
  1425. onCall();
  1426. }
  1427. else
  1428. {
  1429. if( !hasValidUserInfo( m_pRai ) )
  1430. {
  1431. DisplayMsgErr( hwnd, IDS_NO_CALL_LOG_INFO );
  1432. }
  1433. else
  1434. {
  1435. TCHAR szName[CCHMAXSZ_NAME];
  1436. GetEditText(szName, CCHMAX(szName));
  1437. DisplayMsgErr(hwnd, IDS_ERR_CALLTO, szName);
  1438. }
  1439. }
  1440. break;
  1441. }
  1442. case IDCANCEL:
  1443. {
  1444. PostMessage( hwnd, WM_CLOSE, 0, 0 );
  1445. break;
  1446. }
  1447. case IDE_DLGCALL_NAME:
  1448. {
  1449. switch (codeNotify)
  1450. {
  1451. case EN_CHANGE:
  1452. {
  1453. TCHAR sz[CCHMAXSZ];
  1454. sz[0] = _T('\0');
  1455. GetWindowText(m_hwndEdit, sz, CCHMAX(sz));
  1456. int iSel = CFindSomeone::FindSzBySortedColumn(sz);
  1457. if (-1 != iSel)
  1458. {
  1459. HWND hwndCurr = GetHwndList();
  1460. // Select the item found
  1461. ListView_SetItemState(hwndCurr, iSel, LVIS_SELECTED | LVIS_FOCUSED, LVIS_SELECTED | LVIS_FOCUSED);
  1462. int iTop = ListView_GetTopIndex(hwndCurr);
  1463. int cDiff = (iSel - iTop);
  1464. if ((cDiff < 0) || (cDiff > m_cVisible))
  1465. {
  1466. // Scroll the item into view
  1467. RECT rc;
  1468. ListView_GetItemRect(hwndCurr, iSel, &rc, LVIR_BOUNDS);
  1469. ListView_Scroll(hwndCurr, 0, cDiff * (rc.bottom - rc.top));
  1470. }
  1471. }
  1472. break;
  1473. }
  1474. default:
  1475. break;
  1476. } /* switch (HIWORD(wParam)) */
  1477. break;
  1478. }
  1479. case IDM_DLGCALL_NEWWINDOW:
  1480. NewHtmlWindow(m_webView);
  1481. break;
  1482. case IDM_DLGCALL_REFRESH:
  1483. if (IsWindowVisible(m_webView))
  1484. {
  1485. DoAxMethod(m_webView, L"Refresh");
  1486. break;
  1487. }
  1488. // Fall through
  1489. case IDM_DLGCALL_PROPERTIES:
  1490. case IDM_DLGCALL_DELETE:
  1491. case IDM_DLGCALL_SPEEDDIAL:
  1492. case IDM_DLGCALL_WAB:
  1493. case IDM_DLGCALL_STOP:
  1494. {
  1495. if (NULL != m_pAlv)
  1496. {
  1497. m_pAlv->OnCommand(GET_WM_COMMAND_MPS(id, hwndCtl, codeNotify));
  1498. }
  1499. break;
  1500. }
  1501. case IDM_DLGCALL_DELETE_ILS:
  1502. OnDeleteIlsServer();
  1503. break;
  1504. default:
  1505. break;
  1506. }
  1507. }
  1508. int CFindSomeone::GetEditText(LPTSTR psz, int cchMax)
  1509. {
  1510. ASSERT(NULL != m_hwndEdit);
  1511. int cch = GetWindowText(m_hwndEdit, psz, cchMax);
  1512. return TrimSz(psz);
  1513. }
  1514. HRESULT CFindSomeone::HrGetSelection(void)
  1515. {
  1516. HRESULT result = E_FAIL;
  1517. // Check if we have a selection
  1518. if( ListView_GetNextItem( GetHwndList(), -1, LVNI_ALL | LVNI_SELECTED ) != -1 )
  1519. {
  1520. m_pRai = m_pAlv->GetAddrInfo();
  1521. if( hasValidUserInfo( m_pRai ) )
  1522. {
  1523. result = S_OK;
  1524. }
  1525. }
  1526. return( result );
  1527. } // End of CFindSomeone::HrGetSelection.
  1528. /* F I N D S Z */
  1529. /*-------------------------------------------------------------------------
  1530. %%Function: FindSz
  1531. First the item that matches at least the first part of the string
  1532. from the current list.
  1533. -------------------------------------------------------------------------*/
  1534. int CFindSomeone::FindSz(LPCTSTR psz)
  1535. {
  1536. LV_FINDINFO lvfi;
  1537. ClearStruct(&lvfi);
  1538. lvfi.flags = LVFI_PARTIAL;
  1539. lvfi.psz = (LPTSTR) psz;
  1540. return ListView_FindItem(GetHwndList(), -1, &lvfi);
  1541. }
  1542. //--------------------------------------------------------------------------//
  1543. // CFindSomeone::FindSzBySortedColumn. //
  1544. //--------------------------------------------------------------------------//
  1545. int CFindSomeone::FindSzBySortedColumn(LPCTSTR psz)
  1546. {
  1547. ASSERT( psz != NULL );
  1548. TCHAR pszColumnText[ MAX_PATH ];
  1549. HWND hwndList = GetHwndList();
  1550. bool bSorted = hwndList == m_ilsListView;
  1551. int iSearchColumn = bSorted ? m_iSortColumn : 0;
  1552. int iStart = 0;
  1553. int iEnd = ListView_GetItemCount( hwndList ) - 1;
  1554. int iMid = iEnd / 2;
  1555. int iCompareLength = lstrlen( psz );
  1556. int iResult = -1;
  1557. bool bAscending = (m_fSortAscending != FALSE);
  1558. if( iEnd > 0 )
  1559. {
  1560. if( bSorted && ((iSearchColumn == COLUMN_INDEX_AUDIO) || (iSearchColumn == COLUMN_INDEX_VIDEO) ))
  1561. {
  1562. bSorted = false;
  1563. // The list is sorted by av capabilities.... perform a linear search on the first (unsorted) text column...
  1564. int iColumnOrder[ MAX_DIR_COLUMNS ];
  1565. if( ListView_GetColumnOrderArray( m_ilsListView, g_rgDLColumnInfo[ DLT_ULS ].nColumns, iColumnOrder ) )
  1566. {
  1567. if( (iColumnOrder[ 0 ] == COLUMN_INDEX_AUDIO) || (iColumnOrder[ 0 ] == COLUMN_INDEX_VIDEO) )
  1568. {
  1569. if( (iColumnOrder[ 1 ] == COLUMN_INDEX_AUDIO) || (iColumnOrder[ 1 ] == COLUMN_INDEX_VIDEO) )
  1570. {
  1571. iSearchColumn = 2;
  1572. }
  1573. else
  1574. {
  1575. iSearchColumn = 1;
  1576. }
  1577. }
  1578. else
  1579. {
  1580. iSearchColumn = 0;
  1581. }
  1582. iSearchColumn = iColumnOrder[iSearchColumn];
  1583. }
  1584. }
  1585. if (!bSorted)
  1586. {
  1587. // Do a linear search
  1588. for( int nn = 0; nn <= iEnd; nn++ )
  1589. {
  1590. ListView_GetItemText( hwndList, nn, iSearchColumn, pszColumnText, CCHMAX( pszColumnText ) );
  1591. if( iCompareLength < CCHMAX( pszColumnText ) - 1 )
  1592. {
  1593. pszColumnText[ iCompareLength ] = '\0'; // Only compare iCompareLength characters...
  1594. }
  1595. if( StringCompare( psz, pszColumnText ) == 0 )
  1596. {
  1597. iResult = nn;
  1598. break;
  1599. }
  1600. }
  1601. }
  1602. else
  1603. {
  1604. while( iStart <= iEnd )
  1605. {
  1606. pszColumnText[ 0 ] = '\0';
  1607. ListView_GetItemText( hwndList, iMid, iSearchColumn, pszColumnText, CCHMAX( pszColumnText ) );
  1608. if( iCompareLength < CCHMAX( pszColumnText ) - 1 )
  1609. {
  1610. pszColumnText[ iCompareLength ] = '\0'; // Only compare iCompareLength characters...
  1611. }
  1612. int iCompareResult = StringCompare( psz, pszColumnText );
  1613. bool pszIsLess = (iCompareResult <= 0);
  1614. if( iCompareResult == 0 )
  1615. {
  1616. // We've found a match but we will keep going because there may be more than one
  1617. // and we want to find the "first" one...
  1618. iResult = iMid;
  1619. }
  1620. else
  1621. {
  1622. if( !bAscending )
  1623. {
  1624. pszIsLess = !pszIsLess;
  1625. }
  1626. }
  1627. if( pszIsLess )
  1628. {
  1629. iEnd = iMid - 1;
  1630. }
  1631. else
  1632. {
  1633. iStart = iMid + 1;
  1634. }
  1635. iMid = iStart + ((iEnd - iStart) / 2);
  1636. }
  1637. }
  1638. }
  1639. return( iResult );
  1640. } // End of CFindSomeone::FindSzBySortedColumn.
  1641. VOID CFindSomeone::OnEditChangeDirectory(void)
  1642. {
  1643. if (m_fInEdit)
  1644. return;
  1645. TCHAR szEdit[CCHMAXSZ_ADDRESS];
  1646. TCHAR szBuff[CCHMAXSZ_ADDRESS];
  1647. szEdit[0] = _T('\0');
  1648. int cch = GetWindowText(m_hwndCombo, szEdit, CCHMAX(szEdit));
  1649. CharUpperBuff(szEdit, CCHMAX(szEdit));
  1650. if (0 == cch)
  1651. return;
  1652. // Check if anything changed
  1653. PTCHAR pchSrc = m_szDirectory;
  1654. PTCHAR pchDest = szEdit;
  1655. while ((_T('\0') != *pchSrc) && (*pchSrc == *pchDest))
  1656. {
  1657. // REVIEW: not DBCS safe
  1658. pchSrc++;
  1659. pchDest++;
  1660. }
  1661. if (_T('\0') == *pchDest)
  1662. {
  1663. lstrcpy(m_szDirectory, szEdit);
  1664. return; // quick exit - nothing to do
  1665. }
  1666. if (-1 == FindSzCombo(m_hwndCombo, szEdit, szBuff))
  1667. return;
  1668. m_fInEdit = TRUE;
  1669. AutoCompleteCombo(m_hwndCombo, &szBuff[cch]);
  1670. lstrcpyn(m_szDirectory, szBuff, cch+1);
  1671. CharUpperBuff(m_szDirectory, CCHMAX(m_szDirectory));
  1672. m_fInEdit = FALSE;
  1673. }
  1674. /* I N I T A L V */
  1675. /*-------------------------------------------------------------------------
  1676. %%Function: InitAlv
  1677. -------------------------------------------------------------------------*/
  1678. VOID CFindSomeone::InitAlv(void)
  1679. {
  1680. int iSelDefault = LB_ERR;
  1681. ASSERT( NULL == m_pAlv );
  1682. RegEntry reConf(CONFERENCING_KEY, HKEY_CURRENT_USER);
  1683. bool bGkEnabled = CALLING_MODE_GATEKEEPER == reConf.GetNumber(REGVAL_CALLING_MODE, CALLING_MODE_DIRECT );
  1684. #if USE_GAL
  1685. // Add each of the view items to the list
  1686. if ((NULL != m_pGAL) && m_pGAL->FAvailable())
  1687. {
  1688. AddAlv(m_pGAL);
  1689. if( !m_pAlv )
  1690. {
  1691. m_pAlv = m_pGAL;
  1692. }
  1693. }
  1694. #endif // #if USE_GAL
  1695. if((!bGkEnabled) && (NULL != m_pSpeedDial) && m_pSpeedDial->FAvailable())
  1696. {
  1697. AddAlv(m_pSpeedDial);
  1698. if( !m_pAlv )
  1699. {
  1700. m_pAlv = m_pSpeedDial;
  1701. }
  1702. }
  1703. if ( (!bGkEnabled) && ( NULL != m_pHistory) && m_pHistory->FAvailable())
  1704. {
  1705. AddAlv(m_pHistory);
  1706. if( !m_pAlv )
  1707. {
  1708. m_pAlv = m_pHistory;
  1709. }
  1710. }
  1711. if ((NULL != m_pWab) && m_pWab->FAvailable())
  1712. {
  1713. AddAlv(m_pWab);
  1714. if( !m_pAlv )
  1715. {
  1716. m_pAlv = m_pWab;
  1717. }
  1718. }
  1719. m_iIlsFirst = ::SendMessage(m_hwndCombo, CB_GETCOUNT, 0, 0);
  1720. // Add the list of ILS servers
  1721. if (bGkEnabled)
  1722. {
  1723. // If there is a customized Web View, we will show it in GK mode as
  1724. // well as in ILS mode
  1725. if ((NULL != m_pUls) && ConfPolicies::GetWebDirInfo())
  1726. {
  1727. AddAlvSz( m_pUls, CDirectoryManager::get_displayName( CDirectoryManager::get_webDirectoryIls() ));
  1728. }
  1729. }
  1730. else
  1731. {
  1732. if((NULL != m_pUls) && m_pUls->FAvailable() && (NULL != m_pMruServer) && !_IsDirectoryServicePolicyDisabled())
  1733. {
  1734. m_pAlv = m_pUls;
  1735. for( int i = m_pMruServer->GetNumEntries() - 1; i >= 0; i-- )
  1736. {
  1737. AddAlvSz( m_pUls, CDirectoryManager::get_displayName( m_pMruServer->GetNameEntry( i ) ) );
  1738. }
  1739. }
  1740. }
  1741. iSelDefault = CB_ERR;
  1742. if( m_pAlv )
  1743. {
  1744. // We know that there is at lest one directory server in the combobox
  1745. // Get the user's last selection
  1746. RegEntry reMru(DLGCALL_MRU_KEY, HKEY_CURRENT_USER);
  1747. LPTSTR psz = reMru.GetString(REGVAL_DLGCALL_DEFDIR);
  1748. if (!FEmptySz(psz))
  1749. {
  1750. TRACE_OUT(("Last directory was [%s]",psz));
  1751. iSelDefault = ::SendMessage(m_hwndCombo, CB_FINDSTRINGEXACT, (WPARAM) -1, (LPARAM) psz);
  1752. }
  1753. else
  1754. {
  1755. // This may be the first time, and there is no last use one in the registry
  1756. TCHAR szDirectoryName[CCHMAXSZ];
  1757. if( m_pAlv )
  1758. {
  1759. if( m_pAlv != m_pUls )
  1760. {
  1761. m_pAlv->GetName(szDirectoryName, CCHMAX(szDirectoryName));
  1762. iSelDefault = ::SendMessage(m_hwndCombo, CB_FINDSTRINGEXACT, (WPARAM) -1, (LPARAM) szDirectoryName);
  1763. }
  1764. else if( m_pMruServer->GetNumEntries() > 0 )
  1765. {
  1766. iSelDefault = m_iIlsFirst;
  1767. }
  1768. }
  1769. }
  1770. iSelDefault = (CB_ERR != iSelDefault) ? iSelDefault : 0;
  1771. ShowList(iSelDefault);
  1772. // We have to send this twice so that we don't autocomplete..
  1773. SendMessage(m_hwndCombo, CB_SETCURSEL, (WPARAM) iSelDefault, 0);
  1774. SendMessage(m_hwndCombo, CB_SETCURSEL, (WPARAM) iSelDefault, 0);
  1775. }
  1776. else
  1777. {
  1778. // The user has no directory services set up
  1779. DisplayMsgErr( NULL, IDS_NO_DIRECTORY_SERVICES_OR_ADDRESS_BOOKS );
  1780. }
  1781. }
  1782. bool CFindSomeone::_IsDirectoryServicePolicyDisabled()
  1783. {
  1784. RegEntry rePol(POLICIES_KEY, HKEY_CURRENT_USER);
  1785. return rePol.GetNumber( REGVAL_POL_NO_DIRECTORY_SERVICES, DEFAULT_POL_NO_DIRECTORY_SERVICES ) ? true : false;
  1786. }
  1787. int CFindSomeone::AddAlv(CALV * pAlv)
  1788. {
  1789. TCHAR sz[CCHMAXSZ];
  1790. pAlv->GetName(sz, CCHMAX(sz));
  1791. return AddAlvSz(pAlv, sz);
  1792. }
  1793. int CFindSomeone::AddAlvSz(CALV * pAlv, LPCTSTR psz, int cbIndex)
  1794. {
  1795. ASSERT(NULL != pAlv);
  1796. ASSERT(NULL != m_hwndCombo);
  1797. COMBOBOXEXITEM cbi;
  1798. ClearStruct(&cbi);
  1799. cbi.mask = CBEIF_TEXT | CBEIF_IMAGE | CBEIF_SELECTEDIMAGE;
  1800. cbi.iItem = cbIndex;
  1801. cbi.iImage = pAlv->GetIconId(psz);
  1802. cbi.iSelectedImage = cbi.iImage;
  1803. cbi.pszText = (LPTSTR) psz;
  1804. cbi.cchTextMax = lstrlen(cbi.pszText);
  1805. int iIndex = ::SendMessage(m_hwndCombo, CBEM_INSERTITEM, 0, (LPARAM) &cbi);
  1806. if (CB_ERR != iIndex)
  1807. {
  1808. ::SendMessage(m_hwndCombo, CB_SETITEMDATA, iIndex, (LPARAM) pAlv);
  1809. }
  1810. return iIndex;
  1811. }
  1812. /* S H O W L I S T */
  1813. /*-------------------------------------------------------------------------
  1814. %%Function: ShowList
  1815. -------------------------------------------------------------------------*/
  1816. VOID CFindSomeone::ShowList(int iSel)
  1817. {
  1818. HWND hwnd = GetWindow();
  1819. if (CB_ERR == iSel)
  1820. return;
  1821. m_iSel = iSel;
  1822. BOOL bPrevListWasOwnerData = m_pAlv->FOwnerData();
  1823. int nColsPrevListView = _GetCurListViewNumColumns();
  1824. BOOL bPrevChangedNumCols = FALSE;
  1825. // Clear any existing data
  1826. if (NULL != m_pAlv)
  1827. {
  1828. m_pAlv->ClearItems();
  1829. if( nColsPrevListView != _GetCurListViewNumColumns() )
  1830. {
  1831. bPrevChangedNumCols = TRUE;
  1832. }
  1833. }
  1834. // Get the new selection
  1835. m_pAlv = (CALV *) ::SendMessage(m_hwndCombo, CB_GETITEMDATA, iSel, 0);
  1836. ASSERT(m_pAlv != NULL);
  1837. ASSERT(m_pAlv != (CALV *) -1);
  1838. BOOL bCurListIsOwnerData = m_pAlv->FOwnerData();
  1839. bool webDir = false;
  1840. TCHAR szServer[CCHMAXSZ_SERVER];
  1841. szServer[0] = 0;
  1842. if (m_pAlv == m_pUls)
  1843. {
  1844. COMBOBOXEXITEM cbi;
  1845. ClearStruct(&cbi);
  1846. cbi.mask = CBEIF_TEXT;
  1847. cbi.iItem = iSel;
  1848. cbi.pszText = szServer;
  1849. cbi.cchTextMax = ARRAY_ELEMENTS(szServer);
  1850. ::SendMessage(m_hwndCombo, CBEM_GETITEM, iSel, (LPARAM)&cbi);
  1851. TRACE_OUT(("CBEM_GETITEM got %s for text", szServer));
  1852. webDir = CDirectoryManager::isWebDirectory( szServer );
  1853. }
  1854. // Show or hide all windows between HELP and CANCEL
  1855. HWND start = GetDlgItem(hwnd, ID_TB_HELP);
  1856. HWND end = GetDlgItem(hwnd, IDCANCEL);
  1857. if (NULL != start)
  1858. {
  1859. UINT uShow = webDir ? SW_HIDE : SW_SHOW;
  1860. for (HWND child=::GetWindow(start, GW_HWNDNEXT);
  1861. child!=NULL && child!=end; child=::GetWindow(child, GW_HWNDNEXT))
  1862. {
  1863. ShowWindow(child, uShow);
  1864. }
  1865. ShowWindow(m_webView, SW_HIDE+SW_SHOW-uShow);
  1866. }
  1867. EnableWindow(GetDlgItem(hwnd, IDM_DLGCALL_NEWWINDOW), webDir);
  1868. EnableWindow( GetDlgItem( hwnd, IDM_DLGCALL_DELETE_ILS ), FALSE );
  1869. if (m_pAlv == m_pUls)
  1870. {
  1871. ::ShowWindow( m_hwndOwnerDataList, SW_HIDE );
  1872. ::ShowWindow( m_hwndList, SW_HIDE );
  1873. if( webDir )
  1874. {
  1875. ::SetFocus( m_webView );
  1876. }
  1877. else
  1878. {
  1879. m_pUls->SetServer( CDirectoryManager::get_dnsName( szServer ) );
  1880. m_pAlv->ShowItems(GetHwndList());
  1881. ::SetFocus( m_ilsListView );
  1882. RegEntry rePol( POLICIES_KEY, HKEY_CURRENT_USER );
  1883. if (!rePol.GetNumber(REGVAL_POL_NO_ADDING_NEW_ULS, DEFAULT_POL_NO_ADDING_NEW_ULS))
  1884. {
  1885. EnableWindow( GetDlgItem( hwnd, IDM_DLGCALL_DELETE_ILS ), TRUE );
  1886. }
  1887. }
  1888. }
  1889. else
  1890. {
  1891. if( m_ilsListView != NULL )
  1892. {
  1893. ::ShowWindow( m_ilsListView, SW_HIDE );
  1894. ::ShowWindow( m_webView, SW_HIDE );
  1895. m_pUls->CacheServerData();
  1896. }
  1897. if (m_pAlv->FOwnerData())
  1898. {
  1899. ::ShowWindow(m_hwndList, SW_HIDE);
  1900. }
  1901. else
  1902. {
  1903. ::ShowWindow(m_hwndOwnerDataList, SW_HIDE);
  1904. }
  1905. m_pAlv->ShowItems(GetHwndList());
  1906. ::SetFocus( GetHwndList() );
  1907. }
  1908. if( bPrevChangedNumCols || ( bCurListIsOwnerData != bPrevListWasOwnerData ) || ( nColsPrevListView != _GetCurListViewNumColumns() ) )
  1909. {
  1910. Layout();
  1911. }
  1912. }
  1913. int CFindSomeone::_GetCurListViewNumColumns()
  1914. {
  1915. int nListViewColumns = DEFAULT_NUM_LISTVIEW_COLUMNS;
  1916. HWND hListViewHeader = NULL;
  1917. HWND hWndList = GetHwndList();
  1918. if( hWndList )
  1919. {
  1920. hListViewHeader = ListView_GetHeader( hWndList );
  1921. if( hListViewHeader )
  1922. {
  1923. nListViewColumns = Header_GetItemCount( hListViewHeader );
  1924. }
  1925. }
  1926. return nListViewColumns;
  1927. }
  1928. /* D L G C A L L S E T H E A D E R */
  1929. /*-------------------------------------------------------------------------
  1930. %%Function: DlgCallSetHeader
  1931. Set the listbox header text for the "address" / "email" / "status" field.
  1932. -------------------------------------------------------------------------*/
  1933. VOID DlgCallSetHeader(HWND hwndList, int ids)
  1934. {
  1935. HWND hwnd = ListView_GetHeader(hwndList);
  1936. if (NULL == hwnd)
  1937. return;
  1938. TCHAR sz[CCHMAXSZ];
  1939. if (!FLoadString(ids, sz, CCHMAX(sz)))
  1940. return;
  1941. HDITEM hdItem;
  1942. ClearStruct(&hdItem);
  1943. hdItem.mask = HDI_TEXT; // | HDI_IMAGE; hdItem.iImage = II_ASCENDING;
  1944. hdItem.pszText = sz;
  1945. hdItem.cchTextMax = lstrlen(hdItem.pszText);
  1946. Header_SetItem(hwnd, IDI_DLGCALL_ADDRESS, &hdItem);
  1947. }
  1948. /* D L G C A L L A D D I T E M */
  1949. /*-------------------------------------------------------------------------
  1950. %%Function: DlgCallAddItem
  1951. Add the name and address to the list, returning the position.
  1952. -------------------------------------------------------------------------*/
  1953. int DlgCallAddItem(HWND hwndList, LPCTSTR pszName, LPCTSTR pszAddress, int iImage, LPARAM lParam, int iItem, LPCTSTR pszComment)
  1954. {
  1955. LV_ITEM lvItem;
  1956. ClearStruct(&lvItem);
  1957. lvItem.mask = LVIF_TEXT | LVIF_PARAM | LVIF_IMAGE;
  1958. lvItem.iItem = iItem;
  1959. lvItem.lParam = lParam;
  1960. lvItem.iImage = iImage;
  1961. lvItem.pszText = (LPTSTR) pszName;
  1962. lvItem.cchTextMax = lstrlen(lvItem.pszText);
  1963. int iPos = ListView_InsertItem(hwndList, &lvItem);
  1964. if (-1 != iPos)
  1965. {
  1966. ListView_SetItemText(hwndList, iPos, IDI_DLGCALL_ADDRESS, (LPTSTR) pszAddress);
  1967. if( pszComment )
  1968. {
  1969. ListView_SetItemText(hwndList, iPos, IDI_DLGCALL_COMMENT, (LPTSTR) pszComment);
  1970. }
  1971. }
  1972. return iPos;
  1973. }
  1974. /* C A L L T O S Z */
  1975. /*-------------------------------------------------------------------------
  1976. %%Function: CallToSz
  1977. Call an address using "CallTo:"
  1978. -------------------------------------------------------------------------*/
  1979. HRESULT CallToSz(LPCTSTR pcszAddress)
  1980. {
  1981. #ifdef DEBUG
  1982. RegEntry re(DEBUG_KEY, HKEY_LOCAL_MACHINE);
  1983. if (0 != re.GetNumber(REGVAL_DBG_FAKE_CALLTO, DEFAULT_DBG_FAKE_CALLTO))
  1984. {
  1985. MessageBox(NULL, pcszAddress, "Called", MB_OK);
  1986. return S_OK;
  1987. }
  1988. #endif /* DEBUG */
  1989. if (FEmptySz(pcszAddress))
  1990. return E_INVALIDARG;
  1991. TCHAR sz[MAX_PATH];
  1992. lstrcpy(sz, g_cszCallTo);
  1993. int cch = lstrlen(sz);
  1994. if (CCHMAX(sz) <= (cch + lstrlen(pcszAddress)))
  1995. return E_INVALIDARG; // the address won't fit
  1996. lstrcpy(&sz[cch], pcszAddress);
  1997. HINSTANCE hInst = ShellExecute(NULL, NULL, sz, NULL, NULL, SW_SHOWNORMAL);
  1998. return ((INT_PTR)hInst > 32) ? S_OK : E_FAIL;
  1999. }
  2000. //--------------------------------------------------------------------------//
  2001. // GetMruListServer. //
  2002. //--------------------------------------------------------------------------//
  2003. CMRUList *
  2004. GetMruListServer(void)
  2005. {
  2006. CMRUList * pMruList = new CMRUList;
  2007. if( pMruList != NULL )
  2008. {
  2009. pMruList->Load( DIR_MRU_KEY );
  2010. if( CDirectoryManager::isWebDirectoryEnabled() )
  2011. {
  2012. // Make sure the web directory is in the list...
  2013. pMruList->AppendEntry( CDirectoryManager::get_webDirectoryIls() );
  2014. }
  2015. const TCHAR * const defaultServer = CDirectoryManager::get_defaultServer();
  2016. if( lstrlen( defaultServer ) > 0 )
  2017. {
  2018. // Make sure the default server name is in the list and at the top...
  2019. pMruList->AddNewEntry( defaultServer );
  2020. }
  2021. }
  2022. return( pMruList );
  2023. } // End of GetMruListServer.
  2024. // The only purpose for this function is to avoid scrollbar tracking of the data
  2025. /* static */
  2026. LRESULT CALLBACK CFindSomeone::OwnerDataListWndProc( HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam )
  2027. {
  2028. CFindSomeone* pThis = reinterpret_cast< CFindSomeone* >( GetWindowLongPtr( hwnd, GWLP_USERDATA ) );
  2029. WNDPROC lpPrevWndFunc = pThis->m_WndOwnerDataListOldWndProc;
  2030. bool bUserDroppedScrollSlider = false;
  2031. if (WM_VSCROLL == uMsg)
  2032. {
  2033. switch (LOWORD(wParam))
  2034. {
  2035. case SB_THUMBTRACK:
  2036. // don't track
  2037. return FALSE;
  2038. case SB_THUMBPOSITION:
  2039. bUserDroppedScrollSlider = true;
  2040. // fake a final SB_THUMBTRACK notification
  2041. CallWindowProc(lpPrevWndFunc, hwnd, uMsg, MAKEWPARAM(SB_THUMBTRACK, HIWORD(wParam)), lParam);
  2042. break;
  2043. default:
  2044. break;
  2045. }
  2046. }
  2047. LRESULT lRet = CallWindowProc(lpPrevWndFunc, hwnd, uMsg, wParam, lParam);
  2048. if( bUserDroppedScrollSlider )
  2049. {
  2050. ListView_SetItemState(hwnd, GetScrollPos( hwnd, SB_VERT ), LVIS_SELECTED | LVIS_FOCUSED, LVIS_SELECTED | LVIS_FOCUSED);
  2051. }
  2052. return lRet;
  2053. }
  2054. //--------------------------------------------------------------------------//
  2055. //--------------------------------------------------------------------------//
  2056. // copied from 2.11 dirlist.cpp. //
  2057. //--------------------------------------------------------------------------//
  2058. //--------------------------------------------------------------------------//
  2059. //--------------------------------------------------------------------------//
  2060. // CFindSomeone::InitColumns. //
  2061. //--------------------------------------------------------------------------//
  2062. BOOL CFindSomeone::InitColumns(void)
  2063. {
  2064. BOOL bRet = FALSE;
  2065. if (NULL != m_ilsListView)
  2066. {
  2067. // Remove all columns:
  2068. // NOTE: we could optimize by removing only extras and changing the rest
  2069. while( ListView_DeleteColumn( m_ilsListView, 0 ) ){};
  2070. // Now initialize the columns we will need
  2071. // Initialize the LV_COLUMN structure
  2072. // the mask specifies that the .fmt, .ex, width, and .subitem members
  2073. // of the structure are valid,
  2074. LV_COLUMN lvC; // List View Column structure
  2075. TCHAR szText[256]; // place to store some text
  2076. lvC.pszText = szText;
  2077. lvC.fmt = LVCFMT_LEFT; // left align the column
  2078. // Add the columns.
  2079. for (int index = 0; index < g_rgDLColumnInfo[DLT_ULS].nColumns; index++)
  2080. {
  2081. HD_ITEM hdi;
  2082. switch( index )
  2083. {
  2084. case COLUMN_INDEX_AUDIO:
  2085. hdi.iImage = II_AUDIO_COLUMN_HEADER;
  2086. lvC.mask = LVCF_SUBITEM;
  2087. break;
  2088. case COLUMN_INDEX_VIDEO:
  2089. hdi.iImage = II_VIDEO_COLUMN_HEADER;
  2090. lvC.mask = LVCF_SUBITEM;
  2091. break;
  2092. default:
  2093. hdi.iImage = -1;
  2094. lvC.mask = LVCF_TEXT | LVCF_FMT | LVCF_SUBITEM;
  2095. LoadString( ::GetInstanceHandle(),
  2096. ciColumnInfo[ index ].iColumnLabelIds,
  2097. szText,
  2098. CCHMAX(szText));
  2099. break;
  2100. }
  2101. lvC.iSubItem = index;
  2102. if( ListView_InsertColumn( m_ilsListView, index, &lvC ) == -1 )
  2103. {
  2104. WARNING_OUT(("Could not insert column %d in list view", index));
  2105. }
  2106. else if( hdi.iImage != -1 )
  2107. {
  2108. hdi.mask = HDI_IMAGE | HDI_FORMAT;
  2109. hdi.fmt = HDF_IMAGE;
  2110. Header_SetItem( ListView_GetHeader( m_ilsListView ), index, &hdi );
  2111. }
  2112. }
  2113. Header_SetImageList( ListView_GetHeader( m_ilsListView ), m_himlIcon );
  2114. bRet = TRUE;
  2115. }
  2116. // ISSUE: do we want to persist this data?
  2117. m_fSortAscending = TRUE;
  2118. m_iSortColumn = COLUMN_INDEX_LAST_NAME;
  2119. return bRet;
  2120. }
  2121. //--------------------------------------------------------------------------//
  2122. // CFindSomeone::LoadColumnInfo. //
  2123. //--------------------------------------------------------------------------//
  2124. BOOL
  2125. CFindSomeone::LoadColumnInfo(void)
  2126. {
  2127. RegEntry re(UI_KEY, HKEY_CURRENT_USER);
  2128. // load column info from registry:
  2129. LPLONG plColumns = NULL;
  2130. DWORD dwLength = re.GetBinary( g_rgDLColumnInfo[DLT_ULS].pszRVWidths,
  2131. (LPVOID*) &plColumns);
  2132. if (dwLength == (sizeof(LONG) * g_rgDLColumnInfo[DLT_ULS].nColumns))
  2133. {
  2134. // get width of each column
  2135. for (int i = 0; i < g_rgDLColumnInfo[DLT_ULS].nColumns; i++)
  2136. {
  2137. m_alColumns[i] = plColumns[i];
  2138. ListView_SetColumnWidth(m_ilsListView, i, m_alColumns[i]);
  2139. }
  2140. }
  2141. else
  2142. {
  2143. int iMinColumnPixels[ MAX_DIR_COLUMNS ];
  2144. int iPixelsPerChar = GetPixelsPerChar( m_ilsListView );
  2145. int ii;
  2146. // Loop through all the columns setting their minimum widths...
  2147. for( ii = 0; ii < g_rgDLColumnInfo[ DLT_ULS ].nColumns; ii++ )
  2148. {
  2149. if( (ii == COLUMN_INDEX_AUDIO) || (ii == COLUMN_INDEX_VIDEO) )
  2150. {
  2151. // We use a 16x16 icon, plus add 7 pixels for padding.
  2152. // There's also padding built into the icons to make them look more centered....
  2153. // The column header icons are aligned left and the column icons are aligned right...
  2154. iMinColumnPixels[ ii ] = 23;
  2155. }
  2156. else
  2157. {
  2158. iMinColumnPixels[ ii ] = ciColumnInfo[ ii ].iMinColumnChars * iPixelsPerChar;
  2159. }
  2160. }
  2161. // Loop through all but the last column setting each columns to it's minimum width...
  2162. for( ii = 0; ii < g_rgDLColumnInfo[ DLT_ULS ].nColumns - 1; ii++ )
  2163. {
  2164. ListView_SetColumnWidth( m_ilsListView, ii, iMinColumnPixels[ ii ] );
  2165. m_alColumns[ ii ] = ListView_GetColumnWidth( m_ilsListView, ii );
  2166. }
  2167. // The last column gets the rest...
  2168. int iLastColoumnIndex = g_rgDLColumnInfo[ DLT_ULS ].nColumns - 1;
  2169. ListView_SetColumnWidth( m_ilsListView, iLastColoumnIndex, LVSCW_AUTOSIZE_USEHEADER );
  2170. m_alColumns[ iLastColoumnIndex ] = ListView_GetColumnWidth( m_ilsListView, iLastColoumnIndex );
  2171. if( m_alColumns[ iLastColoumnIndex ] < iMinColumnPixels[ iLastColoumnIndex ] )
  2172. {
  2173. ListView_SetColumnWidth( m_ilsListView, iLastColoumnIndex, iMinColumnPixels[ iLastColoumnIndex ] );
  2174. m_alColumns[ iLastColoumnIndex ] = ListView_GetColumnWidth( m_ilsListView, iLastColoumnIndex );
  2175. }
  2176. }
  2177. // set column order
  2178. LPLONG plColumnOrder = NULL;
  2179. dwLength = re.GetBinary( g_rgDLColumnInfo[DLT_ULS].pszRVOrder,
  2180. (LPVOID*) &plColumnOrder);
  2181. if (dwLength == (sizeof(LONG) * g_rgDLColumnInfo[DLT_ULS].nColumns))
  2182. {
  2183. ListView_SetColumnOrderArray( m_ilsListView,
  2184. g_rgDLColumnInfo[DLT_ULS].nColumns,
  2185. plColumnOrder);
  2186. // load the sort column and direction
  2187. m_iSortColumn = re.GetNumber(g_rgDLColumnInfo[DLT_ULS].pszRVSortColumn, COLUMN_INDEX_LAST_NAME );
  2188. ASSERT(m_iSortColumn < g_rgDLColumnInfo[DLT_ULS].nColumns);
  2189. m_fSortAscending = re.GetNumber(g_rgDLColumnInfo[DLT_ULS].pszRVSortAscending, TRUE);
  2190. }
  2191. else
  2192. {
  2193. const static int iDefaultColumnOrder[] = { COLUMN_INDEX_AUDIO, COLUMN_INDEX_VIDEO, COLUMN_INDEX_LAST_NAME, COLUMN_INDEX_FIRST_NAME, COLUMN_INDEX_ADDRESS, COLUMN_INDEX_LOCATION, COLUMN_INDEX_COMMENTS };
  2194. ListView_SetColumnOrderArray( m_ilsListView, ARRAY_ELEMENTS( iDefaultColumnOrder ), iDefaultColumnOrder );
  2195. }
  2196. return TRUE;
  2197. }
  2198. //--------------------------------------------------------------------------//
  2199. // CFindSomeone::CompareWrapper. //
  2200. //--------------------------------------------------------------------------//
  2201. int
  2202. CALLBACK
  2203. CFindSomeone::CompareWrapper
  2204. (
  2205. LPARAM param1,
  2206. LPARAM param2,
  2207. LPARAM This
  2208. ){
  2209. return( ((CFindSomeone *) This)->DirListViewCompareProc( param1, param2 ) );
  2210. } // End of CFindSomeone::CompareWrapper.
  2211. //--------------------------------------------------------------------------//
  2212. // CFindSomeone::DirListViewCompareProc. //
  2213. //--------------------------------------------------------------------------//
  2214. int
  2215. CFindSomeone::DirListViewCompareProc
  2216. (
  2217. LPARAM param1,
  2218. LPARAM param2
  2219. ){
  2220. LV_ITEM lvi;
  2221. int result;
  2222. if( (m_iSortColumn == COLUMN_INDEX_AUDIO) || (m_iSortColumn == COLUMN_INDEX_VIDEO) )
  2223. {
  2224. lvi.mask = LVIF_IMAGE;
  2225. lvi.iItem = LParamToPos( param1 );
  2226. lvi.iSubItem = m_iSortColumn;
  2227. ListView_GetItem( m_ilsListView, &lvi );
  2228. int iImage1 = lvi.iImage;
  2229. lvi.iItem = LParamToPos( param2 );
  2230. ListView_GetItem( m_ilsListView, &lvi );
  2231. result = lvi.iImage - iImage1;
  2232. }
  2233. else
  2234. {
  2235. // BUGBUG: need better constant for max size
  2236. TCHAR szText1[ MAX_PATH ];
  2237. TCHAR szText2[ MAX_PATH ];
  2238. lvi.mask = LVIF_TEXT;
  2239. lvi.iItem = LParamToPos( param1 );
  2240. lvi.iSubItem = m_iSortColumn;
  2241. lvi.pszText = szText1;
  2242. lvi.cchTextMax = CCHMAX( szText1 );
  2243. ListView_GetItem( m_ilsListView, &lvi );
  2244. lvi.iItem = LParamToPos( param2 );
  2245. lvi.pszText = szText2;
  2246. ListView_GetItem( m_ilsListView, &lvi );
  2247. result = StringCompare( szText1, szText2 );
  2248. }
  2249. if( !m_fSortAscending )
  2250. {
  2251. result = -result;
  2252. }
  2253. return( result );
  2254. }
  2255. //--------------------------------------------------------------------------//
  2256. // CFindSomeone::LParamToPos. //
  2257. //--------------------------------------------------------------------------//
  2258. int CFindSomeone::LParamToPos(LPARAM lParam)
  2259. {
  2260. LV_FINDINFO lvF;
  2261. lvF.flags = LVFI_PARAM;
  2262. lvF.lParam = lParam;
  2263. return( ListView_FindItem( m_ilsListView, -1, &lvF ) ); // Note: retuns -1 on failure...
  2264. }
  2265. //--------------------------------------------------------------------------//
  2266. // CFindSomeone::StoreColumnInfo. //
  2267. //--------------------------------------------------------------------------//
  2268. void
  2269. CFindSomeone::StoreColumnInfo(void)
  2270. {
  2271. if( IsWindow( m_ilsListView ) )
  2272. {
  2273. RegEntry re( UI_KEY, HKEY_CURRENT_USER );
  2274. DWORD dwStyle = ::GetWindowLong( m_ilsListView, GWL_STYLE );
  2275. if( (dwStyle & LVS_TYPEMASK) == LVS_REPORT )
  2276. {
  2277. // get width of each column
  2278. for( int i = 0; i < g_rgDLColumnInfo[ DLT_ULS ].nColumns; i++ )
  2279. {
  2280. m_alColumns[ i ] = ListView_GetColumnWidth( m_ilsListView, i );
  2281. }
  2282. }
  2283. // save this back to registry
  2284. re.SetValue( g_rgDLColumnInfo[ DLT_ULS ].pszRVWidths,
  2285. (LPVOID) m_alColumns,
  2286. sizeof( LONG ) * g_rgDLColumnInfo[ DLT_ULS ].nColumns );
  2287. int anColumnOrder[MAX_DIR_COLUMNS];
  2288. if( ListView_GetColumnOrderArray( m_ilsListView,
  2289. g_rgDLColumnInfo[ DLT_ULS ].nColumns,
  2290. anColumnOrder ) )
  2291. {
  2292. // save the column order to registry
  2293. re.SetValue( g_rgDLColumnInfo[ DLT_ULS ].pszRVOrder,
  2294. (LPVOID) anColumnOrder,
  2295. sizeof( LONG ) * g_rgDLColumnInfo[ DLT_ULS ].nColumns );
  2296. }
  2297. // save the sort column and direction
  2298. re.SetValue(g_rgDLColumnInfo[ DLT_ULS ].pszRVSortColumn, m_iSortColumn);
  2299. re.SetValue(g_rgDLColumnInfo[ DLT_ULS ].pszRVSortAscending, m_fSortAscending);
  2300. }
  2301. } // End of CFindSomeone::StoreColumnInfo.
  2302. //--------------------------------------------------------------------------//
  2303. // CFindSomeone::onCall. //
  2304. //--------------------------------------------------------------------------//
  2305. void
  2306. CFindSomeone::onCall(void)
  2307. {
  2308. if( m_pRai != NULL )
  2309. {
  2310. const bool secure = (IsDlgButtonChecked( GetWindow(), IDS_SECURITY_CHECKBOX ) == BST_CHECKED);
  2311. const NM_ADDR_TYPE nmType = static_cast<NM_ADDR_TYPE>(m_pRai->rgDwStr[ 0 ].dw);
  2312. g_pCCallto->Callto( m_pRai->rgDwStr[ 0 ].psz, // pointer to the callto url to try to place the call with...
  2313. m_pRai->szName, // pointer to the display name to use...
  2314. nmType, // callto type to resolve this callto as...
  2315. true, // the pszCallto parameter is to be interpreted as a pre-unescaped addressing component vs a full callto...
  2316. &secure, // security preference, NULL for none. must be "compatible" with secure param if present...
  2317. false, // whether or not save in mru...
  2318. true, // whether or not to perform user interaction on errors...
  2319. GetWindow(), // if bUIEnabled is true this is the window to parent error/status windows to...
  2320. NULL ); // out pointer to INmCall * to receive INmCall * generated by placing call...
  2321. }
  2322. } // End of CFindSomeone::onCall.
  2323. //--------------------------------------------------------------------------//
  2324. // CFindSomeone::OnDeleteIlsServer. //
  2325. //--------------------------------------------------------------------------//
  2326. void CFindSomeone::OnDeleteIlsServer(void)
  2327. {
  2328. if( m_pMruServer != NULL )
  2329. {
  2330. int iSelectionIndex = ::SendMessage( m_hwndCombo, CB_GETCURSEL, 0, 0 );
  2331. if( iSelectionIndex != CB_ERR )
  2332. {
  2333. int iLength = ::SendMessage( m_hwndCombo, CB_GETLBTEXTLEN, iSelectionIndex, 0 );
  2334. TCHAR * pszIls = new TCHAR [ iLength + 1 ];
  2335. if( pszIls != NULL )
  2336. {
  2337. if( ::SendMessage( m_hwndCombo, CB_GETLBTEXT, iSelectionIndex, (LPARAM) pszIls ) != CB_ERR )
  2338. {
  2339. HWND hwndDialog = s_pDlgCall->GetWindow();
  2340. // First make sure this directory server isn't their default server....
  2341. if( lstrcmpi( pszIls, CDirectoryManager::get_defaultServer() ) == 0 )
  2342. {
  2343. ::MessageBox( hwndDialog, RES2T( IDS_DLGCALL_CANT_DELETE_DEFAULT_ILS ), RES2T( IDS_MSGBOX_TITLE ), MB_SETFOREGROUND | MB_OK | MB_ICONEXCLAMATION );
  2344. }
  2345. else
  2346. {
  2347. // Next make them confirm they want to do this...
  2348. if( ::MessageBox( hwndDialog, RES2T( IDS_DLGCALL_CONFIRM_DELETE_ILS ), RES2T( IDS_MSGBOX_TITLE ), MB_SETFOREGROUND | MB_YESNO | MB_ICONQUESTION | MB_DEFBUTTON2 ) == IDYES )
  2349. {
  2350. if( m_pMruServer->DeleteEntry( pszIls ) )
  2351. {
  2352. m_pMruServer->Save();
  2353. int iCount = ::SendMessage( m_hwndCombo, CB_DELETESTRING, iSelectionIndex, 0 );
  2354. if( iCount <= iSelectionIndex )
  2355. {
  2356. iSelectionIndex--;
  2357. }
  2358. ::SendMessage( m_hwndCombo, CB_SETCURSEL, iSelectionIndex, 0 );
  2359. ShowList( iSelectionIndex );
  2360. }
  2361. }
  2362. }
  2363. }
  2364. delete [] pszIls;
  2365. }
  2366. }
  2367. }
  2368. } // End of CFindSomeone::OnDeleteIlsServer.
  2369. void CFindSomeone::OnCallStarted()
  2370. {
  2371. HWND hwnd = GetWindow();
  2372. m_secure = ::IsDlgButtonChecked( hwnd, IDS_SECURITY_CHECKBOX ) != 0;
  2373. UpdateSecurityCheck(m_pConfRoom, hwnd, IDS_SECURITY_CHECKBOX);
  2374. // BUGBUG georgep: We get notified before the actual conference state
  2375. // changes, so we need to disable manually
  2376. ::EnableWindow( GetDlgItem(hwnd, IDS_SECURITY_CHECKBOX), FALSE );
  2377. }
  2378. void CFindSomeone::OnCallEnded()
  2379. {
  2380. HWND hwnd = GetWindow();
  2381. UpdateSecurityCheck(m_pConfRoom, hwnd, IDS_SECURITY_CHECKBOX);
  2382. ::CheckDlgButton( hwnd, IDS_SECURITY_CHECKBOX, m_secure );
  2383. }
  2384. //--------------------------------------------------------------------------//
  2385. // StringCompare. //
  2386. //--------------------------------------------------------------------------//
  2387. int StringCompare( const TCHAR * const psz1, const TCHAR * const psz2 )
  2388. {
  2389. ASSERT( psz1 != NULL );
  2390. ASSERT( psz2 != NULL );
  2391. int iResult = CompareString( LOCALE_USER_DEFAULT, NORM_IGNORECASE | SORT_STRINGSORT, psz1, -1, psz2, -1 );
  2392. ASSERT( iResult != 0);
  2393. if( iResult == CSTR_LESS_THAN )
  2394. {
  2395. iResult = -1;
  2396. }
  2397. else if( iResult == CSTR_GREATER_THAN )
  2398. {
  2399. iResult = 1;
  2400. }
  2401. else
  2402. {
  2403. iResult = 0;
  2404. }
  2405. return( iResult );
  2406. } // End of StringCompare.
  2407. //--------------------------------------------------------------------------//
  2408. // GetDefaultRect. //
  2409. //--------------------------------------------------------------------------//
  2410. void GetDefaultRect(const HWND hwndParent, RECT & rcRect, const int iDefaultWidth, const int iDefaultHeight)
  2411. {
  2412. RECT rcWorkArea;
  2413. bool bResizeAndCenter = true;
  2414. if( !SystemParametersInfo( SPI_GETWORKAREA, 0, &rcWorkArea, 0 ) )
  2415. {
  2416. SetRect( &rcWorkArea, 0, 0, GetSystemMetrics( SM_CXSCREEN ) - 1, GetSystemMetrics( SM_CYSCREEN ) - 1 );
  2417. }
  2418. if( IsWindow( hwndParent ) )
  2419. {
  2420. GetWindowRect( hwndParent, &rcRect );
  2421. rcRect.left += DX_BORDER;
  2422. rcRect.top += DY_BORDER + GetSystemMetrics( SM_CYCAPTION );
  2423. rcRect.right = rcRect.left + iDefaultWidth;
  2424. rcRect.bottom = rcRect.top + iDefaultHeight;
  2425. if( (rcRect.left >= rcWorkArea.left) && (rcRect.top >= rcWorkArea.top) &&
  2426. (rcRect.right <= rcWorkArea.right) && (rcRect.bottom <= rcWorkArea.bottom) )
  2427. {
  2428. bResizeAndCenter = false;
  2429. }
  2430. }
  2431. if( bResizeAndCenter )
  2432. {
  2433. int iWidth = rcWorkArea.right - rcWorkArea.left;
  2434. if( iWidth > iDefaultWidth )
  2435. {
  2436. iWidth -= iDefaultWidth;
  2437. iWidth /= 2;
  2438. rcWorkArea.left += iWidth;
  2439. rcWorkArea.right = rcWorkArea.left + iDefaultWidth;
  2440. }
  2441. else
  2442. {
  2443. iWidth = (rcWorkArea.right - rcWorkArea.left) / 10;
  2444. rcWorkArea.left += iWidth;
  2445. rcWorkArea.right -= iWidth;
  2446. }
  2447. int iHeight = rcWorkArea.bottom - rcWorkArea.top;
  2448. if( iHeight > iDefaultHeight )
  2449. {
  2450. iHeight -= iDefaultHeight;
  2451. iHeight /= 2;
  2452. rcWorkArea.top += iHeight;
  2453. rcWorkArea.bottom = rcWorkArea.top + iDefaultHeight;
  2454. }
  2455. else
  2456. {
  2457. iHeight = (rcWorkArea.bottom - rcWorkArea.top) / 10;
  2458. rcWorkArea.top += iHeight;
  2459. rcWorkArea.bottom -= iHeight;
  2460. }
  2461. rcRect = rcWorkArea;
  2462. }
  2463. } // End of GetDefaultRect.
  2464. //--------------------------------------------------------------------------//
  2465. // GetPixelsPerChar. //
  2466. //--------------------------------------------------------------------------//
  2467. int GetPixelsPerChar(const HWND hwnd)
  2468. {
  2469. int iPixels = 10;
  2470. if( IsWindow( hwnd ) )
  2471. {
  2472. HDC hDC;
  2473. if( (hDC = GetDC( hwnd )) != NULL )
  2474. {
  2475. HFONT hFont;
  2476. if( (hFont = (HFONT) SendMessage( hwnd, WM_GETFONT, 0, 0 )) != NULL )
  2477. {
  2478. hFont = (HFONT) SelectObject( hDC, hFont );
  2479. TEXTMETRIC tmMetrics;
  2480. GetTextMetrics( hDC, &tmMetrics );
  2481. iPixels = tmMetrics.tmAveCharWidth;
  2482. SelectObject( hDC, hFont );
  2483. }
  2484. ReleaseDC( hwnd, hDC );
  2485. }
  2486. }
  2487. return( iPixels );
  2488. } // End of GetPixelsPerChar.