RButtGrp.st
author claus
Sun, 23 Jul 1995 05:03:13 +0200
changeset 133 e58c7c979f33
parent 131 208fa92f434d
child 155 d6f3836d2b51
permissions -rw-r--r--
.

"
 COPYRIGHT (c) 1991 by Claus Gittinger
	      All Rights Reserved

 This software is furnished under a license and may be used
 only in accordance with the terms of that license and with the
 inclusion of the above copyright notice.   This software may not
 be provided or otherwise made available to, or used by, any
 other person.  No title to or ownership of the software is
 hereby transferred.
"

'From Smalltalk/X, Version:2.10.5 on 14-mar-1995 at 11:06:48 am'!

OrderedCollection subclass:#RadioButtonGroup
	 instanceVariableNames:'valueChannel'
	 classVariableNames:''
	 poolDictionaries:''
	 category:'Views-Support'
!

RadioButtonGroup comment:'
COPYRIGHT (c) 1991 by Claus Gittinger
	      All Rights Reserved

$Header: /cvs/stx/stx/libwidg/Attic/RButtGrp.st,v 1.12 1995-06-27 02:23:53 claus Exp $
'!

!RadioButtonGroup class methodsFor:'documentation '!

copyright
"
 COPYRIGHT (c) 1991 by Claus Gittinger
	      All Rights Reserved

 This software is furnished under a license and may be used
 only in accordance with the terms of that license and with the
 inclusion of the above copyright notice.   This software may not
 be provided or otherwise made available to, or used by, any
 other person.  No title to or ownership of the software is
 hereby transferred.
"
!

version
"
$Header: /cvs/stx/stx/libwidg/Attic/RButtGrp.st,v 1.12 1995-06-27 02:23:53 claus Exp $
"
!

documentation
"
    RadioButtonGroups control the interaction between RadioButtons
    turning off other button(s) when one of the group is pressed.
    To group some buttons (and have one-on behavior) use:

	|g|

	g := RadioButtonGroup new.
	...
	b1 := RadioButton label:....
	g add:b1
	...
	b2 := RadioButton label:....
	g add:b2
	...

    A radioButtonGroup installs itself as the model of its toggles or
    radioButtons. Therefore, you can no longer operate the individual
    elements on models.
    Instead, the group now provides a valueHolder which takes as value 
    the index of the selected toggle. You can set/get this valueHolder
    via the valueChannel:/valueChannel messages.
    See the last example for how this is done.
"
!

examples 
"
    example (using Toggles for 'at most one-on behavior'):

	|top panel toggle b g|

	top := StandardSystemView new.
	panel := HorizontalPanelView origin:0.0@0.0 corner:1.0@1.0 in:top.
	g := RadioButtonGroup new.
	b := Toggle label:'one' in:panel.
	g add:b.
	b := Toggle label:'two' in:panel.
	g add:b.
	b := Toggle label:'three' in:panel.
	g add:b.
	top extent:(panel preferredExtent).
	top open.

    example (using RadioButtons for 'one-on behavior'):

	|top panel toggle b g|

	top := StandardSystemView new.
	panel := HorizontalPanelView origin:0.0@0.0 corner:1.0@1.0 in:top.
	g := RadioButtonGroup new.
	b := RadioButton label:'one' in:panel.
	g add:b.
	b := RadioButton label:'two' in:panel.
	g add:b.
	b := RadioButton label:'three' in:panel.
	g add:b.
	top extent:(panel preferredExtent).
	top open.

    example (same, with button 'two' initially on):

	|top panel toggle b g|

	top := StandardSystemView new.
	panel := HorizontalPanelView origin:0.0@0.0 corner:1.0@1.0 in:top.
	g := RadioButtonGroup new.
	b := RadioButton label:'one' in:panel.
	g add:b.
	b := RadioButton label:'two' in:panel.
	g add:b.
	b turnOn.
	b := RadioButton label:'three' in:panel.
	g add:b.
	top extent:(panel preferredExtent).
	top open.

    example (operating on a model)

	|selected dialog group|

	selected := 2 asValue.
	dialog := Dialog new.
	group := RadioButtonGroup new.
	group add:(dialog addCheckBox:'one' on:nil) toggleView.
	group add:(dialog addCheckBox:'two' on:nil) toggleView.
	group add:(dialog addCheckBox:'three' on:nil) toggleView.
	group valueChannel:selected.
	dialog addAbortButton; addOkButton.
	dialog open.
	dialog accepted ifTrue:[
	    Transcript showCr:'you selected: ' , selected value printString
	]

"
! !

!RadioButtonGroup methodsFor:'accessing-mvc'!

valueChannel
    ^ valueChannel
!

valueChannel:aValueHolder
    |oldValue newValue|

    valueChannel notNil ifTrue:[oldValue := valueChannel value].
    valueChannel := aValueHolder.
    (newValue := valueChannel value) ~~ oldValue ifTrue:[
	(self at:newValue) turnOn
    ]
! !

!RadioButtonGroup methodsFor:'adding / removing'!

add:aRadioButton
    super add:aRadioButton.

    "
     this would work, if dependencies were not weak ...
    "
"/    aRadioButton addDependent:self.


    "
     therefore, we have to install ourself as model of the buttons/toggles
     creating a valueChannel, if there is none yet.
    "
    valueChannel isNil ifTrue:[
	valueChannel := 0 asValue
    ].
    aRadioButton isOn ifTrue:[valueChannel value:self size].
    aRadioButton model:self; aspectMessage:nil; changeMessage:#elementChanged:from:.

! !

!RadioButtonGroup methodsFor:'update'!

update:something with:someArgument from:changedButton
    (self includes:changedButton) ifFalse:[^ self].

    "
     a RadioButton in this group has changed - notify the others
    "

"/    "in case we have a toggle in the group, 
"/     and it has been turned off - turn it on again
"/    "
"/    changedButton isOn ifFalse:[
"/        changedButton toggleNoAction.
"/        ^ self
"/    ].
    self elementChanged:(changedButton isOn) from:changedButton
!

elementChanged:aToggle
    "historic compatibility leftover - will vanish"

    ^ self elementChanged:true from:aToggle
!

elementChanged:value from:aToggle
    "some element changed its value - find it, turn the others off
     and update my value"

    |newValue|

    self keysAndValuesDo:[:index :aButton |
	(aButton == aToggle) ifFalse:[
	    aButton isOn ifTrue:[
		aButton turnOff
	    ]
	] ifTrue:[
	    value ifTrue:[newValue := index]   
	]
    ].
    valueChannel value:newValue

! !