* harbour/include/hbapi.h
* harbour/source/rtl/console.c
* changed hb_conOutAlt() from static to global function.
* harbour/include/hbapi.h
* harbour/source/common/hbver.c
+ added hb_verBuildDate()
* harbour/source/rtl/accept.c
! Fixed ACCEPT to respect SET CONSOLE and similar sets.
* harbour/source/rtl/hbgtcore.c
* harbour/contrib/libct/ctwin.c
! fixed chr(8) console output - it should erase character on the screen
* harbour/include/set.ch
* harbour/include/hbset.h
* harbour/source/rtl/set.c
* harbour/source/rtl/filesys.c
+ added _SET_TRIMFILENAME - when enabled low level hb_fs*() functions
strip trailing and leading spaces from file names to emulate DOS
like behavior - switch compatible with xHarbour
* harbour/source/rtl/run.c
* remove compiler type checking - if system() is not supported by
some platform/compiler then I'd prefer to exclude it explicitly.
* harbour/source/rtl/dircmd.prg
+ added support for extended DBF types and replaced some of
Bin2W() by ASC()
* harbour/source/rtl/defpath.c
* use OS_HAS_DRIVE_LETTER macro to detect if platform supports drive
letters
* harbour/source/rtl/philes.c
+ added HB_FCOMMIT(), HB_OSERROR(), HB_OSDRIVESEPARATOR()
Question: why we have HB_F_EOF() instead HB_FEOF()
* harbour/source/rtl/oldbox.c
* harbour/source/rtl/box.c
! fixed __BOX() to be Clipper compatible
* harbour/source/rtl/math.c
* harbour/source/rtl/dateshb.c
* formatting and some minor improvements
* harbour/source/rtl/isprint.c
* harbour/source/vm/itemapi.c
* harbour/source/rtl/ampm.c
* harbour/source/rtl/inkey.c
* harbour/source/rtl/gete.c
* harbour/source/rtl/fkmax.c
* harbour/source/rtl/langapi.c
* harbour/source/rtl/colorind.c
* harbour/source/rtl/mouseapi.c
* harbour/source/rtl/readvar.prg
* harbour/source/rtl/devoutp.prg
* harbour/source/rtl/readkey.prg
* code checking and formatting
! some minor fixes
% some speed improvements
* harbour/source/rtl/menuto.prg
* harbour/source/rtl/radiogrp.prg
* harbour/source/rtl/listbox.prg
* harbour/source/rtl/checkbox.prg
* harbour/source/rtl/pushbtn.prg
* harbour/source/rtl/radiobtn.prg
* code checking and formatting
! added fixes borrowed from xHarbour
! some other fixes
% some speed improvements
* harbour/source/rtl/filehb.c
+ added commment
* harbour/source/rtl/transfrm.c
! fixed integer numbers transformation when _SET_FIXED is on to
be Clipper compatible
* harbour/source/rtl/version.c
+ added HB_PCODEVER() and HB_BUILDDATE()
* harbour/source/rtl/copyfile.c
! fixed __COPYFILE() - source and destination files should respect
_SET_DEFAULT
286 lines
6.5 KiB
Plaintext
286 lines
6.5 KiB
Plaintext
/*
|
|
* $Id$
|
|
*/
|
|
|
|
/*
|
|
* Harbour Project source code:
|
|
* PROMPT/MENU TO commands
|
|
*
|
|
* Released to Public Domain by Phil Barnett <philb@iag.net>
|
|
* www - http://www.harbour-project.org
|
|
*
|
|
*/
|
|
|
|
/* NOTE: Recursive use is supported. */
|
|
|
|
#include "color.ch"
|
|
#include "common.ch"
|
|
#include "inkey.ch"
|
|
#include "hbmemvar.ch"
|
|
#include "setcurs.ch"
|
|
#xtranslate COLORARRAY(<x>) => &( '{"' + strtran(<x>, ',', '","') + '"}' )
|
|
|
|
STATIC s_aLevel := {}
|
|
STATIC s_nPointer := 1
|
|
|
|
FUNCTION __AtPrompt( nRow, nCol, cPrompt, cMsg, cColor )
|
|
|
|
if s_nPointer < 1
|
|
s_nPointer := 1
|
|
endif
|
|
|
|
// add the current level empty array.
|
|
do while len( s_aLevel ) < s_nPointer
|
|
aadd( s_aLevel, {} )
|
|
enddo
|
|
|
|
// add to the static array
|
|
aadd( s_aLevel[ s_nPointer ], { nRow, nCol, cPrompt, cMsg, cColor} )
|
|
|
|
// put this prompt on the screen right now
|
|
DispOutAt( nRow, nCol, cPrompt, cColor )
|
|
|
|
RETURN .f.
|
|
|
|
FUNCTION __MenuTo( bBlock, cVariable )
|
|
|
|
local nKey
|
|
local y
|
|
local q
|
|
local n
|
|
local lExit
|
|
local nArrLen
|
|
local xMsg
|
|
local nMsgCol
|
|
local nMsgRow
|
|
local lMsgCenter
|
|
local nSaveCursor
|
|
local cSaveReadVar
|
|
|
|
local lDeclared
|
|
local bAction
|
|
local nMouseClik
|
|
|
|
local nPointer
|
|
Local aColor
|
|
local cBackColor
|
|
local cFrontColor
|
|
|
|
// Detect if a memvar was passed
|
|
lDeclared := !__mvEXIST( cVariable )
|
|
if lDeclared
|
|
__mvPUBLIC( cVariable )
|
|
endif
|
|
|
|
n := eval( bBlock )
|
|
|
|
// if no prompts were defined, exit with 0
|
|
|
|
if s_nPointer < 1 .or. s_nPointer > len( s_aLevel )
|
|
|
|
n := 0
|
|
|
|
else
|
|
|
|
s_nPointer ++
|
|
nPointer := s_nPointer
|
|
|
|
nArrLen := len( s_aLevel[ nPointer - 1 ] )
|
|
|
|
// put choice in a valid range
|
|
|
|
if !ISNUMBER( n ) .OR. n < 1
|
|
n := 1
|
|
endif
|
|
|
|
if n > nArrLen
|
|
n := nArrLen
|
|
endif
|
|
|
|
//
|
|
|
|
nSaveCursor := setcursor( iif( Set( _SET_INTENSITY ), SC_NONE, NIL ) )
|
|
cSaveReadVar := ReadVar( upper( cVariable ) )
|
|
xMsg := ""
|
|
nMsgCol := 0
|
|
nMsgRow := set( _SET_MESSAGE )
|
|
lMsgCenter := set( _SET_MCENTER )
|
|
lExit := .F.
|
|
|
|
|
|
do while n <> 0
|
|
|
|
// should we display messages?
|
|
if nMsgRow > 0
|
|
|
|
if ! Empty( xMsg )
|
|
DispOutAt( nMsgRow, nMsgCol, Space( Len( xMsg ) ) )
|
|
endif
|
|
|
|
xMsg := s_aLevel[ nPointer - 1, n, 4 ]
|
|
|
|
// Code Block messages ( yes, they are documented! )
|
|
if ISBLOCK( xMsg )
|
|
xMsg := eval( xMsg )
|
|
endif
|
|
|
|
if !ISCHARACTER( xMsg )
|
|
xMsg := ""
|
|
endif
|
|
|
|
if lMsgCenter
|
|
nMsgCol := int( ( maxcol() - len( xMsg ) ) / 2 )
|
|
endif
|
|
|
|
DispOutAt( nMsgRow, nMsgCol, xMsg )
|
|
|
|
endif
|
|
|
|
// save the current row
|
|
q := n
|
|
|
|
if s_aLevel[ s_nPointer - 1, n, 5 ] <> nil
|
|
aColor := COLORARRAY( s_aLevel[ s_nPointer - 1, n, 5 ] )
|
|
cFrontColor := IIF( EMPTY( aColor[ 1 ] ) , NIL , aColor[ 1 ] )
|
|
cBackColor := IIF( LEN( aColor ) > 1 , aColor[2], NIL )
|
|
endif
|
|
|
|
if Set( _SET_INTENSITY )
|
|
if cBackColor == Nil // Only select Color Enhace if no color was passed
|
|
ColorSelect( CLR_ENHANCED )
|
|
endif
|
|
endif
|
|
|
|
// highlight the prompt
|
|
DispOutAt( s_aLevel[ nPointer - 1, n, 1 ],;
|
|
s_aLevel[ nPointer - 1, n, 2 ],;
|
|
s_aLevel[ nPointer - 1, n, 3 ],;
|
|
cBackColor )
|
|
|
|
if Set( _SET_INTENSITY )
|
|
if cFrontColor == NIL // Only select Color Enhace if no color was passed
|
|
ColorSelect( CLR_STANDARD )
|
|
endif
|
|
endif
|
|
|
|
if lExit
|
|
exit
|
|
endif
|
|
|
|
nKey := 0
|
|
do while nKey == 0
|
|
|
|
// wait for a keystroke
|
|
nKey := inkey( 0 )
|
|
|
|
if ( bAction := setkey( nKey ) ) <> NIL
|
|
|
|
eval( bBlock, n )
|
|
eval( bAction, procname( 1 ), procline( 1 ), upper( cVariable ) )
|
|
n := eval( bBlock )
|
|
|
|
if n < 1
|
|
n := 1
|
|
elseif n > nArrLen
|
|
n := nArrLen
|
|
endif
|
|
|
|
nKey := 0
|
|
|
|
endif
|
|
enddo
|
|
|
|
// check for keystrokes
|
|
SWITCH nKey
|
|
case K_MOUSEMOVE
|
|
EXIT
|
|
case K_LBUTTONDOWN
|
|
case K_LDBLCLK
|
|
if ( nMouseClik := hittest( s_aLevel[ nPointer - 1 ], ;
|
|
mrow(), mcol() ) ) > 0
|
|
n := nMouseClik
|
|
endif
|
|
if nKey == K_LDBLCLK
|
|
lExit := .T.
|
|
endif
|
|
EXIT
|
|
case K_DOWN
|
|
case K_RIGHT
|
|
if ++n > nArrLen
|
|
n := iif( Set( _SET_WRAP ), 1, nArrLen )
|
|
endif
|
|
EXIT
|
|
case K_UP
|
|
case K_LEFT
|
|
if --n < 1
|
|
n := iif( Set( _SET_WRAP ), nArrLen, 1 )
|
|
endif
|
|
EXIT
|
|
case K_HOME
|
|
n := 1
|
|
EXIT
|
|
case K_END
|
|
n := nArrLen
|
|
EXIT
|
|
case K_ENTER
|
|
case K_PGUP
|
|
case K_PGDN
|
|
lExit := .T.
|
|
EXIT
|
|
case K_ESC
|
|
n := 0
|
|
EXIT
|
|
otherwise
|
|
// did user hit a hot key?
|
|
for y := 1 to nArrLen
|
|
if upper( left( ltrim( s_aLevel[ nPointer - 1, y, 3 ] ), 1 ) ) == upper( chr( nKey ) )
|
|
n := y
|
|
lExit := .T.
|
|
exit
|
|
endif
|
|
next
|
|
end
|
|
|
|
if n <> 0
|
|
DispOutAt( s_aLevel[ nPointer - 1, q, 1 ],;
|
|
s_aLevel[ nPointer - 1, q, 2 ],;
|
|
s_aLevel[ nPointer - 1, q, 3 ],;
|
|
cFrontColor )
|
|
endif
|
|
|
|
enddo
|
|
|
|
ReadVar( cSaveReadVar )
|
|
SetCursor( nSaveCursor )
|
|
|
|
s_nPointer := nPointer
|
|
s_nPointer --
|
|
asize( s_aLevel, s_nPointer - 1 )
|
|
|
|
endif
|
|
|
|
eval( bBlock, n )
|
|
|
|
if lDeclared
|
|
__mvXRELEASE( cVariable )
|
|
endif
|
|
|
|
SetPos( MaxRow() - 1, 0)
|
|
|
|
RETURN n
|
|
|
|
STATIC FUNCTION HITTEST( aMenu, nMouseRow, nMouseCol )
|
|
|
|
LOCAL aMenuItem
|
|
|
|
FOR EACH aMenuItem IN aMenu
|
|
IF nMouseRow == aMenuItem[ 1 ] .AND. ;
|
|
nMouseCol >= aMenuItem[ 2 ] .AND. ;
|
|
nMouseCol < aMenuItem[ 2 ] + LEN( aMenuItem[ 3 ] )
|
|
|
|
RETURN aMenuItem:__enumIndex()
|
|
ENDIF
|
|
NEXT
|
|
|
|
RETURN 0
|