Attached is the current development code base for Snap! (formerly known as BYOB4). It consists of several JavaScript, HTML and text files, and while some of it may be functional most parts will be in flux and subject to frequent, even fundamental modifications. This document lays out a few simple guidelines ensuring that collaborative code contribution works out.
Please check your code frequently with JSLint, either at JSlint.com or via a locally installed jslint.
For our Snap code set JSLint's settings to:
- assume a browser
- tolerate missing 'use strict' pragma
4
indentation78
maximum line length
If you're working on the core Morphic library you can also
- tolerate eval
- tolerate unfiltered for in
although you'll probably not ever going to need either EVAL or FORIN in your changesets anyway.
There are, of course, other tools - like JSHint and browser debug tools - that help you debug your code. Feel free to use whichever suits you best, but let's all agree on JSLint's (nitpicky!) formatting rules so we get code that's well readable and easily shareable among ourselves.
Snap's codebase is both big and fast moving. We'll continue to churn out several builds per day for a long time, hunting bugs and adding features throughout the whole application. We'll also need to be able to get our heads around the whole codebase. Being able to read and to quickly understand the code is most important, much more so than mathematical elegance.
Let me really stress this point: In creating Snap we're neither playing "Code Golf" (solving a problem using the least number of keystrokes) nor are we trying to outsmart Knuth. Instead we're maintaining a large number of small interchangeable code chunks, therefore:
- accessing the DOM
- frameworks (e.g. JQuery)
- modules and namespaces (e.g. IIFE)
- all of Snap is a single "World" with unique names for everything
- remember: Changesets contain "small interchangeable code chunks"...
- all of Snap is a single "World" with unique names for everything
- passing "thisArg" to functions like
map()
,filter()
orforEach()
- except in
call()
- always use
myself
to referencethis
in an outer scope instead
- except in
- meta-class systems
- use Morphic's way of creating class-like constructors instead
- initialize all attributes either in the constructor or in an
init()
method, so you now the object's structure by casting a single look upon it. Avoid adding adding attributes elsewhere, or if you do, initialize them explicitly to "null" in the constructor orinit()
code
- initialize all attributes either in the constructor or in an
- use Morphic's way of creating class-like constructors instead
- nested ternary operators
- RegEx
- overwriting existing definitions
- except in changesets, duh :-)
- create new constructors instead
- non-descriptive names for variables, functions, objects
- giant functions
- which often are "modules in disguise", especially if they define local helper functions
- create a new constructor instead
(don't worry, I'm not talking about formal UnitTest Suites or other BDSM software fetishes, just about playing with what you're creating while you're doing it)
To test your changes locally, just open index.html
in your browser.
To actually play with your Morphs you can right-click on them and open an inspector on them. You can open more than one inspector on each object. The inspector pretty much works the same as in Smalltalk. It even has an evaluation pane at the bottom, in which you can type in any JS code, mark it with your mouse (or select all with ctrl-a), righ-click on the selection and either "do it", "show it" or "inspect it" (again, like in Squeak).
Needless to say, in the evaluation pane this
always refers to the inspected
object.
Snap! is hosted on Github at https://github.com/jmoenig/Snap. You can make a fork
via the Github "Fork" button and then create a PR by pushing a branch to your fork
and then creating a PR against the master brancd of jmoenig/Snap
. You can see
current PRs here: https://github.com/jmoenig/Snap/pulls
At this stage of development, Snap! can be translated to any LTR language maintaining the current order of inputs (formal parameters in blocks).
Translating Snap! is easy:
Download the sources and extract them into a local folder on your computer: https://github.com/jmoenig/Snap/releases/latest.
Use the German translation file (named 'lang-de.js') as template for your own translations. Start with editing the original file, because that way you will be able to immediately check the results in your browsers while you're working on your translation (keep the local copy of snap.html open in your web browser, and refresh it as you progress with your translation).
Edit the translation file with a regular text editor, or with your favorite JavaScript editor.
In the first non-commented line (the one right below this note) replace "de" with the two-letter ISO 639-1 code for your language, e.g.
fr - French => SnapTranslator.dict.fr = {
it - Italian => SnapTranslator.dict.it = {
pl - Polish => SnapTranslator.dict.pl = {
pt - Portuguese => SnapTranslator.dict.pt = {
es - Spanish => SnapTranslator.dict.es = {
el - Greek => => SnapTranslator.dict.el = {
etc. (see http://en.wikipedia.org/wiki/ISO_639-1)
Then work through the dictionary, replacing the German strings against your translations. The dictionary is a straight-forward JavaScript ad-hoc object, for review purposes it should be formatted as follows:
{
'English string':
'Translation string',
'last key':
} 'last value'
and you only edit the indented value strings. Note that each key-value pair needs to be delimited by a comma, but that there shouldn't be a comma after the last pair (again, just overwrite the template file and you'll be fine).
If something doesn't work, or if you're unsure about the formalities you should check your file with JSLint
This will inform you about any missed commas etc.
Depending on which text editor and which file encoding you use you can directly enter special characters (e.g. Umlaut, accented characters) on your keyboard. However, I've noticed that some browsers may not display special characters correctly, even if other browsers do. So it's best to check your results in several browsers. If you want to be on the safe side, it's even better to escape these characters using Unicode.
See this collection of JavaScript utilities: http://0xcc.net/jsescape/
At this time your translation of block specs will only work correctly, if the order of formal parameters and their types are unchanged. Placeholders for inputs (formal parameters) are indicated by a preceding % prefix and followed by a type abbreviation.
For example:
'say %s for %n secs'
can currently not be changed into
'say %n secs long %s'
and still work as intended.
Similarly
'point towards %dst'
cannot be changed into
'point towards %cst'
without breaking its functionality.
When you're done, rename the edited file by replacing the "de" part of the filename with the two-letter ISO 639-1 code for your language, e.g.
fr - French => lang-fr.js
it - Italian => lang-it.js
pl - Polish => lang-pl.js
pt - Portuguese => lang-pt.js
es - Spanish => lang-es.js
el - Greek => => lang-el.js
and send it to me for inclusion in the official Snap! distribution. Once your translation has been included, Your name will the shown in the "Translators" tab in the "About Snap!" dialog box, and you will be able to directly launch a translated version of Snap! in your browser by appending
lang:xx
to the URL, xx
representing your translations two-letter code.
In some browsers accents or ornaments located in typographic ascenders above the cap height are currently (partially) cut-off.
Enjoy!
-Jens