configSectionNameDialog.py 4.21 KB
Newer Older
1 2 3
"""
Dialog that allows user to specify a new config file section name.
Used to get new highlight theme and keybinding set names.
4 5
The 'return value' for the dialog, used two placed in configDialog.py,
is the .result attribute set in the Ok and Cancel methods.
6
"""
7 8
from tkinter import *
import tkinter.messagebox as tkMessageBox
9 10

class GetCfgSectionNameDialog(Toplevel):
11
    def __init__(self, parent, title, message, used_names):
12 13
        """
        message - string, informational message to display
14
        used_names - string collection, names already in use for validity check
15 16 17
        """
        Toplevel.__init__(self, parent)
        self.configure(borderwidth=5)
18
        self.resizable(height=FALSE, width=FALSE)
19 20 21 22 23
        self.title(title)
        self.transient(parent)
        self.grab_set()
        self.protocol("WM_DELETE_WINDOW", self.Cancel)
        self.parent = parent
24 25 26 27
        self.message = message
        self.used_names = used_names
        self.create_widgets()
        self.withdraw()  #hide while setting geometry
28 29 30
        self.update_idletasks()
        #needs to be done here so that the winfo_reqwidth is valid
        self.messageInfo.config(width=self.frameMain.winfo_reqwidth())
31 32 33 34 35 36 37 38
        self.geometry(
                "+%d+%d" % (
                parent.winfo_rootx() +
                (parent.winfo_width()/2 - self.winfo_reqwidth()/2),
                parent.winfo_rooty() +
                (parent.winfo_height()/2 - self.winfo_reqheight()/2)
                ) )  #centre dialog over parent
        self.deiconify()  #geometry set, unhide
39 40
        self.wait_window()

41 42 43 44 45 46 47 48
    def create_widgets(self):
        self.name = StringVar(self.parent)
        self.fontSize = StringVar(self.parent)
        self.frameMain = Frame(self, borderwidth=2, relief=SUNKEN)
        self.frameMain.pack(side=TOP, expand=TRUE, fill=BOTH)
        self.messageInfo = Message(self.frameMain, anchor=W, justify=LEFT,
                    padx=5, pady=5, text=self.message) #,aspect=200)
        entryName = Entry(self.frameMain, textvariable=self.name, width=30)
49
        entryName.focus_set()
50 51
        self.messageInfo.pack(padx=5, pady=5) #, expand=TRUE, fill=BOTH)
        entryName.pack(padx=5, pady=5)
52

53 54 55 56 57 58 59 60 61 62 63 64 65 66
        frameButtons = Frame(self, pady=2)
        frameButtons.pack(side=BOTTOM)
        self.buttonOk = Button(frameButtons, text='Ok',
                width=8, command=self.Ok)
        self.buttonOk.pack(side=LEFT, padx=5)
        self.buttonCancel = Button(frameButtons, text='Cancel',
                width=8, command=self.Cancel)
        self.buttonCancel.pack(side=RIGHT, padx=5)

    def name_ok(self):
        ''' After stripping entered name, check that it is a  sensible
        ConfigParser file section name. Return it if it is, '' if not.
        '''
        name = self.name.get().strip()
67 68
        if not name: #no name specified
            tkMessageBox.showerror(title='Name Error',
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
69
                    message='No name specified.', parent=self)
70
        elif len(name)>30: #name too long
71
            tkMessageBox.showerror(title='Name Error',
72
                    message='Name too long. It should be no more than '+
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
73
                    '30 characters.', parent=self)
74 75
            name = ''
        elif name in self.used_names:
76
            tkMessageBox.showerror(title='Name Error',
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
77
                    message='This name is already in use.', parent=self)
78 79
            name = ''
        return name
80

81
    def Ok(self, event=None):
82 83 84
        name = self.name_ok()
        if name:
            self.result = name
85
            self.destroy()
86

87
    def Cancel(self, event=None):
88
        self.result = ''
89 90 91
        self.destroy()

if __name__ == '__main__':
92 93
    import unittest
    unittest.main('idlelib.idle_test.test_config_name', verbosity=2, exit=False)
94

95 96
    # also human test the dialog
    root = Tk()
97 98
    def run():
        dlg=GetCfgSectionNameDialog(root,'Get Name',
99 100 101 102
                "After the text entered with [Ok] is stripped, <nothing>, "
                "'abc', or more that 30 chars are errors. "
                "Close with a valid entry (printed), [Cancel], or [X]",
                {'abc'})
103
        print(dlg.result)
104 105
    Message(root, text='').pack()  # will be needed for oher dialog tests
    Button(root, text='Click to begin dialog test', command=run).pack()
106
    root.mainloop()