| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464 | Smalltalk current createPackage: 'Helios-Browser'!HLWidget subclass: #HLBrowser	instanceVariableNames: 'model packagesListWidget classesListWidget protocolsListWidget methodsListWidget sourceWidget bottomDiv'	package: 'Helios-Browser'!!HLBrowser methodsFor: 'accessing'!environment	^ self model environment!model	^ model ifNil: [ model := HLBrowserModel new ]!model: aModel	model := aModel! !!HLBrowser methodsFor: 'actions'!focus	^ self packagesListWidget focus! !!HLBrowser methodsFor: 'keybindings'!registerBindingsOn: aBindingGroup	HLBrowserCommand 		registerConcreteClassesOn: aBindingGroup 		for: self model! !!HLBrowser methodsFor: 'rendering'!renderContentOn: html	html with: (HLContainer with: (HLHorizontalSplitter     	with: (HLVerticalSplitter        	with: (HLVerticalSplitter            	with: self packagesListWidget                with: self classesListWidget)            with: (HLVerticalSplitter            	with: self protocolsListWidget                with: self methodsListWidget))         with: self sourceWidget)).		self packagesListWidget focus! !!HLBrowser methodsFor: 'testing'!canHaveFocus	^ true! !!HLBrowser methodsFor: 'widgets'!classesListWidget	^ classesListWidget ifNil: [      	classesListWidget := HLClassesListWidget on: self model.		classesListWidget next: self protocolsListWidget ]!methodsListWidget	^ methodsListWidget ifNil: [      	methodsListWidget := HLMethodsListWidget on: self model.		methodsListWidget next: self sourceWidget ]!packagesListWidget	^ packagesListWidget ifNil: [      	packagesListWidget := HLPackagesListWidget on: self model.		packagesListWidget next: self classesListWidget ]!protocolsListWidget	^ protocolsListWidget ifNil: [      	protocolsListWidget := HLProtocolsListWidget on: self model.		protocolsListWidget next: self methodsListWidget ]!sourceWidget	^ sourceWidget ifNil: [      	sourceWidget := HLBrowserCodeWidget new			browserModel: self model;			yourself ]! !HLBrowser class instanceVariableNames: 'nextId'!!HLBrowser class methodsFor: 'accessing'!nextId	nextId ifNil: [ nextId := 0 ].    ^ 'browser_', (nextId + 1) asString!tabLabel	^ 'Browser'!tabPriority	^ 0! !!HLBrowser class methodsFor: 'testing'!canBeOpenAsTab	^ true! !HLNavigationListWidget subclass: #HLBrowserListWidget	instanceVariableNames: 'model'	package: 'Helios-Browser'!!HLBrowserListWidget methodsFor: 'accessing'!commandCategory	^ self label!label	^ 'List'!menuCommands	"Answer a collection of commands to be put in the cog menu"		^ (HLBrowserCommand concreteClasses 		collect: [ :each | each for: self model ])		select: [ :each | 			each category = self commandCategory and: [ 				each isAction and: [ each isActive ] ] ]!model	^ model!model: aBrowserModel	model := aBrowserModel.        self 		observeSystem;		observeModel!selectedItem: anItem	"Selection changed, update the cog menu"		super selectedItem: anItem.	self updateMenu! !!HLBrowserListWidget methodsFor: 'actions'!activateListItem: anItem	self model withChangesDo: [ super activateListItem: anItem ]!observeModel!observeSystem! !!HLBrowserListWidget methodsFor: 'rendering'!renderContentOn: html	self renderHeadOn: html.		super renderContentOn: html!renderHeadOn: html	html div 		class: 'list-label';		with: [			html with: self label.			self renderMenuOn: html ]!renderMenuOn: html	| commands |		commands := self menuCommands.	commands isEmpty ifTrue: [ ^ self ].		html div 		class: 'btn-group cog';		with: [			html a				class: 'btn dropdown-toggle';				at: 'data-toggle' put: 'dropdown';				with: [ (html tag: 'i') class: 'icon-cog' ].		html ul 			class: 'dropdown-menu pull-right';			with: [ 				self menuCommands do: [ :each | 					html li with: [ html a 						with: each menuLabel;						onClick: [ self execute: each ] ] ] ] ]! !!HLBrowserListWidget methodsFor: 'updating'!updateMenu	(self wrapper asJQuery find: '.cog') remove.		[ :html | self renderMenuOn: html ] 		appendToJQuery: (self wrapper asJQuery find: '.list-label')! !!HLBrowserListWidget class methodsFor: 'instance creation'!on: aModel	^ self new     	model: aModel;        yourself! !HLBrowserListWidget subclass: #HLClassesListWidget	instanceVariableNames: ''	package: 'Helios-Browser'!!HLClassesListWidget methodsFor: 'accessing'!getChildrenOf: aClass	^ self items select: [ :each | each superclass = aClass ]!getRootClassesOf: aCollection	^ aCollection select: [ :each |    		(aCollection includes: each superclass) not ]!iconForItem: aClass	^ aClass theNonMetaClass comment isEmpty    	ifFalse: [ 'icon-none' ]      	ifTrue: [ 'icon-question-sign' ]!label	^ 'Classes'!showClass	^ self model showInstance not and: [		self model showComment not ]!showComment	^ self model showComment!showInstance	^ self model showInstance and: [		self model showComment not ]! !!HLClassesListWidget methodsFor: 'actions'!focusMethodsListWidget	self model announcer announce: HLMethodsListFocus new!focusProtocolsListWidget	self model announcer announce: HLProtocolsListFocus new!observeModel	self model announcer     	on: HLPackageSelected do: [ :ann | self onPackageSelected: ann item ];    	on: HLShowInstanceToggled do: [ :ann | self onShowInstanceToggled ];		on: HLClassSelected do: [ :ann | self onClassSelected: ann item ];		on: HLClassesFocusRequested do: [ :ann | self onClassesFocusRequested ]!observeSystem	self model systemAnnouncer    	on: ClassAdded        do: [ :ann | self onClassAdded: ann theClass ];        on: ClassRemoved        do: [ :ann | self onClassRemoved: ann theClass ];		on: ClassMoved        do: [ :ann | self onClassMoved: ann theClass from: ann oldPackage ];		on: ClassRenamed        do: [ :ann | self onClassRenamed: ann theClass ]!selectItem: aClass    self model selectedClass: aClass!showComment: aBoolean	self model showComment: aBoolean!showInstance: aBoolean	self model showInstance: aBoolean! !!HLClassesListWidget methodsFor: 'private'!setItemsForPackage: aPackage	self items: (aPackage     	ifNil: [ #() ]  		ifNotNil: [ ((aPackage classes         	collect: [ :each | each theNonMetaClass ]) asSet asArray)             	sort: [:a :b | a name < b name ] ]).!setItemsForSelectedPackage	self setItemsForPackage: self model selectedPackage! !!HLClassesListWidget methodsFor: 'reactions'!onClassAdded: aClass	(aClass package = self model selectedPackage or: [		self items includes: aClass ]) ifFalse: [ ^ self ].        self setItemsForSelectedPackage.    self refresh!onClassMoved: aClass from: aPackage	(aPackage = self model selectedPackage or: [		aClass package = self model selectedPackage ])			ifFalse: [ ^ self ].		aPackage = self model selectedPackage ifTrue: [ 		self 			selectedItem: nil;			selectItem: nil ].        self setItemsForSelectedPackage.    self refresh!onClassRemoved: aClass	aClass package = self model selectedPackage ifFalse: [ ^ self ].    aClass = self model selectedClass ifTrue: [ self selectItem: nil ].        self setItemsForSelectedPackage.    self refresh!onClassRenamed: aClass	aClass package = self model selectedPackage ifFalse: [ ^ self ].        self setItemsForSelectedPackage.    self refresh!onClassSelected: aClass	| selectedClass |		aClass ifNil: [ ^ self ].		selectedClass := aClass theNonMetaClass.	self selectedItem: selectedClass.	self hasFocus ifFalse: [		self 			activateItem: selectedClass;			focus ]!onClassesFocusRequested	self focus!onPackageSelected: aPackage    self selectedItem: nil.        self setItemsForSelectedPackage.    self refresh!onShowInstanceToggled	self refresh! !!HLClassesListWidget methodsFor: 'rendering'!renderButtonsOn: html	html div         class: 'btn-group';		with: [            	html button                 class: (String streamContents: [ :str |                	str nextPutAll: 'btn'.                    self showInstance ifTrue: [                     	str nextPutAll: ' active' ] ]);  				with: 'Instance';                onClick: [ self showInstance: true ].  			html button  				class: (String streamContents: [ :str |                	str nextPutAll: 'btn'.                    self showClass ifTrue: [                     	str nextPutAll: ' active' ] ]);  				with: 'Class';				onClick: [ self showInstance: false ].			html button  				class: (String streamContents: [ :str |                	str nextPutAll: 'btn'.                    self showComment ifTrue: [                     	str nextPutAll: ' active' ] ]);  				with: 'Doc';				onClick: [ self showComment: true ] ]!renderItem: aClass level: anInteger on: html	| li |    	li := html li.	self registerMappingFrom: aClass to: li.	    li    	at: 'list-data' put: (self items indexOf: aClass);    	class: (self cssClassForItem: aClass);        with: [         	html a            	with: [             		(html tag: 'i') class: (self iconForItem: aClass).  					self renderItemLabel: aClass level: anInteger on: html ];				onClick: [                  	self activateListItem: li asJQuery ] ].                        (self getChildrenOf: aClass) do: [ :each |    	self renderItem: each level: anInteger + 1 on: html ]!renderItem: aClass on: html	super renderItem: aClass on: html.    (self getChildrenOf: aClass) do: [ :each |    	self renderItem: each level: 1 on: html ]!renderItemLabel: aClass level: anInteger on: html	html span asJQuery html: (String streamContents: [ :str |		anInteger timesRepeat: [			str nextPutAll: '    '].			str nextPutAll: aClass name ])!renderItemLabel: aClass on: html	self renderItemLabel: aClass level: 0 on: html!renderListOn: html	(self getRootClassesOf: self items)    	do: [ :each | self renderItem: each on: html ]! !HLBrowserListWidget subclass: #HLMethodsListWidget	instanceVariableNames: 'selectorsCache'	package: 'Helios-Browser'!!HLMethodsListWidget methodsFor: 'accessing'!allProtocol	^ self model allProtocol!iconForItem: aSelector	| override overriden method |        method := self methodForSelector: aSelector.    override := self isOverride: method.    overriden := self isOverridden: method.    	^ override    	ifTrue: [ overriden			ifTrue: [ 'icon-resize-vertical' ]			ifFalse: [ 'icon-arrow-up' ] ]		ifFalse: [			overriden			ifTrue: [ 'icon-arrow-down' ]			ifFalse: [ 'icon-none' ] ]!label	^ 'Methods'!methodForSelector: aSelector	^ self model selectedClass    	methodDictionary at: aSelector!methodsInProtocol: aString	self model selectedClass ifNil: [ ^ #() ].    	^ aString = self allProtocol    	ifTrue: [ self model selectedClass methods ]      	ifFalse: [ self model selectedClass methodsInProtocol: aString ]!overrideSelectors	^ self selectorsCache     	at: 'override'        ifAbsentPut: [         	self model selectedClass allSuperclasses				inject: Set new into: [ :acc :each | acc addAll: each selectors; yourself ] ]!overridenSelectors	^ self selectorsCache     	at: 'overriden'        ifAbsentPut: [         	self model selectedClass allSubclasses				inject: Set new into: [ :acc :each | acc addAll: each selectors; yourself ] ]!selectorsCache	^ self class selectorsCache!selectorsInProtocol: aString	^ ((self methodsInProtocol: aString)    	collect: [ :each | each selector ]) sorted! !!HLMethodsListWidget methodsFor: 'actions'!observeModel	self model announcer 		on: HLProtocolSelected 		do: [ :ann | self onProtocolSelected: ann item ];		on: HLShowInstanceToggled 		do: [ :ann | self onProtocolSelected: nil ];		on: HLMethodSelected 		do: [ :ann | self onMethodSelected: ann item ];		on: HLMethodsFocusRequested 		do: [ :ann | self onMethodsFocusRequested ]!observeSystem	self model systemAnnouncer     	on: ProtocolAdded        do: [ :ann | self onProtocolAdded: ann theClass ];    	on: ProtocolRemoved        do: [ :ann | self onProtocolRemoved: ann theClass ];    	on: MethodAdded         do: [ :ann | self onMethodAdded: ann method ];        on: MethodRemoved         do: [ :ann | self onMethodRemoved: ann method ];		on: MethodMoved         do: [ :ann | self onMethodMoved: ann method ]!selectItem: aSelector	aSelector ifNil: [ ^ self model selectedMethod: nil ].   	self model selectedMethod: (self methodForSelector: aSelector)! !!HLMethodsListWidget methodsFor: 'private'!setItemsForProtocol: aString	^ self items: (aString    	ifNil: [ #() ]      	ifNotNil: [ self selectorsInProtocol: aString ])!setItemsForSelectedProtocol	self setItemsForProtocol: self model selectedProtocol! !!HLMethodsListWidget methodsFor: 'reactions'!onMethodAdded: aMethod	self model selectedClass = aMethod methodClass ifFalse: [ ^ self ].        self setItemsForSelectedProtocol.    self refresh!onMethodMoved: aMethod	self model selectedMethod = aMethod ifFalse: [ ^ self ].    	self model selectedProtocol = self model allProtocol ifFalse: [		self 			selectedItem: nil; 			selectItem: nil;			setItemsForSelectedProtocol;    		refresh ]!onMethodRemoved: aMethod	self items detect: [ :each | each = aMethod selector ] ifNone: [ ^ self ].    self selectedItem ifNotNil: [      	(aMethod methodClass = self model selectedClass and: [ aMethod selector = self selectedItem ])  			ifTrue: [ 				self selectedItem: nil; 				selectItem: nil ] ].    self setItemsForSelectedProtocol.	self refresh!onMethodSelected: aMethod	| selector |	selector := aMethod isCompiledMethod 		ifTrue: [ aMethod selector ]		ifFalse: [ nil ].			self 		selectedItem: selector;		activateItem: selector!onMethodsFocusRequested	self focus!onProtocolAdded: aClass	self model selectedClass = aClass ifFalse: [ ^ self ].		self setItemsForSelectedProtocol.    self refresh.	self focus!onProtocolRemoved: aClass	self model selectedClass = aClass ifFalse: [ ^ self ].		self setItemsForSelectedProtocol.    self refresh.	self focus!onProtocolSelected: aString    self selectedItem: nil.    	self setItemsForSelectedProtocol.    self refresh! !!HLMethodsListWidget methodsFor: 'rendering'!renderContentOn: html	self model showInstance    	ifFalse: [ html div         	class: 'class_side';             with: [ super renderContentOn: html ] ]      	ifTrue: [ super renderContentOn: html ]!renderItemLabel: aSelector on: html	html with: aSelector! !!HLMethodsListWidget methodsFor: 'testing'!isOverridden: aMethod   ^ self selectorsCache isOverridden: aMethod!isOverride: aMethod   ^ self selectorsCache isOverride: aMethod! !HLMethodsListWidget class instanceVariableNames: 'selectorsCache'!!HLMethodsListWidget class methodsFor: 'accessing'!selectorsCache	^ HLSelectorsCache current! !HLBrowserListWidget subclass: #HLPackagesListWidget	instanceVariableNames: ''	package: 'Helios-Browser'!!HLPackagesListWidget methodsFor: 'accessing'!items	^ items ifNil: [self initializeItems]!label	^ 'Packages'! !!HLPackagesListWidget methodsFor: 'actions'!commitPackage	self model commitPackage!focusClassesListWidget	self model announcer announce: HLClassesListFocus new!observeModel    self model announcer 		on: HLPackageSelected 		do: [ :ann | self onPackageSelected: ann item ];		on: HLPackagesFocusRequested 		do: [ :ann | self onPackagesFocusRequested ]!observeSystem    self model systemAnnouncer 		on: ClassAdded 		do: [ :ann | self onClassAdded: ann theClass ]!selectItem: aPackage	self model selectedPackage: aPackage! !!HLPackagesListWidget methodsFor: 'initialization'!initializeItems	^ items := self model packages 		sort: [ :a :b | a name < b name ]! !!HLPackagesListWidget methodsFor: 'reactions'!onClassAdded: aClass	"Amber doesn't have yet a global organizer for packages"		(self items includes: aClass package) ifFalse: [ 		self 			initializeItems;			refresh ]!onPackageSelected: aPackage	self selectedItem: aPackage.	self hasFocus ifFalse: [		self			activateItem: aPackage;			focus ]!onPackagesFocusRequested	self focus! !!HLPackagesListWidget methodsFor: 'rendering'!renderButtonsOn: html	html div 		class: 'buttons';		with: [			html button 				class: 'btn';				with: 'Commit';				onClick: [ self commitPackage ] ]!renderItemLabel: aPackage on: html	html with: aPackage name! !HLBrowserListWidget subclass: #HLProtocolsListWidget	instanceVariableNames: ''	package: 'Helios-Browser'!!HLProtocolsListWidget methodsFor: 'accessing'!allProtocol	^ self model allProtocol!label	^ 'Protocols'!selectedItem	^ super selectedItem" ifNil: [ self allProtocol ]"! !!HLProtocolsListWidget methodsFor: 'actions'!observeModel    self model announcer 		on: HLClassSelected 		do: [ :ann | self onClassSelected: ann item ];    	on: HLShowInstanceToggled 		do: [ :ann | self onClassSelected: self model selectedClass ];    	on: HLProtocolSelected		do: [ :ann | self onProtocolSelected: ann item ];		on: HLProtocolsFocusRequested 		do: [ :ann | self onProtocolsFocusRequested ]!observeSystem	self model systemAnnouncer		on: ProtocolAdded 	    do: [ :ann | self onProtocolAdded: ann protocol to: ann theClass ];	    on: ProtocolRemoved	    do: [ :ann | self onProtocolRemoved: ann protocol from: ann theClass ]!selectItem: aString    self model selectedProtocol: aString! !!HLProtocolsListWidget methodsFor: 'private'!setItemsForClass: aClass	self items: (aClass    	ifNil: [ Array with: self allProtocol ]      	ifNotNil: [         	(Array with: self allProtocol)             	addAll: aClass protocols;                 yourself ])!setItemsForSelectedClass	self setItemsForClass: self model selectedClass! !!HLProtocolsListWidget methodsFor: 'reactions'!onClassSelected: aClass    self selectedItem: nil.        self setItemsForSelectedClass.    self refresh!onProtocolAdded: aString to: aClass	aClass = self model selectedClass ifFalse: [ ^ self ].        self setItemsForSelectedClass.    self refresh!onProtocolRemoved: aString from: aClass	aClass = self model selectedClass ifFalse: [ ^ self ].        self model selectedProtocol = aString     	ifTrue: [ 			self 				selectedItem: nil;				selectItem: nil ].            self setItemsForSelectedClass.    self refresh!onProtocolSelected: aString	self selectedItem: aString.	aString ifNil: [ ^ self ].    	self hasFocus ifFalse: [		self 			activateItem: aString;			focus ]!onProtocolsFocusRequested	self focus! !!HLProtocolsListWidget methodsFor: 'rendering'!renderContentOn: html	self model showInstance    	ifFalse: [ html div         	class: 'class_side';             with: [ super renderContentOn: html ] ]      	ifTrue: [ super renderContentOn: html ]! !HLModel subclass: #HLBrowserModel	instanceVariableNames: 'selectedPackage selectedClass selectedProtocol selectedSelector showInstance showComment'	package: 'Helios-Browser'!!HLBrowserModel methodsFor: 'accessing'!allSelectors	^ self environment allSelectors!availableClassNames	^ self environment availableClassNames!availablePackageNames	^ self environment availablePackageNames!availablePackages	^ self environment availablePackageNames!availableProtocols	^ self environment availableProtocolsFor: self selectedClass!handleUnkownVariableError: anError	self announcer announce: (HLUnknownVariableErrorRaised new		error: anError;		yourself)!packages	^ self environment packages!selectedClass	^ selectedClass!selectedClass: aClass	aClass ifNil: [ ^ self ].		self withChangesDo: [		selectedClass = aClass ifTrue: [ 			self selectedProtocol: nil ].    		aClass    			ifNil: [ selectedClass := nil ]    		ifNotNil: [				self showInstance    					ifTrue: [ selectedClass := aClass theNonMetaClass ]     				ifFalse: [ selectedClass := aClass theMetaClass ] ].		self selectedProtocol: nil.		self announcer announce: (HLClassSelected on: self selectedClass) ]!selectedMethod	^ self selectedClass ifNotNil: [ 			self selectedClass methodDictionary 				at: selectedSelector 				ifAbsent: [ nil ] ]!selectedMethod: aCompiledMethod	selectedSelector = aCompiledMethod ifTrue: [ ^ self ].        self withChangesDo: [		aCompiledMethod    		ifNil: [ selectedSelector := nil ]      		ifNotNil: [				selectedSelector := aCompiledMethod selector ].		self announcer announce: (HLMethodSelected on: aCompiledMethod) ]!selectedPackage	^ selectedPackage!selectedPackage: aPackage	selectedPackage = aPackage ifTrue: [ ^ self ].    	self withChangesDo: [		selectedPackage := aPackage.		self selectedClass: nil.		self announcer announce: (HLPackageSelected on: aPackage) ]!selectedProtocol	^ selectedProtocol!selectedProtocol: aString	selectedProtocol = aString ifTrue: [ ^ self ].	self withChangesDo: [		selectedProtocol := aString.		self selectedMethod: nil.		self announcer announce: (HLProtocolSelected on: aString) ]!showComment	^ showComment ifNil: [ false ]!showComment: aBoolean	self withChangesDo: [		showComment := aBoolean.		self announcer announce: HLShowCommentToggled new ]!showInstance	^ showInstance ifNil: [ true ]!showInstance: aBoolean	self withChangesDo: [		showInstance := aBoolean.		showComment := false.    	self selectedClass ifNotNil: [    		self selectedClass: (aBoolean    			ifTrue: [self selectedClass theNonMetaClass ]	    	  	ifFalse: [ self selectedClass theMetaClass ]) ].    		self announcer announce: HLShowInstanceToggled new ]! !!HLBrowserModel methodsFor: 'actions'!addInstVarNamed: aString	self environment addInstVarNamed: aString to: self selectedClass.	self announcer announce: (HLInstVarAdded new		theClass: self selectedClass;		variableName: aString;		yourself)!focusOnClasses	self announcer announce: HLClassesFocusRequested new!focusOnMethods	self announcer announce: HLMethodsFocusRequested new!focusOnPackages	self announcer announce: HLPackagesFocusRequested new!focusOnProtocols	self announcer announce: HLProtocolsFocusRequested new!focusOnSourceCode	self announcer announce: HLSourceCodeFocusRequested new!save: aString	self announcer announce: HLSourceCodeSaved new.		(self shouldCompileClassDefinition: aString)		ifTrue: [ self compileClassDefinition: aString ]		ifFalse: [ self compileMethod: aString ]!saveSourceCode	self announcer announce: HLSaveSourceCode new! !!HLBrowserModel methodsFor: 'commands actions'!commitPackage	self 		withHelperLabelled: 'Committing package ', self selectedPackage name, '...'		do: [ self environment commitPackage: self selectedPackage ]!copyClassTo: aClassName	self withChangesDo: [ 		self environment 			copyClass: self selectedClass theNonMetaClass			to: aClassName ]!moveClassToPackage: aPackageName	self withChangesDo: [		self environment 			moveClass: self selectedClass theNonMetaClass			toPackage: aPackageName ]!moveMethodToClass: aClassName	self withChangesDo: [		self environment 			moveMethod: self selectedMethod 			toClass: aClassName ]!moveMethodToProtocol: aProtocol	self withChangesDo: [		self environment 			moveMethod: self selectedMethod 			toProtocol: aProtocol ]!openClassNamed: aString	| class |		self withChangesDo: [		class := self environment classNamed: aString.		self selectedPackage: class package.		self selectedClass: class ]!removeClass	self withChangesDo: [		(self manager confirm: 'Do you REALLY want to remove class ', self selectedClass name)			ifTrue: [ self environment removeClass: self selectedClass ] ]!removeMethod	self withChangesDo: [		(self manager confirm: 'Do you REALLY want to remove method ', self selectedMethod methodClass name,' >> #', self selectedMethod selector)			ifTrue: [ self environment removeMethod: self selectedMethod ] ]!removeProtocol	self withChangesDo: [		(self manager confirm: 'Do you REALLY want to remove protocol ', self selectedProtocol)			ifTrue: [ self environment 				removeProtocol: self selectedProtocol 				from: self selectedClass ] ]!renameClassTo: aClassName	self withChangesDo: [		self environment 			renameClass: self selectedClass theNonMetaClass			to: aClassName ]!renameProtocolTo: aString	self withChangesDo: [		self environment 			renameProtocol: self selectedProtocol			to: aString			in: self selectedClass ]! !!HLBrowserModel methodsFor: 'compiling'!compileClassComment: aString	self environment 		compileClassComment: aString 		for: self selectedClass!compileClassDefinition: aString	self environment compileClassDefinition: aString!compileMethod: aString	| method |		self withCompileErrorHandling: [ 		method := self environment 			compileMethod: aString 			for: self selectedClass			protocol: self compilationProtocol.		self selectedMethod: method ]! !!HLBrowserModel methodsFor: 'defaults'!allProtocol	^ '-- all --'!unclassifiedProtocol	^ 'as yet unclassified'! !!HLBrowserModel methodsFor: 'error handling'!handleCompileError: anError	self announcer announce: (HLCompileErrorRaised new		error: anError;		yourself)!handleParseError: anError	| split line column messageToInsert |		split := anError messageText tokenize: ' : '.	messageToInsert := split second.	"21 = 'Parse error on line ' size + 1"	split := split first copyFrom: 21 to: split first size.		split := split tokenize: ' column '.	line := split first.	column := split second.		self announcer announce: (HLParseErrorRaised new		line: line asNumber;		column: column asNumber;		message: messageToInsert;		error: anError;		yourself)!withChangesDo: aBlock	[ 		self announcer announce: HLAboutToChange new.		aBlock value	]		on: HLChangeForbidden 		do: [ :ex | ]!withCompileErrorHandling: aBlock	[		[			aBlock 				on: ParseError				do: [:ex | self handleParseError: ex ]		]			on: UnknownVariableError			do: [ :ex | self handleUnkownVariableError: ex ]	]		on: CompilerError		do: [ :ex | self handleCompileError: ex ]! !!HLBrowserModel methodsFor: 'private'!compilationProtocol	| currentProtocol |		currentProtocol := self selectedProtocol.	currentProtocol ifNil: [ currentProtocol := self unclassifiedProtocol ].	self selectedMethod ifNotNil: [ currentProtocol := self selectedMethod protocol ].	^ currentProtocol = self allProtocol		ifTrue: [ self unclassifiedProtocol ]		ifFalse: [ currentProtocol ]!withHelperLabelled: aString do: aBlock	"TODO: doesn't belong here"	(window jQuery: '#helper') remove.	[ :html |		html div 			id: 'helper';			with: aString ] appendToJQuery: 'body' asJQuery.		[		aBlock value.		(window jQuery: '#helper') remove	] 		valueWithTimeout: 10! !!HLBrowserModel methodsFor: 'testing'!shouldCompileClassDefinition: aString	^ self selectedClass isNil or: [		aString first asUppercase = aString first ]! !!HLBrowserModel class methodsFor: 'actions'!on: anEnvironment	^ self new    	environment: anEnvironment;        yourself! !Object subclass: #HLClassCache	instanceVariableNames: 'class selectorsCache overrideCache overriddenCache'	package: 'Helios-Browser'!!HLClassCache methodsFor: 'accessing'!overriddenCache	^ overriddenCache ifNil: [ overriddenCache := HashedCollection new ]!overrideCache	^ overrideCache ifNil: [ overrideCache := HashedCollection new ]!selectorsCache	^ selectorsCache!selectorsCache: aCache	selectorsCache := aCache!theClass	^ class!theClass: aClass	class := aClass! !!HLClassCache methodsFor: 'actions'!invalidateChildrenSelector: aSelector	self theClass subclasses do: [ :each |    	(self selectorsCache cacheFor: each)        	removeSelector: aSelector;        	invalidateChildrenSelector: aSelector ]!invalidateParentSelector: aSelector	self theClass superclass ifNotNil: [    	(self selectorsCache cacheFor: self theClass superclass)        	removeSelector: aSelector;			invalidateParentSelector: aSelector ]!invalidateSelector: aSelector	self     	invalidateParentSelector: aSelector;        invalidateChildrenSelector: aSelector;        removeSelector: aSelector! !!HLClassCache methodsFor: 'private'!removeSelector: aSelector	self overriddenCache     	removeKey: aSelector        ifAbsent: [ ].    self overrideCache     	removeKey: aSelector        ifAbsent: [ ]! !!HLClassCache methodsFor: 'testing'!isOverridden: aMethod	^ self overriddenCache     	at: aMethod selector      	ifAbsentPut: [ aMethod isOverridden ]!isOverride: aMethod	^ self overrideCache    	at: aMethod selector      	ifAbsentPut: [ aMethod isOverride ]! !!HLClassCache class methodsFor: 'instance creation'!on: aClass selectorsCache: aSelectorsCache	^ self new    	theClass: aClass;        selectorsCache: aSelectorsCache;        yourself! !HLFocusableWidget subclass: #HLDocumentationWidget	instanceVariableNames: 'documentation'	package: 'Helios-Browser'!!HLDocumentationWidget methodsFor: 'accessing'!documentation	^ documentation ifNil: [ self defaultDocumentation ]!documentation: aString	documentation := aString! !!HLDocumentationWidget methodsFor: 'defaults'!defaultDocumentation	^ '#No documentation available. ##That''s bad. Seriously.'! !!HLDocumentationWidget methodsFor: 'rendering'!renderContentOn: html	html div 		class: 'markdown';		with: (Showdown makeHtml: self documentation)! !Object subclass: #HLSelectorsCache	instanceVariableNames: 'classesCache'	package: 'Helios-Browser'!!HLSelectorsCache methodsFor: 'accessing'!cacheFor: aClass	aClass ifNil: [ ^ nil ].    	^ self classesCache    	at: aClass name        ifAbsentPut: [ self newCacheFor: aClass ]!classesCache	^ classesCache ifNil: [ classesCache := HashedCollection new ]! !!HLSelectorsCache methodsFor: 'actions'!observeSystem	SystemAnnouncer current		on: MethodAdded        do: [ :ann | self onMethodAdded: ann method ];        on: MethodRemoved        do: [ :ann | self onMethodRemoved: ann method ]! !!HLSelectorsCache methodsFor: 'factory'!newCacheFor: aClass	^ HLClassCache     	on: aClass        selectorsCache: self! !!HLSelectorsCache methodsFor: 'initialization'!initialize	super initialize.    self observeSystem! !!HLSelectorsCache methodsFor: 'private'!invalidateCacheFor: aMethod	(self cacheFor: aMethod methodClass)    	invalidateSelector: aMethod selector! !!HLSelectorsCache methodsFor: 'reactions'!onMethodAdded: aMethod	self invalidateCacheFor: aMethod!onMethodRemoved: aMethod	self invalidateCacheFor: aMethod! !!HLSelectorsCache methodsFor: 'testing'!isOverridden: aMethod	^ (self cacheFor: aMethod methodClass)    	isOverridden: aMethod!isOverride: aMethod	^ (self cacheFor: aMethod methodClass)    	isOverride: aMethod! !HLSelectorsCache class instanceVariableNames: 'current'!!HLSelectorsCache class methodsFor: 'accessing'!current	^ current ifNil: [ current := super new ]!flush	current := nil! !!HLSelectorsCache class methodsFor: 'instance creation'!new	self shouldNotImplement! !
 |