1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539 |
- Smalltalk createPackage: 'Kernel-Collections'!
- Object subclass: #Association
- slots: {#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: #BucketStore
- slots: {#buckets. #hashBlock}
- package: 'Kernel-Collections'!
- !BucketStore commentStamp!
- I am an helper class for hash-based stores.
- I hold buckets which are selected by a hash, specified using `#hashBlock:`.
- The hash can be any object, and
- it is used as a JS property (that is, in ES5
- its toString() value counts).
- ## API
- I maintain a list of buckets. Client code can use this API:
- - `#bucketOfElement:` (to ask a bucket for element, I can return JS null if n/a)
- - `#do:` (to enumerate all elements of all buckets)
- - `#removeAll` (to remove all buckets)
- Client code itself should add/remove elements
- in a bucket. The `nil` object should not be put into any bucket.
- Types of buckets are the responsibility of subclasses via `#newBucket`.!
- !BucketStore methodsFor: 'accessing'!
- bucketOfElement: anObject
- <inlineJS: '
- var hash = $self.hashBlock(anObject);
- if (!!hash) return null;
- var buckets = $self.buckets,
- bucket = buckets[hash];
- if (!!bucket) { bucket = buckets[hash] = $self._newBucket(); }
- return bucket;
- '>
- !
- hashBlock: aBlock
- hashBlock := aBlock
- ! !
- !BucketStore methodsFor: 'adding/removing'!
- removeAll
- <inlineJS: '$self.buckets = Object.create(null);'>
- ! !
- !BucketStore methodsFor: 'enumerating'!
- do: aBlock
- <inlineJS: '
- var buckets = $self.buckets;
- var keys = Object.keys(buckets);
- for (var i = 0; i < keys.length; ++i) { buckets[keys[i]]._do_(aBlock); }
- '>
- ! !
- !BucketStore methodsFor: 'initialization'!
- initialize
- super initialize.
- self removeAll
- ! !
- !BucketStore methodsFor: 'private'!
- newBucket
- self subclassResponsibility
- ! !
- !BucketStore class methodsFor: 'instance creation'!
- hashBlock: aBlock
- ^ self new
- hashBlock: aBlock;
- yourself
- ! !
- BucketStore subclass: #ArrayBucketStore
- slots: {}
- package: 'Kernel-Collections'!
- !ArrayBucketStore commentStamp!
- I am a concrete `BucketStore` with buckets being instance of `Array`.!
- !ArrayBucketStore methodsFor: 'private'!
- newBucket
- ^ #()
- ! !
- Object subclass: #Collection
- slots: {}
- 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'!
- 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 ]
- !
- 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
- !
- single
- "Answer a single element.
- Raise an error if collection holds less or more than one element."
- self ifEmpty: [ self error: 'Collection is empty' ].
- self size > 1 ifTrue: [ self error: 'Collection holds more than one element' ].
- ^ self anyOne
- !
- size
- self subclassResponsibility
- ! !
- !Collection methodsFor: 'adding/removing'!
- add: anObject
- self subclassResponsibility
- !
- addAll: aCollection
- aCollection do: [ :each |
- self add: each ].
- ^ aCollection
- !
- remove: anObject
- ^ self remove: anObject ifAbsent: [ self errorNotFound ]
- !
- remove: anObject ifAbsent: aBlock
- self subclassResponsibility
- !
- removeAll
- self subclassResponsibility
- ! !
- !Collection methodsFor: 'converting'!
- asArray
- ^ Array withAll: self
- !
- asJavaScriptObject
- ^ self asArray collect: [ :each | each asJavaScriptObject ]
- !
- asOrderedCollection
- ^ self asArray
- !
- asSet
- ^ Set withAll: self
- ! !
- !Collection methodsFor: 'copying'!
- , aCollection
- ^ self copy
- addAll: aCollection;
- yourself
- !
- copyEmpty
- ^ self class new
- !
- copyWith: anObject
- ^ self copy add: anObject; yourself
- !
- copyWithAll: aCollection
- self deprecatedAPI: 'Use #, instead.'.
- ^ self, aCollection
- !
- copyWithout: anObject
- "Answer a copy of the receiver that does not contain
- any occurrences of anObject."
- ^ self reject: [ :each | each = anObject ]
- !
- 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 ]
- !
- deepCopy
- ^ self collect: [ :each | each deepCopy ]
- !
- shallowCopy
- ^ self collect: [ :each | 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 do: [ :each | (aBlock value: each) ifTrue: [ ^each ] ].
- ^ anotherBlock value
- !
- 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: 'printing'!
- shortenedPrintString
- ^ self size <= 1
- ifTrue: [ self printString ]
- ifFalse: [ (self copyEmpty copyWith: self anyOne) printString, ' ... ', (self size - 1) asString, ' more items' ]
- ! !
- !Collection methodsFor: 'streaming'!
- putOn: aStream
- self do: [ :each | each putOn: aStream ]
- ! !
- !Collection methodsFor: 'testing'!
- 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" [ aBlock value ]
- ifFalse: [ self ]
- !
- ifEmpty: aBlock ifNotEmpty: anotherBlock
- ^ self isEmpty
- ifTrue: "aBlock" [ aBlock value ]
- ifFalse: [ anotherBlock value: self ]
- !
- ifNotEmpty: aBlock
- ^ self notEmpty
- ifTrue: [ aBlock value: self ]
- ifFalse: [ self ]
- !
- ifNotEmpty: aBlock ifEmpty: anotherBlock
- ^ self notEmpty
- ifTrue: [ aBlock value: self ]
- ifFalse: "anotherBlock" [ anotherBlock value ]
- !
- includes: anObject
- ^ self anySatisfy: [ :each | each = anObject ]
- !
- isEmpty
- ^ self size = 0
- !
- notEmpty
- ^ self isEmpty not
- ! !
- !Collection class methodsFor: 'accessing'!
- classTag
- "Returns a tag or general category for this class.
- Typically used to help tools do some reflection.
- Helios, for example, uses this to decide what icon the class should display."
-
- ^ '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: #AssociativeCollection
- slots: {}
- 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: anAssociativeCollection
- super addAll: anAssociativeCollection associations.
- ^ anAssociativeCollection
- !
- 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'!
- = anAssociativeCollection
- ^ self class = anAssociativeCollection class and: [
- self size = anAssociativeCollection size and: [
- | comparisons |
- comparisons := OrderedCollection new.
- (self associations allSatisfy: [ :each |
- anAssociativeCollection at: each key
- ifPresent: [ :otherValue | comparisons add: { each value. otherValue }. true ]
- ifAbsent: [ false ] ]) and: [
- comparisons allSatisfy: [ :each | each first = each second ] ] ] ]
- ! !
- !AssociativeCollection methodsFor: 'converting'!
- asDictionary
- ^ Dictionary from: self associations
- !
- asHashedCollection
- ^ HashedCollection from: self associations
- !
- asJavaScriptObject
- | hash |
- hash := HashedCollection new.
- self keysAndValuesDo: [ :key :value |
- hash at: key put: value asJavaScriptObject ].
- ^ hash
- ! !
- !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
- !
- select: selectBlock thenCollect: collectBlock
- | newDict |
- newDict := self class new.
- self keysAndValuesDo: [ :key :value |
- (selectBlock value: value) ifTrue: [ newDict at: key put: (collectBlock value: 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: ')'
- !
- shortenedPrintString
- ^ self size <= 1
- ifTrue: [ self printString ]
- ifFalse: [ | key | key := self keys anyOne. (self copyEmpty at: key put: (self at: key); yourself) printString, ' ... ', (self size - 1) asString, ' more items' ]
- ! !
- !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
- slots: {#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
- <inlineJS: '
- var index = $self._positionOfKey_(aKey);
- return index >=0 ? $self.values[index] : aBlock._value();
- '>
- !
- at: aKey put: aValue
- <inlineJS: '
- 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
- <inlineJS: '
- 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
- <inlineJS: '
- 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
- <inlineJS: 'return $self._positionOfKey_(aKey) >= 0;'>
- ! !
- AssociativeCollection subclass: #HashedCollection
- slots: {}
- 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'!
- asJavaScriptSource
- ^ self
- ifEmpty: [ '{}' ]
- ifNotEmpty: [
- String streamContents: [ :str |
- str nextPut: '{'.
- self keysAndValuesDo: [ :key :value |
- str nextPutAll: key asJavaScriptSource; nextPut: ':'; nextPutAll: value asJavaScriptSource; nextPut: ',' ].
- str skip: -1; nextPut: '}' ] ]
- !
- at: aKey ifAbsent: aBlock
- ^ (self includesKey: aKey)
- ifTrue: [ self basicAt: aKey ]
- ifFalse: [ aBlock value ]
- !
- at: aKey put: aValue
- ^ self basicAt: aKey put: aValue
- !
- keys
- <inlineJS: 'return Object.keys(self)'>
- !
- values
- <inlineJS: '
- 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: 'converting'!
- jsonLiteralized
- ^ JSON parse: (JSON stringify: self)
- ! !
- !HashedCollection methodsFor: 'enumerating'!
- keysDo: aBlock
- self keys do: aBlock
- !
- valuesDo: aBlock
- self values do: aBlock
- ! !
- !HashedCollection methodsFor: 'testing'!
- includesKey: aKey
- <inlineJS: 'return self.hasOwnProperty(aKey)'>
- ! !
- Collection subclass: #SequenceableCollection
- slots: {}
- 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
- !
- anyOne
- ^ self at: 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 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 subclassResponsibility
- !
- 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
- self subclassResponsibility
- !
- copyWithFirst: anObject
- ^ (self class with: anObject) addAll: self; yourself
- ! !
- !SequenceableCollection methodsFor: 'enumerating'!
- pairsCollect: aBlock
- "Evaluate aBlock with my elements taken two at a time,
- and return an Array with the results"
- "(#(1 'fred' 2 'charlie' 3 'elmer') pairsCollect: [:a :b | b, ' is number ', a printString]) >>> #('fred is number 1' 'charlie is number 2' 'elmer is number 3')"
- ^ (1 to: self size // 2) collect: [ :index |
- aBlock value: (self at: 2 * index - 1) value: (self at: 2 * index) ]
- !
- pairsDo: aBlock
- "Evaluate aBlock with my elements taken two at a time.
- If there's an odd number of items, ignore the last one.
- Allows use of a flattened array for things that naturally group into pairs.
- See also pairsCollect:"
- "(#(1 'fred' 2 'charlie' 3 'elmer') pairsDo: [:a :b | Transcript cr; show: b, ' is number ', a printString]) >>> #(1 'fred' 2 'charlie' 3 'elmer')"
- 1 to: self size // 2 do: [ :index |
- aBlock value: (self at: 2 * index - 1) value: (self at: 2 * index) ]
- !
- reverseDo: aBlock
- self reversed do: aBlock
- ! !
- !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
- slots: {}
- 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 put: anObject
- <inlineJS: 'return self[anIndex - 1] = anObject'>
- ! !
- !Array methodsFor: 'adding/removing'!
- add: anObject
- <inlineJS: 'self.push(anObject); return anObject;'>
- !
- addAll: aCollection
- <inlineJS: '
- if (Array.isArray(aCollection) && aCollection.length < 65000) self.push.apply(self, aCollection);
- else $globals.Array.superclass.fn.prototype._addAll_.call($self, aCollection);
- return aCollection;
- '>
- !
- addFirst: anObject
- <inlineJS: 'self.unshift(anObject); return anObject;'>
- !
- remove: anObject ifAbsent: aBlock
- | index |
- index := self indexOf: anObject ifAbsent: [ 0 ].
- ^ index = 0
- ifFalse: [ self removeIndex: index. anObject ]
- ifTrue: [ aBlock value ]
- !
- removeAll
- <inlineJS: 'self.length = 0'>
- !
- removeFrom: aNumber to: anotherNumber
- <inlineJS: 'self.splice(aNumber -1, anotherNumber - aNumber + 1)'>
- !
- removeIndex: anInteger
- <inlineJS: 'self.splice(anInteger - 1, 1)'>
- !
- removeLast
- <inlineJS: 'return self.pop();'>
- ! !
- !Array methodsFor: 'converting'!
- asJavaScriptSource
- ^ '[', ((self collect: [:each | each asJavaScriptSource ]) join: ', '), ']'
- !
- reversed
- <inlineJS: 'return self.slice().reverse()'>
- ! !
- !Array methodsFor: 'copying'!
- appendToString: aString
- <inlineJS: '
- for (var i = 0, l = $self.length; i < l; ++i) {
- var el = $self[i];
- if ((typeof el === "string") || $recv(el)._isString()) {
- if (el.length === 1) { aString += el; continue; }
- }
- $self._error_("Not a character.");
- }
- return aString'>
- !
- copyFrom: anIndex to: anotherIndex
- <inlineJS: '
- if (anIndex >= 1 && anotherIndex <= self.length) {
- return self.slice(anIndex - 1, anotherIndex);
- } else {
- self._at_(anIndex);
- self._at_(self.length + 1);
- throw new Error("Incorrect indexes in #copyFrom:to: not caught by #at:");
- }
- '>
- !
- shallowCopy
- <inlineJS: 'return self.slice()'>
- ! !
- !Array methodsFor: 'enumerating'!
- allIn: aBlock
- ^ aBlock valueWithPossibleArguments:
- "collect to match #in: behaviour"
- (self collect: [ :each | each in: [ :x | x ] ])
- !
- collect: aBlock
- "Optimized version"
-
- <inlineJS: 'return self.map(function(each) {return aBlock._value_(each)})'>
- !
- join: aString
- <inlineJS: 'return self.join(aString)'>
- !
- select: aBlock
- "Optimized version"
-
- <inlineJS: 'return self.filter(function(each) {return aBlock._value_(each)})'>
- !
- sort
- ^ self sort: [ :a :b | a < b ]
- !
- sort: aBlock
- <inlineJS: '
- 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 class methodsFor: 'instance creation'!
- new: anInteger
- <inlineJS: '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: #String
- slots: {}
- 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
- <inlineJS: 'return self.charCodeAt(0);'>
- !
- at: anIndex ifAbsent: aBlock
- <inlineJS: 'return String(self)[anIndex - 1] || aBlock._value()'>
- !
- at: anIndex ifPresent: aBlock ifAbsent: anotherBlock
- <inlineJS: '
- var result = String(self)[anIndex - 1];
- return result ? aBlock._value_(result) : anotherBlock._value();
- '>
- !
- at: anIndex put: anObject
- self errorReadOnly
- !
- charCodeAt: anInteger
- <inlineJS: 'return self.charCodeAt(anInteger - 1)'>
- ! !
- !String methodsFor: 'adding/removing'!
- add: anObject
- self errorReadOnly
- !
- remove: anObject
- self errorReadOnly
- !
- remove: anObject ifAbsent: aBlock
- self errorReadOnly
- ! !
- !String methodsFor: 'comparing'!
- < aString
- <inlineJS: 'return typeof aString === "string" ?
- String(self) < aString :
- $recv(aString)._isStringLessThanSelf_(String(self))'>
- !
- <= aString
- <inlineJS: 'return typeof aString === "string" ?
- String(self) <= aString :
- $recv(aString)._isStringLessThanOrEqualToSelf_(String(self))'>
- !
- = aString
- <inlineJS: 'return typeof aString === "string" ?
- String(self) === aString :
- $recv(aString)._isStringEqualToSelf_(String(self))'>
- !
- == aString
- <inlineJS: 'return typeof aString === "string" ?
- String(self) === aString :
- $recv(aString)._isStringEqualToSelf_(String(self))'>
- !
- > aString
- <inlineJS: 'return typeof aString === "string" ?
- String(self) > aString :
- $recv(aString)._isStringGreaterThanSelf_(String(self))'>
- !
- >= aString
- <inlineJS: 'return typeof aString === "string" ?
- String(self) >= aString :
- $recv(aString)._isStringGreaterThanOrEqualSelf_(String(self))'>
- !
- isStringEqualToSelf: aString
- <inlineJS: 'return aString === String(self)'>
- !
- isStringGreaterThanOrEqualToSelf: aString
- <inlineJS: 'return aString >= self'>
- !
- isStringGreaterThanSelf: aString
- <inlineJS: 'return aString > self'>
- !
- isStringLessThanOrEqualToSelf: aString
- <inlineJS: 'return aString <= self'>
- !
- isStringLessThanSelf: aString
- <inlineJS: 'return aString < self'>
- ! !
- !String methodsFor: 'converting'!
- asJavaScriptMethodName
- <inlineJS: 'return $core.st2js(self)'>
- !
- asJavaScriptObject
- ^ self
- !
- asJavaScriptSource
- <inlineJS: '
- 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
- <inlineJS: 'return self.toLowerCase()'>
- !
- asMutator
- "Answer a setter selector. For example,
- #name asMutator returns #name:"
- self last = ':' ifFalse: [ ^ self, ':' ].
- ^ self
- !
- asNumber
- <inlineJS: 'return Number(self)'>
- !
- asRegexp
- ^ RegularExpression fromString: self
- !
- asString
- ^ self
- !
- asSymbol
- ^ self
- !
- asUppercase
- <inlineJS: 'return self.toUpperCase()'>
- !
- capitalized
- ^ self ifNotEmpty: [ self first asUppercase, self allButFirst ]
- !
- crlfSanitized
- ^ self lines join: String lf
- !
- escaped
- <inlineJS: 'return escape(self)'>
- !
- reversed
- <inlineJS: 'return self.split("").reverse().join("")'>
- !
- unescaped
- <inlineJS: 'return unescape(self)'>
- !
- uriComponentDecoded
- <inlineJS: 'return decodeURIComponent(self)'>
- !
- uriComponentEncoded
- <inlineJS: 'return encodeURIComponent(self)'>
- !
- uriDecoded
- <inlineJS: 'return decodeURI(self)'>
- !
- uriEncoded
- <inlineJS: 'return encodeURI(self)'>
- ! !
- !String methodsFor: 'copying'!
- , aString
- <inlineJS: 'return typeof aString === "string" ?
- String(self) + aString :
- $recv(aString)._appendToString_(String(self))'>
- !
- appendToString: aString
- <inlineJS: 'return aString + self'>
- !
- copyFrom: anIndex to: anotherIndex
- <inlineJS: 'return self.substring(anIndex - 1, anotherIndex)'>
- !
- copyWithFirst: anObject
- (anObject isString and: [ anObject size = 1 ]) "character is one-char string in JS"
- ifFalse: [ self error: 'Cannot put ', anObject class name, ' in a String' ].
- ^ anObject, self
- !
- deepCopy
- ^ self shallowCopy
- !
- shallowCopy
- ^ self
- ! !
- !String methodsFor: 'error handling'!
- errorReadOnly
- self error: 'Object is read-only'
- ! !
- !String methodsFor: 'evaluating'!
- value: anObject
- ^ anObject perform: self
- ! !
- !String methodsFor: 'printing'!
- asSymbolPrintOn: aStream
- aStream nextPutAll: '#'.
- self asString isSelector
- ifTrue: [ aStream nextPut: self ]
- ifFalse: [ self printOn: aStream ]
- !
- printNl
- <inlineJS: 'console.log(self)'>
- !
- printOn: aStream
- aStream
- nextPutAll: '''';
- nextPutAll: (self replace: '''' with: '''''');
- nextPutAll: ''''
- !
- shortenedPrintString
- ^ self printString size > 30
- ifTrue: [ (self printString copyFrom: 1 to: 30), '...''' ]
- ifFalse: [ self printString ]
- !
- symbolPrintString
- ^ String streamContents: [ :str | self asSymbolPrintOn: str ]
- ! !
- !String methodsFor: 'regular expressions'!
- match: aRegexp
- <inlineJS: 'return self.search(aRegexp) !!= -1'>
- !
- matchesOf: aRegularExpression
- <inlineJS: 'return self.match(aRegularExpression)'>
- !
- replace: aString with: anotherString
- ^ self replaceRegexp: (RegularExpression fromString: aString flag: 'g') with: anotherString
- !
- replaceRegexp: aRegexp with: aString
- <inlineJS: '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."
- <inlineJS: '
- var result = self.split(/\r\n|\r|\n/);
- if (!!result[result.length-1]) result.pop();
- return result;
- '>
- !
- 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 lines do: aBlock
- !
- subStrings: aString
- ^ self tokenize: aString
- !
- tokenize: aString
- <inlineJS: 'return self.split(aString)'>
- ! !
- !String methodsFor: 'streaming'!
- putOn: aStream
- aStream nextPutString: self
- ! !
- !String methodsFor: 'testing'!
- includesSubString: subString
- <inlineJS: 'return self.indexOf(subString) !!= -1'>
- !
- isCapitalized
- ^ self first asUppercase == self first
- !
- isImmutable
- ^ true
- !
- isSelector
- <inlineJS:
- 'return !!!!self.match(/^([a-zA-Z][a-zA-Z0-9]*|[\\+*/=><,@%~|&-]+|([a-zA-Z][a-zA-Z0-9]*\:)+)$/)'
- >
- !
- 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
- <inlineJS: 'return "\r"'>
- !
- crlf
- <inlineJS: 'return "\r\n"'>
- !
- esc
- ^ self fromCharCode: 27
- !
- lf
- <inlineJS: 'return "\n"'>
- !
- space
- <inlineJS: 'return " "'>
- !
- streamClass
- ^ StringStream
- !
- tab
- <inlineJS: 'return "\t"'>
- ! !
- !String class methodsFor: 'instance creation'!
- fromCharCode: anInteger
- <inlineJS: 'return String.fromCharCode(anInteger)'>
- !
- fromString: aString
- <inlineJS: 'return String(aString)'>
- !
- value: aUTFCharCode
- <inlineJS: 'return String.fromCharCode(aUTFCharCode);'>
- ! !
- !String class methodsFor: 'random'!
- random
- "Returns random alphanumeric string beginning with letter"
- <inlineJS: 'return ((10+22*Math.random())/32).toString(32).slice(2);'>
- !
- randomNotIn: aString
- | result |
- [ result := self random. aString includesSubString: result ] whileTrue.
- ^ result
- ! !
- Collection subclass: #Set
- slots: {#defaultBucket. #slowBucketStores. #fastBuckets. #size}
- package: 'Kernel-Collections'!
- !Set commentStamp!
- I represent an unordered set of objects without duplicates.
- ## Implementation notes
- I put elements into different stores based on their type.
- The goal is to store some elements into native JS object property names to be fast.
- If an unboxed element has typeof 'string', 'boolean' or 'number', or an element is nil, null or undefined,
- I store it as a property name in an empty (== Object.create(null)) JS object, different for each type
- (for simplicity, nil/null/undefined is treated as one and included with the two booleans).
- If element happen to be an object, I try to store them in `ArrayBucketStore`. I have two of them by default,
- one hashed using the Smalltalk class name, the other one using the JS constructor name. It is possible to have more or less
- instances of `ArrayBucketStores`, see `#initializeSlowBucketStores`.
- As a last resort, if none of the `ArrayBucketStore` instances can find a suitable bucket, the `defaultBucket` is used,
- which is an `Array`.!
- !Set methodsFor: 'accessing'!
- size
- ^ size
- ! !
- !Set methodsFor: 'adding/removing'!
- add: anObject
- | bucket |
- bucket := self bucketsOfElement: anObject.
- ^ bucket second
- ifNil: [
- | object slowBucket |
- object := bucket first.
- slowBucket := bucket third.
- slowBucket
- indexOf: object
- ifAbsent: [
- slowBucket add: object.
- size := size + 1 ].
- object ]
- ifNotNil: [ :primitiveBucket |
- self
- add: bucket first
- in: primitiveBucket ]
- !
- remove: anObject ifAbsent: aBlock
- | bucket |
- bucket := self bucketsOfElement: anObject.
- ^ bucket second
- ifNil: [ | obj | obj := bucket first. bucket third remove: obj ifAbsent: [ ^aBlock value ]. size := size - 1. obj ]
- ifNotNil: [ :primitiveBucket | self remove: bucket first in: primitiveBucket ifAbsent: aBlock ]
- !
- removeAll
- <inlineJS: '
- $self.fastBuckets = {
- "boolean": { store: Object.create(null), fn: function (x) { return {"true": true, "false": false, "null": null}[x]; } },
- "number": { store: Object.create(null), fn: Number },
- "string": { store: Object.create(null) }
- };
- $self.slowBucketStores.forEach(function (x) { x._removeAll(); });
- $self.defaultBucket._removeAll();
- $self.size = 0;
- '>
- ! !
- !Set methodsFor: 'comparing'!
- = aCollection
- ^ self class = aCollection class and: [
- self size = aCollection size and: [
- self allSatisfy: [ :each | aCollection includes: each ] ] ]
- ! !
- !Set methodsFor: 'enumerating'!
- collect: aBlock
- | collection |
- collection := self class new.
- self do: [ :each | collection add: (aBlock value: each) ].
- ^ collection
- !
- do: aBlock
- <inlineJS: '
- var el, keys, i;
- el = $self.fastBuckets;
- keys = Object.keys(el);
- for (i = 0; i < keys.length; ++i) {
- var fastBucket = el[keys[i]], fn = fastBucket.fn, store = Object.keys(fastBucket.store);
- if (fn) { for (var j = 0; j < store.length; ++j) { aBlock._value_(fn(store[j])); } }
- else { store._do_(aBlock); }
- }
- el = $self.slowBucketStores;
- for (i = 0; i < el.length; ++i) { el[i]._do_(aBlock); }
- $self.defaultBucket._do_(aBlock);
- '>
- !
- select: aBlock
- | collection |
- collection := self class new.
- self do: [ :each |
- (aBlock value: each) ifTrue: [
- collection add: each ] ].
- ^ collection
- !
- select: selectBlock thenCollect: collectBlock
- | collection |
- collection := self class new.
- self do: [ :each |
- (selectBlock value: each) ifTrue: [
- collection add: (collectBlock value: each) ] ].
- ^ collection
- ! !
- !Set methodsFor: 'initialization'!
- initialize
- super initialize.
-
- defaultBucket := #().
- self
- initializeSlowBucketStores;
- removeAll
- !
- initializeSlowBucketStores
- slowBucketStores := {
- ArrayBucketStore hashBlock: [ :x | self classNameOf: x ].
- ArrayBucketStore hashBlock: [ :x | self jsConstructorNameOf: x ]
- }
- ! !
- !Set methodsFor: 'printing'!
- printOn: aStream
- super printOn: aStream.
-
- aStream nextPutAll: ' ('.
- self
- do: [ :each | each printOn: aStream ]
- separatedBy: [ aStream nextPutAll: ' ' ].
- aStream nextPutAll: ')'
- ! !
- !Set methodsFor: 'private'!
- add: anObject in: anotherObject
- <inlineJS: '
- if (anObject in anotherObject.store) { return anObject; }
- $self.size++;
- anotherObject.store[anObject] = true;
- return anObject;
- '>
- !
- bucketsOfElement: anObject
- "Find the appropriate bucket for `anObject`.
- For optimization purposes, directly answer an array with:
- - the object to be store
- - the primitive bucket
- - the slow bucket"
-
- <inlineJS: '
- // include nil to well-known objects under "boolean" fastBucket
- if (anObject == null || anObject.a$nil) return [ null, $self.fastBuckets.boolean ];
-
- var prim = anObject.valueOf();
- if (typeof prim === "object" || typeof prim === "function" || !!$self.fastBuckets[typeof prim]) {
- var bucket = null;
- $self.slowBucketStores.some(function (store) {
- return bucket = store._bucketOfElement_(anObject);
- });
- return [ anObject, null, bucket || $self.defaultBucket ];
- }
- return [ prim, $self.fastBuckets[typeof prim] ];
- '>
- !
- classNameOf: anObject
- "Answer the class name of `anObject`, or `undefined`
- if `anObject` is not an Smalltalk object"
-
- <inlineJS: 'return anObject.a$cls !!= null && anObject.a$cls.name'>
- !
- includes: anObject in: anotherObject
- <inlineJS: 'return anObject in anotherObject.store'>
- !
- jsConstructorNameOf: anObject
- <inlineJS: 'return anObject.constructor && anObject.constructor.name'>
- !
- remove: anObject in: anotherObject ifAbsent: aBlock
- <inlineJS: '
- if (anObject in anotherObject.store) {
- delete anotherObject.store[anObject];
- $self.size--;
- return anObject;
- } else {
- return aBlock._value();
- }'>
- ! !
- !Set methodsFor: 'testing'!
- includes: anObject
- | bucket |
- bucket := self bucketsOfElement: anObject.
- ^ bucket second
- ifNil: [ bucket third includes: bucket first ]
- ifNotNil: [ :primitiveBucket | self includes: bucket first in: primitiveBucket ]
- ! !
- Object subclass: #ProtoStream
- slots: {}
- package: 'Kernel-Collections'!
- !ProtoStream commentStamp!
- I am the abstract base for different accessor for a sequence of objects. This sequence is referred to as my "contents".
- My instances are read/write streams modifying the contents.!
- !ProtoStream methodsFor: 'accessing'!
- contents
- self subclassResponsibility
- ! !
- !ProtoStream methodsFor: 'actions'!
- reset
- self subclassResponsibility
- !
- resetContents
- self subclassResponsibility
- ! !
- !ProtoStream methodsFor: 'enumerating'!
- do: aBlock
- [ self atEnd ] whileFalse: [ aBlock value: self next ]
- ! !
- !ProtoStream methodsFor: 'positioning'!
- setToEnd
- self subclassResponsibility
- !
- setToStart
- self reset
- ! !
- !ProtoStream methodsFor: 'reading'!
- next
- ^ self atEnd
- ifTrue: [ nil ]
- ifFalse: [ self subclassResponsibility ]
- !
- peek
- ^ self atEnd
- ifTrue: [ nil ]
- ifFalse: [ self subclassResponsibility ]
- ! !
- !ProtoStream methodsFor: 'testing'!
- atEnd
- self subclassResponsibility
- !
- atStart
- self subclassResponsibility
- !
- isEmpty
- ^ self atStart and: [ self atEnd ]
- ! !
- !ProtoStream methodsFor: 'writing'!
- << anObject
- self write: anObject
- !
- nextPut: anObject
- self subclassResponsibility
- !
- nextPutAll: aCollection
- aCollection do: [ :each |
- self nextPut: each ]
- !
- nextPutString: aString
- self nextPut: aString
- !
- write: anObject
- anObject putOn: self
- ! !
- !ProtoStream class methodsFor: 'instance creation'!
- on: aCollection
- ^ self new
- setCollection: aCollection;
- setStreamSize: aCollection size;
- yourself
- ! !
- ProtoStream subclass: #Stream
- slots: {#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: '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'!
- nextPut: anObject
- self position: self position + 1.
- self collection at: self position put: anObject.
- self setStreamSize: (self streamSize max: self position)
- ! !
- !Stream class methodsFor: 'instance creation'!
- on: aCollection
- ^ self new
- setCollection: aCollection;
- setStreamSize: aCollection size;
- yourself
- ! !
- Stream subclass: #StringStream
- slots: {}
- 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 position = self collection size 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
- !
- print: anObject
- anObject printOn: self
- !
- printSymbol: anObject
- anObject asSymbolPrintOn: self
- !
- space
- self nextPut: ' '
- !
- tab
- ^ self nextPutAll: String tab
- ! !
- Object subclass: #Queue
- slots: {#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 ifEmpty: [
- 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
- slots: {}
- package: 'Kernel-Collections'!
- !RegularExpression commentStamp!
- I represent a regular expression object. My instances are JavaScript `RegExp` object.!
- !RegularExpression methodsFor: 'evaluating'!
- compile: aString
- <inlineJS: 'return self.compile(aString)'>
- !
- exec: aString
- <inlineJS: 'return self.exec(aString) || nil'>
- !
- test: aString
- <inlineJS: 'return self.test(aString)'>
- ! !
- !RegularExpression class methodsFor: 'instance creation'!
- fromString: aString
- ^ self fromString: aString flag: ''
- !
- fromString: aString flag: anotherString
- <inlineJS: 'return new RegExp(aString, anotherString)'>
- ! !
- Trait named: #TKeyValueCollection
- package: 'Kernel-Collections'!
- !TKeyValueCollection 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
- ! !
- !TKeyValueCollection methodsFor: 'enumerating'!
- 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
- ! !
- Trait named: #TNativeZeroBasedCollection
- package: 'Kernel-Collections'!
- !TNativeZeroBasedCollection methodsFor: 'accessing'!
- at: anIndex ifAbsent: aBlock
- <inlineJS: '
- return anIndex >= 1 && anIndex <= self.length
- ? self[anIndex - 1]
- : aBlock._value()
- '>
- !
- at: anIndex ifPresent: aBlock ifAbsent: anotherBlock
- <inlineJS: '
- return anIndex >= 1 && anIndex <= self.length
- ? aBlock._value_(self[anIndex - 1])
- : anotherBlock._value()
- '>
- !
- indexOf: anObject ifAbsent: aBlock
- <inlineJS: '
- for(var i=0; i < self.length; i++) {
- if($recv(self[i]).__eq(anObject)) {return i+1}
- };
- return aBlock._value();
- '>
- !
- indexOf: anObject startingAt: start ifAbsent: aBlock
- <inlineJS: '
- for(var i=start - 1; i < self.length; i++){
- if($recv(self[i]).__eq(anObject)) {return i+1}
- }
- return aBlock._value();
- '>
- !
- single
- <inlineJS: '
- if (self.length == 0) throw new Error("Collection is empty");
- if (self.length > 1) throw new Error("Collection holds more than one element.");
- return self[0];
- '>
- !
- size
- <inlineJS: 'return self.length'>
- ! !
- !TNativeZeroBasedCollection methodsFor: 'enumerating'!
- detect: aBlock ifNone: anotherBlock
- <inlineJS: '
- for(var i = 0; i < self.length; i++)
- if(aBlock._value_(self[i]))
- return self[i];
- return anotherBlock._value();
- '>
- !
- do: aBlock
- <inlineJS: '
- for(var i=0; i < self.length; i++) {
- aBlock._value_(self[i]);
- }
- '>
- !
- with: anotherCollection do: aBlock
- <inlineJS: '
- $recv(anotherCollection)._first_(0); // #guardSequenceableCollection
- for(var i=0; i<self.length; i++) {
- aBlock._value_value_(self[i], anotherCollection[i]);
- }
- '>
- !
- withIndexDo: aBlock
- <inlineJS: '
- for(var i=0; i < self.length; i++) {
- aBlock._value_value_(self[i], i+1);
- }
- '>
- ! !
- AssociativeCollection setTraitComposition: {TKeyValueCollection} asTraitComposition!
- SequenceableCollection setTraitComposition: {TKeyValueCollection} asTraitComposition!
- Array setTraitComposition: {TNativeZeroBasedCollection} asTraitComposition!
- String setTraitComposition: {TNativeZeroBasedCollection} asTraitComposition!
- ! !
|