![]() |
ImageTk Paste - Printable Version +- Python Forum (https://python-forum.io) +-- Forum: Python Coding (https://python-forum.io/forum-7.html) +--- Forum: General Coding Help (https://python-forum.io/forum-8.html) +--- Thread: ImageTk Paste (/thread-33838.html) Pages:
1
2
|
RE: ImageTk Paste - KDog - Jun-02-2021 Tried using PIL but still no joy: import tkinter as tk import ImageChops import cv2 from PIL import Image, ImageTk, ImageEnhance import PIL import time from tkinter import filedialog import numpy as np class App: def __init__(self, video_source=0): self.overlay_img = None self.appName = "Kamera" self.window = tk.Tk() self.window.title(self.appName) self.window.resizable(0, 0) # self.window.wm_iconbitmap("cam.ico") self.window['bg'] = 'black' self.video_source = video_source self.vid = MyVideoCapture(self.video_source) # self.label = Label(self.window, text=self.appName, font=15, bg='blue', fg='white').pack(side=TOP, fill=BOTH) self.canvas = tk.Canvas(self.window, width=self.vid.width, height=self.vid.height, bg='red') self.canvas.pack() self.btn_snapshot = tk.Button(self.window, text="Snapshot", width=5, command=self.snapshot) self.btn_snapshot.pack(side=tk.LEFT, padx=10) self.btn_overlay = tk.Button(self.window, text="Overlay", width=7, command=self.overlay) self.btn_overlay.pack(side=tk.LEFT, padx=10) self.btn_settings = tk.Button(self.window, text="Settings", width=5, command=self.settings) self.btn_settings.pack(side=tk.LEFT, padx=10) self.slide_value = 200 self.update() self.window.mainloop() def settings(self): self.newWindow = tk.Toplevel(self.window) self.newWindow.title("Settings") self.newWindow.geometry("400x400") self.btn_flip = tk.Button(self.newWindow, text="Mirror Image", width=10, command=self.flip_img) self.btn_flip.pack(side=tk.LEFT, padx=10) # self.brightness_lbl = Label(self.newWindow, text="Image Brightness") # self.brightness_lbl.pack(anchor=NW) #var = IntVar() self.brightness = tk.Scale( self.newWindow, length=200, from_=0, to=255, orient=tk.HORIZONTAL, label="Image Brightness", command=self.slide) self.brightness.set(200) self.brightness.pack(anchor=tk.NW) def slide(self, var): self.slide_value = self.brightness.get() print(self.slide_value) def flip_img(self): self.vid.flipped = not self.vid.flipped def overlay(self): file = filedialog.askopenfile( mode='rb', defaultextension='.png',title="Choose Overlay Image", filetypes=[("PNG Files", '*.png')]) if file: self.overlay_img = ImageTk.PhotoImage(file=file) self.pil_overlay_img = PIL.Image.open(file, mode='r', formats=None) def snapshot(self): isTrue, frame = self.vid.getFrame() if isTrue: filename = filedialog.asksaveasfilename( defaultextension='.jpg', title="Choose Filename", filetypes=[("JPEG Files", '*.jpg')]) # image = "IMG-" + time.strftime("%H-%M-%S-%d-%m") + ".jpg" cv2.imwrite(filename, cv2.cvtColor(frame, cv2.COLOR_BGR2RGB)) # msg = Label(self.window, text='Image Saved' + image, bg='black', fg='green').place(x=430, y=510) else: messagebox.showerror("paint says", "unable to save image ,\n something went wrong") if isTrue and self.pil_overlay_img: filename = filedialog.asksaveasfilename( defaultextension='.jpg', title="Choose Filename", filetypes=[("JPEG Files", '*.jpg')]) filename = self.photo.paste(self.pil_overlay_img, (0,0)) def update(self): isTrue, frame = self.vid.getFrame() frame = cv2.normalize(frame, frame, 0, self.slide_value, cv2.NORM_MINMAX) if isTrue: self.photo = ImageTk.PhotoImage(image=PIL.Image.fromarray(frame)) #self.pil_photo_img = PIL.Image.open(self.photo, mode='r', formats=None) #self.canvas.tag_lower(self.photo) self.canvas.create_image(0, 0, image=self.photo, anchor=tk.NW) if self.overlay_img: #self.canvas.tag_raise(self.overlay_img) self.canvas.create_image(0,0, image=self.overlay_img, anchor=tk.NW) #self.stamped_img = self.photo.paste(self.overlay_img) self.window.after(10, self.update) class MyVideoCapture: def __init__(self, video_source=0): self.vid = cv2.VideoCapture(video_source) if not self.vid.isOpened(): raise ValueError("Unable to open this camera \n select another video source", video_source) self.width = self.vid.get(cv2.CAP_PROP_FRAME_WIDTH) self.height = self.vid.get(cv2.CAP_PROP_FRAME_HEIGHT) self.flipped = True def getFrame(self): if self.vid.isOpened(): isTrue, frame = self.vid.read() if isTrue and self.flipped: frame = cv2.flip(frame, 1) if isTrue: return (isTrue, cv2.cvtColor(frame, cv2.COLOR_BGR2RGB)) else: return (isTrue, None) else: return (isTrue, None) def __del__(self): if self.vid.isOpened(): self.vid.release() if __name__ == "__main__": App() RE: ImageTk Paste - deanhystad - Jun-02-2021 "no joy" is not very useful. Please describe the problem. RE: ImageTk Paste - KDog - Jun-02-2021 The problem is when i go to save/snapshot an image it doesn't save the overlay on top of the video frame. Also once i click save on the file dialog a new file dialog appears. Here are my efforts at PIL: This is in the update function where I attempt to convert the latest frame to PIL image self.photo = ImageTk.PhotoImage(image=PIL.Image.fromarray(frame))This is in the snapshot function where I attempt to paste the overlay onto the latest frame filename = self.photo.paste(self.pil_overlay_img, (0,0)) RE: ImageTk Paste - deanhystad - Jun-03-2021 The snapshot method opens a dialog, and if successful opens a second dialog. It opens two dialogs because you wrote the code that way. Then it proceeds to perform actions that don't make sense. Your problem is not with the image stuff, it is with everything else that has become a real mess because you've been thrashing away on this problem. Time to step back and get a simple example working. I gutted your code, removing the video capture, and pared it down to a program that opens image files and pastes them on an existing image. The pasted image is centered where the cursor was when the mouse button was pressed inside the canvas object. There is a save button to write the resulting image to a file. import tkinter as tk from PIL import Image, ImageTk from tkinter import filedialog filetypes = (('PNG', '*.png'), ('JPEG', '*.jpg'), ('All', '*.*')) class App: def __init__(self): self.image = None self.tk_image = None window = tk.Tk() self.canvas = tk.Canvas(window, width=360, height=360) self.canvas.pack() self.canvas.bind('<Button>', self.open) button = tk.Button(window, text="Save", command=self.save) button.pack(side=tk.LEFT, padx=10) window.mainloop() def open(self, event): """Open image file and paste to existing image""" filename = filedialog.askopenfile(mode='rb', filetypes=filetypes) if filename: # Open image file. This creates a PIL image file object image = Image.open(filename) if self.image is None: # Set initial image self.image = image else: # Paste new image on existing image x = event.x - image.width//2 y = event.y - image.height//2 Image.Image.paste(self.image, image, (x, y)) # Draw image on canvas. Must convert to Tk image first self.tk_image = ImageTk.PhotoImage(self.image) self.canvas.delete(tk.ALL) self.canvas.create_image(180, 180, image=self.tk_image) def save(self): """Save image to a file""" filename = filedialog.asksaveasfilename(filetypes=filetypes) if filename: self.image.save(filename) if __name__ == "__main__": App()I have never used PIL this way and I was surprised to see that paste is a function and not a method. You pass the base and pasted images as arguments. There may be a paste method for ImageTk objects, I haven't looked, but you should. You can use my short example to experiment around with that. When you are trying something new you should write a short example instead of messing up existing code. The test program should be only as long as it has to be to test the idea. A nice side benefit of this type of development is you can keep all these programs in a folder, like a design notebook, and reference them later when you have questions about how to use different Python libraries. The short examples are also great for posting to the forum. RE: ImageTk Paste - KDog - Jun-27-2021 Many thanks for this deanhystad! I will play around with it. Your second suggestion is a great idea, it will simplify things and leave me with a handy library of mini apps. ![]() |