From 8fecb75a3e107a936ecc5bac653b38a3ad5c2cb2 Mon Sep 17 00:00:00 2001 From: "Jorj X. McKie" Date: Tue, 4 Sep 2018 10:25:08 -0400 Subject: [PATCH 01/14] New general Docuemtn viewer with improved zooming --- Demo_DOC_Viewer_PIL.py | 238 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 238 insertions(+) create mode 100644 Demo_DOC_Viewer_PIL.py diff --git a/Demo_DOC_Viewer_PIL.py b/Demo_DOC_Viewer_PIL.py new file mode 100644 index 00000000..ac2c8232 --- /dev/null +++ b/Demo_DOC_Viewer_PIL.py @@ -0,0 +1,238 @@ +""" +@created: 2018-08-19 18:00:00 +@author: (c) 2018 Jorj X. McKie +Display a PyMuPDF Document using Tkinter +------------------------------------------------------------------------------- +Dependencies: +------------- +PyMuPDF, PySimpleGUI (requires Python 3), Tkinter, PIL +License: +-------- +GNU GPL V3+ +Description +------------ +Get filename and start displaying page 1. Please note that all file types +of MuPDF are supported (including EPUB e-books and HTML files for example). +Pages can be directly jumped to, or buttons can be used for paging. + +This version contains enhancements: +* Use of PIL improves response times by a factor 3 or more. +* Zooming is now flexible: only one button serves as a toggle. Arrow keys can + be used for moving the window when zooming. + +We also interpret keyboard events (PageDown / PageUp) and mouse wheel actions +to support paging as if a button was clicked. Similarly, we do not include +a 'Quit' button. Instead, the ESCAPE key can be used, or cancelling the form. +To improve paging performance, we are not directly creating pixmaps from +pages, but instead from the fitz.DisplayList of the page. A display list +will be stored in a list and looked up by page number. This way, zooming +pixmaps and page re-visits will re-use a once-created display list. + +""" +import sys +import fitz +import PySimpleGUI as sg +import tkinter as tk +from PIL import Image, ImageTk +import time + +if len(sys.argv) == 1: + rc, fname = sg.GetFileBox('Document Browser', 'Document file to open', + file_types = ( + ("PDF Files", "*.pdf"), + ("XPS Files", "*.*xps"), + ("Epub Files", "*.epub"), + ("Fiction Books", "*.fb2"), + ("Comic Books", "*.cbz"), + ("HTML", "*.htm*"), + # add more document types here + ) + ) +else: + fname = sys.argv[1] + +if not fname: + sg.MsgBox("Cancelling:", "No filename supplied") + raise SystemExit("Cancelled: no filename supplied") + +doc = fitz.open(fname) +page_count = len(doc) + +# used for response time statistics only +fitz_img_time = 0.0 +tk_img_time = 0.0 +img_count = 1 + +# allocate storage for page display lists +dlist_tab = [None] * page_count + +title = "PyMuPDF display of '%s', pages: %i" % (fname, page_count) + +def get_page(pno, zoom = False, max_size = None, first = False): + """Return a PNG image for a document page number. + """ + dlist = dlist_tab[pno] # get display list of page number + if not dlist: # create if not yet there + dlist_tab[pno] = doc[pno].getDisplayList() + dlist = dlist_tab[pno] + r = dlist.rect # the page rectangle + clip = r + # ensure image fits screen: + # exploit, but do not exceed width or height + zoom_0 = 1 + if max_size: + zoom_0 = min(1, max_size[0] / r.width, max_size[1] / r.height) + if zoom_0 == 1: + zoom_0 = min(max_size[0] / r.width, max_size[1] / r.height) + mat_0 = fitz.Matrix(zoom_0, zoom_0) + + if not zoom: # show total page + pix = dlist.getPixmap(matrix = mat_0, alpha=False) + else: + mp = r.tl + (r.br - r.tl) * 0.5 # page rect center + w2 = r.width / 2 + h2 = r.height / 2 + clip = r * 0.5 + tl = zoom[0] # old top-left + tl.x += zoom[1] * (w2 / 2) + tl.x = max(0, tl.x) + tl.x = min(w2, tl.x) + tl.y += zoom[2] * (h2 / 2) + tl.y = max(0, tl.y) + tl.y = min(h2, tl.y) + clip = fitz.Rect(tl, tl.x + w2, tl.y + h2) + + mat = mat_0 * fitz.Matrix(2, 2) # zoom matrix + pix = dlist.getPixmap(alpha=False, matrix=mat, clip=clip) + + if first: # first call: tkinter still inactive + img = pix.getPNGData() # so use fitz png output + else: # else take tk photo image + pilimg = Image.frombytes("RGB", [pix.width, pix.height], pix.samples) + img = ImageTk.PhotoImage(pilimg) + + return img, clip.tl # return image, clip position + + +root = tk.Tk() +max_width = root.winfo_screenwidth() - 20 +max_height = root.winfo_screenheight() - 135 +max_size = (max_width, max_height) +root.destroy() +del root + +form = sg.FlexForm(title, return_keyboard_events = True, + location = (0,0), use_default_focus = False) + +cur_page = 0 +data, clip_pos = get_page(cur_page, + zoom = False, + max_size = max_size, + first = True) + +image_elem = sg.Image(data = data) + +goto = sg.InputText(str(cur_page + 1), size=(5, 1), do_not_clear=True, + key = "PageNumber") + +layout = [ + [ + sg.ReadFormButton('Next'), + sg.ReadFormButton('Prev'), + sg.Text('Page:'), + goto, + sg.Text('(%i)' % page_count), + sg.ReadFormButton('Zoom'), + sg.Text('(toggle on/off, use arrows to navigate while zooming)'), + ], + [image_elem], +] + +form.Layout(layout) + +# now define the buttons / events we want to handle +enter_buttons = [chr(13), "Return:13"] +quit_buttons = ["Escape:27", chr(27)] +next_buttons = ["Next", "Next:34", "MouseWheel:Down"] +prev_buttons = ["Prev", "Prior:33", "MouseWheel:Up"] +Up = "Up:38" +Left = "Left:37" +Right = "Right:39" +Down = "Down:40" +zoom_buttons = ["Zoom", Up, Down, Left, Right] + +# all the buttons we will handle +my_keys = enter_buttons + next_buttons + prev_buttons + zoom_buttons + +# old page store and zoom toggle +old_page = 0 +old_zoom = False + +while True: + button, value = form.Read() + if button is None and (value is None or value['PageNumber'] is None): + break + if button in quit_buttons: + break + + zoom_pressed = False + zoom = False + + if button in enter_buttons: + try: + cur_page = int(value['PageNumber']) - 1 # check if valid + while cur_page < 0: + cur_page += page_count + except: + cur_page = 0 # this guy's trying to fool me + + elif button in next_buttons: + cur_page += 1 + elif button in prev_buttons: + cur_page -= 1 + elif button == Up: + zoom = (clip_pos, 0, -1) + elif button == Down: + zoom = (clip_pos, 0, 1) + elif button == Left: + zoom = (clip_pos, -1, 0) + elif button == Right: + zoom = (clip_pos, 1, 0) + elif button == "Zoom": + zoom_pressed = True + zoom = (clip_pos, 0, 0) + + # sanitize page number + if cur_page >= page_count: # wrap around + cur_page = 0 + while cur_page < 0: # pages > 0 look nicer + cur_page += page_count + + if zoom_pressed and old_zoom: + zoom = zoom_pressed = old_zoom = False + + t0 = time.perf_counter() + data, clip_pos = get_page(cur_page, zoom = zoom, max_size = max_size, + first = False) + t1 = time.perf_counter() + image_elem.Update(data = data) + t2 = time.perf_counter() + fitz_img_time += t1 - t0 + tk_img_time += t2 - t1 + img_count += 1 + old_page = cur_page + old_zoom = zoom_pressed or zoom + + # update page number field + if button in my_keys: + goto.Update(str(cur_page + 1)) + + +# print some response time statistics +if img_count > 0: + print("response times for '%s'" % doc.name) + print("%.4f" % (fitz_img_time/img_count), "sec fitz avg. image time") + print("%.4f" % (tk_img_time/img_count), "sec tk avg. image time") + print("%.4f" % ((fitz_img_time + tk_img_time)/img_count), "sec avg. total time") + print(img_count, "images read") + print(page_count, "pages") From 38a18042eaa5e8b1934d665923cf640635b91f53 Mon Sep 17 00:00:00 2001 From: fluxrider Date: Sun, 9 Sep 2018 19:27:29 -0400 Subject: [PATCH 02/14] Listbox: in Update(), point to the new values. --- PySimpleGUI.py | 1 + 1 file changed, 1 insertion(+) diff --git a/PySimpleGUI.py b/PySimpleGUI.py index c9a00233..155e5e1d 100644 --- a/PySimpleGUI.py +++ b/PySimpleGUI.py @@ -420,6 +420,7 @@ class Listbox(Element): for item in values: self.TKListbox.insert(tk.END, item) self.TKListbox.selection_set(0, 0) + self.Values = values def SetValue(self, values): for index, item in enumerate(self.Values): From bd97141255407a421bd6cfccdde5e8cb7df28c30 Mon Sep 17 00:00:00 2001 From: eagleEggs <29800532+eagleEggs@users.noreply.github.com> Date: Mon, 24 Sep 2018 21:37:54 -0400 Subject: [PATCH 03/14] Form2 call fix Corrected the 'ShowTabbedForm() argument which was referencing a 'form' variable which should be 'form2' as it's the declared Window variable for this script. --- Demo_Tabbed_Form.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Demo_Tabbed_Form.py b/Demo_Tabbed_Form.py index 60de7de6..f902152a 100644 --- a/Demo_Tabbed_Form.py +++ b/Demo_Tabbed_Form.py @@ -76,8 +76,8 @@ layout_tab_2.append([sg.InputText(size=(100,1), default_text='gruen -sara -quarz layout_tab_2.append([sg.Text('_' * 100, size=(75, 1))]) layout_tab_2.append([sg.ReadButton('Submit', button_color=('red', 'yellow')), sg.Cancel(button_color=('white', 'blue'))]) -results = sg.ShowTabbedForm('eBay Super Searcher', (form,layout_tab_1,'Where To Save'), (form2, layout_tab_2, 'Categories & Search String')) +results = sg.ShowTabbedForm('eBay Super Searcher', (form2, layout_tab_1,'Where To Save'), (form2, layout_tab_2, 'Categories & Search String')) -sg.Popup('Results', results) \ No newline at end of file +sg.Popup('Results', results) From 0c290058ddee384390ed2f910656c2f1506b980a Mon Sep 17 00:00:00 2001 From: MikeTheWatchGuy Date: Tue, 25 Sep 2018 17:51:44 -0400 Subject: [PATCH 04/14] Delete Demo_Tabbed_Form.py --- Demo_Tabbed_Form.py | 83 --------------------------------------------- 1 file changed, 83 deletions(-) delete mode 100644 Demo_Tabbed_Form.py diff --git a/Demo_Tabbed_Form.py b/Demo_Tabbed_Form.py deleted file mode 100644 index f902152a..00000000 --- a/Demo_Tabbed_Form.py +++ /dev/null @@ -1,83 +0,0 @@ -import PySimpleGUI as sg - -# Drop Down list of options -configs = ('0 - Gruen - Started 2 days ago in Watches', - '1 - Gruen - Currently Active in Watches', - '2 - Alpina - Currently Active in Jewelry', - '3 - Gruen - Ends in 1 day in Watches', - '4 - Gruen - Completed in Watches', - '5 - Gruen - Advertising', - '6 - Gruen - Currently Active in Jewelry', - '7 - Gruen - Price Test', - '8 - Gruen - No brand name specified') - -us_categories = ('Use Default with no change', - 'All - 1', - 'Jewelry - 281', - ' Watches - 14324', - ' Wristwatches - 31387', - ' Pocket Watches - 3937', - 'Advertising - 34', - ' Watch Ads - 165254' - ) - -german_categories =('Use Default with no change', - 'All - 1', - 'Jewelry - 281', - ' Watches - 14324', - ' Wristwatches - 31387', - ' Pocket Watches - 3937', - 'Advertising - 1', - ' Watch Ads - 19823' - ) - - -# the form layout -window = sg.Window('EBay Super Searcher', auto_size_text=True) - -form2 = sg.Window('EBay Super Searcher', auto_size_text=False) - -layout_tab_1 = [[sg.Text('eBay Super Searcher!', size=(60,1), font=('helvetica', 15))], - [sg.Text('Choose base configuration to run')], - [sg.InputCombo(configs)], - [sg.Text('_'*100, size=(80,1))], - [sg.InputText(),sg.Text('Choose Destination Folder'), sg.FolderBrowse(target=(sg.ThisRow,0))], - [sg.InputText(),sg.Text('Custom text to add to folder name')], - [sg.Text('_'*100, size=(80,1))], - [sg.Checkbox('US', default=True, size=(15, 1)), sg.Checkbox('German', size=(15, 1), default=True, )], - [sg.Radio('Active Listings','ActiveComplete', default = True,size=(15, 1)), sg.Radio('Completed Listings', 'ActiveComplete', size=(15, 1))], - [sg.Text('_'*100, size=(80,1))], - [sg.Checkbox('Save Images', size=(15,1)),sg.Checkbox('Save PDFs', size=(15,1)), sg.Checkbox('Extract PDFs', size=(15,1))], - [sg.Text('_'*100, size=(80,1))], - [sg.Text('Time Filters')], - [sg.Radio('No change','time', default=True),sg.Radio('ALL listings','time'),sg.Radio('Started 1 day ago','time', size=(15,1)),sg.Radio('Started 2 days ago','time', size=(15,1)), sg.Radio('Ends in 1 day','time', size=(15,1))], - [sg.Text('_'*100, size=(80,1))], - [sg.Text('Price Range'), sg.InputText(size=(10,1)),sg.Text('To'), sg.InputText(size=(10,1))], - [sg.Text('_'*100, size=(80,1))], - [sg.Submit(button_color=('red', 'yellow')), sg.Cancel(button_color=('white', 'blue'))]] - - -# First category is default (need to special case this) -layout_tab_2 = [[sg.Text('Choose Category')], - [sg.Text('US Categories'),sg.Text('German Categories')], - [sg.Radio(us_categories[0],'CATUS', default=True), sg.Radio(german_categories[0], 'CATDE', default=True)]] - -for i,cat in enumerate(us_categories): - if i == 0: continue # skip first one - layout_tab_2.append([sg.Radio(cat,'CATUS'), sg.Radio(german_categories[i],'CATDE')]) - -layout_tab_2.append([sg.Text('_' * 100, size=(75, 1))]) -layout_tab_2.append([sg.Text('US Search String Override')]) -layout_tab_2.append([sg.InputText(size=(100,1))]) -layout_tab_2.append([sg.Text('German Search String Override')]) -layout_tab_2.append([sg.InputText(size=(100,1))]) -layout_tab_2.append([sg.Text('Typical US Search String')]) -layout_tab_2.append([sg.InputText(size=(100,1), default_text='gruen -sara -quarz -quartz -embassy -bob -robert -elephants -adidas -LED ')]) -layout_tab_2.append([sg.Text('_' * 100, size=(75, 1))]) -layout_tab_2.append([sg.ReadButton('Submit', button_color=('red', 'yellow')), sg.Cancel(button_color=('white', 'blue'))]) - -results = sg.ShowTabbedForm('eBay Super Searcher', (form2, layout_tab_1,'Where To Save'), (form2, layout_tab_2, 'Categories & Search String')) - - - -sg.Popup('Results', results) From 2501c150e5d21ebc94d40b5565f5b489ad0cd3c0 Mon Sep 17 00:00:00 2001 From: MikeTheWatchGuy Date: Tue, 25 Sep 2018 20:03:14 -0400 Subject: [PATCH 05/14] Doc updates to match Python 2.7 support --- docs/cookbook.md | 53 +++++++++++++++++++++++++++++++++++++++++++----- docs/index.md | 24 ++++++++++++++-------- docs/tutorial.md | 17 ++++++++++++++-- readme.md | 24 ++++++++++++++-------- 4 files changed, 95 insertions(+), 23 deletions(-) diff --git a/docs/cookbook.md b/docs/cookbook.md index a171d06a..cf7477de 100644 --- a/docs/cookbook.md +++ b/docs/cookbook.md @@ -6,6 +6,17 @@ You'll find that starting with a Recipe will give you a big jump-start on creating your custom GUI. Copy and paste one of these Recipes and modify it to match your requirements. Study them to get an idea of what design patterns to follow. +The Recipes in this Cookbook all assume you're running on a Python3 machine. If you are running Python 2.7 then your code will differ by 2 character. Replace the import statement: + + import PySimpleGUI as sg + +with + + import PySimpleGUI27 as sg + +There is a short section in the Readme with instruction on installing PySimpleGUI + + ## Simple Data Entry - Return Values As List Same GUI screen except the return values are in a list instead of a dictionary and doesn't have initial values. @@ -42,7 +53,7 @@ A simple GUI with default values. Results returned in a dictionary. [sg.Text('Name', size=(15, 1)), sg.InputText('name', key='name')], [sg.Text('Address', size=(15, 1)), sg.InputText('address', key='address')], [sg.Text('Phone', size=(15, 1)), sg.InputText('phone', key='phone')], - [sg.Submit(), sg.Cancel()] + [sg.Submit(), sg.Cancel()] ] window = sg.Window('Simple data entry GUI').Layout(layout) @@ -302,7 +313,7 @@ This recipe shows just how easy it is to add a progress meter to your code. import PySimpleGUI as sg for i in range(1000): - sg.OneLineProgressMeter('One Line Meter Example', i+1, 1000, 'mymeter') + sg.OneLineProgressMeter('One Line Meter Example', i+1, 1000, 'key') ----- @@ -530,9 +541,10 @@ you can write this line of code for the exact same result (OK, two lines with th -------------------- ## Multiple Columns -Starting in version 2.9 you can use the Column Element. A Column is required when you have a tall element to the left of smaller elements. -This example uses a Column. There is a Listbox on the left that is 3 rows high. To the right of it are 3 single rows of text and input. These 3 rows are in a Column Element. +A Column is required when you have a tall element to the left of smaller elements. + +In this example, there is a Listbox on the left that is 3 rows high. To the right of it are 3 single rows of text and input. These 3 rows are in a Column Element. To make it easier to see the Column in the window, the Column background has been shaded blue. The code is wordier than normal due to the blue shading. Each element in the column needs to have the color set to match blue background. @@ -1222,6 +1234,37 @@ In this example we're defining our graph to be from -100, -100 to +100,+100. Th button, values = window.Read() +## Tabs + +Tabs bring not only an extra level of sophistication to your window layout, they give you extra room to add more elements. Tabs are one of the 3 container Elements, Elements that hold or contain other Elements. The other two are the Column and Frame Elements. + + +![tabs](https://user-images.githubusercontent.com/13696193/46049479-97732f00-c0fc-11e8-8015-5bbed8bd88bb.jpg) + + + +``` +import PySimpleGUI as sg + +tab1_layout = [[sg.T('This is inside tab 1')]] + +tab2_layout = [[sg.T('This is inside tab 2')], + [sg.In(key='in')]] + +layout = [[sg.TabGroup([[sg.Tab('Tab 1', tab1_layout, tooltip='tip'), sg.Tab('Tab 2', tab2_layout)]], tooltip='TIP2')], + [sg.RButton('Read')]] + +window = sg.Window('My window with tabs', default_element_size=(12,1)).Layout(layout) + +while True: + button, v = window.Read() + print(button,values) + if button is None: # always, always give a way out! + break +``` + + + ## Creating a Windows .EXE File @@ -1244,4 +1287,4 @@ That's all... Run your `my_program.exe` file on the Windows machine of your choo > (famous last words that screw up just about anything being referenced) -Your EXE file should run without creating a "shell window". Only the GUI window should show up on your taskbar. \ No newline at end of file +Your EXE file should run without creating a "shell window". Only the GUI window should show up on your taskbar. diff --git a/docs/index.md b/docs/index.md index 67474a73..4aa63cf8 100644 --- a/docs/index.md +++ b/docs/index.md @@ -4,12 +4,21 @@ ![pysimplegui_logo](https://user-images.githubusercontent.com/13696193/43165867-fe02e3b2-8f62-11e8-9fd0-cc7c86b11772.png) -[![Downloads](http://pepy.tech/badge/pysimplegui)](http://pepy.tech/project/pysimplegui) ![Documentation Status](https://readthedocs.org/projects/pysimplegui/badge/?version=latest) [![Python Version](https://img.shields.io/badge/Python-3-brightgreen.svg)](https://www.python.org/downloads/) +[![Downloads](http://pepy.tech/badge/pysimplegui)](http://pepy.tech/project/pysimplegui) ![Documentation Status](https://readthedocs.org/projects/pysimplegui/badge/?version=latest) +![Awesome Meter](https://img.shields.io/badge/Awesomeness_Rating-100%-yellow.svg) + ![Python Version](https://img.shields.io/badge/Python-3-yellow.svg) ![Python Version](https://img.shields.io/badge/Python-2.7-yellow.svg) + + + + # PySimpleGUI -![Python Version](https://img.shields.io/badge/PySimpleGUI_For_Python_3_Version-3.8.0-red.svg?longCache=true&style=for-the-badge) + +## Now supports both Python 2.7 & 3 + +![Python Version](https://img.shields.io/badge/PySimpleGUI_For_Python_3_Version-3.8.1-red.svg?longCache=true&style=for-the-badge) ![Python Version](https://img.shields.io/badge/PySimpleGUI_For_Python_2.7_Version-1.0.0-blue.svg?longCache=true&style=for-the-badge) @@ -21,8 +30,6 @@ [Brief Tutorial](https://pysimplegui.readthedocs.io/en/latest/tutorial/) -[OpenSource Article](https://opensource.com/article/18/8/pysimplegui) - [Latest Demos and Master Branch on GitHub](https://github.com/MikeTheWatchGuy/PySimpleGUI) Super-simple GUI to use... Powerfully customizable. @@ -30,16 +37,17 @@ Super-simple GUI to use... Powerfully customizable. Home of the 1-line custom GUI and 1-line progress meter #### Note regarding Python versions -As of 9/25/2018 **both Python 3 and Python 2.7 are supported**! The Python 3 version is called PySimpleGUI. The Python 2.7 version is called PySimpleGUI27. They are installed separately and the imports are different. See instructions in Installation section for more info. +As of 9/25/2018 **both Python 3 and Python 2.7 are supported**! The Python 3 version is named PySimpleGUI. The Python 2.7 version is named PySimpleGUI27. They are installed separately and the imports are different. See instructions in Installation section for more info. ---------------------------------- +------------------------------------------------------------------------ -Looking for a GUI package to help with + +Looking for a GUI package? * Taking your Python code from the world of command lines and into the convenience of a GUI? * * Have a Raspberry **Pi** with a touchscreen that's going to waste because you don't have the time to learn a GUI SDK? * Into Machine Learning and are sick of the command line? -* How about distributing your Python code to Windows users as a single .EXE file that launches straight into a GUI, much like a WinForms app? +* Would like to distribute your Python code to Windows users as a single .EXE file that launches straight into a GUI, much like a WinForms app? Look no further, **you've found your GUI package**. diff --git a/docs/tutorial.md b/docs/tutorial.md index 9526216f..1d3e7729 100644 --- a/docs/tutorial.md +++ b/docs/tutorial.md @@ -2,7 +2,7 @@ # Add GUIs to your programs and scripts easily with PySimpleGUI -NOTE -- PySimpleGUI requires Python3. If you're running Python2, don't waste your time reading. +PySimpleGUI now supports BOTH Python 2.7 and Python 3 ## Introduction Few people run Python programs by double clicking the .py file as if it were a .exe file. When a typical user (non-programmer types) double clicks an exe file, they expect it to pop open with a window they can interact with. While GUIs, using tkinter, are possible using standard Python installations, it's unlikely many programs do this. @@ -46,7 +46,7 @@ There are more complex GUIs such as those that don't close after a button is cli When is PySimpleGUI useful? ***Immediately***, anytime you've got a GUI need. It will take under 5 minutes for you to create and try your GUI. With those kinds of times, what do you have to lose trying it? The best way to go about making your GUI in under 5 minutes is to copy one of the GUIs from the [PySimpleGUI Cookbook](https://pysimplegui.readthedocs.io/en/latest/cookbook/). Follow these steps: -* Install PySimpleGUI +* Install PySimpleGUI (see short section in readme on installation) * Find a GUI that looks similar to what you want to create * Copy code from Cookbook * Paste into your IDE and run @@ -78,6 +78,19 @@ It's a reasonably sized window. If you only need to collect a few values and they're all basically strings, then you would copy this recipe and modify it to suit your needs. +### Python 2.7 Differences + +The only noticeable difference between PySimpleGUI code running under Python 2.7 and one running on Python 3 is the import statement. + +Python 3.x: + + import PySimpleGUI as sg + +Python 2.7: + + import PySimpleGUI27 as sg + + ## The 5-line GUI Not all GUIs take 5 minutes. Some take 5 lines of code. This is a GUI with a custom layout contained in 5 lines of code. diff --git a/readme.md b/readme.md index 67474a73..4aa63cf8 100644 --- a/readme.md +++ b/readme.md @@ -4,12 +4,21 @@ ![pysimplegui_logo](https://user-images.githubusercontent.com/13696193/43165867-fe02e3b2-8f62-11e8-9fd0-cc7c86b11772.png) -[![Downloads](http://pepy.tech/badge/pysimplegui)](http://pepy.tech/project/pysimplegui) ![Documentation Status](https://readthedocs.org/projects/pysimplegui/badge/?version=latest) [![Python Version](https://img.shields.io/badge/Python-3-brightgreen.svg)](https://www.python.org/downloads/) +[![Downloads](http://pepy.tech/badge/pysimplegui)](http://pepy.tech/project/pysimplegui) ![Documentation Status](https://readthedocs.org/projects/pysimplegui/badge/?version=latest) +![Awesome Meter](https://img.shields.io/badge/Awesomeness_Rating-100%-yellow.svg) + ![Python Version](https://img.shields.io/badge/Python-3-yellow.svg) ![Python Version](https://img.shields.io/badge/Python-2.7-yellow.svg) + + + + # PySimpleGUI -![Python Version](https://img.shields.io/badge/PySimpleGUI_For_Python_3_Version-3.8.0-red.svg?longCache=true&style=for-the-badge) + +## Now supports both Python 2.7 & 3 + +![Python Version](https://img.shields.io/badge/PySimpleGUI_For_Python_3_Version-3.8.1-red.svg?longCache=true&style=for-the-badge) ![Python Version](https://img.shields.io/badge/PySimpleGUI_For_Python_2.7_Version-1.0.0-blue.svg?longCache=true&style=for-the-badge) @@ -21,8 +30,6 @@ [Brief Tutorial](https://pysimplegui.readthedocs.io/en/latest/tutorial/) -[OpenSource Article](https://opensource.com/article/18/8/pysimplegui) - [Latest Demos and Master Branch on GitHub](https://github.com/MikeTheWatchGuy/PySimpleGUI) Super-simple GUI to use... Powerfully customizable. @@ -30,16 +37,17 @@ Super-simple GUI to use... Powerfully customizable. Home of the 1-line custom GUI and 1-line progress meter #### Note regarding Python versions -As of 9/25/2018 **both Python 3 and Python 2.7 are supported**! The Python 3 version is called PySimpleGUI. The Python 2.7 version is called PySimpleGUI27. They are installed separately and the imports are different. See instructions in Installation section for more info. +As of 9/25/2018 **both Python 3 and Python 2.7 are supported**! The Python 3 version is named PySimpleGUI. The Python 2.7 version is named PySimpleGUI27. They are installed separately and the imports are different. See instructions in Installation section for more info. ---------------------------------- +------------------------------------------------------------------------ -Looking for a GUI package to help with + +Looking for a GUI package? * Taking your Python code from the world of command lines and into the convenience of a GUI? * * Have a Raspberry **Pi** with a touchscreen that's going to waste because you don't have the time to learn a GUI SDK? * Into Machine Learning and are sick of the command line? -* How about distributing your Python code to Windows users as a single .EXE file that launches straight into a GUI, much like a WinForms app? +* Would like to distribute your Python code to Windows users as a single .EXE file that launches straight into a GUI, much like a WinForms app? Look no further, **you've found your GUI package**. From bd7cd64cee5f1ddc28e8e24e67b0aba393a4b3a3 Mon Sep 17 00:00:00 2001 From: jackyOO7 <44204857+jackyOO7@users.noreply.github.com> Date: Tue, 16 Oct 2018 17:26:26 +0200 Subject: [PATCH 06/14] Colour Image OpenCV->PySimpleGUI I think I found a nicer way to do it. PySimpleGUI is really neat, thank you for your hard work! Cheers jacky --- Demo_OpenCV_Webcam.py | 23 ++++++----------------- 1 file changed, 6 insertions(+), 17 deletions(-) diff --git a/Demo_OpenCV_Webcam.py b/Demo_OpenCV_Webcam.py index c21eaea0..006c2c47 100644 --- a/Demo_OpenCV_Webcam.py +++ b/Demo_OpenCV_Webcam.py @@ -4,10 +4,8 @@ if sys.version_info[0] >= 3: import PySimpleGUI as sg else: import PySimpleGUI27 as sg -import cv2 as cv -from PIL import Image +import cv2 import numpy as np -import io from sys import exit as exit """ @@ -31,7 +29,7 @@ def main(): window.Layout(layout).Finalize() # ---===--- Event LOOP Read and display frames, operate the GUI --- # - cap = cv.VideoCapture(0) + cap = cv2.VideoCapture(0) recording = False while True: event, values = window.ReadNonBlocking() @@ -42,10 +40,8 @@ def main(): recording = True elif event == 'Stop': recording = False - img = Image.new('RGB', (640, 480), (255, 255, 255)) - bio = io.BytesIO() # a binary memory resident stream - img.save(bio, format='PNG') # save image as png to it - imgbytes = bio.getvalue() + img = np.full((480, 640),255) + imgbytes=cv2.imencode('.png', img)[1].tobytes() #this is faster, shorter and needs less includes window.FindElement('image').Update(data=imgbytes) elif event == 'About': sg.PopupNoWait('Made with PySimpleGUI', @@ -56,14 +52,7 @@ def main(): keep_on_top=True) if recording: ret, frame = cap.read() - - gray = cv.cvtColor(frame, cv.COLOR_BGR2GRAY) - - # let img be the PIL image - img = Image.fromarray(gray) # create PIL image from frame - bio = io.BytesIO() # a binary memory resident stream - img.save(bio, format= 'PNG') # save image as png to it - imgbytes = bio.getvalue() # this can be used by OpenCV hopefully + imgbytes=cv2.imencode('.png', cv2.cvtColor(frame,cv2.COLOR_BGR2RGB))[1].tobytes() #ditto window.FindElement('image').Update(data=imgbytes) -main() \ No newline at end of file +main() From 03ef2aae4ee6ced6598600fa7cdcc4e9a21b3ea1 Mon Sep 17 00:00:00 2001 From: jackyOO7 <44204857+jackyOO7@users.noreply.github.com> Date: Wed, 17 Oct 2018 07:38:53 +0200 Subject: [PATCH 07/14] Colour fix fixed the colours. cvtColor is not needed here as openCV takes care of the BGR issue when saving to PNG. --- Demo_OpenCV_Webcam.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Demo_OpenCV_Webcam.py b/Demo_OpenCV_Webcam.py index 006c2c47..c6fb89c3 100644 --- a/Demo_OpenCV_Webcam.py +++ b/Demo_OpenCV_Webcam.py @@ -52,7 +52,7 @@ def main(): keep_on_top=True) if recording: ret, frame = cap.read() - imgbytes=cv2.imencode('.png', cv2.cvtColor(frame,cv2.COLOR_BGR2RGB))[1].tobytes() #ditto + imgbytes=cv2.imencode('.png', frame)[1].tobytes() #ditto window.FindElement('image').Update(data=imgbytes) main() From f17f7722b730002a70045bf0dc90b063403d29ff Mon Sep 17 00:00:00 2001 From: jackyOO7 <44204857+jackyOO7@users.noreply.github.com> Date: Fri, 19 Oct 2018 15:04:21 +0200 Subject: [PATCH 08/14] Create Demo_Touch_Keyboard I needed a simple keyboard to use the GUI on a touchscreen system without keyboard. Turned out to be quite tricky and I do not think that this is the best way to do it. But in principle it works. I didn't manage to get it working with Multilines and Comboboxes though. Maybe someone with better understanding of PySimpleGUI and tkinter can offer a more universal and elegant way... --- Demo_Touch_Keyboard | 84 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 84 insertions(+) create mode 100644 Demo_Touch_Keyboard diff --git a/Demo_Touch_Keyboard b/Demo_Touch_Keyboard new file mode 100644 index 00000000..ac8eebed --- /dev/null +++ b/Demo_Touch_Keyboard @@ -0,0 +1,84 @@ +import PySimpleGUI as sg + +class keyboard(): + def __init__(self,font=('Arial',16)): + self.font=font + numberRow='1234567890' + topRow='QWERTYUIOP' + midRow='ASDFGHJKL' + bottomRow='ZXCVBNM' + keyboard_layout=[[sg.ReadButton(c, key=c, pad=(0, 0), size=(4, 2), font=self.font) for c in numberRow]+[sg.ReadButton('⌫', key='back', pad=(0, 0), size=(4, 2), font=self.font),sg.ReadButton('Esc', key='close', pad=(0, 0), size=(4, 2), font=self.font)], + [sg.T(' '*4)]+[sg.ReadButton(c, key=c, pad=(0, 0), size=(4, 2), font=self.font) for c in topRow], + [sg.T(' '*11)]+[sg.ReadButton(c, key=c, pad=(0, 0), size=(4, 2), font=self.font) for c in midRow], + [sg.T(' '*18)]+[sg.ReadButton(c, key=c, pad=(0, 0), size=(4, 2), font=self.font) for c in bottomRow]] + + self.window=sg.Window('keyboard', grab_anywhere=True, keep_on_top=True, no_titlebar=True).Layout( + keyboard_layout).Finalize() + self.hide() + + def _keyboardhandler(self): + if self.event is not None: + if self.event=='close': + self.hide() + elif len(self.event)==1: + self.focus.Update(self.focus.Get()+self.event) + elif self.event=='back': + Text=self.focus.Get() + if len(Text)>0: + Text=Text[:-1] + self.focus.Update(Text) + + def hide(self): + self.visible=False + self.window.Disappear() + + def show(self): + self.visible=True + self.window.Reappear() + + def togglevis(self): + if self.visible: + self.hide() + else: + self.show() + + def update(self,focus): + self.event,_=self.window.ReadNonBlocking() + self.focus=focus + self._keyboardhandler() + + def close(self): + self.window.CloseNonBlocking() + + +class GUI(): + def __init__(self): + layout = [[sg.Text('Enter Text')], + [sg.Input(size=(17, 1), key='input1',)], + [sg.InputText(size=(17, 1), key='input2')], + [sg.ReadButton('on-screen keyboard',key='keyboard')], + [sg.ReadButton('close',key='close')]] + + self.mainWindow = sg.Window('On-screen test', grab_anywhere=False,no_titlebar=True).Layout(layout) + self.keyboard=keyboard() + self.focus=None + + + def run(self): + while True: + for row in self.mainWindow.Rows: + for element in row: + if element.Type=='input' and element.TKEntry is not None and element.TKEntry is element.TKEntry.focus_get(): + self.focus=element + event, values=self.mainWindow.ReadNonBlocking() + self.keyboard.update(self.focus) + if event=='keyboard': + self.keyboard.togglevis() + elif event=='close': + break + self.keyboard.close() + self.mainWindow.CloseNonBlocking() + +if __name__ == '__main__': + app=GUI() + app.run() From 07819c15a6fdfbc17a60da29edc6fc8981c7357d Mon Sep 17 00:00:00 2001 From: MikeTheWatchGuy Date: Fri, 19 Oct 2018 13:56:36 -0400 Subject: [PATCH 09/14] Rename Demo_Touch_Keyboard to Demo_Touch_Keyboard.py --- Demo_Touch_Keyboard => Demo_Touch_Keyboard.py | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename Demo_Touch_Keyboard => Demo_Touch_Keyboard.py (100%) diff --git a/Demo_Touch_Keyboard b/Demo_Touch_Keyboard.py similarity index 100% rename from Demo_Touch_Keyboard rename to Demo_Touch_Keyboard.py From d75044f702c5c6a646a5907258df84a38ac2d119 Mon Sep 17 00:00:00 2001 From: MikeTheWatchGuy Date: Sun, 21 Oct 2018 21:34:40 -0400 Subject: [PATCH 10/14] Add files via upload --- Demo_LED_Clock_Weather.py | 140 ++++++++++++++++++++++++++++++++++++++ 1 file changed, 140 insertions(+) create mode 100644 Demo_LED_Clock_Weather.py diff --git a/Demo_LED_Clock_Weather.py b/Demo_LED_Clock_Weather.py new file mode 100644 index 00000000..404262ef --- /dev/null +++ b/Demo_LED_Clock_Weather.py @@ -0,0 +1,140 @@ +#!/usr/bin/env python +import sys +if sys.version_info[0] >= 3: + import PySimpleGUI as sg +else: + import PySimpleGUI27 as sg + sg.PopupError('This program uses Base64 images which are not supported in Python 2.7') + sys.exit() +import datetime +import calendar +import forecastio + +NUM_COLS = 5 + +def led_clock(): + def update_weather(): + api_key = "bb9b798d340e413869ea72ac1bea9742" # *** INSERT YOUR DARKSKY KEY HERE ** + lat = 35.738387 + lng = -79.203797 + forecast = forecastio.load_forecast(api_key, lat, lng) + daily = forecast.daily() + max_temps = [] + min_temps = [] + daily_icons = [] + for daily_data in daily.data: + daily_icons.append(daily_data.d['icon']) + max_temps.append(int(daily_data.d['temperatureMax'])) + min_temps.append(int(daily_data.d['temperatureMin'])) + + for i in range(NUM_COLS): + max_element = window.FindElement('_high_' + str(i)) + min_element = window.FindElement('_low_' + str(i)) + icon_element = window.FindElement('_icon_' + str(i)) + max_element.Update(max_temps[i]) + min_element.Update(min_temps[i]) + icon_element.Update(data=weather_icon_dict[daily_icons[i]][22:]) + + weather_icon_dict ={'clear-day':w1, 'clear-night':w1, 'rain':w3, 'snow':w3, 'sleet':w3, 'wind':w3, 'fog':w3, 'cloudy':w4, 'partly-cloudy-day':w5, 'partly-cloudy-night':w5} + + led_digits = [led0, led1, led2, led3, led4, led5, led6, led7, led8, led9] + sg.SetOptions(border_width=0, text_color='white', background_color='black', text_element_background_color='black') + + clock = [ + [sg.T('', pad=((120,0),0)), + sg.Image(data=ledblank[22:], key='_hour1_'), + sg.Image(data=ledblank[22:], key='_hour2_'), + sg.Image(data=ledblank[22:], key='_colon_'), + sg.Image(data=ledblank[22:], key='_min1_'), + sg.Image(data=ledblank[22:], key='_min2_')], ] + + today_weekday = datetime.datetime.today().weekday() + weather_cols = [] + for i in range(NUM_COLS): + weather_cols.append( + [[sg.T(calendar.day_abbr[(today_weekday + i) % 7], size=(4, 1), font='Any 20', justification='center'), ], + [sg.Image(data=w1[22:], background_color='black', key='_icon_'+str(i), pad=((4, 0), 3)), ], + [sg.T('--', size=(3, 1), justification='center', font='Any 20', key='_high_' + str(i), pad=((10, 0), 3))], + [sg.T('--', size=(3, 1), justification='center', font='Any 20', key='_low_' + str(i), pad=((10, 0), 3))]]) + + layout = [[sg.Column(clock, background_color='black')], + [sg.Column(weather_cols[x], background_color='black') for x in range(NUM_COLS)], + + [sg.RButton('Exit', button_color=('white', 'black'), image_data=orangeround[22:], + tooltip='close window')]] + window = sg.Window('My new window', + background_color='black', + grab_anywhere=True, + use_default_focus=False, + no_titlebar=True + ).Layout(layout).Finalize() + + colon_elem = window.FindElement('_colon_') + hour1 = window.FindElement('_hour1_') + hour2 = window.FindElement('_hour2_') + min1 = window.FindElement('_min1_') + min2 = window.FindElement('_min2_') + + i = last_update_time = 0 + while True: # Event Loop + event, values = window.Read(timeout=1000) + if event in (None, 'Exit'): + break + # update the clock + now = datetime.datetime.now() + real_hour = now.hour - 12 if now.hour > 12 else now.hour + hour1_digit = led_digits[real_hour // 10] + hour1.Update(data=hour1_digit[22:]) + hour2.Update(data=led_digits[real_hour % 10][22:]) + min2.Update(data=led_digits[int(now.minute) % 10][22:]) + min1.Update(data=led_digits[int(now.minute) // 10][22:]) + if i % 2: + colon_elem.Update(data=ledcolon[22:]) + else: + colon_elem.Update(data=ledblank[22:]) + i += 1 + # update weather + if last_update_time == 0 or (now-last_update_time).seconds >= 60*60*6: + print('*** Updating Weather ***') + last_update_time = now + update_weather() + + +led0 = '' + +led1 = '' + +led2 = '' + +led3 = '' + +led4 = '' + +led5 = '' + +led6 = '' + +led7 = '' + +led8 = '' + +led9 = '' + +ledcolon = '' + +ledblank = '' + +w1 = '' + +w2 = '' + +w3 = '' + +w4 = '' + +w5 = '' + +orangeround = '' + +if __name__ == '__main__': + led_clock() From e28652a54800ad3049a27db263a67166588afadc Mon Sep 17 00:00:00 2001 From: MikeTheWatchGuy Date: Sun, 21 Oct 2018 21:34:53 -0400 Subject: [PATCH 11/14] test --- Demo_LED_Indicators.py | 105 ++++++++++++++++++++--------------------- 1 file changed, 51 insertions(+), 54 deletions(-) diff --git a/Demo_LED_Indicators.py b/Demo_LED_Indicators.py index 06961874..be9417dd 100644 --- a/Demo_LED_Indicators.py +++ b/Demo_LED_Indicators.py @@ -1,54 +1,51 @@ -#!/usr/bin/env python -import sys - -if sys.version_info[0] >= 3: - import PySimpleGUI as sg -else: - import PySimpleGUI27 as sg -import time -import random - -""" - Demo program showing how to create your own "LED Indicators" - The LEDIndicator function acts like a new Element that is directly placed in a window's layout - After the Window is created, use the SetLED function to access the LED and set the color - -""" - - -def LEDIndicator(key=None, radius=30): - return sg.Graph(canvas_size=(radius, radius), - graph_bottom_left=(-radius, -radius), - graph_top_right=(radius, radius), - pad=(0, 0), key=key) - -def SetLED(window, key, color): - graph = window.FindElement(key) - graph.Erase() - graph.DrawCircle((0, 0), 12, fill_color=color, line_color=color) - - -layout = [[sg.Text('My LED Status Indicators', size=(20,1))], - [sg.Text('CPU Use'), LEDIndicator('_cpu_')], - [sg.Text('RAM'), LEDIndicator('_ram_')], - [sg.Text('Temperature'), LEDIndicator('_temp_')], - [sg.Text('Server 1'), LEDIndicator('_server1_')], - [sg.RButton('Exit')]] - -window = sg.Window('My new window', default_element_size=(12, 1), auto_size_text=False).Layout(layout).Finalize() - -i = 0 -while True: # Event Loop - event, value = window.ReadNonBlocking() - if event == 'Exit': - window.CloseNonBlocking() - break - if value is None: - break - i += 1 - SetLED(window, '_cpu_', 'green' if random.randint(1, 10) > 5 else 'red') - SetLED(window, '_ram_', 'green' if random.randint(1, 10) > 5 else 'red') - SetLED(window, '_temp_', 'green' if random.randint(1, 10) > 5 else 'red') - SetLED(window, '_server1_', 'green' if random.randint(1, 10) > 5 else 'red') - - time.sleep(.400) +#!/usr/bin/env python +import sys + +if sys.version_info[0] >= 3: + import PySimpleGUI as sg +else: + import PySimpleGUI27 as sg +import time +import random + +""" + Demo program showing how to create your own "LED Indicators" + The LEDIndicator function acts like a new Element that is directly placed in a window's layout + After the Window is created, use the SetLED function to access the LED and set the color + +""" + + +def LEDIndicator(key=None, radius=30): + return sg.Graph(canvas_size=(radius, radius), + graph_bottom_left=(-radius, -radius), + graph_top_right=(radius, radius), + pad=(0, 0), key=key) + +def SetLED(window, key, color): + graph = window.FindElement(key) + graph.Erase() + graph.DrawCircle((0, 0), 12, fill_color=color, line_color=color) + + +layout = [[sg.Text('My LED Status Indicators', size=(20,1))], + [sg.Text('CPU Use'), LEDIndicator('_cpu_')], + [sg.Text('RAM'), LEDIndicator('_ram_')], + [sg.Text('Temperature'), LEDIndicator('_temp_')], + [sg.Text('Server 1'), LEDIndicator('_server1_')], + [sg.RButton('Exit')]] + +window = sg.Window('My new window', default_element_size=(12, 1), auto_size_text=False).Layout(layout).Finalize() + +i = 0 +while True: # Event Loop + event, value = window.Read(timeout=400) + if event == 'Exit' or event is None: + break + if value is None: + break + i += 1 + SetLED(window, '_cpu_', 'green' if random.randint(1, 10) > 5 else 'red') + SetLED(window, '_ram_', 'green' if random.randint(1, 10) > 5 else 'red') + SetLED(window, '_temp_', 'green' if random.randint(1, 10) > 5 else 'red') + SetLED(window, '_server1_', 'green' if random.randint(1, 10) > 5 else 'red') From 8b04b9598cd44c03a1341acbb1f431300174690c Mon Sep 17 00:00:00 2001 From: jackyOO7 <44204857+jackyOO7@users.noreply.github.com> Date: Mon, 22 Oct 2018 11:13:54 +0200 Subject: [PATCH 14/14] Create Demo_OpenCV_Simple_GUI.py A simple example where different OpenCV functions can be controlled via PySimpleGUI in realtime. Is that going in the right direction? --- Demo_OpenCV_Simple_GUI.py | 76 +++++++++++++++++++++++++++++++++++++++ 1 file changed, 76 insertions(+) create mode 100644 Demo_OpenCV_Simple_GUI.py diff --git a/Demo_OpenCV_Simple_GUI.py b/Demo_OpenCV_Simple_GUI.py new file mode 100644 index 00000000..ff051d32 --- /dev/null +++ b/Demo_OpenCV_Simple_GUI.py @@ -0,0 +1,76 @@ +import sys +if sys.version_info[0] >= 3: + import PySimpleGUI as sg +else: + import PySimpleGUI27 as sg +import cv2 +import numpy as np +from sys import exit as exit + +""" +Demo program that displays a webcam using OpenCV and applies some very basic image functions + +- functions from top to bottom - +none: no processing +threshold: simple b/w-threshold on the luma channel, slider sets the threshold value +canny: edge finding with canny, sliders set the two threshold values for the function => edge sensitivity +contour: colour finding in the frame, first slider sets the hue for the colour to find, second the minimum saturation + for the object. Found objects are drawn with a red contour. +blur: simple Gaussian blur, slider sets the sigma, i.e. the amount of blur smear +hue: moves the image hue values by the amount selected on the slider +enhance: applies local contrast enhancement on the luma channel to make the image fancier - slider controls fanciness. +""" +def main(): + + sg.ChangeLookAndFeel('LightGreen') + + # define the window layout + layout = [[sg.Text('OpenCV Demo', size=(40, 1), justification='center')], + [sg.Image(filename='', key='image')], + [sg.Radio('None', 'Radio', True, size=(10, 1))], + [sg.Radio('threshold', 'Radio', size=(10, 1),key='thresh'),sg.Slider((0,255),128,1,orientation='h', size=(40, 15),key='thresh_slider')], + [sg.Radio('canny', 'Radio', size=(10, 1), key='canny'),sg.Slider((0, 255), 128, 1, orientation='h', size=(20, 15), key='canny_slider_a'),sg.Slider((0, 255), 128, 1, orientation='h', size=(20, 15), key='canny_slider_b')], + [sg.Radio('contour', 'Radio', size=(10, 1), key='contour'),sg.Slider((0, 255), 128, 1, orientation='h', size=(20, 15), key='contour_slider'),sg.Slider((0, 255), 80, 1, orientation='h', size=(20, 15), key='base_slider')], + [sg.Radio('blur', 'Radio', size=(10, 1),key='blur'),sg.Slider((1,11),1,1,orientation='h', size=(40, 15),key='blur_slider')], + [sg.Radio('hue', 'Radio', size=(10, 1), key='hue'),sg.Slider((0, 225), 0, 1, orientation='h', size=(40, 15), key='hue_slider')], + [sg.Radio('enhance', 'Radio', size=(10, 1),key='enhance'),sg.Slider((1,255),128,1,orientation='h', size=(40, 15),key='enhance_slider')], + [sg.ReadButton('Exit', size=(10, 1))]] + + # create the window and show it without the plot + window = sg.Window('Demo Application - OpenCV Integration', + location=(800,400)) + window.Layout(layout).Finalize() + + cap = cv2.VideoCapture(0) + while True: + event, values = window.ReadNonBlocking() + if event == 'Exit' or values is None: + sys.exit(0) + ret, frame = cap.read() + if values['thresh']: + frame=cv2.cvtColor(frame,cv2.COLOR_BGR2LAB)[:,:,0] + _,frame=cv2.threshold(frame,values['thresh_slider'],255,cv2.THRESH_BINARY) + if values['canny']: + frame=cv2.Canny(frame,values['canny_slider_a'],values['canny_slider_b']) + if values['blur']: + frame=cv2.GaussianBlur(frame,(21,21),values['blur_slider']) + if values['hue']: + frame=cv2.cvtColor(frame,cv2.COLOR_BGR2HSV) + frame[:,:,0]+=values['hue_slider'] + frame=cv2.cvtColor(frame,cv2.COLOR_HSV2BGR) + if values['enhance']: + enh_val=values['enhance_slider']/40 + clahe=cv2.createCLAHE(clipLimit=enh_val, tileGridSize=(8,8)) + lab=cv2.cvtColor(frame,cv2.COLOR_BGR2LAB) + lab[:,:,0]=clahe.apply(lab[:,:,0]) + frame=cv2.cvtColor(lab,cv2.COLOR_LAB2BGR) + if values['contour']: + hue=cv2.cvtColor(frame,cv2.COLOR_BGR2HSV) + hue=cv2.GaussianBlur(hue,(21,21),1) + hue=cv2.inRange(hue,np.array([values['contour_slider'],values['base_slider'],40]),np.array([values['contour_slider']+30,255,220])) + _,cnts,_=cv2.findContours(hue,cv2.RETR_EXTERNAL,cv2.CHAIN_APPROX_SIMPLE) + cv2.drawContours(frame,cnts,-1,(0,0,255),2) + imgbytes=cv2.imencode('.png', frame)[1].tobytes() #ditto + window.FindElement('image').Update(data=imgbytes) + +main()