|
A question for Steve Terry, Miran, Spellcaster and other GUI people! |
Richard Phipps
Member #1,632
November 2001
|
Hi there, Since you three have some experience of modifying the Allegro GUI I was wondering how you would go about designing a new GUI system? One that's more flexible and skinnable.. Thought I'd better get some ideas before I jump in at the deep end!
|
amarillion
Member #940
January 2001
|
Just download their GUI's and look at the source code. I haven't tried out all tree of them but I know that MASKinG comes with docs explaining the interface in great detail. Copy & paste! -- |
Richard Phipps
Member #1,632
November 2001
|
I was thinking more of listening to their ideas after they have done their libs. Most of us realise the mistakes we've made in design after we've done something! But, you do have a good point there amarillion.. |
amarillion
Member #940
January 2001
|
Try to do write a program using Java SWING. Although it is highly critisized for being very slow, I think it has a very nice, neat structure. For example, the way events are handled or the way data is separated from widgets. -- |
Richard Phipps
Member #1,632
November 2001
|
Can you explain more please? |
spellcaster
Member #1,493
September 2001
|
Ok, let's startr simple: You should use an OO approach. I'll use C++, you can of course use C aswell (but using C++ in the examples will make the examples more readable). The first think you should realize is that every widget (UI Element) should be eable to draw itself, and to deal with any events itself. This leads to the point, that sombody or something needs to tell the widgets that an event has happened. Let's call this thing the EventManager. What the EventManger does is quite simple. Every time he is called (Let's assume there is an EventMessagner::update() method, or something like this) it checks where the mouse is. It then tries to find the widget at this position (if any). These methods might create repaint events which are stored in the EventManagers queue. So, after doing it's normal stuff, it will work on this list as well. And calling the update method of the widgets in question. In order to make painting easy, every widget should only get a sub bitmap of the bitmap everything is drawn on. So, you can always assume that (0,0) is the upper left corner of the widget whichs paint() method is called. Depending on your intention, you might want to use something similar to a graphics context, to store the bitmap to work on, the current font, foreground and background colors, etc. Another key point is that your widgets should be containers. This means they should be able to contain other widgets. So, if you have a textbox, a button and a pop-up menu you should be able to code a combo box simply by combining these elements. Once you have a simple widget, and the EventManager writing the widget itself is pretty simple. In theory, your widget class could look like this:
The getWidgetAtLocation() checks if one of the child widgets is at the given position. If so, it calls the getWidgetAtLocation() of the corresponding child widget. If there's no child at this position, it returns the this pointer. So, your EventManager class can call the getWidgetAtLocation() method to get the widget under the mouse (Some complex widgets, say a combobox, might not call the child class, but simply always return this). And yes, this is of course simplified. -- |
Steve Terry
Member #1,989
March 2002
|
Well I did write some docs for my lib and went over a few of the subsystems in depth. I can safely say I've made many mistakes while making my GUI, but there is really no right or wrong way, just a matter of preference. For instance at one point I had a texture field for all of my skins so you could have one texture used for multiple things, later out of preference I changed it so that each object has a different texture name, this cut down the number of config lines in the config files, but in some instances caused skins to bloat a bit because two textures may look identical, but require different names. If you really want to know how to make the allegro GUI more flexible download my source and look at some of the things I did, also looking at the docs and the "making procedures" section should explain some of the subsystems I had to create to handle stuff. The most important subsystem I had to create so far was the memory manager, without that I would be extremely limited in how flexible I could make the GUI, with it however there is almost no limit. In fact all of the subsystems I use for my GUI are completely portable to any other GUI and are not tied into NAS GUI at all. Of course you are free to extend any part of my GUI as you wish. I'm looking for someone who is willing to extend it to allow for multiple overlapping dialogs and menus. After that all I have to do is finish off the few remaining procs and it'll be nearly complete.
Simple no? We don't even have to worry about handling the mouse as hiding it and such because that is handled by the dirty rectangle system.
This code is entirely non-blocking as well since it uses no while loops. This is because the memory manager allows me to store a little bit of extra information per procedure like the states variable. This allows me to retirive the last mouse clicked position and some other useful information like a status flag telling me that it has been clicked, or the mouse was held down outside of the procedure, etc. Because of this system I can store any type of data and retrieve it per procedure as needed.. pretty nifty right;D ___________________________________ |
Richard Phipps
Member #1,632
November 2001
|
So pretty much the same way as Allegro handles it then? Is this really how all modern GUI's handle things? EDIT: Saw Steve's post after this reply. Thanks Steve that looks very useful. I'm still not clear on why you need a memory manager though? Can you explain it to me a bit more? (sorry I'm a newbie on GUI stuff!) Cheers! |
X-G
Member #856
December 2000
|
Qt and Swing both use a signals/slots mechanism, AFAIK - someone explained this in another thread. -- |
23yrold3yrold
Member #1,134
March 2001
|
I wrote a GUI (remember that ugly horror? ). If I were going to write an Allegro GUI, I'd probably ape the crap out of Win32 since it's all I know. Are any of the Allegro GUI's at a good useable state? I keep forgetting; there's too many -- |
Steve Terry
Member #1,989
March 2002
|
The memory manager is not really needed as you could store teh extra information in any one of the free dp* fields, this however would clutter up the GUI since you'd have to create structures or something to hold multiple types of data in one pointer. This manager allows flexibility without using any dp* fields. In fact I could rewrite the DIALOG structure to not have any dp fields at all and just call helper functions to set callbacks, etc. because those procs could retirieve that information from the memory banks. It just makes the API a lot prettier ___________________________________ |
spellcaster
Member #1,493
September 2001
|
Signal / Slot is also calles Listener and/or Subscriber. Edit: Quote: So pretty much the same way as Allegro handles it then?
Um no. -- |
Steve Terry
Member #1,989
March 2002
|
My GUI lacks z-ordering but as soon as I impliment multiple dialogs that will be there. So far I've managed to abstract the graphics context and made non-blocking procedures, not so sure about containers or how I could do that... [edit] ___________________________________ |
Peter Wang
Member #23
April 2000
|
Attaching on to spellcaster's post: Note the distinction between Events and Signals. Events are from the event manager to the widgets (e.g. "repaint yourself", "the user pushed the left mouse button"). Signals are from the widget to the application (e.g. "hey, I got clicked, time for you to do something useful"). This is the terminology used by GTK. Other toolkits [should] have the same separation, if under different names. Otherwise, you end up with a mess like the Allegro GUI.
|
Andrei Ellman
Member #3,434
April 2003
|
I've noticed several people (including myself) are working on their own GUIs instead of using a pre-existing GUI. Is this because they feel more comfortable using a GUI they completely understand rather than an off-the-shelf GUI, or do they simply find Allegro's GUI to be a bit limited, or do they want to have complete control of it and customise it any way they want? I chose to write my own GUI in my own games for a mixture of these three reasons. AE. -- |
IronBob
Member #3,248
February 2003
|
not only is allegro's gui limiting. but its down-right ugly. |
Steve Terry
Member #1,989
March 2002
|
Which reminds me, an Allegro theme for my GUI would be neat for those who want some nastalgia. Hmm completely black and white buttons and dialogs... fun fun;D ___________________________________ |
X-G
Member #856
December 2000
|
You should call it GEM though, because that's what Allegro is supposed to look like, afaik. -- |
Richard Phipps
Member #1,632
November 2001
|
Well, while working on your own GUI is a lot of work I think it's something that I may have to do at some point. However I really should look at the GUI Libs people have done using Allegro and see if I can adapt them.. (Those that are open source of course). So that's why I was interested to seeing the different design approaches GUI's use. EDIT: GEM (or Graphical Enviroment Manager) was part of the GUI system used on the Atari ST. Pretty soon it looked horribly dated, but that black and white style looks quite retro and attractive recently! |
miran
Member #2,407
June 2002
|
I would just like to add that I based my GUI on a library called DeGUI which was basically a C++ wrapper for the Allegro GUI (that was in the times of Allegro 3.12). The author of DeGUI abandoned it and I picked up on it improving it and adding extra functionality (support for skins etc.) While it's still incomplete and a bit on the buggy side, it does support more or less everything spellcaster said up there except for that z sorting thingy which I don't even know what it is. Or if it is what I think it might be then I have that too to some extent. Anyway, while working on the lib I discovered it's NOT a good idea to make a GUI based on the Allegro GUI code. You will be much better off writing it completely from scratch using spellcaster's code as a starting point. If you decide to make your own GUI I suggest you study a good C++ book very carefully before you start. I was almost a complete newbie in C++ a year ago and that shows in my code. Since then I learned a lot and a while ago I was even tempted to rewrite my lib from scratch too but then I realized it would be too much work for too little benefit and I just quit. I am planning to make major improvements to the lib in the summer though (summer starts in july here)... -- |
Richard Phipps
Member #1,632
November 2001
|
Hmmm.. The problem here is that I don't know C++ (I did try to learn). I know C++ would be a better choice for doing a GUI, but for some reason I just don't like it. I have done some tinkering with the Allegro GUI routines for the interface in Stylepaint. But I find some aspects of it very restricitng. Oh, I thought that LexGui was based around simply adding skin support to the Allegro GUI. Or have I done Lenny an injustice? |
miran
Member #2,407
June 2002
|
Yeah, LexGUI is just an extension of the default Allegro GUI (as far as I can see all the widgets are "derived" from the default ones and they only "overload" the MSG_DRAW message)... -- |
gillius
Member #119
April 2000
|
I've used a lot of Java Swing and only a very brief peppering of Win32 code, but I do know that I love Java Swing. Looking at spellcaster's code that's a really good way to do it, so listen to spellcaster . Signal / slots is awesome, although I've never heard that term except on this fourm. I thought the Java Swing followed MVC (Model-View-Controller). But its event system is subscription based and I love it. I like those concepts so much that I followed them very greatly when I designed my GNE library. I think GUIs is the the best or very close to the best example case for object-oriented programming and one of the best cases for mutli-threaded programming. GUIs are very suited for a lot of really cool programming concepts that I like . Gillius |
Richard Phipps
Member #1,632
November 2001
|
Lenny, apparantly I've got to listen to you.. Using plain C how would you do a GUI system? Or would you simply modify allegro's? (That question is for everyone else too..)
|
miran
Member #2,407
June 2002
|
As I already said, I don't suggest just modifying the Allegro GUI. You'll get results sooner that way but in the long run you'll regret it. If you do want to go down that road then why don't you modify LexGUI instead? Spellcaster obviously stopped working on it a long time ago... -- |
|
|