You would like to deal with many widgets as though they were a single widget.
#lang scheme/gui
(define my-widget%
(class vertical-panel%
(init-field parent)
(super-new [parent parent])
(new button%
[label "Press Me"]
[parent this] [callback void])
(new text-field%
[label "Enter Text"]
[parent this] [callback void])))
(define top-frame (instantiate frame% () (label "Example")))
(define new-my-widget (new my-widget% (parent top-frame)))
(send top-frame show #t)
Addenda
this can be done as a macro;
#lang scheme/gui
(define-syntax define-aggregate-widget
(syntax-rules ()
((_ new-aggregate-widget-classname container . sub-widget-args)
(define new-aggregate-widget-classname
(class container
(init-field parent)
(super-new (parent parent))
.
sub-widget-args )))))
(define-syntax new-widget
(syntax-rules ()
((_ new-widget-classname . args)
(new new-widget-classname (parent this) . args))))
(define-aggregate-widget my3% horizontal-panel%
(new-widget button% (label "Press Me") (callback void) (stretchable-width #t))
(new-widget button% (label "Press Me1") (callback void) (stretchable-width #t))
(new-widget button% (label "Press Me2") (callback void) (stretchable-width #t))
(new-widget button% (label "Press Me3") (callback void) (stretchable-width #t))
(new-widget button% (label "Press Me4") (callback void) (stretchable-width #t))
(new-widget text-field% (label "Enter Text") (callback void) (stretchable-width #t)))
(define top-frame (instantiate frame% () (label "Example")))
(define new-my3 (new my3% (parent top-frame)))
(send top-frame show #t)
--
StephenDeGabrielle - 17 Jul 2008
GUI programming is non-standard. A PLT Solution has been given.
When designing a GUI, it is common to design widget groups that are reused. This is done by creating a subclass of the panel type and defining the widgets in that class.
The above sample displays a push button with the label "Push Me" above a Text Box with the label "Enter Text".
The first thing to do is start scheme with the graphics toolkit enabled:
(this is equivalent to importing the required libraries)
Then, we would like to define a new widget class and bind it to a name. It is customary to name classes with a
% suffix. This will be a subclass of the vertical-panel% widget which will provide layout geometry for child widgets.
(define my-widget%
(class vertical-panel%
))
Since panel widgets cannot be top level windows (
frames in the
MrEd and wxWidget nomenclature), we must give this widget a parent that must be set on instantiation. This is done with the init-field expression:
If we wanted, we could set it to an initial value much the same way we set initial values in a
let expression.
Next instantiate the vertical-panel[Initialize the superclass]. As we mentioned before, panels cannot be top level windows, so they must have parents. That means we need to pass the parent parameter to the super-instantiate expression:
(super-new [parent parent])
Don't worry that the parent expression has the same name as the parameter. The values are resolved by position in the expression as well as name, so Scheme won't be confused.
The next step is to instantiate the component widgets. This is quite straight forward:
(new button%
[label "Press Me"]
[parent this] [callback void])
(new text-field%
[label "Enter Text"]
[parent this] [callback void])
The keyword ''this'' allows us to place the component widgets (button and text-field) within the vertical-panel.
Constructing and showing the window are straight forward:
(define top-frame (instantiate frame% () (label "Example")))
(define new-my-widget (new my-widget% (parent top-frame)))
(send top-frame show #t)
Refactored to include changes suggested by comments and updated for PLT-Scheme 4+. (please see the history)
Thanks to
DanielSilva and
EwanHiggs
--
StephenDeGabrielle - 16 Jul 2008
Here is another example - but but the widget is created from drawing toolkit elements, rather than existing widgets.
(define scatter-chart% (class pane% ()
(init-field parent)
(super-instantiate () (parent parent) [min-width 100] [stretchable-width #f]
(min-height 200) [stretchable-height #f])
(define list-of-points '()) (define canvas (new canvas% [parent this]
(paint-callback (lambda (canvas dc) (draw-chart)))
[min-width 100] [stretchable-width #f]
[min-height 200] [stretchable-height #f]
))
(define dc (send canvas get-dc)) (define no-pen (make-object pen% "BLACK" 1 'transparent))
(define blue-brush (make-object brush% "BLUE" 'solid))
(define yellow-brush (make-object brush% "YELLOW" 'solid))
(define/public (add-point new-point)
(set! list-of-points (cons new-point list-of-points)) (send dc draw-rectangle 0 new-point 200 1)) (define/public (draw-chart)
(send dc set-pen no-pen)
(send dc set-brush blue-brush)
(send dc draw-rectangle 0 0 100 200) (send dc set-brush yellow-brush)
(map
(lambda (y) (send dc draw-rectangle 0 y 200 1)) list-of-points ))
(draw-chart)))
(define frame (new frame% [label "scatter-chart% Example"] [width 100] [height 200]))
(send frame show #t)
(define new-my-widget (new scatter-chart% (parent frame) )) (define new-his-widget (new scatter-chart% (parent frame) ))
(send new-my-widget draw-chart)
(define (go)
(map (lambda (y)
(send new-my-widget add-point y) (sleep/yield 0.05)) '(10 30 22 77 78 79 17 65 97 54 55 56 57 91 33 55 95 99 125 155 157 159 188 187 186 191)))
(go)
--
StephenDeGabrielle - 21 Sep 2007
--
EwanHiggs - 01 Sep 2004