HyperLink.cpp
上传用户:lj3531212
上传日期:2007-06-18
资源大小:346k
文件大小:13k
源码类别:

绘图程序

开发平台:

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, 1998 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 <afxconv.h>           // For LPTSTR -> LPSTR macros
  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, 92, 0);   //::GetSysColor(COLOR_3DHILIGHT);
  35.     m_bOverControl      = FALSE;                // Cursor not yet over control
  36.     m_bVisited          = FALSE;                // Hasn't been visited yet.
  37.     m_bUnderline        = TRUE;                 // Underline the link?
  38.     m_bAdjustToFit      = TRUE;                 // Resize the window to fit the text?
  39.     m_strURL.Empty();
  40. }
  41. CHyperLink::~CHyperLink()
  42. {
  43.     m_Font.DeleteObject();
  44. }
  45. BEGIN_MESSAGE_MAP(CHyperLink, CStatic)
  46.     //{{AFX_MSG_MAP(CHyperLink)
  47.     ON_CONTROL_REFLECT(STN_CLICKED, OnClicked)
  48.     ON_WM_CTLCOLOR_REFLECT()
  49.     ON_WM_SETCURSOR()
  50.     ON_WM_MOUSEMOVE()
  51.     //}}AFX_MSG_MAP
  52. END_MESSAGE_MAP()
  53. /////////////////////////////////////////////////////////////////////////////
  54. // CHyperLink message handlers
  55. BOOL CHyperLink::PreTranslateMessage(MSG* pMsg) 
  56. {
  57.     m_ToolTip.RelayEvent(pMsg);
  58.     return CStatic::PreTranslateMessage(pMsg);
  59. }
  60. void CHyperLink::OnClicked()
  61. {
  62.     m_bOverControl = FALSE;
  63.     int result = (int)GotoURL(m_strURL, SW_SHOW);
  64.     m_bVisited = (result > HINSTANCE_ERROR);
  65.     if (!m_bVisited) {
  66.         MessageBeep(MB_ICONEXCLAMATION);     // Unable to follow link
  67.         ReportError(result);
  68.     } else 
  69.         SetVisited();                        // Repaint to show visited colour
  70. }
  71. HBRUSH CHyperLink::CtlColor(CDC* pDC, UINT nCtlColor) 
  72. {
  73.     ASSERT(nCtlColor == CTLCOLOR_STATIC);
  74.     UNUSED(nCtlColor);
  75.     if (m_bOverControl)
  76.         pDC->SetTextColor(m_crHoverColour);
  77.     else if (m_bVisited)
  78.         pDC->SetTextColor(m_crVisitedColour);
  79.     else
  80.         pDC->SetTextColor(m_crLinkColour);
  81.     // transparent text.
  82.     pDC->SetBkMode(TRANSPARENT);
  83.     return (HBRUSH)GetStockObject(NULL_BRUSH);
  84. }
  85. void CHyperLink::OnMouseMove(UINT nFlags, CPoint point) 
  86. {
  87.     CStatic::OnMouseMove(nFlags, point);
  88.     if (m_bOverControl)        // Cursor is currently over control
  89.     {
  90.         CRect rect;
  91.         GetClientRect(rect);
  92.         if (!rect.PtInRect(point))
  93.         {
  94.             m_bOverControl = FALSE;
  95.             ReleaseCapture();
  96.             RedrawWindow();
  97.             return;
  98.         }
  99.     }
  100.     else                      // Cursor has just moved over control
  101.     {
  102.         m_bOverControl = TRUE;
  103.         RedrawWindow();
  104.         SetCapture();
  105.     }
  106. }
  107. BOOL CHyperLink::OnSetCursor(CWnd* /*pWnd*/, UINT /*nHitTest*/, UINT /*message*/) 
  108. {
  109.     if (m_hLinkCursor)
  110.     {
  111.         ::SetCursor(m_hLinkCursor);
  112.         return TRUE;
  113.     }
  114.     return FALSE;
  115. }
  116. void CHyperLink::PreSubclassWindow() 
  117. {
  118.     // We want to get mouse clicks via STN_CLICKED
  119.     DWORD dwStyle = GetStyle();
  120.     ::SetWindowLong(GetSafeHwnd(), GWL_STYLE, dwStyle | SS_NOTIFY);
  121.     
  122.     // Set the URL as the window text
  123.     if (m_strURL.IsEmpty())
  124.         GetWindowText(m_strURL);
  125.     // Check that the window text isn't empty. If it is, set it as the URL.
  126.     CString strWndText;
  127.     GetWindowText(strWndText);
  128.     if (strWndText.IsEmpty()) {
  129.         ASSERT(!m_strURL.IsEmpty());    // Window and URL both NULL. DUH!
  130.         SetWindowText(m_strURL);
  131.     }
  132.     // Create the font
  133.     LOGFONT lf;
  134.     GetFont()->GetLogFont(&lf);
  135.     lf.lfUnderline = (BYTE) m_bUnderline;
  136.     m_Font.CreateFontIndirect(&lf);
  137.     SetFont(&m_Font);
  138.     PositionWindow();        // Adjust size of window to fit URL if necessary
  139.     SetDefaultCursor();      // Try and load up a "hand" cursor
  140.     // Create the tooltip
  141.     CRect rect; 
  142.     GetClientRect(rect);
  143.     m_ToolTip.Create(this);
  144.     m_ToolTip.AddTool(this, m_strURL, rect, TOOLTIP_ID);
  145.     CStatic::PreSubclassWindow();
  146. }
  147. /////////////////////////////////////////////////////////////////////////////
  148. // CHyperLink operations
  149. void CHyperLink::SetURL(CString strURL)
  150. {
  151.     m_strURL = strURL;
  152.     if (::IsWindow(GetSafeHwnd())) {
  153.         //PositionWindow();
  154.         m_ToolTip.UpdateTipText(strURL, this, TOOLTIP_ID);
  155.     }
  156. }
  157. CString CHyperLink::GetURL() const
  158.     return m_strURL;   
  159. }
  160. void CHyperLink::SetColours(COLORREF crLinkColour, COLORREF crVisitedColour,
  161.                             COLORREF crHoverColour /* = -1 */) 
  162.     m_crLinkColour    = crLinkColour; 
  163.     m_crVisitedColour = crVisitedColour;
  164. if (crHoverColour == -1)
  165. m_crHoverColour = ::GetSysColor(COLOR_HIGHLIGHT);
  166. else
  167. m_crHoverColour = crHoverColour;
  168.     if (::IsWindow(m_hWnd))
  169.         Invalidate(); 
  170. }
  171. COLORREF CHyperLink::GetLinkColour() const
  172.     return m_crLinkColour; 
  173. }
  174. COLORREF CHyperLink::GetVisitedColour() const
  175. {
  176.     return m_crVisitedColour; 
  177. }
  178. COLORREF CHyperLink::GetHoverColour() const
  179. {
  180.     return m_crHoverColour;
  181. }
  182. void CHyperLink::SetVisited(BOOL bVisited /* = TRUE */) 
  183.     m_bVisited = bVisited; 
  184.     if (::IsWindow(GetSafeHwnd()))
  185.         Invalidate(); 
  186. }
  187. BOOL CHyperLink::GetVisited() const
  188.     return m_bVisited; 
  189. }
  190. void CHyperLink::SetLinkCursor(HCURSOR hCursor)
  191.     m_hLinkCursor = hCursor;
  192.     if (m_hLinkCursor == NULL)
  193.         SetDefaultCursor();
  194. }
  195. HCURSOR CHyperLink::GetLinkCursor() const
  196. {
  197.     return m_hLinkCursor;
  198. }
  199. void CHyperLink::SetUnderline(BOOL bUnderline /* = TRUE */)
  200. {
  201.     m_bUnderline = bUnderline;
  202.     if (::IsWindow(GetSafeHwnd()))
  203.     {
  204.         LOGFONT lf;
  205.         GetFont()->GetLogFont(&lf);
  206.         lf.lfUnderline = (BYTE) m_bUnderline;
  207.         m_Font.DeleteObject();
  208.         m_Font.CreateFontIndirect(&lf);
  209.         SetFont(&m_Font);
  210.         Invalidate(); 
  211.     }
  212. }
  213. BOOL CHyperLink::GetUnderline() const
  214.     return m_bUnderline; 
  215. }
  216. void CHyperLink::SetAutoSize(BOOL bAutoSize /* = TRUE */)
  217. {
  218.     m_bAdjustToFit = bAutoSize;
  219.     if (::IsWindow(GetSafeHwnd()))
  220.         PositionWindow();
  221. }
  222. BOOL CHyperLink::GetAutoSize() const
  223.     return m_bAdjustToFit; 
  224. }
  225. // Move and resize the window so that the window is the same size
  226. // as the hyperlink text. This stops the hyperlink cursor being active
  227. // when it is not directly over the text. If the text is left justified
  228. // then the window is merely shrunk, but if it is centred or right
  229. // justified then the window will have to be moved as well.
  230. //
  231. // Suggested by P錶 K. T鴑der 
  232. void CHyperLink::PositionWindow()
  233. {
  234.     if (!::IsWindow(GetSafeHwnd()) || !m_bAdjustToFit) 
  235.         return;
  236.     // Get the current window position
  237.     CRect rect;
  238.     GetWindowRect(rect);
  239.     CWnd* pParent = GetParent();
  240.     if (pParent)
  241.         pParent->ScreenToClient(rect);
  242.     // Get the size of the window text
  243.     CString strWndText;
  244.     GetWindowText(strWndText);
  245.     CDC* pDC = GetDC();
  246.     CFont* pOldFont = pDC->SelectObject(&m_Font);
  247.     CSize Extent = pDC->GetTextExtent(strWndText);
  248.     pDC->SelectObject(pOldFont);
  249.     ReleaseDC(pDC);
  250.     // Get the text justification via the window style
  251.     DWORD dwStyle = GetStyle();
  252.     // Recalc the window size and position based on the text justification
  253.     if (dwStyle & SS_CENTERIMAGE)
  254.         rect.DeflateRect(0, (rect.Height() - Extent.cy)/2);
  255.     else
  256.         rect.bottom = rect.top + Extent.cy;
  257.     if (dwStyle & SS_CENTER)   
  258.         rect.DeflateRect((rect.Width() - Extent.cx)/2, 0);
  259.     else if (dwStyle & SS_RIGHT) 
  260.         rect.left  = rect.right - Extent.cx;
  261.     else // SS_LEFT = 0, so we can't test for it explicitly 
  262.         rect.right = rect.left + Extent.cx;
  263.     // Move the window
  264.     SetWindowPos(NULL, rect.left, rect.top, rect.Width(), rect.Height(), SWP_NOZORDER);
  265. }
  266. /////////////////////////////////////////////////////////////////////////////
  267. // CHyperLink implementation
  268. // The following appeared in Paul DiLascia's Jan 1998 MSJ articles.
  269. // It loads a "hand" cursor from the winhlp32.exe module
  270. void CHyperLink::SetDefaultCursor()
  271. {
  272.     if (m_hLinkCursor == NULL)                // No cursor handle - load our own
  273.     {
  274.         // Get the windows directory
  275.         CString strWndDir;
  276.         GetWindowsDirectory(strWndDir.GetBuffer(MAX_PATH), MAX_PATH);
  277.         strWndDir.ReleaseBuffer();
  278.         strWndDir += _T("\winhlp32.exe");
  279.         // This retrieves cursor #106 from winhlp32.exe, which is a hand pointer
  280.         HMODULE hModule = LoadLibrary(strWndDir);
  281.         if (hModule) {
  282.             HCURSOR hHandCursor = ::LoadCursor(hModule, MAKEINTRESOURCE(106));
  283.             if (hHandCursor)
  284.                 m_hLinkCursor = CopyCursor(hHandCursor);
  285.         }
  286.         FreeLibrary(hModule);
  287.     }
  288. }
  289. LONG CHyperLink::GetRegKey(HKEY key, LPCTSTR subkey, LPTSTR retdata)
  290. {
  291.     HKEY hkey;
  292.     LONG retval = RegOpenKeyEx(key, subkey, 0, KEY_QUERY_VALUE, &hkey);
  293.     if (retval == ERROR_SUCCESS) {
  294.         long datasize = MAX_PATH;
  295.         TCHAR data[MAX_PATH];
  296.         RegQueryValue(hkey, NULL, data, &datasize);
  297.         lstrcpy(retdata,data);
  298.         RegCloseKey(hkey);
  299.     }
  300.     return retval;
  301. }
  302. void CHyperLink::ReportError(int nError)
  303. {
  304.     CString str;
  305.     switch (nError) 
  306.     {
  307.         case 0:                       str = _T("The operating system is outnof memory or resources."); break;
  308.         case SE_ERR_PNF:              str = _T("The specified path was not found."); break;
  309.         case SE_ERR_FNF:              str = _T("The specified file was not found."); break;
  310.         case ERROR_BAD_FORMAT:        str = _T("The .EXE file is invalidn(non-Win32 .EXE or error in .EXE image)."); break;
  311.         case SE_ERR_ACCESSDENIED:     str = _T("The operating system deniednaccess to the specified file."); break;
  312.         case SE_ERR_ASSOCINCOMPLETE:  str = _T("The filename association isnincomplete or invalid."); break;
  313.         case SE_ERR_DDEBUSY:          str = _T("The DDE transaction could notnbe completed because other DDE transactionsnwere being processed."); break;
  314.         case SE_ERR_DDEFAIL:          str = _T("The DDE transaction failed."); break;
  315.         case SE_ERR_DDETIMEOUT:       str = _T("The DDE transaction could notnbe completed because the request timed out."); break;
  316.         case SE_ERR_DLLNOTFOUND:      str = _T("The specified dynamic-link library was not found."); break;
  317.         case SE_ERR_NOASSOC:          str = _T("There is no application associatednwith the given filename extension."); break;
  318.         case SE_ERR_OOM:              str = _T("There was not enough memory to complete the operation."); break;
  319.         case SE_ERR_SHARE:            str = _T("A sharing violation occurred. ");
  320.         default:                      str.Format(_T("Unknown Error (%d) occurred."), nError); break;
  321.     }
  322.     str = _T("Unable to open hyperlink:nn") + str;
  323.     AfxMessageBox(str, MB_ICONEXCLAMATION | MB_OK);
  324. }
  325. HINSTANCE CHyperLink::GotoURL(LPCTSTR url, int showcmd)
  326. {
  327.     TCHAR key[MAX_PATH + MAX_PATH];
  328.     // First try ShellExecute()
  329.     HINSTANCE result = ShellExecute(NULL, _T("open"), url, NULL,NULL, showcmd);
  330.     // If it failed, get the .htm regkey and lookup the program
  331.     if ((UINT)result <= HINSTANCE_ERROR) {
  332.         if (GetRegKey(HKEY_CLASSES_ROOT, _T(".htm"), key) == ERROR_SUCCESS) {
  333.             lstrcat(key, _T("\shell\open\command"));
  334.             if (GetRegKey(HKEY_CLASSES_ROOT,key,key) == ERROR_SUCCESS) {
  335.                 TCHAR *pos;
  336.                 pos = _tcsstr(key, _T(""%1""));
  337.                 if (pos == NULL) {                     // No quotes found
  338.                     pos = _tcsstr(key, _T("%1"));       // Check for %1, without quotes 
  339.                     if (pos == NULL)                   // No parameter at all...
  340.                         pos = key+lstrlen(key)-1;
  341.                     else
  342.                         *pos = '';                   // Remove the parameter
  343.                 }
  344.                 else
  345.                     *pos = '';                       // Remove the parameter
  346.                 lstrcat(pos, _T(" "));
  347.                 lstrcat(pos, url);
  348. #ifdef _UNICODE
  349.                 USES_CONVERSION;
  350.                 result = (HINSTANCE) WinExec(T2A(key),showcmd);
  351. #else
  352.                 result = (HINSTANCE) WinExec(key,showcmd);
  353. #endif
  354.             }
  355.         }
  356.     }
  357.     return result;
  358. }