删除 Tkinter 中的特定元素 canvas
Delete a specific element in Tkinter canvas
我正在使用 tkinter 创建有向图。在此图中,我试图删除所有创建的箭头或特定箭头。
这是代码:
from tkinter import *
import math
class Window:
#Constructor
def __init__(self, window, colour="white", width=1000, height=700):
#Set variables
self.width = width
self.height = height
self.colour = colour
self.ligacao = []
#This dictionary is used to keep track of an item being dragged
self._drag_data = {"x": 0, "y": 0, "item": None}
#Create canvas
self.canvas = Canvas(window, bg=self.colour, height=self.height, width=self.width)
self.canvas.pack()
#Add bindings for clicking, dragging and releasing overo any object with the "group_drag" tag
self.canvas.bind("<ButtonPress-1>", self.OnCircleButtonPress)
self.canvas.bind("<ButtonRelease-1>", self.OnCircleButtonRelease)
self.canvas.bind("<B1-Motion>", self.OnCircleMotion)
#This is used to draw particle objects on the canvas, notice the tag that has been added as an attribute
def _create_circle(self, xcoord, ycoord, r, outline, color, tag):
self.circle = self.canvas.create_oval(xcoord - r, ycoord - r, xcoord + r, ycoord + r,
outline=outline, fill=color, tags=tag)
#This is used to draw text on top of the object on the canvas
def _create_text(self, xcoord, ycoord, text, tag):
self.text = self.canvas.create_text(xcoord, ycoord, text=text, tags=tag)
def delete_create(self, event):
self.canvas.delete(self.ligacao)
self._create_arrow(300, 500, event.x, event.y, 25, "black", "item-20")
def _create_arrow(self, x0, y0, x1, y1, r, colour, tag):
delta = math.sqrt(math.pow((y1 - y0), 2) + math.pow((x1 - x0), 2))
if delta != 0:
self.arrow = {"x0": x0 + r * (x1 - x0) / delta, "y0": y0 + r * (y1 - y0) / delta,
"x1": x1 - r * (x1 - x0) / delta, "y1": y1 - r * (y1 - y0) / delta, "tag": tag}
self.ligacao.append(self.canvas.create_line(self.arrow["x0"], self.arrow["y0"], self.arrow["x1"], self.arrow["y1"],
fill=colour, arrow=LAST, tags=tag))
else:
self.arrow = {"x0": x0, "y0": y0, "x1": x1, "y1": y1, "tag": tag}
self.ligacao.append(self.canvas.create_line(x0, y0, x1, y1, arrow=LAST, fill=colour, tags=tag))
return [self.arrow["x0"], self.arrow["y0"], self.arrow["x1"], self.arrow["y1"], self.arrow["tag"]]
#This uses the find_closest method to get store the x and y positions of the nearest item into the dictionary
def OnCircleButtonPress(self, event):
'''Begin drag of an object'''
# record the item and its location
item = self.canvas.find_closest(event.x, event.y)[0]
tags = self.canvas.gettags(item)
for tag in tags:
if tag.startswith("item-"):
break
#print(tag)
self._drag_data["item"] = tag
self._drag_data["x"] = event.x
self._drag_data["y"] = event.y
#This clears the dictionary once the mouse button has been released
def OnCircleButtonRelease(self, event):
'''End drag of an object'''
self.canvas.tag_bind(self.circle, "<ButtonRelease-1>", self.delete_create)
self.canvas.tag_bind(self.text, "<ButtonRelease-1>", self.delete_create)
# reset the drag information
self._drag_data["item"] = None
self._drag_data["x"] = 0
self._drag_data["y"] = 0
#This moves the item as it is being dragged around the screen
def OnCircleMotion(self, event):
'''Handle dragging of an object'''
# compute how much this object has moved
delta_x = event.x - self._drag_data["x"]
delta_y = event.y - self._drag_data["y"]
# move the object the appropriate amount
self.canvas.move(self._drag_data["item"], delta_x, delta_y)
# record the new position
self._drag_data["x"] = event.x
self._drag_data["y"] = event.y
class Picture:
#Constructor
def __init__(self, window):
self.window = window
def CircleAndText(self, width=100, height=100, r=50, outline="black", colour="#8BF1F5", text="test", t="item-10"):
self.window._create_circle(width, height, r, outline, colour, t)
self.window._create_text(width, height, text, t)
def ArrowAndText(self, x0, y0, x1, y1, r, colour, text, t="item-20"):
s = self.window._create_arrow(x0, y0, x1, y1, r, colour, t)
self.window._create_text(abs(s[0]-s[2])/2 + min(s[0], s[2]),
abs(s[1]-s[3])/2 + min(s[1], s[3]), text, t)
return s
window = Tk()
window.title("Drag & Drop")
#Create an instance of the window class
main_window = Window(window)
dictionary = {"James": [("Paul", 3), ("Julie", 2), ("Lisa", 1)], "Lisa": [("Paul", 5), ("Antonio", 2)],
"Fabian": [("George", 2), ("Diana", 4), ("Julie", 5)], "Julie": [("Paul", 2), ("Antonio", 1)], "Antonio": [("Mary", 1), ("Diana", 2)]}
r = 25
x_meio = int(main_window.width/2)
y_meio = int(main_window.height/2)
i = 1
j = 1
circulo = []
texto_circulo = []
tag_seta = []
seta = []
texto_seta = []
tag = []
keys = dictionary.keys()
dictionary_len = len(keys)
angulo = 2 * math.pi / dictionary_len
raio_x = x_meio - r
raio_y = y_meio - r
for name in dictionary:
k = x_meio + raio_x*math.cos(angulo*(i-1))
l = y_meio + raio_y*math.sin(angulo*(i-1))
m = x_meio + raio_x*math.cos(angulo*i)
n = y_meio + raio_y*math.sin(angulo*i)
Picture(main_window).CircleAndText(k, l, r, "black", "#8BF1F5", name, "item-%s" % int(i*10))
Picture(main_window).CircleAndText(k, l, r, "black", "#8BF1F5", name, "item-%s" % int(i*10))
Picture(main_window).CircleAndText(k, l, r, "black", "#8BF1F5", name, "item-%s" % int(i*10))
Picture(main_window).ArrowAndText(k, l, m, n, r, "black", name, "item-%s" % int(i * 10))
Picture(main_window).ArrowAndText(k, l, m, n, r, "black", name, "item-%s" % int(i * 10))
i += 1
#Start the animation loop
window.mainloop()
当我创建箭头时,self.ligacao
列表创建如下:
def _create_arrow(self, x0, y0, x1, y1, r, colour, tag):
delta = math.sqrt(math.pow((y1 - y0), 2) + math.pow((x1 - x0), 2))
if delta != 0:
self.arrow = {"x0": x0 + r * (x1 - x0) / delta, "y0": y0 + r * (y1 - y0) / delta,
"x1": x1 - r * (x1 - x0) / delta, "y1": y1 - r * (y1 - y0) / delta, "tag": tag}
self.ligacao.append(self.canvas.create_line(self.arrow["x0"], self.arrow["y0"], self.arrow["x1"], self.arrow["y1"],
fill=colour, arrow=LAST, tags=tag))
else:
self.arrow = {"x0": x0, "y0": y0, "x1": x1, "y1": y1, "tag": tag}
self.ligacao.append(self.canvas.create_line(x0, y0, x1, y1, arrow=LAST, fill=colour, tags=tag))
return [self.arrow["x0"], self.arrow["y0"], self.arrow["x1"], self.arrow["y1"], self.arrow["tag"]]
这是我用来删除一个箭头并在新位置创建另一个箭头的方法:
def delete_create(self, event):
self.canvas.delete(self.ligacao)
self._create_arrow(300, 500, event.x, event.y, 25, "black", "item-20")
当我使用下面这段代码时,我只能删除最后一个箭头,释放最后一个圆圈或文本,但我想删除所有箭头,或特定的,更容易完成的代码。
self.canvas.tag_bind(self.circle, "<ButtonRelease-1>", self.delete_create)
self.canvas.tag_bind(self.text, "<ButtonRelease-1>", self.delete_create)
如何使用 tag_bind
函数删除所有箭头或特定箭头?
删除所有箭头的最简单方法是给每个箭头至少两个标签:给它一个 "arrow" 标签以及该箭头唯一的标签。当您将 "arrow"
传递给 delete
方法时,它将删除所有带有该标签的项目。
例如,像这样创建箭头:
def _create_arrow(self, x0, y0, x1, y1, r, colour, tag):
tags = (tag, "arrow")
...
self.ligacao.append(self.canvas.create_line(..., tags=tags))
删除所有箭头就变成了self.canvas.delete("arrow")
我正在使用 tkinter 创建有向图。在此图中,我试图删除所有创建的箭头或特定箭头。
这是代码:
from tkinter import *
import math
class Window:
#Constructor
def __init__(self, window, colour="white", width=1000, height=700):
#Set variables
self.width = width
self.height = height
self.colour = colour
self.ligacao = []
#This dictionary is used to keep track of an item being dragged
self._drag_data = {"x": 0, "y": 0, "item": None}
#Create canvas
self.canvas = Canvas(window, bg=self.colour, height=self.height, width=self.width)
self.canvas.pack()
#Add bindings for clicking, dragging and releasing overo any object with the "group_drag" tag
self.canvas.bind("<ButtonPress-1>", self.OnCircleButtonPress)
self.canvas.bind("<ButtonRelease-1>", self.OnCircleButtonRelease)
self.canvas.bind("<B1-Motion>", self.OnCircleMotion)
#This is used to draw particle objects on the canvas, notice the tag that has been added as an attribute
def _create_circle(self, xcoord, ycoord, r, outline, color, tag):
self.circle = self.canvas.create_oval(xcoord - r, ycoord - r, xcoord + r, ycoord + r,
outline=outline, fill=color, tags=tag)
#This is used to draw text on top of the object on the canvas
def _create_text(self, xcoord, ycoord, text, tag):
self.text = self.canvas.create_text(xcoord, ycoord, text=text, tags=tag)
def delete_create(self, event):
self.canvas.delete(self.ligacao)
self._create_arrow(300, 500, event.x, event.y, 25, "black", "item-20")
def _create_arrow(self, x0, y0, x1, y1, r, colour, tag):
delta = math.sqrt(math.pow((y1 - y0), 2) + math.pow((x1 - x0), 2))
if delta != 0:
self.arrow = {"x0": x0 + r * (x1 - x0) / delta, "y0": y0 + r * (y1 - y0) / delta,
"x1": x1 - r * (x1 - x0) / delta, "y1": y1 - r * (y1 - y0) / delta, "tag": tag}
self.ligacao.append(self.canvas.create_line(self.arrow["x0"], self.arrow["y0"], self.arrow["x1"], self.arrow["y1"],
fill=colour, arrow=LAST, tags=tag))
else:
self.arrow = {"x0": x0, "y0": y0, "x1": x1, "y1": y1, "tag": tag}
self.ligacao.append(self.canvas.create_line(x0, y0, x1, y1, arrow=LAST, fill=colour, tags=tag))
return [self.arrow["x0"], self.arrow["y0"], self.arrow["x1"], self.arrow["y1"], self.arrow["tag"]]
#This uses the find_closest method to get store the x and y positions of the nearest item into the dictionary
def OnCircleButtonPress(self, event):
'''Begin drag of an object'''
# record the item and its location
item = self.canvas.find_closest(event.x, event.y)[0]
tags = self.canvas.gettags(item)
for tag in tags:
if tag.startswith("item-"):
break
#print(tag)
self._drag_data["item"] = tag
self._drag_data["x"] = event.x
self._drag_data["y"] = event.y
#This clears the dictionary once the mouse button has been released
def OnCircleButtonRelease(self, event):
'''End drag of an object'''
self.canvas.tag_bind(self.circle, "<ButtonRelease-1>", self.delete_create)
self.canvas.tag_bind(self.text, "<ButtonRelease-1>", self.delete_create)
# reset the drag information
self._drag_data["item"] = None
self._drag_data["x"] = 0
self._drag_data["y"] = 0
#This moves the item as it is being dragged around the screen
def OnCircleMotion(self, event):
'''Handle dragging of an object'''
# compute how much this object has moved
delta_x = event.x - self._drag_data["x"]
delta_y = event.y - self._drag_data["y"]
# move the object the appropriate amount
self.canvas.move(self._drag_data["item"], delta_x, delta_y)
# record the new position
self._drag_data["x"] = event.x
self._drag_data["y"] = event.y
class Picture:
#Constructor
def __init__(self, window):
self.window = window
def CircleAndText(self, width=100, height=100, r=50, outline="black", colour="#8BF1F5", text="test", t="item-10"):
self.window._create_circle(width, height, r, outline, colour, t)
self.window._create_text(width, height, text, t)
def ArrowAndText(self, x0, y0, x1, y1, r, colour, text, t="item-20"):
s = self.window._create_arrow(x0, y0, x1, y1, r, colour, t)
self.window._create_text(abs(s[0]-s[2])/2 + min(s[0], s[2]),
abs(s[1]-s[3])/2 + min(s[1], s[3]), text, t)
return s
window = Tk()
window.title("Drag & Drop")
#Create an instance of the window class
main_window = Window(window)
dictionary = {"James": [("Paul", 3), ("Julie", 2), ("Lisa", 1)], "Lisa": [("Paul", 5), ("Antonio", 2)],
"Fabian": [("George", 2), ("Diana", 4), ("Julie", 5)], "Julie": [("Paul", 2), ("Antonio", 1)], "Antonio": [("Mary", 1), ("Diana", 2)]}
r = 25
x_meio = int(main_window.width/2)
y_meio = int(main_window.height/2)
i = 1
j = 1
circulo = []
texto_circulo = []
tag_seta = []
seta = []
texto_seta = []
tag = []
keys = dictionary.keys()
dictionary_len = len(keys)
angulo = 2 * math.pi / dictionary_len
raio_x = x_meio - r
raio_y = y_meio - r
for name in dictionary:
k = x_meio + raio_x*math.cos(angulo*(i-1))
l = y_meio + raio_y*math.sin(angulo*(i-1))
m = x_meio + raio_x*math.cos(angulo*i)
n = y_meio + raio_y*math.sin(angulo*i)
Picture(main_window).CircleAndText(k, l, r, "black", "#8BF1F5", name, "item-%s" % int(i*10))
Picture(main_window).CircleAndText(k, l, r, "black", "#8BF1F5", name, "item-%s" % int(i*10))
Picture(main_window).CircleAndText(k, l, r, "black", "#8BF1F5", name, "item-%s" % int(i*10))
Picture(main_window).ArrowAndText(k, l, m, n, r, "black", name, "item-%s" % int(i * 10))
Picture(main_window).ArrowAndText(k, l, m, n, r, "black", name, "item-%s" % int(i * 10))
i += 1
#Start the animation loop
window.mainloop()
当我创建箭头时,self.ligacao
列表创建如下:
def _create_arrow(self, x0, y0, x1, y1, r, colour, tag):
delta = math.sqrt(math.pow((y1 - y0), 2) + math.pow((x1 - x0), 2))
if delta != 0:
self.arrow = {"x0": x0 + r * (x1 - x0) / delta, "y0": y0 + r * (y1 - y0) / delta,
"x1": x1 - r * (x1 - x0) / delta, "y1": y1 - r * (y1 - y0) / delta, "tag": tag}
self.ligacao.append(self.canvas.create_line(self.arrow["x0"], self.arrow["y0"], self.arrow["x1"], self.arrow["y1"],
fill=colour, arrow=LAST, tags=tag))
else:
self.arrow = {"x0": x0, "y0": y0, "x1": x1, "y1": y1, "tag": tag}
self.ligacao.append(self.canvas.create_line(x0, y0, x1, y1, arrow=LAST, fill=colour, tags=tag))
return [self.arrow["x0"], self.arrow["y0"], self.arrow["x1"], self.arrow["y1"], self.arrow["tag"]]
这是我用来删除一个箭头并在新位置创建另一个箭头的方法:
def delete_create(self, event):
self.canvas.delete(self.ligacao)
self._create_arrow(300, 500, event.x, event.y, 25, "black", "item-20")
当我使用下面这段代码时,我只能删除最后一个箭头,释放最后一个圆圈或文本,但我想删除所有箭头,或特定的,更容易完成的代码。
self.canvas.tag_bind(self.circle, "<ButtonRelease-1>", self.delete_create)
self.canvas.tag_bind(self.text, "<ButtonRelease-1>", self.delete_create)
如何使用 tag_bind
函数删除所有箭头或特定箭头?
删除所有箭头的最简单方法是给每个箭头至少两个标签:给它一个 "arrow" 标签以及该箭头唯一的标签。当您将 "arrow"
传递给 delete
方法时,它将删除所有带有该标签的项目。
例如,像这样创建箭头:
def _create_arrow(self, x0, y0, x1, y1, r, colour, tag):
tags = (tag, "arrow")
...
self.ligacao.append(self.canvas.create_line(..., tags=tags))
删除所有箭头就变成了self.canvas.delete("arrow")