【问题标题】:Draggable Tkinter Label With Transparent Image Still Covers Image from Parent Canvas具有透明图像的可拖动 Tkinter 标签仍覆盖父画布中的图像
【发布时间】:2018-06-27 21:46:49
【问题描述】:

我正在努力在背景图像上添加可拖动标签,其中标签是具有透明背景的图像。用于标签本身的图像是透明的,但标签本身相对于其父画布不是透明的。由于标签是可拖动的,因此我不能轻易地将父图像用作标签,并将透明图像粘贴在顶部。

龙是一个可拖动的标签,背景是透明的,但是你看到标签本身是不透明的,并且覆盖了画布图像。

龙图像本身具有透明背景,因为正常背景是蓝色的,所以我知道问题出在标签透明度上。

希望的行为是让标签透明,因此当标签上的图像透明时,标签应该显示到下面的图像。

这是一个最小的例子:

from PIL import Image, ImageTk
import numpy as np
import tkinter as tk

#Creates a blue square with a transparent border
blue_square_transparent_border = [[[0,0,0,0]]*100]*10 + [[[0,0,0,0]]*30 + [[0,0,255,255]]*40 + [[0,0,0,0]]*30]*40 + [[[0,0,0,0]]*100]*10
blue_square_transparent_border = np.array(blue_square_transparent_border, dtype='uint8')
#convert numpy array to PIL image
pil_image = Image.fromarray(blue_square_transparent_border)

root = tk.Tk()
root.configure(background='red')
#convert PIL image to tkinter image
tk_image = ImageTk.PhotoImage(pil_image)
#create label
image_label = tk.Label(root, image=tk_image)
image_label.pack()
root.mainloop()

我希望看到的是红色背景上的蓝色方块,没有边框。但是在上面的例子中,出现了灰色的边框,因为它是通过透明图像看到的标签;调整窗口大小时很容易看到这一点。我怀疑如果标签是透明的,这会解决我的问题。

任何帮助都会很棒,谢谢!

【问题讨论】:

  • 我认为这可能会有所帮助stackoverflow.com/q/10461045/7032856
  • @Nae 很抱歉起初的问题很糟糕!我用一个简单的例子更新了信息。我也查看了您的链接,但我认为它不能解决我的问题!如果您有更多意见,请告诉我!
  • 所有小部件都有背景颜色,您不能将其设置为透明。您只能使用create_image 将透明图像放在Canvas 小部件上。
  • 感谢@furas 提供的信息,我担心这就是答案。那么有没有办法制作一个非矩形的小部件呢?

标签: python tkinter tkinter-canvas


【解决方案1】:

如果您想将部分透明的图像(如龙)拖到背景图像上,您可以使用Canvas

想法不是使用标签,而是Canvascreate_image 方法。 首先,使用canvas.create_image(0, 0, image=background_image, anchor='nw') 显示您的背景图像,然后使用标签'draggable' 显示所有可拖动图像:canvas.create_image(x, y, image=draggable_image, anchor='nw', tag='draggable')。最后,将标签“可拖动”绑定到鼠标事件。

这是一个例子:

import tkinter as tk
import numpy as np
from PIL import Image, ImageTk

# drag callbacks
dragged_item = None
current_coords = 0, 0

def start_drag(event):
    global current_coords
    global dragged_item
    result = canvas.find_withtag('current')
    if result:
        dragged_item = result[0]
        current_coords = canvas.canvasx(event.x), canvas.canvasy(event.y)
    else:
        dragged_item = None

def stop_drag(event):
    dragged_item = None

def drag(event):
    global current_coords
    xc, yc = canvas.canvasx(event.x), canvas.canvasy(event.y)
    dx, dy = xc - current_coords[0], yc - current_coords[1]
    current_coords = xc, yc
    canvas.move(dragged_item, dx, dy)


#Create pictures
blue_square_transparent_border = [[[0,0,0,0]]*100]*10 + [[[0,0,0,0]]*30 + [[0,0,255,255]]*40 + [[0,0,0,0]]*30]*40 + [[[0,0,0,0]]*100]*10
blue_square_transparent_border = np.array(blue_square_transparent_border, dtype='uint8')
pil_image = Image.fromarray(blue_square_transparent_border)

background_data = np.zeros((200, 400, 4))
background_data[:, :, 0] = 255 * np.ones((200, 400))
background_data[:, :, 3] = 255 * np.ones((200, 400))
background_data = np.array(background_data, dtype='uint8')
pil_image_bg = Image.fromarray(background_data)

# create GUI
root = tk.Tk()
background_image = ImageTk.PhotoImage(pil_image_bg)
tk_image = ImageTk.PhotoImage(pil_image)

canvas = tk.Canvas(root, width=400, height=200)
canvas.pack()
# bind 'draggable' tag to mouse events
canvas.tag_bind('draggable', '<ButtonPress-1>', start_drag)
canvas.tag_bind('draggable', '<ButtonRelease-1>', stop_drag)
canvas.tag_bind('draggable', '<B1-Motion>', drag)
# display pictures
canvas.create_image(0, 0, image=background_image, anchor='nw')
canvas.create_image(0, 0, image=tk_image, anchor='nw', tag='draggable')

root.mainloop()

【讨论】:

  • 谢谢!我确实注意到您可以在画布上绘制两个图像,并将它们作为单独的图像进行维护。我曾假设一个会覆盖另一个。这是一个更简单的解决方案!
猜你喜欢
  • 1970-01-01
  • 2014-08-12
  • 2014-02-27
  • 2012-05-13
  • 2017-09-25
  • 2023-03-30
  • 2017-05-20
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多