Skip to content
Snippets Groups Projects
BarnesHutTree.cpp 1.4 KiB
Newer Older
#include "BarnesHutTree.hpp"
#include "TreeNode.hpp"

namespace nbody {
	BarnesHutTree::BarnesHutTree() {
	}

	BarnesHutTree::~BarnesHutTree() {
	}

	vector<Box> BarnesHutTree::splitBB(TreeNode* node) {
		vector<Box> result;

		for (int i = 0; i < 8; i++) {
			Box current = node->getBB();

			for (int j = 0; j < 3; j++) {
				double middle = node->getBB().getMin(j) + (node->getBB().getMax(j) - node->getBB().getMin(j)) / 2.0;

				if (i & (1 >> j)) {
					current.setMin(j, middle);
				} else {
					current.setMax(j, middle);
				}
			}
			result.push_back(current);
		}
		return result;
	}

	void BarnesHutTree::build(vector<Body> bodies) {
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		vector<TreeNode*>::iterator it;

Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		if (bodies.empty()) return;
		this->nodes.push_back(new TreeNode(this));
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		this->nodes.front()->bodies = bodies;
		this->nodes.front()->extendBBforBodies();
		this->nodes.front()->extendBBtoCube();
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		it = this->nodes.begin();
		while (it != this->nodes.end()) {
			(*it)->afterSubtree = it;
			(*it)->afterSubtree++;
			if ((*it)->isSplitable()) {
				vector<Box> subboxes = this->splitBB(*it);
				for (vector<Box>::iterator bit = subboxes.begin(); bit != subboxes.end(); bit++) {
					TreeNode* current = new TreeNode(this);

					current->bb = *bit;
					current->bodies = bit->containedBodies((*it)->getBodies());
					this->nodes.insert((*it)->afterSubtree, current);
				}
				(*it)->bodies.clear();
			}
			it++;
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		}