mirror of
git://git.sv.gnu.org/emacs.git
synced 2025-12-15 10:30:25 -08:00
* .dir-locals.el (c-mode): Add ANDROID_EXPORT noise macro. * .gitignore: Add new files to ignore. * Makefile.in: Adjust for Android. * admin/merge-gnulib: Add new warning. * configure.ac: Detect Android. Run cross-configuration for Android when appropriate. * etc/DEBUG: Document how to debug Emacs on Android. * java/AndroidManifest.xml: * java/Makefile.in: * java/README: * java/debug.sh: * java/org/gnu/emacs/EmacsActivity.java (EmacsActivity): * java/org/gnu/emacs/EmacsApplication.java (EmacsApplication): * java/org/gnu/emacs/EmacsCopyArea.java (EmacsCopyArea): * java/org/gnu/emacs/EmacsDrawLine.java (EmacsDrawLine): * java/org/gnu/emacs/EmacsDrawPoint.java (EmacsDrawPoint): * java/org/gnu/emacs/EmacsDrawRectangle.java (EmacsDrawRectangle): * java/org/gnu/emacs/EmacsDrawable.java (EmacsDrawable): * java/org/gnu/emacs/EmacsFillPolygon.java (EmacsFillPolygon): * java/org/gnu/emacs/EmacsFillRectangle.java (EmacsFillRectangle): * java/org/gnu/emacs/EmacsFontDriver.java (EmacsFontDriver): * java/org/gnu/emacs/EmacsGC.java (EmacsGC): * java/org/gnu/emacs/EmacsHandleObject.java (EmacsHandleObject): * java/org/gnu/emacs/EmacsNative.java (EmacsNative): * java/org/gnu/emacs/EmacsPaintQueue.java (EmacsPaintQueue): * java/org/gnu/emacs/EmacsPaintReq.java (EmacsPaintReq): * java/org/gnu/emacs/EmacsPixmap.java (EmacsPixmap): * java/org/gnu/emacs/EmacsSdk7FontDriver.java (EmacsSdk7FontDriver): * java/org/gnu/emacs/EmacsService.java (class Holder<T>) (EmacsService): * java/org/gnu/emacs/EmacsSurfaceView.java (EmacsSurfaceView): * java/org/gnu/emacs/EmacsThread.java (EmacsThread): * java/org/gnu/emacs/EmacsView.java (EmacsView): * java/org/gnu/emacs/EmacsWindow.java (EmacsWindow): New files and classes. * lib-src/Makefile.in (srcdir): * lib/Makefile.in (VPATH): (HAVE_NATIVE_COMP): (libgnu_a_SOURCES): (DEPFLAGS): Configure correctly for cross-compiling. * lib/faccessat.c: * lib/fpending.c (__fpending): * lib/open.c: * lib/unistd.c (_GL_UNISTD_INLINE): Temporary adjustments to gnulib. * lisp/frame.el (display-graphic-p): (display-screens): (display-pixel-height): (display-pixel-width): (display-mm-height): (display-mm-width): (display-backing-store): (display-save-under): (display-planes): (display-color-cells): (display-visual-class): Adjust for new window system `android'. * lisp/image/wallpaper.el (x-open-connection): Add declaration. * lisp/loadup.el (featurep): Load up files for Android. * lisp/net/eww.el (eww-form-submit, eww-form-file) (eww-form-checkbox, eww-form-select): Adjust faces for android. * lisp/term/android-win.el: New file. * src/Makefile.in: Add new targets emacs.so and android-emacs, then adjust for cross compilation. * src/alloc.c (cleanup_vector): Clean up Android font entities as well. (garbage_collect): Mark androidterm. * src/android-emacs.c (main): * src/android.c (ANDROID_THROW, enum android_fd_table_entry_flags) (struct android_emacs_service, struct android_emacs_pixmap) (struct android_graphics_point, struct android_event_container) (struct android_event_queue, android_run_select_thread) (android_handle_sigusr1, android_init_events, android_pending) (android_next_event, android_write_event, android_select) (android_run_debug_thread, android_user_full_name) (android_get_asset_name, android_fstat, android_fstatat) (android_file_access_p, android_hack_asset_fd, android_open) (android_close, JNICALL, android_init_emacs_service) (android_init_emacs_pixmap, android_init_graphics_point) (MAX_HANDLE, struct android_handle_entry, android_alloc_id) (android_destroy_handle, android_resolve_handle) (android_resolve_handle2, android_change_window_attributes) (android_create_window, android_set_window_background) (android_destroy_window, android_init_android_rect_class) (android_init_emacs_gc_class, android_create_gc, android_free_gc) (android_change_gc, android_set_clip_rectangles) (android_reparent_window, android_lookup_method) (android_clear_window, android_map_window, android_unmap_window) (android_resize_window, android_move_window, android_swap_buffers) (android_get_gc_values, android_set_foreground) (android_fill_rectangle, android_create_pixmap_from_bitmap_data) (android_set_clip_mask, android_set_fill_style, android_copy_area) (android_free_pixmap, android_set_background, android_fill_polygon) (android_draw_rectangle, android_draw_point, android_draw_line) (android_create_pixmap, android_set_ts_origin, android_clear_area): * src/android.h (ANDROID_EXPORT): * src/androidfns.c (android_display_info_for_name) (check_android_display_info, check_x_display_info, gamma_correct) (android_defined_color, android_decode_color) (android_implicitly_set_name, android_explicitly_set_name) (android_set_tool_bar_lines, android_change_tool_bar_height) (android_set_tab_bar_lines, android_change_tab_bar_height) (android_set_scroll_bar_default_height) (android_set_scroll_bar_default_width, android_icon_verify) (android_icon, android_make_gc, android_free_gcs) (unwind_create_frame, do_unwind_create_frame) (android_default_font_parameter, android_create_frame_window) (Fx_create_frame, Fxw_color_defined_p, Fxw_color_values) (Fxw_display_color_p, Fx_display_grayscale_p) (Fx_display_pixel_width, Fx_display_pixel_height) (Fx_display_planes, Fx_display_color_cells, Fx_display_screens) (Fx_display_mm_width, Fx_display_mm_height) (Fx_display_backing_store, Fx_display_visual_class) (Fx_display_monitor_attributes_list, Fx_frame_geometry) (Fx_frame_list_z_order, Fx_frame_restack) (Fx_mouse_absolute_pixel_position) (Fx_set_mouse_absolute_pixel_position, Fandroid_get_connection) (Fx_display_list, Fx_show_tip, Fx_hide_tip) (android_set_background_color, android_set_border_color) (android_set_cursor_color, android_set_cursor_type) (android_set_foreground_color) (android_set_child_frame_border_width) (android_set_internal_border_width, android_set_menu_bar_lines) (android_set_mouse_color, android_set_title, android_set_alpha) (android_frame_parm_handlers, syms_of_androidfns): * src/androidfont.c (struct android_emacs_font_driver) (struct android_emacs_font_spec, struct android_emacs_font_metrics) (struct android_emacs_font_object, struct android_integer) (struct androidfont_info, struct androidfont_entity) (android_init_font_driver, android_init_font_spec) (android_init_font_metrics, android_init_integer) (android_init_font_object, androidfont_get_cache) (androidfont_from_lisp, androidfont_from_java, androidfont_list) (androidfont_match, androidfont_draw, androidfont_open_font) (androidfont_close_font, androidfont_has_char) (androidfont_encode_char, androidfont_text_extents) (androidfont_list_family, androidfont_driver) (syms_of_androidfont_for_pdumper, syms_of_androidfont) (init_androidfont, android_finalize_font_entity): * src/androidgui.h (_ANDROID_GUI_H_, struct android_rectangle) (struct android_point, enum android_gc_function) (enum android_gc_value_mask, enum android_fill_style) (enum android_window_value_mask) (struct android_set_window_attributes, struct android_gc_values) (struct android_gc, enum android_swap_action, enum android_shape) (enum android_coord_mode, struct android_swap_info) (NativeRectangle, struct android_any_event) (struct android_key_event, struct android_configure_event) (union android_event): * src/androidterm.c (android_window_to_frame, android_clear_frame) (android_ring_bell, android_toggle_invisible_pointer) (android_update_begin, android_update_end, show_back_buffer) (android_flush_dirty_back_buffer_on, handle_one_android_event) (android_read_socket, android_frame_up_to_date) (android_buffer_flipping_unblocked_hook) (android_query_frame_background_color, android_parse_color) (android_alloc_nearest_color, android_query_colors) (android_mouse_position, android_get_focus_frame) (android_focus_frame, android_frame_rehighlight) (android_frame_raise_lower, android_make_frame_visible) (android_make_frame_invisible) (android_make_frame_visible_invisible, android_fullscreen_hook) (android_iconify_frame, android_set_window_size_1) (android_set_window_size, android_set_offset, android_set_alpha) (android_new_font, android_bitmap_icon, android_free_pixmap_hook) (android_free_frame_resources, android_delete_frame) (android_delete_terminal, android_scroll_run) (android_after_update_window_line, android_flip_and_flush) (android_clear_rectangle, android_reset_clip_rectangles) (android_clip_to_row, android_draw_fringe_bitmap) (android_set_cursor_gc, android_set_mouse_face_gc) (android_set_mode_line_face_gc, android_set_glyph_string_gc) (android_set_glyph_string_clipping) (android_set_glyph_string_clipping_exactly) (android_compute_glyph_string_overhangs) (android_clear_glyph_string_rect) (android_draw_glyph_string_background, android_fill_triangle) (android_make_point, android_inside_rect_p, android_clear_point) (android_draw_relief_rect, android_draw_box_rect) (HIGHLIGHT_COLOR_DARK_BOOST_LIMIT, android_setup_relief_color) (android_setup_relief_colors, android_draw_glyph_string_box) (android_draw_glyph_string_bg_rect, android_draw_image_relief) (android_draw_image_foreground, android_draw_image_foreground_1) (android_draw_image_glyph_string) (android_draw_stretch_glyph_string, android_draw_underwave) (android_draw_glyph_string_foreground) (android_draw_composite_glyph_string_foreground) (android_draw_glyphless_glyph_string_foreground) (android_draw_glyph_string, android_define_frame_cursor) (android_clear_frame_area, android_clear_under_internal_border) (android_draw_hollow_cursor, android_draw_bar_cursor) (android_draw_window_cursor, android_draw_vertical_window_border) (android_draw_window_divider, android_redisplay_interface) (frame_set_mouse_pixel_position, get_keysym_name) (android_create_terminal, android_term_init, syms_of_androidterm) (mark_androidterm): * src/androidterm.h (_ANDROID_TERM_H_, struct android_display_info) (struct android_output, FRAME_ANDROID_OUTPUT, XSCROLL_BAR): New files. * src/dired.c (file_attributes): Do not use openat on Android. * src/dispextern.h (No_Cursor): Define appropriately on Android. (struct glyph_string, struct face): Make gc field of type struct android_gc on Android. * src/dispnew.c (clear_current_matrices, clear_desired_matrices) (adjust_frame_glyphs_for_window_redisplay, free_glyphs) (update_frame, scrolling, char_ins_del_cost, update_frame_line) (init_display_interactive): Disable text terminal support completely on Android. Fix non-toolkit menus for non-X systems. * src/editfns.c (Fuser_full_name): Call android_user_full_name. * src/emacs.c (android_emacs_init): Make main this on Android. Prohibit argv sorting from exceeding end of argv. * src/epaths.in: Add path definitions for Android. * src/fileio.c (file_access_p): Call android_file_access_p. (file_name_directory): Avoid using openat on Android. (Fcopy_file): Adjust to call sys_fstat instead. (file_directory_p): (Finsert_file_contents): (write_region): Likewise. * src/filelock.c: * src/fns.c (Flocale_info): Pacify warning on Android. * src/font.c (font_make_entity_android): New function. * src/font.h: * src/frame.c (Fframep): (Fwindow_system): Handle new window system `android'. Update doc strings. (Fmake_terminal_frame): Disable on Android. (gui_display_get_resource): Disable get_string_resource_hook on Android. (syms_of_frame): New defsym `android'. * src/frame.h (GCALIGNED_STRUCT): Add new output data for Android. (ENUM_BF): Expand enumerator size. (FRAME_ANDROID_P, FRAME_WINDOW_P, MOUSE_HL_INFO): Add definitions for Android. * src/image.c (GET_PIXEL): (image_create_bitmap_from_file): (image_create_x_image_and_pixmap_1): (image_get_x_image): (slurp_file): (lookup_rgb_color): (image_to_emacs_colors): (image_from_emacs_colors): (image_pixmap_draw_cross): (image_disable_image): (MaskForeground): (gif_load): Add stubs for Android. * src/lisp.h: * src/lread.c (safe_to_load_version, maybe_swap_for_eln1, openp): * src/pdumper.c (pdumper_load): Call sys_fstat instead of fstat. * src/process.c (wait_reading_process_output): Use android_select instead of pselect. * src/scroll.c: Disable on Android. * src/sysdep.c (widen_foreground_group, reset_sys_modes) (init_signals, emacs_fstatat, sys_fstat): New function. (emacs_open, emacs_open_noquit, emacs_close): Implement differently on Android. (close_output_streams): Disable what is not required on Android. * src/term.c (OUTPUT1_IF, encode_terminal_code, string_cost) (string_cost_one_line, per_line_cost, calculate_costs) (struct fkey_table, tty_append_glyph, produce_glyphs) (tty_capable_p, Fsuspend_tty, Fresume_tty, device, init_tty) (maybe_fatal, syms_of_term): Disable text terminal support on Android. * src/termhooks.h (enum output_method): Add android output method. (GCALIGNED_STRUCT, TERMINAL_FONT_CACHE): Define for Android. * src/terminal.c (Fterminal_live_p): Implement for Android. * src/verbose.mk.in (AM_V_GLOBALS): Add JAVAC and DX. * src/xdisp.c (redisplay_internal): Disable text terminals on Android. (display_menu_bar): (display_tty_menu_item): (draw_row_with_mouse_face): (expose_frame): Make the non toolkit menu bar work on Android. * src/xfaces.c (GCGraphicsExposures): (x_create_gc): (x_free_gc): (Fx_load_color_file): Define for Android. * xcompile/Makefile.in (top_srcdir): (top_builddir): * xcompile/README: * xcompile/langinfo.h (nl_langinfo): New files.
588 lines
23 KiB
EmacsLisp
588 lines
23 KiB
EmacsLisp
;;; wallpaper.el --- Change the desktop background -*- lexical-binding: t; -*-
|
||
|
||
;; Copyright (C) 2022 Free Software Foundation, Inc.
|
||
|
||
;; Author: Stefan Kangas <stefankangas@gmail.com>
|
||
;; Keywords: images
|
||
|
||
;; This file is part of GNU Emacs.
|
||
|
||
;; GNU Emacs is free software: you can redistribute it and/or modify
|
||
;; it under the terms of the GNU General Public License as published by
|
||
;; the Free Software Foundation, either version 3 of the License, or
|
||
;; (at your option) any later version.
|
||
|
||
;; GNU Emacs is distributed in the hope that it will be useful,
|
||
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
;; GNU General Public License for more details.
|
||
|
||
;; You should have received a copy of the GNU General Public License
|
||
;; along with GNU Emacs. If not, see <https://www.gnu.org/licenses/>.
|
||
|
||
;;; Commentary:
|
||
|
||
;; This library provides the command `wallpaper-set', which sets the
|
||
;; desktop background.
|
||
;;
|
||
;; On GNU/Linux and other Unix-like systems, it uses an external
|
||
;; command to set the desktop background. This should work seamlessly
|
||
;; on both X and Wayland.
|
||
;;
|
||
;; Finding an external command to use is obviously a bit tricky to get
|
||
;; right, as there is no lack of platforms, window managers, desktop
|
||
;; environments and tools. However, it should be detected
|
||
;; automatically in most cases. If it doesn't work in your
|
||
;; environment, customize the user options `wallpaper-command' and
|
||
;; `wallpaper-command-args'.
|
||
;;
|
||
;; On MS-Windows, it uses the `w32-set-wallpaper' function, and on
|
||
;; Haiku the `haiku-set-wallpaper' function, neither of which relies
|
||
;; on any external commands. The value of `wallpaper-command' and
|
||
;; `wallpaper-command-args' are ignored on such systems.
|
||
;;
|
||
;; On macOS, the "osascript" command is used. You might need to
|
||
;; disable the option "Change picture" in the "Desktop & Screensaver"
|
||
;; preferences for this to work (this was seen with macOS 10.13).
|
||
;; You might also have to tweak some permissions.
|
||
;;
|
||
;; Note: If you find that you need to use a command in your
|
||
;; environment that was not automatically detected, we would love to
|
||
;; hear about it! Please send an email to bug-gnu-emacs@gnu.org and
|
||
;; tell us the command (and all options) that worked for you. You can
|
||
;; also use `M-x report-emacs-bug'.
|
||
|
||
;;; Code:
|
||
|
||
(eval-when-compile (require 'subr-x))
|
||
(require 'xdg)
|
||
(require 'cl-macs)
|
||
|
||
(defvar wallpaper-debug nil
|
||
"If non-nil, display debug messages.")
|
||
|
||
(defun wallpaper-debug (&rest args)
|
||
(when wallpaper-debug
|
||
(apply #'message
|
||
(concat "wallpaper-debug: " (car args))
|
||
(cdr args))))
|
||
|
||
(defvar wallpaper-set-function
|
||
(cond ((fboundp 'w32-set-wallpaper)
|
||
#'w32-set-wallpaper)
|
||
((and (fboundp 'haiku-set-wallpaper)
|
||
(featurep 'haiku))
|
||
'haiku-set-wallpaper)
|
||
(#'wallpaper-default-set-function))
|
||
"Function used by `wallpaper-set' to set the wallpaper.
|
||
The function takes one argument, FILE, which is the file name of
|
||
the image file to set the wallpaper to.")
|
||
|
||
(defun wallpaper--use-default-set-function-p ()
|
||
(eq wallpaper-set-function #'wallpaper-default-set-function))
|
||
|
||
|
||
;;; Finding the wallpaper command
|
||
|
||
(cl-defstruct (wallpaper-setter
|
||
;; Get rid of the default constructor (`make-wallpaper-cmd').
|
||
(:constructor nil)
|
||
(:constructor
|
||
wallpaper-setter-create
|
||
( name command args-raw
|
||
&rest rest-plist
|
||
&aux
|
||
(args (if (or (listp args-raw) (symbolp args-raw))
|
||
args-raw
|
||
(string-split args-raw)))
|
||
(predicate (plist-get rest-plist :predicate))
|
||
(init-action (plist-get rest-plist :init-action))
|
||
(detach (plist-get rest-plist :detach))))
|
||
(:copier wallpaper-setter-copy))
|
||
"Structure containing a method to set the wallpaper.
|
||
|
||
NAME is a description of the setter (e.g. the name of the Desktop
|
||
Environment).
|
||
|
||
COMMAND is the executable to run to set the wallpaper.
|
||
|
||
ARGS is the default list of command line arguments for COMMAND.
|
||
|
||
PREDICATE is a function that will be called without any arguments
|
||
and returns non-nil if this setter should be used.
|
||
|
||
INIT-ACTION is a function that will be called without any
|
||
arguments before trying to set the wallpaper.
|
||
|
||
DETACH, if non-nil, means that the wallpaper process should
|
||
continue running even after exiting Emacs."
|
||
name
|
||
command
|
||
args
|
||
(predicate #'always)
|
||
init-action
|
||
detach)
|
||
|
||
;;;###autoload
|
||
(put 'wallpaper-setter-create 'lisp-indent-function 1)
|
||
|
||
(defun wallpaper--init-action-kill (process-name)
|
||
"Return kill function for `init-action' of a `wallpaper-setter' structure.
|
||
The returned function kills any process named PROCESS-NAME owned
|
||
by the current effective user id."
|
||
(lambda ()
|
||
(when-let ((procs
|
||
(seq-filter (lambda (p) (let-alist p
|
||
(and (= .euid (user-uid))
|
||
(equal .comm process-name))))
|
||
(mapcar (lambda (pid)
|
||
(cons (cons 'pid pid)
|
||
(process-attributes pid)))
|
||
(list-system-processes)))))
|
||
(dolist (proc procs)
|
||
(let-alist proc
|
||
(when (y-or-n-p (format "Kill \"%s\" process with PID %d?" .comm .pid))
|
||
(signal-process .pid 'TERM)))))))
|
||
|
||
(defmacro wallpaper--default-methods-create (&rest items)
|
||
"Helper macro for defining `wallpaper--default-setters'."
|
||
(cons 'list
|
||
(mapcar
|
||
(lambda (item)
|
||
`(wallpaper-setter-create ,@item))
|
||
items)))
|
||
|
||
(defvar wallpaper--default-setters
|
||
(wallpaper--default-methods-create
|
||
|
||
;; macOS.
|
||
;; NB. Should come first to override everything else.
|
||
("macOS"
|
||
"osascript"
|
||
'("-e" "tell application \"Finder\" to set desktop picture to POSIX file \"%f\"")
|
||
:predicate (lambda ()
|
||
(eq system-type 'darwin)))
|
||
|
||
;; Desktop environments.
|
||
("Gnome"
|
||
"gsettings"
|
||
"set org.gnome.desktop.background picture-uri file://%F"
|
||
:predicate (lambda ()
|
||
(or (and (getenv "DESKTOP_SESSION")
|
||
(member (downcase (getenv "DESKTOP_SESSION"))
|
||
'("gnome" "gnome" "gnome-wayland" "gnome-xorg"
|
||
"unity" "ubuntu" "pantheon" "budgie-desktop"
|
||
"pop")))
|
||
(member "GNOME" (xdg-current-desktop))
|
||
(member "Budgie" (xdg-current-desktop))
|
||
(member "GNOME-Classic" (xdg-current-desktop)))))
|
||
|
||
("KDE Plasma"
|
||
"plasma-apply-wallpaperimage" "%f"
|
||
:predicate (lambda ()
|
||
(member "KDE" (xdg-current-desktop))))
|
||
|
||
("XFCE"
|
||
"xfconf-query" #'wallpaper-xfce-command-args
|
||
:predicate (lambda ()
|
||
(or (and (getenv "DESKTOP_SESSION")
|
||
(member (downcase (getenv "DESKTOP_SESSION"))
|
||
'("xubuntu" "ubuntustudio")))
|
||
(member "XFCE" (xdg-current-desktop)))))
|
||
|
||
("LXDE"
|
||
"pcmanfm" "--set-wallpaper=%f"
|
||
:predicate (lambda ()
|
||
(member "LXDE" (xdg-current-desktop))))
|
||
|
||
("LXQt"
|
||
"pcmanfm-qt" "--set-wallpaper=%f" ; "--wallpaper-mode=MODE"
|
||
:predicate (lambda ()
|
||
(or (member (and (getenv "DESKTOP_SESSION")
|
||
(downcase (getenv "DESKTOP_SESSION")))
|
||
'("lubuntu" "lxqt"))
|
||
(member "LXQt" (xdg-current-desktop)))))
|
||
|
||
("Mate"
|
||
"gsettings" "set org.mate.background picture-filename %f"
|
||
:predicate (lambda ()
|
||
(or (and (getenv "DESKTOP_SESSION")
|
||
(equal "mate" (downcase (getenv "DESKTOP_SESSION"))))
|
||
(member "MATE" (xdg-current-desktop)))))
|
||
|
||
("Cinnamon"
|
||
"gsettings" "set org.cinnamon.desktop.background picture-uri file://%F"
|
||
:predicate (lambda ()
|
||
(or (equal "cinnamon" (and (getenv "DESKTOP_SESSION")
|
||
(downcase (getenv "DESKTOP_SESSION"))))
|
||
(member "X-Cinnamon" (xdg-current-desktop)))))
|
||
|
||
("Deepin"
|
||
"gsettings" "set com.deepin.wrap.gnome.desktop.background picture-uri file://%F"
|
||
:predicate (lambda ()
|
||
(member "Deepin" (xdg-current-desktop))))
|
||
|
||
;; Wayland general.
|
||
("Sway (Wayland)"
|
||
"swaybg" "-o * -i %f -m fill"
|
||
:predicate (lambda ()
|
||
(and (getenv "WAYLAND_DISPLAY")
|
||
(getenv "SWAYSOCK")))
|
||
:init-action (wallpaper--init-action-kill "swaybg")
|
||
:detach t)
|
||
|
||
("wbg"
|
||
"wbg" "%f"
|
||
:predicate (lambda ()
|
||
(getenv "WAYLAND_DISPLAY"))
|
||
:init-action (wallpaper--init-action-kill "wbg")
|
||
:detach t)
|
||
|
||
;; X general.
|
||
("GraphicsMagick"
|
||
"gm" "display -size %wx%h -window root %f")
|
||
|
||
("ImageMagick"
|
||
"display" "-resize %wx%h -window root %f")
|
||
|
||
("feh"
|
||
"feh" "--bg-max %f")
|
||
|
||
("fbsetbg"
|
||
"fbsetbg" "-a %f")
|
||
|
||
("xwallpaper"
|
||
"xwallpaper" "--zoom %f")
|
||
|
||
("hsetroot"
|
||
"hsetroot" "-full %f")
|
||
|
||
("xloadimage"
|
||
"xloadimage" "-onroot -fullscreen %f")
|
||
|
||
("xsetbg"
|
||
"xsetbg" "%f")
|
||
)
|
||
"List of setters used for setting the wallpaper.
|
||
Every item in the list is a structure of type
|
||
`wallpaper-setter' (which see).
|
||
|
||
This is used by `wallpaper--find-command' to automatically set
|
||
`wallpaper-command', and by `wallpaper--find-command-args' to set
|
||
`wallpaper-command-args'. The setters will be tested in the
|
||
order in which they appear.")
|
||
|
||
(defun wallpaper-xfce-command-args ()
|
||
(let ((info
|
||
(with-temp-buffer
|
||
(call-process "xfconf-query" nil t nil
|
||
"-c" "xfce4-desktop"
|
||
"-p" "/backdrop/single-workspace-mode")
|
||
(buffer-string))))
|
||
(list "-c" "xfce4-desktop"
|
||
"-p" (format "/backdrop/screen%%S/monitor%%M/workspace%s/last-image"
|
||
(if (equal info "true")
|
||
"0"
|
||
"%W"))
|
||
"-s" "%f")))
|
||
|
||
(defvar wallpaper--current-setter nil)
|
||
|
||
(defun wallpaper--find-setter ()
|
||
(when (wallpaper--use-default-set-function-p)
|
||
(or (and (wallpaper-setter-p wallpaper--current-setter)
|
||
wallpaper--current-setter)
|
||
(setq wallpaper--current-setter
|
||
(catch 'found
|
||
(dolist (setter wallpaper--default-setters)
|
||
(wallpaper-debug "Testing setter %s" (wallpaper-setter-name setter))
|
||
(when (and (executable-find (wallpaper-setter-command setter))
|
||
(if-let ((pred (wallpaper-setter-predicate setter)))
|
||
(funcall pred)
|
||
t))
|
||
(wallpaper-debug "Found setter %s" (wallpaper-setter-name setter))
|
||
(throw 'found setter))))))))
|
||
|
||
(defun wallpaper--find-command ()
|
||
"Return a valid command to set the wallpaper in this environment."
|
||
(when-let ((setter (wallpaper--find-setter)))
|
||
(wallpaper-setter-command setter)))
|
||
|
||
(defun wallpaper--find-command-args ()
|
||
"Return command line arguments matching `wallpaper-command'."
|
||
(when-let ((setter (wallpaper--find-setter)))
|
||
(wallpaper-setter-args setter)))
|
||
|
||
|
||
;;; Customizable variables
|
||
|
||
(defvar wallpaper-command-args) ; silence byte-compiler
|
||
(defun wallpaper--set-wallpaper-command (sym val)
|
||
"Set `wallpaper-command', and update `wallpaper-command-args'.
|
||
Used to set `wallpaper-command'."
|
||
;; Note: `wallpaper-command' is used by `wallpaper--find-command-args'.
|
||
(prog1 (set-default sym val)
|
||
(set-default 'wallpaper-command-args
|
||
(wallpaper--find-command-args))))
|
||
|
||
(defcustom wallpaper-command (wallpaper--find-command)
|
||
"Executable used by `wallpaper-set' for setting the wallpaper.
|
||
A suitable command for your environment should be detected
|
||
automatically, so there is usually no need to customize this.
|
||
|
||
If you set this to any supported command using customize or
|
||
`setopt', the user option `wallpaper-command-args' is
|
||
automatically updated to match. If you need to change this to an
|
||
unsupported command, you will want to manually customize
|
||
`wallpaper-command-args' to match.
|
||
|
||
The value of this variable is ignored on MS-Windows and Haiku
|
||
systems, where a native API is used instead."
|
||
:type
|
||
'(choice
|
||
(radio
|
||
(const :tag "gsettings (GNOME)" "gsettings")
|
||
(const :tag "plasma-apply-wallpaperimage (KDE Plasma)" "plasma-apply-wallpaperimage")
|
||
(const :tag "xfconf-query (XFCE)" "xfconf-query")
|
||
(const :tag "pcmanf (LXDE)" "pcmanf")
|
||
(const :tag "pcmanf-qt (LXQt)" "pcmanf-qt")
|
||
(const :tag "swaybg (Wayland/Sway)" "swaybg")
|
||
(const :tag "wbg (Wayland)" "wbg")
|
||
(const :tag "gm (X Window System)" "gm")
|
||
(const :tag "display (X Window System)" "display")
|
||
(const :tag "feh (X Window System)" "feh")
|
||
(const :tag "fbsetbg (X Window System)" "fbsetbg")
|
||
(const :tag "xwallpaper (X Window System)" "xwallpaper")
|
||
(const :tag "hsetroot (X Window System)" "hsetroot")
|
||
(const :tag "xloadimage (X Window System)" "xloadimage")
|
||
(const :tag "xsetbg (X Window System)" "xsetbg")
|
||
(const :tag "osascript (macOS)" "osascript"))
|
||
(const :tag "Other (specify)" string)
|
||
(const :tag "None" nil))
|
||
:set #'wallpaper--set-wallpaper-command
|
||
:group 'image
|
||
:version "29.1")
|
||
|
||
(defcustom wallpaper-command-args (wallpaper--find-command-args)
|
||
"Command line arguments for `wallpaper-command'.
|
||
A suitable command for your environment should be detected
|
||
automatically, so there is usually no need to customize this.
|
||
However, if you do need to change this, you might also want to
|
||
customize `wallpaper-command' to match.
|
||
|
||
The value is a list of command list arguments to use, or a
|
||
function that returns a list of command line arguments.
|
||
|
||
In each command line argument, these specifiers will be replaced:
|
||
|
||
%f full file name
|
||
%h height of the selected frame's display (as returned
|
||
by `display-pixel-height')
|
||
%w the width of the selected frame's display (as returned
|
||
by `display-pixel-width').
|
||
%F full file name URI-encoded
|
||
%S current X screen (e.g. \"0\")
|
||
%W current workspace (e.g., \"0\")
|
||
%M name of the monitor (e.g., \"0\" or \"LVDS\")
|
||
|
||
If `wallpaper-set' is run from a TTY frame, instead prompt for a
|
||
height and width to use for %h and %w.
|
||
|
||
The value of this variable is ignored on MS-Windows and Haiku
|
||
systems, where a native API is used instead."
|
||
:type '(choice (repeat string)
|
||
function)
|
||
:group 'image
|
||
:version "29.1")
|
||
|
||
|
||
;;; Utility functions
|
||
|
||
(defvar wallpaper-default-width 1080
|
||
"Default width used by `wallpaper-set'.
|
||
This is only used when it can't be detected automatically.
|
||
See also `wallpaper-default-height'.")
|
||
|
||
(defvar wallpaper-default-height 1920
|
||
"Default height used by `wallpaper-set'.
|
||
This is only used when it can't be detected automatically.
|
||
See also `wallpaper-default-width'.")
|
||
|
||
(defun wallpaper--get-height-or-width (desc fun default)
|
||
(cond ((display-graphic-p) (funcall fun))
|
||
(noninteractive default)
|
||
((read-number (format "Wallpaper %s in pixels: " desc) default))))
|
||
|
||
(autoload 'ffap-file-at-point "ffap")
|
||
|
||
(defvar wallpaper-image-file-extensions
|
||
'("bmp" "gif" "heif" "jpeg" "jpg" "png" "tif" "tiff" "webp")
|
||
"List of file extensions that `wallpaper-set' will consider for completion.")
|
||
|
||
(defun wallpaper--image-file-regexp ()
|
||
(rx-to-string '(: "." (eval `(or ,@wallpaper-image-file-extensions)) eos) t))
|
||
|
||
(defun wallpaper--get-default-file ()
|
||
(catch 'found
|
||
(dolist (file (list buffer-file-name (ffap-file-at-point)))
|
||
(when (and file (string-match (wallpaper--image-file-regexp) file))
|
||
(throw 'found (abbreviate-file-name
|
||
(expand-file-name file)))))))
|
||
|
||
|
||
;;; wallpaper-set
|
||
|
||
(declare-function x-open-connection "xfns.c")
|
||
|
||
(defun wallpaper--x-monitor-name ()
|
||
"Get the monitor name for `wallpaper-set'.
|
||
On a graphical display, try using the same monitor as the current
|
||
frame.
|
||
On a non-graphical display, try to get the name by connecting to
|
||
the display server directly, and run \"xrandr\" if that doesn't
|
||
work. Prompt for the monitor name if neither method works.
|
||
|
||
This function is meaningful only on X and is used only there."
|
||
(if (or (display-graphic-p)
|
||
noninteractive)
|
||
(let-alist (car (display-monitor-attributes-list))
|
||
(if (and .name (member .source '("XRandr" "XRandR 1.5" "Gdk")))
|
||
.name
|
||
"0"))
|
||
(if-let ((name
|
||
(and (getenv "DISPLAY")
|
||
(or
|
||
(cdr (assq 'name
|
||
(progn
|
||
(x-open-connection (getenv "DISPLAY"))
|
||
(car (display-monitor-attributes-list
|
||
(car (last (terminal-list))))))))
|
||
(and (executable-find "xrandr")
|
||
(with-temp-buffer
|
||
(call-process "xrandr" nil t nil)
|
||
(goto-char (point-min))
|
||
(re-search-forward (rx bol
|
||
(group (+ (not (in " \n"))))
|
||
" connected")
|
||
nil t)
|
||
(match-string 1)))))))
|
||
;; Prefer "0" to "default" as that works in XFCE.
|
||
(if (equal name "default") "0" name)
|
||
(read-string (format-prompt "Monitor name" nil)))))
|
||
|
||
(defun wallpaper--format-arg (format file)
|
||
"Format a `wallpaper-command-args' argument ARG.
|
||
FILE is the image file name."
|
||
(format-spec
|
||
format
|
||
`((?f . ,(expand-file-name file))
|
||
(?F . ,(lambda ()
|
||
(mapconcat #'url-hexify-string
|
||
(file-name-split file)
|
||
"/")))
|
||
(?h . ,(lambda ()
|
||
(wallpaper--get-height-or-width
|
||
"height"
|
||
#'display-pixel-height
|
||
wallpaper-default-height)))
|
||
(?w . ,(lambda ()
|
||
(wallpaper--get-height-or-width
|
||
"width"
|
||
#'display-pixel-width
|
||
wallpaper-default-width)))
|
||
;; screen number
|
||
(?S . ,(lambda ()
|
||
(let ((display (frame-parameter (selected-frame) 'display)))
|
||
(if (and display
|
||
(string-match (rx ":" (+ (in "0-9")) "."
|
||
(group (+ (in "0-9"))) eos)
|
||
display))
|
||
(match-string 1 display)
|
||
"0"))))
|
||
;; monitor name
|
||
(?M . ,#'wallpaper--x-monitor-name)
|
||
;; workspace
|
||
(?W . ,(lambda ()
|
||
(or (and (fboundp 'x-window-property)
|
||
(display-graphic-p)
|
||
(number-to-string
|
||
(or (x-window-property "_NET_CURRENT_DESKTOP" nil "CARDINAL" 0 nil t)
|
||
(x-window-property "WIN_WORKSPACE" nil "CARDINAL" 0 nil t)
|
||
0)))
|
||
"0"))))))
|
||
|
||
(defun wallpaper-default-set-function (file)
|
||
"Set the wallpaper to FILE using a command.
|
||
This is the default function for `wallpaper-set-function'."
|
||
(unless wallpaper-command
|
||
(error "Couldn't find a command to set the wallpaper with"))
|
||
(let* ((args (if (functionp wallpaper-command-args)
|
||
(funcall wallpaper-command-args)
|
||
wallpaper-command-args))
|
||
(real-args (mapcar (lambda (arg) (wallpaper--format-arg arg file))
|
||
args))
|
||
(bufname (format " *wallpaper-%s*" (random)))
|
||
(setter (and (wallpaper-setter-p wallpaper--current-setter)
|
||
(equal (wallpaper-setter-command wallpaper--current-setter)
|
||
wallpaper-command)
|
||
wallpaper--current-setter))
|
||
(init-action (and setter (wallpaper-setter-init-action setter)))
|
||
(detach (and setter (wallpaper-setter-detach setter)))
|
||
process)
|
||
(when init-action
|
||
(funcall init-action))
|
||
(wallpaper-debug "Using command: \"%s %s\""
|
||
wallpaper-command (string-join real-args " "))
|
||
(if detach
|
||
(apply #'call-process wallpaper-command nil 0 nil real-args)
|
||
(setq process
|
||
(apply #'start-process "set-wallpaper" bufname
|
||
wallpaper-command real-args))
|
||
(setf (process-sentinel process)
|
||
(lambda (process status)
|
||
(unwind-protect
|
||
(if (and (eq (process-status process) 'exit)
|
||
(zerop (process-exit-status process)))
|
||
(message "Desktop wallpaper changed to %s"
|
||
(abbreviate-file-name file))
|
||
(message "command \"%s %s\": %S"
|
||
(string-join (process-command process) " ")
|
||
(string-replace "\n" "" status)
|
||
(with-current-buffer (process-buffer process)
|
||
(string-clean-whitespace (buffer-string)))))
|
||
(ignore-errors
|
||
(kill-buffer (process-buffer process)))))))
|
||
process))
|
||
|
||
;;;###autoload
|
||
(defun wallpaper-set (file)
|
||
"Set the desktop background to FILE in a graphical environment.
|
||
|
||
On GNU/Linux and other Unix-like systems, this relies on an
|
||
external command. Which command to use is automatically detected
|
||
in most cases, but can be manually customized with the user
|
||
options `wallpaper-command' and `wallpaper-command-args'.
|
||
|
||
On MS-Windows and Haiku systems, no external command is needed,
|
||
so the value of `wallpaper-commands' is ignored."
|
||
(interactive
|
||
(let ((default (wallpaper--get-default-file)))
|
||
(list (read-file-name (format-prompt "Set desktop background to" default)
|
||
default-directory default
|
||
t nil
|
||
(let ((re (wallpaper--image-file-regexp)))
|
||
(lambda (file-name)
|
||
(or (file-directory-p file-name)
|
||
(string-match re file-name))))))))
|
||
(when (file-directory-p file)
|
||
(error "Can't set wallpaper to a directory: %s" file))
|
||
(unless (file-exists-p file)
|
||
(error "No such file: %s" file))
|
||
(unless (file-readable-p file)
|
||
(error "File is not readable: %s" file))
|
||
(wallpaper-debug "Using image %S:" file)
|
||
(funcall wallpaper-set-function file))
|
||
|
||
(provide 'wallpaper)
|
||
|
||
;;; wallpaper.el ends here
|