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