Makie to Simulate Turing Machine Tape

Julia Makie supports recording of animated gif, etc. I wish to utilize this feature to create an animation of a Turing machine’s tape. I’d like the tape to appear such as follows (created using HTML) …

My attempt at this in Julia appears below …

using CairoMakie
function mainFunction()
  figure = Figure()
  colgap!( figure.layout, 0 )
  Textbox( figure[1,1], placeholder = "☐", bordercolor = :black, cornerradius = 0, borderwidth = 2 )
  Textbox( figure[1,2], placeholder = "A", bordercolor = :black, cornerradius = 0, borderwidth = 2 )
  Textbox( figure[1,3], placeholder = "B", bordercolor = :black, cornerradius = 0, borderwidth = 2 )
  Textbox( figure[1,4], placeholder = "C", bordercolor = :black, cornerradius = 0, borderwidth = 2 )
  Textbox( figure[1,5], placeholder = "☐", bordercolor = :black, cornerradius = 0, borderwidth = 2 )
  save( "Turing_machine_tape.png", figure )
end # function mainFunction
mainFunction()

The above Julia program creates an image that suffers from the following “deficiencies” …

  • has tape cells farther apart than I would like (as I wish to display as many cells as possible horizontally)
  • does not have uniform cell-width (A, B, C cells seem narrower than :white_medium_square: cells)
  • does not have cell contents in black (as A, B, C, and :white_medium_square: appear to be grey)

How may I correct these “deficiencies” in the Julia-generated image?

Here’s one way you can do this. I wouldn’t use Textbox etc. because those block objects are relatively heavy and not nice for animation. Much easier to handle this with plain plot objects.

using CairoMakie

f = Figure()
ax = Axis(f[1, 1], autolimitaspect = 1)
hidedecorations!(ax)
hidespines!(ax)

letters = string.('A':'Z')

cellsize = (20, 25)
xs = cumsum(fill(cellsize[1], length(letters)))
poly!(
    ax, [Rect2f((x, 0) .- 0.5 .* cellsize, cellsize) for x in xs],
    color = [i == 10 ? :rosybrown1 : :gray95 for i in eachindex(letters)],
    strokecolor = :black,
    strokewidth = 1
)
text!(
    ax, xs, zeros(length(letters)),
    text = letters,
    align = (:center, :center),
    markerspace = :data
)
scatter!(ax, xs[10], cellsize[2] / 2 + 10, marker = :dtriangle, color = :red)
f

4 Likes

Thank you for this information.

I am writing programs as I attempt to understand jules’ solution shown above. Below is one of my programs …

using CairoMakie
function mainFunction()
  figure = Figure()
  axis = Axis( figure[1,1], autolimitaspect = 1 )
  text!( axis, -5, 0, text = "A", align = (:center, :center), markerspace = :data, fontsize = 1.0 )
  text!( axis,  0, 0, text = "B", align = (:center, :center), markerspace = :data, fontsize = 1.0 )
  text!( axis, +5, 0, text = "C", align = (:center, :center), markerspace = :data, fontsize = 1.0 )
  save( "ABC_ok.png", figure )
end # function mainFunction
mainFunction()

The above program yields …

However, the following program yields …

BoundsError: attempt to access 1-element Vector{Makie.GlyphCollection} at index [2]

using CairoMakie
function mainFunction()
  figure = Figure()
  axis = Axis( figure[1,1], autolimitaspect = 1 )
  xs = [-5,0,+5]
  ys = [0,0,0]
  text!( axis, xs, ys, text = "ABC", align = (:center, :center), markerspace = :data, fontsize = 1.0 )
  save( "ABC_NOT_ok.png", figure )
end # function mainFunction
mainFunction()

It seems that I am using function text! in the same manner as jules, but getting an error. Might someone see where I am going wrong?


You cannot pass a single string for three text positions in an array. In that case you need three strings in an array as well.

Thank you. I had (mistakenly) assumed that, when letters = string.('A':'Z'), that typeof(letters) would be String, but I see now that typeof(letters) is Vector{String}.