-
Notifications
You must be signed in to change notification settings - Fork 1
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Support WithEvents
in standard modules
#42
Comments
FWIW, you'd just get the same functionality with a More generally, I think it's a problem with VBx language that it encourages people to not use objects the way it should be used. They are enticed to do everything from modules and promiscuously pollute the global namespace with everything without setting up proper abstraction layer between components. We want to make it easy for people to do the right thing. Handling events from a module isn't it, I think. Modules should be treated as if it was stateless, IMO. |
Modules in VBx are very lightweight very much like plain C source files. Getting rid of standard modules + UDTs style of programming instead of a more object-oriented one will damage the language IMO. Rust and golang stay away of everything-class mantra that C# and Java brought to this world and prefer cheap modules and traits instead of object inheritance. Lastly code in modules is very easily pruned by not-so-advanced linkers too. If a function gets compiled to an .obj file but it is not referenced in any other part of the code the linker is smart enough to remove its codegen from the final executable. This is something that cannot be (cheaply) done for class methods as each of the underlying functions is referenced by the vtable of the default COM interface so the linker thinks these chunks of code are in use. Edit: I wouldn't want to incur the hidden |
But you're suggesting that modules should support events, which basically means it will be treated like an object when as you said, it's not meant to work like that. Events are inherently object-oriented and for that reason, it has to be used with classes, not modules. Otherwise, it'd make the language even more hard to use because there's no real abstraction. I don't think everything ought to be a class, and modules does have its place but that also implies we need to not mix object-oriented ideas into modules lest it becomes a bastardized version of class module, IMO. |
The promised VB6 compatibility is very important in the Twinbasic project. |
Events can be static in C#, i.e. assoicated with the type not an instance. Surprised the heck out of me when I saw that for the first time... What .Net does have is more feature-rich value types ( |
No, this is not what I'm suggesting. The current feature requests boils down to supporting this sample code
It's such |
Would you envisage Module TestModule
Private WithEvents m_oCn As ADODB.Connection
Private Sub Disconnected(adStatus As ADODB.EventStatusEnum, ByVal pConnection As ADODB.Connection) Handles m_oCN.Disconnect
Debug.Print "Disconnected"
End Sub
End Module |
The point is that the code:
is conceptually equivalent to:
so what you are actually asking is that a module supports the What we are really asking, IMO, is an ability to create a true singleton object that can be globally accessible, which we can get there with a In the case of C#'s |
TBH, I'm not foreseeing this one getting implemented in tB. I agree, that it's something that catches beginners out, but I see that more as an opportunity to teach them why WithEvents needs a class, and a little about OOP. To make this work would require tB to implicitly create a class handler, with stubs that forward to the matching module members, and then give that class a global instance tacked on to the global variables for cleanup purposes. At the moment I'm not seeing it as worth the extra effort to implement and write the tests for. |
@bclothier This not how events are implemented in VBx (and possibly TB). Otherwise it would be impossible to compile a class with two sinks of
For each @WaynePhillipsEA Yes, there was a recent thread on vbforums.com where someone wanted to get notifications from an Ax-EXE and everyone was advising to use So to workaround lack of
When there is a There has to be more code executed than simple assignment when the variable is declared But I agree this enhancement is very low priority, probably not worth considering until 1.0 release (or even after this). |
I see what you mean. I was basing this on the fact that the subscriber object must provide an implementation of the event interface during the call to Addendum: From other discussion in tangentially related issue, Wayne cited wqweto's earlier comments which seems to support the idea that the event interface is created per-instance, rather than shared so I stand corrected on this. Thanks for the education! |
I'm not quite sure from where, but upon reading this I was like "well Duh, of course it's like that, because...(?)" ...and then I realized that I couldn't fill in the (?) part. It's in my head that modules don't support/implement the COM iDispatch() interface, maybe? Ergo, supporting Events is a non-starter as there's no class-instance pointer to hand out to callbacks...or something like that? |
Actually there is no such requirement. Classes have a default dual interface (i.e. both vtable base ISomeInterface and an IDispatch which forwards to the same ISomeInterface) and then separately for each WithEvents variable a custom IDispatch is provided (to be advised) with an Invoke method branching on event dispid and calling respective event-handler routine in parent instance. This might be implemented data-driven the way ATL does it with sink tables, sink ids, etc. or it could be codegen for each WithEvents declaration. This codegen can call regular routines in a standard module too and this would be even easier than calling methods on the parent object because currently all Object_EventName event-handlers have a hidden this/Me parameter passed additionally. Anyway, this is so very low priority that probably is not worth it even discussing the possible implementations. |
This feature is something often missing in VBx, not having it complicates things. But if this is to problematic, OK, we will stay using workarounds. About predeclared class modules:
About "this is not a good practice" and al related stuff: VBx is a mix of OOP and structured, functional language. Bas modules are not OOP but functional. Still, they can host objects but are not able to receive their events. The usual workaround is to add a class to host the object, receive the event(s) and make callbacks to the bas module from the class. Not too difficult but a bit cumbersome. |
I'm normally the sort to argue for a request like this one, but oddly I'm just not "feeling it" for this - at least not anymore. perhaps that's because I think that modules should stay "simpler" than classes. Events and callbacks are "more advanced" programming techniques, and in so far as "learning BASIC" as a new skill goes, I can see keeping the two conceptual models apart from one another to help ease the overall learning curve. |
There is nothing to keep apart here. This is a clasical case of abstraction leak. The implementation of standard modules does not allow declaring WithEvents variables which is a limitation that baffles beginner and has no good explanation. It sounds like “Long variables are not allowed in modules” — but why? Why WithEvents variables are not allowed in modules? What problem this prevents me from tripping over a beginner will ask themselves? No explanation is given but we all know that the implementation (codegen) would be non-trivial and this is the reason VBx don’t have it. |
@wqweto |
@mburns08109 I think that makes sense as a position. I do think though we should focus on deciding if this feature is good or bad, yes/no. That is generally the more subjective part which needs group discussion. Then Wayne can prioritise based on perceived value Vs difficulty of implementation. So on the one hand it sounds like you want this eventually just you assess it's low priority given cost benefit, on the other hand it sounds like you might not want the feature at all given it may steepen the learning curve by mixing advanced concepts in. So which is it? Personally I think it's a good feature - low discoverability is almost a good thing as users won't worry about it until they need it and find it just works™. One bad thing in VBA is classes are seen as extra complicated advanced concepts when really they are pretty essential to good code, I think blurring the lines here will help teach arguably more advanced concepts like events in a more familiar setting of a standard module, smoothing the learning curve. Agreed it sounds like a pain to implement but that's Wayne's decision (he is best suited to make that assessment anyway). |
@Greedquest, So, classify me as ambivalent right now - I'm undecided on whether I'd like to see this happen. I'll take up the task of reconsidering it more deeply and get back to you. |
Is your feature request related to a problem? Please describe.
TB should support
WithEvents
on private/global variables in a standard module.The current limitation is inherited from VBx and has always stumbled beginners (and intermediate) users.
Describe the solution you'd like
Cannot see a technical reason why the callback IDispatch cannot be implemented in .bas module codegen, with lifetime connected on the private/global variable that is dimensioned using
WithEvents
modifier.I can only hypothesize that in the original VBx this task was probably flagged a too low priority and remained unimplemented until project's thermal death consequently.
The text was updated successfully, but these errors were encountered: