567 lines
17 KiB
C++
567 lines
17 KiB
C++
// HyperLink.cpp : implementation file
|
||
//
|
||
// HyperLink static control.
|
||
//
|
||
// Copyright (C) 1997, 1998 Giancarlo Iovino (giancarlo@saria.com)
|
||
// All rights reserved. May not be sold for profit.
|
||
//
|
||
// This code is based on CHyperlink by Chris Maunder.
|
||
// "GotoURL" function by Stuart Patterson appeared in the Aug, 1997
|
||
// Windows Developer's Journal.
|
||
// "Default hand cursor" from Paul DiLascia's Jan 1998 MSJ article.
|
||
|
||
#include <stdafx.h>
|
||
#include "HyperLink.h"
|
||
|
||
#if defined(_DEBUG) && !defined(MMGR)
|
||
#define new DEBUG_NEW
|
||
#undef THIS_FILE
|
||
static char THIS_FILE[] = __FILE__;
|
||
#endif
|
||
|
||
#define TOOLTIP_ID 1
|
||
|
||
#define SETBITS(dw, bits) (dw |= bits)
|
||
#define CLEARBITS(dw, bits) (dw &= ~(bits))
|
||
#define BITSET(dw, bit) (((dw) & (bit)) != 0L)
|
||
|
||
/////////////////////////////////////////////////////////////////////////////
|
||
// CHyperLink
|
||
|
||
const DWORD CHyperLink::StyleUnderline = 0x00000001; // Underline bit
|
||
const DWORD CHyperLink::StyleUseHover = 0x00000002; // Hand over coloring bit
|
||
const DWORD CHyperLink::StyleAutoSize = 0x00000004; // Auto size bit
|
||
const DWORD CHyperLink::StyleDownClick = 0x00000008; // Down click mode bit
|
||
const DWORD CHyperLink::StyleGetFocusOnClick = 0x00000010; // Get focus on click bit
|
||
const DWORD CHyperLink::StyleNoHandCursor = 0x00000020; // No hand cursor bit
|
||
const DWORD CHyperLink::StyleNoActiveColor = 0x00000040; // No active color bit
|
||
|
||
COLORREF CHyperLink::g_crLinkColor = RGB(0, 0, 255); // Blue
|
||
COLORREF CHyperLink::g_crActiveColor = RGB(0, 128, 128); // Dark cyan
|
||
COLORREF CHyperLink::g_crVisitedColor = RGB(128, 0, 128); // Purple
|
||
COLORREF CHyperLink::g_crHoverColor = RGB(255, 0, 0 ); // Red
|
||
HCURSOR CHyperLink::g_hLinkCursor = NULL; // No cursor
|
||
|
||
CHyperLink::CHyperLink()
|
||
{
|
||
m_bOverControl = FALSE; // Cursor not yet over control
|
||
m_bVisited = FALSE; // Link has not been visited yet
|
||
m_bLinkActive = FALSE; // Control doesn't own the focus yet
|
||
m_strURL.Empty(); // Set URL to an empty string
|
||
// Set default styles
|
||
m_dwStyle = StyleUnderline|StyleAutoSize|StyleGetFocusOnClick;
|
||
}
|
||
|
||
CHyperLink::~CHyperLink()
|
||
{
|
||
m_Font.DeleteObject();
|
||
}
|
||
|
||
IMPLEMENT_DYNAMIC(CHyperLink, CStatic)
|
||
|
||
BEGIN_MESSAGE_MAP(CHyperLink, CStatic)
|
||
//{{AFX_MSG_MAP(CHyperLink)
|
||
ON_WM_CTLCOLOR_REFLECT()
|
||
ON_WM_SETCURSOR()
|
||
ON_WM_MOUSEMOVE()
|
||
ON_WM_LBUTTONUP()
|
||
ON_WM_SETFOCUS()
|
||
ON_WM_KILLFOCUS()
|
||
ON_WM_KEYDOWN()
|
||
ON_WM_NCHITTEST()
|
||
ON_WM_LBUTTONDOWN()
|
||
//}}AFX_MSG_MAP
|
||
END_MESSAGE_MAP()
|
||
|
||
/////////////////////////////////////////////////////////////////////////////
|
||
// CHyperLink message handlers
|
||
|
||
BOOL CHyperLink::PreTranslateMessage(MSG* pMsg)
|
||
{
|
||
m_ToolTip.RelayEvent(pMsg);
|
||
return CStatic::PreTranslateMessage(pMsg);
|
||
}
|
||
|
||
void CHyperLink::PreSubclassWindow()
|
||
{
|
||
// If the URL string is empty try to set it to the window text
|
||
if (m_strURL.IsEmpty())
|
||
GetWindowText(m_strURL);
|
||
|
||
// Check that the window text isn't empty.
|
||
// If it is, set it as URL string.
|
||
CString strWndText;
|
||
GetWindowText(strWndText);
|
||
if (strWndText.IsEmpty()) {
|
||
// Set the URL string as the window text
|
||
ASSERT(!m_strURL.IsEmpty()); // window text and URL both NULL!
|
||
CStatic::SetWindowText(m_strURL);
|
||
}
|
||
|
||
// Get the current window font
|
||
CFont* pFont = GetFont();
|
||
|
||
if (pFont != NULL) {
|
||
LOGFONT lf;
|
||
pFont->GetLogFont(&lf);
|
||
lf.lfUnderline = BITSET(m_dwStyle, StyleUnderline);
|
||
if (m_Font.CreateFontIndirect(&lf))
|
||
CStatic::SetFont(&m_Font);
|
||
// Adjust window size to fit URL if necessary
|
||
AdjustWindow();
|
||
}
|
||
else {
|
||
// if GetFont() returns NULL then probably the static
|
||
// control is not of a text type: it's better to set
|
||
// auto-resizing off
|
||
CLEARBITS(m_dwStyle,StyleAutoSize);
|
||
}
|
||
|
||
if (!BITSET(m_dwStyle,StyleNoHandCursor))
|
||
SetDefaultCursor(); // Try to load an "hand" cursor
|
||
|
||
// Create the tooltip
|
||
CRect rect;
|
||
GetClientRect(rect);
|
||
m_ToolTip.Create(this);
|
||
|
||
m_ToolTip.AddTool(this, m_strURL, rect, TOOLTIP_ID);
|
||
|
||
CStatic::PreSubclassWindow();
|
||
}
|
||
|
||
// Handler for WM_CTLCOLOR reflected message (see message map)
|
||
HBRUSH CHyperLink::CtlColor(CDC* pDC, UINT nCtlColor)
|
||
{
|
||
ASSERT(nCtlColor == CTLCOLOR_STATIC);
|
||
|
||
if (m_bOverControl && BITSET(m_dwStyle,StyleUseHover))
|
||
pDC->SetTextColor(g_crHoverColor);
|
||
else if (!BITSET(m_dwStyle,StyleNoActiveColor) && m_bLinkActive)
|
||
pDC->SetTextColor(g_crActiveColor);
|
||
else if (m_bVisited)
|
||
pDC->SetTextColor(g_crVisitedColor);
|
||
else
|
||
pDC->SetTextColor(g_crLinkColor);
|
||
|
||
// Set transparent drawing mode
|
||
pDC->SetBkMode(TRANSPARENT);
|
||
return (HBRUSH)GetStockObject(NULL_BRUSH);
|
||
}
|
||
|
||
void CHyperLink::OnMouseMove(UINT nFlags, CPoint point)
|
||
{
|
||
|
||
if (m_bOverControl) // Cursor currently over control
|
||
{
|
||
CRect rect;
|
||
GetClientRect(rect);
|
||
|
||
if (!rect.PtInRect(point))
|
||
{
|
||
m_bOverControl = FALSE;
|
||
ReleaseCapture();
|
||
Invalidate();
|
||
return;
|
||
}
|
||
}
|
||
else // Cursor has left control area
|
||
{
|
||
m_bOverControl = TRUE;
|
||
Invalidate();
|
||
SetCapture();
|
||
}
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////////
|
||
// "Normally, a static control does not get mouse events unless it has
|
||
// SS_NOTIFY. This achieves the same effect as SS_NOTIFY, but it's fewer
|
||
// lines of code and more reliable than turning on SS_NOTIFY in OnCtlColor
|
||
// because Windows doesn't send WM_CTLCOLOR to bitmap static controls."
|
||
// (Paul DiLascia)
|
||
LRESULT CHyperLink::OnNcHitTest(CPoint /*point*/)
|
||
{
|
||
return HTCLIENT;
|
||
}
|
||
|
||
void CHyperLink::OnLButtonDown(UINT /*nFlags*/, CPoint /*point*/)
|
||
{
|
||
if (BITSET(m_dwStyle,StyleGetFocusOnClick))
|
||
SetFocus(); // Set the focus and make the link active
|
||
if (BITSET(m_dwStyle,StyleDownClick))
|
||
FollowLink();
|
||
m_bLinkActive = TRUE;
|
||
}
|
||
|
||
void CHyperLink::OnLButtonUp(UINT /*nFlags*/, CPoint /*point*/)
|
||
{
|
||
if (m_bLinkActive && !BITSET(m_dwStyle,StyleDownClick))
|
||
FollowLink();
|
||
}
|
||
|
||
BOOL CHyperLink::OnSetCursor(CWnd* /*pWnd*/, UINT /*nHitTest*/, UINT /*message*/)
|
||
{
|
||
if (g_hLinkCursor)
|
||
{
|
||
::SetCursor(g_hLinkCursor);
|
||
return TRUE;
|
||
}
|
||
return FALSE;
|
||
}
|
||
|
||
void CHyperLink::OnSetFocus(CWnd* /*pOldWnd*/)
|
||
{
|
||
m_bLinkActive = TRUE;
|
||
Invalidate(); // Repaint to set the focus
|
||
}
|
||
|
||
void CHyperLink::OnKillFocus(CWnd* /*pNewWnd*/)
|
||
{
|
||
// Assume that control lost focus = mouse out
|
||
// this avoid troubles with the Hover color
|
||
m_bOverControl = FALSE;
|
||
m_bLinkActive = FALSE;
|
||
Invalidate(); // Repaint to unset the focus
|
||
}
|
||
|
||
void CHyperLink::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)
|
||
{
|
||
if (nChar == VK_SPACE)
|
||
FollowLink();
|
||
else
|
||
CStatic::OnKeyDown(nChar, nRepCnt, nFlags);
|
||
}
|
||
|
||
/////////////////////////////////////////////////////////////////////////////
|
||
// CHyperLink operations
|
||
|
||
void CHyperLink::SetColors( COLORREF crLinkColor,
|
||
COLORREF crActiveColor,
|
||
COLORREF crVisitedColor,
|
||
COLORREF crHoverColor /* = -1 */)
|
||
{
|
||
g_crLinkColor = crLinkColor;
|
||
g_crActiveColor = crActiveColor;
|
||
g_crVisitedColor = crVisitedColor;
|
||
|
||
if (crHoverColor == -1)
|
||
g_crHoverColor = ::GetSysColor(COLOR_HIGHLIGHT);
|
||
else
|
||
g_crHoverColor = crHoverColor;
|
||
}
|
||
|
||
void CHyperLink::SetColors(HYPERLINKCOLORS& linkColors) {
|
||
g_crLinkColor = linkColors.crLink;
|
||
g_crActiveColor = linkColors.crActive;
|
||
g_crVisitedColor = linkColors.crVisited;
|
||
g_crHoverColor = linkColors.crHover;
|
||
}
|
||
|
||
void CHyperLink::GetColors(HYPERLINKCOLORS& linkColors) {
|
||
linkColors.crLink = g_crLinkColor;
|
||
linkColors.crActive = g_crActiveColor;
|
||
linkColors.crVisited = g_crVisitedColor;
|
||
linkColors.crHover = g_crHoverColor;
|
||
}
|
||
|
||
void CHyperLink::SetLinkCursor(HCURSOR hCursor) {
|
||
ASSERT(hCursor != NULL);
|
||
|
||
g_hLinkCursor = hCursor;
|
||
if (g_hLinkCursor == NULL)
|
||
SetDefaultCursor();
|
||
}
|
||
|
||
HCURSOR CHyperLink::GetLinkCursor() {
|
||
return g_hLinkCursor;
|
||
}
|
||
|
||
BOOL CHyperLink:: ModifyLinkStyle(DWORD dwRemove, DWORD dwAdd,
|
||
BOOL bApply /* =TRUE */)
|
||
{
|
||
// Check if we are adding and removing the same style
|
||
if ((dwRemove & dwAdd) != 0L)
|
||
return FALSE;
|
||
|
||
// Remove old styles and set the new ones
|
||
CLEARBITS(m_dwStyle, dwRemove);
|
||
SETBITS(m_dwStyle, dwAdd);
|
||
|
||
if (bApply && ::IsWindow(GetSafeHwnd())) {
|
||
// If possible, APPLY the new styles on the fly
|
||
if (BITSET(dwAdd,StyleUnderline) || BITSET(dwRemove,StyleUnderline))
|
||
SwitchUnderline();
|
||
if (BITSET(dwAdd,StyleAutoSize))
|
||
AdjustWindow();
|
||
if (BITSET(dwRemove,StyleUseHover))
|
||
Invalidate();
|
||
}
|
||
return TRUE;
|
||
}
|
||
|
||
DWORD CHyperLink::GetLinkStyle() const {
|
||
return m_dwStyle;
|
||
}
|
||
|
||
void CHyperLink::SetURL(CString strURL)
|
||
{
|
||
m_strURL = strURL;
|
||
|
||
if (::IsWindow(GetSafeHwnd())) {
|
||
ShowWindow(SW_HIDE);
|
||
AdjustWindow();
|
||
m_ToolTip.UpdateTipText(strURL, this, TOOLTIP_ID);
|
||
ShowWindow(SW_SHOW);
|
||
}
|
||
}
|
||
|
||
CString CHyperLink::GetURL() const {
|
||
return m_strURL;
|
||
}
|
||
|
||
void CHyperLink::SetWindowText(LPCTSTR lpszText)
|
||
{
|
||
ASSERT(lpszText != NULL);
|
||
|
||
if (::IsWindow(GetSafeHwnd())) {
|
||
// Set the window text and adjust its size while the window
|
||
// is kept hidden in order to allow dynamic modification
|
||
ShowWindow(SW_HIDE); // Hide window
|
||
// Call the base class SetWindowText()
|
||
CStatic::SetWindowText(lpszText);
|
||
// Resize the control if necessary
|
||
AdjustWindow();
|
||
ShowWindow(SW_SHOW); // Show window
|
||
}
|
||
}
|
||
|
||
void CHyperLink::SetFont(CFont* pFont)
|
||
{
|
||
ASSERT(::IsWindow(GetSafeHwnd()));
|
||
ASSERT(pFont != NULL);
|
||
|
||
// Set the window font and adjust its size while the window
|
||
// is kept hidden in order to allow dynamic modification
|
||
ShowWindow(SW_HIDE); // Hide window
|
||
LOGFONT lf;
|
||
// Create the new font
|
||
pFont->GetLogFont(&lf);
|
||
m_Font.DeleteObject();
|
||
m_Font.CreateFontIndirect(&lf);
|
||
// Call the base class SetFont()
|
||
CStatic::SetFont(&m_Font);
|
||
// Resize the control if necessary
|
||
AdjustWindow();
|
||
ShowWindow(SW_SHOW); // Show window
|
||
}
|
||
|
||
// Function to set underline on/off
|
||
void CHyperLink::SwitchUnderline()
|
||
{
|
||
LOGFONT lf;
|
||
CFont* pFont = GetFont();
|
||
if (pFont != NULL) {
|
||
pFont->GetLogFont(&lf);
|
||
lf.lfUnderline = BITSET(m_dwStyle,StyleUnderline);
|
||
m_Font.DeleteObject();
|
||
m_Font.CreateFontIndirect(&lf);
|
||
SetFont(&m_Font);
|
||
}
|
||
}
|
||
|
||
// Move and resize the window so that its client area has the same size
|
||
// as the hyperlink text. This prevents the hyperlink cursor being active
|
||
// when it is not over the text.
|
||
void CHyperLink::AdjustWindow()
|
||
{
|
||
ASSERT(::IsWindow(GetSafeHwnd()));
|
||
|
||
if (!BITSET(m_dwStyle,StyleAutoSize))
|
||
return;
|
||
|
||
// Get the current window rect
|
||
CRect rcWnd;
|
||
GetWindowRect(rcWnd);
|
||
|
||
// For a child CWnd object, window rect is relative to the
|
||
// upper-left corner of the parent window<6F>s client area.
|
||
CWnd* pParent = GetParent();
|
||
if (pParent)
|
||
pParent->ScreenToClient(rcWnd);
|
||
|
||
// Get the current client rect
|
||
CRect rcClient;
|
||
GetClientRect(rcClient);
|
||
|
||
// Calc border size based on window and client rects
|
||
int borderWidth = rcWnd.Width() - rcClient.Width();
|
||
int borderHeight = rcWnd.Height() - rcClient.Height();
|
||
|
||
// Get the extent of window text
|
||
CString strWndText;
|
||
GetWindowText(strWndText);
|
||
|
||
CDC* pDC = GetDC();
|
||
CFont* pOldFont = pDC->SelectObject(&m_Font);
|
||
CSize Extent = pDC->GetTextExtent(strWndText);
|
||
pDC->SelectObject(pOldFont);
|
||
ReleaseDC(pDC);
|
||
|
||
// Get the text justification style
|
||
DWORD dwStyle = GetStyle();
|
||
|
||
// Recalc window size and position based on text justification
|
||
if (BITSET(dwStyle, SS_CENTERIMAGE))
|
||
rcWnd.DeflateRect(0, (rcWnd.Height() - Extent.cy) / 2);
|
||
else
|
||
rcWnd.bottom = rcWnd.top + Extent.cy;
|
||
|
||
if (BITSET(dwStyle, SS_CENTER))
|
||
rcWnd.DeflateRect((rcWnd.Width() - Extent.cx) / 2, 0);
|
||
else if (BITSET(dwStyle,SS_RIGHT))
|
||
rcWnd.left = rcWnd.right - Extent.cx;
|
||
else // SS_LEFT
|
||
rcWnd.right = rcWnd.left + Extent.cx;
|
||
|
||
// Move and resize the window
|
||
MoveWindow(rcWnd.left, rcWnd.top, rcWnd.Width() + borderWidth,
|
||
rcWnd.Height() + borderHeight);
|
||
}
|
||
|
||
void CHyperLink::SetVisited(BOOL bVisited /* = TRUE */) {
|
||
m_bVisited = bVisited;
|
||
}
|
||
|
||
BOOL CHyperLink::IsVisited() const {
|
||
return m_bVisited;
|
||
}
|
||
|
||
/////////////////////////////////////////////////////////////////////////////
|
||
// CHyperLink implementation
|
||
|
||
// The following function appeared in Paul DiLascia's Jan 1998
|
||
// MSJ articles. It loads a "hand" cursor from "winhlp32.exe"
|
||
// resources
|
||
void CHyperLink::SetDefaultCursor()
|
||
{
|
||
if (g_hLinkCursor == NULL) // No cursor handle - load our own
|
||
{
|
||
// Get the windows directory
|
||
CString strWndDir;
|
||
GetWindowsDirectory(strWndDir.GetBuffer(MAX_PATH), MAX_PATH);
|
||
strWndDir.ReleaseBuffer();
|
||
|
||
strWndDir += _T("\\winhlp32.exe");
|
||
// This retrieves cursor #106 from winhlp32.exe, which is a hand pointer
|
||
HMODULE hModule = LoadLibrary(strWndDir);
|
||
if (hModule) {
|
||
HCURSOR hHandCursor = ::LoadCursor(hModule, MAKEINTRESOURCE(106));
|
||
if (hHandCursor)
|
||
g_hLinkCursor = CopyCursor(hHandCursor);
|
||
FreeLibrary(hModule);
|
||
}
|
||
}
|
||
}
|
||
|
||
LONG CHyperLink::GetRegKey(HKEY key, LPCTSTR subkey, LPTSTR retdata)
|
||
{
|
||
HKEY hkey;
|
||
LONG retval = RegOpenKeyEx(key, subkey, 0, KEY_QUERY_VALUE, &hkey);
|
||
|
||
if (retval == ERROR_SUCCESS) {
|
||
long datasize = MAX_PATH;
|
||
TCHAR data[MAX_PATH];
|
||
RegQueryValue(hkey, NULL, data, &datasize);
|
||
lstrcpy(retdata,data);
|
||
RegCloseKey(hkey);
|
||
}
|
||
|
||
return retval;
|
||
}
|
||
|
||
// Error report function
|
||
void CHyperLink::ReportError(int nError)
|
||
{
|
||
CString str;
|
||
|
||
switch (nError) {
|
||
case 0: str = _T("The operating system is out\nof memory or resources."); break;
|
||
case ERROR_FILE_NOT_FOUND: str = _T("The specified file was not found."); break;
|
||
case ERROR_PATH_NOT_FOUND: str = _T("The specified path was not found."); break;
|
||
case ERROR_BAD_FORMAT: str = _T("The .EXE file is invalid\n(non-Win32 .EXE or error in .EXE image)."); break;
|
||
case SE_ERR_ACCESSDENIED: str = _T("The operating system denied\naccess to the specified file."); break;
|
||
case SE_ERR_ASSOCINCOMPLETE: str = _T("The filename association is\nincomplete or invalid."); break;
|
||
case SE_ERR_DDEBUSY: str = _T("The DDE transaction could not\nbe completed because other DDE transactions\nwere being processed."); break;
|
||
case SE_ERR_DDEFAIL: str = _T("The DDE transaction failed."); break;
|
||
case SE_ERR_DDETIMEOUT: str = _T("The DDE transaction could not\nbe completed because the request timed out."); break;
|
||
case SE_ERR_DLLNOTFOUND: str = _T("The specified dynamic-link library was not found."); break;
|
||
//case SE_ERR_FNF: str = _T("Windows 95 only: The specified file was not found."); break;
|
||
case SE_ERR_NOASSOC: str = _T("There is no application associated\nwith the given filename extension."); break;
|
||
case SE_ERR_OOM: str = _T("There was not enough memory to complete the operation."); break;
|
||
//case SE_ERR_PNF: str = _T("The specified path was not found."); break;
|
||
case SE_ERR_SHARE: str = _T("A sharing violation occurred. "); break;
|
||
default: str.Format(_T("Unknown Error (%d) occurred."), nError); break;
|
||
}
|
||
|
||
str = "Can't open link:\n\n" + str;
|
||
AfxMessageBox(str, MB_ICONEXCLAMATION | MB_OK);
|
||
}
|
||
|
||
// "GotoURL" function by Stuart Patterson
|
||
// As seen in the August, 1997 Windows Developer's Journal.
|
||
HINSTANCE CHyperLink::GotoURL(LPCTSTR url, int showcmd)
|
||
{
|
||
USES_CONVERSION;
|
||
|
||
TCHAR key[MAX_PATH + MAX_PATH];
|
||
|
||
// First try ShellExecute()
|
||
HINSTANCE result = ShellExecute(NULL, _T("open"), url, NULL,NULL, showcmd);
|
||
|
||
// If it failed, get the .htm regkey and lookup the program
|
||
if ((UINT)result <= HINSTANCE_ERROR) {
|
||
|
||
if (GetRegKey(HKEY_CLASSES_ROOT, _T(".htm"), key) == ERROR_SUCCESS) {
|
||
lstrcat(key, _T("\\shell\\open\\command"));
|
||
|
||
if (GetRegKey(HKEY_CLASSES_ROOT,key,key) == ERROR_SUCCESS) {
|
||
TCHAR *pos;
|
||
pos = _tcsstr(key, _T("\"%1\""));
|
||
if (pos == NULL) { // No quotes found
|
||
pos = _tcsstr(key, _T("%1")); // Check for %1, without quotes
|
||
if (pos == NULL) // No parameter at all...
|
||
pos = key+lstrlen(key) - 1;
|
||
else
|
||
*pos = '\0'; // Remove the parameter
|
||
}
|
||
else
|
||
*pos = '\0'; // Remove the parameter
|
||
|
||
lstrcat(pos, _T(" "));
|
||
lstrcat(pos, url);
|
||
result = (HINSTANCE)WinExec(T2A(key), showcmd);
|
||
PROCESS_INFORMATION ProcessInformation;
|
||
STARTUPINFO startupinfo;
|
||
memset(&startupinfo, 0, sizeof(startupinfo));
|
||
startupinfo.cb = sizeof(startupinfo);
|
||
CreateProcess(0, key, 0, 0, 0, 0, 0, 0, &startupinfo, &ProcessInformation);
|
||
}
|
||
}
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
// Activate the link
|
||
void CHyperLink::FollowLink()
|
||
{
|
||
int result = (int) GotoURL(m_strURL, SW_SHOW);
|
||
if (result <= HINSTANCE_ERROR) {
|
||
MessageBeep(MB_ICONEXCLAMATION); // Unable to follow link
|
||
ReportError(result);
|
||
} else {
|
||
// Mark link as visited and repaint window
|
||
m_bVisited = TRUE;
|
||
Invalidate();
|
||
}
|
||
} |