Merge pull request #726 from MikeTheWatchGuy/Dev-latest
openCV Demo port! Raw and Base64 image support, text justification, t…
This commit is contained in:
commit
010b7fded5
|
@ -0,0 +1,46 @@
|
||||||
|
# PySimpleGUI-HowDoI
|
||||||
|
|
||||||
|
## Introduction
|
||||||
|
This package contains a GUI front-end to the AMAZING tool, HowDoI. You can ask this tool any programming question and it will tap into the enormous database of programming questions and answers, StackOverflow.
|
||||||
|
|
||||||
|
This program takes you question and returns CODE as a response.
|
||||||
|
|
||||||
|
The way it works is that it searches StackOverflow, gets the results and then finds the highest voted answer. From that answer it takes the code it finds and that is what is returned to you. It works shockingly well.
|
||||||
|
|
||||||
|
To learn more about HowDoI, visit their GitHub site:
|
||||||
|
https://github.com/gleitz/howdoi
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
![snag-0081](https://user-images.githubusercontent.com/13696193/46911287-3a151580-cf25-11e8-8328-f36fda446c4b.jpg)
|
||||||
|
|
||||||
|
|
||||||
|
Check out this example. This was not rehearsed. While typing this readme, an example was needed and a random question, that I've never asked before, was posed. Once again this program delivered a great answer.
|
||||||
|
|
||||||
|
You can copy and paste the solution right into your code if you wish.
|
||||||
|
|
||||||
|
## Installing
|
||||||
|
|
||||||
|
When you install PySimpleGUI-HowDoI, it will install the other components that it requires. To install, on windows, type this into a command prompt:
|
||||||
|
|
||||||
|
pip install pysimplegui-howdoi
|
||||||
|
|
||||||
|
|
||||||
|
## Running the GUI Program
|
||||||
|
|
||||||
|
Afer your Pip install completes you can run the program. Do run it, type this into your command prompt:
|
||||||
|
|
||||||
|
python -m pysimplegui-howdoi.pysimplegui-howdoi
|
||||||
|
|
||||||
|
Once running you simply type in your question and press enter or click the "SEND" button. If you want to ask a question again, you can use the arrow keys or your mouse wheel to access the history of questions you've previously asked.
|
||||||
|
|
||||||
|
Ask ANY question you want for ANY programming language. I recommend starting the question with the programming language.
|
||||||
|
|
||||||
|
|
||||||
|
## PySimpleGUI Project
|
||||||
|
|
||||||
|
This program was built as a sample application of the PySimpleGUI GUI Framework. It quickly became a tool I was unable to live without. I've been trying for some time to bring it to life for others to try.
|
||||||
|
|
||||||
|
## Windows Only?
|
||||||
|
|
||||||
|
This has only been tested using Windows. I have not gotten it to work under Linux. The linkage between the program and the howdoI package was messed up on Linux. If you're able to get a Linux version running, please let me know at info@PySimpleGUI.org
|
|
@ -6,7 +6,20 @@ import textwrap
|
||||||
import pickle
|
import pickle
|
||||||
import base64
|
import base64
|
||||||
import calendar
|
import calendar
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
from PySide2.QtWidgets import QApplication, QLabel, QWidget, QLineEdit, QComboBox, QFormLayout, QVBoxLayout, \
|
||||||
|
QHBoxLayout, QListWidget, QDial, QTableWidget
|
||||||
|
from PySide2.QtWidgets import QSlider, QCheckBox, QRadioButton, QSpinBox, QPushButton, QTextEdit, QMainWindow, QDialog, QAbstractItemView
|
||||||
|
from PySide2.QtWidgets import QSpacerItem, QFrame, QGroupBox, QTextBrowser, QPlainTextEdit, QButtonGroup, QFileDialog, QTableWidget, QTabWidget, QTabBar, QTreeWidget, QTreeWidgetItem, QLayout, QTreeWidgetItemIterator, QProgressBar
|
||||||
|
from PySide2.QtWidgets import QTableWidgetItem, QGraphicsView, QGraphicsScene, QGraphicsItemGroup, QMenu, QMenuBar, QAction, QSystemTrayIcon
|
||||||
|
from PySide2.QtGui import QPainter, QPixmap, QPen, QColor, QBrush, QPainterPath, QFont, QImage, QIcon
|
||||||
|
from PySide2.QtCore import Qt,QProcess, QEvent
|
||||||
|
import PySide2.QtGui as QtGui
|
||||||
|
import PySide2.QtCore as QtCore
|
||||||
|
import PySide2.QtWidgets as QtWidgets
|
||||||
|
using_pyqt5 = False
|
||||||
|
except:
|
||||||
from PyQt5.QtWidgets import QApplication, QLabel, QWidget, QLineEdit, QComboBox, QFormLayout, QVBoxLayout, \
|
from PyQt5.QtWidgets import QApplication, QLabel, QWidget, QLineEdit, QComboBox, QFormLayout, QVBoxLayout, \
|
||||||
QHBoxLayout, QListWidget, QDial, QTableWidget
|
QHBoxLayout, QListWidget, QDial, QTableWidget
|
||||||
from PyQt5.QtWidgets import QSlider, QCheckBox, QRadioButton, QSpinBox, QPushButton, QTextEdit, QMainWindow, QDialog, QAbstractItemView
|
from PyQt5.QtWidgets import QSlider, QCheckBox, QRadioButton, QSpinBox, QPushButton, QTextEdit, QMainWindow, QDialog, QAbstractItemView
|
||||||
|
@ -18,18 +31,6 @@ try:
|
||||||
import PyQt5.QtCore as QtCore
|
import PyQt5.QtCore as QtCore
|
||||||
import PyQt5.QtWidgets as QtWidgets
|
import PyQt5.QtWidgets as QtWidgets
|
||||||
using_pyqt5 = True
|
using_pyqt5 = True
|
||||||
except:
|
|
||||||
from PySide2.QtWidgets import QApplication, QLabel, QWidget, QLineEdit, QComboBox, QFormLayout, QVBoxLayout, \
|
|
||||||
QHBoxLayout, QListWidget, QDial, QTableWidget
|
|
||||||
from PySide2.QtWidgets import QSlider, QCheckBox, QRadioButton, QSpinBox, QPushButton, QTextEdit, QMainWindow, QDialog, QAbstractItemView
|
|
||||||
from PySide2.QtWidgets import QSpacerItem, QFrame, QGroupBox, QTextBrowser, QPlainTextEdit, QButtonGroup, QFileDialog, QTableWidget, QTabWidget, QTabBar, QTreeWidget, QTreeWidgetItem, QLayout, QTreeWidgetItemIterator, QProgressBar
|
|
||||||
from PySide2.QtWidgets import QTableWidgetItem, QGraphicsView, QGraphicsScene, QGraphicsItemGroup, QMenu, QMenuBar, QAction
|
|
||||||
from PySide2.QtGui import QPainter, QPixmap, QPen, QColor, QBrush, QPainterPath, QFont, QImage, QIcon
|
|
||||||
from PySide2.QtCore import Qt,QProcess, QEvent
|
|
||||||
import PySide2.QtGui as QtGui
|
|
||||||
import PySide2.QtCore as QtCore
|
|
||||||
import PySide2.QtWidgets as QtWidgets
|
|
||||||
using_pyqt5 = False
|
|
||||||
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
@ -91,7 +92,7 @@ DEFAULT_FONT = ("Helvetica", 10)
|
||||||
DEFAULT_TEXT_JUSTIFICATION = 'left'
|
DEFAULT_TEXT_JUSTIFICATION = 'left'
|
||||||
DEFAULT_BORDER_WIDTH = 1
|
DEFAULT_BORDER_WIDTH = 1
|
||||||
DEFAULT_AUTOCLOSE_TIME = 3 # time in seconds to show an autoclose form
|
DEFAULT_AUTOCLOSE_TIME = 3 # time in seconds to show an autoclose form
|
||||||
DEFAULT_DEBUG_WINDOW_SIZE = (80, 20)
|
DEFAULT_DEBUG_WINDOW_SIZE = (800, 400)
|
||||||
DEFAULT_WINDOW_LOCATION = (None, None)
|
DEFAULT_WINDOW_LOCATION = (None, None)
|
||||||
MAX_SCROLLED_TEXT_BOX_HEIGHT = 50
|
MAX_SCROLLED_TEXT_BOX_HEIGHT = 50
|
||||||
DEFAULT_TOOLTIP_TIME = 400
|
DEFAULT_TOOLTIP_TIME = 400
|
||||||
|
@ -297,10 +298,7 @@ class Element():
|
||||||
def __init__(self, type, size=(None, None), auto_size_text=None, font=None, background_color=None, text_color=None,
|
def __init__(self, type, size=(None, None), auto_size_text=None, font=None, background_color=None, text_color=None,
|
||||||
key=None, pad=None, tooltip=None):
|
key=None, pad=None, tooltip=None):
|
||||||
|
|
||||||
if size[1] is not None and size[1] < 10: # change from character based size to pixels (roughly)
|
self.Size = convert_tkinter_size_to_Qt(size)
|
||||||
self.Size = size[0]*10, size[1]*25
|
|
||||||
else:
|
|
||||||
self.Size = size
|
|
||||||
self.Type = type
|
self.Type = type
|
||||||
self.AutoSizeText = auto_size_text
|
self.AutoSizeText = auto_size_text
|
||||||
self.Pad = DEFAULT_ELEMENT_PADDING if pad is None else pad
|
self.Pad = DEFAULT_ELEMENT_PADDING if pad is None else pad
|
||||||
|
@ -936,8 +934,12 @@ class Multiline(Element, QWidget):
|
||||||
self.Autoscroll = autoscroll
|
self.Autoscroll = autoscroll
|
||||||
self.Disabled = disabled
|
self.Disabled = disabled
|
||||||
self.ChangeSubmits = change_submits
|
self.ChangeSubmits = change_submits
|
||||||
|
tsize = size # convert tkinter size to pixels
|
||||||
|
if size[0] is not None and size[0] < 100:
|
||||||
|
tsize = convert_tkinter_size_to_Qt(size)
|
||||||
|
|
||||||
super().__init__(ELEM_TYPE_INPUT_MULTILINE, size=size, auto_size_text=auto_size_text, background_color=bg,
|
|
||||||
|
super().__init__(ELEM_TYPE_INPUT_MULTILINE, size=tsize, auto_size_text=auto_size_text, background_color=bg,
|
||||||
text_color=fg, key=key, pad=pad, tooltip=tooltip, font=font or DEFAULT_FONT)
|
text_color=fg, key=key, pad=pad, tooltip=tooltip, font=font or DEFAULT_FONT)
|
||||||
return
|
return
|
||||||
|
|
||||||
|
@ -1139,8 +1141,9 @@ class Output(Element):
|
||||||
bg = background_color if background_color else DEFAULT_INPUT_ELEMENTS_COLOR
|
bg = background_color if background_color else DEFAULT_INPUT_ELEMENTS_COLOR
|
||||||
fg = text_color if text_color is not None else DEFAULT_INPUT_TEXT_COLOR
|
fg = text_color if text_color is not None else DEFAULT_INPUT_TEXT_COLOR
|
||||||
|
|
||||||
|
tsize = convert_tkinter_size_to_Qt(size) if size[0] is not None and size[0] < 100 else size
|
||||||
|
|
||||||
super().__init__(ELEM_TYPE_OUTPUT, size=size, background_color=bg, text_color=fg, pad=pad, font=font,
|
super().__init__(ELEM_TYPE_OUTPUT, size=tsize, background_color=bg, text_color=fg, pad=pad, font=font,
|
||||||
tooltip=tooltip, key=key)
|
tooltip=tooltip, key=key)
|
||||||
|
|
||||||
def reroute_stdout(self):
|
def reroute_stdout(self):
|
||||||
|
@ -1423,10 +1426,16 @@ class ProgressBar(Element):
|
||||||
self.Relief = relief if relief else DEFAULT_PROGRESS_BAR_RELIEF
|
self.Relief = relief if relief else DEFAULT_PROGRESS_BAR_RELIEF
|
||||||
self.BarExpired = False
|
self.BarExpired = False
|
||||||
self.StartValue = start_value
|
self.StartValue = start_value
|
||||||
super().__init__(ELEM_TYPE_PROGRESS_BAR, size=size, auto_size_text=auto_size_text, key=key, pad=pad)
|
tsize = size
|
||||||
|
if size[0] is not None and size[0] < 100:
|
||||||
|
tsize = size[0]*10, size[1]*3
|
||||||
|
|
||||||
|
super().__init__(ELEM_TYPE_PROGRESS_BAR, size=tsize, auto_size_text=auto_size_text, key=key, pad=pad)
|
||||||
|
|
||||||
# returns False if update failed
|
# returns False if update failed
|
||||||
def UpdateBar(self, current_count, max=None):
|
def UpdateBar(self, current_count, max=None):
|
||||||
|
if max is not None:
|
||||||
|
self.QT_QProgressBar.setMaximum(max)
|
||||||
self.QT_QProgressBar.setValue(current_count)
|
self.QT_QProgressBar.setValue(current_count)
|
||||||
self.ParentForm.QTApplication.processEvents() # refresh the window
|
self.ParentForm.QTApplication.processEvents() # refresh the window
|
||||||
|
|
||||||
|
@ -1440,7 +1449,7 @@ class ProgressBar(Element):
|
||||||
# Image #
|
# Image #
|
||||||
# ---------------------------------------------------------------------- #
|
# ---------------------------------------------------------------------- #
|
||||||
class Image(Element):
|
class Image(Element):
|
||||||
def __init__(self, filename=None, data=None, background_color=None, size=(None, None), pad=None, key=None,
|
def __init__(self, filename=None, data=None, data_base64=None, background_color=None, size=(None, None), pad=None, key=None,
|
||||||
tooltip=None):
|
tooltip=None):
|
||||||
'''
|
'''
|
||||||
Image Element
|
Image Element
|
||||||
|
@ -1454,15 +1463,16 @@ class Image(Element):
|
||||||
'''
|
'''
|
||||||
self.Filename = filename
|
self.Filename = filename
|
||||||
self.Data = data
|
self.Data = data
|
||||||
|
self.DataBase64 = data_base64
|
||||||
self.tktext_label = None
|
self.tktext_label = None
|
||||||
self.BackgroundColor = background_color
|
self.BackgroundColor = background_color
|
||||||
if data is None and filename is None:
|
if data is None and filename is None and data_base64 is None:
|
||||||
print('* Warning... no image specified in Image Element! *')
|
print('* Warning... no image specified in Image Element! *')
|
||||||
super().__init__(ELEM_TYPE_IMAGE, size=size, background_color=background_color, pad=pad, key=key,
|
super().__init__(ELEM_TYPE_IMAGE, size=size, background_color=background_color, pad=pad, key=key,
|
||||||
tooltip=tooltip)
|
tooltip=tooltip)
|
||||||
return
|
return
|
||||||
|
|
||||||
def Update(self, filename=None, data=None, size=(None, None)):
|
def Update(self, filename=None, data=None, data_base64=None, size=(None, None)):
|
||||||
if filename is not None:
|
if filename is not None:
|
||||||
qlabel = self.QT_QLabel
|
qlabel = self.QT_QLabel
|
||||||
qlabel.setText('')
|
qlabel.setText('')
|
||||||
|
@ -1473,11 +1483,17 @@ class Image(Element):
|
||||||
elif data is not None:
|
elif data is not None:
|
||||||
qlabel = self.QT_QLabel
|
qlabel = self.QT_QLabel
|
||||||
qlabel.setText('')
|
qlabel.setText('')
|
||||||
ba = QtCore.QByteArray.fromBase64(data)
|
ba = QtCore.QByteArray.fromRawData(data)
|
||||||
|
pixmap = QtGui.QPixmap()
|
||||||
|
pixmap.loadFromData(ba)
|
||||||
|
qlabel.setPixmap(pixmap)
|
||||||
|
elif data_base64 is not None:
|
||||||
|
qlabel = self.QT_QLabel
|
||||||
|
qlabel.setText('')
|
||||||
|
ba = QtCore.QByteArray.fromBase64(data_base64)
|
||||||
pixmap = QtGui.QPixmap()
|
pixmap = QtGui.QPixmap()
|
||||||
pixmap.loadFromData(ba)
|
pixmap.loadFromData(ba)
|
||||||
qlabel.setPixmap(pixmap)
|
qlabel.setPixmap(pixmap)
|
||||||
|
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
super().__del__()
|
super().__del__()
|
||||||
|
@ -1991,7 +2007,8 @@ class Slider(Element):
|
||||||
temp_size = size
|
temp_size = size
|
||||||
if temp_size == (None, None):
|
if temp_size == (None, None):
|
||||||
temp_size = (150, 30) if self.Orientation.startswith('h') else (30, 150)
|
temp_size = (150, 30) if self.Orientation.startswith('h') else (30, 150)
|
||||||
|
elif size[0] is not None and size[0] < 100:
|
||||||
|
temp_size = size[0]*10, size[1]*3
|
||||||
|
|
||||||
|
|
||||||
super().__init__(ELEM_TYPE_INPUT_SLIDER, size=temp_size, font=font, background_color=background_color,
|
super().__init__(ELEM_TYPE_INPUT_SLIDER, size=temp_size, font=font, background_color=background_color,
|
||||||
|
@ -2528,6 +2545,17 @@ class ErrorElement(Element):
|
||||||
super().__del__()
|
super().__del__()
|
||||||
|
|
||||||
|
|
||||||
|
# ------------------------------------------------------------------------- #
|
||||||
|
# Tray CLASS #
|
||||||
|
# ------------------------------------------------------------------------- #
|
||||||
|
class Tray:
|
||||||
|
def __init__(self, title):
|
||||||
|
self.Title = title
|
||||||
|
|
||||||
|
def Read(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
# ------------------------------------------------------------------------- #
|
# ------------------------------------------------------------------------- #
|
||||||
# Window CLASS #
|
# Window CLASS #
|
||||||
# ------------------------------------------------------------------------- #
|
# ------------------------------------------------------------------------- #
|
||||||
|
@ -2568,8 +2596,8 @@ class Window:
|
||||||
self.AutoSizeButtons = auto_size_buttons if auto_size_buttons is not None else DEFAULT_AUTOSIZE_BUTTONS
|
self.AutoSizeButtons = auto_size_buttons if auto_size_buttons is not None else DEFAULT_AUTOSIZE_BUTTONS
|
||||||
self.Title = title
|
self.Title = title
|
||||||
self.Rows = [] # a list of ELEMENTS for this row
|
self.Rows = [] # a list of ELEMENTS for this row
|
||||||
self.DefaultElementSize = default_element_size
|
self.DefaultElementSize = convert_tkinter_size_to_Qt(default_element_size)
|
||||||
self.DefaultButtonElementSize = default_button_element_size if default_button_element_size != (
|
self.DefaultButtonElementSize = convert_tkinter_size_to_Qt(default_button_element_size) if default_button_element_size != (
|
||||||
None, None) else DEFAULT_BUTTON_ELEMENT_SIZE
|
None, None) else DEFAULT_BUTTON_ELEMENT_SIZE
|
||||||
self.Location = location
|
self.Location = location
|
||||||
self.ButtonColor = button_color if button_color else DEFAULT_BUTTON_COLOR
|
self.ButtonColor = button_color if button_color else DEFAULT_BUTTON_COLOR
|
||||||
|
@ -2869,19 +2897,17 @@ class Window:
|
||||||
print('*** Error loading form to disk ***')
|
print('*** Error loading form to disk ***')
|
||||||
|
|
||||||
def GetScreenDimensions(self):
|
def GetScreenDimensions(self):
|
||||||
if self.TKrootDestroyed:
|
# TODO
|
||||||
return None, None
|
|
||||||
screen_width = screen_height = 0
|
screen_width = screen_height = 0
|
||||||
return screen_width, screen_height
|
return screen_width, screen_height
|
||||||
|
|
||||||
def Move(self, x, y):
|
def Move(self, x, y):
|
||||||
try:
|
# TODO
|
||||||
self.TKroot.geometry("+%s+%s" % (x, y))
|
return
|
||||||
except:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def Minimize(self):
|
def Minimize(self):
|
||||||
self.TKroot.iconify()
|
# TODO
|
||||||
|
return
|
||||||
|
|
||||||
def StartMove(self, event):
|
def StartMove(self, event):
|
||||||
try:
|
try:
|
||||||
|
@ -2968,18 +2994,21 @@ class Window:
|
||||||
return
|
return
|
||||||
|
|
||||||
def Disable(self):
|
def Disable(self):
|
||||||
self.TKroot.grab_set_global()
|
# TODO
|
||||||
|
return
|
||||||
|
|
||||||
def Enable(self):
|
def Enable(self):
|
||||||
self.TKroot.grab_release()
|
# TODO
|
||||||
|
return
|
||||||
|
|
||||||
def Hide(self):
|
def Hide(self):
|
||||||
self._Hidden = True
|
self._Hidden = True
|
||||||
self.TKroot.withdraw()
|
# TODO
|
||||||
|
return
|
||||||
|
|
||||||
def UnHide(self):
|
def UnHide(self):
|
||||||
if self._Hidden:
|
if self._Hidden:
|
||||||
self.TKroot.deiconify()
|
# TODO
|
||||||
self._Hidden = False
|
self._Hidden = False
|
||||||
|
|
||||||
def Disappear(self):
|
def Disappear(self):
|
||||||
|
@ -3122,6 +3151,20 @@ def element_callback_quit_mainloop(element):
|
||||||
element.ParentForm.QTApplication.exit() # kick the users out of the mainloop
|
element.ParentForm.QTApplication.exit() # kick the users out of the mainloop
|
||||||
|
|
||||||
|
|
||||||
|
# =========================================================================== #
|
||||||
|
# Stops the mainloop and sets the event information #
|
||||||
|
# =========================================================================== #
|
||||||
|
def convert_tkinter_size_to_Qt(size):
|
||||||
|
"""
|
||||||
|
Converts size in characters to size in pixels
|
||||||
|
:param size: size in characters, rows
|
||||||
|
:return: size in pixels, pixels
|
||||||
|
"""
|
||||||
|
qtsize = size
|
||||||
|
if size[1] is not None and size[1] < 10: # change from character based size to pixels (roughly)
|
||||||
|
qtsize = size[0]*10, size[1]*25
|
||||||
|
return qtsize
|
||||||
|
|
||||||
|
|
||||||
# ################################################################################
|
# ################################################################################
|
||||||
# ################################################################################
|
# ################################################################################
|
||||||
|
@ -3782,9 +3825,17 @@ def PackFormIntoFrame(window, containing_frame, toplevel_win):
|
||||||
# ------------------------- TEXT element ------------------------- #
|
# ------------------------- TEXT element ------------------------- #
|
||||||
elif element_type == ELEM_TYPE_TEXT:
|
elif element_type == ELEM_TYPE_TEXT:
|
||||||
element.QT_Label = QLabel(element.DisplayText, toplevel_win.QTWindow)
|
element.QT_Label = QLabel(element.DisplayText, toplevel_win.QTWindow)
|
||||||
if element.Justification[0] == 'c':
|
|
||||||
|
if element.Justification is not None:
|
||||||
|
justification = element.Justification
|
||||||
|
elif toplevel_win.TextJustification is not None:
|
||||||
|
justification = toplevel_win.TextJustification
|
||||||
|
else:
|
||||||
|
justification = DEFAULT_TEXT_JUSTIFICATION
|
||||||
|
|
||||||
|
if justification[0] == 'c':
|
||||||
element.QT_Label.setAlignment(Qt.AlignCenter)
|
element.QT_Label.setAlignment(Qt.AlignCenter)
|
||||||
elif element.Justification[0] == 'r':
|
elif justification[0] == 'r':
|
||||||
element.QT_Label.setAlignment(Qt.AlignRight)
|
element.QT_Label.setAlignment(Qt.AlignRight)
|
||||||
if not auto_size_text:
|
if not auto_size_text:
|
||||||
if element_size[0] is not None:
|
if element_size[0] is not None:
|
||||||
|
@ -3804,8 +3855,8 @@ def PackFormIntoFrame(window, containing_frame, toplevel_win):
|
||||||
|
|
||||||
if element.Tooltip:
|
if element.Tooltip:
|
||||||
element.QT_Label.setToolTip(element.Tooltip)
|
element.QT_Label.setToolTip(element.Tooltip)
|
||||||
element.QT_Label.setMargin(full_element_pad[0])
|
# element.QT_Label.setMargin(full_element_pad[0])
|
||||||
element.QT_Label.setIndent(full_element_pad[1])
|
# element.QT_Label.setIndent(full_element_pad[1])
|
||||||
qt_row_layout.addWidget(element.QT_Label)
|
qt_row_layout.addWidget(element.QT_Label)
|
||||||
# ------------------------- BUTTON element ------------------------- #
|
# ------------------------- BUTTON element ------------------------- #
|
||||||
elif element_type == ELEM_TYPE_BUTTON:
|
elif element_type == ELEM_TYPE_BUTTON:
|
||||||
|
@ -4141,18 +4192,24 @@ def PackFormIntoFrame(window, containing_frame, toplevel_win):
|
||||||
qt_row_layout.addWidget(element.QT_TextBrowser)
|
qt_row_layout.addWidget(element.QT_TextBrowser)
|
||||||
# ------------------------- IMAGE element ------------------------- #
|
# ------------------------- IMAGE element ------------------------- #
|
||||||
elif element_type == ELEM_TYPE_IMAGE:
|
elif element_type == ELEM_TYPE_IMAGE:
|
||||||
if element.Filename:
|
if element.Filename is not None:
|
||||||
qlabel = QLabel()
|
qlabel = QLabel()
|
||||||
qlabel.setText('')
|
qlabel.setText('')
|
||||||
w = QtGui.QPixmap(element.Filename).width()
|
w = QtGui.QPixmap(element.Filename).width()
|
||||||
h = QtGui.QPixmap(element.Filename).height()
|
h = QtGui.QPixmap(element.Filename).height()
|
||||||
qlabel.setGeometry(QtCore.QRect(0, 0, w, h))
|
qlabel.setGeometry(QtCore.QRect(0, 0, w, h))
|
||||||
qlabel.setPixmap(QtGui.QPixmap(element.Filename))
|
qlabel.setPixmap(QtGui.QPixmap(element.Filename))
|
||||||
elif element.Data:
|
elif element.Data is not None:
|
||||||
qlabel = QLabel()
|
qlabel = QLabel()
|
||||||
qlabel.setText('')
|
qlabel.setText('')
|
||||||
|
ba = QtCore.QByteArray.fromRawData(element.Data)
|
||||||
ba = QtCore.QByteArray.fromBase64(element.Data)
|
pixmap = QtGui.QPixmap()
|
||||||
|
pixmap.loadFromData(ba)
|
||||||
|
qlabel.setPixmap(pixmap)
|
||||||
|
elif element.DataBase64:
|
||||||
|
qlabel = QLabel()
|
||||||
|
qlabel.setText('')
|
||||||
|
ba = QtCore.QByteArray.fromBase64(element.DataBase64)
|
||||||
pixmap = QtGui.QPixmap()
|
pixmap = QtGui.QPixmap()
|
||||||
pixmap.loadFromData(ba)
|
pixmap.loadFromData(ba)
|
||||||
qlabel.setPixmap(pixmap)
|
qlabel.setPixmap(pixmap)
|
||||||
|
@ -4374,8 +4431,8 @@ def PackFormIntoFrame(window, containing_frame, toplevel_win):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
# ............................DONE WITH ROW pack the row of widgets ..........................#
|
# ............................DONE WITH ROW pack the row of widgets ..........................#
|
||||||
containing_frame.setSpacing(toplevel_win.ElementPadding[1])
|
|
||||||
qt_row_layout.setSpacing(toplevel_win.ElementPadding[0])
|
qt_row_layout.setSpacing(toplevel_win.ElementPadding[0])
|
||||||
|
containing_frame.setSpacing(toplevel_win.ElementPadding[1])
|
||||||
containing_frame.addRow('', qt_row_layout)
|
containing_frame.addRow('', qt_row_layout)
|
||||||
|
|
||||||
# done with row, pack the row of widgets
|
# done with row, pack the row of widgets
|
||||||
|
@ -5070,10 +5127,10 @@ def SetOptions(icon=None, button_color=None, element_size=(None, None), button_e
|
||||||
DEFAULT_BUTTON_COLOR = button_color
|
DEFAULT_BUTTON_COLOR = button_color
|
||||||
|
|
||||||
if element_size != (None, None):
|
if element_size != (None, None):
|
||||||
DEFAULT_ELEMENT_SIZE = element_size
|
DEFAULT_ELEMENT_SIZE = convert_tkinter_size_to_Qt(element_size)
|
||||||
|
|
||||||
if button_element_size != (None, None):
|
if button_element_size != (None, None):
|
||||||
DEFAULT_BUTTON_ELEMENT_SIZE = button_element_size
|
DEFAULT_BUTTON_ELEMENT_SIZE = convert_tkinter_size_to_Qt(button_element_size)
|
||||||
|
|
||||||
if margins != (None, None):
|
if margins != (None, None):
|
||||||
DEFAULT_MARGINS = margins
|
DEFAULT_MARGINS = margins
|
||||||
|
@ -5512,8 +5569,8 @@ def ObjToString(obj, extra=' '):
|
||||||
# ----------------------------------- The mighty Popup! ------------------------------------------------------------ #
|
# ----------------------------------- The mighty Popup! ------------------------------------------------------------ #
|
||||||
|
|
||||||
def Popup(*args, button_color=None, background_color=None, text_color=None, button_type=POPUP_BUTTONS_OK,
|
def Popup(*args, button_color=None, background_color=None, text_color=None, button_type=POPUP_BUTTONS_OK,
|
||||||
auto_close=False, auto_close_duration=None, non_blocking=False, icon=DEFAULT_WINDOW_ICON, line_width=None,
|
auto_close=False, auto_close_duration=None, custom_text=(None, None), non_blocking=False, icon=DEFAULT_WINDOW_ICON, line_width=None,
|
||||||
font=None, no_titlebar=False, grab_anywhere=False, keep_on_top=False, location=(None, None)):
|
font=None, no_titlebar=False, grab_anywhere=False, keep_on_top=False, location=(None, None), use_system_tray=False):
|
||||||
"""
|
"""
|
||||||
Popup - Display a popup box with as many parms as you wish to include
|
Popup - Display a popup box with as many parms as you wish to include
|
||||||
:param args:
|
:param args:
|
||||||
|
@ -5535,6 +5592,9 @@ def Popup(*args, button_color=None, background_color=None, text_color=None, butt
|
||||||
"""
|
"""
|
||||||
global _my_windows
|
global _my_windows
|
||||||
|
|
||||||
|
# if use_system_tray:
|
||||||
|
# QSystemTrayIcon.
|
||||||
|
|
||||||
if not args:
|
if not args:
|
||||||
args_to_print = ['']
|
args_to_print = ['']
|
||||||
else:
|
else:
|
||||||
|
@ -5548,6 +5608,7 @@ def Popup(*args, button_color=None, background_color=None, text_color=None, butt
|
||||||
auto_close=auto_close, auto_close_duration=auto_close_duration, icon=icon, font=font,
|
auto_close=auto_close, auto_close_duration=auto_close_duration, icon=icon, font=font,
|
||||||
no_titlebar=no_titlebar, grab_anywhere=grab_anywhere, keep_on_top=keep_on_top, location=location)
|
no_titlebar=no_titlebar, grab_anywhere=grab_anywhere, keep_on_top=keep_on_top, location=location)
|
||||||
max_line_total, total_lines = 0, 0
|
max_line_total, total_lines = 0, 0
|
||||||
|
layout = []
|
||||||
for message in args_to_print:
|
for message in args_to_print:
|
||||||
# fancy code to check if string and convert if not is not need. Just always convert to string :-)
|
# fancy code to check if string and convert if not is not need. Just always convert to string :-)
|
||||||
# if not isinstance(message, str): message = str(message)
|
# if not isinstance(message, str): message = str(message)
|
||||||
|
@ -5562,34 +5623,43 @@ def Popup(*args, button_color=None, background_color=None, text_color=None, butt
|
||||||
max_line_total = max(max_line_total, width_used)
|
max_line_total = max(max_line_total, width_used)
|
||||||
# height = _GetNumLinesNeeded(message, width_used)
|
# height = _GetNumLinesNeeded(message, width_used)
|
||||||
height = message_wrapped_lines
|
height = message_wrapped_lines
|
||||||
window.AddRow(
|
layout.append([Text(message_wrapped, auto_size_text=True, text_color=text_color, background_color=background_color)])
|
||||||
Text(message_wrapped, auto_size_text=True, text_color=text_color, background_color=background_color))
|
|
||||||
total_lines += height
|
total_lines += height
|
||||||
|
|
||||||
if total_lines < 3:
|
if total_lines < 3:
|
||||||
[window.AddRow(Text('')) for i in range(2)]
|
layout.append([Text('')])
|
||||||
|
layout.append([Text('')])
|
||||||
if non_blocking:
|
if non_blocking:
|
||||||
PopupButton = DummyButton # important to use or else button will close other windows too!
|
PopupButton = DummyButton # important to use or else button will close other windows too!
|
||||||
else:
|
else:
|
||||||
PopupButton = CloseButton
|
PopupButton = CloseButton
|
||||||
# show either an OK or Yes/No depending on paramater
|
# show either an OK or Yes/No depending on paramater
|
||||||
if button_type is POPUP_BUTTONS_YES_NO:
|
# show either an OK or Yes/No depending on paramater
|
||||||
window.AddRow(PopupButton('Yes', button_color=button_color, focus=True, bind_return_key=True, pad=((20, 5), 3),
|
if custom_text != (None, None):
|
||||||
size=(60, 20)), PopupButton('No', button_color=button_color, size=(60, 20)))
|
if type(custom_text) is not tuple:
|
||||||
|
window.AddRow(PopupButton(custom_text, button_color=button_color, focus=True, bind_return_key=True))
|
||||||
|
elif custom_text[1] is None:
|
||||||
|
window.AddRow(PopupButton(custom_text[0], button_color=button_color, focus=True, bind_return_key=True))
|
||||||
|
else:
|
||||||
|
window.AddRow(PopupButton(custom_text[0], button_color=button_color, focus=True, bind_return_key=True),
|
||||||
|
PopupButton(custom_text[1], button_color=button_color),Stretch())
|
||||||
|
elif button_type is POPUP_BUTTONS_YES_NO:
|
||||||
|
layout.append([PopupButton('Yes', button_color=button_color, focus=True, bind_return_key=True, pad=((20, 5), 3),
|
||||||
|
size=(60, 20)), PopupButton('No', button_color=button_color, size=(60, 20))])
|
||||||
elif button_type is POPUP_BUTTONS_CANCELLED:
|
elif button_type is POPUP_BUTTONS_CANCELLED:
|
||||||
window.AddRow(
|
layout.append([PopupButton('Cancelled', button_color=button_color, focus=True, bind_return_key=True, pad=((20, 0), 3)), Stretch()])
|
||||||
PopupButton('Cancelled', button_color=button_color, focus=True, bind_return_key=True, pad=((20, 0), 3)))
|
|
||||||
elif button_type is POPUP_BUTTONS_ERROR:
|
elif button_type is POPUP_BUTTONS_ERROR:
|
||||||
window.AddRow(PopupButton('Error', size=(60, 20), button_color=button_color, focus=True, bind_return_key=True,
|
layout.append([PopupButton('Error', size=(60, 20), button_color=button_color, focus=True, bind_return_key=True,
|
||||||
pad=((20, 0), 3)))
|
pad=((20, 0), 3)), Stretch()])
|
||||||
elif button_type is POPUP_BUTTONS_OK_CANCEL:
|
elif button_type is POPUP_BUTTONS_OK_CANCEL:
|
||||||
window.AddRow(PopupButton('OK', size=(60, 20), button_color=button_color, focus=True, bind_return_key=True),
|
layout.append([PopupButton('OK', size=(60, 20), button_color=button_color, focus=True, bind_return_key=True),
|
||||||
PopupButton('Cancel', size=(60, 20), button_color=button_color))
|
PopupButton('Cancel', size=(60, 20), button_color=button_color), Stretch()])
|
||||||
elif button_type is POPUP_BUTTONS_NO_BUTTONS:
|
elif button_type is POPUP_BUTTONS_NO_BUTTONS:
|
||||||
pass
|
pass
|
||||||
else:
|
else:
|
||||||
window.AddRow(PopupButton('OK', size=(60, 20), button_color=button_color, focus=True, bind_return_key=True))
|
layout.append([PopupButton('OK', size=(60, 20), button_color=button_color, focus=True, bind_return_key=True), Stretch()])
|
||||||
|
|
||||||
|
window.Layout(layout)
|
||||||
if non_blocking:
|
if non_blocking:
|
||||||
button, values = window.Read(timeout=0)
|
button, values = window.Read(timeout=0)
|
||||||
_my_windows.active_popups[window] = title
|
_my_windows.active_popups[window] = title
|
||||||
|
@ -5704,7 +5774,7 @@ def PopupQuick(*args, button_type=POPUP_BUTTONS_OK, button_color=None, backgroun
|
||||||
# --------------------------- PopupQuick - a NonBlocking, Self-closing Popup with no titlebar and no buttons ---------------------------
|
# --------------------------- PopupQuick - a NonBlocking, Self-closing Popup with no titlebar and no buttons ---------------------------
|
||||||
def PopupQuickMessage(*args, button_type=POPUP_BUTTONS_NO_BUTTONS, button_color=None, background_color=None,
|
def PopupQuickMessage(*args, button_type=POPUP_BUTTONS_NO_BUTTONS, button_color=None, background_color=None,
|
||||||
text_color=None,
|
text_color=None,
|
||||||
auto_close=True, auto_close_duration=2, non_blocking=True, icon=DEFAULT_WINDOW_ICON,
|
auto_close=True, auto_close_duration=4, non_blocking=True, icon=DEFAULT_WINDOW_ICON,
|
||||||
line_width=None,
|
line_width=None,
|
||||||
font=None, no_titlebar=True, grab_anywhere=False, keep_on_top=False, location=(None, None)):
|
font=None, no_titlebar=True, grab_anywhere=False, keep_on_top=False, location=(None, None)):
|
||||||
"""
|
"""
|
||||||
|
@ -5769,7 +5839,7 @@ PopupAnnoying = PopupNoTitlebar
|
||||||
|
|
||||||
# --------------------------- PopupAutoClose ---------------------------
|
# --------------------------- PopupAutoClose ---------------------------
|
||||||
def PopupAutoClose(*args, button_type=POPUP_BUTTONS_OK, button_color=None, background_color=None, text_color=None,
|
def PopupAutoClose(*args, button_type=POPUP_BUTTONS_OK, button_color=None, background_color=None, text_color=None,
|
||||||
auto_close=True, auto_close_duration=None, non_blocking=False, icon=DEFAULT_WINDOW_ICON,
|
auto_close=True, auto_close_duration=DEFAULT_AUTOCLOSE_TIME, non_blocking=False, icon=DEFAULT_WINDOW_ICON,
|
||||||
line_width=None, font=None, no_titlebar=False, grab_anywhere=False, keep_on_top=False,
|
line_width=None, font=None, no_titlebar=False, grab_anywhere=False, keep_on_top=False,
|
||||||
location=(None, None)):
|
location=(None, None)):
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -0,0 +1,59 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
import sys
|
||||||
|
if sys.version_info[0] >= 3:
|
||||||
|
import PySimpleGUIQt 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
|
||||||
|
"""
|
||||||
|
def main():
|
||||||
|
|
||||||
|
sg.ChangeLookAndFeel('LightGreen')
|
||||||
|
|
||||||
|
# define the window layout
|
||||||
|
layout = [[sg.Text('OpenCV Demo', size=(40, 1), justification='center', font='Helvetica 20')],
|
||||||
|
[sg.Image(filename='', key='image')],
|
||||||
|
[sg.Button('Record', size=(10, 1), font='Helvetica 14'),
|
||||||
|
sg.Button('Stop', size=(10, 1), font='Any 14'),
|
||||||
|
sg.Button('Exit', size=(10, 1), font='Helvetica 14'),
|
||||||
|
sg.Button('About', size=(10,1), font='Any 14')]]
|
||||||
|
|
||||||
|
# create the window and show it without the plot
|
||||||
|
window = sg.Window('Demo Application - OpenCV Integration',
|
||||||
|
location=(800,400))
|
||||||
|
window.Layout(layout)
|
||||||
|
|
||||||
|
# ---===--- Event LOOP Read and display frames, operate the GUI --- #
|
||||||
|
cap = cv2.VideoCapture(0)
|
||||||
|
recording = False
|
||||||
|
while True:
|
||||||
|
event, values = window.Read(timeout=0, timeout_key='timeout')
|
||||||
|
if event == 'Exit' or event is None:
|
||||||
|
sys.exit(0)
|
||||||
|
pass
|
||||||
|
elif event == 'Record':
|
||||||
|
recording = True
|
||||||
|
elif event == 'Stop':
|
||||||
|
recording = False
|
||||||
|
# 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',
|
||||||
|
'www.PySimpleGUI.org',
|
||||||
|
'Check out how the video keeps playing behind this window.',
|
||||||
|
'I finally figured out how to display frames from a webcam.',
|
||||||
|
'ENJOY! Go make something really cool with this... please!',
|
||||||
|
keep_on_top=True)
|
||||||
|
if recording:
|
||||||
|
ret, frame = cap.read()
|
||||||
|
imgbytes=cv2.imencode('.png', frame)[1].tobytes() #ditto
|
||||||
|
window.FindElement('image').Update(data=imgbytes)
|
||||||
|
|
||||||
|
main()
|
||||||
|
exit()
|
|
@ -36,34 +36,47 @@ Welcome to the Alpha Release of PySimpleGUI for Qt!
|
||||||
|
|
||||||
You can use the exact same code that you are running on the older, tkinter, version of PySimpleGUI.
|
You can use the exact same code that you are running on the older, tkinter, version of PySimpleGUI.
|
||||||
|
|
||||||
PySimpleGUIQt uses **PySide2** for access to Qt.
|
PySimpleGUIQt uses **PySide2** OR **PyQt5** for access to Qt.
|
||||||
|
|
||||||
### Differences between PySimpleGUI and PySimpleGUIQt
|
## Porting your PySimpleGUI code to PySimpleGUIQt
|
||||||
|
|
||||||
|
|
||||||
|
To "port" your code from the tkinter implementation. Follow these steps:
|
||||||
|
|
||||||
|
1. Change `import PySimpleGUI` to `PySimpleGUIQt`
|
||||||
|
|
||||||
|
That's it! OK, maybe I should have said step instead of steps.
|
||||||
|
|
||||||
|
|
||||||
|
## Differences between PySimpleGUI and PySimpleGUIQt
|
||||||
|
|
||||||
#### Sizes
|
#### Sizes
|
||||||
IMPORTANT NOTE if you are porting from tkinter to Qt - You will need to make one important change to your code.... **You must change your size parameters to be in PIXELS instead of CHARACTERS**.
|
|
||||||
|
While you can use "Character-based" sizes like you did in tkinter, it's best to use pixel based sizes as that is what Qt uses. PySimpleGUIQt does some very rough / basic conversions from the character sizes to pixel sizes. It's enough that your elements will at least be visible. But the conversion is likely to not be ideal.
|
||||||
|
|
||||||
#### Fonts
|
#### Fonts
|
||||||
|
|
||||||
Fonts should be in the format (font family, size). The original PySimpleGUI also allowed a font string 'Family Size' but that option is not available (yet) in the Qt version. I'll add it though so the code ports straight over.
|
Fonts should be in the format (font family, size). You can use the older string based too, but it will not work with setting like bold and italics. PySimpleGUIQt converts from the string 'Courier 20' to the tuple ('Courier', 20) for you.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### Installing PySimpleGUIQt for Python 3
|
### Installing PySimpleGUIQt for Python 3
|
||||||
|
|
||||||
pip install --upgrade PySimpleGUIQt
|
pip install --upgrade PySimpleGUIQt
|
||||||
|
|
||||||
On some systems you need to run pip3.
|
On Linux systems you need to run pip3.
|
||||||
|
|
||||||
pip3 install --upgrade PySimpleGUIQt
|
pip3 install --upgrade PySimpleGUIQt
|
||||||
|
|
||||||
|
|
||||||
### Installing PySide2 for Python 3
|
### Installing PySide2 or PyQt5 for Python 3
|
||||||
|
|
||||||
|
It is recommended that you use PySide2, however, if that cannot be found, then PyQt5 will be attempted. To install either of these:
|
||||||
|
|
||||||
```pip install PySide2```
|
```pip install PySide2```
|
||||||
|
|
||||||
|
or
|
||||||
|
|
||||||
|
```pip install PyQt5```
|
||||||
|
|
||||||
|
|
||||||
## Testing your installation
|
## Testing your installation
|
||||||
|
@ -85,7 +98,7 @@ Here is the window you should see:
|
||||||
|
|
||||||
## Prerequisites
|
## Prerequisites
|
||||||
Python 3
|
Python 3
|
||||||
PySide2
|
PySide2 or PyQt5
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue