]> Dogcows Code - chaz/openbox/blob - scripts/motion.py
grab early in the cycle so that we dont miss the modifier release
[chaz/openbox] / scripts / motion.py
1 ############################################################################
2 ### Functions that provide callbacks for motion events to move and ###
3 ### resize windows. ###
4 ############################################################################
5
6 #############################################################################
7 ### Options that can be modified to change the functions' behaviors. ###
8 #############################################################################
9 EDGE_RESISTANCE = 10
10 """The amount of resistance to provide to moving a window past a screen
11 boundary. Specify a value of 0 to disable edge resistance."""
12 MOVE_POPUP = 1
13 """Display a coordinates popup when moving windows."""
14 MOVE_RUBBERBAND = 0
15 """NOT IMPLEMENTED (yet?)
16 Display an outline while moving instead of moving the actual window,
17 until the move is completed. Good for slower systems."""
18 RESIZE_POPUP = 1
19 """Display a size popup when resizing windows."""
20 RESIZE_RUBBERBAND = 0
21 """NOT IMPLEMENTED (yet?)
22 Display an outline while resizing instead of resizing the actual
23 window, until the resize is completed. Good for slower systems."""
24 RESIZE_NEAREST = 1
25 """Non-zero to resize from the corner nearest where the mouse is, 0 to
26 resize always from the bottom right corner."""
27 #############################################################################
28
29 def move(data):
30 """Moves the window interactively. This should only be used with
31 MouseAction.Motion events. If MOVE_POPUP or MOVE_RUBBERBAND is enabled,
32 then the end_move function needs to be bound as well."""
33 _move(data)
34
35 def end_move(data):
36 """Complete the interactive move of a window."""
37 _end_move(data)
38
39 def resize(data):
40 """Resizes the window interactively. This should only be used with
41 MouseMotion events. If RESIZE_POPUP or RESIZE_RUBBERBAND is enabled,
42 then the end_resize function needs to be bound as well."""
43 _resize(data)
44
45 def end_resize(data):
46 """Complete the interactive resize of a window."""
47 _end_resize(data)
48
49 ###########################################################################
50 ###########################################################################
51
52 ###########################################################################
53 ### Internal stuff, should not be accessed outside the module. ###
54 ###########################################################################
55
56 import ob
57 import otk
58
59 _popwidget = 0
60 _poplabel = 0
61
62 # motion state
63 _inmove = 0
64 _inresize = 0
65
66 # last motion data
67 _cx = 0
68 _cy = 0
69 _cw = 0
70 _ch = 0
71 _px = 0
72 _py = 0
73 _dx = 0
74 _dy = 0
75 _client = 0
76 _screen = 0
77
78 _motion_mask = 0
79
80 def _motion_grab(data):
81 global _motion_mask, _inmove, _inresize;
82
83 if data.action == ob.KeyAction.Release:
84 # have all the modifiers this started with been released?
85 if not _motion_mask & data.state:
86 if _inmove:
87 _end_move(data)
88 elif _inresize:
89 _end_resize(data)
90 else:
91 raise RuntimeError
92
93 _last_x = 0
94 _last_y = 0
95
96 def _do_move():
97 global _screen, _client, _cx, _cy, _dx, _dy
98
99 # get destination x/y for the *frame*
100 x = _cx + _dx + _client.frame.area().x() - _client.area().x()
101 y = _cy + _dy + _client.frame.area().y() - _client.area().y()
102
103 global _last_x, _last_y
104 if EDGE_RESISTANCE:
105 fs = _client.frame.size()
106 w = _client.area().width() + fs.left + fs.right
107 h = _client.area().height() + fs.top + fs.bottom
108 # use the area based on the struts
109 area = ob.openbox.screen(_screen).area()
110 l = area.left()
111 r = area.right() - w + 1
112 t = area.top()
113 b = area.bottom() - h + 1
114 # left screen edge
115 if _last_x > x and x < l and x >= l - EDGE_RESISTANCE:
116 x = l
117 # right screen edge
118 if _last_x < x and x > r and x <= r + EDGE_RESISTANCE:
119 x = r
120 # top screen edge
121 if _last_y > y and y < t and y >= t - EDGE_RESISTANCE:
122 y = t
123 # right screen edge
124 if _last_y < y and y > b and y <= b + EDGE_RESISTANCE:
125 y = b
126
127 global _inmove
128 if not _inmove:
129 _last_x = 0
130 _last_y = 0
131 else:
132 _last_x = x
133 _last_y = y
134
135 if MOVE_RUBBERBAND:
136 # draw the outline ...
137 f=0
138 else:
139 _client.move(x, y)
140
141 if MOVE_POPUP:
142 global _popwidget, _poplabel
143 text = "X: " + str(x) + " Y: " + str(y)
144 if not _popwidget:
145 _popwidget = otk.Widget(_screen, ob.openbox,
146 otk.Widget.Horizontal, 0, 1)
147 _poplabel = otk.Label(_popwidget)
148 _poplabel.setText(text)
149 scsize = otk.display.screenInfo(_screen).size()
150 size = _poplabel.minSize()
151 _popwidget.resize(_poplabel.minSize())
152 _popwidget.move(otk.Point((scsize.width() - size.width()) / 2,
153 (scsize.height() - size.height()) / 2))
154 _popwidget.show(1)
155
156 def _move(data):
157 if not data.client: return
158
159 # not-normal windows dont get moved
160 if not data.client.normal(): return
161
162 global _screen, _client, _cx, _cy, _dx, _dy
163 _screen = data.screen
164 _client = data.client
165 _cx = data.press_clientx
166 _cy = data.press_clienty
167 _dx = data.xroot - data.pressx
168 _dy = data.yroot - data.pressy
169 _do_move()
170 global _inmove
171 if not _inmove:
172 ob.kgrab(_screen, _motion_grab)
173 _inmove = 1
174
175 def _end_move(data):
176 global MOVE_RUBBERBAND
177 global _inmove, _popwidget, _poplabel
178 if _inmove:
179 r = MOVE_RUBBERBAND
180 MOVE_RUBBERBAND = 0
181 _do_move()
182 MOVE_RUBBERBAND = r
183 _inmove = 0
184 _poplabel = 0
185 _popwidget = 0
186 ob.kungrab()
187
188 def _do_resize():
189 global _screen, _client, _cx, _cy, _cw, _ch, _px, _py, _dx, _dy
190
191 dx = _dx
192 dy = _dy
193
194 # pick a corner to anchor
195 if not (RESIZE_NEAREST or _context == ob.MouseContext.Grip):
196 corner = ob.Client.TopLeft
197 else:
198 x = _px - _cx
199 y = _py - _cy
200 if y < _ch / 2:
201 if x < _cw / 2:
202 corner = ob.Client.BottomRight
203 dx *= -1
204 else:
205 corner = ob.Client.BottomLeft
206 dy *= -1
207 else:
208 if x < _cw / 2:
209 corner = ob.Client.TopRight
210 dx *= -1
211 else:
212 corner = ob.Client.TopLeft
213
214 w = _cw + dx
215 h = _ch + dy
216 if w < 0: w = 0
217 if h < 0: h = 0
218
219 if RESIZE_RUBBERBAND:
220 # draw the outline ...
221 f=0
222 else:
223 _client.resize(corner, w, h)
224
225 if RESIZE_POPUP:
226 global _popwidget, _poplabel
227 ls = _client.logicalSize()
228 text = "W: " + str(ls.width()) + " H: " + str(ls.height())
229 if not _popwidget:
230 _popwidget = otk.Widget(_screen, ob.openbox,
231 otk.Widget.Horizontal, 0, 1)
232 _poplabel = otk.Label(_popwidget)
233 _poplabel.setText(text)
234 scsize = otk.display.screenInfo(_screen).size()
235 size = _poplabel.minSize()
236 _popwidget.resize(_poplabel.minSize())
237 _popwidget.move(otk.Point((scsize.width() - size.width()) / 2,
238 (scsize.height() - size.height()) / 2))
239 _popwidget.show(1)
240
241 def _resize(data):
242 if not data.client: return
243
244 # not-normal windows dont get resized
245 if not data.client.normal(): return
246
247 global _screen, _client, _cx, _cy, _cw, _ch, _px, _py, _dx, _dy
248 _screen = data.screen
249 _client = data.client
250 _cx = data.press_clientx
251 _cy = data.press_clienty
252 _cw = data.press_clientwidth
253 _ch = data.press_clientheight
254 _px = data.pressx
255 _py = data.pressy
256 _dx = data.xroot - _px
257 _dy = data.yroot - _py
258 _do_resize()
259 global _inresize
260 if not _inresize:
261 ob.kgrab(_screen, _motion_grab)
262 _inresize = 1
263
264 def _end_resize(data):
265 global RESIZE_RUBBERBAND, _inresize
266 global _popwidget, _poplabel
267 if _inresize:
268 r = RESIZE_RUBBERBAND
269 RESIZE_RUBBERBAND = 0
270 _do_resize()
271 RESIZE_RUBBERBAND = r
272 _inresize = 0
273 _poplabel = 0
274 _popwidget = 0
275 ob.kungrab()
This page took 0.047718 seconds and 4 git commands to generate.