wrap when moving things to outputs with direction
Patch status: rejected
Patch by Francesco Mazzoli
To apply this patch, use:
curl http://cr.i3wm.org/patch/6/raw.patch | git am
b/include/randr.h
15 |
@@ -88,19 +88,28 @@ Output *get_output_by_name(const char *name); |
16 |
Output *get_output_containing(int x, int y); |
17 |
|
18 |
/** |
19 |
- * Gets the output which is the last one in the given direction, for example |
20 |
- * the output on the most bottom when direction == D_DOWN, the output most |
21 |
- * right when direction == D_RIGHT and so on. |
22 |
+ * Gets the output which is the next one in the given direction. |
23 |
+ * |
24 |
+ * If close_far == CLOSEST_OUTPUT, then the output next to the current one will |
25 |
+ * selected. If close_far == FARTHEST_OUTPUT, the output which is the last one |
26 |
+ * in the given direction will be selected. |
27 |
* |
28 |
- * This function always returns a output. |
29 |
+ * NULL will be returned when no non-active outputs are present in the direction |
30 |
+ * specified (note that current counts as such an output). |
31 |
* |
32 |
*/ |
33 |
-Output *get_output_most(direction_t direction, Output *current); |
34 |
+Output *get_output_next(direction_t direction, Output *current, output_close_far_t close_far); |
35 |
|
36 |
/** |
37 |
- * Gets the output which is the next one in the given direction. |
38 |
+ * Like get_output_next with close_far == CLOSEST_OUTPUT, but wraps. |
39 |
+ * |
40 |
+ * For example if get_output_next(D_DOWN, x, FARTHEST_OUTPUT) = NULL, then |
41 |
+ * get_output_next_wrap(D_DOWN, x) will return the topmost output. |
42 |
+ * |
43 |
+ * This function always returns a output: if no active outputs can be found, |
44 |
+ * current itself is returned. |
45 |
* |
46 |
*/ |
47 |
-Output *get_output_next(direction_t direction, Output *current, output_close_far_t close_far); |
48 |
+Output *get_output_next_wrap(direction_t direction, Output *current); |
49 |
|
50 |
#endif |
b/src/commands.c
55 |
@@ -55,23 +55,15 @@ static bool definitelyGreaterThan(float a, float b, float epsilon) { |
56 |
static Output *get_output_from_string(Output *current_output, const char *output_str) { |
57 |
Output *output; |
58 |
|
59 |
- if (strcasecmp(output_str, "left") == 0) { |
60 |
- output = get_output_next(D_LEFT, current_output, CLOSEST_OUTPUT); |
61 |
- if (!output) |
62 |
- output = get_output_most(D_RIGHT, current_output); |
63 |
- } else if (strcasecmp(output_str, "right") == 0) { |
64 |
- output = get_output_next(D_RIGHT, current_output, CLOSEST_OUTPUT); |
65 |
- if (!output) |
66 |
- output = get_output_most(D_LEFT, current_output); |
67 |
- } else if (strcasecmp(output_str, "up") == 0) { |
68 |
- output = get_output_next(D_UP, current_output, CLOSEST_OUTPUT); |
69 |
- if (!output) |
70 |
- output = get_output_most(D_DOWN, current_output); |
71 |
- } else if (strcasecmp(output_str, "down") == 0) { |
72 |
- output = get_output_next(D_DOWN, current_output, CLOSEST_OUTPUT); |
73 |
- if (!output) |
74 |
- output = get_output_most(D_UP, current_output); |
75 |
- } else output = get_output_by_name(output_str); |
76 |
+ if (strcasecmp(output_str, "left") == 0) |
77 |
+ output = get_output_next_wrap(D_LEFT, current_output); |
78 |
+ else if (strcasecmp(output_str, "right") == 0) |
79 |
+ output = get_output_next_wrap(D_RIGHT, current_output); |
80 |
+ else if (strcasecmp(output_str, "up") == 0) |
81 |
+ output = get_output_next_wrap(D_UP, current_output); |
82 |
+ else if (strcasecmp(output_str, "down") == 0) |
83 |
+ output = get_output_next_wrap(D_DOWN, current_output); |
84 |
+ else output = get_output_by_name(output_str); |
85 |
|
86 |
return output; |
87 |
} |
88 |
@@ -1052,13 +1044,13 @@ void cmd_move_con_to_output(I3_CMD, char *name) { |
89 |
|
90 |
// TODO: clean this up with commands.spec as soon as we switched away from the lex/yacc command parser |
91 |
if (strcasecmp(name, "up") == 0) |
92 |
- output = get_output_next(D_UP, current_output, CLOSEST_OUTPUT); |
93 |
+ output = get_output_next_wrap(D_UP, current_output); |
94 |
else if (strcasecmp(name, "down") == 0) |
95 |
- output = get_output_next(D_DOWN, current_output, CLOSEST_OUTPUT); |
96 |
+ output = get_output_next_wrap(D_DOWN, current_output); |
97 |
else if (strcasecmp(name, "left") == 0) |
98 |
- output = get_output_next(D_LEFT, current_output, CLOSEST_OUTPUT); |
99 |
+ output = get_output_next_wrap(D_LEFT, current_output); |
100 |
else if (strcasecmp(name, "right") == 0) |
101 |
- output = get_output_next(D_RIGHT, current_output, CLOSEST_OUTPUT); |
102 |
+ output = get_output_next_wrap(D_RIGHT, current_output); |
103 |
else |
104 |
output = get_output_by_name(name); |
105 |
|
b/src/randr.c
110 |
@@ -93,15 +93,30 @@ Output *get_output_containing(int x, int y) { |
111 |
} |
112 |
|
113 |
/* |
114 |
- * Gets the output which is the last one in the given direction, for example |
115 |
- * the output on the most bottom when direction == D_DOWN, the output most |
116 |
- * right when direction == D_RIGHT and so on. |
117 |
+ * Like get_output_next with close_far == CLOSEST_OUTPUT, but wraps. |
118 |
* |
119 |
- * This function always returns a output. |
120 |
+ * For example if get_output_next(D_DOWN, x, FARTHEST_OUTPUT) = NULL, then |
121 |
+ * get_output_next_wrap(D_DOWN, x) will return the topmost output. |
122 |
+ * |
123 |
+ * This function always returns a output: if no active outputs can be found, |
124 |
+ * current itself is returned. |
125 |
* |
126 |
*/ |
127 |
-Output *get_output_most(direction_t direction, Output *current) { |
128 |
- Output *best = get_output_next(direction, current, FARTHEST_OUTPUT); |
129 |
+Output *get_output_next_wrap(direction_t direction, Output *current) { |
130 |
+ Output *best = get_output_next(direction, current, CLOSEST_OUTPUT); |
131 |
+ /* If no output can be found, wrap */ |
132 |
+ if (!best) { |
133 |
+ direction_t opposite; |
134 |
+ if (direction == D_RIGHT) |
135 |
+ opposite = D_LEFT; |
136 |
+ else if (direction == D_LEFT) |
137 |
+ opposite = D_RIGHT; |
138 |
+ else if (direction == D_DOWN) |
139 |
+ opposite = D_UP; |
140 |
+ else |
141 |
+ opposite = D_DOWN; |
142 |
+ best = get_output_next(opposite, current, FARTHEST_OUTPUT); |
143 |
+ } |
144 |
if (!best) |
145 |
best = current; |
146 |
DLOG("current = %s, best = %s\n", current->name, best->name); |
147 |
@@ -111,6 +126,13 @@ Output *get_output_most(direction_t direction, Output *current) { |
148 |
/* |
149 |
* Gets the output which is the next one in the given direction. |
150 |
* |
151 |
+ * If close_far == CLOSEST_OUTPUT, then the output next to the current one will |
152 |
+ * selected. If close_far == FARTHEST_OUTPUT, the output which is the last one |
153 |
+ * in the given direction will be selected. |
154 |
+ * |
155 |
+ * NULL will be returned when no non-active outputs are present in the direction |
156 |
+ * specified (note that current counts as such an output). |
157 |
+ * |
158 |
*/ |
159 |
Output *get_output_next(direction_t direction, Output *current, output_close_far_t close_far) { |
160 |
Rect *cur = &(current->rect), |