Skip to main content

A package for n-tree 2D data structures similar to and including quadtree, with mapping to payload data.

Project description

Contributors Forks Stargazers Issues MIT License


9🌳

nontree

A python package for n-tree 2D data structures similar to and including quadtree, with mapping to payload data
Explore the docs »

Report Bug · Request Feature

Table of Contents
  1. About The Project
  2. Getting Started
  3. Usage
  4. Contributing
  5. Testing
  6. License
  7. Project Links

About The Project

NonTree pyplot

This package implements quadtree-like data structures in the classes NonTree, QuadTree and BiTree. More precisely speaking, point-region (PR) quadtrees and variants thereof are implemented. It also provides the class TreeMap to map points from the tree to arbitrary payload data, with a dictionary-like interface.

A point-region quadtree is a recursive data structure to contain points that lie on a two dimensional surface. It allows for efficient collision detection between the containend points and an area. Thus it can answer the question "Which points are within that rectangular area?".

In comparision to looping over all points and comparing the location of each point with the rectangular boundaries, a quadtree delivers the answer in O(log(n)) instead of O(n).

This implementation also provides methods for circular collision detection.

An exemplary application could be to test elements on a map of a 2D game, before rendering each frame, whether they are in the screen area and thus must be drawn, or if the time of drawing them can be saved due to them not beeing in a visible area anyway.

For more information about quadtrees in general, see the Wikipedia article. Or more specifically the paragraph about point region quadtrees.

A quadtree (4-tree) is split into 4 subtrees, if its bucket capacity is full and additional points are added. Thus, it's area gets split into 4 quadrants. The points then are stored in the subtrees, until they are at capacity and get split as well. That's the well known quadtree data structure. This package also implements two unusual variants thereof: The non-tree (9-tree) and the alternating bi-tree (2-tree).

The quadtree splits each tree into 9 segments in a 3 by 3 grid. Due to reduced nesting depth, the retrieval of colliding points can be faster on a densely populated surface.

The bitree splits each tree in only 2 segments, alternatingly horizontally or vertically. This tends to result in a higher nesting depth. But at each level, only two instead of four or nine subtrees have to be considered. Thus it can be faster on a very sparsely populated surface.

The class TreeMap internally contains a NonTree (the default) or a QuadTree or BiTree. It maps the points of the tree to payload data. This allows the retrieval of objects that lie in the searched area.

If no payload data is needed, and only the bare points are of interest, the tree classes can be used directly.

Additionally, a module visualize exists, to plot a graph of the layout of a tree with the help of matplotlib.

Getting Started

Install the package, import a module, and call a function.

Prerequisites

This is a package for python3 (>=3.10). Slightly older versions of python3 might work, but are entirely untested.

The package has no mandatory dependencies of other external packages, but it optionally can make use of the follwing, if present:

  • matplotlib to plot graphs of the tree layout, such as the one shown above. It is used by the module visualize.
  • numba to speed up (jit-compile) some of the calculations for circular collision detection. It gets detected/used automatically, without need for configuration or passing options.

Installation

You can get the raw files from the git repository:

git clone https://github.com/iehgit/nontree.git

It's easier to install the package with pip:

pip install nontree

Demo

To see that the package has been installed and is accessible, you could plot a demo graph, if you also have matplotlib available:

>>> from nontree import visualize
>>> visualize.plot_demo()
>>>

Or you could simply construct a TreeMap, set a datapoint, and get back its data:

>>> from nontree.TreeMap import TreeMap
>>> tree_map = TreeMap((0, 0, 100, 100))
>>> tree_map[(50, 50)] = 'Hello World!'
>>> tree_map[(50, 50)]
'Hello World!'
>>>

Usage

Data Types

Points are defined as tuples in the shape of (x, y).
Data points are defined as tuples in the shape of (point, data), i.e. ((x, y), data), with data beeing any arbitrary object.
Rectangular areas are defined as tuples in the shape of (x, y, width, height), or anything that can be indexed in the same way. This allows the use of pygame.Rect.
Circular areas are defined as tuples in the shape of (x, y, radius).

Constructing A TreeMap

A TreeMap can be constructed like this, with a rectangle as its first parameter to specify its surface:

from nontree.TreeMap import TreeMap

tree_map = TreeMap((0, 0, 100, 100))

Per default, it contains a NonTree. To specify the tree type, you can use the keyword argument mode, like this:

from nontree.TreeMap import TreeMap

tree_map_foo = TreeMap((0, 0, 100, 100), mode=9)  # NonTree
tree_map_bar = TreeMap((0, 0, 100, 100), mode=4)  # QuadTree
tree_map_baz = TreeMap((0, 0, 100, 100), mode=2)  # BiTree

Adding Data Points To A TreeMap

To add multiple data points:

from nontree.TreeMap import TreeMap

a_lot_of_datapoints = [((2, 3), 'dog'), ((17, 80), 'cat'), ((45, 13), 'fish'), ((99, 77), 'rat')]

tree_map = TreeMap((0, 0, 100, 100))
tree_map.add_datapoints(a_lot_of_datapoints)

To add a single data point, the dict-like interface is convenient:

from nontree.TreeMap import TreeMap

tree_map = TreeMap((0, 0, 100, 100))
tree_map[(55, 89)] = 'goat'

Collision Detection

To query the data within an rectangular area of the surface, you can use the method get_rect, which takes a rectangle as its parameter:

from nontree.TreeMap import TreeMap

a_lot_of_datapoints = [((2, 3), 'dog'), ((17, 80), 'cat'), ((45, 13), 'fish'), ((99, 77), 'rat')]

tree_map = TreeMap((0, 0, 100, 100))
tree_map.add_datapoints(a_lot_of_datapoints)
tree_map[(55, 89)] = 'goat'

result = tree_map.get_rect((4, 4, 80, 80))
print(result)

Output of the above:

['fish', 'cat']

For more details, please refer to the Documentation.

Contributing

Contributions are what make the open source community such an amazing place to learn, inspire, and create. Any contributions you make are greatly appreciated.

If you have a suggestion that would make this better, please fork the repo and create a pull request. Please make sure that all unit tests still pass after any changes.

  1. Fork the Project
  2. Create your Feature Branch (git checkout -b feature/AmazingFeature)
  3. Commit your Changes (git commit -m 'Add some AmazingFeature')
  4. Push to the Branch (git push origin feature/AmazingFeature)
  5. Open a Pull Request

You can also simply open an issue with the tag "enhancement".

Testing

To run the unit tests:

cd tests
PYTHONPATH=../src python3 utest.py 

The output should look roughly similar to this:

......................................
-------------------------------------------------------------------------------
Ran 38 tests in 0.308s

PASSED (successes=38)

License

Distributed under the MIT License. See LICENSE for more information.

Project Links

Github

PyPI

API documentation

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

nontree-1.0.5.tar.gz (203.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

nontree-1.0.5-py3-none-any.whl (13.8 kB view details)

Uploaded Python 3

File details

Details for the file nontree-1.0.5.tar.gz.

File metadata

  • Download URL: nontree-1.0.5.tar.gz
  • Upload date:
  • Size: 203.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.1 CPython/3.10.8

File hashes

Hashes for nontree-1.0.5.tar.gz
Algorithm Hash digest
SHA256 d7b149039056cee7da56fe633b8788c5f35a8c919794b69560820174e178416b
MD5 42d7a97a4a91ed6e6a5b62e40e70801e
BLAKE2b-256 78821a1d69fde9d5200f689338913d71d18b9abc5b4a827253c48e7e3f9ecb8c

See more details on using hashes here.

File details

Details for the file nontree-1.0.5-py3-none-any.whl.

File metadata

  • Download URL: nontree-1.0.5-py3-none-any.whl
  • Upload date:
  • Size: 13.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.1 CPython/3.10.8

File hashes

Hashes for nontree-1.0.5-py3-none-any.whl
Algorithm Hash digest
SHA256 257904a01556a979a73d4e1621279581625941d5b50b885bf765049dbf41c3ea
MD5 83abc1b31de46521fafe10196761e8cd
BLAKE2b-256 2dd98dc9884fdac8650e1eb4a0141796fcfe5f5ff6c276f8c54908925dcfd5a0

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page