Makie outline 3e

makie expert, need your help again. Is there away to obtain an “outline” effect like this?

Specifically, in drawing a torus as a surface

I’d like the outer edge and the inner “hole” to have an outline, similar to this:

1 Like
using GLMakie, GeometryBasics

function torus(R, r, N, n)
    ps = Vector{Point3f}(undef, N*n)
    idx = 1
    for i in 1:N
        c = cos(2pi * i/N)
        s = sin(2pi * i/N)
        M = Mat3f(
            c, s, 0,
            -s, c, 0,
            0, 0, 1
        center = Point3f(R*c, R*s, 0)
        for j in 1:n
            ps[idx] = center + M * Point3f(r*cos(2pi * j/n), 0, r*sin(2pi * j/n))
            idx += 1

    faces = Vector{GLTriangleFace}(undef, 2N*n)
    for i in 1:N*n
        faces[2i-1] = GLTriangleFace(
            mod1(i+n, N*n),
            mod1(i+1, N*n) 
        faces[2i] = GLTriangleFace(
            mod1(i+1, N*n), 
            mod1(i+n, N*n),
            mod1(i+n+1, N*n) 

    return normal_mesh(ps, faces)

To my understanding geometry nodes are a way to manipulate the rendering pipeline. Makie doesn’t let you do that, so you have to use or abuse what’s there… I have two ideas how to get something similar:

  1. Abuse the lighting calculation to invert specular reflections:
    torus(1f0, 0.3f0, 100, 30), color = RGBf(1, 1, 1),
    specular = Vec3f(-1e-3), shininess = -4f0

Screenshot from 2022-09-24 16-59-48
You can also amplify color to values larger than 1 to get sharper outlines with this, but you can’t fully get rid of the dark blobs. But maybe those are nice in some situations? (negative shininess puts specular reflections near edges and negative specular results in brightness/color being taken away)

  1. Draw a second mesh “behind” the first:
f, a, p = mesh(
    torus(1f0, 0.32f0, 100, 30), color = :black, depth_shift = 0.01f0
p2 = mesh!(a, torus(1f0, 0.3f0, 100, 30), color = :white)

Screenshot from 2022-09-24 16-58-47
depth_shift = 0.01f0 puts the black torus slightly behind the white one in the final depth buffer, so it only peaks out outside the white torus. Depending on the value of depth_shift you may get more or less peaking out. (Should always be between 0 and 1)



This seems amazing, the only problem is that they way I currently have things set up I’m working with surface instead of mesh, is there a way to achieve something similar with surface that you know of?

Thank you

With a simple mesh you could try just scaling it for approach 2:

p = surface!(ax, ...)
scale!(p, 1.1)

For a more complex mesh like a torus this wouldn’t work though. Something else you could try is translating all the vertex positions towards their normal direction. For example: (constants probably need adjustments)

using CairoMakie, GLMakie
GLMakie.activate!() # just in case

# example mesh
using FileIO
m = FileIO.load(Makie.assetpath("cat.obj"))

# CairoMakie has a (internal) surface -> mesh conversion function you can use
# m = CairoMakie.surface2mesh(xs, ys, zs)

# grow mesh
m2 = deepcopy(m)
scale = 0.0001 * m.position |> norm |> maximum
@. m2.position += scale * m.normals

f, a, p = mesh(m, color = :orange)
mesh!(a, m2, color = :black, depth_shift = 0.005f0)

Screenshot from 2022-09-27 10-18-47