| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751 | Widget subclass: #TabManager	instanceVariableNames: 'selectedTab tabs opened ul'	category: 'IDE'!!TabManager methodsFor: 'accessing'!tabs    ^tabs ifNil: [tabs := Array new]! !!TabManager methodsFor: 'actions'!updateBodyMargin    self setBodyMargin: '#jtalk' asJQuery height + 27!updatePosition    <jQuery('#jtalk').css('top', '').css('bottom', '27px')>!removeBodyMargin    self setBodyMargin: 0!setBodyMargin: anInteger    '.jtalkBody' asJQuery cssAt: 'margin-bottom' put: anInteger asString, 'px'!onResize: aBlock    <jQuery('#jtalk').resizable({	handles: 'n', 	resize: aBlock,	minHeight: 230})>!onWindowResize: aBlock    <jQuery(window).resize(aBlock)>!open    opened ifFalse: [	'body' asJQuery addClass: 'jtalkBody'.	'#jtalk' asJQuery show.	ul asJQuery show.	self updateBodyMargin.	selectedTab show.	opened := true]!close    opened ifTrue: [	'#jtalk' asJQuery hide.	ul asJQuery hide.	selectedTab hide.	self removeBodyMargin.	'body' asJQuery removeClass: 'jtalkBody'.	opened := false]!newBrowserTab    Browser open!selectTab: aWidget    self open.    selectedTab := aWidget.    self tabs do: [:each |	each hide].    aWidget show.	    self update!closeTab: aWidget    self removeTab: aWidget.    self selectTab: self tabs last.    aWidget remove.    self update! !!TabManager methodsFor: 'adding/Removing'!addTab: aWidget    self tabs add: aWidget.    '#jtalk' asJQuery append: aWidget.    aWidget hide!removeTab: aWidget    self tabs remove: aWidget.    self update! !!TabManager methodsFor: 'initialization'!initialize    super initialize.    opened := true.    'body' asJQuery 	append: self;	append: [:html | html div id: 'jtalk'];	addClass: 'jtalkBody'.    self 	addTab: Transcript current;	addTab: Workspace new.    self selectTab: self tabs last.    self 	onResize: [self updateBodyMargin; updatePosition];	onWindowResize: [self updatePosition]! !!TabManager methodsFor: 'rendering'!renderOn: html	ul := html ul		id: 'jtalkTabs';		yourself.	self renderTabs!renderTabFor: aWidget on: html    | li |    li := html li.    selectedTab = aWidget ifTrue: [	li class: 'selected'].    li with: [        aWidget canBeClosed ifTrue: [	    html span 		class: 'close';		with: 'x';		onClick: [self closeTab: aWidget]].	html span	    with: aWidget label;	    onClick: [self selectTab: aWidget]]!renderTabs	ul contents: [:html |	    html li 		class: 'closeAll';		with: 'x';		onClick: [self close].	    self tabs do: [:each |		self renderTabFor: each on: html].	    html li		class: 'newtab';		with: ' + ';		onClick: [self newBrowserTab]]! !!TabManager methodsFor: 'updating'!update	self renderTabs! !TabManager class instanceVariableNames: 'current'!!TabManager class methodsFor: 'instance creation'!current    ^current ifNil: [current := super new]!new    self shouldNotImplement! !Widget subclass: #TabWidget	instanceVariableNames: 'div'	category: 'IDE'!!TabWidget methodsFor: 'accessing'!label    self subclassResponsibility! !!TabWidget methodsFor: 'actions'!open    TabManager current addTab: self.    TabManager current selectTab: self!show	div asJQuery show!hide	div asJQuery hide!remove	div asJQuery remove! !!TabWidget methodsFor: 'rendering'!renderOn: html	div := html div		class: 'jtalkTool';		yourself.	self renderTab!renderBoxOn: html!renderButtonsOn: html!update	self renderTab!renderTab	div contents: [:html |	    html div		class: 'jt_box';		with: [self renderBoxOn: html].	    html div		class: 'jt_buttons';		with: [self renderButtonsOn: html]]! !!TabWidget methodsFor: 'testing'!canBeClosed    ^false! !!TabWidget class methodsFor: 'instance creation'!open    ^self new open! !TabWidget subclass: #Workspace	instanceVariableNames: 'sourceArea'	category: 'IDE'!!Workspace methodsFor: 'accessing'!label    ^'[Workspace]'! !!Workspace methodsFor: 'actions'!clearWorkspace    sourceArea clear!doIt   sourceArea doIt!printIt	sourceArea printIt!inspectIt    sourceArea inspectIt! !!Workspace methodsFor: 'rendering'!renderBoxOn: html    sourceArea := SourceArea new.    sourceArea renderOn: html!renderButtonsOn: html    html button	with: 'DoIt';	title: 'ctrl+d';	onClick: [self doIt].    html button	with: 'PrintIt';	title: 'ctrl+p';	onClick: [self printIt].    html button	with: 'InspectIt';	title: 'ctrl+i';	onClick: [self inspectIt].    html button	with: 'Clear workspace';	onClick: [self clearWorkspace]! !TabWidget subclass: #Transcript	instanceVariableNames: 'textarea'	category: 'IDE'!!Transcript methodsFor: 'accessing'!label    ^'[Transcript]'! !!Transcript methodsFor: 'actions'!show: anObject    textarea asJQuery val: textarea asJQuery val, anObject asString.!cr    textarea asJQuery val: textarea asJQuery val, String cr.!clear    textarea asJQuery val: ''! !!Transcript methodsFor: 'rendering'!renderBoxOn: html    textarea := html textarea.    textarea asJQuery call: 'tabby'.    textarea 	class: 'jt_transcript';	at: 'spellcheck' put: 'false'!renderButtonsOn: html    html button	with: 'Clear transcript';	onClick: [self clear]! !Transcript class instanceVariableNames: 'current'!!Transcript class methodsFor: 'instance creation'!open    self current open!new    self shouldNotImplement!current    ^current ifNil: [current := super new]! !!Transcript class methodsFor: 'printing'!show: anObject    self current show: anObject!cr    self current show: String cr!clear    self current clear! !TabWidget subclass: #Browser	instanceVariableNames: 'selectedCategory selectedClass selectedProtocol selectedMethod commitButton categoriesList classesList protocolsList methodsList sourceArea tabsList selectedTab saveButton classButtons methodButtons unsavedChanges input'	category: 'IDE'!!Browser methodsFor: 'accessing'!label    ^selectedClass 	ifNil: ['Browser (nil)']	ifNotNil: [selectedClass name]!categories    | categories |    categories := Array new.    Smalltalk current classes do: [:each |	(categories includes: each category) ifFalse: [	    categories add: each category]].    ^categories sort!classes    ^(Smalltalk current classes 	select: [:each | each category = selectedCategory])	sort: [:a :b | a name < b name]!protocols    | klass |    selectedClass ifNotNil: [	selectedTab = #comment ifTrue: [^#()].	klass := selectedTab = #instance	    ifTrue: [selectedClass]	    ifFalse: [selectedClass class].	klass methodDictionary isEmpty ifTrue: [	    ^Array with: 'not yet classified'].	^klass protocols].    ^Array new!methods    | klass |    selectedTab = #comment ifTrue: [^#()].    selectedClass ifNotNil: [	klass := selectedTab = #instance	    ifTrue: [selectedClass]	    ifFalse: [selectedClass class]].    ^(selectedProtocol 	ifNil: [	    klass 		ifNil: [#()] 		ifNotNil: [klass methodDictionary values]]	ifNotNil: [	    klass methodDictionary values select: [:each |		each category = selectedProtocol]]) sort: [:a :b | a selector < b selector]!source    selectedTab = #comment ifFalse: [	^(selectedProtocol notNil or: [selectedMethod notNil])	    ifFalse: [self declarationSource]	    ifTrue: [self methodSource]].    ^selectedClass	ifNil: ['']	ifNotNil: [self classCommentSource]!methodSource    ^selectedMethod	ifNil: [self dummyMethodSource]	ifNotNil: [selectedMethod source]!dummyMethodSource    ^'messageSelectorAndArgumentNames	"comment stating purpose of message"	| temporary variable names |	statements'!declarationSource    ^selectedTab = #instance	ifTrue: [self classDeclarationSource]	ifFalse: [self metaclassDeclarationSource]!classDeclarationSource    | stream |    stream := '' writeStream.    selectedClass ifNotNil: [	stream 	    nextPutAll: selectedClass superclass asString;	    nextPutAll: ' subclass: #';	    nextPutAll: selectedClass name;	    nextPutAll: String lf, String tab;	    nextPutAll: 'instanceVariableNames: '''.	selectedClass instanceVariableNames 	    do: [:each | stream nextPutAll: each] 	    separatedBy: [stream nextPutAll: ' '].	stream	    nextPutAll: '''', String lf, String tab;	    nextPutAll: 'category: ''';	    nextPutAll: selectedClass category;	    nextPutAll: ''''].    ^stream contents!metaclassDeclarationSource    | stream |    stream := '' writeStream.    selectedClass ifNotNil: [	stream 	    nextPutAll: selectedClass asString;	    nextPutAll: ' class ';	    nextPutAll: 'instanceVariableNames: '''.	selectedClass class instanceVariableNames	    do: [:each | stream nextPutAll: each]	    separatedBy: [stream nextPutAll: ' '].	stream nextPutAll: ''''].    ^stream contents!classCommentSource    ^selectedClass comment!selectedClass	^selectedClass! !!Browser methodsFor: 'actions'!disableSaveButton    saveButton ifNotNil: [	saveButton at: 'disabled' put: true].    unsavedChanges := false!hideClassButtons    classButtons asJQuery hide!showClassButtons    classButtons asJQuery show!hideMethodButtons    methodButtons asJQuery hide!showMethodButtons    methodButtons asJQuery show!compile    self disableSaveButton.    selectedTab = #comment ifTrue: [	selectedClass ifNotNil: [	    self compileClassComment]].    (selectedProtocol notNil or: [selectedMethod notNil])	ifFalse: [self compileDefinition]	ifTrue: [self compileMethodDefinition]!compileClassComment    selectedClass comment: sourceArea val!compileMethodDefinition    selectedTab = #instance	ifTrue: [self compileMethodDefinitionFor: selectedClass]	ifFalse: [self compileMethodDefinitionFor: selectedClass class]!compileMethodDefinitionFor: aClass    | compiler method source node |    source := sourceArea val.    selectedProtocol ifNil: [selectedProtocol := selectedMethod category].    compiler := Compiler new.    compiler source: source.    node := compiler parse: source.    node isParseFailure ifTrue: [	^self alert: 'PARSE ERROR: ', node reason, ', position: ', node position asString].    compiler currentClass: aClass.    method := compiler eval: (compiler compileNode: node).    method category: selectedProtocol.    compiler unknownVariables do: [:each |	(self confirm: 'Declare ''', each, ''' as instance variable?') ifTrue: [		self addInstanceVariableNamed: each toClass: aClass.		^self compileMethodDefinitionFor: aClass]].    aClass addCompiledMethod: method.    compiler setupClass: aClass.    self updateMethodsList.    self selectMethod: method!compileDefinition    | newClass |    newClass := Compiler new loadExpression: sourceArea val.    self 	resetClassesList;	updateCategoriesList;	updateClassesList!commitCategory    selectedCategory ifNotNil: [	(Ajax url: self class commitPathJs, '/', selectedCategory, '.js')	    at: 'type' put: 'PUT';	    at: 'data' put: (Exporter new exportCategory: selectedCategory);	    at: 'error' put: [self alert: 'Commit failed!!'];	    send.	(Ajax url: self class commitPathJs, '/', selectedCategory, '.deploy.js')	    at: 'type' put: 'PUT';	    at: 'data' put: (StrippedExporter new exportCategory: selectedCategory);	    at: 'error' put: [self alert: 'Commit failed!!'];	    send.	(Ajax url: self class commitPathSt, '/', selectedCategory, '.st')	    at: 'type' put: 'PUT';	    at: 'data' put: (ChunkExporter new exportCategory: selectedCategory);	    at: 'error' put: [self alert: 'Commit failed!!'];	    send]!cancelChanges    ^unsavedChanges 	ifTrue: [self confirm: 'Cancel changes?']	ifFalse: [true]!removeClass    (self confirm: 'Do you really want to remove ', selectedClass name, '?')	ifTrue: [	    Smalltalk current removeClass: selectedClass.	    self resetClassesList.	    self selectClass: nil]!removeMethod    self cancelChanges ifTrue: [	(self confirm: 'Do you really want to remove #', selectedMethod selector, '?')	    ifTrue: [		selectedTab = #instance 			ifTrue: [selectedClass removeCompiledMethod: selectedMethod]			ifFalse: [selectedClass class removeCompiledMethod: selectedMethod].		self selectMethod: nil]]!setMethodProtocol: aString    self cancelChanges ifTrue: [	(self protocols includes: aString)	    ifFalse: [self addNewProtocol]	    ifTrue: [		selectedMethod category: aString.		selectedProtocol := aString.		selectedMethod := selectedMethod.		self 		    updateProtocolsList;		    updateMethodsList;		    updateSourceAndButtons]]!addNewProtocol    | newProtocol |    newProtocol := self prompt: 'New method protocol'.    newProtocol notEmpty ifTrue: [	selectedMethod category: newProtocol.	self setMethodProtocol: newProtocol]!selectCategory: aCategory    self cancelChanges ifTrue: [	selectedCategory := aCategory.	selectedClass := selectedProtocol := selectedMethod :=  nil.	self resetClassesList.	self 	    updateCategoriesList;	    updateClassesList;	    updateProtocolsList;	    updateMethodsList;	    updateSourceAndButtons]!selectClass: aClass    self cancelChanges ifTrue: [	selectedClass := aClass.	selectedProtocol := selectedMethod := nil.	self 	    updateClassesList;	    updateProtocolsList;	    updateMethodsList;	    updateSourceAndButtons]!selectProtocol: aString    self cancelChanges ifTrue: [	selectedProtocol := aString.	selectedMethod := nil.	self 	    updateProtocolsList;	    updateMethodsList;	    updateSourceAndButtons]!selectMethod: aMethod    self cancelChanges ifTrue: [	selectedMethod := aMethod.	self 	    updateProtocolsList;	    updateMethodsList;	    updateSourceAndButtons]!selectTab: aString    self cancelChanges ifTrue: [	selectedTab := aString.	self selectProtocol: nil.	self updateTabsList]!renameClass    | newName |    newName := self prompt: 'Rename class ', selectedClass name.    newName notEmpty ifTrue: [	selectedClass rename: newName.	self 		updateClassesList;		updateSourceAndButtons]!addInstanceVariableNamed: aString toClass: aClass	ClassBuilder new		addSubclassOf: aClass superclass named: aClass name instanceVariableNames: (aClass instanceVariableNames copy add: aString; yourself)!searchReferencesOf: aString	ReferencesBrowser search: aString!searchClassReferences	ReferencesBrowser search: selectedClass name!search: aString	self cancelChanges ifTrue: [| searchedClass |		searchedClass := Smalltalk current at: aString.		searchedClass isClass			ifTrue: [self class openOn: searchedClass]			ifFalse: [self searchReferencesOf: aString]]! !!Browser methodsFor: 'initialization'!initialize    super initialize.    selectedTab := #instance.    unsavedChanges := false! !!Browser methodsFor: 'rendering'!renderBoxOn: html    self 	renderTopPanelOn: html;	renderTabsOn: html;	renderBottomPanelOn: html!renderTopPanelOn: html    html div 	class: 'top'; 	with: [	    self renderInputOn: html.	    categoriesList := html ul class: 'jt_column browser categories'.	    commitButton := html button 		class: 'jt_commit';		title: 'Commit classes in this category to disk';		onClick: [self commitCategory];		with: 'Commit category'.	    classesList := ClassesList on: self.	    classesList renderOn: html.	    protocolsList := html ul class: 'jt_column browser protocols'.	    methodsList := html ul class: 'jt_column browser methods'.	    self		updateCategoriesList;		updateClassesList;		updateProtocolsList;		updateMethodsList.	    html div class: 'jt_clear']!renderTabsOn: html    tabsList := html ul class: 'jt_tabs'.    self updateTabsList.!renderBottomPanelOn: html    html div	class: 'jt_sourceCode';	with: [	    sourceArea := SourceArea new.	    sourceArea renderOn: html.	    sourceArea		onKeyUp: [self updateStatus]]!renderButtonsOn: html    saveButton := html button.    saveButton 	with: 'Save';	onClick: [self compile].    methodButtons := html span.    classButtons := html span.    html div 	class: 'right';	with: [		html button			with: 'DoIt';			onClick: [sourceArea doIt].		html button			with: 'PrintIt';			onClick: [sourceArea printIt].		html button with: 'InspectIt';			onClick: [sourceArea inspectit]].     self updateSourceAndButtons!renderInputOn: html 	input := html input 		class: 'implementors';		yourself.	input onKeyPress: [:event |		event keyCode = 13 ifTrue: [			self search: input asJQuery val]]! !!Browser methodsFor: 'testing'!canBeClosed	^true! !!Browser methodsFor: 'updating'!updateCategoriesList    categoriesList contents: [:html |	self categories do: [:each || li label |	    each isEmpty 		ifTrue: [label := 'Unclassified']		ifFalse: [label := each].	    li := html li.	    selectedCategory = each ifTrue: [		li class: 'selected'].	    li		with: label;		onClick: [self selectCategory: each]]]!updateClassesList    TabManager current update.    classesList updateNodes.    "classesList contents: [:html |	self classes do: [:each || li |	    li := html li.	    selectedClass = each ifTrue: [		li class: 'selected'].	    li		with: each name;		onClick: [self selectClass: each]]]"!updateProtocolsList    protocolsList contents: [:html |	self protocols do: [:each || li |	    li := html li.	    selectedProtocol = each ifTrue: [		li class: 'selected'].	    li 		with: each;		onClick: [self selectProtocol: each]]]!updateMethodsList    methodsList contents: [:html |	self methods do: [:each || li |	    li := html li.	    selectedMethod = each ifTrue: [		li class: 'selected'].	    li		with: each selector;		onClick: [self selectMethod: each]]]!updateTabsList    tabsList contents: [:html || li |	li := html li.	selectedTab = #instance ifTrue: [li class: 'selected'].	li	    with: 'Instance';	    onClick: [self selectTab: #instance].	li := html li.	selectedTab = #class ifTrue: [li class: 'selected'].	li	    with: 'Class';	    onClick: [self selectTab: #class].	li := html li.	selectedTab = #comment ifTrue: [li class: 'selected'].	li	    with: 'Comment';	    onClick: [self selectTab: #comment]]!updateSourceAndButtons	self disableSaveButton.	classButtons contents: [:html |		html button			with: 'Rename class';			onClick: [self renameClass].		html button			with: 'Remove class';			onClick: [self removeClass].		html button			with: 'References';			onClick: [self searchClassReferences]].	methodButtons contents: [:html |		html button			with: 'Remove method';			onClick: [self removeMethod].		html select 	    		onChange: [:e :select | self setMethodProtocol: select val];	    		with: [				html option		    			with: 'Method protocol';					at: 'disabled' put: 'disabled'.				html option		    			class: 'important';		    			with: 'New...'.				self protocols do: [:each |		    			html option with: each]].		selectedMethod isNil ifFalse: [			html select 	    			onChange: [:e :select | self searchReferencesOf: select val];	    			with: [					html option		    				with: 'References';						at: 'disabled' put: 'disabled'.					html option		    				class: 'important';		    				with: selectedMethod selector.					selectedMethod messageSends sorted do: [:each |		    				html option with: each]]]].    	selectedMethod isNil		ifTrue: [	    		self hideMethodButtons.	    			(selectedClass isNil or: [selectedProtocol notNil])					ifTrue: [self hideClassButtons]	    				ifFalse: [self showClassButtons]]		ifFalse: [	    		self hideClassButtons.	    		self showMethodButtons].    	sourceArea val: self source!updateStatus	sourceArea val = self source		ifTrue: [			saveButton ifNotNil: [				saveButton at: 'disabled' put: true].    			unsavedChanges := false]		ifFalse: [			saveButton ifNotNil: [    				saveButton removeAt: 'disabled'].    			unsavedChanges := true]!resetClassesList	classesList resetNodes! !!Browser class methodsFor: 'accessing'!commitPathJs	^'js'!commitPathSt	^'st'! !!Browser class methodsFor: 'convenience'!openOn: aClass    ^self new	open;	selectCategory: aClass category;	selectClass: aClass!open    self new open! !TabWidget subclass: #Inspector	instanceVariableNames: 'label variables object selectedVariable variablesList valueTextarea workspaceTextarea diveButton'	category: 'IDE'!!Inspector methodsFor: 'accessing'!label	^label ifNil: ['Inspector (nil)']!variables	^variables!setVariables: aCollection	variables := aCollection!setLabel: aString	label := aString!selectedVariable	^selectedVariable!selectedVariable: aString	selectedVariable := aString! !!Inspector methodsFor: 'actions'!inspect: anObject	object := anObject.	variables := #().	object inspectOn: self!dive	(self variables at: self selectedVariable) inspect!refresh	self 		inspect: object; 		updateVariablesList;		updateValueTextarea! !!Inspector methodsFor: 'rendering'!renderBoxOn: html	self 		renderTopPanelOn: html;		renderBottomPanelOn: html!renderTopPanelOn: html    html div 	class: 'top'; 	with: [	    variablesList := html ul class: 'jt_column variables'.	    valueTextarea := html textarea class: 'jt_column value'; at: 'readonly' put: 'readonly'.	    self		updateVariablesList;		updateValueTextarea.	    html div class: 'jt_clear']!renderBottomPanelOn: html    html div	class: 'jt_sourceCode';	with: [	    workspaceTextarea := html textarea 		class: 'source';		at: 'spellcheck' put: 'false'.	    workspaceTextarea asJQuery call: 'tabby']!renderButtonsOn: html	html button		with: 'Refresh';		onClick: [self refresh].	diveButton := html button 		with: 'Dive'; 		onClick: [self dive].	self updateButtons! !!Inspector methodsFor: 'testing'!canBeClosed	^true! !!Inspector methodsFor: 'updating'!updateVariablesList	variablesList contents: [:html |		self variables keys do: [:each || li |			li := html li.			li				with: each;				onClick: [self selectVariable: each].			self selectedVariable = each ifTrue: [				li class: 'selected']]]!selectVariable: aString	self selectedVariable: aString.	self 		updateVariablesList;		updateValueTextarea;		updateButtons!updateValueTextarea	valueTextarea asJQuery val: (self selectedVariable isNil		ifTrue: ['']		ifFalse: [(self variables at: self selectedVariable) printString])!updateButtons	(self selectedVariable notNil and: [(self variables at: self selectedVariable) notNil])		ifFalse: [diveButton at: 'disabled' put: true] 		ifTrue: [diveButton removeAt: 'disabled']! !!Inspector class methodsFor: 'instance creation'!on: anObject	^self new		inspect: anObject;		yourself! !TabWidget subclass: #ReferencesBrowser	instanceVariableNames: 'implementors senders implementorsList input timer selector sendersList referencedClasses referencedClassesList'	category: 'IDE'!!ReferencesBrowser methodsFor: 'accessing'!implementors	^implementors ifNil: [implementors := Array new]!label	^'[ReferencesBrowser]'!selector	^selector!senders	^senders ifNil: [senders := Array new]!classesAndMetaclasses	^Smalltalk current classes, (Smalltalk current classes collect: [:each | each class])!referencedClasses	^referencedClasses ifNil: [referencedClasses := Array new]! !!ReferencesBrowser methodsFor: 'actions'!openBrowserOn: aMethod       | browser |       browser := Browser openOn: (aMethod class isMetaclass 		ifTrue: [aMethod methodClass instanceClass] ifFalse: [aMethod methodClass]).       aMethod methodClass isMetaclass ifTrue: [browser selectTab: #class].       browser               selectProtocol: aMethod category;               selectMethod: aMethod!searchReferencesFor: aString	selector := aString.	implementors := Array new.	senders := Array new.	referencedClasses := Array new.	(selector match: '^[A-Z]') 		ifFalse: [self searchSelectorReferencesFor: selector]		ifTrue: [self searchReferencedClassesFor: selector]!search: aString	self 		searchReferencesFor: aString;		updateImplementorsList;		updateSendersList;		updateReferencedClassesList!searchReferencedClassesFor: aString	self classesAndMetaclasses do: [:each |		each methodDictionary values do: [:value |			(((value referencedClasses select: [:each | each notNil])collect: [:each | each name]) includes: selector) ifTrue: [				self referencedClasses add: value]]]!searchSelectorReferencesFor: aString	self classesAndMetaclasses do: [:each | 		each methodDictionary keysAndValuesDo: [:key :value | 			key = selector ifTrue: [self implementors add: value]].		each methodDictionary keysAndValuesDo: [:key :value | 			(value messageSends includes: selector) ifTrue: [				self senders add: value]]]! !!ReferencesBrowser methodsFor: 'initialization'!initialize	super initialize.	selector := ''! !!ReferencesBrowser methodsFor: 'private'!setInputEvents	input		onKeyUp: [timer := [self search: input asJQuery val] valueWithTimeout: 100];		onKeyDown: [timer ifNotNil: [timer clearTimeout]]! !!ReferencesBrowser methodsFor: 'rendering'!renderBoxOn: html	self 		renderInputOn: html;		renderImplementorsOn: html;		renderSendersOn: html;		renderReferencedClassesOn: html!renderInputOn: html	input := html input 		class: 'implementors';		yourself.	input asJQuery val: selector.	self setInputEvents!renderImplementorsOn: html    	implementorsList := html ul class: 'jt_column implementors'.	self updateImplementorsList!renderSendersOn: html    	sendersList := html ul class: 'jt_column senders'.	self updateSendersList!renderReferencedClassesOn: html    	referencedClassesList := html ul class: 'jt_column referenced_classes'.	self updateReferencedClassesList! !!ReferencesBrowser methodsFor: 'testing'!canBeClosed	^true! !!ReferencesBrowser methodsFor: 'updating'!updateImplementorsList    implementorsList contents: [:html |	html li		class: 'column_label'; 		with: 'Implementors (', self implementors size asString, ')';		style: 'font-weight: bold'.	self implementors do: [:each || li |	    li := html li.	    li		with: (each methodClass asString, ' >> ', self selector);		onClick: [self openBrowserOn: each]]]!updateSendersList    	sendersList contents: [:html |	html li		class: 'column_label'; 		with: 'Senders (', self senders size asString, ')';		style: 'font-weight: bold'.	self senders do: [:each |		html li	    		with: (each methodClass asString, ' >> ', each selector);			onClick: [self openBrowserOn: each]]]!updateReferencedClassesList    	referencedClassesList contents: [:html |	html li		class: 'column_label'; 		with: 'Class references (', self referencedClasses size asString, ')';		style: 'font-weight: bold'.	self referencedClasses do: [:each |		html li	    		with: (each methodClass asString, ' >> ', each selector);			onClick: [self openBrowserOn: each]]]! !!ReferencesBrowser class methodsFor: 'instance creation'!search: aString	^self new		searchReferencesFor: aString;		open! !Widget subclass: #SourceArea	instanceVariableNames: 'editor div'	category: 'IDE'!!SourceArea methodsFor: 'accessing'!val    ^editor getValue!val: aString    editor setValue: aString!currentLine    ^editor getLine: (editor getCursor line)!selection	^editor getSelection!selectionEnd   ^textarea element selectionEnd!selectionStart   ^textarea element selectionStart!selectionStart: anInteger   textarea element selectionStart: anInteger!selectionEnd: anInteger   textarea element selectionEnd: anInteger!setEditorOn: aTextarea	<self['@editor'] = CodeMirror.fromTextArea(aTextarea, {        	theme: 'jtalk',                lineNumbers: true,                enterMode: 'classic',                matchBrackets: true,                electricChars: false,	})>!editor	^editor! !!SourceArea methodsFor: 'actions'!clear    textarea asJQuery val: ''!doIt    | selection |    editor somethingSelected	ifFalse: [selection := self currentLine]	ifTrue: [selection := self selection].    ^self eval: selection!eval: aString    | compiler node |    compiler := Compiler new.    node := compiler parseExpression: aString.    node isParseFailure ifTrue: [	^self alert: node reason, ', position: ', node position].    ^compiler loadExpression: aString!handleKeyDown: anEvent    <if(anEvent.ctrlKey) {		if(anEvent.keyCode === 80) { //ctrl+p			self._printIt();			anEvent.preventDefault();			return false;		}		if(anEvent.keyCode === 68) { //ctrl+d			self._doIt();			anEvent.preventDefault();			return false;		}		if(anEvent.keyCode === 73) { //ctrl+i			self._inspectIt();			anEvent.preventDefault();			return false;		}	}>!inspectIt    self doIt inspect!print: aString	| start stop |	start := Dictionary new.	stop := Dictionary new.	start at: 'line' put: (editor getCursor: false) line.	start at: 'ch' put: (editor getCursor: false) ch.	stop at: 'line' put: (start at: 'line').	stop at: 'ch' put: ((start at: 'ch') + aString size + 2).	editor replaceSelection: (editor getSelection, ' ', aString, ' ').	editor setCursor: (editor getCursor: true).	editor setSelection: stop end: start!printIt    self print: self doIt printString! !!SourceArea methodsFor: 'events'!onKeyUp: aBlock	div onKeyUp: aBlock!onKeyDown: aBlock	div onKeyDown: aBlock! !!SourceArea methodsFor: 'rendering'!renderOn: html    | textarea |    div := html div class: 'source'.    div with: [textarea := html textarea].    self setEditorOn: textarea element.    div onKeyDown: [:e | self handleKeyDown: e]! !Widget subclass: #ClassesList	instanceVariableNames: 'browser ul nodes'	category: 'IDE'!!ClassesList methodsFor: 'accessing'!category	^self browser selectedCategory!nodes	nodes ifNil: [nodes := self getNodes].	^nodes!browser	^browser!browser: aBrowser	browser := aBrowser!getNodes	| classes children others |	classes := self browser classes.	children := #().	others := #().	classes do: [:each |		(classes includes: each superclass)			ifFalse: [children add: each]			ifTrue: [others add: each]].	^children collect: [:each |		ClassesListNode on: each browser: self browser classes: others level: 0]!resetNodes	nodes := nil! !!ClassesList methodsFor: 'rendering'!renderOn: html	ul := html ul		class: 'jt_column browser classes';		yourself.	self updateNodes!updateNodes	ul contents: [:html |		self nodes do: [:each |			each renderOn: html]]! !!ClassesList class methodsFor: 'instance creation'!on: aBrowser	^self new 		browser: aBrowser; 		yourself! !Widget subclass: #ClassesListNode	instanceVariableNames: 'browser theClass level nodes'	category: 'IDE'!!ClassesListNode methodsFor: 'accessing'!nodes	^nodes!theClass	^theClass!theClass: aClass	theClass := aClass!browser	^browser!browser: aBrowser	browser := aBrowser!level	^level!level: anInteger	level := anInteger!label	| str |	str := String new writeStream.	self level timesRepeat: [		str nextPutAll: '    '].	str nextPutAll: self theClass name.	^str contents!getNodesFrom: aCollection	| children others |	children := #().	others := #().	aCollection do: [:each |		(each superclass = self theClass)			ifTrue: [children add: each]			ifFalse: [others add: each]].	nodes:= children collect: [:each |		ClassesListNode on: each browser: self browser classes: others level: self level + 1]! !!ClassesListNode methodsFor: 'rendering'!renderOn: html	| li |	li := html li 		onClick: [self browser selectClass: self theClass].	li asJQuery contents: self label.	self browser selectedClass = self theClass ifTrue:  [		li class: 'selected'].	self nodes do: [:each |		each renderOn: html]! !!ClassesListNode class methodsFor: 'instance creation'!on: aClass browser: aBrowser classes: aCollection level: anInteger	^self new		theClass: aClass;		browser: aBrowser;		level: anInteger;		getNodesFrom: aCollection;		yourself! !TabWidget subclass: #Debugger	instanceVariableNames: 'error selectedContext sourceArea ul'	category: 'IDE'!!Debugger methodsFor: 'accessing'!error	^error!error: anError	error := anError!label	^'[Debugger]'! !!Debugger methodsFor: 'actions'!selectContext: aContext	selectedContext := aContext.	self updateContextsList.	self updateSourceArea! !!Debugger methodsFor: 'rendering'!renderBoxOn: html    self 	renderTopPanelOn: html;	renderBottomPanelOn: html!renderTopPanelOn: html	selectedContext := self error context.	html div 		class: 'top'; 		with: [			html div 				class: 'label';				with: self error messageText.	    		ul := html ul 			class: 'jt_column debugger contexts';			with: [self renderContext: self error context on: html]]!renderContext: aContext on: html	| li |	li := html li.	selectedContext = aContext ifTrue: [		li class: 'selected'].	li 		with: aContext asString;		onClick: [self selectContext: aContext].	aContext home ifNotNil: [self renderContext: aContext home on: html]!renderBottomPanelOn: html    html div	class: 'jt_sourceCode';	with: [	    sourceArea := SourceArea new.	    sourceArea renderOn: html].    self updateSourceArea! !!Debugger methodsFor: 'testing'!canBeClosed    ^true! !!Debugger methodsFor: 'updating'!updateContextsList	ul contents: [:html |		self renderContext: self error context on: html]!updateSourceArea	 sourceArea val: (selectedContext receiver class methodAt: selectedContext selector) source! !ErrorHandler subclass: #DebugErrorHandler	instanceVariableNames: ''	category: 'IDE'!!DebugErrorHandler methodsFor: 'error handling'!handleError: anError	[Debugger new		error: anError;		open] on: Error do: [:error |			ErrorHandler new handleError: error]! !!DebugErrorHandler class methodsFor: 'initialization'!initialize	self register! !!Object methodsFor: '*IDE'!inspect	Inspector new 		inspect: self;		open!inspectOn: anInspector	| variables |	variables := Dictionary new.	variables at: '#self' put: self.	self class allInstanceVariableNames do: [:each |		variables at: each put: (self instVarAt: each)].	anInspector 		setLabel: self printString;		setVariables: variables! !!Date methodsFor: '*IDE'!inspectOn: anInspector	| variables |	variables := Dictionary new.	variables at: '#self' put: self.	variables at: '#year' put: self year.	variables at: '#month' put: self month.	variables at: '#day' put: self day.	variables at: '#hours' put: self hours.	variables at: '#minutes' put: self minutes.	variables at: '#seconds' put: self seconds.	variables at: '#milliseconds' put: self milliseconds.	anInspector 		setLabel: self printString;		setVariables: variables! !!Collection methodsFor: '*IDE'!inspectOn: anInspector	| variables |	variables := Dictionary new.	variables at: '#self' put: self.	self withIndexDo: [:each :i |		variables at: i put: each].	anInspector 		setLabel: self printString;		setVariables: variables! !!String methodsFor: '*IDE'!inspectOn: anInspector	| label |	super inspectOn: anInspector.	self printString size > 30 		ifTrue: [label := (self printString copyFrom: 1 to: 30), '...''']		ifFalse: [label := self printString]. 	anInspector setLabel: label! !!MethodContext methodsFor: '*IDE'!inspectOn: anInspector	| variables |	variables := Dictionary new.	variables at: '#self' put: self.	variables at: '#home' put: self home.	variables at: '#receiver' put: self receiver.	variables at: '#selector' put: self selector.	variables at: '#temps' put: self temps.	self class instanceVariableNames do: [:each |		variables at: each put: (self instVarAt: each)].	anInspector 		setLabel: self printString;		setVariables: variables! !!Dictionary methodsFor: '*IDE'!inspectOn: anInspector	| variables |	variables := Dictionary new.	variables at: '#self' put: self.	variables at: '#keys' put: self keys.	self keysAndValuesDo: [:key :value |		variables at: key put: value].	anInspector 		setLabel: self printString;		setVariables: variables! !
 |