2020-09-30 16:53:55 +02:00

1702 lines
56 KiB
C++

/**********************************************************************/
/** Microsoft Windows/NT **/
/** Copyright(c) Microsoft Corporation, 1995 - 1999 **/
/**********************************************************************/
/*
FILE HISTORY:
ipaddr.c - TCP/IP Address custom control
November 9, 1992 Greg Strange
December 13, 1993 Ronald Meijer - Wildcard and readonly style bits
April 18, 1994 Ronald Meijer - Added IP_SETREADONLY, IP_SETFIELD
*/
#include <stdafx.h>
//nclude <windows.h>
//nclude <stdlib.h>
#ifdef IP_CUST_CTRL
#include <custcntl.h>
#endif
#include "ipaddr.h" // Global IPAddress definitions
#include "ipadd.h" // Internal IPAddress definitions
/* global static variables */
static HINSTANCE s_hLibInstance = NULL;
#ifdef IP_CUST_CTRL
HANDLE hLibData;
LPFNSTRTOID lpfnVerId;
LPFNIDTOSTR lpfnIdStr;
#endif
/*
Strings loaded at initialization.
*/
TCHAR szNoMem[MAX_IPNOMEMSTRING]; // Out of memory string
TCHAR szCaption[MAX_IPCAPTION]; // Alert message box caption
#define IPADDRESS_CLASS TEXT("IPAddress")
// The character that is displayed between address fields.
#define FILLER TEXT('.')
#define SZFILLER TEXT(".")
#define SPACE TEXT(' ')
#define WILDCARD TEXT('*')
#define SZWILDCARD TEXT(" *")
#define BACK_SPACE 8
// Min, max values
#define NUM_FIELDS 4
#define CHARS_PER_FIELD 3
#define HEAD_ROOM 1 // space at top of control
#define LEAD_ROOM 1 // space at front of control
#define MIN_FIELD_VALUE 0 // default minimum allowable field value
#define MAX_FIELD_VALUE 255 // default maximum allowable field value
// All the information unique to one control is stuffed in one of these
// structures in global memory and the handle to the memory is stored in the
// Windows extra space.
typedef struct tagFIELD {
HWND hWnd;
WNDPROC lpfnWndProc;
BYTE byLow; // lowest allowed value for this field.
BYTE byHigh; // Highest allowed value for this field.
HFONT hFont; // Handle to the logical Font
} FIELD;
typedef struct tagCONTROL {
HWND hwndParent;
UINT uiFieldWidth;
UINT uiFillerWidth;
BOOL fEnabled;
BOOL fPainted;
BOOL fAllowWildcards;
BOOL fReadOnly;
BOOL fInMessageBox; // Set when a message box is displayed so that
// we don't send a EN_KILLFOCUS message when
// we receive the EN_KILLFOCUS message for the
// current field.
BOOL fModified ; // Indicates whether field has changed
FIELD Children[NUM_FIELDS];
} CONTROL;
// The following macros extract and store the CONTROL structure for a control.
#define IPADDRESS_EXTRA (2 * sizeof(LONG_PTR))
#define GET_CONTROL_HANDLE(hWnd) ((HGLOBAL)(GetWindowLongPtr((hWnd), GWLP_USERDATA)))
#define SAVE_CONTROL_HANDLE(hWnd,x) (SetWindowLongPtr((hWnd), GWLP_USERDATA, (LONG_PTR)x))
#define IPADDR_GET_SUBSTYLE(hwnd) (GetWindowLongPtr((hwnd), sizeof(LONG_PTR) * 1))
#define IPADDR_SET_SUBSTYLE(hwnd, style) (SetWindowLongPtr((hwnd), sizeof(LONG_PTR) * 1, (style)))
/* internal IPAddress function prototypes */
#ifdef IP_CUST_CTRL
BOOL FAR WINAPI IPAddressDlgFn( HWND, WORD, WORD, LONG );
void GetStyleBit(HWND, LPCTLSTYLE, int, DWORD);
#endif
LRESULT FAR WINAPI IPAddressWndFn( HWND, UINT, WPARAM, LPARAM );
LRESULT FAR WINAPI IPAddressFieldProc(HWND, UINT, WPARAM, LPARAM);
BOOL SwitchFields(CONTROL FAR *, int, int, WORD, WORD);
void EnterField(FIELD FAR *, WORD, WORD);
BOOL ExitField(CONTROL FAR *, int iField);
int GetFieldValue(FIELD FAR *);
int FAR CDECL IPAlertPrintf(HWND hwndParent, UINT ids, int iCurrent, int iLow, int iHigh);
BOOL IPLoadOem(HINSTANCE hInst, UINT idResource, TCHAR* lpszBuffer, int cbBuffer);
/*
LibMain() - Called once before anything else.
call
hInstance = library instance handle
wDataSegment = library data segment
wHeapSize = default heap size
lpszCmdLine = command line arguements
When this file is compiled as a DLL, this function is called by Libentry()
when the library is first loaded. See the SDK docs for details.
*/
#ifdef IPDLL
/*
//DLL_BASED BOOL WINAPI IpAddrDllEntry (
DLL_BASED BOOL WINAPI DllMain (
HINSTANCE hDll,
DWORD dwReason,
LPVOID lpReserved
)
{
BOOL bResult = TRUE ;
switch ( dwReason )
{
case DLL_PROCESS_ATTACH:
bResult = IPAddrInit( hDll ) ;
break ;
case DLL_THREAD_ATTACH:
break ;
case DLL_PROCESS_DETACH:
break ;
case DLL_THREAD_DETACH:
break ;
}
return bResult ;
}
*/
#endif
#ifdef FE_SB
BYTE
CodePageToCharSet(
UINT CodePage
)
{
CHARSETINFO csi;
if (!TranslateCharsetInfo((DWORD *)ULongToPtr(CodePage), &csi, TCI_SRCCODEPAGE))
csi.ciCharset = OEM_CHARSET;
return (BYTE)csi.ciCharset;
}
#endif // FE_SB
LOGFONT logfont;
void SetDefaultFont( )
{
LANGID langid = PRIMARYLANGID(GetThreadLocale());
BOOL fIsDbcs = (langid == LANG_CHINESE ||
langid == LANG_JAPANESE ||
langid == LANG_KOREAN);
logfont.lfWidth = 0;
logfont.lfEscapement = 0;
logfont.lfOrientation = 0;
logfont.lfOutPrecision = OUT_DEFAULT_PRECIS;
logfont.lfClipPrecision = CLIP_DEFAULT_PRECIS;
logfont.lfQuality = DEFAULT_QUALITY;
logfont.lfPitchAndFamily = VARIABLE_PITCH | FF_SWISS;
logfont.lfUnderline = 0;
logfont.lfStrikeOut = 0;
logfont.lfItalic = 0;
logfont.lfWeight = FW_NORMAL;
HDC hDC = GetDC(NULL);
if (hDC)
{
if (fIsDbcs)
{
logfont.lfHeight = -(9*GetDeviceCaps(hDC,LOGPIXELSY)/72);
logfont.lfCharSet = DEFAULT_CHARSET;
}
else
{
logfont.lfHeight = -(8*GetDeviceCaps(hDC,LOGPIXELSY)/72);
logfont.lfCharSet = ANSI_CHARSET;
}
// logfont.lfHeight = -(8*GetDeviceCaps(GetDC(NULL),LOGPIXELSY)/72);
//fdef FE_SB
// logfont.lfCharSet = CodePageToCharSet( GetACP() );
//lse
// logfont.lfCharSet = ANSI_CHARSET;
//ndif
lstrcpy(logfont.lfFaceName, TEXT("MS Shell Dlg"));
ReleaseDC(NULL, hDC);
}
}
/*
IPAddrInit() - IPAddress custom control initialization
call
hInstance = library or application instance
return
TRUE on success, FALSE on failure.
This function does all the one time initialization of IPAddress custom
controls. Specifically it creates the IPAddress window class.
*/
DLL_BASED int FAR WINAPI IPAddrInit(HINSTANCE hInstance)
{
HGLOBAL hClassStruct;
LPWNDCLASS lpClassStruct;
/* register IPAddress window if necessary */
if ( s_hLibInstance == NULL ) {
/* allocate memory for class structure */
hClassStruct = GlobalAlloc( GHND, (DWORD)sizeof(WNDCLASS) );
if ( hClassStruct ) {
/* lock it down */
lpClassStruct = (LPWNDCLASS)GlobalLock( hClassStruct );
if ( lpClassStruct ) {
/* define class attributes */
lpClassStruct->lpszClassName = IPADDRESS_CLASS;
lpClassStruct->hCursor = LoadCursor(NULL,IDC_IBEAM);
lpClassStruct->lpszMenuName = (LPCTSTR)NULL;
lpClassStruct->style = CS_HREDRAW|CS_VREDRAW|CS_DBLCLKS|CS_GLOBALCLASS;
lpClassStruct->lpfnWndProc = IPAddressWndFn;
lpClassStruct->hInstance = hInstance;
lpClassStruct->hIcon = NULL;
lpClassStruct->cbWndExtra = IPADDRESS_EXTRA;
lpClassStruct->hbrBackground = (HBRUSH)(COLOR_WINDOW + 1 );
/* register IPAddress window class */
s_hLibInstance = ( RegisterClass(lpClassStruct) ) ? hInstance : NULL;
if (hInstance)
{
/* Load caption and out of memory string before we're
out of memory. */
if (!IPLoadOem(hInstance, IDS_IPNOMEM, szNoMem,
sizeof(szNoMem) / sizeof(*szNoMem))
|| !IPLoadOem(hInstance, IDS_IPMBCAPTION, szCaption,
sizeof(szCaption) / sizeof(*szCaption)))
return FALSE;
}
GlobalUnlock( hClassStruct );
}
GlobalFree( hClassStruct );
}
}
SetDefaultFont();
return s_hLibInstance != NULL ;
}
// Use this function to force the ip address entered to
// be contiguous (series of 1's followed by a series of 0's).
// This is useful for entering valid submasks
//
// Returns NO_ERROR if successful, error code otherwise
//
DWORD APIENTRY IpAddr_ForceContiguous(HWND hwndIpAddr) {
DWORD_PTR dwOldStyle;
// Set the last error information so that we can
// return an error correctly
SetLastError(NO_ERROR);
// Set the extended style of the given window so
// that it descriminates the address entered.
dwOldStyle = IPADDR_GET_SUBSTYLE(hwndIpAddr);
IPADDR_SET_SUBSTYLE(hwndIpAddr, dwOldStyle | IPADDR_EX_STYLE_CONTIGUOUS);
return GetLastError();
}
/*
IPAddressInfo() - Returns various bits of information about the control.
returns
A handle for a CtlInfo structure.
This function is only included in the DLL and is used by the dialog
editor.
*/
#ifdef IP_CUST_CTRL
HANDLE FAR WINAPI IPAddressInfo()
{
HGLOBAL hCtlInfo;
LPCTLINFO lpCtlInfo;
/* allocate space for information structure */
hCtlInfo = GlobalAlloc( GHND, (DWORD)sizeof(CTLINFO) );
if ( hCtlInfo ) {
/* attempt to lock it down */
lpCtlInfo = (LPCTLINFO)GlobalLock( hCtlInfo );
if ( lpCtlInfo ) {
/* define the fixed portion of the structure */
lpCtlInfo->wVersion = 100;
lpCtlInfo->wCtlTypes = 1;
lstrcpy( lpCtlInfo->szClass, IPADDRESS_CLASS );
lstrcpy( lpCtlInfo->szTitle, TEXT("TCP/IP IP Address") );
/* define the variable portion of the structure */
lpCtlInfo->Type[0].wWidth = NUM_FIELDS*(CHARS_PER_FIELD+1) * 4 + 4;
lpCtlInfo->Type[0].wHeight = 13;
lpCtlInfo->Type[0].dwStyle = WS_CHILD | WS_TABSTOP;
lstrcpy( lpCtlInfo->Type[0].szDescr, TEXT("IPAddress") );
/* unlock it */
GlobalUnlock( hCtlInfo );
} else {
GlobalFree( hCtlInfo );
hCtlInfo = NULL;
}
}
/* return result */
return( hCtlInfo );
}
#endif
/*
IPAddressStyle()
call
hWnd handle of parent window
hCtlStyle handle to control style info
lpfnVerifyId pointer to the VerifyId function from dialog editor
lpfnGetIDStr pointer to the GetIdStr function from dialog editor
This function is called by the dialog editor when the user double clicks
on the custom control. Or when the user chooses to edit the control's
styles.
*/
#ifdef IP_CUST_CTRL
BOOL FAR WINAPI IPAddressStyle(
HWND hWnd,
HANDLE hCtlStyle,
LPFNSTRTOID lpfnVerifyId,
LPFNIDTOSTR lpfnGetIdStr )
{
FARPROC lpDlgFn;
HANDLE hNewCtlStyle;
// initialization
hLibData = hCtlStyle;
lpfnVerId = lpfnVerifyId;
lpfnIdStr = lpfnGetIdStr;
// display dialog box
lpDlgFn = MakeProcInstance( (FARPROC)IPAddressDlgFn, s_hLibInstance );
hNewCtlStyle = ( DialogBox(s_hLibInstance,TEXT("IPAddressStyle"),hWnd,lpDlgFn) ) ? hLibData : NULL;
FreeProcInstance( lpDlgFn );
// return updated data block
return( hNewCtlStyle );
}
#endif
/*
IPAddressDlgFn() - Dialog editor style dialog
hDlg styles dialog box handle
wMessage window message
wParam word parameter
lParam long parameter
This is the dialog function for the styles dialog that is displayed when
the user wants to edit an IPAddress control's style from the dialog editor.
*/
#ifdef IP_CUST_CTRL
BOOL FAR WINAPI IPAddressDlgFn(
HWND hDlg,
WORD wMessage,
WORD wParam,
LONG lParam )
{
BOOL bResult;
/* initialization */
bResult = TRUE;
/* process message */
switch( wMessage )
{
case WM_INITDIALOG :
{
HANDLE hCtlStyle;
LPCTLSTYLE lpCtlStyle;
/* disable Ok button & save dialog data handle */
hCtlStyle = hLibData;
/* retrieve & display style parameters */
if ( hCtlStyle ) {
/* add handle to property list */
SetProp( hDlg, MAKEINTRESOURCE(1), hCtlStyle );
/* update dialog box fields */
lpCtlStyle = (LPCTLSTYLE)GlobalLock( hCtlStyle );
lstrcpy( lpCtlStyle->szClass, IPADDRESS_CLASS );
SendDlgItemMessage(hDlg, ID_VISIBLE, BM_SETCHECK,
(WPARAM)((lpCtlStyle->dwStyle & WS_VISIBLE) != 0), 0L);
SendDlgItemMessage(hDlg, ID_GROUP, BM_SETCHECK,
(WPARAM)((lpCtlStyle->dwStyle & WS_GROUP) != 0), 0L);
SendDlgItemMessage(hDlg, ID_DISABLED, BM_SETCHECK,
(WPARAM)((lpCtlStyle->dwStyle & WS_DISABLED) != 0), 0L);
SendDlgItemMessage(hDlg, ID_TABSTOP, BM_SETCHECK,
(WPARAM)((lpCtlStyle->dwStyle & WS_TABSTOP) != 0), 0L);
GlobalUnlock( hCtlStyle );
} else
EndDialog( hDlg, FALSE );
}
break;
case WM_COMMAND :
switch( wParam )
{
case IDCANCEL:
RemoveProp( hDlg, MAKEINTRESOURCE(1) );
EndDialog( hDlg, FALSE );
break;
case IDOK:
{
HANDLE hCtlStyle;
LPCTLSTYLE lpCtlStyle;
hCtlStyle = GetProp( hDlg, MAKEINTRESOURCE(1) );
lpCtlStyle = (LPCTLSTYLE)GlobalLock( hCtlStyle );
GetStyleBit(hDlg, lpCtlStyle, ID_VISIBLE, WS_VISIBLE);
GetStyleBit(hDlg, lpCtlStyle, ID_DISABLED, WS_DISABLED);
GetStyleBit(hDlg, lpCtlStyle, ID_GROUP, WS_GROUP);
GetStyleBit(hDlg, lpCtlStyle, ID_TABSTOP, WS_TABSTOP);
GlobalUnlock( hCtlStyle );
RemoveProp( hDlg, MAKEINTRESOURCE(1) );
hLibData = hCtlStyle;
EndDialog( hDlg, TRUE );
}
break;
default :
bResult = FALSE;
break;
}
break;
default :
bResult = FALSE;
break;
}
return( bResult );
}
#endif
/*
Get the value of a check box and set the appropriate style bit.
*/
#ifdef IP_CUST_CTRL
void GetStyleBit(HWND hDlg, LPCTLSTYLE lpCtlStyle, int iControl, DWORD dwStyle)
{
if (SendDlgItemMessage(hDlg, iControl, BM_GETSTATE, 0, 0L))
lpCtlStyle->dwStyle |= dwStyle;
else
lpCtlStyle->dwStyle &= ~dwStyle;
}
#endif
/*
IPAddressFlags()
call
wFlags class style flags
lpszString class style string
wMaxString maximum size of class style string
This function translates the class style flags provided into a
corresponding text string for output to an RC file. The general
windows flags (contained in the low byte) are not interpreted,
only those in the high byte.
The value returned by this function is the library instance
handle when sucessful, and NULL otherwise.
*/
#ifdef IP_CUST_CTRL
WORD FAR WINAPI IPAddressFlags(
WORD wFlags,
LPSTR lpszString,
WORD wMaxString )
{
lpszString[0] = NULL;
return( 0 );
}
#endif
// This function causes the ip address entered into hwndIpAddr to be
// corrected so that it is contiguous.
DWORD IpAddrMakeContiguous(HWND hwndIpAddr) {
DWORD i, dwNewMask, dwMask;
// Read in the current address
SendMessage(hwndIpAddr, IP_GETADDRESS, 0, (LPARAM)&dwMask);
// Find out where the first '1' is in binary going right to left
dwNewMask = 0;
for (i = 0; i < sizeof(dwMask)*8; i++) {
dwNewMask |= 1 << i;
if (dwNewMask & dwMask) {
break;
}
}
// At this point, dwNewMask is 000...0111... If we inverse it,
// we get a mask that can be or'd with dwMask to fill in all of
// the holes.
dwNewMask = dwMask | ~dwNewMask;
// If the new mask is different, correct it here
if (dwMask != dwNewMask) {
// WCHAR pszAddr[32];
// wsprintfW(pszAddr, L"%d.%d.%d.%d", FIRST_IPADDRESS (dwNewMask),
// SECOND_IPADDRESS(dwNewMask),
// THIRD_IPADDRESS (dwNewMask),
// FOURTH_IPADDRESS(dwNewMask));
SendMessage(hwndIpAddr, IP_SETADDRESS, 0, (LPARAM) dwNewMask);
// SendMessage(hwndIpAddr, IP_SETADDRESS, 0, (LPARAM)pszAddr);
}
return NO_ERROR;
}
/*
IPAddressWndFn() - Main window function for an IPAddress control.
call
hWnd handle to IPAddress window
wMsg message number
wParam word parameter
lParam long parameter
*/
LRESULT FAR WINAPI IPAddressWndFn( HWND hWnd,
UINT wMsg,
WPARAM wParam,
LPARAM lParam )
{
LONG_PTR lResult;
HGLOBAL hControl;
CONTROL *pControl;
int i;
lResult = TRUE;
switch( wMsg )
{
case WM_SETTEXT:
{
static TCHAR szBuf[CHARS_PER_FIELD+1];
DWORD dwValue[4];
int nField, nPos;
BOOL fFinish = FALSE;
TCHAR *pszString = (TCHAR*)lParam;
dwValue[0]=0;
dwValue[1]=0;
dwValue[2]=0;
dwValue[3]=0;
for( nField = 0, nPos = 0; !fFinish; nPos++)
{
if (( pszString[nPos]<TEXT('0')) || (pszString[nPos]>TEXT('9')))
{
// not a number
nField++;
fFinish = (nField == 4);
}
else
{
dwValue[nField] *= 10;
dwValue[nField] += (pszString[nPos]-TEXT('0'));
}
}
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
for (i = 0; i < NUM_FIELDS; ++i)
{
if ( lstrcmp(pszString, TEXT("")) == 0 )
{
wsprintf(szBuf,TEXT(""));
}
else
{
wsprintf(szBuf, TEXT("%d"), dwValue[i]);
}
SendMessage(pControl->Children[i].hWnd, WM_SETTEXT,
0, (LPARAM) (LPSTR) szBuf);
}
GlobalUnlock(hControl);
}
break;
case WM_GETTEXTLENGTH:
case WM_GETTEXT:
{
int iFieldValue;
int srcPos, desPos;
DWORD dwValue[4];
TCHAR pszResult[30];
TCHAR *pszDest = (TCHAR *)lParam;
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
lResult = 0;
dwValue[0] = 0;
dwValue[1] = 0;
dwValue[2] = 0;
dwValue[3] = 0;
for (i = 0; i < NUM_FIELDS; ++i)
{
iFieldValue = GetFieldValue(&(pControl->Children[i]));
if (iFieldValue == -1)
iFieldValue = 0;
else
++lResult;
dwValue[i] = iFieldValue;
}
wsprintf( pszResult, TEXT("%d.%d.%d.%d"), dwValue[0], dwValue[1], dwValue[2], dwValue[3] );
if ( wMsg == WM_GETTEXTLENGTH )
{
lResult = lstrlen( pszResult );
}
else
{
for ( srcPos=0, desPos=0; (srcPos+1<(INT)wParam) && (pszResult[srcPos]!=TEXT('\0')); )
{
pszDest[desPos++] = pszResult[srcPos++];
}
pszDest[desPos]=TEXT('\0');
lResult = desPos;
}
GlobalUnlock(hControl);
}
break;
case WM_GETDLGCODE :
lResult = DLGC_WANTCHARS;
break;
case WM_CREATE : /* create pallette window */
{
HDC hdc;
UINT uiFieldStart;
FARPROC lpfnFieldProc;
hControl = GlobalAlloc(GMEM_MOVEABLE, sizeof(CONTROL));
if (hControl)
{
HFONT OldFont;
RECT rectClient;
#define LPCS ((CREATESTRUCT *)lParam)
pControl = (CONTROL *)GlobalLock(hControl);
pControl->fEnabled = TRUE;
pControl->fPainted = FALSE;
pControl->fModified = FALSE ;
pControl->fInMessageBox = FALSE;
pControl->hwndParent = LPCS->hwndParent;
pControl->fAllowWildcards = (LPCS->style & IPS_ALLOWWILDCARDS);
pControl->fReadOnly = (LPCS->style & IPS_READONLY);
hdc = GetDC(hWnd);
if (hdc)
{
OldFont = (HFONT) SelectObject( hdc, CreateFontIndirect(&logfont) );
GetCharWidth(hdc, FILLER, FILLER,
(int *)(&pControl->uiFillerWidth));
HGDIOBJ hObj = SelectObject(hdc, OldFont );
if (hObj)
DeleteObject( hObj );
ReleaseDC(hWnd, hdc);
// we need to calculate this with the client rect
// because we may have a 3d look and feel which makes
// the client area smaller than the window
GetClientRect(hWnd, &rectClient);
pControl->uiFieldWidth = (rectClient.right - rectClient.left
- LEAD_ROOM
- pControl->uiFillerWidth
*(NUM_FIELDS-1))
/ NUM_FIELDS;
uiFieldStart = LEAD_ROOM;
lpfnFieldProc = MakeProcInstance((FARPROC)IPAddressFieldProc,
LPCS->hInstance);
for (i = 0; i < NUM_FIELDS; ++i)
{
pControl->Children[i].byLow = MIN_FIELD_VALUE;
pControl->Children[i].byHigh = MAX_FIELD_VALUE;
pControl->Children[i].hWnd = CreateWindow(
TEXT("Edit"),
NULL,
WS_CHILD | WS_VISIBLE |
/*ES_MULTILINE |*/ ES_CENTER,
uiFieldStart,
HEAD_ROOM,
pControl->uiFieldWidth,
rectClient.bottom - rectClient.top - (HEAD_ROOM*2),
hWnd,
(HMENU)ULongToPtr(i),
LPCS->hInstance,
(LPVOID)ULongToPtr(NULL));
SendMessage(pControl->Children[i].hWnd, EM_LIMITTEXT,
CHARS_PER_FIELD, 0L);
pControl->Children[i].hFont = CreateFontIndirect(&logfont);
SendMessage(pControl->Children[i].hWnd, WM_SETFONT,
(WPARAM)pControl->Children[i].hFont, TRUE);
pControl->Children[i].lpfnWndProc =
(WNDPROC)GetWindowLongPtr(pControl->Children[i].hWnd,
GWLP_WNDPROC);
SetWindowLongPtr(pControl->Children[i].hWnd,
GWLP_WNDPROC, (LPARAM)lpfnFieldProc);
uiFieldStart += pControl->uiFieldWidth
+ pControl->uiFillerWidth;
}
#undef LPCS
}
GlobalUnlock(hControl);
SAVE_CONTROL_HANDLE(hWnd, hControl);
}
else
DestroyWindow(hWnd);
}
lResult = 0;
break;
case WM_PAINT: /* paint control window */
{
PAINTSTRUCT Ps;
RECT rect;
UINT uiFieldStart;
COLORREF TextColor;
HFONT OldFont;
HBRUSH hBrush;
HBRUSH hOldBrush;
BeginPaint(hWnd, (LPPAINTSTRUCT)&Ps);
OldFont = (HFONT) SelectObject( Ps.hdc, CreateFontIndirect(&logfont));
GetClientRect(hWnd, &rect);
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
// paint the background depending upon if the control is enabled
if (pControl->fEnabled)
hBrush = CreateSolidBrush( GetSysColor( COLOR_WINDOW ));
else
hBrush = CreateSolidBrush( GetSysColor( COLOR_BTNFACE ));
hOldBrush = (HBRUSH) SelectObject( Ps.hdc, hBrush );
if (!(GetWindowLong(hWnd, GWL_EXSTYLE) & WS_EX_CLIENTEDGE))
Rectangle(Ps.hdc, 0, 0, rect.right, rect.bottom);
else
FillRect(Ps.hdc, &rect, hBrush);
HGDIOBJ hObj = SelectObject( Ps.hdc, hOldBrush );
if (hObj)
DeleteObject( hObj );
// now set the text color
if (pControl->fEnabled)
TextColor = GetSysColor(COLOR_WINDOWTEXT);
else
TextColor = GetSysColor(COLOR_GRAYTEXT);
if (TextColor)
SetTextColor(Ps.hdc, TextColor);
// and the background color
if (pControl->fEnabled)
SetBkColor(Ps.hdc, GetSysColor(COLOR_WINDOW));
else
SetBkColor(Ps.hdc, GetSysColor(COLOR_BTNFACE));
uiFieldStart = pControl->uiFieldWidth + LEAD_ROOM;
for (i = 0; i < NUM_FIELDS-1; ++i)
{
TextOut(Ps.hdc, uiFieldStart, HEAD_ROOM, SZFILLER, 1);
uiFieldStart +=pControl->uiFieldWidth + pControl->uiFillerWidth;
}
pControl->fPainted = TRUE;
GlobalUnlock(hControl);
DeleteObject(SelectObject(Ps.hdc, OldFont));
EndPaint(hWnd, &Ps);
}
break;
case WM_SETFOCUS : /* get focus - display caret */
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
EnterField(&(pControl->Children[0]), 0, CHARS_PER_FIELD);
GlobalUnlock(hControl);
break;
case WM_LBUTTONDOWN : /* left button depressed - fall through */
SetFocus(hWnd);
break;
case WM_ENABLE:
{
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
pControl->fEnabled = (BOOL)wParam;
for (i = 0; i < NUM_FIELDS; ++i)
{
EnableWindow(pControl->Children[i].hWnd, (BOOL)wParam);
}
EnableWindow(hWnd, (BOOL) wParam);
if (pControl->fPainted) InvalidateRect(hWnd, NULL, FALSE);
GlobalUnlock(hControl);
}
break;
case WM_DESTROY :
hControl = GET_CONTROL_HANDLE(hWnd);
if (!hControl)
break;
pControl = (CONTROL *)GlobalLock(hControl);
// Restore all the child window procedures before we delete our memory block.
for (i = 0; i < NUM_FIELDS; ++i)
{
SetWindowLongPtr(pControl->Children[i].hWnd, GWLP_WNDPROC,
(LPARAM)pControl->Children[i].lpfnWndProc);
DeleteObject(pControl->Children[i].hFont);
}
GlobalUnlock(hControl);
GlobalFree(hControl);
SAVE_CONTROL_HANDLE(hWnd, NULL);
break;
case WM_COMMAND:
switch (HIWORD(wParam))
{
// One of the fields lost the focus, see if it lost the focus to another field
// of if we've lost the focus altogether. If its lost altogether, we must send
// an EN_KILLFOCUS notification on up the ladder.
case EN_KILLFOCUS:
{
HWND hFocus;
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
if (!pControl->fInMessageBox)
{
hFocus = GetFocus();
for (i = 0; i < NUM_FIELDS; ++i)
if (pControl->Children[i].hWnd == hFocus)
break;
if (i >= NUM_FIELDS)
{
// Before sending the address up the
// ladder, make sure that the ip
// address is contiguous, if needed
if (IPADDR_GET_SUBSTYLE(hWnd) &
IPADDR_EX_STYLE_CONTIGUOUS)
IpAddrMakeContiguous(hWnd);
SendMessage(pControl->hwndParent, WM_COMMAND,
MAKEWPARAM(GetWindowLongPtr(hWnd, GWLP_ID),
EN_KILLFOCUS), (LPARAM)hWnd);
}
}
GlobalUnlock(hControl);
}
break;
case EN_CHANGE:
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
SendMessage(pControl->hwndParent, WM_COMMAND,
MAKEWPARAM(GetWindowLongPtr(hWnd, GWLP_ID), EN_CHANGE), (LPARAM)hWnd);
GlobalUnlock(hControl);
break;
}
break;
// Get the value of the IP Address. The address is placed in the DWORD pointed
// to by lParam and the number of non-blank fields is returned.
case IP_GETADDRESS:
{
int iFieldValue;
DWORD dwValue;
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
lResult = 0;
dwValue = 0;
for (i = 0; i < NUM_FIELDS; ++i)
{
iFieldValue = GetFieldValue(&(pControl->Children[i]));
if (iFieldValue == -1)
iFieldValue = 0;
else
++lResult;
dwValue = (dwValue << 8) + iFieldValue;
}
*((DWORD *)lParam) = dwValue;
GlobalUnlock(hControl);
}
break;
case IP_GETMASK:
{
TCHAR szBuf[CHARS_PER_FIELD+1];
WORD wLength;
lResult = 0;
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
for (i = 0; i < NUM_FIELDS; ++i)
{
*(WORD *)szBuf = (sizeof(szBuf) / sizeof(*szBuf)) - 1;
wLength = (WORD)SendMessage(pControl->Children[i].hWnd,
EM_GETLINE,0, (LPARAM) szBuf);
szBuf[wLength] = TEXT('\0');
if (!lstrcmp(szBuf, SZWILDCARD))
{
lResult |= 1L<<i;
}
}
GlobalUnlock(hControl);
}
break;
case IP_GETMODIFY:
{
hControl = GET_CONTROL_HANDLE(hWnd);
if ( ! hControl )
break ;
pControl = (CONTROL *)GlobalLock(hControl);
lResult = pControl->fModified > 0 ;
for (i = 0 ; i < NUM_FIELDS ; )
{
lResult |= SendMessage( pControl->Children[i++].hWnd, EM_GETMODIFY, 0, 0 ) > 0 ;
}
GlobalUnlock(hControl);
}
break ;
case IP_SETMODIFY:
{
hControl = GET_CONTROL_HANDLE(hWnd);
if ( ! hControl )
break ;
pControl = (CONTROL *)GlobalLock(hControl);
pControl->fModified = wParam > 0 ;
for (i = 0 ; i < NUM_FIELDS ; )
{
SendMessage( pControl->Children[i++].hWnd, EM_GETMODIFY, wParam, 0 ) ;
}
GlobalUnlock(hControl);
}
break ;
// Clear all fields to blanks.
case IP_CLEARADDRESS:
{
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
for (i = 0; i < NUM_FIELDS; ++i)
{
SendMessage(pControl->Children[i].hWnd, WM_SETTEXT,
0, (LPARAM) (LPSTR) TEXT(""));
}
GlobalUnlock(hControl);
}
break;
// Set the value of the IP Address. The address is in the lParam with the
// first address byte being the high byte, the second being the second byte,
// and so on. A lParam value of -1 removes the address.
case IP_SETADDRESS:
{
static TCHAR szBuf[CHARS_PER_FIELD+1];
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
for (i = 0; i < NUM_FIELDS; ++i)
{
wsprintf(szBuf, TEXT("%d"), HIBYTE(HIWORD(lParam)));
SendMessage(pControl->Children[i].hWnd, WM_SETTEXT,
0, (LPARAM) (LPSTR) szBuf);
lParam <<= 8;
}
GlobalUnlock(hControl);
}
break;
case IP_SETREADONLY:
{
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
pControl->fReadOnly = (wParam != 0);
GlobalUnlock(hControl);
}
break;
// Set a single field value. The wparam (0-3) indicates the field,
// the lparam (0-255) indicates the value
case IP_SETFIELD:
{
static TCHAR szBuf[CHARS_PER_FIELD+1] = TEXT("");
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
if (wParam < NUM_FIELDS)
{
if (lParam != -1)
{
wsprintf(szBuf, TEXT("%d"), HIBYTE(HIWORD(lParam)));
}
SendMessage(pControl->Children[wParam].hWnd, WM_SETTEXT,
0, (LPARAM) (LPSTR) szBuf);
}
GlobalUnlock(hControl);
}
break;
case IP_SETMASK:
{
BYTE bMask = (BYTE)wParam;
static TCHAR szBuf[CHARS_PER_FIELD+1];
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
for (i = 0; i < NUM_FIELDS; ++i)
{
if (bMask & 1<<i)
{
SendMessage(pControl->Children[i].hWnd, WM_SETTEXT,
0, (LPARAM)SZWILDCARD);
}
else
{
wsprintf(szBuf, TEXT("%d"), HIBYTE(HIWORD(lParam)));
SendMessage(pControl->Children[i].hWnd, WM_SETTEXT,
0, (LPARAM) (LPSTR) szBuf);
}
lParam <<= 8;
}
GlobalUnlock(hControl);
}
break;
case IP_SETRANGE:
if (wParam < NUM_FIELDS)
{
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
pControl->Children[wParam].byLow = LOBYTE(LOWORD(lParam));
pControl->Children[wParam].byHigh = HIBYTE(LOWORD(lParam));
GlobalUnlock(hControl);
}
break;
// Set the focus to this control.
// wParam = the field number to set focus to, or -1 to set the focus to the
// first non-blank field.
case IP_SETFOCUS:
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
if (wParam >= NUM_FIELDS)
{
for (wParam = 0; wParam < NUM_FIELDS; ++wParam)
if (GetFieldValue(&(pControl->Children[wParam])) == -1)
break;
if (wParam >= NUM_FIELDS) wParam = 0;
}
//
// 0, -1 select the entire control
//
EnterField(&(pControl->Children[wParam]), 0, (WORD)-1);
GlobalUnlock(hControl);
break;
// Determine whether all four subfields are blank
case IP_ISBLANK:
hControl = GET_CONTROL_HANDLE(hWnd);
pControl = (CONTROL *)GlobalLock(hControl);
lResult = TRUE;
for (i = 0; i < NUM_FIELDS; ++i)
{
if (GetFieldValue(&(pControl->Children[i])) != -1)
{
lResult = FALSE;
break;
}
}
GlobalUnlock(hControl);
break;
default:
lResult = DefWindowProc( hWnd, wMsg, wParam, lParam );
break;
}
return( lResult );
}
/*
IPAddressFieldProc() - Edit field window procedure
This function sub-classes each edit field.
*/
LRESULT FAR WINAPI IPAddressFieldProc(HWND hWnd,
UINT wMsg,
WPARAM wParam,
LPARAM lParam)
{
HANDLE hControl;
CONTROL *pControl;
FIELD *pField;
HWND hControlWindow;
WORD wChildID;
LRESULT lresult;
if (!(hControlWindow = GetParent(hWnd)))
return 0;
hControl = GET_CONTROL_HANDLE(hControlWindow);
pControl = (CONTROL *)GlobalLock(hControl);
wChildID = (WORD) GetWindowLong(hWnd, GWL_ID);
pField = &(pControl->Children[wChildID]);
if (pField->hWnd != hWnd)
{
return 0;
}
switch (wMsg)
{
case WM_DESTROY:
DeleteObject( (HGDIOBJ) SendMessage( hWnd, WM_GETFONT, 0, 0 ));
return 0;
case WM_CHAR:
if (pControl->fReadOnly)
{
MessageBeep((UINT)-1);
GlobalUnlock( hControl );
return 0;
}
// Typing in the last digit in a field, skips to the next field.
if (wParam >= TEXT('0') && wParam <= TEXT('9'))
{
DWORD_PTR dwResult;
pControl->fModified = TRUE ;
dwResult = CallWindowProc(pControl->Children[wChildID].lpfnWndProc,
hWnd, wMsg, wParam, lParam);
dwResult = SendMessage(hWnd, EM_GETSEL, 0, 0L);
if (dwResult == MAKELPARAM(CHARS_PER_FIELD, CHARS_PER_FIELD)
&& ExitField(pControl, wChildID)
&& wChildID < NUM_FIELDS-1)
{
EnterField(&(pControl->Children[wChildID+1]),
0, CHARS_PER_FIELD);
}
GlobalUnlock( hControl );
return dwResult;
}
// spaces and periods fills out the current field and then if possible,
// goes to the next field.
else if (wParam == FILLER || wParam == SPACE )
{
DWORD_PTR dwResult;
dwResult = SendMessage(hWnd, EM_GETSEL, 0, 0L);
if (dwResult != 0L && HIWORD(dwResult) == LOWORD(dwResult)
&& ExitField(pControl, wChildID))
{
if (wChildID >= NUM_FIELDS-1)
MessageBeep((UINT)-1);
else
{
EnterField(&(pControl->Children[wChildID+1]),
0, CHARS_PER_FIELD);
}
}
GlobalUnlock( hControl );
return 0;
}
// Backspaces go to the previous field if at the beginning of the current field.
// Also, if the focus shifts to the previous field, the backspace must be
// processed by that field.
else if (wParam == BACK_SPACE)
{
pControl->fModified = TRUE ;
if (wChildID > 0 && SendMessage(hWnd, EM_GETSEL, 0, 0L) == 0L)
{
if (SwitchFields(pControl, wChildID, wChildID-1,
CHARS_PER_FIELD, CHARS_PER_FIELD)
&& SendMessage(pControl->Children[wChildID-1].hWnd,
EM_LINELENGTH, 0, 0L) != 0L)
{
SendMessage(pControl->Children[wChildID-1].hWnd,
wMsg, wParam, lParam);
}
GlobalUnlock( hControl );
return 0;
}
else
{
TCHAR szBuf[CHARS_PER_FIELD+1];
WORD wLength;
*(WORD *)szBuf = (sizeof(szBuf) / sizeof(*szBuf)) - 1;
wLength = (WORD)SendMessage(pControl->Children[wChildID].hWnd,
EM_GETLINE,0, (LPARAM) szBuf);
szBuf[wLength] = TEXT('\0');
if (!lstrcmp(szBuf, SZWILDCARD))
{
SendMessage(pControl->Children[wChildID].hWnd,
WM_SETTEXT, 0, (LPARAM)TEXT(""));
}
}
}
else if ((wParam == WILDCARD) && (pControl->fAllowWildcards))
{
// Only works at the beginning of the line.
if (SendMessage(hWnd, EM_GETSEL, 0, 0L) == 0L)
{
pControl->fModified = TRUE;
SendMessage(pControl->Children[wChildID].hWnd, WM_SETTEXT, 0, (LPARAM)SZWILDCARD);
if (ExitField(pControl, wChildID) && (wChildID < NUM_FIELDS-1))
{
EnterField(&(pControl->Children[wChildID+1]),0, CHARS_PER_FIELD);
}
}
else
{
// Not at the beginning of the line, complain
MessageBeep((UINT)-1);
}
GlobalUnlock( hControl );
return 0;
}
// Any other printable characters are not allowed.
else if (wParam > SPACE)
{
MessageBeep((UINT)-1);
GlobalUnlock( hControl );
return 0;
}
break;
case WM_KEYDOWN:
switch (wParam)
{
case VK_DELETE:
{
TCHAR szBuf[CHARS_PER_FIELD+1];
WORD wLength;
if (pControl->fReadOnly)
{
MessageBeep((UINT)-1);
GlobalUnlock( hControl );
return 0;
}
pControl->fModified = TRUE ;
*(WORD *)szBuf = (sizeof(szBuf) / sizeof(*szBuf)) - 1;
wLength = (WORD)SendMessage(pControl->Children[wChildID].hWnd,
EM_GETLINE,0, (LPARAM) szBuf);
szBuf[wLength] = TEXT('\0');
if (!lstrcmp(szBuf, SZWILDCARD))
{
SendMessage(pControl->Children[wChildID].hWnd,
WM_SETTEXT, 0, (LPARAM)TEXT(""));
GlobalUnlock( hControl );
return 0;
}
}
break;
// Arrow keys move between fields when the end of a field is reached.
case VK_LEFT:
case VK_RIGHT:
case VK_UP:
case VK_DOWN:
if (GetKeyState(VK_CONTROL) < 0)
{
if ((wParam == VK_LEFT || wParam == VK_UP) && wChildID > 0)
{
SwitchFields(pControl, wChildID, wChildID-1,
0, CHARS_PER_FIELD);
GlobalUnlock( hControl );
return 0;
}
else if ((wParam == VK_RIGHT || wParam == VK_DOWN)
&& wChildID < NUM_FIELDS-1)
{
SwitchFields(pControl, wChildID, wChildID+1,
0, CHARS_PER_FIELD);
GlobalUnlock( hControl );
return 0;
}
}
else
{
DWORD_PTR dwResult;
WORD wStart, wEnd;
dwResult = SendMessage(hWnd, EM_GETSEL, 0, 0L);
wStart = LOWORD(dwResult);
wEnd = HIWORD(dwResult);
if (wStart == wEnd)
{
if ((wParam == VK_LEFT || wParam == VK_UP)
&& wStart == 0
&& wChildID > 0)
{
SwitchFields(pControl, wChildID, wChildID-1,
CHARS_PER_FIELD, CHARS_PER_FIELD);
GlobalUnlock( hControl );
return 0;
}
else if ((wParam == VK_RIGHT || wParam == VK_DOWN)
&& wChildID < NUM_FIELDS-1)
{
dwResult = SendMessage(hWnd, EM_LINELENGTH, 0, 0L);
if (wStart >= dwResult)
{
SwitchFields(pControl, wChildID, wChildID+1, 0, 0);
GlobalUnlock( hControl );
return 0;
}
}
}
}
break;
// Home jumps back to the beginning of the first field.
case VK_HOME:
if (wChildID > 0)
{
SwitchFields(pControl, wChildID, 0, 0, 0);
GlobalUnlock( hControl );
return 0;
}
break;
// End scoots to the end of the last field.
case VK_END:
if (wChildID < NUM_FIELDS-1)
{
SwitchFields(pControl, wChildID, NUM_FIELDS-1,
CHARS_PER_FIELD, CHARS_PER_FIELD);
GlobalUnlock( hControl );
return 0;
}
break;
} // switch (wParam)
break;
case WM_KILLFOCUS:
if ( !ExitField( pControl, wChildID ))
{
GlobalUnlock( hControl );
return 0;
}
} // switch (wMsg)
lresult = CallWindowProc(pControl->Children[wChildID].lpfnWndProc,
hWnd, wMsg, wParam, lParam);
GlobalUnlock( hControl );
return lresult;
}
/*
Switch the focus from one field to another.
call
pControl = Pointer to the CONTROL structure.
iOld = Field we're leaving.
iNew = Field we're entering.
hNew = Window of field to goto
wStart = First character selected
wEnd = Last character selected + 1
returns
TRUE on success, FALSE on failure.
Only switches fields if the current field can be validated.
*/
BOOL SwitchFields(CONTROL *pControl, int iOld, int iNew, WORD wStart, WORD wEnd)
{
if (!ExitField(pControl, iOld))
return FALSE;
EnterField(&(pControl->Children[iNew]), wStart, wEnd);
return TRUE;
}
/*
Set the focus to a specific field's window.
call
pField = pointer to field structure for the field.
wStart = First character selected
wEnd = Last character selected + 1
*/
void EnterField(FIELD *pField, WORD wStart, WORD wEnd)
{
SetFocus(pField->hWnd);
SendMessage(pField->hWnd, EM_SETSEL, wStart, wEnd);
}
/*
Exit a field.
call
pControl = pointer to CONTROL structure.
iField = field number being exited.
returns
TRUE if the user may exit the field.
FALSE if he may not.
*/
BOOL ExitField(CONTROL *pControl, int iField)
{
HWND hControlWnd;
HWND hDialog;
WORD wLength;
FIELD *pField;
static TCHAR szBuf[CHARS_PER_FIELD+1];
int i,j;
pField = &(pControl->Children[iField]);
*(WORD *)szBuf = (sizeof(szBuf) / sizeof(*szBuf)) - 1;
wLength = (WORD)SendMessage(pField->hWnd,EM_GETLINE,0,(LPARAM)(LPSTR)szBuf);
if (wLength != 0)
{
szBuf[wLength] = TEXT('\0');
if (pControl->fAllowWildcards && !lstrcmp(szBuf, SZWILDCARD))
{
return TRUE;
}
for (j=0,i=0;j<(INT)wLength;j++)
{
i=i*10+szBuf[j]-TEXT('0');
}
if (i < (int)(UINT)pField->byLow || i > (int)(UINT)pField->byHigh)
{
if ( i < (int)(UINT) pField->byLow )
{
/* too small */
wsprintf(szBuf, TEXT("%d"), (int)(UINT)pField->byLow );
}
else
{
/* must be bigger */
wsprintf(szBuf, TEXT("%d"), (int)(UINT)pField->byHigh );
}
SendMessage(pField->hWnd, WM_SETTEXT, 0, (LPARAM) (LPSTR) szBuf);
if ((hControlWnd = GetParent(pField->hWnd)) != NULL
&& (hDialog = GetParent(hControlWnd)) != NULL)
{
pControl->fInMessageBox = TRUE;
IPAlertPrintf(hDialog, IDS_IPBAD_FIELD_VALUE, i,
pField->byLow, pField->byHigh);
pControl->fInMessageBox = FALSE;
SendMessage(pField->hWnd, EM_SETSEL, 0, CHARS_PER_FIELD);
return FALSE;
}
}
}
if ((hControlWnd = GetParent(pField->hWnd)))
{
if (IPADDR_GET_SUBSTYLE(hControlWnd) & IPADDR_EX_STYLE_CONTIGUOUS)
IpAddrMakeContiguous(hControlWnd);
}
return TRUE;
}
/*
Get the value stored in a field.
call
pField = pointer to the FIELD structure for the field.
returns
The value (0..255) or -1 if the field has not value.
*/
int GetFieldValue(FIELD *pField)
{
WORD wLength;
static TCHAR szBuf[CHARS_PER_FIELD+1];
INT i,j;
//*(WORD *)szBuf = sizeof(szBuf) - 1;
//wLength = (WORD)SendMessage(pField->hWnd,EM_GETLINE,0,(DWORD)(LPSTR)szBuf);
wLength = (WORD)SendMessage(pField->hWnd,WM_GETTEXT,(sizeof(szBuf) / sizeof(*szBuf)),(LPARAM)(LPSTR)szBuf);
if (wLength != 0)
{
szBuf[wLength] = TEXT('\0');
if (!lstrcmp(szBuf, SZWILDCARD))
{
return 255;
}
for (j=0,i=0;j<(INT)wLength;j++)
{
i=i*10+szBuf[j]-TEXT('0');
}
return i;
}
else
return -1;
}
/*
IPAlertPrintf() - Does a printf to a message box.
*/
int FAR CDECL IPAlertPrintf(HWND hwndParent, UINT ids, int iCurrent, int iLow, int iHigh)
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
// Is this large enough?
static TCHAR szBuf[MAX_IPRES*2];
static TCHAR szFormat[MAX_IPRES];
TCHAR * psz;
if (ids != IDS_IPNOMEM &&
//
// Why OEM?
//
//IPLoadOem(s_hLibInstance, ids, szFormat, sizeof(szFormat)))
LoadString(AfxGetResourceHandle(), ids, szFormat, sizeof(szFormat)/sizeof(*szFormat)))
{
wsprintf(szBuf, szFormat, iCurrent, iLow, iHigh);
psz = szBuf;
}
else
{
psz = szNoMem;
}
MessageBeep(MB_ICONEXCLAMATION);
return MessageBox(hwndParent, psz, szCaption, MB_ICONEXCLAMATION);
}
/*
Load an OEM string and convert it to ANSI.
call
hInst = This instance
idResource = The ID of the string to load
lpszBuffer = Pointer to buffer to load string into.
cbBuffer = Length of the buffer.
returns
TRUE if the string is loaded, FALSE if it is not.
*/
BOOL IPLoadOem(HINSTANCE hInst, UINT idResource, TCHAR* lpszBuffer, int cbBuffer)
{
if (LoadString(hInst, idResource, lpszBuffer, cbBuffer))
{
//OemToAnsi(lpszBuffer, lpszBuffer);
return TRUE;
}
else
{
lpszBuffer[0] = 0;
return FALSE;
}
}
__declspec(dllexport) WCHAR * WINAPI
inet_ntoaw(
struct in_addr dwAddress
) {
static WCHAR szAddress[16];
char* pAddr = inet_ntoa(*(struct in_addr *) &dwAddress);
if (pAddr)
{
// mbstowcs(szAddress, inet_ntoa(*(struct in_addr *)&dwAddress), 16);
MultiByteToWideChar(CP_ACP, 0, pAddr, -1, szAddress, 16);
return szAddress;
}
else
return NULL;
}
__declspec(dllexport) DWORD WINAPI
inet_addrw(
LPCWSTR szAddressW
) {
CHAR szAddressA[16];
wcstombs(szAddressA, szAddressW, 16);
return inet_addr(szAddressA);
}