1702 lines
56 KiB
C++
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);
|
|
}
|
|
|