A Native Art Gallery for Your Mac (Take 2)

The challenge: fit a rotating art gallery somewhere into my life.

I love visual art and find it hugely inspiring. Unfortunately, reading art books is too much of a context switch to be a regular distraction, while museums are only appropriate for the rare trip. Instagram helps, but it only lets you see content from artists you follow. There’s still the 99% of art history beyond that sliver!

Sourcing art wasn’t the problem. For years, I had been keeping a fairly large folder of inspiring images from places such as Imgur albums, RuTracker museum collections, /r/ImaginaryNetwork, and /r/museum. But leafing through them wasn’t enough: I needed to put them into a regular and random rotation in a place that was just out of eyeshot, but without becoming an overt distraction.

In 2015, I finally solved the problem by building an app called Backgroundifier, which converted arbitrary-size images into wallpapers by superimposing them onto attractive, blurred backgrounds. By pairing an Automator Folder Action with the native wallpaper cycling functionality of macOS, I could now drop arbitrary images into a directory on my desktop and have them automatically show up in my wallpaper rotation. Peeking at an image was as simple as invoking the Show Desktop shortcut, and if I wanted to see something new, all I had to do was switch to a new Space.

For several years, this scheme worked perfectly fine. But recently, my collection had grown to over 500 images, and I found myself bumping into some slight annoyances. For example, I had no way to retrieve the filename of the current wallpaper, to remove an image from rotation, or to mark it as a favorite. Every maintenance task had to be performed manually.

Finally, I decided to build a menu bar app that would solve all my problems through a unified interface: BackgroundifierBuddy.

BackgroundifierBuddy expects your images to be organized into two directories: one containing your source images, and the other containing their converted, Backgroundified counterparts. This latter directory should be the directory selected for your wallpaper rotation in System Preferences.

To start with, right-clicking on the menu bar icon shows your desktop, and right-clicking again moves onward to the next image. Moving the cursor away from the icon hides the desktop.

With automatic conversion enabled, images dropped into the Source directory are immediately converted into wallpapers in the Output directory, just so long as the app is running. This means that obscure Folder Actions are no longer necessary for automatic conversion to work.

If the current wallpaper is based in the Output directory, and if it has a counterpart in the Source directory, a number of maintenance tasks become available. If you’ve grown tired of an image in your rotation, you can click Delete to trash it together with its source image. If you want to save it for later, or if find that it needs some tweaking, you can click Archive to delete the wallpaper image and move the source image into the Archive directory. (Holding Option allows you to archive the image while still keeping it in rotation.) Clicking Favorite adds a custom Finder tag to the source image, making it easier to locate later.

Finally, Refresh Wallpaper Cache restarts the Dock (which seems to sometimes be necessary to update the wallpaper rotation with new images), while Toggle Desktop Icons shows and hides the icons on your desktop for better image visibility.

I was going for simplicity with my solution, and this is just about as simple as it gets: your bog-standard OS wallpaper cycling functionality and a helper app that builds on basic file system commands. No complexity, no hassles, and everything just works!

Backgroundifier still costs a buck on the App Store, but BackgroundifierBuddy is free and open source. You can find the latest release here. Enjoy!

Technical Details

Although deceptively simple on the surface, the code behind BackgroundifierBuddy has eluded me for some time. The reason is that there’s no public way to query the current wallpaper image when a directory is selected. You can try calling desktopImageURL on NSWorkspace.shared, but this will only return the directory itself, not the displayed image.

In the past, you could pull this info from the com.apple.desktop defaults, but this is no longer an option. Starting with Mavericks, the wallpaper settings are stored in a desktoppicture.db SQLite file located in the ~/Library/Application Support/Dock directory. The layout of this file is a tiny bit confusing, and you can read more about it here. In brief, each image in the data table is associated with a Space UUID and display UUID pair. Unfortunately, there’s no indication of which UUID the current Space might be associated with, nor are the UUIDs stored in order. (So if a new Space is created and then moved over a few spots, it becomes impossible to tell which one it is from the database alone.)

What’s needed is a way to get the UUIDs of the current Space and display, and the classic way to do this is to query the com.apple.spaces defaults. Unfortunately, the data returned appears to be subtly incorrect. Among other faults, the “Current Space” UUID is usually out of date and the “Display Identifier” UUID is outright wrong, at least on my machine. To get the real info dictionary for Spaces, you have to call the private CGSCopyManagedDisplaySpaces function. This gives you up-to-date UUIDs for both the current Space and display.

With these UUIDs in tow, there’s now enough info to run a query against the desktoppicture.db file and retrieve the current wallpaper. Fortunately, we don’t even have to go this far. After sleuthing around on Github for some relevant keywords—_CGSDefaultConnection together with “wallpaper”, I think—I found a scant few references to a private function that did exactly what I needed: DesktopPictureCopyDisplayForSpace. Together with CGSGetDisplayForUUID, you can use this function to retrieve a dictionary with all the wallpaper info for a given space.

Caveat emptor: all this stuff might break in a future macOS release. Fortunately, I’m not making any changes through the private APIs, only requesting data. The only edit I make to the wallpaper settings is when refreshing the current image, and this is simply done by calling the public NSWorkspace method setDesktopImageURL with arguments mirrored from desktopImageURL and desktopImageOptions. Toggle Desktop Icons does make a change to the com.apple.finder defaults, but this functionality is entirely optional.

Note that sandboxing restrictions only allow command line calls to Backgroundifier to process images in the user’s Pictures directory and subdirectories. I haven’t yet found a way to expand an app’s sandbox when called from a non-sandboxed app, so this restriction carries over to the selection of Source and Output directories. (Let me know if you know a way to expand an app’s sandbox from another app!) If this poses a problem, you’ll be able to find a non-sandboxed, command-line version of the Backgroundifier executable in a zip file in the Resources subdirectory of the Backgroundifier.app bundle. Point to it in the BackgroundifierBuddy preferences and you should be good to go for arbitrary Source and Output directories.

You can find a discussion of this article on Hacker News.

Archagon

May 2, 2018