89bd07ee72
If the user puts in a string that doesn't match any of the selectables, do one of the following: 1. When called inside of Emacs, create a new buffer with that name. 2. When called outside of Emacs, do nothing. Change-Id: I92ba985b5cd7805d37d5d0e0631b20fdce7ce479 Reviewed-on: https://cl.tvl.fyi/c/depot/+/12542 Autosubmit: tazjin <tazjin@tvl.su> Tested-by: BuildkiteCI Reviewed-by: tazjin <tazjin@tvl.su>
181 lines
7.5 KiB
EmacsLisp
181 lines
7.5 KiB
EmacsLisp
;;; niri.el --- seamless niri/emacs integration. -*- lexical-binding: t; -*-
|
|
;;
|
|
;; Copyright (C) 2024 The TVL Contributors
|
|
;;
|
|
;; Author: Vincent Ambo <tazjin@tvl.su>
|
|
;; Version: 1.0
|
|
;; Package-Requires: ((emacs "27.1"))
|
|
;;
|
|
;;; Commentary:
|
|
;;
|
|
;; After having used EXWM for many years (7 or so?) it's become second nature
|
|
;; that there is no difference between windows and Emacs buffers. This means
|
|
;; that from any Emacs buffer (or, in the case of EXWM, from any X window) it's
|
|
;; possible to switch to any of the others.
|
|
;;
|
|
;; This implements similar logic for Emacs running in Niri, consisting of two
|
|
;; sides of the integration:
|
|
;;
|
|
;; # In Emacs
|
|
;;
|
|
;; Inside of Emacs, when switching buffers, populate the buffer-switching menu
|
|
;; additionally with all open Niri windows. Selecting a Niri window moves the
|
|
;; screen to that window.
|
|
;;
|
|
;; # Outside of Emacs
|
|
;;
|
|
;; Provides an interface for the same core functionality that can be used from
|
|
;; shell scripts, and bound to selectors like dmenu or rofi.
|
|
;;
|
|
;; # Switching to Emacs buffers
|
|
;;
|
|
;; Some special logic exists for handling the case of switching to an Emacs
|
|
;; buffer. There are several conditions that we can be in, that each have a
|
|
;; predictable result:
|
|
;;
|
|
;; In a non-Emacs window, selecting an Emacs buffer will either switch to an
|
|
;; Emacs frame already displaying this buffer, or launch a new frame for it.
|
|
;;
|
|
;; Inside of Emacs, if *another* frame is already displaying the buffer, switch
|
|
;; to it. Otherwise the behaviour is the same as standard buffer switching.
|
|
|
|
(require 'seq)
|
|
(require 'map)
|
|
|
|
(defun niri-list-windows ()
|
|
"List all currently open Niri windows."
|
|
(json-parse-string
|
|
(shell-command-to-string "niri msg -j windows")
|
|
:false-object nil))
|
|
|
|
(defun niri--window-is-emacs (window)
|
|
(equal (map-elt window "app_id") "emacs"))
|
|
|
|
(defun niri--list-selectables ()
|
|
"Lists all currently selectable things in a format that can work
|
|
with completing-read. Selectable means all open Niri
|
|
windows (except Emacs windows) and all Emacs buffers.
|
|
|
|
Emacs windows are returned separately, as they are required for
|
|
frame navigation."
|
|
(let* (;; all niri windows, with emacs/non-emacs windows split up
|
|
(all-windows (niri-list-windows))
|
|
(windows (seq-filter (lambda (w) (not (niri--window-is-emacs w)))
|
|
all-windows))
|
|
(emacs-windows (seq-filter #'niri--window-is-emacs all-windows))
|
|
|
|
;; all non-hidden buffers
|
|
(buffers (seq-filter (lambda (b) (not (string-prefix-p " " (buffer-name b))))
|
|
(buffer-list)))
|
|
(selectables (make-hash-table :test 'equal :size (+ (length windows)
|
|
(length buffers)))))
|
|
(seq-do (lambda (window)
|
|
(map-put! selectables (map-elt window "title")
|
|
(cons :niri window)))
|
|
windows)
|
|
|
|
(seq-do (lambda (buf)
|
|
(map-put! selectables (buffer-name buf)
|
|
(cons :emacs buf)))
|
|
buffers)
|
|
(cons selectables emacs-windows)))
|
|
|
|
(defun niri--focus-window (window)
|
|
(shell-command (format "niri msg action focus-window --id %d"
|
|
(map-elt window "id"))))
|
|
|
|
(defun niri--target-action-internal (target)
|
|
"Focus the given TARGET (a Niri window or Emacs buffer). This is
|
|
used when called from inside of Emacs. It will NOT correctly
|
|
switch Niri windows when called from outside of Emacs."
|
|
(pcase (car target)
|
|
(:emacs (pop-to-buffer (cdr target) '((display-buffer-reuse-window
|
|
display-buffer-same-window)
|
|
(reusable-frames . 0))))
|
|
(:niri (niri--focus-window (cdr target)))))
|
|
|
|
(defun niri-go-anywhere ()
|
|
"Interactively select and switch to an open Niri window, or an
|
|
Emacs buffer."
|
|
(interactive)
|
|
(let* ((selectables (car (niri--list-selectables)))
|
|
;; Annotate buffers that display remote files. I frequently
|
|
;; want to see it, because I might have identically named
|
|
;; files open locally and remotely at the same time, and it
|
|
;; helps with differentiating them.
|
|
(completion-extra-properties
|
|
'(:annotation-function
|
|
(lambda (name)
|
|
(let ((elt (map-elt selectables name)))
|
|
(pcase (car elt)
|
|
(:emacs
|
|
(if-let* ((file (buffer-file-name (cdr elt)))
|
|
(remote (file-remote-p file)))
|
|
(format " [%s]" remote)))
|
|
(:niri (format " [%s]" (map-elt (cdr elt) "app_id"))))))))
|
|
|
|
(target-key (completing-read "Switch to: " (map-keys selectables)))
|
|
(target (map-elt selectables target-key)))
|
|
(if target
|
|
(niri--target-action-internal target)
|
|
(switch-to-buffer target-key nil t))))
|
|
|
|
|
|
(defun niri--target-action-external (target frames)
|
|
"Focus the given TARGET (a Niri window or Emacs buffer). This
|
|
always behaves correctly, but does more work than the -internal
|
|
variant. It should only be called when invoking the switcher from
|
|
outside of Emacs (i.e. through `emacsclient').
|
|
|
|
FRAMES is the exact list of Emacs frames that existed at the time
|
|
the switcher was invoked."
|
|
(pcase (car target)
|
|
(:niri (niri--focus-window (cdr target)))
|
|
|
|
;; When switching to an Emacs buffer from outside of Emacs, we run into the
|
|
;; additional complication that Wayland does not allow arbitrary
|
|
;; applications to change the focused window. Calling e.g.
|
|
;; `select-frame-set-input-focus' has no effect on Wayland when not called
|
|
;; from within a focused Emacs frame.
|
|
;;
|
|
;; However, due to concurrency, frames may change between the moment when we
|
|
;; start the switcher (and potentially wait for user input), and when the
|
|
;; final selection happens.
|
|
;;
|
|
;; To get around this we try to match the target Emacs frame (if present) to
|
|
;; a Niri window, switch to it optimistically, and *then* execute the final
|
|
;; buffer switching command.
|
|
(:emacs
|
|
(if-let ((window (get-buffer-window (cdr target) t))
|
|
(frame (window-frame window))
|
|
(frame-name (frame-parameter frame 'name))
|
|
(niri-window (seq-find (lambda (w)
|
|
(equal (map-elt w "title") frame-name))
|
|
frames)))
|
|
;; Target frame found and could be matched to a Niri window: Go there!
|
|
(progn (select-window window) ;; ensure the right window in the frame has focus
|
|
(niri--focus-window niri-window)
|
|
(message "Switched to existing window for \"%s\"" (buffer-name (cdr target))))
|
|
|
|
;; Target frame not found; is Emacs the focused program?
|
|
(if (seq-find (lambda (w) (map-elt w "is_focused")) frames)
|
|
(switch-to-buffer (cdr target))
|
|
;; if not, just make a new frame
|
|
(display-buffer (cdr target) '(display-buffer-pop-up-frame)))))))
|
|
|
|
(defun niri-go-anywhere-external ()
|
|
"Use a dmenu-compatible launcher like `fuzzel' to achieve the same
|
|
effect as `niri-go-anywhere', but from outside of Emacs through
|
|
Emacsclient."
|
|
(interactive) ;; TODO no?
|
|
(let* ((all (niri--list-selectables))
|
|
(selectables (car all))
|
|
(target (with-temp-buffer
|
|
(dolist (key (map-keys selectables))
|
|
(insert key "\n"))
|
|
(call-process-region nil nil "fuzzel" t t nil "-d")
|
|
(string-trim (buffer-string)))))
|
|
(when-let ((selectable (map-elt selectables target)))
|
|
(niri--target-action-external selectable (cdr all)))))
|
|
|
|
(provide 'niri)
|