-
Notifications
You must be signed in to change notification settings - Fork 4
Expand file tree
/
Copy pathtk_dragtool.py
More file actions
165 lines (154 loc) · 6.88 KB
/
tk_dragtool.py
File metadata and controls
165 lines (154 loc) · 6.88 KB
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
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
"""提供用鼠标拖动、缩放tkinter控件工具的跨平台模块。
A cross-platform module providing tools to drag
and resize tkinter windows and widgets with the mouse."""
import tkinter as tk
import tkinter.ttk as ttk
__author__="fufvocdi"
__version__="1.1.4"
# tkinter控件的对象能够作为字典键。
# bound的键是dragger, 值是一个列表, 包含了若干个绑定事件, 用于存储绑定数据
# 列表的每个项是一个元组, 包含了tkwidget和其他绑定参数
bound = {}
def __add(wid,data):# 添加绑定数据
bound[wid]=bound.get(wid,[])+[data]
def __remove(wid,key): # 用于从bound中移除绑定
for i in range(len(bound[wid])):
try:
if bound[wid][i][0]==key:
del bound[wid][i]
except IndexError:pass
def __get(wid,key=''): # 用于从bound中获取绑定数据
if not key:return bound[wid][0]
if key=='resize':
for i in range(len(bound[wid])):
for s in 'nwse':
if s in bound[wid][i][0].lower():
return bound[wid][i]
for i in range(len(bound[wid])):
if bound[wid][i][0]==key:
return bound[wid][i]
def move(widget,x=None,y=None,width=None,height=None):
"移动控件或窗口widget至指定坐标, 参数都为可选参数。"
x=x if x!=None else widget.winfo_x()
y=y if y!=None else widget.winfo_y()
width=width if width!=None else widget.winfo_width()
height=height if height!=None else widget.winfo_height()
if isinstance(widget,tk.Wm):
widget.geometry("%dx%d+%d+%d"%(width,height,x,y))
else:
widget.place(x=x,y=y,width=width,height=height)
return x,y,width,height
def _mousedown(event):
if event.widget not in bound:return
lst=bound[event.widget]
for data in lst: # 开始拖动时, 在每一个控件记录位置和控件尺寸
widget=data[1]
widget.mousex,widget.mousey = widget.winfo_pointerxy() # 获取初始鼠标位置
widget.startx,widget.starty = widget.winfo_x(),widget.winfo_y() # 获取相对坐标
widget.start_w=widget.winfo_width()
widget.start_h=widget.winfo_height()
def _drag(event):
if event.widget not in bound:return
lst=bound[event.widget]
for data in lst: # 多个绑定
if data[0]!='drag':return
widget=data[1]
dx = widget.winfo_pointerx()-widget.mousex # 计算鼠标当前位置和开始拖动时位置的差距
# 注: 鼠标位置不能用event.x和event.y
# event.x,event.y与控件的位置、大小有关,不能真实地反映鼠标移动的距离差值
dy = widget.winfo_pointery()+widget.winfo_vrooty()-widget.mousey
move(widget,widget.startx + dx if data[2] else None,
widget.starty + dy if data[3] else None)
def _resize(event):
data=__get(event.widget,'resize')
if data is None:return
widget=data[1]
dx = widget.winfo_pointerx()-widget.mousex # 计算位置差
dy = widget.winfo_pointery()-widget.mousey
type = data[0].lower()
minw,minh = data[2:4]
if 's' in type:
move(widget,height=max(widget.start_h+dy,minh))
elif 'n' in type:
move(widget,y=min(widget.starty+dy,widget.starty+widget.start_h-minh),
height=max(widget.start_h-dy,minh))
__remove(event.widget,data[0])# 取消绑定, 为防止widget.update()中产生新的事件, 避免_resize()被tkinter反复调用
widget.update() # 刷新控件, 使以下左右缩放时, winfo_height()返回的是新的控件坐标, 而不是旧的
__add(event.widget,data) # 重新绑定
if 'e' in type:
move(widget,width=max(widget.start_w+dx,minw))
elif 'w' in type:
move(widget,x=min(widget.startx+dx,widget.startx+widget.start_w-minw),
width=max(widget.start_w-dx,minw))
def draggable(tkwidget,x=True,y=True):
"""调用draggable(tkwidget) 使tkwidget可拖动。
tkwidget: 一个控件(Widget)或一个窗口(Wm)。
x 和 y: 只允许改变x坐标或y坐标。"""
bind_drag(tkwidget,tkwidget,x,y)
def bind_drag(tkwidget,dragger,x=True,y=True):
"""绑定拖曳事件。
tkwidget: 被拖动的控件或窗口,
dragger: 接收鼠标事件的控件,
调用bind_drag后,当鼠标拖动dragger时, tkwidget会被带着拖动, 但dragger
作为接收鼠标事件的控件, 位置不会改变。
x 和 y: 同draggable()函数。"""
dragger.bind("<Button-1>",_mousedown,add='+')
dragger.bind("<B1-Motion>",_drag,add='+')
__add(dragger,('drag',tkwidget,x,y)) # 在bound字典中记录数据
def bind_resize(tkwidget,dragger,anchor,min_w=0,min_h=0,move_dragger=True):
"""绑定缩放事件。
anchor: 缩放"手柄"的方位, 取值为N,S,W,E,NW,NE,SW,SE,分别表示东、西、南、北。
min_w,min_h: 该方向tkwidget缩放的最小宽度(或高度)。
move_dragger: 缩放时是否移动dragger。
其他说明同bind_drag函数。"""
dragger.bind("<Button-1>",_mousedown,add='+')
dragger.bind("<B1-Motion>",_resize,add='+')
data=(anchor,tkwidget,min_w,min_h,move_dragger)
__add(dragger,data)
def test():
btns=[] # 用btns列表存储创建的按钮
def add_button(func,anchor):
# func的作用是计算按钮新坐标
b=ttk.Button(root)
b._func=func
bind_resize(btn,b,anchor)
x,y=func()
b.place(x=x,y=y,width=size,height=size)
b.bind('<B1-Motion>',adjust_button,add='+')
b.bind('<B1-ButtonRelease>',adjust_button,add='+')
btns.append(b)
def adjust_button(event=None):
# 改变大小或拖动后,调整手柄位置
for b in btns:
x,y=b._func()
b.place(x=x,y=y)
root=tk.Tk()
root.title("Test")
root.geometry('500x350')
btn=ttk.Button(root,text="Button")
draggable(root)
draggable(btn)
btn.bind('<B1-Motion>',adjust_button,add='+')
btn.bind('<B1-ButtonRelease>',adjust_button,add='+')
x1=20;y1=20;x2=220;y2=170;size=10
btn.place(x=x1,y=y1,width=x2-x1,height=y2-y1)
root.update()
# 创建各个手柄, 这里是控件缩放的算法
add_button(lambda:(btn.winfo_x()-size, btn.winfo_y()-size),
'nw')
add_button(lambda:(btn.winfo_x()+btn.winfo_width()//2,
btn.winfo_y()-size), 'n')
add_button(lambda:(btn.winfo_x()+btn.winfo_width(), btn.winfo_y()-size),
'ne')
add_button(lambda:(btn.winfo_x()+btn.winfo_width(),
btn.winfo_y()+btn.winfo_height()//2),'e')
add_button(lambda:(btn.winfo_x()+btn.winfo_width(),
btn.winfo_y()+btn.winfo_height()), 'se')
add_button(lambda:(btn.winfo_x()+btn.winfo_width()//2,
btn.winfo_y()+btn.winfo_height()),'s')
add_button(lambda:(btn.winfo_x()-size, btn.winfo_y()+btn.winfo_height()),
'sw')
add_button(lambda:(btn.winfo_x()-size,
btn.winfo_y()+btn.winfo_height()//2), 'w')
root.mainloop()
if __name__=="__main__":test()