ProtocolAdaptor.st
author claus
Wed, 10 May 1995 04:26:18 +0200
changeset 71 5b34cd877517
parent 69 225a9efd50f5
child 75 a53337dc3e19
permissions -rw-r--r--
.
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
69
claus
parents:
diff changeset
     1
'From Smalltalk/X, Version:2.10.5 on 9-may-1995 at 12:03:43 pm'!
claus
parents:
diff changeset
     2
claus
parents:
diff changeset
     3
ValueModel subclass:#ProtocolAdaptor
claus
parents:
diff changeset
     4
	 instanceVariableNames:'accessPath subject subjectChannel subjectSendsUpdates'
claus
parents:
diff changeset
     5
	 classVariableNames:''
claus
parents:
diff changeset
     6
	 poolDictionaries:''
71
claus
parents: 69
diff changeset
     7
	 category:'Interface-Support-Models'
69
claus
parents:
diff changeset
     8
!
claus
parents:
diff changeset
     9
claus
parents:
diff changeset
    10
!ProtocolAdaptor class methodsFor:'documentation'!
claus
parents:
diff changeset
    11
claus
parents:
diff changeset
    12
copyright
claus
parents:
diff changeset
    13
"
claus
parents:
diff changeset
    14
 COPYRIGHT (c) 1995 by Claus Gittinger
71
claus
parents: 69
diff changeset
    15
	      All Rights Reserved
69
claus
parents:
diff changeset
    16
claus
parents:
diff changeset
    17
 This software is furnished under a license and may be used
claus
parents:
diff changeset
    18
 only in accordance with the terms of that license and with the
claus
parents:
diff changeset
    19
 inclusion of the above copyright notice.   This software may not
claus
parents:
diff changeset
    20
 be provided or otherwise made available to, or used by, any
claus
parents:
diff changeset
    21
 other person.  No title to or ownership of the software is
claus
parents:
diff changeset
    22
 hereby transferred.
claus
parents:
diff changeset
    23
"
claus
parents:
diff changeset
    24
!
claus
parents:
diff changeset
    25
claus
parents:
diff changeset
    26
version
claus
parents:
diff changeset
    27
"
71
claus
parents: 69
diff changeset
    28
$Header: /cvs/stx/stx/libview2/ProtocolAdaptor.st,v 1.2 1995-05-10 02:25:43 claus Exp $
69
claus
parents:
diff changeset
    29
"
claus
parents:
diff changeset
    30
!
claus
parents:
diff changeset
    31
claus
parents:
diff changeset
    32
documentation
claus
parents:
diff changeset
    33
"
claus
parents:
diff changeset
    34
    a ProtocolAdaptor allows access to embeded values in a
claus
parents:
diff changeset
    35
    complex model and plays model towards the outside world.
claus
parents:
diff changeset
    36
claus
parents:
diff changeset
    37
    Consider the case where editFields are required for the
claus
parents:
diff changeset
    38
    elements (instance variables) of a more complex model.
claus
parents:
diff changeset
    39
    Using ValueHolders, you had to copy the individual
claus
parents:
diff changeset
    40
    values out-of and into multiple valueHolders.
claus
parents:
diff changeset
    41
    A protocolAdaptor makes this easier, by playing model towards
claus
parents:
diff changeset
    42
    the editField, returning a value from the complex model, 
claus
parents:
diff changeset
    43
    and forwards changes to the complex model.
claus
parents:
diff changeset
    44
claus
parents:
diff changeset
    45
    Notice: since you can specify the aspect- and changeSymbols in most ST/X
claus
parents:
diff changeset
    46
    widgets, ProtocolAdapters are not always needed (at  least, if no access-
claus
parents:
diff changeset
    47
    path is required). 
claus
parents:
diff changeset
    48
    However, if you want to apply widgets on objects which where not originally 
claus
parents:
diff changeset
    49
    designed as models (such as Arrays), ProtocolAdapters are very useful.
claus
parents:
diff changeset
    50
claus
parents:
diff changeset
    51
    Notice: this class was implemented using protocol information
claus
parents:
diff changeset
    52
    from alpha testers - it may not be complete or compatible to
claus
parents:
diff changeset
    53
    the corresponding ST-80 class. If you encounter any incompatibilities,
claus
parents:
diff changeset
    54
    please forward a note to the ST/X team.
claus
parents:
diff changeset
    55
"
claus
parents:
diff changeset
    56
!
claus
parents:
diff changeset
    57
claus
parents:
diff changeset
    58
examples 
claus
parents:
diff changeset
    59
"
claus
parents:
diff changeset
    60
	|a obj|
claus
parents:
diff changeset
    61
claus
parents:
diff changeset
    62
	a := ProtocolAdaptor accessPath:#(1 2 3).
claus
parents:
diff changeset
    63
	obj := Array with:#(11 (121 122 123) 13)
claus
parents:
diff changeset
    64
		     with:#(21 (221 222 223) 23)
claus
parents:
diff changeset
    65
		     with:#(33 (321 322 323) 33).
claus
parents:
diff changeset
    66
	a valueUsingSubject:obj  
claus
parents:
diff changeset
    67
claus
parents:
diff changeset
    68
claus
parents:
diff changeset
    69
claus
parents:
diff changeset
    70
	|a obj|
claus
parents:
diff changeset
    71
claus
parents:
diff changeset
    72
	a := ProtocolAdaptor accessPath:#(1 2 origin).
claus
parents:
diff changeset
    73
	obj := Array with:(Array with:1@1 with:(1@2 corner:100@100))
claus
parents:
diff changeset
    74
		     with:(Array with:2@1 with:2@2)
claus
parents:
diff changeset
    75
		     with:(Array with:3@1 with:3@2).
claus
parents:
diff changeset
    76
	a valueUsingSubject:obj  
claus
parents:
diff changeset
    77
claus
parents:
diff changeset
    78
claus
parents:
diff changeset
    79
claus
parents:
diff changeset
    80
	|a model|
claus
parents:
diff changeset
    81
claus
parents:
diff changeset
    82
	a := ProtocolAdaptor accessPath:#(1 2 origin).
claus
parents:
diff changeset
    83
	model := (Array with:(Array with:1@1 with:(1@2 corner:100@100))
claus
parents:
diff changeset
    84
		     with:(Array with:2@1 with:2@2)
claus
parents:
diff changeset
    85
		     with:(Array with:3@1 with:3@2)) asValue.
claus
parents:
diff changeset
    86
	a subjectChannel:model.
claus
parents:
diff changeset
    87
	a value   
claus
parents:
diff changeset
    88
"
claus
parents:
diff changeset
    89
! !
claus
parents:
diff changeset
    90
claus
parents:
diff changeset
    91
!ProtocolAdaptor class methodsFor:'instance creation'!
claus
parents:
diff changeset
    92
claus
parents:
diff changeset
    93
accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
    94
    ^ (self new) accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
    95
!
claus
parents:
diff changeset
    96
claus
parents:
diff changeset
    97
subject:anObject
claus
parents:
diff changeset
    98
    ^ (self new) subject:anObject
claus
parents:
diff changeset
    99
!
claus
parents:
diff changeset
   100
claus
parents:
diff changeset
   101
subject:anObject sendsUpdates:aBoolean
claus
parents:
diff changeset
   102
    ^ (self new) subject:anObject; sendsUpdates:aBoolean
claus
parents:
diff changeset
   103
!
claus
parents:
diff changeset
   104
claus
parents:
diff changeset
   105
subject:anObject accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
   106
    ^ (self new) subject:anObject; accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
   107
!
claus
parents:
diff changeset
   108
claus
parents:
diff changeset
   109
subject:anObject sendsUpdates:aBoolean accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
   110
    ^ (self new) subject:anObject; sendsUpdates:aBoolean; accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
   111
!
claus
parents:
diff changeset
   112
claus
parents:
diff changeset
   113
subjectChannel:aValueHolder
claus
parents:
diff changeset
   114
    ^ (self new) subjectChannel:aValueHolder
claus
parents:
diff changeset
   115
!
claus
parents:
diff changeset
   116
claus
parents:
diff changeset
   117
subjectChannel:aValueHolder sendsUpdates:aBoolean 
claus
parents:
diff changeset
   118
    ^ (self new) subjectChannel:aValueHolder; sendsUpdates:aBoolean
claus
parents:
diff changeset
   119
!
claus
parents:
diff changeset
   120
claus
parents:
diff changeset
   121
subjectChannel:aValueHolder accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
   122
    ^ (self new) subjectChannel:aValueHolder; accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
   123
!
claus
parents:
diff changeset
   124
claus
parents:
diff changeset
   125
subjectChannel:aValueHolder sendsUpdates:aBoolean accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
   126
    ^ (self new) subjectChannel:aValueHolder; sendsUpdates:aBoolean; accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
   127
! !
claus
parents:
diff changeset
   128
claus
parents:
diff changeset
   129
!ProtocolAdaptor methodsFor:'accessing'!
claus
parents:
diff changeset
   130
claus
parents:
diff changeset
   131
setValue:newValue usingSubject:anObject
claus
parents:
diff changeset
   132
    "set a value in anObject, using the selectors in accessPath.
claus
parents:
diff changeset
   133
     A helper for setValue:."
claus
parents:
diff changeset
   134
claus
parents:
diff changeset
   135
    |obj lastIndex|
claus
parents:
diff changeset
   136
claus
parents:
diff changeset
   137
    obj := anObject.
claus
parents:
diff changeset
   138
    lastIndex := accessPath size.
claus
parents:
diff changeset
   139
    accessPath keysAndValuesDo:[:idx :aSelectorOrIndex |
claus
parents:
diff changeset
   140
	aSelectorOrIndex isInteger ifTrue:[
claus
parents:
diff changeset
   141
	    idx == lastIndex ifTrue:[
claus
parents:
diff changeset
   142
		obj at:aSelectorOrIndex put:newValue
claus
parents:
diff changeset
   143
	    ] ifFalse:[
claus
parents:
diff changeset
   144
		obj := obj at:aSelectorOrIndex
claus
parents:
diff changeset
   145
	    ]
claus
parents:
diff changeset
   146
	] ifFalse:[
claus
parents:
diff changeset
   147
	    idx == lastIndex ifTrue:[
claus
parents:
diff changeset
   148
		obj perform:(aSelectorOrIndex , ':') asSymbol with:newValue
claus
parents:
diff changeset
   149
	    ] ifFalse:[
claus
parents:
diff changeset
   150
		obj := obj perform:aSelectorOrIndex
claus
parents:
diff changeset
   151
	    ]
claus
parents:
diff changeset
   152
	]
claus
parents:
diff changeset
   153
    ].
claus
parents:
diff changeset
   154
    ^ newValue
claus
parents:
diff changeset
   155
!
claus
parents:
diff changeset
   156
claus
parents:
diff changeset
   157
valueUsingSubject:anObject
claus
parents:
diff changeset
   158
    "return the value from anObject, using the selectors in accessPath.
claus
parents:
diff changeset
   159
     A helper for value."
claus
parents:
diff changeset
   160
claus
parents:
diff changeset
   161
    |obj|
claus
parents:
diff changeset
   162
claus
parents:
diff changeset
   163
    obj := anObject.
claus
parents:
diff changeset
   164
    accessPath notNil ifTrue:[
claus
parents:
diff changeset
   165
	accessPath do:[:aSelectorOrIndex |
claus
parents:
diff changeset
   166
	    aSelectorOrIndex isInteger ifTrue:[
claus
parents:
diff changeset
   167
		obj := obj at:aSelectorOrIndex
claus
parents:
diff changeset
   168
	    ] ifFalse:[
claus
parents:
diff changeset
   169
		obj := obj perform:aSelectorOrIndex
claus
parents:
diff changeset
   170
	    ]
claus
parents:
diff changeset
   171
	].
claus
parents:
diff changeset
   172
    ].
claus
parents:
diff changeset
   173
    ^ obj
claus
parents:
diff changeset
   174
!
claus
parents:
diff changeset
   175
claus
parents:
diff changeset
   176
value
claus
parents:
diff changeset
   177
    "return the value from my subject or subjectChannel."
claus
parents:
diff changeset
   178
claus
parents:
diff changeset
   179
    |obj|
claus
parents:
diff changeset
   180
claus
parents:
diff changeset
   181
    subject notNil ifTrue:[
claus
parents:
diff changeset
   182
	obj := subject.
claus
parents:
diff changeset
   183
    ] ifFalse:[
claus
parents:
diff changeset
   184
	obj := subjectChannel value
claus
parents:
diff changeset
   185
    ].
claus
parents:
diff changeset
   186
    ^ self valueUsingSubject:obj
claus
parents:
diff changeset
   187
!
claus
parents:
diff changeset
   188
claus
parents:
diff changeset
   189
setValue:newValue
claus
parents:
diff changeset
   190
    "set the value in my subject or subjectChannel."
claus
parents:
diff changeset
   191
claus
parents:
diff changeset
   192
    |obj|
claus
parents:
diff changeset
   193
claus
parents:
diff changeset
   194
    subject notNil ifTrue:[
claus
parents:
diff changeset
   195
	obj := subject.
claus
parents:
diff changeset
   196
    ] ifFalse:[
claus
parents:
diff changeset
   197
	obj := subjectChannel value
claus
parents:
diff changeset
   198
    ].
claus
parents:
diff changeset
   199
    ^ self setValue:newValue usingSubject:obj
claus
parents:
diff changeset
   200
! !
claus
parents:
diff changeset
   201
claus
parents:
diff changeset
   202
!ProtocolAdaptor methodsFor:'change notification'!
claus
parents:
diff changeset
   203
claus
parents:
diff changeset
   204
notifyChange
claus
parents:
diff changeset
   205
    subjectSendsUpdates ifFalse:[
71
claus
parents: 69
diff changeset
   206
	super notifyChange:#value
69
claus
parents:
diff changeset
   207
    ]
claus
parents:
diff changeset
   208
! !
claus
parents:
diff changeset
   209
claus
parents:
diff changeset
   210
!ProtocolAdaptor methodsFor:'initialization'!
claus
parents:
diff changeset
   211
claus
parents:
diff changeset
   212
initialize
claus
parents:
diff changeset
   213
    super initialize.
claus
parents:
diff changeset
   214
    subjectSendsUpdates := false. 
claus
parents:
diff changeset
   215
! !
claus
parents:
diff changeset
   216
claus
parents:
diff changeset
   217
!ProtocolAdaptor methodsFor:'change & update'!
claus
parents:
diff changeset
   218
claus
parents:
diff changeset
   219
update:something with:aPArameter from:changedObject
claus
parents:
diff changeset
   220
    "translate updates from my subject into value-changes towards
claus
parents:
diff changeset
   221
     my dependents. Since I have no specific aspect, every change is forwarded"
claus
parents:
diff changeset
   222
claus
parents:
diff changeset
   223
    (changedObject == subject
claus
parents:
diff changeset
   224
    or:[changedObject == subjectChannel]) ifTrue:[
claus
parents:
diff changeset
   225
	self changed:#value.
claus
parents:
diff changeset
   226
	^ self
claus
parents:
diff changeset
   227
    ].
claus
parents:
diff changeset
   228
! !
claus
parents:
diff changeset
   229
claus
parents:
diff changeset
   230
!ProtocolAdaptor methodsFor:'accessing-spec'!
claus
parents:
diff changeset
   231
claus
parents:
diff changeset
   232
subjectSendsUpdates
claus
parents:
diff changeset
   233
    "return true, if the subject sends updates itself
claus
parents:
diff changeset
   234
     If true, the receiver will not send updates on changes"
claus
parents:
diff changeset
   235
claus
parents:
diff changeset
   236
    ^ subjectSendsUpdates
claus
parents:
diff changeset
   237
!
claus
parents:
diff changeset
   238
claus
parents:
diff changeset
   239
subjectSendsUpdates:aBoolean
claus
parents:
diff changeset
   240
    "set/clear the flag which states if the subject sends updates itself.
claus
parents:
diff changeset
   241
     If true, the receiver will not send updates on changes"
claus
parents:
diff changeset
   242
claus
parents:
diff changeset
   243
    subjectSendsUpdates := aBoolean.
claus
parents:
diff changeset
   244
!
claus
parents:
diff changeset
   245
claus
parents:
diff changeset
   246
subjectChannel:aValueHolder
claus
parents:
diff changeset
   247
    |oldChannel|
claus
parents:
diff changeset
   248
claus
parents:
diff changeset
   249
    oldChannel := subjectChannel.
claus
parents:
diff changeset
   250
    subjectChannel := aValueHolder.
claus
parents:
diff changeset
   251
    oldChannel notNil ifTrue:[
claus
parents:
diff changeset
   252
	self changed:#value.
claus
parents:
diff changeset
   253
    ].
claus
parents:
diff changeset
   254
!
claus
parents:
diff changeset
   255
claus
parents:
diff changeset
   256
subjectChannel
claus
parents:
diff changeset
   257
    ^ subjectChannel
claus
parents:
diff changeset
   258
!
claus
parents:
diff changeset
   259
claus
parents:
diff changeset
   260
subject:anObject
claus
parents:
diff changeset
   261
    subject notNil ifTrue:[
claus
parents:
diff changeset
   262
	subject removeDependent:self
claus
parents:
diff changeset
   263
    ].
claus
parents:
diff changeset
   264
    subject := anObject.
claus
parents:
diff changeset
   265
    self changed:#value.
claus
parents:
diff changeset
   266
    subject notNil ifTrue:[
claus
parents:
diff changeset
   267
	subject addDependent:self
claus
parents:
diff changeset
   268
    ].
claus
parents:
diff changeset
   269
!
claus
parents:
diff changeset
   270
claus
parents:
diff changeset
   271
subject
claus
parents:
diff changeset
   272
    ^ subject
claus
parents:
diff changeset
   273
!
claus
parents:
diff changeset
   274
claus
parents:
diff changeset
   275
accessPath:aCollectionOfSelectors
claus
parents:
diff changeset
   276
    accessPath := aCollectionOfSelectors
claus
parents:
diff changeset
   277
!
claus
parents:
diff changeset
   278
claus
parents:
diff changeset
   279
accessPath
claus
parents:
diff changeset
   280
    ^ accessPath
claus
parents:
diff changeset
   281
! !
claus
parents:
diff changeset
   282