guitk


Nameguitk JSON
Version 0.4.5 PyPI version JSON
download
home_pageNone
SummaryPython GUI Toolkit for Tk (guitk): simplify the layout and construction of tkinter graphical user interfaces in python.
upload_time2024-09-03 12:19:51
maintainerNone
docs_urlNone
authorRhet Turnbull
requires_python<4.0,>=3.9
licenseMIT
keywords
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            <!--* DO NOT EDIT README.md, instead edit README.mdpp and process with MarkdownPP using doit (see dodo.py) -->

# Python GUI Toolkit for TK (GUITk)

## Synopsis

GUITk is a declarative framework for building nice-looking, cross-platform GUIs with [tkinter](https://docs.python.org/3/library/tkinter.html) inspired by [SwiftUI](https://developer.apple.com/documentation/swiftui).

GUITk allows you to build complete GUI applications with a few lines of code. GUITk makes it easy to layout your GUI elements and respond to events using a declarative syntax. Because GUITk is built on top of tkinter, you can access the underlying tkinter API if you need to but for many use cases, you can build your GUI without needing to know much about tkinter.

GUITk apps are built by subclasses the `guitk.Window` class. Your GUI elements are layed out using a `guitk.HLayout` (horizontal layout) or `guitk.VLayout` (vertical layout) object which takes care of placing all widgets in the window using a declarative syntax. This is much simpler than using the underlying tkinter [grid manager](https://tkdocs.com/shipman/grid.html) or [pack](https://dafarry.github.io/tkinterbook/pack.htm) geometry managers.

GUITk is in early beta but is in constant development so check back frequently if this interests you or open an issue to start a conversation about what pain points this project could help you solve!

Documentation is available at [GUITk](https://rhettbull.github.io/guitk/).

GUITk has no dependencies outside of the Python standard library. (Though if [Pillow](https://pypi.org/project/Pillow/) is installed, GUITk will use it to load images.)

## Code Example

### Simple HLayout

<!--[[[cog
import os
os.system("python3 utils/screenshot.py examples/hello.py HelloWindow docs/images/hello.py.png --overwrite")
]]]-->
<!--[[[end]]]-->
![hello.py example](https://raw.githubusercontent.com/RhetTbull/guitk/main/docs/images/hello.py.png "Hello World example")

```python
"""Simple Hello World example using guitk """

import guitk as ui


# subclass guitk.Window as the starting point for your app's main window
class HelloWindow(ui.Window):
    def config(self):
        """Configure the window"""

        # set the window title
        self.title = "Hello, World"

        # define a layout for the window
        # the layout manager will automatically add widgets to the window
        with ui.HLayout():
            ui.Label("What's your name?")
            ui.Entry(key="name", focus=True)
            ui.Button("Ok", key="ok")

    @ui.on(key="ok")
    def on_ok(self, event: ui.Event):
        """Handle the Ok button click"""
        print("Hello, ", self.get("name").value)


# run your event loop
if __name__ == "__main__":
    HelloWindow().run()
```

## Motivation

The goal of GUITk is to make it very easy to create simple and attractive GUI apps with python. It borrows ideas from several other libraries include [PySimpleGUI](https://www.pysimplegui.org/en/latest/), [SwiftUI](https://developer.apple.com/documentation/swiftui), [textual](https://github.com/Textualize/textual), and [applepy](https://github.com/eduardohleite/applepy). GUITk builds on [tkinter](https://docs.python.org/3/library/tkinter.html) which ships with the Python standard library and works across many platforms. tkinter is a mature and powerful GUI framework but requires a fair bit of boiler plate and understanding of the underlying framework to use effectively. GUITk attempts to simplify this by providing a higher level interface to tkinter while still allowing you to access the underlying tkinter API if you need to.

Though you can build simple apps without knowing much about tkinter, a basic understanding of tkinter will be helpful when building with GUITk. I highly recommend Mark Roseman's excellent [Modern Tkinter for Busy Python Developers](https://tkdocs.com/book.html) book as a starting point.

## Is GUITk for you?

GUITk is still beta software so I would not use it for anything really important. That said, I am using GUITk regularly for small projects and it is working well for me. If you want to build a simple GUI for your Python app using a minimal amount of code and without having to think too much about the geometry of your window, GUITk might be worth trying. GUITk has no dependencies outside the Python standard library so it's easy to add to your project and it should work across many platforms.

If you need complex widgets or native OS features like drag & drop, notifications, etc. or a native OS look & feel, then GUITk is not the best choice as it's built on tkinter which does not offer these features.

## Supported Platforms

GUITk requires Python 3.9+ and is currently tested on macOS Ventura (13.x) and macOS Catalina (10.15.7). It should work on any platform that supports tkinter but this has not been tested. If you run into issues on a particular platform, please open an issue.

## Installation

* `python3 -m pip install guitk`

## Anatomy of a guitk program

<!--[[[cog
import os
os.system("python3 utils/screenshot.py examples/hello2.py HelloWorld docs/images/hello2.py.png --overwrite")
]]]-->
<!--[[[end]]]-->
![hello2.py example](https://raw.githubusercontent.com/RhetTbull/guitk/main/docs/images/hello2.py.png "Hello World example")

```python
"""Hello World example using guitk """

import guitk as ui


class HelloWorld(ui.Window):
    # subclass guitk.Window as the starting point for your app's main window
    def config(self):
        # Your Window class needs to define a config() method that describes the layout, title, etc for your app
        # config() is called by the Window class when the Window is being created

        # Title for the window
        self.title = "Hello, World"

        # optionally set size as a tuple of (width, height)
        self.size = (320, 240)

        # you can also use self.geometry for consistency with tkinter
        # self.geometry = "320x240"

        # Define the window's contents
        # guitk.Label corresponds to a tkinter.ttk.Label, etc.
        # optionally provide a unique key to each element to easily reference the element later
        # use a HLayout or VLayout class to define the layout of the window
        # HLayout arranges widgets horizontally, VLayout arranges widgets vertically
        with ui.VLayout():
            # use a VLayout to stack the widgets vertically
            # standard tkinter layout options such as sticky and weight are supported
            ui.Label("What's your name?", sticky="ew", anchor="center", weightx=1)
            # each widget can be assigned a key, which should be unique, to easily reference the widget later
            # set focus=True so the Entry box has focus when the window is displayed
            ui.Entry(key="entry_name", focus=True, weightx=1, sticky="ew")
            ui.Label("", width=40, key="output")
            with ui.HStack():
                # align these two buttons in a horizontal row using HStack
                ui.Button("Ok")
                ui.Button("Quit")

    # Every Window class has 3 special methods that can be overridden to provide custom behavior
    # you do not need to provide any of these methods if you do not need to customize the default behavior
    # (the default behavior is to do nothing)
    # These special methods are: setup(), teardown(), and handle_event()

    def setup(self):
        """Perform any initialization needed before the Window is displayed"""
        # your setup() method is called by the Window class after config() just before the Window is displayed
        # use this to initialize any internal state you need
        # you do not need to provide a setup() method if no initialization is needed
        print("setup")

    def teardown(self):
        """Perform any cleanup needed before destroying the window"""
        # your teardown() method is called by the Window class after the Window is closed
        # use this to clean up before the Window is destroyed
        # you do not need to provide a teardown() method if no cleanup is needed
        print("teardown")

    def handle_event(self, event: ui.Event):
        """handle_event() is called by the Window class when an event occurs"""
        # you do not need to provide a handle_event() method if you prefer to use
        # the @on decorator to bind functions to events (see below)
        # handle_event() is a useful place to put code that needs to run for every event
        # or for use during debugging
        print(f"handle_event: {event}")

    @ui.on(key="Quit")
    def on_quit(self):
        # return the value of the Entry box
        self.quit(self["entry_name"].value)

    @ui.on(key="Ok")
    @ui.on(event_type=ui.EventType.EntryReturn)
    def on_ok(self):
        # User pressed the OK button or the Return key inside the Entry box
        # the @on decorator can be used to bind a function to an event
        # @on can be repeated to bind the function to multiple events
        # set the output Label to the value of the Entry box
        # individual widgets can be accessed by their key; the window object acts as a dictionary of widgets
        greeting = f"Hello {self['entry_name'].value}! Thanks for trying guitk."

        # if you prefer, you can use get() instead of the dictionary syntax
        self.get("output").value = greeting


if __name__ == "__main__":
    # instantiate your Window class and run it
    name = HelloWorld().run()
    print(f"Hello {name}")
```

## Documentation

Documentation can be found [here](https://rhettbull.github.io/guitk/).  Take a look at the [examples](https://github.com/RhetTbull/guitk/tree/main/examples) directory for a number of examples on use of various widgets.

## Testing

There are currently no automated tests as I haven't figured out how to do these with tkinter. I am working on adding tests and there are several tests that run with `pytest` in the `tests` directory.  These are not automated and require user interaction.

You can also run `python3 -m guitk` which opens a window with examples of all the widgets. I find this useful for quick testing of layout and widget behavior.

## Contributors

Contributions welcome! If this project interests you, open an Issue or send a PR!

## TODO

* [x] Basic prototype
* [x] Frame
* [x] Label
* [x] Entry
* [x] Button
* [x] Checkbutton
* [x] Radiobutton
* [x] Text
* [x] ScrolledText
* [x] Treeview
* [x] Listbox
* [x] Combobox
* [x] Spinner
* [x] Other widgets
* [x] Menus
* [x] Tooltips
* [ ] Documentation
* [x] Add docstrings
* [x] Add type hints to public API
* [ ] Tests

## License

Licensed under the MIT License.

## See Also

* [Textual](https://github.com/Textualize/textual) - An amazing Python framework for building user interfaces in the terminal.
* [PySimpleGUI](https://www.PySimpleGUI.org) - A Python GUI Framework.
* [applepy](https://github.com/eduardohleite/applepy) - A declarative GUI framework for developing native macOS applications in Python 3.


            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "guitk",
    "maintainer": null,
    "docs_url": null,
    "requires_python": "<4.0,>=3.9",
    "maintainer_email": null,
    "keywords": null,
    "author": "Rhet Turnbull",
    "author_email": "rturnbull+git@gmail.com",
    "download_url": "https://files.pythonhosted.org/packages/4c/23/a7faa2c1770e8b579ecf915d123e720da66825587d24b9930be3610bf0b6/guitk-0.4.5.tar.gz",
    "platform": null,
    "description": "<!--* DO NOT EDIT README.md, instead edit README.mdpp and process with MarkdownPP using doit (see dodo.py) -->\n\n# Python GUI Toolkit for TK (GUITk)\n\n## Synopsis\n\nGUITk is a declarative framework for building nice-looking, cross-platform GUIs with [tkinter](https://docs.python.org/3/library/tkinter.html) inspired by [SwiftUI](https://developer.apple.com/documentation/swiftui).\n\nGUITk allows you to build complete GUI applications with a few lines of code. GUITk makes it easy to layout your GUI elements and respond to events using a declarative syntax. Because GUITk is built on top of tkinter, you can access the underlying tkinter API if you need to but for many use cases, you can build your GUI without needing to know much about tkinter.\n\nGUITk apps are built by subclasses the `guitk.Window` class. Your GUI elements are layed out using a `guitk.HLayout` (horizontal layout) or `guitk.VLayout` (vertical layout) object which takes care of placing all widgets in the window using a declarative syntax. This is much simpler than using the underlying tkinter [grid manager](https://tkdocs.com/shipman/grid.html) or [pack](https://dafarry.github.io/tkinterbook/pack.htm) geometry managers.\n\nGUITk is in early beta but is in constant development so check back frequently if this interests you or open an issue to start a conversation about what pain points this project could help you solve!\n\nDocumentation is available at [GUITk](https://rhettbull.github.io/guitk/).\n\nGUITk has no dependencies outside of the Python standard library. (Though if [Pillow](https://pypi.org/project/Pillow/) is installed, GUITk will use it to load images.)\n\n## Code Example\n\n### Simple HLayout\n\n<!--[[[cog\nimport os\nos.system(\"python3 utils/screenshot.py examples/hello.py HelloWindow docs/images/hello.py.png --overwrite\")\n]]]-->\n<!--[[[end]]]-->\n![hello.py example](https://raw.githubusercontent.com/RhetTbull/guitk/main/docs/images/hello.py.png \"Hello World example\")\n\n```python\n\"\"\"Simple Hello World example using guitk \"\"\"\n\nimport guitk as ui\n\n\n# subclass guitk.Window as the starting point for your app's main window\nclass HelloWindow(ui.Window):\n    def config(self):\n        \"\"\"Configure the window\"\"\"\n\n        # set the window title\n        self.title = \"Hello, World\"\n\n        # define a layout for the window\n        # the layout manager will automatically add widgets to the window\n        with ui.HLayout():\n            ui.Label(\"What's your name?\")\n            ui.Entry(key=\"name\", focus=True)\n            ui.Button(\"Ok\", key=\"ok\")\n\n    @ui.on(key=\"ok\")\n    def on_ok(self, event: ui.Event):\n        \"\"\"Handle the Ok button click\"\"\"\n        print(\"Hello, \", self.get(\"name\").value)\n\n\n# run your event loop\nif __name__ == \"__main__\":\n    HelloWindow().run()\n```\n\n## Motivation\n\nThe goal of GUITk is to make it very easy to create simple and attractive GUI apps with python. It borrows ideas from several other libraries include [PySimpleGUI](https://www.pysimplegui.org/en/latest/), [SwiftUI](https://developer.apple.com/documentation/swiftui), [textual](https://github.com/Textualize/textual), and [applepy](https://github.com/eduardohleite/applepy). GUITk builds on [tkinter](https://docs.python.org/3/library/tkinter.html) which ships with the Python standard library and works across many platforms. tkinter is a mature and powerful GUI framework but requires a fair bit of boiler plate and understanding of the underlying framework to use effectively. GUITk attempts to simplify this by providing a higher level interface to tkinter while still allowing you to access the underlying tkinter API if you need to.\n\nThough you can build simple apps without knowing much about tkinter, a basic understanding of tkinter will be helpful when building with GUITk. I highly recommend Mark Roseman's excellent [Modern Tkinter for Busy Python Developers](https://tkdocs.com/book.html) book as a starting point.\n\n## Is GUITk for you?\n\nGUITk is still beta software so I would not use it for anything really important. That said, I am using GUITk regularly for small projects and it is working well for me. If you want to build a simple GUI for your Python app using a minimal amount of code and without having to think too much about the geometry of your window, GUITk might be worth trying. GUITk has no dependencies outside the Python standard library so it's easy to add to your project and it should work across many platforms.\n\nIf you need complex widgets or native OS features like drag & drop, notifications, etc. or a native OS look & feel, then GUITk is not the best choice as it's built on tkinter which does not offer these features.\n\n## Supported Platforms\n\nGUITk requires Python 3.9+ and is currently tested on macOS Ventura (13.x) and macOS Catalina (10.15.7). It should work on any platform that supports tkinter but this has not been tested. If you run into issues on a particular platform, please open an issue.\n\n## Installation\n\n* `python3 -m pip install guitk`\n\n## Anatomy of a guitk program\n\n<!--[[[cog\nimport os\nos.system(\"python3 utils/screenshot.py examples/hello2.py HelloWorld docs/images/hello2.py.png --overwrite\")\n]]]-->\n<!--[[[end]]]-->\n![hello2.py example](https://raw.githubusercontent.com/RhetTbull/guitk/main/docs/images/hello2.py.png \"Hello World example\")\n\n```python\n\"\"\"Hello World example using guitk \"\"\"\n\nimport guitk as ui\n\n\nclass HelloWorld(ui.Window):\n    # subclass guitk.Window as the starting point for your app's main window\n    def config(self):\n        # Your Window class needs to define a config() method that describes the layout, title, etc for your app\n        # config() is called by the Window class when the Window is being created\n\n        # Title for the window\n        self.title = \"Hello, World\"\n\n        # optionally set size as a tuple of (width, height)\n        self.size = (320, 240)\n\n        # you can also use self.geometry for consistency with tkinter\n        # self.geometry = \"320x240\"\n\n        # Define the window's contents\n        # guitk.Label corresponds to a tkinter.ttk.Label, etc.\n        # optionally provide a unique key to each element to easily reference the element later\n        # use a HLayout or VLayout class to define the layout of the window\n        # HLayout arranges widgets horizontally, VLayout arranges widgets vertically\n        with ui.VLayout():\n            # use a VLayout to stack the widgets vertically\n            # standard tkinter layout options such as sticky and weight are supported\n            ui.Label(\"What's your name?\", sticky=\"ew\", anchor=\"center\", weightx=1)\n            # each widget can be assigned a key, which should be unique, to easily reference the widget later\n            # set focus=True so the Entry box has focus when the window is displayed\n            ui.Entry(key=\"entry_name\", focus=True, weightx=1, sticky=\"ew\")\n            ui.Label(\"\", width=40, key=\"output\")\n            with ui.HStack():\n                # align these two buttons in a horizontal row using HStack\n                ui.Button(\"Ok\")\n                ui.Button(\"Quit\")\n\n    # Every Window class has 3 special methods that can be overridden to provide custom behavior\n    # you do not need to provide any of these methods if you do not need to customize the default behavior\n    # (the default behavior is to do nothing)\n    # These special methods are: setup(), teardown(), and handle_event()\n\n    def setup(self):\n        \"\"\"Perform any initialization needed before the Window is displayed\"\"\"\n        # your setup() method is called by the Window class after config() just before the Window is displayed\n        # use this to initialize any internal state you need\n        # you do not need to provide a setup() method if no initialization is needed\n        print(\"setup\")\n\n    def teardown(self):\n        \"\"\"Perform any cleanup needed before destroying the window\"\"\"\n        # your teardown() method is called by the Window class after the Window is closed\n        # use this to clean up before the Window is destroyed\n        # you do not need to provide a teardown() method if no cleanup is needed\n        print(\"teardown\")\n\n    def handle_event(self, event: ui.Event):\n        \"\"\"handle_event() is called by the Window class when an event occurs\"\"\"\n        # you do not need to provide a handle_event() method if you prefer to use\n        # the @on decorator to bind functions to events (see below)\n        # handle_event() is a useful place to put code that needs to run for every event\n        # or for use during debugging\n        print(f\"handle_event: {event}\")\n\n    @ui.on(key=\"Quit\")\n    def on_quit(self):\n        # return the value of the Entry box\n        self.quit(self[\"entry_name\"].value)\n\n    @ui.on(key=\"Ok\")\n    @ui.on(event_type=ui.EventType.EntryReturn)\n    def on_ok(self):\n        # User pressed the OK button or the Return key inside the Entry box\n        # the @on decorator can be used to bind a function to an event\n        # @on can be repeated to bind the function to multiple events\n        # set the output Label to the value of the Entry box\n        # individual widgets can be accessed by their key; the window object acts as a dictionary of widgets\n        greeting = f\"Hello {self['entry_name'].value}! Thanks for trying guitk.\"\n\n        # if you prefer, you can use get() instead of the dictionary syntax\n        self.get(\"output\").value = greeting\n\n\nif __name__ == \"__main__\":\n    # instantiate your Window class and run it\n    name = HelloWorld().run()\n    print(f\"Hello {name}\")\n```\n\n## Documentation\n\nDocumentation can be found [here](https://rhettbull.github.io/guitk/).  Take a look at the [examples](https://github.com/RhetTbull/guitk/tree/main/examples) directory for a number of examples on use of various widgets.\n\n## Testing\n\nThere are currently no automated tests as I haven't figured out how to do these with tkinter. I am working on adding tests and there are several tests that run with `pytest` in the `tests` directory.  These are not automated and require user interaction.\n\nYou can also run `python3 -m guitk` which opens a window with examples of all the widgets. I find this useful for quick testing of layout and widget behavior.\n\n## Contributors\n\nContributions welcome! If this project interests you, open an Issue or send a PR!\n\n## TODO\n\n* [x] Basic prototype\n* [x] Frame\n* [x] Label\n* [x] Entry\n* [x] Button\n* [x] Checkbutton\n* [x] Radiobutton\n* [x] Text\n* [x] ScrolledText\n* [x] Treeview\n* [x] Listbox\n* [x] Combobox\n* [x] Spinner\n* [x] Other widgets\n* [x] Menus\n* [x] Tooltips\n* [ ] Documentation\n* [x] Add docstrings\n* [x] Add type hints to public API\n* [ ] Tests\n\n## License\n\nLicensed under the MIT License.\n\n## See Also\n\n* [Textual](https://github.com/Textualize/textual) - An amazing Python framework for building user interfaces in the terminal.\n* [PySimpleGUI](https://www.PySimpleGUI.org) - A Python GUI Framework.\n* [applepy](https://github.com/eduardohleite/applepy) - A declarative GUI framework for developing native macOS applications in Python 3.\n\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "Python GUI Toolkit for Tk (guitk): simplify the layout and construction of tkinter graphical user interfaces in python.",
    "version": "0.4.5",
    "project_urls": null,
    "split_keywords": [],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "2c7523894c033f554531b653b0bcaca9cfa6e70d404719cacb8f106d9ddf1f3a",
                "md5": "279cc53c007c7e9adad2f91ae6b6a9da",
                "sha256": "544ff335a96e276f99cac2efe07ea456b64f7f7b72371d045bb7c0f92c1307ab"
            },
            "downloads": -1,
            "filename": "guitk-0.4.5-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "279cc53c007c7e9adad2f91ae6b6a9da",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": "<4.0,>=3.9",
            "size": 80619,
            "upload_time": "2024-09-03T12:19:49",
            "upload_time_iso_8601": "2024-09-03T12:19:49.994636Z",
            "url": "https://files.pythonhosted.org/packages/2c/75/23894c033f554531b653b0bcaca9cfa6e70d404719cacb8f106d9ddf1f3a/guitk-0.4.5-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "4c23a7faa2c1770e8b579ecf915d123e720da66825587d24b9930be3610bf0b6",
                "md5": "bcc871a4c1d0d2deeb388b70510eb85b",
                "sha256": "2e077a026309811cfddeac565e98efe4cd2574e4cf0a24512ba33845927bd7f4"
            },
            "downloads": -1,
            "filename": "guitk-0.4.5.tar.gz",
            "has_sig": false,
            "md5_digest": "bcc871a4c1d0d2deeb388b70510eb85b",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": "<4.0,>=3.9",
            "size": 59606,
            "upload_time": "2024-09-03T12:19:51",
            "upload_time_iso_8601": "2024-09-03T12:19:51.393277Z",
            "url": "https://files.pythonhosted.org/packages/4c/23/a7faa2c1770e8b579ecf915d123e720da66825587d24b9930be3610bf0b6/guitk-0.4.5.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-09-03 12:19:51",
    "github": false,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "lcname": "guitk"
}
        
Elapsed time: 1.97675s