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