12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126 |
- Smalltalk current createPackage: 'Kernel-Collections'!
- Object subclass: #Association
- instanceVariableNames: 'key value'
- package: 'Kernel-Collections'!
- !Association commentStamp!
- I represent a pair of associated objects, a key and a value. My instances can serve as entries in a dictionary.
- Instances can be created with the class-side method `#key:value:`!
- !Association methodsFor: 'accessing'!
- key
- ^ key
- !
- key: aKey
- key := aKey
- !
- value
- ^ value
- !
- value: aValue
- value := aValue
- ! !
- !Association methodsFor: 'comparing'!
- = anAssociation
- ^ self class = anAssociation class and: [
- self key = anAssociation key and: [
- self value = anAssociation value ]]
- ! !
- !Association methodsFor: 'printing'!
- printOn: aStream
- self key printOn: aStream.
- aStream nextPutAll: ' -> '.
- self value printOn: aStream
- ! !
- !Association class methodsFor: 'instance creation'!
- key: aKey value: aValue
- ^ self new
- key: aKey;
- value: aValue;
- yourself
- ! !
- Object subclass: #Collection
- instanceVariableNames: ''
- package: 'Kernel-Collections'!
- !Collection commentStamp!
- I am the abstract superclass of all classes that represent a group of elements.
- I provide a set of useful methods to the Collection hierarchy such as enumerating and converting methods.!
- !Collection methodsFor: 'accessing'!
- occurrencesOf: anObject
- "Answer how many of the receiver's elements are equal to anObject."
- | tally |
- tally := 0.
- self do: [ :each | anObject = each ifTrue: [ tally := tally + 1 ]].
- ^ tally
- !
- size
- self subclassResponsibility
- ! !
- !Collection methodsFor: 'adding/removing'!
- add: anObject
- self subclassResponsibility
- !
- addAll: aCollection
- aCollection do: [ :each |
- self add: each ].
- ^ aCollection
- !
- anyOne
- "Answer a representative sample of the receiver. This method can
- be helpful when needing to preinfer the nature of the contents of
- semi-homogeneous collections."
- self ifEmpty: [ self error: 'Collection is empty' ].
- self do: [ :each | ^ each ]
- !
- remove: anObject
- ^ self remove: anObject ifAbsent: [ self errorNotFound ]
- !
- remove: anObject ifAbsent: aBlock
- self subclassResponsibility
- !
- removeAll
- self subclassResponsibility
- ! !
- !Collection methodsFor: 'converting'!
- asArray
- ^ Array withAll: self
- !
- asJSON
- ^ self asArray collect: [ :each | each asJSON ]
- !
- asOrderedCollection
- ^ self asArray
- !
- asSet
- ^ Set withAll: self
- ! !
- !Collection methodsFor: 'copying'!
- , aCollection
- ^ self copy
- addAll: aCollection;
- yourself
- !
- copyWith: anObject
- ^ self copy add: anObject; yourself
- !
- copyWithAll: aCollection
- ^ self copy addAll: aCollection; yourself
- !
- copyWithoutAll: aCollection
- "Answer a copy of the receiver that does not contain any elements
- equal to those in aCollection."
- ^ self reject: [ :each | aCollection includes: each ]
- ! !
- !Collection methodsFor: 'enumerating'!
- allSatisfy: aBlock
- "Evaluate aBlock with the elements of the receiver.
- If aBlock returns false for any element return false.
- Otherwise return true."
- self do: [ :each | (aBlock value: each) ifFalse: [ ^ false ] ].
- ^ true
- !
- anySatisfy: aBlock
- "Evaluate aBlock with the elements of the receiver.
- If aBlock returns true for any element return true.
- Otherwise return false."
- self do: [ :each | (aBlock value: each) ifTrue: [ ^ true ] ].
- ^ false
- !
- collect: aBlock
- | stream |
- stream := self class new writeStream.
- self do: [ :each |
- stream nextPut: (aBlock value: each) ].
- ^ stream contents
- !
- detect: aBlock
- ^ self detect: aBlock ifNone: [ self errorNotFound ]
- !
- detect: aBlock ifNone: anotherBlock
- self subclassResponsibility
- !
- do: aBlock
- self subclassResponsibility
- !
- do: aBlock separatedBy: anotherBlock
- | actionBeforeElement |
- actionBeforeElement := [ actionBeforeElement := anotherBlock ].
- self do: [ :each |
- actionBeforeElement value.
- aBlock value: each ]
- !
- inject: anObject into: aBlock
- | result |
- result := anObject.
- self do: [ :each |
- result := aBlock value: result value: each ].
- ^ result
- !
- intersection: aCollection
- "Answer the set theoretic intersection of two collections."
- | set outputSet |
-
- set := self asSet.
- outputSet := Set new.
-
- aCollection do: [ :each |
- ((set includes: each) and: [ (outputSet includes: each) not ])
- ifTrue: [
- outputSet add: each ]].
-
- ^ self class withAll: outputSet asArray
- !
- noneSatisfy: aBlock
- "Evaluate aBlock with the elements of the receiver.
- If aBlock returns false for all elements return true.
- Otherwise return false"
- self do: [ :item | (aBlock value: item) ifTrue: [ ^ false ] ].
- ^ true
- !
- reject: aBlock
- ^ self select: [ :each | (aBlock value: each) = false ]
- !
- select: aBlock
- | stream |
- stream := self class new writeStream.
- self do: [ :each |
- (aBlock value: each) ifTrue: [
- stream nextPut: each ] ].
- ^ stream contents
- !
- select: selectBlock thenCollect: collectBlock
- | stream |
- stream := self class new writeStream.
- self do: [ :each |
- (selectBlock value: each) ifTrue: [
- stream nextPut: (collectBlock value: each) ] ].
- ^ stream contents
- ! !
- !Collection methodsFor: 'error handling'!
- errorNotFound
- self error: 'Object is not in the collection'
- ! !
- !Collection methodsFor: 'streaming'!
- putOn: aStream
- self do: [ :each | each putOn: aStream ]
- ! !
- !Collection methodsFor: 'testing'!
- contains: aBlock
- self deprecatedAPI.
- ^ self anySatisfy: aBlock
- !
- ifEmpty: aBlock
- "Evaluate the given block with the receiver as argument, answering its value if the receiver is empty, otherwise answer the receiver.
- Note that the fact that this method returns its argument in case the receiver is not empty allows one to write expressions like the following ones:
- self classifyMethodAs:
- (myProtocol ifEmpty: ['As yet unclassified'])"
- ^ self isEmpty
- ifTrue: aBlock
- ifFalse: [ self ]
- !
- ifEmpty: aBlock ifNotEmpty: anotherBlock
- ^ self isEmpty
- ifTrue: aBlock
- ifFalse: anotherBlock
- !
- ifNotEmpty: aBlock
- ^ self notEmpty
- ifTrue: aBlock
- ifFalse: [ self ]
- !
- ifNotEmpty: aBlock ifEmpty: anotherBlock
- ^ self notEmpty
- ifTrue: aBlock
- ifFalse: anotherBlock
- !
- includes: anObject
- ^ self anySatisfy: [ :each | each = anObject ]
- !
- isEmpty
- ^ self size = 0
- !
- notEmpty
- ^ self isEmpty not
- ! !
- !Collection class methodsFor: 'helios'!
- heliosClass
- ^ 'collection'
- ! !
- !Collection class methodsFor: 'instance creation'!
- new: anInteger
- ^ self new
- !
- with: anObject
- ^ self new
- add: anObject;
- yourself
- !
- with: anObject with: anotherObject
- ^ self new
- add: anObject;
- add: anotherObject;
- yourself
- !
- with: firstObject with: secondObject with: thirdObject
- ^ self new
- add: firstObject;
- add: secondObject;
- add: thirdObject;
- yourself
- !
- withAll: aCollection
- ^ self new
- addAll: aCollection;
- yourself
- ! !
- Collection subclass: #IndexableCollection
- instanceVariableNames: ''
- package: 'Kernel-Collections'!
- !IndexableCollection commentStamp!
- I am a key-value store collection, that is,
- I store values under indexes.
- As a rule of thumb, if a collection has `#at:` and `#at:put:`,
- it is an IndexableCollection.!
- !IndexableCollection methodsFor: 'accessing'!
- at: anIndex
- "Lookup the given index in the receiver.
- If it is present, answer the value stored at anIndex.
- Otherwise, raise an error."
- ^ self at: anIndex ifAbsent: [ self errorNotFound ]
- !
- at: anIndex ifAbsent: aBlock
- "Lookup the given index in the receiver.
- If it is present, answer the value stored at anIndex.
- Otherwise, answer the value of aBlock."
- self subclassResponsibility
- !
- at: aKey ifAbsentPut: aBlock
- ^ self at: aKey ifAbsent: [
- self at: aKey put: aBlock value ]
- !
- at: anIndex ifPresent: aBlock
- "Lookup the given index in the receiver.
- If it is present, answer the value of evaluating aBlock with the value stored at anIndex.
- Otherwise, answer nil."
- ^ self at: anIndex ifPresent: aBlock ifAbsent: [ nil ]
- !
- at: anIndex ifPresent: aBlock ifAbsent: anotherBlock
- "Lookup the given index in the receiver.
- If it is present, answer the value of evaluating aBlock with the value stored at anIndex.
- Otherwise, answer the value of anotherBlock."
- self subclassResponsibility
- !
- at: anIndex put: anObject
- "Store anObject under the given index in the receiver."
- self subclassResponsibility
- !
- indexOf: anObject
- "Lookup index at which anObject is stored in the receiver.
- If not present, raise an error."
- ^ self indexOf: anObject ifAbsent: [ self errorNotFound ]
- !
- indexOf: anObject ifAbsent: aBlock
- "Lookup index at which anObject is stored in the receiver.
- If not present, return value of executing aBlock."
- self subclassResponsibility
- ! !
- !IndexableCollection methodsFor: 'enumarating'!
- with: anotherCollection do: aBlock
- "Calls aBlock with every value from self
- and with indetically-indexed value from anotherCollection"
- self withIndexDo: [ :each :index |
- aBlock value: each value: (anotherCollection at: index) ]
- !
- withIndexDo: aBlock
- "Calls aBlock with every value from self
- and with its index as the second argument"
- self subclassResponsibility
- ! !
- IndexableCollection subclass: #AssociativeCollection
- instanceVariableNames: ''
- package: 'Kernel-Collections'!
- !AssociativeCollection commentStamp!
- I am a base class for object-indexed collections (Dictionary et.al.).!
- !AssociativeCollection methodsFor: 'accessing'!
- associations
- | associations |
- associations := #().
- self associationsDo: [ :each | associations add: each ].
- ^ associations
- !
- at: aKey ifPresent: aBlock ifAbsent: anotherBlock
- "Lookup the given key in the receiver.
- If it is present, answer the value of evaluating the oneArgBlock with the value associated with the key,
- otherwise answer the value of absentBlock."
- ^ (self includesKey: aKey)
- ifTrue: [ aBlock value: (self at: aKey) ]
- ifFalse: [ anotherBlock value ]
- !
- indexOf: anObject ifAbsent: aBlock
- ^ self keys detect: [ :each | (self at: each) = anObject ] ifNone: aBlock
- !
- keyAtValue: anObject
- ^ self keyAtValue: anObject ifAbsent: [ self errorNotFound ]
- !
- keyAtValue: anObject ifAbsent: aBlock
- ^ self indexOf: anObject ifAbsent: aBlock
- !
- keys
- self subclassResponsibility
- !
- size
- ^ self keys size
- !
- values
- self subclassResponsibility
- ! !
- !AssociativeCollection methodsFor: 'adding/removing'!
- add: anAssociation
- self at: anAssociation key put: anAssociation value
- !
- addAll: anAssocitativeCollection
- super addAll: anAssocitativeCollection associations.
- ^ anAssocitativeCollection
- !
- remove: aKey ifAbsent: aBlock
- ^ self removeKey: aKey ifAbsent: aBlock
- !
- removeAll
- ^ self keys do: [ :each | self removeKey: each ]
- !
- removeKey: aKey
- ^ self remove: aKey
- !
- removeKey: aKey ifAbsent: aBlock
- self subclassResponsibility
- ! !
- !AssociativeCollection methodsFor: 'comparing'!
- = anAssocitativeCollection
- self class = anAssocitativeCollection class ifFalse: [ ^ false ].
- self size = anAssocitativeCollection size ifFalse: [ ^ false ].
- ^ self associations = anAssocitativeCollection associations
- ! !
- !AssociativeCollection methodsFor: 'converting'!
- asDictionary
- ^ Dictionary from: self associations
- !
- asHashedCollection
- ^ HashedCollection from: self associations
- !
- asJSON
- | c |
- c := HashedCollection new.
- self keysAndValuesDo: [ :key :value |
- c at: key put: value asJSON ].
- ^ c
- ! !
- !AssociativeCollection methodsFor: 'copying'!
- deepCopy
- | copy |
- copy := self class new.
- self keysAndValuesDo: [ :key :value |
- copy at: key put: value deepCopy ].
- ^ copy
- !
- shallowCopy
- | copy |
- copy := self class new.
- self keysAndValuesDo: [ :key :value |
- copy at: key put: value ].
- ^ copy
- ! !
- !AssociativeCollection methodsFor: 'enumerating'!
- associationsDo: aBlock
- self keysAndValuesDo: [ :key :value |
- aBlock value: (Association key: key value: value) ]
- !
- collect: aBlock
- | newDict |
- newDict := self class new.
- self keysAndValuesDo: [ :key :value |
- newDict at: key put: (aBlock value: value) ].
- ^ newDict
- !
- detect: aBlock ifNone: anotherBlock
- ^ self values detect: aBlock ifNone: anotherBlock
- !
- do: aBlock
- self valuesDo: aBlock
- !
- includes: anObject
- ^ self values includes: anObject
- !
- keysAndValuesDo: aBlock
- self keysDo: [ :each |
- aBlock value: each value: (self at: each) ]
- !
- keysDo: aBlock
- self subclassResponsibility
- !
- select: aBlock
- | newDict |
- newDict := self class new.
- self keysAndValuesDo: [ :key :value |
- (aBlock value: value) ifTrue: [ newDict at: key put: value ]].
- ^ newDict
- !
- valuesDo: aBlock
- self subclassResponsibility
- !
- withIndexDo: aBlock
- self keysAndValuesDo: [ :key :value | aBlock value: value value: key ]
- ! !
- !AssociativeCollection methodsFor: 'printing'!
- printOn: aStream
- super printOn: aStream.
-
- aStream nextPutAll: ' ('.
- self associations
- do: [ :each | each printOn: aStream ]
- separatedBy: [ aStream nextPutAll: ' , ' ].
- aStream nextPutAll: ')'
- ! !
- !AssociativeCollection methodsFor: 'testing'!
- includesKey: aKey
- self subclassResponsibility
- ! !
- !AssociativeCollection class methodsFor: 'instance creation'!
- from: aCollection
- | newCollection |
- newCollection := self new.
- aCollection do: [ :each | newCollection add: each ].
- ^ newCollection
- !
- fromPairs: aCollection
- "This message is poorly named and has been replaced by #from:"
- ^ self from: aCollection
- !
- newFromPairs: aCollection
- "Accept an array of elements where every two elements form an
- association - the odd element being the key, and the even element the value."
-
- | newCollection |
-
- aCollection size even ifFalse: [
- self error: '#newFromPairs only accepts arrays of an even length' ].
-
- newCollection := self new.
- ( 1 to: aCollection size by: 2 ) do: [ :each |
- newCollection at: (aCollection at: each) put: (aCollection at: each + 1) ].
-
- ^ newCollection
- ! !
- AssociativeCollection subclass: #Dictionary
- instanceVariableNames: 'keys values'
- package: 'Kernel-Collections'!
- !Dictionary commentStamp!
- I represent a set of elements that can be viewed from one of two perspectives: a set of associations,
- or a container of values that are externally named where the name can be any object that responds to `=`.
- The external name is referred to as the key.!
- !Dictionary methodsFor: 'accessing'!
- at: aKey ifAbsent: aBlock
- <
- var index = self._positionOfKey_(aKey);
- return index >>=0 ? self['@values'][index] : aBlock._value();
- >
- !
- at: aKey put: aValue
- <
- var index = self._positionOfKey_(aKey);
- if(index === -1) {
- var keys = self['@keys'];
- index = keys.length;
- keys.push(aKey);
- }
- return self['@values'][index] = aValue;
- >
- !
- indexOf: anObject ifAbsent: aBlock
- | index |
- index := values
- indexOf: anObject
- ifAbsent: [ 0 ].
- ^ index = 0
- ifTrue: [ aBlock value ]
- ifFalse: [ keys at: index ]
- !
- keys
- ^ keys copy
- !
- values
- ^ values
- ! !
- !Dictionary methodsFor: 'adding/removing'!
- removeAll
- keys removeAll.
- values removeAll
- !
- removeKey: aKey ifAbsent: aBlock
- <
- var index = self._positionOfKey_(aKey);
- if(index === -1) {
- return aBlock._value()
- } else {
- var keys = self['@keys'], values = self['@values'];
- var value = values[index], l = keys.length;
- keys[index] = keys[l-1];
- keys.pop();
- values[index] = values[l-1];
- values.pop();
- return value;
- }
- >
- ! !
- !Dictionary methodsFor: 'enumerating'!
- keysAndValuesDo: aBlock
- ^ keys with: values do: aBlock
- !
- keysDo: aBlock
- ^ keys do: aBlock
- !
- valuesDo: aBlock
- ^ values do: aBlock
- ! !
- !Dictionary methodsFor: 'initialization'!
- initialize
- super initialize.
- keys := #().
- values := #()
- ! !
- !Dictionary methodsFor: 'private'!
- positionOfKey: anObject
- <
- var keys = self['@keys'];
- for(var i=0;i<keys.length;i++){
- if(keys[i].__eq(anObject)) { return i;}
- }
- return -1;
- >
- ! !
- !Dictionary methodsFor: 'testing'!
- includesKey: aKey
- < return self._positionOfKey_(aKey) >>= 0; >
- ! !
- AssociativeCollection subclass: #HashedCollection
- instanceVariableNames: ''
- package: 'Kernel-Collections'!
- !HashedCollection commentStamp!
- I am a traditional JavaScript object, or a Smalltalk `Dictionary`.
- Unlike a `Dictionary`, I can only have strings as keys.!
- !HashedCollection methodsFor: 'accessing'!
- at: aKey ifAbsent: aBlock
- ^ (self includesKey: aKey)
- ifTrue: [ self basicAt: aKey ]
- ifFalse: [ aBlock value ]
- !
- at: aKey put: aValue
- ^ self basicAt: aKey put: aValue
- !
- keys
- <return Object.keys(self)>
- !
- values
- <
- return self._keys().map(function(key){
- return self._at_(key);
- });
- >
- ! !
- !HashedCollection methodsFor: 'adding/removing'!
- removeKey: aKey ifAbsent: aBlock
- ^ self
- at: aKey
- ifPresent: [ :removed | self basicDelete: aKey. removed ]
- ifAbsent: [ aBlock value ]
- ! !
- !HashedCollection methodsFor: 'enumerating'!
- keysDo: aBlock
- self keys do: aBlock
- !
- valuesDo: aBlock
- self values do: aBlock
- ! !
- !HashedCollection methodsFor: 'testing'!
- includesKey: aKey
- <return self.hasOwnProperty(aKey)>
- ! !
- IndexableCollection subclass: #SequenceableCollection
- instanceVariableNames: ''
- package: 'Kernel-Collections'!
- !SequenceableCollection commentStamp!
- I am an IndexableCollection
- with numeric indexes starting with 1.!
- !SequenceableCollection methodsFor: 'accessing'!
- allButFirst
- ^ self copyFrom: 2 to: self size
- !
- allButLast
- ^ self copyFrom: 1 to: self size - 1
- !
- atRandom
- ^ self at: self size atRandom
- !
- first
- ^ self at: 1
- !
- first: aNumber
- "Answer the first `aNumber` elements of the receiver.
- Raise an error if there are not enough elements in the receiver."
- self size < aNumber ifTrue: [ self error: 'Invalid number of elements' ].
- ^ self copyFrom: 1 to: aNumber
- !
- fourth
- ^ self at: 4
- !
- indexOf: anObject ifAbsent: aBlock
- <
- self = self._numericallyIndexable();
- for(var i=0; i < self.length; i++) {
- if(_st(self[i]).__eq(anObject)) {return i+1}
- };
- return aBlock._value();
- >
- !
- indexOf: anObject startingAt: start
- "Answer the index of the first occurence of anElement after start
- within the receiver. If the receiver does not contain anElement,
- answer 0."
- ^ self indexOf: anObject startingAt: start ifAbsent: [ 0 ]
- !
- indexOf: anObject startingAt: start ifAbsent: aBlock
- <
- self = self._numericallyIndexable();
- for(var i=start - 1; i < self.length; i++){
- if(_st(self[i]).__eq(anObject)) {return i+1}
- }
- return aBlock._value();
- >
- !
- last
- ^ self at: self size
- !
- last: aNumber
- "Answer the last aNumber elements of the receiver.
- Raise an error if there are not enough elements in the receiver."
- self size < aNumber ifTrue: [ self error: 'Invalid number of elements' ].
- ^ self copyFrom: self size - aNumber + 1 to: self size
- !
- second
- ^ self at: 2
- !
- third
- ^ self at: 3
- ! !
- !SequenceableCollection methodsFor: 'adding/removing'!
- addLast: anObject
- self add: anObject
- !
- removeLast
- ^ self remove: self last
- ! !
- !SequenceableCollection methodsFor: 'comparing'!
- = aCollection
- (self class = aCollection class and: [
- self size = aCollection size ]) ifFalse: [ ^ false ].
- self withIndexDo: [ :each :i |
- (aCollection at: i) = each ifFalse: [ ^ false ]].
- ^ true
- ! !
- !SequenceableCollection methodsFor: 'converting'!
- reversed
- self subclassResponsibility
- ! !
- !SequenceableCollection methodsFor: 'copying'!
- copyFrom: anIndex to: anotherIndex
- | range newCollection |
- range := anIndex to: anotherIndex.
- newCollection := self class new: range size.
- range withIndexDo: [ :each :i |
- newCollection at: i put: (self at: each) ].
- ^ newCollection
- !
- deepCopy
- | newCollection |
- newCollection := self class new: self size.
- self withIndexDo: [ :each :index |
- newCollection at: index put: each deepCopy ].
- ^ newCollection
- !
- shallowCopy
- | newCollection |
- newCollection := self class new: self size.
- self withIndexDo: [ :each :index |
- newCollection at: index put: each ].
- ^ newCollection
- ! !
- !SequenceableCollection methodsFor: 'enumerating'!
- detect: aBlock ifNone: anotherBlock
- <
- self = self._numericallyIndexable();
- for(var i = 0; i < self.length; i++)
- if(aBlock._value_(self[i]))
- return self[i];
- return anotherBlock._value();
- >
- !
- do: aBlock
- <
- self = self._numericallyIndexable();
- for(var i=0; i < self.length; i++) {
- aBlock._value_(self[i]);
- }
- >
- !
- with: anotherCollection do: aBlock
- <
- self = self._numericallyIndexable();
- anotherCollection = anotherCollection._numericallyIndexable();
- for(var i=0; i<self.length; i++) {
- aBlock._value_value_(self[i], anotherCollection[i]);
- }
- >
- !
- withIndexDo: aBlock
- <
- self = self._numericallyIndexable();
- for(var i=0; i < self.length; i++) {
- aBlock._value_value_(self[i], i+1);
- }
- >
- ! !
- !SequenceableCollection methodsFor: 'private'!
- numericallyIndexable
- "This is an internal converting message.
- It answeres a representation of the receiver
- that can use foo[i] in JavaScript code.
-
- It fixes IE8, where boxed String is unable
- to numerically index its characters,
- but primitive string can."
-
- self subclassResponsibility
- ! !
- !SequenceableCollection methodsFor: 'streaming'!
- newStream
- ^ self streamClass on: self
- !
- readStream
- "For Pharo compatibility"
-
- ^ self stream
- !
- stream
- ^ self newStream
- !
- streamClass
- ^ self class streamClass
- !
- writeStream
- "For Pharo compatibility"
-
- ^ self stream
- ! !
- !SequenceableCollection methodsFor: 'testing'!
- beginsWith: prefix
- self size < prefix size ifTrue: [ ^ false ].
- ^ (self first: prefix size) = prefix
- !
- endsWith: suffix
- self size < suffix size ifTrue: [ ^ false ].
- ^ (self last: suffix size) = suffix
- !
- includes: anObject
- ^ (self indexOf: anObject ifAbsent: [ nil ]) notNil
- ! !
- !SequenceableCollection class methodsFor: 'accessing'!
- streamClass
- ^ Stream
- ! !
- !SequenceableCollection class methodsFor: 'streaming'!
- streamContents: aBlock
- | stream |
- stream := (self streamClass on: self new).
- aBlock value: stream.
- ^ stream contents
- ! !
- SequenceableCollection subclass: #Array
- instanceVariableNames: ''
- package: 'Kernel-Collections'!
- !Array commentStamp!
- I represent a collection of objects ordered by the collector. The size of arrays is dynamic.
- I am directly mapped to JavaScript Number.
- *Note* In Amber, `OrderedCollection` is an alias for `Array`.!
- !Array methodsFor: 'accessing'!
- at: anIndex ifAbsent: aBlock
- <
- return anIndex >>= 1 && anIndex <= self.length
- ? self[anIndex - 1]
- : aBlock._value()
- >
- !
- at: anIndex ifPresent: aBlock ifAbsent: anotherBlock
- <
- return anIndex >>= 1 && anIndex <= self.length
- ? aBlock._value_(self[anIndex - 1])
- : anotherBlock._value()
- >
- !
- at: anIndex put: anObject
- <return self[anIndex - 1] = anObject>
- !
- size
- <return self.length>
- ! !
- !Array methodsFor: 'adding/removing'!
- add: anObject
- <self.push(anObject); return anObject;>
- !
- addFirst: anObject
- <self.unshift(anObject); return anObject;>
- !
- remove: anObject ifAbsent: aBlock
- <
- for(var i=0;i<self.length;i++) {
- if(_st(self[i]).__eq(anObject)) {
- self.splice(i,1);
- return self;
- }
- };
- aBlock._value();
- >
- !
- removeAll
- <self.length = 0>
- !
- removeFrom: aNumber to: anotherNumber
- <self.splice(aNumber -1, anotherNumber - aNumber + 1)>
- !
- removeIndex: anInteger
- <self.splice(anInteger - 1, 1)>
- !
- removeLast
- <return self.pop();>
- ! !
- !Array methodsFor: 'converting'!
- asJavascript
- ^ '[', ((self collect: [:each | each asJavascript ]) join: ', '), ']'
- !
- reversed
- <return self._copy().reverse()>
- ! !
- !Array methodsFor: 'enumerating'!
- collect: aBlock
- "Optimized version"
-
- <return self.map(function(each) {return aBlock._value_(each)})>
- !
- join: aString
- <return self.join(aString)>
- !
- select: aBlock
- "Optimized version"
-
- <
- var result = self.klass._new();
- for(var i=0; i<self.length; i++) {
- if(aBlock._value_(self[i])) {
- result.push(self[i]);
- }
- }
- return result;
- >
- !
- sort
- ^ self basicPerform: 'sort'
- !
- sort: aBlock
- <
- return self.sort(function(a, b) {
- if(aBlock._value_value_(a,b)) {return -1} else {return 1}
- })
- >
- !
- sorted
- ^ self copy sort
- !
- sorted: aBlock
- ^ self copy sort: aBlock
- ! !
- !Array methodsFor: 'printing'!
- printOn: aStream
- super printOn: aStream.
-
- aStream nextPutAll: ' ('.
- self
- do: [ :each | each printOn: aStream ]
- separatedBy: [ aStream nextPutAll: ' ' ].
- aStream nextPutAll: ')'
- ! !
- !Array methodsFor: 'private'!
- numericallyIndexable
- ^ self
- ! !
- !Array class methodsFor: 'instance creation'!
- new: anInteger
- <return new Array(anInteger)>
- !
- with: anObject
- ^ (self new: 1)
- at: 1 put: anObject;
- yourself
- !
- with: anObject with: anObject2
- ^ (self new: 2)
- at: 1 put: anObject;
- at: 2 put: anObject2;
- yourself
- !
- with: anObject with: anObject2 with: anObject3
- ^ (self new: 3)
- at: 1 put: anObject;
- at: 2 put: anObject2;
- at: 3 put: anObject3;
- yourself
- !
- withAll: aCollection
- | instance index |
- index := 1.
- instance := self new: aCollection size.
- aCollection do: [ :each |
- instance at: index put: each.
- index := index + 1 ].
- ^ instance
- ! !
- SequenceableCollection subclass: #CharacterArray
- instanceVariableNames: ''
- package: 'Kernel-Collections'!
- !CharacterArray commentStamp!
- I am the abstract superclass of string-like collections.!
- !CharacterArray methodsFor: 'accessing'!
- at: anIndex put: anObject
- self errorReadOnly
- ! !
- !CharacterArray methodsFor: 'adding/removing'!
- add: anObject
- self errorReadOnly
- !
- remove: anObject
- self errorReadOnly
- ! !
- !CharacterArray methodsFor: 'converting'!
- asLowercase
- ^ self class fromString: self asString asLowercase
- !
- asNumber
- ^ self asString asNumber
- !
- asString
- ^ self subclassResponsibility
- !
- asSymbol
- ^ self asString
- !
- asUppercase
- ^ self class fromString: self asString asUppercase
- ! !
- !CharacterArray methodsFor: 'copying'!
- , aString
- ^ self asString, aString asString
- ! !
- !CharacterArray methodsFor: 'error handling'!
- errorReadOnly
- self error: 'Object is read-only'
- ! !
- !CharacterArray methodsFor: 'printing'!
- printOn: aStream
- self asString printOn: aStream
- ! !
- !CharacterArray methodsFor: 'streaming'!
- putOn: aStream
- aStream nextPutString: self
- ! !
- !CharacterArray class methodsFor: 'instance creation'!
- fromString: aString
- self subclassResponsibility
- ! !
- CharacterArray subclass: #String
- instanceVariableNames: ''
- package: 'Kernel-Collections'!
- !String commentStamp!
- I am an indexed collection of Characters. Unlike most Smalltalk dialects, Amber doesn't provide the Character class. Instead, elements of a String are single character strings.
- String inherits many useful methods from its hierarchy, such as
- `Collection >> #,`!
- !String methodsFor: 'accessing'!
- asciiValue
- <return self.charCodeAt(0);>
- !
- at: anIndex ifAbsent: aBlock
- <return String(self)[anIndex - 1] || aBlock._value()>
- !
- at: anIndex ifPresent: aBlock ifAbsent: anotherBlock
- <
- var result = String(self)[anIndex - 1];
- return result ? aBlock._value_(result) : anotherBlock._value();
- >
- !
- charCodeAt: anInteger
- <return self.charCodeAt(anInteger - 1)>
- !
- identityHash
- ^ self, 's'
- !
- size
- <return self.length>
- ! !
- !String methodsFor: 'comparing'!
- < aString
- <return String(self) < aString._asString()>
- !
- <= aString
- <return String(self) <= aString._asString()>
- !
- = aString
- <
- return aString !!= null &&
- typeof aString._isString === "function" &&
- aString._isString() &&
- String(self) === String(aString)
- >
- !
- == aString
- ^ self = aString
- !
- > aString
- <return String(self) >> aString._asString()>
- !
- >= aString
- <return String(self) >>= aString._asString()>
- ! !
- !String methodsFor: 'converting'!
- asJSON
- ^ self
- !
- asJavascript
- <
- if(self.search(/^[a-zA-Z0-9_:.$ ]*$/) == -1)
- return "\"" + self.replace(/[\x00-\x1f"\\\x7f-\x9f]/g, function(ch){var c=ch.charCodeAt(0);return "\\x"+("0"+c.toString(16)).slice(-2)}) + "\"";
- else
- return "\"" + self + "\"";
- >
- !
- asLowercase
- <return self.toLowerCase()>
- !
- asMutator
- "Answer a setter selector. For example,
- #name asMutator returns #name:"
- self last = ':' ifFalse: [ ^ self, ':' ].
- ^ self
- !
- asNumber
- <return Number(self)>
- !
- asRegexp
- ^ RegularExpression fromString: self
- !
- asSelector
- <return smalltalk.selector(self)>
- !
- asString
- ^ self
- !
- asSymbol
- ^ self
- !
- asUppercase
- <return self.toUpperCase()>
- !
- capitalized
- ^ self isEmpty
- ifTrue: [ self ]
- ifFalse: [ self first asUppercase, self allButFirst ]
- !
- crlfSanitized
- ^ self lines join: String lf
- !
- escaped
- <return escape(self)>
- !
- reversed
- <return self.split("").reverse().join("")>
- !
- unescaped
- <return unescape(self)>
- !
- uriComponentDecoded
- <return decodeURIComponent(self)>
- !
- uriComponentEncoded
- <return encodeURIComponent(self)>
- !
- uriDecoded
- <return decodeURI(self)>
- !
- uriEncoded
- <return encodeURI(self)>
- ! !
- !String methodsFor: 'copying'!
- , aString
- <return String(self) + aString>
- !
- copyFrom: anIndex to: anotherIndex
- <return self.substring(anIndex - 1, anotherIndex)>
- !
- deepCopy
- ^ self shallowCopy
- !
- shallowCopy
- ^ self class fromString: self
- ! !
- !String methodsFor: 'printing'!
- printNl
- <console.log(self)>
- !
- printOn: aStream
- aStream
- nextPutAll: '''';
- nextPutAll: self;
- nextPutAll: ''''
- ! !
- !String methodsFor: 'private'!
- numericallyIndexable
- <return String(self)>
- ! !
- !String methodsFor: 'regular expressions'!
- match: aRegexp
- <return self.search(aRegexp) !!= -1>
- !
- matchesOf: aRegularExpression
- <return self.match(aRegularExpression)>
- !
- replace: aString with: anotherString
- ^ self replaceRegexp: (RegularExpression fromString: aString flag: 'g') with: anotherString
- !
- replaceRegexp: aRegexp with: aString
- <return self.replace(aRegexp, aString)>
- !
- trimBoth
- ^ self trimBoth: '\s'
- !
- trimBoth: separators
- ^ (self trimLeft: separators) trimRight: separators
- !
- trimLeft
- ^ self trimLeft: '\s'
- !
- trimLeft: separators
- ^ self replaceRegexp: (RegularExpression fromString: '^[', separators, ']+' flag: 'g') with: ''
- !
- trimRight
- ^ self trimRight: '\s'
- !
- trimRight: separators
- ^ self replaceRegexp: (RegularExpression fromString: '[', separators, ']+$' flag: 'g') with: ''
- ! !
- !String methodsFor: 'split join'!
- join: aCollection
- ^ String
- streamContents: [ :stream | aCollection
- do: [ :each | stream nextPutAll: each asString ]
- separatedBy: [ stream nextPutAll: self ]]
- !
- lineIndicesDo: aBlock
- "execute aBlock with 3 arguments for each line:
- - start index of line
- - end index of line without line delimiter
- - end index of line including line delimiter(s) CR, LF or CRLF"
-
- | cr lf start sz nextLF nextCR |
- start := 1.
- sz := self size.
- cr := String cr.
- nextCR := self indexOf: cr startingAt: 1.
- lf := String lf.
- nextLF := self indexOf: lf startingAt: 1.
- [ start <= sz ] whileTrue: [
- (nextLF = 0 and: [ nextCR = 0 ])
- ifTrue: [ "No more CR, nor LF, the string is over"
- aBlock value: start value: sz value: sz.
- ^ self ].
- (nextCR = 0 or: [ 0 < nextLF and: [ nextLF < nextCR ] ])
- ifTrue: [ "Found a LF"
- aBlock value: start value: nextLF - 1 value: nextLF.
- start := 1 + nextLF.
- nextLF := self indexOf: lf startingAt: start ]
- ifFalse: [ 1 + nextCR = nextLF
- ifTrue: [ "Found a CR-LF pair"
- aBlock value: start value: nextCR - 1 value: nextLF.
- start := 1 + nextLF.
- nextCR := self indexOf: cr startingAt: start.
- nextLF := self indexOf: lf startingAt: start ]
- ifFalse: [ "Found a CR"
- aBlock value: start value: nextCR - 1 value: nextCR.
- start := 1 + nextCR.
- nextCR := self indexOf: cr startingAt: start ] ]]
- !
- lineNumber: anIndex
- "Answer a string containing the characters in the given line number."
- | lineCount |
- lineCount := 0.
- self lineIndicesDo: [ :start :endWithoutDelimiters :end |
- (lineCount := lineCount + 1) = anIndex ifTrue: [ ^ self copyFrom: start to: endWithoutDelimiters ]].
- ^ nil
- !
- lines
- "Answer an array of lines composing this receiver without the line ending delimiters."
- | lines |
- lines := Array new.
- self linesDo: [ :aLine | lines add: aLine ].
- ^ lines
- !
- linesDo: aBlock
- "Execute aBlock with each line in this string. The terminating line
- delimiters CR, LF or CRLF pairs are not included in what is passed to aBlock"
- self lineIndicesDo: [ :start :endWithoutDelimiters :end |
- aBlock value: (self copyFrom: start to: endWithoutDelimiters) ]
- !
- subStrings: aString
- ^ self tokenize: aString
- !
- tokenize: aString
- <return self.split(aString)>
- ! !
- !String methodsFor: 'testing'!
- includesSubString: subString
- <return self.indexOf(subString) !!= -1>
- !
- isCapitalized
- ^ self first asUppercase == self first
- !
- isImmutable
- ^ true
- !
- isString
- ^ true
- !
- isVowel
- "Answer true if the receiver is a one character string containing a voyel"
-
- ^ self size = 1 and: [ 'aeiou' includes: self asLowercase ]
- ! !
- !String class methodsFor: 'accessing'!
- cr
- <return '\r'>
- !
- crlf
- <return '\r\n'>
- !
- esc
- ^ self fromCharCode: 27
- !
- lf
- <return '\n'>
- !
- space
- <return ' '>
- !
- streamClass
- ^ StringStream
- !
- tab
- <return '\t'>
- ! !
- !String class methodsFor: 'instance creation'!
- fromCharCode: anInteger
- <return String.fromCharCode(anInteger)>
- !
- fromString: aString
- <return String(aString)>
- !
- value: aUTFCharCode
- <return String.fromCharCode(aUTFCharCode);>
- ! !
- !String class methodsFor: 'random'!
- random
- "Returns random alphanumeric string beginning with letter"
- <return (Math.random()*(22/32)+(10/32)).toString(32).slice(2);>
- !
- randomNotIn: aString
- | result |
- [ result := self random. aString includesSubString: result ] whileTrue.
- ^ result
- ! !
- Collection subclass: #Set
- instanceVariableNames: 'elements'
- package: 'Kernel-Collections'!
- !Set commentStamp!
- I represent an unordered set of objects without duplicates.!
- !Set methodsFor: 'accessing'!
- size
- ^ elements size
- ! !
- !Set methodsFor: 'adding/removing'!
- add: anObject
- <
- var found;
- for(var i=0; i < self['@elements'].length; i++) {
- if(_st(anObject).__eq(self['@elements'][i])) {
- found = true;
- break;
- }
- }
- if(!!found) {self['@elements'].push(anObject)}
- >
- !
- remove: anObject
- elements remove: anObject
- !
- remove: anObject ifAbsent: aBlock
- elements remove: anObject ifAbsent: aBlock
- !
- removeAll
- elements removeAll
- ! !
- !Set methodsFor: 'comparing'!
- = aCollection
- self class = aCollection class ifFalse: [ ^ false ].
- self size = aCollection size ifFalse: [ ^ false ].
- self do: [ :each | (aCollection includes: each) ifFalse: [ ^ false ] ].
- ^ true
- ! !
- !Set methodsFor: 'converting'!
- asArray
- ^ elements copy
- ! !
- !Set methodsFor: 'enumerating'!
- collect: aBlock
- ^ self class withAll: (elements collect: aBlock)
- !
- detect: aBlock ifNone: anotherBlock
- ^ elements detect: aBlock ifNone: anotherBlock
- !
- do: aBlock
- elements do: aBlock
- !
- select: aBlock
- | collection |
- collection := self class new.
- self do: [ :each |
- (aBlock value: each) ifTrue: [
- collection add: each ]].
- ^ collection
- ! !
- !Set methodsFor: 'initialization'!
- initialize
- super initialize.
- elements := #()
- ! !
- !Set methodsFor: 'printing'!
- printOn: aStream
- super printOn: aStream.
-
- aStream nextPutAll: ' ('.
- self
- do: [ :each | each printOn: aStream ]
- separatedBy: [ aStream nextPutAll: ' ' ].
- aStream nextPutAll: ')'
- ! !
- !Set methodsFor: 'testing'!
- includes: anObject
- ^ elements includes: anObject
- ! !
- Object subclass: #Queue
- instanceVariableNames: 'read readIndex write'
- package: 'Kernel-Collections'!
- !Queue commentStamp!
- I am a one-sided queue.
- ## Usage
- Use `#nextPut:` to add items to the queue.
- Use `#next` or `#nextIfAbsent:` to get (and remove) the next item in the queue.
- ## Implementation notes
- A Queue uses two OrderedCollections inside,
- `read` is at the front, is not modified and only read using `readIndex`.
- `write` is at the back and is appended new items.
- When `read` is exhausted, `write` is promoted to `read` and new `write` is created.
- As a consequence, no data moving is done by me, write appending may do data moving
- when growing `write`, but this is left to engine to implement as good as it chooses to.!
- !Queue methodsFor: 'accessing'!
- next
- ^ self nextIfAbsent: [ self error: 'Cannot read from empty Queue.' ]
- !
- nextIfAbsent: aBlock
- | result |
- result := read at: readIndex ifAbsent: [
- write isEmpty ifTrue: [
- readIndex > 1 ifTrue: [ read := #(). readIndex := 1 ].
- ^ aBlock value ].
- read := write.
- readIndex := 1.
- write := OrderedCollection new.
- read first ].
- read at: readIndex put: nil.
- readIndex := readIndex + 1.
- ^ result
- !
- nextPut: anObject
- write add: anObject
- ! !
- !Queue methodsFor: 'initialization'!
- initialize
- super initialize.
- read := OrderedCollection new.
- write := OrderedCollection new.
- readIndex := 1
- ! !
- Object subclass: #RegularExpression
- instanceVariableNames: ''
- package: 'Kernel-Collections'!
- !RegularExpression commentStamp!
- I represent a regular expression object. My instances are JavaScript `RegExp` object.!
- !RegularExpression methodsFor: 'evaluating'!
- compile: aString
- <return self.compile(aString)>
- !
- exec: aString
- <return self.exec(aString) || nil>
- !
- test: aString
- <return self.test(aString)>
- ! !
- !RegularExpression class methodsFor: 'instance creation'!
- fromString: aString
- ^ self fromString: aString flag: ''
- !
- fromString: aString flag: anotherString
- <return new RegExp(aString, anotherString)>
- ! !
- Object subclass: #Stream
- instanceVariableNames: 'collection position streamSize'
- package: 'Kernel-Collections'!
- !Stream commentStamp!
- I represent an accessor for a sequence of objects. This sequence is referred to as my "contents".
- My instances are read/write streams to the contents sequence collection.!
- !Stream methodsFor: 'accessing'!
- collection
- ^ collection
- !
- contents
- ^ self collection
- copyFrom: 1
- to: self streamSize
- !
- position
- ^ position ifNil: [ position := 0 ]
- !
- position: anInteger
- position := anInteger
- !
- setCollection: aCollection
- collection := aCollection
- !
- setStreamSize: anInteger
- streamSize := anInteger
- !
- size
- ^ self streamSize
- !
- streamSize
- ^ streamSize
- ! !
- !Stream methodsFor: 'actions'!
- close
- !
- flush
- !
- reset
- self position: 0
- !
- resetContents
- self reset.
- self setStreamSize: 0
- ! !
- !Stream methodsFor: 'enumerating'!
- do: aBlock
- [ self atEnd ] whileFalse: [ aBlock value: self next ]
- ! !
- !Stream methodsFor: 'positioning'!
- setToEnd
- self position: self size
- !
- skip: anInteger
- self position: ((self position + anInteger) min: self size max: 0)
- ! !
- !Stream methodsFor: 'reading'!
- next
- ^ self atEnd
- ifTrue: [ nil ]
- ifFalse: [
- self position: self position + 1.
- collection at: self position ]
- !
- next: anInteger
- | tempCollection |
- tempCollection := self collection class new.
- anInteger timesRepeat: [
- self atEnd ifFalse: [
- tempCollection add: self next ]].
- ^ tempCollection
- !
- peek
- ^ self atEnd ifFalse: [
- self collection at: self position + 1 ]
- ! !
- !Stream methodsFor: 'testing'!
- atEnd
- ^ self position = self size
- !
- atStart
- ^ self position = 0
- !
- isEmpty
- ^ self size = 0
- ! !
- !Stream methodsFor: 'writing'!
- << anObject
- self write: anObject
- !
- nextPut: anObject
- self position: self position + 1.
- self collection at: self position put: anObject.
- self setStreamSize: (self streamSize max: self position)
- !
- nextPutAll: aCollection
- aCollection do: [ :each |
- self nextPut: each ]
- !
- nextPutString: aString
- self nextPut: aString
- !
- write: anObject
- anObject putOn: self
- ! !
- !Stream class methodsFor: 'instance creation'!
- on: aCollection
- ^ self new
- setCollection: aCollection;
- setStreamSize: aCollection size;
- yourself
- ! !
- Stream subclass: #StringStream
- instanceVariableNames: ''
- package: 'Kernel-Collections'!
- !StringStream commentStamp!
- I am a Stream specific to `String` objects.!
- !StringStream methodsFor: 'reading'!
- next: anInteger
- | tempCollection |
- tempCollection := self collection class new.
- anInteger timesRepeat: [
- self atEnd ifFalse: [
- tempCollection := tempCollection, self next ]].
- ^ tempCollection
- ! !
- !StringStream methodsFor: 'writing'!
- cr
- ^ self nextPutAll: String cr
- !
- crlf
- ^ self nextPutAll: String crlf
- !
- lf
- ^ self nextPutAll: String lf
- !
- nextPut: aString
- self nextPutAll: aString
- !
- nextPutAll: aString
- | pre post |
- self atEnd ifTrue: [ self setCollection: self collection, aString ] ifFalse: [
- pre := self collection copyFrom: 1 to: self position.
- post := self collection copyFrom: (self position + 1 + aString size) to: self collection size.
- self setCollection: pre, aString, post
- ].
- self position: self position + aString size.
- self setStreamSize: (self streamSize max: self position)
- !
- nextPutString: aString
- self nextPutAll: aString
- !
- space
- self nextPut: ' '
- !
- tab
- ^ self nextPutAll: String tab
- ! !
|