Metadata-Version: 2.1
Name: lk-qtquick-scaffold
Version: 2.0.0a1
Summary: A flexible toolset to improve QML coding experience for PyQt/PySide development.
Home-page: https://github.com/likianta/lk-qtquick-scaffold
License: MIT
Author: Likianta
Author-email: likianta@foxmail.com
Requires-Python: >=3.8,<4.0
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Provides-Extra: qt5
Provides-Extra: qt6
Requires-Dist: lk-lambdex
Requires-Dist: lk-logger
Requires-Dist: lk-utils
Requires-Dist: pyside2; extra == "qt5"
Requires-Dist: pyside6; extra == "qt6"
Requires-Dist: qtpy
Description-Content-Type: text/markdown

# LK QtQuick Scaffold

Using Python and QtQuick QML to build desktop applications from a series of
predefined tools.

# Highlights

- A layout engine to extend QML layouts.
- Integrate Qt logging with python console.
- Execute Python snippet in QML, and vice versa.
- Easy-to-use register handler to register Python functions to QML side.
- Well type-annotated signal and slot (and more elegant writing style!)
- Hot loader to verify layout changing on the fly.
- Stylesheet manager to thoroughly control application appearance (color, size, 
  motion, typography, and so on).
- A set of built-in widgets/themes to quickly produce beautiful user interface.

# How to install

`lk-qtquick-scaffold` requires Python 3.8+ interpreter version.

Use pip install:

```shell
# the 1.x version. (1.3.0)
pip install lk-qtquick-scaffold

# the next big version is coming soon. currently i've released a preview version.
pip install lk-qtquick-scaffold>=2.0.0a0
```

Note: many features in this document are based on 2.0, currently the 2.0 formal
release is still in progress, you may install the alpha version to taste the
newest features.

## Install Qt backend

Installing lk-qtquick-scaffold doesn't include Python for Qt's library. You 
need to manually install one of the follows:

```shell
# choose one to install
pip install pyside6
pip install pyqt6
pip install pyside2
pip install pyqt5
```

lk-qtquick-scaffold auto detects the Qt backend you've installed (you can also 
explicitly set the specific one), it uses 
[qtpy](https://github.com/spyder-ide/qtpy) to provide an uniform layer 
overrides PySide6/PyQt6/PySide2/PyQt5.

# Examples quick through

## Hello world

view.qml

```qml
import QtQuick
import QtQuick.Window

Window {
    visible: true
    width: 400
    height: 300
    
    Text {
        anchors.centerIn: parent
        text: 'Hello world!'
    }
}
```

main.py

```python
from lk_qtquick_scaffold import app
app.run('view.qml')
```

![](examples/hello_world/screenshot.png)

## Hot loader

The `app.run` method accepts `debug` (bool type) parameter, to enable hot 
loader mode:

```python
from lk_qtquick_scaffold import app
app.run('view.qml', debug=True)
```

It starts a floating window that includes a button "RELOAD", each time when you 
modify "view.qml", click "RELOAD" to refresh your GUI:

![](examples/hot_reloader/screenshot.gif)

BTW you can run "view.qml" in command line:

```shell
# see help
py -m lk_qtquick_scaffold -h

# run
py -m lk_qtquick_scaffold view.qml

# run in debug mode
py -m lk_qtquick_scaffold view.qml --debug
```

It has the same result like above "main.py" does.

## Register funtions to QML

```python
from lk_qtquick_scaffold import QObject, app, pyside, slot

class MyObject(QObject):
    @slot(result=str)
    def hello(self):
        return 'hello world'

# 1. register QObject subclasses by `app.register`
app.register(MyObject())  
#   it will be available as 'MyObject' in QML side.
# 1.1. or use alias
app.register(MyObject(), name='PyObject')
#   it will be available as 'PyObject' in QML side.

# 2. register regular function by `pyside.register`.
def foo(a: int, b: int, c: int):
    return a + b + c
pyside.register(foo)
#   it will be available as 'pyside.call("foo", ...)' in QML side.
# 2.1. or use alias
pyside.register(foo, name='add_abc')
#   it will be available as 'pyside.call("add_abc", ...)' in QML side.
```

view.qml

```qml
import QtQuick

Item {
    Component.onCompleted: {
        console.log(MyObject.hello())  // -> 'hello world'
        console.log(PyObject.hello())  // -> 'hello world'

        console.log(pyside.call("foo", [1, 2, 3]))  // -> 6
        console.log(pyside.call("add_abc", [1, 2, 3]))  // -> 6
    }
}
```

## Integrate qt logging in python console

When you use `console.log` in QML side, it will be printed in Python console:

![](examples/console_print/screenshot.png)

## Signal and Slot

The `signal` and `slot` wrap on Qt's `Signal` and `Slot` decorators, but
extended their functionalities:

1.  You can get the correct type hint in IDE:

    ![](examples/signal_slot/screenshot_1.png)

2.  The `slot` accepts more types as alias to "QObject" and "QVariant" -- it is
    more convenient and more readable:

    ```python
    from lk_qtquick_scaffold import QObject, slot
    
    class MyObject(QObject):

        @slot(int, dict, result=list)  # <- here
        def foo(self, index, data):
            return [index, len(data)]
    
    '''
    it is more readable than:
        @Slot(int, QJSValue, result='QVariant')
        def foo(self, index, data):
            return [index, len(data)]
    '''
    ```

    Here is a full alias list (which is documented in 
    `lk_qtquick_scaffold/qt_core/signal_slot.py`):

    **slot(\*args)**

    | Alias         | Real value    | Note              |
    | ------------- | ------------- |------------------ |
    | `bool`        | `bool`        | basic type        |
    | `float`       | `float`       | basic type        |
    | `int`         | `int`         | basic type        |
    | `str`         | `str`         | basic type        |
    | `QObject`     | `QObject`     | object            |
    | `object`      | `QObject`     | object            |
    | `'item'`      | `QObject`     | object (string)   |
    | `'object'`    | `QObject`     | object (string)   |
    | `'qobject'`   | `QObject`     | object (string)   |
    | `dict`        | `QJSValue`    | qjsvalue          |
    | `list`        | `QJSValue`    | qjsvalue          |
    | `set`         | `QJSValue`    | qjsvalue          |
    | `tuple`       | `QJSValue`    | qjsvalue          |
    | `...`         | `QJSValue`    | qjsvalue          |
    | `'any'`       | `QJSValue`    | qjsvalue (string) |

    **slot(result=...)**

    | Alias     | Real value    | Note          |
    | --------- | ------------- |-------------- |
    | `None`    | `None`        | basic type    |
    | `bool`    | `bool`        | basic type    |
    | `float`   | `float`       | basic type    |
    | `int`     | `int`         | basic type    |
    | `str`     | `str`         | basic type    |
    | `dict`    | `'QVariant'`  | qvariant      |
    | `list`    | `'QVariant'`  | qvariant      |
    | `set`     | `'QVariant'`  | qvariant      |
    | `tuple`   | `'QVariant'`  | qvariant      |
    | `...`     | `'QVariant'`  | qvariant      |

3.  `slot` decorator is non-intrusive -- it means the method been decorated can 
    be called in Python side as usual.

    ```python
    from lk_qtquick_scaffold import QObject, slot

    class MyObject(QObject):
        @slot(int, str, result=list)
        def foo(self, index, name):
            return [index, name]

    my_obj = MyObject()
    # you can call it like a regular method! (just 'ignore' its docorator.)
    my_obj.foo(1, 'hello')  # -> [1, 'hello']
    ```

## Built-in widgets library

`lk-qtquick-scaffold` provides a set of built-in widgets under its `~/widgets` 
directory. 

Basically, you can use it in QML by importing "LKWidgets" (or "LKWidgets 1.0" 
for Qt 5.x):

```qml
import LKWidgets

LKWindow {
    color: '#DBDBF7'  // moon white

    LKRectangle {
        anchors.fill: parent
        anchors.margins: 32
        color: '#ECDEC8'  // parchment yellow

        LKColumn {
            anchors.centerIn: parent
            alignment: 'hcenter'  // horizontally center children

            LKGhostButton {
                text: 'SUNDAY'
            }

            LKButton {
                text: 'MONDAY'
            }

            LKGhostButton {
                text: 'TUESDAY'
            }

            LKButton {
                text: 'WEDNESDAY'
            }

            LKGhostButton {
                text: 'THURSDAY'
            }

            LKButton {
                text: 'FRIDAY'
            }

            LKGhostButton {
                text: 'SATURDAY'
            }
        }
    }
}
```

![](examples/lk_widgets/screenshot_1.gif)

The dark theme:

![](examples/lk_widgets/screenshot_2.png)

More screenshots: see `examples/lk_widgets/screenshot_*`.

All widget names are started with 'LK', the full list is in
`lk_qtquick_scaffold/widgets/LKWidgets/qmldir` file.

Note: the widgets documentation is not ready. Currently you may have a look at 
the `examples/lk_widgets` screenshots, or view its source code for more details.

## High-level model, human-readable API

*TODO*

## Layout engine

Layout engine is powered by `lk_qtquick_scaffold.qmlside.layout_helper`, which 
is registered as `pylayout` in QML side.

```qml
// some_view.qml
import QtQuick

Column {
    height: 100
    
    Item { id: item1; height: 20  }
    Item { id: item2; height: 0.4 }
    Item { id: item3; height: 0   }
    Item { id: item4; height: 0   }

    Component.onCompleted: {
        // horizontally center children
        pylayout.auto_align(this, 'hcenter')

        // auto size children:
        //  width > 1: as pixels
        //  width > 0 and < 1: as percent of left spared space
        //  width = 0: as stretch to fill the left spared space
        pylayout.auto_size_children(this, 'vertical')
        //  the result is:
        //      item1: 20px
        //      item2: (100 - 20) * 0.4 = 32px
        //      item3: (100 - 20 - 32) * 0.5 = 24px
        //      item4: (100 - 20 - 32) * 0.5 = 24px
        //          (item 3 and 4 share the left space equally.)
    }
}
```

## Executing Python snippet in QML, and vice versa

test.py

```python
from lk_qtquick_scaffold import eval_js

def foo(item1: QObject, item2: QObject):
    eval_js('''
        $a.widthChanged.connect(() => {
            $b.width = $a.width * 2
        })
    ''', {'a': item1, 'b': item2})
```

view.qml

```qml
import QtQuick

ListView {
    model: pyside.eval(`
        import os
        files = os.listdir(input('target folder: '))
        return files
    `)
}
```

## Style manager

`lk-qtquick-scaffold` exposes a list of built-in style controlers to QML side
as follows:

| Style         | Description                                               |
| ------------- | --------------------------------------------------------- |
| `pycolor`     | All color specifications defined in a canonical name form |
| `pyfont`      | Font related specifications                               |
| `pysize`      | Width, height, radius, padding, margin, spacing, etc.     |
| `pymotion` | Animation related specifications (duration, easing type, etc.) |

Usage examples (seen in all LKWidgets):

![](examples/lk_widgets/screenshot_3.png)

![](examples/lk_widgets/screenshot_4.png)

You can overwrite the style by giving a YAML file to load, for example a 
"dark-theme.yaml":

```yaml
# this is dark theme color scheme

# == general ==

blue_1: '#e4e5f8'
blue_3: '#5294eb'
blue_5: '#3844e6'
blue_7: '#0f143b'
dark_1: '#424141'
dark_2: '#242529'
dark_3: '#15141a'
dark_5: '#050408'
grey_3: '#e8eaed'
grey_5: '#a9acb0'

# == widgets spec ==

border_active: '#797171'
border_default: '#575757'
border_glow: '$border_active'
button_bg_active: '$blue_5'
button_bg_default: '$panel_bg'
button_bg_hovered: '$dark_1'
button_bg_pressed: '$dark_3'
button_bg_selected: '$button_bg_pressed'
input_bg_active: '$dark_2'
input_bg_default: '$panel_bg'
input_border_active: '$border_active'
input_border_default: '$border_default'
input_indicator_active: '$blue_5'
panel_bg: '$dark_3'
prog_bg: '$blue_1'
prog_fg: '$blue_5'
sidebar_bg: '$panel_bg'
text_default: '$grey_3'
text_disabled: '$grey_5'
text_hint: '$grey_5'
win_bg_default: '$dark_5'
```

The dollar symbol (`$`) is a simple pointer to the other key. 

You don't need to write all colors in the file, `lk-qtquick-scaffold` has a 
great deduction algorithm to automatically call back "defaults" when required
colors are missing from your sheet.

Finally load it by calling `pycolor.update_from_file()`:

```python
from lk_qtquick_scaffold import pycolor
pycolor.update_from_file('dark-theme.yaml')
```

Warning: currently color name style is under refactoring, it is very unstable 
to learn from its style.

# Gallery

![](gallery/widgets_demo/viscous-indicator-anim.gif)

![](gallery/widgets_demo/swipe-view.gif)

![](gallery/widgets_demo/breathing-circle-avatar.gif)

[![](gallery/widgets_demo/password-eye-open.gif)](https://uimovement.com/media/resource_image/image_5213.gif.mp4)

https://user-images.githubusercontent.com/27986259/180829198-7110831e-c060-436a-a9be-c41452f49932.mp4

https://user-images.githubusercontent.com/27986259/180829267-cd497bcc-de38-4d00-bb19-c4a84b251031.mp4

*TODO:AddMoreWidgetsDemo*

