2017-07-27 130 views
1

我正在使用Python GUI w/Tkinter。我試圖將BMP圖像中的四個指定點位置保存爲變量,並創建一個最適合的橢圓,或多或少地通過保存的點。我仍然是一名初學者,工作着W/Tkinter和GUI,所以請耐心等待!Python Tkinter。來自數據點的最佳擬合橢圓

到目前爲止,代碼可以標記點並打印出其位置/座標。我應該在這種情況下使用matplotlib嗎?我能夠使用那個w/tkinter嗎?

這裏是我的代碼:

from tkinter import * 
from PIL import Image, ImageTk 

class Window(Frame): 

    def __init__(self, master=None): 
     Frame.__init__(self, master) 

     self.master = master 
     self.pos = [] 
     self.master.title("GUI") 
     self.pack(fill=BOTH, expand=1) 

     self.counter = 0 

     menu = Menu(self.master) 
     self.master.config(menu=menu) 

     file = Menu(menu) 
     file.add_command(label="Exit", command=self.client_exit) 
     menu.add_cascade(label="File", menu=file) 
     analyze = Menu(menu) 

     analyze.add_command(label="Region of 
     Interest",command=self.regionOfInterest) 
     analyze.add_command(label="Erase", command=self.erasePoints) 

     menu.add_cascade(label="Analyze", menu=analyze) 
     load = Image.open("ap41.ddr.brf.sdat.bmp") 
     render = ImageTk.PhotoImage(load) 

     img = Label(self, image=render) 
     img.image = render 
     img.place(x=0, y=0) 

    def regionOfInterest(self): 
     root.config(cursor="plus") 
     canvas.bind("<Button-1>", self.imgClick) 

    def erasePoints(self): 
     self.pos = [] 

    def client_exit(self): 
     exit() 

    def imgClick(self, event): 

     if self.counter < 4: 
      x = canvas.canvasx(event.x) 
      y = canvas.canvasy(event.y) 
      self.pos.append((x, y)) 
      print(self.pos) 
      canvas.create_line(x - 5, y, x + 5, y, fill="red", 
tags="crosshair") 
      canvas.create_line(x, y - 5, x, y + 5, fill="red", 
tags="crosshair") 
      self.counter += 1 
     else: 
      canvas.unbind("<Button 1>") 
      root.config(cursor="arrow") 
      self.counter = 0 


root = Tk() 
imgSize = Image.open("ap41.ddr.brf.sdat.bmp") 
tkimage = ImageTk.PhotoImage(imgSize) 
w, h = imgSize.size 

canvas = Canvas(root, width=w, height=h) 
canvas.create_image((w/2,h/2),image=tkimage) 
canvas.pack() 

root.geometry("%dx%d"%(w,h)) 
app = Window(root) 
root.mainloop() 
+0

您好再次FLCL。你有沒有嘗試過使用'create_oval()' –

+0

我剛剛更新了我的答案。我有一個需要修復的錯誤。 –

+0

4分是不足以定義一個獨特的橢圓,5是最小的 – f5r5e5d

回答

1

這裏是你可以用和微調玩,但我認爲這將是接近你正在嘗試做的。

首先,我創建了另一個標記爲Create Ellipse的菜單項,它鏈接到計算左上角簾線和右下角簾線的方法,然後將其與create_ovel()命令配合使用以在屏幕上創建橢圓。讓我知道這是否接近你想要做的事情。

下面的新方法會將每個元組的值與基元組進行比較,如果數值較低,它將更改左上角的連線,如果數字較高,則會更改右下角的連線。用這兩組繩索計算出來後,它將創建一個橢圓,以便粗略地適合你所選擇的。

def createEllipse(self): 

    top_left_cords = self.pos[0] 
    bottom_right_cords = self.pos[0] 
    for pos in self.pos: 
     if pos[0] < top_left_cords[0]: 
      top_left_cords = (pos[0], top_left_cords[1]) 

     if pos[1] < top_left_cords[1]: 
      top_left_cords = (top_left_cords[0], pos[1]) 

     if pos[0] > bottom_right_cords[0]: 
      bottom_right_cords = (pos[0], bottom_right_cords[1]) 

     if pos[1] > bottom_right_cords[1]: 
      bottom_right_cords = (bottom_right_cords[0], pos[1]) 

以下是完整代碼:

from tkinter import * 
from PIL import Image, ImageTk 

class Window(Frame): 

    def __init__(self, master=None): 
     Frame.__init__(self, master) 

     self.master = master 
     self.pos = [] 
     self.master.title("GUI") 
     self.pack(fill=BOTH, expand=1) 

     self.counter = 0 

     menu = Menu(self.master) 
     self.master.config(menu=menu) 

     file = Menu(menu) 
     file.add_command(label="Exit", command=self.client_exit) 
     menu.add_cascade(label="File", menu=file) 
     analyze = Menu(menu) 

     analyze.add_command(label="Region of Interest",command=self.regionOfInterest) 
     analyze.add_command(label="Erase", command=self.erasePoints) 
     analyze.add_command(label="Create Ellipse", command=self.createEllipse) 

     menu.add_cascade(label="Analyze", menu=analyze) 
     load = Image.open("./Colors/1.png") 
     render = ImageTk.PhotoImage(load) 


     img = Label(self, image=render) 
     img.image = render 
     img.place(x=0, y=0) 

    def createEllipse(self): 

     top_left_cords = self.pos[0] 
     bottom_right_cords = self.pos[0] 
     for pos in self.pos: 
      if pos[0] < top_left_cords[0]: 
       top_left_cords = (pos[0], top_left_cords[1]) 

      if pos[1] < top_left_cords[1]: 
       top_left_cords = (top_left_cords[0], pos[1]) 

      if pos[0] > bottom_right_cords[0]: 
       bottom_right_cords = (pos[0], bottom_right_cords[1]) 

      if pos[1] > bottom_right_cords[1]: 
       bottom_right_cords = (bottom_right_cords[0], pos[1]) 

     print(top_left_cords, bottom_right_cords) 

     canvas.create_oval(top_left_cords, bottom_right_cords) 

    def regionOfInterest(self): 
     root.config(cursor="plus") 
     canvas.bind("<Button-1>", self.imgClick) 

    def erasePoints(self): 
     self.pos = [] 

    def client_exit(self): 
     exit() 

    def imgClick(self, event): 

     if self.counter < 4: 
      x = canvas.canvasx(event.x) 
      y = canvas.canvasy(event.y) 
      self.pos.append((x, y)) 
      print(self.pos) 
      canvas.create_line(x - 5, y, x + 5, y, fill="red", 
tags="crosshair") 
      canvas.create_line(x, y - 5, x, y + 5, fill="red", 
tags="crosshair") 
      self.counter += 1 
     else: 
      canvas.unbind("<Button 1>") 
      root.config(cursor="arrow") 
      self.counter = 0 


root = Tk() 
imgSize = Image.open("./Colors/1.png") 
tkimage = ImageTk.PhotoImage(imgSize) 
w, h = imgSize.size 

canvas = Canvas(root, width=w, height=h) 
canvas.create_image((w/2,h/2),image=tkimage) 
canvas.pack() 

root.geometry("%dx%d"%(w,h)) 
app = Window(root) 
root.mainloop() 

以下是我對測試樣本圖像的之前和之後。

前:

enter image description here

後:

enter image description here

+0

你好!對於遲到的回覆感到抱歉。感謝您的代碼的詳細運行!不幸的是,當我嘗試運行橢圓時,橢圓不會顯示在帶有標記點的bmp圖像上,但它仍然會打印出top_left_cords&bottom_right_cords的值,所以我確實相信它可行!我有一種感覺,它必須做的帆布或許。 – FLCL

+0

@FLCL嗡嗡聲。那麼它對我有用。我只是將圖像更改爲我可以測試的東西。我在測試的一側添加了一張前後圖像。 –

+0

@FLCL您是否使用新的菜單按鈕來創建橢圓? –