/* proctree.js Copyright (c) 2012, Paul Brunt c++ port Copyright (c) 2015, Jari Komppa All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * Neither the name of proctree.js nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL PAUL BRUNT BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include "proctree.h" #include "core/math/math_funcs.h" #ifndef M_PI #define M_PI 3.1415926535897932384626433832795f #endif namespace Proctree { float length(fvec3 a) { return sqrt(a.x * a.x + a.y * a.y + a.z * a.z); } fvec3 normalize(fvec3 a) { float l = length(a); if (l != 0) { l = 1.0f / l; a.x *= l; a.y *= l; a.z *= l; } return a; } fvec3 cross(fvec3 a, fvec3 b) { fvec3 c = { a.y * b.z - a.z * b.y, a.z * b.x - a.x * b.z, a.x * b.y - a.y * b.x }; return c; } float dot(fvec3 a, fvec3 b) { return a.x * b.x + a.y * b.y + a.z * b.z; } fvec3 sub(fvec3 a, fvec3 b) { a.x -= b.x; a.y -= b.y; a.z -= b.z; return a; } fvec3 add(fvec3 a, fvec3 b) { a.x += b.x; a.y += b.y; a.z += b.z; return a; } fvec3 scaleVec(fvec3 a, float b) { a.x *= b; a.y *= b; a.z *= b; return a; } fvec3 scaleInDirection(fvec3 aVector, fvec3 aDirection, float aScale) { float currentMag = dot(aVector, aDirection); fvec3 change = scaleVec(aDirection, currentMag * aScale - currentMag); return add(aVector, change); } fvec3 vecAxisAngle(fvec3 aVec, fvec3 aAxis, float aAngle) { //v std::cos(T) + (axis x v) * std::sin(T) + axis*(axis . v)(1-std::cos(T) float cosr = Math::cos(aAngle); float sinr = Math::sin(aAngle); return add(add(scaleVec(aVec, cosr), scaleVec(cross(aAxis, aVec), sinr)), scaleVec(aAxis, dot(aAxis, aVec) * (1 - cosr))); } fvec3 mirrorBranch(fvec3 aVec, fvec3 aNorm, Properties &aProperties) { fvec3 v = cross(aNorm, cross(aVec, aNorm)); float s = aProperties.mBranchFactor * dot(v, aVec); fvec3 res = { aVec.x - v.x * s, aVec.y - v.y * s, aVec.z - v.z * s }; return res; } Properties::Properties( float aClumpMax, float aClumpMin, float aLengthFalloffFactor, float aLengthFalloffPower, float aBranchFactor, float aRadiusFalloffRate, float aClimbRate, float aTrunkKink, float aMaxRadius, int aTreeSteps, float aTaperRate, float aTwistRate, int aSegments, int aLevels, float aSweepAmount, float aInitialBranchLength, float aTrunkLength, float aDropAmount, float aGrowAmount, float aVMultiplier, float aTwigScale, int aSeed) { mSeed = aSeed; mSegments = aSegments; mLevels = aLevels; mVMultiplier = aVMultiplier; mTwigScale = aTwigScale; mInitialBranchLength = aInitialBranchLength; mLengthFalloffFactor = aLengthFalloffFactor; mLengthFalloffPower = aLengthFalloffPower; mClumpMax = aClumpMax; mClumpMin = aClumpMin; mBranchFactor = aBranchFactor; mDropAmount = aDropAmount; mGrowAmount = aGrowAmount; mSweepAmount = aSweepAmount; mMaxRadius = aMaxRadius; mClimbRate = aClimbRate; mTrunkKink = aTrunkKink; mTreeSteps = aTreeSteps; mTaperRate = aTaperRate; mRadiusFalloffRate = aRadiusFalloffRate; mTwistRate = aTwistRate; mTrunkLength = aTrunkLength; } Properties::Properties() { mSeed = 262; mSegments = 6; mLevels = 5; mVMultiplier = 0.36f; mTwigScale = 0.39f; mInitialBranchLength = 0.49f; mLengthFalloffFactor = 0.85f; mLengthFalloffPower = 0.99f; mClumpMax = 0.454f; mClumpMin = 0.404f; mBranchFactor = 2.45f; mDropAmount = -0.1f; mGrowAmount = 0.235f; mSweepAmount = 0.01f; mMaxRadius = 0.139f; mClimbRate = 0.371f; mTrunkKink = 0.093f; mTreeSteps = 5; mTaperRate = 0.947f; mRadiusFalloffRate = 0.73f; mTwistRate = 3.02f; mTrunkLength = 2.4f; } float Properties::random(float aFixed) { if (!aFixed) { aFixed = (float)mRseed++; } return Math::abs(Math::cos(aFixed + aFixed * aFixed)); } Branch::~Branch() { delete mChild0; delete mChild1; delete[] mRootRing; delete[] mRing0; delete[] mRing1; delete[] mRing2; } Branch::Branch() { mRootRing = 0; mRing0 = 0; mRing1 = 0; mRing2 = 0; mChild0 = 0; mChild1 = 0; mParent = 0; mLength = 1; mTrunktype = 0; mRadius = 0; mHead = { 0, 0, 0 }; mTangent = { 0, 0, 0 }; mEnd = 0; } Branch::Branch(fvec3 aHead, Branch *aParent) { mRootRing = 0; mRing0 = 0; mRing1 = 0; mRing2 = 0; mChild0 = 0; mChild1 = 0; mLength = 1; mTrunktype = 0; mRadius = 0; mHead = aHead; mTangent = { 0, 0, 0 }; mParent = aParent; mEnd = 0; } void Branch::split(int aLevel, int aSteps, Properties &aProperties, int aL1/* = 1*/, int aL2/* = 1*/) { int rLevel = aProperties.mLevels - aLevel; fvec3 po; if (this->mParent) { po = mParent->mHead; } else { po = { 0, 0, 0 }; mTrunktype = 1; } fvec3 so = mHead; fvec3 dir = normalize(sub(so, po)); fvec3 a = { dir.z, dir.x, dir.y }; fvec3 normal = cross(dir, a); fvec3 tangent = cross(dir, normal); float r = aProperties.random(rLevel * 10 + aL1 * 5.0f + aL2 + aProperties.mSeed); //float r2 = aProperties.random(rLevel * 10 + aL1 * 5.0f + aL2 + 1 + aProperties.seed); // never used fvec3 adj = add(scaleVec(normal, r), scaleVec(tangent, 1 - r)); if (r > 0.5) adj = scaleVec(adj, -1); float clump = (aProperties.mClumpMax - aProperties.mClumpMin) * r + aProperties.mClumpMin; fvec3 newdir = normalize(add(scaleVec(adj, 1 - clump), scaleVec(dir, clump))); fvec3 newdir2 = mirrorBranch(newdir, dir, aProperties); if (r > 0.5) { fvec3 tmp = newdir; newdir = newdir2; newdir2 = tmp; } if (aSteps > 0) { float angle = aSteps / (float)aProperties.mTreeSteps * 2 * M_PI * aProperties.mTwistRate; a = { Math::sin(angle), r, Math::cos(angle) }; newdir2 = normalize(a); } float growAmount = aLevel * aLevel / (float)(aProperties.mLevels * aProperties.mLevels) * aProperties.mGrowAmount; float dropAmount = rLevel * aProperties.mDropAmount; float sweepAmount = rLevel * aProperties.mSweepAmount; a = { sweepAmount, dropAmount + growAmount, 0 }; newdir = normalize(add(newdir, a)); newdir2 = normalize(add(newdir2, a)); fvec3 head0 = add(so, scaleVec(newdir, mLength)); fvec3 head1 = add(so, scaleVec(newdir2, mLength)); mChild0 = new Branch(head0, this); mChild1 = new Branch(head1, this); mChild0->mLength = pow(mLength, aProperties.mLengthFalloffPower) * aProperties.mLengthFalloffFactor; mChild1->mLength = pow(mLength, aProperties.mLengthFalloffPower) * aProperties.mLengthFalloffFactor; if (aLevel > 0) { if (aSteps > 0) { a = { (r - 0.5f) * 2 * aProperties.mTrunkKink, aProperties.mClimbRate, (r - 0.5f) * 2 * aProperties.mTrunkKink }; mChild0->mHead = add(mHead, a); mChild0->mTrunktype = 1; mChild0->mLength = mLength * aProperties.mTaperRate; mChild0->split(aLevel, aSteps - 1, aProperties, aL1 + 1, aL2); } else { mChild0->split(aLevel - 1, 0, aProperties, aL1 + 1, aL2); } mChild1->split(aLevel - 1, 0, aProperties, aL1, aL2 + 1); } } Tree::Tree() { mRoot = 0; mVert = 0; mNormal = 0; mUV = 0; mTwigVert = 0; mTwigNormal = 0; mTwigUV = 0; mFace = 0; mTwigFace = 0; mVertCount = 0; mTwigVertCount = 0; mFaceCount = 0; mTwigFaceCount = 0; } Tree::~Tree() { delete[] mRoot; delete[] mVert; delete[] mNormal; delete[] mUV; delete[] mTwigVert; delete[] mTwigNormal; delete[] mTwigUV; delete[] mFace; delete[] mTwigFace; } void Tree::init() { mVertCount = 0; mTwigVertCount = 0; mFaceCount = 0; mTwigFaceCount = 0; delete[] mRoot; delete[] mVert; delete[] mNormal; delete[] mUV; delete[] mTwigVert; delete[] mTwigNormal; delete[] mTwigUV; delete[] mFace; delete[] mTwigFace; mRoot = 0; mVert = 0; mNormal = 0; mUV = 0; mTwigVert = 0; mTwigNormal = 0; mTwigUV = 0; mFace = 0; mTwigFace = 0; } void Tree::allocVertBuffers() { mVert = new fvec3[mVertCount]; mNormal = new fvec3[mVertCount]; mUV = new fvec2[mVertCount]; mTwigVert = new fvec3[mTwigVertCount]; mTwigNormal = new fvec3[mTwigVertCount]; mTwigUV = new fvec2[mTwigVertCount]; mTwigFace = new ivec3[mTwigFaceCount]; // Reset back to zero, we'll use these as counters mVertCount = 0; mTwigVertCount = 0; mTwigFaceCount = 0; } void Tree::allocFaceBuffers() { mFace = new ivec3[mFaceCount]; // Reset back to zero, we'll use these as counters mFaceCount = 0; } void Tree::generate() { init(); mProperties.mRseed = mProperties.mSeed; fvec3 starthead = { 0, mProperties.mTrunkLength, 0 }; mRoot = new Branch(starthead, 0); mRoot->mLength = mProperties.mInitialBranchLength; mRoot->split(mProperties.mLevels, mProperties.mTreeSteps, mProperties); calcVertSizes(0); allocVertBuffers(); createForks(0, 0); createTwigs(0); calcFaceSizes(0); allocFaceBuffers(); doFaces(0); calcNormals(); fixUVs(); delete mRoot; mRoot = 0; } void Tree::fixUVs() { // There'll never be more than 50% bad vertices int *badverttable = new int[mVertCount / 2]; int i; int badverts = 0; // step 1: find bad verts // - If edge's U coordinate delta is over 0.5, texture has wrapped around. // - The vertex that has zero U is the wrong one // - Care needs to be taken not to tag bad vertex more than once. for (i = 0; i < mFaceCount; i++) { // x/y edges (vertex 0 and 1) if ((Math::abs(mUV[mFace[i].x].u - mUV[mFace[i].y].u) > 0.5f) && (mUV[mFace[i].x].u == 0 || mUV[mFace[i].y].u == 0)) { int found = 0, j; for (j = 0; j < badverts; j++) { if (badverttable[j] == mFace[i].y && mUV[mFace[i].y].u == 0) found = 1; if (badverttable[j] == mFace[i].x && mUV[mFace[i].x].u == 0) found = 1; } if (!found) { if (mUV[mFace[i].x].u == 0) badverttable[badverts] = mFace[i].x; if (mUV[mFace[i].y].u == 0) badverttable[badverts] = mFace[i].y; badverts++; } } // x/z edges (vertex 0 and 2) if ((Math::abs(mUV[mFace[i].x].u - mUV[mFace[i].z].u) > 0.5f) && (mUV[mFace[i].x].u == 0 || mUV[mFace[i].z].u == 0)) { int found = 0, j; for (j = 0; j < badverts; j++) { if (badverttable[j] == mFace[i].z && mUV[mFace[i].z].u == 0) found = 1; if (badverttable[j] == mFace[i].x && mUV[mFace[i].x].u == 0) found = 1; } if (!found) { if (mUV[mFace[i].x].u == 0) badverttable[badverts] = mFace[i].x; if (mUV[mFace[i].z].u == 0) badverttable[badverts] = mFace[i].z; badverts++; } } // y/z edges (vertex 1 and 2) if ((Math::abs(mUV[mFace[i].y].u - mUV[mFace[i].z].u) > 0.5f) && (mUV[mFace[i].y].u == 0 || mUV[mFace[i].z].u == 0)) { int found = 0, j; for (j = 0; j < badverts; j++) { if (badverttable[j] == mFace[i].z && mUV[mFace[i].z].u == 0) found = 1; if (badverttable[j] == mFace[i].y && mUV[mFace[i].y].u == 0) found = 1; } if (!found) { if (mUV[mFace[i].y].u == 0) badverttable[badverts] = mFace[i].y; if (mUV[mFace[i].z].u == 0) badverttable[badverts] = mFace[i].z; badverts++; } } } // step 2: allocate more space for our new duplicate verts fvec3 *nvert = new fvec3[mVertCount + badverts]; memcpy(nvert, mVert, sizeof(fvec3) * mVertCount); delete[] mVert; mVert = nvert; fvec3 *nnorm = new fvec3[mVertCount + badverts]; memcpy(nnorm, mNormal, sizeof(fvec3) * mVertCount); delete[] mNormal; mNormal = nnorm; fvec2 *nuv = new fvec2[mVertCount + badverts]; memcpy(nuv, mUV, sizeof(fvec2) * mVertCount); delete[] mUV; mUV = nuv; // step 3: populate duplicate verts - otherwise identical except for U=1 instead of 0 for (i = 0; i < badverts; i++) { mVert[mVertCount + i] = mVert[badverttable[i]]; mNormal[mVertCount + i] = mNormal[badverttable[i]]; mUV[mVertCount + i] = mUV[badverttable[i]]; mUV[mVertCount + i].u = 1.0f; } // step 4: fix faces for (i = 0; i < mFaceCount; i++) { // x/y edges (vertex 0 and 1) if ((Math::abs(mUV[mFace[i].x].u - mUV[mFace[i].y].u) > 0.5f) && (mUV[mFace[i].x].u == 0 || mUV[mFace[i].y].u == 0)) { int found = 0, j; for (j = 0; j < badverts; j++) { if (badverttable[j] == mFace[i].y && mUV[mFace[i].y].u == 0) found = j; if (badverttable[j] == mFace[i].x && mUV[mFace[i].x].u == 0) found = j; } if (mUV[mFace[i].y].u == 0) mFace[i].y = mVertCount + found; if (mUV[mFace[i].x].u == 0) mFace[i].x = mVertCount + found; } // x/z edges (vertex 0 and 2) if ((Math::abs(mUV[mFace[i].x].u - mUV[mFace[i].z].u) > 0.5f) && (mUV[mFace[i].x].u == 0 || mUV[mFace[i].z].u == 0)) { int found = 0, j; for (j = 0; j < badverts; j++) { if (badverttable[j] == mFace[i].z && mUV[mFace[i].z].u == 0) found = j; if (badverttable[j] == mFace[i].x && mUV[mFace[i].x].u == 0) found = j; } if (mUV[mFace[i].x].u == 0) mFace[i].x = mVertCount + found; if (mUV[mFace[i].z].u == 0) mFace[i].z = mVertCount + found; } // y/z edges (vertex 1 and 2) if ((Math::abs(mUV[mFace[i].y].u - mUV[mFace[i].z].u) > 0.5f) && (mUV[mFace[i].y].u == 0 || mUV[mFace[i].z].u == 0)) { int found = 0, j; for (j = 0; j < badverts; j++) { if (badverttable[j] == mFace[i].z && mUV[mFace[i].z].u == 0) found = j; if (badverttable[j] == mFace[i].y && mUV[mFace[i].y].u == 0) found = j; } if (mUV[mFace[i].y].u == 0) mFace[i].y = mVertCount + found; if (mUV[mFace[i].z].u == 0) mFace[i].z = mVertCount + found; } } // step 5: update vert count mVertCount += badverts; // and cleanup delete[] badverttable; } void Tree::calcVertSizes(Branch *aBranch) { int segments = mProperties.mSegments; if (!aBranch) aBranch = mRoot; if (!aBranch->mParent) { mVertCount += segments; } if (aBranch->mChild0) { mVertCount += 1 + (segments / 2) - 1 + 1 + (segments / 2) - 1 + (segments / 2) - 1; calcVertSizes(aBranch->mChild0); calcVertSizes(aBranch->mChild1); } else { mVertCount++; mTwigVertCount += 8; mTwigFaceCount += 4; } } void Tree::calcFaceSizes(Branch *aBranch) { int segments = mProperties.mSegments; if (!aBranch) aBranch = mRoot; if (!aBranch->mParent) { mFaceCount += segments * 2; } if (aBranch->mChild0->mRing0 != 0) { mFaceCount += segments * 4; calcFaceSizes(aBranch->mChild0); calcFaceSizes(aBranch->mChild1); } else { mFaceCount += segments * 2; } } void Tree::calcNormals() { int *normalCount = new int[mVertCount]; memset(normalCount, 0, sizeof(int) * mVertCount); memset(mNormal, 0, sizeof(fvec3) * mVertCount); int i; for (i = 0; i < (int)mFaceCount; i++) { normalCount[mFace[i].x]++; normalCount[mFace[i].y]++; normalCount[mFace[i].z]++; fvec3 norm = normalize(cross(sub(mVert[mFace[i].y], mVert[mFace[i].z]), sub(mVert[mFace[i].y], mVert[mFace[i].x]))); mNormal[mFace[i].x].x += norm.x; mNormal[mFace[i].x].y += norm.y; mNormal[mFace[i].x].z += norm.z; mNormal[mFace[i].y].x += norm.x; mNormal[mFace[i].y].y += norm.y; mNormal[mFace[i].y].z += norm.z; mNormal[mFace[i].z].x += norm.x; mNormal[mFace[i].z].y += norm.y; mNormal[mFace[i].z].z += norm.z; } for (i = 0; i < (int)mVertCount; i++) { float d = 1.0f / normalCount[i]; mNormal[i].x *= d; mNormal[i].y *= d; mNormal[i].z *= d; } delete[] normalCount; } void Tree::doFaces(Branch *aBranch) { if (!aBranch) { aBranch = mRoot; } int segments = mProperties.mSegments; int i; if (!aBranch->mParent) { fvec3 tangent = normalize(cross(sub(aBranch->mChild0->mHead, aBranch->mHead), sub(aBranch->mChild1->mHead, aBranch->mHead))); fvec3 normal = normalize(aBranch->mHead); fvec3 left = { -1, 0, 0 }; float angle = Math::acos(dot(tangent, left)); if (dot(cross(left, tangent), normal) > 0) { angle = 2 * M_PI - angle; } int segOffset = (int)floor(0.5f + (angle / M_PI / 2 * segments)); for (i = 0; i < segments; i++) { int v1 = aBranch->mRing0[i]; int v2 = aBranch->mRootRing[(i + segOffset + 1) % segments]; int v3 = aBranch->mRootRing[(i + segOffset) % segments]; int v4 = aBranch->mRing0[(i + 1) % segments]; ivec3 a; a = { v1, v4, v3 }; mFace[mFaceCount++] = (a); a = { v4, v2, v3 }; mFace[mFaceCount++] = (a); mUV[(i + segOffset) % segments] = { i / (float)segments, 0 }; float len = length(sub(mVert[aBranch->mRing0[i]], mVert[aBranch->mRootRing[(i + segOffset) % segments]])) * mProperties.mVMultiplier; mUV[aBranch->mRing0[i]] = { i / (float)segments, len }; mUV[aBranch->mRing2[i]] = { i / (float)segments, len }; } } if (aBranch->mChild0->mRing0 != 0) { int segOffset0 = -1, segOffset1 = -1; float match0, match1; fvec3 v1 = normalize(sub(mVert[aBranch->mRing1[0]], aBranch->mHead)); fvec3 v2 = normalize(sub(mVert[aBranch->mRing2[0]], aBranch->mHead)); v1 = scaleInDirection(v1, normalize(sub(aBranch->mChild0->mHead, aBranch->mHead)), 0); v2 = scaleInDirection(v2, normalize(sub(aBranch->mChild1->mHead, aBranch->mHead)), 0); for (i = 0; i < segments; i++) { fvec3 d = normalize(sub(mVert[aBranch->mChild0->mRing0[i]], aBranch->mChild0->mHead)); float l = dot(d, v1); if (segOffset0 == -1 || l > match0) { match0 = l; segOffset0 = segments - i; } d = normalize(sub(mVert[aBranch->mChild1->mRing0[i]], aBranch->mChild1->mHead)); l = dot(d, v2); if (segOffset1 == -1 || l > match1) { match1 = l; segOffset1 = segments - i; } } float UVScale = mProperties.mMaxRadius / aBranch->mRadius; for (i = 0; i < segments; i++) { int lv1 = aBranch->mChild0->mRing0[i]; int lv2 = aBranch->mRing1[(i + segOffset0 + 1) % segments]; int lv3 = aBranch->mRing1[(i + segOffset0) % segments]; int lv4 = aBranch->mChild0->mRing0[(i + 1) % segments]; ivec3 a; a = { lv1, lv4, lv3 }; mFace[mFaceCount++] = (a); a = { lv4, lv2, lv3 }; mFace[mFaceCount++] = (a); lv1 = aBranch->mChild1->mRing0[i]; lv2 = aBranch->mRing2[(i + segOffset1 + 1) % segments]; lv3 = aBranch->mRing2[(i + segOffset1) % segments]; lv4 = aBranch->mChild1->mRing0[(i + 1) % segments]; a = { lv1, lv2, lv3 }; mFace[mFaceCount++] = (a); a = { lv1, lv4, lv2 }; mFace[mFaceCount++] = (a); float len1 = length(sub(mVert[aBranch->mChild0->mRing0[i]], mVert[aBranch->mRing1[(i + segOffset0) % segments]])) * UVScale; fvec2 uv1 = mUV[aBranch->mRing1[(i + segOffset0 - 1) % segments]]; mUV[aBranch->mChild0->mRing0[i]] = { uv1.u, uv1.v + len1 * mProperties.mVMultiplier }; mUV[aBranch->mChild0->mRing2[i]] = { uv1.u, uv1.v + len1 * mProperties.mVMultiplier }; float len2 = length(sub(mVert[aBranch->mChild1->mRing0[i]], mVert[aBranch->mRing2[(i + segOffset1) % segments]])) * UVScale; fvec2 uv2 = mUV[aBranch->mRing2[(i + segOffset1 - 1) % segments]]; mUV[aBranch->mChild1->mRing0[i]] = { uv2.u, uv2.v + len2 * mProperties.mVMultiplier }; mUV[aBranch->mChild1->mRing2[i]] = { uv2.u, uv2.v + len2 * mProperties.mVMultiplier }; } doFaces(aBranch->mChild0); doFaces(aBranch->mChild1); } else { for (i = 0; i < segments; i++) { ivec3 a; a = { aBranch->mChild0->mEnd, aBranch->mRing1[(i + 1) % segments], aBranch->mRing1[i] }; mFace[mFaceCount++] = (a); a = { aBranch->mChild1->mEnd, aBranch->mRing2[(i + 1) % segments], aBranch->mRing2[i] }; mFace[mFaceCount++] = (a); float len = length(sub(mVert[aBranch->mChild0->mEnd], mVert[aBranch->mRing1[i]])); mUV[aBranch->mChild0->mEnd] = { i / (float)segments - 1, len * mProperties.mVMultiplier }; len = length(sub(mVert[aBranch->mChild1->mEnd], mVert[aBranch->mRing2[i]])); mUV[aBranch->mChild1->mEnd] = { i / (float)segments, len * mProperties.mVMultiplier }; } } } void Tree::createTwigs(Branch *aBranch) { if (!aBranch) { aBranch = mRoot; } if (!aBranch->mChild0) { fvec3 tangent = normalize(cross(sub(aBranch->mParent->mChild0->mHead, aBranch->mParent->mHead), sub(aBranch->mParent->mChild1->mHead, aBranch->mParent->mHead))); fvec3 binormal = normalize(sub(aBranch->mHead, aBranch->mParent->mHead)); //fvec3 normal = cross(tangent, binormal); //never used int vert1 = mTwigVertCount; mTwigVert[mTwigVertCount++] = (add(add(aBranch->mHead, scaleVec(tangent, mProperties.mTwigScale)), scaleVec(binormal, mProperties.mTwigScale * 2 - aBranch->mLength))); int vert2 = mTwigVertCount; mTwigVert[mTwigVertCount++] = (add(add(aBranch->mHead, scaleVec(tangent, -mProperties.mTwigScale)), scaleVec(binormal, mProperties.mTwigScale * 2 - aBranch->mLength))); int vert3 = mTwigVertCount; mTwigVert[mTwigVertCount++] = (add(add(aBranch->mHead, scaleVec(tangent, -mProperties.mTwigScale)), scaleVec(binormal, -aBranch->mLength))); int vert4 = mTwigVertCount; mTwigVert[mTwigVertCount++] = (add(add(aBranch->mHead, scaleVec(tangent, mProperties.mTwigScale)), scaleVec(binormal, -aBranch->mLength))); int vert8 = mTwigVertCount; mTwigVert[mTwigVertCount++] = (add(add(aBranch->mHead, scaleVec(tangent, mProperties.mTwigScale)), scaleVec(binormal, mProperties.mTwigScale * 2 - aBranch->mLength))); int vert7 = mTwigVertCount; mTwigVert[mTwigVertCount++] = (add(add(aBranch->mHead, scaleVec(tangent, -mProperties.mTwigScale)), scaleVec(binormal, mProperties.mTwigScale * 2 - aBranch->mLength))); int vert6 = mTwigVertCount; mTwigVert[mTwigVertCount++] = (add(add(aBranch->mHead, scaleVec(tangent, -mProperties.mTwigScale)), scaleVec(binormal, -aBranch->mLength))); int vert5 = mTwigVertCount; mTwigVert[mTwigVertCount++] = (add(add(aBranch->mHead, scaleVec(tangent, mProperties.mTwigScale)), scaleVec(binormal, -aBranch->mLength))); mTwigFace[mTwigFaceCount++] = { vert1, vert2, vert3 }; mTwigFace[mTwigFaceCount++] = { vert4, vert1, vert3 }; mTwigFace[mTwigFaceCount++] = { vert6, vert7, vert8 }; mTwigFace[mTwigFaceCount++] = { vert6, vert8, vert5 }; fvec3 normal = normalize(cross(sub(mTwigVert[vert1], mTwigVert[vert3]), sub(mTwigVert[vert2], mTwigVert[vert3]))); fvec3 normal2 = normalize(cross(sub(mTwigVert[vert7], mTwigVert[vert6]), sub(mTwigVert[vert8], mTwigVert[vert6]))); mTwigNormal[vert1] = (normal); mTwigNormal[vert2] = (normal); mTwigNormal[vert3] = (normal); mTwigNormal[vert4] = (normal); mTwigNormal[vert8] = (normal2); mTwigNormal[vert7] = (normal2); mTwigNormal[vert6] = (normal2); mTwigNormal[vert5] = (normal2); mTwigUV[vert1] = { 0, 0 }; mTwigUV[vert2] = { 1, 0 }; mTwigUV[vert3] = { 1, 1 }; mTwigUV[vert4] = { 0, 1 }; mTwigUV[vert8] = { 0, 0 }; mTwigUV[vert7] = { 1, 0 }; mTwigUV[vert6] = { 1, 1 }; mTwigUV[vert5] = { 0, 1 }; } else { createTwigs(aBranch->mChild0); createTwigs(aBranch->mChild1); } } void Tree::createForks(Branch *aBranch, float aRadius) { if (!aBranch) aBranch = mRoot; if (!aRadius) aRadius = mProperties.mMaxRadius; aBranch->mRadius = aRadius; if (aRadius > aBranch->mLength) aRadius = aBranch->mLength; int segments = mProperties.mSegments; float segmentAngle = M_PI * 2 / (float)segments; if (!aBranch->mParent) { aBranch->mRootRing = new int[segments]; //create the root of the tree //branch.root = []; fvec3 axis = { 0, 1, 0 }; int i; for (i = 0; i < segments; i++) { fvec3 left = { -1, 0, 0 }; fvec3 vec = vecAxisAngle(left, axis, -segmentAngle * i); aBranch->mRootRing[i] = mVertCount; mVert[mVertCount++] = (scaleVec(vec, aRadius / mProperties.mRadiusFalloffRate)); } } //cross the branches to get the left //add the branches to get the up if (aBranch->mChild0) { fvec3 axis; if (aBranch->mParent) { axis = normalize(sub(aBranch->mHead, aBranch->mParent->mHead)); } else { axis = normalize(aBranch->mHead); } fvec3 axis1 = normalize(sub(aBranch->mHead, aBranch->mChild0->mHead)); fvec3 axis2 = normalize(sub(aBranch->mHead, aBranch->mChild1->mHead)); fvec3 tangent = normalize(cross(axis1, axis2)); aBranch->mTangent = tangent; fvec3 axis3 = normalize(cross(tangent, normalize(add(scaleVec(axis1, -1), scaleVec(axis2, -1))))); fvec3 dir = { axis2.x, 0, axis2.z }; fvec3 centerloc = add(aBranch->mHead, scaleVec(dir, -mProperties.mMaxRadius / 2)); aBranch->mRing0 = new int[segments]; aBranch->mRing1 = new int[segments]; aBranch->mRing2 = new int[segments]; int ring0count = 0; int ring1count = 0; int ring2count = 0; float scale = mProperties.mRadiusFalloffRate; if (aBranch->mChild0->mTrunktype || aBranch->mTrunktype) { scale = 1.0f / mProperties.mTaperRate; } //main segment ring int linch0 = mVertCount; aBranch->mRing0[ring0count++] = linch0; aBranch->mRing2[ring2count++] = linch0; mVert[mVertCount++] = (add(centerloc, scaleVec(tangent, aRadius * scale))); int start = mVertCount - 1; fvec3 d1 = vecAxisAngle(tangent, axis2, 1.57f); fvec3 d2 = normalize(cross(tangent, axis)); float s = 1 / dot(d1, d2); int i; for (i = 1; i < segments / 2; i++) { fvec3 vec = vecAxisAngle(tangent, axis2, segmentAngle * i); aBranch->mRing0[ring0count++] = start + i; aBranch->mRing2[ring2count++] = start + i; vec = scaleInDirection(vec, d2, s); mVert[mVertCount++] = (add(centerloc, scaleVec(vec, aRadius * scale))); } int linch1 = mVertCount; aBranch->mRing0[ring0count++] = linch1; aBranch->mRing1[ring1count++] = linch1; mVert[mVertCount++] = (add(centerloc, scaleVec(tangent, -aRadius * scale))); for (i = segments / 2 + 1; i < segments; i++) { fvec3 vec = vecAxisAngle(tangent, axis1, segmentAngle * i); aBranch->mRing0[ring0count++] = mVertCount; aBranch->mRing1[ring1count++] = mVertCount; mVert[mVertCount++] = (add(centerloc, scaleVec(vec, aRadius * scale))); } aBranch->mRing1[ring1count++] = linch0; aBranch->mRing2[ring2count++] = linch1; start = mVertCount - 1; for (i = 1; i < segments / 2; i++) { fvec3 vec = vecAxisAngle(tangent, axis3, segmentAngle * i); aBranch->mRing1[ring1count++] = start + i; aBranch->mRing2[ring2count++] = start + (segments / 2 - i); fvec3 v = scaleVec(vec, aRadius * scale); mVert[mVertCount++] = (add(centerloc, v)); } //child radius is related to the brans direction and the length of the branch //float length0 = length(sub(aBranch->mHead, aBranch->mChild0->mHead)); // never used //float length1 = length(sub(aBranch->mHead, aBranch->mChild1->mHead)); // never used float radius0 = 1 * aRadius * mProperties.mRadiusFalloffRate; float radius1 = 1 * aRadius * mProperties.mRadiusFalloffRate; if (aBranch->mChild0->mTrunktype) { radius0 = aRadius * mProperties.mTaperRate; } createForks(aBranch->mChild0, radius0); createForks(aBranch->mChild1, radius1); } else { //add points for the ends of braches aBranch->mEnd = mVertCount; //branch.head=add(branch.head,scaleVec([this.properties.xBias,this.properties.yBias,this.properties.zBias],branch.length*3)); mVert[mVertCount++] = (aBranch->mHead); } } }