Helios-Core.st 11 KB

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