Re: [Tutor] Tkinter dilemas
Magnus Lycka
magnus at thinkware.se
Tue Apr 20 13:55:00 EDT 2004
I fear that there is no really beginner-friendly way of writing GUI code
in Python. Tkinter isn't great, but the other GUI tool kits are thin
wrappers for C or C++ libraries where quirks caused by the underlying
language shine through. We can't really compete with VB on the beginner
level here... Will someone please write PyNewbieGui or VisualPython?
David Talaga wrote:
> global root
> root=Tk()
The global statement is meaningless here. Use global within a local
scope to declare that a certain name in that (local) scope refers to
a global variable. E.g:
>>> x = "Hello"
>>> def f():
global x
x = 'Hi'
>>> print x
Hello
>>> f()
>>> print x
Hi
There is no way you can stop local variables overriding globals like
you seem to be trying to do. On the other hand, you never assign to
the name "root" in any local scope, so you don't need that anyway.
As a general rule, try to avoid global variables. Your program is
short but still seems messy to me. Maybe you can try to clean it
up a bit. For instance, you use both "from Tkinter import *" and
"import Tkinter". Why all of Tkinter in two scopes? For clarity,
avoid using "from <something> import *".
You should also try to give all variables and functions good names.
Try to spend some time thinking about that. It will make it easier
for others to understand your code, and it might help you make it
clear for yourself what the different parts of the program are really
supposed to do. (If you can't think of a good name for a function or
variable, it's likely that it's not a very good function or variable...)
> import re
..
> sub = re.sub
..
> tmp = sub('\x0D','', i)
Don't do that! Skip the middle step and do
> import re
..
> tmp = re.sub('\x0D','', i)
A call to "re.sub" is meaningful to any experienced Python coder reading
your code. Just "sub" means that he has to look in some other place in the
code (remember that "real" programs get thousands of lines long) to figure
out that your sub is simply the sub in the re module. Also, for a Python
newbie reading "re.sub" it will be clear where to find out more about what
the sub function does.
> def doDialog():
> global result
> result = dialog()
Why this strange two step solution?
#Define the List function that displays the list of "cleaned" files
def Open():
Incorrect comments are worse than no comments... ;)
> def Clean():
> if f:
> fileName = dialog()
> in_file = open(fileName,'r').readlines()
> out_file = open(fileName + '_cleaned.txt', 'w')
> for i in in_file:
> tmp = sub('\x0D','', i)
> out_file.write(tmp)
> print 'File cleaned'
>
> if x:
> out_file.close()
> if l:
> List()
> print fileName
Huh? Neither f, x nor l are defined in this scope (or globally).
In the function where they *are* defined they all refer to Buttons,
and should all be "true" all the time. It seems these if-statements
have nothing to do in this function. Clean should only be called if
you pressed the Clean button anyway.
Have you really understood the concept of callbacks in Tkinter? You
know, the Tkinter.Button(... command = ...) thingie?
> #Make the path where the cleaned files will be stored
> if os.path.isdir("C:\\Cleaned_Files"):
> print "C:\\Cleaned_Files exists"
> else:
> os.mkdir("C:\\Cleaned_Files")
This is exactly the kind of situation where you might possibly
want global variables (but probably not the global statement) after
all. Don't repeat yourself! Note that you should have used this directory
in the Clean() function as well (and in List()?), if it's going to be
meaningful. Defining the same string like that over and over again is a
bug waiting to happen. One day you will want to move the directory, but
forget to change the code in one place. :(
So:
CLEANED_FILES_DIR = "C:\\Cleaned_Files"
..
out_file_name = os.path.join(CLEANED_FILES_DIR, fileName + '_cleaned.txt')
out_file = open(out_file_name, 'w')
..
#Make the path where the cleaned files will be stored
if os.path.isdir(CLEANED_FILES_DIR):
print CLEANED_FILES_DIR, "exists"
else:
os.mkdir(CLEANED_FILES_DIR)
Frankly, your code smells of cargo cult programming*) ;). What I mean
is that it seems you are writing (copying?) code that you don't really
understand. That won't work. Perhaps you need to slow down a little,
and try to make something very small and elementary work, and make
sure you understand exactly what all the parts are for, and how they
interact. Then you can expand on that, a small step at a time.
It seems to me that GUI code is an area where OO code with classes
is a very suitable approach. Classes give you a good tool to retain
values (attributes of the object) between function calls, and still
keep all these values in a well defined place. But maybe it's a bit
too much to introduce that concept right now...
I might do something like below, but perhaps I'm just confusing you
by introducing a class if that is a new concept for you. Also note
that I'm far from a Tkinter guru. Anyway, don't expand further on
this unless/until you feel that you understand it completely.
Feel free to ask more questions of course...
import Tkinter
import sys
import tkFileDialog
class MyMenu:
def __init__(self, root):
# Set up menu
self.cleanButton = Tkinter.Button(root, text = "Clean File",
command = self.onCleanButton)
self.cleanButton.pack()
self.viewButton = Tkinter.Button(root, text = "View List",
command = self.onViewButton)
self.viewButton.pack()
self.exitButton = Tkinter.Button(root, text = "Exit",
command = sys.exit)
self.exitButton.pack()
self.filename = None
def openFile(self):
initDir = 'C:/windows/desktop'
filetype = [('All files', '.*')]
self.fileName = tkFileDialog.askopenfilename(initialdir=initDir,
filetypes=filetype)
def cleanFile(self):
if not self.fileName:
return
# You know what to put here...
print "Process", self.fileName
def onCleanButton(self):
self.openFile()
self.cleanFile()
def onViewButton(self):
print "View List"
# I don't quite understand what you are trying to do in "List()",
# but I suspect you want a new window for this? A Toplevel?
root = Tkinter.Tk()
MyMenu(root)
root.mainloop()
*) See http://www.science.uva.nl/~mes/jargon/c/cargocultprogramming.html
--
Magnus Lycka, Thinkware AB
Alvans vag 99, SE-907 50 UMEA, SWEDEN
phone: int+46 70 582 80 65, fax: int+46 70 612 80 65
http://www.thinkware.se/ mailto:magnus at thinkware.se
More information about the Tutor
mailing list