What is “jank”? It’s artifacts in your UI.
I’ve mainly heard the term used in a frontend web programming context. Ironically, most low level timing and painting is implemented for you on the web. It’s much harder to build something jank-free in a low level cross-platform C++ UI framework such as JUCE.
I spent a week trying to get a tooltip to animate nicely in JUCE.
I spent a couple days trying to track down text rendering issue in a TextEditor.
On the main plugin I’m developing, there’s probably been a couple dozen occasions so far where I resorted to manually logging paint calls to understand why components were painting partially or “animating” strangely.
What is jank?
Jank is jerky.
Jank is glitchy and flickering.
Jank is an unintentional intermediate state.
Jank is jarring and communicates “low quality” and “nobody cared.” It’s not functionally broken, but it’s certainly not nice.
Jank is my nemesis. I want to build UIs that look nice, feel snappy, and are free of papercuts.
Jank & JUCE
A C++ framework like JUCE seems more prone to jank than something like HTML/CSS:
- It’s low level enough where one has to manually manage, time and optimize paint calls
- A dev has to learn the hard way (experience or via the source) when the framework actually repaints components. For example, it’s important to know which methods (such as
setVisible) call repaint….. or that each repaint will call repaint on the component’s parent if the component is transparent, etc.
- JUCE’s out of the box widgets, such as tooltips, were not built with animation in mind.
- There’s very minimal paint debugging in JUCE.
- Component animation support is ad-hoc, rather than built into the concept of a JUCE component.
- Catch22: Most audio plugin UIs are single screens, in part because it’s very high effort to make a nice cross-platform multi-screen app.
What are some examples of jank in JUCE?
Animating one component over another
I wanted a tooltip to move with a slider thumb. I discovered this was hard.
To resolve the jank (after a lot of debugging), I got clinical with paint call timing. I made sure everything was always painted synchronously (via manually calling repaint on siblings).
I had a problem where text entry felt delayed and letters appeared in pieces in two subsequent paint calls.
The root cause was because a lot of my UI was behind the modal form the
TextEditor was sitting on. It turns out
setOpaque wasn’t preventing those parts of the UI from painting, so lots of my UI would repaint when the cursor sang the song of its people. See the forum post.
Popups and Overlays
I’ve run into this many times. Any time a component is being placed on top of another, care has to be given to paint ordering.
Got Jank? Screen Record It
If you are seeing painting jank or glitching but can’t quite understand what’s happening, take the guesswork out of the equation: screen record what’s happening and then step through it frame by frame.
Lightweight ways I resolve jank
DBGin paint calls of relevant children, sibling and parents. Perform relevant UI actions and check out the order.
- If you are using stock JUCE “widgets”, google their implementations for calls to
repaint, you might be surprised when things are repainting.
- All sorts of things can cause a repaint, such as
setBounds, a child’s paint call, etc. Get clear on the order.
- If you are using custom UI components that you wrote, verify where you placed calls to
Heavyweight ways I resolve jank
Log all component paint calls
Step 1: Open up the JUCE component implementation and stick
DBG calls in
Component::paintComponentAndChildren for next to each call to
paint(). Note: don’t be tempted to put the
DBG call into the
Component::paint method, as it could be overridden.
Step 2: Demangle the component type name so you know which component is painting. I wrote a method
melatonin::componentString to do this, which I include in
juce_Component.cpp each time I need to do this.
Step 3: Stick something like the following into
handlePaint method to understand what the paint bounds are for each call:
DBG("Clip bounds at handlePaint(): " + g.getClipBounds().toString())
Note: this doesn’t take hierarchy into account, but it should give you a fuller picture of what’s actually painting.
Use perfetto and stick macros in the same places described above (next to the calls to paint).
This is the highest amount of detail, giving you a visual hierarchy over time.
Some people swear by “Quartz Debug” on MacOS. You can get by downloading “Additional Tools for Xcode” on apple’s developer site. It’s flashes paint call bounds on screen in a way people find more reliable than
Addressing Jank at the framework level
Better support for component debugging would help a lot.
JUCE_ENABLE_REPAINT_DEBUGGING visually shows you what is repainting. But it’s very hard to visually parse timing issues with flickering colors. It would be nice to have a way to debug the when and better understand how the component hierarchy is interacting.
It would be great if we could register a lambda with JUCE that is called with a reference to component on every paint call in debug. This would enable some very cool tooling (and be useful for easy console debugging).
Another improvement would be having first class animation support in the library, including implementing common easing functions, adding stack blur shadows and perhaps taking some inspiration from the greensock API. I believe this would raise the bar and better guarantee that common tasks such as popping up a modal or animating a tooltip are jank free out of the box.