Skip to content
Snippets Groups Projects
TreeNode.cpp 3.24 KiB
Newer Older
#include <cstdlib>
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
#include <iostream>
#include "TreeNode.hpp"

namespace nbody {
	using namespace std;


Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
	TreeNode::TreeNode(Tree* tree) {
		for (int i = 0; i < 3; i++) {
			this->bb.min[i] = 1.0;
			this->bb.max[i] = -1.0;
		}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		this->afterSubtree = NULL;
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		this->prev = this;
		this->next = this;
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		this->leaf = true;
		this->tree = tree;
	}

	Box TreeNode::getBB() {
		return this->bb;
	}

	bool TreeNode::isSplitable() {
		if (this->bodies.size() < 2) {
			return false;
		}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		if (this->bb.volume() <= FLT_EPSILON) {
			return false;
		}
		return true;
	}

	void TreeNode::extendBBforBodies() {
		if (this->bodies.empty()) return;
		for (int i = 0; i < 3; i++) {
			this->bb.min[i] = this->bodies.front().getPosition(i);
			this->bb.max[i] = this->bodies.front().getPosition(i);
		}
		for (vector<Body>::iterator it = this->bodies.begin(); it != this->bodies.end(); it++) {
			for (int i = 0; i < 3; i++) {
				if (it->getPosition(i) < this->bb.min[i]) {
					this->bb.min[i] = it->getPosition(i);
				}
				if (it->getPosition(i) > this->bb.max[i]) {
					this->bb.max[i] = it->getPosition(i);
				}
			}
		}
	}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed

	void TreeNode::extendBBtoCube() {
		this->bb.extendToCube();
	}

	vector<Body> TreeNode::getBodies() {
		return this->bodies;
	}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed

	void TreeNode::insertBefore(TreeNode* node) {
		node->next = this;
		node->prev = this->prev;
		this->prev->next = node;
		this->prev = node;
	}

	void TreeNode::insertAfter(TreeNode* node) {
		this->next->insertBefore(node);
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
	}

	void TreeNode::unlink() {
		this->next->prev = this->prev;
		this->prev->next = this->next;
		this->next = this;
		this->prev = this;
	}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed

	bool TreeNode::isCorrect() {
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		if (this->afterSubtree == NULL) {
			cout << "after subtree null" << endl;
			return false;
		}
		if (!this->bb.isCorrect()) {
			cout << "bb wrong" << endl;
			return false;
		}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		for (int i = 0; i < 3; i++) {
			if (this->bb.min[i] > this->bb.max[i]) {
				cout << "bb " << i << " min " << this->bb.min[i] << " " << this->bb.max[i] << endl;
				return false;
			}
		}
		for (vector<Body>::iterator it = this->bodies.begin(); it != this->bodies.end(); it++) {
			if (!this->bb.isContained(*it)) {
				cout << "bb out of bounds" << endl;
				return false;
			}
		}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		if (!this->leaf) {
			TreeNode* current = this->next;
			int children = 0;
			while (current != NULL && current != this->afterSubtree) {
				current = current->afterSubtree;
				children++;
			}
			if (current == NULL) {
				cout << "afterSubtree null" << endl;
				return false;
			}
			if (children != this->tree->numberOfChildren()) {
				cout << "wrong number of children " << children << endl;
				return false;
			}
			current = this->next;
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
			for (int i = 0; i < this->tree->numberOfChildren(); i++) {
				current = current->afterSubtree;
			}
			if (current != this->afterSubtree) {
				cout << "last sibling afterSubtree inconsistent" << endl;
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
				return false;
			}
		}
		if (!this->leaf && this->bodies.size() > 0) {
			cout << "non-empty inner node" << endl;
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
			return false;
		}
Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
		return true;
	}
	void TreeNode::update() {
		if (this->leaf) {

		}
	}

Paul Heinzlreiter's avatar
Paul Heinzlreiter committed
	void TreeNode::print() {
		this->bb.print();
		for (vector<Body>::iterator it = this->bodies.begin(); it != this->bodies.end(); it++) {
			cout << "  ";
			it->print();
		}
	}