To be able to edit code and run cells, you need to run the notebook yourself. Where would you like to run the notebook?

In the cloud (experimental)

Binder is a free, open source service that runs scientific notebooks in the cloud! It will take a while, usually 2-7 minutes to get a session.

On your computer

(Recommended if you want to store your changes.)

  1. Copy the notebook URL:
  2. Run Pluto

    (Also see: How to install Julia and Pluto)

  3. Paste URL in the Open box

Frontmatter

If you are publishing this notebook on the web, you can set the parameters below to provide HTML metadata. This is useful for search engines and social media.

Author 1
👀 Reading hidden code
begin
using Pkg
Pkg.activate(mktempdir())
Pkg.add([
Pkg.PackageSpec(name="PlutoUI", version="0.6.7-0.6"),
Pkg.PackageSpec(name="JSON"),
])

using PlutoUI
import JSON
end
❔
  Activating new project at `/tmp/jl_ZqEsWP`
    Updating registry at `~/.julia/registries/General.toml`
   Resolving package versions...
    Updating `/tmp/jl_ZqEsWP/Project.toml`
  [682c06a0] + JSON v0.21.4
  [7f904dfe] + PlutoUI v0.6.11
    Updating `/tmp/jl_ZqEsWP/Manifest.toml`
  [682c06a0] + JSON v0.21.4
  [69de0a69] + Parsers v2.8.1
  [7f904dfe] + PlutoUI v0.6.11
  [aea7be01] + PrecompileTools v1.2.1
  [21216c6a] + Preferences v1.4.3
  [fd094767] + Suppressor v0.2.8
  [2a0f44e3] + Base64
  [ade2ca70] + Dates
  [56ddb016] + Logging
  [d6f4376e] + Markdown
  [a63ad114] + Mmap
  [de0858da] + Printf
  [9a3f8284] + Random
  [ea8e919c] + SHA
  [9e88b42a] + Serialization
  [fa267f1f] + TOML
  [cf7118a7] + UUIDs
  [4ec0a83e] + Unicode
Precompiling project...
PlutoUI
  1 dependency successfully precompiled in 1 seconds (5 already precompiled)
3.1 s

this persistence

👀 Reading hidden code
210 μs

Edit x!

👀 Reading hidden code
md"""
Edit `x`!
"""
197 μs
👀 Reading hidden code
x = [80, 200, 500]
13.3 μs
👀 Reading hidden code
157 ms

(More info will follow, sorry!)

👀 Reading hidden code
188 μs

Self updating cells

👀 Reading hidden code
204 μs

Old value: 10

New value:

begin
# we need to reference potato in this cell --before assigning to potato-- to
# register as self-updating cell
if !@isdefined(potato)
# potato not defined means that this is the first run
# when the bond sets a value, the new value will be assigned to potato
# __before__ this cell is run.
# so potato defined means that this cell is running as
# response to the bond update
# set a default value
potato = 10
end
s = @bind potato Slider(1:100, default=potato)
# We now assigned to potato to register as self-update
# (`@bind x a` counts as assignment to `x`)
# bound values will only be set if they are assigned somewhere.
md"""
Old value: $potato
New value: $s
"""
end
👀 Reading hidden code
261 ms

Comments:

You see that you can't slide the slider. This is because a new slider gets rendered every time the cell updates, which resets its internal (cursor) state.

If this is a problem, it can be fixed using the this persistence together with self-updating, see the next section.

👀 Reading hidden code
338 μs

this persistence + self updating

The demo below uses both techniques, but you can see that the animations are not working when you click.

👀 Reading hidden code
272 μs
2×2 Matrix{Float64}:
 0.0  -1.0
 1.0  -0.7
A = [0 -1
1 -.7]
👀 Reading hidden code
24.2 μs
−2.0−1.5−1.0−0.50.00.51.01.52.0−3.0−2.5−2.0−1.5−1.0−0.50.00.51.01.52.02.53.0
begin
if !@isdefined(x0)
x0 = [0.5, 0.5]
end
@bind x0 wow(x0)
end |> with_d3_libs
👀 Reading hidden code
623 ms
x0
👀 Reading hidden code
9.4 μs
wow (generic function with 1 method)
function wow(previous)
path = compute_path(previous)
c = """
<script id="aa">
const path = $(JSON.json(path))

const svg = this == null ? DOM.svg(600,400) : this
const s = this == null ? d3.select(svg) : this.s
svg.value = $(JSON.json(previous))
const xscale = d3.scaleLinear()
.domain([-3,3])
.range([20, 580])
const yscale = d3.scaleLinear()
.domain([-2,2])
.range([380, 20])
if(this == null) {
s.append("g")
.attr("transform", `translate(\${xscale(0)},0)`)
.call(d3.axisLeft(yscale))
s.append("g")
.attr("transform", `translate(0,\${yscale(0)})`)
.call(d3.axisBottom(xscale))
s.append("g").classed("thepath", true)
}
const down_handler =(e) => {
svg.value = [xscale.invert(e.clientX - svg.getBoundingClientRect().left), yscale.invert(e.clientY - svg.getBoundingClientRect().top)]
svg.dispatchEvent(new CustomEvent("input", {}))
}
svg.addEventListener("pointerdown", down_handler)
invalidation.then(() => svg.removeEventListener("pointerdown", down_handler))
s.select("g.thepath").selectAll("path")
.data([path])
.join("path")
.transition()
.duration(300)
.attr("d", d => d3.line()
.x(p => xscale(p[0]))
.y(p => yscale(p[1]))(d))
.attr("stroke", "gray")
.attr('stroke-width', 5)
.attr("fill", "none")

const output = svg
output.s = s
return output
</script>

"""
BondDefault(HTML(c), previous)
end
👀 Reading hidden code
606 μs
500-element LinRange{Float64, Int64}:
 0.0,0.12024,0.240481,0.360721,0.480962,0.601202,…,59.519,59.6393,59.7595,59.8798,60.0
T = LinRange(0.0, 60.0, 500)
👀 Reading hidden code
17.5 μs
0.12024048096192384
ΔT = step(T)
👀 Reading hidden code
22.1 μs
compute_path (generic function with 1 method)
function compute_path(first)
accumulate(T; init=first) do x_prev, t
x_prev + ΔT * f(t,x_prev)
end
end
👀 Reading hidden code
1.0 ms
f (generic function with 1 method)
f(t,x) = A*x
👀 Reading hidden code
388 μs

The next one is kinda broken, come back later!

👀 Reading hidden code
193 μs
let
previous = if @isdefined(pos)
pos
else
100
end
x = previous * (1:10)
c = """
<script id="helloaza">
const x = $(JSON.json(x))

const svg = this == null ? DOM.svg(600,200) : this
const s = this == null ? d3.select(svg) : this.s
svg.value = $(previous)
const down_handler =(e) => {
svg.value = e.clientX - svg.getBoundingClientRect().left
svg.dispatchEvent(new CustomEvent("input", {}))
console.log(svg.value)
}
svg.addEventListener("pointerdown", down_handler)
invalidation.then(() => svg.removeEventListener("pointerdown", down_handler))
s.selectAll("circle")
.data(x)
.join("circle")
.transition()
.duration(300)
.attr("cx", d => d)
.attr("cy", 100)
.attr("r", 10)
.attr("fill", "gray")

const output = svg
output.s = s
return output
</script>

"""
with_d3_libs(@bind pos BondDefault(HTML(c), previous))
end
👀 Reading hidden code
76.7 ms
100
pos
👀 Reading hidden code
10.0 μs
with_d3_libs (generic function with 1 method)
with_d3_libs(content) = HTML("""
<script src="https://cdn.jsdelivr.net/npm/d3@6.2.0/dist/d3.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/d3-scale@3.2.3/dist/d3-scale.min.js"></script>
$(repr(MIME"text/html"(), content))
""")
👀 Reading hidden code
496 μs
begin
struct BondDefault
element
default
end
Base.show(io::IO, m::MIME"text/html", bd::BondDefault) = Base.show(io, m, bd.element)
Base.get(bd::BondDefault) = bd.default
end
👀 Reading hidden code
4.6 ms