2024-06-15 09:57:11 +02:00
|
|
|
(define-module (modules general)
|
|
|
|
#:use-module (sjson parser)
|
|
|
|
#:use-module (swayipc dispatcher)
|
2024-06-22 18:49:34 +02:00
|
|
|
#:use-module (swayipc events)
|
|
|
|
#:use-module (swayipc records)
|
2024-06-15 09:57:11 +02:00
|
|
|
#:use-module (srfi srfi-18)
|
|
|
|
#:use-module (ice-9 hash-table)
|
2024-06-22 18:49:34 +02:00
|
|
|
#:use-module (ice-9 string-fun)
|
2024-06-18 18:05:39 +02:00
|
|
|
#:export (general-keybinding-translator
|
|
|
|
general-configure
|
2024-06-22 18:49:34 +02:00
|
|
|
general-init
|
2024-06-18 18:05:39 +02:00
|
|
|
general-define-keys
|
|
|
|
general-define-key
|
|
|
|
general-keybindings
|
2024-06-22 18:49:34 +02:00
|
|
|
general-submaps
|
2024-06-22 23:10:51 +02:00
|
|
|
general-command-received-hook))
|
2024-06-22 18:49:34 +02:00
|
|
|
|
|
|
|
(define general-command-prefix "echo /general ")
|
|
|
|
(define general-command-signature
|
|
|
|
(string-append "exec " general-command-prefix))
|
2024-06-15 09:57:11 +02:00
|
|
|
|
|
|
|
;; Local copy of keybindings configured by general, it's recommended to
|
|
|
|
;; only use general to assign keybindings, otherwise this hashtable won't
|
|
|
|
;; by synced with actual keybindings available in sway
|
2024-06-18 18:05:39 +02:00
|
|
|
(define general-keybindings (make-hash-table))
|
2024-06-15 09:57:11 +02:00
|
|
|
|
|
|
|
;; Local copy of submaps configured by general, it's recommended to
|
|
|
|
;; only use general to assign submaps, otherwise this hashtable won't
|
2024-06-18 18:05:39 +02:00
|
|
|
;; by synced with actual submaps available in sway
|
|
|
|
(define general-submaps (make-hash-table))
|
2024-06-15 09:57:11 +02:00
|
|
|
|
2024-06-18 18:05:39 +02:00
|
|
|
;; add default submap, this is the default submap in sway
|
|
|
|
(hash-set! general-submaps "" "default")
|
|
|
|
|
2024-06-22 18:49:34 +02:00
|
|
|
;; data received: emitted on new command received via bindings.
|
|
|
|
;; Parameters:
|
|
|
|
;; - arg1: commandd.
|
2024-06-22 23:10:51 +02:00
|
|
|
(define general-command-received-hook
|
2024-06-22 18:49:34 +02:00
|
|
|
(make-hook 1))
|
|
|
|
|
2024-06-18 18:05:39 +02:00
|
|
|
(define (general-keybinding-translator key)
|
|
|
|
"Translate a given key, passing a function can enable easier keybindings
|
|
|
|
like emacs key chords (refer to module modules/kbd.scm). The default implementation
|
|
|
|
doesn't modify passed keybindings"
|
2024-06-15 09:57:11 +02:00
|
|
|
key)
|
|
|
|
|
2024-06-22 18:49:34 +02:00
|
|
|
(define* (general-configure #:key keybinding-translator)
|
|
|
|
"Configure keybinding-translator (refer to general-keybinding-translator)
|
2024-06-18 18:05:39 +02:00
|
|
|
Parameters:
|
2024-06-22 18:49:34 +02:00
|
|
|
- keybinding-translator: a function that takes a key and returns the translated version."
|
2024-06-18 18:05:39 +02:00
|
|
|
(when keybinding-translator
|
2024-06-22 18:49:34 +02:00
|
|
|
(set! general-keybinding-translator keybinding-translator)))
|
|
|
|
|
|
|
|
(define (binding-changed binding-event)
|
|
|
|
(let* ((command (sway-binding-event-binding-command
|
|
|
|
(sway-binding-event-binding binding-event)))
|
|
|
|
(prefix (if (> (string-length command)
|
|
|
|
(string-length general-command-signature))
|
|
|
|
(substring command 0 (string-length general-command-signature))
|
|
|
|
""))
|
|
|
|
(general-command (equal? prefix general-command-signature)))
|
|
|
|
(when general-command
|
2024-06-22 23:10:51 +02:00
|
|
|
(run-hook general-command-received-hook
|
2024-06-22 18:49:34 +02:00
|
|
|
(hex->string
|
|
|
|
(substring command (string-length general-command-signature)))))))
|
|
|
|
|
|
|
|
(define (general-init)
|
|
|
|
;; add sway bindings event hook
|
|
|
|
(add-hook! sway-binding-hook binding-changed)
|
|
|
|
|
|
|
|
;; add a hook to listen to received commands
|
2024-06-22 23:10:51 +02:00
|
|
|
(add-hook! general-command-received-hook
|
2024-06-22 18:49:34 +02:00
|
|
|
(lambda (command)
|
|
|
|
(format #t "executing command ~a\n" command)
|
|
|
|
(with-exception-handler
|
|
|
|
(lambda (exc)
|
|
|
|
(custom-exception-handler exc command))
|
|
|
|
(lambda () (eval-string command))
|
|
|
|
#:unwind? #t))))
|
2024-06-15 18:10:21 +02:00
|
|
|
|
2024-06-15 09:57:11 +02:00
|
|
|
(define (exp->string exp)
|
2024-06-18 18:05:39 +02:00
|
|
|
"Convert a given expression exp to a string."
|
2024-06-15 09:57:11 +02:00
|
|
|
(call-with-output-string (lambda (p)
|
|
|
|
(write exp p))))
|
|
|
|
|
2024-06-18 18:05:39 +02:00
|
|
|
(define* (define-submap chord wk submap parent-submap)
|
|
|
|
"Define a submap, this will be translated to sway by defining a keybinding
|
|
|
|
to switch to the provided submap, and add a default ESC keybinding to return
|
|
|
|
to the default submap.
|
|
|
|
Parameters:
|
|
|
|
- chord: chord string to trigger the keybinding. e.g. (s-SPC f f).
|
|
|
|
- wk: which-key's description.
|
|
|
|
- submap: the name of the submap.
|
|
|
|
- parent-submap: parent submap, default if submap is a child of the default submap."
|
|
|
|
(let* ((chord-ls (map general-keybinding-translator
|
|
|
|
(string-split chord #\Space)))
|
|
|
|
(key (car (last-pair chord-ls))))
|
|
|
|
(format #t "define submap ~a\n" chord)
|
|
|
|
(hash-set! general-submaps chord submap)
|
|
|
|
(define-keybindings chord
|
2024-06-22 18:49:34 +02:00
|
|
|
(list `sway-mode submap)
|
2024-06-18 18:05:39 +02:00
|
|
|
wk parent-submap)
|
|
|
|
(define-keybindings (string-append chord " Esc")
|
|
|
|
`(sway-mode "default")
|
|
|
|
"Escape" submap)))
|
|
|
|
|
|
|
|
(define (last-key chord)
|
|
|
|
"Return last key from a given chord."
|
|
|
|
(car (reverse (string-split chord #\+))))
|
|
|
|
|
|
|
|
(define (general-command exp-str)
|
|
|
|
"Execute a general command (scheme expression)"
|
2024-06-22 18:49:34 +02:00
|
|
|
(string-append general-command-signature
|
|
|
|
(string->hex exp-str)))
|
2024-06-15 09:57:11 +02:00
|
|
|
|
2024-06-18 18:05:39 +02:00
|
|
|
(define* (define-keybindings chord exp wk submap)
|
|
|
|
"Define a sway keybinding.
|
|
|
|
Parameters:
|
|
|
|
- chord: chord string to trigger the keybinding. e.g. (s-SPC f f).
|
|
|
|
- exp: expression to execute when the chord is triggered.
|
|
|
|
- wk: which-key's description.
|
|
|
|
- submap: the name of the submap."
|
2024-06-22 18:49:34 +02:00
|
|
|
(format #t "define-keybindings ~a with expression `~a`\n" chord exp)
|
2024-06-18 18:05:39 +02:00
|
|
|
(let* ((chord-ls (map general-keybinding-translator
|
|
|
|
(string-split chord #\Space)))
|
|
|
|
(key (car (last-pair chord-ls)))
|
|
|
|
(type (if (string->number (last-key key)) "bindcode" "bindsym"))
|
|
|
|
(command (string-append type " " key " " (general-command (exp->string exp))))
|
|
|
|
(esc (string-append type " " key " " (general-command (exp->string `(and ,exp (sway-mode "default")))))))
|
|
|
|
|
2024-06-22 23:10:51 +02:00
|
|
|
(hash-set! general-keybindings chord (list key exp wk submap))
|
2024-06-15 09:57:11 +02:00
|
|
|
(if (equal? submap "default")
|
|
|
|
(dispatch-command command)
|
|
|
|
(begin
|
|
|
|
(sway-mode-subcommand submap command)
|
|
|
|
(unless (equal? "sway-mode" (symbol->string (car exp)))
|
|
|
|
(sway-mode-subcommand submap esc))))))
|
|
|
|
|
|
|
|
(define (find-submap chord)
|
2024-06-18 18:05:39 +02:00
|
|
|
"Return the submap for the provided chord.
|
|
|
|
If the submap isn't found, #f is returned."
|
2024-06-15 09:57:11 +02:00
|
|
|
(or
|
2024-06-18 18:05:39 +02:00
|
|
|
(hash-get-handle general-submaps
|
2024-06-15 09:57:11 +02:00
|
|
|
(string-join
|
|
|
|
(list-head chord (- (length chord) 1)) " "))
|
2024-06-18 18:05:39 +02:00
|
|
|
#f))
|
|
|
|
|
|
|
|
(define* (general-define-key chord #:optional exp #:key wk submap)
|
|
|
|
"Assign a given chord to a given expression.
|
|
|
|
Parameters:
|
|
|
|
- chord: chord string to trigger the keybinding. e.g. (s-SPC f f).
|
|
|
|
- exp: expression to execute when the chord is triggered.
|
|
|
|
- wk: which-key's description.
|
|
|
|
- submap: the name of the submap."
|
2024-06-15 09:57:11 +02:00
|
|
|
(let* ((chord-ls
|
2024-06-18 18:05:39 +02:00
|
|
|
(map general-keybinding-translator
|
2024-06-15 09:57:11 +02:00
|
|
|
(string-split chord #\Space)))
|
|
|
|
(key (car (last-pair chord-ls)))
|
2024-06-18 18:05:39 +02:00
|
|
|
(chord (string-join chord-ls " "))
|
2024-06-15 09:57:11 +02:00
|
|
|
(key-code (string->number key))
|
2024-06-18 18:05:39 +02:00
|
|
|
(description (or wk (or submap (symbol->string (car exp)))))
|
|
|
|
(tsubmap (cdr (find-submap chord-ls))))
|
|
|
|
|
|
|
|
;; if target submap isn't found, throw an error
|
|
|
|
;; the submap must always be defined before defining a keybinding
|
|
|
|
(unless tsubmap
|
|
|
|
(error 'find-submap "chord was not found in submaps, a submap has to be defined for it" chord))
|
|
|
|
|
2024-06-15 09:57:11 +02:00
|
|
|
(if submap
|
2024-06-18 18:05:39 +02:00
|
|
|
;; if submap key is provided, then define a submap (ignore exp)
|
2024-06-22 23:10:51 +02:00
|
|
|
(define-submap chord submap submap
|
2024-06-15 09:57:11 +02:00
|
|
|
(cdr (find-submap chord-ls)))
|
2024-06-18 18:05:39 +02:00
|
|
|
;; otherwise, define a keybinding with exp
|
|
|
|
(define-keybindings chord exp wk
|
2024-06-15 09:57:11 +02:00
|
|
|
(cdr (find-submap chord-ls))))))
|
|
|
|
|
2024-06-18 18:05:39 +02:00
|
|
|
(define* (general-define-keys #:key parent-prefix prefix wk . args )
|
|
|
|
"Assign given list of chords and expressions to a submap.
|
|
|
|
Parameters:
|
|
|
|
- prefix: a prefix that's applied to all the keybindings passed. e.g. (s-Spc).
|
|
|
|
- parent-prefix: a prefix of the parent (passed internally, you should probably use prefix instead).
|
|
|
|
- wk: which-key's description of the submap.
|
|
|
|
- args: list of keybindings to assign to the prefix, these will be passed to general-define-key.
|
|
|
|
|
|
|
|
For example:
|
|
|
|
(general-define-keys
|
|
|
|
#:prefix \"s-Space\" #:wk \"Leader\"
|
|
|
|
`(\"o\" (exec \"rofi -show drun\"))
|
|
|
|
`(\"C-g\" (sway-mode \"default\") #:wk \"abort\")
|
|
|
|
|
|
|
|
`(general-define-keys
|
|
|
|
#:prefix \"r\" #:wk \"Rofi\"
|
|
|
|
(\"p\" (exec \"~/.config/rofi/bin/password-manager\"))))"
|
|
|
|
|
2024-06-15 09:57:11 +02:00
|
|
|
(when parent-prefix
|
|
|
|
(set! prefix (string-append parent-prefix " " prefix)))
|
|
|
|
|
|
|
|
(when prefix
|
2024-06-18 18:05:39 +02:00
|
|
|
(format #t "define prefix submap: ~a, wk ~a\n" prefix wk)
|
|
|
|
(general-define-key prefix #:submap wk))
|
2024-06-15 09:57:11 +02:00
|
|
|
|
|
|
|
(map (lambda (arg)
|
|
|
|
(when (list? arg)
|
|
|
|
(if (and
|
|
|
|
(symbol? (car arg))
|
2024-06-18 18:05:39 +02:00
|
|
|
(equal? "general-define-keys" (symbol->string (car arg))))
|
|
|
|
(apply general-define-keys (append `(#:parent-prefix ,prefix) (cdr arg)))
|
|
|
|
(apply general-define-key
|
2024-06-15 09:57:11 +02:00
|
|
|
(cons (string-append
|
|
|
|
(if prefix (string-append prefix " ") "")
|
|
|
|
(car arg)) (cdr arg))))))
|
|
|
|
args))
|
|
|
|
|
|
|
|
(define (custom-exception-handler exc command-id payload)
|
2024-06-22 18:49:34 +02:00
|
|
|
"Exception handler for evaluating expressions."
|
2024-06-22 13:03:48 +02:00
|
|
|
(format #t "An error occurd while executing the received
|
|
|
|
general command: command: ~a, payload: ~a\n" command-id payload)
|
|
|
|
(format #t "exception: ~a\n" exc))
|
2024-06-15 09:57:11 +02:00
|
|
|
|
2024-06-22 18:49:34 +02:00
|
|
|
;; FIXME: there must be some guile built-in function to
|
|
|
|
;; base64 encode or convert to hex
|
|
|
|
(define (char->hex char)
|
|
|
|
"Convert a character to hex."
|
|
|
|
(let ((hex (number->string (char->integer char) 16)))
|
|
|
|
(if (< (string-length hex) 2)
|
|
|
|
(string-append "0" hex)
|
|
|
|
hex)))
|
|
|
|
|
|
|
|
(define (hex->char hex-pair)
|
|
|
|
"Convert a hex to character."
|
|
|
|
(integer->char (string->number hex-pair 16)))
|
|
|
|
|
|
|
|
(define (string->hex str)
|
|
|
|
"Convert a string to hex."
|
|
|
|
(let loop ((chars (string->list str))
|
|
|
|
(result '()))
|
|
|
|
(if (null? chars)
|
|
|
|
(string-concatenate (reverse result))
|
|
|
|
(loop (cdr chars) (cons (char->hex (car chars)) result)))))
|
|
|
|
|
|
|
|
(define (hex->string hex-str)
|
|
|
|
"Convert a hex to string."
|
|
|
|
(let loop ((chars (string->list hex-str))
|
|
|
|
(result '()))
|
|
|
|
(if (null? chars)
|
|
|
|
(list->string (reverse result))
|
|
|
|
(let ((char1 (car chars))
|
|
|
|
(char2 (cadr chars)))
|
|
|
|
(loop (cddr chars)
|
|
|
|
(cons (hex->char (string char1 char2)) result))))))
|