/**************************************************************************\ *** SCICALC Scientific Calculator for Windows 3.00.12 *** By Kraig Brockschmidt, Microsoft Co-op, Contractor, 1988-1989 *** (c)1989 Microsoft Corporation. All Rights Reserved. *** *** scimain.c *** *** Definitions of all globals, WinMain procedure *** *** Last modification *** Fri 22-Nov-1996 *** *** -by- Jonathan Parati. [jonpa] 22-Nov-1996 *** Converted Calc from floating point to infinite precision. *** The new math engine is in ..\ratpak *** *** *** -by- Amit Chatterjee. [amitc] 05-Jan-1990. *** Calc did not have a floating point exception signal handler. This *** would cause CALC to be forced to exit on a FP exception as that's *** the default. *** The signal handler is defined in SCIFUNC.C, in WinMain we hook the *** the signal. \**************************************************************************/ #include "scicalc.h" #include "calchelp.h" #include "signal.h" #include "unifunc.h" #include "input.h" #include "scidisp.h" #include "strsafe.h" #define BOOLIFY(x) ((x)?1:0) /**************************************************************************/ /*** Global variable declarations and initializations ***/ /**************************************************************************/ int nCalc=0; /* 0=Scientific, 1=Simple. */ BOOL gbUseSep=FALSE; /* display the number with a separator */ ANGLE_TYPE nDecMode=ANGLE_DEG; /* Holder for last used Deg/Rad/Grad mode. */ UINT gnDecGrouping=0x03; /* Holds the decimal digit grouping number */ int nHexMode=0; /* Holder for last used Dword/Word/Byte mode. */ int nTempCom=0, /* Holding place for the last command. */ nParNum=0, /* Number of parenthases. */ nOpCode=0, /* ID value of operation. */ nOp[25], /* Holding array for parenthasis operations. */ nPrecOp[25], /* Holding array for precedence operations. */ nPrecNum=0, /* Current number of precedence ops in holding. */ gcIntDigits; /* Number of digits allowed in the current base */ eNUMOBJ_FMT nFE = FMT_FLOAT; /* Scientific notation conversion flag. */ HWND g_hwndDlg=0, /* Global handle to main window. */ hEdit=0, /* Handle to Clibboard I/O edit control */ hStatBox=0, /* Global handle to statistics box. */ hListBox=0; /* Global handle for statistics list box. */ HMENU g_hHexMenu=NULL; // Global handle for hex menu HMENU g_hDecMenu=NULL; // Global handle for dec menu HANDLE hAccel; // Accelerator handle. HINSTANCE hInst; // Global instance. BOOL bHyp=FALSE, // Hyperbolic on/off flag. bInv=FALSE, // Inverse on/off flag. bError=FALSE, // Error flag. bColor=TRUE; // Flag indicating if color is available. HNUMOBJ ghnoNum=NULL, // Currently displayed number used everywhere. ghnoParNum[25], // Holding array for parenthasis values. ghnoPrecNum[25], // Holding array for precedence values. ghnoMem=NULL, // Current memory value. ghnoLastNum = NULL; // Number before operation (left operand). LONG nPrecision = 32, // number of digits to use in decimal mode nDecimalPlaces = 10, // number of decimal places to show nRadix=10, // the current base (2, 8, 10, or 16) dwWordBitWidth = 64; // # of bits in currently selected word size BOOL g_fHighContrast = FALSE; // Are we in High Contrast mode? HNUMOBJ g_ahnoChopNumbers[4]; // word size inforcement BOOL bFarEast; // true if we need to use Far East localization #ifdef USE_MIRRORING BOOL g_fLayoutRTL = FALSE; #endif extern CALCINPUTOBJ gcio; extern BOOL gbRecord; /* DO NOT LOCALIZE THESE STRINGS. */ TCHAR szAppName[10]=TEXT("SciCalc"), /* Application name. */ szDec[5]=TEXT("."), /* Default decimal character */ gszSep[5]=TEXT(","), /* Default thousand seperator */ szBlank[6]=TEXT(" "); /* Blank space. */ LPTSTR gpszNum = NULL; int gcchNum = 0; static TCHAR szInitNum[] = TEXT("0"); // text to init gpszNum with /* END WARNING */ /* rgpsz[] is an array of pointers to strings in a locally allocated */ /* memory block. This block is fixed such that LocalLock does not need */ /* to be called to use a string. */ TCHAR *rgpsz[CSTRINGS]; RECT rcDeg[6]; void ParseCmdLine( LPSTR pszCmdA ); BOOL InitializeWindowClass( HINSTANCE hPrevInstance ); void InitialOneTimeOnlySetup(); void EverythingResettingNumberSetup(); extern WNDPROC fpOrgDispEditProc; LRESULT CALLBACK SubDispEditProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam); /**************************************************************************/ /*** Main Window Procedure. ***/ /*** ***/ /*** Important functions: ***/ /*** 1) Gets text dimensions and sets conversion units correctly. ***/ /*** ***/ /*** 2) Checks the display device driver for color capability. ***/ /*** If only 2 colors are available (mono, cga), bColor is ***/ /*** set to FALSE, and the background brush is gray. If ***/ /*** color is available, the background brush colors are read ***/ /*** from WIN.INI and the brush is created. ***/ /*** ***/ /*** 3) Window and hidden edit control are created. ***/ /*** ***/ /*** 4) Contains message loop and deletes the brushes used. ***/ /*** ***/ /**************************************************************************/ int APIENTRY WinMain( HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { MSG msg; INT nx; LPTSTR psz; int cch = 0, cchTotal = 0; TCHAR szTempString[100] = {0}; #ifdef USE_MIRRORING DWORD dwLayout; #endif // A bunch of sanity checks to ensure nobody is violating any of the // bazillion // assumptions calc makes about the order of controls. Of course these // asserts // wouldn't prevent a really dedicated person from messing things up but they // should help guide a rational person who might not be aware of calc's // idiosyncrasies. // Anyone who modifies the resource file should hit these asserts which // will then // alert them to the consequences of their actions. // IDC_0 to IDC_F must be in sequential increasing order ASSERT( 15 == (IDC_F - IDC_0) ); // Binary operators IDC_AND through IDC_PWR must be in order ASSERT( (95-86) == (IDC_PWR - IDC_AND) ); // Unary operators IDC_CHOP through IDC_EQU must be in order ASSERT( (112-96) == (IDC_EQU - IDC_CHOP) ); // menu item id's must be in order ASSERT( 5 == (IDM_LASTMENU - IDM_FIRSTMENU) ); #ifdef USE_MIRRORING if (GetProcessDefaultLayout(&dwLayout) && (dwLayout & LAYOUT_RTL)) { SetProcessDefaultLayout(dwLayout & ~LAYOUT_RTL); g_fLayoutRTL = TRUE; } #endif ParseCmdLine( lpCmdLine ); hInst = hInstance; if ( !InitializeWindowClass( hPrevInstance ) ) return FALSE; // Read strings for keys, errors, trig types, etc. // These will be copied from the resources to local memory. A larger // than needed block is allocated first and then reallocated once we // know how much is actually used. try { psz = (LPTSTR) LocalAlloc(LPTR, ByteCountOf(CCHSTRINGSMAX)); if (!psz) throw; int cchResourceBuffer = CCHSTRINGSMAX, cchLeftInBuffer; // build up an offset array in rgpsz for (nx = 0; nx <= CSTRINGS; nx++) { INT_PTR iOffset; Retry: cchLeftInBuffer = cchResourceBuffer - cchTotal; cch = 1 + LoadString(hInstance, (UINT)(IDS_FIRSTKEY + nx), psz + cchTotal, cchLeftInBuffer); if (cch == (cchResourceBuffer - cchTotal)) // woops: buffer was too small { LPTSTR pszTmp = (LPTSTR)LocalReAlloc(psz, ByteCountOf(cchResourceBuffer + CCHSTRINGSMAX), LMEM_MOVEABLE); if (!pszTmp) throw; psz = pszTmp; cchResourceBuffer += CCHSTRINGSMAX; goto Retry; } iOffset = (INT_PTR)cchTotal; rgpsz[nx] = (LPTSTR)iOffset; // first pass is offset array cchTotal += cch; } LPTSTR pszTmp = (LPTSTR)LocalReAlloc(psz, ByteCountOf(cchTotal), LMEM_MOVEABLE); if (!pszTmp) throw; psz = pszTmp; // convert the array of offsets into an array of pointers for (nx = 0 ; nx <= CSTRINGS ; nx++) rgpsz[nx] = psz + (INT_PTR)rgpsz[nx]; } catch ( ... ) { if (psz) LocalFree(psz); if (LoadString(hInst, IDS_NOMEM, szTempString, CharSizeOf(szTempString))) { MessageBox((HWND) NULL, szTempString, NULL, MB_OK | MB_ICONHAND); } return FALSE; } // The display in calc isn't really an edit control so we use this edit // control to simplify cutting to the clipboard hEdit = CreateWindow( TEXT("EDIT"), TEXT("CalcMsgPumpWnd"), WS_OVERLAPPED | WS_VISIBLE, CW_USEDEFAULT,0,CW_USEDEFAULT,0, NULL, NULL, hInst, NULL ); // This initializes things that only need to be set up once, including a // call to ratpak so that ratpak can create any constants it needs InitialOneTimeOnlySetup(); // we store in the win.ini file our desired display mode, Scientific // or Standard nCalc = (INT)GetProfileInt(szAppName, TEXT("layout"), 1); if ((nCalc != 0) && (nCalc != 1)) { // Go to the default value in case of bad values nCalc = 1; } gbUseSep = (INT)GetProfileInt(szAppName, TEXT("UseSep"), 0); if ((gbUseSep != 0) && (gbUseSep != 1)) { // Go to the default value in case of bad values gbUseSep = 0; } // InitSciCalc creates a dialog based on what the value of nCalc is. // A handle to the window that is created is stored in g_hwndDlg InitSciCalc(TRUE); hAccel = LoadAccelerators(hInst, MAKEINTRESOURCE(IDA_ACCELTABLE)); while (GetMessage(&msg, NULL, 0, 0)) { if (!hStatBox || !IsDialogMessage(hStatBox, &msg)) { if ( ((msg.hwnd == g_hwndDlg)||IsChild(g_hwndDlg, msg.hwnd)) && TranslateAccelerator (g_hwndDlg, (HACCEL)hAccel, &msg)) continue; TranslateMessage(&msg); DispatchMessage(&msg); } } LocalFree(psz); return (DWORD)msg.wParam; } /**************************************************************************\ * * Command Line processing routines * * History * 22-Nov-1996 JonPa Wrote it * \**************************************************************************/ #define IsWhiteSpace( ch ) ((ch) == TEXT(' ') || (ch) == TEXT('\t')) #define IsDigit( ch ) ((ch) >= TEXT('0') && (ch) <= TEXT('9')) LPTSTR TtoL( LPTSTR psz, LONG *pl ) { LONG l = 0; while( IsDigit( *psz ) ) { l = l * 10 + (*psz - TEXT('0')); psz = CharNext( psz ); } *pl = l; return psz; } void ParseCmdLine( LPSTR pszCmdA ) { BOOL fInQuote; LPTSTR pszCmdT = GetCommandLine(); // parse cmd line // usage: -p:## -r:## -w:## -e -x -i // -e, -x, and -i currently do nothing. // Skip app name while( *pszCmdT && IsWhiteSpace( *pszCmdT )) { pszCmdT = CharNext( pszCmdT ); } fInQuote = FALSE; while( *pszCmdT && (fInQuote || !IsWhiteSpace(*pszCmdT)) ) { if (*pszCmdT == TEXT('\"')) fInQuote = !fInQuote; pszCmdT = CharNext( pszCmdT ); } while( *pszCmdT ) { switch( *pszCmdT ) { case TEXT('p'): case TEXT('P'): // -p:## precision pszCmdT = CharNext(pszCmdT); // Skip ':' and white space while( *pszCmdT && (*pszCmdT == TEXT(':') || IsWhiteSpace(*pszCmdT)) ) { pszCmdT = CharNext(pszCmdT); } pszCmdT = TtoL( pszCmdT, &nPrecision ); // a percision > C_NUM_MAX_DIGITS will allow a string too long for it's buffer if ( nPrecision > C_NUM_MAX_DIGITS) { ASSERT( nPrecision <= C_NUM_MAX_DIGITS ); nPrecision = C_NUM_MAX_DIGITS; } // NOTE: this code assumes there MUST be a space after the number break; case TEXT('r'): case TEXT('R'): // -r:## Radix pszCmdT = CharNext(pszCmdT); // Skip ':' and white space while( *pszCmdT && (*pszCmdT == TEXT(':') || IsWhiteSpace(*pszCmdT)) ) { pszCmdT = CharNext(pszCmdT); } pszCmdT = TtoL( pszCmdT, &nRadix ); // since the UI only has 16 keys for digit input, we only allow upto base 16 if (nRadix > 16) { ASSERT( nRadix <= 16 ); nRadix = 16; } else if (nRadix < 2) // you know some fool would try for base zero if you let them { ASSERT( nRadix >= 2 ); nRadix = 2; } // NOTE: this code assumes there MUST be a space after the number break; case TEXT('e'): case TEXT('E'): // -e extended mode break; case TEXT('w'): case TEXT('W'): // -w:## Word size in bits pszCmdT = CharNext(pszCmdT); // Skip ':' and white space while( *pszCmdT && (*pszCmdT == TEXT(':') || IsWhiteSpace(*pszCmdT)) ) { pszCmdT = CharNext(pszCmdT); } // Set bit count pszCmdT = TtoL( pszCmdT, &dwWordBitWidth ); // NOTE: this code assumes there MUST be a space after the number break; } pszCmdT = CharNext( pszCmdT ); } } ////////////////////////////////////////////////// // // InitalizeWindowClass // ////////////////////////////////////////////////// BOOL InitializeWindowClass( HINSTANCE hPrevInstance ) { WNDCLASSEX wndclass; if (!hPrevInstance) { wndclass.cbSize = sizeof(wndclass); wndclass.style = 0; wndclass.lpfnWndProc = CalcWndProc; wndclass.cbClsExtra = 0; wndclass.cbWndExtra = DLGWINDOWEXTRA; wndclass.hInstance = hInst; wndclass.hIcon = LoadIcon(hInst, TEXT("SC")); wndclass.hCursor = LoadCursor (NULL, IDC_ARROW); wndclass.hbrBackground = GetSysColorBrush(COLOR_3DFACE); wndclass.lpszMenuName = MAKEINTRESOURCE(IDM_CALCMENU); wndclass.lpszClassName = szAppName; wndclass.hIconSm = NULL; if (!RegisterClassEx(&wndclass)) return FALSE; } return TRUE; } ////////////////////////////////////////////////// // // InitialOneTimeOnlyNumberSetup // ////////////////////////////////////////////////// void InitialOneTimeOnlySetup() { // Initialize the decimal input code. This ends up getting called twice // but it's quick so that shouldn't be a problem. Needs to be done before // SetRadix is called. CIO_vClear( &gcio ); gbRecord = TRUE; // we must now setup all the ratpak constants and our arrayed pointers // to these constants. BaseOrPrecisionChanged(); // these rat numbers are set only once and then never change regardless of // base or precision changes g_ahnoChopNumbers[0] = rat_qword; g_ahnoChopNumbers[1] = rat_dword; g_ahnoChopNumbers[2] = rat_word; g_ahnoChopNumbers[3] = rat_byte; // we can't call this until after we have set the radix (and thus called // ChangeConstants) so we do it last. EverythingResettingNumberSetup(); NumObjAssign( &ghnoMem, HNO_ZERO ); } ////////////////////////////////////////////////// // // EverythingResettingNumberSetup // ////////////////////////////////////////////////// void EverythingResettingNumberSetup() { int i; // Initialize the decimal input code. CIO_vClear( &gcio ); gbRecord = TRUE; NumObjAssign( &ghnoNum, HNO_ZERO ); NumObjAssign( &ghnoLastNum, HNO_ZERO ); // REVIEW: is it just me, or do we speew major memory wheneven this method // executes? // array used to handle ( and ) for( i = 0; i < ARRAYSIZE(ghnoParNum); i++ ) ghnoParNum[i] = NULL; // array used to handle order of operations for( i = 0; i < ARRAYSIZE(ghnoPrecNum); i++ ) ghnoPrecNum[i] = NULL; int cbNum = sizeof(szInitNum); gpszNum = (LPTSTR)NumObjAllocMem( cbNum ); if (gpszNum) { gcchNum = cbNum / sizeof(TCHAR); StringCchCopy(gpszNum, gcchNum, szInitNum); } } ////////////////////////////////////////////////// // // InitSciCalc // ////////////////////////////////////////////////// VOID APIENTRY InitSciCalc(BOOL bViewChange) { TCHAR chLastDec; TCHAR chLastSep; int nLastSepLen; UINT nLastDecGrouping; HMENU hMenu; HWND hDispEdit; BOOL bRepaint=FALSE; RECT rect = {0,0,0,0}; TCHAR szGrouping[32]; EverythingResettingNumberSetup(); // when we switch modes, we need to remind the ui that we are no longer // inputing the number we were inputting before we switched modes. gbRecord = FALSE; // REVIEW: This should not be needed with the new initialization chLastDec = szDec[0]; chLastSep = gszSep[0]; nLastDecGrouping=gnDecGrouping; // SECURITY: szDec can have any value you want, but only the first letter is used. GetProfileString(TEXT("intl"), TEXT("sDecimal"), TEXT("."), szDec, CharSizeOf(szDec)); if (szDec[0] == 0) { szDec[0] = L'.'; } // SECURITY: gszSep can have any value you want, but only the first letter is used. GetProfileString(TEXT("intl"), TEXT("sThousand"), TEXT(","), gszSep, CharSizeOf(gszSep)); if (gszSep[0] == 0) { gszSep[0] = L','; } ZeroMemory(szGrouping, sizeof(szGrouping)); // SECURITY: DigitGroupingStringToGroupingNum is responsible for handling all failure cases GetProfileString(TEXT("intl"), TEXT("sGrouping"), TEXT("3;0"), szGrouping, CharSizeOf(szGrouping)); gnDecGrouping=DigitGroupingStringToGroupingNum(szGrouping); // if the grouping pattern changed we always do the following things if (gnDecGrouping != nLastDecGrouping) { nLastDecGrouping=gnDecGrouping; bRepaint=TRUE; } // if the thousands symbol has changed we always do the following things if ( gszSep[0] != chLastSep ) { chLastSep = gszSep[0]; bRepaint = TRUE; } // if the decimal symbol has changed we always do the following things if ( szDec[0] != chLastDec ) { chLastDec = szDec[0]; // Re-initialize input string's decimal point. CIO_vUpdateDecimalSymbol(&gcio, chLastDec); // put the new decimal symbol into the table used to draw the decimal // key *(rgpsz[IDS_DECIMAL]) = chLastDec; // we need to redraw to update the decimal point button bRepaint = TRUE; } { HIGHCONTRAST hc; hc.cbSize = sizeof(hc); if ( SystemParametersInfo(SPI_GETHIGHCONTRAST, sizeof(hc), &hc, 0) ) { if ( BOOLIFY(hc.dwFlags & HCF_HIGHCONTRASTON) != g_fHighContrast ) { g_fHighContrast = BOOLIFY(hc.dwFlags & HCF_HIGHCONTRASTON); bRepaint = TRUE; } } } if ( bViewChange ) { BOOL bUseOldPos = FALSE; // if we are changing views we destory the old window and create // a new window if ( g_hwndDlg ) { SetMenu(g_hwndDlg, g_hDecMenu); bUseOldPos = TRUE; GetWindowRect( g_hwndDlg, &rect ); DestroyWindow( g_hwndDlg ); DestroyMenu(g_hHexMenu); g_hHexMenu=NULL; } // create the correct window for the mode we're currently in if ( nCalc ) { // switch to standard mode g_hwndDlg = CreateDialog(hInst, MAKEINTRESOURCE(IDD_STANDARD), 0, NULL); g_hDecMenu=GetMenu(g_hwndDlg); #ifdef USE_MIRRORING if (g_fLayoutRTL) { SetWindowLong(g_hwndDlg, GWL_EXSTYLE, GetWindowLong(g_hwndDlg,GWL_EXSTYLE) | \ WS_EX_LAYOUTRTL | WS_EX_NOINHERITLAYOUT); } #endif } else { // switch to scientific mode g_hwndDlg = CreateDialog(hInst, MAKEINTRESOURCE(IDD_SCIENTIFIC), 0, NULL); g_hDecMenu=GetMenu(g_hwndDlg); g_hHexMenu=LoadMenu(hInst, MAKEINTRESOURCE(IDM_HEXCALCMENU)); #ifdef USE_MIRRORING if (g_fLayoutRTL) { SetWindowLong(g_hwndDlg, GWL_EXSTYLE, GetWindowLong(g_hwndDlg,GWL_EXSTYLE) | WS_EX_LAYOUTRTL | WS_EX_NOINHERITLAYOUT); } #endif // Stat box is initially off, disable stat buttons. for ( int iID = IDC_AVE; iID <= IDC_DATA; iID++ ) EnableWindow( GetDlgItem( g_hwndDlg, iID ), FALSE ); SwitchModes(10, nDecMode, nHexMode); // If precision won't fit in display, then resize it if (nPrecision > 32) { HWND hwndDisplay; RECT rc, rcMain; hwndDisplay=GetDlgItem( g_hwndDlg, IDC_DISPLAY ); GetWindowRect( hwndDisplay, &rc ); GetClientRect( g_hwndDlg, &rcMain ); MapWindowPoints( g_hwndDlg, NULL, (LPPOINT)&rcMain, 2); rc.left = rcMain.left + (rcMain.right - rc.right); OffsetRect( &rc, -(rcMain.left), -(rcMain.top) ); SetWindowPos(hwndDisplay, NULL, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, SWP_NOACTIVATE | SWP_NOZORDER ); } } if (hDispEdit = GetDlgItem(g_hwndDlg, IDC_DISPLAY)) { // subclass the Edit Control hide caret and filter out mouse msg fpOrgDispEditProc = (WNDPROC)GetWindowLongPtr(hDispEdit, GWLP_WNDPROC); if (fpOrgDispEditProc) SetWindowLongPtr(hDispEdit, GWLP_WNDPROC, (LONG_PTR)(WNDPROC)SubDispEditProc); } // keep calc in the same place it was before if ( bUseOldPos ) { SetWindowPos( g_hwndDlg, NULL, rect.left, rect.top, 0,0, SWP_NOZORDER | SWP_NOSIZE ); } // ensure the menu items for Scientific and Standard are set correctly CheckMenuRadioItem(g_hDecMenu, IDM_SC, IDM_SSC, (nCalc == 0 ? IDM_SC : IDM_SSC), MF_BYCOMMAND); CheckMenuItem(g_hDecMenu, IDM_USE_SEPARATOR, MF_BYCOMMAND | (gbUseSep ? MF_CHECKED : MF_UNCHECKED)); if (g_hHexMenu) { CheckMenuRadioItem(g_hHexMenu, IDM_SC, IDM_SSC, (nCalc == 0 ? IDM_SC : IDM_SSC), MF_BYCOMMAND); CheckMenuItem(g_hHexMenu, IDM_USE_SEPARATOR, MF_BYCOMMAND | (gbUseSep ? MF_CHECKED:MF_UNCHECKED)); } // To ensure that the call to SetRadix correctly update the active // state of the buttons on // SciCalc we must tell it to forget the previous Radix { extern long oldRadix; oldRadix = (unsigned)-1; } // this will set the correct buttons on the UI SetRadix(10); SetDlgItemText(g_hwndDlg, IDC_MEMTEXT, NumObjIsZero(ghnoMem) ? (szBlank) : (TEXT(" M")) ); SendMessage(g_hwndDlg, WM_CHANGEUISTATE, MAKEWPARAM(UIS_INITIALIZE, 0), 0); ShowWindow( g_hwndDlg, SW_SHOW ); UpdateWindow(g_hwndDlg); } // END if ( bViewChanged ) else if ( bRepaint ) { // no need to repaint if we just changed views InvalidateRect( g_hwndDlg, NULL, TRUE ); } }