AllBASIC Forum

BASIC Developer & Support Resources => Interpreters => Topic started by: Steve A. on December 19, 2010, 09:37:01 AM

Title: sequential vs events driven
Post by: Steve A. on December 19, 2010, 09:37:01 AM
sequential vs events driven

Hey guys,
To all you language developers out there:

In the construction of Bxbasic, since Bxbasic has been up to this point
a Console mode, sequentially driven compiler/interpreter, I've never
had to be bothered about event handling.
i.e.: mouse roll-over, on-click, on-button, etc.

From those of you who are familiar with GUI and event driven models,
I'd like to know just how you are dealing with coding event handlers.

Related to Bxbasic, up to now, I've never done any GUI, event driven coding.
At this moment, I'm working on some GUI stuff, so that I can incorporate
some "Windows" specific GUI to Bxbasic.

One of the things I've uncovered and am toying with, is the notion that
I may not need two distinct compilers/interpreters to handle Console
mode -vs- GUI mode operation.

What I mean by that is, by using a single scripting engine that is
capable of handling both Console mode and GUI mode scripts, I may be
able to construct a single compiler/interpreter. As opposed to a distinct
Console mode engine and a seperate GUI mode engine.

That is, instead of requireing the user to select whether they want to
build a Console mode or GUI mode application, they simply make that
distinction by opening a GUI window or not, within the script.
Assuming that a standard console is the default method.

Or..., conversely,
if GUI is the default method, by not making a window visible at the start,
either a console can be opened or a GUI window can be made visible.

I have already constructed a test platform, whereby a console app can open
a GUI window. And, I know that a GUI window has the ability to open a
Console Mode window, (stdin/stdout and all).
And, I can open both, simultaniously, sending stdout to the console and
displaying text and graphics to the GUI window.

The issue I have to resolve is Windows events.

Programming in C/C++, generally WinProc() handles the events aspect.
Programming a Console mode engine is pretty straight forward, you simply
read-in what the next command is and execute it.

Example: from Bxbasic: function parser():

void parser()
{
    switch(token)
    {   case 1:                      /* LET */
            parse_let();
            break;
        case 2:                      /* CLEAR */
            clr_vars();
            break;
        case 3:                      /* LOCATE */
            locate();
            break;
        case 4:                      /* PRINT */
            parse_print();
            break;
        case 5:                      /* GOTO */
            go_to();
            break;
        case 6:                      /* BEEP */
            beep();
            break;
        case 7:                      /* CLS */
            cls();
            break;
<snip>


The question becomes; in a GUI mode engine, how do you test for events
and handle event messages ?

Those of you who are constructing GUI engines, are you dealing with
and handling events ?
Or, are you programming GUI in a sequential manner ?

Your input is greatly appreciated.
Steve

<cc: other forums as well>
Title: Re: sequential vs events driven
Post by: JRS on December 19, 2010, 11:46:24 PM
Steve,

The trend I have noticed is to use multi-platform GUI toolkit. (Gtk, Qt, wxWidgets, ...) You're kind of stuck with the winAPI (message loops, SDKs, ...) on Windows.

I'm using GTK with ScriptBasic and it works great for the applications I write.

John
Title: Re: sequential vs events driven
Post by: jcfuller on December 20, 2010, 05:11:39 AM
I'm using wxWidgets on both Linux & Windows with Bcx,Ubx,Mbc but it is a c++ library.
With wxWidgets you get native look on windows.

James
Title: Re: sequential vs events driven
Post by: JRS on January 02, 2011, 02:35:26 AM
I believe MDI is MS Windows answer to a multiple window / single process applications. This is why I'm going the multi-threaded route so each window can operate independently.
Title: Re: sequential vs events driven
Post by: JRS on January 02, 2011, 11:19:56 AM
No John...
This is not about MDI ,i know what you mean.
I mean open few windows as separate window or as child .

How would you manage the Windows message pump from multiple windows in a single threaded process? After using Linux as my full time desktop environment for the last few months, I keep saying to myself, was I that lazy and set in my ways that I would accept grief and frustration as a normal aspect of my day?
Title: Re: sequential vs events driven
Post by: JRS on January 03, 2011, 02:18:33 PM
Quote
I have in plan use pointer for message loop.

Maybe one of the Windows gurus out there can shed some light on how to deal with multiple windows in a single process and not use a MDI container.

Good luck, I'm also interested in the feedback you get on this topic.



Title: Re: sequential vs events driven
Post by: Steve A. on January 04, 2011, 05:44:14 PM
Thanks John...
I looking for Steve respond ,maby he have something concrete on his mind.

I don't have the answer for this Aurel.
I'm trying to avoid using (for the moment) GTK, wxWidgets, etc.
I want to learn how it all works using GDI, OpenGL, etc.

I guess what I want to do is create a "message pump" and intercept any app related messages and process them.
The problem I have is that I don't quite know where to begin..., what API functions I need to implement.
Title: Re: sequential vs events driven
Post by: JRS on January 04, 2011, 09:25:22 PM
Quote
I don't have the answer for this Aurel.
I'm trying to avoid using (for the moment) GTK, wxWidgets, etc.

To quote a Godaddy site pitch, "You could be up and running tonight!"

Why don't you give GTK-Server a try. It comes with a Windows generic DLL and installing Gtk runtime under Windows takes less than a minute. Gtk-Server is more than just a Gtk GUI solution. I have scripted complex Linux interfaces with it and added threading to to mix. Until you know where your going, join the GTK-Server fan club and I'll even share my scripts with you.  8)
Title: Re: sequential vs events driven
Post by: Steve A. on January 05, 2011, 09:17:04 AM
Here is code in C++ maby you get some point :
 ...[snip]

In EBasic i can crete one sub which contain adress of window procedure
and inside this sub is intercepted all window messages.

Hey Aurel,
I thoroughly understand the code snippet you provided for WinMain and WinProc.
What I don't understand is how I turn those into User Definable Commands.
Actually, it has more to do with WinProc. WinMain is not that hard to figure out.
I will explain what I mean by that, with an example, in my next post.
Title: Re: sequential vs events driven
Post by: Steve A. on January 05, 2011, 10:36:31 AM
Okay, below is C code for a sample WinProc() callback function.
This is a normal looking WinProc(), which you might expect to see in any normal Windows GUI application.

Code: [Select]
LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
    PAINTSTRUCT ps;
    RECT        rect;
    HDC         hdc;

    switch(uMsg)
    {
        case WM_CREATE:
            hdc = GetDC(hwnd);
            break;

        case WM_PAINT:
            {
                BeginPaint(hwnd, &ps);

                GetClientRect(hwnd, &rect);
                DrawText(hdc, "Demo program using a private DC.", -1, &rect, DT_SINGLELINE | DT_CENTER | DT_VCENTER);
                DrawLine(hwnd);
                MakeNewFont(hwnd);
                DrawBox(hwnd);

                DrawRect(hwnd);
                DrawRndRect(hwnd);
                EndPaint(hwnd, &ps);
            }
            break;

        case WM_LBUTTONDOWN:    // left clicking on the client area of the window (white part)
            break;

        case WM_RBUTTONDOWN:    // right clicking on the client area of the window (white part)
            break;

        case WM_CHAR:   // character key
            break;

        case WM_MOVE:   // moving the window
            break;

        case WM_SIZE:
            break;

        case WM_DESTROY:
            PostQuitMessage(0);
            break;

        default:
            return DefWindowProc(hwnd, uMsg, wParam, lParam);
    }
    return 0;
}
/* ------ end WndProc ------ */


The problem, as I see it, is that a scripting engine/interpreter is not a normal Windows GUI application.
Or, perhaps it is and I'm just missing something.

If you look at the code snippet for the command parser, from my original post, shown here:

Code: [Select]
void parser()
{
    switch(token)
    {   case 1:                      /* LET */
            parse_let();
            break;
        case 2:                      /* CLEAR */
            clr_vars();
            break;
        case 3:                      /* LOCATE */
            locate();
            break;
        case 4:                      /* PRINT */
            parse_print();
            break;
        case 5:                      /* GOTO */
            go_to();
            break;
        case 6:                      /* BEEP */
            beep();
            break;
        case 7:                      /* CLS */
            cls();
            break;
<snip>

you can see that each Bxbasic command is listed sequentially in a switch/case loop.

What I have not figured out is:
1) how to combine the message loop with the command parser loop,
2) assuming I create Bxbasic commands to:
    a) create and open a GUI window,
    b) draw graphics in the client area and
    c) output text to the client area,

how does the user intercept the messages shown in the WinProc example ?
Not only that, but, how does the user define what messages to intercept ?
Title: Re: sequential vs events driven
Post by: JRS on January 05, 2011, 12:30:52 PM
Steve,

Is your plans to remain a Windows only interpreter? It's a bit crowded in that space, don't you think? Even with ScriptBasic being an embeddable Basic, I have had very little interest from Windows users. If BXBasic is just a personal challenge and meant to be a tutorial on writing an interpreter, no need to read any further and may the force be with you.

I'm trying to find a niche by extending SB with dynamic scripting of 3rd party libraries and system APIs. The multi-threaded effort is just a personal challenge trying to find the limits of SB.

John
Title: Re: sequential vs events driven
Post by: Steve A. on January 05, 2011, 12:37:48 PM
Hey Aurel,
Thanks for your help.
I appreciate your assistance and detailed explaining.
I may figure this out yet.
Title: Re: sequential vs events driven
Post by: Steve A. on January 05, 2011, 12:57:05 PM
Hey John,

Is your plans to remain a Windows only interpreter?

No, that's not really it.
Before I commit to using a tool kit, I'd prefer to know how it works, at the lowest levels.
Sure, I could simply Plug'N Play a tool kit, but, that wouldn't tell me anything about how it works.
If a bug were to exist in either my code or in the 'tool kit', I'd prefer to know from an internal perspective how things work.

Quote
It's a bit crowded in that space, don't you think?

Oh, I'm sure.
New avenues are opening almost daily.

Quote
If BXBasic is just a personal challenge and meant to be a tutorial on writing an interpreter, ...

Well, I would like to write Volume-II.
In it I would like to take on the current topic of discussion.
Title: Re: sequential vs events driven
Post by: Steve A. on January 06, 2011, 07:46:03 AM
Here you can see screenshot of testing how i open multiple windows.
http://aurelbasicsoft.ucoz.com/AOnepreview.png

Hmmm..., simple, interesting.
Title: Re: sequential vs events driven
Post by: JRS on January 06, 2011, 06:22:41 PM
Aurel,

Nice job! Do you have all 6 windows sending messages to the IDE message processing loop?

Side Note:

I'm surprised you stayed on with E-Basic with the company being sold over and over.  What motivation keeps you from looking at other options? I understand that E-Basic is a community supported Basic compiler. Wouldn't it make more sense to take on the role of a project manager and support E-Basic as a language rather than creating an interpreter for beginners with it? I see E-Basic following the same road the RapidQ ended on. Does the new owner of the IWBasic version plan on future releases? My guess is Paul will come up with a new compiler that will encompass everything he has done in the past. Basic has found it's niche as a hobbyist language, creating custom applications for small businesses and a teaching aid for leaning how to program.

John

P.S.

E-Basic running under Wine with both a Mac and XP theme.
Title: Re: sequential vs events driven
Post by: JRS on January 06, 2011, 11:44:13 PM
If your not interested in my threaded adventures, here is a multi-window example running as a single process script in scriba.

(http://files.allbasic.info/AllBasic/gtk10.png)

Code: [Select]
DECLARE SUB DLL ALIAS "_idll" LIB "gtk-server"
DECLARE SUB DEFINE ALIAS "_idll_define" LIB "gtk-server"

DEFINE "gtk_init NONE NONE 2 NULL NULL"
DEFINE "gtk_window_new delete-event WIDGET 1 INT"
DEFINE "gtk_window_set_title NONE NONE 2 WIDGET STRING"
DEFINE "gtk_table_new NONE WIDGET 3 INT INT BOOL"
DEFINE "gtk_container_add NONE NONE 2 WIDGET WIDGET"
DEFINE "gtk_button_new_with_label clicked WIDGET 1 STRING"
DEFINE "gtk_table_attach_defaults NONE NONE 6 WIDGET WIDGET INT INT INT INT"
DEFINE "gtk_widget_show_all NONE NONE 1 WIDGET"
DEFINE "gtk_server_callback NONE STRING 1 STRING"
DEFINE "gtk_widget_destroy NONE NONE 1 WIDGET"
DEFINE "gtk_server_exit NONE NONE 0"

DLL("gtk_init NULL NULL")

FOR h = 1 to 10
  win[h] = DLL("gtk_window_new 0")
  DLL("gtk_window_set_title " & win[h] & " \"SB Gtk " & h & "\"")
  tbl[h] = DLL("gtk_table_new 10 10 1")
  DLL("gtk_container_add " & win[h] & " " & tbl[h])
  but[h] = DLL("gtk_button_new_with_label \"Quit\"")
  DLL("gtk_table_attach_defaults " & tbl[h] & " " & but[h] & " 5 9 5 9")
  DLL("gtk_widget_show_all " & win[h])
NEXT h

win_cnt = 10
REPEAT
  event = DLL("gtk_server_callback WAIT")
  FOR e = 1 TO 10
    IF event = win[e] OR event = but[e] THEN
      DLL("gtk_widget_destroy " & win[e])
      win_cnt -= 1
    END IF
  NEXT e
UNTIL win_cnt = 0

DLL("gtk_server_exit")

END
Title: Re: sequential vs events driven
Post by: JRS on January 07, 2011, 12:20:39 AM
Gtk runs on Windows, Linux and the Mac. My Linux example would run untouched on Windows.

(http://files.allbasic.info/ScriptBasic/abmain.png)

(http://files.allbasic.info/ScriptBasic/linux_abmain.png)
Title: Re: sequential vs events driven
Post by: JRS on January 10, 2011, 03:23:42 PM
Aurel and Steve,

Any news on your efforts for Windows native GUI support for BXBasic or ABasic multi-window message handler?

There seemed to be some momentum in this area and I think it's a topic many Basic programmers have had a hard time getting a handle on.

John

Title: Re: sequential vs events driven
Post by: JRS on January 11, 2011, 05:49:58 AM
I found an interesting link that gives a good explanation how the Windows message loop works and how it deals with multiple windows. This seems to be a science all of it's own.

http://www.winprog.org/tutorial/message_loop.html

Years ago James Fuller came up with a message cracker scheme that made dealing with Windows messages easier. Maybe James can chime in and unravel the mystery.
Title: Re: sequential vs events driven
Post by: jcfuller on January 11, 2011, 08:10:23 AM

Years ago James Fuller came up with a message cracker scheme that made dealing with Windows messages easier. Maybe James can chime in and unravel the mystery.

For a compiler using Function Pointers. I doubt it would be applicable?

James
Title: Re: sequential vs events driven
Post by: Steve A. on January 11, 2011, 08:15:13 AM
@ John:
    Thanks for the link. Helpful.

@ James:
    could you explain a little more about it, or, provide a link ?

Steve
Title: Re: sequential vs events driven
Post by: jcfuller on January 11, 2011, 10:25:32 AM
@ James:
    could you explain a little more about it, or, provide a link ?

Steve


Steve,
 A little knowledge of PowerBASIC would help but with your background you should be able to see what I'm doing here.
If not just yell.
Note the original code was created by a code generator not by hand.

James
Code: [Select]
Ok this is how I did it for PowerBASIC.
All Dialog messages would go through the same event function here and use this udt:

#IF NOT %DEF(%cEventInfoType)
  %cEventInfoType = 1
  TYPE EventInfoType
    MaxDlgMsgs AS LONG
    DlgMsgs         AS DWORD PTR
    DlgProcs        AS DWORD PTR
    MaxCtlMsgs      AS LONG
    CtlMsgs         AS DWORD PTR
    CtlProcs        AS DWORD PTR
    MaxNotifyMsgs AS LONG
    NotifyCtlIds AS DWORD PTR
    NotifyProcs AS DWORD PTR
    ExitFlag AS LONG
  END TYPE
#ENDIF
'===========================================================================
'Dummy Declare for CALL DWORD Message Cracker
DECLARE _
  FUNCTION _
    DlgCallBackProc ( _
      BYVAL hDlg      AS DWORD,_
      BYVAL wMsg      AS DWORD,_
      BYVAL wParam    AS DWORD,_
      BYVAL lParam    AS LONG _
    ) AS LONG
'----------------------------------------------------------------------------
'~Dlg_Event_Proc            [Sub/Func Tag]
'----------------------------------------------------------------------------
FUNCTION _
  Dlg_Event_Proc ( _
    BYVAL hDlg AS DWORD,_
    BYVAL wMsg AS DWORD,_
    BYVAL wParam AS DWORD,_
    BYVAL lParam AS LONG _
  )  AS LONG

  DIM i AS LONG, _
      pEventInfo AS EventInfoType PTR, _
      dwTemp AS DWORD
 

  IF wMsg = %WM_INITDIALOG THEN
   
    #IF %DEF(%ADP_MACROS)
      m_SetDataInfo(hDlg,lParam)     
    #ELSE
      SetWindowLong hDlg,%DWL_USER,lParam
    #ENDIF 
   
    pEventInfo = lParam
    dwTemp = @pEventInfo.@DlgProcs[0]
  ELSE
    #IF %DEF(%ADP_MACROS)
      pEventInfo = m_GetDataInfo(hDlg)
    #ELSE
      pEventInfo = GetWindowLong(hDlg,%DWL_USER)
    #ENDIF 
    IF ISFALSE pEventInfo THEN
      EXIT FUNCTION
    END IF
    DO
      IF i > @pEventInfo.MaxDlgMsgs THEN
        EXIT DO
      END IF
      IF @pEventInfo.@DlgMsgs[i] = wMsg THEN
        dwTemp = @pEventInfo.@DlgProcs[i]
        EXIT DO
      END IF
      INCR i
    LOOP
  END IF
  IF dwTemp THEN
    CALL DWORD dwTemp _
      USING DlgCallBackProc(hDlg,_
                            wMsg,_
                            wParam,_
                            lParam) TO i
    FUNCTION = i
    SetWindowLong hDlg, %DWL_MSGRESULT, i
  END IF
  END FUNCTION
'***********************************************************************
'**********************************************************************


This is the Dialog code:


'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=

'                  This Source Code was produced using PBWinADP

'        Target: Win32
'    Basic File: DLG_120.BAS
' Resource File: DLG_120.RC
'  Date Created: 12-31-2000
'  Time Created: 13:45:16

'Control & Menu ID EQUATES FOR DLG_120
'----------------------------------------------------------------------------
%DLG_120_CB_TSFTTP = 104
%DLG_120_CB_TSFTP = 105
%DLG_120_CB_TSATP = 106
%DLG_120_CB_SOST = 109
%DLG_120_CB_SOAT = 110
%DLG_120_CB_PFSUP = 111
%DLG_120_CB_BOAT = 107
%DLG_120_CB__PFSUS = 112
%DLG_120_CB_SOFT = 108
%DLG_120_IDC_BUTTON10 = 102
%DLG_120_IDC_STATICFRAME1 = 103
%DLG_120_IDC_STATICTEXT1 = 113
%DLG_120_IDC_RADIOBUTTON1 = 114
%DLG_120_IDC_RADIOBUTTON2 = 115
%DLG_120_IDC_RADIOBUTTON3 = 116
%DLG_120_IDC_STATICFRAME2 = 117
%DLG_120_CB_TA = 121
%DLG_120_RB_IMPERIAL = 120
%DLG_120_RB_METRIC = 119

'----------------------------------------------------------------------------
'The following UDT is used for encapsulation of message cracker arrays.
'If you need any private data just add to the UDT.
'----------------------------------------------------------------------------
TYPE DLG_120_InfoType
  MaxDlgMsgs           AS LONG
  DlgMsgs              AS DWORD PTR
  DlgProcs             AS DWORD PTR
  MaxCtlMsgs           AS LONG
  CtlMsgs              AS DWORD PTR
  CtlProcs             AS DWORD PTR
  MaxNotifyMsgs        AS LONG
  NotifyCtlIds         AS DWORD PTR
  NotifyProcs          AS DWORD PTR
  ExitFlag             AS LONG
  DlgId    AS LONG
  IsModal    AS LONG
'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
'>> DO NOT ALTER ANYTHING ABOVE. ADD ALL NEW ENTRIES BELOW
'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
END TYPE
'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
'                    FUNCTION PROTOTYPES
'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
'----------------------------------------------------------------------------
'Call the following function to create your dialog.
'The function expects a global instance handle of ghInst.
'This var is created if you use RC2BAS to create your main dialog
'The function will return %FALSE if there was an error
'hWnd is the handle of the calling Window/Dialog.
'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=

'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
'~Create_DLG_120          [Sub/Func Tag]
'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
FUNCTION _
  Create_DLG_120 ( _
    hWnd AS DWORD _
  ) AS LONG

  DIM lpMem AS DWORD,_
      lpDataInfo AS DLG_120_InfoType PTR, _
      MsgIndex AS LONG, _
      MemOffset AS DWORD, _
      RetVal AS LONG
  DIM hDlg AS DWORD
  lpMem = HeapAlloc(ghHeap,0 OR %HEAP_ZERO_MEMORY,LEN(DLG_120_InfoType))
  IF ISFALSE lpMem THEN
    MsgBox "Not Able to Allocate Mem"
    FUNCTION = 0
    EXIT FUNCTION
  END IF
  lpDataInfo = lpMem
  @lpDataInfo.DlgId = 120
  @lpDataInfo.IsModal = 0

'---------------------------------------------------------------------------
'>> Message Crackers
  #INCLUDE "DLG_120.MSG"

  hDlg = CreateDialogParam(ghInst,BYVAL 120,hWnd,CODEPTR(Dlg_Event_Proc),lpMem)
  IF ISFALSE hDlg THEN
    HeapFree ghHeap,0,@lpDataInfo.DlgMsgs
    HeapFree ghHeap,0,lpMem
    FUNCTION = 0
    EXIT FUNCTION
  END IF
  ghMainMenu = hDlg
  ShowWindow hDlg,%SW_SHOW
  FUNCTION = hDlg
END FUNCTION              'Create_DLG_120
'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
'~Dlg_120_ShowPickListForm
'------------------------------------------------------------------------------
FUNCTION _
  Dlg_120_ShowPickListForm ( _
    BYVAL hDlg      AS DWORD,_
    BYVAL wMsg      AS DWORD,_
    BYVAL wParam    AS DWORD,_
    BYVAL lParam    AS LONG _
  ) AS LONG

  LOCAL IdVal&, _
        IdCodePtr AS DWORD, _
        RetVal&, _
        szTitle AS ASCIIZ * 80

  IdVal& = LOWRD(wParam)
  IF IsDlgButtonChecked(hDlg,%DLG_120_IDC_RADIOBUTTON1) THEN
    IF IdVal& = %DLG_120_CB_TSFTTP THEN
      IdCodePtr = %Dlg_220'CODEPTR(Create_Dlg_220)
      szTitle = "SDJ Electra [-CAD4OCS-] Trolley System Fixed Termination Trolley Pole"
    ELSEIF IdVal& = %DLG_120_CB_TSFTP THEN
      IdCodePtr = %Dlg_300'CODEPTR(Create_Dlg_300)
      szTitle = "SDJ Electra [-CAD4OCS-] Trolley System Fixed Termination Pantograph" 
    ELSEIF IdVal& = %DLG_120_CB_TSATP THEN
      IdCodePtr = %Dlg_400'CODEPTR(Create_Dlg_400)
      szTitle = "SDJ Electra [-CAD4OCS-] Trolley System Auto Tensioned Pantograph" 
    ELSEIF IdVal& = %DLG_120_CB_SOST THEN
      IdCodePtr = %Dlg_700'CODEPTR(Create_Dlg_700)
      szTitle = "SDJ Electra [-CAD4OCS-] Simple OCS Semi Tensioned"
    ELSEIF IdVal& = %DLG_120_CB_SOAT THEN
      IdCodePtr = %Dlg_800'CODEPTR(Create_Dlg_800)
      szTitle = "SDJ Electra [-CAD4OCS-] Simple OCS Auto Tensioned"
    ELSEIF IdVal& = %DLG_120_CB_PFSUP THEN
      IdCodePtr = %Dlg_900'CODEPTR(Create_Dlg_900)
      szTitle = "SDJ Electra [-CAD4OCS-] Parallel Feeders Supported"
    ELSEIF IdVal& = %DLG_120_CB_BOAT THEN
      IdCodePtr = %Dlg_500'CODEPTR(Create_Dlg_500)
      szTitle = "SDJ Electra [-CAD4OCS-]  Bridle OCS Auto Tensioned"
    ELSEIF IdVal& = %DLG_120_CB__PFSUS THEN
      IdCodePtr = %Dlg_1000'CODEPTR(Create_Dlg_1000)
      szTitle = "SDJ Electra [-CAD4OCS-] Parallel Feeders Suspended"
    ELSEIF IdVal& = %DLG_120_CB_SOFT THEN
      IdCodePtr = %Dlg_600'CODEPTR(Create_Dlg_600)
      szTitle = "SDJ Electra [-CAD4OCS-] Simple OCS Fixed Termination"
    ELSEIF IdVal& =   %DLG_120_CB_TA THEN
      'MessageBox hDlg,"Shell To Track Alignment Data Input","Track Alignment",%MB_OK
      IdCodePtr = %Dlg_1300'CODEPTR(Create_Dlg_1300)
      szTitle = "SDJ Electra [-CAD4OCS-] Track Alignment"
    END IF
  ELSEIF IsDlgButtonChecked(hDlg,%DLG_120_IDC_RADIOBUTTON2) THEN
    IF IdVal& = %DLG_120_CB_TSFTTP THEN
      IdCodePtr = %Dlg_1600
      szTitle = "SDJ Electra [-CAD4OCS-] Trolley System Fixed Termination Trolley Pole Calculations"
    END IF 
'     IF IdVal& =   %DLG_120_CB_TA THEN
'       IdCodePtr = CODEPTR(Create_Dlg_1200)
       'gTrackAlignFlag = 1
'       MessageBox hDlg,"Do Track Alignment Calculations","Track Alignment",%MB_OK
'       EXIT FUNCTION
'     END IF 
'    IF IdVal& = %DLG_120_CB_TSFTTP THEN
      'Create_Dlg_1100 hDlg
 '   ELSE
 '   MessageBox hDlg,"Calculations are not Implemented at present","Comming soon....",%MB_OK
 '   END IF
  ELSEIF IsDlgButtonChecked(hDlg,%DLG_120_IDC_RADIOBUTTON3) THEN   
    MessageBox hDlg,"Reports are not Implemented at present","No Reports",%MB_OK
  END IF
   
   

  IF IdCodePtr THEN
    ShowWindow hDlg,%SW_HIDE
    ghCurrentPickList& = ghPickList&(IdCodePtr)
    ShowWindow ghCurrentPickList&,%SW_SHOW
    SetWindowText ghHoldWin, szTitle
    ShowWindow ghDlg1400,%SW_SHOW
'    ShowWindow ghDlg1400,%SW_HIDE
'    CALL DWORD IdCodePtr USING Create_Dlg_220(GetParent(hDlg)) 
'    CALL DWORD IdCodePtr USING Create_Dlg_220(ghDlg1400) 
  END IF 
   
   
   
END FUNCTION 
'============================================================================
'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
'Message Cracker Functions For DLG_120
'============================================================================
'>> Dialog Functions
'============================================================================
'============================================================================
'~DLG_120_Onwmactivate            [Sub/Func Tag]
'============================================================================
FUNCTION _
  DLG_120_Onwmactivate ( _
    BYVAL hDlg        AS DWORD,_
    BYVAL wMsg        AS DWORD,_
    BYVAL wParam      AS DWORD,_
    BYVAL lParam      AS DWORD _
  ) AS DWORD

  DIM lpDataInfo       AS DLG_120_InfoType PTR

  lpDataInfo = m_GetDataInfo(hDlg)


 
END FUNCTION
'============================================================================
'~DLG_120_OnWmInitDialog           [Sub/Func Tag]
'============================================================================
FUNCTION _
  DLG_120_OnWmInitDialog ( _
    BYVAL hDlg        AS DWORD,_
    BYVAL wMsg        AS DWORD,_
    BYVAL wParam      AS DWORD,_
    BYVAL lParam      AS LONG _
  ) AS LONG

'  DIM lpDataInfo       AS DLG_120_InfoType PTR

'  lpDataInfo = GetWindowLong(hDlg,%DWL_USER)

  Initialize_Controls hDlg
'---------------------------------------------------------------------------
  CenterWindow hDlg
  CheckDlgButton hDlg,%DLG_120_IDC_RADIOBUTTON1,%BST_CHECKED
  CheckDlgButton hDlg,%DLG_120_RB_METRIC, %BST_CHECKED
'  PostMessage ghHoldWin,%WM_MAKE_1400,0,0
 FUNCTION = 1
END FUNCTION
'============================================================================
'~DLG_120_OnWmDestroy          [Sub/Func Tag]
'============================================================================
FUNCTION _
  DLG_120_OnWmDestroy ( _
    BYVAL hDlg AS DWORD,_
    BYVAL wMsg AS DWORD,_
    BYVAL wParam AS DWORD,_
    BYVAL lParam AS LONG _
  ) AS LONG

  DIM lpDataInfo      AS DLG_120_InfoType PTR

  #IF %TRACEIT = 2
    TRACE ON
    TRACE PRINT "WM _DESTROY IN " + FUNCNAME$
    TRACE OFF
  #ENDIF

  lpDataInfo = m_GetDataInfo(hDlg)'GetWindowLong(hDlg,%DWL_USER)
 
  #IF %DEF(%DLG_120_USE_OD_BUTTONS)
    EnumChildWindows hDlg,CODEPTR(ODButFreeMem2),0
  #ENDIF

  EnumChildWindows hDlg,CODEPTR(Control_FreeMem),0
 
  IF lpDataInfo THEN
   
      HeapFree ghHeap,0,@lpDataInfo.DlgMsgs
      HeapFree ghHeap,0,lpDataInfo
 
     
    m_setDataInfo(hDlg,0)
  END IF 
'  PostMessage GetParent(hDlg),%WM_CLOSE,0,0 
 FUNCTION = 0
END FUNCTION
'============================================================================
'~DLG_120_OnWmClose            [Sub/Func Tag]
'============================================================================
FUNCTION _
  DLG_120_OnWmClose ( _
    BYVAL hDlg        AS DWORD,_
    BYVAL wMsg        AS DWORD,_
    BYVAL wParam      AS DWORD,_
    BYVAL lParam      AS LONG _
  ) AS LONG

  DestroyWindow hDlg
  FUNCTION = 0
END FUNCTION
'============================================================================
'============================================================================
'~DLG_120_OnWmDrawItem            [Sub/Func Tag]
'============================================================================
FUNCTION _
  DLG_120_OnWmDrawItem ( _
    BYVAL hDlg AS DWORD,_
    BYVAL wMsg      AS DWORD,_
    BYVAL wParam    AS DWORD,_
    BYVAL lParam    AS DWORD _
  ) AS DWORD

#IF %DEF(%DLG_120_USE_OD_BUTTONS)

 
  LOCAL lpODButInfo AS ODBut_InfoType PTR ,_
        di AS DRAWITEMSTRUCT PTR,_
        lpCtlInfo AS ControlInfoType PTR, _
        bm AS BITMAP,_
      hBitmap AS DWORD,_
      rFocus AS RECT,_
      DcRect            AS RECT,_
      NoFocusFlag       AS INTEGER

  di = lParam

  lpCtlInfo = GetProp(@di.hWndItem,"CtlInfo")
  IF ISFALSE lpCtlInfo THEN
    EXIT FUNCTION
  END IF
   
  lpODButInfo = @lpCtlInfo.dwx

  IF ISFALSE lpODButInfo THEN
    EXIT FUNCTION
  END IF
  IF (@di.ItemState AND %ODS_SELECTED) THEN
    hBitMap = @lpODButInfo.hBitmapDn
    rFocus = @lpODButInfo.rFocusDn
  ELSE
    hBitmap = @lpODButInfo.hBitmapUp
    rFocus = @lpODButInfo.rFocusUp
  END IF
  GetObject hBitmap,LEN(BITMAP),bm
  DcRect.nRight = bm.bmWidth
  DcRect.nBottom = bm.bmHeight

  IF (@di.ItemAction AND %ODA_DRAWENTIRE) OR _
    (@di.ItemAction AND %ODA_SELECT) THEN
    PaintBitmap @di.hDc,DcRect,hBitMap,DcRect,@lpODButInfo.hPal
  END IF

  'Need some code for focus
  IF ISFALSE @lpODButInfo.NoFocusFlag  THEN
    IF (@di.ItemState AND %ODS_FOCUS) THEN
      FrameRect @di.hDc,rFocus,GetStockObject(%DKGRAY_BRUSH)
    END IF
  END IF
  FUNCTION = 1
#ENDIF
END FUNCTION
'============================================================================
'>> Control Functions
'============================================================================
'~DLG_120_IDC_BUTTON10_OnClick          [Sub/Func Tag]
'============================================================================
FUNCTION _
  DLG_120_IDC_BUTTON10_OnClick ( _
    BYVAL hDlg      AS DWORD,_
    BYVAL wMsg      AS DWORD,_
    BYVAL wParam    AS DWORD,_
    BYVAL lParam    AS LONG _
  ) AS LONG

   'SendMessage ghPickList&(%Dlg_1400),%WM_CLOSE,0,0
   PostMessage GetParent(hDlg),%WM_MAKE_1400,hDlg,0 
END FUNCTION
'============================================================================
'***********************************************************************
'***********************************************************************




And this is the setup for the callabcks:
This is the   #INCLUDE "DLG_120.MSG" file from the code above


'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
'DLG_120
'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
@lpDataInfo.MaxDlgMsgs = 10
@lpDataInfo.MaxCtlMsgs = 11
@lpDataInfo.MaxNotifyMsgs =  0
'---------------------------------------------------------------------------
MemOffset = HeapAlloc(ghHeap,%HEAP_ZERO_MEMORY,(@lpDataInfo.MaxDlgMsgs + _
                               @lpDataInfo.MaxCtlMsgs + _
                               @lpDataInfo.MaxNotifyMsgs)*8)
IF ISFALSE MemOffset THEN
  MsgBox "Message Array memory allocation error"
  FUNCTION = 0
  EXIT FUNCTION
END IF
'---------------------------------------------------------------------------
IF @lpDataInfo.MaxDlgMsgs THEN
   @lpDataInfo.DlgMsgs = MemOffset
   MemOffset = MemOffset + (@lpDataInfo.MaxDlgMsgs * 4)
   @lpDataInfo.DlgProcs = MemOffset
   MemOffset = MemOffset + (@lpDataInfo.MaxDlgMsgs * 4)
END IF
IF @lpDataInfo.MaxCtlMsgs THEN
   @lpDataInfo.CtlMsgs = MemOffset
   MemOffset = MemOffset + (@lpDataInfo.MaxCtlMsgs * 4)
   @lpDataInfo.CtlProcs = MemOffset
   MemOffset = MemOffset + (@lpDataInfo.MaxCtlMsgs * 4)
END IF
IF @lpDataInfo.MaxNotifyMsgs THEN
   @lpDataInfo.NotifyCtlIds =  MemOffset
   MemOffset = MemOffset + (@lpDataInfo.MaxNotifyMsgs * 4)
   @lpDataInfo.NotifyProcs = MemOffset
END IF
'===========================================================================
'>> BEGIN DLG_MSGS
'===========================================================================
  MsgIndex = 0
'---------------------------------------------------------------------------
'~%WM_INITDIALOG   (%WM_INITDIALOG IS ALWAYS FIRST)
'---------------------------------------------------------------------------
  @lpDataInfo.@DlgMsgs[MsgIndex] = %WM_INITDIALOG
  @lpDataInfo.@DlgProcs[MsgIndex] = CODEPTR(DLG_120_OnWmInitDialog)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~%WM_COMMAND
'---------------------------------------------------------------------------
  @lpDataInfo.@DlgMsgs[MsgIndex] = %WM_COMMAND
  @lpDataInfo.@DlgProcs[MsgIndex] = CODEPTR(Ctl_Event_Proc)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~%WM_CTLCOLORSTATIC
'---------------------------------------------------------------------------
  @lpDataInfo.@DlgMsgs[MsgIndex] = %WM_CTLCOLORSTATIC
  @lpDataInfo.@DlgProcs[MsgIndex] = CODEPTR(WM_CTLCOLOR_Proc)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~%WM_CTLCOLORBTN
'---------------------------------------------------------------------------
  @lpDataInfo.@DlgMsgs[MsgIndex] = %WM_CTLCOLORBTN
  @lpDataInfo.@DlgProcs[MsgIndex] = CODEPTR(WM_CTLCOLOR_Proc)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~%WM_CTLCOLOREDIT
'---------------------------------------------------------------------------
  @lpDataInfo.@DlgMsgs[MsgIndex] = %WM_CTLCOLOREDIT
  @lpDataInfo.@DlgProcs[MsgIndex] = CODEPTR(WM_CTLCOLOR_Proc)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~%WM_CTLCOLORLISTBOX
'---------------------------------------------------------------------------
  @lpDataInfo.@DlgMsgs[MsgIndex] = %WM_CTLCOLORLISTBOX
  @lpDataInfo.@DlgProcs[MsgIndex] = CODEPTR(WM_CTLCOLOR_Proc)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~%WM_CLOSE
'---------------------------------------------------------------------------
  @lpDataInfo.@DlgMsgs[MsgIndex] = %WM_CLOSE
  @lpDataInfo.@DlgProcs[MsgIndex] = CODEPTR(DLG_120_OnWmClose)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~%WM_DESTROY
'---------------------------------------------------------------------------
  @lpDataInfo.@DlgMsgs[MsgIndex] = %WM_DESTROY
  @lpDataInfo.@DlgProcs[MsgIndex] = CODEPTR(DLG_120_OnWmDestroy)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~%WM_DRAWITEM
'---------------------------------------------------------------------------
  @lpDataInfo.@DlgMsgs[MsgIndex] = %WM_DRAWITEM
  @lpDataInfo.@DlgProcs[MsgIndex] = CODEPTR(DLG_120_OnWmDrawItem)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~%WM_ACTIVATE
'---------------------------------------------------------------------------
  @lpDataInfo.@DlgMsgs[MsgIndex] = %WM_ACTIVATE
  @lpDataInfo.@DlgProcs[MsgIndex] = CODEPTR(DLG_120_Onwmactivate)
  INCR MsgIndex
'---------------------------------------------------------------------------
'>> END DLG_MSGS
'===========================================================================
'>> BEGIN CTL_MSGS
'===========================================================================
  MsgIndex = 0
'---------------------------------------------------------------------------
'~DLG_120_CB_TSFTTP_OnClick >> %BN_CLICKED <<
'---------------------------------------------------------------------------
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_CB_TSFTTP,%BN_CLICKED)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(Dlg_120_ShowPickListForm)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~DLG_120_CB_TSFTP_OnClick >> %BN_CLICKED <<
'---------------------------------------------------------------------------
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_CB_TSFTP,%BN_CLICKED)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(Dlg_120_ShowPickListForm)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~DLG_120_CB_TSATP_OnClick >> %BN_CLICKED <<
'---------------------------------------------------------------------------
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_CB_TSATP,%BN_CLICKED)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(Dlg_120_ShowPickListForm)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~DLG_120_CB_SOST_OnClick >> %BN_CLICKED <<
'---------------------------------------------------------------------------
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_CB_SOST,%BN_CLICKED)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(Dlg_120_ShowPickListForm)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~DLG_120_CB_SOAT_OnClick >> %BN_CLICKED <<
'---------------------------------------------------------------------------
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_CB_SOAT,%BN_CLICKED)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(Dlg_120_ShowPickListForm)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~DLG_120_CB_PFSUP_OnClick >> %BN_CLICKED <<
'---------------------------------------------------------------------------
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_CB_PFSUP,%BN_CLICKED)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(Dlg_120_ShowPickListForm)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~DLG_120_CB_BOAT_OnClick >> %BN_CLICKED <<
'---------------------------------------------------------------------------
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_CB_BOAT,%BN_CLICKED)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(Dlg_120_ShowPickListForm)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~DLG_120_CB__PFSUS_OnClick >> %BN_CLICKED <<
'---------------------------------------------------------------------------
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_CB__PFSUS,%BN_CLICKED)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(Dlg_120_ShowPickListForm)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~DLG_120_CB_SOFT_OnClick >> %BN_CLICKED <<
'---------------------------------------------------------------------------
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_CB_SOFT,%BN_CLICKED)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(Dlg_120_ShowPickListForm)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~DLG_120_IDC_BUTTON10_OnClick >> %BN_CLICKED <<
'---------------------------------------------------------------------------
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_IDC_BUTTON10,%BN_CLICKED)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(DLG_120_IDC_BUTTON10_OnClick)
  INCR MsgIndex
'---------------------------------------------------------------------------
'~Dlg_120_ShowPickListForm    << MENU >>
  @lpDataInfo.@CtlMsgs[MsgIndex] = MAKDWD(%DLG_120_CB_TA,0)
  @lpDataInfo.@CtlProcs[MsgIndex] = CODEPTR(Dlg_120_ShowPickListForm)
  INCR MsgIndex
''---------------------------------------------------------------------------
'>> END CTL_MSGS
'===========================================================================
'>> BEGIN NOTIFY_MSGS
'===========================================================================
  MsgIndex = 0
'---------------------------------------------------------------------------
'>> END NOTIFY_MSGS
'===========================================================================
'~EOM         End of Messages
'=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=








Title: Re: sequential vs events driven
Post by: JRS on January 11, 2011, 10:09:47 PM
No one said that managing the Windows message loop was going to be easy. It's amazing the number of messages, the rules, the weirdness and then have to catch the few messages you really care about. I'm so glad I made the Linux and Gtk move. I get more done with less code. The other benefit I see using Gtk is if you see something in an application that attracts you, look at the code (doesn't matter what language it is) and see how the Gtk functions are used to help build your own application.
Title: Re: sequential vs events driven
Post by: JRS on January 12, 2011, 11:31:22 AM
I could use some help defining this Windows structure to pass as an argument in ScriptBasic for Windows. I'm going to use the DYC (DynaCall) extension module to script a simple window (winAPI  style) for tutorial purposes. SB is a nice way to prototype a concept.

Code: [Select]
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,
    LPSTR lpCmdLine, int nCmdShow)
{
    WNDCLASSEX wc;
    HWND hwnd;

    // Registering the Window Class
    wc.cbSize        = sizeof(WNDCLASSEX);
    wc.style         = 0;
    wc.lpfnWndProc   = WndProc;
    wc.cbClsExtra    = 0;
    wc.cbWndExtra    = 0;
    wc.hInstance     = hInstance;
    wc.hIcon         = LoadIcon(NULL, IDI_APPLICATION);
    wc.hCursor       = LoadCursor(NULL, IDC_ARROW);
    wc.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
    wc.lpszMenuName  = NULL;
    wc.lpszClassName = g_szClassName;
    wc.hIconSm       = LoadIcon(NULL, IDI_APPLICATION);

    RegisterClassEx(&wc)
    }



lpfnWndProc will be the challenge for SB.

Quote
The window procedure is called for each message, the HWND parameter is the handle of your window, the one that the message applies to. This is important since you might have two or more windows of the same class and they will use the same window procedure (WndProc()). The difference is that the parameter hwnd will be different depending on which window it is. For example when we get the WM_CLOSE message we destroy the window. Since we use the window handle that we received as the first paramter, any other windows will not be affected, only the one that the message was intended for.

I can see right off the bat that creating this Windows class structure and returning a pointer to it is best done at the BCX level. I'm setting up my Wine SB Windows development environment and will post the current SBAPI.dll source to this thread when done. At this point it will have the SB VARPTR and a new GenWinClass helper functions along with the statically linked libscriba.obj.
Title: Re: sequential vs events driven
Post by: JRS on January 12, 2011, 05:33:31 PM
The Windows DYC ScriptBasic extension module works under Linux/Wine. Glad I don't have to do this under Windows.

(http://files.allbasic.info/AllBasic/wine-dyc.png)

Code: [Select]
DECLARE SUB DLL ALIAS "dyc" LIB "dyc"

PRINT DLL("ms,i,USER32.DLL,MessageBox,PZZL", 0, "message text", "title", 3)

C:\scriptbasic\test>
6
C:\scriptbasic\test>

Quote
Titlebar Exit = 2
Yes = 6
No = 7
Cancel = 2

I was curious why the msgbox wasn't using the XP theme that I assigned to Wine. I tried the same thing under Windows and it had the same Win2000 style buttons.

(http://www.scriptbasic.org/forum/link_docs/msgbox.gif)
Title: Re: sequential vs events driven
Post by: JRS on January 12, 2011, 08:33:22 PM
Can a window callback procedure point to an exported DLL function?

I'm trying to use the same concept as Peter did with gtk_server_callback WAIT.

AutoIt has an interesting approach to dealing with callbacks. DllCallBack (http://files.allbasic.info/AllBasic/DllCallBack.txt)  DllStructCreate (http://www.autoitscript.com/autoit3/docs/functions/DllStructCreate.htm)

Here is an example of it's use.

Code: [Select]
#include "DllCallBack.au3"



Global $sOut

$hStub_EnumChildProc = _DllCallBack ("_EnumChildProc", "hwnd;ptr")

Func _EnumChildProc($hWnd, $lParam)

$hWnd = HWnd($hWnd)

$sOut &= "hWnd: " & $hWnd & @CRLF

Return 1

EndFunc   ;==>_EnumChildProc

$hWnd_Parent = WinGetHandle("[ACTIVE]") ; SciTE

DllCall("user32.dll", "int", "EnumChildWindows", "hwnd", $hWnd_Parent, "ptr", $hStub_EnumChildProc, "ptr", 0)

ConsoleWrite($sOut & @CRLF)

_DllCallBack_Free ($hStub_EnumChildProc)