py-trees-js


Namepy-trees-js JSON
Version 0.6.6 PyPI version JSON
download
home_pageNone
Summaryjavascript libraries for visualising behaviour trees
upload_time2025-01-14 02:34:28
maintainerDaniel Stonier
docs_urlNone
authorDaniel Stonier
requires_python<4.0,>=3.8
licenseBSD
keywords py_trees py-trees behaviour-trees
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # PyTrees Js

[[About](#about)] [[Features](#features)] [[Preview](#preview)] [[Exmaple - Simple Web App](#example---simple-web-app)] [[Example - PyQt App](#example---pyqt-app)] [[The JSON Specification](#the-json-specification)]

## About

Libraries for visualisation of runtime or replayed behaviour trees.

* [./js](./js) - a self-contained javascript library to build apps around
* [py_trees_js](./py_trees_js) - a python package that makes the js available as a pyqt resource
* [py_trees_js.viewer](./py_trees_js/viewer) - a demo pyqtwebengine app

See [py_trees_ros_viewer](https://github.com/splintered-reality/py_trees_ros_viewer) for a fully fledged pyqt integration that uses [py_trees_js](./py_trees_js).

## Features

* Visualise the runtime state of a behaviour tree
* Collapsible subtrees
* Zoom and scale contents to fit
* Timeline rewind & resume
* Blackboard key-value storage view
* Activity log view

Although designed for py_trees, the js libs (in particular, the interfaces) are not dependent on py_trees and could be used for other behaviour tree applications.

## Preview

With VSCode DevContainers and on a PC with an NVIDIA GPU:

* Install [VSCode](https://code.visualstudio.com/docs/setup/linux#_debian-and-ubuntu-based-distributions) 
* Install the [NVIDIA Container Toolkit](https://docs.nvidia.com/datacenter/cloud-native/container-toolkit/latest/install-guide.html#docker)
* Clone and launch the demo:

```
$ git clone https://github.com/splintered-reality/py_trees_js
$ code .
$ cd py_trees_js

# Reopen the project in the devcontainer
$ (use CTRL-SHIFT-P if you miss VSCode's helper dialog)

# Setup and launch
$ poetry install
$ poetry run py-trees-demo-viewer
```
<p align="center">
  <img src="images/splash.png" width="80%"/>
  <img src="images/screenshot.png" width="100%"/>
</p>

If you do not have a PC that meets those requirements, some alternative options:

* Install [Poetry](https://python-poetry.org/) and PyQt on your system or in a venv. Clone and launch.
* If you're just interested in seeing the demo viewer, `pip install --user py_trees_js` and launch the viewer
* Create your own devcontainer with something like the [desktop-lite](https://github.com/devcontainers/features/tree/main/src/desktop-lite) feature. If this works, send me a PR!

## Example - Simple Web App

Building a complete application that can render a behaviour tree stream is an effort that can be decomposed into two tasks:

1. Creating the web app for rendering trees and visualising a timeline
2. Wrapping the web app in a framework and connecting it to an external stream

The first stage is purely an exercise with html, css and javascript. The latter will depend on your use case - it could be a qt-js hybrid application (as exemplified here) for developers, an electron application for cross-platform and mobile deployment or a cloud based service.

This section will walk through how to build a web application with the provided js libraries. An example of wrapping the web app within a Qt-Js application will follow.

To get started, let's begin with a basic html page with two divs, one for the tree canvas and one for the timeline:

```xhtml
<!doctype html>
<html>
<head>
  <meta charset="utf-8">
  <title>PyTrees Viewer</title>
  <style>
    html {
      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */
    }
    body {
      margin: 0;
      overflow:hidden;  /* no scrollbars */
      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */
    }
  </style>
</head>
<body>
  <div id="canvas"></div>
  <div id="timeline"></div>
</body>
</html>
```

Next, bring in the javascript libraries. For exemplar purposes, it is assumed here that the libraries
have been made available alongside the html page - how is an integration detail depending on the mode
of deployment (see next section for an example).

Note that the `py_trees-<version>.js` library has only one dependency, [jointjs](https://resources.jointjs.com/docs/jointjs/v3.0/joint.html),
but that in turn has a few dependencies of it's own. The bundled libraries in the `js/jointjs` folder
of this repository correspond to the requirements for a specific version of jointjs and
have been tested to work with the accompany `py_trees-<version>.js` library.

You can verify that the libraries have been properly imported by calling `py_trees.hello()` which
will print version information of the loaded javascript libraries (if found) to the javascript console.

```xhtml
<!doctype html>
<html>
<head>
  <meta charset="utf-8">
  <title>PyTrees Viewer</title>
  <link rel="stylesheet" href="js/py_trees-0.6.0.css"/>
  <link rel="stylesheet" type="text/css" href="js/jointjs/joint-3.0.4.min.css"/>
  <script src="js/jointjs/dagre-0.8.4.min.js"></script>
  <script src="js/jointjs/graphlib-2.1.7.min.js"></script>
  <script src="js/jointjs/jquery-3.4.1.min.js"></script>
  <script src="js/jointjs/lodash-4.17.11.min.js"></script>
  <script src="js/jointjs/backbone-1.4.0.js"></script>
  <script src="js/jointjs/joint-3.0.4.min.js"></script>
  <script src="js/py_trees-0.6.0.js"></script>
  <!-- Web app integration css here -->
  <style>
    html {
      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */
    }
    body {
      margin: 0;
      overflow:hidden;  /* no scrollbars */
      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */
    }
  </style>
</head>
<body>
  <script type="text/javascript">
    py_trees.hello()
  </script>
  <div id="window">
    <div id="canvas"></div>
    <div id="timeline"></div>
  </div>
</body>
</html>
```

Output from `py_trees.hello()`:

```
********************************************************************************
                               Py Trees JS

 A javascript library for visualisation of executing behaviour trees.

 Version & Dependency Info:
  - py_trees:  0.6.0
    - jointjs :  3.1.0
       - backbone:  1.4.0
       - dagre   :  0.8.4
       - jquery  :  3.4.1
       - lodash  :  4.17.11
********************************************************************************
```

In the next iteration, the canvas is initialised and a callback for
accepting incoming trees from an external source is prepared. To test it,
pass it the demo tree provided by the library.

```xhtml
<!doctype html>
<html>
<head>
  <meta charset="utf-8">
  <title>PyTrees Viewer</title>
  <link rel="stylesheet" href="js/py_trees-0.6.0.css"/>
  <link rel="stylesheet" type="text/css" href="js/jointjs/joint-3.0.4.min.css"/>
  <script src="js/jointjs/dagre-0.8.4.min.js"></script>
  <script src="js/jointjs/graphlib-2.1.7.min.js"></script>
  <script src="js/jointjs/jquery-3.4.1.min.js"></script>
  <script src="js/jointjs/lodash-4.17.11.min.js"></script>
  <script src="js/jointjs/backbone-1.4.0.js"></script>
  <script src="js/jointjs/joint-3.0.4.min.js"></script>
  <script src="js/py_trees-0.6.0.js"></script>
  <!-- Web app integration css here -->
  <style>
    html {
      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */
    }
    body {
      margin: 0;
      overflow:hidden;  /* no scrollbars */
      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */
    }
  </style>
</head>
<body>
  <script type="text/javascript">
    py_trees.hello()
  </script>
  <div id="window">
    <div id="canvas"></div>
    <div id="timeline"></div>
  </div>
  <script type="text/javascript">
    // rendering canvas
    canvas_graph = py_trees.canvas.create_graph()
    canvas_paper = py_trees.canvas.create_paper({graph: canvas_graph})

    render_tree = function({tree}) {
      py_trees.canvas.update_graph({graph: canvas_graph, tree: tree})
      py_trees.canvas.layout_graph({graph: canvas_graph})
      if ( canvas_graph.get('scale_content_to_fit') ) {
        py_trees.canvas.scale_content_to_fit(canvas_paper)
      }
      return "rendered"
    }
    render_tree({tree: py_trees.experimental.create_demo_tree_definition()})
  </script>
</body>
</html>
```

At this point, your web app should be visualising a single tree and
zoom/collapse/scale to fit interactions functional. I'm happy, you should be too!

Adding a timeline to the application is optional, but the code does not
change significantly and is a very useful feature to have. The built-in demo
app's [index.html](py_trees_js/viewer/html/index.html) does exactly this. The code is reproduced below for convenience.

```xhtml
<!doctype html>
<html>
<head>
  <meta charset="utf-8">
  <title>PyTrees Viewer</title>
  <link rel="stylesheet" href="js/py_trees-0.6.0.css"/>
  <link rel="stylesheet" type="text/css" href="js/jointjs/joint-3.0.4.min.css"/>
  <script src="js/jointjs/dagre-0.8.4.min.js"></script>
  <script src="js/jointjs/graphlib-2.1.7.min.js"></script>
  <script src="js/jointjs/jquery-3.4.1.min.js"></script>
  <script src="js/jointjs/lodash-4.17.11.min.js"></script>
  <script src="js/jointjs/backbone-1.4.0.js"></script>
  <script src="js/jointjs/joint-3.0.4.min.js"></script>
  <script src="js/py_trees-0.6.0.js"></script>
  <!-- Web app integration css here -->
  <style>
    html {
      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */
    }
    body {
      margin: 0;
      overflow:hidden;  /* no scrollbars */
      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */
    }
  </style>
</head>
<body>
  <script type="text/javascript">
    py_trees.hello()
  </script>
  <div id="window">
    <div id="canvas"></div>
    <div id="timeline"></div>
  </div>
  <script type="text/javascript">
    // rendering canvas
    canvas_graph = py_trees.canvas.create_graph()
    canvas_paper = py_trees.canvas.create_paper({graph: canvas_graph})

    // event timeline
    timeline_graph = py_trees.timeline.create_graph({event_cache_limit: 100});
    timeline_paper = py_trees.timeline.create_paper({
        timeline_graph: timeline_graph,
        canvas_graph: canvas_graph,
        canvas_paper: canvas_paper,
    })

    // react to window resizing events
    $(window).resize(function() {
      py_trees.canvas.on_window_resize(canvas_paper)
      py_trees.timeline.on_window_resize(timeline_paper)
    })

    render_tree = function({tree}) {
      py_trees.timeline.add_tree_to_cache({
          timeline_graph: timeline_graph,
          canvas_graph: canvas_graph,
          canvas_paper: canvas_paper,
          tree: tree
      })
      return "rendered"
    }
  </script>
</body>
</html>
```

## Example - PyQt App

The `py-trees-demo-viewer` app is a qt-js hybrid application using `qtwebengine`.
Every time a qt button is pressed, an internally generated tree snapshot is sent to `render_tree()` in the embedded web application. From here, it is not too hard to
imagine connecting the qt application to an actual external source. The qt layer
then acts as a shim or relay transferring messages to the internal web app.

How does it work? 

* The js libs are made available as a `.qrc` resource [1]
* A simple web app is made available as another `.qrc` resource
* Both resources are consumed by the QWebEngine View to serve the app

[1] This can be made available separately and as a dependency to the actual
pyqt application. For instance, the [py_trees_js](./py_trees_js) package is a dependency of [py_trees_ros_viewer](https://github.com/splintered-reality/py_trees_ros_viewer).

In more detail...

### The JS Libraries

1. Bundle the javascript resources into a `.qrc` file
2. Generate the resources as a c++ library / python module
3. Deploy the c++ library/python module in your development environment

In this case, the py_trees and jointjs javascript libraries have been listed
in [py_trees_js/resources.qrc](py_trees_js/resources.qrc), generated using
[py_trees_js/gen.bash](py_trees_js/gen.bash), resulting in the importable module
[py_trees_js/resources.py](py_trees_js/resources.py). From this point, any pythonic
Qt application wishing to visualise behaviour trees need only import this module from the `py_trees_js` package.

### The Web App

1. Bundle the `.html`/`.css` pages into a `.qrc` file
2. Import into directly into designer when building your Qt application

In this case, our web app ([py_trees_js/viewer/html/index.html](py_trees_js/viewer/html/index.html)) has been rolled into [py_trees_js/viewer/web_app.qrc](py_trees_js/viewer/web_app.qrc) which is directly loaded into [py_trees_js/viewer/web_view.ui](py_trees_js/viewer/web_view.ui) where the URL property of the QWebEngineView widget has been configured with the resources `index.html`.

You could alternatively, generate a module from the `.qrc` and import that into the
relevant python code as was done for the javascript resources.

### The Qt Application

The Qt application can be designed in whatever way you're most comfortable with - via
Designer, pure C++ or python. In this case, Qt's Designer is used to produce the `.ui`
files which are generated into python modules and finally customised and brought together
as a PyQt5 application. Refer to [py_trees_js/viewer](py_trees_js/viewer) for more details
or as a reference example from which to start your own Qt-Js hybrid application.

Key elements:

1. Build your Qt application around a QWebEngineView widget
2. Link/import the javascript module in the web engine view class
3. Load the html page into the QWebEngineView view

Do not use the QWebView widget - this is deprecating in favour of the QWebEngineView widget. The most notable difference is that QWebView uses Qt's old webkit, while QWebEngineView makes use of Chromium's webkit.

Note that the second step automagically makes available the javascript resources to the application
when it's loaded. It's not terribly fussy about where it gets loaded, see [py_trees_js/viewer/web_view.py](py_trees_js/viewer/web_view.py) for an example:

```
# This is the module generated by running pyrcc5 on the js libraries .qrc
# It could have been equivalently deployed in a completely different python package
import py_trees_js.resources
```

Loading the web page can be accomplished in designer. Simply point it at your qresource file
and set the dynamic URL property on the QWebEngineView widget. Alternatively you can import
the resource module and load it via QWebEngineView's `load` api.

#### Qt-Js Interactions

Qt and JS can interact directly over snippets of javascript code (via `runJavaScript()`
or over QWebChannel (a mechanism similar to sigslots) where more complexity is needed.
The example application here calls on the `render_tree()` method we created earlier in
the web application to send trees to the app. Example code from [py_trees_js/viewer/viewer.py](py_trees_js/viewer/viewer.py) which handles button clicks to cycle through a list of
demonstration trees:

```
def send_tree_response(reply):
    console.logdebug("reply: '{}' [viewer]".format(reply))


@qt_core.pyqtSlot()
def send_tree(web_view_page, demo_trees, unused_checked):
    demo_trees[send_tree.index]['timestamp'] = time.time()
    console.logdebug("send: tree '{}' [{}][viewer]".format(
        send_tree.index, demo_trees[send_tree.index]['timestamp'])
    )
    javascript_command = "render_tree({{tree: {}}})".format(demo_trees[send_tree.index])
    web_view_page.runJavaScript(javascript_command, send_tree_response)
    send_tree.index = 0 if send_tree.index == 2 else send_tree.index + 1

send_tree.index = 0
```

## The JSON Specification

TODO: A JSon schema

Roughly, the specification expects json objects of the form:

* timestamp: int
* behaviours: dict[str, dict]
* (optional) visited_path: list[str]
* (optional) blackboard: {
*    behaviours: dict[str, dict[str, str]],
*    data: dict[str, str]
* }
* (optional) activity: list[str]

where each behaviour in the dict has specification:

* id: str
* status: Union[`INVALID`,`FAILURE`, `RUNNING`, `SUCCESS`]
* name: str
* colour: <html style hex code>
* (optional) children: List[str]
* (optional) data: <generic key-value dictionary>

Identification strings (id's) must be unique and are used as both keys for the
behaviours dictionary, children and visited_path variables.

An example (extracted from `py_trees.experimental.create_demo_tree_definition()`):

```
{
    timestamp: 1563938995,
    visited_path: ['1', '2', '3', '4', '5', '7', '8'],
    behaviours: {
        '1': {
            id: '1',
            status: 'RUNNING',
            name: 'Selector',
            colour: '#00FFFF',
            children: ['2', '3', '4', '6'],
            data: {
                Type: 'py_trees.composites.Selector',
                Feedback: "Decision maker",
            },
        },
        '2': {
            id: '2',
            status: 'FAILURE',
            name: 'Worker',
            colour: '#FFA500',
            children: ['7', '8', '9'],
            data: {
                Type: 'py_trees.composites.Sequence',
                Feedback: "Worker"
            },
        },
    }
    'blackboard': {
        'behaviours': {  # key metadata per behaviour
            '2': {
                '/parameters/initial_value': 'r',
                '/state/worker': 'w'
            },
        },
        'data': {
            '/parameters/initial_value': 'foo',
            '/state/worker': 'bar',
        },
    'activity': [
        "<text style='color: blue;'>Worker initialised with 'foo'</text>'",
        "<text style='color: red;'>Worker wrote 'bar'</text>'",
    ]
}


            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "py-trees-js",
    "maintainer": "Daniel Stonier",
    "docs_url": null,
    "requires_python": "<4.0,>=3.8",
    "maintainer_email": "d.stonier@gmail.com",
    "keywords": "py_trees, py-trees, behaviour-trees",
    "author": "Daniel Stonier",
    "author_email": null,
    "download_url": "https://files.pythonhosted.org/packages/3a/cd/e188df9128cac327ee8d736b5d5794849f930b105988854e6f469bb6acc7/py_trees_js-0.6.6.tar.gz",
    "platform": null,
    "description": "# PyTrees Js\n\n[[About](#about)] [[Features](#features)] [[Preview](#preview)] [[Exmaple - Simple Web App](#example---simple-web-app)] [[Example - PyQt App](#example---pyqt-app)] [[The JSON Specification](#the-json-specification)]\n\n## About\n\nLibraries for visualisation of runtime or replayed behaviour trees.\n\n* [./js](./js) - a self-contained javascript library to build apps around\n* [py_trees_js](./py_trees_js) - a python package that makes the js available as a pyqt resource\n* [py_trees_js.viewer](./py_trees_js/viewer) - a demo pyqtwebengine app\n\nSee [py_trees_ros_viewer](https://github.com/splintered-reality/py_trees_ros_viewer) for a fully fledged pyqt integration that uses [py_trees_js](./py_trees_js).\n\n## Features\n\n* Visualise the runtime state of a behaviour tree\n* Collapsible subtrees\n* Zoom and scale contents to fit\n* Timeline rewind & resume\n* Blackboard key-value storage view\n* Activity log view\n\nAlthough designed for py_trees, the js libs (in particular, the interfaces) are not dependent on py_trees and could be used for other behaviour tree applications.\n\n## Preview\n\nWith VSCode DevContainers and on a PC with an NVIDIA GPU:\n\n* Install [VSCode](https://code.visualstudio.com/docs/setup/linux#_debian-and-ubuntu-based-distributions) \n* Install the [NVIDIA Container Toolkit](https://docs.nvidia.com/datacenter/cloud-native/container-toolkit/latest/install-guide.html#docker)\n* Clone and launch the demo:\n\n```\n$ git clone https://github.com/splintered-reality/py_trees_js\n$ code .\n$ cd py_trees_js\n\n# Reopen the project in the devcontainer\n$ (use CTRL-SHIFT-P if you miss VSCode's helper dialog)\n\n# Setup and launch\n$ poetry install\n$ poetry run py-trees-demo-viewer\n```\n<p align=\"center\">\n  <img src=\"images/splash.png\" width=\"80%\"/>\n  <img src=\"images/screenshot.png\" width=\"100%\"/>\n</p>\n\nIf you do not have a PC that meets those requirements, some alternative options:\n\n* Install [Poetry](https://python-poetry.org/) and PyQt on your system or in a venv. Clone and launch.\n* If you're just interested in seeing the demo viewer, `pip install --user py_trees_js` and launch the viewer\n* Create your own devcontainer with something like the [desktop-lite](https://github.com/devcontainers/features/tree/main/src/desktop-lite) feature. If this works, send me a PR!\n\n## Example - Simple Web App\n\nBuilding a complete application that can render a behaviour tree stream is an effort that can be decomposed into two tasks:\n\n1. Creating the web app for rendering trees and visualising a timeline\n2. Wrapping the web app in a framework and connecting it to an external stream\n\nThe first stage is purely an exercise with html, css and javascript. The latter will depend on your use case - it could be a qt-js hybrid application (as exemplified here) for developers, an electron application for cross-platform and mobile deployment or a cloud based service.\n\nThis section will walk through how to build a web application with the provided js libraries. An example of wrapping the web app within a Qt-Js application will follow.\n\nTo get started, let's begin with a basic html page with two divs, one for the tree canvas and one for the timeline:\n\n```xhtml\n<!doctype html>\n<html>\n<head>\n  <meta charset=\"utf-8\">\n  <title>PyTrees Viewer</title>\n  <style>\n    html {\n      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */\n    }\n    body {\n      margin: 0;\n      overflow:hidden;  /* no scrollbars */\n      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */\n    }\n  </style>\n</head>\n<body>\n  <div id=\"canvas\"></div>\n  <div id=\"timeline\"></div>\n</body>\n</html>\n```\n\nNext, bring in the javascript libraries. For exemplar purposes, it is assumed here that the libraries\nhave been made available alongside the html page - how is an integration detail depending on the mode\nof deployment (see next section for an example).\n\nNote that the `py_trees-<version>.js` library has only one dependency, [jointjs](https://resources.jointjs.com/docs/jointjs/v3.0/joint.html),\nbut that in turn has a few dependencies of it's own. The bundled libraries in the `js/jointjs` folder\nof this repository correspond to the requirements for a specific version of jointjs and\nhave been tested to work with the accompany `py_trees-<version>.js` library.\n\nYou can verify that the libraries have been properly imported by calling `py_trees.hello()` which\nwill print version information of the loaded javascript libraries (if found) to the javascript console.\n\n```xhtml\n<!doctype html>\n<html>\n<head>\n  <meta charset=\"utf-8\">\n  <title>PyTrees Viewer</title>\n  <link rel=\"stylesheet\" href=\"js/py_trees-0.6.0.css\"/>\n  <link rel=\"stylesheet\" type=\"text/css\" href=\"js/jointjs/joint-3.0.4.min.css\"/>\n  <script src=\"js/jointjs/dagre-0.8.4.min.js\"></script>\n  <script src=\"js/jointjs/graphlib-2.1.7.min.js\"></script>\n  <script src=\"js/jointjs/jquery-3.4.1.min.js\"></script>\n  <script src=\"js/jointjs/lodash-4.17.11.min.js\"></script>\n  <script src=\"js/jointjs/backbone-1.4.0.js\"></script>\n  <script src=\"js/jointjs/joint-3.0.4.min.js\"></script>\n  <script src=\"js/py_trees-0.6.0.js\"></script>\n  <!-- Web app integration css here -->\n  <style>\n    html {\n      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */\n    }\n    body {\n      margin: 0;\n      overflow:hidden;  /* no scrollbars */\n      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */\n    }\n  </style>\n</head>\n<body>\n  <script type=\"text/javascript\">\n    py_trees.hello()\n  </script>\n  <div id=\"window\">\n    <div id=\"canvas\"></div>\n    <div id=\"timeline\"></div>\n  </div>\n</body>\n</html>\n```\n\nOutput from `py_trees.hello()`:\n\n```\n********************************************************************************\n                               Py Trees JS\n\n A javascript library for visualisation of executing behaviour trees.\n\n Version & Dependency Info:\n  - py_trees:  0.6.0\n    - jointjs :  3.1.0\n       - backbone:  1.4.0\n       - dagre   :  0.8.4\n       - jquery  :  3.4.1\n       - lodash  :  4.17.11\n********************************************************************************\n```\n\nIn the next iteration, the canvas is initialised and a callback for\naccepting incoming trees from an external source is prepared. To test it,\npass it the demo tree provided by the library.\n\n```xhtml\n<!doctype html>\n<html>\n<head>\n  <meta charset=\"utf-8\">\n  <title>PyTrees Viewer</title>\n  <link rel=\"stylesheet\" href=\"js/py_trees-0.6.0.css\"/>\n  <link rel=\"stylesheet\" type=\"text/css\" href=\"js/jointjs/joint-3.0.4.min.css\"/>\n  <script src=\"js/jointjs/dagre-0.8.4.min.js\"></script>\n  <script src=\"js/jointjs/graphlib-2.1.7.min.js\"></script>\n  <script src=\"js/jointjs/jquery-3.4.1.min.js\"></script>\n  <script src=\"js/jointjs/lodash-4.17.11.min.js\"></script>\n  <script src=\"js/jointjs/backbone-1.4.0.js\"></script>\n  <script src=\"js/jointjs/joint-3.0.4.min.js\"></script>\n  <script src=\"js/py_trees-0.6.0.js\"></script>\n  <!-- Web app integration css here -->\n  <style>\n    html {\n      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */\n    }\n    body {\n      margin: 0;\n      overflow:hidden;  /* no scrollbars */\n      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */\n    }\n  </style>\n</head>\n<body>\n  <script type=\"text/javascript\">\n    py_trees.hello()\n  </script>\n  <div id=\"window\">\n    <div id=\"canvas\"></div>\n    <div id=\"timeline\"></div>\n  </div>\n  <script type=\"text/javascript\">\n    // rendering canvas\n    canvas_graph = py_trees.canvas.create_graph()\n    canvas_paper = py_trees.canvas.create_paper({graph: canvas_graph})\n\n    render_tree = function({tree}) {\n      py_trees.canvas.update_graph({graph: canvas_graph, tree: tree})\n      py_trees.canvas.layout_graph({graph: canvas_graph})\n      if ( canvas_graph.get('scale_content_to_fit') ) {\n        py_trees.canvas.scale_content_to_fit(canvas_paper)\n      }\n      return \"rendered\"\n    }\n    render_tree({tree: py_trees.experimental.create_demo_tree_definition()})\n  </script>\n</body>\n</html>\n```\n\nAt this point, your web app should be visualising a single tree and\nzoom/collapse/scale to fit interactions functional. I'm happy, you should be too!\n\nAdding a timeline to the application is optional, but the code does not\nchange significantly and is a very useful feature to have. The built-in demo\napp's [index.html](py_trees_js/viewer/html/index.html) does exactly this. The code is reproduced below for convenience.\n\n```xhtml\n<!doctype html>\n<html>\n<head>\n  <meta charset=\"utf-8\">\n  <title>PyTrees Viewer</title>\n  <link rel=\"stylesheet\" href=\"js/py_trees-0.6.0.css\"/>\n  <link rel=\"stylesheet\" type=\"text/css\" href=\"js/jointjs/joint-3.0.4.min.css\"/>\n  <script src=\"js/jointjs/dagre-0.8.4.min.js\"></script>\n  <script src=\"js/jointjs/graphlib-2.1.7.min.js\"></script>\n  <script src=\"js/jointjs/jquery-3.4.1.min.js\"></script>\n  <script src=\"js/jointjs/lodash-4.17.11.min.js\"></script>\n  <script src=\"js/jointjs/backbone-1.4.0.js\"></script>\n  <script src=\"js/jointjs/joint-3.0.4.min.js\"></script>\n  <script src=\"js/py_trees-0.6.0.js\"></script>\n  <!-- Web app integration css here -->\n  <style>\n    html {\n      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */\n    }\n    body {\n      margin: 0;\n      overflow:hidden;  /* no scrollbars */\n      height: 100%  /* canvas is intended to fill the screen, cascading heights achieves this */\n    }\n  </style>\n</head>\n<body>\n  <script type=\"text/javascript\">\n    py_trees.hello()\n  </script>\n  <div id=\"window\">\n    <div id=\"canvas\"></div>\n    <div id=\"timeline\"></div>\n  </div>\n  <script type=\"text/javascript\">\n    // rendering canvas\n    canvas_graph = py_trees.canvas.create_graph()\n    canvas_paper = py_trees.canvas.create_paper({graph: canvas_graph})\n\n    // event timeline\n    timeline_graph = py_trees.timeline.create_graph({event_cache_limit: 100});\n    timeline_paper = py_trees.timeline.create_paper({\n        timeline_graph: timeline_graph,\n        canvas_graph: canvas_graph,\n        canvas_paper: canvas_paper,\n    })\n\n    // react to window resizing events\n    $(window).resize(function() {\n      py_trees.canvas.on_window_resize(canvas_paper)\n      py_trees.timeline.on_window_resize(timeline_paper)\n    })\n\n    render_tree = function({tree}) {\n      py_trees.timeline.add_tree_to_cache({\n          timeline_graph: timeline_graph,\n          canvas_graph: canvas_graph,\n          canvas_paper: canvas_paper,\n          tree: tree\n      })\n      return \"rendered\"\n    }\n  </script>\n</body>\n</html>\n```\n\n## Example - PyQt App\n\nThe `py-trees-demo-viewer` app is a qt-js hybrid application using `qtwebengine`.\nEvery time a qt button is pressed, an internally generated tree snapshot is sent to `render_tree()` in the embedded web application. From here, it is not too hard to\nimagine connecting the qt application to an actual external source. The qt layer\nthen acts as a shim or relay transferring messages to the internal web app.\n\nHow does it work? \n\n* The js libs are made available as a `.qrc` resource [1]\n* A simple web app is made available as another `.qrc` resource\n* Both resources are consumed by the QWebEngine View to serve the app\n\n[1] This can be made available separately and as a dependency to the actual\npyqt application. For instance, the [py_trees_js](./py_trees_js) package is a dependency of [py_trees_ros_viewer](https://github.com/splintered-reality/py_trees_ros_viewer).\n\nIn more detail...\n\n### The JS Libraries\n\n1. Bundle the javascript resources into a `.qrc` file\n2. Generate the resources as a c++ library / python module\n3. Deploy the c++ library/python module in your development environment\n\nIn this case, the py_trees and jointjs javascript libraries have been listed\nin [py_trees_js/resources.qrc](py_trees_js/resources.qrc), generated using\n[py_trees_js/gen.bash](py_trees_js/gen.bash), resulting in the importable module\n[py_trees_js/resources.py](py_trees_js/resources.py). From this point, any pythonic\nQt application wishing to visualise behaviour trees need only import this module from the `py_trees_js` package.\n\n### The Web App\n\n1. Bundle the `.html`/`.css` pages into a `.qrc` file\n2. Import into directly into designer when building your Qt application\n\nIn this case, our web app ([py_trees_js/viewer/html/index.html](py_trees_js/viewer/html/index.html)) has been rolled into [py_trees_js/viewer/web_app.qrc](py_trees_js/viewer/web_app.qrc) which is directly loaded into [py_trees_js/viewer/web_view.ui](py_trees_js/viewer/web_view.ui) where the URL property of the QWebEngineView widget has been configured with the resources `index.html`.\n\nYou could alternatively, generate a module from the `.qrc` and import that into the\nrelevant python code as was done for the javascript resources.\n\n### The Qt Application\n\nThe Qt application can be designed in whatever way you're most comfortable with - via\nDesigner, pure C++ or python. In this case, Qt's Designer is used to produce the `.ui`\nfiles which are generated into python modules and finally customised and brought together\nas a PyQt5 application. Refer to [py_trees_js/viewer](py_trees_js/viewer) for more details\nor as a reference example from which to start your own Qt-Js hybrid application.\n\nKey elements:\n\n1. Build your Qt application around a QWebEngineView widget\n2. Link/import the javascript module in the web engine view class\n3. Load the html page into the QWebEngineView view\n\nDo not use the QWebView widget - this is deprecating in favour of the QWebEngineView widget. The most notable difference is that QWebView uses Qt's old webkit, while QWebEngineView makes use of Chromium's webkit.\n\nNote that the second step automagically makes available the javascript resources to the application\nwhen it's loaded. It's not terribly fussy about where it gets loaded, see [py_trees_js/viewer/web_view.py](py_trees_js/viewer/web_view.py) for an example:\n\n```\n# This is the module generated by running pyrcc5 on the js libraries .qrc\n# It could have been equivalently deployed in a completely different python package\nimport py_trees_js.resources\n```\n\nLoading the web page can be accomplished in designer. Simply point it at your qresource file\nand set the dynamic URL property on the QWebEngineView widget. Alternatively you can import\nthe resource module and load it via QWebEngineView's `load` api.\n\n#### Qt-Js Interactions\n\nQt and JS can interact directly over snippets of javascript code (via `runJavaScript()`\nor over QWebChannel (a mechanism similar to sigslots) where more complexity is needed.\nThe example application here calls on the `render_tree()` method we created earlier in\nthe web application to send trees to the app. Example code from [py_trees_js/viewer/viewer.py](py_trees_js/viewer/viewer.py) which handles button clicks to cycle through a list of\ndemonstration trees:\n\n```\ndef send_tree_response(reply):\n    console.logdebug(\"reply: '{}' [viewer]\".format(reply))\n\n\n@qt_core.pyqtSlot()\ndef send_tree(web_view_page, demo_trees, unused_checked):\n    demo_trees[send_tree.index]['timestamp'] = time.time()\n    console.logdebug(\"send: tree '{}' [{}][viewer]\".format(\n        send_tree.index, demo_trees[send_tree.index]['timestamp'])\n    )\n    javascript_command = \"render_tree({{tree: {}}})\".format(demo_trees[send_tree.index])\n    web_view_page.runJavaScript(javascript_command, send_tree_response)\n    send_tree.index = 0 if send_tree.index == 2 else send_tree.index + 1\n\nsend_tree.index = 0\n```\n\n## The JSON Specification\n\nTODO: A JSon schema\n\nRoughly, the specification expects json objects of the form:\n\n* timestamp: int\n* behaviours: dict[str, dict]\n* (optional) visited_path: list[str]\n* (optional) blackboard: {\n*    behaviours: dict[str, dict[str, str]],\n*    data: dict[str, str]\n* }\n* (optional) activity: list[str]\n\nwhere each behaviour in the dict has specification:\n\n* id: str\n* status: Union[`INVALID`,`FAILURE`, `RUNNING`, `SUCCESS`]\n* name: str\n* colour: <html style hex code>\n* (optional) children: List[str]\n* (optional) data: <generic key-value dictionary>\n\nIdentification strings (id's) must be unique and are used as both keys for the\nbehaviours dictionary, children and visited_path variables.\n\nAn example (extracted from `py_trees.experimental.create_demo_tree_definition()`):\n\n```\n{\n    timestamp: 1563938995,\n    visited_path: ['1', '2', '3', '4', '5', '7', '8'],\n    behaviours: {\n        '1': {\n            id: '1',\n            status: 'RUNNING',\n            name: 'Selector',\n            colour: '#00FFFF',\n            children: ['2', '3', '4', '6'],\n            data: {\n                Type: 'py_trees.composites.Selector',\n                Feedback: \"Decision maker\",\n            },\n        },\n        '2': {\n            id: '2',\n            status: 'FAILURE',\n            name: 'Worker',\n            colour: '#FFA500',\n            children: ['7', '8', '9'],\n            data: {\n                Type: 'py_trees.composites.Sequence',\n                Feedback: \"Worker\"\n            },\n        },\n    }\n    'blackboard': {\n        'behaviours': {  # key metadata per behaviour\n            '2': {\n                '/parameters/initial_value': 'r',\n                '/state/worker': 'w'\n            },\n        },\n        'data': {\n            '/parameters/initial_value': 'foo',\n            '/state/worker': 'bar',\n        },\n    'activity': [\n        \"<text style='color: blue;'>Worker initialised with 'foo'</text>'\",\n        \"<text style='color: red;'>Worker wrote 'bar'</text>'\",\n    ]\n}\n\n",
    "bugtrack_url": null,
    "license": "BSD",
    "summary": "javascript libraries for visualising behaviour trees",
    "version": "0.6.6",
    "project_urls": {
        "Documentation": "https://github.com/splintered-reality/py_trees_js",
        "Homepage": "https://github.com/splintered-reality/py_trees_js",
        "Repository": "https://github.com/splintered-reality/py_trees_js"
    },
    "split_keywords": [
        "py_trees",
        " py-trees",
        " behaviour-trees"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "6ebd5a9c6c8cc4a656c8a5d44a9dcadcee42ee1f6934d43ad2ff89dc9ee06c65",
                "md5": "5e52ff8b917f2fea704fb06989ca417e",
                "sha256": "4c4024b8d445aa9799fbbf557dde1ee4a4b203b915498fc32904f2d2b75dbed8"
            },
            "downloads": -1,
            "filename": "py_trees_js-0.6.6-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "5e52ff8b917f2fea704fb06989ca417e",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": "<4.0,>=3.8",
            "size": 578242,
            "upload_time": "2025-01-14T02:34:25",
            "upload_time_iso_8601": "2025-01-14T02:34:25.427258Z",
            "url": "https://files.pythonhosted.org/packages/6e/bd/5a9c6c8cc4a656c8a5d44a9dcadcee42ee1f6934d43ad2ff89dc9ee06c65/py_trees_js-0.6.6-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "3acde188df9128cac327ee8d736b5d5794849f930b105988854e6f469bb6acc7",
                "md5": "b4e668e1c276ddf9f4a20f60b4bf976d",
                "sha256": "bcad260bcef72edfa2c8e0ccadf0b25b9915f089a146891bf28ecc2bc6d59239"
            },
            "downloads": -1,
            "filename": "py_trees_js-0.6.6.tar.gz",
            "has_sig": false,
            "md5_digest": "b4e668e1c276ddf9f4a20f60b4bf976d",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": "<4.0,>=3.8",
            "size": 549689,
            "upload_time": "2025-01-14T02:34:28",
            "upload_time_iso_8601": "2025-01-14T02:34:28.157090Z",
            "url": "https://files.pythonhosted.org/packages/3a/cd/e188df9128cac327ee8d736b5d5794849f930b105988854e6f469bb6acc7/py_trees_js-0.6.6.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2025-01-14 02:34:28",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "splintered-reality",
    "github_project": "py_trees_js",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "tox": true,
    "lcname": "py-trees-js"
}
        
Elapsed time: 0.42963s