Передача значений между классами для Tkinter - PullRequest
0 голосов
/ 05 ноября 2018

Я работаю над небольшим графическим интерфейсом для запроса информации из нашей CMDB для отображения для пользователей. Проблема, с которой я столкнулся, заключается в том, что после события в одном классе (кнопка) я хочу обновить комбинированный список в другом классе. Я думал, что должен использовать tk.StringVar () для передачи списка, но в поле со списком отображается только значение «PC_VAR #» и не обновляется. Может ли кто-нибудь предложить какую-либо помощь, пожалуйста?

#!/usr/bin/python

import Tkinter as tk
import ttk
import signal


class LoginUI:
    def __init__(self, frame):
        self.frame = frame

        # Set default list entry
        self.dc_list = tk.StringVar()
        self.dc_list.set(['Login first'])

        # Add a button to log in
        self.button = tk.Button(self.frame, text='Login', command=self.change_combobox)
        self.button.grid(column=0, row=0, pady=5)

    def change_combobox(self):
        # Change combobox values
        dc_list = ['Site_1', 'Site_2', 'Site_3']
        self.dc_list.set(dc_list)


class QueryUI:
    def __init__(self, frame, dc_list):
        self.frame = frame
        self.dc = tk.StringVar()
        self.dc_list = tk.StringVar()
        self.dc_list.set(dc_list)

        # Create site combobox
        tk.Label(self.frame, text='Site:').grid(column=0, row=0, sticky="w")
        self.dc_combobox = ttk.Combobox(
            self.frame,
            textvariable=self.dc,
            width=20,
            state='readonly'
        )
        self.dc_combobox['values'] = self.dc_list.get()
        self.dc_combobox.grid(column=1, row=0, sticky="w")


class App:
    def __init__(self, root):
        self.root = root
        self.root.title('Logging Handler')
        self.root.columnconfigure(0, weight=1)
        self.root.rowconfigure(0, weight=1)

        # Create the left frame panels
        left_frame = tk.Frame(self.root, padx=5, pady=5)
        login_frame = tk.LabelFrame(left_frame, text="Login", borderwidth=2, relief="groove", padx=5, pady=5)
        query_frame = tk.LabelFrame(left_frame, text="Query", borderwidth=2, relief="groove", padx=5, pady=5)

        # Align frames
        left_frame.grid(row=0, column=0, sticky="nw")
        login_frame.grid(row=0, column=0, pady=5, sticky="nw")
        query_frame.grid(row=1, column=0, pady=5, sticky="nw")

        # Initialize all frames
        self.login = LoginUI(login_frame)
        self.query = QueryUI(query_frame, self.login.dc_list)
        self.root.protocol('WM_DELETE_WINDOW', self.quit)
        self.root.bind('<Control-q>', self.quit)
        signal.signal(signal.SIGINT, self.quit)

    def quit(self, *args):
        self.root.destroy()


def main():
    root = tk.Tk()
    app = App(root)
    app.root.mainloop()


if __name__ == '__main__':
    main()

1 Ответ

0 голосов
/ 05 ноября 2018

Что я хотел бы сделать, это передать управляющий класс (приложение) классу, необходимому для обновления поля со списком. Таким образом, мы можем взаимодействовать с ним позже, если это будет необходимо. Передав self из App в LoginUI, мы можем взаимодействовать с атрибутами и методами класса App из LoginUI. Это упрощает обновление комбинированного списка.

Это говорит о том, что вам действительно не нужны все StringVars. Просто пройдите список как список, и вам будет хорошо идти.

import Tkinter as tk
import ttk
import signal


class LoginUI:
    def __init__(self, controller, frame):
        self.controller = controller
        self.frame = frame
        self.dc_list = ['Login first']
        self.button = tk.Button(self.frame, text='Login', command=self.change_combobox)
        self.button.grid(column=0, row=0, pady=5)

    def change_combobox(self):
        self.controller.query.dc_combobox['values'] = ['Site_1', 'Site_2', 'Site_3']
        self.controller.query.dc.set('Site_1')


class QueryUI:
    def __init__(self, frame, dc_list):
        self.frame = frame
        self.dc = tk.StringVar()
        tk.Label(self.frame, text='Site:').grid(column=0, row=0, sticky="w")
        self.dc_combobox = ttk.Combobox(self.frame, textvariable=self.dc, width=20, state='readonly')
        self.dc_combobox['values'] = dc_list
        self.dc_combobox.grid(column=1, row=0, sticky="w")


class App:
    def __init__(self, root):
        self.root = root
        self.root.title('Logging Handler')
        self.root.columnconfigure(0, weight=1)
        self.root.rowconfigure(0, weight=1)

        left_frame = tk.Frame(self.root, padx=5, pady=5)
        login_frame = tk.LabelFrame(left_frame, text="Login", borderwidth=2, relief="groove", padx=5, pady=5)
        query_frame = tk.LabelFrame(left_frame, text="Query", borderwidth=2, relief="groove", padx=5, pady=5)
        left_frame.grid(row=0, column=0, sticky="nw")
        login_frame.grid(row=0, column=0, pady=5, sticky="nw")
        query_frame.grid(row=1, column=0, pady=5, sticky="nw")
        self.login = LoginUI(self, login_frame)
        self.query = QueryUI(query_frame, self.login.dc_list)
        self.root.protocol('WM_DELETE_WINDOW', self.quit)
        self.root.bind('<Control-q>', self.quit)
        signal.signal(signal.SIGINT, self.quit)

    def quit(self, *args):
        self.root.destroy()


if __name__ == '__main__':
    root = tk.Tk()
    app = App(root)
    app.root.mainloop()
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...