Helios-Core.st 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522
  1. Smalltalk current createPackage: 'Helios-Core' properties: #{}!
  2. Object subclass: #HLTab
  3. instanceVariableNames: 'widget label'
  4. package: 'Helios-Core'!
  5. !HLTab methodsFor: 'accessing'!
  6. activate
  7. self manager activate: self
  8. !
  9. add
  10. self manager addTab: self
  11. !
  12. label
  13. ^ label ifNil: [ '' ]
  14. !
  15. label: aString
  16. label := aString
  17. !
  18. manager
  19. ^ HLManager current
  20. !
  21. widget
  22. ^ widget
  23. !
  24. widget: aWidget
  25. widget := aWidget
  26. ! !
  27. !HLTab methodsFor: 'testing'!
  28. isActive
  29. ^ self manager activeTab = self
  30. ! !
  31. !HLTab class methodsFor: 'instance creation'!
  32. on: aWidget labelled: aString
  33. ^ self new
  34. widget: aWidget;
  35. label: aString;
  36. yourself
  37. ! !
  38. Widget subclass: #HLWidget
  39. instanceVariableNames: 'wrapper'
  40. package: 'Helios-Core'!
  41. !HLWidget methodsFor: 'accessing'!
  42. manager
  43. ^ HLManager current
  44. !
  45. wrapper
  46. ^ wrapper
  47. ! !
  48. !HLWidget methodsFor: 'keybindings'!
  49. registerBindings
  50. self registerBindingsOn: self manager keyBinder bindings
  51. !
  52. registerBindingsOn: aBindingGroup
  53. ! !
  54. !HLWidget methodsFor: 'rendering'!
  55. renderContentOn: html
  56. !
  57. renderOn: html
  58. self registerBindings.
  59. wrapper := html div with: [
  60. self renderContentOn: html ]
  61. ! !
  62. !HLWidget methodsFor: 'updating'!
  63. refresh
  64. self wrapper ifNil: [ ^ self ].
  65. self wrapper asJQuery empty.
  66. [ :html | self renderContentOn: html ] appendToJQuery: self wrapper asJQuery
  67. ! !
  68. !HLWidget class methodsFor: 'accessing'!
  69. openAsTab
  70. HLManager current addTab: (HLTab on: self new labelled: self tabLabel)
  71. !
  72. tabLabel
  73. ^ 'Tab'
  74. !
  75. tabPriority
  76. ^ 500
  77. ! !
  78. !HLWidget class methodsFor: 'testing'!
  79. canBeOpenAsTab
  80. ^ false
  81. ! !
  82. HLWidget subclass: #HLDebugger
  83. instanceVariableNames: ''
  84. package: 'Helios-Core'!
  85. HLWidget subclass: #HLFocusableWidget
  86. instanceVariableNames: 'hiddenInput'
  87. package: 'Helios-Core'!
  88. !HLFocusableWidget methodsFor: 'accessing'!
  89. focusClass
  90. ^ 'focused'
  91. ! !
  92. !HLFocusableWidget methodsFor: 'events'!
  93. blur
  94. hiddenInput asJQuery blur
  95. !
  96. focus
  97. hiddenInput asJQuery focus
  98. !
  99. hasFocus
  100. ^ self wrapper notNil and: [ self wrapper asJQuery hasClass: self focusClass ]
  101. ! !
  102. !HLFocusableWidget methodsFor: 'rendering'!
  103. renderContentOn: html
  104. !
  105. renderHiddenInputOn: html
  106. hiddenInput := html input
  107. style: 'position: absolute; left: -100000px;';
  108. onBlur: [ self wrapper asJQuery removeClass: self focusClass ];
  109. onFocus: [ self wrapper asJQuery addClass: self focusClass ]
  110. !
  111. renderOn: html
  112. self registerBindings.
  113. self renderHiddenInputOn: html.
  114. wrapper := html div
  115. class: 'hl_widget';
  116. onClick: [ hiddenInput asJQuery focus ];
  117. with: [
  118. self renderContentOn: html ]
  119. ! !
  120. HLFocusableWidget subclass: #HLListWidget
  121. instanceVariableNames: 'items selectedItem'
  122. package: 'Helios-Core'!
  123. !HLListWidget methodsFor: 'accessing'!
  124. cssClassForItem: anObject
  125. ^ self selectedItem = anObject
  126. ifTrue: [ 'active' ]
  127. ifFalse: [ 'inactive' ]
  128. !
  129. iconForItem: anObject
  130. ^ ''
  131. !
  132. items
  133. ^ items ifNil: [ items := self defaultItems ]
  134. !
  135. items: aCollection
  136. items := aCollection
  137. !
  138. selectedItem
  139. ^ selectedItem
  140. !
  141. selectedItem: anObject
  142. selectedItem := anObject
  143. ! !
  144. !HLListWidget methodsFor: 'actions'!
  145. activateListItem: aListItem
  146. | parent position |
  147. (aListItem get: 0) ifNil: [ ^self ].
  148. <position = aListItem.parent().children().get().indexOf(aListItem.get(0)) + 1>.
  149. parent := aListItem parent.
  150. parent children removeClass: 'active'.
  151. aListItem addClass: 'active'.
  152. "Move the scrollbar to show the active element"
  153. aListItem position top < 0 ifTrue: [
  154. (parent get: 0) scrollTop: ((parent get: 0) scrollTop + aListItem position top - 10) ].
  155. aListItem position top + aListItem height > parent height ifTrue: [
  156. (parent get: 0) scrollTop: ((parent get: 0) scrollTop + aListItem height - (parent height - aListItem position top)) +10 ].
  157. "Activate the corresponding item"
  158. self selectItem: (self items at: (aListItem attr: 'list-data') asNumber)
  159. !
  160. selectItem: anObject
  161. self selectedItem: anObject
  162. ! !
  163. !HLListWidget methodsFor: 'defaults'!
  164. defaultItems
  165. ^ #()
  166. ! !
  167. !HLListWidget methodsFor: 'events'!
  168. setupKeyBindings
  169. | next |
  170. hiddenInput asJQuery unbind: 'keydown'.
  171. hiddenInput asJQuery keydown: [ :e | | selected |
  172. selected := window jQuery: '.focused .nav-pills .active'.
  173. e which = 38 ifTrue: [
  174. self activateListItem: selected prev ].
  175. e which = 40 ifTrue: [
  176. next := selected next.
  177. (next get: 0) ifNil: [ next := window jQuery: '.focused .nav-pills li:first-child' ].
  178. self activateListItem: next ] ]
  179. ! !
  180. !HLListWidget methodsFor: 'rendering'!
  181. renderButtonsOn: html
  182. !
  183. renderContentOn: html
  184. html ul
  185. class: 'nav nav-pills nav-stacked';
  186. with: [ self renderListOn: html ].
  187. html div class: 'pane_actions form-actions'; with: [
  188. self renderButtonsOn: html ].
  189. self setupKeyBindings
  190. !
  191. renderItem: anObject on: html
  192. | li |
  193. li := html li.
  194. li
  195. class: (self cssClassForItem: anObject);
  196. at: 'list-data' put: (self items indexOf: anObject) asString;
  197. with: [
  198. html a
  199. with: [
  200. (html tag: 'i') class: (self iconForItem: anObject).
  201. self renderItemLabel: anObject on: html ];
  202. onClick: [
  203. self activateListItem: li asJQuery ] ]
  204. !
  205. renderItemLabel: anObject on: html
  206. html with: anObject asString
  207. !
  208. renderListOn: html
  209. self items do: [ :each |
  210. self renderItem: each on: html ]
  211. ! !
  212. HLListWidget subclass: #HLNavigationListWidget
  213. instanceVariableNames: 'previous next'
  214. package: 'Helios-Core'!
  215. !HLNavigationListWidget methodsFor: 'accessing'!
  216. next
  217. ^ next
  218. !
  219. next: aWidget
  220. next := aWidget.
  221. aWidget previous = self ifFalse: [ aWidget previous: self ]
  222. !
  223. previous
  224. ^ previous
  225. !
  226. previous: aWidget
  227. previous := aWidget.
  228. aWidget next = self ifFalse: [ aWidget next: self ]
  229. ! !
  230. !HLNavigationListWidget methodsFor: 'actions'!
  231. nextFocus
  232. self next ifNotNil: [ self next focus ]
  233. !
  234. previousFocus
  235. self previous ifNotNil: [ self previous focus ]
  236. ! !
  237. !HLNavigationListWidget methodsFor: 'events'!
  238. setupKeyBindings
  239. super setupKeyBindings.
  240. hiddenInput asJQuery keydown: [ :e |
  241. e which = 39 ifTrue: [
  242. self nextFocus ].
  243. e which = 37 ifTrue: [
  244. self previousFocus ] ]
  245. ! !
  246. HLWidget subclass: #HLManager
  247. instanceVariableNames: 'tabs activeTab keyBinder environment'
  248. package: 'Helios-Core'!
  249. !HLManager methodsFor: 'accessing'!
  250. activeTab
  251. ^ activeTab
  252. !
  253. environment
  254. "The default environment used by all Helios objects"
  255. ^ environment ifNil: [ environment := self defaultEnvironment ]
  256. !
  257. environment: anEnvironment
  258. environment := anEnvironment
  259. !
  260. keyBinder
  261. ^ keyBinder ifNil: [ keyBinder := HLKeyBinder new ]
  262. !
  263. tabs
  264. ^ tabs ifNil: [ tabs := OrderedCollection new ]
  265. ! !
  266. !HLManager methodsFor: 'actions'!
  267. activate: aTab
  268. self keyBinder flushBindings.
  269. activeTab := aTab.
  270. self
  271. refresh;
  272. show: aTab
  273. !
  274. addTab: aTab
  275. self tabs add: aTab.
  276. self activate: aTab
  277. !
  278. removeTab: aTab
  279. "Todo: activate the previously activated tab. Keep a history of tabs selection"
  280. (self tabs includes: aTab) ifFalse: [ ^ self ].
  281. self tabs remove: aTab.
  282. self refresh
  283. ! !
  284. !HLManager methodsFor: 'defaults'!
  285. defaultEnvironment
  286. ^ HLLocalEnvironment new
  287. ! !
  288. !HLManager methodsFor: 'initialization'!
  289. initialize
  290. super initialize.
  291. self keyBinder setupEvents
  292. ! !
  293. !HLManager methodsFor: 'rendering'!
  294. refresh
  295. (window jQuery: '.navbar') remove.
  296. (window jQuery: '#container') remove.
  297. self appendToJQuery: 'body' asJQuery
  298. !
  299. renderAddOn: html
  300. html li
  301. class: 'dropdown';
  302. with: [
  303. html a
  304. class: 'dropdown-toggle';
  305. at: 'data-toggle' put: 'dropdown';
  306. with: [
  307. html with: 'Open...'.
  308. (html tag: 'b') class: 'caret' ].
  309. html ul
  310. class: 'dropdown-menu';
  311. with: [
  312. ((HLWidget withAllSubclasses
  313. select: [ :each | each canBeOpenAsTab ])
  314. sorted: [ :a :b | a tabPriority < b tabPriority ])
  315. do: [ :each |
  316. html li with: [
  317. html a
  318. with: each tabLabel;
  319. onClick: [ each openAsTab ] ] ] ] ]
  320. !
  321. renderContentOn: html
  322. html div
  323. class: 'navbar navbar-fixed-top';
  324. with: [ html div
  325. class: 'navbar-inner';
  326. with: [ self renderTabsOn: html ] ].
  327. html div id: 'container'
  328. !
  329. renderTabsOn: html
  330. html ul
  331. class: 'nav';
  332. with: [
  333. self tabs do: [ :each |
  334. html li
  335. class: (each isActive ifTrue: [ 'active' ] ifFalse: [ 'inactive' ]);
  336. with: [
  337. html a
  338. with: [
  339. ((html tag: 'i') class: 'icon-remove-circle')
  340. onClick: [ self removeTab: each ].
  341. html with: each label ];
  342. onClick: [ each activate ] ] ].
  343. self renderAddOn: html ]
  344. !
  345. show: aTab
  346. (window jQuery: '#container') empty.
  347. aTab widget appendToJQuery: '#container' asJQuery
  348. ! !
  349. HLManager class instanceVariableNames: 'current'!
  350. !HLManager class methodsFor: 'accessing'!
  351. current
  352. ^ current ifNil: [ current := self basicNew initialize ]
  353. ! !
  354. !HLManager class methodsFor: 'initialization'!
  355. initialize
  356. self current appendToJQuery: 'body' asJQuery
  357. ! !
  358. !HLManager class methodsFor: 'instance creation'!
  359. new
  360. "Use current instead"
  361. self shouldNotImplement
  362. ! !
  363. HLWidget subclass: #HLSUnit
  364. instanceVariableNames: ''
  365. package: 'Helios-Core'!
  366. !HLSUnit class methodsFor: 'accessing'!
  367. tabLabel
  368. ^ 'SUnit'
  369. !
  370. tabPriority
  371. ^ 1000
  372. ! !
  373. !HLSUnit class methodsFor: 'testing'!
  374. canBeOpenAsTab
  375. ^ true
  376. ! !
  377. HLWidget subclass: #HLTranscript
  378. instanceVariableNames: ''
  379. package: 'Helios-Core'!
  380. !HLTranscript class methodsFor: 'accessing'!
  381. tabLabel
  382. ^ 'Transcript'
  383. !
  384. tabPriority
  385. ^ 600
  386. ! !
  387. !HLTranscript class methodsFor: 'testing'!
  388. canBeOpenAsTab
  389. ^ true
  390. ! !