-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathswm-grid.lisp
123 lines (103 loc) · 3.38 KB
/
swm-grid.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
;;;; swm-grid.lisp
(in-package #:swm-grid)
(export '(*width*
*height*
*default-group*
create-groups
left-group
right-group
down-group
up-group))
;;
;; Default to a 3x3 grid, opening at center.
;;
(defvar *width* 3)
(defvar *height* 3)
(defvar *default-group* 4)
(defun create-groups ()
"Create enough workspaces to fill the grid."
(let*
((groups (stumpwm::sort-groups (stumpwm:current-screen)))
(ngroups (length groups)))
(dotimes (i (- (* *width* *height*) ngroups))
(stumpwm:gnew (write-to-string i))))
(let
((groups (stumpwm::sort-groups (stumpwm:current-screen))))
(stumpwm::switch-to-group (nth *default-group* groups))))
(defun gnav-get-y (i)
"Get the row given a group index."
(floor (/ i *width*)))
(defun gnav-get-x (i)
"Get the column given a group index."
(mod i *width*))
(defun gnav-get-i (x y)
"Get the group index of a row, column pair."
(+ (* y *width*) x))
(defun gnav-step (i direction)
"From group index I, step in one direction in '(:right :left :down :up)"
(let*
((curx (gnav-get-x i))
(cury (gnav-get-y i))
(newx (case direction
(:left (max 0 (min (- *width* 1) (- curx 1))))
(:right (max 0 (min (- *width* 1) (+ curx 1))))
(otherwise curx)))
(newy (case direction
(:up (max 0 (min (- *height* 1) (- cury 1))))
(:down (max 0 (min (- *height* 1) (+ cury 1))))
(otherwise cury)))
(newi (gnav-get-i newx newy)))
newi))
(defun gnav-current-index ()
"Get the current group index."
(position (current-group) (stumpwm::sort-groups (current-screen))))
(defun navigate-groups (direction)
"Move groups as though they're in an MxN grid"
(let*
((groups (stumpwm::sort-groups (current-screen)))
(curi (position (current-group) groups))
(newi (gnav-step curi direction))
(new-group (nth newi groups)))
(progn
(if (not (eq newi curi))
(stumpwm::switch-to-group new-group))
(gnav-echo-graph))))
(defun nl-join (items)
"Join on new lines."
(format nil (concatenate 'string "~{~A~^" (string #\newline) "~}") items))
(defun join (delim items)
"Join strings on delimiter."
(format nil (concatenate 'string "~{~A~^" (string delim) "~}") items))
(defun gnav-get-fmt-row (y)
(let*
((curi (gnav-current-index))
(firsti (gnav-get-i 0 y))
(lasti (gnav-get-i *width* y)))
(join " " (mapcar
(lambda (i)
(if (eq i curi)
(concatenate 'string "^R" (write-to-string i) "^r")
i))
(range lasti :min firsti)))))
(defun gnav-echo-graph ()
"Echo an ascii workspace indicator."
(let*
((rows (mapcar 'gnav-get-fmt-row (range *height*)))
(img (nl-join rows)))
(echo img)))
(defun range (max &key (min 0) (step 1))
"https://stackoverflow.com/questions/13937520/pythons-range-analog-in-common-lisp"
(loop for n from min below max by step
collect n))
(defcommand left-group ()
() ""
(navigate-groups :left))
(defcommand right-group ()
() ""
(navigate-groups :right))
(defcommand down-group ()
() ""
(navigate-groups :down))
(defcommand up-group ()
() ""
(navigate-groups :up))