Skip to content

Instantly share code, notes, and snippets.

@armenr
Last active January 17, 2025 23:54
Show Gist options
  • Save armenr/81b77587c1dda1d00d1c1c9541dcda94 to your computer and use it in GitHub Desktop.
Save armenr/81b77587c1dda1d00d1c1c9541dcda94 to your computer and use it in GitHub Desktop.
ClipSync (2-way clipboard sync between Wayland and X11 + cliphy support)
#!/usr/bin/env sh
#
# Two-way clipboard syncronization between Wayland and X11, with cliphy support!
# !! Recommended use: Drop this file off @ /usr/local/bin/clipsync && make it executable
# Requires: wl-clipboard, xclip, clipnotify.
# Modified from: https://github.com/hyprwm/Hyprland/issues/6132#issuecomment-2127153823
#
# Usage:
# clipsync watch [with-notifications|without-notifications] - run in background.
# clipsync stop - kill all background processes.
# echo -n any | clipsync insert [with-notifications|without-notifications] - insert clipboard content from stdin.
# clipsync help - display help information.
#
# Workaround for issue:
# "Clipboard synchronization between wayland and xwayland clients broken"
# https://github.com/hyprwm/Hyprland/issues/6132
#
# Also pertains to:
# https://github.com/hyprwm/Hyprland/issues/6247
# https://github.com/hyprwm/Hyprland/issues/6443
# https://github.com/hyprwm/Hyprland/issues/6148
# Updates clipboard content of both Wayland and X11 if current clipboard content differs.
# Usage: echo -e "1\n2" | clipsync insert [with-notifications|without-notifications]
get_mime_type() {
wl-paste --list-types | head -n 1
}
insert() {
mime_type=$(get_mime_type)
case "$mime_type" in
text/*)
value=$(wl-paste -n | tr -d '\0')
insert_text "$value" "$1"
;;
image/*)
insert_image "$1"
;;
*)
insert_file "$1"
;;
esac
}
insert_text() {
value="$1"
notification_mode="$2"
wValue=$(wl-paste -n 2>/dev/null | tr -d '\0' || printf "")
xValue=$(xclip -o -selection clipboard 2>/dev/null | tr -d '\0' || printf "")
notify() {
if [ "$notification_mode" != "without-notifications" ]; then
truncated_value=$(printf "%.50s" "$value")
notify-send -u low -c clipboard "Text copied" "$truncated_value"
fi
}
update_clipboard() {
if [ "$value" != "$1" ]; then
notify
printf "%s" "$value" | $2
command -v cliphist >/dev/null 2>&1 && printf "%s" "$value" | cliphist store
fi
}
update_clipboard "$wValue" "wl-copy"
update_clipboard "$xValue" "xclip -selection clipboard"
}
insert_image() {
notification_mode="$1"
tmp_file=$(mktemp)
wl-paste >"$tmp_file"
[ "$notification_mode" != "without-notifications" ] && notify-send -u low -c clipboard "Image copied" "Image data copied to clipboard"
wl-copy <"$tmp_file"
xclip -selection clipboard -t "$(get_mime_type)" <"$tmp_file"
command -v cliphist >/dev/null 2>&1 && wl-copy <"$tmp_file" | cliphist store
rm "$tmp_file"
}
insert_file() {
notification_mode="$1"
tmp_file=$(mktemp)
wl-paste >"$tmp_file"
[ "$notification_mode" != "without-notifications" ] && notify-send -u low -c clipboard "File copied" "File data copied to clipboard"
wl-copy <"$tmp_file"
xclip -selection clipboard -t "$(get_mime_type)" <"$tmp_file"
command -v cliphist >/dev/null 2>&1 && wl-copy <"$tmp_file" | cliphist store
rm "$tmp_file"
}
# Watch for clipboard changes and synchronize between Wayland and X11
# Usage: clipsync watch [with-notifications|without-notifications]
watch() {
sleep 1
notification_mode=${1:-with-notifications}
watch_clipboard() {
$1 | while read -r _; do
clipsync insert "$notification_mode"
done &
}
watch_clipboard "wl-paste --watch printf ''"
watch_clipboard "wl-paste --primary --watch printf ''"
watch_clipboard "clipnotify"
watch_clipboard "clipnotify -s PRIMARY"
}
# Kill all background processes related to clipsync
stop_clipsync() {
pkill -f "wl-paste --type text --watch"
pkill clipnotify
pkill -f "xclip -selection clipboard"
pkill -f "clipsync insert"
}
help() {
cat <<EOF
clipsync - Two-way clipboard synchronization between Wayland and X11, with cliphist support
Usage:
clipsync watch [with-notifications|without-notifications]
Run clipboard synchronization in the background.
Options:
with-notifications (default): Show desktop notifications for clipboard changes.
without-notifications: Operate silently without notifications.
clipsync stop
Stop all background processes related to clipsync.
echo -n "text" | clipsync insert [with-notifications|without-notifications]
Insert clipboard content from stdin.
Notification options work the same as in the watch command.
clipsync help
Display this help information.
Requirements: wl-clipboard, xclip, clipnotify
Optional: cliphist (for Hyprland users)
EOF
}
case "$1" in
watch)
watch "$2"
;;
stop)
stop_clipsync
;;
insert)
insert "$2"
;;
help)
help
;;
*)
echo "Usage: $0 {watch [with-notifications|without-notifications]|stop|insert [with-notifications|without-notifications]|help}"
echo "Run '$0 help' for more information."
exit 1
;;
esac
@armenr
Copy link
Author

armenr commented Sep 11, 2024

Updated to fix a few bugs.

It now correctly does 2-way sync. It also handles images, files, etc just as you would expect anywhere in your graphical user/desktop session.

@Vaisakhkm2625
Copy link

thanks for making this

though butter to update kill function to not to kill all the instance of wl-clip watch

❯ ps aux | grep wl-paste
vaisakh     1853  0.0  0.0   2692  1792 tty2     S+   17:14   0:00 wl-paste --type text --watch cliphist store
vaisakh     1854  0.0  0.0   2692  1536 tty2     S+   17:14   0:00 wl-paste --type image --watch cliphist store
vaisakh     1855  0.0  0.0   2692  1664 tty2     S+   17:14   0:00 wl-paste -t text --watch bash -c kdeconnect-cli -d a6780b4540540a1c --share-text "$(</dev/stdin)"
vaisakh    42525  0.0  0.0   6688  2560 pts/7    S+   19:58   0:00 grep wl-paste

@armenr
Copy link
Author

armenr commented Nov 23, 2024

Oh damn. Good point @Vaisakhkm2625 - I'll try to refactor this in order to avoid accidentally doing that! :)

@PasVegan
Copy link

This doesn't work for copying from brave to vscode on Hyprland

@andron3000
Copy link

Doesn't work.

@jrayhawk
Copy link

A lot of this can be simplified with idiomatic POSIX:

https://x.org/~jrayhawk/clipsync3.sh

...But fundamentally this should get rewritten with tooling that can natively bridge multi-type clipboard operations and cache NULs in-memory.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment