Виджет для ноутбука в Tkinter

Немного поиграв с Tkinter и wxPython, мне очень нравится Tkinter с точки зрения того, насколько выглядит мой исходный код. Однако, похоже, у него не так много функций; в частности, он не имеет вкладок (например, вкладки в верхней части окна Firefox).

Небольшой поисковый запрос по этому вопросу предлагает несколько предложений. Там запись в поваренной книге с классом, позволяющим использовать вкладки, но это очень примитивно. Там также Python megawidgets на SourceForge, хотя это кажется очень старым и давало мне ошибки во время установки.

Есть ли у кого-нибудь опыт создания графических интерфейсов с вкладками в Tkinter? Что вы использовали? Или это просто случай, когда любой, кому нужны более мощные компоненты окон, должен использовать wxPython?

Ответ 1

В последних версиях Python ( > 2.7) вы можете использовать модуль ttk, который обеспечивает доступ к набору виджета Tk themed, который был введен в Tk 8.5.

Здесь вы импортируете ttk в Python 2:

import ttk

help(ttk.Notebook)

В Python 3 модуль ttk поставляется со стандартными дистрибутивами как подмодуль tkinter.

Вот простой рабочий пример, основанный на примере с TkDocs:

from tkinter import ttk
import tkinter as tk
from tkinter.scrolledtext import ScrolledText


def demo():
    root = tk.Tk()
    root.title("ttk.Notebook")

    nb = ttk.Notebook(root)

    # adding Frames as pages for the ttk.Notebook 
    # first page, which would get widgets gridded into it
    page1 = ttk.Frame(nb)

    # second page
    page2 = ttk.Frame(nb)
    text = ScrolledText(page2)
    text.pack(expand=1, fill="both")

    nb.add(page1, text='One')
    nb.add(page2, text='Two')

    nb.pack(expand=1, fill="both")

    root.mainloop()

if __name__ == "__main__":
    demo()

Другой альтернативой является использование виджета NoteBook из библиотеки tkinter.tix. Чтобы использовать tkinter.tix, вы должны установить виджеты Tix, как правило, рядом с установкой виджетов Tk. Чтобы протестировать вашу установку, попробуйте выполнить следующее:

from tkinter import tix
root = tix.Tk()
root.tk.eval('package require Tix')

Для получения дополнительной информации ознакомьтесь с этой веб-страницей на веб-сайте PSF.

Обратите внимание, что Tix довольно старый и не очень хорошо поддерживается, поэтому лучше всего выбрать ttk.Notebook.

Ответ 2

Если кто-то еще смотрит, у меня это работает как Tab в tkinter. Играйте с кодом, чтобы он работал так, как вы хотите (например, вы можете добавить кнопку, чтобы добавить новую вкладку):

from tkinter import *

class Tabs(Frame):

    """Tabs for testgen output"""

    def __init__(self, parent):
        super(Tabs, self).__init__()
        self.parent = parent
        self.columnconfigure(10, weight=1)
        self.rowconfigure(3, weight=1)
        self.curtab = None
        self.tabs = {}
        self.addTab()                
        self.pack(fill=BOTH, expand=1, padx=5, pady=5)

    def addTab(self):
        tabslen = len(self.tabs)
        if tabslen < 10:
            tab = {}
            btn = Button(self, text="Tab "+str(tabslen), command=lambda: self.raiseTab(tabslen))
            btn.grid(row=0, column=tabslen, sticky=W+E)

            textbox = Text(self.parent)
            textbox.grid(row=1, column=0, columnspan=10, rowspan=2, sticky=W+E+N+S, in_=self)

            # Y axis scroll bar
            scrollby = Scrollbar(self, command=textbox.yview)
            scrollby.grid(row=7, column=5, rowspan=2, columnspan=1, sticky=N+S+E)
            textbox['yscrollcommand'] = scrollby.set

            tab['id']=tabslen
            tab['btn']=btn
            tab['txtbx']=textbox
            self.tabs[tabslen] = tab
            self.raiseTab(tabslen)

    def raiseTab(self, tabid):
        print(tabid)
        print("curtab"+str(self.curtab))
        if self.curtab!= None and self.curtab != tabid and len(self.tabs)>1:
                self.tabs[tabid]['txtbx'].lift(self)
                self.tabs[self.curtab]['txtbx'].lower(self)
        self.curtab = tabid


def main():
    root = Tk()
    root.geometry("600x450+300+300")
    t = Tabs(root)
    t.addTab()
    root.mainloop()

if __name__ == '__main__':
    main()

Ответ 3

Пока это может не помочь вам, tk 8.5 поставляется с расширенным набором виджетов. Этот расширенный набор доступен с tk 8.4 с помощью расширения, известного как "плитка". В расширенный набор виджетов входит виджет для ноутбука. К сожалению, в это время Tkinter по умолчанию использует довольно старую версию Tk, которая не поставляется с этими виджетами.

Были предприняты попытки сделать плитки доступными для Tkinter. Проверьте http://tkinter.unpythonic.net/wiki/TileWrapper. Для другого подобного усилия см. http://pypi.python.org/pypi/pyttk. Кроме того, для удобства просмотра этих виджетов (в Ruby, Perl и Tcl) см. http://www.tkdocs.com/.

Tk 8.5 - это огромное улучшение по сравнению с запасом Tk. В нем представлены несколько новых виджетов, собственные виджеты и движок для тематики. Надеюсь, он скоро будет доступен по умолчанию в Tkinter. Слишком плохо, что мир Python отстает от других языков.

update: последние версии Python теперь включают в себя поддержку тематических виджетов из коробки. _

Ответ 4

"Или это просто случай, когда любой, кому нужны более мощные компоненты окон, должен использовать wxPython?"
Короткий ответ: да.

Длинный ответ: Для вашего кода wxPython может потребоваться некоторая практика, чтобы чувствовать себя "чистым", но это лучше и намного мощнее, чем Tkinter. Вы также получите лучшую поддержку, поскольку в наши дни все больше людей используют ее.

Ответ 5

Какие у вас проблемы с pmw? Это старый, да, но это чистый питон, поэтому он должен работать.

Обратите внимание, что Tix не работает с py2exe, если это проблема для вас.