for the past 350 years i have been sitting through vendors pitching software that will definitely solve 100% of all of my problems. i take a lot of screenshots and paste those into obsidian to have reference points for my notes.
as iâve hoarded piles of powerpoint screenshots, something has started bothering me1: each screenshot is at least 1 MB because macOS thinks i would appreciate a pixel-perfect lossless representation of a vendorâs feature matrix compared to their vastly inferior competitor who they know weâll be meeting with in an hour.
in this case, and in most cases, i do not need lossless pngs! iâm perfectly happy with lower resolution compressed garbage. perfect fidelity is the exception for me.
i picked up Retrobatch and Acorn a earlier this month after reading a blog post talking about them and i appreciated being able to pay a reasonable price one-time to permanently own software made by an independent developer. i hadnât really used Retrobatch yet but this problem seemed like the ideal job for it!
it took a lot of fiddling, but i have a workflow that works great:
- take a screenshot
cmd+space
to open Raycastlrc <enter>
to launch theLowRes Clipboard.app
droplet made from Retrobatch- paste a scaled down 75% quality webp2 instead of a lossless png
what eventually worked
weâll come back to all the failed attempts, but letâs start with what ended up working
data:image/s3,"s3://crabby-images/39981/399814efbb7237f7a65ae1b88dd054a0e1ece668" alt=""
the workflow relies on two scripts, timestamp-file
and pbfile
that iâll put in the appendix at the bottom of this post. the first script renames files to have an RFC3339ish timestamp prefixed3, and the latter is a wrapper around osascript
to stick a file on the clipboard.
the distinction between a file and image data from a file is important in this case. sticking image data on the clipboard did not work at all for me!
the copy the file
method has another benefit i didnât immediately realize: itâs a context-aware paste! if i paste into a program that accepts image data, the image data is pasted. otherwise, the file path is pasted. iâve found this useful!
failed attempt #1: pure clip->clip workflow, no intermediate files
what i initially wanted was to go straight from image data on the clipboard to transformed image data on the clipboard with no intermediate files. i tried connecting a read clipboard
node to a write clipboard
node.
data:image/s3,"s3://crabby-images/25bc1/25bc1b6b3c1f2b62e340310a1fd829644b795176" alt=""
this does not work because writing to clipboard doesnât allow for transformations. this is a bummer! but having spent some a little time fucking around in ObjectiveC (spoiler alert), i might understand why.
failed attempt #2: clip->file->native clip (no scripts)
fine iâll use an intermediate file.
data:image/s3,"s3://crabby-images/b7c9e/b7c9e937240cf48487dec147bcfe84167fa3cdfe" alt=""
this didnât work either. Raycast clipboard history lets me see the size of whatâs on my clipboard4.
data:image/s3,"s3://crabby-images/4671a/4671afdf6026152939f293635e07ea7a55cafe0c" alt=""
126 KB is too many kilobytes. i have the file, itâs 14 KB. i guessed it might be converting to png, and yeah, if i paste into obsidian iâm getting a .png
, and the file
command agrees.
failed attempt #3: clip -> file -> pbcopy
pbcopy
does not play nice with binary data. it pretends to work, but it does a conversion which i do not want. the man
page is explicit about this:
The input is placed in the pasteboard as plain text data unless it begins with the Encapsulated PostScript (EPS) file header or the Rich Text Format (RTF) file header, in which case it is placed in the pasteboard as one of those data types
also, lemme just say love the output of pbcopy -help
; pbcopy -help
2025-01-29 17:29:42.920 pbcopy[89866:22467314] Usage: pbcopy [-help]
# that's it, that's the full output
đ very helpful.
failed attempt #4: clip -> file with âoverwrite originalâ set
yeah Retrobatch really doesnât like it when you use âWrite back to original imagesâ when using Clipboard as a source, it straight up crashes. i filed my crash report with them, weâll see what comes of it.
update: they got back to me!
as i was drafting this post they emailed me back thanking me for the crash report, saying they investigated, fixed the bug, and built a new version which they linked me to!
i deeply appreciate the fast responseâi filed the report just a few hours ago, and iâm frankly used to never hearing back when i file issues with software vendors so i didnât expect anything. big shoutout to flying meat!
failed attempt #5: some loose ObjC i found jangling around on the internet
i found this gist that folks in the comments said worked for them, so maybe it will work for me? only time will tell!
foreshadowing: this is under a header that says failed attempt
.
it works better than pbcopy
, but it suffers the same problem as Retrobatch: it converts the image data to png, defeating the whole point of this cursed sidequest.
failed attempt #6: plead with a robot to please make this loose ObjC i found jangling around on the internet do what i need and please double check your work no bugs please thank you!!
i barely know ObjectiveCâonly ever had to modify it, never write it from scratchâand i definitely donât know macOS framework bullshit, so i threw the script into an llm and said âplz help, my clipboard, itâs dyingâ.
i learned a bunch, but was ultimately unsuccessful in convincing it to produce something that didnât result in a png. eventually i asked, âhey, you think this isâŠmaybe impossible?â and it said âlol yeah probablyâ:
NSImage also treats images as renderable content, not raw file formats, so copying NSImage to the clipboard always results in PNG.
welp, that might explain why Retrobatch canât do this either!
despite watching me flounder about with a solution that it knew5 wouldnât work, the robot did end up suggesting something useful: âUse AppleScript to Simulate Finder Copy (Hacky but Works)â. you know iâm about that âhacky but worksâ life so thatâs what I went with and it was hacky but it worked. that suggestion eventually became pbfile
.
escape hatches rule
even though Retrobatch couldnât do what i needed with standard nodes, letting me drop down to custom scripts in my workflow made this possible in the end. i love tools that give me reasonable 2-way escape hatches! i can escape, but i can also come back, e.g. outputting outputImagePath
from my script lets me return data back to the workflow. sometimes tools make you choose between blessed path and off-roading, so i like when tools consider reentry.
another thing i liked was being able to save the workflow as an executable .app
let me easily run this from Raycast!
oh also btw i did try imagemagick first since i hadnât really used Retrobatch yet and i figured this was a magick
one-liner
; pbpaste | magick png:- output.webp
magick: improper image header `/var/folders/dz/t5n_l_h57tdg4784rrrwddcw0000gn/T/magick-Nhp35
2dTBLEFyvfyis1KskY7ZEAZ96H2' @ error/png.c/ReadPNGImage/3941
i couldnât find anything useful about this error in a few minutes of searching so i just switched to Retrobatch thinking itâd be faster than trying to debug this.
current listening
Deafheaven is screaming again, we are so back6
appendix i: timestamp-file
#!/usr/bin/env bash
set -euo pipefail
# Check if at least one argument is provided
if [[ $# -eq 0 ]]; then
echo "Usage: $0 file1 [file2 ...]"
exit 1
fi
# Regex to detect an ISO-like timestamp at the beginning of a filename
timestamp_regex='^[0-9]{4}-[0-9]{2}-[0-9]{2}T[0-9]{6}_'
for file in "$@"; do
# Check if the file exists and is a regular file
if [[ ! -f "$file" ]]; then
echo "Error: File not found or not a regular file: $file"
exit 1
fi
# Extract the filename (without directory)
base=$(basename "$file")
# Skip renaming if the filename already starts with a timestamp
if [[ "$base" =~ $timestamp_regex ]]; then
echo "Skipping: $file (already contains a timestamp)"
continue
fi
# Get the creation time in RFC 3339 format (UTC)
ctime=$(stat -f "%SB" -t "%Y-%m-%dT%H%M%S" "$file")
if [[ -z "$ctime" ]]; then
echo "Error: Unable to retrieve creation time for $file"
exit 1
fi
# Extract directory
dir=$(dirname "$file")
# Construct new filename
new_name="${dir}/${ctime}_${base}"
if [[ -e "$new_name" ]]; then
echo "Error: Target filename already exists: $new_name"
exit 1
fi
# Rename the file
mv "$file" "$new_name"
# Required to chain the next Retrobatch step
echo "outputImagePath: $new_name"
done
appendix ii: pbfile
#!/usr/bin/env bash
set -euo pipefail
# Check if a file argument is provided
if [[ $# -ne 1 ]]; then
echo "Usage: pbfile <file-path>"
exit 1
fi
# Ensure the file exists
file_path="$1"
if [[ ! -f "$file_path" ]]; then
echo "Error: File not found: $file_path"
exit 1
fi
# Convert to absolute path (handles relative paths correctly)
abs_path=$(realpath "$file_path")
# Use AppleScript to copy the file reference to clipboard
osascript -e "set the clipboard to (POSIX file \"$abs_path\")"
echo "Success: Copied file reference to clipboard: $abs_path"
Footnotes
-
other than every vendor feeling like they need to shoehorn AI into their tools and presentation đ â©
-
why not avif? beyond having to do extra work to make the âimage hasnât loaded yetâ experience good that i talked about in a prior post, iâm running into a bunch of things that do not support avif! for example my rss client inoreader will pick up the first image from a post to use as a preview for the post, but it doesnât work for avif. iâm gonna keep filing issues where i find gaps in avif support, but it might be a little too early to go all in. â©
-
at some point I need to write about how I love to timestamp everything and how it improves my quality of life by at least 18%[citation needed] â©
-
huge shoutout to clipboard history tools, couldnât computer without âem. if youâre not using a clipboard history tool, get one today to improve your quality of life by 18%[citation needed] â©
-
i know that llms donât actually know anything, let me live â©
-
in the âmusic of Deafheavenâ sense, not the gestures around sense. honestly tho i wasnât an Infinite Granite hater, thereâs some bangers on there imo! â©