HyperLink.cpp
上传用户:xiuanze55
上传日期:2013-06-16
资源大小:85k
文件大小:12k
源码类别:

其他数据库

开发平台:

Visual C++

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