Desktop-spanning backgrounds in KDE 4.x

One of the simplest ways to make a multi-monitor system look more impressive is to set a desktop background which spans across all of the monitors. Just scale the image up to the smallest size that fills all of the visible desktop space, then cut out monitor-shaped pieces to display.

Every system (even lightweight, older Linux desktops) supports this… every system except one. When KDE went from 3.5.x to 4.x, they dropped support for spanning a background across all desktops and, to this day, they still haven’t brought it back.

…and since everything more lightweight has bugs with the desktop layout I’m currently dealing with while I wait for the new monitor bracket I ordered, that’s a problem. (Even worse, there’s no API I could find to programmatically set a background either!)

The solution I settled on was to write a little Python script which uses maybe half a dozen lines of PyQt 4.x/5.x calls (before line-wrapping and boilerplate) to do what KDE should have, then spits out image files to be set as per-monitor backgrounds.

I also implemented a --randomize option which can be used with cron to produce input for KDE’s slideshow mode. (Basically, you set each monitor background to be a single-entry slideshow that the script updates)

Just give it an image (or --randomize and a list of files and/or folders) and an output directory. (See --help for more details)

UPDATE: It now also gives more control over how the background is matched to the desktop’s aspect ratio via --gravity and I included, as an example, the .desktop file I use to integrate it with Geeqie via Zenity. (KDialog’s equivalent to zenity --list is inferior.)

[Desktop Entry]
# IMPORTANT: THIS FILE IS A ONE-OFF HACK THAT I WROTE FOR MYSELF AND I AM
# SHARING IT AS AN EXAMPLE. I DO NOT GUARANTEE THAT IT WILL WORK AS-IS ON
# YOUR DESKTOP.
#
# Requirements:
# - Geeqie
# - Zenity
# - ~/bin/set_background.py
#
# Instructions:
# 1. `xdg-desktop-menu install geeqie-prepare-background.desktop`
# 2. `mkdir -p ~/.local/share/images/background`
Version=1.0
Type=Application
Name=Prepare Desktop Background
# call the helper script
Exec=sh -c 'python ~/bin/set_background.py --gravity $(zenity --list --title="Select Gravity" --column="Gravity" --text="Select Background Alignment" --hide-header --height=300 top-left top-center top-right middle-left middle-center middle-right bottom-left bottom-middle bottom-right | cut -d\| -f1) %F ~/.local/share/images/background'
# Desktop files that are usable only in Geeqie should be marked like this:
Categories=X-Geeqie;
OnlyShowIn=X-Geeqie;
# Show in "Edit" menu
X-Geeqie-Menu-Path=EditMenu
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""Simple PyQt5 script to chop up multi-monitor backgrounds for KDE."""
from __future__ import (absolute_import, division, print_function,
with_statement, unicode_literals)
__author__ = "Stephan Sokolow (deitarion/SSokolow)"
__version__ = "0.2.1"
__license__ = "MIT"
IMG_EXTS = [
'.bmp', '.gif',
'.jpg', '.jpe', '.jpeg',
'.png', '.pbm', '.pgm', '.ppm',
'.xbm', '.xpm',
'.svg'
]
import logging
log = logging.getLogger(__name__)
import errno, os, random
try:
from PyQt5.QtCore import QRect, QPoint, Qt
from PyQt5.QtGui import QImage
from PyQt5.QtWidgets import QApplication
except ImportError:
from PyQt4.QtCore import QRect, QPoint, Qt
from PyQt4.QtGui import QApplication, QImage
V_GRAV = {
'top': Qt.AlignTop,
'middle': Qt.AlignVCenter,
'bottom': Qt.AlignBottom
}
H_GRAV = {
'left': Qt.AlignLeft,
'center': Qt.AlignHCenter,
'right': Qt.AlignRight
}
GRAVITIES = {'%s-%s' % (x, y): V_GRAV[x] | H_GRAV[y]
for x in V_GRAV for y in H_GRAV}
def get_desktop_geom():
"""Get the box we must upscale to fill and the pieces to crop out"""
app = QApplication([])
deskwid = app.desktop()
desktop = QRect(0, 0, deskwid.width(), deskwid.height())
log.debug("Desktop rectangle: %s", desktop)
monitors = [deskwid.screenGeometry(x)
for x in range(deskwid.screenCount())]
log.debug("Monitor rectangles: %s", monitors)
return desktop, monitors
def write_backgrounds(image, monitors, outdir):
"""Given a QImage, a list of QRects, and a target directory, crop out
a bunch of numbered PNG files.
"""
for idx, geom in enumerate(monitors):
# We can't cleanly force a refresh in plasma, so we must instead
# provide single-image sets to Plasma's randomizer so it can notice
# the changes
outparent = os.path.join(outdir, '%d' % idx)
if not os.path.exists(outparent):
log.debug("Creating directory: %s", outparent)
os.makedirs(outparent)
outpath = os.path.join(outparent, 'piece.png')
log.info("Extracting %s and writing as %s", geom, outpath)
if not image.copy(geom).save(outpath):
log.warning("Failed to write image!")
def apply_gravity(image, tgt_rect, gravity):
"""Crop an image to a target rect based on a given alignment"""
# Verticals
if gravity & Qt.AlignTop:
tgt_rect.moveTop(0)
if gravity & Qt.AlignVCenter:
tgt_rect.moveCenter(QPoint(tgt_rect.center().x(),
int(image.height() / 2)))
elif gravity & Qt.AlignBottom:
tgt_rect.moveBottom(image.height())
# Horizontals
if gravity & Qt.AlignLeft:
tgt_rect.moveLeft(0)
if gravity & Qt.AlignHCenter:
tgt_rect.moveCenter(QPoint(int(image.width() / 2),
tgt_rect.center().y()))
elif gravity & Qt.AlignRight:
tgt_rect.moveRight(image.width())
log.debug("Cropping to %s", tgt_rect)
return image.copy(tgt_rect)
def set_background(img_path, outdir, gravity=0):
"""Given a path, [re]generate output files matching the monitors."""
if not os.path.exists(outdir):
raise IOError(errno.ENOENT, "Destination directory does not exist",
outdir)
if not os.path.isdir(outdir):
raise IOError(errno.ENOTDIR, "Destination path is not a directory",
outdir)
if not os.access(outdir, os.W_OK):
raise IOError(errno.EACCES, "Destination directory not writable",
outdir)
if not os.access(img_path, os.R_OK):
raise IOError(errno.EACCES, "Source file not readable",
outdir)
log.debug("Loading image: %s", img_path)
infile = QImage(img_path)
if infile.isNull():
raise ValueError("Could not load image: %s", img_path)
desktop, monitors = get_desktop_geom()
log.debug("Image size before: %s", infile.size())
infile = infile.scaled(desktop.size(),
Qt.KeepAspectRatioByExpanding,
Qt.SmoothTransformation)
log.debug("Image size after fitting to desktop with "
"KeepAspectRatioByExpanding: %s", infile.size())
infile = apply_gravity(infile, desktop, gravity)
write_backgrounds(infile, monitors, outdir)
def is_image(path):
"""Determine whether the given path represents a potential background"""
return os.path.splitext(path)[1].lower() in IMG_EXTS
def pick_recursive(roots):
"""Recursively traverse a list of paths and pick an image file."""
log.debug("Picking random file from: %s", roots)
potentials = []
for root in roots:
if os.path.isfile(root):
potentials.append(root)
elif os.path.isdir(root):
for path, _, files in os.walk(root):
for fname in files:
fpath = os.path.join(path, fname)
if is_image(fpath):
potentials.append(fpath)
return random.choice(potentials)
def main():
"""The main entry point, compatible with setuptools entry points."""
from argparse import ArgumentParser
parser = ArgumentParser(
description=__doc__.replace('\r\n', '\n').split('\n--snip--\n')[0])
parser.add_argument('--version', action='version',
version='%%(prog)s v%s' % __version__)
parser.add_argument('-v', '--verbose', action="count", default=2,
help="Increase verbosity. Use twice for extra effect")
parser.add_argument('-q', '--quiet', action="count", default=0,
help="Decrease verbosity. Use twice for extra effect")
parser.add_argument('--gravity', default='top-left', choices=GRAVITIES,
help="Choose how to align the image before cropping if"
" its aspect ratio doesn't match the desktop.")
parser.add_argument('--randomize', action="store_true", default=False,
help="Randomly select an image from the given paths")
parser.add_argument('paths', nargs='+')
parser.add_argument('outpath')
args = parser.parse_args()
# Set up clean logging to stderr
log_levels = [logging.CRITICAL, logging.ERROR, logging.WARNING,
logging.INFO, logging.DEBUG]
args.verbose = min(args.verbose - args.quiet, len(log_levels) - 1)
args.verbose = max(args.verbose, 0)
logging.basicConfig(level=log_levels[args.verbose],
format='%(levelname)s: %(message)s')
if args.randomize:
args.paths = [pick_recursive(args.paths)]
if args.paths:
path = args.paths[0]
if os.path.isfile(path):
log.debug("Requested alignment/gravity: %s", args.gravity)
try:
set_background(args.paths[0], args.outpath,
GRAVITIES[args.gravity])
except IOError as err:
log.error(err)
else:
log.error("Not a valid file: %s", path)
if __name__ == '__main__':
main()
# vim: set sw=4 sts=4 expandtab :

CC BY-SA 4.0 Desktop-spanning backgrounds in KDE 4.x by Stephan Sokolow is licensed under a Creative Commons Attribution-ShareAlike 4.0 International License.

This entry was posted in Geek Stuff. Bookmark the permalink.

2 Responses to Desktop-spanning backgrounds in KDE 4.x

  1. Nice effort, I tested this code, and it works decent for simple rectangular monitor configurations but isn’t robust enough to handle non-rectangular monitor configurations(KDE 4.13 clearly isn’t either as while I was testing your code, after switching my monitor orientation multiple times during testing, plasma finally got confused and all monitors were stuck in a state where nothing was responsive, no problem though as my mighty reboot powers thwarted the problem).

    Here are some suggestions:

    1) The images and background folder simply do not exist on my system, your code assumes that they do and raises an exception. I’m not exactly sure why images/background was chosen but if it needs to exist due to FreeDesktop specs or something of the like then, check if they exist and are writable first (don’t rely on exceptions, prevent them from ever occurring, because in this particular case, there is nothing exceptional about missing data, and your code will be more portable and robust as a result).

    IOError: [Errno 13] Destination directory not writable: %s: '/home/user/.local/share/images/background'

    2) Since your .desktop file is specific to geeqie, I would suggest mentioning that it needs to be installed in the /usr/share/geeqie/applications directory(I have no previous experience with geeqie). Furthermore, since you took the time to integrate it with geeqie to that extent, you really shouldn’t be looking in the users directory for the script.

    3) Rather than assuming the monitors will be in a rectangular configuration, you should be researching how to grab the entire multi-monitor resolution from KDE’s “Display Configuration” (It’s likely stored in rc config somewhere in ~/home/user/.kde/share/config/ , likely in plasma-desktoprc judging from a quick peek) and then scale the wallpaper accordingly and crop out the “dead zones” as you mentioned on the KDE forums( https://forum.kde.org/viewtopic.php?f=67&t=93190&sid=376c69f95bc29e2228305fabd886087d&start=15 ). I.E. all monitors are of equal resolution, top-left and bottom-right monitors in vertical orientation, top-right and bottom-left monitors are in horizontal orientation, thus leaving a “dead zone” in the center.

    4a) Your code needs to take into account monitor bezel width. KDE 4.13 “Display Configuration” does takes this into account but not in physical units, but in pixels. Unfortunately it does not allow you to manually specify what the pixel-bezel gap is between the monitors, but I’m sure this could be found and modified in the plasma-desktoprc file mentioned above. I.E. take the 4-monitor configuration from above and make it 5 by adding a monitor filling the “dead zone” (note that the resolution of this monitor would affect the “bezel” width. Based on the image cropping algorithm and monitor orientation listed in suggestion 3 above for how to crop images properly, the monitor in the “dead zone” should be set at whatever the maximum resolution of the “dead zone” is minus the center monitor’s bezel width). Grok that.

    4b) Now understand that every monitor can be set at a different resolution. For this next example, we are going to show the monitor configuration. See the attached image depicting the configuration: ( https://drive.google.com/file/d/0B8EHPTGJzAQ5ZV9XNll1TEdHWFE/view?usp=sharing ) The four outer monitors are drawn at a 800X600 size and the center monitor is drawn at a 180X180 size, all with a bezel size of 10(making a 20 gap between). In this configuration, the outer monitors are at a 4:3 ratio and the center monitor at a 1:1 ratio. While it’s possible that the center monitor would actually be a 1:1 ratio monitor with 180×180 resolution, it’s highly unlikely in almost any other user’s configuration other than this example.

    4c) Lets say that for the example the center monitor is set at a higher resolution capability than the outer monitors (1920×1920 center and 800×600 outer). What should the total background wallpaper resolution be? You should include bezel width since the center monitor bezel width cannot be ignored. 800+600+20=1420 virtual pixel width.

    4d) You finally have a desktop that handles a rectangle properly(or so I think). I’m not aware fully of how Wayland will work compared to X11, but I think this would be a step in the right direction for KDE. I’m a bit surprised that they consider this a low priority, but I’m sure it will elevate to a higher priority as they understand the problem better.

    I hope you and others have found this educational (if not essential) and improves KDE in the long term.

    Cheers,
    Jonathan

    Disclosure: I’m one of the developers for Embroidermodder 2 and working on a KDE4 thumbnailer for embroidery files FWIW. I’m a Life Member at Digital Blasphemy who was researching this problem a day after Stephan posted this.

    • Trust me, I wrote it specifically because I have a non-rectangular desktop and, based on what I know about the APIs I’m using, it should work for anything. If it’s failing for you, it’s probably a misconfiguration or bug outside my script but I’d be happy to investigate.

      I’ve posted an update to add log.debug() output so, when explaining your symptoms, please run it with -vv and paste the output.

      As for your points:

      1. ~/.local/ mirrors /usr/ and I mirrored /usr/share/images/ because that’s where Lubuntu was putting custom theming elements like panel backgrounds back when I started customizing my desktop’s appearance.

      (In other words, it’s XDG-inspired but, in this particular case, there’s no explicit technical reason for it)

      As for “Destination directory not writable”, that’s just it falling back to the general-case error message I wrote because I was too tired to decide whether there were cases where an os.makedirs() call would be unwelcome (given the potential for problems if there are typos and that no other command-line utility just blindly creates missing directories, with ancestors, like you suggest) and I then forgot about it the next morning.

      I’ve made the following changes:

      1. I suppressed the traceback and fed it through log.error() to make it more clear that it’s an expected error caused by bad input rather than an unexpected error caused by bad code.
      2. I added clearer error messages for “destination directory does not exist” and “destination path is not a directory”
      3. I’ve added screaming bold formatting to the part where it says that the .desktop file is intended as an example.
      4. I’ve added an all-caps disclaimer to the .desktop file indicating that it’s intended as an example and may require changes to work on your system.

      (Also, thanks for drawing attention to that formatting bug I introduced when I replaced a logging call (with lazy string formatting evaulation) as an IOError (which doesn’t support lazy string formatting evaluation but does have a three-argument form that prevented it from erroring out).

      2. The whole point of Geeqie using .desktop files to populate the Edit menu is that it does not need them to be in a special location. The .desktop file just specifies OnlyShowIn=X-Geeqie and anywhere on the normal .desktop lookup path will do. (That’s why I just used xdg-desktop-menu --install to drop it into ~/.local/share/applications/ and update the cache on my system.)

      However, I did accidentally post an old version which still had an accidental Hidden=true line, so I apologize for that. I’ve also added instructions to the .desktop file.

      3. The whole reason I wrote this is for non-rectangular desktops. If I had a rectangular desktop, I’d still be using PCManFM for desktop icons and LXPanel for my panels. (LXPanel doesn’t like non-rectangular desktops but lets me span a single panel and taskbar widget across all monitors.)

      In fact, when I wrote this, I hadn’t yet dug up a monitor raiser and my left-hand monitor only had about 2/3rds vertical overlap with the other two monitors. I have no clue why it isn’t working for you.

      Also, the monitor configuration is stored in the X11 server itself and accessed via the XRandR protocol extension. That’s why you can query and set it using non-KDE tools like the command-line xrandr utility or the nvidia-settings GUI.

      If you look at get_desktop_geom() (lines , you’ll see that it looks up the entire desktop’s rectangle (for scaling) and the geometry of each monitor within that rectangle (for copying out the individual monitor images).

      If you look at write_backgrounds(), you’ll see that it takes that geometry and uses it to determine where each individual image should be grabbed from within the desktop rectangle. I don’t know why that’s going wrong on your system, but my code is simple enough and works properly and reliably enough on my system (tested on both PyQt 4.x and PyQt 5.x) that it’s much more likely that a bug in X11 or Qt is causing your system to lie to my script about the monitor geometry.

      Also, there’s no need to quote “dead zones” when not talking about the term itself. It makes it sound like you’re saying it while making air quotes with your fingers but “dead space”, “dead areas”, or “dead zones” are the actual technical terms used.

      4a, 4b, 4c. If you look at something less primitive than the KDE display control panel (I use nvidia-settings and then copy the generated MetaModes line into my xorg.conf so it’s not DE-specific), you’ll see that you can set monitor positions without needing to fill the gaps. Under the hood, it’s just setting integer coordinates for rectangles within a larger rectangle. In fact, you can even overlap monitors. (Useful for things like hacking together a presentation where you want to mirror just one window on your desktop to the projector.)

      My script works as follows:

      1. Ask X11 for the desktop rectangle (the smallest rectangle which encompasses all monitors).
      2. Use aspect ratio-preserving scaling to fill the desktop (this makes it bigger than the desktop in one dimension unless the aspect ratios match perfectly)
      3. Crop the scaled image to the desktop shape to attain the alignment requested via --gravity
      4. Ask X11 for the (x, y, width, height) rectangles representing all defined monitors and then crop out and save each one as a background.

      (Note: All querying X11 is actually done at the beginning but, for simplicity’s sake, I haven’t written it that way.)

      Everything you’re saying I have to do is already done by X11 itself. In fact, the only thing I didn’t bother to do is provide an option to scale pieces of the background to compensate for varying physical pixel sizes. (Because it’s a lot of complexity and testing for a situation I’d never find myself in. it’d drive me crazy to have monitors of different DPIs on my system and I was too lazy to temporarily borrow a 17″ 1280×1024 monitor to set up next to my 19″ 1280×1024 monitors.)

      4d. Given that the APIs I use are used by things like games to figure out what window geometry to request, it’s quite likely that XWayland will expose them. As I’m on the nVidia binary drivers and, as a KDE user, wary of libinput’s “have one good set of defaults” mentality, Wayland hasn’t been a priority for me. (You’re talking to someone who hacked together a replacement for the update manager because they removed the ability to disable the “Reboot to update your kernel” nags. It’s entirely possible I’ll end up maintaining a fork of libinput to get some behaviour they decided wasn’t worth supporting.)

Leave a Reply

Your email address will not be published. Required fields are marked *

By submitting a comment here you grant this site a perpetual license to reproduce your words and name/web site in attribution under the same terms as the associated post.

All comments are moderated. If your comment is generic enough to apply to any post, it will be assumed to be spam. Borderline comments will have their URL field erased before being approved.