/* * $Id$ */ /* * Harbour Project source code: * ALERT() function * * Released to Public Domain by Vladimir Kazimirchik * www - http://www.harbour-project.org * */ /* * The following parts are Copyright of the individual authors. * www - http://www.harbour-project.org * * Copyright 1999 Victor Szel * Changes for higher Clipper compatibility, console mode, extensions * __NONOALERT() * * Copyright 1999 Chen Kedem * Documentation * * See doc/license.txt for licensing terms. * */ #include "hbsetup.ch" #include "box.ch" #include "common.ch" #include "inkey.ch" #include "setcurs.ch" // ; TOFIX: Clipper defines a clipped window for Alert() // ; Clipper will return NIL if the first parameter is not a string, but // this is not documented. This implementation converts the first parameter // to a string if another type was passed. You can switch back to // Clipper compatible mode by defining constant // HARBOUR_STRICT_CLIPPER_COMPATIBILITY. // ; Clipper handles these buttons { "Ok", "", "Cancel" } in a buggy way. // This is fixed. // ; nDelay parameter is a Harbour addition. STATIC s_lNoAlert := NIL /* $DOC$ * $FUNCNAME$ * ALERT() * $CATEGORY$ * Data input and output * $ONELINER$ * Display a dialog box with a message * $SYNTAX$ * ALERT( , [], [], * [] ) --> nChoice or NIL * $ARGUMENTS$ * Message to display in the dialog box. can be * of any Harbour type. * If is an array of Character strings, each element would * be displayed in a new line. is converted to Character * string, you could split the message to several lines by placing * semicolon (;) in the desired places. * * Array with available response. Each element should be * Character string. If omitted, default is { "Ok" }. * * Color string to paint the dialog box with. * If omitted, default color is "W+/R". * * Number of seconds to wait to user response before abort. * Default value is 0, that wait forever. * $RETURNS$ * ALERT() return Numeric value representing option number chosen. * If ESC was pressed, return value is zero. The return value is NIL * if ALERT() is called with no parameters, or if type is * not Character and HARBOUR_STRICT_CLIPPER_COMPATIBILITY option was * used. If seconds had passed without user response, the * return value is 1. * $DESCRIPTION$ * ALERT() display simple dialog box on screen and let the user select * one option. The user can move the highlight bar using arrow keys or * TAB key. To select an option the user can press ENTER, SPACE or the * first letter of the option. * * If the program is executed with the //NOALERT command line switch, * nothing is displayed and it simply returns NIL. This switch could * be overridden with __NONOALERT(). * * If the GT system is linked in, ALERT() display the message using * the full screen I/O system, if not, the information is printed to * the standard output using OUTSTD(). * $EXAMPLES$ * LOCAL cMessage, aOptions, nChoice * * // harmless message * cMessage := "Major Database Corruption Detected!;" + ; * "(deadline in few hours);;" + ; * "where DO you want to go today?" * * // define response option * aOptions := { "Ok", "www.jobs.com", "Oops" } * * // show message and let end user select panic level * nChoice := ALERT( cMessage, aOptions ) * DO CASE * CASE nChoice == 0 * // do nothing, blame it on some one else * CASE nChoice == 1 * ? "Please call home and tell them you're gonn'a be late" * CASE nChoice == 2 * // make sure your resume is up to date * CASE nChoice == 3 * ? "Oops mode is not working in this version" * ENDCASE * $TESTS$ * $STATUS$ * $COMPLIANCE$ * This function is sensitive to HARBOUR_STRICT_CLIPPER_COMPATIBILITY * settings. * * ON: accept Character values only and return NIL if other * types are passed, * OFF: could be any type, and internally converted to * Character string. If type is Array, multi-line message is * displayed. * * ON: Only the first four valid are taken. * OFF: could contain as many as needed options. * * is an Harbour extension, or at least un-documented * in Clipper 5.2 NG. * * is an Harbour extension. * $SEEALSO$ * @...PROMPT, MENU TO, STDOUT(), __NONOALERT() * $END$ */ FUNCTION Alert( xMessage, aOptions, cColorNorm, nDelay ) LOCAL nChoice LOCAL aSay, nPos, nWidth, nOpWidth, nInitRow, nInitCol, nEval LOCAL nKey, aPos, nCurrent, aHotkey, aOptionsOK LOCAL cColorHigh LOCAL nOldRow LOCAL nOldCol LOCAL nOldCursor LOCAL cOldScreen LOCAL nOldDispCount LOCAL nCount /* TOFIX: Clipper decides at runtime, whether the GT is linked in, */ /* if it is not, the console mode is choosen here */ LOCAL lConsole := .F. DEFAULT s_lNoAlert TO __argCheck( "NOALERT" ) IF s_lNoAlert RETURN NIL ENDIF aSay := {} #ifdef HARBOUR_STRICT_CLIPPER_COMPATIBILITY IF !ISCHARACTER( xMessage ) RETURN NIL ENDIF DO WHILE ( nPos := At( ';', xMessage ) ) != 0 AAdd( aSay, Left( xMessage, nPos - 1 ) ) xMessage := SubStr( xMessage, nPos + 1 ) ENDDO AAdd( aSay, xMessage ) #else IF PCount() == 0 RETURN NIL ENDIF IF ISARRAY( xMessage ) FOR nEval := 1 TO Len( xMessage ) IF ISCHARACTER( xMessage[ nEval ] ) AAdd( aSay, xMessage[ nEval ] ) ENDIF NEXT ELSE DO CASE CASE ValType( xMessage ) $ "CM" /* Do nothing, just speed up things */ CASE ValType( xMessage ) == "N" ; xMessage := LTrim( Str( xMessage ) ) CASE ValType( xMessage ) == "D" ; xMessage := DToC( xMessage ) CASE ValType( xMessage ) == "L" ; xMessage := iif( xMessage, ".T.", ".F." ) CASE ValType( xMessage ) == "O" ; xMessage := xMessage:className + " Object" CASE ValType( xMessage ) == "B" ; xMessage := "{||...}" OTHERWISE ; xMessage := "NIL" ENDCASE DO WHILE ( nPos := At( ';', xMessage ) ) != 0 AAdd( aSay, Left( xMessage, nPos - 1 ) ) xMessage := SubStr( xMessage, nPos + 1 ) ENDDO AAdd( aSay, xMessage ) ENDIF #endif IF !ISARRAY( aOptions ) aOptions := {} ENDIF IF !ISCHARACTER( cColorNorm ) cColorNorm := "W+/R" cColorHigh := "W+/B" ELSE cColorHigh := StrTran( StrTran( iif( At( "/", cColorNorm ) == 0, "N", SubStr( cColorNorm, At( "/", cColorNorm ) + 1 ) ) + "/" +; iif( At( "/", cColorNorm ) == 0, cColorNorm, Left( cColorNorm, At( "/", cColorNorm ) - 1 ) ), "+", "" ), "*", "" ) ENDIF IF nDelay == NIL nDelay := 0 ENDIF /* The longest line */ nWidth := 0 AEval( aSay, {| x | nWidth := Max( Len( x ), nWidth ) } ) /* Cleanup the button array */ aOptionsOK := {} FOR nEval := 1 TO Len( aOptions ) IF ISCHARACTER( aOptions[ nEval ] ) .AND. !Empty( aOptions[ nEval ] ) AAdd( aOptionsOK, aOptions[ nEval ] ) ENDIF NEXT IF Len( aOptionsOK ) == 0 aOptionsOK := { 'Ok' } #ifdef HARBOUR_STRICT_CLIPPER_COMPATIBILITY // ; Clipper allows only four options ELSEIF Len( aOptionsOK ) > 4 aSize( aOptionsOK, 4 ) #endif ENDIF /* Total width of the botton line (the one with choices) */ nOpWidth := 0 AEval( aOptionsOK, {| x | nOpWidth += Len( x ) + 4 } ) /* what's wider ? */ nWidth := Max( nWidth + 2 + iif( Len( aSay ) == 1, 4, 0 ), nOpWidth + 2 ) /* box coordinates */ nInitRow := Int( ( ( MaxRow() - ( Len( aSay ) + 4 ) ) / 2 ) + .5 ) nInitCol := Int( ( ( MaxCol() - ( nWidth + 2 ) ) / 2 ) + .5 ) /* detect prompts positions */ aPos := {} aHotkey := {} nCurrent := nInitCol + Int( ( nWidth - nOpWidth ) / 2 ) + 2 AEval( aOptionsOK, {| x | AAdd( aPos, nCurrent ), AAdd( aHotKey, Upper( Left( x, 1 ) ) ), nCurrent += Len( x ) + 4 } ) nChoice := 1 IF lConsole FOR nEval := 1 TO Len( aSay ) OutStd( aSay[ nEval ] ) IF nEval < Len( aSay ) OutStd( hb_OSNewLine() ) ENDIF NEXT OutStd( " (" ) FOR nEval := 1 TO Len( aOptionsOK ) OutStd( aOptionsOK[ nEval ] ) IF nEval < Len( aOptionsOK ) OutStd( ", " ) ENDIF NEXT OutStd( ") " ) /* choice loop */ DO WHILE .T. nKey := Inkey( nDelay ) DO CASE CASE nKey == 0 EXIT CASE nKey == K_ESC nChoice := 0 EXIT CASE aScan( aHotkey, {| x | x == Upper( Chr( nKey ) ) } ) > 0 nChoice := aScan( aHotkey, {| x | x == Upper( Chr( nKey ) ) } ) EXIT ENDCASE ENDDO OutStd( Chr( nKey ) ) ELSE /* PreExt */ nCount := nOldDispCount := DispCount() DO WHILE nCount-- != 0 DispEnd() ENDDO /* save status */ nOldRow := Row() nOldCol := Col() nOldCursor := SetCursor( SC_NONE ) cOldScreen := SaveScreen( nInitRow, nInitCol, nInitRow + Len( aSay ) + 3, nInitCol + nWidth + 1 ) /* draw box */ DispBox( nInitRow, nInitCol, nInitRow + Len( aSay ) + 3, nInitCol + nWidth + 1, B_SINGLE + ' ', cColorNorm ) FOR nEval := 1 TO Len( aSay ) DispOutAt( nInitRow + nEval, nInitCol + 1 + Int( ( ( nWidth - Len( aSay[ nEval ] ) ) / 2 ) + .5 ), aSay[ nEval ], cColorNorm ) NEXT /* choice loop */ DO WHILE .T. FOR nEval := 1 TO Len( aOptionsOK ) DispOutAt( nInitRow + Len( aSay ) + 2, aPos[ nEval ], " " + aOptionsOK[ nEval ] + " ",; iif( nEval == nChoice, cColorHigh, cColorNorm ) ) NEXT nKey := Inkey( nDelay ) DO CASE CASE nKey == K_ENTER .OR. ; nKey == K_SPACE .OR. ; nKey == 0 EXIT CASE nKey == K_ESC nChoice := 0 EXIT CASE ( nKey == K_LEFT .OR. nKey == K_SH_TAB ) .AND. Len( aOptionsOK ) > 1 nChoice-- IF nChoice == 0 nChoice := Len( aOptionsOK ) ENDIF nDelay := 0 CASE ( nKey == K_RIGHT .OR. nKey == K_TAB ) .AND. Len( aOptionsOK ) > 1 nChoice++ IF nChoice > Len( aOptionsOK ) nChoice := 1 ENDIF nDelay := 0 CASE aScan( aHotkey, {| x | x == Upper( Chr( nKey ) ) } ) > 0 nChoice := aScan( aHotkey, {| x | x == Upper( Chr( nKey ) ) } ) EXIT ENDCASE ENDDO /* Restore status */ RestScreen( nInitRow, nInitCol, nInitRow + Len( aSay ) + 3, nInitCol + nWidth + 1, cOldScreen ) SetCursor( nOldCursor ) SetPos( nOldRow, nOldCol ) /* PostExt */ DO WHILE nOldDispCount-- != 0 DispBegin() ENDDO ENDIF RETURN nChoice /* $DOC$ * $FUNCNAME$ * __NONOALERT() * $CATEGORY$ * Data input and output * $ONELINER$ * Override //NOALERT command line switch * $SYNTAX$ * __NONOALERT() --> NIL * $ARGUMENTS$ * This function take no arguments. * $RETURNS$ * __NONOALERT() always return NIL. * $DESCRIPTION$ * The //NOALERT command line switch cause Clipper to ignore calls to * the ALERT() function, this function override this behavior * and always display ALERT() dialog box. * $EXAMPLES$ * // make sure alert are been displayed * __NONOALERT() * $TESTS$ * $STATUS$ * $COMPLIANCE$ * __NONOALERT() is an Undocumented CA-Clipper function * $SEEALSO$ * $END$ */ PROCEDURE __NONOALERT() s_lNoAlert := .F. RETURN