Observable JavaScript
NotebooksThe JavaScript dialect used in Observable notebooks is almost—but not entirely—vanilla. This is intentional: by building on the native language of the web, Observable is familiar. And you can use the libraries you know and love, such as D3, Lodash, and Apache Arrow. Yet for dataflow, Observable needed to change JavaScript in a few ways.
Note
Observable JavaScript is used in notebooks only. Observable Framework uses vanilla JavaScript.
Here’s a quick overview of what’s different from vanilla.
Cells are separate scripts
Each cell in a notebook is a separate script that runs independently. A syntax error in one cell won’t prevent other cells from running.
data:image/s3,"s3://crabby-images/df85b/df85bff30f0f922aa624174514188aed631de4e9" alt="Two open cells, the first showing a syntax error and then the second showing a variable assignment. Even though the first cell has a syntax error it has no affect on the other cell in the notebook."
The same holds true for a runtime error:
data:image/s3,"s3://crabby-images/c4adc/c4adcb5b52f5a90b939cad29a0d54d612c0eae4d" alt="Two open cells, the first showing a runtime error and then the second showing a variable assignment. Even though the first cell has a runtime error it has no affect on the other cell in the notebook."
Likewise, local variables are only visible to the cell that defines them. Here in the following screenshot, you can see the constant local
is defined within curly braces and is therefore local to the cell/block. Therefore, when the variable is called in the next cell, there is a runtime error:
data:image/s3,"s3://crabby-images/39477/394774f6aebb479cc4785857d627395c2dd69e97" alt="Two open cells, the first defines a local variable, `local`. Because `local` is local to the other cell, calling it from the second cell causes a runtime error."
Cells run in topological order
In vanilla JavaScript, code runs from top to bottom. Not so here; Observable runs like a spreadsheet, so you can define your cells in whatever order makes sense.
data:image/s3,"s3://crabby-images/cddf1/cddf1615292eac9be9957324337ea98932784bad" alt="Two open cells, the first uses a variable, `a`, in a calculation. The second cell defines `a`. This shows that the order of cells doesn't matter with respect to variable definitions."
By extension, circular definitions are not allowed:
data:image/s3,"s3://crabby-images/a9c0e/a9c0ecda9a6604d8c9f9bdfe825a7e3bb8102db2" alt="Two open cells, the first which defines `c1.` in terms of `c2` and the second of which defines `c2` in terms of `c1`. Both cells show errors because circular definitions are not allowed in Observable."
Cells re-run when any referenced cell changes
You don’t have to run cells explicitly when you edit or interact—the notebook updates automatically. Run the cell below by clicking the play button , or by focusing and hitting Shift-Enter . Only the referencing cells run, then their referencing cells, and so on—other cells are unaffected.
If a cell allocates resources that won’t be automatically cleaned up by the garbage collector, such as an animation loop or event listener, use the invalidation promise to dispose of these resources manually and avoid leaks.
data:image/s3,"s3://crabby-images/4e4ff/4e4ffc90ea6c436daa63e3a2307f14a2ad849bc8" alt="An open cell, in the presentation section we see the value of the cell is `undefined`. In the code section of the cell there is some code using the invalidation promise."
Cells implicitly await promises
You can define a cell whose value is a promise:
data:image/s3,"s3://crabby-images/19e4d/19e4d718a1e8a4976dae5aa1dd48f8c9b07548ce" alt="An open cell, the presentation section shows the message 'hello there' and the code section shows a new promise object being created such that it resolves after 30000 ms to 'hello there'."
If you reference such a cell, you don’t need to await
; the referencing cell won’t run until the value resolves.
Cells implicitly iterate over generators
If a cell yield
s, any referencing cell will see the most recently yielded value.
data:image/s3,"s3://crabby-images/706f2/706f2ee08f3aed0c1bc3f31abedce4b0e4f73b85" alt="Two open cells, the first showing `c` having a value of 3 in the presentation mode and a block of text with three `yield`s consecutively, with values 1,2, and 3 respectively. The second cell shows `c` being called in the code section, while showing the value 3 in the presentation section, which is the most recent of the `yield`ed values of the block above."
Also, yields occur no more than once every animation frame: typically sixty times a second, which makes generators handy for animation. If you yield a DOM element, it will be added to the DOM before the generator resumes.
Named cells are declarations, not assignments
Named cells look like, and function almost like, assignment expressions in vanilla JavaScript. But cells can be defined in any order, so think of them as hoisted function declarations.
You can’t assign the value of another cell (though see mutables below):
data:image/s3,"s3://crabby-images/4532c/4532c9f6c209fdd1246c24486d88ac2593232667" alt="Two open cells, the first showing a variable assignment of `foo` being set to the number 2. The second showing the same variable being assigned a different value inside curly braces."
Cell names must also be unique. If two or more cells share the same name, they will all error:
data:image/s3,"s3://crabby-images/4532c/4532c9f6c209fdd1246c24486d88ac2593232667" alt="Two open cells, the first showing a variable assignment of `foo` being set to the number 2. The second showing the same variable being assigned a different value, 3,inside curly braces."
Note
Observable doesn’t yet support destructuring assignment to declare multiple names, but we hope to add that soon.
Statements need curly braces, and return or yield
A cell body can be a simple expression, such as a number or string literal, or a function call. But sometimes you want statements, such as for loops. For that you’ll need curly braces, and a return
statement to give the cell a value. Think of a cell as a function, except the function has no arguments.
data:image/s3,"s3://crabby-images/56866/56866eb0ec0a99bb4e51bcb890f12c17875cba28" alt="An open cell with the value 45 shown in the presentation section of the cell and a statement in curly braces in the code section of the cell with a return statement that returns the value of `sum`, which in this case is 45. So the value of the return statement is the same as the cell's value."
For the same reason, you’ll need to wrap object literals in parentheses, or use a block statement with a return
:
data:image/s3,"s3://crabby-images/158df/158dfb57461f1fc27d77c1b482f989afa4b9715d" alt="Two open cells each with properly defined object literals. The first cell uses parentheses around the object in the code section of the cell and the second cell uses a return statement to return the object without parentheses"
The below cell is interpreted as a block with a single labeled statement, followed by a string literal expression. The cell value is undefined because the block doesn’t return anything.
data:image/s3,"s3://crabby-images/39098/390981d64a96f86accd40f79d3a420ba25418242" alt="An open cell `label` with the value of `undefined` shown in the presentation section of the cell, while the code section shows the object `label` being defined as an object literal without parentheses, which are required in Observable."
Cells can be views
Observable has a special viewof
operator which lets you define interactive values. A view is a cell with two faces: its user interface, and its programmatic value.
data:image/s3,"s3://crabby-images/0f25f/0f25f22514824aff52881c61098cd029a4f1468f" alt="Two open cells, the first featuring the `viewof` operator set to `text` in the code section which renders a text field in the presentation section of the cell. The second cell calls `text` in the code section and provides the value of the above cell's text field as a result."
Cells can be mutables
Observable has a special mutable
operator so you can opt-in to mutable state: you can set the value of a mutable from another cell:
data:image/s3,"s3://crabby-images/09707/09707b0a764e054318cc371946ee1cf14a223020" alt="Two open cells, the first using the `mutable` operator to define the variable `thing` set to value 0, and the next cell modifying thing by adding 1 successfully."
Observable has a standard library
Observable provides a small standard library for essential features, such as a reactive width
and Inputs.
Cells can be imported from other notebooks
You can import any named cell from any notebook, with syntax similar to static ES imports. But Observable imports are lazy: if you don’t use it, it won’t run.
data:image/s3,"s3://crabby-images/73f84/73f844bedd5dab974bb0c43fc08c0bec8ba62246" alt="Two open cells, the first importing a cell called ramp from another notebook, the second calling the imported cell and using it to create a visual."
Also, you can import-with, which allows you to inject cells from the current notebook into the imported notebook, overriding the original definition. You can treat any notebook as an extensible template!
Static ES imports are not supported; use dynamic imports
Since everything in Observable is inherently dynamic, there’s not really a need for static ES imports—though, we might add support in the future. Note that only the most-recent browsers support dynamic imports, so you might consider using require for now.
data:image/s3,"s3://crabby-images/fded0/fded0d744518dd455fa57d74fe7229c866638d84" alt="Two open cells, the first importing the lodash module. The second one is using the lodash module to put a string into camel case."