python - tkinter 中的 SQL 查询结果

标签 python python-3.x tkinter

我有一个 tkinter 界面,我需要在其中显示一些查询结果,并且我需要用户能够修改列并提交结果。目前为了提取查询,我正在做这样的事情:

conn = connection_info_goes_here
cur = conn.cursor()
cur.execute(query_goes_here)

这是我的查询:

SELECT id, reviewer, task, num_seconds, start_time, end_time
FROM hours
WHERE DATE(start_time) = '2014-12-18'
AND reviewer = 'john'

用户需要修改的字段是num_seconds(只是数字)。我的问题是,如何使查询结果显示在网格中以及如何使其中一个字段可以通过按钮进行修改以提交更改?

附加信息:我已经使用 exec() 以非常困惑的方式完成了此操作,并以编程方式为每个字段创建了变量。它变得非常冗长和令人困惑,我真的认为必须有更好、更简单的方法来做到这一点。

感谢任何帮助。谢谢!!

快速更新:因为这被搁置了,我将添加一张与我正在寻找的东西相似的图片:

enter image description here

当我将它们上传回数据库时,条目标签中的值必须替换右侧列中的值。

当我说我以困惑的方式做这件事时,是因为我这样做了(我能想到的唯一方式):

def cor_window():
    corrections = Tk()
    corrections.title("Corrections")
    corrections_frame = ttk.Frame(corrections)

    cor_values = []
    count=0
    cor_count=0
    for x in results:
        count2=0
        for y in results[count]:

            if count2 == 3:
                exec('int' + str(cor_count) + '=tkinter.StringVar')
                exec('int' + str(cor_count) + '_entry = ttk.Entry(corrections, width=20, textvariable=int' + str(cor_count) + ')')
                exec('int' + str(cor_count) + '_entry.grid(column=count2, row=count+2)')

                cor_count = cor_count+1
                cor_values.append('int' + str(cor_count) + '_entry')

                ttk.Label(corrections, width=20, anchor=CENTER, relief=SUNKEN, borderwidth=1, text= results[count][count2]).grid(column=count2+1, row=count+2)

            elif count2 > 3:
                ttk.Label(corrections, width=20, anchor=CENTER, relief=SUNKEN, borderwidth=1, text= results[count][count2]).grid(column=count2+1, row=count+2)
            else:
                ttk.Label(corrections, width=20, anchor=CENTER, relief=SUNKEN, borderwidth=1, text= results[count][count2]).grid(column=count2, row=count+2)
            count2=count2+1
        count=count+1

    ttk.Button(corrections, text="Done!", command=upload_cor).grid(column=0, row=1)

results 是包含查询结果的列表,upload_cor 是将更改上传到数据库的函数。由于我使用了exec,即使用户修改了输入框,我也无法使用.get()获取用户输入的内容。当我尝试使用 .get() 时,即使在输入框中输入了某些内容,我也只会得到 None

我只是需要一种不同的方法来做到这一点,再次欢迎任何想法。

最佳答案

您绝对不想使用exec,也不需要使用textvariable 选项。这两者都只会增加困惑。只需将您的小部件存储为字典,直接从条目小部件获取数据,这一切都变得非常易于管理。

这是一个工作示例:

import tkinter as tk

class Example(tk.Frame):
    def __init__(self, parent):
        tk.Frame.__init__(self, parent)
        b = tk.Button(self, text="Done!", command=self.upload_cor)
        b.pack()
        table = tk.Frame(self)
        table.pack(side="top", fill="both", expand=True)

        data = (
            (45417, "rodringof", "CSP L2 Review", 0.000394, "2014-12-19 10:08:12", "2014-12-19 10:08:12"),
            (45418, "rodringof", "CSP L2 Review", 0.000394, "2014-12-19 10:08:12", "2014-12-19 10:08:12"),
            (45419, "rodringof", "CSP L2 Review", 0.000394, "2014-12-19 10:08:12", "2014-12-19 10:08:12"),
            (45420, "rodringof", "CSP L2 Review", 0.000394, "2014-12-19 10:08:12", "2014-12-19 10:08:12"),
            (45421, "rodringof", "CSP L2 Review", 0.000394, "2014-12-19 10:08:12", "2014-12-19 10:08:12"),
            (45422, "rodringof", "CSP L2 Review", 0.000394, "2014-12-19 10:08:12", "2014-12-19 10:08:12"),
            (45423, "rodringof", "CSP L2 Review", 0.000394, "2014-12-19 10:08:12", "2014-12-19 10:08:12"),
        )

        self.widgets = {}
        row = 0
        for rowid, reviewer, task, num_seconds, start_time, end_time in (data):
            row += 1
            self.widgets[rowid] = {
                "rowid": tk.Label(table, text=rowid),
                "reviewer": tk.Label(table, text=reviewer),
                "task": tk.Label(table, text=task),
                "num_seconds_correction": tk.Entry(table),
                "num_seconds": tk.Label(table, text=num_seconds),
                "start_time": tk.Label(table, text=start_time),
                "end_time": tk.Label(table, text=start_time)
            }

            self.widgets[rowid]["rowid"].grid(row=row, column=0, sticky="nsew")
            self.widgets[rowid]["reviewer"].grid(row=row, column=1, sticky="nsew")
            self.widgets[rowid]["task"].grid(row=row, column=2, sticky="nsew")
            self.widgets[rowid]["num_seconds_correction"].grid(row=row, column=3, sticky="nsew")
            self.widgets[rowid]["num_seconds"].grid(row=row, column=4, sticky="nsew")
            self.widgets[rowid]["start_time"].grid(row=row, column=5, sticky="nsew")
            self.widgets[rowid]["end_time"].grid(row=row, column=6, sticky="nsew")

        table.grid_columnconfigure(1, weight=1)
        table.grid_columnconfigure(2, weight=1)
        # invisible row after last row gets all extra space
        table.grid_rowconfigure(row+1, weight=1)

    def upload_cor(self):
        for rowid in sorted(self.widgets.keys()):
            entry_widget = self.widgets[rowid]["num_seconds_correction"]
            new_value = entry_widget.get()
            print("%s: %s" % (rowid, new_value))

if __name__ == "__main__":
    root = tk.Tk()
    Example(root).pack(fill="both", expand=True)
    root.mainloop()

实际上,我会通过使用 add_row 方法创建一个 Table 类来实现这一点,但我不想变得太复杂。无论您是创建一个 Table 类,在一个类中完成所有操作,还是按程序进行,基本思想都是相同的 - 创建一个字典来表示您的数据。您也可以使用嵌套列表,但我发现字典更易于使用。它们也是 self 记录的,因为您通过符号名称引用事物,而不是只知道第 4 列是开始时间。

关于python - tkinter 中的 SQL 查询结果,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27558482/

相关文章:

python - kivy 相当于 Tkinter after()

python - tkinter 文本小部件 : how to indent automatically after a soft line wrap

python - Scrapy 通过 IP 地址爬取本地网站

python - 如何解决chrome headless无法下载文件的问题?

python - python 中的瓷砖游戏

python - 如何在 python 中替换/删除字符串

python - 从具有多个页面结果的网站抓取网页

Python 列表扩展和变量赋值

python - BeautifulSoup 不在 html 页面中显示某些标签

python-3.x - 通过 Tkinter 中的列表变量修改列表框条目