diff --git a/src/GDLInterpreter.cpp b/src/GDLInterpreter.cpp index dfe899ab6..291ff1364 100644 --- a/src/GDLInterpreter.cpp +++ b/src/GDLInterpreter.cpp @@ -81,38 +81,22 @@ GDLInterpreter::GDLInterpreter() try { // for error handling - //optimize speed: differentiate inner loops to avoid one externally implicit comparison on controlc for each loop (if possible) - if (interruptEnable) { //will test for sigControlC - { - do { - last = _retTree; - callStack.back()->SetLineNumber(last->getLine()); // track actual line number - retCode = last->Run(); // Run() sets _retTree - } while (_retTree != NULL && retCode == RC_OK && !(sigControlC) && (debugMode <= DEBUG_RETURN)); //loops if debug_clear or debug_return - if (_retTree != NULL) last = _retTree; //this is OK see https://github.com/gnudatalanguage/gdl/issues/1403#issuecomment-1326490113 - goto afterStatement; - } - } else { //will not test for sigControlC - { - do { - last = _retTree; - callStack.back()->SetLineNumber(last->getLine()); // track actual line number - retCode = last->Run(); // Run() sets _retTree - } while (_retTree != NULL && retCode == RC_OK && (debugMode <= DEBUG_RETURN)); //loops if debug_clear or debug_return - if (_retTree != NULL) last = _retTree; //this is OK see https://github.com/gnudatalanguage/gdl/issues/1403#issuecomment-1326490113 + //optimize speed: differentiate inner loops to avoid one externally implicit comparison on controlc for each loop (if possible) + if (interruptEnable) { //will test for sigControlC + do { + last = _retTree; + callStack.back()->SetLineNumber(last->getLine()); // track actual line number + retCode = last->Run(); // Run() sets _retTree + } while (_retTree != NULL && retCode == RC_OK && !(sigControlC) && (debugMode <= DEBUG_RETURN)); //loops if debug_clear or debug_return + } else { //will not test for sigControlC + do { + last = _retTree; + callStack.back()->SetLineNumber(last->getLine()); // track actual line number + retCode = last->Run(); // Run() sets _retTree + } while (_retTree != NULL && retCode == RC_OK && (debugMode <= DEBUG_RETURN)); //loops if debug_clear or debug_return + } + if (_retTree != NULL && debugMode != DEBUG_RETURN ) last = _retTree; goto afterStatement; - } - } - // original single loop with all checks - // { - // do { - // last = _retTree; - // callStack.back()->SetLineNumber(last->getLine()); // track actual line number - // retCode = last->Run(); // Run() sets _retTree - // } while (_retTree != NULL && retCode == RC_OK && !(sigControlC && interruptEnable) && (debugMode <= DEBUG_RETURN)); //loops if debug_clear or debug_return - // if (_retTree != NULL) last = _retTree; //this is OK see https://github.com/gnudatalanguage/gdl/issues/1403#issuecomment-1326490113 - // goto afterStatement; - // } // The following code (up to afterStatement is never executed - here only because ANTLR code needs it, although it should not (not optimized parser) @@ -358,72 +342,73 @@ GDLInterpreter::GDLInterpreter() } } - afterStatement: - // tested a possible optimization: make CtrlCHandler produce a DEBUG_STOP, and do not test sigControlC here at all. - // Apparently does not perform noticeably better and has adverse effects. Forget it. - if (interruptEnable && sigControlC) { - DebugMsg(last, "Interrupted at: "); - sigControlC = false; - retCode = NewInterpreterInstance(last->getLine()); //-1); - } else if (interruptEnable && _retTree == NULL && (debugMode == DEBUG_RETURN || debugMode == DEBUG_OUT)) { - if (debugMode == DEBUG_RETURN) { - if (callStack.back()->GetProName() == MyProName) { - DebugMsg(last, "Return encountered: "); - debugMode = DEBUG_CLEAR; - return NewInterpreterInstance(last->getLine()); //-1); - } - } else { //DEBUG_OUT --> just do an additional .step if we are at MyProName - if (callStack.back()->GetProName() == MyProName) { - debugMode = DEBUG_STEP; - stepCount=1; - return retCode; //continue - } - } - } else if (debugMode != DEBUG_CLEAR) { - if (debugMode == DEBUG_STOP) { - DebugMsg(last, "Stop encountered: "); - if (!interruptEnable) debugMode = DEBUG_PROCESS_STOP; - } else if (debugMode == DEBUG_STOP_SILENT) { - if (!interruptEnable) debugMode = DEBUG_PROCESS_STOP; - } - - if (debugMode == DEBUG_STEP) { - if (stepCount == 1) { - stepCount = 0; - DebugMsg(last, "Stepped to: "); - - debugMode = DEBUG_CLEAR; - - retCode = NewInterpreterInstance(last->getLine()); //-1); - } else { - --stepCount; + afterStatement: + // tested a possible optimization: make CtrlCHandler produce a DEBUG_STOP, and do not test sigControlC here at all. + // Apparently does not perform noticeably better and has adverse effects. Forget it. + if (interruptEnable && sigControlC) { + DebugMsg(last, "Interrupted at: "); + sigControlC = false; + retCode = NewInterpreterInstance(last->getLine()); //-1); + } else if (interruptEnable && _retTree == NULL && (debugMode == DEBUG_RETURN || debugMode == DEBUG_OUT)) { + if (debugMode == DEBUG_RETURN) { + if (callStack.back()->GetProName() == MyProName) { + DebugMsg(last, "Return encountered: "); + debugMode = DEBUG_STOP_SILENT; + return retCode; + } + } else { //DEBUG_OUT --> just do an additional .step if we are at MyProName + if (callStack.back()->GetProName() == MyProName) { + debugMode = DEBUG_STEP; + stepCount=1; + return retCode; //continue + } + } + } else if (debugMode != DEBUG_CLEAR) { + if (debugMode == DEBUG_STOP) { + DebugMsg(last, "Stop encountered: "); + if (!interruptEnable) debugMode = DEBUG_PROCESS_STOP; + } else if (debugMode == DEBUG_STOP_SILENT) { + if (!interruptEnable) debugMode = DEBUG_PROCESS_STOP; + } + if (debugMode == DEBUG_STEP) { + if (stepCount == 1) { + stepCount = 0; + DebugMsg(last, "Stepped to: "); + debugMode = DEBUG_CLEAR; + retCode = NewInterpreterInstance(last->getLine()); //-1); + } else { + --stepCount; #ifdef GDL_DEBUG - std::cout << "stepCount-- = " << stepCount << std::endl; + std::cout << "stepCount-- = " << stepCount << std::endl; #endif - } - } else if (debugMode == DEBUG_STEPOVER) { - if (callStack.back()->GetProName() == MyProName) { //we count only in current level - if (stepCount == 1) { - stepCount = 0; - DebugMsg(last, "Stepped to: "); - - debugMode = DEBUG_CLEAR; - MyProName=""; - retCode = NewInterpreterInstance(last->getLine()); //-1); - } else { - --stepCount; + } + } else if (debugMode == DEBUG_STEPOVER) { + if (callStack.back()->GetProName() == MyProName) { //we count only in current level + if (stepCount == 1) { + stepCount = 0; + DebugMsg(last, "Stepped to: "); + debugMode = DEBUG_CLEAR; + MyProName=""; + retCode = NewInterpreterInstance(last->getLine()); //-1); + } else { + --stepCount; #ifdef GDL_DEBUG - std::cout << "stepCount-- = " << stepCount << std::endl; + std::cout << "stepCount-- = " << stepCount << std::endl; #endif - } - } - } else if (interruptEnable) { - if (debugMode == DEBUG_PROCESS_STOP) DebugMsg(last, "Stepped to: "); - debugMode = DEBUG_CLEAR; - retCode = NewInterpreterInstance(last->getLine()); //-1); - } else { - retCode = RC_ABORT; - } + } + } + } else if (interruptEnable) { + if (debugMode == DEBUG_PROCESS_STOP) DebugMsg(last, "Stepped to: "); + debugMode = DEBUG_CLEAR; + retCode = NewInterpreterInstance(last->getLine()); //-1); + } else { + if (debugMode == DEBUG_PROCESS_STOP) { + debugMode = DEBUG_CLEAR; + retCode = NewInterpreterInstance(last->getLine()); + } else { + retCode = RC_ABORT; + } + } } return retCode; diff --git a/src/basic_pro_jmg.cpp b/src/basic_pro_jmg.cpp index 52c143a59..b2b7f0656 100644 --- a/src/basic_pro_jmg.cpp +++ b/src/basic_pro_jmg.cpp @@ -21,7 +21,7 @@ #include #include #include -#include "gdleventhandler.hpp" +#include "gdleventhandler.hpp" //for gdlwait_responsive #include "dinterpreter.hpp" #include "basic_pro_jmg.hpp" @@ -483,6 +483,76 @@ namespace lib { DllContainer::clear(); } + //idem WAIT_PRO, but needed in idlneturl as we wait while some widgets events must happen. + //should NEVER be used elsewhere (the idlneturl.pro procedure should disappear in favor of a pure C/C++ use of the CURL library) + //The real WAIT blocks absolutely anything --- as it should. + void gdlwait_responsive(EnvT* e) { + e->NParam(1); //, "WAIT"); + + DDouble waittime; + e->AssureDoubleScalarPar(0, waittime); + + if (waittime < 0) + throw GDLException(e->CallingNode(), + "WAIT: Argument must be non-negative" + + e->GetParString(0)); + +#ifdef _WIN32 + LARGE_INTEGER Frequency; + LARGE_INTEGER BeginTime; + LARGE_INTEGER Endtime; + LARGE_INTEGER elapsed; + LARGE_INTEGER waittime_us; + waittime_us.QuadPart = waittime * 1e6; + + QueryPerformanceFrequency(&Frequency); + QueryPerformanceCounter(&BeginTime); + + while (1) { + QueryPerformanceCounter(&Endtime); + elapsed.QuadPart = (Endtime.QuadPart - BeginTime.QuadPart) / (Frequency.QuadPart / 1000000); + if (elapsed.QuadPart >= waittime_us.QuadPart) break; + else if (elapsed.QuadPart > 100) Sleep(80); + } +#else + int old_version = 0; + + if (waittime <= 0.005) old_version = 1; + + // AC 2010-09-16 + // this version is OK and very accurate for small durations + // but used 100% of one CPU :(( + if (old_version == 1) { + struct timeval tval; + struct timezone tzone; + + // derivated from the current version of SYSTIME() + gettimeofday(&tval, &tzone); + double t_start = tval.tv_sec + tval.tv_usec / 1e+6; // time in UTC seconds + double t_current = 0.0; + + double diff = 0.0; + while (diff < waittime) { + + gettimeofday(&tval, &tzone); + t_current = tval.tv_sec + tval.tv_usec / 1e+6; + diff = t_current - t_start; + } + } + + // AC 2010-09-16 this version should used much less CPU ! + if (old_version == 0) { + //cout << floor(waittime) << " " << waittime-floor(waittime) << endl; + struct timespec tv; + tv.tv_sec = floor(waittime); + tv.tv_nsec = (waittime - floor(waittime))*1e9; + int retval; + retval = nanosleep(&tv, NULL); + } +#endif + GDLEventHandler(); //this is probably not OK, but is at the moment needed to permit delivery of callback functions in idlneturl__define.pro where a waiting loop uses teh WAIT command. + } + void wait_pro( EnvT* e) { e->NParam( 1);//, "WAIT"); @@ -548,7 +618,6 @@ namespace lib { retval=nanosleep(&tv,NULL); } #endif - GDLEventHandler(); //this is probably not OK, but is at the moment needed to permit delivery of callback functions in idlneturl__define.pro where a waiting loop uses teh WAIT command. } diff --git a/src/basic_pro_jmg.hpp b/src/basic_pro_jmg.hpp index 0c215ded1..68070232e 100644 --- a/src/basic_pro_jmg.hpp +++ b/src/basic_pro_jmg.hpp @@ -35,6 +35,7 @@ namespace lib { void unlinksymbol( EnvT* e ); void ResetDLLs( void ); void wait_pro( EnvT* e); + void gdlwait_responsive( EnvT* e); void kwtest( EnvT* e); diff --git a/src/devicewx.hpp b/src/devicewx.hpp index d9b2a287b..e6010e436 100644 --- a/src/devicewx.hpp +++ b/src/devicewx.hpp @@ -231,9 +231,8 @@ if(hide) { return gcFunction; } - DLongGDL* GetScreenSize(char* disp) { - DLongGDL* res; - res = new DLongGDL(2, BaseGDL::NOZERO); + DIntGDL* GetScreenSize(char* disp) { + DIntGDL* res = new DIntGDL(2, BaseGDL::NOZERO); (*res)[0] = wxSystemSettings::GetMetric(wxSYS_SCREEN_X); (*res)[1] = wxSystemSettings::GetMetric(wxSYS_SCREEN_Y); return res; diff --git a/src/devicex.hpp b/src/devicex.hpp index e1ea6677f..06cac8e7e 100644 --- a/src/devicex.hpp +++ b/src/devicex.hpp @@ -182,10 +182,9 @@ class DeviceX : public GraphicsMultiDevice { return gcFunction; } - DLongGDL* GetScreenSize(char* disp) { + DIntGDL* GetScreenSize(char* disp) { Display* display = XOpenDisplay(disp); int screen_num, screen_width, screen_height; - DLongGDL* res; if (display == NULL) { screen_width = 0; @@ -196,7 +195,7 @@ class DeviceX : public GraphicsMultiDevice { screen_height = DisplayHeight(display, screen_num); XCloseDisplay(display); } - res = new DLongGDL(2, BaseGDL::NOZERO); + DIntGDL* res = new DIntGDL(2, BaseGDL::NOZERO); (*res)[0]= screen_width; (*res)[1]= screen_height; return res; diff --git a/src/devicez.hpp b/src/devicez.hpp index f8a48b3b0..380fd9495 100644 --- a/src/devicez.hpp +++ b/src/devicez.hpp @@ -220,7 +220,7 @@ class DeviceZ: public GraphicsDevice memBuffer[i] = bColor; } - DLong GetPixelDepth() { return 24;} + DInt GetPixelDepth() { return 24;} bool SetPixelDepth(DInt value) { static int displayed=0; diff --git a/src/dpro.hpp b/src/dpro.hpp index 94a806236..068a8a4df 100644 --- a/src/dpro.hpp +++ b/src/dpro.hpp @@ -318,7 +318,7 @@ class DLibFunRetNewTP: public DLibFun bool RetConstant() { return this->retConstant;} }; // direct call functions must have: -// ony one parameter, no keywords +// only one parameter, no keywords // these functions are called "direct", no environment is created class DLibFunDirect: public DLibFunRetNew { diff --git a/src/gdlc.i.g b/src/gdlc.i.g index 92e27f423..76272e019 100644 --- a/src/gdlc.i.g +++ b/src/gdlc.i.g @@ -1102,39 +1102,23 @@ statement returns[ RetCode retCode] _retTree = _t; } : - { - //optimize speed: differentiate inner loops to avoid one externally implicit comparison on controlc for each loop (if possible) - if (interruptEnable) { //will test for sigControlC - { - do { - last = _retTree; - callStack.back()->SetLineNumber(last->getLine()); // track actual line number - retCode = last->Run(); // Run() sets _retTree - } while (_retTree != NULL && retCode == RC_OK && !(sigControlC) && (debugMode <= DEBUG_RETURN)); //loops if debug_clear or debug_return - if (_retTree != NULL) last = _retTree; //this is OK see https://github.com/gnudatalanguage/gdl/issues/1403#issuecomment-1326490113 - goto afterStatement; - } - } else { //will not test for sigControlC - { - do { - last = _retTree; - callStack.back()->SetLineNumber(last->getLine()); // track actual line number - retCode = last->Run(); // Run() sets _retTree - } while (_retTree != NULL && retCode == RC_OK && (debugMode <= DEBUG_RETURN)); //loops if debug_clear or debug_return - if (_retTree != NULL) last = _retTree; //this is OK see https://github.com/gnudatalanguage/gdl/issues/1403#issuecomment-1326490113 +{ + //optimize speed: differentiate inner loops to avoid one externally implicit comparison on controlc for each loop (if possible) + if (interruptEnable) { //will test for sigControlC + do { + last = _retTree; + callStack.back()->SetLineNumber(last->getLine()); // track actual line number + retCode = last->Run(); // Run() sets _retTree + } while (_retTree != NULL && retCode == RC_OK && !(sigControlC) && (debugMode <= DEBUG_RETURN)); //loops if debug_clear or debug_return + } else { //will not test for sigControlC + do { + last = _retTree; + callStack.back()->SetLineNumber(last->getLine()); // track actual line number + retCode = last->Run(); // Run() sets _retTree + } while (_retTree != NULL && retCode == RC_OK && (debugMode <= DEBUG_RETURN)); //loops if debug_clear or debug_return + } + if (_retTree != NULL && debugMode != DEBUG_RETURN ) last = _retTree; goto afterStatement; - } - } -// original single loop with all checks -// { -// do { -// last = _retTree; -// callStack.back()->SetLineNumber(last->getLine()); // track actual line number -// retCode = last->Run(); // Run() sets _retTree -// } while (_retTree != NULL && retCode == RC_OK && !(sigControlC && interruptEnable) && (debugMode <= DEBUG_RETURN)); //loops if debug_clear or debug_return -// if (_retTree != NULL) last = _retTree; //this is OK see https://github.com/gnudatalanguage/gdl/issues/1403#issuecomment-1326490113 -// goto afterStatement; -// } // The following code (up to afterStatement is never executed - here only because ANTLR code needs it, although it should not (not optimized parser) } @@ -1178,75 +1162,76 @@ statement returns[ RetCode retCode] // control-c and debugging { - afterStatement: - // tested a possible optimization: make CtrlCHandler produce a DEBUG_STOP, and do not test sigControlC here at all. - // Apparently does not perform noticeably better and has adverse effects. Forget it. - if (interruptEnable && sigControlC) { - DebugMsg(last, "Interrupted at: "); - sigControlC = false; - retCode = NewInterpreterInstance(last->getLine()); //-1); - } else if (interruptEnable && _retTree == NULL && (debugMode == DEBUG_RETURN || debugMode == DEBUG_OUT)) { - if (debugMode == DEBUG_RETURN) { - if (callStack.back()->GetProName() == MyProName) { - DebugMsg(last, "Return encountered: "); - debugMode = DEBUG_CLEAR; - return NewInterpreterInstance(last->getLine()); //-1); - } - } else { //DEBUG_OUT --> just do an additional .step if we are at MyProName - if (callStack.back()->GetProName() == MyProName) { - debugMode = DEBUG_STEP; - stepCount=1; - return retCode; //continue - } - } - } else if (debugMode != DEBUG_CLEAR) { - if (debugMode == DEBUG_STOP) { - DebugMsg(last, "Stop encountered: "); - if (!interruptEnable) debugMode = DEBUG_PROCESS_STOP; - } else if (debugMode == DEBUG_STOP_SILENT) { - if (!interruptEnable) debugMode = DEBUG_PROCESS_STOP; - } - - if (debugMode == DEBUG_STEP) { - if (stepCount == 1) { - stepCount = 0; - DebugMsg(last, "Stepped to: "); - - debugMode = DEBUG_CLEAR; - - retCode = NewInterpreterInstance(last->getLine()); //-1); - } else { - --stepCount; + afterStatement: + // tested a possible optimization: make CtrlCHandler produce a DEBUG_STOP, and do not test sigControlC here at all. + // Apparently does not perform noticeably better and has adverse effects. Forget it. + if (interruptEnable && sigControlC) { + DebugMsg(last, "Interrupted at: "); + sigControlC = false; + retCode = NewInterpreterInstance(last->getLine()); //-1); + } else if (interruptEnable && _retTree == NULL && (debugMode == DEBUG_RETURN || debugMode == DEBUG_OUT)) { + if (debugMode == DEBUG_RETURN) { + if (callStack.back()->GetProName() == MyProName) { + DebugMsg(last, "Return encountered: "); + debugMode = DEBUG_STOP_SILENT; + return retCode; + } + } else { //DEBUG_OUT --> just do an additional .step if we are at MyProName + if (callStack.back()->GetProName() == MyProName) { + debugMode = DEBUG_STEP; + stepCount=1; + return retCode; //continue + } + } + } else if (debugMode != DEBUG_CLEAR) { + if (debugMode == DEBUG_STOP) { + DebugMsg(last, "Stop encountered: "); + if (!interruptEnable) debugMode = DEBUG_PROCESS_STOP; + } else if (debugMode == DEBUG_STOP_SILENT) { + if (!interruptEnable) debugMode = DEBUG_PROCESS_STOP; + } + if (debugMode == DEBUG_STEP) { + if (stepCount == 1) { + stepCount = 0; + DebugMsg(last, "Stepped to: "); + debugMode = DEBUG_CLEAR; + retCode = NewInterpreterInstance(last->getLine()); //-1); + } else { + --stepCount; #ifdef GDL_DEBUG - std::cout << "stepCount-- = " << stepCount << std::endl; + std::cout << "stepCount-- = " << stepCount << std::endl; #endif - } - } else if (debugMode == DEBUG_STEPOVER) { - if (callStack.back()->GetProName() == MyProName) { //we count only in current level - if (stepCount == 1) { - stepCount = 0; - DebugMsg(last, "Stepped to: "); - - debugMode = DEBUG_CLEAR; - MyProName=""; - retCode = NewInterpreterInstance(last->getLine()); //-1); - } else { - --stepCount; + } + } else if (debugMode == DEBUG_STEPOVER) { + if (callStack.back()->GetProName() == MyProName) { //we count only in current level + if (stepCount == 1) { + stepCount = 0; + DebugMsg(last, "Stepped to: "); + debugMode = DEBUG_CLEAR; + MyProName=""; + retCode = NewInterpreterInstance(last->getLine()); //-1); + } else { + --stepCount; #ifdef GDL_DEBUG - std::cout << "stepCount-- = " << stepCount << std::endl; + std::cout << "stepCount-- = " << stepCount << std::endl; #endif - } - } - } else if (interruptEnable) { - if (debugMode == DEBUG_PROCESS_STOP) DebugMsg(last, "Stepped to: "); - debugMode = DEBUG_CLEAR; - retCode = NewInterpreterInstance(last->getLine()); //-1); - } else { - retCode = RC_ABORT; - } + } + } + } else if (interruptEnable) { + if (debugMode == DEBUG_PROCESS_STOP) DebugMsg(last, "Stepped to: "); + debugMode = DEBUG_CLEAR; + retCode = NewInterpreterInstance(last->getLine()); //-1); + } else { + if (debugMode == DEBUG_PROCESS_STOP) { + debugMode = DEBUG_CLEAR; + retCode = NewInterpreterInstance(last->getLine()); + } else { + retCode = RC_ABORT; + } + } } return retCode; - } +} ; exception catch [ GDLException& e] diff --git a/src/gdleventhandler.cpp b/src/gdleventhandler.cpp index 12c47cb0c..dcba50544 100644 --- a/src/gdleventhandler.cpp +++ b/src/gdleventhandler.cpp @@ -45,11 +45,15 @@ int GDLEventHandler() if (useWxWidgets) GDLWidget::HandleUnblockedWidgetEvents(); #endif GraphicsDevice::HandleEvents(); - #ifdef _WIN32 - Sleep(10); // this just to quiet down the character input from readline. 2 was not enough. 20 was ok. -#endif - + Sleep(10); // this just to quiet down the character input from readline. 2 was not enough. 20 was ok. +#else + const long SLEEP = 10000000; // 10ms + struct timespec delay; + delay.tv_sec = 0; + delay.tv_nsec = SLEEP; + nanosleep(&delay, NULL); +#endif return 0; } diff --git a/src/gdlhelp.cpp b/src/gdlhelp.cpp index e1cbdffe1..d43f14482 100644 --- a/src/gdlhelp.cpp +++ b/src/gdlhelp.cpp @@ -138,14 +138,14 @@ struct CompProName: public std::function } }; -static bool CompareWithJokers(string names, string sourceFiles) { +static bool CompareWithJokers(string theString, string thePattern) { #ifdef _WIN32 WCHAR wnames[MAX_PATH]; WCHAR wsourceFiles[MAX_PATH]; - const char* cnames = names.c_str(); - const char* csourceFiles = sourceFiles.c_str(); + const char* cnames = theString.c_str(); + const char* csourceFiles = thePattern.c_str(); MultiByteToWideChar(CP_UTF8, 0, cnames, -1, (LPWSTR) wnames, MAX_PATH); @@ -153,7 +153,11 @@ static bool CompareWithJokers(string names, string sourceFiles) { (LPWSTR) wsourceFiles, MAX_PATH); int match = 1 - PathMatchSpecW(wsourceFiles, wnames); #else - int match = fnmatch(names.c_str(), sourceFiles.c_str(), 0 ); + int flags=0; +#ifdef __GNUC__ + flags=FNM_CASEFOLD; +#endif + int match = fnmatch(theString.c_str(), thePattern.c_str(), flags ); #endif if ( match == 0) return true; else return false; @@ -803,6 +807,13 @@ void help_help(EnvT* e) } void help_pro(EnvT* e) { + //unsupported: just return + //BREAKPOINTS","DLM", "MESSAGES", "LAMBDA", + static int BREAKPOINTS = e->KeywordIx("BREAKPOINTS"); if (e->KeywordPresent(BREAKPOINTS)) { Message("Unsupported Keyword."); return;} + static int DLM = e->KeywordIx("DLM"); if (e->KeywordPresent(DLM)) { Message("Unsupported Keyword."); return;} + static int MESSAGES = e->KeywordIx("MESSAGES"); if (e->KeywordPresent(MESSAGES)) { Message("Unsupported Keyword."); return;} + static int LAMBDA = e->KeywordIx("LAMBDA"); if (e->KeywordPresent(LAMBDA)) { Message("Unsupported Keyword."); return;} + // in order of priority bool kw = false; static int lastmKWIx = e->KeywordIx("LAST_MESSAGE"); @@ -834,6 +845,15 @@ void help_help(EnvT* e) return; } + static int namesIx = e->KeywordIx("NAMES"); + bool isKWSetNames = e->KeywordPresent(namesIx); + DString names = ""; + if (isKWSetNames) { + e->AssureStringScalarKWIfPresent(namesIx, names); + // since routines and var. are stored in Maj, we convert ... + names = StrUpCase(names); + } + static int allkeysIx = e->KeywordIx("ALL_KEYS"); static int keysIx = e->KeywordIx("KEYS"); if (e->KeywordSet(allkeysIx) || e->KeywordSet(keysIx)) // ALL_KEYS is an obsolete keyword @@ -892,7 +912,7 @@ void help_help(EnvT* e) return; } - static int heapIx = e->KeywordIx("HEAP"); + static int heapIx = e->KeywordIx("HEAP_VARIABLES"); if (e->KeywordSet(heapIx)) { help_heap_obj_ptr_head(e, *ostrp); if (briefKW) return; @@ -911,9 +931,7 @@ void help_help(EnvT* e) #endif return; } - static int namesIx = e->KeywordIx("NAMES"); - bool isKWSetNames = e->KeywordPresent(namesIx); - + static int sysvarIx = e->KeywordIx("SYSTEM_VARIABLES"); if (e->KeywordSet(sysvarIx)) { help_sysvar(*ostrp, briefKW); @@ -962,12 +980,6 @@ void help_help(EnvT* e) return; } - DString names = ""; - if (isKWSetNames) { - e->AssureStringScalarKWIfPresent(namesIx, names); - // since routines and var. are stored in Maj, we convert ... - names = StrUpCase(names); - } static int sourceFilesKWIx = e->KeywordIx("SOURCE_FILES"); bool sourceFilesKW = e->KeywordPresent(sourceFilesKWIx); @@ -1153,6 +1165,7 @@ void help_help(EnvT* e) } else { //bug 1628 char ctmp; int nb_lines = TermHeight(); + if (SysVar::More() == 0) nb_lines=10000; *ostrp << "Recall buffer length: " << nEl2 << '\n'; for (SizeT i = 0; i < nEl2; ++i) { if (isKWSetNames and !CompareWithJokers(names, (*previous_commands)[i])) continue; @@ -1207,7 +1220,7 @@ void help_help(EnvT* e) if ((nParam == 0 and !isKWSetMemory) or isKWSetFunctions or isKWSetProcedures) { - if (nParam == 0 and !isKWSetFunctions and !isKWSetProcedures and !routinesKW) SimpleDumpStack(e, *ostrp); + if (nParam == 0 and !isKWSetFunctions and !isKWSetProcedures and !routinesKW and !isKWSetNames) SimpleDumpStack(e, *ostrp); if (isKWSetProcedures or routinesKW) { *ostrp << "Compiled Procedures:\n$MAIN$\n"; @@ -1456,7 +1469,7 @@ void help_help(EnvT* e) if (routinesKW and briefKW) kw = true; if (nParam == 0 and !kw) { - routinesKW = true; + routinesKW = !isKWSetNames; //when "NAMES" is present, ROUTINES ARE NOT GIVEN. briefKW = true; // list all variables of caller diff --git a/src/gdlwidget.cpp b/src/gdlwidget.cpp index 03b54012c..766777f37 100644 --- a/src/gdlwidget.cpp +++ b/src/gdlwidget.cpp @@ -937,9 +937,6 @@ inline wxSize GDLWidgetList::computeWidgetSize() widgetSize.y = lineHeight; } - if (wSize.x > 0 && maxlinelength > wSize.x) widgetSize.y += gdlSCROLL_HEIGHT_X; - if (nlines > wSize.y) widgetSize.x += gdlSCROLL_WIDTH_Y; - widgetSize.y += 10; //but.. if (wScreenSize.x > 0 ) widgetSize.x = wScreenSize.x; //we need an integer value if (wScreenSize.y > 0) widgetSize.y = wScreenSize.y; @@ -1004,12 +1001,12 @@ inline wxSize GDLWidget::computeWidgetSize() //here is a good place to make dynamic widgets static, since dynamic_resize is permitted only if size is not given. if (wSize.x > 0 || wSize.y > 0 || wScreenSize.x > 0 || wScreenSize.y > 0) dynamicResize=-1; wxSize widgetSize; - if ( wSize.x > 0 ) widgetSize.x = wSize.x*unitConversionFactor.x; + if ( wSize.x > 0 ) widgetSize.x = wSize.x; else widgetSize.x = wxDefaultSize.x; //but.. if (wScreenSize.x > 0) widgetSize.x=wScreenSize.x; - if ( wSize.y > 0 ) widgetSize.y = wSize.y * unitConversionFactor.y; + if ( wSize.y > 0 ) widgetSize.y = wSize.y; else widgetSize.y = wxDefaultSize.y; //but.. if (wScreenSize.y > 0) widgetSize.y=wScreenSize.y; @@ -1243,8 +1240,8 @@ void GDLWidget::SendWidgetTimerEvent(int millisecs) { if (m_windowTimer == NULL) { m_windowTimer = new wxTimer(w->GetEventHandler(), widgetID); } -#ifdef GDL_DEBUG_WIDGETS - std::cerr << "sending event," << widgetID << "," << m_windowTimer << std::endl; +#ifdef GDL_DEBUG_WIDGETS_TIMER + std::cerr << "SendWidgetTimerEvent: " << widgetID << "," << m_windowTimer << std::endl; #endif m_windowTimer->StartOnce(millisecs); } @@ -1260,21 +1257,40 @@ void GDLWidget::HandleUnblockedWidgetEvents() CallWXEventLoop(); //treat our GDL events... DStructGDL* ev = NULL; - while( (ev = GDLWidget::widgetEventQueue.Pop()) != NULL) - { - ev = CallEventHandler( ev ); - - if( ev != NULL) + while( (ev = GDLWidget::widgetEventQueue.Pop()) != NULL) { + //check event's TOP base is either not managed or is tagged interactive (done in xmanager.pro with "undocumented" keyword /XMANAGER_ACTIVE_COMMAND) + static int topIx = ev->Desc()->TagIndex("TOP"); + WidgetIDT top = (*static_cast (ev->GetTag(topIx, 0)))[0]; // get its id + GDLWidget* w = GDLWidget::GetWidget(top); + if (w //exists + && ( + (w->GetManaged() == false) // not managed at all + || (w->IsUsingInteractiveEventLoop()))) //managed but with a nonblocking xmanager { - GDLDelete( ev ); - ev = NULL; - } - CallWXEventLoop(); // eneble results of above in the widgets + ev = CallEventHandler(ev); + if (ev != NULL) { + GDLDelete(ev); + ev = NULL; + } + CallWXEventLoop(); // enable results of above in the widgets + + } + if (wxIsBusy()) wxEndBusyCursor(); + // avoid looping like crazy +#ifdef _WIN32 + Sleep(1); // this just to quiet down the character input from readline. 2 was not enough. 20 was ok. +#else + const long SLEEP = 1000000; // 1ms + struct timespec delay; + delay.tv_sec = 0; + delay.tv_nsec = SLEEP; + nanosleep(&delay, NULL); +#endif + } - if (wxIsBusy()) wxEndBusyCursor( ); } -void GDLWidget::PushEvent( WidgetIDT baseWidgetID, DStructGDL* ev) { +void GDLWidget::PushEvent(DStructGDL* ev) { widgetEventQueue.PushBack( ev ); } @@ -1628,7 +1644,8 @@ void GDLWidget::SetFocus() //gives focus to the CHILD of the panel. wxWindow *me=dynamic_cast(this->GetWxWidget()); if (me!=NULL) me->SetFocus(); else cerr<<"Setting Focus for unknown widget!\n"; } void GDLWidget::SetWidgetPosition(DLong posx, DLong posy){ - // -1: not set ---> keep the same + START_CHANGESIZE_NOEVENT + // -1: not set ---> keep the same wxWindow* me=static_cast(theWxContainer); if (me) { wxPoint where=me->GetPosition(); @@ -1636,7 +1653,7 @@ void GDLWidget::SetFocus() //gives focus to the CHILD of the panel. if (posy >= 0) where.y=posy; me->Move(where); } else cerr<<"set offset on non-existent widget!"< (theWxContainer); - if (p) { + if (p) { //there is a container. +// if (theWxWidget != theWxContainer) { +// // from theWxWidget to theWxContainer, fit all intermediate containers +// wxWindow* w = static_cast (theWxWidget); +//// std::cerr << " start from theWxWidget: " << theWxWidget << ","; +// while (w && w != p) { +// wxSizer* ws = w->GetSizer(); +// if (ws == NULL) { +// break; +// } +// ws->Fit(w); +// w = w->GetParent(); +//// std::cerr << ", upwards: " << w << ","; +// } +//// std::cerr << std::endl; +// } wxSizer* s = p->GetSizer(); if (s) s->Fit(p); else { @@ -1840,9 +1872,9 @@ bool GDLWidget::IsRealized() { } } } -#ifdef GDL_DEBUG_WIDGETS +//#ifdef GDL_DEBUG_WIDGETS else wxMessageOutputStderr().Printf(_T("Unknown Container for (%s) widget ID %d\n"), widgetName, widgetID); -#endif +//#endif if (notifyRealize != "") { //insure it is called once only for this. std::string note = notifyRealize; @@ -2166,19 +2198,16 @@ GDLWidgetBase::GDLWidgetBase(WidgetIDT parentID, EnvT* e, ULong eventFlags_, , ypad(ypad_) , doMap(mapWid) { // All bases can receive events: EV_CONTEXT, EV_KBRD_FOCUS, EV_TRACKING - wSize = computeWidgetSize(); //get immediately rid of scroll sizes in case of scroll or not... Here is the logic: - if (x_scroll_size > 0) {scrolled=true;x_scroll_size*=unitConversionFactor.x;x_scroll_size+=gdlSCROLL_WIDTH_Y;} - if (y_scroll_size > 0) {scrolled=true;y_scroll_size*=unitConversionFactor.y;y_scroll_size+=gdlSCROLL_HEIGHT_X;} + if (x_scroll_size > 0) {scrolled=true;x_scroll_size+=gdlSCROLL_WIDTH_Y;} else xfree=true; + if (y_scroll_size > 0) {scrolled=true;y_scroll_size+=gdlSCROLL_HEIGHT_X;} else yfree=true; if (scrolled) { if (x_scroll_size < 1) x_scroll_size = gdlDEFAULT_XSIZE+gdlSCROLL_WIDTH_Y; if (y_scroll_size < 1) y_scroll_size = gdlDEFAULT_YSIZE+gdlSCROLL_HEIGHT_X; } wScrollSize = scrolled ? wxSize(x_scroll_size , y_scroll_size ) : wSize; //y_scroll_size + gdlSCROLL_HEIGHT_X); - xfree=(wScrollSize.x <= 0); - yfree=(wScrollSize.y <= 0); // Set exclusiveMode // If exclusive then set to -1 to signal first radiobutton if ( exclusiveMode_ == BGEXCLUSIVE ) @@ -2206,33 +2235,37 @@ GDLWidgetBase::GDLWidgetBase(WidgetIDT parentID, EnvT* e, ULong eventFlags_, // NOTE: a=widget_base(/base_align_center, /scroll,xsize=300,ysize=300) will NOT produce a blank 300x300 panel inside a 100x100 scolled window, as we use only ONE // panel, and the internal size (the VirtualSize) will be determined by what WILL BE created as childrens (there exist no // This is completely different from IDL's Motif widgets, and there is no SetMinVirtualSize() -void GDLWidgetBase::CreateBase(wxWindow* parent){ -//the container is a ScrollPanel -// bool doFrame=true; //!(this->IsTopBase()); //IDL Prevents topBases to be framed (?). + +void GDLWidgetBase::CreateBase(wxWindow* parent) { + //the container is a ScrollPanel + if (scrolled && frameWidth < 1) frameWidth=1; if (frameWidth > 0) { - wxPanel* frame = new wxPanel(parent, wxID_ANY, wOffset, wxDefaultSize, gdlBORDER_EXT); - theWxContainer=frame; + wxPanel* frame = new wxPanel(parent, wxID_ANY, wOffset, wxDefaultSize, gdlBORDER_EXT); +// std::cerr<<" frame panel: "<SetSizer(panelsz); - wxBoxSizer* sz_inside=panelsz; - wxPanel* frame_inside=frame; - DLong newframewidth=frameWidth; -// Fancy variant: - if (frameWidth > 1 /*&& tryToMimicOriginalWidgets*/ ) { - newframewidth=frameWidth/2; -// frame->SetBackgroundColour(*wxBLACK); //will show a strong frame as does IDL - frame->SetBackgroundColour(wxColour(127,127,127)); //will show a strong frame as does IDL + wxBoxSizer* sz_inside = panelsz; + wxPanel* frame_inside = frame; + DLong newframewidth = frameWidth; + // Fancy variant: + if (frameWidth > 1 /*&& tryToMimicOriginalWidgets*/) { + newframewidth = frameWidth / 2; + // frame->SetBackgroundColour(*wxBLACK); //will show a strong frame as does IDL + frame->SetBackgroundColour(wxColour(127, 127, 127)); //will show a strong frame as does IDL int mode = wxBORDER_NONE; frame_inside = new wxPanel(frame, wxID_ANY, wxDefaultPosition, wxDefaultSize, mode); +// std::cerr<<" frame_inside : "<Add(frame_inside, FRAME_ALLOWSTRETCH, wxALL | wxEXPAND, newframewidth); frame_inside->SetBackgroundColour(*wxLIGHT_GREY); - panelsz->Add(frame_inside, FRAME_ALLOWSTRETCH, wxALL|wxEXPAND, newframewidth); - sz_inside = new wxBoxSizer(wxVERTICAL); frame_inside->SetSizer(sz_inside); - } + } if (xpad > 0 || ypad > 0) { - wxScrolled* padxpady = new wxScrolled(frame_inside); + wxScrolled* padxpady = new wxScrolled(frame_inside);//, wxID_ANY, wxDefaultPosition, wxDefaultSize, scrollmode); +// std::cerr<<" padxpady: "<Add(padxpady,FRAME_ALLOWSTRETCH, wxALL | wxEXPAND, newframewidth);//gdlFRAME_MARGIN); #ifdef GDL_DEBUG_WIDGETS_COLORIZE padxpady->SetBackgroundColour(wxColour(0xa7, 0x3d, 0x0f)); //orange fonce #else @@ -2244,60 +2277,55 @@ void GDLWidgetBase::CreateBase(wxWindow* parent){ if (xpad > 1) sz->Add(0, 0, wxGBPosition(1, 2)); if (ypad > 1) sz->Add(0, 0, wxGBPosition(2, 1)); - widgetPanel = new wxScrolled(padxpady, widgetID, wOffset, wxDefaultSize); + widgetPanel = new wxScrolled(padxpady, widgetID, wOffset, wxDefaultSize); +// std::cerr<<" widgetPanel: "<Add(widgetPanel, wxGBPosition(1, 1)); #ifdef GDL_DEBUG_WIDGETS_COLORIZE widgetPanel->SetBackgroundColour(RandomWxColour()); #endif - // widgetPanel->SetVirtualSize(wSize); - padxpady->SetSize(wScrollSize); padxpady->SetMinSize(wScrollSize); //Just Enable scrollBars if scrolling is necessary if (scrolled) { padxpady->SetScrollbars(gdlSCROLL_RATE, gdlSCROLL_RATE, wSize.x / gdlSCROLL_RATE, wSize.y / gdlSCROLL_RATE); - padxpady->ShowScrollbars(wxSHOW_SB_ALWAYS, wxSHOW_SB_ALWAYS); + padxpady->ShowScrollbars((xfree) ? wxSHOW_SB_DEFAULT : wxSHOW_SB_ALWAYS, (yfree) ? wxSHOW_SB_DEFAULT : wxSHOW_SB_ALWAYS); } - sz_inside->Add(padxpady, FRAME_ALLOWSTRETCH, wxALL | wxEXPAND, newframewidth);//gdlFRAME_MARGIN); - sz_inside->Fit(padxpady); } else { - widgetPanel = new wxScrolled(frame_inside, widgetID, wOffset, wxDefaultSize); + widgetPanel = new wxScrolled(frame_inside, widgetID, wOffset);//, wxDefaultSize, scrollmode); +// std::cerr<<" widgetPanel: "<Add(widgetPanel,FRAME_ALLOWSTRETCH, wxALL | wxEXPAND, newframewidth);//gdlFRAME_MARGIN); #ifdef GDL_DEBUG_WIDGETS_COLORIZE widgetPanel->SetBackgroundColour(RandomWxColour()); #else if (frameWidth > 1 /*&& tryToMimicOriginalWidgets*/) widgetPanel->SetBackgroundColour(wxColour(sysPanelDefaultColour)); #endif - // widgetPanel->SetVirtualSize(wSize); - widgetPanel->SetSize(wScrollSize); widgetPanel->SetMinSize(wScrollSize); //Just Enable scrollBars if scrolling is necessary if (scrolled) { widgetPanel->SetScrollbars(gdlSCROLL_RATE, gdlSCROLL_RATE, wSize.x / gdlSCROLL_RATE, wSize.y / gdlSCROLL_RATE); - widgetPanel->ShowScrollbars(wxSHOW_SB_ALWAYS, wxSHOW_SB_ALWAYS); + widgetPanel->ShowScrollbars((xfree) ? wxSHOW_SB_DEFAULT : wxSHOW_SB_ALWAYS, (yfree) ? wxSHOW_SB_DEFAULT : wxSHOW_SB_ALWAYS); } - sz_inside->Add(widgetPanel, FRAME_ALLOWSTRETCH, wxALL | wxEXPAND, newframewidth);//gdlFRAME_MARGIN); - sz_inside->Fit(widgetPanel); } theWxWidget = widgetPanel; } else { - if (xpad > 0 || ypad > 0 ) { - wxScrolled* padxpady = new wxScrolled(parent); + if (xpad > 0 || ypad > 0) { + wxScrolled* padxpady = new wxScrolled(parent);//, wxID_ANY, wxDefaultPosition, wxDefaultSize, scrollmode); +// std::cerr<<" padxpady: "<SetBackgroundColour(wxColour(0xa7, 0x3d, 0x0f)); //orange fonce - #endif +#endif wxGridBagSizer* sz = new wxGridBagSizer(ypad, xpad); padxpady->SetSizer(sz); sz->SetEmptyCellSize(wxSize(0, 0)); if (xpad > 1) sz->Add(0, 0, wxGBPosition(1, 2)); if (ypad > 1) sz->Add(0, 0, wxGBPosition(2, 1)); - widgetPanel = new wxScrolled(padxpady, widgetID, wOffset, wxDefaultSize); - sz->Add(widgetPanel, wxGBPosition(1, 1)); + widgetPanel = new wxScrolled(padxpady, widgetID, wOffset, wxDefaultSize); +// std::cerr<<" widgetPanel: "<Add(widgetPanel, wxGBPosition(1, 1)); #ifdef GDL_DEBUG_WIDGETS_COLORIZE widgetPanel->SetBackgroundColour(RandomWxColour()); #endif - // widgetPanel->SetVirtualSize(wSize); - padxpady->SetSize(wScrollSize); padxpady->SetMinSize(wScrollSize); //Just Enable scrollBars if scrolling is necessary if (scrolled) { @@ -2305,37 +2333,35 @@ void GDLWidgetBase::CreateBase(wxWindow* parent){ padxpady->ShowScrollbars(wxSHOW_SB_ALWAYS, wxSHOW_SB_ALWAYS); } } else { - widgetPanel = new wxScrolled(parent, widgetID, wOffset, wxDefaultSize); + widgetPanel = new wxScrolled(parent, widgetID, wOffset);//, wxDefaultSize, scrollmode); theWxContainer = widgetPanel; #ifdef GDL_DEBUG_WIDGETS_COLORIZE widgetPanel->SetBackgroundColour(RandomWxColour()); #endif -// widgetPanel->SetVirtualSize(wSize); - widgetPanel->SetSize(wScrollSize); widgetPanel->SetMinSize(wScrollSize); //Just Enable scrollBars if scrolling is necessary if (scrolled) { widgetPanel->SetScrollbars(gdlSCROLL_RATE, gdlSCROLL_RATE, wSize.x / gdlSCROLL_RATE, wSize.y / gdlSCROLL_RATE); - widgetPanel->ShowScrollbars(wxSHOW_SB_ALWAYS, wxSHOW_SB_ALWAYS); + widgetPanel->ShowScrollbars((xfree) ? wxSHOW_SB_DEFAULT : wxSHOW_SB_ALWAYS, (yfree) ? wxSHOW_SB_DEFAULT : wxSHOW_SB_ALWAYS); } } - + theWxWidget = widgetPanel; } - -//define inside sizer. +//if (theWxWidget != theWxContainer) std::cerr<SetSizer(widgetSizer); + if (widgetSizer) widgetPanel->SetSizer(widgetSizer); else if (scrolled) { //no col or row but scroll: need a sizer otherwise the content size will not be propagated to the scrolled window. - widgetSizer=new wxBoxSizer(wxHORIZONTAL); + widgetSizer = new wxBoxSizer(wxHORIZONTAL); widgetPanel->SetSizer(widgetSizer); - nrows=1; //do not forget as there would be problems with alignment checks afterwards + nrows = 1; //do not forget as there would be problems with alignment checks afterwards } wxSizer* parentSizer = parent->GetSizer(); - if (parentSizer) parentSizer->Add(static_cast(theWxContainer), ALLOWSTRETCH, wxALL | widgetAlignment(), gdlSPACE); - } + if (parentSizer) parentSizer->Add(static_cast (theWxContainer), ALLOWSTRETCH, wxALL | widgetAlignment(), gdlSPACE); +} void GDLWidgetBase::SetWidgetSize(DLong sizex, DLong sizey) { @@ -2424,7 +2450,7 @@ long style = (wxMINIMIZE_BOX | wxMAXIMIZE_BOX | wxRESIZE_BORDER | wxCAPTION | wx if (wOffset.x < 0) wOffset.x =x.x+x.width/2; if (wOffset.y < 0) wOffset.y =x.y+x.height/2; } - topFrame = new gdlwxFrame(NULL, this, widgetID, titleWxString, wOffset, wxDefaultSize, style, modal); + topFrame = new gdlwxFrame(NULL, this, widgetID, titleWxString, unitConversionFactor, wOffset, wxDefaultSize, style, modal); #ifdef __WXMAC__ //does not work. @@ -2654,14 +2680,14 @@ DStructGDL* GDLWidgetBase::GetGeometry(wxRealPoint fact) { xoffset = panel_xoff+margin; yoffset = panel_yoff+margin; } else { - xoffset = z.x+margin; - yoffset = z.y+margin; + xoffset = 0; + yoffset = 0; } - iscr_xsize = z.width-2*margin; - iscr_ysize = z.height-2*margin; + iscr_xsize = z.width;//-2*margin; + iscr_ysize = z.height;//-2*margin; wxSize s = w->GetClientSize(); - ixsize = s.x-2*margin; - iysize = s.y-2*margin; + ixsize = s.x;//-2*margin; + iysize = s.y;//-2*margin; //size is in pixels, pass in requested units (1.0 default) xsize = ixsize / fact.x; ysize = iysize / fact.y; @@ -2788,6 +2814,25 @@ DStructGDL* GDLWidgetBase::GetGeometry(wxRealPoint fact) { wxWindow* me=dynamic_cast(theWxContainer); if (me) me->Show(val); else {cerr<<"Warning: GDLWidgetTabbedBase::mapBase(): Non-existent widget!\n"; return;} } + // Order in which widgets are plotted is different btw wxWindow and IDL if one does not do the following tricks: + //DOES NOT WORK + void GDLWidgetBase::invertStack() { + wxScrolled* parent=this->GetParentPanel(); + if (parent == NULL) return; + int nchild = this->NChildren(); + if (nchild < 2) return; + std::vector windowlist; + //get children list... + DLongGDL* gdlwList= GetChildrenList(); + for (int i = 0; i < nchild; ++i) { + GDLWidget* g=GetWidget((*gdlwList)[i]); + wxWindow* w=static_cast(g->GetWxContainer()); + windowlist.push_back(w); + } + GDLDelete(gdlwList); + //THEN DO SOMETHING ! + //for (auto i = nchild; i>0; --i) DO SOMETHING + } // Order in which widgets are plotted is different btw wxWindow and IDL if one does not do the following tricks: void GDLWidgetBase::DoReorderColWidgets(int code,int style, int border) { int nchild = this->NChildren(); @@ -2874,8 +2919,8 @@ void GDLWidgetBase::ReorderWidgets() { // NULL widget Sizer means 1) no row no col was asked for (so, no sizer) or, if col>1, we have to create the sizer here and add children in specific order. // do *not* forget to give back the sizer pointer instead of the previous NULL to the base widget! - if (widgetSizer == NULL) return; - if (ncols > 1) DoReorderColWidgets(0,0,space); //need to reorder widget for /COL only + if (widgetSizer == NULL) return; //invertStack(); else + if (ncols > 1) DoReorderColWidgets(0,0,space); //need to reorder widget for /COL only } void GDLWidgetBase::ReorderForANewWidget(wxWindow* w, int code,int style, int border) @@ -3011,8 +3056,8 @@ DLongGDL* rowHeights_, DStringGDL* rowLabels_, //DLong tabMode_, BaseGDL* value_, -DLong xScrollSize_, -DLong yScrollSize_, +DLong xScrollColumn_, +DLong yScrollRow_, DStringGDL* valueAsStrings, DULong eventFlags_ ) @@ -3037,8 +3082,8 @@ DULong eventFlags_ , rowHeights( rowHeights_ ) , rowLabels( rowLabels_ ) //, tabMode( tabMode_ ) -, x_scroll_size_columns( xScrollSize_ ) -, y_scroll_size_rows( yScrollSize_) +, x_scroll_column( xScrollColumn_ ) +, y_scroll_row( yScrollRow_) , updating(false) { GDLWidget* gdlParent = GetWidget(parentID); widgetPanel = GetParentPanel(); @@ -3176,58 +3221,68 @@ DULong eventFlags_ ////Not Useful? //grid->SetInitialSize(wClientSize); - //Size will be WindowSize, as wSize is in columns. - wxSize windowSize(100, 100); //default value, avoids warnings in GTK - //wxSize in columns, transform to units - int xsize_columns = wSize.x; - int ysize_rows = wSize.y; + //Size will be WindowSize, as wSize and ScrollSize are is in columns. + int xsize_columns = grid_ncols; + int ysize_rows = grid_nrows; + wxSize windowSize; + // size of value is the default size: + { // no 'if' + int windowsizex = currentRowLabelWidth; + for (SizeT i = 0; i < xsize_columns; ++i) windowsizex += (i < grid_ncols) ? grid->GetColSize(i) : grid->GetDefaultColSize(); + windowSize.x = windowsizex; + } + { // no 'if' + int windowsizey = currentColLabelHeight; + for (SizeT j = 0; j < ysize_rows; ++j) windowsizey += (j < grid_nrows) ? grid->GetRowHeight(j) : grid->GetDefaultRowSize(); + windowSize.y = windowsizey; + } + + //then xsize, ysize in columns, win on above values + xsize_columns = wSize.x; + ysize_rows = wSize.y; if (xsize_columns > 0) { //defined - int windowsizex = currentRowLabelWidth; - for (SizeT i = 0; i < xsize_columns; ++i) windowsizex += (i < grid_ncols) ? grid->GetColSize(i) : grid->GetDefaultColSize(); - windowSize = wxSize(windowsizex, windowSize.y); + int windowsizex = currentRowLabelWidth; + for (SizeT i = 0; i < xsize_columns; ++i) windowsizex += (i < grid_ncols) ? grid->GetColSize(i) : grid->GetDefaultColSize(); + windowSize.x = windowsizex; } if (ysize_rows > 0) { //defined - int windowsizey = currentColLabelHeight; - for (SizeT j = 0; j < ysize_rows; ++j) windowsizey += (j < grid_nrows) ? grid->GetRowHeight(j) : grid->GetDefaultRowSize(); - windowSize = wxSize(windowSize.x, windowsizey); - } - - //Scrolled? - x_scroll_size_columns = (x_scroll_size_columns > grid_ncols) ? grid_ncols : x_scroll_size_columns; - y_scroll_size_rows = (y_scroll_size_rows > grid_nrows) ? grid_nrows : y_scroll_size_rows; - if (x_scroll_size_columns > 0) { //scroll size is in columns - scrolled = true; + int windowsizey = currentColLabelHeight; + for (SizeT j = 0; j < ysize_rows; ++j) windowsizey += (j < grid_nrows) ? grid->GetRowHeight(j) : grid->GetDefaultRowSize(); + windowSize.y = windowsizey; } - if (y_scroll_size_rows > 0) { //rows - scrolled = true; + + //then scroll in columns, win on above value if SMALLER + xsize_columns = x_scroll_column; + ysize_rows = y_scroll_row; + if (xsize_columns > 0) { //defined + scrolled = true; //sets scrolling + if (xsize_columns < windowSize.x) { + int windowsizex = currentRowLabelWidth; + for (SizeT i = 0; i < xsize_columns; ++i) windowsizex += (i < grid_ncols) ? grid->GetColSize(i) : grid->GetDefaultColSize(); + windowSize.x = windowsizex; + } } - if (scrolled) { //size to pass is given by scrolled - if (x_scroll_size_columns > 0) { //scroll size is in columns - int scrollsizex = currentRowLabelWidth + gdlSCROLL_WIDTH_Y; - for (SizeT i = 0; i < x_scroll_size_columns; ++i) scrollsizex += (i < grid_ncols) ? grid->GetColSize(i) : grid->GetDefaultColSize(); - windowSize = wxSize(scrollsizex, windowSize.y); - } - if (y_scroll_size_rows > 0) { //rows - int scrollsizey = currentColLabelHeight + gdlSCROLL_HEIGHT_X; - for (SizeT j = 0; j < y_scroll_size_rows; ++j) scrollsizey += (j < grid_nrows) ? grid->GetRowHeight(j) : grid->GetDefaultRowSize(); - windowSize = wxSize(windowSize.x, scrollsizey); - } + if (ysize_rows > 0) { //defined + scrolled = true;//sets scrolling + if (ysize_rows < windowSize.y) { + int windowsizey = currentColLabelHeight; + for (SizeT j = 0; j < ysize_rows; ++j) windowsizey += (j < grid_nrows) ? grid->GetRowHeight(j) : grid->GetDefaultRowSize(); + windowSize.y = windowsizey; + } } - //scr_xsize or ysize: replaces value in windowSize: - if (wScreenSize.x > 0) windowSize = wxSize(wScreenSize.x, windowSize.y); - if (wScreenSize.y > 0) windowSize = wxSize(windowSize.x, wScreenSize.y); - + //scr_xsize or ysize: replaces value in windowSize, these are IN PIXELS, after using UNITS. + if (wScreenSize.x > 0) windowSize.x = wScreenSize.x; + if (wScreenSize.y > 0) windowSize.y = wScreenSize.y; //wxGrid IS a scrolled window if (scrolled) { - // int visibx=x_scroll_size>0?wxSHOW_SB_ALWAYS:wxSHOW_SB_NEVER; - // int visiby=y_scroll_size>0?wxSHOW_SB_ALWAYS:wxSHOW_SB_NEVER; - grid->ShowScrollbars(wxSHOW_SB_ALWAYS, wxSHOW_SB_ALWAYS); + grid->ShowScrollbars(wxSHOW_SB_DEFAULT, wxSHOW_SB_DEFAULT); } grid->SetScrollLineX(grid->GetColSize(0)); grid->SetScrollLineY(grid->GetRowHeight(0)); grid->SetSize(windowSize); grid->SetMinSize(windowSize); + grid->SetMaxSize(windowSize); #if wxCHECK_VERSION(3,1,4) grid->SetDefaultCellFitMode(wxGridFitMode::Clip()); #endif @@ -3247,6 +3302,8 @@ DULong eventFlags_ std::vector GDLWidgetTable::GetSortedSelectedRowsOrColsList(DLongGDL* selection, bool doCol = true) { wxGridGDL * grid = dynamic_cast (theWxWidget); assert(grid != NULL); + int ncolsmax = this->GetDataNCols(); + int nrowsmax = this->GetDataNRows(); std::vector list; if (selection == NULL) { std::vector wxList = grid->GetSelectedDisjointCellsList(); @@ -3276,9 +3333,19 @@ std::vector GDLWidgetTable::GetSortedSelectedRowsOrColsList(DLongGDL* selec int index = 0; for (SizeT n = 0, l = 0; n < MAX(selection->Dim(1), 1); ++n) { index = (*selection)[l++]; - if (doCol) allRowsOrCols.push_back(index); + if (doCol) { + if (index == -1) { + for (auto i=0; i< ncolsmax; ++i) allRowsOrCols.push_back(i); //all cols + } + else allRowsOrCols.push_back(index); + } index = (*selection)[l++]; - if (!doCol) allRowsOrCols.push_back(index); + if (!doCol) { + if (index == -1) { + for (auto i=0; i< nrowsmax; ++i) allRowsOrCols.push_back(i); //all cols + } + else allRowsOrCols.push_back(index); + } } std::sort(allRowsOrCols.begin(), allRowsOrCols.end()); int theRowOrCol = -1; @@ -3289,8 +3356,8 @@ std::vector GDLWidgetTable::GetSortedSelectedRowsOrColsList(DLongGDL* selec } } } else { //4 values, cols are contiguous, easy. - int colTL = (*selection)[0]; - int colBR = (*selection)[2]; + int colTL = (*selection)[0]; if (colTL==-1) colTL=0; + int colBR = (*selection)[2]; if (colBR==-1) colBR=ncolsmax-1; int count = colBR - colTL + 1; for (auto i = colTL; i < colBR + 1; ++i) list.push_back(i); } @@ -3822,18 +3889,18 @@ bool GDLWidgetTable::GetValidTableSelection(DLongGDL* &selection) { for (auto j = 0; j < MAX(selection->Dim(1), 1); ++j) { DLong col = (*selection)[j * 2 + 0]; DLong row = (*selection)[j * 2 + 1]; - if (col < 0 || col > ncolsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); - if (row < 0 || row > nrowsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); + if ((col < 0 && col != -1) || col > ncolsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); + if ((row < 0 && row != -1)|| row > nrowsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); } } else { //4 values DLong val = (*selection)[0]; - if (val < 0 || val > ncolsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); + if ((val < 0 && val != -1) || val > ncolsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); val = (*selection)[1]; - if (val < 0 || val > nrowsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); + if ((val < 0 && val != -1) || val > nrowsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); val = (*selection)[2]; - if (val < 0 || val > ncolsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); + if ((val < 0 && val != -1) || val > ncolsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); val = (*selection)[3]; - if (val < 0 || val > nrowsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); + if ((val < 0 && val != -1) || val > nrowsmax) ThrowGDLException("USE_TABLE_SELECT value out of range."); } return true; } @@ -5190,6 +5257,8 @@ void GDLWidgetTable::SetSelection(DLongGDL* selection) { assert(grid != NULL); grid->BeginBatch(); updating = true; //prevent sending unwanted events + int ncolsmax = grid->GetNumberCols(); + int nrowsmax = grid->GetNumberRows(); grid->ClearSelection(); wxPoint firstVisible = wxPoint(0, 0); if (disjointSelection) { //pairs lists @@ -5197,10 +5266,14 @@ void GDLWidgetTable::SetSelection(DLongGDL* selection) { for (SizeT i = 0; i < MAX(selection->Dim(1), 1); ++i) { int col = (*selection)[k++]; int row = (*selection)[k++]; - grid->SelectBlock(row, col, row, col, true); + int rowmin=(row==-1)?0:row; + int rowmax=(row==-1)?nrowsmax-1:row; + int colmin=(col==-1)?0:col; + int colmax=(col==-1)?ncolsmax-1:col; + grid->SelectBlock(rowmin, colmin, rowmax, colmax, true); if (k == 2) { - firstVisible.x = row; - firstVisible.y = col; + firstVisible.x = rowmin; + firstVisible.y = colmin; } } } else { @@ -5208,9 +5281,13 @@ void GDLWidgetTable::SetSelection(DLongGDL* selection) { int rowTL = (*selection)[1]; int colBR = (*selection)[2]; int rowBR = (*selection)[3]; - grid->SelectBlock(rowTL, colTL, rowBR, colBR, false); - firstVisible.x = rowTL; - firstVisible.y = colTL; + int rowmin=(rowTL==-1)?0:rowTL; + int rowmax=(rowBR==-1)?nrowsmax-1:rowBR; + int colmin=(colTL==-1)?0:colTL; + int colmax=(colBR==-1)?ncolsmax-1:colBR; + grid->SelectBlock(rowmin, colmin, rowmax, colmax, false); + firstVisible.x = rowmin; + firstVisible.y = colmin; } grid->EndBatch(); grid->MakeCellVisible(firstVisible.x, firstVisible.y); @@ -5882,6 +5959,7 @@ GDLWidgetSlider::GDLWidgetSlider( WidgetIDT p, EnvT* e, DLong value_ const wxString s=wxString( title.c_str( ), wxConvUTF8 ); wxSlider* slider; + wxSize minsize=(vertical)?wxSize(-1,100):wxSize(100,-1); //get defined sizes if any wSize=computeWidgetSize( ); @@ -5897,6 +5975,7 @@ GDLWidgetSlider::GDLWidgetSlider( WidgetIDT p, EnvT* e, DLong value_ theWxContainer = comboPanel; //else stays as panel_if_framed. comboPanel->SetFont(font); //enable fancy font slider= new wxSlider( comboPanel, widgetID, value, minimum, maximum, wxDefaultPosition, wxDefaultSize, style); + slider->SetMinSize(minsize); combosz->Add(slider,DONOTALLOWSTRETCH,wxEXPAND|wxALL,gdlSPACE); int w=0; int h=0; @@ -5914,7 +5993,7 @@ GDLWidgetSlider::GDLWidgetSlider( WidgetIDT p, EnvT* e, DLong value_ } else { slider= new wxSlider( widgetPanel, widgetID, value, minimum, maximum, wxDefaultPosition, wxDefaultSize, style); slider->SetSize(wSize); - slider->SetMinSize(wSize); + slider->SetMinSize(minsize); theWxContainer = theWxWidget=slider; } widgetPanel->Fit(); @@ -6466,14 +6545,14 @@ GDLWidgetList::GDLWidgetList( WidgetIDT p, EnvT* e, BaseGDL *value, DLong style, } choices.Add( wxString( (*val)[i].c_str( ), wxConvUTF8 ) ); } - wxListBox * list=new wxListBox(); - theWxContainer = theWxWidget = list; - //ok now size can be computed - list->Create(widgetPanel, widgetID, wxDefaultPosition, wxDefaultSize , choices, style|wxLB_NEEDED_SB|wxLB_SINGLE); //|wxLB_MULTIPLE ); this->setFont(); //set fancy font before computing sizes! - wSize=computeWidgetSize(); - list->SetClientSize(wSize); - list->SetMinClientSize(wSize); + style |= wxLB_SINGLE; + wxSize wLocalSize = computeWidgetSize(); + if ((frameWidth > 0) && (wLocalSize.y > 2*gdlSCROLL_HEIGHT_X)) //IDL seems to do that + style|=wxLB_ALWAYS_SB; else style|=wxLB_NO_SB; + + wxListBox * list=new wxListBox(widgetPanel, widgetID, wxDefaultPosition, wLocalSize , choices, style); + theWxContainer = theWxWidget = list; list->SetSelection(wxNOT_FOUND);//necessary to avoid spurious event on first click. this->AddToDesiredEvents( wxEVT_COMMAND_LISTBOX_DOUBLECLICKED,wxCommandEventHandler(gdlwxFrame::OnListBoxDoubleClicked),list); this->AddToDesiredEvents( wxEVT_COMMAND_LISTBOX_SELECTED,wxCommandEventHandler(gdlwxFrame::OnListBox),list); @@ -6748,12 +6827,12 @@ inline wxSize GDLWidgetComboBox::computeWidgetSize() { //here is a good place to make dynamic widgets static, since dynamic_resize is permitted only if size is not given. if (wSize.x > 0 || wSize.y > 0 || wScreenSize.x > 0 || wScreenSize.y > 0) dynamicResize = -1; wxSize widgetSize; - if (wSize.x > 0) widgetSize.x = wSize.x * unitConversionFactor.x; + if (wSize.x > 0) widgetSize.x = wSize.x; else widgetSize.x = wxDefaultSize.x; //but.. if (wScreenSize.x > 0) widgetSize.x = wScreenSize.x; - if (wSize.y > 0) widgetSize.y = wSize.y * unitConversionFactor.y; + if (wSize.y > 0) widgetSize.y = wSize.y; else widgetSize.y = wxDefaultSize.y; //but.. if (wScreenSize.y > 0) widgetSize.y = wScreenSize.y; @@ -7357,14 +7436,17 @@ void GDLWidgetLabel::SetLabelValue(const DString& value_) { #endif // GDL widgets ===================================================== // GDLFrame ======================================================== -gdlwxFrame::gdlwxFrame( wxWindow* parent, GDLWidgetTopBase* gdlOwner_, wxWindowID id, const wxString& title , const wxPoint& pos, const wxSize& size, long style, bool modal) +gdlwxFrame::gdlwxFrame( wxWindow* parent, GDLWidgetTopBase* gdlOwner_, wxWindowID id, const wxString& title , const wxRealPoint& conv, const wxRealPoint& pos, const wxSize& size, long style, bool modal) : wxFrame() , mapped( false ) , frameSize(size) , gdlOwner( gdlOwner_), m_windowDisabler(NULL) { - Create ( parent, id, title, pos, size, style ); + wxPoint posint; + posint.x=pos.x*conv.x; + posint.y=pos.y*conv.y; + Create ( parent, id, title, posint, size, style ); m_resizeTimer = new wxTimer(this,RESIZE_TIMER); if (modal) m_windowDisabler = new wxWindowDisabler(this); } @@ -7583,8 +7665,8 @@ GDLWidgetDraw::GDLWidgetDraw( WidgetIDT p, EnvT* e, int windowIndex, //get immediately rid of scroll sizes in case of scroll or not... Here is the logic: // Note that IDL reports wrong dimensions when size and scr_xsize are used simultaneously for at least widget_draw. - if (x_scroll_size > 0) {scrolled=true;x_scroll_size*=unitConversionFactor.x;x_scroll_size+=gdlSCROLL_WIDTH_Y;} - if (y_scroll_size > 0) {scrolled=true;y_scroll_size*=unitConversionFactor.y;y_scroll_size+=gdlSCROLL_HEIGHT_X;} + if (x_scroll_size > 0) {scrolled=true;x_scroll_size+=gdlSCROLL_WIDTH_Y;} + if (y_scroll_size > 0) {scrolled=true;y_scroll_size+=gdlSCROLL_HEIGHT_X;} if (scrolled) { if (x_scroll_size < 1) x_scroll_size = gdlDEFAULT_XSIZE+gdlSCROLL_WIDTH_Y; if (y_scroll_size < 1) y_scroll_size = gdlDEFAULT_YSIZE+gdlSCROLL_HEIGHT_X; diff --git a/src/gdlwidget.hpp b/src/gdlwidget.hpp index 66efc6a12..51d1b13e5 100644 --- a/src/gdlwidget.hpp +++ b/src/gdlwidget.hpp @@ -21,6 +21,7 @@ #ifdef HAVE_LIBWXWIDGETS // #define GDL_DEBUG_WIDGETS +// #define GDL_DEBUG_WIDGETS_TIMER //only to show when a timer event is EMITTED. // #define GDL_DEBUG_WIDGETS_COLORIZE // use "plain menubars" instead of 'taskbars used as menubars'. taskbars permit to change font in bar, put pixmaps instead of text, and will work @@ -84,8 +85,8 @@ #define gdlCOMBOBOX_ARROW_WIDTH sysComboboxArrow #define gdlDEFAULT_SCROLL_SIZE 100 //gdlDEFAULT_XSIZE+gdlSCROLL_HEIGHT_X #define gdlFRAME_MARGIN 0 -#define gdlPAD 0 //3 //default padding -#define gdlSPACE 0 +#define gdlPAD 0 //default padding +#define gdlSPACE 3 #define gdlLABEL_SPACE 2 #define gdlTEXT_SPACE 4 #define gdlBUTTON_SPACE 4 @@ -286,7 +287,7 @@ class gdlwxFrame : public wxFrame { public: // ctor(s) - gdlwxFrame(wxWindow* parent, GDLWidgetTopBase* gdlOwner_, wxWindowID id, const wxString& title, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxDefaultSize, long style=wxDEFAULT_FRAME_STYLE, bool modal=false); + gdlwxFrame(wxWindow* parent, GDLWidgetTopBase* gdlOwner_, wxWindowID id, const wxString& title, const wxRealPoint& conv, const wxRealPoint& pos, const wxSize& size = wxDefaultSize, long style=wxDEFAULT_FRAME_STYLE, bool modal=false); ~gdlwxFrame(); // called from ~GDLWidgetBase void NullGDLOwner() { @@ -395,9 +396,8 @@ class GDLWidget public: static wxFont defaultFont; static wxFont systemFont; - static GDLEventQueue BlockingEventQueue; static GDLEventQueue widgetEventQueue; - static void PushEvent( WidgetIDT baseWidgetID, DStructGDL* ev); + static void PushEvent(DStructGDL* ev); static void InformAuthorities(const std::string& message); static void HandleUnblockedWidgetEvents(); @@ -465,7 +465,7 @@ class GDLWidget wxObject* theWxContainer; //the external wx Container (wxFrame mostly) that contains everything wx that must be destroyed, or created, and is what is seen. //theWxContainer is subject to framing (AddFrame) and scrolling (AddScroll) //position & size - wxPoint wOffset; + wxRealPoint wOffset; wxSize wSize; wxSize wScreenSize; @@ -597,18 +597,18 @@ class GDLWidget this->setFont(); this->SetSensitive(sensitive); // if (this->GetRealized()) this->RefreshWidget(); - if (eventPro.size() > 0 ) { -#ifdef GDL_DEBUG_WIDGETS - wxMessageOutputStderr().Printf(_T("Realize: SetEventPro: \"%s\" for %d\n"), eventPro, widgetID); -#endif - bool found=GDLInterpreter::SearchCompilePro(eventPro, true); // true -> search for procedure - } - if (eventFun.size() > 0 ) { -#ifdef GDL_DEBUG_WIDGETS - wxMessageOutputStderr().Printf(_T("Realize: SetEventFun: \"%s\" for %d\n"), eventFun, widgetID); -#endif - bool found=GDLInterpreter::SearchCompilePro(eventFun, false); // false -> search for function - } +// if (eventPro.size() > 0 ) { +//#ifdef GDL_DEBUG_WIDGETS +// wxMessageOutputStderr().Printf(_T("Realize: SetEventPro: \"%s\" for %d\n"), eventPro, widgetID); +//#endif +// bool found=GDLInterpreter::SearchCompilePro(eventPro, true); // true -> search for procedure +// } +// if (eventFun.size() > 0 ) { +//#ifdef GDL_DEBUG_WIDGETS +// wxMessageOutputStderr().Printf(_T("Realize: SetEventFun: \"%s\" for %d\n"), eventFun, widgetID); +//#endif +// bool found=GDLInterpreter::SearchCompilePro(eventFun, false); // false -> search for function +// } if( notifyRealize != "") { //insure it is called once only for this. std::string note=notifyRealize; notifyRealize.clear(); @@ -705,23 +705,31 @@ class GDLWidget #ifdef GDL_DEBUG_WIDGETS wxMessageOutputStderr().Printf(_T("SetEventPro: \"%s\" for %d\n"), eventPro, widgetID); #endif - if (eventPro.size() > 0) { - bool found = GDLInterpreter::SearchCompilePro(eventPro, true); // true -> search for procedure - } +// if (eventPro.size() > 0) { +// bool found = GDLInterpreter::SearchCompilePro(eventPro, true); // true -> search for procedure +// } } - const DString& GetEventPro() const { return eventPro;}; + const DString& GetEventPro() const { +#ifdef GDL_DEBUG_WIDGETS + wxMessageOutputStderr().Printf(_T("GetEventPro: \"%s\" for %d\n"),eventPro, widgetID); +#endif + return eventPro;}; void SetEventFun( const DString& eFun) { eventFun = StrUpCase(eFun); #ifdef GDL_DEBUG_WIDGETS wxMessageOutputStderr().Printf(_T("SetEventFun: \"%s\" for %d\n"),eventFun, widgetID); #endif - if (eventFun.size() > 0) { - bool found = GDLInterpreter::SearchCompilePro(eventFun, false); // false -> search for function - } +// if (eventFun.size() > 0) { +// bool found = GDLInterpreter::SearchCompilePro(eventFun, false); // false -> search for function +// } } - const DString& GetEventFun() const { return eventFun;} + const DString& GetEventFun() const { +#ifdef GDL_DEBUG_WIDGETS + wxMessageOutputStderr().Printf(_T("GetEventFun: \"%s\" for %d\n"),eventFun, widgetID); +#endif +return eventFun;} void SetNotifyRealize( const DString& eNR) { notifyRealize = StrUpCase( eNR);} const DString& GetNotifyRealize() const { return notifyRealize;} void SetKillNotify( const DString& eKN) { killNotify = StrUpCase( eKN);} @@ -737,13 +745,15 @@ class GDLWidget wxSizer* GetParentSizer() { GDLWidget* gdlParent = GetWidget(parentID); - return gdlParent->GetSizer(); + if (gdlParent) return gdlParent->GetSizer(); + return NULL; } wxSizer* GetSizer() { return widgetSizer;} wxScrolled* GetParentPanel() { GDLWidget* gdlParent = GetWidget(parentID); - return gdlParent->GetPanel(); + if (gdlParent) return gdlParent->GetPanel(); + return NULL; } wxScrolled* GetPanel() { return widgetPanel;} @@ -900,15 +910,16 @@ class GDLWidgetBase: public GDLWidgetContainer bool IsContainer() const final { return true;} bool IsScrolled() const { return scrolled;} long getChildrenAlignment(){return childrenAlignment;} - long getSpace(){return space;} - long getXPad(){return xpad;} - long getYPad(){return ypad;} + int getSpace(){return space;} + int getXPad(){return xpad;} + int getYPad(){return ypad;} virtual void mapBase(bool val); DStructGDL* GetGeometry(wxRealPoint fact=wxRealPoint(1.0,1.0)) final; // wxScrolled* AddBaseFrame(wxScrolled* wxParent, int width=0); wxScrolled* AddXYPad(wxScrolled* wxParent, int xpad=0, int ypad=0); //Apparently children of a base are plotted in reverse order in IDL (last first) void DoReorderColWidgets(int code=0,int style=0, int border=0); + void invertStack(); void ReorderWidgets(); void ReorderForANewWidget(wxWindow* w, int code,int style, int border); // int widgetAlignment(); @@ -1642,8 +1653,8 @@ class GDLWidgetTable: public GDLWidget DLongGDL* rowHeights; DStringGDL* rowLabels; // DLong tabMode; - DLong x_scroll_size_columns; - DLong y_scroll_size_rows; + DLong x_scroll_column; + DLong y_scroll_row; bool updating; //widget is modified by program (avoid sending events) public: @@ -1671,8 +1682,8 @@ class GDLWidgetTable: public GDLWidget DStringGDL* rowLabels_, // DLong tabMode_, BaseGDL* value_, - DLong xScrollSize_, - DLong yScrollSize_, + DLong xScrollColumn_, + DLong yScrollRow_, DStringGDL* valueAsStrings, DULong eventFlags_ ); @@ -1730,14 +1741,18 @@ class GDLWidgetTable: public GDLWidget void SetSelection(DLongGDL* selection); int GetDataNCols(){ - if ( majority == NONE_MAJOR) return vValue->Dim(0); - else if ( majority == COLUMN_MAJOR) return vValue->Dim(0); + SizeT dim0=vValue->Dim(0); if (dim0==0) dim0=1; + SizeT dim1=vValue->Dim(1); if (dim1==0) dim1=1; + if ( majority == NONE_MAJOR) return dim0; + else if ( majority == COLUMN_MAJOR) return dim1; else return static_cast(vValue)->NTags(); } int GetDataNRows(){ - if ( majority == NONE_MAJOR) return vValue->Dim(1); + SizeT dim0=vValue->Dim(0); if (dim0==0) dim0=1; + SizeT dim1=vValue->Dim(1); if (dim1==0) dim1=1; + if ( majority == NONE_MAJOR) return dim1; else if ( majority == COLUMN_MAJOR) return static_cast(vValue)->NTags(); - else return vValue->Dim(0); + else return dim0; } template DString GetRawEditingValue(T1* value, const int nEl, const int n) { diff --git a/src/gdlwidgeteventhandler.cpp b/src/gdlwidgeteventhandler.cpp index 2ef4e4625..9ed710da1 100644 --- a/src/gdlwidgeteventhandler.cpp +++ b/src/gdlwidgeteventhandler.cpp @@ -138,14 +138,14 @@ void wxTextCtrlGDL::OnChar(wxKeyEvent& event ) { if ( event.GetKeyCode( ) == WXK_RETURN ) { //only widg = new DStructGDL( "WIDGET_TEXT_CH" ); widg->InitTag( "ID", DLongGDL( event.GetId( ) ) ); + widg->InitTag( "HANDLER", DLongGDL( event.GetId( ) ) ); widg->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widg->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); widg->InitTag( "TYPE", DIntGDL( 0 ) ); // selection widg->InitTag( "OFFSET", DLongGDL( this->GetInsertionPoint() ) ); widg->InitTag( "CH", DByteGDL( 10 ) ); - GDLWidget::PushEvent( baseWidgetID, widg ); + GDLWidget::PushEvent(widg); if (txt->IsMultiline()) event.Skip( ); - return; + return; } event.Skip( ); return; @@ -171,12 +171,12 @@ void wxTextCtrlGDL::OnChar(wxKeyEvent& event ) { if ( newpos >= 0 && newpos <= this->GetLastPosition( ) ) this->SetInsertionPoint( newpos ); widg = new DStructGDL( "WIDGET_TEXT_SEL" ); widg->InitTag( "ID", DLongGDL( event.GetId( ) ) ); + widg->InitTag( "HANDLER", DLongGDL( event.GetId( ) ) ); widg->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widg->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); widg->InitTag( "TYPE", DIntGDL( 3 ) ); // selection widg->InitTag( "OFFSET", DLongGDL( this->GetInsertionPoint( ) ) ); widg->InitTag( "LENGTH", DLongGDL( 0 ) ); - GDLWidget::PushEvent( baseWidgetID, widg ); + GDLWidget::PushEvent(widg); if (edit) event.Skip( ); return; break; @@ -191,40 +191,48 @@ void wxTextCtrlGDL::OnChar(wxKeyEvent& event ) { if ( oldpos > 0 ) { widg = new DStructGDL( "WIDGET_TEXT_DEL" ); widg->InitTag( "ID", DLongGDL( event.GetId( ) ) ); + widg->InitTag( "HANDLER", DLongGDL( event.GetId( ) ) ); widg->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widg->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); widg->InitTag( "TYPE", DIntGDL( 2 ) ); // selection widg->InitTag( "OFFSET", DLongGDL( from - 1 ) ); widg->InitTag( "LENGTH", DLongGDL( to - from + 1 ) ); - GDLWidget::PushEvent( baseWidgetID, widg ); + GDLWidget::PushEvent(widg); } if (edit) event.Skip( ); //do it! return; break; case WXK_TAB: + //On UNIX,For single-line text widgets, the value of the TAB_MODE keyword is ignored, keypress events are not generated for the Tab key, and tab characters are not inserted into the text field. + if (!(txt->IsMultiline())) return; widg = new DStructGDL( "WIDGET_TEXT_CH" ); widg->InitTag( "ID", DLongGDL( event.GetId( ) ) ); + widg->InitTag( "HANDLER", DLongGDL( event.GetId( ) ) ); widg->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widg->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); widg->InitTag( "TYPE", DIntGDL( 0 ) ); // selection widg->InitTag( "OFFSET", DLongGDL( from ) ); widg->InitTag( "CH", DByteGDL( 9 ) ); - GDLWidget::PushEvent( baseWidgetID, widg ); - //if (!edit) event.Skip( ); //do NOT skip - if (edit) this->WriteText("\t"); //necessary! + GDLWidget::PushEvent(widg); + if (edit) { + this->SetEvtHandlerEnabled(false); + this->WriteText("\t"); //necessary! + this->SetEvtHandlerEnabled(true); + } return; break; case WXK_RETURN: //important, *DL returns CH=10 instead of 13 for widg = new DStructGDL( "WIDGET_TEXT_CH" ); widg->InitTag( "ID", DLongGDL( event.GetId( ) ) ); + widg->InitTag( "HANDLER", DLongGDL( event.GetId( ) ) ); widg->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widg->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); widg->InitTag( "TYPE", DIntGDL( 0 ) ); // selection widg->InitTag( "OFFSET", DLongGDL( from ) ); widg->InitTag( "CH", DByteGDL( 10 ) ); - GDLWidget::PushEvent( baseWidgetID, widg ); - if (edit) this->WriteText("\n"); //necessary! - if (edit) event.Skip( ); + GDLWidget::PushEvent(widg); + if (edit && txt->IsMultiline()) { + this->SetEvtHandlerEnabled(false); + this->WriteText("\n"); //necessary! + this->SetEvtHandlerEnabled(true); + } return; break; } @@ -233,13 +241,13 @@ void wxTextCtrlGDL::OnChar(wxKeyEvent& event ) { //else return a CHAR event for most keys (as *DL) widg = new DStructGDL( "WIDGET_TEXT_CH" ); - widg->InitTag( "ID", DLongGDL( event.GetId( ) ) ); + widg->InitTag( "ID", DLongGDL( event.GetId( ) ) ); + widg->InitTag( "HANDLER", DLongGDL( event.GetId( ) ) ); widg->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widg->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); widg->InitTag( "TYPE", DIntGDL( 0 ) ); // single char widg->InitTag( "OFFSET", DLongGDL( this->GetInsertionPoint( ) ) ); widg->InitTag( "CH", DByteGDL( event.GetKeyCode( ) ) ); - GDLWidget::PushEvent( baseWidgetID, widg ); + GDLWidget::PushEvent(widg); if (edit) event.Skip( ); } @@ -271,7 +279,7 @@ void wxTextCtrlGDL::OnMouseEvents( wxMouseEvent& event) widgtimer->InitTag( "ID", DLongGDL(*originating_id)); widgtimer->InitTag( "TOP", DLongGDL( baseWidgetID ) ); widgtimer->InitTag( "HANDLER", DLongGDL(*originating_id) ); //will be set elsewhere - GDLWidget::PushEvent( baseWidgetID, widgtimer); + GDLWidget::PushEvent(widgtimer); } void gdlwxFrame::OnShowRequest( wxCommandEvent& event) @@ -349,11 +357,11 @@ void gdlwxFrame::OnButton( wxCommandEvent& event) // create GDL event struct DStructGDL* widgbut = new DStructGDL( "WIDGET_BUTTON"); widgbut->InitTag("ID", DLongGDL( event.GetId())); + widgbut->InitTag("HANDLER", DLongGDL( event.GetId())); widgbut->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgbut->InitTag("HANDLER", DLongGDL( baseWidgetID )); widgbut->InitTag("SELECT", DLongGDL( 1)); - GDLWidget::PushEvent( baseWidgetID, widgbut); + GDLWidget::PushEvent(widgbut); } //change bitmap or parent button? @@ -367,11 +375,11 @@ void gdlwxFrame::OnMenu( wxCommandEvent& event) // create GDL event struct DStructGDL* widgbut = new DStructGDL( "WIDGET_BUTTON"); widgbut->InitTag("ID", DLongGDL( event.GetId())); + widgbut->InitTag("HANDLER", DLongGDL( event.GetId())); widgbut->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgbut->InitTag("HANDLER", DLongGDL( baseWidgetID )); widgbut->InitTag("SELECT", DLongGDL( 1)); - GDLWidget::PushEvent( baseWidgetID, widgbut); + GDLWidget::PushEvent(widgbut); } void gdlwxFrame::OnRadioButton( wxCommandEvent& event) @@ -395,21 +403,21 @@ void gdlwxFrame::OnRadioButton( wxCommandEvent& event) DStructGDL* widgbut = new DStructGDL( "WIDGET_BUTTON"); widgbut->InitTag("ID", DLongGDL( lastSelection)); widgbut->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgbut->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widgbut->InitTag("HANDLER", DLongGDL( event.GetId() )); widgbut->InitTag("SELECT", DLongGDL( 0)); GDLWidgetNormalButton* widget = dynamic_cast(GDLWidget::GetWidget( lastSelection)); assert(widget!=NULL); widget->SetRadioButton( false); - if (widget->getNoReleaseOption()==false) GDLWidget::PushEvent( baseWidgetID, widgbut); + if (widget->getNoReleaseOption()==false) GDLWidget::PushEvent(widgbut); } // create GDL event struct DStructGDL* widgbut = new DStructGDL( "WIDGET_BUTTON"); widgbut->InitTag("ID", DLongGDL( event.GetId())); + widgbut->InitTag("HANDLER", DLongGDL( event.GetId())); widgbut->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgbut->InitTag("HANDLER", DLongGDL( baseWidgetID )); // widgbut->InitTag("SELECT", DLongGDL( selectValue ? 1 : 0)); widgbut->InitTag("SELECT", DLongGDL( 1)); @@ -418,7 +426,7 @@ void gdlwxFrame::OnRadioButton( wxCommandEvent& event) GDLWidget* widget = GDLWidget::GetWidget( event.GetId()); assert(widget->IsButton()); static_cast(widget)->SetRadioButton( true); - GDLWidget::PushEvent( baseWidgetID, widgbut); + GDLWidget::PushEvent(widgbut); } void gdlwxFrame::OnCheckBox( wxCommandEvent& event) @@ -440,11 +448,11 @@ void gdlwxFrame::OnCheckBox( wxCommandEvent& event) // create GDL event struct DStructGDL* widgbut = new DStructGDL( "WIDGET_BUTTON"); widgbut->InitTag("ID", DLongGDL( event.GetId())); + widgbut->InitTag("HANDLER", DLongGDL( event.GetId())); widgbut->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgbut->InitTag("HANDLER", DLongGDL( baseWidgetID )); widgbut->InitTag("SELECT", DLongGDL( selectValue ? 1 : 0)); - GDLWidget::PushEvent( baseWidgetID, widgbut); + GDLWidget::PushEvent(widgbut); } void gdlwxFrame::OnComboBox( wxCommandEvent& event) @@ -463,11 +471,11 @@ void gdlwxFrame::OnComboBox( wxCommandEvent& event) widgcbox = new DStructGDL( "WIDGET_COMBOBOX"); widgcbox->InitTag("ID", DLongGDL( event.GetId())); widgcbox->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgcbox->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widgcbox->InitTag("HANDLER", DLongGDL( event.GetId() )); widgcbox->InitTag("INDEX", DLongGDL( selectValue)); widgcbox->InitTag("STR", DStringGDL( std::string(strValue.mb_str(wxConvUTF8)) )); - GDLWidget::PushEvent( baseWidgetID, widgcbox); + GDLWidget::PushEvent(widgcbox); } void gdlwxFrame::OnComboBoxTextEnter( wxCommandEvent& event) { @@ -485,11 +493,11 @@ void gdlwxFrame::OnComboBoxTextEnter( wxCommandEvent& event) widgcbox = new DStructGDL( "WIDGET_COMBOBOX"); widgcbox->InitTag("ID", DLongGDL( event.GetId())); widgcbox->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgcbox->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widgcbox->InitTag("HANDLER", DLongGDL( event.GetId() )); widgcbox->InitTag("INDEX", DLongGDL( selectValue)); widgcbox->InitTag("STR", DStringGDL( std::string(strValue.mb_str(wxConvUTF8)) )); - GDLWidget::PushEvent( baseWidgetID, widgcbox); + GDLWidget::PushEvent(widgcbox); } void gdlwxFrame::OnDropList( wxCommandEvent& event) { @@ -504,10 +512,10 @@ void gdlwxFrame::OnDropList( wxCommandEvent& event) widdrplst = new DStructGDL( "WIDGET_DROPLIST"); widdrplst->InitTag("ID", DLongGDL( event.GetId())); widdrplst->InitTag("TOP", DLongGDL( baseWidgetID)); -// widdrplst->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widdrplst->InitTag("HANDLER", DLongGDL( event.GetId() )); widdrplst->InitTag("INDEX", DLongGDL( selectValue)); - GDLWidget::PushEvent( baseWidgetID, widdrplst); + GDLWidget::PushEvent(widdrplst); } void gdlwxFrame::OnListBoxDo( wxCommandEvent& event, DLong clicks) @@ -525,11 +533,11 @@ void gdlwxFrame::OnListBoxDo( wxCommandEvent& event, DLong clicks) widgcbox = new DStructGDL( "WIDGET_LIST"); widgcbox->InitTag("ID", DLongGDL( event.GetId())); widgcbox->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgcbox->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widgcbox->InitTag("HANDLER", DLongGDL( event.GetId() )); widgcbox->InitTag("INDEX", DLongGDL( selectValue)); widgcbox->InitTag("CLICKS", DLongGDL( clicks)); - GDLWidget::PushEvent( baseWidgetID, widgcbox); + GDLWidget::PushEvent(widgcbox); } void gdlwxFrame::OnListBox( wxCommandEvent& event) { @@ -581,12 +589,12 @@ void gdlwxFrame::OnTextMouseEvents( wxMouseEvent& event) // widg = new DStructGDL( "WIDGET_TEXT_SEL"); // widg->InitTag("ID", DLongGDL( event.GetId())); // widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); +// widg->InitTag("HANDLER", DLongGDL( event.GetId() )); // widg->InitTag("TYPE", DIntGDL( 3)); // selection // from=textCtrl->GetInsertionPoint(); // widg->InitTag("OFFSET", DLongGDL( from )); // widg->InitTag("LENGTH", DLongGDL( 0 )); -// GDLWidget::PushEvent( baseWidgetID, widg); +// GDLWidget::PushEvent(widg); // NOTE: "Process a wxEVT_LEFT_DOWN event. The handler of this event should normally call event.Skip() // to allow the default processing to take place as otherwise the window under mouse wouldn't get the focus. " event.Skip( ); @@ -594,12 +602,12 @@ void gdlwxFrame::OnTextMouseEvents( wxMouseEvent& event) widg = new DStructGDL( "WIDGET_TEXT_SEL"); widg->InitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("TYPE", DIntGDL( 3)); // selection textCtrl->GetSelection(&from,&to); widg->InitTag("OFFSET", DLongGDL( from )); widg->InitTag("LENGTH", DLongGDL( to-from )); - GDLWidget::PushEvent( baseWidgetID, widg); + GDLWidget::PushEvent(widg); // NOTE: "Process a wxEVT_LEFT_DOWN event. The handler of this event should normally call event.Skip() // to allow the default processing to take place as otherwise the window under mouse wouldn't get the focus. " // if (event.ButtonDown(wxMOUSE_BTN_LEFT)) @@ -646,11 +654,11 @@ void gdlwxFrame::OnTextPaste( wxClipboardTextEvent& event) widg = new DStructGDL( "WIDGET_TEXT_STR"); widg->InitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("TYPE", DIntGDL( 1)); // multiple char widg->InitTag("OFFSET", DLongGDL( pos+s.length() )); widg->InitTag("STR", DStringGDL(std::string(s.c_str()))); - GDLWidget::PushEvent( baseWidgetID, widg); + GDLWidget::PushEvent(widg); if (edit) event.Skip(); return; } @@ -764,7 +772,7 @@ void gdlwxFrame::OnText( wxCommandEvent& event) widg = new DStructGDL( "WIDGET_TEXT_SEL"); widg->InitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("TYPE", DIntGDL( 3)); // selection widg->InitTag("OFFSET", DLongGDL( offset)); widg->InitTag("LENGTH", DLongGDL( selEnd-selStart)); @@ -777,7 +785,7 @@ void gdlwxFrame::OnText( wxCommandEvent& event) widg = new DStructGDL( "WIDGET_TEXT_DEL"); widg->InitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("TYPE", DIntGDL( 2)); // delete widg->InitTag("OFFSET", DLongGDL( offset-1)); widg->InitTag("LENGTH", DLongGDL( -lengthDiff)); @@ -788,18 +796,18 @@ void gdlwxFrame::OnText( wxCommandEvent& event) widg = new DStructGDL( "WIDGET_TEXT_DEL"); widg->InitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("TYPE", DIntGDL( 2)); // delete widg->InitTag("OFFSET", DLongGDL( 0)); widg->InitTag("LENGTH", DLongGDL( lastValue.length())); - GDLWidget::PushEvent( baseWidgetID, widg); + GDLWidget::PushEvent(widg); // 2nd insert new widg = new DStructGDL( "WIDGET_TEXT_STR"); widg->InitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("TYPE", DIntGDL( 1)); // multiple char widg->InitTag("OFFSET", DLongGDL( 0)); widg->InitTag("STR", DStringGDL( newValue)); @@ -809,7 +817,7 @@ void gdlwxFrame::OnText( wxCommandEvent& event) widg = new DStructGDL( "WIDGET_TEXT_CH"); widg->InitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("TYPE", DIntGDL( 0)); // single char widg->InitTag("OFFSET", DLongGDL( offset+1)); widg->InitTag("CH", DByteGDL( newValue[offsetInitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("TYPE", DIntGDL( 1)); // multiple char widg->InitTag("OFFSET", DLongGDL( offset)); widg->InitTag("STR", DStringGDL( str)); } } - GDLWidget::PushEvent( baseWidgetID, widg); + GDLWidget::PushEvent(widg); } void gdlwxFrame::OnTextEnter( wxCommandEvent& event) @@ -875,12 +883,12 @@ void gdlwxFrame::OnTextEnter( wxCommandEvent& event) // widg = new DStructGDL( "WIDGET_TEXT_CH"); // widg->InitTag("ID", DLongGDL( event.GetId())); // widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); +// widg->InitTag("HANDLER", DLongGDL( event.GetId() )); // widg->InitTag("TYPE", DIntGDL( 0)); // single char // widg->InitTag("OFFSET", DLongGDL( offset)); // widg->InitTag("CH", DByteGDL( '\n')); // -// GDLWidget::PushEvent( baseWidgetID, widg); +// GDLWidget::PushEvent(widg); // } // else if( widget->IsComboBox()) // { @@ -892,11 +900,11 @@ void gdlwxFrame::OnTextEnter( wxCommandEvent& event) // widgcbox = new DStructGDL( "WIDGET_COMBOBOX"); // widgcbox->InitTag("ID", DLongGDL( event.GetId())); // widgcbox->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgcbox->InitTag("HANDLER", DLongGDL( baseWidgetID )); +// widgcbox->InitTag("HANDLER", DLongGDL( event.GetId() )); // widgcbox->InitTag("INDEX", DLongGDL( selectValue)); // widgcbox->InitTag("STR", DStringGDL( std::string(strValue.mb_str(wxConvUTF8)) )); // -// GDLWidget::PushEvent( baseWidgetID, widgcbox); +// GDLWidget::PushEvent(widgcbox); // } // else // { @@ -907,10 +915,10 @@ void gdlwxFrame::OnTextEnter( wxCommandEvent& event) // widdrplst = new DStructGDL( "WIDGET_DROPLIST"); // widdrplst->InitTag("ID", DLongGDL( event.GetId())); // widdrplst->InitTag("TOP", DLongGDL( baseWidgetID)); -// widdrplst->InitTag("HANDLER", DLongGDL( baseWidgetID )); +// widdrplst->InitTag("HANDLER", DLongGDL( event.GetId() )); // widdrplst->InitTag("INDEX", DLongGDL( selectValue)); // -// GDLWidget::PushEvent( baseWidgetID, widdrplst); +// GDLWidget::PushEvent(widdrplst); // } //} @@ -927,10 +935,10 @@ void gdlwxFrame::OnPageChanged( wxNotebookEvent& event) widg = new DStructGDL( "WIDGET_TAB"); widg->InitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("TAB", DLongGDL( event.GetSelection())); - GDLWidget::PushEvent( baseWidgetID, widg); + GDLWidget::PushEvent(widg); event.Skip(); } @@ -1040,10 +1048,10 @@ void gdlwxFrame::OnTimerResize(wxTimerEvent& event) DStructGDL* widgbase = new DStructGDL("WIDGET_BASE"); widgbase->InitTag("ID", DLongGDL(owner->GetWidgetID())); widgbase->InitTag("TOP", DLongGDL(baseWidgetID)); -// widgbase->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widgbase->InitTag("HANDLER", DLongGDL(event.GetId())); widgbase->InitTag("X", DLongGDL(frameSize.x)); widgbase->InitTag("Y", DLongGDL(frameSize.y)); - GDLWidget::PushEvent(baseWidgetID, widgbase); + GDLWidget::PushEvent(widgbase); } else event.Skip(); } @@ -1102,10 +1110,10 @@ void gdlwxFrame::OnSizeWithTimer(wxSizeEvent& event) DStructGDL* widgbase = new DStructGDL("WIDGET_BASE"); widgbase->InitTag("ID", DLongGDL(event.GetId())); widgbase->InitTag("TOP", DLongGDL(baseWidgetID)); -// widgbase->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widgbase->InitTag("HANDLER", DLongGDL(event.GetId())); widgbase->InitTag("X", DLongGDL(frameSize.x)); widgbase->InitTag("Y", DLongGDL(frameSize.y)); - GDLWidget::PushEvent(baseWidgetID, widgbase); + GDLWidget::PushEvent(widgbase); } else event.Skip(); //important, pass to others! #if (GDL_DEBUG_ALL_EVENTS || GDL_DEBUG_SIZE_EVENTS) wxMessageOutputStderr().Printf(_T("Processed.\n")); @@ -1136,11 +1144,11 @@ void gdlwxFrame::OnThumbTrack( wxScrollEvent& event) widg = new DStructGDL( "WIDGET_SLIDER"); widg->InitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("VALUE", DLongGDL( newSelection)); widg->InitTag("DRAG", DIntGDL( 1)); // dragging events from here - GDLWidget::PushEvent( baseWidgetID, widg); + GDLWidget::PushEvent(widg); } return; // } @@ -1168,11 +1176,11 @@ void gdlwxFrame::OnThumbRelease( wxScrollEvent& event) widg = new DStructGDL( "WIDGET_SLIDER"); widg->InitTag("ID", DLongGDL( event.GetId())); widg->InitTag("TOP", DLongGDL( baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widg->InitTag("HANDLER", DLongGDL( event.GetId() )); widg->InitTag("VALUE", DLongGDL( newSelection)); widg->InitTag("DRAG", DIntGDL( 0)); // set events from here - GDLWidget::PushEvent( baseWidgetID, widg); + GDLWidget::PushEvent(widg); return; } @@ -1192,9 +1200,9 @@ void gdlwxFrame::OnEnterWindow( wxMouseEvent &event ) { DStructGDL* widgtracking = new DStructGDL( "WIDGET_TRACKING" ); widgtracking->InitTag( "ID", DLongGDL( event.GetId( ) ) ); widgtracking->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgtracking->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgtracking->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); widgtracking->InitTag( "ENTER", DIntGDL( 1 ) ); - GDLWidget::PushEvent( baseWidgetID, widgtracking ); + GDLWidget::PushEvent(widgtracking); } event.Skip(); } @@ -1213,9 +1221,9 @@ void gdlwxFrame::OnLeaveWindow( wxMouseEvent &event ) { DStructGDL* widgtracking = new DStructGDL( "WIDGET_TRACKING" ); widgtracking->InitTag( "ID", DLongGDL( event.GetId( ) ) ); widgtracking->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgtracking->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgtracking->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); widgtracking->InitTag( "ENTER", DIntGDL( 0 ) ); - GDLWidget::PushEvent( baseWidgetID, widgtracking ); + GDLWidget::PushEvent(widgtracking); } event.Skip(); } @@ -1231,10 +1239,10 @@ void gdlwxFrame::OnKBRDFocusChange( wxFocusEvent &event ) { DStructGDL* widgkbrdfocus = new DStructGDL( "WIDGET_KBRD_FOCUS" ); widgkbrdfocus->InitTag( "ID", DLongGDL( event.GetId( ) ) ); widgkbrdfocus->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgkbrdfocus->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgkbrdfocus->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); int enter=(event.GetEventType()==wxEVT_SET_FOCUS); widgkbrdfocus->InitTag( "ENTER", DIntGDL( enter ) ); - GDLWidget::PushEvent( baseWidgetID, widgkbrdfocus ); + GDLWidget::PushEvent(widgkbrdfocus); } event.Skip(); //"The focus event handlers should almost invariably call wxEvent::Skip() on their event argument to allow the default handling to take place. Failure to do this may result in incorrect behaviour of the native controls." } @@ -1266,7 +1274,7 @@ void gdlwxFrame::OnContextEvent( wxContextMenuEvent& event) { DStructGDL* widgcontext = new DStructGDL( "WIDGET_CONTEXT" ); widgcontext->InitTag( "ID", DLongGDL( eventID ) ); widgcontext->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgcontext->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgcontext->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); wxPoint position=event.GetPosition( ); if (position==wxDefaultPosition) { //issued from keyboard position=wxGetMousePosition(); @@ -1281,7 +1289,7 @@ void gdlwxFrame::OnContextEvent( wxContextMenuEvent& event) { widgcontext->InitTag( "ROW", DLongGDL( row ) ); widgcontext->InitTag( "COL", DLongGDL( col ) ); } - GDLWidget::PushEvent( baseWidgetID, widgcontext ); + GDLWidget::PushEvent(widgcontext); } else event.Skip();//normal end of event processing! } @@ -1301,9 +1309,9 @@ void gdlwxFrame::OnIconize( wxIconizeEvent & event) DStructGDL* widgtlb_iconify_events = new DStructGDL( "WIDGET_TLB_ICONIFY" ); widgtlb_iconify_events->InitTag( "ID", DLongGDL( event.GetId( ) ) ); widgtlb_iconify_events->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgtlb_iconify_events->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgtlb_iconify_events->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); widgtlb_iconify_events->InitTag( "ICONIFIED", DIntGDL( event.IsIconized() ) ); - GDLWidget::PushEvent( baseWidgetID, widgtlb_iconify_events ); + GDLWidget::PushEvent(widgtlb_iconify_events); } else event.Skip(); } @@ -1324,10 +1332,10 @@ void gdlwxFrame::OnMove( wxMoveEvent & event) DStructGDL* widgtlb_move_events = new DStructGDL( "WIDGET_TLB_MOVE" ); widgtlb_move_events->InitTag( "ID", DLongGDL( event.GetId( ) ) ); widgtlb_move_events->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgtlb_move_events->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgtlb_move_events->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); widgtlb_move_events->InitTag( "X", DLongGDL( event.GetPosition().x ) ); widgtlb_move_events->InitTag( "Y", DLongGDL( event.GetPosition().y ) ); - GDLWidget::PushEvent( baseWidgetID, widgtlb_move_events ); + GDLWidget::PushEvent(widgtlb_move_events); } else event.Skip(); } @@ -1348,8 +1356,8 @@ void gdlwxFrame::OnCloseFrame( wxCloseEvent & event) { DStructGDL* widgtlb_kill_request_events = new DStructGDL("WIDGET_KILL_REQUEST"); widgtlb_kill_request_events->InitTag("ID", DLongGDL(event.GetId())); widgtlb_kill_request_events->InitTag("TOP", DLongGDL(baseWidgetID)); -// widgtlb_kill_request_events->InitTag("HANDLER", DLongGDL(baseWidgetID)); - GDLWidget::PushEvent(baseWidgetID, widgtlb_kill_request_events); + widgtlb_kill_request_events->InitTag("HANDLER", DLongGDL(event.GetId())); + GDLWidget::PushEvent(widgtlb_kill_request_events); } void gdlwxFrame::OnUnhandledCloseFrame( wxCloseEvent & event) @@ -1477,7 +1485,7 @@ void gdlwxDrawPanel::OnFakeDropFileEvent(wxDropFilesEvent& event){ DStructGDL* drawdrop = new DStructGDL("WIDGET_DROP"); drawdrop->InitTag("ID", DLongGDL( myWidgetDraw->GetWidgetID() )); //ID of the destination drawdrop->InitTag("TOP", DLongGDL(baseWidgetID)); -// drawdrop->InitTag("HANDLER", DLongGDL(baseWidgetID)); + drawdrop->InitTag("HANDLER", DLongGDL(event.GetId())); drawdrop->InitTag("DRAG_ID", DLongGDL(droppedID)); // ID of the source drawdrop->InitTag("POSITION", DIntGDL(1)); // 1 above 2 on 4 below destination widget wxPoint where=CalcUnscrolledPosition(event.GetPosition()); @@ -1485,7 +1493,7 @@ void gdlwxDrawPanel::OnFakeDropFileEvent(wxDropFilesEvent& event){ drawdrop->InitTag("Y", DLongGDL(drawSize.y-where.y)); drawdrop->InitTag("MODIFIERS", DIntGDL(GetModifiers())); //mask with 1 shift 2 control 4 caps lock 8 alt // insert into structList - GDLWidget::PushEvent(baseWidgetID, drawdrop); + GDLWidget::PushEvent(drawdrop); } } @@ -1497,13 +1505,13 @@ void gdlwxDrawPanel::OnMouseMove( wxMouseEvent &event ) { DStructGDL* widgdraw = new DStructGDL( "WIDGET_DRAW" ); widgdraw->InitTag( "ID", DLongGDL( myWidgetDraw->GetWidgetID() ) ); widgdraw->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgdraw->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgdraw->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); widgdraw->InitTag( "TYPE", DIntGDL( 2 ) ); //motion wxPoint where=WhereIsMouse(event); widgdraw->InitTag( "X", DLongGDL( where.x ) ); widgdraw->InitTag( "Y", DLongGDL( drawSize.y-where.y ) ); widgdraw->InitTag( "MODIFIERS", DLongGDL( RemapModifiers(event))); - GDLWidget::PushEvent( baseWidgetID, widgdraw ); + GDLWidget::PushEvent(widgdraw); } else event.Skip(); //normal end of event processing! #if (GDL_DEBUG_ALL_EVENTS || GDL_DEBUG_MOVE_EVENTS) wxMessageOutputStderr().Printf(_T("in gdlwxDrawPanel::OnMouseMove: ID=%d, coords=(%d,%d)\n"),event.GetId(),event.GetX(),event.GetY()); @@ -1521,7 +1529,7 @@ void gdlwxDrawPanel::OnMouseDown( wxMouseEvent &event ) { DStructGDL* widgdraw = new DStructGDL( "WIDGET_DRAW" ); widgdraw->InitTag( "ID", DLongGDL( myWidgetDraw->GetWidgetID() ) ); widgdraw->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgdraw->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgdraw->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); widgdraw->InitTag( "TYPE", DIntGDL( 0 ) ); //button Press wxPoint where=WhereIsMouse(event); widgdraw->InitTag( "X", DLongGDL( where.x ) ); @@ -1533,7 +1541,7 @@ void gdlwxDrawPanel::OnMouseDown( wxMouseEvent &event ) { widgdraw->InitTag( "MODIFIERS", DLongGDL( RemapModifiers(event))); widgdraw->InitTag( "CH", DByteGDL( 0 ) ); widgdraw->InitTag( "KEY", DLongGDL( 0 ) ); - GDLWidget::PushEvent( baseWidgetID, widgdraw ); + GDLWidget::PushEvent(widgdraw); } else event.Skip(); //normal end of event processing! } @@ -1549,7 +1557,7 @@ void gdlwxDrawPanel::OnMouseDownDble(wxMouseEvent &event) { DStructGDL* widgdraw = new DStructGDL("WIDGET_DRAW"); widgdraw->InitTag("ID", DLongGDL(myWidgetDraw->GetWidgetID())); widgdraw->InitTag("TOP", DLongGDL(baseWidgetID)); -// widgdraw->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widgdraw->InitTag("HANDLER", DLongGDL(event.GetId())); widgdraw->InitTag("TYPE", DIntGDL(0)); //button Press wxPoint where = WhereIsMouse(event); widgdraw->InitTag("X", DLongGDL(where.x)); @@ -1561,7 +1569,7 @@ void gdlwxDrawPanel::OnMouseDownDble(wxMouseEvent &event) { widgdraw->InitTag("MODIFIERS", DLongGDL(RemapModifiers(event))); widgdraw->InitTag("CH", DByteGDL(0)); widgdraw->InitTag("KEY", DLongGDL(0)); - GDLWidget::PushEvent(baseWidgetID, widgdraw); + GDLWidget::PushEvent(widgdraw); } else event.Skip(); //normal end of event processing! } @@ -1576,7 +1584,7 @@ void gdlwxDrawPanel::OnMouseUp( wxMouseEvent &event ) { DStructGDL* widgdraw = new DStructGDL( "WIDGET_DRAW" ); widgdraw->InitTag( "ID", DLongGDL( myWidgetDraw->GetWidgetID() ) ); widgdraw->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgdraw->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgdraw->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); widgdraw->InitTag( "TYPE", DIntGDL( 1 ) ); //button Release wxPoint where=WhereIsMouse(event); widgdraw->InitTag( "X", DLongGDL( where.x ) ); @@ -1588,7 +1596,7 @@ void gdlwxDrawPanel::OnMouseUp( wxMouseEvent &event ) { widgdraw->InitTag( "MODIFIERS", DLongGDL( RemapModifiers(event))); widgdraw->InitTag( "CH", DByteGDL( 0 ) ); widgdraw->InitTag( "KEY", DLongGDL( 0 ) ); - GDLWidget::PushEvent( baseWidgetID, widgdraw ); + GDLWidget::PushEvent(widgdraw); } else event.Skip();//normal end of event processing! } @@ -1603,7 +1611,7 @@ void gdlwxDrawPanel::OnMouseWheel( wxMouseEvent &event ) { DStructGDL* widgdraw = new DStructGDL( "WIDGET_DRAW" ); widgdraw->InitTag( "ID", DLongGDL( myWidgetDraw->GetWidgetID() ) ); widgdraw->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgdraw->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgdraw->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); widgdraw->InitTag( "TYPE", DIntGDL( 7 ) ); //wheel event wxPoint where=WhereIsMouse(event); widgdraw->InitTag( "X", DLongGDL( where.x ) ); @@ -1614,7 +1622,7 @@ void gdlwxDrawPanel::OnMouseWheel( wxMouseEvent &event ) { widgdraw->InitTag( "MODIFIERS", DLongGDL(RemapModifiers(event)) ); widgdraw->InitTag( "CH", DByteGDL( 0 ) ); widgdraw->InitTag( "KEY", DLongGDL( 0 ) ); - GDLWidget::PushEvent( baseWidgetID, widgdraw ); + GDLWidget::PushEvent(widgdraw); } else event.Skip(); } @@ -1632,7 +1640,7 @@ void gdlwxDrawPanel::OnKey( wxKeyEvent &event ) { DStructGDL* widgdraw = new DStructGDL( "WIDGET_DRAW" ); widgdraw->InitTag( "ID", DLongGDL( myWidgetDraw->GetWidgetID() ) ); widgdraw->InitTag( "TOP", DLongGDL( baseWidgetID ) ); -// widgdraw->InitTag( "HANDLER", DLongGDL( baseWidgetID ) ); + widgdraw->InitTag( "HANDLER", DLongGDL( event.GetId() ) ); wxPoint where=WhereIsMouse(event); widgdraw->InitTag( "X", DLongGDL( where.x ) ); widgdraw->InitTag( "Y", DLongGDL( drawSize.y-where.y ) ); @@ -1673,7 +1681,7 @@ void gdlwxDrawPanel::OnKey( wxKeyEvent &event ) { widgdraw->InitTag( "KEY", DLongGDL( start ) ); widgdraw->InitTag( "CH", DByteGDL( 0 ) ); widgdraw->InitTag( "MODIFIERS", DLongGDL(0) ); - GDLWidget::PushEvent( baseWidgetID, widgdraw ); + GDLWidget::PushEvent(widgdraw); } break; default: @@ -1681,7 +1689,7 @@ void gdlwxDrawPanel::OnKey( wxKeyEvent &event ) { widgdraw->InitTag( "KEY", DLongGDL( 0 ) ); widgdraw->InitTag( "CH", DByteGDL( event.GetKeyCode() & 0xFF ) ); widgdraw->InitTag( "MODIFIERS", DLongGDL(GetModifiers() ) ); - GDLWidget::PushEvent( baseWidgetID, widgdraw ); + GDLWidget::PushEvent(widgdraw); } } else event.Skip(); } @@ -1697,12 +1705,12 @@ void wxGridGDL::OnTableRowResizing(wxGridSizeEvent & event){ DStructGDL* widgtablerowheight = new DStructGDL( "WIDGET_TABLE_ROW_HEIGHT"); widgtablerowheight->InitTag("ID", DLongGDL( event.GetId( ) )); widgtablerowheight->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgtablerowheight->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widgtablerowheight->InitTag("HANDLER", DLongGDL( event.GetId() )); widgtablerowheight->InitTag("TYPE", DIntGDL(6)); // 6 widgtablerowheight->InitTag("ROW", DLongGDL( event.GetRowOrCol() )); widgtablerowheight->InitTag("HEIGHT",DLongGDL( this->GetRowSize(event.GetRowOrCol()))); // insert into structList - GDLWidget::PushEvent( baseWidgetID, widgtablerowheight ); + GDLWidget::PushEvent(widgtablerowheight); } else event.Skip(); } @@ -1717,12 +1725,12 @@ void wxGridGDL::OnTableColResizing(wxGridSizeEvent & event){ DStructGDL* widgtablerowheight = new DStructGDL( "WIDGET_TABLE_COL_WIDTH"); widgtablerowheight->InitTag("ID", DLongGDL( event.GetId( ) )); widgtablerowheight->InitTag("TOP", DLongGDL( baseWidgetID)); -// widgtablerowheight->InitTag("HANDLER", DLongGDL( baseWidgetID )); + widgtablerowheight->InitTag("HANDLER", DLongGDL( event.GetId() )); widgtablerowheight->InitTag("TYPE", DIntGDL(7)); // 7 widgtablerowheight->InitTag("COL", DLongGDL( event.GetRowOrCol() )); widgtablerowheight->InitTag("WIDTH",DLongGDL( this->GetColSize(event.GetRowOrCol()))); // insert into structList - GDLWidget::PushEvent( baseWidgetID, widgtablerowheight ); + GDLWidget::PushEvent(widgtablerowheight); } else event.Skip(); } @@ -1748,13 +1756,13 @@ void wxGridGDL::OnTableRangeSelection(wxGridRangeSelectEvent & event) { DStructGDL* widgtablecelsel2 = new DStructGDL("WIDGET_TABLE_CELL_SEL"); //sel widgtablecelsel2->InitTag("ID", DLongGDL(event.GetId())); widgtablecelsel2->InitTag("TOP", DLongGDL(baseWidgetID)); -// widgtablecelsel2->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widgtablecelsel2->InitTag("HANDLER", DLongGDL(event.GetId())); widgtablecelsel2->InitTag("TYPE", DIntGDL(4)); // 4 or 9 widgtablecelsel2->InitTag("SEL_LEFT", DLongGDL(lc)); widgtablecelsel2->InitTag("SEL_TOP", DLongGDL(tr)); widgtablecelsel2->InitTag("SEL_RIGHT", DLongGDL(rc)); widgtablecelsel2->InitTag("SEL_BOTTOM", DLongGDL(br)); - GDLWidget::PushEvent(baseWidgetID, widgtablecelsel2); + GDLWidget::PushEvent(widgtablecelsel2); } } else { //remove if (table->GetDisjointSelection() && event.ControlDown()) { @@ -1762,26 +1770,26 @@ void wxGridGDL::OnTableRangeSelection(wxGridRangeSelectEvent & event) { DStructGDL* widgtablecelsel = new DStructGDL("WIDGET_TABLE_CELL_DESEL"); widgtablecelsel->InitTag("ID", DLongGDL(event.GetId())); widgtablecelsel->InitTag("TOP", DLongGDL(baseWidgetID)); -// widgtablecelsel->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widgtablecelsel->InitTag("HANDLER", DLongGDL(event.GetId())); widgtablecelsel->InitTag("TYPE", DIntGDL(9)); // 9 widgtablecelsel->InitTag("SEL_LEFT", DLongGDL(lc)); widgtablecelsel->InitTag("SEL_TOP", DLongGDL(tr)); widgtablecelsel->InitTag("SEL_RIGHT", DLongGDL(rc)); widgtablecelsel->InitTag("SEL_BOTTOM", DLongGDL(br)); - GDLWidget::PushEvent(baseWidgetID, widgtablecelsel); + GDLWidget::PushEvent(widgtablecelsel); } } else { if (!(table->GetDisjointSelection() && event.ControlDown())) { //desel DStructGDL* widgtablecelsel = new DStructGDL("WIDGET_TABLE_CELL_SEL"); widgtablecelsel->InitTag("ID", DLongGDL(event.GetId())); widgtablecelsel->InitTag("TOP", DLongGDL(baseWidgetID)); -// widgtablecelsel->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widgtablecelsel->InitTag("HANDLER", DLongGDL(event.GetId())); widgtablecelsel->InitTag("TYPE", DIntGDL(4)); // widgtablecelsel->InitTag("SEL_LEFT", DLongGDL(-1)); widgtablecelsel->InitTag("SEL_TOP", DLongGDL(-1)); widgtablecelsel->InitTag("SEL_RIGHT", DLongGDL(-1)); widgtablecelsel->InitTag("SEL_BOTTOM", DLongGDL(-1)); - GDLWidget::PushEvent(baseWidgetID, widgtablecelsel); + GDLWidget::PushEvent(widgtablecelsel); } } } @@ -1844,39 +1852,39 @@ void wxGridGDL::OnTextChanged(wxGridEvent & event) { widg = new DStructGDL("WIDGET_TABLE_DEL"); widg->InitTag("ID", DLongGDL(GDLWidgetTableID)); widg->InitTag("TOP", DLongGDL(baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widg->InitTag("HANDLER", DLongGDL(event.GetId())); widg->InitTag("TYPE", DIntGDL(2)); // delete widg->InitTag("OFFSET", DLongGDL(offset)); widg->InitTag("LENGTH", DLongGDL(-lengthDiff)); widg->InitTag("X", DLongGDL(x)); widg->InitTag("Y", DLongGDL(y)); - GDLWidget::PushEvent(baseWidgetID, widg); + GDLWidget::PushEvent(widg); return; } else if (lengthDiff == 0) //empty to empty, but RETURN entered { widg = new DStructGDL("WIDGET_TABLE_CH"); widg->InitTag("ID", DLongGDL(GDLWidgetTableID)); widg->InitTag("TOP", DLongGDL(baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widg->InitTag("HANDLER", DLongGDL(event.GetId())); widg->InitTag("TYPE", DIntGDL(0)); // single char widg->InitTag("OFFSET", DLongGDL(offset)); widg->InitTag("CH", DByteGDL(10)); //newline widg->InitTag("X", DLongGDL(x)); widg->InitTag("Y", DLongGDL(y)); - GDLWidget::PushEvent(baseWidgetID, widg); + GDLWidget::PushEvent(widg); return; } else if (lengthDiff == 1) //only character added only { widg = new DStructGDL("WIDGET_TABLE_CH"); widg->InitTag("ID", DLongGDL(GDLWidgetTableID)); widg->InitTag("TOP", DLongGDL(baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widg->InitTag("HANDLER", DLongGDL(event.GetId())); widg->InitTag("TYPE", DIntGDL(0)); // single char widg->InitTag("OFFSET", DLongGDL(offset)); widg->InitTag("CH", DByteGDL((newValue.substr(offset, offset).c_str())[0])); widg->InitTag("X", DLongGDL(x)); widg->InitTag("Y", DLongGDL(y)); - GDLWidget::PushEvent(baseWidgetID, widg); + GDLWidget::PushEvent(widg); return; } else // lengthDiff > 1 We just imagine it is deleted from there to the end, and the new characters are added { @@ -1885,27 +1893,27 @@ void wxGridGDL::OnTextChanged(wxGridEvent & event) { widg = new DStructGDL("WIDGET_TABLE_DEL"); widg->InitTag("ID", DLongGDL(GDLWidgetTableID)); widg->InitTag("TOP", DLongGDL(baseWidgetID)); -// widg->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widg->InitTag("HANDLER", DLongGDL(event.GetId())); widg->InitTag("TYPE", DIntGDL(2)); // delete widg->InitTag("OFFSET", DLongGDL(offset)); widg->InitTag("LENGTH", DLongGDL(previousValue.length() - offset)); widg->InitTag("X", DLongGDL(x)); widg->InitTag("Y", DLongGDL(y)); - GDLWidget::PushEvent(baseWidgetID, widg); + GDLWidget::PushEvent(widg); } DStructGDL* widg2; // 2nd insert new widg2 = new DStructGDL("WIDGET_TABLE_STR"); widg2->InitTag("ID", DLongGDL(GDLWidgetTableID)); widg2->InitTag("TOP", DLongGDL(baseWidgetID)); -// widg2->InitTag("HANDLER", DLongGDL(baseWidgetID)); + widg2->InitTag("HANDLER", DLongGDL(event.GetId())); widg2->InitTag("TYPE", DIntGDL(1)); // multiple char widg2->InitTag("OFFSET", DLongGDL(offset)); widg2->InitTag("STR", DStringGDL(newValue.substr(offset))); widg2->InitTag("X", DLongGDL(x)); widg2->InitTag("Y", DLongGDL(y)); - GDLWidget::PushEvent(baseWidgetID, widg2); + GDLWidget::PushEvent(widg2); return; } } @@ -1927,7 +1935,7 @@ void wxTreeCtrlGDL::OnItemActivated(wxTreeEvent & event){ treeselect->InitTag("TYPE", DIntGDL(0)); // 0 treeselect->InitTag("CLICKS",DLongGDL(2)); // insert into structList - GDLWidget::PushEvent( baseWidgetID, treeselect ); + GDLWidget::PushEvent(treeselect); event.Skip(); me->Refresh(); } @@ -1949,7 +1957,7 @@ void wxTreeCtrlGDL::OnItemSelected(wxTreeEvent & event){ treeselect->InitTag("TYPE", DIntGDL(0)); // 0 treeselect->InitTag("CLICKS",DLongGDL(1)); // insert into structList - GDLWidget::PushEvent( baseWidgetID, treeselect ); + GDLWidget::PushEvent(treeselect); event.Skip(); me->Refresh(); } @@ -2116,7 +2124,7 @@ void wxTreeCtrlGDL::OnTreeCtrlDrop(wxTreeEvent & event){ treedrop->InitTag("Y",DLongGDL(position.y)); treedrop->InitTag("MODIFIERS",DIntGDL(GetModifiers())); //mask with 1 shift 2 control 4 caps lock 8 alt // insert into structList - GDLWidget::PushEvent( baseWidgetID, treedrop ); + GDLWidget::PushEvent(treedrop); } //unset dragged itemDragging=nullptr; @@ -2148,7 +2156,7 @@ void wxTreeCtrlGDL::OnForeignDrop(WidgetIDT selected, WidgetIDT drag_id){ treedrop->InitTag("Y",DLongGDL(0)); treedrop->InitTag("MODIFIERS",DIntGDL(GetModifiers())); //mask with 1 shift 2 control 4 caps lock 8 alt // insert into structList - GDLWidget::PushEvent( baseWidgetID, treedrop ); + GDLWidget::PushEvent(treedrop); } Refresh(); } @@ -2172,7 +2180,7 @@ void wxTreeCtrlGDL::OnItemExpanded(wxTreeEvent & event){ treeexpand->InitTag("TYPE", DIntGDL(1)); // 1 treeexpand->InitTag("EXPAND",DLongGDL(1)); //1 expand // insert into structList - GDLWidget::PushEvent( baseWidgetID, treeexpand ); + GDLWidget::PushEvent(treeexpand); event.Skip(); me->Refresh(); } @@ -2196,7 +2204,7 @@ void wxTreeCtrlGDL::OnItemCollapsed(wxTreeEvent & event){ treeexpand->InitTag("TYPE", DIntGDL(1)); // 1 treeexpand->InitTag("EXPAND",DLongGDL(0)); //0 collapse // insert into structList - GDLWidget::PushEvent( baseWidgetID, treeexpand ); + GDLWidget::PushEvent(treeexpand); event.Skip(); me->Refresh(); } @@ -2221,7 +2229,7 @@ void wxTreeCtrlGDL::OnItemCollapsed(wxTreeEvent & event){ treeechecked->InitTag("TYPE", DIntGDL(2)); // 2 treeechecked->InitTag("STATE",DLongGDL(this->GetItemState(event.GetItem()))); //on off // insert into structList - GDLWidget::PushEvent( baseWidgetID, treeechecked ); + GDLWidget::PushEvent(treeechecked); event.Skip(); me->Refresh(); } diff --git a/src/graphicsdevice.hpp b/src/graphicsdevice.hpp index c096433f7..c92953a37 100644 --- a/src/graphicsdevice.hpp +++ b/src/graphicsdevice.hpp @@ -226,7 +226,7 @@ class GraphicsDevice virtual DString GetCurrentFont() {return "__$";} virtual DLong GetGraphicsFunction() { return -1;} virtual DIntGDL* GetPageSize() { return NULL;} - virtual DLong GetPixelDepth() { return -1;} + virtual DInt GetPixelDepth() { return -1;} virtual bool SetPixelDepth(DInt depth) { return false;} virtual DDoubleGDL* GetScreenResolution(char* disp=NULL) //fake a basic screen if not implemented: { @@ -237,10 +237,9 @@ class GraphicsDevice return res; } // virtual DFloatGDL* GetScreenSize(char* disp=NULL) { return NULL;} - virtual DLongGDL* GetScreenSize(char* disp=NULL) //fake a basic screen if not implemented: + virtual DIntGDL* GetScreenSize(char* disp=NULL) //fake a basic screen if not implemented: { - DLongGDL* res; - res = new DLongGDL(2, BaseGDL::NOZERO); + DIntGDL* res = new DIntGDL(2, BaseGDL::NOZERO); (*res)[0]=640; (*res)[1]=480; return res; diff --git a/src/initsysvar.cpp b/src/initsysvar.cpp index 8e6ce980b..e8315df61 100644 --- a/src/initsysvar.cpp +++ b/src/initsysvar.cpp @@ -59,7 +59,7 @@ namespace SysVar } // the index of some system variables UInt nullIx, trueIx, falseIx, pathIx, helppathIx, promptIx, edit_inputIx, quietIx, moreIx, - dIx, pIx, xIx, yIx, zIx, vIx, gdlWarningIx, gdlIx, cIx, MouseIx, + dIx, pIx, xIx, yIx, zIx, vIx, gdlIx, cIx, MouseIx, errorStateIx, errorIx, errIx, err_stringIx, valuesIx, journalIx, exceptIx, mapIx, cpuIx, dirIx, stimeIx, warnIx, usersymIx, orderIx, MakeDllIx, colorIx; @@ -225,18 +225,18 @@ namespace SysVar return static_cast(*qSysVar.Data())[0]; } + DInt More() + { + DVar& qSysVar=*sysVarList[moreIx]; + return static_cast(*qSysVar.Data())[0]; + } + void Make_Quiet() { DVar& qSysVar=*sysVarList[quietIx]; static_cast(*qSysVar.Data())[0]=1; } - DLong GDL_Warning() - { - DVar& gwSysVar=*sysVarList[gdlWarningIx]; - return static_cast(*gwSysVar.Data())[0]; - } - void SetC( DLong cVal) { DVar& cSysVar=*sysVarList[cIx]; @@ -657,14 +657,6 @@ namespace SysVar orderIx = sysVarList.size(); sysVarList.push_back( order); - // !GDL_WARNING (to be used in VOIGT() and BeselIJKY() to warm on - // different behaviour between IDL and GDL - DLongGDL *gdlWarningData = new DLongGDL( 1 ); - DVar *gdlWarning = new DVar( "GDL_WARNING", gdlWarningData); - gdlWarningIx = sysVarList.size(); - sysVarList.push_back( gdlWarning); - sysVarNoSaveList.push_back( gdlWarning); - // !GDL (to allow distinguish IDL/GDL with DEFSYSV, '!gdl', exists=exists ) DStructGDL* gdlStruct = new DStructGDL( "!GNUDATALANGUAGE"); gdlStruct->NewTag("RELEASE", new DStringGDL( (char*)VERSION + (VERSION[0] == 'v' ? 1 : 0))); diff --git a/src/initsysvar.hpp b/src/initsysvar.hpp index 5e97456a1..95e2b7690 100644 --- a/src/initsysvar.hpp +++ b/src/initsysvar.hpp @@ -93,15 +93,12 @@ namespace SysVar void Suppress_Edit_Input(); DLong Quiet(); void Make_Quiet(); - DLong GDL_Warning(); - + DInt More(); + // get and set !JOURNAL (journal file LUN) DLong JournalLUN(); void JournalLUN( DLong jLUN); - // get and set USERSYM - DStructGDL* USYM(); - // get the !ORDER variable DLong TV_ORDER(); diff --git a/src/libinit.cpp b/src/libinit.cpp index eb4b2824d..ef5037b45 100644 --- a/src/libinit.cpp +++ b/src/libinit.cpp @@ -316,9 +316,8 @@ void LibInit() "INTERNAL_LIB_GDL","KEYS","LAST_MESSAGE","LIB","MEMORY","NAMES", "OBJECTS","OUTPUT","PATH_CACHE","PREFERENCES","PROCEDURES", "RECALL_COMMANDS","ROUTINES","SOURCE_FILES","STRUCTURES", - "SYSTEM_VARIABLES","TRACEBACK", "COMMON","LEVEL", "SHARED_MEMORY", KLISTEND}; - const string helpWarnKey[]={"BREAKPOINTS","DLM", "MESSAGES", KLISTEND}; - new DLibPro(lib::help_pro,string("HELP"),-1,helpKey,helpWarnKey); + "SYSTEM_VARIABLES","TRACEBACK", "COMMON","LEVEL", "SHARED_MEMORY", "BREAKPOINTS","DLM", "MESSAGES", "LAMBDA", KLISTEND}; + new DLibPro(lib::help_pro,string("HELP"),-1,helpKey); new DLibPro(lib::delvar_pro,string("DELVAR"),-1,NULL,NULL); DLibPro* hide = new DLibPro(lib::findvar_pro,string("FINDVAR"),-1,NULL,NULL); diff --git a/src/libinit_jmg.cpp b/src/libinit_jmg.cpp index f981b5451..d0f0c3c99 100644 --- a/src/libinit_jmg.cpp +++ b/src/libinit_jmg.cpp @@ -247,6 +247,7 @@ void LibInit_jmg() new DLibPro( lib::unlinksymbol, string("UNLINKSYMBOL"), 2, unlinksymbolKey, NULL, 2 ); new DLibPro(lib::wait_pro,string("WAIT"),1); + new DLibPro(lib::gdlwait_responsive,string("GDLWAIT_RESPONSIVE"),1); #if defined(USE_HDF) new DLibFunRetNew(lib::hdf_ishdf,string("HDF_ISHDF"),1); diff --git a/src/math_fun_ng.cpp b/src/math_fun_ng.cpp index 7e1f521cc..bb535796c 100644 --- a/src/math_fun_ng.cpp +++ b/src/math_fun_ng.cpp @@ -135,13 +135,6 @@ namespace lib { //Voigt ( scalar , array ) if (nElp0 == 1 && nElp1 > 1) { - /* obsolete !! if ((A->Rank() > 0) && (SysVar::GDL_Warning())) { - Warning ( "You are using a case where IDL and GDL don't work similarly:"); - Warning ( "in Voigt(A,U), A is a singleton array and U a true array: check your code !"); - Warning ( "in GDL: Voigt(1,[0,1,2]) == Voigt([1],[0,1,2]) == Voigt([1,1,1],[0,1,2])."); - Warning ( "You can turn OFF this warning changing !GDL_WARNING to 0."); - } - */ for (i=0;i 1 && nElp1 == 1) { - /* obsolete !! - if ((U->Rank() > 0) && (SysVar::GDL_Warning())) { - Warning ( "You are using a case where IDL and GDL don't work similarly:"); - Warning ( "in Voigt(A,U), U is a singleton array and A a true array: check your code !"); - Warning ( "in GDL: Voigt([0,1,2], 1) == Voigt([0,1,2],[1]) == Voigt([0,1,2],[1,1,1])."); - Warning ( "You can turn OFF this warning changing !GDL_WARNING to 0."); - } - */ for (i=0;i w) @@ -192,7 +192,8 @@ void OutScientific(std::ostream& os, const T &val, const int w, const int d, con // TODO: IDL handles both lower and upper case "E" (tracker item no. 3147155) if ( code & fmtSHOWPOS ) oss << std::showpos; if ( code & fmtUPPER ) oss << std::uppercase ; - oss << std::scientific << std::setprecision(d) << val; + //IDL uses the showpoint mode + oss << std::scientific<< std::showpoint << std::setprecision(d) << val; if (w == 0 ) os << oss.str(); else if (oss.tellp() > w) diff --git a/src/plotting_device.cpp b/src/plotting_device.cpp index abce42ace..8b310577b 100644 --- a/src/plotting_device.cpp +++ b/src/plotting_device.cpp @@ -124,11 +124,11 @@ namespace lib { { static int get_pixel_depthIx = e->KeywordIx("GET_PIXEL_DEPTH"); if (e->KeywordPresent(get_pixel_depthIx)) { - DLong value = actDevice->GetPixelDepth(); + DInt value = actDevice->GetPixelDepth(); if (value == -1) e->Throw("Keyword GET_PIXEL_DEPTH not allowed for call to: DEVICE"); else - e->SetKW(get_pixel_depthIx, new DLongGDL(value)); + e->SetKW(get_pixel_depthIx, new DIntGDL(value)); } } @@ -136,7 +136,7 @@ namespace lib { { static int get_screen_sizeIx = e->KeywordIx("GET_SCREEN_SIZE"); if (e->KeywordPresent(get_screen_sizeIx)) { - DLongGDL* fvalue = actDevice->GetScreenSize(); + DIntGDL* fvalue = actDevice->GetScreenSize(); if (fvalue == NULL) e->Throw("Keyword GET_SCREEN_SIZE not allowed for call to: DEVICE"); else { diff --git a/src/pro/cw_field.pro b/src/pro/cw_field.pro new file mode 100644 index 000000000..cbbb0e1f3 --- /dev/null +++ b/src/pro/cw_field.pro @@ -0,0 +1,798 @@ +;+ +; NAME: +; CW_FIELD +; +; PURPOSE: +; +; The purpose of this compound widget is to provide an alternative +; to the CW_FIELD widget offered in the IDL distribution. What has +; always annoyed me about CW_FIELD is that the text widgets do not +; look editable to the users on Windows platforms. This program +; corrects that deficiency and adds some features that I think +; would be helpful. For example, you can now assign an event handler +; to the compound widget. +; +; AUTHOR: +; FANNING SOFTWARE CONSULTING +; David Fanning, Ph.D. +; 2642 Bradbury Court +; Fort Collins, CO 80521 USA +; Phone: 970-221-0438 +; E-mail: davidf@dfanning.com +; Coyote's Guide to IDL Programming: http://www.dfanning.com/ +; +; CATEGORY: +; +; General programming. +; +; CALLING SEQUENCE: +; +; fieldID = Cw_Field(parent, Title='X Size: ", Value=256, /IntegerValue) +; +; INPUT PARAMETERS: +; +; parent -- The parent widget ID of the compound widget. Required. +; +; INPUT KEYWORDS: +; +; Column -- Set this keyword to have the Label Widget above the Text Widget. +; CR_Only -- Set this keyword if you only want Carriage Return events. If +; this keyword is not set, all events are returned. No events +; are returned unless the EVENT_PRO or EVENT_FUNC keywords are used. +; DoubleValue -- Set this keyword if you want DOUBLE values returned. +; Decimal -- Set this keyword to the number of digits to the right of the decimal +; point in FLOATVALUE and DOUBLEVALUE numbers. +; Digits -- Set this keyword to the number of digits permitted in INTERGERVALUE and LONGVALUE numbers. +; Event_Func -- Set this keyword to the name of an Event Function. If this +; keyword is undefined and the Event_Pro keyword is undefined, +; all compound widget events are handled internally and not +; passed on to the parent widget. +; Event_Pro -- Set this keyword to the name of an Event Procedure. If this +; keyword is undefined and the Event_Func keyword is undefined, +; all compound widget events are handled internally and not +; passed on to the parent widget. +; FieldFont -- The font name for the text in the Text Widget. +; FloatValue -- Set this keyword for FLOAT values. +; Frame -- Set this keyword to put a frame around the compound widget. +; IntegerValue -- Set this keyword for INTEGER values. +; LabelFont -- The font name for the text in the Label Widget. +; LabelSize -- The X screen size of the Label Widget. +; LongValue -- Set this keyword for LONG values. +; Row=row -- Set this keyword to have the Label beside the Text Widget. (The default.) +; Scr_XSize -- The X screen size of the compound widget. +; Scr_YSize -- The Y screen size of the compound widget. +; StringValue -- Set this keyword for STRING values. (The default.) +; Title -- The text to go on the Label Widget. +; UValue -- A user value for any purpose. +; Value -- The "value" of the compound widget. +; XSize -- The X size of the Text Widget. +; YSize -- The Y size of the Text Widget. +; classic -- added by GD, 2025, "classic" events returned. +; +; COMMON BLOCKS: +; +; None. +; +; RESTRICTIONS: +; +; None. +; +; EVENT STRUCTURE: +; +; All events are handled internally unless either the Event_Pro or Event_Func +; keywords are used to assign an event handler to the compound widget. By +; default all events generated by the text widget are passed to the assigned +; event handler. If you wish to receive only Carriage Return events, set the +; CR_Only keyword. +; +; event = { DF_CW_FIELD, $ ; The name of the event structure. +; ID: 0L, $ ; The ID of the compound widget's top-level base. +; TOP: 0L, $ ; The widget ID of the top-level base of the hierarchy. +; HANDLER: 0L, $ ; The event handler ID. Filled out by IDL. +; Value: Ptr_New(), $ ; A pointer to the widget value. +; Type:"" ; A string indicating the type of data in the VALUE field. +; } ; Values are "INT", "LONG", "FLOAT", "DOUBLE", or "STRING". +; +; EXAMPLE: +; +; An example program is provided at the end of the DF_CW_FIELD code. To run it, +; type this commands: +; +; IDL> .run Df_Cw_Field +; +; MODIFICATION HISTORY: +; +; Written by: David Fanning, 17 NOV 1999. +; Added check to make float and double values finite. 18 NOV 1999. DWF. +; Fixed a bug when selecting and deleting all numerical text. 19 NOV 1999. DWF. +; Added DECIMAL and DIGITS keywords. 2 Jan 2000. DWF. +; Added the POSITIVE keyword. 12 Jan 2000. DWF. +; Fixed a few minor bugs with delete and digits. 12 Jan 2000. DWF. +; Made GET_VALUE function return pointer to data, instead of data. 12 Jan 2000. DWF. +; Fixed a small typo: "aveDecimal" to "haveDecimal". 10 March 2000. DWF. +; Added to GDL pro suite to provide "cw_field", 2025 Gilles Duvert, while preserving +; as uch as possible the original work by D. Fanning +;- +; +;########################################################################### +; +; LICENSE +; +; This software is OSI Certified Open Source Software. +; OSI Certified is a certification mark of the Open Source Initiative. +; +; Copyright © 2000 Fanning Software Consulting. +; +; This software is provided "as-is", without any express or +; implied warranty. In no event will the authors be held liable +; for any damages arising from the use of this software. +; +; Permission is granted to anyone to use this software for any +; purpose, including commercial applications, and to alter it and +; redistribute it freely, subject to the following restrictions: +; +; 1. The origin of this software must not be misrepresented; you must +; not claim you wrote the original software. If you use this software +; in a product, an acknowledgment in the product documentation +; would be appreciated, but is not required. +; +; 2. Altered source versions must be plainly marked as such, and must +; not be misrepresented as being the original software. +; +; 3. This notice may not be removed or altered from any source distribution. +; +; For more information on Open Source Software, visit the Open Source +; web site: http://www.opensource.org. +; +;########################################################################### + + +Function Df_Cw_Field_ReturnValue, inputValue, dataType + +; This utility routine takes a string and turns it into a number, +; depending upon the required data type. + +ON_IOERROR, CatchIt + +IF (Byte(inputValue))[0] EQ 32B THEN RETURN, "" + +CASE dataType OF + 'INT': IF inputValue EQ "" OR inputValue EQ "-" OR inputValue EQ "+" THEN $ + retValue = 'NULLVALUE' ELSE retValue = Fix(inputValue) + 'LONG': IF inputValue EQ "" OR inputValue EQ "-" OR inputValue EQ "+" THEN $ + retValue = 'NULLVALUE' ELSE retValue = Long(inputValue) + 'FLOAT' : IF inputValue EQ "" OR inputValue EQ "-" OR inputValue EQ "+" THEN $ + retValue = 'NULLVALUE' ELSE retValue = Float(inputValue) + 'DOUBLE': IF inputValue EQ "" OR inputValue EQ "-" OR inputValue EQ "+" THEN $ + retValue = 'NULLVALUE' ELSE retValue = Double(inputValue) + 'STRING' : retValue = inputValue +ENDCASE +RETURN, retValue + +CatchIt: + retValue = 'NULLVALUE' + RETURN, retValue +END ;---------------------------------------------------------------------------- + + + +Function Df_Cw_Field_Validate, value, dataType, Decimal=decimal, Digits=digits, $ + Positive=positive + +; This function eliminates illegal characters from a string that represents +; a number. The return value is a properly formatted string that can be turned into +; an INT, LONG, FLOAT, or DOUBLE value. +; +; + 43B +; - 45B +; . 46B +; 0 - 9 48B -57B +; 'eEdD' [101B, 69B, 100B, 68B] + + ; A null string should be returned at once. + +IF N_Elements(value) EQ 0 THEN value = "" +value = value[0] +IF value EQ "" THEN RETURN, String(value) + + ; No leading or trainnig blank characters to evaluate. + +value = StrTrim(value, 2) + +IF N_Elements(dataType) EQ 0 THEN dataType = 'STRING' + + ; A string value should be returned at once. Nothing to check. + +IF StrUpCase(dataType) EQ 'STRING' THEN RETURN, value + + ; Check integers and longs. A "-" or "+" in the first character is allowed. Otherwise, + ; only number between 0 and 9, or 43B to 57B. + +IF StrUpCase(dataType) EQ 'INT' OR StrUpCase(dataType) EQ 'LONG' THEN BEGIN + + returnValue = Ptr_New(/Allocate_Heap) + asBytes = Byte(value) + IF positive THEN BEGIN + IF (asBytes[0] EQ 43B) OR $ + (asBytes[0] GE 48B AND asBytes[0] LE 57B) THEN *returnValue = [asBytes[0]] + ENDIF ELSE BEGIN + IF (asBytes[0] EQ 45B) OR (asBytes[0] EQ 43B) OR $ + (asBytes[0] GE 48B AND asBytes[0] LE 57B) THEN *returnValue = [asBytes[0]] + ENDELSE + length = StrLen(asBytes) + IF length EQ 1 THEN BEGIN + IF N_Elements(*returnValue) EQ 0 THEN *returnValue = [32B] ELSE $ + *returnValue = [asBytes[0]] + ENDIF ELSE BEGIN + FOR j=1,length-1 DO BEGIN + IF (asBytes[j] GE 48B AND asBytes[j] LE 57B) THEN BEGIN + IF N_Elements(*returnValue) EQ 0 THEN *returnValue = [asBytes[j]] ELSE $ + *returnValue = [*returnValue, asBytes[j]] + ENDIF + ENDFOR + ENDELSE + IF N_Elements(*returnValue) NE 0 THEN retValue = String(*returnValue) ELSE retValue = "" + Ptr_Free, returnValue + + ; Check for digit restrictions. + + IF digits GT 0 THEN BEGIN + retValue = StrTrim(retValue, 2) + IF StrMid(retValue, 0, 1) EQ "-" THEN digits = digits + 1 + retValue = StrMid(retValue, 0, digits) + ENDIF + + RETURN, retValue + +ENDIF + + ; Check floating and double values. (+,-) in first character or after 'eEdD'. + ; Only numbers, signs, decimal points, and 'eEdD' allowed. + +IF StrUpCase(dataType) EQ 'FLOAT' OR StrUpCase(dataType) EQ 'DOUBLE' THEN BEGIN + returnValue = Ptr_New(/Allocate_Heap) + asBytes = Byte(value) + IF positive THEN BEGIN + IF (asBytes[0] EQ 43B) OR $ + (asBytes[0] GE 48B AND asBytes[0] LE 57B) OR $ + (asBytes[0] EQ 46B) THEN *returnValue = [asBytes[0]] + IF (asBytes[0] EQ 46B) THEN haveDecimal = 1 ELSE haveDecimal = 0 + ENDIF ELSE BEGIN + IF (asBytes[0] EQ 45B) OR (asBytes[0] EQ 43B) OR $ + (asBytes[0] GE 48B AND asBytes[0] LE 57B) OR $ + (asBytes[0] EQ 46B) THEN *returnValue = [asBytes[0]] + IF (asBytes[0] EQ 46B) THEN haveDecimal = 1 ELSE haveDecimal = 0 + ENDELSE + haveExponent = 0 + length = StrLen(asBytes) + prevByte = asBytes[0] + exponents = Byte('eEdD') + IF length EQ 1 THEN BEGIN + IF N_Elements(*returnValue) EQ 0 THEN *returnValue = [32B] ELSE $ + *returnValue = [asBytes[0]] + ENDIF ELSE BEGIN + FOR j=1,length-1 DO BEGIN + IF (asBytes[j] GE 48B AND asBytes[j] LE 57B) THEN BEGIN + IF N_Elements(*returnValue) EQ 0 THEN *returnValue = [asBytes[j]] ELSE $ + *returnValue = [*returnValue, asBytes[j]] + prevByte = asBytes[j] + ENDIF ELSE BEGIN + + ; What kind of thing is it? + + IF (asBytes[j] EQ 46B) THEN BEGIN ; A decimal point. + IF haveDecimal EQ 0 THEN BEGIN + *returnValue = [*returnValue, asBytes[j]] + haveDecimal = 1 + prevByte = asBytes[j] + ENDIF + ENDIF + + IF (asBytes[j] EQ 45B) OR (asBytes[j] EQ 43B) THEN BEGIN ; A + or - sign. + index = Where(exponents EQ prevByte, count) + IF count EQ 1 AND haveExponent THEN BEGIN + *returnValue = [*returnValue, asBytes[j]] + haveDecimal = 1 + prevByte = asBytes[j] + ENDIF + ENDIF + + index = Where(exponents EQ asBytes[j], count) + IF count EQ 1 AND haveExponent EQ 0 THEN BEGIN ; An exponent + *returnValue = [*returnValue, asBytes[j]] + haveExponent = 1 + prevByte = asBytes[j] + ENDIF + ENDELSE + ENDFOR + ENDELSE + IF N_Elements(*returnValue) NE 0 THEN BEGIN + + retValue = String(*returnValue) + retValue = StrTrim(retValue, 2) + + ; Check for decimal restrictions + + IF decimal GE 0 THEN BEGIN + theDecimalPt = StrPos(retValue, '.') + IF theDecimalPt NE -1 THEN retValue = StrMid(retValue, 0, theDecimalPt + decimal + 1) + ENDIF + + ENDIF ELSE retValue = "" + Ptr_Free, returnValue + + ; Is this a representable number? + + testValue = Df_Cw_Field_ReturnValue(retValue, dataType) + IF String(testValue) NE 'NULLVALUE' THEN numCheck = Finite(testValue) ELSE numCheck = 1 + IF numCheck THEN BEGIN + RETURN, retValue + ENDIF ELSE BEGIN + Message, 'The requested number is not representable.', /Informational + RETURN, "" + ENDELSE +ENDIF + +END ;---------------------------------------------------------------------------- + + + +Pro Df_Cw_Field__Define + +; The Df_Cw_Field Event Structure. + + event = { DF_CW_FIELD, $ ; The name of the event structure. + ID: 0L, $ ; The ID of the compound widget's top-level base. + TOP: 0L, $ ; The widget ID of the top-level base of the hierarchy. + HANDLER: 0L, $ ; The event handler ID. Filled out by IDL. + Value: Ptr_New(), $ ; A pointer to the widget value. + Type:"" $ ; A string indicating the type of data in the VALUE field. + } ; Values are "INT", "LONG", "FLOAT", "DOUBLE", or "STRING". + +END ;---------------------------------------------------------------------------- + + + +Pro Df_Cw_Field_Kill_Notify, ID + +; This routine cleans up the pointer when the compound widget is destroyed. + +Widget_Control, ID, Get_UValue=info, /No_Copy +Ptr_Free, info.theValue +END ;---------------------------------------------------------------------------- + + + +Pro Df_Cw_Field_Set_Value, cw_tlb, value + +; This procedure sets a value for the compound widget. The value +; is a value appropriate for the data type or a string. + + ; Get info structure. + +info_carrier = Widget_Info(cw_tlb, Find_by_UName='INFO_CARRIER') +Widget_Control, info_carrier, Get_UValue=info, /No_Copy + + ; Validate the value. + +theText = Strtrim(value, 2) +theText = Df_Cw_Field_Validate(theText, info.dataType, Decimal=info.decimal, $ + Digits=info.digits, Positive=info.positive) + + ; Load the value in the widget. + +Widget_Control, info.textID, Set_Value=theText, Set_Text_Select=[StrLen(theText),0] +info.theText = theText + + ; Set the actual value of the compound widget. + +*info.theValue = Df_Cw_Field_ReturnValue(info.theText, info.dataType) + +Widget_Control, info_carrier, Set_UValue=info, /No_Copy +END ;---------------------------------------------------------------------------- + +; added by GD, 2025 for GDL +Function Cw_Field_Get_Value, cw_tlb + +; This function returns the numerical or string value of the +; compound widget. + +info_carrier = Widget_Info(cw_tlb, Find_by_UName='INFO_CARRIER') +Widget_Control, info_carrier, Get_UValue=info, /No_Copy +value = info.theValue +Widget_Control, info_carrier, Set_UValue=info, /No_Copy +RETURN, (*value) +END ;---------------------------------------------------------------------------- + + +Function Df_Cw_Field_Get_Value, cw_tlb + +; This function returns the numerical or string value of the +; compound widget. + +info_carrier = Widget_Info(cw_tlb, Find_by_UName='INFO_CARRIER') +Widget_Control, info_carrier, Get_UValue=info, /No_Copy +value = info.theValue +Widget_Control, info_carrier, Set_UValue=info, /No_Copy +RETURN, value +END ;---------------------------------------------------------------------------- + + + + +PRO Df_Cw_Field_Event_Handler, event + +; The main event handler for the compound widget. + + ; Get the info structure. Get the previous text, the current + ; cursor location in the text widget, and indicate this is not + ; a Carriage Return event. + +Widget_Control, event.ID, Get_UValue=info, /No_Copy +previousText = info.theText +textLocation = Widget_Info(event.id, /Text_Select) +cr_event = 0 + + ; What kind of event is this? + +possibleTypes = ['INSERT SINGLE CHARACTER', 'INSERT MULTIPLE CHARACTERS', 'DELETE TEXT', 'SELECT TEXT'] +thisType = possibleTypes[event.type] + + ; Branch on event type. + +CASE thisType OF + + 'INSERT SINGLE CHARACTER': BEGIN + + ; Get the current contents of text widget. Validate it. + + Widget_Control, info.textID, Get_Value=newText + newText = newText[0] + validText = Df_Cw_Field_Validate(newText, info.dataType, Decimal=info.decimal, $ + Digits=info.digits, Positive=info.positive) + + ; If it is valid, leave it alone. If not, go back to previous text. + + IF validText NE newText THEN BEGIN + Widget_Control, info.textID, Set_Value=previousText, Set_Text_Select=[textLocation[0]-1,0] + ENDIF ELSE BEGIN + info.theText = validText + testValue = Df_Cw_Field_ReturnValue(validText, info.dataType) + IF String(testValue) EQ "NULLVALUE" THEN BEGIN + Ptr_Free, info.theValue + info.theValue = Ptr_New(/Allocate_Heap) + ENDIF ELSE *info.theValue = testValue + ENDELSE + ; Is this a Carriage Return event? + + IF event.ch EQ 10B then cr_event = 1 + ENDCASE + + 'INSERT MULTIPLE CHARACTERS': BEGIN + + ; Same thing as above, but for all the characters you are inserting. + + Widget_Control, info.textID, Get_Value=newText + newText = newText[0] + validText = Df_Cw_Field_Validate(newText, info.dataType, Decimal=info.decimal, $ + Digits=info.digits, Positive=info.positive) + IF validText NE newText THEN BEGIN + Widget_Control, info.textID, Set_Value=previousText, Set_Text_Select=[textLocation[0]-1,0] + ENDIF ELSE BEGIN + info.theText = validText + testValue = Df_Cw_Field_ReturnValue(validText, info.dataType) + IF String(testValue) EQ "NULLVALUE" THEN BEGIN + Ptr_Free, info.theValue + info.theValue = Ptr_New(/Allocate_Heap) + ENDIF ELSE *info.theValue = testValue + ENDELSE + ENDCASE + + 'DELETE TEXT': BEGIN + + ; Just get the new text and update the info stucture. + + Widget_Control, info.textID, Get_Value=theText + theText = theText[0] + validText = Df_Cw_Field_Validate(theText, info.dataType, Decimal=info.decimal, $ + Digits=info.digits, Positive=info.positive) + + ; Load the valid text. + + Widget_Control, info.textID, Set_Value=validText, Set_Text_Select=[textLocation[0],0] + info.theText = validText + testValue = Df_Cw_Field_ReturnValue(info.theText, info.dataType) + IF String(testValue) EQ "NULLVALUE" THEN BEGIN + Ptr_Free, info.theValue + info.theValue = Ptr_New(/Allocate_Heap) + ENDIF ELSE *info.theValue = testValue + ENDCASE + + 'SELECT TEXT': ; Nothing to do. + +ENDCASE + + ; Do you report all events, or only Carriage Return events? + +if info.classic then begin + val=(*info.thevalue) + dtype=0 + case info.datatype of + 'STRING':dtype=0 + 'INT':dtype=2 + 'LONG':dtype=3 + 'FLOAT':dtype=1 + 'DOUBLE':dtype=5 + endcase + if (info.positive) then begin + dtype=4 + val=fix(val,/ULONG) + endif + if isa(val,/NULL) then val=0 + thisEvent = {id:info.cw_tlb, top:event.top, handler:0L, value:val, type:dtype, update:cr_event eq 1} +endif else thisEvent = {Df_Cw_Field, info.cw_tlb, event.top, 0L, info.theValue, info.dataType} + +IF info.cr_only THEN BEGIN +; IF info.event_func NE "" THEN BEGIN + IF cr_event THEN Widget_Control, info.cw_tlb, Send_Event=thisEvent +; ENDIF +; IF info.event_pro NE "" THEN BEGIN +; IF cr_event THEN Widget_Control, info.cw_tlb, Send_Event=thisEvent +; ENDIF +ENDIF ELSE BEGIN +; IF info.event_func NE "" THEN BEGIN + Widget_Control, info.cw_tlb, Send_Event=thisEvent +; ENDIF +; +; IF info.event_pro NE "" THEN BEGIN +; Widget_Control, info.cw_tlb, Send_Event=thisEvent +; ENDIF +ENDELSE + + ; Out of here. + +Widget_Control, event.ID, Set_UValue=info, /No_Copy +END ;---------------------------------------------------------------------------- + + + +Function Df_Cw_Field, $ ; The compound widget Df_Cw_Field. + parent, $ ; The parent widget. Required for all compound widgets. + Column=column, $ ; Set this keyword to have Label above Text Widget. + CR_Only=cr_only, $ ; Set this keyword if you only want Carriage Return events. + Digits=digits, $ ; Set this keyword to number of allowed digits in INT and LONG values. + Decimal=decimal, $ ; Set to the number of digits to right of decimal point. + DoubleValue=doublevalue, $ ; Set this keyword if you want DOUBLE values returned. + Event_Func=event_func, $ ; Set this keyword to the name of an Event Function. + Event_Pro=event_pro, $ ; Set this keyword to the name of an Event Procedure. + FieldFont=fieldfont, $ ; The font name for the text in the Text Widget. + FloatValue=floatvalue, $ ; Set this keyword for FLOAT values. + Frame=frame, $ ; Set this keyword to put a frame around the compound widget. + IntegerValue=integervalue, $ ; Set this keyword for INTEGER values. + LabelFont=labelfont, $ ; The fon name for the text in the Label Widget. + LabelSize=labelsize, $ ; The X screen size of the Label Widget. + LongValue=longvalue, $ ; Set this keyword for LONG values. + Positive=positive, $ ; Set this keyword to only allow positive number values. + Row=row, $ ; Set this keyword to have the Label beside the Text Widget. (The default.) + Scr_XSize=scr_xsize, $ ; The X screen size of the compound widget. + Scr_YSize=scr_ysize, $ ; The Y screen size of the compound widget. + StringValue=stringvalue, $ ; Set this keyword for STRING values. (The default.) + Title=title, $ ; The text to go on the Label Widget. + UValue=uvalue, $ ; A user value for any purpose. + Value=value, $ ; The "value" of the compound widget. + XSize=xsize,$ ; The X size of the Text Widget. + YSize=ysize,$ ; The X size of the Text Widget. + textID=textID, $ + floating=floating,$ + noedit=noedit,$ + classic=classic + + ; A parent is required. + +IF N_Elements(parent) EQ 0 THEN BEGIN + Message, 'A PARENT argument is required. Returning...', /Informational + RETURN, -1L +ENDIF + + ; Check keyword values. +IF N_elements(classic) eq 0 then classic = 0 else classic = 1 +IF N_elements(noedit) eq 0 then editable = 1 else editable = 0 +IF N_Elements(column) EQ 0 THEN column = 0 +IF N_Elements(digits) EQ 0 THEN digits = 0 ELSE digits = Fix(digits) +IF N_Elements(decimal) EQ 0 THEN decimal = -1 ELSE decimal = Fix(decimal) +IF N_Elements(event_func) EQ 0 THEN event_func = "" +IF N_Elements(event_pro) EQ 0 THEN event_pro = "" +IF N_Elements(fieldfont) EQ 0 THEN fieldfont = "" +IF N_Elements(frame) EQ 0 THEN frame = 0 +IF N_Elements(labelfont) EQ 0 THEN labelfont = "" +IF N_Elements(labelsize) EQ 0 THEN labelsize = 0 +IF N_Elements(scr_xsize) EQ 0 THEN scr_xsize = 0 +IF N_Elements(scr_ysize) EQ 0 THEN scr_ysize = 0 +IF N_Elements(title) EQ 0 THEN title = "Input Value: " +IF N_Elements(uvalue) EQ 0 THEN uvalue = "" +IF N_Elements(value) EQ 0 THEN value = "" ELSE value = StrTrim(value,2) +IF N_Elements(xsize) EQ 0 THEN xsize = 0 +IF N_Elements(ysize) EQ 0 THEN ysize = 0 +IF N_Elements(row) EQ 0 AND column EQ 0 THEN row = 1 ELSE row = 0 +positive = Keyword_Set(positive) + + ; What data type are we looking for? + +dataType = 'STRING' +IF Keyword_Set(stringvalue) THEN dataType = 'STRING' +IF Keyword_Set(integervalue) THEN dataType = 'INT' +IF Keyword_Set(longvalue) THEN dataType = 'LONG' +IF Keyword_Set(floatvalue) THEN dataType = 'FLOAT' +IF Keyword_Set(doublevalue) THEN dataType = 'DOUBLE' + + ; Validate the input value. + +value = Df_Cw_Field_Validate(value, dataType, Decimal=decimal, Digits=digits, Positive=positive) + + ; Create the widgets. + +cw_tlb = Widget_Base( parent, $ ; The top-level base of the compound widget. + Pro_Set_Value='Df_Cw_Field_Set_Value', $ + Func_Get_Value='Cw_Field_Get_Value', $ ; modified: converts pointer to value + Frame=frame, $ + Row=row, $ + Column=Keyword_Set(column), $ + Base_Align_Center=1, $ + UValue=uvalue, $ + Event_Pro=event_pro, $ + Event_Func=event_func ) + +labelID = Widget_Label( cw_tlb, Value=title, Font=labelfont, $ ; The Label Widget. + Scr_XSize=labelsize) + +textID = Widget_Text( cw_tlb, $ ; The Text Widget. + Value=value, $ + XSize=xsize, $ + YSize=ysize, $ + Scr_XSize=scr_xsize, $ + Scr_YSize=scr_ysize, $ + Font=fieldfont, $ + All_Events=1, $ + Event_Pro='Df_Cw_Field_Event_Handler', $ + UName='INFO_CARRIER', $ + Kill_Notify='Df_Cw_Field_Kill_Notify', $ + Editable=editable ) + + ; Set the actual return value of the compound widget. + +theValue = Df_Cw_Field_ReturnValue(value, dataType) + + ; The info structure. + +info = { theText:value, $ ; The text in the Text Widget. + theValue:Ptr_New(theValue), $ ; The real value of the Text Widget. + cw_tlb:cw_tlb, $ ; The top-level base of the compound widget. + event_func:event_func, $ ; The name of the event handler function. + event_pro:event_pro, $ ; The name of the event handler procedure. + cr_only:Keyword_Set(cr_only), $ ; A flag to return events only on CR events. + dataType:dataType, $ ; The type of data wanted in the compound widget. + decimal:decimal, $ ; The number of digits in decimal numbers. + digits:digits, $ ; The number of digits in integer numbers. + positive:positive, $ ; Flag to indicate positive number values. + textID:textID, $ ; The widget identifier of the Text Widget. + classic:classic } ; returns "classic" event structures + + ; Store info structure in Text Widget. + +Widget_Control, textID, Set_UValue=info, /No_Copy +RETURN, cw_tlb +END ;---------------------------------------------------------------------------- + + +; The GDL cw_field replacement just calls David Fanning's cw_field (renamed df_cw_field) with proper values, +; and takes advantage of the event_func trapping to convert df_cw_field event to an basic idl event. +; this to avoid maiming the original work of D.Fanning + +FUNCTION CW_FIELD, Parent, COLUMN=Column, ROW=Row, $ + EVENT_FUNC = event_func, $ + DOUBLE=doublevalue, ULONG=ulongvalue, $ + FOCUS_EVENTS=focusEventsIn, $ + FLOATING=floatvalue, INTEGER=Integervalue, LONG=Longvalue, STRING=Stringvalue, $ + FONT=LabelFont, FRAME=Frame, TITLE=Title, UVALUE=UValue, VALUE=value, $ + RETURN_EVENTS=Return_Events, ALL_EVENTS=all_events, $ + FIELDFONT=FieldFont, NOEDIT=NoEdit, TEXT_FRAME=Text_Frame, $ + XSIZE=XSize, YSIZE=YSize, UNAME=uname, TAB_MODE=tab_mode, $ + IGNORE_ACCELERATORS=ignoreAccel + + ; Check keyword values. + +dataType = 'STRING' +positive=0 +IF Keyword_Set(stringvalue) THEN dataType = 'STRING' +IF Keyword_Set(integervalue) THEN dataType = 'INT' +IF Keyword_Set(longvalue) THEN dataType = 'LONG' +IF Keyword_Set(ulongvalue) THEN begin & dataType = 'LONG' & positive=1 & end +IF Keyword_Set(floatvalue) THEN dataType = 'FLOAT' +IF Keyword_Set(doublevalue) THEN dataType = 'DOUBLE' +IF Keyword_set(return_events) THEN cr_Only=1 +IF Keyword_set(all_events) THEN cr_Only=0 + +return,Df_Cw_Field( $ ; The compound widget Df_Cw_Field. + parent, $ ; The parent widget. Required for all compound widgets. + Column=column, $ ; Set this keyword to have Label above Text Widget. + CR_Only=cr_only, $ ; Set this keyword if you only want Carriage Return events. + Digits=digits, $ ; Set this keyword to number of allowed digits in INT and LONG values. + Decimal=decimal, $ ; Set to the number of digits to right of decimal point. + DoubleValue=doublevalue, $ ; Set this keyword if you want DOUBLE values returned. + Event_Func=event_func, $ ; Set this keyword to the name of an Event Function. +; Event_Pro=event_pro, $ ; Set this keyword to the name of an Event Procedure. + FieldFont=fieldfont, $ ; The font name for the text in the Text Widget. + FloatValue=floatvalue, $ ; Set this keyword for FLOAT values. + Frame=frame, $ ; Set this keyword to put a frame around the compound widget. + IntegerValue=integervalue, $ ; Set this keyword for INTEGER values. + LabelFont=labelfont, $ ; The fon name for the text in the Label Widget. + LabelSize=labelsize, $ ; The X screen size of the Label Widget. + LongValue=longvalue, $ ; Set this keyword for LONG values. + Positive=positive, $ ; Set this keyword to only allow positive number values. + Row=row, $ ; Set this keyword to have the Label beside the Text Widget. (The default.) + Scr_XSize=scr_xsize, $ ; The X screen size of the compound widget. + Scr_YSize=scr_ysize, $ ; The Y screen size of the compound widget. + StringValue=stringvalue, $ ; Set this keyword for STRING values. (The default.) + Title=title, $ ; The text to go on the Label Widget. + UValue=uvalue, $ ; A user value for any purpose. + Value=value, $ ; The "value" of the compound widget. + XSize=xsize,$ ; The X size of the Text Widget. + YSize=ysize,$ ; The X size of the Text Widget. + textID=textID, $ + floating=floating,$ + noedit=noedit,$ + classic=1) +end + +PRO Df_Cw_Field_Example_Event, event + +; An example event handler for Df_Cw_Field. + +Widget_Control, event.top, Get_UValue=info +Widget_Control, event.id, Get_UValue=thisEvent +CASE thisEvent OF + 'Field 1 Event': BEGIN + Print, '' + Print, 'Field 1 Event Value: ', *event.value + END + 'Field 2 Event': BEGIN + Print, '' + IF N_Elements(*event.value) EQ 0 THEN Print,'Field 2 is Undefined' ELSE $ + Print, 'Field 2 Event Value: ', *event.value + END + 'Print It': BEGIN + Widget_Control, info.field3, Get_Value=theValue + Print, '' + Print, 'Field 3 Value: ', theValue + END + 'PrintFloat': BEGIN + Widget_Control, info.field2, Get_Value=theValue + IF N_Elements(*theValue) EQ 0 THEN Print, 'Field 2 Value is Undefined' ELSE $ + Print, 'Field 2 Value: ', *theValue + END + 'Set It': BEGIN + Widget_Control, info.field3, Set_Value='Coyote Rules!' + END + 'Quit': Widget_Control, event.top, /Destroy +ENDCASE +END ;---------------------------------------------------------------------------- + + +; An example program to exercise some of the features of DF_CW_FIELD. +; All events are returned for Field 1, which only allows INTEGER values. +; Only Carriage Return events are returned for Field 2, which allows FLOAT values. +; Field 3 allows you to obtain and set values through the normal WIDGET_CONTROL mechanism. + +tlb = Widget_Base(Column=1) +field1 = Df_Cw_Field(tlb, Title='Integer:', LabelSize=50, digits=4, $ + Value=5, /IntegerValue, Event_Pro='Df_Cw_Field_Example_Event', UValue='Field 1 Event') +field2 = Df_Cw_Field(tlb, Title='Float:', LabelSize=50, Value=45.6, $ + /FloatValue, Event_Pro='Df_Cw_Field_Example_Event', /CR_Only, UValue='Field 2 Event', Decimal=3, /Positive) +field3 = Df_Cw_Field(tlb, Title='String:', LabelSize=50, Value='Coyote Rules!') +button = Widget_Button(tlb, Value='Print Value of String', UValue="Print It") +button = Widget_Button(tlb, Value='Set Value of String', UValue='Set It') +button = Widget_Button(tlb, Value='Print Floating Value', UValue='PrintFloat') +button = Widget_Button(tlb, Value='Quit', UValue='Quit') +Widget_Control, tlb, /Realize, Set_UValue={field1:field1, field2:field2, field3:field3} +XManager, 'Df_Cw_Field_Example', tlb, /No_Block +END diff --git a/src/pro/utilities/idlneturl__define.pro b/src/pro/utilities/idlneturl__define.pro index 45b549afa..991edecfe 100644 --- a/src/pro/utilities/idlneturl__define.pro +++ b/src/pro/utilities/idlneturl__define.pro @@ -224,7 +224,7 @@ function idlneturl::Get, BUFFER=buffer, FILENAME=filename, STRI StatusInfo="Downloading..." time=0d while (file_test(pb) eq 0 and time le 5 ) do begin - wait,0.1 + gdlwait_responsive,0.1 ;special wait command, never use elsewhere time+=0.1 end if (time ge 5) then goto,done ; curl did not even create the progressbar (? to fast or problem?) , skip callback @@ -238,7 +238,7 @@ function idlneturl::Get, BUFFER=buffer, FILENAME=filename, STRI free_lun,lun goto, fin endif - wait,0.1 + gdlwait_responsive,0.1 ;special wait command, never use elsewhere endwhile free_lun,lun done: @@ -284,7 +284,7 @@ fin: return, response endif else if KEYWORD_SET(string_array) then begin n=1 - istext=strpos('text/',self.content_type) ge 0 + istext=strpos(self.content_type,'text/') ge 0 if istext then n=file_lines(filename) s="" & resp=strarr(n) openr,lun,filename,/get @@ -446,7 +446,7 @@ function idlneturl::Put, data, BUFFER=buffer, FILENAME=filename, POST=post, STRI ; return: if KEYWORD_SET(string_array) then begin n=1 - istext=strpos('text/',self.content_type) ge 0 + istext=strpos(self.content_type,'text/') ge 0 if istext then n=file_lines(filename) resp=strarr(n) & s='' openr,lun,oufile,/get diff --git a/src/widget.cpp b/src/widget.cpp index 473b1fac3..2868d78fc 100644 --- a/src/widget.cpp +++ b/src/widget.cpp @@ -81,7 +81,7 @@ wxBitmap * GetBitmapFromPassedBytes(EnvT* e, DByteGDL* passedBytes, bool doMask= wxRealPoint GetRequestedUnitConversionFactor( EnvT* e){ int the_units = 0; - static int unitsIx = e->KeywordIx( "UNITS" ); + int unitsIx = e->KeywordIx( "UNITS" ); //not static as the calling finction may change e->AssureLongScalarKWIfPresent( unitsIx, the_units ); //convert unit to the factor in pixels DDouble sx=wxGetDisplaySizeMM().x; @@ -98,7 +98,7 @@ wxRealPoint GetRequestedUnitConversionFactor( EnvT* e){ void GDLWidget::ChangeUnitConversionFactor( EnvT* e) { int the_units = 0; - static int unitsIx = e->KeywordIx( "UNITS" ); + int unitsIx = e->KeywordIx( "UNITS" ); //not static as the calling finction may change e->AssureLongScalarKWIfPresent( unitsIx, the_units ); //convert unit to the factor in pixels DDouble sx=wxGetDisplaySizeMM().x; @@ -186,17 +186,36 @@ void GDLWidget::GetCommonKeywords( EnvT* e) DDouble sy=wxGetDisplaySizeMM().y; sx=wxGetDisplaySize().x/sx; //pix per mm sy=wxGetDisplaySize().y/sy; - + if (the_units==0) unitConversionFactor=wxRealPoint(1,1); if (the_units==1) unitConversionFactor=wxRealPoint(sx*25.4,sy*25.4); if (the_units==2) unitConversionFactor=wxRealPoint(sx*10.0,sy*10.0); - - e->AssureLongScalarKWIfPresent( scr_xsizeIx, wScreenSize.x ); if (wScreenSize.x<=0) wScreenSize.x=wxDefaultSize.x; - e->AssureLongScalarKWIfPresent( xsizeIx, wSize.x ); if (wSize.x<=0) wSize.x=wxDefaultSize.x; - e->AssureLongScalarKWIfPresent( scr_ysizeIx, wScreenSize.y ); if (wScreenSize.y<=0) wScreenSize.y=wxDefaultSize.y; - e->AssureLongScalarKWIfPresent( ysizeIx, wSize.y ); if (wSize.y<=0) wSize.y=wxDefaultSize.y; - e->AssureLongScalarKWIfPresent( xoffsetIx, wOffset.x ); if (wOffset.x<=0) wOffset.x=wxDefaultPosition.x; - e->AssureLongScalarKWIfPresent( yoffsetIx, wOffset.y ); if (wOffset.y<=0) wOffset.y=wxDefaultPosition.y; + double widget_xsize=-1; + double widget_ysize=-1; + double widget_screen_xsize=-1; + double widget_screen_ysize=-1; + e->AssureDoubleScalarKWIfPresent( scr_xsizeIx, widget_screen_xsize ); + e->AssureDoubleScalarKWIfPresent( scr_ysizeIx, widget_screen_ysize ); + wScreenSize.x=widget_screen_xsize*unitConversionFactor.x; + wScreenSize.y=widget_screen_ysize*unitConversionFactor.y; + if (wScreenSize.x<=0) wScreenSize.x=wxDefaultSize.x; + if (wScreenSize.y<=0) wScreenSize.y=wxDefaultSize.y; + + //check ProName as XSIZE and YSIZE are not in UNITS if WIDGET_LIST, WIDGET_TABLE, or WIDGET_TEXT + bool useScale=true; + std::string proname=e->GetProName(); + if (proname == "WIDGET_TABLE") useScale=false; + if (proname == "WIDGET_LIST") useScale=false; + if (proname == "WIDGET_TEXT") useScale=false; + e->AssureDoubleScalarKWIfPresent( xsizeIx, widget_xsize); + e->AssureDoubleScalarKWIfPresent( ysizeIx, widget_ysize ); + wSize.x=widget_xsize*(useScale?unitConversionFactor.x:1); + wSize.y=widget_ysize*(useScale?unitConversionFactor.y:1); + if (wSize.x<=0) wSize.x=-1; + if (wSize.y<=0) wSize.y=-1; + + e->AssureDoubleScalarKWIfPresent( xoffsetIx, wOffset.x ); if (wOffset.x<=0) wOffset.x=wxDefaultPosition.x; + e->AssureDoubleScalarKWIfPresent( yoffsetIx, wOffset.y ); if (wOffset.y<=0) wOffset.y=wxDefaultPosition.y; uValue = e->GetKW( uvalueIx ); //IDL: a !NULL uvalue is ignored. if ( uValue != NULL ) { @@ -228,23 +247,20 @@ void GDLWidget::GetCommonKeywords( EnvT* e) // these reside here because gdlwidget.hpp is only included if wxWidgets are used // and hence putting them there would cause a compiler error without wxWidgets -//GD not sure if the environment at the end of called PRO or FUNC is OK (nested procedures?) TBC. -BaseGDL* CallEventFunc( const std::string& f, BaseGDL* ev) -{ - StackGuard guard( BaseGDL::interpreter->CallStack( ) ); - - int funIx = GDLInterpreter::GetFunIx( f ); - - ProgNodeP callingNode = NULL; //BaseGDL::interpreter->GetRetTree(); - - EnvUDT* newEnv = new EnvUDT( callingNode, funList[ funIx], NULL ); - newEnv->SetNextPar( ev ); // pass as local - - BaseGDL::interpreter->CallStack( ).push_back( newEnv ); +BaseGDL* CallEventFunc( const std::string& f, BaseGDL* ev) { + DInterpreter* myInterpreter = ev->interpreter; + int funIx = myInterpreter->GetFunIx(f); + StackGuard guard(myInterpreter->CallStack()); + // GD: The function is called as LRFUNCTION to permit functions returning NULL (undefined) values, + // which are the equivalent of a procedure and terminate the upward-transmission of events in the widget_event loop. + // This seems an undocumented feature + // Calling with another CallContext will not permit this (particular?) behaviour. + EnvUDT* newEnv = new EnvUDT(myInterpreter->GetRetTree(), funList[ funIx], EnvUDT::LRFUNCTION); + newEnv->SetNextPar(ev); // pass as local + ev->interpreter->CallStack().push_back(newEnv); // make the call - newEnv->SetCallContext( EnvUDT::RFUNCTION ); - BaseGDL* res = BaseGDL::interpreter->call_fun( static_cast (newEnv->GetPro( ))->GetTree( ) ); + BaseGDL* res = myInterpreter->call_fun(funList[funIx]->GetTree()); return res; } @@ -272,7 +288,7 @@ void CallEventPro( const std::string& p, BaseGDL* p0, BaseGDL* p1 ) { #endif } -DStructGDL* CallEventHandler( DStructGDL* ev ) { +DStructGDL* CallEventHandler(DStructGDL* ev) { // Must work in good harmony with WIDGET_EVENT requirements. // for one event, start from the originating widget and go through the list of parents, // and process the first event-related procedure associated. @@ -281,14 +297,9 @@ DStructGDL* CallEventHandler( DStructGDL* ev ) { // If the top of the hierarchy is attained without ev being swallowed by an event handler, return ev. // Empty events (success) are returned in any other case. #ifdef HAVE_LIBWXWIDGETS - - - static int idIx = 0 ; //ev->Desc( )->TagIndex( "ID" ); // 0 - static int topIx = 1; //ev->Desc( )->TagIndex( "TOP" ); // 1 - static int handlerIx = 2; //ev->Desc( )->TagIndex( "HANDLER" ); // 2 - - DLong actID = (*static_cast (ev->GetTag( idIx, 0 )))[0]; + DLong actID = (*static_cast (ev->GetTag( 0, 0 )))[0]; + DLong topRecurs = (*static_cast (ev->GetTag( 1, 0 )))[0]; //run-time errors (throws by interpreter etc but in widget's loop) if (ev->Desc( )->Name( ) == "*WIDGET_RUNTIME_ERROR*" ) { #ifdef GDL_DEBUG_WIDGETS @@ -323,8 +334,7 @@ DStructGDL* CallEventHandler( DStructGDL* ev ) { return NULL; } - static int handlerIx = ev->Desc( )->TagIndex( "HANDLER" ); - DLong handlerCode = (*static_cast (ev->GetTag( handlerIx, 0 )))[0]; + DLong handlerCode = (*static_cast (ev->GetTag( 2, 0 )))[0]; GDLDelete( ev ); @@ -338,10 +348,8 @@ DStructGDL* CallEventHandler( DStructGDL* ev ) { return NULL; //= OK } - - //No handler yet: set value to 0 - (*static_cast (ev->GetTag(handlerIx, 0)))[0] = 0; do { + (*static_cast (ev->GetTag(2, 0)))[0] = actID; //handler ID marked. #ifdef GDL_DEBUG_WIDGETS std::cout << "searching event handler with: " + i2s(actID) << std::endl; #endif @@ -354,42 +362,42 @@ DStructGDL* CallEventHandler( DStructGDL* ev ) { } DString eventHandlerPro = widget->GetEventPro(); if (eventHandlerPro != "") { - (*static_cast (ev->GetTag(handlerIx, 0)))[0] = actID; //handler ID marked. #ifdef GDL_DEBUG_WIDGETS std::cout << "CallEventPro: " + eventHandlerPro + " on " + i2s(actID) << std::endl; #endif - CallEventPro(eventHandlerPro, ev->Dup()); // swallows ev according to the doc, thus: - break; // out of while + CallEventPro(eventHandlerPro, ev); // swallows ev according to the doc, thus: + return NULL; // out of while } DString eventHandlerFun = widget->GetEventFun(); if (eventHandlerFun != "") { - //this a posteriori (not issued in gdlwidgeteventhandler, where handler=topFrame is the default) will define me (actID) as the handler of this event, - //which is OK as long as the ID of the originating event is eitehr me or one of my children.. - (*static_cast (ev->GetTag(handlerIx, 0)))[0] = actID; //handler ID marked. #ifdef GDL_DEBUG_WIDGETS std::cout << "CallEventFunc: " + eventHandlerFun + " on " + i2s(actID) << std::endl; #endif - DStructGDL* evstart=(DStructGDL*)(ev)->Dup(); BaseGDL* retVal = CallEventFunc(eventHandlerFun, ev); // grabs ev - //will test if ev is unchanged: + if (retVal == NULL) return NULL; //same as a procedure, has swallowed the event + //will test if ev is unchanged: if (retVal->Type() == GDL_STRUCT) { - ev = static_cast (retVal); - if (ev->Desc()->TagIndex("ID") != idIx || ev->Desc()->TagIndex("TOP") != topIx || ev->Desc()->TagIndex("HANDLER") != handlerIx) { + ev = dynamic_cast (retVal); + if (ev == NULL) { //no struct + GDLDelete(retVal); + return NULL; + } + //struct + if (ev->Desc()->TagIndex("ID") != 0 || ev->Desc()->TagIndex("TOP") != 1 || ev->Desc()->TagIndex("HANDLER") != 2) { GDLDelete(ev); throw GDLException(eventHandlerFun + ": Event handler return struct must contain ID, TOP, HANDLER as first tags."); } - GDLDelete(evstart); - } else { //not a struct, same as a procedure, has swallowed the event - ev = NULL; - return ev; + // here ev struct is passed to parent until top is reached + } else { //not a struct or no recusrsivity, same as a procedure, has swallowed the event + GDLDelete(retVal); + return NULL; } - // returned struct is a new ev: - // FUNCTION --> no break, will go up to the top or exit if consumed.! + // FUNCTION --> no break, will go up to the topRecurs or exit if consumed.! } actID = widget->GetParentID(); //go upper in hierarchy - } while (actID != GDLWidget::NullID); - // if we arrve here, all the hierarchy has been traversed - (*static_cast (ev->GetTag(handlerIx, 0)))[0] = 0; + } while (actID >= topRecurs); // not good if for some reason the increasing ID logic is changed one day. + // if we arrive here, all the hierarchy has been traversed + (*static_cast (ev->GetTag(2, 0)))[0] = 0; #endif return ev; } @@ -565,12 +573,12 @@ BaseGDL* widget_table( EnvT* e) if (e->KeywordSet(COLUMN_MAJOR)) majority = GDLWidgetTable::COLUMN_MAJOR; // if (e->KeywordSet(ROW_MAJOR)) majority = GDLWidgetTable::ROW_MAJOR; //order of preference - static int x_scroll_sizeIx = e->KeywordIx( "X_SCROLL_SIZE" ); - DLong x_scroll_size = 0; - e->AssureLongScalarKWIfPresent( x_scroll_sizeIx, x_scroll_size ); - static int y_scroll_sizeIx = e->KeywordIx( "Y_SCROLL_SIZE" ); - DLong y_scroll_size = 0; - e->AssureLongScalarKWIfPresent( y_scroll_sizeIx, y_scroll_size ); + static int x_scroll_columnIx = e->KeywordIx( "X_SCROLL_SIZE" ); //the scroll size is in COLUMNS + DLong x_scroll_column = -1; + e->AssureLongScalarKWIfPresent( x_scroll_columnIx, x_scroll_column ); + static int y_scroll_rowIx = e->KeywordIx( "Y_SCROLL_SIZE" ); //the scroll size is in ROWS + DLong y_scroll_row = -1; + e->AssureLongScalarKWIfPresent( y_scroll_rowIx, y_scroll_row ); //common for all widgets static int TRACKING_EVENTS = e->KeywordIx( "TRACKING_EVENTS" ); @@ -614,8 +622,8 @@ BaseGDL* widget_table( EnvT* e) rowLabels, // tabMode, value, - x_scroll_size, - y_scroll_size, + x_scroll_column, + y_scroll_row, valueAsStrings, eventFlags ); @@ -763,10 +771,15 @@ BaseGDL* widget_draw( EnvT* e ) { bool app_scroll = e->KeywordSet(APP_SCROLL); static int x_scroll_sizeIx = e->KeywordIx( "X_SCROLL_SIZE" ); DLong x_scroll_size = -1; - e->AssureLongScalarKWIfPresent( x_scroll_sizeIx, x_scroll_size ); + double dx_scroll_size= -1; + e->AssureDoubleScalarKWIfPresent( x_scroll_sizeIx, dx_scroll_size ); + x_scroll_size=dx_scroll_size*GetRequestedUnitConversionFactor(e).x; + static int y_scroll_sizeIx = e->KeywordIx( "Y_SCROLL_SIZE" ); DLong y_scroll_size = -1; - e->AssureLongScalarKWIfPresent( y_scroll_sizeIx, y_scroll_size ); + double dy_scroll_size = -1; + e->AssureDoubleScalarKWIfPresent( y_scroll_sizeIx, dy_scroll_size ); + y_scroll_size=dy_scroll_size*GetRequestedUnitConversionFactor(e).y; static int TOOLTIP = e->KeywordIx( "TOOLTIP" ); @@ -803,7 +816,10 @@ BaseGDL* widget_draw( EnvT* e ) { else if (val==1) {eventFlags |= GDLWidget::EV_KEYBOARD;} } DStringGDL* tooltipgdl=NULL; - if (e->KeywordPresentAndDefined(TOOLTIP)) tooltipgdl = e->GetKWAs(TOOLTIP) ; + if (e->KeywordPresentAndDefined(TOOLTIP)) { + tooltipgdl = e->GetKWAs(TOOLTIP) ; + if (tooltipgdl->N_Elements() > 1) e->Throw("Expression must be a scalar or 1 element array in this context: " + e->GetString(TOOLTIP)); + } GDLWidgetDraw* draw=new GDLWidgetDraw( parentID, e, -1, x_scroll_size, y_scroll_size, app_scroll, eventFlags, tooltipgdl); if (e->KeywordPresent(COLOR_MODEL)) static_cast(draw->GetWxWidget())->SetUndecomposed(); if (draw->GetWidgetType()==GDLWidget::WIDGET_UNKNOWN ) draw->SetWidgetType( GDLWidget::WIDGET_DRAW ); @@ -910,10 +926,14 @@ BaseGDL* widget_draw( EnvT* e ) { DLong frame_attr=0; e->AssureLongScalarKWIfPresent( tlb_frame_attrIx, frame_attr ); DLong x_scroll_size = -1; - e->AssureLongScalarKWIfPresent( x_scroll_sizeIx, x_scroll_size ); + double dx_scroll_size = -1; + e->AssureDoubleScalarKWIfPresent( x_scroll_sizeIx, dx_scroll_size ); + x_scroll_size=dx_scroll_size*GetRequestedUnitConversionFactor(e).x; DLong y_scroll_size = -1; - e->AssureLongScalarKWIfPresent( y_scroll_sizeIx, y_scroll_size ); - + double dy_scroll_size = -1; + e->AssureDoubleScalarKWIfPresent( y_scroll_sizeIx, dy_scroll_size ); + y_scroll_size=dy_scroll_size*GetRequestedUnitConversionFactor(e).y; + bool mbarPresent = e->KeywordPresent( mbarIx )||e->KeywordPresent( obsolete_app_mbarIx ); // consistency @@ -922,21 +942,30 @@ BaseGDL* widget_draw( EnvT* e ) { //besides, SPACE, XPAD and YPAD are ignored. //according to doc, Exclusive and non-exclusive base admit only button widget children, but simple tests show it is not the case for IDL up to now. - //xpad, ypad and space default to gdlPAD if not precised: - - DLong space=gdlSPACE; - if ( e->KeywordPresent(spaceIx) && !nonexclusive && !exclusive ) e->AssureLongScalarKWIfPresent( spaceIx, space ); - DLong xpad = space; - if ( !nonexclusive && !exclusive ) e->AssureLongScalarKWIfPresent( xpadIx, xpad ); - DLong ypad = space; - if ( !nonexclusive && !exclusive ) e->AssureLongScalarKWIfPresent( ypadIx, ypad ); + DLong column = 0; + e->AssureLongScalarKWIfPresent(columnIx, column); + DLong row = 0; + e->AssureLongScalarKWIfPresent(rowIx, row); + + if (column > 0 && row > 0) e->Throw("Conflicting keywords: row vs. col"); + + //xpad, ypad and space are subject to UNITS too + + DLong space = 0; + DLong xpad = 0; + DLong ypad = 0; + if (!nonexclusive && !exclusive) { + DDouble dspace = 0; + DDouble dxpad = 0; + DDouble dypad = 0; + e->AssureDoubleScalarKWIfPresent(spaceIx, dspace); + space = (column > 0) ? dspace * GetRequestedUnitConversionFactor(e).y: dspace * GetRequestedUnitConversionFactor(e).x; + e->AssureDoubleScalarKWIfPresent(xpadIx, dxpad); + xpad = dxpad * GetRequestedUnitConversionFactor(e).x; + e->AssureDoubleScalarKWIfPresent(ypadIx, dypad); + ypad = dypad * GetRequestedUnitConversionFactor(e).y; + } - DLong column = 0; - e->AssureLongScalarKWIfPresent( columnIx, column ); - DLong row = 0; - e->AssureLongScalarKWIfPresent( rowIx, row ); - - if (column>0 && row>0) e->Throw( "Conflicting keywords: row vs. col" ); DString resource_name = ""; DString rname_mbar = ""; @@ -1139,8 +1168,10 @@ BaseGDL* widget_draw( EnvT* e ) { DStringGDL* tooltipgdl = NULL; GDLWidgetButton* button; - if (e->KeywordPresentAndDefined(TOOLTIP)) tooltipgdl = e->GetKWAs(TOOLTIP); - + if (e->KeywordPresentAndDefined(TOOLTIP)) { + tooltipgdl = e->GetKWAs(TOOLTIP); + if (tooltipgdl->N_Elements() > 1) e->Throw("Expression must be a scalar or 1 element array in this context: " + e->GetString(TOOLTIP)); + } DString strvalue = "button"+i2s(buttonNumber++); //tested default! @@ -1824,20 +1855,6 @@ BaseGDL* widget_info( EnvT* e ) { static int STRING_SIZE=e->KeywordIx("STRING_SIZE"); bool getStringSize=e->KeywordPresent(STRING_SIZE); static int SIBLING=e->KeywordIx("SIBLING"); bool sibling=e->KeywordPresent(SIBLING); - //find a string, return a long - if (findbyuname) { - DStringGDL* myUname = e->GetKWAs(findbyunameIx); - if (myUname == NULL) return new DLongGDL( 0 ); - DLongGDL* list = static_cast( GDLWidget::GetWidgetsList( ) ); - Guard guard_list(list); - for (SizeT i=0; i< list->N_Elements(); ++i) { - GDLWidget* widget = GDLWidget::GetWidget( (*list)[i] ); - if ( widget != NULL ){ - if ((*myUname)[0] == widget->GetUname() ) return new DLongGDL(widget->GetWidgetID()); - } - } - return new DLongGDL( 0 ); - } if ( nParam > 0 ) { p0L = e->GetParAs(0); @@ -1923,8 +1940,24 @@ BaseGDL* widget_info( EnvT* e ) { (*res)[i]=os.str(); } return res; - } - + } + //find a string, return a long + if (findbyuname) { + DStringGDL* myUname = e->GetKWAs(findbyunameIx); + if (myUname == NULL) return new DLongGDL(0); + WidgetIDT widgetID = (*p0L)[0]; + GDLWidget *widget = GDLWidget::GetWidget( widgetID ); + DLongGDL* list = widget->GetAllHeirs(); + Guard guard_list(list); + for (SizeT i = 0; i < list->N_Elements(); ++i) { + GDLWidget* widget = GDLWidget::GetWidget((*list)[i]); + if (widget != NULL) { + if ((*myUname)[0] == widget->GetUname()) return new DLongGDL(widget->GetWidgetID()); + } + } + return new DLongGDL(0); + } + // Returns a String, empty if no result: // UNAME, FONTNAME keywords if ( uname || fontname || name ||eventpro || eventfun) { @@ -2296,7 +2329,7 @@ BaseGDL* widget_info( EnvT* e ) { else if (dropevents) return new DLongGDL(thisTreeItem->GetDroppableValue()); else if (dragnotify) return new DStringGDL(thisTreeItem->GetDragNotifyValue()); //other not implemented! } else { - if (treebitmap) e->Throw("Expression must be a scalar in this context."); + if (treebitmap) e->Throw("Expression must be a scalar in this context: "+e->GetString(TREE_BITMAP)); BaseGDL* myres; bool atLeastOneFound = false; // Array Input @@ -2439,7 +2472,8 @@ BaseGDL* widget_info( EnvT* e ) { if (dobadid) e->AssureGlobalKW(badidIx); SizeT nParam = e->NParam(); - std::vector widgetIDList; + std::vector WatchedWidgetIDList; + std::vector TopWidgetIDList; //upward-looking for handler will stop here std::vector has_children; DLongGDL* p0L = NULL; SizeT nEl = 0; @@ -2459,7 +2493,8 @@ BaseGDL* widget_info( EnvT* e ) { e->Throw("Invalid widget identifier:" + i2s((*p0L)[i])); } } - widgetIDList.push_back((*p0L)[i]); + WatchedWidgetIDList.push_back((*p0L)[i]); + TopWidgetIDList.push_back((*p0L)[i]); if (widget->NChildren() > 0) has_children.push_back(true); //NChildren() is more general than IsContainer(). //At some point, remove the discrepancy between Containers and Menus/Submenus. //The latter having a problem wrt the general structure of widgets in that they are on the stack and cannot be treated as "permanent" widgets, @@ -2467,22 +2502,24 @@ BaseGDL* widget_info( EnvT* e ) { else has_children.push_back(false); } //loop on this list, and add recursively all children when widget is a container. - SizeT currentVectorSize = widgetIDList.size(); + SizeT currentVectorSize = WatchedWidgetIDList.size(); while (1) { for (SizeT i = 0; i < currentVectorSize; i++) { if (has_children.at(i)) { + WidgetIDT topbase=TopWidgetIDList.at(i); has_children.at(i) = false; - GDLWidget *widget = GDLWidget::GetWidget(widgetIDList.at(i)); + GDLWidget *widget = GDLWidget::GetWidget(WatchedWidgetIDList.at(i)); DLongGDL* list = static_cast (widget)->GetChildrenList(); for (SizeT j = 0; j < list->N_Elements(); j++) { - widgetIDList.push_back((*list)[j]); + WatchedWidgetIDList.push_back((*list)[j]); + TopWidgetIDList.push_back(topbase); if (GDLWidget::GetWidget((*list)[j])->NChildren() > 0) has_children.push_back(true); else has_children.push_back(false); } } } - if (widgetIDList.size() == currentVectorSize) break; //no changes - currentVectorSize = widgetIDList.size(); + if (WatchedWidgetIDList.size() == currentVectorSize) break; //no changes + currentVectorSize = WatchedWidgetIDList.size(); } } else { //return default zero struct if there is no MANAGED widget on screen DLongGDL* res = static_cast( GDLWidget::GetWidgetsList( ) ); @@ -2502,56 +2539,75 @@ BaseGDL* widget_info( EnvT* e ) { DLong id; int infinity = (nowait) ? 0 : 1; DStructGDL* ev; - + + WidgetIDT top=-1; + do { // outer while loop, will run once if NOWAIT while (1) { //inner loop, catch controlC, default return if no event trapped in nowait mode GDLWidget::CallWXEventLoop(); + top=GDLWidget::NullID; if (!all) { //specific widget(s) while ((ev = GDLWidget::widgetEventQueue.Pop()) != NULL) { // get event - static int idIx = ev->Desc()->TagIndex("ID"); + static int idIx = ev->Desc()->TagIndex("ID"); id = (*static_cast (ev->GetTag(idIx, 0)))[0]; // get its id - for (SizeT i = 0; i < widgetIDList.size(); i++) { //is ID corresponding to any widget in list? - if (widgetIDList.at(i) == id) { //if yes + //check TLW destroyed here + if (ev->Desc()->Name() == "*TOPLEVEL_DESTROYED*") { + for (SizeT i = 0; i < WatchedWidgetIDList.size(); i++) { //is ID corresponding to a TLB of any widget in list? + WidgetIDT top=TopWidgetIDList.at(i); + if (top == id) { //if yes, all is consumed. + GDLDelete(ev); + return defaultRes; + } + } + } + for (SizeT i = 0; i < WatchedWidgetIDList.size(); i++) { //is ID corresponding to any widget in list? + if (WatchedWidgetIDList.at(i) == id) { //if yes + WidgetIDT top=TopWidgetIDList.at(i); + // top must be the TOP of the event, as we look for top and children here + (*static_cast (ev->GetTag(1, 0)))[0] = top ; //1:TOP goto endwait; } } - GDLWidget::widgetEventQueue.PushBack(ev); - GDLWidget::CallWXEventLoop(); - // avoid looping like crazy -#ifdef _WIN32 - Sleep(10); // this just to quiet down the character input from readline. 2 was not enough. 20 was ok. -#else - const long SLEEP = 10000000; // 10ms - struct timespec delay; - delay.tv_sec = 0; - delay.tv_nsec = SLEEP; // 20ms - nanosleep(&delay, NULL); -#endif +// GD 2025: suppress mouse movements while in this loop (if they have been not trapped above) + if (ev->Desc()->Name() != "WIDGET_DRAW" ) GDLWidget::widgetEventQueue.PushBack(ev); + else { + DInt type = (*static_cast (ev->GetTag( 3, 0 )))[0]; //TYPE Tag + if (type != 2) GDLWidget::widgetEventQueue.PushBack(ev); //remove unhandled mouse MOVEMENTS + } + GDLWidget::CallWXEventLoop(); } } else { - //wait for ALL . This is the case of /XMANAGER_BLOCK for example. + //wait for ALL . This is the case /XMANAGER_BLOCK for example. if ((ev = GDLWidget::widgetEventQueue.Pop()) != NULL) goto endwait; } if (nowait) return defaultRes; if (sigControlC) return defaultRes; + // avoid looping like crazy +#ifdef _WIN32 + Sleep(1); // this just to quiet down the character input from readline. 2 was not enough. 20 was ok. +#else + const long SLEEP = 1000000; // 1ms + struct timespec delay; + delay.tv_sec = 0; + delay.tv_nsec = SLEEP; + nanosleep(&delay, NULL); +#endif } //end inner loop //here we got a real event, process it, walking back the hierachy (in CallEventHandler()) for modified ev in case of function handlers. endwait: - if (blockedByXmanager && ev->Desc()->Name() == "*TOPLEVEL_DESTROYED*") { + if (/* blockedByXmanager && */ ev->Desc()->Name() == "*TOPLEVEL_DESTROYED*") { // deleted widgets list are hopefully handled internally by xmanager GDLDelete(ev); return defaultRes; } - ev = CallEventHandler(ev); //process it recursively (going up hierarchy) in eventHandler. Should block waiting for xmanager. - // examine return: - if (ev == NULL) { //swallowed by a procedure or non-event-stucture returning function + ev = CallEventHandler(ev); //process it recursively (going up hierarchy up to ev.top or before if so programmed) in eventHandler. + // examine return: + if (ev == NULL) { //swallowed by a procedure or non-event-stucture returning function : looping wait for another event if (nowait) return defaultRes; //else will loop again } else { // untreated or modified by a function return ev; } - GDLWidget::CallWXEventLoop(); - } while (infinity); return NULL; //pacifier. #endif //HAVE_LIBWXWIDGETS @@ -2936,31 +2992,20 @@ void widget_control( EnvT* e ) { if (hasScr_xsize || hasScr_ysize) { //simple: direct sizing in pixels or UNITS for ALL widgets DLong xsize=-1, ysize=-1; - if (hasScr_xsize) {xsize= (*e->GetKWAs(SCR_XSIZE))[0]; if (xsize<0) e->Throw("Illegal keyword value for SCR_XSIZE.");} - if (hasScr_ysize) {ysize= (*e->GetKWAs(SCR_YSIZE))[0]; if (ysize<0) e->Throw("Illegal keyword value for SCR_YSIZE.");} + if (hasScr_xsize) {xsize= (*e->GetKWAs(SCR_XSIZE))[0]*GetRequestedUnitConversionFactor(e).x; if (xsize<0) e->Throw("Illegal keyword value for SCR_XSIZE.");} + if (hasScr_ysize) {ysize= (*e->GetKWAs(SCR_YSIZE))[0]*GetRequestedUnitConversionFactor(e).y; if (ysize<0) e->Throw("Illegal keyword value for SCR_YSIZE.");} wxWindow* me=dynamic_cast(widget->GetWxWidget()); if (me==NULL) e->Throw("Geometry request not allowed for menubar or pulldown menus."); - - wxRealPoint fact = wxRealPoint(1.,1.); - if ( unitsGiven ) { - fact = GetRequestedUnitConversionFactor( e ); - if (hasScr_xsize) xsize*=fact.x; - if (hasScr_ysize) ysize*=fact.y; - } + widget->SetWidgetScreenSize(xsize,ysize); } if ( (hasDraw_xsize || hasDraw_ysize ) && widget->IsDraw() ) { DLong xsize=-1, ysize=-1; - if (hasDraw_xsize) {xsize= (*e->GetKWAs(DRAW_XSIZE))[0]; if (xsize<0) e->Throw("Illegal keyword value for DRAW_XSIZE.");} - if (hasDraw_ysize) {ysize= (*e->GetKWAs(DRAW_YSIZE))[0]; if (ysize<0) e->Throw("Illegal keyword value for DRAW_YSIZE.");} - wxRealPoint fact = wxRealPoint(1.,1.); - if ( unitsGiven ) { - fact = GetRequestedUnitConversionFactor( e ); - if (hasDraw_xsize) xsize*=fact.x; - if (hasDraw_ysize) ysize*=fact.y; - } + if (hasDraw_xsize) {xsize= (*e->GetKWAs(DRAW_XSIZE))[0]*GetRequestedUnitConversionFactor(e).x; if (xsize<0) e->Throw("Illegal keyword value for DRAW_XSIZE.");} + if (hasDraw_ysize) {ysize= (*e->GetKWAs(DRAW_YSIZE))[0]*GetRequestedUnitConversionFactor(e).y; if (ysize<0) e->Throw("Illegal keyword value for DRAW_YSIZE.");} + widget->SetWidgetVirtualSize(xsize,ysize); } @@ -2971,20 +3016,17 @@ void widget_control( EnvT* e ) { if (whatSortofBut->IsMenu() || whatSortofBut->IsEntry()) e->Throw("Geometry request not allowed for menubar or pulldown menus."); } DLong xsize=-1, ysize=-1; - if (hasXsize) {xsize= (*e->GetKWAs(XSIZE))[0]; if (xsize<0) e->Throw("Illegal keyword value for XSIZE.");} - if (hasYsize) {ysize= (*e->GetKWAs(YSIZE))[0]; if (ysize<0) e->Throw("Illegal keyword value for YSIZE.");} wxWindow* me=dynamic_cast(widget->GetWxWidget()); if (!me) e->Throw("Geometry request not allowed for menubar or pulldown menus."); if (!(widget->IsList() || widget->IsTable() || widget->IsText())) { - wxRealPoint fact = wxRealPoint(1.,1.); - if ( unitsGiven ) { - fact = GetRequestedUnitConversionFactor( e ); - if (hasXsize) xsize*=fact.x; - if (hasYsize) ysize*=fact.y; - } - } + if (hasXsize) {xsize= (*e->GetKWAs(XSIZE))[0]*GetRequestedUnitConversionFactor(e).x; if (xsize<0) e->Throw("Illegal keyword value for XSIZE.");} + if (hasYsize) {ysize= (*e->GetKWAs(YSIZE))[0]*GetRequestedUnitConversionFactor(e).y; if (ysize<0) e->Throw("Illegal keyword value for YSIZE.");} + } else { + if (hasXsize) {xsize= (*e->GetKWAs(XSIZE))[0]; if (xsize<0) e->Throw("Illegal keyword value for XSIZE.");} + if (hasYsize) {ysize= (*e->GetKWAs(YSIZE))[0]; if (ysize<0) e->Throw("Illegal keyword value for YSIZE.");} + } widget->SetWidgetSize(xsize,ysize); } @@ -3164,14 +3206,7 @@ void widget_control( EnvT* e ) { if ((val1->Type() != GDL_LONG) || (val2->Type() != GDL_LONG) || (val3->Type() != GDL_LONG)) { e->Throw("Invalid SEND_EVENT value."); } - DLongGDL* lval1 = static_cast (val1); - DLongGDL* lval2 = static_cast (val2); - DLongGDL* lval3 = static_cast (val3); - WidgetIDT baseWidgetID = widget->GetBaseId(widgetID); - if ((*lval1)[0] == 0) (*lval1)[0] = widgetID; - if ((*lval2)[0] == 0) (*lval2)[0] = baseWidgetID; - if ((*lval3)[0] == 0) (*lval3)[0] = baseWidgetID; - GDLWidget::PushEvent(baseWidgetID, ev); + GDLWidget::PushEvent(ev); } if (clear_events) widget->ClearEvents(); @@ -3384,7 +3419,8 @@ void widget_control( EnvT* e ) { DStringGDL* tlbTitle=e->GetKWAs( tlbsettitleIx ); wxString tlbName = wxString( (*tlbTitle)[0].c_str( ), wxConvUTF8 ); topFrame->SetTitle(tlbName); - } + } else { + topFrame->SetEvtHandlerEnabled(false); if (settlbxoffset) { DLongGDL* xoffset=e->GetKWAs( tlbsetxoffsetIx ); if (unitsGiven) { @@ -3398,6 +3434,8 @@ void widget_control( EnvT* e ) { wxRealPoint fact=GetRequestedUnitConversionFactor(e); topFrame->Move(topFrame->GetPosition().x, (*yoffset)[0]*fact.y ); } else topFrame->Move(topFrame->GetPosition().x, (*yoffset)[0] ); + } + topFrame->SetEvtHandlerEnabled(true); } } @@ -3543,7 +3581,7 @@ void widget_control( EnvT* e ) { static int SET_DROPLIST_SELECT = e->KeywordIx( "SET_DROPLIST_SELECT" ); if (e->KeywordPresentAndDefined(SET_DROPLIST_SELECT)) { DLongGDL* droplistSelection = e->GetKWAs(SET_DROPLIST_SELECT); - if (droplistSelection->N_Elements() > 1) e->Throw( "Expression must be a scalar or 1 element array in this context:"); + if (droplistSelection->N_Elements() > 1) e->Throw( "Expression must be a scalar or 1 element array in this context: " + e->GetString(SET_DROPLIST_SELECT)); droplist->SelectEntry((*droplistSelection)[0]); } } @@ -3581,7 +3619,7 @@ void widget_control( EnvT* e ) { static int SET_COMBOBOX_SELECT = e->KeywordIx( "SET_COMBOBOX_SELECT" ); if (e->KeywordPresentAndDefined(SET_COMBOBOX_SELECT)) { DLongGDL* comboSelection = e->GetKWAs(SET_COMBOBOX_SELECT); - if (comboSelection->N_Elements() > 1) e->Throw( "Expression must be a scalar or 1 element array in this context:"); + if (comboSelection->N_Elements() > 1) e->Throw( "Expression must be a scalar or 1 element array in this context: " + e->GetString(SET_COMBOBOX_SELECT)); combo->SelectEntry((*comboSelection)[0]); } static int COMBOBOX_ADDITEM = e->KeywordIx( "COMBOBOX_ADDITEM" ); diff --git a/src/widget.hpp b/src/widget.hpp index 7b5fcd6f0..ac54dd7b5 100644 --- a/src/widget.hpp +++ b/src/widget.hpp @@ -50,5 +50,5 @@ namespace lib { BaseGDL* CallEventFunc( const std::string& f, BaseGDL* ev); void CallEventPro( const std::string& p, BaseGDL* p0, BaseGDL* p1 = NULL); -DStructGDL* CallEventHandler( /*DLong id,*/ DStructGDL* ev); +DStructGDL* CallEventHandler(DStructGDL* ev); #endif diff --git a/testsuite/interactive_tests/example_event_handling.pro b/testsuite/interactive_tests/example_event_handling.pro new file mode 100644 index 000000000..153d23bbe --- /dev/null +++ b/testsuite/interactive_tests/example_event_handling.pro @@ -0,0 +1,57 @@ +; this example shows 3 different event 'history': +; - an event_pro, exit immediately when 'treated' +; - an event_func, passed from level while reporting the level +; - an event_func as above, but changed to event_pro at some level +pro tlb_catch_final_event,ev + print,"tlb_catch_final_event called at "+strtrim(ev.handler,2) +; help,ev + if size(ev,/sname) eq "WIDGET_TLB_MOVE" then widget_control,ev.handler,tlb_set_xoff=ev.x,tlb_set_yoff=ev.y else print,"not a TLB_MOVE_EVENT" +end +function show,ev + print,"show called at "+strtrim(ev.handler,2) +; help,ev + return,ev +end + +pro report_procedure_event,ev + print,'report_procedure_event called at '+strtrim(ev.id,2)+" handled by "+strtrim(ev.handler,2) +end + +function report_function_event,ev + print,"report_function_event called at "+strtrim(ev.id,2)+" handled by "+strtrim(ev.handler,2) +; help,ev +; return,ev + nev={WIDGET_TLB_MOVE} + nev.handler=44; widget_info(ev.id,/parent) + nev.id=ev.id + nev.top=1 + nev.x=1000*randomu(seed) + nev.y=1000*randomu(seed) +; if ev.handler eq 5 then return,ev + return,nev +end + +function changemind,ev + print,"changemind called at "+strtrim(ev.id,2)+" handled by "+strtrim(ev.handler,2) + widget_control,ev.top+4,event_fun="" + widget_control,ev.top+4,event_pro="tlb_catch_final_event" + return,ev;report_function_event(ev) +end + +pro done,ev + widget_control,ev.top,/destroy +end + +n=8 & bases=lonarr(n) +bases[0]=widget_base(/col,event_pro="tlb_catch_final_event",/tlb_move) +for i=1,n-1 do bases[i]=widget_base(bases[i-1],/col,event_func="show") +b1=widget_button(bases[n-1],value="function_event",event_func="report_function_event",tooltip="event function: will traverse the "+strtrim(n,2)+" bases up to top and make the widget bounce" ) +b2=widget_button(bases[n-1],value="procedure_event",event_pro="report_procedure_event",tooltip="event procedure: will be trapped only at tlb level") +b3=widget_button(bases[n-1],value="function_to_procedure",event_fun="changemind",tooltip="event function, transformed to procedure at tlb+4" ) +b4=widget_button(bases[n-1],value="exit",event_pro="done") + +;foreach i,[2L,3,4,5] do widget_control,bases[i],event_func="" +widget_control,bases[0],/realize +z=widget_event(bases[0]) +;xmanager,"report_procedure",bases[0] +end diff --git a/testsuite/interactive_tests/test_widgets.pro b/testsuite/interactive_tests/test_widgets.pro index bef311b2d..4bf5a7cb5 100644 --- a/testsuite/interactive_tests/test_widgets.pro +++ b/testsuite/interactive_tests/test_widgets.pro @@ -81,27 +81,25 @@ pro exit_gui,ev widget_control,ev.top,/DESTROY end pro cleanup_xmanager, id - print,"Cleaning up called by xmanager..." + print,"Cleaning up "+string(id)+" called by xmanager..." widget_control,id,/DESTROY end pro cleanup, id - print,"Cleaning up called by base widget..." + print,"Cleaning up "+string(id)+" called by base widget..." widget_control,id,/DESTROY end pro cleanup_control, id - print,"Cleaning up called by widget_control..." + print,"Cleaning up "+string(id)+" called by widget_control..." widget_control,id,/DESTROY end pro i_am_realized, id print,"Widget "+string(id)+" is realized now." end -pro base_event_base, id - print,"event in bases base" - help,id +pro base_event_base, ev + print,"event in bases base from id="+strtrim(ev.id,2) end -pro base_event, id - print,"event in top base" - help,id +pro base_event, ev + print,"event in top base from id="+strtrim(ev.id,2) end pro slider_killed,id print,"slider "+string(id)+" was killed" @@ -348,7 +346,7 @@ pro test_widgets,table,help=help,nocanvas=nocanvas,notree=notree,block=block,fon green_bmp= bytarr(7,7,3)& green_bmp[*,*,1] = 255& & green_bmp[0,0,1] = 254 red_bmp= bytarr(7,7,3)& red_bmp[*,*,0] = 255& & red_bmp[0,0,0] = 254 if (n_elements(select) gt 0) then present=select - if (n_elements(present) eq 0) then present=['TEXT','LIST','DRAW','SLIDER','BUTTON','TABLE','TREE','LABEL','DROPLIST','COMBOBOX','BASE'] + if (n_elements(present) gt 0) then present=strupcase(temporary(present)) else present=['TEXT','LIST','DRAW','SLIDER','BUTTON','TABLE','TREE','LABEL','DROPLIST','COMBOBOX','BASE'] count=0 title='GDL Widget Examples' DEFSYSV, '!gdl', exists=isGDL @@ -384,7 +382,7 @@ ev = {vEv,type:'',pos:[0,0]} ;Create a base widget. -base = WIDGET_BASE(/col,MBAR=mbar,title=title,event_pro='base_event_nok',kill_notify='cleanup',/tlb_kill_request_events,/tlb_size_events) ; ---> PROBLEM: ,/tlb_size_events) ;,/scroll) +base = WIDGET_BASE(/col,MBAR=mbar,title=title,event_pro='base_event',kill_notify='cleanup',/tlb_kill_request_events,/tlb_size_events,/tracking_events) ; ---> PROBLEM: ,/tlb_size_events) ;,/scroll) doMbar,mbar,fontname ;mysize=widget_info(base,string_size='012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234') ; add a progress bar to test timer @@ -426,6 +424,8 @@ if total(strcmp('TEXT',present,/fold)) then begin xsize=20, ysize=5,/wrap,/all_events) & offy+=50 ; label=widget_label(yoff=offy,text_base,value='Is EDITABLE + ALL_EVENTS: input change widget content and events are generated') & offy+=20 ; text1=widget_text(yoff=offy,text_base,VALUE=["EDITABLE, all_events","Line 2","line 3","line 4","line 5"],/EDITABLE,/ALL_EVENTS,ysize=3) & offy+=30 ; + label=widget_label(yoff=offy,text_base,value='As above but ony one line: TAB events are not generated') & offy+=20 ; + text1=widget_text(yoff=offy,text_base,VALUE=["EDITABLE, all_events but one line"],/EDITABLE,/ALL_EVENTS) & offy+=30 ; label=widget_label(yoff=offy,text_base,value='Is NOT EDITABLE + ALL_EVENTS: no change but events are generated') & offy+=20 ; text1=widget_text(yoff=offy,text_base,VALUE=["NOT EDITABLE, all_events","Line 2","line 3","line 4","line 5"],EDITABLE=0,/ALL_EVENTS) & offy+=30 ; label=widget_label(yoff=offy,text_base,value='Is only EDITABLE: input change widget content and generates an event.') & offy+=20 ; @@ -435,23 +435,25 @@ endif if total(strcmp('LIST',present,/fold)) then begin ;LIST list_base = widget_base( tabbed_base, TITLE="LISTs",_extra=extra) & offy=0 ; - + textlist=["Lorem ipsum dolor sit amet,", "consectetur adipiscing elit,", "sed do eiusmod tempor", "ut labore et dolore magna.", "Ut enim ad minim,", "quis nostrud exercita", "ullamco laboris", "ex ea commodo."] label=widget_label(yoff=offy,list_base,value='3 lines, placement inherited') & offy+=10 ; - list=widget_list(yoff=offy,list_base,value=["Simple rather long text of widget_list, ysize=3","created","with","WIDGET_LIST","widget_list, ysize=3","created","with","WIDGET_LIST", "ysize=3","created","with","WIDGET_LIST","widget_list, ysize=3","created","with","WIDGET_LIST","widget_list, ysize=3","created","with","WIDGET_LIST","widget_list, ysize=3","created","with","WIDGET_LIST", "ysize=3","created","with","WIDGET_LIST","widget_list, ysize=3","created","with","WIDGET_LIST"],/multiple,YSIZE=3) & offy+=100 ; - label=widget_label(yoff=offy,list_base,value='idem,fancy,framed(32 pix)') & offy+=10 ; - list=widget_list(yoff=offy,list_base,value=["FANCY FONT long text widget_list, ysize=3","created","with","WIDGET_LIST","widget_list, ysize=3","created","with","WIDGET_LIST", "ysize=3","created","with","WIDGET_LIST","widget_list, ysize=3","created","with","WIDGET_LIST"],/multiple,YSIZE=3,frame=32,font=fontname) & offy+=200 ; - label=widget_label(yoff=offy,list_base,value='3 lines, placement inherited, xsize=12 chars (scroll)') & offy+=10 ; - list=widget_list(yoff=offy,list_base,value=["widget_list, ysize=3","created","with","WIDGET_LIST"],/multiple,YSIZE=3,xsize=12) & offy+=150 ; - label=widget_label(yoff=offy,list_base,value='1 line, placement inherited, xsize=2 chars (scroll)') & offy+=10 ; - list=widget_list(yoff=offy,list_base,value=["widget_list, ysize=3","created","with","WIDGET_LIST"],/multiple,xsize=2) & offy+=150 ; - label=widget_label(yoff=offy,list_base,value='no size,fancy,framed(32 pix)') & offy+=10 ; - list=widget_list(yoff=offy,list_base,value=["widget_list","(no YSIZE, frame=32)"],frame=32,font=fontname) & offy+=50 ; - label=widget_label(yoff=offy,list_base,value='centered,frame=1',/align_center) & offy+=10 ; - list=widget_list(yoff=offy,list_base,value=["(centered)"," frame=1"],/align_center,frame=1) & offy+=50 ; + list=widget_list(yoff=offy,list_base,value=textlist,/multiple,YSIZE=3) & offy+=100 ; + label=widget_label(yoff=offy,list_base,value='idem,fancy,framed(32 pix)->scrollbar') & offy+=10 ; + list=widget_list(yoff=offy,list_base,value=textlist,/multiple,XSIZE=20,YSIZE=3,frame=32,font=fontname) & offy+=200 ; + label=widget_label(yoff=offy,list_base,value='3 lines, placement inherited') & offy+=10 ; + list=widget_list(yoff=offy,list_base,value=textlist,/multiple,YSIZE=3) & offy+=150 ; + label=widget_label(yoff=offy,list_base,value='1 line, placement inherited, xsize=32 (ignored)') & offy+=10 ; + list=widget_list(yoff=offy,list_base,value=textlist,/multiple,xsize=32, ysize=1) & offy+=150 ; + label=widget_label(yoff=offy,list_base,value='no size,fancy,framed(32 pix) -> scrollbar') & offy+=10 ; + list=widget_list(yoff=offy,list_base,value=textlist,frame=32,font=fontname) & offy+=50 ; + label=widget_label(yoff=offy,list_base,value='centered,ysize=1,xsize=40,frame=1 ->scrollbar',/align_center) & offy+=10 ; + list=widget_list(yoff=offy,list_base,value=textlist,ysize=1,xsize=40,/align_center,frame=1) & offy+=50 ; + label=widget_label(yoff=offy,list_base,value='centered,ysize=1,no xsize,frame=1 ->scrollbar',/align_center) & offy+=10 ; + list=widget_list(yoff=offy,list_base,value=textlist,ysize=1,/align_center,frame=1) & offy+=50 ; label=widget_label(yoff=offy,list_base,value='basic') & offy+=10 ; - list=widget_list(yoff=offy,list_base,value=["default"," basic"]) & offy+=50 ; + list=widget_list(yoff=offy,list_base,value=textlist) & offy+=50 ; label=widget_label(yoff=offy,list_base,value='centered,basic',/align_center) & offy+=10 ; - list=widget_list(yoff=offy,list_base,value=["(centered)","basic"],/align_center) & offy+=10 ; + list=widget_list(yoff=offy,list_base,value=textlist,/align_center) & offy+=10 ; endif if total(strcmp('DRAW',present,/fold)) then begin @@ -467,10 +469,10 @@ endif draw3 = WIDGET_DRAW(xoff=200,yoff=offy,draw_base,/BUTTON_EVENTS, xsize=100,ysize=100,tooltip="WIGDET_DRAW",EVENT_PRO = 'draw_context',uname='drawToBeDeleted') & offy+=100 ; contextBase = WIDGET_BASE(yoff=offy,draw3, /CONTEXT_MENU,col=2,TITLE="ZZZZZZZZZZZZZZ",UNAME = 'drawContext') & offy+=10; b1 = WIDGET_BUTTON(yoff=offy,contextBase, VALUE = 'Delete this draw widget', /SEPARATOR, EVENT_PRO = 'DeleteDraw') - b2 = WIDGET_BUTTON(contextBase, VALUE = 'just an entry') & offy+=10; - b2 = WIDGET_BUTTON(contextBase, VALUE = 'just an entry, checked', /check) & offy+=10 ; + b2 = WIDGET_BUTTON(contextBase, VALUE = 'deleting this widget (above)') & offy+=10; + b2 = WIDGET_BUTTON(contextBase, VALUE = 'will generate an error', /check) & offy+=10 ; widget_control,b2,/set_button - b2 = WIDGET_BUTTON(contextBase, VALUE = 'just an entry') & offy+=10; + b2 = WIDGET_BUTTON(contextBase, VALUE = 'when deleting the other, this is normal.') & offy+=10; b3 = WIDGET_BUTTON(contextBase, VALUE = 'a menu', /menu) & offy+=10; b4 = WIDGET_BUTTON(b3 , VALUE = 'an item.') & offy+=10; b5 = WIDGET_BUTTON(b3 , VALUE = 'Delete this draw widget (again)', /SEPARATOR, EVENT_PRO = 'DeleteDraw') @@ -512,7 +514,7 @@ endif button_base00 = widget_base( tabbed_base, TITLE="BUTTONs", COL=2, $ SPACE=10, XPAD=10, YPAD=10) & offy=10 - button_base01 = widget_base(button_base00, TITLE="BUTTONs",/COL, event_func='catch_passed_event_example') & offy=10 + button_base01 = widget_base(button_base00, TITLE="BUTTON+EVENT",/COL, event_func='catch_passed_event_example') & offy=10 button_base02 = widget_base(button_base00, TITLE="BUTTONs",/COL) & offy=10 ; BUTTONs tmp=widget_label(yoff=offy,button_base01,value="Simple ON/OFF Button") & offy+=10 ; @@ -524,7 +526,7 @@ endif tmp=widget_label(yoff=offy,button_base01,value="Bitmap Simple Button") & offy+=10 ; tmp=widget_button(yoff=offy,button_base01,value=myBitmap()) & offy+=10 ; tmp=widget_label(yoff=offy,button_base01,value="Fancy Simple Button") & offy+=10 ; - tmp=widget_button(yoff=offy,button_base01,value="Fancy Button",font=fontname, event_func='test_func_button') & offy+=10 ; + tmp=widget_button(yoff=offy,button_base01,value="Fancy Button+Event",font=fontname, event_func='test_func_button') & offy+=10 ; tmp=widget_label(yoff=offy,button_base01,value="Exclusive base, framed 30") & offy+=10 ; radio=widget_base(yoff=offy,button_base01,/EXCLUSIVE,COL=1,frame=30) & offy+=150 ; rb1=widget_button(radio,VALUE="button in EXCLUSIVE base 1",uvalue={vEv,'rb1',[8,0]}, font=fontname) @@ -753,8 +755,6 @@ widget_control, labeltoupdate, set_value="Text2 has been replaced with widget_co ; overwrite buttons; for iRow=0,nRows-1 do widget_control,fileButtons[iRow], set_value="Y" -;; - endif @@ -799,5 +799,5 @@ endif ; create a timer event --- otherwise will not be active and thus not catched in eventloop widget_control,progressbar,timer=0 -xmanager,"handle",base,cleanup="cleanup_xmanager";,no_block=~block +xmanager,"handle",base,cleanup="cleanup_xmanager",no_block=~block end