aboutsummaryrefslogtreecommitdiff
path: root/src/MinOverlapPlacement.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/MinOverlapPlacement.cc')
-rw-r--r--src/MinOverlapPlacement.cc176
1 files changed, 105 insertions, 71 deletions
diff --git a/src/MinOverlapPlacement.cc b/src/MinOverlapPlacement.cc
index 559ff11..78cadb4 100644
--- a/src/MinOverlapPlacement.cc
+++ b/src/MinOverlapPlacement.cc
@@ -25,17 +25,75 @@
25#include "Window.hh" 25#include "Window.hh"
26#include "Screen.hh" 26#include "Screen.hh"
27 27
28ScreenPlacement::PlacementPolicy MinOverlapPlacement::s_policy = ScreenPlacement::ROWMINOVERLAPPLACEMENT; 28namespace {
29ScreenPlacement::RowDirection MinOverlapPlacement::s_row_dir = ScreenPlacement::LEFTRIGHT;
30ScreenPlacement::ColumnDirection MinOverlapPlacement::s_col_dir = ScreenPlacement::TOPBOTTOM;
31 29
32MinOverlapPlacement::MinOverlapPlacement(ScreenPlacement::PlacementPolicy policy) { 30inline void getWindowDimensions(const FluxboxWindow& win, int& left, int& top, int& right, int& bottom) {
33 s_policy = policy; 31
32 const int bw = 2 * win.frame().window().borderWidth();
33 left = win.x() - win.xOffset();
34 top = win.y() - win.yOffset();
35 right = left + win.width() + bw + win.widthOffset();
36 bottom = top + win.height() + bw + win.heightOffset();
34} 37}
35 38
39class Area {
40public:
41
42 enum Corner {
43 TOPLEFT,
44 TOPRIGHT,
45 BOTTOMLEFT,
46 BOTTOMRIGHT
47 } corner; // indicates the corner of the window that will be placed
48
49 Area(Corner _corner, int _x, int _y):
50 corner(_corner), x(_x), y(_y) { };
51
52 // do all STL set implementations use this for sorting?
53 bool operator <(const Area &o) const {
54 switch (s_policy) {
55 case ScreenPlacement::ROWMINOVERLAPPLACEMENT:
56 // if we're making rows, y-value is most important
57 if (y != o.y)
58 return ((y < o.y) ^ (s_col_dir == ScreenPlacement::BOTTOMTOP));
59 if (x != o.x)
60 return ((x < o.x) ^ (s_row_dir == ScreenPlacement::RIGHTLEFT));
61 return (corner < o.corner);
62 case ScreenPlacement::COLMINOVERLAPPLACEMENT:
63 // if we're making columns, x-value is most important
64 if (x != o.x)
65 return ((x < o.x) ^ (s_row_dir == ScreenPlacement::RIGHTLEFT));
66 if (y != o.y)
67 return ((y < o.y) ^ (s_col_dir == ScreenPlacement::BOTTOMTOP));
68 return (corner < o.corner);
69 default:
70 return false;
71 }
72 }
73
74 // position where the top left corner of the window will be placed
75 int x, y;
76
77 static ScreenPlacement::RowDirection s_row_dir;
78 static ScreenPlacement::ColumnDirection s_col_dir;
79 static ScreenPlacement::PlacementPolicy s_policy;
80};
81
82ScreenPlacement::RowDirection Area::s_row_dir = ScreenPlacement::LEFTRIGHT;
83ScreenPlacement::ColumnDirection Area::s_col_dir = ScreenPlacement::TOPBOTTOM;
84ScreenPlacement::PlacementPolicy Area::s_policy = ScreenPlacement::ROWMINOVERLAPPLACEMENT;
85
86} // end of anonymous namespace
87
88
36bool MinOverlapPlacement::placeWindow(const FluxboxWindow &win, int head, 89bool MinOverlapPlacement::placeWindow(const FluxboxWindow &win, int head,
37 int &place_x, int &place_y) { 90 int &place_x, int &place_y) {
38 91
92 int left;
93 int top;
94 int right;
95 int bottom;
96
39 std::list<FluxboxWindow *> windowlist; 97 std::list<FluxboxWindow *> windowlist;
40 const std::list<Focusable *> focusables = 98 const std::list<Focusable *> focusables =
41 win.screen().focusControl().focusedOrderWinList().clientList(); 99 win.screen().focusControl().focusedOrderWinList().clientList();
@@ -56,24 +114,26 @@ bool MinOverlapPlacement::placeWindow(const FluxboxWindow &win, int head,
56 int head_top = (signed) win.screen().maxTop(head); 114 int head_top = (signed) win.screen().maxTop(head);
57 int head_bot = (signed) win.screen().maxBottom(head); 115 int head_bot = (signed) win.screen().maxBottom(head);
58 116
59 const ScreenPlacement &screen_placement = win.screen().placementStrategy();
60 s_row_dir = screen_placement.rowDirection();
61 s_col_dir = screen_placement.colDirection();
62
63 int win_w = win.normalWidth() + win.fbWindow().borderWidth()*2 + 117 int win_w = win.normalWidth() + win.fbWindow().borderWidth()*2 +
64 win.widthOffset(); 118 win.widthOffset();
65 int win_h = win.normalHeight() + win.fbWindow().borderWidth()*2 + 119 int win_h = win.normalHeight() + win.fbWindow().borderWidth()*2 +
66 win.heightOffset(); 120 win.heightOffset();
67 121
68 // we keep a set of open spaces on the desktop, sorted by size/location 122 // we keep a set of open spaces on the desktop, sorted by size/location
69 std::set<Region> region_set; 123 std::set<Area> areas;
124
125 // setup stuff in order to make Area::operator< work
126 const ScreenPlacement& p = win.screen().placementStrategy();
127 Area::s_policy = p.placementPolicy();
128 Area::s_row_dir = p.rowDirection();
129 Area::s_col_dir = p.colDirection();
130
70 131
71 // initialize the set of regions to contain the entire head 132 // initialize the set of areas to contain the entire head
72 region_set.insert(Region(Region::TOPLEFT, head_left, head_top)); 133 areas.insert(Area(Area::TOPLEFT, head_left, head_top));
73 region_set.insert(Region(Region::TOPRIGHT, head_right - win_w, head_top)); 134 areas.insert(Area(Area::TOPRIGHT, head_right - win_w, head_top));
74 region_set.insert(Region(Region::BOTTOMLEFT, head_left, head_bot - win_h)); 135 areas.insert(Area(Area::BOTTOMLEFT, head_left, head_bot - win_h));
75 region_set.insert(Region(Region::BOTTOMRIGHT, head_right - win_w, 136 areas.insert(Area(Area::BOTTOMRIGHT, head_right - win_w, head_bot - win_h));
76 head_bot - win_h));
77 137
78 // go through the list of windows, creating other reasonable placements 138 // go through the list of windows, creating other reasonable placements
79 // at the end, we'll find the one with minimum overlap 139 // at the end, we'll find the one with minimum overlap
@@ -85,61 +145,45 @@ bool MinOverlapPlacement::placeWindow(const FluxboxWindow &win, int head,
85 for (; it != it_end; ++it) { 145 for (; it != it_end; ++it) {
86 if (*it == &win) continue; 146 if (*it == &win) continue;
87 147
88 // get the dimensions of the window 148 getWindowDimensions(*(*it), left, top, right, bottom);
89 int left = (*it)->x() - (*it)->xOffset();
90 int top = (*it)->y() - (*it)->yOffset();
91 int right = left + (*it)->width() +
92 2*(*it)->frame().window().borderWidth() +
93 (*it)->widthOffset();
94 int bottom = top + (*it)->height() +
95 2*(*it)->frame().window().borderWidth() +
96 (*it)->heightOffset();
97 149
98 // go through the list of regions 150 // go through the list of regions
99 // if this window overlaps that region and the new window still fits, 151 // if this window overlaps that region and the new window still fits,
100 // it will create new regions to test 152 // it will create new regions to test
101 std::set<Region>::iterator reg_it = region_set.begin(); 153 std::set<Area>::iterator ar_it = areas.begin();
102 for (; reg_it != region_set.end(); ++reg_it) { 154 for (; ar_it != areas.end(); ++ar_it) {
103 155
104 switch (reg_it->corner) { 156 switch (ar_it->corner) {
105 case Region::TOPLEFT: 157 case Area::TOPLEFT:
106 if (right > reg_it->x && bottom > reg_it->y) { 158 if (right > ar_it->x && bottom > ar_it->y) {
107 if (bottom + win_h <= head_bot) 159 if (bottom + win_h <= head_bot)
108 region_set.insert(Region(Region::TOPLEFT, 160 areas.insert(Area(Area::TOPLEFT, ar_it->x, bottom));
109 reg_it->x, bottom));
110 if (right + win_w <= head_right) 161 if (right + win_w <= head_right)
111 region_set.insert(Region(Region::TOPLEFT, 162 areas.insert(Area(Area::TOPLEFT, right, ar_it->y));
112 right, reg_it->y));
113 } 163 }
114 break; 164 break;
115 case Region::TOPRIGHT: 165 case Area::TOPRIGHT:
116 if (left < reg_it->x + win_w && bottom > reg_it->y) { 166 if (left < ar_it->x + win_w && bottom > ar_it->y) {
117 if (bottom + win_h <= head_bot) 167 if (bottom + win_h <= head_bot)
118 region_set.insert(Region(Region::TOPRIGHT, 168 areas.insert(Area(Area::TOPRIGHT, ar_it->x, bottom));
119 reg_it->x, bottom));
120 if (left - win_w >= head_left) 169 if (left - win_w >= head_left)
121 region_set.insert(Region(Region::TOPRIGHT, 170 areas.insert(Area(Area::TOPRIGHT, left - win_w, ar_it->y));
122 left - win_w, reg_it->y));
123 } 171 }
124 break; 172 break;
125 case Region::BOTTOMRIGHT: 173 case Area::BOTTOMRIGHT:
126 if (left < reg_it->x + win_w && top < reg_it->y + win_h) { 174 if (left < ar_it->x + win_w && top < ar_it->y + win_h) {
127 if (top - win_h >= head_top) 175 if (top - win_h >= head_top)
128 region_set.insert(Region(Region::BOTTOMRIGHT, 176 areas.insert(Area(Area::BOTTOMRIGHT, ar_it->x, top - win_h));
129 reg_it->x, top - win_h));
130 if (left - win_w >= head_left) 177 if (left - win_w >= head_left)
131 region_set.insert(Region(Region::BOTTOMRIGHT, 178 areas.insert(Area(Area::BOTTOMRIGHT, left - win_w, ar_it->y));
132 left - win_w, reg_it->y));
133 } 179 }
134 break; 180 break;
135 case Region::BOTTOMLEFT: 181 case Area::BOTTOMLEFT:
136 if (right > reg_it->x && top < reg_it->y + win_h) { 182 if (right > ar_it->x && top < ar_it->y + win_h) {
137 if (top - win_h >= head_top) 183 if (top - win_h >= head_top)
138 region_set.insert(Region(Region::BOTTOMLEFT, 184 areas.insert(Area(Area::BOTTOMLEFT, ar_it->x, top - win_h));
139 reg_it->x, top - win_h));
140 if (right + win_w <= head_right) 185 if (right + win_w <= head_right)
141 region_set.insert(Region(Region::BOTTOMLEFT, 186 areas.insert(Area(Area::BOTTOMLEFT, right, ar_it->y));
142 right, reg_it->y));
143 } 187 }
144 break; 188 break;
145 } 189 }
@@ -149,32 +193,22 @@ bool MinOverlapPlacement::placeWindow(const FluxboxWindow &win, int head,
149 193
150 // choose the region with minimum overlap 194 // choose the region with minimum overlap
151 int min_so_far = win_w * win_h * windowlist.size() + 1; 195 int min_so_far = win_w * win_h * windowlist.size() + 1;
152 std::set<Region>::iterator min_reg = region_set.end(); 196 std::set<Area>::iterator min_reg = areas.end();
153 197
154 std::set<Region>::iterator reg_it = region_set.begin(); 198 std::set<Area>::iterator ar_it = areas.begin();
155 for (; reg_it != region_set.end(); ++reg_it) { 199 for (; ar_it != areas.end(); ++ar_it) {
156 200
157 int overlap = 0; 201 int overlap = 0;
158 it = const_windowlist.rbegin(); 202 it = const_windowlist.rbegin();
159 for (; it != it_end; ++it) { 203 for (; it != it_end; ++it) {
160 204
161 // get the dimensions of the window 205 getWindowDimensions(*(*it), left, top, right, bottom);
162 int left = (*it)->x() - (*it)->xOffset();
163 int top = (*it)->y() - (*it)->yOffset();
164 int right = left + (*it)->width() +
165 2*(*it)->frame().window().borderWidth() +
166 (*it)->widthOffset();
167 int bottom = top + (*it)->height() +
168 2*(*it)->frame().window().borderWidth() +
169 (*it)->heightOffset();
170 206
171 // get the coordinates of the overlap region 207 // get the coordinates of the overlap region
172 int min_right = (right > reg_it->x + win_w) ? 208 int min_right = std::min(right, ar_it->x + win_w);
173 reg_it->x + win_w : right; 209 int min_bottom = std::min(bottom, ar_it->y + win_h);
174 int min_bottom = (bottom > reg_it->y + win_h) ? 210 int max_left = std::max(left, ar_it->x);
175 reg_it->y + win_h : bottom; 211 int max_top = std::max(top, ar_it->y);
176 int max_left = (left > reg_it->x) ? left : reg_it->x;
177 int max_top = (top > reg_it->y) ? top : reg_it->y;
178 212
179 // now compute the overlap and add to running total 213 // now compute the overlap and add to running total
180 if (min_right > max_left && min_bottom > max_top) 214 if (min_right > max_left && min_bottom > max_top)
@@ -184,7 +218,7 @@ bool MinOverlapPlacement::placeWindow(const FluxboxWindow &win, int head,
184 218
185 // if this placement is better, use it 219 // if this placement is better, use it
186 if (overlap < min_so_far) { 220 if (overlap < min_so_far) {
187 min_reg = reg_it; 221 min_reg = ar_it;
188 min_so_far = overlap; 222 min_so_far = overlap;
189 if (overlap == 0) // can't do better than this 223 if (overlap == 0) // can't do better than this
190 break; 224 break;