From d71b949468cc94e320f33d6da630d0c7b87beee5 Mon Sep 17 00:00:00 2001 From: Amos <48657826+Mauler125@users.noreply.github.com> Date: Sun, 13 Mar 2022 01:15:52 +0100 Subject: [PATCH] Add Recast & Detour navmesh editor to SDK --- r5dev/naveditor/ChunkyTriMesh.cpp | 349 +++ r5dev/naveditor/ConvexVolumeTool.cpp | 297 +++ r5dev/naveditor/CrowdTool.cpp | 1107 ++++++++ r5dev/naveditor/Filelist.cpp | 78 + r5dev/naveditor/InputGeom.cpp | 661 +++++ r5dev/naveditor/MeshLoaderBsp.cpp | 85 + r5dev/naveditor/MeshLoaderObj.cpp | 257 ++ r5dev/naveditor/MeshLoaderPly.cpp | 155 ++ r5dev/naveditor/NavMeshPruneTool.cpp | 323 +++ r5dev/naveditor/NavMeshTesterTool.cpp | 1420 ++++++++++ r5dev/naveditor/OffMeshConnectionTool.cpp | 177 ++ r5dev/naveditor/PerfTimer.cpp | 59 + r5dev/naveditor/Sample.cpp | 698 +++++ r5dev/naveditor/SampleInterfaces.cpp | 318 +++ r5dev/naveditor/Sample_Debug.cpp | 387 +++ r5dev/naveditor/Sample_SoloMesh.cpp | 755 ++++++ r5dev/naveditor/Sample_TempObstacles.cpp | 1532 +++++++++++ r5dev/naveditor/Sample_TileMesh.cpp | 1221 +++++++++ r5dev/naveditor/TestCase.cpp | 463 ++++ r5dev/naveditor/ValueHistory.cpp | 114 + r5dev/naveditor/imgui.cpp | 676 +++++ r5dev/naveditor/imguiRenderGL.cpp | 529 ++++ r5dev/naveditor/include/ChunkyTriMesh.h | 62 + r5dev/naveditor/include/ConvexVolumeTool.h | 55 + r5dev/naveditor/include/CrowdTool.h | 144 ++ r5dev/naveditor/include/DroidSans.h | 2273 +++++++++++++++++ r5dev/naveditor/include/Filelist.h | 28 + r5dev/naveditor/include/InputGeom.h | 151 ++ r5dev/naveditor/include/MeshLoaderBsp.h | 51 + r5dev/naveditor/include/MeshLoaderObj.h | 74 + r5dev/naveditor/include/MeshLoaderPly.h | 50 + r5dev/naveditor/include/NavMeshPruneTool.h | 56 + r5dev/naveditor/include/NavMeshTesterTool.h | 113 + .../naveditor/include/OffMeshConnectionTool.h | 50 + r5dev/naveditor/include/PerfTimer.h | 32 + r5dev/naveditor/include/Sample.h | 206 ++ r5dev/naveditor/include/SampleInterfaces.h | 99 + r5dev/naveditor/include/Sample_Debug.h | 63 + r5dev/naveditor/include/Sample_SoloMesh.h | 86 + .../naveditor/include/Sample_TempObstacles.h | 98 + r5dev/naveditor/include/Sample_TileMesh.h | 114 + r5dev/naveditor/include/TestCase.h | 114 + r5dev/naveditor/include/ValueHistory.h | 50 + r5dev/naveditor/include/imgui.h | 108 + r5dev/naveditor/include/imguiRenderGL.h | 26 + r5dev/naveditor/main.cpp | 1234 +++++++++ r5dev/thirdparty/fastlz/fastlz.c | 556 ++++ r5dev/thirdparty/fastlz/fastlz.h | 100 + r5dev/thirdparty/sdl/src/stdlib/SDL_stdlib.c | 12 +- r5dev/vproj/libsdl.vcxproj | 1 + r5dev/vproj/libsdl.vcxproj.filters | 9 + r5dev/vproj/naveditor.vcxproj | 152 ++ r5dev/vproj/naveditor.vcxproj.filters | 110 + r5sdk.sln | 21 +- 54 files changed, 17952 insertions(+), 7 deletions(-) create mode 100644 r5dev/naveditor/ChunkyTriMesh.cpp create mode 100644 r5dev/naveditor/ConvexVolumeTool.cpp create mode 100644 r5dev/naveditor/CrowdTool.cpp create mode 100644 r5dev/naveditor/Filelist.cpp create mode 100644 r5dev/naveditor/InputGeom.cpp create mode 100644 r5dev/naveditor/MeshLoaderBsp.cpp create mode 100644 r5dev/naveditor/MeshLoaderObj.cpp create mode 100644 r5dev/naveditor/MeshLoaderPly.cpp create mode 100644 r5dev/naveditor/NavMeshPruneTool.cpp create mode 100644 r5dev/naveditor/NavMeshTesterTool.cpp create mode 100644 r5dev/naveditor/OffMeshConnectionTool.cpp create mode 100644 r5dev/naveditor/PerfTimer.cpp create mode 100644 r5dev/naveditor/Sample.cpp create mode 100644 r5dev/naveditor/SampleInterfaces.cpp create mode 100644 r5dev/naveditor/Sample_Debug.cpp create mode 100644 r5dev/naveditor/Sample_SoloMesh.cpp create mode 100644 r5dev/naveditor/Sample_TempObstacles.cpp create mode 100644 r5dev/naveditor/Sample_TileMesh.cpp create mode 100644 r5dev/naveditor/TestCase.cpp create mode 100644 r5dev/naveditor/ValueHistory.cpp create mode 100644 r5dev/naveditor/imgui.cpp create mode 100644 r5dev/naveditor/imguiRenderGL.cpp create mode 100644 r5dev/naveditor/include/ChunkyTriMesh.h create mode 100644 r5dev/naveditor/include/ConvexVolumeTool.h create mode 100644 r5dev/naveditor/include/CrowdTool.h create mode 100644 r5dev/naveditor/include/DroidSans.h create mode 100644 r5dev/naveditor/include/Filelist.h create mode 100644 r5dev/naveditor/include/InputGeom.h create mode 100644 r5dev/naveditor/include/MeshLoaderBsp.h create mode 100644 r5dev/naveditor/include/MeshLoaderObj.h create mode 100644 r5dev/naveditor/include/MeshLoaderPly.h create mode 100644 r5dev/naveditor/include/NavMeshPruneTool.h create mode 100644 r5dev/naveditor/include/NavMeshTesterTool.h create mode 100644 r5dev/naveditor/include/OffMeshConnectionTool.h create mode 100644 r5dev/naveditor/include/PerfTimer.h create mode 100644 r5dev/naveditor/include/Sample.h create mode 100644 r5dev/naveditor/include/SampleInterfaces.h create mode 100644 r5dev/naveditor/include/Sample_Debug.h create mode 100644 r5dev/naveditor/include/Sample_SoloMesh.h create mode 100644 r5dev/naveditor/include/Sample_TempObstacles.h create mode 100644 r5dev/naveditor/include/Sample_TileMesh.h create mode 100644 r5dev/naveditor/include/TestCase.h create mode 100644 r5dev/naveditor/include/ValueHistory.h create mode 100644 r5dev/naveditor/include/imgui.h create mode 100644 r5dev/naveditor/include/imguiRenderGL.h create mode 100644 r5dev/naveditor/main.cpp create mode 100644 r5dev/thirdparty/fastlz/fastlz.c create mode 100644 r5dev/thirdparty/fastlz/fastlz.h create mode 100644 r5dev/vproj/naveditor.vcxproj create mode 100644 r5dev/vproj/naveditor.vcxproj.filters diff --git a/r5dev/naveditor/ChunkyTriMesh.cpp b/r5dev/naveditor/ChunkyTriMesh.cpp new file mode 100644 index 00000000..52944e0f --- /dev/null +++ b/r5dev/naveditor/ChunkyTriMesh.cpp @@ -0,0 +1,349 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#include "NavEditor/Include/ChunkyTriMesh.h" +#include +#include +#include + +struct BoundsItem +{ + float bmin[2]; + float bmax[2]; + int i; +}; + +static int compareItemX(const void* va, const void* vb) +{ + const BoundsItem* a = (const BoundsItem*)va; + const BoundsItem* b = (const BoundsItem*)vb; + if (a->bmin[0] < b->bmin[0]) + return -1; + if (a->bmin[0] > b->bmin[0]) + return 1; + return 0; +} + +static int compareItemY(const void* va, const void* vb) +{ + const BoundsItem* a = (const BoundsItem*)va; + const BoundsItem* b = (const BoundsItem*)vb; + if (a->bmin[1] < b->bmin[1]) + return -1; + if (a->bmin[1] > b->bmin[1]) + return 1; + return 0; +} + +static void calcExtends(const BoundsItem* items, const int /*nitems*/, + const int imin, const int imax, + float* bmin, float* bmax) +{ + bmin[0] = items[imin].bmin[0]; + bmin[1] = items[imin].bmin[1]; + + bmax[0] = items[imin].bmax[0]; + bmax[1] = items[imin].bmax[1]; + + for (int i = imin+1; i < imax; ++i) + { + const BoundsItem& it = items[i]; + if (it.bmin[0] < bmin[0]) bmin[0] = it.bmin[0]; + if (it.bmin[1] < bmin[1]) bmin[1] = it.bmin[1]; + + if (it.bmax[0] > bmax[0]) bmax[0] = it.bmax[0]; + if (it.bmax[1] > bmax[1]) bmax[1] = it.bmax[1]; + } +} + +inline int longestAxis(float x, float y) +{ + return y > x ? 1 : 0; +} + +static void subdivide(BoundsItem* items, int nitems, int imin, int imax, int trisPerChunk, + int& curNode, rcChunkyTriMeshNode* nodes, const int maxNodes, + int& curTri, int* outTris, const int* inTris) +{ + int inum = imax - imin; + int icur = curNode; + + if (curNode >= maxNodes) + return; + + rcChunkyTriMeshNode& node = nodes[curNode++]; + + if (inum <= trisPerChunk) + { + // Leaf + calcExtends(items, nitems, imin, imax, node.bmin, node.bmax); + + // Copy triangles. + node.i = curTri; + node.n = inum; + + for (int i = imin; i < imax; ++i) + { + const int* src = &inTris[items[i].i*3]; + int* dst = &outTris[curTri*3]; + curTri++; + dst[0] = src[0]; + dst[1] = src[1]; + dst[2] = src[2]; + } + } + else + { + // Split + calcExtends(items, nitems, imin, imax, node.bmin, node.bmax); + + int axis = longestAxis(node.bmax[0] - node.bmin[0], + node.bmax[1] - node.bmin[1]); + + if (axis == 0) + { + // Sort along x-axis + qsort(items+imin, static_cast(inum), sizeof(BoundsItem), compareItemX); + } + else if (axis == 1) + { + // Sort along y-axis + qsort(items+imin, static_cast(inum), sizeof(BoundsItem), compareItemY); + } + + int isplit = imin+inum/2; + + // Left + subdivide(items, nitems, imin, isplit, trisPerChunk, curNode, nodes, maxNodes, curTri, outTris, inTris); + // Right + subdivide(items, nitems, isplit, imax, trisPerChunk, curNode, nodes, maxNodes, curTri, outTris, inTris); + + int iescape = curNode - icur; + // Negative index means escape. + node.i = -iescape; + } +} + +bool rcCreateChunkyTriMesh(const float* verts, const int* tris, int ntris, + int trisPerChunk, rcChunkyTriMesh* cm) +{ + int nchunks = (ntris + trisPerChunk-1) / trisPerChunk; + + cm->nodes = new rcChunkyTriMeshNode[nchunks*4]; + if (!cm->nodes) + return false; + + cm->tris = new int[ntris*3]; + if (!cm->tris) + return false; + + cm->ntris = ntris; + + // Build tree + BoundsItem* items = new BoundsItem[ntris]; + if (!items) + return false; + + for (int i = 0; i < ntris; i++) + { + const int* t = &tris[i*3]; + BoundsItem& it = items[i]; + it.i = i; + // Calc triangle XY bounds. + it.bmin[0] = it.bmax[0] = verts[t[0]*3+0]; + it.bmin[1] = it.bmax[1] = verts[t[0]*3+1]; + for (int j = 1; j < 3; ++j) + { + const float* v = &verts[t[j]*3]; + if (v[0] < it.bmin[0]) it.bmin[0] = v[0]; + if (v[1] < it.bmin[1]) it.bmin[1] = v[1]; + + if (v[0] > it.bmax[0]) it.bmax[0] = v[0]; + if (v[1] > it.bmax[1]) it.bmax[1] = v[1]; + } + } + + int curTri = 0; + int curNode = 0; + subdivide(items, ntris, 0, ntris, trisPerChunk, curNode, cm->nodes, nchunks*4, curTri, cm->tris, tris); + + delete [] items; + + cm->nnodes = curNode; + + // Calc max tris per node. + cm->maxTrisPerChunk = 0; + for (int i = 0; i < cm->nnodes; ++i) + { + rcChunkyTriMeshNode& node = cm->nodes[i]; + const bool isLeaf = node.i >= 0; + if (!isLeaf) continue; + if (node.n > cm->maxTrisPerChunk) + cm->maxTrisPerChunk = node.n; + } + + return true; +} + + +inline bool checkOverlapRect(const float amin[2], const float amax[2], + const float bmin[2], const float bmax[2]) +{ + bool overlap = true; + overlap = (amin[0] > bmax[0] || amax[0] < bmin[0]) ? false : overlap; + overlap = (amin[1] > bmax[1] || amax[1] < bmin[1]) ? false : overlap; + return overlap; +} + +int rcGetChunksOverlappingRect(const rcChunkyTriMesh* cm, + float bmin[2], float bmax[2], + int* ids, const int maxIds) +{ + // Traverse tree + int i = 0; + int n = 0; + while (i < cm->nnodes) + { + const rcChunkyTriMeshNode* node = &cm->nodes[i]; + const bool overlap = checkOverlapRect(bmin, bmax, node->bmin, node->bmax); + const bool isLeafNode = node->i >= 0; + + if (isLeafNode && overlap) + { + if (n < maxIds) + { + ids[n] = i; + n++; + } + } + + if (overlap || isLeafNode) + i++; + else + { + const int escapeIndex = -node->i; + i += escapeIndex; + } + } + + return n; +} + +int rcGetChunksOverlappingRect(const rcChunkyTriMesh * cm, float bmin[2], float bmax[2], int * ids, const int maxIds, int& count_returned, int & current_idx) +{ + // Traverse tree + while (current_idx < cm->nnodes) + { + const rcChunkyTriMeshNode* node = &cm->nodes[current_idx]; + const bool overlap = checkOverlapRect(bmin, bmax, node->bmin, node->bmax); + const bool isLeafNode = node->i >= 0; + + if (isLeafNode && overlap) + { + if (count_returned < maxIds) + { + ids[count_returned] = current_idx; + count_returned++; + } + } + + if (overlap || isLeafNode) + current_idx++; + else + { + const int escapeIndex = -node->i; + current_idx += escapeIndex; + } + if (count_returned == maxIds) + { + return 0; + } + } + //done with tree + return 1; +} + + + +static bool checkOverlapSegment(const float p[2], const float q[2], + const float bmin[2], const float bmax[2]) +{ + static const float EPSILON = 1e-6f; + + float tmin = 0; + float tmax = 1; + float d[2]; + d[0] = q[0] - p[0]; + d[1] = q[1] - p[1]; + + for (int i = 0; i < 2; i++) + { + if (fabsf(d[i]) < EPSILON) + { + // Ray is parallel to slab. No hit if origin not within slab + if (p[i] < bmin[i] || p[i] > bmax[i]) + return false; + } + else + { + // Compute intersection t value of ray with near and far plane of slab + float ood = 1.0f / d[i]; + float t1 = (bmin[i] - p[i]) * ood; + float t2 = (bmax[i] - p[i]) * ood; + if (t1 > t2) { float tmp = t1; t1 = t2; t2 = tmp; } + if (t1 > tmin) tmin = t1; + if (t2 < tmax) tmax = t2; + if (tmin > tmax) return false; + } + } + return true; +} + +int rcGetChunksOverlappingSegment(const rcChunkyTriMesh* cm, + float p[2], float q[2], + int* ids, const int maxIds) +{ + // Traverse tree + int i = 0; + int n = 0; + while (i < cm->nnodes) + { + const rcChunkyTriMeshNode* node = &cm->nodes[i]; + const bool overlap = checkOverlapSegment(p, q, node->bmin, node->bmax); + const bool isLeafNode = node->i >= 0; + + if (isLeafNode && overlap) + { + if (n < maxIds) + { + ids[n] = i; + n++; + } + } + + if (overlap || isLeafNode) + i++; + else + { + const int escapeIndex = -node->i; + i += escapeIndex; + } + } + + return n; +} diff --git a/r5dev/naveditor/ConvexVolumeTool.cpp b/r5dev/naveditor/ConvexVolumeTool.cpp new file mode 100644 index 00000000..14ef2d86 --- /dev/null +++ b/r5dev/naveditor/ConvexVolumeTool.cpp @@ -0,0 +1,297 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include +#include +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" +#include "NavEditor/Include/imgui.h" +#include "NavEditor/Include/ConvexVolumeTool.h" +#include "NavEditor/Include/InputGeom.h" +#include "NavEditor/Include/Sample.h" +#include "Recast/Include/Recast.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "DebugUtils/Include/DetourDebugDraw.h" + +#ifdef WIN32 +# define snprintf _snprintf +#endif + +// Quick and dirty convex hull. + +// Returns true if 'c' is left of line 'a'-'b'. +inline bool left(const float* a, const float* b, const float* c) +{ + const float u1 = b[0] - a[0]; + const float v1 = b[2] - a[2]; + const float u2 = c[0] - a[0]; + const float v2 = c[2] - a[2]; + return u1 * v2 - v1 * u2 < 0; +} + +// Returns true if 'a' is more lower-left than 'b'. +inline bool cmppt(const float* a, const float* b) +{ + if (a[0] < b[0]) return true; + if (a[0] > b[0]) return false; + if (a[2] < b[2]) return true; + if (a[2] > b[2]) return false; + return false; +} +// Calculates convex hull on xz-plane of points on 'pts', +// stores the indices of the resulting hull in 'out' and +// returns number of points on hull. +static int convexhull(const float* pts, int npts, int* out) +{ + // Find lower-leftmost point. + int hull = 0; + for (int i = 1; i < npts; ++i) + if (cmppt(&pts[i*3], &pts[hull*3])) + hull = i; + // Gift wrap hull. + int endpt = 0; + int i = 0; + do + { + out[i++] = hull; + endpt = 0; + for (int j = 1; j < npts; ++j) + if (hull == endpt || left(&pts[hull*3], &pts[endpt*3], &pts[j*3])) + endpt = j; + hull = endpt; + } + while (endpt != out[0]); + + return i; +} + +static int pointInPoly(int nvert, const float* verts, const float* p) +{ + int i, j, c = 0; + for (i = 0, j = nvert-1; i < nvert; j = i++) + { + const float* vi = &verts[i*3]; + const float* vj = &verts[j*3]; + if (((vi[2] > p[2]) != (vj[2] > p[2])) && + (p[0] < (vj[0]-vi[0]) * (p[2]-vi[2]) / (vj[2]-vi[2]) + vi[0]) ) + c = !c; + } + return c; +} + + +ConvexVolumeTool::ConvexVolumeTool() : + m_sample(0), + m_areaType(SAMPLE_POLYAREA_GRASS), + m_polyOffset(0.0f), + m_boxHeight(6.0f), + m_boxDescent(1.0f), + m_npts(0), + m_nhull(0) +{ +} + +void ConvexVolumeTool::init(Sample* sample) +{ + m_sample = sample; +} + +void ConvexVolumeTool::reset() +{ + m_npts = 0; + m_nhull = 0; +} + +void ConvexVolumeTool::handleMenu() +{ + imguiSlider("Shape Height", &m_boxHeight, 0.1f, 20.0f, 0.1f); + imguiSlider("Shape Descent", &m_boxDescent, 0.1f, 20.0f, 0.1f); + imguiSlider("Poly Offset", &m_polyOffset, 0.0f, 10.0f, 0.1f); + + imguiSeparator(); + + imguiLabel("Area Type"); + imguiIndent(); + if (imguiCheck("Ground", m_areaType == SAMPLE_POLYAREA_GROUND)) + m_areaType = SAMPLE_POLYAREA_GROUND; + if (imguiCheck("Water", m_areaType == SAMPLE_POLYAREA_WATER)) + m_areaType = SAMPLE_POLYAREA_WATER; + if (imguiCheck("Road", m_areaType == SAMPLE_POLYAREA_ROAD)) + m_areaType = SAMPLE_POLYAREA_ROAD; + if (imguiCheck("Door", m_areaType == SAMPLE_POLYAREA_DOOR)) + m_areaType = SAMPLE_POLYAREA_DOOR; + if (imguiCheck("Grass", m_areaType == SAMPLE_POLYAREA_GRASS)) + m_areaType = SAMPLE_POLYAREA_GRASS; + if (imguiCheck("Jump", m_areaType == SAMPLE_POLYAREA_JUMP)) + m_areaType = SAMPLE_POLYAREA_JUMP; + imguiUnindent(); + + imguiSeparator(); + + if (imguiButton("Clear Shape")) + { + m_npts = 0; + m_nhull = 0; + } +} + +void ConvexVolumeTool::handleClick(const float* /*s*/, const float* p, bool shift) +{ + if (!m_sample) return; + InputGeom* geom = m_sample->getInputGeom(); + if (!geom) return; + + if (shift) + { + // Delete + int nearestIndex = -1; + const ConvexVolume* vols = geom->getConvexVolumes(); + for (int i = 0; i < geom->getConvexVolumeCount(); ++i) + { + if (pointInPoly(vols[i].nverts, vols[i].verts, p) && + p[1] >= vols[i].hmin && p[1] <= vols[i].hmax) + { + nearestIndex = i; + } + } + // If end point close enough, delete it. + if (nearestIndex != -1) + { + geom->deleteConvexVolume(nearestIndex); + } + } + else + { + // Create + + // If clicked on that last pt, create the shape. + if (m_npts && rcVdistSqr(p, &m_pts[(m_npts-1)*3]) < rcSqr(0.2f)) + { + if (m_nhull > 2) + { + // Create shape. + float verts[MAX_PTS*3]; + for (int i = 0; i < m_nhull; ++i) + rcVcopy(&verts[i*3], &m_pts[m_hull[i]*3]); + + float minh = FLT_MAX, maxh = 0; + for (int i = 0; i < m_nhull; ++i) + minh = rcMin(minh, verts[i*3+1]); + minh -= m_boxDescent; + maxh = minh + m_boxHeight; + + if (m_polyOffset > 0.01f) + { + float offset[MAX_PTS*2*3]; + int noffset = rcOffsetPoly(verts, m_nhull, m_polyOffset, offset, MAX_PTS*2); + if (noffset > 0) + geom->addConvexVolume(offset, noffset, minh, maxh, (unsigned char)m_areaType); + } + else + { + geom->addConvexVolume(verts, m_nhull, minh, maxh, (unsigned char)m_areaType); + } + } + + m_npts = 0; + m_nhull = 0; + } + else + { + // Add new point + if (m_npts < MAX_PTS) + { + rcVcopy(&m_pts[m_npts*3], p); + m_npts++; + // Update hull. + if (m_npts > 1) + m_nhull = convexhull(m_pts, m_npts, m_hull); + else + m_nhull = 0; + } + } + } + +} + +void ConvexVolumeTool::handleToggle() +{ +} + +void ConvexVolumeTool::handleStep() +{ +} + +void ConvexVolumeTool::handleUpdate(const float /*dt*/) +{ +} + +void ConvexVolumeTool::handleRender() +{ + duDebugDraw& dd = m_sample->getDebugDraw(); + + // Find height extent of the shape. + float minh = FLT_MAX, maxh = 0; + for (int i = 0; i < m_npts; ++i) + minh = rcMin(minh, m_pts[i*3+1]); + minh -= m_boxDescent; + maxh = minh + m_boxHeight; + + dd.begin(DU_DRAW_POINTS, 4.0f); + for (int i = 0; i < m_npts; ++i) + { + unsigned int col = duRGBA(255,255,255,255); + if (i == m_npts-1) + col = duRGBA(240,32,16,255); + dd.vertex(m_pts[i*3+0],m_pts[i*3+1],m_pts[i*3+2] + 0.1f, col); + } + dd.end(); + + dd.begin(DU_DRAW_LINES, 2.0f); + for (int i = 0, j = m_nhull-1; i < m_nhull; j = i++) + { + const float* vi = &m_pts[m_hull[j]*3]; + const float* vj = &m_pts[m_hull[i]*3]; + dd.vertex(vj[0],vj[1], minh, duRGBA(255, 255, 255, 64)); + dd.vertex(vi[0], vi[1], minh, duRGBA(255, 255, 255, 64)); + dd.vertex(vj[0], vj[1], maxh, duRGBA(255, 255, 255, 64)); + dd.vertex(vi[0], vi[1], maxh, duRGBA(255, 255, 255, 64)); + dd.vertex(vj[0], vj[1], minh, duRGBA(255, 255, 255, 64)); + dd.vertex(vj[0], vj[1], maxh, duRGBA(255,255,255,64)); + } + dd.end(); +} + +void ConvexVolumeTool::handleRenderOverlay(double* /*proj*/, double* /*model*/, int* view) +{ + // Tool help + const int h = view[3]; + if (!m_npts) + { + imguiDrawText(280, h-40, IMGUI_ALIGN_LEFT, "LMB: Create new shape. SHIFT+LMB: Delete existing shape (click inside a shape).", imguiRGBA(255,255,255,192)); + } + else + { + imguiDrawText(280, h-40, IMGUI_ALIGN_LEFT, "Click LMB to add new points. Click on the red point to finish the shape.", imguiRGBA(255,255,255,192)); + imguiDrawText(280, h-60, IMGUI_ALIGN_LEFT, "The shape will be convex hull of all added points.", imguiRGBA(255,255,255,192)); + } + +} diff --git a/r5dev/naveditor/CrowdTool.cpp b/r5dev/naveditor/CrowdTool.cpp new file mode 100644 index 00000000..f6b22a4d --- /dev/null +++ b/r5dev/naveditor/CrowdTool.cpp @@ -0,0 +1,1107 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include +#include +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" +#ifdef __APPLE__ +# include +#else +# include +#endif +#include "NavEditor/Include/imgui.h" +#include "NavEditor/Include/CrowdTool.h" +#include "NavEditor/Include/InputGeom.h" +#include "NavEditor/Include/Sample.h" +#include "DetourCrowd/Include/DetourCrowd.h" +#include "DebugUtils/Include/DetourDebugDraw.h" +#include "DetourCrowd/Include/DetourObstacleAvoidance.h" +#include "Detour/Include/DetourCommon.h" +#include "Detour/Include/DetourNode.h" +#include "NavEditor/Include/SampleInterfaces.h" + +#ifdef WIN32 +# define snprintf _snprintf +#endif + +static bool isectSegAABB(const float* sp, const float* sq, + const float* amin, const float* amax, + float& tmin, float& tmax) +{ + static const float EPS = 1e-6f; + + float d[3]; + dtVsub(d, sq, sp); + tmin = 0; // set to -FLT_MAX to get first hit on line + tmax = FLT_MAX; // set to max distance ray can travel (for segment) + + // For all three slabs + for (int i = 0; i < 3; i++) + { + if (fabsf(d[i]) < EPS) + { + // Ray is parallel to slab. No hit if origin not within slab + if (sp[i] < amin[i] || sp[i] > amax[i]) + return false; + } + else + { + // Compute intersection t value of ray with near and far plane of slab + const float ood = 1.0f / d[i]; + float t1 = (amin[i] - sp[i]) * ood; + float t2 = (amax[i] - sp[i]) * ood; + // Make t1 be intersection with near plane, t2 with far plane + if (t1 > t2) dtSwap(t1, t2); + // Compute the intersection of slab intersections intervals + if (t1 > tmin) tmin = t1; + if (t2 < tmax) tmax = t2; + // Exit with no collision as soon as slab intersection becomes empty + if (tmin > tmax) return false; + } + } + + return true; +} + +static void getAgentBounds(const dtCrowdAgent* ag, float* bmin, float* bmax) +{ + const float* p = ag->npos; + const float r = ag->params.radius; + const float h = ag->params.height; + bmin[0] = p[0] - r; + bmin[1] = p[1]; + bmin[2] = p[2] - r; + bmax[0] = p[0] + r; + bmax[1] = p[1] + h; + bmax[2] = p[2] + r; +} + +CrowdToolState::CrowdToolState() : + m_sample(0), + m_nav(0), + m_crowd(0), + m_targetRef(0), + m_run(true) +{ + m_toolParams.m_expandSelectedDebugDraw = true; + m_toolParams.m_showCorners = false; + m_toolParams.m_showCollisionSegments = false; + m_toolParams.m_showPath = false; + m_toolParams.m_showVO = false; + m_toolParams.m_showOpt = false; + m_toolParams.m_showNeis = false; + m_toolParams.m_expandDebugDraw = false; + m_toolParams.m_showLabels = false; + m_toolParams.m_showGrid = false; + m_toolParams.m_showNodes = false; + m_toolParams.m_showPerfGraph = false; + m_toolParams.m_showDetailAll = false; + m_toolParams.m_expandOptions = true; + m_toolParams.m_anticipateTurns = true; + m_toolParams.m_optimizeVis = true; + m_toolParams.m_optimizeTopo = true; + m_toolParams.m_obstacleAvoidance = true; + m_toolParams.m_obstacleAvoidanceType = 3.0f; + m_toolParams.m_separation = false; + m_toolParams.m_separationWeight = 2.0f; + + memset(m_trails, 0, sizeof(m_trails)); + + m_vod = dtAllocObstacleAvoidanceDebugData(); + m_vod->init(2048); + + memset(&m_agentDebug, 0, sizeof(m_agentDebug)); + m_agentDebug.idx = -1; + m_agentDebug.vod = m_vod; +} + +CrowdToolState::~CrowdToolState() +{ + dtFreeObstacleAvoidanceDebugData(m_vod); +} + +void CrowdToolState::init(class Sample* sample) +{ + if (m_sample != sample) + { + m_sample = sample; + } + + dtNavMesh* nav = m_sample->getNavMesh(); + dtCrowd* crowd = m_sample->getCrowd(); + + if (nav && crowd && (m_nav != nav || m_crowd != crowd)) + { + m_nav = nav; + m_crowd = crowd; + + crowd->init(MAX_AGENTS, m_sample->getAgentRadius(), nav); + + // Make polygons with 'disabled' flag invalid. + crowd->getEditableFilter(0)->setExcludeFlags(SAMPLE_POLYFLAGS_DISABLED); + + // Setup local avoidance params to different qualities. + dtObstacleAvoidanceParams params; + // Use mostly default settings, copy from dtCrowd. + memcpy(¶ms, crowd->getObstacleAvoidanceParams(0), sizeof(dtObstacleAvoidanceParams)); + + // Low (11) + params.velBias = 0.5f; + params.adaptiveDivs = 5; + params.adaptiveRings = 2; + params.adaptiveDepth = 1; + crowd->setObstacleAvoidanceParams(0, ¶ms); + + // Medium (22) + params.velBias = 0.5f; + params.adaptiveDivs = 5; + params.adaptiveRings = 2; + params.adaptiveDepth = 2; + crowd->setObstacleAvoidanceParams(1, ¶ms); + + // Good (45) + params.velBias = 0.5f; + params.adaptiveDivs = 7; + params.adaptiveRings = 2; + params.adaptiveDepth = 3; + crowd->setObstacleAvoidanceParams(2, ¶ms); + + // High (66) + params.velBias = 0.5f; + params.adaptiveDivs = 7; + params.adaptiveRings = 3; + params.adaptiveDepth = 3; + + crowd->setObstacleAvoidanceParams(3, ¶ms); + } +} + +void CrowdToolState::reset() +{ +} + +void CrowdToolState::handleRender() +{ + duDebugDraw& dd = m_sample->getDebugDraw(); + const float rad = m_sample->getAgentRadius(); + + dtNavMesh* nav = m_sample->getNavMesh(); + dtCrowd* crowd = m_sample->getCrowd(); + if (!nav || !crowd) + return; + + if (m_toolParams.m_showNodes && crowd->getPathQueue()) + { + const dtNavMeshQuery* navquery = crowd->getPathQueue()->getNavQuery(); + if (navquery) + duDebugDrawNavMeshNodes(&dd, *navquery); + } + + dd.depthMask(false); + + // Draw paths + if (m_toolParams.m_showPath) + { + for (int i = 0; i < crowd->getAgentCount(); i++) + { + if (m_toolParams.m_showDetailAll == false && i != m_agentDebug.idx) + continue; + const dtCrowdAgent* ag =crowd->getAgent(i); + if (!ag->active) + continue; + const dtPolyRef* path = ag->corridor.getPath(); + const int npath = ag->corridor.getPathCount(); + for (int j = 0; j < npath; ++j) + duDebugDrawNavMeshPoly(&dd, *nav, path[j], duRGBA(255,255,255,24)); + } + } + + if (m_targetRef) + duDebugDrawCross(&dd, m_targetPos[0],m_targetPos[1]+0.1f,m_targetPos[2], rad, duRGBA(255,255,255,192), 2.0f); + + // Occupancy grid. + if (m_toolParams.m_showGrid) + { + float gridy = -FLT_MAX; + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + if (!ag->active) continue; + const float* pos = ag->corridor.getPos(); + gridy = dtMax(gridy, pos[1]); + } + gridy += 1.0f; + + dd.begin(DU_DRAW_QUADS); + const dtProximityGrid* grid = crowd->getGrid(); + const int* bounds = grid->getBounds(); + const float cs = grid->getCellSize(); + for (int y = bounds[1]; y <= bounds[3]; ++y) + { + for (int x = bounds[0]; x <= bounds[2]; ++x) + { + const int count = grid->getItemCountAt(x,y); + if (!count) continue; + unsigned int col = duRGBA(128,0,0,dtMin(count*40,255)); + dd.vertex(x*cs, gridy, y*cs, col); + dd.vertex(x*cs, gridy, y*cs+cs, col); + dd.vertex(x*cs+cs, gridy, y*cs+cs, col); + dd.vertex(x*cs+cs, gridy, y*cs, col); + } + } + dd.end(); + } + + // Trail + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + if (!ag->active) continue; + + const AgentTrail* trail = &m_trails[i]; + const float* pos = ag->npos; + + dd.begin(DU_DRAW_LINES,3.0f); + float prev[3], preva = 1; + dtVcopy(prev, pos); + for (int j = 0; j < AGENT_MAX_TRAIL-1; ++j) + { + const int idx = (trail->htrail + AGENT_MAX_TRAIL-j) % AGENT_MAX_TRAIL; + const float* v = &trail->trail[idx*3]; + float a = 1 - j/(float)AGENT_MAX_TRAIL; + dd.vertex(prev[0],prev[1]+0.1f,prev[2], duRGBA(0,0,0,(int)(128*preva))); + dd.vertex(v[0],v[1]+0.1f,v[2], duRGBA(0,0,0,(int)(128*a))); + preva = a; + dtVcopy(prev, v); + } + dd.end(); + + } + + // Corners & co + for (int i = 0; i < crowd->getAgentCount(); i++) + { + if (m_toolParams.m_showDetailAll == false && i != m_agentDebug.idx) + continue; + const dtCrowdAgent* ag =crowd->getAgent(i); + if (!ag->active) + continue; + + const float radius = ag->params.radius; + const float* pos = ag->npos; + + if (m_toolParams.m_showCorners) + { + if (ag->ncorners) + { + dd.begin(DU_DRAW_LINES, 2.0f); + for (int j = 0; j < ag->ncorners; ++j) + { + const float* va = j == 0 ? pos : &ag->cornerVerts[(j-1)*3]; + const float* vb = &ag->cornerVerts[j*3]; + dd.vertex(va[0],va[1]+radius,va[2], duRGBA(128,0,0,192)); + dd.vertex(vb[0],vb[1]+radius,vb[2], duRGBA(128,0,0,192)); + } + if (ag->ncorners && ag->cornerFlags[ag->ncorners-1] & DT_STRAIGHTPATH_OFFMESH_CONNECTION) + { + const float* v = &ag->cornerVerts[(ag->ncorners-1)*3]; + dd.vertex(v[0],v[1],v[2], duRGBA(192,0,0,192)); + dd.vertex(v[0],v[1]+radius*2,v[2], duRGBA(192,0,0,192)); + } + + dd.end(); + + + if (m_toolParams.m_anticipateTurns) + { + /* float dvel[3], pos[3]; + calcSmoothSteerDirection(ag->pos, ag->cornerVerts, ag->ncorners, dvel); + pos[0] = ag->pos[0] + dvel[0]; + pos[1] = ag->pos[1] + dvel[1]; + pos[2] = ag->pos[2] + dvel[2]; + + const float off = ag->radius+0.1f; + const float* tgt = &ag->cornerVerts[0]; + const float y = ag->pos[1]+off; + + dd.begin(DU_DRAW_LINES, 2.0f); + + dd.vertex(ag->pos[0],y,ag->pos[2], duRGBA(255,0,0,192)); + dd.vertex(pos[0],y,pos[2], duRGBA(255,0,0,192)); + + dd.vertex(pos[0],y,pos[2], duRGBA(255,0,0,192)); + dd.vertex(tgt[0],y,tgt[2], duRGBA(255,0,0,192)); + + dd.end();*/ + } + } + } + + if (m_toolParams.m_showCollisionSegments) + { + const float* center = ag->boundary.getCenter(); + duDebugDrawCross(&dd, center[0],center[1]+radius,center[2], 0.2f, duRGBA(192,0,128,255), 2.0f); + duDebugDrawCircle(&dd, center[0],center[1]+radius,center[2], ag->params.collisionQueryRange, + duRGBA(192,0,128,128), 2.0f); + + dd.begin(DU_DRAW_LINES, 3.0f); + for (int j = 0; j < ag->boundary.getSegmentCount(); ++j) + { + const float* s = ag->boundary.getSegment(j); + unsigned int col = duRGBA(192,0,128,192); + if (dtTriArea2D(pos, s, s+3) < 0.0f) + col = duDarkenCol(col); + + duAppendArrow(&dd, s[0],s[1]+0.2f,s[2], s[3],s[4]+0.2f,s[5], 0.0f, 0.3f, col); + } + dd.end(); + } + + if (m_toolParams.m_showNeis) + { + duDebugDrawCircle(&dd, pos[0],pos[1]+radius,pos[2], ag->params.collisionQueryRange, + duRGBA(0,192,128,128), 2.0f); + + dd.begin(DU_DRAW_LINES, 2.0f); + for (int j = 0; j < ag->nneis; ++j) + { + // Get 'n'th active agent. + // TODO: fix this properly. + const dtCrowdAgent* nei = crowd->getAgent(ag->neis[j].idx); + if (nei) + { + dd.vertex(pos[0],pos[1]+radius,pos[2], duRGBA(0,192,128,128)); + dd.vertex(nei->npos[0],nei->npos[1]+radius,nei->npos[2], duRGBA(0,192,128,128)); + } + } + dd.end(); + } + + if (m_toolParams.m_showOpt) + { + dd.begin(DU_DRAW_LINES, 2.0f); + dd.vertex(m_agentDebug.optStart[0],m_agentDebug.optStart[1]+0.3f,m_agentDebug.optStart[2], duRGBA(0,128,0,192)); + dd.vertex(m_agentDebug.optEnd[0],m_agentDebug.optEnd[1]+0.3f,m_agentDebug.optEnd[2], duRGBA(0,128,0,192)); + dd.end(); + } + } + + // Agent cylinders. + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + if (!ag->active) continue; + + const float radius = ag->params.radius; + const float* pos = ag->npos; + + unsigned int col = duRGBA(0,0,0,32); + if (m_agentDebug.idx == i) + col = duRGBA(255,0,0,128); + + duDebugDrawCircle(&dd, pos[0], pos[1], pos[2], radius, col, 2.0f); + } + + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + if (!ag->active) continue; + + const float height = ag->params.height; + const float radius = ag->params.radius; + const float* pos = ag->npos; + + unsigned int col = duRGBA(220,220,220,128); + if (ag->targetState == DT_CROWDAGENT_TARGET_REQUESTING || ag->targetState == DT_CROWDAGENT_TARGET_WAITING_FOR_QUEUE) + col = duLerpCol(col, duRGBA(128,0,255,128), 32); + else if (ag->targetState == DT_CROWDAGENT_TARGET_WAITING_FOR_PATH) + col = duLerpCol(col, duRGBA(128,0,255,128), 128); + else if (ag->targetState == DT_CROWDAGENT_TARGET_FAILED) + col = duRGBA(255,32,16,128); + else if (ag->targetState == DT_CROWDAGENT_TARGET_VELOCITY) + col = duLerpCol(col, duRGBA(64,255,0,128), 128); + + duDebugDrawCylinder(&dd, pos[0]-radius, pos[1]+radius*0.1f, pos[2]-radius, + pos[0]+radius, pos[1]+height, pos[2]+radius, col); + } + + + if (m_toolParams.m_showVO) + { + for (int i = 0; i < crowd->getAgentCount(); i++) + { + if (m_toolParams.m_showDetailAll == false && i != m_agentDebug.idx) + continue; + const dtCrowdAgent* ag =crowd->getAgent(i); + if (!ag->active) + continue; + + // Draw detail about agent sela + const dtObstacleAvoidanceDebugData* vod = m_agentDebug.vod; + + const float dx = ag->npos[0]; + const float dy = ag->npos[1]+ag->params.height; + const float dz = ag->npos[2]; + + duDebugDrawCircle(&dd, dx,dy,dz, ag->params.maxSpeed, duRGBA(255,255,255,64), 2.0f); + + dd.begin(DU_DRAW_QUADS); + for (int j = 0; j < vod->getSampleCount(); ++j) + { + const float* p = vod->getSampleVelocity(j); + const float sr = vod->getSampleSize(j); + const float pen = vod->getSamplePenalty(j); + const float pen2 = vod->getSamplePreferredSidePenalty(j); + unsigned int col = duLerpCol(duRGBA(255,255,255,220), duRGBA(128,96,0,220), (int)(pen*255)); + col = duLerpCol(col, duRGBA(128,0,0,220), (int)(pen2*128)); + dd.vertex(dx+p[0]-sr, dy, dz+p[2]-sr, col); + dd.vertex(dx+p[0]-sr, dy, dz+p[2]+sr, col); + dd.vertex(dx+p[0]+sr, dy, dz+p[2]+sr, col); + dd.vertex(dx+p[0]+sr, dy, dz+p[2]-sr, col); + } + dd.end(); + } + } + + // Velocity stuff. + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + if (!ag->active) continue; + + const float radius = ag->params.radius; + const float height = ag->params.height; + const float* pos = ag->npos; + const float* vel = ag->vel; + const float* dvel = ag->dvel; + + unsigned int col = duRGBA(220,220,220,192); + if (ag->targetState == DT_CROWDAGENT_TARGET_REQUESTING || ag->targetState == DT_CROWDAGENT_TARGET_WAITING_FOR_QUEUE) + col = duLerpCol(col, duRGBA(128,0,255,192), 32); + else if (ag->targetState == DT_CROWDAGENT_TARGET_WAITING_FOR_PATH) + col = duLerpCol(col, duRGBA(128,0,255,192), 128); + else if (ag->targetState == DT_CROWDAGENT_TARGET_FAILED) + col = duRGBA(255,32,16,192); + else if (ag->targetState == DT_CROWDAGENT_TARGET_VELOCITY) + col = duLerpCol(col, duRGBA(64,255,0,192), 128); + + duDebugDrawCircle(&dd, pos[0], pos[1]+height, pos[2], radius, col, 2.0f); + + duDebugDrawArrow(&dd, pos[0],pos[1]+height,pos[2], + pos[0]+dvel[0],pos[1]+height+dvel[1],pos[2]+dvel[2], + 0.0f, 0.4f, duRGBA(0,192,255,192), (m_agentDebug.idx == i) ? 2.0f : 1.0f); + + duDebugDrawArrow(&dd, pos[0],pos[1]+height,pos[2], + pos[0]+vel[0],pos[1]+height+vel[1],pos[2]+vel[2], + 0.0f, 0.4f, duRGBA(0,0,0,160), 2.0f); + } + + dd.depthMask(true); +} + + +void CrowdToolState::handleRenderOverlay(double* proj, double* model, int* view) +{ + GLdouble x, y, z; + + // Draw start and end point labels + if (m_targetRef && gluProject((GLdouble)m_targetPos[0], (GLdouble)m_targetPos[1], (GLdouble)m_targetPos[2], + model, proj, view, &x, &y, &z)) + { + imguiDrawText((int)x, (int)(y+25), IMGUI_ALIGN_CENTER, "TARGET", imguiRGBA(0,0,0,220)); + } + + char label[32]; + + if (m_toolParams.m_showNodes) + { + dtCrowd* crowd = m_sample->getCrowd(); + if (crowd && crowd->getPathQueue()) + { + const dtNavMeshQuery* navquery = crowd->getPathQueue()->getNavQuery(); + const dtNodePool* pool = navquery->getNodePool(); + if (pool) + { + const float off = 0.5f; + for (int i = 0; i < pool->getHashSize(); ++i) + { + for (dtNodeIndex j = pool->getFirst(i); j != DT_NULL_IDX; j = pool->getNext(j)) + { + const dtNode* node = pool->getNodeAtIdx(j+1); + if (!node) continue; + + if (gluProject((GLdouble)node->pos[0],(GLdouble)node->pos[1]+off,(GLdouble)node->pos[2], + model, proj, view, &x, &y, &z)) + { + const float heuristic = node->total;// - node->cost; + snprintf(label, 32, "%.2f", heuristic); + imguiDrawText((int)x, (int)y+15, IMGUI_ALIGN_CENTER, label, imguiRGBA(0,0,0,220)); + } + } + } + } + } + } + + if (m_toolParams.m_showLabels) + { + dtCrowd* crowd = m_sample->getCrowd(); + if (crowd) + { + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + if (!ag->active) continue; + const float* pos = ag->npos; + const float h = ag->params.height; + if (gluProject((GLdouble)pos[0], (GLdouble)pos[1]+h, (GLdouble)pos[2], + model, proj, view, &x, &y, &z)) + { + snprintf(label, 32, "%d", i); + imguiDrawText((int)x, (int)y+15, IMGUI_ALIGN_CENTER, label, imguiRGBA(0,0,0,220)); + } + } + } + } + if (m_agentDebug.idx != -1) + { + dtCrowd* crowd = m_sample->getCrowd(); + if (crowd) + { + for (int i = 0; i < crowd->getAgentCount(); i++) + { + if (m_toolParams.m_showDetailAll == false && i != m_agentDebug.idx) + continue; + const dtCrowdAgent* ag =crowd->getAgent(i); + if (!ag->active) + continue; + const float radius = ag->params.radius; + if (m_toolParams.m_showNeis) + { + for (int j = 0; j < ag->nneis; ++j) + { + const dtCrowdAgent* nei = crowd->getAgent(ag->neis[j].idx); + if (!nei->active) continue; + + if (gluProject((GLdouble)nei->npos[0], (GLdouble)nei->npos[1]+radius, (GLdouble)nei->npos[2], + model, proj, view, &x, &y, &z)) + { + snprintf(label, 32, "%.3f", ag->neis[j].dist); + imguiDrawText((int)x, (int)y+15, IMGUI_ALIGN_CENTER, label, imguiRGBA(255,255,255,220)); + } + } + } + } + } + } + + + if (m_toolParams.m_showPerfGraph) + { + GraphParams gp; + gp.setRect(300, 10, 500, 200, 8); + gp.setValueRange(0.0f, 2.0f, 4, "ms"); + + drawGraphBackground(&gp); + drawGraph(&gp, &m_crowdTotalTime, 1, "Total", duRGBA(255,128,0,255)); + + gp.setRect(300, 10, 500, 50, 8); + gp.setValueRange(0.0f, 2000.0f, 1, ""); + drawGraph(&gp, &m_crowdSampleCount, 0, "Sample Count", duRGBA(96,96,96,128)); + } + +} + +void CrowdToolState::handleUpdate(const float dt) +{ + if (m_run) + updateTick(dt); +} + +void CrowdToolState::addAgent(const float* p) +{ + if (!m_sample) return; + dtCrowd* crowd = m_sample->getCrowd(); + + dtCrowdAgentParams ap; + memset(&ap, 0, sizeof(ap)); + ap.radius = m_sample->getAgentRadius(); + ap.height = m_sample->getAgentHeight(); + ap.maxAcceleration = 8.0f; + ap.maxSpeed = 3.5f; + ap.collisionQueryRange = ap.radius * 12.0f; + ap.pathOptimizationRange = ap.radius * 30.0f; + ap.updateFlags = 0; + if (m_toolParams.m_anticipateTurns) + ap.updateFlags |= DT_CROWD_ANTICIPATE_TURNS; + if (m_toolParams.m_optimizeVis) + ap.updateFlags |= DT_CROWD_OPTIMIZE_VIS; + if (m_toolParams.m_optimizeTopo) + ap.updateFlags |= DT_CROWD_OPTIMIZE_TOPO; + if (m_toolParams.m_obstacleAvoidance) + ap.updateFlags |= DT_CROWD_OBSTACLE_AVOIDANCE; + if (m_toolParams.m_separation) + ap.updateFlags |= DT_CROWD_SEPARATION; + ap.obstacleAvoidanceType = (unsigned char)m_toolParams.m_obstacleAvoidanceType; + ap.separationWeight = m_toolParams.m_separationWeight; + + int idx = crowd->addAgent(p, &ap); + if (idx != -1) + { + if (m_targetRef) + crowd->requestMoveTarget(idx, m_targetRef, m_targetPos); + + // Init trail + AgentTrail* trail = &m_trails[idx]; + for (int i = 0; i < AGENT_MAX_TRAIL; ++i) + dtVcopy(&trail->trail[i*3], p); + trail->htrail = 0; + } +} + +void CrowdToolState::removeAgent(const int idx) +{ + if (!m_sample) return; + dtCrowd* crowd = m_sample->getCrowd(); + + crowd->removeAgent(idx); + + if (idx == m_agentDebug.idx) + m_agentDebug.idx = -1; +} + +void CrowdToolState::hilightAgent(const int idx) +{ + m_agentDebug.idx = idx; +} + +static void calcVel(float* vel, const float* pos, const float* tgt, const float speed) +{ + dtVsub(vel, tgt, pos); + vel[1] = 0.0; + dtVnormalize(vel); + dtVscale(vel, vel, speed); +} + +void CrowdToolState::setMoveTarget(const float* p, bool adjust) +{ + if (!m_sample) return; + + // Find nearest point on navmesh and set move request to that location. + dtNavMeshQuery* navquery = m_sample->getNavMeshQuery(); + dtCrowd* crowd = m_sample->getCrowd(); + const dtQueryFilter* filter = crowd->getFilter(0); + const float* halfExtents = crowd->getQueryExtents(); + + if (adjust) + { + float vel[3]; + // Request velocity + if (m_agentDebug.idx != -1) + { + const dtCrowdAgent* ag = crowd->getAgent(m_agentDebug.idx); + if (ag && ag->active) + { + calcVel(vel, ag->npos, p, ag->params.maxSpeed); + crowd->requestMoveVelocity(m_agentDebug.idx, vel); + } + } + else + { + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + if (!ag->active) continue; + calcVel(vel, ag->npos, p, ag->params.maxSpeed); + crowd->requestMoveVelocity(i, vel); + } + } + } + else + { + navquery->findNearestPoly(p, halfExtents, filter, &m_targetRef, m_targetPos); + + if (m_agentDebug.idx != -1) + { + const dtCrowdAgent* ag = crowd->getAgent(m_agentDebug.idx); + if (ag && ag->active) + crowd->requestMoveTarget(m_agentDebug.idx, m_targetRef, m_targetPos); + } + else + { + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + if (!ag->active) continue; + crowd->requestMoveTarget(i, m_targetRef, m_targetPos); + } + } + } +} + +int CrowdToolState::hitTestAgents(const float* s, const float* p) +{ + if (!m_sample) return -1; + dtCrowd* crowd = m_sample->getCrowd(); + + int isel = -1; + float tsel = FLT_MAX; + + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + if (!ag->active) continue; + float bmin[3], bmax[3]; + getAgentBounds(ag, bmin, bmax); + float tmin, tmax; + if (isectSegAABB(s, p, bmin,bmax, tmin, tmax)) + { + if (tmin > 0 && tmin < tsel) + { + isel = i; + tsel = tmin; + } + } + } + + return isel; +} + +void CrowdToolState::updateAgentParams() +{ + if (!m_sample) return; + dtCrowd* crowd = m_sample->getCrowd(); + if (!crowd) return; + + unsigned char updateFlags = 0; + unsigned char obstacleAvoidanceType = 0; + + if (m_toolParams.m_anticipateTurns) + updateFlags |= DT_CROWD_ANTICIPATE_TURNS; + if (m_toolParams.m_optimizeVis) + updateFlags |= DT_CROWD_OPTIMIZE_VIS; + if (m_toolParams.m_optimizeTopo) + updateFlags |= DT_CROWD_OPTIMIZE_TOPO; + if (m_toolParams.m_obstacleAvoidance) + updateFlags |= DT_CROWD_OBSTACLE_AVOIDANCE; + if (m_toolParams.m_obstacleAvoidance) + updateFlags |= DT_CROWD_OBSTACLE_AVOIDANCE; + if (m_toolParams.m_separation) + updateFlags |= DT_CROWD_SEPARATION; + + obstacleAvoidanceType = (unsigned char)m_toolParams.m_obstacleAvoidanceType; + + dtCrowdAgentParams params; + + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + if (!ag->active) continue; + memcpy(¶ms, &ag->params, sizeof(dtCrowdAgentParams)); + params.updateFlags = updateFlags; + params.obstacleAvoidanceType = obstacleAvoidanceType; + params.separationWeight = m_toolParams.m_separationWeight; + crowd->updateAgentParameters(i, ¶ms); + } +} + +void CrowdToolState::updateTick(const float dt) +{ + if (!m_sample) return; + dtNavMesh* nav = m_sample->getNavMesh(); + dtCrowd* crowd = m_sample->getCrowd(); + if (!nav || !crowd) return; + + TimeVal startTime = getPerfTime(); + + crowd->update(dt, &m_agentDebug); + + TimeVal endTime = getPerfTime(); + + // Update agent trails + for (int i = 0; i < crowd->getAgentCount(); ++i) + { + const dtCrowdAgent* ag = crowd->getAgent(i); + AgentTrail* trail = &m_trails[i]; + if (!ag->active) + continue; + // Update agent movement trail. + trail->htrail = (trail->htrail + 1) % AGENT_MAX_TRAIL; + dtVcopy(&trail->trail[trail->htrail*3], ag->npos); + } + + m_agentDebug.vod->normalizeSamples(); + + m_crowdSampleCount.addSample((float)crowd->getVelocitySampleCount()); + m_crowdTotalTime.addSample(getPerfTimeUsec(endTime - startTime) / 1000.0f); +} + + + + +CrowdTool::CrowdTool() : + m_sample(0), + m_state(0), + m_mode(TOOLMODE_CREATE) +{ +} + +void CrowdTool::init(Sample* sample) +{ + if (m_sample != sample) + { + m_sample = sample; + } + + if (!sample) + return; + + m_state = (CrowdToolState*)sample->getToolState(type()); + if (!m_state) + { + m_state = new CrowdToolState(); + sample->setToolState(type(), m_state); + } + m_state->init(sample); +} + +void CrowdTool::reset() +{ +} + +void CrowdTool::handleMenu() +{ + if (!m_state) + return; + CrowdToolParams* params = m_state->getToolParams(); + + if (imguiCheck("Create Agents", m_mode == TOOLMODE_CREATE)) + m_mode = TOOLMODE_CREATE; + if (imguiCheck("Move Target", m_mode == TOOLMODE_MOVE_TARGET)) + m_mode = TOOLMODE_MOVE_TARGET; + if (imguiCheck("Select Agent", m_mode == TOOLMODE_SELECT)) + m_mode = TOOLMODE_SELECT; + if (imguiCheck("Toggle Polys", m_mode == TOOLMODE_TOGGLE_POLYS)) + m_mode = TOOLMODE_TOGGLE_POLYS; + + imguiSeparatorLine(); + + if (imguiCollapse("Options", 0, params->m_expandOptions)) + params->m_expandOptions = !params->m_expandOptions; + + if (params->m_expandOptions) + { + imguiIndent(); + if (imguiCheck("Optimize Visibility", params->m_optimizeVis)) + { + params->m_optimizeVis = !params->m_optimizeVis; + m_state->updateAgentParams(); + } + if (imguiCheck("Optimize Topology", params->m_optimizeTopo)) + { + params->m_optimizeTopo = !params->m_optimizeTopo; + m_state->updateAgentParams(); + } + if (imguiCheck("Anticipate Turns", params->m_anticipateTurns)) + { + params->m_anticipateTurns = !params->m_anticipateTurns; + m_state->updateAgentParams(); + } + if (imguiCheck("Obstacle Avoidance", params->m_obstacleAvoidance)) + { + params->m_obstacleAvoidance = !params->m_obstacleAvoidance; + m_state->updateAgentParams(); + } + if (imguiSlider("Avoidance Quality", ¶ms->m_obstacleAvoidanceType, 0.0f, 3.0f, 1.0f)) + { + m_state->updateAgentParams(); + } + if (imguiCheck("Separation", params->m_separation)) + { + params->m_separation = !params->m_separation; + m_state->updateAgentParams(); + } + if (imguiSlider("Separation Weight", ¶ms->m_separationWeight, 0.0f, 20.0f, 0.01f)) + { + m_state->updateAgentParams(); + } + + imguiUnindent(); + } + + if (imguiCollapse("Selected Debug Draw", 0, params->m_expandSelectedDebugDraw)) + params->m_expandSelectedDebugDraw = !params->m_expandSelectedDebugDraw; + + if (params->m_expandSelectedDebugDraw) + { + imguiIndent(); + if (imguiCheck("Show Corners", params->m_showCorners)) + params->m_showCorners = !params->m_showCorners; + if (imguiCheck("Show Collision Segs", params->m_showCollisionSegments)) + params->m_showCollisionSegments = !params->m_showCollisionSegments; + if (imguiCheck("Show Path", params->m_showPath)) + params->m_showPath = !params->m_showPath; + if (imguiCheck("Show VO", params->m_showVO)) + params->m_showVO = !params->m_showVO; + if (imguiCheck("Show Path Optimization", params->m_showOpt)) + params->m_showOpt = !params->m_showOpt; + if (imguiCheck("Show Neighbours", params->m_showNeis)) + params->m_showNeis = !params->m_showNeis; + imguiUnindent(); + } + + if (imguiCollapse("Debug Draw", 0, params->m_expandDebugDraw)) + params->m_expandDebugDraw = !params->m_expandDebugDraw; + + if (params->m_expandDebugDraw) + { + imguiIndent(); + if (imguiCheck("Show Labels", params->m_showLabels)) + params->m_showLabels = !params->m_showLabels; + if (imguiCheck("Show Prox Grid", params->m_showGrid)) + params->m_showGrid = !params->m_showGrid; + if (imguiCheck("Show Nodes", params->m_showNodes)) + params->m_showNodes = !params->m_showNodes; + if (imguiCheck("Show Perf Graph", params->m_showPerfGraph)) + params->m_showPerfGraph = !params->m_showPerfGraph; + if (imguiCheck("Show Detail All", params->m_showDetailAll)) + params->m_showDetailAll = !params->m_showDetailAll; + imguiUnindent(); + } +} + +void CrowdTool::handleClick(const float* s, const float* p, bool shift) +{ + if (!m_sample) return; + if (!m_state) return; + InputGeom* geom = m_sample->getInputGeom(); + if (!geom) return; + dtCrowd* crowd = m_sample->getCrowd(); + if (!crowd) return; + + if (m_mode == TOOLMODE_CREATE) + { + if (shift) + { + // Delete + int ahit = m_state->hitTestAgents(s,p); + if (ahit != -1) + m_state->removeAgent(ahit); + } + else + { + // Add + m_state->addAgent(p); + } + } + else if (m_mode == TOOLMODE_MOVE_TARGET) + { + m_state->setMoveTarget(p, shift); + } + else if (m_mode == TOOLMODE_SELECT) + { + // Highlight + int ahit = m_state->hitTestAgents(s,p); + m_state->hilightAgent(ahit); + } + else if (m_mode == TOOLMODE_TOGGLE_POLYS) + { + dtNavMesh* nav = m_sample->getNavMesh(); + dtNavMeshQuery* navquery = m_sample->getNavMeshQuery(); + if (nav && navquery) + { + dtQueryFilter filter; + const float* halfExtents = crowd->getQueryExtents(); + float tgt[3]; + dtPolyRef ref; + navquery->findNearestPoly(p, halfExtents, &filter, &ref, tgt); + if (ref) + { + unsigned short flags = 0; + if (dtStatusSucceed(nav->getPolyFlags(ref, &flags))) + { + flags ^= SAMPLE_POLYFLAGS_DISABLED; + nav->setPolyFlags(ref, flags); + } + } + } + } + +} + +void CrowdTool::handleStep() +{ + if (!m_state) return; + + const float dt = 1.0f/20.0f; + m_state->updateTick(dt); + + m_state->setRunning(false); +} + +void CrowdTool::handleToggle() +{ + if (!m_state) return; + m_state->setRunning(!m_state->isRunning()); +} + +void CrowdTool::handleUpdate(const float dt) +{ + rcIgnoreUnused(dt); +} + +void CrowdTool::handleRender() +{ +} + +void CrowdTool::handleRenderOverlay(double* proj, double* model, int* view) +{ + rcIgnoreUnused(model); + rcIgnoreUnused(proj); + + // Tool help + const int h = view[3]; + int ty = h-40; + + if (m_mode == TOOLMODE_CREATE) + { + imguiDrawText(280, ty, IMGUI_ALIGN_LEFT, "LMB: add agent. Shift+LMB: remove agent.", imguiRGBA(255,255,255,192)); + } + else if (m_mode == TOOLMODE_MOVE_TARGET) + { + imguiDrawText(280, ty, IMGUI_ALIGN_LEFT, "LMB: set move target. Shift+LMB: adjust set velocity.", imguiRGBA(255,255,255,192)); + ty -= 20; + imguiDrawText(280, ty, IMGUI_ALIGN_LEFT, "Setting velocity will move the agents without pathfinder.", imguiRGBA(255,255,255,192)); + } + else if (m_mode == TOOLMODE_SELECT) + { + imguiDrawText(280, ty, IMGUI_ALIGN_LEFT, "LMB: select agent.", imguiRGBA(255,255,255,192)); + } + ty -= 20; + imguiDrawText(280, ty, IMGUI_ALIGN_LEFT, "SPACE: Run/Pause simulation. 1: Step simulation.", imguiRGBA(255,255,255,192)); + ty -= 20; + + if (m_state && m_state->isRunning()) + imguiDrawText(280, ty, IMGUI_ALIGN_LEFT, "- RUNNING -", imguiRGBA(255,32,16,255)); + else + imguiDrawText(280, ty, IMGUI_ALIGN_LEFT, "- PAUSED -", imguiRGBA(255,255,255,128)); +} diff --git a/r5dev/naveditor/Filelist.cpp b/r5dev/naveditor/Filelist.cpp new file mode 100644 index 00000000..170600a1 --- /dev/null +++ b/r5dev/naveditor/Filelist.cpp @@ -0,0 +1,78 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#include "NavEditor/Include/Filelist.h" + +#include +#ifdef WIN32 +# include +#else +# include +# include +#endif + +using std::vector; +using std::string; + +void scanDirectoryAppend(const string& path, const string& ext, vector& filelist) +{ +#ifdef WIN32 + string pathWithExt = path + "/*" + ext; + + _finddata_t dir; + intptr_t fh = _findfirst(pathWithExt.c_str(), &dir); + if (fh == -1L) + { + return; + } + + do + { + filelist.push_back(dir.name); + } + while (_findnext(fh, &dir) == 0); + _findclose(fh); +#else + dirent* current = 0; + DIR* dp = opendir(path.c_str()); + if (!dp) + { + return; + } + + int extLen = strlen(ext.c_str()); + while ((current = readdir(dp)) != 0) + { + int len = strlen(current->d_name); + if (len > extLen && strncmp(current->d_name + len - extLen, ext.c_str(), extLen) == 0) + { + filelist.push_back(current->d_name); + } + } + closedir(dp); +#endif + + // Sort the list of files alphabetically. + std::sort(filelist.begin(), filelist.end()); +} + +void scanDirectory(const string& path, const string& ext, vector& filelist) +{ + filelist.clear(); + scanDirectoryAppend(path, ext, filelist); +} diff --git a/r5dev/naveditor/InputGeom.cpp b/r5dev/naveditor/InputGeom.cpp new file mode 100644 index 00000000..b7bbaa3c --- /dev/null +++ b/r5dev/naveditor/InputGeom.cpp @@ -0,0 +1,661 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include +#include +#include +#include "Recast/Include/Recast.h" +#include "NavEditor/Include/InputGeom.h" +#include "NavEditor/Include/ChunkyTriMesh.h" +#include "NavEditor/Include/MeshLoaderObj.h" +#include "NavEditor/Include/MeshLoaderPly.h" +#include "DebugUtils/Include/DebugDraw.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "Detour/Include/DetourNavMesh.h" +#include "NavEditor/Include/Sample.h" + +static bool intersectSegmentTriangle(const float* sp, const float* sq, + const float* a, const float* b, const float* c, + float &t) +{ + float v, w; + float ab[3], ac[3], qp[3], ap[3], norm[3], e[3]; + rcVsub(ab, b, a); + rcVsub(ac, c, a); + rcVsub(qp, sp, sq); + + // Compute triangle normal. Can be precalculated or cached if + // intersecting multiple segments against the same triangle + rcVcross(norm, ab, ac); + + // Compute denominator d. If d <= 0, segment is parallel to or points + // away from triangle, so exit early + float d = rcVdot(qp, norm); + if (d <= 0.0f) return false; + + // Compute intersection t value of pq with plane of triangle. A ray + // intersects iff 0 <= t. Segment intersects iff 0 <= t <= 1. Delay + // dividing by d until intersection has been found to pierce triangle + rcVsub(ap, sp, a); + t = rcVdot(ap, norm); + if (t < 0.0f) return false; + if (t > d) return false; // For segment; exclude this code line for a ray test + + // Compute barycentric coordinate components and test if within bounds + rcVcross(e, qp, ap); + v = rcVdot(ac, e); + if (v < 0.0f || v > d) return false; + w = -rcVdot(ab, e); + if (w < 0.0f || v + w > d) return false; + + // Segment/ray intersects triangle. Perform delayed division + t /= d; + + return true; +} + +static char* parseRow(char* buf, char* bufEnd, char* row, int len) +{ + bool start = true; + bool done = false; + int n = 0; + while (!done && buf < bufEnd) + { + char c = *buf; + buf++; + // multirow + switch (c) + { + case '\n': + if (start) break; + done = true; + break; + case '\r': + break; + case '\t': + case ' ': + if (start) break; + // else falls through + default: + start = false; + row[n++] = c; + if (n >= len-1) + done = true; + break; + } + } + row[n] = '\0'; + return buf; +} + + + +InputGeom::InputGeom() : + m_chunkyMesh(0), + m_mesh(0), + m_hasBuildSettings(false), + m_offMeshConCount(0), + m_volumeCount(0) +{ +} + +InputGeom::~InputGeom() +{ + delete m_chunkyMesh; + delete m_mesh; +} + +bool InputGeom::loadMesh(rcContext* ctx, const std::string& filepath,bool is_tf2) +{ + if (m_mesh) + { + delete m_chunkyMesh; + m_chunkyMesh = 0; + delete m_mesh; + m_mesh = 0; + } + m_offMeshConCount = 0; + m_volumeCount = 0; + + m_mesh = new rcMeshLoaderObj; + //m_mesh->m_flip_tris = is_tf2; + m_mesh->m_tf2_import_flip = is_tf2; + if (!m_mesh) + { + ctx->log(RC_LOG_ERROR, "loadMesh: Out of memory 'm_mesh'."); + return false; + } + if (!m_mesh->load(filepath)) + { + ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not load '%s'", filepath.c_str()); + return false; + } + + rcCalcBounds(m_mesh->getVerts(), m_mesh->getVertCount(), m_meshBMin, m_meshBMax); + + m_chunkyMesh = new rcChunkyTriMesh; + if (!m_chunkyMesh) + { + ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Out of memory 'm_chunkyMesh'."); + return false; + } + if (!rcCreateChunkyTriMesh(m_mesh->getVerts(), m_mesh->getTris(), m_mesh->getTriCount(), 256, m_chunkyMesh)) + { + ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Failed to build chunky mesh."); + return false; + } + + return true; +} +bool InputGeom::loadPlyMesh(rcContext* ctx, const std::string& filepath, bool is_tf2) +{ + if (m_mesh) + { + delete m_chunkyMesh; + m_chunkyMesh = 0; + delete m_mesh; + m_mesh = 0; + } + m_offMeshConCount = 0; + m_volumeCount = 0; + + m_mesh = new rcMeshLoaderPly; + //m_mesh->m_flip_tris = is_tf2; + m_mesh->m_tf2_import_flip = is_tf2; + if (!m_mesh) + { + ctx->log(RC_LOG_ERROR, "loadMesh: Out of memory 'm_mesh'."); + return false; + } + if (!m_mesh->load(filepath)) + { + ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not load '%s'", filepath.c_str()); + return false; + } + + rcCalcBounds(m_mesh->getVerts(), m_mesh->getVertCount(), m_meshBMin, m_meshBMax); + + m_chunkyMesh = new rcChunkyTriMesh; + if (!m_chunkyMesh) + { + ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Out of memory 'm_chunkyMesh'."); + return false; + } + if (!rcCreateChunkyTriMesh(m_mesh->getVerts(), m_mesh->getTris(), m_mesh->getTriCount(), 256, m_chunkyMesh)) + { + ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Failed to build chunky mesh."); + return false; + } + + return true; +} +bool InputGeom::loadGeomSet(rcContext* ctx, const std::string& filepath,bool is_tf2) +{ + //NB(warmist): tf2 not implemented here + char* buf = 0; + FILE* fp = fopen(filepath.c_str(), "rb"); + if (!fp) + { + return false; + } + if (fseek(fp, 0, SEEK_END) != 0) + { + fclose(fp); + return false; + } + + long bufSize = ftell(fp); + if (bufSize < 0) + { + fclose(fp); + return false; + } + if (fseek(fp, 0, SEEK_SET) != 0) + { + fclose(fp); + return false; + } + buf = new char[bufSize]; + if (!buf) + { + fclose(fp); + return false; + } + size_t readLen = fread(buf, bufSize, 1, fp); + fclose(fp); + if (readLen != 1) + { + delete[] buf; + return false; + } + + m_offMeshConCount = 0; + m_volumeCount = 0; + delete m_mesh; + m_mesh = 0; + + char* src = buf; + char* srcEnd = buf + bufSize; + char row[512]; + while (src < srcEnd) + { + // Parse one row + row[0] = '\0'; + src = parseRow(src, srcEnd, row, sizeof(row)/sizeof(char)); + if (row[0] == 'f') + { + // File name. + const char* name = row+1; + // Skip white spaces + while (*name && isspace(*name)) + name++; + if (*name) + { + if (!loadMesh(ctx, name, is_tf2)) + { + delete [] buf; + return false; + } + } + } + else if (row[0] == 'c') + { + // Off-mesh connection + if (m_offMeshConCount < MAX_OFFMESH_CONNECTIONS) + { + float* v = &m_offMeshConVerts[m_offMeshConCount*3*2]; + int bidir, area = 0, flags = 0; + float rad; + sscanf(row+1, "%f %f %f %f %f %f %f %d %d %d", + &v[0], &v[1], &v[2], &v[3], &v[4], &v[5], &rad, &bidir, &area, &flags); + m_offMeshConRads[m_offMeshConCount] = rad; + m_offMeshConDirs[m_offMeshConCount] = (unsigned char)bidir; + m_offMeshConAreas[m_offMeshConCount] = (unsigned char)area; + m_offMeshConFlags[m_offMeshConCount] = (unsigned short)flags; + m_offMeshConCount++; + } + } + else if (row[0] == 'v') + { + // Convex volumes + if (m_volumeCount < MAX_VOLUMES) + { + ConvexVolume* vol = &m_volumes[m_volumeCount++]; + sscanf(row+1, "%d %d %f %f", &vol->nverts, &vol->area, &vol->hmin, &vol->hmax); + for (int i = 0; i < vol->nverts; ++i) + { + row[0] = '\0'; + src = parseRow(src, srcEnd, row, sizeof(row)/sizeof(char)); + sscanf(row, "%f %f %f", &vol->verts[i*3+0], &vol->verts[i*3+1], &vol->verts[i*3+2]); + } + } + } + else if (row[0] == 's') + { + // Settings + m_hasBuildSettings = true; + sscanf(row + 1, "%f %f %f %f %f %f %f %f %f %f %f %f %f %d %f %f %f %f %f %f %f", + &m_buildSettings.cellSize, + &m_buildSettings.cellHeight, + &m_buildSettings.agentHeight, + &m_buildSettings.agentRadius, + &m_buildSettings.agentMaxClimb, + &m_buildSettings.agentMaxSlope, + &m_buildSettings.regionMinSize, + &m_buildSettings.regionMergeSize, + &m_buildSettings.edgeMaxLen, + &m_buildSettings.edgeMaxError, + &m_buildSettings.vertsPerPoly, + &m_buildSettings.detailSampleDist, + &m_buildSettings.detailSampleMaxError, + &m_buildSettings.partitionType, + &m_buildSettings.navMeshBMin[0], + &m_buildSettings.navMeshBMin[1], + &m_buildSettings.navMeshBMin[2], + &m_buildSettings.navMeshBMax[0], + &m_buildSettings.navMeshBMax[1], + &m_buildSettings.navMeshBMax[2], + &m_buildSettings.tileSize); + } + } + + delete [] buf; + + return true; +} + +bool InputGeom::load(rcContext* ctx, const std::string& filepath,bool is_tf2) +{ + size_t extensionPos = filepath.find_last_of('.'); + if (extensionPos == std::string::npos) + return false; + + std::string extension = filepath.substr(extensionPos); + std::transform(extension.begin(), extension.end(), extension.begin(), tolower); + + if (extension == ".gset") + return loadGeomSet(ctx, filepath, is_tf2); + if (extension == ".obj") + return loadMesh(ctx, filepath, is_tf2); + if (extension == ".ply") + return loadPlyMesh(ctx, filepath, is_tf2); + + return false; +} + +bool InputGeom::saveGeomSet(const BuildSettings* settings) +{ + if (!m_mesh) return false; + + // Change extension + std::string filepath = m_mesh->getFileName(); + size_t extPos = filepath.find_last_of('.'); + if (extPos != std::string::npos) + filepath = filepath.substr(0, extPos); + + filepath += ".gset"; + + FILE* fp = fopen(filepath.c_str(), "w"); + if (!fp) return false; + + // Store mesh filename. + fprintf(fp, "f %s\n", m_mesh->getFileName().c_str()); + + // Store settings if any + if (settings) + { + fprintf(fp, + "s %f %f %f %f %f %f %f %f %f %f %f %f %f %d %f %f %f %f %f %f %f\n", + settings->cellSize, + settings->cellHeight, + settings->agentHeight, + settings->agentRadius, + settings->agentMaxClimb, + settings->agentMaxSlope, + settings->regionMinSize, + settings->regionMergeSize, + settings->edgeMaxLen, + settings->edgeMaxError, + settings->vertsPerPoly, + settings->detailSampleDist, + settings->detailSampleMaxError, + settings->partitionType, + settings->navMeshBMin[0], + settings->navMeshBMin[1], + settings->navMeshBMin[2], + settings->navMeshBMax[0], + settings->navMeshBMax[1], + settings->navMeshBMax[2], + settings->tileSize); + } + + // Store off-mesh links. + for (int i = 0; i < m_offMeshConCount; ++i) + { + const float* v = &m_offMeshConVerts[i*3*2]; + const float rad = m_offMeshConRads[i]; + const int bidir = m_offMeshConDirs[i]; + const int area = m_offMeshConAreas[i]; + const int flags = m_offMeshConFlags[i]; + fprintf(fp, "c %f %f %f %f %f %f %f %d %d %d\n", + v[0], v[1], v[2], v[3], v[4], v[5], rad, bidir, area, flags); + } + + // Convex volumes + for (int i = 0; i < m_volumeCount; ++i) + { + ConvexVolume* vol = &m_volumes[i]; + fprintf(fp, "v %d %d %f %f\n", vol->nverts, vol->area, vol->hmin, vol->hmax); + for (int j = 0; j < vol->nverts; ++j) + fprintf(fp, "%f %f %f\n", vol->verts[j*3+0], vol->verts[j*3+1], vol->verts[j*3+2]); + } + + fclose(fp); + + return true; +} + +static bool isectSegAABB(const float* sp, const float* sq, + const float* amin, const float* amax, + float& tmin, float& tmax) +{ + static const float EPS = 1e-6f; + + float d[3]; + d[0] = sq[0] - sp[0]; + d[1] = sq[1] - sp[1]; + d[2] = sq[2] - sp[2]; + tmin = 0.0; + tmax = 1.0f; + + for (int i = 0; i < 3; i++) + { + if (fabsf(d[i]) < EPS) + { + if (sp[i] < amin[i] || sp[i] > amax[i]) + return false; + } + else + { + const float ood = 1.0f / d[i]; + float t1 = (amin[i] - sp[i]) * ood; + float t2 = (amax[i] - sp[i]) * ood; + if (t1 > t2) { float tmp = t1; t1 = t2; t2 = tmp; } + if (t1 > tmin) tmin = t1; + if (t2 < tmax) tmax = t2; + if (tmin > tmax) return false; + } + } + + return true; +} + + +bool InputGeom::raycastMesh(float* src, float* dst, float& tmin) +{ + // Prune hit ray. + float btmin, btmax; + if (!isectSegAABB(src, dst, m_meshBMin, m_meshBMax, btmin, btmax)) + return false; + float p[2], q[2]; + p[0] = src[0] + (dst[0]-src[0])*btmin; + p[1] = src[1] + (dst[1]-src[1])*btmin; + q[0] = src[0] + (dst[0]-src[0])*btmax; + q[1] = src[1] + (dst[1]-src[1])*btmax; + + int cid[4096]; //stack is crying for help + const int ncid = rcGetChunksOverlappingSegment(m_chunkyMesh, p, q, cid, 4096); + if (!ncid) + return false; + + tmin = 1.0f; + bool hit = false; + const float* verts = m_mesh->getVerts(); + + for (int i = 0; i < ncid; ++i) + { + const rcChunkyTriMeshNode& node = m_chunkyMesh->nodes[cid[i]]; + const int* tris = &m_chunkyMesh->tris[node.i*3]; + const int ntris = node.n; + + for (int j = 0; j < ntris*3; j += 3) + { + float t = 1; + if (intersectSegmentTriangle(src, dst, + &verts[tris[j]*3], + &verts[tris[j+1]*3], + &verts[tris[j+2]*3], t)) + { + if (t < tmin) + tmin = t; + hit = true; + } + } + } + + return hit; +} + +void InputGeom::addOffMeshConnection(const float* spos, const float* epos, const float rad, + unsigned char bidir, unsigned char area, unsigned short flags) +{ + if (m_offMeshConCount >= MAX_OFFMESH_CONNECTIONS) return; + float* v = &m_offMeshConVerts[m_offMeshConCount*3*2]; + m_offMeshConRads[m_offMeshConCount] = rad; + m_offMeshConDirs[m_offMeshConCount] = bidir; + m_offMeshConAreas[m_offMeshConCount] = area; + m_offMeshConFlags[m_offMeshConCount] = flags; + m_offMeshConId[m_offMeshConCount] = 1000 + m_offMeshConCount; + rcVcopy(&v[0], spos); + rcVcopy(&v[3], epos); + m_offMeshConCount++; +} + +void InputGeom::deleteOffMeshConnection(int i) +{ + m_offMeshConCount--; + float* src = &m_offMeshConVerts[m_offMeshConCount*3*2]; + float* dst = &m_offMeshConVerts[i*3*2]; + rcVcopy(&dst[0], &src[0]); + rcVcopy(&dst[3], &src[3]); + m_offMeshConRads[i] = m_offMeshConRads[m_offMeshConCount]; + m_offMeshConDirs[i] = m_offMeshConDirs[m_offMeshConCount]; + m_offMeshConAreas[i] = m_offMeshConAreas[m_offMeshConCount]; + m_offMeshConFlags[i] = m_offMeshConFlags[m_offMeshConCount]; +} + +void InputGeom::drawOffMeshConnections(duDebugDraw* dd, bool hilight) +{ + unsigned int conColor = duRGBA(192,0,128,192); + unsigned int baseColor = duRGBA(0,0,0,64); + dd->depthMask(false); + + dd->begin(DU_DRAW_LINES, 2.0f); + for (int i = 0; i < m_offMeshConCount; ++i) + { + float* v = &m_offMeshConVerts[i*3*2]; + + dd->vertex(v[0],v[1],v[2], baseColor); + dd->vertex(v[0],v[1]+0.2f,v[2], baseColor); + + dd->vertex(v[3],v[4],v[5], baseColor); + dd->vertex(v[3],v[4]+0.2f,v[5], baseColor); + + duAppendCircle(dd, v[0],v[1]+0.1f,v[2], m_offMeshConRads[i], baseColor); + duAppendCircle(dd, v[3],v[4]+0.1f,v[5], m_offMeshConRads[i], baseColor); + + if (hilight) + { + duAppendArc(dd, v[0],v[1],v[2], v[3],v[4],v[5], 0.25f, + (m_offMeshConDirs[i]&1) ? 0.6f : 0.0f, 0.6f, conColor); + } + } + dd->end(); + + dd->depthMask(true); +} + +void InputGeom::addConvexVolume(const float* verts, const int nverts, + const float minh, const float maxh, unsigned char area) +{ + if (m_volumeCount >= MAX_VOLUMES) return; + ConvexVolume* vol = &m_volumes[m_volumeCount++]; + memset(vol, 0, sizeof(ConvexVolume)); + memcpy(vol->verts, verts, sizeof(float)*3*nverts); + vol->hmin = minh; + vol->hmax = maxh; + vol->nverts = nverts; + vol->area = area; +} + +void InputGeom::deleteConvexVolume(int i) +{ + m_volumeCount--; + m_volumes[i] = m_volumes[m_volumeCount]; +} + +void InputGeom::drawConvexVolumes(struct duDebugDraw* dd, bool /*hilight*/) +{ + dd->depthMask(false); + + dd->begin(DU_DRAW_TRIS); + + for (int i = 0; i < m_volumeCount; ++i) + { + const ConvexVolume* vol = &m_volumes[i]; + unsigned int col = duTransCol(dd->areaToCol(vol->area), 32); + for (int j = 0, k = vol->nverts-1; j < vol->nverts; k = j++) + { + const float* va = &vol->verts[k*3]; + const float* vb = &vol->verts[j*3]; + + dd->vertex(vol->verts[0],vol->hmax,vol->verts[2], col); + dd->vertex(vb[0],vol->hmax,vb[2], col); + dd->vertex(va[0],vol->hmax,va[2], col); + + dd->vertex(va[0],vol->hmin,va[2], duDarkenCol(col)); + dd->vertex(va[0],vol->hmax,va[2], col); + dd->vertex(vb[0],vol->hmax,vb[2], col); + + dd->vertex(va[0],vol->hmin,va[2], duDarkenCol(col)); + dd->vertex(vb[0],vol->hmax,vb[2], col); + dd->vertex(vb[0],vol->hmin,vb[2], duDarkenCol(col)); + } + } + + dd->end(); + + dd->begin(DU_DRAW_LINES, 2.0f); + for (int i = 0; i < m_volumeCount; ++i) + { + const ConvexVolume* vol = &m_volumes[i]; + unsigned int col = duTransCol(dd->areaToCol(vol->area), 220); + for (int j = 0, k = vol->nverts-1; j < vol->nverts; k = j++) + { + const float* va = &vol->verts[k*3]; + const float* vb = &vol->verts[j*3]; + dd->vertex(va[0],vol->hmin,va[2], duDarkenCol(col)); + dd->vertex(vb[0],vol->hmin,vb[2], duDarkenCol(col)); + dd->vertex(va[0],vol->hmax,va[2], col); + dd->vertex(vb[0],vol->hmax,vb[2], col); + dd->vertex(va[0],vol->hmin,va[2], duDarkenCol(col)); + dd->vertex(va[0],vol->hmax,va[2], col); + } + } + dd->end(); + + dd->begin(DU_DRAW_POINTS, 3.0f); + for (int i = 0; i < m_volumeCount; ++i) + { + const ConvexVolume* vol = &m_volumes[i]; + unsigned int col = duDarkenCol(duTransCol(dd->areaToCol(vol->area), 220)); + for (int j = 0; j < vol->nverts; ++j) + { + dd->vertex(vol->verts[j*3+0],vol->verts[j*3+1]+0.1f,vol->verts[j*3+2], col); + dd->vertex(vol->verts[j*3+0],vol->hmin,vol->verts[j*3+2], col); + dd->vertex(vol->verts[j*3+0],vol->hmax,vol->verts[j*3+2], col); + } + } + dd->end(); + + + dd->depthMask(true); +} diff --git a/r5dev/naveditor/MeshLoaderBsp.cpp b/r5dev/naveditor/MeshLoaderBsp.cpp new file mode 100644 index 00000000..e9974ffd --- /dev/null +++ b/r5dev/naveditor/MeshLoaderBsp.cpp @@ -0,0 +1,85 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#include "NavEditor/Include/MeshLoaderBsp.h" +#include +#include +#include +#define _USE_MATH_DEFINES +#include + + +bool rcMeshLoaderBsp::load(const std::string& filename) +{ +#if 0 + //we expect lumps to be in same dir + + using namespace std; + std::string vertex_lump_path = filename + ".0003.bsp_lump"; + auto vf=fopen(vertex_lump_path.c_str(), "rb"); + + if (!vf) + return false; + + fseek(vf, 0, SEEK_END); + int fsize=ftell(vf); + fseek(vf, 0, SEEK_SET); + + m_verts.resize(fsize / sizeof(float)); + if (m_verts[i] = fread(&m_verts[i], 4, m_verts.size(), vf) != m_verts.size()) + { + fclose(vf); + return false; + } + fclose(vf); + + //TODO: triangles + + + // Calculate normals. + m_normals.resize(m_triCount*3); + for (int i = 0; i < m_triCount*3; i += 3) + { + const float* v0 = &m_verts[m_tris[i]*3]; + const float* v1 = &m_verts[m_tris[i+1]*3]; + const float* v2 = &m_verts[m_tris[i+2]*3]; + float e0[3], e1[3]; + for (int j = 0; j < 3; ++j) + { + e0[j] = v1[j] - v0[j]; + e1[j] = v2[j] - v0[j]; + } + float* n = &m_normals[i]; + n[0] = e0[1]*e1[2] - e0[2]*e1[1]; + n[1] = e0[2]*e1[0] - e0[0]*e1[2]; + n[2] = e0[0]*e1[1] - e0[1]*e1[0]; + float d = sqrtf(n[0]*n[0] + n[1]*n[1] + n[2]*n[2]); + if (d > 0) + { + d = 1.0f/d; + n[0] *= d; + n[1] *= d; + n[2] *= d; + } + } + + m_filename = filename; + return true; +#endif + return false; +} diff --git a/r5dev/naveditor/MeshLoaderObj.cpp b/r5dev/naveditor/MeshLoaderObj.cpp new file mode 100644 index 00000000..f2d027ab --- /dev/null +++ b/r5dev/naveditor/MeshLoaderObj.cpp @@ -0,0 +1,257 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#include "NavEditor/Include/MeshLoaderObj.h" +#include +#include +#include +#define _USE_MATH_DEFINES +#include + +rcMeshLoaderObj::rcMeshLoaderObj() : + m_scale(1.0f), + m_verts(0), + m_tris(0), + m_normals(0), + m_vertCount(0), + m_triCount(0) +{ +} + +rcMeshLoaderObj::~rcMeshLoaderObj() +{ + delete [] m_verts; + delete [] m_normals; + delete [] m_tris; +} + +void rcMeshLoaderObj::addVertex(float x, float y, float z, int& cap) +{ + if (m_vertCount+1 > cap) + { + cap = !cap ? 8 : cap*2; + float* nv = new float[cap*3]; + if (m_vertCount) + memcpy(nv, m_verts, m_vertCount*3*sizeof(float)); + delete [] m_verts; + m_verts = nv; + } + float* dst = &m_verts[m_vertCount*3]; + if (m_tf2_import_flip) + { + *dst++ = x * m_scale; + *dst++ = z * m_scale; + *dst++ = -y * m_scale; + } + else + { + *dst++ = x * m_scale; + *dst++ = y * m_scale; + *dst++ = z * m_scale; + } + m_vertCount++; +} + +void rcMeshLoaderObj::addTriangle(int a, int b, int c, int& cap) +{ + if (m_triCount+1 > cap) + { + cap = !cap ? 8 : cap*2; + int* nv = new int[cap*3]; + if (m_triCount) + memcpy(nv, m_tris, m_triCount*3*sizeof(int)); + delete [] m_tris; + m_tris = nv; + } + int* dst = &m_tris[m_triCount*3]; + *dst++ = a; + *dst++ = b; + *dst++ = c; + m_triCount++; +} + +static char* parseRow(char* buf, char* bufEnd, char* row, int len) +{ + bool start = true; + bool done = false; + int n = 0; + while (!done && buf < bufEnd) + { + char c = *buf; + buf++; + // multirow + switch (c) + { + case '\\': + break; + case '\n': + if (start) break; + done = true; + break; + case '\r': + break; + case '\t': + case ' ': + if (start) break; + // else falls through + default: + start = false; + row[n++] = c; + if (n >= len-1) + done = true; + break; + } + } + row[n] = '\0'; + return buf; +} + +static int parseFace(char* row, int* data, int n, int vcnt) +{ + int j = 0; + while (*row != '\0') + { + // Skip initial white space + while (*row != '\0' && (*row == ' ' || *row == '\t')) + row++; + char* s = row; + // Find vertex delimiter and terminated the string there for conversion. + while (*row != '\0' && *row != ' ' && *row != '\t') + { + if (*row == '/') *row = '\0'; + row++; + } + if (*s == '\0') + continue; + int vi = atoi(s); + data[j++] = vi < 0 ? vi+vcnt : vi-1; + if (j >= n) return j; + } + return j; +} + +bool rcMeshLoaderObj::load(const std::string& filename) +{ + char* buf = 0; + FILE* fp = fopen(filename.c_str(), "rb"); + if (!fp) + return false; + if (fseek(fp, 0, SEEK_END) != 0) + { + fclose(fp); + return false; + } + long bufSize = ftell(fp); + if (bufSize < 0) + { + fclose(fp); + return false; + } + if (fseek(fp, 0, SEEK_SET) != 0) + { + fclose(fp); + return false; + } + buf = new char[bufSize]; + if (!buf) + { + fclose(fp); + return false; + } + size_t readLen = fread(buf, bufSize, 1, fp); + fclose(fp); + + if (readLen != 1) + { + delete[] buf; + return false; + } + + char* src = buf; + char* srcEnd = buf + bufSize; + char row[512]; + int face[32]; + float x,y,z; + int nv; + int vcap = 0; + int tcap = 0; + + while (src < srcEnd) + { + // Parse one row + row[0] = '\0'; + src = parseRow(src, srcEnd, row, sizeof(row)/sizeof(char)); + // Skip comments + if (row[0] == '#') continue; + if (row[0] == 'v' && row[1] != 'n' && row[1] != 't') + { + // Vertex pos + sscanf(row+1, "%f %f %f", &x, &y, &z); + addVertex(x, y, z, vcap); + } + if (row[0] == 'f') + { + // Faces + nv = parseFace(row+1, face, 32, m_vertCount); + for (int i = 2; i < nv; ++i) + { + const int a = face[0]; + const int b = face[i-1]; + const int c = face[i]; + if (a < 0 || a >= m_vertCount || b < 0 || b >= m_vertCount || c < 0 || c >= m_vertCount) + continue; + if(m_flip_tris) + addTriangle(a, c, b, tcap); + else + addTriangle(a, b, c, tcap); + } + } + } + + delete [] buf; + + // Calculate normals. + m_normals = new float[m_triCount*3]; + for (int i = 0; i < m_triCount*3; i += 3) + { + const float* v0 = &m_verts[m_tris[i]*3]; + const float* v1 = &m_verts[m_tris[i+1]*3]; + const float* v2 = &m_verts[m_tris[i+2]*3]; + float e0[3], e1[3]; + for (int j = 0; j < 3; ++j) + { + e0[j] = v1[j] - v0[j]; + e1[j] = v2[j] - v0[j]; + } + float* n = &m_normals[i]; + n[0] = e0[1]*e1[2] - e0[2]*e1[1]; + n[1] = e0[2]*e1[0] - e0[0]*e1[2]; + n[2] = e0[0]*e1[1] - e0[1]*e1[0]; + float d = sqrtf(n[0]*n[0] + n[1]*n[1] + n[2]*n[2]); + if (d > 0) + { + d = 1.0f/d; + n[0] *= d; + n[1] *= d; + n[2] *= d; + } + } + + m_filename = filename; + return true; +} diff --git a/r5dev/naveditor/MeshLoaderPly.cpp b/r5dev/naveditor/MeshLoaderPly.cpp new file mode 100644 index 00000000..3a435387 --- /dev/null +++ b/r5dev/naveditor/MeshLoaderPly.cpp @@ -0,0 +1,155 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#include "NavEditor/Include/MeshLoaderPly.h" +#include +#include +#include +#define _USE_MATH_DEFINES +#include + +#include +#include + +bool rcMeshLoaderPly::load(const std::string& filename) +{ + using namespace std; + + ifstream input(filename,std::ios::binary); + + if (!input.is_open()) + return false; +//we expect and only support! +/* +ply +format binary_little_endian 1.0 +element vertex %d +property float x +property float y +property float z +element face %d +property list uchar int vertex_index +end_header +*/ + std::string line; + getline(input, line); + if (line != "ply") + return false; + getline(input, line); + if (line != "format binary_little_endian 1.0") + return false; + while (true) + { + input >> line; + if (line == "element") + { + input >> line; + if (line == "vertex") + { + input >> m_vertCount; + m_verts.resize(m_vertCount * 3); + } + else if (line == "face") + { + input >> m_triCount; + m_tris.resize(m_triCount * 3); + } + } + else if (line == "end_header") + { + break; + } + else + { + //skip rest of the line + getline(input, line); + } + } + + //skip newline + input.seekg(1, ios_base::cur); + + for (size_t i = 0; i < m_vertCount; i++) + { + //TODO: m_scale? + if (m_tf2_import_flip) + { + input.read((char*)&m_verts[i * 3 + 0], sizeof(float)); + input.read((char*)&m_verts[i * 3 + 2], sizeof(float)); + input.read((char*)&m_verts[i * 3 + 1], sizeof(float)); + m_verts[i * 3 + 1] *= -1; + } + else + { + input.read((char*)&m_verts[i * 3 + 0], sizeof(float)); + input.read((char*)&m_verts[i * 3 + 1], sizeof(float)); + input.read((char*)&m_verts[i * 3 + 2], sizeof(float)); + } + + } + + for (size_t i = 0; i < m_triCount; i++) + { + char count; + input.read(&count, 1); + if (count != 3) + return false; + if (m_flip_tris) + { + input.read((char*)&m_tris[i * 3 + 0], sizeof(int)); + input.read((char*)&m_tris[i * 3 + 2], sizeof(int)); + input.read((char*)&m_tris[i * 3 + 1], sizeof(int)); + } + else + { + input.read((char*)&m_tris[i * 3 + 0], sizeof(int)); + input.read((char*)&m_tris[i * 3 + 1], sizeof(int)); + input.read((char*)&m_tris[i * 3 + 2], sizeof(int)); + } + } + + // Calculate normals. + m_normals.resize(m_triCount*3); + for (int i = 0; i < m_triCount*3; i += 3) + { + const float* v0 = &m_verts[m_tris[i]*3]; + const float* v1 = &m_verts[m_tris[i+1]*3]; + const float* v2 = &m_verts[m_tris[i+2]*3]; + float e0[3], e1[3]; + for (int j = 0; j < 3; ++j) + { + e0[j] = v1[j] - v0[j]; + e1[j] = v2[j] - v0[j]; + } + float* n = &m_normals[i]; + n[0] = e0[1]*e1[2] - e0[2]*e1[1]; + n[1] = e0[2]*e1[0] - e0[0]*e1[2]; + n[2] = e0[0]*e1[1] - e0[1]*e1[0]; + float d = sqrtf(n[0]*n[0] + n[1]*n[1] + n[2]*n[2]); + if (d > 0) + { + d = 1.0f/d; + n[0] *= d; + n[1] *= d; + n[2] *= d; + } + } + + m_filename = filename; + return true; +} diff --git a/r5dev/naveditor/NavMeshPruneTool.cpp b/r5dev/naveditor/NavMeshPruneTool.cpp new file mode 100644 index 00000000..ceecaf7b --- /dev/null +++ b/r5dev/naveditor/NavMeshPruneTool.cpp @@ -0,0 +1,323 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include +#include +#include +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" +#include "NavEditor/Include/imgui.h" +#include "NavEditor/Include/NavMeshPruneTool.h" +#include "NavEditor/Include/InputGeom.h" +#include "NavEditor/Include/Sample.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Detour/Include/DetourCommon.h" +#include "Detour/Include/DetourAssert.h" +#include "DebugUtils/Include/DetourDebugDraw.h" + +#ifdef WIN32 +# define snprintf _snprintf +#endif + +class NavmeshFlags +{ + struct TileFlags + { + inline void purge() { dtFree(flags); } + unsigned char* flags; + int nflags; + dtPolyRef base; + }; + + const dtNavMesh* m_nav; + TileFlags* m_tiles; + int m_ntiles; + +public: + NavmeshFlags() : + m_nav(0), m_tiles(0), m_ntiles(0) + { + } + + ~NavmeshFlags() + { + for (int i = 0; i < m_ntiles; ++i) + m_tiles[i].purge(); + dtFree(m_tiles); + } + + bool init(const dtNavMesh* nav) + { + m_ntiles = nav->getMaxTiles(); + if (!m_ntiles) + return true; + m_tiles = (TileFlags*)dtAlloc(sizeof(TileFlags)*m_ntiles, DT_ALLOC_TEMP); + if (!m_tiles) + { + return false; + } + memset(m_tiles, 0, sizeof(TileFlags)*m_ntiles); + + // Alloc flags for each tile. + for (int i = 0; i < nav->getMaxTiles(); ++i) + { + const dtMeshTile* tile = nav->getTile(i); + if (!tile->header) continue; + TileFlags* tf = &m_tiles[i]; + tf->nflags = tile->header->polyCount; + tf->base = nav->getPolyRefBase(tile); + if (tf->nflags) + { + tf->flags = (unsigned char*)dtAlloc(tf->nflags, DT_ALLOC_TEMP); + if (!tf->flags) + return false; + memset(tf->flags, 0, tf->nflags); + } + } + + m_nav = nav; + + return false; + } + + inline void clearAllFlags() + { + for (int i = 0; i < m_ntiles; ++i) + { + TileFlags* tf = &m_tiles[i]; + if (tf->nflags) + memset(tf->flags, 0, tf->nflags); + } + } + + inline unsigned char getFlags(dtPolyRef ref) + { + dtAssert(m_nav); + dtAssert(m_ntiles); + // Assume the ref is valid, no bounds checks. + unsigned int salt, it, ip; + m_nav->decodePolyId(ref, salt, it, ip); + return m_tiles[it].flags[ip]; + } + + inline void setFlags(dtPolyRef ref, unsigned char flags) + { + dtAssert(m_nav); + dtAssert(m_ntiles); + // Assume the ref is valid, no bounds checks. + unsigned int salt, it, ip; + m_nav->decodePolyId(ref, salt, it, ip); + m_tiles[it].flags[ip] = flags; + } + +}; + +static void floodNavmesh(dtNavMesh* nav, NavmeshFlags* flags, dtPolyRef start, unsigned char flag) +{ + // If already visited, skip. + if (flags->getFlags(start)) + return; + + flags->setFlags(start, flag); + + std::vector openList; + openList.push_back(start); + + while (openList.size()) + { + const dtPolyRef ref = openList.back(); + openList.pop_back(); + + // Get current poly and tile. + // The API input has been cheked already, skip checking internal data. + const dtMeshTile* tile = 0; + const dtPoly* poly = 0; + nav->getTileAndPolyByRefUnsafe(ref, &tile, &poly); + + // Visit linked polygons. + for (unsigned int i = poly->firstLink; i != DT_NULL_LINK; i = tile->links[i].next) + { + const dtPolyRef neiRef = tile->links[i].ref; + // Skip invalid and already visited. + if (!neiRef || flags->getFlags(neiRef)) + continue; + // Mark as visited + flags->setFlags(neiRef, flag); + // Visit neighbours + openList.push_back(neiRef); + } + } +} + +static void disableUnvisitedPolys(dtNavMesh* nav, NavmeshFlags* flags) +{ + for (int i = 0; i < nav->getMaxTiles(); ++i) + { + const dtMeshTile* tile = ((const dtNavMesh*)nav)->getTile(i); + if (!tile->header) continue; + const dtPolyRef base = nav->getPolyRefBase(tile); + for (int j = 0; j < tile->header->polyCount; ++j) + { + const dtPolyRef ref = base | (unsigned int)j; + if (!flags->getFlags(ref)) + { + unsigned short f = 0; + nav->getPolyFlags(ref, &f); + nav->setPolyFlags(ref, f | SAMPLE_POLYFLAGS_DISABLED); + } + } + } +} + +NavMeshPruneTool::NavMeshPruneTool() : + m_sample(0), + m_flags(0), + m_hitPosSet(false) +{ +} + +NavMeshPruneTool::~NavMeshPruneTool() +{ + delete m_flags; +} + +void NavMeshPruneTool::init(Sample* sample) +{ + m_sample = sample; +} + +void NavMeshPruneTool::reset() +{ + m_hitPosSet = false; + delete m_flags; + m_flags = 0; +} + +void NavMeshPruneTool::handleMenu() +{ + dtNavMesh* nav = m_sample->getNavMesh(); + if (!nav) return; + if (!m_flags) return; + + if (imguiButton("Clear Selection")) + { + m_flags->clearAllFlags(); + } + + if (imguiButton("Prune Unselected")) + { + disableUnvisitedPolys(nav, m_flags); + delete m_flags; + m_flags = 0; + } +} + +void NavMeshPruneTool::handleClick(const float* s, const float* p, bool shift) +{ + rcIgnoreUnused(s); + rcIgnoreUnused(shift); + + if (!m_sample) return; + InputGeom* geom = m_sample->getInputGeom(); + if (!geom) return; + dtNavMesh* nav = m_sample->getNavMesh(); + if (!nav) return; + dtNavMeshQuery* query = m_sample->getNavMeshQuery(); + if (!query) return; + + dtVcopy(m_hitPos, p); + m_hitPosSet = true; + + if (!m_flags) + { + m_flags = new NavmeshFlags; + m_flags->init(nav); + } + + const float halfExtents[3] = { 2, 4, 2 }; + dtQueryFilter filter; + dtPolyRef ref = 0; + query->findNearestPoly(p, halfExtents, &filter, &ref, 0); + + floodNavmesh(nav, m_flags, ref, 1); +} + +void NavMeshPruneTool::handleToggle() +{ +} + +void NavMeshPruneTool::handleStep() +{ +} + +void NavMeshPruneTool::handleUpdate(const float /*dt*/) +{ +} + +void NavMeshPruneTool::handleRender() +{ + duDebugDraw& dd = m_sample->getDebugDraw(); + + if (m_hitPosSet) + { + const float s = m_sample->getAgentRadius(); + const unsigned int col = duRGBA(255,255,255,255); + dd.begin(DU_DRAW_LINES); + dd.vertex(m_hitPos[0]-s,m_hitPos[1],m_hitPos[2], col); + dd.vertex(m_hitPos[0]+s,m_hitPos[1],m_hitPos[2], col); + dd.vertex(m_hitPos[0],m_hitPos[1]-s,m_hitPos[2], col); + dd.vertex(m_hitPos[0],m_hitPos[1]+s,m_hitPos[2], col); + dd.vertex(m_hitPos[0],m_hitPos[1],m_hitPos[2]-s, col); + dd.vertex(m_hitPos[0],m_hitPos[1],m_hitPos[2]+s, col); + dd.end(); + } + + const dtNavMesh* nav = m_sample->getNavMesh(); + if (m_flags && nav) + { + for (int i = 0; i < nav->getMaxTiles(); ++i) + { + const dtMeshTile* tile = nav->getTile(i); + if (!tile->header) continue; + const dtPolyRef base = nav->getPolyRefBase(tile); + for (int j = 0; j < tile->header->polyCount; ++j) + { + const dtPolyRef ref = base | (unsigned int)j; + if (m_flags->getFlags(ref)) + { + duDebugDrawNavMeshPoly(&dd, *nav, ref, duRGBA(255,255,255,128)); + } + } + } + } + +} + +void NavMeshPruneTool::handleRenderOverlay(double* proj, double* model, int* view) +{ + rcIgnoreUnused(model); + rcIgnoreUnused(proj); + + // Tool help + const int h = view[3]; + + imguiDrawText(280, h-40, IMGUI_ALIGN_LEFT, "LMB: Click fill area.", imguiRGBA(255,255,255,192)); +} diff --git a/r5dev/naveditor/NavMeshTesterTool.cpp b/r5dev/naveditor/NavMeshTesterTool.cpp new file mode 100644 index 00000000..a4df041c --- /dev/null +++ b/r5dev/naveditor/NavMeshTesterTool.cpp @@ -0,0 +1,1420 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include +#include +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" +#ifdef __APPLE__ +# include +#else +# include +#endif +#include "NavEditor/Include/imgui.h" +#include "NavEditor/Include/NavMeshTesterTool.h" +#include "NavEditor/Include/Sample.h" +#include "Recast/Include/Recast.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Detour/Include/DetourNavMeshBuilder.h" +#include "DebugUtils/Include/DetourDebugDraw.h" +#include "Detour/Include/DetourCommon.h" + +#ifdef WIN32 +# define snprintf _snprintf +#endif + +// Uncomment this to dump all the requests in stdout. +#define DUMP_REQS + +// Returns a random number [0..1] +static float frand() +{ +// return ((float)(rand() & 0xffff)/(float)0xffff); + return (float)rand()/(float)RAND_MAX; +} + +inline bool inRange(const float* v1, const float* v2, const float r, const float h) +{ + const float dx = v2[0] - v1[0]; + const float dy = v2[1] - v1[1]; + const float dz = v2[2] - v1[2]; + return (dx*dx + dz*dz) < r*r && fabsf(dy) < h; +} + + +static int fixupCorridor(dtPolyRef* path, const int npath, const int maxPath, + const dtPolyRef* visited, const int nvisited) +{ + int furthestPath = -1; + int furthestVisited = -1; + + // Find furthest common polygon. + for (int i = npath-1; i >= 0; --i) + { + bool found = false; + for (int j = nvisited-1; j >= 0; --j) + { + if (path[i] == visited[j]) + { + furthestPath = i; + furthestVisited = j; + found = true; + } + } + if (found) + break; + } + + // If no intersection found just return current path. + if (furthestPath == -1 || furthestVisited == -1) + return npath; + + // Concatenate paths. + + // Adjust beginning of the buffer to include the visited. + const int req = nvisited - furthestVisited; + const int orig = rcMin(furthestPath+1, npath); + int size = rcMax(0, npath-orig); + if (req+size > maxPath) + size = maxPath-req; + if (size) + memmove(path+req, path+orig, size*sizeof(dtPolyRef)); + + // Store visited + for (int i = 0; i < req; ++i) + path[i] = visited[(nvisited-1)-i]; + + return req+size; +} + +// This function checks if the path has a small U-turn, that is, +// a polygon further in the path is adjacent to the first polygon +// in the path. If that happens, a shortcut is taken. +// This can happen if the target (T) location is at tile boundary, +// and we're (S) approaching it parallel to the tile edge. +// The choice at the vertex can be arbitrary, +// +---+---+ +// |:::|:::| +// +-S-+-T-+ +// |:::| | <-- the step can end up in here, resulting U-turn path. +// +---+---+ +static int fixupShortcuts(dtPolyRef* path, int npath, dtNavMeshQuery* navQuery) +{ + if (npath < 3) + return npath; + + // Get connected polygons + static const int maxNeis = 16; + dtPolyRef neis[maxNeis]; + int nneis = 0; + + const dtMeshTile* tile = 0; + const dtPoly* poly = 0; + if (dtStatusFailed(navQuery->getAttachedNavMesh()->getTileAndPolyByRef(path[0], &tile, &poly))) + return npath; + + for (unsigned int k = poly->firstLink; k != DT_NULL_LINK; k = tile->links[k].next) + { + const dtLink* link = &tile->links[k]; + if (link->ref != 0) + { + if (nneis < maxNeis) + neis[nneis++] = link->ref; + } + } + + // If any of the neighbour polygons is within the next few polygons + // in the path, short cut to that polygon directly. + static const int maxLookAhead = 6; + int cut = 0; + for (int i = dtMin(maxLookAhead, npath) - 1; i > 1 && cut == 0; i--) { + for (int j = 0; j < nneis; j++) + { + if (path[i] == neis[j]) { + cut = i; + break; + } + } + } + if (cut > 1) + { + int offset = cut-1; + npath -= offset; + for (int i = 1; i < npath; i++) + path[i] = path[i+offset]; + } + + return npath; +} + +static bool getSteerTarget(dtNavMeshQuery* navQuery, const float* startPos, const float* endPos, + const float minTargetDist, + const dtPolyRef* path, const int pathSize, + float* steerPos, unsigned char& steerPosFlag, dtPolyRef& steerPosRef, + float* outPoints = 0, int* outPointCount = 0) +{ + // Find steer target. + static const int MAX_STEER_POINTS = 3; + float steerPath[MAX_STEER_POINTS*3]; + unsigned char steerPathFlags[MAX_STEER_POINTS]; + dtPolyRef steerPathPolys[MAX_STEER_POINTS]; + int nsteerPath = 0; + navQuery->findStraightPath(startPos, endPos, path, pathSize, + steerPath, steerPathFlags, steerPathPolys, &nsteerPath, MAX_STEER_POINTS); + if (!nsteerPath) + return false; + + if (outPoints && outPointCount) + { + *outPointCount = nsteerPath; + for (int i = 0; i < nsteerPath; ++i) + dtVcopy(&outPoints[i*3], &steerPath[i*3]); + } + + + // Find vertex far enough to steer to. + int ns = 0; + while (ns < nsteerPath) + { + // Stop at Off-Mesh link or when point is further than slop away. + if ((steerPathFlags[ns] & DT_STRAIGHTPATH_OFFMESH_CONNECTION) || + !inRange(&steerPath[ns*3], startPos, minTargetDist, 1000.0f)) + break; + ns++; + } + // Failed to find good point to steer to. + if (ns >= nsteerPath) + return false; + + dtVcopy(steerPos, &steerPath[ns*3]); + steerPos[2] = startPos[2]; + steerPosFlag = steerPathFlags[ns]; + steerPosRef = steerPathPolys[ns]; + + return true; +} + + +NavMeshTesterTool::NavMeshTesterTool() : + m_sample(0), + m_navMesh(0), + m_navQuery(0), + m_pathFindStatus(DT_FAILURE), + m_toolMode(TOOLMODE_PATHFIND_FOLLOW), + m_straightPathOptions(0), + m_startRef(0), + m_endRef(0), + m_npolys(0), + m_nstraightPath(0), + m_nsmoothPath(0), + m_nrandPoints(0), + m_randPointsInCircle(false), + m_hitResult(false), + m_distanceToWall(0), + m_sposSet(false), + m_eposSet(false), + m_pathIterNum(0), + m_pathIterPolyCount(0), + m_steerPointCount(0) +{ + m_filter.setIncludeFlags(SAMPLE_POLYFLAGS_ALL ^ SAMPLE_POLYFLAGS_DISABLED); + m_filter.setExcludeFlags(0); + + m_polyPickExt[0] = 2; + m_polyPickExt[1] = 2; + m_polyPickExt[2] = 4; + + m_neighbourhoodRadius = 2.5f; + m_randomRadius = 5.0f; +} + +void NavMeshTesterTool::init(Sample* sample) +{ + m_sample = sample; + m_navMesh = sample->getNavMesh(); + m_navQuery = sample->getNavMeshQuery(); + recalc(); + + if (m_navQuery) + { + // Change costs. + m_filter.setAreaCost(SAMPLE_POLYAREA_GROUND, 1.0f); + m_filter.setAreaCost(SAMPLE_POLYAREA_WATER, 10.0f); + m_filter.setAreaCost(SAMPLE_POLYAREA_ROAD, 1.0f); + m_filter.setAreaCost(SAMPLE_POLYAREA_DOOR, 1.0f); + m_filter.setAreaCost(SAMPLE_POLYAREA_GRASS, 2.0f); + m_filter.setAreaCost(SAMPLE_POLYAREA_JUMP, 1.5f); + } + + m_neighbourhoodRadius = sample->getAgentRadius() * 20.0f; + m_randomRadius = sample->getAgentRadius() * 30.0f; +} + +void NavMeshTesterTool::handleMenu() +{ + if (imguiCheck("Pathfind Follow", m_toolMode == TOOLMODE_PATHFIND_FOLLOW)) + { + m_toolMode = TOOLMODE_PATHFIND_FOLLOW; + recalc(); + } + if (imguiCheck("Pathfind Straight", m_toolMode == TOOLMODE_PATHFIND_STRAIGHT)) + { + m_toolMode = TOOLMODE_PATHFIND_STRAIGHT; + recalc(); + } + if (m_toolMode == TOOLMODE_PATHFIND_STRAIGHT) + { + imguiIndent(); + imguiLabel("Vertices at crossings"); + if (imguiCheck("None", m_straightPathOptions == 0)) + { + m_straightPathOptions = 0; + recalc(); + } + if (imguiCheck("Area", m_straightPathOptions == DT_STRAIGHTPATH_AREA_CROSSINGS)) + { + m_straightPathOptions = DT_STRAIGHTPATH_AREA_CROSSINGS; + recalc(); + } + if (imguiCheck("All", m_straightPathOptions == DT_STRAIGHTPATH_ALL_CROSSINGS)) + { + m_straightPathOptions = DT_STRAIGHTPATH_ALL_CROSSINGS; + recalc(); + } + + imguiUnindent(); + } + if (imguiCheck("Pathfind Sliced", m_toolMode == TOOLMODE_PATHFIND_SLICED)) + { + m_toolMode = TOOLMODE_PATHFIND_SLICED; + recalc(); + } + + imguiSeparator(); + + if (imguiCheck("Distance to Wall", m_toolMode == TOOLMODE_DISTANCE_TO_WALL)) + { + m_toolMode = TOOLMODE_DISTANCE_TO_WALL; + recalc(); + } + + imguiSeparator(); + + if (imguiCheck("Raycast", m_toolMode == TOOLMODE_RAYCAST)) + { + m_toolMode = TOOLMODE_RAYCAST; + recalc(); + } + + imguiSeparator(); + + if (imguiCheck("Find Polys in Circle", m_toolMode == TOOLMODE_FIND_POLYS_IN_CIRCLE)) + { + m_toolMode = TOOLMODE_FIND_POLYS_IN_CIRCLE; + recalc(); + } + if (imguiCheck("Find Polys in Shape", m_toolMode == TOOLMODE_FIND_POLYS_IN_SHAPE)) + { + m_toolMode = TOOLMODE_FIND_POLYS_IN_SHAPE; + recalc(); + } + + imguiSeparator(); + + if (imguiCheck("Find Local Neighbourhood", m_toolMode == TOOLMODE_FIND_LOCAL_NEIGHBOURHOOD)) + { + m_toolMode = TOOLMODE_FIND_LOCAL_NEIGHBOURHOOD; + recalc(); + } + + imguiSeparator(); + + if (imguiButton("Set Random Start")) + { + dtStatus status = m_navQuery->findRandomPoint(&m_filter, frand, &m_startRef, m_spos); + if (dtStatusSucceed(status)) + { + m_sposSet = true; + recalc(); + } + } + if (imguiButton("Set Random End", m_sposSet)) + { + if (m_sposSet) + { + dtStatus status = m_navQuery->findRandomPointAroundCircle(m_startRef, m_spos, m_randomRadius, &m_filter, frand, &m_endRef, m_epos); + if (dtStatusSucceed(status)) + { + m_eposSet = true; + recalc(); + } + } + } + + imguiSeparator(); + + if (imguiButton("Make Random Points")) + { + m_randPointsInCircle = false; + m_nrandPoints = 0; + for (int i = 0; i < MAX_RAND_POINTS; i++) + { + float pt[3]; + dtPolyRef ref; + dtStatus status = m_navQuery->findRandomPoint(&m_filter, frand, &ref, pt); + if (dtStatusSucceed(status)) + { + dtVcopy(&m_randPoints[m_nrandPoints*3], pt); + m_nrandPoints++; + } + } + } + if (imguiButton("Make Random Points Around", m_sposSet)) + { + if (m_sposSet) + { + m_nrandPoints = 0; + m_randPointsInCircle = true; + for (int i = 0; i < MAX_RAND_POINTS; i++) + { + float pt[3]; + dtPolyRef ref; + dtStatus status = m_navQuery->findRandomPointAroundCircle(m_startRef, m_spos, m_randomRadius, &m_filter, frand, &ref, pt); + if (dtStatusSucceed(status)) + { + dtVcopy(&m_randPoints[m_nrandPoints*3], pt); + m_nrandPoints++; + } + } + } + } + + + imguiSeparator(); + + imguiLabel("Include Flags"); + + imguiIndent(); + if (imguiCheck("Walk", (m_filter.getIncludeFlags() & SAMPLE_POLYFLAGS_WALK) != 0)) + { + m_filter.setIncludeFlags(m_filter.getIncludeFlags() ^ SAMPLE_POLYFLAGS_WALK); + recalc(); + } + if (imguiCheck("Swim", (m_filter.getIncludeFlags() & SAMPLE_POLYFLAGS_SWIM) != 0)) + { + m_filter.setIncludeFlags(m_filter.getIncludeFlags() ^ SAMPLE_POLYFLAGS_SWIM); + recalc(); + } + if (imguiCheck("Door", (m_filter.getIncludeFlags() & SAMPLE_POLYFLAGS_DOOR) != 0)) + { + m_filter.setIncludeFlags(m_filter.getIncludeFlags() ^ SAMPLE_POLYFLAGS_DOOR); + recalc(); + } + if (imguiCheck("Jump", (m_filter.getIncludeFlags() & SAMPLE_POLYFLAGS_JUMP) != 0)) + { + m_filter.setIncludeFlags(m_filter.getIncludeFlags() ^ SAMPLE_POLYFLAGS_JUMP); + recalc(); + } + imguiUnindent(); + + imguiSeparator(); + imguiLabel("Exclude Flags"); + + imguiIndent(); + if (imguiCheck("Walk", (m_filter.getExcludeFlags() & SAMPLE_POLYFLAGS_WALK) != 0)) + { + m_filter.setExcludeFlags(m_filter.getExcludeFlags() ^ SAMPLE_POLYFLAGS_WALK); + recalc(); + } + if (imguiCheck("Swim", (m_filter.getExcludeFlags() & SAMPLE_POLYFLAGS_SWIM) != 0)) + { + m_filter.setExcludeFlags(m_filter.getExcludeFlags() ^ SAMPLE_POLYFLAGS_SWIM); + recalc(); + } + if (imguiCheck("Door", (m_filter.getExcludeFlags() & SAMPLE_POLYFLAGS_DOOR) != 0)) + { + m_filter.setExcludeFlags(m_filter.getExcludeFlags() ^ SAMPLE_POLYFLAGS_DOOR); + recalc(); + } + if (imguiCheck("Jump", (m_filter.getExcludeFlags() & SAMPLE_POLYFLAGS_JUMP) != 0)) + { + m_filter.setExcludeFlags(m_filter.getExcludeFlags() ^ SAMPLE_POLYFLAGS_JUMP); + recalc(); + } + imguiUnindent(); + + imguiSeparator(); +} + +void NavMeshTesterTool::handleClick(const float* /*s*/, const float* p, bool shift) +{ + if (shift) + { + m_sposSet = true; + dtVcopy(m_spos, p); + } + else + { + m_eposSet = true; + dtVcopy(m_epos, p); + } + recalc(); +} + +void NavMeshTesterTool::handleStep() +{ +} + +void NavMeshTesterTool::handleToggle() +{ + // TODO: merge separate to a path iterator. Use same code in recalc() too. + if (m_toolMode != TOOLMODE_PATHFIND_FOLLOW) + return; + + if (!m_sposSet || !m_eposSet || !m_startRef || !m_endRef) + return; + + static const float STEP_SIZE = 0.5f; + static const float SLOP = 0.01f; + + if (m_pathIterNum == 0) + { + m_navQuery->findPath(m_startRef, m_endRef, m_spos, m_epos, &m_filter, m_polys, &m_npolys, MAX_POLYS); + m_nsmoothPath = 0; + + m_pathIterPolyCount = m_npolys; + if (m_pathIterPolyCount) + memcpy(m_pathIterPolys, m_polys, sizeof(dtPolyRef)*m_pathIterPolyCount); + + if (m_pathIterPolyCount) + { + // Iterate over the path to find smooth path on the detail mesh surface. + m_navQuery->closestPointOnPoly(m_startRef, m_spos, m_iterPos, 0); + m_navQuery->closestPointOnPoly(m_pathIterPolys[m_pathIterPolyCount-1], m_epos, m_targetPos, 0); + + m_nsmoothPath = 0; + + dtVcopy(&m_smoothPath[m_nsmoothPath*3], m_iterPos); + m_nsmoothPath++; + } + } + + dtVcopy(m_prevIterPos, m_iterPos); + + m_pathIterNum++; + + if (!m_pathIterPolyCount) + return; + + if (m_nsmoothPath >= MAX_SMOOTH) + return; + + // Move towards target a small advancement at a time until target reached or + // when ran out of memory to store the path. + + // Find location to steer towards. + float steerPos[3]; + unsigned char steerPosFlag; + dtPolyRef steerPosRef; + + if (!getSteerTarget(m_navQuery, m_iterPos, m_targetPos, SLOP, + m_pathIterPolys, m_pathIterPolyCount, steerPos, steerPosFlag, steerPosRef, + m_steerPoints, &m_steerPointCount)) + return; + + dtVcopy(m_steerPos, steerPos); + + bool endOfPath = (steerPosFlag & DT_STRAIGHTPATH_END) ? true : false; + bool offMeshConnection = (steerPosFlag & DT_STRAIGHTPATH_OFFMESH_CONNECTION) ? true : false; + + // Find movement delta. + float delta[3], len; + dtVsub(delta, steerPos, m_iterPos); + len = sqrtf(dtVdot(delta,delta)); + // If the steer target is end of path or off-mesh link, do not move past the location. + if ((endOfPath || offMeshConnection) && len < STEP_SIZE) + len = 1; + else + len = STEP_SIZE / len; + float moveTgt[3]; + dtVmad(moveTgt, m_iterPos, delta, len); + + // Move + float result[3]; + dtPolyRef visited[16]; + int nvisited = 0; + m_navQuery->moveAlongSurface(m_pathIterPolys[0], m_iterPos, moveTgt, &m_filter, + result, visited, &nvisited, 16); + m_pathIterPolyCount = fixupCorridor(m_pathIterPolys, m_pathIterPolyCount, MAX_POLYS, visited, nvisited); + m_pathIterPolyCount = fixupShortcuts(m_pathIterPolys, m_pathIterPolyCount, m_navQuery); + + float h = 0; + m_navQuery->getPolyHeight(m_pathIterPolys[0], result, &h); + result[1] = h; + dtVcopy(m_iterPos, result); + + // Handle end of path and off-mesh links when close enough. + if (endOfPath && inRange(m_iterPos, steerPos, SLOP, 1.0f)) + { + // Reached end of path. + dtVcopy(m_iterPos, m_targetPos); + if (m_nsmoothPath < MAX_SMOOTH) + { + dtVcopy(&m_smoothPath[m_nsmoothPath*3], m_iterPos); + m_nsmoothPath++; + } + return; + } + else if (offMeshConnection && inRange(m_iterPos, steerPos, SLOP, 1.0f)) + { + // Reached off-mesh connection. + float startPos[3], endPos[3]; + + // Advance the path up to and over the off-mesh connection. + dtPolyRef prevRef = 0, polyRef = m_pathIterPolys[0]; + int npos = 0; + while (npos < m_pathIterPolyCount && polyRef != steerPosRef) + { + prevRef = polyRef; + polyRef = m_pathIterPolys[npos]; + npos++; + } + for (int i = npos; i < m_pathIterPolyCount; ++i) + m_pathIterPolys[i-npos] = m_pathIterPolys[i]; + m_pathIterPolyCount -= npos; + + // Handle the connection. + dtStatus status = m_navMesh->getOffMeshConnectionPolyEndPoints(prevRef, polyRef, startPos, endPos); + if (dtStatusSucceed(status)) + { + if (m_nsmoothPath < MAX_SMOOTH) + { + dtVcopy(&m_smoothPath[m_nsmoothPath*3], startPos); + m_nsmoothPath++; + // Hack to make the dotted path not visible during off-mesh connection. + if (m_nsmoothPath & 1) + { + dtVcopy(&m_smoothPath[m_nsmoothPath*3], startPos); + m_nsmoothPath++; + } + } + // Move position at the other side of the off-mesh link. + dtVcopy(m_iterPos, endPos); + float eh = 0.0f; + m_navQuery->getPolyHeight(m_pathIterPolys[0], m_iterPos, &eh); + m_iterPos[1] = eh; + } + } + + // Store results. + if (m_nsmoothPath < MAX_SMOOTH) + { + dtVcopy(&m_smoothPath[m_nsmoothPath*3], m_iterPos); + m_nsmoothPath++; + } + +} + +void NavMeshTesterTool::handleUpdate(const float /*dt*/) +{ + if (m_toolMode == TOOLMODE_PATHFIND_SLICED) + { + if (dtStatusInProgress(m_pathFindStatus)) + { + m_pathFindStatus = m_navQuery->updateSlicedFindPath(1,0); + } + if (dtStatusSucceed(m_pathFindStatus)) + { + m_navQuery->finalizeSlicedFindPath(m_polys, &m_npolys, MAX_POLYS); + m_nstraightPath = 0; + if (m_npolys) + { + // In case of partial path, make sure the end point is clamped to the last polygon. + float epos[3]; + dtVcopy(epos, m_epos); + if (m_polys[m_npolys-1] != m_endRef) + m_navQuery->closestPointOnPoly(m_polys[m_npolys-1], m_epos, epos, 0); + + m_navQuery->findStraightPath(m_spos, epos, m_polys, m_npolys, + m_straightPath, m_straightPathFlags, + m_straightPathPolys, &m_nstraightPath, MAX_POLYS, DT_STRAIGHTPATH_ALL_CROSSINGS); + } + + m_pathFindStatus = DT_FAILURE; + } + } +} + +void NavMeshTesterTool::reset() +{ + m_startRef = 0; + m_endRef = 0; + m_npolys = 0; + m_nstraightPath = 0; + m_nsmoothPath = 0; + memset(m_hitPos, 0, sizeof(m_hitPos)); + memset(m_hitNormal, 0, sizeof(m_hitNormal)); + m_distanceToWall = 0; +} + + +void NavMeshTesterTool::recalc() +{ + if (!m_navMesh) + return; + + if (m_sposSet) + m_navQuery->findNearestPoly(m_spos, m_polyPickExt, &m_filter, &m_startRef, 0); + else + m_startRef = 0; + + if (m_eposSet) + m_navQuery->findNearestPoly(m_epos, m_polyPickExt, &m_filter, &m_endRef, 0); + else + m_endRef = 0; + + m_pathFindStatus = DT_FAILURE; + + if (m_toolMode == TOOLMODE_PATHFIND_FOLLOW) + { + m_pathIterNum = 0; + if (m_sposSet && m_eposSet && m_startRef && m_endRef) + { +#ifdef DUMP_REQS + printf("pi %f %f %f %f %f %f 0x%x 0x%x\n", + m_spos[0],m_spos[1],m_spos[2], m_epos[0],m_epos[1],m_epos[2], + m_filter.getIncludeFlags(), m_filter.getExcludeFlags()); +#endif + + m_navQuery->findPath(m_startRef, m_endRef, m_spos, m_epos, &m_filter, m_polys, &m_npolys, MAX_POLYS); + + m_nsmoothPath = 0; + + if (m_npolys) + { + // Iterate over the path to find smooth path on the detail mesh surface. + dtPolyRef polys[MAX_POLYS]; + memcpy(polys, m_polys, sizeof(dtPolyRef)*m_npolys); + int npolys = m_npolys; + + float iterPos[3], targetPos[3]; + m_navQuery->closestPointOnPoly(m_startRef, m_spos, iterPos, 0); + m_navQuery->closestPointOnPoly(polys[npolys-1], m_epos, targetPos, 0); + + static const float STEP_SIZE = 2.0f; + static const float SLOP = 0.1f; + + m_nsmoothPath = 0; + + dtVcopy(&m_smoothPath[m_nsmoothPath*3], iterPos); + m_nsmoothPath++; + + // Move towards target a small advancement at a time until target reached or + // when ran out of memory to store the path. + while (npolys && m_nsmoothPath < MAX_SMOOTH) + { + // Find location to steer towards. + float steerPos[3]; + unsigned char steerPosFlag; + dtPolyRef steerPosRef; + + if (!getSteerTarget(m_navQuery, iterPos, targetPos, SLOP, + polys, npolys, steerPos, steerPosFlag, steerPosRef)) + break; + + bool endOfPath = (steerPosFlag & DT_STRAIGHTPATH_END) ? true : false; + bool offMeshConnection = (steerPosFlag & DT_STRAIGHTPATH_OFFMESH_CONNECTION) ? true : false; + + // Find movement delta. + float delta[3], len; + dtVsub(delta, steerPos, iterPos); + len = dtMathSqrtf(dtVdot(delta, delta)); + // If the steer target is end of path or off-mesh link, do not move past the location. + if ((endOfPath || offMeshConnection) && len < STEP_SIZE) + len = 1; + else + len = STEP_SIZE / len; + float moveTgt[3]; + dtVmad(moveTgt, iterPos, delta, len); + + // Move + float result[3]; + dtPolyRef visited[16]; + int nvisited = 0; + m_navQuery->moveAlongSurface(polys[0], iterPos, moveTgt, &m_filter, + result, visited, &nvisited, 16); + + npolys = fixupCorridor(polys, npolys, MAX_POLYS, visited, nvisited); + npolys = fixupShortcuts(polys, npolys, m_navQuery); + + float h = 0; + m_navQuery->getPolyHeight(polys[0], result, &h); + result[2] = h; + dtVcopy(iterPos, result); + + // Handle end of path and off-mesh links when close enough. + if (endOfPath && inRange(iterPos, steerPos, SLOP, 1.0f)) + { + // Reached end of path. + dtVcopy(iterPos, targetPos); + if (m_nsmoothPath < MAX_SMOOTH) + { + dtVcopy(&m_smoothPath[m_nsmoothPath*3], iterPos); + m_nsmoothPath++; + } + break; + } + else if (offMeshConnection && inRange(iterPos, steerPos, SLOP, 1.0f)) + { + // Reached off-mesh connection. + float startPos[3], endPos[3]; + + // Advance the path up to and over the off-mesh connection. + dtPolyRef prevRef = 0, polyRef = polys[0]; + int npos = 0; + while (npos < npolys && polyRef != steerPosRef) + { + prevRef = polyRef; + polyRef = polys[npos]; + npos++; + } + for (int i = npos; i < npolys; ++i) + polys[i-npos] = polys[i]; + npolys -= npos; + + // Handle the connection. + dtStatus status = m_navMesh->getOffMeshConnectionPolyEndPoints(prevRef, polyRef, startPos, endPos); + if (dtStatusSucceed(status)) + { + if (m_nsmoothPath < MAX_SMOOTH) + { + dtVcopy(&m_smoothPath[m_nsmoothPath*3], startPos); + m_nsmoothPath++; + // Hack to make the dotted path not visible during off-mesh connection. + if (m_nsmoothPath & 1) + { + dtVcopy(&m_smoothPath[m_nsmoothPath*3], startPos); + m_nsmoothPath++; + } + } + // Move position at the other side of the off-mesh link. + dtVcopy(iterPos, endPos); + float eh = 0.0f; + m_navQuery->getPolyHeight(polys[0], iterPos, &eh); + iterPos[2] = eh; + } + } + + // Store results. + if (m_nsmoothPath < MAX_SMOOTH) + { + dtVcopy(&m_smoothPath[m_nsmoothPath*3], iterPos); + m_nsmoothPath++; + } + } + } + + } + else + { + m_npolys = 0; + m_nsmoothPath = 0; + } + } + else if (m_toolMode == TOOLMODE_PATHFIND_STRAIGHT) + { + if (m_sposSet && m_eposSet && m_startRef && m_endRef) + { +#ifdef DUMP_REQS + printf("ps %f %f %f %f %f %f 0x%x 0x%x\n", + m_spos[0],m_spos[1],m_spos[2], m_epos[0],m_epos[1],m_epos[2], + m_filter.getIncludeFlags(), m_filter.getExcludeFlags()); +#endif + m_navQuery->findPath(m_startRef, m_endRef, m_spos, m_epos, &m_filter, m_polys, &m_npolys, MAX_POLYS); + m_nstraightPath = 0; + if (m_npolys) + { + // In case of partial path, make sure the end point is clamped to the last polygon. + float epos[3]; + dtVcopy(epos, m_epos); + if (m_polys[m_npolys-1] != m_endRef) + m_navQuery->closestPointOnPoly(m_polys[m_npolys-1], m_epos, epos, 0); + + m_navQuery->findStraightPath(m_spos, epos, m_polys, m_npolys, + m_straightPath, m_straightPathFlags, + m_straightPathPolys, &m_nstraightPath, MAX_POLYS, m_straightPathOptions); + } + } + else + { + m_npolys = 0; + m_nstraightPath = 0; + } + } + else if (m_toolMode == TOOLMODE_PATHFIND_SLICED) + { + if (m_sposSet && m_eposSet && m_startRef && m_endRef) + { +#ifdef DUMP_REQS + printf("ps %f %f %f %f %f %f 0x%x 0x%x\n", + m_spos[0],m_spos[1],m_spos[2], m_epos[0],m_epos[1],m_epos[2], + m_filter.getIncludeFlags(), m_filter.getExcludeFlags()); +#endif + m_npolys = 0; + m_nstraightPath = 0; + + m_pathFindStatus = m_navQuery->initSlicedFindPath(m_startRef, m_endRef, m_spos, m_epos, &m_filter, DT_FINDPATH_ANY_ANGLE); + } + else + { + m_npolys = 0; + m_nstraightPath = 0; + } + } + else if (m_toolMode == TOOLMODE_RAYCAST) + { + m_nstraightPath = 0; + if (m_sposSet && m_eposSet && m_startRef) + { +#ifdef DUMP_REQS + printf("rc %f %f %f %f %f %f 0x%x 0x%x\n", + m_spos[0],m_spos[1],m_spos[2], m_epos[0],m_epos[1],m_epos[2], + m_filter.getIncludeFlags(), m_filter.getExcludeFlags()); +#endif + float t = 0; + m_npolys = 0; + m_nstraightPath = 2; + m_straightPath[0] = m_spos[0]; + m_straightPath[1] = m_spos[1]; + m_straightPath[2] = m_spos[2]; + m_navQuery->raycast(m_startRef, m_spos, m_epos, &m_filter, &t, m_hitNormal, m_polys, &m_npolys, MAX_POLYS); + if (t > 1) + { + // No hit + dtVcopy(m_hitPos, m_epos); + m_hitResult = false; + } + else + { + // Hit + dtVlerp(m_hitPos, m_spos, m_epos, t); + m_hitResult = true; + } + // Adjust height. + if (m_npolys > 0) + { + float h = 0; + m_navQuery->getPolyHeight(m_polys[m_npolys-1], m_hitPos, &h); + m_hitPos[2] = h; + } + dtVcopy(&m_straightPath[3], m_hitPos); + } + } + else if (m_toolMode == TOOLMODE_DISTANCE_TO_WALL) + { + m_distanceToWall = 0; + if (m_sposSet && m_startRef) + { +#ifdef DUMP_REQS + printf("dw %f %f %f %f 0x%x 0x%x\n", + m_spos[0],m_spos[1],m_spos[2], 100.0f, + m_filter.getIncludeFlags(), m_filter.getExcludeFlags()); +#endif + m_distanceToWall = 0.0f; + m_navQuery->findDistanceToWall(m_startRef, m_spos, 100.0f, &m_filter, &m_distanceToWall, m_hitPos, m_hitNormal); + } + } + else if (m_toolMode == TOOLMODE_FIND_POLYS_IN_CIRCLE) + { + if (m_sposSet && m_startRef && m_eposSet) + { + const float dx = m_epos[0] - m_spos[0]; + const float dz = m_epos[2] - m_spos[2]; + float dist = sqrtf(dx*dx + dz*dz); +#ifdef DUMP_REQS + printf("fpc %f %f %f %f 0x%x 0x%x\n", + m_spos[0],m_spos[1],m_spos[2], dist, + m_filter.getIncludeFlags(), m_filter.getExcludeFlags()); +#endif + m_navQuery->findPolysAroundCircle(m_startRef, m_spos, dist, &m_filter, + m_polys, m_parent, 0, &m_npolys, MAX_POLYS); + + } + } + else if (m_toolMode == TOOLMODE_FIND_POLYS_IN_SHAPE) + { + if (m_sposSet && m_startRef && m_eposSet) + { + const float nx = (m_epos[2] - m_spos[2])*0.25f; + const float nz = -(m_epos[0] - m_spos[0])*0.25f; + const float agentHeight = m_sample ? m_sample->getAgentHeight() : 0; + + m_queryPoly[0] = m_spos[0] + nx*1.2f; + m_queryPoly[1] = m_spos[1] + agentHeight/2; + m_queryPoly[2] = m_spos[2] + nz*1.2f; + + m_queryPoly[3] = m_spos[0] - nx*1.3f; + m_queryPoly[4] = m_spos[1] + agentHeight/2; + m_queryPoly[5] = m_spos[2] - nz*1.3f; + + m_queryPoly[6] = m_epos[0] - nx*0.8f; + m_queryPoly[7] = m_epos[1] + agentHeight/2; + m_queryPoly[8] = m_epos[2] - nz*0.8f; + + m_queryPoly[9] = m_epos[0] + nx; + m_queryPoly[10] = m_epos[1] + agentHeight/2; + m_queryPoly[11] = m_epos[2] + nz; + +#ifdef DUMP_REQS + printf("fpp %f %f %f %f %f %f %f %f %f %f %f %f 0x%x 0x%x\n", + m_queryPoly[0],m_queryPoly[1],m_queryPoly[2], + m_queryPoly[3],m_queryPoly[4],m_queryPoly[5], + m_queryPoly[6],m_queryPoly[7],m_queryPoly[8], + m_queryPoly[9],m_queryPoly[10],m_queryPoly[11], + m_filter.getIncludeFlags(), m_filter.getExcludeFlags()); +#endif + m_navQuery->findPolysAroundShape(m_startRef, m_queryPoly, 4, &m_filter, + m_polys, m_parent, 0, &m_npolys, MAX_POLYS); + } + } + else if (m_toolMode == TOOLMODE_FIND_LOCAL_NEIGHBOURHOOD) + { + if (m_sposSet && m_startRef) + { +#ifdef DUMP_REQS + printf("fln %f %f %f %f 0x%x 0x%x\n", + m_spos[0],m_spos[1],m_spos[2], m_neighbourhoodRadius, + m_filter.getIncludeFlags(), m_filter.getExcludeFlags()); +#endif + m_navQuery->findLocalNeighbourhood(m_startRef, m_spos, m_neighbourhoodRadius, &m_filter, + m_polys, m_parent, &m_npolys, MAX_POLYS); + } + } +} + +static void getPolyCenter(dtNavMesh* navMesh, dtPolyRef ref, float* center) +{ + center[0] = 0; + center[1] = 0; + center[2] = 0; + + const dtMeshTile* tile = 0; + const dtPoly* poly = 0; + dtStatus status = navMesh->getTileAndPolyByRef(ref, &tile, &poly); + if (dtStatusFailed(status)) + return; + + for (int i = 0; i < (int)poly->vertCount; ++i) + { + const float* v = &tile->verts[poly->verts[i]*3]; + center[0] += v[0]; + center[1] += v[1]; + center[2] += v[2]; + } + const float s = 1.0f / poly->vertCount; + center[0] *= s; + center[1] *= s; + center[2] *= s; +} + + + +void NavMeshTesterTool::handleRender() +{ + duDebugDraw& dd = m_sample->getDebugDraw(); + + static const unsigned int startCol = duRGBA(128,25,0,192); + static const unsigned int endCol = duRGBA(51,102,0,129); + static const unsigned int pathCol = duRGBA(0,0,0,64); + + const float agentRadius = m_sample->getAgentRadius(); + const float agentHeight = m_sample->getAgentHeight(); + const float agentClimb = m_sample->getAgentClimb(); + + dd.depthMask(false); + if (m_sposSet) + drawAgent(m_spos, agentRadius, agentHeight, agentClimb, startCol); + if (m_eposSet) + drawAgent(m_epos, agentRadius, agentHeight, agentClimb, endCol); + dd.depthMask(true); + + if (!m_navMesh) + { + return; + } + + if (m_toolMode == TOOLMODE_PATHFIND_FOLLOW) + { + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_startRef, startCol); + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_endRef, endCol); + + if (m_npolys) + { + for (int i = 0; i < m_npolys; ++i) + { + if (m_polys[i] == m_startRef || m_polys[i] == m_endRef) + continue; + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_polys[i], pathCol); + } + } + + if (m_nsmoothPath) + { + dd.depthMask(false); + const unsigned int spathCol = duRGBA(0,0,0,220); + dd.begin(DU_DRAW_LINES, 3.0f); + for (int i = 0; i < m_nsmoothPath; ++i) + dd.vertex(m_smoothPath[i*3], m_smoothPath[i*3+1], m_smoothPath[i*3+2] + 0.1f, spathCol); + dd.end(); + dd.depthMask(true); + } + + if (m_pathIterNum) + { + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_pathIterPolys[0], duRGBA(255,255,255,128)); + + dd.depthMask(false); + dd.begin(DU_DRAW_LINES, 1.0f); + + const unsigned int prevCol = duRGBA(255,192,0,220); + const unsigned int curCol = duRGBA(255,255,255,220); + const unsigned int steerCol = duRGBA(0,192,255,220); + + dd.vertex(m_prevIterPos[0],m_prevIterPos[1],m_prevIterPos[2] - 0.3f, prevCol); + dd.vertex(m_prevIterPos[0],m_prevIterPos[1],m_prevIterPos[2] + 0.3f, prevCol); + + dd.vertex(m_iterPos[0],m_iterPos[1],m_iterPos[2] - 0.3f, curCol); + dd.vertex(m_iterPos[0],m_iterPos[1],m_iterPos[2] + 0.3f, curCol); + + dd.vertex(m_prevIterPos[0],m_prevIterPos[1],m_prevIterPos[2] + 0.3f, prevCol); + dd.vertex(m_iterPos[0],m_iterPos[1],m_iterPos[2] + 0.3f, prevCol); + + dd.vertex(m_prevIterPos[0],m_prevIterPos[1],m_prevIterPos[2] + 0.3f, steerCol); + dd.vertex(m_steerPos[0],m_steerPos[1],m_steerPos[2] + 0.3f, steerCol); + + for (int i = 0; i < m_steerPointCount-1; ++i) + { + dd.vertex(m_steerPoints[i*3+0],m_steerPoints[i*3+1],m_steerPoints[i*3+2] + 0.2f, duDarkenCol(steerCol)); + dd.vertex(m_steerPoints[(i+1)*3+0],m_steerPoints[(i+1)*3+1],m_steerPoints[(i+1)*3+2] + 0.2f, duDarkenCol(steerCol)); + } + + dd.end(); + dd.depthMask(true); + } + } + else if (m_toolMode == TOOLMODE_PATHFIND_STRAIGHT || + m_toolMode == TOOLMODE_PATHFIND_SLICED) + { + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_startRef, startCol); + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_endRef, endCol); + + if (m_npolys) + { + for (int i = 0; i < m_npolys; ++i) + { + if (m_polys[i] == m_startRef || m_polys[i] == m_endRef) + continue; + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_polys[i], pathCol); + } + } + + if (m_nstraightPath) + { + dd.depthMask(false); + const unsigned int spathCol = duRGBA(64,16,0,220); + const unsigned int offMeshCol = duRGBA(128,96,0,220); + dd.begin(DU_DRAW_LINES, 2.0f); + for (int i = 0; i < m_nstraightPath-1; ++i) + { + unsigned int col; + if (m_straightPathFlags[i] & DT_STRAIGHTPATH_OFFMESH_CONNECTION) + col = offMeshCol; + else + col = spathCol; + + dd.vertex(m_straightPath[i*3], m_straightPath[i*3+1], m_straightPath[i*3+2] + 0.4f, col); + dd.vertex(m_straightPath[(i+1)*3], m_straightPath[(i+1)*3+1], m_straightPath[(i+1)*3+2] + 0.4f, col); + } + dd.end(); + dd.begin(DU_DRAW_POINTS, 6.0f); + for (int i = 0; i < m_nstraightPath; ++i) + { + unsigned int col; + if (m_straightPathFlags[i] & DT_STRAIGHTPATH_START) + col = startCol; + else if (m_straightPathFlags[i] & DT_STRAIGHTPATH_END) + col = endCol; + else if (m_straightPathFlags[i] & DT_STRAIGHTPATH_OFFMESH_CONNECTION) + col = offMeshCol; + else + col = spathCol; + dd.vertex(m_straightPath[i*3], m_straightPath[i*3+1], m_straightPath[i*3+2] + 0.4f, col); + } + dd.end(); + dd.depthMask(true); + } + } + else if (m_toolMode == TOOLMODE_RAYCAST) + { + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_startRef, startCol); + + if (m_nstraightPath) + { + for (int i = 1; i < m_npolys; ++i) + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_polys[i], pathCol); + + dd.depthMask(false); + const unsigned int spathCol = m_hitResult ? duRGBA(64,16,0,220) : duRGBA(240,240,240,220); + dd.begin(DU_DRAW_LINES, 2.0f); + for (int i = 0; i < m_nstraightPath-1; ++i) + { + dd.vertex(m_straightPath[i*3], m_straightPath[i*3+1], m_straightPath[i*3+2] + 0.4f, spathCol); + dd.vertex(m_straightPath[(i+1)*3], m_straightPath[(i+1)*3+1], m_straightPath[(i+1)*3+2] + 0.4f, spathCol); + } + dd.end(); + dd.begin(DU_DRAW_POINTS, 4.0f); + for (int i = 0; i < m_nstraightPath; ++i) + dd.vertex(m_straightPath[i*3], m_straightPath[i*3+1], m_straightPath[i*3+2] + 0.4f, spathCol); + dd.end(); + + if (m_hitResult) + { + const unsigned int hitCol = duRGBA(0,0,0,128); + dd.begin(DU_DRAW_LINES, 2.0f); + dd.vertex(m_hitPos[0], m_hitPos[1] , m_hitPos[2] + 0.4f, hitCol); + dd.vertex(m_hitPos[0] + m_hitNormal[0]*agentRadius, + m_hitPos[1] + m_hitNormal[1] * agentRadius , + m_hitPos[2] + 0.4f + m_hitNormal[2] * agentRadius, hitCol); + dd.end(); + } + dd.depthMask(true); + } + } + else if (m_toolMode == TOOLMODE_DISTANCE_TO_WALL) + { + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_startRef, startCol); + dd.depthMask(false); + duDebugDrawCircle(&dd, m_spos[0], m_spos[1], m_spos[2] + agentHeight / 2, m_distanceToWall, duRGBA(64,16,0,220), 2.0f); + dd.begin(DU_DRAW_LINES, 3.0f); + dd.vertex(m_hitPos[0], m_hitPos[1] , m_hitPos[2] + 0.02f, duRGBA(0,0,0,192)); + dd.vertex(m_hitPos[0], m_hitPos[1] , m_hitPos[2] + agentHeight, duRGBA(0,0,0,192)); + dd.end(); + dd.depthMask(true); + } + else if (m_toolMode == TOOLMODE_FIND_POLYS_IN_CIRCLE) + { + for (int i = 0; i < m_npolys; ++i) + { + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_polys[i], pathCol); + dd.depthMask(false); + if (m_parent[i]) + { + float p0[3], p1[3]; + dd.depthMask(false); + getPolyCenter(m_navMesh, m_parent[i], p0); + getPolyCenter(m_navMesh, m_polys[i], p1); + duDebugDrawArc(&dd, p0[0],p0[1],p0[2], p1[0],p1[1],p1[2], 0.25f, 0.0f, 0.4f, duRGBA(0,0,0,128), 2.0f); + dd.depthMask(true); + } + dd.depthMask(true); + } + + if (m_sposSet && m_eposSet) + { + dd.depthMask(false); + const float dx = m_epos[0] - m_spos[0]; + const float dy = m_epos[1] - m_spos[1]; + const float dist = sqrtf(dx*dx + dy*dy); + duDebugDrawCircle(&dd, m_spos[0], m_spos[1], m_spos[2] + agentHeight / 2, dist, duRGBA(64,16,0,220), 2.0f); + dd.depthMask(true); + } + } + else if (m_toolMode == TOOLMODE_FIND_POLYS_IN_SHAPE) + { + for (int i = 0; i < m_npolys; ++i) + { + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_polys[i], pathCol); + dd.depthMask(false); + if (m_parent[i]) + { + float p0[3], p1[3]; + dd.depthMask(false); + getPolyCenter(m_navMesh, m_parent[i], p0); + getPolyCenter(m_navMesh, m_polys[i], p1); + duDebugDrawArc(&dd, p0[0],p0[1],p0[2], p1[0],p1[1],p1[2], 0.25f, 0.0f, 0.4f, duRGBA(0,0,0,128), 2.0f); + dd.depthMask(true); + } + dd.depthMask(true); + } + + if (m_sposSet && m_eposSet) + { + dd.depthMask(false); + const unsigned int col = duRGBA(64,16,0,220); + dd.begin(DU_DRAW_LINES, 2.0f); + for (int i = 0, j = 3; i < 4; j=i++) + { + const float* p0 = &m_queryPoly[j*3]; + const float* p1 = &m_queryPoly[i*3]; + dd.vertex(p0, col); + dd.vertex(p1, col); + } + dd.end(); + dd.depthMask(true); + } + } + else if (m_toolMode == TOOLMODE_FIND_LOCAL_NEIGHBOURHOOD) + { + for (int i = 0; i < m_npolys; ++i) + { + duDebugDrawNavMeshPoly(&dd, *m_navMesh, m_polys[i], pathCol); + dd.depthMask(false); + if (m_parent[i]) + { + float p0[3], p1[3]; + dd.depthMask(false); + getPolyCenter(m_navMesh, m_parent[i], p0); + getPolyCenter(m_navMesh, m_polys[i], p1); + duDebugDrawArc(&dd, p0[0],p0[1],p0[2], p1[0],p1[1],p1[2], 0.25f, 0.0f, 0.4f, duRGBA(0,0,0,128), 2.0f); + dd.depthMask(true); + } + + static const int MAX_SEGS = DT_VERTS_PER_POLYGON*4; + float segs[MAX_SEGS*6]; + dtPolyRef refs[MAX_SEGS]; + memset(refs, 0, sizeof(dtPolyRef)*MAX_SEGS); + int nsegs = 0; + m_navQuery->getPolyWallSegments(m_polys[i], &m_filter, segs, refs, &nsegs, MAX_SEGS); + dd.begin(DU_DRAW_LINES, 2.0f); + for (int j = 0; j < nsegs; ++j) + { + const float* s = &segs[j*6]; + + // Skip too distant segments. + float tseg; + float distSqr = dtDistancePtSegSqr2D(m_spos, s, s+3, tseg); + if (distSqr > dtSqr(m_neighbourhoodRadius)) + continue; + + float delta[3], norm[3], p0[3], p1[3]; + dtVsub(delta, s+3,s); + dtVmad(p0, s, delta, 0.5f); + norm[0] = delta[2]; + norm[1] = -delta[0]; //NB(warmist):unsure TODO + norm[2] = 0; + dtVnormalize(norm); + dtVmad(p1, p0, norm, agentRadius*0.5f); + + // Skip backfacing segments. + if (refs[j]) + { + unsigned int col = duRGBA(255,255,255,32); + dd.vertex(s[0],s[1],s[2] + agentClimb,col); + dd.vertex(s[3],s[4],s[5] + agentClimb,col); + } + else + { + unsigned int col = duRGBA(192,32,16,192); + if (dtTriArea2D(m_spos, s, s+3) < 0.0f) + col = duRGBA(96,32,16,192); + + dd.vertex(p0[0],p0[1],p0[2] + agentClimb,col); + dd.vertex(p1[0],p1[1],p1[2] + agentClimb,col); + + dd.vertex(s[0],s[1],s[2] + agentClimb,col); + dd.vertex(s[3],s[4],s[5] + agentClimb,col); + } + } + dd.end(); + + dd.depthMask(true); + } + + if (m_sposSet) + { + dd.depthMask(false); + duDebugDrawCircle(&dd, m_spos[0], m_spos[1], m_spos[2] + agentHeight / 2, m_neighbourhoodRadius, duRGBA(64,16,0,220), 2.0f); + dd.depthMask(true); + } + } + + if (m_nrandPoints > 0) + { + dd.begin(DU_DRAW_POINTS, 6.0f); + for (int i = 0; i < m_nrandPoints; i++) + { + const float* p = &m_randPoints[i*3]; + dd.vertex(p[0],p[1],p[2] + 0.1f, duRGBA(220,32,16,192)); + } + dd.end(); + + if (m_randPointsInCircle && m_sposSet) + { + duDebugDrawCircle(&dd, m_spos[0], m_spos[1], m_spos[2] + agentHeight / 2, m_randomRadius, duRGBA(64,16,0,220), 2.0f); + } + } +} + +void NavMeshTesterTool::handleRenderOverlay(double* proj, double* model, int* view) +{ + GLdouble x, y, z; + + // Draw start and end point labels + if (m_sposSet && gluProject((GLdouble)m_spos[0], (GLdouble)m_spos[1], (GLdouble)m_spos[2], + model, proj, view, &x, &y, &z)) + { + imguiDrawText((int)x, (int)(y-25), IMGUI_ALIGN_CENTER, "Start", imguiRGBA(0,0,0,220)); + } + if (m_eposSet && gluProject((GLdouble)m_epos[0], (GLdouble)m_epos[1], (GLdouble)m_epos[2], + model, proj, view, &x, &y, &z)) + { + imguiDrawText((int)x, (int)(y-25), IMGUI_ALIGN_CENTER, "End", imguiRGBA(0,0,0,220)); + } + + // Tool help + const int h = view[3]; + imguiDrawText(280, h-40, IMGUI_ALIGN_LEFT, "LMB+SHIFT: Set start location LMB: Set end location", imguiRGBA(255,255,255,192)); +} + +void NavMeshTesterTool::drawAgent(const float* pos, float r, float h, float c, const unsigned int col) +{ + duDebugDraw& dd = m_sample->getDebugDraw(); + + dd.depthMask(false); + + // Agent dimensions. + duDebugDrawCylinderWire(&dd, pos[0]-r, pos[1]-r, pos[2] + 0.02f, pos[0]+r, pos[1]+r, pos[2]+h, col, 2.0f); + + duDebugDrawCircle(&dd, pos[0],pos[1],pos[2] + c,r,duRGBA(0,0,0,64),1.0f); + + unsigned int colb = duRGBA(0,0,0,196); + dd.begin(DU_DRAW_LINES); + dd.vertex(pos[0], pos[1], pos[2] - c, colb); + dd.vertex(pos[0], pos[1], pos[2] + c, colb); + dd.vertex(pos[0]-r/2, pos[1], pos[2] + 0.02f, colb); + dd.vertex(pos[0]+r/2, pos[1], pos[2] + 0.02f, colb); + dd.vertex(pos[0], pos[1] - r / 2, pos[2] + 0.02f, colb); + dd.vertex(pos[0], pos[1] + r / 2, pos[2] + 0.02f, colb); + dd.end(); + + dd.depthMask(true); +} diff --git a/r5dev/naveditor/OffMeshConnectionTool.cpp b/r5dev/naveditor/OffMeshConnectionTool.cpp new file mode 100644 index 00000000..1013b4a2 --- /dev/null +++ b/r5dev/naveditor/OffMeshConnectionTool.cpp @@ -0,0 +1,177 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include +#include +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" +#ifdef __APPLE__ +# include +#else +# include +#endif +#include "NavEditor/Include/imgui.h" +#include "NavEditor/Include/OffMeshConnectionTool.h" +#include "NavEditor/Include/InputGeom.h" +#include "NavEditor/Include/Sample.h" +#include "Recast/Include/Recast.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "DebugUtils/Include/DetourDebugDraw.h" + +#ifdef WIN32 +# define snprintf _snprintf +#endif + +OffMeshConnectionTool::OffMeshConnectionTool() : + m_sample(0), + m_hitPosSet(0), + m_bidir(true), + m_oldFlags(0) +{ +} + +OffMeshConnectionTool::~OffMeshConnectionTool() +{ + if (m_sample) + { + m_sample->setNavMeshDrawFlags(m_oldFlags); + } +} + +void OffMeshConnectionTool::init(Sample* sample) +{ + if (m_sample != sample) + { + m_sample = sample; + m_oldFlags = m_sample->getNavMeshDrawFlags(); + m_sample->setNavMeshDrawFlags(m_oldFlags & ~DU_DRAWNAVMESH_OFFMESHCONS); + } +} + +void OffMeshConnectionTool::reset() +{ + m_hitPosSet = false; +} + +void OffMeshConnectionTool::handleMenu() +{ + if (imguiCheck("One Way", !m_bidir)) + m_bidir = false; + if (imguiCheck("Bidirectional", m_bidir)) + m_bidir = true; +} + +void OffMeshConnectionTool::handleClick(const float* /*s*/, const float* p, bool shift) +{ + if (!m_sample) return; + InputGeom* geom = m_sample->getInputGeom(); + if (!geom) return; + + if (shift) + { + // Delete + // Find nearest link end-point + float nearestDist = FLT_MAX; + int nearestIndex = -1; + const float* verts = geom->getOffMeshConnectionVerts(); + for (int i = 0; i < geom->getOffMeshConnectionCount()*2; ++i) + { + const float* v = &verts[i*3]; + float d = rcVdistSqr(p, v); + if (d < nearestDist) + { + nearestDist = d; + nearestIndex = i/2; // Each link has two vertices. + } + } + // If end point close enough, delete it. + if (nearestIndex != -1 && + sqrtf(nearestDist) < m_sample->getAgentRadius()) + { + geom->deleteOffMeshConnection(nearestIndex); + } + } + else + { + // Create + if (!m_hitPosSet) + { + rcVcopy(m_hitPos, p); + m_hitPosSet = true; + } + else + { + const unsigned char area = SAMPLE_POLYAREA_JUMP; + const unsigned short flags = SAMPLE_POLYFLAGS_JUMP; + geom->addOffMeshConnection(m_hitPos, p, m_sample->getAgentRadius(), m_bidir ? 1 : 0, area, flags); + m_hitPosSet = false; + } + } + +} + +void OffMeshConnectionTool::handleToggle() +{ +} + +void OffMeshConnectionTool::handleStep() +{ +} + +void OffMeshConnectionTool::handleUpdate(const float /*dt*/) +{ +} + +void OffMeshConnectionTool::handleRender() +{ + duDebugDraw& dd = m_sample->getDebugDraw(); + const float s = m_sample->getAgentRadius(); + + if (m_hitPosSet) + duDebugDrawCross(&dd, m_hitPos[0],m_hitPos[1]+0.1f,m_hitPos[2], s, duRGBA(0,0,0,128), 2.0f); + + InputGeom* geom = m_sample->getInputGeom(); + if (geom) + geom->drawOffMeshConnections(&dd, true); +} + +void OffMeshConnectionTool::handleRenderOverlay(double* proj, double* model, int* view) +{ + GLdouble x, y, z; + + // Draw start and end point labels + if (m_hitPosSet && gluProject((GLdouble)m_hitPos[0], (GLdouble)m_hitPos[1], (GLdouble)m_hitPos[2], + model, proj, view, &x, &y, &z)) + { + imguiDrawText((int)x, (int)(y-25), IMGUI_ALIGN_CENTER, "Start", imguiRGBA(0,0,0,220)); + } + + // Tool help + const int h = view[3]; + if (!m_hitPosSet) + { + imguiDrawText(280, h-40, IMGUI_ALIGN_LEFT, "LMB: Create new connection. SHIFT+LMB: Delete existing connection, click close to start or end point.", imguiRGBA(255,255,255,192)); + } + else + { + imguiDrawText(280, h-40, IMGUI_ALIGN_LEFT, "LMB: Set connection end point and finish.", imguiRGBA(255,255,255,192)); + } +} diff --git a/r5dev/naveditor/PerfTimer.cpp b/r5dev/naveditor/PerfTimer.cpp new file mode 100644 index 00000000..d0f93f87 --- /dev/null +++ b/r5dev/naveditor/PerfTimer.cpp @@ -0,0 +1,59 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#include "NavEditor/Include/PerfTimer.h" + +#if defined(WIN32) + +// Win32 +#include + +TimeVal getPerfTime() +{ + __int64 count; + QueryPerformanceCounter((LARGE_INTEGER*)&count); + return count; +} + +int getPerfTimeUsec(const TimeVal duration) +{ + static __int64 freq = 0; + if (freq == 0) + QueryPerformanceFrequency((LARGE_INTEGER*)&freq); + return (int)(duration*1000000 / freq); +} + +#else + +// Linux, BSD, OSX + +#include + +TimeVal getPerfTime() +{ + timeval now; + gettimeofday(&now, 0); + return (TimeVal)now.tv_sec*1000000L + (TimeVal)now.tv_usec; +} + +int getPerfTimeUsec(const TimeVal duration) +{ + return (int)duration; +} + +#endif diff --git a/r5dev/naveditor/Sample.cpp b/r5dev/naveditor/Sample.cpp new file mode 100644 index 00000000..3eb53409 --- /dev/null +++ b/r5dev/naveditor/Sample.cpp @@ -0,0 +1,698 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include "NavEditor/Include/Sample.h" +#include "NavEditor/Include/InputGeom.h" +#include "Recast/Include/Recast.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "DebugUtils/Include/DetourDebugDraw.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Detour/Include/DetourNavMeshQuery.h" +#include "DetourCrowd/Include/DetourCrowd.h" +#include "NavEditor/Include/imgui.h" +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" + +#include +#include +#ifdef WIN32 +# define snprintf _snprintf +#endif + +unsigned int SampleDebugDraw::areaToCol(unsigned int area) +{ + switch(area) + { + // Ground (0) : light blue + case SAMPLE_POLYAREA_GROUND: return duRGBA(0, 120, 255, 255); + // Water : blue + case SAMPLE_POLYAREA_WATER: return duRGBA(0, 0, 255, 255); + // Road : brown + case SAMPLE_POLYAREA_ROAD: return duRGBA(50, 20, 12, 255); + // Door : cyan + case SAMPLE_POLYAREA_DOOR: return duRGBA(0, 255, 255, 255); + // Grass : green + case SAMPLE_POLYAREA_GRASS: return duRGBA(0, 255, 0, 255); + // Jump : yellow + case SAMPLE_POLYAREA_JUMP: return duRGBA(255, 255, 0, 255); + // Unexpected : red + default: return duRGBA(255, 0, 0, 255); + } +} + +Sample::Sample() : + m_geom(0), + m_navMesh(0), + m_navQuery(0), + m_crowd(0), + m_navMeshDrawFlags(DU_DRAWNAVMESH_OFFMESHCONS|DU_DRAWNAVMESH_CLOSEDLIST), + m_filterLowHangingObstacles(true), + m_filterLedgeSpans(true), + m_filterWalkableLowHeightSpans(true), + m_tool(0), + m_ctx(0) +{ + resetCommonSettings(); + m_navQuery = dtAllocNavMeshQuery(); + m_crowd = dtAllocCrowd(); + + for (int i = 0; i < MAX_TOOLS; i++) + m_toolStates[i] = 0; +} + +Sample::~Sample() +{ + dtFreeNavMeshQuery(m_navQuery); + dtFreeNavMesh(m_navMesh); + dtFreeCrowd(m_crowd); + delete m_tool; + for (int i = 0; i < MAX_TOOLS; i++) + delete m_toolStates[i]; +} + +void Sample::setTool(SampleTool* tool) +{ + delete m_tool; + m_tool = tool; + if (tool) + m_tool->init(this); +} + +void Sample::handleSettings() +{ +} + +void Sample::handleTools() +{ +} + +void Sample::handleDebugMode() +{ +} + +void Sample::handleRender() +{ + if (!m_geom) + return; + + // Draw mesh + duDebugDrawTriMesh(&m_dd, m_geom->getMesh()->getVerts(), m_geom->getMesh()->getVertCount(), + m_geom->getMesh()->getTris(), m_geom->getMesh()->getNormals(), m_geom->getMesh()->getTriCount(), 0, 1.0f); + // Draw bounds + const float* bmin = m_geom->getMeshBoundsMin(); + const float* bmax = m_geom->getMeshBoundsMax(); + duDebugDrawBoxWire(&m_dd, bmin[0],bmin[1],bmin[2], bmax[0],bmax[1],bmax[2], duRGBA(255,255,255,128), 1.0f); +} + +void Sample::handleRenderOverlay(double* /*proj*/, double* /*model*/, int* /*view*/) +{ +} + +void Sample::handleMeshChanged(InputGeom* geom) +{ + m_geom = geom; + + const BuildSettings* buildSettings = geom->getBuildSettings(); + if (buildSettings) + { + m_cellSize = buildSettings->cellSize; + m_cellHeight = buildSettings->cellHeight; + m_agentHeight = buildSettings->agentHeight; + m_agentRadius = buildSettings->agentRadius; + m_agentMaxClimb = buildSettings->agentMaxClimb; + m_agentMaxSlope = buildSettings->agentMaxSlope; + m_regionMinSize = buildSettings->regionMinSize; + m_regionMergeSize = buildSettings->regionMergeSize; + m_edgeMaxLen = buildSettings->edgeMaxLen; + m_edgeMaxError = buildSettings->edgeMaxError; + m_vertsPerPoly = buildSettings->vertsPerPoly; + m_detailSampleDist = buildSettings->detailSampleDist; + m_detailSampleMaxError = buildSettings->detailSampleMaxError; + m_partitionType = buildSettings->partitionType; + } +} + +void Sample::collectSettings(BuildSettings& settings) +{ + settings.cellSize = m_cellSize; + settings.cellHeight = m_cellHeight; + settings.agentHeight = m_agentHeight; + settings.agentRadius = m_agentRadius; + settings.agentMaxClimb = m_agentMaxClimb; + settings.agentMaxSlope = m_agentMaxSlope; + settings.regionMinSize = m_regionMinSize; + settings.regionMergeSize = m_regionMergeSize; + settings.edgeMaxLen = m_edgeMaxLen; + settings.edgeMaxError = m_edgeMaxError; + settings.vertsPerPoly = m_vertsPerPoly; + settings.detailSampleDist = m_detailSampleDist; + settings.detailSampleMaxError = m_detailSampleMaxError; + settings.partitionType = m_partitionType; +} + + +void Sample::resetCommonSettings() +{ + m_cellSize = 15.0f; + m_cellHeight = 4.0f; + m_agentHeight = 2.0f; + m_agentRadius = 0.6f; + m_agentMaxClimb = 0.9f; + m_agentMaxSlope = 45.0f; + m_regionMinSize = 8; + m_regionMergeSize = 20; + m_edgeMaxLen = 12.0f; + m_edgeMaxError = 1.3f; + m_vertsPerPoly = 6.0f; + m_detailSampleDist = 6.0f; + m_detailSampleMaxError = 1.0f; + m_partitionType = SAMPLE_PARTITION_WATERSHED; + m_count_reachability_tables = 1; +} +hulldef hulls[5] = { + {"small",8,72*0.5,70,512.0f}, + {"med_short",20,72*0.5,75,512.0f}, + {"medium",48,150*0.5,77,512.0f}, + {"large",60,235*0.5,80,960.0f}, + {"extra_large",88,235*0.5,80,960.0f}, +}; +void Sample::handleCommonSettings() +{ + bool is_human = true; + for (auto& h : hulls) + { + if (imguiButton(h.name)) + { + m_agentRadius = h.radius; + m_agentMaxClimb = h.climb_height; + m_agentHeight = h.height; + if (is_human) + m_count_reachability_tables = 4; + m_navmesh_name = h.name; + } + is_human = false; + } + imguiLabel("Rasterization"); + imguiSlider("Cell Size", &m_cellSize, 0.1f, 100.0f, 0.01f); + imguiSlider("Cell Height", &m_cellHeight, 0.1f, 100.0f, 0.01f); + + if (m_geom) + { + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + int gw = 0, gh = 0; + rcCalcGridSize(bmin, bmax, m_cellSize, &gw, &gh); + char text[64]; + snprintf(text, 64, "Voxels %d x %d", gw, gh); + imguiValue(text); + } + + imguiSeparator(); + imguiLabel("Agent"); + imguiSlider("Height", &m_agentHeight, 0.1f, 300.0f, 0.1f); + imguiSlider("Radius", &m_agentRadius, 0.0f, 100.0f, 0.1f); + imguiSlider("Max Climb", &m_agentMaxClimb, 0.1f, 120.0f, 0.1f); + imguiSlider("Max Slope", &m_agentMaxSlope, 0.0f, 90.0f, 1.0f); + + imguiSeparator(); + imguiLabel("Region"); + imguiSlider("Min Region Size", &m_regionMinSize, 0.0f, 150.0f, 1.0f); + imguiSlider("Merged Region Size", &m_regionMergeSize, 0.0f, 150.0f, 1.0f); + + imguiSeparator(); + imguiLabel("Partitioning"); + if (imguiCheck("Watershed", m_partitionType == SAMPLE_PARTITION_WATERSHED)) + m_partitionType = SAMPLE_PARTITION_WATERSHED; + if (imguiCheck("Monotone", m_partitionType == SAMPLE_PARTITION_MONOTONE)) + m_partitionType = SAMPLE_PARTITION_MONOTONE; + if (imguiCheck("Layers", m_partitionType == SAMPLE_PARTITION_LAYERS)) + m_partitionType = SAMPLE_PARTITION_LAYERS; + + imguiSeparator(); + imguiLabel("Filtering"); + if (imguiCheck("Low Hanging Obstacles", m_filterLowHangingObstacles)) + m_filterLowHangingObstacles = !m_filterLowHangingObstacles; + if (imguiCheck("Ledge Spans", m_filterLedgeSpans)) + m_filterLedgeSpans= !m_filterLedgeSpans; + if (imguiCheck("Walkable Low Height Spans", m_filterWalkableLowHeightSpans)) + m_filterWalkableLowHeightSpans = !m_filterWalkableLowHeightSpans; + + imguiSeparator(); + imguiLabel("Polygonization"); + imguiSlider("Max Edge Length", &m_edgeMaxLen, 0.0f, 50.0f, 1.0f); + imguiSlider("Max Edge Error", &m_edgeMaxError, 0.1f, 3.0f, 0.1f); + imguiSlider("Verts Per Poly", &m_vertsPerPoly, 3.0f, 12.0f, 1.0f,false); + + imguiSeparator(); + imguiLabel("Detail Mesh"); + imguiSlider("Sample Distance", &m_detailSampleDist, 0.0f, 16.0f, 1.0f); + imguiSlider("Max Sample Error", &m_detailSampleMaxError, 0.0f, 16.0f, 1.0f); + + imguiSeparator(); +} + +void Sample::handleClick(const float* s, const float* p, bool shift) +{ + if (m_tool) + m_tool->handleClick(s, p, shift); +} + +void Sample::handleToggle() +{ + if (m_tool) + m_tool->handleToggle(); +} + +void Sample::handleStep() +{ + if (m_tool) + m_tool->handleStep(); +} + +bool Sample::handleBuild() +{ + return true; +} + +void Sample::handleUpdate(const float dt) +{ + if (m_tool) + m_tool->handleUpdate(dt); + updateToolStates(dt); +} + + +void Sample::updateToolStates(const float dt) +{ + for (int i = 0; i < MAX_TOOLS; i++) + { + if (m_toolStates[i]) + m_toolStates[i]->handleUpdate(dt); + } +} + +void Sample::initToolStates(Sample* sample) +{ + for (int i = 0; i < MAX_TOOLS; i++) + { + if (m_toolStates[i]) + m_toolStates[i]->init(sample); + } +} + +void Sample::resetToolStates() +{ + for (int i = 0; i < MAX_TOOLS; i++) + { + if (m_toolStates[i]) + m_toolStates[i]->reset(); + } +} + +void Sample::renderToolStates() +{ + for (int i = 0; i < MAX_TOOLS; i++) + { + if (m_toolStates[i]) + m_toolStates[i]->handleRender(); + } +} + +void Sample::renderOverlayToolStates(double* proj, double* model, int* view) +{ + for (int i = 0; i < MAX_TOOLS; i++) + { + if (m_toolStates[i]) + m_toolStates[i]->handleRenderOverlay(proj, model, view); + } +} + +static const int NAVMESHSET_MAGIC = 'M'<<24 | 'S'<<16 | 'E'<<8 | 'T'; //'MSET'; +static const int NAVMESHSET_VERSION = 8; + +struct NavMeshSetHeader +{ + int magic; + int version; + int numTiles; + dtNavMeshParams params; + int unk0 = 0; +}; + +struct NavMeshTileHeader +{ + dtTileRef tileRef; + int dataSize; +}; + +void coord_tf_fix(float* c) +{ + std::swap(c[1], c[2]); + c[2] *= -1; +} +void coord_tf_unfix(float* c) +{ + c[2] *= -1; + std::swap(c[1], c[2]); +} +void coord_short_tf_fix(unsigned short* c) +{ + std::swap(c[1], c[2]); + c[2] = std::numeric_limits::max() - c[2]; +} +void coord_short_tf_unfix(unsigned short* c) +{ + c[2] = std::numeric_limits::max() - c[2]; + std::swap(c[1], c[2]); +} +void patch_headertf2(NavMeshSetHeader& h) +{ + coord_tf_fix(h.params.orig); +} +void unpatch_headertf2(NavMeshSetHeader& h) +{ + coord_tf_unfix(h.params.orig); +} + +void patch_tiletf2(dtMeshTile* t) +{ + coord_tf_fix(t->header->bmin); + coord_tf_fix(t->header->bmax); + + for (size_t i = 0; i < t->header->vertCount * 3; i += 3) + coord_tf_fix(t->verts + i); + for (size_t i = 0; i < t->header->detailVertCount * 3; i += 3) + coord_tf_fix(t->detailVerts + i); + for (size_t i = 0; i < t->header->polyCount; i++) + coord_tf_fix(t->polys[i].org); + //might be wrong because of coord change might break tree layout + for (size_t i = 0; i < t->header->bvNodeCount; i++) + { + coord_short_tf_fix(t->bvTree[i].bmax); + coord_short_tf_fix(t->bvTree[i].bmin); + } + for (size_t i = 0; i < t->header->offMeshConCount; i++) + { + coord_tf_fix(t->offMeshCons[i].pos); + coord_tf_fix(t->offMeshCons[i].pos + 3); + coord_tf_fix(t->offMeshCons[i].unk); + } +} +void unpatch_tiletf2(dtMeshTile* t) +{ + coord_tf_unfix(t->header->bmin); + coord_tf_unfix(t->header->bmax); + + for (size_t i = 0; i < t->header->vertCount * 3; i += 3) + coord_tf_unfix(t->verts + i); + for (size_t i = 0; i < t->header->detailVertCount * 3; i += 3) + coord_tf_unfix(t->detailVerts + i); + for (size_t i = 0; i < t->header->polyCount; i++) + coord_tf_unfix(t->polys[i].org); + //might be wrong because of coord change might break tree layout + for (size_t i = 0; i < t->header->bvNodeCount; i++) + { + coord_short_tf_unfix(t->bvTree[i].bmax); + coord_short_tf_unfix(t->bvTree[i].bmin); + } + for (size_t i = 0; i < t->header->offMeshConCount; i++) + { + coord_tf_unfix(t->offMeshCons[i].pos); + coord_tf_unfix(t->offMeshCons[i].pos+3); + coord_tf_unfix(t->offMeshCons[i].unk); + } +} +dtNavMesh* Sample::loadAll(const char* path) +{ + + char buffer[256]; + sprintf(buffer, "%s_%s.nm", path, m_navmesh_name); + + FILE* fp = fopen(buffer, "rb"); + if (!fp) return 0; + + // Read header. + NavMeshSetHeader header; + size_t readLen = fread(&header, sizeof(NavMeshSetHeader), 1, fp); + if (readLen != 1) + { + fclose(fp); + return 0; + } + if (header.magic != NAVMESHSET_MAGIC) + { + fclose(fp); + return 0; + } + if (header.version != NAVMESHSET_VERSION) + { + fclose(fp); + return 0; + } + + dtNavMesh* mesh = dtAllocNavMesh(); + if (!mesh) + { + fclose(fp); + return 0; + } + if(*is_tf2) patch_headertf2(header); + dtStatus status = mesh->init(&header.params); + if (dtStatusFailed(status)) + { + fclose(fp); + return 0; + } + + // Read tiles. + for (int i = 0; i < header.numTiles; ++i) + { + NavMeshTileHeader tileHeader; + readLen = fread(&tileHeader, sizeof(tileHeader), 1, fp); + if (readLen != 1) + { + fclose(fp); + return 0; + } + + if (!tileHeader.tileRef || !tileHeader.dataSize) + break; + + unsigned char* data = (unsigned char*)dtAlloc(tileHeader.dataSize, DT_ALLOC_PERM); + if (!data) break; + memset(data, 0, tileHeader.dataSize); + readLen = fread(data, tileHeader.dataSize, 1, fp); + if (readLen != 1) + { + dtFree(data); + fclose(fp); + return 0; + } + dtTileRef result; + mesh->addTile(data, tileHeader.dataSize, DT_TILE_FREE_DATA, tileHeader.tileRef, &result); + auto tile = const_cast(mesh->getTileByRef(result)); + if (*is_tf2) patch_tiletf2(tile); + } + + fclose(fp); + + return mesh; +} +struct link_table_data +{ + //disjoint set algo from some crappy site because i'm too lazy to think + int set_count = 0; + std::vector rank; + std::vector parent; + void init(int size) + { + rank.resize(size); + parent.resize(size); + + for (int i = 0; i < parent.size(); i++) + parent[i] = i; + } + int insert_new() + { + rank.push_back(0); + parent.push_back(set_count); + return set_count++; + } + int find(int id) + { + if (parent[id] != id) + return find(parent[id]); + return id; + } + void set_union(int x, int y) + { + int sx = find(x); + int sy = find(y); + if (sx == sy) //same set already + return; + + if (rank[sx] < rank[sy]) + parent[sx] = sy; + else if (rank[sx] > rank[sy]) + parent[sy] = sx; + else + { + parent[sy] = sx; + rank[sx] += 1; + } + } +}; +void build_link_table(dtNavMesh* mesh, link_table_data& data) +{ + //clear all labels + for (int i = 0; i < mesh->getMaxTiles(); ++i) + { + dtMeshTile* tile = mesh->getTile(i); + if (!tile || !tile->header || !tile->dataSize) continue; + auto pcount = tile->header->polyCount; + for (int j = 0; j < pcount; j++) + { + auto& poly = tile->polys[j]; + poly.link_table_idx = -1; + } + } + //first pass + std::set nlabels; + for (int i = 0; i < mesh->getMaxTiles(); ++i) + { + dtMeshTile* tile = mesh->getTile(i); + if (!tile || !tile->header || !tile->dataSize) continue; + auto pcount = tile->header->polyCount; + for (int j = 0; j < pcount; j++) + { + auto& poly = tile->polys[j]; + auto plink = poly.firstLink; + while (plink != DT_NULL_LINK) + { + auto l=tile->links[plink]; + const dtMeshTile *t; + const dtPoly *p; + mesh->getTileAndPolyByRefUnsafe(l.ref, &t, &p); + + if(p->link_table_idx != (unsigned short)-1) + nlabels.insert(p->link_table_idx); + plink = l.next; + } + if (nlabels.empty()) + { + poly.link_table_idx = data.insert_new(); + } + else + { + auto l = *nlabels.begin(); + poly.link_table_idx = l; + for (auto nl : nlabels) + data.set_union(l, nl); + } + nlabels.clear(); + } + } + //second pass + for (int i = 0; i < mesh->getMaxTiles(); ++i) + { + dtMeshTile* tile = mesh->getTile(i); + if (!tile || !tile->header || !tile->dataSize) continue; + auto pcount = tile->header->polyCount; + for (int j = 0; j < pcount; j++) + { + auto& poly = tile->polys[j]; + auto id = data.find(poly.link_table_idx); + poly.link_table_idx = id; + } + } +} +void set_reachable(std::vector& data,int count, int id1, int id2, bool value) +{ + int w = ((count + 31) / 32); + auto& cell = data[id1*w + id2 / 32]; + uint32_t value_mask = ~(1<<(id2 & 0x1f)); + if (!value) + cell = (cell & value_mask); + else + cell = (cell & value_mask) | (1 << (id2 & 0x1f)); +} +void Sample::saveAll(const char* path,dtNavMesh* mesh) +{ + if (!mesh) return; + char buffer[256]; + sprintf(buffer, "%s_%s.nm", path, m_navmesh_name); + + FILE* fp = fopen(buffer, "wb"); + if (!fp) + return; + + // Store header. + NavMeshSetHeader header; + header.magic = NAVMESHSET_MAGIC; + header.version = NAVMESHSET_VERSION; + header.numTiles = 0; + + for (int i = 0; i < mesh->getMaxTiles(); ++i) + { + dtMeshTile* tile = mesh->getTile(i); + if (!tile || !tile->header || !tile->dataSize) continue; + header.numTiles++; + } + memcpy(&header.params, mesh->getParams(), sizeof(dtNavMeshParams)); + + link_table_data link_data; + build_link_table(mesh, link_data); + int table_size = ((link_data.set_count + 31) / 32)*link_data.set_count * 32; + header.params.disjoint_poly_group_count = link_data.set_count; + header.params.reachability_table_count = m_count_reachability_tables; + header.params.reachability_table_size = table_size; + + if (*is_tf2)unpatch_headertf2(header); + fwrite(&header, sizeof(NavMeshSetHeader), 1, fp); + + // Store tiles. + for (int i = 0; i < mesh->getMaxTiles(); ++i) + { + dtMeshTile* tile = mesh->getTile(i); + if (!tile || !tile->header || !tile->dataSize) continue; + + NavMeshTileHeader tileHeader; + tileHeader.tileRef = mesh->getTileRef(tile); + tileHeader.dataSize = tile->dataSize; + fwrite(&tileHeader, sizeof(tileHeader), 1, fp); + + if (*is_tf2)unpatch_tiletf2(const_cast(tile)); + fwrite(tile->data, tile->dataSize, 1, fp); + if (*is_tf2)patch_tiletf2(const_cast(tile)); + } + + //still dont know what this thing is... + int header_sth=0; + for(int i=0;i reachability(table_size,0); + for (int i = 0; i < link_data.set_count; i++) + set_reachable(reachability, link_data.set_count, i, i, true); + for(int i=0;i< header.params.reachability_table_count;i++) + fwrite(reachability.data(), sizeof(int), (table_size /4), fp); + fclose(fp); +} diff --git a/r5dev/naveditor/SampleInterfaces.cpp b/r5dev/naveditor/SampleInterfaces.cpp new file mode 100644 index 00000000..d53fbf06 --- /dev/null +++ b/r5dev/naveditor/SampleInterfaces.cpp @@ -0,0 +1,318 @@ +#define _USE_MATH_DEFINES +#include +#include +#include +#include "NavEditor/Include/SampleInterfaces.h" +#include "Recast/Include/Recast.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "DebugUtils/Include/DetourDebugDraw.h" +#include "NavEditor/Include/PerfTimer.h" + +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" + +#ifdef WIN32 +# define snprintf _snprintf +#endif + +//////////////////////////////////////////////////////////////////////////////////////////////////// + +BuildContext::BuildContext() : + m_messageCount(0), + m_textPoolSize(0) +{ + memset(m_messages, 0, sizeof(char*) * MAX_MESSAGES); + + resetTimers(); +} + +// Virtual functions for custom implementations. +void BuildContext::doResetLog() +{ + m_messageCount = 0; + m_textPoolSize = 0; +} + +void BuildContext::doLog(const rcLogCategory category, const char* msg, const int len) +{ + if (!len) return; + if (m_messageCount >= MAX_MESSAGES) + return; + char* dst = &m_textPool[m_textPoolSize]; + int n = TEXT_POOL_SIZE - m_textPoolSize; + if (n < 2) + return; + char* cat = dst; + char* text = dst+1; + const int maxtext = n-1; + // Store category + *cat = (char)category; + // Store message + const int count = rcMin(len+1, maxtext); + memcpy(text, msg, count); + text[count-1] = '\0'; + m_textPoolSize += 1 + count; + m_messages[m_messageCount++] = dst; +} + +void BuildContext::doResetTimers() +{ + for (int i = 0; i < RC_MAX_TIMERS; ++i) + m_accTime[i] = -1; +} + +void BuildContext::doStartTimer(const rcTimerLabel label) +{ + m_startTime[label] = getPerfTime(); +} + +void BuildContext::doStopTimer(const rcTimerLabel label) +{ + const TimeVal endTime = getPerfTime(); + const TimeVal deltaTime = endTime - m_startTime[label]; + if (m_accTime[label] == -1) + m_accTime[label] = deltaTime; + else + m_accTime[label] += deltaTime; +} + +int BuildContext::doGetAccumulatedTime(const rcTimerLabel label) const +{ + return getPerfTimeUsec(m_accTime[label]); +} + +void BuildContext::dumpLog(const char* format, ...) +{ + // Print header. + va_list ap; + va_start(ap, format); + vprintf(format, ap); + va_end(ap); + printf("\n"); + + // Print messages + const int TAB_STOPS[4] = { 28, 36, 44, 52 }; + for (int i = 0; i < m_messageCount; ++i) + { + const char* msg = m_messages[i]+1; + int n = 0; + while (*msg) + { + if (*msg == '\t') + { + int count = 1; + for (int j = 0; j < 4; ++j) + { + if (n < TAB_STOPS[j]) + { + count = TAB_STOPS[j] - n; + break; + } + } + while (--count) + { + putchar(' '); + n++; + } + } + else + { + putchar(*msg); + n++; + } + msg++; + } + putchar('\n'); + } +} + +int BuildContext::getLogCount() const +{ + return m_messageCount; +} + +const char* BuildContext::getLogText(const int i) const +{ + return m_messages[i]+1; +} + +//////////////////////////////////////////////////////////////////////////////////////////////////// + +class GLCheckerTexture +{ + unsigned int m_texId; +public: + GLCheckerTexture() : m_texId(0) + { + } + + ~GLCheckerTexture() + { + if (m_texId != 0) + glDeleteTextures(1, &m_texId); + } + void bind() + { + if (m_texId == 0) + { + // Create checker pattern. + const unsigned int col0 = duRGBA(215,215,215,255); + const unsigned int col1 = duRGBA(255,255,255,255); + static const int TSIZE = 64; + unsigned int data[TSIZE*TSIZE]; + + glGenTextures(1, &m_texId); + glBindTexture(GL_TEXTURE_2D, m_texId); + + int level = 0; + int size = TSIZE; + while (size > 0) + { + for (int y = 0; y < size; ++y) + for (int x = 0; x < size; ++x) + data[x+y*size] = (x==0 || y==0) ? col0 : col1; + glTexImage2D(GL_TEXTURE_2D, level, GL_RGBA, size,size, 0, GL_RGBA, GL_UNSIGNED_BYTE, data); + size /= 2; + level++; + } + + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_NEAREST); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + } + else + { + glBindTexture(GL_TEXTURE_2D, m_texId); + } + } +}; +GLCheckerTexture g_tex; + + +void DebugDrawGL::depthMask(bool state) +{ + glDepthMask(state ? GL_TRUE : GL_FALSE); +} + +void DebugDrawGL::texture(bool state) +{ + if (state) + { + glEnable(GL_TEXTURE_2D); + g_tex.bind(); + } + else + { + glDisable(GL_TEXTURE_2D); + } +} + +void DebugDrawGL::begin(duDebugDrawPrimitives prim, float size) +{ + switch (prim) + { + case DU_DRAW_POINTS: + glPointSize(size); + glBegin(GL_POINTS); + break; + case DU_DRAW_LINES: + glLineWidth(size); + glBegin(GL_LINES); + break; + case DU_DRAW_TRIS: + glBegin(GL_TRIANGLES); + break; + case DU_DRAW_QUADS: + glBegin(GL_QUADS); + break; + }; +} + +void DebugDrawGL::vertex(const float* pos, unsigned int color) +{ + glColor4ubv((GLubyte*)&color); + glVertex3fv(pos); +} + +void DebugDrawGL::vertex(const float x, const float y, const float z, unsigned int color) +{ + glColor4ubv((GLubyte*)&color); + glVertex3f(x,y,z); +} + +void DebugDrawGL::vertex(const float* pos, unsigned int color, const float* uv) +{ + glColor4ubv((GLubyte*)&color); + glTexCoord2fv(uv); + glVertex3fv(pos); +} + +void DebugDrawGL::vertex(const float x, const float y, const float z, unsigned int color, const float u, const float v) +{ + glColor4ubv((GLubyte*)&color); + glTexCoord2f(u,v); + glVertex3f(x,y,z); +} + +void DebugDrawGL::end() +{ + glEnd(); + glLineWidth(1.0f); + glPointSize(1.0f); +} + +//////////////////////////////////////////////////////////////////////////////////////////////////// + +FileIO::FileIO() : + m_fp(0), + m_mode(-1) +{ +} + +FileIO::~FileIO() +{ + if (m_fp) fclose(m_fp); +} + +bool FileIO::openForWrite(const char* path) +{ + if (m_fp) return false; + m_fp = fopen(path, "wb"); + if (!m_fp) return false; + m_mode = 1; + return true; +} + +bool FileIO::openForRead(const char* path) +{ + if (m_fp) return false; + m_fp = fopen(path, "rb"); + if (!m_fp) return false; + m_mode = 2; + return true; +} + +bool FileIO::isWriting() const +{ + return m_mode == 1; +} + +bool FileIO::isReading() const +{ + return m_mode == 2; +} + +bool FileIO::write(const void* ptr, const size_t size) +{ + if (!m_fp || m_mode != 1) return false; + fwrite(ptr, size, 1, m_fp); + return true; +} + +bool FileIO::read(void* ptr, const size_t size) +{ + if (!m_fp || m_mode != 2) return false; + size_t readLen = fread(ptr, size, 1, m_fp); + return readLen == 1; +} + + diff --git a/r5dev/naveditor/Sample_Debug.cpp b/r5dev/naveditor/Sample_Debug.cpp new file mode 100644 index 00000000..e59a9eab --- /dev/null +++ b/r5dev/naveditor/Sample_Debug.cpp @@ -0,0 +1,387 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include "NavEditor/Include/Sample_Debug.h" +#include "NavEditor/Include/InputGeom.h" +#include "Recast/Include/Recast.h" +#include "Detour/Include/DetourNavMesh.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "DebugUtils/Include/DetourDebugDraw.h" +#include "DebugUtils/Include/RecastDump.h" +#include "NavEditor/Include/imgui.h" +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" + +#ifdef WIN32 +# define snprintf _snprintf +#endif + +/* +static int loadBin(const char* path, unsigned char** data) +{ + FILE* fp = fopen(path, "rb"); + if (!fp) return 0; + fseek(fp, 0, SEEK_END); + int size = ftell(fp); + fseek(fp, 0, SEEK_SET); + *data = new unsigned char[size]; + fread(*data, size, 1, fp); + fclose(fp); + return size; +} +*/ + +Sample_Debug::Sample_Debug() : + m_chf(0), + m_cset(0), + m_pmesh(0) +{ + resetCommonSettings(); + + // Test +/* m_chf = rcAllocCompactHeightfield(); + FileIO io; + if (!io.openForRead("test.chf")) + { + delete m_chf; + m_chf = 0; + } + else + { + if (!duReadCompactHeightfield(*m_chf, &io)) + { + delete m_chf; + m_chf = 0; + } + }*/ + +/* if (m_chf) + { + unsigned short ymin = 0xffff; + unsigned short ymax = 0; + for (int i = 0; i < m_chf->spanCount; ++i) + { + const rcCompactSpan& s = m_chf->spans[i]; + if (s.y < ymin) ymin = s.y; + if (s.y > ymax) ymax = s.y; + } + printf("ymin=%d ymax=%d\n", (int)ymin, (int)ymax); + + int maxSpans = 0; + for (int i = 0; i < m_chf->width*m_chf->height; ++i) + { + maxSpans = rcMax(maxSpans, (int)m_chf->cells[i].count); + } + printf("maxSpans = %d\n", maxSpans); + }*/ + + +/* const float orig[3] = {0,0,0}; + m_navMesh = new dtNavMesh; + m_navMesh->init(orig, 133.333f,133.333f, 2048, 4096, 4096); + + unsigned char* data = 0; + int dataSize = 0; + + // Tile_-13_-14.bin is basically just the bytes that was output by Detour. It should be loaded at X: -13 and Y: -14. + + dataSize = loadBin("Tile_-13_-13.bin", &data); + if (dataSize > 0) + { + m_navMesh->addTileAt(-13,-13, data, dataSize, true); + dtMeshHeader* header = (dtMeshHeader*)data; + vcopy(m_bmin, header->bmin); + vcopy(m_bmax, header->bmax); + } + + dataSize = loadBin("Tile_-13_-14.bin", &data); + if (dataSize > 0) + { + m_navMesh->addTileAt(-13,-14, data, dataSize, true); + } + + dataSize = loadBin("Tile_-14_-14.bin", &data); + if (dataSize > 0) + { + m_navMesh->addTileAt(-14,-14, data, dataSize, true); + } + + const float halfExtents[3] = {40,100,40}; + const float center[3] = { -1667.9491f, 135.52649f, -1680.6149f }; + dtQueryFilter filter; + m_ref = m_navMesh->findNearestPoly(center, halfExtents, &filter, 0); + + vcopy(m_halfExtents, halfExtents); + vcopy(m_center, center);*/ + + + { + m_cset = rcAllocContourSet(); + if (m_cset) + { + FileIO io; + if (io.openForRead("PathSet_TMP_NA_PathingTestAReg1_1_2_CS.rc")) + { + duReadContourSet(*m_cset, &io); + + printf("bmin=(%f,%f,%f) bmax=(%f,%f,%f)\n", + m_cset->bmin[0], m_cset->bmin[1], m_cset->bmin[2], + m_cset->bmax[0], m_cset->bmax[1], m_cset->bmax[2]); + printf("cs=%f ch=%f\n", m_cset->cs, m_cset->ch); + } + else + { + printf("could not open test.cset\n"); + } + } + else + { + printf("Could not alloc cset\n"); + } + + +/* if (m_cset) + { + m_pmesh = rcAllocPolyMesh(); + if (m_pmesh) + { + rcBuildPolyMesh(m_ctx, *m_cset, 6, *m_pmesh); + } + }*/ + } + +} + +Sample_Debug::~Sample_Debug() +{ + rcFreeCompactHeightfield(m_chf); + rcFreeContourSet(m_cset); + rcFreePolyMesh(m_pmesh); +} + +void Sample_Debug::handleSettings() +{ +} + +void Sample_Debug::handleTools() +{ +} + +void Sample_Debug::handleDebugMode() +{ +} + +void Sample_Debug::handleRender() +{ + if (m_chf) + { + duDebugDrawCompactHeightfieldRegions(&m_dd, *m_chf); +// duDebugDrawCompactHeightfieldSolid(&dd, *m_chf); + } + + if (m_navMesh) + duDebugDrawNavMesh(&m_dd, *m_navMesh, DU_DRAWNAVMESH_OFFMESHCONS); + + if (m_ref && m_navMesh) + duDebugDrawNavMeshPoly(&m_dd, *m_navMesh, m_ref, duRGBA(255,0,0,128)); + +/* float bmin[3], bmax[3]; + rcVsub(bmin, m_center, m_halfExtents); + rcVadd(bmax, m_center, m_halfExtents); + duDebugDrawBoxWire(&dd, bmin[0],bmin[1],bmin[2], bmax[0],bmax[1],bmax[2], duRGBA(255,255,255,128), 1.0f); + duDebugDrawCross(&dd, m_center[0], m_center[1], m_center[2], 1.0f, duRGBA(255,255,255,128), 2.0f);*/ + + if (m_cset) + { + duDebugDrawRawContours(&m_dd, *m_cset, 0.25f); + duDebugDrawContours(&m_dd, *m_cset); + } + + if (m_pmesh) + { + duDebugDrawPolyMesh(&m_dd, *m_pmesh); + } + + /* + dd.depthMask(false); + { + const float bmin[3] = {-32.000004f,-11.488281f,-115.343544f}; + const float cs = 0.300000f; + const float ch = 0.200000f; + const int verts[] = { + 158,46,336,0, + 157,47,331,0, + 161,53,330,0, + 162,52,335,0, + 158,46,336,0, + 154,46,339,5, + 161,46,365,5, + 171,46,385,5, + 174,46,400,5, + 177,46,404,5, + 177,46,410,5, + 183,46,416,5, + 188,49,416,5, + 193,52,411,6, + 194,53,382,6, + 188,52,376,6, + 188,57,363,6, + 174,57,349,6, + 174,60,342,6, + 168,58,336,6, + 167,59,328,6, + 162,55,324,6, + 159,53,324,5, + 152,46,328,5, + 151,46,336,5, + 154,46,339,5, + 158,46,336,0, + 160,46,340,0, + 164,52,339,0, + 168,55,343,0, + 168,50,351,0, + 182,54,364,0, + 182,47,378,0, + 188,50,383,0, + 188,49,409,0, + 183,46,409,0, + 183,46,403,0, + 180,46,399,0, + 177,46,384,0, + 165,46,359,0, + 160,46,340,0, + }; + const int nverts = sizeof(verts)/(sizeof(int)*4); + + const unsigned int colln = duRGBA(255,255,255,128); + dd.begin(DU_DRAW_LINES, 1.0f); + for (int i = 0, j = nverts-1; i < nverts; j=i++) + { + const int* va = &verts[j*4]; + const int* vb = &verts[i*4]; + dd.vertex(bmin[0]+va[0]*cs, bmin[1]+va[1]*ch+j*0.01f, bmin[2]+va[2]*cs, colln); + dd.vertex(bmin[0]+vb[0]*cs, bmin[1]+vb[1]*ch+i*0.01f, bmin[2]+vb[2]*cs, colln); + } + dd.end(); + + const unsigned int colpt = duRGBA(255,255,255,255); + dd.begin(DU_DRAW_POINTS, 3.0f); + for (int i = 0, j = nverts-1; i < nverts; j=i++) + { + const int* va = &verts[j*4]; + dd.vertex(bmin[0]+va[0]*cs, bmin[1]+va[1]*ch+j*0.01f, bmin[2]+va[2]*cs, colpt); + } + dd.end(); + + extern int triangulate(int n, const int* verts, int* indices, int* tris); + + static int indices[nverts]; + static int tris[nverts*3]; + for (int j = 0; j < nverts; ++j) + indices[j] = j; + + static int ntris = 0; + if (!ntris) + { + ntris = triangulate(nverts, verts, &indices[0], &tris[0]); + if (ntris < 0) ntris = -ntris; + } + + const unsigned int coltri = duRGBA(255,255,255,64); + dd.begin(DU_DRAW_TRIS); + for (int i = 0; i < ntris*3; ++i) + { + const int* va = &verts[indices[tris[i]]*4]; + dd.vertex(bmin[0]+va[0]*cs, bmin[1]+va[1]*ch, bmin[2]+va[2]*cs, coltri); + } + dd.end(); + + } + dd.depthMask(true);*/ +} + +void Sample_Debug::handleRenderOverlay(double* /*proj*/, double* /*model*/, int* /*view*/) +{ +} + +void Sample_Debug::handleMeshChanged(InputGeom* geom) +{ + m_geom = geom; +} + +const float* Sample_Debug::getBoundsMin() +{ + if (m_cset) + return m_cset->bmin; + if (m_chf) + return m_chf->bmin; + if (m_navMesh) + return m_bmin; + return 0; +} + +const float* Sample_Debug::getBoundsMax() +{ + if (m_cset) + return m_cset->bmax; + if (m_chf) + return m_chf->bmax; + if (m_navMesh) + return m_bmax; + return 0; +} + +void Sample_Debug::handleClick(const float* s, const float* p, bool shift) +{ + if (m_tool) + m_tool->handleClick(s, p, shift); +} + +void Sample_Debug::handleToggle() +{ + if (m_tool) + m_tool->handleToggle(); +} + +bool Sample_Debug::handleBuild() +{ + + if (m_chf) + { + rcFreeContourSet(m_cset); + m_cset = 0; + + // Create contours. + m_cset = rcAllocContourSet(); + if (!m_cset) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'cset'."); + return false; + } + if (!rcBuildContours(m_ctx, *m_chf, /*m_cfg.maxSimplificationError*/1.3f, /*m_cfg.maxEdgeLen*/12, *m_cset)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not create contours."); + return false; + } + } + + return true; +} diff --git a/r5dev/naveditor/Sample_SoloMesh.cpp b/r5dev/naveditor/Sample_SoloMesh.cpp new file mode 100644 index 00000000..c010cc6f --- /dev/null +++ b/r5dev/naveditor/Sample_SoloMesh.cpp @@ -0,0 +1,755 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" +#include "NavEditor/Include/imgui.h" +#include "NavEditor/Include/InputGeom.h" +#include "NavEditor/Include/Sample.h" +#include "NavEditor/Include/Sample_SoloMesh.h" +#include "Recast/Include/Recast.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "DebugUtils/Include/RecastDump.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Detour/Include/DetourNavMeshBuilder.h" +#include "DebugUtils/Include/DetourDebugDraw.h" +#include "NavEditor/Include/NavMeshTesterTool.h" +#include "NavEditor/Include/NavMeshPruneTool.h" +#include "NavEditor/Include/OffMeshConnectionTool.h" +#include "NavEditor/Include/ConvexVolumeTool.h" +#include "NavEditor/Include/CrowdTool.h" + +#ifdef WIN32 +# define snprintf _snprintf +#endif + + +Sample_SoloMesh::Sample_SoloMesh() : + m_keepInterResults(true), + m_totalBuildTimeMs(0), + m_triareas(0), + m_solid(0), + m_chf(0), + m_cset(0), + m_pmesh(0), + m_dmesh(0), + m_drawMode(DRAWMODE_NAVMESH) +{ + setTool(new NavMeshTesterTool); +} + +Sample_SoloMesh::~Sample_SoloMesh() +{ + cleanup(); +} + +void Sample_SoloMesh::cleanup() +{ + delete [] m_triareas; + m_triareas = 0; + rcFreeHeightField(m_solid); + m_solid = 0; + rcFreeCompactHeightfield(m_chf); + m_chf = 0; + rcFreeContourSet(m_cset); + m_cset = 0; + rcFreePolyMesh(m_pmesh); + m_pmesh = 0; + rcFreePolyMeshDetail(m_dmesh); + m_dmesh = 0; + dtFreeNavMesh(m_navMesh); + m_navMesh = 0; +} + +void Sample_SoloMesh::handleSettings() +{ + Sample::handleCommonSettings(); + + if (imguiCheck("Keep Itermediate Results", m_keepInterResults)) + m_keepInterResults = !m_keepInterResults; + + imguiSeparator(); + + imguiIndent(); + imguiIndent(); + + if (imguiButton("Save")) + { + Sample::saveAll("solo_navmesh.bin", m_navMesh); + } + + if (imguiButton("Load")) + { + dtFreeNavMesh(m_navMesh); + m_navMesh = Sample::loadAll("solo_navmesh.bin"); + m_navQuery->init(m_navMesh, 2048); + } + + imguiUnindent(); + imguiUnindent(); + + char msg[64]; + snprintf(msg, 64, "Build Time: %.1fms", m_totalBuildTimeMs); + imguiLabel(msg); + + imguiSeparator(); +} + +void Sample_SoloMesh::handleTools() +{ + int type = !m_tool ? TOOL_NONE : m_tool->type(); + + if (imguiCheck("Test Navmesh", type == TOOL_NAVMESH_TESTER)) + { + setTool(new NavMeshTesterTool); + } + if (imguiCheck("Prune Navmesh", type == TOOL_NAVMESH_PRUNE)) + { + setTool(new NavMeshPruneTool); + } + if (imguiCheck("Create Off-Mesh Connections", type == TOOL_OFFMESH_CONNECTION)) + { + setTool(new OffMeshConnectionTool); + } + if (imguiCheck("Create Convex Volumes", type == TOOL_CONVEX_VOLUME)) + { + setTool(new ConvexVolumeTool); + } + if (imguiCheck("Create Crowds", type == TOOL_CROWD)) + { + setTool(new CrowdTool); + } + + imguiSeparatorLine(); + + imguiIndent(); + + if (m_tool) + m_tool->handleMenu(); + + imguiUnindent(); + +} + +void Sample_SoloMesh::handleDebugMode() +{ + // Check which modes are valid. + bool valid[MAX_DRAWMODE]; + for (int i = 0; i < MAX_DRAWMODE; ++i) + valid[i] = false; + + if (m_geom) + { + valid[DRAWMODE_NAVMESH] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_TRANS] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_BVTREE] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_NODES] = m_navQuery != 0; + valid[DRAWMODE_NAVMESH_INVIS] = m_navMesh != 0; + valid[DRAWMODE_MESH] = true; + valid[DRAWMODE_VOXELS] = m_solid != 0; + valid[DRAWMODE_VOXELS_WALKABLE] = m_solid != 0; + valid[DRAWMODE_COMPACT] = m_chf != 0; + valid[DRAWMODE_COMPACT_DISTANCE] = m_chf != 0; + valid[DRAWMODE_COMPACT_REGIONS] = m_chf != 0; + valid[DRAWMODE_REGION_CONNECTIONS] = m_cset != 0; + valid[DRAWMODE_RAW_CONTOURS] = m_cset != 0; + valid[DRAWMODE_BOTH_CONTOURS] = m_cset != 0; + valid[DRAWMODE_CONTOURS] = m_cset != 0; + valid[DRAWMODE_POLYMESH] = m_pmesh != 0; + valid[DRAWMODE_POLYMESH_DETAIL] = m_dmesh != 0; + } + + int unavail = 0; + for (int i = 0; i < MAX_DRAWMODE; ++i) + if (!valid[i]) unavail++; + + if (unavail == MAX_DRAWMODE) + return; + + imguiLabel("Draw"); + if (imguiCheck("Input Mesh", m_drawMode == DRAWMODE_MESH, valid[DRAWMODE_MESH])) + m_drawMode = DRAWMODE_MESH; + if (imguiCheck("Navmesh", m_drawMode == DRAWMODE_NAVMESH, valid[DRAWMODE_NAVMESH])) + m_drawMode = DRAWMODE_NAVMESH; + if (imguiCheck("Navmesh Invis", m_drawMode == DRAWMODE_NAVMESH_INVIS, valid[DRAWMODE_NAVMESH_INVIS])) + m_drawMode = DRAWMODE_NAVMESH_INVIS; + if (imguiCheck("Navmesh Trans", m_drawMode == DRAWMODE_NAVMESH_TRANS, valid[DRAWMODE_NAVMESH_TRANS])) + m_drawMode = DRAWMODE_NAVMESH_TRANS; + if (imguiCheck("Navmesh BVTree", m_drawMode == DRAWMODE_NAVMESH_BVTREE, valid[DRAWMODE_NAVMESH_BVTREE])) + m_drawMode = DRAWMODE_NAVMESH_BVTREE; + if (imguiCheck("Navmesh Nodes", m_drawMode == DRAWMODE_NAVMESH_NODES, valid[DRAWMODE_NAVMESH_NODES])) + m_drawMode = DRAWMODE_NAVMESH_NODES; + if (imguiCheck("Voxels", m_drawMode == DRAWMODE_VOXELS, valid[DRAWMODE_VOXELS])) + m_drawMode = DRAWMODE_VOXELS; + if (imguiCheck("Walkable Voxels", m_drawMode == DRAWMODE_VOXELS_WALKABLE, valid[DRAWMODE_VOXELS_WALKABLE])) + m_drawMode = DRAWMODE_VOXELS_WALKABLE; + if (imguiCheck("Compact", m_drawMode == DRAWMODE_COMPACT, valid[DRAWMODE_COMPACT])) + m_drawMode = DRAWMODE_COMPACT; + if (imguiCheck("Compact Distance", m_drawMode == DRAWMODE_COMPACT_DISTANCE, valid[DRAWMODE_COMPACT_DISTANCE])) + m_drawMode = DRAWMODE_COMPACT_DISTANCE; + if (imguiCheck("Compact Regions", m_drawMode == DRAWMODE_COMPACT_REGIONS, valid[DRAWMODE_COMPACT_REGIONS])) + m_drawMode = DRAWMODE_COMPACT_REGIONS; + if (imguiCheck("Region Connections", m_drawMode == DRAWMODE_REGION_CONNECTIONS, valid[DRAWMODE_REGION_CONNECTIONS])) + m_drawMode = DRAWMODE_REGION_CONNECTIONS; + if (imguiCheck("Raw Contours", m_drawMode == DRAWMODE_RAW_CONTOURS, valid[DRAWMODE_RAW_CONTOURS])) + m_drawMode = DRAWMODE_RAW_CONTOURS; + if (imguiCheck("Both Contours", m_drawMode == DRAWMODE_BOTH_CONTOURS, valid[DRAWMODE_BOTH_CONTOURS])) + m_drawMode = DRAWMODE_BOTH_CONTOURS; + if (imguiCheck("Contours", m_drawMode == DRAWMODE_CONTOURS, valid[DRAWMODE_CONTOURS])) + m_drawMode = DRAWMODE_CONTOURS; + if (imguiCheck("Poly Mesh", m_drawMode == DRAWMODE_POLYMESH, valid[DRAWMODE_POLYMESH])) + m_drawMode = DRAWMODE_POLYMESH; + if (imguiCheck("Poly Mesh Detail", m_drawMode == DRAWMODE_POLYMESH_DETAIL, valid[DRAWMODE_POLYMESH_DETAIL])) + m_drawMode = DRAWMODE_POLYMESH_DETAIL; + + if (unavail) + { + imguiValue("Tick 'Keep Itermediate Results'"); + imguiValue("to see more debug mode options."); + } +} + +void Sample_SoloMesh::handleRender() +{ + if (!m_geom || !m_geom->getMesh()) + return; + + glEnable(GL_FOG); + glDepthMask(GL_TRUE); + + const float texScale = 1.0f / (m_cellSize * 10.0f); + + if (m_drawMode != DRAWMODE_NAVMESH_TRANS) + { + // Draw mesh + duDebugDrawTriMeshSlope(&m_dd, m_geom->getMesh()->getVerts(), m_geom->getMesh()->getVertCount(), + m_geom->getMesh()->getTris(), m_geom->getMesh()->getNormals(), m_geom->getMesh()->getTriCount(), + m_agentMaxSlope, texScale); + m_geom->drawOffMeshConnections(&m_dd); + } + + glDisable(GL_FOG); + glDepthMask(GL_FALSE); + + // Draw bounds + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + duDebugDrawBoxWire(&m_dd, bmin[0],bmin[1],bmin[2], bmax[0],bmax[1],bmax[2], duRGBA(255,255,255,128), 1.0f); + m_dd.begin(DU_DRAW_POINTS, 5.0f); + m_dd.vertex(bmin[0],bmin[1],bmin[2],duRGBA(255,255,255,128)); + m_dd.end(); + + if (m_navMesh && m_navQuery && + (m_drawMode == DRAWMODE_NAVMESH || + m_drawMode == DRAWMODE_NAVMESH_TRANS || + m_drawMode == DRAWMODE_NAVMESH_BVTREE || + m_drawMode == DRAWMODE_NAVMESH_NODES || + m_drawMode == DRAWMODE_NAVMESH_INVIS)) + { + if (m_drawMode != DRAWMODE_NAVMESH_INVIS) + duDebugDrawNavMeshWithClosedList(&m_dd, *m_navMesh, *m_navQuery, m_navMeshDrawFlags); + if (m_drawMode == DRAWMODE_NAVMESH_BVTREE) + duDebugDrawNavMeshBVTree(&m_dd, *m_navMesh); + if (m_drawMode == DRAWMODE_NAVMESH_NODES) + duDebugDrawNavMeshNodes(&m_dd, *m_navQuery); + duDebugDrawNavMeshPolysWithFlags(&m_dd, *m_navMesh, SAMPLE_POLYFLAGS_DISABLED, duRGBA(0,0,0,128)); + } + + glDepthMask(GL_TRUE); + + if (m_chf && m_drawMode == DRAWMODE_COMPACT) + duDebugDrawCompactHeightfieldSolid(&m_dd, *m_chf); + + if (m_chf && m_drawMode == DRAWMODE_COMPACT_DISTANCE) + duDebugDrawCompactHeightfieldDistance(&m_dd, *m_chf); + if (m_chf && m_drawMode == DRAWMODE_COMPACT_REGIONS) + duDebugDrawCompactHeightfieldRegions(&m_dd, *m_chf); + if (m_solid && m_drawMode == DRAWMODE_VOXELS) + { + glEnable(GL_FOG); + duDebugDrawHeightfieldSolid(&m_dd, *m_solid); + glDisable(GL_FOG); + } + if (m_solid && m_drawMode == DRAWMODE_VOXELS_WALKABLE) + { + glEnable(GL_FOG); + duDebugDrawHeightfieldWalkable(&m_dd, *m_solid); + glDisable(GL_FOG); + } + if (m_cset && m_drawMode == DRAWMODE_RAW_CONTOURS) + { + glDepthMask(GL_FALSE); + duDebugDrawRawContours(&m_dd, *m_cset); + glDepthMask(GL_TRUE); + } + if (m_cset && m_drawMode == DRAWMODE_BOTH_CONTOURS) + { + glDepthMask(GL_FALSE); + duDebugDrawRawContours(&m_dd, *m_cset, 0.5f); + duDebugDrawContours(&m_dd, *m_cset); + glDepthMask(GL_TRUE); + } + if (m_cset && m_drawMode == DRAWMODE_CONTOURS) + { + glDepthMask(GL_FALSE); + duDebugDrawContours(&m_dd, *m_cset); + glDepthMask(GL_TRUE); + } + if (m_chf && m_cset && m_drawMode == DRAWMODE_REGION_CONNECTIONS) + { + duDebugDrawCompactHeightfieldRegions(&m_dd, *m_chf); + + glDepthMask(GL_FALSE); + duDebugDrawRegionConnections(&m_dd, *m_cset); + glDepthMask(GL_TRUE); + } + if (m_pmesh && m_drawMode == DRAWMODE_POLYMESH) + { + glDepthMask(GL_FALSE); + duDebugDrawPolyMesh(&m_dd, *m_pmesh); + glDepthMask(GL_TRUE); + } + if (m_dmesh && m_drawMode == DRAWMODE_POLYMESH_DETAIL) + { + glDepthMask(GL_FALSE); + duDebugDrawPolyMeshDetail(&m_dd, *m_dmesh); + glDepthMask(GL_TRUE); + } + + m_geom->drawConvexVolumes(&m_dd); + + if (m_tool) + m_tool->handleRender(); + renderToolStates(); + + glDepthMask(GL_TRUE); +} + +void Sample_SoloMesh::handleRenderOverlay(double* proj, double* model, int* view) +{ + if (m_tool) + m_tool->handleRenderOverlay(proj, model, view); + renderOverlayToolStates(proj, model, view); +} + +void Sample_SoloMesh::handleMeshChanged(class InputGeom* geom) +{ + Sample::handleMeshChanged(geom); + + dtFreeNavMesh(m_navMesh); + m_navMesh = 0; + + if (m_tool) + { + m_tool->reset(); + m_tool->init(this); + } + resetToolStates(); + initToolStates(this); +} + + +bool Sample_SoloMesh::handleBuild() +{ + if (!m_geom || !m_geom->getMesh()) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Input mesh is not specified."); + return false; + } + + cleanup(); + + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + const float* verts = m_geom->getMesh()->getVerts(); + const int nverts = m_geom->getMesh()->getVertCount(); + const int* tris = m_geom->getMesh()->getTris(); + const int ntris = m_geom->getMesh()->getTriCount(); + + // + // Step 1. Initialize build config. + // + + // Init build configuration from GUI + memset(&m_cfg, 0, sizeof(m_cfg)); + m_cfg.cs = m_cellSize; + m_cfg.ch = m_cellHeight; + m_cfg.walkableSlopeAngle = m_agentMaxSlope; + m_cfg.walkableHeight = (int)ceilf(m_agentHeight / m_cfg.ch); + m_cfg.walkableClimb = (int)floorf(m_agentMaxClimb / m_cfg.ch); + m_cfg.walkableRadius = (int)ceilf(m_agentRadius / m_cfg.cs); + m_cfg.maxEdgeLen = (int)(m_edgeMaxLen / m_cellSize); + m_cfg.maxSimplificationError = m_edgeMaxError; + m_cfg.minRegionArea = (int)rcSqr(m_regionMinSize); // Note: area = size*size + m_cfg.mergeRegionArea = (int)rcSqr(m_regionMergeSize); // Note: area = size*size + m_cfg.maxVertsPerPoly = (int)m_vertsPerPoly; + m_cfg.detailSampleDist = m_detailSampleDist < 0.9f ? 0 : m_cellSize * m_detailSampleDist; + m_cfg.detailSampleMaxError = m_cellHeight * m_detailSampleMaxError; + + // Set the area where the navigation will be build. + // Here the bounds of the input mesh are used, but the + // area could be specified by an user defined box, etc. + rcVcopy(m_cfg.bmin, bmin); + rcVcopy(m_cfg.bmax, bmax); + rcCalcGridSize(m_cfg.bmin, m_cfg.bmax, m_cfg.cs, &m_cfg.width, &m_cfg.height); + + // Reset build times gathering. + m_ctx->resetTimers(); + + // Start the build process. + m_ctx->startTimer(RC_TIMER_TOTAL); + + m_ctx->log(RC_LOG_PROGRESS, "Building navigation:"); + m_ctx->log(RC_LOG_PROGRESS, " - %d x %d cells", m_cfg.width, m_cfg.height); + m_ctx->log(RC_LOG_PROGRESS, " - %.1fK verts, %.1fK tris", nverts/1000.0f, ntris/1000.0f); + + // + // Step 2. Rasterize input polygon soup. + // + + // Allocate voxel heightfield where we rasterize our input data to. + m_solid = rcAllocHeightfield(); + if (!m_solid) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'solid'."); + return false; + } + if (!rcCreateHeightfield(m_ctx, *m_solid, m_cfg.width, m_cfg.height, m_cfg.bmin, m_cfg.bmax, m_cfg.cs, m_cfg.ch)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not create solid heightfield."); + return false; + } + + // Allocate array that can hold triangle area types. + // If you have multiple meshes you need to process, allocate + // and array which can hold the max number of triangles you need to process. + m_triareas = new unsigned char[ntris]; + if (!m_triareas) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'm_triareas' (%d).", ntris); + return false; + } + + // Find triangles which are walkable based on their slope and rasterize them. + // If your input data is multiple meshes, you can transform them here, calculate + // the are type for each of the meshes and rasterize them. + memset(m_triareas, 0, ntris*sizeof(unsigned char)); + rcMarkWalkableTriangles(m_ctx, m_cfg.walkableSlopeAngle, verts, nverts, tris, ntris, m_triareas); + if (!rcRasterizeTriangles(m_ctx, verts, nverts, tris, m_triareas, ntris, *m_solid, m_cfg.walkableClimb)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not rasterize triangles."); + return false; + } + + if (!m_keepInterResults) + { + delete [] m_triareas; + m_triareas = 0; + } + + // + // Step 3. Filter walkables surfaces. + // + + // Once all geoemtry is rasterized, we do initial pass of filtering to + // remove unwanted overhangs caused by the conservative rasterization + // as well as filter spans where the character cannot possibly stand. + if (m_filterLowHangingObstacles) + rcFilterLowHangingWalkableObstacles(m_ctx, m_cfg.walkableClimb, *m_solid); + if (m_filterLedgeSpans) + rcFilterLedgeSpans(m_ctx, m_cfg.walkableHeight, m_cfg.walkableClimb, *m_solid); + if (m_filterWalkableLowHeightSpans) + rcFilterWalkableLowHeightSpans(m_ctx, m_cfg.walkableHeight, *m_solid); + + + // + // Step 4. Partition walkable surface to simple regions. + // + + // Compact the heightfield so that it is faster to handle from now on. + // This will result more cache coherent data as well as the neighbours + // between walkable cells will be calculated. + m_chf = rcAllocCompactHeightfield(); + if (!m_chf) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'chf'."); + return false; + } + if (!rcBuildCompactHeightfield(m_ctx, m_cfg.walkableHeight, m_cfg.walkableClimb, *m_solid, *m_chf)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build compact data."); + return false; + } + + if (!m_keepInterResults) + { + rcFreeHeightField(m_solid); + m_solid = 0; + } + + // Erode the walkable area by agent radius. + if (!rcErodeWalkableArea(m_ctx, m_cfg.walkableRadius, *m_chf)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not erode."); + return false; + } + + // (Optional) Mark areas. + const ConvexVolume* vols = m_geom->getConvexVolumes(); + for (int i = 0; i < m_geom->getConvexVolumeCount(); ++i) + rcMarkConvexPolyArea(m_ctx, vols[i].verts, vols[i].nverts, vols[i].hmin, vols[i].hmax, (unsigned char)vols[i].area, *m_chf); + + + // Partition the heightfield so that we can use simple algorithm later to triangulate the walkable areas. + // There are 3 martitioning methods, each with some pros and cons: + // 1) Watershed partitioning + // - the classic Recast partitioning + // - creates the nicest tessellation + // - usually slowest + // - partitions the heightfield into nice regions without holes or overlaps + // - the are some corner cases where this method creates produces holes and overlaps + // - holes may appear when a small obstacles is close to large open area (triangulation can handle this) + // - overlaps may occur if you have narrow spiral corridors (i.e stairs), this make triangulation to fail + // * generally the best choice if you precompute the nacmesh, use this if you have large open areas + // 2) Monotone partioning + // - fastest + // - partitions the heightfield into regions without holes and overlaps (guaranteed) + // - creates long thin polygons, which sometimes causes paths with detours + // * use this if you want fast navmesh generation + // 3) Layer partitoining + // - quite fast + // - partitions the heighfield into non-overlapping regions + // - relies on the triangulation code to cope with holes (thus slower than monotone partitioning) + // - produces better triangles than monotone partitioning + // - does not have the corner cases of watershed partitioning + // - can be slow and create a bit ugly tessellation (still better than monotone) + // if you have large open areas with small obstacles (not a problem if you use tiles) + // * good choice to use for tiled navmesh with medium and small sized tiles + + if (m_partitionType == SAMPLE_PARTITION_WATERSHED) + { + // Prepare for region partitioning, by calculating distance field along the walkable surface. + if (!rcBuildDistanceField(m_ctx, *m_chf)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build distance field."); + return false; + } + + // Partition the walkable surface into simple regions without holes. + if (!rcBuildRegions(m_ctx, *m_chf, 0, m_cfg.minRegionArea, m_cfg.mergeRegionArea)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build watershed regions."); + return false; + } + } + else if (m_partitionType == SAMPLE_PARTITION_MONOTONE) + { + // Partition the walkable surface into simple regions without holes. + // Monotone partitioning does not need distancefield. + if (!rcBuildRegionsMonotone(m_ctx, *m_chf, 0, m_cfg.minRegionArea, m_cfg.mergeRegionArea)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build monotone regions."); + return false; + } + } + else // SAMPLE_PARTITION_LAYERS + { + // Partition the walkable surface into simple regions without holes. + if (!rcBuildLayerRegions(m_ctx, *m_chf, 0, m_cfg.minRegionArea)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build layer regions."); + return false; + } + } + + // + // Step 5. Trace and simplify region contours. + // + + // Create contours. + m_cset = rcAllocContourSet(); + if (!m_cset) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'cset'."); + return false; + } + if (!rcBuildContours(m_ctx, *m_chf, m_cfg.maxSimplificationError, m_cfg.maxEdgeLen, *m_cset)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not create contours."); + return false; + } + + // + // Step 6. Build polygons mesh from contours. + // + + // Build polygon navmesh from the contours. + m_pmesh = rcAllocPolyMesh(); + if (!m_pmesh) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'pmesh'."); + return false; + } + if (!rcBuildPolyMesh(m_ctx, *m_cset, m_cfg.maxVertsPerPoly, *m_pmesh)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not triangulate contours."); + return false; + } + + // + // Step 7. Create detail mesh which allows to access approximate height on each polygon. + // + + m_dmesh = rcAllocPolyMeshDetail(); + if (!m_dmesh) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'pmdtl'."); + return false; + } + + if (!rcBuildPolyMeshDetail(m_ctx, *m_pmesh, *m_chf, m_cfg.detailSampleDist, m_cfg.detailSampleMaxError, *m_dmesh)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build detail mesh."); + return false; + } + + if (!m_keepInterResults) + { + rcFreeCompactHeightfield(m_chf); + m_chf = 0; + rcFreeContourSet(m_cset); + m_cset = 0; + } + + // At this point the navigation mesh data is ready, you can access it from m_pmesh. + // See duDebugDrawPolyMesh or dtCreateNavMeshData as examples how to access the data. + + // + // (Optional) Step 8. Create Detour data from Recast poly mesh. + // + + // The GUI may allow more max points per polygon than Detour can handle. + // Only build the detour navmesh if we do not exceed the limit. + if (m_cfg.maxVertsPerPoly <= DT_VERTS_PER_POLYGON) + { + unsigned char* navData = 0; + int navDataSize = 0; + + // Update poly flags from areas. + for (int i = 0; i < m_pmesh->npolys; ++i) + { + if (m_pmesh->areas[i] == RC_WALKABLE_AREA) + m_pmesh->areas[i] = SAMPLE_POLYAREA_GROUND; + + if (m_pmesh->areas[i] == SAMPLE_POLYAREA_GROUND || + m_pmesh->areas[i] == SAMPLE_POLYAREA_GRASS || + m_pmesh->areas[i] == SAMPLE_POLYAREA_ROAD) + { + m_pmesh->flags[i] = SAMPLE_POLYFLAGS_WALK; + } + else if (m_pmesh->areas[i] == SAMPLE_POLYAREA_WATER) + { + m_pmesh->flags[i] = SAMPLE_POLYFLAGS_SWIM; + } + else if (m_pmesh->areas[i] == SAMPLE_POLYAREA_DOOR) + { + m_pmesh->flags[i] = SAMPLE_POLYFLAGS_WALK | SAMPLE_POLYFLAGS_DOOR; + } + } + + + dtNavMeshCreateParams params; + memset(¶ms, 0, sizeof(params)); + params.verts = m_pmesh->verts; + params.vertCount = m_pmesh->nverts; + params.polys = m_pmesh->polys; + params.polyAreas = m_pmesh->areas; + params.polyFlags = m_pmesh->flags; + params.polyCount = m_pmesh->npolys; + params.nvp = m_pmesh->nvp; + params.detailMeshes = m_dmesh->meshes; + params.detailVerts = m_dmesh->verts; + params.detailVertsCount = m_dmesh->nverts; + params.detailTris = m_dmesh->tris; + params.detailTriCount = m_dmesh->ntris; + params.offMeshConVerts = m_geom->getOffMeshConnectionVerts(); + params.offMeshConRad = m_geom->getOffMeshConnectionRads(); + params.offMeshConDir = m_geom->getOffMeshConnectionDirs(); + params.offMeshConAreas = m_geom->getOffMeshConnectionAreas(); + params.offMeshConFlags = m_geom->getOffMeshConnectionFlags(); + params.offMeshConUserID = m_geom->getOffMeshConnectionId(); + params.offMeshConCount = m_geom->getOffMeshConnectionCount(); + params.walkableHeight = m_agentHeight; + params.walkableRadius = m_agentRadius; + params.walkableClimb = m_agentMaxClimb; + rcVcopy(params.bmin, m_pmesh->bmin); + rcVcopy(params.bmax, m_pmesh->bmax); + params.cs = m_cfg.cs; + params.ch = m_cfg.ch; + params.buildBvTree = true; + + if (!dtCreateNavMeshData(¶ms, &navData, &navDataSize)) + { + m_ctx->log(RC_LOG_ERROR, "Could not build Detour navmesh."); + return false; + } + + m_navMesh = dtAllocNavMesh(); + if (!m_navMesh) + { + dtFree(navData); + m_ctx->log(RC_LOG_ERROR, "Could not create Detour navmesh"); + return false; + } + + dtStatus status; + + status = m_navMesh->init(navData, navDataSize, DT_TILE_FREE_DATA); + if (dtStatusFailed(status)) + { + dtFree(navData); + m_ctx->log(RC_LOG_ERROR, "Could not init Detour navmesh"); + return false; + } + + status = m_navQuery->init(m_navMesh, 2048); + if (dtStatusFailed(status)) + { + m_ctx->log(RC_LOG_ERROR, "Could not init Detour navmesh query"); + return false; + } + } + + m_ctx->stopTimer(RC_TIMER_TOTAL); + + // Show performance stats. + duLogBuildTimes(*m_ctx, m_ctx->getAccumulatedTime(RC_TIMER_TOTAL)); + m_ctx->log(RC_LOG_PROGRESS, ">> Polymesh: %d vertices %d polygons", m_pmesh->nverts, m_pmesh->npolys); + + m_totalBuildTimeMs = m_ctx->getAccumulatedTime(RC_TIMER_TOTAL)/1000.0f; + + if (m_tool) + m_tool->init(this); + initToolStates(this); + + return true; +} diff --git a/r5dev/naveditor/Sample_TempObstacles.cpp b/r5dev/naveditor/Sample_TempObstacles.cpp new file mode 100644 index 00000000..35994359 --- /dev/null +++ b/r5dev/naveditor/Sample_TempObstacles.cpp @@ -0,0 +1,1532 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include +#include +#include +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" +#ifdef __APPLE__ +# include +#else +# include +#endif +#include "NavEditor/Include/imgui.h" +#include "NavEditor/Include/InputGeom.h" +#include "NavEditor/Include/Sample.h" +#include "NavEditor/Include/Sample_TempObstacles.h" +#include "Recast/Include/Recast.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "Detour/Include/DetourAssert.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Detour/Include/DetourNavMeshBuilder.h" +#include "DebugUtils/Include/DetourDebugDraw.h" +#include "Detour/Include/DetourCommon.h" +#include "DetourTileCache/Include/DetourTileCache.h" +#include "NavEditor/Include/NavMeshTesterTool.h" +#include "NavEditor/Include/OffMeshConnectionTool.h" +#include "NavEditor/Include/ConvexVolumeTool.h" +#include "NavEditor/Include/CrowdTool.h" +#include "Recast/Include/RecastAlloc.h" +#include "Recast/Include/RecastAssert.h" +#include "thirdparty/fastlz/fastlz.h" + +#ifdef WIN32 +# define snprintf _snprintf +#endif + + +// This value specifies how many layers (or "floors") each navmesh tile is expected to have. +static const int EXPECTED_LAYERS_PER_TILE = 4; + + +static bool isectSegAABB(const float* sp, const float* sq, + const float* amin, const float* amax, + float& tmin, float& tmax) +{ + static const float EPS = 1e-6f; + + float d[3]; + rcVsub(d, sq, sp); + tmin = 0; // set to -FLT_MAX to get first hit on line + tmax = FLT_MAX; // set to max distance ray can travel (for segment) + + // For all three slabs + for (int i = 0; i < 3; i++) + { + if (fabsf(d[i]) < EPS) + { + // Ray is parallel to slab. No hit if origin not within slab + if (sp[i] < amin[i] || sp[i] > amax[i]) + return false; + } + else + { + // Compute intersection t value of ray with near and far plane of slab + const float ood = 1.0f / d[i]; + float t1 = (amin[i] - sp[i]) * ood; + float t2 = (amax[i] - sp[i]) * ood; + // Make t1 be intersection with near plane, t2 with far plane + if (t1 > t2) rcSwap(t1, t2); + // Compute the intersection of slab intersections intervals + if (t1 > tmin) tmin = t1; + if (t2 < tmax) tmax = t2; + // Exit with no collision as soon as slab intersection becomes empty + if (tmin > tmax) return false; + } + } + + return true; +} + +static int calcLayerBufferSize(const int gridWidth, const int gridHeight) +{ + const int headerSize = dtAlign4(sizeof(dtTileCacheLayerHeader)); + const int gridSize = gridWidth * gridHeight; + return headerSize + gridSize*4; +} + + + + +struct FastLZCompressor : public dtTileCacheCompressor +{ + virtual int maxCompressedSize(const int bufferSize) + { + return (int)(bufferSize* 1.05f); + } + + virtual dtStatus compress(const unsigned char* buffer, const int bufferSize, + unsigned char* compressed, const int /*maxCompressedSize*/, int* compressedSize) + { + *compressedSize = fastlz_compress((const void *const)buffer, bufferSize, compressed); + return DT_SUCCESS; + } + + virtual dtStatus decompress(const unsigned char* compressed, const int compressedSize, + unsigned char* buffer, const int maxBufferSize, int* bufferSize) + { + *bufferSize = fastlz_decompress(compressed, compressedSize, buffer, maxBufferSize); + return *bufferSize < 0 ? DT_FAILURE : DT_SUCCESS; + } +}; + +struct LinearAllocator : public dtTileCacheAlloc +{ + unsigned char* buffer; + size_t capacity; + size_t top; + size_t high; + + LinearAllocator(const size_t cap) : buffer(0), capacity(0), top(0), high(0) + { + resize(cap); + } + + ~LinearAllocator() + { + dtFree(buffer); + } + + void resize(const size_t cap) + { + if (buffer) dtFree(buffer); + buffer = (unsigned char*)dtAlloc(cap, DT_ALLOC_PERM); + capacity = cap; + } + + virtual void reset() + { + high = dtMax(high, top); + top = 0; + } + + virtual void* alloc(const size_t size) + { + if (!buffer) + return 0; + if (top+size > capacity) + return 0; + unsigned char* mem = &buffer[top]; + top += size; + return mem; + } + + virtual void free(void* /*ptr*/) + { + // Empty + } +}; + +struct MeshProcess : public dtTileCacheMeshProcess +{ + InputGeom* m_geom; + + inline MeshProcess() : m_geom(0) + { + } + + inline void init(InputGeom* geom) + { + m_geom = geom; + } + + virtual void process(struct dtNavMeshCreateParams* params, + unsigned char* polyAreas, unsigned short* polyFlags) + { + // Update poly flags from areas. + for (int i = 0; i < params->polyCount; ++i) + { + if (polyAreas[i] == DT_TILECACHE_WALKABLE_AREA) + polyAreas[i] = SAMPLE_POLYAREA_GROUND; + + if (polyAreas[i] == SAMPLE_POLYAREA_GROUND || + polyAreas[i] == SAMPLE_POLYAREA_GRASS || + polyAreas[i] == SAMPLE_POLYAREA_ROAD) + { + polyFlags[i] = SAMPLE_POLYFLAGS_WALK; + } + else if (polyAreas[i] == SAMPLE_POLYAREA_WATER) + { + polyFlags[i] = SAMPLE_POLYFLAGS_SWIM; + } + else if (polyAreas[i] == SAMPLE_POLYAREA_DOOR) + { + polyFlags[i] = SAMPLE_POLYFLAGS_WALK | SAMPLE_POLYFLAGS_DOOR; + } + } + + // Pass in off-mesh connections. + if (m_geom) + { + params->offMeshConVerts = m_geom->getOffMeshConnectionVerts(); + params->offMeshConRad = m_geom->getOffMeshConnectionRads(); + params->offMeshConDir = m_geom->getOffMeshConnectionDirs(); + params->offMeshConAreas = m_geom->getOffMeshConnectionAreas(); + params->offMeshConFlags = m_geom->getOffMeshConnectionFlags(); + params->offMeshConUserID = m_geom->getOffMeshConnectionId(); + params->offMeshConCount = m_geom->getOffMeshConnectionCount(); + } + } +}; + + + + +static const int MAX_LAYERS = 32; + +struct TileCacheData +{ + unsigned char* data; + int dataSize; +}; + +struct RasterizationContext +{ + RasterizationContext() : + solid(0), + triareas(0), + lset(0), + chf(0), + ntiles(0) + { + memset(tiles, 0, sizeof(TileCacheData)*MAX_LAYERS); + } + + ~RasterizationContext() + { + rcFreeHeightField(solid); + delete [] triareas; + rcFreeHeightfieldLayerSet(lset); + rcFreeCompactHeightfield(chf); + for (int i = 0; i < MAX_LAYERS; ++i) + { + dtFree(tiles[i].data); + tiles[i].data = 0; + } + } + + rcHeightfield* solid; + unsigned char* triareas; + rcHeightfieldLayerSet* lset; + rcCompactHeightfield* chf; + TileCacheData tiles[MAX_LAYERS]; + int ntiles; +}; + +int Sample_TempObstacles::rasterizeTileLayers( + const int tx, const int ty, + const rcConfig& cfg, + TileCacheData* tiles, + const int maxTiles) +{ + if (!m_geom || !m_geom->getMesh() || !m_geom->getChunkyMesh()) + { + m_ctx->log(RC_LOG_ERROR, "buildTile: Input mesh is not specified."); + return 0; + } + + FastLZCompressor comp; + RasterizationContext rc; + + const float* verts = m_geom->getMesh()->getVerts(); + const int nverts = m_geom->getMesh()->getVertCount(); + const rcChunkyTriMesh* chunkyMesh = m_geom->getChunkyMesh(); + + // Tile bounds. + const float tcs = cfg.tileSize * cfg.cs; + + rcConfig tcfg; + memcpy(&tcfg, &cfg, sizeof(tcfg)); + + tcfg.bmin[0] = cfg.bmin[0] + tx*tcs; + tcfg.bmin[1] = cfg.bmin[1]; + tcfg.bmin[2] = cfg.bmin[2] + ty*tcs; + tcfg.bmax[0] = cfg.bmin[0] + (tx+1)*tcs; + tcfg.bmax[1] = cfg.bmax[1]; + tcfg.bmax[2] = cfg.bmin[2] + (ty+1)*tcs; + tcfg.bmin[0] -= tcfg.borderSize*tcfg.cs; + tcfg.bmin[2] -= tcfg.borderSize*tcfg.cs; + tcfg.bmax[0] += tcfg.borderSize*tcfg.cs; + tcfg.bmax[2] += tcfg.borderSize*tcfg.cs; + + // Allocate voxel heightfield where we rasterize our input data to. + rc.solid = rcAllocHeightfield(); + if (!rc.solid) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'solid'."); + return 0; + } + if (!rcCreateHeightfield(m_ctx, *rc.solid, tcfg.width, tcfg.height, tcfg.bmin, tcfg.bmax, tcfg.cs, tcfg.ch)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not create solid heightfield."); + return 0; + } + + // Allocate array that can hold triangle flags. + // If you have multiple meshes you need to process, allocate + // and array which can hold the max number of triangles you need to process. + rc.triareas = new unsigned char[chunkyMesh->maxTrisPerChunk]; + if (!rc.triareas) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'm_triareas' (%d).", chunkyMesh->maxTrisPerChunk); + return 0; + } + + float tbmin[2], tbmax[2]; + tbmin[0] = tcfg.bmin[0]; + tbmin[1] = tcfg.bmin[2]; + tbmax[0] = tcfg.bmax[0]; + tbmax[1] = tcfg.bmax[2]; + int cid[512];// TODO: Make grow when returning too many items. + const int ncid = rcGetChunksOverlappingRect(chunkyMesh, tbmin, tbmax, cid, 512); + if (!ncid) + { + return 0; // empty + } + + for (int i = 0; i < ncid; ++i) + { + const rcChunkyTriMeshNode& node = chunkyMesh->nodes[cid[i]]; + const int* tris = &chunkyMesh->tris[node.i*3]; + const int ntris = node.n; + + memset(rc.triareas, 0, ntris*sizeof(unsigned char)); + rcMarkWalkableTriangles(m_ctx, tcfg.walkableSlopeAngle, + verts, nverts, tris, ntris, rc.triareas); + + if (!rcRasterizeTriangles(m_ctx, verts, nverts, tris, rc.triareas, ntris, *rc.solid, tcfg.walkableClimb)) + return 0; + } + + // Once all geometry is rasterized, we do initial pass of filtering to + // remove unwanted overhangs caused by the conservative rasterization + // as well as filter spans where the character cannot possibly stand. + if (m_filterLowHangingObstacles) + rcFilterLowHangingWalkableObstacles(m_ctx, tcfg.walkableClimb, *rc.solid); + if (m_filterLedgeSpans) + rcFilterLedgeSpans(m_ctx, tcfg.walkableHeight, tcfg.walkableClimb, *rc.solid); + if (m_filterWalkableLowHeightSpans) + rcFilterWalkableLowHeightSpans(m_ctx, tcfg.walkableHeight, *rc.solid); + + + rc.chf = rcAllocCompactHeightfield(); + if (!rc.chf) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'chf'."); + return 0; + } + if (!rcBuildCompactHeightfield(m_ctx, tcfg.walkableHeight, tcfg.walkableClimb, *rc.solid, *rc.chf)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build compact data."); + return 0; + } + + // Erode the walkable area by agent radius. + if (!rcErodeWalkableArea(m_ctx, tcfg.walkableRadius, *rc.chf)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not erode."); + return 0; + } + + // (Optional) Mark areas. + const ConvexVolume* vols = m_geom->getConvexVolumes(); + for (int i = 0; i < m_geom->getConvexVolumeCount(); ++i) + { + rcMarkConvexPolyArea(m_ctx, vols[i].verts, vols[i].nverts, + vols[i].hmin, vols[i].hmax, + (unsigned char)vols[i].area, *rc.chf); + } + + rc.lset = rcAllocHeightfieldLayerSet(); + if (!rc.lset) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'lset'."); + return 0; + } + if (!rcBuildHeightfieldLayers(m_ctx, *rc.chf, tcfg.borderSize, tcfg.walkableHeight, *rc.lset)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build heighfield layers."); + return 0; + } + + rc.ntiles = 0; + for (int i = 0; i < rcMin(rc.lset->nlayers, MAX_LAYERS); ++i) + { + TileCacheData* tile = &rc.tiles[rc.ntiles++]; + const rcHeightfieldLayer* layer = &rc.lset->layers[i]; + + // Store header + dtTileCacheLayerHeader header; + header.magic = DT_TILECACHE_MAGIC; + header.version = DT_TILECACHE_VERSION; + + // Tile layer location in the navmesh. + header.tx = tx; + header.ty = ty; + header.tlayer = i; + dtVcopy(header.bmin, layer->bmin); + dtVcopy(header.bmax, layer->bmax); + + // Tile info. + header.width = (unsigned char)layer->width; + header.height = (unsigned char)layer->height; + header.minx = (unsigned char)layer->minx; + header.maxx = (unsigned char)layer->maxx; + header.miny = (unsigned char)layer->miny; + header.maxy = (unsigned char)layer->maxy; + header.hmin = (unsigned short)layer->hmin; + header.hmax = (unsigned short)layer->hmax; + + dtStatus status = dtBuildTileCacheLayer(&comp, &header, layer->heights, layer->areas, layer->cons, + &tile->data, &tile->dataSize); + if (dtStatusFailed(status)) + { + return 0; + } + } + + // Transfer ownsership of tile data from build context to the caller. + int n = 0; + for (int i = 0; i < rcMin(rc.ntiles, maxTiles); ++i) + { + tiles[n++] = rc.tiles[i]; + rc.tiles[i].data = 0; + rc.tiles[i].dataSize = 0; + } + + return n; +} + + +void drawTiles(duDebugDraw* dd, dtTileCache* tc) +{ + unsigned int fcol[6]; + float bmin[3], bmax[3]; + + for (int i = 0; i < tc->getTileCount(); ++i) + { + const dtCompressedTile* tile = tc->getTile(i); + if (!tile->header) continue; + + tc->calcTightTileBounds(tile->header, bmin, bmax); + + const unsigned int col = duIntToCol(i,64); + duCalcBoxColors(fcol, col, col); + duDebugDrawBox(dd, bmin[0],bmin[1],bmin[2], bmax[0],bmax[1],bmax[2], fcol); + } + + for (int i = 0; i < tc->getTileCount(); ++i) + { + const dtCompressedTile* tile = tc->getTile(i); + if (!tile->header) continue; + + tc->calcTightTileBounds(tile->header, bmin, bmax); + + const unsigned int col = duIntToCol(i,255); + const float pad = tc->getParams()->cs * 0.1f; + duDebugDrawBoxWire(dd, bmin[0]-pad,bmin[1]-pad,bmin[2]-pad, + bmax[0]+pad,bmax[1]+pad,bmax[2]+pad, col, 2.0f); + } + +} + +enum DrawDetailType +{ + DRAWDETAIL_AREAS, + DRAWDETAIL_REGIONS, + DRAWDETAIL_CONTOURS, + DRAWDETAIL_MESH, +}; + +void drawDetail(duDebugDraw* dd, dtTileCache* tc, const int tx, const int ty, int type) +{ + struct TileCacheBuildContext + { + inline TileCacheBuildContext(struct dtTileCacheAlloc* a) : layer(0), lcset(0), lmesh(0), alloc(a) {} + inline ~TileCacheBuildContext() { purge(); } + void purge() + { + dtFreeTileCacheLayer(alloc, layer); + layer = 0; + dtFreeTileCacheContourSet(alloc, lcset); + lcset = 0; + dtFreeTileCachePolyMesh(alloc, lmesh); + lmesh = 0; + } + struct dtTileCacheLayer* layer; + struct dtTileCacheContourSet* lcset; + struct dtTileCachePolyMesh* lmesh; + struct dtTileCacheAlloc* alloc; + }; + + dtCompressedTileRef tiles[MAX_LAYERS]; + const int ntiles = tc->getTilesAt(tx,ty,tiles,MAX_LAYERS); + + dtTileCacheAlloc* talloc = tc->getAlloc(); + dtTileCacheCompressor* tcomp = tc->getCompressor(); + const dtTileCacheParams* params = tc->getParams(); + + for (int i = 0; i < ntiles; ++i) + { + const dtCompressedTile* tile = tc->getTileByRef(tiles[i]); + + talloc->reset(); + + TileCacheBuildContext bc(talloc); + const int walkableClimbVx = (int)(params->walkableClimb / params->ch); + dtStatus status; + + // Decompress tile layer data. + status = dtDecompressTileCacheLayer(talloc, tcomp, tile->data, tile->dataSize, &bc.layer); + if (dtStatusFailed(status)) + return; + if (type == DRAWDETAIL_AREAS) + { + duDebugDrawTileCacheLayerAreas(dd, *bc.layer, params->cs, params->ch); + continue; + } + + // Build navmesh + status = dtBuildTileCacheRegions(talloc, *bc.layer, walkableClimbVx); + if (dtStatusFailed(status)) + return; + if (type == DRAWDETAIL_REGIONS) + { + duDebugDrawTileCacheLayerRegions(dd, *bc.layer, params->cs, params->ch); + continue; + } + + bc.lcset = dtAllocTileCacheContourSet(talloc); + if (!bc.lcset) + return; + status = dtBuildTileCacheContours(talloc, *bc.layer, walkableClimbVx, + params->maxSimplificationError, *bc.lcset); + if (dtStatusFailed(status)) + return; + if (type == DRAWDETAIL_CONTOURS) + { + duDebugDrawTileCacheContours(dd, *bc.lcset, tile->header->bmin, params->cs, params->ch); + continue; + } + + bc.lmesh = dtAllocTileCachePolyMesh(talloc); + if (!bc.lmesh) + return; + status = dtBuildTileCachePolyMesh(talloc, *bc.lcset, *bc.lmesh); + if (dtStatusFailed(status)) + return; + + if (type == DRAWDETAIL_MESH) + { + duDebugDrawTileCachePolyMesh(dd, *bc.lmesh, tile->header->bmin, params->cs, params->ch); + continue; + } + + } +} + + +void drawDetailOverlay(const dtTileCache* tc, const int tx, const int ty, double* proj, double* model, int* view) +{ + dtCompressedTileRef tiles[MAX_LAYERS]; + const int ntiles = tc->getTilesAt(tx,ty,tiles,MAX_LAYERS); + if (!ntiles) + return; + + const int rawSize = calcLayerBufferSize(tc->getParams()->width, tc->getParams()->height); + + char text[128]; + + for (int i = 0; i < ntiles; ++i) + { + const dtCompressedTile* tile = tc->getTileByRef(tiles[i]); + + float pos[3]; + pos[0] = (tile->header->bmin[0]+tile->header->bmax[0])/2.0f; + pos[1] = tile->header->bmin[1]; + pos[2] = (tile->header->bmin[2]+tile->header->bmax[2])/2.0f; + + GLdouble x, y, z; + if (gluProject((GLdouble)pos[0], (GLdouble)pos[1], (GLdouble)pos[2], + model, proj, view, &x, &y, &z)) + { + snprintf(text,128,"(%d,%d)/%d", tile->header->tx,tile->header->ty,tile->header->tlayer); + imguiDrawText((int)x, (int)y-25, IMGUI_ALIGN_CENTER, text, imguiRGBA(0,0,0,220)); + snprintf(text,128,"Compressed: %.1f kB", tile->dataSize/1024.0f); + imguiDrawText((int)x, (int)y-45, IMGUI_ALIGN_CENTER, text, imguiRGBA(0,0,0,128)); + snprintf(text,128,"Raw:%.1fkB", rawSize/1024.0f); + imguiDrawText((int)x, (int)y-65, IMGUI_ALIGN_CENTER, text, imguiRGBA(0,0,0,128)); + } + } +} + +dtObstacleRef hitTestObstacle(const dtTileCache* tc, const float* sp, const float* sq) +{ + float tmin = FLT_MAX; + const dtTileCacheObstacle* obmin = 0; + for (int i = 0; i < tc->getObstacleCount(); ++i) + { + const dtTileCacheObstacle* ob = tc->getObstacle(i); + if (ob->state == DT_OBSTACLE_EMPTY) + continue; + + float bmin[3], bmax[3], t0,t1; + tc->getObstacleBounds(ob, bmin,bmax); + + if (isectSegAABB(sp,sq, bmin,bmax, t0,t1)) + { + if (t0 < tmin) + { + tmin = t0; + obmin = ob; + } + } + } + return tc->getObstacleRef(obmin); +} + +void drawObstacles(duDebugDraw* dd, const dtTileCache* tc) +{ + // Draw obstacles + for (int i = 0; i < tc->getObstacleCount(); ++i) + { + const dtTileCacheObstacle* ob = tc->getObstacle(i); + if (ob->state == DT_OBSTACLE_EMPTY) continue; + float bmin[3], bmax[3]; + tc->getObstacleBounds(ob, bmin,bmax); + + unsigned int col = 0; + if (ob->state == DT_OBSTACLE_PROCESSING) + col = duRGBA(255,255,0,128); + else if (ob->state == DT_OBSTACLE_PROCESSED) + col = duRGBA(255,192,0,192); + else if (ob->state == DT_OBSTACLE_REMOVING) + col = duRGBA(220,0,0,128); + + duDebugDrawCylinder(dd, bmin[0],bmin[1],bmin[2], bmax[0],bmax[1],bmax[2], col); + duDebugDrawCylinderWire(dd, bmin[0],bmin[1],bmin[2], bmax[0],bmax[1],bmax[2], duDarkenCol(col), 2); + } +} + + + + +class TempObstacleHilightTool : public SampleTool +{ + Sample_TempObstacles* m_sample; + float m_hitPos[3]; + bool m_hitPosSet; + int m_drawType; + +public: + + TempObstacleHilightTool() : + m_sample(0), + m_hitPosSet(false), + m_drawType(DRAWDETAIL_AREAS) + { + m_hitPos[0] = m_hitPos[1] = m_hitPos[2] = 0; + } + + virtual ~TempObstacleHilightTool() + { + } + + virtual int type() { return TOOL_TILE_HIGHLIGHT; } + + virtual void init(Sample* sample) + { + m_sample = (Sample_TempObstacles*)sample; + } + + virtual void reset() {} + + virtual void handleMenu() + { + imguiLabel("Highlight Tile Cache"); + imguiValue("Click LMB to highlight a tile."); + imguiSeparator(); + if (imguiCheck("Draw Areas", m_drawType == DRAWDETAIL_AREAS)) + m_drawType = DRAWDETAIL_AREAS; + if (imguiCheck("Draw Regions", m_drawType == DRAWDETAIL_REGIONS)) + m_drawType = DRAWDETAIL_REGIONS; + if (imguiCheck("Draw Contours", m_drawType == DRAWDETAIL_CONTOURS)) + m_drawType = DRAWDETAIL_CONTOURS; + if (imguiCheck("Draw Mesh", m_drawType == DRAWDETAIL_MESH)) + m_drawType = DRAWDETAIL_MESH; + } + + virtual void handleClick(const float* /*s*/, const float* p, bool /*shift*/) + { + m_hitPosSet = true; + rcVcopy(m_hitPos,p); + } + + virtual void handleToggle() {} + + virtual void handleStep() {} + + virtual void handleUpdate(const float /*dt*/) {} + + virtual void handleRender() + { + if (m_hitPosSet && m_sample) + { + const float s = m_sample->getAgentRadius(); + glColor4ub(0,0,0,128); + glLineWidth(2.0f); + glBegin(GL_LINES); + glVertex3f(m_hitPos[0]-s,m_hitPos[1]+0.1f,m_hitPos[2]); + glVertex3f(m_hitPos[0]+s,m_hitPos[1]+0.1f,m_hitPos[2]); + glVertex3f(m_hitPos[0],m_hitPos[1]-s+0.1f,m_hitPos[2]); + glVertex3f(m_hitPos[0],m_hitPos[1]+s+0.1f,m_hitPos[2]); + glVertex3f(m_hitPos[0],m_hitPos[1]+0.1f,m_hitPos[2]-s); + glVertex3f(m_hitPos[0],m_hitPos[1]+0.1f,m_hitPos[2]+s); + glEnd(); + glLineWidth(1.0f); + + int tx=0, ty=0; + m_sample->getTilePos(m_hitPos, tx, ty); + m_sample->renderCachedTile(tx,ty,m_drawType); + } + } + + virtual void handleRenderOverlay(double* proj, double* model, int* view) + { + if (m_hitPosSet) + { + if (m_sample) + { + int tx=0, ty=0; + m_sample->getTilePos(m_hitPos, tx, ty); + m_sample->renderCachedTileOverlay(tx,ty,proj,model,view); + } + } + } +}; + + +class TempObstacleCreateTool : public SampleTool +{ + Sample_TempObstacles* m_sample; + +public: + + TempObstacleCreateTool() : m_sample(0) + { + } + + virtual ~TempObstacleCreateTool() + { + } + + virtual int type() { return TOOL_TEMP_OBSTACLE; } + + virtual void init(Sample* sample) + { + m_sample = (Sample_TempObstacles*)sample; + } + + virtual void reset() {} + + virtual void handleMenu() + { + imguiLabel("Create Temp Obstacles"); + + if (imguiButton("Remove All")) + m_sample->clearAllTempObstacles(); + + imguiSeparator(); + + imguiValue("Click LMB to create an obstacle."); + imguiValue("Shift+LMB to remove an obstacle."); + } + + virtual void handleClick(const float* s, const float* p, bool shift) + { + if (m_sample) + { + if (shift) + m_sample->removeTempObstacle(s,p); + else + m_sample->addTempObstacle(p); + } + } + + virtual void handleToggle() {} + virtual void handleStep() {} + virtual void handleUpdate(const float /*dt*/) {} + virtual void handleRender() {} + virtual void handleRenderOverlay(double* /*proj*/, double* /*model*/, int* /*view*/) { } +}; + + + + + +Sample_TempObstacles::Sample_TempObstacles() : + m_keepInterResults(false), + m_tileCache(0), + m_cacheBuildTimeMs(0), + m_cacheCompressedSize(0), + m_cacheRawSize(0), + m_cacheLayerCount(0), + m_cacheBuildMemUsage(0), + m_drawMode(DRAWMODE_NAVMESH), + m_maxTiles(0), + m_maxPolysPerTile(0), + m_tileSize(48) +{ + resetCommonSettings(); + + m_talloc = new LinearAllocator(32000); + m_tcomp = new FastLZCompressor; + m_tmproc = new MeshProcess; + + setTool(new TempObstacleCreateTool); +} + +Sample_TempObstacles::~Sample_TempObstacles() +{ + dtFreeNavMesh(m_navMesh); + m_navMesh = 0; + dtFreeTileCache(m_tileCache); +} + +void Sample_TempObstacles::handleSettings() +{ + Sample::handleCommonSettings(); + + if (imguiCheck("Keep Itermediate Results", m_keepInterResults)) + m_keepInterResults = !m_keepInterResults; + + imguiLabel("Tiling"); + imguiSlider("TileSize", &m_tileSize, 16.0f, 128.0f, 8.0f); + + int gridSize = 1; + if (m_geom) + { + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + char text[64]; + int gw = 0, gh = 0; + rcCalcGridSize(bmin, bmax, m_cellSize, &gw, &gh); + const int ts = (int)m_tileSize; + const int tw = (gw + ts-1) / ts; + const int th = (gh + ts-1) / ts; + snprintf(text, 64, "Tiles %d x %d", tw, th); + imguiValue(text); + // Max tiles and max polys affect how the tile IDs are caculated. + // There are 22 bits available for identifying a tile and a polygon. + int tileBits = rcMin((int)dtIlog2(dtNextPow2(tw*th*EXPECTED_LAYERS_PER_TILE)), 14); + if (tileBits > 14) tileBits = 14; + int polyBits = 22 - tileBits; + m_maxTiles = 1 << tileBits; + m_maxPolysPerTile = 1 << polyBits; + snprintf(text, 64, "Max Tiles %d", m_maxTiles); + imguiValue(text); + snprintf(text, 64, "Max Polys %d", m_maxPolysPerTile); + imguiValue(text); + gridSize = tw*th; + } + else + { + m_maxTiles = 0; + m_maxPolysPerTile = 0; + } + + imguiSeparator(); + + imguiLabel("Tile Cache"); + char msg[64]; + + const float compressionRatio = (float)m_cacheCompressedSize / (float)(m_cacheRawSize+1); + + snprintf(msg, 64, "Layers %d", m_cacheLayerCount); + imguiValue(msg); + snprintf(msg, 64, "Layers (per tile) %.1f", (float)m_cacheLayerCount/(float)gridSize); + imguiValue(msg); + + snprintf(msg, 64, "Memory %.1f kB / %.1f kB (%.1f%%)", m_cacheCompressedSize/1024.0f, m_cacheRawSize/1024.0f, compressionRatio*100.0f); + imguiValue(msg); + snprintf(msg, 64, "Navmesh Build Time %.1f ms", m_cacheBuildTimeMs); + imguiValue(msg); + snprintf(msg, 64, "Build Peak Mem Usage %.1f kB", m_cacheBuildMemUsage/1024.0f); + imguiValue(msg); + + imguiSeparator(); + + imguiIndent(); + imguiIndent(); + + if (imguiButton("Save")) + { + saveAll("all_tiles_tilecache.bin"); + } + + if (imguiButton("Load")) + { + dtFreeNavMesh(m_navMesh); + dtFreeTileCache(m_tileCache); + loadAll("all_tiles_tilecache.bin"); + m_navQuery->init(m_navMesh, 2048); + } + + imguiUnindent(); + imguiUnindent(); + + imguiSeparator(); +} + +void Sample_TempObstacles::handleTools() +{ + int type = !m_tool ? TOOL_NONE : m_tool->type(); + + if (imguiCheck("Test Navmesh", type == TOOL_NAVMESH_TESTER)) + { + setTool(new NavMeshTesterTool); + } + if (imguiCheck("Highlight Tile Cache", type == TOOL_TILE_HIGHLIGHT)) + { + setTool(new TempObstacleHilightTool); + } + if (imguiCheck("Create Temp Obstacles", type == TOOL_TEMP_OBSTACLE)) + { + setTool(new TempObstacleCreateTool); + } + if (imguiCheck("Create Off-Mesh Links", type == TOOL_OFFMESH_CONNECTION)) + { + setTool(new OffMeshConnectionTool); + } + if (imguiCheck("Create Convex Volumes", type == TOOL_CONVEX_VOLUME)) + { + setTool(new ConvexVolumeTool); + } + if (imguiCheck("Create Crowds", type == TOOL_CROWD)) + { + setTool(new CrowdTool); + } + + imguiSeparatorLine(); + + imguiIndent(); + + if (m_tool) + m_tool->handleMenu(); + + imguiUnindent(); +} + +void Sample_TempObstacles::handleDebugMode() +{ + // Check which modes are valid. + bool valid[MAX_DRAWMODE]; + for (int i = 0; i < MAX_DRAWMODE; ++i) + valid[i] = false; + + if (m_geom) + { + valid[DRAWMODE_NAVMESH] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_TRANS] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_BVTREE] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_NODES] = m_navQuery != 0; + valid[DRAWMODE_NAVMESH_PORTALS] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_INVIS] = m_navMesh != 0; + valid[DRAWMODE_MESH] = true; + valid[DRAWMODE_CACHE_BOUNDS] = true; + } + + int unavail = 0; + for (int i = 0; i < MAX_DRAWMODE; ++i) + if (!valid[i]) unavail++; + + if (unavail == MAX_DRAWMODE) + return; + + imguiLabel("Draw"); + if (imguiCheck("Input Mesh", m_drawMode == DRAWMODE_MESH, valid[DRAWMODE_MESH])) + m_drawMode = DRAWMODE_MESH; + if (imguiCheck("Navmesh", m_drawMode == DRAWMODE_NAVMESH, valid[DRAWMODE_NAVMESH])) + m_drawMode = DRAWMODE_NAVMESH; + if (imguiCheck("Navmesh Invis", m_drawMode == DRAWMODE_NAVMESH_INVIS, valid[DRAWMODE_NAVMESH_INVIS])) + m_drawMode = DRAWMODE_NAVMESH_INVIS; + if (imguiCheck("Navmesh Trans", m_drawMode == DRAWMODE_NAVMESH_TRANS, valid[DRAWMODE_NAVMESH_TRANS])) + m_drawMode = DRAWMODE_NAVMESH_TRANS; + if (imguiCheck("Navmesh BVTree", m_drawMode == DRAWMODE_NAVMESH_BVTREE, valid[DRAWMODE_NAVMESH_BVTREE])) + m_drawMode = DRAWMODE_NAVMESH_BVTREE; + if (imguiCheck("Navmesh Nodes", m_drawMode == DRAWMODE_NAVMESH_NODES, valid[DRAWMODE_NAVMESH_NODES])) + m_drawMode = DRAWMODE_NAVMESH_NODES; + if (imguiCheck("Navmesh Portals", m_drawMode == DRAWMODE_NAVMESH_PORTALS, valid[DRAWMODE_NAVMESH_PORTALS])) + m_drawMode = DRAWMODE_NAVMESH_PORTALS; + if (imguiCheck("Cache Bounds", m_drawMode == DRAWMODE_CACHE_BOUNDS, valid[DRAWMODE_CACHE_BOUNDS])) + m_drawMode = DRAWMODE_CACHE_BOUNDS; + + if (unavail) + { + imguiValue("Tick 'Keep Itermediate Results'"); + imguiValue("rebuild some tiles to see"); + imguiValue("more debug mode options."); + } +} + +void Sample_TempObstacles::handleRender() +{ + if (!m_geom || !m_geom->getMesh()) + return; + + const float texScale = 1.0f / (m_cellSize * 10.0f); + + // Draw mesh + if (m_drawMode != DRAWMODE_NAVMESH_TRANS) + { + // Draw mesh + duDebugDrawTriMeshSlope(&m_dd, m_geom->getMesh()->getVerts(), m_geom->getMesh()->getVertCount(), + m_geom->getMesh()->getTris(), m_geom->getMesh()->getNormals(), m_geom->getMesh()->getTriCount(), + m_agentMaxSlope, texScale); + m_geom->drawOffMeshConnections(&m_dd); + } + + if (m_tileCache && m_drawMode == DRAWMODE_CACHE_BOUNDS) + drawTiles(&m_dd, m_tileCache); + + if (m_tileCache) + drawObstacles(&m_dd, m_tileCache); + + + glDepthMask(GL_FALSE); + + // Draw bounds + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + duDebugDrawBoxWire(&m_dd, bmin[0],bmin[1],bmin[2], bmax[0],bmax[1],bmax[2], duRGBA(255,255,255,128), 1.0f); + + // Tiling grid. + int gw = 0, gh = 0; + rcCalcGridSize(bmin, bmax, m_cellSize, &gw, &gh); + const int tw = (gw + (int)m_tileSize-1) / (int)m_tileSize; + const int th = (gh + (int)m_tileSize-1) / (int)m_tileSize; + const float s = m_tileSize*m_cellSize; + duDebugDrawGridXZ(&m_dd, bmin[0],bmin[1],bmin[2], tw,th, s, duRGBA(0,0,0,64), 1.0f); + + if (m_navMesh && m_navQuery && + (m_drawMode == DRAWMODE_NAVMESH || + m_drawMode == DRAWMODE_NAVMESH_TRANS || + m_drawMode == DRAWMODE_NAVMESH_BVTREE || + m_drawMode == DRAWMODE_NAVMESH_NODES || + m_drawMode == DRAWMODE_NAVMESH_PORTALS || + m_drawMode == DRAWMODE_NAVMESH_INVIS)) + { + if (m_drawMode != DRAWMODE_NAVMESH_INVIS) + duDebugDrawNavMeshWithClosedList(&m_dd, *m_navMesh, *m_navQuery, m_navMeshDrawFlags/*|DU_DRAWNAVMESH_COLOR_TILES*/); + if (m_drawMode == DRAWMODE_NAVMESH_BVTREE) + duDebugDrawNavMeshBVTree(&m_dd, *m_navMesh); + if (m_drawMode == DRAWMODE_NAVMESH_PORTALS) + duDebugDrawNavMeshPortals(&m_dd, *m_navMesh); + if (m_drawMode == DRAWMODE_NAVMESH_NODES) + duDebugDrawNavMeshNodes(&m_dd, *m_navQuery); + duDebugDrawNavMeshPolysWithFlags(&m_dd, *m_navMesh, SAMPLE_POLYFLAGS_DISABLED, duRGBA(0,0,0,128)); + } + + + glDepthMask(GL_TRUE); + + m_geom->drawConvexVolumes(&m_dd); + + if (m_tool) + m_tool->handleRender(); + renderToolStates(); + + glDepthMask(GL_TRUE); +} + +void Sample_TempObstacles::renderCachedTile(const int tx, const int ty, const int type) +{ + if (m_tileCache) + drawDetail(&m_dd,m_tileCache,tx,ty,type); +} + +void Sample_TempObstacles::renderCachedTileOverlay(const int tx, const int ty, double* proj, double* model, int* view) +{ + if (m_tileCache) + drawDetailOverlay(m_tileCache, tx, ty, proj, model, view); +} + +void Sample_TempObstacles::handleRenderOverlay(double* proj, double* model, int* view) +{ + if (m_tool) + m_tool->handleRenderOverlay(proj, model, view); + renderOverlayToolStates(proj, model, view); + + // Stats +/* imguiDrawRect(280,10,300,100,imguiRGBA(0,0,0,64)); + + char text[64]; + int y = 110-30; + + snprintf(text,64,"Lean Data: %.1fkB", m_tileCache->getRawSize()/1024.0f); + imguiDrawText(300, y, IMGUI_ALIGN_LEFT, text, imguiRGBA(255,255,255,255)); + y -= 20; + + snprintf(text,64,"Compressed: %.1fkB (%.1f%%)", m_tileCache->getCompressedSize()/1024.0f, + m_tileCache->getRawSize() > 0 ? 100.0f*(float)m_tileCache->getCompressedSize()/(float)m_tileCache->getRawSize() : 0); + imguiDrawText(300, y, IMGUI_ALIGN_LEFT, text, imguiRGBA(255,255,255,255)); + y -= 20; + + if (m_rebuildTileCount > 0 && m_rebuildTime > 0.0f) + { + snprintf(text,64,"Changed obstacles, rebuild %d tiles: %.3f ms", m_rebuildTileCount, m_rebuildTime); + imguiDrawText(300, y, IMGUI_ALIGN_LEFT, text, imguiRGBA(255,192,0,255)); + y -= 20; + } + */ +} + +void Sample_TempObstacles::handleMeshChanged(class InputGeom* geom) +{ + Sample::handleMeshChanged(geom); + + dtFreeTileCache(m_tileCache); + m_tileCache = 0; + + dtFreeNavMesh(m_navMesh); + m_navMesh = 0; + + if (m_tool) + { + m_tool->reset(); + m_tool->init(this); + m_tmproc->init(m_geom); + } + resetToolStates(); + initToolStates(this); +} + +void Sample_TempObstacles::addTempObstacle(const float* pos) +{ + if (!m_tileCache) + return; + float p[3]; + dtVcopy(p, pos); + p[1] -= 0.5f; + m_tileCache->addObstacle(p, 1.0f, 2.0f, 0); +} + +void Sample_TempObstacles::removeTempObstacle(const float* sp, const float* sq) +{ + if (!m_tileCache) + return; + dtObstacleRef ref = hitTestObstacle(m_tileCache, sp, sq); + m_tileCache->removeObstacle(ref); +} + +void Sample_TempObstacles::clearAllTempObstacles() +{ + if (!m_tileCache) + return; + for (int i = 0; i < m_tileCache->getObstacleCount(); ++i) + { + const dtTileCacheObstacle* ob = m_tileCache->getObstacle(i); + if (ob->state == DT_OBSTACLE_EMPTY) continue; + m_tileCache->removeObstacle(m_tileCache->getObstacleRef(ob)); + } +} + +bool Sample_TempObstacles::handleBuild() +{ + dtStatus status; + + if (!m_geom || !m_geom->getMesh()) + { + m_ctx->log(RC_LOG_ERROR, "buildTiledNavigation: No vertices and triangles."); + return false; + } + + m_tmproc->init(m_geom); + + // Init cache + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + int gw = 0, gh = 0; + rcCalcGridSize(bmin, bmax, m_cellSize, &gw, &gh); + const int ts = (int)m_tileSize; + const int tw = (gw + ts-1) / ts; + const int th = (gh + ts-1) / ts; + + // Generation params. + rcConfig cfg; + memset(&cfg, 0, sizeof(cfg)); + cfg.cs = m_cellSize; + cfg.ch = m_cellHeight; + cfg.walkableSlopeAngle = m_agentMaxSlope; + cfg.walkableHeight = (int)ceilf(m_agentHeight / cfg.ch); + cfg.walkableClimb = (int)floorf(m_agentMaxClimb / cfg.ch); + cfg.walkableRadius = (int)ceilf(m_agentRadius / cfg.cs); + cfg.maxEdgeLen = (int)(m_edgeMaxLen / m_cellSize); + cfg.maxSimplificationError = m_edgeMaxError; + cfg.minRegionArea = (int)rcSqr(m_regionMinSize); // Note: area = size*size + cfg.mergeRegionArea = (int)rcSqr(m_regionMergeSize); // Note: area = size*size + cfg.maxVertsPerPoly = (int)m_vertsPerPoly; + cfg.tileSize = (int)m_tileSize; + cfg.borderSize = cfg.walkableRadius + 3; // Reserve enough padding. + cfg.width = cfg.tileSize + cfg.borderSize*2; + cfg.height = cfg.tileSize + cfg.borderSize*2; + cfg.detailSampleDist = m_detailSampleDist < 0.9f ? 0 : m_cellSize * m_detailSampleDist; + cfg.detailSampleMaxError = m_cellHeight * m_detailSampleMaxError; + rcVcopy(cfg.bmin, bmin); + rcVcopy(cfg.bmax, bmax); + + // Tile cache params. + dtTileCacheParams tcparams; + memset(&tcparams, 0, sizeof(tcparams)); + rcVcopy(tcparams.orig, bmin); + tcparams.cs = m_cellSize; + tcparams.ch = m_cellHeight; + tcparams.width = (int)m_tileSize; + tcparams.height = (int)m_tileSize; + tcparams.walkableHeight = m_agentHeight; + tcparams.walkableRadius = m_agentRadius; + tcparams.walkableClimb = m_agentMaxClimb; + tcparams.maxSimplificationError = m_edgeMaxError; + tcparams.maxTiles = tw*th*EXPECTED_LAYERS_PER_TILE; + tcparams.maxObstacles = 128; + + dtFreeTileCache(m_tileCache); + + m_tileCache = dtAllocTileCache(); + if (!m_tileCache) + { + m_ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not allocate tile cache."); + return false; + } + status = m_tileCache->init(&tcparams, m_talloc, m_tcomp, m_tmproc); + if (dtStatusFailed(status)) + { + m_ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not init tile cache."); + return false; + } + + dtFreeNavMesh(m_navMesh); + + m_navMesh = dtAllocNavMesh(); + if (!m_navMesh) + { + m_ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not allocate navmesh."); + return false; + } + + dtNavMeshParams params; + memset(¶ms, 0, sizeof(params)); + rcVcopy(params.orig, bmin); + params.tileWidth = m_tileSize*m_cellSize; + params.tileHeight = m_tileSize*m_cellSize; + params.maxTiles = m_maxTiles; + params.maxPolys = m_maxPolysPerTile; + + status = m_navMesh->init(¶ms); + if (dtStatusFailed(status)) + { + m_ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not init navmesh."); + return false; + } + + status = m_navQuery->init(m_navMesh, 2048); + if (dtStatusFailed(status)) + { + m_ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not init Detour navmesh query"); + return false; + } + + + // Preprocess tiles. + + m_ctx->resetTimers(); + + m_cacheLayerCount = 0; + m_cacheCompressedSize = 0; + m_cacheRawSize = 0; + + for (int y = 0; y < th; ++y) + { + for (int x = 0; x < tw; ++x) + { + TileCacheData tiles[MAX_LAYERS]; + memset(tiles, 0, sizeof(tiles)); + int ntiles = rasterizeTileLayers(x, y, cfg, tiles, MAX_LAYERS); + + for (int i = 0; i < ntiles; ++i) + { + TileCacheData* tile = &tiles[i]; + status = m_tileCache->addTile(tile->data, tile->dataSize, DT_COMPRESSEDTILE_FREE_DATA, 0); + if (dtStatusFailed(status)) + { + dtFree(tile->data); + tile->data = 0; + continue; + } + + m_cacheLayerCount++; + m_cacheCompressedSize += tile->dataSize; + m_cacheRawSize += calcLayerBufferSize(tcparams.width, tcparams.height); + } + } + } + + // Build initial meshes + m_ctx->startTimer(RC_TIMER_TOTAL); + for (int y = 0; y < th; ++y) + for (int x = 0; x < tw; ++x) + m_tileCache->buildNavMeshTilesAt(x,y, m_navMesh); + m_ctx->stopTimer(RC_TIMER_TOTAL); + + m_cacheBuildTimeMs = m_ctx->getAccumulatedTime(RC_TIMER_TOTAL)/1000.0f; + m_cacheBuildMemUsage = static_cast(m_talloc->high); + + + const dtNavMesh* nav = m_navMesh; + int navmeshMemUsage = 0; + for (int i = 0; i < nav->getMaxTiles(); ++i) + { + const dtMeshTile* tile = nav->getTile(i); + if (tile->header) + navmeshMemUsage += tile->dataSize; + } + printf("navmeshMemUsage = %.1f kB", navmeshMemUsage/1024.0f); + + + if (m_tool) + m_tool->init(this); + initToolStates(this); + + return true; +} + +void Sample_TempObstacles::handleUpdate(const float dt) +{ + Sample::handleUpdate(dt); + + if (!m_navMesh) + return; + if (!m_tileCache) + return; + + m_tileCache->update(dt, m_navMesh); +} + +void Sample_TempObstacles::getTilePos(const float* pos, int& tx, int& ty) +{ + if (!m_geom) return; + + const float* bmin = m_geom->getNavMeshBoundsMin(); + + const float ts = m_tileSize*m_cellSize; + tx = (int)((pos[0] - bmin[0]) / ts); + ty = (int)((pos[2] - bmin[2]) / ts); +} + +static const int TILECACHESET_MAGIC = 'T'<<24 | 'S'<<16 | 'E'<<8 | 'T'; //'TSET'; +static const int TILECACHESET_VERSION = 1; + +struct TileCacheSetHeader +{ + int magic; + int version; + int numTiles; + dtNavMeshParams meshParams; + dtTileCacheParams cacheParams; +}; + +struct TileCacheTileHeader +{ + dtCompressedTileRef tileRef; + int dataSize; +}; + +void Sample_TempObstacles::saveAll(const char* path) +{ + if (!m_tileCache) return; + + FILE* fp = fopen(path, "wb"); + if (!fp) + return; + + // Store header. + TileCacheSetHeader header; + header.magic = TILECACHESET_MAGIC; + header.version = TILECACHESET_VERSION; + header.numTiles = 0; + for (int i = 0; i < m_tileCache->getTileCount(); ++i) + { + const dtCompressedTile* tile = m_tileCache->getTile(i); + if (!tile || !tile->header || !tile->dataSize) continue; + header.numTiles++; + } + memcpy(&header.cacheParams, m_tileCache->getParams(), sizeof(dtTileCacheParams)); + memcpy(&header.meshParams, m_navMesh->getParams(), sizeof(dtNavMeshParams)); + fwrite(&header, sizeof(TileCacheSetHeader), 1, fp); + + // Store tiles. + for (int i = 0; i < m_tileCache->getTileCount(); ++i) + { + const dtCompressedTile* tile = m_tileCache->getTile(i); + if (!tile || !tile->header || !tile->dataSize) continue; + + TileCacheTileHeader tileHeader; + tileHeader.tileRef = m_tileCache->getTileRef(tile); + tileHeader.dataSize = tile->dataSize; + fwrite(&tileHeader, sizeof(tileHeader), 1, fp); + + fwrite(tile->data, tile->dataSize, 1, fp); + } + + fclose(fp); +} + +void Sample_TempObstacles::loadAll(const char* path) +{ + FILE* fp = fopen(path, "rb"); + if (!fp) return; + + // Read header. + TileCacheSetHeader header; + size_t headerReadReturnCode = fread(&header, sizeof(TileCacheSetHeader), 1, fp); + if( headerReadReturnCode != 1) + { + // Error or early EOF + fclose(fp); + return; + } + if (header.magic != TILECACHESET_MAGIC) + { + fclose(fp); + return; + } + if (header.version != TILECACHESET_VERSION) + { + fclose(fp); + return; + } + + m_navMesh = dtAllocNavMesh(); + if (!m_navMesh) + { + fclose(fp); + return; + } + dtStatus status = m_navMesh->init(&header.meshParams); + if (dtStatusFailed(status)) + { + fclose(fp); + return; + } + + m_tileCache = dtAllocTileCache(); + if (!m_tileCache) + { + fclose(fp); + return; + } + status = m_tileCache->init(&header.cacheParams, m_talloc, m_tcomp, m_tmproc); + if (dtStatusFailed(status)) + { + fclose(fp); + return; + } + + // Read tiles. + for (int i = 0; i < header.numTiles; ++i) + { + TileCacheTileHeader tileHeader; + size_t tileHeaderReadReturnCode = fread(&tileHeader, sizeof(tileHeader), 1, fp); + if( tileHeaderReadReturnCode != 1) + { + // Error or early EOF + fclose(fp); + return; + } + if (!tileHeader.tileRef || !tileHeader.dataSize) + break; + + unsigned char* data = (unsigned char*)dtAlloc(tileHeader.dataSize, DT_ALLOC_PERM); + if (!data) break; + memset(data, 0, tileHeader.dataSize); + size_t tileDataReadReturnCode = fread(data, tileHeader.dataSize, 1, fp); + if( tileDataReadReturnCode != 1) + { + // Error or early EOF + dtFree(data); + fclose(fp); + return; + } + + dtCompressedTileRef tile = 0; + dtStatus addTileStatus = m_tileCache->addTile(data, tileHeader.dataSize, DT_COMPRESSEDTILE_FREE_DATA, &tile); + if (dtStatusFailed(addTileStatus)) + { + dtFree(data); + } + + if (tile) + m_tileCache->buildNavMeshTile(tile, m_navMesh); + } + + fclose(fp); +} diff --git a/r5dev/naveditor/Sample_TileMesh.cpp b/r5dev/naveditor/Sample_TileMesh.cpp new file mode 100644 index 00000000..e80cc4ab --- /dev/null +++ b/r5dev/naveditor/Sample_TileMesh.cpp @@ -0,0 +1,1221 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" +#ifdef __APPLE__ +# include +#else +# include +#endif +#include "NavEditor/Include/imgui.h" +#include "NavEditor/Include/InputGeom.h" +#include "NavEditor/Include/Sample.h" +#include "NavEditor/Include/Sample_TileMesh.h" +#include "Recast/Include/Recast.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Detour/Include/DetourNavMeshBuilder.h" +#include "DebugUtils/Include/DetourDebugDraw.h" +#include "NavEditor/Include/NavMeshTesterTool.h" +#include "NavEditor/Include/NavMeshPruneTool.h" +#include "NavEditor/Include/OffMeshConnectionTool.h" +#include "NavEditor/Include/ConvexVolumeTool.h" +#include "NavEditor/Include/CrowdTool.h" + + +#ifdef WIN32 +# define snprintf _snprintf +#endif + + +inline unsigned int nextPow2(unsigned int v) +{ + v--; + v |= v >> 1; + v |= v >> 2; + v |= v >> 4; + v |= v >> 8; + v |= v >> 16; + v++; + return v; +} + +inline unsigned int ilog2(unsigned int v) +{ + unsigned int r; + unsigned int shift; + r = (v > 0xffff) << 4; v >>= r; + shift = (v > 0xff) << 3; v >>= shift; r |= shift; + shift = (v > 0xf) << 2; v >>= shift; r |= shift; + shift = (v > 0x3) << 1; v >>= shift; r |= shift; + r |= (v >> 1); + return r; +} + +class NavMeshTileTool : public SampleTool +{ + Sample_TileMesh* m_sample; + float m_hitPos[3]; + bool m_hitPosSet; + +public: + + NavMeshTileTool() : + m_sample(0), + m_hitPosSet(false) + { + m_hitPos[0] = m_hitPos[1] = m_hitPos[2] = 0; + } + + virtual ~NavMeshTileTool() + { + } + + virtual int type() { return TOOL_TILE_EDIT; } + + virtual void init(Sample* sample) + { + m_sample = (Sample_TileMesh*)sample; + } + + virtual void reset() {} + + virtual void handleMenu() + { + imguiLabel("Create Tiles"); + if (imguiButton("Create All")) + { + if (m_sample) + m_sample->buildAllTiles(); + } + if (imguiButton("Remove All")) + { + if (m_sample) + m_sample->removeAllTiles(); + } + } + + virtual void handleClick(const float* /*s*/, const float* p, bool shift) + { + m_hitPosSet = true; + rcVcopy(m_hitPos,p); + if (m_sample) + { + if (shift) + m_sample->removeTile(m_hitPos); + else + m_sample->buildTile(m_hitPos); + } + } + + virtual void handleToggle() {} + + virtual void handleStep() {} + + virtual void handleUpdate(const float /*dt*/) {} + + virtual void handleRender() + { + if (m_hitPosSet) + { + const float s = m_sample->getAgentRadius(); + glColor4ub(0,0,0,128); + glLineWidth(2.0f); + glBegin(GL_LINES); + glVertex3f(m_hitPos[0]-s,m_hitPos[1]+0.1f,m_hitPos[2]); + glVertex3f(m_hitPos[0]+s,m_hitPos[1]+0.1f,m_hitPos[2]); + glVertex3f(m_hitPos[0],m_hitPos[1]-s+0.1f,m_hitPos[2]); + glVertex3f(m_hitPos[0],m_hitPos[1]+s+0.1f,m_hitPos[2]); + glVertex3f(m_hitPos[0],m_hitPos[1]+0.1f,m_hitPos[2]-s); + glVertex3f(m_hitPos[0],m_hitPos[1]+0.1f,m_hitPos[2]+s); + glEnd(); + glLineWidth(1.0f); + } + } + + virtual void handleRenderOverlay(double* proj, double* model, int* view) + { + GLdouble x, y, z; + if (m_hitPosSet && gluProject((GLdouble)m_hitPos[0], (GLdouble)m_hitPos[1], (GLdouble)m_hitPos[2], + model, proj, view, &x, &y, &z)) + { + int tx=0, ty=0; + m_sample->getTilePos(m_hitPos, tx, ty); + char text[32]; + snprintf(text,32,"(%d,%d)", tx,ty); + imguiDrawText((int)x, (int)y-25, IMGUI_ALIGN_CENTER, text, imguiRGBA(0,0,0,220)); + } + + // Tool help + const int h = view[3]; + imguiDrawText(280, h-40, IMGUI_ALIGN_LEFT, "LMB: Rebuild hit tile. Shift+LMB: Clear hit tile.", imguiRGBA(255,255,255,192)); + } +}; + + + + +Sample_TileMesh::Sample_TileMesh() : + m_keepInterResults(false), + m_buildAll(true), + m_totalBuildTimeMs(0), + m_triareas(0), + m_solid(0), + m_chf(0), + m_cset(0), + m_pmesh(0), + m_dmesh(0), + m_drawMode(DRAWMODE_NAVMESH), + m_maxTiles(0), + m_maxPolysPerTile(0), + m_tileSize(32), + m_tileCol(duRGBA(0,0,0,32)), + m_tileBuildTime(0), + m_tileMemUsage(0), + m_tileTriCount(0) +{ + resetCommonSettings(); + memset(m_lastBuiltTileBmin, 0, sizeof(m_lastBuiltTileBmin)); + memset(m_lastBuiltTileBmax, 0, sizeof(m_lastBuiltTileBmax)); + + setTool(new NavMeshTileTool); +} + +Sample_TileMesh::~Sample_TileMesh() +{ + cleanup(); + dtFreeNavMesh(m_navMesh); + m_navMesh = 0; +} + +void Sample_TileMesh::cleanup() +{ + delete [] m_triareas; + m_triareas = 0; + rcFreeHeightField(m_solid); + m_solid = 0; + rcFreeCompactHeightfield(m_chf); + m_chf = 0; + rcFreeContourSet(m_cset); + m_cset = 0; + rcFreePolyMesh(m_pmesh); + m_pmesh = 0; + rcFreePolyMeshDetail(m_dmesh); + m_dmesh = 0; +} + +void Sample_TileMesh::handleSettings() +{ + Sample::handleCommonSettings(); + + if (imguiCheck("Keep Itermediate Results", m_keepInterResults)) + m_keepInterResults = !m_keepInterResults; + + if (imguiCheck("Build All Tiles", m_buildAll)) + m_buildAll = !m_buildAll; + + imguiLabel("Tiling"); + imguiSlider("TileSize", &m_tileSize, 16.0f, 1024.0f, 16.0f); + + if (m_geom) + { + char text[64]; + int gw = 0, gh = 0; + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + rcCalcGridSize(bmin, bmax, m_cellSize, &gw, &gh); + const int ts = (int)m_tileSize; + const int tw = (gw + ts-1) / ts; + const int th = (gh + ts-1) / ts; + snprintf(text, 64, "Tiles %d x %d", tw, th); + imguiValue(text); + snprintf(text, 64, "Tile Sizes %g x %g", tw*m_cellSize, th*m_cellSize); + imguiValue(text); + // Max tiles and max polys affect how the tile IDs are caculated. + // There are 22 bits available for identifying a tile and a polygon. + int tileBits = rcMin((int)ilog2(nextPow2(tw*th)), 14); + if (tileBits > 14) tileBits = 14; + int polyBits = 22 - tileBits; + m_maxTiles = 1 << tileBits; + m_maxPolysPerTile = 1 << polyBits; + snprintf(text, 64, "Max Tiles %d", m_maxTiles); + imguiValue(text); + snprintf(text, 64, "Max Polys %d", m_maxPolysPerTile); + imguiValue(text); + } + else + { + m_maxTiles = 0; + m_maxPolysPerTile = 0; + } + + imguiSeparator(); + + imguiIndent(); + imguiIndent(); + + if (imguiButton("Save")) + { + Sample::saveAll(m_model_name.c_str(), m_navMesh); + } + + if (imguiButton("Load")) + { + dtFreeNavMesh(m_navMesh); + m_navMesh = Sample::loadAll(m_model_name.c_str()); + m_navQuery->init(m_navMesh, 2048); + } + + imguiUnindent(); + imguiUnindent(); + + char msg[64]; + snprintf(msg, 64, "Build Time: %.1fms", m_totalBuildTimeMs); + imguiLabel(msg); + + imguiSeparator(); + + imguiSeparator(); + +} + +void Sample_TileMesh::handleTools() +{ + int type = !m_tool ? TOOL_NONE : m_tool->type(); + + if (imguiCheck("Test Navmesh", type == TOOL_NAVMESH_TESTER)) + { + setTool(new NavMeshTesterTool); + } + if (imguiCheck("Prune Navmesh", type == TOOL_NAVMESH_PRUNE)) + { + setTool(new NavMeshPruneTool); + } + if (imguiCheck("Create Tiles", type == TOOL_TILE_EDIT)) + { + setTool(new NavMeshTileTool); + } + if (imguiCheck("Create Off-Mesh Links", type == TOOL_OFFMESH_CONNECTION)) + { + setTool(new OffMeshConnectionTool); + } + if (imguiCheck("Create Convex Volumes", type == TOOL_CONVEX_VOLUME)) + { + setTool(new ConvexVolumeTool); + } + if (imguiCheck("Create Crowds", type == TOOL_CROWD)) + { + setTool(new CrowdTool); + } + + imguiSeparatorLine(); + + imguiIndent(); + + if (m_tool) + m_tool->handleMenu(); + + imguiUnindent(); +} + +void Sample_TileMesh::handleDebugMode() +{ + // Check which modes are valid. + bool valid[MAX_DRAWMODE]; + for (int i = 0; i < MAX_DRAWMODE; ++i) + valid[i] = false; + + if (m_geom) + { + valid[DRAWMODE_NAVMESH] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_TRANS] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_BVTREE] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_NODES] = m_navQuery != 0; + valid[DRAWMODE_NAVMESH_PORTALS] = m_navMesh != 0; + valid[DRAWMODE_NAVMESH_INVIS] = m_navMesh != 0; + valid[DRAWMODE_MESH] = true; + valid[DRAWMODE_VOXELS] = m_solid != 0; + valid[DRAWMODE_VOXELS_WALKABLE] = m_solid != 0; + valid[DRAWMODE_COMPACT] = m_chf != 0; + valid[DRAWMODE_COMPACT_DISTANCE] = m_chf != 0; + valid[DRAWMODE_COMPACT_REGIONS] = m_chf != 0; + valid[DRAWMODE_REGION_CONNECTIONS] = m_cset != 0; + valid[DRAWMODE_RAW_CONTOURS] = m_cset != 0; + valid[DRAWMODE_BOTH_CONTOURS] = m_cset != 0; + valid[DRAWMODE_CONTOURS] = m_cset != 0; + valid[DRAWMODE_POLYMESH] = m_pmesh != 0; + valid[DRAWMODE_POLYMESH_DETAIL] = m_dmesh != 0; + } + + int unavail = 0; + for (int i = 0; i < MAX_DRAWMODE; ++i) + if (!valid[i]) unavail++; + + if (unavail == MAX_DRAWMODE) + return; + + imguiLabel("Draw"); + if (imguiCheck("Input Mesh", m_drawMode == DRAWMODE_MESH, valid[DRAWMODE_MESH])) + m_drawMode = DRAWMODE_MESH; + if (imguiCheck("Navmesh", m_drawMode == DRAWMODE_NAVMESH, valid[DRAWMODE_NAVMESH])) + m_drawMode = DRAWMODE_NAVMESH; + if (imguiCheck("Navmesh Invis", m_drawMode == DRAWMODE_NAVMESH_INVIS, valid[DRAWMODE_NAVMESH_INVIS])) + m_drawMode = DRAWMODE_NAVMESH_INVIS; + if (imguiCheck("Navmesh Trans", m_drawMode == DRAWMODE_NAVMESH_TRANS, valid[DRAWMODE_NAVMESH_TRANS])) + m_drawMode = DRAWMODE_NAVMESH_TRANS; + if (imguiCheck("Navmesh BVTree", m_drawMode == DRAWMODE_NAVMESH_BVTREE, valid[DRAWMODE_NAVMESH_BVTREE])) + m_drawMode = DRAWMODE_NAVMESH_BVTREE; + if (imguiCheck("Navmesh Nodes", m_drawMode == DRAWMODE_NAVMESH_NODES, valid[DRAWMODE_NAVMESH_NODES])) + m_drawMode = DRAWMODE_NAVMESH_NODES; + if (imguiCheck("Navmesh Portals", m_drawMode == DRAWMODE_NAVMESH_PORTALS, valid[DRAWMODE_NAVMESH_PORTALS])) + m_drawMode = DRAWMODE_NAVMESH_PORTALS; + if (imguiCheck("Voxels", m_drawMode == DRAWMODE_VOXELS, valid[DRAWMODE_VOXELS])) + m_drawMode = DRAWMODE_VOXELS; + if (imguiCheck("Walkable Voxels", m_drawMode == DRAWMODE_VOXELS_WALKABLE, valid[DRAWMODE_VOXELS_WALKABLE])) + m_drawMode = DRAWMODE_VOXELS_WALKABLE; + if (imguiCheck("Compact", m_drawMode == DRAWMODE_COMPACT, valid[DRAWMODE_COMPACT])) + m_drawMode = DRAWMODE_COMPACT; + if (imguiCheck("Compact Distance", m_drawMode == DRAWMODE_COMPACT_DISTANCE, valid[DRAWMODE_COMPACT_DISTANCE])) + m_drawMode = DRAWMODE_COMPACT_DISTANCE; + if (imguiCheck("Compact Regions", m_drawMode == DRAWMODE_COMPACT_REGIONS, valid[DRAWMODE_COMPACT_REGIONS])) + m_drawMode = DRAWMODE_COMPACT_REGIONS; + if (imguiCheck("Region Connections", m_drawMode == DRAWMODE_REGION_CONNECTIONS, valid[DRAWMODE_REGION_CONNECTIONS])) + m_drawMode = DRAWMODE_REGION_CONNECTIONS; + if (imguiCheck("Raw Contours", m_drawMode == DRAWMODE_RAW_CONTOURS, valid[DRAWMODE_RAW_CONTOURS])) + m_drawMode = DRAWMODE_RAW_CONTOURS; + if (imguiCheck("Both Contours", m_drawMode == DRAWMODE_BOTH_CONTOURS, valid[DRAWMODE_BOTH_CONTOURS])) + m_drawMode = DRAWMODE_BOTH_CONTOURS; + if (imguiCheck("Contours", m_drawMode == DRAWMODE_CONTOURS, valid[DRAWMODE_CONTOURS])) + m_drawMode = DRAWMODE_CONTOURS; + if (imguiCheck("Poly Mesh", m_drawMode == DRAWMODE_POLYMESH, valid[DRAWMODE_POLYMESH])) + m_drawMode = DRAWMODE_POLYMESH; + if (imguiCheck("Poly Mesh Detail", m_drawMode == DRAWMODE_POLYMESH_DETAIL, valid[DRAWMODE_POLYMESH_DETAIL])) + m_drawMode = DRAWMODE_POLYMESH_DETAIL; + + if (unavail) + { + imguiValue("Tick 'Keep Itermediate Results'"); + imguiValue("rebuild some tiles to see"); + imguiValue("more debug mode options."); + } +} + +void Sample_TileMesh::handleRender() +{ + if (!m_geom || !m_geom->getMesh()) + return; + + const float texScale = 1.0f / (m_cellSize * 10.0f); + + // Draw mesh + if (m_drawMode != DRAWMODE_NAVMESH_TRANS) + { + // Draw mesh + duDebugDrawTriMeshSlope(&m_dd, m_geom->getMesh()->getVerts(), m_geom->getMesh()->getVertCount(), + m_geom->getMesh()->getTris(), m_geom->getMesh()->getNormals(), m_geom->getMesh()->getTriCount(), + m_agentMaxSlope, texScale); + m_geom->drawOffMeshConnections(&m_dd); + } + + glDepthMask(GL_FALSE); + + // Draw bounds + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + duDebugDrawBoxWire(&m_dd, bmin[0],bmin[1],bmin[2], bmax[0],bmax[1],bmax[2], duRGBA(255,255,255,128), 1.0f); + + // Tiling grid. + int gw = 0, gh = 0; + rcCalcGridSize(bmin, bmax, m_cellSize, &gw, &gh); + const int tw = (gw + (int)m_tileSize-1) / (int)m_tileSize; + const int th = (gh + (int)m_tileSize-1) / (int)m_tileSize; + const float s = m_tileSize*m_cellSize; + duDebugDrawGridXY_TF2(&m_dd, bmax[0],bmin[1],bmin[2], tw,th, s, duRGBA(0,0,0,64), 1.0f); + + // Draw active tile + duDebugDrawBoxWire(&m_dd, m_lastBuiltTileBmin[0],m_lastBuiltTileBmin[1],m_lastBuiltTileBmin[2], + m_lastBuiltTileBmax[0],m_lastBuiltTileBmax[1],m_lastBuiltTileBmax[2], m_tileCol, 1.0f); + + if (m_navMesh && m_navQuery && + (m_drawMode == DRAWMODE_NAVMESH || + m_drawMode == DRAWMODE_NAVMESH_TRANS || + m_drawMode == DRAWMODE_NAVMESH_BVTREE || + m_drawMode == DRAWMODE_NAVMESH_NODES || + m_drawMode == DRAWMODE_NAVMESH_PORTALS || + m_drawMode == DRAWMODE_NAVMESH_INVIS)) + { + if (m_drawMode != DRAWMODE_NAVMESH_INVIS) + duDebugDrawNavMeshWithClosedList(&m_dd, *m_navMesh, *m_navQuery, m_navMeshDrawFlags); + if (m_drawMode == DRAWMODE_NAVMESH_BVTREE) + duDebugDrawNavMeshBVTree(&m_dd, *m_navMesh); + if (m_drawMode == DRAWMODE_NAVMESH_PORTALS) + duDebugDrawNavMeshPortals(&m_dd, *m_navMesh); + if (m_drawMode == DRAWMODE_NAVMESH_NODES) + duDebugDrawNavMeshNodes(&m_dd, *m_navQuery); + duDebugDrawNavMeshPolysWithFlags(&m_dd, *m_navMesh, SAMPLE_POLYFLAGS_DISABLED, duRGBA(0,0,0,128)); + } + + + glDepthMask(GL_TRUE); + + if (m_chf && m_drawMode == DRAWMODE_COMPACT) + duDebugDrawCompactHeightfieldSolid(&m_dd, *m_chf); + + if (m_chf && m_drawMode == DRAWMODE_COMPACT_DISTANCE) + duDebugDrawCompactHeightfieldDistance(&m_dd, *m_chf); + if (m_chf && m_drawMode == DRAWMODE_COMPACT_REGIONS) + duDebugDrawCompactHeightfieldRegions(&m_dd, *m_chf); + if (m_solid && m_drawMode == DRAWMODE_VOXELS) + { + glEnable(GL_FOG); + duDebugDrawHeightfieldSolid(&m_dd, *m_solid); + glDisable(GL_FOG); + } + if (m_solid && m_drawMode == DRAWMODE_VOXELS_WALKABLE) + { + glEnable(GL_FOG); + duDebugDrawHeightfieldWalkable(&m_dd, *m_solid); + glDisable(GL_FOG); + } + + if (m_cset && m_drawMode == DRAWMODE_RAW_CONTOURS) + { + glDepthMask(GL_FALSE); + duDebugDrawRawContours(&m_dd, *m_cset); + glDepthMask(GL_TRUE); + } + + if (m_cset && m_drawMode == DRAWMODE_BOTH_CONTOURS) + { + glDepthMask(GL_FALSE); + duDebugDrawRawContours(&m_dd, *m_cset, 0.5f); + duDebugDrawContours(&m_dd, *m_cset); + glDepthMask(GL_TRUE); + } + if (m_cset && m_drawMode == DRAWMODE_CONTOURS) + { + glDepthMask(GL_FALSE); + duDebugDrawContours(&m_dd, *m_cset); + glDepthMask(GL_TRUE); + } + if (m_chf && m_cset && m_drawMode == DRAWMODE_REGION_CONNECTIONS) + { + duDebugDrawCompactHeightfieldRegions(&m_dd, *m_chf); + + glDepthMask(GL_FALSE); + duDebugDrawRegionConnections(&m_dd, *m_cset); + glDepthMask(GL_TRUE); + } + if (m_pmesh && m_drawMode == DRAWMODE_POLYMESH) + { + glDepthMask(GL_FALSE); + duDebugDrawPolyMesh(&m_dd, *m_pmesh); + glDepthMask(GL_TRUE); + } + if (m_dmesh && m_drawMode == DRAWMODE_POLYMESH_DETAIL) + { + glDepthMask(GL_FALSE); + duDebugDrawPolyMeshDetail(&m_dd, *m_dmesh); + glDepthMask(GL_TRUE); + } + + m_geom->drawConvexVolumes(&m_dd); + + if (m_tool) + m_tool->handleRender(); + renderToolStates(); + + glDepthMask(GL_TRUE); +} + +void Sample_TileMesh::handleRenderOverlay(double* proj, double* model, int* view) +{ + GLdouble x, y, z; + + // Draw start and end point labels + if (m_tileBuildTime > 0.0f && gluProject((GLdouble)(m_lastBuiltTileBmin[0]+m_lastBuiltTileBmax[0])/2, (GLdouble)(m_lastBuiltTileBmin[1]+m_lastBuiltTileBmax[1])/2, (GLdouble)(m_lastBuiltTileBmin[2]+m_lastBuiltTileBmax[2])/2, + model, proj, view, &x, &y, &z)) + { + char text[32]; + snprintf(text,32,"%.3fms / %dTris / %.1fkB", m_tileBuildTime, m_tileTriCount, m_tileMemUsage); + imguiDrawText((int)x, (int)y-25, IMGUI_ALIGN_CENTER, text, imguiRGBA(0,0,0,220)); + } + + if (m_tool) + m_tool->handleRenderOverlay(proj, model, view); + renderOverlayToolStates(proj, model, view); +} + +void Sample_TileMesh::handleMeshChanged(InputGeom* geom) +{ + Sample::handleMeshChanged(geom); + + const BuildSettings* buildSettings = geom->getBuildSettings(); + if (buildSettings && buildSettings->tileSize > 0) + m_tileSize = buildSettings->tileSize; + + cleanup(); + + dtFreeNavMesh(m_navMesh); + m_navMesh = 0; + + if (m_tool) + { + m_tool->reset(); + m_tool->init(this); + } + resetToolStates(); + initToolStates(this); +} + +bool Sample_TileMesh::handleBuild() +{ + if (!m_geom || !m_geom->getMesh()) + { + m_ctx->log(RC_LOG_ERROR, "buildTiledNavigation: No vertices and triangles."); + return false; + } + + dtFreeNavMesh(m_navMesh); + + m_navMesh = dtAllocNavMesh(); + if (!m_navMesh) + { + m_ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not allocate navmesh."); + return false; + } + + dtNavMeshParams params; + rcVcopy(params.orig, m_geom->getNavMeshBoundsMin()); + + params.orig[0] = m_geom->getNavMeshBoundsMax()[0]; + + params.tileWidth = m_tileSize*m_cellSize; + params.tileHeight = m_tileSize*m_cellSize; + params.maxTiles = m_maxTiles; + params.maxPolys = m_maxPolysPerTile; + + dtStatus status; + + status = m_navMesh->init(¶ms); + if (dtStatusFailed(status)) + { + m_ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not init navmesh."); + return false; + } + + status = m_navQuery->init(m_navMesh, 2048); + if (dtStatusFailed(status)) + { + m_ctx->log(RC_LOG_ERROR, "buildTiledNavigation: Could not init Detour navmesh query"); + return false; + } + + if (m_buildAll) + buildAllTiles(); + + if (m_tool) + m_tool->init(this); + initToolStates(this); + + return true; +} + +void Sample_TileMesh::collectSettings(BuildSettings& settings) +{ + Sample::collectSettings(settings); + + settings.tileSize = m_tileSize; +} + +void Sample_TileMesh::buildTile(const float* pos) +{ + if (!m_geom) return; + if (!m_navMesh) return; + + int tx, ty; + getTilePos(pos, tx, ty); + getTileExtents(tx, ty, m_lastBuiltTileBmin, m_lastBuiltTileBmax); + + m_tileCol = duRGBA(255,255,255,64); + + m_ctx->resetLog(); + + int dataSize = 0; + unsigned char* data = buildTileMesh(tx, ty, m_lastBuiltTileBmin, m_lastBuiltTileBmax, dataSize); + + // Remove any previous data (navmesh owns and deletes the data). + m_navMesh->removeTile(m_navMesh->getTileRefAt(tx,ty,0),0,0); + + // Add tile, or leave the location empty. + if (data) + { + // Let the navmesh own the data. + dtStatus status = m_navMesh->addTile(data,dataSize,DT_TILE_FREE_DATA,0,0); + if (dtStatusFailed(status)) + dtFree(data); + } + + m_ctx->dumpLog("Build Tile (%d,%d):", tx,ty); +} + +void Sample_TileMesh::getTileExtents(int tx, int ty, float* tmin, float* tmax) +{ + const float ts = m_tileSize * m_cellSize; + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + tmin[0] = bmax[0] - (tx+1)*ts; + //tmin[0] = bmin[0] + tx * ts; + tmin[1] = bmin[1] + ty * ts; + tmin[2] = bmin[2]; + + tmax[0] = bmax[0] - (tx)*ts; + //tmax[0] = bmin[0] + (tx + 1)*ts; + tmax[1] = bmin[1] + (ty + 1)*ts; + tmax[2] = bmax[2]; +} +void Sample_TileMesh::getTilePos(const float* pos, int& tx, int& ty) +{ + if (!m_geom) return; + + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + + const float ts = m_tileSize*m_cellSize; + tx = (int)((bmax[0]- pos[0]) / ts); + //tx = (int)((pos[0] - bmin[0]) / ts); + ty = (int)((pos[1] - bmin[1]) / ts); +} + +void Sample_TileMesh::removeTile(const float* pos) +{ + if (!m_geom) return; + if (!m_navMesh) return; + + int tx, ty; + getTilePos(pos, tx, ty); + getTileExtents(tx, ty, m_lastBuiltTileBmin, m_lastBuiltTileBmax); + + m_tileCol = duRGBA(128,32,16,64); + + m_navMesh->removeTile(m_navMesh->getTileRefAt(tx,ty,0),0,0); +} + +void Sample_TileMesh::buildAllTiles() +{ + if (!m_geom) return; + if (!m_navMesh) return; + + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + int gw = 0, gh = 0; + rcCalcGridSize(bmin, bmax, m_cellSize, &gw, &gh); + const int ts = (int)m_tileSize; + const int tw = (gw + ts-1) / ts; + const int th = (gh + ts-1) / ts; + const float tcs = m_tileSize*m_cellSize; + + + // Start the build process. + m_ctx->startTimer(RC_TIMER_TEMP); + + for (int y = 0; y < th; ++y) + { + for (int x = 0; x < tw; ++x) + { + getTileExtents(x, y, m_lastBuiltTileBmin, m_lastBuiltTileBmax); + + int dataSize = 0; + unsigned char* data = buildTileMesh(x, y, m_lastBuiltTileBmin, m_lastBuiltTileBmax, dataSize); + if (data) + { + // Remove any previous data (navmesh owns and deletes the data). + m_navMesh->removeTile(m_navMesh->getTileRefAt(x,y,0),0,0); + // Let the navmesh own the data. + dtStatus status = m_navMesh->addTile(data,dataSize,DT_TILE_FREE_DATA,0,0); + if (dtStatusFailed(status)) + dtFree(data); + } + } + } + + // Start the build process. + m_ctx->stopTimer(RC_TIMER_TEMP); + + m_totalBuildTimeMs = m_ctx->getAccumulatedTime(RC_TIMER_TEMP)/1000.0f; + +} + +//CLEANUP: copied from sample +extern hulldef hulls[4];/* = { + {"small",8,72 * 0.5,18,512.0f}, + {"med_short",20,72 * 0.5,18,512.0f}, + {"medium",48,150 * 0.5,32,512.0f}, + {"large",60,235 * 0.5,80,960.0f}, +};*/ +void Sample_TileMesh::build_n_SaveAllHulls() +{ + bool is_human = true; + for (auto& h : hulls) + { + m_agentRadius = h.radius; + m_agentMaxClimb = h.climb_height; + m_agentHeight = h.height; + if (is_human) + m_count_reachability_tables = 4; + m_navmesh_name = h.name; + is_human = false; + + handleSettings(); + handleBuild(); + Sample::saveAll(m_model_name.c_str(), m_navMesh); + } +} +void Sample_TileMesh::removeAllTiles() +{ + if (!m_geom || !m_navMesh) + return; + + const float* bmin = m_geom->getNavMeshBoundsMin(); + const float* bmax = m_geom->getNavMeshBoundsMax(); + int gw = 0, gh = 0; + rcCalcGridSize(bmin, bmax, m_cellSize, &gw, &gh); + const int ts = (int)m_tileSize; + const int tw = (gw + ts-1) / ts; + const int th = (gh + ts-1) / ts; + + for (int y = 0; y < th; ++y) + for (int x = 0; x < tw; ++x) + m_navMesh->removeTile(m_navMesh->getTileRefAt(x,y,0),0,0); +} + + +unsigned char* Sample_TileMesh::buildTileMesh(const int tx, const int ty, const float* bmin, const float* bmax, int& dataSize) +{ + if (!m_geom || !m_geom->getMesh() || !m_geom->getChunkyMesh()) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Input mesh is not specified."); + return 0; + } + + m_tileMemUsage = 0; + m_tileBuildTime = 0; + + cleanup(); + + const float* verts = m_geom->getMesh()->getVerts(); + const int nverts = m_geom->getMesh()->getVertCount(); + const int ntris = m_geom->getMesh()->getTriCount(); + const rcChunkyTriMesh* chunkyMesh = m_geom->getChunkyMesh(); + + // Init build configuration from GUI + memset(&m_cfg, 0, sizeof(m_cfg)); + m_cfg.cs = m_cellSize; + m_cfg.ch = m_cellHeight; + m_cfg.walkableSlopeAngle = m_agentMaxSlope; + m_cfg.walkableHeight = (int)ceilf(m_agentHeight / m_cfg.ch); + m_cfg.walkableClimb = (int)floorf(m_agentMaxClimb / m_cfg.ch); + m_cfg.walkableRadius = (int)ceilf(m_agentRadius / m_cfg.cs); + m_cfg.maxEdgeLen = (int)(m_edgeMaxLen / m_cellSize); + m_cfg.maxSimplificationError = m_edgeMaxError; + m_cfg.minRegionArea = (int)rcSqr(m_regionMinSize); // Note: area = size*size + m_cfg.mergeRegionArea = (int)rcSqr(m_regionMergeSize); // Note: area = size*size + m_cfg.maxVertsPerPoly = (int)m_vertsPerPoly; + m_cfg.tileSize = (int)m_tileSize; + m_cfg.borderSize = m_cfg.walkableRadius + 3; // Reserve enough padding. + m_cfg.width = m_cfg.tileSize + m_cfg.borderSize*2; + m_cfg.height = m_cfg.tileSize + m_cfg.borderSize*2; + m_cfg.detailSampleDist = m_detailSampleDist < 0.9f ? 0 : m_cellSize * m_detailSampleDist; + m_cfg.detailSampleMaxError = m_cellHeight * m_detailSampleMaxError; + + // Expand the heighfield bounding box by border size to find the extents of geometry we need to build this tile. + // + // This is done in order to make sure that the navmesh tiles connect correctly at the borders, + // and the obstacles close to the border work correctly with the dilation process. + // No polygons (or contours) will be created on the border area. + // + // IMPORTANT! + // + // :''''''''': + // : +-----+ : + // : | | : + // : | |<--- tile to build + // : | | : + // : +-----+ :<-- geometry needed + // :.........: + // + // You should use this bounding box to query your input geometry. + // + // For example if you build a navmesh for terrain, and want the navmesh tiles to match the terrain tile size + // you will need to pass in data from neighbour terrain tiles too! In a simple case, just pass in all the 8 neighbours, + // or use the bounding box below to only pass in a sliver of each of the 8 neighbours. + rcVcopy(m_cfg.bmin, bmin); + rcVcopy(m_cfg.bmax, bmax); + m_cfg.bmin[0] -= m_cfg.borderSize*m_cfg.cs; + m_cfg.bmin[1] -= m_cfg.borderSize*m_cfg.cs; + m_cfg.bmax[0] += m_cfg.borderSize*m_cfg.cs; + m_cfg.bmax[1] += m_cfg.borderSize*m_cfg.cs; + + // Reset build times gathering. + m_ctx->resetTimers(); + + // Start the build process. + m_ctx->startTimer(RC_TIMER_TOTAL); + + m_ctx->log(RC_LOG_PROGRESS, "Building navigation:"); + m_ctx->log(RC_LOG_PROGRESS, " - %d x %d cells", m_cfg.width, m_cfg.height); + m_ctx->log(RC_LOG_PROGRESS, " - %.1fK verts, %.1fK tris", nverts/1000.0f, ntris/1000.0f); + + // Allocate voxel heightfield where we rasterize our input data to. + m_solid = rcAllocHeightfield(); + if (!m_solid) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'solid'."); + return 0; + } + if (!rcCreateHeightfield(m_ctx, *m_solid, m_cfg.width, m_cfg.height, m_cfg.bmin, m_cfg.bmax, m_cfg.cs, m_cfg.ch)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not create solid heightfield."); + return 0; + } + + // Allocate array that can hold triangle flags. + // If you have multiple meshes you need to process, allocate + // and array which can hold the max number of triangles you need to process. + m_triareas = new unsigned char[chunkyMesh->maxTrisPerChunk]; + if (!m_triareas) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'm_triareas' (%d).", chunkyMesh->maxTrisPerChunk); + return 0; + } + + float tbmin[2], tbmax[2]; + tbmin[0] = m_cfg.bmin[0]; + tbmin[1] = m_cfg.bmin[1]; + tbmax[0] = m_cfg.bmax[0]; + tbmax[1] = m_cfg.bmax[1]; +#if 0 //NOTE(warmist): original algo + int cid[2048];// TODO: Make grow when returning too many items. + const int ncid = rcGetChunksOverlappingRect(chunkyMesh, tbmin, tbmax, cid, 2048); + if (!ncid) + return 0; + + m_tileTriCount = 0; + + for (int i = 0; i < ncid; ++i) + { + const rcChunkyTriMeshNode& node = chunkyMesh->nodes[cid[i]]; + const int* ctris = &chunkyMesh->tris[node.i*3]; + const int nctris = node.n; + + m_tileTriCount += nctris; + + memset(m_triareas, 0, nctris*sizeof(unsigned char)); + rcMarkWalkableTriangles(m_ctx, m_cfg.walkableSlopeAngle, + verts, nverts, ctris, nctris, m_triareas); + + if (!rcRasterizeTriangles(m_ctx, verts, nverts, ctris, m_triareas, nctris, *m_solid, m_cfg.walkableClimb)) + return 0; + } +#else //NOTE(warmist): algo with limited return but can be reinvoked to continue the query + int cid[1024];//NOTE: we don't grow it but we reuse it (e.g. like a yieldable function or iterator or sth) + int current_node = 0; + + bool done = false; + m_tileTriCount = 0; + do{ + int current_count = 0; + done=rcGetChunksOverlappingRect(chunkyMesh, tbmin, tbmax, cid, 1024,current_count,current_node); + for (int i = 0; i < current_count; ++i) + { + const rcChunkyTriMeshNode& node = chunkyMesh->nodes[cid[i]]; + const int* ctris = &chunkyMesh->tris[node.i * 3]; + const int nctris = node.n; + + m_tileTriCount += nctris; + + memset(m_triareas, 0, nctris * sizeof(unsigned char)); + rcMarkWalkableTriangles(m_ctx, m_cfg.walkableSlopeAngle, + verts, nverts, ctris, nctris, m_triareas); + + if (!rcRasterizeTriangles(m_ctx, verts, nverts, ctris, m_triareas, nctris, *m_solid, m_cfg.walkableClimb)) + return 0; + } + } while (!done); + + if (m_tileTriCount == 0) + return 0; +#endif + if (!m_keepInterResults) + { + delete [] m_triareas; + m_triareas = 0; + } + + // Once all geometry is rasterized, we do initial pass of filtering to + // remove unwanted overhangs caused by the conservative rasterization + // as well as filter spans where the character cannot possibly stand. + if (m_filterLowHangingObstacles) + rcFilterLowHangingWalkableObstacles(m_ctx, m_cfg.walkableClimb, *m_solid); + if (m_filterLedgeSpans) + rcFilterLedgeSpans(m_ctx, m_cfg.walkableHeight, m_cfg.walkableClimb, *m_solid); + if (m_filterWalkableLowHeightSpans) + rcFilterWalkableLowHeightSpans(m_ctx, m_cfg.walkableHeight, *m_solid); + + // Compact the heightfield so that it is faster to handle from now on. + // This will result more cache coherent data as well as the neighbours + // between walkable cells will be calculated. + m_chf = rcAllocCompactHeightfield(); + if (!m_chf) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'chf'."); + return 0; + } + if (!rcBuildCompactHeightfield(m_ctx, m_cfg.walkableHeight, m_cfg.walkableClimb, *m_solid, *m_chf)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build compact data."); + return 0; + } + + if (!m_keepInterResults) + { + rcFreeHeightField(m_solid); + m_solid = 0; + } + + // Erode the walkable area by agent radius. + if (!rcErodeWalkableArea(m_ctx, m_cfg.walkableRadius, *m_chf)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not erode."); + return 0; + } + + // (Optional) Mark areas. + const ConvexVolume* vols = m_geom->getConvexVolumes(); + for (int i = 0; i < m_geom->getConvexVolumeCount(); ++i) + rcMarkConvexPolyArea(m_ctx, vols[i].verts, vols[i].nverts, vols[i].hmin, vols[i].hmax, (unsigned char)vols[i].area, *m_chf); + + + // Partition the heightfield so that we can use simple algorithm later to triangulate the walkable areas. + // There are 3 martitioning methods, each with some pros and cons: + // 1) Watershed partitioning + // - the classic Recast partitioning + // - creates the nicest tessellation + // - usually slowest + // - partitions the heightfield into nice regions without holes or overlaps + // - the are some corner cases where this method creates produces holes and overlaps + // - holes may appear when a small obstacles is close to large open area (triangulation can handle this) + // - overlaps may occur if you have narrow spiral corridors (i.e stairs), this make triangulation to fail + // * generally the best choice if you precompute the nacmesh, use this if you have large open areas + // 2) Monotone partioning + // - fastest + // - partitions the heightfield into regions without holes and overlaps (guaranteed) + // - creates long thin polygons, which sometimes causes paths with detours + // * use this if you want fast navmesh generation + // 3) Layer partitoining + // - quite fast + // - partitions the heighfield into non-overlapping regions + // - relies on the triangulation code to cope with holes (thus slower than monotone partitioning) + // - produces better triangles than monotone partitioning + // - does not have the corner cases of watershed partitioning + // - can be slow and create a bit ugly tessellation (still better than monotone) + // if you have large open areas with small obstacles (not a problem if you use tiles) + // * good choice to use for tiled navmesh with medium and small sized tiles + + if (m_partitionType == SAMPLE_PARTITION_WATERSHED) + { + // Prepare for region partitioning, by calculating distance field along the walkable surface. + if (!rcBuildDistanceField(m_ctx, *m_chf)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build distance field."); + return 0; + } + + // Partition the walkable surface into simple regions without holes. + if (!rcBuildRegions(m_ctx, *m_chf, m_cfg.borderSize, m_cfg.minRegionArea, m_cfg.mergeRegionArea)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build watershed regions."); + return 0; + } + } + else if (m_partitionType == SAMPLE_PARTITION_MONOTONE) + { + // Partition the walkable surface into simple regions without holes. + // Monotone partitioning does not need distancefield. + if (!rcBuildRegionsMonotone(m_ctx, *m_chf, m_cfg.borderSize, m_cfg.minRegionArea, m_cfg.mergeRegionArea)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build monotone regions."); + return 0; + } + } + else // SAMPLE_PARTITION_LAYERS + { + // Partition the walkable surface into simple regions without holes. + if (!rcBuildLayerRegions(m_ctx, *m_chf, m_cfg.borderSize, m_cfg.minRegionArea)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not build layer regions."); + return 0; + } + } + + // Create contours. + m_cset = rcAllocContourSet(); + if (!m_cset) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'cset'."); + return 0; + } + if (!rcBuildContours(m_ctx, *m_chf, m_cfg.maxSimplificationError, m_cfg.maxEdgeLen, *m_cset)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not create contours."); + return 0; + } + + if (m_cset->nconts == 0) + { + return 0; + } + + // Build polygon navmesh from the contours. + m_pmesh = rcAllocPolyMesh(); + if (!m_pmesh) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'pmesh'."); + return 0; + } + if (!rcBuildPolyMesh(m_ctx, *m_cset, m_cfg.maxVertsPerPoly, *m_pmesh)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could not triangulate contours."); + return 0; + } + + // Build detail mesh. + m_dmesh = rcAllocPolyMeshDetail(); + if (!m_dmesh) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Out of memory 'dmesh'."); + return 0; + } + rcFlipPolyMesh(*m_pmesh); + if (!rcBuildPolyMeshDetail(m_ctx, *m_pmesh, *m_chf, + m_cfg.detailSampleDist, m_cfg.detailSampleMaxError, + *m_dmesh)) + { + m_ctx->log(RC_LOG_ERROR, "buildNavigation: Could build polymesh detail."); + return 0; + } + + //rcFlipPolyMeshDetail(*m_dmesh,m_pmesh->nverts); + if (!m_keepInterResults) + { + rcFreeCompactHeightfield(m_chf); + m_chf = 0; + rcFreeContourSet(m_cset); + m_cset = 0; + } + + unsigned char* navData = 0; + int navDataSize = 0; + if (m_cfg.maxVertsPerPoly <= DT_VERTS_PER_POLYGON) + { + if (m_pmesh->nverts >= 0xffff) + { + // The vertex indices are ushorts, and cannot point to more than 0xffff vertices. + m_ctx->log(RC_LOG_ERROR, "Too many vertices per tile %d (max: %d).", m_pmesh->nverts, 0xffff); + return 0; + } + + // Update poly flags from areas. + for (int i = 0; i < m_pmesh->npolys; ++i) + { + if (m_pmesh->areas[i] == RC_WALKABLE_AREA) + m_pmesh->areas[i] = SAMPLE_POLYAREA_GROUND; + + if (m_pmesh->areas[i] == SAMPLE_POLYAREA_GROUND || + m_pmesh->areas[i] == SAMPLE_POLYAREA_GRASS || + m_pmesh->areas[i] == SAMPLE_POLYAREA_ROAD) + { + m_pmesh->flags[i] = SAMPLE_POLYFLAGS_WALK; + } + else if (m_pmesh->areas[i] == SAMPLE_POLYAREA_WATER) + { + m_pmesh->flags[i] = SAMPLE_POLYFLAGS_SWIM; + } + else if (m_pmesh->areas[i] == SAMPLE_POLYAREA_DOOR) + { + m_pmesh->flags[i] = SAMPLE_POLYFLAGS_WALK | SAMPLE_POLYFLAGS_DOOR; + } + } + + dtNavMeshCreateParams params; + memset(¶ms, 0, sizeof(params)); + params.verts = m_pmesh->verts; + params.vertCount = m_pmesh->nverts; + params.polys = m_pmesh->polys; + params.polyAreas = m_pmesh->areas; + params.polyFlags = m_pmesh->flags; + params.polyCount = m_pmesh->npolys; + params.nvp = m_pmesh->nvp; + params.detailMeshes = m_dmesh->meshes; + params.detailVerts = m_dmesh->verts; + params.detailVertsCount = m_dmesh->nverts; + params.detailTris = m_dmesh->tris; + params.detailTriCount = m_dmesh->ntris; + params.offMeshConVerts = m_geom->getOffMeshConnectionVerts(); + params.offMeshConRad = m_geom->getOffMeshConnectionRads(); + params.offMeshConDir = m_geom->getOffMeshConnectionDirs(); + params.offMeshConAreas = m_geom->getOffMeshConnectionAreas(); + params.offMeshConFlags = m_geom->getOffMeshConnectionFlags(); + params.offMeshConUserID = m_geom->getOffMeshConnectionId(); + params.offMeshConCount = m_geom->getOffMeshConnectionCount(); + params.walkableHeight = m_agentHeight; + params.walkableRadius = m_agentRadius; + params.walkableClimb = m_agentMaxClimb; + params.tileX = tx; + params.tileY = ty; + params.tileLayer = 0; + rcVcopy(params.bmin, m_pmesh->bmin); + rcVcopy(params.bmax, m_pmesh->bmax); + params.cs = m_cfg.cs; + params.ch = m_cfg.ch; + params.buildBvTree = true; + + if (!dtCreateNavMeshData(¶ms, &navData, &navDataSize)) + { + m_ctx->log(RC_LOG_ERROR, "Could not build Detour navmesh."); + return 0; + } + } + m_tileMemUsage = navDataSize/1024.0f; + + m_ctx->stopTimer(RC_TIMER_TOTAL); + + // Show performance stats. + duLogBuildTimes(*m_ctx, m_ctx->getAccumulatedTime(RC_TIMER_TOTAL)); + m_ctx->log(RC_LOG_PROGRESS, ">> Polymesh: %d vertices %d polygons", m_pmesh->nverts, m_pmesh->npolys); + + m_tileBuildTime = m_ctx->getAccumulatedTime(RC_TIMER_TOTAL)/1000.0f; + + dataSize = navDataSize; + return navData; +} diff --git a/r5dev/naveditor/TestCase.cpp b/r5dev/naveditor/TestCase.cpp new file mode 100644 index 00000000..75309d84 --- /dev/null +++ b/r5dev/naveditor/TestCase.cpp @@ -0,0 +1,463 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#include +#include +#include +#include +#include "NavEditor/Include/TestCase.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Detour/Include/DetourNavMeshQuery.h" +#include "Detour/Include/DetourCommon.h" + +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" +#ifdef __APPLE__ +# include +#else +# include +#endif +#include "NavEditor/Include/imgui.h" +#include "NavEditor/Include/PerfTimer.h" + +#ifdef WIN32 +#define snprintf _snprintf +#endif + +TestCase::TestCase() : + m_tests(0) +{ +} + +TestCase::~TestCase() +{ + Test* iter = m_tests; + while (iter) + { + Test* next = iter->next; + delete iter; + iter = next; + } +} + + +static char* parseRow(char* buf, char* bufEnd, char* row, int len) +{ + bool start = true; + bool done = false; + int n = 0; + while (!done && buf < bufEnd) + { + char c = *buf; + buf++; + // multirow + switch (c) + { + case '\n': + if (start) break; + done = true; + break; + case '\r': + break; + case '\t': + case ' ': + if (start) break; + // else falls through + default: + start = false; + row[n++] = c; + if (n >= len-1) + done = true; + break; + } + } + row[n] = '\0'; + return buf; +} + +static void copyName(std::string& dst, const char* src) +{ + // Skip white spaces + while (*src && isspace(*src)) + src++; + dst = src; +} + +bool TestCase::load(const std::string& filePath) +{ + char* buf = 0; + FILE* fp = fopen(filePath.c_str(), "rb"); + if (!fp) + return false; + if (fseek(fp, 0, SEEK_END) != 0) + { + fclose(fp); + return false; + } + long bufSize = ftell(fp); + if (bufSize < 0) + { + fclose(fp); + return false; + } + if (fseek(fp, 0, SEEK_SET) != 0) + { + fclose(fp); + return false; + } + buf = new char[bufSize]; + if (!buf) + { + fclose(fp); + return false; + } + size_t readLen = fread(buf, bufSize, 1, fp); + fclose(fp); + if (readLen != 1) + { + delete[] buf; + return false; + } + + char* src = buf; + char* srcEnd = buf + bufSize; + char row[512]; + while (src < srcEnd) + { + // Parse one row + row[0] = '\0'; + src = parseRow(src, srcEnd, row, sizeof(row)/sizeof(char)); + if (row[0] == 's') + { + // Sample name. + copyName(m_sampleName, row+1); + } + else if (row[0] == 'f') + { + // File name. + copyName(m_geomFileName, row+1); + } + else if (row[0] == 'p' && row[1] == 'f') + { + // Pathfind test. + Test* test = new Test(); + test->type = TEST_PATHFIND; + test->expand = false; + test->next = m_tests; + m_tests = test; + sscanf(row+2, "%f %f %f %f %f %f %hx %hx", + &test->spos[0], &test->spos[1], &test->spos[2], + &test->epos[0], &test->epos[1], &test->epos[2], + &test->includeFlags, &test->excludeFlags); + } + else if (row[0] == 'r' && row[1] == 'c') + { + // Pathfind test. + Test* test = new Test(); + test->type = TEST_RAYCAST; + test->expand = false; + test->next = m_tests; + m_tests = test; + sscanf(row+2, "%f %f %f %f %f %f %hx %hx", + &test->spos[0], &test->spos[1], &test->spos[2], + &test->epos[0], &test->epos[1], &test->epos[2], + &test->includeFlags, &test->excludeFlags); + } + } + + delete [] buf; + + return true; +} + +void TestCase::resetTimes() +{ + for (Test* iter = m_tests; iter; iter = iter->next) + { + iter->findNearestPolyTime = 0; + iter->findPathTime = 0; + iter->findStraightPathTime = 0; + } +} + +void TestCase::doTests(dtNavMesh* navmesh, dtNavMeshQuery* navquery) +{ + if (!navmesh || !navquery) + return; + + resetTimes(); + + static const int MAX_POLYS = 256; + dtPolyRef polys[MAX_POLYS]; + float straight[MAX_POLYS*3]; + const float polyPickExt[3] = {2,4,2}; + + for (Test* iter = m_tests; iter; iter = iter->next) + { + delete [] iter->polys; + iter->polys = 0; + iter->npolys = 0; + delete [] iter->straight; + iter->straight = 0; + iter->nstraight = 0; + + dtQueryFilter filter; + filter.setIncludeFlags(iter->includeFlags); + filter.setExcludeFlags(iter->excludeFlags); + + // Find start points + TimeVal findNearestPolyStart = getPerfTime(); + + dtPolyRef startRef, endRef; + navquery->findNearestPoly(iter->spos, polyPickExt, &filter, &startRef, iter->nspos); + navquery->findNearestPoly(iter->epos, polyPickExt, &filter, &endRef, iter->nepos); + + TimeVal findNearestPolyEnd = getPerfTime(); + iter->findNearestPolyTime += getPerfTimeUsec(findNearestPolyEnd - findNearestPolyStart); + + if (!startRef || ! endRef) + continue; + + if (iter->type == TEST_PATHFIND) + { + // Find path + TimeVal findPathStart = getPerfTime(); + + navquery->findPath(startRef, endRef, iter->spos, iter->epos, &filter, polys, &iter->npolys, MAX_POLYS); + + TimeVal findPathEnd = getPerfTime(); + iter->findPathTime += getPerfTimeUsec(findPathEnd - findPathStart); + + // Find straight path + if (iter->npolys) + { + TimeVal findStraightPathStart = getPerfTime(); + + navquery->findStraightPath(iter->spos, iter->epos, polys, iter->npolys, + straight, 0, 0, &iter->nstraight, MAX_POLYS); + TimeVal findStraightPathEnd = getPerfTime(); + iter->findStraightPathTime += getPerfTimeUsec(findStraightPathEnd - findStraightPathStart); + } + + // Copy results + if (iter->npolys) + { + iter->polys = new dtPolyRef[iter->npolys]; + memcpy(iter->polys, polys, sizeof(dtPolyRef)*iter->npolys); + } + if (iter->nstraight) + { + iter->straight = new float[iter->nstraight*3]; + memcpy(iter->straight, straight, sizeof(float)*3*iter->nstraight); + } + } + else if (iter->type == TEST_RAYCAST) + { + float t = 0; + float hitNormal[3], hitPos[3]; + + iter->straight = new float[2*3]; + iter->nstraight = 2; + + iter->straight[0] = iter->spos[0]; + iter->straight[1] = iter->spos[1]; + iter->straight[2] = iter->spos[2]; + + TimeVal findPathStart = getPerfTime(); + + navquery->raycast(startRef, iter->spos, iter->epos, &filter, &t, hitNormal, polys, &iter->npolys, MAX_POLYS); + + TimeVal findPathEnd = getPerfTime(); + iter->findPathTime += getPerfTimeUsec(findPathEnd - findPathStart); + + if (t > 1) + { + // No hit + dtVcopy(hitPos, iter->epos); + } + else + { + // Hit + dtVlerp(hitPos, iter->spos, iter->epos, t); + } + // Adjust height. + if (iter->npolys > 0) + { + float h = 0; + navquery->getPolyHeight(polys[iter->npolys-1], hitPos, &h); + hitPos[1] = h; + } + dtVcopy(&iter->straight[3], hitPos); + + if (iter->npolys) + { + iter->polys = new dtPolyRef[iter->npolys]; + memcpy(iter->polys, polys, sizeof(dtPolyRef)*iter->npolys); + } + } + } + + + printf("Test Results:\n"); + int n = 0; + for (Test* iter = m_tests; iter; iter = iter->next) + { + const int total = iter->findNearestPolyTime + iter->findPathTime + iter->findStraightPathTime; + printf(" - Path %02d: %.4f ms\n", n, (float)total/1000.0f); + printf(" - poly: %.4f ms\n", (float)iter->findNearestPolyTime/1000.0f); + printf(" - path: %.4f ms\n", (float)iter->findPathTime/1000.0f); + printf(" - straight: %.4f ms\n", (float)iter->findStraightPathTime/1000.0f); + n++; + } +} + +void TestCase::handleRender() +{ + glLineWidth(2.0f); + glBegin(GL_LINES); + for (Test* iter = m_tests; iter; iter = iter->next) + { + float dir[3]; + dtVsub(dir, iter->epos, iter->spos); + dtVnormalize(dir); + glColor4ub(128,25,0,192); + glVertex3f(iter->spos[0],iter->spos[1]-0.3f,iter->spos[2]); + glVertex3f(iter->spos[0],iter->spos[1]+0.3f,iter->spos[2]); + glVertex3f(iter->spos[0],iter->spos[1]+0.3f,iter->spos[2]); + glVertex3f(iter->spos[0]+dir[0]*0.3f,iter->spos[1]+0.3f+dir[1]*0.3f,iter->spos[2]+dir[2]*0.3f); + glColor4ub(51,102,0,129); + glVertex3f(iter->epos[0],iter->epos[1]-0.3f,iter->epos[2]); + glVertex3f(iter->epos[0],iter->epos[1]+0.3f,iter->epos[2]); + + if (iter->expand) + { + const float s = 0.1f; + glColor4ub(255,32,0,128); + glVertex3f(iter->spos[0]-s,iter->spos[1],iter->spos[2]); + glVertex3f(iter->spos[0]+s,iter->spos[1],iter->spos[2]); + glVertex3f(iter->spos[0],iter->spos[1],iter->spos[2]-s); + glVertex3f(iter->spos[0],iter->spos[1],iter->spos[2]+s); + glColor4ub(255,192,0,255); + glVertex3f(iter->nspos[0]-s,iter->nspos[1],iter->nspos[2]); + glVertex3f(iter->nspos[0]+s,iter->nspos[1],iter->nspos[2]); + glVertex3f(iter->nspos[0],iter->nspos[1],iter->nspos[2]-s); + glVertex3f(iter->nspos[0],iter->nspos[1],iter->nspos[2]+s); + + glColor4ub(255,32,0,128); + glVertex3f(iter->epos[0]-s,iter->epos[1],iter->epos[2]); + glVertex3f(iter->epos[0]+s,iter->epos[1],iter->epos[2]); + glVertex3f(iter->epos[0],iter->epos[1],iter->epos[2]-s); + glVertex3f(iter->epos[0],iter->epos[1],iter->epos[2]+s); + glColor4ub(255,192,0,255); + glVertex3f(iter->nepos[0]-s,iter->nepos[1],iter->nepos[2]); + glVertex3f(iter->nepos[0]+s,iter->nepos[1],iter->nepos[2]); + glVertex3f(iter->nepos[0],iter->nepos[1],iter->nepos[2]-s); + glVertex3f(iter->nepos[0],iter->nepos[1],iter->nepos[2]+s); + } + + if (iter->expand) + glColor4ub(255,192,0,255); + else + glColor4ub(0,0,0,64); + + for (int i = 0; i < iter->nstraight-1; ++i) + { + glVertex3f(iter->straight[i*3+0],iter->straight[i*3+1]+0.3f,iter->straight[i*3+2]); + glVertex3f(iter->straight[(i+1)*3+0],iter->straight[(i+1)*3+1]+0.3f,iter->straight[(i+1)*3+2]); + } + } + glEnd(); + glLineWidth(1.0f); +} + +bool TestCase::handleRenderOverlay(double* proj, double* model, int* view) +{ + GLdouble x, y, z; + char text[64], subtext[64]; + int n = 0; + + static const float LABEL_DIST = 1.0f; + + for (Test* iter = m_tests; iter; iter = iter->next) + { + float pt[3], dir[3]; + if (iter->nstraight) + { + dtVcopy(pt, &iter->straight[3]); + if (dtVdist(pt, iter->spos) > LABEL_DIST) + { + dtVsub(dir, pt, iter->spos); + dtVnormalize(dir); + dtVmad(pt, iter->spos, dir, LABEL_DIST); + } + pt[1]+=0.5f; + } + else + { + dtVsub(dir, iter->epos, iter->spos); + dtVnormalize(dir); + dtVmad(pt, iter->spos, dir, LABEL_DIST); + pt[1]+=0.5f; + } + + if (gluProject((GLdouble)pt[0], (GLdouble)pt[1], (GLdouble)pt[2], + model, proj, view, &x, &y, &z)) + { + snprintf(text, 64, "Path %d\n", n); + unsigned int col = imguiRGBA(0,0,0,128); + if (iter->expand) + col = imguiRGBA(255,192,0,220); + imguiDrawText((int)x, (int)(y-25), IMGUI_ALIGN_CENTER, text, col); + } + n++; + } + + static int resScroll = 0; + bool mouseOverMenu = imguiBeginScrollArea("Test Results", 10, view[3] - 10 - 350, 200, 350, &resScroll); +// mouseOverMenu = true; + + n = 0; + for (Test* iter = m_tests; iter; iter = iter->next) + { + const int total = iter->findNearestPolyTime + iter->findPathTime + iter->findStraightPathTime; + snprintf(subtext, 64, "%.4f ms", (float)total/1000.0f); + snprintf(text, 64, "Path %d", n); + + if (imguiCollapse(text, subtext, iter->expand)) + iter->expand = !iter->expand; + if (iter->expand) + { + snprintf(text, 64, "Poly: %.4f ms", (float)iter->findNearestPolyTime/1000.0f); + imguiValue(text); + + snprintf(text, 64, "Path: %.4f ms", (float)iter->findPathTime/1000.0f); + imguiValue(text); + + snprintf(text, 64, "Straight: %.4f ms", (float)iter->findStraightPathTime/1000.0f); + imguiValue(text); + + imguiSeparator(); + } + + n++; + } + + imguiEndScrollArea(); + + return mouseOverMenu; +} diff --git a/r5dev/naveditor/ValueHistory.cpp b/r5dev/naveditor/ValueHistory.cpp new file mode 100644 index 00000000..48e67e8e --- /dev/null +++ b/r5dev/naveditor/ValueHistory.cpp @@ -0,0 +1,114 @@ +#include "NavEditor/Include/ValueHistory.h" +#include "NavEditor/Include/imgui.h" +#include +#include + +#ifdef WIN32 +# define snprintf _snprintf +#endif + +ValueHistory::ValueHistory() : + m_hsamples(0) +{ + for (int i = 0; i < MAX_HISTORY; ++i) + m_samples[i] = 0; +} + +float ValueHistory::getSampleMin() const +{ + float val = m_samples[0]; + for (int i = 1; i < MAX_HISTORY; ++i) + if (m_samples[i] < val) + val = m_samples[i]; + return val; +} + +float ValueHistory::getSampleMax() const +{ + float val = m_samples[0]; + for (int i = 1; i < MAX_HISTORY; ++i) + if (m_samples[i] > val) + val = m_samples[i]; + return val; +} + +float ValueHistory::getAverage() const +{ + float val = 0; + for (int i = 0; i < MAX_HISTORY; ++i) + val += m_samples[i]; + return val/(float)MAX_HISTORY; +} + +void GraphParams::setRect(int ix, int iy, int iw, int ih, int ipad) +{ + x = ix; + y = iy; + w = iw; + h = ih; + pad = ipad; +} + +void GraphParams::setValueRange(float ivmin, float ivmax, int indiv, const char* iunits) +{ + vmin = ivmin; + vmax = ivmax; + ndiv = indiv; + strcpy(units, iunits); +} + +void drawGraphBackground(const GraphParams* p) +{ + // BG + imguiDrawRoundedRect((float)p->x, (float)p->y, (float)p->w, (float)p->h, (float)p->pad, imguiRGBA(64,64,64,128)); + + const float sy = (p->h-p->pad*2) / (p->vmax-p->vmin); + const float oy = p->y+p->pad-p->vmin*sy; + + char text[64]; + + // Divider Lines + for (int i = 0; i <= p->ndiv; ++i) + { + const float u = (float)i/(float)p->ndiv; + const float v = p->vmin + (p->vmax-p->vmin)*u; + snprintf(text, 64, "%.2f %s", v, p->units); + const float fy = oy + v*sy; + imguiDrawText(p->x + p->w - p->pad, (int)fy-4, IMGUI_ALIGN_RIGHT, text, imguiRGBA(0,0,0,255)); + imguiDrawLine((float)p->x + (float)p->pad, fy, (float)p->x + (float)p->w - (float)p->pad - 50, fy, 1.0f, imguiRGBA(0,0,0,64)); + } +} + +void drawGraph(const GraphParams* p, const ValueHistory* graph, + int idx, const char* label, const unsigned int col) +{ + const float sx = (p->w - p->pad*2) / (float)graph->getSampleCount(); + const float sy = (p->h - p->pad*2) / (p->vmax - p->vmin); + const float ox = (float)p->x + (float)p->pad; + const float oy = (float)p->y + (float)p->pad - p->vmin*sy; + + // Values + float px=0, py=0; + for (int i = 0; i < graph->getSampleCount()-1; ++i) + { + const float x = ox + i*sx; + const float y = oy + graph->getSample(i)*sy; + if (i > 0) + imguiDrawLine(px,py, x,y, 2.0f, col); + px = x; + py = y; + } + + // Label + const int size = 15; + const int spacing = 10; + int ix = p->x + p->w + 5; + int iy = p->y + p->h - (idx+1)*(size+spacing); + + imguiDrawRoundedRect((float)ix, (float)iy, (float)size, (float)size, 2.0f, col); + + char text[64]; + snprintf(text, 64, "%.2f %s", graph->getAverage(), p->units); + imguiDrawText(ix+size+5, iy+3, IMGUI_ALIGN_LEFT, label, imguiRGBA(255,255,255,192)); + imguiDrawText(ix+size+150, iy+3, IMGUI_ALIGN_RIGHT, text, imguiRGBA(255,255,255,128)); +} diff --git a/r5dev/naveditor/imgui.cpp b/r5dev/naveditor/imgui.cpp new file mode 100644 index 00000000..e1c6e5f7 --- /dev/null +++ b/r5dev/naveditor/imgui.cpp @@ -0,0 +1,676 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#include +#include +#define _USE_MATH_DEFINES +#include +#include "NavEditor/Include/imgui.h" + +#ifdef WIN32 +# define snprintf _snprintf +#endif + +////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + +static const unsigned TEXT_POOL_SIZE = 50000; +static char g_textPool[TEXT_POOL_SIZE]; +static unsigned g_textPoolSize = 0; +static const char* allocText(const char* text) +{ + unsigned len = static_cast(strlen(text)+1); + if (g_textPoolSize + len >= TEXT_POOL_SIZE) + return 0; + char* dst = &g_textPool[g_textPoolSize]; + memcpy(dst, text, len); + g_textPoolSize += len; + return dst; +} + +static const unsigned GFXCMD_QUEUE_SIZE = 5000; +static imguiGfxCmd g_gfxCmdQueue[GFXCMD_QUEUE_SIZE]; +static unsigned g_gfxCmdQueueSize = 0; + +static void resetGfxCmdQueue() +{ + g_gfxCmdQueueSize = 0; + g_textPoolSize = 0; +} + +static void addGfxCmdScissor(int x, int y, int w, int h) +{ + if (g_gfxCmdQueueSize >= GFXCMD_QUEUE_SIZE) + return; + imguiGfxCmd& cmd = g_gfxCmdQueue[g_gfxCmdQueueSize++]; + cmd.type = IMGUI_GFXCMD_SCISSOR; + cmd.flags = x < 0 ? 0 : 1; // on/off flag. + cmd.col = 0; + cmd.rect.x = (short)x; + cmd.rect.y = (short)y; + cmd.rect.w = (short)w; + cmd.rect.h = (short)h; +} + +static void addGfxCmdRect(float x, float y, float w, float h, unsigned int color) +{ + if (g_gfxCmdQueueSize >= GFXCMD_QUEUE_SIZE) + return; + imguiGfxCmd& cmd = g_gfxCmdQueue[g_gfxCmdQueueSize++]; + cmd.type = IMGUI_GFXCMD_RECT; + cmd.flags = 0; + cmd.col = color; + cmd.rect.x = (short)(x*8.0f); + cmd.rect.y = (short)(y*8.0f); + cmd.rect.w = (short)(w*8.0f); + cmd.rect.h = (short)(h*8.0f); + cmd.rect.r = 0; +} + +static void addGfxCmdLine(float x0, float y0, float x1, float y1, float r, unsigned int color) +{ + if (g_gfxCmdQueueSize >= GFXCMD_QUEUE_SIZE) + return; + imguiGfxCmd& cmd = g_gfxCmdQueue[g_gfxCmdQueueSize++]; + cmd.type = IMGUI_GFXCMD_LINE; + cmd.flags = 0; + cmd.col = color; + cmd.line.x0 = (short)(x0*8.0f); + cmd.line.y0 = (short)(y0*8.0f); + cmd.line.x1 = (short)(x1*8.0f); + cmd.line.y1 = (short)(y1*8.0f); + cmd.line.r = (short)(r*8.0f); +} + +static void addGfxCmdRoundedRect(float x, float y, float w, float h, float r, unsigned int color) +{ + if (g_gfxCmdQueueSize >= GFXCMD_QUEUE_SIZE) + return; + imguiGfxCmd& cmd = g_gfxCmdQueue[g_gfxCmdQueueSize++]; + cmd.type = IMGUI_GFXCMD_RECT; + cmd.flags = 0; + cmd.col = color; + cmd.rect.x = (short)(x*8.0f); + cmd.rect.y = (short)(y*8.0f); + cmd.rect.w = (short)(w*8.0f); + cmd.rect.h = (short)(h*8.0f); + cmd.rect.r = (short)(r*8.0f); +} + +static void addGfxCmdTriangle(int x, int y, int w, int h, int flags, unsigned int color) +{ + if (g_gfxCmdQueueSize >= GFXCMD_QUEUE_SIZE) + return; + imguiGfxCmd& cmd = g_gfxCmdQueue[g_gfxCmdQueueSize++]; + cmd.type = IMGUI_GFXCMD_TRIANGLE; + cmd.flags = (char)flags; + cmd.col = color; + cmd.rect.x = (short)(x*8.0f); + cmd.rect.y = (short)(y*8.0f); + cmd.rect.w = (short)(w*8.0f); + cmd.rect.h = (short)(h*8.0f); +} + +static void addGfxCmdText(int x, int y, int align, const char* text, unsigned int color) +{ + if (g_gfxCmdQueueSize >= GFXCMD_QUEUE_SIZE) + return; + imguiGfxCmd& cmd = g_gfxCmdQueue[g_gfxCmdQueueSize++]; + cmd.type = IMGUI_GFXCMD_TEXT; + cmd.flags = 0; + cmd.col = color; + cmd.text.x = (short)x; + cmd.text.y = (short)y; + cmd.text.align = (short)align; + cmd.text.text = allocText(text); +} + +////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// +struct GuiState +{ + GuiState() : + left(false), leftPressed(false), leftReleased(false), + mx(-1), my(-1), scroll(0), + active(0), hot(0), hotToBe(0), isHot(false), isActive(false), wentActive(false), + dragX(0), dragY(0), dragOrig(0), widgetX(0), widgetY(0), widgetW(100), + insideCurrentScroll(false), areaId(0), widgetId(0) + { + } + + bool left; + bool leftPressed, leftReleased; + int mx,my; + int scroll; + unsigned int active; + unsigned int hot; + unsigned int hotToBe; + bool isHot; + bool isActive; + bool wentActive; + int dragX, dragY; + float dragOrig; + int widgetX, widgetY, widgetW; + bool insideCurrentScroll; + + unsigned int areaId; + unsigned int widgetId; +}; + +static GuiState g_state; + +inline bool anyActive() +{ + return g_state.active != 0; +} + +inline bool isActive(unsigned int id) +{ + return g_state.active == id; +} + +inline bool isHot(unsigned int id) +{ + return g_state.hot == id; +} + +inline bool inRect(int x, int y, int w, int h, bool checkScroll = true) +{ + return (!checkScroll || g_state.insideCurrentScroll) && g_state.mx >= x && g_state.mx <= x+w && g_state.my >= y && g_state.my <= y+h; +} + +inline void clearInput() +{ + g_state.leftPressed = false; + g_state.leftReleased = false; + g_state.scroll = 0; +} + +inline void clearActive() +{ + g_state.active = 0; + // mark all UI for this frame as processed + clearInput(); +} + +inline void setActive(unsigned int id) +{ + g_state.active = id; + g_state.wentActive = true; +} + +inline void setHot(unsigned int id) +{ + g_state.hotToBe = id; +} + + +static bool buttonLogic(unsigned int id, bool over) +{ + bool res = false; + // process down + if (!anyActive()) + { + if (over) + setHot(id); + if (isHot(id) && g_state.leftPressed) + setActive(id); + } + + // if button is active, then react on left up + if (isActive(id)) + { + g_state.isActive = true; + if (over) + setHot(id); + if (g_state.leftReleased) + { + if (isHot(id)) + res = true; + clearActive(); + } + } + + if (isHot(id)) + g_state.isHot = true; + + return res; +} + +static void updateInput(int mx, int my, unsigned char mbut, int scroll) +{ + bool left = (mbut & IMGUI_MBUT_LEFT) != 0; + + g_state.mx = mx; + g_state.my = my; + g_state.leftPressed = !g_state.left && left; + g_state.leftReleased = g_state.left && !left; + g_state.left = left; + + g_state.scroll = scroll; +} + +void imguiBeginFrame(int mx, int my, unsigned char mbut, int scroll) +{ + updateInput(mx,my,mbut,scroll); + + g_state.hot = g_state.hotToBe; + g_state.hotToBe = 0; + + g_state.wentActive = false; + g_state.isActive = false; + g_state.isHot = false; + + g_state.widgetX = 0; + g_state.widgetY = 0; + g_state.widgetW = 0; + + g_state.areaId = 1; + g_state.widgetId = 1; + + resetGfxCmdQueue(); +} + +void imguiEndFrame() +{ + clearInput(); +} + +const imguiGfxCmd* imguiGetRenderQueue() +{ + return g_gfxCmdQueue; +} + +int imguiGetRenderQueueSize() +{ + return g_gfxCmdQueueSize; +} + + +////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// +static const int BUTTON_HEIGHT = 20; +static const int SLIDER_HEIGHT = 20; +static const int SLIDER_MARKER_WIDTH = 10; +static const int CHECK_SIZE = 8; +static const int DEFAULT_SPACING = 4; +static const int TEXT_HEIGHT = 8; +static const int SCROLL_AREA_PADDING = 6; +static const int INDENT_SIZE = 16; +static const int AREA_HEADER = 28; + +static int g_scrollTop = 0; +static int g_scrollBottom = 0; +static int g_scrollRight = 0; +static int g_scrollAreaTop = 0; +static int* g_scrollVal = 0; +static int g_focusTop = 0; +static int g_focusBottom = 0; +static unsigned int g_scrollId = 0; +static bool g_insideScrollArea = false; + +bool imguiBeginScrollArea(const char* name, int x, int y, int w, int h, int* scroll) +{ + g_state.areaId++; + g_state.widgetId = 0; + g_scrollId = (g_state.areaId<<16) | g_state.widgetId; + + g_state.widgetX = x + SCROLL_AREA_PADDING; + g_state.widgetY = y+h-AREA_HEADER + (*scroll); + g_state.widgetW = w - SCROLL_AREA_PADDING*4; + g_scrollTop = y-AREA_HEADER+h; + g_scrollBottom = y+SCROLL_AREA_PADDING; + g_scrollRight = x+w - SCROLL_AREA_PADDING*3; + g_scrollVal = scroll; + + g_scrollAreaTop = g_state.widgetY; + + g_focusTop = y-AREA_HEADER; + g_focusBottom = y-AREA_HEADER+h; + + g_insideScrollArea = inRect(x, y, w, h, false); + g_state.insideCurrentScroll = g_insideScrollArea; + + addGfxCmdRoundedRect((float)x, (float)y, (float)w, (float)h, 6, imguiRGBA(0,0,0,192)); + + addGfxCmdText(x+AREA_HEADER/2, y+h-AREA_HEADER/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, name, imguiRGBA(255,255,255,128)); + + addGfxCmdScissor(x+SCROLL_AREA_PADDING, y+SCROLL_AREA_PADDING, w-SCROLL_AREA_PADDING*4, h-AREA_HEADER-SCROLL_AREA_PADDING); + + return g_insideScrollArea; +} + +void imguiEndScrollArea() +{ + // Disable scissoring. + addGfxCmdScissor(-1,-1,-1,-1); + + // Draw scroll bar + int x = g_scrollRight+SCROLL_AREA_PADDING/2; + int y = g_scrollBottom; + int w = SCROLL_AREA_PADDING*2; + int h = g_scrollTop - g_scrollBottom; + + int stop = g_scrollAreaTop; + int sbot = g_state.widgetY; + int sh = stop - sbot; // The scrollable area height. + + float barHeight = (float)h/(float)sh; + + if (barHeight < 1) + { + float barY = (float)(y - sbot)/(float)sh; + if (barY < 0) barY = 0; + if (barY > 1) barY = 1; + + // Handle scroll bar logic. + unsigned int hid = g_scrollId; + int hx = x; + int hy = y + (int)(barY*h); + int hw = w; + int hh = (int)(barHeight*h); + + const int range = h - (hh-1); + bool over = inRect(hx, hy, hw, hh); + buttonLogic(hid, over); + if (isActive(hid)) + { + float u = (float)(hy-y) / (float)range; + if (g_state.wentActive) + { + g_state.dragY = g_state.my; + g_state.dragOrig = u; + } + if (g_state.dragY != g_state.my) + { + u = g_state.dragOrig + (g_state.my - g_state.dragY) / (float)range; + if (u < 0) u = 0; + if (u > 1) u = 1; + *g_scrollVal = (int)((1-u) * (sh - h)); + } + } + + // BG + addGfxCmdRoundedRect((float)x, (float)y, (float)w, (float)h, (float)w/2-1, imguiRGBA(0,0,0,196)); + // Bar + if (isActive(hid)) + addGfxCmdRoundedRect((float)hx, (float)hy, (float)hw, (float)hh, (float)w/2-1, imguiRGBA(255,196,0,196)); + else + addGfxCmdRoundedRect((float)hx, (float)hy, (float)hw, (float)hh, (float)w/2-1, isHot(hid) ? imguiRGBA(255,196,0,96) : imguiRGBA(255,255,255,64)); + + // Handle mouse scrolling. + if (g_insideScrollArea) // && !anyActive()) + { + if (g_state.scroll) + { + *g_scrollVal += 120*g_state.scroll; + if (*g_scrollVal < 0) *g_scrollVal = 0; + if (*g_scrollVal > (sh - h)) *g_scrollVal = (sh - h); + } + } + } + g_state.insideCurrentScroll = false; +} + +bool imguiButton(const char* text, bool enabled) +{ + g_state.widgetId++; + unsigned int id = (g_state.areaId<<16) | g_state.widgetId; + + int x = g_state.widgetX; + int y = g_state.widgetY - BUTTON_HEIGHT; + int w = g_state.widgetW; + int h = BUTTON_HEIGHT; + g_state.widgetY -= BUTTON_HEIGHT + DEFAULT_SPACING; + + bool over = enabled && inRect(x, y, w, h); + bool res = buttonLogic(id, over); + + addGfxCmdRoundedRect((float)x, (float)y, (float)w, (float)h, (float)BUTTON_HEIGHT/2-1, imguiRGBA(128,128,128, isActive(id)?196:96)); + if (enabled) + addGfxCmdText(x+BUTTON_HEIGHT/2, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, isHot(id) ? imguiRGBA(255,196,0,255) : imguiRGBA(255,255,255,200)); + else + addGfxCmdText(x+BUTTON_HEIGHT/2, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, imguiRGBA(128,128,128,200)); + + return res; +} + +bool imguiItem(const char* text, bool enabled) +{ + g_state.widgetId++; + unsigned int id = (g_state.areaId<<16) | g_state.widgetId; + + int x = g_state.widgetX; + int y = g_state.widgetY - BUTTON_HEIGHT; + int w = g_state.widgetW; + int h = BUTTON_HEIGHT; + g_state.widgetY -= BUTTON_HEIGHT + DEFAULT_SPACING; + + bool over = enabled && inRect(x, y, w, h); + bool res = buttonLogic(id, over); + + if (isHot(id)) + addGfxCmdRoundedRect((float)x, (float)y, (float)w, (float)h, 2.0f, imguiRGBA(255,196,0,isActive(id)?196:96)); + + if (enabled) + addGfxCmdText(x+BUTTON_HEIGHT/2, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, imguiRGBA(255,255,255,200)); + else + addGfxCmdText(x+BUTTON_HEIGHT/2, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, imguiRGBA(128,128,128,200)); + + return res; +} + +bool imguiCheck(const char* text, bool checked, bool enabled) +{ + g_state.widgetId++; + unsigned int id = (g_state.areaId<<16) | g_state.widgetId; + + int x = g_state.widgetX; + int y = g_state.widgetY - BUTTON_HEIGHT; + int w = g_state.widgetW; + int h = BUTTON_HEIGHT; + g_state.widgetY -= BUTTON_HEIGHT + DEFAULT_SPACING; + + bool over = enabled && inRect(x, y, w, h); + bool res = buttonLogic(id, over); + + const int cx = x+BUTTON_HEIGHT/2-CHECK_SIZE/2; + const int cy = y+BUTTON_HEIGHT/2-CHECK_SIZE/2; + addGfxCmdRoundedRect((float)cx-3, (float)cy-3, (float)CHECK_SIZE+6, (float)CHECK_SIZE+6, 4, imguiRGBA(128,128,128, isActive(id)?196:96)); + if (checked) + { + if (enabled) + addGfxCmdRoundedRect((float)cx, (float)cy, (float)CHECK_SIZE, (float)CHECK_SIZE, (float)CHECK_SIZE/2-1, imguiRGBA(255,255,255,isActive(id)?255:200)); + else + addGfxCmdRoundedRect((float)cx, (float)cy, (float)CHECK_SIZE, (float)CHECK_SIZE, (float)CHECK_SIZE/2-1, imguiRGBA(128,128,128,200)); + } + + if (enabled) + addGfxCmdText(x+BUTTON_HEIGHT, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, isHot(id) ? imguiRGBA(255,196,0,255) : imguiRGBA(255,255,255,200)); + else + addGfxCmdText(x+BUTTON_HEIGHT, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, imguiRGBA(128,128,128,200)); + + return res; +} + +bool imguiCollapse(const char* text, const char* subtext, bool checked, bool enabled) +{ + g_state.widgetId++; + unsigned int id = (g_state.areaId<<16) | g_state.widgetId; + + int x = g_state.widgetX; + int y = g_state.widgetY - BUTTON_HEIGHT; + int w = g_state.widgetW; + int h = BUTTON_HEIGHT; + g_state.widgetY -= BUTTON_HEIGHT; // + DEFAULT_SPACING; + + const int cx = x+BUTTON_HEIGHT/2-CHECK_SIZE/2; + const int cy = y+BUTTON_HEIGHT/2-CHECK_SIZE/2; + + bool over = enabled && inRect(x, y, w, h); + bool res = buttonLogic(id, over); + + if (checked) + addGfxCmdTriangle(cx, cy, CHECK_SIZE, CHECK_SIZE, 2, imguiRGBA(255,255,255,isActive(id)?255:200)); + else + addGfxCmdTriangle(cx, cy, CHECK_SIZE, CHECK_SIZE, 1, imguiRGBA(255,255,255,isActive(id)?255:200)); + + if (enabled) + addGfxCmdText(x+BUTTON_HEIGHT, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, isHot(id) ? imguiRGBA(255,196,0,255) : imguiRGBA(255,255,255,200)); + else + addGfxCmdText(x+BUTTON_HEIGHT, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, imguiRGBA(128,128,128,200)); + + if (subtext) + addGfxCmdText(x+w-BUTTON_HEIGHT/2, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_RIGHT, subtext, imguiRGBA(255,255,255,128)); + + return res; +} + +void imguiLabel(const char* text) +{ + int x = g_state.widgetX; + int y = g_state.widgetY - BUTTON_HEIGHT; + g_state.widgetY -= BUTTON_HEIGHT; + addGfxCmdText(x, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, imguiRGBA(255,255,255,255)); +} + +void imguiValue(const char* text) +{ + const int x = g_state.widgetX; + const int y = g_state.widgetY - BUTTON_HEIGHT; + const int w = g_state.widgetW; + g_state.widgetY -= BUTTON_HEIGHT; + + addGfxCmdText(x+w-BUTTON_HEIGHT/2, y+BUTTON_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_RIGHT, text, imguiRGBA(255,255,255,200)); +} + +bool imguiSlider(const char* text, float* val, float vmin, float vmax, float vinc, bool enabled) +{ + g_state.widgetId++; + unsigned int id = (g_state.areaId<<16) | g_state.widgetId; + + int x = g_state.widgetX; + int y = g_state.widgetY - BUTTON_HEIGHT; + int w = g_state.widgetW; + int h = SLIDER_HEIGHT; + g_state.widgetY -= SLIDER_HEIGHT + DEFAULT_SPACING; + + addGfxCmdRoundedRect((float)x, (float)y, (float)w, (float)h, 4.0f, imguiRGBA(0,0,0,128)); + + const int range = w - SLIDER_MARKER_WIDTH; + + float u = (*val - vmin) / (vmax-vmin); + if (u < 0) u = 0; + if (u > 1) u = 1; + int m = (int)(u * range); + + bool over = enabled && inRect(x+m, y, SLIDER_MARKER_WIDTH, SLIDER_HEIGHT); + bool res = buttonLogic(id, over); + bool valChanged = false; + + if (isActive(id)) + { + if (g_state.wentActive) + { + g_state.dragX = g_state.mx; + g_state.dragOrig = u; + } + if (g_state.dragX != g_state.mx) + { + u = g_state.dragOrig + (float)(g_state.mx - g_state.dragX) / (float)range; + if (u < 0) u = 0; + if (u > 1) u = 1; + *val = vmin + u*(vmax-vmin); + *val = floorf(*val/vinc+0.5f)*vinc; // Snap to vinc + m = (int)(u * range); + valChanged = true; + } + } + + if (isActive(id)) + addGfxCmdRoundedRect((float)(x+m), (float)y, (float)SLIDER_MARKER_WIDTH, (float)SLIDER_HEIGHT, 4.0f, imguiRGBA(255,255,255,255)); + else + addGfxCmdRoundedRect((float)(x+m), (float)y, (float)SLIDER_MARKER_WIDTH, (float)SLIDER_HEIGHT, 4.0f, isHot(id) ? imguiRGBA(255,196,0,128) : imguiRGBA(255,255,255,64)); + + // TODO: fix this, take a look at 'nicenum'. + int digits = (int)(ceilf(log10f(vinc))); + char fmt[16]; + snprintf(fmt, 16, "%%.%df", digits >= 0 ? 0 : -digits); + char msg[128]; + snprintf(msg, 128, fmt, *val); + + if (enabled) + { + addGfxCmdText(x+SLIDER_HEIGHT/2, y+SLIDER_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, isHot(id) ? imguiRGBA(255,196,0,255) : imguiRGBA(255,255,255,200)); + addGfxCmdText(x+w-SLIDER_HEIGHT/2, y+SLIDER_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_RIGHT, msg, isHot(id) ? imguiRGBA(255,196,0,255) : imguiRGBA(255,255,255,200)); + } + else + { + addGfxCmdText(x+SLIDER_HEIGHT/2, y+SLIDER_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_LEFT, text, imguiRGBA(128,128,128,200)); + addGfxCmdText(x+w-SLIDER_HEIGHT/2, y+SLIDER_HEIGHT/2-TEXT_HEIGHT/2, IMGUI_ALIGN_RIGHT, msg, imguiRGBA(128,128,128,200)); + } + + return res || valChanged; +} + + +void imguiIndent() +{ + g_state.widgetX += INDENT_SIZE; + g_state.widgetW -= INDENT_SIZE; +} + +void imguiUnindent() +{ + g_state.widgetX -= INDENT_SIZE; + g_state.widgetW += INDENT_SIZE; +} + +void imguiSeparator() +{ + g_state.widgetY -= DEFAULT_SPACING*3; +} + +void imguiSeparatorLine() +{ + int x = g_state.widgetX; + int y = g_state.widgetY - DEFAULT_SPACING*2; + int w = g_state.widgetW; + int h = 1; + g_state.widgetY -= DEFAULT_SPACING*4; + + addGfxCmdRect((float)x, (float)y, (float)w, (float)h, imguiRGBA(255,255,255,32)); +} + +void imguiDrawText(int x, int y, int align, const char* text, unsigned int color) +{ + addGfxCmdText(x, y, align, text, color); +} + +void imguiDrawLine(float x0, float y0, float x1, float y1, float r, unsigned int color) +{ + addGfxCmdLine(x0, y0, x1, y1, r, color); +} + +void imguiDrawRect(float x, float y, float w, float h, unsigned int color) +{ + addGfxCmdRect(x, y, w, h, color); +} + +void imguiDrawRoundedRect(float x, float y, float w, float h, float r, unsigned int color) +{ + addGfxCmdRoundedRect(x, y, w, h, r, color); +} + diff --git a/r5dev/naveditor/imguiRenderGL.cpp b/r5dev/naveditor/imguiRenderGL.cpp new file mode 100644 index 00000000..640a98fb --- /dev/null +++ b/r5dev/naveditor/imguiRenderGL.cpp @@ -0,0 +1,529 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#define _USE_MATH_DEFINES +#include +#include +#include "NavEditor/Include/imgui.h" +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_opengl.h" + +// Some math headers don't have PI defined. +static const float PI = 3.14159265f; + +void imguifree(void* ptr, void* userptr); +void* imguimalloc(size_t size, void* userptr); + +#define STBTT_malloc(x,y) imguimalloc(x,y) +#define STBTT_free(x,y) imguifree(x,y) +#define STB_TRUETYPE_IMPLEMENTATION +#include "thirdparty/imgui/include/imstb_truetype.h" + +void imguifree(void* ptr, void* /*userptr*/) +{ + free(ptr); +} + +void* imguimalloc(size_t size, void* /*userptr*/) +{ + return malloc(size); +} + +static const unsigned TEMP_COORD_COUNT = 100; +static float g_tempCoords[TEMP_COORD_COUNT*2]; +static float g_tempNormals[TEMP_COORD_COUNT*2]; + +static const int CIRCLE_VERTS = 8*4; +static float g_circleVerts[CIRCLE_VERTS*2]; + +static stbtt_bakedchar g_cdata[96]; // ASCII 32..126 is 95 glyphs +static GLuint g_ftex = 0; + +inline unsigned int RGBA(unsigned char r, unsigned char g, unsigned char b, unsigned char a) +{ + return (r) | (g << 8) | (b << 16) | (a << 24); +} + +static void drawPolygon(const float* coords, unsigned numCoords, float r, unsigned int col) +{ + if (numCoords > TEMP_COORD_COUNT) numCoords = TEMP_COORD_COUNT; + + for (unsigned i = 0, j = numCoords-1; i < numCoords; j=i++) + { + const float* v0 = &coords[j*2]; + const float* v1 = &coords[i*2]; + float dx = v1[0] - v0[0]; + float dy = v1[1] - v0[1]; + float d = sqrtf(dx*dx+dy*dy); + if (d > 0) + { + d = 1.0f/d; + dx *= d; + dy *= d; + } + g_tempNormals[j*2+0] = dy; + g_tempNormals[j*2+1] = -dx; + } + + for (unsigned i = 0, j = numCoords-1; i < numCoords; j=i++) + { + float dlx0 = g_tempNormals[j*2+0]; + float dly0 = g_tempNormals[j*2+1]; + float dlx1 = g_tempNormals[i*2+0]; + float dly1 = g_tempNormals[i*2+1]; + float dmx = (dlx0 + dlx1) * 0.5f; + float dmy = (dly0 + dly1) * 0.5f; + float dmr2 = dmx*dmx + dmy*dmy; + if (dmr2 > 0.000001f) + { + float scale = 1.0f / dmr2; + if (scale > 10.0f) scale = 10.0f; + dmx *= scale; + dmy *= scale; + } + g_tempCoords[i*2+0] = coords[i*2+0]+dmx*r; + g_tempCoords[i*2+1] = coords[i*2+1]+dmy*r; + } + + unsigned int colTrans = RGBA(col&0xff, (col>>8)&0xff, (col>>16)&0xff, 0); + + glBegin(GL_TRIANGLES); + + glColor4ubv((GLubyte*)&col); + + for (unsigned i = 0, j = numCoords-1; i < numCoords; j=i++) + { + glVertex2fv(&coords[i*2]); + glVertex2fv(&coords[j*2]); + glColor4ubv((GLubyte*)&colTrans); + glVertex2fv(&g_tempCoords[j*2]); + + glVertex2fv(&g_tempCoords[j*2]); + glVertex2fv(&g_tempCoords[i*2]); + + glColor4ubv((GLubyte*)&col); + glVertex2fv(&coords[i*2]); + } + + glColor4ubv((GLubyte*)&col); + for (unsigned i = 2; i < numCoords; ++i) + { + glVertex2fv(&coords[0]); + glVertex2fv(&coords[(i-1)*2]); + glVertex2fv(&coords[i*2]); + } + + glEnd(); +} + +static void drawRect(float x, float y, float w, float h, float fth, unsigned int col) +{ + float verts[4*2] = + { + x+0.5f, y+0.5f, + x+w-0.5f, y+0.5f, + x+w-0.5f, y+h-0.5f, + x+0.5f, y+h-0.5f, + }; + drawPolygon(verts, 4, fth, col); +} + +/* +static void drawEllipse(float x, float y, float w, float h, float fth, unsigned int col) +{ + float verts[CIRCLE_VERTS*2]; + const float* cverts = g_circleVerts; + float* v = verts; + + for (int i = 0; i < CIRCLE_VERTS; ++i) + { + *v++ = x + cverts[i*2]*w; + *v++ = y + cverts[i*2+1]*h; + } + + drawPolygon(verts, CIRCLE_VERTS, fth, col); +} +*/ + +static void drawRoundedRect(float x, float y, float w, float h, float r, float fth, unsigned int col) +{ + const unsigned n = CIRCLE_VERTS/4; + float verts[(n+1)*4*2]; + const float* cverts = g_circleVerts; + float* v = verts; + + for (unsigned i = 0; i <= n; ++i) + { + *v++ = x+w-r + cverts[i*2]*r; + *v++ = y+h-r + cverts[i*2+1]*r; + } + + for (unsigned i = n; i <= n*2; ++i) + { + *v++ = x+r + cverts[i*2]*r; + *v++ = y+h-r + cverts[i*2+1]*r; + } + + for (unsigned i = n*2; i <= n*3; ++i) + { + *v++ = x+r + cverts[i*2]*r; + *v++ = y+r + cverts[i*2+1]*r; + } + + for (unsigned i = n*3; i < n*4; ++i) + { + *v++ = x+w-r + cverts[i*2]*r; + *v++ = y+r + cverts[i*2+1]*r; + } + *v++ = x+w-r + cverts[0]*r; + *v++ = y+r + cverts[1]*r; + + drawPolygon(verts, (n+1)*4, fth, col); +} + + +static void drawLine(float x0, float y0, float x1, float y1, float r, float fth, unsigned int col) +{ + float dx = x1-x0; + float dy = y1-y0; + float d = sqrtf(dx*dx+dy*dy); + if (d > 0.0001f) + { + d = 1.0f/d; + dx *= d; + dy *= d; + } + float nx = dy; + float ny = -dx; + float verts[4*2]; + r -= fth; + r *= 0.5f; + if (r < 0.01f) r = 0.01f; + dx *= r; + dy *= r; + nx *= r; + ny *= r; + + verts[0] = x0-dx-nx; + verts[1] = y0-dy-ny; + + verts[2] = x0-dx+nx; + verts[3] = y0-dy+ny; + + verts[4] = x1+dx+nx; + verts[5] = y1+dy+ny; + + verts[6] = x1+dx-nx; + verts[7] = y1+dy-ny; + + drawPolygon(verts, 4, fth, col); +} + + +//bool imguiRenderGLInit(const char* fontpath) +//{ +// for (int i = 0; i < CIRCLE_VERTS; ++i) +// { +// float a = (float)i/(float)CIRCLE_VERTS * PI*2; +// g_circleVerts[i*2+0] = cosf(a); +// g_circleVerts[i*2+1] = sinf(a); +// } +// +// // Load font. +// FILE* fp = fopen(fontpath, "rb"); +// if (!fp) return false; +// if (fseek(fp, 0, SEEK_END) != 0) +// { +// fclose(fp); +// return false; +// } +// long size = ftell(fp); +// if (size < 0) +// { +// fclose(fp); +// return false; +// } +// if (fseek(fp, 0, SEEK_SET) != 0) +// { +// fclose(fp); +// return false; +// } +// +// unsigned char* ttfBuffer = (unsigned char*)malloc(size); +// if (!ttfBuffer) +// { +// fclose(fp); +// return false; +// } +// +// size_t readLen = fread(ttfBuffer, 1, size, fp); +// fclose(fp); +// if (readLen != static_cast(size)) +// { +// free(ttfBuffer); +// return false; +// } +// +// fp = 0; +// +// unsigned char* bmap = (unsigned char*)malloc(512*512); +// if (!bmap) +// { +// free(ttfBuffer); +// return false; +// } +// +// stbtt_BakeFontBitmap(ttfBuffer,0, 15.0f, bmap,512,512, 32,96, g_cdata); +// +// // can free ttf_buffer at this point +// glGenTextures(1, &g_ftex); +// glBindTexture(GL_TEXTURE_2D, g_ftex); +// glTexImage2D(GL_TEXTURE_2D, 0, GL_ALPHA, 512,512, 0, GL_ALPHA, GL_UNSIGNED_BYTE, bmap); +// glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); +// glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); +// +// free(ttfBuffer); +// free(bmap); +// +// return true; +//} + +bool imguiRenderGLInit(const unsigned int* ttfBuffer) +{ + for (int i = 0; i < CIRCLE_VERTS; ++i) + { + float a = (float)i / (float)CIRCLE_VERTS * PI * 2; + g_circleVerts[i * 2 + 0] = cosf(a); + g_circleVerts[i * 2 + 1] = sinf(a); + } + + unsigned char* bmap = (unsigned char*)malloc(512 * 512); + if (!bmap) + { + return false; + } + + stbtt_BakeFontBitmap((unsigned char*)ttfBuffer, 0, 15.0f, bmap, 512, 512, 32, 96, g_cdata); + + // can free ttf_buffer at this point + glGenTextures(1, &g_ftex); + glBindTexture(GL_TEXTURE_2D, g_ftex); + glTexImage2D(GL_TEXTURE_2D, 0, GL_ALPHA, 512, 512, 0, GL_ALPHA, GL_UNSIGNED_BYTE, bmap); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + + free(bmap); + + return true; +} + +void imguiRenderGLDestroy() +{ + if (g_ftex) + { + glDeleteTextures(1, &g_ftex); + g_ftex = 0; + } +} + +static void getBakedQuad(stbtt_bakedchar *chardata, int pw, int ph, int char_index, + float *xpos, float *ypos, stbtt_aligned_quad *q) +{ + stbtt_bakedchar *b = chardata + char_index; + int round_x = STBTT_ifloor(*xpos + b->xoff); + int round_y = STBTT_ifloor(*ypos - b->yoff); + + q->x0 = (float)round_x; + q->y0 = (float)round_y; + q->x1 = (float)round_x + b->x1 - b->x0; + q->y1 = (float)round_y - b->y1 + b->y0; + + q->s0 = b->x0 / (float)pw; + q->t0 = b->y0 / (float)pw; + q->s1 = b->x1 / (float)ph; + q->t1 = b->y1 / (float)ph; + + *xpos += b->xadvance; +} + +static const float g_tabStops[4] = {150, 210, 270, 330}; + +static float getTextLength(stbtt_bakedchar *chardata, const char* text) +{ + float xpos = 0; + float len = 0; + while (*text) + { + int c = (unsigned char)*text; + if (c == '\t') + { + for (int i = 0; i < 4; ++i) + { + if (xpos < g_tabStops[i]) + { + xpos = g_tabStops[i]; + break; + } + } + } + else if (c >= 32 && c < 128) + { + stbtt_bakedchar *b = chardata + c-32; + int round_x = STBTT_ifloor((xpos + b->xoff) + 0.5); + len = round_x + b->x1 - b->x0 + 0.5f; + xpos += b->xadvance; + } + ++text; + } + return len; +} + +static void drawText(float x, float y, const char *text, int align, unsigned int col) +{ + if (!g_ftex) return; + if (!text) return; + + if (align == IMGUI_ALIGN_CENTER) + x -= getTextLength(g_cdata, text)/2; + else if (align == IMGUI_ALIGN_RIGHT) + x -= getTextLength(g_cdata, text); + + glColor4ub(col&0xff, (col>>8)&0xff, (col>>16)&0xff, (col>>24)&0xff); + + glEnable(GL_TEXTURE_2D); + + // assume orthographic projection with units = screen pixels, origin at top left + glBindTexture(GL_TEXTURE_2D, g_ftex); + + glBegin(GL_TRIANGLES); + + const float ox = x; + + while (*text) + { + int c = (unsigned char)*text; + if (c == '\t') + { + for (int i = 0; i < 4; ++i) + { + if (x < g_tabStops[i]+ox) + { + x = g_tabStops[i]+ox; + break; + } + } + } + else if (c >= 32 && c < 128) + { + stbtt_aligned_quad q; + getBakedQuad(g_cdata, 512,512, c-32, &x,&y,&q); + + glTexCoord2f(q.s0, q.t0); + glVertex2f(q.x0, q.y0); + glTexCoord2f(q.s1, q.t1); + glVertex2f(q.x1, q.y1); + glTexCoord2f(q.s1, q.t0); + glVertex2f(q.x1, q.y0); + + glTexCoord2f(q.s0, q.t0); + glVertex2f(q.x0, q.y0); + glTexCoord2f(q.s0, q.t1); + glVertex2f(q.x0, q.y1); + glTexCoord2f(q.s1, q.t1); + glVertex2f(q.x1, q.y1); + } + ++text; + } + + glEnd(); + glDisable(GL_TEXTURE_2D); +} + + +void imguiRenderGLDraw() +{ + const imguiGfxCmd* q = imguiGetRenderQueue(); + int nq = imguiGetRenderQueueSize(); + + const float s = 1.0f/8.0f; + + glDisable(GL_SCISSOR_TEST); + for (int i = 0; i < nq; ++i) + { + const imguiGfxCmd& cmd = q[i]; + if (cmd.type == IMGUI_GFXCMD_RECT) + { + if (cmd.rect.r == 0) + { + drawRect((float)cmd.rect.x*s+0.5f, (float)cmd.rect.y*s+0.5f, + (float)cmd.rect.w*s-1, (float)cmd.rect.h*s-1, + 1.0f, cmd.col); + } + else + { + drawRoundedRect((float)cmd.rect.x*s+0.5f, (float)cmd.rect.y*s+0.5f, + (float)cmd.rect.w*s-1, (float)cmd.rect.h*s-1, + (float)cmd.rect.r*s, 1.0f, cmd.col); + } + } + else if (cmd.type == IMGUI_GFXCMD_LINE) + { + drawLine(cmd.line.x0*s, cmd.line.y0*s, cmd.line.x1*s, cmd.line.y1*s, cmd.line.r*s, 1.0f, cmd.col); + } + else if (cmd.type == IMGUI_GFXCMD_TRIANGLE) + { + if (cmd.flags == 1) + { + const float verts[3*2] = + { + (float)cmd.rect.x*s+0.5f, (float)cmd.rect.y*s+0.5f, + (float)cmd.rect.x*s+0.5f+(float)cmd.rect.w*s-1, (float)cmd.rect.y*s+0.5f+(float)cmd.rect.h*s/2-0.5f, + (float)cmd.rect.x*s+0.5f, (float)cmd.rect.y*s+0.5f+(float)cmd.rect.h*s-1, + }; + drawPolygon(verts, 3, 1.0f, cmd.col); + } + if (cmd.flags == 2) + { + const float verts[3*2] = + { + (float)cmd.rect.x*s+0.5f, (float)cmd.rect.y*s+0.5f+(float)cmd.rect.h*s-1, + (float)cmd.rect.x*s+0.5f+(float)cmd.rect.w*s/2-0.5f, (float)cmd.rect.y*s+0.5f, + (float)cmd.rect.x*s+0.5f+(float)cmd.rect.w*s-1, (float)cmd.rect.y*s+0.5f+(float)cmd.rect.h*s-1, + }; + drawPolygon(verts, 3, 1.0f, cmd.col); + } + } + else if (cmd.type == IMGUI_GFXCMD_TEXT) + { + drawText(cmd.text.x, cmd.text.y, cmd.text.text, cmd.text.align, cmd.col); + } + else if (cmd.type == IMGUI_GFXCMD_SCISSOR) + { + if (cmd.flags) + { + glEnable(GL_SCISSOR_TEST); + glScissor(cmd.rect.x, cmd.rect.y, cmd.rect.w, cmd.rect.h); + } + else + { + glDisable(GL_SCISSOR_TEST); + } + } + } + glDisable(GL_SCISSOR_TEST); +} diff --git a/r5dev/naveditor/include/ChunkyTriMesh.h b/r5dev/naveditor/include/ChunkyTriMesh.h new file mode 100644 index 00000000..e63330bc --- /dev/null +++ b/r5dev/naveditor/include/ChunkyTriMesh.h @@ -0,0 +1,62 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef CHUNKYTRIMESH_H +#define CHUNKYTRIMESH_H + +struct rcChunkyTriMeshNode +{ + float bmin[2]; + float bmax[2]; + int i; + int n; +}; + +struct rcChunkyTriMesh +{ + inline rcChunkyTriMesh() : nodes(0), nnodes(0), tris(0), ntris(0), maxTrisPerChunk(0) {}; + inline ~rcChunkyTriMesh() { delete [] nodes; delete [] tris; } + + rcChunkyTriMeshNode* nodes; + int nnodes; + int* tris; + int ntris; + int maxTrisPerChunk; + +private: + // Explicitly disabled copy constructor and copy assignment operator. + rcChunkyTriMesh(const rcChunkyTriMesh&); + rcChunkyTriMesh& operator=(const rcChunkyTriMesh&); +}; + +/// Creates partitioned triangle mesh (AABB tree), +/// where each node contains at max trisPerChunk triangles. +bool rcCreateChunkyTriMesh(const float* verts, const int* tris, int ntris, + int trisPerChunk, rcChunkyTriMesh* cm); + +/// Returns the chunk indices which overlap the input rectable. +int rcGetChunksOverlappingRect(const rcChunkyTriMesh* cm, float bmin[2], float bmax[2], int* ids, const int maxIds); + +/// Returns the chunk indices which overlap the input rectable. Return value is "we are done". Can be reinvoked to continue +int rcGetChunksOverlappingRect(const rcChunkyTriMesh* cm, float bmin[2], float bmax[2], int* ids, const int maxIds,int& count_returned,int& current_idx); + +/// Returns the chunk indices which overlap the input segment. +int rcGetChunksOverlappingSegment(const rcChunkyTriMesh* cm, float p[2], float q[2], int* ids, const int maxIds); + + +#endif // CHUNKYTRIMESH_H diff --git a/r5dev/naveditor/include/ConvexVolumeTool.h b/r5dev/naveditor/include/ConvexVolumeTool.h new file mode 100644 index 00000000..1f56cb07 --- /dev/null +++ b/r5dev/naveditor/include/ConvexVolumeTool.h @@ -0,0 +1,55 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef CONVEXVOLUMETOOL_H +#define CONVEXVOLUMETOOL_H + +#include "Sample.h" + +// Tool to create convex volumess for InputGeom + +class ConvexVolumeTool : public SampleTool +{ + Sample* m_sample; + int m_areaType; + float m_polyOffset; + float m_boxHeight; + float m_boxDescent; + + static const int MAX_PTS = 12; + float m_pts[MAX_PTS*3]; + int m_npts; + int m_hull[MAX_PTS]; + int m_nhull; + +public: + ConvexVolumeTool(); + + virtual int type() { return TOOL_CONVEX_VOLUME; } + virtual void init(Sample* sample); + virtual void reset(); + virtual void handleMenu(); + virtual void handleClick(const float* s, const float* p, bool shift); + virtual void handleToggle(); + virtual void handleStep(); + virtual void handleUpdate(const float dt); + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); +}; + +#endif // CONVEXVOLUMETOOL_H diff --git a/r5dev/naveditor/include/CrowdTool.h b/r5dev/naveditor/include/CrowdTool.h new file mode 100644 index 00000000..7af2a5b1 --- /dev/null +++ b/r5dev/naveditor/include/CrowdTool.h @@ -0,0 +1,144 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef CROWDTOOL_H +#define CROWDTOOL_H + +#include "NavEditor/Include/Sample.h" +#include "Detour/Include/DetourNavMesh.h" +#include "DetourCrowd/Include/DetourObstacleAvoidance.h" +#include "NavEditor/Include/ValueHistory.h" +#include "DetourCrowd/Include/DetourCrowd.h" + +// Tool to create crowds. + +struct CrowdToolParams +{ + bool m_expandSelectedDebugDraw; + bool m_showCorners; + bool m_showCollisionSegments; + bool m_showPath; + bool m_showVO; + bool m_showOpt; + bool m_showNeis; + + bool m_expandDebugDraw; + bool m_showLabels; + bool m_showGrid; + bool m_showNodes; + bool m_showPerfGraph; + bool m_showDetailAll; + + bool m_expandOptions; + bool m_anticipateTurns; + bool m_optimizeVis; + bool m_optimizeTopo; + bool m_obstacleAvoidance; + float m_obstacleAvoidanceType; + bool m_separation; + float m_separationWeight; +}; + +class CrowdToolState : public SampleToolState +{ + Sample* m_sample; + dtNavMesh* m_nav; + dtCrowd* m_crowd; + + float m_targetPos[3]; + dtPolyRef m_targetRef; + + dtCrowdAgentDebugInfo m_agentDebug; + dtObstacleAvoidanceDebugData* m_vod; + + static const int AGENT_MAX_TRAIL = 64; + static const int MAX_AGENTS = 128; + struct AgentTrail + { + float trail[AGENT_MAX_TRAIL*3]; + int htrail; + }; + AgentTrail m_trails[MAX_AGENTS]; + + ValueHistory m_crowdTotalTime; + ValueHistory m_crowdSampleCount; + + CrowdToolParams m_toolParams; + + bool m_run; + +public: + CrowdToolState(); + virtual ~CrowdToolState(); + + virtual void init(class Sample* sample); + virtual void reset(); + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); + virtual void handleUpdate(const float dt); + + inline bool isRunning() const { return m_run; } + inline void setRunning(const bool s) { m_run = s; } + + void addAgent(const float* pos); + void removeAgent(const int idx); + void hilightAgent(const int idx); + void updateAgentParams(); + int hitTestAgents(const float* s, const float* p); + void setMoveTarget(const float* p, bool adjust); + void updateTick(const float dt); + + inline CrowdToolParams* getToolParams() { return &m_toolParams; } + +private: + // Explicitly disabled copy constructor and copy assignment operator. + CrowdToolState(const CrowdToolState&); + CrowdToolState& operator=(const CrowdToolState&); +}; + + +class CrowdTool : public SampleTool +{ + Sample* m_sample; + CrowdToolState* m_state; + + enum ToolMode + { + TOOLMODE_CREATE, + TOOLMODE_MOVE_TARGET, + TOOLMODE_SELECT, + TOOLMODE_TOGGLE_POLYS, + }; + ToolMode m_mode; + +public: + CrowdTool(); + + virtual int type() { return TOOL_CROWD; } + virtual void init(Sample* sample); + virtual void reset(); + virtual void handleMenu(); + virtual void handleClick(const float* s, const float* p, bool shift); + virtual void handleToggle(); + virtual void handleStep(); + virtual void handleUpdate(const float dt); + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); +}; + +#endif // CROWDTOOL_H diff --git a/r5dev/naveditor/include/DroidSans.h b/r5dev/naveditor/include/DroidSans.h new file mode 100644 index 00000000..f448817e --- /dev/null +++ b/r5dev/naveditor/include/DroidSans.h @@ -0,0 +1,2273 @@ +// File: 'DroidSans.ttf' (108796 bytes) +// Exported using binary_to_compressed_c.cpp +static const unsigned int droidsans_size = 108796; +static const unsigned int droidsans_data[108796 / 4] = +{ + 0x00000100, 0x00011100, 0x10000400, 0x4853544c, 0xcab629e9, 0x3c0b0000, 0x55020000, 0x322f534f, 0x5855839f, 0x98010000, 0x60000000, 0x70616d63, + 0x22df2fb7, 0xd8390000, 0xf0030000, 0x20747663, 0xc03d9239, 0xb0470000, 0xfc010000, 0x6d677066, 0xb023d373, 0xc83d0000, 0x05070000, 0x70736167, + 0x07000400, 0xf0a80100, 0x0c000000, 0x66796c67, 0xca80c86e, 0x504e0000, 0xbc060100, 0x786d6468, 0x35627744, 0x940d0000, 0x442c0000, 0x64616568, + 0x58575cea, 0x1c010000, 0x36000000, 0x61656868, 0x6a08040f, 0x54010000, 0x24000000, 0x78746d68, 0xefee7120, 0xf8010000, 0x42090000, 0x6e72656b, + 0x708b22a2, 0x0c550100, 0x80370000, 0x61636f6c, 0x84e860a6, 0xac490000, 0xa4040000, 0x7078616d, 0x3e02d604, 0x78010000, 0x20000000, 0x656d616e, + 0x04a4c6f9, 0x8c8c0100, 0x6a0b0000, 0x74736f70, 0x41bc9e78, 0xf8970100, 0xf6100000, 0x70657270, 0xa9eef93b, 0xd0440000, 0xdd020000, 0x00000100, + 0x00000100, 0x57b78b77, 0xf53c0f5f, 0x00081900, 0x00000000, 0x00339ac1, 0x00000000, 0x70d900c2, 0x14fea0fe, 0xd907f408, 0x06000000, 0x00000200, + 0x00000000, 0x00000100, 0x1dfe6d07, 0x5a090000, 0xa2fea0fe, 0x0100f408, 0x00000000, 0x00000000, 0x00000000, 0x50020000, 0x00000100, 0x91005102, + 0x7a001600, 0x02000500, 0x2f001000, 0x00005a00, 0x0101fc01, 0x01000300, 0x68040300, 0x05009001, 0x9a050000, 0x00003305, 0x9a051e01, 0x00003305, + 0x6600d003, 0x0000f201, 0x06060b02, 0x02040803, 0x00a00402, 0x0040af02, 0x00005b20, 0x00002800, 0x41310000, 0x40004353, 0xfdff2000, 0x14fe1f06, + 0x6d078400, 0x0020e301, 0x00009f00, 0x4a040000, 0x0000b605, 0x00002000, 0xc100cd04, 0x00000000, 0x00001404, 0x00001402, 0x93002702, 0x85003703, + 0x33002b05, 0x7b006804, 0x66009a06, 0x6d009e05, 0x8500cf01, 0x52006802, 0x3d006802, 0x52006804, 0x66006804, 0x3f000002, 0x52009302, 0x93002502, + 0x1400fc02, 0x62006804, 0xb2006804, 0x60006804, 0x52006804, 0x17006804, 0x83006804, 0x71006804, 0x5a006804, 0x6a006804, 0x6a006804, 0x93002502, + 0x3f002502, 0x66006804, 0x66006804, 0x66006804, 0x25006803, 0x6d00ee06, 0x0000dd04, 0xc700f804, 0x7d00d304, 0xc7007905, 0xc7003904, 0xc700ee03, + 0x7d008505, 0xc7009c05, 0x5200b602, 0x48ff2b02, 0xc700a204, 0xc700ee03, 0xc700f606, 0xc700d505, 0x7d00f005, 0xc7009c04, 0x7d00ee05, 0xc700b804, + 0x68002704, 0x14002704, 0xb8009605, 0x00008b04, 0x14001207, 0x00006004, 0x00003704, 0x52005004, 0xa4006d02, 0x1700fc02, 0x33006d02, 0x29004204, + 0xfcff4a03, 0x89019e04, 0x5e003f04, 0xae00b004, 0x7100b403, 0x7100b004, 0x71004804, 0x1d00a202, 0x25002504, 0xae00b604, 0xa0001202, 0xbcff1202, + 0xae00f803, 0xae001202, 0xae002b07, 0xae00b604, 0x71009e04, 0xae00b004, 0x7100b004, 0xae003103, 0x5a009c03, 0x2100b602, 0xa400b604, 0x0000d503, + 0x1400f805, 0x23000004, 0x0a00e903, 0x52008703, 0x3d00d502, 0xe9016804, 0x3300d502, 0x66006804, 0x00001402, 0x93002702, 0xbc006804, 0x44006804, + 0x7b006804, 0x1d006804, 0xe9016804, 0x7900e303, 0x33019e04, 0x6400a806, 0x4400a602, 0x5200e503, 0x66006804, 0x52009302, 0x6400a806, 0xfaff0004, + 0x7b006d03, 0x66006804, 0x3100a602, 0x1f00a602, 0x89019e04, 0xae00c104, 0x71003d05, 0x93002502, 0x2300a401, 0x3f00a602, 0x4200cd02, 0x5400e503, + 0x3f00e505, 0x2c00e505, 0x1f00e505, 0x44006803, 0x0000dd04, 0x0000dd04, 0x0000dd04, 0x0000dd04, 0x0000dd04, 0x0000dd04, 0xfeffd106, 0x7d00d304, + 0xc7003904, 0xc7003904, 0xc7003904, 0xc7003904, 0x3e00b602, 0x5200b602, 0x1100b602, 0x4000b602, 0x2f007905, 0xc700d505, 0x7d00f005, 0x7d00f005, + 0x7d00f005, 0x7d00f005, 0x7d00f005, 0x8d006804, 0x7d00f005, 0xb8009605, 0xb8009605, 0xb8009605, 0xb8009605, 0x00003704, 0xc7009c04, 0xae00d104, + 0x5e003f04, 0x5e003f04, 0x5e003f04, 0x5e003f04, 0x5e003f04, 0x5e003f04, 0x5e00aa06, 0x7100b403, 0x71004804, 0x71004804, 0x71004804, 0x71004804, + 0xdeff1202, 0xae001202, 0xbdff1202, 0xeeff1202, 0x6f009e04, 0xae00b604, 0x71009e04, 0x71009e04, 0x71009e04, 0x71009e04, 0x71009e04, 0x66006804, + 0x73009e04, 0xa400b604, 0xa400b604, 0xa400b604, 0xa400b604, 0x0a00e903, 0xae00b004, 0x0a00e903, 0x0000dd04, 0x5e003f04, 0x0000dd04, 0x5e003f04, + 0x0000dd04, 0x5e003f04, 0x7d00d304, 0x7100b403, 0x7d00d304, 0x7100b403, 0x7d00d304, 0x7100b403, 0x7d00d304, 0x7100b403, 0xc7007905, 0x71000005, + 0x2f007905, 0x7100b004, 0xc7003904, 0x71004804, 0xc7003904, 0x71004804, 0xc7003904, 0x71004804, 0xc7003904, 0x71004804, 0xc7003904, 0x71004804, + 0x7d008505, 0x25002504, 0x7d008505, 0x25002504, 0x7d008505, 0x25002504, 0x7d008505, 0x25002504, 0xc7009c05, 0xae00b604, 0x00009c05, 0x1200b604, + 0xf5ffb602, 0xa3ff1202, 0x3d00b602, 0xebff1202, 0x3000b602, 0xdeff1202, 0x5200b602, 0x44001202, 0x5200b602, 0xae001202, 0x5200e104, 0xa0002504, + 0x48ff2b02, 0xbcff1202, 0xc700a204, 0xae00f803, 0xae00f803, 0xc700ee03, 0xab001202, 0xc700ee03, 0x66001202, 0xc700ee03, 0xae006202, 0xc700ee03, + 0xae006602, 0x1d00ee03, 0xf6ff1202, 0xc700d505, 0xae00b604, 0xc700d505, 0xae00b604, 0xc700d505, 0xae00b604, 0xffff4605, 0xc700d505, 0xae00b604, + 0x7d00f005, 0x71009e04, 0x7d00f005, 0x71009e04, 0x7d00f005, 0x71009e04, 0x7d001f07, 0x71004807, 0xc700b804, 0xae003103, 0xc700b804, 0x60003103, + 0xc700b804, 0x72003103, 0x68002704, 0x5a009c03, 0x68002704, 0x5a009c03, 0x68002704, 0x5a009c03, 0x68002704, 0x5a009c03, 0x14002704, 0x2100b602, + 0x14002704, 0x2100b602, 0x14002704, 0x2100b602, 0xb8009605, 0xa400b604, 0xb8009605, 0xa400b604, 0xb8009605, 0xa400b604, 0xb8009605, 0xa400b604, + 0xb8009605, 0xa400b604, 0xb8009605, 0xa400b604, 0x14001207, 0x1400f805, 0x00003704, 0x0a00e903, 0x00003704, 0x52005004, 0x52008703, 0x52005004, + 0x52008703, 0x52005004, 0x52008703, 0xae004c02, 0xcb006804, 0x0000dd04, 0x5e003f04, 0xfeffd106, 0x5e00aa06, 0x7d00f005, 0x73009e04, 0x68002704, + 0x5a009c03, 0x02019e04, 0x02019e04, 0x1b017504, 0x21019e04, 0xa0001202, 0x6d019e04, 0x1f007b01, 0x02019e04, 0xdf009e04, 0xf8019e04, 0x14019e04, + 0xe9ffdd04, 0x93002502, 0xe7ffc904, 0xe7ff1706, 0xe7ffb203, 0xe7ff4206, 0xe7ff5405, 0xe7ff4a06, 0xe4ff9e02, 0x0000dd04, 0xc700f804, 0xc700ee03, + 0x25008b04, 0xc7003904, 0x52005004, 0xc7009c05, 0x7d00f005, 0x5200b602, 0xc700a204, 0x00008b04, 0xc700f606, 0xc700d505, 0x52003f04, 0x7d00f005, + 0xc7008705, 0xc7009c04, 0x4e004204, 0x14002704, 0x00003704, 0x68002306, 0x00006004, 0x68002306, 0x4e00f005, 0x4000b602, 0x00003704, 0x7100b004, + 0x5a00a003, 0xae00b604, 0xa4009e02, 0xa400b804, 0x7100b004, 0xae00d104, 0x0a00e903, 0x6f009e04, 0x5a00a003, 0x7100a403, 0xae00b604, 0x71008704, + 0xa4009e02, 0xae00f803, 0xf2ff1704, 0xae00c104, 0x00002304, 0x7100a403, 0x71009e04, 0x19000a05, 0xa4009e04, 0x7100a403, 0x7100b404, 0x12009803, + 0xa400b804, 0x71009605, 0xecff3104, 0xa400ec05, 0x71000606, 0x15009e02, 0xa400b804, 0x71009e04, 0xa400b804, 0x71000606, 0xc7003904, 0x1400a005, + 0xc700ee03, 0x7d00d304, 0x68002704, 0x5200b602, 0x4000b602, 0x48ff2b02, 0x00004807, 0xc7007507, 0x1400a005, 0xc700a204, 0x1900ae04, 0xc7008705, + 0x0000dd04, 0xc7009c04, 0xc700f804, 0xc700ee03, 0x0e003305, 0xc7003904, 0x04008506, 0x48005c04, 0xc900d505, 0xc900d505, 0xc700a204, 0x00005a05, + 0xc700f606, 0xc7009c05, 0x7d00f005, 0xc7008705, 0xc7009c04, 0x7d00d304, 0x14002704, 0x1900ae04, 0x68002306, 0x00006004, 0xc7009a05, 0xa6004805, + 0xc700fa07, 0xc700fa07, 0x14003d05, 0xc7009606, 0xc7009c04, 0x3b00c104, 0xc7001d08, 0x1900b804, 0x5e003f04, 0x75008f04, 0xae008904, 0xae003303, + 0x29006004, 0x71004804, 0x0400c305, 0x4400ae03, 0xae00e904, 0xae00e904, 0xae00ee03, 0x10005e04, 0xae00be05, 0xae00e904, 0x71009e04, 0xae00c104, + 0xae00b004, 0x7100b403, 0x29008703, 0x0a00e903, 0x71009305, 0x23000004, 0xae00d104, 0x9a00ac04, 0xae00f406, 0xae000407, 0x29006d05, 0xae00fe05, + 0xae008904, 0x3700b203, 0xae007906, 0x23003f04, 0x71004804, 0x1200b604, 0xae003303, 0x7100b403, 0x5a009c03, 0xa0001202, 0xeeff1202, 0xbcff1202, + 0x10008306, 0xae00e506, 0x1200b604, 0xae00ee03, 0x0a00e903, 0xae00c104, 0xc700ee03, 0xae003303, 0x14001207, 0x1400f805, 0x14001207, 0x1400f805, + 0x14001207, 0x1400f805, 0x00003704, 0x0a00e903, 0x52000004, 0x52000008, 0x52000008, 0xfcff4a03, 0x17006601, 0x17006601, 0x3f000002, 0x19006601, + 0x1700e702, 0x1700e702, 0x3f008103, 0x7b00e303, 0x7b00f803, 0x96000203, 0x93006f06, 0x66005a09, 0x8500cf01, 0x85003703, 0x52004e02, 0x52004e02, + 0x9300f603, 0xa0fe0a01, 0x6a00f802, 0x60006804, 0x44006804, 0x96000006, 0x3f006804, 0x8d006806, 0x77000004, 0xc700e707, 0x2500fe05, 0x4e00f005, + 0x6600f404, 0x53000e06, 0x33000e06, 0x4f000e06, 0x71000e06, 0x6200a604, 0x29008b04, 0xc700ee05, 0x4a000c05, 0x66006804, 0x25006404, 0x7700a805, + 0x10001903, 0x66006804, 0x66006804, 0x66006804, 0x66006804, 0x6d00aa04, 0x1d00b404, 0x1d00b404, 0xcf009e04, 0xbcff1202, 0x87010004, 0x6f010004, + 0x7d010004, 0x2500a602, 0x0c00a602, 0x3b00a602, 0x2500a602, 0x2f00a602, 0x3100a602, 0x2100a602, 0x00000004, 0x00000008, 0x00000004, 0x00000008, + 0x0000ab02, 0x00000002, 0x00005501, 0x00006804, 0x00002502, 0x00009a01, 0x0000cd00, 0x00000000, 0x00000000, 0x54000008, 0x00005400, 0x51020000, + 0x1a010101, 0x01011828, 0x012b010a, 0x160b0a01, 0x0b051401, 0x0b0b0b0b, 0x0b0b0b0b, 0x0b14140b, 0x0b1c0b0b, 0x0b142226, 0x150b1201, 0x14151519, + 0x1322191c, 0x1b1b161e, 0x10101715, 0x05011226, 0x01012001, 0x0b1c0b11, 0x2e192a15, 0x0c0a0c0c, 0x0b152e18, 0x0b2a120b, 0x1a0b192e, 0x18191216, + 0x281a0b19, 0x0b0b0b0b, 0x04011518, 0x010b2714, 0x0b100104, 0x12011111, 0x1101140b, 0x0101270a, 0x26261c0a, 0x26262626, 0x01011401, 0x19190101, + 0x190b1919, 0x22222222, 0x15220b22, 0x26151515, 0x11111613, 0x11111111, 0x15151c16, 0x0c0c1515, 0x2e150c0c, 0x15151515, 0x2e150b15, 0x162e2e2e, + 0x1126160b, 0x11261126, 0x1c141c14, 0x1c141c14, 0x0b0b100b, 0x15011501, 0x15011501, 0x190b1501, 0x190b190b, 0x2e15190b, 0x0c192e15, 0x0c190c19, + 0x0c190c19, 0x0c151715, 0x140a0a15, 0x140c140c, 0x14241410, 0x192e190c, 0x182e192e, 0x15222e19, 0x15221522, 0x1216282a, 0x12161216, 0x2a1b2a1b, + 0x2a1b2a1b, 0x0b1b0b1b, 0x2e150b1b, 0x2e152e15, 0x2e152e15, 0x0b102e15, 0x12261626, 0x12121212, 0x260b3012, 0x22160111, 0x012a1b15, 0x01012401, + 0x01010101, 0x14260101, 0x150c2d15, 0x2616150d, 0x01171922, 0x19221512, 0x191c1915, 0x13152211, 0x19261b20, 0x19221910, 0x2e160b26, 0x0c0b1216, + 0x16161516, 0x0a16172e, 0x1818122d, 0x18131815, 0x19120118, 0x16181313, 0x18121512, 0x14190e01, 0x1519191b, 0x150e1928, 0x13261530, 0x011f1922, + 0x19191218, 0x151c1015, 0x14131522, 0x1019301b, 0x19191415, 0x16191017, 0x14111624, 0x15161419, 0x19191a19, 0x1913160a, 0x1c0b1215, 0x1a191619, + 0x19191216, 0x11191817, 0x2e151119, 0x0c2a1c14, 0x18180c0c, 0x12160a2e, 0x0b100117, 0x0b100b10, 0x01011626, 0x15150101, 0x15151519, 0x19191919, + 0x182b1419, 0x010c1919, 0x110b0b19, 0x2a01010b, 0x0c012219, 0x260c0c0c, 0x012b1e17, 0x0b171701, 0x190b0b0b, 0x0c191414, 0x11010101, 0x11111111, + 0x01011111, 0x01010101, 0x0101012e, 0x01010101, 0x00000001, 0x13000000, 0x54020000, 0x00040706, 0x02020203, 0x04050304, 0x03020201, 0x02020203, + 0x03030302, 0x03030303, 0x02030303, 0x03030302, 0x04040503, 0x03030404, 0x02020404, 0x04050303, 0x04040404, 0x04040303, 0x03030305, 0x03020202, + 0x04040302, 0x02030403, 0x02020403, 0x04050203, 0x02040404, 0x03040203, 0x03030404, 0x03020302, 0x03030202, 0x03030303, 0x03020503, 0x03050203, + 0x03030303, 0x02040403, 0x03020301, 0x03040404, 0x04040404, 0x04050404, 0x03030303, 0x02020202, 0x04040404, 0x03040404, 0x04040404, 0x04030304, + 0x04040404, 0x03050404, 0x03030303, 0x02020202, 0x04040404, 0x03040404, 0x04040404, 0x03040304, 0x04040404, 0x03040404, 0x03040304, 0x05040304, + 0x03030404, 0x03030303, 0x03030303, 0x03040304, 0x03040304, 0x04040404, 0x02020202, 0x02020202, 0x04040202, 0x03030202, 0x03020303, 0x03030302, + 0x04020302, 0x04040404, 0x04040504, 0x04040404, 0x05050404, 0x02040204, 0x03030204, 0x03030303, 0x02030303, 0x02030203, 0x04040404, 0x04040404, + 0x04040404, 0x03030405, 0x03030303, 0x02030303, 0x05030403, 0x03040405, 0x03030303, 0x01030203, 0x03030303, 0x05040204, 0x05040503, 0x03040402, + 0x04030304, 0x04030204, 0x04030405, 0x03040404, 0x06030603, 0x04030204, 0x04020403, 0x04030404, 0x04040303, 0x04040302, 0x04040304, 0x03040304, + 0x05040504, 0x04040206, 0x05030604, 0x02030403, 0x06060202, 0x04040305, 0x03040404, 0x03050304, 0x04030404, 0x04040405, 0x04030404, 0x04040306, + 0x06050606, 0x04060404, 0x02040404, 0x03040303, 0x04030404, 0x04040405, 0x03030304, 0x04040405, 0x05040606, 0x04060303, 0x03020403, 0x02020203, + 0x03040505, 0x02030403, 0x04050405, 0x03030405, 0x02060603, 0x01020101, 0x03030202, 0x07060203, 0x02020201, 0x03020103, 0x05030503, 0x04050603, + 0x05050504, 0x04040405, 0x04030304, 0x03030302, 0x04040403, 0x03030203, 0x03030303, 0x03030303, 0x06030603, 0x03010202, 0x00010102, 0x00060600, + 0x00040807, 0x03020204, 0x05060405, 0x04020202, 0x02020204, 0x04040403, 0x04040404, 0x02040404, 0x04040402, 0x04050603, 0x04040504, 0x02040505, + 0x05060304, 0x04050405, 0x05050404, 0x04040406, 0x04020302, 0x04040403, 0x03040403, 0x02020404, 0x04060204, 0x03040404, 0x04040203, 0x03040406, + 0x04020402, 0x04040202, 0x03040404, 0x03020604, 0x04060204, 0x03030403, 0x02050404, 0x03020301, 0x03050505, 0x05050505, 0x04060505, 0x04040404, + 0x04040404, 0x05050505, 0x04050505, 0x05050505, 0x04040405, 0x04040404, 0x03060404, 0x04040404, 0x02020202, 0x04040404, 0x04040404, 0x04040404, + 0x04040404, 0x04050405, 0x03040405, 0x03040304, 0x05050304, 0x04040405, 0x04040404, 0x04040404, 0x04050405, 0x04050405, 0x04050405, 0x02040204, + 0x02040204, 0x04050204, 0x04040202, 0x03020304, 0x03030302, 0x05020302, 0x05040504, 0x04050504, 0x04050405, 0x06060405, 0x03040304, 0x03040304, + 0x03040304, 0x02040304, 0x02040204, 0x04050405, 0x04050405, 0x04050405, 0x04040606, 0x04030404, 0x02030403, 0x06040504, 0x04040506, 0x04040403, + 0x01040204, 0x04040404, 0x05040205, 0x06050505, 0x03040502, 0x05040405, 0x05040405, 0x05040506, 0x04040405, 0x06040604, 0x04040405, 0x04020403, + 0x04040404, 0x04040303, 0x04040402, 0x04040304, 0x03040304, 0x06040604, 0x04040206, 0x05040604, 0x04040403, 0x07070204, 0x05040405, 0x03040405, + 0x04060405, 0x05040505, 0x05050506, 0x04040404, 0x05050406, 0x06050808, 0x04070404, 0x03040404, 0x03060404, 0x04030404, 0x04040405, 0x04030304, + 0x04040406, 0x06050606, 0x04060304, 0x03030404, 0x02020203, 0x03040606, 0x03030404, 0x06060606, 0x04040606, 0x03070704, 0x01020101, 0x03030303, + 0x08060303, 0x02020302, 0x04030103, 0x06040504, 0x05050704, 0x05050504, 0x05050405, 0x05040404, 0x04040403, 0x04040404, 0x04040204, 0x03030304, + 0x03030303, 0x07040704, 0x04010202, 0x00010102, 0x00070700, 0x00050908, 0x03020204, 0x06070405, 0x04020202, 0x02030204, 0x04040403, 0x04040404, + 0x02040404, 0x04040402, 0x05060704, 0x04040505, 0x02040606, 0x06080405, 0x05060506, 0x06060404, 0x05040508, 0x04020302, 0x05040503, 0x03040504, + 0x02020504, 0x05080204, 0x03050505, 0x04050304, 0x04040406, 0x04030403, 0x04040202, 0x04040404, 0x04030705, 0x04070304, 0x04040403, 0x02050505, + 0x04030402, 0x04060606, 0x06060606, 0x05070606, 0x04040404, 0x04040404, 0x06060605, 0x04060606, 0x06060606, 0x05050406, 0x04040404, 0x04070404, + 0x04040404, 0x02020202, 0x05050505, 0x04050505, 0x05050505, 0x04050405, 0x04060406, 0x04050406, 0x04050405, 0x06050405, 0x04040505, 0x04040404, + 0x04040404, 0x04060406, 0x04060406, 0x05060506, 0x02040204, 0x02040204, 0x04050204, 0x04050202, 0x04020404, 0x04030402, 0x06020402, 0x06050605, + 0x05060605, 0x05060506, 0x07070506, 0x03050305, 0x04040305, 0x04040404, 0x03040404, 0x03040304, 0x05060506, 0x05060506, 0x05060506, 0x04040608, + 0x05040504, 0x02040504, 0x07040604, 0x04050607, 0x04050504, 0x01050205, 0x05050505, 0x06050206, 0x06050605, 0x04050602, 0x06050406, 0x06050406, + 0x06040608, 0x04050506, 0x06050604, 0x05040406, 0x05020504, 0x05040505, 0x05050404, 0x05040402, 0x05050404, 0x04050405, 0x06040605, 0x05050206, + 0x06040605, 0x04040504, 0x08080204, 0x06050506, 0x04050506, 0x04080406, 0x06050606, 0x06060608, 0x05040505, 0x05060506, 0x07060808, 0x05080505, + 0x03050504, 0x04060404, 0x05040505, 0x05050506, 0x04040405, 0x05050406, 0x06050808, 0x04070404, 0x04030504, 0x02020204, 0x04050707, 0x03040504, + 0x06080608, 0x04040608, 0x03080804, 0x01020101, 0x04040303, 0x09060304, 0x02020302, 0x04030104, 0x06040604, 0x06060804, 0x06060605, 0x06060506, + 0x06040405, 0x04040403, 0x05050504, 0x04040205, 0x04040404, 0x04040404, 0x08040804, 0x04010203, 0x00010202, 0x00080800, 0x00050b09, 0x04020205, + 0x06080506, 0x06030302, 0x02030205, 0x05050503, 0x05050505, 0x02050505, 0x05050502, 0x05060804, 0x05050605, 0x03040606, 0x06080405, 0x05070507, + 0x06060405, 0x05060608, 0x05030303, 0x05050504, 0x04050504, 0x02020505, 0x05080205, 0x04050505, 0x05050404, 0x04050506, 0x05030503, 0x05050202, + 0x04050505, 0x05040705, 0x05070305, 0x04040504, 0x02060505, 0x05040402, 0x04080707, 0x06060606, 0x05080606, 0x05050505, 0x04040404, 0x07070606, + 0x05070707, 0x06060607, 0x05050606, 0x05050505, 0x04080505, 0x05050505, 0x02020202, 0x05050505, 0x05050505, 0x05050505, 0x05050505, 0x05060506, + 0x04050506, 0x04050405, 0x06060405, 0x05050506, 0x05050505, 0x05050505, 0x05060506, 0x05060506, 0x05060506, 0x02040204, 0x02040204, 0x04060204, + 0x05050203, 0x04020405, 0x04030402, 0x06020403, 0x06050605, 0x05060605, 0x05070507, 0x08090507, 0x04050405, 0x04050405, 0x04050405, 0x04040405, + 0x04040404, 0x05060506, 0x05060506, 0x05060506, 0x05060608, 0x05040506, 0x02040504, 0x08050605, 0x05050708, 0x05050504, 0x02050205, 0x05050505, + 0x07060206, 0x08070805, 0x05050602, 0x06050506, 0x06050407, 0x07050608, 0x04050506, 0x06060606, 0x05060407, 0x05020504, 0x05050505, 0x05050404, + 0x05050502, 0x06050405, 0x04050405, 0x06060605, 0x05050206, 0x06050605, 0x04050505, 0x09090304, 0x06060506, 0x05050506, 0x05080506, 0x07050606, + 0x06070608, 0x06040505, 0x06060606, 0x07070808, 0x05090505, 0x04050505, 0x04060505, 0x05050505, 0x05050506, 0x05040405, 0x05050506, 0x07060808, + 0x05070405, 0x04040505, 0x02020204, 0x05050808, 0x04050505, 0x06080608, 0x05060608, 0x04090905, 0x02020202, 0x04040404, 0x0b060304, 0x03030402, + 0x05040104, 0x07050805, 0x07080905, 0x08080806, 0x06060508, 0x07050506, 0x05050503, 0x06060605, 0x05050204, 0x04040405, 0x04040404, 0x09050905, + 0x05020203, 0x00010202, 0x00090900, 0x00060c0a, 0x04020305, 0x07080606, 0x06030302, 0x02030205, 0x05050504, 0x05050505, 0x02050505, 0x05050502, + 0x06060804, 0x05050606, 0x03040606, 0x06080405, 0x05070507, 0x06060405, 0x05060608, 0x05030403, 0x05050604, 0x04050504, 0x02020505, 0x05080205, + 0x04050505, 0x05050404, 0x04050506, 0x05040504, 0x05050203, 0x05050505, 0x05040806, 0x05080305, 0x04040504, 0x02060506, 0x05040402, 0x04070707, + 0x06060606, 0x06090606, 0x05050505, 0x04040404, 0x07070606, 0x05070707, 0x06060607, 0x05050606, 0x05050505, 0x04080505, 0x05050505, 0x02020202, + 0x05050505, 0x05050505, 0x05050505, 0x05050505, 0x05060506, 0x04060506, 0x04060406, 0x07060406, 0x05050506, 0x05050505, 0x05050505, 0x05060506, + 0x05060506, 0x05060506, 0x02040204, 0x02040204, 0x05060204, 0x05050203, 0x04020405, 0x04030402, 0x06020404, 0x06050605, 0x05060705, 0x05070507, + 0x08090507, 0x04050405, 0x04050405, 0x04050405, 0x04040405, 0x04040404, 0x05060506, 0x05060506, 0x05060506, 0x05060608, 0x05040506, 0x02040504, + 0x09050605, 0x05050708, 0x06060604, 0x02060306, 0x06060606, 0x07060206, 0x08070805, 0x05060602, 0x06050506, 0x06050407, 0x07050608, 0x04050506, + 0x08060806, 0x05060407, 0x05020504, 0x05050505, 0x05050404, 0x05050502, 0x06050405, 0x04050405, 0x06060605, 0x05050206, 0x07050605, 0x04050605, + 0x09090304, 0x06060507, 0x05060606, 0x05080506, 0x07050606, 0x06070608, 0x06040605, 0x06060608, 0x08070a0a, 0x050a0606, 0x04050505, 0x04060505, + 0x05050505, 0x05050506, 0x05040405, 0x05050506, 0x07070808, 0x05080405, 0x04040505, 0x02020204, 0x05050808, 0x04050505, 0x06080608, 0x05060608, + 0x040a0a05, 0x02020202, 0x04040404, 0x0c090304, 0x03030402, 0x05040104, 0x08050805, 0x07070905, 0x08080806, 0x07060508, 0x07050606, 0x05050503, + 0x06060605, 0x05050206, 0x04040405, 0x04040404, 0x0a050a05, 0x06020303, 0x00010203, 0x000a0a00, 0x00070d0b, 0x05030306, 0x08090607, 0x06030303, + 0x03040306, 0x06060604, 0x06060606, 0x03060606, 0x06060603, 0x07070a05, 0x05060807, 0x03030808, 0x080a0507, 0x07090609, 0x07080506, 0x07070609, + 0x06030403, 0x06060605, 0x04060605, 0x02020606, 0x060a0205, 0x04060606, 0x06060405, 0x05060608, 0x06040504, 0x06060303, 0x05050606, 0x05040906, + 0x06090406, 0x04040605, 0x03070606, 0x05040402, 0x05080808, 0x07070707, 0x07090707, 0x06060606, 0x03030303, 0x09090808, 0x06090909, 0x08080809, + 0x06060708, 0x06060606, 0x050a0606, 0x06060606, 0x02020202, 0x06060606, 0x06060606, 0x06060606, 0x06060606, 0x06070607, 0x05070607, 0x05070507, + 0x07080507, 0x06060608, 0x06060606, 0x06060606, 0x06080608, 0x06080608, 0x06080608, 0x02030203, 0x02030203, 0x05070203, 0x05070203, 0x05020505, + 0x05030502, 0x08020503, 0x08060806, 0x06080806, 0x06090609, 0x0a0a0609, 0x04070407, 0x05060407, 0x05060506, 0x04050506, 0x04050405, 0x06080608, + 0x06080608, 0x06080608, 0x06070809, 0x07050707, 0x02050705, 0x09060706, 0x0606090a, 0x06060605, 0x02060306, 0x06060606, 0x09070307, 0x09080904, + 0x06070704, 0x08070607, 0x07070309, 0x0905080a, 0x05060608, 0x09060907, 0x06070309, 0x06040605, 0x06060606, 0x06060505, 0x06060504, 0x08060506, + 0x05060506, 0x08060806, 0x06060408, 0x08060806, 0x03060706, 0x0a0a0303, 0x08060708, 0x06070707, 0x06090607, 0x07070808, 0x0809080a, 0x06050706, + 0x08080609, 0x09080b0b, 0x070c0707, 0x04060606, 0x05080607, 0x06050606, 0x06060607, 0x06060506, 0x06060608, 0x08080a0a, 0x06090506, 0x05040606, + 0x02020205, 0x0506090a, 0x04050606, 0x08090809, 0x06070809, 0x050b0b06, 0x02020202, 0x05040404, 0x0d090505, 0x03030503, 0x06040106, 0x09060906, + 0x09090b06, 0x09090907, 0x08070609, 0x08060607, 0x06060604, 0x06060706, 0x06060206, 0x04040406, 0x04040404, 0x0b060b06, 0x06020304, 0x00010203, + 0x000b0b00, 0x00070f0c, 0x05030306, 0x080a0708, 0x07040403, 0x03040307, 0x07070704, 0x07070707, 0x03070707, 0x07070703, 0x07070a05, 0x06060807, + 0x03040808, 0x090b0607, 0x07090709, 0x07080707, 0x0707070b, 0x06040404, 0x07070705, 0x03070706, 0x03030706, 0x070b0306, 0x04070707, 0x05070406, + 0x06050609, 0x07040704, 0x07070303, 0x06070707, 0x06040a07, 0x060a0407, 0x04040705, 0x03080707, 0x06040402, 0x05090909, 0x07070707, 0x070a0707, + 0x06060606, 0x04040404, 0x09090908, 0x07090909, 0x08080809, 0x07070708, 0x07070707, 0x060b0707, 0x07070707, 0x03030303, 0x07070707, 0x07070707, + 0x07070707, 0x05070507, 0x07070707, 0x06070707, 0x06070607, 0x08080607, 0x07060708, 0x07060706, 0x07060706, 0x06080608, 0x06080608, 0x07080708, + 0x03040304, 0x03040304, 0x06070304, 0x06070303, 0x06030606, 0x06040603, 0x09030604, 0x09070907, 0x07090807, 0x07090709, 0x0b0b0709, 0x04070407, + 0x06070407, 0x06070607, 0x04070607, 0x04070407, 0x07080708, 0x07080708, 0x07080708, 0x0507090b, 0x07060707, 0x03060706, 0x0a070707, 0x0707090b, + 0x07070706, 0x02070307, 0x07070707, 0x09070307, 0x09090906, 0x06070704, 0x08070607, 0x07070409, 0x0907090b, 0x07070708, 0x09070907, 0x07070409, + 0x07040706, 0x07050707, 0x07070506, 0x07050604, 0x08070606, 0x05070607, 0x09060907, 0x07070409, 0x09060907, 0x04070706, 0x0b0b0304, 0x08070709, + 0x06070707, 0x07090608, 0x08070909, 0x0809080b, 0x07070707, 0x08080709, 0x0a080b0b, 0x070c0707, 0x05070707, 0x06090706, 0x07060707, 0x07070709, + 0x05050607, 0x07070609, 0x09080b0b, 0x070a0607, 0x06050707, 0x03030306, 0x06070b0b, 0x05060705, 0x090b090b, 0x0507090b, 0x050c0c06, 0x02020202, + 0x05040404, 0x0f090505, 0x04040503, 0x07040206, 0x0a070a07, 0x090a0d06, 0x09090907, 0x08070709, 0x09070708, 0x07070705, 0x07070707, 0x06060307, + 0x04040406, 0x04040404, 0x0c060c06, 0x07020304, 0x00010203, 0x000c0c00, 0x0008100d, 0x05040307, 0x090b0708, 0x07040403, 0x03040307, 0x07070705, + 0x07070707, 0x03070707, 0x07070703, 0x08080b06, 0x06070908, 0x03050909, 0x090b0608, 0x080a070a, 0x07090707, 0x0707070b, 0x07040504, 0x08070805, + 0x04070806, 0x03030807, 0x080b0306, 0x05080808, 0x06080406, 0x0606070a, 0x07050705, 0x07070403, 0x06070707, 0x06050b08, 0x070b0407, 0x05050706, + 0x03090808, 0x06050503, 0x060a0a0a, 0x08080808, 0x080b0808, 0x07070707, 0x05050505, 0x0a0a0909, 0x070a0a0a, 0x0909090a, 0x08070709, 0x07070707, + 0x060b0707, 0x07070707, 0x03030303, 0x08080808, 0x07080808, 0x08080808, 0x06080608, 0x07080708, 0x06080708, 0x06080608, 0x09090608, 0x07070809, + 0x07070707, 0x07070707, 0x07090709, 0x07090709, 0x08090809, 0x03050305, 0x03050305, 0x06080305, 0x06080303, 0x06030606, 0x06040603, 0x09030604, + 0x09080908, 0x08090908, 0x080a080a, 0x0c0c080a, 0x05080508, 0x06070508, 0x06070607, 0x04070607, 0x04070407, 0x08090809, 0x08090809, 0x08090809, + 0x06070a0b, 0x07060707, 0x04060706, 0x0b070807, 0x07080a0b, 0x07080806, 0x02080308, 0x08080808, 0x0a080308, 0x0a090a06, 0x07080804, 0x09070707, + 0x0808050a, 0x0a07090b, 0x07070709, 0x0a070a07, 0x0807050a, 0x08040806, 0x08060808, 0x08080606, 0x08070604, 0x08080607, 0x06080608, 0x0a070a08, + 0x0808040b, 0x0a070b08, 0x05070807, 0x0c0c0305, 0x0908080a, 0x07080808, 0x070b0708, 0x09080909, 0x090a090b, 0x08070807, 0x0909070a, 0x0b090d0d, + 0x080d0808, 0x05070807, 0x06090707, 0x07060808, 0x08080809, 0x06050608, 0x08080709, 0x0a080b0b, 0x070b0607, 0x06050807, 0x03030306, 0x06080b0b, + 0x05060806, 0x0a0b0a0b, 0x06070a0b, 0x050d0d07, 0x02020202, 0x05050505, 0x10090505, 0x04040503, 0x07050206, 0x0a070b07, 0x0a0a0d07, 0x0a0a0a08, + 0x0907080a, 0x09070708, 0x07070705, 0x07070707, 0x07070307, 0x05050507, 0x05050505, 0x0d070d07, 0x07020304, 0x00010303, 0x000d0d00, 0x0008110e, + 0x06040407, 0x0a0c0809, 0x08040403, 0x04050408, 0x08080805, 0x08080808, 0x04080808, 0x08080804, 0x09090c06, 0x07070a08, 0x03050a0a, 0x0a0c0708, + 0x080b080b, 0x080a0707, 0x0807080d, 0x07040504, 0x08070806, 0x04080807, 0x04040807, 0x080d0407, 0x05080808, 0x07080507, 0x0707070a, 0x08050705, + 0x08080404, 0x07070808, 0x07050c08, 0x070c0508, 0x05050806, 0x04090808, 0x07050503, 0x060a0a0a, 0x09090909, 0x080c0909, 0x07070707, 0x05050505, + 0x0b0b0a0a, 0x080b0b0b, 0x0a0a0a0b, 0x0808070a, 0x07070707, 0x070c0707, 0x08080808, 0x04040404, 0x08080808, 0x08080808, 0x08080808, 0x07080708, + 0x07090709, 0x07080709, 0x07080708, 0x090a0708, 0x0807080a, 0x08070807, 0x08070807, 0x070a070a, 0x070a070a, 0x080a080a, 0x04050405, 0x04050405, + 0x08080405, 0x07080403, 0x07040707, 0x07050704, 0x0a040705, 0x0a080a08, 0x080a0a08, 0x080b080b, 0x0d0c080b, 0x05080508, 0x07070508, 0x07070707, + 0x05070707, 0x05070507, 0x080a080a, 0x080a080a, 0x080a080a, 0x07070a0d, 0x08070807, 0x04070807, 0x0c070908, 0x07080b0c, 0x08080807, 0x03080408, + 0x08080808, 0x0b080409, 0x0b090b06, 0x07090904, 0x0a080709, 0x0908050b, 0x0b080a0c, 0x0708080a, 0x0b080b07, 0x0807050b, 0x08040807, 0x08070808, + 0x08080607, 0x08070704, 0x09080707, 0x06080708, 0x0a070a08, 0x0808040b, 0x0a070b08, 0x05070807, 0x0d0d0305, 0x0a08080a, 0x07090809, 0x080b0709, + 0x09080a0a, 0x0a0b0a0c, 0x08070808, 0x090a080b, 0x0c090d0d, 0x080f0809, 0x05080807, 0x07090808, 0x08070808, 0x0808080a, 0x07050708, 0x08080709, + 0x0b090b0b, 0x080b0708, 0x07050808, 0x04040407, 0x07080c0c, 0x06070807, 0x0a0d0a0d, 0x07070a0d, 0x060e0e07, 0x03030303, 0x07060606, 0x110c0607, + 0x04040603, 0x08050207, 0x0b080c08, 0x0b0a0e07, 0x0b0b0b09, 0x0a09080b, 0x0a080809, 0x08080805, 0x08080708, 0x07070408, 0x05050507, 0x05050505, + 0x0e070e07, 0x08020405, 0x00010304, 0x000e0e00, 0x0009120f, 0x07040408, 0x0b0c080a, 0x08050504, 0x04050408, 0x08080806, 0x08080808, 0x04080808, + 0x08080804, 0x0a0a0d06, 0x08080a09, 0x04060b0a, 0x0b0d0709, 0x090b090b, 0x090b0807, 0x0808090e, 0x08050605, 0x09080906, 0x04080907, 0x04040808, + 0x080d0407, 0x06090909, 0x07080507, 0x0707070b, 0x08050705, 0x08080404, 0x07070808, 0x07050c09, 0x080c0508, 0x05050807, 0x040a0809, 0x07050503, + 0x060b0b0b, 0x0a0a0a0a, 0x090d0a0a, 0x08080808, 0x06060606, 0x0b0b0b0a, 0x080b0b0b, 0x0b0b0b0b, 0x0909080b, 0x08080808, 0x070d0808, 0x08080808, + 0x04040404, 0x09090809, 0x08090909, 0x08080809, 0x07090708, 0x080a080a, 0x0709080a, 0x07090709, 0x0a0a0709, 0x0808090a, 0x08080808, 0x08080808, + 0x080a080a, 0x080a080a, 0x080b080b, 0x04060406, 0x04060406, 0x08090406, 0x07090404, 0x07040707, 0x07050704, 0x0b040705, 0x0b080b08, 0x080b0a08, + 0x090b090b, 0x0e0d090b, 0x06090609, 0x07070609, 0x07070707, 0x05080707, 0x05080508, 0x080b080b, 0x080b080b, 0x080b080b, 0x07080b0e, 0x08070808, + 0x04070807, 0x0d080a08, 0x07090b0d, 0x08090907, 0x03090409, 0x09090909, 0x0c09040a, 0x0c0a0c07, 0x080a0a04, 0x0b08080a, 0x0a09060b, 0x0b080b0d, + 0x0808090b, 0x0b090b08, 0x0908060b, 0x08040807, 0x09070909, 0x09080707, 0x08070704, 0x09090708, 0x07090709, 0x0a080b08, 0x0908040c, 0x0b080c08, + 0x06070908, 0x0f0e0406, 0x0b09090b, 0x080a090a, 0x080c080a, 0x0b090b0b, 0x0b0b0b0d, 0x09080909, 0x0a0b090b, 0x0d0a0e0e, 0x0910090a, 0x06080908, + 0x070b0808, 0x08070808, 0x0809080b, 0x07070709, 0x0808070b, 0x0b0a0d0d, 0x080c0708, 0x07060808, 0x04040407, 0x07080d0c, 0x06070807, 0x0b0e0b0e, + 0x07080b0e, 0x060f0f08, 0x03030303, 0x08060606, 0x120c0708, 0x04040704, 0x08050207, 0x0c080c08, 0x0b0c0f08, 0x0b0b0b09, 0x0b0a090b, 0x0b080809, + 0x08080806, 0x09090908, 0x08080409, 0x05050508, 0x05050505, 0x0f080f08, 0x08030405, 0x00020304, 0x000f0f00, 0x000a1410, 0x07040408, 0x0b0d090a, + 0x09050504, 0x04050409, 0x09090906, 0x09090909, 0x04090909, 0x09090904, 0x0b0b0e07, 0x08080b0a, 0x04060b0b, 0x0c0e0809, 0x0a0b090b, 0x0a0b0808, + 0x0908090e, 0x09050605, 0x09090907, 0x04090908, 0x04040908, 0x090e0408, 0x06090909, 0x08090507, 0x0808080c, 0x09060906, 0x09090404, 0x08090909, + 0x08060d09, 0x080d0509, 0x06060907, 0x040a0909, 0x08060603, 0x070c0c0c, 0x0b0b0b0b, 0x0a0e0b0b, 0x08080808, 0x06060606, 0x0b0b0c0b, 0x090b0b0b, + 0x0b0b0b0b, 0x0a09080b, 0x09090909, 0x080e0909, 0x09090909, 0x04040404, 0x09090909, 0x09090909, 0x09090909, 0x08090809, 0x090b090b, 0x080a090b, + 0x080a080a, 0x0a0b080a, 0x0908090b, 0x09080908, 0x09080908, 0x080b080b, 0x080b080b, 0x090b090b, 0x04060406, 0x04060406, 0x080a0406, 0x08090404, + 0x08040808, 0x08050804, 0x0c040806, 0x0c090c09, 0x090c0b09, 0x090b090b, 0x0f0e090b, 0x060a060a, 0x0708060a, 0x07080708, 0x05080708, 0x05080508, + 0x090b090b, 0x090b090b, 0x090b090b, 0x08080c0e, 0x09080908, 0x05080908, 0x0e090b09, 0x08090b0e, 0x09090907, 0x03090409, 0x09090909, 0x0c0a040b, + 0x0d0b0d07, 0x080b0b05, 0x0b09080a, 0x0a09060b, 0x0b090c0e, 0x0809090b, 0x0d090d08, 0x0908060b, 0x09050907, 0x09080a09, 0x09090707, 0x09080805, + 0x0a090808, 0x07090809, 0x0c080b09, 0x0909050c, 0x0b080c09, 0x06080a08, 0x0f0f0406, 0x0b0a090b, 0x080b090b, 0x090e080a, 0x0b090c0c, 0x0b0b0b0e, + 0x0a080a09, 0x0a0b090d, 0x0d0b1010, 0x0a100a0a, 0x06090909, 0x070b0909, 0x09080909, 0x0909090b, 0x08070809, 0x0909080b, 0x0c0b0e0e, 0x090d0809, + 0x08060909, 0x04040407, 0x08090e0e, 0x06080908, 0x0c0e0c0e, 0x08080c0e, 0x07101008, 0x03030303, 0x08060606, 0x140c0708, 0x05050704, 0x09060208, + 0x0d090d09, 0x0b0d1008, 0x0c0c0c0a, 0x0b0a090c, 0x0b09090a, 0x09090906, 0x09090909, 0x0808040a, 0x06060608, 0x06060606, 0x10081008, 0x09030405, + 0x00020304, 0x00101000, 0x000a1411, 0x07040409, 0x0c0e090b, 0x09050504, 0x04050409, 0x09090906, 0x09090909, 0x04090909, 0x09090904, 0x0b0a0f07, + 0x09090c0a, 0x05060c0c, 0x0d0f080a, 0x0b0d0a0d, 0x0a0c0909, 0x0a09090f, 0x09050605, 0x0a090a07, 0x06090a08, 0x04040908, 0x090f0408, 0x060a0a0a, + 0x08090608, 0x0808080d, 0x09060906, 0x09090404, 0x08090909, 0x08060e0a, 0x090e0509, 0x06060907, 0x040b090a, 0x08060603, 0x070d0d0d, 0x0a0a0a0a, + 0x0a0e0a0a, 0x09090909, 0x06060606, 0x0d0d0d0c, 0x090d0d0d, 0x0c0c0c0d, 0x0a0a090c, 0x09090909, 0x080e0909, 0x09090909, 0x04040404, 0x0a0a090a, + 0x090a0a0a, 0x0909090a, 0x080a0809, 0x090a090a, 0x080a090a, 0x080a080a, 0x0b0c080a, 0x09090a0c, 0x09090909, 0x09090909, 0x080c080c, 0x080c080c, + 0x090c090c, 0x04060406, 0x04060406, 0x080b0406, 0x080a0405, 0x08040808, 0x08050804, 0x0d040806, 0x0d090d09, 0x090d0b09, 0x0a0d0a0d, 0x0f0f0a0d, + 0x060b060b, 0x0809060b, 0x08090809, 0x06090809, 0x06090609, 0x090c090c, 0x090c090c, 0x090c090c, 0x08090d0f, 0x0a080a09, 0x05080a08, 0x0e090a09, + 0x090a0d0e, 0x090a0a08, 0x030a040a, 0x0a0a0a0a, 0x0d0a040a, 0x0d0b0d08, 0x090b0a05, 0x0c0a090a, 0x0a0a060d, 0x0d090d0f, 0x09090a0c, 0x0e090e09, + 0x0a09060d, 0x09050908, 0x0a080a0a, 0x0a090808, 0x09090805, 0x0b0a0808, 0x080a080a, 0x0c090c09, 0x0a09050d, 0x0c090d09, 0x06090a09, 0x100f0506, + 0x0c0a0a0c, 0x090b0a0a, 0x0a0e090b, 0x0b0a0d0d, 0x0c0d0c0f, 0x0a090a0a, 0x0b0c090e, 0x0e0b1111, 0x0b120a0b, 0x06090a09, 0x080c0909, 0x09080909, + 0x090a090c, 0x0808080a, 0x0909080c, 0x0c0b0f0f, 0x090e0809, 0x08060909, 0x04040408, 0x08090e0e, 0x07080908, 0x0d0f0d0f, 0x08090d0f, 0x07111109, + 0x03030303, 0x08060606, 0x140f0608, 0x05050704, 0x09060208, 0x0e090d09, 0x0d0e1209, 0x0d0d0d0b, 0x0c0a0a0d, 0x0c09090b, 0x09090907, 0x0a0a0909, + 0x0909040a, 0x06060609, 0x06060606, 0x11091109, 0x09030406, 0x00020305, 0x00111100, 0x000b1512, 0x07040509, 0x0d0f0a0c, 0x0a050504, 0x0406040a, + 0x0a0a0a07, 0x0a0a0a0a, 0x040a0a0a, 0x0a0a0a04, 0x0b0b1008, 0x090a0c0b, 0x05060d0c, 0x0d10090a, 0x0b0d0b0d, 0x0a0d0909, 0x0a090a10, 0x0a050705, + 0x0b0a0a07, 0x070a0b08, 0x05050b0a, 0x0b100509, 0x070b0b0a, 0x0a0b0609, 0x080a090d, 0x0a060a06, 0x0a0a0405, 0x090a0a0a, 0x0a060f0a, 0x090f060a, + 0x06060a08, 0x040c0b0a, 0x0a060604, 0x080d0d0d, 0x0b0b0b0b, 0x0b0f0b0b, 0x0a0a0a0a, 0x06060606, 0x0d0d0d0c, 0x0a0d0d0d, 0x0d0d0d0d, 0x0b0b090d, + 0x0a0a0a0a, 0x080f0a0a, 0x0a0a0a0a, 0x05050505, 0x0a0a0b0a, 0x0a0a0a0a, 0x0b0b0b0a, 0x0a0b0a0b, 0x0a0b0a0b, 0x080b0a0b, 0x080b080b, 0x0b0c080b, + 0x0a0a0b0c, 0x0a0a0a0a, 0x0a0a0a0a, 0x0a0c0a0c, 0x0a0c0a0c, 0x0b0d0b0d, 0x05060506, 0x05060506, 0x0a0b0506, 0x090a0505, 0x09050909, 0x09050905, + 0x0d050906, 0x0d0b0d0b, 0x0b0d0d0b, 0x0a0d0a0d, 0x10100a0d, 0x070b070b, 0x0909070b, 0x09090909, 0x06090909, 0x06090609, 0x0b0d0b0d, 0x0b0d0b0d, + 0x0b0d0b0d, 0x0a090d10, 0x0a080a09, 0x05080a08, 0x0f0a0b0a, 0x090a0d0f, 0x0a0a0a09, 0x030a050a, 0x0a0a0a0a, 0x0e0b040b, 0x0e0c0e08, 0x0a0b0b06, + 0x0d0a0a0b, 0x0b0a060d, 0x0d0a0d10, 0x090a0b0d, 0x0e0a0e09, 0x0b09060d, 0x0b060b08, 0x0a0a0b0b, 0x0a0b0808, 0x0b090906, 0x0c0a0809, 0x080a080b, + 0x0c0a0d0b, 0x0a0b060d, 0x0d0a0d0b, 0x06090b0a, 0x11100506, 0x0d0b0a0d, 0x0a0b0b0b, 0x0a0e0a0c, 0x0c0a0d0d, 0x0d0d0d10, 0x0b090b0b, 0x0c0d0a0e, + 0x0f0c1212, 0x0b130b0b, 0x070a0a0a, 0x080d0a0a, 0x0a090b0b, 0x0b0a0b0e, 0x0a08080b, 0x0b0b090d, 0x0e0c1010, 0x0a0f080a, 0x08070b0a, 0x05050509, + 0x090b100f, 0x07090b0a, 0x0d100d10, 0x0a090d10, 0x07121209, 0x03040303, 0x08060606, 0x150f0708, 0x06060704, 0x0a070209, 0x0e0a0e0a, 0x0d0e1309, + 0x0e0e0e0b, 0x0d0b0a0e, 0x0d0a0a0b, 0x0a0a0a07, 0x0b0b0b0a, 0x0909050b, 0x06060609, 0x06060606, 0x12091209, 0x0a030506, 0x00020405, 0x00121200, + 0x000b1713, 0x0804050a, 0x0d100a0c, 0x0a060604, 0x0406040a, 0x0a0a0a07, 0x0a0a0a0a, 0x040a0a0a, 0x0a0a0a04, 0x0c0c1008, 0x090a0d0c, 0x05060e0d, + 0x0e110a0b, 0x0c0e0b0e, 0x0b0e0a0a, 0x0a0a0a11, 0x0a060706, 0x0b0a0b08, 0x070a0b09, 0x05050b0a, 0x0b110509, 0x080b0b0a, 0x0a0b0609, 0x080a0a0e, + 0x0a070a07, 0x0a0a0405, 0x0a0a0a0a, 0x0a07100b, 0x0a10060a, 0x06060a08, 0x040c0b0b, 0x0a070604, 0x080e0e0e, 0x0c0c0c0c, 0x0c100c0c, 0x0a0a0a0a, + 0x06060606, 0x0e0e0e0d, 0x0a0e0e0e, 0x0e0e0e0e, 0x0b0b0a0e, 0x0a0a0a0a, 0x09100a0a, 0x0a0a0a0a, 0x05050505, 0x0a0a0b0a, 0x0a0a0a0a, 0x0b0b0b0a, + 0x0a0b0a0b, 0x0a0c0a0c, 0x090c0a0c, 0x090c090c, 0x0c0d090c, 0x0a0a0b0d, 0x0a0a0a0a, 0x0a0a0a0a, 0x0a0d0a0d, 0x0a0d0a0d, 0x0b0e0b0e, 0x05060506, + 0x05060506, 0x0a0b0506, 0x090b0505, 0x0a050a09, 0x0a060a05, 0x0e050a06, 0x0e0b0e0b, 0x0b0e0d0b, 0x0a0e0a0e, 0x11110a0e, 0x080c080c, 0x090a080c, + 0x090a090a, 0x060a090a, 0x060a060a, 0x0b0e0b0e, 0x0b0e0b0e, 0x0b0e0b0e, 0x0a0a0e11, 0x0a080a0a, 0x05080a08, 0x100a0c0a, 0x0a0a0e10, 0x0b0b0b09, + 0x040b050b, 0x0b0b0b0b, 0x0e0b040c, 0x0f0d0f09, 0x0a0c0c06, 0x0e0a0a0b, 0x0c0b060e, 0x0e0a0e11, 0x0a0a0b0e, 0x0e0a0e0a, 0x0b0a060e, 0x0b060b08, + 0x0a0a0b0b, 0x0a0b0908, 0x0b0a0906, 0x0c0a090a, 0x090a090b, 0x0e0a0d0b, 0x0a0b060e, 0x0d0a0e0b, 0x060a0c0a, 0x12110506, 0x0e0b0b0d, 0x0a0c0b0c, + 0x0a100a0c, 0x0d0b0e0e, 0x0e0e0e11, 0x0b0a0c0b, 0x0c0e0a0e, 0x100c1212, 0x0c140c0c, 0x070a0a0a, 0x080e0a0b, 0x0a090b0b, 0x0b0a0b0e, 0x0a09090b, + 0x0b0b0a0d, 0x0f0d1111, 0x0a0f090a, 0x09070b0a, 0x05050509, 0x090b100f, 0x080a0b0a, 0x0e110e11, 0x0a0a0e11, 0x0813130a, 0x03040303, 0x08070707, + 0x170f0708, 0x06060804, 0x0a070209, 0x0f0a0e0a, 0x0e0e130a, 0x0e0e0e0c, 0x0e0b0b0e, 0x0d0a0a0b, 0x0a0a0a07, 0x0c0c0b0a, 0x0a0a050b, 0x0606060a, + 0x06060606, 0x130a130a, 0x0a030506, 0x00020405, 0x00131300, 0x000c1714, 0x0805050a, 0x0e110b0d, 0x0b060605, 0x0506050b, 0x0b0b0b07, 0x0b0b0b0b, + 0x050b0b0b, 0x0b0b0b05, 0x0c0d1109, 0x0a0b0e0c, 0x06070d0e, 0x0e110a0b, 0x0c100c10, 0x0c0d0b0b, 0x0b0b0b12, 0x0b060706, 0x0c0b0c08, 0x070a0c09, + 0x05050c0b, 0x0c13050a, 0x080c0c0b, 0x0a0c0709, 0x090a0a0f, 0x0b070b07, 0x0b0b0505, 0x090b0b0b, 0x0a07110c, 0x0a11060b, 0x07070b09, 0x050d0c0c, + 0x0a070704, 0x090f0f0f, 0x0d0d0d0d, 0x0c110d0d, 0x0b0b0b0b, 0x07070707, 0x10100e0e, 0x0b101010, 0x0d0d0d10, 0x0c0c0b0d, 0x0b0b0b0b, 0x09110b0b, + 0x0a0a0a0a, 0x05050505, 0x0b0b0c0b, 0x0b0b0b0b, 0x0c0c0c0b, 0x0a0c0a0c, 0x0b0d0b0d, 0x090c0b0d, 0x090c090c, 0x0d0e090c, 0x0a0b0c0e, 0x0a0b0a0b, + 0x0a0b0a0b, 0x0b0e0b0e, 0x0b0e0b0e, 0x0c0d0c0d, 0x05070507, 0x05070507, 0x0a0d0507, 0x0a0b0506, 0x0a050a0a, 0x0a060a05, 0x0e050a06, 0x0e0c0e0c, + 0x0c0e0e0c, 0x0b100b10, 0x12120b10, 0x080c080c, 0x090b080c, 0x090b090b, 0x070b090b, 0x070b070b, 0x0c0d0c0d, 0x0c0d0c0d, 0x0c0d0c0d, 0x0a0b0f12, + 0x0b090b0b, 0x06090b09, 0x110b0d0b, 0x0b0b1011, 0x0b0c0c09, 0x040c050c, 0x0c0c0c0c, 0x0f0d050d, 0x110d1109, 0x0b0c0d06, 0x0d0b0b0c, 0x0c0b0710, + 0x100b0e11, 0x0b0b0c0d, 0x0f0b0f0b, 0x0c0b0710, 0x0c060c08, 0x0b0b0c0c, 0x0b0c0908, 0x0c0a0a06, 0x0d0b090a, 0x090b090c, 0x0f0a0d0c, 0x0b0c060f, + 0x0e0b0f0c, 0x070b0c0b, 0x13130607, 0x0d0c0b0e, 0x0b0c0c0d, 0x0b110b0d, 0x0d0b0e0e, 0x0d100d11, 0x0c0b0c0c, 0x0d0d0b0f, 0x100d1313, 0x0c150d0c, + 0x080b0b0b, 0x080f0a0b, 0x0b0a0c0c, 0x0c0b0c0e, 0x0a09090c, 0x0c0c0a0f, 0x100e1111, 0x0b11090b, 0x09080c0a, 0x05050509, 0x0a0c1111, 0x08090c0a, + 0x0f120f12, 0x0a0b0f12, 0x0814140a, 0x04040404, 0x0a080808, 0x170f080a, 0x06060805, 0x0b07030a, 0x100b0f0b, 0x100f140a, 0x0f0f0f0c, 0x0e0c0c0f, + 0x0e0b0b0c, 0x0b0b0b08, 0x0c0c0b0b, 0x0a0a050c, 0x0707070a, 0x07070707, 0x140a140a, 0x0b030507, 0x00020405, 0x00141400, 0x000d1915, 0x0806050b, + 0x0f110c0e, 0x0c060604, 0x0607060c, 0x0c0c0c08, 0x0c0c0c0c, 0x060c0c0c, 0x0c0c0c06, 0x0d0d1209, 0x0a0b0e0d, 0x06080f0e, 0x0f120a0c, 0x0d0f0c0f, + 0x0c0f0a0b, 0x0b0c0b13, 0x0b060806, 0x0c0b0c09, 0x070b0c0a, 0x05050c0b, 0x0c13050a, 0x080c0c0c, 0x0b0c0709, 0x090b0b10, 0x0c070c07, 0x0c0c0605, + 0x0a0c0c0c, 0x0a07110c, 0x0b11070c, 0x07070c09, 0x060e0c0c, 0x0a070704, 0x090f0f0f, 0x0d0d0d0d, 0x0d120d0d, 0x0b0b0b0b, 0x08080808, 0x0f0f0f0e, + 0x0c0f0f0f, 0x0f0f0f0f, 0x0c0c0c0f, 0x0b0b0b0b, 0x0a120b0b, 0x0b0b0b0b, 0x05050505, 0x0c0c0c0c, 0x0c0c0c0c, 0x0c0c0c0c, 0x0b0c0b0c, 0x0b0d0b0d, + 0x0a0d0b0d, 0x0a0d0a0d, 0x0d0e0a0d, 0x0b0b0c0e, 0x0b0b0b0b, 0x0b0b0b0b, 0x0b0e0b0e, 0x0b0e0b0e, 0x0c0f0c0f, 0x05080508, 0x05080508, 0x0b0d0508, + 0x0a0c0506, 0x0a050a0a, 0x0a060a05, 0x0f050a06, 0x0f0c0f0c, 0x0c0f0e0c, 0x0c0f0c0f, 0x14130c0f, 0x080d080d, 0x090b080d, 0x090b090b, 0x070a090b, + 0x070a070a, 0x0c0f0c0f, 0x0c0f0c0f, 0x0c0f0c0f, 0x0b0c1013, 0x0b090b0c, 0x06090b09, 0x120b0d0c, 0x0b0c0f12, 0x0c0c0c09, 0x040c050c, 0x0c0c0c0c, + 0x100d060d, 0x110e100a, 0x0b0d0d06, 0x0f0b0b0c, 0x0d0c080f, 0x0f0b0f12, 0x0a0b0c0f, 0x100b100c, 0x0c0c080f, 0x0c060c09, 0x0c0b0d0c, 0x0c0c0909, + 0x0c0b0a06, 0x0d0c0a0b, 0x090c0a0c, 0x100b0f0c, 0x0c0c060f, 0x0f0b0f0c, 0x080b0d0b, 0x14130608, 0x0f0c0c0f, 0x0b0d0c0d, 0x0b120b0e, 0x0e0c0f0f, + 0x0f0f0f12, 0x0c0a0d0c, 0x0e0f0b10, 0x110e1414, 0x0d150d0d, 0x080b0c0b, 0x090f0b0c, 0x0c0a0c0c, 0x0c0c0c0f, 0x0b090a0c, 0x0c0c0b0f, 0x100e1111, + 0x0b120a0b, 0x0a080c0b, 0x05050509, 0x0a0c1211, 0x080a0c0b, 0x10131013, 0x0b0c1013, 0x0915150b, 0x04040404, 0x0a080808, 0x1912090a, 0x06060804, + 0x0c07030a, 0x110c100c, 0x0f11150b, 0x1010100d, 0x0f0c0c10, 0x0f0c0c0c, 0x0c0c0c08, 0x0c0c0b0c, 0x0b0b050c, 0x0707070b, 0x07070707, 0x150b150b, + 0x0c040507, 0x00020406, 0x00151500, 0x000d1a16, 0x0806060b, 0x0f120c0e, 0x0c070704, 0x0607060c, 0x0c0c0c08, 0x0c0c0c0c, 0x060c0c0c, 0x0c0c0c06, + 0x0e0d1309, 0x0b0c0f0d, 0x06080f0f, 0x10130b0d, 0x0d100d10, 0x0d0f0b0b, 0x0c0c0c13, 0x0c070807, 0x0d0c0d09, 0x070c0d0a, 0x06060d0b, 0x0d14060b, + 0x090d0d0d, 0x0b0d070a, 0x0a0b0b10, 0x0c080c08, 0x0c0c0606, 0x0b0c0c0c, 0x0b07120d, 0x0b12070c, 0x07070c09, 0x060e0d0d, 0x0b080705, 0x09101010, + 0x0d0d0d0d, 0x0d130d0d, 0x0c0c0c0c, 0x08080808, 0x1010100f, 0x0c101010, 0x0f0f0f10, 0x0d0d0c0f, 0x0c0c0c0c, 0x0a120c0c, 0x0c0c0c0c, 0x06060606, + 0x0d0d0d0d, 0x0c0d0d0d, 0x0d0d0d0d, 0x0b0d0b0d, 0x0c0d0c0d, 0x0a0d0c0d, 0x0a0d0a0d, 0x0e0f0a0d, 0x0c0c0d0f, 0x0c0c0c0c, 0x0c0c0c0c, 0x0b0f0b0f, + 0x0b0f0b0f, 0x0d0f0d0f, 0x06080608, 0x06080608, 0x0c0d0608, 0x0b0d0606, 0x0b060b0b, 0x0b070b06, 0x10060b07, 0x100d100d, 0x0d100f0d, 0x0d100d10, + 0x15140d10, 0x090d090d, 0x0a0b090d, 0x0a0b0a0b, 0x070b0a0b, 0x070b070b, 0x0d0f0d0f, 0x0d0f0d0f, 0x0d0f0d0f, 0x0b0c1013, 0x0c0a0c0c, 0x060a0c0a, + 0x130c0d0c, 0x0b0d1012, 0x0c0d0d0a, 0x040d060d, 0x0d0d0d0d, 0x110d060d, 0x110f110a, 0x0c0e0d07, 0x0f0c0c0d, 0x0d0d0810, 0x100c1013, 0x0b0c0d0f, + 0x100c100c, 0x0d0c0810, 0x0d070d0a, 0x0d0b0d0d, 0x0d0d0a0a, 0x0d0b0b07, 0x0e0d0a0b, 0x0a0d0a0d, 0x100c100d, 0x0d0d0710, 0x0f0c100d, 0x080b0d0c, + 0x15140608, 0x0f0d0d0f, 0x0c0e0d0d, 0x0c120c0e, 0x0f0d1010, 0x0f100f13, 0x0d0b0d0d, 0x0f0f0c10, 0x120f1616, 0x0d170d0e, 0x090c0d0c, 0x0a100c0c, + 0x0c0b0d0d, 0x0d0d0d10, 0x0b0a0a0d, 0x0d0d0b10, 0x110e1414, 0x0c130a0c, 0x0a090d0c, 0x0606060a, 0x0b0d1312, 0x090a0d0b, 0x10131013, 0x0b0c1013, + 0x0916160b, 0x04050404, 0x0b080808, 0x1a12080b, 0x07070804, 0x0c08030b, 0x120c110c, 0x1011160b, 0x1111110e, 0x0f0d0d11, 0x0f0c0c0d, 0x0c0c0c08, + 0x0d0d0d0c, 0x0b0b060c, 0x0707070b, 0x07070707, 0x160b160b, 0x0c040607, 0x00020406, 0x00161600, 0x000e1b17, 0x0806060c, 0x10130d0f, 0x0d070704, + 0x0607060d, 0x0d0d0d09, 0x0d0d0d0d, 0x060d0d0d, 0x0d0d0d06, 0x0e0e140a, 0x0b0c100e, 0x06081010, 0x11140b0d, 0x0e110d11, 0x0d100c0c, 0x0c0c0d14, + 0x0c070907, 0x0d0c0d09, 0x080c0d0b, 0x06060e0c, 0x0e14060b, 0x090d0d0d, 0x0b0e080a, 0x0a0b0b11, 0x0d080c08, 0x0d0d0606, 0x0b0c0d0d, 0x0b08130d, + 0x0c13070d, 0x08080d0a, 0x060f0e0d, 0x0b080805, 0x0a111111, 0x0e0e0e0e, 0x0e140e0e, 0x0c0c0c0c, 0x08080808, 0x11111110, 0x0d111111, 0x10101011, + 0x0e0d0c10, 0x0c0c0c0c, 0x0b130c0c, 0x0c0c0c0c, 0x06060606, 0x0d0d0e0d, 0x0d0d0d0d, 0x0e0e0e0d, 0x0b0d0b0e, 0x0c0e0c0e, 0x0b0e0c0e, 0x0b0e0b0e, + 0x0e100b0e, 0x0c0c0d10, 0x0c0c0c0c, 0x0c0c0c0c, 0x0c100c10, 0x0c100c10, 0x0e100e10, 0x06080608, 0x06080608, 0x0c0e0608, 0x0b0d0606, 0x0b060b0b, + 0x0b070b06, 0x11060b08, 0x110e110e, 0x0e11100e, 0x0d110d11, 0x15140d11, 0x090e090e, 0x0a0c090e, 0x0a0c0a0c, 0x080c0a0c, 0x080c080c, 0x0e100e10, + 0x0e100e10, 0x0e100e10, 0x0b0c1114, 0x0c0a0c0c, 0x070a0c0a, 0x140c0e0d, 0x0c0d1113, 0x0d0d0d0a, 0x040d060d, 0x0d0d0d0d, 0x120e060e, 0x120f120b, + 0x0c0e0e08, 0x100c0c0d, 0x0e0d0811, 0x110c1114, 0x0c0c0d10, 0x120d120c, 0x0d0c0811, 0x0e080e0a, 0x0d0b0e0d, 0x0d0e0a0a, 0x0e0c0b08, 0x0f0d0b0b, + 0x0a0d0b0d, 0x110c100e, 0x0d0e0810, 0x100c100e, 0x080c0e0c, 0x16150608, 0x100d0d10, 0x0c0e0d0e, 0x0d120c0f, 0x0f0d1111, 0x10111014, 0x0d0c0e0d, + 0x0f100d12, 0x130f1616, 0x0e180e0e, 0x090d0d0c, 0x0b100c0d, 0x0d0b0e0e, 0x0e0d0e11, 0x0b0a0b0d, 0x0d0e0b10, 0x12101414, 0x0c130b0d, 0x0b090e0c, + 0x0606060a, 0x0b0e1514, 0x090b0e0b, 0x11141114, 0x0b0c1114, 0x0917170c, 0x04050404, 0x0a080808, 0x1b120a0a, 0x08080804, 0x0d08030b, 0x120d110d, + 0x1112170c, 0x1111110e, 0x100d0d11, 0x100d0d0e, 0x0d0d0d09, 0x0e0e0d0d, 0x0c0c060d, 0x0808080c, 0x08080808, 0x170c170c, 0x0d040608, 0x00020506, + 0x00171700, 0x000e1c18, 0x0a06060c, 0x11140d10, 0x0d070705, 0x0608060d, 0x0d0d0d09, 0x0d0d0d0d, 0x060d0d0d, 0x0d0d0d06, 0x0f0f150a, 0x0c0d100e, + 0x07091111, 0x12150c0e, 0x0e120e12, 0x0e110d0d, 0x0d0d0d15, 0x0d070907, 0x0e0d0e0a, 0x080d0e0b, 0x06060e0d, 0x0e16060c, 0x0a0e0e0e, 0x0c0e080b, + 0x0b0c0c12, 0x0d090d09, 0x0d0d0606, 0x0c0d0d0d, 0x0c08140e, 0x0c14080d, 0x08080d0a, 0x06100e0e, 0x0c080805, 0x0a121212, 0x0f0f0f0f, 0x0e140f0f, + 0x0d0d0d0d, 0x09090909, 0x12121210, 0x0d121212, 0x11111112, 0x0e0e0d11, 0x0d0d0d0d, 0x0b140d0d, 0x0d0d0d0d, 0x06060606, 0x0e0e0e0e, 0x0d0e0e0e, + 0x0e0e0e0e, 0x0c0e0c0e, 0x0d0f0d0f, 0x0b0e0d0f, 0x0b0e0b0e, 0x0f100b0e, 0x0d0d0e10, 0x0d0d0d0d, 0x0d0d0d0d, 0x0d110d11, 0x0d110d11, 0x0e110e11, + 0x06090609, 0x06090609, 0x0c0f0609, 0x0c0e0607, 0x0c060c0c, 0x0c070c06, 0x12060c08, 0x120e120e, 0x0e12100e, 0x0e120e12, 0x16150e12, 0x0a0e0a0e, + 0x0b0d0a0e, 0x0b0d0b0d, 0x080d0b0d, 0x080d080d, 0x0e110e11, 0x0e110e11, 0x0e110e11, 0x0c0d1215, 0x0d0b0d0d, 0x070b0d0b, 0x140d0f0d, 0x0d0e1214, + 0x0d0e0e0b, 0x040e060e, 0x0e0e0e0e, 0x120e060f, 0x1310130b, 0x0d0f0f08, 0x110d0d0e, 0x0f0e0912, 0x120d1215, 0x0d0d0e11, 0x130d130d, 0x0e0d0912, + 0x0e080e0b, 0x0e0c0e0e, 0x0e0e0b0b, 0x0e0c0c08, 0x0f0e0b0c, 0x0b0e0b0e, 0x120d100e, 0x0e0e0812, 0x110d120e, 0x090d0e0d, 0x17160709, 0x110e0e11, + 0x0d0f0e0f, 0x0d140d10, 0x100e1212, 0x11121115, 0x0e0d0e0e, 0x10110d13, 0x14101717, 0x0e190e0f, 0x0a0d0e0d, 0x0b100d0d, 0x0d0c0e0e, 0x0e0e0e11, + 0x0c0a0b0e, 0x0e0e0c10, 0x12101414, 0x0d140b0d, 0x0b0a0e0d, 0x0606060b, 0x0c0e1514, 0x0a0c0e0c, 0x12151215, 0x0c0d1215, 0x0a18180c, 0x04050404, + 0x0b090909, 0x1c120a0b, 0x08080a05, 0x0d08030c, 0x130d120d, 0x1213180c, 0x1212120f, 0x110e0e12, 0x110d0d0e, 0x0d0d0d09, 0x0e0e0f0d, 0x0c0c060d, + 0x0808080c, 0x08080808, 0x180c180c, 0x0d040608, 0x00020506, 0x00181800, 0x03000000, 0x03000000, 0x1c000000, 0x00000100, 0xe4010000, 0x01000300, + 0x1c000000, 0xc8010400, 0x6e000000, 0x05004000, 0x7e002e00, 0x92017f01, 0x1902ff01, 0xc902c702, 0x8a03dd02, 0xa1038c03, 0x0c04ce03, 0x5c044f04, + 0x91045f04, 0xf31e851e, 0x15200b20, 0x22201e20, 0x30202620, 0x3a203320, 0x44203c20, 0xa4207f20, 0xac20a720, 0x13210521, 0x22211621, 0x2e212621, + 0x02225e21, 0x0f220622, 0x1a221222, 0x2b221e22, 0x60224822, 0xca256522, 0xfffe02fb, 0xfffffdff, 0x20000000, 0x9201a000, 0x1802fa01, 0xc902c602, + 0x8403d802, 0x8e038c03, 0x0104a303, 0x51040e04, 0x90045e04, 0xf21e801e, 0x13200020, 0x20201720, 0x30202620, 0x39203220, 0x44203c20, 0xa3207f20, + 0xac20a720, 0x13210521, 0x22211621, 0x2e212621, 0x02225b21, 0x0f220622, 0x1a221122, 0x2b221e22, 0x60224822, 0xca256422, 0xfffe01fb, 0xfffffcff, + 0xc2ffe3ff, 0x49ffb0ff, 0x85fe31ff, 0x76fe84fe, 0xcffdd0fd, 0xcdfdcefd, 0x9afd9bfd, 0x98fd99fd, 0x7ae368fd, 0x42e20ee3, 0xeee1efe1, 0xeae1ede1, + 0xe0e1e1e1, 0xdae1dbe1, 0x99e1d3e1, 0x74e176e1, 0x18e170e1, 0x09e10be1, 0xfbe0fee0, 0xc8e0f4e0, 0x22e025e0, 0x19e01ae0, 0x0fe012e0, 0xe7df03e0, + 0xcddfd0df, 0x330769dc, 0x53024f03, 0x00000100, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x0c020600, 0x00000000, 0x01000001, 0x00000000, 0x00000000, + 0x00000000, 0x01000000, 0x00000200, 0x00000000, 0x00000200, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000100, 0x03000000, 0x05000400, 0x07000600, 0x09000800, 0x0b000a00, 0x0d000c00, 0x0f000e00, 0x11001000, 0x13001200, 0x15001400, 0x17001600, + 0x19001800, 0x1b001a00, 0x1d001c00, 0x1f001e00, 0x21002000, 0x23002200, 0x25002400, 0x27002600, 0x29002800, 0x2b002a00, 0x2d002c00, 0x2f002e00, + 0x31003000, 0x33003200, 0x35003400, 0x37003600, 0x39003800, 0x3b003a00, 0x3d003c00, 0x3f003e00, 0x41004000, 0x43004200, 0x45004400, 0x47004600, + 0x49004800, 0x4b004a00, 0x4d004c00, 0x4f004e00, 0x51005000, 0x53005200, 0x55005400, 0x57005600, 0x59005800, 0x5b005a00, 0x5d005c00, 0x5f005e00, + 0x61006000, 0x86000000, 0x89008700, 0x93008b00, 0x9e009800, 0xa200a300, 0xa600a400, 0xa700a500, 0xab00a900, 0xac00aa00, 0xaf00ad00, 0xb000ae00, + 0xb300b100, 0xb400b500, 0xb800b600, 0xbc00b700, 0xbd00bb00, 0x0d02be00, 0x64007200, 0x69006500, 0x78000f02, 0x7000a100, 0x20026b00, 0x6a007600, + 0x88003002, 0x2d029a00, 0x31027300, 0x67003202, 0x27027700, 0x29022a02, 0x2e028d01, 0x7c006c00, 0xa8002102, 0x8100ba00, 0x6e006300, 0x42012c02, + 0x28022f02, 0x7d006d00, 0x03001002, 0x85008200, 0x14019700, 0x02021501, 0x0a020302, 0x06020b02, 0xb9000702, 0xc1003302, 0x17023a01, 0x14026600, + 0x34021502, 0x0e023502, 0x08027900, 0x11020c02, 0x8c008400, 0x8d008300, 0x8f008a00, 0x91009000, 0x95008e00, 0x00009600, 0x9c009400, 0x9b009d00, + 0x4b01f300, 0x71005201, 0x4f014e01, 0x7a005001, 0x51015301, 0x00004c01, 0x58594540, 0x52535455, 0x4e4f5051, 0x4a4b4c4d, 0x46474849, 0x42434445, + 0x3e3f4041, 0x3a3b3c3d, 0x36373839, 0x2f303135, 0x282c2d2e, 0x24252627, 0x1f212223, 0x10111418, 0x0b0d0e0f, 0x0708090a, 0x03040506, 0x2c000102, + 0x60462345, 0x6026b020, 0x232604b0, 0x2c2d4848, 0x23462345, 0x26b02061, 0x2604b061, 0x2d484823, 0x4623452c, 0x6120b060, 0x6046b020, 0x232604b0, + 0x2c2d4848, 0x23462345, 0x6020b061, 0x6126b020, 0xb06120b0, 0x48232604, 0x452c2d48, 0xb0604623, 0xb0206140, 0x04b06066, 0x48482326, 0x23452c2d, + 0xb0612346, 0xb0206040, 0x40b06126, 0x2604b061, 0x2d484823, 0x2010012c, 0x2d3c003c, 0x2345202c, 0x44cdb020, 0x01b82023, 0x2358515a, 0x448db020, + 0xb0205923, 0x235851ed, 0x444db020, 0xb0205923, 0x58512604, 0x0db02023, 0x21592344, 0x202c2d21, 0x68184520, 0x01b02044, 0xb0452060, 0x8a687646, + 0x2d446045, 0x0bb1012c, 0x4323430a, 0x2c2d0a65, 0x0b0ab100, 0x0b432343, 0xb0002c2d, 0xb1702328, 0x013e2801, 0x702328b0, 0x452802b1, 0x0002b13a, + 0x2c2d0d08, 0x03b04520, 0x64614525, 0x585150b0, 0x211b4445, 0x2c2d5921, 0x230eb049, 0x202c2d44, 0x4300b045, 0x2c2d4460, 0x4306b001, 0x654307b0, + 0x202c2d0a, 0x6140b069, 0x208b00b0, 0x8ac02cb1, 0x0010b88c, 0x0c2b6062, 0x61642364, 0x03b0585c, 0x2c2d5961, 0x8a45038a, 0x11b0878a, 0x2329b02b, + 0x7a29b044, 0x2c2d18e4, 0x2cb06545, 0xb0454423, 0x2d44232b, 0x58524b2c, 0x211b4445, 0x2c2d5921, 0x4558514b, 0x21211b44, 0x012c2d59, 0x102505b0, + 0xf58a2023, 0x6001b000, 0x2deced23, 0x05b0012c, 0x20231025, 0xb000f58a, 0xed236101, 0x012c2dec, 0x102506b0, 0xeced00f5, 0x23462c2d, 0x8a8a6046, + 0x46202346, 0x618a608a, 0x6280ffb8, 0x23102023, 0x0c0cb18a, 0x6045708a, 0x5000b020, 0x6101b058, 0x8bbaffb8, 0x8c46b01b, 0x6010b059, 0x2d3a0168, + 0xb045202c, 0x52462503, 0x5113b04b, 0x02b0585b, 0x68204625, 0x2503b061, 0x3f2503b0, 0x1b382123, 0x2d591121, 0xb045202c, 0x50462503, 0x2502b058, + 0x61682046, 0xb02503b0, 0x233f2503, 0x211b3821, 0x2c2d5911, 0x4307b000, 0x0b4306b0, 0x21212c2d, 0x6423640c, 0x0040b88b, 0x212c2d62, 0x585180b0, + 0x6423640c, 0x0020b88b, 0x00b21b62, 0x592b2f40, 0x2d6002b0, 0xc0b0212c, 0x640c5851, 0xb88b6423, 0x1b625515, 0x2f8000b2, 0x02b0592b, 0x0c2c2d60, + 0x8b642364, 0x620040b8, 0x2d212360, 0x58534b2c, 0x2504b08a, 0x45236449, 0x8b40b069, 0x6280b061, 0x6a6120b0, 0x44230eb0, 0x0eb01023, 0x23211bf6, + 0x2011128a, 0x2d592f39, 0x58534b2c, 0x2503b020, 0x20696449, 0xb02605b0, 0x64492506, 0x80b06123, 0x6120b062, 0x230eb06a, 0x2604b044, 0xf60eb010, + 0x0eb0108a, 0x0eb04423, 0x230eb0f6, 0xed0eb044, 0x04b08a1b, 0x20121126, 0x39202339, 0x2d592f2f, 0x4523452c, 0x60452360, 0x23604523, 0xb0186876, + 0x2d206280, 0x2b48b02c, 0x45202c2d, 0x585400b0, 0x204440b0, 0x6140b045, 0x21211b44, 0x452c2d59, 0x452f30b1, 0x60614523, 0x696001b0, 0x4b2c2d44, + 0x2fb05851, 0x14b07023, 0x211b4223, 0x2c2d5921, 0x2058514b, 0x452503b0, 0x44585369, 0x5921211b, 0x5921211b, 0xb0452c2d, 0x00b04314, 0x01b06360, + 0x2d446960, 0x452fb02c, 0x452c2d44, 0x8a452023, 0x2c2d4460, 0x60452345, 0x4b2c2d44, 0xb9585123, 0xe0ff3300, 0x1b2034b1, 0x340033b3, 0x44445900, + 0x16b02c2d, 0x03b05843, 0x588a4526, 0x1fb06664, 0xb0641b60, 0x20666020, 0xb0211b58, 0x01b05940, 0x58235961, 0x29b05965, 0x10234423, 0x1be029b0, + 0x21212121, 0x2c2d5921, 0x544302b0, 0x23534b58, 0x585a514b, 0x21211b38, 0x21211b59, 0x2d592121, 0x4316b02c, 0x2504b058, 0x20b06445, 0x58206660, + 0x40b0211b, 0x6101b059, 0x651b5823, 0x2329b059, 0x2505b044, 0x082508b0, 0x1b025820, 0x04b05903, 0x05b01025, 0xb0462025, 0x42232504, 0x2504b03c, + 0x082507b0, 0x102507b0, 0x202506b0, 0x2504b046, 0x236001b0, 0x58203c42, 0x59001b01, 0x102504b0, 0xb02505b0, 0x29b0e029, 0x44654520, 0x102507b0, + 0xb02506b0, 0x05b0e029, 0x2508b025, 0x02582008, 0xb059031b, 0x03b02505, 0xb0484325, 0x07b02504, 0x06b00825, 0x2503b025, 0x436001b0, 0x59211b48, + 0x21212121, 0x2d212121, 0x04b0022c, 0x46202025, 0x232504b0, 0x2505b042, 0x2503b008, 0x21214845, 0x2c2d2121, 0x2503b002, 0x2504b020, 0x2502b008, + 0x21214843, 0x452c2d21, 0x18452023, 0x5000b020, 0x65235820, 0x68235923, 0x5040b020, 0x40b02158, 0x65582359, 0x44608a59, 0x534b2c2d, 0x5a514b23, + 0x8a452058, 0x211b4460, 0x2c2d5921, 0x2058544b, 0x44608a45, 0x5921211b, 0x534b2c2d, 0x5a514b23, 0x211b3858, 0x2c2d5921, 0x4b2100b0, 0x1b385854, + 0x2d592121, 0x4302b02c, 0x46b05854, 0x21211b2b, 0x2d592121, 0x4302b02c, 0x47b05854, 0x21211b2b, 0x2c2d5921, 0x544302b0, 0x2b48b058, 0x2121211b, + 0x2c2d5921, 0x544302b0, 0x2b49b058, 0x2121211b, 0x202c2d59, 0x4b23088a, 0x514b8a53, 0x3823585a, 0x5921211b, 0xb0002c2d, 0xb0492502, 0x20585300, + 0x113840b0, 0x2d59211b, 0x2346012c, 0x46236046, 0x10202361, 0x618a4620, 0x6280ffb8, 0x4040b18a, 0x6045708a, 0x2c2d3a68, 0x49238a20, 0x53238a64, + 0x211b3c58, 0x4b2c2d59, 0x1b7d5852, 0x2c2d597a, 0x4b0012b0, 0x42544b01, 0x02b12c2d, 0x23b14200, 0xb1518801, 0x53880140, 0x10b9585a, 0x88200000, + 0x02b25854, 0x60430201, 0x24b15942, 0x58518801, 0x000020b9, 0x58548840, 0x020202b2, 0xb1426043, 0x54880124, 0x2002b258, 0x42604302, 0x4b014b00, + 0x02b25852, 0x60430208, 0xb91b5942, 0x80000040, 0xb2585488, 0x43020402, 0xb9594260, 0x80000040, 0x0001b863, 0xb2585488, 0x43020802, 0xb9594260, + 0x00010040, 0x0002b863, 0xb2585488, 0x43021002, 0xb9594260, 0x00020040, 0x0004b863, 0xb2585488, 0x43024002, 0x59594260, 0x2d595959, 0x6818452c, + 0x58514b23, 0x20452023, 0x5040b064, 0x68597c58, 0x4459608a, 0x00b02c2d, 0x2502b016, 0x012502b0, 0x3e2301b0, 0x2302b000, 0x0201b13e, 0x0ab00c06, + 0xb0426523, 0x0142230b, 0x3f2301b0, 0x2302b000, 0x0201b13f, 0x06b00c06, 0xb0426523, 0xb0422307, 0x2d011601, 0x108a7a2c, 0x18f52345, 0x0000002d, + 0xf8091040, 0x8f1fff03, 0x02f79ff7, 0x6001f37f, 0xffb801f2, 0xeb2b40e8, 0xdf46100c, 0xde55dd33, 0x3055dcff, 0x01dd01dd, 0x03dc5503, 0xc2301ffa, + 0xefc06f01, 0xb6fc02c0, 0xb7301f18, 0x80b76001, 0xffb802b7, 0xb73840c0, 0xe746130f, 0xaf1f01b1, 0xaf3faf2f, 0x5faf4f03, 0x03af6faf, 0x130faf40, + 0x1851ac46, 0x5f9c1f1f, 0x9be0029c, 0x9a2b0301, 0x019a1f01, 0x9aa09a90, 0x839a7302, 0xb805029a, 0x1940eaff, 0x460b099a, 0x97bf97af, 0x962b0302, + 0x01961f01, 0x96af969f, 0x01967c02, 0xeaffb805, 0x09968540, 0x922f460b, 0x924f923f, 0x0c924003, 0x912f460f, 0x01919f01, 0x1f188687, 0x7c507c40, + 0x74100302, 0x74307420, 0x01740203, 0x0a0174f2, 0x6fff016f, 0x016fa901, 0x75016f97, 0x026f856f, 0x0a016f4b, 0x6eff016e, 0x016ea901, 0x4b016e97, + 0x1a06016e, 0x19551801, 0x071fff13, 0x061fff04, 0x3f1fff03, 0x671f0167, 0x673f672f, 0x400467ff, 0xa0665066, 0x0466b066, 0x0f01653f, 0x0265af65, + 0xe064a005, 0xb8030264, 0x4b40c0ff, 0x460a0664, 0x1f475f60, 0x1f22505f, 0xec015bf7, 0x5b54015b, 0x49025b84, 0x5b3b015b, 0x015af901, 0x6b015aef, + 0x5a4b015a, 0x015a3b01, 0x12331306, 0x03010555, 0x03330455, 0x01031f55, 0x033f030f, 0x0f0303af, 0x2f571f57, 0xb8030357, 0x56b3c0ff, 0xb8461512, + 0x56b3e0ff, 0xb8460b07, 0x54b3c0ff, 0xb8461512, 0x6540c0ff, 0x460b0654, 0x504f503f, 0xfa03505f, 0x48ef0148, 0x01488701, 0x56014865, 0x483a0148, + 0x0147fa01, 0x870147ef, 0x473b0147, 0x1b1c0601, 0x33161fff, 0x01115515, 0x3310550f, 0x0102550f, 0x47015500, 0x1bfa5500, 0x0f0f1f12, 0xcf0f1f01, + 0x0f0f020f, 0x06020fff, 0x007f006f, 0x00ef00af, 0x01001004, 0x05011680, 0x9001b801, 0x2b5354b1, 0x07b84b2b, 0xb04b52ff, 0xb05b5006, 0x25b08801, + 0x8801b053, 0x5a5140b0, 0xb08806b0, 0x5b5a5500, 0x0101b158, 0x8d85598e, 0x1d42008d, 0x5332b04b, 0x1d60b058, 0x64b04b59, 0x40b05853, 0xb04b591d, + 0xb0585380, 0x16b11d10, 0x73594200, 0x74735e73, 0x2b2b2b75, 0x2b2b2b2b, 0x73735f01, 0x73737373, 0x73737373, 0x2b017300, 0x5f2b2b2b, 0x74730073, + 0x012b2b2b, 0x7373735f, 0x73737373, 0x00737373, 0x2b012b2b, 0x735e735f, 0x74737374, 0x2b2b2b00, 0x735f012b, 0x74737373, 0x73737373, 0x73007473, + 0x5f017474, 0x73002b73, 0x01732b74, 0x73735f2b, 0x735f7474, 0x73735f2b, 0x5f007474, 0x2b017373, 0x74732b00, 0x2b007301, 0x012b7473, 0x2b730073, + 0x2b2b732b, 0x73732b01, 0x182b0073, 0x0000005e, 0x0b001406, 0xb6054e00, 0x75001700, 0xcd05b605, 0x00000000, 0x00000000, 0x00000000, 0x4a040000, + 0x8f001400, 0xecff0000, 0x00000000, 0x0000ecff, 0xecff0000, 0x14fe0000, 0x000014fe, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00080000, 0x00000000, 0xb600ac00, 0x0000bc00, 0x0000d500, 0x00000000, 0x83005500, 0x9f009700, 0xe5007d00, 0xae00ae00, 0x71007100, + 0x00000000, 0xc500ba00, 0x0000ba00, 0xa4000000, 0x00009f00, 0x00000000, 0xc700c700, 0x7d007d00, 0x00000000, 0x00000000, 0x00000000, 0xb900b000, + 0x00008a00, 0x9b000000, 0x8f00a600, 0x00007700, 0x00000000, 0x00000000, 0x00009600, 0x00000000, 0x00000000, 0x6e006900, 0xb4009000, 0xd500c100, + 0x00000000, 0x00000000, 0x6f006600, 0x96007800, 0xd500c000, 0x00004701, 0x00000000, 0x3a01fe00, 0x7800c500, 0x1601fe00, 0x0000f601, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x0000ee00, 0x88009600, 0x9600ae00, 0x0c018900, 0x18019600, 0x1d030000, 0x5a029400, 0x96038200, + 0xa8000000, 0x00008c00, 0x79020000, 0xb400d900, 0x00000a01, 0x6d008301, 0xa0007f00, 0x00000000, 0x88006d00, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0xa0009300, 0x82000000, 0x00008900, 0x00000000, 0x00000000, 0x94fcb605, 0xefff1100, 0x8f008300, 0x00000000, 0x7b006d00, 0x00000000, + 0x00000000, 0xbc000000, 0x5403aa01, 0x00000000, 0xb600bc00, 0x9501d701, 0x96000000, 0xae000001, 0xbcfeb605, 0x83fe6ffe, 0xad026f00, 0x2a000000, + 0x2a002a00, 0x6f002a00, 0x2901a100, 0x7102e201, 0x4d032903, 0xac037c03, 0x0f04e203, 0x5b044404, 0xb4049504, 0x3f050605, 0x12069405, 0xcc066506, + 0x67073f07, 0x82081108, 0x3d09da08, 0xbb097909, 0x6a0af709, 0xa00b340b, 0x740c170c, 0xec0cb30c, 0x970d390d, 0x100ecd0d, 0x970e490e, 0x2b0fb70e, + 0xd50f7c0f, 0x8b102210, 0x6611fc10, 0xe511a711, 0xee123812, 0x9e134d13, 0xf713d113, 0x3b141714, 0x70145914, 0x08159514, 0xae156915, 0x6b160e16, + 0xa217ce16, 0x1418e317, 0xaf185f18, 0x3919c818, 0xc0197719, 0x831a221a, 0x391bc91a, 0xcf1b8e1b, 0xd41c271c, 0xce1d651d, 0x751e1b1e, 0xf51e991e, + 0x481f481f, 0xf41f901f, 0x01216a20, 0xa4217421, 0xa1225d22, 0xb4234a23, 0x35240824, 0x0c253d24, 0x7e252325, 0x0a26ba25, 0xa4267f26, 0x2b27ee26, + 0xab276527, 0x2a28e227, 0xa3287a28, 0xf728c828, 0x85296d29, 0xb4299c29, 0xe429cb29, 0x722a092a, 0x9d2a852a, 0xcc2ab42a, 0xfd2ae52a, 0x2b2b142b, + 0xa82b442b, 0xd72bbf2b, 0x052cee2b, 0x362c1d2c, 0x1c2d9b2c, 0x4b2d342d, 0x7c2d632d, 0xdd2d932d, 0x8f2e782e, 0xbb2ea52e, 0xe92ed12e, 0xb12f012f, + 0xdb2fc42f, 0x0730f12f, 0x37301f30, 0x65304e30, 0x0e317e30, 0x3b312431, 0x68315131, 0x98317f31, 0x82320932, 0xaf329932, 0xde32c532, 0x5933f432, + 0x89337133, 0xb6339f33, 0xe333cc33, 0x17340034, 0x46342f34, 0x75345e34, 0xa5348e34, 0xd534bd34, 0xf934f134, 0x89357235, 0xb6359f35, 0xe335cc35, + 0x0e36fa35, 0x46362f36, 0x73365c36, 0xa0368936, 0xce36b736, 0xf936e636, 0x27370f37, 0x95374437, 0x0738f037, 0x35381e38, 0x63384c38, 0x8d387a38, + 0xb838a138, 0xfa38d138, 0x35391e39, 0x5f394c39, 0xbb397239, 0xf039d339, 0x173a033a, 0x4e3a2f3a, 0x843a613a, 0x1e3bd13a, 0x4b3b353b, 0x723b5f3b, + 0xa03b893b, 0x233cc23b, 0x933c7c3c, 0xc03ca93c, 0xf03cd73c, 0x793d083d, 0x1f3e073e, 0x483e353e, 0x743e5c3e, 0xa33e8c3e, 0xd03eb93e, 0xfa3ee63e, + 0x243f0d3f, 0x4d3f3a3f, 0x783f613f, 0xe13f8a3f, 0x63404b40, 0x91407940, 0xbe40a740, 0xed40d440, 0x1e410541, 0x49413641, 0x73415c41, 0xa1418a41, + 0xd041b741, 0xfd41e741, 0x2b421442, 0x58424242, 0xef428b42, 0xa644d143, 0xd544be44, 0x0245ec44, 0x28451545, 0x94455e45, 0xe745ac45, 0x5e460e46, + 0xd2468b46, 0x42471547, 0xcc47b347, 0x2b480648, 0x92485148, 0xf748c348, 0x38491f49, 0x48494049, 0xc6497a49, 0xd649ce49, 0x4d4ade49, 0x5d4a554a, + 0xb14aa94a, 0x064bb94a, 0x394b0e4b, 0x894b414b, 0x994b914b, 0x304c284c, 0x204d9e4c, 0x524d394d, 0x7e4d684d, 0xac4d944d, 0x334ec14d, 0x144fb54e, + 0x18509c4f, 0xbe507e50, 0x6e513051, 0xf2517651, 0x82524052, 0x2a532253, 0xd8537e53, 0xa0544754, 0x2c55ea54, 0x3e56aa55, 0x29579f56, 0x5b574257, + 0x87577157, 0xbe57a557, 0x4f583858, 0xc258ba58, 0xe358ca58, 0x8259eb58, 0x3f5ae159, 0x6d5a565a, 0xbe5ab65a, 0x195b115b, 0x945b215b, 0x295c9c5b, + 0x0b5dac5c, 0x6b5d235d, 0xd75dcf5d, 0xe75ddf5d, 0xf75def5d, 0x075eff5d, 0x7f5e775e, 0xbc5e875e, 0x4b5f035f, 0xf75fa15f, 0x9b604d60, 0x7561fe60, + 0xfa61f261, 0xe7627562, 0x75630f63, 0xf2637d63, 0xb4647064, 0x0565cb64, 0xb5655065, 0x0366fb65, 0x34662c66, 0x7b663c66, 0x00678366, 0x42670867, + 0xcd678267, 0x83682668, 0x2f69e568, 0x0c6a9869, 0x7b6a636a, 0x006bea6a, 0x646b5c6b, 0x856b6c6b, 0x056c8d6b, 0xbe6c646c, 0xea6cd46c, 0x646d2f6d, + 0xa76d8f6d, 0xd56dbe6d, 0x066eec6d, 0x386e206e, 0x6b6e506e, 0xa16e866e, 0xf56eca6e, 0x546f216f, 0xde6f826f, 0xa4703a70, 0x8571ff70, 0x6d72c871, + 0x42733a73, 0x7d734a73, 0xc473b273, 0x2574e273, 0x07758074, 0x25769175, 0x5177d376, 0x8e780778, 0xf9789678, 0x51792279, 0xab798079, 0x727a257a, + 0xeb7aa17a, 0x3d7b117b, 0x2a7cd97b, 0x2b7db37c, 0xb47d707d, 0x1f7e007e, 0x807e3d7e, 0xd97eb37e, 0x237ffe7e, 0xaf7f667f, 0x6b800c80, 0x37819a80, + 0xa181a181, 0xa181a181, 0xa181a181, 0xa181a181, 0xa181a181, 0xa181a181, 0xa181a181, 0x5e83eb82, 0xc1000200, 0x0a040000, 0x0300b605, 0x2b000700, + 0x02050b40, 0x70040902, 0x02038003, 0xc0ffb803, 0x09060940, 0x03040348, 0x00030007, 0x332f323f, 0x5d2b2f01, 0x2f331133, 0x13303133, 0x37211121, + 0xc1211121, 0xb7fc4903, 0xfd790268, 0xfab60587, 0xe604684a, 0x93000200, 0x9101e3ff, 0x0300b605, 0x3a001700, 0xff0100b9, 0x0a1340f0, 0x19104814, + 0x19901980, 0x030419a0, 0x02049a0e, 0xffb80402, 0x070a40c0, 0x0104480a, 0x02139b09, 0x2f3f0003, 0x2f01cef5, 0x102f332b, 0x315d32e1, 0x23012b30, + 0x34033303, 0x3233023e, 0x1415021e, 0x2223020e, 0x5001022e, 0xf0df3379, 0x1b2e2214, 0x14222f1a, 0x1a2f2214, 0x14222e1b, 0x18049e01, 0x3526b9fa, + 0x210f0f21, 0x35252635, 0x22101022, 0x02000035, 0xa6038500, 0xb605b202, 0x07000300, 0x23403700, 0x07079804, 0xe009d009, 0x092f0209, 0x097f096f, + 0x00980003, 0xe0031003, 0x0403f003, 0x02020603, 0x00030307, 0x2f33333f, 0x5d2f0133, 0x125d5de1, 0x31e12f39, 0x23030130, 0x23032103, 0x294a0103, + 0x2d022973, 0x05297229, 0x02f0fdb6, 0x02f0fd10, 0x02000010, 0x00003300, 0xb605f804, 0x1f001b00, 0x58409900, 0x1a1a0303, 0x1d1e1618, 0x17060407, + 0x00190617, 0x05040401, 0x211818b1, 0x081c1f15, 0x14140904, 0x0b0e0f12, 0x0ab11304, 0x10011050, 0x090c0c10, 0x0a010a50, 0x0d48011c, 0x0cae0d01, + 0x1f0c0408, 0x11ae1000, 0x3f111519, 0xdf114f11, 0x110c0311, 0x1705110c, 0x050a0613, 0x3f332f00, 0x39391233, 0x115d2f2f, 0xe1103333, 0x33113232, + 0x5de11033, 0x2f013232, 0x2f33335d, 0x105d2f33, 0x323917e4, 0x39171211, 0xe42f3311, 0x33323917, 0x39171211, 0x332f3311, 0x0130312f, 0x21152103, + 0x21132303, 0x21132303, 0x21132135, 0x33132135, 0x33132103, 0x01152103, 0x03211321, 0x18013fd7, 0x9352cdfe, 0x52ddfe54, 0xfefe4e90, 0xfe411d01, + 0x522b01ee, 0x25015293, 0x01549054, 0x01ebfc06, 0xddfe4023, 0xb8fe7d03, 0x0154fe89, 0x0154feac, 0x480189ac, 0xfeb00189, 0xfeb00150, 0xb8fe8950, + 0x03004801, 0x89ff7b00, 0x1206d903, 0x36002d00, 0xb4003f00, 0x2f343340, 0x2f290129, 0x06210121, 0x013c2f70, 0x011e2f3c, 0x0720131e, 0x0d070701, + 0x0f246e2e, 0x02001f00, 0x000100ff, 0x00070100, 0x370d0d41, 0x0119e06e, 0xc0ffb819, 0x0b083040, 0x14331948, 0x3c3c2914, 0x08372e13, 0x2173283d, + 0x401f2125, 0x1f48110e, 0x21501e1f, 0x1f210f01, 0x21080221, 0x08731334, 0x0606080e, 0xffb80805, 0x0d0ab3c0, 0x2f000848, 0x2f33332b, 0xe1103311, + 0x5d5e2f32, 0x2f33335d, 0x1033112b, 0x391232e1, 0x11391239, 0x33113333, 0x5d2b2f01, 0x102f33e1, 0x5d5d5ed6, 0x11e13271, 0x335d2f39, 0x5d335d33, + 0x325d32e1, 0x3031325d, 0x020e1401, 0x35231507, 0x27022e22, 0x33031e35, 0x35032e11, 0x37023e34, 0x16153335, 0x26071716, 0x1e112726, 0x2e340703, + 0x36112702, 0x1e140136, 0x06111702, 0x32d90306, 0x8a54855d, 0x54606632, 0x60572120, 0x83592f65, 0x5b312a56, 0x648a4f81, 0x384243a9, 0x87584a8c, + 0x14b02e5b, 0x5d33462b, 0x1112fe5b, 0x59314228, 0x46be0153, 0x0c375472, 0x1209dde6, 0x10ac111a, 0x01111a21, 0x55421eb2, 0x6f434a6e, 0xb4093553, + 0x1f2a05b0, 0x06291991, 0x421f5afe, 0x21486b53, 0x12262d37, 0x620e8bfe, 0x3924a302, 0x0111262f, 0x00591071, 0x66000500, 0x3306ecff, 0x0900cb05, + 0x27001d00, 0x3f003b00, 0x3cb25d00, 0xffb83e10, 0x3c3340f0, 0x283e3c3e, 0x32b41e14, 0x28b423b5, 0x01410f41, 0xb50ab405, 0x1410b400, 0x14301420, + 0x063f1403, 0xb625183e, 0xb621b737, 0xb603192d, 0xb607b70f, 0x3f000719, 0x3fe1f4e1, 0x3fe1f4e1, 0x5d2f013f, 0x5de1f4e1, 0xf4e1de10, 0x391211e1, + 0x382f2f39, 0x13303138, 0x32331614, 0x22231011, 0x0e140506, 0x2e222302, 0x3e343502, 0x1e323302, 0x16140102, 0x10113233, 0x05062223, 0x23020e14, + 0x35022e22, 0x33023e34, 0x01021e32, 0xfa012301, 0x9c9c5047, 0xc7014750, 0x4f734a24, 0x264c7049, 0x4e714923, 0x274d714b, 0x5047ac01, 0x47509c9c, + 0x4a23c601, 0x704a4f73, 0x4923264b, 0x714b4e71, 0x00ff274c, 0x039ed5fc, 0xa502042c, 0x014a01a5, 0x6ca5a348, 0x3f3f76ac, 0x6c6cac76, 0x3e3e75aa, + 0x4afdaa75, 0x4901a4a5, 0xa5a34801, 0x3f76ab6c, 0x6cab763f, 0x3e75aa6c, 0x03aa753e, 0x054afa92, 0x030000b6, 0xecff6d00, 0xcd057d05, 0x21001100, + 0x80005300, 0x18274d40, 0x49044a17, 0x0a48492c, 0x47413647, 0x01422042, 0x42364236, 0x3c3b051d, 0x00480447, 0x102c0047, 0x2c08022c, 0x4820222c, + 0x1d554801, 0x20221048, 0x41220222, 0x51123141, 0x1827164f, 0x0f044a17, 0x053b3c47, 0x15493104, 0x0031500f, 0x123fe12f, 0x17123917, 0x11e13f39, + 0x2f012f39, 0xc610e15d, 0x2f39115d, 0x11e15d5e, 0x39123917, 0x5d2f2f39, 0xe110e110, 0x12113311, 0x30313917, 0x021e1401, 0x35033e17, 0x23022e34, + 0x32130622, 0x0137023e, 0x1415030e, 0x3425021e, 0x2e37023e, 0x3e343503, 0x1e323302, 0x0e141502, 0x3e010702, 0x0e333703, 0x23010703, 0x23030e27, + 0x01022e22, 0x342110a6, 0x38563b24, 0x422f191c, 0x8764562a, 0x4854623a, 0x347dfe20, 0x231c3750, 0x7dfe6042, 0x476f4d28, 0x1c2d3c1f, 0x588a5e32, + 0x305b8353, 0x3c6d5432, 0x2b1b6001, 0xb80a1b22, 0x4135290f, 0xe1150127, 0x6c6031a8, 0xa7694e7c, 0x8d043d73, 0x43414122, 0x403e2325, 0x3d242946, + 0xfb59192c, 0x362817af, 0x2197011f, 0x3855483f, 0x24415b36, 0x647a4ef0, 0x4d242a56, 0x4b396357, 0x2b2b5377, 0x404b7753, 0x244f5d6d, 0x3c1d8cfe, + 0x422f4e44, 0x2955626f, 0x2dacdbfe, 0x351b3147, 0x01009567, 0xa6038500, 0xb6054a01, 0x2a000300, 0x05c01c40, 0x05e005d0, 0x6f052f03, 0x98000205, + 0x03100300, 0x03f003e0, 0x02020304, 0x3f000303, 0x2f012f33, 0x5d5de15d, 0x03013031, 0x4a010323, 0x05297329, 0x02f0fdb6, 0x01000010, 0xbcfe5200, + 0xb6052b02, 0x1a001300, 0x0e060d40, 0x3ff009f2, 0x0e000100, 0x00f805f9, 0x2f013f3f, 0x32e4e15d, 0x34133031, 0x3337023e, 0x14150206, 0x2317021e, + 0x2452032e, 0xac4e714a, 0x4725918c, 0x4eaa456a, 0x02244a71, 0xe5f37d31, 0xfec15dd3, 0xec77f432, 0x5a5ed4e2, 0x00f0e1ce, 0x3d000100, 0x1702bcfe, + 0x1300b605, 0x0e401c00, 0x0bf20e06, 0x0100b0f0, 0xf80e1500, 0x3f00f905, 0xde10013f, 0x32e4e15d, 0x14013031, 0x2307020e, 0x3435033e, 0x1e332702, + 0x24170203, 0xaa4e714b, 0x24486a45, 0x4eac8d90, 0x02244b71, 0xe1f07c31, 0xd45e5ace, 0xf477ece2, 0x5dc1ce01, 0x00f3e5d3, 0x52000100, 0x14047702, + 0x0e001406, 0x15402400, 0x0001101f, 0x800e0098, 0x030e900e, 0x061f0e08, 0x00060601, 0x323f0000, 0x2f015d2f, 0x5de55d5e, 0x03013031, 0x13051725, + 0x27030307, 0x05372513, 0x2b980203, 0xfe1a8d01, 0xb0b2f586, 0xfef2b89e, 0x87011d89, 0xfe14062b, 0x1cc16f77, 0x0160bafe, 0x609afe66, 0xc11c4601, + 0x0089016f, 0x66000100, 0x02040601, 0x0b00a204, 0x18402900, 0x06010d10, 0xef03aa09, 0x00200100, 0x00a00060, 0x00090003, 0xb30306ad, 0xe1333f00, + 0x5d2f0132, 0x32e1325d, 0x0130315d, 0x11213521, 0x15211133, 0x01231121, 0x017dfee9, 0x83019683, 0x02967dfe, 0x85019687, 0xfe967bfe, 0x0100007f, + 0xf8fe3f00, 0xee007901, 0x38000c00, 0x0ecf1440, 0x900e1001, 0x030ea00e, 0x0c2b0c1b, 0x97010c02, 0xffb80706, 0x100d40c0, 0x075f4814, 0x01071001, + 0x0c9c0607, 0x01ed2f00, 0x2b5d5d2f, 0x5d32ed33, 0x30315d5d, 0x030e1725, 0x033e2307, 0x0f6a0137, 0x332f270e, 0x1d0f8a19, 0xee08161b, 0x7c7a3617, + 0x843d387b, 0x00357d83, 0x52000100, 0x4202d101, 0x03007902, 0x09401500, 0x00400502, 0xb9000001, 0xe12f0001, 0x105d2f01, 0x133031ce, 0x52152135, + 0xd101f001, 0x0100a8a8, 0xe3ff9300, 0xfa009101, 0x35001300, 0x15801b40, 0x15a01590, 0x01151103, 0x00c0960a, 0x340200d0, 0x64004400, 0x04007400, + 0xc0ffb800, 0x480a07b6, 0x0f9b0500, 0x01ed2f00, 0x5d5d2b2f, 0x315d5ded, 0x3e343730, 0x1e323302, 0x0e141502, 0x2e222302, 0x22149302, 0x2f1a1b2e, + 0x22141422, 0x2e1b1a2f, 0x266f1422, 0x0f0f2135, 0x25263521, 0x10102235, 0x01003522, 0x00001400, 0xb605e702, 0x1e000300, 0xb80201b1, 0x0940f0ff, + 0x10000302, 0x00010500, 0x2f3f0003, 0x38330111, 0x33382f32, 0x01013031, 0xe7020123, 0x02b3e0fd, 0xfab60521, 0x00b6054a, 0x62000200, 0x0804ecff, + 0x1300cd05, 0x26002700, 0x6f1e1540, 0x29102900, 0x206f1401, 0x230a010a, 0x19070f73, 0x00190573, 0xe13fe13f, 0xe15d2f01, 0xe1de105d, 0x14013031, + 0x23060602, 0x02262622, 0x36123435, 0x16323336, 0x14051216, 0x3233021e, 0x3435023e, 0x2223022e, 0x0804020e, 0x7fb27133, 0x3973af76, 0x7eb16f33, + 0x3a74b077, 0x421e13fd, 0x6c4d4d6b, 0x451f1f45, 0x6b4d4d6c, 0xdd021e42, 0xc2e8feb1, 0x01c26666, 0x01b1b118, 0x6566c118, 0xb2e8fec1, 0x4b95e096, + 0x97e1944a, 0x4a94e096, 0x00e0944a, 0xb2000100, 0xc7020000, 0x1000b605, 0x21403500, 0x0f011240, 0x000e0e01, 0xff01bf6e, 0x017e0201, 0x10010001, + 0x40012001, 0x01060401, 0x00060f0d, 0x3f3f0018, 0x5e2f01cd, 0xe15d5d5d, 0x33112f33, 0x2130315d, 0x3e341123, 0x030e3702, 0x01270707, 0xb0c70233, + 0x01030301, 0x1e1b1a11, 0x01609415, 0x9103967f, 0x5961622b, 0x181a1222, 0x7b79121b, 0x01002b01, 0x00006000, 0xcb05f003, 0x3c002300, 0x08232040, + 0x251b1b6f, 0x22012510, 0x0121016f, 0x01201111, 0x22080101, 0x16730d10, 0x74220207, 0x3f001801, 0xe13f32e1, 0x01391233, 0x2f335d2f, 0xed103311, + 0x2f33115d, 0x303133e1, 0x01352121, 0x3435033e, 0x2223022e, 0x3e270706, 0x1e323303, 0x0e141502, 0x15010702, 0xfcf00321, 0x4b5e0170, 0x222c5376, + 0x5f35563f, 0x28664599, 0x41766a5c, 0x3b6c9b60, 0x4b815d35, 0xb102e7fe, 0x517d019c, 0x4c818086, 0x203f5a3b, 0x24773c4d, 0x361b2e3f, 0x555b9165, + 0x5196959a, 0x0008d5fe, 0x52000100, 0xee03ecff, 0x3900cb05, 0x0b405d00, 0x30213021, 0x096f1a12, 0xb8006f27, 0x2840c0ff, 0x00481714, 0x203b0900, + 0x124f013b, 0x20061201, 0x0121ab73, 0x0b012179, 0x21080121, 0x2c2f1521, 0x15073573, 0x190e1273, 0xe1333f00, 0x1233e13f, 0x5d5e2f39, 0x39e15d5d, + 0x5d5d2f01, 0x2f32ce10, 0xe110e12b, 0x2f393911, 0x0130312f, 0x07020e14, 0x15161615, 0x23020e14, 0x35272622, 0x32331616, 0x3435023e, 0x2323022e, + 0x3e323335, 0x2e343502, 0x06222302, 0x033e2707, 0x021e3233, 0x532ec103, 0xb8b14774, 0x8aca8441, 0x5755c16d, 0x865c5dcb, 0x62352957, 0x8585598d, + 0x2c557e51, 0x385c4224, 0x5c4aa36b, 0x7d6e5d26, 0x6ea36c46, 0x49600438, 0x0c395878, 0x91b51606, 0x4074a060, 0x2eaa2d22, 0x6c4a2832, 0x3f614443, + 0x4a28971e, 0x52343d66, 0x36431e39, 0x29361f7d, 0x85613618, 0x17000200, 0x3f040000, 0x0a00be05, 0x4e001800, 0x56092c40, 0x00000100, 0x0c116e02, + 0x0320070b, 0x1a030301, 0x77011a10, 0x02188718, 0x01055f18, 0x18060905, 0x05050174, 0x06071102, 0x3f001802, 0x3912333f, 0x32e1332f, 0x5d2f0132, + 0x125d5d33, 0x335d2f39, 0xe1333333, 0x325d2f32, 0x23013031, 0x21112311, 0x11330135, 0x34112133, 0x2337023e, 0x0107030e, 0xb0d53f04, 0x97025dfd, + 0x7bfed5bc, 0x01050403, 0x19150709, 0x65fe0b1a, 0xb8fe4801, 0x039f4801, 0x0130fcd7, 0x757b3864, 0x31142266, 0xfd102e31, 0x010000a0, 0xecff8300, + 0xb605f603, 0x4e002a00, 0x1a261840, 0x102c056f, 0x2427012c, 0x23682824, 0x01235901, 0x0ff02323, 0xffb80f01, 0x081240c0, 0x1d0f480b, 0x15000073, + 0x06247427, 0x0a107315, 0x333f0019, 0x12e13fe1, 0x01e12f39, 0x335d2b2f, 0x335d5d2f, 0x5d331133, 0x33e1de10, 0x32013031, 0x1415021e, 0x2223020e, + 0x3527022e, 0x3233031e, 0x3435023e, 0x0e222326, 0x13270702, 0x03211521, 0x21023636, 0x487fab63, 0x80c58644, 0x525b6333, 0x62592121, 0x7c4f2a63, + 0xa8b02e56, 0x393f3f1b, 0x02375a15, 0x27ecfdb2, 0x81036920, 0x69a06c37, 0x437eb672, 0x141e130a, 0x182417ac, 0x764e250d, 0x05978f51, 0x39040908, + 0xfea6b002, 0x000e065d, 0x71000200, 0x0a04ecff, 0x2b00cb05, 0x37003f00, 0x6e312040, 0x1041220c, 0x3b170141, 0x1000006f, 0x03002000, 0x1d753600, + 0x732c071d, 0x73101927, 0x3f000707, 0x11e13fe1, 0x01e12f39, 0x32e15d2f, 0x32de105d, 0x133031e1, 0x33043e34, 0x17021e32, 0x23262615, 0x07040e22, + 0x33033e33, 0x15021e32, 0x23020e14, 0x01022e22, 0x35023e32, 0x23022e34, 0x15020e22, 0x71021e14, 0x8e5c3515, 0x2e1385c6, 0x23112b2f, 0x895a2b58, + 0x142a4364, 0x39140c03, 0x5f3b5f4c, 0x3e3b6c9a, 0x6466a474, 0x014a80af, 0x48633cdb, 0x63422127, 0x4e6f4342, 0x6e49252b, 0xd0697102, 0x4579a4bf, + 0x05070502, 0x2b0c0c9b, 0x94836c4e, 0x2d3f2450, 0xa5723b1a, 0x7fb6726a, 0xf2a04e44, 0x5329b9fe, 0x6f46577f, 0x4b2f2a4e, 0x85433060, 0x0100436a, + 0x00005a00, 0xb6050604, 0x2b000600, 0xb80006b1, 0x1140f0ff, 0x01020000, 0x08100805, 0x02050201, 0x00060374, 0x3f3f0018, 0x2f0132e1, 0x32ce105d, + 0x382f3911, 0x21303133, 0x21352101, 0x19010115, 0x0efd3302, 0xd5fdac03, 0x91a61005, 0x0300dbfa, 0xecff6a00, 0xcd050004, 0x3a002700, 0x80004a00, + 0x231e5340, 0x0a0f6e32, 0xc36e4828, 0x0205d305, 0x050105b5, 0x104c0f05, 0x6e28014c, 0xd56e3e19, 0x23cc0123, 0x0123ba01, 0x19102323, 0x19021920, + 0x38681e0a, 0x59023898, 0x38280138, 0x38483838, 0x43933803, 0x56432601, 0x43430243, 0x14732d00, 0x00733b19, 0xe13f0007, 0x3911e13f, 0xc15d5d2f, + 0x395d5d5d, 0x5d2f0139, 0x5d5d2f33, 0xe110e15d, 0x32ce105d, 0xe15d5d2f, 0xe1103912, 0x30313911, 0x021e3201, 0x020e1415, 0x15031e07, 0x23020e14, + 0x35022e22, 0x37023e34, 0x3435032e, 0x1403023e, 0x3233021e, 0x3435023e, 0x2727022e, 0x22010606, 0x1e141506, 0x033e1702, 0x02263435, 0x71955435, + 0x60462842, 0x576f3a38, 0xa9794335, 0x75ab6e66, 0x684c2d3d, 0x3f56313a, 0x95724325, 0x684420c7, 0x486b4648, 0x66492724, 0x807e1e3f, 0x7d6a1601, + 0x33573e23, 0x243f5530, 0x2ccd057e, 0x43588458, 0x1c45576c, 0x765f4c1f, 0x68955c49, 0x92653638, 0x60784b5c, 0x491f1c4a, 0x57426d5a, 0xfb2c5883, + 0x3f5935a6, 0x5c412323, 0x48543438, 0x3c0e1f40, 0x6a54039b, 0x40523965, 0x34161833, 0x65365442, 0x0200006a, 0xecff6a00, 0xcb050404, 0x3d002900, + 0x1e403500, 0x006f1539, 0x013f103f, 0x100c6e2f, 0x02202020, 0x1b753420, 0x732a071b, 0x75100725, 0x3f001a07, 0x11e13fe1, 0x01e12f39, 0xe1335d2f, + 0xe1de105d, 0x01303132, 0x23040e14, 0x27022e22, 0x33161635, 0x37023e32, 0x23030e23, 0x35022e22, 0x33023e34, 0x01021e32, 0x15020e22, 0x33021e14, + 0x35023e32, 0x04022e34, 0x5c351504, 0x1385c68e, 0x112c2e2e, 0x872b5823, 0x052b66ae, 0x4c38140d, 0x9a5f3b60, 0x733f3b6c, 0xae6566a5, 0x25fe4a80, + 0x2748633c, 0x42634221, 0x2b4e6e44, 0x036e4925, 0xbed16946, 0x024578a5, 0x9c050605, 0xa15e0c0d, 0x3e2477d6, 0x723b1a2e, 0xb7726aa5, 0xa04e447f, + 0x284701f3, 0x46577f54, 0x2f2a4e6f, 0x4330604b, 0x00426b85, 0x93000200, 0x9101e3ff, 0x13006604, 0x3e002700, 0x29101c40, 0x29902980, 0x1e0429a0, + 0xc014960a, 0x0200d000, 0x00440034, 0x00740064, 0xffb80004, 0x070b40c0, 0x2300480a, 0x0510199b, 0x2f000f9b, 0x01ed3fed, 0x5d5d2b2f, 0x5d32e533, + 0x34373031, 0x3233023e, 0x1415021e, 0x2223020e, 0x3411022e, 0x3233023e, 0x1415021e, 0x2223020e, 0x1493022e, 0x1a1b2e22, 0x1414222f, 0x1b1a2f22, + 0x1414222e, 0x1a1b2e22, 0x1414222f, 0x1b1a2f22, 0x6f14222e, 0x0f213526, 0x2635210f, 0x10223525, 0x03352210, 0x21352791, 0x35210e0e, 0x22342527, + 0x34221010, 0x3f000200, 0x9101f8fe, 0x0c006604, 0x61002000, 0x22102f40, 0x22902280, 0x170422a0, 0xd00dc096, 0x0d64020d, 0x50020d74, 0x0d44010d, + 0x010d3b01, 0x0d2f0d1f, 0x1b0d0d02, 0x020c2b0c, 0x0697010c, 0xc0ffb807, 0x14101140, 0x01075f48, 0x07010710, 0x10129b1c, 0x000c9c06, 0xed3fed2f, + 0x5d5d2f01, 0x32ed332b, 0x5d2f335d, 0x5d5d5d5d, 0x315de55d, 0x0e172530, 0x3e230703, 0x34033703, 0x3233023e, 0x1415021e, 0x2223020e, 0x6a01022e, + 0x2f270e0f, 0x0f8a1933, 0x08161b1d, 0x2e221411, 0x222f1a1b, 0x2f221414, 0x222e1b1a, 0x3617ee14, 0x387b7c7a, 0x7d83843d, 0x27ed0235, 0x0e0e2135, + 0x25273521, 0x10102234, 0x01003422, 0xee006600, 0xdd040204, 0x4e000600, 0x08003040, 0x40010840, 0x02010101, 0x03050501, 0x7f006f06, 0x00300200, + 0x04000001, 0x50010320, 0x80037003, 0xf003d003, 0x033f0503, 0x01030001, 0x2f000306, 0x5d5d5d5e, 0x2f333371, 0x12325d5d, 0x332f3d39, 0x2f180133, + 0xce105d5d, 0x01253031, 0x01150135, 0xfc020401, 0xfd9c0364, 0xeedf0221, 0x0166a801, 0x94fea0e1, 0x0200befe, 0xba016600, 0xe9030204, 0x07000300, + 0x3d405c00, 0x40090207, 0xc6040109, 0x00bb0100, 0x0100a901, 0x7b010086, 0x00680100, 0x01004201, 0x00010039, 0x051fad04, 0x7f02052f, 0x05000105, + 0x06020510, 0xad000505, 0x0f0101f0, 0x02016f01, 0x5d2f0001, 0x2f33e15d, 0x715d5d5e, 0x5d2f01e1, 0x5d5d5d5d, 0x335d5d5d, 0x32ce105d, 0x35133031, + 0x35011521, 0x03661521, 0x0364fc9c, 0x9554039c, 0x9666fe95, 0x01000096, 0xee006600, 0xdd040204, 0x4e000600, 0x08053040, 0x40010840, 0x05060106, + 0x03010104, 0x7f066f00, 0x06300206, 0x02060601, 0x50010320, 0x80037003, 0xf003d003, 0x033f0503, 0x01030001, 0x2f000306, 0x5d5d5d5e, 0x2f333371, + 0x12335d5d, 0x332f3d39, 0x2f180133, 0xce105d5d, 0x01133031, 0x15013501, 0xe0026601, 0x9c0320fd, 0x8f0164fc, 0x6c014201, 0x661ffea0, 0x020058fe, + 0xe3ff2500, 0xcb052503, 0x3b002700, 0x21403e00, 0x28289a32, 0x00004627, 0x1c460b14, 0x013d2f3d, 0x0f170b14, 0x00060100, 0x379b2d00, 0x17511013, + 0xe13f0004, 0x32e52f33, 0x125d5e2f, 0x5d2f0139, 0x11e1de10, 0x33e12f39, 0x3031e12f, 0x3e343501, 0x033e3702, 0x022e3435, 0x07062223, 0x33363627, + 0x15021e32, 0x07020e14, 0x1515030e, 0x023e3403, 0x021e3233, 0x020e1415, 0x022e2223, 0x270f1901, 0x44303242, 0x391e152b, 0x96533855, 0xbc513f46, + 0x68955d61, 0x50361b38, 0x26423436, 0x2214bb0e, 0x2f1a1b2e, 0x22141422, 0x2e1b1a2f, 0x9e011422, 0x505c3925, 0x43292a4d, 0x30354f45, 0x341f394f, + 0x3b2a9122, 0x578b6033, 0x545a6943, 0x3f432d2f, 0xfe122c42, 0x213526d1, 0x35210f0f, 0x22352526, 0x35221010, 0x6d000200, 0x81064aff, 0x5700b605, + 0x6f006800, 0x17583f40, 0x171f2760, 0x01277f01, 0x46462717, 0x4e031727, 0x01002031, 0x6a406a00, 0x4e403b01, 0x0c2c4e01, 0x1207125b, 0x120f1c64, + 0x12bf121f, 0x1c000603, 0x1c120701, 0x36401c12, 0x45400353, 0x332f0049, 0x12c13fc1, 0x2f2f3939, 0x5d5e5d5e, 0x3311c110, 0x3232c110, 0xc15d2f01, + 0x71de105d, 0x391711c1, 0x5d2f2f2f, 0x10c1105d, 0x013031c1, 0x23040e14, 0x27022e22, 0x23030e23, 0x35022e22, 0x33023e34, 0x17021e32, 0x14060603, + 0x1e141514, 0x3e323302, 0x2e343502, 0x04222302, 0x14150206, 0x3233021e, 0x1537023e, 0x22230606, 0x35022624, 0x24361234, 0x16043233, 0x16140112, + 0x023e3233, 0x26263737, 0x020e2223, 0x25138106, 0x3a614c39, 0x2134492d, 0x36120406, 0x4d355947, 0x3b2b5277, 0x2d629e6f, 0x1745525a, 0x15010117, + 0x2e172b22, 0x56182f46, 0xa97bd198, 0x5aaffefe, 0x93e3994f, 0x646f773d, 0x82d8562b, 0xc3e7feb3, 0x01db7666, 0x019cc137, 0xfc6abf06, 0x37556515, + 0x041a324e, 0x2a4d1c0e, 0x1c3f654a, 0x7d3edb02, 0x29486171, 0x2341321e, 0x1c314225, 0x568e6538, 0x447aa865, 0x08110e08, 0x1b1660fe, 0x35030810, + 0x3d0f2844, 0x8e4e8c68, 0x6f4f98dd, 0xa2effec7, 0x52a0ea97, 0x111f180e, 0x662c268d, 0xb31901c3, 0xee4501bc, 0xfebd6588, 0x85d5fef1, 0x73532d77, + 0x0d08fd45, 0x00785e3a, 0x00000200, 0xdd040000, 0x0700bc05, 0x84001400, 0x05062440, 0x46010246, 0x14020114, 0x01084903, 0x08010149, 0x0e0e0001, + 0x10000003, 0x07800107, 0x07d00790, 0xffb80703, 0x061840c0, 0x1007480a, 0x0f160707, 0x2f161f16, 0x9f168f16, 0x0616df16, 0xb8040307, 0x1140f0ff, + 0x0e5f0204, 0x480e0a20, 0x1414050e, 0x00040305, 0x323f0012, 0x122f393f, 0x01e12b39, 0x5e33382f, 0x2f33115d, 0x715d2b38, 0x39121133, 0x39122f3d, + 0x125d5d39, 0x5d5d3939, 0x30313333, 0x03210321, 0x01330123, 0x032e0301, 0x07030e27, 0xa01f0403, 0xbca2dffd, 0x02aa1902, 0x9467fe1a, 0x12121106, + 0x12120708, 0x01910611, 0x053bfec5, 0x0244fabc, 0x12a8016a, 0x1f413c34, 0x333d421f, 0x0058fe11, 0xc7000300, 0x87040000, 0x1700b605, 0x2f002200, + 0x3e406200, 0x055a1e0b, 0x06e50106, 0xd60206f5, 0x06060106, 0x705b2a24, 0x02118011, 0x7f316711, 0x02318f31, 0x18013110, 0x64175a24, 0x60230b30, + 0x0b011879, 0x18080118, 0x60240018, 0x60221217, 0x3f000300, 0x11e13fe1, 0x5d5e2f39, 0x0139e15d, 0x32e1f610, 0xf6105d5d, 0x3912e15d, 0x715d5d2f, + 0x303132e1, 0x1e322113, 0x0e141502, 0x1e150702, 0x0e141503, 0x13212302, 0x023e3233, 0x23263435, 0x21111123, 0x35023e32, 0x23022e34, 0x808f01c7, + 0x274283c3, 0x45456d4a, 0x41345a79, 0xfe6fb07b, 0x54f4ba1b, 0x9a1f4672, 0x0a01dfa6, 0x20497758, 0x5c7c4b21, 0x5727b605, 0x6c3e678d, 0x0a093752, + 0x784f2d0c, 0x6d9d6456, 0x1e4a033a, 0x783b593b, 0xfd97fd68, 0x654828f0, 0x435e383d, 0x01000025, 0xecff7d00, 0xcb059804, 0x4c002300, 0x0eaf1440, + 0x15400e01, 0x0e0e4818, 0x0120ba18, 0x20702060, 0xffb82002, 0x061840c0, 0x2020480a, 0x0125af25, 0x66185b05, 0x5f002124, 0x0a0d041d, 0x0013135f, + 0x3f33e13f, 0x100133e1, 0x115de1f6, 0x5d2b2f33, 0x2f39125d, 0x30315d2b, 0x020e2201, 0x021e1415, 0x37363233, 0x23030e15, 0x02262622, 0x36123435, + 0x16323336, 0x26260717, 0xae6b1903, 0x763b437b, 0xa05976b0, 0x554e274e, 0xf0a43b61, 0xa9574c9d, 0xc46ca2fa, 0x943f4e4f, 0x98512705, 0xdb8d89da, + 0x17234e96, 0x0e170fa2, 0x01c66c07, 0x01a6a916, 0x2c6ec614, 0x2e209c2a, 0xc7000200, 0xfc040000, 0x0c00b605, 0x26001700, 0x5b0d1540, 0x10196700, + 0x5a140119, 0x13186406, 0x14030760, 0x00120660, 0xe13fe13f, 0xe1f61001, 0xe1f6105d, 0x14013031, 0x23040602, 0x32211121, 0x07121616, 0x23022e34, + 0x20331123, 0x60fc0400, 0xa8f7feb6, 0x970192fe, 0x5faef899, 0xb87e42c5, 0x01a2c975, 0x020c0108, 0xe9feb9e9, 0xb6055ebb, 0xf4feb55c, 0x8ad592b6, + 0x0189fb43, 0x01000024, 0x0000c700, 0xb605be03, 0x42000b00, 0x08142640, 0x01080801, 0x0d670004, 0x015a0a06, 0x5f090c64, 0x0f01064f, 0x0206af06, + 0x0a060608, 0x03025f05, 0x12015f0a, 0x3fe13f00, 0x2f3912e1, 0xe1715d5e, 0xe1f61001, 0x32e61032, 0x5d2f3911, 0x21213031, 0x21152111, 0x21152111, + 0xbe032111, 0xf70209fd, 0x1702c3fd, 0x3d02e9fd, 0xfea4b605, 0xf8fda23c, 0xc7000100, 0xbe030000, 0x0900b605, 0x11407000, 0x0f010808, 0x03ff0103, + 0x90038001, 0x0303d003, 0xc0ffb803, 0x0a073840, 0x0b030348, 0x0b2f0b0f, 0x0baf0b8f, 0x00060704, 0x0a64015a, 0x060f5f09, 0x3f060f01, 0xff066f06, + 0x06080406, 0x481d1a40, 0x15104006, 0x00060648, 0x03025f05, 0x3f001200, 0x3912e13f, 0x5e2b2b2f, 0x01e1715d, 0x32e1f610, 0x33115d5e, 0x5d5d2b2f, + 0x2f391271, 0x23213031, 0x21152111, 0x21152111, 0x02ba8101, 0x02c3fdf7, 0x05e9fd17, 0xfcfda4b6, 0x010000a4, 0xecff7d00, 0xcb05f204, 0x37002b00, + 0x2b2b1e40, 0x145a290c, 0x102d6702, 0x5b1f012d, 0x2b2c660c, 0x2400005f, 0x04115f1a, 0x13075f24, 0x3fe13f00, 0x2f3912e1, 0xf61001e1, 0xf6105de1, + 0x3911e132, 0x0130312f, 0x030e1121, 0x26262223, 0x12343502, 0x32332436, 0x2e071716, 0x0e222303, 0x1e141502, 0x3e323302, 0x21113702, 0xe4010e03, + 0x82767037, 0xa6f29d4b, 0x01b65f56, 0xcc6fab0b, 0x53244858, 0x7a2e5d58, 0x37427fbc, 0x2c86be78, 0x1a373e49, 0x0403d5fe, 0x1c1233fd, 0xc3690a13, + 0xacae1701, 0x69c31601, 0x11a22a2c, 0x510e171e, 0x8289da98, 0x05569cd8, 0x01050b08, 0x010000b4, 0x0000c700, 0xb605d504, 0x3d000b00, 0x01092340, + 0x0d65005a, 0xbf010dc0, 0x0d20010d, 0x5a040801, 0x030c6405, 0x01080f5f, 0x0a080808, 0x00050306, 0x323f0012, 0x2f39333f, 0x01e15d5e, 0x32e1f610, + 0x105d5d5d, 0x3132e1f6, 0x11232130, 0x11231121, 0x11211133, 0xbad50433, 0xbaba66fd, 0x02ba9a02, 0x0556fdaa, 0x0298fdb6, 0x01000068, 0x00005200, + 0xb6056402, 0x57000b00, 0x0d0b2640, 0x7b020d2b, 0xab0d9b0d, 0x040dfb0d, 0x2b010d54, 0x4b0d3b0d, 0x0d1f030d, 0x0b080201, 0x02055a0a, 0x030103c9, + 0x40f8ffb8, 0x48100d10, 0x06010300, 0x06040903, 0x000a0303, 0xc13f0012, 0x32c13f32, 0x5d5e2f01, 0x32c15d2b, 0x5f32c1f1, 0x5d5d5d5d, 0x21303171, + 0x11373521, 0x15213527, 0x02171107, 0xaceefd64, 0xac1202ac, 0x042966ac, 0x66662998, 0x2968fb29, 0x48ff0100, 0x73017bfe, 0x1300b605, 0x1c402f00, + 0x600115df, 0x02157015, 0x0f01152f, 0x03030c5a, 0x0c100c00, 0x0d0c0702, 0x005f0703, 0x3fe12f00, 0x5d5e2f01, 0xe1102f33, 0x315d5d5d, 0x26220330, + 0x16163527, 0x023e3233, 0x11331135, 0x1d020e14, 0x221c4c33, 0x4b252d4e, 0x3bbb263d, 0x7bfe9369, 0x09a00b0d, 0x5832130b, 0xfab60544, 0x659a695e, + 0x01000031, 0x0000c700, 0xb605a204, 0x64000c00, 0x0c022d40, 0x0c010c66, 0x100b0a00, 0x00010b0b, 0x02001000, 0x00100007, 0x0eb00e00, 0x010e2f01, + 0x08010e10, 0x64055a04, 0x0b10020d, 0xb8084810, 0x0c40f0ff, 0x0248100b, 0x060a0508, 0x12050003, 0x3f333f00, 0x39391233, 0x10012b2b, 0x5d32e1f6, + 0x33115d5d, 0x5d5e382f, 0x382f3333, 0x5d391133, 0x30313311, 0x07012321, 0x33112311, 0x33013711, 0xd3a20401, 0xba8b3dfe, 0xc40179ba, 0x02f8fdd1, + 0xb8fd72ba, 0x25fdb605, 0xfd3302a8, 0x01000083, 0x0000c700, 0xb605be03, 0x23000500, 0x07041340, 0x100107af, 0x5a030107, 0x01066400, 0x005f0303, + 0xe13f0012, 0xf610013f, 0x115d5de1, 0x33303133, 0x21113311, 0x02bac715, 0xfab6053d, 0x0100a6f0, 0x0000c700, 0xb6052f06, 0x8b001900, 0x19361340, + 0x01003901, 0x0c080e17, 0x0e39480f, 0x5a110e01, 0xf8ffb819, 0x0f0c1c40, 0x08001948, 0x00480f0c, 0x090c0d0d, 0x4f1b6510, 0x1b20011b, 0x011b0f01, + 0xffb80b08, 0x0c1a40f8, 0x0b26480f, 0x08020b01, 0x1a64095a, 0x10010118, 0x01481209, 0x11030b0e, 0xf0ffb80c, 0x481209b6, 0x1200080c, 0x32323f00, + 0x333f322b, 0x33112b33, 0xe1f61001, 0x2b5d3232, 0x5d5d5d5e, 0x3911f610, 0x2b331139, 0x32e12b33, 0x31322b5d, 0x215d5d30, 0x17162301, 0x11151616, + 0x01211123, 0x11210133, 0x36341123, 0x23373637, 0xfe230301, 0x04060845, 0x01ac0504, 0x069c0114, 0x14019e01, 0x040304ba, 0x41fe0803, 0x494a0005, + 0xfc398b3f, 0xfbb60596, 0xfaa80458, 0x3477034a, 0x49473d86, 0x010002fb, 0x0000c700, 0xb6050e05, 0x51001700, 0x280e2940, 0x15010101, 0x1965005a, + 0x8f0119b0, 0x19000119, 0x27021910, 0x030c010c, 0x640a5a09, 0x10021618, 0x02481806, 0xb80d030b, 0x06b6f0ff, 0x0a0d4818, 0x3f001200, 0x3f2b3232, + 0x01332b33, 0x32e1f610, 0x5d5d5d32, 0xe1f6105d, 0x31325d32, 0x01232130, 0x16171623, 0x23111516, 0x33013311, 0x032e2726, 0x05331135, 0x31fdd70e, + 0x04040608, 0x02d5ac05, 0x040307cc, 0x01030301, 0x4dba04ae, 0x398e414c, 0xb605e7fc, 0x4a4c4cfb, 0x3e434320, 0x0020031a, 0x7d000200, 0x7105ecff, + 0x1300cd05, 0x34002700, 0x5b1e2040, 0xc0296700, 0x29bf0129, 0x01297001, 0x295f292f, 0x0a5b1402, 0x5f232866, 0x5f19040f, 0x3f001305, 0x01e13fe1, + 0x5de1f610, 0x105d5d5d, 0x3031e1f6, 0x06021401, 0x26222306, 0x34350226, 0x33363612, 0x12161632, 0x021e1405, 0x023e3233, 0x022e3435, 0x020e2223, + 0xa0517105, 0xefa39bed, 0x9e4c4c9d, 0xeb9ba3f0, 0xd1fb51a0, 0x72a56b34, 0x326ba572, 0x72a46a32, 0x346ca672, 0xfea9dd02, 0x6c6cc6ea, 0xaa1701c6, + 0xc41501aa, 0xfec56b6b, 0xdb89abeb, 0x99515199, 0xda8a89db, 0x97515197, 0x020000da, 0x0000c700, 0xb6053304, 0x19000e00, 0x2c404600, 0x00285b15, + 0x00480038, 0x1b670003, 0x40011bcf, 0x1b0f011b, 0x070f0601, 0x1a64085a, 0x0630600f, 0x06020640, 0x60190706, 0x12070309, 0xe13f3f00, 0x5d2f3911, + 0xf61001e1, 0x5d5e32e1, 0xf6105d5d, 0x3031e15d, 0x020e1401, 0x23112323, 0x1e322111, 0x32330102, 0x3435023e, 0x04232326, 0xcf7e3733, 0x01ba9698, + 0x7ec2866a, 0x814efd3c, 0x2e5b8b5d, 0x04a0aea4, 0x81a85b0a, 0x05c7fd4d, 0xa06d39b6, 0x472067fe, 0x898e5171, 0x7d000200, 0x710562fe, 0x1d00cd05, + 0x38003100, 0x5b282240, 0xc0336700, 0x33bf0133, 0x01337001, 0x335f332f, 0x145b1e02, 0x5f2d3266, 0x5f230419, 0x09130f05, 0x333f2f00, 0x01e13fe1, + 0x5de1f610, 0x105d5d5d, 0x3031e1f6, 0x020e1401, 0x17161607, 0x27262607, 0x22230606, 0x35022626, 0x36361234, 0x16163233, 0x1e140512, 0x3e323302, + 0x2e343502, 0x0e222302, 0x31710502, 0x2b5d8e5f, 0x67795a89, 0x291133ad, 0x9defa312, 0xf09e4c4c, 0xa0eb9ba3, 0x34d1fb51, 0x7272a56b, 0x32326ba5, + 0x7272a46a, 0x02346ca6, 0xb5e283dd, 0x8f5e2684, 0xc6498e3c, 0x6c02027f, 0xaa1701c6, 0xc41501aa, 0xfec56b6b, 0xdb89abeb, 0x99515199, 0xda8a89db, + 0x97515197, 0x020000da, 0x0000c700, 0xb605a004, 0x1c000f00, 0x56408200, 0x0f190f09, 0x010ff902, 0x0f0b080f, 0x090c0f48, 0x0c07010c, 0x085b1601, + 0x02071807, 0x0e090707, 0xf90ee901, 0x080e020e, 0x0e480f0b, 0x1e0d100d, 0x1e8f1e3f, 0x1ebf1e9f, 0x20051edf, 0x0110011e, 0x1d64025a, 0x6010030c, + 0x08010000, 0x1c010000, 0x0e030360, 0x3f001201, 0x11e13f33, 0x5d5e2f39, 0x013912e1, 0x32e1f610, 0xce105d5d, 0x5d2b3238, 0x5d2f3271, 0x5e3912e1, + 0x2b33115d, 0x3031715d, 0x11231101, 0x15162021, 0x07020e14, 0x27012301, 0x023e3233, 0x022e3435, 0x81012323, 0x016401ba, 0x5131fe0a, 0x8e013768, + 0xe5a1fedb, 0x517e5aa4, 0x7f532925, 0x5c02a057, 0xb605a4fd, 0x8257d1ce, 0xfd143e5d, 0x9e5c0271, 0x45674523, 0x1d406448, 0x68000100, 0xc903ecff, + 0x3300cb05, 0x27404200, 0x23012359, 0x67005a11, 0xff35bf35, 0x35600235, 0x01353f01, 0x1b095a2a, 0x2a113466, 0x245f2705, 0x600e0420, 0x00130509, + 0x3fe1333f, 0x3912e133, 0xf6100139, 0x5d5de132, 0xe1f6105d, 0x30315d33, 0x020e1401, 0x27262223, 0x33031e35, 0x34353632, 0x2e27022e, 0x3e343503, + 0x16323302, 0x26260717, 0x15062223, 0x17021e14, 0xc903031e, 0x73b88045, 0x2241c16f, 0x32666057, 0x491d99a0, 0x83595d7a, 0x74402955, 0xbe7761a1, + 0xa541434a, 0x1e867a58, 0x5b547346, 0x012f5c89, 0x6a996187, 0xb2222337, 0x0f181f10, 0x50367078, 0x23253f43, 0x54846853, 0x325f8a58, 0x1d9c232d, + 0x3960712b, 0x213b4353, 0x7e604c24, 0x14000100, 0x12040000, 0x0700b605, 0x32405e00, 0xd001090f, 0x094f0109, 0x100209cf, 0x30092009, 0x06af0309, + 0x840206ef, 0x06060106, 0x40025a07, 0x0203e003, 0x0801030f, 0x02570303, 0x02770267, 0xffb80203, 0x070b40c0, 0x0702480a, 0x03045f03, 0x3f001200, + 0x0132e13f, 0x335d2b2f, 0x5d5d5e2f, 0x2f32e110, 0x5d5d5d5d, 0x3031715d, 0x21112321, 0x21152135, 0xfebb7102, 0xfefe035e, 0xa412055f, 0x010000a4, + 0xecffb800, 0xb805dd04, 0x2f001700, 0x5a161c40, 0xb0196501, 0x196f0119, 0x100219af, 0x5a0e0119, 0x1118640b, 0x0c13065f, 0x3f000300, 0x01e13f32, + 0x5de1f610, 0xf6105d5d, 0x013031e1, 0x020e1411, 0x022e2223, 0x11331135, 0x32331614, 0x1137023e, 0x8542dd04, 0xc48088c9, 0xadbb4485, 0x528059af, + 0xb8050128, 0xc4724cfc, 0x8e4d5290, 0xae037ac7, 0xc0af48fc, 0x51886236, 0x0100b803, 0x00000000, 0xb6058b04, 0x6c000c00, 0x02031040, 0x00040909, + 0x01700160, 0x01f001b0, 0xffb80104, 0x061640c0, 0x1001480a, 0x2f0e0101, 0xbf0e7f0e, 0x400e030e, 0x05480906, 0xf0ffb804, 0x040504b4, 0xffb80903, + 0x110ab3e0, 0xffb80948, 0x060a40f0, 0x02094809, 0x01001203, 0x333f0003, 0x2b33333f, 0x01333f2b, 0x2b33382f, 0x2f33115d, 0x335d2b38, 0x2f3d3912, + 0x30313333, 0x23013301, 0x16013301, 0x36361716, 0xc6c50337, 0xfebb17fe, 0x2701c519, 0x0f112a1d, 0xb6051f2e, 0xb6054afa, 0xa95b61fc, 0x61a94a4a, + 0x14000100, 0xfe060000, 0x2a00b605, 0x10b6df00, 0x48181508, 0xffb80f10, 0x1815b5f8, 0x01070f48, 0x40f8ffb8, 0x48181512, 0x15080001, 0x16004818, + 0x1815081d, 0xb81c1d48, 0x2f40f8ff, 0x1c481815, 0x01071425, 0x07440704, 0x070307b4, 0x25142504, 0x25442524, 0x07052554, 0x16252516, 0x0d1e0307, + 0x70010e00, 0xc00e800e, 0xb80e030e, 0x1840c0ff, 0x0e480a07, 0x2c0e0e10, 0x2c7f2c6f, 0x302c2002, 0x2c0f022c, 0x1e1f0801, 0x40f0ffb8, 0x20161e13, + 0x1648110a, 0x48090610, 0x1e000d16, 0x25250703, 0xb3e0ffb8, 0x2548110a, 0xb6f0ffb8, 0x25480906, 0x00121d10, 0x2b33333f, 0x3f33112b, 0x2b333333, + 0x382f012b, 0x5d5d5e33, 0x2f33115d, 0x715d2b38, 0x39171233, 0x2f2f2f3d, 0x715d5e5d, 0x332b3311, 0x2b33112b, 0x33112b33, 0x312b332b, 0x13330130, + 0x3e17031e, 0x33133703, 0x26032301, 0x27262726, 0x06060706, 0x01230307, 0x031e1333, 0x37033e17, 0xe5c52903, 0x13191d0f, 0x100c0406, 0xc7c80b13, + 0xfebc91fe, 0x0c0b1a0e, 0x0a0b0b0b, 0xbcf20e19, 0xdfc57efe, 0x0e11140c, 0x140f0505, 0xb6050d17, 0x7038a8fc, 0x26265e69, 0x3167635a, 0x4afa7203, + 0x6c33aa03, 0x3334372f, 0x36702f37, 0xb6055cfc, 0x632e87fc, 0x25265b62, 0x316f6c62, 0x00000100, 0x60040000, 0x0b00b605, 0x1d408100, 0x0a100a09, + 0x0b37000a, 0x38080b01, 0x02050105, 0x00000102, 0x80007001, 0x0300c000, 0xc0ffb800, 0x0a071440, 0x00100048, 0x0d0f0d00, 0x0d2f0d1f, 0x08040d7f, + 0xffb80607, 0x0606b3f0, 0xffb80403, 0x041040f0, 0x27010228, 0x08020108, 0x03060904, 0x00120004, 0x333f323f, 0x5d393912, 0x382f015d, 0x382f3232, + 0x115d5e33, 0x2b382f33, 0x3933715d, 0x5d332f3d, 0x115d3333, 0x382f1833, 0x21303133, 0x23010123, 0x01330101, 0x04013301, 0x9efed360, 0x01bc91fe, + 0xc65afec5, 0x4e014c01, 0x025bfebe, 0x0285fd7b, 0xfdba02fc, 0xfd2f02d1, 0x0100004c, 0x00000000, 0xb6053704, 0x73000800, 0x0aef1940, 0x09400a01, + 0xab08480c, 0x07980107, 0x01074001, 0x0f01071b, 0xb8070107, 0x2f40f0ff, 0x01050707, 0x4f010280, 0x021b0102, 0x02100201, 0x5a040002, 0x05870577, + 0x4f030597, 0x05000105, 0x07020510, 0x01003605, 0x12040100, 0x00030107, 0x123f333f, 0x2f015d39, 0x5d5d5d5e, 0x2f3239e1, 0x5d5d5d38, 0x2f331133, + 0x5d5d5d38, 0x2b335d5d, 0x0130315d, 0x11013301, 0x33011123, 0x54011b02, 0xbb42fec8, 0x02cb42fe, 0xfce302d3, 0x02c7fd83, 0x0087032f, 0x52000100, + 0xfe030000, 0x0900b605, 0x20403800, 0x07030909, 0x0b0f0b67, 0x0b4f0b3f, 0x08040b9f, 0x01040408, 0x04070a66, 0x0203055f, 0x12015f08, 0x39e13f00, + 0x0139e13f, 0x2f32e610, 0x105d5e32, 0x2f3232e6, 0x21213031, 0x35210135, 0x21011521, 0x54fcfe03, 0x4dfdc702, 0x3afd8303, 0x0491db02, 0xfb91a67f, + 0x01000081, 0xbcfea400, 0xb6053902, 0x26000700, 0x00041740, 0x00f106f3, 0xb0011001, 0x0401c001, 0x02f50501, 0x01f506f8, 0xe13f00f9, 0x2f01e13f, + 0x32ede15d, 0x21013031, 0x23152111, 0x39023311, 0x95016bfe, 0xbcfedfdf, 0xfa95fa06, 0x01000031, 0x00001700, 0xb605e902, 0x21000300, 0x010102b7, + 0x00050110, 0xf0ffb803, 0x010203b4, 0x3f000300, 0x2f012f2f, 0x33113338, 0x31331138, 0x23011330, 0x2002c901, 0x05e0fdb2, 0x054afab6, 0x010000b6, + 0xbcfe3300, 0xb605c901, 0x24000700, 0x00031440, 0x60f101f3, 0x02067006, 0xf5000906, 0xf503f907, 0x3f00f804, 0x01e13fe1, 0xe15dd610, 0x303132ed, + 0x23113317, 0x21112135, 0x01dfdf33, 0xae6afe96, 0xf995cf05, 0x01000006, 0x25022900, 0xc1051904, 0x12000600, 0x080303b6, 0x06010000, 0x01cd3f00, + 0x2f33112f, 0x01133031, 0x01230133, 0xcb012901, 0xa1bf0166, 0xa3feaffe, 0x9c032502, 0xdf0264fc, 0x010021fd, 0xbcfefcff, 0x48ff4e03, 0x12000300, + 0x050000b6, 0x02ba0101, 0x01e12f00, 0x2f33112f, 0x21013031, 0x4e032135, 0x5203aefc, 0x008cbcfe, 0x89010100, 0x1203d904, 0x0d002106, 0x0a401600, + 0x80080600, 0x005f000f, 0x2f000002, 0x01cc1a5d, 0x3031cd2f, 0x032e2301, 0x1e333527, 0x12031703, 0x4d522378, 0x10db103f, 0x15302e2b, 0x531cd904, + 0x151b5158, 0x4c515122, 0x0200001d, 0xecff5e00, 0x5e049c03, 0x32002300, 0x11405400, 0x47290110, 0x0f345523, 0x02346f34, 0x0c483006, 0xd0ffb81a, + 0x110d1e40, 0x09101a48, 0x1a1a480c, 0x1933560c, 0x2a1d5016, 0x1d101052, 0x02502410, 0x15001607, 0x333f3f00, 0x2f393fe1, 0x32e110e1, 0x32e61001, + 0x102b2b2f, 0x105d5ee1, 0x3232e1f6, 0x27213031, 0x23030e23, 0x35022e22, 0x37373634, 0x022e3435, 0x07062223, 0x33363627, 0x15021e32, 0x3e322511, + 0x07353502, 0x1415030e, 0x25190316, 0x4e422108, 0x74453f60, 0xece73055, 0x51371db8, 0x428f5334, 0x64b64a40, 0x30619566, 0x683d2ffe, 0x5a8f2b4c, + 0x6120497a, 0x2a412d98, 0x7b512714, 0x08b0a454, 0x5a434507, 0x22301837, 0x29382889, 0xfd628a59, 0x4d267f10, 0x07634f75, 0x51392004, 0x00565c33, + 0xae000200, 0x3f04ecff, 0x1f001406, 0x38002f00, 0x05482db5, 0xb8313157, 0x1740b8ff, 0x1015490a, 0x54124725, 0x12001330, 0x0f502a15, 0x5020160a, + 0x0010001b, 0x3fe1323f, 0x3f3fe133, 0xe1f61001, 0x102b3232, 0x3031e1f6, 0x021e3201, 0x020e1415, 0x022e2223, 0x23072327, 0x14113311, 0x07060706, + 0x17033e33, 0x15020e22, 0x33021e14, 0x34353632, 0x5e9e0226, 0x3c3c6d9a, 0x3b5e9a6d, 0x173b4d60, 0xb685250c, 0x02020202, 0x4d3a1708, 0x6b501960, + 0x411b1b41, 0x7f87516c, 0x485e047f, 0x8d8cd48f, 0x1a4990d5, 0x8b203a2b, 0x88fe1406, 0x28224f23, 0x2c3c2326, 0x68339719, 0x9d65699c, 0xccda376b, + 0x0100ced0, 0xecff7100, 0x5e046f03, 0x2a001f00, 0x0d1d1840, 0x7f215f21, 0x21100221, 0x05481601, 0x51132056, 0x5119100a, 0x3f001600, 0x01e13fe1, + 0x5de1f610, 0x32ce105d, 0x22053031, 0x3435022e, 0x3233023e, 0x2e071716, 0x06222303, 0x33161415, 0x15373632, 0x52020606, 0x4a82b065, 0x66b2854c, + 0x3632954e, 0x3a3c3817, 0x91909d1a, 0x36835194, 0x3f147b36, 0x9d96d589, 0x223e89db, 0x130a9a19, 0xd4c9090f, 0x1925c3d3, 0x001e1da2, 0x71000200, + 0x0204ecff, 0x1f001406, 0x34003000, 0x00261d40, 0x551e471b, 0x01321032, 0x560b482e, 0x1c151f31, 0x16502b00, 0x50201010, 0x00160601, 0x3fe1333f, + 0x3f3fe133, 0xe1f61001, 0xe1f6105d, 0x30313232, 0x030e2325, 0x022e2223, 0x023e3435, 0x021e3233, 0x27263317, 0x11352626, 0x25231133, 0x37023e32, + 0x022e3435, 0x15062223, 0x54031614, 0x4d3b1608, 0x9a5d3c60, 0x6e3c3c6e, 0x603b5d9a, 0x0c163c4d, 0x04020303, 0xc6fe93b6, 0x1f41694c, 0x6c411b02, + 0x7f7f8751, 0x2e3d2293, 0xd48f481a, 0x90d58d8c, 0x3a2c1a49, 0x1a1f2220, 0xb4011037, 0x2e83ecf9, 0x295e8d5e, 0x376b9d65, 0xcdd1ccda, 0x71000200, + 0xe103ecff, 0x1e005e04, 0x43002700, 0x48232840, 0x29571019, 0x6f0129e0, 0x11220129, 0x28560548, 0x221b5011, 0x0f02222b, 0x22060122, 0x501f1422, + 0x5114100a, 0x3f001600, 0x12e13fe1, 0x5d5e2f39, 0x1001e15d, 0x5d32e1f6, 0x32f6105d, 0x053031e1, 0x35022e22, 0x33023e34, 0x15021e32, 0x16162115, + 0x023e3233, 0x030e1537, 0x07062203, 0x022e3421, 0xb66e6002, 0x78424883, 0x9e6365a7, 0x4cfd3b6e, 0x33979905, 0x274c5157, 0x57514d28, 0x0b857260, + 0x391bec01, 0x8e4a1458, 0xd68887d2, 0x81474e95, 0xc1716eb5, 0x1d130ab6, 0x1c13a212, 0xdb030812, 0x7144959c, 0x01002c50, 0x00001d00, 0x1f06f002, + 0x70001b00, 0x1dcf4e40, 0x60021ddf, 0x901d801d, 0x041da01d, 0x1d3f1d1f, 0x1b031d4f, 0xbf107f1b, 0x10100210, 0x0347021a, 0x050f0307, 0x052f051f, + 0x050405af, 0x10030005, 0x80032003, 0xa0039003, 0x03060603, 0x074f0501, 0x07011a00, 0x50140f1a, 0x1502010d, 0xe13f3f00, 0x335d5e3f, 0x2f0132e1, + 0x2f335d5e, 0x1033115d, 0x2f3232e1, 0x5d2f395d, 0x30315d5d, 0x23112301, 0x37352311, 0x023e3435, 0x17163233, 0x23262607, 0x15020e22, 0x8b023315, + 0xc2c2b7f5, 0x4e7c552d, 0x2f27633b, 0x2828491f, 0xf513263a, 0x3ffcc103, 0x444bc103, 0x548d6b60, 0x8d0e1723, 0x3013110b, 0x00684153, 0x25000300, + 0xfc0314fe, 0x3f005e04, 0x5e005200, 0x1940a700, 0x5305320d, 0x2f123747, 0x37703760, 0x37033780, 0x272f372f, 0x47591d48, 0xc0ffb805, 0x0a074d40, + 0x01050548, 0xfd011d0a, 0x1db0011d, 0x011d8801, 0x1d301d20, 0x1d031d40, 0x601f601d, 0xdf60bf01, 0x60a00260, 0x40274001, 0x27480f0c, 0x0d320502, + 0x52563c04, 0x5c220a0a, 0x4e2c3c4f, 0x0701170f, 0x103c1717, 0x1b224f45, 0x000f0001, 0xe13fc13f, 0x5e2f393f, 0x1039c15d, 0x2f3911e1, 0x391712e1, + 0xc12b2f01, 0x11715d5d, 0x5d5d2f33, 0x33715d5d, 0xe12b2f39, 0x3911c110, 0x5d2f2f39, 0xe110c110, 0x31393912, 0x07150130, 0x14151616, 0x2223020e, + 0x030e2726, 0x021e1415, 0x1e323333, 0x0e141502, 0x2e222302, 0x3e343502, 0x26263702, 0x37363435, 0x3435032e, 0x3233023e, 0x14011716, 0x3233021e, + 0x2e343536, 0x22232302, 0x1413020e, 0x36323316, 0x23263435, 0xfc030622, 0x2f261cc5, 0x165d8c5f, 0x21110e2c, 0x2918111b, 0x5db01f38, 0x41245180, + 0x6b8bcd86, 0x27356aa0, 0x2a2f5742, 0x2b454036, 0x321b3147, 0x25619262, 0x40fe1b4f, 0x48613b1a, 0x3718b9ba, 0x23b0415a, 0x5c293f4c, 0x6764636c, + 0x6a636469, 0x1b714a04, 0x4c456d23, 0x01355e81, 0x20190a03, 0x211b1828, 0x502f0612, 0x8c583d6d, 0x502a3461, 0x5b3c4771, 0x130b2a42, 0x593d3552, + 0x513f122a, 0x8c593360, 0x090b3462, 0x402502fb, 0x6c731b2e, 0x0c213a2e, 0x034d2c10, 0x6f707360, 0x78747b77, 0xae000100, 0x12040000, 0x19001406, + 0x1d403200, 0x55194700, 0x601b101b, 0x031b801b, 0x470a0e0f, 0x101a540b, 0x10155004, 0x000b000c, 0x323f0015, 0x33e13f3f, 0xe1f61001, 0x105d3232, + 0x3031e1f6, 0x26341121, 0x020e2223, 0x11231115, 0x33071133, 0x3233033e, 0x03111516, 0x5170695c, 0xb61d436e, 0x190a08b6, 0x305c5245, 0xc302b9b7, + 0x66348282, 0xc7fd6094, 0x32fe1406, 0x2a3f2b90, 0xfdd2bf14, 0x02000033, 0x0000a000, 0xe5057501, 0x11000300, 0x14402500, 0x13201310, 0x47000c02, + 0x12540104, 0x0f0f5307, 0x15000f02, 0x333f3f00, 0x1001e52f, 0x32e132f6, 0x2130315d, 0x03331123, 0x32333634, 0x1415021e, 0x26222306, 0xb6b66401, + 0x162d3dc4, 0x3f111d27, 0x043d2d2c, 0x3c29014a, 0x2b1c0d36, 0x38393a1e, 0xbcff0200, 0x750114fe, 0x1300e505, 0x2e002100, 0x23101940, 0x1c022320, + 0x140c470f, 0x540c0303, 0x1f531722, 0x070f0d1f, 0x001b0050, 0x333fe13f, 0x1001e52f, 0x322f32e6, 0x5d32e110, 0x22133031, 0x16352726, 0x3e323316, + 0x33113502, 0x020e1411, 0x33363413, 0x15021e32, 0x22230614, 0x3f304226, 0x23361a17, 0x13232e1b, 0x6d4822b6, 0x162d3d13, 0x3f111d27, 0xfe3d2d2c, + 0x940b0e14, 0x270f0b0a, 0xf4043341, 0x7b4d18fb, 0x5f072f57, 0x1c0d363c, 0x393a1e2b, 0x01000038, 0x0000ae00, 0x1406f003, 0x5e000e00, 0x04070b40, + 0x03030204, 0x01054406, 0xc0ffb805, 0x0a071740, 0x05100548, 0x100f1005, 0x0702102f, 0x0a47090d, 0x000b0f54, 0xf8ffb800, 0x0f0c1040, 0x0c080748, + 0x0700480f, 0x150a0603, 0x3f000f03, 0x3912333f, 0x3f2b2b39, 0xe1f61001, 0x115d5e32, 0x2b382f33, 0x1133335d, 0x33113933, 0x37013031, 0x01013301, + 0x11070123, 0x11331123, 0x87560103, 0xfed32501, 0xd1ac016f, 0xb46db0fe, 0x370210b4, 0xfe6901aa, 0x0191fd25, 0x5afe52f8, 0x36fd1406, 0x0100edfe, + 0x0000ae00, 0x14066401, 0x1a000300, 0x05100e40, 0x00020520, 0x04540147, 0x15000002, 0x013f3f00, 0x5de1f610, 0x23213031, 0x64013311, 0x1406b6b6, + 0xae000100, 0x87060000, 0x2c005e04, 0x3f406500, 0xb9470a23, 0x0b96010b, 0x89020ba6, 0x0b67010b, 0x0b020b77, 0x4700160b, 0xf02e552c, 0x2ecf012e, + 0x502e2001, 0x2e0f022e, 0x15190801, 0x2d541647, 0x041a1a23, 0x1f28500f, 0x160f1710, 0x0015000b, 0x3f32323f, 0x32e1333f, 0x01331132, 0x32e1f610, + 0x5d5d5d5e, 0xe1f6105d, 0x5d2f3911, 0xe15d5d5d, 0x21303132, 0x23263411, 0x15020e22, 0x34112311, 0x0e222326, 0x23111502, 0x33173311, 0x3233033e, + 0x3e331716, 0x16323303, 0xd1051115, 0x66496964, 0x63b71e41, 0x3f684d69, 0x1a94b61b, 0x4f42180a, 0x9f782e59, 0x491a0826, 0xaf326057, 0x82c302b1, + 0x875b2f82, 0x02a2fd58, 0x348282c3, 0xfd609466, 0x944a04c7, 0x142a3f2b, 0x442f5e58, 0xd2bf162d, 0x010033fd, 0x0000ae00, 0x5e041204, 0x30001800, + 0x47001c40, 0x101a5518, 0x801a601a, 0x0a0e031a, 0x19540b47, 0x1450040f, 0x0b0f0c10, 0x3f001500, 0xe13f3f32, 0xf6100133, 0x105d32e1, 0x3031e1f6, + 0x26341121, 0x020e2223, 0x11231115, 0x3e331733, 0x16323303, 0x5c031115, 0x6e517069, 0x94b61d43, 0x45190a1a, 0xb7305c52, 0x82c302b9, 0x94663482, + 0x04c7fd60, 0x3f2b944a, 0xd2bf142a, 0x020033fd, 0xecff7100, 0x5e042d04, 0x1f001300, 0x1d403000, 0x5700481a, 0xd0214021, 0x0321e021, 0x0601210f, + 0x560a4814, 0x0f501d20, 0x05501710, 0xe13f0016, 0x1001e13f, 0x5d5ee1f6, 0xe1f6105d, 0x14013031, 0x2223020e, 0x3435022e, 0x3233023e, 0x1405021e, + 0x36323316, 0x23263435, 0x2d040622, 0x6fb27d43, 0x477fae67, 0x6fb37c43, 0x477fae67, 0x9a8900fd, 0x9a89879a, 0x2702879a, 0x4c91d589, 0x89d5914c, + 0x4b91d388, 0x88d3914b, 0xd1d3d3d1, 0x00cfcfd1, 0xae000200, 0x3f0414fe, 0x1f005e04, 0x36003000, 0x482e1e40, 0x1032571b, 0x10260132, 0x0d470c06, + 0x50203154, 0x0e101611, 0x2b1b0c0f, 0x16000550, 0xe1323f00, 0x333f3f3f, 0xf61001e1, 0x323232e1, 0xe1f6105d, 0x22053031, 0x2327022e, 0x16161716, + 0x11231115, 0x3e331733, 0x1e323303, 0x0e141502, 0x0e220302, 0x14150702, 0x3233021e, 0x26343536, 0x603b9e02, 0x0c173b4d, 0x04020303, 0x081a94b6, + 0x604d3a17, 0x6d9a5e3c, 0x9a6d3c3c, 0x41694c81, 0x411b021f, 0x7f87516c, 0x2b1a147f, 0x1f22203a, 0xfe10371a, 0x9436062b, 0x1b2d3d23, 0x8cd48f48, + 0x4990d58d, 0x5e2edb03, 0x65295f8c, 0xda376b9d, 0x00ced0cc, 0x71000200, 0x020414fe, 0x10005e04, 0x36003000, 0x202b1e40, 0x24472505, 0x32103255, + 0x16480e01, 0x1b243156, 0x0b200f22, 0x2c101b50, 0x16115000, 0x33e13f00, 0x3f33e13f, 0xf610013f, 0xf6105de1, 0x323232e1, 0x32253031, 0x3537023e, + 0x23022e34, 0x14150622, 0x2e221716, 0x3e343502, 0x1e323302, 0x37331702, 0x11231133, 0x36373634, 0x030e2337, 0x694c3502, 0x1b021f41, 0x87516c41, + 0x5d667f7f, 0x3c3c6e9a, 0x3b5d9a6e, 0x173c4c60, 0xb6931b08, 0x03030204, 0x4d3b160c, 0x5e2e8360, 0x65295e8d, 0xda376b9d, 0x97cdd1cc, 0x8cd48f48, + 0x4990d58d, 0x233d2d1b, 0x01caf994, 0x1b3a13d5, 0x3d222220, 0x01001a2e, 0x0000ae00, 0x5e040803, 0x41001600, 0x03b20b40, 0x900203c2, 0x0203a003, + 0xc0ffb803, 0x0b071b40, 0x18030348, 0x18501830, 0x11031880, 0x540e470d, 0x0d0f0f17, 0x00071215, 0xc13f0010, 0x013f3f33, 0x32e1f610, 0x2f33115d, + 0x315d5d2b, 0x16320130, 0x26260717, 0x020e2223, 0x11231115, 0x3e331733, 0x1d890203, 0x1c181a48, 0x683f1a3b, 0x94b6294b, 0x39190816, 0x5e045847, + 0x05a80505, 0x855f3307, 0x04b0fd51, 0x502bc94a, 0x0100253d, 0xecff5a00, 0x5e043f03, 0x48003500, 0x13252d40, 0xa0009047, 0x57000200, 0x5f373f37, + 0x03379f37, 0x2c013710, 0x1d9f0947, 0x1d021daf, 0x29263656, 0x052c1350, 0x0e091022, 0x00160550, 0x3f32e13f, 0xe1393912, 0xf6100132, 0x5de1325d, + 0x5df6105d, 0x303133e1, 0x020e1401, 0x27262223, 0x33031e35, 0x35023e32, 0x27022e34, 0x3435032e, 0x3233023e, 0x26071716, 0x06222326, 0x021e1415, + 0x03031e17, 0x9a6d3a3f, 0x3b9c6d60, 0x59544c1f, 0x395b412c, 0x5c35141a, 0x50734848, 0x8c64372b, 0x48a16156, 0x4789413f, 0x38176266, 0x7148465e, + 0x2d012a50, 0x28517850, 0x10a62223, 0x160f181f, 0x1f243b29, 0x1f323132, 0x614a3c1f, 0x4a6d4643, 0x93222a26, 0x3e432b1d, 0x2f2e3423, 0x4b3c1e1d, + 0x01000060, 0xecff2100, 0x46058f02, 0x50001d00, 0xb80519b1, 0x2f40c0ff, 0x05480b08, 0x1f3f1f05, 0x17021f4f, 0x1f14471b, 0x02122f12, 0x10100012, + 0xb0102010, 0xd010c010, 0x10060610, 0x164f121a, 0x000f1714, 0x00160b50, 0x333fe13f, 0x0132e133, 0xc65d5e2f, 0x32e1335d, 0x2f33115d, 0x3031332b, + 0x023e3225, 0x030e1537, 0x022e2223, 0x35231135, 0x15333737, 0x11211521, 0xfa011614, 0x232a2d12, 0x30280d09, 0x6a3e1934, 0x9b9b2c4d, 0x1401694e, + 0x813fecfe, 0x03080604, 0x090c068a, 0x854e2005, 0x517d0265, 0x89fce64e, 0x626183fd, 0xa4000100, 0x0804ecff, 0x1a004a04, 0x1c403000, 0x1a471701, + 0x1c101c55, 0x1c801c60, 0x0c470f03, 0x0d181b54, 0x0250120f, 0x15001607, 0x333f3f00, 0x01333fe1, 0x5de1f610, 0x32e1f610, 0x27213031, 0x23030e23, + 0x35022e22, 0x14113311, 0x3e323316, 0x33113502, 0x1b750311, 0x5245190a, 0x8a5b305c, 0x6ab62f5c, 0x436e516f, 0x2b93b61d, 0x2e14293f, 0x02699862, + 0x823dfdcd, 0x94653482, 0xfb3a0260, 0x010000b6, 0x00000000, 0x4a04d503, 0x6d001100, 0xff1100b9, 0x0a0f40f8, 0x0011480e, 0x480e0a08, 0x01090900, + 0xffb8100f, 0x1512b3c0, 0xffb81048, 0x071c40c0, 0x1010480b, 0xbf131010, 0xef13cf13, 0x13500313, 0x2f130f01, 0x03134f13, 0xb8010207, 0x0a40f0ff, + 0x01094701, 0x0f010f09, 0x3f001500, 0x5d39333f, 0xc1382f01, 0x5d5d5d5e, 0x382f3311, 0x12c12b2b, 0x332f3d39, 0x312b332b, 0x33012130, 0x17031e13, + 0x37033e33, 0x01013313, 0xbc89fe77, 0x1e1e0bc7, 0x05070419, 0x0b1e1e18, 0x89febcc7, 0x9dfd4a04, 0x606c6821, 0x6c601919, 0x63022168, 0x0100b6fb, + 0x00001400, 0x4a04e305, 0xc3002f00, 0xff2f00b9, 0x0a0c40f8, 0x002f480e, 0x480e0908, 0xb8202700, 0x1240f8ff, 0x20480e09, 0x0e09081f, 0x10091f48, + 0x480e0a08, 0xffb80f10, 0x090940f8, 0x180f480e, 0x27012754, 0x40e0ffb8, 0x480a0715, 0x1801185b, 0x480a0720, 0x18180927, 0x11032709, 0xffb82e2d, + 0x1512b3c0, 0xffb82e48, 0x071340c0, 0x102e480b, 0x20312e2e, 0x02313031, 0x0701310f, 0xffb81112, 0x111640f0, 0x09091f2d, 0x0f110901, 0x19061927, + 0x19761966, 0x10001903, 0x333f0015, 0x33115d33, 0x335d333f, 0x382f0133, 0x5d5d5e33, 0x382f3311, 0x12332b2b, 0x2f3d3917, 0x5d2b2f2f, 0x33115d2b, + 0x112b332b, 0x2b332b33, 0x332b3311, 0x2130312b, 0x27032e03, 0x06232726, 0x07060607, 0x33012303, 0x17031e13, 0x37033e33, 0x1e133313, 0x3e331703, + 0x33133703, 0xa8f00301, 0x0d0c0c04, 0x060f0e06, 0x190b0d0e, 0xfed3ac0b, 0x0a83bfe7, 0x040e1214, 0x15110506, 0xc4b30a16, 0x161709ac, 0x03060412, + 0x0b15120d, 0xe4feba89, 0x2d126802, 0x3a193432, 0x323a3f3e, 0x9cfd256a, 0xb8fd4a04, 0x5b67692d, 0x61571a1d, 0x6b02215f, 0x5c2295fd, 0x1a1d585f, + 0x2f6d6857, 0xb6fb4802, 0x23000100, 0xdb030000, 0x0b004a04, 0xa140e500, 0x86010989, 0x04060103, 0x0104f701, 0x360104e5, 0x05040104, 0x060106e8, + 0x0100e703, 0x02090900, 0x0102f801, 0x390102ea, 0x01020102, 0x057b056b, 0x01055702, 0x054a053a, 0x74016402, 0x01580201, 0x45013501, 0x01050201, + 0x05090109, 0x08060b03, 0x0108f701, 0x360108e5, 0x07080108, 0x48191640, 0x110e4007, 0x7b076b48, 0x07570207, 0x4a073a01, 0x0d070207, 0x0d300d10, + 0xb00d9002, 0x0d0f020d, 0x0ad90601, 0x010ac801, 0x09010aba, 0x3b0a010a, 0x020b4b0b, 0x05010b28, 0x020b150b, 0x0115070b, 0x3f3f000f, 0x5d5d2f01, + 0x5d5dc15d, 0x5d5e5d5d, 0xde10715d, 0x2b5d5d5d, 0x5d5dc12b, 0x1712715d, 0x182f3d39, 0x5d5d2f2f, 0x5d5d5d5d, 0x5d5dc110, 0x3311715d, 0x5d33335d, + 0x5d5dc110, 0x3031715d, 0x015d5d00, 0x13133301, 0x23010133, 0x01230101, 0xcf9ffe98, 0xfecffafa, 0xcf75019d, 0xf2fef4fe, 0x023302cf, 0x0166fe17, + 0xfde9fd9a, 0xfeb401cd, 0x0100004c, 0x14fe0a00, 0x4a04df03, 0x64002200, 0x081022b6, 0x0f0e0008, 0xb3c0ffb8, 0x0f481512, 0x40c0ffb8, 0x480b071d, + 0x0f0f100f, 0xcf24bf24, 0x0324ef24, 0x0f012450, 0x4f242f24, 0x18070324, 0xffb80001, 0x000c40f0, 0x23081022, 0x1b15501c, 0x000f000e, 0xe13f323f, + 0x33333311, 0xc1382f01, 0x5d5d5e33, 0x2f33115d, 0xc12b2b38, 0x2f3d3912, 0x30313333, 0x1e133313, 0x3e331703, 0x33133703, 0x23030e01, 0x35272622, + 0x32331616, 0x3737023e, 0x0ed7bd0a, 0x0412191d, 0x1b160506, 0xbcc70b1d, 0x411c4efe, 0x34507456, 0x40151b4c, 0x34463023, 0x04390f25, 0x289bfd4a, + 0x23525858, 0x5e615619, 0xfb630221, 0x5a815127, 0x91060b31, 0x2c170705, 0x00a02940, 0x52000100, 0x35030000, 0x09004a04, 0x0b406c00, 0x01039709, + 0x0d090803, 0xb8070348, 0x1140c0ff, 0x07480a07, 0x0b3f0b07, 0x0b7f0b5f, 0x01089803, 0xf8ffb808, 0x480d09b5, 0xb8020408, 0x12b7c0ff, 0x023f4815, + 0xb8070201, 0x1240f0ff, 0x07480c07, 0x0f054f04, 0x0c071002, 0x4f080248, 0x3f001501, 0x3f2b32e1, 0x012b32e1, 0x332b5d2f, 0x5d5d2b33, 0x2b2f3311, + 0x335d2b33, 0x21213031, 0x35210135, 0x21011521, 0x1dfd3503, 0x09fe1802, 0xf4fdb002, 0x037d1e02, 0xfc928944, 0x010000d1, 0xbcfe3d00, 0xb605a202, + 0x40002700, 0x051a2540, 0x2720f705, 0x0f1323f1, 0x010c10f6, 0xf50f230c, 0x0f0110d9, 0x02105f10, 0x1a291010, 0x05f819f5, 0x00f906f5, 0xe13fe13f, + 0x5d2f3912, 0x0139e15d, 0x33e65d2f, 0xe232f133, 0x3031322f, 0x021e1405, 0x032e1517, 0x26341135, 0x36323523, 0x3e341135, 0x0e153702, 0x14111503, + 0x16150706, 0xf4011516, 0x28412d18, 0x365f834d, 0x837d7d83, 0x4d835f36, 0x182d4128, 0x77737377, 0x233d3010, 0x0196010d, 0x4e6e4721, 0x56674e01, + 0x0167569b, 0x476e4e4d, 0x01950121, 0x303d230d, 0x7b69b4fe, 0x7a140c14, 0x0100006a, 0x14fee901, 0x14067f02, 0x2d000300, 0x05001f40, 0x40053001, + 0x80057005, 0xaa020405, 0x03100300, 0x03800340, 0x070503c0, 0x001b0203, 0x3f3f0000, 0x5d5e2f01, 0x31715de1, 0x11330130, 0x96e90123, 0xf8140696, + 0x01000000, 0xbcfe3300, 0xb6059802, 0x40002900, 0x240d2540, 0x0007f724, 0x14f61af1, 0x011d9003, 0xf51a041d, 0x19ff19ef, 0x0119d902, 0x240e1919, + 0x0df923f5, 0x00f80ef5, 0xe13fe13f, 0x5d2f3911, 0x0139e15d, 0x33335d2f, 0xe232f1e6, 0x3031322f, 0x37363413, 0x35262635, 0x022e3411, 0x031e3527, + 0x1e141115, 0x22153302, 0x14111506, 0x3507020e, 0xe135033e, 0x77737377, 0x28412d18, 0x365f834d, 0x3e604121, 0x5f36837d, 0x41284d83, 0x3b01182d, + 0x0c147a6a, 0x01697b14, 0x233d304c, 0x0195010d, 0x4e6e4721, 0x4834b3fe, 0x569b142d, 0x4eb2fe67, 0x0121476e, 0x230d0196, 0x0100303d, 0x4a026600, + 0x5a030204, 0x3c002300, 0x251d0d40, 0x10012510, 0x170a010a, 0xb81f0aad, 0x1640c0ff, 0x1f481310, 0x1cad051f, 0x0d1f0d0f, 0x0d4f0d3f, 0x0d8f0d6f, + 0x2f000d06, 0xc8f1335d, 0xe1322b2f, 0x5d5d2f01, 0x3031ce10, 0x23032e01, 0x07020e22, 0x32333635, 0x1e17021e, 0x3e323303, 0x06153702, 0x022e2223, + 0x37251202, 0x1c16292d, 0x19383b3c, 0x321d9464, 0x252f4337, 0x16282f37, 0x383b3c1c, 0x1d956318, 0x02433732, 0x0d16108b, 0x2c211305, 0x056ca219, + 0x1014190d, 0x13050d16, 0xa2192c21, 0x190d056c, 0x93000200, 0x91018bfe, 0x03005e04, 0x41001700, 0xff0000b9, 0x0a1340f0, 0x19304814, 0x19b019a0, + 0x020419c0, 0x030e9a04, 0xffb80e03, 0x070f40c0, 0x000e480a, 0x00139b09, 0x02021002, 0x2f000207, 0xf52f5d5e, 0x2b2f01ce, 0xe1102f33, 0x30315d32, + 0x1333132b, 0x0e141323, 0x2e222302, 0x3e343502, 0x1e323302, 0x3379d502, 0x2313efdf, 0x2e1a1b2e, 0x23141423, 0x2e1b1a2e, 0xa4021323, 0x4805e7fb, + 0x0f213526, 0x2635210f, 0x10223425, 0x00342210, 0xbc000100, 0xba03ecff, 0x2500cb05, 0x25405a00, 0x0f460312, 0x250a0404, 0x27402715, 0x00481e01, + 0x400a300a, 0x040ad00a, 0x731b0a06, 0x210f120f, 0x05020573, 0xc0ffb80f, 0x120f0c40, 0x0f050f48, 0x07100305, 0x3f001903, 0x3939123f, 0x112b2f2f, + 0x11e11033, 0x01e11033, 0xe15d5e2f, 0x32c6105d, 0x332f3911, 0x303132e1, 0x15070624, 0x032e3523, 0x023e3435, 0x15333537, 0x07171616, 0x2223032e, + 0x16141506, 0x37363233, 0x6e760315, 0x8a57894c, 0x61353462, 0x4889568b, 0x17352e88, 0x193b3c38, 0x9491909d, 0xd4368351, 0xcec8021e, 0xc7854b0d, + 0x88cb8d89, 0xa4ac0d4b, 0x9a172103, 0x090f130a, 0xc3d2d4ca, 0x00a11825, 0x44000100, 0x23040000, 0x2800c905, 0x11407500, 0x236f110d, 0x0f1f0f0f, + 0x0f1f0702, 0x03190f1f, 0xc0ffb817, 0x480e0ab3, 0xc8ffb817, 0x09063040, 0x2a171748, 0x21012a10, 0x0e0b4019, 0x29191948, 0x0d752110, 0x227f222f, + 0x22af228f, 0x22df22bf, 0x220722ff, 0x74160022, 0x73071819, 0x3f000700, 0x11e13fe1, 0x335d2f39, 0x011132e1, 0x332b2f33, 0x2f33115d, 0x12332b2b, + 0x2f2f3939, 0xe1335d5e, 0x01303132, 0x07171632, 0x22232626, 0x1115020e, 0x15211521, 0x07020e14, 0x35211521, 0x3535033e, 0x11333523, 0x02023e34, + 0x42ae6a9a, 0x4b8d3842, 0x213c5230, 0x88fe7801, 0x1b332717, 0x21fcec02, 0x1e35492c, 0x693bc6c6, 0x2dc90592, 0x2b1d9023, 0x425e3b1b, 0xd389d9fe, + 0x2b40593e, 0x0b9aa610, 0x43614429, 0x440189d5, 0x325f8957, 0x7b000200, 0xec031d01, 0x23008b04, 0x86003700, 0x8f0e2340, 0x16160116, 0x0f15ab2e, + 0x1e06180c, 0x00080321, 0x12011270, 0x39103912, 0x24200401, 0x010080aa, 0xc0ffb800, 0x0a063140, 0x38000048, 0x011f8017, 0x18060c1f, 0x04060f1e, + 0x00ae2909, 0x0d1b011b, 0xcfae3305, 0x0209ef09, 0x09a00990, 0x1f0309b0, 0x6f093f09, 0x00090309, 0x5d5d5d2f, 0x2f32c6e1, 0x1712e15d, 0xc6331139, + 0x0111325d, 0x5d2b2f33, 0x5d32c6e1, 0x5d2f3311, 0xf1391712, 0x325d2fc0, 0x34133031, 0x37273736, 0x33363617, 0x37171632, 0x16160717, 0x07061415, + 0x06270717, 0x26222306, 0x37270727, 0x14372626, 0x3233021e, 0x3435023e, 0x2223022e, 0x23ba020e, 0x7f62811f, 0x3c3c6c2f, 0x63812e6b, 0x23251f82, + 0x81607f21, 0x3c3c6b2e, 0x607f2d6e, 0x8a231f7f, 0x32574125, 0x26425933, 0x33594226, 0x25415732, 0x6b3bd302, 0x8162812f, 0x20242420, 0x2e816081, + 0x6e3c3c6d, 0x7f607f2d, 0x2024231f, 0x2d7f607f, 0x57323c6c, 0x41262641, 0x59333257, 0x42262642, 0x01000059, 0x00001d00, 0xb6054c04, 0x95001600, + 0x18305b40, 0x0c130f01, 0x0115aa16, 0x040c1515, 0x02ab0108, 0x01021f01, 0x00980202, 0x03070001, 0x14995a0b, 0x0f101401, 0x0c8a010c, 0x100c0001, + 0x030c400c, 0x0e0a0c07, 0x0f070f60, 0x13601206, 0x7f130003, 0x020f8f0f, 0xd0011330, 0x130f0113, 0x0f02131f, 0x01130f13, 0x0115120b, 0x333f0003, + 0x3939123f, 0x5d5d2f2f, 0x33115d71, 0x32e11033, 0xe1103311, 0x5e2f0132, 0x33715d5d, 0x32e15d33, 0x325d3932, 0x335d5d2f, 0x33113939, 0x12335d2f, + 0x315d3939, 0x33010130, 0x21153301, 0x21152115, 0x21112311, 0x21352135, 0x33013335, 0x5a013302, 0xef7ffebf, 0x2d01d3fe, 0xfeb2d3fe, 0xfe2d01d3, + 0x85feead3, 0x02d302c0, 0x8900fde3, 0xfafe899e, 0x9e890601, 0x00000389, 0xe9010200, 0x7f0214fe, 0x03001406, 0x39000700, 0x09002540, 0x40093001, + 0x80097009, 0x06020409, 0x070003aa, 0x07400710, 0x07c00780, 0x04070705, 0x06030403, 0x0000001b, 0x39393f3f, 0x2f012f2f, 0xe1335d5e, 0x31715d32, + 0x11330130, 0x11331123, 0x96e90123, 0x06969696, 0xfef4fc14, 0x00f3fc19, 0x79000200, 0x5e03f6ff, 0x45001f06, 0x79005a00, 0x56054b40, 0x21465108, + 0x3c464c26, 0x10292946, 0xa00121ff, 0x0221b021, 0x5cc05c21, 0x015cbf01, 0x17004646, 0x31080846, 0x00010000, 0x30001000, 0xd000c000, 0x0600f000, + 0x265b0008, 0x1756054c, 0x04294c56, 0x2e50370d, 0x0d501415, 0xe13f0001, 0x1712e13f, 0x11331139, 0xc6100133, 0x32715d5e, 0x10e12f32, 0x105d5de1, + 0x325d5dce, 0x12e12f32, 0xe1103939, 0x31393911, 0x3e341330, 0x26263702, 0x023e3435, 0x17163233, 0x23262607, 0x14150622, 0x1e17021e, 0x0e141503, + 0x16160702, 0x020e1415, 0x27262223, 0x33031e35, 0x35023e32, 0x27022e34, 0x1437032e, 0x1717021e, 0x3435033e, 0x0e27022e, 0x2d1a8903, 0x554b1f3a, + 0x568c6437, 0x38489d61, 0x63478c41, 0x5f391866, 0x4e714846, 0x3429182a, 0x3b4c451c, 0x6c609b6c, 0x4c1f3b9c, 0x452b5954, 0x1117375d, 0x494c5e33, + 0x9a295073, 0x48653f1c, 0x21291423, 0x6c411a15, 0x262f1952, 0x33290317, 0x0f2d4053, 0x3d547226, 0x28254462, 0x271c8b20, 0x2e1b393b, 0x1c1d2f2c, + 0x3e614e41, 0x31445534, 0x4e6d2610, 0x284d6f47, 0x0f9e2021, 0x180e171e, 0x1d1b3327, 0x1f312d2d, 0x644e3e1f, 0x3a3f2559, 0x0d0f1e37, 0x22382e24, + 0x393b4026, 0x2d1f081e, 0x0200003a, 0x0c053301, 0xd9056a03, 0x19000b00, 0x21403500, 0x14af860c, 0x06c01401, 0x10000086, 0x50004000, 0x00060400, + 0x1791030f, 0x09cf099f, 0x01093002, 0x5d2f0009, 0x32e5335d, 0x5d5e2f01, 0x5ddc1ae1, 0x013031e1, 0x32333634, 0x06141516, 0x25262223, 0x32333634, + 0x1415021e, 0x26222306, 0x28383301, 0x273a3a27, 0x77013828, 0x23132838, 0x263a101a, 0x73053828, 0x35313036, 0x35323235, 0x190c3036, 0x32351b26, + 0x03000032, 0xecff6400, 0xcb054406, 0x41002500, 0x6a005500, 0xc5054340, 0x220f0f1a, 0x221a221a, 0x00c34c26, 0x34c00134, 0x42573401, 0xc90a26c3, + 0x1fc90015, 0x151f150f, 0x157f152f, 0x159f158f, 0x1f000806, 0x1f601f10, 0x1f801f70, 0x151f1505, 0xc8472d1f, 0x2dc8513b, 0xe13f0004, 0x3911e12f, + 0x5d2f2f39, 0xe1105d5e, 0x2f01e110, 0x5dde10e1, 0x3911e171, 0x112f2f39, 0xe1102f33, 0x22013031, 0x1415020e, 0x3233021e, 0x1537023e, 0x2223030e, + 0x3435022e, 0x3233023e, 0x26071716, 0x3e340126, 0x1e323304, 0x0e141504, 0x2e222304, 0x1e143704, 0x3e323302, 0x2e343502, 0x0e222302, 0x3d7b0302, + 0x1d21405e, 0x17435f3d, 0x19383936, 0x3c353118, 0x65986623, 0x99693633, 0x3b843f64, 0xfc61343e, 0x8a6136be, 0x6868c0a7, 0x618aa7c0, 0x8a613636, + 0x6868c0a7, 0x618aa7c0, 0xaf656d36, 0xea8585ea, 0xaf6565af, 0xea8585ea, 0x1d0465af, 0x4b78532c, 0x2b52784e, 0x09110c07, 0x0e120b83, 0xaa7a4207, + 0x78a76567, 0x7f1d2143, 0xbefe1c1a, 0x8aa7c068, 0x61363661, 0x68c0a78a, 0x89a7c068, 0x62353562, 0x68c0a789, 0x65afea85, 0x85eaaf65, 0x65afea85, + 0x00eaaf65, 0x44000200, 0x42021003, 0x1e00c705, 0x4e002d00, 0x012d2f40, 0x1d00e00f, 0x1d021d10, 0x1f2f0f2f, 0x7f2f4f2f, 0x052faf2f, 0x170be024, + 0x010b6017, 0xe42d2e0b, 0x011a0f0f, 0x0600e427, 0x1ae413c0, 0xe13f00de, 0xe1c4dc1a, 0x2f391139, 0xc61001e1, 0x102f325d, 0xd6105de1, 0x3232e15d, + 0x27013031, 0x2223030e, 0x3435022e, 0x35373736, 0x22232634, 0x36270706, 0x16323336, 0x0e031115, 0x16141503, 0x023e3233, 0xe7013535, 0x2f27121c, + 0x482b2338, 0x8f8d1d34, 0x30383d63, 0x33302a5a, 0x777d3c75, 0x294433c9, 0x222a3212, 0x03192b3a, 0x2316521d, 0x331a0d19, 0x6c66334d, 0x481f0405, + 0x64161d39, 0x7a6a241a, 0x39013afe, 0x2b1e1203, 0x152d331d, 0x312c412c, 0x52000200, 0x93037300, 0x0600c703, 0x60000d00, 0x04021140, 0x500aeb0d, + 0x02046004, 0x0a040a04, 0xb8090b06, 0x2140c0ff, 0x09480c09, 0x9f0f0f0f, 0x030faf0f, 0x039feb06, 0x00060301, 0x0a070d03, 0x0303050a, 0x08050c01, + 0x332f0001, 0x3912332f, 0x39122f3d, 0x11333311, 0x18013333, 0x5de15d2f, 0x322bc610, 0x2f393911, 0xe1105d2f, 0x30313311, 0x03170113, 0x25010713, + 0x13031701, 0x01520107, 0xeeee7535, 0x01cbfe75, 0x74360197, 0xfe74eded, 0x012902ca, 0xa4fe4e9e, 0x014ea4fe, 0x9e011b9b, 0xfea4fe4e, 0x9b014ea4, + 0x66000100, 0x02040601, 0x05001d03, 0x24403900, 0x0701aa02, 0x96010710, 0x048b0104, 0x01047901, 0x4b010456, 0x04380104, 0x01041201, 0x04010409, + 0xb305ad04, 0x01e13f00, 0x5d5d5d2f, 0x5d5d5d5d, 0xde105d5d, 0x013031e1, 0x21112311, 0x95020435, 0x1d03f9fc, 0x8101e9fd, 0xffff0096, 0xd1015200, + 0x79024202, 0x10000602, 0x04000000, 0xecff6400, 0xcb054406, 0x1e000800, 0x4e003a00, 0x7d40c200, 0x16b416a4, 0xb40316c4, 0x0217c417, 0x52011617, + 0x0e170e15, 0x0e15c516, 0x0e151514, 0x1900090e, 0xc5091ac5, 0x00041504, 0x1a00011a, 0x1ad01ac0, 0x048f0703, 0x1a041a01, 0xc3451f04, 0xc0012d00, + 0x502d012d, 0x0e1fc33b, 0x0000c918, 0x15161b16, 0x1bc9081a, 0x0f011a00, 0x2f1a1f1a, 0x8f1a7f1a, 0x061a9f1a, 0x101b0008, 0x701b601b, 0x051b801b, + 0x1b1a1b1a, 0x34c84026, 0x26c84a13, 0xe13f0004, 0x3911e13f, 0x5d2f2f39, 0x10715d5e, 0x333311e1, 0x10391211, 0x2f0132e1, 0x5dde10e1, 0x3911e171, + 0x5d2f2f39, 0x11715d5e, 0x10e11039, 0x391132e1, 0x102b1087, 0x0587c100, 0x01c4102b, 0x5d30315d, 0x36323301, 0x23263435, 0x0e140523, 0x17160702, + 0x1717021e, 0x11230323, 0x32331123, 0x3e340116, 0x1e323304, 0x0e141504, 0x2e222304, 0x1e143704, 0x3e323302, 0x2e343502, 0x0e222302, 0x48e70202, + 0x59534f5b, 0x1b920146, 0x431f392d, 0x212a1735, 0xceb30a0a, 0xa8e99d5f, 0x36ebfb9e, 0xc0a78a61, 0xa7c06868, 0x3636618a, 0xc0a78a61, 0xa7c06868, + 0x6d36618a, 0x85eaaf65, 0x65afea85, 0x85eaaf65, 0x65afea85, 0x45480003, 0x30813b4a, 0x0d28394b, 0x4725576e, 0x01111138, 0x03a0fe60, 0xc3fe827d, + 0x8aa7c068, 0x61363661, 0x68c0a78a, 0x89a7c068, 0x62353562, 0x68c0a789, 0x65afea85, 0x85eaaf65, 0x65afea85, 0x00eaaf65, 0xfaff0100, 0x06041406, + 0x0300a006, 0x00b61200, 0x01010500, 0x2f0002ba, 0x112f01e1, 0x30312f33, 0x21352101, 0xf4fb0604, 0x14060c04, 0x0200008c, 0x56037b00, 0xcb05f202, + 0x27001300, 0x2c404300, 0x290aab1e, 0x1401299f, 0x400030aa, 0x19000200, 0x200f10ae, 0x0fe0020f, 0x6f020ff0, 0x0f00010f, 0x0f200f10, 0x0f0f0603, + 0x0405ae23, 0x33e13f00, 0x5d5d5e2f, 0x01e1715d, 0x5de15d2f, 0x31e1d610, 0x3e341330, 0x1e323302, 0x0e141502, 0x2e222302, 0x1e143702, 0x3e323302, + 0x2e343502, 0x0e222302, 0x55327b02, 0x73414173, 0x56323256, 0x73414173, 0x1e7b3255, 0x28284634, 0x1e1e3546, 0x28284635, 0x041e3446, 0x5673418f, + 0x73563232, 0x55724141, 0x72553131, 0x34452741, 0x45341e1e, 0x35472827, 0x47351f1f, 0x66000200, 0x02040000, 0x0b00a204, 0x3a000f00, 0x11102140, + 0x08080f01, 0x0caa0906, 0xef030101, 0x00200100, 0x00a00060, 0xad0d0003, 0xad00090c, 0x00b30306, 0x32e1333f, 0x2f01e12f, 0x33335d5d, 0x32e13311, + 0x5d331132, 0x21013031, 0x33112135, 0x21152111, 0x35012311, 0xe9011521, 0x83017dfe, 0xfe830196, 0x7dfe967d, 0x87029c03, 0xfe850196, 0x7ffe967b, + 0x9696fafe, 0x31000100, 0x6d024a02, 0x1e00c905, 0x15404000, 0x1700e108, 0x7f204f20, 0x40200220, 0x1d480a06, 0x0f0f01e1, 0xc0ffb801, 0x18150e40, + 0x1d080148, 0xde12e50b, 0xdd01e51d, 0x3fe13f00, 0x013912e1, 0x2f332b2f, 0x5d2be110, 0xe132de10, 0x21013031, 0x033e3735, 0x23263435, 0x27070622, + 0x32333636, 0x1415021e, 0x0707020e, 0xfd6d0221, 0x4839d1c4, 0x36420f28, 0x4e2d5d33, 0x3c528536, 0x1d254461, 0x94334f36, 0x4a028c01, 0x553ee470, + 0x41223b43, 0x5e263240, 0x3f214130, 0x5632395b, 0x9d375b55, 0x1f000100, 0x68023902, 0x3000c905, 0x3c406100, 0x19190003, 0xe11e060e, 0xe1150000, + 0x325f3206, 0x3202328f, 0x480a0640, 0x400e2727, 0x0e482019, 0x0fe41903, 0x2f1a1f1a, 0xdf1a5f1a, 0x1a08051a, 0x2326121a, 0x12de2ce5, 0xdf0b0fe5, + 0xe1333f00, 0x1233e13f, 0x5d5e2f39, 0x2f0139e1, 0x2b2f332b, 0xe1de105d, 0x11e12f33, 0x122f3912, 0x01303139, 0x16070614, 0x0e141516, 0x26222302, + 0x16163527, 0x35363233, 0x23232634, 0x36323335, 0x022e3435, 0x07062223, 0x33033e27, 0x02021e32, 0x5845514e, 0x7e532858, 0x397b4656, 0x6235843f, + 0x62606b58, 0x14545c62, 0x3b1b2f23, 0x1d453361, 0x2c4c443d, 0x23466945, 0x6a4ee704, 0x4e6a1718, 0x2847643c, 0x22851f19, 0x4a495326, 0x404f7143, + 0x0e1e2f20, 0x17602529, 0x220f1a25, 0x0100533c, 0xd9048901, 0x21061203, 0x16000d00, 0x00060a40, 0x0c0f8005, 0x0c020c5f, 0x1a5d2f00, 0xcd2f01cd, + 0x3e013031, 0x15333703, 0x2307030e, 0x2f168901, 0xdb102a2f, 0x514d3f10, 0xf4047923, 0x51514c1d, 0x511b1522, 0x001c5358, 0xae000100, 0x120414fe, + 0x1d004a04, 0x22403700, 0x0a47090d, 0x1f101f55, 0x1f601f20, 0x1f801f70, 0x471d1405, 0x1a1e541c, 0x1150031b, 0x1c150b16, 0x3f000f09, 0xe13f3f33, + 0xf610013f, 0x105d32e1, 0x3132e1f6, 0x16140130, 0x023e3233, 0x11331135, 0x06232723, 0x26222306, 0x16171627, 0x23111516, 0x64013311, 0x6e526f6a, + 0x93b61c43, 0x90300a1b, 0x236a4867, 0x01020201, 0x8701b6b6, 0x65348282, 0x3a026094, 0x5393b6fb, 0x262a2e54, 0x2a552328, 0x3606c0fe, 0x71000100, + 0x6604fcfe, 0x13001406, 0x21403700, 0x05009904, 0x05400530, 0x06040550, 0x010d0505, 0x10150099, 0x0d000115, 0x0d020d10, 0x05001203, 0x322f0000, + 0x2f01c13f, 0xd6105d5d, 0x2f3912e1, 0x31e15d5e, 0x11230130, 0x11231123, 0x2e222306, 0x3e343502, 0x04213302, 0x79cf7866, 0x9b5f553d, 0x77413c6d, + 0x330264a6, 0x7906fcfe, 0x330387f9, 0xc0763312, 0x78c5938c, 0x01000032, 0x48029300, 0x5e039101, 0x33001300, 0x15101a40, 0x15901580, 0x0a0415a0, + 0xd000c096, 0x00340200, 0x00640044, 0x00040074, 0xb6c0ffb8, 0x00480a07, 0x000f9b05, 0x2f01e52f, 0xed5d5d2b, 0x1330315d, 0x33023e34, 0x15021e32, + 0x23020e14, 0x93022e22, 0x1b2e2214, 0x14222f1a, 0x1a2f2214, 0x14222e1b, 0x3526d302, 0x210f0f21, 0x34252635, 0x22101022, 0x01000034, 0x14fe2300, + 0x00009801, 0x39001900, 0x13141f40, 0x127f1513, 0x1202128f, 0x840d0612, 0x1a061b00, 0x40158c12, 0x15480e09, 0x8d0a1315, 0xe12f0003, 0x2b2f392f, + 0xc61001e1, 0x11e1d610, 0x335d2f39, 0x31331133, 0x06140130, 0x27262223, 0x33161635, 0x34353632, 0x3727022e, 0x031e0733, 0x968d9801, 0x0f0f2d16, + 0x50471031, 0x253f2e1a, 0x2239795a, 0xfe192b3a, 0x066c61e1, 0x03036c03, 0x2318312b, 0xb009131a, 0x291a0873, 0x0100003a, 0x4a023f00, 0xb605ba01, + 0x34000e00, 0x104f2140, 0x1002107f, 0x480a0640, 0xe1020e0e, 0x8f037f00, 0x03200203, 0x03020330, 0x090ddd02, 0x00dc00e5, 0x3fcde13f, 0x5d5d2f01, + 0x2f33e133, 0x30315d2b, 0x23113301, 0x023e3411, 0x07060637, 0x33012707, 0x03019187, 0x260e0103, 0x054a5e16, 0x0294fcb6, 0x3c3c1904, 0x28111638, + 0x00604911, 0x42000200, 0x8b021003, 0x1300c705, 0x2e001f00, 0x00e01ab2, 0x40c0ffb8, 0x480f0914, 0x210f2100, 0x0ae01401, 0x05e41720, 0x0fe41dc0, + 0xe13f00de, 0x01e1dc1a, 0x5de1d610, 0xe12bd610, 0x14013031, 0x2223020e, 0x3435022e, 0x3233023e, 0x1405021e, 0x36323316, 0x23263435, 0x8b020622, + 0x446d4d29, 0x2b4e6a3f, 0x446d4c29, 0x2c4e6b3e, 0x564b3afe, 0x554b4b55, 0x6d044b56, 0x2f598253, 0x5382592f, 0x2e588153, 0x5381582e, 0x77797977, + 0x00737378, 0x54000200, 0x96037300, 0x0600c703, 0x56000d00, 0x0f0f2f40, 0x0faf0f9f, 0x04eb0703, 0x0a020a02, 0x090b0302, 0x9feb000e, 0x03100103, + 0x03400320, 0x070d0303, 0x0300060a, 0x030a030a, 0x08050c01, 0x332f0001, 0x3912332f, 0x2f2f3d39, 0x11333311, 0x18013333, 0xe15d5d2f, 0x1132c610, + 0x2f2f3939, 0x5de13311, 0x01013031, 0x37031327, 0x27010501, 0x01370313, 0xcafe9603, 0x74eded74, 0x68fe3601, 0xee75cbfe, 0x350175ee, 0x65fe0e02, + 0x015c014e, 0x62fe4e5c, 0x4e65fe1b, 0x5c015c01, 0x0062fe4e, 0x3f00ffff, 0x8b050000, 0x2600b605, 0x00007b00, 0x17022700, 0x00004a02, 0x3c020701, + 0xb7fdfc02, 0x1d403000, 0x18160203, 0x16bf0203, 0x01168f01, 0x1601163f, 0x01114001, 0x11011100, 0x01004000, 0x355d1100, 0x355d5d11, 0x5d5d5d11, + 0x3f003535, 0xffff3535, 0x00002c00, 0xb605a005, 0x7b002600, 0x270000ed, 0x35021702, 0x07010000, 0x33037400, 0x2800b7fd, 0x14021840, 0x14000218, + 0xb0011401, 0x11400111, 0x70001101, 0x00400100, 0x5d110001, 0x5d11355d, 0x5d11355d, 0x353f0035, 0x1f00ffff, 0xce050000, 0x2600c905, 0x00007500, + 0x17022700, 0x0000a802, 0x3c020701, 0xb7fd3f03, 0x27403c00, 0x18380203, 0x38700203, 0x01385001, 0x33b40138, 0x0133a401, 0x64013384, 0x33500133, + 0x01333001, 0x33013320, 0x5d014c0f, 0x5d5d5d11, 0x5d5d5d5d, 0x5d5d1135, 0x3f003535, 0x02003535, 0x77fe4400, 0x5e044403, 0x3b002700, 0x12404400, + 0x28289a32, 0x00004627, 0x0f3d140b, 0x0b08013d, 0xffb81c46, 0x0f1040c0, 0x0b1c481b, 0x2d272717, 0x1310379b, 0x00175110, 0x3f33e12f, 0x122f32e5, + 0x2b2f0139, 0x105d5ee1, 0x2f3911ce, 0xe12f33e1, 0x15013031, 0x07020e14, 0x1415030e, 0x3233021e, 0x06173736, 0x2e222306, 0x3e343502, 0x033e3702, + 0x14133535, 0x2223020e, 0x3435022e, 0x3233023e, 0x5002021e, 0x32412710, 0x152b4430, 0x3755391e, 0x40459654, 0x5d61bc52, 0x1b386795, 0x34365135, + 0xba0e2642, 0x1b2e2313, 0x14232e1a, 0x1a2e2314, 0x13232e1b, 0x3a25a402, 0x2a4c515b, 0x4f454329, 0x394f3035, 0x9223331f, 0x60333a2a, 0x6844588a, + 0x2d2f545a, 0x2b433e43, 0x262f0113, 0x0f0f2135, 0x25263521, 0x10102234, 0xffff3422, 0x00000000, 0x7307dd04, 0x24002602, 0x07010000, 0xbdff4300, + 0x15005201, 0x051502b4, 0xffb80226, 0x151bb49c, 0x01250704, 0x2b00352b, 0xffff0035, 0x00000000, 0x7307dd04, 0x24002602, 0x07010000, 0x8d007600, + 0x13005201, 0x21020b40, 0x6c022605, 0x07041b15, 0x352b0125, 0x00352b00, 0x0000ffff, 0xdd040000, 0x26027307, 0x00002400, 0x4b010701, 0x52011f00, + 0x02b41500, 0x02260515, 0xb4ffffb8, 0x0704151d, 0x352b0125, 0x00352b00, 0x0000ffff, 0xdd040000, 0x26023507, 0x00002400, 0x52010701, 0x52010600, + 0x0b401300, 0x26051d02, 0x2c1e0102, 0x01250704, 0x2b00352b, 0xffff0035, 0x00000000, 0x2b07dd04, 0x24002602, 0x07010000, 0x21006a00, 0x17005201, + 0x02030d40, 0x0326051e, 0x29150102, 0x01250704, 0x0035352b, 0x0035352b, 0x0000ffff, 0xdd040000, 0x26020407, 0x00002400, 0x50010601, 0x31007d1f, + 0x02032040, 0xdf011aef, 0x1a50011a, 0x011a4001, 0x10011a20, 0x1a00011a, 0x02031a01, 0x04151f00, 0x2b012507, 0x11003535, 0x5d5d5d5d, 0x355d5d5d, + 0x02000035, 0x0000feff, 0xb6055606, 0x13000f00, 0x2a408400, 0x0e0a1306, 0x0111015a, 0x13040310, 0x240113a9, 0x54133413, 0x01100313, 0x010c1401, + 0x0c0c0113, 0x05031301, 0x15670008, 0xffb80504, 0x052040f0, 0x065f1309, 0x0d105f03, 0x0a4f0a5f, 0xaf0a0f01, 0x1008020a, 0x060a100a, 0x5f0e0403, + 0x00120105, 0x2fe1333f, 0x2f39393f, 0x715d5e2f, 0xe110e110, 0x0132e110, 0x1033382f, 0x171132e6, 0x2f2f2f39, 0x5d5d5d5d, 0xc4c4877d, 0x10330111, + 0x331132e1, 0x21213031, 0x23032111, 0x21152101, 0x21152111, 0x21012111, 0x56062311, 0x25fe08fd, 0x8f02bacb, 0xc3fdc903, 0xeafd1602, 0x75fb3d02, + 0x016c9301, 0x053bfec5, 0x3cfea4b6, 0x01f8fda2, 0x00a802c6, 0x7d00ffff, 0x980414fe, 0x2602cb05, 0x00002600, 0x7a000701, 0x0000fc01, 0x01b60b00, + 0x18242a4f, 0x2b012520, 0xffff0035, 0x0000c700, 0x7307be03, 0x28002602, 0x07010000, 0xb7ff4300, 0x15005201, 0x050c01b4, 0xffb80126, 0x0c12b4c2, + 0x01250001, 0x2b00352b, 0xffff0035, 0x0000c700, 0x7307be03, 0x28002602, 0x07010000, 0x3f007600, 0x13005201, 0x18010b40, 0x4a012605, 0x0001120c, + 0x352b0125, 0x00352b00, 0xc700ffff, 0xbe030000, 0x26027307, 0x00002800, 0x4b010701, 0x5201f1ff, 0x01b41500, 0x0126050c, 0xb4fdffb8, 0x00010c14, + 0x352b0125, 0x00352b00, 0xc700ffff, 0xbe030000, 0x26022b07, 0x00002800, 0x6a000701, 0x5201f5ff, 0x0d401700, 0x05150102, 0x01010226, 0x0001200c, + 0x352b0125, 0x352b0035, 0xffff0035, 0x00003e00, 0x73076402, 0x2c002602, 0x07010000, 0xb5fe4300, 0x15005201, 0x050c01b4, 0xffb80126, 0x0c12b4a8, + 0x01250001, 0x2b00352b, 0xffff0035, 0x00005200, 0x73078a02, 0x2c002602, 0x07010000, 0x78ff7600, 0x13005201, 0x18010b40, 0x6a012605, 0x0001120c, + 0x352b0125, 0x00352b00, 0x1100ffff, 0xa9020000, 0x26027307, 0x00002c00, 0x4b010701, 0x52010fff, 0x0b401300, 0x26050c01, 0x0c140201, 0x01250001, + 0x2b00352b, 0xffff0035, 0x00004000, 0x2b077702, 0x2c002602, 0x07010000, 0x0dff6a00, 0x17005201, 0x01020d40, 0x02260515, 0x200c0001, 0x01250001, + 0x0035352b, 0x0035352b, 0x2f000200, 0xfc040000, 0x1000b605, 0x5d001f00, 0x1a1a3a40, 0x085b110e, 0x21202167, 0x5a1c1801, 0x0110100e, 0x1b20640e, + 0x0f185f10, 0x000f0100, 0x006f003f, 0x00df00af, 0x080600ff, 0x1d1a4000, 0x02000048, 0x120e601c, 0x03026017, 0x3fe13f00, 0x2f3911e1, 0x715d5e2b, + 0x0132e133, 0x3232e610, 0x32e1102f, 0xe1f6105d, 0x312f3911, 0x11331330, 0x16163221, 0x02141512, 0x21230406, 0x34252311, 0x2323022e, 0x21152111, + 0x00203311, 0x9701982f, 0x5faef899, 0xf7feb660, 0x9892fea8, 0x7e420804, 0x01c975b8, 0xa2b0fe50, 0x0c010801, 0x91022503, 0xf4feb55c, 0xe9feb9b0, + 0x83025ebb, 0x8ad59260, 0xa20efe43, 0x24011dfe, 0xc700ffff, 0x0e050000, 0x26023507, 0x00003100, 0x52010701, 0x52018b00, 0x0b401300, 0x26052001, + 0x2f210a01, 0x0125000a, 0x2b00352b, 0xffff0035, 0xecff7d00, 0x73077105, 0x32002602, 0x07010000, 0x54004300, 0x15005201, 0x052802b4, 0xffb80226, + 0x282eb4ab, 0x0125000a, 0x2b00352b, 0xffff0035, 0xecff7d00, 0x73077105, 0x32002602, 0x07010000, 0x02017600, 0x13005201, 0x34020b40, 0x58022605, + 0x000a2e28, 0x352b0125, 0x00352b00, 0x7d00ffff, 0x7105ecff, 0x26027307, 0x00003200, 0x4b010701, 0x5201ae00, 0x0b401300, 0x26052802, 0x28300502, + 0x0125000a, 0x2b00352b, 0xffff0035, 0xecff7d00, 0x35077105, 0x32002602, 0x07010000, 0x7d005201, 0x15005201, 0x053002b4, 0xffb80226, 0x3f31b4f0, + 0x0125000a, 0x2b00352b, 0xffff0035, 0xecff7d00, 0x2b077105, 0x32002602, 0x07010000, 0xaa006a00, 0x17005201, 0x02030d40, 0x03260531, 0x3c280102, + 0x0125000a, 0x0035352b, 0x0035352b, 0x8d000100, 0xdd032d01, 0x0b007b04, 0x00b98700, 0xb3f0ff06, 0x06481714, 0x40e0ffb8, 0x48120f18, 0x17141000, + 0x0f200048, 0x10094812, 0x09481714, 0x48120f20, 0xf0ffb803, 0x481714b3, 0xe0ffb803, 0x120f3040, 0x010d4048, 0x03050507, 0x5001010b, 0x08030103, + 0x02040a0a, 0x01002002, 0x00200000, 0x00700050, 0x00a00080, 0x00d000c0, 0x060900f0, 0x1900b300, 0x715d5e3f, 0x32331132, 0x2f013311, 0x3311335d, + 0x33113311, 0x0030315d, 0x2b2b2b2b, 0x2b2b2b01, 0x3701012b, 0x01170101, 0x01010701, 0xfecb0127, 0x3d0169c2, 0xfe684201, 0x663f01bf, 0xc3febefe, + 0x01d30267, 0xc2fe693f, 0xfe673e01, 0x66c0febf, 0xc5fe3d01, 0x03000067, 0xb4ff7d00, 0xfc057105, 0x26001a00, 0x5c003100, 0x1f293a40, 0x1b041e2a, + 0x19015b27, 0x11040e0b, 0xc0336704, 0x33bf0133, 0x01337001, 0x335f332f, 0x115b1b02, 0x291f3266, 0x2d042a1e, 0x01195f22, 0x09040b0e, 0x2d04161a, + 0x13090c5f, 0xe1333f00, 0x1712333f, 0x1711e139, 0xf6100139, 0x5d5d5de1, 0x11f6105d, 0x11e13917, 0x30313917, 0x12160701, 0x06021415, 0x27222306, + 0x26372707, 0x12343502, 0x32333636, 0x01371716, 0x01171614, 0x22232626, 0x1005020e, 0x16160127, 0x023e3233, 0x5b5c1405, 0xeda0515e, 0x4e85bd9b, + 0x5b615a89, 0xa3f09e4c, 0x5042a15e, 0x302eb7fc, 0x72304302, 0x6ca67247, 0x586a0334, 0x722fbefd, 0x6ba57245, 0x95ae0532, 0xb7defe63, 0xc6eafea9, + 0x4e7f476c, 0x2a016491, 0x1501aabe, 0x262a6bc4, 0x83e1fc7f, 0xb1034ed1, 0x9751201d, 0x01018ada, 0x1c54fc97, 0xdb99511e, 0xb800ffff, 0xdd04ecff, + 0x26027307, 0x00003800, 0x43000701, 0x52013d00, 0x01b41500, 0x01260518, 0xb4c0ffb8, 0x000b181e, 0x352b0125, 0x00352b00, 0xb800ffff, 0xdd04ecff, + 0x26027307, 0x00003800, 0x76000701, 0x5201c500, 0x0b401300, 0x26052401, 0x1e184801, 0x0125000b, 0x2b00352b, 0xffff0035, 0xecffb800, 0x7307dd04, + 0x38002602, 0x07010000, 0x79004b01, 0x15005201, 0x051801b4, 0xffb80126, 0x1820b4fd, 0x0125000b, 0x2b00352b, 0xffff0035, 0xecffb800, 0x2b07dd04, + 0x38002602, 0x07010000, 0x7d006a00, 0x17005201, 0x01020d40, 0x02260521, 0x2c180101, 0x0125000b, 0x0035352b, 0x0035352b, 0x0000ffff, 0x37040000, + 0x26027307, 0x00003c00, 0x76000701, 0x52013100, 0x0b401300, 0x26051501, 0x0f096301, 0x01250207, 0x2b00352b, 0x02000035, 0x0000c700, 0xb6053304, + 0x1b001000, 0x21403c00, 0x67005b17, 0x011d9f1d, 0x11011d10, 0x085a070b, 0x60111c64, 0x0b601b06, 0x0b060b06, 0x07030907, 0x3f3f0012, 0x2f393912, + 0x10e1102f, 0xf61001e1, 0x5d3232e1, 0xe1f6105d, 0x14013031, 0x2323020e, 0x33112311, 0x1e323315, 0x32330102, 0x3435023e, 0x04232326, 0xcf7e3733, + 0xbaba9698, 0x7ec286b0, 0x814efd3c, 0x2e5b8b5d, 0x03a0aea4, 0x81a85b0e, 0x05c3fe4d, 0x6d39fcb6, 0x2067fea0, 0x8f517147, 0x01000088, 0xecffae00, + 0x1f067504, 0x6d004b00, 0x46074840, 0x0047352e, 0x191f190f, 0x1903192f, 0x48130d40, 0x2e6f2e5f, 0x1f000f02, 0x03002f00, 0x002e1908, 0x03192e00, + 0x11472441, 0x4d104d57, 0x4dc04d20, 0x41474003, 0x07244c54, 0x3a160335, 0x41014750, 0x1a501f15, 0x3f001616, 0x3f3fe133, 0x391712e1, 0xe1f61001, + 0xe1f6105d, 0x2f391712, 0x5d5e2f2f, 0x105d2b5d, 0x31e110e1, 0x0e140130, 0x1e141504, 0x031e1702, 0x020e1415, 0x27262223, 0x33031e35, 0x35023e32, + 0x27022e34, 0x3435032e, 0x3435043e, 0x2223022e, 0x1115020e, 0x3e341123, 0x1e323302, 0x2bf20302, 0x2b3f4b3f, 0x3946270e, 0x213d5838, 0x558d6538, + 0x1a358b61, 0x254c4841, 0x18345138, 0x38482b11, 0x1635553f, 0x3e483e29, 0x533c2129, 0x42583133, 0x713fb627, 0x985c5c9c, 0xec043c6c, 0x3c495939, + 0x151e3637, 0x26312721, 0x60524825, 0x517e573e, 0xa6222327, 0x0f181f10, 0x28402d19, 0x3a383b24, 0x43442823, 0x4f362a46, 0x433a363f, 0x293e2a2c, + 0x53301313, 0x044efb41, 0x558d68b0, 0x744c2625, 0x5e00ffff, 0x9c03ecff, 0x26022106, 0x00004400, 0x43000601, 0x15000094, 0x113302b4, 0xffb80226, + 0x3339b4e5, 0x0125220c, 0x2b00352b, 0xffff0035, 0xecff5e00, 0x21069c03, 0x44002602, 0x06010000, 0x00357600, 0x0b401300, 0x26113f02, 0x39338502, + 0x0125220c, 0x2b00352b, 0xffff0035, 0xecff5e00, 0x21069c03, 0x44002602, 0x06010000, 0x00e24b01, 0x0b401300, 0x26113302, 0x333b3302, 0x0125220c, + 0x2b00352b, 0xffff0035, 0xecff5e00, 0xe3059c03, 0x44002602, 0x06010000, 0x00bd5201, 0x0b401300, 0x26113b02, 0x4a3c2902, 0x0125220c, 0x2b00352b, + 0xffff0035, 0xecff5e00, 0xd9059c03, 0x44002602, 0x06010000, 0x00de6a00, 0x0d401700, 0x113c0203, 0x2f020326, 0x220c4733, 0x352b0125, 0x352b0035, + 0xffff0035, 0xecff5e00, 0x87069c03, 0x44002602, 0x06010000, 0x00de5001, 0x0d401700, 0x11380203, 0x30020326, 0x220c333d, 0x352b0125, 0x352b0035, + 0x03000035, 0xecff5e00, 0x5e044406, 0x47003800, 0x7c005000, 0x142f1a40, 0x04481e4b, 0x4c004242, 0x571d2648, 0x5f524f52, 0x03529f52, 0x0e004839, + 0x40c0ffb8, 0x48110d2e, 0x56000e0e, 0x10420051, 0x1e420242, 0x10040050, 0x0f040204, 0x024b1f4b, 0x114b4b07, 0x2f503c21, 0x4816342c, 0x17500a0d, + 0x00101114, 0xe133333f, 0x333f3232, 0x1132e133, 0x5d5e2f39, 0x32e15d33, 0xe610015d, 0x102b2f32, 0xf6105de1, 0x3911e132, 0x32e1332f, 0x30313939, + 0x37363413, 0x2e343537, 0x06222302, 0x36362707, 0x17163233, 0x32333636, 0x1515021e, 0x33161621, 0x37023e32, 0x23030e15, 0x0e272622, 0x2e222303, + 0x16143702, 0x023e3233, 0x0e073535, 0x06220103, 0x2e342107, 0xece75e02, 0x51371db8, 0x428f5334, 0x64b64a40, 0x332ba683, 0x9a6167a6, 0x60fd396c, + 0x31939305, 0x254a4e55, 0x554f4b27, 0x3eca8a31, 0x745f4c22, 0x5a7b474a, 0x4f61bd34, 0x2b4c683d, 0x497a5a8f, 0x6e850320, 0xd7010b7f, 0x0154371a, + 0x08b0a433, 0x5a434507, 0x22301837, 0x55382889, 0x475d555d, 0x716eb581, 0x130ab6c1, 0x13a2121d, 0x7208121c, 0x3b553673, 0x7b51271f, 0x26565c52, + 0x634f754d, 0x39200407, 0x9c630251, 0x50714495, 0xffff002c, 0x14fe7100, 0x5e046f03, 0x46002602, 0x07010000, 0x42017a00, 0x0b000000, 0x262f01b6, + 0x250d0520, 0x00352b01, 0x7100ffff, 0xe103ecff, 0x26022106, 0x00004800, 0x43000601, 0x15000094, 0x112802b4, 0xffb80226, 0x282eb4b9, 0x01250f05, + 0x2b00352b, 0xffff0035, 0xecff7100, 0x2106e103, 0x48002602, 0x06010000, 0x00527600, 0x0b401300, 0x26113402, 0x2e287602, 0x01250f05, 0x2b00352b, + 0xffff0035, 0xecff7100, 0x2106e103, 0x48002602, 0x06010000, 0x00de4b01, 0x0b401300, 0x26112802, 0x28300302, 0x01250f05, 0x2b00352b, 0xffff0035, + 0xecff7100, 0xd905e103, 0x48002602, 0x06010000, 0x00da6a00, 0x0d401700, 0x11310203, 0x00020326, 0x0f053c28, 0x352b0125, 0x352b0035, 0xffff0035, + 0x0000deff, 0x21066701, 0xf3002602, 0x07010000, 0x55fe4300, 0x15000000, 0x110401b4, 0xffb80126, 0x040ab49a, 0x01250001, 0x2b00352b, 0xffff0035, + 0x0000ae00, 0x21064202, 0xf3002602, 0x07010000, 0x30ff7600, 0x13000000, 0x10010b40, 0x74012611, 0x00010a04, 0x352b0125, 0x00352b00, 0xbdffffff, + 0x55020000, 0x26022106, 0x0000f300, 0x4b010701, 0x0000bbfe, 0x0b401300, 0x26110401, 0x040c0001, 0x01250001, 0x2b00352b, 0xffff0035, 0x0000eeff, + 0xd9052502, 0xf3002602, 0x07010000, 0xbbfe6a00, 0x17000000, 0x01020d40, 0x0226110d, 0x18040001, 0x01250001, 0x0035352b, 0x0035352b, 0x6f000200, + 0x2d04ecff, 0x27002306, 0x74003900, 0x28124640, 0x20230048, 0x1c041916, 0x221c1822, 0x03181c22, 0x3b57000a, 0x3bd03b40, 0x0f033be0, 0x3206013b, + 0x3a560a48, 0x16231920, 0x21211704, 0x0f502d1d, 0x0faf0f12, 0x30020fbf, 0x0f17010f, 0x011d0f17, 0x16055037, 0x3fe13f00, 0x2f2f3939, 0x33115d5d, + 0x3311e110, 0x39171211, 0xe1f61001, 0x105d5d5e, 0x391711e6, 0x112f2f2f, 0x10391712, 0x303132e1, 0x020e1401, 0x022e2223, 0x023e3435, 0x17163233, + 0x27262637, 0x26372705, 0x16372726, 0x17371716, 0x07031e07, 0x23022e34, 0x15020e22, 0x33021e14, 0x2d043632, 0x6fb27d43, 0x477faf68, 0x69a8763f, + 0x082b9a66, 0xff5a781f, 0x28d94a00, 0x41462f55, 0x4ae33b7a, 0x4f6f43c3, 0x4622bc2c, 0x6d4d4b6e, 0x47212146, 0x879a4c6d, 0xdc8e3d02, 0x7f424f98, + 0xb87777b9, 0x3c3b417e, 0x51c07604, 0x1c837299, 0x207b1a37, 0x718a2c48, 0xbb9c4175, 0x6b38b0dd, 0x582e3252, 0x7d4c5583, 0x00c7315a, 0xae00ffff, + 0x12040000, 0x2602e305, 0x00005100, 0x52010601, 0x130000f9, 0x21010b40, 0x02012611, 0x170b3022, 0x352b0125, 0x00352b00, 0x7100ffff, 0x2d04ecff, + 0x26022106, 0x00005200, 0x43000601, 0x150000d8, 0x112002b4, 0xffb80226, 0x2026b4d7, 0x0125000a, 0x2b00352b, 0xffff0035, 0xecff7100, 0x21062d04, + 0x52002602, 0x06010000, 0x00507600, 0x0b401300, 0x26112c02, 0x26204e02, 0x0125000a, 0x2b00352b, 0xffff0035, 0xecff7100, 0x21062d04, 0x52002602, + 0x06010000, 0x00fb4b01, 0x02b41500, 0x02261120, 0xb4faffb8, 0x000a2028, 0x352b0125, 0x00352b00, 0x7100ffff, 0x2d04ecff, 0x2602e305, 0x00005200, + 0x52010601, 0x150000e2, 0x112802b4, 0xffb80226, 0x3729b4fd, 0x0125000a, 0x2b00352b, 0xffff0035, 0xecff7100, 0xd9052d04, 0x52002602, 0x06010000, + 0x00f96a00, 0x03b61900, 0x26112902, 0xffb80203, 0x3420b4f9, 0x0125000a, 0x0035352b, 0x0035352b, 0x66000300, 0x0204f800, 0x0300ac04, 0x2b001700, + 0x15406000, 0x22012d30, 0x0e1818aa, 0x660356aa, 0x03280203, 0x03020338, 0xf0ffb800, 0x0d092840, 0x27040048, 0x011d10ad, 0x1d011d0f, 0xad09011d, + 0x13101300, 0x13601320, 0x13c013b0, 0x070713d0, 0xad001313, 0x3f00b301, 0x5e2f33e1, 0x3311e15d, 0xe1715d2f, 0x2b332f01, 0xe15d5d33, 0x5de12f33, + 0x35133031, 0x34011521, 0x3233023e, 0x1415021e, 0x2223020e, 0x3411022e, 0x3233023e, 0x1415021e, 0x2223020e, 0x0366022e, 0x12bffd9c, 0x1718291f, + 0x1212202a, 0x18172a20, 0x12121f29, 0x1718291f, 0x1212202a, 0x18172a20, 0x02121f29, 0xfe969687, 0x1e2f23ee, 0x2f1e0d0d, 0x1f2f2123, 0x2f1f0e0e, + 0x2f23db02, 0x1e0d0d1e, 0x2f21232f, 0x1f0e0e1f, 0x0300002f, 0xb4ff7300, 0x91042f04, 0x24001a00, 0x5c002d00, 0x1f273b40, 0x1b041e28, 0x18174825, + 0x08071516, 0x0d08090a, 0x402f5700, 0xe02fd02f, 0x2f0f032f, 0x481b0601, 0x282e560d, 0x041f271e, 0x0750222b, 0x0415180a, 0x10121605, 0x0508502b, + 0xc63f0016, 0x12c63fe1, 0x11e13917, 0x10013917, 0x5d5ee1f6, 0x11f6105d, 0x11e13917, 0x30313917, 0x020e1401, 0x07272223, 0x26263727, 0x023e3435, + 0x17163233, 0x16071737, 0x16140516, 0x26260117, 0x05062223, 0x16012734, 0x36323316, 0x7d432f04, 0x627d6fb2, 0x3f508344, 0xb37c4346, 0x31713f6f, + 0x3e508344, 0x1300fd45, 0x1d8d0116, 0x879a2d4b, 0xfe274402, 0x2d481f72, 0x2702879a, 0x4c91d589, 0x834a6d35, 0x8889d548, 0x1d4b91d3, 0x81496c1c, + 0x5486d149, 0x87023383, 0xd1cf1211, 0x7bfd639f, 0x00d31011, 0xa400ffff, 0x0804ecff, 0x26022106, 0x00005800, 0x43000601, 0x150000a3, 0x111b01b4, + 0xffb80126, 0x1b21b49b, 0x0125190c, 0x2b00352b, 0xffff0035, 0xecffa400, 0x21060804, 0x58002602, 0x06010000, 0x00607600, 0x0b401300, 0x26112701, + 0x211b5701, 0x0125190c, 0x2b00352b, 0xffff0035, 0xecffa400, 0x21060804, 0x58002602, 0x06010000, 0x00084b01, 0x0b401300, 0x26111b01, 0x1b230001, + 0x0125190c, 0x2b00352b, 0xffff0035, 0xecffa400, 0xd9050804, 0x58002602, 0x06010000, 0x00026a00, 0x02b61900, 0x26112401, 0xffb80102, 0x2f1bb4fb, + 0x0125190c, 0x0035352b, 0x0035352b, 0x0a00ffff, 0xdf0314fe, 0x26022106, 0x00005c00, 0x76000601, 0x1300000e, 0x2f010b40, 0x67012611, 0x0f002923, + 0x352b0125, 0x00352b00, 0xae000200, 0x3f0414fe, 0x20001406, 0x38003100, 0x482f1f40, 0x1033570a, 0x20270133, 0x471b151f, 0x1d32541c, 0x2c1b1b00, + 0x160f1550, 0x05005021, 0x333f0010, 0xe1333fe1, 0x10013f3f, 0x3232e1f6, 0x105d3232, 0x3031e1f6, 0x33033e01, 0x15021e32, 0x23020e14, 0x27022e22, + 0x16171623, 0x23111516, 0x07113311, 0x020e2225, 0x1e141507, 0x36323302, 0x01263435, 0x4d3a1764, 0x9a5e3c60, 0x6d3c3c6d, 0x603b5e9a, 0x0c173b4d, + 0x04020303, 0x0108b6b6, 0x41694c1f, 0x411b021f, 0x7f87516c, 0x23b6037f, 0x481b2d3d, 0x8d8cd48f, 0x1a4990d5, 0x22203a2b, 0x10371a1f, 0x00082bfe, + 0x119436fe, 0x5f8c5e2e, 0x6b9d6529, 0xd0ccda37, 0xffff00ce, 0x14fe0a00, 0xd905df03, 0x5c002602, 0x06010000, 0x00b16a00, 0x0d401700, 0x112c0102, + 0x0b010226, 0x0f003723, 0x352b0125, 0x352b0035, 0xffff0035, 0x00000000, 0xc106dd04, 0x24002602, 0x07010000, 0x33004d01, 0x15005201, 0x051702b4, + 0xffb80226, 0x1615b4ff, 0x01250704, 0x2b00352b, 0xffff0035, 0xecff5e00, 0x6f059c03, 0x44002602, 0x06010000, 0x00e64d01, 0x0b401300, 0x26113502, + 0x34332302, 0x0125220c, 0x2b00352b, 0xffff0035, 0x00000000, 0x4007dd04, 0x24002602, 0x07010000, 0x21004e01, 0x13005201, 0x1a020b40, 0x00022605, + 0x0704151f, 0x352b0125, 0x00352b00, 0x5e00ffff, 0x9c03ecff, 0x2602ee05, 0x00004400, 0x4e010601, 0x130000d4, 0x38020b40, 0x25022611, 0x220c333d, + 0x352b0125, 0x00352b00, 0x0000ffff, 0xdd0442fe, 0x2602bc05, 0x00002400, 0x51010701, 0x00006203, 0x0d401400, 0x01278002, 0x00012730, 0x00002727, + 0x5d2b0125, 0xffff355d, 0x42fe5e00, 0x5e04b803, 0x44002602, 0x07010000, 0x5a025101, 0x20000000, 0xf0021640, 0x45a00145, 0x01459001, 0x30014580, + 0x45000145, 0x25000045, 0x5d5d2b01, 0x355d5d5d, 0x7d00ffff, 0x9804ecff, 0x26027307, 0x00002600, 0x76000701, 0x52010a01, 0x0b401300, 0x26053001, + 0x2a24cd01, 0x01252018, 0x2b00352b, 0xffff0035, 0xecff7100, 0x21066f03, 0x46002602, 0x06010000, 0x004a7600, 0x0e401700, 0x26112c01, 0x01201001, + 0x052620a7, 0x2b01250d, 0x2b00355d, 0xffff0035, 0xecff7d00, 0x73079804, 0x26002602, 0x07010000, 0xb4004b01, 0x13005201, 0x24010b40, 0x77012605, + 0x2018242c, 0x352b0125, 0x00352b00, 0x7100ffff, 0x8f03ecff, 0x26022106, 0x00004600, 0x4b010601, 0x170000f5, 0x20010e40, 0x10012611, 0x28530128, + 0x250d0520, 0x355d2b01, 0x00352b00, 0x7d00ffff, 0x9804ecff, 0x26023707, 0x00002600, 0x4f010701, 0x52010002, 0x0b401300, 0x26052f01, 0x2c248001, + 0x01252018, 0x2b00352b, 0xffff0035, 0xecff7100, 0xe5056f03, 0x46002602, 0x07010000, 0x33014f01, 0x17000000, 0x2b010e40, 0x10012611, 0x204d0120, + 0x250d0528, 0x355d2b01, 0x00352b00, 0x7d00ffff, 0x9804ecff, 0x26027307, 0x00002600, 0x4c010701, 0x5201a800, 0x0b401300, 0x26052901, 0x242f6b01, + 0x01252018, 0x2b00352b, 0xffff0035, 0xecff7100, 0x2106a003, 0x46002602, 0x06010000, 0x00064c01, 0x0e401700, 0x26112501, 0x012b1001, 0x05202b64, + 0x2b01250d, 0x2b00355d, 0xffff0035, 0x0000c700, 0x7307fc04, 0x27002602, 0x07010000, 0x4a004c01, 0x15005201, 0x051d02b4, 0xffb80226, 0x1823b4b7, + 0x01250006, 0x2b00352b, 0xffff0035, 0xecff7100, 0x14065205, 0x47002600, 0x07010000, 0xd9023802, 0x1e000000, 0x406f1540, 0x01404f01, 0x4001401f, + 0x48080640, 0x3e3e5e02, 0x01251d1d, 0x5d2b352b, 0xffff5d5d, 0x00002f00, 0xb605fc04, 0x92000602, 0x02000000, 0xecff7100, 0x14069e04, 0x38002700, + 0x31405500, 0x2d001b1f, 0x1d1d2647, 0x5526220b, 0x013a103a, 0x560b4836, 0x25152739, 0x1e224f1d, 0x0020101e, 0x00165033, 0x20101010, 0x10100310, + 0x06005028, 0x333f0016, 0x335d3fe1, 0x39123fe1, 0x32e1332f, 0xf610013f, 0xe6105de1, 0x2f391132, 0x3232e110, 0x25303132, 0x23030e23, 0x35022e22, + 0x33023e34, 0x17021e32, 0x26272633, 0x21353526, 0x33352135, 0x23153315, 0x32252311, 0x3537023e, 0x23022e34, 0x14150622, 0x08540316, 0x604d3b16, + 0x6e9a5d3c, 0x9a6e3c3c, 0x4d603b5d, 0x030c163c, 0xfe040203, 0xb67b0185, 0xfe939c9c, 0x41694cc6, 0x411b021f, 0x7f87516c, 0x3d22937f, 0x8a461a2e, + 0xce8786cc, 0x2c19478b, 0x1f21203a, 0x9e10371a, 0x89b6b689, 0x2c832bfb, 0x275a8559, 0x34669561, 0xc3c6c3cf, 0xc700ffff, 0xbe030000, 0x2602c106, + 0x00002800, 0x4d010701, 0x52010a00, 0x0b401300, 0x26050e01, 0x0d0c0201, 0x01250001, 0x2b00352b, 0xffff0035, 0xecff7100, 0x6f05e103, 0x48002602, + 0x06010000, 0x00004d01, 0x0b401300, 0x26112a02, 0x29281102, 0x01250f05, 0x2b00352b, 0xffff0035, 0x0000c700, 0x4007be03, 0x28002602, 0x07010000, + 0xfbff4e01, 0x13005201, 0x11010b40, 0x06012605, 0x00010c16, 0x352b0125, 0x00352b00, 0x7100ffff, 0xe103ecff, 0x2602ee05, 0x00004800, 0x4e010601, + 0x130000ea, 0x2d020b40, 0x0f022611, 0x0f052832, 0x352b0125, 0x00352b00, 0xc700ffff, 0xbe030000, 0x26021a07, 0x00002800, 0x4f010701, 0x35013f01, + 0x0b401300, 0x26051701, 0x140c0701, 0x01250001, 0x2b00352b, 0xffff0035, 0xecff7100, 0xe505e103, 0x48002602, 0x07010000, 0x2b014f01, 0x13000000, + 0x33020b40, 0x0c022611, 0x0f053028, 0x352b0125, 0x00352b00, 0xc700ffff, 0xbe0342fe, 0x2602b605, 0x00002800, 0x51010701, 0x00002102, 0x00b90e00, + 0xb4c1ff01, 0x00001212, 0x352b0125, 0x7100ffff, 0xe10361fe, 0x26025e04, 0x00004800, 0x51010701, 0x1f003b02, 0x13402700, 0x013bf002, 0xa0013be0, + 0x3b90013b, 0x013b5001, 0xb8013b00, 0x3bb4caff, 0x251a1a3b, 0x5d5d2b01, 0x5d5d5d5d, 0xffff0035, 0x0000c700, 0x7307be03, 0x28002602, 0x07010000, + 0xf9ff4c01, 0x13005201, 0x11010b40, 0x04012605, 0x00010c17, 0x352b0125, 0x00352b00, 0x7100ffff, 0xe103ecff, 0x26022106, 0x00004800, 0x4c010601, + 0x130000e4, 0x2d020b40, 0x09022611, 0x0f052833, 0x352b0125, 0x00352b00, 0x7d00ffff, 0xf204ecff, 0x26027307, 0x00002a00, 0x4b010701, 0x5201d700, + 0x0b401300, 0x26052c01, 0x2c346d01, 0x0125010c, 0x2b00352b, 0xffff0035, 0x14fe2500, 0x2106fc03, 0x4a002602, 0x06010000, 0x00c64b01, 0x0b401300, + 0x26115f03, 0x5f670303, 0x01250027, 0x2b00352b, 0xffff0035, 0xecff7d00, 0x4007f204, 0x2a002602, 0x07010000, 0xc9004e01, 0x13005201, 0x31010b40, + 0x5f012605, 0x010c2c36, 0x352b0125, 0x00352b00, 0x2500ffff, 0xfc0314fe, 0x2602ee05, 0x00004a00, 0x4e010601, 0x150000b1, 0x116403b4, 0xffb80326, + 0x5f69b4ef, 0x01250027, 0x2b00352b, 0xffff0035, 0xecff7d00, 0x3707f204, 0x2a002602, 0x07010000, 0x0c024f01, 0x13005201, 0x37010b40, 0x5f012605, + 0x010c342c, 0x352b0125, 0x00352b00, 0x2500ffff, 0xfc0314fe, 0x2602e505, 0x00004a00, 0x4f010701, 0x0000f200, 0x03b41500, 0x0326116a, 0xb4ecffb8, + 0x0027675f, 0x352b0125, 0x00352b00, 0x7d00ffff, 0xf2043bfe, 0x2602cb05, 0x00002a00, 0x39020701, 0x00001701, 0x01b60b00, 0x0c322c51, 0x2b012501, + 0xffff0035, 0x14fe2500, 0x2106fc03, 0x4a002602, 0x06010000, 0x002d3a02, 0x0b401300, 0x26116403, 0x5f651c03, 0x01250027, 0x2b00352b, 0xffff0035, + 0x0000c700, 0x7307d504, 0x2b002602, 0x07010000, 0x7f004b01, 0x15005201, 0x050c01b4, 0xffb80126, 0x0c14b4ff, 0x01250005, 0x2b00352b, 0xffff0035, + 0x0000ae00, 0xaa071204, 0x4b002602, 0x07010000, 0x2b004b01, 0x1f008901, 0x0d701440, 0x010c7001, 0x011a7001, 0x0126021a, 0x0b1a2219, 0x2b012518, + 0x5d2b0035, 0x005d5d35, 0x00000200, 0x9c050000, 0x1300b605, 0x56001700, 0x04142f40, 0x0b075a0c, 0x19b01965, 0x0119af01, 0x17011910, 0x005a0f03, + 0x0e186410, 0x0a16175f, 0x07136012, 0x13171303, 0x10011317, 0x0105120b, 0x333f0003, 0x3912333f, 0x112f2f39, 0xe1103333, 0xe1103232, 0x32f61001, + 0x5d3232e1, 0xf6105d5d, 0x3232e132, 0x35133031, 0x35211533, 0x15331533, 0x11231123, 0x11231121, 0x35013523, 0xbac71521, 0xc7ba9a02, 0x66fdbac7, + 0x1b04c7ba, 0xc30466fd, 0xf3f3f3f3, 0x02d3fb96, 0x0456fdaa, 0x8bfe962d, 0x0100dfdf, 0x00001200, 0x14061204, 0x55002100, 0x0b133440, 0x55214700, + 0x60231023, 0x03238023, 0x0a121617, 0x540b0f47, 0x4f0d1522, 0xcf0ebf12, 0x0e0e020e, 0x50041710, 0x1d101d00, 0x1d031d20, 0x0b001010, 0x3f001500, + 0x5d3f3f32, 0x391233e1, 0xe1335d2f, 0xf6100132, 0x3232e132, 0xf6105d32, 0x313912e1, 0x34112130, 0x0e222326, 0x23111502, 0x33352311, 0x21153335, + 0x07152115, 0x33033e33, 0x11151632, 0x70695c03, 0x1d436e51, 0xb69c9cb6, 0x85fe7b01, 0x45190a08, 0xb7305c52, 0x829a02b9, 0x94663482, 0x04f0fd60, + 0xb6b689d5, 0x2b90b889, 0xbf142a3f, 0x005cfdd2, 0xf5ffffff, 0xc4020000, 0x26023507, 0x00002c00, 0x52010701, 0x5201f3fe, 0x0b401300, 0x26051401, + 0x23150101, 0x01250001, 0x2b00352b, 0xffff0035, 0x0000a3ff, 0xe3057202, 0xf3002602, 0x07010000, 0xa1fe5201, 0x13000000, 0x0c010b40, 0x01012611, + 0x00011b0d, 0x352b0125, 0x00352b00, 0x3d00ffff, 0x7c020000, 0x2602c106, 0x00002c00, 0x4d010701, 0x520122ff, 0x0b401300, 0x26050e01, 0x0d0c0101, + 0x01250001, 0x2b00352b, 0xffff0035, 0x0000ebff, 0x6f052a02, 0xf3002602, 0x07010000, 0xd0fe4d01, 0x13000000, 0x06010b40, 0x01012611, 0x00010504, + 0x352b0125, 0x00352b00, 0x3000ffff, 0x8a020000, 0x26024007, 0x00002c00, 0x4e010701, 0x52010fff, 0x0b401300, 0x26051101, 0x0c160201, 0x01250001, + 0x2b00352b, 0xffff0035, 0x0000deff, 0xee053802, 0xf3002602, 0x07010000, 0xbdfe4e01, 0x13000000, 0x09010b40, 0x02012611, 0x0001040e, 0x352b0125, + 0x00352b00, 0x5200ffff, 0x640242fe, 0x2602b605, 0x00002c00, 0x51010701, 0x00009c00, 0x01b60b00, 0x01121900, 0x2b012500, 0xffff0035, 0x42fe4400, + 0xe5058301, 0x4c002602, 0x06010000, 0x00255101, 0x0a401000, 0x01256f02, 0x00252500, 0x2b012500, 0xffff355d, 0x00005200, 0x37076402, 0x2c002602, + 0x07010000, 0x50004f01, 0x13005201, 0x17010b40, 0x00012605, 0x0001140c, 0x352b0125, 0x00352b00, 0xae000100, 0x64010000, 0x03004a04, 0x0e401a00, + 0x05200510, 0x01470002, 0x0f020454, 0x3f001500, 0xf610013f, 0x30315de1, 0x33112321, 0xb6b66401, 0xffff4a04, 0x7bfe5200, 0xb6052904, 0x2c002600, + 0x07010000, 0xb6022d00, 0x38000000, 0x18010e40, 0x4f01187f, 0x40180118, 0x18480707, 0x40c0ffb8, 0x48060617, 0xa00121bf, 0x218f0121, 0x01215001, + 0x2101210f, 0x48070740, 0x5d5d5d2b, 0x2b2b5d5d, 0x35115d5d, 0xa000ffff, 0x870314fe, 0x2600e505, 0x00004c00, 0x4d000701, 0x00001202, 0x20402e00, + 0xcf1e0203, 0x35df011e, 0x01359f01, 0x5f013580, 0x35400135, 0x01352001, 0x35013500, 0x48060640, 0x5d5d5d2b, 0x5d5d5d5d, 0x3535115d, 0x48ffffff, + 0x64027bfe, 0x26027307, 0x00002d00, 0x4b010701, 0x5201cafe, 0x0b401300, 0x26051401, 0x141c0001, 0x01250e0d, 0x2b00352b, 0xffff0035, 0x14febcff, + 0x21065702, 0x37022602, 0x07010000, 0xbdfe4b01, 0x13000000, 0x14010b40, 0x00012611, 0x0e0d141c, 0x352b0125, 0x00352b00, 0xc700ffff, 0xa2043bfe, + 0x2602b605, 0x00002e00, 0x39020601, 0x0e000073, 0xff0100b9, 0x130db4b1, 0x01250005, 0xffff352b, 0x3bfeae00, 0x1406f003, 0x4e002602, 0x06010000, + 0x00213902, 0x00b90e00, 0xb4c4ff01, 0x050a150f, 0x352b0125, 0xae000100, 0xf0030000, 0x11004a04, 0x0a404900, 0x0e101001, 0x11440f0f, 0xffb81101, + 0x071c40c0, 0x1011480a, 0x0f131111, 0x02132f13, 0x47030707, 0x0c125404, 0x0f050e01, 0x00150004, 0x333f323f, 0x10013939, 0x5e32e1f6, 0x2f33115d, + 0x335d2b38, 0x11393311, 0x21303133, 0x23110701, 0x14113311, 0x07060706, 0x01330137, 0xfe230301, 0xb4b46dac, 0x04040305, 0xcd330183, 0xac016ffe, + 0xfe51e901, 0xfe4a0468, 0x26663ce7, 0x01b0242c, 0xfd16fe81, 0xffff00a0, 0x0000c700, 0x7307be03, 0x2f002602, 0x07010000, 0x65ff7600, 0x15005201, + 0x051201b4, 0xffb80126, 0x0c06b470, 0x01250400, 0x2b00352b, 0xffff0035, 0x0000ab00, 0xac073402, 0x4f002602, 0x07010000, 0x22ff7600, 0x1f008b01, + 0x03701440, 0x01027001, 0x01107001, 0x01260210, 0x010a0466, 0x2b012500, 0x5d2b0035, 0x005d5d35, 0xc700ffff, 0xbe033bfe, 0x2602b605, 0x00002f00, + 0x39020601, 0x0e00002d, 0xff0100b9, 0x0c06b4dd, 0x01250400, 0xffff352b, 0x3bfe6600, 0x14066c01, 0x4f002602, 0x07010000, 0xf7fe3902, 0x0e000000, + 0xff0100b9, 0x0a04b4e0, 0x01250001, 0xffff352b, 0x0000c700, 0xb705be03, 0x2f002602, 0x07010000, 0x00013802, 0x1600a3ff, 0x0b010d40, 0x06100103, + 0x0c066001, 0x01250402, 0x00355d2b, 0xffff353f, 0x0000ae00, 0x1406b402, 0x4f002600, 0x06010000, 0x003b3802, 0x1c402600, 0x5f01136f, 0x40130113, + 0x01480806, 0x100e4004, 0x06400448, 0x045e4809, 0x25030304, 0x2b2b2b01, 0x5d5d2b35, 0xc700ffff, 0xbe030000, 0x2602b605, 0x00002f00, 0x4f010701, + 0x65fde301, 0x01b60b00, 0x000e06ab, 0x2b012504, 0xffff0035, 0x0000ae00, 0x1406a202, 0x4f002600, 0x07010000, 0x2d014f01, 0x2b0038fd, 0x13ff1640, + 0x0113ef01, 0x8f01139f, 0x137f0113, 0x01135f01, 0x0101131f, 0xb42e01b8, 0x00010c04, 0x352b0125, 0x5d5d5d5d, 0x005d5d5d, 0x1d000100, 0xbe030000, + 0x0d00b605, 0x32406100, 0x0c05060b, 0x04030005, 0x0504040d, 0x08080a05, 0x010faf0f, 0x0c010f10, 0x03060d0d, 0x000b5a07, 0x050e640a, 0x040c0d04, + 0x010c040c, 0x120a5f07, 0x3f000301, 0x3912e13f, 0x112f2f39, 0x01331133, 0x3232f610, 0x333232e1, 0x5d5d332f, 0x12113311, 0x10332f39, 0xc404877d, + 0xc48710c4, 0x133031c4, 0x37113311, 0x21110517, 0x07112115, 0xe3bac727, 0x02cffe4e, 0x6109fd3d, 0x03a40249, 0x8f63fd12, 0x39febe7d, 0x3cf801a6, + 0x0100007d, 0x0000f6ff, 0x14061d02, 0x69000b00, 0x0d403b40, 0x09020d50, 0x050a0506, 0x0b040300, 0x080b0b0a, 0x0105dd04, 0x05cb05bb, 0x1f050f02, + 0x03052f05, 0x06050506, 0x09470703, 0x0c540800, 0x04050a0b, 0x040a040a, 0x01150701, 0x3f3f0000, 0x2f393912, 0x1133112f, 0xf6100133, 0x32e13232, + 0x5e2f3232, 0x335d5d5d, 0x332f3311, 0xc404877d, 0xc48710c4, 0x315d01c4, 0x33111330, 0x07173711, 0x07112311, 0x6bb6ae27, 0x6ab6b94e, 0x03f6024e, + 0x4559fd1e, 0x3ffd7879, 0x79464a02, 0xc700ffff, 0x0e050000, 0x26027307, 0x00003100, 0x76000701, 0x5201f200, 0x0b401300, 0x26052401, 0x1e185501, + 0x0125000a, 0x2b00352b, 0xffff0035, 0x0000ae00, 0x21061204, 0x51002602, 0x06010000, 0x006f7600, 0x0b401300, 0x26112501, 0x1f195c01, 0x0125170b, + 0x2b00352b, 0xffff0035, 0x3bfec700, 0xb6050e05, 0x31002602, 0x07010000, 0xd9003902, 0x0e000000, 0xff0100b9, 0x1e18b4e1, 0x0125000a, 0xffff352b, + 0x3bfeae00, 0x5e041204, 0x51002602, 0x06010000, 0x004c3902, 0x00b90e00, 0xb4deff01, 0x170b1f19, 0x352b0125, 0xc700ffff, 0x0e050000, 0x26027307, + 0x00003100, 0x4c010701, 0x52019c00, 0x0b401300, 0x26051d01, 0x18230001, 0x0125000a, 0x2b00352b, 0xffff0035, 0x0000ae00, 0x21061204, 0x51002602, + 0x06010000, 0x000c4c01, 0x01b41500, 0x0126111e, 0xb4faffb8, 0x170b1924, 0x352b0125, 0x00352b00, 0xffffffff, 0xa1040000, 0x2700b605, 0x8f005100, + 0x06010000, 0x00e80702, 0x1d402900, 0x27551800, 0xbf0127ef, 0x279f0127, 0x01275f01, 0x2701272f, 0x48080840, 0x06064027, 0x5d2b2b48, 0x5d5d5d5d, + 0x0034f610, 0xc7000100, 0x0e057bfe, 0x2500b605, 0x2a404a00, 0x5a1f0c18, 0x15040420, 0xb0276520, 0x278f0127, 0x10270001, 0x13170227, 0x155a140e, + 0x0d1f2664, 0x12150316, 0x07120c18, 0x2f00005f, 0x3f333fe1, 0x0133333f, 0x32e1f610, 0x5d5d3232, 0x11e6105d, 0xe1102f39, 0x30313232, 0x27262201, + 0x33161635, 0x35023e32, 0x17162301, 0x11151616, 0x01331123, 0x26272633, 0x33113526, 0x020e1411, 0x4d338b03, 0x2d4e221b, 0x263d4b25, 0x060808fd, + 0xac050404, 0x07cc02d5, 0x05030403, 0x8f6437ae, 0x0b0d7bfe, 0x130b09a0, 0x04445832, 0x414c4dba, 0xe7fc398e, 0x75fbb605, 0x7d384141, 0xfa200334, + 0x659a695e, 0x01000031, 0x14feae00, 0x5e041204, 0x3a002800, 0x03032240, 0x24470c17, 0x2a102a55, 0x2a802a60, 0x47161a03, 0x1a295417, 0x10205010, + 0x15170f18, 0x1b005007, 0x3fe13f00, 0x33e13f3f, 0xe1f61001, 0xf6105d32, 0x2f3911e1, 0x22013031, 0x16352726, 0x3e323316, 0x34133502, 0x0e222326, + 0x23111502, 0x33173311, 0x3233033e, 0x14031516, 0xee02020e, 0x1a173f30, 0x2e1b2336, 0x69021323, 0x436e5170, 0x1a94b61d, 0x5245190a, 0xb9b7305c, + 0x6d482202, 0x0b0e14fe, 0x0f0b0a94, 0x03334127, 0x3482826d, 0xfd609466, 0x944a04c7, 0x142a3f2b, 0x95fcd2bf, 0x2f577b4d, 0x7d00ffff, 0x7105ecff, + 0x2602c106, 0x00003200, 0x4d010701, 0x5201bc00, 0x0b401300, 0x26052a02, 0x29280002, 0x0125000a, 0x2b00352b, 0xffff0035, 0xecff7100, 0x6f052d04, + 0x52002602, 0x06010000, 0x00144d01, 0x0b401300, 0x26112202, 0x21200002, 0x0125000a, 0x2b00352b, 0xffff0035, 0xecff7d00, 0x40077105, 0x32002602, + 0x07010000, 0xaa004e01, 0x13005201, 0x2d020b40, 0x01022605, 0x000a2832, 0x352b0125, 0x00352b00, 0x7100ffff, 0x2d04ecff, 0x2602ee05, 0x00005200, + 0x4e010601, 0x15000000, 0x112502b4, 0xffb80226, 0x202ab4ff, 0x0125000a, 0x2b00352b, 0xffff0035, 0xecff7d00, 0x73077105, 0x32002602, 0x07010000, + 0xec005301, 0x17005201, 0x02030d40, 0x03260534, 0x3c284302, 0x0125000a, 0x0035352b, 0x0035352b, 0x7100ffff, 0x2d04ecff, 0x26022106, 0x00005200, + 0x53010601, 0x17000046, 0x02030d40, 0x0326112c, 0x34204502, 0x0125000a, 0x0035352b, 0x0035352b, 0x7d000200, 0xa406ecff, 0x1900cd05, 0x5b002a00, + 0x16143640, 0x00161601, 0x275a1814, 0x00120927, 0x2c1f2c67, 0x095b1f01, 0x5f172b66, 0x0f01144f, 0x0214af14, 0x19141408, 0x03115f12, 0x040e5f1a, + 0x13045f24, 0x12005f19, 0x3fe13f00, 0x3fe13fe1, 0x2f3912e1, 0xe1715d5e, 0xe1f61001, 0x32e6105d, 0xe12f3911, 0x2f391132, 0x2130315d, 0x23060621, + 0x02262622, 0x36123435, 0x17323336, 0x11211521, 0x11211521, 0x0e220121, 0x1e141502, 0x36323302, 0x26261137, 0x0afda406, 0xa3305b2b, 0x4c4c9def, + 0x62a3f09e, 0xfdf40254, 0xfd1702c2, 0xfc3e02e9, 0x6ca67256, 0xa56b3434, 0x265a3472, 0x0b095926, 0x1701c66c, 0x1501aaaa, 0xa4176bc4, 0xfda23cfe, + 0x518504f8, 0x898ada97, 0x125199db, 0x1058040f, 0x03000011, 0xecff7100, 0x5e04e106, 0x36002a00, 0x64003f00, 0x03133b40, 0xd0481d3f, 0x31310131, + 0x2548370b, 0x0f41571c, 0x41ff0141, 0x01414001, 0x560b482b, 0x1b501d40, 0x023f2b3f, 0x06013f0f, 0x3c2e3f3f, 0x13165034, 0x2e201010, 0x06000350, + 0x333f0016, 0x3f32e133, 0x32e13333, 0x5e2f3912, 0x01e15d5d, 0x5de1f610, 0xf610715d, 0x3911e132, 0x32e15d2f, 0x30313939, 0x27262205, 0x22230606, + 0x3435022e, 0x3233023e, 0x36361716, 0x021e3233, 0x16211515, 0x3e323316, 0x0e153702, 0x16140103, 0x35363233, 0x22232634, 0x2e340506, 0x06222302, + 0x82600507, 0xc63f41cc, 0x7fae6780, 0xb37c4347, 0x3fc3796f, 0x6375b93c, 0xfd3b6e9e, 0x9799054c, 0x4c515733, 0x514d2827, 0x899afb57, 0x8c8b969a, + 0x04879a97, 0x58391bf0, 0x0b85723e, 0x6d6d7014, 0xd5914c70, 0x91d38889, 0x696a6f4b, 0xb5814770, 0xb6c1716e, 0x121d130a, 0x121c13a2, 0xd13b0208, + 0xdcd1c9d3, 0x4462cfce, 0x9c2c5071, 0xffff0095, 0x0000c700, 0x7307a004, 0x35002602, 0x07010000, 0x54007600, 0x15005201, 0x052902b4, 0xffb80226, + 0x231db4ee, 0x01250d02, 0x2b00352b, 0xffff0035, 0x0000ae00, 0x21060803, 0x55002602, 0x06010000, 0x00ce7600, 0x0b401300, 0x26112301, 0x1d174001, + 0x0125030e, 0x2b00352b, 0xffff0035, 0x3bfec700, 0xb605a004, 0x35002602, 0x06010000, 0x00773902, 0x00b90e00, 0xb4b6ff02, 0x0d02231d, 0x352b0125, + 0x6000ffff, 0x08033bfe, 0x26025e04, 0x00005500, 0x39020701, 0x0000f1fe, 0x00b90e00, 0xb408ff01, 0x030e1d17, 0x352b0125, 0xc700ffff, 0xa0040000, + 0x26027307, 0x00003500, 0x4c010701, 0x52010e00, 0x02b41500, 0x02260522, 0xb4a9ffb8, 0x0d021d28, 0x352b0125, 0x00352b00, 0x7200ffff, 0x0a030000, + 0x26022106, 0x00005500, 0x4c010701, 0x000070ff, 0x01b41500, 0x0126111c, 0xb4e3ffb8, 0x030e1722, 0x352b0125, 0x00352b00, 0x6800ffff, 0xc903ecff, + 0x26027307, 0x00003600, 0x76000701, 0x52013b00, 0x0b401300, 0x26054001, 0x3a347001, 0x01250008, 0x2b00352b, 0xffff0035, 0xecff5a00, 0x21063f03, + 0x56002602, 0x06010000, 0x00dc7600, 0x0b401300, 0x26114201, 0x3c365d01, 0x01250008, 0x2b00352b, 0xffff0035, 0xecff6800, 0x7307c903, 0x36002602, + 0x07010000, 0xedff4b01, 0x13005201, 0x34010b40, 0x22012605, 0x0008343c, 0x352b0125, 0x00352b00, 0x5a00ffff, 0x3f03ecff, 0x26022106, 0x00005600, + 0x4b010601, 0x1300009b, 0x36010b40, 0x1c012611, 0x0008363e, 0x352b0125, 0x00352b00, 0x6800ffff, 0xc90314fe, 0x2602cb05, 0x00003600, 0x7a000701, + 0x00003501, 0x00b90e00, 0xb4faff01, 0x0008343a, 0x352b0125, 0x5a00ffff, 0x3f0314fe, 0x26025e04, 0x00005600, 0x7a000701, 0x0000f000, 0x01b60b00, + 0x08363c01, 0x2b012500, 0xffff0035, 0xecff6800, 0x7307c903, 0x36002602, 0x07010000, 0xe6ff4c01, 0x13005201, 0x39010b40, 0x1b012605, 0x0008343f, + 0x352b0125, 0x00352b00, 0x5a00ffff, 0x3f03ecff, 0x26022106, 0x00005600, 0x4c010601, 0x13000097, 0x3b010b40, 0x18012611, 0x00083641, 0x352b0125, + 0x00352b00, 0x1400ffff, 0x12043bfe, 0x2602b605, 0x00003700, 0x39020601, 0x0e0000f9, 0xff0100b9, 0x0e08b4d8, 0x01250503, 0xffff352b, 0x3bfe2100, + 0x46058f02, 0x57002602, 0x07010000, 0x5dff3902, 0x0e000000, 0xff0100b9, 0x241eb4f7, 0x01250512, 0xffff352b, 0x00001400, 0x73071204, 0x37002602, + 0x07010000, 0xc8ff4c01, 0x13005201, 0x0d010b40, 0x03012605, 0x05030813, 0x352b0125, 0x00352b00, 0x2100ffff, 0xe802ecff, 0x26021406, 0x00005700, + 0x38020601, 0x0b00006f, 0x246101b6, 0x25181824, 0x00352b01, 0x14000100, 0x12040000, 0x0f00b605, 0x4c407300, 0xe001111f, 0x115f0111, 0x200211df, + 0x40113011, 0xbf080311, 0x0205ff05, 0x060a0505, 0x010f015a, 0x02f00250, 0x01021f02, 0x770d0202, 0x97018701, 0x01310301, 0x10010001, 0x03012001, + 0x0d0a0107, 0x00000760, 0x06120b03, 0x03035f02, 0x32e13f00, 0x2f39123f, 0x0132e133, 0x5d5d5e2f, 0x2f33335d, 0x39125d5d, 0x3232e110, 0x5d395d2f, + 0x31715d5d, 0x21110130, 0x21152135, 0x21152111, 0x21112311, 0xfeb60135, 0xfefe035e, 0xfe27015f, 0xd7febbd9, 0xdf013303, 0x21fea4a4, 0x0262fd95, + 0x0100959e, 0xecff2100, 0x46058f02, 0x6a002500, 0x011f14b6, 0x051c1c20, 0x40c0ffb8, 0x480b0837, 0x2f270505, 0x02273f27, 0x47231b1f, 0x1f160f13, + 0x18160216, 0x10100014, 0xb010a010, 0x0510c010, 0x22111007, 0x1f1f144f, 0x4f1e150b, 0x0f1b181a, 0x160b5000, 0x3fe13f00, 0x32e13333, 0x332f3912, + 0x2f0132e1, 0x33335d5e, 0xe1325dc4, 0x115d3232, 0x332b2f33, 0x5d003311, 0x32253031, 0x1537023e, 0x2223030e, 0x3535022e, 0x35333523, 0x37373523, + 0x15211533, 0x15211521, 0x16141521, 0x2d12fa01, 0x0d09232a, 0x19343028, 0x2c4d6a3e, 0x9b9b8b8b, 0x1401694e, 0x0201ecfe, 0x813ffefe, 0x03080604, + 0x090c068a, 0x854e2005, 0xf88afb65, 0xfce64e51, 0xfb8af889, 0xffff6261, 0xecffb800, 0x3507dd04, 0x38002602, 0x07010000, 0x60005201, 0x15005201, + 0x052001b4, 0xffb80126, 0x2f21b4ff, 0x0125000b, 0x2b00352b, 0xffff0035, 0xecffa400, 0xe3050804, 0x58002602, 0x06010000, 0x00f35201, 0x0b401300, + 0x26112301, 0x32240601, 0x0125190c, 0x2b00352b, 0xffff0035, 0xecffb800, 0xc106dd04, 0x38002602, 0x07010000, 0x8f004d01, 0x15005201, 0x051a01b4, + 0xffb80126, 0x1918b4ff, 0x0125000b, 0x2b00352b, 0xffff0035, 0xecffa400, 0x6f050804, 0x58002602, 0x06010000, 0x00214d01, 0x0b401300, 0x26111d01, + 0x1c1b0501, 0x0125190c, 0x2b00352b, 0xffff0035, 0xecffb800, 0x4007dd04, 0x38002602, 0x07010000, 0x7d004e01, 0x13005201, 0x1d010b40, 0x00012605, + 0x000b1822, 0x352b0125, 0x00352b00, 0xa400ffff, 0x0804ecff, 0x2602ee05, 0x00005800, 0x4e010601, 0x1300000c, 0x20010b40, 0x04012611, 0x190c1b25, + 0x352b0125, 0x00352b00, 0xb800ffff, 0xdd04ecff, 0x2602d907, 0x00003800, 0x50010701, 0x52017b00, 0x0d401700, 0x051d0102, 0x00010226, 0x000b1822, + 0x352b0125, 0x352b0035, 0xffff0035, 0xecffa400, 0x87060804, 0x58002602, 0x06010000, 0x000c5001, 0x0d401700, 0x11200102, 0x05010226, 0x190c1b25, + 0x352b0125, 0x352b0035, 0xffff0035, 0xecffb800, 0x7307dd04, 0x38002602, 0x07010000, 0xb2005301, 0x17005201, 0x01020d40, 0x02260524, 0x2c183601, + 0x0125000b, 0x0035352b, 0x0035352b, 0xa400ffff, 0x0e04ecff, 0x26022106, 0x00005800, 0x53010601, 0x17000050, 0x01020d40, 0x02261127, 0x2f1b4801, + 0x0125190c, 0x0035352b, 0x0035352b, 0xb800ffff, 0xdd0442fe, 0x2602b805, 0x00003800, 0x51010701, 0x00001902, 0x01b60b00, 0x0b1e250d, 0x2b012500, + 0xffff0035, 0x42fea400, 0x4a041604, 0x58002602, 0x07010000, 0xb8025101, 0x0b000000, 0x210e01b6, 0x251a1a21, 0x00352b01, 0x1400ffff, 0xfe060000, + 0x26027307, 0x00003a00, 0x4b010701, 0x52013b01, 0x0b401300, 0x26052b01, 0x2b330001, 0x01250e1e, 0x2b00352b, 0xffff0035, 0x00001400, 0x2106e305, + 0x5a002602, 0x07010000, 0xae004b01, 0x13000000, 0x30010b40, 0x00012611, 0x2e113038, 0x352b0125, 0x00352b00, 0x0000ffff, 0x37040000, 0x26027307, + 0x00003c00, 0x4b010701, 0x5201d0ff, 0x0b401300, 0x26050901, 0x09110201, 0x01250207, 0x2b00352b, 0xffff0035, 0x14fe0a00, 0x2106df03, 0x5c002602, + 0x06010000, 0x00a94b01, 0x0b401300, 0x26112301, 0x232b0201, 0x01250f00, 0x2b00352b, 0xffff0035, 0x00000000, 0x2b073704, 0x3c002602, 0x07010000, + 0xceff6a00, 0x17005201, 0x01020d40, 0x02260512, 0x1d090101, 0x01250207, 0x0035352b, 0x0035352b, 0x5200ffff, 0xfe030000, 0x26027307, 0x00003d00, + 0x76000701, 0x52013900, 0x0b401300, 0x26051601, 0x100a5e01, 0x01250001, 0x2b00352b, 0xffff0035, 0x00005200, 0x21063503, 0x5d002602, 0x06010000, + 0x00d67600, 0x0b401300, 0x26111601, 0x100a6001, 0x01250001, 0x2b00352b, 0xffff0035, 0x00005200, 0x3707fe03, 0x3d002602, 0x07010000, 0x29014f01, + 0x13005201, 0x15010b40, 0x0b012605, 0x0001120a, 0x352b0125, 0x00352b00, 0x5200ffff, 0x35030000, 0x2602e505, 0x00005d00, 0x4f010701, 0x0000cd00, + 0x0b401300, 0x26111501, 0x120a1401, 0x01250001, 0x2b00352b, 0xffff0035, 0x00005200, 0x7307fe03, 0x3d002602, 0x07010000, 0xe0ff4c01, 0x13005201, + 0x0f010b40, 0x06012605, 0x00010a15, 0x352b0125, 0x00352b00, 0x5200ffff, 0x35030000, 0x26022106, 0x00005d00, 0x4c010601, 0x13000086, 0x0f010b40, + 0x10012611, 0x00010a15, 0x352b0125, 0x00352b00, 0xae000100, 0xbe020000, 0x13001f06, 0x15402500, 0x00150909, 0x20151015, 0x47130315, 0x0d145400, + 0x00010650, 0x3f3f0015, 0xf61001e1, 0x33115de1, 0x3330312f, 0x023e3411, 0x17163233, 0x23262607, 0x15020e22, 0x552dae11, 0x633b4e7c, 0x481f2f26, + 0x273a2828, 0x6bb00413, 0x1723548d, 0x110b8d0e, 0x41533013, 0x01004efb, 0x14fecb00, 0xcb05e903, 0x4a002b00, 0x2d302a40, 0x1f1f2901, 0x16124701, + 0x12141412, 0x12000a0a, 0x12301210, 0x07041240, 0x60001312, 0x0d292916, 0x071c5023, 0x1b06500d, 0x3fe13f00, 0x2f3912e1, 0x0132e133, 0x335d5e2f, + 0x2f33112f, 0xe1103311, 0x5d322f32, 0x11013031, 0x23020e14, 0x35272622, 0x32331616, 0x1135023e, 0x35373523, 0x33023e34, 0x07171632, 0x22232626, + 0x1515020e, 0x8f021533, 0x4e7b552d, 0x1a1c3d20, 0x3b281f39, 0xc2c21326, 0x4e7c552d, 0x2f26633b, 0x2828481f, 0xf613273a, 0x3ffc4403, 0x23548d6b, + 0x08990609, 0x5330130a, 0x4bc30341, 0x8d6b8944, 0x0e172354, 0x13110b8d, 0x91415330, 0x04000089, 0x00000000, 0xaa07dd04, 0x23001600, 0x3e003100, + 0x5e400101, 0x29012989, 0x302a2abe, 0x381d3131, 0x1f0d0f83, 0x0d10020d, 0x0d020d20, 0x6623560d, 0x07230223, 0x69175908, 0x17060217, 0x83321d05, + 0x001f000f, 0x0003006f, 0x010ad600, 0x0a470a37, 0x0ac70a57, 0x03d90a04, 0x48033801, 0xc8035803, 0x1d030403, 0x1005091d, 0x04800104, 0x04d00490, + 0xffb80403, 0x061840c0, 0x1004480a, 0x0f400404, 0x2f401f40, 0x9f408f40, 0x0640df40, 0xb8090807, 0x3140f0ff, 0x235f0709, 0x0e0a201d, 0x030a1d48, + 0x0135193c, 0x10128c35, 0x128f013c, 0x600212ef, 0x31100112, 0x23023120, 0x3131123c, 0x04233c12, 0x04092904, 0x333f0012, 0x3917122f, 0x2f2f2f2f, + 0x5d5d5d5d, 0x115de110, 0x2b333333, 0x2f01e110, 0x5d5e3338, 0x382f3311, 0x33715d2b, 0x2f3d3912, 0x335d5d33, 0x18335d5d, 0x11e15d2f, 0x5d393912, + 0x5d393912, 0x715d2f33, 0x2f3311e1, 0xed2f3333, 0x0130315d, 0x01070614, 0x03210323, 0x26260123, 0x023e3435, 0x021e3233, 0x032e0303, 0x07030e27, + 0x033e1303, 0x0e153337, 0x13230703, 0x22232634, 0x16141506, 0x36323317, 0x333c5003, 0x9abefc01, 0xbc9cd3fd, 0x3a33f801, 0x32523b20, 0x233e5431, + 0x1106940c, 0x07081212, 0x06111212, 0x30158591, 0xdb102a2f, 0x514d3f10, 0x3fbc7923, 0x333f3132, 0x3f320c31, 0x60469c05, 0x0123fb19, 0x0479fe87, + 0x466019db, 0x1d384f33, 0xfc4f371d, 0x107d015f, 0x1c3b362e, 0x2e363c1c, 0x0483fe0f, 0x413d188b, 0x16101c40, 0x16424741, 0x3c34f6fe, 0x3b33343c, + 0x05003c03, 0xecff5e00, 0xaa079c03, 0x31000d00, 0x54004000, 0x9c006000, 0x83552340, 0x411f410f, 0x4103412f, 0x4b835b41, 0x0d4b0d06, 0x1e1a0d4b, + 0x3147370f, 0x623f6255, 0x1a483e01, 0xd0ffb828, 0x110d3a40, 0x09102848, 0x2828480c, 0x5861561a, 0x0d50508c, 0x0d700d60, 0x500d0d03, 0x8c5e0550, + 0x46504640, 0x27464602, 0x382b5024, 0x2b1e1e52, 0x0f503210, 0x150e1615, 0x0501050f, 0x3f5d2f00, 0x3fe1333f, 0x10e12f39, 0x2f3332e1, 0x3911e15d, + 0x5d2f332f, 0x1001e110, 0x2b2f32e6, 0x5de1102b, 0x32e1f610, 0x39391132, 0xcd102f2f, 0x2f33e110, 0x3031e15d, 0x37033e01, 0x030e1533, 0x27012307, + 0x23030e23, 0x35022e22, 0x37373634, 0x022e3435, 0x07062223, 0x33363627, 0x15021e32, 0x3e322511, 0x07353502, 0x1415030e, 0x0e140116, 0x2e222302, + 0x3e343502, 0x1e323302, 0x26340702, 0x15062223, 0x32331614, 0x15c70136, 0x102a2f30, 0x4d3f10db, 0x01782352, 0x21082552, 0x3f604e42, 0x30557445, + 0x1db8ece7, 0x53345137, 0x4a40428f, 0x956664b6, 0x2ffe3061, 0x2b4c683d, 0x497a5a8f, 0x86016120, 0x31543d23, 0x213b5232, 0x32523b21, 0x233e5430, + 0x32313f75, 0x3138393f, 0x18b8063f, 0x1c40413d, 0x47411610, 0x5cf91642, 0x2a412d98, 0x7b512714, 0x08b0a454, 0x5a434507, 0x22301837, 0x29382889, + 0xfd628a59, 0x4d267f10, 0x07634f75, 0x51392004, 0x05565c33, 0x38513308, 0x4f381d1d, 0x384f3333, 0x4f371d1d, 0x3c3c3534, 0x3c3c3535, 0xfeffffff, + 0x56060000, 0x26027307, 0x00008800, 0x76000701, 0x52012502, 0x02b41500, 0x02260520, 0xb44801b8, 0x00051a14, 0x352b0125, 0x00352b00, 0x5e00ffff, + 0x4406ecff, 0x26022106, 0x0000a800, 0x76000701, 0x00007501, 0x0b401300, 0x26115d03, 0x57517103, 0x01251c00, 0x2b00352b, 0xffff0035, 0xb4ff7d00, + 0x73077105, 0x9a002602, 0x07010000, 0x02017600, 0x13005201, 0x3e030b40, 0x58032605, 0x04113832, 0x352b0125, 0x00352b00, 0x7300ffff, 0x2f04b4ff, + 0x26022106, 0x0000ba00, 0x76000601, 0x13000050, 0x3a030b40, 0x4c032611, 0x000d342e, 0x352b0125, 0x00352b00, 0x6800ffff, 0xc9033bfe, 0x2602cb05, + 0x00003600, 0x39020601, 0x0e000000, 0xff0100b9, 0x3a34b4da, 0x01250008, 0xffff352b, 0x3bfe5a00, 0x5e043f03, 0x56002602, 0x06010000, 0x00bb3902, + 0x00b90e00, 0xb4e1ff01, 0x00083c36, 0x352b0125, 0x02010100, 0x9a03d904, 0x14002106, 0x11402500, 0x04040e0f, 0x0408c000, 0x0f00800e, 0x02085f08, + 0x5d2f0008, 0x32cc1a33, 0xcc1a2f01, 0x332f3d39, 0x01303133, 0x27262623, 0x23070606, 0x37033e35, 0x17031e33, 0x33799a03, 0x6a36346c, 0x441a7933, + 0xc0103b43, 0x45433b10, 0x22d90419, 0x61373761, 0x4c1d1b22, 0x22225151, 0x1d4c5151, 0x02010100, 0x9a03d904, 0x14002106, 0x11402500, 0x10100506, + 0x130bc000, 0x0f10800d, 0x02065f06, 0x5d2f0006, 0x32cd1a33, 0xcc1a2f01, 0x332f3d39, 0x01303133, 0x2307030e, 0x3527032e, 0x17161633, 0x33373636, + 0x45199a03, 0xc0103b43, 0x44433b10, 0x6a33791a, 0x336c3436, 0x1d060679, 0x2350514c, 0x4c515023, 0x61221b1d, 0x22613838, 0x1b010100, 0x5a03d904, + 0x03006f05, 0x0a401500, 0x8f000001, 0x035f030f, 0x2f000302, 0x2f01e15d, 0x013031cd, 0x01211521, 0xfd3f021b, 0x966f05c1, 0x21010100, 0x7b03d904, + 0x1500ee05, 0x1c402f00, 0x00ff8315, 0x0bc00001, 0x7f000a83, 0x9f0a8f0a, 0x800a030a, 0x050f8f10, 0x0502055f, 0xe15d2f00, 0x325dcd1a, 0x1ae12f01, + 0x31e15ddc, 0x030e0130, 0x022e2223, 0x031e3327, 0x023e3233, 0x047b0337, 0x476d4d2c, 0x27496d49, 0x1c046c03, 0x242c4330, 0x04223341, 0x653dee05, + 0x4927294a, 0x322b3f66, 0x1b090719, 0x01002831, 0x0005a000, 0xe5057501, 0x1c000d00, 0x87081040, 0x9f910300, 0xef0bcf0b, 0x0b30030b, 0x2f000b01, + 0x01e55d5d, 0x3031e12f, 0x33363413, 0x15021e32, 0x22230614, 0x2d3da026, 0x111d2716, 0x3d2d2c3f, 0x363c7305, 0x1e2b1c0d, 0x0038393a, 0x6d010200, + 0x3103d904, 0x13008706, 0x40001f00, 0x83142d40, 0x003f000f, 0x005f004f, 0x831a0004, 0x0a010a30, 0x0f0f8c17, 0x0f3f0f1f, 0x0f5f0f4f, 0x0fff0faf, + 0x1d0f0607, 0x5f050f8c, 0x00050205, 0xd4e15d2f, 0x01e15d5e, 0xd4e15d2f, 0x3031e15d, 0x020e1401, 0x022e2223, 0x023e3435, 0x021e3233, 0x23263407, + 0x14150622, 0x36323316, 0x3d233103, 0x52323154, 0x3b20203b, 0x54303252, 0x3f75233e, 0x393f3231, 0x053f3138, 0x385133b2, 0x4f381d1d, 0x384f3333, + 0x4f371d1d, 0x3c3c3534, 0x3c3c3535, 0x1f000100, 0x5e0142fe, 0x14000000, 0x09401600, 0x84008006, 0x8e03120d, 0xe12f000a, 0xe12f012f, 0x3031cc1a, + 0x33161417, 0x15373632, 0x22230606, 0x3e343526, 0x06333702, 0x192234b4, 0x401a0e2d, 0x1e64641d, 0x811a382f, 0x2b2dee8b, 0x08710405, 0x2a5a6808, + 0x1334404b, 0x01000085, 0xd9040201, 0xe305d103, 0x38001b00, 0x170f2340, 0x1702172f, 0x09200900, 0x16090702, 0x400e8f05, 0x0e481310, 0x480b0740, + 0x8f130e0e, 0x01000f09, 0x5d2f0000, 0x2f33e132, 0x33e12b2b, 0x5d5e2f01, 0x30315dcc, 0x022e2201, 0x07062223, 0x33033e23, 0x33021e32, 0x33373632, + 0xfe02030e, 0x464c4f28, 0x0e302d20, 0x35210568, 0x512a2e4a, 0x2d1d454c, 0x05690f2e, 0x044a3521, 0x232b23db, 0x623c3e35, 0x2a232545, 0x3c3e3423, + 0x00264561, 0xdf000200, 0xbe03d904, 0x0d002106, 0x2d001b00, 0x0e141940, 0x5040070e, 0x003f0100, 0x0002004f, 0x80920513, 0x5f0c0f1b, 0x000c020c, + 0x1a335d2f, 0x2f0132ed, 0xcd1a5d5d, 0x31cd2f39, 0x033e1330, 0x0e153337, 0x25230703, 0x3337033e, 0x07030e15, 0x2f16df23, 0xc7102a2f, 0x514d3f10, + 0x6b016523, 0x2a2f3015, 0x3f10c610, 0x6423514d, 0x4c1df404, 0x15225151, 0x5358511b, 0x4c1d1b1c, 0x15225151, 0x5358511b, 0x0100001c, 0xd904f801, + 0x21060403, 0x27000d00, 0x10051640, 0x0548110c, 0x000c4006, 0x05000100, 0x0c0f8092, 0x0c020c5f, 0x1a5d2f00, 0x5d2f01ed, 0x32cd1a33, 0x0130312b, + 0x3337033e, 0x07030e15, 0x0af80123, 0x04101314, 0x2d2106c7, 0x046c1834, 0x514d1ef4, 0x18152150, 0x2056574e, 0x14010300, 0x89030c05, 0x0d00b406, + 0x29001b00, 0x19406600, 0x240f841c, 0x0f02241f, 0xcf241f24, 0x40240324, 0x24480c09, 0x84160d24, 0xc0ffb80e, 0x0d092740, 0x050e0e48, 0x48110c10, + 0x0d0c0605, 0x0c0c9205, 0x2791111f, 0x600119ff, 0xd019b019, 0x190f0319, 0x0602191f, 0x5e2f0019, 0x335d5d5d, 0x2f3932e5, 0x332f01ed, 0x332b32cd, + 0x11e12b2f, 0x5d2b2f33, 0x3031e171, 0x37033e01, 0x030e1533, 0x34272307, 0x1e323336, 0x06141502, 0x25262223, 0x32333634, 0x1415021e, 0x26222306, + 0x190bfe01, 0xcf081618, 0x3d383012, 0x38ea5220, 0x1a231329, 0x29263a10, 0x38b50138, 0x1a231328, 0x28263a10, 0x1e870538, 0x24514f4b, 0x504d2014, + 0x36062551, 0x26190c30, 0x3232351b, 0x0c303635, 0x351b2619, 0xffff3232, 0x0000e9ff, 0xbc05dd04, 0x24002602, 0x07010000, 0xf1fd5401, 0x180097ff, + 0x031a02b6, 0x01223002, 0xb4e9ffb8, 0x04042222, 0x5d2b0125, 0x353f0035, 0x93000100, 0x91014802, 0x13005e03, 0x1a403300, 0x15801510, 0x15a01590, + 0xc0960a04, 0x0200d000, 0x00440034, 0x00740064, 0xffb80004, 0x0a07b6c0, 0x9b050048, 0xe52f000f, 0x5d2b2f01, 0x315ded5d, 0x3e341330, 0x1e323302, + 0x0e141502, 0x2e222302, 0x22149302, 0x2f1a1b2e, 0x22141422, 0x2e1b1a2f, 0xd3021422, 0x0f213526, 0x2635210f, 0x10223425, 0x00342210, 0xe7ffffff, + 0x4d040000, 0x2700b805, 0x8f002800, 0x07010000, 0xeffd5401, 0x2e0097ff, 0x031101b4, 0xffb81201, 0x0f08b2c0, 0x9dffb848, 0x12120f40, 0x00250202, + 0xbf1b6700, 0x1b0f011b, 0x5d5d0101, 0x0134fe10, 0x00352b2b, 0xffff353f, 0x0000e7ff, 0xb8055005, 0x2b002600, 0x0701007b, 0xeffd5401, 0x310097ff, + 0x031101b4, 0xffb81201, 0x1108b2c0, 0xb1ffb848, 0x12121240, 0x00250606, 0x1b6f6500, 0x011b2f01, 0x1b011b10, 0x5d5d1001, 0x2b34fe5d, 0x3f00352b, + 0xffff0035, 0x0000e7ff, 0xb8056003, 0x2c002700, 0x0000fc00, 0x54010701, 0x97ffeffd, 0x16406600, 0x01031101, 0xa00112c0, 0x12300112, 0x01122001, + 0x00011210, 0xffb80112, 0x122f40a5, 0x25060612, 0x0100f000, 0xc00100d0, 0x00af0100, 0x01007001, 0x50010060, 0x00400100, 0x01003f01, 0x011b2000, + 0x0707401b, 0x07401b48, 0x2b014809, 0x5d115d2b, 0x5d5d5d5d, 0x5d5d5d5d, 0x5d5d2b35, 0x5d5d5d5d, 0x353f0035, 0xe7ffffff, 0xc305ecff, 0x2600cd05, + 0x00523200, 0x54010701, 0x97ffeffd, 0x30404700, 0x02032d02, 0x80012ea0, 0x2e70012e, 0x012e5001, 0x10012e20, 0x2e00012e, 0x2e2e2401, 0x01250a0a, + 0x01005000, 0x00010050, 0x0137bf67, 0x3701370f, 0x5d5d1001, 0x345d5dfe, 0x5d5d2b34, 0x5d5d5d5d, 0x3f00355d, 0xffff0035, 0x0000e7ff, 0xb8055405, + 0x3c002700, 0x00001d01, 0x54010701, 0x97ffeffd, 0x01b44b00, 0x0f01030e, 0xb6c0ffb8, 0xa0481716, 0xb80f010f, 0x09b2c0ff, 0xffb84811, 0x0f1c40d6, + 0x2507070f, 0x0105c000, 0x7f0105b0, 0x05200105, 0x183f0501, 0x01182f01, 0x0101180f, 0x115d5d5d, 0x5d5d5d5d, 0x5d2b2b35, 0x3f00352b, 0xffff0035, + 0x0000e7ff, 0xcd050006, 0x76012600, 0x0701005a, 0xeffd5401, 0x350097ff, 0x35012340, 0x36200103, 0x01361001, 0x16013600, 0x13133636, 0x1d500025, + 0x011d5001, 0x3fbf671d, 0x013f0f01, 0x5d10013f, 0x5d5dfe5d, 0x5d5d2b34, 0x3f00355d, 0xffff0035, 0xecffe4ff, 0xb4067702, 0x86012602, 0x07010000, + 0xd0fe5501, 0x18000000, 0x02030f40, 0x0124c001, 0x20012440, 0x00153a24, 0x5d2b0125, 0x3535355d, 0x0000ffff, 0xdd040000, 0x0602bc05, 0x00002400, + 0xc700ffff, 0x87040000, 0x0602b605, 0x00002500, 0xc7000100, 0xbe030000, 0x0500b605, 0x0a404300, 0x00f400c4, 0x0100b002, 0xffb80002, 0x081f40c0, + 0x00004813, 0x2f070f07, 0x6f074f07, 0x40070407, 0x03481811, 0x0664045a, 0x03055f02, 0x3f001203, 0x1001e13f, 0x5d2be1f6, 0x2b2f3311, 0x315d5d5f, + 0x21150130, 0x03112311, 0xbac3fdbe, 0xfaa6b605, 0x00b605f0, 0x25000200, 0x68040000, 0x0500b605, 0x58000e00, 0x04051140, 0x0b020606, 0x7001605b, + 0xf001b001, 0xb8010401, 0x2740c0ff, 0x01480a06, 0x102f1001, 0x106f105f, 0x109f107f, 0x100610bf, 0x48090640, 0x06025b0a, 0x48110a20, 0x0a030406, + 0x0012025f, 0x333fe13f, 0xe12f012b, 0x33115d2b, 0xe15d2b2f, 0x2f3d3912, 0x30313333, 0x35211525, 0x06073301, 0x21030706, 0x04262603, 0x01bdfb68, + 0x115ebbc2, 0x02fe1d2a, 0x2e1ffcae, 0x05857d7d, 0xa84acd31, 0x020afd5b, 0x00a861f0, 0xc700ffff, 0xbe030000, 0x0602b605, 0x00002800, 0x5200ffff, + 0xfe030000, 0x0602b605, 0x00003d00, 0xc700ffff, 0xd5040000, 0x0602b605, 0x00002b00, 0x7d000300, 0x7105ecff, 0x0300cd05, 0x2b001700, 0x34405300, + 0x01000100, 0x045b220e, 0x2dd02d67, 0x012dcf01, 0x3f012d80, 0x022d6f2d, 0x660e5b18, 0x0f60032c, 0xdf00af00, 0x0400ff00, 0x1d000008, 0x04135f27, + 0x13095f1d, 0x3fe13f00, 0x2f3912e1, 0x01e15d5e, 0x5de1f610, 0x105d5d5d, 0x3912e1f6, 0x312f2f39, 0x15210130, 0x02142521, 0x22230606, 0x35022626, + 0x36361234, 0x16163233, 0x1e140512, 0x3e323302, 0x2e343502, 0x0e222302, 0x02ec0102, 0x03eafd16, 0xeda05185, 0x9defa39b, 0xf09e4c4c, 0xa0eb9ba3, + 0x34d1fb51, 0x7272a56b, 0x32326ba5, 0x7272a46a, 0x03346ca6, 0xa945a139, 0x6cc6eafe, 0x1701c66c, 0x1501aaaa, 0xc56b6bc4, 0x89abebfe, 0x515199db, + 0x8a89db99, 0x515197da, 0xffffda97, 0x00005200, 0xb6056402, 0x2c000602, 0xffff0000, 0x0000c700, 0xb605a204, 0x2e000602, 0x01000000, 0x00000000, + 0xb6058b04, 0x5f000c00, 0x0e2f1c40, 0x0eff0eef, 0x12400e03, 0x400e4815, 0x0e48100d, 0x48090640, 0x05050b0c, 0xffb80a09, 0x0a2240f0, 0x4f002f01, + 0x9f005f00, 0xcf00af00, 0x0700ef00, 0x00010010, 0x20050010, 0x05480e0a, 0x000a030b, 0x323f0012, 0x012b333f, 0x5d5d382f, 0x33382f32, 0x332f3d39, + 0x2b2b2b33, 0x2130315d, 0x26260123, 0x07060627, 0x33012301, 0xfec68b04, 0x0f2e1fdb, 0xfe1d2a11, 0xe701c5d9, 0x619a03bb, 0xa84b4ba8, 0x0560fc5b, + 0xffff00b6, 0x0000c700, 0xb6052f06, 0x30000602, 0xffff0000, 0x0000c700, 0xb6050e05, 0x31000602, 0x03000000, 0x00005200, 0xb605ee03, 0x07000300, + 0x63000b00, 0x00143e40, 0x01011b01, 0x01000100, 0x0006060b, 0x08c00108, 0x01083f01, 0x0d4f0d08, 0x30020d6f, 0x0707010d, 0x0b800b70, 0x010b4f02, + 0x0f5f030b, 0x0200af00, 0x04000008, 0x120a5f0b, 0x03045f07, 0x3fe13f00, 0x2f3911e1, 0x01e15d5e, 0x335d5d2f, 0x105d5d2f, 0x715d5dce, 0x39112f32, + 0x5d2f2f39, 0x1330315d, 0x03211521, 0x01211521, 0xcd352115, 0x5afda602, 0xfc4a0352, 0xfc7303b6, 0xa24e0364, 0xfba40a03, 0x00a4a492, 0x7d00ffff, + 0x7105ecff, 0x0602cd05, 0x00003200, 0xc7000100, 0xc1040000, 0x0700b605, 0x1e403100, 0x65005a01, 0x0109b009, 0x099f096f, 0x100309af, 0x5a040109, + 0x03086405, 0x0503065f, 0x3f001200, 0x01e13f32, 0x5de1f610, 0xf6105d5d, 0x213031e1, 0x11211123, 0x04211123, 0x7bfdbbc1, 0x05fa03ba, 0x05eefa12, + 0xffff00b6, 0x0000c700, 0xb6053304, 0x33000602, 0x01000000, 0x00004e00, 0xb6051204, 0x5a000b00, 0x5b080d40, 0x17142006, 0x02060248, 0xb80a0106, + 0x2640c0ff, 0x0a481506, 0x0d0f0d0a, 0x0d6f0d2f, 0x06040d8f, 0x035b0907, 0x013f012f, 0x02080102, 0x5f070902, 0x5f090304, 0x3f001200, 0x12e13fe1, + 0x332f3d39, 0x5d2f1801, 0x5e32e133, 0x2f33115d, 0x3939122b, 0xe12b2f2f, 0x35333031, 0x21350101, 0x01012115, 0x014e1521, 0x036efe9e, 0x0158fd7d, + 0x0267fe85, 0x660298f7, 0xa4932502, 0xa4fdeefd, 0xffff00a4, 0x00001400, 0xb6051204, 0x37000602, 0xffff0000, 0x00000000, 0xb6053704, 0x3c000602, + 0x03000000, 0xecff6800, 0xcb05ba05, 0x2e002100, 0x81003b00, 0x02225040, 0x11011167, 0x9a213b5a, 0x12870112, 0x01127a01, 0x281b1212, 0x3d67085b, + 0x00013d2f, 0x3dd0013d, 0xbf3daf01, 0x3d90023d, 0x013d5f01, 0x0f013d30, 0x023d1f3d, 0x1b5b3506, 0x3b223c66, 0x13101360, 0x21602f2e, 0x21132102, + 0x13112113, 0x3f000400, 0x2f39393f, 0x1033112f, 0x331132e1, 0x0132e110, 0x5ee1f610, 0x5d5d5d5d, 0x71715d5d, 0x11e1f610, 0x5d5d2f39, 0xe133335d, + 0x3132325d, 0x15330130, 0x021e3233, 0x040e1415, 0x23152323, 0x2e222335, 0x3e343504, 0x13333302, 0x023e3233, 0x022e3435, 0x0e22032b, 0x1e141502, + 0x02333302, 0x8649bbb4, 0x183c7ec2, 0x9f785434, 0x2fbb2f65, 0x54789f65, 0x7e3d1835, 0xbb4a85c2, 0x5b8b5d1a, 0x7f53292e, 0x39bb3957, 0x28547f57, + 0x5d8b5b2e, 0xb4cb051a, 0x66c4995e, 0x6e7b813d, 0xe1e13052, 0x7b6e5230, 0xc4663d81, 0x53fc5e99, 0x5894693b, 0x3c688b4e, 0x4e8b683c, 0x3b699458, + 0x0000ffff, 0x60040000, 0x0602b605, 0x00003b00, 0x68000100, 0xba050000, 0x2500b605, 0x47407000, 0x0109671f, 0x9a1c5a09, 0x0a87010a, 0x010a7a01, + 0x23130a0a, 0x2767005b, 0x0001272f, 0x27d00127, 0xbf27af01, 0x27900227, 0x01275f01, 0x0f012730, 0x02271f27, 0x135b1606, 0x1c1f2666, 0x0b0b0860, + 0x141d2409, 0x00120903, 0x33333f3f, 0x332f3912, 0x100132e1, 0x5d5ee1f6, 0x5d5d5d5d, 0x1071715d, 0x3911e1f6, 0x5d5d5d2f, 0x325de133, 0x14013031, + 0x2323040e, 0x23112311, 0x35042e22, 0x14113311, 0x3333021e, 0x33113311, 0x11353632, 0x18ba0533, 0x9f785434, 0x2fbb2f65, 0x54789f65, 0x2ec11835, + 0x1a5d8b5b, 0xb7ba1abb, 0x3dd703c0, 0x4e6a777f, 0x0142fe2e, 0x684e2dbe, 0x013d7e77, 0x5821fee3, 0x0334628d, 0xc6a6fc5a, 0x00e301b1, 0x4e000100, + 0xa6050000, 0x2f00cd05, 0x51407c00, 0x220b5b0e, 0x0b20255b, 0x04020b30, 0x252f010b, 0x25df253f, 0x0b0425ef, 0x0b070125, 0x13250b25, 0x23065b2b, + 0x23022316, 0xd031671d, 0x31cf0131, 0x01318001, 0x316f313f, 0x190d0902, 0x0d08020d, 0x66135b05, 0x185f0030, 0x0a222604, 0x0c255f0d, 0x333f0012, + 0x323232e1, 0x1001e13f, 0x5e33e1f6, 0x5d5d5d5d, 0x32f6105d, 0x3911e15d, 0x5e2f2f39, 0x5d5d5d5d, 0xe110e110, 0x22013031, 0x1415020e, 0x1517021e, + 0x2e213521, 0x3e343503, 0x1e323302, 0x0e141502, 0x15210702, 0x033e3521, 0x022e3435, 0xa472fa02, 0x5123326a, 0xb0fd5f83, 0x6f406201, 0xa0512e50, + 0xeb9b9aec, 0x502e51a0, 0x6201416e, 0x835fb0fd, 0x6a322351, 0x442905a4, 0x6475ba81, 0x4197abbb, 0x8730a493, 0x966fc7a8, 0x5e5eacf4, 0x6f96f4ac, + 0x3087a8c7, 0x974193a4, 0x7564bbab, 0x004481ba, 0x4000ffff, 0x77020000, 0x26022b07, 0x00002c00, 0x6a000701, 0x52010dff, 0x0d401700, 0x05150102, + 0x00010226, 0x0001200c, 0x352b0125, 0x352b0035, 0xffff0035, 0x00000000, 0x2b073704, 0x3c002602, 0x07010000, 0xceff6a00, 0x17005201, 0x01020d40, + 0x02260512, 0x1d090101, 0x01250207, 0x0035352b, 0x0035352b, 0x7100ffff, 0x9104ecff, 0x26022106, 0x00007e01, 0x54010601, 0x13000019, 0x4a020b40, + 0x16022611, 0x2f16443e, 0x352b0125, 0x00352b00, 0x5a00ffff, 0x5c03ecff, 0x26022106, 0x00008201, 0x54010601, 0x130000ca, 0x46010b40, 0x6d012611, + 0x2d18403a, 0x352b0125, 0x00352b00, 0xae00ffff, 0x120414fe, 0x26022106, 0x00008401, 0x54010601, 0x13000044, 0x25010b40, 0x62012611, 0x170b1f19, + 0x352b0125, 0x00352b00, 0xa400ffff, 0x7702ecff, 0x26022106, 0x00008601, 0x54010701, 0x0000cefe, 0x01b41500, 0x01261122, 0xb4fdffb8, 0x00152216, + 0x352b0125, 0x00352b00, 0xa400ffff, 0x3d04ecff, 0x2602b406, 0x00009201, 0x55010601, 0x12000010, 0x010203b2, 0xb4eeffb8, 0x1b05422c, 0x352b0125, + 0x02003535, 0xecff7100, 0x5e049104, 0x3d001000, 0x1e403800, 0x47051e38, 0x103f282f, 0x480e013f, 0x223e5616, 0x500b1e0f, 0x2f380f1b, 0x3350002c, + 0x3f001611, 0x3232e133, 0x33e13f32, 0xf610013f, 0xd4105de1, 0x3232e132, 0x32253031, 0x3537023e, 0x23022e34, 0x14150622, 0x2e221716, 0x3e343502, + 0x16323302, 0x36363317, 0x030e3337, 0x16141115, 0x37363233, 0x23060615, 0x27022e22, 0x02030e23, 0x41694c35, 0x411b021f, 0x7f87516c, 0x9a5d667f, + 0x713d3c6e, 0x907064a2, 0x210a0c31, 0x130b8f19, 0x2232080d, 0x0f08250e, 0x3f262241, 0x0c0c2432, 0x604d3b16, 0x98622f83, 0x9d650f68, 0xccda376b, + 0x4897cdd1, 0x8d8cd48f, 0x534990d5, 0x1f522355, 0x7e796821, 0x3c5dfe37, 0x85030733, 0x27101109, 0x3d223040, 0x02001a2e, 0x14feae00, 0x1f067504, + 0x3d001b00, 0x34405900, 0x08054839, 0x2f331f05, 0x05330233, 0x2c150533, 0x3f570c47, 0x3f303f20, 0x47152202, 0x153e5416, 0x5032081b, 0x0801330f, + 0x22003333, 0x16115027, 0x0100501c, 0x3fe13f00, 0x391132e1, 0xe15d5e2f, 0x10013f39, 0x5d32e1f6, 0x12e1f610, 0x2f2f3939, 0x1039125d, 0x013031e1, + 0x15021e32, 0x15070614, 0x14151616, 0x2223020e, 0x23112726, 0x023e3411, 0x020e2217, 0x031e1115, 0x023e3233, 0x022e3435, 0x33352323, 0x35023e32, + 0x02022e34, 0x76a36077, 0xb08f9843, 0xad783fb9, 0x3ca4606d, 0xa87945b6, 0x4b643863, 0x524e202c, 0x6f502452, 0x60351f46, 0x4d664f84, 0x254e7752, + 0x065f4426, 0x9462311f, 0x17ad9562, 0xbaca1506, 0x376da26c, 0xe9fd1f20, 0xb27b3406, 0x1f963773, 0xfc607f4c, 0x151e1292, 0x6f4d280b, 0x4d755047, + 0x49289825, 0x5e3f3d66, 0x01001f3e, 0x14fe0a00, 0x4a04df03, 0x63001b00, 0x401d1640, 0x50481512, 0x1d0f011d, 0x1d4f1d2f, 0x051a0703, 0x06480501, + 0x40f0ffb8, 0x0b060615, 0x0c131300, 0x201b001a, 0x031b401b, 0x1b101b08, 0xffb80c0d, 0x0c0a40f0, 0x130f0c1a, 0x05150b00, 0x3f3f001b, 0x333f3333, + 0x33382f01, 0x5d5e382f, 0x3d391233, 0x3333332f, 0xe1382f18, 0x5d5d5e5d, 0x2530312b, 0x2315030e, 0x37023e34, 0x1e133301, 0x3e331703, 0x33133703, + 0x22166002, 0x0fbe0c18, 0xfe15251b, 0x0ed9bd60, 0x04141a1e, 0x19140506, 0xbcc70b1b, 0x86883e12, 0x792a347e, 0x043f8d89, 0x28bafd3e, 0x1b505a5d, + 0x5a5d521a, 0x004c0221, 0x6f000200, 0x2d04ecff, 0x32001f06, 0x51004400, 0x461a2e40, 0x05053800, 0x0f48332e, 0x40465724, 0xe046d046, 0x460f0346, + 0x483d0601, 0x0045562e, 0x0a1a4224, 0x420a3838, 0x15162950, 0x010a1050, 0xe1333f00, 0x3911e13f, 0x1239122f, 0x10013339, 0x5d5ee1f6, 0x32f6105d, + 0x2f3911e1, 0x31e13939, 0x032e0130, 0x023e3435, 0x021e3233, 0x032e0717, 0x020e2223, 0x021e1415, 0x15031e17, 0x23020e14, 0x35022e22, 0x01023e34, + 0x27022e34, 0x1415030e, 0x3233021e, 0x38140236, 0x3325435d, 0x444a7d5c, 0x23516071, 0x5149204a, 0x3c2a335b, 0x3c1b1227, 0x86554762, 0x7f45315c, + 0xae676db2, 0x7243477f, 0x1eb40199, 0x3a355239, 0x2644687e, 0x8e436949, 0x20aa0399, 0x3962564b, 0x22466948, 0x13271f14, 0x1e261491, 0x32251712, + 0x3d3e251b, 0x6c312941, 0x7f58937e, 0x3d4383c1, 0x6a6eaa75, 0xfe537aa3, 0x596f4551, 0x3a101f49, 0x42679162, 0xb42d516f, 0x5a000100, 0x5c03ecff, + 0x39005e04, 0x37405900, 0x0101231d, 0x572d1018, 0xb03ba03b, 0x033bc03b, 0x3b5f3b3f, 0x013b1002, 0x23234634, 0x56184705, 0x50021e3a, 0x790139bf, + 0x02398939, 0x310a3939, 0x0a102850, 0x00161350, 0xe13fe13f, 0x5d2f3912, 0x0139e15d, 0x33e1f610, 0x5d5de12f, 0x32e6105d, 0x122f3911, 0x01303139, + 0x06222315, 0x021e1415, 0x023e3233, 0x06061537, 0x022e2223, 0x023e3435, 0x032e3537, 0x023e3435, 0x021e3233, 0x26260717, 0x15062223, 0x33021e14, + 0x8a819c02, 0x60462985, 0x515c3337, 0x9e3b1f47, 0x6ca6716d, 0x523f2635, 0x33462b2c, 0x90693a1c, 0x525a3556, 0x4b3f284d, 0x736c4781, 0x3d644827, + 0x5b998702, 0x2a45335e, 0x1f180f12, 0x2922a010, 0x43755631, 0x293e583e, 0x2b0e0b0f, 0x4632503e, 0x09264a6d, 0x93141c13, 0x4d4d2622, 0x1227402d, + 0x71000100, 0x6a036ffe, 0x33001406, 0x1f403900, 0x1f03032e, 0x14201a46, 0x10351401, 0x0235c035, 0x0029480a, 0x34562900, 0x00032e19, 0x00000150, + 0x3232e13f, 0xe610012f, 0xe1102f32, 0x5dd6105d, 0x2f33e132, 0x13303133, 0x0e152135, 0x1e141505, 0x031e1702, 0x020e1415, 0x033e2307, 0x022e3435, + 0x35032e27, 0x36361234, 0x06070637, 0x02ae2306, 0x7eb880b6, 0x2d0d284c, 0x474b7854, 0x15193a5f, 0xaa162b22, 0x14222b18, 0x4d5b300e, 0x2e5c8759, + 0x6cc99a5c, 0x60282f2b, 0x997b052f, 0xb3cd768d, 0x306f859a, 0x23406c59, 0x3b2e0f11, 0x582d2a48, 0x201f4b52, 0x1d434848, 0x1f222514, 0x6c42110f, + 0x01946698, 0x6cd9e801, 0x04020303, 0xae000100, 0x120414fe, 0x18005e04, 0x1d403100, 0x55184700, 0x601a101a, 0x031a801a, 0x0b470a0e, 0x040e1954, + 0x0c101450, 0x00150b0f, 0x3f3f001b, 0x33e13f3f, 0xe1f61001, 0xf6105d32, 0x013031e1, 0x23263411, 0x15020e22, 0x33112311, 0x033e3317, 0x15163233, + 0x695c0311, 0x436e5170, 0x1a94b61d, 0x5245190a, 0xb9b7305c, 0xaf0414fe, 0x66348282, 0xc7fd6094, 0x2b944a04, 0xbf142a3f, 0x0047fbd2, 0x71000300, + 0x1704ecff, 0x13002b06, 0x27001e00, 0x3a405b00, 0x00471925, 0x290f2957, 0x0129d001, 0x299f297f, 0x01294002, 0x0601290f, 0x0a471a24, 0x501a2856, + 0xba0124cb, 0x24890124, 0x0f022499, 0x24080124, 0x501f1424, 0x5014010f, 0x3f001605, 0x12e13fe1, 0x5d5e2f39, 0xe15d5d5d, 0xe1f61001, 0x5d5d5e32, + 0x10715d5d, 0x3132e1f6, 0x02140130, 0x22230606, 0x35022626, 0x36361234, 0x16163233, 0x3e320112, 0x1e213702, 0x0e221303, 0x02210702, 0x34170402, + 0x767eb370, 0x333874af, 0x767eb16f, 0xfe3a75b0, 0x4568492b, 0xcbfd0323, 0x68442102, 0x4466474a, 0x33020523, 0x0c038409, 0xced7febc, 0x01ce6d6d, + 0x01bcbc29, 0x6b6cce29, 0xfcd7fecd, 0xdf944ab9, 0x95df9395, 0x4510054b, 0x0189cd89, 0x00130111, 0xa4000100, 0x7702ecff, 0x15004804, 0x23403500, + 0x09010930, 0x17001709, 0x17201710, 0x17701760, 0x17c01790, 0x080817d0, 0x54144701, 0x0f500416, 0x000f0016, 0x01e13f3f, 0x5ee1f610, 0x2f33115d, + 0x0130315d, 0x33161411, 0x37023e32, 0x23030e15, 0x35022e22, 0x3f5a0111, 0x2a2d1248, 0x290d0924, 0x3e183430, 0x042c4d6a, 0x61fcfc48, 0x08060462, + 0x0c068a03, 0x4e200509, 0x04036585, 0xae00ffff, 0xf0030000, 0x06024a04, 0x0000fa00, 0xf2ff0100, 0x1704ecff, 0x2e002106, 0x2a406800, 0x1a151522, + 0x29011212, 0x290a290a, 0x011a5000, 0x1a441a34, 0x101a0002, 0x031a201a, 0x60301a1a, 0x02309030, 0x0601300f, 0xffb8002e, 0x000f40f0, 0x0129152e, + 0x50170e01, 0x5007161e, 0x3f00010e, 0x11e13fe1, 0x3f332f39, 0x33382f01, 0x115d5d5e, 0x5d5d2f33, 0x3939125d, 0x2f182f3d, 0x11333311, 0x33113912, + 0x01233031, 0x23032e27, 0x35070622, 0x32333636, 0x0117021e, 0x32331616, 0x06153736, 0x2e222306, 0x2e032702, 0x06232703, 0x0e030706, 0x0d35b801, + 0x2b3d2b1f, 0x1a143222, 0x694a2342, 0x011f3f4f, 0x262e1248, 0x1708260e, 0x3b272537, 0x870f252f, 0x161b1d0b, 0x350d0605, 0x3304e51f, 0x314329a0, + 0x9105071b, 0x592a0a07, 0x36fc5d87, 0x03073936, 0x110e0c85, 0x012e4028, 0x615e21a2, 0xaa4d1957, 0x00c1fd4f, 0xae000100, 0x120414fe, 0x1d004a04, + 0x26403f00, 0x47090c0d, 0x101f550a, 0x601f201f, 0x801f701f, 0x0014051f, 0x541b471a, 0x141b1a1e, 0x1150030d, 0x1c150b16, 0x3f000f09, 0xe13f3f33, + 0x013f3333, 0x32e1f610, 0xf6105d32, 0x313232e1, 0x16140130, 0x023e3233, 0x11331135, 0x06232723, 0x26222306, 0x16171627, 0x23111516, 0x64013311, + 0x6e526f6a, 0x93b61c43, 0x90300a1b, 0x236a4867, 0x01020201, 0x8701b6b6, 0x65348282, 0x3a026094, 0x5393b6fb, 0x262a2e54, 0x2a552328, 0x3606c0fe, + 0x00000100, 0xcf030000, 0x14004a04, 0x16403b00, 0x07071314, 0x0e470d00, 0x16101657, 0x0116b001, 0x0601160f, 0xffb80001, 0x0700b6f0, 0x000d1514, + 0x323f000f, 0x2f01333f, 0x5d5e3238, 0xf610715d, 0x3d3912e1, 0x3133332f, 0x13331130, 0x3317031e, 0x3512023e, 0x02021433, 0xbc230706, 0x1e1f0cc9, + 0x5a060519, 0xb6194272, 0x759a5c25, 0xfd4a04c0, 0x676321b0, 0xcd60195c, 0x970501e7, 0xfee0fea3, 0x007ffdf5, 0x71000100, 0x6a036ffe, 0x46001406, + 0x4e407e00, 0x113b3b32, 0x4641012a, 0x242d2d38, 0x1001001c, 0x03012001, 0x1c010108, 0x01112046, 0x483f4811, 0x487f485f, 0x100448ef, 0x48070148, + 0x3c475624, 0x39503832, 0x1c49002a, 0x1c691c59, 0x011c3803, 0x031c2407, 0x004f0116, 0x00391600, 0x3f00fa16, 0x2f39123f, 0x391711e1, 0x39115d5d, + 0x3232e110, 0xe1f61001, 0xd6105d5d, 0x2fc0f15d, 0x12115d5e, 0xe1332f39, 0x33113912, 0x3031332f, 0x22231501, 0x1415020e, 0x1e17021e, 0x0e141503, + 0x3e230702, 0x2e343503, 0x26262702, 0x023e3435, 0x26263537, 0x023e3435, 0x06070637, 0x35232306, 0x22231521, 0x1415020e, 0x0333021e, 0x7f51a223, + 0x542d2d57, 0x5f474b78, 0x2215193a, 0x18aa162b, 0x0e14222b, 0xb24d5b30, 0x664e2fb8, 0x2d726537, 0x223e6b4f, 0x2e552127, 0x3781023e, 0x47739149, + 0x51724620, 0x31897703, 0x4e417155, 0x11223b60, 0x483b2e0f, 0x52582d2a, 0x48201f4b, 0x141d4348, 0x0f1f2225, 0x4db5c722, 0x1143637f, 0x75861c0c, + 0x31496648, 0x02030313, 0x268d9904, 0x354e754e, 0x001c3751, 0x7100ffff, 0x2d04ecff, 0x06025e04, 0x00005200, 0x19000100, 0xc904ecff, 0x18004a04, + 0x38405800, 0x168f470a, 0x0e161601, 0x00141403, 0x02031003, 0x2f1a0303, 0xef1a9f1a, 0x0d10031a, 0xef0e2f47, 0x030eff0e, 0x1510400e, 0x19540e48, + 0x50100c15, 0x150e0f12, 0x16075000, 0x3fe13f00, 0x3232e13f, 0x2bf61001, 0x5d33e15d, 0x5d2f3311, 0x12112f33, 0xe15d2f39, 0x32253031, 0x06153736, + 0x26222306, 0x11211135, 0x35231123, 0x23152137, 0x04161411, 0x0f2f1c50, 0x71304a0f, 0xb66dfe71, 0x270489dd, 0x0d812fd3, 0x11099008, 0xbe028284, + 0xb00350fc, 0xfd9a504a, 0x0037464e, 0xa4000200, 0x2d0414fe, 0x1a005e04, 0x39002700, 0x48252140, 0xcf295700, 0x29400129, 0x01290f01, 0x0f091f06, + 0x28541047, 0x1016501b, 0x50221b0f, 0x00160508, 0x3fe1333f, 0x1001e13f, 0x3232e1f6, 0x5d5d5d5e, 0x31e1f610, 0x0e140130, 0x26222302, 0x17162327, + 0x11151616, 0x3e341123, 0x1e323302, 0x06220102, 0x16161107, 0x35363233, 0x2d042634, 0x68a7753f, 0x06368f4b, 0x01020201, 0xa97841b6, 0x77a36169, + 0x893bfe43, 0x8f360382, 0x7d7b8c4b, 0xd5892702, 0x2d2d4c91, 0x61272e2b, 0x04ddfe2e, 0x91d38813, 0xd3914b4b, 0xc8c41801, 0x3133acfe, 0xcfd1d1d3, + 0x71000100, 0x6f036ffe, 0x31005e04, 0x36405100, 0x46152727, 0x100a0010, 0x030a200a, 0x333f330a, 0x337f335f, 0x100433ef, 0x48000133, 0x2832561f, + 0x0a37512d, 0x38152801, 0x03154815, 0x001f0a15, 0x10240f04, 0x3f00fa0f, 0x3917123f, 0x32e15d5d, 0xe1f61001, 0xd6105d5d, 0x33e1325d, 0x0130312f, + 0x17021e14, 0x1415031e, 0x2307020e, 0x3435033e, 0x2e27022e, 0x3e343503, 0x16323302, 0x032e0717, 0x020e2223, 0x491c2d01, 0x5f47617e, 0x2215193a, + 0x18aa162b, 0x0e14222b, 0x4d4d5b30, 0x49376284, 0x4e6ab383, 0x17363295, 0x1a3a3c38, 0x22497250, 0x79600602, 0x0f162f4e, 0x2a483b2e, 0x4b52582d, + 0x4848201f, 0x25141d43, 0x0f0f1f22, 0x82b47641, 0x4995e29a, 0x0a9a1922, 0x39090f13, 0x0200a470, 0xecff7100, 0x4a048504, 0x29001600, 0x21403600, + 0x12482124, 0x2b570010, 0x2b902b10, 0x2bb02ba0, 0x17052be0, 0x2a560a48, 0x0f502512, 0x05501c0f, 0xe13f0016, 0x0132e13f, 0x5de1f610, 0x3232f610, + 0x303132e1, 0x020e1401, 0x022e2223, 0x023e3435, 0x21152133, 0x1405031e, 0x3233021e, 0x3435023e, 0x22232726, 0x2d04020e, 0x75b47a3e, 0x447daf6b, + 0x80ce904d, 0xf6fee901, 0x1c304125, 0x472100fd, 0x6d4c4d6e, 0x4b4f2147, 0x5a8b5f3b, 0x6ff8012b, 0x46508dc0, 0x9c85cb8a, 0x9a3e8ada, 0x7c6c5d29, + 0x69905536, 0x8662373b, 0x59d7904e, 0x00a1632c, 0x12000100, 0x6603e5ff, 0x1c004a04, 0x24403a00, 0x1f0d0f01, 0x0d06020d, 0x1e101e57, 0x3b1a2b01, + 0x031a021a, 0xe0181047, 0x02180218, 0x0f1c5019, 0x16135008, 0x3fe13f00, 0x2f0132e1, 0x5dc6e15d, 0x5ee6105d, 0x3031325d, 0x11211501, 0x33021e14, + 0x37023e32, 0x23030e15, 0x35022e22, 0x37352111, 0x8cfe6603, 0x243e2d19, 0x24282a12, 0x2e270d0d, 0x75401b35, 0xd7fe3459, 0x9a4a0486, 0x523ba2fd, + 0x06031733, 0x06870407, 0x1d060b0c, 0x0268864d, 0x00504a73, 0xa4000100, 0x3d04ecff, 0x1d004a04, 0x18402900, 0x571b4710, 0x601f101f, 0x031f801f, + 0x54054708, 0x0f06151e, 0x1600500d, 0x3fe13f00, 0xf6100133, 0xf6105de1, 0x053031e1, 0x35022e22, 0x14113311, 0x3233021e, 0x2e343536, 0x1e332702, + 0x02101503, 0xaa815c02, 0x20b62964, 0x92466642, 0x1710098d, 0x180fb60f, 0x14f40810, 0x6dbd8c50, 0xb2fd5802, 0x36638c56, 0x7d47f7e8, 0x3d3d7475, + 0x4b817772, 0xd3fec1fe, 0x71000200, 0x250514fe, 0x25005e04, 0x5e003500, 0x25313c40, 0x00961647, 0x000200a6, 0x48260600, 0x2037571f, 0x370f0137, + 0x0137e001, 0x377f376f, 0x4003379f, 0x370f0137, 0x0602371f, 0x56064711, 0x0b502b36, 0x1631101a, 0x16012450, 0x3f001b00, 0x32e1333f, 0x01e1333f, + 0x5ee1f610, 0x5d5d5d5d, 0xf6107171, 0x2f3911e1, 0x32e1335d, 0x11013031, 0x3435032e, 0x1737023e, 0x1415030e, 0x1117021e, 0x32333634, 0x1415021e, + 0x1107020e, 0x022e3401, 0x020e2223, 0x033e1115, 0xb76d6602, 0x38204b86, 0x258d2d4e, 0x33192d3f, 0xa5407458, 0x66925a91, 0xbe905637, 0x1e4f0168, + 0x1b2b4936, 0x48162531, 0xfe33597b, 0x05da0114, 0x97d58943, 0x7d8a9853, 0x68336037, 0x6f477d72, 0x07306199, 0xc1bc5e02, 0x7ac99050, 0x4a90d995, + 0x0426fe05, 0x65946225, 0x57351733, 0x07a0fd41, 0x009f6b3b, 0xecff0100, 0x250414fe, 0x28004e04, 0x1a40a000, 0x1f781f68, 0x09671f02, 0x09c70977, + 0x76060903, 0x1c37011c, 0x1e1c1c01, 0xffb80807, 0x130e40c0, 0x08344816, 0x08100801, 0x0112c008, 0xc0ffb812, 0x0b061940, 0x2a121248, 0x2a3f2a0f, + 0x2acf2a4f, 0x25b00604, 0x250225c0, 0xb81e1d25, 0x1f40f0ff, 0x1e291e1e, 0x2706171b, 0x1c180206, 0x1c021c28, 0x04061f09, 0x16500f00, 0x230f071b, + 0x000f0050, 0x3f3fe13f, 0x391712e1, 0x113f5d5d, 0x382f3301, 0x5d2f3333, 0x33115d5e, 0x335d2b2f, 0x2b5d382f, 0x3d391233, 0x335d5d2f, 0x5d335d33, + 0x32133031, 0x1317021e, 0x13013301, 0x3233031e, 0x06153736, 0x2e222306, 0x01032702, 0x26030123, 0x07222326, 0xbc363635, 0x313a492d, 0x1f017b15, + 0xb273feb2, 0x34261d0e, 0x102e1a25, 0x43283916, 0x1733475f, 0xc2b6fe83, 0x1ba0c601, 0x1c243547, 0x4e043e16, 0x3d5c3e1f, 0x4a02a8fe, 0x20fef8fc, + 0x1d334226, 0x068d0305, 0x6646260b, 0xfd6a0141, 0x013e0383, 0x0a594fbe, 0x000a078f, 0xa4000100, 0x710514fe, 0x27001206, 0x2f404c00, 0x26471701, + 0x061e1818, 0x29571147, 0x29402930, 0x29b02990, 0x0f0529c0, 0x02291f29, 0x1e472106, 0x1f0b2854, 0x5026010f, 0x17161916, 0x0000001b, 0x333f3f3f, + 0x333f32e1, 0xe1f61001, 0x105d5d5e, 0x3911e1f6, 0x32e1332f, 0x11013031, 0x3435033e, 0x3327022e, 0x1415031e, 0x1107020e, 0x032e1123, 0x11331135, + 0x17021e14, 0x4e560311, 0x08345f83, 0xb610180f, 0x08101810, 0x6dc49357, 0x89bc6fb2, 0x5a36b64c, 0x12064278, 0x3c0973fa, 0x47679c6c, 0x49837e80, + 0x7e7e8348, 0x8cdc9d44, 0x26fe0745, 0x4104da01, 0x029ad688, 0x72d9fd1f, 0x052e619a, 0x01008f05, 0xecff7100, 0x4a049605, 0x70003b00, 0x67384840, + 0x471d011d, 0xb7011ad5, 0x1aa9011a, 0x011a9a01, 0x051a1a03, 0x57304825, 0x143d043d, 0x3dc4023d, 0x7b023df4, 0x023d9b3d, 0x02013d40, 0x0f013d30, + 0x023d1f3d, 0x05481006, 0x1b1b3c56, 0x200f0a2a, 0x35395015, 0x3f001600, 0x32e13232, 0x2f39333f, 0xe1f61001, 0x5f5d5d5e, 0x715d5d5d, 0x11e1f610, + 0x5d5f2f39, 0xe15d5d5d, 0x3031395d, 0x022e2205, 0x023e3435, 0x030e3337, 0x021e1415, 0x023e3233, 0x11331135, 0x32331614, 0x3435023e, 0x3327022e, + 0x1415031e, 0x2223020e, 0x06232726, 0x5ae90106, 0x0f32608c, 0xba202f1f, 0x0f1f2f20, 0x2d4b361d, 0x162c442e, 0x2d5163b2, 0x0f1d364b, 0xbb202f1f, + 0x0f1f2f20, 0x5a8d6032, 0x0a1f8b6b, 0x52148b1f, 0x5179cb92, 0x4887878f, 0x8e878649, 0x65966252, 0x62482934, 0xfe320139, 0x348b81ce, 0x52629665, + 0x4986878e, 0x8f878748, 0x92cb7951, 0x5b5b5752, 0xffff0057, 0xecff1500, 0xd9057702, 0x86012602, 0x07010000, 0xe2fe6a00, 0x17000000, 0x01020d40, + 0x0226111f, 0x2a163201, 0x01250014, 0x0035352b, 0x0035352b, 0xa400ffff, 0x3d04ecff, 0x2602d905, 0x00009201, 0x6a000601, 0x19000000, 0x270102b6, + 0x01022611, 0xb4deffb8, 0x1b05321e, 0x352b0125, 0x352b0035, 0xffff0035, 0xecff7100, 0x21062d04, 0x52002602, 0x06010000, 0x000e5401, 0x0b401300, + 0x26112c02, 0x26203d02, 0x0125000a, 0x2b00352b, 0xffff0035, 0xecffa400, 0x21063d04, 0x92012602, 0x06010000, 0x00105401, 0x0b401300, 0x26112a01, + 0x241e1d01, 0x01251b05, 0x2b00352b, 0xffff0035, 0xecff7100, 0x21069605, 0x96012602, 0x07010000, 0xc3005401, 0x21000000, 0x48010940, 0x60012611, + 0xb83c013c, 0x0940c0ff, 0x3d480b09, 0x3005423c, 0x2b2b0125, 0x2b00355d, 0xffff0035, 0x0000c700, 0x2b07be03, 0x28002602, 0x07010000, 0xf5ff6a00, + 0x17005201, 0x01020d40, 0x02260515, 0x200c0101, 0x01250001, 0x0035352b, 0x0035352b, 0x14000100, 0x0605ecff, 0x2500b605, 0x5a408500, 0x0c130404, + 0x01210f5a, 0x21ff21df, 0x27552102, 0x276f270f, 0x0603277f, 0x0118f41a, 0x18b018a0, 0x01187402, 0x5a121818, 0x30150013, 0x50154015, 0x0515c015, + 0x13b01515, 0x77136701, 0x135e0213, 0x4f133f01, 0x13000213, 0x5f111301, 0x19071a1a, 0x03165f15, 0x5f071213, 0x3f001300, 0xe13f3fe1, 0x2f391232, + 0x5d2f01e1, 0x5d5d5d5d, 0x105d2f33, 0x5d2f32e1, 0x5e325d5d, 0x5df6105d, 0x3912e171, 0x0530312f, 0x35272622, 0x32331616, 0x3535023e, 0x21232634, + 0x21112311, 0x21152135, 0x1e322111, 0x14151502, 0x9e03020e, 0x1d19462f, 0x40202744, 0x7b692133, 0xfebbbbfe, 0xfeb103ac, 0x5d5a015e, 0x35356592, + 0x0d14845f, 0x0c09a00b, 0x45573313, 0xfd747385, 0xa4120523, 0x316ffea4, 0x89598b5e, 0x31659a69, 0xc700ffff, 0xbe030000, 0x26027307, 0x00006101, + 0x76000701, 0x52013300, 0x0b401300, 0x26051201, 0x0c063e01, 0x01250004, 0x2b00352b, 0x01000035, 0xecff7d00, 0xcb059804, 0x5f002600, 0x40111340, + 0x07481813, 0x1b110711, 0x700123ca, 0x02238023, 0xc0ffb823, 0x0a072440, 0x28232348, 0x050128bf, 0x661b5b08, 0x0f5f0827, 0x0205af05, 0x0d050508, + 0x205f0024, 0x105f0d04, 0x3f001316, 0xe13fe133, 0x2f391233, 0x01e15d5e, 0x32e1f610, 0x2f33115d, 0x125d5d2b, 0x2f2f3939, 0x0130312b, 0x07020e22, + 0x1e211521, 0x36323303, 0x030e1537, 0x26262223, 0x12343502, 0x32333636, 0x26071716, 0x5f190326, 0x0d4d78a0, 0x86fd7402, 0xab764005, 0x4ea05971, + 0x61554e27, 0x9df0a43b, 0xfaa9574c, 0x4fc46ca2, 0x05943f4e, 0xaf7a4127, 0xc982a26f, 0x1723488a, 0x0e170fa2, 0x01c66c07, 0x01a6a916, 0x2c6ec614, + 0x2e209c2a, 0x6800ffff, 0xc903ecff, 0x0602cb05, 0x00003600, 0x5200ffff, 0x64020000, 0x0602b605, 0x00002c00, 0x4000ffff, 0x77020000, 0x26022b07, + 0x00002c00, 0x6a000701, 0x52010dff, 0x0d401700, 0x05150102, 0x00010226, 0x0001200c, 0x352b0125, 0x352b0035, 0xffff0035, 0x7bfe48ff, 0xb6057301, + 0x2d000602, 0x02000000, 0xe9ff0000, 0xb605df06, 0x35002a00, 0x25b58d00, 0x0d065a2b, 0xf8ffb808, 0x17145340, 0x235a0848, 0x1e011e99, 0x00231e10, + 0x02069006, 0x0123e006, 0x23c423b4, 0x000323d4, 0x90235023, 0x0423a023, 0x06230607, 0x5b2f1523, 0xcf376700, 0x40370137, 0x15480906, 0x5f353615, + 0x25af250f, 0x25250802, 0x235f082b, 0x12601903, 0x06602b13, 0xe13f0012, 0xe13fe13f, 0x5e2f3912, 0x0111e15d, 0x5d2b2f33, 0x11e1f610, 0x2f2f3939, + 0x5d5d5d5e, 0x33115d5e, 0xe1105d38, 0xe110322b, 0x01303132, 0x23020e14, 0x0e211121, 0x030e0703, 0x27262223, 0x33161635, 0x37023e32, 0x12123636, + 0x33112137, 0x01021e32, 0x35363233, 0x23022e34, 0x3cdf0623, 0xfe86c27e, 0x0fc2febf, 0x1022211f, 0x6e4f3514, 0x1c4a234e, 0x30203a17, 0x0c1a2a3e, + 0x2b27200c, 0x6d930216, 0x377ecf98, 0xae7777fd, 0x8b5b2ea4, 0xac01585d, 0x3d709e61, 0xf4711205, 0x6751d4ec, 0x0e3b6fa2, 0x100d9a0b, 0x3675623f, + 0x0601c839, 0xfdaa3a01, 0x98714298, 0x88849dfe, 0x1b3c6146, 0xc7000200, 0x0c070000, 0x1600b605, 0x53002100, 0x17113240, 0x06060e5a, 0x005b1b0a, + 0x230f2367, 0x1f230f01, 0xff237f23, 0x090d0423, 0x22640a5a, 0x115f0821, 0x0daf0d0f, 0x0d0d0802, 0x17030b0f, 0x12060a60, 0xe1333f00, 0x2f39333f, + 0xe1335d5e, 0xf6100132, 0x715d32e1, 0x11e1f610, 0xe1332f39, 0x01303132, 0x23020e14, 0x11211121, 0x11331123, 0x11331121, 0x021e3233, 0x36323301, + 0x022e3435, 0x0c072323, 0x86c27e3c, 0xb8fdbffe, 0x4802baba, 0xcf986dba, 0x77fd377e, 0x2ea4ae77, 0x585d8b5b, 0x9e61ac01, 0xaa023d70, 0xb60556fd, + 0x680298fd, 0x714298fd, 0x849dfe98, 0x3c614688, 0x0100001b, 0x00001400, 0xb6050605, 0x73001500, 0x5a084d40, 0x0f175507, 0x7f176f17, 0xf4060317, + 0x14a00114, 0x740214b0, 0x14140114, 0x0f5a0e00, 0x11301100, 0x11501140, 0x110511c0, 0x010fb011, 0x0f770f67, 0x010f5e02, 0x0f4f0f3f, 0x010f0002, + 0x5f11150f, 0x005f0d12, 0x03120700, 0x0012070f, 0x123f333f, 0x10e12f39, 0x2f0132e1, 0x5d5d5d5d, 0x5d2f335d, 0x3232e110, 0x5d5d5d2f, 0xf6105d5e, + 0x013031e1, 0x021e3221, 0x11231115, 0x21232634, 0x21112311, 0x21152135, 0x5a012302, 0x3565925d, 0xfe7b69ba, 0xacfebbbb, 0x5efeb103, 0x5e318103, + 0xf2fd598b, 0x7473f601, 0x120523fd, 0xffffa4a4, 0x0000c700, 0x7307a204, 0xb4012602, 0x07010000, 0x89007600, 0x13005201, 0x17010b40, 0x22012605, + 0x0004110b, 0x352b0125, 0x00352b00, 0x1900ffff, 0xae04ecff, 0x26026907, 0x0000bd01, 0x36020701, 0x52012300, 0x0b401300, 0x26052701, 0x222c0d01, + 0x01250013, 0x2b00352b, 0x01000035, 0x7ffec700, 0xb605c104, 0x63000b00, 0x02c64140, 0x020202d6, 0xd503c55a, 0x03a80203, 0x01039601, 0x110e1003, + 0x01034748, 0x0301033a, 0x5a090503, 0xb00d6500, 0x0d6f010d, 0x0daf0d9f, 0x010d1003, 0x64055a08, 0x03060a0c, 0x02055f08, 0x00050102, 0x2f33332f, + 0x333fe110, 0xe1f61001, 0x105d5d5d, 0x3911e1f6, 0x2b5d5d2f, 0xe15d5d5d, 0x2130315d, 0x11231121, 0x11331121, 0x04331121, 0xb15afec1, 0x02ba5dfe, + 0x7ffebb85, 0xb6058101, 0x1005f0fa, 0x0000ffff, 0xdd040000, 0x0602bc05, 0x00002400, 0xc7000200, 0x33040000, 0x1000b605, 0x4b001b00, 0x085f2e40, + 0x11080801, 0x67005b15, 0x5f1d0f1d, 0xcf1d7f1d, 0x0b06041d, 0x64065a11, 0x0f5f1b1c, 0x020baf0b, 0x110b0b08, 0x03075f0a, 0x12066011, 0x3fe13f00, + 0x2f3912e1, 0x01e15d5e, 0x32e1f610, 0xf6105d5e, 0x2f3912e1, 0x0130315d, 0x23020e14, 0x15211121, 0x32331121, 0x3301021e, 0x34353632, 0x2323022e, + 0x7e3c3304, 0x96fe86c2, 0xa6fd1403, 0x7ecf9896, 0xa04efd37, 0x5b2ea4ae, 0x01815d8b, 0x709e61ac, 0xa4b6053d, 0x71423cfe, 0x849dfe98, 0x3c614688, + 0xffff001b, 0x0000c700, 0xb6058704, 0x25000602, 0xffff0000, 0x0000c700, 0xb605be03, 0x61010602, 0x02000000, 0x7ffe0e00, 0xb6050a05, 0x1b001100, + 0x0b408b00, 0x1b281b18, 0x37071b02, 0xb8140114, 0x0b40f8ff, 0x14481714, 0x070f0e5a, 0xb8070701, 0x1040f0ff, 0x0ec00e30, 0x070e0702, 0x5a12050e, + 0x005a0110, 0x40c0ffb8, 0x48151025, 0x65100000, 0x011d201d, 0x08011d0f, 0x05055a04, 0x0e5f141c, 0x08510903, 0x5f06101b, 0x05050003, 0x3f001203, + 0x10332f33, 0x2f3232e1, 0x11e13fe2, 0xe12f3301, 0x105d5d5e, 0x2b2f32e4, 0x11e110e1, 0x2f2f3939, 0x5d5e385d, 0x5d2be110, 0x315d3311, 0x11230130, + 0x11231121, 0x37053e33, 0x21331121, 0x050e2111, 0xb00a0507, 0x71b064fc, 0x414d562f, 0x02041d30, 0x83fec265, 0x1f04fafe, 0x4d463d2e, 0x017ffe27, + 0x027ffe81, 0xd9c85527, 0x69e3e8e6, 0x6a04f0fa, 0xd1cab94c, 0x004bb7c8, 0xc700ffff, 0xbe030000, 0x0602b605, 0x00002800, 0x04000100, 0x81060000, + 0x1100b605, 0x4e40cb00, 0x09760966, 0x37030986, 0x57094709, 0x06090309, 0xf70d4706, 0x5a0d020d, 0x00790069, 0x38030089, 0x58004800, 0x03000300, + 0xb80ea803, 0x0e99020e, 0x010e8601, 0x0e670e57, 0x0e030e77, 0x070a110e, 0x000b0808, 0x0a70010a, 0x0ac00a80, 0xffb80a03, 0x072640c0, 0x100a480a, + 0xf0130a0a, 0x13bf0113, 0x13df13cf, 0x0113a003, 0x4001138f, 0x130f0113, 0x0802131f, 0x10010102, 0xf0ffb811, 0x11110f40, 0x03030612, 0x0e0a1104, + 0x04010712, 0x333f0003, 0x33333f33, 0x33113912, 0x2f330111, 0x11333338, 0x5d5d5e33, 0x5d5d5d5d, 0x382f3311, 0x33715d2b, 0x11331133, 0x5d2f3912, + 0x335d5d5d, 0x5d5d3311, 0x11325de1, 0x315d5d33, 0x33010130, 0x11331101, 0x01013301, 0x23110123, 0x02230111, 0xcdedfd25, 0x02b30a02, 0xedfdcd0a, + 0xfdd32102, 0xeefdb3ee, 0x02f202d3, 0x023cfdc4, 0x023cfdc4, 0xfd3cfdc4, 0xfde5020e, 0xfde5021b, 0x0100001b, 0xecff4800, 0xcb05ec03, 0x68003900, + 0x5b273f40, 0x00210500, 0x00002130, 0x13033021, 0x670b5b1c, 0x013b9f3b, 0x13801370, 0x4f133f02, 0x03135f13, 0x053a1313, 0x21aa6020, 0x01217801, + 0x0801210f, 0x2f192121, 0x0435602c, 0x10146019, 0x333f0013, 0x33e13fe1, 0x5e2f3912, 0xe15d5d5d, 0x33011139, 0x5d5d5d2f, 0x12e1f610, 0x2f2f3917, + 0x3912112f, 0x3031e110, 0x020e1401, 0x031e1507, 0x020e1415, 0x27262223, 0x33031e35, 0x34353632, 0x35232326, 0x023e3233, 0x022e3435, 0x07062223, + 0x33033e27, 0x03021e32, 0x7d5b33d1, 0x5e8a574b, 0xcd884332, 0x55c06e89, 0x6363602b, 0xcfb0b22e, 0x5cb0bfba, 0x2532618e, 0x6e3a5f44, 0x265c4ba9, + 0x47837462, 0x3971a66d, 0x78496004, 0x060c3958, 0x7759390b, 0x74a06048, 0xaa2d2240, 0x0d182417, 0x81878794, 0x65482797, 0x3a53363d, 0x7d36431e, + 0x1829361f, 0x00856136, 0xc9000100, 0x10050000, 0x1700b605, 0x1b406b00, 0x5a0e0b14, 0xd019650d, 0x19af0119, 0x30192001, 0x09160219, 0x64175a02, + 0xb8091718, 0x14b3e8ff, 0xb8094818, 0x2340e0ff, 0x0648130a, 0x26091609, 0x0d090309, 0x14181412, 0x20144818, 0x0948130a, 0x29141914, 0x14070314, + 0x0003000b, 0x5e32323f, 0x3f2b2b5d, 0x2b2b5d33, 0xf6100133, 0x5d3232e1, 0xf6105d5d, 0x313232e1, 0x11331330, 0x07020e14, 0x01330706, 0x11231133, + 0x36373634, 0x23012337, 0x0201aec9, 0x04040202, 0xd5cc0207, 0x040406ac, 0x31fd0805, 0xfcb605d7, 0x433e1ae0, 0x4c4a2043, 0x4afab404, 0x8e391903, + 0xfb4d4c41, 0xffff0046, 0x0000c900, 0x69071005, 0xb2012602, 0x07010000, 0x89003602, 0x15005201, 0x051d01b4, 0xffb80126, 0x1822b4eb, 0x01250c00, + 0x2b00352b, 0x01000035, 0x0000c700, 0xb605a204, 0x5e000a00, 0x09082140, 0x01091009, 0x00100000, 0x10000702, 0xb00c0000, 0x0c2f010c, 0x010c1001, + 0x56010a97, 0xb80a010a, 0x1440f8ff, 0x0a480c08, 0x5a030707, 0x070b6404, 0x05080402, 0x12000403, 0x3f323f00, 0x39391233, 0xe1f61001, 0x2b331132, + 0x5d5d5d5d, 0x2f33115d, 0x335d5e38, 0x33113833, 0x23213031, 0x11231101, 0x33011133, 0xdba20401, 0xbababafd, 0xfdcf3502, 0xfde502cb, 0xfdb6051b, + 0xfdc4023c, 0x01000042, 0xe9ff0000, 0xb6059304, 0x62001f00, 0xb80308b1, 0x3b40f8ff, 0x03481714, 0x0119895a, 0xe0191019, 0x1eb4011e, 0x1ed41ec4, + 0x501e0003, 0x031e801e, 0x101e1e07, 0x65005a01, 0x1f210f21, 0x03217f21, 0x20101008, 0x031e5f03, 0x0d116014, 0x00120013, 0xe1333f3f, 0x0111e13f, + 0x5d5e2f33, 0x11e1f610, 0x5d5e2f39, 0x38335d5d, 0x322be15d, 0x23213031, 0x030e2111, 0x23030e07, 0x35272622, 0x32331616, 0x3637023e, 0x37121236, + 0xba930421, 0x1f0f85fe, 0x14102221, 0x4e6e4f35, 0x171c4a23, 0x3e30203a, 0x0c0c1a2a, 0x162b2720, 0x1205d002, 0xd4ecf471, 0x6fa26751, 0x9a0b0e3b, + 0x623f100d, 0xc8393675, 0x3a010601, 0xffff00aa, 0x0000c700, 0xb6052f06, 0x30000602, 0xffff0000, 0x0000c700, 0xb605d504, 0x2b000602, 0xffff0000, + 0xecff7d00, 0xcd057105, 0x32000602, 0xffff0000, 0x0000c700, 0xb605c104, 0x6e010602, 0xffff0000, 0x0000c700, 0xb6053304, 0x33000602, 0xffff0000, + 0xecff7d00, 0xcb059804, 0x26000602, 0xffff0000, 0x00001400, 0xb6051204, 0x37000602, 0x01000000, 0xecff1900, 0xb605ae04, 0x6f002100, 0x01121040, + 0x21131b1b, 0x00700060, 0x00f000b0, 0xffb80004, 0x061040c0, 0x1000480a, 0x0f230000, 0x02237f23, 0xb8130914, 0x0c40f0ff, 0xe7221313, 0x021af71a, + 0x1a011ad6, 0x40e0ffb8, 0x480e0a10, 0x031a0112, 0x0a5f0d00, 0x00131306, 0x323f0003, 0x11e1333f, 0x5d2b3917, 0x3301115d, 0x3333382f, 0x2f33115d, + 0x335d2b38, 0x33113912, 0x01303133, 0x23030e01, 0x35272622, 0x32331616, 0x0137023e, 0x031e0133, 0x033e3317, 0xae040137, 0x532654fe, 0x336f9c70, + 0x5925255a, 0x42523534, 0xeefd1c38, 0x057f01cc, 0x0409090a, 0x0b0b0302, 0x3701030b, 0xfafbb605, 0x487ba45d, 0x14b90f0f, 0x56351719, 0xfc3f043f, + 0x1a180ad7, 0x1f090917, 0x03071d21, 0xffff0018, 0xecff6800, 0xcb05ba05, 0x73010602, 0xffff0000, 0x00000000, 0xb6056004, 0x3b000602, 0x01000000, + 0x7ffec700, 0xb6057105, 0x3b000b00, 0x5a032140, 0x5a090202, 0x0db00d00, 0x010daf01, 0x08010d10, 0x0c64055a, 0x0003060a, 0x02055f08, 0x00120502, + 0x102f333f, 0x333f32e1, 0xe1f61001, 0x105d5d5d, 0x2f33e1d4, 0x253031e1, 0x11231133, 0x11331121, 0x04331121, 0xfcb0b0c1, 0x8502ba06, 0xd9fda6bb, + 0xb6058101, 0x1005f0fa, 0xa6000100, 0x81040000, 0x1500b605, 0x1d404800, 0x005a0113, 0x17701765, 0x2f0217a0, 0x17100117, 0x605a0d01, 0xa00a700a, + 0x040ab00a, 0xc0ffb80a, 0x0a070e40, 0x10130a48, 0x1405055f, 0x1200030b, 0x333f3f00, 0x33e12f39, 0x5d2b2f01, 0x5d5d5de1, 0x32e1f610, 0x23213031, + 0x23060611, 0x35022e22, 0x14113311, 0x36323316, 0x04331137, 0xc373ba81, 0x65925d62, 0x7b69ba35, 0xba70b95a, 0x2e2c5602, 0x598a5f31, 0xd1fd4702, + 0x28287473, 0x0100c602, 0x0000c700, 0xb6053307, 0x60000b00, 0x08564240, 0x175a0801, 0x05860105, 0x05a60596, 0x77056703, 0x05050205, 0x005a0901, + 0x0d000d65, 0x20020d10, 0x700d500d, 0xc00d800d, 0xe00dd00d, 0x0d0f070d, 0x5a040701, 0x0a0c6401, 0x08030206, 0x12015f04, 0x32e13f00, 0x0133333f, + 0x5ee1f610, 0x10715d5d, 0x3911e1f6, 0x715d5d2f, 0x30315de1, 0x33112121, 0x33112111, 0x33112111, 0x94f93307, 0xba1f02ba, 0x05ba1f02, 0x05f0fab6, + 0x05f0fa10, 0x01000010, 0x7ffec700, 0xb605e307, 0x71000f00, 0x0c564b40, 0x175a0c01, 0x09860109, 0x09a60996, 0x77096703, 0x09090209, 0x005a0d05, + 0x02025a03, 0x00116500, 0x02111011, 0x11501120, 0x11801170, 0x11d011c0, 0x0f0711e0, 0x08070111, 0x1064055a, 0x03060a0e, 0x5f08000c, 0x05020205, + 0x333f0012, 0x32e1102f, 0x33333f32, 0xe1f61001, 0x715d5d5e, 0x2f32e410, 0x11e110e1, 0x5d5d2f39, 0x315de171, 0x11332530, 0x11211123, 0x11211133, + 0x11211133, 0xb0330733, 0xba94f9b0, 0x02ba1f02, 0xfda6ba1f, 0x058101d9, 0x05f0fab6, 0x05f0fa10, 0x02000010, 0x00001400, 0xb605d504, 0x1b001000, + 0x33405200, 0x905a110b, 0x06060106, 0x005b1508, 0x1d0f1d67, 0x2f1d0f01, 0x5f1d3f1d, 0x051daf1d, 0x1c080806, 0x0b0f5f1b, 0x08020baf, 0x08110b0b, + 0x1103095f, 0x00120660, 0xe13fe13f, 0x5e2f3912, 0x0111e15d, 0x5d5e2f33, 0xe1f61071, 0x5d2f3911, 0x303132e1, 0x020e1401, 0x21112123, 0x33112135, + 0x01021e32, 0x35363233, 0x23022e34, 0x3cd50423, 0xfe86c27e, 0x02acfe95, 0xcf98950f, 0x4efd377f, 0x2da3aea0, 0x815d8a5c, 0x9e61ac01, 0x12053d70, + 0x4298fda4, 0x9dfe9871, 0x61468884, 0x03001b3c, 0x0000c700, 0xb605cf05, 0x19000e00, 0x49001d00, 0x5b132b40, 0x00010000, 0x5a1a0600, 0x0f1f651d, + 0x021f9f1f, 0x5a0f0906, 0x191e6406, 0xaf090f5f, 0x09080209, 0x03071b09, 0x061a600f, 0x333f0012, 0x39333fe1, 0xe15d5e2f, 0xe1f61001, 0x105d5e32, + 0x3911e1f6, 0x31e15d2f, 0x0e140130, 0x11212302, 0x32331133, 0x3301021e, 0x34353632, 0x2323022e, 0x11331101, 0x7e3c3304, 0x96fe86c2, 0xcf9896ba, + 0x4efd377e, 0x2ea4aea0, 0x815d8b5b, 0x01bb9303, 0x709e61ac, 0xfdb6053d, 0x98714298, 0x88849dfe, 0x1b3c6146, 0xb60556fd, 0x02004afa, 0x0000c700, + 0xb6053304, 0x19000e00, 0x2e404800, 0x00905b13, 0x1b670001, 0x1b5f1b0f, 0x401b0602, 0x1b481815, 0x48130d40, 0x065a0f09, 0x5f191a64, 0x09af090f, + 0x09090802, 0x600f0307, 0x3f001206, 0x2f393fe1, 0x01e15d5e, 0x32e1f610, 0x5d5e2b2b, 0xe15df610, 0x14013031, 0x2123020e, 0x33113311, 0x01021e32, + 0x35363233, 0x23022e34, 0x3c330423, 0xfe86c27e, 0x9896ba96, 0xfd377ecf, 0xa4aea04e, 0x5d8b5b2e, 0x61ac0181, 0x053d709e, 0x4298fdb6, 0x9dfe9871, + 0x61468884, 0x01001b3c, 0xecff3b00, 0xcb054404, 0x51002400, 0x201d2f40, 0x1e160c5b, 0x0c041e16, 0x26bf2667, 0x26df26cf, 0x25040403, 0x1faa5f1e, + 0x011f7801, 0x08011f0f, 0x1a071f1f, 0x1311175f, 0x075f0003, 0xe13f0004, 0xe1333f33, 0x5e2f3911, 0xe15d5d5d, 0x2f330111, 0x11e6105d, 0x2f2f3939, + 0x3132e110, 0x06220130, 0x36362707, 0x16163233, 0x02141512, 0x22230606, 0x3527022e, 0x32331616, 0x35211312, 0x01032e21, 0x3f945eba, 0x6cc44f4e, + 0x52a2f4a2, 0xa4f7a552, 0x4e56613a, 0x59a04e27, 0xfd06f7ed, 0x0971028b, 0x059d7347, 0x9c202e27, 0xc16d2c2a, 0xb39df6fe, 0x6dcae0fe, 0x0f170e07, + 0x012317a2, 0xa2060117, 0x407ab06f, 0xc7000200, 0x9e07ecff, 0x1a00cd05, 0x5e002e00, 0x5b1b3c40, 0x0111900a, 0x250d1111, 0x3067005b, 0x3f01300f, + 0x7f306f30, 0xff30ef30, 0x40300530, 0x10481411, 0x640d5a0c, 0x165f2a2f, 0x0f5f0b04, 0x10080110, 0x030e0d10, 0x5f20120d, 0x3f001305, 0x123f3fe1, + 0x5d5e2f39, 0x01e13fe1, 0x32e1f610, 0x10715d2b, 0x3911e1f6, 0xe1335d2f, 0x14013031, 0x23060602, 0x02262622, 0x23112127, 0x21113311, 0x3233033e, + 0x05121616, 0x33021e14, 0x35023e32, 0x23022e34, 0x07020e22, 0xe59b4f9e, 0x98e19796, 0xaefe0650, 0x5601baba, 0xdb98550b, 0x9ae59591, 0x31fafb4f, + 0x6d6c9d67, 0x3030669d, 0x6d6c9d65, 0x0231679e, 0xeafea9dd, 0xb7646cc6, 0xfda00301, 0xfdb60556, 0xa7ec9298, 0xfec56b5a, 0xdb89abeb, 0x99515199, + 0xda8a89db, 0x97515197, 0x020000da, 0x00001900, 0xb605f203, 0x1c000f00, 0x09409b00, 0xe6010006, 0x0200f600, 0xf8ffb800, 0x0f0b2440, 0x5a0e1c48, + 0x08000003, 0x3f1e650d, 0x9f1e8f1e, 0xdf1ebf1e, 0x1e20051e, 0x01010601, 0x01f601e6, 0xffb80102, 0x0b0d40f8, 0x4b01480f, 0x40020102, 0x02480b08, + 0x40f0ffb8, 0x16020226, 0xef08df5b, 0x0308ff08, 0x130f4008, 0x17080748, 0x08080208, 0x601c1d66, 0x010f0f03, 0x030b6011, 0x0012010e, 0xe13f333f, + 0x332f3912, 0xf61001e1, 0x5d2b5d5e, 0x382f33e1, 0x2b335d2b, 0x5d5d715d, 0x3911f610, 0x32e13311, 0x31715d2b, 0x23010130, 0x35032e01, 0x21333634, + 0x11112311, 0x020e2223, 0x021e1415, 0x42023333, 0x01dbb2fe, 0x4b63377d, 0x01fbfd2c, 0x4abcbb75, 0x27294f73, 0xb451774e, 0xa4fd5c02, 0x3e148f02, + 0xc65d8861, 0x024afac9, 0x1dbb025c, 0x4244613e, 0x00284a69, 0x5e00ffff, 0x9c03ecff, 0x06025e04, 0x00004400, 0x75000200, 0x2104ecff, 0x27002306, + 0x49003b00, 0x482d2c40, 0x3d571e09, 0x7f013dd0, 0x3d40013d, 0x013d0f01, 0x48371306, 0x323c5600, 0x19191350, 0x23502808, 0x0b080916, 0x51094810, + 0x3f000108, 0xe13f2be1, 0x332f3911, 0xf61001e1, 0x5d5e32e1, 0x105d5d5d, 0x31e132f6, 0x12341330, 0x36373636, 0x0e173736, 0x030e0703, 0x033e3307, + 0x021e3233, 0x020e1415, 0x022e2223, 0x023e3201, 0x022e3435, 0x020e2223, 0x021e1407, 0xa2683175, 0x66f27d71, 0x7d7b3221, 0x643e2c75, 0x0d042948, + 0x6b543e12, 0x65966340, 0xae7c4532, 0x7bad6b68, 0x3ce10142, 0x1a264662, 0x3d415a3a, 0x0d39536a, 0x02724119, 0x1901b593, 0x1b197ecb, 0x08a01431, + 0x0a181816, 0xa571420e, 0x313c1d71, 0xb4834920, 0x8ace896c, 0xfdad5945, 0x582794fe, 0x8654698f, 0x4430335f, 0xbc641a49, 0x03005993, 0x0000ae00, + 0x4a041904, 0x20001500, 0x5c002b00, 0x1b033a40, 0x00004621, 0x0947160f, 0x2d202d57, 0x902d4001, 0xd02db02d, 0x052df02d, 0x06012d0f, 0x0f471b26, + 0x1a032c54, 0x0126bf50, 0x26892679, 0x1b262602, 0x0f105025, 0x150f501b, 0x3fe13f00, 0x2f3912e1, 0x39e15d5d, 0xe1f61001, 0x5d5d5e32, 0xe1f61071, + 0xe12f3912, 0x30313912, 0x07061401, 0x15031e15, 0x23020e14, 0x32211121, 0x3403021e, 0x11212326, 0x023e3221, 0x23263403, 0x32331121, 0xf603023e, + 0x61386b7b, 0x652f2947, 0x39fe719f, 0x8e53c501, 0x82903b67, 0x0100ff80, 0x435d3804, 0x6c6d1e26, 0x3eecf5fe, 0x031f3f5c, 0x12756835, 0x3d240707, + 0x74433f5b, 0x4a043056, 0xfd6b401c, 0xfe485cc7, 0x452a12a8, 0x404d0c02, 0x240fd7fe, 0x0100003c, 0x0000ae00, 0x4a040a03, 0x2f000500, 0x00001c40, + 0x00020010, 0x07a00700, 0x01078101, 0x02010750, 0x06540347, 0x0f045001, 0x3f001502, 0x1001e13f, 0x5d5de1f6, 0x2f33115d, 0x0130315d, 0x11231121, + 0xfe0a0321, 0x5c02b65a, 0x50fcb003, 0x02004a04, 0x83fe2900, 0x4a043704, 0x17000f00, 0x10407900, 0x46120717, 0x3f072f0c, 0x070b0207, 0x0702071b, + 0x40f0ffb8, 0xb20ca239, 0x0c04020c, 0x07020c14, 0x050c070c, 0x010e4710, 0x20000046, 0x550e010e, 0xcf19af19, 0x19300219, 0x01192f01, 0x05054604, + 0x0c4f1218, 0x060e170f, 0x05150350, 0x3f00fb00, 0x32e13f32, 0x11e13f32, 0xe12f3301, 0x105d5d5d, 0x2f325de4, 0x11e110e1, 0x2f2f3939, 0x5d385d5d, + 0x11e1105d, 0x01303133, 0x11211123, 0x3e331123, 0x21371202, 0x11213311, 0x07030e23, 0xfdae3704, 0x4156ae4e, 0x01224363, 0xfe971702, 0x2407c0b2, + 0xfe34513a, 0xfe7d0183, 0x5f170283, 0x0001f3df, 0x0350fc7f, 0xd1d76724, 0xffff53c2, 0xecff7100, 0x5e04e103, 0x48000602, 0x01000000, 0x00000400, + 0x4a04be05, 0x9c001100, 0x03061f40, 0x0f460a03, 0x0bd60000, 0xc90bb901, 0x0ba6020b, 0x010b9701, 0x070e0b0b, 0x08050504, 0xc0ffb807, 0x481512b3, + 0xc0ffb807, 0x0b072440, 0x01070f48, 0x07100706, 0x13001307, 0xdf021310, 0x13300113, 0x13901370, 0x01130f03, 0x10101107, 0xffb80e0d, 0x0e0f40f0, + 0x0003120e, 0x080d0100, 0x0411150a, 0x3f000f01, 0x333f3333, 0x11391233, 0x33011133, 0x3333382f, 0x5d5e3311, 0x11715d5d, 0x5e382f33, 0x332b2b5d, + 0x11331133, 0x5d2f3912, 0x335d5d5d, 0x32e13311, 0x30313311, 0x11331101, 0x01013301, 0x23110123, 0x01230111, 0x8f023301, 0xc5a001a4, 0xce0158fe, + 0xa443fece, 0x01cf44fe, 0xc558fecf, 0x15023502, 0x1502ebfd, 0xcbfdebfd, 0xd3fd2d02, 0xd3fd2d02, 0x15023502, 0x44000100, 0x5203ecff, 0x39005e04, + 0x3a405f00, 0x1b164605, 0x39163916, 0x47340c16, 0xb03b5721, 0xd03bc03b, 0x3b6f033b, 0x013b2001, 0x3a560c29, 0xcf50381b, 0x39890139, 0x0f023999, + 0x39080139, 0x502f1139, 0x50081626, 0x3f001011, 0x11e13fe1, 0x5d5e2f39, 0x39e15d5d, 0x32e61001, 0x105d5d5d, 0x3912e1f6, 0x122f2f39, 0x31e11039, + 0x3e320130, 0x26343502, 0x07062223, 0x33033e27, 0x15021e32, 0x07020e14, 0x15031e15, 0x23020e14, 0x35272622, 0x3233031e, 0x3435023e, 0x35232326, + 0x643d7101, 0x6c732748, 0x3f4b8147, 0x5a524d28, 0x69905635, 0x46331c3a, 0x3f522c2b, 0xa66c3526, 0x3aaf6d71, 0x61574c1f, 0x46603733, 0x818a8529, + 0x27128702, 0x4d4d2d40, 0x14932226, 0x2609131c, 0x32466d4a, 0x0e293b4d, 0x422c0e0b, 0x75433e5a, 0x22233156, 0x181f10a6, 0x452a120f, 0x995b5e33, + 0xae000100, 0x3b040000, 0x0b004a04, 0x1b405100, 0x18091003, 0x46060348, 0x000d5505, 0x200d100d, 0xc00da00d, 0xf00dd00d, 0xb80d070d, 0x0eb3c0ff, + 0xb8094811, 0x1140f0ff, 0x09481809, 0x540a4600, 0x0b03080c, 0x0a02060f, 0x333f0015, 0x33333f33, 0xe1f61001, 0x5d2b2b32, 0x32e1f610, 0x0130312b, + 0x33010711, 0x13112311, 0x01112301, 0x04020c5a, 0xfd0bace9, 0x4a04eafe, 0x03fc64fd, 0x02b6fb98, 0xfc0f0187, 0x004a046a, 0xae00ffff, 0x3b040000, + 0x26021706, 0x0000d201, 0x36020601, 0x15000025, 0x111101b4, 0xffb80126, 0x0c16b4ff, 0x0125040a, 0x2b00352b, 0x01000035, 0x0000ae00, 0x4a04e503, + 0x41000a00, 0x01002340, 0x03000401, 0x03020310, 0x0c030310, 0x0a010c0f, 0x07470602, 0x02050b54, 0x00070a0a, 0x07040f08, 0x333f0015, 0x3912333f, + 0x01333311, 0x32e1f610, 0x33115d32, 0x335d382f, 0x31331133, 0x01330130, 0x11012301, 0x11331123, 0xfec4fa02, 0xcec60161, 0xb4b44bfe, 0xf1fd4a04, + 0x2d02c5fd, 0x4a04d3fd, 0x0100ebfd, 0xf2ff1000, 0x4a04b003, 0x4c001400, 0x46033140, 0xf4011304, 0x13800113, 0x13c01390, 0x540413d0, 0x13130113, + 0x0047010b, 0x160f1655, 0x16ff162f, 0x0b0b0603, 0x13500315, 0x084f0e0f, 0x00150116, 0x3fe13f3f, 0x330111e1, 0x105d5e2f, 0x3911e1f6, 0x5d5d5d2f, + 0x3031e171, 0x21112321, 0x06060202, 0x27262223, 0x32331635, 0x37123636, 0xb6b00321, 0x4014fefe, 0x1c56825f, 0x1b171031, 0x33455937, 0x03540212, + 0xfefdfeb0, 0x0669e694, 0x75068306, 0xfa6f01f1, 0xae000100, 0x10050000, 0x20004a04, 0x3f405e00, 0x0046011f, 0x22cf2255, 0x50224001, 0x220f0222, + 0x10130601, 0x21541146, 0x0f3b0f02, 0x0f5b0f4b, 0x190f0903, 0x030f290f, 0x0f121f0f, 0x19441934, 0x06031954, 0x26191619, 0x09190319, 0x00151101, + 0x3333333f, 0x333f5d5d, 0x115d5d33, 0xf6100133, 0x5d5e32e1, 0xf6105d5d, 0x303132e1, 0x0e112321, 0x23010703, 0x27032e01, 0x33112311, 0x17031e01, + 0x0137033e, 0xa1100533, 0x12141106, 0x87f8fe07, 0x1105f7fe, 0xa2071214, 0x0d0b01dd, 0x040f1316, 0x16130e06, 0xd90e010d, 0x39158103, 0xfd12363b, + 0x0db00250, 0x1a3e3b31, 0x4a047ffc, 0x432247fd, 0x1110323c, 0x21413b31, 0x0100bd02, 0x0000ae00, 0x4a043b04, 0x5b000b00, 0x06021740, 0x0d550547, + 0x0d200d00, 0x400d3002, 0xd00d500d, 0x050df00d, 0xc0ffb80d, 0x110e2340, 0x47090148, 0x080c540a, 0x0101eb50, 0x01b901a9, 0x1f010f02, 0x03012f01, + 0x0a010106, 0x00031505, 0x323f000f, 0x2f39333f, 0x5d5d5d5e, 0xf61001e1, 0x5d2b32e1, 0xe1f61071, 0x01303132, 0x33112111, 0x21112311, 0x01112311, + 0xb6210264, 0xb6dffdb6, 0x39fe4a04, 0xb6fbc701, 0x17fee901, 0xffff4a04, 0xecff7100, 0x5e042d04, 0x52000602, 0x01000000, 0x0000ae00, 0x4a041204, + 0x2d000700, 0x47051c40, 0x10095504, 0x60092009, 0x80097009, 0x47000509, 0x07085401, 0x050f0250, 0x3f001500, 0x01e13f32, 0x5de1f610, 0x31e1f610, + 0x11232130, 0x11231121, 0xb6640121, 0xfeb66403, 0xfb4a0408, 0x00b003b6, 0xae00ffff, 0x3f0414fe, 0x06025e04, 0x00005300, 0x7100ffff, 0x6f03ecff, + 0x06025e04, 0x00004600, 0x29000100, 0x5e030000, 0x07004a04, 0x3c405800, 0xdf010920, 0x09700109, 0x5f020980, 0x000f0109, 0x0100cc01, 0x00bf000f, + 0x00000602, 0x60034702, 0xd0057005, 0x051f0305, 0x00050501, 0xb0031003, 0x0403c003, 0x50050103, 0x15020f06, 0xe13f3f00, 0x5d2f0132, 0x5d5d2f33, + 0x2f32e110, 0x715d5d5e, 0x715d5d5d, 0x21013031, 0x21112311, 0x5e032135, 0xfeb7c1fe, 0x033503c1, 0x0350fcb0, 0xffff9ab0, 0x14fe0a00, 0x4a04df03, + 0x5c000602, 0x03000000, 0x14fe7100, 0x14062305, 0x22001900, 0x74002900, 0x00274b40, 0x171d460b, 0x49130d0c, 0x0cc90cb9, 0xa70c9702, 0x0c0c020c, + 0x05482312, 0x2b202b57, 0x012b0f01, 0x2bef2b9f, 0x012b8002, 0x40012b6f, 0x2b0f012b, 0x06022b1f, 0x5612481a, 0x2600182a, 0x1700501e, 0x501d2710, + 0x0b160d0a, 0x3f3f001b, 0x3f32e133, 0x3f32e133, 0xe1f61001, 0x5d5d5d5e, 0x71715d5d, 0x11e1f610, 0x5d5d2f39, 0xe133332b, 0x30313232, 0x15031e01, + 0x07020e14, 0x2e112311, 0x3e343503, 0x33113702, 0x17161401, 0x05030e11, 0x11272634, 0x21033636, 0x4c88bd71, 0x77be8647, 0x89bf72aa, 0xc185474c, + 0x0cfeaa79, 0x7d58ae9c, 0x39032550, 0x99acaa9b, 0x550c5a04, 0x7d7dc68f, 0x0b5691c8, 0xdc0124fe, 0xc990560a, 0x8fc77d7e, 0xba010b55, 0xd2bb13fc, + 0x093c0313, 0x5e91673d, 0xfc14ccba, 0x00d114c6, 0x2300ffff, 0xdb030000, 0x06024a04, 0x00005b00, 0xae000100, 0xb20483fe, 0x0b004a04, 0x2b404400, + 0x000f4701, 0x00000601, 0x0a8f4707, 0x100d0a01, 0x600d200d, 0x800d700d, 0x060dc00d, 0x54034706, 0x0f04080c, 0x0350060a, 0x00fb0115, 0x32e13f3f, + 0x1001333f, 0x105de1f6, 0x33e15dd4, 0xe15d5e2f, 0x23013031, 0x33112111, 0x33112111, 0xb2043311, 0xb6b2fcb6, 0xa0b6f801, 0x7d0183fe, 0x50fc4a04, + 0x50fcb003, 0x9a000100, 0xfe030000, 0x18004a04, 0x1c403100, 0x0b470c08, 0x1a101a55, 0x1a801a60, 0x17470103, 0x03081954, 0x0b121250, 0x0f000915, + 0x3f323f00, 0x33e12f39, 0xe1f61001, 0xe1f6105d, 0x01303132, 0x32331411, 0x1137023e, 0x11231133, 0x2223030e, 0x1135022e, 0x2db85001, 0x2a4d4c50, + 0x502cb6b6, 0x4e3a5e55, 0x042b527a, 0xae66fe4a, 0x1c2c1c0f, 0xb6fbd501, 0x301ee901, 0x542f1120, 0xa6014473, 0xae000100, 0x46060000, 0x0b004a04, + 0x43406600, 0x09b64700, 0xa90209c6, 0x74030109, 0x94098409, 0x09670309, 0x09090601, 0x04470105, 0x0d200d55, 0x010d0f01, 0x0df00d80, 0x010d6f02, + 0x0f010d40, 0x020d2f0d, 0x05470806, 0x08000c54, 0x0a150550, 0x000f0206, 0x3f33333f, 0x100132e1, 0x5d5ee1f6, 0x715d5d5d, 0xe1f61071, 0x5f2f3911, + 0x5d5f5d5d, 0x3031e15d, 0x33112125, 0x33112111, 0x33112111, 0xba01d503, 0xb668fab7, 0x9ab6bb01, 0xb6fbb003, 0x50fc4a04, 0x0100b003, 0x83feae00, + 0x4a04e506, 0x77000f00, 0x470a4e40, 0x07c607b6, 0x0107a902, 0x84077403, 0x03079407, 0x06010767, 0x0b030707, 0x47010e47, 0x0001000e, 0x010ebf00, + 0x1120110e, 0x01110f01, 0x11801140, 0x11f011e0, 0x2f110f04, 0x06060211, 0x10540347, 0x0f04080c, 0x50060a0e, 0xfb011503, 0xe13f3f00, 0x333f3232, + 0xf6100133, 0x5d5d5ee1, 0xc4107171, 0x5d2f325d, 0x11e110e1, 0x5d5f2f39, 0x5d5d5f5d, 0x013031e1, 0x11211123, 0x11211133, 0x11211133, 0x06331133, + 0x7ffab6e5, 0xb6bb01b6, 0x9fb7ba01, 0x7d0183fe, 0x50fc4a04, 0x50fcb003, 0x50fcb003, 0x29000200, 0xfc040000, 0x0e004a04, 0x5c001b00, 0x470f3e40, + 0x501d5704, 0x901d601d, 0x041de01d, 0x00011d3f, 0x000a4716, 0x020c100c, 0x0cf00cb0, 0x000c0c02, 0x500a100a, 0xa00a600a, 0x0a08050a, 0x00bf5015, + 0x16000001, 0x0f0d500c, 0x150a5016, 0x3fe13f00, 0x2f3912e1, 0x2f01e15d, 0x2f335d5e, 0xe110715d, 0x105d5d32, 0x3031e1f6, 0x16322101, 0x020e1415, + 0x21112123, 0x34012135, 0x2123022e, 0x3e322111, 0x01480202, 0x2fccd414, 0xfe719f65, 0x0298fe39, 0x2102021f, 0xff406041, 0x37040100, 0x0226445d, + 0x4d9b9c87, 0x032f587c, 0xf8fc9ab0, 0x11284231, 0x2911a8fe, 0x03000042, 0x0000ae00, 0x4a045005, 0x10000c00, 0x61001d00, 0x47184040, 0x04100400, + 0x04700420, 0x060504d0, 0x0e0a0404, 0x1f550d47, 0xbf011f0f, 0xdf1fcf1f, 0x041fff1f, 0x0f011fa0, 0x021f7f1f, 0x47120007, 0x111e540a, 0x0100bf50, + 0x0b0f0000, 0x0e50120f, 0x3f00150a, 0x333fe133, 0xe15d2f39, 0xe1f61001, 0x5d5d5e32, 0xf610715d, 0x2f3911e1, 0x31e15d5e, 0x32330130, 0x0e141516, + 0x11212302, 0x11230133, 0x33110133, 0x35023e32, 0x23022e34, 0xd4ec6401, 0x9f652fcc, 0xb662fe71, 0xb6b6ec03, 0x38db14fc, 0x2126435d, 0x02406041, + 0x4d9b9c87, 0x042f587c, 0x04b6fb4a, 0xfea4fd4a, 0x422911a8, 0x28423130, 0x02000011, 0x0000ae00, 0x4a041904, 0x19000c00, 0x29404000, 0x57044714, + 0x011b001b, 0x1b201b00, 0x1b901b70, 0x1bd01bb0, 0x0e000806, 0x1a540a47, 0x00bf500d, 0x0b000001, 0x0a500e0f, 0xe13f0015, 0x5d2f393f, 0xf61001e1, + 0x5d5e32e1, 0xe1f61071, 0x21013031, 0x14151632, 0x2123020e, 0x11113311, 0x023e3221, 0x022e3435, 0x01640123, 0x2fccd415, 0xfe719f65, 0x0401b639, + 0x26435d38, 0x40604121, 0x9b9c8702, 0x2f587c4d, 0xa4fd4a04, 0x2911a8fe, 0x42313042, 0x01001128, 0xecff3700, 0x5e044203, 0x5e002600, 0x0e0e3e40, + 0x480f0c04, 0x8f285722, 0x0228af28, 0x18012840, 0x041f040f, 0x0403042f, 0x48110c40, 0x0d275804, 0x010eef50, 0x0eb90ea9, 0x1f0e0f02, 0x030e2f0e, + 0x070e0e06, 0x101d5112, 0x16005107, 0x3fe13f00, 0x2f3912e1, 0x5d5d5d5e, 0xe61001e1, 0x5d32712b, 0xe1f6105d, 0x2f391132, 0x22053031, 0x16352726, + 0x3e323316, 0x35213702, 0x23262621, 0x07020e22, 0x33033e27, 0x15021e32, 0x01020e14, 0x367c5942, 0x49528336, 0x04305274, 0xdb0123fe, 0x198c920d, + 0x17383c3b, 0x48401835, 0xb366264e, 0x8c514d84, 0x1d1e14ba, 0x282519a2, 0x9a5c8454, 0x0f0997a6, 0x0c9a0a13, 0x3e091016, 0x929ddb89, 0x00438ad4, + 0xae000200, 0x0806ecff, 0x1a005e04, 0x72002600, 0x481b4b40, 0x0111200a, 0x1101114f, 0x48210d11, 0x0f285700, 0x02282f28, 0x28ff28ef, 0x01284702, + 0x282f280f, 0x470c1002, 0x2427540d, 0x0b101650, 0x0110ef50, 0x10b910a9, 0x01102c02, 0x101b100b, 0x10100602, 0x0d0f0e0d, 0x05501e15, 0xe13f0016, + 0x39123f3f, 0x5d5d5e2f, 0x3fe15d5d, 0xf61001e1, 0x5d5d32e1, 0xf610715d, 0x2f3911e1, 0xe133715d, 0x14013031, 0x2223020e, 0x2127022e, 0x33112311, + 0x033e2111, 0x021e3233, 0x33161405, 0x34353632, 0x06222326, 0x78400806, 0xa2626aab, 0xfe074675, 0x01b6b6ef, 0x76480a13, 0xa76263a0, 0x29fd4479, + 0x7e8f8e80, 0x7e8e8f80, 0xd5892702, 0x81424c91, 0x17fe7cbe, 0x39fe4a04, 0x3e78b174, 0x88d3914b, 0xd1d3d3d1, 0x00cfcfd1, 0x23000200, 0x91030000, + 0x11004a04, 0x53001c00, 0x11021440, 0x0f180711, 0x1e550e47, 0x1e6f1e0f, 0x06031e8f, 0xffb80100, 0x011940f0, 0x07461201, 0x48151240, 0x021d5607, + 0x18185010, 0x0d501900, 0x15000f0f, 0x3f323f00, 0x2f3912e1, 0x100132e1, 0x33e12bf6, 0x5e33382f, 0xe1f6105d, 0x11391132, 0x33303133, 0x032e0123, + 0x023e3435, 0x23112133, 0x14032311, 0x3333021e, 0x06222311, 0x2301c6e9, 0x2841542d, 0x538a6438, 0xf4b6bc01, 0x5a4125dd, 0x6bfedb36, 0x0ccd0168, + 0x476c4c30, 0x2a51794e, 0xb001b6fb, 0x46305401, 0x6601162e, 0xffff005f, 0xecff7100, 0xd905e103, 0x48002602, 0x06010000, 0x00da6a00, 0x0d401700, + 0x11310203, 0x00020326, 0x0f053c28, 0x352b0125, 0x352b0035, 0x01000035, 0x14fe1200, 0x14061204, 0x53003100, 0x171f3240, 0x552d470c, 0x60331033, + 0x03338033, 0x161e2223, 0x54171b47, 0x4f192132, 0x1c1a1a1e, 0x00501023, 0x20291029, 0x0f290329, 0x1517001c, 0x1b005007, 0x3fe13f00, 0xe15d3f3f, + 0x2f391233, 0x0132e133, 0xe132f610, 0x5d323232, 0x12e1f610, 0x01303139, 0x35272622, 0x32331616, 0x1135023e, 0x22232634, 0x1115020e, 0x35231123, + 0x15333533, 0x15211521, 0x033e3307, 0x15163233, 0x020e1411, 0x3f30f002, 0x23361a17, 0x13232e1b, 0x6e517069, 0x9cb61d43, 0x7b01b69c, 0x0a0885fe, + 0x5c524519, 0x22b9b730, 0x14fe6d48, 0x0a940b0e, 0x41270f0b, 0x82440333, 0x94663482, 0x04f0fd60, 0xb6b689d5, 0x2b90b889, 0xbf142a3f, 0x4dbefcd2, + 0x002f577b, 0xae00ffff, 0x0a030000, 0x26022106, 0x0000cd01, 0x76000601, 0x130000d4, 0x12010b40, 0x45012611, 0x00030c06, 0x352b0125, 0x00352b00, + 0x71000100, 0x6f03ecff, 0x22005e04, 0x31404f00, 0x1f161717, 0x248f240d, 0x400224af, 0x19160124, 0x23560548, 0x16ef5019, 0xb916a901, 0x160f0216, + 0x162f161f, 0x16160603, 0x0a51131c, 0x00511c10, 0xe13f0016, 0x3912e13f, 0x5d5d5e2f, 0x1001e15d, 0x5d32e1f6, 0x32ce105d, 0x312f3911, 0x2e220530, + 0x3e343502, 0x16323302, 0x032e0717, 0x07062223, 0x16211521, 0x36323316, 0x06061537, 0xb0655202, 0x854c4a82, 0x954e66b2, 0x38173632, 0x8b1a3a3c, + 0xdb010e92, 0x920923fe, 0x3683518a, 0x3f147b36, 0x9d96d589, 0x223e89db, 0x130a9a19, 0xa697090f, 0x25a4b89a, 0x1e1da219, 0x5a00ffff, 0x3f03ecff, + 0x06025e04, 0x00005600, 0xa000ffff, 0x75010000, 0x0602e505, 0x00004c00, 0xeeffffff, 0x25020000, 0x2602d905, 0x0000f300, 0x6a000701, 0x0000bbfe, + 0x0d401700, 0x110d0102, 0x00010226, 0x00011804, 0x352b0125, 0x352b0035, 0xffff0035, 0x14febcff, 0xe5057501, 0x4d000602, 0x02000000, 0xf2ff1000, + 0x4a041206, 0x2a001d00, 0x45406b00, 0x001c460c, 0xd40a4725, 0x1c90011c, 0x24021ca0, 0x1c00011c, 0x1c0a1c01, 0x471e140a, 0x0f2c5704, 0x3f2c2f2c, + 0x7f2c5f2c, 0xef2cbf2c, 0x1414072c, 0xbf50242b, 0x00000100, 0x1c500c25, 0x114f170f, 0x0a502516, 0xe13f0015, 0xe13fe13f, 0x5d2f3912, 0x330111e1, + 0xf6105d2f, 0x393911e1, 0x5d5d2f2f, 0xe1105d5d, 0x31e11032, 0x32330130, 0x0e141516, 0x11212302, 0x06020223, 0x26222306, 0x33163527, 0x12363632, + 0x34012137, 0x2323022e, 0x3e323311, 0xec870302, 0x652ecbd4, 0x62fe719f, 0x5f4014d9, 0x311c5682, 0x371b1710, 0x12334559, 0xd9012b02, 0x40604121, + 0x5d38dbd7, 0x87022643, 0x7c4d9b9c, 0xb0032f58, 0x94fefdfe, 0x060669e6, 0xf1750683, 0xfcfa6f01, 0x284231f8, 0x11a8fe11, 0x02004229, 0x0000ae00, + 0x4a047506, 0x21001400, 0x30405300, 0x02471c05, 0x15130f0f, 0x23570947, 0x800123ef, 0x233f0123, 0x01231001, 0x13471201, 0x111b2254, 0x01bf0550, + 0x00010101, 0x0f13501c, 0x0f000315, 0x3f323f00, 0x3911e133, 0xe1335d2f, 0xf6100132, 0x5d5d32e1, 0xf6105d5d, 0x2f3911e1, 0x3132e133, 0x21110130, + 0x33113311, 0x14151632, 0x2123020e, 0x23112111, 0x2e340111, 0x11232302, 0x023e3233, 0xcf016401, 0xccd4ecb6, 0x719f652f, 0x31fe62fe, 0x211505b6, + 0xd83f6141, 0x445d37dc, 0xfe4a0426, 0xfec70139, 0x4d9b9c3d, 0x012f587c, 0x0417fee9, 0x31f8fc4a, 0xfe112842, 0x422911a8, 0x12000100, 0x12040000, + 0x21001406, 0x32405300, 0x14011468, 0x2147000b, 0x23102355, 0x23802360, 0x12161703, 0x0b0f470a, 0x0d152254, 0x0e0e124f, 0x50041710, 0x1d101d00, + 0x1d031d20, 0x0b00100f, 0x3f001500, 0x5d3f3f32, 0x391233e1, 0x32e1332f, 0x32f61001, 0x323232e1, 0xe1f6105d, 0x315d3911, 0x34112130, 0x0e222326, + 0x23111502, 0x33352311, 0x21153335, 0x07152115, 0x33033e33, 0x11151632, 0x70695c03, 0x1d436e51, 0xb69c9cb6, 0x85fe7b01, 0x45190a08, 0xb7305c52, + 0x829a02b9, 0x94663482, 0x04f0fd60, 0xb6b689d5, 0x2b90b889, 0xbf142a3f, 0x005cfdd2, 0xae00ffff, 0xe5030000, 0x26022106, 0x0000d401, 0x76000601, + 0x13000008, 0x17010b40, 0x0c012611, 0x0307110b, 0x352b0125, 0x00352b00, 0x0a00ffff, 0xdf0314fe, 0x26021706, 0x00005c00, 0x36020601, 0x130000b1, + 0x28010b40, 0x0a012611, 0x0f00232d, 0x352b0125, 0x00352b00, 0xae000100, 0x120483fe, 0x0b004a04, 0x3d405900, 0xb70106d6, 0x0206c706, 0x07b54706, + 0x07d507c5, 0x86077603, 0x074a0207, 0x0702075a, 0x47010907, 0x100d5504, 0x600d200d, 0x800d700d, 0x4700050d, 0x050c5409, 0x15095000, 0x020afb06, + 0x333f000f, 0x33e13f3f, 0xe1f61001, 0xe1f6105d, 0x5d2f3911, 0x5de15d5d, 0x2530315d, 0x11331121, 0x11231121, 0x01331121, 0xb6f80164, 0xfeb7b3fe, + 0x039ab6a0, 0xfeb6fbb0, 0x047d0183, 0x0100004a, 0x0000c700, 0xe306be03, 0x44000700, 0x5a002c40, 0x03ef03df, 0x000303ff, 0x02031003, 0xb0090303, + 0x0209d009, 0x09af090f, 0x065a0502, 0x5f040864, 0x01012007, 0x03070101, 0x3f001206, 0x5d2f333f, 0x1001e110, 0x5d5de1f6, 0x5d2f3311, 0x3031e15d, + 0x11331101, 0x11231121, 0xfdb00e03, 0xb605bac3, 0x2dfe2d01, 0xb605f0fa, 0xae000100, 0x0a030000, 0x07008905, 0x1c403000, 0x00004705, 0x00020010, + 0x47020900, 0x10085403, 0x02062006, 0x4f010606, 0x15020f04, 0xe13f3f00, 0x015d2f33, 0x11e1f610, 0xe15d2f33, 0x21013031, 0x21112311, 0x0a033311, + 0x01b65afe, 0xc103b6a6, 0x4a043ffc, 0xffff3f01, 0x00001400, 0x7307fe06, 0x3a002602, 0x07010000, 0xe7004300, 0x15005201, 0x052b01b4, 0xffb80126, + 0x2b31b4ac, 0x01250e1e, 0x2b00352b, 0xffff0035, 0x00001400, 0x2106e305, 0x5a002602, 0x06010000, 0x004c4300, 0x01b41500, 0x01261130, 0xb49effb8, + 0x2e113036, 0x352b0125, 0x00352b00, 0x1400ffff, 0xfe060000, 0x26027307, 0x00003a00, 0x76000701, 0x5201a001, 0x0b401300, 0x26053701, 0x312b6401, + 0x01250e1e, 0x2b00352b, 0xffff0035, 0x00001400, 0x2106e305, 0x5a002602, 0x07010000, 0x17017600, 0x13000000, 0x3c010b40, 0x69012611, 0x2e113630, + 0x352b0125, 0x00352b00, 0x1400ffff, 0xfe060000, 0x26022b07, 0x00003a00, 0x6a000701, 0x52013901, 0x02b61900, 0x26053401, 0xffb80102, 0x3f2bb4ff, + 0x01250e1e, 0x0035352b, 0x0035352b, 0x1400ffff, 0xe3050000, 0x2602d905, 0x00005a00, 0x6a000701, 0x0000ac00, 0x02b61900, 0x26113901, 0xffb80102, + 0x4430b4ff, 0x01252e11, 0x0035352b, 0x0035352b, 0x0000ffff, 0x37040000, 0x26027307, 0x00003c00, 0x43000701, 0x520172ff, 0x01b41500, 0x01260509, + 0xb4a4ffb8, 0x0207090f, 0x352b0125, 0x00352b00, 0x0a00ffff, 0xdf0314fe, 0x26022106, 0x00005c00, 0x43000701, 0x00004bff, 0x01b41500, 0x01261123, + 0xb4a4ffb8, 0x0f002329, 0x352b0125, 0x00352b00, 0x52000100, 0xae03d101, 0x03007902, 0x00b91d00, 0x40c0ff02, 0x480c060b, 0x00050202, 0xbd01b900, + 0x01e13f00, 0x2f33112f, 0x1330312b, 0x52152135, 0xd1015c03, 0x0100a8a8, 0xd1015200, 0x7902ae07, 0x1d000300, 0xff0200b9, 0x060b40c0, 0x0202480c, + 0xb9000005, 0x3f00bd01, 0x112f01e1, 0x312b2f33, 0x21351330, 0x5c075215, 0xa8a8d101, 0x52000100, 0xae07d101, 0x03007902, 0x00b91d00, 0x40c0ff02, + 0x480c060b, 0x00050202, 0xbd01b900, 0x01e13f00, 0x2f33112f, 0x1330312b, 0x52152135, 0xd1015c07, 0x0200a8a8, 0x31fefcff, 0xd3ff4e03, 0x07000300, + 0x18402a00, 0x09000004, 0xba050105, 0x06ff06ef, 0xba020602, 0x01100100, 0x01030120, 0xe15d2f00, 0x01e15d2f, 0x3311332f, 0x3031332f, 0x21352101, + 0x21352135, 0xaefc4e03, 0xaefc5203, 0x31fe5203, 0x008b8c8b, 0x17000100, 0x5001c103, 0x0c00b605, 0x17402500, 0x06010e5f, 0x075f070f, 0x07bf076f, + 0x070507cf, 0x0c01980c, 0x0003069c, 0x2f01e53f, 0x335d2fe1, 0x1330315d, 0x37033e27, 0x07030e33, 0x270e0e25, 0x8919342e, 0x161a1d0f, 0x16c10308, + 0x7b7c7a36, 0x83843d38, 0x0100357c, 0xc1031700, 0xb6055001, 0x25000c00, 0x0e5f1740, 0x0c070601, 0x5f010f98, 0xbf016f01, 0x0501cf01, 0x009c0601, + 0xe53f0003, 0xe15d2f01, 0x315d332f, 0x0e170130, 0x3e230703, 0x42013703, 0x2f270e0e, 0x0e891933, 0x08161b1d, 0x3716b605, 0x387a7d79, 0x7c84843c, + 0x01000035, 0xf8fe3f00, 0xee007901, 0x35000c00, 0xff0e00b9, 0x0a1440c0, 0x980c4818, 0x015f010f, 0x017f016f, 0x010501cf, 0xb8070601, 0x10b7c0ff, + 0x06074815, 0x00a8009c, 0x2f01e53f, 0x2f33332b, 0x312be15d, 0x0e172530, 0x3e230703, 0x6a013703, 0x2f270e0f, 0x0f8a1933, 0x08161b1d, 0x7a3617ee, + 0x3d387b7c, 0x357d8384, 0x19000100, 0x5201c103, 0x0c00b605, 0x1c402b00, 0x06010e5f, 0x055f050f, 0x057f056f, 0x05cf05bf, 0x050705df, 0x0b980005, + 0x03009c06, 0x01e53f00, 0x2f33e12f, 0x315d335d, 0x031e1330, 0x032e2317, 0x08ee3727, 0x0f1d1a16, 0x2e341989, 0x050e0e27, 0x847c35b6, 0x7a383c84, + 0x1637797d, 0x17000200, 0xd102c103, 0x0c00b605, 0x62001900, 0x1bbf4840, 0x011b9001, 0x1b5f1b0f, 0x13031b6f, 0x145f140f, 0x147f146f, 0x14cf14bf, + 0x140714df, 0x0e981914, 0x0100980c, 0x01600150, 0x01b00170, 0x01d001c0, 0x06010107, 0x075f070f, 0x07bf076f, 0x070507cf, 0x139c0c19, 0x3f000306, + 0x0132e533, 0x33335d2f, 0x2fe15d2f, 0x5d2f33e1, 0x5d5d5d33, 0x27013031, 0x3337033e, 0x2107030e, 0x37033e27, 0x07030e33, 0x0e0ea601, 0x19342e27, + 0x1a1d0f89, 0xb8fd0816, 0x2e270e0e, 0x0f891934, 0x08161a1d, 0x3616c103, 0x387b7c7a, 0x7c83843d, 0x7a361635, 0x3d387b7c, 0x357c8384, 0x17000200, + 0xd102c103, 0x0c00b605, 0x62001900, 0x1bbf4840, 0x011b9001, 0x1b5f1b0f, 0x13031b6f, 0x14501400, 0x14701460, 0x14c014b0, 0x140714d0, 0x0f981914, + 0x6f0e5f0e, 0xcf0ebf0e, 0x0c0e050e, 0x5f010f98, 0x7f016f01, 0xcf01bf01, 0x0701df01, 0x07060101, 0x0d9c0613, 0x3f000300, 0x0132e532, 0x2f33332f, + 0x5d2fe15d, 0x5d2f33e1, 0x5d5d5d33, 0x17013031, 0x2307030e, 0x2137033e, 0x07030e17, 0x37033e23, 0x0e0e4201, 0x19332f27, 0x1b1d0e89, 0x48020816, + 0x2f270e0e, 0x0e891933, 0x08161b1d, 0x3716b605, 0x387a7d79, 0x7c84843c, 0x79371635, 0x3c387a7d, 0x357c8484, 0x3f000200, 0xfa02f8fe, 0x0c00ee00, + 0x7e001900, 0x1bd05140, 0x1bf01be0, 0xb41ba403, 0x031bc41b, 0x02011b90, 0x1b301b20, 0x1b601b40, 0x1b801b70, 0x14001306, 0x14601450, 0x14c01470, + 0x140614d0, 0x90981914, 0xf00ee00e, 0x0e0f030e, 0x0e020e5f, 0x010f980c, 0x016f015f, 0x01cf017f, 0x010601df, 0xb8070601, 0x0a40c0ff, 0x07481810, + 0x0d9c0613, 0x3f00a800, 0x0132e532, 0x33332b2f, 0x2fe15d2f, 0x33e15d5d, 0x5d335d2f, 0x5d5d5d5f, 0x17253031, 0x2307030e, 0x2137033e, 0x07030e17, + 0x37033e23, 0x0e0f6a01, 0x19332f27, 0x1b1d0f8a, 0x48020816, 0x2f270e0e, 0x0e891933, 0x08161b1d, 0x7a3617ee, 0x3d387b7c, 0x357d8384, 0x7c7a3617, + 0x843d387b, 0x00357d83, 0x7b000100, 0x68030000, 0x0b001406, 0x52407c00, 0x0da00d90, 0x0df00de0, 0x010d6f04, 0x0d300d10, 0x07030d40, 0x0804c006, + 0x0be40308, 0xd6020bf4, 0x0b77010b, 0x010b6a01, 0x45010b54, 0x0b26010b, 0x09c00b01, 0x0909010a, 0x02010236, 0x100300be, 0x0303a003, 0x00010403, + 0x0a0bbf05, 0x08c20607, 0x2f000300, 0x3232f63f, 0x3232e132, 0x5d2f0132, 0x11325de1, 0xe6103333, 0x5d5d5d5d, 0x115d5d5d, 0xe6332f33, 0x5d5d5d33, + 0x25013031, 0x05132313, 0x33030535, 0x68032503, 0xd937b5fe, 0x01c9fe37, 0x37d93737, 0xdd034b01, 0x0304fc1f, 0x1eb41ffc, 0x5ffea101, 0x0100001e, + 0x00007b00, 0x14067d03, 0xb0001500, 0x17007140, 0xb017a001, 0x0317f017, 0x2001177f, 0x50174017, 0x0d080317, 0x0f0b0ec0, 0x0401050f, 0x12f40112, + 0x0112e601, 0x7a011287, 0x12640112, 0x01125501, 0x12461236, 0x10c01202, 0x10101114, 0x04000315, 0x0a0a05be, 0x05000609, 0x10050001, 0x50052005, + 0xb005a005, 0x05080605, 0x0c0b1314, 0x0e1112bf, 0x000fc20d, 0x08000109, 0x020306bf, 0x0005c207, 0x3232f62f, 0x3232e132, 0x32f63f32, 0x32e13232, + 0x2f013232, 0x33715d5e, 0x102f3333, 0x323232e1, 0x33331132, 0x5d5de610, 0x5d5d5d5d, 0x1132715d, 0x33332f33, 0x5d5d32e6, 0x3031715d, 0x25152501, + 0x05132313, 0x13030535, 0x03053505, 0x15250333, 0x31021325, 0xb4fe4c01, 0xfe38d937, 0x2f4c01b4, 0x01b4fe2f, 0x37d9384c, 0xb4fe4c01, 0x1ef0012f, + 0x87fe1fb4, 0xb41f7901, 0x0122011e, 0x1eb41f13, 0x88fe7801, 0xfe1fb41e, 0x010000ed, 0xe5019600, 0xf2036d02, 0x46001300, 0x152f2440, 0x156f155f, + 0x15cf157f, 0x15ff15ef, 0x01151007, 0x0a6f0a5f, 0x0aaf0a9f, 0x0aef0adf, 0x00d00a06, 0xffb80001, 0x070c40c0, 0x1f00480a, 0x100f010f, 0x00050105, + 0x5dc55d2f, 0x5d2b2f01, 0x5d5d5dc5, 0x34133031, 0x3233023e, 0x1415021e, 0x2223020e, 0x2496022e, 0x3132563f, 0x25254056, 0x32315640, 0x02243f56, + 0x3f6447ec, 0x643f1c1c, 0x3f644647, 0x643f1e1e, 0x93000300, 0xdb05e3ff, 0x1300fa00, 0x39002700, 0x7640a800, 0x3b243b14, 0x013bfb02, 0xbb013be4, + 0x023bcb3b, 0x8b013ba4, 0x3b64013b, 0x4b023b74, 0x3b34013b, 0x013b0b01, 0x1466961e, 0x14021476, 0x96280014, 0xfb013224, 0x32e00132, 0x0132d401, + 0xa40132bb, 0x328b0132, 0x01327201, 0x4b013266, 0x32300132, 0x32200201, 0x01320f01, 0x960a3206, 0x00f000e0, 0x64005402, 0x00000200, 0x00200010, + 0x2d000703, 0x379b0519, 0x2f000f23, 0x32ed3333, 0x5e2f0132, 0xed5d5d5d, 0x5d5d5e2f, 0x5d5d5d5f, 0x5d5d5d5d, 0x715d5d5d, 0x2f3911ed, 0x5d5ded5d, + 0x5d5d5d5d, 0x715d5d5d, 0x34373031, 0x3233023e, 0x1415021e, 0x2223020e, 0x3425022e, 0x3233023e, 0x1415021e, 0x2223020e, 0x3425022e, 0x3233023e, + 0x1415021e, 0x2223020e, 0x22149326, 0x2f1a1b2e, 0x22141422, 0x2e1b1a2f, 0x25021422, 0x1b2e2214, 0x14222f1a, 0x1a2f2214, 0x14222e1b, 0x23132502, + 0x2f1a1b2e, 0x22141422, 0x49361a2f, 0x2135266f, 0x35210f0f, 0x22352526, 0x35221010, 0x21352625, 0x35210f0f, 0x22352526, 0x35221010, 0x21352625, + 0x35210f0f, 0x22352526, 0x07004210, 0xecff6600, 0xcb05f408, 0x1d000900, 0x3b002700, 0x49003f00, 0x89005d00, 0xb4405c40, 0x1e4ab554, 0x23b532b4, + 0x3ea028b4, 0x403c3001, 0x033cb03c, 0x28402830, 0x283c3e02, 0x033e3c28, 0x4ab44514, 0x4f5f3f5f, 0x7f5f5f5f, 0x055faf5f, 0xb50ab405, 0x1410b400, + 0x14301420, 0xb6471403, 0xb643b759, 0x063f194f, 0xb625183e, 0xb621b737, 0xb603192d, 0xb607b70f, 0x3f000719, 0x3fe1f4e1, 0x3fe1f4e1, 0xf4e13f3f, + 0x5d2f01e1, 0x5de1f4e1, 0x12e1de10, 0x2f2f3917, 0x5d5d5d2f, 0xe1f4e110, 0x31e1f410, 0x16141330, 0x10113233, 0x05062223, 0x23020e14, 0x35022e22, + 0x33023e34, 0x01021e32, 0x32331614, 0x22231011, 0x0e140506, 0x2e222302, 0x3e343502, 0x1e323302, 0x23010102, 0x16140101, 0x10113233, 0x05062223, + 0x23020e14, 0x35022e22, 0x33023e34, 0xfa021e32, 0x9c9c5047, 0xc7014750, 0x4f734a24, 0x264c7049, 0x4e714923, 0x274d714b, 0x5047ac01, 0x47509c9c, + 0x4a23c601, 0x704a4f73, 0x4923264b, 0x714b4e71, 0x00ff274c, 0x039ed5fc, 0x4797022c, 0x519b9b51, 0x23c70147, 0x4a4f734a, 0x23264b70, 0x4a4e7149, + 0x04274c72, 0x01a5a502, 0xa348014a, 0x76ac6ca5, 0xac763f3f, 0x75aa6c6c, 0xaa753e3e, 0xa4a54afd, 0x48014901, 0xab6ca5a3, 0x763f3f76, 0xaa6c6cab, + 0x753e3e75, 0xfa9203aa, 0xfcb6054a, 0x01a4a502, 0xa3480149, 0x76ab6ca5, 0xab763f3f, 0x75aa6c6c, 0xaa753e3e, 0x8500ffff, 0x4a01a603, 0x0602b605, + 0x00000a00, 0x8500ffff, 0xb202a603, 0x0602b605, 0x00000500, 0x52000100, 0xfc017300, 0x0600c703, 0x04b13c00, 0xc0ffb802, 0x0c091f40, 0x3f080248, + 0xaf089f08, 0xef08df08, 0x0608ff08, 0x039feb06, 0x00060301, 0x05010303, 0x2f2f0001, 0x2f3d3912, 0x18013333, 0x5de15d2f, 0x322bc610, 0x01133031, + 0x07130317, 0x35015201, 0x75eeee75, 0x2902cbfe, 0xfe4e9e01, 0x4ea4fea4, 0x01009b01, 0x73005200, 0xc703fc01, 0x3f000600, 0xeb002840, 0x03ef03df, + 0x100303ff, 0x02032003, 0x9f083f03, 0xdf08af08, 0xff08ef08, 0x3f040608, 0x06020102, 0x01030300, 0x2f000105, 0x3d39122f, 0x0133332f, 0x335d2f18, + 0x5d5d2f5d, 0x013031e1, 0x03132701, 0xfc010137, 0xed75cbfe, 0x350175ed, 0x65fe0e02, 0x015c014e, 0x62fe4e5c, 0x9300ffff, 0x6203e3ff, 0x2700b605, + 0xd1010400, 0x06010000, 0x00000400, 0x2fb50900, 0x31100131, 0x005d5d01, 0xa0fe0100, 0x68020000, 0x0300b605, 0x01b11d00, 0xf0ffb802, 0x03020940, + 0x01001000, 0x00030012, 0x2f013f3f, 0x382f3238, 0x01303133, 0x02012301, 0x9dd5fc68, 0xb6052b03, 0xb6054afa, 0x6a000100, 0x93021d03, 0x1400c705, + 0x17404100, 0x1400e000, 0x60161401, 0xe016c016, 0x0416f016, 0x0e01160f, 0xb80be00a, 0x0e40c0ff, 0x0b480f0a, 0x0ec0000a, 0xde11e404, 0x3f00dc0c, + 0x1a33e13f, 0x2f0132cc, 0x5d32e12b, 0x5dd6105d, 0x013031e1, 0x23263411, 0x15020e22, 0x33112311, 0x33363317, 0x02111532, 0x2d3f3f14, 0x7f152a41, + 0x40091069, 0x1d03e582, 0x4451a601, 0x41573415, 0x9d02a6fe, 0xfefa6558, 0x01000050, 0x00006000, 0xb6050004, 0x76001100, 0x10024840, 0x00051002, + 0x200b100b, 0x0b0b030b, 0x01133013, 0x5a04000e, 0xdf050905, 0x07070107, 0x05100500, 0x070305c0, 0x60070305, 0x11080008, 0x0e0f0e5f, 0x0e6f0e3f, + 0x400e0803, 0x08481510, 0x040e080e, 0x030a5f0d, 0x3f001204, 0x3912e13f, 0x2b2f2f39, 0xe1105d5e, 0xe1103311, 0x5e2f0132, 0x5d2f335d, 0xe1103311, + 0x115d3232, 0x125d2f33, 0x2f2f3939, 0x21013031, 0x23112115, 0x33352311, 0x21152111, 0x21152111, 0x2401c301, 0xb0b3dcfe, 0xfdf002b0, 0xfd1602c3, + 0x898f01ea, 0x0601fafe, 0xa4270489, 0x00a4fcfd, 0x44000100, 0x23040000, 0x3000c905, 0x10408700, 0x6f150d11, 0x0f23272b, 0x23132313, 0x1b031e13, + 0x40c0ffb8, 0x480e093e, 0x10321b1b, 0x25290132, 0x0e0b401e, 0x25141e48, 0x26112675, 0x2a752910, 0x266f2a0d, 0x012ad001, 0x2a1f2a0f, 0x2a4f2a3f, + 0x2abf2aaf, 0x262a2606, 0x1a1e002a, 0x07181d74, 0x00070073, 0xe13fe13f, 0x39391132, 0x5d5d2f2f, 0x1033115d, 0x331132e1, 0x0132e110, 0x33332b2f, + 0x2f33115d, 0x3912332b, 0x112f2f39, 0x33331133, 0x313232e1, 0x16320130, 0x26260717, 0x020e2223, 0x15211515, 0x15211521, 0x0e141521, 0x15210702, + 0x033e3521, 0x35233535, 0x35233533, 0x3e343533, 0x6a9a0202, 0x384242ae, 0x52304b8d, 0x7801213c, 0x780188fe, 0x271788fe, 0xec021b33, 0x492c21fc, + 0xc6c61e35, 0x693bc6c6, 0x2dc90592, 0x2b1d9023, 0x425e3b1b, 0x899e89c0, 0x40593e13, 0x9aa6102b, 0x6144290b, 0x9e891543, 0x8957dd89, 0x0300325f, + 0xecff9600, 0xb605b605, 0x2a001b00, 0x79003500, 0x6f311840, 0x0f10121c, 0x1815186e, 0x181c181c, 0x0105ff24, 0x15124005, 0xffb80548, 0x092c40c0, + 0x0505480e, 0x01370f37, 0x100b4037, 0x6e232b48, 0x0f367c24, 0x12157518, 0x22732b15, 0x22152215, 0x25733524, 0x00182406, 0x00190975, 0x3f3fe13f, + 0x393912e1, 0xe1102f2f, 0xe1103311, 0xf6100132, 0x5d2b32e1, 0x2b2f3311, 0x39125d2b, 0x112f2f39, 0x32e11033, 0x31e11032, 0x3e322530, 0x06153702, + 0x2e222306, 0x23113502, 0x33373735, 0x23153315, 0x01161411, 0x23020e14, 0x11231123, 0x021e3233, 0x3e323301, 0x26343502, 0x3b052323, 0x1e23240f, + 0x3f501707, 0x24405835, 0x6b419c9c, 0xfe34d1d1, 0xcc7c348c, 0xfbb22f98, 0x3a7bbf86, 0x5d1abdfd, 0xa42e5b8b, 0x048139ae, 0x8a030806, 0x4622140c, + 0xbf01486a, 0xd3bd4d52, 0x4c56fe89, 0x5b89034e, 0xfd4d81a8, 0x39b605c7, 0x67fea06d, 0x51714720, 0x0100898e, 0xecff3f00, 0xcb054a04, 0x86003900, + 0x280712b4, 0xffb81936, 0x094b40c0, 0x1919480d, 0x013b103b, 0x1323232c, 0x2e6f0d05, 0x01286322, 0x133a2828, 0x10246023, 0x602c0824, 0x5f2d052d, + 0x8f247f24, 0xff24ef24, 0x2d0f0524, 0x2d2f2d1f, 0x2d9f2d5f, 0x24062dcf, 0x162d242d, 0x33740037, 0x195f1607, 0x3f00191d, 0xe13fe133, 0x39391233, + 0x5d5d2f2f, 0xe1103311, 0x10331132, 0x011132e1, 0x335d2f33, 0x3232e133, 0x5d331133, 0x2b2f3311, 0x39391233, 0x22013031, 0x2107020e, 0x07142115, + 0x14151406, 0x15211714, 0x33161621, 0x15373632, 0x22230606, 0x2327022e, 0x34263335, 0x35363435, 0x3e333523, 0x16323303, 0x26260717, 0x76450803, + 0x0112435e, 0x0141feb0, 0x81010201, 0xb92292fe, 0x3b874b95, 0x735b853b, 0x155887b6, 0x020294a4, 0x5812a094, 0x6172b887, 0x33504fa0, 0x34270577, + 0x895b8f63, 0x1a0c0e0f, 0x16291309, 0x20b8af89, 0x1f1ca21a, 0x79c18749, 0x1d1e1789, 0x89082e16, 0x4e90ca7d, 0x1f92312b, 0x0400002b, 0xf8ff8d00, + 0xc105db05, 0x17000300, 0x48002b00, 0x0a408600, 0x46390201, 0x030eb418, 0xffb80200, 0x704a40f0, 0xa0469046, 0x0446d046, 0x00010edf, 0x0e460210, + 0x460e0000, 0x22310402, 0x01040fb4, 0x04ff04ef, 0x404a0402, 0x013100b4, 0x31103100, 0x31e03120, 0x080531f0, 0x2cfc4231, 0x36fc3dfd, 0x13fc2707, + 0x09fc1dfd, 0x01060319, 0x3f3f0018, 0xe1f4e13f, 0xe1f4e13f, 0x5d5e2f01, 0xde10e171, 0x11e1715d, 0x2f2f3917, 0x5d382f2f, 0x3311385d, 0x3311e110, + 0x30313311, 0x01230101, 0x020e1401, 0x022e2223, 0x023e3435, 0x021e3233, 0x021e1405, 0x023e3233, 0x022e3435, 0x020e2223, 0x022e2201, 0x023e3435, + 0x17163233, 0x23262607, 0x14150622, 0x37363233, 0x05060615, 0x9dd5fc0a, 0x6e012b03, 0x4470502d, 0x2f516e3f, 0x4471502c, 0x2f526e3e, 0x261133fe, + 0x3c2a2b3c, 0x25111125, 0x3c2b2a3c, 0xcbfd1126, 0x345a7945, 0x487d5c35, 0x21206433, 0x6722551f, 0x5a33c25f, 0x05632323, 0x054afab6, 0x5398fbb6, + 0x2d2d577f, 0x53537f57, 0x2d2d577f, 0x33537f57, 0x23233e56, 0x3433563e, 0x21213d55, 0x9201553d, 0x5a805226, 0x2652845f, 0x0d6b0f16, 0xe5767414, + 0x126b1011, 0x02000013, 0xecff7700, 0xcb057b03, 0x3a002d00, 0x32405500, 0x4023702e, 0x23481410, 0x06051123, 0x6e28353c, 0x11101419, 0x11301120, + 0x11901180, 0x15181105, 0x11147636, 0x30002828, 0x00071e76, 0x06060b75, 0x3f00190b, 0xe1102f33, 0x3912e13f, 0xe133332f, 0x2f013232, 0xe133c55d, + 0xc1d61032, 0x2b2f3912, 0x253031e1, 0x37023e32, 0x23030e33, 0x35022e22, 0x07060635, 0x37363635, 0x023e3411, 0x021e3233, 0x020e1415, 0x1e141107, + 0x23341302, 0x15020e22, 0x02033e11, 0x2e39206f, 0x0464031e, 0x46694626, 0x30516a3a, 0x3431612e, 0x401c2d5f, 0x57364b68, 0x5c35213d, 0x2411477c, + 0x20666337, 0x36091729, 0x7718334e, 0x3b523216, 0x305a8353, 0x63885425, 0x0c1c11e7, 0x0f1e0e79, 0x6c3bee01, 0x4f2a3153, 0xa663456f, 0xfe266886, + 0x3b5130d3, 0xbc210421, 0x2a45321b, 0x4e216afe, 0x04007962, 0x0000c700, 0xb6058907, 0x2b001700, 0x3b003700, 0x7240ba00, 0x5a15010e, 0x2c393900, + 0x001022e1, 0x0100b001, 0x3001009f, 0x02004000, 0x22002200, 0x18e1320a, 0x3a1f3a0f, 0x3a033a2f, 0x0118ef3a, 0x6f3d6718, 0xcf3dbf3d, 0x3d40033d, + 0x03080c01, 0x640a5a09, 0x1de52f3c, 0xcf27e535, 0xef1ddf1d, 0x401d031d, 0x0f480b06, 0x5f271f27, 0x7f276f27, 0x0627ef27, 0x1d271d06, 0x10031627, + 0x03481806, 0xe539030b, 0xffb80e38, 0x1806b6f0, 0x0a010e48, 0x333f0012, 0xe0332b33, 0x332b333f, 0x2f2f3939, 0x5d2b5d5e, 0xe110e110, 0xe1f61001, + 0x5d323232, 0x5de6105d, 0x10712f32, 0x393911e1, 0x5d5d2f2f, 0xe110715d, 0xe1102f33, 0x30313232, 0x23012321, 0x16161716, 0x11231115, 0x26330133, + 0x35032e27, 0x14013311, 0x2223020e, 0x3435022e, 0x3233023e, 0x1405021e, 0x36323316, 0x23263435, 0x35030622, 0xa8041521, 0x0887fdcd, 0x05040406, + 0x7602cfa6, 0x01040306, 0xa4010203, 0x4d29e102, 0x6b3f446d, 0x4d292c4d, 0x6b3e446d, 0x42fe2c4e, 0x47515148, 0x48515147, 0x04f00163, 0x414c4dba, + 0xe7fc398e, 0x4cfbb605, 0x43204a4c, 0x031a3e43, 0x53b9fc20, 0x2f2f5982, 0x53538259, 0x2e2e5881, 0x71538158, 0x72717373, 0x1ffd6d6d, 0x02009393, + 0xe5022500, 0xb6055005, 0x20000700, 0x5d40a300, 0x1301c400, 0x11c4100a, 0x06d006c0, 0x6f0306e0, 0x06300106, 0x06500640, 0x01061f03, 0x20010604, + 0x0211d011, 0x11110601, 0x03030106, 0x1815191e, 0xaf179fc4, 0x22170217, 0x222f220f, 0x2203223f, 0x48181540, 0x120f4022, 0xef03cf48, 0x1f030203, + 0x0c200909, 0x07094818, 0x1804c803, 0xe0ffb814, 0x180c0c40, 0x08111448, 0x12150101, 0x3f000304, 0x2f333333, 0x2b333333, 0x32e11033, 0x33112b33, + 0x2b5d2f01, 0xd6105d2b, 0x3232e15d, 0x39171132, 0x5d2f2f2f, 0x5d5d5d5d, 0x32e1105d, 0x31e11032, 0x11230130, 0x15213523, 0x23030123, 0x1115031e, + 0x13331123, 0x23113313, 0x023e3411, 0x01032337, 0x02c47f68, 0x4002c70a, 0x020107ba, 0xba7b0101, 0x7fb2bfb4, 0x01010101, 0xe502c306, 0x6c6c6502, + 0x25029bfd, 0x181e2210, 0x0249fe06, 0x02d9fdd1, 0x012ffd27, 0x231e09ac, 0xdbfd0c23, 0x4e00ffff, 0xa6050000, 0x0602cd05, 0x00007601, 0x66000200, + 0x8b04ddff, 0x22004804, 0x3b002f00, 0x4a232140, 0x2f31111e, 0x05104a13, 0x1d1d0501, 0x2b4e1218, 0x022f3b2f, 0x29182f2f, 0x180f0c4e, 0x0016004e, + 0xe13fe13f, 0x5d2f3912, 0x2f3911e1, 0xe15d2f01, 0x32de1032, 0x053031e1, 0x35022e22, 0x33043e34, 0x15021e32, 0x031e1121, 0x023e3233, 0x030e1737, + 0x032e1113, 0x020e2223, 0x79021107, 0x4586c682, 0x76664c2c, 0xc2713f80, 0xc5fc518e, 0x584d4016, 0x5e744a2e, 0x2448224d, 0xcc8f6e53, 0x5d4d3b13, + 0x49573335, 0x5e23183c, 0x636ecc9d, 0x3c5c7ea0, 0xd1934f1d, 0x189cfe83, 0x2015222c, 0x2937573c, 0x2c4d6539, 0x15018b02, 0x16232a14, 0x172a2113, + 0xffffe9fe, 0xecff5300, 0xb605d305, 0x7b002600, 0x27000014, 0x5e021702, 0x07010000, 0x62034002, 0x3000b3fd, 0x03041f40, 0x01192702, 0xe40111f4, + 0x11d00111, 0x0111b001, 0x300111a0, 0x2f110111, 0x590f0159, 0x115d5d01, 0x5d5d5d5d, 0x00355d5d, 0x3535353f, 0x3300ffff, 0xd305ecff, 0x2600c905, + 0x00147500, 0x17022700, 0x00008b02, 0x40020701, 0xb3fd6203, 0x28403c00, 0x49020304, 0x33f00119, 0x0133d001, 0xb40133c4, 0x33a00133, 0x01339401, + 0x30013360, 0x33240133, 0x7b2f3301, 0x017b0f01, 0x5d115d5d, 0x5d5d5d5d, 0x5d5d5d5d, 0x353f0035, 0xffff3535, 0xecff4f00, 0xb605d305, 0x3d022600, + 0x27000014, 0x8b021702, 0x07010000, 0x62034002, 0x3c00b3fd, 0x03042840, 0x01193d02, 0xd00127f0, 0x27c40127, 0x0127b401, 0x940127a0, 0x27600127, + 0x01273001, 0x27012724, 0x0f016f2f, 0x5d5d016f, 0x5d5d5d11, 0x5d5d5d5d, 0x00355d5d, 0x3535353f, 0x7100ffff, 0xd305ecff, 0x2600b605, 0x00423f02, + 0x17022700, 0x00003902, 0x40020701, 0xb3fd6203, 0x22403400, 0x1f020304, 0x09f40119, 0x0109eb01, 0xb40109cb, 0x099b0109, 0x01097f01, 0x0901092b, + 0x0f01512f, 0x5d5d0151, 0x5d5d5d11, 0x5d5d5d5d, 0x353f0035, 0x02003535, 0xecff6200, 0xc5053b04, 0x43002b00, 0x21403b00, 0x00471933, 0x45404557, + 0x01450f01, 0x21483f06, 0x3844560c, 0x2c13134f, 0x27501c21, 0x07502c04, 0xe13f0016, 0x1233e13f, 0x01e12f39, 0xe132f610, 0x105d5d5e, 0x3132e1f6, + 0x0e140130, 0x2e222304, 0x3e343502, 0x16323304, 0x35343617, 0x22232634, 0x3507020e, 0x3233033e, 0x3201021e, 0x2e37043e, 0x0e222303, 0x1e141504, + 0x203b0402, 0xb18a6541, 0x538c6a6d, 0x51321722, 0x5b629973, 0x8b022d93, 0x43441f89, 0x43181b41, 0x7d254f4b, 0xfd2a63a6, 0x515e359e, 0x0a253543, + 0x493b290a, 0x4f663e2a, 0x1512253a, 0xa603462d, 0xbcd4e16a, 0x6e42528d, 0x8d3c4c8f, 0x3d65838d, 0x2a13454f, 0x0cc9be0b, 0xae131f16, 0x0a11170c, + 0xfcc49659, 0x684d2c73, 0x2843857a, 0x301e3345, 0x6e6f6851, 0x4056322f, 0x02000024, 0x00002900, 0xb6056204, 0x0e000500, 0x11405800, 0x09090102, + 0x605b0e05, 0xb0047004, 0x0404f004, 0xc0ffb804, 0x0a062740, 0x10040448, 0x105f102f, 0x107f106f, 0x10bf109f, 0x06401006, 0x5b0d4809, 0x055f0d05, + 0x0a200912, 0x01094811, 0x333f0003, 0x01e13f2b, 0x5d2be12f, 0x2b2f3311, 0x3912e15d, 0x33332f3d, 0x01373031, 0x21150133, 0x27262601, 0x03070606, + 0xbe012921, 0xfbc001bb, 0x207702c7, 0x2a110f2d, 0xa602fa1d, 0xfa450571, 0x9a036fb9, 0x4b4ba861, 0x04fd5ba8, 0xc7000100, 0x270514fe, 0x0700b605, + 0x11403700, 0x65075a00, 0x30090009, 0x70094009, 0x05098009, 0xc0ffb809, 0x18140f40, 0x045a0348, 0x5f020864, 0x00040305, 0x323f001b, 0x1001e13f, + 0x5d2be1f6, 0x31e1f610, 0x21110130, 0x21112311, 0xfd6d0411, 0x6004ba14, 0xfc0614fe, 0xa20704f9, 0x01005ef8, 0x14fe4a00, 0xb605e304, 0x5c000b00, + 0x5b080b40, 0x02010670, 0x00060206, 0xc0ffb80a, 0x480e09b7, 0x070d0a0a, 0xe0ffb809, 0x180f0940, 0x00030948, 0xb80d0c00, 0x0f40c0ff, 0x08481813, + 0x07090202, 0x0903045f, 0x001b005f, 0xe13fe13f, 0x2f3d3912, 0x112b0133, 0x332f1833, 0x11322b33, 0x122b2f33, 0x2f2f3939, 0x3031e15d, 0x01013513, + 0x21152135, 0x15210101, 0xfd70024a, 0xfc4804a0, 0xfd3a02bc, 0xfe9b03b0, 0x92037314, 0xa4722b03, 0x9dfc09fd, 0x010000a4, 0x87026600, 0x1d030204, + 0x32000300, 0x05021f40, 0x8b010096, 0x00790100, 0x01005601, 0x3801004b, 0x00140100, 0x01000901, 0x01ad0000, 0xe13f00b3, 0x5d5d2f01, 0x5d5d5d5d, + 0xce105d5d, 0x35133031, 0x03661521, 0x9687029c, 0x01000096, 0xf2ff2500, 0xa006c504, 0x2c000800, 0x00011340, 0x06020506, 0x08070602, 0x03030810, + 0x070404ae, 0x2f2f0000, 0x01e12f39, 0x33382f2f, 0x2f193939, 0x11332f18, 0x30313333, 0x23012305, 0x01132135, 0x85730233, 0x01b4ebfe, 0x0002e529, + 0x0a030e92, 0x0569fd8f, 0x030000ac, 0x91017700, 0x0e043105, 0x33002300, 0x76004300, 0x17494f40, 0x010d4801, 0x2f0145f0, 0x02455f45, 0x274a273a, + 0x2a03276a, 0x1a370137, 0x3f042708, 0x2f972f77, 0x20aa2f02, 0x02123012, 0x013f7812, 0x3400ab3f, 0x3a17ae2a, 0x3708ae24, 0x17041a27, 0x100d0005, + 0x0d07020d, 0x179f1f0d, 0x01172001, 0x5d2f0017, 0x2f33335d, 0x12335d5e, 0x32e13917, 0x0132e110, 0x2f5de12f, 0x125de15d, 0x5d5d3917, 0x30315d5d, + 0x14015d5d, 0x2223020e, 0x030e2726, 0x022e2223, 0x023e3435, 0x17163233, 0x3233033e, 0x3205021e, 0x26263736, 0x020e2223, 0x021e1415, 0x07062201, + 0x32331616, 0x3435023e, 0x3105022e, 0x426d4d2b, 0x1d419b5d, 0x2b534e46, 0x2d4f6e41, 0x436f4e2b, 0x1d3e9e55, 0x30574f44, 0x2b4d6d42, 0x6c3f7bfc, + 0x456b3134, 0x182c4028, 0x02412c15, 0x376b3f7c, 0x27446c33, 0x19182d40, 0xcd02402e, 0x3457723f, 0x4f307369, 0x522c1e38, 0x73414875, 0x706b3156, + 0x1f384e30, 0xf975522d, 0x5a5e6157, 0x2644331d, 0x1e324224, 0x61576a01, 0x331d5c5d, 0x43262744, 0x01001c31, 0x14fe1000, 0x14060603, 0x37002300, + 0x25102340, 0x140225d0, 0x04060104, 0xac0d0401, 0x0901151b, 0x10150115, 0x021e201e, 0x12ae191e, 0x00ae071c, 0xe13f0000, 0x2f01e13f, 0x5d5d335d, + 0x5d5d32e1, 0x0130315d, 0x15171632, 0x22232626, 0x1115020e, 0x23020e14, 0x35272622, 0x32331616, 0x1135023e, 0x02023e34, 0x164b2283, 0x33243d12, + 0x32102742, 0x244a7c59, 0x3e14174b, 0x2a453323, 0x79572f12, 0x09091406, 0x27110993, 0xfa2d5441, 0x56865ed7, 0x93080b29, 0x40251008, 0x27053054, + 0x2856865e, 0x66000200, 0x02047b01, 0x23002504, 0x4b004700, 0x1d412e40, 0x01491049, 0x200a102e, 0x400a020a, 0xaf31ad29, 0x0aad1743, 0x051caf1f, + 0xef0ddfad, 0x030dff0d, 0x120f400d, 0x3b0d0d48, 0xb3432ead, 0xe1333f00, 0x5d2b2f33, 0x32f533e1, 0xe1f510e1, 0x5d2f0133, 0xce105d33, 0x01303132, + 0x2223032e, 0x3507020e, 0x1e323336, 0x031e1702, 0x023e3233, 0x23061537, 0x03022e22, 0x2223032e, 0x3507020e, 0x1e323336, 0x031e1702, 0x023e3233, + 0x23061537, 0x02022e22, 0x2d372512, 0x3c1c1629, 0x6419383b, 0x37321d94, 0x37252f43, 0x1c16282f, 0x18383b3c, 0x321d9563, 0x252f4337, 0x16292d37, + 0x383b3c1c, 0x1d946419, 0x2f433732, 0x282f3725, 0x3b3c1c16, 0x97611838, 0x4337321d, 0x1610bc01, 0x2113050d, 0x6ca2192c, 0x14190d05, 0x050d1610, + 0x192c2113, 0x0d056ca2, 0x10ae0119, 0x13050d15, 0xa21a2c20, 0x190e056d, 0x0d151014, 0x2d201305, 0x056ca219, 0x0100190d, 0xa4006600, 0x04050204, + 0xa6001300, 0x0d0a1140, 0x0912110e, 0x03001212, 0x13080704, 0xb8130908, 0x4f40f0ff, 0x09131009, 0x10060913, 0x1540150b, 0x06c60101, 0x0106bb01, + 0x860106a9, 0x067b0106, 0x01066801, 0x39010642, 0x09060106, 0x040d0808, 0x121307ad, 0xad001112, 0x2f031f0e, 0x037f0203, 0x10030001, 0x03060203, + 0x07f00a03, 0x6f070f01, 0x00070207, 0x335d5d2f, 0x5d5e2f33, 0xe133715d, 0x33113232, 0x3332e110, 0x2f013311, 0x5d5d5d5d, 0x5d5d5d5d, 0xce105d33, + 0x39391132, 0x38382f2f, 0x877d3311, 0xc4c4c4c4, 0x87103301, 0xc4c4c4c4, 0x23013031, 0x21132135, 0x17132135, 0x21153307, 0x21152103, 0x5e012703, + 0x793e01f8, 0xfc0149fe, 0xfa698a85, 0x017bc1fe, 0x8100feba, 0x96ba0189, 0x01950401, 0x95e03b1b, 0xfe96fcfe, 0x020039ea, 0x00006600, 0xdd040204, + 0x0a000600, 0x32405200, 0x400c0009, 0x4008010c, 0x02010101, 0x03050501, 0x7f006f06, 0x00300200, 0x04000001, 0x50010320, 0x80037003, 0xf003d003, + 0x033f0503, 0x01030001, 0x2f000306, 0x5d5d5d5e, 0x2f333371, 0x12325d5d, 0x332f3d39, 0x2f180133, 0x105d335d, 0x303132ce, 0x01350125, 0x35020915, + 0x02041521, 0x9c0364fc, 0xdf0221fd, 0x9c0364fc, 0x66a801ee, 0xfea0e101, 0xfebefe94, 0x00969671, 0x66000200, 0x02040000, 0x0600dd04, 0x52000a00, + 0x05093240, 0x010c400c, 0x01064008, 0x01040506, 0x6f000301, 0x02067f06, 0x06010630, 0x03200206, 0x70035001, 0xd0038003, 0x0503f003, 0x0001033f, + 0x03060103, 0x5d5e2f00, 0x33715d5d, 0x5d5d2f33, 0x3d391233, 0x0133332f, 0x335d2f18, 0x32ce105d, 0x01133031, 0x15013501, 0x21351501, 0xe2026615, + 0x9c031efd, 0x9c0364fc, 0x42018f01, 0xfea26a01, 0x58fe661f, 0x009696ee, 0x6d000200, 0x3f040000, 0x0500c305, 0x5d000900, 0x01023640, 0x07040509, + 0x06030709, 0xff0b00ac, 0x0b40010b, 0x0b800b50, 0x0bc00bb0, 0x2f0b0f05, 0xaa08020b, 0x03010310, 0x06000803, 0x06080608, 0x04ad0702, 0x02ad0907, + 0x3fe12f00, 0x393912e1, 0x112f2f3d, 0x01331133, 0xe15d2f18, 0x105d5d5d, 0x3912e1de, 0x33331139, 0x31333311, 0x23010130, 0x09330101, 0xfe3f0403, + 0x3efe4c3c, 0x014cc201, 0xfecffe0c, 0x023101cf, 0x021ffde1, 0xfde402df, 0xfe00021e, 0x00fefd00, 0x1d00ffff, 0x17040000, 0x26001f06, 0x00004900, + 0x4c000701, 0x0000a202, 0x18402400, 0x1daf0102, 0x011d5001, 0x0f011d1f, 0xaf1d011d, 0x2f40012f, 0x012f0f01, 0x115d5d5d, 0x5d5d5d5d, 0xffff3535, + 0x00001d00, 0x1f060604, 0x49002600, 0x07010000, 0xa2024f00, 0x22000000, 0xaf011740, 0x1d50011d, 0x011d1f01, 0x1d011d0f, 0x400121af, 0x210f0121, + 0x5d5d5d01, 0x5d5d5d11, 0x0100355d, 0xd904cf00, 0x1706cd03, 0x3f001500, 0x17002940, 0xaf021720, 0x0217cf17, 0x17701730, 0x01000f02, 0x0a000000, + 0x06020a30, 0x007f0a0a, 0x06400001, 0x00004809, 0x00058e10, 0x2f33e12f, 0x01335d2b, 0x335d5e2f, 0x5d5d5d2f, 0x01303171, 0x2223030e, 0x3327022e, + 0x3233031e, 0x0337023e, 0x5d3106cd, 0x8c67648d, 0xaa052958, 0x4c321b05, 0x36492c35, 0x17060622, 0x2951764e, 0x50775027, 0x10294939, 0x35482b13, + 0xbcff0100, 0x640114fe, 0x13004a04, 0x13402300, 0x15201510, 0x0c470f02, 0x540c0303, 0x070f0d14, 0x001b0050, 0x013fe13f, 0x2f32e610, 0x315de110, + 0x26221330, 0x16163527, 0x023e3233, 0x11331135, 0x42020e14, 0x1a173f30, 0x2e1b2336, 0x22b61323, 0x14fe6d48, 0x0a940b0e, 0x41270f0b, 0xfbf40433, + 0x577b4d18, 0x0100002f, 0xcd048701, 0x14067902, 0x19000d00, 0x85050c40, 0x0601063f, 0x920c000c, 0x3f000005, 0x332f01e5, 0x31e15ddd, 0x033e0130, + 0x0e153337, 0x01230703, 0x11120987, 0x06b4040e, 0x162e291f, 0x1ee70460, 0x2150514d, 0x564e1914, 0x01002056, 0x3bfe6f01, 0x83ff7502, 0x16000d00, + 0x000c0940, 0x06850500, 0x0005920c, 0x2f01e52f, 0x332f32e1, 0x3e013031, 0x15333703, 0x2307030e, 0x1a0b6f01, 0xb2041318, 0x362c1f06, 0x56fe621d, + 0x53544c19, 0x4d191421, 0x00215657, 0x7d010100, 0x8302d904, 0x0d002106, 0x0a401700, 0x0500000c, 0x92050685, 0x3f00000c, 0xe12f01e5, 0x31332f32, + 0x030e0130, 0x3e352307, 0x02333703, 0x181a0c83, 0x05b20412, 0x1c372d1f, 0x19060662, 0x2153544c, 0x564e1815, 0x02002156, 0x39022500, 0xc7057f02, + 0x1d000b00, 0x19402a00, 0x1f0ce106, 0x1f7f1f4f, 0x06401f02, 0xe100480a, 0x1be50916, 0x11e503de, 0xe13f00df, 0x2f01e13f, 0x105d2be1, 0x3031e1de, + 0x33161413, 0x34353632, 0x06222326, 0x020e1405, 0x022e2223, 0x023e3435, 0xb8163233, 0x4e4e5147, 0x47514e4e, 0x4a23c701, 0x704a4f73, 0x4923264b, + 0x9a954e71, 0xa1a50204, 0x9fa5a79f, 0xaa6ca59f, 0x753e3e75, 0xa96c6caa, 0x00ec3c74, 0x0c000200, 0x8f024a02, 0x0a00bc05, 0x46001500, 0x02092a40, + 0x03070be1, 0x175f1703, 0x1702178f, 0x480a0640, 0x0105e115, 0x0f09e504, 0x2f0b1f0b, 0x0b08030b, 0xe50f020b, 0xdd02dc07, 0xe13f3f00, 0x5e2f3912, + 0x32e1335d, 0x2be12f01, 0x2f39125d, 0x32e13333, 0x23013031, 0x21352315, 0x11330135, 0x34352133, 0x030e3736, 0x8f020707, 0x89fe8f7d, 0x7d8d7901, + 0x0303f4fe, 0x18161405, 0x0a039b09, 0x026fc0c0, 0xc3cdfd43, 0x0b31632a, 0x0f282a25, 0x010000f0, 0x37023b00, 0xb6056602, 0x4a002400, 0x22211740, + 0x1d1d1e22, 0xe114200b, 0x264f2605, 0x2602267f, 0x480a0640, 0xc0ffb80b, 0x18121140, 0xe5170b48, 0x21110000, 0x11dc1ee5, 0x00df08e5, 0xe13fe13f, + 0xe12f3912, 0x2b2b2f01, 0xe1de105d, 0x2f391233, 0x33113333, 0x32013031, 0x1415021e, 0x26222306, 0x031e3527, 0x35363233, 0x22232634, 0x2707020e, + 0x21152113, 0x01363607, 0x4f6b3d42, 0x3fa1a62d, 0x3c1a2c79, 0x55183b3d, 0x0d595f5f, 0x0e252725, 0xba012143, 0x1412befe, 0x236d0439, 0x8c416544, + 0x8d1a1c9d, 0x0a131b12, 0x554d584c, 0x02070604, 0x7ba8012b, 0x000603d7, 0x25000200, 0x7f023902, 0x1d00d505, 0x38002d00, 0x0a102040, 0x2f00e121, + 0x2f7f2f4f, 0x06402f02, 0x2915480a, 0xe4240ae1, 0x10191915, 0xe51ede0f, 0x3f00df05, 0x39333fe1, 0x01e1332f, 0x2b32e12f, 0xe1de105d, 0x30313912, + 0x020e1401, 0x022e2223, 0x023e3435, 0x030e1737, 0x36363307, 0x021e3233, 0x35363205, 0x22232634, 0x1415020e, 0x7f02021e, 0x446f4e2a, 0x2f526e40, + 0x6eb27e45, 0x607f4f2d, 0x1b0b0f40, 0x5a364a63, 0xdbfe2440, 0x474d5444, 0x192d3f27, 0x033d2d19, 0x506f4166, 0x80582d2d, 0x8fb36e53, 0x1b6d276d, + 0x3b604f40, 0x4825312b, 0x565cf769, 0x29195a52, 0x492c1e37, 0x01001d35, 0x4a022f00, 0xb6056402, 0x38000600, 0xff0000b9, 0x001f40f0, 0xe1010200, + 0x084f0805, 0x0802087f, 0x480a0640, 0x02e002d0, 0x020302f0, 0xdc03e502, 0x3f00dd00, 0x2f01e13f, 0x105d2b5d, 0x3911e1de, 0x3031382f, 0x35210113, + 0x8d011521, 0x62fe4001, 0xbffe3502, 0xf1024a02, 0xf8fc647b, 0x31000300, 0x71023902, 0x2500c705, 0x44003400, 0x22407700, 0x4221e138, 0x3d0a05e1, + 0x05041c32, 0x05210521, 0x0fe12c17, 0x7f464f46, 0x40460246, 0x26480a06, 0xffb817e1, 0x152740c0, 0x0a174818, 0x4b323d1c, 0x6b325b32, 0xab329b32, + 0x14320532, 0xc63db635, 0x033dd63d, 0xe529003d, 0xe535df14, 0x3f00de00, 0x11e13fe1, 0x12115d39, 0x12115d39, 0x2f013939, 0x5d2be12b, 0x12e1de10, + 0x2f2f3939, 0x10391712, 0x31e110e1, 0x1e320130, 0x0e141502, 0x031e0702, 0x020e1415, 0x35262223, 0x37023e34, 0x3435032e, 0x1403023e, 0x36323316, + 0x022e3435, 0x06062727, 0x15062213, 0x17021e14, 0x3435033e, 0x34520126, 0x162a465e, 0x221e3427, 0x2a1c303e, 0x8b3f694d, 0x382a1996, 0x232e1b1f, + 0x5f482b13, 0x4a484a62, 0x3727164b, 0x42410f21, 0x123f3894, 0x181a2d20, 0x4113202c, 0x371bc705, 0x3e253854, 0x13112832, 0x2a45372d, 0x24425d39, + 0x452a7383, 0x14112b38, 0x253e352b, 0x1c375338, 0x463b68fd, 0x301e3b46, 0x060f2027, 0xe8014d22, 0x2a1b3637, 0x0c0d1d24, 0x1c2c231c, 0x02003736, + 0x39022100, 0xc9057b02, 0x35002500, 0x22403b00, 0x09e01131, 0x4f37001c, 0x02377f37, 0x0a064037, 0x1ce12948, 0x1711e42c, 0xe5260517, 0xe50cde21, + 0x3f00df05, 0x12e13fe1, 0xe1332f39, 0x2be12f01, 0x11de105d, 0x3132e139, 0x0e140130, 0x26222302, 0x16163527, 0x023e3233, 0x030e2337, 0x022e2223, + 0x023e3435, 0x021e3233, 0x15062225, 0x32331614, 0x3435023e, 0x7b02022e, 0x81a25a21, 0x15154019, 0x6c531936, 0x09031d40, 0x3d30240d, 0x44603c26, + 0x6c4b2924, 0x54724143, 0x44cbfe30, 0x264c4a56, 0x15182d3f, 0x42043e2a, 0x5b93bb60, 0x0d7d0708, 0x7257350a, 0x1b24123d, 0x66472511, 0x516f4241, + 0x93612f2d, 0x4c585eac, 0x33291958, 0x3b48241a, 0x16000024, 0x81fe5400, 0xee05c107, 0x0b000500, 0x17001100, 0x1f001b00, 0x27002300, 0x2f002b00, + 0x37003300, 0x3f003b00, 0x47004300, 0x5f005300, 0x78006f00, 0x90008100, 0x5940e400, 0x1c202c34, 0x51571838, 0x746f707a, 0x82646467, 0x51706b7e, + 0x51d051c0, 0x6b6f5103, 0x03516f6b, 0x8c4f895d, 0x8c028c5f, 0x2428448c, 0x505d1709, 0x024b604b, 0x3d414b4b, 0x2a0c0031, 0x263e4642, 0x8f8f8632, + 0x7067796f, 0x7a607a70, 0x4848546f, 0x6f806f70, 0x016f6102, 0xc0ffb86f, 0x0c072540, 0x606f6f48, 0x8a4e4e5a, 0x2f601f78, 0x6f603f60, 0x60600460, + 0x181c2c01, 0x35390c12, 0x010f0721, 0x00010701, 0x335d5e2f, 0x2f333333, 0x33333333, 0x5d2f3912, 0x2f333333, 0x2f331133, 0x335d5d2b, 0x3311332f, + 0x2f391211, 0x33113333, 0x332f332f, 0x332f332f, 0x33332f01, 0x2f333333, 0x332f335d, 0x33333333, 0x12335d2f, 0x2f2f3917, 0x33115d2f, 0x332f3333, + 0x33331133, 0x332f3311, 0x332f332f, 0x11133031, 0x15231521, 0x11213525, 0x11013523, 0x15331533, 0x35333521, 0x35211133, 0x35211521, 0x35011521, + 0x23011521, 0x23113311, 0x35013311, 0x23011521, 0x35013311, 0x35331521, 0x23011521, 0x23353311, 0x23013311, 0x22013311, 0x36343526, 0x15163237, + 0x32270614, 0x26343536, 0x15062223, 0x33011614, 0x14151632, 0x16150706, 0x06141516, 0x33132323, 0x34353632, 0x15232326, 0x36323315, 0x23263435, + 0x16163505, 0x35363233, 0x14113311, 0x54222306, 0x05c02f01, 0x6d3001ce, 0xc06f00f9, 0x6dc30e05, 0x110149fd, 0x0e01e1fb, 0x0e01f2fe, 0x6d6db704, + 0xc2fb6d6d, 0x30fc1001, 0xc0026f6f, 0x01771001, 0x6fa8fa11, 0x066f6f6f, 0xfa6d6dfe, 0x87877f99, 0x87877f7f, 0x3f3f487f, 0x42424548, 0x6dac9f01, + 0x382d2d70, 0xcf5e6d33, 0x242e427b, 0x4a3b3029, 0x34262531, 0x20100e01, 0x7d312514, 0x043d5f68, 0x6f3001be, 0xfe6fc1c1, 0x02f9c1d0, 0x6dc22f01, + 0xd1fec26d, 0x6d6d6d6d, 0x6f6ffe06, 0x0e01a8fa, 0x0f010202, 0x6d6d3bfa, 0x0e01a601, 0x6f6f4a04, 0x2ffc6f6f, 0x01791001, 0x0168fd0f, 0x9f8afe10, + 0x019b918e, 0x9f8e919c, 0x5e5e6768, 0x5e5e6666, 0x43ea0167, 0x08443153, 0x3a440b04, 0x62015951, 0x1d222022, 0x252b9ae3, 0x66fc2a20, 0x32240503, + 0x72fe9201, 0x0300645e, 0xc1fe5400, 0x1406aa07, 0x23000300, 0x50002f00, 0x23042840, 0x2a202423, 0x2a402a30, 0x4f2a2a03, 0x24240124, 0x0b0b1c02, + 0x02151502, 0x2d302323, 0x272d2d01, 0x10190327, 0x19000310, 0x2f18332f, 0x2f331133, 0x335d2f33, 0x2f19012f, 0x112f1833, 0x11332f33, 0x335d2f33, + 0x33115d2f, 0x3031332f, 0x35050309, 0x36373634, 0x2e343536, 0x0e222302, 0x36170702, 0x16323336, 0x07061415, 0x15150606, 0x33161403, 0x34353632, + 0x06222326, 0xac03fe03, 0x56fc54fc, 0x4c21eb03, 0x5b314d63, 0x5a2b5183, 0x52225257, 0x3f387e44, 0x4552273e, 0x46471b4a, 0x44474744, 0x14064746, + 0x57fc56fc, 0x2ffba903, 0x4c3a3e2c, 0x6b455983, 0x1b10274a, 0x22b21423, 0x312f3a2e, 0x79354144, 0xedfe3b50, 0x3e49493e, 0x00494940, 0x01000000, + 0x7c370000, 0x3d090100, 0x0b000030, 0x05006e07, 0xaeff2400, 0x2c000500, 0x05002900, 0x52003700, 0x39000500, 0x05005200, 0x66003a00, 0x3b000500, + 0x05002900, 0x52003c00, 0x3d000500, 0x05002900, 0xc3ff4600, 0x47000500, 0x0500c3ff, 0xc3ff4800, 0x4a000500, 0x0500d7ff, 0xc3ff5200, 0x54000500, + 0x0500c3ff, 0x29005700, 0x59000500, 0x05002900, 0x14005a00, 0x5c000500, 0x05002900, 0xaeff8200, 0x83000500, 0x0500aeff, 0xaeff8400, 0x85000500, + 0x0500aeff, 0xaeff8600, 0x87000500, 0x0500aeff, 0x5cff8800, 0x8e000500, 0x05002900, 0x29008f00, 0x90000500, 0x05002900, 0x29009100, 0x9f000500, + 0x05005200, 0xc3ffa800, 0xa9000500, 0x0500c3ff, 0xc3ffaa00, 0xab000500, 0x0500c3ff, 0xc3ffac00, 0xad000500, 0x0500c3ff, 0xc3ffb400, 0xb5000500, + 0x0500c3ff, 0xc3ffb600, 0xb7000500, 0x0500c3ff, 0xc3ffb800, 0xba000500, 0x0500c3ff, 0x2900bf00, 0xc1000500, 0x05002900, 0xc3ff1501, 0x3a010500, + 0x05005200, 0x29003f01, 0x28020500, 0x0a00aeff, 0xaeff2400, 0x2c000a00, 0x0a002900, 0x52003700, 0x39000a00, 0x0a005200, 0x66003a00, 0x3b000a00, + 0x0a002900, 0x52003c00, 0x3d000a00, 0x0a002900, 0xc3ff4600, 0x47000a00, 0x0a00c3ff, 0xc3ff4800, 0x4a000a00, 0x0a00d7ff, 0xc3ff5200, 0x54000a00, + 0x0a00c3ff, 0x29005700, 0x59000a00, 0x0a002900, 0x14005a00, 0x5c000a00, 0x0a002900, 0xaeff8200, 0x83000a00, 0x0a00aeff, 0xaeff8400, 0x85000a00, + 0x0a00aeff, 0xaeff8600, 0x87000a00, 0x0a00aeff, 0x5cff8800, 0x8e000a00, 0x0a002900, 0x29008f00, 0x90000a00, 0x0a002900, 0x29009100, 0x9f000a00, + 0x0a005200, 0xc3ffa800, 0xa9000a00, 0x0a00c3ff, 0xc3ffaa00, 0xab000a00, 0x0a00c3ff, 0xc3ffac00, 0xad000a00, 0x0a00c3ff, 0xc3ffb400, 0xb5000a00, + 0x0a00c3ff, 0xc3ffb600, 0xb7000a00, 0x0a00c3ff, 0xc3ffb800, 0xba000a00, 0x0a00c3ff, 0x2900bf00, 0xc1000a00, 0x0a002900, 0xc3ff1501, 0x3a010a00, + 0x0a005200, 0x29003f01, 0x28020a00, 0x0b00aeff, 0x7b002d00, 0x26000f00, 0x0f00c3ff, 0xc3ff2a00, 0x32000f00, 0x0f00c3ff, 0xc3ff3400, 0x37000f00, + 0x0f009aff, 0xd7ff3800, 0x39000f00, 0x0f009aff, 0xaeff3a00, 0x3c000f00, 0x0f009aff, 0xc3ff8900, 0x94000f00, 0x0f00c3ff, 0xc3ff9500, 0x96000f00, + 0x0f00c3ff, 0xc3ff9700, 0x98000f00, 0x0f00c3ff, 0xc3ff9a00, 0x9b000f00, 0x0f00d7ff, 0xd7ff9c00, 0x9d000f00, 0x0f00d7ff, 0xd7ff9e00, 0x9f000f00, + 0x0f009aff, 0x9aff3a01, 0x26001100, 0x1100c3ff, 0xc3ff2a00, 0x32001100, 0x1100c3ff, 0xc3ff3400, 0x37001100, 0x11009aff, 0xd7ff3800, 0x39001100, + 0x11009aff, 0xaeff3a00, 0x3c001100, 0x11009aff, 0xc3ff8900, 0x94001100, 0x1100c3ff, 0xc3ff9500, 0x96001100, 0x1100c3ff, 0xc3ff9700, 0x98001100, + 0x1100c3ff, 0xc3ff9a00, 0x9b001100, 0x1100d7ff, 0xd7ff9c00, 0x9d001100, 0x1100d7ff, 0xd7ff9e00, 0x9f001100, 0x11009aff, 0x9aff3a01, 0x05002400, + 0x2400aeff, 0xaeff0a00, 0x26002400, 0x2400ecff, 0xecff2a00, 0x32002400, 0x2400ecff, 0xecff3400, 0x37002400, 0x240085ff, 0xecff3800, 0x39002400, + 0x2400c3ff, 0xd7ff3a00, 0x3c002400, 0x24009aff, 0xecff8900, 0x94002400, 0x2400ecff, 0xecff9500, 0x96002400, 0x2400ecff, 0xecff9700, 0x98002400, + 0x2400ecff, 0xecff9a00, 0x9b002400, 0x2400ecff, 0xecff9c00, 0x9d002400, 0x2400ecff, 0xecff9e00, 0x9f002400, 0x24009aff, 0xd7ff1401, 0x3a012400, + 0x24009aff, 0xaeff0702, 0x0b022400, 0x2500aeff, 0xecff3700, 0x39002500, 0x2500ecff, 0xecff3b00, 0x3c002500, 0x2500ecff, 0xecff9f00, 0x3a012500, + 0x2600ecff, 0x29000500, 0x0a002600, 0x26002900, 0x29000c00, 0x26002600, 0x2600d7ff, 0xd7ff2a00, 0x32002600, 0x2600d7ff, 0xd7ff3400, 0x40002600, + 0x26002900, 0x29006000, 0x89002600, 0x2600d7ff, 0xd7ff9400, 0x95002600, 0x2600d7ff, 0xd7ff9600, 0x97002600, 0x2600d7ff, 0xd7ff9800, 0x9a002600, + 0x2600d7ff, 0x29000702, 0x0b022600, 0x27002900, 0xc3ff0f00, 0x11002700, 0x2700c3ff, 0xecff2400, 0x2c002700, 0x2700ecff, 0xc3ff3700, 0x39002700, + 0x2700ecff, 0xecff3a00, 0x3b002700, 0x2700ecff, 0xd7ff3c00, 0x3d002700, 0x2700ecff, 0xecff8200, 0x83002700, 0x2700ecff, 0xecff8400, 0x85002700, + 0x2700ecff, 0xecff8600, 0x87002700, 0x2700ecff, 0xc3ff8800, 0x8e002700, 0x2700ecff, 0xecff8f00, 0x90002700, 0x2700ecff, 0xecff9100, 0x9f002700, + 0x2700d7ff, 0xd7ff3a01, 0x3f012700, 0x2700ecff, 0xecff2802, 0x05002900, 0x29003d00, 0x3d000a00, 0x0c002900, 0x29002900, 0x9aff0f00, 0x11002900, + 0x29009aff, 0x29002200, 0x24002900, 0x2900d7ff, 0x14003900, 0x3a002900, 0x29001400, 0x14003c00, 0x40002900, 0x29002900, 0x29006000, 0x82002900, + 0x2900d7ff, 0xd7ff8300, 0x84002900, 0x2900d7ff, 0xd7ff8500, 0x86002900, 0x2900d7ff, 0xd7ff8700, 0x88002900, 0x2900c3ff, 0x14009f00, 0x3a012900, + 0x29001400, 0x3d000702, 0x0b022900, 0x29003d00, 0xd7ff2802, 0x05002c00, 0x2c002900, 0x29000a00, 0x26002c00, 0x2c00ecff, 0xecff2a00, 0x32002c00, + 0x2c00ecff, 0xecff3400, 0x89002c00, 0x2c00ecff, 0xecff9400, 0x95002c00, 0x2c00ecff, 0xecff9600, 0x97002c00, 0x2c00ecff, 0xecff9800, 0x9a002c00, + 0x2c00ecff, 0xecff1401, 0x07022c00, 0x2c002900, 0x29000b02, 0x05002e00, 0x2e002900, 0x29000a00, 0x26002e00, 0x2e00d7ff, 0xd7ff2a00, 0x32002e00, + 0x2e00d7ff, 0xd7ff3400, 0x89002e00, 0x2e00d7ff, 0xd7ff9400, 0x95002e00, 0x2e00d7ff, 0xd7ff9600, 0x97002e00, 0x2e00d7ff, 0xd7ff9800, 0x9a002e00, + 0x2e00d7ff, 0xd7ff1401, 0x07022e00, 0x2e002900, 0x29000b02, 0x05002f00, 0x2f009aff, 0x9aff0a00, 0x26002f00, 0x2f00ecff, 0xecff2a00, 0x32002f00, + 0x2f00ecff, 0xecff3400, 0x37002f00, 0x2f0085ff, 0xecff3800, 0x39002f00, 0x2f00aeff, 0xc3ff3a00, 0x3c002f00, 0x2f009aff, 0xecff8900, 0x94002f00, + 0x2f00ecff, 0xecff9500, 0x96002f00, 0x2f00ecff, 0xecff9700, 0x98002f00, 0x2f00ecff, 0xecff9a00, 0x9b002f00, 0x2f00ecff, 0xecff9c00, 0x9d002f00, + 0x2f00ecff, 0xecff9e00, 0x9f002f00, 0x2f009aff, 0xecff1401, 0x3a012f00, 0x2f009aff, 0x9aff0702, 0x0b022f00, 0x32009aff, 0xc3ff0f00, 0x11003200, + 0x3200c3ff, 0xecff2400, 0x2c003200, 0x3200ecff, 0xc3ff3700, 0x39003200, 0x3200d7ff, 0xecff3a00, 0x3b003200, 0x3200d7ff, 0xd7ff3c00, 0x3d003200, + 0x3200ecff, 0xecff8200, 0x83003200, 0x3200ecff, 0xecff8400, 0x85003200, 0x3200ecff, 0xecff8600, 0x87003200, 0x3200ecff, 0xd7ff8800, 0x8e003200, + 0x3200ecff, 0xecff8f00, 0x90003200, 0x3200ecff, 0xecff9100, 0x9f003200, 0x3200d7ff, 0xd7ff3a01, 0x3f013200, 0x3200ecff, 0xecff2802, 0x0f003300, + 0x330033ff, 0x33ff1100, 0x24003300, 0x3300aeff, 0xecff2600, 0x3b003300, 0x3300ecff, 0xecff3c00, 0x3d003300, 0x3300d7ff, 0xaeff8200, 0x83003300, + 0x3300aeff, 0xaeff8400, 0x85003300, 0x3300aeff, 0xaeff8600, 0x87003300, 0x3300aeff, 0x71ff8800, 0x89003300, 0x3300ecff, 0xecff9f00, 0x3a013300, + 0x3300ecff, 0xd7ff3f01, 0x28023300, 0x3400aeff, 0xc3ff0f00, 0x11003400, 0x3400c3ff, 0xecff2400, 0x2c003400, 0x3400ecff, 0xc3ff3700, 0x39003400, + 0x3400d7ff, 0xecff3a00, 0x3b003400, 0x3400d7ff, 0xd7ff3c00, 0x3d003400, 0x3400ecff, 0xecff8200, 0x83003400, 0x3400ecff, 0xecff8400, 0x85003400, + 0x3400ecff, 0xecff8600, 0x87003400, 0x3400ecff, 0xc3ff8800, 0x8e003400, 0x3400ecff, 0xecff8f00, 0x90003400, 0x3400ecff, 0xecff9100, 0x9f003400, + 0x3400d7ff, 0xd7ff3a01, 0x3f013400, 0x3400ecff, 0xecff2802, 0x37003500, 0x3700ecff, 0x52000500, 0x0a003700, 0x37005200, 0x9aff0f00, 0x11003700, + 0x37009aff, 0x29002200, 0x24003700, 0x370085ff, 0xc3ff2600, 0x2a003700, 0x3700c3ff, 0xc3ff3200, 0x34003700, 0x3700c3ff, 0xecff3600, 0x37003700, + 0x37001400, 0x85ff4400, 0x46003700, 0x370085ff, 0x85ff4700, 0x48003700, 0x370085ff, 0x9aff4a00, 0x50003700, 0x3700aeff, 0xaeff5100, 0x52003700, + 0x370085ff, 0xaeff5300, 0x54003700, 0x370085ff, 0xaeff5500, 0x56003700, 0x370085ff, 0xaeff5800, 0x59003700, 0x3700c3ff, 0xc3ff5a00, 0x5b003700, + 0x3700c3ff, 0xc3ff5c00, 0x5d003700, 0x3700c3ff, 0x85ff8200, 0x83003700, 0x370085ff, 0x85ff8400, 0x85003700, 0x370085ff, 0x85ff8600, 0x87003700, + 0x370085ff, 0x71ff8800, 0x89003700, 0x3700c3ff, 0xc3ff9400, 0x95003700, 0x3700c3ff, 0xc3ff9600, 0x97003700, 0x3700c3ff, 0xc3ff9800, 0x9a003700, + 0x3700c3ff, 0x85ffa200, 0xa3003700, 0x370085ff, 0x85ffa400, 0xa5003700, 0x370085ff, 0x85ffa600, 0xa7003700, 0x370085ff, 0x85ffa800, 0xa9003700, + 0x370085ff, 0x85ffaa00, 0xab003700, 0x370085ff, 0x85ffac00, 0xad003700, 0x370085ff, 0xaeffb300, 0xb4003700, 0x370085ff, 0x85ffb500, 0xb6003700, + 0x370085ff, 0x85ffb700, 0xb8003700, 0x370085ff, 0x85ffba00, 0xbb003700, 0x3700aeff, 0xaeffbc00, 0xbd003700, 0x3700aeff, 0xaeffbe00, 0xbf003700, + 0x3700c3ff, 0xc3ffc100, 0x14013700, 0x3700c3ff, 0x85ff1501, 0x22013700, 0x3700ecff, 0x85ff2301, 0x40013700, 0x3700c3ff, 0x52000702, 0x0b023700, + 0x37005200, 0x85ff2802, 0x0f003800, 0x3800d7ff, 0xd7ff1100, 0x24003800, 0x3800ecff, 0xecff8200, 0x83003800, 0x3800ecff, 0xecff8400, 0x85003800, + 0x3800ecff, 0xecff8600, 0x87003800, 0x3800ecff, 0xd7ff8800, 0x28023800, 0x3900ecff, 0x52000500, 0x0a003900, 0x39005200, 0x9aff0f00, 0x11003900, + 0x39009aff, 0x29002200, 0x24003900, 0x3900c3ff, 0xd7ff2600, 0x2a003900, 0x3900d7ff, 0xd7ff3200, 0x34003900, 0x3900d7ff, 0xc3ff4400, 0x46003900, + 0x3900c3ff, 0xc3ff4700, 0x48003900, 0x3900c3ff, 0xc3ff4a00, 0x50003900, 0x3900d7ff, 0xd7ff5100, 0x52003900, 0x3900c3ff, 0xd7ff5300, 0x54003900, + 0x3900c3ff, 0xd7ff5500, 0x56003900, 0x3900d7ff, 0xd7ff5800, 0x82003900, 0x3900c3ff, 0xc3ff8300, 0x84003900, 0x3900c3ff, 0xc3ff8500, 0x86003900, + 0x3900c3ff, 0xc3ff8700, 0x88003900, 0x390085ff, 0xd7ff8900, 0x94003900, 0x3900d7ff, 0xd7ff9500, 0x96003900, 0x3900d7ff, 0xd7ff9700, 0x98003900, + 0x3900d7ff, 0xd7ff9a00, 0xa2003900, 0x3900c3ff, 0xc3ffa300, 0xa4003900, 0x3900c3ff, 0xc3ffa500, 0xa6003900, 0x3900c3ff, 0xc3ffa700, 0xa8003900, + 0x3900c3ff, 0xc3ffa900, 0xaa003900, 0x3900c3ff, 0xc3ffab00, 0xac003900, 0x3900c3ff, 0xc3ffad00, 0xb3003900, 0x3900d7ff, 0xc3ffb400, 0xb5003900, + 0x3900c3ff, 0xc3ffb600, 0xb7003900, 0x3900c3ff, 0xc3ffb800, 0xba003900, 0x3900c3ff, 0xd7ffbb00, 0xbc003900, 0x3900d7ff, 0xd7ffbd00, 0xbe003900, + 0x3900d7ff, 0xd7ff1401, 0x15013900, 0x3900c3ff, 0xd7ff2301, 0x07023900, 0x39005200, 0x52000b02, 0x28023900, 0x3a00c3ff, 0x66000500, 0x0a003a00, + 0x3a006600, 0xaeff0f00, 0x11003a00, 0x3a00aeff, 0xd7ff2400, 0x26003a00, 0x3a00ecff, 0xecff2a00, 0x32003a00, 0x3a00ecff, 0xecff3400, 0x44003a00, + 0x3a00d7ff, 0xd7ff4600, 0x47003a00, 0x3a00d7ff, 0xd7ff4800, 0x4a003a00, 0x3a00ecff, 0xecff5000, 0x51003a00, 0x3a00ecff, 0xd7ff5200, 0x53003a00, + 0x3a00ecff, 0xd7ff5400, 0x55003a00, 0x3a00ecff, 0xd7ff5600, 0x58003a00, 0x3a00ecff, 0xecff5d00, 0x82003a00, 0x3a00d7ff, 0xd7ff8300, 0x84003a00, + 0x3a00d7ff, 0xd7ff8500, 0x86003a00, 0x3a00d7ff, 0xd7ff8700, 0x88003a00, 0x3a00aeff, 0xecff8900, 0x94003a00, 0x3a00ecff, 0xecff9500, 0x96003a00, + 0x3a00ecff, 0xecff9700, 0x98003a00, 0x3a00ecff, 0xecff9a00, 0xa2003a00, 0x3a00d7ff, 0xd7ffa300, 0xa4003a00, 0x3a00d7ff, 0xd7ffa500, 0xa6003a00, + 0x3a00d7ff, 0xd7ffa700, 0xa8003a00, 0x3a00d7ff, 0xd7ffa900, 0xaa003a00, 0x3a00d7ff, 0xd7ffab00, 0xac003a00, 0x3a00d7ff, 0xd7ffad00, 0xb3003a00, + 0x3a00ecff, 0xd7ffb400, 0xb5003a00, 0x3a00d7ff, 0xd7ffb600, 0xb7003a00, 0x3a00d7ff, 0xd7ffb800, 0xba003a00, 0x3a00d7ff, 0xecffbb00, 0xbc003a00, + 0x3a00ecff, 0xecffbd00, 0xbe003a00, 0x3a00ecff, 0xecff1401, 0x15013a00, 0x3a00d7ff, 0xd7ff2301, 0x40013a00, 0x3a00ecff, 0x66000702, 0x0b023a00, + 0x3a006600, 0xd7ff2802, 0x05003b00, 0x3b002900, 0x29000a00, 0x26003b00, 0x3b00d7ff, 0xd7ff2a00, 0x32003b00, 0x3b00d7ff, 0xd7ff3400, 0x46003b00, + 0x3b00ecff, 0xecff4700, 0x48003b00, 0x3b00ecff, 0xecff5200, 0x54003b00, 0x3b00ecff, 0xd7ff8900, 0x94003b00, 0x3b00d7ff, 0xd7ff9500, 0x96003b00, + 0x3b00d7ff, 0xd7ff9700, 0x98003b00, 0x3b00d7ff, 0xd7ff9a00, 0xa8003b00, 0x3b00ecff, 0xecffa900, 0xaa003b00, 0x3b00ecff, 0xecffab00, 0xac003b00, + 0x3b00ecff, 0xecffad00, 0xb4003b00, 0x3b00ecff, 0xecffb500, 0xb6003b00, 0x3b00ecff, 0xecffb700, 0xb8003b00, 0x3b00ecff, 0xecffba00, 0x14013b00, + 0x3b00d7ff, 0xecff1501, 0x07023b00, 0x3b002900, 0x29000b02, 0x05003c00, 0x3c005200, 0x52000a00, 0x0f003c00, 0x3c009aff, 0x9aff1100, 0x22003c00, + 0x3c002900, 0x9aff2400, 0x26003c00, 0x3c00d7ff, 0xd7ff2a00, 0x32003c00, 0x3c00d7ff, 0xd7ff3400, 0x36003c00, 0x3c00ecff, 0x9aff4400, 0x46003c00, + 0x3c009aff, 0x9aff4700, 0x48003c00, 0x3c009aff, 0x9aff4a00, 0x50003c00, 0x3c00c3ff, 0xc3ff5100, 0x52003c00, 0x3c009aff, 0xc3ff5300, 0x54003c00, + 0x3c009aff, 0xc3ff5500, 0x56003c00, 0x3c00aeff, 0xc3ff5800, 0x5b003c00, 0x3c00d7ff, 0xecff5c00, 0x5d003c00, 0x3c00c3ff, 0x9aff8200, 0x83003c00, + 0x3c009aff, 0x9aff8400, 0x85003c00, 0x3c009aff, 0x9aff8600, 0x87003c00, 0x3c009aff, 0x71ff8800, 0x89003c00, 0x3c00d7ff, 0xd7ff9400, 0x95003c00, + 0x3c00d7ff, 0xd7ff9600, 0x97003c00, 0x3c00d7ff, 0xd7ff9800, 0x9a003c00, 0x3c00d7ff, 0x9affa200, 0xa3003c00, 0x3c009aff, 0x9affa400, 0xa5003c00, + 0x3c009aff, 0x9affa600, 0xa7003c00, 0x3c009aff, 0x9affa800, 0xa9003c00, 0x3c009aff, 0x9affaa00, 0xab003c00, 0x3c009aff, 0x9affac00, 0xad003c00, + 0x3c009aff, 0xc3ffb300, 0xb4003c00, 0x3c009aff, 0x9affb500, 0xb6003c00, 0x3c009aff, 0x9affb700, 0xb8003c00, 0x3c009aff, 0x9affba00, 0xbb003c00, + 0x3c00c3ff, 0xc3ffbc00, 0xbd003c00, 0x3c00c3ff, 0xc3ffbe00, 0xbf003c00, 0x3c00ecff, 0xecffc100, 0x14013c00, 0x3c00d7ff, 0x9aff1501, 0x22013c00, + 0x3c00ecff, 0xaeff2301, 0x40013c00, 0x3c00c3ff, 0x52000702, 0x0b023c00, 0x3c005200, 0x9aff2802, 0x05003d00, 0x3d002900, 0x29000a00, 0x26003d00, + 0x3d00ecff, 0xecff2a00, 0x32003d00, 0x3d00ecff, 0xecff3400, 0x89003d00, 0x3d00ecff, 0xecff9400, 0x95003d00, 0x3d00ecff, 0xecff9600, 0x97003d00, + 0x3d00ecff, 0xecff9800, 0x9a003d00, 0x3d00ecff, 0xd7ff1401, 0x07023d00, 0x3d002900, 0x29000b02, 0x2d003e00, 0x44007b00, 0xd7ff0a00, 0x0a004500, + 0x4600d7ff, 0x29000500, 0x0a004600, 0x46002900, 0x3d000702, 0x0b024600, 0x48003d00, 0xd7ff0a00, 0x05004900, 0x49006600, 0x66000a00, 0x59004900, + 0x49001400, 0x14005a00, 0x5c004900, 0x49001400, 0x1400bf00, 0xc1004900, 0x49001400, 0x66000702, 0x0b024900, 0x4a006600, 0x29000500, 0x0a004a00, + 0x4a002900, 0x14004a00, 0x07024a00, 0x4a002900, 0x29000b02, 0x0a004b00, 0x4e00c3ff, 0x29000500, 0x0a004e00, 0x4e002900, 0x29000702, 0x0b024e00, + 0x50002900, 0xd7ff0a00, 0x0a005100, 0x5200d7ff, 0xd7ff5b00, 0x5d005200, 0x5200ecff, 0xecff4001, 0x05005500, 0x55006600, 0x66000a00, 0x44005500, + 0x5500ecff, 0xecff4a00, 0xa2005500, 0x5500ecff, 0xecffa300, 0xa4005500, 0x5500ecff, 0xecffa500, 0xa6005500, 0x5500ecff, 0xecffa700, 0x07025500, + 0x55006600, 0x66000b02, 0x05005600, 0x56003d00, 0x3d000a00, 0x07025600, 0x56003d00, 0x3d000b02, 0x05005700, 0x57005200, 0x52000a00, 0x57005700, + 0x57001400, 0x52000702, 0x0b025700, 0x59005200, 0x29000500, 0x0a005900, 0x59002900, 0x14004900, 0x07025900, 0x59005200, 0x52000b02, 0x05005a00, + 0x5a005200, 0x52000a00, 0x49005a00, 0x5a001400, 0x52000702, 0x0b025a00, 0x5b005200, 0x29000500, 0x0a005b00, 0x5b002900, 0xd7ff5200, 0xa8005b00, + 0x5b00d7ff, 0xd7ffb400, 0xb5005b00, 0x5b00d7ff, 0xd7ffb600, 0xb7005b00, 0x5b00d7ff, 0xd7ffb800, 0xba005b00, 0x5b00d7ff, 0xd7ff1501, 0x07025b00, + 0x5b002900, 0x29000b02, 0x05005c00, 0x5c002900, 0x29000a00, 0x49005c00, 0x5c001400, 0x3d000702, 0x0b025c00, 0x5d003d00, 0xecff5200, 0xa8005d00, + 0x5d00ecff, 0xecffb400, 0xb5005d00, 0x5d00ecff, 0xecffb600, 0xb7005d00, 0x5d00ecff, 0xecffb800, 0xba005d00, 0x5d00ecff, 0xecff1501, 0x2d005e00, + 0x82007b00, 0xaeff0500, 0x0a008200, 0x8200aeff, 0x85ff0d00, 0x0f008200, 0x82004400, 0x44001e00, 0x22008200, 0x8200c3ff, 0xecff2600, 0x2a008200, + 0x8200ecff, 0x5e002d00, 0x32008200, 0x8200ecff, 0xecff3400, 0x37008200, 0x820085ff, 0xecff3800, 0x39008200, 0x8200c3ff, 0xd7ff3a00, 0x3c008200, + 0x82009aff, 0x3b003d00, 0x49008200, 0x8200ecff, 0xecff5700, 0x59008200, 0x8200d7ff, 0xecff5a00, 0x5c008200, 0x8200d7ff, 0xd7ff8200, 0x89008200, + 0x8200ecff, 0xecff9400, 0x95008200, 0x8200ecff, 0xecff9600, 0x97008200, 0x8200ecff, 0xecff9800, 0x9a008200, 0x8200ecff, 0xecff9b00, 0x9c008200, + 0x8200ecff, 0xecff9d00, 0x9e008200, 0x8200ecff, 0x9aff9f00, 0xbf008200, 0x8200d7ff, 0xd7ffc100, 0x14018200, 0x8200d7ff, 0x9aff3a01, 0x3f018200, + 0x82003b00, 0xaeff0702, 0x0b028200, 0x8300aeff, 0xaeff0500, 0x0a008300, 0x8300aeff, 0x85ff0d00, 0x0f008300, 0x83004400, 0x44001e00, 0x22008300, + 0x8300c3ff, 0xecff2600, 0x2a008300, 0x8300ecff, 0x5e002d00, 0x32008300, 0x8300ecff, 0xecff3400, 0x37008300, 0x830085ff, 0xecff3800, 0x39008300, + 0x8300c3ff, 0xd7ff3a00, 0x3c008300, 0x83009aff, 0x3b003d00, 0x49008300, 0x8300ecff, 0xecff5700, 0x59008300, 0x8300d7ff, 0xecff5a00, 0x5c008300, + 0x8300d7ff, 0xe5ff8400, 0x89008300, 0x8300ecff, 0xd7ff8a00, 0x8f008300, 0x8300d7ff, 0xe5ff9000, 0x94008300, 0x8300ecff, 0xecff9500, 0x96008300, + 0x8300ecff, 0xecff9700, 0x98008300, 0x8300ecff, 0xecff9a00, 0x9b008300, 0x8300ecff, 0xecff9c00, 0x9d008300, 0x8300ecff, 0xecff9e00, 0x9f008300, + 0x83009aff, 0xd7ffbf00, 0xc1008300, 0x8300d7ff, 0xd7ff1401, 0x3a018300, 0x83009aff, 0x3b003f01, 0x07028300, 0x8300aeff, 0xaeff0b02, 0x05008400, + 0x8400aeff, 0xaeff0a00, 0x0d008400, 0x840085ff, 0x44000f00, 0x1e008400, 0x84004400, 0xc3ff2200, 0x26008400, 0x8400ecff, 0xecff2a00, 0x2d008400, + 0x84005e00, 0xecff3200, 0x34008400, 0x8400ecff, 0x85ff3700, 0x38008400, 0x8400ecff, 0xc3ff3900, 0x3a008400, 0x8400d7ff, 0x9aff3c00, 0x3d008400, + 0x84003b00, 0xecff4900, 0x57008400, 0x8400ecff, 0xd7ff5900, 0x5a008400, 0x8400ecff, 0xd7ff5c00, 0x84008400, 0x8400e5ff, 0xecff8900, 0x8a008400, + 0x8400d7ff, 0xd7ff8f00, 0x90008400, 0x8400e5ff, 0xecff9400, 0x95008400, 0x8400ecff, 0xecff9600, 0x97008400, 0x8400ecff, 0xecff9800, 0x9a008400, + 0x8400ecff, 0xecff9b00, 0x9c008400, 0x8400ecff, 0xecff9d00, 0x9e008400, 0x8400ecff, 0x9aff9f00, 0xbf008400, 0x8400d7ff, 0xd7ffc100, 0x14018400, + 0x8400d7ff, 0x9aff3a01, 0x3f018400, 0x84003b00, 0xaeff0702, 0x0b028400, 0x8500aeff, 0xaeff0500, 0x0a008500, 0x8500aeff, 0x85ff0d00, 0x0f008500, + 0x85004400, 0x44001e00, 0x22008500, 0x8500c3ff, 0xecff2600, 0x2a008500, 0x8500ecff, 0x5e002d00, 0x32008500, 0x8500ecff, 0xecff3400, 0x37008500, + 0x850085ff, 0xecff3800, 0x39008500, 0x8500c3ff, 0xd7ff3a00, 0x3c008500, 0x85009aff, 0x3b003d00, 0x49008500, 0x8500ecff, 0xecff5700, 0x59008500, + 0x8500d7ff, 0xecff5a00, 0x5c008500, 0x8500d7ff, 0xd7ff8200, 0x89008500, 0x8500ecff, 0xecff9400, 0x95008500, 0x8500ecff, 0xecff9600, 0x97008500, + 0x8500ecff, 0xecff9800, 0x9a008500, 0x8500ecff, 0xecff9b00, 0x9c008500, 0x8500ecff, 0xecff9d00, 0x9e008500, 0x8500ecff, 0x9aff9f00, 0xbf008500, + 0x8500d7ff, 0xd7ffc100, 0x14018500, 0x8500d7ff, 0x9aff3a01, 0x3f018500, 0x85003b00, 0xaeff0702, 0x0b028500, 0x8600aeff, 0x85ff0500, 0x0a008600, + 0x860085ff, 0x7fff0d00, 0x0f008600, 0x86004400, 0x44001e00, 0x22008600, 0x8600d7ff, 0xecff2600, 0x2a008600, 0x8600ecff, 0x5e002d00, 0x32008600, + 0x8600ecff, 0xecff3400, 0x37008600, 0x860085ff, 0xecff3800, 0x39008600, 0x8600c3ff, 0xd7ff3a00, 0x3c008600, 0x86009aff, 0x3b003d00, 0x57008600, + 0x8600e5ff, 0xd5ff5900, 0x5a008600, 0x8600e5ff, 0xdbff5c00, 0x89008600, 0x8600e5ff, 0xe5ff9400, 0x95008600, 0x8600e5ff, 0xe5ff9600, 0x97008600, + 0x8600e5ff, 0xe5ff9800, 0x9a008600, 0x8600e5ff, 0xe5ff9b00, 0x9c008600, 0x8600e5ff, 0xe5ff9d00, 0x9e008600, 0x8600e5ff, 0x9aff9f00, 0xbf008600, + 0x8600dbff, 0xdbffc100, 0x14018600, 0x8600d7ff, 0x9aff3a01, 0x3f018600, 0x86003b00, 0xaeff0702, 0x0b028600, 0x870085ff, 0x66ff0500, 0x0a008700, + 0x870066ff, 0x7fff0d00, 0x0f008700, 0x87004400, 0x44001e00, 0x22008700, 0x8700d7ff, 0xecff2600, 0x2a008700, 0x8700ecff, 0x5e002d00, 0x32008700, + 0x8700ecff, 0xecff3400, 0x37008700, 0x870085ff, 0xecff3800, 0x39008700, 0x8700c3ff, 0xd7ff3a00, 0x3c008700, 0x87009aff, 0x3b003d00, 0x57008700, + 0x8700e5ff, 0xd5ff5900, 0x5a008700, 0x8700e5ff, 0xdbff5c00, 0x89008700, 0x8700e5ff, 0xe5ff9400, 0x95008700, 0x8700e5ff, 0xe5ff9600, 0x97008700, + 0x8700e5ff, 0xe5ff9800, 0x9a008700, 0x8700e5ff, 0xe5ff9b00, 0x9c008700, 0x8700e5ff, 0xe5ff9d00, 0x9e008700, 0x8700e5ff, 0x9aff9f00, 0xbf008700, + 0x8700dbff, 0xdbffc100, 0x14018700, 0x8700d7ff, 0x9aff3a01, 0x3f018700, 0x87003b00, 0x66ff0602, 0x07028700, 0x8700aeff, 0x66ff0a02, 0x0b028700, + 0x870085ff, 0x9aff2002, 0x05008900, 0x89002900, 0x29000a00, 0x0c008900, 0x89002900, 0xd7ff2600, 0x2a008900, 0x8900d7ff, 0xd7ff3200, 0x34008900, + 0x8900d7ff, 0x29004000, 0x60008900, 0x89002900, 0xd7ff8900, 0x94008900, 0x8900d7ff, 0xd7ff9500, 0x96008900, 0x8900d7ff, 0xd7ff9700, 0x98008900, + 0x8900d7ff, 0xd7ff9a00, 0x07028900, 0x89002900, 0x29000b02, 0x05008a00, 0x8a002900, 0x29000a00, 0x10008a00, 0x8a00d7ff, 0xecff2600, 0x2d008a00, + 0x8a003d00, 0xecff3200, 0x34008a00, 0x8a00ecff, 0xecff8400, 0x89008a00, 0x8a00ecff, 0xecff8a00, 0x8f008a00, 0x8a00ecff, 0xecff9400, 0x95008a00, + 0x8a00ecff, 0xecff9600, 0x97008a00, 0x8a00ecff, 0xecff9800, 0x9a008a00, 0x8a00ecff, 0x29000702, 0x0b028a00, 0x8b002900, 0x29000500, 0x0a008b00, + 0x8b002900, 0xd7ff1000, 0x26008b00, 0x8b00ecff, 0x3d002d00, 0x32008b00, 0x8b00ecff, 0xecff3400, 0x89008b00, 0x8b00ecff, 0xecff8b00, 0x94008b00, + 0x8b00ecff, 0xecff9500, 0x96008b00, 0x8b00ecff, 0xecff9700, 0x98008b00, 0x8b00ecff, 0xecff9a00, 0x07028b00, 0x8b002900, 0x29000b02, 0x05008c00, + 0x8c002900, 0x29000a00, 0x10008c00, 0x8c00d7ff, 0xecff2600, 0x2d008c00, 0x8c003d00, 0xecff3200, 0x34008c00, 0x8c00ecff, 0xecff8400, 0x89008c00, + 0x8c00ecff, 0xecff8a00, 0x8f008c00, 0x8c00ecff, 0xecff9400, 0x95008c00, 0x8c00ecff, 0xecff9600, 0x97008c00, 0x8c00ecff, 0xecff9800, 0x9a008c00, + 0x8c00ecff, 0x29000702, 0x0b028c00, 0x8d002900, 0x29000500, 0x0a008d00, 0x8d002900, 0xd7ff1000, 0x26008d00, 0x8d00ecff, 0x3d002d00, 0x32008d00, + 0x8d00ecff, 0xecff3400, 0x84008d00, 0x8d00ecff, 0xecff8900, 0x8a008d00, 0x8d00ecff, 0xecff8f00, 0x94008d00, 0x8d00ecff, 0xecff9500, 0x96008d00, + 0x8d00ecff, 0xecff9700, 0x98008d00, 0x8d00ecff, 0xecff9a00, 0x07028d00, 0x8d002900, 0x29000b02, 0x05008e00, 0x8e002900, 0x29000a00, 0x26008e00, + 0x8e00ecff, 0xecff2a00, 0x32008e00, 0x8e00ecff, 0xecff3400, 0x89008e00, 0x8e00ecff, 0xecff9400, 0x95008e00, 0x8e00ecff, 0xecff9600, 0x97008e00, + 0x8e00ecff, 0xecff9800, 0x9a008e00, 0x8e00ecff, 0xecff1401, 0x07028e00, 0x8e002900, 0x29000b02, 0x05008f00, 0x8f002900, 0x29000a00, 0x26008f00, + 0x8f00ecff, 0xecff2a00, 0x32008f00, 0x8f00ecff, 0xecff3400, 0x89008f00, 0x8f00ecff, 0xecff9400, 0x95008f00, 0x8f00ecff, 0xecff9600, 0x97008f00, + 0x8f00ecff, 0xecff9800, 0x9a008f00, 0x8f00ecff, 0xecff1401, 0x07028f00, 0x8f002900, 0x29000b02, 0x05009000, 0x90002900, 0x29000a00, 0x26009000, + 0x9000ecff, 0xecff2a00, 0x32009000, 0x9000ecff, 0xecff3400, 0x89009000, 0x9000ecff, 0xecff9400, 0x95009000, 0x9000ecff, 0xecff9600, 0x97009000, + 0x9000ecff, 0xecff9800, 0x9a009000, 0x9000ecff, 0xecff1401, 0x07029000, 0x90002900, 0x29000b02, 0x05009100, 0x91002900, 0x29000a00, 0x26009100, + 0x9100ecff, 0xecff2a00, 0x32009100, 0x9100ecff, 0xecff3400, 0x89009100, 0x9100ecff, 0xecff9400, 0x95009100, 0x9100ecff, 0xecff9600, 0x97009100, + 0x9100ecff, 0xecff9800, 0x9a009100, 0x9100ecff, 0xecff1401, 0x07029100, 0x91002900, 0x29000b02, 0x24009200, 0x9200ecff, 0xecff8200, 0x83009200, + 0x9200ecff, 0xecff8400, 0x85009200, 0x9200ecff, 0xecff8600, 0x87009200, 0x9200ecff, 0xecff2802, 0x0c009400, 0x9400d7ff, 0xaeff0f00, 0x11009400, + 0x9400c3ff, 0xecff2400, 0x2c009400, 0x9400ecff, 0xf6ff2d00, 0x36009400, 0x9400ecff, 0xc3ff3700, 0x39009400, 0x9400d7ff, 0xecff3a00, 0x3b009400, + 0x9400d7ff, 0xd7ff3c00, 0x3d009400, 0x9400ecff, 0xd7ff4000, 0x60009400, 0x9400d7ff, 0xecff8200, 0x83009400, 0x9400ecff, 0xecff8400, 0x85009400, + 0x9400ecff, 0xecff8600, 0x87009400, 0x9400ecff, 0xd7ff8800, 0x8e009400, 0x9400ecff, 0xecff8f00, 0x90009400, 0x9400ecff, 0xecff9100, 0x9f009400, + 0x9400c3ff, 0xecff2201, 0x3a019400, 0x9400c3ff, 0xd7ff3f01, 0x28029400, 0x9500ecff, 0xd7ff0c00, 0x0f009500, 0x9500aeff, 0xc3ff1100, 0x24009500, + 0x9500ecff, 0xecff2c00, 0x2d009500, 0x9500f6ff, 0xecff3600, 0x37009500, 0x9500c3ff, 0xd7ff3900, 0x3a009500, 0x9500ecff, 0xd7ff3b00, 0x3c009500, + 0x9500d7ff, 0xecff3d00, 0x40009500, 0x9500d7ff, 0xd7ff6000, 0x82009500, 0x9500ecff, 0xecff8300, 0x84009500, 0x9500d7ff, 0xecff8500, 0x86009500, + 0x9500ecff, 0xecff8700, 0x88009500, 0x9500d7ff, 0xecff8e00, 0x8f009500, 0x9500ecff, 0xecff9000, 0x91009500, 0x9500ecff, 0xc3ff9f00, 0x22019500, + 0x9500ecff, 0xc3ff3a01, 0x3f019500, 0x9500d7ff, 0xecff2802, 0x0c009600, 0x9600d7ff, 0xaeff0f00, 0x11009600, 0x9600c3ff, 0xecff2400, 0x2c009600, + 0x9600ecff, 0xf6ff2d00, 0x36009600, 0x9600ecff, 0xc3ff3700, 0x39009600, 0x9600d7ff, 0xecff3a00, 0x3b009600, 0x9600d7ff, 0xd7ff3c00, 0x3d009600, + 0x9600ecff, 0xd7ff4000, 0x60009600, 0x9600d7ff, 0xecff8200, 0x83009600, 0x9600ecff, 0xd7ff8400, 0x85009600, 0x9600ecff, 0xecff8600, 0x87009600, + 0x9600ecff, 0xd7ff8800, 0x8e009600, 0x9600ecff, 0xecff8f00, 0x90009600, 0x9600ecff, 0xecff9100, 0x9f009600, 0x9600c3ff, 0xecff2201, 0x3a019600, + 0x9600c3ff, 0xd7ff3f01, 0x28029600, 0x9700ecff, 0xd7ff0c00, 0x0f009700, 0x9700aeff, 0xc3ff1100, 0x24009700, 0x9700ecff, 0xecff2c00, 0x2d009700, + 0x9700f6ff, 0xecff3600, 0x37009700, 0x9700c3ff, 0xd7ff3900, 0x3a009700, 0x9700ecff, 0xd7ff3b00, 0x3c009700, 0x9700d7ff, 0xecff3d00, 0x40009700, + 0x9700d7ff, 0xd7ff6000, 0x82009700, 0x9700d7ff, 0xecff8300, 0x84009700, 0x9700ecff, 0xecff8500, 0x86009700, 0x9700ecff, 0xecff8700, 0x88009700, + 0x9700d7ff, 0xecff8e00, 0x8f009700, 0x9700ecff, 0xecff9000, 0x91009700, 0x9700ecff, 0xc3ff9f00, 0x22019700, 0x9700ecff, 0xc3ff3a01, 0x3f019700, + 0x9700d7ff, 0xecff2802, 0x0c009800, 0x9800d7ff, 0xaeff0f00, 0x11009800, 0x9800c3ff, 0xecff2400, 0x2c009800, 0x9800ecff, 0xf6ff2d00, 0x36009800, + 0x9800ecff, 0xc3ff3700, 0x39009800, 0x9800d7ff, 0xecff3a00, 0x3b009800, 0x9800d7ff, 0xd7ff3c00, 0x3d009800, 0x9800ecff, 0xd7ff4000, 0x60009800, + 0x9800d7ff, 0xecff8200, 0x83009800, 0x9800ecff, 0xecff8400, 0x85009800, 0x9800ecff, 0xd7ff8600, 0x87009800, 0x9800ecff, 0xd7ff8800, 0x8e009800, + 0x9800ecff, 0xecff8f00, 0x90009800, 0x9800ecff, 0xecff9100, 0x9f009800, 0x9800c3ff, 0xecff2201, 0x3a019800, 0x9800c3ff, 0xd7ff3f01, 0x28029800, + 0x9a00ecff, 0xc3ff0f00, 0x11009a00, 0x9a00c3ff, 0xecff2400, 0x2c009a00, 0x9a00ecff, 0xc3ff3700, 0x39009a00, 0x9a00d7ff, 0xecff3a00, 0x3b009a00, + 0x9a00d7ff, 0xd7ff3c00, 0x3d009a00, 0x9a00ecff, 0xecff8200, 0x83009a00, 0x9a00ecff, 0xecff8400, 0x85009a00, 0x9a00ecff, 0xecff8600, 0x87009a00, + 0x9a00ecff, 0xd7ff8800, 0x8e009a00, 0x9a00ecff, 0xecff8f00, 0x90009a00, 0x9a00ecff, 0xecff9100, 0x9f009a00, 0x9a00d7ff, 0xd7ff3a01, 0x3f019a00, + 0x9a00ecff, 0xecff2802, 0x0f009b00, 0x9b00c3ff, 0xd7ff1100, 0x24009b00, 0x9b00ecff, 0xecff3000, 0x3d009b00, 0x9b00ecff, 0xecff4400, 0x82009b00, + 0x9b00ecff, 0xecff8300, 0x84009b00, 0x9b00ecff, 0xecff8500, 0x86009b00, 0x9b00ecff, 0xecff8700, 0x88009b00, 0x9b00d7ff, 0xecffa200, 0xa3009b00, + 0x9b00ecff, 0xecffa400, 0xa5009b00, 0x9b00ecff, 0xecffa600, 0xa7009b00, 0x9b00ecff, 0xecff3f01, 0x28029b00, 0x9c00ecff, 0xc3ff0f00, 0x11009c00, + 0x9c00d7ff, 0xecff2400, 0x30009c00, 0x9c00ecff, 0xecff3d00, 0x44009c00, 0x9c00ecff, 0xecff8200, 0x83009c00, 0x9c00ecff, 0xecff8400, 0x85009c00, + 0x9c00ecff, 0xecff8600, 0x87009c00, 0x9c00ecff, 0xd7ff8800, 0xa2009c00, 0x9c00ecff, 0xecffa300, 0xa4009c00, 0x9c00ecff, 0xecffa500, 0xa6009c00, + 0x9c00ecff, 0xecffa700, 0x3f019c00, 0x9c00ecff, 0xecff2802, 0x0f009d00, 0x9d00c3ff, 0xd7ff1100, 0x24009d00, 0x9d00ecff, 0xecff3000, 0x3d009d00, + 0x9d00ecff, 0xecff4400, 0x82009d00, 0x9d00ecff, 0xecff8300, 0x84009d00, 0x9d00ecff, 0xecff8500, 0x86009d00, 0x9d00ecff, 0xecff8700, 0x88009d00, + 0x9d00d7ff, 0xecffa200, 0xa3009d00, 0x9d00ecff, 0xecffa400, 0xa5009d00, 0x9d00ecff, 0xecffa600, 0xa7009d00, 0x9d00ecff, 0xecff3f01, 0x28029d00, + 0x9e00ecff, 0xc3ff0f00, 0x11009e00, 0x9e00d7ff, 0xecff2400, 0x30009e00, 0x9e00ecff, 0xecff3d00, 0x44009e00, 0x9e00ecff, 0xecff8200, 0x83009e00, + 0x9e00ecff, 0xecff8400, 0x85009e00, 0x9e00ecff, 0xecff8600, 0x87009e00, 0x9e00ecff, 0xd7ff8800, 0xa2009e00, 0x9e00ecff, 0xecffa300, 0xa4009e00, + 0x9e00ecff, 0xecffa500, 0xa6009e00, 0x9e00ecff, 0xecffa700, 0x3f019e00, 0x9e00ecff, 0xecff2802, 0x05009f00, 0x9f005200, 0xc3ff0900, 0x0a009f00, + 0x9f005200, 0x3d000c00, 0x0d009f00, 0x9f002900, 0x5cff0f00, 0x10009f00, 0x9f009aff, 0x9aff1100, 0x22009f00, 0x9f002900, 0x9aff2400, 0x26009f00, + 0x9f00d7ff, 0xd7ff2a00, 0x2d009f00, 0x9f00beff, 0xc3ff3000, 0x32009f00, 0x9f00d7ff, 0xd7ff3400, 0x36009f00, 0x9f00ecff, 0x27003700, 0x39009f00, + 0x9f002900, 0x14003a00, 0x40009f00, 0x9f003d00, 0x9aff4400, 0x46009f00, 0x9f009aff, 0x9aff4700, 0x48009f00, 0x9f009aff, 0xe5ff4900, 0x4a009f00, + 0x9f009aff, 0xc3ff5000, 0x51009f00, 0x9f00c3ff, 0x9aff5200, 0x53009f00, 0x9f00c3ff, 0x9aff5400, 0x55009f00, 0x9f00c3ff, 0xaeff5600, 0x58009f00, + 0x9f00c3ff, 0xd7ff5900, 0x5a009f00, 0x9f00ecff, 0xd7ff5b00, 0x5c009f00, 0x9f00ecff, 0xc3ff5d00, 0x60009f00, 0x9f003d00, 0x9aff8200, 0x83009f00, + 0x9f009aff, 0x9aff8400, 0x85009f00, 0x9f009aff, 0x9aff8600, 0x87009f00, 0x9f009aff, 0x71ff8800, 0x89009f00, 0x9f00c3ff, 0xc3ff9400, 0x95009f00, + 0x9f00c3ff, 0xc3ff9600, 0x97009f00, 0x9f00c3ff, 0xc3ff9800, 0x9a009f00, 0x9f00c3ff, 0x71ffa200, 0xa3009f00, 0x9f0071ff, 0x71ffa400, 0xa5009f00, + 0x9f0071ff, 0x71ffa600, 0xa7009f00, 0x9f0071ff, 0x9affa800, 0xa9009f00, 0x9f009aff, 0x9affaa00, 0xab009f00, 0x9f009aff, 0x9affac00, 0xad009f00, + 0x9f009aff, 0xc3ffb300, 0xb4009f00, 0x9f009aff, 0x9affb500, 0xb6009f00, 0x9f009aff, 0x9affb700, 0xb8009f00, 0x9f009aff, 0x9affba00, 0xbb009f00, + 0x9f00c3ff, 0xc3ffbc00, 0xbd009f00, 0x9f00c3ff, 0xc3ffbe00, 0xbf009f00, 0x9f00d7ff, 0xd7ffc100, 0x14019f00, 0x9f00d7ff, 0x9aff1501, 0x22019f00, + 0x9f007bff, 0xaeff2301, 0x40019f00, 0x9f00c3ff, 0x52000702, 0x0b029f00, 0x9f005200, 0x9aff2802, 0x0a00a200, 0xa300d7ff, 0xd7ff0a00, 0x0a00a400, + 0xa500d7ff, 0xd7ff0a00, 0x0a00a600, 0xa700d7ff, 0xd7ff0a00, 0x5b00a800, 0xa800d7ff, 0xecff5d00, 0x4001a800, 0xa900ecff, 0x29000500, 0x0a00a900, + 0xa9002900, 0x3d000702, 0x0b02a900, 0xaa003d00, 0x98ff0500, 0x0a00aa00, 0xab00d7ff, 0x98ff0500, 0x0a00ab00, 0xac00d7ff, 0x98ff0500, 0x0a00ac00, + 0xad00d7ff, 0x98ff0500, 0x0a00ad00, 0xb300d7ff, 0x98ff0500, 0x0a00b300, 0xb300d7ff, 0xd7ff0b02, 0x0500b400, 0xb4006fff, 0x6fff0a00, 0x4900b400, + 0xb400dbff, 0xd7ff5b00, 0x5d00b400, 0xb400ecff, 0xecff4001, 0x0500b500, 0xb5006fff, 0x6fff0a00, 0x4900b500, 0xb500dbff, 0xd7ff5b00, 0x5d00b500, + 0xb500ecff, 0xecff4001, 0x0500b600, 0xb6006fff, 0x6fff0a00, 0x4900b600, 0xb600dbff, 0xd7ff5b00, 0x5d00b600, 0xb600ecff, 0xecff4001, 0x0500b700, + 0xb7006fff, 0x6fff0a00, 0x4900b700, 0xb700dbff, 0xd7ff5b00, 0x5d00b700, 0xb700ecff, 0xecff4001, 0x0500b800, 0xb8006fff, 0x6fff0a00, 0x4900b800, + 0xb800dbff, 0xd7ff5b00, 0x5d00b800, 0xb800ecff, 0xecff4001, 0x5b00ba00, 0xba00d7ff, 0xecff5d00, 0x4001ba00, 0xbb00ecff, 0xbeff0500, 0x0a00bb00, + 0xbc00beff, 0xbeff0500, 0x0a00bc00, 0xbd00beff, 0xbeff0500, 0x0a00bd00, 0xbe00beff, 0xbeff0500, 0x0a00be00, 0xbf00beff, 0x29000500, 0x0a00bf00, + 0xbf002900, 0xbeff0f00, 0x1100bf00, 0xbf00beff, 0xb4ff2200, 0x4600bf00, 0xbf00f6ff, 0xf6ff4700, 0x4800bf00, 0xbf00f6ff, 0x14004900, 0x4a00bf00, + 0xbf00f6ff, 0xf6ff5200, 0x5400bf00, 0xbf00f6ff, 0x06005700, 0xa800bf00, 0xbf00f6ff, 0xf6ffa900, 0xaa00bf00, 0xbf00f6ff, 0xf6ffab00, 0xac00bf00, + 0xbf00f6ff, 0xf6ffad00, 0xb400bf00, 0xbf00f6ff, 0xf6ffb500, 0xb600bf00, 0xbf00f6ff, 0xf6ffb700, 0xb800bf00, 0xbf00f6ff, 0xf6ffba00, 0x1501bf00, + 0xbf00f6ff, 0x3d000702, 0x0802bf00, 0xbf008dff, 0x3d000b02, 0x0c02bf00, 0xbf008dff, 0x81ff1002, 0x1502bf00, 0xc1000c00, 0x29000500, 0x0a00c100, + 0xc1002900, 0xbeff0f00, 0x1100c100, 0xc100beff, 0x14004900, 0x0702c100, 0xc1003d00, 0x3d000b02, 0x05000301, 0x03019aff, 0x9aff0a00, 0x26000301, + 0x0301ecff, 0xecff2a00, 0x32000301, 0x0301ecff, 0xecff3400, 0x37000301, 0x030185ff, 0xecff3800, 0x39000301, 0x0301aeff, 0xc3ff3a00, 0x3c000301, + 0x03019aff, 0xecff8900, 0x94000301, 0x0301ecff, 0xecff9500, 0x96000301, 0x0301ecff, 0xecff9700, 0x98000301, 0x0301ecff, 0xecff9a00, 0x9b000301, + 0x0301ecff, 0xecff9c00, 0x9d000301, 0x0301ecff, 0xecff9e00, 0x9f000301, 0x03019aff, 0xecff1401, 0x3a010301, 0x03019aff, 0x9aff0702, 0x0b020301, + 0x15019aff, 0xd7ff5b00, 0x5d001501, 0x1501ecff, 0xecff4001, 0x05002201, 0x22012900, 0x29000a00, 0x30002201, 0x22011400, 0xecff3600, 0x37002201, + 0x2201d7ff, 0xecff3900, 0x3a002201, 0x2201ecff, 0xd7ff3b00, 0x3c002201, 0x2201d7ff, 0xecff5900, 0x5c002201, 0x2201ecff, 0xd7ff9f00, 0xbf002201, + 0x2201ecff, 0xecffc100, 0x22012201, 0x2201ecff, 0xecff2301, 0x3a012201, 0x2201d7ff, 0x29000702, 0x0b022201, 0x23012900, 0x3d000500, 0x0a002301, + 0x23013d00, 0x3d000702, 0x0b022301, 0x3a013d00, 0x52000500, 0x09003a01, 0x3a01c3ff, 0x52000a00, 0x0c003a01, 0x3a013d00, 0x29000d00, 0x0f003a01, + 0x3a013dff, 0x5cff1000, 0x11003a01, 0x3a019aff, 0x29002200, 0x24003a01, 0x3a019aff, 0xd7ff2600, 0x2a003a01, 0x3a01d7ff, 0xbeff2d00, 0x30003a01, + 0x3a01ecff, 0xd7ff3200, 0x34003a01, 0x3a01d7ff, 0xecff3600, 0x37003a01, 0x3a012700, 0x3d004000, 0x44003a01, 0x3a019aff, 0x9aff4600, 0x47003a01, + 0x3a019aff, 0x9aff4800, 0x49003a01, 0x3a01e5ff, 0x9aff4a00, 0x50003a01, 0x3a01c3ff, 0xc3ff5100, 0x52003a01, 0x3a019aff, 0xc3ff5300, 0x54003a01, + 0x3a019aff, 0xc3ff5500, 0x56003a01, 0x3a01aeff, 0xc3ff5800, 0x5b003a01, 0x3a01d7ff, 0xecff5c00, 0x5d003a01, 0x3a01c3ff, 0x3d006000, 0x82003a01, + 0x3a019aff, 0x9aff8300, 0x84003a01, 0x3a019aff, 0x9aff8500, 0x86003a01, 0x3a019aff, 0x9aff8700, 0x88003a01, 0x3a0171ff, 0xd3ff8900, 0x94003a01, + 0x3a01d3ff, 0xd3ff9500, 0x96003a01, 0x3a01d3ff, 0xd3ff9700, 0x98003a01, 0x3a01d3ff, 0xd3ff9a00, 0xa2003a01, 0x3a0139ff, 0x39ffa300, 0xa4003a01, + 0x3a0139ff, 0x39ffa500, 0xa6003a01, 0x3a0139ff, 0x39ffa700, 0xa8003a01, 0x3a014cff, 0x4cffa900, 0xaa003a01, 0x3a014cff, 0x4cffab00, 0xac003a01, + 0x3a014cff, 0x4cffad00, 0xb3003a01, 0x3a0173ff, 0x4cffb400, 0xb5003a01, 0x3a014cff, 0x4cffb600, 0xb7003a01, 0x3a014cff, 0x4cffb800, 0xba003a01, + 0x3a014cff, 0x73ffbb00, 0xbc003a01, 0x3a0173ff, 0x73ffbd00, 0xbe003a01, 0x3a0173ff, 0xecffbf00, 0xc1003a01, 0x3a01ecff, 0xd7ff1401, 0x15013a01, + 0x3a014cff, 0xe5ff2201, 0x23013a01, 0x3a017bff, 0xc3ff4001, 0x07023a01, 0x3a015200, 0x52000b02, 0x0f023a01, 0x3a01d7ff, 0x9aff2802, 0x05003f01, + 0x3f011400, 0x14000a00, 0x10003f01, 0x3f01c3ff, 0xecff2600, 0x2a003f01, 0x3f01ecff, 0xecff3200, 0x34003f01, 0x3f01ecff, 0xd7ff3600, 0x38003f01, + 0x3f01ecff, 0xecff4600, 0x47003f01, 0x3f01ecff, 0xecff4800, 0x52003f01, 0x3f01ecff, 0xecff5400, 0x59003f01, 0x3f01d7ff, 0xd7ff5a00, 0x5c003f01, + 0x3f01d7ff, 0xd7ff8900, 0x94003f01, 0x3f01d7ff, 0xd7ff9500, 0x96003f01, 0x3f01d7ff, 0xd7ff9700, 0x98003f01, 0x3f01d7ff, 0xd7ff9a00, 0x9b003f01, + 0x3f01ecff, 0xecff9c00, 0x9d003f01, 0x3f01ecff, 0xecff9e00, 0xa8003f01, 0x3f01ecff, 0xecffa900, 0xaa003f01, 0x3f01ecff, 0xecffab00, 0xac003f01, + 0x3f01ecff, 0xecffad00, 0xb4003f01, 0x3f01ecff, 0xecffb500, 0xb6003f01, 0x3f01ecff, 0xecffb700, 0xb8003f01, 0x3f01ecff, 0xecffba00, 0xbf003f01, + 0x3f01d7ff, 0xd7ffc100, 0x14013f01, 0x3f01d7ff, 0xecff1501, 0x22013f01, 0x3f01d7ff, 0x29000702, 0x0b023f01, 0x40011400, 0xecff5200, 0xa8004001, + 0x4001ecff, 0xecffb400, 0xb5004001, 0x4001ecff, 0xecffb600, 0xb7004001, 0x4001ecff, 0xecffb800, 0xba004001, 0x4001ecff, 0xecff1501, 0x0b024001, + 0x61011400, 0x9aff0500, 0x0a006101, 0x61019aff, 0xecff2600, 0x2a006101, 0x6101ecff, 0xecff3200, 0x34006101, 0x6101ecff, 0x85ff3700, 0x38006101, + 0x6101ecff, 0xaeff3900, 0x3a006101, 0x6101c3ff, 0x9aff3c00, 0x89006101, 0x6101ecff, 0xecff9400, 0x95006101, 0x6101ecff, 0xecff9600, 0x97006101, + 0x6101ecff, 0xecff9800, 0x9a006101, 0x6101ecff, 0xecff9b00, 0x9c006101, 0x6101ecff, 0xecff9d00, 0x9e006101, 0x6101ecff, 0x9aff9f00, 0x14016101, + 0x6101ecff, 0x9aff3a01, 0x07026101, 0x61019aff, 0x9aff0b02, 0x05006201, 0x62015200, 0x52000a00, 0x0f006201, 0x62019aff, 0x9aff1100, 0x22006201, + 0x62012900, 0xc3ff2400, 0x26006201, 0x6201d7ff, 0xd7ff2a00, 0x32006201, 0x6201d7ff, 0xd7ff3400, 0x44006201, 0x6201c3ff, 0xc3ff4600, 0x47006201, + 0x6201c3ff, 0xc3ff4800, 0x4a006201, 0x6201c3ff, 0xd7ff5000, 0x51006201, 0x6201d7ff, 0xc3ff5200, 0x53006201, 0x6201d7ff, 0xc3ff5400, 0x55006201, + 0x6201d7ff, 0xd7ff5600, 0x58006201, 0x6201d7ff, 0xc3ff8200, 0x83006201, 0x6201c3ff, 0xc3ff8400, 0x85006201, 0x6201c3ff, 0xc3ff8600, 0x87006201, + 0x6201c3ff, 0x85ff8800, 0x89006201, 0x6201d7ff, 0xd7ff9400, 0x95006201, 0x6201d7ff, 0xd7ff9600, 0x97006201, 0x6201d7ff, 0xd7ff9800, 0x9a006201, + 0x6201d7ff, 0xc3ffa200, 0xa3006201, 0x6201c3ff, 0xc3ffa400, 0xa5006201, 0x6201c3ff, 0xc3ffa600, 0xa7006201, 0x6201c3ff, 0xc3ffa800, 0xa9006201, + 0x6201c3ff, 0xc3ffaa00, 0xab006201, 0x6201c3ff, 0xc3ffac00, 0xad006201, 0x6201c3ff, 0xd7ffb300, 0xb4006201, 0x6201c3ff, 0xc3ffb500, 0xb6006201, + 0x6201c3ff, 0xc3ffb700, 0xb8006201, 0x6201c3ff, 0xc3ffba00, 0xbb006201, 0x6201d7ff, 0xd7ffbc00, 0xbd006201, 0x6201d7ff, 0xd7ffbe00, 0x14016201, + 0x6201d7ff, 0xc3ff1501, 0x23016201, 0x6201d7ff, 0x52000702, 0x0b026201, 0x62015200, 0xc3ff2802, 0x05006901, 0x69015200, 0x52000a00, 0x0f006901, + 0x69019aff, 0x9aff1100, 0x22006901, 0x69012900, 0xc3ff2400, 0x26006901, 0x6901d7ff, 0xd7ff2a00, 0x32006901, 0x6901d7ff, 0xd7ff3400, 0x44006901, + 0x6901c3ff, 0xc3ff4600, 0x47006901, 0x6901c3ff, 0xc3ff4800, 0x4a006901, 0x6901c3ff, 0xd7ff5000, 0x51006901, 0x6901d7ff, 0xc3ff5200, 0x53006901, + 0x6901d7ff, 0xc3ff5400, 0x55006901, 0x6901d7ff, 0xd7ff5600, 0x58006901, 0x6901d7ff, 0xc3ff8200, 0x83006901, 0x6901c3ff, 0xc3ff8400, 0x85006901, + 0x6901c3ff, 0xc3ff8600, 0x87006901, 0x6901c3ff, 0x85ff8800, 0x89006901, 0x6901d7ff, 0xd7ff9400, 0x95006901, 0x6901d7ff, 0xd7ff9600, 0x97006901, + 0x6901d7ff, 0xd7ff9800, 0x9a006901, 0x6901d7ff, 0xc3ffa200, 0xa3006901, 0x6901c3ff, 0xc3ffa400, 0xa5006901, 0x6901c3ff, 0xc3ffa600, 0xa7006901, + 0x6901c3ff, 0xc3ffa800, 0xa9006901, 0x6901c3ff, 0xc3ffaa00, 0xab006901, 0x6901c3ff, 0xc3ffac00, 0xad006901, 0x6901c3ff, 0xd7ffb300, 0xb4006901, + 0x6901c3ff, 0xc3ffb500, 0xb6006901, 0x6901c3ff, 0xc3ffb700, 0xb8006901, 0x6901c3ff, 0xc3ffba00, 0xbb006901, 0x6901d7ff, 0xd7ffbc00, 0xbd006901, + 0x6901d7ff, 0xd7ffbe00, 0x14016901, 0x6901d7ff, 0xc3ff1501, 0x23016901, 0x6901d7ff, 0x52000702, 0x0b026901, 0x69015200, 0xc3ff2802, 0x05007701, + 0x77012900, 0x29000a00, 0x26007701, 0x7701ecff, 0xecff2a00, 0x32007701, 0x7701ecff, 0xecff3400, 0x89007701, 0x7701ecff, 0xecff9400, 0x95007701, + 0x7701ecff, 0xecff9600, 0x97007701, 0x7701ecff, 0xecff9800, 0x9a007701, 0x7701ecff, 0xecff1401, 0x07027701, 0x77012900, 0x29000b02, 0x05007801, + 0x78015200, 0xc3ff0900, 0x0a007801, 0x78015200, 0x3d000c00, 0x0d007801, 0x78012900, 0x3dff0f00, 0x10007801, 0x78015cff, 0x9aff1100, 0x22007801, + 0x78012900, 0x9aff2400, 0x26007801, 0x7801d7ff, 0xd7ff2a00, 0x2d007801, 0x7801beff, 0xecff3000, 0x32007801, 0x7801d7ff, 0xd7ff3400, 0x36007801, + 0x7801ecff, 0x27003700, 0x40007801, 0x78013d00, 0x9aff4400, 0x46007801, 0x78019aff, 0x9aff4700, 0x48007801, 0x78019aff, 0xe5ff4900, 0x4a007801, + 0x78019aff, 0xc3ff5000, 0x51007801, 0x7801c3ff, 0x9aff5200, 0x53007801, 0x7801c3ff, 0x9aff5400, 0x55007801, 0x7801c3ff, 0xaeff5600, 0x58007801, + 0x7801c3ff, 0xd7ff5b00, 0x5c007801, 0x7801ecff, 0xc3ff5d00, 0x60007801, 0x78013d00, 0x9aff8200, 0x83007801, 0x78019aff, 0x9aff8400, 0x85007801, + 0x78019aff, 0x9aff8600, 0x87007801, 0x78019aff, 0x71ff8800, 0x89007801, 0x7801d3ff, 0xd3ff9400, 0x95007801, 0x7801d3ff, 0xd3ff9600, 0x97007801, + 0x7801d3ff, 0xd3ff9800, 0x9a007801, 0x7801d3ff, 0x39ffa200, 0xa3007801, 0x780139ff, 0x39ffa400, 0xa5007801, 0x780139ff, 0x39ffa600, 0xa7007801, + 0x780139ff, 0x4cffa800, 0xa9007801, 0x78014cff, 0x4cffaa00, 0xab007801, 0x78014cff, 0x4cffac00, 0xad007801, 0x78014cff, 0x73ffb300, 0xb4007801, + 0x78014cff, 0x4cffb500, 0xb6007801, 0x78014cff, 0x4cffb700, 0xb8007801, 0x78014cff, 0x4cffba00, 0xbb007801, 0x780173ff, 0x73ffbc00, 0xbd007801, + 0x780173ff, 0x73ffbe00, 0xbf007801, 0x7801ecff, 0xecffc100, 0x14017801, 0x7801d7ff, 0x4cff1501, 0x22017801, 0x7801e5ff, 0x7bff2301, 0x40017801, + 0x7801c3ff, 0x52000702, 0x0b027801, 0x78015200, 0xd7ff0f02, 0x28027801, 0x80019aff, 0x29000500, 0x0a008001, 0x80012900, 0x14004900, 0x07028001, + 0x80013d00, 0x3d000b02, 0x05008201, 0x82013d00, 0x3d000a00, 0x07028201, 0x82013d00, 0x3d000b02, 0x0a008401, 0x8601d7ff, 0x52000500, 0x0a008601, + 0x86015200, 0x14008601, 0x07028601, 0x86015200, 0x52000b02, 0x05008701, 0x87012900, 0x29000a00, 0x07028701, 0x87012900, 0x29000b02, 0x05008801, + 0x88012900, 0x29000a00, 0x49008801, 0x88011400, 0x3d000702, 0x0b028801, 0x8a013d00, 0x29000500, 0x0a008a01, 0x8a012900, 0x14004900, 0x07028a01, + 0x8a015200, 0x52000b02, 0x05009101, 0x91015200, 0x52000a00, 0x91019101, 0x91011400, 0x52000702, 0x0b029101, 0x93015200, 0xd7ff5b00, 0x5d009301, + 0x9301ecff, 0xecff4001, 0x05009401, 0x94012900, 0x29000a00, 0x49009401, 0x94011400, 0x3d000702, 0x0b029401, 0x95013d00, 0xd7ff5b00, 0x5d009501, + 0x9501ecff, 0xecff4001, 0x05009c01, 0x9c012900, 0x29000a00, 0x10009c01, 0x9c01d7ff, 0xecff2600, 0x2d009c01, 0x9c013d00, 0xecff3200, 0x34009c01, + 0x9c01ecff, 0xecff8400, 0x89009c01, 0x9c01ecff, 0xecff8a00, 0x8f009c01, 0x9c01ecff, 0xecff9400, 0x95009c01, 0x9c01ecff, 0xecff9600, 0x97009c01, + 0x9c01ecff, 0xecff9800, 0x9a009c01, 0x9c01ecff, 0x29000702, 0x0b029c01, 0xa2012900, 0x29000500, 0x0a00a201, 0xa2012900, 0xecff2600, 0x2a00a201, + 0xa201ecff, 0xecff3200, 0x3400a201, 0xa201ecff, 0xecff8900, 0x9400a201, 0xa201ecff, 0xecff9500, 0x9600a201, 0xa201ecff, 0xecff9700, 0x9800a201, + 0xa201ecff, 0xecff9a00, 0x1401a201, 0xa201ecff, 0x29000702, 0x0b02a201, 0xcc012900, 0x3d000500, 0x0a00cc01, 0xcc013d00, 0x3d000702, 0x0b02cc01, + 0xd0013d00, 0x29000500, 0x0a00d001, 0xd0012900, 0x29000702, 0x0b02d001, 0xd1012900, 0x3d000500, 0x0a00d101, 0xd1013d00, 0x3d000702, 0x0b02d101, + 0xd4013d00, 0x29000500, 0x0a00d401, 0xd4012900, 0x29000702, 0x0b02d401, 0xe5012900, 0x3d000500, 0x0a00e501, 0xe5013d00, 0x3d000702, 0x0b02e501, + 0xe6013d00, 0x3d000500, 0x0a00e601, 0xe6013d00, 0x3d000702, 0x0b02e601, 0xe9013d00, 0x3d000500, 0x0a00e901, 0xe9013d00, 0x3d000702, 0x0b02e901, + 0xea013d00, 0x98ff0500, 0x0a00ea01, 0x0602d7ff, 0xaeff2400, 0x2c000602, 0x06022900, 0x52003700, 0x39000602, 0x06025200, 0x66003a00, 0x3b000602, + 0x06022900, 0x52003c00, 0x3d000602, 0x06022900, 0xc3ff4600, 0x47000602, 0x0602c3ff, 0xc3ff4800, 0x4a000602, 0x0602d7ff, 0xc3ff5200, 0x54000602, + 0x0602c3ff, 0x29005700, 0x59000602, 0x06022900, 0x14005a00, 0x82000602, 0x0602aeff, 0xaeff8300, 0x84000602, 0x0602aeff, 0xaeff8500, 0x86000602, + 0x0602aeff, 0xaeff8700, 0x88000602, 0x06025cff, 0x29008e00, 0x8f000602, 0x06022900, 0x29009000, 0x91000602, 0x06022900, 0x52009f00, 0xa8000602, + 0x0602c3ff, 0xc3ffa900, 0xaa000602, 0x0602c3ff, 0xc3ffab00, 0xac000602, 0x0602c3ff, 0xc3ffad00, 0xb4000602, 0x0602c3ff, 0xc3ffb500, 0xb6000602, + 0x0602c3ff, 0xc3ffb700, 0xb8000602, 0x0602c3ff, 0xc3ffba00, 0x15010602, 0x0602c3ff, 0x52003a01, 0x3f010602, 0x06022900, 0xaeff2802, 0x24000a02, + 0x0a02aeff, 0x29002c00, 0x37000a02, 0x0a025200, 0x52003900, 0x3a000a02, 0x0a026600, 0x29003b00, 0x3c000a02, 0x0a025200, 0x29003d00, 0x46000a02, + 0x0a02c3ff, 0xc3ff4700, 0x48000a02, 0x0a02c3ff, 0xd7ff4a00, 0x52000a02, 0x0a02c3ff, 0xc3ff5400, 0x57000a02, 0x0a022900, 0x29005900, 0x5a000a02, + 0x0a021400, 0xaeff8200, 0x83000a02, 0x0a02aeff, 0xaeff8400, 0x85000a02, 0x0a02aeff, 0xaeff8600, 0x87000a02, 0x0a02aeff, 0x5cff8800, 0x8e000a02, + 0x0a022900, 0x29008f00, 0x90000a02, 0x0a022900, 0x29009100, 0x9f000a02, 0x0a025200, 0xc3ffa800, 0xa9000a02, 0x0a02c3ff, 0xc3ffaa00, 0xab000a02, + 0x0a02c3ff, 0xc3ffac00, 0xad000a02, 0x0a02c3ff, 0xc3ffb400, 0xb5000a02, 0x0a02c3ff, 0xc3ffb600, 0xb7000a02, 0x0a02c3ff, 0xc3ffb800, 0xba000a02, + 0x0a02c3ff, 0xc3ff1501, 0x3a010a02, 0x0a025200, 0x29003f01, 0x28020a02, 0x2802aeff, 0xaeff0500, 0x0a002802, 0x2802aeff, 0xecff2600, 0x2a002802, + 0x2802ecff, 0xecff3200, 0x34002802, 0x2802ecff, 0x85ff3700, 0x38002802, 0x2802ecff, 0xc3ff3900, 0x3a002802, 0x2802d7ff, 0x9aff3c00, 0x89002802, + 0x2802ecff, 0xecff9400, 0x95002802, 0x2802ecff, 0xecff9600, 0x97002802, 0x2802ecff, 0xecff9800, 0x9a002802, 0x2802ecff, 0xecff9b00, 0x9c002802, + 0x2802ecff, 0xecff9d00, 0x9e002802, 0x2802ecff, 0x9aff9f00, 0x14012802, 0x2802d7ff, 0x9aff3a01, 0x07022802, 0x2802aeff, 0xaeff0b02, 0x19000000, + 0x01003201, 0x00000000, 0x34000000, 0x01000000, 0x00000000, 0x0a000100, 0x01003400, 0x00000000, 0x07000200, 0x01003e00, 0x00000000, 0x15000300, + 0x01004500, 0x00000000, 0x0a000400, 0x01003400, 0x00000000, 0x0c000500, 0x01005a00, 0x00000000, 0x09000600, 0x01006600, 0x00000000, 0x4e000700, + 0x01006f00, 0x00000000, 0x14000800, 0x0100bd00, 0x00000000, 0x67000a00, 0x0100d100, 0x00000000, 0x0a001200, 0x03003400, 0x09040100, 0x68000000, + 0x03003801, 0x09040100, 0x14000100, 0x0300a001, 0x09040100, 0x0e000200, 0x0300b401, 0x09040100, 0x2a000300, 0x0300c201, 0x09040100, 0x14000400, + 0x0300a001, 0x09040100, 0x2c000500, 0x0300ec01, 0x09040100, 0x12000600, 0x03001802, 0x09040100, 0x9c000700, 0x03002a02, 0x09040100, 0x28000800, + 0x0300c602, 0x09040100, 0xce000a00, 0x0300ee02, 0x09040100, 0x38000b00, 0x0300bc03, 0x09040100, 0x5c000c00, 0x0300f403, 0x09040100, 0xa2050d00, + 0x03005004, 0x09040100, 0x46000e00, 0x6944f209, 0x69746967, 0x2064657a, 0x61746164, 0x706f6320, 0x67697279, 0xa9207468, 0x30303220, 0x47202c36, + 0x6c676f6f, 0x6f432065, 0x726f7072, 0x6f697461, 0x72442e6e, 0x2064696f, 0x736e6153, 0x75676552, 0x4172616c, 0x6e656373, 0x20726564, 0x7244202d, + 0x2064696f, 0x736e6153, 0x73726556, 0x206e6f69, 0x30302e31, 0x696f7244, 0x6e615364, 0x6f724473, 0x69206469, 0x20612073, 0x64617274, 0x72616d65, + 0x666f206b, 0x6f6f4720, 0x20656c67, 0x20646e61, 0x2079616d, 0x72206562, 0x73696765, 0x65726574, 0x6e692064, 0x72656320, 0x6e696174, 0x72756a20, + 0x69647369, 0x6f697463, 0x412e736e, 0x6e656373, 0x20726564, 0x70726f43, 0x7461726f, 0x446e6f69, 0x64696f72, 0x6e615320, 0x73692073, 0x68206120, + 0x6e616d75, 0x20747369, 0x736e6173, 0x72657320, 0x74206669, 0x66657079, 0x20656361, 0x69736564, 0x64656e67, 0x726f6620, 0x65737520, 0x6e692072, + 0x66726574, 0x73656361, 0x646e6120, 0x656c6520, 0x6f727463, 0x2063696e, 0x6d6d6f63, 0x63696e75, 0x6f697461, 0x44002e6e, 0x67006900, 0x74006900, + 0x7a006900, 0x64006500, 0x64002000, 0x74006100, 0x20006100, 0x6f006300, 0x79007000, 0x69007200, 0x68006700, 0x20007400, 0x2000a900, 0x30003200, + 0x36003000, 0x20002c00, 0x6f004700, 0x67006f00, 0x65006c00, 0x43002000, 0x72006f00, 0x6f007000, 0x61007200, 0x69007400, 0x6e006f00, 0x44002e00, + 0x6f007200, 0x64006900, 0x53002000, 0x6e006100, 0x52007300, 0x67006500, 0x6c007500, 0x72006100, 0x73004100, 0x65006300, 0x64006e00, 0x72006500, + 0x2d002000, 0x44002000, 0x6f007200, 0x64006900, 0x53002000, 0x6e006100, 0x56007300, 0x72006500, 0x69007300, 0x6e006f00, 0x31002000, 0x30002e00, + 0x20003000, 0x75006200, 0x6c006900, 0x20006400, 0x30003100, 0x44003500, 0x6f007200, 0x64006900, 0x61005300, 0x73006e00, 0x72004400, 0x69006f00, + 0x20006400, 0x73006900, 0x61002000, 0x74002000, 0x61007200, 0x65006400, 0x61006d00, 0x6b007200, 0x6f002000, 0x20006600, 0x6f004700, 0x67006f00, + 0x65006c00, 0x61002000, 0x64006e00, 0x6d002000, 0x79006100, 0x62002000, 0x20006500, 0x65007200, 0x69006700, 0x74007300, 0x72006500, 0x64006500, + 0x69002000, 0x20006e00, 0x65006300, 0x74007200, 0x69006100, 0x20006e00, 0x75006a00, 0x69007200, 0x64007300, 0x63006900, 0x69007400, 0x6e006f00, + 0x2e007300, 0x73004100, 0x65006300, 0x64006e00, 0x72006500, 0x43002000, 0x72006f00, 0x6f007000, 0x61007200, 0x69007400, 0x6e006f00, 0x72004400, + 0x69006f00, 0x20006400, 0x61005300, 0x73006e00, 0x69002000, 0x20007300, 0x20006100, 0x75006800, 0x61006d00, 0x69006e00, 0x74007300, 0x73002000, + 0x6e006100, 0x20007300, 0x65007300, 0x69007200, 0x20006600, 0x79007400, 0x65007000, 0x61006600, 0x65006300, 0x64002000, 0x73006500, 0x67006900, + 0x65006e00, 0x20006400, 0x6f006600, 0x20007200, 0x73007500, 0x72006500, 0x69002000, 0x74006e00, 0x72006500, 0x61006600, 0x65006300, 0x20007300, + 0x6e006100, 0x20006400, 0x6c006500, 0x63006500, 0x72007400, 0x6e006f00, 0x63006900, 0x63002000, 0x6d006f00, 0x75006d00, 0x69006e00, 0x61006300, + 0x69007400, 0x6e006f00, 0x68002e00, 0x74007400, 0x3a007000, 0x2f002f00, 0x77007700, 0x2e007700, 0x73006100, 0x65006300, 0x64006e00, 0x72006500, + 0x6f006300, 0x70007200, 0x63002e00, 0x6d006f00, 0x68002f00, 0x74007400, 0x3a007000, 0x2f002f00, 0x77007700, 0x2e007700, 0x73006100, 0x65006300, + 0x64006e00, 0x72006500, 0x6f006300, 0x70007200, 0x63002e00, 0x6d006f00, 0x74002f00, 0x70007900, 0x64006500, 0x73006500, 0x67006900, 0x65006e00, + 0x73007200, 0x68002e00, 0x6d007400, 0x54006c00, 0x69006800, 0x20007300, 0x6f006600, 0x74006e00, 0x73002000, 0x66006f00, 0x77007400, 0x72006100, + 0x20006500, 0x73006900, 0x74002000, 0x65006800, 0x76002000, 0x6c006100, 0x61007500, 0x6c006200, 0x20006500, 0x72007000, 0x70006f00, 0x72006500, + 0x79007400, 0x6f002000, 0x20006600, 0x73004100, 0x65006300, 0x64006e00, 0x72006500, 0x43002000, 0x72006f00, 0x6f007000, 0x61007200, 0x69007400, + 0x6e006f00, 0x61002000, 0x64006e00, 0x6f002f00, 0x20007200, 0x74006900, 0x20007300, 0x75007300, 0x70007000, 0x69006c00, 0x72006500, 0x20007300, + 0x6e006100, 0x20006400, 0x74006900, 0x20007300, 0x73007500, 0x20006500, 0x79006200, 0x79002000, 0x75006f00, 0x69002000, 0x20007300, 0x6f006300, + 0x65007600, 0x65007200, 0x20006400, 0x6e007500, 0x65006400, 0x20007200, 0x68007400, 0x20006500, 0x65007400, 0x6d007200, 0x20007300, 0x66006f00, + 0x61002000, 0x6c002000, 0x63006900, 0x6e006500, 0x65007300, 0x61002000, 0x72006700, 0x65006500, 0x65006d00, 0x74006e00, 0x20002e00, 0x68005400, + 0x73006900, 0x66002000, 0x6e006f00, 0x20007400, 0x6f007300, 0x74006600, 0x61007700, 0x65007200, 0x69002000, 0x20007300, 0x69006c00, 0x65006300, + 0x73006e00, 0x64006500, 0x74002000, 0x20006f00, 0x6f007900, 0x20007500, 0x79006200, 0x41002000, 0x63007300, 0x6e006500, 0x65006400, 0x20007200, + 0x6f004300, 0x70007200, 0x72006f00, 0x74006100, 0x6f006900, 0x20006e00, 0x6f006600, 0x20007200, 0x6f007900, 0x72007500, 0x70002000, 0x72006500, + 0x6f007300, 0x61006e00, 0x20006c00, 0x72006f00, 0x62002000, 0x73007500, 0x6e006900, 0x73006500, 0x20007300, 0x73007500, 0x20006500, 0x6e006f00, + 0x75002000, 0x20007000, 0x6f007400, 0x66002000, 0x76006900, 0x20006500, 0x65007000, 0x73007200, 0x6e006f00, 0x6c006100, 0x63002000, 0x6d006f00, + 0x75007000, 0x65007400, 0x73007200, 0x20002e00, 0x6f005900, 0x20007500, 0x61006d00, 0x20007900, 0x6f006e00, 0x20007400, 0x73007500, 0x20006500, + 0x68007400, 0x73006900, 0x66002000, 0x6e006f00, 0x20007400, 0x6f007300, 0x74006600, 0x61007700, 0x65007200, 0x6f002000, 0x20006e00, 0x6f006d00, + 0x65007200, 0x74002000, 0x61006800, 0x20006e00, 0x69006600, 0x65007600, 0x70002000, 0x72006500, 0x6f007300, 0x61006e00, 0x20006c00, 0x6f006300, + 0x70006d00, 0x74007500, 0x72006500, 0x20007300, 0x6e007500, 0x65006c00, 0x73007300, 0x79002000, 0x75006f00, 0x68002000, 0x76006100, 0x20006500, + 0x62006f00, 0x61007400, 0x6e006900, 0x64006500, 0x61002000, 0x6c002000, 0x63006900, 0x6e006500, 0x65007300, 0x66002000, 0x6f007200, 0x20006d00, + 0x73004100, 0x65006300, 0x64006e00, 0x72006500, 0x74002000, 0x20006f00, 0x6f006400, 0x73002000, 0x2e006f00, 0x45002000, 0x63007800, 0x70006500, + 0x20007400, 0x73006100, 0x73002000, 0x65007000, 0x69006300, 0x69006600, 0x61006300, 0x6c006c00, 0x20007900, 0x65007000, 0x6d007200, 0x74006900, + 0x65007400, 0x20006400, 0x79006200, 0x74002000, 0x65006800, 0x6c002000, 0x63006900, 0x6e006500, 0x65007300, 0x20002c00, 0x6f007900, 0x20007500, + 0x61006d00, 0x20007900, 0x6f006e00, 0x20007400, 0x6f006300, 0x79007000, 0x74002000, 0x69006800, 0x20007300, 0x6f006600, 0x74006e00, 0x73002000, + 0x66006f00, 0x77007400, 0x72006100, 0x2e006500, 0x0a000a00, 0x66004900, 0x79002000, 0x75006f00, 0x68002000, 0x76006100, 0x20006500, 0x6e006100, + 0x20007900, 0x75007100, 0x73006500, 0x69007400, 0x6e006f00, 0x2c007300, 0x70002000, 0x65006c00, 0x73006100, 0x20006500, 0x65007200, 0x69007600, + 0x77006500, 0x74002000, 0x65006800, 0x6c002000, 0x63006900, 0x6e006500, 0x65007300, 0x61002000, 0x72006700, 0x65006500, 0x65006d00, 0x74006e00, + 0x79002000, 0x75006f00, 0x72002000, 0x63006500, 0x69006500, 0x65007600, 0x20006400, 0x69007700, 0x68007400, 0x74002000, 0x69006800, 0x20007300, + 0x6f006600, 0x74006e00, 0x73002000, 0x66006f00, 0x77007400, 0x72006100, 0x2c006500, 0x61002000, 0x64006e00, 0x6f002f00, 0x20007200, 0x6f006300, + 0x74006e00, 0x63006100, 0x20007400, 0x73004100, 0x65006300, 0x64006e00, 0x72006500, 0x43002000, 0x72006f00, 0x6f007000, 0x61007200, 0x69007400, + 0x6e006f00, 0x20002e00, 0x0a000a00, 0x6f004300, 0x74006e00, 0x63006100, 0x20007400, 0x6e004900, 0x6f006600, 0x6d007200, 0x74006100, 0x6f006900, + 0x3a006e00, 0x41000a00, 0x63007300, 0x6e006500, 0x65006400, 0x20007200, 0x6f004300, 0x70007200, 0x72006f00, 0x74006100, 0x6f006900, 0x0a006e00, + 0x65005700, 0x20006200, 0x74006800, 0x70007400, 0x2f003a00, 0x77002f00, 0x77007700, 0x61002e00, 0x63007300, 0x6e006500, 0x65006400, 0x63007200, + 0x72006f00, 0x2e007000, 0x6f006300, 0x2f006d00, 0x74006800, 0x70007400, 0x2f003a00, 0x61002f00, 0x63007300, 0x6e006500, 0x65006400, 0x63007200, + 0x72006f00, 0x2e007000, 0x6f006300, 0x2f006d00, 0x75006500, 0x61006c00, 0x30003100, 0x68002e00, 0x6d007400, 0x00006c00, 0x00000200, 0x00000000, + 0x660066ff, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00005102, 0x02000201, 0x04000300, 0x06000500, 0x08000700, 0x0a000900, + 0x0c000b00, 0x0e000d00, 0x10000f00, 0x12001100, 0x14001300, 0x16001500, 0x18001700, 0x1a001900, 0x1c001b00, 0x1e001d00, 0x20001f00, 0x22002100, + 0x24002300, 0x26002500, 0x28002700, 0x2a002900, 0x2c002b00, 0x2e002d00, 0x30002f00, 0x32003100, 0x34003300, 0x36003500, 0x38003700, 0x3a003900, + 0x3c003b00, 0x3e003d00, 0x40003f00, 0x42004100, 0x44004300, 0x46004500, 0x48004700, 0x4a004900, 0x4c004b00, 0x4e004d00, 0x50004f00, 0x52005100, + 0x54005300, 0x56005500, 0x58005700, 0x5a005900, 0x5c005b00, 0x5e005d00, 0x60005f00, 0xac006100, 0x8400a300, 0xbd008500, 0xe8009600, 0x8e008600, + 0x9d008b00, 0xa400a900, 0x8a000301, 0x83000401, 0xf2009300, 0x8d00f300, 0x88009700, 0xde000501, 0x9e00f100, 0xf500aa00, 0xf600f400, 0xad00a200, + 0xc700c900, 0x6200ae00, 0x90006300, 0xcb006400, 0xc8006500, 0xcf00ca00, 0xcd00cc00, 0xe900ce00, 0xd3006600, 0xd100d000, 0x6700af00, 0x9100f000, + 0xd400d600, 0x6800d500, 0xed00eb00, 0x6a008900, 0x6b006900, 0x6c006d00, 0xa0006e00, 0x71006f00, 0x72007000, 0x75007300, 0x76007400, 0xea007700, + 0x7a007800, 0x7b007900, 0x7c007d00, 0xa100b800, 0x7e007f00, 0x81008000, 0xee00ec00, 0x0601ba00, 0x08010701, 0x0a010901, 0xfd000b01, 0x0c01fe00, + 0x0e010d01, 0xff000f01, 0x10010001, 0x12011101, 0x14011301, 0x16011501, 0x18011701, 0x1a011901, 0x1c011b01, 0x1e011d01, 0xf8001f01, 0x2001f900, + 0x22012101, 0x24012301, 0x26012501, 0x28012701, 0x2a012901, 0x2c012b01, 0x2e012d01, 0x30012f01, 0x3101d700, 0x33013201, 0x35013401, 0x37013601, + 0x39013801, 0x3b013a01, 0x3d013c01, 0x3f013e01, 0xe300e200, 0x41014001, 0x43014201, 0x45014401, 0x47014601, 0x49014801, 0x4b014a01, 0x4d014c01, + 0xb0004e01, 0x4f01b100, 0x51015001, 0x53015201, 0x55015401, 0x57015601, 0xfb005801, 0xe400fc00, 0x5901e500, 0x5b015a01, 0x5d015c01, 0x5f015e01, + 0x61016001, 0x63016201, 0x65016401, 0x67016601, 0x69016801, 0x6b016a01, 0x6d016c01, 0xbb006e01, 0x70016f01, 0x72017101, 0xe700e600, 0xa6007301, + 0x75017401, 0x77017601, 0x79017801, 0x7b017a01, 0xe100d800, 0xdb007c01, 0xdd00dc00, 0xd900e000, 0x7d01df00, 0x7f017e01, 0x81018001, 0x83018201, + 0x85018401, 0x87018601, 0x89018801, 0x8b018a01, 0x8d018c01, 0x8f018e01, 0x91019001, 0x93019201, 0x95019401, 0x97019601, 0x99019801, 0x9b019a01, + 0x9d019c01, 0x9f019e01, 0xa101a001, 0xa301a201, 0xa501a401, 0xa701a601, 0xa901a801, 0xab01aa01, 0xad01ac01, 0xaf01ae01, 0xb101b001, 0xb301b201, + 0xb501b401, 0xb6019b00, 0xb801b701, 0xba01b901, 0xbc01bb01, 0xbe01bd01, 0xc001bf01, 0xc201c101, 0xc401c301, 0xc601c501, 0xc801c701, 0xca01c901, + 0xcc01cb01, 0xce01cd01, 0xd001cf01, 0xd201d101, 0xd401d301, 0xd601d501, 0xd801d701, 0xda01d901, 0xdc01db01, 0xde01dd01, 0xe001df01, 0xe201e101, + 0xe401e301, 0xe601e501, 0xe801e701, 0xea01e901, 0xec01eb01, 0xee01ed01, 0xf001ef01, 0xf201f101, 0xf401f301, 0xf601f501, 0xf801f701, 0xfa01f901, + 0xfc01fb01, 0xfe01fd01, 0x0002ff01, 0x02020102, 0x04020302, 0x06020502, 0x08020702, 0x0a020902, 0x0c020b02, 0x0e020d02, 0x10020f02, 0x12021102, + 0x14021302, 0x16021502, 0x18021702, 0x1a021902, 0x1c021b02, 0x1e021d02, 0x20021f02, 0x22022102, 0x24022302, 0x26022502, 0x28022702, 0xb2002902, + 0x2a02b300, 0xb6002b02, 0xc400b700, 0xb4002c02, 0xc500b500, 0xc2008200, 0xab008700, 0x2d02c600, 0xbe002e02, 0x2f02bf00, 0x3002bc00, 0x3102f700, + 0x33023202, 0x35023402, 0x8c003602, 0x37029f00, 0x39023802, 0x3b023a02, 0x3c029800, 0x99009a00, 0xa500ef00, 0x9c009200, 0x8f00a700, 0x95009400, + 0x3d02b900, 0x3f023e02, 0x41024002, 0x43024202, 0x45024402, 0x47024602, 0x49024802, 0x4b024a02, 0x4d024c02, 0x4f024e02, 0x51025002, 0x53025202, + 0x55025402, 0x57025602, 0x59025802, 0x756e2e05, 0x75076c6c, 0x3030696e, 0x6f094441, 0x73726576, 0x65726f63, 0x7265700e, 0x63646f69, 0x65746e65, + 0x07646572, 0x63616d41, 0x076e6f72, 0x63616d61, 0x066e6f72, 0x65726241, 0x61066576, 0x76657262, 0x6f410765, 0x656e6f67, 0x6f61076b, 0x656e6f67, + 0x63430b6b, 0x75637269, 0x656c666d, 0x63630b78, 0x75637269, 0x656c666d, 0x64430478, 0x6304746f, 0x06746f64, 0x72616344, 0x64066e6f, 0x6f726163, + 0x6344066e, 0x74616f72, 0x72636406, 0x0774616f, 0x63616d45, 0x076e6f72, 0x63616d65, 0x066e6f72, 0x65726245, 0x65066576, 0x76657262, 0x64450a65, + 0x6361746f, 0x746e6563, 0x6f64650a, 0x63636174, 0x07746e65, 0x6f676f45, 0x076b656e, 0x6f676f65, 0x066b656e, 0x72616345, 0x65066e6f, 0x6f726163, + 0x63470b6e, 0x75637269, 0x656c666d, 0x63670b78, 0x75637269, 0x656c666d, 0x64470478, 0x6704746f, 0x0c746f64, 0x6d6f6347, 0x6361616d, 0x746e6563, + 0x6f63670c, 0x61616d6d, 0x6e656363, 0x63480b74, 0x75637269, 0x656c666d, 0x63680b78, 0x75637269, 0x656c666d, 0x62480478, 0x68047261, 0x06726162, + 0x6c697449, 0x69066564, 0x646c6974, 0x6d490765, 0x6f726361, 0x6d69076e, 0x6f726361, 0x6249066e, 0x65766572, 0x72626906, 0x07657665, 0x6f676f49, + 0x076b656e, 0x6f676f69, 0x0a6b656e, 0x746f6449, 0x65636361, 0x4902746e, 0x6a69024a, 0x69634a0b, 0x6d756372, 0x78656c66, 0x69636a0b, 0x6d756372, + 0x78656c66, 0x6f634b0c, 0x61616d6d, 0x6e656363, 0x636b0c74, 0x616d6d6f, 0x65636361, 0x6b0c746e, 0x65657267, 0x6e616c6e, 0x06636964, 0x7563614c, + 0x6c066574, 0x74756361, 0x634c0c65, 0x616d6d6f, 0x65636361, 0x6c0c746e, 0x6d6d6f63, 0x63636161, 0x06746e65, 0x7261634c, 0x6c066e6f, 0x6f726163, + 0x644c046e, 0x6c04746f, 0x06746f64, 0x7563614e, 0x6e066574, 0x74756361, 0x634e0c65, 0x616d6d6f, 0x65636361, 0x6e0c746e, 0x6d6d6f63, 0x63636161, + 0x06746e65, 0x7261634e, 0x6e066e6f, 0x6f726163, 0x616e0b6e, 0x74736f70, 0x68706f72, 0x6e450365, 0x6e650367, 0x6d4f0767, 0x6f726361, 0x6d6f076e, + 0x6f726361, 0x624f066e, 0x65766572, 0x72626f06, 0x0d657665, 0x6e75684f, 0x75726167, 0x75616c6d, 0x686f0d74, 0x61676e75, 0x6c6d7572, 0x06747561, + 0x75636152, 0x72066574, 0x74756361, 0x63520c65, 0x616d6d6f, 0x65636361, 0x720c746e, 0x6d6d6f63, 0x63636161, 0x06746e65, 0x72616352, 0x72066e6f, + 0x6f726163, 0x6153066e, 0x65747563, 0x63617306, 0x0b657475, 0x72696353, 0x666d7563, 0x0b78656c, 0x72696373, 0x666d7563, 0x0c78656c, 0x6d6f6354, + 0x6361616d, 0x746e6563, 0x6f63740c, 0x61616d6d, 0x6e656363, 0x63540674, 0x6e6f7261, 0x61637406, 0x046e6f72, 0x72616254, 0x61627404, 0x74550672, + 0x65646c69, 0x69747506, 0x0765646c, 0x63616d55, 0x076e6f72, 0x63616d75, 0x066e6f72, 0x65726255, 0x75066576, 0x76657262, 0x72550565, 0x05676e69, + 0x6e697275, 0x68550d67, 0x61676e75, 0x6c6d7572, 0x0d747561, 0x6e756875, 0x75726167, 0x75616c6d, 0x6f550774, 0x656e6f67, 0x6f75076b, 0x656e6f67, + 0x63570b6b, 0x75637269, 0x656c666d, 0x63770b78, 0x75637269, 0x656c666d, 0x63590b78, 0x75637269, 0x656c666d, 0x63790b78, 0x75637269, 0x656c666d, + 0x615a0678, 0x65747563, 0x63617a06, 0x0a657475, 0x746f645a, 0x65636361, 0x7a0a746e, 0x61746f64, 0x6e656363, 0x6f6c0574, 0x0a73676e, 0x6e697241, + 0x75636167, 0x610a6574, 0x676e6972, 0x74756361, 0x45410765, 0x74756361, 0x65610765, 0x74756361, 0x734f0b65, 0x6873616c, 0x74756361, 0x736f0b65, + 0x6873616c, 0x74756361, 0x63530c65, 0x616d6d6f, 0x65636361, 0x730c746e, 0x6d6d6f63, 0x63636161, 0x06746e65, 0x7263616d, 0x74056e6f, 0x736f6e6f, + 0x6569640d, 0x69736572, 0x6e6f7473, 0x410a736f, 0x6168706c, 0x6f6e6f74, 0x6e610973, 0x6c65746f, 0x0c616965, 0x69737045, 0x746e6f6c, 0x736f6e6f, + 0x61744508, 0x6f6e6f74, 0x6f490973, 0x6f746174, 0x0c736f6e, 0x63696d4f, 0x746e6f72, 0x736f6e6f, 0x7370550c, 0x6e6f6c69, 0x6f6e6f74, 0x6d4f0a73, + 0x74616765, 0x736f6e6f, 0x746f6911, 0x65696461, 0x69736572, 0x6e6f7473, 0x4105736f, 0x6168706c, 0x74654204, 0x61470561, 0x07616d6d, 0x30696e75, + 0x07343933, 0x69737045, 0x046e6f6c, 0x6174655a, 0x61744503, 0x65685405, 0x49046174, 0x0561746f, 0x7070614b, 0x614c0661, 0x6164626d, 0x02754d02, + 0x5802754e, 0x6d4f0769, 0x6f726369, 0x6950026e, 0x6f685203, 0x67695305, 0x5403616d, 0x55077561, 0x6c697370, 0x50036e6f, 0x43036968, 0x50036968, + 0x75076973, 0x3330696e, 0x490c3941, 0x6461746f, 0x65726569, 0x0f736973, 0x69737055, 0x646e6f6c, 0x65726569, 0x0a736973, 0x68706c61, 0x6e6f7461, + 0x650c736f, 0x6c697370, 0x6f746e6f, 0x08736f6e, 0x74617465, 0x736f6e6f, 0x746f6909, 0x6e6f7461, 0x7514736f, 0x6c697370, 0x69646e6f, 0x73657265, + 0x6f747369, 0x05736f6e, 0x68706c61, 0x65620461, 0x67056174, 0x616d6d61, 0x6c656405, 0x65076174, 0x6c697370, 0x7a046e6f, 0x03617465, 0x05617465, + 0x74656874, 0x6f690461, 0x6b056174, 0x61707061, 0x6d616c06, 0x07616462, 0x30696e75, 0x02434233, 0x7802756e, 0x6d6f0769, 0x6f726369, 0x6872036e, + 0x6973066f, 0x31616d67, 0x67697305, 0x7403616d, 0x75077561, 0x6c697370, 0x70036e6f, 0x63036968, 0x70036968, 0x6f056973, 0x6167656d, 0x746f690c, + 0x65696461, 0x69736572, 0x70750f73, 0x6f6c6973, 0x6569646e, 0x69736572, 0x6d6f0c73, 0x6f726369, 0x6e6f746e, 0x750c736f, 0x6c697370, 0x6f746e6f, + 0x0a736f6e, 0x67656d6f, 0x6e6f7461, 0x6109736f, 0x31696966, 0x33323030, 0x69666109, 0x30303169, 0x61093135, 0x31696966, 0x32353030, 0x69666109, + 0x30303169, 0x61093335, 0x31696966, 0x34353030, 0x69666109, 0x30303169, 0x61093535, 0x31696966, 0x36353030, 0x69666109, 0x30303169, 0x61093735, + 0x31696966, 0x38353030, 0x69666109, 0x30303169, 0x61093935, 0x31696966, 0x30363030, 0x69666109, 0x30303169, 0x61093136, 0x31696966, 0x32363030, + 0x69666109, 0x31303169, 0x61093534, 0x31696966, 0x37313030, 0x69666109, 0x30303169, 0x61093831, 0x31696966, 0x39313030, 0x69666109, 0x30303169, + 0x61093032, 0x31696966, 0x31323030, 0x69666109, 0x30303169, 0x61093232, 0x31696966, 0x34323030, 0x69666109, 0x30303169, 0x61093532, 0x31696966, + 0x36323030, 0x69666109, 0x30303169, 0x61093732, 0x31696966, 0x38323030, 0x69666109, 0x30303169, 0x61093932, 0x31696966, 0x30333030, 0x69666109, + 0x30303169, 0x61093133, 0x31696966, 0x32333030, 0x69666109, 0x30303169, 0x61093333, 0x31696966, 0x34333030, 0x69666109, 0x30303169, 0x61093533, + 0x31696966, 0x36333030, 0x69666109, 0x30303169, 0x61093733, 0x31696966, 0x38333030, 0x69666109, 0x30303169, 0x61093933, 0x31696966, 0x30343030, + 0x69666109, 0x30303169, 0x61093134, 0x31696966, 0x32343030, 0x69666109, 0x30303169, 0x61093334, 0x31696966, 0x34343030, 0x69666109, 0x30303169, + 0x61093534, 0x31696966, 0x36343030, 0x69666109, 0x30303169, 0x61093734, 0x31696966, 0x38343030, 0x69666109, 0x30303169, 0x61093934, 0x31696966, + 0x35363030, 0x69666109, 0x30303169, 0x61093636, 0x31696966, 0x37363030, 0x69666109, 0x30303169, 0x61093836, 0x31696966, 0x39363030, 0x69666109, + 0x30303169, 0x61093037, 0x31696966, 0x32373030, 0x69666109, 0x30303169, 0x61093337, 0x31696966, 0x34373030, 0x69666109, 0x30303169, 0x61093537, + 0x31696966, 0x36373030, 0x69666109, 0x30303169, 0x61093737, 0x31696966, 0x38373030, 0x69666109, 0x30303169, 0x61093937, 0x31696966, 0x30383030, + 0x69666109, 0x30303169, 0x61093138, 0x31696966, 0x32383030, 0x69666109, 0x30303169, 0x61093338, 0x31696966, 0x34383030, 0x69666109, 0x30303169, + 0x61093538, 0x31696966, 0x36383030, 0x69666109, 0x30303169, 0x61093738, 0x31696966, 0x38383030, 0x69666109, 0x30303169, 0x61093938, 0x31696966, + 0x30393030, 0x69666109, 0x30303169, 0x61093139, 0x31696966, 0x32393030, 0x69666109, 0x30303169, 0x61093339, 0x31696966, 0x34393030, 0x69666109, + 0x30303169, 0x61093539, 0x31696966, 0x36393030, 0x69666109, 0x30303169, 0x61093739, 0x31696966, 0x31373030, 0x69666109, 0x30303169, 0x61093939, + 0x31696966, 0x30303130, 0x69666109, 0x31303169, 0x61093130, 0x31696966, 0x32303130, 0x69666109, 0x31303169, 0x61093330, 0x31696966, 0x34303130, + 0x69666109, 0x31303169, 0x61093530, 0x31696966, 0x36303130, 0x69666109, 0x31303169, 0x61093730, 0x31696966, 0x38303130, 0x69666109, 0x31303169, + 0x61093930, 0x31696966, 0x30313130, 0x69666109, 0x31303169, 0x61093339, 0x31696966, 0x30353030, 0x69666109, 0x30303169, 0x57063839, 0x76617267, + 0x67770665, 0x65766172, 0x63615706, 0x06657475, 0x75636177, 0x57096574, 0x72656964, 0x73697365, 0x69647709, 0x73657265, 0x59067369, 0x76617267, + 0x67790665, 0x65766172, 0x69666109, 0x32303069, 0x750d3830, 0x7265646e, 0x726f6373, 0x6c626465, 0x6f75710d, 0x65726574, 0x73726576, 0x6d066465, + 0x74756e69, 0x65730665, 0x646e6f63, 0x63786509, 0x646d616c, 0x6e096c62, 0x65707573, 0x726f6972, 0x69666109, 0x39383069, 0x70063134, 0x74657365, + 0x75450461, 0x61096f72, 0x36696966, 0x38343231, 0x69666109, 0x32313669, 0x61093938, 0x36696966, 0x32353331, 0x74736509, 0x74616d69, 0x6f096465, + 0x6965656e, 0x68746867, 0x7268740c, 0x69656565, 0x68746867, 0x69660b73, 0x69656576, 0x68746867, 0x65730c73, 0x656e6576, 0x74686769, 0x44057368, + 0x61746c65, 0x696e7507, 0x31304246, 0x696e7507, 0x32304246, 0x7279630d, 0x696c6c69, 0x65726263, 0x64086576, 0x656c746f, 0x106a7373, 0x6f726163, + 0x6d6f636e, 0x6361616d, 0x746e6563, 0x6d6f630b, 0x6361616d, 0x746e6563, 0x6d6f6311, 0x6361616d, 0x746e6563, 0x61746f72, 0x7a0c6574, 0x736f7265, + 0x72657075, 0x0c726f69, 0x72756f66, 0x65707573, 0x726f6972, 0x7669660c, 0x70757365, 0x6f697265, 0x69730b72, 0x70757378, 0x6f697265, 0x65730d72, + 0x736e6576, 0x72657075, 0x0d726f69, 0x68676965, 0x70757374, 0x6f697265, 0x696e0c72, 0x7573656e, 0x69726570, 0x7507726f, 0x3032696e, 0x75073030, + 0x3032696e, 0x75073130, 0x3032696e, 0x75073230, 0x3032696e, 0x75073330, 0x3032696e, 0x75073430, 0x3032696e, 0x75073530, 0x3032696e, 0x75073630, + 0x3032696e, 0x75073730, 0x3032696e, 0x75073830, 0x3032696e, 0x75073930, 0x3032696e, 0x75074130, 0x3032696e, 0x75074230, 0x4546696e, 0x75074646, + 0x4646696e, 0x75074346, 0x4646696e, 0x00004446, 0x02000000, 0x02000500, 0x0300ffff, +}; diff --git a/r5dev/naveditor/include/Filelist.h b/r5dev/naveditor/include/Filelist.h new file mode 100644 index 00000000..30ade6ec --- /dev/null +++ b/r5dev/naveditor/include/Filelist.h @@ -0,0 +1,28 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef FILELIST_H +#define FILELIST_H + +#include +#include + +void scanDirectoryAppend(const std::string& path, const std::string& ext, std::vector& fileList); +void scanDirectory(const std::string& path, const std::string& ext, std::vector& fileList); + +#endif // FILELIST_H diff --git a/r5dev/naveditor/include/InputGeom.h b/r5dev/naveditor/include/InputGeom.h new file mode 100644 index 00000000..0c93ed5b --- /dev/null +++ b/r5dev/naveditor/include/InputGeom.h @@ -0,0 +1,151 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef INPUTGEOM_H +#define INPUTGEOM_H + +#include "NavEditor/Include/ChunkyTriMesh.h" +#include "NavEditor/Include/MeshLoaderObj.h" + +static const int MAX_CONVEXVOL_PTS = 12; +struct ConvexVolume +{ + float verts[MAX_CONVEXVOL_PTS*3]; + float hmin, hmax; + int nverts; + int area; +}; + +struct BuildSettings +{ + // Cell size in world units + float cellSize; + // Cell height in world units + float cellHeight; + // Agent height in world units + float agentHeight; + // Agent radius in world units + float agentRadius; + // Agent max climb in world units + float agentMaxClimb; + // Agent max slope in degrees + float agentMaxSlope; + // Region minimum size in voxels. + // regionMinSize = sqrt(regionMinArea) + float regionMinSize; + // Region merge size in voxels. + // regionMergeSize = sqrt(regionMergeArea) + float regionMergeSize; + // Edge max length in world units + float edgeMaxLen; + // Edge max error in voxels + float edgeMaxError; + float vertsPerPoly; + // Detail sample distance in voxels + float detailSampleDist; + // Detail sample max error in voxel heights. + float detailSampleMaxError; + // Partition type, see SamplePartitionType + int partitionType; + // Bounds of the area to mesh + float navMeshBMin[3]; + float navMeshBMax[3]; + // Size of the tiles in voxels + float tileSize; +}; + +class InputGeom +{ + rcChunkyTriMesh* m_chunkyMesh; + IMeshLoader* m_mesh; + float m_meshBMin[3], m_meshBMax[3]; + BuildSettings m_buildSettings; + bool m_hasBuildSettings; + + /// @name Off-Mesh connections. + ///@{ + static const int MAX_OFFMESH_CONNECTIONS = 256; + float m_offMeshConVerts[MAX_OFFMESH_CONNECTIONS*3*2]; + float m_offMeshConRads[MAX_OFFMESH_CONNECTIONS]; + unsigned char m_offMeshConDirs[MAX_OFFMESH_CONNECTIONS]; + unsigned char m_offMeshConAreas[MAX_OFFMESH_CONNECTIONS]; + unsigned short m_offMeshConFlags[MAX_OFFMESH_CONNECTIONS]; + unsigned int m_offMeshConId[MAX_OFFMESH_CONNECTIONS]; + int m_offMeshConCount; + ///@} + + /// @name Convex Volumes. + ///@{ + static const int MAX_VOLUMES = 256; + ConvexVolume m_volumes[MAX_VOLUMES]; + int m_volumeCount; + ///@} + + bool loadMesh(class rcContext* ctx, const std::string& filepath,bool is_tf2); + bool loadPlyMesh(class rcContext* ctx, const std::string& filepath, bool is_tf2); + bool loadGeomSet(class rcContext* ctx, const std::string& filepath,bool is_tf2); +public: + InputGeom(); + ~InputGeom(); + + + bool load(class rcContext* ctx, const std::string& filepath, bool is_tf2); + bool saveGeomSet(const BuildSettings* settings); + + /// Method to return static mesh data. + const IMeshLoader* getMesh() const { return m_mesh; } + const float* getMeshBoundsMin() const { return m_meshBMin; } + const float* getMeshBoundsMax() const { return m_meshBMax; } + const float* getNavMeshBoundsMin() const { return m_hasBuildSettings ? m_buildSettings.navMeshBMin : m_meshBMin; } + const float* getNavMeshBoundsMax() const { return m_hasBuildSettings ? m_buildSettings.navMeshBMax : m_meshBMax; } + const rcChunkyTriMesh* getChunkyMesh() const { return m_chunkyMesh; } + const BuildSettings* getBuildSettings() const { return m_hasBuildSettings ? &m_buildSettings : 0; } + bool raycastMesh(float* src, float* dst, float& tmin); + + /// @name Off-Mesh connections. + ///@{ + int getOffMeshConnectionCount() const { return m_offMeshConCount; } + const float* getOffMeshConnectionVerts() const { return m_offMeshConVerts; } + const float* getOffMeshConnectionRads() const { return m_offMeshConRads; } + const unsigned char* getOffMeshConnectionDirs() const { return m_offMeshConDirs; } + const unsigned char* getOffMeshConnectionAreas() const { return m_offMeshConAreas; } + const unsigned short* getOffMeshConnectionFlags() const { return m_offMeshConFlags; } + const unsigned int* getOffMeshConnectionId() const { return m_offMeshConId; } + void addOffMeshConnection(const float* spos, const float* epos, const float rad, + unsigned char bidir, unsigned char area, unsigned short flags); + void deleteOffMeshConnection(int i); + void drawOffMeshConnections(struct duDebugDraw* dd, bool hilight = false); + ///@} + + /// @name Box Volumes. + ///@{ + int getConvexVolumeCount() const { return m_volumeCount; } + const ConvexVolume* getConvexVolumes() const { return m_volumes; } + void addConvexVolume(const float* verts, const int nverts, + const float minh, const float maxh, unsigned char area); + void deleteConvexVolume(int i); + void drawConvexVolumes(struct duDebugDraw* dd, bool hilight = false); + ///@} + +private: + // Explicitly disabled copy constructor and copy assignment operator. + InputGeom(const InputGeom&); + InputGeom& operator=(const InputGeom&); +}; + +#endif // INPUTGEOM_H diff --git a/r5dev/naveditor/include/MeshLoaderBsp.h b/r5dev/naveditor/include/MeshLoaderBsp.h new file mode 100644 index 00000000..02864025 --- /dev/null +++ b/r5dev/naveditor/include/MeshLoaderBsp.h @@ -0,0 +1,51 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// +#pragma once + +#include +#include +#include + +class rcMeshLoaderBsp:public IMeshLoader +{ +public: + rcMeshLoaderBsp() = default; + // Explicitly disabled copy constructor and copy assignment operator. + rcMeshLoaderBsp(const rcMeshLoaderBsp&) =delete; + rcMeshLoaderBsp& operator=(const rcMeshLoaderBsp&) =delete; + + bool load(const std::string& fileName); + + const float* getVerts() const { return m_verts.data(); } + const float* getNormals() const { return m_normals.data(); } + const int* getTris() const { return m_tris.data(); } + int getVertCount() const { return m_vertCount; } + int getTriCount() const { return m_triCount; } + const std::string& getFileName() const { return m_filename; } + +private: + std::string m_filename; + float m_scale = 1.0; + std::vector m_verts; + std::vector m_tris; + std::vector m_normals; + int m_vertCount = 0; + int m_triCount = 0; + + +}; diff --git a/r5dev/naveditor/include/MeshLoaderObj.h b/r5dev/naveditor/include/MeshLoaderObj.h new file mode 100644 index 00000000..dfeaeba1 --- /dev/null +++ b/r5dev/naveditor/include/MeshLoaderObj.h @@ -0,0 +1,74 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef MESHLOADER_OBJ +#define MESHLOADER_OBJ + +#include +class IMeshLoader +{ +public: + virtual ~IMeshLoader() {}; + + virtual bool load(const std::string& fileName)=0; + + virtual const float* getVerts() const = 0; + virtual const float* getNormals() const = 0; + virtual const int* getTris() const = 0; + virtual int getVertCount() const = 0; + virtual int getTriCount() const = 0; + virtual const std::string& getFileName() const = 0; + + bool m_tf2_import_flip = false; // !TODO: ImGui import option. + bool m_flip_tris = false; // !TODO: ImGui import option. +}; +class rcMeshLoaderObj:public IMeshLoader +{ +public: + rcMeshLoaderObj(); + virtual ~rcMeshLoaderObj(); + + bool load(const std::string& fileName); + + const float* getVerts() const { return m_verts; } + const float* getNormals() const { return m_normals; } + const int* getTris() const { return m_tris; } + int getVertCount() const { return m_vertCount; } + int getTriCount() const { return m_triCount; } + const std::string& getFileName() const { return m_filename; } + +private: + // Explicitly disabled copy constructor and copy assignment operator. + rcMeshLoaderObj(const rcMeshLoaderObj&); + rcMeshLoaderObj& operator=(const rcMeshLoaderObj&); + + void addVertex(float x, float y, float z, int& cap); + void addTriangle(int a, int b, int c, int& cap); + + std::string m_filename; + float m_scale; + float* m_verts; + int* m_tris; + float* m_normals; + int m_vertCount; + int m_triCount; + + +}; + +#endif // MESHLOADER_OBJ diff --git a/r5dev/naveditor/include/MeshLoaderPly.h b/r5dev/naveditor/include/MeshLoaderPly.h new file mode 100644 index 00000000..00509b84 --- /dev/null +++ b/r5dev/naveditor/include/MeshLoaderPly.h @@ -0,0 +1,50 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// +#pragma once + +#include +#include +#include + +class rcMeshLoaderPly:public IMeshLoader +{ +public: + rcMeshLoaderPly() = default; + // Explicitly disabled copy constructor and copy assignment operator. + rcMeshLoaderPly(const rcMeshLoaderPly&) =delete; + rcMeshLoaderPly& operator=(const rcMeshLoaderPly&) =delete; + + bool load(const std::string& fileName); + + const float* getVerts() const { return m_verts.data(); } + const float* getNormals() const { return m_normals.data(); } + const int* getTris() const { return m_tris.data(); } + int getVertCount() const { return m_vertCount; } + int getTriCount() const { return m_triCount; } + const std::string& getFileName() const { return m_filename; } + +private: + + std::string m_filename; + float m_scale = 1.0; + std::vector m_verts; + std::vector m_tris; + std::vector m_normals; + int m_vertCount = 0; + int m_triCount = 0; +}; diff --git a/r5dev/naveditor/include/NavMeshPruneTool.h b/r5dev/naveditor/include/NavMeshPruneTool.h new file mode 100644 index 00000000..a8bc74c7 --- /dev/null +++ b/r5dev/naveditor/include/NavMeshPruneTool.h @@ -0,0 +1,56 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef NAVMESHPRUNETOOL_H +#define NAVMESHPRUNETOOL_H + +#include "NavEditor/Include/Sample.h" + +// Prune navmesh to accessible locations from a point. + +class NavMeshPruneTool : public SampleTool +{ + Sample* m_sample; + + class NavmeshFlags* m_flags; + + float m_hitPos[3]; + bool m_hitPosSet; + +public: + NavMeshPruneTool(); + virtual ~NavMeshPruneTool(); + + virtual int type() { return TOOL_NAVMESH_PRUNE; } + virtual void init(Sample* sample); + virtual void reset(); + virtual void handleMenu(); + virtual void handleClick(const float* s, const float* p, bool shift); + virtual void handleToggle(); + virtual void handleStep(); + virtual void handleUpdate(const float dt); + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); + +private: + // Explicitly disabled copy constructor and copy assignment operator. + NavMeshPruneTool(const NavMeshPruneTool&); + NavMeshPruneTool& operator=(const NavMeshPruneTool&); +}; + +#endif // NAVMESHPRUNETOOL_H diff --git a/r5dev/naveditor/include/NavMeshTesterTool.h b/r5dev/naveditor/include/NavMeshTesterTool.h new file mode 100644 index 00000000..56644846 --- /dev/null +++ b/r5dev/naveditor/include/NavMeshTesterTool.h @@ -0,0 +1,113 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef NAVMESHTESTERTOOL_H +#define NAVMESHTESTERTOOL_H + +#include "NavEditor/Include/Sample.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Detour/Include/DetourNavMeshQuery.h" + +class NavMeshTesterTool : public SampleTool +{ + Sample* m_sample; + + dtNavMesh* m_navMesh; + dtNavMeshQuery* m_navQuery; + + dtQueryFilter m_filter; + + dtStatus m_pathFindStatus; + + enum ToolMode + { + TOOLMODE_PATHFIND_FOLLOW, + TOOLMODE_PATHFIND_STRAIGHT, + TOOLMODE_PATHFIND_SLICED, + TOOLMODE_RAYCAST, + TOOLMODE_DISTANCE_TO_WALL, + TOOLMODE_FIND_POLYS_IN_CIRCLE, + TOOLMODE_FIND_POLYS_IN_SHAPE, + TOOLMODE_FIND_LOCAL_NEIGHBOURHOOD, + }; + + ToolMode m_toolMode; + + int m_straightPathOptions; + + static const int MAX_POLYS = 256; + static const int MAX_SMOOTH = 2048; + + dtPolyRef m_startRef; + dtPolyRef m_endRef; + dtPolyRef m_polys[MAX_POLYS]; + dtPolyRef m_parent[MAX_POLYS]; + int m_npolys; + float m_straightPath[MAX_POLYS*3]; + unsigned char m_straightPathFlags[MAX_POLYS]; + dtPolyRef m_straightPathPolys[MAX_POLYS]; + int m_nstraightPath; + float m_polyPickExt[3]; + float m_smoothPath[MAX_SMOOTH*3]; + int m_nsmoothPath; + float m_queryPoly[4*3]; + + static const int MAX_RAND_POINTS = 64; + float m_randPoints[MAX_RAND_POINTS*3]; + int m_nrandPoints; + bool m_randPointsInCircle; + + float m_spos[3]; + float m_epos[3]; + float m_hitPos[3]; + float m_hitNormal[3]; + bool m_hitResult; + float m_distanceToWall; + float m_neighbourhoodRadius; + float m_randomRadius; + bool m_sposSet; + bool m_eposSet; + + int m_pathIterNum; + dtPolyRef m_pathIterPolys[MAX_POLYS]; + int m_pathIterPolyCount; + float m_prevIterPos[3], m_iterPos[3], m_steerPos[3], m_targetPos[3]; + + static const int MAX_STEER_POINTS = 10; + float m_steerPoints[MAX_STEER_POINTS*3]; + int m_steerPointCount; + +public: + NavMeshTesterTool(); + + virtual int type() { return TOOL_NAVMESH_TESTER; } + virtual void init(Sample* sample); + virtual void reset(); + virtual void handleMenu(); + virtual void handleClick(const float* s, const float* p, bool shift); + virtual void handleToggle(); + virtual void handleStep(); + virtual void handleUpdate(const float dt); + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); + + void recalc(); + void drawAgent(const float* pos, float r, float h, float c, const unsigned int col); +}; + +#endif // NAVMESHTESTERTOOL_H \ No newline at end of file diff --git a/r5dev/naveditor/include/OffMeshConnectionTool.h b/r5dev/naveditor/include/OffMeshConnectionTool.h new file mode 100644 index 00000000..fa166c5d --- /dev/null +++ b/r5dev/naveditor/include/OffMeshConnectionTool.h @@ -0,0 +1,50 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef OFFMESHCONNECTIONTOOL_H +#define OFFMESHCONNECTIONTOOL_H + +#include "NavEditor/Include/Sample.h" + +// Tool to create off-mesh connection for InputGeom + +class OffMeshConnectionTool : public SampleTool +{ + Sample* m_sample; + float m_hitPos[3]; + bool m_hitPosSet; + bool m_bidir; + unsigned char m_oldFlags; + +public: + OffMeshConnectionTool(); + ~OffMeshConnectionTool(); + + virtual int type() { return TOOL_OFFMESH_CONNECTION; } + virtual void init(Sample* sample); + virtual void reset(); + virtual void handleMenu(); + virtual void handleClick(const float* s, const float* p, bool shift); + virtual void handleToggle(); + virtual void handleStep(); + virtual void handleUpdate(const float dt); + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); +}; + +#endif // OFFMESHCONNECTIONTOOL_H diff --git a/r5dev/naveditor/include/PerfTimer.h b/r5dev/naveditor/include/PerfTimer.h new file mode 100644 index 00000000..ed62c180 --- /dev/null +++ b/r5dev/naveditor/include/PerfTimer.h @@ -0,0 +1,32 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef PERFTIMER_H +#define PERFTIMER_H + +#ifdef __GNUC__ +#include +typedef int64_t TimeVal; +#else +typedef __int64 TimeVal; +#endif + +TimeVal getPerfTime(); +int getPerfTimeUsec(const TimeVal duration); + +#endif // PERFTIMER_H \ No newline at end of file diff --git a/r5dev/naveditor/include/Sample.h b/r5dev/naveditor/include/Sample.h new file mode 100644 index 00000000..5a4fab1b --- /dev/null +++ b/r5dev/naveditor/include/Sample.h @@ -0,0 +1,206 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef RECASTSAMPLE_H +#define RECASTSAMPLE_H + +#include "Recast/Include/Recast.h" +#include "NavEditor/Include/SampleInterfaces.h" +#include + +struct hulldef +{ + const char* name; + float radius; + float height; + float climb_height; + + float tile_size; + //TODO: voxel size, tile size +}; +/// Tool types. +enum SampleToolType +{ + TOOL_NONE = 0, + TOOL_TILE_EDIT, + TOOL_TILE_HIGHLIGHT, + TOOL_TEMP_OBSTACLE, + TOOL_NAVMESH_TESTER, + TOOL_NAVMESH_PRUNE, + TOOL_OFFMESH_CONNECTION, + TOOL_CONVEX_VOLUME, + TOOL_CROWD, + MAX_TOOLS +}; + +/// These are just sample areas to use consistent values across the samples. +/// The use should specify these base on his needs. +enum SamplePolyAreas +{ + SAMPLE_POLYAREA_GROUND, + SAMPLE_POLYAREA_WATER, + SAMPLE_POLYAREA_ROAD, + SAMPLE_POLYAREA_DOOR, + SAMPLE_POLYAREA_GRASS, + SAMPLE_POLYAREA_JUMP, +}; +enum SamplePolyFlags +{ + SAMPLE_POLYFLAGS_WALK = 0x01, // Ability to walk (ground, grass, road) + SAMPLE_POLYFLAGS_SWIM = 0x02, // Ability to swim (water). + SAMPLE_POLYFLAGS_DOOR = 0x04, // Ability to move through doors. + SAMPLE_POLYFLAGS_JUMP = 0x08, // Ability to jump. + SAMPLE_POLYFLAGS_DISABLED = 0x10, // Disabled polygon + SAMPLE_POLYFLAGS_ALL = 0xffff // All abilities. +}; + +class SampleDebugDraw : public DebugDrawGL +{ +public: + virtual unsigned int areaToCol(unsigned int area); +}; + +enum SamplePartitionType +{ + SAMPLE_PARTITION_WATERSHED, + SAMPLE_PARTITION_MONOTONE, + SAMPLE_PARTITION_LAYERS, +}; + +struct SampleTool +{ + virtual ~SampleTool() {} + virtual int type() = 0; + virtual void init(class Sample* sample) = 0; + virtual void reset() = 0; + virtual void handleMenu() = 0; + virtual void handleClick(const float* s, const float* p, bool shift) = 0; + virtual void handleRender() = 0; + virtual void handleRenderOverlay(double* proj, double* model, int* view) = 0; + virtual void handleToggle() = 0; + virtual void handleStep() = 0; + virtual void handleUpdate(const float dt) = 0; +}; + +struct SampleToolState { + virtual ~SampleToolState() {} + virtual void init(class Sample* sample) = 0; + virtual void reset() = 0; + virtual void handleRender() = 0; + virtual void handleRenderOverlay(double* proj, double* model, int* view) = 0; + virtual void handleUpdate(const float dt) = 0; +}; + +class Sample +{ +protected: + class InputGeom* m_geom; + class dtNavMesh* m_navMesh; + class dtNavMeshQuery* m_navQuery; + class dtCrowd* m_crowd; + + unsigned char m_navMeshDrawFlags; + + float m_cellSize; + float m_cellHeight; + float m_agentHeight; + float m_agentRadius; + float m_agentMaxClimb; + float m_agentMaxSlope; + float m_regionMinSize; + float m_regionMergeSize; + float m_edgeMaxLen; + float m_edgeMaxError; + float m_vertsPerPoly; + float m_detailSampleDist; + float m_detailSampleMaxError; + int m_partitionType; + int m_count_reachability_tables; + const char* m_navmesh_name="unk"; + + bool m_filterLowHangingObstacles; + bool m_filterLedgeSpans; + bool m_filterWalkableLowHeightSpans; + + SampleTool* m_tool; + SampleToolState* m_toolStates[MAX_TOOLS]; + + BuildContext* m_ctx; + + SampleDebugDraw m_dd; + + dtNavMesh* loadAll(const char* path); + void saveAll(const char* path,dtNavMesh* mesh); + +public: + std::string m_model_name; + + Sample(); + virtual ~Sample(); + + void setContext(BuildContext* ctx) { m_ctx = ctx; } + + void setTool(SampleTool* tool); + SampleToolState* getToolState(int type) { return m_toolStates[type]; } + void setToolState(int type, SampleToolState* s) { m_toolStates[type] = s; } + + SampleDebugDraw& getDebugDraw() { return m_dd; } + + virtual void handleSettings(); + virtual void handleTools(); + virtual void handleDebugMode(); + virtual void handleClick(const float* s, const float* p, bool shift); + virtual void handleToggle(); + virtual void handleStep(); + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); + virtual void handleMeshChanged(class InputGeom* geom); + virtual bool handleBuild(); + virtual void handleUpdate(const float dt); + virtual void collectSettings(struct BuildSettings& settings); + + virtual class InputGeom* getInputGeom() { return m_geom; } + virtual class dtNavMesh* getNavMesh() { return m_navMesh; } + virtual class dtNavMeshQuery* getNavMeshQuery() { return m_navQuery; } + virtual class dtCrowd* getCrowd() { return m_crowd; } + virtual float getAgentRadius() { return m_agentRadius; } + virtual float getAgentHeight() { return m_agentHeight; } + virtual float getAgentClimb() { return m_agentMaxClimb; } + + unsigned char getNavMeshDrawFlags() const { return m_navMeshDrawFlags; } + void setNavMeshDrawFlags(unsigned char flags) { m_navMeshDrawFlags = flags; } + + void updateToolStates(const float dt); + void initToolStates(Sample* sample); + void resetToolStates(); + void renderToolStates(); + void renderOverlayToolStates(double* proj, double* model, int* view); + + void resetCommonSettings(); + void handleCommonSettings(); + + //don't do this kids, this is bad cpp + bool* is_tf2=nullptr; +private: + // Explicitly disabled copy constructor and copy assignment operator. + Sample(const Sample&); + Sample& operator=(const Sample&); +}; + + +#endif // RECASTSAMPLE_H diff --git a/r5dev/naveditor/include/SampleInterfaces.h b/r5dev/naveditor/include/SampleInterfaces.h new file mode 100644 index 00000000..3816566a --- /dev/null +++ b/r5dev/naveditor/include/SampleInterfaces.h @@ -0,0 +1,99 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef SAMPLEINTERFACES_H +#define SAMPLEINTERFACES_H + +#include "DebugUtils/Include/DebugDraw.h" +#include "Recast/Include/Recast.h" +#include "DebugUtils/Include/RecastDump.h" +#include "NavEditor/Include/PerfTimer.h" + +// These are example implementations of various interfaces used in Recast and Detour. + +/// Recast build context. +class BuildContext : public rcContext +{ + TimeVal m_startTime[RC_MAX_TIMERS]; + TimeVal m_accTime[RC_MAX_TIMERS]; + + static const int MAX_MESSAGES = 1000; + const char* m_messages[MAX_MESSAGES]; + int m_messageCount; + static const int TEXT_POOL_SIZE = 8000; + char m_textPool[TEXT_POOL_SIZE]; + int m_textPoolSize; + +public: + BuildContext(); + + /// Dumps the log to stdout. + void dumpLog(const char* format, ...); + /// Returns number of log messages. + int getLogCount() const; + /// Returns log message text. + const char* getLogText(const int i) const; + +protected: + /// Virtual functions for custom implementations. + ///@{ + virtual void doResetLog(); + virtual void doLog(const rcLogCategory category, const char* msg, const int len); + virtual void doResetTimers(); + virtual void doStartTimer(const rcTimerLabel label); + virtual void doStopTimer(const rcTimerLabel label); + virtual int doGetAccumulatedTime(const rcTimerLabel label) const; + ///@} +}; + +/// OpenGL debug draw implementation. +class DebugDrawGL : public duDebugDraw +{ +public: + virtual void depthMask(bool state); + virtual void texture(bool state); + virtual void begin(duDebugDrawPrimitives prim, float size = 1.0f); + virtual void vertex(const float* pos, unsigned int color); + virtual void vertex(const float x, const float y, const float z, unsigned int color); + virtual void vertex(const float* pos, unsigned int color, const float* uv); + virtual void vertex(const float x, const float y, const float z, unsigned int color, const float u, const float v); + virtual void end(); +}; + +/// stdio file implementation. +class FileIO : public duFileIO +{ + FILE* m_fp; + int m_mode; +public: + FileIO(); + virtual ~FileIO(); + bool openForWrite(const char* path); + bool openForRead(const char* path); + virtual bool isWriting() const; + virtual bool isReading() const; + virtual bool write(const void* ptr, const size_t size); + virtual bool read(void* ptr, const size_t size); +private: + // Explicitly disabled copy constructor and copy assignment operator. + FileIO(const FileIO&); + FileIO& operator=(const FileIO&); +}; + +#endif // SAMPLEINTERFACES_H + diff --git a/r5dev/naveditor/include/Sample_Debug.h b/r5dev/naveditor/include/Sample_Debug.h new file mode 100644 index 00000000..c790aa54 --- /dev/null +++ b/r5dev/naveditor/include/Sample_Debug.h @@ -0,0 +1,63 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef RECASTSAMPLEDEBUG_H +#define RECASTSAMPLEDEBUG_H + +#include "NavEditor/Include/Sample.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Recast/Include/Recast.h" + +/// Sample used for random debugging. +class Sample_Debug : public Sample +{ +protected: + rcCompactHeightfield* m_chf; + rcContourSet* m_cset; + rcPolyMesh* m_pmesh; + + float m_halfExtents[3]; + float m_center[3]; + float m_bmin[3], m_bmax[3]; + dtPolyRef m_ref; + +public: + Sample_Debug(); + virtual ~Sample_Debug(); + + virtual void handleSettings(); + virtual void handleTools(); + virtual void handleDebugMode(); + virtual void handleClick(const float* s, const float* p, bool shift); + virtual void handleToggle(); + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); + virtual void handleMeshChanged(class InputGeom* geom); + virtual bool handleBuild(); + + virtual const float* getBoundsMin(); + virtual const float* getBoundsMax(); + +private: + // Explicitly disabled copy constructor and copy assignment operator. + Sample_Debug(const Sample_Debug&); + Sample_Debug& operator=(const Sample_Debug&); +}; + + +#endif // RECASTSAMPLE_H diff --git a/r5dev/naveditor/include/Sample_SoloMesh.h b/r5dev/naveditor/include/Sample_SoloMesh.h new file mode 100644 index 00000000..b60281ee --- /dev/null +++ b/r5dev/naveditor/include/Sample_SoloMesh.h @@ -0,0 +1,86 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef RECASTSAMPLESOLOMESH_H +#define RECASTSAMPLESOLOMESH_H + +#include "NavEditor/Include/Sample.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Recast/Include/Recast.h" + +class Sample_SoloMesh : public Sample +{ +protected: + bool m_keepInterResults; + float m_totalBuildTimeMs; + + unsigned char* m_triareas; + rcHeightfield* m_solid; + rcCompactHeightfield* m_chf; + rcContourSet* m_cset; + rcPolyMesh* m_pmesh; + rcConfig m_cfg; + rcPolyMeshDetail* m_dmesh; + + enum DrawMode + { + DRAWMODE_NAVMESH, + DRAWMODE_NAVMESH_TRANS, + DRAWMODE_NAVMESH_BVTREE, + DRAWMODE_NAVMESH_NODES, + DRAWMODE_NAVMESH_INVIS, + DRAWMODE_MESH, + DRAWMODE_VOXELS, + DRAWMODE_VOXELS_WALKABLE, + DRAWMODE_COMPACT, + DRAWMODE_COMPACT_DISTANCE, + DRAWMODE_COMPACT_REGIONS, + DRAWMODE_REGION_CONNECTIONS, + DRAWMODE_RAW_CONTOURS, + DRAWMODE_BOTH_CONTOURS, + DRAWMODE_CONTOURS, + DRAWMODE_POLYMESH, + DRAWMODE_POLYMESH_DETAIL, + MAX_DRAWMODE + }; + + DrawMode m_drawMode; + + void cleanup(); + +public: + Sample_SoloMesh(); + virtual ~Sample_SoloMesh(); + + virtual void handleSettings(); + virtual void handleTools(); + virtual void handleDebugMode(); + + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); + virtual void handleMeshChanged(class InputGeom* geom); + virtual bool handleBuild(); + +private: + // Explicitly disabled copy constructor and copy assignment operator. + Sample_SoloMesh(const Sample_SoloMesh&); + Sample_SoloMesh& operator=(const Sample_SoloMesh&); +}; + + +#endif // RECASTSAMPLESOLOMESHSIMPLE_H diff --git a/r5dev/naveditor/include/Sample_TempObstacles.h b/r5dev/naveditor/include/Sample_TempObstacles.h new file mode 100644 index 00000000..11a2679e --- /dev/null +++ b/r5dev/naveditor/include/Sample_TempObstacles.h @@ -0,0 +1,98 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef RECASTSAMPLETEMPOBSTACLE_H +#define RECASTSAMPLETEMPOBSTACLE_H + +#include "NavEditor/Include/Sample.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Recast/Include/Recast.h" +#include "NavEditor/Include/ChunkyTriMesh.h" + + +class Sample_TempObstacles : public Sample +{ +protected: + bool m_keepInterResults; + + struct LinearAllocator* m_talloc; + struct FastLZCompressor* m_tcomp; + struct MeshProcess* m_tmproc; + + class dtTileCache* m_tileCache; + + float m_cacheBuildTimeMs; + int m_cacheCompressedSize; + int m_cacheRawSize; + int m_cacheLayerCount; + unsigned int m_cacheBuildMemUsage; + + enum DrawMode + { + DRAWMODE_NAVMESH, + DRAWMODE_NAVMESH_TRANS, + DRAWMODE_NAVMESH_BVTREE, + DRAWMODE_NAVMESH_NODES, + DRAWMODE_NAVMESH_PORTALS, + DRAWMODE_NAVMESH_INVIS, + DRAWMODE_MESH, + DRAWMODE_CACHE_BOUNDS, + MAX_DRAWMODE + }; + + DrawMode m_drawMode; + + int m_maxTiles; + int m_maxPolysPerTile; + float m_tileSize; + +public: + Sample_TempObstacles(); + virtual ~Sample_TempObstacles(); + + virtual void handleSettings(); + virtual void handleTools(); + virtual void handleDebugMode(); + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); + virtual void handleMeshChanged(class InputGeom* geom); + virtual bool handleBuild(); + virtual void handleUpdate(const float dt); + + void getTilePos(const float* pos, int& tx, int& ty); + + void renderCachedTile(const int tx, const int ty, const int type); + void renderCachedTileOverlay(const int tx, const int ty, double* proj, double* model, int* view); + + void addTempObstacle(const float* pos); + void removeTempObstacle(const float* sp, const float* sq); + void clearAllTempObstacles(); + + void saveAll(const char* path); + void loadAll(const char* path); + +private: + // Explicitly disabled copy constructor and copy assignment operator. + Sample_TempObstacles(const Sample_TempObstacles&); + Sample_TempObstacles& operator=(const Sample_TempObstacles&); + + int rasterizeTileLayers(const int tx, const int ty, const rcConfig& cfg, struct TileCacheData* tiles, const int maxTiles); +}; + + +#endif // RECASTSAMPLETEMPOBSTACLE_H diff --git a/r5dev/naveditor/include/Sample_TileMesh.h b/r5dev/naveditor/include/Sample_TileMesh.h new file mode 100644 index 00000000..8c0cbfa9 --- /dev/null +++ b/r5dev/naveditor/include/Sample_TileMesh.h @@ -0,0 +1,114 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef RECASTSAMPLETILEMESH_H +#define RECASTSAMPLETILEMESH_H + +#include "NavEditor/Include/Sample.h" +#include "Detour/Include/DetourNavMesh.h" +#include "Recast/Include/Recast.h" +#include "NavEditor/Include/ChunkyTriMesh.h" + +class Sample_TileMesh : public Sample +{ +protected: + bool m_keepInterResults; + bool m_buildAll; + float m_totalBuildTimeMs; + + unsigned char* m_triareas; + rcHeightfield* m_solid; + rcCompactHeightfield* m_chf; + rcContourSet* m_cset; + rcPolyMesh* m_pmesh; + rcPolyMeshDetail* m_dmesh; + rcConfig m_cfg; + + enum DrawMode + { + DRAWMODE_NAVMESH, + DRAWMODE_NAVMESH_TRANS, + DRAWMODE_NAVMESH_BVTREE, + DRAWMODE_NAVMESH_NODES, + DRAWMODE_NAVMESH_PORTALS, + DRAWMODE_NAVMESH_INVIS, + DRAWMODE_MESH, + DRAWMODE_VOXELS, + DRAWMODE_VOXELS_WALKABLE, + DRAWMODE_COMPACT, + DRAWMODE_COMPACT_DISTANCE, + DRAWMODE_COMPACT_REGIONS, + DRAWMODE_REGION_CONNECTIONS, + DRAWMODE_RAW_CONTOURS, + DRAWMODE_BOTH_CONTOURS, + DRAWMODE_CONTOURS, + DRAWMODE_POLYMESH, + DRAWMODE_POLYMESH_DETAIL, + MAX_DRAWMODE + }; + + DrawMode m_drawMode; + + int m_maxTiles; + int m_maxPolysPerTile; + float m_tileSize; + + unsigned int m_tileCol; + float m_lastBuiltTileBmin[3]; + float m_lastBuiltTileBmax[3]; + float m_tileBuildTime; + float m_tileMemUsage; + int m_tileTriCount; + + unsigned char* buildTileMesh(const int tx, const int ty, const float* bmin, const float* bmax, int& dataSize); + + void cleanup(); + + void saveAll(const char* path, const dtNavMesh* mesh); + dtNavMesh* loadAll(const char* path); + +public: + Sample_TileMesh(); + virtual ~Sample_TileMesh(); + + virtual void handleSettings(); + virtual void handleTools(); + virtual void handleDebugMode(); + virtual void handleRender(); + virtual void handleRenderOverlay(double* proj, double* model, int* view); + virtual void handleMeshChanged(class InputGeom* geom); + virtual bool handleBuild(); + virtual void collectSettings(struct BuildSettings& settings); + + void getTilePos(const float* pos, int& tx, int& ty); + void getTileExtents(int tx, int ty, float* bmin, float* bmax); + + void buildTile(const float* pos); + void removeTile(const float* pos); + void buildAllTiles(); + void removeAllTiles(); + + void build_n_SaveAllHulls(); +private: + // Explicitly disabled copy constructor and copy assignment operator. + Sample_TileMesh(const Sample_TileMesh&); + Sample_TileMesh& operator=(const Sample_TileMesh&); +}; + + +#endif // RECASTSAMPLETILEMESH_H diff --git a/r5dev/naveditor/include/TestCase.h b/r5dev/naveditor/include/TestCase.h new file mode 100644 index 00000000..1bd6fc87 --- /dev/null +++ b/r5dev/naveditor/include/TestCase.h @@ -0,0 +1,114 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef TESTCASE_H +#define TESTCASE_H + +#include +#include "Detour/Include/DetourNavMesh.h" + +class TestCase +{ + enum TestType + { + TEST_PATHFIND, + TEST_RAYCAST, + }; + + struct Test + { + Test() : + type(), + spos(), + epos(), + nspos(), + nepos(), + radius(0), + includeFlags(0), + excludeFlags(0), + expand(false), + straight(0), + nstraight(0), + polys(0), + npolys(0), + findNearestPolyTime(0), + findPathTime(0), + findStraightPathTime(0), + next(0) + { + } + + ~Test() + { + delete [] straight; + delete [] polys; + } + + TestType type; + float spos[3]; + float epos[3]; + float nspos[3]; + float nepos[3]; + float radius; + unsigned short includeFlags; + unsigned short excludeFlags; + bool expand; + + float* straight; + int nstraight; + dtPolyRef* polys; + int npolys; + + int findNearestPolyTime; + int findPathTime; + int findStraightPathTime; + + Test* next; + private: + // Explicitly disabled copy constructor and copy assignment operator. + Test(const Test&); + Test& operator=(const Test&); + }; + + std::string m_sampleName; + std::string m_geomFileName; + Test* m_tests; + + void resetTimes(); + +public: + TestCase(); + ~TestCase(); + + bool load(const std::string& filePath); + + const std::string& getSampleName() const { return m_sampleName; } + const std::string& getGeomFileName() const { return m_geomFileName; } + + void doTests(class dtNavMesh* navmesh, class dtNavMeshQuery* navquery); + + void handleRender(); + bool handleRenderOverlay(double* proj, double* model, int* view); + +private: + // Explicitly disabled copy constructor and copy assignment operator. + TestCase(const TestCase&); + TestCase& operator=(const TestCase&); +}; + +#endif // TESTCASE_H \ No newline at end of file diff --git a/r5dev/naveditor/include/ValueHistory.h b/r5dev/naveditor/include/ValueHistory.h new file mode 100644 index 00000000..2da53902 --- /dev/null +++ b/r5dev/naveditor/include/ValueHistory.h @@ -0,0 +1,50 @@ +#ifndef VALUEHISTORY_H +#define VALUEHISTORY_H + +class ValueHistory +{ + static const int MAX_HISTORY = 256; + float m_samples[MAX_HISTORY]; + int m_hsamples; +public: + ValueHistory(); + + inline void addSample(const float val) + { + m_hsamples = (m_hsamples+MAX_HISTORY-1) % MAX_HISTORY; + m_samples[m_hsamples] = val; + } + + inline int getSampleCount() const + { + return MAX_HISTORY; + } + + inline float getSample(const int i) const + { + return m_samples[(m_hsamples+i) % MAX_HISTORY]; + } + + float getSampleMin() const; + float getSampleMax() const; + float getAverage() const; +}; + +struct GraphParams +{ + void setRect(int ix, int iy, int iw, int ih, int ipad); + void setValueRange(float ivmin, float ivmax, int indiv, const char* iunits); + + int x, y, w, h, pad; + float vmin, vmax; + int ndiv; + char units[16]; +}; + +void drawGraphBackground(const GraphParams* p); + +void drawGraph(const GraphParams* p, const ValueHistory* graph, + int idx, const char* label, const unsigned int col); + + +#endif // VALUEHISTORY_H \ No newline at end of file diff --git a/r5dev/naveditor/include/imgui.h b/r5dev/naveditor/include/imgui.h new file mode 100644 index 00000000..325829b3 --- /dev/null +++ b/r5dev/naveditor/include/imgui.h @@ -0,0 +1,108 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef IMGUI_H +#define IMGUI_H + +enum imguiMouseButton +{ + IMGUI_MBUT_LEFT = 0x01, + IMGUI_MBUT_RIGHT = 0x02, +}; + +enum imguiTextAlign +{ + IMGUI_ALIGN_LEFT, + IMGUI_ALIGN_CENTER, + IMGUI_ALIGN_RIGHT, +}; + +inline unsigned int imguiRGBA(unsigned char r, unsigned char g, unsigned char b, unsigned char a=255) +{ + return (r) | (g << 8) | (b << 16) | (a << 24); +} + +void imguiBeginFrame(int mx, int my, unsigned char mbut, int scroll); +void imguiEndFrame(); + +bool imguiBeginScrollArea(const char* name, int x, int y, int w, int h, int* scroll); +void imguiEndScrollArea(); + +void imguiIndent(); +void imguiUnindent(); +void imguiSeparator(); +void imguiSeparatorLine(); + +bool imguiButton(const char* text, bool enabled = true); +bool imguiItem(const char* text, bool enabled = true); +bool imguiCheck(const char* text, bool checked, bool enabled = true); +bool imguiCollapse(const char* text, const char* subtext, bool checked, bool enabled = true); +void imguiLabel(const char* text); +void imguiValue(const char* text); +bool imguiSlider(const char* text, float* val, float vmin, float vmax, float vinc, bool enabled = true); + +void imguiDrawText(int x, int y, int align, const char* text, unsigned int color); +void imguiDrawLine(float x0, float y0, float x1, float y1, float r, unsigned int color); +void imguiDrawRoundedRect(float x, float y, float w, float h, float r, unsigned int color); +void imguiDrawRect(float x, float y, float w, float h, unsigned int color); + +// Pull render interface. +enum imguiGfxCmdType +{ + IMGUI_GFXCMD_RECT, + IMGUI_GFXCMD_TRIANGLE, + IMGUI_GFXCMD_LINE, + IMGUI_GFXCMD_TEXT, + IMGUI_GFXCMD_SCISSOR, +}; + +struct imguiGfxRect +{ + short x,y,w,h,r; +}; + +struct imguiGfxText +{ + short x,y,align; + const char* text; +}; + +struct imguiGfxLine +{ + short x0,y0,x1,y1,r; +}; + +struct imguiGfxCmd +{ + char type; + char flags; + char pad[2]; + unsigned int col; + union + { + imguiGfxLine line; + imguiGfxRect rect; + imguiGfxText text; + }; +}; + +const imguiGfxCmd* imguiGetRenderQueue(); +int imguiGetRenderQueueSize(); + + +#endif // IMGUI_H diff --git a/r5dev/naveditor/include/imguiRenderGL.h b/r5dev/naveditor/include/imguiRenderGL.h new file mode 100644 index 00000000..89a57271 --- /dev/null +++ b/r5dev/naveditor/include/imguiRenderGL.h @@ -0,0 +1,26 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#ifndef IMGUI_RENDER_GL_H +#define IMGUI_RENDER_GL_H + +bool imguiRenderGLInit(const unsigned int* ttfBuffer); +void imguiRenderGLDestroy(); +void imguiRenderGLDraw(); + +#endif // IMGUI_RENDER_GL_H \ No newline at end of file diff --git a/r5dev/naveditor/main.cpp b/r5dev/naveditor/main.cpp new file mode 100644 index 00000000..34f7ee49 --- /dev/null +++ b/r5dev/naveditor/main.cpp @@ -0,0 +1,1234 @@ +// +// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// + +#include +#define _USE_MATH_DEFINES +#include + +#include "thirdparty/sdl/include/SDL.h" +#include "thirdparty/sdl/include/SDL_syswm.h" +#include "thirdparty/sdl/include/SDL_opengl.h" +#ifdef __APPLE__ +# include +#else +# include +#endif + +#include +#include + +#include "NavEditor/Include/imgui.h" +#include "NavEditor/Include/imguiRenderGL.h" + +#include "Recast/Include/Recast.h" +#include "DebugUtils/Include/RecastDebugDraw.h" +#include "NavEditor/Include/InputGeom.h" +#include "NavEditor/Include/TestCase.h" +#include "NavEditor/Include/Filelist.h" +#include "NavEditor/Include/Sample_SoloMesh.h" +#include "NavEditor/Include/Sample_TileMesh.h" +#include "NavEditor/Include/Sample_TempObstacles.h" +#include "NavEditor/Include/Sample_Debug.h" +#include "NavEditor/include/DroidSans.h" + +#include "Recast/Include/RecastAlloc.h" + + +#ifdef WIN32 +# define snprintf _snprintf +# define putenv _putenv +#include "commdlg.h" + +#endif + +using std::string; +using std::vector; + +struct SampleItem +{ + Sample* (*create)(); + const string name; +}; +Sample* createSolo() { return new Sample_SoloMesh(); } +Sample* createTile() { return new Sample_TileMesh(); } +Sample* createTempObstacle() { return new Sample_TempObstacles(); } +Sample* createDebug() { return new Sample_Debug(); } +static SampleItem g_samples[] = +{ + { createSolo, "Solo Mesh" }, + { createTile, "Tile Mesh" }, + { createTempObstacle, "Temp Obstacles" }, +}; +static const int g_nsamples = sizeof(g_samples) / sizeof(SampleItem); + +void save_ply(std::vector& pts,std::vector& colors,rcIntArray& tris) +{ + static int counter = 0; + char fname[255]; + sprintf(fname, "out_%d.ply", counter); + counter++; + auto f = fopen(fname, "wb"); + fprintf(f, +R"(ply +format ascii 1.0 +element vertex %d +property float x +property float y +property float z +property uchar red +property uchar green +property uchar blue +element face %d +property list uchar int vertex_index +end_header +)",pts.size()/3,tris.size()/3); + + for (size_t i = 0; i < pts.size(); i+=3) + { + auto c = colors[i / 3]; + fprintf(f, "%g %g %g %d %d %d\n", pts[i], pts[i + 1], pts[i + 2], c & 0xff, (c >> 8) & 0xff, (c >> 16) & 0xff); + } + for (size_t i = 0; i < tris.size(); i += 3) + { + fprintf(f, "3 %d %d %d\n", tris[i], tris[i + 1], tris[i + 2]); + } + + fclose(f); +} +float area2(const float* a, const float* b, const float* c) +{ + return (b[0] - a[0]) * (c[1] - a[1]) - (c[0] - a[0]) * (b[1] - a[1]); +} +void convex_hull(std::vector& pts, std::vector& hull) +{ + int pt_count = pts.size() / 3; + int cur_pt = 0; + float min_x = pts[0]; + for(size_t i=0;i 0) //reverse this comparison for flipped hull direction + endpoint = i; + } + point_on_hull = endpoint; + } while (endpoint != hull[0]); +} +float frand() +{ + return rand() / (float)RAND_MAX; +} +void generate_points(float* pts, int count, float dx, float dy, float dz) +{ + for (int i = 0; i < count; i++) + { + pts[i * 3+0] = frand()*dx * 2 - dx; + pts[i * 3+1] = frand()*dy * 2 - dy; + pts[i * 3+2] = frand()*dz * 2 - dz; + } +} + +void do_auto_load(const char* path, BuildContext& ctx,Sample* sample,InputGeom*& geom, string& meshName,bool& tf2_transforms) +{ + string geom_path = std::string(path); + meshName = geom_path.substr(geom_path.rfind("\\") + 1); + geom = new InputGeom; + if (!geom->load(&ctx, geom_path, tf2_transforms)) + { + delete geom; + geom = 0; + + // Destroy the sample if it already had geometry loaded, as we've just deleted it! + /*if (sample && sample->getInputGeom()) + { + delete sample; + sample = 0; + }*/ + ctx.dumpLog("Geom load log %s:", meshName.c_str()); + } + if (sample && geom) + { + sample->handleMeshChanged(geom); + sample->m_model_name = meshName.substr(0, meshName.size() - 4); + } +} + +void update_camera(const float* bmin, const float* bmax,float* cameraPos,float* cameraEulers,float& camr) +{ + // Reset camera and fog to match the mesh bounds. + if (bmin && bmax) + { + camr = sqrtf(rcSqr(bmax[0] - bmin[0]) + + rcSqr(bmax[1] - bmin[1]) + + rcSqr(bmax[2] - bmin[2])) / 2; + cameraPos[0] = (bmax[0] + bmin[0]) / 2 + camr; + cameraPos[1] = (bmax[1] + bmin[1]) / 2 + camr; + cameraPos[2] = (bmax[2] + bmin[2]) / 2 + camr; + camr *= 3; + } + cameraEulers[0] = 45; + cameraEulers[1] = -125; + glFogf(GL_FOG_START, camr * 0.1f); + glFogf(GL_FOG_END, camr * 1.25f); +} +#if 1 +int main(int argc, char** argv) +#else +#define STB_IMAGE_WRITE_IMPLEMENTATION +#include "stb_image_write.h" +//just quick tests for stuff + + + +extern void delaunayHull(rcContext* ctx, const int npts, const float* pts, + const int nhull, const int* hull, + rcIntArray& tris, rcIntArray& edges); + +int main_test_delaunay(int /*argc*/, char** /*argv*/) +{ + rcContext ctx; + std::vector pts(25*3); + std::vector colors(pts.size() / 3, 0xffffffff); + generate_points(pts.data(), pts.size()/3, 10, 10, 2); + + + std::vector hull; + convex_hull(pts,hull); + + for (auto h : hull) + colors[h] = 0xffff0000; + + rcIntArray tris; + save_ply(pts, colors, tris); + + rcIntArray edges; + delaunayHull(&ctx, pts.size()/3, pts.data(), hull.size(), hull.data(), tris, edges); + save_ply(pts, colors, tris); + return 0; +} +void compact_tris(rcIntArray& tris) +{ + int j = 3; + for (int i = 4; i < tris.size(); i++) + { + if (i % 4 == 3) continue; + tris[j] = tris[i]; + j++; + } + tris.resize(j); +} +int main(int argc, char** argv) +{ + srand(17); + rcContext ctx; + std::vector pts(8 * 3); + std::vector colors(pts.size() / 3, 0xffffffff); + generate_points(pts.data(), pts.size() / 3, 10, 10, 10); + + + std::vector hull; + convex_hull(pts, hull); + + for (auto h : hull) + colors[h] = 0xffff0000; + + rcIntArray tris; + //save_ply(pts, colors, tris); + + rcIntArray edges; + delaunayHull(&ctx, pts.size() / 3, pts.data(), hull.size(), hull.data(), tris, edges); + compact_tris(tris); + save_ply(pts, colors, tris); + int tri_count = tris.size() / 3; + std::vector areas; + areas.resize(tri_count); + for (int i = 0; i < tri_count; i++) + areas[i] = i; + + + float bmin[3]; + float bmax[3]; + rcCalcBounds(pts.data(), pts.size()/3, bmin, bmax); + + float cellSize = .05f; + float cellHeight = .05f; + + int width; + int height; + + rcCalcGridSize(bmin, bmax, cellSize, &width, &height); + + rcHeightfield solid; + rcCreateHeightfield(&ctx, solid, width, height, bmin, bmax, cellSize, cellHeight); + + int flagMergeThr = 1; + + rcRasterizeTriangles(&ctx, pts.data(), pts.size() / 3, tris.data(), areas.data(), tri_count, solid, flagMergeThr); + + std::vector img_data(width*height); + float zdelta = bmax[2] - bmin[2]; + for(int x=0;xarea*(235/ (float)tri_count) +20; +#else + img_data[x + y * width]=((s->smax*cellHeight)/zdelta)*255; +#endif + //img_data[x + y * width] = 255; + } + } + + stbi_write_png("hmap.png", width, height, 1, img_data.data(), width); + + return 0; +} +int not_main(int argc, char** argv) +#endif +{ + // Init SDL + if (SDL_Init(SDL_INIT_EVERYTHING) != 0) + { + printf("Could not initialise SDL.\nError: %s\n", SDL_GetError()); + return -1; + } + + // Enable depth buffer. + SDL_GL_SetAttribute(SDL_GL_DOUBLEBUFFER, 1); + SDL_GL_SetAttribute(SDL_GL_DEPTH_SIZE, 24); + + // Set color channel depth. + SDL_GL_SetAttribute(SDL_GL_RED_SIZE, 8); + SDL_GL_SetAttribute(SDL_GL_GREEN_SIZE, 8); + SDL_GL_SetAttribute(SDL_GL_BLUE_SIZE, 8); + SDL_GL_SetAttribute(SDL_GL_ALPHA_SIZE, 8); + + // 4x MSAA. + SDL_GL_SetAttribute(SDL_GL_MULTISAMPLEBUFFERS, 1); + SDL_GL_SetAttribute(SDL_GL_MULTISAMPLESAMPLES, 4); + + SDL_DisplayMode displayMode; + SDL_GetCurrentDisplayMode(0, &displayMode); + + bool presentationMode = false; + Uint32 flags = SDL_WINDOW_OPENGL | SDL_RENDERER_PRESENTVSYNC; + int width; + int height; + if (presentationMode) + { + // Create a fullscreen window at the native resolution. + width = displayMode.w; + height = displayMode.h; + flags |= SDL_WINDOW_FULLSCREEN; + } + else + { + float aspect = 16.0f / 9.0f; + width = rcMin(displayMode.w, (int)(displayMode.h * aspect)) - 80; + height = displayMode.h - 80; + } + + SDL_Window* window; + SDL_Renderer* renderer; + int errorCode = SDL_CreateWindowAndRenderer(width, height, flags, &window, &renderer); + + if (errorCode != 0 || !window || !renderer) + { + printf("Could not initialise SDL opengl\nError: %s\n", SDL_GetError()); + return -1; + } + + SDL_SetWindowPosition(window, SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED); + SDL_GL_CreateContext(window); + + if (!imguiRenderGLInit(droidsans_data)) + { + printf("Could not init GUI renderer.\n"); + SDL_Quit(); + return -1; + } + + const char* auto_load = nullptr; + if (argc > 1) + { + if (strcmp(argv[1], "-console") == 0) + { + if (argc > 2) + { + auto_load = argv[2]; + } + } + else + { + FreeConsole(); + auto_load = argv[1]; + } + } + + float t = 0.0f; + float timeAcc = 0.0f; + Uint32 prevFrameTime = SDL_GetTicks(); + int mousePos[2] = {0, 0}; + int origMousePos[2] = {0, 0}; // Used to compute mouse movement totals across frames. + + float cameraEulers[] = {45, 45}; + float cameraPos[] = {0, 0, 0}; + float camr = 1000; + float origCameraEulers[] = {0, 0}; // Used to compute rotational changes across frames. + + float moveFront = 0.0f, moveBack = 0.0f, moveLeft = 0.0f, moveRight = 0.0f, moveUp = 0.0f, moveDown = 0.0f; + + float scrollZoom = 0; + bool rotate = false; + bool movedDuringRotate = false; + float rayStart[3]; + float rayEnd[3]; + bool mouseOverMenu = false; + + bool showMenu = !presentationMode; + bool showLog = false; + bool showTools = true; + bool showLevels = false; + bool showSample = false; + bool showTestCases = false; + + // Window scroll positions. + int propScroll = 0; + int logScroll = 0; + int toolsScroll = 0; + + string sampleName = "Choose Sample..."; + + vector files; + const string meshesFolder = "Meshes"; + string meshName = "Choose Mesh..."; + + float markerPosition[3] = {0, 0, 0}; + bool markerPositionSet = false; + + InputGeom* geom = 0; + Sample* sample = 0; + bool tf2_transforms = false; + + const string testCasesFolder = "TestCases"; + TestCase* test = 0; + + BuildContext ctx; + + //Load tiled sample + + sample = g_samples[1].create(); + sampleName = g_samples[1].name; + sample->is_tf2 = &tf2_transforms; + sample->setContext(&ctx); + if (geom) + { + sample->handleMeshChanged(geom); + } + if (auto_load) + { + do_auto_load(auto_load, ctx, sample, geom, meshName, tf2_transforms); + if (geom || sample) + { + const float* bmin = 0; + const float* bmax = 0; + if (geom) + { + bmin = geom->getNavMeshBoundsMin(); + bmax = geom->getNavMeshBoundsMax(); + } + update_camera(bmin, bmax, cameraPos, cameraEulers, camr); + } + if (argc > 2) + { + auto ts = dynamic_cast(sample); + ts->build_n_SaveAllHulls(); + return 0; + } + } + // Fog. + float fogColor[4] = { 0.32f, 0.31f, 0.30f, 1.0f }; + glEnable(GL_FOG); + glFogi(GL_FOG_MODE, GL_LINEAR); + glFogf(GL_FOG_START, camr * 0.1f); + glFogf(GL_FOG_END, camr * 1.25f); + glFogfv(GL_FOG_COLOR, fogColor); + + glEnable(GL_CULL_FACE); + glDepthFunc(GL_LEQUAL); + + bool done = false; + while(!done) + { + // Handle input events. + int mouseScroll = 0; + bool processHitTest = false; + bool processHitTestShift = false; + + SDL_Event event; + + while (SDL_PollEvent(&event)) + { + switch (event.type) + { + case SDL_KEYDOWN: + // Handle any key presses here. + if (event.key.keysym.sym == SDLK_ESCAPE) + { + done = true; + } + else if (event.key.keysym.sym == SDLK_t) + { + showLevels = false; + showSample = false; + showTestCases = true; + scanDirectory(testCasesFolder, ".txt", files); + } + else if (event.key.keysym.sym == SDLK_TAB) + { + showMenu = !showMenu; + } + else if (event.key.keysym.sym == SDLK_SPACE) + { + if (sample) + sample->handleToggle(); + } + else if (event.key.keysym.sym == SDLK_1) + { + if (sample) + sample->handleStep(); + } + else if (event.key.keysym.sym == SDLK_9) + { + if (sample && geom) + { + string savePath = meshesFolder + "/"; + BuildSettings settings; + memset(&settings, 0, sizeof(settings)); + + rcVcopy(settings.navMeshBMin, geom->getNavMeshBoundsMin()); + rcVcopy(settings.navMeshBMax, geom->getNavMeshBoundsMax()); + + sample->collectSettings(settings); + + geom->saveGeomSet(&settings); + } + } + break; + + case SDL_MOUSEWHEEL: + if (event.wheel.y < 0) + { + // wheel down + if (mouseOverMenu) + { + mouseScroll++; + } + else + { + scrollZoom += 1.0f; + } + } + else + { + if (mouseOverMenu) + { + mouseScroll--; + } + else + { + scrollZoom -= 1.0f; + } + } + break; + case SDL_MOUSEBUTTONDOWN: + if (event.button.button == SDL_BUTTON_RIGHT) + { + if (!mouseOverMenu) + { + // Rotate view + rotate = true; + movedDuringRotate = false; + origMousePos[0] = mousePos[0]; + origMousePos[1] = mousePos[1]; + origCameraEulers[0] = cameraEulers[0]; + origCameraEulers[1] = cameraEulers[1]; + } + } + break; + + case SDL_MOUSEBUTTONUP: + // Handle mouse clicks here. + if (event.button.button == SDL_BUTTON_RIGHT) + { + rotate = false; + if (!mouseOverMenu) + { + if (!movedDuringRotate) + { + processHitTest = true; + processHitTestShift = true; + } + } + } + else if (event.button.button == SDL_BUTTON_LEFT) + { + if (!mouseOverMenu) + { + processHitTest = true; + processHitTestShift = (SDL_GetModState() & KMOD_SHIFT) ? true : false; + } + } + + break; + + case SDL_MOUSEMOTION: + mousePos[0] = event.motion.x; + mousePos[1] = height-1 - event.motion.y; + + if (rotate) + { + int dx = mousePos[0] - origMousePos[0]; + int dy = mousePos[1] - origMousePos[1]; + cameraEulers[0] = origCameraEulers[0] - dy * 0.25f; + cameraEulers[1] = origCameraEulers[1] + dx * 0.25f; + if (dx * dx + dy * dy > 3 * 3) + { + movedDuringRotate = true; + } + } + break; + + case SDL_QUIT: + done = true; + break; + + default: + break; + } + } + + unsigned char mouseButtonMask = 0; + if (SDL_GetMouseState(0, 0) & SDL_BUTTON_LMASK) + mouseButtonMask |= IMGUI_MBUT_LEFT; + if (SDL_GetMouseState(0, 0) & SDL_BUTTON_RMASK) + mouseButtonMask |= IMGUI_MBUT_RIGHT; + + Uint32 time = SDL_GetTicks(); + float dt = (time - prevFrameTime) / 1000.0f; + prevFrameTime = time; + + t += dt; + + // Hit test mesh. + if (processHitTest && geom && sample) + { + float hitTime; + bool hit = geom->raycastMesh(rayStart, rayEnd, hitTime); + + if (hit) + { + if (SDL_GetModState() & KMOD_CTRL) + { + // Marker + markerPositionSet = true; + markerPosition[0] = rayStart[0] + (rayEnd[0] - rayStart[0]) * hitTime; + markerPosition[1] = rayStart[1] + (rayEnd[1] - rayStart[1]) * hitTime; + markerPosition[2] = rayStart[2] + (rayEnd[2] - rayStart[2]) * hitTime; + } + else + { + float pos[3]; + pos[0] = rayStart[0] + (rayEnd[0] - rayStart[0]) * hitTime; + pos[1] = rayStart[1] + (rayEnd[1] - rayStart[1]) * hitTime; + pos[2] = rayStart[2] + (rayEnd[2] - rayStart[2]) * hitTime; + sample->handleClick(rayStart, pos, processHitTestShift); + } + } + else + { + if (SDL_GetModState() & KMOD_CTRL) + { + // Marker + markerPositionSet = false; + } + } + } + + // Update sample simulation. + const float SIM_RATE = 20; + const float DELTA_TIME = 1.0f / SIM_RATE; + timeAcc = rcClamp(timeAcc + dt, -1.0f, 1.0f); + int simIter = 0; + while (timeAcc > DELTA_TIME) + { + timeAcc -= DELTA_TIME; + if (simIter < 5 && sample) + { + sample->handleUpdate(DELTA_TIME); + } + simIter++; + } + + // Clamp the framerate so that we do not hog all the CPU. + const float MIN_FRAME_TIME = 1.0f / 40.0f; + if (dt < MIN_FRAME_TIME) + { + int ms = (int)((MIN_FRAME_TIME - dt) * 1000.0f); + if (ms > 10) ms = 10; + if (ms >= 0) SDL_Delay(ms); + } + + // Set the viewport. + glViewport(0, 0, width, height); + GLint viewport[4]; + glGetIntegerv(GL_VIEWPORT, viewport); + + // Clear the screen + glClearColor(0.3f, 0.3f, 0.32f, 1.0f); + glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); + glEnable(GL_BLEND); + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + glDisable(GL_TEXTURE_2D); + glEnable(GL_DEPTH_TEST); + + // Compute the projection matrix. + glMatrixMode(GL_PROJECTION); + glLoadIdentity(); + gluPerspective(50.0f, (float)width/(float)height, 1.0f, camr); + GLdouble projectionMatrix[16]; + glGetDoublev(GL_PROJECTION_MATRIX, projectionMatrix); + + // Compute the modelview matrix. + glMatrixMode(GL_MODELVIEW); + glLoadIdentity(); + glRotatef(cameraEulers[0], 1, 0, 0); + glRotatef(cameraEulers[1], 0, 1, 0); + float mXZY_to_XYZ[16] = + { + 1,0,0,0, + 0,0,-1,0, //tbh not sure why this is needed, the tri flips again? something is very stupid... + 0,1,0,0, + 0,0,0,1 + }; + glMultMatrixf(mXZY_to_XYZ); + glTranslatef(-cameraPos[0], -cameraPos[1], -cameraPos[2]); + GLdouble modelviewMatrix[16]; + glGetDoublev(GL_MODELVIEW_MATRIX, modelviewMatrix); + + // Get hit ray position and direction. + GLdouble x, y, z; + gluUnProject(mousePos[0], mousePos[1], 0.0f, modelviewMatrix, projectionMatrix, viewport, &x, &y, &z); + rayStart[0] = (float)x; + rayStart[1] = (float)y; + rayStart[2] = (float)z; + gluUnProject(mousePos[0], mousePos[1], 1.0f, modelviewMatrix, projectionMatrix, viewport, &x, &y, &z); + rayEnd[0] = (float)x; + rayEnd[1] = (float)y; + rayEnd[2] = (float)z; + + // Handle keyboard movement. + const Uint8* keystate = SDL_GetKeyboardState(NULL); + moveFront = rcClamp(moveFront + dt * 4 * ((keystate[SDL_SCANCODE_W] || keystate[SDL_SCANCODE_UP ]) ? 1 : -1), 0.0f, 1.0f); + moveLeft = rcClamp(moveLeft + dt * 4 * ((keystate[SDL_SCANCODE_A] || keystate[SDL_SCANCODE_LEFT ]) ? 1 : -1), 0.0f, 1.0f); + moveBack = rcClamp(moveBack + dt * 4 * ((keystate[SDL_SCANCODE_S] || keystate[SDL_SCANCODE_DOWN ]) ? 1 : -1), 0.0f, 1.0f); + moveRight = rcClamp(moveRight + dt * 4 * ((keystate[SDL_SCANCODE_D] || keystate[SDL_SCANCODE_RIGHT ]) ? 1 : -1), 0.0f, 1.0f); + moveUp = rcClamp(moveUp + dt * 4 * ((keystate[SDL_SCANCODE_Q] || keystate[SDL_SCANCODE_PAGEUP ]) ? 1 : -1), 0.0f, 1.0f); + moveDown = rcClamp(moveDown + dt * 4 * ((keystate[SDL_SCANCODE_E] || keystate[SDL_SCANCODE_PAGEDOWN ]) ? 1 : -1), 0.0f, 1.0f); + + float keybSpeed = 8800.0f; + if (SDL_GetModState() & KMOD_SHIFT) + { + keybSpeed *= 2.0f; + } + + float movex = (moveRight - moveLeft) * keybSpeed * dt; + float movey = (moveBack - moveFront) * keybSpeed * dt + scrollZoom * 2.0f; + scrollZoom = 0; + + cameraPos[0] += movex * (float)modelviewMatrix[0]; + cameraPos[1] += movex * (float)modelviewMatrix[4]; + cameraPos[2] += movex * (float)modelviewMatrix[8]; + + cameraPos[0] += movey * (float)modelviewMatrix[2]; + cameraPos[1] += movey * (float)modelviewMatrix[6]; + cameraPos[2] += movey * (float)modelviewMatrix[10]; + + cameraPos[1] += (moveUp - moveDown) * keybSpeed * dt; + + glEnable(GL_FOG); + + if (sample) + sample->handleRender(); + if (test) + test->handleRender(); + + glDisable(GL_FOG); + + // Render GUI + glDisable(GL_DEPTH_TEST); + glMatrixMode(GL_PROJECTION); + glLoadIdentity(); + gluOrtho2D(0, width, 0, height); + glMatrixMode(GL_MODELVIEW); + glLoadIdentity(); + + mouseOverMenu = false; + + imguiBeginFrame(mousePos[0], mousePos[1], mouseButtonMask, mouseScroll); + + if (sample) + { + sample->handleRenderOverlay((double*)projectionMatrix, (double*)modelviewMatrix, (int*)viewport); + } + if (test) + { + if (test->handleRenderOverlay((double*)projectionMatrix, (double*)modelviewMatrix, (int*)viewport)) + mouseOverMenu = true; + } + + // Help text. + if (showMenu) + { + const char msg[] = "W/S/A/D: Move RMB: Rotate"; + imguiDrawText(280, height-20, IMGUI_ALIGN_LEFT, msg, imguiRGBA(255,255,255,128)); + } + string geom_path; + if (showMenu) + { + if (imguiBeginScrollArea("Properties", width-250-10, 10, 250, height-20, &propScroll)) + mouseOverMenu = true; + + if (imguiCheck("Show Log", showLog)) + showLog = !showLog; + if (imguiCheck("Show Tools", showTools)) + showTools = !showTools; + + imguiSeparator(); + imguiLabel("Sample"); + if (imguiButton(sampleName.c_str())) + { + if (showSample) + { + showSample = false; + } + else + { + showSample = true; + showLevels = false; + showTestCases = false; + } + } + + imguiSeparator(); + //if (imguiCheck("Import/Export TF2", tf2_transforms, true)) + // tf2_transforms = !tf2_transforms; + imguiLabel("Input Mesh"); + if (imguiButton("Load mesh...")) + { + char szFile[260]; + OPENFILENAMEA diag = { 0 }; + diag.lStructSize = sizeof(diag); + + SDL_SysWMinfo sdlinfo; + SDL_version sdlver; + SDL_VERSION(&sdlver); + sdlinfo.version = sdlver; + SDL_GetWindowWMInfo(window, &sdlinfo); + + diag.hwndOwner = sdlinfo.info.win.window; + + diag.lpstrFile = szFile; + diag.lpstrFile[0] = 0; + diag.nMaxFile = sizeof(szFile); + diag.lpstrFilter = "OBJ\0*.obj\0Ply\0*.ply\0All\0*.*\0"; //TODO: BSP\0*.bsp\0 + diag.nFilterIndex = 1; + diag.lpstrFileTitle = NULL; + diag.nMaxFileTitle = 0; + diag.lpstrInitialDir = NULL; + diag.Flags = OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST; + + if (GetOpenFileNameA(&diag)) + { + geom_path = std::string(szFile); + meshName = geom_path.substr(geom_path.rfind("\\")+1); + } + } + if (imguiButton(meshName.c_str())) + { + if (showLevels) + { + showLevels = false; + } + else + { + showSample = false; + showTestCases = false; + showLevels = true; + scanDirectory(meshesFolder, ".obj", files); + scanDirectoryAppend(meshesFolder, ".gset", files); + scanDirectoryAppend(meshesFolder, ".ply", files); + } + } + if (geom) + { + char text[64]; + snprintf(text, 64, "Verts: %.1fk Tris: %.1fk", + geom->getMesh()->getVertCount()/1000.0f, + geom->getMesh()->getTriCount()/1000.0f); + imguiValue(text); + } + imguiSeparator(); + + if (geom && sample) + { + imguiSeparatorLine(); + + sample->handleSettings(); + + if (imguiButton("Build")) + { + ctx.resetLog(); + if (!sample->handleBuild()) + { + showLog = true; + logScroll = 0; + } + ctx.dumpLog("Build log %s:", meshName.c_str()); + + // Clear test. + delete test; + test = 0; + } + + imguiSeparator(); + } + + if (sample) + { + imguiSeparatorLine(); + sample->handleDebugMode(); + } + + imguiEndScrollArea(); + } + + // Sample selection dialog. + if (showSample) + { + static int levelScroll = 0; + if (imguiBeginScrollArea("Choose Sample", width-10-250-10-200, height-10-250, 200, 250, &levelScroll)) + mouseOverMenu = true; + + Sample* newSample = 0; + for (int i = 0; i < g_nsamples; ++i) + { + if (imguiItem(g_samples[i].name.c_str())) + { + newSample = g_samples[i].create(); + if (newSample) + { + sampleName = g_samples[i].name; + newSample->is_tf2 = &tf2_transforms; + } + } + } + if (newSample) + { + delete sample; + sample = newSample; + sample->setContext(&ctx); + if (geom) + { + sample->handleMeshChanged(geom); + } + showSample = false; + } + + if (geom || sample) + { + const float* bmin = 0; + const float* bmax = 0; + if (geom) + { + bmin = geom->getNavMeshBoundsMin(); + bmax = geom->getNavMeshBoundsMax(); + } + // Reset camera and fog to match the mesh bounds. + update_camera(bmin, bmax, cameraPos, cameraEulers, camr); + } + + imguiEndScrollArea(); + } + + // Level selection dialog. + if (showLevels) + { + static int levelScroll = 0; + if (imguiBeginScrollArea("Choose Level", width - 10 - 250 - 10 - 200, height - 10 - 450, 200, 450, &levelScroll)) + mouseOverMenu = true; + + vector::const_iterator fileIter = files.begin(); + vector::const_iterator filesEnd = files.end(); + vector::const_iterator levelToLoad = filesEnd; + for (; fileIter != filesEnd; ++fileIter) + { + if (imguiItem(fileIter->c_str())) + { + levelToLoad = fileIter; + } + } + + if (levelToLoad != filesEnd) + { + meshName = *levelToLoad; + showLevels = false; + + delete geom; + geom = 0; + + geom_path= meshesFolder + "/" + meshName; + } + + imguiEndScrollArea(); + + } + if (!geom_path.empty()) + { + geom = new InputGeom; + if (!geom->load(&ctx, geom_path, tf2_transforms)) + { + delete geom; + geom = 0; + + // Destroy the sample if it already had geometry loaded, as we've just deleted it! + if (sample && sample->getInputGeom()) + { + delete sample; + sample = 0; + } + + showLog = true; + logScroll = 0; + ctx.dumpLog("Geom load log %s:", meshName.c_str()); + } + if (sample && geom) + { + sample->handleMeshChanged(geom); + sample->m_model_name = meshName.substr(0, meshName.size() - 4); + } + + if (geom || sample) + { + const float* bmin = 0; + const float* bmax = 0; + if (geom) + { + bmin = geom->getNavMeshBoundsMin(); + bmax = geom->getNavMeshBoundsMax(); + } + // Reset camera and fog to match the mesh bounds. + update_camera(bmin, bmax, cameraPos, cameraEulers, camr); + } + } + // Test cases + if (showTestCases) + { + static int testScroll = 0; + if (imguiBeginScrollArea("Choose Test To Run", width-10-250-10-200, height-10-450, 200, 450, &testScroll)) + mouseOverMenu = true; + + vector::const_iterator fileIter = files.begin(); + vector::const_iterator filesEnd = files.end(); + vector::const_iterator testToLoad = filesEnd; + for (; fileIter != filesEnd; ++fileIter) + { + if (imguiItem(fileIter->c_str())) + { + testToLoad = fileIter; + } + } + + if (testToLoad != filesEnd) + { + string path = testCasesFolder + "/" + *testToLoad; + test = new TestCase; + if (test) + { + // Load the test. + if (!test->load(path)) + { + delete test; + test = 0; + } + + // Create sample + Sample* newSample = 0; + for (int i = 0; i < g_nsamples; ++i) + { + if (g_samples[i].name == test->getSampleName()) + { + newSample = g_samples[i].create(); + if (newSample) + { + sampleName = g_samples[i].name; + newSample->is_tf2 = &tf2_transforms; + } + } + } + + delete sample; + sample = newSample; + + if (sample) + { + sample->setContext(&ctx); + showSample = false; + } + + // Load geom. + meshName = test->getGeomFileName(); + + + path = meshesFolder + "/" + meshName; + + delete geom; + geom = new InputGeom; + if (!geom || !geom->load(&ctx, path,tf2_transforms)) + { + delete geom; + geom = 0; + delete sample; + sample = 0; + showLog = true; + logScroll = 0; + ctx.dumpLog("Geom load log %s:", meshName.c_str()); + } + if (sample && geom) + { + sample->handleMeshChanged(geom); + sample->m_model_name = meshName.substr(0, meshName.size() - 4); + } + + // This will ensure that tile & poly bits are updated in tiled sample. + if (sample) + sample->handleSettings(); + + ctx.resetLog(); + if (sample && !sample->handleBuild()) + { + ctx.dumpLog("Build log %s:", meshName.c_str()); + } + + if (geom || sample) + { + const float* bmin = 0; + const float* bmax = 0; + if (geom) + { + bmin = geom->getNavMeshBoundsMin(); + bmax = geom->getNavMeshBoundsMax(); + } + // Reset camera and fog to match the mesh bounds. + update_camera(bmin, bmax, cameraPos, cameraEulers, camr); + } + + // Do the tests. + if (sample) + test->doTests(sample->getNavMesh(), sample->getNavMeshQuery()); + } + } + + imguiEndScrollArea(); + } + + + // Log + if (showLog && showMenu) + { + if (imguiBeginScrollArea("Log", 250 + 20, 10, width - 300 - 250, 200, &logScroll)) + mouseOverMenu = true; + for (int i = 0; i < ctx.getLogCount(); ++i) + imguiLabel(ctx.getLogText(i)); + imguiEndScrollArea(); + } + + // Left column tools menu + if (!showTestCases && showTools && showMenu) // && geom && sample) + { + if (imguiBeginScrollArea("Tools", 10, 10, 250, height - 20, &toolsScroll)) + mouseOverMenu = true; + + if (sample) + sample->handleTools(); + + imguiEndScrollArea(); + } + + // Marker + if (markerPositionSet && gluProject((GLdouble)markerPosition[0], (GLdouble)markerPosition[1], (GLdouble)markerPosition[2], + modelviewMatrix, projectionMatrix, viewport, &x, &y, &z)) + { + // Draw marker circle + glLineWidth(5.0f); + glColor4ub(240,220,0,196); + glBegin(GL_LINE_LOOP); + const float r = 25.0f; + for (int i = 0; i < 20; ++i) + { + const float a = (float)i / 20.0f * RC_PI*2; + const float fx = (float)x + cosf(a)*r; + const float fy = (float)y + sinf(a)*r; + glVertex2f(fx,fy); + } + glEnd(); + glLineWidth(1.0f); + } + + imguiEndFrame(); + imguiRenderGLDraw(); + + glEnable(GL_DEPTH_TEST); + SDL_GL_SwapWindow(window); + + } + + imguiRenderGLDestroy(); + + SDL_Quit(); + + delete sample; + delete geom; + + return 0; +} diff --git a/r5dev/thirdparty/fastlz/fastlz.c b/r5dev/thirdparty/fastlz/fastlz.c new file mode 100644 index 00000000..645d2947 --- /dev/null +++ b/r5dev/thirdparty/fastlz/fastlz.c @@ -0,0 +1,556 @@ +/* + FastLZ - lightning-fast lossless compression library + + Copyright (C) 2007 Ariya Hidayat (ariya@kde.org) + Copyright (C) 2006 Ariya Hidayat (ariya@kde.org) + Copyright (C) 2005 Ariya Hidayat (ariya@kde.org) + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in + all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + THE SOFTWARE. +*/ + +#if !defined(FASTLZ__COMPRESSOR) && !defined(FASTLZ_DECOMPRESSOR) + +/* + * Always check for bound when decompressing. + * Generally it is best to leave it defined. + */ +#define FASTLZ_SAFE + +/* + * Give hints to the compiler for branch prediction optimization. + */ +#if defined(__GNUC__) && (__GNUC__ > 2) +#define FASTLZ_EXPECT_CONDITIONAL(c) (__builtin_expect((c), 1)) +#define FASTLZ_UNEXPECT_CONDITIONAL(c) (__builtin_expect((c), 0)) +#else +#define FASTLZ_EXPECT_CONDITIONAL(c) (c) +#define FASTLZ_UNEXPECT_CONDITIONAL(c) (c) +#endif + +/* + * Use inlined functions for supported systems. + */ +#if defined(__GNUC__) || defined(__DMC__) || defined(__POCC__) || defined(__WATCOMC__) || defined(__SUNPRO_C) +#define FASTLZ_INLINE inline +#elif defined(__BORLANDC__) || defined(_MSC_VER) || defined(__LCC__) +#define FASTLZ_INLINE __inline +#else +#define FASTLZ_INLINE +#endif + +/* + * Prevent accessing more than 8-bit at once, except on x86 architectures. + */ +#if !defined(FASTLZ_STRICT_ALIGN) +#define FASTLZ_STRICT_ALIGN +#if defined(__i386__) || defined(__386) /* GNU C, Sun Studio */ +#undef FASTLZ_STRICT_ALIGN +#elif defined(__i486__) || defined(__i586__) || defined(__i686__) /* GNU C */ +#undef FASTLZ_STRICT_ALIGN +#elif defined(_M_IX86) /* Intel, MSVC */ +#undef FASTLZ_STRICT_ALIGN +#elif defined(__386) +#undef FASTLZ_STRICT_ALIGN +#elif defined(_X86_) /* MinGW */ +#undef FASTLZ_STRICT_ALIGN +#elif defined(__I86__) /* Digital Mars */ +#undef FASTLZ_STRICT_ALIGN +#endif +#endif + +/* + * FIXME: use preprocessor magic to set this on different platforms! + */ +typedef unsigned char flzuint8; +typedef unsigned short flzuint16; +typedef unsigned int flzuint32; + +/* Disable "conversion from A to B, possible loss of data" warning when using MSVC */ +#if defined(_MSC_VER) +#pragma warning(disable: 4244) +#endif + +/* prototypes */ +int fastlz_compress(const void* input, int length, void* output); +int fastlz_compress_level(int level, const void* input, int length, void* output); +int fastlz_decompress(const void* input, int length, void* output, int maxout); + +#define MAX_COPY 32 +#define MAX_LEN 264 /* 256 + 8 */ +#define MAX_DISTANCE 8192 + +#if !defined(FASTLZ_STRICT_ALIGN) +#define FASTLZ_READU16(p) *((const flzuint16*)(p)) +#else +#define FASTLZ_READU16(p) ((p)[0] | (p)[1]<<8) +#endif + +#define HASH_LOG 13 +#define HASH_SIZE (1<< HASH_LOG) +#define HASH_MASK (HASH_SIZE-1) +#define HASH_FUNCTION(v,p) { v = FASTLZ_READU16(p); v ^= FASTLZ_READU16(p+1)^(v>>(16-HASH_LOG));v &= HASH_MASK; } + +#undef FASTLZ_LEVEL +#define FASTLZ_LEVEL 1 + +#undef FASTLZ_COMPRESSOR +#undef FASTLZ_DECOMPRESSOR +#define FASTLZ_COMPRESSOR fastlz1_compress +#define FASTLZ_DECOMPRESSOR fastlz1_decompress +static FASTLZ_INLINE int FASTLZ_COMPRESSOR(const void* input, int length, void* output); +static FASTLZ_INLINE int FASTLZ_DECOMPRESSOR(const void* input, int length, void* output, int maxout); +#include "fastlz.c" + +#undef FASTLZ_LEVEL +#define FASTLZ_LEVEL 2 + +#undef MAX_DISTANCE +#define MAX_DISTANCE 8191 +#define MAX_FARDISTANCE (65535+MAX_DISTANCE-1) + +#undef FASTLZ_COMPRESSOR +#undef FASTLZ_DECOMPRESSOR +#define FASTLZ_COMPRESSOR fastlz2_compress +#define FASTLZ_DECOMPRESSOR fastlz2_decompress +static FASTLZ_INLINE int FASTLZ_COMPRESSOR(const void* input, int length, void* output); +static FASTLZ_INLINE int FASTLZ_DECOMPRESSOR(const void* input, int length, void* output, int maxout); +#include "fastlz.c" + +int fastlz_compress(const void* input, int length, void* output) +{ + /* for short block, choose fastlz1 */ + if(length < 65536) + return fastlz1_compress(input, length, output); + + /* else... */ + return fastlz2_compress(input, length, output); +} + +int fastlz_decompress(const void* input, int length, void* output, int maxout) +{ + /* magic identifier for compression level */ + int level = ((*(const flzuint8*)input) >> 5) + 1; + + if(level == 1) + return fastlz1_decompress(input, length, output, maxout); + if(level == 2) + return fastlz2_decompress(input, length, output, maxout); + + /* unknown level, trigger error */ + return 0; +} + +int fastlz_compress_level(int level, const void* input, int length, void* output) +{ + if(level == 1) + return fastlz1_compress(input, length, output); + if(level == 2) + return fastlz2_compress(input, length, output); + + return 0; +} + +#else /* !defined(FASTLZ_COMPRESSOR) && !defined(FASTLZ_DECOMPRESSOR) */ + +static FASTLZ_INLINE int FASTLZ_COMPRESSOR(const void* input, int length, void* output) +{ + const flzuint8* ip = (const flzuint8*) input; + const flzuint8* ip_bound = ip + length - 2; + const flzuint8* ip_limit = ip + length - 12; + flzuint8* op = (flzuint8*) output; + + const flzuint8* htab[HASH_SIZE]; + const flzuint8** hslot; + flzuint32 hval; + + flzuint32 copy; + + /* sanity check */ + if(FASTLZ_UNEXPECT_CONDITIONAL(length < 4)) + { + if(length) + { + /* create literal copy only */ + *op++ = length-1; + ip_bound++; + while(ip <= ip_bound) + *op++ = *ip++; + return length+1; + } + else + return 0; + } + + /* initializes hash table */ + for (hslot = htab; hslot < htab + HASH_SIZE; hslot++) + *hslot = ip; + + /* we start with literal copy */ + copy = 2; + *op++ = MAX_COPY-1; + *op++ = *ip++; + *op++ = *ip++; + + /* main loop */ + while(FASTLZ_EXPECT_CONDITIONAL(ip < ip_limit)) + { + const flzuint8* ref; + flzuint32 distance; + + /* minimum match length */ + flzuint32 len = 3; + + /* comparison starting-point */ + const flzuint8* anchor = ip; + + /* check for a run */ +#if FASTLZ_LEVEL==2 + if(ip[0] == ip[-1] && FASTLZ_READU16(ip-1)==FASTLZ_READU16(ip+1)) + { + distance = 1; + ip += 3; + ref = anchor - 1 + 3; + goto match; + } +#endif + + /* find potential match */ + HASH_FUNCTION(hval,ip); + hslot = htab + hval; + ref = htab[hval]; + + /* calculate distance to the match */ + distance = anchor - ref; + + /* update hash table */ + *hslot = anchor; + + /* is this a match? check the first 3 bytes */ + if(distance==0 || +#if FASTLZ_LEVEL==1 + (distance >= MAX_DISTANCE) || +#else + (distance >= MAX_FARDISTANCE) || +#endif + *ref++ != *ip++ || *ref++!=*ip++ || *ref++!=*ip++) + goto literal; + +#if FASTLZ_LEVEL==2 + /* far, needs at least 5-byte match */ + if(distance >= MAX_DISTANCE) + { + if(*ip++ != *ref++ || *ip++!= *ref++) + goto literal; + len += 2; + } + + match: +#endif + + /* last matched byte */ + ip = anchor + len; + + /* distance is biased */ + distance--; + + if(!distance) + { + /* zero distance means a run */ + flzuint8 x = ip[-1]; + while(ip < ip_bound) + if(*ref++ != x) break; else ip++; + } + else + for(;;) + { + /* safe because the outer check against ip limit */ + if(*ref++ != *ip++) break; + if(*ref++ != *ip++) break; + if(*ref++ != *ip++) break; + if(*ref++ != *ip++) break; + if(*ref++ != *ip++) break; + if(*ref++ != *ip++) break; + if(*ref++ != *ip++) break; + if(*ref++ != *ip++) break; + while(ip < ip_bound) + if(*ref++ != *ip++) break; + break; + } + + /* if we have copied something, adjust the copy count */ + if(copy) + /* copy is biased, '0' means 1 byte copy */ + *(op-copy-1) = copy-1; + else + /* back, to overwrite the copy count */ + op--; + + /* reset literal counter */ + copy = 0; + + /* length is biased, '1' means a match of 3 bytes */ + ip -= 3; + len = ip - anchor; + + /* encode the match */ +#if FASTLZ_LEVEL==2 + if(distance < MAX_DISTANCE) + { + if(len < 7) + { + *op++ = (len << 5) + (distance >> 8); + *op++ = (distance & 255); + } + else + { + *op++ = (7 << 5) + (distance >> 8); + for(len-=7; len >= 255; len-= 255) + *op++ = 255; + *op++ = len; + *op++ = (distance & 255); + } + } + else + { + /* far away, but not yet in the another galaxy... */ + if(len < 7) + { + distance -= MAX_DISTANCE; + *op++ = (len << 5) + 31; + *op++ = 255; + *op++ = distance >> 8; + *op++ = distance & 255; + } + else + { + distance -= MAX_DISTANCE; + *op++ = (7 << 5) + 31; + for(len-=7; len >= 255; len-= 255) + *op++ = 255; + *op++ = len; + *op++ = 255; + *op++ = distance >> 8; + *op++ = distance & 255; + } + } +#else + + if(FASTLZ_UNEXPECT_CONDITIONAL(len > MAX_LEN-2)) + while(len > MAX_LEN-2) + { + *op++ = (7 << 5) + (distance >> 8); + *op++ = MAX_LEN - 2 - 7 -2; + *op++ = (distance & 255); + len -= MAX_LEN-2; + } + + if(len < 7) + { + *op++ = (len << 5) + (distance >> 8); + *op++ = (distance & 255); + } + else + { + *op++ = (7 << 5) + (distance >> 8); + *op++ = len - 7; + *op++ = (distance & 255); + } +#endif + + /* update the hash at match boundary */ + HASH_FUNCTION(hval,ip); + htab[hval] = ip++; + HASH_FUNCTION(hval,ip); + htab[hval] = ip++; + + /* assuming literal copy */ + *op++ = MAX_COPY-1; + + continue; + + literal: + *op++ = *anchor++; + ip = anchor; + copy++; + if(FASTLZ_UNEXPECT_CONDITIONAL(copy == MAX_COPY)) + { + copy = 0; + *op++ = MAX_COPY-1; + } + } + + /* left-over as literal copy */ + ip_bound++; + while(ip <= ip_bound) + { + *op++ = *ip++; + copy++; + if(copy == MAX_COPY) + { + copy = 0; + *op++ = MAX_COPY-1; + } + } + + /* if we have copied something, adjust the copy length */ + if(copy) + *(op-copy-1) = copy-1; + else + op--; + +#if FASTLZ_LEVEL==2 + /* marker for fastlz2 */ + *(flzuint8*)output |= (1 << 5); +#endif + + return op - (flzuint8*)output; +} + +static FASTLZ_INLINE int FASTLZ_DECOMPRESSOR(const void* input, int length, void* output, int maxout) +{ + const flzuint8* ip = (const flzuint8*) input; + const flzuint8* ip_limit = ip + length; + flzuint8* op = (flzuint8*) output; + flzuint8* op_limit = op + maxout; + flzuint32 ctrl = (*ip++) & 31; + int loop = 1; + + do + { + const flzuint8* ref = op; + flzuint32 len = ctrl >> 5; + flzuint32 ofs = (ctrl & 31) << 8; + + if(ctrl >= 32) + { +#if FASTLZ_LEVEL==2 + flzuint8 code; +#endif + len--; + ref -= ofs; + if (len == 7-1) +#if FASTLZ_LEVEL==1 + len += *ip++; + ref -= *ip++; +#else + do + { + code = *ip++; + len += code; + } while (code==255); + code = *ip++; + ref -= code; + + /* match from 16-bit distance */ + if(FASTLZ_UNEXPECT_CONDITIONAL(code==255)) + if(FASTLZ_EXPECT_CONDITIONAL(ofs==(31 << 8))) + { + ofs = (*ip++) << 8; + ofs += *ip++; + ref = op - ofs - MAX_DISTANCE; + } +#endif + +#ifdef FASTLZ_SAFE + if (FASTLZ_UNEXPECT_CONDITIONAL(op + len + 3 > op_limit)) + return 0; + + if (FASTLZ_UNEXPECT_CONDITIONAL(ref-1 < (flzuint8 *)output)) + return 0; +#endif + + if(FASTLZ_EXPECT_CONDITIONAL(ip < ip_limit)) + ctrl = *ip++; + else + loop = 0; + + if(ref == op) + { + /* optimize copy for a run */ + flzuint8 b = ref[-1]; + *op++ = b; + *op++ = b; + *op++ = b; + for(; len; --len) + *op++ = b; + } + else + { +#if !defined(FASTLZ_STRICT_ALIGN) + const flzuint16* p; + flzuint16* q; +#endif + /* copy from reference */ + ref--; + *op++ = *ref++; + *op++ = *ref++; + *op++ = *ref++; + +#if !defined(FASTLZ_STRICT_ALIGN) + /* copy a byte, so that now it's word aligned */ + if(len & 1) + { + *op++ = *ref++; + len--; + } + + /* copy 16-bit at once */ + q = (flzuint16*) op; + op += len; + p = (const flzuint16*) ref; + for(len>>=1; len > 4; len-=4) + { + *q++ = *p++; + *q++ = *p++; + *q++ = *p++; + *q++ = *p++; + } + for(; len; --len) + *q++ = *p++; +#else + for(; len; --len) + *op++ = *ref++; +#endif + } + } + else + { + ctrl++; +#ifdef FASTLZ_SAFE + if (FASTLZ_UNEXPECT_CONDITIONAL(op + ctrl > op_limit)) + return 0; + if (FASTLZ_UNEXPECT_CONDITIONAL(ip + ctrl > ip_limit)) + return 0; +#endif + + *op++ = *ip++; + for(--ctrl; ctrl; ctrl--) + *op++ = *ip++; + + loop = FASTLZ_EXPECT_CONDITIONAL(ip < ip_limit); + if(loop) + ctrl = *ip++; + } + } + while(FASTLZ_EXPECT_CONDITIONAL(loop)); + + return op - (flzuint8*)output; +} + +#endif /* !defined(FASTLZ_COMPRESSOR) && !defined(FASTLZ_DECOMPRESSOR) */ diff --git a/r5dev/thirdparty/fastlz/fastlz.h b/r5dev/thirdparty/fastlz/fastlz.h new file mode 100644 index 00000000..f87bc7be --- /dev/null +++ b/r5dev/thirdparty/fastlz/fastlz.h @@ -0,0 +1,100 @@ +/* + FastLZ - lightning-fast lossless compression library + + Copyright (C) 2007 Ariya Hidayat (ariya@kde.org) + Copyright (C) 2006 Ariya Hidayat (ariya@kde.org) + Copyright (C) 2005 Ariya Hidayat (ariya@kde.org) + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in + all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + THE SOFTWARE. +*/ + +#ifndef FASTLZ_H +#define FASTLZ_H + +#define FASTLZ_VERSION 0x000100 + +#define FASTLZ_VERSION_MAJOR 0 +#define FASTLZ_VERSION_MINOR 0 +#define FASTLZ_VERSION_REVISION 0 + +#define FASTLZ_VERSION_STRING "0.1.0" + +#if defined (__cplusplus) +extern "C" { +#endif + +/** + Compress a block of data in the input buffer and returns the size of + compressed block. The size of input buffer is specified by length. The + minimum input buffer size is 16. + + The output buffer must be at least 5% larger than the input buffer + and can not be smaller than 66 bytes. + + If the input is not compressible, the return value might be larger than + length (input buffer size). + + The input buffer and the output buffer can not overlap. +*/ + +int fastlz_compress(const void* input, int length, void* output); + +/** + Decompress a block of compressed data and returns the size of the + decompressed block. If error occurs, e.g. the compressed data is + corrupted or the output buffer is not large enough, then 0 (zero) + will be returned instead. + + The input buffer and the output buffer can not overlap. + + Decompression is memory safe and guaranteed not to write the output buffer + more than what is specified in maxout. + */ + +int fastlz_decompress(const void* input, int length, void* output, int maxout); + +/** + Compress a block of data in the input buffer and returns the size of + compressed block. The size of input buffer is specified by length. The + minimum input buffer size is 16. + + The output buffer must be at least 5% larger than the input buffer + and can not be smaller than 66 bytes. + + If the input is not compressible, the return value might be larger than + length (input buffer size). + + The input buffer and the output buffer can not overlap. + + Compression level can be specified in parameter level. At the moment, + only level 1 and level 2 are supported. + Level 1 is the fastest compression and generally useful for short data. + Level 2 is slightly slower but it gives better compression ratio. + + Note that the compressed data, regardless of the level, can always be + decompressed using the function fastlz_decompress above. +*/ + +int fastlz_compress_level(int level, const void* input, int length, void* output); + +#if defined (__cplusplus) +} +#endif + +#endif /* FASTLZ_H */ diff --git a/r5dev/thirdparty/sdl/src/stdlib/SDL_stdlib.c b/r5dev/thirdparty/sdl/src/stdlib/SDL_stdlib.c index 9d785aad..aec9efce 100644 --- a/r5dev/thirdparty/sdl/src/stdlib/SDL_stdlib.c +++ b/r5dev/thirdparty/sdl/src/stdlib/SDL_stdlib.c @@ -551,15 +551,15 @@ __declspec(selectany) int _fltused = 1; /* The optimizer on Visual Studio 2005 and later generates memcpy() and memset() calls */ #if _MSC_VER >= 1400 -extern void *memcpy(void* dst, const void* src, size_t len); +//extern void *memcpy(void* dst, const void* src, size_t len); #pragma intrinsic(memcpy) #pragma function(memcpy) -void * -memcpy(void *dst, const void *src, size_t len) -{ - return SDL_memcpy(dst, src, len); -} +//void * +//memcpy(void *dst, const void *src, size_t len) +//{ +// return SDL_memcpy(dst, src, len); +//} extern void *memset(void* dst, int c, size_t len); #pragma intrinsic(memset) diff --git a/r5dev/vproj/libsdl.vcxproj b/r5dev/vproj/libsdl.vcxproj index 53cea767..88284c98 100644 --- a/r5dev/vproj/libsdl.vcxproj +++ b/r5dev/vproj/libsdl.vcxproj @@ -530,6 +530,7 @@ + diff --git a/r5dev/vproj/libsdl.vcxproj.filters b/r5dev/vproj/libsdl.vcxproj.filters index f82d9462..9a19e88a 100644 --- a/r5dev/vproj/libsdl.vcxproj.filters +++ b/r5dev/vproj/libsdl.vcxproj.filters @@ -169,6 +169,12 @@ {4755f3a6-49ac-46d6-86be-21f5c21f2197} + + {b83a78cc-4e49-439f-b61a-90798f3012db} + + + {cf91bc2a-7739-4654-a08e-51cbcc3878de} + @@ -1314,6 +1320,9 @@ power\windows + + main\windows + diff --git a/r5dev/vproj/naveditor.vcxproj b/r5dev/vproj/naveditor.vcxproj new file mode 100644 index 00000000..9940d164 --- /dev/null +++ b/r5dev/vproj/naveditor.vcxproj @@ -0,0 +1,152 @@ + + + + + Debug + x64 + + + Release + x64 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 16.0 + Win32Proj + {1942083A-03D9-4D76-B644-A3FA2A118A35} + naveditor + 10.0 + + + + Application + true + v143 + Unicode + Static + + + Application + false + v143 + true + Unicode + Static + + + + + + + + + + + + + + + true + $(SolutionDir)r5dev\;$(SolutionDir)r5dev\thirdparty\recast\;$(IncludePath) + $(SolutionDir)build\$(ProjectName)\$(Configuration)\ + $(SolutionDir)bin\$(Configuration)\ + $(ProjectName) + + + false + $(SolutionDir)r5dev\;$(SolutionDir)r5dev\thirdparty\recast\;$(IncludePath) + $(SolutionDir)build\$(ProjectName)\$(Configuration)\ + $(SolutionDir)bin\$(Configuration)\ + $(ProjectName) + + + + Level3 + true + _DEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + /D _CRT_SECURE_NO_WARNINGS /D WIN32 %(AdditionalOptions) + stdcpp17 + + + Console + true + $(SolutionDir)lib\$(Configuration)\ + librecast_x64.lib;libdtdetour_x64.lib;libdetourcrowd_x64.lib;libdetourtilecache_x64.lib;libdtdebugutils_x64.lib;libsdl2_x64.lib;OpenGL32.lib;Glu32.lib;Gdi32.lib;User32.lib;Shell32.lib;Comdlg32.lib;Kernel32.lib;Winmm.lib;Setupapi.lib;Advapi32.lib;Version.lib;Ole32.lib;Oleaut32.lib;Imm32.lib;%(AdditionalDependencies) + + + + + Level3 + true + true + true + NDEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + /D _CRT_SECURE_NO_WARNINGS /D WIN32 %(AdditionalOptions) + stdcpp17 + + + Console + true + true + true + $(SolutionDir)lib\$(Configuration)\ + librecast_x64.lib;libdtdetour_x64.lib;libdetourcrowd_x64.lib;libdetourtilecache_x64.lib;libdtdebugutils_x64.lib;libsdl2_x64.lib;OpenGL32.lib;Glu32.lib;Gdi32.lib;User32.lib;Shell32.lib;Comdlg32.lib;Kernel32.lib;Winmm.lib;Setupapi.lib;Advapi32.lib;Version.lib;Ole32.lib;Oleaut32.lib;Imm32.lib;%(AdditionalDependencies) + + + + + + \ No newline at end of file diff --git a/r5dev/vproj/naveditor.vcxproj.filters b/r5dev/vproj/naveditor.vcxproj.filters new file mode 100644 index 00000000..fd6e9843 --- /dev/null +++ b/r5dev/vproj/naveditor.vcxproj.filters @@ -0,0 +1,110 @@ + + + + + {9ab87056-e082-4b08-96fc-91d0b5e14f3b} + + + {1ac9f8e6-8a00-4875-aa9e-0b439faaff62} + + + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Include + + + Conrtib + + + + + + + + + + + + + + + + + + + + + + + + + + + + Conrtib + + + \ No newline at end of file diff --git a/r5sdk.sln b/r5sdk.sln index b2f74d58..710f1086 100644 --- a/r5sdk.sln +++ b/r5sdk.sln @@ -54,7 +54,18 @@ Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "librecast", "r5dev\vproj\li EndProject Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "libdebugutils", "r5dev\vproj\libdebugutils.vcxproj", "{0E701104-CD9A-45C0-8E32-3284DBDEAF5E}" EndProject -Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "SDL2", "r5dev\vproj\libsdl.vcxproj", "{81CE8DAF-EBB2-4761-8E45-B71ABCCA8C68}" +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "libsdl2", "r5dev\vproj\libsdl.vcxproj", "{81CE8DAF-EBB2-4761-8E45-B71ABCCA8C68}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "naveditor", "r5dev\vproj\naveditor.vcxproj", "{1942083A-03D9-4D76-B644-A3FA2A118A35}" + ProjectSection(ProjectDependencies) = postProject + {0E701104-CD9A-45C0-8E32-3284DBDEAF5E} = {0E701104-CD9A-45C0-8E32-3284DBDEAF5E} + {DC456E49-7FC6-4BB9-B8A1-C879A37F2A1C} = {DC456E49-7FC6-4BB9-B8A1-C879A37F2A1C} + {31FB1B73-F4C5-414B-A27D-AB0DC194BC61} = {31FB1B73-F4C5-414B-A27D-AB0DC194BC61} + {DC72AD9E-F12F-4802-8BB8-F17A16BFCAEB} = {DC72AD9E-F12F-4802-8BB8-F17A16BFCAEB} + {6A8085A2-4DD0-4726-A667-ED873020AAB7} = {6A8085A2-4DD0-4726-A667-ED873020AAB7} + {81CE8DAF-EBB2-4761-8E45-B71ABCCA8C68} = {81CE8DAF-EBB2-4761-8E45-B71ABCCA8C68} + {6DC4E2AF-1740-480B-A9E4-BA766BC6B58D} = {6DC4E2AF-1740-480B-A9E4-BA766BC6B58D} + EndProjectSection EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution @@ -176,6 +187,14 @@ Global {81CE8DAF-EBB2-4761-8E45-B71ABCCA8C68}.Release|x64.Build.0 = Release|x64 {81CE8DAF-EBB2-4761-8E45-B71ABCCA8C68}.Release|x86.ActiveCfg = Release|Win32 {81CE8DAF-EBB2-4761-8E45-B71ABCCA8C68}.Release|x86.Build.0 = Release|Win32 + {1942083A-03D9-4D76-B644-A3FA2A118A35}.Debug|x64.ActiveCfg = Debug|x64 + {1942083A-03D9-4D76-B644-A3FA2A118A35}.Debug|x64.Build.0 = Debug|x64 + {1942083A-03D9-4D76-B644-A3FA2A118A35}.Debug|x86.ActiveCfg = Debug|x64 + {1942083A-03D9-4D76-B644-A3FA2A118A35}.Debug|x86.Build.0 = Debug|x64 + {1942083A-03D9-4D76-B644-A3FA2A118A35}.Release|x64.ActiveCfg = Release|x64 + {1942083A-03D9-4D76-B644-A3FA2A118A35}.Release|x64.Build.0 = Release|x64 + {1942083A-03D9-4D76-B644-A3FA2A118A35}.Release|x86.ActiveCfg = Release|x64 + {1942083A-03D9-4D76-B644-A3FA2A118A35}.Release|x86.Build.0 = Release|x64 EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE