resin 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (83) hide show
  1. data/README.markdown +52 -0
  2. data/amber/css/amber.css +519 -0
  3. data/amber/css/documentation.css +84 -0
  4. data/amber/css/profstef.css +75 -0
  5. data/amber/css/style.css +313 -0
  6. data/amber/images/amber.png +0 -0
  7. data/amber/images/amber_small.png +0 -0
  8. data/amber/images/off.png +0 -0
  9. data/amber/images/offHover.png +0 -0
  10. data/amber/images/presentation.png +0 -0
  11. data/amber/images/profstef.png +0 -0
  12. data/amber/images/sprite.png +0 -0
  13. data/amber/images/tinylogo.png +0 -0
  14. data/amber/images/twitterwall.png +0 -0
  15. data/amber/js/Additional-Examples.deploy.js +15 -0
  16. data/amber/js/Additional-Examples.js +21 -0
  17. data/amber/js/Benchfib.deploy.js +132 -0
  18. data/amber/js/Benchfib.js +167 -0
  19. data/amber/js/Canvas.deploy.js +1304 -0
  20. data/amber/js/Canvas.js +1885 -0
  21. data/amber/js/Compiler.deploy.js +1871 -0
  22. data/amber/js/Compiler.js +2616 -0
  23. data/amber/js/Documentation.deploy.js +961 -0
  24. data/amber/js/Documentation.js +1376 -0
  25. data/amber/js/Examples.deploy.js +53 -0
  26. data/amber/js/Examples.js +73 -0
  27. data/amber/js/IDE.deploy.js +3468 -0
  28. data/amber/js/IDE.js +4883 -0
  29. data/amber/js/Kernel-Announcements.deploy.js +107 -0
  30. data/amber/js/Kernel-Announcements.js +152 -0
  31. data/amber/js/Kernel-Classes.deploy.js +675 -0
  32. data/amber/js/Kernel-Classes.js +956 -0
  33. data/amber/js/Kernel-Collections.deploy.js +3273 -0
  34. data/amber/js/Kernel-Collections.js +4644 -0
  35. data/amber/js/Kernel-Exceptions.deploy.js +244 -0
  36. data/amber/js/Kernel-Exceptions.js +349 -0
  37. data/amber/js/Kernel-Methods.deploy.js +510 -0
  38. data/amber/js/Kernel-Methods.js +739 -0
  39. data/amber/js/Kernel-Objects.deploy.js +2698 -0
  40. data/amber/js/Kernel-Objects.js +3858 -0
  41. data/amber/js/Kernel-Tests.deploy.js +1419 -0
  42. data/amber/js/Kernel-Tests.js +1929 -0
  43. data/amber/js/Kernel-Transcript.deploy.js +142 -0
  44. data/amber/js/Kernel-Transcript.js +202 -0
  45. data/amber/js/SUnit.deploy.js +351 -0
  46. data/amber/js/SUnit.js +501 -0
  47. data/amber/js/amber.js +250 -0
  48. data/amber/js/boot.js +587 -0
  49. data/amber/js/compat.js +22 -0
  50. data/amber/js/init.js +8 -0
  51. data/amber/js/lib/CodeMirror/LICENSE +19 -0
  52. data/amber/js/lib/CodeMirror/amber.css +21 -0
  53. data/amber/js/lib/CodeMirror/codemirror.css +67 -0
  54. data/amber/js/lib/CodeMirror/codemirror.js +2144 -0
  55. data/amber/js/lib/CodeMirror/smalltalk.js +134 -0
  56. data/amber/js/lib/jQuery/jquery-1.4.4.min.js +167 -0
  57. data/amber/js/lib/jQuery/jquery-1.6.4.min.js +4 -0
  58. data/amber/js/lib/jQuery/jquery-ui-1.8.16.custom.min.js +791 -0
  59. data/amber/js/lib/jQuery/jquery.textarea.js +267 -0
  60. data/amber/js/lib/peg-0.6.2.min.js +2 -0
  61. data/amber/js/lib/showdown.js +419 -0
  62. data/amber/js/parser.js +4005 -0
  63. data/amber/js/parser.pegjs +220 -0
  64. data/amber/st/Benchfib.st +124 -0
  65. data/amber/st/Canvas.st +556 -0
  66. data/amber/st/Compiler.st +1425 -0
  67. data/amber/st/Documentation.st +758 -0
  68. data/amber/st/Examples.st +38 -0
  69. data/amber/st/IDE.st +2336 -0
  70. data/amber/st/Kernel-Announcements.st +61 -0
  71. data/amber/st/Kernel-Classes.st +403 -0
  72. data/amber/st/Kernel-Collections.st +1673 -0
  73. data/amber/st/Kernel-Exceptions.st +124 -0
  74. data/amber/st/Kernel-Methods.st +287 -0
  75. data/amber/st/Kernel-Objects.st +1489 -0
  76. data/amber/st/Kernel-Tests.st +892 -0
  77. data/amber/st/Kernel-Transcript.st +70 -0
  78. data/amber/st/SUnit.st +172 -0
  79. data/bin/runresin +12 -0
  80. data/lib/resin.rb +0 -0
  81. data/lib/resin/app/app.rb +121 -0
  82. data/lib/resin/app/views/index.haml +10 -0
  83. metadata +216 -0
@@ -0,0 +1,758 @@
1
+ Smalltalk current createPackage: 'Documentation' properties: #{}!
2
+ Object subclass: #DocumentationBuilder
3
+ instanceVariableNames: 'chapters announcer widget'
4
+ category: 'Documentation'!
5
+
6
+ !DocumentationBuilder methodsFor: 'accessing'!
7
+
8
+ chapters
9
+ ^chapters ifNil: [chapters := self buildChapters]
10
+ !
11
+
12
+ announcer
13
+ ^announcer ifNil: [announcer := Announcer new]
14
+ !
15
+
16
+ widget
17
+ ^widget ifNil: [widget := DocumentationWidget on: self]
18
+ ! !
19
+
20
+ !DocumentationBuilder methodsFor: 'building'!
21
+
22
+ buildChapters
23
+ ^((self class methodDictionary values sorted: [:a :b | a selector < b selector])
24
+ select: [:each | each category = 'chapters'])
25
+ collect: [:each | self perform: each selector]
26
+ !
27
+
28
+ buildOn: aCanvas
29
+ aCanvas with: self widget.
30
+ self
31
+ checkHashChange;
32
+ checkHash
33
+ !
34
+
35
+ buildOnJQuery: aJQuery
36
+ self buildOn: (HTMLCanvas onJQuery: aJQuery)
37
+ !
38
+
39
+ build
40
+ self buildOnJQuery: ('body' asJQuery)
41
+ ! !
42
+
43
+ !DocumentationBuilder methodsFor: 'chapters'!
44
+
45
+ ch1introduction
46
+ ^DocChapter new
47
+ title: 'Introduction';
48
+ contents: '
49
+
50
+ ##Amber Smalltalk in a nutshell
51
+
52
+ Amber is an implementation of the Smalltalk-80 language. It is designed to make client-side web development **faster, easier and more fun** as it allows developers to write HTML5 applications in a live Smalltalk environment!!
53
+
54
+ Amber is written in itself, including the IDE and the compiler and it runs **directly inside your browser**. The IDE is fairly complete with a class browser, workspace, transcript, unit test runner, object inspectors, cross reference tools and even a debugger.
55
+
56
+ Noteworthy features:
57
+
58
+ - Amber is semantically and syntactically very close to [Pharo Smalltalk](http://www.pharo-project.org). Pharo is considered the reference implementation.
59
+ - Amber **seamlessly interacts with JavaScript** and can use its full eco system of libraries without any glue code needed.
60
+ - Amber **has no dependencies** and can be used in any JavaScript runtime, not only inside browsers. An important example is [Node.js](http://nodejs.org).
61
+ - Amber is a live Smalltalk that **compiles incrementally into efficient JavaScript** often mapping one-to-one with JavaScript equivalents.
62
+ - Amber has a **Seaside influenced canvas library** to dynamically generate HTML.
63
+
64
+ ## Arguments for using Amber
65
+ In our humble opinion the main arguments for using Amber are:
66
+
67
+ - JavaScript is quite a broken language with lots of traps and odd quirks. It is the assembler of the Internet which is cool, but we don''t want to write in it.
68
+ - Smalltalk as a language is immensely cleaner and more mature, both syntactically and semantically.
69
+ - Smalltalk has a simple class model with a lightweight syntax for closures, it is in many ways a perfect match for the Good Parts of JavaScript.
70
+ - Having a true live interactive incremental development environment where you can build your application directly in the browser is unbeatable.
71
+
72
+ ## Disclaimer
73
+
74
+ This documentation doesn''t aim to teach Smalltalk.
75
+ Knowledge of Smalltalk is needed to understand the topics covered in this documentation.
76
+ If you want to learn the Smalltalk language, you can read the excellent [Pharo By Example](http://www.pharobyexample.org) book.
77
+ '
78
+ !
79
+
80
+ ch2differencesWithOtherSmalltalks
81
+ ^DocChapter new
82
+ title: 'Differences with other Smalltalks';
83
+ contents: '
84
+ Amber has some differences with other Smalltalk implementations. This makes porting code a non-trivial thing, but still quite manageable.
85
+ Because it maps Smalltalk constructs one-to-one with the JavaScript equivalent, including Smalltalk classes to JavaScript constructors, the core class library is simplified compared to Pharo Smalltalk.
86
+ And since we want Amber to be useful in building lean browser apps we can''t let it bloat too much.
87
+
88
+ But apart from missing things other Smalltalks may have, there are also things that are plain different:
89
+
90
+ - The collection class hierarchy is much simpler compared to most Smalltalk implementations. In part this is because we want to map reasonably well with JavaScript counter parts.
91
+ - As of today, there is no SortedCollection. The size of arrays is dynamic, and they behave like an ordered collection. They can also be sorted with the `#sort*` methods.
92
+ - The `Date` class behaves like the `Date` and `TimeStamp` classes in Pharo Smalltalk. Therefore both `Date today` and `Date now` are valid in Amber.
93
+ - Amber does not have class Character, but `String` does implement some of Character behavior so a single character String can work as a Character.
94
+ - Amber does support **class instance variables**, but not class variables.
95
+ - Amber only has global classes and packages, but not arbitrary objects. Use classes instead like `Smalltalk current` instead of `Smalltalk` etc.
96
+ - Amber does not support pool dictionaries.
97
+ - Amber uses **< ...javascript code... >** to inline JavaScript code and does not have pragmas.
98
+ - Amber does not have class categories. The left side in the browser lists real Packages, but they feel much the same.
99
+ '
100
+ !
101
+
102
+ ch3GettingStarted
103
+ ^DocChapter new
104
+ title: 'Getting started';
105
+ contents: '
106
+ To get started hacking in Amber you can basically take three routes, independent of your platform:
107
+
108
+ 1. Just **try it out directly** at [www.amber-lang.net](http://www.amber-lang.net) - click the **Class browser** button there. But you will **not be able to save any code you write**!!
109
+ Still, it works fine for looking at the IDE and playing around. Just **don''t press F5/reload** - it will lose any code you have written.
110
+ 2. Download an Amber zip-ball, install [Nodejs](http://www.nodejs.org), fire up the Amber server and then open Amber from localhost - then you **can save code**. Detailed instructions are below!!
111
+ 3. Same as above but install git first and get a proper clone from [http://github.com/NicolasPetton/amber](http://github.com/NicolasPetton/amber) instead of a zip/tar-ball.
112
+ If you want to **contribute to Amber itself** this is really what you want to do. In fact, in most cases this is what you want to do. It requires installing git first, but it is quite simple - although we leave this bit as an "exercise to the reader" :)
113
+
114
+ **PLEASE NOTE:** Amber core developers use Linux.
115
+ We do not want to introduce dependencies that aren''t cross platform - but currently amberc (the command line compiler) is a bash script and we also use Makefiles
116
+ (for building Amber itself and server side examples) written on Linux/Unix. So using Windows is currently a bit limited - you can''t run "make" in the .st directory to rebuild whole of Amber for example.
117
+ BUT... if you only want to use Amber to build web client apps and not really get involved in hacking Amber itself - then you should be fine!!
118
+
119
+ ## Downloading Amber
120
+ Currently you can download in zip or tar-ball format, either cutting edge or a release. [Downloads are available here](https://github.com/NicolasPetton/amber/archives/amber).
121
+
122
+ Unpack wherever you like, but I would rename the directory that is unpacked to something slightly shorter - like say "amber". :)
123
+ And yes, at this point you can double click the index.html file in the amber directory to get the IDE up, but again, **you will not be able to save code**. So please continue below :)
124
+
125
+ ## Installing Node.js
126
+ [Node](http://www.nodejs.org) (for short) is simply the V8 Javascript VM from Google (used in Chrome) hooked together with some hard core C-libraries for doing "evented I/O".
127
+ Basically it''s JavaScript for the server - on asynch steroids. Amber runs fine in Node and we use it for several Amber tools, like amberc (the command line Amber compiler) or the Amber server (see below).
128
+ There are also several Amber-Node examples to look at if you want to play with running Amber programs server side. **In short - you really want to install Nodejs. :)**
129
+
130
+ - Installing Node on Linux can be done using your package tool of choice (`apt-get install nodejs` for example) or any other way described at [the download page](http://nodejs.org/#download).
131
+ - Installing Node on MacOS or Windows is probably done best by using the [installers available at Nodejs.org](http://nodejs.org/#download).
132
+
133
+ ## Starting Amber server
134
+ Nicolas has written a minimal webDAV server that is the easiest way to get up and running Amber with the ability to save code. This little server is written in... Amber!!
135
+ And it runs on top of Node. So to start it up serving your brand new directory tree of sweet Amber you do:
136
+
137
+ cd amber (or whatever you called the directory you unpackaged)
138
+ ./bin/server (in windows you type `node server\server.js` instead)
139
+
140
+ It should say it is listening on port 4000. If it does, hooray!! That means both Node and Amber are good. In Windows you might get a question about opening that port in the local firewall - yep, do it!!
141
+
142
+ ## Firing up Amber
143
+ The Amber IDE is written in... Amber. It uses [jQuery](http://jquery.com) and runs right in your browser as a ... well, a web page.
144
+ We could open it up just using a file url - but the reason we performed the previous steps is so that we can load the IDE web page from a server that can handle PUTs (webDAV) of source code.
145
+ According to web security Amber can only do PUT back to the same server it was loaded from. Thus we instead want to open it [through our little server now listening on port 4000](http://localhost:4000/index.html).
146
+ Clicking that link and then pressing the **Class browser** should get your Amber IDE running with the ability to commit modified packages locally.
147
+
148
+ To verify that you can indeed commit now - just select a Package in the browser, like say "Examples" and press the **Commit** button below. **If all goes well nothing happens :)**.
149
+ So in order to really know if it worked we can check the modified date on the files **amber/st/Examples.st**, **amber/js/Examples.js** and **amber/js/Examples.deploy.js** - they should be brand new.
150
+
151
+ NOTE: We can use any webDAV server and Apache2 has been used earlier and works fine. But the Amber server is smaller and simpler to start.
152
+ '
153
+ !
154
+
155
+ ch5Index
156
+ ^ClassesIndexChapter new
157
+ !
158
+
159
+ ch6KernelObjects
160
+ ^PackageDocChapter on: (Package named: 'Kernel-Objects')
161
+ !
162
+
163
+ ch7KernelClasses
164
+ ^PackageDocChapter on: (Package named: 'Kernel-Classes')
165
+ !
166
+
167
+ ch4Tutorials
168
+ ^TutorialsChapter new
169
+ !
170
+
171
+ ch8KernelCollection
172
+ ^PackageDocChapter on: (Package named: 'Kernel-Collections')
173
+ !
174
+
175
+ ch9KernelMethods
176
+ ^PackageDocChapter on: (Package named: 'Kernel-Methods')
177
+ ! !
178
+
179
+ !DocumentationBuilder methodsFor: 'routing'!
180
+
181
+ checkHashChange
182
+ (window jQuery: window) bind: 'hashchange' do: [self checkHash]
183
+ !
184
+
185
+ checkHash
186
+ | hash presentation |
187
+ hash := document location hash replace: '^#' with: ''.
188
+ self announcer announce: (ChapterSelectionAnnouncement new
189
+ id: hash;
190
+ yourself)
191
+ ! !
192
+
193
+ !DocumentationBuilder methodsFor: 'updating'!
194
+
195
+ update
196
+ chapters := nil.
197
+ announcer := nil.
198
+ widget := nil.
199
+ (window jQuery: '.documentation') remove.
200
+ self build
201
+ ! !
202
+
203
+ DocumentationBuilder class instanceVariableNames: 'current'!
204
+
205
+ !DocumentationBuilder class methodsFor: 'accessing'!
206
+
207
+ current
208
+ ^current ifNil: [current := self new]
209
+ ! !
210
+
211
+ !DocumentationBuilder class methodsFor: 'initialization'!
212
+
213
+ initialize
214
+ self current build
215
+ ! !
216
+
217
+ Widget subclass: #DocChapter
218
+ instanceVariableNames: 'title contents parent'
219
+ category: 'Documentation'!
220
+
221
+ !DocChapter methodsFor: 'accessing'!
222
+
223
+ title
224
+ ^title ifNil: ['']
225
+ !
226
+
227
+ title: aString
228
+ title := aString
229
+ !
230
+
231
+ contents
232
+ ^contents ifNil: ['']
233
+ !
234
+
235
+ contents: aString
236
+ contents := aString
237
+ !
238
+
239
+ htmlContents
240
+ ^(Showdown at: #converter) new makeHtml: self contents
241
+ !
242
+
243
+ chapters
244
+ "A doc chapter can contain sub chapters"
245
+ ^#()
246
+ !
247
+
248
+ cssClass
249
+ ^'doc_chapter'
250
+ !
251
+
252
+ level
253
+ ^self parent ifNil: [1] ifNotNil: [self parent level +1]
254
+ !
255
+
256
+ level: anInteger
257
+ level := anInteger
258
+ !
259
+
260
+ parent
261
+ ^parent
262
+ !
263
+
264
+ parent: aChapter
265
+ parent := aChapter
266
+ !
267
+
268
+ id
269
+ "The id is used in url fragments.
270
+ It must be unique amoung all chapters"
271
+ ^self title replace: ' ' with: '-'
272
+ !
273
+
274
+ announcer
275
+ ^DocumentationBuilder current announcer
276
+ ! !
277
+
278
+ !DocChapter methodsFor: 'actions'!
279
+
280
+ selectClass: aClass
281
+ DocumentationBuilder current announcer announce: (ClassSelectionAnnouncement on: aClass)
282
+ !
283
+
284
+ selectChapter: aChapter
285
+ document location hash: aChapter id
286
+ !
287
+
288
+ displayChapter: aChapter
289
+ DocumentationBuilder current widget displayChapter: aChapter
290
+ ! !
291
+
292
+ !DocChapter methodsFor: 'initialization'!
293
+
294
+ initialize
295
+ super initialize.
296
+ self subscribe
297
+ ! !
298
+
299
+ !DocChapter methodsFor: 'rendering'!
300
+
301
+ renderOn: html
302
+ html div
303
+ class: self cssClass;
304
+ with: [
305
+ self renderDocOn: html.
306
+ self renderLinksOn: html]
307
+ !
308
+
309
+ renderDocOn: html
310
+ | div |
311
+ html h1 with: self title.
312
+ self renderNavigationOn: html.
313
+ div := html div class: 'contents'.
314
+ div asJQuery html: self htmlContents
315
+ !
316
+
317
+ renderNavigationOn: html
318
+ self parent ifNotNil: [
319
+ html div
320
+ class: 'navigation'; with: [
321
+ html a
322
+ with: '← back to ', self parent title;
323
+ onClick: [self selectChapter: self parent]]]
324
+ !
325
+
326
+ renderLinksOn: html
327
+ html ul
328
+ class: 'links';
329
+ with: [
330
+ self chapters do: [:each |
331
+ html li with: [
332
+ html a
333
+ with: each title;
334
+ onClick: [self selectChapter: each]]]]
335
+ ! !
336
+
337
+ !DocChapter methodsFor: 'subscriptions'!
338
+
339
+ subscribe
340
+ self announcer on: ChapterSelectionAnnouncement do: [:ann |
341
+ ann id = self id ifTrue: [self displayChapter: self]]
342
+ ! !
343
+
344
+ DocChapter subclass: #PackageDocChapter
345
+ instanceVariableNames: 'package chapters'
346
+ category: 'Documentation'!
347
+
348
+ !PackageDocChapter methodsFor: 'accessing'!
349
+
350
+ package
351
+ ^package
352
+ !
353
+
354
+ title
355
+ ^'Package ', self package name
356
+ !
357
+
358
+ chapters
359
+ ^chapters
360
+ !
361
+
362
+ contents
363
+ ^'Classes in package ', self package name, ':'
364
+ ! !
365
+
366
+ !PackageDocChapter methodsFor: 'initialization'!
367
+
368
+ initializeWithPackage: aPackage
369
+ package := aPackage.
370
+ chapters := (aPackage classes sorted: [:a :b | a name < b name]) collect: [:each |
371
+ (ClassDocChapter on: each)
372
+ parent: self;
373
+ yourself]
374
+ ! !
375
+
376
+ !PackageDocChapter class methodsFor: 'instance creation'!
377
+
378
+ on: aPackage
379
+ ^self basicNew
380
+ initializeWithPackage: aPackage;
381
+ initialize;
382
+ yourself
383
+ ! !
384
+
385
+ DocChapter subclass: #ClassDocChapter
386
+ instanceVariableNames: 'theClass'
387
+ category: 'Documentation'!
388
+
389
+ !ClassDocChapter methodsFor: 'accessing'!
390
+
391
+ theClass
392
+ ^theClass
393
+ !
394
+
395
+ contents
396
+ ^self theClass comment isEmpty
397
+ ifTrue: [self theClass name, ' is not documented yet.']
398
+ ifFalse: [self theClass comment]
399
+ !
400
+
401
+ cssClass
402
+ ^'doc_class ', super cssClass
403
+ !
404
+
405
+ title
406
+ ^self theClass name
407
+ !
408
+
409
+ initializeWithClass: aClass
410
+ theClass := aClass
411
+ ! !
412
+
413
+ !ClassDocChapter methodsFor: 'rendering'!
414
+
415
+ renderLinksOn: html
416
+ html ul
417
+ class: 'links';
418
+ with: [
419
+ html li with: [html a
420
+ with: 'Browse this class';
421
+ onClick: [Browser openOn: self theClass]]]
422
+ ! !
423
+
424
+ !ClassDocChapter methodsFor: 'subscriptions'!
425
+
426
+ subscribe
427
+ super subscribe.
428
+ self announcer
429
+ on: ClassSelectionAnnouncement do: [:ann |
430
+ ann theClass = self theClass ifTrue: [
431
+ self selectChapter: self]]
432
+ ! !
433
+
434
+ !ClassDocChapter class methodsFor: 'accessing'!
435
+
436
+ on: aClass
437
+ ^self basicNew
438
+ initializeWithClass: aClass;
439
+ initialize;
440
+ yourself
441
+ ! !
442
+
443
+ Widget subclass: #DocumentationWidget
444
+ instanceVariableNames: 'builder selectedChapter chapterDiv'
445
+ category: 'Documentation'!
446
+
447
+ !DocumentationWidget methodsFor: 'accessing'!
448
+
449
+ builder
450
+ ^builder
451
+ !
452
+
453
+ builder: aDocumentationBuilder
454
+ builder := aDocumentationBuilder
455
+ !
456
+
457
+ chapters
458
+ ^self builder chapters
459
+ !
460
+
461
+ selectedChapter
462
+ ^selectedChapter ifNil: [selectedChapter := self chapters first]
463
+ !
464
+
465
+ selectedChapter: aChapter
466
+ ^selectedChapter := aChapter
467
+ ! !
468
+
469
+ !DocumentationWidget methodsFor: 'actions'!
470
+
471
+ displayChapter: aChapter
472
+ self selectedChapter: aChapter.
473
+ self updateChapterDiv
474
+ !
475
+
476
+ selectChapter: aChapter
477
+ document location hash: aChapter id
478
+ ! !
479
+
480
+ !DocumentationWidget methodsFor: 'rendering'!
481
+
482
+ renderOn: html
483
+ html div
484
+ class: 'documentation';
485
+ with: [
486
+ self renderMenuOn: html.
487
+ chapterDiv := html div.
488
+ self updateChapterDiv]
489
+ !
490
+
491
+ renderMenuOn: html
492
+ html div
493
+ class: 'menu';
494
+ with: [
495
+ html ol with: [
496
+ self chapters do: [:each |
497
+ html li with: [
498
+ self renderChapterMenu: each on: html]]]]
499
+ !
500
+
501
+ renderChapterMenu: aChapter on: html
502
+ html a
503
+ with: aChapter title;
504
+ onClick: [
505
+ self selectChapter: aChapter].
506
+ html ol with: [
507
+ aChapter chapters do: [:each |
508
+ html li with: [
509
+ self renderChapterMenu: each on: html]]]
510
+ ! !
511
+
512
+ !DocumentationWidget methodsFor: 'updating'!
513
+
514
+ updateChapterDiv
515
+ chapterDiv contents: [:html |
516
+ html with: self selectedChapter]
517
+ ! !
518
+
519
+ !DocumentationWidget class methodsFor: 'instance creation'!
520
+
521
+ on: aBuilder
522
+ ^self new
523
+ builder: aBuilder;
524
+ yourself
525
+ ! !
526
+
527
+ DocChapter subclass: #ClassesIndexChapter
528
+ instanceVariableNames: ''
529
+ category: 'Documentation'!
530
+
531
+ !ClassesIndexChapter methodsFor: 'accessing'!
532
+
533
+ cssClass
534
+ ^'index_doc ', super cssClass
535
+ !
536
+
537
+ title
538
+ ^'Smalltalk classes by index'
539
+ !
540
+
541
+ alphabet
542
+ ^'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
543
+ ! !
544
+
545
+ !ClassesIndexChapter methodsFor: 'rendering'!
546
+
547
+ renderDocOn: html
548
+ html h1 with: self title.
549
+ self alphabet do: [:letter || classes |
550
+ classes := Smalltalk current classes select: [:each | each name first = letter].
551
+ classes ifNotEmpty: [html h2 with: letter].
552
+ html ul with: [
553
+ (classes sorted: [:a :b | a name < b name])
554
+ do: [:each |
555
+ html li with: [html a
556
+ with: each name;
557
+ onClick: [self selectClass: each]]]]]
558
+ ! !
559
+
560
+ Object subclass: #ClassSelectionAnnouncement
561
+ instanceVariableNames: 'theClass'
562
+ category: 'Documentation'!
563
+
564
+ !ClassSelectionAnnouncement methodsFor: 'accessing'!
565
+
566
+ theClass
567
+ ^theClass
568
+ !
569
+
570
+ theClass: aClass
571
+ theClass := aClass
572
+ ! !
573
+
574
+ !ClassSelectionAnnouncement class methodsFor: 'instance creation'!
575
+
576
+ on: aClass
577
+ ^self new
578
+ theClass: aClass;
579
+ yourself
580
+ ! !
581
+
582
+ Object subclass: #ChapterSelectionAnnouncement
583
+ instanceVariableNames: 'id'
584
+ category: 'Documentation'!
585
+
586
+ !ChapterSelectionAnnouncement methodsFor: 'accessing'!
587
+
588
+ id
589
+ ^id
590
+ !
591
+
592
+ id: aString
593
+ id := aString
594
+ ! !
595
+
596
+ DocChapter subclass: #TutorialsChapter
597
+ instanceVariableNames: ''
598
+ category: 'Documentation'!
599
+
600
+ !TutorialsChapter methodsFor: 'accessing'!
601
+
602
+ title
603
+ ^'Tutorials'
604
+ !
605
+
606
+ contents
607
+ ^'Here''s a serie of tutorials. If you are new to Smalltalk, you can also learn Amber online with [ProfStef](http://www.amber-lang.net/learn.html)'
608
+ !
609
+
610
+ chapters
611
+ ^{ self firstAppChapter. self counterChapter }
612
+ !
613
+
614
+ firstAppChapter
615
+ ^DocChapter new
616
+ title: 'A first application';
617
+ contents: '
618
+
619
+ Let''s make Hello World in Amber.
620
+
621
+ First, you need a place for your new project. I made a new directory under amber:
622
+
623
+ amber/projects/hello
624
+
625
+ This will store your project files. To get started, add a new index.html file to this folder, as well as empty js and st folders.
626
+
627
+ Your index.html can be really basic. The most important thing it does is include amber.js and run loadAmber. Here is a basic index.html you can use:
628
+
629
+
630
+ <!!DOCTYPE html>
631
+ <html>
632
+ <head>
633
+ <title>My First Amber Project</title>
634
+ <script src="../../js/amber.js" type="text/javascript"></script>
635
+ <script type="text/javascript">
636
+ loadAmber({
637
+ files: [],
638
+ prefix: ''projects/hello/js'',
639
+ ready: function() {
640
+
641
+ }});
642
+ </script>
643
+ </head>
644
+ <body>
645
+ <article>
646
+ <h1>My First Amber Project</h1>
647
+ <button onclick="smalltalk.Browser._open()">class browser</button>
648
+ <button id="sayHello">say hello</button>
649
+ </article>
650
+ </body>
651
+ </html>
652
+
653
+ Now start up amber with node.js and navigate to http://localhost:4000/projects/hello/index.html
654
+
655
+ It''s boring so far, so lets write some code. Click the button to open the class browser. Find an existing class and change its name to Hello and its package to HelloApp.
656
+ Then click save. This creates a new class and leaves the old one intact, it doesn''t overwrite it. Your class will look like this:
657
+
658
+ Object subclass: #Hello
659
+ instanceVariableNames: ''''
660
+ package: ''HelloApp''
661
+
662
+ Now click save and navigate to your new class in its new package.
663
+ Then click ''commit package''. You just created a new class and saved your work.
664
+ On your file system check out your js and st folders. Your new class is now saved in both JavaScript and Smalltalk.
665
+
666
+ Now, refresh your browser page and reopen the class browser. Oh no, your new class is gone!! To load your new class automatically, you have to add it in index.html. Make your JavaScript look like this:
667
+
668
+
669
+ loadAmber({
670
+ files: [''HelloApp.js''],
671
+ prefix: ''projects/hello/js'',
672
+ ready: function() {
673
+ }});
674
+
675
+ Save and refresh again. Now your class is loaded and shows up in the class browser.
676
+
677
+ Now, let''s make this class do something. Create a new message in the class browser by navigating to your class, then clicking ''not yet classified'' and fill in a simple message. Try this for example:
678
+
679
+ begin
680
+ "Makes me say hello to the user."
681
+
682
+ | msg button |
683
+ msg := ''Hello world!!''.
684
+ button := ''#sayHello'' asJQuery.
685
+ button click: [button after: ''<p>'' , msg , ''</p>''].
686
+
687
+ Your message isn''t too helpful if it doesn''t get called. Save it, commit the package, then edit index.html again. You can write JavaScript code that sends a message to Smalltalk:
688
+
689
+ loadAmber({
690
+ files: [''HelloApp.js''],
691
+ prefix: ''projects/hello/js'', // path for js files i think
692
+ ready: function() {
693
+ $(function() {
694
+ smalltalk.Hello._new()._begin();
695
+ });
696
+ }});
697
+
698
+ From there, you can create new Smalltalk classes and messages to build up your app. Enjoy!!
699
+ '
700
+ !
701
+
702
+ counterChapter
703
+ ^DocChapter new
704
+ title: 'The counter application';
705
+ contents: '
706
+
707
+ This tutorial will teach you how to build HTML with Amber using jQuery and the HTMLCanvas API. It is freely adapted from
708
+ the [Seaside counter example](http://www.seaside.st/about/examples/counter)
709
+
710
+ ##The counter widget
711
+
712
+ The counter is the most basic example of a widget. It allows to increment and decrement a number by clicking a button.
713
+
714
+ Amber already comes with a counter example in the `Examples` package. To avoid class name conflict, we''ll name our counter class `TCounter`.
715
+
716
+ Widget subclass: #TCounter
717
+ instanceVariableNames: ''count header''
718
+ package: ''Tutorials''
719
+
720
+ The first method is used to initialize the component with the default state, in this case we set the counter to 0:
721
+
722
+ initialize
723
+ super initialize.
724
+ count := 0
725
+
726
+ The method used for rendering a widget is `#renderOn:`. It takes an instance of HTMLCanvas as parameter.
727
+ The `header` h1 kept as an instance variable, so when the count value change, we can update it''s contents accordingly.
728
+
729
+ renderOn: html
730
+ header := html h1
731
+ with: count asString;
732
+ yourself.
733
+ html button
734
+ with: ''++'';
735
+ onClick: [self increase].
736
+ html button
737
+ with: ''--'';
738
+ onClick: [self decrease]
739
+
740
+ The counter is almost ready. All we need now is to implement the two action methods `#increase` and `#decrease` to change the state
741
+ of our counter and update its header.
742
+
743
+ increase
744
+ count := count + 1.
745
+ header contents: [:html | html with: count asString]
746
+
747
+ decrease
748
+ count := count - 1.
749
+ header contents: [:html | html with: count asString]
750
+
751
+
752
+ That''s it!! We can now display an instance of TCounter by rendering it on the page using jQuery:
753
+
754
+ TCounter new appendToJQuery: ''body'' asJQuery
755
+
756
+ '
757
+ ! !
758
+