HyperLink.cpp
上传用户:dzyhzl
上传日期:2019-04-29
资源大小:56270k
文件大小:14k
源码类别:

模拟服务器

开发平台:

C/C++

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