HyperLink.cpp
上传用户:lds876
上传日期:2013-05-25
资源大小:567k
文件大小:14k
源码类别:

P2P编程

开发平台:

Visual C++

  1. // HyperLink.cpp : implementation file
  2. //
  3. // HyperLink static control. Will open the default browser with the given URL
  4. // when the user clicks on the link.
  5. //
  6. // Copyright (C) 1997 - 1999 Chris Maunder
  7. // All rights reserved. May not be sold for profit.
  8. //
  9. // Thanks to P錶 K. T鴑der for auto-size and window caption changes.
  10. //
  11. // "GotoURL" function by Stuart Patterson
  12. // As seen in the August, 1997 Windows Developer's Journal.
  13. // Copyright 1997 by Miller Freeman, Inc. All rights reserved.
  14. // Modified by Chris Maunder to use TCHARs instead of chars.
  15. //
  16. // "Default hand cursor" from Paul DiLascia's Jan 1998 MSJ article.
  17. //
  18. #include "stdafx.h"
  19. #include "HyperLink.h"
  20. #include "testBT.h"
  21. #ifdef _DEBUG
  22. #define new DEBUG_NEW
  23. #undef THIS_FILE
  24. static char THIS_FILE[] = __FILE__;
  25. #endif
  26. #define TOOLTIP_ID 1
  27. /////////////////////////////////////////////////////////////////////////////
  28. // CHyperLink
  29. CHyperLink::CHyperLink()
  30. {
  31.     m_hLinkCursor       = NULL;                 // No cursor as yet
  32.     m_crLinkColour      = RGB(  0,   0, 238);   // Blue
  33.     m_crVisitedColour   = RGB( 85,  26, 139);   // Purple
  34.     m_crHoverColour     = RGB(255,   0,   0);   // Red
  35.     m_bOverControl      = FALSE;                // Cursor not yet over control
  36.     m_bVisited          = FALSE;                // Hasn't been visited yet.
  37.     m_nUnderline        = ulHover;              // Underline the link?
  38.     m_bAdjustToFit      = TRUE;                 // Resize the window to fit the text?
  39.     m_strURL.Empty();
  40.     // m_pStdFont          = NULL;
  41.     m_nTimerID          = 100;
  42. }
  43. CHyperLink::~CHyperLink()
  44. {
  45.     m_UnderlineFont.DeleteObject();
  46. m_StdFont.DeleteObject();
  47. if (m_hLinkCursor)
  48. {
  49. BOOL bRet = DestroyCursor(m_hLinkCursor);
  50. ASSERT(bRet);
  51. }
  52. }
  53. /////////////////////////////////////////////////////////////////////////////
  54. // CHyperLink overrides
  55. BOOL CHyperLink::DestroyWindow() 
  56. {
  57.     KillTimer(m_nTimerID);
  58. return CStatic::DestroyWindow();
  59. }
  60. BOOL CHyperLink::PreTranslateMessage(MSG* pMsg) 
  61. {
  62.     m_ToolTip.RelayEvent(pMsg);
  63.     return CStatic::PreTranslateMessage(pMsg);
  64. }
  65. void CHyperLink::PreSubclassWindow() 
  66. {
  67.     // We want to get mouse clicks via STN_CLICKED
  68.     DWORD dwStyle = GetStyle();
  69.     ::SetWindowLong(GetSafeHwnd(), GWL_STYLE, dwStyle | SS_NOTIFY);
  70.     
  71.     // Set the URL as the window text
  72.     if (m_strURL.IsEmpty())
  73.         GetWindowText(m_strURL);
  74.     // Check that the window text isn't empty. If it is, set it as the URL.
  75.     CString strWndText;
  76.     GetWindowText(strWndText);
  77.     if (strWndText.IsEmpty()) 
  78.     {
  79.         ASSERT(!m_strURL.IsEmpty());    // Window and URL both NULL. DUH!
  80.         SetWindowText(m_strURL);
  81.     }
  82.     // m_pStdFont = GetFont();
  83. CFont* pTempFont = GetFont();
  84.     ASSERT(pTempFont);
  85.     // Create the underline font
  86.     LOGFONT lf;
  87.     pTempFont->GetLogFont(&lf);
  88. m_StdFont.CreateFontIndirect(&lf);
  89.     lf.lfUnderline = (BYTE) TRUE;
  90.     m_UnderlineFont.CreateFontIndirect(&lf);
  91.     PositionWindow();        // Adjust size of window to fit URL if necessary
  92.     SetDefaultCursor();      // Try and load up a "hand" cursor
  93.     SetUnderline();
  94.     // Create the tooltip
  95.     CRect rect; 
  96.     GetClientRect(rect);
  97.     m_ToolTip.Create(this);
  98.     m_ToolTip.AddTool(this, m_strURL, rect, TOOLTIP_ID);
  99.     CStatic::PreSubclassWindow();
  100. }
  101. BEGIN_MESSAGE_MAP(CHyperLink, CStatic)
  102.     //{{AFX_MSG_MAP(CHyperLink)
  103.     ON_WM_CTLCOLOR_REFLECT()
  104.     ON_WM_SETCURSOR()
  105.     ON_WM_MOUSEMOVE()
  106. ON_WM_TIMER()
  107.     ON_CONTROL_REFLECT(STN_CLICKED, OnClicked)
  108. ON_WM_ERASEBKGND()
  109. //}}AFX_MSG_MAP
  110. END_MESSAGE_MAP()
  111. /////////////////////////////////////////////////////////////////////////////
  112. // CHyperLink message handlers
  113. void CHyperLink::OnClicked()
  114. {
  115.     m_bOverControl = FALSE;
  116.     int result = (int)GotoURL(m_strURL, SW_SHOW);
  117.     m_bVisited = (result > HINSTANCE_ERROR);
  118.     if (!m_bVisited)
  119.     {
  120.         MessageBeep(MB_ICONEXCLAMATION);     // Unable to follow link
  121.         ReportError(result);
  122.     }
  123.     else 
  124.         SetVisited();                        // Repaint to show visited colour
  125. }
  126. HBRUSH CHyperLink::CtlColor(CDC* pDC, UINT nCtlColor) 
  127. {
  128. nCtlColor;
  129.     ASSERT(nCtlColor == CTLCOLOR_STATIC);
  130.     if (m_bOverControl)
  131.         pDC->SetTextColor(m_crHoverColour);
  132.     else if (m_bVisited)
  133.         pDC->SetTextColor(m_crVisitedColour);
  134.     else
  135.         pDC->SetTextColor(m_crLinkColour);
  136.     // transparent text.
  137.     pDC->SetBkMode(TRANSPARENT);
  138.     return (HBRUSH)GetStockObject(NULL_BRUSH);
  139. }
  140. void CHyperLink::OnMouseMove(UINT nFlags, CPoint point) 
  141. {
  142.     if (!m_bOverControl)        // Cursor has just moved over control
  143.     {
  144.         m_bOverControl = TRUE;
  145.         if (m_nUnderline == ulHover)
  146.             SetFont(&m_UnderlineFont);
  147.         Invalidate();
  148.         SetTimer(m_nTimerID, 100, NULL);
  149.     }
  150.     CStatic::OnMouseMove(nFlags, point);
  151. }
  152. void CHyperLink::OnTimer(UINT nIDEvent) 
  153. {
  154.     CPoint p(GetMessagePos());
  155.     ScreenToClient(&p);
  156.     CRect rect;
  157.     GetClientRect(rect);
  158.     if (!rect.PtInRect(p))
  159.     {
  160.         m_bOverControl = FALSE;
  161.         KillTimer(m_nTimerID);
  162.         if (m_nUnderline != ulAlways)
  163. {
  164. ASSERT((HFONT)m_StdFont);
  165.             SetFont(&m_StdFont);
  166. }
  167.         rect.bottom+=10;
  168.         InvalidateRect(rect);
  169.     }
  170.     
  171. CStatic::OnTimer(nIDEvent);
  172. }
  173. BOOL CHyperLink::OnSetCursor(CWnd* /*pWnd*/, UINT /*nHitTest*/, UINT /*message*/) 
  174. {
  175.     if (m_hLinkCursor)
  176.     {
  177.         ::SetCursor(m_hLinkCursor);
  178.         return TRUE;
  179.     }
  180.     return FALSE;
  181. }
  182. BOOL CHyperLink::OnEraseBkgnd(CDC* pDC) 
  183. {
  184.     CRect rect;
  185.     GetClientRect(rect);
  186.     pDC->FillSolidRect(rect, ::GetSysColor(COLOR_3DFACE));
  187.     return TRUE;
  188. }
  189. /////////////////////////////////////////////////////////////////////////////
  190. // CHyperLink operations
  191. void CHyperLink::SetURL(CString strURL)
  192. {
  193.     m_strURL = strURL;
  194.     if (::IsWindow(GetSafeHwnd())) {
  195.         PositionWindow();
  196.         m_ToolTip.UpdateTipText(strURL, this, TOOLTIP_ID);
  197.     }
  198. }
  199. CString CHyperLink::GetURL() const
  200.     return m_strURL;   
  201. }
  202. void CHyperLink::SetColours(COLORREF crLinkColour, COLORREF crVisitedColour,
  203.                             COLORREF crHoverColour /* = -1 */) 
  204.     m_crLinkColour    = crLinkColour; 
  205.     m_crVisitedColour = crVisitedColour;
  206. if (crHoverColour == -1)
  207. m_crHoverColour = ::GetSysColor(COLOR_HIGHLIGHT);
  208. else
  209. m_crHoverColour = crHoverColour;
  210.     if (::IsWindow(m_hWnd))
  211.         Invalidate(); 
  212. }
  213. COLORREF CHyperLink::GetLinkColour() const
  214.     return m_crLinkColour; 
  215. }
  216. COLORREF CHyperLink::GetVisitedColour() const
  217. {
  218.     return m_crVisitedColour; 
  219. }
  220. COLORREF CHyperLink::GetHoverColour() const
  221. {
  222.     return m_crHoverColour;
  223. }
  224. void CHyperLink::SetVisited(BOOL bVisited /* = TRUE */) 
  225.     m_bVisited = bVisited; 
  226.     if (::IsWindow(GetSafeHwnd()))
  227.         Invalidate(); 
  228. }
  229. BOOL CHyperLink::GetVisited() const
  230.     return m_bVisited; 
  231. }
  232. void CHyperLink::SetLinkCursor(HCURSOR hCursor)
  233. if (m_hLinkCursor)
  234. {
  235. BOOL bRet = DestroyCursor(m_hLinkCursor);
  236. ASSERT(bRet);
  237. }
  238.     m_hLinkCursor = hCursor;
  239.     if (m_hLinkCursor == NULL)
  240.         SetDefaultCursor();
  241. }
  242. HCURSOR CHyperLink::GetLinkCursor() const
  243. {
  244.     return m_hLinkCursor;
  245. }
  246. void CHyperLink::SetUnderline(int nUnderline /*=ulHover*/)
  247. {
  248.     if (m_nUnderline == nUnderline)
  249.         return;
  250.     if (::IsWindow(GetSafeHwnd()))
  251.     {
  252.         if (nUnderline == ulAlways)
  253.             SetFont(&m_UnderlineFont);
  254.         else
  255. {
  256. ASSERT((HFONT)m_StdFont);
  257.             SetFont(&m_StdFont);
  258. }
  259.         Invalidate(); 
  260.     }
  261.     m_nUnderline = nUnderline;
  262. }
  263. int CHyperLink::GetUnderline() const
  264.     return m_nUnderline; 
  265. }
  266. void CHyperLink::SetAutoSize(BOOL bAutoSize /* = TRUE */)
  267. {
  268.     m_bAdjustToFit = bAutoSize;
  269.     if (::IsWindow(GetSafeHwnd()))
  270.         PositionWindow();
  271. }
  272. BOOL CHyperLink::GetAutoSize() const
  273.     return m_bAdjustToFit; 
  274. }
  275. // Move and resize the window so that the window is the same size
  276. // as the hyperlink text. This stops the hyperlink cursor being active
  277. // when it is not directly over the text. If the text is left justified
  278. // then the window is merely shrunk, but if it is centred or right
  279. // justified then the window will have to be moved as well.
  280. //
  281. // Suggested by P錶 K. T鴑der 
  282. void CHyperLink::PositionWindow()
  283. {
  284.     if (!::IsWindow(GetSafeHwnd()) || !m_bAdjustToFit) 
  285.         return;
  286.     // Get the current window position
  287.     CRect WndRect, ClientRect;
  288.     GetWindowRect(WndRect);
  289.     GetClientRect(ClientRect);
  290.     ClientToScreen(ClientRect);
  291.     CWnd* pParent = GetParent();
  292.     if (pParent)
  293.     {
  294.         pParent->ScreenToClient(WndRect);
  295.         pParent->ScreenToClient(ClientRect);
  296.     }
  297.     // Get the size of the window text
  298.     CString strWndText;
  299.     GetWindowText(strWndText);
  300.     CDC* pDC = GetDC();
  301.     CFont* pOldFont = pDC->SelectObject(&m_UnderlineFont);
  302.     CSize Extent = pDC->GetTextExtent(strWndText);
  303.     pDC->SelectObject(pOldFont);
  304.     ReleaseDC(pDC);
  305.     // Adjust for window borders
  306.     Extent.cx += WndRect.Width() - ClientRect.Width(); 
  307.     Extent.cy += WndRect.Height() - ClientRect.Height(); 
  308.     // Get the text justification via the window style
  309.     DWORD dwStyle = GetStyle();
  310.     // Recalc the window size and position based on the text justification
  311.     if (dwStyle & SS_CENTERIMAGE)
  312.         WndRect.DeflateRect(0, (WndRect.Height() - Extent.cy)/2);
  313.     else
  314.         WndRect.bottom = WndRect.top + Extent.cy;
  315.     if (dwStyle & SS_CENTER)   
  316.         WndRect.DeflateRect((WndRect.Width() - Extent.cx)/2, 0);
  317.     else if (dwStyle & SS_RIGHT) 
  318.         WndRect.left  = WndRect.right - Extent.cx;
  319.     else // SS_LEFT = 0, so we can't test for it explicitly 
  320.         WndRect.right = WndRect.left + Extent.cx;
  321.     // Move the window
  322.     SetWindowPos(NULL, WndRect.left, WndRect.top, WndRect.Width(), WndRect.Height(), SWP_NOZORDER);
  323. }
  324. /////////////////////////////////////////////////////////////////////////////
  325. // CHyperLink implementation
  326. // The following appeared in Paul DiLascia's Jan 1998 MSJ articles.
  327. // It loads a "hand" cursor from the winhlp32.exe module
  328. void CHyperLink::SetDefaultCursor()
  329. {
  330.     if (m_hLinkCursor == NULL)                // No cursor handle - load our own
  331.     {
  332.         // Get the windows directory
  333.         CString strWndDir;
  334.         GetWindowsDirectory(strWndDir.GetBuffer(MAX_PATH), MAX_PATH);
  335.         strWndDir.ReleaseBuffer();
  336.         strWndDir += _T("\winhlp32.exe");
  337.         // This retrieves cursor #106 from winhlp32.exe, which is a hand pointer
  338.         HMODULE hModule = LoadLibrary(strWndDir);
  339.         if (hModule) {
  340.             HCURSOR hHandCursor = ::LoadCursor(hModule, MAKEINTRESOURCE(106));
  341.             if (hHandCursor)
  342. {
  343.                 m_hLinkCursor = CopyCursor(hHandCursor);
  344. BOOL bRet = DestroyCursor(hHandCursor);
  345. ASSERT(bRet);
  346. }
  347.         }
  348.         FreeLibrary(hModule);
  349.     }
  350. else
  351. {
  352. ASSERT(FALSE);
  353. }
  354. }
  355. LONG CHyperLink::GetRegKey(HKEY key, LPCTSTR subkey, LPTSTR retdata)
  356. {
  357.     HKEY hkey;
  358.     LONG retval = RegOpenKeyEx(key, subkey, 0, KEY_QUERY_VALUE, &hkey);
  359.     if (retval == ERROR_SUCCESS) {
  360.         long datasize = MAX_PATH;
  361.         TCHAR data[MAX_PATH];
  362.         RegQueryValue(hkey, NULL, data, &datasize);
  363.         lstrcpy(retdata,data);
  364.         RegCloseKey(hkey);
  365.     }
  366.     return retval;
  367. }
  368. void CHyperLink::ReportError(int nError)
  369. {
  370.     CString str;
  371.     switch (nError) {
  372.         case 0:                       str = "The operating system is outnof memory or resources."; break;
  373.         case SE_ERR_PNF:              str = "The specified path was not found."; break;
  374.         case SE_ERR_FNF:              str = "The specified file was not found."; break;
  375.         case ERROR_BAD_FORMAT:        str = "The .EXE file is invalidn(non-Win32 .EXE or error in .EXE image)."; break;
  376.         case SE_ERR_ACCESSDENIED:     str = "The operating system deniednaccess to the specified file."; break;
  377.         case SE_ERR_ASSOCINCOMPLETE:  str = "The filename association isnincomplete or invalid."; break;
  378.         case SE_ERR_DDEBUSY:          str = "The DDE transaction could notnbe completed because other DDE transactionsnwere being processed."; break;
  379.         case SE_ERR_DDEFAIL:          str = "The DDE transaction failed."; break;
  380.         case SE_ERR_DDETIMEOUT:       str = "The DDE transaction could notnbe completed because the request timed out."; break;
  381.         case SE_ERR_DLLNOTFOUND:      str = "The specified dynamic-link library was not found."; break;
  382.         case SE_ERR_NOASSOC:          str = "There is no application associatednwith the given filename extension."; break;
  383.         case SE_ERR_OOM:              str = "There was not enough memory to complete the operation."; break;
  384.         case SE_ERR_SHARE:            str = "A sharing violation occurred. ";
  385.         default:                      str.Format("Unknown Error (%d) occurred.", nError); break;
  386.     }
  387.     str = "Unable to open hyperlink:nn" + str;
  388.     AfxMessageBox(str, MB_ICONEXCLAMATION | MB_OK);
  389. }
  390. HINSTANCE CHyperLink::GotoURL(LPCTSTR url, int showcmd)
  391. {
  392.     TCHAR key[MAX_PATH + MAX_PATH];
  393.     // First try ShellExecute()
  394.     HINSTANCE result = ShellExecute(NULL, _T("open"), url, NULL,NULL, showcmd);
  395.     // If it failed, get the .htm regkey and lookup the program
  396.     if ((UINT)result <= HINSTANCE_ERROR) {
  397.         if (GetRegKey(HKEY_CLASSES_ROOT, _T(".htm"), key) == ERROR_SUCCESS) {
  398.             lstrcat(key, _T("\shell\open\command"));
  399.             if (GetRegKey(HKEY_CLASSES_ROOT,key,key) == ERROR_SUCCESS) {
  400.                 TCHAR *pos;
  401.                 pos = _tcsstr(key, _T(""%1""));
  402.                 if (pos == NULL) {                     // No quotes found
  403.                     pos = strstr(key, _T("%1"));       // Check for %1, without quotes 
  404.                     if (pos == NULL)                   // No parameter at all...
  405.                         pos = key+lstrlen(key)-1;
  406.                     else
  407.                         *pos = '';                   // Remove the parameter
  408.                 }
  409.                 else
  410.                     *pos = '';                       // Remove the parameter
  411.                 lstrcat(pos, _T(" "));
  412.                 lstrcat(pos, url);
  413.                 result = (HINSTANCE) WinExec(key,showcmd);
  414.             }
  415.         }
  416.     }
  417.     return result;
  418. }