123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598 |
- Smalltalk current createPackage: 'Helios-Workspace'!
- Object subclass: #HLCodeModel
- instanceVariableNames: 'announcer environment receiver'
- package: 'Helios-Workspace'!
- !HLCodeModel methodsFor: 'accessing'!
- announcer
- ^ announcer ifNil: [ announcer := Announcer new ]
- !
- environment
- ^ environment ifNil: [ HLManager current environment ]
- !
- environment: anEnvironment
- environment := anEnvironment
- !
- receiver
- ^ receiver ifNil: [ receiver := self defaultReceiver ]
- !
- receiver: anObject
- receiver := anObject
- ! !
- !HLCodeModel methodsFor: 'actions'!
- doIt: someCode
- ^ self environment eval: someCode on: self receiver
- !
- subscribe: aWidget
- aWidget subscribeTo: self announcer
- ! !
- !HLCodeModel methodsFor: 'defaults'!
- defaultReceiver
- ^ DoIt new
- ! !
- !HLCodeModel class methodsFor: 'actions'!
- on: anEnvironment
- ^ self new
- environment: anEnvironment;
- yourself
- ! !
- HLWidget subclass: #HLCodeWidget
- instanceVariableNames: 'model wrapper code editor'
- package: 'Helios-Workspace'!
- !HLCodeWidget methodsFor: 'accessing'!
- announcer
- ^ self model announcer
- !
- contents
- ^ editor getValue
- !
- contents: aString
- editor setValue: aString
- !
- currentLine
- ^editor getLine: (editor getCursor line)
- !
- currentLineOrSelection
- ^editor somethingSelected
- ifFalse: [ self currentLine ]
- ifTrue: [ self selection ]
- !
- model
- ^ model ifNil: [ model := HLCodeModel new ]
- !
- model: aModel
- model := aModel
- !
- receiver
- ^ self model receiver
- !
- receiver: anObject
- self model receiver: anObject
- !
- selection
- ^editor getSelection
- !
- selectionEnd
- ^code element selectionEnd
- !
- selectionEnd: anInteger
- code element selectionEnd: anInteger
- !
- selectionStart
- ^code element selectionStart
- !
- selectionStart: anInteger
- code element selectionStart: anInteger
- ! !
- !HLCodeWidget methodsFor: 'actions'!
- clear
- self contents: ''
- !
- configureEditor
- self editor at: 'amberCodeWidget' put: self
- !
- doIt
- | result |
- self announcer announce: (HLDoItRequested on: model).
- result:= model doIt: self currentLineOrSelection.
- self announcer announce: (HLDoItExecuted on: model).
- ^ result
- !
- editor
- ^editor
- !
- focus
- editor focus
- !
- inspectIt
- | newInspector |
-
- self announcer announce: (HLInspectItRequested on: model).
- newInspector := self makeInspectorOn: self doIt.
- newInspector open
- !
- makeInspectorOn: anObject
- ^ HLInspector new
- inspect: anObject;
- yourself
- !
- print: aString
- | start stop currentLine |
- currentLine := (editor getCursor: false) line.
- start := HashedCollection new.
- start at: 'line' put: currentLine.
- start at: 'ch' put: (editor getCursor: false) ch.
- (editor getSelection) ifEmpty: [
- "select current line if selection is empty"
- start at: 'ch' put: (editor getLine: currentLine) size.
- editor setSelection: #{'line' -> currentLine. 'ch' -> 0} end: start.
- ].
- stop := HashedCollection new.
- stop at: 'line' put: currentLine.
- 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
- | result |
- result:= self doIt.
-
- self announcer announce: (HLPrintItRequested on: model).
- self print: result printString.
- self focus.
- !
- saveIt
- "I do not do anything"
- !
- setEditorOn: aTextarea
- <self['@editor'] = CodeMirror.fromTextArea(aTextarea, {
- theme: 'amber',
- lineNumbers: true,
- enterMode: 'flat',
- indentWithTabs: true,
- indentUnit: 4,
- matchBrackets: true,
- electricChars: false,
- keyMap: 'Amber',
- extraKeys: {"Shift-Space": "autocomplete"}
- })>
- ! !
- !HLCodeWidget methodsFor: 'hints'!
- messageHintFor: anEditor token: aToken
- ^ ((Smalltalk current at: 'allSelectors') value asSet asArray
- select: [ :each | each includesSubString: aToken string ])
- reject: [ :each | each = aToken string ]
- !
- variableHintFor: anEditor token: aToken
- | variables classNames pseudoVariables |
-
- variables := ((window jQuery: anEditor display wrapper) find: 'span.cm-variable') get
- collect: [ :each | (window jQuery: each) html ].
-
- classNames := Smalltalk current classes collect: [ :each | each name ].
- pseudoVariables := Smalltalk current pseudoVariableNames.
-
- ^ ((variables, classNames, pseudoVariables) asSet asArray
- select: [ :each | each includesSubString: aToken string ])
- reject: [ :each | each = aToken string ]
- ! !
- !HLCodeWidget methodsFor: 'reactions'!
- onDoIt
-
- self doIt
- !
- onInspectIt
- self inspectIt
- !
- onPrintIt
- self printIt
- !
- onSaveIt
- "I do not do anything"
- ! !
- !HLCodeWidget methodsFor: 'rendering'!
- renderContentOn: html
- code := html textarea.
- self setEditorOn: code element.
- self configureEditor
- ! !
- !HLCodeWidget methodsFor: 'testing'!
- canHaveFocus
- ^ true
- !
- hasFocus
- ^ code asJQuery is: ':active'
- ! !
- !HLCodeWidget class methodsFor: 'accessing'!
- keyMap
- ^ HLManager current keyBinder systemIsMac
- ifTrue: [ self macKeyMap ]
- ifFalse: [ self pcKeyMap ]
- !
- macKeyMap
- ^ #{
- 'Alt-Backspace' -> 'delWordBefore'.
- 'Alt-Delete' -> 'delWordAfter'.
- 'Alt-Left' -> 'goWordBoundaryLeft'.
- 'Alt-Right' -> 'goWordBoundaryRight'.
- 'Cmd-A' -> 'selectAll'.
- 'Cmd-Alt-F' -> 'replace'.
- 'Cmd-D' -> 'doIt'.
- 'Cmd-Down' -> 'goDocEnd'.
- 'Cmd-End' -> 'goDocEnd'.
- 'Cmd-F' -> 'find'.
- 'Cmd-G' -> 'findNext'.
- 'Cmd-I' -> 'inspectIt'.
- 'Cmd-Left' -> 'goLineStart'.
- 'Cmd-P' -> 'printIt'.
- 'Cmd-Right' -> 'goLineEnd'.
- 'Cmd-S' -> 'saveIt'.
- 'Cmd-Up' -> 'goDocStart'.
- 'Cmd-Y' -> 'redo'.
- 'Cmd-Z' -> 'undo'.
- 'Cmd-[' -> 'indentLess'.
- 'Cmd-]' -> 'indentMore'.
- 'Ctrl-Alt-Backspace' -> 'delWordAfter'.
- 'Shift-Cmd-Alt-F' -> 'replaceAll'.
- 'Shift-Cmd-G' -> 'findPrev'.
- 'Shift-Cmd-Z' -> 'redo'.
- 'fallthrough' -> { 'basic'. 'emacsy' }
- }
- !
- pcKeyMap
- ^ {
- 'Alt-Left' -> 'goLineStart'.
- 'Alt-Right' -> 'goLineEnd'.
- 'Alt-Up' -> 'goDocStart'.
- 'Ctrl-A' -> 'selectAll'.
- 'Ctrl-Backspace' -> 'delWordBefore'.
- 'Ctrl-D' -> 'doIt'.
- 'Ctrl-Delete' -> 'delWordAfter'.
- 'Ctrl-Down' -> 'goDocEnd'.
- 'Ctrl-End' -> 'goDocEnd'.
- 'Ctrl-F' -> 'find'.
- 'Ctrl-G' -> 'findNext'.
- 'Ctrl-I' -> 'inspectIt'.
- 'Ctrl-Home' -> 'goDocStart'.
- 'Ctrl-Left' -> 'goWordBoundaryLeft'.
- 'Ctrl-P' -> 'printIt'.
- 'Ctrl-Right' -> 'goWordBoundaryRight'.
- 'Ctrl-S' -> 'saveIt'.
- 'Ctrl-Y' -> 'redo'.
- 'Ctrl-Z' -> 'undo'.
- 'Ctrl-[' -> 'indentLess'.
- 'Ctrl-]' -> 'indentMore'.
- 'Shift-Ctrl-F' -> 'replace'.
- 'Shift-Ctrl-G' -> 'findPrev'.
- 'Shift-Ctrl-R' -> 'replaceAll'.
- 'Shift-Ctrl-Z' -> 'redo'.
- 'fallthrough' -> #('basic')
- }
- !
- tabLabel
- ^ 'Workspace'
- !
- tabPriority
- ^ 10
- ! !
- !HLCodeWidget class methodsFor: 'hints'!
- hintFor: anEditor options: options
- | cursor token completions |
-
- cursor := anEditor getCursor.
- token := anEditor getTokenAt: cursor.
- token at: 'state' put: ((CodeMirror basicAt: 'innerMode')
- value: anEditor getMode value: (token at: 'state')) state.
-
- completions := token type = 'variable'
- ifTrue: [ HLCodeWidget variableHintFor: anEditor token: token ]
- ifFalse: [ HLCodeWidget messageHintFor: anEditor token: token ].
-
- ^ #{
- 'list' -> completions.
- 'from' -> ((CodeMirror basicAt: 'Pos') value: cursor line value: token end).
- 'to' -> ((CodeMirror basicAt: 'Pos') value: cursor line value: token start)
- }
- !
- messageHintFor: anEditor token: aToken
- ^ (anEditor at: 'amberCodeWidget')
- messageHintFor: anEditor token: aToken
- !
- variableHintFor: anEditor token: aToken
- ^ (anEditor at: 'amberCodeWidget')
- variableHintFor: anEditor token: aToken
- ! !
- !HLCodeWidget class methodsFor: 'initialization'!
- initialize
- super initialize.
- self
- setupCodeMirror;
- setupCommands;
- setupKeyMaps.
- !
- setupCodeMirror
- <
- CodeMirror.keyMap.default.fallthrough = ["basic"];
- CodeMirror.commands.autocomplete = function(cm) {
- CodeMirror.showHint(cm, self._hintFor_options_);
- }
- >
- !
- setupCommands
- (CodeMirror basicAt: 'commands')
- at: 'doIt' put: [ :cm | cm amberCodeWidget doIt ];
- at: 'inspectIt' put: [ :cm | cm amberCodeWidget inspectIt ];
- at: 'printIt' put: [ :cm | cm amberCodeWidget printIt ];
- at: 'saveIt' put: [ :cm | cm amberCodeWidget saveIt ]
- !
- setupKeyMaps
- <CodeMirror.keyMap['Amber'] = self._keyMap()>
- ! !
- !HLCodeWidget class methodsFor: 'testing'!
- canBeOpenAsTab
- ^ true
- ! !
- HLCodeWidget subclass: #HLSourceCodeWidget
- instanceVariableNames: 'browserModel methodContents'
- package: 'Helios-Workspace'!
- !HLSourceCodeWidget methodsFor: 'accessing'!
- browserModel
- ^ browserModel
- !
- browserModel: aBrowserModel
- browserModel := aBrowserModel.
- self
- observeSystem;
- observeBrowserModel
- !
- methodContents
- ^ methodContents
- !
- methodContents: aString
- methodContents := aString
- !
- previous
- "for browser lists widget"
- !
- previous: aWidget
- "for browser lists widget"
- ! !
- !HLSourceCodeWidget methodsFor: 'actions'!
- observeBrowserModel
- self browserModel announcer
- on: HLSaveSourceCode
- do: [ :ann | self onSaveIt ];
- on: HLParseErrorRaised
- do: [ :ann | self onParseError: ann ];
- on: HLCompileErrorRaised
- do: [ :ann | self onCompileError: ann error ];
- on: HLUnknownVariableErrorRaised
- do: [ :ann | self onUnknownVariableError: ann error ];
- on: HLInstVarAdded
- do: [ :ann | self onInstVarAdded ];
- on: HLMethodSelected
- do: [ :ann | self onMethodSelected: ann item ];
- on: HLClassSelected
- do: [ :ann | self onClassSelected: ann item ];
- on: HLProtocolSelected
- do: [ :ann | self onProtocolSelected: ann item ];
- on: HLSourceCodeFocusRequested
- do: [ :ann | self onSourceCodeFocusRequested ]
- !
- observeSystem
- self browserModel systemAnnouncer
- on: MethodModified
- do: [ :ann | self onMethodModified: ann method ];
- on: HLMethodSelected
- do: [ :ann | self onMethodSelected: ann item ];
- on: HLClassSelected
- do: [ :ann | self onClassSelected: ann item ];
- on: HLProtocolSelected
- do: [ :ann | self onProtocolSelected: ann item ];
- on: HLSourceCodeFocusRequested
- do: [ :ann | self onSourceCodeFocusRequested ]
- !
- refresh
- self hasModification ifTrue: [ ^ self ].
- self hasFocus ifTrue: [ ^ self ].
- self contents: self model selectedMethod source
- !
- saveIt
- self browserModel saveSourceCode
- ! !
- !HLSourceCodeWidget methodsFor: 'reactions'!
- onClassSelected: aClass
- aClass ifNil: [ ^ self contents: '' ].
-
- self contents: aClass definition
- !
- onCompileError: anError
- self alert: anError messageText
- !
- onInstVarAdded
- self browserModel save: self contents
- !
- onMethodModified: aMethod
- self browserModel selectedClass = aMethod methodClass ifFalse: [ ^ self ].
- self browserModel selectedMethod ifNil: [ ^ self ].
- self browserModel selectedMethod selector = aMethod selector ifFalse: [ ^ self ].
- self refresh
- !
- onMethodSelected: aCompiledMethod
- aCompiledMethod ifNil: [ ^ self contents: '' ].
-
- self contents: aCompiledMethod source
- !
- onParseError: anAnnouncement
- | lineIndex newContents |
-
- lineIndex := 1.
-
- self contents: (String streamContents: [ :stream |
- self contents linesDo: [ :each |
- lineIndex = anAnnouncement line
- ifTrue: [
- stream
- nextPutAll: (each copyFrom: 1 to: anAnnouncement column);
- nextPutAll: '<- ';
- nextPutAll: anAnnouncement message;
- nextPutAll: ' ';
- nextPutAll: (each copyFrom: anAnnouncement column + 1 to: each size) ]
- ifFalse: [ stream nextPutAll: each ].
- stream nextPutAll: String cr.
- lineIndex := lineIndex + 1 ] ])
- !
- onProtocolSelected: aString
- self browserModel selectedClass ifNil: [ ^ self contents: '' ].
-
- self contents: self browserModel selectedClass definition
- !
- onSaveIt
- self browserModel save: self contents
- !
- onSourceCodeFocusRequested
- self focus
- !
- onUnknownVariableError: anError
- | confirm |
- confirm := self confirm: (String streamContents: [ :stream |
- stream
- nextPutAll: anError messageText;
- nextPutAll: String cr;
- nextPutAll: 'Would you like to define an instance variable?' ]).
-
- confirm ifFalse: [ ^ self ].
-
- self browserModel addInstVarNamed: anError variableName
- ! !
- !HLSourceCodeWidget methodsFor: 'testing'!
- hasModification
- ^ (self methodContents = self contents) not
- ! !
- !HLSourceCodeWidget class methodsFor: 'instance creation'!
- on: aBrowserModel
- ^ self new
- browserModel: aBrowserModel;
- yourself
- ! !
- !HLSourceCodeWidget class methodsFor: 'testing'!
- canBeOpenAsTab
- ^ false
- ! !
|