From 88e680e8e7247b679e3c1e23d8fcf999eb615bf0 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 21 Mar 2013 23:52:44 +0100 Subject: [PATCH 01/66] Add profiler files --- src/engine/Profiler.cpp | 1 + src/engine/Profiler.h | 87 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 88 insertions(+) create mode 100644 src/engine/Profiler.cpp create mode 100644 src/engine/Profiler.h diff --git a/src/engine/Profiler.cpp b/src/engine/Profiler.cpp new file mode 100644 index 00000000..8b137891 --- /dev/null +++ b/src/engine/Profiler.cpp @@ -0,0 +1 @@ + diff --git a/src/engine/Profiler.h b/src/engine/Profiler.h new file mode 100644 index 00000000..51bac2c1 --- /dev/null +++ b/src/engine/Profiler.h @@ -0,0 +1,87 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 PROFILER_H +#define PROFILER_H + +// Libraries + +// Class ProfileSample +/** + * This class is used to represent a profile sample. It is constructed at the + * beginning of a code block we want to profile and destructed at the end of the + * scope to profile. + */ +class ProfileSample { + + public : + + /// Constructor + ProfileSample(const char* name) { + + // Ask the profiler to start profiling a block of code + Profiler::startProfilingBlock(); + } + + /// Destructor + ~ProfileSample() { + + // Tell the profiler to stop profiling a block of code + Profiler::stopProfilingBlock(); + } +}; + +/// Use this macro to start profile a block of code +#define PROFILE(name) ProfileSample profileSample(name) + +// Class ProfileNode +/** + * Node of the profiler tree + */ +class ProfileNode { + +}; + +// Class Profiler +/** + * This is the main class of the profiler + */ +class Profiler { + + public : + + /// Method called when we want to start profiling a block of code. + static void startProfilingBlock() { + + } + + /// Method called at the end of the scope where the startProfilingBlock() method + /// has been called. + static void stopProfilingBlock() { + + } +}; + +#endif From aa9bd2098da67690a8d8e32cb9ff203123af79a0 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sun, 24 Mar 2013 19:45:48 +0100 Subject: [PATCH 02/66] Continue to work on the profiler --- CMakeLists.txt | 5 ++ src/engine/Profiler.cpp | 140 +++++++++++++++++++++++++++++ src/engine/Profiler.h | 192 +++++++++++++++++++++++++++++++++++----- src/engine/Timer.cpp | 17 ++++ src/engine/Timer.h | 44 +++------ 5 files changed, 346 insertions(+), 52 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index f874bc4d..68307462 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -10,10 +10,15 @@ SET(LIBRARY_OUTPUT_PATH lib/) # Options OPTION(COMPILE_EXAMPLES "Select this if you want to build the examples" OFF) OPTION(COMPILE_TESTS "Select this if you want to build the tests" OFF) +OPTION(PROFILING_ENABLED "Select this if you want to compile with enabled profiling" OFF) # Headers INCLUDE_DIRECTORIES(src) +IF (PROFILING_ENABLED) + ADD_DEFINITIONS(-DIS_PROFILING_ACTIVE) +ENDIF (PROFILING_ENABLED) + # Library configuration file ( GLOB_RECURSE diff --git a/src/engine/Profiler.cpp b/src/engine/Profiler.cpp index 8b137891..0c915469 100644 --- a/src/engine/Profiler.cpp +++ b/src/engine/Profiler.cpp @@ -1 +1,141 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ +#ifdef IS_PROFILING_ACTIVE + +// Libraries +#include "Profiler.h" + +using namespace reactphysics3d; + +// Initialization of static variables +ProfileNode Profiler::mRootNode("Root", NULL); +ProfileNode* Profiler::mCurrentNode = &Profiler::mRootNode; +uint Profiler::mFrameCounter = 0; + +// Constructor +ProfileNode::ProfileNode(const char* name, ProfileNode* parentNode) + :mName(name), mNbTotalCalls(0), mStartingTime(0), mTotalTime(0), + mRecursionCounter(0), mParentNode(parentNode), mChildNode(NULL), + mSiblingNode(NULL) { + +} + +// Destructor +ProfileNode::~ProfileNode() { + + delete mChildNode; + delete mSiblingNode; +} + +// Return a pointer to a sub node with a given name +ProfileNode* ProfileNode::findSubNode(const char* name) { + + // Try to find the node among the child nodes + ProfileNode* child = mChildNode; + while (child != NULL) { + if (child->mName == name) { + return child; + } + child = child->mSiblingNode; + } + + // The nose has not been found. Therefore, we create it + // and add it to the profiler tree + ProfileNode* newNode = new ProfileNode(name, this); + newNode->mSiblingNode = child; + child = newNode; + + return newNode; +} + +// Called when we enter the block of code corresponding to this profile node +void ProfileNode::enterBlockOfCode() { + mNbTotalCalls++; + + // If the current code is not called recursively + if (mRecursionCounter == 0) { + + // Get the current system time to initialize the starting time of + // the profiling of the current block of code + mStartingTime = Timer::getCurrentSystemTime(); + } + + mRecursionCounter++; +} + +// Called when we exit the block of code corresponding to this profile node +bool ProfileNode::exitBlockOfCode() { + mRecursionCounter--; + + if (mRecursionCounter == 0 && mNbTotalCalls != 0) { + + // Get the current system time + long double currentTime = Timer::getCurrentSystemTime(); + + // Increase the total elasped time in the current block of code + mTotalTime += currentTime - mStartingTime; + } + + // Return true if the current code is not recursing + return (mRecursionCounter == 0); +} + +// Method called when we want to start profiling a block of code. +void Profiler::startProfilingBlock(const char* name) { + + // Look for the node in the tree that corresponds to the block of + // code to profile + if (name != mCurrentNode->getName()) { + mCurrentNode = mCurrentNode->findSubNode(name); + } + + // Start profile the node + mCurrentNode->enterBlockOfCode(); +} + +// Method called at the end of the scope where the +// startProfilingBlock() method has been called. +void Profiler::stopProfilingBlock() { + + // Go to the parent node unless if the current block + // of code is recursing + if (mCurrentNode->exitBlockOfCode()) { + mCurrentNode = mCurrentNode->getParentNode(); + } + +} + +// Return an iterator over the profiler tree starting at the root +ProfileNodeIterator* Profiler::getIterator() { + return new ProfileNodeIterator(&mRootNode); +} + +// Print the report of the profiler in a given output stream +void printReport(std::ostream& outputStream) { + +} + +#endif diff --git a/src/engine/Profiler.h b/src/engine/Profiler.h index 51bac2c1..1768b01e 100644 --- a/src/engine/Profiler.h +++ b/src/engine/Profiler.h @@ -26,7 +26,146 @@ #ifndef PROFILER_H #define PROFILER_H +#ifdef IS_PROFILING_ACTIVE + // Libraries +#include "../configuration.h" +#include "Timer.h" + +/// ReactPhysics3D namespace +namespace reactphysics3d { + +// Class ProfileNode +/** + * It represents a profile sample in the profiler tree. + */ +class ProfileNode { + + private : + + /// Name of the node + const char* mName; + + /// Total number of calls of this node + uint mNbTotalCalls; + + /// Starting time of the sampling of corresponding block of code + long double mStartingTime; + + /// Total time spent in the block of code + long double mTotalTime; + + /// Recursion counter + int mRecursionCounter; + + /// Pointer to the parent node + ProfileNode* mParentNode; + + /// Pointer to a child node + ProfileNode* mChildNode; + + /// Pointer to a sibling node + ProfileNode* mSiblingNode; + + public : + + /// Constructor + ProfileNode(const char* name, ProfileNode* parentNode); + + /// Destructor + ~ProfileNode(); + + /// Return a pointer to a sub node + ProfileNode* findSubNode(const char* name); + + /// Return a pointer to the parent node + ProfileNode* getParentNode(); + + /// Return a pointer to a sibling node + ProfileNode* getSiblingNode(); + + /// Return a pointer to a child node + ProfileNode* getChildNode(); + + /// Return the name of the node + const char* getName(); + + /// Return the total number of call of the corresponding block of code + uint getNbTotalCalls() const; + + /// Return the total time spent in the block of code + long double getTotalTime() const; + + /// Called when we enter the block of code corresponding to this profile node + void enterBlockOfCode(); + + /// Called when we exit the block of code corresponding to this profile node + bool exitBlockOfCode(); +}; + +// Class ProfileNodeIterator +/** + * This class allow us to iterator over the profiler tree. + */ +class ProfileNodeIterator { + + private : + + /// Current parent node + ProfileNode* mCurrentParent; + + /// Current child node + ProfileNode* mCurrentChild; + + public : + + /// Constructor + ProfileNodeIterator(ProfileNode* startingNode); + + /// Go to the first node + void first(); + + /// Go to the next node + void next(); + + /// Enter a given child node + void enterChild(); + +}; + +// Class Profiler +/** + * This is the main class of the profiler. This profiler is based on "Real-Time Hierarchical + * Profiling" article from "Game Programming Gems 3" by Greg Hjelstrom and Byon Garrabrant. + */ +class Profiler { + + private : + + /// Root node of the profiler tree + static ProfileNode mRootNode; + + /// Current node in the current execution + static ProfileNode* mCurrentNode; + + /// Frame counter + static uint mFrameCounter; + + public : + + /// Method called when we want to start profiling a block of code. + static void startProfilingBlock(const char *name); + + /// Method called at the end of the scope where the + /// startProfilingBlock() method has been called. + static void stopProfilingBlock(); + + /// Return an iterator over the profiler tree starting at the root + static ProfileNodeIterator* getIterator(); + + /// Print the report of the profiler in a given output stream + static void printReport(std::ostream& outputStream); +}; // Class ProfileSample /** @@ -42,7 +181,7 @@ class ProfileSample { ProfileSample(const char* name) { // Ask the profiler to start profiling a block of code - Profiler::startProfilingBlock(); + Profiler::startProfilingBlock(name); } /// Destructor @@ -56,32 +195,43 @@ class ProfileSample { /// Use this macro to start profile a block of code #define PROFILE(name) ProfileSample profileSample(name) -// Class ProfileNode -/** - * Node of the profiler tree - */ -class ProfileNode { +/// Return a pointer to the parent node +ProfileNode* ProfileNode::getParentNode() { + return mParentNode; +} -}; +/// Return a pointer to a sibling node +ProfileNode* ProfileNode::getSiblingNode() { + return mSiblingNode; +} -// Class Profiler -/** - * This is the main class of the profiler - */ -class Profiler { +/// Return a pointer to a child node +ProfileNode* ProfileNode::getChildNode() { + return mChildNode; +} - public : +/// Return the name of the node +const char* ProfileNode::getName() { + return mName; +} - /// Method called when we want to start profiling a block of code. - static void startProfilingBlock() { +/// Return the total number of call of the corresponding block of code +uint ProfileNode::getNbTotalCalls() const { + return mNbTotalCalls; +} - } +/// Return the total time spent in the block of code +long double ProfileNode::getTotalTime() const { + return mTotalTime; +} - /// Method called at the end of the scope where the startProfilingBlock() method - /// has been called. - static void stopProfilingBlock() { +} - } -}; +#else // In profile is not active + +// Empty macro in case profiling is not active +#define PROFILE(name) + +#endif #endif diff --git a/src/engine/Timer.cpp b/src/engine/Timer.cpp index 007a4240..74c08086 100644 --- a/src/engine/Timer.cpp +++ b/src/engine/Timer.cpp @@ -39,6 +39,23 @@ Timer::~Timer() { } +// Return the current time of the system +long double Timer::getCurrentSystemTime() { + + #if defined(WINDOWS_OS) + LARGE_INTEGER ticksPerSecond; + LARGE_INTEGER ticks; + QueryPerformanceFrequency(&ticksPerSecond); + QueryPerformanceCounter(&ticks); + return (long double(ticks.QuadPart) / long double(ticksPerSecond.QuadPart)); + #else + // Initialize the lastUpdateTime with the current time in seconds + timeval timeValue; + gettimeofday(&timeValue, NULL); + return (timeValue.tv_sec + (timeValue.tv_usec / 1000000.0)); + #endif +} + diff --git a/src/engine/Timer.h b/src/engine/Timer.h index d165dbc6..7c70b52c 100644 --- a/src/engine/Timer.h +++ b/src/engine/Timer.h @@ -74,6 +74,7 @@ class Timer { /// True if the timer is running bool mIsRunning; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -98,8 +99,8 @@ class Timer { /// Set the timestep of the physics engine void setTimeStep(double timeStep); - /// Return the current time - long double getTime() const; + /// Return the current time of the physics engine + long double getPhysicsTime() const; /// Start the timer void start(); @@ -121,6 +122,9 @@ class Timer { /// Compute the interpolation factor decimal computeInterpolationFactor(); + + /// Return the current time of the system + static long double getCurrentSystemTime(); }; // Return the timestep of the physics engine @@ -135,7 +139,7 @@ inline void Timer::setTimeStep(double timeStep) { } // Return the current time -inline long double Timer::getTime() const { +inline long double Timer::getPhysicsTime() const { return mTime; } @@ -147,19 +151,9 @@ inline bool Timer::getIsRunning() const { // Start the timer inline void Timer::start() { if (!mIsRunning) { - -#if defined(WINDOWS_OS) - LARGE_INTEGER ticksPerSecond; - LARGE_INTEGER ticks; - QueryPerformanceFrequency(&ticksPerSecond); - QueryPerformanceCounter(&ticks); - mLastUpdateTime = double(ticks.QuadPart) / double(ticksPerSecond.QuadPart); -#else - // Initialize the lastUpdateTime with the current time in seconds - timeval timeValue; - gettimeofday(&timeValue, NULL); - mLastUpdateTime = timeValue.tv_sec + (timeValue.tv_usec / 1000000.0); -#endif + + // Get the current system time + mLastUpdateTime = getCurrentSystemTime(); mAccumulator = 0.0; mIsRunning = true; @@ -168,7 +162,6 @@ inline void Timer::start() { // Stop the timer inline void Timer::stop() { - std::cout << "Timer stop" << std::endl; mIsRunning = false; } @@ -195,20 +188,9 @@ inline decimal Timer::computeInterpolationFactor() { // Compute the time since the last update() call and add it to the accumulator inline void Timer::update() { - long double currentTime; - -#if defined(WINDOWS_OS) - LARGE_INTEGER ticksPerSecond; - LARGE_INTEGER ticks; - QueryPerformanceFrequency(&ticksPerSecond); - QueryPerformanceCounter(&ticks); - currentTime = double(ticks.QuadPart) / double(ticksPerSecond.QuadPart); -#else - // Compute the current time is seconds - timeval timeValue; - gettimeofday(&timeValue, NULL); - currentTime = timeValue.tv_sec + (timeValue.tv_usec / 1000000.0); -#endif + + // Get the current system time + long double currentTime = getCurrentSystemTime(); // Compute the delta display time between two display frames mDeltaTime = currentTime - mLastUpdateTime; From f479c5edf3c5dc3adac0c8e4961aabbbd95492bc Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 26 Mar 2013 21:37:55 +0100 Subject: [PATCH 03/66] Finish the implementation of the profiler --- src/collision/CollisionDetection.cpp | 7 + src/engine/CollisionWorld.h | 1 + src/engine/ContactSolver.cpp | 3 + src/engine/DynamicsWorld.cpp | 24 ++++ src/engine/Profiler.cpp | 136 +++++++++++++++++-- src/engine/Profiler.h | 187 ++++++++++++++++++++++++--- src/engine/Timer.cpp | 2 +- src/engine/Timer.h | 3 +- 8 files changed, 333 insertions(+), 30 deletions(-) diff --git a/src/collision/CollisionDetection.cpp b/src/collision/CollisionDetection.cpp index 9d0254d2..44e24810 100644 --- a/src/collision/CollisionDetection.cpp +++ b/src/collision/CollisionDetection.cpp @@ -61,6 +61,8 @@ CollisionDetection::~CollisionDetection() { // Compute the collision detection void CollisionDetection::computeCollisionDetection() { + + PROFILE("CollisionDetection::computeCollisionDetection()"); // Compute the broad-phase collision detection computeBroadPhase(); @@ -72,6 +74,8 @@ void CollisionDetection::computeCollisionDetection() { // Compute the broad-phase collision detection void CollisionDetection::computeBroadPhase() { + PROFILE("CollisionDetection::computeBroadPhase()"); + // Notify the broad-phase algorithm about the bodies that have moved since last frame for (set::iterator it = mWorld->getBodiesBeginIterator(); it != mWorld->getBodiesEndIterator(); it++) { @@ -87,6 +91,9 @@ void CollisionDetection::computeBroadPhase() { // Compute the narrow-phase collision detection void CollisionDetection::computeNarrowPhase() { + + PROFILE("CollisionDetection::computeNarrowPhase()"); + map::iterator it; // For each possible collision pair of bodies diff --git a/src/engine/CollisionWorld.h b/src/engine/CollisionWorld.h index dde08972..4a929a3a 100644 --- a/src/engine/CollisionWorld.h +++ b/src/engine/CollisionWorld.h @@ -31,6 +31,7 @@ #include #include #include "../mathematics/mathematics.h" +#include "Profiler.h" #include "../body/CollisionBody.h" #include "OverlappingPair.h" #include "../collision/CollisionDetection.h" diff --git a/src/engine/ContactSolver.cpp b/src/engine/ContactSolver.cpp index a8004d12..e90ed978 100644 --- a/src/engine/ContactSolver.cpp +++ b/src/engine/ContactSolver.cpp @@ -27,6 +27,7 @@ #include "ContactSolver.h" #include "DynamicsWorld.h" #include "../body/RigidBody.h" +#include "Profiler.h" #include using namespace reactphysics3d; @@ -748,6 +749,8 @@ void ContactSolver::solveContactConstraints() { // Solve the constraints void ContactSolver::solve(decimal timeStep) { + PROFILE("ContactSolver::solve()"); + // Set the current time step mTimeStep = timeStep; diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 884ac74c..48940f82 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -52,11 +52,28 @@ DynamicsWorld::~DynamicsWorld() { // Free the allocated memory for the constrained velocities cleanupConstrainedVelocitiesArray(); + +#ifdef IS_PROFILING_ACTIVE + + // Print the profiling report + Profiler::printReport(std::cout); + + // Destroy the profiler (release the allocated memory) + Profiler::destroy(); +#endif + } // Update the physics simulation void DynamicsWorld::update() { +#ifdef IS_PROFILING_ACTIVE + // Increment the frame counter of the profiler + Profiler::incrementFrameCounter(); +#endif + + PROFILE("DynamicsWorld::update()"); + assert(mTimer.getIsRunning()); // Compute the time since the last update() call and update the timer @@ -106,6 +123,9 @@ void DynamicsWorld::update() { // Update the position and orientation of the rigid bodies void DynamicsWorld::updateRigidBodiesPositionAndOrientation() { + + PROFILE("DynamicsWorld::updateRigidBodiesPositionAndOrientation()"); + decimal dt = static_cast(mTimer.getTimeStep()); // For each rigid body of the world @@ -159,6 +179,8 @@ void DynamicsWorld::updateRigidBodiesPositionAndOrientation() { // Compute and set the interpolation factor to all bodies void DynamicsWorld::setInterpolationFactorToAllBodies() { + + PROFILE("DynamicsWorld::setInterpolationFactorToAllBodies()"); // Compute the interpolation factor decimal factor = mTimer.computeInterpolationFactor(); @@ -215,6 +237,8 @@ void DynamicsWorld::cleanupConstrainedVelocitiesArray() { // Apply the gravity force to all bodies of the physics world void DynamicsWorld::applyGravity() { + PROFILE("DynamicsWorld::applyGravity()"); + // For each body of the physics world set::iterator it; for (it = getRigidBodiesBeginIterator(); it != getRigidBodiesEndIterator(); ++it) { diff --git a/src/engine/Profiler.cpp b/src/engine/Profiler.cpp index 0c915469..2759976a 100644 --- a/src/engine/Profiler.cpp +++ b/src/engine/Profiler.cpp @@ -33,6 +33,7 @@ using namespace reactphysics3d; // Initialization of static variables ProfileNode Profiler::mRootNode("Root", NULL); ProfileNode* Profiler::mCurrentNode = &Profiler::mRootNode; +long double Profiler::mProfilingStartTime = Timer::getCurrentSystemTime() * 1000.0; uint Profiler::mFrameCounter = 0; // Constructor @@ -40,7 +41,7 @@ ProfileNode::ProfileNode(const char* name, ProfileNode* parentNode) :mName(name), mNbTotalCalls(0), mStartingTime(0), mTotalTime(0), mRecursionCounter(0), mParentNode(parentNode), mChildNode(NULL), mSiblingNode(NULL) { - + reset(); } // Destructor @@ -65,8 +66,8 @@ ProfileNode* ProfileNode::findSubNode(const char* name) { // The nose has not been found. Therefore, we create it // and add it to the profiler tree ProfileNode* newNode = new ProfileNode(name, this); - newNode->mSiblingNode = child; - child = newNode; + newNode->mSiblingNode = mChildNode; + mChildNode = newNode; return newNode; } @@ -80,7 +81,7 @@ void ProfileNode::enterBlockOfCode() { // Get the current system time to initialize the starting time of // the profiling of the current block of code - mStartingTime = Timer::getCurrentSystemTime(); + mStartingTime = Timer::getCurrentSystemTime() * 1000.0; } mRecursionCounter++; @@ -93,7 +94,7 @@ bool ProfileNode::exitBlockOfCode() { if (mRecursionCounter == 0 && mNbTotalCalls != 0) { // Get the current system time - long double currentTime = Timer::getCurrentSystemTime(); + long double currentTime = Timer::getCurrentSystemTime() * 1000.0; // Increase the total elasped time in the current block of code mTotalTime += currentTime - mStartingTime; @@ -103,6 +104,59 @@ bool ProfileNode::exitBlockOfCode() { return (mRecursionCounter == 0); } +// Reset the profiling of the node +void ProfileNode::reset() { + mNbTotalCalls = 0; + mTotalTime = 0.0; + + // Reset the child node + if (mChildNode != NULL) { + mChildNode->reset(); + } + + // Reset the sibling node + if (mSiblingNode != NULL) { + mSiblingNode->reset(); + } +} + +// Destroy the node +void ProfileNode::destroy() { + delete mChildNode; + mChildNode = NULL; + delete mSiblingNode; + mSiblingNode = NULL; +} + +// Constructor +ProfileNodeIterator::ProfileNodeIterator(ProfileNode* startingNode) + :mCurrentParentNode(startingNode), + mCurrentChildNode(mCurrentParentNode->getChildNode()){ + +} + +// Enter a given child node +void ProfileNodeIterator::enterChild(int index) { + mCurrentChildNode = mCurrentParentNode->getChildNode(); + while ((mCurrentChildNode != NULL) && (index != 0)) { + index--; + mCurrentChildNode = mCurrentChildNode->getSiblingNode(); + } + + if (mCurrentChildNode != NULL) { + mCurrentParentNode = mCurrentChildNode; + mCurrentChildNode = mCurrentParentNode->getChildNode(); + } +} + +// Enter a given parent node +void ProfileNodeIterator::enterParent() { + if (mCurrentParentNode->getParentNode() != NULL) { + mCurrentParentNode = mCurrentParentNode->getParentNode(); + } + mCurrentChildNode = mCurrentParentNode->getChildNode(); +} + // Method called when we want to start profiling a block of code. void Profiler::startProfilingBlock(const char* name) { @@ -125,17 +179,79 @@ void Profiler::stopProfilingBlock() { if (mCurrentNode->exitBlockOfCode()) { mCurrentNode = mCurrentNode->getParentNode(); } - } -// Return an iterator over the profiler tree starting at the root -ProfileNodeIterator* Profiler::getIterator() { - return new ProfileNodeIterator(&mRootNode); +// Reset the timing data of the profiler (but not the profiler tree structure) +void Profiler::reset() { + mRootNode.reset(); + mRootNode.enterBlockOfCode(); + mFrameCounter = 0; + mProfilingStartTime = Timer::getCurrentSystemTime() * 1000.0; } // Print the report of the profiler in a given output stream -void printReport(std::ostream& outputStream) { +void Profiler::printReport(std::ostream& outputStream) { + ProfileNodeIterator* iterator = Profiler::getIterator(); + // Recursively print the report of each node of the profiler tree + printRecursiveNodeReport(iterator, 0, outputStream); + + // Destroy the iterator + destroyIterator(iterator); +} + +// Recursively print the report of a given node of the profiler tree +void Profiler::printRecursiveNodeReport(ProfileNodeIterator* iterator, + int spacing, + std::ostream& outputStream) { + iterator->first(); + + // If we are at the end of a branch in the profiler tree + if (iterator->isEnd()) { + return; + } + + long double parentTime = iterator->isRoot() ? getElapsedTimeSinceStart() : + iterator->getCurrentParentTotalTime(); + long double accumulatedTime = 0.0; + uint nbFrames = Profiler::getNbFrames(); + for (int i=0; igetCurrentParentName() << + " (total running time : " << parentTime << " ms) ---" << std::endl; + long double totalTime = 0.0; + + // Recurse over the children of the current node + int nbChildren = 0; + for (int i=0; !iterator->isEnd(); i++, iterator->next()) { + nbChildren++; + long double currentTotalTime = iterator->getCurrentTotalTime(); + accumulatedTime += currentTotalTime; + long double fraction = parentTime > std::numeric_limits::epsilon() ? + (currentTotalTime / parentTime) * 100.0 : 0.0; + for (int j=0; jgetCurrentName() << " : " << + fraction << " % | " << (currentTotalTime / long double(nbFrames)) << + " ms/frame (" << iterator->getCurrentNbTotalCalls() << " calls)" << + std::endl; + totalTime += currentTotalTime; + } + + if (parentTime < accumulatedTime) { + outputStream << "Something is wrong !" << std::endl; + } + for (int i=0; i std::numeric_limits::epsilon() ? + ((parentTime - accumulatedTime) / parentTime) * 100.0 : 0.0; + long double difference = parentTime - accumulatedTime; + outputStream << "| Unaccounted : " << difference << " ms (" << percentage << " %)" << std::endl; + + for (int i=0; ienterChild(i); + printRecursiveNodeReport(iterator, spacing + 3, outputStream); + iterator->enterParent(); + } } #endif diff --git a/src/engine/Profiler.h b/src/engine/Profiler.h index 1768b01e..286e909b 100644 --- a/src/engine/Profiler.h +++ b/src/engine/Profiler.h @@ -43,6 +43,8 @@ class ProfileNode { private : + // -------------------- Attributes -------------------- // + /// Name of the node const char* mName; @@ -69,6 +71,8 @@ class ProfileNode { public : + // -------------------- Methods -------------------- // + /// Constructor ProfileNode(const char* name, ProfileNode* parentNode); @@ -101,24 +105,34 @@ class ProfileNode { /// Called when we exit the block of code corresponding to this profile node bool exitBlockOfCode(); + + /// Reset the profiling of the node + void reset(); + + /// Destroy the node + void destroy(); }; // Class ProfileNodeIterator /** - * This class allow us to iterator over the profiler tree. + * This class allows us to iterator over the profiler tree. */ class ProfileNodeIterator { private : + // -------------------- Attributes -------------------- // + /// Current parent node - ProfileNode* mCurrentParent; + ProfileNode* mCurrentParentNode; /// Current child node - ProfileNode* mCurrentChild; + ProfileNode* mCurrentChildNode; public : + // -------------------- Methods -------------------- // + /// Constructor ProfileNodeIterator(ProfileNode* startingNode); @@ -129,8 +143,34 @@ class ProfileNodeIterator { void next(); /// Enter a given child node - void enterChild(); + void enterChild(int index); + /// Enter a given parent node + void enterParent(); + + /// Return true if we are at the root of the profiler tree + bool isRoot(); + + /// Return true if we are at the end of a branch of the profiler tree + bool isEnd(); + + /// Return the name of the current node + const char* getCurrentName(); + + /// Return the total time of the current node + long double getCurrentTotalTime(); + + /// Return the total number of calls of the current node + uint getCurrentNbTotalCalls(); + + /// Return the name of the current parent node + const char* getCurrentParentName(); + + /// Return the total time of the current parent node + long double getCurrentParentTotalTime(); + + /// Return the total number of calls of the current parent node + uint getCurrentParentNbTotalCalls(); }; // Class Profiler @@ -142,6 +182,8 @@ class Profiler { private : + // -------------------- Attributes -------------------- // + /// Root node of the profiler tree static ProfileNode mRootNode; @@ -151,8 +193,18 @@ class Profiler { /// Frame counter static uint mFrameCounter; + /// Starting profiling time + static long double mProfilingStartTime; + + /// Recursively print the report of a given node of the profiler tree + static void printRecursiveNodeReport(ProfileNodeIterator* iterator, + int spacing, + std::ostream& outputStream); + public : + // -------------------- Methods -------------------- // + /// Method called when we want to start profiling a block of code. static void startProfilingBlock(const char *name); @@ -160,11 +212,29 @@ class Profiler { /// startProfilingBlock() method has been called. static void stopProfilingBlock(); + /// Reset the timing data of the profiler (but not the profiler tree structure) + static void reset(); + + /// Return the number of frames + static uint getNbFrames(); + + /// Return the elasped time since the start/reset of the profiling + static long double getElapsedTimeSinceStart(); + + /// Increment the frame counter + static void incrementFrameCounter(); + /// Return an iterator over the profiler tree starting at the root static ProfileNodeIterator* getIterator(); /// Print the report of the profiler in a given output stream static void printReport(std::ostream& outputStream); + + /// Destroy a previously allocated iterator + static void destroyIterator(ProfileNodeIterator* iterator); + + /// Destroy the profiler (release the memory) + static void destroy(); }; // Class ProfileSample @@ -177,6 +247,8 @@ class ProfileSample { public : + // -------------------- Methods -------------------- // + /// Constructor ProfileSample(const char* name) { @@ -192,39 +264,120 @@ class ProfileSample { } }; -/// Use this macro to start profile a block of code +// Use this macro to start profile a block of code #define PROFILE(name) ProfileSample profileSample(name) -/// Return a pointer to the parent node -ProfileNode* ProfileNode::getParentNode() { +// Return true if we are at the root of the profiler tree +inline bool ProfileNodeIterator::isRoot() { + return (mCurrentParentNode->getParentNode() == NULL); +} + +// Return true if we are at the end of a branch of the profiler tree +inline bool ProfileNodeIterator::isEnd() { + return (mCurrentChildNode == NULL); +} + +// Return the name of the current node +inline const char* ProfileNodeIterator::getCurrentName() { + return mCurrentChildNode->getName(); +} + +// Return the total time of the current node +inline long double ProfileNodeIterator::getCurrentTotalTime() { + return mCurrentChildNode->getTotalTime(); +} + +// Return the total number of calls of the current node +inline uint ProfileNodeIterator::getCurrentNbTotalCalls() { + return mCurrentChildNode->getNbTotalCalls(); +} + +// Return the name of the current parent node +inline const char* ProfileNodeIterator::getCurrentParentName() { + return mCurrentParentNode->getName(); +} + +// Return the total time of the current parent node +inline long double ProfileNodeIterator::getCurrentParentTotalTime() { + return mCurrentParentNode->getTotalTime(); +} + +// Return the total number of calls of the current parent node +inline uint ProfileNodeIterator::getCurrentParentNbTotalCalls() { + return mCurrentParentNode->getNbTotalCalls(); +} + +// Go to the first node +inline void ProfileNodeIterator::first() { + mCurrentChildNode = mCurrentParentNode->getChildNode(); +} + +// Go to the next node +inline void ProfileNodeIterator::next() { + mCurrentChildNode = mCurrentChildNode->getSiblingNode(); +} + +// Return a pointer to the parent node +inline ProfileNode* ProfileNode::getParentNode() { return mParentNode; } -/// Return a pointer to a sibling node -ProfileNode* ProfileNode::getSiblingNode() { +// Return a pointer to a sibling node +inline ProfileNode* ProfileNode::getSiblingNode() { return mSiblingNode; } -/// Return a pointer to a child node -ProfileNode* ProfileNode::getChildNode() { +// Return a pointer to a child node +inline ProfileNode* ProfileNode::getChildNode() { return mChildNode; } -/// Return the name of the node -const char* ProfileNode::getName() { +// Return the name of the node +inline const char* ProfileNode::getName() { return mName; } -/// Return the total number of call of the corresponding block of code -uint ProfileNode::getNbTotalCalls() const { +// Return the total number of call of the corresponding block of code +inline uint ProfileNode::getNbTotalCalls() const { return mNbTotalCalls; } -/// Return the total time spent in the block of code -long double ProfileNode::getTotalTime() const { +// Return the total time spent in the block of code +inline long double ProfileNode::getTotalTime() const { return mTotalTime; } +// Return the number of frames +inline uint Profiler::getNbFrames() { + return mFrameCounter; +} + +// Return the elasped time since the start/reset of the profiling +inline long double Profiler::getElapsedTimeSinceStart() { + long double currentTime = Timer::getCurrentSystemTime() * 1000.0; + return currentTime - mProfilingStartTime; +} + +// Increment the frame counter +inline void Profiler::incrementFrameCounter() { + mFrameCounter++; +} + +// Return an iterator over the profiler tree starting at the root +inline ProfileNodeIterator* Profiler::getIterator() { + return new ProfileNodeIterator(&mRootNode); +} + +// Destroy a previously allocated iterator +inline void Profiler::destroyIterator(ProfileNodeIterator* iterator) { + delete iterator; +} + +// Destroy the profiler (release the memory) +inline void Profiler::destroy() { + mRootNode.destroy(); +} + } #else // In profile is not active diff --git a/src/engine/Timer.cpp b/src/engine/Timer.cpp index 74c08086..16dbb04c 100644 --- a/src/engine/Timer.cpp +++ b/src/engine/Timer.cpp @@ -39,7 +39,7 @@ Timer::~Timer() { } -// Return the current time of the system +// Return the current time of the system in seconds long double Timer::getCurrentSystemTime() { #if defined(WINDOWS_OS) diff --git a/src/engine/Timer.h b/src/engine/Timer.h index 7c70b52c..6c90ab05 100644 --- a/src/engine/Timer.h +++ b/src/engine/Timer.h @@ -74,7 +74,6 @@ class Timer { /// True if the timer is running bool mIsRunning; - // -------------------- Methods -------------------- // /// Private copy-constructor @@ -123,7 +122,7 @@ class Timer { /// Compute the interpolation factor decimal computeInterpolationFactor(); - /// Return the current time of the system + /// Return the current time of the system in seconds static long double getCurrentSystemTime(); }; From f784dfb3204c50db7c4dfb9fd5aefb73a7d72cf3 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 26 Mar 2013 23:04:03 +0100 Subject: [PATCH 04/66] Start to implement the memory allocator --- src/memory/MemoryAllocator.cpp | 48 ++++++++++++++++++++++++++ src/memory/MemoryAllocator.h | 63 ++++++++++++++++++++++++++++++++++ 2 files changed, 111 insertions(+) create mode 100644 src/memory/MemoryAllocator.cpp create mode 100644 src/memory/MemoryAllocator.h diff --git a/src/memory/MemoryAllocator.cpp b/src/memory/MemoryAllocator.cpp new file mode 100644 index 00000000..073255e3 --- /dev/null +++ b/src/memory/MemoryAllocator.cpp @@ -0,0 +1,48 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "MemoryAllocator.h" + +// Constructor +MemoryAllocator::MemoryAllocator() { + +} + +// Destructor +MemoryAllocator::~MemoryAllocator() { + +} + +// Allocate memory of a given size (in bytes) and return a pointer to the +// allocated memory. +void* MemoryAllocator::allocate(uint size) { + +} + +// Free previously allocated memory. +void MemoryAllocator::free(void* pointer) { + +} diff --git a/src/memory/MemoryAllocator.h b/src/memory/MemoryAllocator.h new file mode 100644 index 00000000..3b380066 --- /dev/null +++ b/src/memory/MemoryAllocator.h @@ -0,0 +1,63 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 MEMORY_ALLOCATOR_H +#define MEMORY_ALLOCATOR_H + +// Libraries + +// Class MemoryAllocator +/** + * This class is used to efficiently allocate memory on the heap. + * It allows us to allocate small blocks of memory (smaller than 1024 bytes) + * efficiently. This implementation is based on the small block allocator + * described here : http://www.codeproject.com/useritems/Small_Block_Allocator.asp + */ +class MemoryAllocator { + + private : + + // -------------------- Attributes -------------------- // + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + MemoryAllocator(); + + /// Destructor + ~MemoryAllocator(); + + /// Allocate memory of a given size (in bytes) and return a pointer to the + /// allocated memory. + void* allocate(uint size); + + /// Free previously allocated memory. + void free(void* pointer); + +}; + +#endif From e03888ea25f565011e52eb5ce104c7089faad538 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Mon, 1 Apr 2013 23:43:50 +0200 Subject: [PATCH 05/66] Finish the implementation of the memory allocator --- src/collision/CollisionDetection.cpp | 17 +- src/collision/CollisionDetection.h | 13 +- .../narrowphase/EPA/EPAAlgorithm.cpp | 6 +- src/collision/narrowphase/EPA/EPAAlgorithm.h | 8 +- .../narrowphase/GJK/GJKAlgorithm.cpp | 12 +- src/collision/narrowphase/GJK/GJKAlgorithm.h | 2 +- .../narrowphase/NarrowPhaseAlgorithm.cpp | 4 +- .../narrowphase/NarrowPhaseAlgorithm.h | 8 +- .../narrowphase/SphereVsSphereAlgorithm.cpp | 6 +- .../narrowphase/SphereVsSphereAlgorithm.h | 2 +- src/constraint/ContactPoint.h | 1 - src/engine/CollisionWorld.cpp | 9 +- src/engine/CollisionWorld.h | 8 +- src/engine/ContactManifold.cpp | 14 +- src/engine/ContactManifold.h | 8 +- src/engine/DynamicsWorld.cpp | 20 +- src/engine/DynamicsWorld.h | 9 - src/engine/OverlappingPair.cpp | 4 +- src/engine/OverlappingPair.h | 2 +- src/memory/MemoryAllocator.cpp | 151 +++++++++- src/memory/MemoryAllocator.h | 91 +++++- src/memory/MemoryPool.h | 283 ------------------ 22 files changed, 306 insertions(+), 372 deletions(-) delete mode 100644 src/memory/MemoryPool.h diff --git a/src/collision/CollisionDetection.cpp b/src/collision/CollisionDetection.cpp index 44e24810..e5359860 100644 --- a/src/collision/CollisionDetection.cpp +++ b/src/collision/CollisionDetection.cpp @@ -43,9 +43,10 @@ using namespace reactphysics3d; using namespace std; // Constructor -CollisionDetection::CollisionDetection(CollisionWorld* world) - : mWorld(world), mNarrowPhaseGJKAlgorithm(mMemoryPoolContactInfos), - mNarrowPhaseSphereVsSphereAlgorithm(mMemoryPoolContactInfos) { +CollisionDetection::CollisionDetection(CollisionWorld* world, MemoryAllocator& memoryAllocator) + : mWorld(world), mMemoryAllocator(memoryAllocator), + mNarrowPhaseGJKAlgorithm(memoryAllocator), + mNarrowPhaseSphereVsSphereAlgorithm(memoryAllocator) { // Create the broad-phase algorithm that will be used (Sweep and Prune with AABB) mBroadPhaseAlgorithm = new SweepAndPruneAlgorithm(*this); @@ -127,9 +128,9 @@ void CollisionDetection::computeNarrowPhase() { // Notify the world about the new narrow-phase contact mWorld->notifyNewContact(pair, contactInfo); - // Delete and remove the contact info from the memory pool + // Delete and remove the contact info from the memory allocator contactInfo->ContactInfo::~ContactInfo(); - mMemoryPoolContactInfos.freeObject(contactInfo); + mMemoryAllocator.release(contactInfo, sizeof(ContactInfo)); } } } @@ -142,7 +143,7 @@ void CollisionDetection::broadPhaseNotifyAddedOverlappingPair(BodyPair* addedPai bodyindexpair indexPair = addedPair->getBodiesIndexPair(); // Create the corresponding broad-phase pair object - BroadPhasePair* broadPhasePair = new (mMemoryPoolBroadPhasePairs.allocateObject()) + BroadPhasePair* broadPhasePair = new (mMemoryAllocator.allocate(sizeof(BroadPhasePair))) BroadPhasePair(addedPair->body1, addedPair->body2); assert(broadPhasePair != NULL); @@ -169,8 +170,8 @@ void CollisionDetection::broadPhaseNotifyRemovedOverlappingPair(BodyPair* remove // Notify the world about the removed broad-phase pair mWorld->notifyRemovedOverlappingPair(broadPhasePair); - // Remove the overlapping pair from the memory pool + // Remove the overlapping pair from the memory allocator broadPhasePair->BroadPhasePair::~BroadPhasePair(); - mMemoryPoolBroadPhasePairs.freeObject(broadPhasePair); + mMemoryAllocator.release(broadPhasePair, sizeof(BroadPhasePair)); mOverlappingPairs.erase(indexPair); } diff --git a/src/collision/CollisionDetection.h b/src/collision/CollisionDetection.h index d44d1b94..71dda4bf 100644 --- a/src/collision/CollisionDetection.h +++ b/src/collision/CollisionDetection.h @@ -30,9 +30,9 @@ #include "../body/CollisionBody.h" #include "broadphase/BroadPhaseAlgorithm.h" #include "BroadPhasePair.h" -#include "../memory/MemoryPool.h" #include "narrowphase/GJK/GJKAlgorithm.h" #include "narrowphase/SphereVsSphereAlgorithm.h" +#include "../memory/MemoryAllocator.h" #include "ContactInfo.h" #include #include @@ -64,6 +64,9 @@ class CollisionDetection { /// Pointer to the physics world CollisionWorld* mWorld; + /// Memory allocator + MemoryAllocator& mMemoryAllocator; + /// Broad-phase overlapping pairs std::map mOverlappingPairs; @@ -76,12 +79,6 @@ class CollisionDetection { /// Narrow-phase Sphere vs Sphere algorithm SphereVsSphereAlgorithm mNarrowPhaseSphereVsSphereAlgorithm; - /// Memory pool for contactinfo - MemoryPool mMemoryPoolContactInfos; - - /// Memory pool for broad-phase pairs - MemoryPool mMemoryPoolBroadPhasePairs; - // -------------------- Methods -------------------- // /// Private copy-constructor @@ -105,7 +102,7 @@ class CollisionDetection { // -------------------- Methods -------------------- // /// Constructor - CollisionDetection(CollisionWorld* world); + CollisionDetection(CollisionWorld* world, MemoryAllocator& memoryAllocator); /// Destructor ~CollisionDetection(); diff --git a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp index 151a7f6c..d14aff7b 100644 --- a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp +++ b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp @@ -32,8 +32,8 @@ using namespace reactphysics3d; // Constructor -EPAAlgorithm::EPAAlgorithm(MemoryPool& memoryPoolContactInfos) - : mMemoryPoolContactInfos(memoryPoolContactInfos) { +EPAAlgorithm::EPAAlgorithm(MemoryAllocator& memoryAllocator) + : mMemoryAllocator(memoryAllocator) { } @@ -400,7 +400,7 @@ bool EPAAlgorithm::computePenetrationDepthAndContactPoints(const Simplex& simple assert(penetrationDepth > 0.0); // Create the contact info object - contactInfo = new (mMemoryPoolContactInfos.allocateObject()) ContactInfo(normal, + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo(normal, penetrationDepth, pALocal, pBLocal); diff --git a/src/collision/narrowphase/EPA/EPAAlgorithm.h b/src/collision/narrowphase/EPA/EPAAlgorithm.h index 4fcdd3be..c9edc3b9 100644 --- a/src/collision/narrowphase/EPA/EPAAlgorithm.h +++ b/src/collision/narrowphase/EPA/EPAAlgorithm.h @@ -32,7 +32,7 @@ #include "../../ContactInfo.h" #include "../../../mathematics/mathematics.h" #include "TriangleEPA.h" -#include "../../../memory/MemoryPool.h" +#include "../../../memory/MemoryAllocator.h" #include /// ReactPhysics3D namespace @@ -85,8 +85,8 @@ class EPAAlgorithm { // -------------------- Attributes -------------------- // - /// Reference to the memory pool - MemoryPool& mMemoryPoolContactInfos; + /// Reference to the memory allocator + MemoryAllocator& mMemoryAllocator; /// Triangle comparison operator TriangleComparison mTriangleComparison; @@ -112,7 +112,7 @@ class EPAAlgorithm { // -------------------- Methods -------------------- // /// Constructor - EPAAlgorithm(MemoryPool& memoryPoolContactInfos); + EPAAlgorithm(MemoryAllocator& memoryAllocator); /// Destructor ~EPAAlgorithm(); diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.cpp b/src/collision/narrowphase/GJK/GJKAlgorithm.cpp index 7e6814d1..b03f3953 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.cpp +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.cpp @@ -37,8 +37,8 @@ using namespace reactphysics3d; // Constructor -GJKAlgorithm::GJKAlgorithm(MemoryPool& memoryPoolContactInfos) - :NarrowPhaseAlgorithm(memoryPoolContactInfos), mAlgoEPA(memoryPoolContactInfos) { +GJKAlgorithm::GJKAlgorithm(MemoryAllocator& memoryAllocator) + :NarrowPhaseAlgorithm(memoryAllocator), mAlgoEPA(memoryAllocator) { } @@ -137,7 +137,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, if (penetrationDepth <= 0.0) return false; // Create the contact info object - contactInfo = new (mMemoryPoolContactInfos.allocateObject()) ContactInfo(normal, + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo(normal, penetrationDepth, pA, pB); @@ -169,7 +169,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, if (penetrationDepth <= 0.0) return false; // Create the contact info object - contactInfo = new (mMemoryPoolContactInfos.allocateObject()) ContactInfo(normal, + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo(normal, penetrationDepth, pA, pB); @@ -199,7 +199,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, if (penetrationDepth <= 0.0) return false; // Create the contact info object - contactInfo = new (mMemoryPoolContactInfos.allocateObject()) ContactInfo(normal, + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo(normal, penetrationDepth, pA, pB); @@ -236,7 +236,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, if (penetrationDepth <= 0.0) return false; // Create the contact info object - contactInfo = new (mMemoryPoolContactInfos.allocateObject()) ContactInfo(normal, + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo(normal, penetrationDepth, pA, pB); diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.h b/src/collision/narrowphase/GJK/GJKAlgorithm.h index 6214d825..fc4af52e 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.h +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.h @@ -85,7 +85,7 @@ class GJKAlgorithm : public NarrowPhaseAlgorithm { // -------------------- Methods -------------------- // /// Constructor - GJKAlgorithm(MemoryPool& memoryPoolContactInfos); + GJKAlgorithm(MemoryAllocator& memoryAllocator); /// Destructor ~GJKAlgorithm(); diff --git a/src/collision/narrowphase/NarrowPhaseAlgorithm.cpp b/src/collision/narrowphase/NarrowPhaseAlgorithm.cpp index 0b214ea0..91a1e4d5 100644 --- a/src/collision/narrowphase/NarrowPhaseAlgorithm.cpp +++ b/src/collision/narrowphase/NarrowPhaseAlgorithm.cpp @@ -30,8 +30,8 @@ using namespace reactphysics3d; // Constructor -NarrowPhaseAlgorithm::NarrowPhaseAlgorithm(MemoryPool& memoryPool) - :mMemoryPoolContactInfos(memoryPool), mCurrentOverlappingPair(NULL) { +NarrowPhaseAlgorithm::NarrowPhaseAlgorithm(MemoryAllocator& memoryAllocator) + :mMemoryAllocator(memoryAllocator), mCurrentOverlappingPair(NULL) { } diff --git a/src/collision/narrowphase/NarrowPhaseAlgorithm.h b/src/collision/narrowphase/NarrowPhaseAlgorithm.h index 8dea2ca8..d68f3f28 100644 --- a/src/collision/narrowphase/NarrowPhaseAlgorithm.h +++ b/src/collision/narrowphase/NarrowPhaseAlgorithm.h @@ -30,7 +30,7 @@ #include "../../body/Body.h" #include "../ContactInfo.h" #include "../broadphase/PairManager.h" -#include "../../memory/MemoryPool.h" +#include "../../memory/MemoryAllocator.h" #include "../BroadPhasePair.h" @@ -50,8 +50,8 @@ class NarrowPhaseAlgorithm { // -------------------- Attributes -------------------- // - /// Reference to the memory pool - MemoryPool& mMemoryPoolContactInfos; + /// Reference to the memory allocator + MemoryAllocator& mMemoryAllocator; /// Overlapping pair of the bodies currently tested for collision BroadPhasePair* mCurrentOverlappingPair; @@ -69,7 +69,7 @@ class NarrowPhaseAlgorithm { // -------------------- Methods -------------------- // /// Constructor - NarrowPhaseAlgorithm(MemoryPool& memoryPool); + NarrowPhaseAlgorithm(MemoryAllocator& memoryAllocator); /// Destructor virtual ~NarrowPhaseAlgorithm(); diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp index cded8912..2f950e04 100644 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp @@ -31,8 +31,8 @@ using namespace reactphysics3d; // Constructor -SphereVsSphereAlgorithm::SphereVsSphereAlgorithm(MemoryPool& memoryPoolContactInfos) - :NarrowPhaseAlgorithm(memoryPoolContactInfos) { +SphereVsSphereAlgorithm::SphereVsSphereAlgorithm(MemoryAllocator& memoryAllocator) + :NarrowPhaseAlgorithm(memoryAllocator) { } @@ -69,7 +69,7 @@ bool SphereVsSphereAlgorithm::testCollision(const CollisionShape* collisionShape decimal penetrationDepth = sumRadius - std::sqrt(squaredDistanceBetweenCenters); // Create the contact info object - contactInfo = new (mMemoryPoolContactInfos.allocateObject()) ContactInfo( + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo( vectorBetweenCenters.getUnit(), penetrationDepth, intersectionOnBody1, intersectionOnBody2); diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.h b/src/collision/narrowphase/SphereVsSphereAlgorithm.h index bb966df1..1c50fb1f 100644 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.h +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.h @@ -57,7 +57,7 @@ class SphereVsSphereAlgorithm : public NarrowPhaseAlgorithm { // -------------------- Methods -------------------- // /// Constructor - SphereVsSphereAlgorithm(MemoryPool& memoryPoolContactInfos); + SphereVsSphereAlgorithm(MemoryAllocator& memoryAllocator); /// Destructor virtual ~SphereVsSphereAlgorithm(); diff --git a/src/constraint/ContactPoint.h b/src/constraint/ContactPoint.h index e501e036..0900d480 100644 --- a/src/constraint/ContactPoint.h +++ b/src/constraint/ContactPoint.h @@ -32,7 +32,6 @@ #include "../body/RigidBody.h" #include "../configuration.h" #include "../mathematics/mathematics.h" -#include "../memory/MemoryPool.h" #include "../configuration.h" #if defined(VISUAL_DEBUG) diff --git a/src/engine/CollisionWorld.cpp b/src/engine/CollisionWorld.cpp index af7eb189..39b9dc74 100644 --- a/src/engine/CollisionWorld.cpp +++ b/src/engine/CollisionWorld.cpp @@ -32,7 +32,8 @@ using namespace reactphysics3d; using namespace std; // Constructor -CollisionWorld::CollisionWorld() : mCollisionDetection(this), mCurrentBodyID(0) { +CollisionWorld::CollisionWorld() + : mCollisionDetection(this, mMemoryAllocator), mCurrentBodyID(0) { } // Destructor @@ -75,7 +76,7 @@ CollisionBody* CollisionWorld::createCollisionBody(const Transform& transform, assert(bodyID < std::numeric_limits::max()); // Create the collision body - CollisionBody* collisionBody = new (mMemoryPoolCollisionBodies.allocateObject()) + CollisionBody* collisionBody = new (mMemoryAllocator.allocate(sizeof(CollisionBody))) CollisionBody(transform, collisionShape, bodyID); assert(collisionBody != NULL); @@ -105,8 +106,8 @@ void CollisionWorld::destroyCollisionBody(CollisionBody* collisionBody) { // Remove the collision body from the list of bodies mBodies.erase(collisionBody); // TODO : Maybe use a set to make this faster - // Free the object from the memory pool - mMemoryPoolCollisionBodies.freeObject(collisionBody); + // Free the object from the memory allocator + mMemoryAllocator.release(collisionBody, sizeof(CollisionBody)); } // Return the next available body ID diff --git a/src/engine/CollisionWorld.h b/src/engine/CollisionWorld.h index 4a929a3a..00de6cdf 100644 --- a/src/engine/CollisionWorld.h +++ b/src/engine/CollisionWorld.h @@ -37,7 +37,7 @@ #include "../collision/CollisionDetection.h" #include "../constraint/Constraint.h" #include "../constraint/ContactPoint.h" -#include "../memory/MemoryPool.h" +#include "../memory/MemoryAllocator.h" /// Namespace reactphysics3d namespace reactphysics3d { @@ -66,12 +66,12 @@ class CollisionWorld { /// Current body ID bodyindex mCurrentBodyID; - /// Memory pool - MemoryPool mMemoryPoolCollisionBodies; - /// List of free ID for rigid bodies std::vector mFreeBodiesIDs; + /// Memory allocator + MemoryAllocator mMemoryAllocator; + // -------------------- Methods -------------------- // /// Private copy-constructor diff --git a/src/engine/ContactManifold.cpp b/src/engine/ContactManifold.cpp index be08a488..69a67e97 100644 --- a/src/engine/ContactManifold.cpp +++ b/src/engine/ContactManifold.cpp @@ -31,10 +31,10 @@ using namespace reactphysics3d; // Constructor ContactManifold::ContactManifold(Body* const body1, Body* const body2, - MemoryPool& memoryPoolContacts) + MemoryAllocator& memoryAllocator) : mBody1(body1), mBody2(body2), mNbContactPoints(0), mFrictionImpulse1(0.0), mFrictionImpulse2(0.0), mFrictionTwistImpulse(0.0), - mMemoryPoolContacts(memoryPoolContacts) { + mMemoryAllocator(memoryAllocator) { } @@ -57,7 +57,7 @@ void ContactManifold::addContactPoint(ContactPoint* contact) { // Delete the new contact contact->ContactPoint::~ContactPoint(); - mMemoryPoolContacts.freeObject(contact); + mMemoryAllocator.release(contact, sizeof(ContactPoint)); //removeContact(i); return; @@ -82,10 +82,10 @@ void ContactManifold::removeContactPoint(uint index) { assert(index < mNbContactPoints); assert(mNbContactPoints > 0); - // Call the destructor explicitly and tell the memory pool that + // Call the destructor explicitly and tell the memory allocator that // the corresponding memory block is now free mContactPoints[index]->ContactPoint::~ContactPoint(); - mMemoryPoolContacts.freeObject(mContactPoints[index]); + mMemoryAllocator.release(mContactPoints[index], sizeof(ContactPoint)); // If we don't remove the last index if (index < mNbContactPoints - 1) { @@ -243,10 +243,10 @@ int ContactManifold::getMaxArea(decimal area0, decimal area1, decimal area2, dec void ContactManifold::clear() { for (uint i=0; iContactPoint::~ContactPoint(); - mMemoryPoolContacts.freeObject(mContactPoints[i]); + mMemoryAllocator.release(mContactPoints[i], sizeof(ContactPoint)); } mNbContactPoints = 0; } diff --git a/src/engine/ContactManifold.h b/src/engine/ContactManifold.h index 02ffbf63..68c1d2f3 100644 --- a/src/engine/ContactManifold.h +++ b/src/engine/ContactManifold.h @@ -30,6 +30,7 @@ #include #include "../body/Body.h" #include "../constraint/ContactPoint.h" +#include "../memory/MemoryAllocator.h" /// ReactPhysics3D namespace namespace reactphysics3d { @@ -85,8 +86,8 @@ class ContactManifold { /// Twist friction constraint accumulated impulse decimal mFrictionTwistImpulse; - /// Reference to the memory pool with the contacts - MemoryPool& mMemoryPoolContacts; + /// Reference to the memory allocator + MemoryAllocator& mMemoryAllocator; // -------------------- Methods -------------------- // @@ -116,8 +117,7 @@ class ContactManifold { // -------------------- Methods -------------------- // /// Constructor - ContactManifold(Body* const mBody1, Body* const mBody2, - MemoryPool& mMemoryPoolContacts); + ContactManifold(Body* const mBody1, Body* const mBody2, MemoryAllocator& memoryAllocator); /// Destructor ~ContactManifold(); diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 48940f82..27184f98 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -47,7 +47,7 @@ DynamicsWorld::~DynamicsWorld() { for (it = mOverlappingPairs.begin(); it != mOverlappingPairs.end(); it++) { // Delete the overlapping pair (*it).second->OverlappingPair::~OverlappingPair(); - mMemoryPoolOverlappingPairs.freeObject((*it).second); + mMemoryAllocator.release((*it).second, sizeof(OverlappingPair)); } // Free the allocated memory for the constrained velocities @@ -267,7 +267,8 @@ RigidBody* DynamicsWorld::createRigidBody(const Transform& transform, decimal ma assert(bodyID < std::numeric_limits::max()); // Create the rigid body - RigidBody* rigidBody = new (mMemoryPoolRigidBodies.allocateObject()) RigidBody(transform, mass, + RigidBody* rigidBody = new (mMemoryAllocator.allocate(sizeof(RigidBody))) RigidBody(transform, + mass, inertiaTensorLocal, collisionShape, bodyID); @@ -300,8 +301,8 @@ void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { mBodies.erase(rigidBody); mRigidBodies.erase(rigidBody); - // Free the object from the memory pool - mMemoryPoolRigidBodies.freeObject(rigidBody); + // Free the object from the memory allocator + mMemoryAllocator.release(rigidBody, sizeof(RigidBody)); } // Remove all constraints in the physics world @@ -316,8 +317,8 @@ void DynamicsWorld::notifyAddedOverlappingPair(const BroadPhasePair* addedPair) bodyindexpair indexPair = addedPair->getBodiesIndexPair(); // Add the pair into the set of overlapping pairs (if not there yet) - OverlappingPair* newPair = new (mMemoryPoolOverlappingPairs.allocateObject()) OverlappingPair( - addedPair->body1, addedPair->body2, mMemoryPoolContacts); + OverlappingPair* newPair = new (mMemoryAllocator.allocate(sizeof(OverlappingPair))) OverlappingPair( + addedPair->body1, addedPair->body2, mMemoryAllocator); assert(newPair != NULL); std::pair::iterator, bool> check = mOverlappingPairs.insert(make_pair(indexPair, newPair)); @@ -330,9 +331,9 @@ void DynamicsWorld::notifyRemovedOverlappingPair(const BroadPhasePair* removedPa // Get the pair of body index std::pair indexPair = removedPair->getBodiesIndexPair(); - // Remove the overlapping pair from the memory pool + // Remove the overlapping pair from the memory allocator mOverlappingPairs.find(indexPair)->second->OverlappingPair::~OverlappingPair(); - mMemoryPoolOverlappingPairs.freeObject(mOverlappingPairs[indexPair]); + mMemoryAllocator.release(mOverlappingPairs[indexPair], sizeof(OverlappingPair)); mOverlappingPairs.erase(indexPair); } @@ -347,7 +348,8 @@ void DynamicsWorld::notifyNewContact(const BroadPhasePair* broadPhasePair, assert(rigidBody2 != NULL); // Create a new contact - ContactPoint* contact = new (mMemoryPoolContacts.allocateObject()) ContactPoint(rigidBody1, + ContactPoint* contact = new (mMemoryAllocator.allocate(sizeof(ContactPoint))) ContactPoint( + rigidBody1, rigidBody2, contactInfo); assert(contact != NULL); diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index 03abe3f8..fb66aa7b 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -73,15 +73,6 @@ class DynamicsWorld : public CollisionWorld { /// True if the gravity force is on bool mIsGravityOn; - /// Memory pool for the overlapping pairs - MemoryPool mMemoryPoolOverlappingPairs; - - /// Memory pool for rigid bodies memory allocation - MemoryPool mMemoryPoolRigidBodies; - - /// Memory pool for the contacts - MemoryPool mMemoryPoolContacts; - /// Array of constrained linear velocities (state of the linear velocities /// after solving the constraints) std::vector mConstrainedLinearVelocities; diff --git a/src/engine/OverlappingPair.cpp b/src/engine/OverlappingPair.cpp index 8ca9b511..69a459bb 100644 --- a/src/engine/OverlappingPair.cpp +++ b/src/engine/OverlappingPair.cpp @@ -31,8 +31,8 @@ using namespace reactphysics3d; // Constructor OverlappingPair::OverlappingPair(CollisionBody* body1, CollisionBody* body2, - MemoryPool& memoryPoolContacts) - : mBody1(body1), mBody2(body2), mContactManifold(body1, body2, memoryPoolContacts), + MemoryAllocator &memoryAllocator) + : mBody1(body1), mBody2(body2), mContactManifold(body1, body2, memoryAllocator), mCachedSeparatingAxis(1.0, 1.0, 1.0) { } diff --git a/src/engine/OverlappingPair.h b/src/engine/OverlappingPair.h index 36beba56..b3c16880 100644 --- a/src/engine/OverlappingPair.h +++ b/src/engine/OverlappingPair.h @@ -72,7 +72,7 @@ class OverlappingPair { /// Constructor OverlappingPair(CollisionBody* body1, CollisionBody* body2, - MemoryPool& memoryPoolContacts); + MemoryAllocator& memoryAllocator); /// Destructor ~OverlappingPair(); diff --git a/src/memory/MemoryAllocator.cpp b/src/memory/MemoryAllocator.cpp index 073255e3..e715a157 100644 --- a/src/memory/MemoryAllocator.cpp +++ b/src/memory/MemoryAllocator.cpp @@ -25,24 +25,169 @@ // Libraries #include "MemoryAllocator.h" +#include +#include + +using namespace reactphysics3d; + +// Initialization of static variables +bool MemoryAllocator::isMapSizeToHeadIndexInitialized = false; +size_t MemoryAllocator::mUnitSizes[NB_HEAPS]; +int MemoryAllocator::mMapSizeToHeapIndex[MAX_UNIT_SIZE + 1]; // Constructor MemoryAllocator::MemoryAllocator() { + // Allocate some memory to manage the blocks + mNbAllocatedMemoryBlocks = 64; + mNbCurrentMemoryBlocks = 0; + const size_t sizeToAllocate = mNbAllocatedMemoryBlocks * sizeof(MemoryBlock); + mMemoryBlocks = (MemoryBlock*) malloc(sizeToAllocate); + memset(mMemoryBlocks, NULL, sizeToAllocate); + memset(mFreeMemoryUnits, NULL, sizeof(mFreeMemoryUnits)); + +#ifndef NDEBUG + mNbTimesAllocateMethodCalled = 0; +#endif + + // If the mMapSizeToHeapIndex has not been initialized yet + if (!isMapSizeToHeadIndexInitialized) { + + // Initialize the array that contains the sizes the memory units that will + // be allocated in each different heap + for (uint i=0; i < NB_HEAPS; i++) { + mUnitSizes[i] = (i+1) * 8; + } + + // Initialize the lookup table that maps the size to allocated to the + // corresponding heap we will use for the allocation + uint j = 0; + mMapSizeToHeapIndex[0] = -1; // This element should not be used + for (uint i=1; i <= MAX_UNIT_SIZE; i++) { + if (i <= mUnitSizes[j]) { + mMapSizeToHeapIndex[i] = j; + } + else { + j++; + mMapSizeToHeapIndex[i] = j; + } + } + + isMapSizeToHeadIndexInitialized = true; + } } // Destructor MemoryAllocator::~MemoryAllocator() { + // Release the memory allocated for each block + for (uint i=0; i MAX_UNIT_SIZE) { + + // Allocate memory using standard malloc() function + return malloc(size); + } + + // Get the index of the heap that will take care of the allocation request + int indexHeap = mMapSizeToHeapIndex[size]; + assert(indexHeap >= 0 && indexHeap < NB_HEAPS); + + // If there still are free memory units in the corresponding heap + if (mFreeMemoryUnits[indexHeap] != NULL) { + + // Return a pointer to the memory unit + MemoryUnit* unit = mFreeMemoryUnits[indexHeap]; + mFreeMemoryUnits[indexHeap] = unit->nextUnit; + return unit; + } + else { // If there is no more free memory units in the corresponding heap + + // If we need to allocate more memory to containsthe blocks + if (mNbCurrentMemoryBlocks == mNbAllocatedMemoryBlocks) { + + // Allocate more memory to contain the blocks + MemoryBlock* currentMemoryBlocks = mMemoryBlocks; + mNbAllocatedMemoryBlocks += 64; + mMemoryBlocks = (MemoryBlock*) malloc(mNbAllocatedMemoryBlocks * sizeof(MemoryBlock)); + memcpy(mMemoryBlocks, currentMemoryBlocks,mNbCurrentMemoryBlocks * sizeof(MemoryBlock)); + memset(mMemoryBlocks + mNbCurrentMemoryBlocks, 0, 64 * sizeof(MemoryBlock)); + free(currentMemoryBlocks); + } + + // Allocate a new memory blocks for the corresponding heap and divide it in many + // memory units + MemoryBlock* newBlock = mMemoryBlocks + mNbCurrentMemoryBlocks; + newBlock->memoryUnits = (MemoryUnit*) malloc(BLOCK_SIZE); + assert(newBlock->memoryUnits != NULL); + size_t unitSize = mUnitSizes[indexHeap]; + uint nbUnits = BLOCK_SIZE / unitSize; + assert(nbUnits * unitSize <= BLOCK_SIZE); + for (size_t i=0; i < nbUnits - 1; i++) { + MemoryUnit* unit = (MemoryUnit*) ((size_t)newBlock->memoryUnits + unitSize * i); + MemoryUnit* nextUnit = (MemoryUnit*) ((size_t)newBlock->memoryUnits + unitSize * (i + 1)); + unit->nextUnit = nextUnit; + } + MemoryUnit* lastUnit = (MemoryUnit*) ((size_t)newBlock->memoryUnits + unitSize * (nbUnits - 1)); + lastUnit->nextUnit = NULL; + + // Add the new allocated block into the list of free memory units in the heap + mFreeMemoryUnits[indexHeap] = newBlock->memoryUnits->nextUnit; + mNbCurrentMemoryBlocks++; + + // Return the pointer to the first memory unit of the new allocated block + return newBlock->memoryUnits; + } } -// Free previously allocated memory. -void MemoryAllocator::free(void* pointer) { +// Release previously allocated memory. +void MemoryAllocator::release(void* pointer, size_t size) { + // Cannot release a 0-byte allocated memory + if (size == 0) return; + +#ifndef NDEBUG + mNbTimesAllocateMethodCalled--; +#endif + + // If the size is larger than the maximum memory unit size + if (size > MAX_UNIT_SIZE) { + + // Release the memory using the standard free() function + free(pointer); + return; + } + + // Get the index of the heap that has handled the corresponding allocation request + int indexHeap = mMapSizeToHeapIndex[size]; + assert(indexHeap >= 0 && indexHeap < NB_HEAPS); + + // Insert the released memory unit into the list of free memory units of the + // corresponding heap + MemoryUnit* releasedUnit = (MemoryUnit*) pointer; + releasedUnit->nextUnit = mFreeMemoryUnits[indexHeap]; + mFreeMemoryUnits[indexHeap] = releasedUnit; } diff --git a/src/memory/MemoryAllocator.h b/src/memory/MemoryAllocator.h index 3b380066..8ef90e14 100644 --- a/src/memory/MemoryAllocator.h +++ b/src/memory/MemoryAllocator.h @@ -27,20 +27,99 @@ #define MEMORY_ALLOCATOR_H // Libraries +#include "../configuration.h" + +/// ReactPhysics3D namespace +namespace reactphysics3d { // Class MemoryAllocator /** * This class is used to efficiently allocate memory on the heap. - * It allows us to allocate small blocks of memory (smaller than 1024 bytes) - * efficiently. This implementation is based on the small block allocator + * It allows us to allocate small blocks of memory (smaller or equal to 1024 bytes) + * efficiently. This implementation is inspired by the small block allocator * described here : http://www.codeproject.com/useritems/Small_Block_Allocator.asp */ class MemoryAllocator { private : + // -------------------- Internal Classes -------------------- // + + // Structure MemoryUnit + /** + * Represent a memory unit that is used for a single memory allocation + * request. + */ + struct MemoryUnit { + + public : + + // -------------------- Attributes -------------------- // + + /// Pointer to the next memory unit inside a memory block + MemoryUnit* nextUnit; + + }; + + // Structure MemoryBlock + /** + * A memory block is a large piece of memory that is allocated once and that + * will contain multiple memory unity. + */ + struct MemoryBlock { + + public : + + /// Pointer to the first element of a linked-list of memory unity. + MemoryUnit* memoryUnits; + }; + + // -------------------- Constants -------------------- // + + /// Number of heaps + static const uint NB_HEAPS = 128; + + /// Maximum memory unit size. An allocation request of a size smaller or equal to + /// this size will be handled using the small block allocator. However, for an + /// allocation request larger than the maximum block size, the standard malloc() + /// will be used. + static const size_t MAX_UNIT_SIZE = 1024; + + /// Size a memory chunk + static const size_t BLOCK_SIZE = 16 * MAX_UNIT_SIZE; + // -------------------- Attributes -------------------- // + /// Size of the memory units that each heap is responsible to allocate + static size_t mUnitSizes[NB_HEAPS]; + + /// Lookup table that mape size to allocate to the index of the + /// corresponding heap we will use for the allocation. + static int mMapSizeToHeapIndex[MAX_UNIT_SIZE + 1]; + + /// True if the mMapSizeToHeapIndex array has already been initialized + static bool isMapSizeToHeadIndexInitialized; + + /// Pointers to the first free memory unit for each heap + MemoryUnit* mFreeMemoryUnits[NB_HEAPS]; + + /// All the allocated memory blocks + MemoryBlock* mMemoryBlocks; + + /// Number of allocated memory blocks + uint mNbAllocatedMemoryBlocks; + + /// Current number of used memory blocks + uint mNbCurrentMemoryBlocks; + +#ifndef NDEBUG + /// This variable is incremented by one when the allocate() method has been + /// called and decreased by one when the release() method has been called. + /// This variable is used in debug mode to check that the allocate() and release() + /// methods are called the same number of times + int mNbTimesAllocateMethodCalled; +#endif + public : // -------------------- Methods -------------------- // @@ -53,11 +132,13 @@ class MemoryAllocator { /// Allocate memory of a given size (in bytes) and return a pointer to the /// allocated memory. - void* allocate(uint size); + void* allocate(size_t size); - /// Free previously allocated memory. - void free(void* pointer); + /// Release previously allocated memory. + void release(void* pointer, size_t size); }; +} + #endif diff --git a/src/memory/MemoryPool.h b/src/memory/MemoryPool.h deleted file mode 100644 index 1a0ff546..00000000 --- a/src/memory/MemoryPool.h +++ /dev/null @@ -1,283 +0,0 @@ -/******************************************************************************** -* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * -* Copyright (c) 2010-2013 Daniel Chappuis * -********************************************************************************* -* * -* 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 MEMORY_POOL_H -#define MEMORY_POOL_H - -// Libraries -#include "../configuration.h" -#include -#include -#include -#include - -// TODO : Check that casting is done correctly in this class using -// C++ cast operator like reinterpret_cast<>, ... - - -/// ReactPhysics3D namespace -namespace reactphysics3d { - -// Class MemoryPool -/** - * This class represents a memory pool. This memory pool allows us to allocate - * dynamic memory at the beginning of the application in order to - * avoid memory fragmentation and also a large number of allocation - * and deallocation. - */ -template -class MemoryPool { - - private: - - /// MemoryUnit represents a unit of memory - struct MemoryUnit { - struct MemoryUnit* pNext; // Pointer to the next memory unit - struct MemoryUnit* pPrevious; // Pointer to the previous memory unit - }; - - /// Memory block (that contains several memory units) - struct MemoryBlock { - struct MemoryBlock* pNext; // Pointer to the next memory block - }; - - // -------------------- Constants -------------------- // - - // Number of objects allocated in the first block - static const uint NB_OBJECTS_FIRST_BLOCK; - - // -------------------- Attributes -------------------- // - - /// Pointer to the first allocated memory block - void* mPBlocks; - - /// Pointer to the first allocated memory unit - MemoryUnit* mPAllocatedUnits; - - /// Pointer to the first free memory unit - MemoryUnit* mPFreeUnits; - - /// Current number of objects in the pool - uint mCurrentNbObjects; - - /// Current maximum number of objects that can be allocated in the pool - uint mCapacity; - - /// Number of objects to allocate in the next block - uint mNbObjectsNextBlock; - - // -------------------- Methods -------------------- // - - /// Private copy-constructor - MemoryPool(const MemoryPool& body); - - /// Private assignment operator - MemoryPool& operator=(const MemoryPool& timer); - - /// Allocate more memory (more blocks) when needed - void allocateMemory(); - - public: - - // -------------------- Methods -------------------- // - - /// Constructor - MemoryPool(uint capacity = 0) throw(std::bad_alloc); - - /// Destructor - ~MemoryPool(); - - /// Return the current maximum number of objects allowed in the pool - uint getCapacity() const; - - /// Return the current number of objects in the pool - uint getCurrentNbObjects() const; - - /// Return a pointer to an memory allocated location to store a new object - void* allocateObject(); - - /// Tell the pool that an object doesn't need to be store in the pool anymore - void freeObject(void* pObjectToFree); -}; - -// static member definition -template const uint MemoryPool::NB_OBJECTS_FIRST_BLOCK = 100; - -// Constructor -/// Allocate a large block of memory in order to contain -/// a given number of object of the template type T -template -MemoryPool::MemoryPool(uint capacity) throw(std::bad_alloc) - : mCurrentNbObjects(0), mCapacity(capacity) { - mPBlocks = NULL; - mPAllocatedUnits = NULL; - mPFreeUnits = NULL; - mNbObjectsNextBlock = (capacity == 0) ? NB_OBJECTS_FIRST_BLOCK : capacity; - - // Allocate the first memory block if the capacity is - // different from zero - if (capacity) allocateMemory(); -} - -// Destructor -/// Deallocate the blocks of memory that have been allocated previously -template -MemoryPool::~MemoryPool() { - - // Check if we have a memory leak - assert(mCurrentNbObjects == 0); - - // Release all the allocated memory blocks - MemoryBlock* currentBlock = (MemoryBlock*) mPBlocks; - while(currentBlock) { - MemoryBlock* tempBlock = currentBlock->pNext; - free(currentBlock); - currentBlock = tempBlock; - } -} - -// Return a pointer to a memory allocated location to store a new object. -/// This method only allocates memory if needed and it returns a pointer -/// to a location in an allocated block of memory where a new object can be stored -template -void* MemoryPool::allocateObject() { - - // If there is not enough allocated memory in the pool - if (mCurrentNbObjects == mCapacity) { - - // Allocate a new memory block - allocateMemory(); - } - - assert(mCurrentNbObjects < mCapacity); - assert(mPFreeUnits != NULL); - - MemoryUnit* currentUnit = mPFreeUnits; - mPFreeUnits = currentUnit->pNext; - if (mPFreeUnits) { - mPFreeUnits->pPrevious = NULL; - } - - currentUnit->pNext = mPAllocatedUnits; - if (mPAllocatedUnits) { - mPAllocatedUnits->pPrevious = currentUnit; - } - mPAllocatedUnits = currentUnit; - - mCurrentNbObjects++; - - // Return a pointer to the allocated memory unit - return (void*)((char*)currentUnit + sizeof(MemoryUnit)); -} - -// Tell the pool that an object does not need to be stored in the pool anymore -/// This method does not deallocate memory because it will be done only at the -/// end but it notifies the memory pool that an object that was stored in the pool -/// does not need to be stored anymore and therefore we can use the corresponding -/// location in the pool for another object -template -void MemoryPool::freeObject(void* pObjectToFree) { - - // The pointer location must be inside the memory block - //assert(pBlockspNext; - if (mPAllocatedUnits) { - mPAllocatedUnits->pPrevious = NULL; - } - - currentUnit->pNext = mPFreeUnits; - if (mPFreeUnits) { - mPFreeUnits->pPrevious = currentUnit; - } - mPFreeUnits = currentUnit; - - mCurrentNbObjects--; -} - -// Allocate more memory (more blocks) when needed -/// This method is called when there are no -/// free memory units available anymore. Therefore, we need to allocate -/// a new memory block in order to be able to store several new memory units. -template -void MemoryPool::allocateMemory() { - - // Compute the size of the new - size_t sizeBlock = mNbObjectsNextBlock * (sizeof(MemoryUnit) + sizeof(T)); - - MemoryBlock* tempBlocks = (MemoryBlock*) mPBlocks; - - // Allocate a new memory block - mPBlocks = malloc(sizeBlock); - - // Check that the allocation did not fail - if (mPBlocks == NULL) throw std::bad_alloc(); - - MemoryBlock* block = (MemoryBlock*) mPBlocks; - block->pNext = tempBlocks; - - // For each allocated memory unit in the new block - for (uint i=0; ipPrevious = NULL; - currentUnit->pNext = mPFreeUnits; - - if (mPFreeUnits) { - mPFreeUnits->pPrevious = currentUnit; - } - - mPFreeUnits = currentUnit; - } - - // Update the current capacity of the memory pool - mCapacity += mNbObjectsNextBlock; - - // The next block will be two times the size of the last - // allocated memory block - mNbObjectsNextBlock *= 2; -} - - -// Return the maximum number of objects allowed in the pool -template -uint MemoryPool::getCapacity() const { - return mCapacity; -} - -// Return the current number of objects in the pool -template -uint MemoryPool::getCurrentNbObjects() const { - return mCurrentNbObjects; -} - -} - -#endif - From a1bd7f7be9071b3e33f98456f01f99aa3910f6c6 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sun, 14 Apr 2013 21:32:40 +0200 Subject: [PATCH 06/66] Use the opengl-framework library for the examples --- CMakeLists.txt | 2 +- examples/CMakeLists.txt | 5 + examples/fallingcubes/Box.cpp | 157 +- examples/fallingcubes/Box.h | 103 +- examples/fallingcubes/CMakeLists.txt | 49 +- examples/fallingcubes/Scene.cpp | 169 + examples/fallingcubes/Scene.h | 83 + examples/fallingcubes/main.cpp | 315 +- examples/opengl-framework/CMakeLists.txt | 55 + .../opengl-framework/freeglut/CMakeLists.txt | 24 + .../opengl-framework/freeglut/COPYING.txt | 27 + .../opengl-framework/freeglut/GL/freeglut.h | 22 + .../freeglut/GL/freeglut_ext.h | 236 ++ .../freeglut/GL/freeglut_std.h | 628 ++++ examples/opengl-framework/freeglut/GL/glut.h | 21 + .../opengl-framework/freeglut/VERSION.txt | 1 + .../freeglut/freeglut_callbacks.c | 412 +++ .../freeglut/freeglut_cursor.c | 282 ++ .../freeglut/freeglut_display.c | 103 + .../opengl-framework/freeglut/freeglut_ext.c | 233 ++ .../opengl-framework/freeglut/freeglut_font.c | 384 +++ .../freeglut/freeglut_font_data.c | 2020 ++++++++++++ .../freeglut/freeglut_gamemode.c | 819 +++++ .../freeglut/freeglut_geometry.c | 1215 +++++++ .../freeglut/freeglut_glutfont_definitions.c | 108 + .../opengl-framework/freeglut/freeglut_init.c | 1180 +++++++ .../freeglut/freeglut_input_devices.c | 378 +++ .../freeglut/freeglut_internal.h | 1023 ++++++ .../freeglut/freeglut_joystick.c | 1800 +++++++++++ .../opengl-framework/freeglut/freeglut_main.c | 2511 +++++++++++++++ .../opengl-framework/freeglut/freeglut_menu.c | 1002 ++++++ .../opengl-framework/freeglut/freeglut_misc.c | 214 ++ .../freeglut/freeglut_overlay.c | 45 + .../freeglut/freeglut_spaceball.c | 471 +++ .../freeglut/freeglut_state.c | 887 +++++ .../freeglut/freeglut_stroke_mono_roman.c | 2849 +++++++++++++++++ .../freeglut/freeglut_stroke_roman.c | 2849 +++++++++++++++++ .../freeglut/freeglut_structure.c | 598 ++++ .../freeglut/freeglut_teapot.c | 200 ++ .../freeglut/freeglut_teapot_data.h | 2429 ++++++++++++++ .../freeglut/freeglut_videoresize.c | 50 + .../freeglut/freeglut_window.c | 2162 +++++++++++++ .../freeglut/freeglut_xinput.c | 219 ++ examples/opengl-framework/freeglut/glut.h | 21 + examples/opengl-framework/src/Camera.cpp | 97 + examples/opengl-framework/src/Camera.h | 181 ++ .../src/FrameBufferObject.cpp | 115 + .../opengl-framework/src/FrameBufferObject.h | 105 + examples/opengl-framework/src/GlutViewer.cpp | 260 ++ examples/opengl-framework/src/GlutViewer.h | 160 + examples/opengl-framework/src/Light.cpp | 100 + examples/opengl-framework/src/Light.h | 181 ++ examples/opengl-framework/src/Mesh.cpp | 177 + examples/opengl-framework/src/Mesh.h | 411 +++ .../opengl-framework/src/MeshReaderWriter.cpp | 397 +++ .../opengl-framework/src/MeshReaderWriter.h | 100 + examples/opengl-framework/src/Object3D.cpp | 41 + examples/opengl-framework/src/Object3D.h | 149 + examples/opengl-framework/src/Shader.cpp | 207 ++ examples/opengl-framework/src/Shader.h | 243 ++ examples/opengl-framework/src/Texture2D.cpp | 84 + examples/opengl-framework/src/Texture2D.h | 142 + .../src/TextureReaderWriter.cpp | 329 ++ .../src/TextureReaderWriter.h | 87 + .../src/VertexBufferObject.cpp | 84 + .../opengl-framework/src/VertexBufferObject.h | 106 + examples/opengl-framework/src/definitions.h | 39 + examples/opengl-framework/src/maths/Color.h | 76 + examples/opengl-framework/src/maths/Matrix3.h | 279 ++ examples/opengl-framework/src/maths/Matrix4.h | 419 +++ examples/opengl-framework/src/maths/Vector2.h | 159 + examples/opengl-framework/src/maths/Vector3.h | 203 ++ examples/opengl-framework/src/maths/Vector4.h | 167 + .../opengl-framework/src/openglframework.h | 49 + .../opengl-framework/src/shaders/depth.frag | 32 + .../opengl-framework/src/shaders/depth.vert | 36 + .../opengl-framework/src/shaders/phong.frag | 63 + .../opengl-framework/src/shaders/phong.vert | 51 + 78 files changed, 33401 insertions(+), 289 deletions(-) create mode 100644 examples/CMakeLists.txt mode change 100755 => 100644 examples/fallingcubes/CMakeLists.txt create mode 100644 examples/fallingcubes/Scene.cpp create mode 100644 examples/fallingcubes/Scene.h create mode 100644 examples/opengl-framework/CMakeLists.txt create mode 100644 examples/opengl-framework/freeglut/CMakeLists.txt create mode 100644 examples/opengl-framework/freeglut/COPYING.txt create mode 100644 examples/opengl-framework/freeglut/GL/freeglut.h create mode 100644 examples/opengl-framework/freeglut/GL/freeglut_ext.h create mode 100644 examples/opengl-framework/freeglut/GL/freeglut_std.h create mode 100644 examples/opengl-framework/freeglut/GL/glut.h create mode 100644 examples/opengl-framework/freeglut/VERSION.txt create mode 100644 examples/opengl-framework/freeglut/freeglut_callbacks.c create mode 100644 examples/opengl-framework/freeglut/freeglut_cursor.c create mode 100644 examples/opengl-framework/freeglut/freeglut_display.c create mode 100644 examples/opengl-framework/freeglut/freeglut_ext.c create mode 100644 examples/opengl-framework/freeglut/freeglut_font.c create mode 100644 examples/opengl-framework/freeglut/freeglut_font_data.c create mode 100644 examples/opengl-framework/freeglut/freeglut_gamemode.c create mode 100644 examples/opengl-framework/freeglut/freeglut_geometry.c create mode 100644 examples/opengl-framework/freeglut/freeglut_glutfont_definitions.c create mode 100644 examples/opengl-framework/freeglut/freeglut_init.c create mode 100644 examples/opengl-framework/freeglut/freeglut_input_devices.c create mode 100644 examples/opengl-framework/freeglut/freeglut_internal.h create mode 100644 examples/opengl-framework/freeglut/freeglut_joystick.c create mode 100644 examples/opengl-framework/freeglut/freeglut_main.c create mode 100644 examples/opengl-framework/freeglut/freeglut_menu.c create mode 100644 examples/opengl-framework/freeglut/freeglut_misc.c create mode 100644 examples/opengl-framework/freeglut/freeglut_overlay.c create mode 100644 examples/opengl-framework/freeglut/freeglut_spaceball.c create mode 100644 examples/opengl-framework/freeglut/freeglut_state.c create mode 100644 examples/opengl-framework/freeglut/freeglut_stroke_mono_roman.c create mode 100644 examples/opengl-framework/freeglut/freeglut_stroke_roman.c create mode 100644 examples/opengl-framework/freeglut/freeglut_structure.c create mode 100644 examples/opengl-framework/freeglut/freeglut_teapot.c create mode 100644 examples/opengl-framework/freeglut/freeglut_teapot_data.h create mode 100644 examples/opengl-framework/freeglut/freeglut_videoresize.c create mode 100644 examples/opengl-framework/freeglut/freeglut_window.c create mode 100644 examples/opengl-framework/freeglut/freeglut_xinput.c create mode 100644 examples/opengl-framework/freeglut/glut.h create mode 100644 examples/opengl-framework/src/Camera.cpp create mode 100644 examples/opengl-framework/src/Camera.h create mode 100644 examples/opengl-framework/src/FrameBufferObject.cpp create mode 100644 examples/opengl-framework/src/FrameBufferObject.h create mode 100644 examples/opengl-framework/src/GlutViewer.cpp create mode 100644 examples/opengl-framework/src/GlutViewer.h create mode 100644 examples/opengl-framework/src/Light.cpp create mode 100644 examples/opengl-framework/src/Light.h create mode 100644 examples/opengl-framework/src/Mesh.cpp create mode 100644 examples/opengl-framework/src/Mesh.h create mode 100644 examples/opengl-framework/src/MeshReaderWriter.cpp create mode 100644 examples/opengl-framework/src/MeshReaderWriter.h create mode 100644 examples/opengl-framework/src/Object3D.cpp create mode 100644 examples/opengl-framework/src/Object3D.h create mode 100644 examples/opengl-framework/src/Shader.cpp create mode 100644 examples/opengl-framework/src/Shader.h create mode 100644 examples/opengl-framework/src/Texture2D.cpp create mode 100644 examples/opengl-framework/src/Texture2D.h create mode 100644 examples/opengl-framework/src/TextureReaderWriter.cpp create mode 100644 examples/opengl-framework/src/TextureReaderWriter.h create mode 100644 examples/opengl-framework/src/VertexBufferObject.cpp create mode 100644 examples/opengl-framework/src/VertexBufferObject.h create mode 100644 examples/opengl-framework/src/definitions.h create mode 100644 examples/opengl-framework/src/maths/Color.h create mode 100644 examples/opengl-framework/src/maths/Matrix3.h create mode 100644 examples/opengl-framework/src/maths/Matrix4.h create mode 100644 examples/opengl-framework/src/maths/Vector2.h create mode 100644 examples/opengl-framework/src/maths/Vector3.h create mode 100644 examples/opengl-framework/src/maths/Vector4.h create mode 100644 examples/opengl-framework/src/openglframework.h create mode 100644 examples/opengl-framework/src/shaders/depth.frag create mode 100644 examples/opengl-framework/src/shaders/depth.vert create mode 100644 examples/opengl-framework/src/shaders/phong.frag create mode 100644 examples/opengl-framework/src/shaders/phong.vert diff --git a/CMakeLists.txt b/CMakeLists.txt index 68307462..7986b1ce 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -35,7 +35,7 @@ ADD_LIBRARY ( # If we need to compile the examples IF (COMPILE_EXAMPLES) - add_subdirectory(examples/fallingcubes) + add_subdirectory(examples/) ENDIF (COMPILE_EXAMPLES) # If we need to compile the tests diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt new file mode 100644 index 00000000..4f0cab67 --- /dev/null +++ b/examples/CMakeLists.txt @@ -0,0 +1,5 @@ +# Minimum cmake version required +CMAKE_MINIMUM_REQUIRED(VERSION 2.6) + +add_subdirectory(opengl-framework/) +add_subdirectory(fallingcubes/) \ No newline at end of file diff --git a/examples/fallingcubes/Box.cpp b/examples/fallingcubes/Box.cpp index 6aa3a675..d4169c9a 100644 --- a/examples/fallingcubes/Box.cpp +++ b/examples/fallingcubes/Box.cpp @@ -26,38 +26,155 @@ // Libraries #include "Box.h" -// Use the ReactPhysics3D namespace -using namespace reactphysics3d; +// Macros +#define MEMBER_OFFSET(s,m) ((char *)NULL + (offsetof(s,m))) + +// Initialize static variables +openglframework::VertexBufferObject Box::mVBOVertices(GL_ARRAY_BUFFER); +openglframework::VertexBufferObject Box::mVBOIndices(GL_ELEMENT_ARRAY_BUFFER); +bool Box::areVBOsCreated = false; +VertexData Box::mCubeVertices[8] = { + {openglframework::Vector3(1,1,1),openglframework::Vector3(1,1,1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(-1,1,1),openglframework::Vector3(-1,1,1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(-1,-1,1),openglframework::Vector3(-1,-1,1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(1,-1,1),openglframework::Vector3(1,-1,1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(1,-1,-1),openglframework::Vector3(1,-1,-1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(-1,-1,-1),openglframework::Vector3(-1,-1,-1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(-1,1,-1),openglframework::Vector3(-1,1,-1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(1,1,-1),openglframework::Vector3(1,1,-1),openglframework::Color(0,0,1,1)} +}; +GLuint Box::mCubeIndices[36] = { 0, 1, 2, + 2, 3, 0, + 7, 4, 5, + 5, 6, 7, + 6, 5, 2, + 2, 1, 6, + 7, 0, 3, + 3, 4, 7, + 7, 6, 1, + 1, 0, 7, + 3, 2, 5, + 5, 4, 3}; // Constructor -Box::Box(double size, RigidBody* rigidBody) { - mSize = size; - mRigidBody = rigidBody; +Box::Box(const openglframework::Vector3& size, const openglframework::Vector3 &position, + float mass, reactphysics3d::DynamicsWorld* dynamicsWorld) + : openglframework::Object3D() { + + // Initialize the size of the box + mSize[0] = size.x * 0.5f; + mSize[1] = size.y * 0.5f; + mSize[2] = size.z * 0.5f; + + // Compute the scaling matrix + mScalingMatrix = openglframework::Matrix4(mSize[0], 0, 0, 0, + 0, mSize[1], 0, 0, + 0, 0, mSize[2], 0, + 0, 0, 0, 1); + + // Initialize the position where the cube will be rendered + translateWorld(position); + + // Create the collision shape for the rigid body (box shape) + mCollisionShape = new rp3d::BoxShape(rp3d::Vector3(mSize[0], mSize[1], mSize[2])); + + // Compute the inertia tensor of the body using its collision shape + rp3d::Matrix3x3 inertiaTensor; + mCollisionShape->computeLocalInertiaTensor(inertiaTensor, mass); + + // Initial position and orientation of the rigid body + rp3d::Vector3 initPosition(position.x, position.y, position.z); + rp3d::Quaternion initOrientation = rp3d::Quaternion::identity(); + rp3d::Transform transform(initPosition, initOrientation); + + // Create a rigid body corresponding to the cube in the dynamics world + mRigidBody = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, mCollisionShape); + + // If the Vertex Buffer object has not been created yet + if (!areVBOsCreated) { + // Create the Vertex Buffer + createVBO(); + } } // Destructor Box::~Box() { + // Destroy the collision shape + delete mCollisionShape; } -// Draw the box -void Box::draw() const { +// Render the cube at the correct position and with the correct orientation +void Box::render(openglframework::Shader& shader) { + + // Bind the shader + shader.bind(); + + // Set the model to World matrix + shader.setMatrix4x4Uniform("modelToWorldMatrix", mTransformMatrix); + + // Bind the vertices VBO + mVBOVertices.bind(); + + // Enable the vertex, normal and color arrays + glEnableClientState(GL_VERTEX_ARRAY); + glEnableClientState(GL_COLOR_ARRAY); + glEnableClientState(GL_NORMAL_ARRAY); + + // Set the arrays pointers + glVertexPointer(3, GL_FLOAT, sizeof(VertexData), MEMBER_OFFSET(VertexData, position)); + glNormalPointer(GL_FLOAT, sizeof(VertexData), MEMBER_OFFSET(VertexData, normal)); + glColorPointer(3, GL_FLOAT, sizeof(VertexData), MEMBER_OFFSET(VertexData, color)); + + // Bind the indices VBO + mVBOIndices.bind(); + + // Draw the geometry of the box + glDrawElements(GL_TRIANGLES, 36, GL_UNSIGNED_INT, (char*)NULL); + + // Unbind the VBOs + mVBOVertices.unbind(); + mVBOIndices.unbind(); + + // Disable the arrays + glDisableClientState(GL_VERTEX_ARRAY); + glDisableClientState(GL_COLOR_ARRAY); + glDisableClientState(GL_NORMAL_ARRAY); + + // Unbind the shader + shader.unbind(); +} + +// Update the transform matrix of the box +void Box::updateTransform() { // Get the interpolated transform of the rigid body - Transform transform = mRigidBody->getInterpolatedTransform(); + rp3d::Transform transform = mRigidBody->getInterpolatedTransform(); - // Get the corresponding OpenGL matrix - float openGLMatrix[16]; - transform.getOpenGLMatrix(openGLMatrix); + // Compute the transform used for rendering the box + float matrix[16]; + transform.getOpenGLMatrix(matrix); + openglframework::Matrix4 newMatrix(matrix[0], matrix[4], matrix[8], matrix[12], + matrix[1], matrix[5], matrix[9], matrix[13], + matrix[2], matrix[6], matrix[10], matrix[14], + matrix[3], matrix[7], matrix[11], matrix[15]); - glPushMatrix(); - - // Multiply by the OpenGL transform matrix - glMultMatrixf(openGLMatrix); - - // Draw the cube - glutSolidCube(mSize); - - glPopMatrix(); + // Apply the scaling matrix to have the correct box dimensions + mTransformMatrix = newMatrix * mScalingMatrix; } +// Create the Vertex Buffer Objects used to render to box with OpenGL. +/// We create two VBOs (one for vertices and one for indices) to render all the boxes +/// in the simulation. +void Box::createVBO() { + + // Create the VBOs + mVBOVertices.create(); + mVBOIndices.create(); + + // Copy the data into the VBOs + mVBOVertices.copyDataIntoVBO(sizeof(mCubeVertices), mCubeVertices, GL_STATIC_DRAW); + mVBOIndices.copyDataIntoVBO(sizeof(mCubeIndices), mCubeIndices, GL_STATIC_DRAW); + + areVBOsCreated = true; +} diff --git a/examples/fallingcubes/Box.h b/examples/fallingcubes/Box.h index e8b9bd98..d287c3cb 100644 --- a/examples/fallingcubes/Box.h +++ b/examples/fallingcubes/Box.h @@ -23,60 +23,89 @@ * * ********************************************************************************/ -// Libraries -#include -#ifdef APPLE_OS - #include -#else - #include -#endif -#ifdef USE_FREEGLUT - #ifdef APPLE_OS - #include - #else - #include - #endif -#else - #ifdef APPLE_OS - #include - #else - #include - #endif -#endif - #ifndef BOX_H -#define BOX_H +#define BOX_H -/// This class represents a cube box -class Box { +// Libraries +#include "openglframework.h" +#include "reactphysics3d.h" - private: +// Structure VertexData +struct VertexData { - /// Size of the box - double mSize; + /// Vertex position + openglframework::Vector3 position; - /// Pointer to the rigid body associated to the box + /// Vertex normal + openglframework::Vector3 normal; + + // Vertex color + openglframework::Color color; +}; + +// Class Box +class Box : public openglframework::Object3D { + + private : + + // -------------------- Attributes -------------------- // + + /// Size of each side of the box + float mSize[3]; + + /// Rigid body used to simulate the dynamics of the box rp3d::RigidBody* mRigidBody; - public: + /// Collision shape of the rigid body + rp3d::BoxShape* mCollisionShape; + + /// Scaling matrix (applied to a cube to obtain the correct box dimensions) + openglframework::Matrix4 mScalingMatrix; + + /// Vertex Buffer Object for the vertices data used to render the box with OpenGL + static openglframework::VertexBufferObject mVBOVertices; + + /// Vertex Buffer Object for the indices used to render the box with OpenGL + static openglframework::VertexBufferObject mVBOIndices; + + /// Vertex data for each vertex of the cube (used to render the box) + static VertexData mCubeVertices[8]; + + /// Indices of the cube (used to render the box) + static GLuint mCubeIndices[36]; + + /// True if the VBOs have already been created + static bool areVBOsCreated; + + // -------------------- Methods -------------------- // + + /// Create a Vertex Buffer Object to render to box with OpenGL + static void createVBO(); + + public : + + // -------------------- Methods -------------------- // /// Constructor - Box(double size, rp3d::RigidBody* rigidBody); + Box(const openglframework::Vector3& size, const openglframework::Vector3& position, + float mass, rp3d::DynamicsWorld* dynamicsWorld); /// Destructor ~Box(); - /// Return the pointer to the rigid body - rp3d::RigidBody* getRigidBodyPointer() const; + /// Return a pointer to the rigid body of the box + rp3d::RigidBody* getRigidBody(); - /// Draw the box - void draw() const; + /// Update the transform matrix of the box + void updateTransform(); + + /// Render the cube at the correct position and with the correct orientation + void render(openglframework::Shader& shader); }; -// Return the pointer to the rigid body -inline rp3d::RigidBody* Box::getRigidBodyPointer() const { +// Return a pointer to the rigid body of the box +inline rp3d::RigidBody* Box::getRigidBody() { return mRigidBody; } #endif - diff --git a/examples/fallingcubes/CMakeLists.txt b/examples/fallingcubes/CMakeLists.txt old mode 100755 new mode 100644 index e716a9d7..4dcb291f --- a/examples/fallingcubes/CMakeLists.txt +++ b/examples/fallingcubes/CMakeLists.txt @@ -4,53 +4,14 @@ cmake_minimum_required(VERSION 2.6) # Project configuration PROJECT(FallingCubes) -# Find Glut or Freeglut -FIND_PACKAGE(GLUT) -if (NOT GLUT_FOUND) - # Find the Freeglut library - FIND_PATH(FREEGLUT_INCLUDE_DIR NAMES GL/freeglut.h - /usr/include/GL - ) - FIND_LIBRARY(FREEGLUT_LIBRARY NAMES freeglut glut) - INCLUDE(FindPackageHandleStandardArgs) - FIND_PACKAGE_HANDLE_STANDARD_ARGS(FREEGLUT DEFAULT_MSG FREEGLUT_LIBRARY FREEGLUT_INCLUDE_DIR) - IF(FREEGLUT_FOUND) - MESSAGE("Freeglut found") - SET(FREEGLUT_LIBRARIES ${FREEGLUT_LIBRARY}) - SET(FREEGLUT_INCLUDE_DIRS ${FREEGLUT_INCLUDE_DIR}) - else() - MESSAGE("Freeglut not found (required to build the examples)") - SET(FREEGLUT_LIBRARIES) - SET(FREEGLUT_INCLUDE_DIRS) - endif() -endif() - -# Find OpenGL -FIND_PACKAGE(OpenGL) -if(OPENGL_FOUND) - MESSAGE("OpenGL found") -else() - MESSAGE("OpenGL not found") -endif() +# Copy the shaders used for the demo into the build directory +FILE(COPY "../opengl-framework/src/shaders/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/shaders/") # Headers -if(GLUT_FOUND) - INCLUDE_DIRECTORIES(${REACTPHYSICS3D_SOURCE_DIR}/src ${GLUT_INCLUDE_DIR}) -elseif(FREEGLUT_FOUND) - INCLUDE_DIRECTORIES(${REACTPHYSICS3D_SOURCE_DIR}/src ${FREEGLUT_INCLUDE_DIRS}) -endif() - -# Definitions -if (FREEGLUT_FOUND) - ADD_DEFINITIONS(-DUSE_FREEGLUT) -endif() +INCLUDE_DIRECTORIES("../opengl-framework/src/") # Create the example executable using the # compiled reactphysics3d static library -ADD_EXECUTABLE(fallingcubes main.cpp Box.cpp Box.h) +ADD_EXECUTABLE(fallingcubes main.cpp Scene.cpp Scene.h Box.cpp Box.h) -if(GLUT_FOUND) - TARGET_LINK_LIBRARIES(fallingcubes reactphysics3d ${GLUT_LIBRARY} ${OPENGL_LIBRARY}) -elseif(FREEGLUT_FOUND) - TARGET_LINK_LIBRARIES(fallingcubes reactphysics3d ${FREEGLUT_LIBRARY} ${OPENGL_LIBRARY}) -endif() +TARGET_LINK_LIBRARIES(fallingcubes reactphysics3d openglframework) diff --git a/examples/fallingcubes/Scene.cpp b/examples/fallingcubes/Scene.cpp new file mode 100644 index 00000000..71a84821 --- /dev/null +++ b/examples/fallingcubes/Scene.cpp @@ -0,0 +1,169 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Scene.h" + +// Namespaces +using namespace openglframework; + +// Constructor +Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), + mPhongShader("shaders/phong.vert", + "shaders/phong.frag"){ + + // Move the light 0 + mLight0.translateWorld(Vector3(7, 15, 15)); + + // Compute the radius and the center of the scene + float radius = 10.0f; + openglframework::Vector3 center(0, 5, 0); + + // Set the center of the scene + mViewer->setScenePosition(center, radius); + + // Gravity vector in the dynamics world + rp3d::Vector3 gravity(0, -9.81, 0); + + // Time step for the physics simulation + rp3d::decimal timeStep = 1.0f / 80.0f; + + // Create the dynamics world for the physics simulation + mDynamicsWorld = new rp3d::DynamicsWorld(gravity, timeStep); + + // Create all the cubes of the scene + for (int i=0; igetRigidBody()->setIsMotionEnabled(true); + + // Set the bouncing factor of the box + cube->getRigidBody()->setRestitution(0.4); + + // Add the box the list of box in the scene + mBoxes.push_back(cube); + } + + // Create the floor + openglframework::Vector3 floorPosition(0, 0, 0); + mFloor = new Box(FLOOR_SIZE, floorPosition, FLOOR_MASS, mDynamicsWorld); + + // The floor must be a non-moving rigid body + mFloor->getRigidBody()->setIsMotionEnabled(false); + + // Set the bouncing factor of the floor + mFloor->getRigidBody()->setRestitution(0.3); + + // Start the simulation + mDynamicsWorld->start(); +} + +// Destructor +Scene::~Scene() { + + // Stop the physics simulation + mDynamicsWorld->stop(); + + // Destroy the shader + mPhongShader.destroy(); + + // Destroy all the cubes of the scene + for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + + // Destroy the corresponding rigid body from the dynamics world + mDynamicsWorld->destroyRigidBody((*it)->getRigidBody()); + + // Destroy the cube + delete (*it); + } + + // Destroy the rigid body of the floor + mDynamicsWorld->destroyRigidBody(mFloor->getRigidBody()); + + // Destroy the floor + delete mFloor; + + // Destroy the dynamics world + delete mDynamicsWorld; +} + +// Take a step for the simulation +void Scene::simulate() { + + // Take a simulation step + mDynamicsWorld->update(); + + // Update the position and orientation of the boxes + for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + + // Update the transform used for the rendering + (*it)->updateTransform(); + } + + mFloor->updateTransform(); +} + +// Render the scene +void Scene::render() { + + glEnable(GL_DEPTH_TEST); + glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); + glDisable(GL_CULL_FACE); + + // Bind the shader + mPhongShader.bind(); + + // Set the variables of the shader + const Camera& camera = mViewer->getCamera(); + Matrix4 matrixIdentity; + matrixIdentity.setToIdentity(); + mPhongShader.setVector3Uniform("cameraWorldPosition", mViewer->getCamera().getOrigin()); + mPhongShader.setMatrix4x4Uniform("worldToCameraMatrix", camera.getTransformMatrix().getInverse()); + mPhongShader.setMatrix4x4Uniform("projectionMatrix", camera.getProjectionMatrix()); + mPhongShader.setVector3Uniform("lightWorldPosition", mLight0.getOrigin()); + mPhongShader.setVector3Uniform("lightAmbientColor", Vector3(0.3f, 0.3f, 0.3f)); + Color& diffCol = mLight0.getDiffuseColor(); + Color& specCol = mLight0.getSpecularColor(); + mPhongShader.setVector3Uniform("lightDiffuseColor", Vector3(diffCol.r, diffCol.g, diffCol.b)); + mPhongShader.setVector3Uniform("lightSpecularColor", Vector3(specCol.r, specCol.g, specCol.b)); + mPhongShader.setFloatUniform("shininess", 60.0f); + + // Render all the cubes of the scene + for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + (*it)->render(mPhongShader); + } + + // Render the floor + mFloor->render(mPhongShader); + + // Unbind the shader + mPhongShader.unbind(); +} diff --git a/examples/fallingcubes/Scene.h b/examples/fallingcubes/Scene.h new file mode 100644 index 00000000..a865d8e0 --- /dev/null +++ b/examples/fallingcubes/Scene.h @@ -0,0 +1,83 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 SCENE_H +#define SCENE_H + +// Libraries +#include "openglframework.h" +#include "reactphysics3d.h" +#include "Box.h" + +// Constants +const int NB_BOXES = 10; // Number of boxes in the scene +const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters +const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters +const float BOX_MASS = 1.0f; // Box mass in kilograms +const float FLOOR_MASS = 100.0f; // Floor mass in kilograms + +// Class Scene +class Scene { + + private : + + // -------------------- Attributes -------------------- // + + // Pointer to the viewer + openglframework::GlutViewer* mViewer; + + // Light 0 + openglframework::Light mLight0; + + // Phong shader + openglframework::Shader mPhongShader; + + /// All the boxes of the scene + std::vector mBoxes; + + /// Box for the floor + Box* mFloor; + + /// Dynamics world used for the physics simulation + rp3d::DynamicsWorld* mDynamicsWorld; + + public: + + // -------------------- Methods -------------------- // + + /// Constructor + Scene(openglframework::GlutViewer* viewer); + + /// Destructor + ~Scene(); + + /// Take a step for the simulation + void simulate(); + + /// Render the scene + void render(); +}; + +#endif diff --git a/examples/fallingcubes/main.cpp b/examples/fallingcubes/main.cpp index c3f05100..6dbfc7f4 100644 --- a/examples/fallingcubes/main.cpp +++ b/examples/fallingcubes/main.cpp @@ -24,225 +24,166 @@ ********************************************************************************/ // Libraries -#include -#include -#include "Box.h" +#include "Scene.h" +#include -// Prototypes -void init(); -void display(); +// Declarations void simulate(); -void clean(); -void reshape(int w, int h); +void display(); +void displayFPS(); +void computeFPS(); +void reshape(int width, int height); +void mouseButton(int button, int state, int x, int y); +void mouseMotion(int x, int y); +void keyboardSpecial(int key, int x, int y); +void init(); -// Use the ReactPhysics3D namespace -using namespace reactphysics3d; - -// Constants -const double FLOOR_SIZE = 20; -const double FLOOR_THICKNESS = 0.02; +// Namespaces +using namespace openglframework; // Global variables -DynamicsWorld* dynamicsWorld; // Dynamics world -Box* boxes[2]; // Falling boxes -BoxShape* collisionShapeBox1; // Collision shape of the first box -BoxShape* collisionShapeBox2; // Collision shape of the second box -BoxShape* collisionShapeFloor; // Collision shape of the floor -RigidBody* floorRigidBody; // Rigid body corresponding the floor - - -// Simulation function -void simulate() { - - // Update the physics simulation - dynamicsWorld->update(); - - // Display the scene - display(); -} +GlutViewer* viewer; +Scene* scene; +int fps; +int nbFrames; +int currentTime; +int previousTime; +int width, height; // Main function int main(int argc, char** argv) { - // Initialize GLUT - glutInit(&argc, argv); - glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGBA | GLUT_DEPTH); - glutInitWindowSize(800, 600); - glutInitWindowPosition(100, 100); - glutCreateWindow("ReactPhysics3D Example - Falling Cubes"); + // Create and initialize the Viewer + viewer = new GlutViewer(); + Vector2 windowsSize = Vector2(800, 600); + Vector2 windowsPosition = Vector2(100, 100); + width = windowsSize.x; + height = windowsSize.y; + bool initOK = viewer->init(argc, argv, "ReactPhysics3D Examples - Falling Cubes", windowsSize, windowsPosition); + if (!initOK) return 1; + + // Create the scene + scene = new Scene(viewer); init(); + nbFrames = 0; + + // Glut Idle function that is continuously called glutIdleFunc(simulate); glutDisplayFunc(display); glutReshapeFunc(reshape); + glutMouseFunc(mouseButton); + glutMotionFunc(mouseMotion); + glutSpecialFunc(keyboardSpecial); + + // Glut main looop glutMainLoop(); - // Stop the physics simulation - dynamicsWorld->stop(); + delete viewer; + delete scene; - clean(); - return 0; } -// Initialization function -void init() { +// Simulate function +void simulate() { - glClearColor(0.0, 0.0, 0.0, 0.0); + // Physics simulation + scene->simulate(); - // Light - glShadeModel(GL_SMOOTH); - GLfloat light_position[] = {5.0f, 5.0f, 5.0f, 1.0f}; - glLightfv(GL_LIGHT0, GL_POSITION, light_position); + computeFPS(); - glEnable(GL_LIGHTING); - glEnable(GL_LIGHT0); - glEnable(GL_DEPTH_TEST); - - // Gravity vector of the physics world - Vector3 gravity(0.0, -9.81, 0.0); - - // Timestep of the simulation - decimal timeStep = 1.0/60.0; - - // Create the dynamics world - dynamicsWorld = new DynamicsWorld(gravity, timeStep); - - // --- Create a falling box with a size of 1m and weight of 3kg --- // - - float size = 1.0f; - - // Initial position and orientation of the box - Vector3 positionBox1(-2.0f, 7.0f, 0.0f); - Quaternion orientationBox1(0.3, 1.0, 0.8, 0.0); - Transform initTransform(positionBox1, orientationBox1); - - // Create a box collision shape for the box (used for collision detection) - collisionShapeBox1 = new BoxShape(Vector3(size/2.0f, size/2.0f, size/2.0f)); - - // Compute the inertia tensor of the box using the collision shape - Matrix3x3 inertiaTensorBox1; - float massBox1 = 3.0f; - collisionShapeBox1->computeLocalInertiaTensor(inertiaTensorBox1, massBox1); - - // Create the rigid body associated with the box in the dynamics world - RigidBody* rigidBody = dynamicsWorld->createRigidBody(initTransform, massBox1, - inertiaTensorBox1, collisionShapeBox1); - - // Set the contact velocity restitution factor of the rigid body - rigidBody->setRestitution(0.5f); - - // Create the box object (used for display) - boxes[0] = new Box(size, rigidBody); - - // --- Create a second falling box with a size of 1.5m and weight of 4.5kg --- // - - size = 1.5; - - // Initial position and orientation of the box - Vector3 positionBox2(2.0, 4.0, 0.0); - Quaternion orientationBox2(1.0, 1.0, 0.5, 0.0); - Transform initTransform2(positionBox2, orientationBox2); - - // Create a box collision shape for the box (used for collision detection) - collisionShapeBox2 = new BoxShape(Vector3(size/2.0f, size/2.0f, size/2.0f)); - - // Compute the inertia tensor using the collision shape - Matrix3x3 inertiaTensorBox2; - float massBox2 = 4.5f; - collisionShapeBox2->computeLocalInertiaTensor(inertiaTensorBox2, massBox2); - - // Create the rigid body associated with the box in the dynamcis world - RigidBody* rigidBody2 = dynamicsWorld->createRigidBody(initTransform2, massBox2, - inertiaTensorBox2, collisionShapeBox2); - - // Set the contact velocity restitution factor of the rigid body - rigidBody2->setRestitution(0.5); - - // Create the box object (used for display) - boxes[1] = new Box(size, rigidBody2); - - // --- Create the rigid body corresponding to the floor --- // - - // Initial position and orientation of the floor - Vector3 positionFloor(0.0, 0.0, 0.0); - Quaternion orientationFloor(0.0, 1.0, 0.0, 0.0); - Transform initTransformFloor(positionFloor, orientationFloor); - - // Create a box collision shape for the floor (used for collision detection) - collisionShapeFloor = new BoxShape(Vector3(FLOOR_SIZE, FLOOR_THICKNESS, FLOOR_SIZE)); - - // Compute the inertia tensor of the floor using the collision shape - float massFloor = 100.0f; - rp3d::Matrix3x3 inertiaTensorFloor; - collisionShapeFloor->computeLocalInertiaTensor(inertiaTensorFloor, massFloor); - - // Create the rigid body associated with the floor in the dynamcis world - floorRigidBody = dynamicsWorld->createRigidBody(initTransformFloor, massFloor, - inertiaTensorFloor, collisionShapeFloor); - - // The floor is a rigid body that cannot move - floorRigidBody->setIsMotionEnabled(false); - - // Set the contact velocity restitution factor of the floor - floorRigidBody->setRestitution(0.5); - - // Start the dynamics simulation - dynamicsWorld->start(); + // Ask GLUT to render the scene + glutPostRedisplay (); } -// Display function -void display() { +// Initialization +void init() { - glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); - - // Display each falling box of the scene - for (int i=0; i<2; i++) { - boxes[i]->draw(); - } - - // Display the plane for the floor - glBegin(GL_POLYGON); - glNormal3f(0.0, 1.0, 0.0); - glVertex3f(-FLOOR_SIZE/2, 0.0, -FLOOR_SIZE/2); - glVertex3f(-FLOOR_SIZE/2, 0.0, FLOOR_SIZE/2); - glVertex3f(FLOOR_SIZE/2, 0.0, FLOOR_SIZE/2); - glVertex3f(FLOOR_SIZE/2, 0.0, -FLOOR_SIZE/2); - glEnd(); - - glutSwapBuffers(); + // Define the background color (black) + glClearColor(0.0, 0.0, 0.0, 1.0); } // Reshape function -void reshape(int w, int h) { - float ratio = ((float)w / h); +void reshape(int newWidth, int newHeight) { + viewer->reshape(newWidth, newHeight); + width = newWidth; + height = newHeight; +} + +// Called when a mouse button event occurs +void mouseButton(int button, int state, int x, int y) { + viewer->mouseButtonEvent(button, state, x, y); +} + +// Called when a mouse motion event occurs +void mouseMotion(int x, int y) { + viewer->mouseMotionEvent(x, y); +} + +// Called when the user hits a special key on the keyboard +void keyboardSpecial(int key, int x, int y) { + /* + if(key=='0') + exit(0); + if(key== GLUT_KEY_RIGHT) { + */ +} + +// Display the scene +void display() { + + // Render the scene + scene->render(); + + // Display the FPS + displayFPS(); + + // Swap the buffers + glutSwapBuffers(); + + // Check the OpenGL errors + GlutViewer::checkOpenGLErrors(); +} + +// Compute the FPS +void computeFPS() { + nbFrames++; + + // Get the number of milliseconds since glutInit called + currentTime = glutGet(GLUT_ELAPSED_TIME); + + // Calculate time passed + int timeInterval = currentTime - previousTime; + + // Update the FPS counter each second + if(timeInterval > 1000){ + + // calculate the number of frames per second + fps = nbFrames / (timeInterval / 1000.0f); + + // Set time + previousTime = currentTime; + + // Reset frame count + nbFrames = 0; + } +} + +// Display the FPS +void displayFPS() { + glMatrixMode(GL_PROJECTION); glLoadIdentity(); - glViewport(0, 0, w, h); - gluPerspective(45, ratio,1,1000); + glOrtho(0, width, height, 0, -1, 1); + glMatrixMode(GL_MODELVIEW); glLoadIdentity(); - gluLookAt(20.0, 4.0, 20.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0); -} - -// Clean the memory allocation -void clean() { - - // Destroy the rigid bodies from the dynamics world - dynamicsWorld->destroyRigidBody(boxes[0]->getRigidBodyPointer()); - dynamicsWorld->destroyRigidBody(boxes[1]->getRigidBodyPointer()); - dynamicsWorld->destroyRigidBody(floorRigidBody); - - // Destroy the dynamics world - delete dynamicsWorld; - - // Destroy the boxes - delete boxes[0]; - delete boxes[1]; - - // Destroy the collision shapes - delete collisionShapeBox1; - delete collisionShapeBox2; - delete collisionShapeFloor; + + glRasterPos2i(10, 20); + glColor4f(1.0f, 1.0f, 1.0f, 1.0f); + std::stringstream ss; + ss << "FPS : " << fps; + glutBitmapString(GLUT_BITMAP_HELVETICA_12, (const unsigned char*)ss.str().c_str()); } diff --git a/examples/opengl-framework/CMakeLists.txt b/examples/opengl-framework/CMakeLists.txt new file mode 100644 index 00000000..82cba467 --- /dev/null +++ b/examples/opengl-framework/CMakeLists.txt @@ -0,0 +1,55 @@ +# Minimum cmake version required +CMAKE_MINIMUM_REQUIRED(VERSION 2.6) + +# Project configuration +PROJECT(OPENGLFRAMEWORK) + +# Where to find the module to find special packages/libraries +set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/") + +# Find OpenGL +FIND_PACKAGE(OpenGL REQUIRED) +if(OPENGL_FOUND) + MESSAGE("OpenGL found") +else() + MESSAGE("OpenGL not found") +endif() + +# Find the GLEW library +FIND_PACKAGE(GLEW REQUIRED) +if(GLEW_FOUND) + MESSAGE("GLEW found") +else() + MESSAGE("GLEW not found") +endif() + +# Find the LIBJPEG library +FIND_PACKAGE(JPEG REQUIRED) +if(JPEG_FOUND) + MESSAGE("LIBJPEG found") +else() + MESSAGE("LIBJPEG not found") +endif() + +# Freeglut +add_subdirectory(freeglut) + +# Headers +INCLUDE_DIRECTORIES(src freeglut ${JPEG_INCLUDE_DIR}) + +# Library configuration +file ( + GLOB_RECURSE + OPENGLFRAMEWORK_SOURCES_FILES + src/* +) + + +# Require the opengl-framework code to be compiled in a static library +ADD_LIBRARY ( + openglframework + STATIC + ${OPENGLFRAMEWORK_SOURCES_FILES} +) + +TARGET_LINK_LIBRARIES(openglframework ${GLEW_LIBRARIES} ${OPENGL_LIBRARY} freeglut_static) diff --git a/examples/opengl-framework/freeglut/CMakeLists.txt b/examples/opengl-framework/freeglut/CMakeLists.txt new file mode 100644 index 00000000..bfa701ed --- /dev/null +++ b/examples/opengl-framework/freeglut/CMakeLists.txt @@ -0,0 +1,24 @@ +add_definitions( -DFREEGLUT_EXPORTS -DFREEGLUT_STATIC -D_CRT_SECURE_NO_WARNINGS ) + +if(APPLE) + include_directories( /usr/X11/include ) +endif(APPLE) + +if(UNIX) + add_definitions( -D__unix__ -DHAVE_FCNTL_H -DHAVE_GETTIMEOFDAY ) +endif(UNIX) + +file ( + GLOB_RECURSE + FREEGLUT_SOURCES + ./* +) + +include_directories ( + ${OPENGL_INCLUDE_DIR} + . +) + +add_library(freeglut_static + ${FREEGLUT_SOURCES} +) diff --git a/examples/opengl-framework/freeglut/COPYING.txt b/examples/opengl-framework/freeglut/COPYING.txt new file mode 100644 index 00000000..fc36ad99 --- /dev/null +++ b/examples/opengl-framework/freeglut/COPYING.txt @@ -0,0 +1,27 @@ + + Freeglut Copyright + ------------------ + + Freeglut code without an explicit copyright is covered by the following + copyright: + + Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + 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 or substantial portions of the Software. + + 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 + PAWEL W. OLSZTA 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. + + Except as contained in this notice, the name of Pawel W. Olszta shall not be + used in advertising or otherwise to promote the sale, use or other dealings + in this Software without prior written authorization from Pawel W. Olszta. diff --git a/examples/opengl-framework/freeglut/GL/freeglut.h b/examples/opengl-framework/freeglut/GL/freeglut.h new file mode 100644 index 00000000..0e6f8c6a --- /dev/null +++ b/examples/opengl-framework/freeglut/GL/freeglut.h @@ -0,0 +1,22 @@ +#ifndef __FREEGLUT_H__ +#define __FREEGLUT_H__ + +/* + * freeglut.h + * + * The freeglut library include file + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include "freeglut_std.h" +#include "freeglut_ext.h" + +/*** END OF FILE ***/ + +#endif /* __FREEGLUT_H__ */ diff --git a/examples/opengl-framework/freeglut/GL/freeglut_ext.h b/examples/opengl-framework/freeglut/GL/freeglut_ext.h new file mode 100644 index 00000000..6bf84b86 --- /dev/null +++ b/examples/opengl-framework/freeglut/GL/freeglut_ext.h @@ -0,0 +1,236 @@ +#ifndef __FREEGLUT_EXT_H__ +#define __FREEGLUT_EXT_H__ + +/* + * freeglut_ext.h + * + * The non-GLUT-compatible extensions to the freeglut library include file + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 2 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#ifdef __cplusplus + extern "C" { +#endif + +/* + * Additional GLUT Key definitions for the Special key function + */ +#define GLUT_KEY_NUM_LOCK 0x006D +#define GLUT_KEY_BEGIN 0x006E +#define GLUT_KEY_DELETE 0x006F +#define GLUT_KEY_SHIFT_L 0x0070 +#define GLUT_KEY_SHIFT_R 0x0071 +#define GLUT_KEY_CTRL_L 0x0072 +#define GLUT_KEY_CTRL_R 0x0073 +#define GLUT_KEY_ALT_L 0x0074 +#define GLUT_KEY_ALT_R 0x0075 + +/* + * GLUT API Extension macro definitions -- behaviour when the user clicks on an "x" to close a window + */ +#define GLUT_ACTION_EXIT 0 +#define GLUT_ACTION_GLUTMAINLOOP_RETURNS 1 +#define GLUT_ACTION_CONTINUE_EXECUTION 2 + +/* + * Create a new rendering context when the user opens a new window? + */ +#define GLUT_CREATE_NEW_CONTEXT 0 +#define GLUT_USE_CURRENT_CONTEXT 1 + +/* + * Direct/Indirect rendering context options (has meaning only in Unix/X11) + */ +#define GLUT_FORCE_INDIRECT_CONTEXT 0 +#define GLUT_ALLOW_DIRECT_CONTEXT 1 +#define GLUT_TRY_DIRECT_CONTEXT 2 +#define GLUT_FORCE_DIRECT_CONTEXT 3 + +/* + * GLUT API Extension macro definitions -- the glutGet parameters + */ +#define GLUT_INIT_STATE 0x007C + +#define GLUT_ACTION_ON_WINDOW_CLOSE 0x01F9 + +#define GLUT_WINDOW_BORDER_WIDTH 0x01FA +#define GLUT_WINDOW_HEADER_HEIGHT 0x01FB + +#define GLUT_VERSION 0x01FC + +#define GLUT_RENDERING_CONTEXT 0x01FD +#define GLUT_DIRECT_RENDERING 0x01FE + +#define GLUT_FULL_SCREEN 0x01FF + +/* + * New tokens for glutInitDisplayMode. + * Only one GLUT_AUXn bit may be used at a time. + * Value 0x0400 is defined in OpenGLUT. + */ +#define GLUT_AUX 0x1000 + +#define GLUT_AUX1 0x1000 +#define GLUT_AUX2 0x2000 +#define GLUT_AUX3 0x4000 +#define GLUT_AUX4 0x8000 + +/* + * Context-related flags, see freeglut_state.c + */ +#define GLUT_INIT_MAJOR_VERSION 0x0200 +#define GLUT_INIT_MINOR_VERSION 0x0201 +#define GLUT_INIT_FLAGS 0x0202 +#define GLUT_INIT_PROFILE 0x0203 + +/* + * Flags for glutInitContextFlags, see freeglut_init.c + */ +#define GLUT_DEBUG 0x0001 +#define GLUT_FORWARD_COMPATIBLE 0x0002 + + +/* + * Flags for glutInitContextProfile, see freeglut_init.c + */ +#define GLUT_CORE_PROFILE 0x0001 +#define GLUT_COMPATIBILITY_PROFILE 0x0002 + +/* + * Process loop function, see freeglut_main.c + */ +FGAPI void FGAPIENTRY glutMainLoopEvent( void ); +FGAPI void FGAPIENTRY glutLeaveMainLoop( void ); +FGAPI void FGAPIENTRY glutExit ( void ); + +/* + * Window management functions, see freeglut_window.c + */ +FGAPI void FGAPIENTRY glutFullScreenToggle( void ); +FGAPI void FGAPIENTRY glutLeaveFullScreen( void ); + +/* + * Window-specific callback functions, see freeglut_callbacks.c + */ +FGAPI void FGAPIENTRY glutMouseWheelFunc( void (* callback)( int, int, int, int ) ); +FGAPI void FGAPIENTRY glutCloseFunc( void (* callback)( void ) ); +FGAPI void FGAPIENTRY glutWMCloseFunc( void (* callback)( void ) ); +/* A. Donev: Also a destruction callback for menus */ +FGAPI void FGAPIENTRY glutMenuDestroyFunc( void (* callback)( void ) ); + +/* + * State setting and retrieval functions, see freeglut_state.c + */ +FGAPI void FGAPIENTRY glutSetOption ( GLenum option_flag, int value ); +FGAPI int * FGAPIENTRY glutGetModeValues(GLenum mode, int * size); +/* A.Donev: User-data manipulation */ +FGAPI void* FGAPIENTRY glutGetWindowData( void ); +FGAPI void FGAPIENTRY glutSetWindowData(void* data); +FGAPI void* FGAPIENTRY glutGetMenuData( void ); +FGAPI void FGAPIENTRY glutSetMenuData(void* data); + +/* + * Font stuff, see freeglut_font.c + */ +FGAPI int FGAPIENTRY glutBitmapHeight( void* font ); +FGAPI GLfloat FGAPIENTRY glutStrokeHeight( void* font ); +FGAPI void FGAPIENTRY glutBitmapString( void* font, const unsigned char *string ); +FGAPI void FGAPIENTRY glutStrokeString( void* font, const unsigned char *string ); + +/* + * Geometry functions, see freeglut_geometry.c + */ +FGAPI void FGAPIENTRY glutWireRhombicDodecahedron( void ); +FGAPI void FGAPIENTRY glutSolidRhombicDodecahedron( void ); +FGAPI void FGAPIENTRY glutWireSierpinskiSponge ( int num_levels, GLdouble offset[3], GLdouble scale ); +FGAPI void FGAPIENTRY glutSolidSierpinskiSponge ( int num_levels, GLdouble offset[3], GLdouble scale ); +FGAPI void FGAPIENTRY glutWireCylinder( GLdouble radius, GLdouble height, GLint slices, GLint stacks); +FGAPI void FGAPIENTRY glutSolidCylinder( GLdouble radius, GLdouble height, GLint slices, GLint stacks); + +/* + * Extension functions, see freeglut_ext.c + */ +typedef void (*GLUTproc)(); +FGAPI GLUTproc FGAPIENTRY glutGetProcAddress( const char *procName ); + +/* + * Multi-touch/multi-pointer extensions + */ + +#define GLUT_HAS_MULTI 1 + +FGAPI void FGAPIENTRY glutMultiEntryFunc( void (* callback)( int, int ) ); +FGAPI void FGAPIENTRY glutMultiButtonFunc( void (* callback)( int, int, int, int, int ) ); +FGAPI void FGAPIENTRY glutMultiMotionFunc( void (* callback)( int, int, int ) ); +FGAPI void FGAPIENTRY glutMultiPassiveFunc( void (* callback)( int, int, int ) ); + +/* + * Joystick functions, see freeglut_joystick.c + */ +/* USE OF THESE FUNCTIONS IS DEPRECATED !!!!! */ +/* If you have a serious need for these functions in your application, please either + * contact the "freeglut" developer community at freeglut-developer@lists.sourceforge.net, + * switch to the OpenGLUT library, or else port your joystick functionality over to PLIB's + * "js" library. + */ +int glutJoystickGetNumAxes( int ident ); +int glutJoystickGetNumButtons( int ident ); +int glutJoystickNotWorking( int ident ); +float glutJoystickGetDeadBand( int ident, int axis ); +void glutJoystickSetDeadBand( int ident, int axis, float db ); +float glutJoystickGetSaturation( int ident, int axis ); +void glutJoystickSetSaturation( int ident, int axis, float st ); +void glutJoystickSetMinRange( int ident, float *axes ); +void glutJoystickSetMaxRange( int ident, float *axes ); +void glutJoystickSetCenter( int ident, float *axes ); +void glutJoystickGetMinRange( int ident, float *axes ); +void glutJoystickGetMaxRange( int ident, float *axes ); +void glutJoystickGetCenter( int ident, float *axes ); + +/* + * Initialization functions, see freeglut_init.c + */ +FGAPI void FGAPIENTRY glutInitContextVersion( int majorVersion, int minorVersion ); +FGAPI void FGAPIENTRY glutInitContextFlags( int flags ); +FGAPI void FGAPIENTRY glutInitContextProfile( int profile ); + +/* to get the typedef for va_list */ +#include + +FGAPI void FGAPIENTRY glutInitErrorFunc( void (* vError)( const char *fmt, va_list ap ) ); +FGAPI void FGAPIENTRY glutInitWarningFunc( void (* vWarning)( const char *fmt, va_list ap ) ); + +/* + * GLUT API macro definitions -- the display mode definitions + */ +#define GLUT_CAPTIONLESS 0x0400 +#define GLUT_BORDERLESS 0x0800 +#define GLUT_SRGB 0x1000 + +#ifdef __cplusplus + } +#endif + +/*** END OF FILE ***/ + +#endif /* __FREEGLUT_EXT_H__ */ diff --git a/examples/opengl-framework/freeglut/GL/freeglut_std.h b/examples/opengl-framework/freeglut/GL/freeglut_std.h new file mode 100644 index 00000000..ba1b7165 --- /dev/null +++ b/examples/opengl-framework/freeglut/GL/freeglut_std.h @@ -0,0 +1,628 @@ +#ifndef __FREEGLUT_STD_H__ +#define __FREEGLUT_STD_H__ + +/* + * freeglut_std.h + * + * The GLUT-compatible part of the freeglut library include file + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 2 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#ifdef __cplusplus + extern "C" { +#endif + +/* + * Under windows, we have to differentiate between static and dynamic libraries + */ +#ifdef _WIN32 +/* #pragma may not be supported by some compilers. + * Discussion by FreeGLUT developers suggests that + * Visual C++ specific code involving pragmas may + * need to move to a separate header. 24th Dec 2003 + */ + +/* Define FREEGLUT_LIB_PRAGMAS to 1 to include library + * pragmas or to 0 to exclude library pragmas. + * The default behavior depends on the compiler/platform. + */ +# ifndef FREEGLUT_LIB_PRAGMAS +# if ( defined(_MSC_VER) || defined(__WATCOMC__) ) && !defined(_WIN32_WCE) +# define FREEGLUT_LIB_PRAGMAS 1 +# else +# define FREEGLUT_LIB_PRAGMAS 0 +# endif +# endif + +# ifndef WIN32_LEAN_AND_MEAN +# define WIN32_LEAN_AND_MEAN 1 +# endif +# ifndef NOMINMAX +# define NOMINMAX +# endif +# include + +/* Windows static library */ +# ifdef FREEGLUT_STATIC + +# define FGAPI +# define FGAPIENTRY + + /* Link with Win32 static freeglut lib */ +# if FREEGLUT_LIB_PRAGMAS +# pragma comment (lib, "freeglut_static.lib") +# endif + +/* Windows shared library (DLL) */ +# else + +# define FGAPIENTRY __stdcall +# if defined(FREEGLUT_EXPORTS) +# define FGAPI __declspec(dllexport) +# else +# define FGAPI __declspec(dllimport) + + /* Link with Win32 shared freeglut lib */ +# if FREEGLUT_LIB_PRAGMAS +# pragma comment (lib, "freeglut.lib") +# endif + +# endif + +# endif + +/* Drag in other Windows libraries as required by FreeGLUT */ +# if FREEGLUT_LIB_PRAGMAS +# pragma comment (lib, "glu32.lib") /* link OpenGL Utility lib */ +# pragma comment (lib, "opengl32.lib") /* link Microsoft OpenGL lib */ +# pragma comment (lib, "gdi32.lib") /* link Windows GDI lib */ +# pragma comment (lib, "winmm.lib") /* link Windows MultiMedia lib */ +# pragma comment (lib, "user32.lib") /* link Windows user lib */ +# endif + +#else + +/* Non-Windows definition of FGAPI and FGAPIENTRY */ +# define FGAPI +# define FGAPIENTRY + +#endif + +/* + * The freeglut and GLUT API versions + */ +#define FREEGLUT 1 +#define GLUT_API_VERSION 4 +#define FREEGLUT_VERSION_2_0 1 +#define GLUT_XLIB_IMPLEMENTATION 13 + +/* + * Always include OpenGL and GLU headers + */ +#include +#include + +/* + * GLUT API macro definitions -- the special key codes: + */ +#define GLUT_KEY_F1 0x0001 +#define GLUT_KEY_F2 0x0002 +#define GLUT_KEY_F3 0x0003 +#define GLUT_KEY_F4 0x0004 +#define GLUT_KEY_F5 0x0005 +#define GLUT_KEY_F6 0x0006 +#define GLUT_KEY_F7 0x0007 +#define GLUT_KEY_F8 0x0008 +#define GLUT_KEY_F9 0x0009 +#define GLUT_KEY_F10 0x000A +#define GLUT_KEY_F11 0x000B +#define GLUT_KEY_F12 0x000C +#define GLUT_KEY_LEFT 0x0064 +#define GLUT_KEY_UP 0x0065 +#define GLUT_KEY_RIGHT 0x0066 +#define GLUT_KEY_DOWN 0x0067 +#define GLUT_KEY_PAGE_UP 0x0068 +#define GLUT_KEY_PAGE_DOWN 0x0069 +#define GLUT_KEY_HOME 0x006A +#define GLUT_KEY_END 0x006B +#define GLUT_KEY_INSERT 0x006C + +/* + * GLUT API macro definitions -- mouse state definitions + */ +#define GLUT_LEFT_BUTTON 0x0000 +#define GLUT_MIDDLE_BUTTON 0x0001 +#define GLUT_RIGHT_BUTTON 0x0002 +#define GLUT_DOWN 0x0000 +#define GLUT_UP 0x0001 +#define GLUT_LEFT 0x0000 +#define GLUT_ENTERED 0x0001 + +/* + * GLUT API macro definitions -- the display mode definitions + */ +#define GLUT_RGB 0x0000 +#define GLUT_RGBA 0x0000 +#define GLUT_INDEX 0x0001 +#define GLUT_SINGLE 0x0000 +#define GLUT_DOUBLE 0x0002 +#define GLUT_ACCUM 0x0004 +#define GLUT_ALPHA 0x0008 +#define GLUT_DEPTH 0x0010 +#define GLUT_STENCIL 0x0020 +#define GLUT_MULTISAMPLE 0x0080 +#define GLUT_STEREO 0x0100 +#define GLUT_LUMINANCE 0x0200 + +/* + * GLUT API macro definitions -- windows and menu related definitions + */ +#define GLUT_MENU_NOT_IN_USE 0x0000 +#define GLUT_MENU_IN_USE 0x0001 +#define GLUT_NOT_VISIBLE 0x0000 +#define GLUT_VISIBLE 0x0001 +#define GLUT_HIDDEN 0x0000 +#define GLUT_FULLY_RETAINED 0x0001 +#define GLUT_PARTIALLY_RETAINED 0x0002 +#define GLUT_FULLY_COVERED 0x0003 + +/* + * GLUT API macro definitions -- fonts definitions + * + * Steve Baker suggested to make it binary compatible with GLUT: + */ +#if defined(_MSC_VER) || defined(__CYGWIN__) || defined(__MINGW32__) || defined(__WATCOMC__) +# define GLUT_STROKE_ROMAN ((void *)0x0000) +# define GLUT_STROKE_MONO_ROMAN ((void *)0x0001) +# define GLUT_BITMAP_9_BY_15 ((void *)0x0002) +# define GLUT_BITMAP_8_BY_13 ((void *)0x0003) +# define GLUT_BITMAP_TIMES_ROMAN_10 ((void *)0x0004) +# define GLUT_BITMAP_TIMES_ROMAN_24 ((void *)0x0005) +# define GLUT_BITMAP_HELVETICA_10 ((void *)0x0006) +# define GLUT_BITMAP_HELVETICA_12 ((void *)0x0007) +# define GLUT_BITMAP_HELVETICA_18 ((void *)0x0008) +#else + /* + * I don't really know if it's a good idea... But here it goes: + */ + extern void* glutStrokeRoman; + extern void* glutStrokeMonoRoman; + extern void* glutBitmap9By15; + extern void* glutBitmap8By13; + extern void* glutBitmapTimesRoman10; + extern void* glutBitmapTimesRoman24; + extern void* glutBitmapHelvetica10; + extern void* glutBitmapHelvetica12; + extern void* glutBitmapHelvetica18; + + /* + * Those pointers will be used by following definitions: + */ +# define GLUT_STROKE_ROMAN ((void *) &glutStrokeRoman) +# define GLUT_STROKE_MONO_ROMAN ((void *) &glutStrokeMonoRoman) +# define GLUT_BITMAP_9_BY_15 ((void *) &glutBitmap9By15) +# define GLUT_BITMAP_8_BY_13 ((void *) &glutBitmap8By13) +# define GLUT_BITMAP_TIMES_ROMAN_10 ((void *) &glutBitmapTimesRoman10) +# define GLUT_BITMAP_TIMES_ROMAN_24 ((void *) &glutBitmapTimesRoman24) +# define GLUT_BITMAP_HELVETICA_10 ((void *) &glutBitmapHelvetica10) +# define GLUT_BITMAP_HELVETICA_12 ((void *) &glutBitmapHelvetica12) +# define GLUT_BITMAP_HELVETICA_18 ((void *) &glutBitmapHelvetica18) +#endif + +/* + * GLUT API macro definitions -- the glutGet parameters + */ +#define GLUT_WINDOW_X 0x0064 +#define GLUT_WINDOW_Y 0x0065 +#define GLUT_WINDOW_WIDTH 0x0066 +#define GLUT_WINDOW_HEIGHT 0x0067 +#define GLUT_WINDOW_BUFFER_SIZE 0x0068 +#define GLUT_WINDOW_STENCIL_SIZE 0x0069 +#define GLUT_WINDOW_DEPTH_SIZE 0x006A +#define GLUT_WINDOW_RED_SIZE 0x006B +#define GLUT_WINDOW_GREEN_SIZE 0x006C +#define GLUT_WINDOW_BLUE_SIZE 0x006D +#define GLUT_WINDOW_ALPHA_SIZE 0x006E +#define GLUT_WINDOW_ACCUM_RED_SIZE 0x006F +#define GLUT_WINDOW_ACCUM_GREEN_SIZE 0x0070 +#define GLUT_WINDOW_ACCUM_BLUE_SIZE 0x0071 +#define GLUT_WINDOW_ACCUM_ALPHA_SIZE 0x0072 +#define GLUT_WINDOW_DOUBLEBUFFER 0x0073 +#define GLUT_WINDOW_RGBA 0x0074 +#define GLUT_WINDOW_PARENT 0x0075 +#define GLUT_WINDOW_NUM_CHILDREN 0x0076 +#define GLUT_WINDOW_COLORMAP_SIZE 0x0077 +#define GLUT_WINDOW_NUM_SAMPLES 0x0078 +#define GLUT_WINDOW_STEREO 0x0079 +#define GLUT_WINDOW_CURSOR 0x007A + +#define GLUT_SCREEN_WIDTH 0x00C8 +#define GLUT_SCREEN_HEIGHT 0x00C9 +#define GLUT_SCREEN_WIDTH_MM 0x00CA +#define GLUT_SCREEN_HEIGHT_MM 0x00CB +#define GLUT_MENU_NUM_ITEMS 0x012C +#define GLUT_DISPLAY_MODE_POSSIBLE 0x0190 +#define GLUT_INIT_WINDOW_X 0x01F4 +#define GLUT_INIT_WINDOW_Y 0x01F5 +#define GLUT_INIT_WINDOW_WIDTH 0x01F6 +#define GLUT_INIT_WINDOW_HEIGHT 0x01F7 +#define GLUT_INIT_DISPLAY_MODE 0x01F8 +#define GLUT_ELAPSED_TIME 0x02BC +#define GLUT_WINDOW_FORMAT_ID 0x007B + +/* + * GLUT API macro definitions -- the glutDeviceGet parameters + */ +#define GLUT_HAS_KEYBOARD 0x0258 +#define GLUT_HAS_MOUSE 0x0259 +#define GLUT_HAS_SPACEBALL 0x025A +#define GLUT_HAS_DIAL_AND_BUTTON_BOX 0x025B +#define GLUT_HAS_TABLET 0x025C +#define GLUT_NUM_MOUSE_BUTTONS 0x025D +#define GLUT_NUM_SPACEBALL_BUTTONS 0x025E +#define GLUT_NUM_BUTTON_BOX_BUTTONS 0x025F +#define GLUT_NUM_DIALS 0x0260 +#define GLUT_NUM_TABLET_BUTTONS 0x0261 +#define GLUT_DEVICE_IGNORE_KEY_REPEAT 0x0262 +#define GLUT_DEVICE_KEY_REPEAT 0x0263 +#define GLUT_HAS_JOYSTICK 0x0264 +#define GLUT_OWNS_JOYSTICK 0x0265 +#define GLUT_JOYSTICK_BUTTONS 0x0266 +#define GLUT_JOYSTICK_AXES 0x0267 +#define GLUT_JOYSTICK_POLL_RATE 0x0268 + +/* + * GLUT API macro definitions -- the glutLayerGet parameters + */ +#define GLUT_OVERLAY_POSSIBLE 0x0320 +#define GLUT_LAYER_IN_USE 0x0321 +#define GLUT_HAS_OVERLAY 0x0322 +#define GLUT_TRANSPARENT_INDEX 0x0323 +#define GLUT_NORMAL_DAMAGED 0x0324 +#define GLUT_OVERLAY_DAMAGED 0x0325 + +/* + * GLUT API macro definitions -- the glutVideoResizeGet parameters + */ +#define GLUT_VIDEO_RESIZE_POSSIBLE 0x0384 +#define GLUT_VIDEO_RESIZE_IN_USE 0x0385 +#define GLUT_VIDEO_RESIZE_X_DELTA 0x0386 +#define GLUT_VIDEO_RESIZE_Y_DELTA 0x0387 +#define GLUT_VIDEO_RESIZE_WIDTH_DELTA 0x0388 +#define GLUT_VIDEO_RESIZE_HEIGHT_DELTA 0x0389 +#define GLUT_VIDEO_RESIZE_X 0x038A +#define GLUT_VIDEO_RESIZE_Y 0x038B +#define GLUT_VIDEO_RESIZE_WIDTH 0x038C +#define GLUT_VIDEO_RESIZE_HEIGHT 0x038D + +/* + * GLUT API macro definitions -- the glutUseLayer parameters + */ +#define GLUT_NORMAL 0x0000 +#define GLUT_OVERLAY 0x0001 + +/* + * GLUT API macro definitions -- the glutGetModifiers parameters + */ +#define GLUT_ACTIVE_SHIFT 0x0001 +#define GLUT_ACTIVE_CTRL 0x0002 +#define GLUT_ACTIVE_ALT 0x0004 + +/* + * GLUT API macro definitions -- the glutSetCursor parameters + */ +#define GLUT_CURSOR_RIGHT_ARROW 0x0000 +#define GLUT_CURSOR_LEFT_ARROW 0x0001 +#define GLUT_CURSOR_INFO 0x0002 +#define GLUT_CURSOR_DESTROY 0x0003 +#define GLUT_CURSOR_HELP 0x0004 +#define GLUT_CURSOR_CYCLE 0x0005 +#define GLUT_CURSOR_SPRAY 0x0006 +#define GLUT_CURSOR_WAIT 0x0007 +#define GLUT_CURSOR_TEXT 0x0008 +#define GLUT_CURSOR_CROSSHAIR 0x0009 +#define GLUT_CURSOR_UP_DOWN 0x000A +#define GLUT_CURSOR_LEFT_RIGHT 0x000B +#define GLUT_CURSOR_TOP_SIDE 0x000C +#define GLUT_CURSOR_BOTTOM_SIDE 0x000D +#define GLUT_CURSOR_LEFT_SIDE 0x000E +#define GLUT_CURSOR_RIGHT_SIDE 0x000F +#define GLUT_CURSOR_TOP_LEFT_CORNER 0x0010 +#define GLUT_CURSOR_TOP_RIGHT_CORNER 0x0011 +#define GLUT_CURSOR_BOTTOM_RIGHT_CORNER 0x0012 +#define GLUT_CURSOR_BOTTOM_LEFT_CORNER 0x0013 +#define GLUT_CURSOR_INHERIT 0x0064 +#define GLUT_CURSOR_NONE 0x0065 +#define GLUT_CURSOR_FULL_CROSSHAIR 0x0066 + +/* + * GLUT API macro definitions -- RGB color component specification definitions + */ +#define GLUT_RED 0x0000 +#define GLUT_GREEN 0x0001 +#define GLUT_BLUE 0x0002 + +/* + * GLUT API macro definitions -- additional keyboard and joystick definitions + */ +#define GLUT_KEY_REPEAT_OFF 0x0000 +#define GLUT_KEY_REPEAT_ON 0x0001 +#define GLUT_KEY_REPEAT_DEFAULT 0x0002 + +#define GLUT_JOYSTICK_BUTTON_A 0x0001 +#define GLUT_JOYSTICK_BUTTON_B 0x0002 +#define GLUT_JOYSTICK_BUTTON_C 0x0004 +#define GLUT_JOYSTICK_BUTTON_D 0x0008 + +/* + * GLUT API macro definitions -- game mode definitions + */ +#define GLUT_GAME_MODE_ACTIVE 0x0000 +#define GLUT_GAME_MODE_POSSIBLE 0x0001 +#define GLUT_GAME_MODE_WIDTH 0x0002 +#define GLUT_GAME_MODE_HEIGHT 0x0003 +#define GLUT_GAME_MODE_PIXEL_DEPTH 0x0004 +#define GLUT_GAME_MODE_REFRESH_RATE 0x0005 +#define GLUT_GAME_MODE_DISPLAY_CHANGED 0x0006 + +/* + * Initialization functions, see fglut_init.c + */ +FGAPI void FGAPIENTRY glutInit( int* pargc, char** argv ); +FGAPI void FGAPIENTRY glutInitWindowPosition( int x, int y ); +FGAPI void FGAPIENTRY glutInitWindowSize( int width, int height ); +FGAPI void FGAPIENTRY glutInitDisplayMode( unsigned int displayMode ); +FGAPI void FGAPIENTRY glutInitDisplayString( const char* displayMode ); + +/* + * Process loop function, see freeglut_main.c + */ +FGAPI void FGAPIENTRY glutMainLoop( void ); + +/* + * Window management functions, see freeglut_window.c + */ +FGAPI int FGAPIENTRY glutCreateWindow( const char* title ); +FGAPI int FGAPIENTRY glutCreateSubWindow( int window, int x, int y, int width, int height ); +FGAPI void FGAPIENTRY glutDestroyWindow( int window ); +FGAPI void FGAPIENTRY glutSetWindow( int window ); +FGAPI int FGAPIENTRY glutGetWindow( void ); +FGAPI void FGAPIENTRY glutSetWindowTitle( const char* title ); +FGAPI void FGAPIENTRY glutSetIconTitle( const char* title ); +FGAPI void FGAPIENTRY glutReshapeWindow( int width, int height ); +FGAPI void FGAPIENTRY glutPositionWindow( int x, int y ); +FGAPI void FGAPIENTRY glutShowWindow( void ); +FGAPI void FGAPIENTRY glutHideWindow( void ); +FGAPI void FGAPIENTRY glutIconifyWindow( void ); +FGAPI void FGAPIENTRY glutPushWindow( void ); +FGAPI void FGAPIENTRY glutPopWindow( void ); +FGAPI void FGAPIENTRY glutFullScreen( void ); + +/* + * Display-connected functions, see freeglut_display.c + */ +FGAPI void FGAPIENTRY glutPostWindowRedisplay( int window ); +FGAPI void FGAPIENTRY glutPostRedisplay( void ); +FGAPI void FGAPIENTRY glutSwapBuffers( void ); + +/* + * Mouse cursor functions, see freeglut_cursor.c + */ +FGAPI void FGAPIENTRY glutWarpPointer( int x, int y ); +FGAPI void FGAPIENTRY glutSetCursor( int cursor ); + +/* + * Overlay stuff, see freeglut_overlay.c + */ +FGAPI void FGAPIENTRY glutEstablishOverlay( void ); +FGAPI void FGAPIENTRY glutRemoveOverlay( void ); +FGAPI void FGAPIENTRY glutUseLayer( GLenum layer ); +FGAPI void FGAPIENTRY glutPostOverlayRedisplay( void ); +FGAPI void FGAPIENTRY glutPostWindowOverlayRedisplay( int window ); +FGAPI void FGAPIENTRY glutShowOverlay( void ); +FGAPI void FGAPIENTRY glutHideOverlay( void ); + +/* + * Menu stuff, see freeglut_menu.c + */ +FGAPI int FGAPIENTRY glutCreateMenu( void (* callback)( int menu ) ); +FGAPI void FGAPIENTRY glutDestroyMenu( int menu ); +FGAPI int FGAPIENTRY glutGetMenu( void ); +FGAPI void FGAPIENTRY glutSetMenu( int menu ); +FGAPI void FGAPIENTRY glutAddMenuEntry( const char* label, int value ); +FGAPI void FGAPIENTRY glutAddSubMenu( const char* label, int subMenu ); +FGAPI void FGAPIENTRY glutChangeToMenuEntry( int item, const char* label, int value ); +FGAPI void FGAPIENTRY glutChangeToSubMenu( int item, const char* label, int value ); +FGAPI void FGAPIENTRY glutRemoveMenuItem( int item ); +FGAPI void FGAPIENTRY glutAttachMenu( int button ); +FGAPI void FGAPIENTRY glutDetachMenu( int button ); + +/* + * Global callback functions, see freeglut_callbacks.c + */ +FGAPI void FGAPIENTRY glutTimerFunc( unsigned int time, void (* callback)( int ), int value ); +FGAPI void FGAPIENTRY glutIdleFunc( void (* callback)( void ) ); + +/* + * Window-specific callback functions, see freeglut_callbacks.c + */ +FGAPI void FGAPIENTRY glutKeyboardFunc( void (* callback)( unsigned char, int, int ) ); +FGAPI void FGAPIENTRY glutSpecialFunc( void (* callback)( int, int, int ) ); +FGAPI void FGAPIENTRY glutReshapeFunc( void (* callback)( int, int ) ); +FGAPI void FGAPIENTRY glutVisibilityFunc( void (* callback)( int ) ); +FGAPI void FGAPIENTRY glutDisplayFunc( void (* callback)( void ) ); +FGAPI void FGAPIENTRY glutMouseFunc( void (* callback)( int, int, int, int ) ); +FGAPI void FGAPIENTRY glutMotionFunc( void (* callback)( int, int ) ); +FGAPI void FGAPIENTRY glutPassiveMotionFunc( void (* callback)( int, int ) ); +FGAPI void FGAPIENTRY glutEntryFunc( void (* callback)( int ) ); + +FGAPI void FGAPIENTRY glutKeyboardUpFunc( void (* callback)( unsigned char, int, int ) ); +FGAPI void FGAPIENTRY glutSpecialUpFunc( void (* callback)( int, int, int ) ); +FGAPI void FGAPIENTRY glutJoystickFunc( void (* callback)( unsigned int, int, int, int ), int pollInterval ); +FGAPI void FGAPIENTRY glutMenuStateFunc( void (* callback)( int ) ); +FGAPI void FGAPIENTRY glutMenuStatusFunc( void (* callback)( int, int, int ) ); +FGAPI void FGAPIENTRY glutOverlayDisplayFunc( void (* callback)( void ) ); +FGAPI void FGAPIENTRY glutWindowStatusFunc( void (* callback)( int ) ); + +FGAPI void FGAPIENTRY glutSpaceballMotionFunc( void (* callback)( int, int, int ) ); +FGAPI void FGAPIENTRY glutSpaceballRotateFunc( void (* callback)( int, int, int ) ); +FGAPI void FGAPIENTRY glutSpaceballButtonFunc( void (* callback)( int, int ) ); +FGAPI void FGAPIENTRY glutButtonBoxFunc( void (* callback)( int, int ) ); +FGAPI void FGAPIENTRY glutDialsFunc( void (* callback)( int, int ) ); +FGAPI void FGAPIENTRY glutTabletMotionFunc( void (* callback)( int, int ) ); +FGAPI void FGAPIENTRY glutTabletButtonFunc( void (* callback)( int, int, int, int ) ); + +/* + * State setting and retrieval functions, see freeglut_state.c + */ +FGAPI int FGAPIENTRY glutGet( GLenum query ); +FGAPI int FGAPIENTRY glutDeviceGet( GLenum query ); +FGAPI int FGAPIENTRY glutGetModifiers( void ); +FGAPI int FGAPIENTRY glutLayerGet( GLenum query ); + +/* + * Font stuff, see freeglut_font.c + */ +FGAPI void FGAPIENTRY glutBitmapCharacter( void* font, int character ); +FGAPI int FGAPIENTRY glutBitmapWidth( void* font, int character ); +FGAPI void FGAPIENTRY glutStrokeCharacter( void* font, int character ); +FGAPI int FGAPIENTRY glutStrokeWidth( void* font, int character ); +FGAPI int FGAPIENTRY glutBitmapLength( void* font, const unsigned char* string ); +FGAPI int FGAPIENTRY glutStrokeLength( void* font, const unsigned char* string ); + +/* + * Geometry functions, see freeglut_geometry.c + */ +FGAPI void FGAPIENTRY glutWireCube( GLdouble size ); +FGAPI void FGAPIENTRY glutSolidCube( GLdouble size ); +FGAPI void FGAPIENTRY glutWireSphere( GLdouble radius, GLint slices, GLint stacks ); +FGAPI void FGAPIENTRY glutSolidSphere( GLdouble radius, GLint slices, GLint stacks ); +FGAPI void FGAPIENTRY glutWireCone( GLdouble base, GLdouble height, GLint slices, GLint stacks ); +FGAPI void FGAPIENTRY glutSolidCone( GLdouble base, GLdouble height, GLint slices, GLint stacks ); + +FGAPI void FGAPIENTRY glutWireTorus( GLdouble innerRadius, GLdouble outerRadius, GLint sides, GLint rings ); +FGAPI void FGAPIENTRY glutSolidTorus( GLdouble innerRadius, GLdouble outerRadius, GLint sides, GLint rings ); +FGAPI void FGAPIENTRY glutWireDodecahedron( void ); +FGAPI void FGAPIENTRY glutSolidDodecahedron( void ); +FGAPI void FGAPIENTRY glutWireOctahedron( void ); +FGAPI void FGAPIENTRY glutSolidOctahedron( void ); +FGAPI void FGAPIENTRY glutWireTetrahedron( void ); +FGAPI void FGAPIENTRY glutSolidTetrahedron( void ); +FGAPI void FGAPIENTRY glutWireIcosahedron( void ); +FGAPI void FGAPIENTRY glutSolidIcosahedron( void ); + +/* + * Teapot rendering functions, found in freeglut_teapot.c + */ +FGAPI void FGAPIENTRY glutWireTeapot( GLdouble size ); +FGAPI void FGAPIENTRY glutSolidTeapot( GLdouble size ); + +/* + * Game mode functions, see freeglut_gamemode.c + */ +FGAPI void FGAPIENTRY glutGameModeString( const char* string ); +FGAPI int FGAPIENTRY glutEnterGameMode( void ); +FGAPI void FGAPIENTRY glutLeaveGameMode( void ); +FGAPI int FGAPIENTRY glutGameModeGet( GLenum query ); + +/* + * Video resize functions, see freeglut_videoresize.c + */ +FGAPI int FGAPIENTRY glutVideoResizeGet( GLenum query ); +FGAPI void FGAPIENTRY glutSetupVideoResizing( void ); +FGAPI void FGAPIENTRY glutStopVideoResizing( void ); +FGAPI void FGAPIENTRY glutVideoResize( int x, int y, int width, int height ); +FGAPI void FGAPIENTRY glutVideoPan( int x, int y, int width, int height ); + +/* + * Colormap functions, see freeglut_misc.c + */ +FGAPI void FGAPIENTRY glutSetColor( int color, GLfloat red, GLfloat green, GLfloat blue ); +FGAPI GLfloat FGAPIENTRY glutGetColor( int color, int component ); +FGAPI void FGAPIENTRY glutCopyColormap( int window ); + +/* + * Misc keyboard and joystick functions, see freeglut_misc.c + */ +FGAPI void FGAPIENTRY glutIgnoreKeyRepeat( int ignore ); +FGAPI void FGAPIENTRY glutSetKeyRepeat( int repeatMode ); +FGAPI void FGAPIENTRY glutForceJoystickFunc( void ); + +/* + * Misc functions, see freeglut_misc.c + */ +FGAPI int FGAPIENTRY glutExtensionSupported( const char* extension ); +FGAPI void FGAPIENTRY glutReportErrors( void ); + +/* Comment from glut.h of classic GLUT: + + Win32 has an annoying issue where there are multiple C run-time + libraries (CRTs). If the executable is linked with a different CRT + from the GLUT DLL, the GLUT DLL will not share the same CRT static + data seen by the executable. In particular, atexit callbacks registered + in the executable will not be called if GLUT calls its (different) + exit routine). GLUT is typically built with the + "/MD" option (the CRT with multithreading DLL support), but the Visual + C++ linker default is "/ML" (the single threaded CRT). + + One workaround to this issue is requiring users to always link with + the same CRT as GLUT is compiled with. That requires users supply a + non-standard option. GLUT 3.7 has its own built-in workaround where + the executable's "exit" function pointer is covertly passed to GLUT. + GLUT then calls the executable's exit function pointer to ensure that + any "atexit" calls registered by the application are called if GLUT + needs to exit. + + Note that the __glut*WithExit routines should NEVER be called directly. + To avoid the atexit workaround, #define GLUT_DISABLE_ATEXIT_HACK. */ + +/* to get the prototype for exit() */ +#include + +#if defined(_WIN32) && !defined(GLUT_DISABLE_ATEXIT_HACK) && !defined(__WATCOMC__) +FGAPI void FGAPIENTRY __glutInitWithExit(int *argcp, char **argv, void (__cdecl *exitfunc)(int)); +FGAPI int FGAPIENTRY __glutCreateWindowWithExit(const char *title, void (__cdecl *exitfunc)(int)); +FGAPI int FGAPIENTRY __glutCreateMenuWithExit(void (* func)(int), void (__cdecl *exitfunc)(int)); +#ifndef FREEGLUT_BUILDING_LIB +#if defined(__GNUC__) +#define FGUNUSED __attribute__((unused)) +#else +#define FGUNUSED +#endif +static void FGAPIENTRY FGUNUSED glutInit_ATEXIT_HACK(int *argcp, char **argv) { __glutInitWithExit(argcp, argv, exit); } +#define glutInit glutInit_ATEXIT_HACK +static int FGAPIENTRY FGUNUSED glutCreateWindow_ATEXIT_HACK(const char *title) { return __glutCreateWindowWithExit(title, exit); } +#define glutCreateWindow glutCreateWindow_ATEXIT_HACK +static int FGAPIENTRY FGUNUSED glutCreateMenu_ATEXIT_HACK(void (* func)(int)) { return __glutCreateMenuWithExit(func, exit); } +#define glutCreateMenu glutCreateMenu_ATEXIT_HACK +#endif +#endif + +#ifdef __cplusplus + } +#endif + +/*** END OF FILE ***/ + +#endif /* __FREEGLUT_STD_H__ */ + diff --git a/examples/opengl-framework/freeglut/GL/glut.h b/examples/opengl-framework/freeglut/GL/glut.h new file mode 100644 index 00000000..6191f77b --- /dev/null +++ b/examples/opengl-framework/freeglut/GL/glut.h @@ -0,0 +1,21 @@ +#ifndef __GLUT_H__ +#define __GLUT_H__ + +/* + * glut.h + * + * The freeglut library include file + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include "freeglut_std.h" + +/*** END OF FILE ***/ + +#endif /* __GLUT_H__ */ diff --git a/examples/opengl-framework/freeglut/VERSION.txt b/examples/opengl-framework/freeglut/VERSION.txt new file mode 100644 index 00000000..6533b668 --- /dev/null +++ b/examples/opengl-framework/freeglut/VERSION.txt @@ -0,0 +1 @@ +2.8.0 \ No newline at end of file diff --git a/examples/opengl-framework/freeglut/freeglut_callbacks.c b/examples/opengl-framework/freeglut/freeglut_callbacks.c new file mode 100644 index 00000000..a40f7daf --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_callbacks.c @@ -0,0 +1,412 @@ +/* + * freeglut_callbacks.c + * + * The callbacks setting methods. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Fri Dec 3 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * All of the callbacks setting methods can be generalized to this: + */ +#define SET_CALLBACK(a) \ +do \ +{ \ + if( fgStructure.CurrentWindow == NULL ) \ + return; \ + SET_WCB( ( *( fgStructure.CurrentWindow ) ), a, callback ); \ +} while( 0 ) + +/* + * Sets the Display callback for the current window + */ +void FGAPIENTRY glutDisplayFunc( void (* callback)( void ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDisplayFunc" ); + if( !callback ) + fgError( "Fatal error in program. NULL display callback not " + "permitted in GLUT 3.0+ or freeglut 2.0.1+" ); + SET_CALLBACK( Display ); +} + +/* + * Sets the Reshape callback for the current window + */ +void FGAPIENTRY glutReshapeFunc( void (* callback)( int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutReshapeFunc" ); + SET_CALLBACK( Reshape ); +} + +/* + * Sets the Keyboard callback for the current window + */ +void FGAPIENTRY glutKeyboardFunc( void (* callback) + ( unsigned char, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutKeyboardFunc" ); + SET_CALLBACK( Keyboard ); +} + +/* + * Sets the Special callback for the current window + */ +void FGAPIENTRY glutSpecialFunc( void (* callback)( int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSpecialFunc" ); + SET_CALLBACK( Special ); +} + +/* + * Sets the global idle callback + */ +void FGAPIENTRY glutIdleFunc( void (* callback)( void ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutIdleFunc" ); + fgState.IdleCallback = callback; +} + +/* + * Sets the Timer callback for the current window + */ +void FGAPIENTRY glutTimerFunc( unsigned int timeOut, void (* callback)( int ), + int timerID ) +{ + SFG_Timer *timer, *node; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutTimerFunc" ); + + if( (timer = fgState.FreeTimers.Last) ) + { + fgListRemove( &fgState.FreeTimers, &timer->Node ); + } + else + { + if( ! (timer = malloc(sizeof(SFG_Timer))) ) + fgError( "Fatal error: " + "Memory allocation failure in glutTimerFunc()" ); + } + + timer->Callback = callback; + timer->ID = timerID; + timer->TriggerTime = fgElapsedTime() + timeOut; + + for( node = fgState.Timers.First; node; node = node->Node.Next ) + { + if( node->TriggerTime > timer->TriggerTime ) + break; + } + + fgListInsert( &fgState.Timers, &node->Node, &timer->Node ); +} + +/* + * Sets the Visibility callback for the current window. + */ +static void fghVisibility( int status ) +{ + int glut_status = GLUT_VISIBLE; + + FREEGLUT_INTERNAL_ERROR_EXIT_IF_NOT_INITIALISED ( "Visibility Callback" ); + freeglut_return_if_fail( fgStructure.CurrentWindow ); + + if( ( GLUT_HIDDEN == status ) || ( GLUT_FULLY_COVERED == status ) ) + glut_status = GLUT_NOT_VISIBLE; + INVOKE_WCB( *( fgStructure.CurrentWindow ), Visibility, ( glut_status ) ); +} + +void FGAPIENTRY glutVisibilityFunc( void (* callback)( int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutVisibilityFunc" ); + SET_CALLBACK( Visibility ); + + if( callback ) + glutWindowStatusFunc( fghVisibility ); + else + glutWindowStatusFunc( NULL ); +} + +/* + * Sets the keyboard key release callback for the current window + */ +void FGAPIENTRY glutKeyboardUpFunc( void (* callback) + ( unsigned char, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutKeyboardUpFunc" ); + SET_CALLBACK( KeyboardUp ); +} + +/* + * Sets the special key release callback for the current window + */ +void FGAPIENTRY glutSpecialUpFunc( void (* callback)( int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSpecialUpFunc" ); + SET_CALLBACK( SpecialUp ); +} + +/* + * Sets the joystick callback and polling rate for the current window + */ +void FGAPIENTRY glutJoystickFunc( void (* callback) + ( unsigned int, int, int, int ), + int pollInterval ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickFunc" ); + fgInitialiseJoysticks (); + + if ( ( ( fgStructure.CurrentWindow->State.JoystickPollRate < 0 ) || + !FETCH_WCB(*fgStructure.CurrentWindow,Joystick) ) && /* Joystick callback was disabled */ + ( callback && ( pollInterval >= 0 ) ) ) /* but is now enabled */ + ++fgState.NumActiveJoysticks; + else if ( ( ( fgStructure.CurrentWindow->State.JoystickPollRate >= 0 ) && + FETCH_WCB(*fgStructure.CurrentWindow,Joystick) ) && /* Joystick callback was enabled */ + ( !callback || ( pollInterval < 0 ) ) ) /* but is now disabled */ + --fgState.NumActiveJoysticks; + + SET_CALLBACK( Joystick ); + fgStructure.CurrentWindow->State.JoystickPollRate = pollInterval; + + fgStructure.CurrentWindow->State.JoystickLastPoll = + fgElapsedTime() - fgStructure.CurrentWindow->State.JoystickPollRate; + + if( fgStructure.CurrentWindow->State.JoystickLastPoll < 0 ) + fgStructure.CurrentWindow->State.JoystickLastPoll = 0; +} + +/* + * Sets the mouse callback for the current window + */ +void FGAPIENTRY glutMouseFunc( void (* callback)( int, int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMouseFunc" ); + SET_CALLBACK( Mouse ); +} + +/* + * Sets the mouse wheel callback for the current window + */ +void FGAPIENTRY glutMouseWheelFunc( void (* callback)( int, int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMouseWheelFunc" ); + SET_CALLBACK( MouseWheel ); +} + +/* + * Sets the mouse motion callback for the current window (one or more buttons + * are pressed) + */ +void FGAPIENTRY glutMotionFunc( void (* callback)( int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMotionFunc" ); + SET_CALLBACK( Motion ); +} + +/* + * Sets the passive mouse motion callback for the current window (no mouse + * buttons are pressed) + */ +void FGAPIENTRY glutPassiveMotionFunc( void (* callback)( int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPassiveMotionFunc" ); + SET_CALLBACK( Passive ); +} + +/* + * Window mouse entry/leave callback + */ +void FGAPIENTRY glutEntryFunc( void (* callback)( int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutEntryFunc" ); + SET_CALLBACK( Entry ); +} + +/* + * Window destruction callbacks + */ +void FGAPIENTRY glutCloseFunc( void (* callback)( void ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutCloseFunc" ); + SET_CALLBACK( Destroy ); +} + +void FGAPIENTRY glutWMCloseFunc( void (* callback)( void ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWMCloseFunc" ); + glutCloseFunc( callback ); +} + +/* A. Donev: Destruction callback for menus */ +void FGAPIENTRY glutMenuDestroyFunc( void (* callback)( void ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMenuDestroyFunc" ); + if( fgStructure.CurrentMenu ) + fgStructure.CurrentMenu->Destroy = callback; +} + +/* + * Deprecated version of glutMenuStatusFunc callback setting method + */ +void FGAPIENTRY glutMenuStateFunc( void (* callback)( int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMenuStateFunc" ); + fgState.MenuStateCallback = callback; +} + +/* + * Sets the global menu status callback for the current window + */ +void FGAPIENTRY glutMenuStatusFunc( void (* callback)( int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMenuStatusFunc" ); + fgState.MenuStatusCallback = callback; +} + +/* + * Sets the overlay display callback for the current window + */ +void FGAPIENTRY glutOverlayDisplayFunc( void (* callback)( void ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutOverlayDisplayFunc" ); + SET_CALLBACK( OverlayDisplay ); +} + +/* + * Sets the window status callback for the current window + */ +void FGAPIENTRY glutWindowStatusFunc( void (* callback)( int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWindowStatusFunc" ); + SET_CALLBACK( WindowStatus ); +} + +/* + * Sets the spaceball motion callback for the current window + */ +void FGAPIENTRY glutSpaceballMotionFunc( void (* callback)( int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSpaceballMotionFunc" ); + fgInitialiseSpaceball(); + + SET_CALLBACK( SpaceMotion ); +} + +/* + * Sets the spaceball rotate callback for the current window + */ +void FGAPIENTRY glutSpaceballRotateFunc( void (* callback)( int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSpaceballRotateFunc" ); + fgInitialiseSpaceball(); + + SET_CALLBACK( SpaceRotation ); +} + +/* + * Sets the spaceball button callback for the current window + */ +void FGAPIENTRY glutSpaceballButtonFunc( void (* callback)( int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSpaceballButtonFunc" ); + fgInitialiseSpaceball(); + + SET_CALLBACK( SpaceButton ); +} + +/* + * Sets the button box callback for the current window + */ +void FGAPIENTRY glutButtonBoxFunc( void (* callback)( int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutButtonBoxFunc" ); + SET_CALLBACK( ButtonBox ); +} + +/* + * Sets the dials box callback for the current window + */ +void FGAPIENTRY glutDialsFunc( void (* callback)( int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDialsFunc" ); + SET_CALLBACK( Dials ); +} + +/* + * Sets the tablet motion callback for the current window + */ +void FGAPIENTRY glutTabletMotionFunc( void (* callback)( int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutTabletMotionFunc" ); + SET_CALLBACK( TabletMotion ); +} + +/* + * Sets the tablet buttons callback for the current window + */ +void FGAPIENTRY glutTabletButtonFunc( void (* callback)( int, int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutTabletButtonFunc" ); + SET_CALLBACK( TabletButton ); +} + +/* + * Sets the multi-pointer entry callback for the current window + */ +void FGAPIENTRY glutMultiEntryFunc( void (* callback)(int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMultiEntryFunc" ); + SET_CALLBACK( MultiEntry ); +} + +/* + * Sets the multi-pointer button callback for the current window + */ +void FGAPIENTRY glutMultiButtonFunc( void (* callback)(int, int, int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMultiButtonFunc" ); + SET_CALLBACK( MultiButton ); +} + +/* + * Sets the multi-pointer motion callback for the current window + */ +void FGAPIENTRY glutMultiMotionFunc( void (* callback)(int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMultiMotionFunc" ); + SET_CALLBACK( MultiMotion ); +} + +/* + * Sets the multi-pointer passive motion callback for the current window + */ +void FGAPIENTRY glutMultiPassiveFunc( void (* callback)(int, int, int ) ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMultiPassiveFunc" ); + SET_CALLBACK( MultiPassive ); +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_cursor.c b/examples/opengl-framework/freeglut/freeglut_cursor.c new file mode 100644 index 00000000..33708319 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_cursor.c @@ -0,0 +1,282 @@ +/* + * freeglut_cursor.c + * + * The mouse cursor related stuff. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 16 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* + * TODO BEFORE THE STABLE RELEASE: + * glutSetCursor() -- Win32 mappings are incomplete. + * + * It would be good to use custom mouse cursor shapes, and introduce + * an option to display them using glBitmap() and/or texture mapping, + * apart from the windowing system version. + */ + +/* -- PRIVATE FUNCTIONS --------------------------------------------------- */ + +#if TARGET_HOST_POSIX_X11 || TARGET_HOST_MAC_OSX || TARGET_HOST_SOLARIS + #include + +/* + * A factory method for an empty cursor + */ +static Cursor getEmptyCursor( void ) +{ + static Cursor cursorNone = None; + if( cursorNone == None ) { + char cursorNoneBits[ 32 ]; + XColor dontCare; + Pixmap cursorNonePixmap; + memset( cursorNoneBits, 0, sizeof( cursorNoneBits ) ); + memset( &dontCare, 0, sizeof( dontCare ) ); + cursorNonePixmap = XCreateBitmapFromData ( fgDisplay.Display, + fgDisplay.RootWindow, + cursorNoneBits, 16, 16 ); + if( cursorNonePixmap != None ) { + cursorNone = XCreatePixmapCursor( fgDisplay.Display, + cursorNonePixmap, cursorNonePixmap, + &dontCare, &dontCare, 0, 0 ); + XFreePixmap( fgDisplay.Display, cursorNonePixmap ); + } + } + return cursorNone; +} + +typedef struct tag_cursorCacheEntry cursorCacheEntry; +struct tag_cursorCacheEntry { + unsigned int cursorShape; /* an XC_foo value */ + Cursor cachedCursor; /* None if the corresponding cursor has + not been created yet */ +}; + +/* + * Note: The arrangement of the table below depends on the fact that + * the "normal" GLUT_CURSOR_* values start a 0 and are consecutive. + */ +static cursorCacheEntry cursorCache[] = { + { XC_arrow, None }, /* GLUT_CURSOR_RIGHT_ARROW */ + { XC_top_left_arrow, None }, /* GLUT_CURSOR_LEFT_ARROW */ + { XC_hand1, None }, /* GLUT_CURSOR_INFO */ + { XC_pirate, None }, /* GLUT_CURSOR_DESTROY */ + { XC_question_arrow, None }, /* GLUT_CURSOR_HELP */ + { XC_exchange, None }, /* GLUT_CURSOR_CYCLE */ + { XC_spraycan, None }, /* GLUT_CURSOR_SPRAY */ + { XC_watch, None }, /* GLUT_CURSOR_WAIT */ + { XC_xterm, None }, /* GLUT_CURSOR_TEXT */ + { XC_crosshair, None }, /* GLUT_CURSOR_CROSSHAIR */ + { XC_sb_v_double_arrow, None }, /* GLUT_CURSOR_UP_DOWN */ + { XC_sb_h_double_arrow, None }, /* GLUT_CURSOR_LEFT_RIGHT */ + { XC_top_side, None }, /* GLUT_CURSOR_TOP_SIDE */ + { XC_bottom_side, None }, /* GLUT_CURSOR_BOTTOM_SIDE */ + { XC_left_side, None }, /* GLUT_CURSOR_LEFT_SIDE */ + { XC_right_side, None }, /* GLUT_CURSOR_RIGHT_SIDE */ + { XC_top_left_corner, None }, /* GLUT_CURSOR_TOP_LEFT_CORNER */ + { XC_top_right_corner, None }, /* GLUT_CURSOR_TOP_RIGHT_CORNER */ + { XC_bottom_right_corner, None }, /* GLUT_CURSOR_BOTTOM_RIGHT_CORNER */ + { XC_bottom_left_corner, None } /* GLUT_CURSOR_BOTTOM_LEFT_CORNER */ +}; + +static void fghSetCursor ( SFG_Window *window, int cursorID ) +{ + Cursor cursor; + /* + * XXX FULL_CROSSHAIR demotes to plain CROSSHAIR. Old GLUT allows + * for this, but if there is a system that easily supports a full- + * window (or full-screen) crosshair, we might consider it. + */ + int cursorIDToUse = + ( cursorID == GLUT_CURSOR_FULL_CROSSHAIR ) ? GLUT_CURSOR_CROSSHAIR : cursorID; + + if( ( cursorIDToUse >= 0 ) && + ( cursorIDToUse < sizeof( cursorCache ) / sizeof( cursorCache[0] ) ) ) { + cursorCacheEntry *entry = &cursorCache[ cursorIDToUse ]; + if( entry->cachedCursor == None ) { + entry->cachedCursor = + XCreateFontCursor( fgDisplay.Display, entry->cursorShape ); + } + cursor = entry->cachedCursor; + } else { + switch( cursorIDToUse ) + { + case GLUT_CURSOR_NONE: + cursor = getEmptyCursor( ); + break; + + case GLUT_CURSOR_INHERIT: + cursor = None; + break; + + default: + fgError( "Unknown cursor type: %d", cursorIDToUse ); + return; + } + } + + if ( cursorIDToUse == GLUT_CURSOR_INHERIT ) { + XUndefineCursor( fgDisplay.Display, window->Window.Handle ); + } else if ( cursor != None ) { + XDefineCursor( fgDisplay.Display, window->Window.Handle, cursor ); + } else if ( cursorIDToUse != GLUT_CURSOR_NONE ) { + fgError( "Failed to create cursor" ); + } +} + + +static void fghWarpPointer ( int x, int y ) +{ + XWarpPointer( + fgDisplay.Display, + None, + fgStructure.CurrentWindow->Window.Handle, + 0, 0, 0, 0, + x, y + ); + /* Make the warp visible immediately. */ + XFlush( fgDisplay.Display ); +} +#endif + + +#if TARGET_HOST_MS_WINDOWS +static void fghSetCursor ( SFG_Window *window, int cursorID ) +{ + /* + * Joe Krahn is re-writing the following code. + */ + /* Set the cursor AND change it for this window class. */ +#if !defined(__MINGW64__) && _MSC_VER <= 1200 +# define MAP_CURSOR(a,b) \ + case a: \ + SetCursor( LoadCursor( NULL, b ) ); \ + SetClassLong( window->Window.Handle, \ + GCL_HCURSOR, \ + ( LONG )LoadCursor( NULL, b ) ); \ + break; + /* Nuke the cursor AND change it for this window class. */ +# define ZAP_CURSOR(a,b) \ + case a: \ + SetCursor( NULL ); \ + SetClassLong( window->Window.Handle, \ + GCL_HCURSOR, ( LONG )NULL ); \ + break; +#else +# define MAP_CURSOR(a,b) \ + case a: \ + SetCursor( LoadCursor( NULL, b ) ); \ + SetClassLongPtr( window->Window.Handle, \ + GCLP_HCURSOR, \ + ( LONG )( LONG_PTR )LoadCursor( NULL, b ) ); \ + break; + /* Nuke the cursor AND change it for this window class. */ +# define ZAP_CURSOR(a,b) \ + case a: \ + SetCursor( NULL ); \ + SetClassLongPtr( window->Window.Handle, \ + GCLP_HCURSOR, ( LONG )( LONG_PTR )NULL ); \ + break; +#endif + + switch( cursorID ) + { + MAP_CURSOR( GLUT_CURSOR_RIGHT_ARROW, IDC_ARROW ); + MAP_CURSOR( GLUT_CURSOR_LEFT_ARROW, IDC_ARROW ); + MAP_CURSOR( GLUT_CURSOR_INFO, IDC_HELP ); + MAP_CURSOR( GLUT_CURSOR_DESTROY, IDC_CROSS ); + MAP_CURSOR( GLUT_CURSOR_HELP, IDC_HELP ); + MAP_CURSOR( GLUT_CURSOR_CYCLE, IDC_SIZEALL ); + MAP_CURSOR( GLUT_CURSOR_SPRAY, IDC_CROSS ); + MAP_CURSOR( GLUT_CURSOR_WAIT, IDC_WAIT ); + MAP_CURSOR( GLUT_CURSOR_TEXT, IDC_IBEAM ); + MAP_CURSOR( GLUT_CURSOR_CROSSHAIR, IDC_CROSS ); + MAP_CURSOR( GLUT_CURSOR_UP_DOWN, IDC_SIZENS ); + MAP_CURSOR( GLUT_CURSOR_LEFT_RIGHT, IDC_SIZEWE ); + MAP_CURSOR( GLUT_CURSOR_TOP_SIDE, IDC_ARROW ); /* XXX ToDo */ + MAP_CURSOR( GLUT_CURSOR_BOTTOM_SIDE, IDC_ARROW ); /* XXX ToDo */ + MAP_CURSOR( GLUT_CURSOR_LEFT_SIDE, IDC_ARROW ); /* XXX ToDo */ + MAP_CURSOR( GLUT_CURSOR_RIGHT_SIDE, IDC_ARROW ); /* XXX ToDo */ + MAP_CURSOR( GLUT_CURSOR_TOP_LEFT_CORNER, IDC_SIZENWSE ); + MAP_CURSOR( GLUT_CURSOR_TOP_RIGHT_CORNER, IDC_SIZENESW ); + MAP_CURSOR( GLUT_CURSOR_BOTTOM_RIGHT_CORNER, IDC_SIZENWSE ); + MAP_CURSOR( GLUT_CURSOR_BOTTOM_LEFT_CORNER, IDC_SIZENESW ); + MAP_CURSOR( GLUT_CURSOR_INHERIT, IDC_ARROW ); /* XXX ToDo */ + ZAP_CURSOR( GLUT_CURSOR_NONE, NULL ); + MAP_CURSOR( GLUT_CURSOR_FULL_CROSSHAIR, IDC_CROSS ); /* XXX ToDo */ + + default: + fgError( "Unknown cursor type: %d", cursorID ); + break; + } +} + + +static void fghWarpPointer ( int x, int y ) +{ + POINT coords; + coords.x = x; + coords.y = y; + + /* ClientToScreen() translates {coords} for us. */ + ClientToScreen( fgStructure.CurrentWindow->Window.Handle, &coords ); + SetCursorPos( coords.x, coords.y ); +} +#endif + + +/* -- INTERNAL FUNCTIONS ---------------------------------------------------- */ +void fgSetCursor ( SFG_Window *window, int cursorID ) +{ + fghSetCursor ( window, cursorID ); +} + + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * Set the cursor image to be used for the current window + */ +void FGAPIENTRY glutSetCursor( int cursorID ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetCursor" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutSetCursor" ); + + fghSetCursor ( fgStructure.CurrentWindow, cursorID ); + fgStructure.CurrentWindow->State.Cursor = cursorID; +} + +/* + * Moves the mouse pointer to given window coordinates + */ +void FGAPIENTRY glutWarpPointer( int x, int y ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWarpPointer" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutWarpPointer" ); + + fghWarpPointer ( x, y ); +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_display.c b/examples/opengl-framework/freeglut/freeglut_display.c new file mode 100644 index 00000000..2ba0ec54 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_display.c @@ -0,0 +1,103 @@ +/* + * freeglut_display.c + * + * Display message posting, context buffer swapping. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Fri Dec 3 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * Marks the current window to have the redisplay performed when possible... + */ +void FGAPIENTRY glutPostRedisplay( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPostRedisplay" ); + if ( ! fgStructure.CurrentWindow ) + { + fgError ( " ERROR: Function <%s> called" + " with no current window defined.", "glutPostRedisplay" ) ; + } + + fgStructure.CurrentWindow->State.Redisplay = GL_TRUE; +} + +/* + * Swaps the buffers for the current window (if any) + */ +void FGAPIENTRY glutSwapBuffers( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSwapBuffers" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutSwapBuffers" ); + + /* + * "glXSwapBuffers" already performs an implicit call to "glFlush". What + * about "SwapBuffers"? + */ + glFlush( ); + if( ! fgStructure.CurrentWindow->Window.DoubleBuffered ) + return; + +#if TARGET_HOST_POSIX_X11 + glXSwapBuffers( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle ); +#elif TARGET_HOST_MS_WINDOWS + SwapBuffers( fgStructure.CurrentWindow->Window.Device ); +#endif + + /* GLUT_FPS env var support */ + if( fgState.FPSInterval ) + { + GLint t = glutGet( GLUT_ELAPSED_TIME ); + fgState.SwapCount++; + if( fgState.SwapTime == 0 ) + fgState.SwapTime = t; + else if( t - fgState.SwapTime > fgState.FPSInterval ) + { + float time = 0.001f * ( t - fgState.SwapTime ); + float fps = ( float )fgState.SwapCount / time; + fprintf( stderr, + "freeglut: %d frames in %.2f seconds = %.2f FPS\n", + fgState.SwapCount, time, fps ); + fgState.SwapTime = t; + fgState.SwapCount = 0; + } + } +} + +/* + * Mark appropriate window to be displayed + */ +void FGAPIENTRY glutPostWindowRedisplay( int windowID ) +{ + SFG_Window* window; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPostWindowRedisplay" ); + window = fgWindowByID( windowID ); + freeglut_return_if_fail( window ); + window->State.Redisplay = GL_TRUE; +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_ext.c b/examples/opengl-framework/freeglut/freeglut_ext.c new file mode 100644 index 00000000..d8aca45e --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_ext.c @@ -0,0 +1,233 @@ +/* + * freeglut_ext.c + * + * Functions related to OpenGL extensions. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 9 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#define GLX_GLXEXT_PROTOTYPES +#include +#include "freeglut_internal.h" + +static GLUTproc fghGetGLUTProcAddress( const char* procName ) +{ + /* optimization: quick initial check */ + if( strncmp( procName, "glut", 4 ) != 0 ) + return NULL; + +#define CHECK_NAME(x) if( strcmp( procName, #x ) == 0) return (GLUTproc)x; + CHECK_NAME(glutInit); + CHECK_NAME(glutInitDisplayMode); + CHECK_NAME(glutInitDisplayString); + CHECK_NAME(glutInitWindowPosition); + CHECK_NAME(glutInitWindowSize); + CHECK_NAME(glutMainLoop); + CHECK_NAME(glutExit); + CHECK_NAME(glutCreateWindow); + CHECK_NAME(glutCreateSubWindow); + CHECK_NAME(glutDestroyWindow); + CHECK_NAME(glutPostRedisplay); + CHECK_NAME(glutPostWindowRedisplay); + CHECK_NAME(glutSwapBuffers); + CHECK_NAME(glutGetWindow); + CHECK_NAME(glutSetWindow); + CHECK_NAME(glutSetWindowTitle); + CHECK_NAME(glutSetIconTitle); + CHECK_NAME(glutPositionWindow); + CHECK_NAME(glutReshapeWindow); + CHECK_NAME(glutPopWindow); + CHECK_NAME(glutPushWindow); + CHECK_NAME(glutIconifyWindow); + CHECK_NAME(glutShowWindow); + CHECK_NAME(glutHideWindow); + CHECK_NAME(glutFullScreen); + CHECK_NAME(glutSetCursor); + CHECK_NAME(glutWarpPointer); + CHECK_NAME(glutEstablishOverlay); + CHECK_NAME(glutRemoveOverlay); + CHECK_NAME(glutUseLayer); + CHECK_NAME(glutPostOverlayRedisplay); + CHECK_NAME(glutPostWindowOverlayRedisplay); + CHECK_NAME(glutShowOverlay); + CHECK_NAME(glutHideOverlay); + CHECK_NAME(glutCreateMenu); + CHECK_NAME(glutDestroyMenu); + CHECK_NAME(glutGetMenu); + CHECK_NAME(glutSetMenu); + CHECK_NAME(glutAddMenuEntry); + CHECK_NAME(glutAddSubMenu); + CHECK_NAME(glutChangeToMenuEntry); + CHECK_NAME(glutChangeToSubMenu); + CHECK_NAME(glutRemoveMenuItem); + CHECK_NAME(glutAttachMenu); + CHECK_NAME(glutDetachMenu); + CHECK_NAME(glutDisplayFunc); + CHECK_NAME(glutReshapeFunc); + CHECK_NAME(glutKeyboardFunc); + CHECK_NAME(glutMouseFunc); + CHECK_NAME(glutMultiEntryFunc); + CHECK_NAME(glutMultiMotionFunc); + CHECK_NAME(glutMultiButtonFunc); + CHECK_NAME(glutMultiPassiveFunc); + CHECK_NAME(glutMotionFunc); + CHECK_NAME(glutPassiveMotionFunc); + CHECK_NAME(glutEntryFunc); + CHECK_NAME(glutVisibilityFunc); + CHECK_NAME(glutIdleFunc); + CHECK_NAME(glutTimerFunc); + CHECK_NAME(glutMenuStateFunc); + CHECK_NAME(glutSpecialFunc); + CHECK_NAME(glutSpaceballMotionFunc); + CHECK_NAME(glutSpaceballRotateFunc); + CHECK_NAME(glutSpaceballButtonFunc); + CHECK_NAME(glutButtonBoxFunc); + CHECK_NAME(glutDialsFunc); + CHECK_NAME(glutTabletMotionFunc); + CHECK_NAME(glutTabletButtonFunc); + CHECK_NAME(glutMenuStatusFunc); + CHECK_NAME(glutOverlayDisplayFunc); + CHECK_NAME(glutWindowStatusFunc); + CHECK_NAME(glutKeyboardUpFunc); + CHECK_NAME(glutSpecialUpFunc); +#if !defined(_WIN32_WCE) + CHECK_NAME(glutJoystickFunc); +#endif /* !defined(_WIN32_WCE) */ + CHECK_NAME(glutSetColor); + CHECK_NAME(glutGetColor); + CHECK_NAME(glutCopyColormap); + CHECK_NAME(glutGet); + CHECK_NAME(glutDeviceGet); + CHECK_NAME(glutExtensionSupported); + CHECK_NAME(glutGetModifiers); + CHECK_NAME(glutLayerGet); + CHECK_NAME(glutBitmapCharacter); + CHECK_NAME(glutBitmapWidth); + CHECK_NAME(glutStrokeCharacter); + CHECK_NAME(glutStrokeWidth); + CHECK_NAME(glutBitmapLength); + CHECK_NAME(glutStrokeLength); + CHECK_NAME(glutWireSphere); + CHECK_NAME(glutSolidSphere); + CHECK_NAME(glutWireCone); + CHECK_NAME(glutSolidCone); + CHECK_NAME(glutWireCube); + CHECK_NAME(glutSolidCube); + CHECK_NAME(glutWireTorus); + CHECK_NAME(glutSolidTorus); + CHECK_NAME(glutWireDodecahedron); + CHECK_NAME(glutSolidDodecahedron); + CHECK_NAME(glutWireTeapot); + CHECK_NAME(glutSolidTeapot); + CHECK_NAME(glutWireOctahedron); + CHECK_NAME(glutSolidOctahedron); + CHECK_NAME(glutWireTetrahedron); + CHECK_NAME(glutSolidTetrahedron); + CHECK_NAME(glutWireIcosahedron); + CHECK_NAME(glutSolidIcosahedron); + CHECK_NAME(glutVideoResizeGet); + CHECK_NAME(glutSetupVideoResizing); + CHECK_NAME(glutStopVideoResizing); + CHECK_NAME(glutVideoResize); + CHECK_NAME(glutVideoPan); + CHECK_NAME(glutReportErrors); + CHECK_NAME(glutIgnoreKeyRepeat); + CHECK_NAME(glutSetKeyRepeat); +#if !defined(_WIN32_WCE) + CHECK_NAME(glutForceJoystickFunc); + CHECK_NAME(glutGameModeString); + CHECK_NAME(glutEnterGameMode); + CHECK_NAME(glutLeaveGameMode); + CHECK_NAME(glutGameModeGet); +#endif /* !defined(_WIN32_WCE) */ + /* freeglut extensions */ + CHECK_NAME(glutMainLoopEvent); + CHECK_NAME(glutLeaveMainLoop); + CHECK_NAME(glutCloseFunc); + CHECK_NAME(glutWMCloseFunc); + CHECK_NAME(glutMenuDestroyFunc); + CHECK_NAME(glutFullScreenToggle); + CHECK_NAME(glutLeaveFullScreen); + CHECK_NAME(glutSetOption); + CHECK_NAME(glutGetModeValues); + CHECK_NAME(glutSetWindowData); + CHECK_NAME(glutGetWindowData); + CHECK_NAME(glutSetMenuData); + CHECK_NAME(glutGetMenuData); + CHECK_NAME(glutBitmapHeight); + CHECK_NAME(glutStrokeHeight); + CHECK_NAME(glutBitmapString); + CHECK_NAME(glutStrokeString); + CHECK_NAME(glutWireRhombicDodecahedron); + CHECK_NAME(glutSolidRhombicDodecahedron); + CHECK_NAME(glutWireSierpinskiSponge); + CHECK_NAME(glutSolidSierpinskiSponge); + CHECK_NAME(glutWireCylinder); + CHECK_NAME(glutSolidCylinder); + CHECK_NAME(glutGetProcAddress); + CHECK_NAME(glutMouseWheelFunc); + CHECK_NAME(glutJoystickGetNumAxes); + CHECK_NAME(glutJoystickGetNumButtons); + CHECK_NAME(glutJoystickNotWorking); + CHECK_NAME(glutJoystickGetDeadBand); + CHECK_NAME(glutJoystickSetDeadBand); + CHECK_NAME(glutJoystickGetSaturation); + CHECK_NAME(glutJoystickSetSaturation); + CHECK_NAME(glutJoystickSetMinRange); + CHECK_NAME(glutJoystickSetMaxRange); + CHECK_NAME(glutJoystickSetCenter); + CHECK_NAME(glutJoystickGetMinRange); + CHECK_NAME(glutJoystickGetMaxRange); + CHECK_NAME(glutJoystickGetCenter); + CHECK_NAME(glutInitContextVersion); + CHECK_NAME(glutInitContextFlags); + CHECK_NAME(glutInitContextProfile); + CHECK_NAME(glutInitErrorFunc); + CHECK_NAME(glutInitWarningFunc); +#undef CHECK_NAME + + return NULL; +} + + +SFG_Proc fghGetProcAddress( const char *procName ) +{ +#if TARGET_HOST_MS_WINDOWS + return (SFG_Proc)wglGetProcAddress( ( LPCSTR )procName ); +#elif TARGET_HOST_POSIX_X11 && defined( GLX_ARB_get_proc_address ) + return (SFG_Proc)glXGetProcAddressARB( ( const GLubyte * )procName ); +#else + return NULL; +#endif +} + + +GLUTproc FGAPIENTRY +glutGetProcAddress( const char *procName ) +{ + GLUTproc p; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetProcAddress" ); + + /* Try GLUT functions first, then core GL functions */ + p = fghGetGLUTProcAddress( procName ); + return ( p != NULL ) ? p : fghGetProcAddress( procName ); +} diff --git a/examples/opengl-framework/freeglut/freeglut_font.c b/examples/opengl-framework/freeglut/freeglut_font.c new file mode 100644 index 00000000..e956d5b0 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_font.c @@ -0,0 +1,384 @@ +/* + * freeglut_font.c + * + * Bitmap and stroke fonts displaying. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 16 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* + * TODO BEFORE THE STABLE RELEASE: + * + * Test things out ... + */ + +/* -- IMPORT DECLARATIONS -------------------------------------------------- */ + +/* + * These are the font faces defined in freeglut_font_data.c file: + */ +extern SFG_Font fgFontFixed8x13; +extern SFG_Font fgFontFixed9x15; +extern SFG_Font fgFontHelvetica10; +extern SFG_Font fgFontHelvetica12; +extern SFG_Font fgFontHelvetica18; +extern SFG_Font fgFontTimesRoman10; +extern SFG_Font fgFontTimesRoman24; +extern SFG_StrokeFont fgStrokeRoman; +extern SFG_StrokeFont fgStrokeMonoRoman; + + +/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ + +/* + * Matches a font ID with a SFG_Font structure pointer. + * This was changed to match the GLUT header style. + */ +static SFG_Font* fghFontByID( void* font ) +{ + if( font == GLUT_BITMAP_8_BY_13 ) + return &fgFontFixed8x13; + if( font == GLUT_BITMAP_9_BY_15 ) + return &fgFontFixed9x15; + if( font == GLUT_BITMAP_HELVETICA_10 ) + return &fgFontHelvetica10; + if( font == GLUT_BITMAP_HELVETICA_12 ) + return &fgFontHelvetica12; + if( font == GLUT_BITMAP_HELVETICA_18 ) + return &fgFontHelvetica18; + if( font == GLUT_BITMAP_TIMES_ROMAN_10 ) + return &fgFontTimesRoman10; + if( font == GLUT_BITMAP_TIMES_ROMAN_24 ) + return &fgFontTimesRoman24; + + fgWarning( "font 0x%08x not found", font ); + return 0; +} + +/* + * Matches a font ID with a SFG_StrokeFont structure pointer. + * This was changed to match the GLUT header style. + */ +static SFG_StrokeFont* fghStrokeByID( void* font ) +{ + if( font == GLUT_STROKE_ROMAN ) + return &fgStrokeRoman; + if( font == GLUT_STROKE_MONO_ROMAN ) + return &fgStrokeMonoRoman; + + fgWarning( "stroke font 0x%08x not found", font ); + return 0; +} + + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * Draw a bitmap character + */ +void FGAPIENTRY glutBitmapCharacter( void* fontID, int character ) +{ + const GLubyte* face; + SFG_Font* font; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutBitmapCharacter" ); + font = fghFontByID( fontID ); + freeglut_return_if_fail( ( character >= 1 )&&( character < 256 ) ); + freeglut_return_if_fail( font ); + + /* + * Find the character we want to draw (???) + */ + face = font->Characters[ character ]; + + glPushClientAttrib( GL_CLIENT_PIXEL_STORE_BIT ); + glPixelStorei( GL_UNPACK_SWAP_BYTES, GL_FALSE ); + glPixelStorei( GL_UNPACK_LSB_FIRST, GL_FALSE ); + glPixelStorei( GL_UNPACK_ROW_LENGTH, 0 ); + glPixelStorei( GL_UNPACK_SKIP_ROWS, 0 ); + glPixelStorei( GL_UNPACK_SKIP_PIXELS, 0 ); + glPixelStorei( GL_UNPACK_ALIGNMENT, 1 ); + glBitmap( + face[ 0 ], font->Height, /* The bitmap's width and height */ + font->xorig, font->yorig, /* The origin in the font glyph */ + ( float )( face[ 0 ] ), 0.0, /* The raster advance -- inc. x,y */ + ( face + 1 ) /* The packed bitmap data... */ + ); + glPopClientAttrib( ); +} + +void FGAPIENTRY glutBitmapString( void* fontID, const unsigned char *string ) +{ + unsigned char c; + float x = 0.0f ; + SFG_Font* font; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutBitmapString" ); + font = fghFontByID( fontID ); + freeglut_return_if_fail( font ); + if ( !string || ! *string ) + return; + + glPushClientAttrib( GL_CLIENT_PIXEL_STORE_BIT ); + glPixelStorei( GL_UNPACK_SWAP_BYTES, GL_FALSE ); + glPixelStorei( GL_UNPACK_LSB_FIRST, GL_FALSE ); + glPixelStorei( GL_UNPACK_ROW_LENGTH, 0 ); + glPixelStorei( GL_UNPACK_SKIP_ROWS, 0 ); + glPixelStorei( GL_UNPACK_SKIP_PIXELS, 0 ); + glPixelStorei( GL_UNPACK_ALIGNMENT, 1 ); + + /* + * Step through the string, drawing each character. + * A newline will simply translate the next character's insertion + * point back to the start of the line and down one line. + */ + while( ( c = *string++) ) + if( c == '\n' ) + { + glBitmap ( 0, 0, 0, 0, -x, (float) -font->Height, NULL ); + x = 0.0f; + } + else /* Not an EOL, draw the bitmap character */ + { + const GLubyte* face = font->Characters[ c ]; + + glBitmap( + face[ 0 ], font->Height, /* Bitmap's width and height */ + font->xorig, font->yorig, /* The origin in the font glyph */ + ( float )( face[ 0 ] ), 0.0, /* The raster advance; inc. x,y */ + ( face + 1 ) /* The packed bitmap data... */ + ); + + x += ( float )( face[ 0 ] ); + } + + glPopClientAttrib( ); +} + +/* + * Returns the width in pixels of a font's character + */ +int FGAPIENTRY glutBitmapWidth( void* fontID, int character ) +{ + SFG_Font* font; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutBitmapWidth" ); + font = fghFontByID( fontID ); + freeglut_return_val_if_fail( character > 0 && character < 256, 0 ); + freeglut_return_val_if_fail( font, 0 ); + return *( font->Characters[ character ] ); +} + +/* + * Return the width of a string drawn using a bitmap font + */ +int FGAPIENTRY glutBitmapLength( void* fontID, const unsigned char* string ) +{ + unsigned char c; + int length = 0, this_line_length = 0; + SFG_Font* font; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutBitmapLength" ); + font = fghFontByID( fontID ); + freeglut_return_val_if_fail( font, 0 ); + if ( !string || ! *string ) + return 0; + + while( ( c = *string++) ) + { + if( c != '\n' )/* Not an EOL, increment length of line */ + this_line_length += *( font->Characters[ c ]); + else /* EOL; reset the length of this line */ + { + if( length < this_line_length ) + length = this_line_length; + this_line_length = 0; + } + } + if ( length < this_line_length ) + length = this_line_length; + + return length; +} + +/* + * Returns the height of a bitmap font + */ +int FGAPIENTRY glutBitmapHeight( void* fontID ) +{ + SFG_Font* font; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutBitmapHeight" ); + font = fghFontByID( fontID ); + freeglut_return_val_if_fail( font, 0 ); + return font->Height; +} + +/* + * Draw a stroke character + */ +void FGAPIENTRY glutStrokeCharacter( void* fontID, int character ) +{ + const SFG_StrokeChar *schar; + const SFG_StrokeStrip *strip; + int i, j; + SFG_StrokeFont* font; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutStrokeCharacter" ); + font = fghStrokeByID( fontID ); + freeglut_return_if_fail( character >= 0 ); + freeglut_return_if_fail( character < font->Quantity ); + freeglut_return_if_fail( font ); + + schar = font->Characters[ character ]; + freeglut_return_if_fail( schar ); + strip = schar->Strips; + + for( i = 0; i < schar->Number; i++, strip++ ) + { + glBegin( GL_LINE_STRIP ); + for( j = 0; j < strip->Number; j++ ) + glVertex2f( strip->Vertices[ j ].X, strip->Vertices[ j ].Y ); + glEnd( ); + glBegin( GL_POINTS ); + for( j = 0; j < strip->Number; j++ ) + glVertex2f( strip->Vertices[ j ].X, strip->Vertices[ j ].Y ); + glEnd( ); + } + glTranslatef( schar->Right, 0.0, 0.0 ); +} + +void FGAPIENTRY glutStrokeString( void* fontID, const unsigned char *string ) +{ + unsigned char c; + int i, j; + float length = 0.0; + SFG_StrokeFont* font; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutStrokeString" ); + font = fghStrokeByID( fontID ); + freeglut_return_if_fail( font ); + if ( !string || ! *string ) + return; + + /* + * Step through the string, drawing each character. + * A newline will simply translate the next character's insertion + * point back to the start of the line and down one line. + */ + while( ( c = *string++) ) + if( c < font->Quantity ) + { + if( c == '\n' ) + { + glTranslatef ( -length, -( float )( font->Height ), 0.0 ); + length = 0.0; + } + else /* Not an EOL, draw the bitmap character */ + { + const SFG_StrokeChar *schar = font->Characters[ c ]; + if( schar ) + { + const SFG_StrokeStrip *strip = schar->Strips; + + for( i = 0; i < schar->Number; i++, strip++ ) + { + glBegin( GL_LINE_STRIP ); + for( j = 0; j < strip->Number; j++ ) + glVertex2f( strip->Vertices[ j ].X, + strip->Vertices[ j ].Y); + + glEnd( ); + } + + length += schar->Right; + glTranslatef( schar->Right, 0.0, 0.0 ); + } + } + } +} + +/* + * Return the width in pixels of a stroke character + */ +int FGAPIENTRY glutStrokeWidth( void* fontID, int character ) +{ + const SFG_StrokeChar *schar; + SFG_StrokeFont* font; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutStrokeWidth" ); + font = fghStrokeByID( fontID ); + freeglut_return_val_if_fail( ( character >= 0 ) && + ( character < font->Quantity ), + 0 + ); + freeglut_return_val_if_fail( font, 0 ); + schar = font->Characters[ character ]; + freeglut_return_val_if_fail( schar, 0 ); + + return ( int )( schar->Right + 0.5 ); +} + +/* + * Return the width of a string drawn using a stroke font + */ +int FGAPIENTRY glutStrokeLength( void* fontID, const unsigned char* string ) +{ + unsigned char c; + float length = 0.0; + float this_line_length = 0.0; + SFG_StrokeFont* font; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutStrokeLength" ); + font = fghStrokeByID( fontID ); + freeglut_return_val_if_fail( font, 0 ); + if ( !string || ! *string ) + return 0; + + while( ( c = *string++) ) + if( c < font->Quantity ) + { + if( c == '\n' ) /* EOL; reset the length of this line */ + { + if( length < this_line_length ) + length = this_line_length; + this_line_length = 0.0; + } + else /* Not an EOL, increment the length of this line */ + { + const SFG_StrokeChar *schar = font->Characters[ c ]; + if( schar ) + this_line_length += schar->Right; + } + } + if( length < this_line_length ) + length = this_line_length; + return( int )( length + 0.5 ); +} + +/* + * Returns the height of a stroke font + */ +GLfloat FGAPIENTRY glutStrokeHeight( void* fontID ) +{ + SFG_StrokeFont* font; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutStrokeHeight" ); + font = fghStrokeByID( fontID ); + freeglut_return_val_if_fail( font, 0.0 ); + return font->Height; +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_font_data.c b/examples/opengl-framework/freeglut/freeglut_font_data.c new file mode 100644 index 00000000..35dfc797 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_font_data.c @@ -0,0 +1,2020 @@ +/* + \file og_font_data.c + \brief Bitmapped font data for OpenGLUT fonts. +*/ + +/* + * This file has been automatically generated by the + * genfonts utility. + * + * The legal status of this file is a bit vague. The font glyphs + * themselves come from XFree86 v4.3.0 (as of this writing), and as + * part of the X server may be subject to the XFree86 copyrights. + * The original freeglut fonts were extracted by a utility written + * by Pawel W. Olszta (see below) and the generated fonts contained + * his copyright exclusively. Steve Baker asserts that Pawel + * assigned intellectual property rights to Steve Baker. Steve + * Baker also asserts that fonts cannot be copyrighted. He has + * neither stripped the copyright from the freeglut fonts nor + * formally retitled anything in his name. Since that time, the + * OpenGLUT project has branched from freeglut, and has made + * necessary modifications to Pawel's ``genfonts'' utility. + * To that extent, OpenGLUT may have some title to this file. + * What is fairly clear is that the font data is licensed under + * the XFree86 license (which is variously termed ``XFree'' and + * ``MIT'' by the freeglut project). It is believed that all + * title holders wish this file to be as useful as possible, and + * that either the ``XFree'' or ``MIT'' license works. + * + * Portions copyright (c) 2004, the OpenGLUT project contributors. + * OpenGLUT branched from freeglut in February, 2004. + * + * Copyright (c) 1999-2000 by Pawel W. Olszta + * Written by Pawel W. Olszta, + * + * 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 Sotware. + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* + * The following bitmapped fonts are defined in this file: + * + * 1. fgFontFixed8x13 + * -misc-fixed-medium-r-normal--13-120-75-75-C-80-iso8859-1 + * 2. fgFontFixed9x15 + * -misc-fixed-medium-r-normal--15-140-75-75-C-90-iso8859-1 + * 3. fgFontHelvetica10 + * -adobe-helvetica-medium-r-normal--10-100-75-75-p-56-iso8859-1 + * 4. fgFontHelvetica12 + * -adobe-helvetica-medium-r-normal--12-120-75-75-p-67-iso8859-1 + * 5. fgFontHelvetica18 + * -adobe-helvetica-medium-r-normal--18-180-75-75-p-98-iso8859-1 + * 6. fgFontTimesRoman10 + * -adobe-times-medium-r-normal--10-100-75-75-p-54-iso8859-1 + * 7. fgFontTimesRoman24 + * -adobe-times-medium-r-normal--24-240-75-75-p-124-iso8859-1 + */ + +static const GLubyte Fixed8x13_Character_000[] = { 8, 0, 0, 0,170, 0,130, 0,130, 0,130, 0,170, 0, 0}; +static const GLubyte Fixed8x13_Character_001[] = { 8, 0, 0, 0, 0, 16, 56,124,254,124, 56, 16, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_002[] = { 8, 0,170, 85,170, 85,170, 85,170, 85,170, 85,170, 85,170}; +static const GLubyte Fixed8x13_Character_003[] = { 8, 0, 0, 0, 4, 4, 4, 4,174,160,224,160,160, 0, 0}; +static const GLubyte Fixed8x13_Character_004[] = { 8, 0, 0, 0, 8, 8, 12, 8,142,128,192,128,224, 0, 0}; +static const GLubyte Fixed8x13_Character_005[] = { 8, 0, 0, 0, 10, 10, 12, 10,108,128,128,128, 96, 0, 0}; +static const GLubyte Fixed8x13_Character_006[] = { 8, 0, 0, 0, 8, 8, 12, 8,238,128,128,128,128, 0, 0}; +static const GLubyte Fixed8x13_Character_007[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 24, 36, 36, 24, 0, 0}; +static const GLubyte Fixed8x13_Character_008[] = { 8, 0, 0, 0, 0,124, 0, 16, 16,124, 16, 16, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_009[] = { 8, 0, 0, 0, 14, 8, 8, 8,168,160,160,160,192, 0, 0}; +static const GLubyte Fixed8x13_Character_010[] = { 8, 0, 0, 0, 4, 4, 4, 4, 46, 80, 80,136,136, 0, 0}; +static const GLubyte Fixed8x13_Character_011[] = { 8, 0, 0, 0, 0, 0, 0, 0,240, 16, 16, 16, 16, 16, 16}; +static const GLubyte Fixed8x13_Character_012[] = { 8, 0, 16, 16, 16, 16, 16, 16,240, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_013[] = { 8, 0, 16, 16, 16, 16, 16, 16, 31, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_014[] = { 8, 0, 0, 0, 0, 0, 0, 0, 31, 16, 16, 16, 16, 16, 16}; +static const GLubyte Fixed8x13_Character_015[] = { 8, 0, 16, 16, 16, 16, 16, 16,255, 16, 16, 16, 16, 16, 16}; +static const GLubyte Fixed8x13_Character_016[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255}; +static const GLubyte Fixed8x13_Character_017[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_018[] = { 8, 0, 0, 0, 0, 0, 0, 0,255, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_019[] = { 8, 0, 0, 0, 0,255, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_020[] = { 8, 0,255, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_021[] = { 8, 0, 16, 16, 16, 16, 16, 16, 31, 16, 16, 16, 16, 16, 16}; +static const GLubyte Fixed8x13_Character_022[] = { 8, 0, 16, 16, 16, 16, 16, 16,240, 16, 16, 16, 16, 16, 16}; +static const GLubyte Fixed8x13_Character_023[] = { 8, 0, 0, 0, 0, 0, 0, 0,255, 16, 16, 16, 16, 16, 16}; +static const GLubyte Fixed8x13_Character_024[] = { 8, 0, 16, 16, 16, 16, 16, 16,255, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_025[] = { 8, 0, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16}; +static const GLubyte Fixed8x13_Character_026[] = { 8, 0, 0, 0,254, 0, 14, 48,192, 48, 14, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_027[] = { 8, 0, 0, 0,254, 0,224, 24, 6, 24,224, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_028[] = { 8, 0, 0, 0, 68, 68, 68, 68, 68,254, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_029[] = { 8, 0, 0, 0, 32, 32,126, 16, 8,126, 4, 4, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_030[] = { 8, 0, 0, 0,220, 98, 32, 32, 32,112, 32, 34, 28, 0, 0}; +static const GLubyte Fixed8x13_Character_031[] = { 8, 0, 0, 0, 0, 0, 0, 0, 24, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_032[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_033[] = { 8, 0, 0, 0, 16, 0, 16, 16, 16, 16, 16, 16, 16, 0, 0}; +static const GLubyte Fixed8x13_Character_034[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 36, 36, 36, 0, 0}; +static const GLubyte Fixed8x13_Character_035[] = { 8, 0, 0, 0, 0, 36, 36,126, 36,126, 36, 36, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_036[] = { 8, 0, 0, 0, 16,120, 20, 20, 56, 80, 80, 60, 16, 0, 0}; +static const GLubyte Fixed8x13_Character_037[] = { 8, 0, 0, 0, 68, 42, 36, 16, 8, 8, 36, 82, 34, 0, 0}; +static const GLubyte Fixed8x13_Character_038[] = { 8, 0, 0, 0, 58, 68, 74, 48, 72, 72, 48, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_039[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 64, 48, 56, 0, 0}; +static const GLubyte Fixed8x13_Character_040[] = { 8, 0, 0, 0, 4, 8, 8, 16, 16, 16, 8, 8, 4, 0, 0}; +static const GLubyte Fixed8x13_Character_041[] = { 8, 0, 0, 0, 32, 16, 16, 8, 8, 8, 16, 16, 32, 0, 0}; +static const GLubyte Fixed8x13_Character_042[] = { 8, 0, 0, 0, 0, 0, 36, 24,126, 24, 36, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_043[] = { 8, 0, 0, 0, 0, 0, 16, 16,124, 16, 16, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_044[] = { 8, 0, 0, 64, 48, 56, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_045[] = { 8, 0, 0, 0, 0, 0, 0, 0,126, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_046[] = { 8, 0, 0, 16, 56, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_047[] = { 8, 0, 0, 0,128,128, 64, 32, 16, 8, 4, 2, 2, 0, 0}; +static const GLubyte Fixed8x13_Character_048[] = { 8, 0, 0, 0, 24, 36, 66, 66, 66, 66, 66, 36, 24, 0, 0}; +static const GLubyte Fixed8x13_Character_049[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16, 80, 48, 16, 0, 0}; +static const GLubyte Fixed8x13_Character_050[] = { 8, 0, 0, 0,126, 64, 32, 24, 4, 2, 66, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_051[] = { 8, 0, 0, 0, 60, 66, 2, 2, 28, 8, 4, 2,126, 0, 0}; +static const GLubyte Fixed8x13_Character_052[] = { 8, 0, 0, 0, 4, 4,126, 68, 68, 36, 20, 12, 4, 0, 0}; +static const GLubyte Fixed8x13_Character_053[] = { 8, 0, 0, 0, 60, 66, 2, 2, 98, 92, 64, 64,126, 0, 0}; +static const GLubyte Fixed8x13_Character_054[] = { 8, 0, 0, 0, 60, 66, 66, 98, 92, 64, 64, 32, 28, 0, 0}; +static const GLubyte Fixed8x13_Character_055[] = { 8, 0, 0, 0, 32, 32, 16, 16, 8, 8, 4, 2,126, 0, 0}; +static const GLubyte Fixed8x13_Character_056[] = { 8, 0, 0, 0, 60, 66, 66, 66, 60, 66, 66, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_057[] = { 8, 0, 0, 0, 56, 4, 2, 2, 58, 70, 66, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_058[] = { 8, 0, 0, 16, 56, 16, 0, 0, 16, 56, 16, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_059[] = { 8, 0, 0, 64, 48, 56, 0, 0, 16, 56, 16, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_060[] = { 8, 0, 0, 0, 2, 4, 8, 16, 32, 16, 8, 4, 2, 0, 0}; +static const GLubyte Fixed8x13_Character_061[] = { 8, 0, 0, 0, 0, 0,126, 0, 0,126, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_062[] = { 8, 0, 0, 0, 64, 32, 16, 8, 4, 8, 16, 32, 64, 0, 0}; +static const GLubyte Fixed8x13_Character_063[] = { 8, 0, 0, 0, 8, 0, 8, 8, 4, 2, 66, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_064[] = { 8, 0, 0, 0, 60, 64, 74, 86, 82, 78, 66, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_065[] = { 8, 0, 0, 0, 66, 66, 66,126, 66, 66, 66, 36, 24, 0, 0}; +static const GLubyte Fixed8x13_Character_066[] = { 8, 0, 0, 0,252, 66, 66, 66,124, 66, 66, 66,252, 0, 0}; +static const GLubyte Fixed8x13_Character_067[] = { 8, 0, 0, 0, 60, 66, 64, 64, 64, 64, 64, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_068[] = { 8, 0, 0, 0,252, 66, 66, 66, 66, 66, 66, 66,252, 0, 0}; +static const GLubyte Fixed8x13_Character_069[] = { 8, 0, 0, 0,126, 64, 64, 64,120, 64, 64, 64,126, 0, 0}; +static const GLubyte Fixed8x13_Character_070[] = { 8, 0, 0, 0, 64, 64, 64, 64,120, 64, 64, 64,126, 0, 0}; +static const GLubyte Fixed8x13_Character_071[] = { 8, 0, 0, 0, 58, 70, 66, 78, 64, 64, 64, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_072[] = { 8, 0, 0, 0, 66, 66, 66, 66,126, 66, 66, 66, 66, 0, 0}; +static const GLubyte Fixed8x13_Character_073[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16, 16, 16,124, 0, 0}; +static const GLubyte Fixed8x13_Character_074[] = { 8, 0, 0, 0, 56, 68, 4, 4, 4, 4, 4, 4, 31, 0, 0}; +static const GLubyte Fixed8x13_Character_075[] = { 8, 0, 0, 0, 66, 68, 72, 80, 96, 80, 72, 68, 66, 0, 0}; +static const GLubyte Fixed8x13_Character_076[] = { 8, 0, 0, 0,126, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0}; +static const GLubyte Fixed8x13_Character_077[] = { 8, 0, 0, 0,130,130,130,146,146,170,198,130,130, 0, 0}; +static const GLubyte Fixed8x13_Character_078[] = { 8, 0, 0, 0, 66, 66, 66, 70, 74, 82, 98, 66, 66, 0, 0}; +static const GLubyte Fixed8x13_Character_079[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_080[] = { 8, 0, 0, 0, 64, 64, 64, 64,124, 66, 66, 66,124, 0, 0}; +static const GLubyte Fixed8x13_Character_081[] = { 8, 0, 0, 2, 60, 74, 82, 66, 66, 66, 66, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_082[] = { 8, 0, 0, 0, 66, 68, 72, 80,124, 66, 66, 66,124, 0, 0}; +static const GLubyte Fixed8x13_Character_083[] = { 8, 0, 0, 0, 60, 66, 2, 2, 60, 64, 64, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_084[] = { 8, 0, 0, 0, 16, 16, 16, 16, 16, 16, 16, 16,254, 0, 0}; +static const GLubyte Fixed8x13_Character_085[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 0}; +static const GLubyte Fixed8x13_Character_086[] = { 8, 0, 0, 0, 16, 40, 40, 40, 68, 68, 68,130,130, 0, 0}; +static const GLubyte Fixed8x13_Character_087[] = { 8, 0, 0, 0, 68,170,146,146,146,130,130,130,130, 0, 0}; +static const GLubyte Fixed8x13_Character_088[] = { 8, 0, 0, 0,130,130, 68, 40, 16, 40, 68,130,130, 0, 0}; +static const GLubyte Fixed8x13_Character_089[] = { 8, 0, 0, 0, 16, 16, 16, 16, 16, 40, 68,130,130, 0, 0}; +static const GLubyte Fixed8x13_Character_090[] = { 8, 0, 0, 0,126, 64, 64, 32, 16, 8, 4, 2,126, 0, 0}; +static const GLubyte Fixed8x13_Character_091[] = { 8, 0, 0, 0, 60, 32, 32, 32, 32, 32, 32, 32, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_092[] = { 8, 0, 0, 0, 2, 2, 4, 8, 16, 32, 64,128,128, 0, 0}; +static const GLubyte Fixed8x13_Character_093[] = { 8, 0, 0, 0,120, 8, 8, 8, 8, 8, 8, 8,120, 0, 0}; +static const GLubyte Fixed8x13_Character_094[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 68, 40, 16, 0, 0}; +static const GLubyte Fixed8x13_Character_095[] = { 8, 0, 0,254, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_096[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4, 24, 56, 0, 0}; +static const GLubyte Fixed8x13_Character_097[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_098[] = { 8, 0, 0, 0, 92, 98, 66, 66, 98, 92, 64, 64, 64, 0, 0}; +static const GLubyte Fixed8x13_Character_099[] = { 8, 0, 0, 0, 60, 66, 64, 64, 66, 60, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_100[] = { 8, 0, 0, 0, 58, 70, 66, 66, 70, 58, 2, 2, 2, 0, 0}; +static const GLubyte Fixed8x13_Character_101[] = { 8, 0, 0, 0, 60, 66, 64,126, 66, 60, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_102[] = { 8, 0, 0, 0, 32, 32, 32, 32,124, 32, 32, 34, 28, 0, 0}; +static const GLubyte Fixed8x13_Character_103[] = { 8, 0, 60, 66, 60, 64, 56, 68, 68, 58, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_104[] = { 8, 0, 0, 0, 66, 66, 66, 66, 98, 92, 64, 64, 64, 0, 0}; +static const GLubyte Fixed8x13_Character_105[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 48, 0, 16, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_106[] = { 8, 0, 56, 68, 68, 4, 4, 4, 4, 12, 0, 4, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_107[] = { 8, 0, 0, 0, 66, 68, 72,112, 72, 68, 64, 64, 64, 0, 0}; +static const GLubyte Fixed8x13_Character_108[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16, 16, 16, 48, 0, 0}; +static const GLubyte Fixed8x13_Character_109[] = { 8, 0, 0, 0,130,146,146,146,146,236, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_110[] = { 8, 0, 0, 0, 66, 66, 66, 66, 98, 92, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_111[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_112[] = { 8, 0, 64, 64, 64, 92, 98, 66, 98, 92, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_113[] = { 8, 0, 2, 2, 2, 58, 70, 66, 70, 58, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_114[] = { 8, 0, 0, 0, 32, 32, 32, 32, 34, 92, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_115[] = { 8, 0, 0, 0, 60, 66, 12, 48, 66, 60, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_116[] = { 8, 0, 0, 0, 28, 34, 32, 32, 32,124, 32, 32, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_117[] = { 8, 0, 0, 0, 58, 68, 68, 68, 68, 68, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_118[] = { 8, 0, 0, 0, 16, 40, 40, 68, 68, 68, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_119[] = { 8, 0, 0, 0, 68,170,146,146,130,130, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_120[] = { 8, 0, 0, 0, 66, 36, 24, 24, 36, 66, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_121[] = { 8, 0, 60, 66, 2, 58, 70, 66, 66, 66, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_122[] = { 8, 0, 0, 0,126, 32, 16, 8, 4,126, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_123[] = { 8, 0, 0, 0, 14, 16, 16, 8, 48, 8, 16, 16, 14, 0, 0}; +static const GLubyte Fixed8x13_Character_124[] = { 8, 0, 0, 0, 16, 16, 16, 16, 16, 16, 16, 16, 16, 0, 0}; +static const GLubyte Fixed8x13_Character_125[] = { 8, 0, 0, 0,112, 8, 8, 16, 12, 16, 8, 8,112, 0, 0}; +static const GLubyte Fixed8x13_Character_126[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 72, 84, 36, 0, 0}; +static const GLubyte Fixed8x13_Character_127[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_128[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_129[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_130[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_131[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_132[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_133[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_134[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_135[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_136[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_137[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_138[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_139[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_140[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_141[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_142[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_143[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_144[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_145[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_146[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_147[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_148[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_149[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_150[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_151[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_152[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_153[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_154[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_155[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_156[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_157[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_158[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_159[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_160[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_161[] = { 8, 0, 0, 0, 16, 16, 16, 16, 16, 16, 16, 0, 16, 0, 0}; +static const GLubyte Fixed8x13_Character_162[] = { 8, 0, 0, 0, 0, 16, 56, 84, 80, 80, 84, 56, 16, 0, 0}; +static const GLubyte Fixed8x13_Character_163[] = { 8, 0, 0, 0,220, 98, 32, 32, 32,112, 32, 34, 28, 0, 0}; +static const GLubyte Fixed8x13_Character_164[] = { 8, 0, 0, 0, 0, 66, 60, 36, 36, 60, 66, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_165[] = { 8, 0, 0, 0, 16, 16,124, 16,124, 40, 68,130,130, 0, 0}; +static const GLubyte Fixed8x13_Character_166[] = { 8, 0, 0, 0, 16, 16, 16, 16, 0, 16, 16, 16, 16, 0, 0}; +static const GLubyte Fixed8x13_Character_167[] = { 8, 0, 0, 0, 24, 36, 4, 24, 36, 36, 24, 32, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_168[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,108, 0, 0}; +static const GLubyte Fixed8x13_Character_169[] = { 8, 0, 0, 0, 0, 56, 68,146,170,162,170,146, 68, 56, 0}; +static const GLubyte Fixed8x13_Character_170[] = { 8, 0, 0, 0, 0, 0,124, 0, 60, 68, 60, 4, 56, 0, 0}; +static const GLubyte Fixed8x13_Character_171[] = { 8, 0, 0, 0, 0, 18, 36, 72,144, 72, 36, 18, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_172[] = { 8, 0, 0, 0, 0, 2, 2, 2,126, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_173[] = { 8, 0, 0, 0, 0, 0, 0, 0, 60, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_174[] = { 8, 0, 0, 0, 0, 56, 68,170,178,170,170,146, 68, 56, 0}; +static const GLubyte Fixed8x13_Character_175[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 0}; +static const GLubyte Fixed8x13_Character_176[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 24, 36, 36, 24, 0, 0}; +static const GLubyte Fixed8x13_Character_177[] = { 8, 0, 0, 0, 0,124, 0, 16, 16,124, 16, 16, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_178[] = { 8, 0, 0, 0, 0, 0, 0, 0,120, 64, 48, 8, 72, 48, 0}; +static const GLubyte Fixed8x13_Character_179[] = { 8, 0, 0, 0, 0, 0, 0, 0, 48, 72, 8, 16, 72, 48, 0}; +static const GLubyte Fixed8x13_Character_180[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_181[] = { 8, 0, 0, 64, 90,102, 66, 66, 66, 66, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_182[] = { 8, 0, 0, 0, 20, 20, 20, 20, 52,116,116,116, 62, 0, 0}; +static const GLubyte Fixed8x13_Character_183[] = { 8, 0, 0, 0, 0, 0, 0, 0, 24, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_184[] = { 8, 0, 24, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_185[] = { 8, 0, 0, 0, 0, 0, 0, 0,112, 32, 32, 32, 96, 32, 0}; +static const GLubyte Fixed8x13_Character_186[] = { 8, 0, 0, 0, 0, 0, 0,120, 0, 48, 72, 72, 48, 0, 0}; +static const GLubyte Fixed8x13_Character_187[] = { 8, 0, 0, 0, 0,144, 72, 36, 18, 36, 72,144, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_188[] = { 8, 0, 0, 0, 6, 26, 18, 10,230, 66, 64, 64,192, 64, 0}; +static const GLubyte Fixed8x13_Character_189[] = { 8, 0, 0, 0, 30, 16, 12, 2,242, 76, 64, 64,192, 64, 0}; +static const GLubyte Fixed8x13_Character_190[] = { 8, 0, 0, 0, 6, 26, 18, 10,102,146, 16, 32,144, 96, 0}; +static const GLubyte Fixed8x13_Character_191[] = { 8, 0, 0, 0, 60, 66, 66, 64, 32, 16, 16, 0, 16, 0, 0}; +static const GLubyte Fixed8x13_Character_192[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 0, 8, 16, 0}; +static const GLubyte Fixed8x13_Character_193[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_194[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 0, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_195[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 0, 76, 50, 0}; +static const GLubyte Fixed8x13_Character_196[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 0, 36, 36, 0}; +static const GLubyte Fixed8x13_Character_197[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 24, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_198[] = { 8, 0, 0, 0,158,144,144,240,156,144,144,144,110, 0, 0}; +static const GLubyte Fixed8x13_Character_199[] = { 8, 0, 16, 8, 60, 66, 64, 64, 64, 64, 64, 66, 60, 0, 0}; +static const GLubyte Fixed8x13_Character_200[] = { 8, 0, 0, 0,126, 64, 64,120, 64, 64,126, 0, 8, 16, 0}; +static const GLubyte Fixed8x13_Character_201[] = { 8, 0, 0, 0,126, 64, 64,120, 64, 64,126, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_202[] = { 8, 0, 0, 0,126, 64, 64,120, 64, 64,126, 0, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_203[] = { 8, 0, 0, 0,126, 64, 64,120, 64, 64,126, 0, 36, 36, 0}; +static const GLubyte Fixed8x13_Character_204[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16,124, 0, 16, 32, 0}; +static const GLubyte Fixed8x13_Character_205[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16,124, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_206[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16,124, 0, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_207[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16,124, 0, 40, 40, 0}; +static const GLubyte Fixed8x13_Character_208[] = { 8, 0, 0, 0,120, 68, 66, 66,226, 66, 66, 68,120, 0, 0}; +static const GLubyte Fixed8x13_Character_209[] = { 8, 0, 0, 0,130,134,138,146,162,194,130, 0,152,100, 0}; +static const GLubyte Fixed8x13_Character_210[] = { 8, 0, 0, 0,124,130,130,130,130,130,124, 0, 16, 32, 0}; +static const GLubyte Fixed8x13_Character_211[] = { 8, 0, 0, 0,124,130,130,130,130,130,124, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_212[] = { 8, 0, 0, 0,124,130,130,130,130,130,124, 0, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_213[] = { 8, 0, 0, 0,124,130,130,130,130,130,124, 0,152,100, 0}; +static const GLubyte Fixed8x13_Character_214[] = { 8, 0, 0, 0,124,130,130,130,130,130,124, 0, 40, 40, 0}; +static const GLubyte Fixed8x13_Character_215[] = { 8, 0, 0, 0, 0, 66, 36, 24, 24, 36, 66, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_216[] = { 8, 0, 0, 64, 60, 98, 82, 82, 82, 74, 74, 70, 60, 2, 0}; +static const GLubyte Fixed8x13_Character_217[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 0, 8, 16, 0}; +static const GLubyte Fixed8x13_Character_218[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_219[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 0, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_220[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 0, 36, 36, 0}; +static const GLubyte Fixed8x13_Character_221[] = { 8, 0, 0, 0, 16, 16, 16, 16, 40, 68, 68, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_222[] = { 8, 0, 0, 0, 64, 64, 64,124, 66, 66, 66,124, 64, 0, 0}; +static const GLubyte Fixed8x13_Character_223[] = { 8, 0, 0, 0, 92, 66, 66, 76, 80, 72, 68, 68, 56, 0, 0}; +static const GLubyte Fixed8x13_Character_224[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 8, 16, 0}; +static const GLubyte Fixed8x13_Character_225[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 8, 4, 0}; +static const GLubyte Fixed8x13_Character_226[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_227[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 76, 50, 0}; +static const GLubyte Fixed8x13_Character_228[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 36, 36, 0}; +static const GLubyte Fixed8x13_Character_229[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 24, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_230[] = { 8, 0, 0, 0,108,146,144,124, 18,108, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_231[] = { 8, 0, 16, 8, 60, 66, 64, 64, 66, 60, 0, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_232[] = { 8, 0, 0, 0, 60, 66, 64,126, 66, 60, 0, 0, 8, 16, 0}; +static const GLubyte Fixed8x13_Character_233[] = { 8, 0, 0, 0, 60, 66, 64,126, 66, 60, 0, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_234[] = { 8, 0, 0, 0, 60, 66, 64,126, 66, 60, 0, 0, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_235[] = { 8, 0, 0, 0, 60, 66, 64,126, 66, 60, 0, 0, 36, 36, 0}; +static const GLubyte Fixed8x13_Character_236[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 48, 0, 0, 16, 32, 0}; +static const GLubyte Fixed8x13_Character_237[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 48, 0, 0, 32, 16, 0}; +static const GLubyte Fixed8x13_Character_238[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 48, 0, 0, 72, 48, 0}; +static const GLubyte Fixed8x13_Character_239[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 48, 0, 0, 40, 40, 0}; +static const GLubyte Fixed8x13_Character_240[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 4, 40, 24, 36, 0}; +static const GLubyte Fixed8x13_Character_241[] = { 8, 0, 0, 0, 66, 66, 66, 66, 98, 92, 0, 0, 76, 50, 0}; +static const GLubyte Fixed8x13_Character_242[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 16, 32, 0}; +static const GLubyte Fixed8x13_Character_243[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_244[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_245[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 76, 50, 0}; +static const GLubyte Fixed8x13_Character_246[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 36, 36, 0}; +static const GLubyte Fixed8x13_Character_247[] = { 8, 0, 0, 0, 0, 16, 16, 0,124, 0, 16, 16, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_248[] = { 8, 0, 0, 64, 60, 98, 82, 74, 70, 60, 2, 0, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_249[] = { 8, 0, 0, 0, 58, 68, 68, 68, 68, 68, 0, 0, 16, 32, 0}; +static const GLubyte Fixed8x13_Character_250[] = { 8, 0, 0, 0, 58, 68, 68, 68, 68, 68, 0, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_251[] = { 8, 0, 0, 0, 58, 68, 68, 68, 68, 68, 0, 0, 36, 24, 0}; +static const GLubyte Fixed8x13_Character_252[] = { 8, 0, 0, 0, 58, 68, 68, 68, 68, 68, 0, 0, 40, 40, 0}; +static const GLubyte Fixed8x13_Character_253[] = { 8, 0, 60, 66, 2, 58, 70, 66, 66, 66, 0, 0, 16, 8, 0}; +static const GLubyte Fixed8x13_Character_254[] = { 8, 0, 64, 64, 92, 98, 66, 66, 98, 92, 64, 64, 0, 0, 0}; +static const GLubyte Fixed8x13_Character_255[] = { 8, 0, 60, 66, 2, 58, 70, 66, 66, 66, 0, 0, 36, 36, 0}; + +/* The font characters mapping: */ +static const GLubyte* Fixed8x13_Character_Map[] = {Fixed8x13_Character_000,Fixed8x13_Character_001,Fixed8x13_Character_002,Fixed8x13_Character_003,Fixed8x13_Character_004,Fixed8x13_Character_005,Fixed8x13_Character_006,Fixed8x13_Character_007,Fixed8x13_Character_008,Fixed8x13_Character_009,Fixed8x13_Character_010,Fixed8x13_Character_011,Fixed8x13_Character_012,Fixed8x13_Character_013,Fixed8x13_Character_014,Fixed8x13_Character_015, + Fixed8x13_Character_016,Fixed8x13_Character_017,Fixed8x13_Character_018,Fixed8x13_Character_019,Fixed8x13_Character_020,Fixed8x13_Character_021,Fixed8x13_Character_022,Fixed8x13_Character_023,Fixed8x13_Character_024,Fixed8x13_Character_025,Fixed8x13_Character_026,Fixed8x13_Character_027,Fixed8x13_Character_028,Fixed8x13_Character_029,Fixed8x13_Character_030,Fixed8x13_Character_031, + Fixed8x13_Character_032,Fixed8x13_Character_033,Fixed8x13_Character_034,Fixed8x13_Character_035,Fixed8x13_Character_036,Fixed8x13_Character_037,Fixed8x13_Character_038,Fixed8x13_Character_039,Fixed8x13_Character_040,Fixed8x13_Character_041,Fixed8x13_Character_042,Fixed8x13_Character_043,Fixed8x13_Character_044,Fixed8x13_Character_045,Fixed8x13_Character_046,Fixed8x13_Character_047, + Fixed8x13_Character_048,Fixed8x13_Character_049,Fixed8x13_Character_050,Fixed8x13_Character_051,Fixed8x13_Character_052,Fixed8x13_Character_053,Fixed8x13_Character_054,Fixed8x13_Character_055,Fixed8x13_Character_056,Fixed8x13_Character_057,Fixed8x13_Character_058,Fixed8x13_Character_059,Fixed8x13_Character_060,Fixed8x13_Character_061,Fixed8x13_Character_062,Fixed8x13_Character_063, + Fixed8x13_Character_064,Fixed8x13_Character_065,Fixed8x13_Character_066,Fixed8x13_Character_067,Fixed8x13_Character_068,Fixed8x13_Character_069,Fixed8x13_Character_070,Fixed8x13_Character_071,Fixed8x13_Character_072,Fixed8x13_Character_073,Fixed8x13_Character_074,Fixed8x13_Character_075,Fixed8x13_Character_076,Fixed8x13_Character_077,Fixed8x13_Character_078,Fixed8x13_Character_079, + Fixed8x13_Character_080,Fixed8x13_Character_081,Fixed8x13_Character_082,Fixed8x13_Character_083,Fixed8x13_Character_084,Fixed8x13_Character_085,Fixed8x13_Character_086,Fixed8x13_Character_087,Fixed8x13_Character_088,Fixed8x13_Character_089,Fixed8x13_Character_090,Fixed8x13_Character_091,Fixed8x13_Character_092,Fixed8x13_Character_093,Fixed8x13_Character_094,Fixed8x13_Character_095, + Fixed8x13_Character_096,Fixed8x13_Character_097,Fixed8x13_Character_098,Fixed8x13_Character_099,Fixed8x13_Character_100,Fixed8x13_Character_101,Fixed8x13_Character_102,Fixed8x13_Character_103,Fixed8x13_Character_104,Fixed8x13_Character_105,Fixed8x13_Character_106,Fixed8x13_Character_107,Fixed8x13_Character_108,Fixed8x13_Character_109,Fixed8x13_Character_110,Fixed8x13_Character_111, + Fixed8x13_Character_112,Fixed8x13_Character_113,Fixed8x13_Character_114,Fixed8x13_Character_115,Fixed8x13_Character_116,Fixed8x13_Character_117,Fixed8x13_Character_118,Fixed8x13_Character_119,Fixed8x13_Character_120,Fixed8x13_Character_121,Fixed8x13_Character_122,Fixed8x13_Character_123,Fixed8x13_Character_124,Fixed8x13_Character_125,Fixed8x13_Character_126,Fixed8x13_Character_032, + Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032, + Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032, + Fixed8x13_Character_160,Fixed8x13_Character_161,Fixed8x13_Character_162,Fixed8x13_Character_163,Fixed8x13_Character_164,Fixed8x13_Character_165,Fixed8x13_Character_166,Fixed8x13_Character_167,Fixed8x13_Character_168,Fixed8x13_Character_169,Fixed8x13_Character_170,Fixed8x13_Character_171,Fixed8x13_Character_172,Fixed8x13_Character_173,Fixed8x13_Character_174,Fixed8x13_Character_175, + Fixed8x13_Character_176,Fixed8x13_Character_177,Fixed8x13_Character_178,Fixed8x13_Character_179,Fixed8x13_Character_180,Fixed8x13_Character_181,Fixed8x13_Character_182,Fixed8x13_Character_183,Fixed8x13_Character_184,Fixed8x13_Character_185,Fixed8x13_Character_186,Fixed8x13_Character_187,Fixed8x13_Character_188,Fixed8x13_Character_189,Fixed8x13_Character_190,Fixed8x13_Character_191, + Fixed8x13_Character_192,Fixed8x13_Character_193,Fixed8x13_Character_194,Fixed8x13_Character_195,Fixed8x13_Character_196,Fixed8x13_Character_197,Fixed8x13_Character_198,Fixed8x13_Character_199,Fixed8x13_Character_200,Fixed8x13_Character_201,Fixed8x13_Character_202,Fixed8x13_Character_203,Fixed8x13_Character_204,Fixed8x13_Character_205,Fixed8x13_Character_206,Fixed8x13_Character_207, + Fixed8x13_Character_208,Fixed8x13_Character_209,Fixed8x13_Character_210,Fixed8x13_Character_211,Fixed8x13_Character_212,Fixed8x13_Character_213,Fixed8x13_Character_214,Fixed8x13_Character_215,Fixed8x13_Character_216,Fixed8x13_Character_217,Fixed8x13_Character_218,Fixed8x13_Character_219,Fixed8x13_Character_220,Fixed8x13_Character_221,Fixed8x13_Character_222,Fixed8x13_Character_223, + Fixed8x13_Character_224,Fixed8x13_Character_225,Fixed8x13_Character_226,Fixed8x13_Character_227,Fixed8x13_Character_228,Fixed8x13_Character_229,Fixed8x13_Character_230,Fixed8x13_Character_231,Fixed8x13_Character_232,Fixed8x13_Character_233,Fixed8x13_Character_234,Fixed8x13_Character_235,Fixed8x13_Character_236,Fixed8x13_Character_237,Fixed8x13_Character_238,Fixed8x13_Character_239, + Fixed8x13_Character_240,Fixed8x13_Character_241,Fixed8x13_Character_242,Fixed8x13_Character_243,Fixed8x13_Character_244,Fixed8x13_Character_245,Fixed8x13_Character_246,Fixed8x13_Character_247,Fixed8x13_Character_248,Fixed8x13_Character_249,Fixed8x13_Character_250,Fixed8x13_Character_251,Fixed8x13_Character_252,Fixed8x13_Character_253,Fixed8x13_Character_254,Fixed8x13_Character_255,NULL}; + +/* The font structure: */ +const SFG_Font fgFontFixed8x13 = { "-misc-fixed-medium-r-normal--13-120-75-75-C-80-iso8859-1", 256, 14, Fixed8x13_Character_Map, 0, 3 }; + +static const GLubyte Fixed9x15_Character_000[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_001[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 28, 0, 62, 0,127, 0,255,128,127, 0, 62, 0, 28, 0, 8, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_002[] = { 9, 0, 0,170,128, 85, 0,170,128, 85, 0,170,128, 85, 0,170,128, 85, 0,170,128, 85, 0,170,128, 85, 0,170,128, 85, 0,170,128}; +static const GLubyte Fixed9x15_Character_003[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 4, 0, 4, 0, 4, 0, 4, 0, 31, 0, 0, 0, 72, 0, 72, 0,120, 0, 72, 0, 72, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_004[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 14, 0, 72, 0, 79, 0, 64, 0,112, 0, 64, 0,120, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_005[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 9, 0, 14, 0, 9, 0, 14, 0, 0, 0, 56, 0, 64, 0, 64, 0, 64, 0, 56, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_006[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 14, 0, 8, 0, 15, 0, 0, 0,120, 0, 64, 0, 64, 0, 64, 0, 64, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_007[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 18, 0, 18, 0, 12, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_008[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 8, 0, 8, 0, 8, 0,127, 0, 8, 0, 8, 0, 8, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_009[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0, 68, 0, 76, 0, 84, 0,100, 0, 68, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_010[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 4, 0, 4, 0, 4, 0, 4, 0, 31, 0, 0, 0, 16, 0, 40, 0, 40, 0, 68, 0, 68, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_011[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,248, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_012[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_013[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_014[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_015[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0,255,128, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_016[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255,128}; +static const GLubyte Fixed9x15_Character_017[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255,128, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_018[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_019[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,255,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_020[] = { 9, 0, 0,255,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_021[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 15,128, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_022[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0,248, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_023[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255,128, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_024[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0,255,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_025[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_026[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 3, 0, 28, 0, 96, 0, 28, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_027[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 96, 0, 28, 0, 3, 0, 28, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_028[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 34, 0, 34, 0, 34, 0, 34, 0, 34, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_029[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16, 0, 16, 0,127, 0, 8, 0,127, 0, 4, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_030[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 46, 0, 81, 0, 48, 0, 16, 0, 16, 0,124, 0, 16, 0, 16, 0, 17, 0, 14, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_031[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_032[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_033[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_034[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 18, 0, 18, 0, 18, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_035[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 36, 0, 36, 0,126, 0, 36, 0, 36, 0,126, 0, 36, 0, 36, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_036[] = { 9, 0, 0, 0, 0, 0, 0, 8, 0, 62, 0, 73, 0, 9, 0, 9, 0, 10, 0, 28, 0, 40, 0, 72, 0, 73, 0, 62, 0, 8, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_037[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 66, 0, 37, 0, 37, 0, 18, 0, 8, 0, 8, 0, 36, 0, 82, 0, 82, 0, 33, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_038[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 49, 0, 74, 0, 68, 0, 74, 0, 49, 0, 48, 0, 72, 0, 72, 0, 72, 0, 48, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_039[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16, 0, 8, 0, 4, 0, 6, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_040[] = { 9, 0, 0, 0, 0, 0, 0, 4, 0, 8, 0, 8, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 8, 0, 8, 0, 4, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_041[] = { 9, 0, 0, 0, 0, 0, 0, 16, 0, 8, 0, 8, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 8, 0, 8, 0, 16, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_042[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 73, 0, 42, 0, 28, 0, 42, 0, 73, 0, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_043[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0,127, 0, 8, 0, 8, 0, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_044[] = { 9, 0, 0, 8, 0, 4, 0, 4, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_045[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_046[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_047[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 64, 0, 32, 0, 32, 0, 16, 0, 8, 0, 8, 0, 4, 0, 2, 0, 2, 0, 1, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_048[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_049[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 72, 0, 40, 0, 24, 0, 8, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_050[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 64, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_051[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 1, 0, 1, 0, 1, 0, 14, 0, 4, 0, 2, 0, 1, 0,127, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_052[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 2, 0, 2, 0,127, 0, 66, 0, 34, 0, 18, 0, 10, 0, 6, 0, 2, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_053[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 1, 0, 1, 0, 1, 0, 97, 0, 94, 0, 64, 0, 64, 0,127, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_054[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 64, 0, 64, 0, 32, 0, 30, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_055[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 32, 0, 16, 0, 16, 0, 8, 0, 4, 0, 2, 0, 1, 0, 1, 0,127, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_056[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 65, 0, 65, 0, 34, 0, 28, 0, 34, 0, 65, 0, 34, 0, 28, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_057[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0, 2, 0, 1, 0, 1, 0, 61, 0, 67, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_058[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_059[] = { 9, 0, 0, 8, 0, 4, 0, 4, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_060[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 4, 0, 8, 0, 16, 0, 32, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_061[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 0, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_062[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0, 2, 0, 4, 0, 8, 0, 16, 0, 32, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_063[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 0, 0, 8, 0, 8, 0, 4, 0, 2, 0, 1, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_064[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0, 77, 0, 83, 0, 81, 0, 79, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_065[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 65, 0, 34, 0, 20, 0, 8, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_066[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 33, 0, 33, 0, 33, 0, 33, 0,126, 0, 33, 0, 33, 0, 33, 0,126, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_067[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_068[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 33, 0, 33, 0, 33, 0, 33, 0, 33, 0, 33, 0, 33, 0, 33, 0,126, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_069[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0, 32, 0,127, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_070[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 32, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0, 32, 0,127, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_071[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 71, 0, 64, 0, 64, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_072[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_073[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_074[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0, 66, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 15,128, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_075[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 66, 0, 68, 0, 72, 0, 80, 0,112, 0, 72, 0, 68, 0, 66, 0, 65, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_076[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_077[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 73, 0, 73, 0, 85, 0, 85, 0, 99, 0, 65, 0, 65, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_078[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 67, 0, 69, 0, 73, 0, 81, 0, 97, 0, 65, 0, 65, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_079[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_080[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0,126, 0, 65, 0, 65, 0, 65, 0,126, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_081[] = { 9, 0, 0, 0, 0, 3, 0, 4, 0, 62, 0, 73, 0, 81, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_082[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 66, 0, 68, 0, 72, 0,126, 0, 65, 0, 65, 0, 65, 0,126, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_083[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 1, 0, 6, 0, 56, 0, 64, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_084[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0,127, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_085[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_086[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 20, 0, 20, 0, 20, 0, 34, 0, 34, 0, 34, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_087[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 85, 0, 73, 0, 73, 0, 73, 0, 73, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_088[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 34, 0, 20, 0, 8, 0, 8, 0, 20, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_089[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 20, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_090[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 64, 0, 64, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0, 1, 0,127, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_091[] = { 9, 0, 0, 0, 0, 0, 0, 30, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 30, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_092[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 2, 0, 2, 0, 4, 0, 8, 0, 8, 0, 16, 0, 32, 0, 32, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_093[] = { 9, 0, 0, 0, 0, 0, 0, 60, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 60, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_094[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 34, 0, 20, 0, 8, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_095[] = { 9, 0, 0, 0, 0, 0, 0,255, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_096[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4, 0, 8, 0, 16, 0, 48, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_097[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_098[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 94, 0, 97, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 64, 0, 64, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_099[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 64, 0, 64, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_100[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 65, 0, 65, 0, 67, 0, 61, 0, 1, 0, 1, 0, 1, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_101[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0,127, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_102[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 16, 0, 16, 0, 16, 0, 16, 0,124, 0, 16, 0, 16, 0, 17, 0, 17, 0, 14, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_103[] = { 9, 0, 0, 62, 0, 65, 0, 65, 0, 62, 0, 64, 0, 60, 0, 66, 0, 66, 0, 66, 0, 61, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_104[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 64, 0, 64, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_105[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 0, 0, 24, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_106[] = { 9, 0, 0, 60, 0, 66, 0, 66, 0, 66, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 14, 0, 0, 0, 0, 0, 6, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_107[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 70, 0, 88, 0, 96, 0, 88, 0, 70, 0, 65, 0, 64, 0, 64, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_108[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_109[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 73, 0, 73, 0, 73, 0, 73, 0, 73, 0,118, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_110[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_111[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_112[] = { 9, 0, 0, 64, 0, 64, 0, 64, 0, 94, 0, 97, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_113[] = { 9, 0, 0, 1, 0, 1, 0, 1, 0, 61, 0, 67, 0, 65, 0, 65, 0, 65, 0, 67, 0, 61, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_114[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 32, 0, 32, 0, 32, 0, 33, 0, 49, 0, 78, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_115[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 1, 0, 62, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_116[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 14, 0, 17, 0, 16, 0, 16, 0, 16, 0, 16, 0,126, 0, 16, 0, 16, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_117[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_118[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 20, 0, 20, 0, 34, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_119[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 85, 0, 73, 0, 73, 0, 73, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_120[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 34, 0, 20, 0, 8, 0, 20, 0, 34, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_121[] = { 9, 0, 0, 60, 0, 66, 0, 2, 0, 58, 0, 70, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_122[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_123[] = { 9, 0, 0, 0, 0, 0, 0, 7, 0, 8, 0, 8, 0, 8, 0, 4, 0, 24, 0, 24, 0, 4, 0, 8, 0, 8, 0, 8, 0, 7, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_124[] = { 9, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_125[] = { 9, 0, 0, 0, 0, 0, 0,112, 0, 8, 0, 8, 0, 8, 0, 16, 0, 12, 0, 12, 0, 16, 0, 8, 0, 8, 0, 8, 0,112, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_126[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 70, 0, 73, 0, 49, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_127[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_128[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_129[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_130[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_131[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_132[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_133[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_134[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_135[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_136[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_137[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_138[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_139[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_140[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_141[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_142[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_143[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_144[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_145[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_146[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_147[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_148[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_149[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_150[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_151[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_152[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_153[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_154[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_155[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_156[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_157[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_158[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_159[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_160[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_161[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0, 0, 0, 8, 0, 8, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_162[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 60, 0, 82, 0, 80, 0, 72, 0, 74, 0, 60, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_163[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 46, 0, 81, 0, 48, 0, 16, 0, 16, 0,124, 0, 16, 0, 16, 0, 17, 0, 14, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_164[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 62, 0, 34, 0, 34, 0, 62, 0, 65, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_165[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 62, 0, 8, 0, 62, 0, 20, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_166[] = { 9, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_167[] = { 9, 0, 0, 0, 0, 0, 0, 24, 0, 36, 0, 4, 0, 24, 0, 36, 0, 36, 0, 36, 0, 24, 0, 32, 0, 36, 0, 24, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_168[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 34, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_169[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0, 66, 0,153, 0,165, 0,161, 0,165, 0,153, 0, 66, 0, 60, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_170[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 0, 0, 60, 0, 72, 0, 56, 0, 72, 0, 48, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_171[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 18, 0, 36, 0, 72, 0, 72, 0, 36, 0, 18, 0, 9, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_172[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 2, 0, 2, 0,126, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_173[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_174[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0, 66, 0,165, 0,169, 0,185, 0,165, 0,185, 0, 66, 0, 60, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_175[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_176[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 18, 0, 18, 0, 12, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_177[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 8, 0, 8, 0, 8, 0,127, 0, 8, 0, 8, 0, 8, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_178[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120, 0, 64, 0, 48, 0, 8, 0, 72, 0, 48, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_179[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 0, 72, 0, 8, 0, 16, 0, 72, 0, 48, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_180[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16, 0, 8, 0, 4, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_181[] = { 9, 0, 0, 0, 0, 64, 0, 64, 0, 93, 0, 99, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_182[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 5, 0, 5, 0, 5, 0, 5, 0, 5, 0, 61, 0, 69, 0, 69, 0, 69, 0, 63, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_183[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_184[] = { 9, 0, 0, 24, 0, 36, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_185[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,112, 0, 32, 0, 32, 0, 32, 0, 96, 0, 32, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_186[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 0, 0, 56, 0, 68, 0, 68, 0, 56, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_187[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 72, 0, 36, 0, 18, 0, 9, 0, 9, 0, 18, 0, 36, 0, 72, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_188[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 13, 0, 9, 0, 5, 0,115, 0, 33, 0, 32, 0, 32, 0, 96, 0, 32, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_189[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 8, 0, 6, 0, 1, 0,121, 0, 38, 0, 32, 0, 32, 0, 96, 0, 32, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_190[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 13, 0, 9, 0, 5, 0, 51, 0, 73, 0, 8, 0, 16, 0, 72, 0, 48, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_191[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 64, 0, 32, 0, 16, 0, 8, 0, 8, 0, 0, 0, 8, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_192[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 8, 0, 16, 0, 32, 0}; +static const GLubyte Fixed9x15_Character_193[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 8, 0, 4, 0, 2, 0}; +static const GLubyte Fixed9x15_Character_194[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 34, 0, 20, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_195[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 78, 0, 49, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_196[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 34, 0, 34, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_197[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 20, 0, 28, 0, 34, 0, 28, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_198[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 79, 0, 72, 0, 72, 0, 72, 0,126, 0, 72, 0, 72, 0, 72, 0, 72, 0, 55, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_199[] = { 9, 0, 0, 24, 0, 36, 0, 12, 0, 62, 0, 65, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_200[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0,127, 0, 0, 0, 8, 0, 16, 0, 32, 0}; +static const GLubyte Fixed9x15_Character_201[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0,127, 0, 0, 0, 8, 0, 4, 0, 2, 0}; +static const GLubyte Fixed9x15_Character_202[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0,127, 0, 0, 0, 34, 0, 20, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_203[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0,127, 0, 0, 0, 34, 0, 34, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_204[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 62, 0, 0, 0, 8, 0, 16, 0, 32, 0}; +static const GLubyte Fixed9x15_Character_205[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 62, 0, 0, 0, 8, 0, 4, 0, 2, 0}; +static const GLubyte Fixed9x15_Character_206[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 62, 0, 0, 0, 34, 0, 20, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_207[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 62, 0, 0, 0, 34, 0, 34, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_208[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 33, 0, 33, 0, 33, 0, 33, 0,225, 0, 33, 0, 33, 0, 33, 0,124, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_209[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 67, 0, 69, 0, 73, 0, 73, 0, 81, 0, 97, 0, 65, 0, 0, 0, 78, 0, 49, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_210[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 16, 0, 32, 0}; +static const GLubyte Fixed9x15_Character_211[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 4, 0, 2, 0}; +static const GLubyte Fixed9x15_Character_212[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 20, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_213[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 78, 0, 49, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_214[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 34, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_215[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 34, 0, 20, 0, 8, 0, 20, 0, 34, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_216[] = { 9, 0, 0, 0, 0, 0, 0, 64, 0, 62, 0, 97, 0, 81, 0, 81, 0, 73, 0, 73, 0, 69, 0, 69, 0, 67, 0, 62, 0, 1, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_217[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 8, 0, 16, 0, 32, 0}; +static const GLubyte Fixed9x15_Character_218[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 8, 0, 4, 0, 2, 0}; +static const GLubyte Fixed9x15_Character_219[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 34, 0, 20, 0, 8, 0}; +static const GLubyte Fixed9x15_Character_220[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 34, 0, 34, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_221[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 20, 0, 34, 0, 65, 0, 65, 0, 0, 0, 8, 0, 4, 0, 2, 0}; +static const GLubyte Fixed9x15_Character_222[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 64, 0, 64, 0, 64, 0,126, 0, 65, 0, 65, 0, 65, 0,126, 0, 64, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_223[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 44, 0, 34, 0, 34, 0, 34, 0, 36, 0,104, 0, 36, 0, 34, 0, 34, 0, 28, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_224[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 4, 0, 8, 0, 16, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_225[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 8, 0, 4, 0, 2, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_226[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 34, 0, 20, 0, 8, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_227[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 38, 0, 25, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_228[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 34, 0, 34, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_229[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 12, 0, 18, 0, 12, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_230[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 55, 0, 73, 0, 72, 0, 62, 0, 9, 0, 73, 0, 54, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_231[] = { 9, 0, 0, 24, 0, 36, 0, 12, 0, 62, 0, 65, 0, 64, 0, 64, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_232[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0,127, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 16, 0, 32, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_233[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0,127, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 4, 0, 2, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_234[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0,127, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 20, 0, 8, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_235[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0,127, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 34, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_236[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 8, 0, 16, 0, 32, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_237[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 16, 0, 8, 0, 4, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_238[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 68, 0, 40, 0, 16, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_239[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 36, 0, 36, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_240[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 4, 0, 40, 0, 24, 0, 36, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_241[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 0, 0, 78, 0, 49, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_242[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 16, 0, 32, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_243[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 4, 0, 2, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_244[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 20, 0, 8, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_245[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 78, 0, 49, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_246[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 34, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_247[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 28, 0, 8, 0, 0, 0,127, 0, 0, 0, 8, 0, 28, 0, 8, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_248[] = { 9, 0, 0, 0, 0, 0, 0, 64, 0, 62, 0, 81, 0, 81, 0, 73, 0, 69, 0, 69, 0, 62, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_249[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 8, 0, 16, 0, 32, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_250[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 8, 0, 4, 0, 2, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_251[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 34, 0, 20, 0, 8, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_252[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 36, 0, 36, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_253[] = { 9, 0, 0, 60, 0, 66, 0, 2, 0, 58, 0, 70, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 16, 0, 8, 0, 4, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_254[] = { 9, 0, 0, 64, 0, 64, 0, 64, 0, 94, 0, 97, 0, 65, 0, 65, 0, 97, 0, 94, 0, 64, 0, 64, 0, 64, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Fixed9x15_Character_255[] = { 9, 0, 0, 60, 0, 66, 0, 2, 0, 58, 0, 70, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 36, 0, 36, 0, 0, 0, 0, 0}; + +/* The font characters mapping: */ +static const GLubyte* Fixed9x15_Character_Map[] = {Fixed9x15_Character_000,Fixed9x15_Character_001,Fixed9x15_Character_002,Fixed9x15_Character_003,Fixed9x15_Character_004,Fixed9x15_Character_005,Fixed9x15_Character_006,Fixed9x15_Character_007,Fixed9x15_Character_008,Fixed9x15_Character_009,Fixed9x15_Character_010,Fixed9x15_Character_011,Fixed9x15_Character_012,Fixed9x15_Character_013,Fixed9x15_Character_014,Fixed9x15_Character_015, + Fixed9x15_Character_016,Fixed9x15_Character_017,Fixed9x15_Character_018,Fixed9x15_Character_019,Fixed9x15_Character_020,Fixed9x15_Character_021,Fixed9x15_Character_022,Fixed9x15_Character_023,Fixed9x15_Character_024,Fixed9x15_Character_025,Fixed9x15_Character_026,Fixed9x15_Character_027,Fixed9x15_Character_028,Fixed9x15_Character_029,Fixed9x15_Character_030,Fixed9x15_Character_031, + Fixed9x15_Character_032,Fixed9x15_Character_033,Fixed9x15_Character_034,Fixed9x15_Character_035,Fixed9x15_Character_036,Fixed9x15_Character_037,Fixed9x15_Character_038,Fixed9x15_Character_039,Fixed9x15_Character_040,Fixed9x15_Character_041,Fixed9x15_Character_042,Fixed9x15_Character_043,Fixed9x15_Character_044,Fixed9x15_Character_045,Fixed9x15_Character_046,Fixed9x15_Character_047, + Fixed9x15_Character_048,Fixed9x15_Character_049,Fixed9x15_Character_050,Fixed9x15_Character_051,Fixed9x15_Character_052,Fixed9x15_Character_053,Fixed9x15_Character_054,Fixed9x15_Character_055,Fixed9x15_Character_056,Fixed9x15_Character_057,Fixed9x15_Character_058,Fixed9x15_Character_059,Fixed9x15_Character_060,Fixed9x15_Character_061,Fixed9x15_Character_062,Fixed9x15_Character_063, + Fixed9x15_Character_064,Fixed9x15_Character_065,Fixed9x15_Character_066,Fixed9x15_Character_067,Fixed9x15_Character_068,Fixed9x15_Character_069,Fixed9x15_Character_070,Fixed9x15_Character_071,Fixed9x15_Character_072,Fixed9x15_Character_073,Fixed9x15_Character_074,Fixed9x15_Character_075,Fixed9x15_Character_076,Fixed9x15_Character_077,Fixed9x15_Character_078,Fixed9x15_Character_079, + Fixed9x15_Character_080,Fixed9x15_Character_081,Fixed9x15_Character_082,Fixed9x15_Character_083,Fixed9x15_Character_084,Fixed9x15_Character_085,Fixed9x15_Character_086,Fixed9x15_Character_087,Fixed9x15_Character_088,Fixed9x15_Character_089,Fixed9x15_Character_090,Fixed9x15_Character_091,Fixed9x15_Character_092,Fixed9x15_Character_093,Fixed9x15_Character_094,Fixed9x15_Character_095, + Fixed9x15_Character_096,Fixed9x15_Character_097,Fixed9x15_Character_098,Fixed9x15_Character_099,Fixed9x15_Character_100,Fixed9x15_Character_101,Fixed9x15_Character_102,Fixed9x15_Character_103,Fixed9x15_Character_104,Fixed9x15_Character_105,Fixed9x15_Character_106,Fixed9x15_Character_107,Fixed9x15_Character_108,Fixed9x15_Character_109,Fixed9x15_Character_110,Fixed9x15_Character_111, + Fixed9x15_Character_112,Fixed9x15_Character_113,Fixed9x15_Character_114,Fixed9x15_Character_115,Fixed9x15_Character_116,Fixed9x15_Character_117,Fixed9x15_Character_118,Fixed9x15_Character_119,Fixed9x15_Character_120,Fixed9x15_Character_121,Fixed9x15_Character_122,Fixed9x15_Character_123,Fixed9x15_Character_124,Fixed9x15_Character_125,Fixed9x15_Character_126,Fixed9x15_Character_032, + Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032, + Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032, + Fixed9x15_Character_160,Fixed9x15_Character_161,Fixed9x15_Character_162,Fixed9x15_Character_163,Fixed9x15_Character_164,Fixed9x15_Character_165,Fixed9x15_Character_166,Fixed9x15_Character_167,Fixed9x15_Character_168,Fixed9x15_Character_169,Fixed9x15_Character_170,Fixed9x15_Character_171,Fixed9x15_Character_172,Fixed9x15_Character_173,Fixed9x15_Character_174,Fixed9x15_Character_175, + Fixed9x15_Character_176,Fixed9x15_Character_177,Fixed9x15_Character_178,Fixed9x15_Character_179,Fixed9x15_Character_180,Fixed9x15_Character_181,Fixed9x15_Character_182,Fixed9x15_Character_183,Fixed9x15_Character_184,Fixed9x15_Character_185,Fixed9x15_Character_186,Fixed9x15_Character_187,Fixed9x15_Character_188,Fixed9x15_Character_189,Fixed9x15_Character_190,Fixed9x15_Character_191, + Fixed9x15_Character_192,Fixed9x15_Character_193,Fixed9x15_Character_194,Fixed9x15_Character_195,Fixed9x15_Character_196,Fixed9x15_Character_197,Fixed9x15_Character_198,Fixed9x15_Character_199,Fixed9x15_Character_200,Fixed9x15_Character_201,Fixed9x15_Character_202,Fixed9x15_Character_203,Fixed9x15_Character_204,Fixed9x15_Character_205,Fixed9x15_Character_206,Fixed9x15_Character_207, + Fixed9x15_Character_208,Fixed9x15_Character_209,Fixed9x15_Character_210,Fixed9x15_Character_211,Fixed9x15_Character_212,Fixed9x15_Character_213,Fixed9x15_Character_214,Fixed9x15_Character_215,Fixed9x15_Character_216,Fixed9x15_Character_217,Fixed9x15_Character_218,Fixed9x15_Character_219,Fixed9x15_Character_220,Fixed9x15_Character_221,Fixed9x15_Character_222,Fixed9x15_Character_223, + Fixed9x15_Character_224,Fixed9x15_Character_225,Fixed9x15_Character_226,Fixed9x15_Character_227,Fixed9x15_Character_228,Fixed9x15_Character_229,Fixed9x15_Character_230,Fixed9x15_Character_231,Fixed9x15_Character_232,Fixed9x15_Character_233,Fixed9x15_Character_234,Fixed9x15_Character_235,Fixed9x15_Character_236,Fixed9x15_Character_237,Fixed9x15_Character_238,Fixed9x15_Character_239, + Fixed9x15_Character_240,Fixed9x15_Character_241,Fixed9x15_Character_242,Fixed9x15_Character_243,Fixed9x15_Character_244,Fixed9x15_Character_245,Fixed9x15_Character_246,Fixed9x15_Character_247,Fixed9x15_Character_248,Fixed9x15_Character_249,Fixed9x15_Character_250,Fixed9x15_Character_251,Fixed9x15_Character_252,Fixed9x15_Character_253,Fixed9x15_Character_254,Fixed9x15_Character_255,NULL}; + +/* The font structure: */ +const SFG_Font fgFontFixed9x15 = { "-misc-fixed-medium-r-normal--15-140-75-75-C-90-iso8859-1", 256, 16, Fixed9x15_Character_Map, 0, 4 }; + +static const GLubyte Helvetica10_Character_000[] = { 8, 0, 0, 0, 84, 0, 68, 0, 68, 0, 84, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_001[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_002[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_003[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_004[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_005[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_006[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_007[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_008[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_009[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_010[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_011[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_012[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_013[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_014[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_015[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_016[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_017[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_018[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_019[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_020[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_021[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_022[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_023[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_024[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_025[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_026[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_027[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_028[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_029[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_030[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_031[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_032[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_033[] = { 3, 0, 0, 0, 64, 0, 64, 64, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica10_Character_034[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 80, 80, 0, 0, 0}; +static const GLubyte Helvetica10_Character_035[] = { 6, 0, 0, 0, 80, 80,248, 40,124, 40, 40, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_036[] = { 6, 0, 0, 32,112,168, 40,112,160,168,112, 32, 0, 0, 0}; +static const GLubyte Helvetica10_Character_037[] = { 9, 0, 0, 0, 0, 0, 0, 38, 0, 41, 0, 22, 0, 16, 0, 8, 0,104, 0,148, 0,100, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_038[] = { 8, 0, 0, 0, 50, 76, 76, 82, 48, 40, 40, 16, 0, 0, 0}; +static const GLubyte Helvetica10_Character_039[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 64, 32, 32, 0, 0, 0}; +static const GLubyte Helvetica10_Character_040[] = { 4, 0, 32, 64, 64,128,128,128,128, 64, 64, 32, 0, 0, 0}; +static const GLubyte Helvetica10_Character_041[] = { 4, 0, 64, 32, 32, 16, 16, 16, 16, 32, 32, 64, 0, 0, 0}; +static const GLubyte Helvetica10_Character_042[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0,160, 64,160, 0, 0, 0}; +static const GLubyte Helvetica10_Character_043[] = { 6, 0, 0, 0, 0, 32, 32,248, 32, 32, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_044[] = { 3, 0,128, 64, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_045[] = { 7, 0, 0, 0, 0, 0, 0,124, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_046[] = { 3, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_047[] = { 3, 0, 0, 0,128,128, 64, 64, 64, 64, 32, 32, 0, 0, 0}; +static const GLubyte Helvetica10_Character_048[] = { 6, 0, 0, 0,112,136,136,136,136,136,136,112, 0, 0, 0}; +static const GLubyte Helvetica10_Character_049[] = { 6, 0, 0, 0, 32, 32, 32, 32, 32, 32, 96, 32, 0, 0, 0}; +static const GLubyte Helvetica10_Character_050[] = { 6, 0, 0, 0,248,128, 64, 48, 8, 8,136,112, 0, 0, 0}; +static const GLubyte Helvetica10_Character_051[] = { 6, 0, 0, 0,112,136, 8, 8, 48, 8,136,112, 0, 0, 0}; +static const GLubyte Helvetica10_Character_052[] = { 6, 0, 0, 0, 16, 16,248,144, 80, 80, 48, 16, 0, 0, 0}; +static const GLubyte Helvetica10_Character_053[] = { 6, 0, 0, 0,112,136, 8, 8,240,128,128,248, 0, 0, 0}; +static const GLubyte Helvetica10_Character_054[] = { 6, 0, 0, 0,112,136,136,200,176,128,136,112, 0, 0, 0}; +static const GLubyte Helvetica10_Character_055[] = { 6, 0, 0, 0, 64, 64, 32, 32, 16, 16, 8,248, 0, 0, 0}; +static const GLubyte Helvetica10_Character_056[] = { 6, 0, 0, 0,112,136,136,136,112,136,136,112, 0, 0, 0}; +static const GLubyte Helvetica10_Character_057[] = { 6, 0, 0, 0,112,136, 8,104,152,136,136,112, 0, 0, 0}; +static const GLubyte Helvetica10_Character_058[] = { 3, 0, 0, 0, 64, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_059[] = { 3, 0,128, 64, 64, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_060[] = { 6, 0, 0, 0, 0, 16, 32, 64, 32, 16, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_061[] = { 5, 0, 0, 0, 0, 0,240, 0,240, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_062[] = { 6, 0, 0, 0, 0, 64, 32, 16, 32, 64, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_063[] = { 6, 0, 0, 0, 32, 0, 32, 32, 16, 8, 72, 48, 0, 0, 0}; +static const GLubyte Helvetica10_Character_064[] = { 11, 0, 0, 62, 0, 64, 0,155, 0,164,128,164,128,162, 64,146, 64, 77, 64, 32,128, 31, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_065[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 0, 0}; +static const GLubyte Helvetica10_Character_066[] = { 7, 0, 0, 0,120, 68, 68, 68,120, 68, 68,120, 0, 0, 0}; +static const GLubyte Helvetica10_Character_067[] = { 8, 0, 0, 0, 60, 66, 64, 64, 64, 64, 66, 60, 0, 0, 0}; +static const GLubyte Helvetica10_Character_068[] = { 8, 0, 0, 0,120, 68, 66, 66, 66, 66, 68,120, 0, 0, 0}; +static const GLubyte Helvetica10_Character_069[] = { 7, 0, 0, 0,124, 64, 64, 64,124, 64, 64,124, 0, 0, 0}; +static const GLubyte Helvetica10_Character_070[] = { 6, 0, 0, 0, 64, 64, 64, 64,120, 64, 64,124, 0, 0, 0}; +static const GLubyte Helvetica10_Character_071[] = { 8, 0, 0, 0, 58, 70, 66, 70, 64, 64, 66, 60, 0, 0, 0}; +static const GLubyte Helvetica10_Character_072[] = { 8, 0, 0, 0, 66, 66, 66, 66,126, 66, 66, 66, 0, 0, 0}; +static const GLubyte Helvetica10_Character_073[] = { 3, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica10_Character_074[] = { 5, 0, 0, 0, 96,144, 16, 16, 16, 16, 16, 16, 0, 0, 0}; +static const GLubyte Helvetica10_Character_075[] = { 7, 0, 0, 0, 68, 68, 72, 72,112, 80, 72, 68, 0, 0, 0}; +static const GLubyte Helvetica10_Character_076[] = { 6, 0, 0, 0,120, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica10_Character_077[] = { 9, 0, 0, 0, 0, 0, 0, 73, 0, 73, 0, 73, 0, 85, 0, 85, 0, 99, 0, 99, 0, 65, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_078[] = { 8, 0, 0, 0, 70, 70, 74, 74, 82, 82, 98, 98, 0, 0, 0}; +static const GLubyte Helvetica10_Character_079[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 0, 0}; +static const GLubyte Helvetica10_Character_080[] = { 7, 0, 0, 0, 64, 64, 64, 64,120, 68, 68,120, 0, 0, 0}; +static const GLubyte Helvetica10_Character_081[] = { 8, 0, 0, 1, 62, 70, 74, 66, 66, 66, 66, 60, 0, 0, 0}; +static const GLubyte Helvetica10_Character_082[] = { 7, 0, 0, 0, 68, 68, 68, 68,120, 68, 68,120, 0, 0, 0}; +static const GLubyte Helvetica10_Character_083[] = { 7, 0, 0, 0, 56, 68, 68, 4, 56, 64, 68, 56, 0, 0, 0}; +static const GLubyte Helvetica10_Character_084[] = { 5, 0, 0, 0, 32, 32, 32, 32, 32, 32, 32,248, 0, 0, 0}; +static const GLubyte Helvetica10_Character_085[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 0, 0, 0}; +static const GLubyte Helvetica10_Character_086[] = { 7, 0, 0, 0, 16, 40, 40, 68, 68, 68,130,130, 0, 0, 0}; +static const GLubyte Helvetica10_Character_087[] = { 9, 0, 0, 0, 0, 0, 0, 34, 0, 34, 0, 34, 0, 85, 0, 73, 0, 73, 0,136,128,136,128, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_088[] = { 7, 0, 0, 0, 68, 68, 40, 40, 16, 40, 68, 68, 0, 0, 0}; +static const GLubyte Helvetica10_Character_089[] = { 7, 0, 0, 0, 16, 16, 16, 40, 40, 68, 68,130, 0, 0, 0}; +static const GLubyte Helvetica10_Character_090[] = { 7, 0, 0, 0,124, 64, 32, 16, 16, 8, 4,124, 0, 0, 0}; +static const GLubyte Helvetica10_Character_091[] = { 3, 0, 96, 64, 64, 64, 64, 64, 64, 64, 64, 96, 0, 0, 0}; +static const GLubyte Helvetica10_Character_092[] = { 3, 0, 0, 0, 32, 32, 64, 64, 64, 64,128,128, 0, 0, 0}; +static const GLubyte Helvetica10_Character_093[] = { 3, 0,192, 64, 64, 64, 64, 64, 64, 64, 64,192, 0, 0, 0}; +static const GLubyte Helvetica10_Character_094[] = { 6, 0, 0, 0, 0, 0, 0,136, 80, 80, 32, 32, 0, 0, 0}; +static const GLubyte Helvetica10_Character_095[] = { 6, 0,252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_096[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 63, 64, 32, 0, 0, 0}; +static const GLubyte Helvetica10_Character_097[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_098[] = { 6, 0, 0, 0,176,200,136,136,200,176,128,128, 0, 0, 0}; +static const GLubyte Helvetica10_Character_099[] = { 5, 0, 0, 0, 96,144,128,128,144, 96, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_100[] = { 6, 0, 0, 0,104,152,136,136,152,104, 8, 8, 0, 0, 0}; +static const GLubyte Helvetica10_Character_101[] = { 5, 0, 0, 0, 96,144,128,240,144, 96, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_102[] = { 4, 0, 0, 0, 64, 64, 64, 64, 64,224, 64, 48, 0, 0, 0}; +static const GLubyte Helvetica10_Character_103[] = { 6, 0,112, 8,104,152,136,136,152,104, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_104[] = { 6, 0, 0, 0,136,136,136,136,200,176,128,128, 0, 0, 0}; +static const GLubyte Helvetica10_Character_105[] = { 2, 0, 0, 0,128,128,128,128,128,128, 0,128, 0, 0, 0}; +static const GLubyte Helvetica10_Character_106[] = { 2, 0, 0,128,128,128,128,128,128,128, 0,128, 0, 0, 0}; +static const GLubyte Helvetica10_Character_107[] = { 5, 0, 0, 0,144,144,160,192,160,144,128,128, 0, 0, 0}; +static const GLubyte Helvetica10_Character_108[] = { 2, 0, 0, 0,128,128,128,128,128,128,128,128, 0, 0, 0}; +static const GLubyte Helvetica10_Character_109[] = { 8, 0, 0, 0,146,146,146,146,146,236, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_110[] = { 6, 0, 0, 0,136,136,136,136,200,176, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_111[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_112[] = { 6, 0,128,128,176,200,136,136,200,176, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_113[] = { 6, 0, 8, 8,104,152,136,136,152,104, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_114[] = { 4, 0, 0, 0,128,128,128,128,192,160, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_115[] = { 5, 0, 0, 0, 96,144, 16, 96,144, 96, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_116[] = { 4, 0, 0, 0, 96, 64, 64, 64, 64,224, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica10_Character_117[] = { 5, 0, 0, 0,112,144,144,144,144,144, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_118[] = { 6, 0, 0, 0, 32, 32, 80, 80,136,136, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_119[] = { 8, 0, 0, 0, 40, 40, 84, 84,146,146, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_120[] = { 6, 0, 0, 0,136,136, 80, 32, 80,136, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_121[] = { 5, 0,128, 64, 64, 96,160,160,144,144, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_122[] = { 5, 0, 0, 0,240,128, 64, 32, 16,240, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_123[] = { 3, 0, 32, 64, 64, 64, 64,128, 64, 64, 64, 32, 0, 0, 0}; +static const GLubyte Helvetica10_Character_124[] = { 3, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica10_Character_125[] = { 3, 0,128, 64, 64, 64, 64, 32, 64, 64, 64,128, 0, 0, 0}; +static const GLubyte Helvetica10_Character_126[] = { 7, 0, 0, 0, 0, 0, 0,152,100, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_127[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_128[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_129[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_130[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_131[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_132[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_133[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_134[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_135[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_136[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_137[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_138[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_139[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_140[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_141[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_142[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_143[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_144[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_145[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_146[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_147[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_148[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_149[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_150[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_151[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_152[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_153[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_154[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_155[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_156[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_157[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_158[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_159[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_160[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_161[] = { 3, 0, 64, 64, 64, 64, 64, 64, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_162[] = { 6, 0, 0, 64,112,168,160,160,168,112, 16, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_163[] = { 6, 0, 0, 0,176, 72, 64, 64,224, 64, 72, 48, 0, 0, 0}; +static const GLubyte Helvetica10_Character_164[] = { 5, 0, 0, 0, 0,144, 96,144,144, 96,144, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_165[] = { 6, 0, 0, 0, 32,248, 32,248, 80, 80,136,136, 0, 0, 0}; +static const GLubyte Helvetica10_Character_166[] = { 3, 0, 64, 64, 64, 64, 0, 0, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica10_Character_167[] = { 6, 0,112,136, 24,112,200,152,112,192,136,112, 0, 0, 0}; +static const GLubyte Helvetica10_Character_168[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,160, 0, 0, 0}; +static const GLubyte Helvetica10_Character_169[] = { 9, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 77, 0, 81, 0, 77, 0, 34, 0, 28, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_170[] = { 4, 0, 0, 0, 0, 0, 0,224, 0,160, 32,224, 0, 0, 0}; +static const GLubyte Helvetica10_Character_171[] = { 6, 0, 0, 0, 40, 80,160, 80, 40, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_172[] = { 7, 0, 0, 0, 0, 0, 4, 4,124, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_173[] = { 4, 0, 0, 0, 0, 0, 0,224, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_174[] = { 9, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 85, 0, 89, 0, 93, 0, 34, 0, 28, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_175[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,224, 0, 0, 0}; +static const GLubyte Helvetica10_Character_176[] = { 4, 0, 0, 0, 0, 0, 0, 96,144,144, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_177[] = { 6, 0, 0, 0,248, 0, 32, 32,248, 32, 32, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_178[] = { 3, 0, 0, 0, 0, 0, 0,224, 64,160, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_179[] = { 3, 0, 0, 0, 0, 0, 0,192, 32, 64,224, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_180[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0,128, 64, 0, 0, 0}; +static const GLubyte Helvetica10_Character_181[] = { 5, 0,128,128,240,144,144,144,144,144, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_182[] = { 6, 0, 40, 40, 40, 40, 40,104,232,232,232,124, 0, 0, 0}; +static const GLubyte Helvetica10_Character_183[] = { 3, 0, 0, 0, 0, 0, 0,192, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_184[] = { 3, 0,192, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_185[] = { 3, 0, 0, 0, 0, 0, 0, 64, 64,192, 64, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_186[] = { 4, 0, 0, 0, 0, 0, 0,224, 0,224,160,224, 0, 0, 0}; +static const GLubyte Helvetica10_Character_187[] = { 6, 0, 0, 0,160, 80, 40, 80,160, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_188[] = { 9, 0, 0, 0, 0, 0, 0, 33, 0, 23,128, 19, 0, 9, 0, 72, 0, 68, 0,196, 0, 66, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_189[] = { 9, 0, 0, 0, 0, 0, 0, 39, 0, 18, 0, 21, 0, 11, 0, 72, 0, 68, 0,196, 0, 66, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_190[] = { 9, 0, 0, 0, 0, 0, 0, 33, 0, 23,128, 19, 0, 9, 0,200, 0, 36, 0, 68, 0,226, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_191[] = { 6, 0, 48, 72, 64, 32, 16, 16, 0, 16, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_192[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 16, 32}; +static const GLubyte Helvetica10_Character_193[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 16, 8}; +static const GLubyte Helvetica10_Character_194[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 40, 16}; +static const GLubyte Helvetica10_Character_195[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 40, 20}; +static const GLubyte Helvetica10_Character_196[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 40, 0}; +static const GLubyte Helvetica10_Character_197[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 16, 40, 16}; +static const GLubyte Helvetica10_Character_198[] = { 10, 0, 0, 0, 0, 0, 0,143,128,136, 0,120, 0, 72, 0, 47,128, 40, 0, 24, 0, 31,128, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_199[] = { 8, 0, 24, 8, 60, 66, 64, 64, 64, 64, 66, 60, 0, 0, 0}; +static const GLubyte Helvetica10_Character_200[] = { 7, 0, 0, 0,124, 64, 64, 64,124, 64, 64,124, 0, 16, 32}; +static const GLubyte Helvetica10_Character_201[] = { 7, 0, 0, 0,124, 64, 64, 64,124, 64, 64,124, 0, 16, 8}; +static const GLubyte Helvetica10_Character_202[] = { 7, 0, 0, 0,124, 64, 64,124, 64, 64, 64,124, 0, 40, 16}; +static const GLubyte Helvetica10_Character_203[] = { 7, 0, 0, 0,124, 64, 64, 64,124, 64, 64,124, 0, 40, 0}; +static const GLubyte Helvetica10_Character_204[] = { 3, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64,128}; +static const GLubyte Helvetica10_Character_205[] = { 3, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64, 32}; +static const GLubyte Helvetica10_Character_206[] = { 3, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0,160, 64}; +static const GLubyte Helvetica10_Character_207[] = { 3, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0,160, 0}; +static const GLubyte Helvetica10_Character_208[] = { 8, 0, 0, 0,120, 68, 66, 66,242, 66, 68,120, 0, 0, 0}; +static const GLubyte Helvetica10_Character_209[] = { 8, 0, 0, 0, 70, 70, 74, 74, 82, 82, 98, 98, 0, 40, 20}; +static const GLubyte Helvetica10_Character_210[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 8, 16}; +static const GLubyte Helvetica10_Character_211[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 8, 4}; +static const GLubyte Helvetica10_Character_212[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 20, 8}; +static const GLubyte Helvetica10_Character_213[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 40, 20}; +static const GLubyte Helvetica10_Character_214[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 36, 0}; +static const GLubyte Helvetica10_Character_215[] = { 6, 0, 0, 0, 0,136, 80, 32, 80,136, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_216[] = { 8, 0, 0, 64, 60, 98, 82, 82, 74, 74, 70, 60, 2, 0, 0}; +static const GLubyte Helvetica10_Character_217[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 0, 8, 16}; +static const GLubyte Helvetica10_Character_218[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 0, 16, 8}; +static const GLubyte Helvetica10_Character_219[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 0, 20, 8}; +static const GLubyte Helvetica10_Character_220[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 0, 36, 0}; +static const GLubyte Helvetica10_Character_221[] = { 7, 0, 0, 0, 16, 16, 16, 40, 40, 68, 68,130, 0, 16, 8}; +static const GLubyte Helvetica10_Character_222[] = { 7, 0, 0, 0, 64, 64,120, 68, 68,120, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica10_Character_223[] = { 5, 0, 0, 0,160,144,144,144,160,144,144, 96, 0, 0, 0}; +static const GLubyte Helvetica10_Character_224[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0, 32, 64, 0, 0}; +static const GLubyte Helvetica10_Character_225[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0, 32, 16, 0, 0}; +static const GLubyte Helvetica10_Character_226[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0, 80, 32, 0, 0}; +static const GLubyte Helvetica10_Character_227[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0,160, 80, 0, 0}; +static const GLubyte Helvetica10_Character_228[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0, 80, 0, 0, 0}; +static const GLubyte Helvetica10_Character_229[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 32, 80, 32, 0, 0}; +static const GLubyte Helvetica10_Character_230[] = { 8, 0, 0, 0,108,146,144,126, 18,236, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_231[] = { 5, 0, 96, 32, 96,144,128,128,144, 96, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_232[] = { 5, 0, 0, 0, 96,144,128,240,144, 96, 0, 32, 64, 0, 0}; +static const GLubyte Helvetica10_Character_233[] = { 5, 0, 0, 0, 96,144,128,240,144, 96, 0, 64, 32, 0, 0}; +static const GLubyte Helvetica10_Character_234[] = { 5, 0, 0, 0, 96,144,128,240,144, 96, 0, 80, 32, 0, 0}; +static const GLubyte Helvetica10_Character_235[] = { 5, 0, 0, 0, 96,144,128,240,144, 96, 0, 80, 0, 0, 0}; +static const GLubyte Helvetica10_Character_236[] = { 2, 0, 0, 0,128,128,128,128,128,128, 0,128, 0, 0, 0}; +static const GLubyte Helvetica10_Character_237[] = { 2, 0, 0, 0,128,128,128,128,128,128, 0,128, 64, 0, 0}; +static const GLubyte Helvetica10_Character_238[] = { 2, 0, 0, 0,128,128,128,128,128,128, 0, 64,128, 0, 0}; +static const GLubyte Helvetica10_Character_239[] = { 2, 0, 0, 0, 64, 64, 64, 64, 64, 64, 0,160, 0, 0, 0}; +static const GLubyte Helvetica10_Character_240[] = { 6, 0, 0, 0,112,136,136,136,136,120,144, 96, 80, 0, 0}; +static const GLubyte Helvetica10_Character_241[] = { 5, 0, 0, 0,144,144,144,144,144,224, 0,160, 80, 0, 0}; +static const GLubyte Helvetica10_Character_242[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 32, 64, 0, 0}; +static const GLubyte Helvetica10_Character_243[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 32, 16, 0, 0}; +static const GLubyte Helvetica10_Character_244[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 80, 32, 0, 0}; +static const GLubyte Helvetica10_Character_245[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 80, 40, 0, 0}; +static const GLubyte Helvetica10_Character_246[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 80, 0, 0, 0}; +static const GLubyte Helvetica10_Character_247[] = { 6, 0, 0, 0, 0, 32, 0,248, 0, 32, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_248[] = { 6, 0, 0, 0,112,136,200,168,152,116, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica10_Character_249[] = { 5, 0, 0, 0,112,144,144,144,144,144, 0, 32, 64, 0, 0}; +static const GLubyte Helvetica10_Character_250[] = { 5, 0, 0, 0,112,144,144,144,144,144, 0, 64, 32, 0, 0}; +static const GLubyte Helvetica10_Character_251[] = { 5, 0, 0, 0,112,144,144,144,144,144, 0, 80, 32, 0, 0}; +static const GLubyte Helvetica10_Character_252[] = { 5, 0, 0, 0,112,144,144,144,144,144, 0, 80, 0, 0, 0}; +static const GLubyte Helvetica10_Character_253[] = { 5, 0,128, 64, 64, 96,160,160,144,144, 0, 32, 16, 0, 0}; +static const GLubyte Helvetica10_Character_254[] = { 6, 0,128,128,176,200,136,136,200,176,128,128, 0, 0, 0}; +static const GLubyte Helvetica10_Character_255[] = { 5, 0,128, 64, 64, 96,160,160,144,144, 0, 80, 0, 0, 0}; + +/* The font characters mapping: */ +static const GLubyte* Helvetica10_Character_Map[] = {Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032, + Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032, + Helvetica10_Character_032,Helvetica10_Character_033,Helvetica10_Character_034,Helvetica10_Character_035,Helvetica10_Character_036,Helvetica10_Character_037,Helvetica10_Character_038,Helvetica10_Character_039,Helvetica10_Character_040,Helvetica10_Character_041,Helvetica10_Character_042,Helvetica10_Character_043,Helvetica10_Character_044,Helvetica10_Character_045,Helvetica10_Character_046,Helvetica10_Character_047, + Helvetica10_Character_048,Helvetica10_Character_049,Helvetica10_Character_050,Helvetica10_Character_051,Helvetica10_Character_052,Helvetica10_Character_053,Helvetica10_Character_054,Helvetica10_Character_055,Helvetica10_Character_056,Helvetica10_Character_057,Helvetica10_Character_058,Helvetica10_Character_059,Helvetica10_Character_060,Helvetica10_Character_061,Helvetica10_Character_062,Helvetica10_Character_063, + Helvetica10_Character_064,Helvetica10_Character_065,Helvetica10_Character_066,Helvetica10_Character_067,Helvetica10_Character_068,Helvetica10_Character_069,Helvetica10_Character_070,Helvetica10_Character_071,Helvetica10_Character_072,Helvetica10_Character_073,Helvetica10_Character_074,Helvetica10_Character_075,Helvetica10_Character_076,Helvetica10_Character_077,Helvetica10_Character_078,Helvetica10_Character_079, + Helvetica10_Character_080,Helvetica10_Character_081,Helvetica10_Character_082,Helvetica10_Character_083,Helvetica10_Character_084,Helvetica10_Character_085,Helvetica10_Character_086,Helvetica10_Character_087,Helvetica10_Character_088,Helvetica10_Character_089,Helvetica10_Character_090,Helvetica10_Character_091,Helvetica10_Character_092,Helvetica10_Character_093,Helvetica10_Character_094,Helvetica10_Character_095, + Helvetica10_Character_096,Helvetica10_Character_097,Helvetica10_Character_098,Helvetica10_Character_099,Helvetica10_Character_100,Helvetica10_Character_101,Helvetica10_Character_102,Helvetica10_Character_103,Helvetica10_Character_104,Helvetica10_Character_105,Helvetica10_Character_106,Helvetica10_Character_107,Helvetica10_Character_108,Helvetica10_Character_109,Helvetica10_Character_110,Helvetica10_Character_111, + Helvetica10_Character_112,Helvetica10_Character_113,Helvetica10_Character_114,Helvetica10_Character_115,Helvetica10_Character_116,Helvetica10_Character_117,Helvetica10_Character_118,Helvetica10_Character_119,Helvetica10_Character_120,Helvetica10_Character_121,Helvetica10_Character_122,Helvetica10_Character_123,Helvetica10_Character_124,Helvetica10_Character_125,Helvetica10_Character_126,Helvetica10_Character_032, + Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032, + Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032, + Helvetica10_Character_160,Helvetica10_Character_161,Helvetica10_Character_162,Helvetica10_Character_163,Helvetica10_Character_164,Helvetica10_Character_165,Helvetica10_Character_166,Helvetica10_Character_167,Helvetica10_Character_168,Helvetica10_Character_169,Helvetica10_Character_170,Helvetica10_Character_171,Helvetica10_Character_172,Helvetica10_Character_173,Helvetica10_Character_174,Helvetica10_Character_175, + Helvetica10_Character_176,Helvetica10_Character_177,Helvetica10_Character_178,Helvetica10_Character_179,Helvetica10_Character_180,Helvetica10_Character_181,Helvetica10_Character_182,Helvetica10_Character_183,Helvetica10_Character_184,Helvetica10_Character_185,Helvetica10_Character_186,Helvetica10_Character_187,Helvetica10_Character_188,Helvetica10_Character_189,Helvetica10_Character_190,Helvetica10_Character_191, + Helvetica10_Character_192,Helvetica10_Character_193,Helvetica10_Character_194,Helvetica10_Character_195,Helvetica10_Character_196,Helvetica10_Character_197,Helvetica10_Character_198,Helvetica10_Character_199,Helvetica10_Character_200,Helvetica10_Character_201,Helvetica10_Character_202,Helvetica10_Character_203,Helvetica10_Character_204,Helvetica10_Character_205,Helvetica10_Character_206,Helvetica10_Character_207, + Helvetica10_Character_208,Helvetica10_Character_209,Helvetica10_Character_210,Helvetica10_Character_211,Helvetica10_Character_212,Helvetica10_Character_213,Helvetica10_Character_214,Helvetica10_Character_215,Helvetica10_Character_216,Helvetica10_Character_217,Helvetica10_Character_218,Helvetica10_Character_219,Helvetica10_Character_220,Helvetica10_Character_221,Helvetica10_Character_222,Helvetica10_Character_223, + Helvetica10_Character_224,Helvetica10_Character_225,Helvetica10_Character_226,Helvetica10_Character_227,Helvetica10_Character_228,Helvetica10_Character_229,Helvetica10_Character_230,Helvetica10_Character_231,Helvetica10_Character_232,Helvetica10_Character_233,Helvetica10_Character_234,Helvetica10_Character_235,Helvetica10_Character_236,Helvetica10_Character_237,Helvetica10_Character_238,Helvetica10_Character_239, + Helvetica10_Character_240,Helvetica10_Character_241,Helvetica10_Character_242,Helvetica10_Character_243,Helvetica10_Character_244,Helvetica10_Character_245,Helvetica10_Character_246,Helvetica10_Character_247,Helvetica10_Character_248,Helvetica10_Character_249,Helvetica10_Character_250,Helvetica10_Character_251,Helvetica10_Character_252,Helvetica10_Character_253,Helvetica10_Character_254,Helvetica10_Character_255,NULL}; + +/* The font structure: */ +const SFG_Font fgFontHelvetica10 = { "-adobe-helvetica-medium-r-normal--10-100-75-75-p-56-iso8859-1", 256, 14, Helvetica10_Character_Map, 0, 3 }; + +static const GLubyte Helvetica12_Character_000[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_001[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_002[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_003[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_004[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_005[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_006[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_007[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_008[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_009[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_010[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_011[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_012[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_013[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_014[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_015[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_016[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_017[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_018[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_019[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_020[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_021[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_022[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_023[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_024[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_025[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_026[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_027[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_028[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_029[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_030[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_031[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_032[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_033[] = { 3, 0, 0, 0, 0, 64, 0, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_034[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 80, 80, 80, 0, 0, 0}; +static const GLubyte Helvetica12_Character_035[] = { 7, 0, 0, 0, 0, 80, 80, 80,252, 40,252, 40, 40, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_036[] = { 7, 0, 0, 0, 16, 56, 84, 84, 20, 56, 80, 84, 56, 16, 0, 0, 0}; +static const GLubyte Helvetica12_Character_037[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 17,128, 10, 64, 10, 64, 9,128, 4, 0, 52, 0, 74, 0, 74, 0, 49, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_038[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 57, 0, 70, 0, 66, 0, 69, 0, 40, 0, 24, 0, 36, 0, 36, 0, 24, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_039[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 64, 32, 96, 0, 0, 0}; +static const GLubyte Helvetica12_Character_040[] = { 4, 0, 16, 32, 32, 64, 64, 64, 64, 64, 64, 32, 32, 16, 0, 0, 0}; +static const GLubyte Helvetica12_Character_041[] = { 4, 0,128, 64, 64, 32, 32, 32, 32, 32, 32, 64, 64,128, 0, 0, 0}; +static const GLubyte Helvetica12_Character_042[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 80, 32, 80, 0, 0, 0}; +static const GLubyte Helvetica12_Character_043[] = { 7, 0, 0, 0, 0, 0, 16, 16,124, 16, 16, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_044[] = { 4, 0, 0, 64, 32, 32, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_045[] = { 8, 0, 0, 0, 0, 0, 0, 0,124, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_046[] = { 3, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_047[] = { 4, 0, 0, 0, 0,128,128, 64, 64, 64, 32, 32, 16, 16, 0, 0, 0}; +static const GLubyte Helvetica12_Character_048[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 68, 68, 56, 0, 0, 0}; +static const GLubyte Helvetica12_Character_049[] = { 7, 0, 0, 0, 0, 16, 16, 16, 16, 16, 16, 16,112, 16, 0, 0, 0}; +static const GLubyte Helvetica12_Character_050[] = { 7, 0, 0, 0, 0,124, 64, 64, 32, 16, 8, 4, 68, 56, 0, 0, 0}; +static const GLubyte Helvetica12_Character_051[] = { 7, 0, 0, 0, 0, 56, 68, 68, 4, 4, 24, 4, 68, 56, 0, 0, 0}; +static const GLubyte Helvetica12_Character_052[] = { 7, 0, 0, 0, 0, 8, 8,252,136, 72, 40, 40, 24, 8, 0, 0, 0}; +static const GLubyte Helvetica12_Character_053[] = { 7, 0, 0, 0, 0, 56, 68, 68, 4, 4,120, 64, 64,124, 0, 0, 0}; +static const GLubyte Helvetica12_Character_054[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68,100, 88, 64, 68, 56, 0, 0, 0}; +static const GLubyte Helvetica12_Character_055[] = { 7, 0, 0, 0, 0, 32, 32, 16, 16, 16, 8, 8, 4,124, 0, 0, 0}; +static const GLubyte Helvetica12_Character_056[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 56, 68, 68, 56, 0, 0, 0}; +static const GLubyte Helvetica12_Character_057[] = { 7, 0, 0, 0, 0, 56, 68, 4, 4, 60, 68, 68, 68, 56, 0, 0, 0}; +static const GLubyte Helvetica12_Character_058[] = { 3, 0, 0, 0, 0, 64, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_059[] = { 3, 0, 0,128, 64, 64, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_060[] = { 7, 0, 0, 0, 0, 0, 12, 48,192, 48, 12, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_061[] = { 7, 0, 0, 0, 0, 0, 0,124, 0,124, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_062[] = { 7, 0, 0, 0, 0, 0, 96, 24, 6, 24, 96, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_063[] = { 7, 0, 0, 0, 0, 16, 0, 16, 16, 8, 8, 68, 68, 56, 0, 0, 0}; +static const GLubyte Helvetica12_Character_064[] = { 12, 0, 0, 0, 0, 0, 0, 31, 0, 32, 0, 77,128, 83, 64, 81, 32, 81, 32, 73, 32, 38,160, 48, 64, 15,128, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_065[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 20, 0, 8, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_066[] = { 8, 0, 0, 0, 0,124, 66, 66, 66,124, 66, 66, 66,124, 0, 0, 0}; +static const GLubyte Helvetica12_Character_067[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 33, 0, 30, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_068[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 66, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 66, 0,124, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_069[] = { 8, 0, 0, 0, 0,126, 64, 64, 64,126, 64, 64, 64,126, 0, 0, 0}; +static const GLubyte Helvetica12_Character_070[] = { 8, 0, 0, 0, 0, 64, 64, 64, 64,124, 64, 64, 64,126, 0, 0, 0}; +static const GLubyte Helvetica12_Character_071[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 29, 0, 35, 0, 65, 0, 65, 0, 71, 0, 64, 0, 64, 0, 33, 0, 30, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_072[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_073[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_074[] = { 7, 0, 0, 0, 0, 56, 68, 68, 4, 4, 4, 4, 4, 4, 0, 0, 0}; +static const GLubyte Helvetica12_Character_075[] = { 8, 0, 0, 0, 0, 65, 66, 68, 72,112, 80, 72, 68, 66, 0, 0, 0}; +static const GLubyte Helvetica12_Character_076[] = { 7, 0, 0, 0, 0,124, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_077[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 68, 64, 68, 64, 74, 64, 74, 64, 81, 64, 81, 64, 96,192, 96,192, 64, 64, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_078[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 67, 0, 69, 0, 69, 0, 73, 0, 81, 0, 81, 0, 97, 0, 65, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_079[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_080[] = { 8, 0, 0, 0, 0, 64, 64, 64, 64,124, 66, 66, 66,124, 0, 0, 0}; +static const GLubyte Helvetica12_Character_081[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30,128, 33, 0, 66,128, 68,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_082[] = { 8, 0, 0, 0, 0, 66, 66, 66, 68,124, 66, 66, 66,124, 0, 0, 0}; +static const GLubyte Helvetica12_Character_083[] = { 8, 0, 0, 0, 0, 60, 66, 66, 2, 12, 48, 64, 66, 60, 0, 0, 0}; +static const GLubyte Helvetica12_Character_084[] = { 7, 0, 0, 0, 0, 16, 16, 16, 16, 16, 16, 16, 16,254, 0, 0, 0}; +static const GLubyte Helvetica12_Character_085[] = { 8, 0, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 0, 0}; +static const GLubyte Helvetica12_Character_086[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 20, 0, 20, 0, 34, 0, 34, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_087[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 17, 0, 17, 0, 17, 0, 42,128, 42,128, 36,128, 68, 64, 68, 64, 68, 64, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_088[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 34, 0, 34, 0, 20, 0, 8, 0, 20, 0, 34, 0, 34, 0, 65, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_089[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 20, 0, 34, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_090[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 64, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0, 1, 0,127, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_091[] = { 3, 0, 96, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 96, 0, 0, 0}; +static const GLubyte Helvetica12_Character_092[] = { 4, 0, 0, 0, 0, 16, 16, 32, 32, 32, 64, 64,128,128, 0, 0, 0}; +static const GLubyte Helvetica12_Character_093[] = { 3, 0,192, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64,192, 0, 0, 0}; +static const GLubyte Helvetica12_Character_094[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0,136, 80, 32, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_095[] = { 7, 0, 0,254, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_096[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192,128, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_097[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_098[] = { 7, 0, 0, 0, 0, 88,100, 68, 68, 68,100, 88, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_099[] = { 7, 0, 0, 0, 0, 56, 68, 64, 64, 64, 68, 56, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_100[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 76, 52, 4, 4, 0, 0, 0}; +static const GLubyte Helvetica12_Character_101[] = { 7, 0, 0, 0, 0, 56, 68, 64,124, 68, 68, 56, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_102[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64,224, 64, 48, 0, 0, 0}; +static const GLubyte Helvetica12_Character_103[] = { 7, 0, 56, 68, 4, 52, 76, 68, 68, 68, 76, 52, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_104[] = { 7, 0, 0, 0, 0, 68, 68, 68, 68, 68,100, 88, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_105[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 0, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_106[] = { 3, 0,128, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_107[] = { 6, 0, 0, 0, 0, 68, 72, 80, 96, 96, 80, 72, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_108[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_109[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 73, 0, 73, 0, 73, 0, 73, 0, 73, 0,109, 0, 82, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_110[] = { 7, 0, 0, 0, 0, 68, 68, 68, 68, 68,100, 88, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_111[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_112[] = { 7, 0, 64, 64, 64, 88,100, 68, 68, 68,100, 88, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_113[] = { 7, 0, 4, 4, 4, 52, 76, 68, 68, 68, 76, 52, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_114[] = { 4, 0, 0, 0, 0, 64, 64, 64, 64, 64, 96, 80, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_115[] = { 6, 0, 0, 0, 0, 48, 72, 8, 48, 64, 72, 48, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_116[] = { 3, 0, 0, 0, 0, 96, 64, 64, 64, 64, 64,224, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_117[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 68, 68, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_118[] = { 7, 0, 0, 0, 0, 16, 16, 40, 40, 68, 68, 68, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_119[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 34, 0, 85, 0, 73, 0, 73, 0,136,128,136,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_120[] = { 6, 0, 0, 0, 0,132,132, 72, 48, 48, 72,132, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_121[] = { 7, 0, 64, 32, 16, 16, 40, 40, 72, 68, 68, 68, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_122[] = { 6, 0, 0, 0, 0,120, 64, 32, 32, 16, 8,120, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_123[] = { 4, 0, 48, 64, 64, 64, 64, 64,128, 64, 64, 64, 64, 48, 0, 0, 0}; +static const GLubyte Helvetica12_Character_124[] = { 3, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_125[] = { 4, 0,192, 32, 32, 32, 32, 32, 16, 32, 32, 32, 32,192, 0, 0, 0}; +static const GLubyte Helvetica12_Character_126[] = { 7, 0, 0, 0, 0, 0, 0, 0,152,100, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_127[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_128[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_129[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_130[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_131[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_132[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_133[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_134[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_135[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_136[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_137[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_138[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_139[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_140[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_141[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_142[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_143[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_144[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_145[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_146[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_147[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_148[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_149[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_150[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_151[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_152[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_153[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_154[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_155[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_156[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_157[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_158[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_159[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_160[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_161[] = { 3, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_162[] = { 7, 0, 0, 0, 32, 56,100, 80, 80, 80, 84, 56, 8, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_163[] = { 7, 0, 0, 0, 0, 88, 36, 16, 16,120, 32, 32, 36, 24, 0, 0, 0}; +static const GLubyte Helvetica12_Character_164[] = { 7, 0, 0, 0, 0, 0,132,120, 72, 72,120,132, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_165[] = { 7, 0, 0, 0, 0, 16, 16,124, 16,124, 16, 40, 68, 68, 0, 0, 0}; +static const GLubyte Helvetica12_Character_166[] = { 3, 0, 0, 64, 64, 64, 64, 0, 0, 0, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_167[] = { 6, 0,112,136, 8, 48, 72,136,136,144, 96,128,136,112, 0, 0, 0}; +static const GLubyte Helvetica12_Character_168[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,160, 0, 0, 0}; +static const GLubyte Helvetica12_Character_169[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 32,128, 78, 64, 81, 64, 80, 64, 81, 64, 78, 64, 32,128, 31, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_170[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0,112, 0, 80, 16,112, 0, 0, 0}; +static const GLubyte Helvetica12_Character_171[] = { 7, 0, 0, 0, 0, 0, 20, 40, 80, 40, 20, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_172[] = { 8, 0, 0, 0, 0, 0, 0, 2, 2, 2,126, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_173[] = { 5, 0, 0, 0, 0, 0, 0, 0,240, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_174[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 32,128, 74, 64, 74, 64, 76, 64, 74, 64, 78, 64, 32,128, 31, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_175[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,240, 0, 0, 0}; +static const GLubyte Helvetica12_Character_176[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 96,144,144, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_177[] = { 7, 0, 0, 0, 0,124, 0, 16, 16,124, 16, 16, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_178[] = { 4, 0, 0, 0, 0, 0, 0, 0,240, 64, 32,144, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_179[] = { 4, 0, 0, 0, 0, 0, 0, 0,192, 32, 64, 32,224, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_180[] = { 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,128, 64, 0, 0}; +static const GLubyte Helvetica12_Character_181[] = { 7, 0, 64, 64, 64,116, 76, 68, 68, 68, 68, 68, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_182[] = { 7, 0, 40, 40, 40, 40, 40, 40,104,232,232,232,104, 60, 0, 0, 0}; +static const GLubyte Helvetica12_Character_183[] = { 3, 0, 0, 0, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_184[] = { 3, 0,192, 32, 32, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_185[] = { 4, 0, 0, 0, 0, 0, 0, 0, 32, 32, 32, 96, 32, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_186[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0,112, 0,112, 80,112, 0, 0, 0}; +static const GLubyte Helvetica12_Character_187[] = { 7, 0, 0, 0, 0, 0, 80, 40, 20, 40, 80, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_188[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 39,128, 21, 0, 19, 0, 73, 0, 68, 0, 68, 0,194, 0, 65, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_189[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 71,128, 34, 0, 17, 0, 20,128, 75, 0, 72, 0, 68, 0,194, 0, 65, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_190[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 33, 0, 23,128, 21, 0, 11, 0,201, 0, 36, 0, 68, 0, 34, 0,225, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_191[] = { 7, 0, 56, 68, 68, 32, 32, 16, 16, 0, 16, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_192[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 0, 0, 8, 0, 16, 0}; +static const GLubyte Helvetica12_Character_193[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 0, 0, 8, 0, 4, 0}; +static const GLubyte Helvetica12_Character_194[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 0, 0, 20, 0, 8, 0}; +static const GLubyte Helvetica12_Character_195[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 0, 0, 20, 0, 10, 0}; +static const GLubyte Helvetica12_Character_196[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 0, 0, 20, 0, 0, 0}; +static const GLubyte Helvetica12_Character_197[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 8, 0, 20, 0, 8, 0}; +static const GLubyte Helvetica12_Character_198[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 71,192, 68, 0, 68, 0, 60, 0, 39,192, 36, 0, 20, 0, 20, 0, 15,192, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_199[] = { 9, 0, 0, 24, 0, 4, 0, 4, 0, 30, 0, 33, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 33, 0, 30, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_200[] = { 8, 0, 0, 0, 0,126, 64, 64, 64,126, 64, 64, 64,126, 0, 8, 16}; +static const GLubyte Helvetica12_Character_201[] = { 8, 0, 0, 0, 0,126, 64, 64, 64,126, 64, 64, 64,126, 0, 8, 4}; +static const GLubyte Helvetica12_Character_202[] = { 8, 0, 0, 0, 0,126, 64, 64, 64,126, 64, 64, 64,126, 0, 20, 8}; +static const GLubyte Helvetica12_Character_203[] = { 8, 0, 0, 0, 0,126, 64, 64, 64,126, 64, 64, 64,126, 0, 20, 0}; +static const GLubyte Helvetica12_Character_204[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64,128}; +static const GLubyte Helvetica12_Character_205[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64, 32}; +static const GLubyte Helvetica12_Character_206[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0,160, 64}; +static const GLubyte Helvetica12_Character_207[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0,160, 0}; +static const GLubyte Helvetica12_Character_208[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 66, 0, 65, 0, 65, 0,241, 0, 65, 0, 65, 0, 66, 0,124, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_209[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 67, 0, 69, 0, 69, 0, 73, 0, 81, 0, 81, 0, 97, 0, 65, 0, 0, 0, 20, 0, 10, 0}; +static const GLubyte Helvetica12_Character_210[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 4, 0, 8, 0}; +static const GLubyte Helvetica12_Character_211[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 4, 0, 2, 0}; +static const GLubyte Helvetica12_Character_212[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 10, 0, 4, 0}; +static const GLubyte Helvetica12_Character_213[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 20, 0, 10, 0}; +static const GLubyte Helvetica12_Character_214[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 18, 0, 0, 0}; +static const GLubyte Helvetica12_Character_215[] = { 7, 0, 0, 0, 0, 0, 68, 40, 16, 40, 68, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_216[] = { 10, 0, 0, 0, 0, 0, 0,128, 0, 94, 0, 33, 0, 80,128, 72,128, 68,128, 68,128, 66,128, 33, 0, 30,128, 0, 64, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_217[] = { 8, 0, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 8, 16}; +static const GLubyte Helvetica12_Character_218[] = { 8, 0, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 8, 4}; +static const GLubyte Helvetica12_Character_219[] = { 8, 0, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 20, 8}; +static const GLubyte Helvetica12_Character_220[] = { 8, 0, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 36, 0}; +static const GLubyte Helvetica12_Character_221[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 20, 0, 34, 0, 34, 0, 65, 0, 65, 0, 0, 0, 8, 0, 4, 0}; +static const GLubyte Helvetica12_Character_222[] = { 8, 0, 0, 0, 0, 64, 64,124, 66, 66, 66,124, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_223[] = { 7, 0, 0, 0, 0, 88, 68, 68, 68, 88, 68, 68, 68, 56, 0, 0, 0}; +static const GLubyte Helvetica12_Character_224[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 8, 16, 0, 0}; +static const GLubyte Helvetica12_Character_225[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 16, 8, 0, 0}; +static const GLubyte Helvetica12_Character_226[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 40, 16, 0, 0}; +static const GLubyte Helvetica12_Character_227[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 40, 20, 0, 0}; +static const GLubyte Helvetica12_Character_228[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 40, 0, 0, 0}; +static const GLubyte Helvetica12_Character_229[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 24, 36, 24, 0, 0}; +static const GLubyte Helvetica12_Character_230[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 59,128, 68, 64, 68, 0, 63,192, 4, 64, 68, 64, 59,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_231[] = { 7, 0, 48, 8, 16, 56, 68, 64, 64, 64, 68, 56, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_232[] = { 7, 0, 0, 0, 0, 56, 68, 64,124, 68, 68, 56, 0, 16, 32, 0, 0}; +static const GLubyte Helvetica12_Character_233[] = { 7, 0, 0, 0, 0, 56, 68, 64,124, 68, 68, 56, 0, 16, 8, 0, 0}; +static const GLubyte Helvetica12_Character_234[] = { 7, 0, 0, 0, 0, 56, 68, 64,124, 68, 68, 56, 0, 40, 16, 0, 0}; +static const GLubyte Helvetica12_Character_235[] = { 7, 0, 0, 0, 0, 56, 68, 64,124, 68, 68, 56, 0, 40, 0, 0, 0}; +static const GLubyte Helvetica12_Character_236[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 0, 64,128, 0, 0}; +static const GLubyte Helvetica12_Character_237[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 0, 64, 32, 0, 0}; +static const GLubyte Helvetica12_Character_238[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 0,160, 64, 0, 0}; +static const GLubyte Helvetica12_Character_239[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 0,160, 0, 0, 0}; +static const GLubyte Helvetica12_Character_240[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 60, 4, 40, 24, 52, 0, 0}; +static const GLubyte Helvetica12_Character_241[] = { 7, 0, 0, 0, 0, 68, 68, 68, 68, 68,100, 88, 0, 40, 20, 0, 0}; +static const GLubyte Helvetica12_Character_242[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 16, 32, 0, 0}; +static const GLubyte Helvetica12_Character_243[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 16, 8, 0, 0}; +static const GLubyte Helvetica12_Character_244[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 40, 16, 0, 0}; +static const GLubyte Helvetica12_Character_245[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 40, 20, 0, 0}; +static const GLubyte Helvetica12_Character_246[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 40, 0, 0, 0}; +static const GLubyte Helvetica12_Character_247[] = { 7, 0, 0, 0, 0, 0, 16, 0,124, 0, 16, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_248[] = { 7, 0, 0, 0, 0,184, 68,100, 84, 76, 68, 58, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica12_Character_249[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 68, 68, 0, 16, 32, 0, 0}; +static const GLubyte Helvetica12_Character_250[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 68, 68, 0, 16, 8, 0, 0}; +static const GLubyte Helvetica12_Character_251[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 68, 68, 0, 40, 16, 0, 0}; +static const GLubyte Helvetica12_Character_252[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 68, 68, 0, 40, 0, 0, 0}; +static const GLubyte Helvetica12_Character_253[] = { 7, 0, 64, 32, 16, 16, 40, 40, 72, 68, 68, 68, 0, 16, 8, 0, 0}; +static const GLubyte Helvetica12_Character_254[] = { 7, 0, 64, 64, 64, 88,100, 68, 68, 68,100, 88, 64, 64, 0, 0, 0}; +static const GLubyte Helvetica12_Character_255[] = { 7, 0, 96, 16, 16, 16, 24, 40, 40, 36, 68, 68, 0, 40, 0, 0, 0}; + +/* The font characters mapping: */ +static const GLubyte* Helvetica12_Character_Map[] = {Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032, + Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032, + Helvetica12_Character_032,Helvetica12_Character_033,Helvetica12_Character_034,Helvetica12_Character_035,Helvetica12_Character_036,Helvetica12_Character_037,Helvetica12_Character_038,Helvetica12_Character_039,Helvetica12_Character_040,Helvetica12_Character_041,Helvetica12_Character_042,Helvetica12_Character_043,Helvetica12_Character_044,Helvetica12_Character_045,Helvetica12_Character_046,Helvetica12_Character_047, + Helvetica12_Character_048,Helvetica12_Character_049,Helvetica12_Character_050,Helvetica12_Character_051,Helvetica12_Character_052,Helvetica12_Character_053,Helvetica12_Character_054,Helvetica12_Character_055,Helvetica12_Character_056,Helvetica12_Character_057,Helvetica12_Character_058,Helvetica12_Character_059,Helvetica12_Character_060,Helvetica12_Character_061,Helvetica12_Character_062,Helvetica12_Character_063, + Helvetica12_Character_064,Helvetica12_Character_065,Helvetica12_Character_066,Helvetica12_Character_067,Helvetica12_Character_068,Helvetica12_Character_069,Helvetica12_Character_070,Helvetica12_Character_071,Helvetica12_Character_072,Helvetica12_Character_073,Helvetica12_Character_074,Helvetica12_Character_075,Helvetica12_Character_076,Helvetica12_Character_077,Helvetica12_Character_078,Helvetica12_Character_079, + Helvetica12_Character_080,Helvetica12_Character_081,Helvetica12_Character_082,Helvetica12_Character_083,Helvetica12_Character_084,Helvetica12_Character_085,Helvetica12_Character_086,Helvetica12_Character_087,Helvetica12_Character_088,Helvetica12_Character_089,Helvetica12_Character_090,Helvetica12_Character_091,Helvetica12_Character_092,Helvetica12_Character_093,Helvetica12_Character_094,Helvetica12_Character_095, + Helvetica12_Character_096,Helvetica12_Character_097,Helvetica12_Character_098,Helvetica12_Character_099,Helvetica12_Character_100,Helvetica12_Character_101,Helvetica12_Character_102,Helvetica12_Character_103,Helvetica12_Character_104,Helvetica12_Character_105,Helvetica12_Character_106,Helvetica12_Character_107,Helvetica12_Character_108,Helvetica12_Character_109,Helvetica12_Character_110,Helvetica12_Character_111, + Helvetica12_Character_112,Helvetica12_Character_113,Helvetica12_Character_114,Helvetica12_Character_115,Helvetica12_Character_116,Helvetica12_Character_117,Helvetica12_Character_118,Helvetica12_Character_119,Helvetica12_Character_120,Helvetica12_Character_121,Helvetica12_Character_122,Helvetica12_Character_123,Helvetica12_Character_124,Helvetica12_Character_125,Helvetica12_Character_126,Helvetica12_Character_032, + Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032, + Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032, + Helvetica12_Character_160,Helvetica12_Character_161,Helvetica12_Character_162,Helvetica12_Character_163,Helvetica12_Character_164,Helvetica12_Character_165,Helvetica12_Character_166,Helvetica12_Character_167,Helvetica12_Character_168,Helvetica12_Character_169,Helvetica12_Character_170,Helvetica12_Character_171,Helvetica12_Character_172,Helvetica12_Character_173,Helvetica12_Character_174,Helvetica12_Character_175, + Helvetica12_Character_176,Helvetica12_Character_177,Helvetica12_Character_178,Helvetica12_Character_179,Helvetica12_Character_180,Helvetica12_Character_181,Helvetica12_Character_182,Helvetica12_Character_183,Helvetica12_Character_184,Helvetica12_Character_185,Helvetica12_Character_186,Helvetica12_Character_187,Helvetica12_Character_188,Helvetica12_Character_189,Helvetica12_Character_190,Helvetica12_Character_191, + Helvetica12_Character_192,Helvetica12_Character_193,Helvetica12_Character_194,Helvetica12_Character_195,Helvetica12_Character_196,Helvetica12_Character_197,Helvetica12_Character_198,Helvetica12_Character_199,Helvetica12_Character_200,Helvetica12_Character_201,Helvetica12_Character_202,Helvetica12_Character_203,Helvetica12_Character_204,Helvetica12_Character_205,Helvetica12_Character_206,Helvetica12_Character_207, + Helvetica12_Character_208,Helvetica12_Character_209,Helvetica12_Character_210,Helvetica12_Character_211,Helvetica12_Character_212,Helvetica12_Character_213,Helvetica12_Character_214,Helvetica12_Character_215,Helvetica12_Character_216,Helvetica12_Character_217,Helvetica12_Character_218,Helvetica12_Character_219,Helvetica12_Character_220,Helvetica12_Character_221,Helvetica12_Character_222,Helvetica12_Character_223, + Helvetica12_Character_224,Helvetica12_Character_225,Helvetica12_Character_226,Helvetica12_Character_227,Helvetica12_Character_228,Helvetica12_Character_229,Helvetica12_Character_230,Helvetica12_Character_231,Helvetica12_Character_232,Helvetica12_Character_233,Helvetica12_Character_234,Helvetica12_Character_235,Helvetica12_Character_236,Helvetica12_Character_237,Helvetica12_Character_238,Helvetica12_Character_239, + Helvetica12_Character_240,Helvetica12_Character_241,Helvetica12_Character_242,Helvetica12_Character_243,Helvetica12_Character_244,Helvetica12_Character_245,Helvetica12_Character_246,Helvetica12_Character_247,Helvetica12_Character_248,Helvetica12_Character_249,Helvetica12_Character_250,Helvetica12_Character_251,Helvetica12_Character_252,Helvetica12_Character_253,Helvetica12_Character_254,Helvetica12_Character_255,NULL}; + +/* The font structure: */ +const SFG_Font fgFontHelvetica12 = { "-adobe-helvetica-medium-r-normal--12-120-75-75-p-67-iso8859-1", 256, 16, Helvetica12_Character_Map, 0, 4 }; + +static const GLubyte Helvetica18_Character_000[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 64, 16, 0, 0, 64, 16, 0, 0, 64, 16, 0, 0, 64, 16, 0, 0, 64, 16, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_001[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_002[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_003[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_004[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_005[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_006[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_007[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_008[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_009[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_010[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_011[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_012[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_013[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_014[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_015[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_016[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_017[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_018[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_019[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_020[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_021[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_022[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_023[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_024[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_025[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_026[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_027[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_028[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_029[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_030[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_031[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_032[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_033[] = { 6, 0, 0, 0, 0, 0, 48, 48, 0, 0, 32, 32, 48, 48, 48, 48, 48, 48, 48, 48, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_034[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,144,144,216,216,216, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_035[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 36, 0, 36, 0, 36, 0,255,128,255,128, 18, 0, 18, 0, 18, 0,127,192,127,192, 9, 0, 9, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_036[] = { 10, 0, 0, 0, 0, 0, 0, 4, 0, 4, 0, 31, 0, 63,128,117,192,100,192, 4,192, 7,128, 31, 0, 60, 0,116, 0,100, 0,101,128, 63,128, 31, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_037[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 60, 12,126, 6,102, 6,102, 3,126, 3, 60, 1,128, 61,128,126,192,102,192,102, 96,126, 96, 60, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_038[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 56, 63,112,115,224, 97,192, 97,224, 99, 96,119, 96, 62, 0, 30, 0, 51, 0, 51, 0, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_039[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 64, 32, 32, 96, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_040[] = { 6, 0, 8, 24, 48, 48, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48, 48, 24, 8, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_041[] = { 6, 0, 64, 96, 48, 48, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 48, 48, 96, 64, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_042[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 68, 56, 56,124, 16, 16, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_043[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 12, 0, 12, 0,127,128,127,128, 12, 0, 12, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_044[] = { 5, 0, 0, 64, 32, 32, 96, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_045[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_046[] = { 5, 0, 0, 0, 0, 0, 96, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_047[] = { 5, 0, 0, 0, 0, 0,192,192, 64, 64, 96, 96, 32, 32, 48, 48, 16, 16, 24, 24, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_048[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63, 0, 51, 0, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 51, 0, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_049[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 62, 0, 62, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_050[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,128, 96, 0,112, 0, 56, 0, 28, 0, 14, 0, 7, 0, 3,128, 1,128, 97,128,127, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_051[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63, 0, 99,128, 97,128, 1,128, 3,128, 15, 0, 14, 0, 3, 0, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_052[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,128, 1,128, 1,128,127,192,127,192, 97,128, 49,128, 25,128, 25,128, 13,128, 7,128, 3,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_053[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0,127, 0, 99,128, 97,128, 1,128, 1,128, 99,128,127, 0,126, 0, 96, 0, 96, 0,127, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_054[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63, 0,113,128, 97,128, 97,128, 97,128,127, 0,110, 0, 96, 0, 96, 0, 49,128, 63,128, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_055[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 0, 48, 0, 24, 0, 24, 0, 24, 0, 12, 0, 12, 0, 6, 0, 6, 0, 3, 0, 1,128,127,128,127,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_056[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63, 0,115,128, 97,128, 97,128, 51, 0, 63, 0, 51, 0, 97,128, 97,128,115,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_057[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0,127, 0, 99, 0, 1,128, 1,128, 29,128, 63,128, 97,128, 97,128, 97,128, 99,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_058[] = { 5, 0, 0, 0, 0, 0, 96, 96, 0, 0, 0, 0, 0, 0, 96, 96, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_059[] = { 5, 0, 0, 64, 32, 32, 96, 96, 0, 0, 0, 0, 0, 0, 96, 96, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_060[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,128, 7,128, 30, 0, 56, 0, 96, 0, 56, 0, 30, 0, 7,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_061[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 63,128, 63,128, 0, 0, 0, 0, 63,128, 63,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_062[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0,120, 0, 30, 0, 7, 0, 1,128, 7, 0, 30, 0,120, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_063[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 0, 24, 0, 0, 0, 0, 0, 24, 0, 24, 0, 24, 0, 28, 0, 14, 0, 7, 0, 99, 0, 99, 0,127, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_064[] = { 18, 0, 0, 0, 0, 0, 0, 3,240, 0, 15,248, 0, 28, 0, 0, 56, 0, 0, 51,184, 0,103,252, 0,102,102, 0,102, 51, 0,102, 51, 0,102, 49,128, 99, 25,128, 51,185,128, 49,217,128, 24, 3, 0, 14, 7, 0, 7,254, 0, 1,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_065[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_066[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,224, 96,112, 96, 48, 96, 48, 96,112,127,224,127,192, 96,192, 96, 96, 96, 96, 96,224,127,192,127,128, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_067[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 0, 96, 0, 96, 0, 96, 0, 96, 0,112, 0, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_068[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,192, 96,224, 96, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 96, 96,224,127,192,127,128, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_069[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,192, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_070[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_071[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,216, 31,248, 56, 56, 48, 24,112, 24, 96,248, 96,248, 96, 0, 96, 0,112, 24, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_072[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48,127,240,127,240, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_073[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_074[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63, 0,115,128, 97,128, 97,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_075[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 56, 96,112, 96,224, 97,192, 99,128,103, 0,126, 0,124, 0,110, 0,103, 0, 99,128, 97,192, 96,224, 96,112, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_076[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_077[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,134, 97,134, 99,198, 98, 70,102,102,102,102,108, 54,108, 54,120, 30,120, 30,112, 14,112, 14, 96, 6, 96, 6, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_078[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 48, 96,112, 96,240, 96,240, 97,176, 99, 48, 99, 48,102, 48,102, 48,108, 48,120, 48,120, 48,112, 48, 96, 48, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_079[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_080[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,192, 96,224, 96, 96, 96, 96, 96,224,127,192,127,128, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_081[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 7,216, 31,240, 56,120, 48,216,112,220, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_082[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96,192, 96,192,127,128,127,192, 96,224, 96, 96, 96, 96, 96,224,127,192,127,128, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_083[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31,128, 63,224,112,112, 96, 48, 0, 48, 0,112, 1,224, 15,128, 62, 0,112, 0, 96, 48,112,112, 63,224, 15,128, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_084[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0,127,224,127,224, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_085[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 63,224, 48, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_086[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 7,128, 7,128, 12,192, 12,192, 12,192, 24, 96, 24, 96, 24, 96, 48, 48, 48, 48, 48, 48, 96, 24, 96, 24, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_087[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 12, 0, 12, 12, 0, 14, 28, 0, 26, 22, 0, 27, 54, 0, 27, 54, 0, 51, 51, 0, 51, 51, 0, 49, 35, 0, 49,227, 0, 97,225,128, 96,193,128, 96,193,128, 96,193,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_088[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 48,112,112, 48, 96, 56,224, 24,192, 13,128, 7, 0, 7, 0, 13,128, 24,192, 56,224, 48, 96,112,112, 96, 48, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_089[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 7,128, 12,192, 24, 96, 24, 96, 48, 48, 48, 48, 96, 24, 96, 24, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_090[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,224,127,224, 96, 0, 48, 0, 24, 0, 12, 0, 14, 0, 6, 0, 3, 0, 1,128, 0,192, 0, 96,127,224,127,224, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_091[] = { 5, 0,120,120, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96,120,120, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_092[] = { 5, 0, 0, 0, 0, 0, 24, 24, 16, 16, 48, 48, 32, 32, 96, 96, 64, 64,192,192, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_093[] = { 5, 0,240,240, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,240,240, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_094[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 99, 0, 54, 0, 28, 0, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_095[] = { 10, 0, 0,255,192,255,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_096[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 96, 64, 64, 32, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_097[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_098[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,111, 0,127,128,113,128, 96,192, 96,192, 96,192, 96,192,113,128,127,128,111, 0, 96, 0, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_099[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96, 0, 96, 0, 96, 0, 96, 0, 49,128, 63,128, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_100[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30,192, 63,192, 49,192, 96,192, 96,192, 96,192, 96,192, 49,192, 63,192, 30,192, 0,192, 0,192, 0,192, 0,192, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_101[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63,128,113,128, 96, 0, 96, 0,127,128, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_102[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48,252,252, 48, 48, 60, 28, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_103[] = { 11, 0, 0, 14, 0, 63,128, 49,128, 0,192, 30,192, 63,192, 49,192, 96,192, 96,192, 96,192, 96,192, 48,192, 63,192, 30,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_104[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128,113,128,111,128,103, 0, 96, 0, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_105[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0, 0, 96, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_106[] = { 4, 0,192,224, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0, 0, 96, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_107[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 99,128, 99, 0,103, 0,102, 0,108, 0,124, 0,120, 0,108, 0,102, 0, 99, 0, 96, 0, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_108[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_109[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 99, 24, 99, 24, 99, 24, 99, 24, 99, 24, 99, 24, 99, 24,115,152,111,120,102, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_110[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128,113,128,111,128,103, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_111[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_112[] = { 11, 0, 0, 96, 0, 96, 0, 96, 0, 96, 0,111, 0,127,128,113,128, 96,192, 96,192, 96,192, 96,192,113,128,127,128,111, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_113[] = { 11, 0, 0, 0,192, 0,192, 0,192, 0,192, 30,192, 63,192, 49,192, 96,192, 96,192, 96,192, 96,192, 49,192, 63,192, 30,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_114[] = { 6, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96,112,108,108, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_115[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0,126, 0, 99, 0, 3, 0, 31, 0,126, 0, 96, 0, 99, 0, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_116[] = { 6, 0, 0, 0, 0, 0, 24, 56, 48, 48, 48, 48, 48, 48,252,252, 48, 48, 48, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_117[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 57,128,125,128, 99,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_118[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 30, 0, 18, 0, 51, 0, 51, 0, 51, 0, 97,128, 97,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_119[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12,192, 12,192, 28,224, 20,160, 52,176, 51, 48, 51, 48, 99, 24, 99, 24, 99, 24, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_120[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,128,115,128, 51, 0, 30, 0, 12, 0, 12, 0, 30, 0, 51, 0,115,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_121[] = { 10, 0, 0, 56, 0, 56, 0, 12, 0, 12, 0, 12, 0, 12, 0, 30, 0, 18, 0, 51, 0, 51, 0, 51, 0, 97,128, 97,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_122[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0,127, 0, 96, 0, 48, 0, 24, 0, 12, 0, 6, 0, 3, 0,127, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_123[] = { 6, 0, 12, 24, 48, 48, 48, 48, 48, 48, 96,192, 96, 48, 48, 48, 48, 48, 24, 12, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_124[] = { 4, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_125[] = { 6, 0,192, 96, 48, 48, 48, 48, 48, 48, 24, 12, 24, 48, 48, 48, 48, 48, 96,192, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_126[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,102, 0, 63, 0, 25,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_127[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_128[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_129[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_130[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_131[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_132[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_133[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_134[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_135[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_136[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_137[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_138[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_139[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_140[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_141[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_142[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_143[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_144[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_145[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_146[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_147[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_148[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_149[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_150[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_151[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_152[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_153[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_154[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_155[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_156[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_157[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_158[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_159[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_160[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_161[] = { 6, 0, 48, 48, 48, 48, 48, 48, 48, 48, 16, 16, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_162[] = { 10, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 31, 0, 63,128, 53,128,100, 0,100, 0,100, 0,100, 0, 53,128, 63,128, 31, 0, 2, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_163[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,223, 0,255,128, 96,128, 48, 0, 24, 0, 24, 0,126, 0, 48, 0, 96, 0, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_164[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,128,127,128, 51, 0, 51, 0, 51, 0,127,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_165[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 12, 0, 12, 0,127,128, 12, 0,127,128, 30, 0, 51, 0, 51, 0, 51, 0, 97,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_166[] = { 4, 0, 0, 96, 96, 96, 96, 96, 96, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_167[] = { 10, 0, 0, 30, 0, 63, 0, 97,128, 97,128, 3,128, 7, 0, 31, 0, 57,128,113,128, 97,128, 99,128, 55, 0, 62, 0,120, 0, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_168[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,216,216, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_169[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 24, 48, 32, 8, 35,136, 68, 68, 72, 4, 72, 4, 72, 4, 68, 68, 35,136, 32, 8, 24, 48, 7,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_170[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 52,108, 36, 28,100, 56, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_171[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 27, 0, 54, 0,108, 0,108, 0, 54, 0, 27, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_172[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 0,192, 0,192,127,192,127,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_173[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0,124,124, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_174[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 24, 48, 32, 8, 36, 40, 68, 68, 68,132, 71,196, 68, 36, 68, 36, 39,200, 32, 8, 24, 48, 7,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_175[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,248, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_176[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,108, 68,108, 56, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_177[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,128, 0, 0, 12, 0, 12, 0, 12, 0,127,128,127,128, 12, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_178[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,248,248, 96, 48, 24,152,248,112, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_179[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,112,248,152, 48, 48,152,248,112, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_180[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 96, 48, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_181[] = { 10, 0, 0, 96, 0, 96, 0, 96, 0, 96, 0,109,128,127,128,115,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_182[] = { 10, 0, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 25, 0, 57, 0,121, 0,121, 0,121, 0,121, 0, 57, 0, 31,128, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_183[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_184[] = { 5, 0,240,216, 24,112, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_185[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48,112,112, 48, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_186[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 56,108, 68, 68,108, 56, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_187[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 72, 0,108, 0, 54, 0, 27, 0, 27, 0, 54, 0,108, 0, 72, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_188[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 24, 24, 24, 12,252, 6,216, 6,120, 51, 56, 49, 24, 49,136, 48,192, 48,192,112, 96,112, 48, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_189[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24,124, 24,124, 12, 48, 6, 24, 6, 12, 51, 76, 49,124, 49,184, 48,192, 48,192,112, 96,112, 48, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_190[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 24, 24, 24, 12,252, 6,216, 6,120,115, 56,249, 24,153,136, 48,192, 48,192,152, 96,248, 48,112, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_191[] = { 10, 0, 0, 62, 0,127, 0, 99, 0, 99, 0,112, 0, 56, 0, 28, 0, 12, 0, 12, 0, 12, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_192[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 6, 0, 12, 0, 24, 0}; +static const GLubyte Helvetica18_Character_193[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 6, 0, 3, 0, 1,128}; +static const GLubyte Helvetica18_Character_194[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 25,128, 15, 0, 6, 0}; +static const GLubyte Helvetica18_Character_195[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 19, 0, 22,128, 12,128}; +static const GLubyte Helvetica18_Character_196[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 25,128, 25,128, 0, 0}; +static const GLubyte Helvetica18_Character_197[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 15, 0, 25,128, 25,128, 15, 0}; +static const GLubyte Helvetica18_Character_198[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96,255,128, 96,255,128, 48,192, 0, 48,192, 0, 63,192, 0, 31,192, 0, 24,255, 0, 24,255, 0, 12,192, 0, 12,192, 0, 6,192, 0, 6,192, 0, 3,255,128, 3,255,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_199[] = { 14, 0, 0, 15, 0, 13,128, 1,128, 7, 0, 7,192, 31,240, 56, 56, 48, 24,112, 0, 96, 0, 96, 0, 96, 0, 96, 0,112, 0, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_200[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,192, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 6, 0, 12, 0, 24, 0}; +static const GLubyte Helvetica18_Character_201[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,192, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 6, 0, 3, 0, 1,128}; +static const GLubyte Helvetica18_Character_202[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,192, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 25,128, 15, 0, 6, 0}; +static const GLubyte Helvetica18_Character_203[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,192, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 25,128, 25,128, 0, 0}; +static const GLubyte Helvetica18_Character_204[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0, 48, 96,192}; +static const GLubyte Helvetica18_Character_205[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0, 48, 24, 12}; +static const GLubyte Helvetica18_Character_206[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0,204,120, 48}; +static const GLubyte Helvetica18_Character_207[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0,204,204, 0}; +static const GLubyte Helvetica18_Character_208[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,192, 96,224, 96, 96, 96, 48, 96, 48,252, 48,252, 48, 96, 48, 96, 48, 96, 96, 96,224,127,192,127,128, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_209[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 48, 96,112, 96,240, 96,240, 97,176, 99, 48, 99, 48,102, 48,102, 48,108, 48,108, 48,120, 48,112, 48,112, 48, 0, 0, 9,128, 11, 64, 6, 64}; +static const GLubyte Helvetica18_Character_210[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 1,128, 3, 0, 6, 0}; +static const GLubyte Helvetica18_Character_211[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 1,128, 0,192, 0, 96}; +static const GLubyte Helvetica18_Character_212[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 6, 96, 3,192, 1,128}; +static const GLubyte Helvetica18_Character_213[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 4,192, 5,160, 3, 32}; +static const GLubyte Helvetica18_Character_214[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 6,192, 6,192, 0, 0}; +static const GLubyte Helvetica18_Character_215[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192,192, 97,128, 51, 0, 30, 0, 12, 0, 30, 0, 51, 0, 97,128,192,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_216[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,199,192,255,240,120, 56, 56, 24,108, 28,110, 12,103, 12, 99,140, 97,204,112,220, 48,120, 56, 56, 31,252, 7,204, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_217[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 63,224, 48, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 3, 0, 6, 0, 12, 0}; +static const GLubyte Helvetica18_Character_218[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 63,224, 48, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 6, 0, 3, 0, 1,128}; +static const GLubyte Helvetica18_Character_219[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 63,224, 48, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 12,192, 7,128, 3, 0}; +static const GLubyte Helvetica18_Character_220[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 63,224, 48, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 12,192, 12,192, 0, 0}; +static const GLubyte Helvetica18_Character_221[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 7,128, 12,192, 24, 96, 24, 96, 48, 48, 48, 48, 96, 24, 96, 24, 0, 0, 3, 0, 1,128, 0,192}; +static const GLubyte Helvetica18_Character_222[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0, 96, 0, 96, 0,127,128,127,192, 96,224, 96, 96, 96, 96, 96,224,127,192,127,128, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_223[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,110, 0,111, 0, 99, 0, 99, 0, 99, 0, 99, 0,110, 0,110, 0, 99, 0, 99, 0, 99, 0, 99, 0, 62, 0, 28, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_224[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 12, 0, 24, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_225[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 24, 0, 12, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_226[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 51, 0, 30, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_227[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 38, 0, 45, 0, 25, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_228[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 54, 0, 54, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_229[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 28, 0, 54, 0, 54, 0, 28, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_230[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 58,240,119,252, 99,140, 99, 0,115, 0, 63,252, 7, 12, 99, 12,119,248, 62,240, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_231[] = { 10, 0, 0, 60, 0, 54, 0, 6, 0, 28, 0, 31, 0, 63,128, 49,128, 96, 0, 96, 0, 96, 0, 96, 0, 49,128, 63,128, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_232[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63,128,113,128, 96, 0, 96, 0,127,128, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 12, 0, 24, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_233[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63,128,113,128, 96, 0, 96, 0,127,128, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 12, 0, 6, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_234[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63,128,113,128, 96, 0, 96, 0,127,128, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 51, 0, 30, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_235[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63,128,113,128, 96, 0, 96, 0,127,128, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 27, 0, 27, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_236[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0, 48, 96,192, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_237[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0,192, 96, 48, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_238[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0,144,240, 96, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_239[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0,216,216, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_240[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 38, 0, 28, 0, 27, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_241[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128,113,128,111,128,103, 0, 0, 0, 38, 0, 45, 0, 25, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_242[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 6, 0, 12, 0, 24, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_243[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 12, 0, 6, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_244[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 25,128, 15, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_245[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 19, 0, 22,128, 12,128, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_246[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 27, 0, 27, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_247[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0,127,128,127,128, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_248[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,206, 0,127,128, 49,128,120,192,108,192,102,192, 99,192, 49,128, 63,192, 14, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_249[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 57,128,125,128, 99,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 6, 0, 12, 0, 24, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_250[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 57,128,125,128, 99,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 12, 0, 6, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_251[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 57,128,125,128, 99,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 51, 0, 30, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_252[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 57,128,125,128, 99,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 51, 0, 51, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_253[] = { 10, 0, 0, 56, 0, 56, 0, 12, 0, 12, 0, 12, 0, 12, 0, 30, 0, 18, 0, 51, 0, 51, 0, 51, 0, 97,128, 97,128, 97,128, 0, 0, 12, 0, 6, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_254[] = { 11, 0, 0, 96, 0, 96, 0, 96, 0, 96, 0,111, 0,127,128,113,128, 96,192, 96,192, 96,192, 96,192,113,128,127,128,111, 0, 96, 0, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte Helvetica18_Character_255[] = { 10, 0, 0, 56, 0, 56, 0, 12, 0, 12, 0, 12, 0, 12, 0, 30, 0, 18, 0, 51, 0, 51, 0, 51, 0, 97,128, 97,128, 97,128, 0, 0, 51, 0, 51, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; + +/* The font characters mapping: */ +static const GLubyte* Helvetica18_Character_Map[] = {Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032, + Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032, + Helvetica18_Character_032,Helvetica18_Character_033,Helvetica18_Character_034,Helvetica18_Character_035,Helvetica18_Character_036,Helvetica18_Character_037,Helvetica18_Character_038,Helvetica18_Character_039,Helvetica18_Character_040,Helvetica18_Character_041,Helvetica18_Character_042,Helvetica18_Character_043,Helvetica18_Character_044,Helvetica18_Character_045,Helvetica18_Character_046,Helvetica18_Character_047, + Helvetica18_Character_048,Helvetica18_Character_049,Helvetica18_Character_050,Helvetica18_Character_051,Helvetica18_Character_052,Helvetica18_Character_053,Helvetica18_Character_054,Helvetica18_Character_055,Helvetica18_Character_056,Helvetica18_Character_057,Helvetica18_Character_058,Helvetica18_Character_059,Helvetica18_Character_060,Helvetica18_Character_061,Helvetica18_Character_062,Helvetica18_Character_063, + Helvetica18_Character_064,Helvetica18_Character_065,Helvetica18_Character_066,Helvetica18_Character_067,Helvetica18_Character_068,Helvetica18_Character_069,Helvetica18_Character_070,Helvetica18_Character_071,Helvetica18_Character_072,Helvetica18_Character_073,Helvetica18_Character_074,Helvetica18_Character_075,Helvetica18_Character_076,Helvetica18_Character_077,Helvetica18_Character_078,Helvetica18_Character_079, + Helvetica18_Character_080,Helvetica18_Character_081,Helvetica18_Character_082,Helvetica18_Character_083,Helvetica18_Character_084,Helvetica18_Character_085,Helvetica18_Character_086,Helvetica18_Character_087,Helvetica18_Character_088,Helvetica18_Character_089,Helvetica18_Character_090,Helvetica18_Character_091,Helvetica18_Character_092,Helvetica18_Character_093,Helvetica18_Character_094,Helvetica18_Character_095, + Helvetica18_Character_096,Helvetica18_Character_097,Helvetica18_Character_098,Helvetica18_Character_099,Helvetica18_Character_100,Helvetica18_Character_101,Helvetica18_Character_102,Helvetica18_Character_103,Helvetica18_Character_104,Helvetica18_Character_105,Helvetica18_Character_106,Helvetica18_Character_107,Helvetica18_Character_108,Helvetica18_Character_109,Helvetica18_Character_110,Helvetica18_Character_111, + Helvetica18_Character_112,Helvetica18_Character_113,Helvetica18_Character_114,Helvetica18_Character_115,Helvetica18_Character_116,Helvetica18_Character_117,Helvetica18_Character_118,Helvetica18_Character_119,Helvetica18_Character_120,Helvetica18_Character_121,Helvetica18_Character_122,Helvetica18_Character_123,Helvetica18_Character_124,Helvetica18_Character_125,Helvetica18_Character_126,Helvetica18_Character_032, + Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032, + Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032, + Helvetica18_Character_160,Helvetica18_Character_161,Helvetica18_Character_162,Helvetica18_Character_163,Helvetica18_Character_164,Helvetica18_Character_165,Helvetica18_Character_166,Helvetica18_Character_167,Helvetica18_Character_168,Helvetica18_Character_169,Helvetica18_Character_170,Helvetica18_Character_171,Helvetica18_Character_172,Helvetica18_Character_173,Helvetica18_Character_174,Helvetica18_Character_175, + Helvetica18_Character_176,Helvetica18_Character_177,Helvetica18_Character_178,Helvetica18_Character_179,Helvetica18_Character_180,Helvetica18_Character_181,Helvetica18_Character_182,Helvetica18_Character_183,Helvetica18_Character_184,Helvetica18_Character_185,Helvetica18_Character_186,Helvetica18_Character_187,Helvetica18_Character_188,Helvetica18_Character_189,Helvetica18_Character_190,Helvetica18_Character_191, + Helvetica18_Character_192,Helvetica18_Character_193,Helvetica18_Character_194,Helvetica18_Character_195,Helvetica18_Character_196,Helvetica18_Character_197,Helvetica18_Character_198,Helvetica18_Character_199,Helvetica18_Character_200,Helvetica18_Character_201,Helvetica18_Character_202,Helvetica18_Character_203,Helvetica18_Character_204,Helvetica18_Character_205,Helvetica18_Character_206,Helvetica18_Character_207, + Helvetica18_Character_208,Helvetica18_Character_209,Helvetica18_Character_210,Helvetica18_Character_211,Helvetica18_Character_212,Helvetica18_Character_213,Helvetica18_Character_214,Helvetica18_Character_215,Helvetica18_Character_216,Helvetica18_Character_217,Helvetica18_Character_218,Helvetica18_Character_219,Helvetica18_Character_220,Helvetica18_Character_221,Helvetica18_Character_222,Helvetica18_Character_223, + Helvetica18_Character_224,Helvetica18_Character_225,Helvetica18_Character_226,Helvetica18_Character_227,Helvetica18_Character_228,Helvetica18_Character_229,Helvetica18_Character_230,Helvetica18_Character_231,Helvetica18_Character_232,Helvetica18_Character_233,Helvetica18_Character_234,Helvetica18_Character_235,Helvetica18_Character_236,Helvetica18_Character_237,Helvetica18_Character_238,Helvetica18_Character_239, + Helvetica18_Character_240,Helvetica18_Character_241,Helvetica18_Character_242,Helvetica18_Character_243,Helvetica18_Character_244,Helvetica18_Character_245,Helvetica18_Character_246,Helvetica18_Character_247,Helvetica18_Character_248,Helvetica18_Character_249,Helvetica18_Character_250,Helvetica18_Character_251,Helvetica18_Character_252,Helvetica18_Character_253,Helvetica18_Character_254,Helvetica18_Character_255,NULL}; + +/* The font structure: */ +const SFG_Font fgFontHelvetica18 = { "-adobe-helvetica-medium-r-normal--18-180-75-75-p-98-iso8859-1", 256, 23, Helvetica18_Character_Map, 0, 5 }; + +static const GLubyte TimesRoman10_Character_000[] = { 8, 0, 0, 0, 0,170, 0,130, 0,130, 0,170, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_001[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_002[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_003[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_004[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_005[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_006[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_007[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_008[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_009[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_010[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_011[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_012[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_013[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_014[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_015[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_016[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_017[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_018[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_019[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_020[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_021[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_022[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_023[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_024[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_025[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_026[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_027[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_028[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_029[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_030[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_031[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_032[] = { 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_033[] = { 3, 0, 0, 0, 0, 64, 0, 64, 64, 64, 64, 64, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_034[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0,160,160, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_035[] = { 5, 0, 0, 0, 0, 80, 80,248, 80,248, 80, 80, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_036[] = { 5, 0, 0, 0, 32,224,144, 16, 96,128,144,112, 32, 0, 0}; +static const GLubyte TimesRoman10_Character_037[] = { 8, 0, 0, 0, 0, 68, 42, 42, 86,168,164,126, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_038[] = { 8, 0, 0, 0, 0,118,141,152,116,110, 80, 48, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_039[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 64,192, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_040[] = { 4, 0, 0, 32, 64, 64,128,128,128, 64, 64, 32, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_041[] = { 4, 0, 0,128, 64, 64, 32, 32, 32, 64, 64,128, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_042[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0,160, 64,160, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_043[] = { 6, 0, 0, 0, 0, 32, 32,248, 32, 32, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_044[] = { 3, 0, 0, 64, 64, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_045[] = { 7, 0, 0, 0, 0, 0, 0,120, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_046[] = { 3, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_047[] = { 3, 0, 0, 0, 0,128,128, 64, 64, 64, 32, 32, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_048[] = { 5, 0, 0, 0, 0, 96,144,144,144,144,144, 96, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_049[] = { 5, 0, 0, 0, 0,112, 32, 32, 32, 32, 96, 32, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_050[] = { 5, 0, 0, 0, 0,240, 64, 32, 32, 16,144, 96, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_051[] = { 5, 0, 0, 0, 0,224, 16, 16, 96, 16,144, 96, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_052[] = { 5, 0, 0, 0, 0, 16, 16,248,144, 80, 48, 16, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_053[] = { 5, 0, 0, 0, 0,224,144, 16, 16,224, 64,112, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_054[] = { 5, 0, 0, 0, 0, 96,144,144,144,224, 64, 48, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_055[] = { 5, 0, 0, 0, 0, 64, 64, 64, 32, 32,144,240, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_056[] = { 5, 0, 0, 0, 0, 96,144,144, 96,144,144, 96, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_057[] = { 5, 0, 0, 0, 0,192, 32,112,144,144,144, 96, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_058[] = { 3, 0, 0, 0, 0, 64, 0, 0, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_059[] = { 3, 0, 0, 64, 64, 64, 0, 0, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_060[] = { 5, 0, 0, 0, 0, 16, 32, 64, 32, 16, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_061[] = { 6, 0, 0, 0, 0, 0,248, 0,248, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_062[] = { 5, 0, 0, 0, 0,128, 64, 32, 64,128, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_063[] = { 4, 0, 0, 0, 0, 64, 0, 64, 64, 32,160,224, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_064[] = { 9, 0, 0, 0, 0, 62, 0, 64, 0,146, 0,173, 0,165, 0,165, 0,157, 0, 66, 0, 60, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_065[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_066[] = { 6, 0, 0, 0, 0,240, 72, 72,112, 72, 72,240, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_067[] = { 7, 0, 0, 0, 0,120,196,128,128,128,196,124, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_068[] = { 7, 0, 0, 0, 0,248, 76, 68, 68, 68, 76,248, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_069[] = { 6, 0, 0, 0, 0,248, 72, 64,112, 64, 72,248, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_070[] = { 6, 0, 0, 0, 0,224, 64, 64,112, 64, 72,248, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_071[] = { 7, 0, 0, 0, 0,120,196,132,156,128,196,124, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_072[] = { 8, 0, 0, 0, 0,238, 68, 68,124, 68, 68,238, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_073[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,224, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_074[] = { 4, 0, 0, 0, 0,192,160, 32, 32, 32, 32,112, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_075[] = { 7, 0, 0, 0, 0,236, 72, 80, 96, 80, 72,236, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_076[] = { 6, 0, 0, 0, 0,248, 72, 64, 64, 64, 64,224, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_077[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0,235,128, 73, 0, 85, 0, 85, 0, 99, 0, 99, 0,227,128, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_078[] = { 8, 0, 0, 0, 0,228, 76, 76, 84, 84,100,238, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_079[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_080[] = { 6, 0, 0, 0, 0,224, 64, 64,112, 72, 72,240, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_081[] = { 7, 0, 0, 12, 24,112,204,132,132,132,204,120, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_082[] = { 7, 0, 0, 0, 0,236, 72, 80,112, 72, 72,240, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_083[] = { 5, 0, 0, 0, 0,224,144, 16, 96,192,144,112, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_084[] = { 6, 0, 0, 0, 0,112, 32, 32, 32, 32,168,248, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_085[] = { 8, 0, 0, 0, 0, 56,108, 68, 68, 68, 68,238, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_086[] = { 8, 0, 0, 0, 0, 16, 16, 40, 40,108, 68,238, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_087[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 34, 0, 85, 0, 85, 0,201,128,136,128,221,192, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_088[] = { 8, 0, 0, 0, 0,238, 68, 40, 16, 40, 68,238, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_089[] = { 8, 0, 0, 0, 0, 56, 16, 16, 40, 40, 68,238, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_090[] = { 6, 0, 0, 0, 0,248,136, 64, 32, 16,136,248, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_091[] = { 3, 0, 0,192,128,128,128,128,128,128,128,192, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_092[] = { 3, 0, 0, 0, 0, 32, 32, 64, 64, 64,128,128, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_093[] = { 3, 0, 0,192, 64, 64, 64, 64, 64, 64, 64,192, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_094[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 80, 80, 32, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_095[] = { 5, 0,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_096[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192,128, 0, 0}; +static const GLubyte TimesRoman10_Character_097[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_098[] = { 5, 0, 0, 0, 0,224,144,144,144,224,128,128, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_099[] = { 4, 0, 0, 0, 0, 96,128,128,128, 96, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_100[] = { 5, 0, 0, 0, 0,104,144,144,144,112, 16, 48, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_101[] = { 4, 0, 0, 0, 0, 96,128,192,160, 96, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_102[] = { 4, 0, 0, 0, 0,224, 64, 64, 64,224, 64, 48, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_103[] = { 5, 0, 0,224,144, 96, 64,160,160,112, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_104[] = { 5, 0, 0, 0, 0,216,144,144,144,224,128,128, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_105[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64,192, 0, 64, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_106[] = { 3, 0, 0,128, 64, 64, 64, 64, 64,192, 0, 64, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_107[] = { 5, 0, 0, 0, 0,152,144,224,160,144,128,128, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_108[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,192, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_109[] = { 8, 0, 0, 0, 0,219,146,146,146,236, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_110[] = { 5, 0, 0, 0, 0,216,144,144,144,224, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_111[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_112[] = { 5, 0, 0,192,128,224,144,144,144,224, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_113[] = { 5, 0, 0, 56, 16,112,144,144,144,112, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_114[] = { 4, 0, 0, 0, 0,224, 64, 64, 96,160, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_115[] = { 4, 0, 0, 0, 0,224, 32, 96,128,224, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_116[] = { 4, 0, 0, 0, 0, 48, 64, 64, 64,224, 64, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_117[] = { 5, 0, 0, 0, 0,104,144,144,144,144, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_118[] = { 5, 0, 0, 0, 0, 32, 96, 80,144,216, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_119[] = { 8, 0, 0, 0, 0, 40,108, 84,146,219, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_120[] = { 6, 0, 0, 0, 0,216, 80, 32, 80,216, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_121[] = { 5, 0, 0,128,128, 64, 96,160,144,184, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_122[] = { 5, 0, 0, 0, 0,240,144, 64, 32,240, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_123[] = { 4, 0, 0, 32, 64, 64, 64,128, 64, 64, 64, 32, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_124[] = { 2, 0, 0,128,128,128,128,128,128,128,128,128, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_125[] = { 4, 0, 0,128, 64, 64, 64, 32, 64, 64, 64,128, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_126[] = { 7, 0, 0, 0, 0, 0, 0,152,100, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_127[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_128[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_129[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_130[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_131[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_132[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_133[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_134[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_135[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_136[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_137[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_138[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_139[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_140[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_141[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_142[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_143[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_144[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_145[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_146[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_147[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_148[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_149[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_150[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_151[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_152[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_153[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_154[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_155[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_156[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_157[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_158[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_159[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_160[] = { 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_161[] = { 3, 0, 0, 64, 64, 64, 64, 64, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_162[] = { 5, 0, 0, 0,128,224,144,128,144,112, 16, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_163[] = { 5, 0, 0, 0, 0,240,200, 64,224, 64, 80, 48, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_164[] = { 5, 0, 0, 0, 0, 0,136,112, 80, 80,112,136, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_165[] = { 5, 0, 0, 0, 0,112, 32,248, 32,216, 80,136, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_166[] = { 2, 0, 0, 0, 0,128,128,128, 0,128,128,128, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_167[] = { 5, 0, 0, 0,224,144, 32, 80,144,160, 64,144,112, 0, 0}; +static const GLubyte TimesRoman10_Character_168[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 80, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_169[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 77, 0, 81, 0, 77, 0, 34, 0, 28, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_170[] = { 4, 0, 0, 0, 0, 0, 0,224, 0,160, 32,192, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_171[] = { 5, 0, 0, 0, 0, 0, 80,160,160, 80, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_172[] = { 7, 0, 0, 0, 0, 0, 4, 4,124, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_173[] = { 4, 0, 0, 0, 0, 0, 0,224, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_174[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 85, 0, 89, 0, 93, 0, 34, 0, 28, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_175[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,224, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_176[] = { 4, 0, 0, 0, 0, 0, 0, 0, 96,144,144, 96, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_177[] = { 6, 0, 0, 0, 0,248, 0, 32, 32,248, 32, 32, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_178[] = { 3, 0, 0, 0, 0, 0, 0, 0,224, 64,160, 96, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_179[] = { 3, 0, 0, 0, 0, 0, 0, 0,192, 32, 64,224, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_180[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0,128, 64, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_181[] = { 5, 0, 0,128,128,232,144,144,144,144, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_182[] = { 6, 0, 0, 40, 40, 40, 40,104,232,232,232,124, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_183[] = { 2, 0, 0, 0, 0, 0, 0,128, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_184[] = { 4, 0,192, 32, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_185[] = { 3, 0, 0, 0, 0, 0, 0, 0,224, 64,192, 64, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_186[] = { 4, 0, 0, 0, 0, 0, 0,224, 0, 64,160, 64, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_187[] = { 5, 0, 0, 0, 0, 0,160, 80, 80,160, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_188[] = { 8, 0, 0, 0, 0, 68, 62, 44,244, 72,200, 68, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_189[] = { 8, 0, 0, 0, 0, 78, 36, 42,246, 72,200, 68, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_190[] = { 8, 0, 0, 0, 0, 68, 62, 44,212, 40, 72,228, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_191[] = { 4, 0, 0,224,160,128, 64, 64, 0, 64, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_192[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 16, 32}; +static const GLubyte TimesRoman10_Character_193[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 16, 8}; +static const GLubyte TimesRoman10_Character_194[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 40, 16}; +static const GLubyte TimesRoman10_Character_195[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 40, 20}; +static const GLubyte TimesRoman10_Character_196[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 40, 0}; +static const GLubyte TimesRoman10_Character_197[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 16, 40, 16}; +static const GLubyte TimesRoman10_Character_198[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,239, 0, 73, 0,120, 0, 46, 0, 40, 0, 57, 0, 31, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_199[] = { 7, 0, 96, 16, 32,120,196,128,128,128,196,124, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_200[] = { 6, 0, 0, 0, 0,248, 72, 64,112, 64, 72,248, 0, 32, 64}; +static const GLubyte TimesRoman10_Character_201[] = { 6, 0, 0, 0, 0,248, 72, 64,112, 64, 72,248, 0, 32, 16}; +static const GLubyte TimesRoman10_Character_202[] = { 6, 0, 0, 0, 0,248, 72, 64,112, 64, 72,248, 0, 80, 32}; +static const GLubyte TimesRoman10_Character_203[] = { 6, 0, 0, 0, 0,248, 72, 64,112, 64, 72,248, 0, 80, 0}; +static const GLubyte TimesRoman10_Character_204[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,224, 0, 64,128}; +static const GLubyte TimesRoman10_Character_205[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,224, 0, 64, 32}; +static const GLubyte TimesRoman10_Character_206[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,224, 0,160, 64}; +static const GLubyte TimesRoman10_Character_207[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,224, 0,160, 0}; +static const GLubyte TimesRoman10_Character_208[] = { 7, 0, 0, 0, 0,248, 76, 68,228, 68, 76,248, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_209[] = { 8, 0, 0, 0, 0,228, 76, 76, 84, 84,100,238, 0, 80, 40}; +static const GLubyte TimesRoman10_Character_210[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 32, 64}; +static const GLubyte TimesRoman10_Character_211[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 16, 8}; +static const GLubyte TimesRoman10_Character_212[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 80, 32}; +static const GLubyte TimesRoman10_Character_213[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 80, 40}; +static const GLubyte TimesRoman10_Character_214[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 80, 0}; +static const GLubyte TimesRoman10_Character_215[] = { 6, 0, 0, 0, 0,136, 80, 32, 80,136, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_216[] = { 8, 0, 0, 0,128,124,102, 82, 82, 74,102, 62, 1, 0, 0}; +static const GLubyte TimesRoman10_Character_217[] = { 8, 0, 0, 0, 0, 56,108, 68, 68, 68, 68,238, 0, 16, 32}; +static const GLubyte TimesRoman10_Character_218[] = { 8, 0, 0, 0, 0, 56,108, 68, 68, 68, 68,238, 0, 16, 8}; +static const GLubyte TimesRoman10_Character_219[] = { 8, 0, 0, 0, 0, 56,108, 68, 68, 68, 68,238, 0, 40, 16}; +static const GLubyte TimesRoman10_Character_220[] = { 8, 0, 0, 0, 0, 56,108, 68, 68, 68, 68,238, 0, 40, 0}; +static const GLubyte TimesRoman10_Character_221[] = { 8, 0, 0, 0, 0, 56, 16, 16, 40, 40, 68,238, 0, 16, 8}; +static const GLubyte TimesRoman10_Character_222[] = { 6, 0, 0, 0, 0,224, 64,112, 72,112, 64,224, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_223[] = { 5, 0, 0, 0, 0,224, 80, 80, 96, 80, 80, 32, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_224[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0, 64,128, 0, 0}; +static const GLubyte TimesRoman10_Character_225[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0, 64, 32, 0, 0}; +static const GLubyte TimesRoman10_Character_226[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0,160, 64, 0, 0}; +static const GLubyte TimesRoman10_Character_227[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0,160, 80, 0, 0}; +static const GLubyte TimesRoman10_Character_228[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0,160, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_229[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 64,160, 64, 0, 0}; +static const GLubyte TimesRoman10_Character_230[] = { 6, 0, 0, 0, 0,216,160,112, 40,216, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_231[] = { 4, 0,192, 32, 64, 96,128,128,128, 96, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_232[] = { 4, 0, 0, 0, 0, 96,128,192,160, 96, 0, 64,128, 0, 0}; +static const GLubyte TimesRoman10_Character_233[] = { 4, 0, 0, 0, 0, 96,128,192,160, 96, 0, 64, 32, 0, 0}; +static const GLubyte TimesRoman10_Character_234[] = { 4, 0, 0, 0, 0, 96,128,192,160, 96, 0,160, 64, 0, 0}; +static const GLubyte TimesRoman10_Character_235[] = { 4, 0, 0, 0, 0, 96,128,192,160, 96, 0,160, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_236[] = { 4, 0, 0, 0, 0,224, 64, 64, 64,192, 0, 64,128, 0, 0}; +static const GLubyte TimesRoman10_Character_237[] = { 4, 0, 0, 0, 0,224, 64, 64, 64,192, 0, 64, 32, 0, 0}; +static const GLubyte TimesRoman10_Character_238[] = { 4, 0, 0, 0, 0,224, 64, 64, 64,192, 0,160, 64, 0, 0}; +static const GLubyte TimesRoman10_Character_239[] = { 4, 0, 0, 0, 0,224, 64, 64, 64,192, 0,160, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_240[] = { 5, 0, 0, 0, 0, 96,144,144,144,112,160,112, 64, 0, 0}; +static const GLubyte TimesRoman10_Character_241[] = { 5, 0, 0, 0, 0,216,144,144,144,224, 0,160, 80, 0, 0}; +static const GLubyte TimesRoman10_Character_242[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0, 32, 64, 0, 0}; +static const GLubyte TimesRoman10_Character_243[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0, 64, 32, 0, 0}; +static const GLubyte TimesRoman10_Character_244[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0,160, 64, 0, 0}; +static const GLubyte TimesRoman10_Character_245[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0,160, 80, 0, 0}; +static const GLubyte TimesRoman10_Character_246[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0,160, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_247[] = { 6, 0, 0, 0, 0, 32, 0,248, 0, 32, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_248[] = { 5, 0, 0, 0, 0,224,144,144,144,112, 8, 0, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_249[] = { 5, 0, 0, 0, 0,104,144,144,144,144, 0, 32, 64, 0, 0}; +static const GLubyte TimesRoman10_Character_250[] = { 5, 0, 0, 0, 0,104,144,144,144,144, 0, 64, 32, 0, 0}; +static const GLubyte TimesRoman10_Character_251[] = { 5, 0, 0, 0, 0,104,144,144,144,144, 0, 80, 32, 0, 0}; +static const GLubyte TimesRoman10_Character_252[] = { 5, 0, 0, 0, 0,104,144,144,144,144, 0, 80, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_253[] = { 5, 0, 0,128,192, 64, 96,160,144,184, 0, 32, 16, 0, 0}; +static const GLubyte TimesRoman10_Character_254[] = { 5, 0, 0,192,128,224,144,144,144,224,128,128, 0, 0, 0}; +static const GLubyte TimesRoman10_Character_255[] = { 5, 0, 0,128,192, 64, 96,160,144,184, 0,160, 0, 0, 0}; + +/* The font characters mapping: */ +static const GLubyte* TimesRoman10_Character_Map[] = {TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032, + TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032, + TimesRoman10_Character_032,TimesRoman10_Character_033,TimesRoman10_Character_034,TimesRoman10_Character_035,TimesRoman10_Character_036,TimesRoman10_Character_037,TimesRoman10_Character_038,TimesRoman10_Character_039,TimesRoman10_Character_040,TimesRoman10_Character_041,TimesRoman10_Character_042,TimesRoman10_Character_043,TimesRoman10_Character_044,TimesRoman10_Character_045,TimesRoman10_Character_046,TimesRoman10_Character_047, + TimesRoman10_Character_048,TimesRoman10_Character_049,TimesRoman10_Character_050,TimesRoman10_Character_051,TimesRoman10_Character_052,TimesRoman10_Character_053,TimesRoman10_Character_054,TimesRoman10_Character_055,TimesRoman10_Character_056,TimesRoman10_Character_057,TimesRoman10_Character_058,TimesRoman10_Character_059,TimesRoman10_Character_060,TimesRoman10_Character_061,TimesRoman10_Character_062,TimesRoman10_Character_063, + TimesRoman10_Character_064,TimesRoman10_Character_065,TimesRoman10_Character_066,TimesRoman10_Character_067,TimesRoman10_Character_068,TimesRoman10_Character_069,TimesRoman10_Character_070,TimesRoman10_Character_071,TimesRoman10_Character_072,TimesRoman10_Character_073,TimesRoman10_Character_074,TimesRoman10_Character_075,TimesRoman10_Character_076,TimesRoman10_Character_077,TimesRoman10_Character_078,TimesRoman10_Character_079, + TimesRoman10_Character_080,TimesRoman10_Character_081,TimesRoman10_Character_082,TimesRoman10_Character_083,TimesRoman10_Character_084,TimesRoman10_Character_085,TimesRoman10_Character_086,TimesRoman10_Character_087,TimesRoman10_Character_088,TimesRoman10_Character_089,TimesRoman10_Character_090,TimesRoman10_Character_091,TimesRoman10_Character_092,TimesRoman10_Character_093,TimesRoman10_Character_094,TimesRoman10_Character_095, + TimesRoman10_Character_096,TimesRoman10_Character_097,TimesRoman10_Character_098,TimesRoman10_Character_099,TimesRoman10_Character_100,TimesRoman10_Character_101,TimesRoman10_Character_102,TimesRoman10_Character_103,TimesRoman10_Character_104,TimesRoman10_Character_105,TimesRoman10_Character_106,TimesRoman10_Character_107,TimesRoman10_Character_108,TimesRoman10_Character_109,TimesRoman10_Character_110,TimesRoman10_Character_111, + TimesRoman10_Character_112,TimesRoman10_Character_113,TimesRoman10_Character_114,TimesRoman10_Character_115,TimesRoman10_Character_116,TimesRoman10_Character_117,TimesRoman10_Character_118,TimesRoman10_Character_119,TimesRoman10_Character_120,TimesRoman10_Character_121,TimesRoman10_Character_122,TimesRoman10_Character_123,TimesRoman10_Character_124,TimesRoman10_Character_125,TimesRoman10_Character_126,TimesRoman10_Character_032, + TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032, + TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032, + TimesRoman10_Character_160,TimesRoman10_Character_161,TimesRoman10_Character_162,TimesRoman10_Character_163,TimesRoman10_Character_164,TimesRoman10_Character_165,TimesRoman10_Character_166,TimesRoman10_Character_167,TimesRoman10_Character_168,TimesRoman10_Character_169,TimesRoman10_Character_170,TimesRoman10_Character_171,TimesRoman10_Character_172,TimesRoman10_Character_173,TimesRoman10_Character_174,TimesRoman10_Character_175, + TimesRoman10_Character_176,TimesRoman10_Character_177,TimesRoman10_Character_178,TimesRoman10_Character_179,TimesRoman10_Character_180,TimesRoman10_Character_181,TimesRoman10_Character_182,TimesRoman10_Character_183,TimesRoman10_Character_184,TimesRoman10_Character_185,TimesRoman10_Character_186,TimesRoman10_Character_187,TimesRoman10_Character_188,TimesRoman10_Character_189,TimesRoman10_Character_190,TimesRoman10_Character_191, + TimesRoman10_Character_192,TimesRoman10_Character_193,TimesRoman10_Character_194,TimesRoman10_Character_195,TimesRoman10_Character_196,TimesRoman10_Character_197,TimesRoman10_Character_198,TimesRoman10_Character_199,TimesRoman10_Character_200,TimesRoman10_Character_201,TimesRoman10_Character_202,TimesRoman10_Character_203,TimesRoman10_Character_204,TimesRoman10_Character_205,TimesRoman10_Character_206,TimesRoman10_Character_207, + TimesRoman10_Character_208,TimesRoman10_Character_209,TimesRoman10_Character_210,TimesRoman10_Character_211,TimesRoman10_Character_212,TimesRoman10_Character_213,TimesRoman10_Character_214,TimesRoman10_Character_215,TimesRoman10_Character_216,TimesRoman10_Character_217,TimesRoman10_Character_218,TimesRoman10_Character_219,TimesRoman10_Character_220,TimesRoman10_Character_221,TimesRoman10_Character_222,TimesRoman10_Character_223, + TimesRoman10_Character_224,TimesRoman10_Character_225,TimesRoman10_Character_226,TimesRoman10_Character_227,TimesRoman10_Character_228,TimesRoman10_Character_229,TimesRoman10_Character_230,TimesRoman10_Character_231,TimesRoman10_Character_232,TimesRoman10_Character_233,TimesRoman10_Character_234,TimesRoman10_Character_235,TimesRoman10_Character_236,TimesRoman10_Character_237,TimesRoman10_Character_238,TimesRoman10_Character_239, + TimesRoman10_Character_240,TimesRoman10_Character_241,TimesRoman10_Character_242,TimesRoman10_Character_243,TimesRoman10_Character_244,TimesRoman10_Character_245,TimesRoman10_Character_246,TimesRoman10_Character_247,TimesRoman10_Character_248,TimesRoman10_Character_249,TimesRoman10_Character_250,TimesRoman10_Character_251,TimesRoman10_Character_252,TimesRoman10_Character_253,TimesRoman10_Character_254,TimesRoman10_Character_255,NULL}; + +/* The font structure: */ +const SFG_Font fgFontTimesRoman10 = { "-adobe-times-medium-r-normal--10-100-75-75-p-54-iso8859-1", 256, 14, TimesRoman10_Character_Map, 0, 4 }; + +static const GLubyte TimesRoman24_Character_000[] = { 19, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_001[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_002[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_003[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_004[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_005[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_006[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_007[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_008[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_009[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_010[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_011[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_012[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_013[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_014[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_015[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_016[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_017[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_018[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_019[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_020[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_021[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_022[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_023[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_024[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_025[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_026[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_027[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_028[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_029[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_030[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_031[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_032[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_033[] = { 8, 0, 0, 0, 0, 0, 0, 0, 24, 24, 0, 0, 0, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_034[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 68, 0,102, 0,102, 0,102, 0,102, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_035[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 17, 0, 17, 0, 17, 0, 17, 0, 17, 0,127,224,127,224, 8,128, 8,128, 8,128, 63,240, 63,240, 4, 64, 4, 64, 4, 64, 4, 64, 4, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_036[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4, 0, 4, 0, 63, 0,229,192,196,192,132, 96,132, 96, 4, 96, 4,224, 7,192, 7,128, 30, 0, 60, 0,116, 0,100, 0,100, 32,100, 96, 52,224, 31,128, 4, 0, 4, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_037[] = { 19, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 30, 0, 12, 57, 0, 6, 48,128, 2, 48, 64, 3, 48, 64, 1,152, 64, 0,140,192, 0,199,128, 60, 96, 0,114, 32, 0, 97, 48, 0, 96,152, 0, 96,136, 0, 48,140, 0, 25,254, 0, 15, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_038[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 30, 0, 63,191, 0,112,240,128, 96, 96, 0, 96,224, 0, 96,208, 0, 49,144, 0, 27,136, 0, 15, 12, 0, 7, 31, 0, 7,128, 0, 14,192, 0, 12, 96, 0, 12, 32, 0, 12, 32, 0, 6, 96, 0, 3,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_039[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 12, 4, 28, 24, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_040[] = { 8, 0, 0, 2, 4, 8, 24, 16, 48, 48, 96, 96, 96, 96, 96, 96, 96, 96, 48, 48, 16, 24, 8, 4, 2, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_041[] = { 8, 0, 0, 64, 32, 16, 24, 8, 12, 12, 6, 6, 6, 6, 6, 6, 6, 6, 12, 12, 8, 24, 16, 32, 64, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_042[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 7, 0, 50, 96, 58,224, 7, 0, 58,224, 50, 96, 7, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_043[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0,127,248,127,248, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_044[] = { 7, 0, 0, 0, 0, 48, 24, 8, 56, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_045[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,248,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_046[] = { 6, 0, 0, 0, 0, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_047[] = { 7, 0, 0, 0, 0,192,192,192, 64, 96, 96, 32, 48, 48, 16, 24, 24, 8, 12, 12, 4, 6, 6, 6, 6, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_048[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 25,128, 48,192, 48,192,112,224, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 48,192, 25,128, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_049[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 63,192, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 30, 0, 6, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_050[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,224, 48, 32, 24, 0, 12, 0, 6, 0, 2, 0, 3, 0, 1,128, 1,128, 0,192, 0,192, 64,192, 64,192, 33,192, 63,128, 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_051[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0,115, 0, 97,128, 0,128, 0,192, 0,192, 0,192, 1,192, 3,128, 15, 0, 6, 0, 3, 0, 65,128, 65,128, 35,128, 63, 0, 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_052[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,128, 1,128, 1,128, 1,128,127,224,127,224, 97,128, 33,128, 49,128, 17,128, 25,128, 9,128, 13,128, 5,128, 3,128, 3,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_053[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 63, 0,113,192, 96,192, 0, 96, 0, 96, 0, 96, 0, 96, 0,224, 1,192, 7,192, 63, 0, 60, 0, 48, 0, 16, 0, 16, 0, 15,192, 15,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_054[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 61,192, 48,192,112, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96,192,121,192,119, 0, 48, 0, 56, 0, 24, 0, 12, 0, 7, 0, 1,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_055[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 6, 0, 6, 0, 6, 0, 2, 0, 3, 0, 3, 0, 1, 0, 1,128, 1,128, 0,128, 0,192, 64,192, 96, 96,127,224, 63,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_056[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192,112,192, 96, 96, 96, 96, 96, 96, 32,224, 48,192, 27,128, 15, 0, 15, 0, 25,128, 48,192, 48,192, 48,192, 25,128, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_057[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120, 0, 14, 0, 3, 0, 1,128, 1,192, 0,192, 14,192, 57,224, 48,224, 96, 96, 96, 96, 96, 96, 96, 96, 96,224, 48,192, 59,192, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_058[] = { 6, 0, 0, 0, 0, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_059[] = { 7, 0, 0, 0, 0, 48, 24, 8, 56, 48, 0, 0, 0, 0, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_060[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 0,224, 3,128, 14, 0, 56, 0, 96, 0, 56, 0, 14, 0, 3,128, 0,224, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_061[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,248,127,248, 0, 0, 0, 0,127,248,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_062[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0, 56, 0, 14, 0, 3,128, 0,224, 0, 48, 0,224, 3,128, 14, 0, 56, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_063[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 4, 0, 4, 0, 4, 0, 6, 0, 6, 0, 3, 0, 3,128, 1,192, 48,192, 48,192, 32,192, 49,128, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_064[] = { 22, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 0, 3,131, 0, 6, 0, 0, 12, 0, 0, 24,119,128, 24,222,192, 49,142, 96, 49,134, 32, 49,134, 48, 49,134, 16, 49,131, 16, 48,195, 16, 48,227, 16, 56,127, 16, 24, 59, 48, 28, 0, 32, 14, 0, 96, 7, 0,192, 3,195,128, 0,254, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_065[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_066[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,240, 24, 60, 24, 12, 24, 6, 24, 6, 24, 6, 24, 12, 24, 28, 31,240, 24, 32, 24, 24, 24, 12, 24, 12, 24, 12, 24, 24, 24, 56,127,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_067[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 15, 28, 28, 4, 48, 2, 48, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 48, 2, 48, 2, 28, 6, 14, 30, 3,242, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_068[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,224, 0, 24, 56, 0, 24, 28, 0, 24, 6, 0, 24, 6, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 6, 0, 24, 6, 0, 24, 28, 0, 24, 56, 0,127,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_069[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 32, 24, 32, 31,224, 24, 32, 24, 32, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_070[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 16, 24, 16, 31,240, 24, 16, 24, 16, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_071[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 15, 28, 0, 28, 14, 0, 48, 6, 0, 48, 6, 0, 96, 6, 0, 96, 6, 0, 96, 31,128, 96, 0, 0, 96, 0, 0, 96, 0, 0, 96, 0, 0, 48, 2, 0, 48, 2, 0, 28, 6, 0, 14, 30, 0, 3,242, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_072[] = { 19, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 15,192, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 31,255, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0,126, 15,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_073[] = { 8, 0, 0, 0, 0, 0, 0, 0,126, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,126, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_074[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0,102, 0, 99, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 15,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_075[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 15,128, 24, 7, 0, 24, 14, 0, 24, 28, 0, 24, 56, 0, 24,112, 0, 24,224, 0, 25,192, 0, 31,128, 0, 31, 0, 0, 25,128, 0, 24,192, 0, 24, 96, 0, 24, 48, 0, 24, 24, 0, 24, 12, 0,126, 63, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_076[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0,126, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_077[] = { 22, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 16,252, 16, 48, 48, 16, 48, 48, 16,104, 48, 16,104, 48, 16,196, 48, 16,196, 48, 17,132, 48, 17,130, 48, 19, 2, 48, 19, 1, 48, 22, 1, 48, 22, 1, 48, 28, 0,176, 28, 0,176, 24, 0,112,120, 0,124, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_078[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 6, 0, 16, 14, 0, 16, 14, 0, 16, 26, 0, 16, 50, 0, 16, 50, 0, 16, 98, 0, 16,194, 0, 16,194, 0, 17,130, 0, 19, 2, 0, 19, 2, 0, 22, 2, 0, 28, 2, 0, 28, 2, 0, 24, 2, 0,120, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_079[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_080[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 31,224, 24, 56, 24, 24, 24, 12, 24, 12, 24, 12, 24, 24, 24, 56,127,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_081[] = { 18, 0, 0, 0, 0, 0, 0, 0, 7,128, 0, 28, 0, 0, 56, 0, 0,112, 0, 0,224, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_082[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 15, 24, 14, 24, 28, 24, 56, 24, 48, 24, 96, 24,224, 25,192, 31,224, 24, 56, 24, 24, 24, 28, 24, 12, 24, 28, 24, 24, 24, 56,127,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_083[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 79, 0,120,192, 96, 96, 64, 48, 64, 48, 0, 48, 0,112, 1,224, 7,192, 15, 0, 60, 0,112, 0, 96, 32, 96, 32, 96, 96, 49,224, 15, 32, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_084[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,224, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 65,130, 65,130, 97,134,127,254, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_085[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 24, 0, 12, 4, 0, 24, 4, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0,126, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_086[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,128, 0, 1,128, 0, 1,128, 0, 3,192, 0, 3, 64, 0, 3, 96, 0, 6, 32, 0, 6, 32, 0, 6, 48, 0, 12, 16, 0, 12, 24, 0, 24, 8, 0, 24, 8, 0, 24, 12, 0, 48, 4, 0, 48, 6, 0,252, 31,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_087[] = { 23, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,131, 0, 1,131, 0, 1,131,128, 3,135,128, 3, 70,128, 3, 70,192, 6, 70, 64, 6, 76, 64, 6, 76, 96, 12, 44, 96, 12, 44, 32, 24, 44, 32, 24, 24, 48, 24, 24, 16, 48, 24, 16, 48, 24, 24,252,126,126, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_088[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 15,192, 48, 3,128, 24, 7, 0, 8, 14, 0, 4, 12, 0, 6, 24, 0, 2, 56, 0, 1,112, 0, 0,224, 0, 0,192, 0, 1,192, 0, 3,160, 0, 3, 16, 0, 6, 8, 0, 14, 12, 0, 28, 6, 0,126, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_089[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,224, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 3,192, 3, 64, 6, 96, 6, 32, 12, 48, 28, 16, 24, 24, 56, 8, 48, 12,252, 63, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_090[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252,112, 12, 56, 4, 24, 4, 28, 0, 12, 0, 14, 0, 7, 0, 3, 0, 3,128, 1,128, 1,192, 0,224, 64, 96, 64,112, 96, 56,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_091[] = { 8, 0, 0, 0, 62, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 62, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_092[] = { 7, 0, 0, 0, 0, 0, 0, 0, 6, 6, 4, 12, 12, 8, 24, 24, 16, 48, 48, 32, 96, 96, 64,192,192, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_093[] = { 8, 0, 0, 0,124, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12,124, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_094[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 64, 64, 96,192, 32,128, 49,128, 17, 0, 27, 0, 10, 0, 14, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_095[] = { 13, 0, 0, 0, 0,255,248,255,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_096[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48,112, 64, 96, 48, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_097[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_098[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 47, 0, 57,192, 48,192, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48,192, 57,192, 55, 0, 48, 0, 48, 0, 48, 0, 48, 0,112, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_099[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 32,192, 49,192, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_100[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 96, 57,192, 48,192, 96,192, 96,192, 96,192, 96,192, 96,192, 96,192, 48,192, 57,192, 14,192, 0,192, 0,192, 0,192, 0,192, 1,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_101[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0,127,192, 96,192, 32,192, 49,128, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_102[] = { 7, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,254, 48, 48, 48, 22, 14, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_103[] = { 12, 0, 0, 0, 0, 31,128,120,224, 96, 48, 96, 16, 48, 48, 31,224, 63,128, 48, 0, 24, 0, 31, 0, 25,128, 48,192, 48,192, 48,192, 48,192, 25,128, 15,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_104[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120,240, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 56,224, 55,192, 51,128, 48, 0, 48, 0, 48, 0, 48, 0,112, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_105[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_106[] = { 6, 0, 0,192,224, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_107[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,121,240, 48,224, 49,192, 51,128, 55, 0, 54, 0, 60, 0, 52, 0, 50, 0, 51, 0, 49,128, 51,224, 48, 0, 48, 0, 48, 0, 48, 0,112, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_108[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_109[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120,241,224, 48, 96,192, 48, 96,192, 48, 96,192, 48, 96,192, 48, 96,192, 48, 96,192, 48, 96,192, 48, 96,192, 56,241,192, 55,207,128,115,135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_110[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120,240, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 56,224, 55,192,115,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_111[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_112[] = { 12, 0, 0, 0, 0,120, 0, 48, 0, 48, 0, 48, 0, 48, 0, 55, 0, 57,192, 48,192, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48,192, 57,192,119, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_113[] = { 12, 0, 0, 0, 0, 1,224, 0,192, 0,192, 0,192, 0,192, 14,192, 57,192, 48,192, 96,192, 96,192, 96,192, 96,192, 96,192, 96,192, 48,192, 57,192, 14,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_114[] = { 8, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 59, 55,115, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_115[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 99, 0, 65,128, 1,128, 3,128, 15, 0, 62, 0, 56, 0,112, 0, 97, 0, 51, 0, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_116[] = { 7, 0, 0, 0, 0, 0, 0, 0, 28, 50, 48, 48, 48, 48, 48, 48, 48, 48, 48,254,112, 48, 16, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_117[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14,112, 31, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_118[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4, 0, 14, 0, 14, 0, 26, 0, 25, 0, 25, 0, 49, 0, 48,128, 48,128, 96,128, 96,192,241,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_119[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4, 16, 0, 14, 56, 0, 14, 56, 0, 26, 40, 0, 26,100, 0, 25,100, 0, 49,100, 0, 48,194, 0, 48,194, 0, 96,194, 0, 96,195, 0,241,231,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_120[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120,240, 48, 96, 16,192, 25,192, 13,128, 7, 0, 6, 0, 13, 0, 28,128, 24,192, 48, 96,120,240, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_121[] = { 11, 0, 0, 0, 0,224, 0,240, 0, 24, 0, 8, 0, 12, 0, 4, 0, 14, 0, 14, 0, 26, 0, 25, 0, 25, 0, 49, 0, 48,128, 48,128, 96,128, 96,192,241,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_122[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128, 97,128, 48,128, 56, 0, 24, 0, 28, 0, 12, 0, 14, 0, 7, 0, 67, 0, 97,128,127,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_123[] = { 10, 0, 0, 0, 0, 3,128, 6, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 8, 0, 24, 0, 16, 0, 96, 0, 16, 0, 24, 0, 8, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 6, 0, 3,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_124[] = { 6, 0, 0, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_125[] = { 10, 0, 0, 0, 0,112, 0, 24, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 4, 0, 6, 0, 2, 0, 1,128, 2, 0, 6, 0, 4, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 24, 0,112, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_126[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 65,192, 99,224, 62, 48, 28, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_127[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_128[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_129[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_130[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_131[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_132[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_133[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_134[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_135[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_136[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_137[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_138[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_139[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_140[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_141[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_142[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_143[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_144[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_145[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_146[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_147[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_148[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_149[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_150[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_151[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_152[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_153[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_154[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_155[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_156[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_157[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_158[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_159[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_160[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_161[] = { 8, 0, 0, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 0, 0, 0, 12, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_162[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 32, 0, 31, 0, 63,128, 56, 64,104, 0,100, 0,100, 0,100, 0, 98, 0, 98, 0, 33,192, 49,192, 15,128, 0,128, 0,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_163[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,115,192, 95, 96, 60, 32, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0,126, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24,192, 12,192, 7,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_164[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 48,119,112, 63,224, 24,192, 48, 96, 48, 96, 48, 96, 48, 96, 24,192, 63,224,119,112, 96, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_165[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,192, 3, 0, 3, 0, 3, 0, 3, 0, 31,224, 3, 0, 31,224, 3, 0, 7,128, 12,128, 12,192, 24, 64, 24, 96, 48, 32,112, 48,248,124, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_166[] = { 6, 0, 0, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_167[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14, 0, 25, 0, 24,128, 1,128, 3,128, 7, 0, 14, 0, 29, 0, 56,128, 48,192, 32,192, 33,192, 19,128, 15, 0, 14, 0, 28, 0, 24, 0, 17,128, 9,128, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_168[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,102,102, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_169[] = { 19, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,248, 0, 14, 14, 0, 24, 3, 0, 48,225,128, 35,184,128, 98, 12,192, 70, 0, 64, 68, 0, 64, 68, 0, 64, 68, 0, 64, 70, 0, 64, 98, 12,192, 35,152,128, 48,241,128, 24, 3, 0, 14, 14, 0, 3,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_170[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0,118,204,204,124, 12,204,120, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_171[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 32, 6, 96, 12,192, 25,128, 51, 0, 51, 0, 25,128, 12,192, 6, 96, 2, 32, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_172[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24,127,248,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_173[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_174[] = { 19, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,248, 0, 14, 14, 0, 24, 3, 0, 48, 1,128, 35,140,128, 97, 24,192, 65, 16, 64, 65, 32, 64, 65,240, 64, 65, 24, 64, 65, 8, 64, 97, 8,192, 33, 24,128, 51,241,128, 24, 3, 0, 14, 14, 0, 3,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_175[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126,126, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_176[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 65, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_177[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,248,127,248, 0, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0,127,248,127,248, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_178[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 68, 32, 48, 16, 8, 12,140, 76, 56, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_179[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,112,136,140, 12, 8, 48, 8,140, 76, 56, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_180[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32, 24, 14, 6, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_181[] = { 13, 0, 0, 0, 0, 32, 0,112, 0, 96, 0, 32, 0, 32, 0, 46,112, 63, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_182[] = { 11, 0, 0, 0, 0, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 12,128, 28,128, 60,128, 60,128,124,128,124,128,124,128, 60,128, 60,128, 28,128, 15,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_183[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_184[] = { 8, 0, 60,102, 6, 30, 24, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_185[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 16, 16, 16, 16, 16, 16, 80, 48, 16, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_186[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 60,102,102,102,102,102, 60, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_187[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 51, 0, 25,128, 12,192, 6, 96, 6, 96, 12,192, 25,128, 51, 0, 34, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_188[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 2, 0, 8, 2, 0, 12,127,128, 4, 34, 0, 6, 50, 0, 3, 18, 0, 1, 10, 0,125,142, 0, 16,134, 0, 16,194, 0, 16, 96, 0, 16, 32, 0, 16, 48, 0, 16, 16, 0, 80, 24, 0, 48, 12, 0, 16, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_189[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 63, 0, 8, 17, 0, 12, 8, 0, 4, 12, 0, 6, 4, 0, 3, 2, 0, 1, 3, 0,125,163, 0, 16,147, 0, 16,206, 0, 16, 96, 0, 16, 32, 0, 16, 48, 0, 16, 16, 0, 80, 24, 0, 48, 12, 0, 16, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_190[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 2, 0, 8, 2, 0, 12,127,128, 4, 34, 0, 6, 50, 0, 3, 18, 0, 1, 10, 0,113,142, 0,136,134, 0,140,194, 0, 12, 96, 0, 8, 32, 0, 48, 48, 0, 8, 16, 0,140, 24, 0, 76, 12, 0, 56, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_191[] = { 11, 0, 0, 0, 0, 31, 0, 49,128, 96,128, 97,128, 97,128,112, 0, 56, 0, 24, 0, 28, 0, 12, 0, 12, 0, 4, 0, 4, 0, 0, 0, 0, 0, 6, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_192[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 0, 32, 0, 0,192, 0, 3,128, 0, 3, 0, 0}; +static const GLubyte TimesRoman24_Character_193[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 1, 0, 0, 0,192, 0, 0,112, 0, 0, 48, 0}; +static const GLubyte TimesRoman24_Character_194[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 8, 16, 0, 6, 96, 0, 3,192, 0, 1,128, 0}; +static const GLubyte TimesRoman24_Character_195[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 7, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 0, 0, 0, 4,224, 0, 3,144, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_196[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 0, 0, 0, 6, 48, 0, 6, 48, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_197[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 1,192, 0, 2, 32, 0, 2, 32, 0, 1,192, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_198[] = { 21, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,249,255,240, 48, 96, 48, 16, 96, 16, 16, 96, 16, 24, 96, 0, 8, 96, 0, 15,224,128, 12, 96,128, 4,127,128, 4, 96,128, 6, 96,128, 2, 96, 0, 2, 96, 0, 1, 96, 32, 1, 96, 32, 1,224, 96, 3,255,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_199[] = { 16, 0, 0, 3,192, 6, 96, 0, 96, 1,224, 1,128, 0,128, 3,240, 15, 28, 28, 4, 48, 2, 48, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 48, 2, 48, 2, 28, 6, 14, 30, 3,242, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_200[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 32, 24, 32, 31,224, 24, 32, 24, 32, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 0,128, 3, 0, 14, 0, 12, 0}; +static const GLubyte TimesRoman24_Character_201[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 32, 24, 32, 31,224, 24, 32, 24, 32, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 2, 0, 1,128, 0,224, 0, 96}; +static const GLubyte TimesRoman24_Character_202[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 32, 24, 32, 31,224, 24, 32, 24, 32, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 8, 16, 6, 96, 3,192, 1,128}; +static const GLubyte TimesRoman24_Character_203[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 32, 24, 32, 31,224, 24, 32, 24, 32, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 0, 0, 12,192, 12,192, 0, 0}; +static const GLubyte TimesRoman24_Character_204[] = { 8, 0, 0, 0, 0, 0, 0, 0,126, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,126, 0, 4, 24,112, 96}; +static const GLubyte TimesRoman24_Character_205[] = { 8, 0, 0, 0, 0, 0, 0, 0,126, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,126, 0, 32, 24, 14, 6}; +static const GLubyte TimesRoman24_Character_206[] = { 8, 0, 0, 0, 0, 0, 0, 0, 63, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 63, 0, 64, 51, 30, 12}; +static const GLubyte TimesRoman24_Character_207[] = { 8, 0, 0, 0, 0, 0, 0, 0,126, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,126, 0, 0,102,102, 0}; +static const GLubyte TimesRoman24_Character_208[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,224, 0, 24, 56, 0, 24, 28, 0, 24, 6, 0, 24, 6, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0,255, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 6, 0, 24, 6, 0, 24, 28, 0, 24, 56, 0,127,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_209[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 6, 0, 16, 14, 0, 16, 14, 0, 16, 26, 0, 16, 50, 0, 16, 50, 0, 16, 98, 0, 16,194, 0, 16,194, 0, 17,130, 0, 19, 2, 0, 19, 2, 0, 22, 2, 0, 28, 2, 0, 28, 2, 0, 24, 2, 0,120, 15,128, 0, 0, 0, 0, 0, 0, 2,112, 0, 1,200, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_210[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0, 32, 0, 0,192, 0, 3,128, 0, 3, 0, 0}; +static const GLubyte TimesRoman24_Character_211[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0,128, 0, 0, 96, 0, 0, 56, 0, 0, 24, 0}; +static const GLubyte TimesRoman24_Character_212[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 4, 8, 0, 3, 48, 0, 1,224, 0, 0,192, 0}; +static const GLubyte TimesRoman24_Character_213[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0, 0, 0, 2,112, 0, 1,200, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_214[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0, 0, 0, 3, 48, 0, 3, 48, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_215[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32, 16, 48, 48, 24, 96, 12,192, 7,128, 3, 0, 7,128, 12,192, 24, 96, 48, 48, 32, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_216[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16, 0, 0, 19,240, 0, 14, 28, 0, 28, 14, 0, 52, 3, 0, 50, 3, 0, 97, 1,128, 97, 1,128, 96,129,128, 96,129,128, 96, 65,128, 96, 65,128, 96, 33,128, 48, 35, 0, 48, 19, 0, 28, 14, 0, 14, 28, 0, 3,242, 0, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_217[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 24, 0, 12, 4, 0, 24, 4, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0,126, 15,128, 0, 0, 0, 0, 32, 0, 0,192, 0, 3,128, 0, 3, 0, 0}; +static const GLubyte TimesRoman24_Character_218[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 24, 0, 12, 4, 0, 24, 4, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0,126, 15,128, 0, 0, 0, 0,128, 0, 0, 96, 0, 0, 56, 0, 0, 24, 0}; +static const GLubyte TimesRoman24_Character_219[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 24, 0, 12, 4, 0, 24, 4, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0,126, 15,128, 0, 0, 0, 4, 8, 0, 3, 48, 0, 1,224, 0, 0,192, 0}; +static const GLubyte TimesRoman24_Character_220[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 24, 0, 12, 4, 0, 24, 4, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0,126, 15,128, 0, 0, 0, 0, 0, 0, 3, 24, 0, 3, 24, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_221[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,224, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 3,192, 3, 64, 6, 96, 6, 32, 12, 48, 28, 16, 24, 24, 56, 8, 48, 12,252, 63, 0, 0, 1, 0, 0,192, 0,112, 0, 48}; +static const GLubyte TimesRoman24_Character_222[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 24, 0, 24, 0, 24, 0, 31,224, 24, 56, 24, 24, 24, 12, 24, 12, 24, 12, 24, 24, 24, 56, 31,224, 24, 0, 24, 0, 24, 0,126, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_223[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,115,128, 54, 64, 54, 96, 48, 96, 48, 96, 48,224, 48,192, 49,192, 51,128, 54, 0, 49,128, 48,192, 48,192, 48,192, 48,192, 25,128, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_224[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 2, 0, 12, 0, 56, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_225[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 8, 0, 6, 0, 3,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_226[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 33, 0, 18, 0, 30, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_227[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 0, 0, 46, 0, 29, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_228[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 0, 0, 51, 0, 51, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_229[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 14, 0, 17, 0, 17, 0, 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_230[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,120,125,252, 99,194, 97,128, 97,128, 49,128, 29,128, 7,254, 1,134, 49,134, 51,204, 30,120, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_231[] = { 11, 0, 0, 30, 0, 51, 0, 3, 0, 15, 0, 12, 0, 4, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 32,192, 49,192, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_232[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0,127,192, 96,192, 32,192, 49,128, 15, 0, 0, 0, 2, 0, 12, 0, 56, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_233[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0,127,192, 96,192, 32,192, 49,128, 15, 0, 0, 0, 8, 0, 6, 0, 3,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_234[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0,127,192, 96,192, 32,192, 49,128, 15, 0, 0, 0, 16,128, 9, 0, 15, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_235[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0,127,192, 96,192, 32,192, 49,128, 15, 0, 0, 0, 0, 0, 25,128, 25,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_236[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 8, 48,224,192, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_237[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 64, 48, 28, 12, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_238[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0,132, 72,120, 48, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_239[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 0,204,204, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_240[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15,128, 99, 0, 30, 0, 15, 0, 56,192, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_241[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120,240, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 56,224, 55,192,115,128, 0, 0, 0, 0, 19,128, 14, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_242[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 1, 0, 6, 0, 28, 0, 24, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_243[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 4, 0, 3, 0, 1,192, 0,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_244[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 16,128, 9, 0, 15, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_245[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 0, 0, 19,128, 14, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_246[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 0, 0, 25,128, 25,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_247[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 3, 0, 0, 0, 0, 0,127,248,127,248, 0, 0, 0, 0, 3, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_248[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0,111, 0, 57,192, 56,192,104, 96,108, 96,100, 96,102, 96, 98, 96, 99, 96, 49,192, 57,192, 15, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_249[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14,112, 31, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 1, 0, 6, 0, 28, 0, 24, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_250[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14,112, 31, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 4, 0, 3, 0, 1,192, 0,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_251[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14,112, 31, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 16,128, 9, 0, 15, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_252[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14,112, 31, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 0, 0, 25,128, 25,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_253[] = { 11, 0, 0, 0, 0,224, 0,240, 0, 24, 0, 8, 0, 12, 0, 4, 0, 14, 0, 14, 0, 26, 0, 25, 0, 25, 0, 49, 0, 48,128, 48,128, 96,128, 96,192,241,224, 0, 0, 8, 0, 6, 0, 3,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_254[] = { 12, 0, 0, 0, 0,120, 0, 48, 0, 48, 0, 48, 0, 48, 0, 55, 0, 57,192, 48,192, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48,192, 57,192, 55, 0, 48, 0, 48, 0, 48, 0, 48, 0,112, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +static const GLubyte TimesRoman24_Character_255[] = { 11, 0, 0, 0, 0,224, 0,240, 0, 24, 0, 8, 0, 12, 0, 4, 0, 14, 0, 14, 0, 26, 0, 25, 0, 25, 0, 49, 0, 48,128, 48,128, 96,128, 96,192,241,224, 0, 0, 0, 0, 51, 0, 51, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; + +/* The font characters mapping: */ +static const GLubyte* TimesRoman24_Character_Map[] = {TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032, + TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032, + TimesRoman24_Character_032,TimesRoman24_Character_033,TimesRoman24_Character_034,TimesRoman24_Character_035,TimesRoman24_Character_036,TimesRoman24_Character_037,TimesRoman24_Character_038,TimesRoman24_Character_039,TimesRoman24_Character_040,TimesRoman24_Character_041,TimesRoman24_Character_042,TimesRoman24_Character_043,TimesRoman24_Character_044,TimesRoman24_Character_045,TimesRoman24_Character_046,TimesRoman24_Character_047, + TimesRoman24_Character_048,TimesRoman24_Character_049,TimesRoman24_Character_050,TimesRoman24_Character_051,TimesRoman24_Character_052,TimesRoman24_Character_053,TimesRoman24_Character_054,TimesRoman24_Character_055,TimesRoman24_Character_056,TimesRoman24_Character_057,TimesRoman24_Character_058,TimesRoman24_Character_059,TimesRoman24_Character_060,TimesRoman24_Character_061,TimesRoman24_Character_062,TimesRoman24_Character_063, + TimesRoman24_Character_064,TimesRoman24_Character_065,TimesRoman24_Character_066,TimesRoman24_Character_067,TimesRoman24_Character_068,TimesRoman24_Character_069,TimesRoman24_Character_070,TimesRoman24_Character_071,TimesRoman24_Character_072,TimesRoman24_Character_073,TimesRoman24_Character_074,TimesRoman24_Character_075,TimesRoman24_Character_076,TimesRoman24_Character_077,TimesRoman24_Character_078,TimesRoman24_Character_079, + TimesRoman24_Character_080,TimesRoman24_Character_081,TimesRoman24_Character_082,TimesRoman24_Character_083,TimesRoman24_Character_084,TimesRoman24_Character_085,TimesRoman24_Character_086,TimesRoman24_Character_087,TimesRoman24_Character_088,TimesRoman24_Character_089,TimesRoman24_Character_090,TimesRoman24_Character_091,TimesRoman24_Character_092,TimesRoman24_Character_093,TimesRoman24_Character_094,TimesRoman24_Character_095, + TimesRoman24_Character_096,TimesRoman24_Character_097,TimesRoman24_Character_098,TimesRoman24_Character_099,TimesRoman24_Character_100,TimesRoman24_Character_101,TimesRoman24_Character_102,TimesRoman24_Character_103,TimesRoman24_Character_104,TimesRoman24_Character_105,TimesRoman24_Character_106,TimesRoman24_Character_107,TimesRoman24_Character_108,TimesRoman24_Character_109,TimesRoman24_Character_110,TimesRoman24_Character_111, + TimesRoman24_Character_112,TimesRoman24_Character_113,TimesRoman24_Character_114,TimesRoman24_Character_115,TimesRoman24_Character_116,TimesRoman24_Character_117,TimesRoman24_Character_118,TimesRoman24_Character_119,TimesRoman24_Character_120,TimesRoman24_Character_121,TimesRoman24_Character_122,TimesRoman24_Character_123,TimesRoman24_Character_124,TimesRoman24_Character_125,TimesRoman24_Character_126,TimesRoman24_Character_032, + TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032, + TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032, + TimesRoman24_Character_160,TimesRoman24_Character_161,TimesRoman24_Character_162,TimesRoman24_Character_163,TimesRoman24_Character_164,TimesRoman24_Character_165,TimesRoman24_Character_166,TimesRoman24_Character_167,TimesRoman24_Character_168,TimesRoman24_Character_169,TimesRoman24_Character_170,TimesRoman24_Character_171,TimesRoman24_Character_172,TimesRoman24_Character_173,TimesRoman24_Character_174,TimesRoman24_Character_175, + TimesRoman24_Character_176,TimesRoman24_Character_177,TimesRoman24_Character_178,TimesRoman24_Character_179,TimesRoman24_Character_180,TimesRoman24_Character_181,TimesRoman24_Character_182,TimesRoman24_Character_183,TimesRoman24_Character_184,TimesRoman24_Character_185,TimesRoman24_Character_186,TimesRoman24_Character_187,TimesRoman24_Character_188,TimesRoman24_Character_189,TimesRoman24_Character_190,TimesRoman24_Character_191, + TimesRoman24_Character_192,TimesRoman24_Character_193,TimesRoman24_Character_194,TimesRoman24_Character_195,TimesRoman24_Character_196,TimesRoman24_Character_197,TimesRoman24_Character_198,TimesRoman24_Character_199,TimesRoman24_Character_200,TimesRoman24_Character_201,TimesRoman24_Character_202,TimesRoman24_Character_203,TimesRoman24_Character_204,TimesRoman24_Character_205,TimesRoman24_Character_206,TimesRoman24_Character_207, + TimesRoman24_Character_208,TimesRoman24_Character_209,TimesRoman24_Character_210,TimesRoman24_Character_211,TimesRoman24_Character_212,TimesRoman24_Character_213,TimesRoman24_Character_214,TimesRoman24_Character_215,TimesRoman24_Character_216,TimesRoman24_Character_217,TimesRoman24_Character_218,TimesRoman24_Character_219,TimesRoman24_Character_220,TimesRoman24_Character_221,TimesRoman24_Character_222,TimesRoman24_Character_223, + TimesRoman24_Character_224,TimesRoman24_Character_225,TimesRoman24_Character_226,TimesRoman24_Character_227,TimesRoman24_Character_228,TimesRoman24_Character_229,TimesRoman24_Character_230,TimesRoman24_Character_231,TimesRoman24_Character_232,TimesRoman24_Character_233,TimesRoman24_Character_234,TimesRoman24_Character_235,TimesRoman24_Character_236,TimesRoman24_Character_237,TimesRoman24_Character_238,TimesRoman24_Character_239, + TimesRoman24_Character_240,TimesRoman24_Character_241,TimesRoman24_Character_242,TimesRoman24_Character_243,TimesRoman24_Character_244,TimesRoman24_Character_245,TimesRoman24_Character_246,TimesRoman24_Character_247,TimesRoman24_Character_248,TimesRoman24_Character_249,TimesRoman24_Character_250,TimesRoman24_Character_251,TimesRoman24_Character_252,TimesRoman24_Character_253,TimesRoman24_Character_254,TimesRoman24_Character_255,NULL}; + +/* The font structure: */ +const SFG_Font fgFontTimesRoman24 = { "-adobe-times-medium-r-normal--24-240-75-75-p-124-iso8859-1", 256, 29, TimesRoman24_Character_Map, 0, 7 }; + diff --git a/examples/opengl-framework/freeglut/freeglut_gamemode.c b/examples/opengl-framework/freeglut/freeglut_gamemode.c new file mode 100644 index 00000000..803026b0 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_gamemode.c @@ -0,0 +1,819 @@ +/* + * freeglut_gamemode.c + * + * The game mode handling code. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 16 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* + * TODO BEFORE THE STABLE RELEASE: + * + * glutGameModeString() -- missing + * glutEnterGameMode() -- X11 version + * glutLeaveGameMode() -- is that correct? + * glutGameModeGet() -- is that correct? + */ + + +/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ + +#if TARGET_HOST_POSIX_X11 +static int xrandr_resize(int xsz, int ysz, int rate, int just_checking) +{ +#ifdef HAVE_X11_EXTENSIONS_XRANDR_H + int event_base, error_base, ver_major, ver_minor, use_rate; + XRRScreenConfiguration *xrr_config = 0; + Status result = -1; + + /* must check at runtime for the availability of the extension */ + if(!XRRQueryExtension(fgDisplay.Display, &event_base, &error_base)) { + return -1; + } + + XRRQueryVersion(fgDisplay.Display, &ver_major, &ver_minor); + + /* we only heed the rate if we CAN actually use it (Xrandr >= 1.1) and + * the user actually cares about it (rate > 0) + */ + use_rate = ( rate > 0 ) && ( ( ver_major >= 1 ) || + ( ( ver_major == 1 ) && ( ver_minor >= 1 ) ) ); + + /* this loop is only so that the whole thing will be repeated if someone + * else changes video mode between our query of the current information and + * the attempt to change it. + */ + do { + XRRScreenSize *ssizes; + short *rates; + Rotation rot; + int i, ssizes_count, rates_count, curr, res_idx = -1; + Time timestamp, cfg_timestamp; + + if(xrr_config) { + XRRFreeScreenConfigInfo(xrr_config); + } + + if(!(xrr_config = XRRGetScreenInfo(fgDisplay.Display, fgDisplay.RootWindow))) { + fgWarning("XRRGetScreenInfo failed"); + break; + } + ssizes = XRRConfigSizes(xrr_config, &ssizes_count); + curr = XRRConfigCurrentConfiguration(xrr_config, &rot); + timestamp = XRRConfigTimes(xrr_config, &cfg_timestamp); + + /* if either of xsz or ysz are unspecified, use the current values */ + if(xsz <= 0) + xsz = fgState.GameModeSize.X = ssizes[curr].width; + if(ysz <= 0) + ysz = fgState.GameModeSize.Y = ssizes[curr].height; + + + if(xsz == ssizes[curr].width && ysz == ssizes[curr].height) { + /* no need to switch, we're already in the requested resolution */ + res_idx = curr; + } else { + for(i=0; i= 1 ) || ( ( RANDR_MAJOR == 1 ) && ( RANDR_MINOR >= 1 ) ) + if(use_rate) { + rate = fgState.GameModeRefresh; + + /* for the selected resolution, let's find out if there is + * a matching refresh rate available. + */ + rates = XRRConfigRates(xrr_config, res_idx, &rates_count); + + for(i=0; i= 1 ) || ( ( RANDR_MAJOR == 1 ) && ( RANDR_MINOR >= 1 ) ) + if(use_rate) + result = XRRSetScreenConfigAndRate(fgDisplay.Display, xrr_config, + fgDisplay.RootWindow, res_idx, rot, rate, timestamp); + else +#endif + result = XRRSetScreenConfig(fgDisplay.Display, xrr_config, + fgDisplay.RootWindow, res_idx, rot, timestamp); + + } while(result == RRSetConfigInvalidTime); + + if(xrr_config) { + XRRFreeScreenConfigInfo(xrr_config); + } + + if(result == 0) { + return 0; + } + +#endif /* HAVE_X11_EXTENSIONS_XRANDR_H */ + return -1; +} +#endif /* TARGET_HOST_POSIX_X11 */ + +/* + * Remembers the current visual settings, so that + * we can change them and restore later... + */ +static void fghRememberState( void ) +{ +#if TARGET_HOST_POSIX_X11 + int event_base, error_base; + + /* + * Remember the current pointer location before going fullscreen + * for restoring it later: + */ + Window junk_window; + unsigned int junk_mask; + + XQueryPointer(fgDisplay.Display, fgDisplay.RootWindow, + &junk_window, &junk_window, + &fgDisplay.DisplayPointerX, &fgDisplay.DisplayPointerY, + &fgDisplay.DisplayPointerX, &fgDisplay.DisplayPointerY, &junk_mask); + +# ifdef HAVE_X11_EXTENSIONS_XRANDR_H + if(XRRQueryExtension(fgDisplay.Display, &event_base, &error_base)) { + XRRScreenConfiguration *xrr_config; + XRRScreenSize *ssizes; + Rotation rot; + int ssize_count, curr; + + if((xrr_config = XRRGetScreenInfo(fgDisplay.Display, fgDisplay.RootWindow))) { + ssizes = XRRConfigSizes(xrr_config, &ssize_count); + curr = XRRConfigCurrentConfiguration(xrr_config, &rot); + + fgDisplay.prev_xsz = ssizes[curr].width; + fgDisplay.prev_ysz = ssizes[curr].height; + fgDisplay.prev_refresh = -1; + +# if ( RANDR_MAJOR >= 1 ) || ( ( RANDR_MAJOR == 1 ) && ( RANDR_MINOR >= 1 ) ) + if(fgState.GameModeRefresh != -1) { + fgDisplay.prev_refresh = XRRConfigCurrentRate(xrr_config); + } +# endif + + fgDisplay.prev_size_valid = 1; + + XRRFreeScreenConfigInfo(xrr_config); + } + } +# endif + + /* + * This highly depends on the XFree86 extensions, + * not approved as X Consortium standards + */ +# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H + if(!XF86VidModeQueryExtension(fgDisplay.Display, &event_base, &error_base)) { + return; + } + + /* + * Remember the current ViewPort location of the screen to be able to + * restore the ViewPort on LeaveGameMode(): + */ + if( !XF86VidModeGetViewPort( + fgDisplay.Display, + fgDisplay.Screen, + &fgDisplay.DisplayViewPortX, + &fgDisplay.DisplayViewPortY ) ) + fgWarning( "XF86VidModeGetViewPort failed" ); + + + /* Query the current display settings: */ + fgDisplay.DisplayModeValid = + XF86VidModeGetModeLine( + fgDisplay.Display, + fgDisplay.Screen, + &fgDisplay.DisplayModeClock, + &fgDisplay.DisplayMode + ); + + if( !fgDisplay.DisplayModeValid ) + fgWarning( "XF86VidModeGetModeLine failed" ); +# endif + +#elif TARGET_HOST_MS_WINDOWS + +/* DEVMODE devMode; */ + + /* Grab the current desktop settings... */ + +/* hack to get around my stupid cross-gcc headers */ +#define FREEGLUT_ENUM_CURRENT_SETTINGS -1 + + EnumDisplaySettings( fgDisplay.DisplayName, FREEGLUT_ENUM_CURRENT_SETTINGS, + &fgDisplay.DisplayMode ); + + /* Make sure we will be restoring all settings needed */ + fgDisplay.DisplayMode.dmFields |= + DM_PELSWIDTH | DM_PELSHEIGHT | DM_BITSPERPEL | DM_DISPLAYFREQUENCY; + +#endif +} + +/* + * Restores the previously remembered visual settings + */ +static void fghRestoreState( void ) +{ +#if TARGET_HOST_POSIX_X11 + /* Restore the remembered pointer position: */ + XWarpPointer( + fgDisplay.Display, None, fgDisplay.RootWindow, 0, 0, 0, 0, + fgDisplay.DisplayPointerX, fgDisplay.DisplayPointerY + ); + + +# ifdef HAVE_X11_EXTENSIONS_XRANDR_H + if(fgDisplay.prev_size_valid) { + if(xrandr_resize(fgDisplay.prev_xsz, fgDisplay.prev_ysz, fgDisplay.prev_refresh, 0) != -1) { + fgDisplay.prev_size_valid = 0; +# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H + fgDisplay.DisplayModeValid = 0; +# endif + return; + } + } +# endif + + + +# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H + /* + * This highly depends on the XFree86 extensions, + * not approved as X Consortium standards + */ + + if( fgDisplay.DisplayModeValid ) + { + XF86VidModeModeInfo** displayModes; + int i, displayModesCount; + + if( !XF86VidModeGetAllModeLines( + fgDisplay.Display, + fgDisplay.Screen, + &displayModesCount, + &displayModes ) ) + { + fgWarning( "XF86VidModeGetAllModeLines failed" ); + return; + } + + + /* + * Check every of the modes looking for one that matches our demands. + * If we find one, switch to it and restore the remembered viewport. + */ + for( i = 0; i < displayModesCount; i++ ) + { + if(displayModes[ i ]->hdisplay == fgDisplay.DisplayMode.hdisplay && + displayModes[ i ]->vdisplay == fgDisplay.DisplayMode.vdisplay && + displayModes[ i ]->dotclock == fgDisplay.DisplayModeClock ) + { + if( !XF86VidModeSwitchToMode( + fgDisplay.Display, + fgDisplay.Screen, + displayModes[ i ] ) ) + { + fgWarning( "XF86VidModeSwitchToMode failed" ); + break; + } + + if( !XF86VidModeSetViewPort( + fgDisplay.Display, + fgDisplay.Screen, + fgDisplay.DisplayViewPortX, + fgDisplay.DisplayViewPortY ) ) + fgWarning( "XF86VidModeSetViewPort failed" ); + + + /* + * For the case this would be the last X11 call the application + * calls exit() we've to flush the X11 output queue to have the + * commands sent to the X server before the application exits. + */ + XFlush( fgDisplay.Display ); + + fgDisplay.DisplayModeValid = 0; +# ifdef HAVE_X11_EXTENSIONS_XRANDR_H + fgDisplay.prev_size_valid = 0; +# endif + + break; + } + } + XFree( displayModes ); + } + +# endif + +#elif TARGET_HOST_MS_WINDOWS + + /* Restore the previously remembered desktop display settings */ + ChangeDisplaySettingsEx( fgDisplay.DisplayName,&fgDisplay.DisplayMode, 0,0,0 ); + +#endif +} + +#if TARGET_HOST_POSIX_X11 +#ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H + +/* + * Checks a single display mode settings against user's preferences. + */ +static GLboolean fghCheckDisplayMode( int width, int height, int depth, int refresh ) +{ + /* The desired values should be stored in fgState structure... */ + return ( width == fgState.GameModeSize.X ) && + ( height == fgState.GameModeSize.Y ) && + ( depth == fgState.GameModeDepth ) && + ( refresh == fgState.GameModeRefresh ); +} + +/* + * Checks all display modes settings against user's preferences. + * Returns the mode number found or -1 if none could be found. + */ +static int fghCheckDisplayModes( GLboolean exactMatch, int displayModesCount, XF86VidModeModeInfo** displayModes ) +{ + int i; + for( i = 0; i < displayModesCount; i++ ) + { + /* Compute the displays refresh rate, dotclock comes in kHz. */ + int refresh = ( displayModes[ i ]->dotclock * 1000 ) / + ( displayModes[ i ]->htotal * displayModes[ i ]->vtotal ); + + if( fghCheckDisplayMode( displayModes[ i ]->hdisplay, + displayModes[ i ]->vdisplay, + fgState.GameModeDepth, + ( exactMatch ? refresh : fgState.GameModeRefresh ) ) ) { + if (!exactMatch) + { + /* Update the chosen refresh rate, otherwise a + * glutGameModeGet(GLUT_GAME_MODE_REFRESH_RATE) would not + * return the right values + */ + fgState.GameModeRefresh = refresh; + } + + return i; + } + } + return -1; +} + +#endif +#endif + +/* + * Changes the current display mode to match user's settings + */ +static GLboolean fghChangeDisplayMode( GLboolean haveToTest ) +{ + GLboolean success = GL_FALSE; +#if TARGET_HOST_POSIX_X11 + + /* first try to use XRandR, then fallback to XF86VidMode */ +# ifdef HAVE_X11_EXTENSIONS_XRANDR_H + if(xrandr_resize(fgState.GameModeSize.X, fgState.GameModeSize.Y, + fgState.GameModeRefresh, haveToTest) != -1) { + return GL_TRUE; + } +# endif + + + /* + * This highly depends on the XFree86 extensions, + * not approved as X Consortium standards + */ +# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H + + /* + * This is also used by applications which check modes by calling + * glutGameModeGet(GLUT_GAME_MODE_POSSIBLE), so allow the check: + */ + if( haveToTest || fgDisplay.DisplayModeValid ) + { + XF86VidModeModeInfo** displayModes; + int i, displayModesCount; + + /* If we don't have a valid modeline in the display structure, which + * can happen if this is called from glutGameModeGet instead of + * glutEnterGameMode, then we need to query the current mode, to make + * unspecified settings to default to their current values. + */ + if(!fgDisplay.DisplayModeValid) { + if(!XF86VidModeGetModeLine(fgDisplay.Display, fgDisplay.Screen, + &fgDisplay.DisplayModeClock, &fgDisplay.DisplayMode)) { + return success; + } + } + + if (fgState.GameModeSize.X == -1) + { + fgState.GameModeSize.X = fgDisplay.DisplayMode.hdisplay; + } + if (fgState.GameModeSize.Y == -1) + { + fgState.GameModeSize.Y = fgDisplay.DisplayMode.vdisplay; + } + if (fgState.GameModeDepth == -1) + { + /* can't get color depth from this, nor can we change it, do nothing + * TODO: get with XGetVisualInfo()? but then how to set? + */ + } + if (fgState.GameModeRefresh == -1) + { + /* Compute the displays refresh rate, dotclock comes in kHz. */ + int refresh = ( fgDisplay.DisplayModeClock * 1000 ) / + ( fgDisplay.DisplayMode.htotal * fgDisplay.DisplayMode.vtotal ); + + fgState.GameModeRefresh = refresh; + } + + /* query all possible display modes */ + if( !XF86VidModeGetAllModeLines( + fgDisplay.Display, + fgDisplay.Screen, + &displayModesCount, + &displayModes ) ) + { + fgWarning( "XF86VidModeGetAllModeLines failed" ); + return success; + } + + + /* + * Check every of the modes looking for one that matches our demands, + * ignoring the refresh rate if no exact match could be found. + */ + i = fghCheckDisplayModes( GL_TRUE, displayModesCount, displayModes ); + if( i < 0 ) { + i = fghCheckDisplayModes( GL_FALSE, displayModesCount, displayModes ); + } + success = ( i < 0 ) ? GL_FALSE : GL_TRUE; + + if( !haveToTest && success ) { + if( !XF86VidModeSwitchToMode( + fgDisplay.Display, + fgDisplay.Screen, + displayModes[ i ] ) ) + fgWarning( "XF86VidModeSwitchToMode failed" ); + } + + XFree( displayModes ); + } + +# endif + + +#elif TARGET_HOST_MS_WINDOWS + + DEVMODE devMode; + char *fggmstr = NULL; + char displayMode[300]; + + success = GL_FALSE; + + EnumDisplaySettings( fgDisplay.DisplayName, -1, &devMode ); + devMode.dmFields = 0; + + if (fgState.GameModeSize.X!=-1) + { + devMode.dmPelsWidth = fgState.GameModeSize.X; + devMode.dmFields |= DM_PELSWIDTH; + } + if (fgState.GameModeSize.Y!=-1) + { + devMode.dmPelsHeight = fgState.GameModeSize.Y; + devMode.dmFields |= DM_PELSHEIGHT; + } + if (fgState.GameModeDepth!=-1) + { + devMode.dmBitsPerPel = fgState.GameModeDepth; + devMode.dmFields |= DM_BITSPERPEL; + } + if (fgState.GameModeRefresh!=-1) + { + devMode.dmDisplayFrequency = fgState.GameModeRefresh; + devMode.dmFields |= DM_DISPLAYFREQUENCY; + } + + switch ( ChangeDisplaySettingsEx(fgDisplay.DisplayName, &devMode, NULL, haveToTest ? CDS_TEST : CDS_FULLSCREEN , NULL) ) + { + case DISP_CHANGE_SUCCESSFUL: + success = GL_TRUE; + + if (!haveToTest) + { + /* update vars in case if windows switched to proper mode */ + EnumDisplaySettings( fgDisplay.DisplayName, FREEGLUT_ENUM_CURRENT_SETTINGS, &devMode ); + fgState.GameModeSize.X = devMode.dmPelsWidth; + fgState.GameModeSize.Y = devMode.dmPelsHeight; + fgState.GameModeDepth = devMode.dmBitsPerPel; + fgState.GameModeRefresh = devMode.dmDisplayFrequency; + } + break; + case DISP_CHANGE_RESTART: + fggmstr = "The computer must be restarted for the graphics mode to work."; + break; + case DISP_CHANGE_BADFLAGS: + fggmstr = "An invalid set of flags was passed in."; + break; + case DISP_CHANGE_BADPARAM: + fggmstr = "An invalid parameter was passed in. This can include an invalid flag or combination of flags."; + break; + case DISP_CHANGE_FAILED: + fggmstr = "The display driver failed the specified graphics mode."; + break; + case DISP_CHANGE_BADMODE: + fggmstr = "The graphics mode is not supported."; + break; + default: + fggmstr = "Unknown error in graphics mode???"; /* dunno if it is possible,MSDN does not mention any other error */ + break; + } + + if ( !success ) + { + /* I'd rather get info whats going on in my program than wonder about */ + /* magic happenings behind my back, its lib for devels at last ;) */ + + /* append display mode to error to make things more informative */ + sprintf(displayMode,"%s Problem with requested mode: %ix%i:%i@%i", fggmstr, devMode.dmPelsWidth, devMode.dmPelsHeight, devMode.dmBitsPerPel, devMode.dmDisplayFrequency); + fgWarning(displayMode); + } +#endif + + return success; +} + + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * Sets the game mode display string + */ +void FGAPIENTRY glutGameModeString( const char* string ) +{ + int width = -1, height = -1, depth = -1, refresh = -1; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGameModeString" ); + + /* + * This one seems a bit easier than glutInitDisplayString. The bad thing + * about it that I was unable to find the game mode string definition, so + * that I assumed it is: "[width]x[height]:[depth]@[refresh rate]", which + * appears in all GLUT game mode programs I have seen to date. + */ + if( sscanf( string, "%ix%i:%i@%i", &width, &height, &depth, &refresh ) != + 4 ) + if( sscanf( string, "%ix%i:%i", &width, &height, &depth ) != 3 ) + if( sscanf( string, "%ix%i@%i", &width, &height, &refresh ) != 3 ) + if( sscanf( string, "%ix%i", &width, &height ) != 2 ) + if( sscanf( string, ":%i@%i", &depth, &refresh ) != 2 ) + if( sscanf( string, ":%i", &depth ) != 1 ) + if( sscanf( string, "@%i", &refresh ) != 1 ) + fgWarning( + "unable to parse game mode string `%s'", + string + ); + + /* All values not specified are now set to -1, which means those + * aspects of the current display mode are not changed in + * fghChangeDisplayMode() above. + */ + fgState.GameModeSize.X = width; + fgState.GameModeSize.Y = height; + fgState.GameModeDepth = depth; + fgState.GameModeRefresh = refresh; +} + + + +/* + * Enters the game mode + */ +int FGAPIENTRY glutEnterGameMode( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutEnterGameMode" ); + + if( fgStructure.GameModeWindow ) + fgAddToWindowDestroyList( fgStructure.GameModeWindow ); + else + fghRememberState( ); + + if( ! fghChangeDisplayMode( GL_FALSE ) ) + { + fgWarning( "failed to change screen settings" ); + return 0; + } + + fgStructure.GameModeWindow = fgCreateWindow( + NULL, "FREEGLUT", GL_TRUE, 0, 0, + GL_TRUE, fgState.GameModeSize.X, fgState.GameModeSize.Y, + GL_TRUE, GL_FALSE + ); + + fgStructure.GameModeWindow->State.Width = fgState.GameModeSize.X; + fgStructure.GameModeWindow->State.Height = fgState.GameModeSize.Y; + fgStructure.GameModeWindow->State.NeedToResize = GL_TRUE; + +#if TARGET_HOST_POSIX_X11 + + /* + * Sync needed to avoid a real race, the Xserver must have really created + * the window before we can grab the pointer into it: + */ + XSync( fgDisplay.Display, False ); + /* + * Grab the pointer to confine it into the window after the calls to + * XWrapPointer() which ensure that the pointer really enters the window. + * + * We also need to wait here until XGrabPointer() returns GrabSuccess, + * otherwise the new window is not viewable yet and if the next function + * (XSetInputFocus) is called with a not yet viewable window, it will exit + * the application which we have to aviod, so wait until it's viewable: + */ + while( GrabSuccess != XGrabPointer( + fgDisplay.Display, fgStructure.GameModeWindow->Window.Handle, + TRUE, + ButtonPressMask | ButtonReleaseMask | ButtonMotionMask + | PointerMotionMask, + GrabModeAsync, GrabModeAsync, + fgStructure.GameModeWindow->Window.Handle, None, CurrentTime) ) + usleep( 100 ); + /* + * Change input focus to the new window. This will exit the application + * if the new window is not viewable yet, see the XGrabPointer loop above. + */ + XSetInputFocus( + fgDisplay.Display, + fgStructure.GameModeWindow->Window.Handle, + RevertToNone, + CurrentTime + ); + + /* Move the Pointer to the middle of the fullscreen window */ + XWarpPointer( + fgDisplay.Display, + None, + fgDisplay.RootWindow, + 0, 0, 0, 0, + fgState.GameModeSize.X/2, fgState.GameModeSize.Y/2 + ); + +# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H + + if( fgDisplay.DisplayModeValid ) + { + int x, y; + Window child; + + /* Change to viewport to the window topleft edge: */ + if( !XF86VidModeSetViewPort( fgDisplay.Display, fgDisplay.Screen, 0, 0 ) ) + fgWarning( "XF86VidModeSetViewPort failed" ); + + /* + * Final window repositioning: It could be avoided using an undecorated + * window using override_redirect, but this * would possily require + * more changes and investigation. + */ + + /* Get the current postion of the drawable area on screen */ + XTranslateCoordinates( + fgDisplay.Display, + fgStructure.CurrentWindow->Window.Handle, + fgDisplay.RootWindow, + 0, 0, &x, &y, + &child + ); + + /* Move the decorataions out of the topleft corner of the display */ + XMoveWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle, + -x, -y); + } + +#endif + + /* Grab the keyboard, too */ + XGrabKeyboard( + fgDisplay.Display, + fgStructure.GameModeWindow->Window.Handle, + FALSE, + GrabModeAsync, GrabModeAsync, + CurrentTime + ); + +#endif + + return fgStructure.GameModeWindow->ID; +} + +/* + * Leaves the game mode + */ +void FGAPIENTRY glutLeaveGameMode( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutLeaveGameMode" ); + + freeglut_return_if_fail( fgStructure.GameModeWindow ); + + fgAddToWindowDestroyList( fgStructure.GameModeWindow ); + fgStructure.GameModeWindow = NULL; + +#if TARGET_HOST_POSIX_X11 + + XUngrabPointer( fgDisplay.Display, CurrentTime ); + XUngrabKeyboard( fgDisplay.Display, CurrentTime ); + +#endif + + fghRestoreState(); +} + +/* + * Returns information concerning the freeglut game mode + */ +int FGAPIENTRY glutGameModeGet( GLenum eWhat ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGameModeGet" ); + + switch( eWhat ) + { + case GLUT_GAME_MODE_ACTIVE: + return !!fgStructure.GameModeWindow; + + case GLUT_GAME_MODE_POSSIBLE: + return fghChangeDisplayMode( GL_TRUE ); + + case GLUT_GAME_MODE_WIDTH: + return fgState.GameModeSize.X; + + case GLUT_GAME_MODE_HEIGHT: + return fgState.GameModeSize.Y; + + case GLUT_GAME_MODE_PIXEL_DEPTH: + return fgState.GameModeDepth; + + case GLUT_GAME_MODE_REFRESH_RATE: + return fgState.GameModeRefresh; + + case GLUT_GAME_MODE_DISPLAY_CHANGED: + /* + * This is true if the game mode has been activated successfully.. + */ + return !!fgStructure.GameModeWindow; + } + + fgWarning( "Unknown gamemode get: %d", eWhat ); + return -1; +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_geometry.c b/examples/opengl-framework/freeglut/freeglut_geometry.c new file mode 100644 index 00000000..0f9f5c53 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_geometry.c @@ -0,0 +1,1215 @@ +/* + * freeglut_geometry.c + * + * Freeglut geometry rendering methods. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Fri Dec 3 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* + * TODO BEFORE THE STABLE RELEASE: + * + * Following functions have been contributed by Andreas Umbach. + * + * glutWireCube() -- looks OK + * glutSolidCube() -- OK + * + * Those functions have been implemented by John Fay. + * + * glutWireTorus() -- looks OK + * glutSolidTorus() -- looks OK + * glutWireDodecahedron() -- looks OK + * glutSolidDodecahedron() -- looks OK + * glutWireOctahedron() -- looks OK + * glutSolidOctahedron() -- looks OK + * glutWireTetrahedron() -- looks OK + * glutSolidTetrahedron() -- looks OK + * glutWireIcosahedron() -- looks OK + * glutSolidIcosahedron() -- looks OK + * + * The Following functions have been updated by Nigel Stewart, based + * on FreeGLUT 2.0.0 implementations: + * + * glutWireSphere() -- looks OK + * glutSolidSphere() -- looks OK + * glutWireCone() -- looks OK + * glutSolidCone() -- looks OK + */ + + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * Draws a wireframed cube. Code contributed by Andreas Umbach + */ +void FGAPIENTRY glutWireCube( GLdouble dSize ) +{ + double size = dSize * 0.5; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWireCube" ); + +# define V(a,b,c) glVertex3d( a size, b size, c size ); +# define N(a,b,c) glNormal3d( a, b, c ); + + /* PWO: I dared to convert the code to use macros... */ + glBegin( GL_LINE_LOOP ); N( 1.0, 0.0, 0.0); V(+,-,+); V(+,-,-); V(+,+,-); V(+,+,+); glEnd(); + glBegin( GL_LINE_LOOP ); N( 0.0, 1.0, 0.0); V(+,+,+); V(+,+,-); V(-,+,-); V(-,+,+); glEnd(); + glBegin( GL_LINE_LOOP ); N( 0.0, 0.0, 1.0); V(+,+,+); V(-,+,+); V(-,-,+); V(+,-,+); glEnd(); + glBegin( GL_LINE_LOOP ); N(-1.0, 0.0, 0.0); V(-,-,+); V(-,+,+); V(-,+,-); V(-,-,-); glEnd(); + glBegin( GL_LINE_LOOP ); N( 0.0,-1.0, 0.0); V(-,-,+); V(-,-,-); V(+,-,-); V(+,-,+); glEnd(); + glBegin( GL_LINE_LOOP ); N( 0.0, 0.0,-1.0); V(-,-,-); V(-,+,-); V(+,+,-); V(+,-,-); glEnd(); + +# undef V +# undef N +} + +/* + * Draws a solid cube. Code contributed by Andreas Umbach + */ +void FGAPIENTRY glutSolidCube( GLdouble dSize ) +{ + double size = dSize * 0.5; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSolidCube" ); + +# define V(a,b,c) glVertex3d( a size, b size, c size ); +# define N(a,b,c) glNormal3d( a, b, c ); + + /* PWO: Again, I dared to convert the code to use macros... */ + glBegin( GL_QUADS ); + N( 1.0, 0.0, 0.0); V(+,-,+); V(+,-,-); V(+,+,-); V(+,+,+); + N( 0.0, 1.0, 0.0); V(+,+,+); V(+,+,-); V(-,+,-); V(-,+,+); + N( 0.0, 0.0, 1.0); V(+,+,+); V(-,+,+); V(-,-,+); V(+,-,+); + N(-1.0, 0.0, 0.0); V(-,-,+); V(-,+,+); V(-,+,-); V(-,-,-); + N( 0.0,-1.0, 0.0); V(-,-,+); V(-,-,-); V(+,-,-); V(+,-,+); + N( 0.0, 0.0,-1.0); V(-,-,-); V(-,+,-); V(+,+,-); V(+,-,-); + glEnd(); + +# undef V +# undef N +} + +/* + * Compute lookup table of cos and sin values forming a cirle + * + * Notes: + * It is the responsibility of the caller to free these tables + * The size of the table is (n+1) to form a connected loop + * The last entry is exactly the same as the first + * The sign of n can be flipped to get the reverse loop + */ + +static void fghCircleTable(double **sint,double **cost,const int n) +{ + int i; + + /* Table size, the sign of n flips the circle direction */ + + const int size = abs(n); + + /* Determine the angle between samples */ + + const double angle = 2*M_PI/(double)( ( n == 0 ) ? 1 : n ); + + /* Allocate memory for n samples, plus duplicate of first entry at the end */ + + *sint = (double *) calloc(sizeof(double), size+1); + *cost = (double *) calloc(sizeof(double), size+1); + + /* Bail out if memory allocation fails, fgError never returns */ + + if (!(*sint) || !(*cost)) + { + free(*sint); + free(*cost); + fgError("Failed to allocate memory in fghCircleTable"); + } + + /* Compute cos and sin around the circle */ + + (*sint)[0] = 0.0; + (*cost)[0] = 1.0; + + for (i=1; i0)?1:0]; + r0 = 0.0; + r1 = sint2[(stacks>0)?1:0]; + + glBegin(GL_TRIANGLE_FAN); + + glNormal3d(0,0,1); + glVertex3d(0,0,radius); + + for (j=slices; j>=0; j--) + { + glNormal3d(cost1[j]*r1, sint1[j]*r1, z1 ); + glVertex3d(cost1[j]*r1*radius, sint1[j]*r1*radius, z1*radius); + } + + glEnd(); + + /* Cover each stack with a quad strip, except the top and bottom stacks */ + + for( i=1; i 0 ) ? stacks : 1 ); + const double rStep = base / ( ( stacks > 0 ) ? stacks : 1 ); + + /* Scaling factors for vertex normals */ + + const double cosn = ( height / sqrt ( height * height + base * base )); + const double sinn = ( base / sqrt ( height * height + base * base )); + + /* Pre-computed circle */ + + double *sint,*cost; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSolidCone" ); + + fghCircleTable(&sint,&cost,-slices); + + /* Cover the circular base with a triangle fan... */ + + z0 = 0.0; + z1 = zStep; + + r0 = base; + r1 = r0 - rStep; + + glBegin(GL_TRIANGLE_FAN); + + glNormal3d(0.0,0.0,-1.0); + glVertex3d(0.0,0.0, z0 ); + + for (j=0; j<=slices; j++) + glVertex3d(cost[j]*r0, sint[j]*r0, z0); + + glEnd(); + + /* Cover each stack with a quad strip, except the top stack */ + + for( i=0; i 0 ) ? stacks : 1 ); + const double rStep = base / ( ( stacks > 0 ) ? stacks : 1 ); + + /* Scaling factors for vertex normals */ + + const double cosn = ( height / sqrt ( height * height + base * base )); + const double sinn = ( base / sqrt ( height * height + base * base )); + + /* Pre-computed circle */ + + double *sint,*cost; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWireCone" ); + + fghCircleTable(&sint,&cost,-slices); + + /* Draw the stacks... */ + + for (i=0; i 0 ) ? stacks : 1 ); + + /* Pre-computed circle */ + + double *sint,*cost; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSolidCylinder" ); + + fghCircleTable(&sint,&cost,-slices); + + /* Cover the base and top */ + + glBegin(GL_TRIANGLE_FAN); + glNormal3d(0.0, 0.0, -1.0 ); + glVertex3d(0.0, 0.0, 0.0 ); + for (j=0; j<=slices; j++) + glVertex3d(cost[j]*radius, sint[j]*radius, 0.0); + glEnd(); + + glBegin(GL_TRIANGLE_FAN); + glNormal3d(0.0, 0.0, 1.0 ); + glVertex3d(0.0, 0.0, height); + for (j=slices; j>=0; j--) + glVertex3d(cost[j]*radius, sint[j]*radius, height); + glEnd(); + + /* Do the stacks */ + + z0 = 0.0; + z1 = zStep; + + for (i=1; i<=stacks; i++) + { + if (i==stacks) + z1 = height; + + glBegin(GL_QUAD_STRIP); + for (j=0; j<=slices; j++ ) + { + glNormal3d(cost[j], sint[j], 0.0 ); + glVertex3d(cost[j]*radius, sint[j]*radius, z0 ); + glVertex3d(cost[j]*radius, sint[j]*radius, z1 ); + } + glEnd(); + + z0 = z1; z1 += zStep; + } + + /* Release sin and cos tables */ + + free(sint); + free(cost); +} + +/* + * Draws a wire cylinder + */ +void FGAPIENTRY glutWireCylinder(GLdouble radius, GLdouble height, GLint slices, GLint stacks) +{ + int i,j; + + /* Step in z and radius as stacks are drawn. */ + + double z = 0.0; + const double zStep = height / ( ( stacks > 0 ) ? stacks : 1 ); + + /* Pre-computed circle */ + + double *sint,*cost; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWireCylinder" ); + + fghCircleTable(&sint,&cost,-slices); + + /* Draw the stacks... */ + + for (i=0; i<=stacks; i++) + { + if (i==stacks) + z = height; + + glBegin(GL_LINE_LOOP); + + for( j=0; j 0 ) + { + GLdouble local_offset[3] ; /* Use a local variable to avoid buildup of roundoff errors */ + num_levels -- ; + scale /= 2.0 ; + for ( i = 0 ; i < NUM_TETR_FACES ; i++ ) + { + local_offset[0] = offset[0] + scale * tet_r[i][0] ; + local_offset[1] = offset[1] + scale * tet_r[i][1] ; + local_offset[2] = offset[2] + scale * tet_r[i][2] ; + glutWireSierpinskiSponge ( num_levels, local_offset, scale ) ; + } + } +} + +void FGAPIENTRY glutSolidSierpinskiSponge ( int num_levels, GLdouble offset[3], GLdouble scale ) +{ + int i, j ; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSolidSierpinskiSponge" ); + + if ( num_levels == 0 ) + { + glBegin ( GL_TRIANGLES ) ; + + for ( i = 0 ; i < NUM_TETR_FACES ; i++ ) + { + glNormal3d ( -tet_r[i][0], -tet_r[i][1], -tet_r[i][2] ) ; + for ( j = 0; j < 3; j++ ) + { + double x = offset[0] + scale * tet_r[tet_i[i][j]][0] ; + double y = offset[1] + scale * tet_r[tet_i[i][j]][1] ; + double z = offset[2] + scale * tet_r[tet_i[i][j]][2] ; + glVertex3d ( x, y, z ) ; + } + } + + glEnd () ; + } + else if ( num_levels > 0 ) + { + GLdouble local_offset[3] ; /* Use a local variable to avoid buildup of roundoff errors */ + num_levels -- ; + scale /= 2.0 ; + for ( i = 0 ; i < NUM_TETR_FACES ; i++ ) + { + local_offset[0] = offset[0] + scale * tet_r[i][0] ; + local_offset[1] = offset[1] + scale * tet_r[i][1] ; + local_offset[2] = offset[2] + scale * tet_r[i][2] ; + glutSolidSierpinskiSponge ( num_levels, local_offset, scale ) ; + } + } +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_glutfont_definitions.c b/examples/opengl-framework/freeglut/freeglut_glutfont_definitions.c new file mode 100644 index 00000000..454ecb73 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_glutfont_definitions.c @@ -0,0 +1,108 @@ +/* + * freeglut_glutfont_definitions.c + * + * Bitmap and stroke fonts displaying. + * + * Copyright (c) 2003 Stephen J. Baker (whether he wants it or not). + * All Rights Reserved. + * Written by John F. Fay , who releases the + * copyright over to the "freeglut" project lead. + * Creation date: Mon July 21 2003 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +/* + * This file is necessary for the *nix version of "freeglut" because the + * original GLUT defined its font variables in rather an unusual way. + * Publicly, in "glut.h", they were defined as "void *". Privately, + * in one of the source code files, they were defined as pointers to a + * structure. Most compilers and linkers are satisfied with the "void *" + * and don't go any farther, but some of them balked. In particular, + * when compiling with "freeglut" and then trying to run using the GLUT + * ".so" library, some of them would give an error. So we are having to + * create this file to define the variables as pointers to an unusual + * structure to match GLUT. + */ + +/* + * freeglut_internal.h uses some GL types, but including the GL header portably + * is a bit tricky, so we include freeglut_std.h here, which contains the + * necessary machinery. But this poses another problem, caused by the ugly + * original defintion of the font constants in "classic" GLUT: They are defined + * as void* externally, so we move them temporarily out of the way by AN EXTREME + * CPP HACK. + */ + +#define glutStrokeRoman glutStrokeRomanIGNOREME +#define glutStrokeMonoRoman glutStrokeMonoRomanIGNOREME +#define glutBitmap9By15 glutBitmap9By15IGNOREME +#define glutBitmap8By13 glutBitmap8By13IGNOREME +#define glutBitmapTimesRoman10 glutBitmapTimesRoman10IGNOREME +#define glutBitmapTimesRoman24 glutBitmapTimesRoman24IGNOREME +#define glutBitmapHelvetica10 glutBitmapHelvetica10IGNOREME +#define glutBitmapHelvetica12 glutBitmapHelvetica12IGNOREME +#define glutBitmapHelvetica18 glutBitmapHelvetica18IGNOREME + +#include + +#undef glutStrokeRoman +#undef glutStrokeMonoRoman +#undef glutBitmap9By15 +#undef glutBitmap8By13 +#undef glutBitmapTimesRoman10 +#undef glutBitmapTimesRoman24 +#undef glutBitmapHelvetica10 +#undef glutBitmapHelvetica12 +#undef glutBitmapHelvetica18 + +#include "freeglut_internal.h" + +#if TARGET_HOST_POSIX_X11 + +struct freeglutStrokeFont +{ + const char *name ; + int num_chars ; + void *ch ; + float top ; + float bottom ; +}; + +struct freeglutBitmapFont +{ + const char *name ; + const int num_chars ; + const int first ; + const void *ch ; +}; + + +struct freeglutStrokeFont glutStrokeRoman ; +struct freeglutStrokeFont glutStrokeMonoRoman ; + +struct freeglutBitmapFont glutBitmap9By15 ; +struct freeglutBitmapFont glutBitmap8By13 ; +struct freeglutBitmapFont glutBitmapTimesRoman10 ; +struct freeglutBitmapFont glutBitmapTimesRoman24 ; +struct freeglutBitmapFont glutBitmapHelvetica10 ; +struct freeglutBitmapFont glutBitmapHelvetica12 ; +struct freeglutBitmapFont glutBitmapHelvetica18 ; + +#endif + diff --git a/examples/opengl-framework/freeglut/freeglut_init.c b/examples/opengl-framework/freeglut/freeglut_init.c new file mode 100644 index 00000000..f59a456c --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_init.c @@ -0,0 +1,1180 @@ +/* + * freeglut_init.c + * + * Various freeglut initialization functions. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 2 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#define FREEGLUT_BUILDING_LIB +#include +#include "freeglut_internal.h" + +#if TARGET_HOST_POSIX_X11 +#include /* LONG_MAX */ +#endif + +/* + * TODO BEFORE THE STABLE RELEASE: + * + * fgDeinitialize() -- Win32's OK, X11 needs the OS-specific + * deinitialization done + * glutInitDisplayString() -- display mode string parsing + * + * Wouldn't it be cool to use gettext() for error messages? I just love + * bash saying "nie znaleziono pliku" instead of "file not found" :) + * Is gettext easily portable? + */ + +/* -- GLOBAL VARIABLES ----------------------------------------------------- */ + +/* + * A structure pointed by g_pDisplay holds all information + * regarding the display, screen, root window etc. + */ +SFG_Display fgDisplay; + +/* + * The settings for the current freeglut session + */ +SFG_State fgState = { { -1, -1, GL_FALSE }, /* Position */ + { 300, 300, GL_TRUE }, /* Size */ + GLUT_RGBA | GLUT_SINGLE | GLUT_DEPTH, /* DisplayMode */ + GL_FALSE, /* Initialised */ + GLUT_TRY_DIRECT_CONTEXT, /* DirectContext */ + GL_FALSE, /* ForceIconic */ + GL_FALSE, /* UseCurrentContext */ + GL_FALSE, /* GLDebugSwitch */ + GL_FALSE, /* XSyncSwitch */ + GLUT_KEY_REPEAT_ON, /* KeyRepeat */ + INVALID_MODIFIERS, /* Modifiers */ + 0, /* FPSInterval */ + 0, /* SwapCount */ + 0, /* SwapTime */ + 0, /* Time */ + { NULL, NULL }, /* Timers */ + { NULL, NULL }, /* FreeTimers */ + NULL, /* IdleCallback */ + 0, /* ActiveMenus */ + NULL, /* MenuStateCallback */ + NULL, /* MenuStatusCallback */ + { 640, 480, GL_TRUE }, /* GameModeSize */ + 16, /* GameModeDepth */ + 72, /* GameModeRefresh */ + GLUT_ACTION_EXIT, /* ActionOnWindowClose */ + GLUT_EXEC_STATE_INIT, /* ExecState */ + NULL, /* ProgramName */ + GL_FALSE, /* JoysticksInitialised */ + 0, /* NumActiveJoysticks */ + GL_FALSE, /* InputDevsInitialised */ + 0, /* MouseWheelTicks */ + 1, /* AuxiliaryBufferNumber */ + 4, /* SampleNumber */ + 1, /* MajorVersion */ + 0, /* MinorVersion */ + 0, /* ContextFlags */ + 0, /* ContextProfile */ + NULL, /* ErrorFunc */ + NULL /* WarningFunc */ +}; + + +/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ + +#if TARGET_HOST_POSIX_X11 + +/* Return the atom associated with "name". */ +static Atom fghGetAtom(const char * name) +{ + return XInternAtom(fgDisplay.Display, name, False); +} + +/* + * Check if "property" is set on "window". The property's values are returned + * through "data". If the property is set and is of type "type", return the + * number of elements in "data". Return zero otherwise. In both cases, use + * "Xfree()" to free "data". + */ +static int fghGetWindowProperty(Window window, + Atom property, + Atom type, + unsigned char ** data) +{ + /* + * Caller always has to use "Xfree()" to free "data", since + * "XGetWindowProperty() always allocates one extra byte in prop_return + * [i.e. "data"] (even if the property is zero length) [..]". + */ + + int status; /* Returned by "XGetWindowProperty". */ + + Atom type_returned; + int temp_format; /* Not used. */ + unsigned long number_of_elements; + unsigned long temp_bytes_after; /* Not used. */ + + + status = XGetWindowProperty(fgDisplay.Display, + window, + property, + 0, + LONG_MAX, + False, + type, + &type_returned, + &temp_format, + &number_of_elements, + &temp_bytes_after, + data); + + FREEGLUT_INTERNAL_ERROR_EXIT(status == Success, + "XGetWindowProperty failled", + "fghGetWindowProperty"); + + if (type_returned != type) + { + number_of_elements = 0; + } + + return number_of_elements; +} + +/* Check if the window manager is NET WM compliant. */ +static int fghNetWMSupported(void) +{ + Atom wm_check; + Window ** window_ptr_1; + + int number_of_windows; + int net_wm_supported; + + + net_wm_supported = 0; + + wm_check = fghGetAtom("_NET_SUPPORTING_WM_CHECK"); + window_ptr_1 = malloc(sizeof(Window *)); + + /* + * Check that the window manager has set this property on the root window. + * The property must be the ID of a child window. + */ + number_of_windows = fghGetWindowProperty(fgDisplay.RootWindow, + wm_check, + XA_WINDOW, + (unsigned char **) window_ptr_1); + if (number_of_windows == 1) + { + Window ** window_ptr_2; + + window_ptr_2 = malloc(sizeof(Window *)); + + /* Check that the window has the same property set to the same value. */ + number_of_windows = fghGetWindowProperty(**window_ptr_1, + wm_check, + XA_WINDOW, + (unsigned char **) window_ptr_2); + if ((number_of_windows == 1) && (**window_ptr_1 == **window_ptr_2)) + { + /* NET WM compliant */ + net_wm_supported = 1; + } + + XFree(*window_ptr_2); + free(window_ptr_2); + } + + XFree(*window_ptr_1); + free(window_ptr_1); + + return net_wm_supported; +} + +/* Check if "hint" is present in "property" for "window". */ +int fgHintPresent(Window window, Atom property, Atom hint) +{ + Atom *atoms; + int number_of_atoms; + int supported; + int i; + + supported = 0; + + number_of_atoms = fghGetWindowProperty(window, + property, + XA_ATOM, + (unsigned char **) &atoms); + for (i = 0; i < number_of_atoms; i++) + { + if (atoms[i] == hint) + { + supported = 1; + break; + } + } + + XFree(atoms); + return supported; +} + +#endif /* TARGET_HOST_POSIX_X11 */ + + +/* + * A call to this function should initialize all the display stuff... + */ +static void fghInitialize( const char* displayName ) +{ +#if TARGET_HOST_POSIX_X11 + fgDisplay.Display = XOpenDisplay( displayName ); + + if( fgDisplay.Display == NULL ) + fgError( "failed to open display '%s'", XDisplayName( displayName ) ); + + if( !glXQueryExtension( fgDisplay.Display, NULL, NULL ) ) + fgError( "OpenGL GLX extension not supported by display '%s'", + XDisplayName( displayName ) ); + + fgDisplay.Screen = DefaultScreen( fgDisplay.Display ); + fgDisplay.RootWindow = RootWindow( + fgDisplay.Display, + fgDisplay.Screen + ); + + fgDisplay.ScreenWidth = DisplayWidth( + fgDisplay.Display, + fgDisplay.Screen + ); + fgDisplay.ScreenHeight = DisplayHeight( + fgDisplay.Display, + fgDisplay.Screen + ); + + fgDisplay.ScreenWidthMM = DisplayWidthMM( + fgDisplay.Display, + fgDisplay.Screen + ); + fgDisplay.ScreenHeightMM = DisplayHeightMM( + fgDisplay.Display, + fgDisplay.Screen + ); + + fgDisplay.Connection = ConnectionNumber( fgDisplay.Display ); + + /* Create the window deletion atom */ + fgDisplay.DeleteWindow = fghGetAtom("WM_DELETE_WINDOW"); + + /* Create the state and full screen atoms */ + fgDisplay.State = None; + fgDisplay.StateFullScreen = None; + + if (fghNetWMSupported()) + { + const Atom supported = fghGetAtom("_NET_SUPPORTED"); + const Atom state = fghGetAtom("_NET_WM_STATE"); + + /* Check if the state hint is supported. */ + if (fgHintPresent(fgDisplay.RootWindow, supported, state)) + { + const Atom full_screen = fghGetAtom("_NET_WM_STATE_FULLSCREEN"); + + fgDisplay.State = state; + + /* Check if the window manager supports full screen. */ + /** Check "_NET_WM_ALLOWED_ACTIONS" on our window instead? **/ + if (fgHintPresent(fgDisplay.RootWindow, supported, full_screen)) + { + fgDisplay.StateFullScreen = full_screen; + } + } + } + +#elif TARGET_HOST_MS_WINDOWS + + WNDCLASS wc; + ATOM atom; + + /* What we need to do is to initialize the fgDisplay global structure here. */ + fgDisplay.Instance = GetModuleHandle( NULL ); + fgDisplay.DisplayName= displayName ? strdup(displayName) : 0 ; + atom = GetClassInfo( fgDisplay.Instance, _T("FREEGLUT"), &wc ); + + if( atom == 0 ) + { + ZeroMemory( &wc, sizeof(WNDCLASS) ); + + /* + * Each of the windows should have its own device context, and we + * want redraw events during Vertical and Horizontal Resizes by + * the user. + * + * XXX Old code had "| CS_DBCLCKS" commented out. Plans for the + * XXX future? Dead-end idea? + */ + wc.lpfnWndProc = fgWindowProc; + wc.cbClsExtra = 0; + wc.cbWndExtra = 0; + wc.hInstance = fgDisplay.Instance; + wc.hIcon = LoadIcon( fgDisplay.Instance, _T("GLUT_ICON") ); + +#if defined(_WIN32_WCE) + wc.style = CS_HREDRAW | CS_VREDRAW; +#else + wc.style = CS_OWNDC | CS_HREDRAW | CS_VREDRAW; + if (!wc.hIcon) + wc.hIcon = LoadIcon( NULL, IDI_WINLOGO ); +#endif + + wc.hCursor = LoadCursor( NULL, IDC_ARROW ); + wc.hbrBackground = NULL; + wc.lpszMenuName = NULL; + wc.lpszClassName = _T("FREEGLUT"); + + /* Register the window class */ + atom = RegisterClass( &wc ); + FREEGLUT_INTERNAL_ERROR_EXIT ( atom, "Window Class Not Registered", "fghInitialize" ); + } + + /* The screen dimensions can be obtained via GetSystemMetrics() calls */ + fgDisplay.ScreenWidth = GetSystemMetrics( SM_CXSCREEN ); + fgDisplay.ScreenHeight = GetSystemMetrics( SM_CYSCREEN ); + + { + HWND desktop = GetDesktopWindow( ); + HDC context = GetDC( desktop ); + + fgDisplay.ScreenWidthMM = GetDeviceCaps( context, HORZSIZE ); + fgDisplay.ScreenHeightMM = GetDeviceCaps( context, VERTSIZE ); + + ReleaseDC( desktop, context ); + } + /* If we have a DisplayName try to use it for metrics */ + if( fgDisplay.DisplayName ) + { + HDC context = CreateDC(fgDisplay.DisplayName,0,0,0); + if( context ) + { + fgDisplay.ScreenWidth = GetDeviceCaps( context, HORZRES ); + fgDisplay.ScreenHeight = GetDeviceCaps( context, VERTRES ); + fgDisplay.ScreenWidthMM = GetDeviceCaps( context, HORZSIZE ); + fgDisplay.ScreenHeightMM = GetDeviceCaps( context, VERTSIZE ); + DeleteDC(context); + } + else + fgWarning("fghInitialize: " + "CreateDC failed, Screen size info may be incorrect\n" + "This is quite likely caused by a bad '-display' parameter"); + + } + /* Set the timer granularity to 1 ms */ + timeBeginPeriod ( 1 ); + +#endif + + fgState.Initialised = GL_TRUE; + + /* Avoid registering atexit callback on Win32 as it results in an access + * violation due to calling into a module which has been unloaded. */ +#if ( TARGET_HOST_MS_WINDOWS == 0 ) + atexit(fgDeinitialize); +#endif + + /* InputDevice uses GlutTimerFunc(), so fgState.Initialised must be TRUE */ + fgInitialiseInputDevices(); +} + +/* + * Perform the freeglut deinitialization... + */ +void fgDeinitialize( void ) +{ + SFG_Timer *timer; + + if( !fgState.Initialised ) + { + return; + } + + /* If we're in game mode, we want to leave game mode */ + if( fgStructure.GameModeWindow ) { + glutLeaveGameMode(); + } + + /* If there was a menu created, destroy the rendering context */ + if( fgStructure.MenuContext ) + { +#if TARGET_HOST_POSIX_X11 + /* Note that the MVisualInfo is not owned by the MenuContext! */ + glXDestroyContext( fgDisplay.Display, fgStructure.MenuContext->MContext ); +#endif + free( fgStructure.MenuContext ); + fgStructure.MenuContext = NULL; + } + + fgDestroyStructure( ); + + while( ( timer = fgState.Timers.First) ) + { + fgListRemove( &fgState.Timers, &timer->Node ); + free( timer ); + } + + while( ( timer = fgState.FreeTimers.First) ) + { + fgListRemove( &fgState.FreeTimers, &timer->Node ); + free( timer ); + } + +#if !defined(_WIN32_WCE) + if ( fgState.JoysticksInitialised ) + fgJoystickClose( ); + + if ( fgState.InputDevsInitialised ) + fgInputDeviceClose( ); +#endif /* !defined(_WIN32_WCE) */ + fgState.JoysticksInitialised = GL_FALSE; + fgState.InputDevsInitialised = GL_FALSE; + + fgState.MouseWheelTicks = 0; + + fgState.MajorVersion = 1; + fgState.MinorVersion = 0; + fgState.ContextFlags = 0; + fgState.ContextProfile = 0; + + fgState.Initialised = GL_FALSE; + + fgState.Position.X = -1; + fgState.Position.Y = -1; + fgState.Position.Use = GL_FALSE; + + fgState.Size.X = 300; + fgState.Size.Y = 300; + fgState.Size.Use = GL_TRUE; + + fgState.DisplayMode = GLUT_RGBA | GLUT_SINGLE | GLUT_DEPTH; + + fgState.DirectContext = GLUT_TRY_DIRECT_CONTEXT; + fgState.ForceIconic = GL_FALSE; + fgState.UseCurrentContext = GL_FALSE; + fgState.GLDebugSwitch = GL_FALSE; + fgState.XSyncSwitch = GL_FALSE; + fgState.ActionOnWindowClose = GLUT_ACTION_EXIT; + fgState.ExecState = GLUT_EXEC_STATE_INIT; + + fgState.KeyRepeat = GLUT_KEY_REPEAT_ON; + fgState.Modifiers = INVALID_MODIFIERS; + + fgState.GameModeSize.X = 640; + fgState.GameModeSize.Y = 480; + fgState.GameModeDepth = 16; + fgState.GameModeRefresh = 72; + + fgListInit( &fgState.Timers ); + fgListInit( &fgState.FreeTimers ); + + fgState.IdleCallback = NULL; + fgState.MenuStateCallback = ( FGCBMenuState )NULL; + fgState.MenuStatusCallback = ( FGCBMenuStatus )NULL; + + fgState.SwapCount = 0; + fgState.SwapTime = 0; + fgState.FPSInterval = 0; + + if( fgState.ProgramName ) + { + free( fgState.ProgramName ); + fgState.ProgramName = NULL; + } + +#if TARGET_HOST_POSIX_X11 + + /* + * Make sure all X-client data we have created will be destroyed on + * display closing + */ + XSetCloseDownMode( fgDisplay.Display, DestroyAll ); + + /* + * Close the display connection, destroying all windows we have + * created so far + */ + XCloseDisplay( fgDisplay.Display ); + +#elif TARGET_HOST_MS_WINDOWS + if( fgDisplay.DisplayName ) + { + free( fgDisplay.DisplayName ); + fgDisplay.DisplayName = NULL; + } + + /* Reset the timer granularity */ + timeEndPeriod ( 1 ); + +#endif + + fgState.Initialised = GL_FALSE; +} + +/* + * Everything inside the following #ifndef is copied from the X sources. + */ + +#if TARGET_HOST_MS_WINDOWS + +/* + +Copyright 1985, 1986, 1987,1998 The Open Group + +Permission to use, copy, modify, distribute, and sell this software and its +documentation for any purpose is hereby granted without fee, provided that +the above copyright notice appear in all copies and that both that +copyright notice and this permission notice appear in supporting +documentation. + +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 OPEN GROUP 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. + +Except as contained in this notice, the name of The Open Group shall +not be used in advertising or otherwise to promote the sale, use or +other dealings in this Software without prior written authorization +from The Open Group. + +*/ + +#define NoValue 0x0000 +#define XValue 0x0001 +#define YValue 0x0002 +#define WidthValue 0x0004 +#define HeightValue 0x0008 +#define AllValues 0x000F +#define XNegative 0x0010 +#define YNegative 0x0020 + +/* + * XParseGeometry parses strings of the form + * "=x{+-}{+-}", where + * width, height, xoffset, and yoffset are unsigned integers. + * Example: "=80x24+300-49" + * The equal sign is optional. + * It returns a bitmask that indicates which of the four values + * were actually found in the string. For each value found, + * the corresponding argument is updated; for each value + * not found, the corresponding argument is left unchanged. + */ + +static int +ReadInteger(char *string, char **NextString) +{ + register int Result = 0; + int Sign = 1; + + if (*string == '+') + string++; + else if (*string == '-') + { + string++; + Sign = -1; + } + for (; (*string >= '0') && (*string <= '9'); string++) + { + Result = (Result * 10) + (*string - '0'); + } + *NextString = string; + if (Sign >= 0) + return Result; + else + return -Result; +} + +static int XParseGeometry ( + const char *string, + int *x, + int *y, + unsigned int *width, /* RETURN */ + unsigned int *height) /* RETURN */ +{ + int mask = NoValue; + register char *strind; + unsigned int tempWidth = 0, tempHeight = 0; + int tempX = 0, tempY = 0; + char *nextCharacter; + + if ( (string == NULL) || (*string == '\0')) + return mask; + if (*string == '=') + string++; /* ignore possible '=' at beg of geometry spec */ + + strind = (char *)string; + if (*strind != '+' && *strind != '-' && *strind != 'x') { + tempWidth = ReadInteger(strind, &nextCharacter); + if (strind == nextCharacter) + return 0; + strind = nextCharacter; + mask |= WidthValue; + } + + if (*strind == 'x' || *strind == 'X') { + strind++; + tempHeight = ReadInteger(strind, &nextCharacter); + if (strind == nextCharacter) + return 0; + strind = nextCharacter; + mask |= HeightValue; + } + + if ((*strind == '+') || (*strind == '-')) { + if (*strind == '-') { + strind++; + tempX = -ReadInteger(strind, &nextCharacter); + if (strind == nextCharacter) + return 0; + strind = nextCharacter; + mask |= XNegative; + } + else + { + strind++; + tempX = ReadInteger(strind, &nextCharacter); + if (strind == nextCharacter) + return 0; + strind = nextCharacter; + } + mask |= XValue; + if ((*strind == '+') || (*strind == '-')) { + if (*strind == '-') { + strind++; + tempY = -ReadInteger(strind, &nextCharacter); + if (strind == nextCharacter) + return 0; + strind = nextCharacter; + mask |= YNegative; + } + else + { + strind++; + tempY = ReadInteger(strind, &nextCharacter); + if (strind == nextCharacter) + return 0; + strind = nextCharacter; + } + mask |= YValue; + } + } + + /* If strind isn't at the end of the string the it's an invalid + geometry specification. */ + + if (*strind != '\0') return 0; + + if (mask & XValue) + *x = tempX; + if (mask & YValue) + *y = tempY; + if (mask & WidthValue) + *width = tempWidth; + if (mask & HeightValue) + *height = tempHeight; + return mask; +} +#endif + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * Perform initialization. This usually happens on the program startup + * and restarting after glutMainLoop termination... + */ +void FGAPIENTRY glutInit( int* pargc, char** argv ) +{ + char* displayName = NULL; + char* geometry = NULL; + int i, j, argc = *pargc; + + if( fgState.Initialised ) + fgError( "illegal glutInit() reinitialization attempt" ); + + if (pargc && *pargc && argv && *argv && **argv) + { + fgState.ProgramName = strdup (*argv); + + if( !fgState.ProgramName ) + fgError ("Could not allocate space for the program's name."); + } + + fgCreateStructure( ); + + /* Get start time */ + fgState.Time = fgSystemTime(); + + /* check if GLUT_FPS env var is set */ +#ifndef _WIN32_WCE + { + const char *fps = getenv( "GLUT_FPS" ); + + if( fps ) + { + int interval; + sscanf( fps, "%d", &interval ); + + if( interval <= 0 ) + fgState.FPSInterval = 5000; /* 5000 millisecond default */ + else + fgState.FPSInterval = interval; + } + } + + displayName = getenv( "DISPLAY" ); + + for( i = 1; i < argc; i++ ) + { + if( strcmp( argv[ i ], "-display" ) == 0 ) + { + if( ++i >= argc ) + fgError( "-display parameter must be followed by display name" ); + + displayName = argv[ i ]; + + argv[ i - 1 ] = NULL; + argv[ i ] = NULL; + ( *pargc ) -= 2; + } + else if( strcmp( argv[ i ], "-geometry" ) == 0 ) + { + if( ++i >= argc ) + fgError( "-geometry parameter must be followed by window " + "geometry settings" ); + + geometry = argv[ i ]; + + argv[ i - 1 ] = NULL; + argv[ i ] = NULL; + ( *pargc ) -= 2; + } + else if( strcmp( argv[ i ], "-direct" ) == 0) + { + if( fgState.DirectContext == GLUT_FORCE_INDIRECT_CONTEXT ) + fgError( "parameters ambiguity, -direct and -indirect " + "cannot be both specified" ); + + fgState.DirectContext = GLUT_FORCE_DIRECT_CONTEXT; + argv[ i ] = NULL; + ( *pargc )--; + } + else if( strcmp( argv[ i ], "-indirect" ) == 0 ) + { + if( fgState.DirectContext == GLUT_FORCE_DIRECT_CONTEXT ) + fgError( "parameters ambiguity, -direct and -indirect " + "cannot be both specified" ); + + fgState.DirectContext = GLUT_FORCE_INDIRECT_CONTEXT; + argv[ i ] = NULL; + (*pargc)--; + } + else if( strcmp( argv[ i ], "-iconic" ) == 0 ) + { + fgState.ForceIconic = GL_TRUE; + argv[ i ] = NULL; + ( *pargc )--; + } + else if( strcmp( argv[ i ], "-gldebug" ) == 0 ) + { + fgState.GLDebugSwitch = GL_TRUE; + argv[ i ] = NULL; + ( *pargc )--; + } + else if( strcmp( argv[ i ], "-sync" ) == 0 ) + { + fgState.XSyncSwitch = GL_TRUE; + argv[ i ] = NULL; + ( *pargc )--; + } + } + + /* Compact {argv}. */ + for( i = j = 1; i < *pargc; i++, j++ ) + { + /* Guaranteed to end because there are "*pargc" arguments left */ + while ( argv[ j ] == NULL ) + j++; + if ( i != j ) + argv[ i ] = argv[ j ]; + } + +#endif /* _WIN32_WCE */ + + /* + * Have the display created now. If there wasn't a "-display" + * in the program arguments, we will use the DISPLAY environment + * variable for opening the X display (see code above): + */ + fghInitialize( displayName ); + + /* + * Geometry parsing deffered until here because we may need the screen + * size. + */ + + if (geometry ) + { + unsigned int parsedWidth, parsedHeight; + int mask = XParseGeometry( geometry, + &fgState.Position.X, &fgState.Position.Y, + &parsedWidth, &parsedHeight ); + /* TODO: Check for overflow? */ + fgState.Size.X = parsedWidth; + fgState.Size.Y = parsedHeight; + + if( (mask & (WidthValue|HeightValue)) == (WidthValue|HeightValue) ) + fgState.Size.Use = GL_TRUE; + + if( mask & XNegative ) + fgState.Position.X += fgDisplay.ScreenWidth - fgState.Size.X; + + if( mask & YNegative ) + fgState.Position.Y += fgDisplay.ScreenHeight - fgState.Size.Y; + + if( (mask & (XValue|YValue)) == (XValue|YValue) ) + fgState.Position.Use = GL_TRUE; + } +} + +#if TARGET_HOST_MS_WINDOWS +void (__cdecl *__glutExitFunc)( int return_value ) = NULL; + +void FGAPIENTRY __glutInitWithExit( int *pargc, char **argv, void (__cdecl *exit_function)(int) ) +{ + __glutExitFunc = exit_function; + glutInit(pargc, argv); +} +#endif + +/* + * Undoes all the "glutInit" stuff + */ +void FGAPIENTRY glutExit ( void ) +{ + fgDeinitialize (); +} + +/* + * Sets the default initial window position for new windows + */ +void FGAPIENTRY glutInitWindowPosition( int x, int y ) +{ + fgState.Position.X = x; + fgState.Position.Y = y; + + if( ( x >= 0 ) && ( y >= 0 ) ) + fgState.Position.Use = GL_TRUE; + else + fgState.Position.Use = GL_FALSE; +} + +/* + * Sets the default initial window size for new windows + */ +void FGAPIENTRY glutInitWindowSize( int width, int height ) +{ + fgState.Size.X = width; + fgState.Size.Y = height; + + if( ( width > 0 ) && ( height > 0 ) ) + fgState.Size.Use = GL_TRUE; + else + fgState.Size.Use = GL_FALSE; +} + +/* + * Sets the default display mode for all new windows + */ +void FGAPIENTRY glutInitDisplayMode( unsigned int displayMode ) +{ + /* We will make use of this value when creating a new OpenGL context... */ + fgState.DisplayMode = displayMode; +} + + +/* -- INIT DISPLAY STRING PARSING ------------------------------------------ */ + +static char* Tokens[] = +{ + "alpha", "acca", "acc", "blue", "buffer", "conformant", "depth", "double", + "green", "index", "num", "red", "rgba", "rgb", "luminance", "stencil", + "single", "stereo", "samples", "slow", "win32pdf", "win32pfd", "xvisual", + "xstaticgray", "xgrayscale", "xstaticcolor", "xpseudocolor", + "xtruecolor", "xdirectcolor", + "xstaticgrey", "xgreyscale", "xstaticcolour", "xpseudocolour", + "xtruecolour", "xdirectcolour", "borderless", "aux" +}; +#define NUM_TOKENS (sizeof(Tokens) / sizeof(*Tokens)) + +void FGAPIENTRY glutInitDisplayString( const char* displayMode ) +{ + int glut_state_flag = 0 ; + /* + * Unpack a lot of options from a character string. The options are + * delimited by blanks or tabs. + */ + char *token ; + size_t len = strlen ( displayMode ); + char *buffer = (char *)malloc ( (len+1) * sizeof(char) ); + memcpy ( buffer, displayMode, len ); + buffer[len] = '\0'; + + token = strtok ( buffer, " \t" ); + + while ( token ) + { + /* Process this token */ + int i ; + + /* Temporary fix: Ignore any length specifications and at least + * process the basic token + * TODO: Fix this permanently + */ + size_t cleanlength = strcspn ( token, "=<>~!" ); + + for ( i = 0; i < NUM_TOKENS; i++ ) + { + if ( strncmp ( token, Tokens[i], cleanlength ) == 0 ) break ; + } + + switch ( i ) + { + case 0 : /* "alpha": Alpha color buffer precision in bits */ + glut_state_flag |= GLUT_ALPHA ; /* Somebody fix this for me! */ + break ; + + case 1 : /* "acca": Red, green, blue, and alpha accumulation buffer + precision in bits */ + break ; + + case 2 : /* "acc": Red, green, and blue accumulation buffer precision + in bits with zero bits alpha */ + glut_state_flag |= GLUT_ACCUM ; /* Somebody fix this for me! */ + break ; + + case 3 : /* "blue": Blue color buffer precision in bits */ + break ; + + case 4 : /* "buffer": Number of bits in the color index color buffer + */ + break ; + + case 5 : /* "conformant": Boolean indicating if the frame buffer + configuration is conformant or not */ + break ; + + case 6 : /* "depth": Number of bits of precsion in the depth buffer */ + glut_state_flag |= GLUT_DEPTH ; /* Somebody fix this for me! */ + break ; + + case 7 : /* "double": Boolean indicating if the color buffer is + double buffered */ + glut_state_flag |= GLUT_DOUBLE ; + break ; + + case 8 : /* "green": Green color buffer precision in bits */ + break ; + + case 9 : /* "index": Boolean if the color model is color index or not + */ + glut_state_flag |= GLUT_INDEX ; + break ; + + case 10 : /* "num": A special capability name indicating where the + value represents the Nth frame buffer configuration + matching the description string */ + break ; + + case 11 : /* "red": Red color buffer precision in bits */ + break ; + + case 12 : /* "rgba": Number of bits of red, green, blue, and alpha in + the RGBA color buffer */ + glut_state_flag |= GLUT_RGBA ; /* Somebody fix this for me! */ + break ; + + case 13 : /* "rgb": Number of bits of red, green, and blue in the + RGBA color buffer with zero bits alpha */ + glut_state_flag |= GLUT_RGB ; /* Somebody fix this for me! */ + break ; + + case 14 : /* "luminance": Number of bits of red in the RGBA and zero + bits of green, blue (alpha not specified) of color buffer + precision */ + glut_state_flag |= GLUT_LUMINANCE ; /* Somebody fix this for me! */ + break ; + + case 15 : /* "stencil": Number of bits in the stencil buffer */ + glut_state_flag |= GLUT_STENCIL; /* Somebody fix this for me! */ + break ; + + case 16 : /* "single": Boolean indicate the color buffer is single + buffered */ + glut_state_flag |= GLUT_SINGLE ; + break ; + + case 17 : /* "stereo": Boolean indicating the color buffer supports + OpenGL-style stereo */ + glut_state_flag |= GLUT_STEREO ; + break ; + + case 18 : /* "samples": Indicates the number of multisamples to use + based on GLX's SGIS_multisample extension (for + antialiasing) */ + glut_state_flag |= GLUT_MULTISAMPLE ; /*Somebody fix this for me!*/ + break ; + + case 19 : /* "slow": Boolean indicating if the frame buffer + configuration is slow or not */ + break ; + + case 20 : /* "win32pdf": (incorrect spelling but was there before */ + case 21 : /* "win32pfd": matches the Win32 Pixel Format Descriptor by + number */ +#if TARGET_HOST_MS_WINDOWS +#endif + break ; + + case 22 : /* "xvisual": matches the X visual ID by number */ +#if TARGET_HOST_POSIX_X11 +#endif + break ; + + case 23 : /* "xstaticgray": */ + case 29 : /* "xstaticgrey": boolean indicating if the frame buffer + configuration's X visual is of type StaticGray */ +#if TARGET_HOST_POSIX_X11 +#endif + break ; + + case 24 : /* "xgrayscale": */ + case 30 : /* "xgreyscale": boolean indicating if the frame buffer + configuration's X visual is of type GrayScale */ +#if TARGET_HOST_POSIX_X11 +#endif + break ; + + case 25 : /* "xstaticcolor": */ + case 31 : /* "xstaticcolour": boolean indicating if the frame buffer + configuration's X visual is of type StaticColor */ +#if TARGET_HOST_POSIX_X11 +#endif + break ; + + case 26 : /* "xpseudocolor": */ + case 32 : /* "xpseudocolour": boolean indicating if the frame buffer + configuration's X visual is of type PseudoColor */ +#if TARGET_HOST_POSIX_X11 +#endif + break ; + + case 27 : /* "xtruecolor": */ + case 33 : /* "xtruecolour": boolean indicating if the frame buffer + configuration's X visual is of type TrueColor */ +#if TARGET_HOST_POSIX_X11 +#endif + break ; + + case 28 : /* "xdirectcolor": */ + case 34 : /* "xdirectcolour": boolean indicating if the frame buffer + configuration's X visual is of type DirectColor */ +#if TARGET_HOST_POSIX_X11 +#endif + break ; + + case 35 : /* "borderless": windows should not have borders */ +#if TARGET_HOST_POSIX_X11 +#endif + break ; + + case 36 : /* "aux": some number of aux buffers */ + glut_state_flag |= GLUT_AUX; + break ; + + case 37 : /* Unrecognized */ + fgWarning ( "WARNING - Display string token not recognized: %s", + token ); + break ; + } + + token = strtok ( NULL, " \t" ); + } + + free ( buffer ); + + /* We will make use of this value when creating a new OpenGL context... */ + fgState.DisplayMode = glut_state_flag; +} + +/* -- SETTING OPENGL 3.0 CONTEXT CREATION PARAMETERS ---------------------- */ + +void FGAPIENTRY glutInitContextVersion( int majorVersion, int minorVersion ) +{ + /* We will make use of these valuse when creating a new OpenGL context... */ + fgState.MajorVersion = majorVersion; + fgState.MinorVersion = minorVersion; +} + + +void FGAPIENTRY glutInitContextFlags( int flags ) +{ + /* We will make use of this value when creating a new OpenGL context... */ + fgState.ContextFlags = flags; +} + +void FGAPIENTRY glutInitContextProfile( int profile ) +{ + /* We will make use of this value when creating a new OpenGL context... */ + fgState.ContextProfile = profile; +} + +/* -------------- User Defined Error/Warning Handler Support -------------- */ + +/* + * Sets the user error handler (note the use of va_list for the args to the fmt) + */ +void FGAPIENTRY glutInitErrorFunc( void (* vfgError) ( const char *fmt, va_list ap ) ) +{ + /* This allows user programs to handle freeglut errors */ + fgState.ErrorFunc = vfgError; +} + +/* + * Sets the user warning handler (note the use of va_list for the args to the fmt) + */ +void FGAPIENTRY glutInitWarningFunc( void (* vfgWarning) ( const char *fmt, va_list ap ) ) +{ + /* This allows user programs to handle freeglut warnings */ + fgState.WarningFunc = vfgWarning; +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_input_devices.c b/examples/opengl-framework/freeglut/freeglut_input_devices.c new file mode 100644 index 00000000..04f5fd5d --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_input_devices.c @@ -0,0 +1,378 @@ +/* + * freeglut_input_devices.c + * + * Handles miscellaneous input devices via direct serial-port access. + * Proper X11 XInput device support is not yet supported. + * Also lacks Mac support. + * + * Written by Joe Krahn 2005 + * + * Copyright (c) 2005 Stephen J. Baker. All Rights Reserved. + * + * 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 + * PAWEL W. OLSZTA OR STEPHEN J. BAKER 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. + */ + +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + +#include +#include "freeglut_internal.h" + +#if TARGET_HOST_POSIX_X11 +#ifdef HAVE_ERRNO_H +#include +#endif +#include +#include +#include +#include +#include +#include + +typedef struct { + int fd; + struct termios termio, termio_save; +} SERIALPORT; + +#elif TARGET_HOST_MS_WINDOWS +#include +#include +typedef struct { + HANDLE fh; + COMMTIMEOUTS timeouts_save; + DCB dcb_save; +} SERIALPORT; + +#endif + +/********************* Dialbox definitions ***********************/ + +#define DIAL_NUM_VALUATORS 8 + +/* dial parser state machine states */ +#define DIAL_NEW (-1) +#define DIAL_WHICH_DEVICE 0 +#define DIAL_VALUE_HIGH 1 +#define DIAL_VALUE_LOW 2 + +/* dial/button box commands */ +#define DIAL_INITIALIZE 0x20 +#define DIAL_SET_LEDS 0x75 +#define DIAL_SET_TEXT 0x61 +#define DIAL_SET_AUTO_DIALS 0x50 +#define DIAL_SET_AUTO_DELTA_DIALS 0x51 +#define DIAL_SET_FILTER 0x53 +#define DIAL_SET_BUTTONS_MOM_TYPE 0x71 +#define DIAL_SET_AUTO_MOM_BUTTONS 0x73 +#define DIAL_SET_ALL_LEDS 0x4b +#define DIAL_CLEAR_ALL_LEDS 0x4c + +/* dial/button box replies and events */ +#define DIAL_INITIALIZED 0x20 +#define DIAL_BASE 0x30 +#define DIAL_DELTA_BASE 0x40 +#define DIAL_PRESS_BASE 0xc0 +#define DIAL_RELEASE_BASE 0xe0 + +/* macros to determine reply type */ +#define IS_DIAL_EVENT(ch) (((ch)>=DIAL_BASE)&&((ch)=DIAL_PRESS_BASE)&&((ch)=DIAL_RELEASE_BASE)&&((ch)Callbacks[CB_Dials] */ + INVOKE_WCB ( *window,Dials, ( ((int*)enumerator->data)[0], ((int*)enumerator->data)[1]) ); + fgEnumSubWindows ( window, fghcbEnumDialCallbacks, enumerator ); +} + +static void send_dial_event ( int num, int value ) +{ + SFG_Enumerator enumerator; + int data[2]; + data[0] = num; + data[1] = value; + enumerator.found = GL_FALSE; + enumerator.data = data; + fgEnumWindows ( fghcbEnumDialCallbacks, &enumerator ); +} + +/********************************************************************/ +static void poll_dials ( int id ) +{ + int data; + static int dial_state = DIAL_NEW; + static int dial_which; + static int dial_value; + static int dials[8] = { 0, 0, 0, 0, 0, 0, 0, 0 }; + + if ( !dialbox_port ) return; + + while ( (data=serial_getchar(dialbox_port)) != EOF ) + { + if ( ( dial_state > DIAL_WHICH_DEVICE ) || IS_DIAL_EVENT ( data ) ) + { + switch ( dial_state ) + { + case DIAL_WHICH_DEVICE: + dial_which = data - DIAL_BASE; + dial_state++; + break; + case DIAL_VALUE_HIGH: + dial_value = ( data << 8 ); + dial_state++; + break; + case DIAL_VALUE_LOW: + dial_value |= data; + if ( dial_value & 0x8000 ) dial_value -= 0x10000; + dials[dial_which] = dial_value; + send_dial_event ( dial_which + 1, dial_value * 360 / 256 ); + dial_state = DIAL_WHICH_DEVICE; + break; + default: + /* error: Impossible state value! */ + break; + } + } + else if ( data == DIAL_INITIALIZED ) + { + fgState.InputDevsInitialised = GL_TRUE; + dial_state = DIAL_WHICH_DEVICE; + serial_putchar(dialbox_port,DIAL_SET_AUTO_DIALS); + serial_putchar(dialbox_port,0xff); + serial_putchar(dialbox_port,0xff); + } + else /* Unknown data; try flushing. */ + serial_flush(dialbox_port); + } + + glutTimerFunc ( 2, poll_dials, 0 ); +} + + +/******** OS Specific Serial I/O routines *******/ +#if TARGET_HOST_POSIX_X11 /* ==> Linux/BSD/UNIX POSIX serial I/O */ +static SERIALPORT *serial_open ( const char *device ) +{ + int fd; + struct termios termio; + SERIALPORT *port; + + fd = open(device, O_RDWR | O_NONBLOCK ); + if (fd <0) { + perror(device); + return NULL; + } + + port = malloc(sizeof(SERIALPORT)); + memset(port, 0, sizeof(SERIALPORT)); + port->fd = fd; + + /* save current port settings */ + tcgetattr(fd,&port->termio_save); + + memset(&termio, 0, sizeof(termio)); + termio.c_cflag = CS8 | CREAD | HUPCL ; + termio.c_iflag = IGNPAR | IGNBRK ; + termio.c_cc[VTIME] = 0; /* inter-character timer */ + termio.c_cc[VMIN] = 1; /* block read until 1 chars received, when blocking I/O */ + + cfsetispeed(&termio, B9600); + cfsetospeed(&termio, B9600); + tcsetattr(fd,TCSANOW,&termio); + + serial_flush(port); + return port; +} + +static void serial_close(SERIALPORT *port) +{ + if (port) + { + /* restore old port settings */ + tcsetattr(port->fd,TCSANOW,&port->termio_save); + close(port->fd); + free(port); + } +} + +static int serial_getchar(SERIALPORT *port) +{ + unsigned char ch; + if (!port) return EOF; + if (read(port->fd,&ch,1)) return ch; + return EOF; +} + +static int serial_putchar(SERIALPORT *port, unsigned char ch){ + if (!port) return 0; + return write(port->fd,&ch,1); +} + +static void serial_flush ( SERIALPORT *port ) +{ + tcflush ( port->fd, TCIOFLUSH ); +} + +#elif TARGET_HOST_MS_WINDOWS + +static SERIALPORT *serial_open(const char *device){ + HANDLE fh; + DCB dcb={sizeof(DCB)}; + COMMTIMEOUTS timeouts; + SERIALPORT *port; + + fh = CreateFile(device,GENERIC_READ|GENERIC_WRITE,0,NULL, + OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL); + if (!fh) return NULL; + + port = malloc(sizeof(SERIALPORT)); + ZeroMemory(port, sizeof(SERIALPORT)); + port->fh = fh; + + /* save current port settings */ + GetCommState(fh,&port->dcb_save); + GetCommTimeouts(fh,&port->timeouts_save); + + dcb.DCBlength=sizeof(DCB); + BuildCommDCB("96,n,8,1",&dcb); + SetCommState(fh,&dcb); + + ZeroMemory(&timeouts,sizeof(timeouts)); + timeouts.ReadTotalTimeoutConstant=1; + timeouts.WriteTotalTimeoutConstant=1; + SetCommTimeouts(fh,&timeouts); + + serial_flush(port); + + return port; +} + +static void serial_close(SERIALPORT *port){ + if (port){ + /* restore old port settings */ + SetCommState(port->fh,&port->dcb_save); + SetCommTimeouts(port->fh,&port->timeouts_save); + CloseHandle(port->fh); + free(port); + } +} + +static int serial_getchar(SERIALPORT *port){ + DWORD n; + unsigned char ch; + if (!port) return EOF; + if (!ReadFile(port->fh,&ch,1,&n,NULL)) return EOF; + if (n==1) return ch; + return EOF; +} + +static int serial_putchar(SERIALPORT *port, unsigned char ch){ + DWORD n; + if (!port) return 0; + return WriteFile(port->fh,&ch,1,&n,NULL); +} + +static void serial_flush ( SERIALPORT *port ) +{ + FlushFileBuffers(port->fh); +} + +#endif diff --git a/examples/opengl-framework/freeglut/freeglut_internal.h b/examples/opengl-framework/freeglut/freeglut_internal.h new file mode 100644 index 00000000..540f75f5 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_internal.h @@ -0,0 +1,1023 @@ +/* + * freeglut_internal.h + * + * The freeglut library private include file. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 2 1999 + * + * 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 + * PAWEL W. OLSZTA 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 FREEGLUT_INTERNAL_H +#define FREEGLUT_INTERNAL_H + +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + +/* XXX Update these for each release! */ +#define VERSION_MAJOR 2 +#define VERSION_MINOR 7 +#define VERSION_PATCH 0 + +/* Freeglut is intended to function under all Unix/X11 and Win32 platforms. */ +/* XXX: Don't all MS-Windows compilers (except Cygwin) have _WIN32 defined? + * XXX: If so, remove the first set of defined()'s below. + */ +#if !defined(TARGET_HOST_POSIX_X11) && !defined(TARGET_HOST_MS_WINDOWS) && !defined(TARGET_HOST_MAC_OSX) && !defined(TARGET_HOST_SOLARIS) +#if defined(_MSC_VER) || defined(__WATCOMC__) || defined(__MINGW32__) \ + || defined(_WIN32) || defined(_WIN32_WCE) \ + || ( defined(__CYGWIN__) && defined(X_DISPLAY_MISSING) ) +# define TARGET_HOST_MS_WINDOWS 1 + +#elif defined(__posix__) || defined(__unix__) || defined(__linux__) || defined(__sun) +# define TARGET_HOST_POSIX_X11 1 + +#elif defined(__APPLE__) +/* This is a placeholder until we get native OSX support ironed out -- JFF 11/18/09 */ +# define TARGET_HOST_POSIX_X11 1 +/* # define TARGET_HOST_MAC_OSX 1 */ + +#else +# error "Unrecognized target host!" + +#endif +#endif + +/* Detect both SunPro and gcc compilers on Sun Solaris */ +#if defined (__SVR4) && defined (__sun) +# define TARGET_HOST_SOLARIS 1 +#endif + +#ifndef TARGET_HOST_MS_WINDOWS +# define TARGET_HOST_MS_WINDOWS 0 +#endif + +#ifndef TARGET_HOST_POSIX_X11 +# define TARGET_HOST_POSIX_X11 0 +#endif + +#ifndef TARGET_HOST_MAC_OSX +# define TARGET_HOST_MAC_OSX 0 +#endif + +#ifndef TARGET_HOST_SOLARIS +# define TARGET_HOST_SOLARIS 0 +#endif + +/* -- FIXED CONFIGURATION LIMITS ------------------------------------------- */ + +#define FREEGLUT_MAX_MENUS 3 + +/* -- PLATFORM-SPECIFIC INCLUDES ------------------------------------------- */ + +/* All Win32 headers depend on the huge windows.h recursive include. + * Note: Lower-case header names are used, for best cross-platform + * compatibility. + */ +#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) +# include +# include +# include +/* CYGWIN does not have tchar.h, but has TEXT(x), defined in winnt.h. */ +# ifndef __CYGWIN__ +# include +# else +# define _TEXT(x) TEXT(x) +# define _T(x) TEXT(x) +# endif + +#elif TARGET_HOST_POSIX_X11 +# include +# include +# include +# include +# include +# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H +# include +# endif +# ifdef HAVE_X11_EXTENSIONS_XRANDR_H +# include +# endif +/* If GLX is too old, we will fail during runtime when multisampling + is requested, but at least freeglut compiles. */ +# ifndef GLX_SAMPLE_BUFFERS +# define GLX_SAMPLE_BUFFERS 0x80A8 +# endif +# ifndef GLX_SAMPLES +# define GLX_SAMPLES 0x80A9 +# endif + +#endif + +/* These files should be available on every platform. */ +#include +#include +#include +#include +#include + +/* These are included based on autoconf directives. */ +#ifdef HAVE_SYS_TYPES_H +# include +#endif +#ifdef HAVE_UNISTD_H +# include +#endif +#ifdef TIME_WITH_SYS_TIME +# include +# include +#elif defined(HAVE_SYS_TIME_H) +# include +#else +# include +#endif + +/* -- AUTOCONF HACKS --------------------------------------------------------*/ + +/* XXX: Update autoconf to avoid these. + * XXX: Are non-POSIX platforms intended not to use autoconf? + * If so, perhaps there should be a config_guess.h for them. Alternatively, + * config guesses could be placed above, just after the config.h exclusion. + */ +#if defined(__FreeBSD__) || defined(__NetBSD__) +# define HAVE_USB_JS 1 +# if defined(__NetBSD__) || ( defined(__FreeBSD__) && __FreeBSD_version >= 500000) +# define HAVE_USBHID_H 1 +# endif +#endif + +#if TARGET_HOST_MS_WINDOWS +# define HAVE_VFPRINTF 1 +#endif + +/* MinGW may lack a prototype for ChangeDisplaySettingsEx() (depending on the version?) */ +#if TARGET_HOST_MS_WINDOWS && !defined(ChangeDisplaySettingsEx) +LONG WINAPI ChangeDisplaySettingsExA(LPCSTR,LPDEVMODEA,HWND,DWORD,LPVOID); +LONG WINAPI ChangeDisplaySettingsExW(LPCWSTR,LPDEVMODEW,HWND,DWORD,LPVOID); +# ifdef UNICODE +# define ChangeDisplaySettingsEx ChangeDisplaySettingsExW +# else +# define ChangeDisplaySettingsEx ChangeDisplaySettingsExA +# endif +#endif + +#if defined(_MSC_VER) || defined(__WATCOMC__) +/* strdup() is non-standard, for all but POSIX-2001 */ +#define strdup _strdup +#endif + +/* M_PI is non-standard (defined by BSD, not ISO-C) */ +#ifndef M_PI +# define M_PI 3.14159265358979323846 +#endif + +#ifdef HAVE_STDBOOL_H +# include +# ifndef TRUE +# define TRUE true +# endif +# ifndef FALSE +# define FALSE false +# endif +#else +# ifndef TRUE +# define TRUE 1 +# endif +# ifndef FALSE +# define FALSE 0 +# endif +#endif + +/* General defines */ + +#define INVALID_MODIFIERS 0xffffffff + +/* -- GLOBAL TYPE DEFINITIONS ---------------------------------------------- */ + +/* Freeglut callbacks type definitions */ +typedef void (* FGCBDisplay )( void ); +typedef void (* FGCBReshape )( int, int ); +typedef void (* FGCBVisibility )( int ); +typedef void (* FGCBKeyboard )( unsigned char, int, int ); +typedef void (* FGCBSpecial )( int, int, int ); +typedef void (* FGCBMouse )( int, int, int, int ); +typedef void (* FGCBMouseWheel )( int, int, int, int ); +typedef void (* FGCBMotion )( int, int ); +typedef void (* FGCBPassive )( int, int ); +typedef void (* FGCBEntry )( int ); +typedef void (* FGCBWindowStatus )( int ); +typedef void (* FGCBSelect )( int, int, int ); +typedef void (* FGCBJoystick )( unsigned int, int, int, int ); +typedef void (* FGCBKeyboardUp )( unsigned char, int, int ); +typedef void (* FGCBSpecialUp )( int, int, int ); +typedef void (* FGCBOverlayDisplay)( void ); +typedef void (* FGCBSpaceMotion )( int, int, int ); +typedef void (* FGCBSpaceRotation )( int, int, int ); +typedef void (* FGCBSpaceButton )( int, int ); +typedef void (* FGCBDials )( int, int ); +typedef void (* FGCBButtonBox )( int, int ); +typedef void (* FGCBTabletMotion )( int, int ); +typedef void (* FGCBTabletButton )( int, int, int, int ); +typedef void (* FGCBDestroy )( void ); + +typedef void (* FGCBMultiEntry )( int, int ); +typedef void (* FGCBMultiButton )( int, int, int, int, int ); +typedef void (* FGCBMultiMotion )( int, int, int ); +typedef void (* FGCBMultiPassive )( int, int, int ); + +/* The global callbacks type definitions */ +typedef void (* FGCBIdle )( void ); +typedef void (* FGCBTimer )( int ); +typedef void (* FGCBMenuState )( int ); +typedef void (* FGCBMenuStatus )( int, int, int ); + +/* The callback used when creating/using menus */ +typedef void (* FGCBMenu )( int ); + +/* The FreeGLUT error/warning handler type definition */ +typedef void (* FGError ) ( const char *fmt, va_list ap); +typedef void (* FGWarning ) ( const char *fmt, va_list ap); + + +/* A list structure */ +typedef struct tagSFG_List SFG_List; +struct tagSFG_List +{ + void *First; + void *Last; +}; + +/* A list node structure */ +typedef struct tagSFG_Node SFG_Node; +struct tagSFG_Node +{ + void *Next; + void *Prev; +}; + +/* A helper structure holding two ints and a boolean */ +typedef struct tagSFG_XYUse SFG_XYUse; +struct tagSFG_XYUse +{ + GLint X, Y; /* The two integers... */ + GLboolean Use; /* ...and a single boolean. */ +}; + +/* + * An enumeration containing the state of the GLUT execution: + * initializing, running, or stopping + */ +typedef enum +{ + GLUT_EXEC_STATE_INIT, + GLUT_EXEC_STATE_RUNNING, + GLUT_EXEC_STATE_STOP +} fgExecutionState ; + +/* This structure holds different freeglut settings */ +typedef struct tagSFG_State SFG_State; +struct tagSFG_State +{ + SFG_XYUse Position; /* The default windows' position */ + SFG_XYUse Size; /* The default windows' size */ + unsigned int DisplayMode; /* Display mode for new windows */ + + GLboolean Initialised; /* freeglut has been initialised */ + + int DirectContext; /* Direct rendering state */ + + GLboolean ForceIconic; /* New top windows are iconified */ + GLboolean UseCurrentContext; /* New windows share with current */ + + GLboolean GLDebugSwitch; /* OpenGL state debugging switch */ + GLboolean XSyncSwitch; /* X11 sync protocol switch */ + + int KeyRepeat; /* Global key repeat mode. */ + int Modifiers; /* Current ALT/SHIFT/CTRL state */ + + GLuint FPSInterval; /* Interval between FPS printfs */ + GLuint SwapCount; /* Count of glutSwapBuffer calls */ + GLuint SwapTime; /* Time of last SwapBuffers */ + + unsigned long Time; /* Time that glutInit was called */ + SFG_List Timers; /* The freeglut timer hooks */ + SFG_List FreeTimers; /* The unused timer hooks */ + + FGCBIdle IdleCallback; /* The global idle callback */ + + int ActiveMenus; /* Num. of currently active menus */ + FGCBMenuState MenuStateCallback; /* Menu callbacks are global */ + FGCBMenuStatus MenuStatusCallback; + + SFG_XYUse GameModeSize; /* Game mode screen's dimensions */ + int GameModeDepth; /* The pixel depth for game mode */ + int GameModeRefresh; /* The refresh rate for game mode */ + + int ActionOnWindowClose; /* Action when user closes window */ + + fgExecutionState ExecState; /* Used for GLUT termination */ + char *ProgramName; /* Name of the invoking program */ + GLboolean JoysticksInitialised; /* Only initialize if application calls for them */ + int NumActiveJoysticks; /* Number of active joysticks -- if zero, don't poll joysticks */ + GLboolean InputDevsInitialised; /* Only initialize if application calls for them */ + + int MouseWheelTicks; /* Number of ticks the mouse wheel has turned */ + + int AuxiliaryBufferNumber; /* Number of auxiliary buffers */ + int SampleNumber; /* Number of samples per pixel */ + + int MajorVersion; /* Major OpenGL context version */ + int MinorVersion; /* Minor OpenGL context version */ + int ContextFlags; /* OpenGL context flags */ + int ContextProfile; /* OpenGL context profile */ + FGError ErrorFunc; /* User defined error handler */ + FGWarning WarningFunc; /* User defined warning handler */ +}; + +/* The structure used by display initialization in freeglut_init.c */ +typedef struct tagSFG_Display SFG_Display; +struct tagSFG_Display +{ +#if TARGET_HOST_POSIX_X11 + Display* Display; /* The display we are being run in. */ + int Screen; /* The screen we are about to use. */ + Window RootWindow; /* The screen's root window. */ + int Connection; /* The display's connection number */ + Atom DeleteWindow; /* The window deletion atom */ + Atom State; /* The state atom */ + Atom StateFullScreen; /* The full screen atom */ + +#ifdef HAVE_X11_EXTENSIONS_XRANDR_H + int prev_xsz, prev_ysz; + int prev_refresh; + int prev_size_valid; +#endif /* HAVE_X11_EXTENSIONS_XRANDR_H */ + +#ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H + /* + * XF86VidMode may be compilable even if it fails at runtime. Therefore, + * the validity of the VidMode has to be tracked + */ + int DisplayModeValid; /* Flag that indicates runtime status*/ + XF86VidModeModeLine DisplayMode; /* Current screen's display settings */ + int DisplayModeClock; /* The display mode's refresh rate */ + int DisplayViewPortX; /* saved X location of the viewport */ + int DisplayViewPortY; /* saved Y location of the viewport */ +#endif /* HAVE_X11_EXTENSIONS_XF86VMODE_H */ + + int DisplayPointerX; /* saved X location of the pointer */ + int DisplayPointerY; /* saved Y location of the pointer */ + +#elif TARGET_HOST_MS_WINDOWS + HINSTANCE Instance; /* The application's instance */ + DEVMODE DisplayMode; /* Desktop's display settings */ + char *DisplayName; /* Display name for multi display support*/ + +#endif + + int ScreenWidth; /* The screen's width in pixels */ + int ScreenHeight; /* The screen's height in pixels */ + int ScreenWidthMM; /* The screen's width in milimeters */ + int ScreenHeightMM; /* The screen's height in milimeters */ +}; + + +/* The user can create any number of timer hooks */ +typedef struct tagSFG_Timer SFG_Timer; +struct tagSFG_Timer +{ + SFG_Node Node; + int ID; /* The timer ID integer */ + FGCBTimer Callback; /* The timer callback */ + long TriggerTime; /* The timer trigger time */ +}; + +/* + * Make "freeglut" window handle and context types so that we don't need so + * much conditionally-compiled code later in the library. + */ +#if TARGET_HOST_POSIX_X11 + +typedef Window SFG_WindowHandleType ; +typedef GLXContext SFG_WindowContextType ; + +#elif TARGET_HOST_MS_WINDOWS + +typedef HWND SFG_WindowHandleType ; +typedef HGLRC SFG_WindowContextType ; + +#endif + +/* + * A window and its OpenGL context. The contents of this structure + * are highly dependant on the target operating system we aim at... + */ +typedef struct tagSFG_Context SFG_Context; +struct tagSFG_Context +{ + SFG_WindowHandleType Handle; /* The window's handle */ + SFG_WindowContextType Context; /* The window's OpenGL/WGL context */ + +#if TARGET_HOST_POSIX_X11 + GLXFBConfig* FBConfig; /* The window's FBConfig */ +#elif TARGET_HOST_MS_WINDOWS + HDC Device; /* The window's device context */ +#endif + + int DoubleBuffered; /* Treat the window as double-buffered */ +}; + +/* Window's state description. This structure should be kept portable. */ +typedef struct tagSFG_WindowState SFG_WindowState; +struct tagSFG_WindowState +{ + /* Note that on Windows, sizes always refer to the client area, thus without the window decorations */ + int Width; /* Window's width in pixels */ + int Height; /* The same about the height */ +#if TARGET_HOST_POSIX_X11 + int OldWidth; /* Window width from before a resize */ + int OldHeight; /* " height " " " " */ +#elif TARGET_HOST_MS_WINDOWS + RECT OldRect; /* window rect - stored before the window is made fullscreen */ + DWORD OldStyle; /* window style - stored before the window is made fullscreen */ +#endif + + GLboolean Redisplay; /* Do we have to redisplay? */ + GLboolean Visible; /* Is the window visible now */ + + int Cursor; /* The currently selected cursor */ + + long JoystickPollRate; /* The joystick polling rate */ + long JoystickLastPoll; /* When the last poll happened */ + + int MouseX, MouseY; /* The most recent mouse position */ + + GLboolean IgnoreKeyRepeat; /* Whether to ignore key repeat. */ + GLboolean KeyRepeating; /* Currently in repeat mode */ + + GLboolean NeedToResize; /* Do we need to resize the window? */ + + GLboolean IsFullscreen; /* is the window fullscreen? */ +}; + + +/* + * A generic function pointer. We should really use the GLUTproc type + * defined in freeglut_ext.h, but if we include that header in this file + * a bunch of other stuff (font-related) blows up! + */ +typedef void (*SFG_Proc)(); + + +/* + * SET_WCB() is used as: + * + * SET_WCB( window, cbname, func ); + * + * ...where {window} is the freeglut window to set the callback, + * {cbname} is the window-specific callback to set, + * {func} is a function-pointer. + * + * Originally, {FETCH_WCB( ... ) = func} was rather sloppily used, + * but this can cause warnings because the FETCH_WCB() macro type- + * casts its result, and a type-cast value shouldn't be an lvalue. + * + * The {if( FETCH_WCB( ... ) != func )} test is to do type-checking + * and for no other reason. Since it's hidden in the macro, the + * ugliness is felt to be rather benign. + */ +#define SET_WCB(window,cbname,func) \ +do \ +{ \ + if( FETCH_WCB( window, cbname ) != (SFG_Proc)(func) ) \ + (((window).CallBacks[CB_ ## cbname]) = (SFG_Proc)(func)); \ +} while( 0 ) + +/* + * FETCH_WCB() is used as: + * + * FETCH_WCB( window, cbname ); + * + * ...where {window} is the freeglut window to fetch the callback from, + * {cbname} is the window-specific callback to fetch. + * + * The result is correctly type-cast to the callback function pointer + * type. + */ +#define FETCH_WCB(window,cbname) \ + ((window).CallBacks[CB_ ## cbname]) + +/* + * INVOKE_WCB() is used as: + * + * INVOKE_WCB( window, cbname, ( arg_list ) ); + * + * ...where {window} is the freeglut window, + * {cbname} is the window-specific callback to be invoked, + * {(arg_list)} is the parameter list. + * + * The callback is invoked as: + * + * callback( arg_list ); + * + * ...so the parentheses are REQUIRED in the {arg_list}. + * + * NOTE that it does a sanity-check and also sets the + * current window. + * + */ +#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) /* FIXME: also WinCE? */ +#define INVOKE_WCB(window,cbname,arg_list) \ +do \ +{ \ + if( FETCH_WCB( window, cbname ) ) \ + { \ + FGCB ## cbname func = (FGCB ## cbname)(FETCH_WCB( window, cbname )); \ + fgSetWindow( &window ); \ + func arg_list; \ + } \ +} while( 0 ) +#else +#define INVOKE_WCB(window,cbname,arg_list) \ +do \ +{ \ + if( FETCH_WCB( window, cbname ) ) \ + { \ + fgSetWindow( &window ); \ + ((FGCB ## cbname)FETCH_WCB( window, cbname )) arg_list; \ + } \ +} while( 0 ) +#endif + +/* + * The window callbacks the user can supply us with. Should be kept portable. + * + * This enumeration provides the freeglut CallBack numbers. + * The symbolic constants are indices into a window's array of + * function callbacks. The names are formed by splicing a common + * prefix onto the callback's base name. (This was originally + * done so that an early stage of development could live side-by- + * side with the old callback code. The old callback code used + * the bare callback's name as a structure member, so I used a + * prefix for the array index name.) + * + * XXX For consistancy, perhaps the prefix should match the + * XXX FETCH* and INVOKE* macro suffices. I.e., WCB_, rather than + * XXX CB_. + */ +enum +{ + CB_Display, + CB_Reshape, + CB_Keyboard, + CB_KeyboardUp, + CB_Special, + CB_SpecialUp, + CB_Mouse, + CB_MouseWheel, + CB_Motion, + CB_Passive, + CB_Entry, + CB_Visibility, + CB_WindowStatus, + CB_Joystick, + CB_Destroy, + + /* MPX-related */ + CB_MultiEntry, + CB_MultiButton, + CB_MultiMotion, + CB_MultiPassive, + + /* Presently ignored */ + CB_Select, + CB_OverlayDisplay, + CB_SpaceMotion, /* presently implemented only on UNIX/X11 */ + CB_SpaceRotation, /* presently implemented only on UNIX/X11 */ + CB_SpaceButton, /* presently implemented only on UNIX/X11 */ + CB_Dials, + CB_ButtonBox, + CB_TabletMotion, + CB_TabletButton, + + /* Always make this the LAST one */ + TOTAL_CALLBACKS +}; + + +/* This structure holds the OpenGL rendering context for all the menu windows */ +typedef struct tagSFG_MenuContext SFG_MenuContext; +struct tagSFG_MenuContext +{ + SFG_WindowContextType MContext; /* The menu window's WGL context */ +}; + +/* This structure describes a menu */ +typedef struct tagSFG_Window SFG_Window; +typedef struct tagSFG_MenuEntry SFG_MenuEntry; +typedef struct tagSFG_Menu SFG_Menu; +struct tagSFG_Menu +{ + SFG_Node Node; + void *UserData; /* User data passed back at callback */ + int ID; /* The global menu ID */ + SFG_List Entries; /* The menu entries list */ + FGCBMenu Callback; /* The menu callback */ + FGCBDestroy Destroy; /* Destruction callback */ + GLboolean IsActive; /* Is the menu selected? */ + int Width; /* Menu box width in pixels */ + int Height; /* Menu box height in pixels */ + int X, Y; /* Menu box raster position */ + + SFG_MenuEntry *ActiveEntry; /* Currently active entry in the menu */ + SFG_Window *Window; /* Window for menu */ + SFG_Window *ParentWindow; /* Window in which the menu is invoked */ +}; + +/* This is a menu entry */ +struct tagSFG_MenuEntry +{ + SFG_Node Node; + int ID; /* The menu entry ID (local) */ + int Ordinal; /* The menu's ordinal number */ + char* Text; /* The text to be displayed */ + SFG_Menu* SubMenu; /* Optional sub-menu tree */ + GLboolean IsActive; /* Is the entry highlighted? */ + int Width; /* Label's width in pixels */ +}; + +/* + * A window, making part of freeglut windows hierarchy. + * Should be kept portable. + * + * NOTE that ActiveMenu is set to menu itself if the window is a menu. + */ +struct tagSFG_Window +{ + SFG_Node Node; + int ID; /* Window's ID number */ + + SFG_Context Window; /* Window and OpenGL context */ + SFG_WindowState State; /* The window state */ + SFG_Proc CallBacks[ TOTAL_CALLBACKS ]; /* Array of window callbacks */ + void *UserData ; /* For use by user */ + + SFG_Menu* Menu[ FREEGLUT_MAX_MENUS ]; /* Menus appended to window */ + SFG_Menu* ActiveMenu; /* The window's active menu */ + + SFG_Window* Parent; /* The parent to this window */ + SFG_List Children; /* The subwindows d.l. list */ + + GLboolean IsMenu; /* Set to 1 if we are a menu */ +}; + + +/* A linked list structure of windows */ +typedef struct tagSFG_WindowList SFG_WindowList ; +struct tagSFG_WindowList +{ + SFG_Node node; + SFG_Window *window ; +}; + +/* This holds information about all the windows, menus etc. */ +typedef struct tagSFG_Structure SFG_Structure; +struct tagSFG_Structure +{ + SFG_List Windows; /* The global windows list */ + SFG_List Menus; /* The global menus list */ + SFG_List WindowsToDestroy; + + SFG_Window* CurrentWindow; /* The currently set window */ + SFG_Menu* CurrentMenu; /* Same, but menu... */ + + SFG_MenuContext* MenuContext; /* OpenGL rendering context for menus */ + + SFG_Window* GameModeWindow; /* The game mode window */ + + int WindowID; /* The new current window ID */ + int MenuID; /* The new current menu ID */ +}; + +/* + * This structure is used for the enumeration purposes. + * You can easily extend its functionalities by declaring + * a structure containing enumerator's contents and custom + * data, then casting its pointer to (SFG_Enumerator *). + */ +typedef struct tagSFG_Enumerator SFG_Enumerator; +struct tagSFG_Enumerator +{ + GLboolean found; /* Used to terminate search */ + void* data; /* Custom data pointer */ +}; +typedef void (* FGCBenumerator )( SFG_Window *, SFG_Enumerator * ); + +/* The bitmap font structure */ +typedef struct tagSFG_Font SFG_Font; +struct tagSFG_Font +{ + char* Name; /* The source font name */ + int Quantity; /* Number of chars in font */ + int Height; /* Height of the characters */ + const GLubyte** Characters; /* The characters mapping */ + + float xorig, yorig; /* Relative origin of the character */ +}; + +/* The stroke font structures */ + +typedef struct tagSFG_StrokeVertex SFG_StrokeVertex; +struct tagSFG_StrokeVertex +{ + GLfloat X, Y; +}; + +typedef struct tagSFG_StrokeStrip SFG_StrokeStrip; +struct tagSFG_StrokeStrip +{ + int Number; + const SFG_StrokeVertex* Vertices; +}; + +typedef struct tagSFG_StrokeChar SFG_StrokeChar; +struct tagSFG_StrokeChar +{ + GLfloat Right; + int Number; + const SFG_StrokeStrip* Strips; +}; + +typedef struct tagSFG_StrokeFont SFG_StrokeFont; +struct tagSFG_StrokeFont +{ + char* Name; /* The source font name */ + int Quantity; /* Number of chars in font */ + GLfloat Height; /* Height of the characters */ + const SFG_StrokeChar** Characters; /* The characters mapping */ +}; + +/* -- GLOBAL VARIABLES EXPORTS --------------------------------------------- */ + +/* Freeglut display related stuff (initialized once per session) */ +extern SFG_Display fgDisplay; + +/* Freeglut internal structure */ +extern SFG_Structure fgStructure; + +/* The current freeglut settings */ +extern SFG_State fgState; + + +/* -- PRIVATE FUNCTION DECLARATIONS ---------------------------------------- */ + +/* + * A call to this function makes us sure that the Display and Structure + * subsystems have been properly initialized and are ready to be used + */ +#define FREEGLUT_EXIT_IF_NOT_INITIALISED( string ) \ + if ( ! fgState.Initialised ) \ + { \ + fgError ( " ERROR: Function <%s> called" \ + " without first calling 'glutInit'.", (string) ) ; \ + } + +#define FREEGLUT_INTERNAL_ERROR_EXIT_IF_NOT_INITIALISED( string ) \ + if ( ! fgState.Initialised ) \ + { \ + fgError ( " ERROR: Internal <%s> function called" \ + " without first calling 'glutInit'.", (string) ) ; \ + } + +#define FREEGLUT_INTERNAL_ERROR_EXIT( cond, string, function ) \ + if ( ! ( cond ) ) \ + { \ + fgError ( " ERROR: Internal error <%s> in function %s", \ + (string), (function) ) ; \ + } + +/* + * Following definitions are somewhat similiar to GLib's, + * but do not generate any log messages: + */ +#define freeglut_return_if_fail( expr ) \ + if( !(expr) ) \ + return; +#define freeglut_return_val_if_fail( expr, val ) \ + if( !(expr) ) \ + return val ; + +/* + * A call to those macros assures us that there is a current + * window set, respectively: + */ +#define FREEGLUT_EXIT_IF_NO_WINDOW( string ) \ + if ( ! fgStructure.CurrentWindow && \ + ( fgState.ActionOnWindowClose != GLUT_ACTION_CONTINUE_EXECUTION ) ) \ + { \ + fgError ( " ERROR: Function <%s> called" \ + " with no current window defined.", (string) ) ; \ + } + +/* + * The deinitialize function gets called on glutMainLoop() end. It should clean up + * everything inside of the freeglut + */ +void fgDeinitialize( void ); + +/* + * Those two functions are used to create/destroy the freeglut internal + * structures. This actually happens when calling glutInit() and when + * quitting the glutMainLoop() (which actually happens, when all windows + * have been closed). + */ +void fgCreateStructure( void ); +void fgDestroyStructure( void ); + +/* A helper function to check if a display mode is possible to use */ +#if TARGET_HOST_POSIX_X11 +GLXFBConfig* fgChooseFBConfig( int* numcfgs ); +#endif + +/* The window procedure for Win32 events handling */ +#if TARGET_HOST_MS_WINDOWS +LRESULT CALLBACK fgWindowProc( HWND hWnd, UINT uMsg, + WPARAM wParam, LPARAM lParam ); +void fgNewWGLCreateContext( SFG_Window* window ); +GLboolean fgSetupPixelFormat( SFG_Window* window, GLboolean checkOnly, + unsigned char layer_type ); +#endif + +/* + * Window creation, opening, closing and destruction. + * Also CallBack clearing/initialization. + * Defined in freeglut_structure.c, freeglut_window.c. + */ +SFG_Window* fgCreateWindow( SFG_Window* parent, const char* title, + GLboolean positionUse, int x, int y, + GLboolean sizeUse, int w, int h, + GLboolean gameMode, GLboolean isMenu ); +void fgSetWindow ( SFG_Window *window ); +void fgOpenWindow( SFG_Window* window, const char* title, + GLboolean positionUse, int x, int y, + GLboolean sizeUse, int w, int h, + GLboolean gameMode, GLboolean isSubWindow ); +void fgCloseWindow( SFG_Window* window ); +void fgAddToWindowDestroyList ( SFG_Window* window ); +void fgCloseWindows (); +void fgDestroyWindow( SFG_Window* window ); + +/* Menu creation and destruction. Defined in freeglut_structure.c */ +SFG_Menu* fgCreateMenu( FGCBMenu menuCallback ); +void fgDestroyMenu( SFG_Menu* menu ); + +/* Joystick device management functions, defined in freeglut_joystick.c */ +int fgJoystickDetect( void ); +void fgInitialiseJoysticks( void ); +void fgJoystickClose( void ); +void fgJoystickPollWindow( SFG_Window* window ); + +/* InputDevice Initialisation and Closure */ +int fgInputDeviceDetect( void ); +void fgInitialiseInputDevices( void ); +void fgInputDeviceClose( void ); + +/* spaceball device functions, defined in freeglut_spaceball.c */ +void fgInitialiseSpaceball( void ); +void fgSpaceballClose( void ); +void fgSpaceballSetWindow( SFG_Window *window ); + +int fgHasSpaceball( void ); +int fgSpaceballNumButtons( void ); + +#if TARGET_HOST_POSIX_X11 +int fgIsSpaceballXEvent( const XEvent *ev ); +void fgSpaceballHandleXEvent( const XEvent *ev ); +#endif + +/* Setting the cursor for a given window */ +void fgSetCursor ( SFG_Window *window, int cursorID ); + +/* + * Helper function to enumerate through all registered windows + * and one to enumerate all of a window's subwindows... + * + * The GFunc callback for those functions will be defined as: + * + * void enumCallback( gpointer window, gpointer enumerator ); + * + * where window is the enumerated (sub)window pointer (SFG_Window *), + * and userData is the a custom user-supplied pointer. Functions + * are defined and exported from freeglut_structure.c file. + */ +void fgEnumWindows( FGCBenumerator enumCallback, SFG_Enumerator* enumerator ); +void fgEnumSubWindows( SFG_Window* window, FGCBenumerator enumCallback, + SFG_Enumerator* enumerator ); + +#if TARGET_HOST_MS_WINDOWS +/* + * Helper functions for getting client area from the window rect + * and the window rect from the client area given the style of the window + * (or a valid window pointer from which the style can be queried). + */ +void fghComputeWindowRectFromClientArea_UseStyle ( const DWORD windowStyle , RECT *clientRect, BOOL posIsOutside ); +void fghComputeWindowRectFromClientArea_QueryWindow( const SFG_Window *window, RECT *clientRect, BOOL posIsOutside ); +void fghComputeClientAreaFromWindowRect ( const SFG_Window *window, RECT *windowRect, BOOL wantPosOutside ); +RECT fghGetClientArea ( const SFG_Window *window, BOOL wantPosOutside ); +void fghGetBorderWidth(const DWORD windowStyle, int* xBorderWidth, int* yBorderWidth); +#endif + +/* + * fgWindowByHandle returns a (SFG_Window *) value pointing to the + * first window in the queue matching the specified window handle. + * The function is defined in freeglut_structure.c file. + */ +SFG_Window* fgWindowByHandle( SFG_WindowHandleType hWindow ); + +/* + * This function is similiar to the previous one, except it is + * looking for a specified (sub)window identifier. The function + * is defined in freeglut_structure.c file. + */ +SFG_Window* fgWindowByID( int windowID ); + +/* + * Looks up a menu given its ID. This is easier than fgWindowByXXX + * as all menus are placed in a single doubly linked list... + */ +SFG_Menu* fgMenuByID( int menuID ); + +/* + * The menu activation and deactivation the code. This is the meat + * of the menu user interface handling code... + */ +void fgUpdateMenuHighlight ( SFG_Menu *menu ); +GLboolean fgCheckActiveMenu ( SFG_Window *window, int button, GLboolean pressed, + int mouse_x, int mouse_y ); +void fgDeactivateMenu( SFG_Window *window ); + +/* + * This function gets called just before the buffers swap, so that + * freeglut can display the pull-down menus via OpenGL. The function + * is defined in freeglut_menu.c file. + */ +void fgDisplayMenu( void ); + +/* Elapsed time as per glutGet(GLUT_ELAPSED_TIME). */ +long fgElapsedTime( void ); + +/* System time in milliseconds */ +long unsigned fgSystemTime(void); + +/* List functions */ +void fgListInit(SFG_List *list); +void fgListAppend(SFG_List *list, SFG_Node *node); +void fgListRemove(SFG_List *list, SFG_Node *node); +int fgListLength(SFG_List *list); +void fgListInsert(SFG_List *list, SFG_Node *next, SFG_Node *node); + +/* Error Message functions */ +void fgError( const char *fmt, ... ); +void fgWarning( const char *fmt, ... ); + +/* + * Check if "hint" is present in "property" for "window". See freeglut_init.c + */ +#if TARGET_HOST_POSIX_X11 +int fgHintPresent(Window window, Atom property, Atom hint); + +/* Handler for X extension Events */ +#ifdef HAVE_X11_EXTENSIONS_XINPUT2_H + void fgHandleExtensionEvents( XEvent * ev ); + void fgRegisterDevices( Display* dpy, Window* win ); +#endif + +#endif + +SFG_Proc fghGetProcAddress( const char *procName ); + +#if TARGET_HOST_MS_WINDOWS +extern void (__cdecl *__glutExitFunc)( int return_value ); +#endif + +#endif /* FREEGLUT_INTERNAL_H */ + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_joystick.c b/examples/opengl-framework/freeglut/freeglut_joystick.c new file mode 100644 index 00000000..f48dc3f2 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_joystick.c @@ -0,0 +1,1800 @@ +/* + * freeglut_joystick.c + * + * Joystick handling code + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Steve Baker, + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +/* + * FreeBSD port by Stephen Montgomery-Smith + * + * Redone by John Fay 2/4/04 with another look from the PLIB "js" library. + * Many thanks for Steve Baker for permission to pull from that library. + */ + +#include +#include "freeglut_internal.h" +#ifdef HAVE_SYS_PARAM_H +# include +#endif + +/* + * Initial defines from "js.h" starting around line 33 with the existing "freeglut_joystick.c" + * interspersed + */ + +/* XXX It might be better to poll the operating system for the numbers of buttons and + * XXX axes and then dynamically allocate the arrays. + */ +#define _JS_MAX_BUTTONS 32 + +#if TARGET_HOST_MACINTOSH +# define _JS_MAX_AXES 9 +# include +#endif + +#if TARGET_HOST_MAC_OSX +# define _JS_MAX_AXES 16 +# include +# include +# include +#endif + +#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) +# define _JS_MAX_AXES 8 +# include +# include +# include + +#endif + +#if TARGET_HOST_POSIX_X11 +# define _JS_MAX_AXES 16 +# ifdef HAVE_SYS_IOCTL_H +# include +# endif +# ifdef HAVE_FCNTL_H +# include +# endif +# ifdef HAVE_ERRNO_H +# include +# include +# endif +# if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__NetBSD__) +/* XXX The below hack is done until freeglut's autoconf is updated. */ +# define HAVE_USB_JS 1 + +# if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) +# include +# else +/* + * XXX NetBSD/amd64 systems may find that they have to steal the + * XXX /usr/include/machine/joystick.h from a NetBSD/i386 system. + * XXX I cannot comment whether that works for the interface, but + * XXX it lets you compile...(^& I do not think that we can do away + * XXX with this header. + */ +# include /* For analog joysticks */ +# endif +# define JS_DATA_TYPE joystick +# define JS_RETURN (sizeof(struct JS_DATA_TYPE)) +# endif + +# if defined(__linux__) +# include + +/* check the joystick driver version */ +# if defined(JS_VERSION) && JS_VERSION >= 0x010000 +# define JS_NEW +# endif +# else /* Not BSD or Linux */ +# ifndef JS_RETURN + + /* + * We'll put these values in and that should + * allow the code to at least compile when there is + * no support. The JS open routine should error out + * and shut off all the code downstream anyway and if + * the application doesn't use a joystick we'll be fine. + */ + + struct JS_DATA_TYPE + { + int buttons; + int x; + int y; + }; + +# define JS_RETURN (sizeof(struct JS_DATA_TYPE)) +# endif +# endif +#endif + +#define JS_TRUE 1 +#define JS_FALSE 0 + +/* BSD defines from "jsBSD.cxx" around lines 42-270 */ + +#if defined(__NetBSD__) || defined(__FreeBSD__) || defined(__FreeBSD_kernel__) + +# ifdef HAVE_USB_JS +# if defined(__NetBSD__) +/* XXX The below hack is done until freeglut's autoconf is updated. */ +# define HAVE_USBHID_H 1 +# ifdef HAVE_USBHID_H +# include +# else +# include +# endif +# elif defined(__FreeBSD__) || defined(__FreeBSD_kernel__) +# ifdef HAVE_USBHID_H +# include +# else +# include +# endif +# endif +# include +# include + +/* Compatibility with older usb.h revisions */ +# if !defined(USB_MAX_DEVNAMES) && defined(MAXDEVNAMES) +# define USB_MAX_DEVNAMES MAXDEVNAMES +# endif +# endif + +static int hatmap_x[9] = { 0, 0, 1, 1, 1, 0, -1, -1, -1 }; +static int hatmap_y[9] = { 0, 1, 1, 0, -1, -1, -1, 0, 1 }; +struct os_specific_s { + char fname [128 ]; + int fd; + int is_analog; + /* The following structure members are specific to analog joysticks */ + struct joystick ajs; +# ifdef HAVE_USB_JS + /* The following structure members are specific to USB joysticks */ + struct hid_item *hids; + int hid_dlen; + int hid_offset; + char *hid_data_buf; + int axes_usage [ _JS_MAX_AXES ]; +# endif + /* We keep button and axes state ourselves, as they might not be updated + * on every read of a USB device + */ + int cache_buttons; + float cache_axes [ _JS_MAX_AXES ]; +}; + +/* Idents lower than USB_IDENT_OFFSET are for analog joysticks. */ +# define USB_IDENT_OFFSET 2 + +# define USBDEV "/dev/usb" +# define UHIDDEV "/dev/uhid" +# define AJSDEV "/dev/joy" + +# ifdef HAVE_USB_JS +/* + * fghJoystickFindUSBdev (and its helper, fghJoystickWalkUSBdev) try to locate + * the full name of a USB device. If /dev/usbN isn't readable, we punt and + * return the uhidN device name. We warn the user of this situation once. + */ +static char *fghJoystickWalkUSBdev(int f, char *dev, char *out, int outlen) +{ + struct usb_device_info di; + int i, a; + char *cp; + + for (a = 1; a < USB_MAX_DEVICES; a++) { + di.udi_addr = a; + if (ioctl(f, USB_DEVICEINFO, &di) != 0) + return NULL; + for (i = 0; i < USB_MAX_DEVNAMES; i++) + if (di.udi_devnames[i][0] && + strcmp(di.udi_devnames[i], dev) == 0) { + cp = calloc( 1, strlen(di.udi_vendor) + strlen(di.udi_product) + 2); + strcpy(cp, di.udi_vendor); + strcat(cp, " "); + strcat(cp, di.udi_product); + strncpy(out, cp, outlen - 1); + out[outlen - 1] = 0; + free( cp ); + return out; + } + } + return NULL; +} + +static int fghJoystickFindUSBdev(char *name, char *out, int outlen) +{ + int i, f; + char buf[50]; + char *cp; + static int protection_warned = 0; + + for (i = 0; i < 16; i++) { + snprintf(buf, sizeof(buf), "%s%d", USBDEV, i); + f = open(buf, O_RDONLY); + if (f >= 0) { + cp = fghJoystickWalkUSBdev(f, name, out, outlen); + close(f); + if (cp) + return 1; + } +#ifdef HAVE_ERRNO_H + else if (errno == EACCES) { + if (!protection_warned) { + fgWarning ( "Can't open %s for read!", buf ); + protection_warned = 1; + } + } +#endif + } + return 0; +} + +static int fghJoystickInitializeHID(struct os_specific_s *os, + int *num_axes, int *num_buttons) +{ + int size, is_joystick; +# ifdef HAVE_USBHID_H + int report_id = 0; +# endif + struct hid_data *d; + struct hid_item h; + report_desc_t rd; + + if ( ( rd = hid_get_report_desc( os->fd ) ) == 0 ) + { +#ifdef HAVE_ERRNO_H + fgWarning ( "error: %s: %s", os->fname, strerror( errno ) ); +#else + fgWarning ( "error: %s", os->fname ); +#endif + return FALSE; + } + + os->hids = NULL; + +# ifdef HAVE_USBHID_H + if( ioctl( os->fd, USB_GET_REPORT_ID, &report_id ) < 0) + { + /*** XXX {report_id} may not be the right variable? ***/ +#ifdef HAVE_ERRNO_H + fgWarning ( "error: %s%d: %s", UHIDDEV, report_id, strerror( errno ) ); +#else + fgWarning ( "error: %s%d", UHIDDEV, report_id ); +#endif + return FALSE; + } + + size = hid_report_size( rd, hid_input, report_id ); +# else + size = hid_report_size( rd, 0, hid_input ); +# endif + os->hid_data_buf = calloc( 1, size ); + os->hid_dlen = size; + + is_joystick = 0; +# ifdef HAVE_USBHID_H + d = hid_start_parse( rd, 1 << hid_input, report_id ); +# else + d = hid_start_parse( rd, 1 << hid_input ); +# endif + while( hid_get_item( d, &h ) ) + { + int usage, page, interesting_hid; + + page = HID_PAGE( h.usage ); + usage = HID_USAGE( h.usage ); + + /* This test is somewhat too simplistic, but this is how MicroSoft + * does, so I guess it works for all joysticks/game pads. */ + is_joystick = is_joystick || + ( h.kind == hid_collection && + page == HUP_GENERIC_DESKTOP && + ( usage == HUG_JOYSTICK || usage == HUG_GAME_PAD ) ); + + if( h.kind != hid_input ) + continue; + + if( !is_joystick ) + continue; + + interesting_hid = TRUE; + if( page == HUP_GENERIC_DESKTOP ) + { + switch( usage ) + { + case HUG_X: + case HUG_RX: + case HUG_Y: + case HUG_RY: + case HUG_Z: + case HUG_RZ: + case HUG_SLIDER: + if( *num_axes < _JS_MAX_AXES ) + { + os->axes_usage[ *num_axes ] = usage; + ( *num_axes )++; + } + break; + case HUG_HAT_SWITCH: + /* Allocate two axes for a hat */ + if( *num_axes + 1 < _JS_MAX_AXES ) + { + os->axes_usage[ *num_axes ] = usage; + (*num_axes)++; + os->axes_usage[ *num_axes ] = usage; + (*num_axes)++; + } + break; + default: + interesting_hid = FALSE; + break; + } + } + else if( page == HUP_BUTTON ) + { + interesting_hid = ( usage > 0 ) && + ( usage <= _JS_MAX_BUTTONS ); + + if( interesting_hid && usage - 1 > *num_buttons ) + *num_buttons = usage - 1; + } + + if( interesting_hid ) + { + h.next = os->hids; + os->hids = calloc( 1, sizeof ( struct hid_item ) ); + *os->hids = h; + } + } + hid_end_parse( d ); + + return os->hids != NULL; +} +# endif +#endif + +/* + * Definition of "SFG_Joystick" structure -- based on JS's "jsJoystick" object class. + * See "js.h" lines 80-178. + */ +typedef struct tagSFG_Joystick SFG_Joystick; +struct tagSFG_Joystick +{ +#if TARGET_HOST_MACINTOSH +#define ISP_NUM_AXIS 9 +#define ISP_NUM_NEEDS 41 + ISpElementReference isp_elem [ ISP_NUM_NEEDS ]; + ISpNeed isp_needs [ ISP_NUM_NEEDS ]; +#endif + +#if TARGET_HOST_MAC_OSX + IOHIDDeviceInterface ** hidDev; + IOHIDElementCookie buttonCookies[41]; + IOHIDElementCookie axisCookies[_JS_MAX_AXES]; + long minReport[_JS_MAX_AXES], + maxReport[_JS_MAX_AXES]; +#endif + +#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) + JOYCAPS jsCaps; + JOYINFOEX js; + UINT js_id; +#endif + + +#if TARGET_HOST_POSIX_X11 +# if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__NetBSD__) + struct os_specific_s *os; +# endif + +# ifdef JS_NEW + struct js_event js; + int tmp_buttons; + float tmp_axes [ _JS_MAX_AXES ]; +# else + struct JS_DATA_TYPE js; +# endif + + char fname [ 128 ]; + int fd; +#endif + + int id; + GLboolean error; + char name [ 128 ]; + int num_axes; + int num_buttons; + + float dead_band[ _JS_MAX_AXES ]; + float saturate [ _JS_MAX_AXES ]; + float center [ _JS_MAX_AXES ]; + float max [ _JS_MAX_AXES ]; + float min [ _JS_MAX_AXES ]; +}; + +/* + * Functions associated with the "jsJoystick" class in PLIB + */ +#if TARGET_HOST_MAC_OSX +#define K_NUM_DEVICES 32 +int numDevices; +io_object_t ioDevices[K_NUM_DEVICES]; + +static void fghJoystickFindDevices ( SFG_Joystick* joy, mach_port_t ); +static CFDictionaryRef fghJoystickGetCFProperties ( SFG_Joystick* joy, io_object_t ); + +static void fghJoystickEnumerateElements ( SFG_Joystick* joy, CFTypeRef element ); +/* callback for CFArrayApply */ +static void fghJoystickElementEnumerator ( SFG_Joystick* joy, void *element, void* vjs ); + +static void fghJoystickAddAxisElement ( SFG_Joystick* joy, CFDictionaryRef axis ); +static void fghJoystickAddButtonElement ( SFG_Joystick* joy, CFDictionaryRef button ); +static void fghJoystickAddHatElement ( SFG_Joystick* joy, CFDictionaryRef hat ); +#endif + + +/* + * The static joystick structure pointer + */ +#define MAX_NUM_JOYSTICKS 2 +static SFG_Joystick *fgJoystick [ MAX_NUM_JOYSTICKS ]; + + +/* + * Read the raw joystick data + */ +static void fghJoystickRawRead( SFG_Joystick* joy, int* buttons, float* axes ) +{ +#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) + MMRESULT status; +#else + int status; +#endif + +#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__NetBSD__) + int len; +#endif + + int i; + + /* Defaults */ + if( buttons ) + *buttons = 0; + + if( axes ) + for( i = 0; i < joy->num_axes; i++ ) + axes[ i ] = 1500.0f; + + if( joy->error ) + return; + +#if TARGET_HOST_MACINTOSH + if ( buttons ) + { + *buttons = 0; + + for ( i = 0; i < joy->num_buttons; i++ ) + { + UInt32 state; + int err = ISpElement_GetSimpleState ( isp_elem [ i + isp_num_axis ], &state); + ISP_CHECK_ERR(err) + + *buttons |= state << i; + } + } + + if ( axes ) + { + for ( i = 0; i < joy->num_axes; i++ ) + { + UInt32 state; + int err = ISpElement_GetSimpleState ( isp_elem [ i ], &state ); + ISP_CHECK_ERR(err) + + axes [i] = (float) state; + } + } +#endif + +#if TARGET_HOST_MAC_OSX + if ( buttons != NULL ) + { + *buttons = 0; + + for ( i = 0; i < joy->num_buttons; i++ ) + { + IOHIDEventStruct hidEvent; + (*(joy->hidDev))->getElementValue ( joy->hidDev, buttonCookies[i], &hidEvent ); + if ( hidEvent.value ) + *buttons |= 1 << i; + } + } + + if ( axes != NULL ) + { + for ( i = 0; i < joy->num_axes; i++ ) + { + IOHIDEventStruct hidEvent; + (*(joy->hidDev))->getElementValue ( joy->hidDev, axisCookies[i], &hidEvent ); + axes[i] = hidEvent.value; + } + } +#endif + +#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) + status = joyGetPosEx( joy->js_id, &joy->js ); + + if ( status != JOYERR_NOERROR ) + { + joy->error = GL_TRUE; + return; + } + + if ( buttons ) + *buttons = joy->js.dwButtons; + + if ( axes ) + { + /* + * WARNING - Fall through case clauses!! + */ + switch ( joy->num_axes ) + { + case 8: + /* Generate two POV axes from the POV hat angle. + * Low 16 bits of js.dwPOV gives heading (clockwise from ahead) in + * hundredths of a degree, or 0xFFFF when idle. + */ + if ( ( joy->js.dwPOV & 0xFFFF ) == 0xFFFF ) + { + axes [ 6 ] = 0.0; + axes [ 7 ] = 0.0; + } + else + { + /* This is the contentious bit: how to convert angle to X/Y. + * wk: I know of no define for PI that we could use here: + * SG_PI would pull in sg, M_PI is undefined for MSVC + * But the accuracy of the value of PI is very unimportant at + * this point. + */ + float s = (float) sin ( ( joy->js.dwPOV & 0xFFFF ) * ( 0.01 * 3.1415926535f / 180.0f ) ); + float c = (float) cos ( ( joy->js.dwPOV & 0xFFFF ) * ( 0.01 * 3.1415926535f / 180.0f ) ); + + /* Convert to coordinates on a square so that North-East + * is (1,1) not (.7,.7), etc. + * s and c cannot both be zero so we won't divide by zero. + */ + if ( fabs ( s ) < fabs ( c ) ) + { + axes [ 6 ] = ( c < 0.0 ) ? -s/c : s/c ; + axes [ 7 ] = ( c < 0.0 ) ? -1.0f : 1.0f; + } + else + { + axes [ 6 ] = ( s < 0.0 ) ? -1.0f : 1.0f; + axes [ 7 ] = ( s < 0.0 ) ? -c/s : c/s ; + } + } + + case 6: axes[5] = (float) joy->js.dwVpos; + case 5: axes[4] = (float) joy->js.dwUpos; + case 4: axes[3] = (float) joy->js.dwRpos; + case 3: axes[2] = (float) joy->js.dwZpos; + case 2: axes[1] = (float) joy->js.dwYpos; + case 1: axes[0] = (float) joy->js.dwXpos; + } + } +#endif + +#if TARGET_HOST_POSIX_X11 +# if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__NetBSD__) + if ( joy->os->is_analog ) + { + int status = read ( joy->os->fd, &joy->os->ajs, sizeof(joy->os->ajs) ); + if ( status != sizeof(joy->os->ajs) ) { + perror ( joy->os->fname ); + joy->error = GL_TRUE; + return; + } + if ( buttons != NULL ) + *buttons = ( joy->os->ajs.b1 ? 1 : 0 ) | ( joy->os->ajs.b2 ? 2 : 0 ); + + if ( axes != NULL ) + { + axes[0] = (float) joy->os->ajs.x; + axes[1] = (float) joy->os->ajs.y; + } + + return; + } + +# ifdef HAVE_USB_JS + while ( ( len = read ( joy->os->fd, joy->os->hid_data_buf, joy->os->hid_dlen ) ) == joy->os->hid_dlen ) + { + struct hid_item *h; + + for ( h = joy->os->hids; h; h = h->next ) + { + int d = hid_get_data ( joy->os->hid_data_buf, h ); + + int page = HID_PAGE ( h->usage ); + int usage = HID_USAGE ( h->usage ); + + if ( page == HUP_GENERIC_DESKTOP ) + { + int i; + for ( i = 0; i < joy->num_axes; i++ ) + if (joy->os->axes_usage[i] == usage) + { + if (usage == HUG_HAT_SWITCH) + { + if (d < 0 || d > 8) + d = 0; /* safety */ + joy->os->cache_axes[i] = (float)hatmap_x[d]; + joy->os->cache_axes[i + 1] = (float)hatmap_y[d]; + } + else + { + joy->os->cache_axes[i] = (float)d; + } + break; + } + } + else if (page == HUP_BUTTON) + { + if (usage > 0 && usage < _JS_MAX_BUTTONS + 1) + { + if (d) + joy->os->cache_buttons |= (1 << ( usage - 1 )); + else + joy->os->cache_buttons &= ~(1 << ( usage - 1 )); + } + } + } + } +#ifdef HAVE_ERRNO_H + if ( len < 0 && errno != EAGAIN ) +#else + if ( len < 0 ) +#endif + { + perror( joy->os->fname ); + joy->error = 1; + } + if ( buttons != NULL ) *buttons = joy->os->cache_buttons; + if ( axes != NULL ) + memcpy ( axes, joy->os->cache_axes, sizeof(float) * joy->num_axes ); +# endif +# endif + +# ifdef JS_NEW + + while ( 1 ) + { + status = read ( joy->fd, &joy->js, sizeof(struct js_event) ); + + if ( status != sizeof( struct js_event ) ) + { +#ifdef HAVE_ERRNO_H + if ( errno == EAGAIN ) + { + /* Use the old values */ + if ( buttons ) + *buttons = joy->tmp_buttons; + if ( axes ) + memcpy( axes, joy->tmp_axes, + sizeof( float ) * joy->num_axes ); + return; + } +#endif + + fgWarning ( "%s", joy->fname ); + joy->error = GL_TRUE; + return; + } + + switch ( joy->js.type & ~JS_EVENT_INIT ) + { + case JS_EVENT_BUTTON: + if( joy->js.value == 0 ) /* clear the flag */ + joy->tmp_buttons &= ~( 1 << joy->js.number ); + else + joy->tmp_buttons |= ( 1 << joy->js.number ); + break; + + case JS_EVENT_AXIS: + if ( joy->js.number < joy->num_axes ) + { + joy->tmp_axes[ joy->js.number ] = ( float )joy->js.value; + + if( axes ) + memcpy( axes, joy->tmp_axes, sizeof(float) * joy->num_axes ); + } + break; + + default: + fgWarning ( "PLIB_JS: Unrecognised /dev/js return!?!" ); + + /* use the old values */ + + if ( buttons != NULL ) *buttons = joy->tmp_buttons; + if ( axes != NULL ) + memcpy ( axes, joy->tmp_axes, sizeof(float) * joy->num_axes ); + + return; + } + + if( buttons ) + *buttons = joy->tmp_buttons; + } +# else + + status = read( joy->fd, &joy->js, JS_RETURN ); + + if ( status != JS_RETURN ) + { + fgWarning( "%s", joy->fname ); + joy->error = GL_TRUE; + return; + } + + if ( buttons ) +# if defined( __FreeBSD__ ) || defined(__FreeBSD_kernel__) || defined( __NetBSD__ ) + *buttons = ( joy->js.b1 ? 1 : 0 ) | ( joy->js.b2 ? 2 : 0 ); /* XXX Should not be here -- BSD is handled earlier */ +# else + *buttons = joy->js.buttons; +# endif + + if ( axes ) + { + axes[ 0 ] = (float) joy->js.x; + axes[ 1 ] = (float) joy->js.y; + } +# endif +#endif +} + +/* + * Correct the joystick axis data + */ +static float fghJoystickFudgeAxis( SFG_Joystick* joy, float value, int axis ) +{ + if( value < joy->center[ axis ] ) + { + float xx = ( value - joy->center[ axis ] ) / ( joy->center[ axis ] - + joy->min[ axis ] ); + + if( xx < -joy->saturate[ axis ] ) + return -1.0f; + + if( xx > -joy->dead_band [ axis ] ) + return 0.0f; + + xx = ( xx + joy->dead_band[ axis ] ) / ( joy->saturate[ axis ] - + joy->dead_band[ axis ] ); + + return ( xx < -1.0f ) ? -1.0f : xx; + } + else + { + float xx = ( value - joy->center [ axis ] ) / ( joy->max[ axis ] - + joy->center[ axis ] ); + + if( xx > joy->saturate[ axis ] ) + return 1.0f; + + if( xx < joy->dead_band[ axis ] ) + return 0.0f; + + xx = ( xx - joy->dead_band[ axis ] ) / ( joy->saturate[ axis ] - + joy->dead_band[ axis ] ); + + return ( xx > 1.0f ) ? 1.0f : xx; + } +} + +/* + * Read the corrected joystick data + */ +static void fghJoystickRead( SFG_Joystick* joy, int* buttons, float* axes ) +{ + float raw_axes[ _JS_MAX_AXES ]; + int i; + + if( joy->error ) + { + if( buttons ) + *buttons = 0; + + if( axes ) + for ( i=0; inum_axes; i++ ) + axes[ i ] = 0.0f; + } + + fghJoystickRawRead( joy, buttons, raw_axes ); + + if( axes ) + for( i=0; inum_axes; i++ ) + axes[ i ] = fghJoystickFudgeAxis( joy, raw_axes[ i ], i ); +} + +/* + * Happy happy happy joy joy joy (happy new year toudi :D) + */ + + +#if TARGET_HOST_MAC_OSX +/** open the IOKit connection, enumerate all the HID devices, add their +interface references to the static array. We then use the array index +as the device number when we come to open() the joystick. */ +static int fghJoystickFindDevices ( SFG_Joystick *joy, mach_port_t masterPort ) +{ + CFMutableDictionaryRef hidMatch = NULL; + IOReturn rv = kIOReturnSuccess; + + io_iterator_t hidIterator; + io_object_t ioDev; + + /* build a dictionary matching HID devices */ + hidMatch = IOServiceMatching(kIOHIDDeviceKey); + + rv = IOServiceGetMatchingServices(masterPort, hidMatch, &hidIterator); + if (rv != kIOReturnSuccess || !hidIterator) { + fgWarning( "no joystick (HID) devices found" ); + return; + } + + /* iterate */ + while ((ioDev = IOIteratorNext(hidIterator))) { + /* filter out keyboard and mouse devices */ + CFDictionaryRef properties = getCFProperties(ioDev); + long usage, page; + + CFTypeRef refPage = CFDictionaryGetValue (properties, CFSTR(kIOHIDPrimaryUsagePageKey)); + CFTypeRef refUsage = CFDictionaryGetValue (properties, CFSTR(kIOHIDPrimaryUsageKey)); + CFNumberGetValue((CFNumberRef) refUsage, kCFNumberLongType, &usage); + CFNumberGetValue((CFNumberRef) refPage, kCFNumberLongType, &page); + + /* keep only joystick devices */ + if ( ( page == kHIDPage_GenericDesktop ) && ( + (usage == kHIDUsage_GD_Joystick) + || (usage == kHIDUsage_GD_GamePad) + || (usage == kHIDUsage_GD_MultiAxisController) + || (usage == kHIDUsage_GD_Hatswitch) /* last two necessary ? */ + /* add it to the array */ + ioDevices[numDevices++] = ioDev; + } + + IOObjectRelease(hidIterator); +} + +static CFDictionaryRef fghJoystickGetCFProperties ( SFG_Joystick *joy, io_object_t ioDev ) +{ + IOReturn rv; + CFMutableDictionaryRef cfProperties; + +#if 0 + /* comment copied from darwin/SDL_sysjoystick.c */ + /* Mac OS X currently is not mirroring all USB properties to HID page so need to look at USB device page also + * get dictionary for usb properties: step up two levels and get CF dictionary for USB properties + */ + + io_registry_entry_t parent1, parent2; + + rv = IORegistryEntryGetParentEntry (ioDev, kIOServicePlane, &parent1); + if (rv != kIOReturnSuccess) { + fgWarning ( "error getting device entry parent"); + return NULL; + } + + rv = IORegistryEntryGetParentEntry (parent1, kIOServicePlane, &parent2); + if (rv != kIOReturnSuccess) { + fgWarning ( "error getting device entry parent 2"); + return NULL; + } +#endif + + rv = IORegistryEntryCreateCFProperties( ioDev /*parent2*/, + &cfProperties, kCFAllocatorDefault, kNilOptions); + if (rv != kIOReturnSuccess || !cfProperties) { + fgWarning ( "error getting device properties"); + return NULL; + } + + return cfProperties; +} + +static void fghJoystickElementEnumerator ( SFG_Joystick *joy, void *element, void* vjs ) +{ + if (CFGetTypeID((CFTypeRef) element) != CFDictionaryGetTypeID()) { + fgError ( "%s", "element enumerator passed non-dictionary value"); + return; + } + + static_cast(vjs)->parseElement ( (CFDictionaryRef) element ); +} + +/** element enumerator function : pass NULL for top-level*/ +static void fghJoystickEnumerateElements ( SFG_Joystick *joy, CFTypeRef element ) +{ + FREEGLUT_INTERNAL_ERROR_EXIT( (CFGetTypeID(element) == CFArrayGetTypeID(), + "Joystick element type mismatch", + "fghJoystickEnumerateElements" ); + + CFRange range = {0, CFArrayGetCount ((CFArrayRef)element)}; + CFArrayApplyFunction((CFArrayRef) element, range, + &fghJoystickElementEnumerator, joy ); +} + +static void fghJoystickAddAxisElement ( SFG_Joystick *joy, CFDictionaryRef axis ) +{ + long cookie, lmin, lmax; + int index = joy->num_axes++; + + CFNumberGetValue ((CFNumberRef) + CFDictionaryGetValue ( axis, CFSTR(kIOHIDElementCookieKey) ), + kCFNumberLongType, &cookie); + + axisCookies[index] = (IOHIDElementCookie) cookie; + + CFNumberGetValue ((CFNumberRef) + CFDictionaryGetValue ( axis, CFSTR(kIOHIDElementMinKey) ), + kCFNumberLongType, &lmin); + + CFNumberGetValue ((CFNumberRef) + CFDictionaryGetValue ( axis, CFSTR(kIOHIDElementMaxKey) ), + kCFNumberLongType, &lmax); + + joy->min[index] = lmin; + joy->max[index] = lmax; + joy->dead_band[index] = 0.0; + joy->saturate[index] = 1.0; + joy->center[index] = (lmax + lmin) * 0.5; +} + +static void fghJoystickAddButtonElement ( SFG_Joystick *joy, CFDictionaryRef button ) +{ + long cookie; + CFNumberGetValue ((CFNumberRef) + CFDictionaryGetValue ( button, CFSTR(kIOHIDElementCookieKey) ), + kCFNumberLongType, &cookie); + + joy->buttonCookies[num_buttons++] = (IOHIDElementCookie) cookie; + /* anything else for buttons? */ +} + +static void fghJoystickAddHatElement ( SFG_Joystick *joy, CFDictionaryRef button ) +{ + /* hatCookies[num_hats++] = (IOHIDElementCookie) cookie; */ + /* do we map hats to axes or buttons? */ +} +#endif + +#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) +/* Inspired by + http://msdn.microsoft.com/archive/en-us/dnargame/html/msdn_sidewind3d.asp + */ +# if FREEGLUT_LIB_PRAGMAS +# pragma comment (lib, "advapi32.lib") +# endif + +static int fghJoystickGetOEMProductName ( SFG_Joystick* joy, char *buf, int buf_sz ) +{ + char buffer [ 256 ]; + + char OEMKey [ 256 ]; + + HKEY hKey; + DWORD dwcb; + LONG lr; + + if ( joy->error ) + return 0; + + /* Open .. MediaResources\CurrentJoystickSettings */ + _snprintf ( buffer, sizeof(buffer), "%s\\%s\\%s", + REGSTR_PATH_JOYCONFIG, joy->jsCaps.szRegKey, + REGSTR_KEY_JOYCURR ); + + lr = RegOpenKeyEx ( HKEY_LOCAL_MACHINE, buffer, 0, KEY_QUERY_VALUE, &hKey); + + if ( lr != ERROR_SUCCESS ) return 0; + + /* Get OEM Key name */ + dwcb = sizeof(OEMKey); + + /* JOYSTICKID1-16 is zero-based; registry entries for VJOYD are 1-based. */ + _snprintf ( buffer, sizeof(buffer), "Joystick%d%s", joy->js_id + 1, REGSTR_VAL_JOYOEMNAME ); + + lr = RegQueryValueEx ( hKey, buffer, 0, 0, (LPBYTE) OEMKey, &dwcb); + RegCloseKey ( hKey ); + + if ( lr != ERROR_SUCCESS ) return 0; + + /* Open OEM Key from ...MediaProperties */ + _snprintf ( buffer, sizeof(buffer), "%s\\%s", REGSTR_PATH_JOYOEM, OEMKey ); + + lr = RegOpenKeyEx ( HKEY_LOCAL_MACHINE, buffer, 0, KEY_QUERY_VALUE, &hKey ); + + if ( lr != ERROR_SUCCESS ) return 0; + + /* Get OEM Name */ + dwcb = buf_sz; + + lr = RegQueryValueEx ( hKey, REGSTR_VAL_JOYOEMNAME, 0, 0, (LPBYTE) buf, + &dwcb ); + RegCloseKey ( hKey ); + + if ( lr != ERROR_SUCCESS ) return 0; + + return 1; +} +#endif + + +static void fghJoystickOpen( SFG_Joystick* joy ) +{ + int i = 0; +#if TARGET_HOST_MACINTOSH + OSStatus err; +#endif +#if TARGET_HOST_MAC_OSX + IOReturn rv; + SInt32 score; + IOCFPlugInInterface **plugin; + + HRESULT pluginResult; + + CFDictionaryRef props; + CFTypeRef topLevelElement; +#endif +#if TARGET_HOST_POSIX_X11 +# if defined( __FreeBSD__ ) || defined(__FreeBSD_kernel__) || defined( __NetBSD__ ) + char *cp; +# endif +# ifdef JS_NEW + unsigned char u; +# else +# if defined( __linux__ ) || TARGET_HOST_SOLARIS + int counter = 0; +# endif +# endif +#endif + + /* Silence gcc, the correct #ifdefs would be too fragile... */ + (void)i; + + /* + * Default values (for no joystick -- each conditional will reset the + * error flag) + */ + joy->error = TRUE; + joy->num_axes = joy->num_buttons = 0; + joy->name[ 0 ] = '\0'; + +#if TARGET_HOST_MACINTOSH + /* XXX FIXME: get joystick name in Mac */ + + err = ISpStartup( ); + + if( err == noErr ) + { +#define ISP_CHECK_ERR(x) if( x != noErr ) { joy->error = GL_TRUE; return; } + + joy->error = GL_TRUE; + + /* initialize the needs structure */ + ISpNeed temp_isp_needs[ isp_num_needs ] = + { + { "\pX-Axis", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, + { "\pY-Axis", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, + { "\pZ-Axis", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, + { "\pR-Axis", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, + { "\pAxis 4", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, + { "\pAxis 5", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, + { "\pAxis 6", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, + { "\pAxis 7", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, + { "\pAxis 8", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, + + { "\pButton 0", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 1", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 2", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 3", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 4", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 5", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 6", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 7", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 8", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 9", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 10", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 11", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 12", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 13", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 14", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 15", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 16", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 17", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 18", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 19", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 20", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 21", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 22", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 23", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 24", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 25", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 26", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 27", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 28", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 29", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 30", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + { "\pButton 31", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, + }; + + memcpy( joy->isp_needs, temp_isp_needs, sizeof (temp_isp_needs ) ); + + + /* next two calls allow keyboard and mouse to emulate other input + * devices (gamepads, joysticks, etc) + */ + /* + err = ISpDevices_ActivateClass ( kISpDeviceClass_Keyboard ); + ISP_CHECK_ERR(err) + + + err = ISpDevices_ActivateClass ( kISpDeviceClass_Mouse ); + ISP_CHECK_ERR(err) + */ + + err = ISpElement_NewVirtualFromNeeds( joy->isp_num_needs, + joy->isp_needs, joy->isp_elem, + 0 ); + ISP_CHECK_ERR( err ) + + err = ISpInit( joy->isp_num_needs, joy->isp_needs, joy->isp_elem, + 'freeglut', nil, 0, 128, 0 ); + ISP_CHECK_ERR( err ) + + joy->num_buttons = joy->isp_num_needs - joy->isp_num_axis; + joy->num_axes = joy->isp_num_axis; + + for( i = 0; i < joy->num_axes; i++ ) + { + joy->dead_band[ i ] = 0; + joy->saturate [ i ] = 1; + joy->center [ i ] = kISpAxisMiddle; + joy->max [ i ] = kISpAxisMaximum; + joy->min [ i ] = kISpAxisMinimum; + } + + joy->error = GL_FALSE; + } + else + joy->num_buttons = joy->num_axes = 0; +#endif + +#if TARGET_HOST_MAC_OSX + if( joy->id >= numDevices ) + { + fgWarning( "device index out of range in fgJoystickOpen()" ); + return; + } + + /* create device interface */ + rv = IOCreatePlugInInterfaceForService( ioDevices[ joy->id ], + kIOHIDDeviceUserClientTypeID, + kIOCFPlugInInterfaceID, + &plugin, &score ); + + if( rv != kIOReturnSuccess ) + { + fgWarning( "error creating plugin for io device" ); + return; + } + + pluginResult = ( *plugin )->QueryInterface( + plugin, + CFUUIDGetUUIDBytes(kIOHIDDeviceInterfaceID), + &( LPVOID )joy->hidDev + ); + + if( pluginResult != S_OK ) + fgWarning ( "QI-ing IO plugin to HID Device interface failed" ); + + ( *plugin )->Release( plugin ); /* don't leak a ref */ + if( joy->hidDev == NULL ) + return; + + /* store the interface in this instance */ + rv = ( *( joy->hidDev ) )->open( joy->hidDev, 0 ); + if( rv != kIOReturnSuccess ) + { + fgWarning( "error opening device interface"); + return; + } + + props = getCFProperties( ioDevices[ joy->id ] ); + + /* recursively enumerate all the bits */ + CFTypeRef topLevelElement = + CFDictionaryGetValue( props, CFSTR( kIOHIDElementKey ) ); + enumerateElements( topLevelElement ); + + CFRelease( props ); +#endif + +#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) + joy->js.dwFlags = JOY_RETURNALL; + joy->js.dwSize = sizeof( joy->js ); + + memset( &joy->jsCaps, 0, sizeof( joy->jsCaps ) ); + + joy->error = + ( joyGetDevCaps( joy->js_id, &joy->jsCaps, sizeof( joy->jsCaps ) ) != + JOYERR_NOERROR ); + + if( joy->jsCaps.wNumAxes == 0 ) + { + joy->num_axes = 0; + joy->error = GL_TRUE; + } + else + { + /* Device name from jsCaps is often "Microsoft PC-joystick driver", + * at least for USB. Try to get the real name from the registry. + */ + if ( ! fghJoystickGetOEMProductName( joy, joy->name, + sizeof( joy->name ) ) ) + { + fgWarning( "JS: Failed to read joystick name from registry" ); + strncpy( joy->name, joy->jsCaps.szPname, sizeof( joy->name ) ); + } + + /* Windows joystick drivers may provide any combination of + * X,Y,Z,R,U,V,POV - not necessarily the first n of these. + */ + if( joy->jsCaps.wCaps & JOYCAPS_HASPOV ) + { + joy->num_axes = _JS_MAX_AXES; + joy->min[ 7 ] = -1.0; joy->max[ 7 ] = 1.0; /* POV Y */ + joy->min[ 6 ] = -1.0; joy->max[ 6 ] = 1.0; /* POV X */ + } + else + joy->num_axes = 6; + + joy->min[ 5 ] = ( float )joy->jsCaps.wVmin; + joy->max[ 5 ] = ( float )joy->jsCaps.wVmax; + joy->min[ 4 ] = ( float )joy->jsCaps.wUmin; + joy->max[ 4 ] = ( float )joy->jsCaps.wUmax; + joy->min[ 3 ] = ( float )joy->jsCaps.wRmin; + joy->max[ 3 ] = ( float )joy->jsCaps.wRmax; + joy->min[ 2 ] = ( float )joy->jsCaps.wZmin; + joy->max[ 2 ] = ( float )joy->jsCaps.wZmax; + joy->min[ 1 ] = ( float )joy->jsCaps.wYmin; + joy->max[ 1 ] = ( float )joy->jsCaps.wYmax; + joy->min[ 0 ] = ( float )joy->jsCaps.wXmin; + joy->max[ 0 ] = ( float )joy->jsCaps.wXmax; + } + + /* Guess all the rest judging on the axes extremals */ + for( i = 0; i < joy->num_axes; i++ ) + { + joy->center [ i ] = ( joy->max[ i ] + joy->min[ i ] ) * 0.5f; + joy->dead_band[ i ] = 0.0f; + joy->saturate [ i ] = 1.0f; + } +#endif + +#if TARGET_HOST_POSIX_X11 +#if defined( __FreeBSD__ ) || defined(__FreeBSD_kernel__) || defined( __NetBSD__ ) + for( i = 0; i < _JS_MAX_AXES; i++ ) + joy->os->cache_axes[ i ] = 0.0f; + + joy->os->cache_buttons = 0; + + joy->os->fd = open( joy->os->fname, O_RDONLY | O_NONBLOCK); + +#ifdef HAVE_ERRNO_H + if( joy->os->fd < 0 && errno == EACCES ) + fgWarning ( "%s exists but is not readable by you", joy->os->fname ); +#endif + + joy->error =( joy->os->fd < 0 ); + + if( joy->error ) + return; + + joy->num_axes = 0; + joy->num_buttons = 0; + if( joy->os->is_analog ) + { + FILE *joyfile; + char joyfname[ 1024 ]; + int noargs, in_no_axes; + + float axes [ _JS_MAX_AXES ]; + int buttons[ _JS_MAX_AXES ]; + + joy->num_axes = 2; + joy->num_buttons = 32; + + fghJoystickRawRead( joy, buttons, axes ); + joy->error = axes[ 0 ] < -1000000000.0f; + if( joy->error ) + return; + + snprintf( joyfname, sizeof(joyfname), "%s/.joy%drc", getenv( "HOME" ), joy->id ); + + joyfile = fopen( joyfname, "r" ); + joy->error =( joyfile == NULL ); + if( joy->error ) + return; + + noargs = fscanf( joyfile, "%d%f%f%f%f%f%f", &in_no_axes, + &joy->min[ 0 ], &joy->center[ 0 ], &joy->max[ 0 ], + &joy->min[ 1 ], &joy->center[ 1 ], &joy->max[ 1 ] ); + joy->error = noargs != 7 || in_no_axes != _JS_MAX_AXES; + fclose( joyfile ); + if( joy->error ) + return; + + for( i = 0; i < _JS_MAX_AXES; i++ ) + { + joy->dead_band[ i ] = 0.0f; + joy->saturate [ i ] = 1.0f; + } + + return; /* End of analog code */ + } + +# ifdef HAVE_USB_JS + if( ! fghJoystickInitializeHID( joy->os, &joy->num_axes, + &joy->num_buttons ) ) + { + close( joy->os->fd ); + joy->error = GL_TRUE; + return; + } + + cp = strrchr( joy->os->fname, '/' ); + if( cp ) + { + if( fghJoystickFindUSBdev( &cp[1], joy->name, sizeof( joy->name ) ) == + 0 ) + strcpy( joy->name, &cp[1] ); + } + + if( joy->num_axes > _JS_MAX_AXES ) + joy->num_axes = _JS_MAX_AXES; + + for( i = 0; i < _JS_MAX_AXES; i++ ) + { + /* We really should get this from the HID, but that data seems + * to be quite unreliable for analog-to-USB converters. Punt for + * now. + */ + if( joy->os->axes_usage[ i ] == HUG_HAT_SWITCH ) + { + joy->max [ i ] = 1.0f; + joy->center[ i ] = 0.0f; + joy->min [ i ] = -1.0f; + } + else + { + joy->max [ i ] = 255.0f; + joy->center[ i ] = 127.0f; + joy->min [ i ] = 0.0f; + } + + joy->dead_band[ i ] = 0.0f; + joy->saturate[ i ] = 1.0f; + } +# endif +#endif + +#if defined( __linux__ ) || TARGET_HOST_SOLARIS + /* Default for older Linux systems. */ + joy->num_axes = 2; + joy->num_buttons = 32; + +# ifdef JS_NEW + for( i = 0; i < _JS_MAX_AXES; i++ ) + joy->tmp_axes[ i ] = 0.0f; + + joy->tmp_buttons = 0; +# endif + + joy->fd = open( joy->fname, O_RDONLY ); + + joy->error =( joy->fd < 0 ); + + if( joy->error ) + return; + + /* Set the correct number of axes for the linux driver */ +# ifdef JS_NEW + /* Melchior Franz's fixes for big-endian Linuxes since writing + * to the upper byte of an uninitialized word doesn't work. + * 9 April 2003 + */ + ioctl( joy->fd, JSIOCGAXES, &u ); + joy->num_axes = u; + ioctl( joy->fd, JSIOCGBUTTONS, &u ); + joy->num_buttons = u; + ioctl( joy->fd, JSIOCGNAME( sizeof( joy->name ) ), joy->name ); + fcntl( joy->fd, F_SETFL, O_NONBLOCK ); +# endif + + /* + * The Linux driver seems to return 512 for all axes + * when no stick is present - but there is a chance + * that could happen by accident - so it's gotta happen + * on both axes for at least 100 attempts. + * + * PWO: shouldn't be that done somehow wiser on the kernel level? + */ +# ifndef JS_NEW + counter = 0; + + do + { + fghJoystickRawRead( joy, NULL, joy->center ); + counter++; + } while( !joy->error && + counter < 100 && + joy->center[ 0 ] == 512.0f && + joy->center[ 1 ] == 512.0f ); + + if ( counter >= 100 ) + joy->error = GL_TRUE; +# endif + + for( i = 0; i < _JS_MAX_AXES; i++ ) + { +# ifdef JS_NEW + joy->max [ i ] = 32767.0f; + joy->center[ i ] = 0.0f; + joy->min [ i ] = -32767.0f; +# else + joy->max[ i ] = joy->center[ i ] * 2.0f; + joy->min[ i ] = 0.0f; +# endif + joy->dead_band[ i ] = 0.0f; + joy->saturate [ i ] = 1.0f; + } +#endif +#endif +} + +/* + * This function replaces the constructor method in the JS library. + */ +static void fghJoystickInit( int ident ) +{ + if( ident >= MAX_NUM_JOYSTICKS ) + fgError( "Too large a joystick number: %d", ident ); + + if( fgJoystick[ ident ] ) + fgError( "illegal attempt to initialize joystick device again" ); + + fgJoystick[ ident ] = + ( SFG_Joystick * )calloc( sizeof( SFG_Joystick ), 1 ); + + /* Set defaults */ + fgJoystick[ ident ]->num_axes = fgJoystick[ ident ]->num_buttons = 0; + fgJoystick[ ident ]->error = GL_TRUE; + +#if TARGET_HOST_MACINTOSH + fgJoystick[ ident ]->id = ident; + snprintf( fgJoystick[ ident ]->fname, sizeof(fgJoystick[ ident ]->fname), "/dev/js%d", ident ); /* FIXME */ + fgJoystick[ ident ]->error = GL_FALSE; +#endif + +#if TARGET_HOST_MAC_OSX + fgJoystick[ ident ]->id = ident; + fgJoystick[ ident ]->error = GL_FALSE; + fgJoystick[ ident ]->num_axes = 0; + fgJoystick[ ident ]->num_buttons = 0; + + if( numDevices < 0 ) + { + /* do first-time init (since we can't over-ride jsInit, hmm */ + numDevices = 0; + + mach_port_t masterPort; + IOReturn rv = IOMasterPort( bootstrap_port, &masterPort ); + if( rv != kIOReturnSuccess ) + { + fgWarning( "error getting master Mach port" ); + return; + } + fghJoystickFindDevices( masterPort ); + } + + if ( ident >= numDevices ) + { + fgJoystick[ ident ]->error = GL_TRUE; + return; + } + + /* get the name now too */ + CFDictionaryRef properties = getCFProperties( ioDevices[ ident ] ); + CFTypeRef ref = CFDictionaryGetValue( properties, + CFSTR( kIOHIDProductKey ) ); + if (!ref) + ref = CFDictionaryGetValue(properties, CFSTR( "USB Product Name" ) ); + + if( !ref || + !CFStringGetCString( ( CFStringRef )ref, name, 128, + CFStringGetSystemEncoding( ) ) ) + { + fgWarning( "error getting device name" ); + name[ 0 ] = '\0'; + } +#endif + +#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) + switch( ident ) + { + case 0: + fgJoystick[ ident ]->js_id = JOYSTICKID1; + fgJoystick[ ident ]->error = GL_FALSE; + break; + case 1: + fgJoystick[ ident ]->js_id = JOYSTICKID2; + fgJoystick[ ident ]->error = GL_FALSE; + break; + default: + fgJoystick[ ident ]->num_axes = 0; + fgJoystick[ ident ]->error = GL_TRUE; + return; + } +#endif + +#if TARGET_HOST_POSIX_X11 +# if defined( __FreeBSD__ ) || defined(__FreeBSD_kernel__) || defined( __NetBSD__ ) + fgJoystick[ ident ]->id = ident; + fgJoystick[ ident ]->error = GL_FALSE; + + fgJoystick[ ident ]->os = calloc( 1, sizeof( struct os_specific_s ) ); + memset( fgJoystick[ ident ]->os, 0, sizeof( struct os_specific_s ) ); + if( ident < USB_IDENT_OFFSET ) + fgJoystick[ ident ]->os->is_analog = 1; + if( fgJoystick[ ident ]->os->is_analog ) + snprintf( fgJoystick[ ident ]->os->fname, sizeof(fgJoystick[ ident ]->os->fname), "%s%d", AJSDEV, ident ); + else + snprintf( fgJoystick[ ident ]->os->fname, sizeof(fgJoystick[ ident ]->os->fname), "%s%d", UHIDDEV, + ident - USB_IDENT_OFFSET ); +# elif defined( __linux__ ) + fgJoystick[ ident ]->id = ident; + fgJoystick[ ident ]->error = GL_FALSE; + + snprintf( fgJoystick[ident]->fname, sizeof(fgJoystick[ident]->fname), "/dev/input/js%d", ident ); + + if( access( fgJoystick[ ident ]->fname, F_OK ) != 0 ) + snprintf( fgJoystick[ ident ]->fname, sizeof(fgJoystick[ ident ]->fname), "/dev/js%d", ident ); +# endif +#endif + + fghJoystickOpen( fgJoystick[ ident ] ); +} + +/* + * Try initializing all the joysticks (well, both of them) + */ +void fgInitialiseJoysticks ( void ) +{ + if( !fgState.JoysticksInitialised ) + { + int ident ; + for ( ident = 0; ident < MAX_NUM_JOYSTICKS; ident++ ) + fghJoystickInit( ident ); + + fgState.JoysticksInitialised = GL_TRUE; + } +} + +/* + * + */ +void fgJoystickClose( void ) +{ + int ident ; + for( ident = 0; ident < MAX_NUM_JOYSTICKS; ident++ ) + { + if( fgJoystick[ ident ] ) + { + +#if TARGET_HOST_MACINTOSH + ISpSuspend( ); + ISpStop( ); + ISpShutdown( ); +#endif + +#if TARGET_HOST_MAC_OSX + ( *( fgJoystick[ ident ]->hidDev ) )-> + close( fgJoystick[ ident ]->hidDev ); +#endif + +#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) + /* Do nothing special */ +#endif + +#if TARGET_HOST_POSIX_X11 +#if defined( __FreeBSD__ ) || defined(__FreeBSD_kernel__) || defined( __NetBSD__ ) + if( fgJoystick[ident]->os ) + { + if( ! fgJoystick[ ident ]->error ) + close( fgJoystick[ ident ]->os->fd ); +#ifdef HAVE_USB_JS + if( fgJoystick[ ident ]->os->hids ) + free (fgJoystick[ ident ]->os->hids); + if( fgJoystick[ ident ]->os->hid_data_buf ) + free( fgJoystick[ ident ]->os->hid_data_buf ); +#endif + free( fgJoystick[ident]->os ); + } +#endif + + if( ! fgJoystick[ident]->error ) + close( fgJoystick[ ident ]->fd ); +#endif + + free( fgJoystick[ ident ] ); + fgJoystick[ ident ] = NULL; + /* show joystick has been deinitialized */ + } + } +} + +/* + * Polls the joystick and executes the joystick callback hooked to the + * window specified in the function's parameter: + */ +void fgJoystickPollWindow( SFG_Window* window ) +{ + float axes[ _JS_MAX_AXES ]; + int buttons; + int ident; + + freeglut_return_if_fail( window ); + freeglut_return_if_fail( FETCH_WCB( *window, Joystick ) ); + + for( ident = 0; ident < MAX_NUM_JOYSTICKS; ident++ ) + { + if( fgJoystick[ident] ) + { + fghJoystickRead( fgJoystick[ident], &buttons, axes ); + + if( !fgJoystick[ident]->error ) + INVOKE_WCB( *window, Joystick, + ( buttons, + (int) ( axes[ 0 ] * 1000.0f ), + (int) ( axes[ 1 ] * 1000.0f ), + (int) ( axes[ 2 ] * 1000.0f ) ) + ); + } + } +} + +/* + * Implementation for glutDeviceGet(GLUT_HAS_JOYSTICK) + */ +int fgJoystickDetect( void ) +{ + int ident; + + fgInitialiseJoysticks (); + + if ( !fgState.JoysticksInitialised ) + return 0; + + for( ident=0; identerror ) + return 1; + + return 0; +} + +/* + * Joystick information functions + */ +int glutJoystickGetNumAxes( int ident ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetNumAxes" ); + return fgJoystick[ ident ]->num_axes; +} +int glutJoystickGetNumButtons( int ident ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetNumButtons" ); + return fgJoystick[ ident ]->num_buttons; +} +int glutJoystickNotWorking( int ident ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickNotWorking" ); + return fgJoystick[ ident ]->error; +} + +float glutJoystickGetDeadBand( int ident, int axis ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetDeadBand" ); + return fgJoystick[ ident ]->dead_band [ axis ]; +} +void glutJoystickSetDeadBand( int ident, int axis, float db ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickSetDeadBand" ); + fgJoystick[ ident ]->dead_band[ axis ] = db; +} + +float glutJoystickGetSaturation( int ident, int axis ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetSaturation" ); + return fgJoystick[ ident ]->saturate[ axis ]; +} +void glutJoystickSetSaturation( int ident, int axis, float st ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickSetSaturation" ); + fgJoystick[ ident ]->saturate [ axis ] = st; +} + +void glutJoystickSetMinRange( int ident, float *axes ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickSetMinRange" ); + memcpy( fgJoystick[ ident ]->min, axes, + fgJoystick[ ident ]->num_axes * sizeof( float ) ); +} +void glutJoystickSetMaxRange( int ident, float *axes ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickSetMaxRange" ); + memcpy( fgJoystick[ ident ]->max, axes, + fgJoystick[ ident ]->num_axes * sizeof( float ) ); +} +void glutJoystickSetCenter( int ident, float *axes ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickSetCenter" ); + memcpy( fgJoystick[ ident ]->center, axes, + fgJoystick[ ident ]->num_axes * sizeof( float ) ); +} + +void glutJoystickGetMinRange( int ident, float *axes ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetMinRange" ); + memcpy( axes, fgJoystick[ ident ]->min, + fgJoystick[ ident ]->num_axes * sizeof( float ) ); +} +void glutJoystickGetMaxRange( int ident, float *axes ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetMaxRange" ); + memcpy( axes, fgJoystick[ ident ]->max, + fgJoystick[ ident ]->num_axes * sizeof( float ) ); +} +void glutJoystickGetCenter( int ident, float *axes ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetCenter" ); + memcpy( axes, fgJoystick[ ident ]->center, + fgJoystick[ ident ]->num_axes * sizeof( float ) ); +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_main.c b/examples/opengl-framework/freeglut/freeglut_main.c new file mode 100644 index 00000000..5f9b0a5d --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_main.c @@ -0,0 +1,2511 @@ +/* + * freeglut_main.c + * + * The windows message processing methods. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Fri Dec 3 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" +#ifdef HAVE_ERRNO_H +# include +#endif +#include +#ifdef HAVE_VFPRINTF +# define VFPRINTF(s,f,a) vfprintf((s),(f),(a)) +#elif defined(HAVE__DOPRNT) +# define VFPRINTF(s,f,a) _doprnt((f),(a),(s)) +#else +# define VFPRINTF(s,f,a) +#endif + +#ifdef _WIN32_WCE + +typedef struct GXDisplayProperties GXDisplayProperties; +typedef struct GXKeyList GXKeyList; +#include + +typedef struct GXKeyList (*GXGETDEFAULTKEYS)(int); +typedef int (*GXOPENINPUT)(); + +GXGETDEFAULTKEYS GXGetDefaultKeys_ = NULL; +GXOPENINPUT GXOpenInput_ = NULL; + +struct GXKeyList gxKeyList; + +#endif /* _WIN32_WCE */ + +/* + * Try to get the maximum value allowed for ints, falling back to the minimum + * guaranteed by ISO C99 if there is no suitable header. + */ +#ifdef HAVE_LIMITS_H +# include +#endif +#ifndef INT_MAX +# define INT_MAX 32767 +#endif + +#ifndef MIN +# define MIN(a,b) (((a)<(b)) ? (a) : (b)) +#endif + +#ifdef WM_TOUCH + typedef BOOL (WINAPI *pGetTouchInputInfo)(HTOUCHINPUT,UINT,PTOUCHINPUT,int); + typedef BOOL (WINAPI *pCloseTouchInputHandle)(HTOUCHINPUT); + static pGetTouchInputInfo fghGetTouchInputInfo = (pGetTouchInputInfo)0xDEADBEEF; + static pCloseTouchInputHandle fghCloseTouchInputHandle = (pCloseTouchInputHandle)0xDEADBEEF; +#endif + +/* + * TODO BEFORE THE STABLE RELEASE: + * + * There are some issues concerning window redrawing under X11, and maybe + * some events are not handled. The Win32 version lacks some more features, + * but seems acceptable for not demanding purposes. + * + * Need to investigate why the X11 version breaks out with an error when + * closing a window (using the window manager, not glutDestroyWindow)... + */ + +/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ + +/* + * Handle a window configuration change. When no reshape + * callback is hooked, the viewport size is updated to + * match the new window size. + */ +static void fghReshapeWindow ( SFG_Window *window, int width, int height ) +{ + SFG_Window *current_window = fgStructure.CurrentWindow; + + freeglut_return_if_fail( window != NULL ); + +#if TARGET_HOST_POSIX_X11 + + XResizeWindow( fgDisplay.Display, window->Window.Handle, + width, height ); + XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ + +#elif TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) + { + RECT windowRect; + + /* + * For windowed mode, get the current position of the + * window and resize taking the size of the frame + * decorations into account. + */ + + /* "GetWindowRect" returns the pixel coordinates of the outside of the window */ + GetWindowRect( window->Window.Handle, &windowRect ); + + /* Create rect in FreeGLUT format, (X,Y) topleft outside window, WxH of client area */ + windowRect.right = windowRect.left+width; + windowRect.bottom = windowRect.top+height; + + if (window->Parent == NULL) + /* get the window rect from this to feed to SetWindowPos, correct for window decorations */ + fghComputeWindowRectFromClientArea_QueryWindow(window,&windowRect,TRUE); + else + { + /* correct rect for position client area of parent window + * (SetWindowPos input for child windows is in coordinates + * relative to the parent's client area). + * Child windows don't have decoration, so no need to correct + * for them. + */ + RECT parentRect; + parentRect = fghGetClientArea( window->Parent, FALSE ); + windowRect.left -= parentRect.left; + windowRect.right -= parentRect.left; + windowRect.top -= parentRect.top; + windowRect.bottom -= parentRect.top; + } + + /* Do the actual resizing */ + SetWindowPos( window->Window.Handle, + HWND_TOP, + windowRect.left, windowRect.top, + windowRect.right - windowRect.left, + windowRect.bottom- windowRect.top, + SWP_NOACTIVATE | SWP_NOOWNERZORDER | SWP_NOSENDCHANGING | + SWP_NOZORDER + ); + } +#endif + + if( FETCH_WCB( *window, Reshape ) ) + INVOKE_WCB( *window, Reshape, ( width, height ) ); + else + { + fgSetWindow( window ); + glViewport( 0, 0, width, height ); + } + + /* + * Force a window redraw. In Windows at least this is only a partial + * solution: if the window is increasing in size in either dimension, + * the already-drawn part does not get drawn again and things look funny. + * But without this we get this bad behaviour whenever we resize the + * window. + */ + window->State.Redisplay = GL_TRUE; + + if( window->IsMenu ) + fgSetWindow( current_window ); +} + +/* + * Calls a window's redraw method. This is used when + * a redraw is forced by the incoming window messages. + */ +static void fghRedrawWindow ( SFG_Window *window ) +{ + SFG_Window *current_window = fgStructure.CurrentWindow; + + freeglut_return_if_fail( window ); + freeglut_return_if_fail( FETCH_WCB ( *window, Display ) ); + + window->State.Redisplay = GL_FALSE; + + freeglut_return_if_fail( window->State.Visible ); + + fgSetWindow( window ); + + if( window->State.NeedToResize ) + { + fghReshapeWindow( + window, + window->State.Width, + window->State.Height + ); + + window->State.NeedToResize = GL_FALSE; + } + + INVOKE_WCB( *window, Display, ( ) ); + + fgSetWindow( current_window ); +} + +/* + * A static helper function to execute display callback for a window + */ +static void fghcbDisplayWindow( SFG_Window *window, + SFG_Enumerator *enumerator ) +{ + if( window->State.Redisplay && + window->State.Visible ) + { + window->State.Redisplay = GL_FALSE; + +#if TARGET_HOST_POSIX_X11 + fghRedrawWindow ( window ) ; +#elif TARGET_HOST_MS_WINDOWS + + RedrawWindow( + window->Window.Handle, NULL, NULL, + RDW_NOERASE | RDW_INTERNALPAINT | RDW_INVALIDATE | RDW_UPDATENOW + ); +#endif + } + + fgEnumSubWindows( window, fghcbDisplayWindow, enumerator ); +} + +/* + * Make all windows perform a display call + */ +static void fghDisplayAll( void ) +{ + SFG_Enumerator enumerator; + + enumerator.found = GL_FALSE; + enumerator.data = NULL; + + fgEnumWindows( fghcbDisplayWindow, &enumerator ); +} + +/* + * Window enumerator callback to check for the joystick polling code + */ +static void fghcbCheckJoystickPolls( SFG_Window *window, + SFG_Enumerator *enumerator ) +{ + long int checkTime = fgElapsedTime( ); + + if( window->State.JoystickLastPoll + window->State.JoystickPollRate <= + checkTime ) + { +#if !defined(_WIN32_WCE) + fgJoystickPollWindow( window ); +#endif /* !defined(_WIN32_WCE) */ + window->State.JoystickLastPoll = checkTime; + } + + fgEnumSubWindows( window, fghcbCheckJoystickPolls, enumerator ); +} + +/* + * Check all windows for joystick polling + */ +static void fghCheckJoystickPolls( void ) +{ + SFG_Enumerator enumerator; + + enumerator.found = GL_FALSE; + enumerator.data = NULL; + + fgEnumWindows( fghcbCheckJoystickPolls, &enumerator ); +} + +/* + * Check the global timers + */ +static void fghCheckTimers( void ) +{ + long checkTime = fgElapsedTime( ); + + while( fgState.Timers.First ) + { + SFG_Timer *timer = fgState.Timers.First; + + if( timer->TriggerTime > checkTime ) + break; + + fgListRemove( &fgState.Timers, &timer->Node ); + fgListAppend( &fgState.FreeTimers, &timer->Node ); + + timer->Callback( timer->ID ); + } +} + + +/* Platform-dependent time in milliseconds, as an unsigned 32-bit integer. + * This value wraps every 49.7 days, but integer overflows cancel + * when subtracting an initial start time, unless the total time exceeds + * 32-bit, where the GLUT API return value is also overflowed. + */ +unsigned long fgSystemTime(void) { +#if TARGET_HOST_SOLARIS || HAVE_GETTIMEOFDAY + struct timeval now; + gettimeofday( &now, NULL ); + return now.tv_usec/1000 + now.tv_sec*1000; +#elif TARGET_HOST_MS_WINDOWS +# if defined(_WIN32_WCE) + return GetTickCount(); +# else + return timeGetTime(); +# endif +#endif +} + +/* + * Elapsed Time + */ +long fgElapsedTime( void ) +{ + return (long) (fgSystemTime() - fgState.Time); +} + +/* + * Error Messages. + */ +void fgError( const char *fmt, ... ) +{ + va_list ap; + + if (fgState.ErrorFunc) { + + va_start( ap, fmt ); + + /* call user set error handler here */ + fgState.ErrorFunc(fmt, ap); + + va_end( ap ); + + } else { + + va_start( ap, fmt ); + + fprintf( stderr, "freeglut "); + if( fgState.ProgramName ) + fprintf( stderr, "(%s): ", fgState.ProgramName ); + VFPRINTF( stderr, fmt, ap ); + fprintf( stderr, "\n" ); + + va_end( ap ); + + if ( fgState.Initialised ) + fgDeinitialize (); + + exit( 1 ); + } +} + +void fgWarning( const char *fmt, ... ) +{ + va_list ap; + + if (fgState.WarningFunc) { + + va_start( ap, fmt ); + + /* call user set warning handler here */ + fgState.WarningFunc(fmt, ap); + + va_end( ap ); + + } else { + + va_start( ap, fmt ); + + fprintf( stderr, "freeglut "); + if( fgState.ProgramName ) + fprintf( stderr, "(%s): ", fgState.ProgramName ); + VFPRINTF( stderr, fmt, ap ); + fprintf( stderr, "\n" ); + + va_end( ap ); + } +} + + +/* + * Indicates whether Joystick events are being used by ANY window. + * + * The current mechanism is to walk all of the windows and ask if + * there is a joystick callback. We have a short-circuit early + * return if we find any joystick handler registered. + * + * The real way to do this is to make use of the glutTimer() API + * to more cleanly re-implement the joystick API. Then, this code + * and all other "joystick timer" code can be yanked. + * + */ +static void fghCheckJoystickCallback( SFG_Window* w, SFG_Enumerator* e) +{ + if( FETCH_WCB( *w, Joystick ) ) + { + e->found = GL_TRUE; + e->data = w; + } + fgEnumSubWindows( w, fghCheckJoystickCallback, e ); +} +static int fghHaveJoystick( void ) +{ + SFG_Enumerator enumerator; + + enumerator.found = GL_FALSE; + enumerator.data = NULL; + fgEnumWindows( fghCheckJoystickCallback, &enumerator ); + return !!enumerator.data; +} +static void fghHavePendingRedisplaysCallback( SFG_Window* w, SFG_Enumerator* e) +{ + if( w->State.Redisplay && w->State.Visible ) + { + e->found = GL_TRUE; + e->data = w; + } + fgEnumSubWindows( w, fghHavePendingRedisplaysCallback, e ); +} +static int fghHavePendingRedisplays (void) +{ + SFG_Enumerator enumerator; + + enumerator.found = GL_FALSE; + enumerator.data = NULL; + fgEnumWindows( fghHavePendingRedisplaysCallback, &enumerator ); + return !!enumerator.data; +} +/* + * Returns the number of GLUT ticks (milliseconds) till the next timer event. + */ +static long fghNextTimer( void ) +{ + long ret = INT_MAX; + SFG_Timer *timer = fgState.Timers.First; + + if( timer ) + ret = timer->TriggerTime - fgElapsedTime(); + if( ret < 0 ) + ret = 0; + + return ret; +} +/* + * Does the magic required to relinquish the CPU until something interesting + * happens. + */ +static void fghSleepForEvents( void ) +{ + long msec; + + if( fgState.IdleCallback || fghHavePendingRedisplays( ) ) + return; + + msec = fghNextTimer( ); + /* XXX Use GLUT timers for joysticks... */ + /* XXX Dumb; forces granularity to .01sec */ + if( fghHaveJoystick( ) && ( msec > 10 ) ) + msec = 10; + +#if TARGET_HOST_POSIX_X11 + /* + * Possibly due to aggressive use of XFlush() and friends, + * it is possible to have our socket drained but still have + * unprocessed events. (Or, this may just be normal with + * X, anyway?) We do non-trivial processing of X events + * after the event-reading loop, in any case, so we + * need to allow that we may have an empty socket but non- + * empty event queue. + */ + if( ! XPending( fgDisplay.Display ) ) + { + fd_set fdset; + int err; + int socket; + struct timeval wait; + + socket = ConnectionNumber( fgDisplay.Display ); + FD_ZERO( &fdset ); + FD_SET( socket, &fdset ); + wait.tv_sec = msec / 1000; + wait.tv_usec = (msec % 1000) * 1000; + err = select( socket+1, &fdset, NULL, NULL, &wait ); + +#ifdef HAVE_ERRNO_H + if( ( -1 == err ) && ( errno != EINTR ) ) + fgWarning ( "freeglut select() error: %d", errno ); +#endif + } +#elif TARGET_HOST_MS_WINDOWS + MsgWaitForMultipleObjects( 0, NULL, FALSE, msec, QS_ALLINPUT ); +#endif +} + +#if TARGET_HOST_POSIX_X11 +/* + * Returns GLUT modifier mask for the state field of an X11 event. + */ +int fghGetXModifiers( int state ) +{ + int ret = 0; + + if( state & ( ShiftMask | LockMask ) ) + ret |= GLUT_ACTIVE_SHIFT; + if( state & ControlMask ) + ret |= GLUT_ACTIVE_CTRL; + if( state & Mod1Mask ) + ret |= GLUT_ACTIVE_ALT; + + return ret; +} +#endif + + +#if TARGET_HOST_POSIX_X11 && _DEBUG + +static const char* fghTypeToString( int type ) +{ + switch( type ) { + case KeyPress: return "KeyPress"; + case KeyRelease: return "KeyRelease"; + case ButtonPress: return "ButtonPress"; + case ButtonRelease: return "ButtonRelease"; + case MotionNotify: return "MotionNotify"; + case EnterNotify: return "EnterNotify"; + case LeaveNotify: return "LeaveNotify"; + case FocusIn: return "FocusIn"; + case FocusOut: return "FocusOut"; + case KeymapNotify: return "KeymapNotify"; + case Expose: return "Expose"; + case GraphicsExpose: return "GraphicsExpose"; + case NoExpose: return "NoExpose"; + case VisibilityNotify: return "VisibilityNotify"; + case CreateNotify: return "CreateNotify"; + case DestroyNotify: return "DestroyNotify"; + case UnmapNotify: return "UnmapNotify"; + case MapNotify: return "MapNotify"; + case MapRequest: return "MapRequest"; + case ReparentNotify: return "ReparentNotify"; + case ConfigureNotify: return "ConfigureNotify"; + case ConfigureRequest: return "ConfigureRequest"; + case GravityNotify: return "GravityNotify"; + case ResizeRequest: return "ResizeRequest"; + case CirculateNotify: return "CirculateNotify"; + case CirculateRequest: return "CirculateRequest"; + case PropertyNotify: return "PropertyNotify"; + case SelectionClear: return "SelectionClear"; + case SelectionRequest: return "SelectionRequest"; + case SelectionNotify: return "SelectionNotify"; + case ColormapNotify: return "ColormapNotify"; + case ClientMessage: return "ClientMessage"; + case MappingNotify: return "MappingNotify"; + default: return "UNKNOWN"; + } +} + +static const char* fghBoolToString( Bool b ) +{ + return b == False ? "False" : "True"; +} + +static const char* fghNotifyHintToString( char is_hint ) +{ + switch( is_hint ) { + case NotifyNormal: return "NotifyNormal"; + case NotifyHint: return "NotifyHint"; + default: return "UNKNOWN"; + } +} + +static const char* fghNotifyModeToString( int mode ) +{ + switch( mode ) { + case NotifyNormal: return "NotifyNormal"; + case NotifyGrab: return "NotifyGrab"; + case NotifyUngrab: return "NotifyUngrab"; + case NotifyWhileGrabbed: return "NotifyWhileGrabbed"; + default: return "UNKNOWN"; + } +} + +static const char* fghNotifyDetailToString( int detail ) +{ + switch( detail ) { + case NotifyAncestor: return "NotifyAncestor"; + case NotifyVirtual: return "NotifyVirtual"; + case NotifyInferior: return "NotifyInferior"; + case NotifyNonlinear: return "NotifyNonlinear"; + case NotifyNonlinearVirtual: return "NotifyNonlinearVirtual"; + case NotifyPointer: return "NotifyPointer"; + case NotifyPointerRoot: return "NotifyPointerRoot"; + case NotifyDetailNone: return "NotifyDetailNone"; + default: return "UNKNOWN"; + } +} + +static const char* fghVisibilityToString( int state ) { + switch( state ) { + case VisibilityUnobscured: return "VisibilityUnobscured"; + case VisibilityPartiallyObscured: return "VisibilityPartiallyObscured"; + case VisibilityFullyObscured: return "VisibilityFullyObscured"; + default: return "UNKNOWN"; + } +} + +static const char* fghConfigureDetailToString( int detail ) +{ + switch( detail ) { + case Above: return "Above"; + case Below: return "Below"; + case TopIf: return "TopIf"; + case BottomIf: return "BottomIf"; + case Opposite: return "Opposite"; + default: return "UNKNOWN"; + } +} + +static const char* fghPlaceToString( int place ) +{ + switch( place ) { + case PlaceOnTop: return "PlaceOnTop"; + case PlaceOnBottom: return "PlaceOnBottom"; + default: return "UNKNOWN"; + } +} + +static const char* fghMappingRequestToString( int request ) +{ + switch( request ) { + case MappingModifier: return "MappingModifier"; + case MappingKeyboard: return "MappingKeyboard"; + case MappingPointer: return "MappingPointer"; + default: return "UNKNOWN"; + } +} + +static const char* fghPropertyStateToString( int state ) +{ + switch( state ) { + case PropertyNewValue: return "PropertyNewValue"; + case PropertyDelete: return "PropertyDelete"; + default: return "UNKNOWN"; + } +} + +static const char* fghColormapStateToString( int state ) +{ + switch( state ) { + case ColormapUninstalled: return "ColormapUninstalled"; + case ColormapInstalled: return "ColormapInstalled"; + default: return "UNKNOWN"; + } +} + +static void fghPrintEvent( XEvent *event ) +{ + switch( event->type ) { + + case KeyPress: + case KeyRelease: { + XKeyEvent *e = &event->xkey; + fgWarning( "%s: window=0x%x, root=0x%x, subwindow=0x%x, time=%lu, " + "(x,y)=(%d,%d), (x_root,y_root)=(%d,%d), state=0x%x, " + "keycode=%u, same_screen=%s", fghTypeToString( e->type ), + e->window, e->root, e->subwindow, (unsigned long)e->time, + e->x, e->y, e->x_root, e->y_root, e->state, e->keycode, + fghBoolToString( e->same_screen ) ); + break; + } + + case ButtonPress: + case ButtonRelease: { + XButtonEvent *e = &event->xbutton; + fgWarning( "%s: window=0x%x, root=0x%x, subwindow=0x%x, time=%lu, " + "(x,y)=(%d,%d), (x_root,y_root)=(%d,%d), state=0x%x, " + "button=%u, same_screen=%d", fghTypeToString( e->type ), + e->window, e->root, e->subwindow, (unsigned long)e->time, + e->x, e->y, e->x_root, e->y_root, e->state, e->button, + fghBoolToString( e->same_screen ) ); + break; + } + + case MotionNotify: { + XMotionEvent *e = &event->xmotion; + fgWarning( "%s: window=0x%x, root=0x%x, subwindow=0x%x, time=%lu, " + "(x,y)=(%d,%d), (x_root,y_root)=(%d,%d), state=0x%x, " + "is_hint=%s, same_screen=%d", fghTypeToString( e->type ), + e->window, e->root, e->subwindow, (unsigned long)e->time, + e->x, e->y, e->x_root, e->y_root, e->state, + fghNotifyHintToString( e->is_hint ), + fghBoolToString( e->same_screen ) ); + break; + } + + case EnterNotify: + case LeaveNotify: { + XCrossingEvent *e = &event->xcrossing; + fgWarning( "%s: window=0x%x, root=0x%x, subwindow=0x%x, time=%lu, " + "(x,y)=(%d,%d), mode=%s, detail=%s, same_screen=%d, " + "focus=%d, state=0x%x", fghTypeToString( e->type ), + e->window, e->root, e->subwindow, (unsigned long)e->time, + e->x, e->y, fghNotifyModeToString( e->mode ), + fghNotifyDetailToString( e->detail ), (int)e->same_screen, + (int)e->focus, e->state ); + break; + } + + case FocusIn: + case FocusOut: { + XFocusChangeEvent *e = &event->xfocus; + fgWarning( "%s: window=0x%x, mode=%s, detail=%s", + fghTypeToString( e->type ), e->window, + fghNotifyModeToString( e->mode ), + fghNotifyDetailToString( e->detail ) ); + break; + } + + case KeymapNotify: { + XKeymapEvent *e = &event->xkeymap; + char buf[32 * 2 + 1]; + int i; + for ( i = 0; i < 32; i++ ) { + snprintf( &buf[ i * 2 ], sizeof( buf ) - i * 2, + "%02x", e->key_vector[ i ] ); + } + buf[ i ] = '\0'; + fgWarning( "%s: window=0x%x, %s", fghTypeToString( e->type ), e->window, + buf ); + break; + } + + case Expose: { + XExposeEvent *e = &event->xexpose; + fgWarning( "%s: window=0x%x, (x,y)=(%d,%d), (width,height)=(%d,%d), " + "count=%d", fghTypeToString( e->type ), e->window, e->x, + e->y, e->width, e->height, e->count ); + break; + } + + case GraphicsExpose: { + XGraphicsExposeEvent *e = &event->xgraphicsexpose; + fgWarning( "%s: drawable=0x%x, (x,y)=(%d,%d), (width,height)=(%d,%d), " + "count=%d, (major_code,minor_code)=(%d,%d)", + fghTypeToString( e->type ), e->drawable, e->x, e->y, + e->width, e->height, e->count, e->major_code, + e->minor_code ); + break; + } + + case NoExpose: { + XNoExposeEvent *e = &event->xnoexpose; + fgWarning( "%s: drawable=0x%x, (major_code,minor_code)=(%d,%d)", + fghTypeToString( e->type ), e->drawable, e->major_code, + e->minor_code ); + break; + } + + case VisibilityNotify: { + XVisibilityEvent *e = &event->xvisibility; + fgWarning( "%s: window=0x%x, state=%s", fghTypeToString( e->type ), + e->window, fghVisibilityToString( e->state) ); + break; + } + + case CreateNotify: { + XCreateWindowEvent *e = &event->xcreatewindow; + fgWarning( "%s: (x,y)=(%d,%d), (width,height)=(%d,%d), border_width=%d, " + "window=0x%x, override_redirect=%s", + fghTypeToString( e->type ), e->x, e->y, e->width, e->height, + e->border_width, e->window, + fghBoolToString( e->override_redirect ) ); + break; + } + + case DestroyNotify: { + XDestroyWindowEvent *e = &event->xdestroywindow; + fgWarning( "%s: event=0x%x, window=0x%x", + fghTypeToString( e->type ), e->event, e->window ); + break; + } + + case UnmapNotify: { + XUnmapEvent *e = &event->xunmap; + fgWarning( "%s: event=0x%x, window=0x%x, from_configure=%s", + fghTypeToString( e->type ), e->event, e->window, + fghBoolToString( e->from_configure ) ); + break; + } + + case MapNotify: { + XMapEvent *e = &event->xmap; + fgWarning( "%s: event=0x%x, window=0x%x, override_redirect=%s", + fghTypeToString( e->type ), e->event, e->window, + fghBoolToString( e->override_redirect ) ); + break; + } + + case MapRequest: { + XMapRequestEvent *e = &event->xmaprequest; + fgWarning( "%s: parent=0x%x, window=0x%x", + fghTypeToString( event->type ), e->parent, e->window ); + break; + } + + case ReparentNotify: { + XReparentEvent *e = &event->xreparent; + fgWarning( "%s: event=0x%x, window=0x%x, parent=0x%x, (x,y)=(%d,%d), " + "override_redirect=%s", fghTypeToString( e->type ), + e->event, e->window, e->parent, e->x, e->y, + fghBoolToString( e->override_redirect ) ); + break; + } + + case ConfigureNotify: { + XConfigureEvent *e = &event->xconfigure; + fgWarning( "%s: event=0x%x, window=0x%x, (x,y)=(%d,%d), " + "(width,height)=(%d,%d), border_width=%d, above=0x%x, " + "override_redirect=%s", fghTypeToString( e->type ), e->event, + e->window, e->x, e->y, e->width, e->height, e->border_width, + e->above, fghBoolToString( e->override_redirect ) ); + break; + } + + case ConfigureRequest: { + XConfigureRequestEvent *e = &event->xconfigurerequest; + fgWarning( "%s: parent=0x%x, window=0x%x, (x,y)=(%d,%d), " + "(width,height)=(%d,%d), border_width=%d, above=0x%x, " + "detail=%s, value_mask=%lx", fghTypeToString( e->type ), + e->parent, e->window, e->x, e->y, e->width, e->height, + e->border_width, e->above, + fghConfigureDetailToString( e->detail ), e->value_mask ); + break; + } + + case GravityNotify: { + XGravityEvent *e = &event->xgravity; + fgWarning( "%s: event=0x%x, window=0x%x, (x,y)=(%d,%d)", + fghTypeToString( e->type ), e->event, e->window, e->x, e->y ); + break; + } + + case ResizeRequest: { + XResizeRequestEvent *e = &event->xresizerequest; + fgWarning( "%s: window=0x%x, (width,height)=(%d,%d)", + fghTypeToString( e->type ), e->window, e->width, e->height ); + break; + } + + case CirculateNotify: { + XCirculateEvent *e = &event->xcirculate; + fgWarning( "%s: event=0x%x, window=0x%x, place=%s", + fghTypeToString( e->type ), e->event, e->window, + fghPlaceToString( e->place ) ); + break; + } + + case CirculateRequest: { + XCirculateRequestEvent *e = &event->xcirculaterequest; + fgWarning( "%s: parent=0x%x, window=0x%x, place=%s", + fghTypeToString( e->type ), e->parent, e->window, + fghPlaceToString( e->place ) ); + break; + } + + case PropertyNotify: { + XPropertyEvent *e = &event->xproperty; + fgWarning( "%s: window=0x%x, atom=%lu, time=%lu, state=%s", + fghTypeToString( e->type ), e->window, + (unsigned long)e->atom, (unsigned long)e->time, + fghPropertyStateToString( e->state ) ); + break; + } + + case SelectionClear: { + XSelectionClearEvent *e = &event->xselectionclear; + fgWarning( "%s: window=0x%x, selection=%lu, time=%lu", + fghTypeToString( e->type ), e->window, + (unsigned long)e->selection, (unsigned long)e->time ); + break; + } + + case SelectionRequest: { + XSelectionRequestEvent *e = &event->xselectionrequest; + fgWarning( "%s: owner=0x%x, requestor=0x%x, selection=0x%x, " + "target=0x%x, property=%lu, time=%lu", + fghTypeToString( e->type ), e->owner, e->requestor, + (unsigned long)e->selection, (unsigned long)e->target, + (unsigned long)e->property, (unsigned long)e->time ); + break; + } + + case SelectionNotify: { + XSelectionEvent *e = &event->xselection; + fgWarning( "%s: requestor=0x%x, selection=0x%x, target=0x%x, " + "property=%lu, time=%lu", fghTypeToString( e->type ), + e->requestor, (unsigned long)e->selection, + (unsigned long)e->target, (unsigned long)e->property, + (unsigned long)e->time ); + break; + } + + case ColormapNotify: { + XColormapEvent *e = &event->xcolormap; + fgWarning( "%s: window=0x%x, colormap=%lu, new=%s, state=%s", + fghTypeToString( e->type ), e->window, + (unsigned long)e->colormap, fghBoolToString( e->new ), + fghColormapStateToString( e->state ) ); + break; + } + + case ClientMessage: { + XClientMessageEvent *e = &event->xclient; + char buf[ 61 ]; + char* p = buf; + char* end = buf + sizeof( buf ); + int i; + switch( e->format ) { + case 8: + for ( i = 0; i < 20; i++, p += 3 ) { + snprintf( p, end - p, " %02x", e->data.b[ i ] ); + } + break; + case 16: + for ( i = 0; i < 10; i++, p += 5 ) { + snprintf( p, end - p, " %04x", e->data.s[ i ] ); + } + break; + case 32: + for ( i = 0; i < 5; i++, p += 9 ) { + snprintf( p, end - p, " %08lx", e->data.l[ i ] ); + } + break; + } + *p = '\0'; + fgWarning( "%s: window=0x%x, message_type=%lu, format=%d, data=(%s )", + fghTypeToString( e->type ), e->window, + (unsigned long)e->message_type, e->format, buf ); + break; + } + + case MappingNotify: { + XMappingEvent *e = &event->xmapping; + fgWarning( "%s: window=0x%x, request=%s, first_keycode=%d, count=%d", + fghTypeToString( e->type ), e->window, + fghMappingRequestToString( e->request ), e->first_keycode, + e->count ); + break; + } + + default: { + fgWarning( "%s", fghTypeToString( event->type ) ); + break; + } + } +} + +#endif + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * Executes a single iteration in the freeglut processing loop. + */ +void FGAPIENTRY glutMainLoopEvent( void ) +{ +#if TARGET_HOST_POSIX_X11 + SFG_Window* window; + XEvent event; + + /* This code was repeated constantly, so here it goes into a definition: */ +#define GETWINDOW(a) \ + window = fgWindowByHandle( event.a.window ); \ + if( window == NULL ) \ + break; + +#define GETMOUSE(a) \ + window->State.MouseX = event.a.x; \ + window->State.MouseY = event.a.y; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMainLoopEvent" ); + + while( XPending( fgDisplay.Display ) ) + { + XNextEvent( fgDisplay.Display, &event ); +#if _DEBUG + fghPrintEvent( &event ); +#endif + + switch( event.type ) + { + case ClientMessage: + if(fgIsSpaceballXEvent(&event)) { + fgSpaceballHandleXEvent(&event); + break; + } + /* Destroy the window when the WM_DELETE_WINDOW message arrives */ + if( (Atom) event.xclient.data.l[ 0 ] == fgDisplay.DeleteWindow ) + { + GETWINDOW( xclient ); + + fgDestroyWindow ( window ); + + if( fgState.ActionOnWindowClose == GLUT_ACTION_EXIT ) + { + fgDeinitialize( ); + exit( 0 ); + } + else if( fgState.ActionOnWindowClose == GLUT_ACTION_GLUTMAINLOOP_RETURNS ) + fgState.ExecState = GLUT_EXEC_STATE_STOP; + + return; + } + break; + + /* + * CreateNotify causes a configure-event so that sub-windows are + * handled compatibly with GLUT. Otherwise, your sub-windows + * (in freeglut only) will not get an initial reshape event, + * which can break things. + * + * GLUT presumably does this because it generally tries to treat + * sub-windows the same as windows. + */ + case CreateNotify: + case ConfigureNotify: + { + int width, height; + if( event.type == CreateNotify ) { + GETWINDOW( xcreatewindow ); + width = event.xcreatewindow.width; + height = event.xcreatewindow.height; + } else { + GETWINDOW( xconfigure ); + width = event.xconfigure.width; + height = event.xconfigure.height; + } + + if( ( width != window->State.OldWidth ) || + ( height != window->State.OldHeight ) ) + { + SFG_Window *current_window = fgStructure.CurrentWindow; + + window->State.OldWidth = width; + window->State.OldHeight = height; + if( FETCH_WCB( *window, Reshape ) ) + INVOKE_WCB( *window, Reshape, ( width, height ) ); + else + { + fgSetWindow( window ); + glViewport( 0, 0, width, height ); + } + glutPostRedisplay( ); + if( window->IsMenu ) + fgSetWindow( current_window ); + } + } + break; + + case DestroyNotify: + /* + * This is sent to confirm the XDestroyWindow call. + * + * XXX WHY is this commented out? Should we re-enable it? + */ + /* fgAddToWindowDestroyList ( window ); */ + break; + + case Expose: + /* + * We are too dumb to process partial exposes... + * + * XXX Well, we could do it. However, it seems to only + * XXX be potentially useful for single-buffered (since + * XXX double-buffered does not respect viewport when we + * XXX do a buffer-swap). + * + */ + if( event.xexpose.count == 0 ) + { + GETWINDOW( xexpose ); + window->State.Redisplay = GL_TRUE; + } + break; + + case MapNotify: + break; + + case UnmapNotify: + /* We get this when iconifying a window. */ + GETWINDOW( xunmap ); + INVOKE_WCB( *window, WindowStatus, ( GLUT_HIDDEN ) ); + window->State.Visible = GL_FALSE; + break; + + case MappingNotify: + /* + * Have the client's keyboard knowledge updated (xlib.ps, + * page 206, says that's a good thing to do) + */ + XRefreshKeyboardMapping( (XMappingEvent *) &event ); + break; + + case VisibilityNotify: + { + /* + * Sending this event, the X server can notify us that the window + * has just acquired one of the three possible visibility states: + * VisibilityUnobscured, VisibilityPartiallyObscured or + * VisibilityFullyObscured. Note that we DO NOT receive a + * VisibilityNotify event when iconifying a window, we only get an + * UnmapNotify then. + */ + GETWINDOW( xvisibility ); + switch( event.xvisibility.state ) + { + case VisibilityUnobscured: + INVOKE_WCB( *window, WindowStatus, ( GLUT_FULLY_RETAINED ) ); + window->State.Visible = GL_TRUE; + break; + + case VisibilityPartiallyObscured: + INVOKE_WCB( *window, WindowStatus, + ( GLUT_PARTIALLY_RETAINED ) ); + window->State.Visible = GL_TRUE; + break; + + case VisibilityFullyObscured: + INVOKE_WCB( *window, WindowStatus, ( GLUT_FULLY_COVERED ) ); + window->State.Visible = GL_FALSE; + break; + + default: + fgWarning( "Unknown X visibility state: %d", + event.xvisibility.state ); + break; + } + } + break; + + case EnterNotify: + case LeaveNotify: + GETWINDOW( xcrossing ); + GETMOUSE( xcrossing ); + if( ( event.type == LeaveNotify ) && window->IsMenu && + window->ActiveMenu && window->ActiveMenu->IsActive ) + fgUpdateMenuHighlight( window->ActiveMenu ); + + INVOKE_WCB( *window, Entry, ( ( EnterNotify == event.type ) ? + GLUT_ENTERED : + GLUT_LEFT ) ); + break; + + case MotionNotify: + { + GETWINDOW( xmotion ); + GETMOUSE( xmotion ); + + if( window->ActiveMenu ) + { + if( window == window->ActiveMenu->ParentWindow ) + { + window->ActiveMenu->Window->State.MouseX = + event.xmotion.x_root - window->ActiveMenu->X; + window->ActiveMenu->Window->State.MouseY = + event.xmotion.y_root - window->ActiveMenu->Y; + } + + fgUpdateMenuHighlight( window->ActiveMenu ); + + break; + } + + /* + * XXX For more than 5 buttons, just check {event.xmotion.state}, + * XXX rather than a host of bit-masks? Or maybe we need to + * XXX track ButtonPress/ButtonRelease events in our own + * XXX bit-mask? + */ + fgState.Modifiers = fghGetXModifiers( event.xmotion.state ); + if ( event.xmotion.state & ( Button1Mask | Button2Mask | Button3Mask | Button4Mask | Button5Mask ) ) { + INVOKE_WCB( *window, Motion, ( event.xmotion.x, + event.xmotion.y ) ); + } else { + INVOKE_WCB( *window, Passive, ( event.xmotion.x, + event.xmotion.y ) ); + } + fgState.Modifiers = INVALID_MODIFIERS; + } + break; + + case ButtonRelease: + case ButtonPress: + { + GLboolean pressed = GL_TRUE; + int button; + + if( event.type == ButtonRelease ) + pressed = GL_FALSE ; + + /* + * A mouse button has been pressed or released. Traditionally, + * break if the window was found within the freeglut structures. + */ + GETWINDOW( xbutton ); + GETMOUSE( xbutton ); + + /* + * An X button (at least in XFree86) is numbered from 1. + * A GLUT button is numbered from 0. + * Old GLUT passed through buttons other than just the first + * three, though it only gave symbolic names and official + * support to the first three. + */ + button = event.xbutton.button - 1; + + /* + * Do not execute the application's mouse callback if a menu + * is hooked to this button. In that case an appropriate + * private call should be generated. + */ + if( fgCheckActiveMenu( window, button, pressed, + event.xbutton.x_root, event.xbutton.y_root ) ) + break; + + /* + * Check if there is a mouse or mouse wheel callback hooked to the + * window + */ + if( ! FETCH_WCB( *window, Mouse ) && + ! FETCH_WCB( *window, MouseWheel ) ) + break; + + fgState.Modifiers = fghGetXModifiers( event.xbutton.state ); + + /* Finally execute the mouse or mouse wheel callback */ + if( ( button < glutDeviceGet ( GLUT_NUM_MOUSE_BUTTONS ) ) || ( ! FETCH_WCB( *window, MouseWheel ) ) ) + INVOKE_WCB( *window, Mouse, ( button, + pressed ? GLUT_DOWN : GLUT_UP, + event.xbutton.x, + event.xbutton.y ) + ); + else + { + /* + * Map 4 and 5 to wheel zero; EVEN to +1, ODD to -1 + * " 6 and 7 " " one; ... + * + * XXX This *should* be behind some variables/macros, + * XXX since the order and numbering isn't certain + * XXX See XFree86 configuration docs (even back in the + * XXX 3.x days, and especially with 4.x). + * + * XXX Note that {button} has already been decremented + * XXX in mapping from X button numbering to GLUT. + * + * XXX Should add support for partial wheel turns as Windows does -- 5/27/11 + */ + int wheel_number = (button - glutDeviceGet ( GLUT_NUM_MOUSE_BUTTONS )) / 2; + int direction = -1; + if( button % 2 ) + direction = 1; + + if( pressed ) + INVOKE_WCB( *window, MouseWheel, ( wheel_number, + direction, + event.xbutton.x, + event.xbutton.y ) + ); + } + fgState.Modifiers = INVALID_MODIFIERS; + } + break; + + case KeyRelease: + case KeyPress: + { + FGCBKeyboard keyboard_cb; + FGCBSpecial special_cb; + + GETWINDOW( xkey ); + GETMOUSE( xkey ); + + /* Detect auto repeated keys, if configured globally or per-window */ + + if ( fgState.KeyRepeat==GLUT_KEY_REPEAT_OFF || window->State.IgnoreKeyRepeat==GL_TRUE ) + { + if (event.type==KeyRelease) + { + /* + * Look at X11 keystate to detect repeat mode. + * While X11 says the key is actually held down, we'll ignore KeyRelease/KeyPress pairs. + */ + + char keys[32]; + XQueryKeymap( fgDisplay.Display, keys ); /* Look at X11 keystate to detect repeat mode */ + + if ( event.xkey.keycode<256 ) /* XQueryKeymap is limited to 256 keycodes */ + { + if ( keys[event.xkey.keycode>>3] & (1<<(event.xkey.keycode%8)) ) + window->State.KeyRepeating = GL_TRUE; + else + window->State.KeyRepeating = GL_FALSE; + } + } + } + else + window->State.KeyRepeating = GL_FALSE; + + /* Cease processing this event if it is auto repeated */ + + if (window->State.KeyRepeating) + { + if (event.type == KeyPress) window->State.KeyRepeating = GL_FALSE; + break; + } + + if( event.type == KeyPress ) + { + keyboard_cb = (FGCBKeyboard)( FETCH_WCB( *window, Keyboard )); + special_cb = (FGCBSpecial) ( FETCH_WCB( *window, Special )); + } + else + { + keyboard_cb = (FGCBKeyboard)( FETCH_WCB( *window, KeyboardUp )); + special_cb = (FGCBSpecial) ( FETCH_WCB( *window, SpecialUp )); + } + + /* Is there a keyboard/special callback hooked for this window? */ + if( keyboard_cb || special_cb ) + { + XComposeStatus composeStatus; + char asciiCode[ 32 ]; + KeySym keySym; + int len; + + /* Check for the ASCII/KeySym codes associated with the event: */ + len = XLookupString( &event.xkey, asciiCode, sizeof(asciiCode), + &keySym, &composeStatus + ); + + /* GLUT API tells us to have two separate callbacks... */ + if( len > 0 ) + { + /* ...one for the ASCII translateable keypresses... */ + if( keyboard_cb ) + { + fgSetWindow( window ); + fgState.Modifiers = fghGetXModifiers( event.xkey.state ); + keyboard_cb( asciiCode[ 0 ], + event.xkey.x, event.xkey.y + ); + fgState.Modifiers = INVALID_MODIFIERS; + } + } + else + { + int special = -1; + + /* + * ...and one for all the others, which need to be + * translated to GLUT_KEY_Xs... + */ + switch( keySym ) + { + case XK_F1: special = GLUT_KEY_F1; break; + case XK_F2: special = GLUT_KEY_F2; break; + case XK_F3: special = GLUT_KEY_F3; break; + case XK_F4: special = GLUT_KEY_F4; break; + case XK_F5: special = GLUT_KEY_F5; break; + case XK_F6: special = GLUT_KEY_F6; break; + case XK_F7: special = GLUT_KEY_F7; break; + case XK_F8: special = GLUT_KEY_F8; break; + case XK_F9: special = GLUT_KEY_F9; break; + case XK_F10: special = GLUT_KEY_F10; break; + case XK_F11: special = GLUT_KEY_F11; break; + case XK_F12: special = GLUT_KEY_F12; break; + + case XK_KP_Left: + case XK_Left: special = GLUT_KEY_LEFT; break; + case XK_KP_Right: + case XK_Right: special = GLUT_KEY_RIGHT; break; + case XK_KP_Up: + case XK_Up: special = GLUT_KEY_UP; break; + case XK_KP_Down: + case XK_Down: special = GLUT_KEY_DOWN; break; + + case XK_KP_Prior: + case XK_Prior: special = GLUT_KEY_PAGE_UP; break; + case XK_KP_Next: + case XK_Next: special = GLUT_KEY_PAGE_DOWN; break; + case XK_KP_Home: + case XK_Home: special = GLUT_KEY_HOME; break; + case XK_KP_End: + case XK_End: special = GLUT_KEY_END; break; + case XK_KP_Insert: + case XK_Insert: special = GLUT_KEY_INSERT; break; + + case XK_Num_Lock : special = GLUT_KEY_NUM_LOCK; break; + case XK_KP_Begin : special = GLUT_KEY_BEGIN; break; + case XK_KP_Delete: special = GLUT_KEY_DELETE; break; + + case XK_Shift_L: special = GLUT_KEY_SHIFT_L; break; + case XK_Shift_R: special = GLUT_KEY_SHIFT_R; break; + case XK_Control_L: special = GLUT_KEY_CTRL_L; break; + case XK_Control_R: special = GLUT_KEY_CTRL_R; break; + case XK_Alt_L: special = GLUT_KEY_ALT_L; break; + case XK_Alt_R: special = GLUT_KEY_ALT_R; break; + } + + /* + * Execute the callback (if one has been specified), + * given that the special code seems to be valid... + */ + if( special_cb && (special != -1) ) + { + fgSetWindow( window ); + fgState.Modifiers = fghGetXModifiers( event.xkey.state ); + special_cb( special, event.xkey.x, event.xkey.y ); + fgState.Modifiers = INVALID_MODIFIERS; + } + } + } + } + break; + + case ReparentNotify: + break; /* XXX Should disable this event */ + + /* Not handled */ + case GravityNotify: + break; + + default: + /* enter handling of Extension Events here */ + #ifdef HAVE_X11_EXTENSIONS_XINPUT2_H + fgHandleExtensionEvents( &event ); + #endif + break; + } + } + +#elif TARGET_HOST_MS_WINDOWS + + MSG stMsg; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMainLoopEvent" ); + + while( PeekMessage( &stMsg, NULL, 0, 0, PM_NOREMOVE ) ) + { + if( GetMessage( &stMsg, NULL, 0, 0 ) == 0 ) + { + if( fgState.ActionOnWindowClose == GLUT_ACTION_EXIT ) + { + fgDeinitialize( ); + exit( 0 ); + } + else if( fgState.ActionOnWindowClose == GLUT_ACTION_GLUTMAINLOOP_RETURNS ) + fgState.ExecState = GLUT_EXEC_STATE_STOP; + + return; + } + + TranslateMessage( &stMsg ); + DispatchMessage( &stMsg ); + } +#endif + + if( fgState.Timers.First ) + fghCheckTimers( ); + fghCheckJoystickPolls( ); + fghDisplayAll( ); + + fgCloseWindows( ); +} + +/* + * Enters the freeglut processing loop. + * Stays until the "ExecState" changes to "GLUT_EXEC_STATE_STOP". + */ +void FGAPIENTRY glutMainLoop( void ) +{ + int action; + +#if TARGET_HOST_MS_WINDOWS + SFG_Window *window = (SFG_Window *)fgStructure.Windows.First ; +#endif + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMainLoop" ); + +#if TARGET_HOST_MS_WINDOWS + /* + * Processing before the main loop: If there is a window which is open and + * which has a visibility callback, call it. I know this is an ugly hack, + * but I'm not sure what else to do about it. Ideally we should leave + * something uninitialized in the create window code and initialize it in + * the main loop, and have that initialization create a "WM_ACTIVATE" + * message. Then we would put the visibility callback code in the + * "case WM_ACTIVATE" block below. - John Fay -- 10/24/02 + */ + while( window ) + { + if ( FETCH_WCB( *window, Visibility ) ) + { + SFG_Window *current_window = fgStructure.CurrentWindow ; + + INVOKE_WCB( *window, Visibility, ( window->State.Visible ) ); + fgSetWindow( current_window ); + } + + window = (SFG_Window *)window->Node.Next ; + } +#endif + + fgState.ExecState = GLUT_EXEC_STATE_RUNNING ; + while( fgState.ExecState == GLUT_EXEC_STATE_RUNNING ) + { + SFG_Window *window; + + glutMainLoopEvent( ); + /* + * Step through the list of windows, seeing if there are any + * that are not menus + */ + for( window = ( SFG_Window * )fgStructure.Windows.First; + window; + window = ( SFG_Window * )window->Node.Next ) + if ( ! ( window->IsMenu ) ) + break; + + if( ! window ) + fgState.ExecState = GLUT_EXEC_STATE_STOP; + else + { + if( fgState.IdleCallback ) + { + if( fgStructure.CurrentWindow && + fgStructure.CurrentWindow->IsMenu ) + /* fail safe */ + fgSetWindow( window ); + fgState.IdleCallback( ); + } + + fghSleepForEvents( ); + } + } + + /* + * When this loop terminates, destroy the display, state and structure + * of a freeglut session, so that another glutInit() call can happen + * + * Save the "ActionOnWindowClose" because "fgDeinitialize" resets it. + */ + action = fgState.ActionOnWindowClose; + fgDeinitialize( ); + if( action == GLUT_ACTION_EXIT ) + exit( 0 ); +} + +/* + * Leaves the freeglut processing loop. + */ +void FGAPIENTRY glutLeaveMainLoop( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutLeaveMainLoop" ); + fgState.ExecState = GLUT_EXEC_STATE_STOP ; +} + + +#if TARGET_HOST_MS_WINDOWS +/* + * Determine a GLUT modifer mask based on MS-WINDOWS system info. + */ +static int fghGetWin32Modifiers (void) +{ + return + ( ( ( GetKeyState( VK_LSHIFT ) < 0 ) || + ( GetKeyState( VK_RSHIFT ) < 0 )) ? GLUT_ACTIVE_SHIFT : 0 ) | + ( ( ( GetKeyState( VK_LCONTROL ) < 0 ) || + ( GetKeyState( VK_RCONTROL ) < 0 )) ? GLUT_ACTIVE_CTRL : 0 ) | + ( ( ( GetKeyState( VK_LMENU ) < 0 ) || + ( GetKeyState( VK_RMENU ) < 0 )) ? GLUT_ACTIVE_ALT : 0 ); +} + +/* + * The window procedure for handling Win32 events + */ +LRESULT CALLBACK fgWindowProc( HWND hWnd, UINT uMsg, WPARAM wParam, + LPARAM lParam ) +{ + static unsigned char lControl = 0, rControl = 0, lShift = 0, + rShift = 0, lAlt = 0, rAlt = 0; + + SFG_Window* window; + PAINTSTRUCT ps; + LRESULT lRet = 1; + + FREEGLUT_INTERNAL_ERROR_EXIT_IF_NOT_INITIALISED ( "Event Handler" ) ; + + window = fgWindowByHandle( hWnd ); + + if ( ( window == NULL ) && ( uMsg != WM_CREATE ) ) + return DefWindowProc( hWnd, uMsg, wParam, lParam ); + + /* printf ( "Window %3d message <%04x> %12d %12d\n", window?window->ID:0, + uMsg, wParam, lParam ); */ + + if ( window ) + { + /* Checking for CTRL, ALT, and SHIFT key positions: Key Down! */ + if ( !lControl && GetAsyncKeyState ( VK_LCONTROL ) ) + { + INVOKE_WCB ( *window, Special, + ( GLUT_KEY_CTRL_L, window->State.MouseX, window->State.MouseY ) + ); + + lControl = 1; + } + + if ( !rControl && GetAsyncKeyState ( VK_RCONTROL ) ) + { + INVOKE_WCB ( *window, Special, + ( GLUT_KEY_CTRL_R, window->State.MouseX, window->State.MouseY ) + ); + + rControl = 1; + } + + if ( !lShift && GetAsyncKeyState ( VK_LSHIFT ) ) + { + INVOKE_WCB ( *window, Special, + ( GLUT_KEY_SHIFT_L, window->State.MouseX, window->State.MouseY ) + ); + + lShift = 1; + } + + if ( !rShift && GetAsyncKeyState ( VK_RSHIFT ) ) + { + INVOKE_WCB ( *window, Special, + ( GLUT_KEY_SHIFT_R, window->State.MouseX, window->State.MouseY ) + ); + + rShift = 1; + } + + if ( !lAlt && GetAsyncKeyState ( VK_LMENU ) ) + { + INVOKE_WCB ( *window, Special, + ( GLUT_KEY_ALT_L, window->State.MouseX, window->State.MouseY ) + ); + + lAlt = 1; + } + + if ( !rAlt && GetAsyncKeyState ( VK_RMENU ) ) + { + INVOKE_WCB ( *window, Special, + ( GLUT_KEY_ALT_R, window->State.MouseX, window->State.MouseY ) + ); + + rAlt = 1; + } + + /* Checking for CTRL, ALT, and SHIFT key positions: Key Up! */ + if ( lControl && !GetAsyncKeyState ( VK_LCONTROL ) ) + { + INVOKE_WCB ( *window, SpecialUp, + ( GLUT_KEY_CTRL_L, window->State.MouseX, window->State.MouseY ) + ); + + lControl = 0; + } + + if ( rControl && !GetAsyncKeyState ( VK_RCONTROL ) ) + { + INVOKE_WCB ( *window, SpecialUp, + ( GLUT_KEY_CTRL_R, window->State.MouseX, window->State.MouseY ) + ); + + rControl = 0; + } + + if ( lShift && !GetAsyncKeyState ( VK_LSHIFT ) ) + { + INVOKE_WCB ( *window, SpecialUp, + ( GLUT_KEY_SHIFT_L, window->State.MouseX, window->State.MouseY ) + ); + + lShift = 0; + } + + if ( rShift && !GetAsyncKeyState ( VK_RSHIFT ) ) + { + INVOKE_WCB ( *window, SpecialUp, + ( GLUT_KEY_SHIFT_R, window->State.MouseX, window->State.MouseY ) + ); + + rShift = 0; + } + + if ( lAlt && !GetAsyncKeyState ( VK_LMENU ) ) + { + INVOKE_WCB ( *window, SpecialUp, + ( GLUT_KEY_ALT_L, window->State.MouseX, window->State.MouseY ) + ); + + lAlt = 0; + } + + if ( rAlt && !GetAsyncKeyState ( VK_RMENU ) ) + { + INVOKE_WCB ( *window, SpecialUp, + ( GLUT_KEY_ALT_R, window->State.MouseX, window->State.MouseY ) + ); + + rAlt = 0; + } + } + + switch( uMsg ) + { + case WM_CREATE: + /* The window structure is passed as the creation structure parameter... */ + window = (SFG_Window *) (((LPCREATESTRUCT) lParam)->lpCreateParams); + FREEGLUT_INTERNAL_ERROR_EXIT ( ( window != NULL ), "Cannot create window", + "fgWindowProc" ); + + window->Window.Handle = hWnd; + window->Window.Device = GetDC( hWnd ); + if( window->IsMenu ) + { + unsigned int current_DisplayMode = fgState.DisplayMode; + fgState.DisplayMode = GLUT_DOUBLE | GLUT_RGB | GLUT_DEPTH; +#if !defined(_WIN32_WCE) + fgSetupPixelFormat( window, GL_FALSE, PFD_MAIN_PLANE ); +#endif + fgState.DisplayMode = current_DisplayMode; + + if( fgStructure.MenuContext ) + wglMakeCurrent( window->Window.Device, + fgStructure.MenuContext->MContext + ); + else + { + fgStructure.MenuContext = + (SFG_MenuContext *)malloc( sizeof(SFG_MenuContext) ); + fgStructure.MenuContext->MContext = + wglCreateContext( window->Window.Device ); + } + + /* window->Window.Context = wglGetCurrentContext (); */ + window->Window.Context = wglCreateContext( window->Window.Device ); + } + else + { +#if !defined(_WIN32_WCE) + fgSetupPixelFormat( window, GL_FALSE, PFD_MAIN_PLANE ); +#endif + + if( ! fgState.UseCurrentContext ) + window->Window.Context = + wglCreateContext( window->Window.Device ); + else + { + window->Window.Context = wglGetCurrentContext( ); + if( ! window->Window.Context ) + window->Window.Context = + wglCreateContext( window->Window.Device ); + } + +#if !defined(_WIN32_WCE) + fgNewWGLCreateContext( window ); +#endif + } + + window->State.NeedToResize = GL_TRUE; + /* if we used CW_USEDEFAULT (thats a negative value) for the size + * of the window, query the window now for the size at which it + * was created. + */ + if( ( window->State.Width < 0 ) || ( window->State.Height < 0 ) ) + { + SFG_Window *current_window = fgStructure.CurrentWindow; + + fgSetWindow( window ); + window->State.Width = glutGet( GLUT_WINDOW_WIDTH ); + window->State.Height = glutGet( GLUT_WINDOW_HEIGHT ); + fgSetWindow( current_window ); + } + + ReleaseDC( window->Window.Handle, window->Window.Device ); + +#if defined(_WIN32_WCE) + /* Take over button handling */ + { + HINSTANCE dxDllLib=LoadLibrary(_T("gx.dll")); + if (dxDllLib) + { + GXGetDefaultKeys_=(GXGETDEFAULTKEYS)GetProcAddress(dxDllLib, _T("?GXGetDefaultKeys@@YA?AUGXKeyList@@H@Z")); + GXOpenInput_=(GXOPENINPUT)GetProcAddress(dxDllLib, _T("?GXOpenInput@@YAHXZ")); + } + + if(GXOpenInput_) + (*GXOpenInput_)(); + if(GXGetDefaultKeys_) + gxKeyList = (*GXGetDefaultKeys_)(GX_LANDSCAPEKEYS); + } + +#endif /* defined(_WIN32_WCE) */ + break; + + case WM_SIZE: + /* + * If the window is visible, then it is the user manually resizing it. + * If it is not, then it is the system sending us a dummy resize with + * zero dimensions on a "glutIconifyWindow" call. + */ + if( window->State.Visible ) + { + window->State.NeedToResize = GL_TRUE; +#if defined(_WIN32_WCE) + window->State.Width = HIWORD(lParam); + window->State.Height = LOWORD(lParam); +#else + window->State.Width = LOWORD(lParam); + window->State.Height = HIWORD(lParam); +#endif /* defined(_WIN32_WCE) */ + } + + break; + + case WM_SETFOCUS: +/* printf("WM_SETFOCUS: %p\n", window ); */ + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + INVOKE_WCB( *window, Entry, ( GLUT_ENTERED ) ); + break; + + case WM_KILLFOCUS: +/* printf("WM_KILLFOCUS: %p\n", window ); */ + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + INVOKE_WCB( *window, Entry, ( GLUT_LEFT ) ); + + if( window->IsMenu && + window->ActiveMenu && window->ActiveMenu->IsActive ) + fgUpdateMenuHighlight( window->ActiveMenu ); + + break; + +#if 0 + case WM_ACTIVATE: + if (LOWORD(wParam) != WA_INACTIVE) + { +/* printf("WM_ACTIVATE: fgSetCursor( %p, %d)\n", window, + window->State.Cursor ); */ + fgSetCursor( window, window->State.Cursor ); + } + + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + break; +#endif + + case WM_SETCURSOR: +/* printf ( "Cursor event %x %x %x %x\n", window, window->State.Cursor, lParam, wParam ) ; */ + if( LOWORD( lParam ) == HTCLIENT ) + fgSetCursor ( window, window->State.Cursor ) ; + else + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + break; + + case WM_SHOWWINDOW: + window->State.Visible = GL_TRUE; + window->State.Redisplay = GL_TRUE; + break; + + case WM_PAINT: + /* Turn on the visibility in case it was turned off somehow */ + window->State.Visible = GL_TRUE; + BeginPaint( hWnd, &ps ); + fghRedrawWindow( window ); + EndPaint( hWnd, &ps ); + break; + + case WM_CLOSE: + fgDestroyWindow ( window ); + if ( fgState.ActionOnWindowClose != GLUT_ACTION_CONTINUE_EXECUTION ) + PostQuitMessage(0); + break; + + case WM_DESTROY: + /* + * The window already got destroyed, so don't bother with it. + */ + return 0; + + case WM_MOUSEMOVE: + { +#if defined(_WIN32_WCE) + window->State.MouseX = 320-HIWORD( lParam ); + window->State.MouseY = LOWORD( lParam ); +#else + window->State.MouseX = LOWORD( lParam ); + window->State.MouseY = HIWORD( lParam ); +#endif /* defined(_WIN32_WCE) */ + /* Restrict to [-32768, 32767] to match X11 behaviour */ + /* See comment in "freeglut_developer" mailing list 10/4/04 */ + if ( window->State.MouseX > 32767 ) window->State.MouseX -= 65536; + if ( window->State.MouseY > 32767 ) window->State.MouseY -= 65536; + + if ( window->ActiveMenu ) + { + fgUpdateMenuHighlight( window->ActiveMenu ); + break; + } + SetFocus(window->Window.Handle); + + fgState.Modifiers = fghGetWin32Modifiers( ); + + if( ( wParam & MK_LBUTTON ) || + ( wParam & MK_MBUTTON ) || + ( wParam & MK_RBUTTON ) ) + INVOKE_WCB( *window, Motion, ( window->State.MouseX, + window->State.MouseY ) ); + else + INVOKE_WCB( *window, Passive, ( window->State.MouseX, + window->State.MouseY ) ); + + fgState.Modifiers = INVALID_MODIFIERS; + } + break; + + case WM_LBUTTONDOWN: + case WM_MBUTTONDOWN: + case WM_RBUTTONDOWN: + case WM_LBUTTONUP: + case WM_MBUTTONUP: + case WM_RBUTTONUP: + { + GLboolean pressed = GL_TRUE; + int button; + +#if defined(_WIN32_WCE) + window->State.MouseX = 320-HIWORD( lParam ); + window->State.MouseY = LOWORD( lParam ); +#else + window->State.MouseX = LOWORD( lParam ); + window->State.MouseY = HIWORD( lParam ); +#endif /* defined(_WIN32_WCE) */ + + /* Restrict to [-32768, 32767] to match X11 behaviour */ + /* See comment in "freeglut_developer" mailing list 10/4/04 */ + if ( window->State.MouseX > 32767 ) window->State.MouseX -= 65536; + if ( window->State.MouseY > 32767 ) window->State.MouseY -= 65536; + + switch( uMsg ) + { + case WM_LBUTTONDOWN: + pressed = GL_TRUE; + button = GLUT_LEFT_BUTTON; + break; + case WM_MBUTTONDOWN: + pressed = GL_TRUE; + button = GLUT_MIDDLE_BUTTON; + break; + case WM_RBUTTONDOWN: + pressed = GL_TRUE; + button = GLUT_RIGHT_BUTTON; + break; + case WM_LBUTTONUP: + pressed = GL_FALSE; + button = GLUT_LEFT_BUTTON; + break; + case WM_MBUTTONUP: + pressed = GL_FALSE; + button = GLUT_MIDDLE_BUTTON; + break; + case WM_RBUTTONUP: + pressed = GL_FALSE; + button = GLUT_RIGHT_BUTTON; + break; + default: + pressed = GL_FALSE; + button = -1; + break; + } + +#if !defined(_WIN32_WCE) + if( GetSystemMetrics( SM_SWAPBUTTON ) ) + { + if( button == GLUT_LEFT_BUTTON ) + button = GLUT_RIGHT_BUTTON; + else + if( button == GLUT_RIGHT_BUTTON ) + button = GLUT_LEFT_BUTTON; + } +#endif /* !defined(_WIN32_WCE) */ + + if( button == -1 ) + return DefWindowProc( hWnd, uMsg, lParam, wParam ); + + /* + * Do not execute the application's mouse callback if a menu + * is hooked to this button. In that case an appropriate + * private call should be generated. + */ + if( fgCheckActiveMenu( window, button, pressed, + window->State.MouseX, window->State.MouseY ) ) + break; + + /* Set capture so that the window captures all the mouse messages */ + /* + * XXX - Multiple button support: Under X11, the mouse is not released + * XXX - from the window until all buttons have been released, even if the + * XXX - user presses a button in another window. This will take more + * XXX - code changes than I am up to at the moment (10/5/04). The present + * XXX - is a 90 percent solution. + */ + if ( pressed == GL_TRUE ) + SetCapture ( window->Window.Handle ) ; + else + ReleaseCapture () ; + + if( ! FETCH_WCB( *window, Mouse ) ) + break; + + fgSetWindow( window ); + fgState.Modifiers = fghGetWin32Modifiers( ); + + INVOKE_WCB( + *window, Mouse, + ( button, + pressed ? GLUT_DOWN : GLUT_UP, + window->State.MouseX, + window->State.MouseY + ) + ); + + fgState.Modifiers = INVALID_MODIFIERS; + } + break; + + case 0x020a: + /* Should be WM_MOUSEWHEEL but my compiler doesn't recognize it */ + { + int wheel_number = LOWORD( wParam ); + short ticks = ( short )HIWORD( wParam ); + fgState.MouseWheelTicks += ticks; + + /* + * XXX Should use WHEEL_DELTA instead of 120 + */ + if ( abs ( fgState.MouseWheelTicks ) > 120 ) + { + int direction = ( fgState.MouseWheelTicks > 0 ) ? 1 : -1; + + if( ! FETCH_WCB( *window, MouseWheel ) && + ! FETCH_WCB( *window, Mouse ) ) + break; + + fgSetWindow( window ); + fgState.Modifiers = fghGetWin32Modifiers( ); + + /* + * XXX Should use WHEEL_DELTA instead of 120 + */ + while( abs ( fgState.MouseWheelTicks ) > 120 ) + { + if( FETCH_WCB( *window, MouseWheel ) ) + INVOKE_WCB( *window, MouseWheel, + ( wheel_number, + direction, + window->State.MouseX, + window->State.MouseY + ) + ); + else /* No mouse wheel, call the mouse button callback twice */ + { + /* + * Map wheel zero to button 3 and 4; +1 to 3, -1 to 4 + * " " one +1 to 5, -1 to 6, ... + * + * XXX The below assumes that you have no more than 3 mouse + * XXX buttons. Sorry. + */ + int button = wheel_number * 2 + 3; + if( direction < 0 ) + ++button; + INVOKE_WCB( *window, Mouse, + ( button, GLUT_DOWN, + window->State.MouseX, window->State.MouseY ) + ); + INVOKE_WCB( *window, Mouse, + ( button, GLUT_UP, + window->State.MouseX, window->State.MouseY ) + ); + } + + /* + * XXX Should use WHEEL_DELTA instead of 120 + */ + fgState.MouseWheelTicks -= 120 * direction; + } + + fgState.Modifiers = INVALID_MODIFIERS; + } + } + break ; + + case WM_SYSKEYDOWN: + case WM_KEYDOWN: + { + int keypress = -1; + POINT mouse_pos ; + + if( ( fgState.KeyRepeat==GLUT_KEY_REPEAT_OFF || window->State.IgnoreKeyRepeat==GL_TRUE ) && (HIWORD(lParam) & KF_REPEAT) ) + break; + + /* + * Remember the current modifiers state. This is done here in order + * to make sure the VK_DELETE keyboard callback is executed properly. + */ + fgState.Modifiers = fghGetWin32Modifiers( ); + + GetCursorPos( &mouse_pos ); + ScreenToClient( window->Window.Handle, &mouse_pos ); + + window->State.MouseX = mouse_pos.x; + window->State.MouseY = mouse_pos.y; + + /* Convert the Win32 keystroke codes to GLUTtish way */ +# define KEY(a,b) case a: keypress = b; break; + + switch( wParam ) + { + KEY( VK_F1, GLUT_KEY_F1 ); + KEY( VK_F2, GLUT_KEY_F2 ); + KEY( VK_F3, GLUT_KEY_F3 ); + KEY( VK_F4, GLUT_KEY_F4 ); + KEY( VK_F5, GLUT_KEY_F5 ); + KEY( VK_F6, GLUT_KEY_F6 ); + KEY( VK_F7, GLUT_KEY_F7 ); + KEY( VK_F8, GLUT_KEY_F8 ); + KEY( VK_F9, GLUT_KEY_F9 ); + KEY( VK_F10, GLUT_KEY_F10 ); + KEY( VK_F11, GLUT_KEY_F11 ); + KEY( VK_F12, GLUT_KEY_F12 ); + KEY( VK_PRIOR, GLUT_KEY_PAGE_UP ); + KEY( VK_NEXT, GLUT_KEY_PAGE_DOWN ); + KEY( VK_HOME, GLUT_KEY_HOME ); + KEY( VK_END, GLUT_KEY_END ); + KEY( VK_LEFT, GLUT_KEY_LEFT ); + KEY( VK_UP, GLUT_KEY_UP ); + KEY( VK_RIGHT, GLUT_KEY_RIGHT ); + KEY( VK_DOWN, GLUT_KEY_DOWN ); + KEY( VK_INSERT, GLUT_KEY_INSERT ); + KEY( VK_LCONTROL, GLUT_KEY_CTRL_L ); + KEY( VK_RCONTROL, GLUT_KEY_CTRL_R ); + KEY( VK_LSHIFT, GLUT_KEY_SHIFT_L ); + KEY( VK_RSHIFT, GLUT_KEY_SHIFT_R ); + KEY( VK_LMENU, GLUT_KEY_ALT_L ); + KEY( VK_RMENU, GLUT_KEY_ALT_R ); + + case VK_DELETE: + /* The delete key should be treated as an ASCII keypress: */ + INVOKE_WCB( *window, Keyboard, + ( 127, window->State.MouseX, window->State.MouseY ) + ); + } + +#if defined(_WIN32_WCE) + if(!(lParam & 0x40000000)) /* Prevent auto-repeat */ + { + if(wParam==(unsigned)gxKeyList.vkRight) + keypress = GLUT_KEY_RIGHT; + else if(wParam==(unsigned)gxKeyList.vkLeft) + keypress = GLUT_KEY_LEFT; + else if(wParam==(unsigned)gxKeyList.vkUp) + keypress = GLUT_KEY_UP; + else if(wParam==(unsigned)gxKeyList.vkDown) + keypress = GLUT_KEY_DOWN; + else if(wParam==(unsigned)gxKeyList.vkA) + keypress = GLUT_KEY_F1; + else if(wParam==(unsigned)gxKeyList.vkB) + keypress = GLUT_KEY_F2; + else if(wParam==(unsigned)gxKeyList.vkC) + keypress = GLUT_KEY_F3; + else if(wParam==(unsigned)gxKeyList.vkStart) + keypress = GLUT_KEY_F4; + } +#endif + + if( keypress != -1 ) + INVOKE_WCB( *window, Special, + ( keypress, + window->State.MouseX, window->State.MouseY ) + ); + + fgState.Modifiers = INVALID_MODIFIERS; + } + break; + + case WM_SYSKEYUP: + case WM_KEYUP: + { + int keypress = -1; + POINT mouse_pos; + + /* + * Remember the current modifiers state. This is done here in order + * to make sure the VK_DELETE keyboard callback is executed properly. + */ + fgState.Modifiers = fghGetWin32Modifiers( ); + + GetCursorPos( &mouse_pos ); + ScreenToClient( window->Window.Handle, &mouse_pos ); + + window->State.MouseX = mouse_pos.x; + window->State.MouseY = mouse_pos.y; + + /* + * Convert the Win32 keystroke codes to GLUTtish way. + * "KEY(a,b)" was defined under "WM_KEYDOWN" + */ + + switch( wParam ) + { + KEY( VK_F1, GLUT_KEY_F1 ); + KEY( VK_F2, GLUT_KEY_F2 ); + KEY( VK_F3, GLUT_KEY_F3 ); + KEY( VK_F4, GLUT_KEY_F4 ); + KEY( VK_F5, GLUT_KEY_F5 ); + KEY( VK_F6, GLUT_KEY_F6 ); + KEY( VK_F7, GLUT_KEY_F7 ); + KEY( VK_F8, GLUT_KEY_F8 ); + KEY( VK_F9, GLUT_KEY_F9 ); + KEY( VK_F10, GLUT_KEY_F10 ); + KEY( VK_F11, GLUT_KEY_F11 ); + KEY( VK_F12, GLUT_KEY_F12 ); + KEY( VK_PRIOR, GLUT_KEY_PAGE_UP ); + KEY( VK_NEXT, GLUT_KEY_PAGE_DOWN ); + KEY( VK_HOME, GLUT_KEY_HOME ); + KEY( VK_END, GLUT_KEY_END ); + KEY( VK_LEFT, GLUT_KEY_LEFT ); + KEY( VK_UP, GLUT_KEY_UP ); + KEY( VK_RIGHT, GLUT_KEY_RIGHT ); + KEY( VK_DOWN, GLUT_KEY_DOWN ); + KEY( VK_INSERT, GLUT_KEY_INSERT ); + KEY( VK_LCONTROL, GLUT_KEY_CTRL_L ); + KEY( VK_RCONTROL, GLUT_KEY_CTRL_R ); + KEY( VK_LSHIFT, GLUT_KEY_SHIFT_L ); + KEY( VK_RSHIFT, GLUT_KEY_SHIFT_R ); + KEY( VK_LMENU, GLUT_KEY_ALT_L ); + KEY( VK_RMENU, GLUT_KEY_ALT_R ); + + case VK_DELETE: + /* The delete key should be treated as an ASCII keypress: */ + INVOKE_WCB( *window, KeyboardUp, + ( 127, window->State.MouseX, window->State.MouseY ) + ); + break; + + default: + { +#if !defined(_WIN32_WCE) + BYTE state[ 256 ]; + WORD code[ 2 ]; + + GetKeyboardState( state ); + + if( ToAscii( (UINT)wParam, 0, state, code, 0 ) == 1 ) + wParam=code[ 0 ]; + + INVOKE_WCB( *window, KeyboardUp, + ( (char)wParam, + window->State.MouseX, window->State.MouseY ) + ); +#endif /* !defined(_WIN32_WCE) */ + } + } + + if( keypress != -1 ) + INVOKE_WCB( *window, SpecialUp, + ( keypress, + window->State.MouseX, window->State.MouseY ) + ); + + fgState.Modifiers = INVALID_MODIFIERS; + } + break; + + case WM_SYSCHAR: + case WM_CHAR: + { + if( (fgState.KeyRepeat==GLUT_KEY_REPEAT_OFF || window->State.IgnoreKeyRepeat==GL_TRUE) && (HIWORD(lParam) & KF_REPEAT) ) + break; + + fgState.Modifiers = fghGetWin32Modifiers( ); + INVOKE_WCB( *window, Keyboard, + ( (char)wParam, + window->State.MouseX, window->State.MouseY ) + ); + fgState.Modifiers = INVALID_MODIFIERS; + } + break; + + case WM_CAPTURECHANGED: + /* User has finished resizing the window, force a redraw */ + INVOKE_WCB( *window, Display, ( ) ); + + /*lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); */ + break; + + /* Other messages that I have seen and which are not handled already */ + case WM_SETTEXT: /* 0x000c */ + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + /* Pass it on to "DefWindowProc" to set the window text */ + break; + + case WM_GETTEXT: /* 0x000d */ + /* Ideally we would copy the title of the window into "lParam" */ + /* strncpy ( (char *)lParam, "Window Title", wParam ); + lRet = ( wParam > 12 ) ? 12 : wParam; */ + /* the number of characters copied */ + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + break; + + case WM_GETTEXTLENGTH: /* 0x000e */ + /* Ideally we would get the length of the title of the window */ + lRet = 12; + /* the number of characters in "Window Title\0" (see above) */ + break; + + case WM_ERASEBKGND: /* 0x0014 */ + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + break; + +#if !defined(_WIN32_WCE) + case WM_SYNCPAINT: /* 0x0088 */ + /* Another window has moved, need to update this one */ + window->State.Redisplay = GL_TRUE; + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + /* Help screen says this message must be passed to "DefWindowProc" */ + break; + + case WM_NCPAINT: /* 0x0085 */ + /* Need to update the border of this window */ + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + /* Pass it on to "DefWindowProc" to repaint a standard border */ + break; + + case WM_SYSCOMMAND : /* 0x0112 */ + { + /* + * We have received a system command message. Try to act on it. + * The commands are passed in through the "wParam" parameter: + * The least significant digit seems to be which edge of the window + * is being used for a resize event: + * 4 3 5 + * 1 2 + * 7 6 8 + * Congratulations and thanks to Richard Rauch for figuring this out.. + */ + switch ( wParam & 0xfff0 ) + { + case SC_SIZE : + break ; + + case SC_MOVE : + break ; + + case SC_MINIMIZE : + /* User has clicked on the "-" to minimize the window */ + /* Turn off the visibility */ + window->State.Visible = GL_FALSE ; + + break ; + + case SC_MAXIMIZE : + break ; + + case SC_NEXTWINDOW : + break ; + + case SC_PREVWINDOW : + break ; + + case SC_CLOSE : + /* Followed very closely by a WM_CLOSE message */ + break ; + + case SC_VSCROLL : + break ; + + case SC_HSCROLL : + break ; + + case SC_MOUSEMENU : + break ; + + case SC_KEYMENU : + break ; + + case SC_ARRANGE : + break ; + + case SC_RESTORE : + break ; + + case SC_TASKLIST : + break ; + + case SC_SCREENSAVE : + break ; + + case SC_HOTKEY : + break ; + +#if(WINVER >= 0x0400) + case SC_DEFAULT : + break ; + + case SC_MONITORPOWER : + break ; + + case SC_CONTEXTHELP : + break ; +#endif /* WINVER >= 0x0400 */ + + default: +#if _DEBUG + fgWarning( "Unknown wParam type 0x%x", wParam ); +#endif + break; + } + } +#endif /* !defined(_WIN32_WCE) */ + + /* We need to pass the message on to the operating system as well */ + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + break; + +#ifdef WM_TOUCH + /* handle multi-touch messages */ + case WM_TOUCH: + { + unsigned int numInputs = (unsigned int)wParam; + unsigned int i = 0; + TOUCHINPUT* ti = (TOUCHINPUT*)malloc( sizeof(TOUCHINPUT)*numInputs); + + if (fghGetTouchInputInfo == (pGetTouchInputInfo)0xDEADBEEF) { + fghGetTouchInputInfo = (pGetTouchInputInfo)GetProcAddress(GetModuleHandle("user32"),"GetTouchInputInfo"); + fghCloseTouchInputHandle = (pCloseTouchInputHandle)GetProcAddress(GetModuleHandle("user32"),"CloseTouchInputHandle"); + } + + if (!fghGetTouchInputInfo) { + free( (void*)ti ); + break; + } + + if (fghGetTouchInputInfo( (HTOUCHINPUT)lParam, numInputs, ti, sizeof(TOUCHINPUT) )) { + /* Handle each contact point */ + for (i = 0; i < numInputs; ++i ) { + + POINT tp; + tp.x = TOUCH_COORD_TO_PIXEL(ti[i].x); + tp.y = TOUCH_COORD_TO_PIXEL(ti[i].y); + ScreenToClient( hWnd, &tp ); + + ti[i].dwID = ti[i].dwID * 2; + + if (ti[i].dwFlags & TOUCHEVENTF_DOWN) { + INVOKE_WCB( *window, MultiEntry, ( ti[i].dwID, GLUT_ENTERED ) ); + INVOKE_WCB( *window, MultiButton, ( ti[i].dwID, tp.x, tp.y, 0, GLUT_DOWN ) ); + } else if (ti[i].dwFlags & TOUCHEVENTF_MOVE) { + INVOKE_WCB( *window, MultiMotion, ( ti[i].dwID, tp.x, tp.y ) ); + } else if (ti[i].dwFlags & TOUCHEVENTF_UP) { + INVOKE_WCB( *window, MultiButton, ( ti[i].dwID, tp.x, tp.y, 0, GLUT_UP ) ); + INVOKE_WCB( *window, MultiEntry, ( ti[i].dwID, GLUT_LEFT ) ); + } + } + } + fghCloseTouchInputHandle((HTOUCHINPUT)lParam); + free( (void*)ti ); + lRet = 0; /*DefWindowProc( hWnd, uMsg, wParam, lParam );*/ + break; + } +#endif + default: + /* Handle unhandled messages */ + lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); + break; + } + + return lRet; +} +#endif + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_menu.c b/examples/opengl-framework/freeglut/freeglut_menu.c new file mode 100644 index 00000000..ea5837e8 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_menu.c @@ -0,0 +1,1002 @@ +/* + * freeglut_menu.c + * + * Pull-down menu creation and handling. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 16 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#define FREEGLUT_BUILDING_LIB +#include +#include "freeglut_internal.h" + +/* -- DEFINITIONS ---------------------------------------------------------- */ + +/* + * FREEGLUT_MENU_FONT can be any freeglut bitmapped font. + * (Stroked fonts would not be out of the question, but we'd need to alter + * code, since GLUT (hence freeglut) does not quite unify stroked and + * bitmapped font handling.) + * Old UNIX/X11 GLUT (BSD, UNIX, IRIX, LINUX, HPUX, ...) used a system + * font best approximated by an 18-pixel HELVETICA, I think. MS-WINDOWS + * GLUT used something closest to the 8x13 fixed-width font. (Old + * GLUT apparently uses host-system menus rather than building its own. + * freeglut is building its own menus from scratch.) + * + * FREEGLUT_MENU_HEIGHT gives the height of ONE menu box. This should be + * the distances between two adjacent menu entries. It should scale + * automatically with the font choice, so you needn't alter it---unless you + * use a stroked font. + * + * FREEGLUT_MENU_BORDER says how many pixels to allow around the edge of a + * menu. (It also seems to be the same as the number of pixels used as + * a border around *items* to separate them from neighbors. John says + * that that wasn't the original intent...if not, perhaps we need another + * symbolic constant, FREEGLUT_MENU_ITEM_BORDER, or such.) + */ +#if TARGET_HOST_MS_WINDOWS +#define FREEGLUT_MENU_FONT GLUT_BITMAP_8_BY_13 +#else +#define FREEGLUT_MENU_FONT GLUT_BITMAP_HELVETICA_18 +#endif + +#define FREEGLUT_MENU_HEIGHT (glutBitmapHeight(FREEGLUT_MENU_FONT) + \ + FREEGLUT_MENU_BORDER) +#define FREEGLUT_MENU_BORDER 2 + + +/* + * These variables are for rendering the freeglut menu items. + * + * The choices are fore- and background, with and without h for Highlighting. + * Old GLUT appeared to be system-dependant for its colors (sigh) so we are + * too. These variables should be stuffed into global state and initialized + * via the glutInit*() system. + */ +#if TARGET_HOST_MS_WINDOWS +static float menu_pen_fore [4] = {0.0f, 0.0f, 0.0f, 1.0f}; +static float menu_pen_back [4] = {0.85f, 0.85f, 0.85f, 1.0f}; +static float menu_pen_hfore [4] = {1.0f, 1.0f, 1.0f, 1.0f}; +static float menu_pen_hback [4] = {0.15f, 0.15f, 0.45f, 1.0f}; +#else +static float menu_pen_fore [4] = {0.0f, 0.0f, 0.0f, 1.0f}; +static float menu_pen_back [4] = {0.70f, 0.70f, 0.70f, 1.0f}; +static float menu_pen_hfore [4] = {0.0f, 0.0f, 0.0f, 1.0f}; +static float menu_pen_hback [4] = {1.0f, 1.0f, 1.0f, 1.0f}; +#endif + +/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ + +/* + * Private function to find a menu entry by index + */ +static SFG_MenuEntry *fghFindMenuEntry( SFG_Menu* menu, int index ) +{ + SFG_MenuEntry *entry; + int i = 1; + + for( entry = (SFG_MenuEntry *)menu->Entries.First; + entry; + entry = (SFG_MenuEntry *)entry->Node.Next ) + { + if( i == index ) + break; + ++i; + } + + return entry; +} + +/* + * Deactivates a menu pointed by the function argument. + */ +static void fghDeactivateSubMenu( SFG_MenuEntry *menuEntry ) +{ + SFG_MenuEntry *subMenuIter; + /* Hide the present menu's window */ + fgSetWindow( menuEntry->SubMenu->Window ); + glutHideWindow( ); + + /* Forget about having that menu active anymore, now: */ + menuEntry->SubMenu->Window->ActiveMenu = NULL; + menuEntry->SubMenu->IsActive = GL_FALSE; + menuEntry->SubMenu->ActiveEntry = NULL; + + /* Hide all submenu windows, and the root menu's window. */ + for ( subMenuIter = (SFG_MenuEntry *)menuEntry->SubMenu->Entries.First; + subMenuIter; + subMenuIter = (SFG_MenuEntry *)subMenuIter->Node.Next ) + { + subMenuIter->IsActive = GL_FALSE; + + /* Is that an active submenu by any case? */ + if( subMenuIter->SubMenu ) + fghDeactivateSubMenu( subMenuIter ); + } + + fgSetWindow ( menuEntry->SubMenu->ParentWindow ) ; +} + +/* + * Private function to get the virtual maximum screen extent + */ +static GLvoid fghGetVMaxExtent( SFG_Window* window, int* x, int* y ) +{ + if( fgStructure.GameModeWindow ) + { +#if TARGET_HOST_POSIX_X11 + int wx, wy; + Window w; + + XTranslateCoordinates( + fgDisplay.Display, + window->Window.Handle, + fgDisplay.RootWindow, + 0, 0, &wx, &wy, &w); + + *x = fgState.GameModeSize.X + wx; + *y = fgState.GameModeSize.Y + wy; +#else + *x = glutGet ( GLUT_SCREEN_WIDTH ); + *y = glutGet ( GLUT_SCREEN_HEIGHT ); +#endif + } + else + { + *x = fgDisplay.ScreenWidth; + *y = fgDisplay.ScreenHeight; + } +} + +/* + * Private function to check for the current menu/sub menu activity state + */ +static GLboolean fghCheckMenuStatus( SFG_Menu* menu ) +{ + SFG_MenuEntry* menuEntry; + int x, y; + + /* First of all check any of the active sub menus... */ + for( menuEntry = (SFG_MenuEntry *)menu->Entries.First; + menuEntry; + menuEntry = (SFG_MenuEntry *)menuEntry->Node.Next ) + { + if( menuEntry->SubMenu && menuEntry->IsActive ) + { + /* + * OK, have the sub-menu checked, too. If it returns GL_TRUE, it + * will mean that it caught the mouse cursor and we do not need + * to regenerate the activity list, and so our parents do... + */ + GLboolean return_status; + + menuEntry->SubMenu->Window->State.MouseX = + menu->Window->State.MouseX + menu->X - menuEntry->SubMenu->X; + menuEntry->SubMenu->Window->State.MouseY = + menu->Window->State.MouseY + menu->Y - menuEntry->SubMenu->Y; + return_status = fghCheckMenuStatus( menuEntry->SubMenu ); + + if ( return_status ) + return GL_TRUE; + } + } + + /* That much about our sub menus, let's get to checking the current menu: */ + x = menu->Window->State.MouseX; + y = menu->Window->State.MouseY; + + /* Check if the mouse cursor is contained within the current menu box */ + if( ( x >= FREEGLUT_MENU_BORDER ) && + ( x < menu->Width - FREEGLUT_MENU_BORDER ) && + ( y >= FREEGLUT_MENU_BORDER ) && + ( y < menu->Height - FREEGLUT_MENU_BORDER ) ) + { + int menuID = ( y - FREEGLUT_MENU_BORDER ) / FREEGLUT_MENU_HEIGHT; + + /* The mouse cursor is somewhere over our box, check it out. */ + menuEntry = fghFindMenuEntry( menu, menuID + 1 ); + FREEGLUT_INTERNAL_ERROR_EXIT( menuEntry, "Cannot find menu entry", + "fghCheckMenuStatus" ); + + menuEntry->IsActive = GL_TRUE; + menuEntry->Ordinal = menuID; + + /* + * If this is not the same as the last active menu entry, deactivate + * the previous entry. Specifically, if the previous active entry + * was a submenu then deactivate it. + */ + if( menu->ActiveEntry && ( menuEntry != menu->ActiveEntry ) ) + if( menu->ActiveEntry->SubMenu ) + fghDeactivateSubMenu( menu->ActiveEntry ); + + if( menuEntry != menu->ActiveEntry ) + { + menu->Window->State.Redisplay = GL_TRUE; + if( menu->ActiveEntry ) + menu->ActiveEntry->IsActive = GL_FALSE; + } + + menu->ActiveEntry = menuEntry; + menu->IsActive = GL_TRUE; /* XXX Do we need this? */ + + /* + * OKi, we have marked that entry as active, but it would be also + * nice to have its contents updated, in case it's a sub menu. + * Also, ignore the return value of the check function: + */ + if( menuEntry->SubMenu ) + { + if ( ! menuEntry->SubMenu->IsActive ) + { + int max_x, max_y; + SFG_Window *current_window = fgStructure.CurrentWindow; + + /* Set up the initial menu position now... */ + menuEntry->SubMenu->IsActive = GL_TRUE; + + /* Set up the initial submenu position now: */ + fghGetVMaxExtent(menu->ParentWindow, &max_x, &max_y); + menuEntry->SubMenu->X = menu->X + menu->Width; + menuEntry->SubMenu->Y = menu->Y + + menuEntry->Ordinal * FREEGLUT_MENU_HEIGHT; + + if( menuEntry->SubMenu->X + menuEntry->SubMenu->Width > max_x ) + menuEntry->SubMenu->X = menu->X - menuEntry->SubMenu->Width; + + if( menuEntry->SubMenu->Y + menuEntry->SubMenu->Height > max_y ) + { + menuEntry->SubMenu->Y -= ( menuEntry->SubMenu->Height - + FREEGLUT_MENU_HEIGHT - + 2 * FREEGLUT_MENU_BORDER ); + if( menuEntry->SubMenu->Y < 0 ) + menuEntry->SubMenu->Y = 0; + } + + fgSetWindow( menuEntry->SubMenu->Window ); + glutPositionWindow( menuEntry->SubMenu->X, + menuEntry->SubMenu->Y ); + glutReshapeWindow( menuEntry->SubMenu->Width, + menuEntry->SubMenu->Height ); + glutPopWindow( ); + glutShowWindow( ); + menuEntry->SubMenu->Window->ActiveMenu = menuEntry->SubMenu; + fgSetWindow( current_window ); + menuEntry->SubMenu->Window->State.MouseX = + x + menu->X - menuEntry->SubMenu->X; + menuEntry->SubMenu->Window->State.MouseY = + y + menu->Y - menuEntry->SubMenu->Y; + fghCheckMenuStatus( menuEntry->SubMenu ); + } + + /* Activate it because its parent entry is active */ + menuEntry->SubMenu->IsActive = GL_TRUE; /* XXX Do we need this? */ + } + + /* Report back that we have caught the menu cursor */ + return GL_TRUE; + } + + /* Looks like the menu cursor is somewhere else... */ + if( menu->ActiveEntry && menu->ActiveEntry->IsActive && + ( !menu->ActiveEntry->SubMenu || + !menu->ActiveEntry->SubMenu->IsActive ) ) + { + menu->Window->State.Redisplay = GL_TRUE; + menu->ActiveEntry->IsActive = GL_FALSE; + menu->ActiveEntry = NULL; + } + + return GL_FALSE; +} + +/* + * Displays a menu box and all of its submenus (if they are active) + */ +static void fghDisplayMenuBox( SFG_Menu* menu ) +{ + SFG_MenuEntry *menuEntry; + int i; + int border = FREEGLUT_MENU_BORDER; + + /* + * Have the menu box drawn first. The +- values are + * here just to make it more nice-looking... + */ + /* a non-black dark version of the below. */ + glColor4f( 1.0f, 1.0f, 1.0f, 1.0f ); + glBegin( GL_QUAD_STRIP ); + glVertex2i( menu->Width , 0 ); + glVertex2i( menu->Width - border, border); + glVertex2i( 0 , 0 ); + glVertex2i( border, border); + glVertex2i( 0 , menu->Height ); + glVertex2i( border, menu->Height - border); + glEnd( ); + + /* a non-black dark version of the below. */ + glColor4f( 0.5f, 0.5f, 0.5f, 1.0f ); + glBegin( GL_QUAD_STRIP ); + glVertex2i( 0 , menu->Height ); + glVertex2i( border, menu->Height - border); + glVertex2i( menu->Width , menu->Height ); + glVertex2i( menu->Width - border, menu->Height - border); + glVertex2i( menu->Width , 0 ); + glVertex2i( menu->Width - border, border); + glEnd( ); + + glColor4fv( menu_pen_back ); + glBegin( GL_QUADS ); + glVertex2i( border, border); + glVertex2i( menu->Width - border, border); + glVertex2i( menu->Width - border, menu->Height - border); + glVertex2i( border, menu->Height - border); + glEnd( ); + + /* Check if any of the submenus is currently active... */ + for( menuEntry = (SFG_MenuEntry *)menu->Entries.First; + menuEntry; + menuEntry = (SFG_MenuEntry *)menuEntry->Node.Next ) + { + /* Has the menu been marked as active, maybe? */ + if( menuEntry->IsActive ) + { + /* + * That's truly right, and we need to have it highlighted. + * There is an assumption that mouse cursor didn't move + * since the last check of menu activity state: + */ + int menuID = menuEntry->Ordinal; + + /* So have the highlight drawn... */ + glColor4fv( menu_pen_hback ); + glBegin( GL_QUADS ); + glVertex2i( border, + (menuID + 0)*FREEGLUT_MENU_HEIGHT + border ); + glVertex2i( menu->Width - border, + (menuID + 0)*FREEGLUT_MENU_HEIGHT + border ); + glVertex2i( menu->Width - border, + (menuID + 1)*FREEGLUT_MENU_HEIGHT + border ); + glVertex2i( border, + (menuID + 1)*FREEGLUT_MENU_HEIGHT + border ); + glEnd( ); + } + } + + /* Print the menu entries now... */ + + glColor4fv( menu_pen_fore ); + + for( menuEntry = (SFG_MenuEntry *)menu->Entries.First, i = 0; + menuEntry; + menuEntry = (SFG_MenuEntry *)menuEntry->Node.Next, ++i ) + { + /* If the menu entry is active, set the color to white */ + if( menuEntry->IsActive ) + glColor4fv( menu_pen_hfore ); + + /* Move the raster into position... */ + /* Try to center the text - JCJ 31 July 2003*/ + glRasterPos2i( + 2 * border, + ( i + 1 )*FREEGLUT_MENU_HEIGHT - + ( int )( FREEGLUT_MENU_HEIGHT*0.3 - border ) + ); + + /* Have the label drawn, character after character: */ + glutBitmapString( FREEGLUT_MENU_FONT, + (unsigned char *)menuEntry->Text); + + /* If it's a submenu, draw a right arrow */ + if( menuEntry->SubMenu ) + { + int width = glutBitmapWidth( FREEGLUT_MENU_FONT, '_' ); + int x_base = menu->Width - 2 - width; + int y_base = i*FREEGLUT_MENU_HEIGHT + border; + glBegin( GL_TRIANGLES ); + glVertex2i( x_base, y_base + 2*border); + glVertex2i( menu->Width - 2, y_base + + ( FREEGLUT_MENU_HEIGHT + border) / 2 ); + glVertex2i( x_base, y_base + FREEGLUT_MENU_HEIGHT - border ); + glEnd( ); + } + + /* If the menu entry is active, reset the color */ + if( menuEntry->IsActive ) + glColor4fv( menu_pen_fore ); + } +} + +/* + * Private static function to set the parent window of a submenu and all + * of its submenus + */ +static void fghSetMenuParentWindow( SFG_Window *window, SFG_Menu *menu ) +{ + SFG_MenuEntry *menuEntry; + + menu->ParentWindow = window; + + for( menuEntry = ( SFG_MenuEntry * )menu->Entries.First; + menuEntry; + menuEntry = ( SFG_MenuEntry * )menuEntry->Node.Next ) + if( menuEntry->SubMenu ) + fghSetMenuParentWindow( window, menuEntry->SubMenu ); +} + +/* + * Function to check for menu entry selection on menu deactivation + */ +static void fghExecuteMenuCallback( SFG_Menu* menu ) +{ + SFG_MenuEntry *menuEntry; + + /* First of all check any of the active sub menus... */ + for( menuEntry = (SFG_MenuEntry *)menu->Entries.First; + menuEntry; + menuEntry = (SFG_MenuEntry *)menuEntry->Node.Next) + { + if( menuEntry->IsActive ) + { + if( menuEntry->SubMenu ) + fghExecuteMenuCallback( menuEntry->SubMenu ); + else + if( menu->Callback ) + { + SFG_Menu *save_menu = fgStructure.CurrentMenu; + fgStructure.CurrentMenu = menu; + menu->Callback( menuEntry->ID ); + fgStructure.CurrentMenu = save_menu; + } + + return; + } + } +} + + +/* + * Displays the currently active menu for the current window + */ +void fgDisplayMenu( void ) +{ + SFG_Window* window = fgStructure.CurrentWindow; + SFG_Menu* menu = NULL; + + FREEGLUT_INTERNAL_ERROR_EXIT ( fgStructure.CurrentWindow, "Displaying menu in nonexistent window", + "fgDisplayMenu" ); + + /* Check if there is an active menu attached to this window... */ + menu = window->ActiveMenu; + freeglut_return_if_fail( menu ); + + fgSetWindow( menu->Window ); + + glPushAttrib( GL_DEPTH_BUFFER_BIT | GL_TEXTURE_BIT | GL_LIGHTING_BIT | + GL_POLYGON_BIT ); + + glDisable( GL_DEPTH_TEST ); + glDisable( GL_TEXTURE_2D ); + glDisable( GL_LIGHTING ); + glDisable( GL_CULL_FACE ); + + glMatrixMode( GL_PROJECTION ); + glPushMatrix( ); + glLoadIdentity( ); + glOrtho( + 0, glutGet( GLUT_WINDOW_WIDTH ), + glutGet( GLUT_WINDOW_HEIGHT ), 0, + -1, 1 + ); + + glMatrixMode( GL_MODELVIEW ); + glPushMatrix( ); + glLoadIdentity( ); + + fghDisplayMenuBox( menu ); + + glPopAttrib( ); + + glMatrixMode( GL_PROJECTION ); + glPopMatrix( ); + glMatrixMode( GL_MODELVIEW ); + glPopMatrix( ); + + glutSwapBuffers( ); + + fgSetWindow ( window ); +} + +/* + * Activates a menu pointed by the function argument + */ +static void fghActivateMenu( SFG_Window* window, int button ) +{ + int max_x, max_y; + + /* We'll be referencing this menu a lot, so remember its address: */ + SFG_Menu* menu = window->Menu[ button ]; + SFG_Window* current_window = fgStructure.CurrentWindow; + + /* If the menu is already active in another window, deactivate it there */ + if ( menu->ParentWindow ) + menu->ParentWindow->ActiveMenu = NULL ; + + /* Mark the menu as active, so that it gets displayed: */ + window->ActiveMenu = menu; + menu->IsActive = GL_TRUE; + fghSetMenuParentWindow ( window, menu ); + fgState.ActiveMenus++; + + /* Set up the initial menu position now: */ + fghGetVMaxExtent(menu->ParentWindow, &max_x, &max_y); + fgSetWindow( window ); + menu->X = window->State.MouseX + glutGet( GLUT_WINDOW_X ); + menu->Y = window->State.MouseY + glutGet( GLUT_WINDOW_Y ); + + if( menu->X + menu->Width > max_x ) + menu->X -=menu->Width; + + if( menu->Y + menu->Height > max_y ) + { + menu->Y -=menu->Height; + if( menu->Y < 0 ) + menu->Y = 0; + } + + menu->Window->State.MouseX = + window->State.MouseX + glutGet( GLUT_WINDOW_X ) - menu->X; + menu->Window->State.MouseY = + window->State.MouseY + glutGet( GLUT_WINDOW_Y ) - menu->Y; + + fgSetWindow( menu->Window ); + glutPositionWindow( menu->X, menu->Y ); + glutReshapeWindow( menu->Width, menu->Height ); + glutPopWindow( ); + glutShowWindow( ); + menu->Window->ActiveMenu = menu; + fghCheckMenuStatus( menu ); + fgSetWindow( current_window ); +} + +/* + * Update Highlight states of the menu + * + * Current mouse position is in menu->Window->State.MouseX/Y. + */ +void fgUpdateMenuHighlight ( SFG_Menu *menu ) +{ + fghCheckMenuStatus( menu ); +} + +/* + * Check whether an active menu absorbs a mouse click + */ +GLboolean fgCheckActiveMenu ( SFG_Window *window, int button, GLboolean pressed, + int mouse_x, int mouse_y ) +{ + /* + * Near as I can tell, this is the menu behaviour: + * - Down-click the menu button, menu not active: activate + * the menu with its upper left-hand corner at the mouse + * location. + * - Down-click any button outside the menu, menu active: + * deactivate the menu + * - Down-click any button inside the menu, menu active: + * select the menu entry and deactivate the menu + * - Up-click the menu button, menu not active: nothing happens + * - Up-click the menu button outside the menu, menu active: + * nothing happens + * - Up-click the menu button inside the menu, menu active: + * select the menu entry and deactivate the menu + * Since menus can have submenus, we need to check this recursively. + */ + if( window->ActiveMenu ) + { + if( window == window->ActiveMenu->ParentWindow ) + { + window->ActiveMenu->Window->State.MouseX = + mouse_x - window->ActiveMenu->X; + window->ActiveMenu->Window->State.MouseY = + mouse_y - window->ActiveMenu->Y; + } + + /* In the menu, invoke the callback and deactivate the menu */ + if( fghCheckMenuStatus( window->ActiveMenu ) ) + { + /* + * Save the current window and menu and set the current + * window to the window whose menu this is + */ + SFG_Window *save_window = fgStructure.CurrentWindow; + SFG_Menu *save_menu = fgStructure.CurrentMenu; + SFG_Window *parent_window = window->ActiveMenu->ParentWindow; + fgSetWindow( parent_window ); + fgStructure.CurrentMenu = window->ActiveMenu; + + /* Execute the menu callback */ + fghExecuteMenuCallback( window->ActiveMenu ); + fgDeactivateMenu( parent_window ); + + /* Restore the current window and menu */ + fgSetWindow( save_window ); + fgStructure.CurrentMenu = save_menu; + } + else if( pressed ) + /* + * Outside the menu, deactivate if it's a downclick + * + * XXX This isn't enough. A downclick outside of + * XXX the interior of our freeglut windows should also + * XXX deactivate the menu. This is more complicated. + */ + fgDeactivateMenu( window->ActiveMenu->ParentWindow ); + + /* + * XXX Why does an active menu require a redisplay at + * XXX this point? If this can come out cleanly, then + * XXX it probably should do so; if not, a comment should + * XXX explain it. + */ + if( ! window->IsMenu ) + window->State.Redisplay = GL_TRUE; + + return GL_TRUE; + } + + /* No active menu, let's check whether we need to activate one. */ + if( ( 0 <= button ) && + ( FREEGLUT_MAX_MENUS > button ) && + ( window->Menu[ button ] ) && + pressed ) + { + /* XXX Posting a requisite Redisplay seems bogus. */ + window->State.Redisplay = GL_TRUE; + fghActivateMenu( window, button ); + return GL_TRUE; + } + + return GL_FALSE; +} + +/* + * Deactivates a menu pointed by the function argument. + */ +void fgDeactivateMenu( SFG_Window *window ) +{ + SFG_Window *parent_window = NULL; + + /* Check if there is an active menu attached to this window... */ + SFG_Menu* menu = window->ActiveMenu; + SFG_MenuEntry *menuEntry; + + /* Did we find an active window? */ + freeglut_return_if_fail( menu ); + + parent_window = menu->ParentWindow; + + /* Hide the present menu's window */ + fgSetWindow( menu->Window ); + glutHideWindow( ); + + /* Forget about having that menu active anymore, now: */ + menu->Window->ActiveMenu = NULL; + menu->ParentWindow->ActiveMenu = NULL; + fghSetMenuParentWindow ( NULL, menu ); + menu->IsActive = GL_FALSE; + menu->ActiveEntry = NULL; + + fgState.ActiveMenus--; + + /* Hide all submenu windows, and the root menu's window. */ + for ( menuEntry = ( SFG_MenuEntry * )menu->Entries.First; + menuEntry; + menuEntry = ( SFG_MenuEntry * )menuEntry->Node.Next ) + { + menuEntry->IsActive = GL_FALSE; + + /* Is that an active submenu by any case? */ + if( menuEntry->SubMenu ) + fghDeactivateSubMenu( menuEntry ); + } + + fgSetWindow ( parent_window ) ; +} + +/* + * Recalculates current menu's box size + */ +void fghCalculateMenuBoxSize( void ) +{ + SFG_MenuEntry* menuEntry; + int width = 0, height = 0; + + /* Make sure there is a current menu set */ + freeglut_return_if_fail( fgStructure.CurrentMenu ); + + /* The menu's box size depends on the menu entries: */ + for( menuEntry = ( SFG_MenuEntry * )fgStructure.CurrentMenu->Entries.First; + menuEntry; + menuEntry = ( SFG_MenuEntry * )menuEntry->Node.Next ) + { + /* Update the menu entry's width value */ + menuEntry->Width = glutBitmapLength( + FREEGLUT_MENU_FONT, + (unsigned char *)menuEntry->Text + ); + + /* + * If the entry is a submenu, then it needs to be wider to + * accomodate the arrow. JCJ 31 July 2003 + */ + if (menuEntry->SubMenu ) + menuEntry->Width += glutBitmapLength( + FREEGLUT_MENU_FONT, + (unsigned char *)"_" + ); + + /* Check if it's the biggest we've found */ + if( menuEntry->Width > width ) + width = menuEntry->Width; + + height += FREEGLUT_MENU_HEIGHT; + } + + /* Store the menu's box size now: */ + fgStructure.CurrentMenu->Height = height + 2 * FREEGLUT_MENU_BORDER; + fgStructure.CurrentMenu->Width = width + 4 * FREEGLUT_MENU_BORDER; +} + + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * Creates a new menu object, adding it to the freeglut structure + */ +int FGAPIENTRY glutCreateMenu( void(* callback)( int ) ) +{ + /* The menu object creation code resides in freeglut_structure.c */ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutCreateMenu" ); + return fgCreateMenu( callback )->ID; +} + +#if TARGET_HOST_MS_WINDOWS +int FGAPIENTRY __glutCreateMenuWithExit( void(* callback)( int ), void (__cdecl *exit_function)(int) ) +{ + __glutExitFunc = exit_function; + return glutCreateMenu( callback ); +} +#endif + +/* + * Destroys a menu object, removing all references to it + */ +void FGAPIENTRY glutDestroyMenu( int menuID ) +{ + SFG_Menu* menu; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDestroyMenu" ); + menu = fgMenuByID( menuID ); + + freeglut_return_if_fail( menu ); + + /* The menu object destruction code resides in freeglut_structure.c */ + fgDestroyMenu( menu ); +} + +/* + * Returns the ID number of the currently active menu + */ +int FGAPIENTRY glutGetMenu( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetMenu" ); + + if( fgStructure.CurrentMenu ) + return fgStructure.CurrentMenu->ID; + + return 0; +} + +/* + * Sets the current menu given its menu ID + */ +void FGAPIENTRY glutSetMenu( int menuID ) +{ + SFG_Menu* menu; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetMenu" ); + menu = fgMenuByID( menuID ); + + freeglut_return_if_fail( menu ); + + fgStructure.CurrentMenu = menu; +} + +/* + * Adds a menu entry to the bottom of the current menu + */ +void FGAPIENTRY glutAddMenuEntry( const char* label, int value ) +{ + SFG_MenuEntry* menuEntry; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutAddMenuEntry" ); + menuEntry = (SFG_MenuEntry *)calloc( sizeof(SFG_MenuEntry), 1 ); + freeglut_return_if_fail( fgStructure.CurrentMenu ); + + menuEntry->Text = strdup( label ); + menuEntry->ID = value; + + /* Have the new menu entry attached to the current menu */ + fgListAppend( &fgStructure.CurrentMenu->Entries, &menuEntry->Node ); + + fghCalculateMenuBoxSize( ); +} + +/* + * Add a sub menu to the bottom of the current menu + */ +void FGAPIENTRY glutAddSubMenu( const char *label, int subMenuID ) +{ + SFG_MenuEntry *menuEntry; + SFG_Menu *subMenu; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutAddSubMenu" ); + menuEntry = ( SFG_MenuEntry * )calloc( sizeof( SFG_MenuEntry ), 1 ); + subMenu = fgMenuByID( subMenuID ); + + freeglut_return_if_fail( fgStructure.CurrentMenu ); + freeglut_return_if_fail( subMenu ); + + menuEntry->Text = strdup( label ); + menuEntry->SubMenu = subMenu; + menuEntry->ID = -1; + + fgListAppend( &fgStructure.CurrentMenu->Entries, &menuEntry->Node ); + fghCalculateMenuBoxSize( ); +} + +/* + * Changes the specified menu item in the current menu into a menu entry + */ +void FGAPIENTRY glutChangeToMenuEntry( int item, const char* label, int value ) +{ + SFG_MenuEntry* menuEntry = NULL; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutChangeToMenuEntry" ); + freeglut_return_if_fail( fgStructure.CurrentMenu ); + + /* Get n-th menu entry in the current menu, starting from one: */ + menuEntry = fghFindMenuEntry( fgStructure.CurrentMenu, item ); + + freeglut_return_if_fail( menuEntry ); + + /* We want it to become a normal menu entry, so: */ + if( menuEntry->Text ) + free( menuEntry->Text ); + + menuEntry->Text = strdup( label ); + menuEntry->ID = value; + menuEntry->SubMenu = NULL; + fghCalculateMenuBoxSize( ); +} + +/* + * Changes the specified menu item in the current menu into a sub-menu trigger. + */ +void FGAPIENTRY glutChangeToSubMenu( int item, const char* label, + int subMenuID ) +{ + SFG_Menu* subMenu; + SFG_MenuEntry* menuEntry; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutChangeToSubMenu" ); + subMenu = fgMenuByID( subMenuID ); + menuEntry = NULL; + + freeglut_return_if_fail( fgStructure.CurrentMenu ); + freeglut_return_if_fail( subMenu ); + + /* Get n-th menu entry in the current menu, starting from one: */ + menuEntry = fghFindMenuEntry( fgStructure.CurrentMenu, item ); + + freeglut_return_if_fail( menuEntry ); + + /* We want it to become a sub menu entry, so: */ + if( menuEntry->Text ) + free( menuEntry->Text ); + + menuEntry->Text = strdup( label ); + menuEntry->SubMenu = subMenu; + menuEntry->ID = -1; + fghCalculateMenuBoxSize( ); +} + +/* + * Removes the specified menu item from the current menu + */ +void FGAPIENTRY glutRemoveMenuItem( int item ) +{ + SFG_MenuEntry* menuEntry; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutRemoveMenuItem" ); + freeglut_return_if_fail( fgStructure.CurrentMenu ); + + /* Get n-th menu entry in the current menu, starting from one: */ + menuEntry = fghFindMenuEntry( fgStructure.CurrentMenu, item ); + + freeglut_return_if_fail( menuEntry ); + + fgListRemove( &fgStructure.CurrentMenu->Entries, &menuEntry->Node ); + if ( menuEntry->Text ) + free( menuEntry->Text ); + + free( menuEntry ); + fghCalculateMenuBoxSize( ); +} + +/* + * Attaches a menu to the current window + */ +void FGAPIENTRY glutAttachMenu( int button ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutAttachMenu" ); + + freeglut_return_if_fail( fgStructure.CurrentWindow ); + freeglut_return_if_fail( fgStructure.CurrentMenu ); + + freeglut_return_if_fail( button >= 0 ); + freeglut_return_if_fail( button < FREEGLUT_MAX_MENUS ); + + fgStructure.CurrentWindow->Menu[ button ] = fgStructure.CurrentMenu; +} + +/* + * Detaches a menu from the current window + */ +void FGAPIENTRY glutDetachMenu( int button ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDetachMenu" ); + + freeglut_return_if_fail( fgStructure.CurrentWindow ); + freeglut_return_if_fail( fgStructure.CurrentMenu ); + + freeglut_return_if_fail( button >= 0 ); + freeglut_return_if_fail( button < FREEGLUT_MAX_MENUS ); + + fgStructure.CurrentWindow->Menu[ button ] = NULL; +} + +/* + * A.Donev: Set and retrieve the menu's user data + */ +void* FGAPIENTRY glutGetMenuData( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetMenuData" ); + return fgStructure.CurrentMenu->UserData; +} + +void FGAPIENTRY glutSetMenuData(void* data) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetMenuData" ); + fgStructure.CurrentMenu->UserData=data; +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_misc.c b/examples/opengl-framework/freeglut/freeglut_misc.c new file mode 100644 index 00000000..4aa809ac --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_misc.c @@ -0,0 +1,214 @@ +/* + * freeglut_misc.c + * + * Functions that didn't fit anywhere else... + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 9 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* + * TODO BEFORE THE STABLE RELEASE: + * + * glutSetColor() -- + * glutGetColor() -- + * glutCopyColormap() -- + * glutSetKeyRepeat() -- this is evil and should be removed from API + */ + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * This functions checks if an OpenGL extension is supported or not + * + * XXX Wouldn't this be simpler and clearer if we used strtok()? + */ +int FGAPIENTRY glutExtensionSupported( const char* extension ) +{ + const char *extensions, *start; + const size_t len = strlen( extension ); + + /* Make sure there is a current window, and thus a current context available */ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutExtensionSupported" ); + freeglut_return_val_if_fail( fgStructure.CurrentWindow != NULL, 0 ); + + if (strchr(extension, ' ')) + return 0; + start = extensions = (const char *) glGetString(GL_EXTENSIONS); + + /* XXX consider printing a warning to stderr that there's no current + * rendering context. + */ + freeglut_return_val_if_fail( extensions != NULL, 0 ); + + while (1) { + const char *p = strstr(extensions, extension); + if (!p) + return 0; /* not found */ + /* check that the match isn't a super string */ + if ((p == start || p[-1] == ' ') && (p[len] == ' ' || p[len] == 0)) + return 1; + /* skip the false match and continue */ + extensions = p + len; + } + + return 0 ; +} + +#ifndef GL_INVALID_FRAMEBUFFER_OPERATION +#ifdef GL_INVALID_FRAMEBUFFER_OPERATION_EXT +#define GL_INVALID_FRAMEBUFFER_OPERATION GL_INVALID_FRAMEBUFFER_OPERATION_EXT +#else +#define GL_INVALID_FRAMEBUFFER_OPERATION 0x0506 +#endif +#endif + +#ifndef GL_TABLE_TOO_LARGE +#ifdef GL_TABLE_TOO_LARGE_EXT +#define GL_TABLE_TOO_LARGE GL_TABLE_TOO_LARGE_EXT +#else +#define GL_TABLE_TOO_LARGE 0x8031 +#endif +#endif + +#ifndef GL_TEXTURE_TOO_LARGE +#ifdef GL_TEXTURE_TOO_LARGE_EXT +#define GL_TEXTURE_TOO_LARGE GL_TEXTURE_TOO_LARGE_EXT +#else +#define GL_TEXTURE_TOO_LARGE 0x8065 +#endif +#endif + +/* + * A cut-down local version of gluErrorString to avoid depending on GLU. + */ +static const char* fghErrorString( GLenum error ) +{ + switch ( error ) { + case GL_INVALID_ENUM: return "invalid enumerant"; + case GL_INVALID_VALUE: return "invalid value"; + case GL_INVALID_OPERATION: return "invalid operation"; + case GL_STACK_OVERFLOW: return "stack overflow"; + case GL_STACK_UNDERFLOW: return "stack underflow"; + case GL_OUT_OF_MEMORY: return "out of memory"; + case GL_TABLE_TOO_LARGE: return "table too large"; + case GL_INVALID_FRAMEBUFFER_OPERATION: return "invalid framebuffer operation"; + case GL_TEXTURE_TOO_LARGE: return "texture too large"; + default: return "unknown GL error"; + } +} + +/* + * This function reports all the OpenGL errors that happened till now + */ +void FGAPIENTRY glutReportErrors( void ) +{ + GLenum error; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutReportErrors" ); + while( ( error = glGetError() ) != GL_NO_ERROR ) + fgWarning( "GL error: %s", fghErrorString( error ) ); +} + +/* + * Control the auto-repeat of keystrokes to the current window + */ +void FGAPIENTRY glutIgnoreKeyRepeat( int ignore ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutIgnoreKeyRepeat" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutIgnoreKeyRepeat" ); + + fgStructure.CurrentWindow->State.IgnoreKeyRepeat = ignore ? GL_TRUE : GL_FALSE; +} + +/* + * Set global auto-repeat of keystrokes + * + * RepeatMode should be either: + * GLUT_KEY_REPEAT_OFF + * GLUT_KEY_REPEAT_ON + * GLUT_KEY_REPEAT_DEFAULT + */ +void FGAPIENTRY glutSetKeyRepeat( int repeatMode ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetKeyRepeat" ); + + switch( repeatMode ) + { + case GLUT_KEY_REPEAT_OFF: + case GLUT_KEY_REPEAT_ON: + fgState.KeyRepeat = repeatMode; + break; + + case GLUT_KEY_REPEAT_DEFAULT: + fgState.KeyRepeat = GLUT_KEY_REPEAT_ON; + break; + + default: + fgError ("Invalid glutSetKeyRepeat mode: %d", repeatMode); + break; + } +} + +/* + * Forces the joystick callback to be executed + */ +void FGAPIENTRY glutForceJoystickFunc( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutForceJoystickFunc" ); +#if !defined(_WIN32_WCE) + freeglut_return_if_fail( fgStructure.CurrentWindow != NULL ); + freeglut_return_if_fail( FETCH_WCB( *( fgStructure.CurrentWindow ), Joystick ) ); + fgJoystickPollWindow( fgStructure.CurrentWindow ); +#endif /* !defined(_WIN32_WCE) */ +} + +/* + * + */ +void FGAPIENTRY glutSetColor( int nColor, GLfloat red, GLfloat green, GLfloat blue ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetColor" ); + /* We really need to do something here. */ +} + +/* + * + */ +GLfloat FGAPIENTRY glutGetColor( int color, int component ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetColor" ); + /* We really need to do something here. */ + return( 0.0f ); +} + +/* + * + */ +void FGAPIENTRY glutCopyColormap( int window ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutCopyColormap" ); + /* We really need to do something here. */ +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_overlay.c b/examples/opengl-framework/freeglut/freeglut_overlay.c new file mode 100644 index 00000000..2a1314a4 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_overlay.c @@ -0,0 +1,45 @@ +/* + * freeglut_overlay.c + * + * Overlay management functions (as defined by GLUT API) + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 16 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* + * NOTE: functions declared in this file probably will not be implemented. + */ + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +void FGAPIENTRY glutEstablishOverlay( void ) { /* Not implemented */ } +void FGAPIENTRY glutRemoveOverlay( void ) { /* Not implemented */ } +void FGAPIENTRY glutUseLayer( GLenum layer ) { /* Not implemented */ } +void FGAPIENTRY glutPostOverlayRedisplay( void ) { /* Not implemented */ } +void FGAPIENTRY glutPostWindowOverlayRedisplay( int ID ) { /* Not implemented */ } +void FGAPIENTRY glutShowOverlay( void ) { /* Not implemented */ } +void FGAPIENTRY glutHideOverlay( void ) { /* Not implemented */ } + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_spaceball.c b/examples/opengl-framework/freeglut/freeglut_spaceball.c new file mode 100644 index 00000000..be0021f0 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_spaceball.c @@ -0,0 +1,471 @@ +/* Spaceball support for Linux. + * Written by John Tsiombikas + * + * This code supports 3Dconnexion's 6-dof space-whatever devices. + * It can communicate with either the proprietary 3Dconnexion daemon (3dxsrv) + * free spacenavd (http://spacenav.sourceforge.net), through the "standard" + * magellan X-based protocol. + */ + +#include +#include "freeglut_internal.h" + +/* -- PRIVATE FUNCTIONS --------------------------------------------------- */ + +#if TARGET_HOST_POSIX_X11 +#include + +enum { + SPNAV_EVENT_ANY, /* used by spnav_remove_events() */ + SPNAV_EVENT_MOTION, + SPNAV_EVENT_BUTTON /* includes both press and release */ +}; + +struct spnav_event_motion { + int type; + int x, y, z; + int rx, ry, rz; + unsigned int period; + int *data; +}; + +struct spnav_event_button { + int type; + int press; + int bnum; +}; + +typedef union spnav_event { + int type; + struct spnav_event_motion motion; + struct spnav_event_button button; +} spnav_event; + + +static int spnav_x11_open(Display *dpy, Window win); +static int spnav_x11_window(Window win); +static int spnav_x11_event(const XEvent *xev, spnav_event *event); +static int spnav_close(void); +static int spnav_fd(void); +static int spnav_remove_events(int type); + +static SFG_Window *spnav_win; +#endif + +/* Flag telling whether we have a spaceball: + * 0 - haven't tried initializing + * 1 - have successfully initialized + * -1 - have tried to initialize but not succeeded + */ +static int sball_initialized = 0; + + +void fgInitialiseSpaceball(void) +{ + if(sball_initialized != 0) { + return; + } + +#if TARGET_HOST_POSIX_X11 + { + Window w; + + if(!fgStructure.CurrentWindow) + { + sball_initialized = -1; + return; + } + + w = fgStructure.CurrentWindow->Window.Handle; + if(spnav_x11_open(fgDisplay.Display, w) == -1) + { + sball_initialized = -1; + return; + } + } +#endif + + sball_initialized = 1; +} + +void fgSpaceballClose(void) +{ +#if TARGET_HOST_POSIX_X11 + spnav_close(); +#endif +} + +int fgHasSpaceball(void) +{ + if(sball_initialized == 0) { + fgInitialiseSpaceball(); + if(sball_initialized != 1) { + fgWarning("fgInitialiseSpaceball failed\n"); + return 0; + } + } + +#if TARGET_HOST_POSIX_X11 + /* XXX this function should somehow query the driver if there's a device + * plugged in, as opposed to just checking if there's a driver to talk to. + */ + return spnav_fd() == -1 ? 0 : 1; +#else + return 0; +#endif +} + +int fgSpaceballNumButtons(void) +{ + if(sball_initialized == 0) { + fgInitialiseSpaceball(); + if(sball_initialized != 1) { + fgWarning("fgInitialiseSpaceball failed\n"); + return 0; + } + } + +#if TARGET_HOST_POSIX_X11 + return 2; /* TODO implement this properly */ +#else + return 0; +#endif +} + +void fgSpaceballSetWindow(SFG_Window *window) +{ + if(sball_initialized == 0) { + fgInitialiseSpaceball(); + if(sball_initialized != 1) { + return; + } + } + +#if TARGET_HOST_POSIX_X11 + if(spnav_win != window) { + spnav_x11_window(window->Window.Handle); + spnav_win = window; + } +#endif +} + + +#if TARGET_HOST_POSIX_X11 +int fgIsSpaceballXEvent(const XEvent *xev) +{ + spnav_event sev; + + if(spnav_win != fgStructure.CurrentWindow) { + /* this will also initialize spaceball if needed (first call) */ + fgSpaceballSetWindow(fgStructure.CurrentWindow); + } + + if(sball_initialized != 1) { + return 0; + } + + return spnav_x11_event(xev, &sev); +} + +void fgSpaceballHandleXEvent(const XEvent *xev) +{ + spnav_event sev; + + if(sball_initialized == 0) { + fgInitialiseSpaceball(); + if(sball_initialized != 1) { + return; + } + } + + if(spnav_x11_event(xev, &sev)) { + switch(sev.type) { + case SPNAV_EVENT_MOTION: + if(sev.motion.x | sev.motion.y | sev.motion.z) { + INVOKE_WCB(*spnav_win, SpaceMotion, (sev.motion.x, sev.motion.y, sev.motion.z)); + } + if(sev.motion.rx | sev.motion.ry | sev.motion.rz) { + INVOKE_WCB(*spnav_win, SpaceRotation, (sev.motion.rx, sev.motion.ry, sev.motion.rz)); + } + spnav_remove_events(SPNAV_EVENT_MOTION); + break; + + case SPNAV_EVENT_BUTTON: + INVOKE_WCB(*spnav_win, SpaceButton, (sev.button.bnum, sev.button.press ? GLUT_DOWN : GLUT_UP)); + break; + + default: + break; + } + } +} + +/* +The following code is part of libspnav, part of the spacenav project (spacenav.sf.net) +Copyright (C) 2007-2009 John Tsiombikas + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + +1. Redistributions of source code must retain the above copyright notice, this + list of conditions and the following disclaimer. +2. Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. +3. The name of the author may not be used to endorse or promote products + derived from this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED +WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO +EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, +EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT +OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS +INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN +CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING +IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY +OF SUCH DAMAGE. +*/ +#include +#include +#include + +#ifdef HAVE_ERRNO_H +#include +#endif + +#include +#include + +static Window get_daemon_window(Display *dpy); +static int catch_badwin(Display *dpy, XErrorEvent *err); + +static Display *dpy; +static Window app_win; +static Atom motion_event, button_press_event, button_release_event, command_event; + +enum { + CMD_APP_WINDOW = 27695, + CMD_APP_SENS +}; + +#define IS_OPEN dpy + +struct event_node { + spnav_event event; + struct event_node *next; +}; + +static int spnav_x11_open(Display *display, Window win) +{ + if(IS_OPEN) { + return -1; + } + + dpy = display; + + motion_event = XInternAtom(dpy, "MotionEvent", True); + button_press_event = XInternAtom(dpy, "ButtonPressEvent", True); + button_release_event = XInternAtom(dpy, "ButtonReleaseEvent", True); + command_event = XInternAtom(dpy, "CommandEvent", True); + + if(!motion_event || !button_press_event || !button_release_event || !command_event) { + dpy = 0; + return -1; /* daemon not started */ + } + + if(spnav_x11_window(win) == -1) { + dpy = 0; + return -1; /* daemon not started */ + } + + app_win = win; + return 0; +} + +static int spnav_close(void) +{ + if(dpy) { + spnav_x11_window(DefaultRootWindow(dpy)); + app_win = 0; + dpy = 0; + return 0; + } + return -1; +} + +static int spnav_x11_window(Window win) +{ + int (*prev_xerr_handler)(Display*, XErrorEvent*); + XEvent xev; + Window daemon_win; + + if(!IS_OPEN) { + return -1; + } + + if(!(daemon_win = get_daemon_window(dpy))) { + return -1; + } + + prev_xerr_handler = XSetErrorHandler(catch_badwin); + + xev.type = ClientMessage; + xev.xclient.send_event = False; + xev.xclient.display = dpy; + xev.xclient.window = win; + xev.xclient.message_type = command_event; + xev.xclient.format = 16; + xev.xclient.data.s[0] = ((unsigned int)win & 0xffff0000) >> 16; + xev.xclient.data.s[1] = (unsigned int)win & 0xffff; + xev.xclient.data.s[2] = CMD_APP_WINDOW; + + XSendEvent(dpy, daemon_win, False, 0, &xev); + XSync(dpy, False); + + XSetErrorHandler(prev_xerr_handler); + return 0; +} + +static int spnav_fd(void) +{ + if(dpy) { + return ConnectionNumber(dpy); + } + return -1; +} + +/*static int spnav_wait_event(spnav_event *event) +{ + if(dpy) { + for(;;) { + XEvent xev; + XNextEvent(dpy, &xev); + + if(spnav_x11_event(&xev, event) > 0) { + return event->type; + } + } + } + return 0; +} + +static int spnav_poll_event(spnav_event *event) +{ + if(dpy) { + if(XPending(dpy)) { + XEvent xev; + XNextEvent(dpy, &xev); + + return spnav_x11_event(&xev, event); + } + } + return 0; +}*/ + +static Bool match_events(Display *dpy, XEvent *xev, char *arg) +{ + int evtype = *(int*)arg; + + if(xev->type != ClientMessage) { + return False; + } + + if(xev->xclient.message_type == motion_event) { + return !evtype || evtype == SPNAV_EVENT_MOTION ? True : False; + } + if(xev->xclient.message_type == button_press_event || + xev->xclient.message_type == button_release_event) { + return !evtype || evtype == SPNAV_EVENT_BUTTON ? True : False; + } + return False; +} + +static int spnav_remove_events(int type) +{ + int rm_count = 0; + + if(dpy) { + XEvent xev; + + while(XCheckIfEvent(dpy, &xev, match_events, (char*)&type)) { + rm_count++; + } + return rm_count; + } + return 0; +} + +static int spnav_x11_event(const XEvent *xev, spnav_event *event) +{ + int i; + int xmsg_type; + + if(xev->type != ClientMessage) { + return 0; + } + + xmsg_type = xev->xclient.message_type; + + if(xmsg_type != motion_event && xmsg_type != button_press_event && + xmsg_type != button_release_event) { + return 0; + } + + if(xmsg_type == motion_event) { + event->type = SPNAV_EVENT_MOTION; + event->motion.data = &event->motion.x; + + for(i=0; i<6; i++) { + event->motion.data[i] = xev->xclient.data.s[i + 2]; + } + event->motion.period = xev->xclient.data.s[8]; + } else { + event->type = SPNAV_EVENT_BUTTON; + event->button.press = xmsg_type == button_press_event ? 1 : 0; + event->button.bnum = xev->xclient.data.s[2]; + } + return event->type; +} + + +static Window get_daemon_window(Display *dpy) +{ + Window win, root_win; + XTextProperty wname; + Atom type; + int fmt; + unsigned long nitems, bytes_after; + unsigned char *prop; + + root_win = DefaultRootWindow(dpy); + + XGetWindowProperty(dpy, root_win, command_event, 0, 1, False, AnyPropertyType, &type, &fmt, &nitems, &bytes_after, &prop); + if(!prop) { + return 0; + } + + win = *(Window*)prop; + XFree(prop); + + if(!XGetWMName(dpy, win, &wname) || strcmp("Magellan Window", (char*)wname.value) != 0) { + return 0; + } + + return win; +} + +static int catch_badwin(Display *dpy, XErrorEvent *err) +{ + char buf[256]; + + if(err->error_code == BadWindow) { + /* do nothing? */ + } else { + XGetErrorText(dpy, err->error_code, buf, sizeof buf); + fprintf(stderr, "Caught unexpected X error: %s\n", buf); + } + return 0; +} + +#endif /* TARGET_HOST_POSIX_X11 */ diff --git a/examples/opengl-framework/freeglut/freeglut_state.c b/examples/opengl-framework/freeglut/freeglut_state.c new file mode 100644 index 00000000..9e3a627d --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_state.c @@ -0,0 +1,887 @@ +/* + * freeglut_state.c + * + * Freeglut state query methods. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 16 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* + * TODO BEFORE THE STABLE RELEASE: + * + * glutGet() -- X11 tests passed, but check if all enums + * handled (what about Win32?) + * glutDeviceGet() -- X11 tests passed, but check if all enums + * handled (what about Win32?) + * glutGetModifiers() -- OK, but could also remove the limitation + * glutLayerGet() -- what about GLUT_NORMAL_DAMAGED? + * + * The fail-on-call policy will help adding the most needed things imho. + */ + +/* -- LOCAL DEFINITIONS ---------------------------------------------------- */ + +/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ + +#if TARGET_HOST_POSIX_X11 +/* + * Queries the GL context about some attributes + */ +static int fghGetConfig( int attribute ) +{ + int returnValue = 0; + int result; /* Not checked */ + + if( fgStructure.CurrentWindow ) + result = glXGetFBConfigAttrib( fgDisplay.Display, + *(fgStructure.CurrentWindow->Window.FBConfig), + attribute, + &returnValue ); + + return returnValue; +} +#endif + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * General settings assignment method + */ +void FGAPIENTRY glutSetOption( GLenum eWhat, int value ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetOption" ); + + /* + * XXX In chronological code add order. (WHY in that order?) + */ + switch( eWhat ) + { + case GLUT_INIT_WINDOW_X: + fgState.Position.X = (GLint)value; + break; + + case GLUT_INIT_WINDOW_Y: + fgState.Position.Y = (GLint)value; + break; + + case GLUT_INIT_WINDOW_WIDTH: + fgState.Size.X = (GLint)value; + break; + + case GLUT_INIT_WINDOW_HEIGHT: + fgState.Size.Y = (GLint)value; + break; + + case GLUT_INIT_DISPLAY_MODE: + fgState.DisplayMode = (unsigned int)value; + break; + + case GLUT_ACTION_ON_WINDOW_CLOSE: + fgState.ActionOnWindowClose = value; + break; + + case GLUT_RENDERING_CONTEXT: + fgState.UseCurrentContext = + ( value == GLUT_USE_CURRENT_CONTEXT ) ? GL_TRUE : GL_FALSE; + break; + + case GLUT_DIRECT_RENDERING: + fgState.DirectContext = value; + break; + + case GLUT_WINDOW_CURSOR: + if( fgStructure.CurrentWindow != NULL ) + fgStructure.CurrentWindow->State.Cursor = value; + break; + + case GLUT_AUX: + fgState.AuxiliaryBufferNumber = value; + break; + + case GLUT_MULTISAMPLE: + fgState.SampleNumber = value; + break; + + default: + fgWarning( "glutSetOption(): missing enum handle %d", eWhat ); + break; + } +} + +#if TARGET_HOST_MS_WINDOWS +/* The following include file is available from SGI but is not standard: + * #include + * So we copy the necessary parts out of it to support the multisampling query + */ +#define WGL_SAMPLES_ARB 0x2042 +#endif + + +/* + * General settings query method + */ +int FGAPIENTRY glutGet( GLenum eWhat ) +{ +#if TARGET_HOST_MS_WINDOWS + int returnValue ; + GLboolean boolValue ; +#endif + + int nsamples = 0; + + switch (eWhat) + { + case GLUT_INIT_STATE: + return fgState.Initialised; + + case GLUT_ELAPSED_TIME: + return fgElapsedTime(); + } + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGet" ); + + /* XXX In chronological code add order. (WHY in that order?) */ + switch( eWhat ) + { + /* Following values are stored in fgState and fgDisplay global structures */ + case GLUT_SCREEN_WIDTH: return fgDisplay.ScreenWidth ; + case GLUT_SCREEN_HEIGHT: return fgDisplay.ScreenHeight ; + case GLUT_SCREEN_WIDTH_MM: return fgDisplay.ScreenWidthMM ; + case GLUT_SCREEN_HEIGHT_MM: return fgDisplay.ScreenHeightMM; + case GLUT_INIT_WINDOW_X: return fgState.Position.Use ? + fgState.Position.X : -1 ; + case GLUT_INIT_WINDOW_Y: return fgState.Position.Use ? + fgState.Position.Y : -1 ; + case GLUT_INIT_WINDOW_WIDTH: return fgState.Size.Use ? + fgState.Size.X : -1 ; + case GLUT_INIT_WINDOW_HEIGHT: return fgState.Size.Use ? + fgState.Size.Y : -1 ; + case GLUT_INIT_DISPLAY_MODE: return fgState.DisplayMode ; + case GLUT_INIT_MAJOR_VERSION: return fgState.MajorVersion ; + case GLUT_INIT_MINOR_VERSION: return fgState.MinorVersion ; + case GLUT_INIT_FLAGS: return fgState.ContextFlags ; + case GLUT_INIT_PROFILE: return fgState.ContextProfile ; + +#if TARGET_HOST_POSIX_X11 + /* + * The window/context specific queries are handled mostly by + * fghGetConfig(). + */ + case GLUT_WINDOW_NUM_SAMPLES: +#ifdef GLX_VERSION_1_3 + glGetIntegerv(GL_SAMPLES, &nsamples); +#endif + return nsamples; + + /* + * The rest of GLX queries under X are general enough to use a macro to + * check them + */ +# define GLX_QUERY(a,b) case a: return fghGetConfig( b ); + + GLX_QUERY( GLUT_WINDOW_RGBA, GLX_RGBA ); + GLX_QUERY( GLUT_WINDOW_DOUBLEBUFFER, GLX_DOUBLEBUFFER ); + GLX_QUERY( GLUT_WINDOW_BUFFER_SIZE, GLX_BUFFER_SIZE ); + GLX_QUERY( GLUT_WINDOW_STENCIL_SIZE, GLX_STENCIL_SIZE ); + GLX_QUERY( GLUT_WINDOW_DEPTH_SIZE, GLX_DEPTH_SIZE ); + GLX_QUERY( GLUT_WINDOW_RED_SIZE, GLX_RED_SIZE ); + GLX_QUERY( GLUT_WINDOW_GREEN_SIZE, GLX_GREEN_SIZE ); + GLX_QUERY( GLUT_WINDOW_BLUE_SIZE, GLX_BLUE_SIZE ); + GLX_QUERY( GLUT_WINDOW_ALPHA_SIZE, GLX_ALPHA_SIZE ); + GLX_QUERY( GLUT_WINDOW_ACCUM_RED_SIZE, GLX_ACCUM_RED_SIZE ); + GLX_QUERY( GLUT_WINDOW_ACCUM_GREEN_SIZE, GLX_ACCUM_GREEN_SIZE ); + GLX_QUERY( GLUT_WINDOW_ACCUM_BLUE_SIZE, GLX_ACCUM_BLUE_SIZE ); + GLX_QUERY( GLUT_WINDOW_ACCUM_ALPHA_SIZE, GLX_ACCUM_ALPHA_SIZE ); + GLX_QUERY( GLUT_WINDOW_STEREO, GLX_STEREO ); + +# undef GLX_QUERY + + /* Colormap size is handled in a bit different way than all the rest */ + case GLUT_WINDOW_COLORMAP_SIZE: + if( (fghGetConfig( GLX_RGBA )) || (fgStructure.CurrentWindow == NULL) ) + { + /* + * We've got a RGBA visual, so there is no colormap at all. + * The other possibility is that we have no current window set. + */ + return 0; + } + else + { + const GLXFBConfig * fbconfig = + fgStructure.CurrentWindow->Window.FBConfig; + + XVisualInfo * visualInfo = + glXGetVisualFromFBConfig( fgDisplay.Display, *fbconfig ); + + const int result = visualInfo->visual->map_entries; + + XFree(visualInfo); + + return result; + } + + /* + * Those calls are somewhat similiar, as they use XGetWindowAttributes() + * function + */ + case GLUT_WINDOW_X: + case GLUT_WINDOW_Y: + case GLUT_WINDOW_BORDER_WIDTH: + case GLUT_WINDOW_HEADER_HEIGHT: + { + int x, y; + Window w; + + if( fgStructure.CurrentWindow == NULL ) + return 0; + + XTranslateCoordinates( + fgDisplay.Display, + fgStructure.CurrentWindow->Window.Handle, + fgDisplay.RootWindow, + 0, 0, &x, &y, &w); + + switch ( eWhat ) + { + case GLUT_WINDOW_X: return x; + case GLUT_WINDOW_Y: return y; + } + + if ( w == 0 ) + return 0; + XTranslateCoordinates( + fgDisplay.Display, + fgStructure.CurrentWindow->Window.Handle, + w, 0, 0, &x, &y, &w); + + switch ( eWhat ) + { + case GLUT_WINDOW_BORDER_WIDTH: return x; + case GLUT_WINDOW_HEADER_HEIGHT: return y; + } + } + + case GLUT_WINDOW_WIDTH: + case GLUT_WINDOW_HEIGHT: + { + XWindowAttributes winAttributes; + + if( fgStructure.CurrentWindow == NULL ) + return 0; + XGetWindowAttributes( + fgDisplay.Display, + fgStructure.CurrentWindow->Window.Handle, + &winAttributes + ); + switch ( eWhat ) + { + case GLUT_WINDOW_WIDTH: return winAttributes.width ; + case GLUT_WINDOW_HEIGHT: return winAttributes.height ; + } + } + + /* I do not know yet if there will be a fgChooseVisual() function for Win32 */ + case GLUT_DISPLAY_MODE_POSSIBLE: + { + /* We should not have to call fgChooseFBConfig again here. */ + GLXFBConfig * fbconfig; + int isPossible; + + fbconfig = fgChooseFBConfig(NULL); + + if (fbconfig == NULL) + { + isPossible = 0; + } + else + { + isPossible = 1; + XFree(fbconfig); + } + + return isPossible; + } + + /* This is system-dependant */ + case GLUT_WINDOW_FORMAT_ID: + if( fgStructure.CurrentWindow == NULL ) + return 0; + + return fghGetConfig( GLX_VISUAL_ID ); + +#elif TARGET_HOST_MS_WINDOWS + + case GLUT_WINDOW_NUM_SAMPLES: + glGetIntegerv(WGL_SAMPLES_ARB, &nsamples); + return nsamples; + + /* Handle the OpenGL inquiries */ + case GLUT_WINDOW_RGBA: +#if defined(_WIN32_WCE) + boolValue = (GLboolean)0; /* WinCE doesn't support this feature */ +#else + glGetBooleanv ( GL_RGBA_MODE, &boolValue ); + returnValue = boolValue ? 1 : 0; +#endif + return returnValue; + case GLUT_WINDOW_DOUBLEBUFFER: +#if defined(_WIN32_WCE) + boolValue = (GLboolean)0; /* WinCE doesn't support this feature */ +#else + glGetBooleanv ( GL_DOUBLEBUFFER, &boolValue ); + returnValue = boolValue ? 1 : 0; +#endif + return returnValue; + case GLUT_WINDOW_STEREO: +#if defined(_WIN32_WCE) + boolValue = (GLboolean)0; /* WinCE doesn't support this feature */ +#else + glGetBooleanv ( GL_STEREO, &boolValue ); + returnValue = boolValue ? 1 : 0; +#endif + return returnValue; + + case GLUT_WINDOW_RED_SIZE: + glGetIntegerv ( GL_RED_BITS, &returnValue ); + return returnValue; + case GLUT_WINDOW_GREEN_SIZE: + glGetIntegerv ( GL_GREEN_BITS, &returnValue ); + return returnValue; + case GLUT_WINDOW_BLUE_SIZE: + glGetIntegerv ( GL_BLUE_BITS, &returnValue ); + return returnValue; + case GLUT_WINDOW_ALPHA_SIZE: + glGetIntegerv ( GL_ALPHA_BITS, &returnValue ); + return returnValue; + case GLUT_WINDOW_ACCUM_RED_SIZE: +#if defined(_WIN32_WCE) + returnValue = 0; /* WinCE doesn't support this feature */ +#else + glGetIntegerv ( GL_ACCUM_RED_BITS, &returnValue ); +#endif + return returnValue; + case GLUT_WINDOW_ACCUM_GREEN_SIZE: +#if defined(_WIN32_WCE) + returnValue = 0; /* WinCE doesn't support this feature */ +#else + glGetIntegerv ( GL_ACCUM_GREEN_BITS, &returnValue ); +#endif + return returnValue; + case GLUT_WINDOW_ACCUM_BLUE_SIZE: +#if defined(_WIN32_WCE) + returnValue = 0; /* WinCE doesn't support this feature */ +#else + glGetIntegerv ( GL_ACCUM_BLUE_BITS, &returnValue ); +#endif + return returnValue; + case GLUT_WINDOW_ACCUM_ALPHA_SIZE: +#if defined(_WIN32_WCE) + returnValue = 0; /* WinCE doesn't support this feature */ +#else + glGetIntegerv ( GL_ACCUM_ALPHA_BITS, &returnValue ); +#endif + return returnValue; + case GLUT_WINDOW_DEPTH_SIZE: + glGetIntegerv ( GL_DEPTH_BITS, &returnValue ); + return returnValue; + + case GLUT_WINDOW_BUFFER_SIZE: + returnValue = 1 ; /* ????? */ + return returnValue; + case GLUT_WINDOW_STENCIL_SIZE: + returnValue = 0 ; /* ????? */ + return returnValue; + + case GLUT_WINDOW_X: + case GLUT_WINDOW_Y: + case GLUT_WINDOW_WIDTH: + case GLUT_WINDOW_HEIGHT: + { + /* + * There is considerable confusion about the "right thing to + * do" concerning window size and position. GLUT itself is + * not consistent between Windows and UNIX/X11; since + * platform independence is a virtue for "freeglut", we + * decided to break with GLUT's behaviour. + * + * Under UNIX/X11, it is apparently not possible to get the + * window border sizes in order to subtract them off the + * window's initial position until some time after the window + * has been created. Therefore we decided on the following + * behaviour, both under Windows and under UNIX/X11: + * - When you create a window with position (x,y) and size + * (w,h), the upper left hand corner of the outside of the + * window is at (x,y) and the size of the drawable area is + * (w,h). + * - When you query the size and position of the window--as + * is happening here for Windows--"freeglut" will return + * the size of the drawable area--the (w,h) that you + * specified when you created the window--and the coordinates + * of the upper left hand corner of the drawable + * area--which is NOT the (x,y) you specified. + */ + + RECT winRect; + + freeglut_return_val_if_fail( fgStructure.CurrentWindow != NULL, 0 ); + +#if defined(_WIN32_WCE) + GetWindowRect( fgStructure.CurrentWindow->Window.Handle, &winRect ); +#else + winRect = fghGetClientArea(fgStructure.CurrentWindow, FALSE); +#endif /* defined(_WIN32_WCE) */ + + switch( eWhat ) + { + case GLUT_WINDOW_X: return winRect.left ; + case GLUT_WINDOW_Y: return winRect.top ; + case GLUT_WINDOW_WIDTH: return winRect.right - winRect.left; + case GLUT_WINDOW_HEIGHT: return winRect.bottom - winRect.top; + } + } + break; + + case GLUT_WINDOW_BORDER_WIDTH : + case GLUT_WINDOW_HEADER_HEIGHT : +#if defined(_WIN32_WCE) + return 0; +#else + { + DWORD windowStyle; + + if (fgStructure.CurrentWindow && fgStructure.CurrentWindow->Window.Handle) + windowStyle = GetWindowLong(fgStructure.CurrentWindow->Window.Handle, GWL_STYLE); + else + /* If no window, return sizes for a default window with title bar and border */ + windowStyle = WS_OVERLAPPEDWINDOW; + + switch( eWhat ) + { + case GLUT_WINDOW_BORDER_WIDTH: + { + int xBorderWidth, yBorderWidth; + fghGetBorderWidth(windowStyle, &xBorderWidth, &yBorderWidth); + return xBorderWidth; + } + case GLUT_WINDOW_HEADER_HEIGHT: + /* Need to query for WS_MAXIMIZEBOX to see if we have a title bar, the WS_CAPTION query is also true for a WS_DLGFRAME only... */ + return (windowStyle & WS_MAXIMIZEBOX)? GetSystemMetrics( SM_CYCAPTION ) : 0; + } + } +#endif /* defined(_WIN32_WCE) */ + + case GLUT_DISPLAY_MODE_POSSIBLE: +#if defined(_WIN32_WCE) + return 0; +#else + return fgSetupPixelFormat( fgStructure.CurrentWindow, GL_TRUE, + PFD_MAIN_PLANE ); +#endif /* defined(_WIN32_WCE) */ + + + case GLUT_WINDOW_FORMAT_ID: +#if !defined(_WIN32_WCE) + if( fgStructure.CurrentWindow != NULL ) + return GetPixelFormat( fgStructure.CurrentWindow->Window.Device ); +#endif /* defined(_WIN32_WCE) */ + return 0; + +#endif + + /* The window structure queries */ + case GLUT_WINDOW_PARENT: + if( fgStructure.CurrentWindow == NULL ) return 0; + if( fgStructure.CurrentWindow->Parent == NULL ) return 0; + return fgStructure.CurrentWindow->Parent->ID; + + case GLUT_WINDOW_NUM_CHILDREN: + if( fgStructure.CurrentWindow == NULL ) + return 0; + return fgListLength( &fgStructure.CurrentWindow->Children ); + + case GLUT_WINDOW_CURSOR: + if( fgStructure.CurrentWindow == NULL ) + return 0; + return fgStructure.CurrentWindow->State.Cursor; + + case GLUT_MENU_NUM_ITEMS: + if( fgStructure.CurrentMenu == NULL ) + return 0; + return fgListLength( &fgStructure.CurrentMenu->Entries ); + + case GLUT_ACTION_ON_WINDOW_CLOSE: + return fgState.ActionOnWindowClose; + + case GLUT_VERSION : + return VERSION_MAJOR * 10000 + VERSION_MINOR * 100 + VERSION_PATCH; + + case GLUT_RENDERING_CONTEXT: + return fgState.UseCurrentContext ? GLUT_USE_CURRENT_CONTEXT + : GLUT_CREATE_NEW_CONTEXT; + + case GLUT_DIRECT_RENDERING: + return fgState.DirectContext; + + case GLUT_FULL_SCREEN: + return fgStructure.CurrentWindow->State.IsFullscreen; + + case GLUT_AUX: + return fgState.AuxiliaryBufferNumber; + + case GLUT_MULTISAMPLE: + return fgState.SampleNumber; + + default: + fgWarning( "glutGet(): missing enum handle %d", eWhat ); + break; + } + return -1; +} + +/* + * Returns various device information. + */ +int FGAPIENTRY glutDeviceGet( GLenum eWhat ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDeviceGet" ); + + /* XXX WARNING: we are mostly lying in this function. */ + switch( eWhat ) + { + case GLUT_HAS_KEYBOARD: + /* + * Win32 is assumed a keyboard, and this cannot be queried, + * except for WindowsCE. + * + * X11 has a core keyboard by definition, although it can + * be present as a virtual/dummy keyboard. For now, there + * is no reliable way to tell if a real keyboard is present. + */ +#if defined(_WIN32_CE) + return ( GetKeyboardStatus() & KBDI_KEYBOARD_PRESENT ) ? 1 : 0; +# if FREEGLUT_LIB_PRAGMAS +# pragma comment (lib,"Kbdui.lib") +# endif + +#else + return 1; +#endif + +#if TARGET_HOST_POSIX_X11 + + /* X11 has a mouse by definition */ + case GLUT_HAS_MOUSE: + return 1 ; + + case GLUT_NUM_MOUSE_BUTTONS: + /* We should be able to pass NULL when the last argument is zero, + * but at least one X server has a bug where this causes a segfault. + * + * In XFree86/Xorg servers, a mouse wheel is seen as two buttons + * rather than an Axis; "freeglut_main.c" expects this when + * checking for a wheel event. + */ + { + unsigned char map; + int nbuttons = XGetPointerMapping(fgDisplay.Display, &map,0); + return nbuttons; + } + +#elif TARGET_HOST_MS_WINDOWS + + case GLUT_HAS_MOUSE: + /* + * MS Windows can be booted without a mouse. + */ + return GetSystemMetrics( SM_MOUSEPRESENT ); + + case GLUT_NUM_MOUSE_BUTTONS: +# if defined(_WIN32_WCE) + return 1; +# else + return GetSystemMetrics( SM_CMOUSEBUTTONS ); +# endif +#endif + + case GLUT_HAS_JOYSTICK: + return fgJoystickDetect (); + + case GLUT_OWNS_JOYSTICK: + return fgState.JoysticksInitialised; + + case GLUT_JOYSTICK_POLL_RATE: + return fgStructure.CurrentWindow ? fgStructure.CurrentWindow->State.JoystickPollRate : 0; + + /* XXX The following two are only for Joystick 0 but this is an improvement */ + case GLUT_JOYSTICK_BUTTONS: + return glutJoystickGetNumButtons ( 0 ); + + case GLUT_JOYSTICK_AXES: + return glutJoystickGetNumAxes ( 0 ); + + case GLUT_HAS_DIAL_AND_BUTTON_BOX: + return fgInputDeviceDetect (); + + case GLUT_NUM_DIALS: + if ( fgState.InputDevsInitialised ) return 8; + return 0; + + case GLUT_NUM_BUTTON_BOX_BUTTONS: + return 0; + + case GLUT_HAS_SPACEBALL: + return fgHasSpaceball(); + + case GLUT_HAS_TABLET: + return 0; + + case GLUT_NUM_SPACEBALL_BUTTONS: + return fgSpaceballNumButtons(); + + case GLUT_NUM_TABLET_BUTTONS: + return 0; + + case GLUT_DEVICE_IGNORE_KEY_REPEAT: + return fgStructure.CurrentWindow ? fgStructure.CurrentWindow->State.IgnoreKeyRepeat : 0; + + case GLUT_DEVICE_KEY_REPEAT: + return fgState.KeyRepeat; + + default: + fgWarning( "glutDeviceGet(): missing enum handle %d", eWhat ); + break; + } + + /* And now -- the failure. */ + return -1; +} + +/* + * This should return the current state of ALT, SHIFT and CTRL keys. + */ +int FGAPIENTRY glutGetModifiers( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetModifiers" ); + if( fgState.Modifiers == INVALID_MODIFIERS ) + { + fgWarning( "glutGetModifiers() called outside an input callback" ); + return 0; + } + + return fgState.Modifiers; +} + +/* + * Return the state of the GLUT API overlay subsystem. A misery ;-) + */ +int FGAPIENTRY glutLayerGet( GLenum eWhat ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutLayerGet" ); + + /* + * This is easy as layers are not implemented ;-) + * + * XXX Can we merge the UNIX/X11 and WIN32 sections? Or + * XXX is overlay support planned? + */ + switch( eWhat ) + { + +#if TARGET_HOST_POSIX_X11 + + case GLUT_OVERLAY_POSSIBLE: + return 0; + + case GLUT_LAYER_IN_USE: + return GLUT_NORMAL; + + case GLUT_HAS_OVERLAY: + return 0; + + case GLUT_TRANSPARENT_INDEX: + /* + * Return just anything, which is always defined as zero + * + * XXX HUH? + */ + return 0; + + case GLUT_NORMAL_DAMAGED: + /* XXX Actually I do not know. Maybe. */ + return 0; + + case GLUT_OVERLAY_DAMAGED: + return -1; + +#elif TARGET_HOST_MS_WINDOWS + + case GLUT_OVERLAY_POSSIBLE: +/* return fgSetupPixelFormat( fgStructure.CurrentWindow, GL_TRUE, + PFD_OVERLAY_PLANE ); */ + return 0 ; + + case GLUT_LAYER_IN_USE: + return GLUT_NORMAL; + + case GLUT_HAS_OVERLAY: + return 0; + + case GLUT_TRANSPARENT_INDEX: + /* + * Return just anything, which is always defined as zero + * + * XXX HUH? + */ + return 0; + + case GLUT_NORMAL_DAMAGED: + /* XXX Actually I do not know. Maybe. */ + return 0; + + case GLUT_OVERLAY_DAMAGED: + return -1; +#endif + + default: + fgWarning( "glutLayerGet(): missing enum handle %d", eWhat ); + break; + } + + /* And fail. That's good. Programs do love failing. */ + return -1; +} + +int * FGAPIENTRY glutGetModeValues(GLenum eWhat, int * size) +{ + int * array; + +#if TARGET_HOST_POSIX_X11 + int attributes[9]; + GLXFBConfig * fbconfigArray; /* Array of FBConfigs */ + int fbconfigArraySize; /* Number of FBConfigs in the array */ + int attribute_name = 0; +#endif + + FREEGLUT_EXIT_IF_NOT_INITIALISED("glutGetModeValues"); + + array = NULL; + *size = 0; + + switch (eWhat) + { +#if TARGET_HOST_POSIX_X11 + case GLUT_AUX: + case GLUT_MULTISAMPLE: + + attributes[0] = GLX_BUFFER_SIZE; + attributes[1] = GLX_DONT_CARE; + + switch (eWhat) + { + case GLUT_AUX: + /* + FBConfigs are now sorted by increasing number of auxiliary + buffers. We want at least one buffer. + */ + attributes[2] = GLX_AUX_BUFFERS; + attributes[3] = 1; + attributes[4] = None; + + attribute_name = GLX_AUX_BUFFERS; + + break; + + + case GLUT_MULTISAMPLE: + attributes[2] = GLX_AUX_BUFFERS; + attributes[3] = GLX_DONT_CARE; + attributes[4] = GLX_SAMPLE_BUFFERS; + attributes[5] = 1; + /* + FBConfigs are now sorted by increasing number of samples per + pixel. We want at least one sample. + */ + attributes[6] = GLX_SAMPLES; + attributes[7] = 1; + attributes[8] = None; + + attribute_name = GLX_SAMPLES; + + break; + } + + fbconfigArray = glXChooseFBConfig(fgDisplay.Display, + fgDisplay.Screen, + attributes, + &fbconfigArraySize); + + if (fbconfigArray != NULL) + { + int * temp_array; + int result; /* Returned by glXGetFBConfigAttrib. Not checked. */ + int previous_value; + int i; + + temp_array = malloc(sizeof(int) * fbconfigArraySize); + previous_value = 0; + + for (i = 0; i < fbconfigArraySize; i++) + { + int value; + + result = glXGetFBConfigAttrib(fgDisplay.Display, + fbconfigArray[i], + attribute_name, + &value); + if (value > previous_value) + { + temp_array[*size] = value; + previous_value = value; + (*size)++; + } + } + + array = malloc(sizeof(int) * (*size)); + for (i = 0; i < *size; i++) + { + array[i] = temp_array[i]; + } + + free(temp_array); + XFree(fbconfigArray); + } + + break; +#endif + + default: + break; + } + + return array; +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_stroke_mono_roman.c b/examples/opengl-framework/freeglut/freeglut_stroke_mono_roman.c new file mode 100644 index 00000000..ec86d56f --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_stroke_mono_roman.c @@ -0,0 +1,2849 @@ +/* + * freeglut_stroke_mono_roman.c + * + * freeglut Monospace Roman stroke font definition + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 16 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + + +/* This file has been automatically generated by the genstroke utility. */ + +#include +#include "freeglut_internal.h" + +/* char: 0x20 */ + +static const SFG_StrokeStrip ch32st[] = +{ + { 0, NULL } +}; + +static const SFG_StrokeChar ch32 = {104.762f,0,ch32st}; + +/* char: 0x21 */ + +static const SFG_StrokeVertex ch33st0[] = +{ + {52.381f,100.0f}, + {52.381f,33.3333f} +}; + +static const SFG_StrokeVertex ch33st1[] = +{ + {52.381f,9.5238f}, + {47.6191f,4.7619f}, + {52.381f,0.0f}, + {57.1429f,4.7619f}, + {52.381f,9.5238f} +}; + +static const SFG_StrokeStrip ch33st[] = +{ + {2,ch33st0}, + {5,ch33st1} +}; + +static const SFG_StrokeChar ch33 = {104.762f,2,ch33st}; + +/* char: 0x22 */ + +static const SFG_StrokeVertex ch34st0[] = +{ + {33.3334f,100.0f}, + {33.3334f,66.6667f} +}; + +static const SFG_StrokeVertex ch34st1[] = +{ + {71.4286f,100.0f}, + {71.4286f,66.6667f} +}; + +static const SFG_StrokeStrip ch34st[] = +{ + {2,ch34st0}, + {2,ch34st1} +}; + +static const SFG_StrokeChar ch34 = {104.762f,2,ch34st}; + +/* char: 0x23 */ + +static const SFG_StrokeVertex ch35st0[] = +{ + {54.7619f,119.048f}, + {21.4286f,-33.3333f} +}; + +static const SFG_StrokeVertex ch35st1[] = +{ + {83.3334f,119.048f}, + {50.0f,-33.3333f} +}; + +static const SFG_StrokeVertex ch35st2[] = +{ + {21.4286f,57.1429f}, + {88.0952f,57.1429f} +}; + +static const SFG_StrokeVertex ch35st3[] = +{ + {16.6667f,28.5714f}, + {83.3334f,28.5714f} +}; + +static const SFG_StrokeStrip ch35st[] = +{ + {2,ch35st0}, + {2,ch35st1}, + {2,ch35st2}, + {2,ch35st3} +}; + +static const SFG_StrokeChar ch35 = {104.762f,4,ch35st}; + +/* char: 0x24 */ + +static const SFG_StrokeVertex ch36st0[] = +{ + {42.8571f,119.048f}, + {42.8571f,-19.0476f} +}; + +static const SFG_StrokeVertex ch36st1[] = +{ + {61.9047f,119.048f}, + {61.9047f,-19.0476f} +}; + +static const SFG_StrokeVertex ch36st2[] = +{ + {85.7143f,85.7143f}, + {76.1905f,95.2381f}, + {61.9047f,100.0f}, + {42.8571f,100.0f}, + {28.5714f,95.2381f}, + {19.0476f,85.7143f}, + {19.0476f,76.1905f}, + {23.8095f,66.6667f}, + {28.5714f,61.9048f}, + {38.0952f,57.1429f}, + {66.6666f,47.619f}, + {76.1905f,42.8571f}, + {80.9524f,38.0952f}, + {85.7143f,28.5714f}, + {85.7143f,14.2857f}, + {76.1905f,4.7619f}, + {61.9047f,0.0f}, + {42.8571f,0.0f}, + {28.5714f,4.7619f}, + {19.0476f,14.2857f} +}; + +static const SFG_StrokeStrip ch36st[] = +{ + {2,ch36st0}, + {2,ch36st1}, + {20,ch36st2} +}; + +static const SFG_StrokeChar ch36 = {104.762f,3,ch36st}; + +/* char: 0x25 */ + +static const SFG_StrokeVertex ch37st0[] = +{ + {95.2381f,100.0f}, + {9.5238f,0.0f} +}; + +static const SFG_StrokeVertex ch37st1[] = +{ + {33.3333f,100.0f}, + {42.8571f,90.4762f}, + {42.8571f,80.9524f}, + {38.0952f,71.4286f}, + {28.5714f,66.6667f}, + {19.0476f,66.6667f}, + {9.5238f,76.1905f}, + {9.5238f,85.7143f}, + {14.2857f,95.2381f}, + {23.8095f,100.0f}, + {33.3333f,100.0f}, + {42.8571f,95.2381f}, + {57.1428f,90.4762f}, + {71.4286f,90.4762f}, + {85.7143f,95.2381f}, + {95.2381f,100.0f} +}; + +static const SFG_StrokeVertex ch37st2[] = +{ + {76.1905f,33.3333f}, + {66.6667f,28.5714f}, + {61.9048f,19.0476f}, + {61.9048f,9.5238f}, + {71.4286f,0.0f}, + {80.9524f,0.0f}, + {90.4762f,4.7619f}, + {95.2381f,14.2857f}, + {95.2381f,23.8095f}, + {85.7143f,33.3333f}, + {76.1905f,33.3333f} +}; + +static const SFG_StrokeStrip ch37st[] = +{ + {2,ch37st0}, + {16,ch37st1}, + {11,ch37st2} +}; + +static const SFG_StrokeChar ch37 = {104.762f,3,ch37st}; + +/* char: 0x26 */ + +static const SFG_StrokeVertex ch38st0[] = +{ + {100.0f,57.1429f}, + {100.0f,61.9048f}, + {95.2381f,66.6667f}, + {90.4762f,66.6667f}, + {85.7143f,61.9048f}, + {80.9524f,52.381f}, + {71.4286f,28.5714f}, + {61.9048f,14.2857f}, + {52.3809f,4.7619f}, + {42.8571f,0.0f}, + {23.8095f,0.0f}, + {14.2857f,4.7619f}, + {9.5238f,9.5238f}, + {4.7619f,19.0476f}, + {4.7619f,28.5714f}, + {9.5238f,38.0952f}, + {14.2857f,42.8571f}, + {47.619f,61.9048f}, + {52.3809f,66.6667f}, + {57.1429f,76.1905f}, + {57.1429f,85.7143f}, + {52.3809f,95.2381f}, + {42.8571f,100.0f}, + {33.3333f,95.2381f}, + {28.5714f,85.7143f}, + {28.5714f,76.1905f}, + {33.3333f,61.9048f}, + {42.8571f,47.619f}, + {66.6667f,14.2857f}, + {76.1905f,4.7619f}, + {85.7143f,0.0f}, + {95.2381f,0.0f}, + {100.0f,4.7619f}, + {100.0f,9.5238f} +}; + +static const SFG_StrokeStrip ch38st[] = +{ + {34,ch38st0} +}; + +static const SFG_StrokeChar ch38 = {104.762f,1,ch38st}; + +/* char: 0x27 */ + +static const SFG_StrokeVertex ch39st0[] = +{ + {52.381f,100.0f}, + {52.381f,66.6667f} +}; + +static const SFG_StrokeStrip ch39st[] = +{ + {2,ch39st0} +}; + +static const SFG_StrokeChar ch39 = {104.762f,1,ch39st}; + +/* char: 0x28 */ + +static const SFG_StrokeVertex ch40st0[] = +{ + {69.0476f,119.048f}, + {59.5238f,109.524f}, + {50.0f,95.2381f}, + {40.4762f,76.1905f}, + {35.7143f,52.381f}, + {35.7143f,33.3333f}, + {40.4762f,9.5238f}, + {50.0f,-9.5238f}, + {59.5238f,-23.8095f}, + {69.0476f,-33.3333f} +}; + +static const SFG_StrokeStrip ch40st[] = +{ + {10,ch40st0} +}; + +static const SFG_StrokeChar ch40 = {104.762f,1,ch40st}; + +/* char: 0x29 */ + +static const SFG_StrokeVertex ch41st0[] = +{ + {35.7143f,119.048f}, + {45.2381f,109.524f}, + {54.7619f,95.2381f}, + {64.2857f,76.1905f}, + {69.0476f,52.381f}, + {69.0476f,33.3333f}, + {64.2857f,9.5238f}, + {54.7619f,-9.5238f}, + {45.2381f,-23.8095f}, + {35.7143f,-33.3333f} +}; + +static const SFG_StrokeStrip ch41st[] = +{ + {10,ch41st0} +}; + +static const SFG_StrokeChar ch41 = {104.762f,1,ch41st}; + +/* char: 0x2a */ + +static const SFG_StrokeVertex ch42st0[] = +{ + {52.381f,71.4286f}, + {52.381f,14.2857f} +}; + +static const SFG_StrokeVertex ch42st1[] = +{ + {28.5715f,57.1429f}, + {76.1905f,28.5714f} +}; + +static const SFG_StrokeVertex ch42st2[] = +{ + {76.1905f,57.1429f}, + {28.5715f,28.5714f} +}; + +static const SFG_StrokeStrip ch42st[] = +{ + {2,ch42st0}, + {2,ch42st1}, + {2,ch42st2} +}; + +static const SFG_StrokeChar ch42 = {104.762f,3,ch42st}; + +/* char: 0x2b */ + +static const SFG_StrokeVertex ch43st0[] = +{ + {52.3809f,85.7143f}, + {52.3809f,0.0f} +}; + +static const SFG_StrokeVertex ch43st1[] = +{ + {9.5238f,42.8571f}, + {95.2381f,42.8571f} +}; + +static const SFG_StrokeStrip ch43st[] = +{ + {2,ch43st0}, + {2,ch43st1} +}; + +static const SFG_StrokeChar ch43 = {104.762f,2,ch43st}; + +/* char: 0x2c */ + +static const SFG_StrokeVertex ch44st0[] = +{ + {57.1429f,4.7619f}, + {52.381f,0.0f}, + {47.6191f,4.7619f}, + {52.381f,9.5238f}, + {57.1429f,4.7619f}, + {57.1429f,-4.7619f}, + {52.381f,-14.2857f}, + {47.6191f,-19.0476f} +}; + +static const SFG_StrokeStrip ch44st[] = +{ + {8,ch44st0} +}; + +static const SFG_StrokeChar ch44 = {104.762f,1,ch44st}; + +/* char: 0x2d */ + +static const SFG_StrokeVertex ch45st0[] = +{ + {9.5238f,42.8571f}, + {95.2381f,42.8571f} +}; + +static const SFG_StrokeStrip ch45st[] = +{ + {2,ch45st0} +}; + +static const SFG_StrokeChar ch45 = {104.762f,1,ch45st}; + +/* char: 0x2e */ + +static const SFG_StrokeVertex ch46st0[] = +{ + {52.381f,9.5238f}, + {47.6191f,4.7619f}, + {52.381f,0.0f}, + {57.1429f,4.7619f}, + {52.381f,9.5238f} +}; + +static const SFG_StrokeStrip ch46st[] = +{ + {5,ch46st0} +}; + +static const SFG_StrokeChar ch46 = {104.762f,1,ch46st}; + +/* char: 0x2f */ + +static const SFG_StrokeVertex ch47st0[] = +{ + {19.0476f,-14.2857f}, + {85.7143f,100.0f} +}; + +static const SFG_StrokeStrip ch47st[] = +{ + {2,ch47st0} +}; + +static const SFG_StrokeChar ch47 = {104.762f,1,ch47st}; + +/* char: 0x30 */ + +static const SFG_StrokeVertex ch48st0[] = +{ + {47.619f,100.0f}, + {33.3333f,95.2381f}, + {23.8095f,80.9524f}, + {19.0476f,57.1429f}, + {19.0476f,42.8571f}, + {23.8095f,19.0476f}, + {33.3333f,4.7619f}, + {47.619f,0.0f}, + {57.1428f,0.0f}, + {71.4286f,4.7619f}, + {80.9524f,19.0476f}, + {85.7143f,42.8571f}, + {85.7143f,57.1429f}, + {80.9524f,80.9524f}, + {71.4286f,95.2381f}, + {57.1428f,100.0f}, + {47.619f,100.0f} +}; + +static const SFG_StrokeStrip ch48st[] = +{ + {17,ch48st0} +}; + +static const SFG_StrokeChar ch48 = {104.762f,1,ch48st}; + +/* char: 0x31 */ + +static const SFG_StrokeVertex ch49st0[] = +{ + {40.4762f,80.9524f}, + {50.0f,85.7143f}, + {64.2857f,100.0f}, + {64.2857f,0.0f} +}; + +static const SFG_StrokeStrip ch49st[] = +{ + {4,ch49st0} +}; + +static const SFG_StrokeChar ch49 = {104.762f,1,ch49st}; + +/* char: 0x32 */ + +static const SFG_StrokeVertex ch50st0[] = +{ + {23.8095f,76.1905f}, + {23.8095f,80.9524f}, + {28.5714f,90.4762f}, + {33.3333f,95.2381f}, + {42.8571f,100.0f}, + {61.9047f,100.0f}, + {71.4286f,95.2381f}, + {76.1905f,90.4762f}, + {80.9524f,80.9524f}, + {80.9524f,71.4286f}, + {76.1905f,61.9048f}, + {66.6666f,47.619f}, + {19.0476f,0.0f}, + {85.7143f,0.0f} +}; + +static const SFG_StrokeStrip ch50st[] = +{ + {14,ch50st0} +}; + +static const SFG_StrokeChar ch50 = {104.762f,1,ch50st}; + +/* char: 0x33 */ + +static const SFG_StrokeVertex ch51st0[] = +{ + {28.5714f,100.0f}, + {80.9524f,100.0f}, + {52.3809f,61.9048f}, + {66.6666f,61.9048f}, + {76.1905f,57.1429f}, + {80.9524f,52.381f}, + {85.7143f,38.0952f}, + {85.7143f,28.5714f}, + {80.9524f,14.2857f}, + {71.4286f,4.7619f}, + {57.1428f,0.0f}, + {42.8571f,0.0f}, + {28.5714f,4.7619f}, + {23.8095f,9.5238f}, + {19.0476f,19.0476f} +}; + +static const SFG_StrokeStrip ch51st[] = +{ + {15,ch51st0} +}; + +static const SFG_StrokeChar ch51 = {104.762f,1,ch51st}; + +/* char: 0x34 */ + +static const SFG_StrokeVertex ch52st0[] = +{ + {64.2857f,100.0f}, + {16.6667f,33.3333f}, + {88.0952f,33.3333f} +}; + +static const SFG_StrokeVertex ch52st1[] = +{ + {64.2857f,100.0f}, + {64.2857f,0.0f} +}; + +static const SFG_StrokeStrip ch52st[] = +{ + {3,ch52st0}, + {2,ch52st1} +}; + +static const SFG_StrokeChar ch52 = {104.762f,2,ch52st}; + +/* char: 0x35 */ + +static const SFG_StrokeVertex ch53st0[] = +{ + {76.1905f,100.0f}, + {28.5714f,100.0f}, + {23.8095f,57.1429f}, + {28.5714f,61.9048f}, + {42.8571f,66.6667f}, + {57.1428f,66.6667f}, + {71.4286f,61.9048f}, + {80.9524f,52.381f}, + {85.7143f,38.0952f}, + {85.7143f,28.5714f}, + {80.9524f,14.2857f}, + {71.4286f,4.7619f}, + {57.1428f,0.0f}, + {42.8571f,0.0f}, + {28.5714f,4.7619f}, + {23.8095f,9.5238f}, + {19.0476f,19.0476f} +}; + +static const SFG_StrokeStrip ch53st[] = +{ + {17,ch53st0} +}; + +static const SFG_StrokeChar ch53 = {104.762f,1,ch53st}; + +/* char: 0x36 */ + +static const SFG_StrokeVertex ch54st0[] = +{ + {78.5714f,85.7143f}, + {73.8096f,95.2381f}, + {59.5238f,100.0f}, + {50.0f,100.0f}, + {35.7143f,95.2381f}, + {26.1905f,80.9524f}, + {21.4286f,57.1429f}, + {21.4286f,33.3333f}, + {26.1905f,14.2857f}, + {35.7143f,4.7619f}, + {50.0f,0.0f}, + {54.7619f,0.0f}, + {69.0476f,4.7619f}, + {78.5714f,14.2857f}, + {83.3334f,28.5714f}, + {83.3334f,33.3333f}, + {78.5714f,47.619f}, + {69.0476f,57.1429f}, + {54.7619f,61.9048f}, + {50.0f,61.9048f}, + {35.7143f,57.1429f}, + {26.1905f,47.619f}, + {21.4286f,33.3333f} +}; + +static const SFG_StrokeStrip ch54st[] = +{ + {23,ch54st0} +}; + +static const SFG_StrokeChar ch54 = {104.762f,1,ch54st}; + +/* char: 0x37 */ + +static const SFG_StrokeVertex ch55st0[] = +{ + {85.7143f,100.0f}, + {38.0952f,0.0f} +}; + +static const SFG_StrokeVertex ch55st1[] = +{ + {19.0476f,100.0f}, + {85.7143f,100.0f} +}; + +static const SFG_StrokeStrip ch55st[] = +{ + {2,ch55st0}, + {2,ch55st1} +}; + +static const SFG_StrokeChar ch55 = {104.762f,2,ch55st}; + +/* char: 0x38 */ + +static const SFG_StrokeVertex ch56st0[] = +{ + {42.8571f,100.0f}, + {28.5714f,95.2381f}, + {23.8095f,85.7143f}, + {23.8095f,76.1905f}, + {28.5714f,66.6667f}, + {38.0952f,61.9048f}, + {57.1428f,57.1429f}, + {71.4286f,52.381f}, + {80.9524f,42.8571f}, + {85.7143f,33.3333f}, + {85.7143f,19.0476f}, + {80.9524f,9.5238f}, + {76.1905f,4.7619f}, + {61.9047f,0.0f}, + {42.8571f,0.0f}, + {28.5714f,4.7619f}, + {23.8095f,9.5238f}, + {19.0476f,19.0476f}, + {19.0476f,33.3333f}, + {23.8095f,42.8571f}, + {33.3333f,52.381f}, + {47.619f,57.1429f}, + {66.6666f,61.9048f}, + {76.1905f,66.6667f}, + {80.9524f,76.1905f}, + {80.9524f,85.7143f}, + {76.1905f,95.2381f}, + {61.9047f,100.0f}, + {42.8571f,100.0f} +}; + +static const SFG_StrokeStrip ch56st[] = +{ + {29,ch56st0} +}; + +static const SFG_StrokeChar ch56 = {104.762f,1,ch56st}; + +/* char: 0x39 */ + +static const SFG_StrokeVertex ch57st0[] = +{ + {83.3334f,66.6667f}, + {78.5714f,52.381f}, + {69.0476f,42.8571f}, + {54.7619f,38.0952f}, + {50.0f,38.0952f}, + {35.7143f,42.8571f}, + {26.1905f,52.381f}, + {21.4286f,66.6667f}, + {21.4286f,71.4286f}, + {26.1905f,85.7143f}, + {35.7143f,95.2381f}, + {50.0f,100.0f}, + {54.7619f,100.0f}, + {69.0476f,95.2381f}, + {78.5714f,85.7143f}, + {83.3334f,66.6667f}, + {83.3334f,42.8571f}, + {78.5714f,19.0476f}, + {69.0476f,4.7619f}, + {54.7619f,0.0f}, + {45.2381f,0.0f}, + {30.9524f,4.7619f}, + {26.1905f,14.2857f} +}; + +static const SFG_StrokeStrip ch57st[] = +{ + {23,ch57st0} +}; + +static const SFG_StrokeChar ch57 = {104.762f,1,ch57st}; + +/* char: 0x3a */ + +static const SFG_StrokeVertex ch58st0[] = +{ + {52.381f,66.6667f}, + {47.6191f,61.9048f}, + {52.381f,57.1429f}, + {57.1429f,61.9048f}, + {52.381f,66.6667f} +}; + +static const SFG_StrokeVertex ch58st1[] = +{ + {52.381f,9.5238f}, + {47.6191f,4.7619f}, + {52.381f,0.0f}, + {57.1429f,4.7619f}, + {52.381f,9.5238f} +}; + +static const SFG_StrokeStrip ch58st[] = +{ + {5,ch58st0}, + {5,ch58st1} +}; + +static const SFG_StrokeChar ch58 = {104.762f,2,ch58st}; + +/* char: 0x3b */ + +static const SFG_StrokeVertex ch59st0[] = +{ + {52.381f,66.6667f}, + {47.6191f,61.9048f}, + {52.381f,57.1429f}, + {57.1429f,61.9048f}, + {52.381f,66.6667f} +}; + +static const SFG_StrokeVertex ch59st1[] = +{ + {57.1429f,4.7619f}, + {52.381f,0.0f}, + {47.6191f,4.7619f}, + {52.381f,9.5238f}, + {57.1429f,4.7619f}, + {57.1429f,-4.7619f}, + {52.381f,-14.2857f}, + {47.6191f,-19.0476f} +}; + +static const SFG_StrokeStrip ch59st[] = +{ + {5,ch59st0}, + {8,ch59st1} +}; + +static const SFG_StrokeChar ch59 = {104.762f,2,ch59st}; + +/* char: 0x3c */ + +static const SFG_StrokeVertex ch60st0[] = +{ + {90.4762f,85.7143f}, + {14.2857f,42.8571f}, + {90.4762f,0.0f} +}; + +static const SFG_StrokeStrip ch60st[] = +{ + {3,ch60st0} +}; + +static const SFG_StrokeChar ch60 = {104.762f,1,ch60st}; + +/* char: 0x3d */ + +static const SFG_StrokeVertex ch61st0[] = +{ + {9.5238f,57.1429f}, + {95.2381f,57.1429f} +}; + +static const SFG_StrokeVertex ch61st1[] = +{ + {9.5238f,28.5714f}, + {95.2381f,28.5714f} +}; + +static const SFG_StrokeStrip ch61st[] = +{ + {2,ch61st0}, + {2,ch61st1} +}; + +static const SFG_StrokeChar ch61 = {104.762f,2,ch61st}; + +/* char: 0x3e */ + +static const SFG_StrokeVertex ch62st0[] = +{ + {14.2857f,85.7143f}, + {90.4762f,42.8571f}, + {14.2857f,0.0f} +}; + +static const SFG_StrokeStrip ch62st[] = +{ + {3,ch62st0} +}; + +static const SFG_StrokeChar ch62 = {104.762f,1,ch62st}; + +/* char: 0x3f */ + +static const SFG_StrokeVertex ch63st0[] = +{ + {23.8095f,76.1905f}, + {23.8095f,80.9524f}, + {28.5714f,90.4762f}, + {33.3333f,95.2381f}, + {42.8571f,100.0f}, + {61.9047f,100.0f}, + {71.4285f,95.2381f}, + {76.1905f,90.4762f}, + {80.9524f,80.9524f}, + {80.9524f,71.4286f}, + {76.1905f,61.9048f}, + {71.4285f,57.1429f}, + {52.3809f,47.619f}, + {52.3809f,33.3333f} +}; + +static const SFG_StrokeVertex ch63st1[] = +{ + {52.3809f,9.5238f}, + {47.619f,4.7619f}, + {52.3809f,0.0f}, + {57.1428f,4.7619f}, + {52.3809f,9.5238f} +}; + +static const SFG_StrokeStrip ch63st[] = +{ + {14,ch63st0}, + {5,ch63st1} +}; + +static const SFG_StrokeChar ch63 = {104.762f,2,ch63st}; + +/* char: 0x40 */ + +static const SFG_StrokeVertex ch64st0[] = +{ + {64.2857f,52.381f}, + {54.7619f,57.1429f}, + {45.2381f,57.1429f}, + {40.4762f,47.619f}, + {40.4762f,42.8571f}, + {45.2381f,33.3333f}, + {54.7619f,33.3333f}, + {64.2857f,38.0952f} +}; + +static const SFG_StrokeVertex ch64st1[] = +{ + {64.2857f,57.1429f}, + {64.2857f,38.0952f}, + {69.0476f,33.3333f}, + {78.5714f,33.3333f}, + {83.3334f,42.8571f}, + {83.3334f,47.619f}, + {78.5714f,61.9048f}, + {69.0476f,71.4286f}, + {54.7619f,76.1905f}, + {50.0f,76.1905f}, + {35.7143f,71.4286f}, + {26.1905f,61.9048f}, + {21.4286f,47.619f}, + {21.4286f,42.8571f}, + {26.1905f,28.5714f}, + {35.7143f,19.0476f}, + {50.0f,14.2857f}, + {54.7619f,14.2857f}, + {69.0476f,19.0476f} +}; + +static const SFG_StrokeStrip ch64st[] = +{ + {8,ch64st0}, + {19,ch64st1} +}; + +static const SFG_StrokeChar ch64 = {104.762f,2,ch64st}; + +/* char: 0x41 */ + +static const SFG_StrokeVertex ch65st0[] = +{ + {52.3809f,100.0f}, + {14.2857f,0.0f} +}; + +static const SFG_StrokeVertex ch65st1[] = +{ + {52.3809f,100.0f}, + {90.4762f,0.0f} +}; + +static const SFG_StrokeVertex ch65st2[] = +{ + {28.5714f,33.3333f}, + {76.1905f,33.3333f} +}; + +static const SFG_StrokeStrip ch65st[] = +{ + {2,ch65st0}, + {2,ch65st1}, + {2,ch65st2} +}; + +static const SFG_StrokeChar ch65 = {104.762f,3,ch65st}; + +/* char: 0x42 */ + +static const SFG_StrokeVertex ch66st0[] = +{ + {19.0476f,100.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeVertex ch66st1[] = +{ + {19.0476f,100.0f}, + {61.9047f,100.0f}, + {76.1905f,95.2381f}, + {80.9524f,90.4762f}, + {85.7143f,80.9524f}, + {85.7143f,71.4286f}, + {80.9524f,61.9048f}, + {76.1905f,57.1429f}, + {61.9047f,52.381f} +}; + +static const SFG_StrokeVertex ch66st2[] = +{ + {19.0476f,52.381f}, + {61.9047f,52.381f}, + {76.1905f,47.619f}, + {80.9524f,42.8571f}, + {85.7143f,33.3333f}, + {85.7143f,19.0476f}, + {80.9524f,9.5238f}, + {76.1905f,4.7619f}, + {61.9047f,0.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeStrip ch66st[] = +{ + {2,ch66st0}, + {9,ch66st1}, + {10,ch66st2} +}; + +static const SFG_StrokeChar ch66 = {104.762f,3,ch66st}; + +/* char: 0x43 */ + +static const SFG_StrokeVertex ch67st0[] = +{ + {88.0952f,76.1905f}, + {83.3334f,85.7143f}, + {73.8096f,95.2381f}, + {64.2857f,100.0f}, + {45.2381f,100.0f}, + {35.7143f,95.2381f}, + {26.1905f,85.7143f}, + {21.4286f,76.1905f}, + {16.6667f,61.9048f}, + {16.6667f,38.0952f}, + {21.4286f,23.8095f}, + {26.1905f,14.2857f}, + {35.7143f,4.7619f}, + {45.2381f,0.0f}, + {64.2857f,0.0f}, + {73.8096f,4.7619f}, + {83.3334f,14.2857f}, + {88.0952f,23.8095f} +}; + +static const SFG_StrokeStrip ch67st[] = +{ + {18,ch67st0} +}; + +static const SFG_StrokeChar ch67 = {104.762f,1,ch67st}; + +/* char: 0x44 */ + +static const SFG_StrokeVertex ch68st0[] = +{ + {19.0476f,100.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeVertex ch68st1[] = +{ + {19.0476f,100.0f}, + {52.3809f,100.0f}, + {66.6666f,95.2381f}, + {76.1905f,85.7143f}, + {80.9524f,76.1905f}, + {85.7143f,61.9048f}, + {85.7143f,38.0952f}, + {80.9524f,23.8095f}, + {76.1905f,14.2857f}, + {66.6666f,4.7619f}, + {52.3809f,0.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeStrip ch68st[] = +{ + {2,ch68st0}, + {12,ch68st1} +}; + +static const SFG_StrokeChar ch68 = {104.762f,2,ch68st}; + +/* char: 0x45 */ + +static const SFG_StrokeVertex ch69st0[] = +{ + {21.4286f,100.0f}, + {21.4286f,0.0f} +}; + +static const SFG_StrokeVertex ch69st1[] = +{ + {21.4286f,100.0f}, + {83.3334f,100.0f} +}; + +static const SFG_StrokeVertex ch69st2[] = +{ + {21.4286f,52.381f}, + {59.5238f,52.381f} +}; + +static const SFG_StrokeVertex ch69st3[] = +{ + {21.4286f,0.0f}, + {83.3334f,0.0f} +}; + +static const SFG_StrokeStrip ch69st[] = +{ + {2,ch69st0}, + {2,ch69st1}, + {2,ch69st2}, + {2,ch69st3} +}; + +static const SFG_StrokeChar ch69 = {104.762f,4,ch69st}; + +/* char: 0x46 */ + +static const SFG_StrokeVertex ch70st0[] = +{ + {21.4286f,100.0f}, + {21.4286f,0.0f} +}; + +static const SFG_StrokeVertex ch70st1[] = +{ + {21.4286f,100.0f}, + {83.3334f,100.0f} +}; + +static const SFG_StrokeVertex ch70st2[] = +{ + {21.4286f,52.381f}, + {59.5238f,52.381f} +}; + +static const SFG_StrokeStrip ch70st[] = +{ + {2,ch70st0}, + {2,ch70st1}, + {2,ch70st2} +}; + +static const SFG_StrokeChar ch70 = {104.762f,3,ch70st}; + +/* char: 0x47 */ + +static const SFG_StrokeVertex ch71st0[] = +{ + {88.0952f,76.1905f}, + {83.3334f,85.7143f}, + {73.8096f,95.2381f}, + {64.2857f,100.0f}, + {45.2381f,100.0f}, + {35.7143f,95.2381f}, + {26.1905f,85.7143f}, + {21.4286f,76.1905f}, + {16.6667f,61.9048f}, + {16.6667f,38.0952f}, + {21.4286f,23.8095f}, + {26.1905f,14.2857f}, + {35.7143f,4.7619f}, + {45.2381f,0.0f}, + {64.2857f,0.0f}, + {73.8096f,4.7619f}, + {83.3334f,14.2857f}, + {88.0952f,23.8095f}, + {88.0952f,38.0952f} +}; + +static const SFG_StrokeVertex ch71st1[] = +{ + {64.2857f,38.0952f}, + {88.0952f,38.0952f} +}; + +static const SFG_StrokeStrip ch71st[] = +{ + {19,ch71st0}, + {2,ch71st1} +}; + +static const SFG_StrokeChar ch71 = {104.762f,2,ch71st}; + +/* char: 0x48 */ + +static const SFG_StrokeVertex ch72st0[] = +{ + {19.0476f,100.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeVertex ch72st1[] = +{ + {85.7143f,100.0f}, + {85.7143f,0.0f} +}; + +static const SFG_StrokeVertex ch72st2[] = +{ + {19.0476f,52.381f}, + {85.7143f,52.381f} +}; + +static const SFG_StrokeStrip ch72st[] = +{ + {2,ch72st0}, + {2,ch72st1}, + {2,ch72st2} +}; + +static const SFG_StrokeChar ch72 = {104.762f,3,ch72st}; + +/* char: 0x49 */ + +static const SFG_StrokeVertex ch73st0[] = +{ + {52.381f,100.0f}, + {52.381f,0.0f} +}; + +static const SFG_StrokeStrip ch73st[] = +{ + {2,ch73st0} +}; + +static const SFG_StrokeChar ch73 = {104.762f,1,ch73st}; + +/* char: 0x4a */ + +static const SFG_StrokeVertex ch74st0[] = +{ + {76.1905f,100.0f}, + {76.1905f,23.8095f}, + {71.4286f,9.5238f}, + {66.6667f,4.7619f}, + {57.1429f,0.0f}, + {47.6191f,0.0f}, + {38.0953f,4.7619f}, + {33.3334f,9.5238f}, + {28.5715f,23.8095f}, + {28.5715f,33.3333f} +}; + +static const SFG_StrokeStrip ch74st[] = +{ + {10,ch74st0} +}; + +static const SFG_StrokeChar ch74 = {104.762f,1,ch74st}; + +/* char: 0x4b */ + +static const SFG_StrokeVertex ch75st0[] = +{ + {19.0476f,100.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeVertex ch75st1[] = +{ + {85.7143f,100.0f}, + {19.0476f,33.3333f} +}; + +static const SFG_StrokeVertex ch75st2[] = +{ + {42.8571f,57.1429f}, + {85.7143f,0.0f} +}; + +static const SFG_StrokeStrip ch75st[] = +{ + {2,ch75st0}, + {2,ch75st1}, + {2,ch75st2} +}; + +static const SFG_StrokeChar ch75 = {104.762f,3,ch75st}; + +/* char: 0x4c */ + +static const SFG_StrokeVertex ch76st0[] = +{ + {23.8095f,100.0f}, + {23.8095f,0.0f} +}; + +static const SFG_StrokeVertex ch76st1[] = +{ + {23.8095f,0.0f}, + {80.9524f,0.0f} +}; + +static const SFG_StrokeStrip ch76st[] = +{ + {2,ch76st0}, + {2,ch76st1} +}; + +static const SFG_StrokeChar ch76 = {104.762f,2,ch76st}; + +/* char: 0x4d */ + +static const SFG_StrokeVertex ch77st0[] = +{ + {14.2857f,100.0f}, + {14.2857f,0.0f} +}; + +static const SFG_StrokeVertex ch77st1[] = +{ + {14.2857f,100.0f}, + {52.3809f,0.0f} +}; + +static const SFG_StrokeVertex ch77st2[] = +{ + {90.4762f,100.0f}, + {52.3809f,0.0f} +}; + +static const SFG_StrokeVertex ch77st3[] = +{ + {90.4762f,100.0f}, + {90.4762f,0.0f} +}; + +static const SFG_StrokeStrip ch77st[] = +{ + {2,ch77st0}, + {2,ch77st1}, + {2,ch77st2}, + {2,ch77st3} +}; + +static const SFG_StrokeChar ch77 = {104.762f,4,ch77st}; + +/* char: 0x4e */ + +static const SFG_StrokeVertex ch78st0[] = +{ + {19.0476f,100.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeVertex ch78st1[] = +{ + {19.0476f,100.0f}, + {85.7143f,0.0f} +}; + +static const SFG_StrokeVertex ch78st2[] = +{ + {85.7143f,100.0f}, + {85.7143f,0.0f} +}; + +static const SFG_StrokeStrip ch78st[] = +{ + {2,ch78st0}, + {2,ch78st1}, + {2,ch78st2} +}; + +static const SFG_StrokeChar ch78 = {104.762f,3,ch78st}; + +/* char: 0x4f */ + +static const SFG_StrokeVertex ch79st0[] = +{ + {42.8571f,100.0f}, + {33.3333f,95.2381f}, + {23.8095f,85.7143f}, + {19.0476f,76.1905f}, + {14.2857f,61.9048f}, + {14.2857f,38.0952f}, + {19.0476f,23.8095f}, + {23.8095f,14.2857f}, + {33.3333f,4.7619f}, + {42.8571f,0.0f}, + {61.9047f,0.0f}, + {71.4286f,4.7619f}, + {80.9524f,14.2857f}, + {85.7143f,23.8095f}, + {90.4762f,38.0952f}, + {90.4762f,61.9048f}, + {85.7143f,76.1905f}, + {80.9524f,85.7143f}, + {71.4286f,95.2381f}, + {61.9047f,100.0f}, + {42.8571f,100.0f} +}; + +static const SFG_StrokeStrip ch79st[] = +{ + {21,ch79st0} +}; + +static const SFG_StrokeChar ch79 = {104.762f,1,ch79st}; + +/* char: 0x50 */ + +static const SFG_StrokeVertex ch80st0[] = +{ + {19.0476f,100.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeVertex ch80st1[] = +{ + {19.0476f,100.0f}, + {61.9047f,100.0f}, + {76.1905f,95.2381f}, + {80.9524f,90.4762f}, + {85.7143f,80.9524f}, + {85.7143f,66.6667f}, + {80.9524f,57.1429f}, + {76.1905f,52.381f}, + {61.9047f,47.619f}, + {19.0476f,47.619f} +}; + +static const SFG_StrokeStrip ch80st[] = +{ + {2,ch80st0}, + {10,ch80st1} +}; + +static const SFG_StrokeChar ch80 = {104.762f,2,ch80st}; + +/* char: 0x51 */ + +static const SFG_StrokeVertex ch81st0[] = +{ + {42.8571f,100.0f}, + {33.3333f,95.2381f}, + {23.8095f,85.7143f}, + {19.0476f,76.1905f}, + {14.2857f,61.9048f}, + {14.2857f,38.0952f}, + {19.0476f,23.8095f}, + {23.8095f,14.2857f}, + {33.3333f,4.7619f}, + {42.8571f,0.0f}, + {61.9047f,0.0f}, + {71.4286f,4.7619f}, + {80.9524f,14.2857f}, + {85.7143f,23.8095f}, + {90.4762f,38.0952f}, + {90.4762f,61.9048f}, + {85.7143f,76.1905f}, + {80.9524f,85.7143f}, + {71.4286f,95.2381f}, + {61.9047f,100.0f}, + {42.8571f,100.0f} +}; + +static const SFG_StrokeVertex ch81st1[] = +{ + {57.1428f,19.0476f}, + {85.7143f,-9.5238f} +}; + +static const SFG_StrokeStrip ch81st[] = +{ + {21,ch81st0}, + {2,ch81st1} +}; + +static const SFG_StrokeChar ch81 = {104.762f,2,ch81st}; + +/* char: 0x52 */ + +static const SFG_StrokeVertex ch82st0[] = +{ + {19.0476f,100.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeVertex ch82st1[] = +{ + {19.0476f,100.0f}, + {61.9047f,100.0f}, + {76.1905f,95.2381f}, + {80.9524f,90.4762f}, + {85.7143f,80.9524f}, + {85.7143f,71.4286f}, + {80.9524f,61.9048f}, + {76.1905f,57.1429f}, + {61.9047f,52.381f}, + {19.0476f,52.381f} +}; + +static const SFG_StrokeVertex ch82st2[] = +{ + {52.3809f,52.381f}, + {85.7143f,0.0f} +}; + +static const SFG_StrokeStrip ch82st[] = +{ + {2,ch82st0}, + {10,ch82st1}, + {2,ch82st2} +}; + +static const SFG_StrokeChar ch82 = {104.762f,3,ch82st}; + +/* char: 0x53 */ + +static const SFG_StrokeVertex ch83st0[] = +{ + {85.7143f,85.7143f}, + {76.1905f,95.2381f}, + {61.9047f,100.0f}, + {42.8571f,100.0f}, + {28.5714f,95.2381f}, + {19.0476f,85.7143f}, + {19.0476f,76.1905f}, + {23.8095f,66.6667f}, + {28.5714f,61.9048f}, + {38.0952f,57.1429f}, + {66.6666f,47.619f}, + {76.1905f,42.8571f}, + {80.9524f,38.0952f}, + {85.7143f,28.5714f}, + {85.7143f,14.2857f}, + {76.1905f,4.7619f}, + {61.9047f,0.0f}, + {42.8571f,0.0f}, + {28.5714f,4.7619f}, + {19.0476f,14.2857f} +}; + +static const SFG_StrokeStrip ch83st[] = +{ + {20,ch83st0} +}; + +static const SFG_StrokeChar ch83 = {104.762f,1,ch83st}; + +/* char: 0x54 */ + +static const SFG_StrokeVertex ch84st0[] = +{ + {52.3809f,100.0f}, + {52.3809f,0.0f} +}; + +static const SFG_StrokeVertex ch84st1[] = +{ + {19.0476f,100.0f}, + {85.7143f,100.0f} +}; + +static const SFG_StrokeStrip ch84st[] = +{ + {2,ch84st0}, + {2,ch84st1} +}; + +static const SFG_StrokeChar ch84 = {104.762f,2,ch84st}; + +/* char: 0x55 */ + +static const SFG_StrokeVertex ch85st0[] = +{ + {19.0476f,100.0f}, + {19.0476f,28.5714f}, + {23.8095f,14.2857f}, + {33.3333f,4.7619f}, + {47.619f,0.0f}, + {57.1428f,0.0f}, + {71.4286f,4.7619f}, + {80.9524f,14.2857f}, + {85.7143f,28.5714f}, + {85.7143f,100.0f} +}; + +static const SFG_StrokeStrip ch85st[] = +{ + {10,ch85st0} +}; + +static const SFG_StrokeChar ch85 = {104.762f,1,ch85st}; + +/* char: 0x56 */ + +static const SFG_StrokeVertex ch86st0[] = +{ + {14.2857f,100.0f}, + {52.3809f,0.0f} +}; + +static const SFG_StrokeVertex ch86st1[] = +{ + {90.4762f,100.0f}, + {52.3809f,0.0f} +}; + +static const SFG_StrokeStrip ch86st[] = +{ + {2,ch86st0}, + {2,ch86st1} +}; + +static const SFG_StrokeChar ch86 = {104.762f,2,ch86st}; + +/* char: 0x57 */ + +static const SFG_StrokeVertex ch87st0[] = +{ + {4.7619f,100.0f}, + {28.5714f,0.0f} +}; + +static const SFG_StrokeVertex ch87st1[] = +{ + {52.3809f,100.0f}, + {28.5714f,0.0f} +}; + +static const SFG_StrokeVertex ch87st2[] = +{ + {52.3809f,100.0f}, + {76.1905f,0.0f} +}; + +static const SFG_StrokeVertex ch87st3[] = +{ + {100.0f,100.0f}, + {76.1905f,0.0f} +}; + +static const SFG_StrokeStrip ch87st[] = +{ + {2,ch87st0}, + {2,ch87st1}, + {2,ch87st2}, + {2,ch87st3} +}; + +static const SFG_StrokeChar ch87 = {104.762f,4,ch87st}; + +/* char: 0x58 */ + +static const SFG_StrokeVertex ch88st0[] = +{ + {19.0476f,100.0f}, + {85.7143f,0.0f} +}; + +static const SFG_StrokeVertex ch88st1[] = +{ + {85.7143f,100.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeStrip ch88st[] = +{ + {2,ch88st0}, + {2,ch88st1} +}; + +static const SFG_StrokeChar ch88 = {104.762f,2,ch88st}; + +/* char: 0x59 */ + +static const SFG_StrokeVertex ch89st0[] = +{ + {14.2857f,100.0f}, + {52.3809f,52.381f}, + {52.3809f,0.0f} +}; + +static const SFG_StrokeVertex ch89st1[] = +{ + {90.4762f,100.0f}, + {52.3809f,52.381f} +}; + +static const SFG_StrokeStrip ch89st[] = +{ + {3,ch89st0}, + {2,ch89st1} +}; + +static const SFG_StrokeChar ch89 = {104.762f,2,ch89st}; + +/* char: 0x5a */ + +static const SFG_StrokeVertex ch90st0[] = +{ + {85.7143f,100.0f}, + {19.0476f,0.0f} +}; + +static const SFG_StrokeVertex ch90st1[] = +{ + {19.0476f,100.0f}, + {85.7143f,100.0f} +}; + +static const SFG_StrokeVertex ch90st2[] = +{ + {19.0476f,0.0f}, + {85.7143f,0.0f} +}; + +static const SFG_StrokeStrip ch90st[] = +{ + {2,ch90st0}, + {2,ch90st1}, + {2,ch90st2} +}; + +static const SFG_StrokeChar ch90 = {104.762f,3,ch90st}; + +/* char: 0x5b */ + +static const SFG_StrokeVertex ch91st0[] = +{ + {35.7143f,119.048f}, + {35.7143f,-33.3333f} +}; + +static const SFG_StrokeVertex ch91st1[] = +{ + {40.4762f,119.048f}, + {40.4762f,-33.3333f} +}; + +static const SFG_StrokeVertex ch91st2[] = +{ + {35.7143f,119.048f}, + {69.0476f,119.048f} +}; + +static const SFG_StrokeVertex ch91st3[] = +{ + {35.7143f,-33.3333f}, + {69.0476f,-33.3333f} +}; + +static const SFG_StrokeStrip ch91st[] = +{ + {2,ch91st0}, + {2,ch91st1}, + {2,ch91st2}, + {2,ch91st3} +}; + +static const SFG_StrokeChar ch91 = {104.762f,4,ch91st}; + +/* char: 0x5c */ + +static const SFG_StrokeVertex ch92st0[] = +{ + {19.0476f,100.0f}, + {85.7143f,-14.2857f} +}; + +static const SFG_StrokeStrip ch92st[] = +{ + {2,ch92st0} +}; + +static const SFG_StrokeChar ch92 = {104.762f,1,ch92st}; + +/* char: 0x5d */ + +static const SFG_StrokeVertex ch93st0[] = +{ + {64.2857f,119.048f}, + {64.2857f,-33.3333f} +}; + +static const SFG_StrokeVertex ch93st1[] = +{ + {69.0476f,119.048f}, + {69.0476f,-33.3333f} +}; + +static const SFG_StrokeVertex ch93st2[] = +{ + {35.7143f,119.048f}, + {69.0476f,119.048f} +}; + +static const SFG_StrokeVertex ch93st3[] = +{ + {35.7143f,-33.3333f}, + {69.0476f,-33.3333f} +}; + +static const SFG_StrokeStrip ch93st[] = +{ + {2,ch93st0}, + {2,ch93st1}, + {2,ch93st2}, + {2,ch93st3} +}; + +static const SFG_StrokeChar ch93 = {104.762f,4,ch93st}; + +/* char: 0x5e */ + +static const SFG_StrokeVertex ch94st0[] = +{ + {52.3809f,109.524f}, + {14.2857f,42.8571f} +}; + +static const SFG_StrokeVertex ch94st1[] = +{ + {52.3809f,109.524f}, + {90.4762f,42.8571f} +}; + +static const SFG_StrokeStrip ch94st[] = +{ + {2,ch94st0}, + {2,ch94st1} +}; + +static const SFG_StrokeChar ch94 = {104.762f,2,ch94st}; + +/* char: 0x5f */ + +static const SFG_StrokeVertex ch95st0[] = +{ + {0,-33.3333f}, + {104.762f,-33.3333f}, + {104.762f,-28.5714f}, + {0,-28.5714f}, + {0,-33.3333f} +}; + +static const SFG_StrokeStrip ch95st[] = +{ + {5,ch95st0} +}; + +static const SFG_StrokeChar ch95 = {104.762f,1,ch95st}; + +/* char: 0x60 */ + +static const SFG_StrokeVertex ch96st0[] = +{ + {42.8572f,100.0f}, + {66.6667f,71.4286f} +}; + +static const SFG_StrokeVertex ch96st1[] = +{ + {42.8572f,100.0f}, + {38.0953f,95.2381f}, + {66.6667f,71.4286f} +}; + +static const SFG_StrokeStrip ch96st[] = +{ + {2,ch96st0}, + {3,ch96st1} +}; + +static const SFG_StrokeChar ch96 = {104.762f,2,ch96st}; + +/* char: 0x61 */ + +static const SFG_StrokeVertex ch97st0[] = +{ + {80.9524f,66.6667f}, + {80.9524f,0.0f} +}; + +static const SFG_StrokeVertex ch97st1[] = +{ + {80.9524f,52.381f}, + {71.4285f,61.9048f}, + {61.9047f,66.6667f}, + {47.619f,66.6667f}, + {38.0952f,61.9048f}, + {28.5714f,52.381f}, + {23.8095f,38.0952f}, + {23.8095f,28.5714f}, + {28.5714f,14.2857f}, + {38.0952f,4.7619f}, + {47.619f,0.0f}, + {61.9047f,0.0f}, + {71.4285f,4.7619f}, + {80.9524f,14.2857f} +}; + +static const SFG_StrokeStrip ch97st[] = +{ + {2,ch97st0}, + {14,ch97st1} +}; + +static const SFG_StrokeChar ch97 = {104.762f,2,ch97st}; + +/* char: 0x62 */ + +static const SFG_StrokeVertex ch98st0[] = +{ + {23.8095f,100.0f}, + {23.8095f,0.0f} +}; + +static const SFG_StrokeVertex ch98st1[] = +{ + {23.8095f,52.381f}, + {33.3333f,61.9048f}, + {42.8571f,66.6667f}, + {57.1428f,66.6667f}, + {66.6666f,61.9048f}, + {76.1905f,52.381f}, + {80.9524f,38.0952f}, + {80.9524f,28.5714f}, + {76.1905f,14.2857f}, + {66.6666f,4.7619f}, + {57.1428f,0.0f}, + {42.8571f,0.0f}, + {33.3333f,4.7619f}, + {23.8095f,14.2857f} +}; + +static const SFG_StrokeStrip ch98st[] = +{ + {2,ch98st0}, + {14,ch98st1} +}; + +static const SFG_StrokeChar ch98 = {104.762f,2,ch98st}; + +/* char: 0x63 */ + +static const SFG_StrokeVertex ch99st0[] = +{ + {80.9524f,52.381f}, + {71.4285f,61.9048f}, + {61.9047f,66.6667f}, + {47.619f,66.6667f}, + {38.0952f,61.9048f}, + {28.5714f,52.381f}, + {23.8095f,38.0952f}, + {23.8095f,28.5714f}, + {28.5714f,14.2857f}, + {38.0952f,4.7619f}, + {47.619f,0.0f}, + {61.9047f,0.0f}, + {71.4285f,4.7619f}, + {80.9524f,14.2857f} +}; + +static const SFG_StrokeStrip ch99st[] = +{ + {14,ch99st0} +}; + +static const SFG_StrokeChar ch99 = {104.762f,1,ch99st}; + +/* char: 0x64 */ + +static const SFG_StrokeVertex ch100st0[] = +{ + {80.9524f,100.0f}, + {80.9524f,0.0f} +}; + +static const SFG_StrokeVertex ch100st1[] = +{ + {80.9524f,52.381f}, + {71.4285f,61.9048f}, + {61.9047f,66.6667f}, + {47.619f,66.6667f}, + {38.0952f,61.9048f}, + {28.5714f,52.381f}, + {23.8095f,38.0952f}, + {23.8095f,28.5714f}, + {28.5714f,14.2857f}, + {38.0952f,4.7619f}, + {47.619f,0.0f}, + {61.9047f,0.0f}, + {71.4285f,4.7619f}, + {80.9524f,14.2857f} +}; + +static const SFG_StrokeStrip ch100st[] = +{ + {2,ch100st0}, + {14,ch100st1} +}; + +static const SFG_StrokeChar ch100 = {104.762f,2,ch100st}; + +/* char: 0x65 */ + +static const SFG_StrokeVertex ch101st0[] = +{ + {23.8095f,38.0952f}, + {80.9524f,38.0952f}, + {80.9524f,47.619f}, + {76.1905f,57.1429f}, + {71.4285f,61.9048f}, + {61.9047f,66.6667f}, + {47.619f,66.6667f}, + {38.0952f,61.9048f}, + {28.5714f,52.381f}, + {23.8095f,38.0952f}, + {23.8095f,28.5714f}, + {28.5714f,14.2857f}, + {38.0952f,4.7619f}, + {47.619f,0.0f}, + {61.9047f,0.0f}, + {71.4285f,4.7619f}, + {80.9524f,14.2857f} +}; + +static const SFG_StrokeStrip ch101st[] = +{ + {17,ch101st0} +}; + +static const SFG_StrokeChar ch101 = {104.762f,1,ch101st}; + +/* char: 0x66 */ + +static const SFG_StrokeVertex ch102st0[] = +{ + {71.4286f,100.0f}, + {61.9048f,100.0f}, + {52.381f,95.2381f}, + {47.6191f,80.9524f}, + {47.6191f,0.0f} +}; + +static const SFG_StrokeVertex ch102st1[] = +{ + {33.3334f,66.6667f}, + {66.6667f,66.6667f} +}; + +static const SFG_StrokeStrip ch102st[] = +{ + {5,ch102st0}, + {2,ch102st1} +}; + +static const SFG_StrokeChar ch102 = {104.762f,2,ch102st}; + +/* char: 0x67 */ + +static const SFG_StrokeVertex ch103st0[] = +{ + {80.9524f,66.6667f}, + {80.9524f,-9.5238f}, + {76.1905f,-23.8095f}, + {71.4285f,-28.5714f}, + {61.9047f,-33.3333f}, + {47.619f,-33.3333f}, + {38.0952f,-28.5714f} +}; + +static const SFG_StrokeVertex ch103st1[] = +{ + {80.9524f,52.381f}, + {71.4285f,61.9048f}, + {61.9047f,66.6667f}, + {47.619f,66.6667f}, + {38.0952f,61.9048f}, + {28.5714f,52.381f}, + {23.8095f,38.0952f}, + {23.8095f,28.5714f}, + {28.5714f,14.2857f}, + {38.0952f,4.7619f}, + {47.619f,0.0f}, + {61.9047f,0.0f}, + {71.4285f,4.7619f}, + {80.9524f,14.2857f} +}; + +static const SFG_StrokeStrip ch103st[] = +{ + {7,ch103st0}, + {14,ch103st1} +}; + +static const SFG_StrokeChar ch103 = {104.762f,2,ch103st}; + +/* char: 0x68 */ + +static const SFG_StrokeVertex ch104st0[] = +{ + {26.1905f,100.0f}, + {26.1905f,0.0f} +}; + +static const SFG_StrokeVertex ch104st1[] = +{ + {26.1905f,47.619f}, + {40.4762f,61.9048f}, + {50.0f,66.6667f}, + {64.2857f,66.6667f}, + {73.8095f,61.9048f}, + {78.5715f,47.619f}, + {78.5715f,0.0f} +}; + +static const SFG_StrokeStrip ch104st[] = +{ + {2,ch104st0}, + {7,ch104st1} +}; + +static const SFG_StrokeChar ch104 = {104.762f,2,ch104st}; + +/* char: 0x69 */ + +static const SFG_StrokeVertex ch105st0[] = +{ + {47.6191f,100.0f}, + {52.381f,95.2381f}, + {57.1429f,100.0f}, + {52.381f,104.762f}, + {47.6191f,100.0f} +}; + +static const SFG_StrokeVertex ch105st1[] = +{ + {52.381f,66.6667f}, + {52.381f,0.0f} +}; + +static const SFG_StrokeStrip ch105st[] = +{ + {5,ch105st0}, + {2,ch105st1} +}; + +static const SFG_StrokeChar ch105 = {104.762f,2,ch105st}; + +/* char: 0x6a */ + +static const SFG_StrokeVertex ch106st0[] = +{ + {57.1429f,100.0f}, + {61.9048f,95.2381f}, + {66.6667f,100.0f}, + {61.9048f,104.762f}, + {57.1429f,100.0f} +}; + +static const SFG_StrokeVertex ch106st1[] = +{ + {61.9048f,66.6667f}, + {61.9048f,-14.2857f}, + {57.1429f,-28.5714f}, + {47.6191f,-33.3333f}, + {38.0953f,-33.3333f} +}; + +static const SFG_StrokeStrip ch106st[] = +{ + {5,ch106st0}, + {5,ch106st1} +}; + +static const SFG_StrokeChar ch106 = {104.762f,2,ch106st}; + +/* char: 0x6b */ + +static const SFG_StrokeVertex ch107st0[] = +{ + {26.1905f,100.0f}, + {26.1905f,0.0f} +}; + +static const SFG_StrokeVertex ch107st1[] = +{ + {73.8095f,66.6667f}, + {26.1905f,19.0476f} +}; + +static const SFG_StrokeVertex ch107st2[] = +{ + {45.2381f,38.0952f}, + {78.5715f,0.0f} +}; + +static const SFG_StrokeStrip ch107st[] = +{ + {2,ch107st0}, + {2,ch107st1}, + {2,ch107st2} +}; + +static const SFG_StrokeChar ch107 = {104.762f,3,ch107st}; + +/* char: 0x6c */ + +static const SFG_StrokeVertex ch108st0[] = +{ + {52.381f,100.0f}, + {52.381f,0.0f} +}; + +static const SFG_StrokeStrip ch108st[] = +{ + {2,ch108st0} +}; + +static const SFG_StrokeChar ch108 = {104.762f,1,ch108st}; + +/* char: 0x6d */ + +static const SFG_StrokeVertex ch109st0[] = +{ + {0,66.6667f}, + {0,0.0f} +}; + +static const SFG_StrokeVertex ch109st1[] = +{ + {0,47.619f}, + {14.2857f,61.9048f}, + {23.8095f,66.6667f}, + {38.0952f,66.6667f}, + {47.619f,61.9048f}, + {52.381f,47.619f}, + {52.381f,0.0f} +}; + +static const SFG_StrokeVertex ch109st2[] = +{ + {52.381f,47.619f}, + {66.6667f,61.9048f}, + {76.1905f,66.6667f}, + {90.4762f,66.6667f}, + {100.0f,61.9048f}, + {104.762f,47.619f}, + {104.762f,0.0f} +}; + +static const SFG_StrokeStrip ch109st[] = +{ + {2,ch109st0}, + {7,ch109st1}, + {7,ch109st2} +}; + +static const SFG_StrokeChar ch109 = {104.762f,3,ch109st}; + +/* char: 0x6e */ + +static const SFG_StrokeVertex ch110st0[] = +{ + {26.1905f,66.6667f}, + {26.1905f,0.0f} +}; + +static const SFG_StrokeVertex ch110st1[] = +{ + {26.1905f,47.619f}, + {40.4762f,61.9048f}, + {50.0f,66.6667f}, + {64.2857f,66.6667f}, + {73.8095f,61.9048f}, + {78.5715f,47.619f}, + {78.5715f,0.0f} +}; + +static const SFG_StrokeStrip ch110st[] = +{ + {2,ch110st0}, + {7,ch110st1} +}; + +static const SFG_StrokeChar ch110 = {104.762f,2,ch110st}; + +/* char: 0x6f */ + +static const SFG_StrokeVertex ch111st0[] = +{ + {45.2381f,66.6667f}, + {35.7143f,61.9048f}, + {26.1905f,52.381f}, + {21.4286f,38.0952f}, + {21.4286f,28.5714f}, + {26.1905f,14.2857f}, + {35.7143f,4.7619f}, + {45.2381f,0.0f}, + {59.5238f,0.0f}, + {69.0476f,4.7619f}, + {78.5714f,14.2857f}, + {83.3334f,28.5714f}, + {83.3334f,38.0952f}, + {78.5714f,52.381f}, + {69.0476f,61.9048f}, + {59.5238f,66.6667f}, + {45.2381f,66.6667f} +}; + +static const SFG_StrokeStrip ch111st[] = +{ + {17,ch111st0} +}; + +static const SFG_StrokeChar ch111 = {104.762f,1,ch111st}; + +/* char: 0x70 */ + +static const SFG_StrokeVertex ch112st0[] = +{ + {23.8095f,66.6667f}, + {23.8095f,-33.3333f} +}; + +static const SFG_StrokeVertex ch112st1[] = +{ + {23.8095f,52.381f}, + {33.3333f,61.9048f}, + {42.8571f,66.6667f}, + {57.1428f,66.6667f}, + {66.6666f,61.9048f}, + {76.1905f,52.381f}, + {80.9524f,38.0952f}, + {80.9524f,28.5714f}, + {76.1905f,14.2857f}, + {66.6666f,4.7619f}, + {57.1428f,0.0f}, + {42.8571f,0.0f}, + {33.3333f,4.7619f}, + {23.8095f,14.2857f} +}; + +static const SFG_StrokeStrip ch112st[] = +{ + {2,ch112st0}, + {14,ch112st1} +}; + +static const SFG_StrokeChar ch112 = {104.762f,2,ch112st}; + +/* char: 0x71 */ + +static const SFG_StrokeVertex ch113st0[] = +{ + {80.9524f,66.6667f}, + {80.9524f,-33.3333f} +}; + +static const SFG_StrokeVertex ch113st1[] = +{ + {80.9524f,52.381f}, + {71.4285f,61.9048f}, + {61.9047f,66.6667f}, + {47.619f,66.6667f}, + {38.0952f,61.9048f}, + {28.5714f,52.381f}, + {23.8095f,38.0952f}, + {23.8095f,28.5714f}, + {28.5714f,14.2857f}, + {38.0952f,4.7619f}, + {47.619f,0.0f}, + {61.9047f,0.0f}, + {71.4285f,4.7619f}, + {80.9524f,14.2857f} +}; + +static const SFG_StrokeStrip ch113st[] = +{ + {2,ch113st0}, + {14,ch113st1} +}; + +static const SFG_StrokeChar ch113 = {104.762f,2,ch113st}; + +/* char: 0x72 */ + +static const SFG_StrokeVertex ch114st0[] = +{ + {33.3334f,66.6667f}, + {33.3334f,0.0f} +}; + +static const SFG_StrokeVertex ch114st1[] = +{ + {33.3334f,38.0952f}, + {38.0953f,52.381f}, + {47.6191f,61.9048f}, + {57.1429f,66.6667f}, + {71.4286f,66.6667f} +}; + +static const SFG_StrokeStrip ch114st[] = +{ + {2,ch114st0}, + {5,ch114st1} +}; + +static const SFG_StrokeChar ch114 = {104.762f,2,ch114st}; + +/* char: 0x73 */ + +static const SFG_StrokeVertex ch115st0[] = +{ + {78.5715f,52.381f}, + {73.8095f,61.9048f}, + {59.5238f,66.6667f}, + {45.2381f,66.6667f}, + {30.9524f,61.9048f}, + {26.1905f,52.381f}, + {30.9524f,42.8571f}, + {40.4762f,38.0952f}, + {64.2857f,33.3333f}, + {73.8095f,28.5714f}, + {78.5715f,19.0476f}, + {78.5715f,14.2857f}, + {73.8095f,4.7619f}, + {59.5238f,0.0f}, + {45.2381f,0.0f}, + {30.9524f,4.7619f}, + {26.1905f,14.2857f} +}; + +static const SFG_StrokeStrip ch115st[] = +{ + {17,ch115st0} +}; + +static const SFG_StrokeChar ch115 = {104.762f,1,ch115st}; + +/* char: 0x74 */ + +static const SFG_StrokeVertex ch116st0[] = +{ + {47.6191f,100.0f}, + {47.6191f,19.0476f}, + {52.381f,4.7619f}, + {61.9048f,0.0f}, + {71.4286f,0.0f} +}; + +static const SFG_StrokeVertex ch116st1[] = +{ + {33.3334f,66.6667f}, + {66.6667f,66.6667f} +}; + +static const SFG_StrokeStrip ch116st[] = +{ + {5,ch116st0}, + {2,ch116st1} +}; + +static const SFG_StrokeChar ch116 = {104.762f,2,ch116st}; + +/* char: 0x75 */ + +static const SFG_StrokeVertex ch117st0[] = +{ + {26.1905f,66.6667f}, + {26.1905f,19.0476f}, + {30.9524f,4.7619f}, + {40.4762f,0.0f}, + {54.7619f,0.0f}, + {64.2857f,4.7619f}, + {78.5715f,19.0476f} +}; + +static const SFG_StrokeVertex ch117st1[] = +{ + {78.5715f,66.6667f}, + {78.5715f,0.0f} +}; + +static const SFG_StrokeStrip ch117st[] = +{ + {7,ch117st0}, + {2,ch117st1} +}; + +static const SFG_StrokeChar ch117 = {104.762f,2,ch117st}; + +/* char: 0x76 */ + +static const SFG_StrokeVertex ch118st0[] = +{ + {23.8095f,66.6667f}, + {52.3809f,0.0f} +}; + +static const SFG_StrokeVertex ch118st1[] = +{ + {80.9524f,66.6667f}, + {52.3809f,0.0f} +}; + +static const SFG_StrokeStrip ch118st[] = +{ + {2,ch118st0}, + {2,ch118st1} +}; + +static const SFG_StrokeChar ch118 = {104.762f,2,ch118st}; + +/* char: 0x77 */ + +static const SFG_StrokeVertex ch119st0[] = +{ + {14.2857f,66.6667f}, + {33.3333f,0.0f} +}; + +static const SFG_StrokeVertex ch119st1[] = +{ + {52.3809f,66.6667f}, + {33.3333f,0.0f} +}; + +static const SFG_StrokeVertex ch119st2[] = +{ + {52.3809f,66.6667f}, + {71.4286f,0.0f} +}; + +static const SFG_StrokeVertex ch119st3[] = +{ + {90.4762f,66.6667f}, + {71.4286f,0.0f} +}; + +static const SFG_StrokeStrip ch119st[] = +{ + {2,ch119st0}, + {2,ch119st1}, + {2,ch119st2}, + {2,ch119st3} +}; + +static const SFG_StrokeChar ch119 = {104.762f,4,ch119st}; + +/* char: 0x78 */ + +static const SFG_StrokeVertex ch120st0[] = +{ + {26.1905f,66.6667f}, + {78.5715f,0.0f} +}; + +static const SFG_StrokeVertex ch120st1[] = +{ + {78.5715f,66.6667f}, + {26.1905f,0.0f} +}; + +static const SFG_StrokeStrip ch120st[] = +{ + {2,ch120st0}, + {2,ch120st1} +}; + +static const SFG_StrokeChar ch120 = {104.762f,2,ch120st}; + +/* char: 0x79 */ + +static const SFG_StrokeVertex ch121st0[] = +{ + {26.1905f,66.6667f}, + {54.7619f,0.0f} +}; + +static const SFG_StrokeVertex ch121st1[] = +{ + {83.3334f,66.6667f}, + {54.7619f,0.0f}, + {45.2381f,-19.0476f}, + {35.7143f,-28.5714f}, + {26.1905f,-33.3333f}, + {21.4286f,-33.3333f} +}; + +static const SFG_StrokeStrip ch121st[] = +{ + {2,ch121st0}, + {6,ch121st1} +}; + +static const SFG_StrokeChar ch121 = {104.762f,2,ch121st}; + +/* char: 0x7a */ + +static const SFG_StrokeVertex ch122st0[] = +{ + {78.5715f,66.6667f}, + {26.1905f,0.0f} +}; + +static const SFG_StrokeVertex ch122st1[] = +{ + {26.1905f,66.6667f}, + {78.5715f,66.6667f} +}; + +static const SFG_StrokeVertex ch122st2[] = +{ + {26.1905f,0.0f}, + {78.5715f,0.0f} +}; + +static const SFG_StrokeStrip ch122st[] = +{ + {2,ch122st0}, + {2,ch122st1}, + {2,ch122st2} +}; + +static const SFG_StrokeChar ch122 = {104.762f,3,ch122st}; + +/* char: 0x7b */ + +static const SFG_StrokeVertex ch123st0[] = +{ + {64.2857f,119.048f}, + {54.7619f,114.286f}, + {50.0f,109.524f}, + {45.2381f,100.0f}, + {45.2381f,90.4762f}, + {50.0f,80.9524f}, + {54.7619f,76.1905f}, + {59.5238f,66.6667f}, + {59.5238f,57.1429f}, + {50.0f,47.619f} +}; + +static const SFG_StrokeVertex ch123st1[] = +{ + {54.7619f,114.286f}, + {50.0f,104.762f}, + {50.0f,95.2381f}, + {54.7619f,85.7143f}, + {59.5238f,80.9524f}, + {64.2857f,71.4286f}, + {64.2857f,61.9048f}, + {59.5238f,52.381f}, + {40.4762f,42.8571f}, + {59.5238f,33.3333f}, + {64.2857f,23.8095f}, + {64.2857f,14.2857f}, + {59.5238f,4.7619f}, + {54.7619f,0.0f}, + {50.0f,-9.5238f}, + {50.0f,-19.0476f}, + {54.7619f,-28.5714f} +}; + +static const SFG_StrokeVertex ch123st2[] = +{ + {50.0f,38.0952f}, + {59.5238f,28.5714f}, + {59.5238f,19.0476f}, + {54.7619f,9.5238f}, + {50.0f,4.7619f}, + {45.2381f,-4.7619f}, + {45.2381f,-14.2857f}, + {50.0f,-23.8095f}, + {54.7619f,-28.5714f}, + {64.2857f,-33.3333f} +}; + +static const SFG_StrokeStrip ch123st[] = +{ + {10,ch123st0}, + {17,ch123st1}, + {10,ch123st2} +}; + +static const SFG_StrokeChar ch123 = {104.762f,3,ch123st}; + +/* char: 0x7c */ + +static const SFG_StrokeVertex ch124st0[] = +{ + {52.381f,119.048f}, + {52.381f,-33.3333f} +}; + +static const SFG_StrokeStrip ch124st[] = +{ + {2,ch124st0} +}; + +static const SFG_StrokeChar ch124 = {104.762f,1,ch124st}; + +/* char: 0x7d */ + +static const SFG_StrokeVertex ch125st0[] = +{ + {40.4762f,119.048f}, + {50.0f,114.286f}, + {54.7619f,109.524f}, + {59.5238f,100.0f}, + {59.5238f,90.4762f}, + {54.7619f,80.9524f}, + {50.0f,76.1905f}, + {45.2381f,66.6667f}, + {45.2381f,57.1429f}, + {54.7619f,47.619f} +}; + +static const SFG_StrokeVertex ch125st1[] = +{ + {50.0f,114.286f}, + {54.7619f,104.762f}, + {54.7619f,95.2381f}, + {50.0f,85.7143f}, + {45.2381f,80.9524f}, + {40.4762f,71.4286f}, + {40.4762f,61.9048f}, + {45.2381f,52.381f}, + {64.2857f,42.8571f}, + {45.2381f,33.3333f}, + {40.4762f,23.8095f}, + {40.4762f,14.2857f}, + {45.2381f,4.7619f}, + {50.0f,0.0f}, + {54.7619f,-9.5238f}, + {54.7619f,-19.0476f}, + {50.0f,-28.5714f} +}; + +static const SFG_StrokeVertex ch125st2[] = +{ + {54.7619f,38.0952f}, + {45.2381f,28.5714f}, + {45.2381f,19.0476f}, + {50.0f,9.5238f}, + {54.7619f,4.7619f}, + {59.5238f,-4.7619f}, + {59.5238f,-14.2857f}, + {54.7619f,-23.8095f}, + {50.0f,-28.5714f}, + {40.4762f,-33.3333f} +}; + +static const SFG_StrokeStrip ch125st[] = +{ + {10,ch125st0}, + {17,ch125st1}, + {10,ch125st2} +}; + +static const SFG_StrokeChar ch125 = {104.762f,3,ch125st}; + +/* char: 0x7e */ + +static const SFG_StrokeVertex ch126st0[] = +{ + {9.5238f,28.5714f}, + {9.5238f,38.0952f}, + {14.2857f,52.381f}, + {23.8095f,57.1429f}, + {33.3333f,57.1429f}, + {42.8571f,52.381f}, + {61.9048f,38.0952f}, + {71.4286f,33.3333f}, + {80.9524f,33.3333f}, + {90.4762f,38.0952f}, + {95.2381f,47.619f} +}; + +static const SFG_StrokeVertex ch126st1[] = +{ + {9.5238f,38.0952f}, + {14.2857f,47.619f}, + {23.8095f,52.381f}, + {33.3333f,52.381f}, + {42.8571f,47.619f}, + {61.9048f,33.3333f}, + {71.4286f,28.5714f}, + {80.9524f,28.5714f}, + {90.4762f,33.3333f}, + {95.2381f,47.619f}, + {95.2381f,57.1429f} +}; + +static const SFG_StrokeStrip ch126st[] = +{ + {11,ch126st0}, + {11,ch126st1} +}; + +static const SFG_StrokeChar ch126 = {104.762f,2,ch126st}; + +/* char: 0x7f */ + +static const SFG_StrokeVertex ch127st0[] = +{ + {71.4286f,100.0f}, + {33.3333f,-33.3333f} +}; + +static const SFG_StrokeVertex ch127st1[] = +{ + {47.619f,66.6667f}, + {33.3333f,61.9048f}, + {23.8095f,52.381f}, + {19.0476f,38.0952f}, + {19.0476f,23.8095f}, + {23.8095f,14.2857f}, + {33.3333f,4.7619f}, + {47.619f,0.0f}, + {57.1428f,0.0f}, + {71.4286f,4.7619f}, + {80.9524f,14.2857f}, + {85.7143f,28.5714f}, + {85.7143f,42.8571f}, + {80.9524f,52.381f}, + {71.4286f,61.9048f}, + {57.1428f,66.6667f}, + {47.619f,66.6667f} +}; + +static const SFG_StrokeStrip ch127st[] = +{ + {2,ch127st0}, + {17,ch127st1} +}; + +static const SFG_StrokeChar ch127 = {104.762f,2,ch127st}; + +static const SFG_StrokeChar *chars[] = +{ + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + &ch32, &ch33, &ch34, &ch35, &ch36, &ch37, &ch38, &ch39, + &ch40, &ch41, &ch42, &ch43, &ch44, &ch45, &ch46, &ch47, + &ch48, &ch49, &ch50, &ch51, &ch52, &ch53, &ch54, &ch55, + &ch56, &ch57, &ch58, &ch59, &ch60, &ch61, &ch62, &ch63, + &ch64, &ch65, &ch66, &ch67, &ch68, &ch69, &ch70, &ch71, + &ch72, &ch73, &ch74, &ch75, &ch76, &ch77, &ch78, &ch79, + &ch80, &ch81, &ch82, &ch83, &ch84, &ch85, &ch86, &ch87, + &ch88, &ch89, &ch90, &ch91, &ch92, &ch93, &ch94, &ch95, + &ch96, &ch97, &ch98, &ch99, &ch100, &ch101, &ch102, &ch103, + &ch104, &ch105, &ch106, &ch107, &ch108, &ch109, &ch110, &ch111, + &ch112, &ch113, &ch114, &ch115, &ch116, &ch117, &ch118, &ch119, + &ch120, &ch121, &ch122, &ch123, &ch124, &ch125, &ch126, &ch127 +}; + +const SFG_StrokeFont fgStrokeMonoRoman = {"MonoRoman",128,152.381f,chars}; diff --git a/examples/opengl-framework/freeglut/freeglut_stroke_roman.c b/examples/opengl-framework/freeglut/freeglut_stroke_roman.c new file mode 100644 index 00000000..da4e0302 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_stroke_roman.c @@ -0,0 +1,2849 @@ +/* + * freeglut_stroke_roman.c + * + * freeglut Roman stroke font definition + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 16 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + + +/* This file has been automatically generated by the genstroke utility. */ + +#include +#include "freeglut_internal.h" + +/* char: 0x20 */ + +static const SFG_StrokeStrip ch32st[] = +{ + { 0, NULL } +}; + +static const SFG_StrokeChar ch32 = {104.762f,0,ch32st}; + +/* char: 0x21 */ + +static const SFG_StrokeVertex ch33st0[] = +{ + {13.3819f,100.0f}, + {13.3819f,33.3333f} +}; + +static const SFG_StrokeVertex ch33st1[] = +{ + {13.3819f,9.5238f}, + {8.62f,4.7619f}, + {13.3819f,0.0f}, + {18.1438f,4.7619f}, + {13.3819f,9.5238f} +}; + +static const SFG_StrokeStrip ch33st[] = +{ + {2,ch33st0}, + {5,ch33st1} +}; + +static const SFG_StrokeChar ch33 = {26.6238f,2,ch33st}; + +/* char: 0x22 */ + +static const SFG_StrokeVertex ch34st0[] = +{ + {4.02f,100.0f}, + {4.02f,66.6667f} +}; + +static const SFG_StrokeVertex ch34st1[] = +{ + {42.1152f,100.0f}, + {42.1152f,66.6667f} +}; + +static const SFG_StrokeStrip ch34st[] = +{ + {2,ch34st0}, + {2,ch34st1} +}; + +static const SFG_StrokeChar ch34 = {51.4352f,2,ch34st}; + +/* char: 0x23 */ + +static const SFG_StrokeVertex ch35st0[] = +{ + {41.2952f,119.048f}, + {7.9619f,-33.3333f} +}; + +static const SFG_StrokeVertex ch35st1[] = +{ + {69.8667f,119.048f}, + {36.5333f,-33.3333f} +}; + +static const SFG_StrokeVertex ch35st2[] = +{ + {7.9619f,57.1429f}, + {74.6286f,57.1429f} +}; + +static const SFG_StrokeVertex ch35st3[] = +{ + {3.2f,28.5714f}, + {69.8667f,28.5714f} +}; + +static const SFG_StrokeStrip ch35st[] = +{ + {2,ch35st0}, + {2,ch35st1}, + {2,ch35st2}, + {2,ch35st3} +}; + +static const SFG_StrokeChar ch35 = {79.4886f,4,ch35st}; + +/* char: 0x24 */ + +static const SFG_StrokeVertex ch36st0[] = +{ + {28.6295f,119.048f}, + {28.6295f,-19.0476f} +}; + +static const SFG_StrokeVertex ch36st1[] = +{ + {47.6771f,119.048f}, + {47.6771f,-19.0476f} +}; + +static const SFG_StrokeVertex ch36st2[] = +{ + {71.4867f,85.7143f}, + {61.9629f,95.2381f}, + {47.6771f,100.0f}, + {28.6295f,100.0f}, + {14.3438f,95.2381f}, + {4.82f,85.7143f}, + {4.82f,76.1905f}, + {9.5819f,66.6667f}, + {14.3438f,61.9048f}, + {23.8676f,57.1429f}, + {52.439f,47.619f}, + {61.9629f,42.8571f}, + {66.7248f,38.0952f}, + {71.4867f,28.5714f}, + {71.4867f,14.2857f}, + {61.9629f,4.7619f}, + {47.6771f,0.0f}, + {28.6295f,0.0f}, + {14.3438f,4.7619f}, + {4.82f,14.2857f} +}; + +static const SFG_StrokeStrip ch36st[] = +{ + {2,ch36st0}, + {2,ch36st1}, + {20,ch36st2} +}; + +static const SFG_StrokeChar ch36 = {76.2067f,3,ch36st}; + +/* char: 0x25 */ + +static const SFG_StrokeVertex ch37st0[] = +{ + {92.0743f,100.0f}, + {6.36f,0.0f} +}; + +static const SFG_StrokeVertex ch37st1[] = +{ + {30.1695f,100.0f}, + {39.6933f,90.4762f}, + {39.6933f,80.9524f}, + {34.9314f,71.4286f}, + {25.4076f,66.6667f}, + {15.8838f,66.6667f}, + {6.36f,76.1905f}, + {6.36f,85.7143f}, + {11.1219f,95.2381f}, + {20.6457f,100.0f}, + {30.1695f,100.0f}, + {39.6933f,95.2381f}, + {53.979f,90.4762f}, + {68.2648f,90.4762f}, + {82.5505f,95.2381f}, + {92.0743f,100.0f} +}; + +static const SFG_StrokeVertex ch37st2[] = +{ + {73.0267f,33.3333f}, + {63.5029f,28.5714f}, + {58.741f,19.0476f}, + {58.741f,9.5238f}, + {68.2648f,0.0f}, + {77.7886f,0.0f}, + {87.3124f,4.7619f}, + {92.0743f,14.2857f}, + {92.0743f,23.8095f}, + {82.5505f,33.3333f}, + {73.0267f,33.3333f} +}; + +static const SFG_StrokeStrip ch37st[] = +{ + {2,ch37st0}, + {16,ch37st1}, + {11,ch37st2} +}; + +static const SFG_StrokeChar ch37 = {96.5743f,3,ch37st}; + +/* char: 0x26 */ + +static const SFG_StrokeVertex ch38st0[] = +{ + {101.218f,57.1429f}, + {101.218f,61.9048f}, + {96.4562f,66.6667f}, + {91.6943f,66.6667f}, + {86.9324f,61.9048f}, + {82.1705f,52.381f}, + {72.6467f,28.5714f}, + {63.1229f,14.2857f}, + {53.599f,4.7619f}, + {44.0752f,0.0f}, + {25.0276f,0.0f}, + {15.5038f,4.7619f}, + {10.7419f,9.5238f}, + {5.98f,19.0476f}, + {5.98f,28.5714f}, + {10.7419f,38.0952f}, + {15.5038f,42.8571f}, + {48.8371f,61.9048f}, + {53.599f,66.6667f}, + {58.361f,76.1905f}, + {58.361f,85.7143f}, + {53.599f,95.2381f}, + {44.0752f,100.0f}, + {34.5514f,95.2381f}, + {29.7895f,85.7143f}, + {29.7895f,76.1905f}, + {34.5514f,61.9048f}, + {44.0752f,47.619f}, + {67.8848f,14.2857f}, + {77.4086f,4.7619f}, + {86.9324f,0.0f}, + {96.4562f,0.0f}, + {101.218f,4.7619f}, + {101.218f,9.5238f} +}; + +static const SFG_StrokeStrip ch38st[] = +{ + {34,ch38st0} +}; + +static const SFG_StrokeChar ch38 = {101.758f,1,ch38st}; + +/* char: 0x27 */ + +static const SFG_StrokeVertex ch39st0[] = +{ + {4.44f,100.0f}, + {4.44f,66.6667f} +}; + +static const SFG_StrokeStrip ch39st[] = +{ + {2,ch39st0} +}; + +static const SFG_StrokeChar ch39 = {13.62f,1,ch39st}; + +/* char: 0x28 */ + +static const SFG_StrokeVertex ch40st0[] = +{ + {40.9133f,119.048f}, + {31.3895f,109.524f}, + {21.8657f,95.2381f}, + {12.3419f,76.1905f}, + {7.58f,52.381f}, + {7.58f,33.3333f}, + {12.3419f,9.5238f}, + {21.8657f,-9.5238f}, + {31.3895f,-23.8095f}, + {40.9133f,-33.3333f} +}; + +static const SFG_StrokeStrip ch40st[] = +{ + {10,ch40st0} +}; + +static const SFG_StrokeChar ch40 = {47.1733f,1,ch40st}; + +/* char: 0x29 */ + +static const SFG_StrokeVertex ch41st0[] = +{ + {5.28f,119.048f}, + {14.8038f,109.524f}, + {24.3276f,95.2381f}, + {33.8514f,76.1905f}, + {38.6133f,52.381f}, + {38.6133f,33.3333f}, + {33.8514f,9.5238f}, + {24.3276f,-9.5238f}, + {14.8038f,-23.8095f}, + {5.28f,-33.3333f} +}; + +static const SFG_StrokeStrip ch41st[] = +{ + {10,ch41st0} +}; + +static const SFG_StrokeChar ch41 = {47.5333f,1,ch41st}; + +/* char: 0x2a */ + +static const SFG_StrokeVertex ch42st0[] = +{ + {30.7695f,71.4286f}, + {30.7695f,14.2857f} +}; + +static const SFG_StrokeVertex ch42st1[] = +{ + {6.96f,57.1429f}, + {54.579f,28.5714f} +}; + +static const SFG_StrokeVertex ch42st2[] = +{ + {54.579f,57.1429f}, + {6.96f,28.5714f} +}; + +static const SFG_StrokeStrip ch42st[] = +{ + {2,ch42st0}, + {2,ch42st1}, + {2,ch42st2} +}; + +static const SFG_StrokeChar ch42 = {59.439f,3,ch42st}; + +/* char: 0x2b */ + +static const SFG_StrokeVertex ch43st0[] = +{ + {48.8371f,85.7143f}, + {48.8371f,0.0f} +}; + +static const SFG_StrokeVertex ch43st1[] = +{ + {5.98f,42.8571f}, + {91.6943f,42.8571f} +}; + +static const SFG_StrokeStrip ch43st[] = +{ + {2,ch43st0}, + {2,ch43st1} +}; + +static const SFG_StrokeChar ch43 = {97.2543f,2,ch43st}; + +/* char: 0x2c */ + +static const SFG_StrokeVertex ch44st0[] = +{ + {18.2838f,4.7619f}, + {13.5219f,0.0f}, + {8.76f,4.7619f}, + {13.5219f,9.5238f}, + {18.2838f,4.7619f}, + {18.2838f,-4.7619f}, + {13.5219f,-14.2857f}, + {8.76f,-19.0476f} +}; + +static const SFG_StrokeStrip ch44st[] = +{ + {8,ch44st0} +}; + +static const SFG_StrokeChar ch44 = {26.0638f,1,ch44st}; + +/* char: 0x2d */ + +static const SFG_StrokeVertex ch45st0[] = +{ + {7.38f,42.8571f}, + {93.0943f,42.8571f} +}; + +static const SFG_StrokeStrip ch45st[] = +{ + {2,ch45st0} +}; + +static const SFG_StrokeChar ch45 = {100.754f,1,ch45st}; + +/* char: 0x2e */ + +static const SFG_StrokeVertex ch46st0[] = +{ + {13.1019f,9.5238f}, + {8.34f,4.7619f}, + {13.1019f,0.0f}, + {17.8638f,4.7619f}, + {13.1019f,9.5238f} +}; + +static const SFG_StrokeStrip ch46st[] = +{ + {5,ch46st0} +}; + +static const SFG_StrokeChar ch46 = {26.4838f,1,ch46st}; + +/* char: 0x2f */ + +static const SFG_StrokeVertex ch47st0[] = +{ + {7.24f,-14.2857f}, + {73.9067f,100.0f} +}; + +static const SFG_StrokeStrip ch47st[] = +{ + {2,ch47st0} +}; + +static const SFG_StrokeChar ch47 = {82.1067f,1,ch47st}; + +/* char: 0x30 */ + +static const SFG_StrokeVertex ch48st0[] = +{ + {33.5514f,100.0f}, + {19.2657f,95.2381f}, + {9.7419f,80.9524f}, + {4.98f,57.1429f}, + {4.98f,42.8571f}, + {9.7419f,19.0476f}, + {19.2657f,4.7619f}, + {33.5514f,0.0f}, + {43.0752f,0.0f}, + {57.361f,4.7619f}, + {66.8848f,19.0476f}, + {71.6467f,42.8571f}, + {71.6467f,57.1429f}, + {66.8848f,80.9524f}, + {57.361f,95.2381f}, + {43.0752f,100.0f}, + {33.5514f,100.0f} +}; + +static const SFG_StrokeStrip ch48st[] = +{ + {17,ch48st0} +}; + +static const SFG_StrokeChar ch48 = {77.0667f,1,ch48st}; + +/* char: 0x31 */ + +static const SFG_StrokeVertex ch49st0[] = +{ + {11.82f,80.9524f}, + {21.3438f,85.7143f}, + {35.6295f,100.0f}, + {35.6295f,0.0f} +}; + +static const SFG_StrokeStrip ch49st[] = +{ + {4,ch49st0} +}; + +static const SFG_StrokeChar ch49 = {66.5295f,1,ch49st}; + +/* char: 0x32 */ + +static const SFG_StrokeVertex ch50st0[] = +{ + {10.1819f,76.1905f}, + {10.1819f,80.9524f}, + {14.9438f,90.4762f}, + {19.7057f,95.2381f}, + {29.2295f,100.0f}, + {48.2771f,100.0f}, + {57.801f,95.2381f}, + {62.5629f,90.4762f}, + {67.3248f,80.9524f}, + {67.3248f,71.4286f}, + {62.5629f,61.9048f}, + {53.039f,47.619f}, + {5.42f,0.0f}, + {72.0867f,0.0f} +}; + +static const SFG_StrokeStrip ch50st[] = +{ + {14,ch50st0} +}; + +static const SFG_StrokeChar ch50 = {77.6467f,1,ch50st}; + +/* char: 0x33 */ + +static const SFG_StrokeVertex ch51st0[] = +{ + {14.5238f,100.0f}, + {66.9048f,100.0f}, + {38.3333f,61.9048f}, + {52.619f,61.9048f}, + {62.1429f,57.1429f}, + {66.9048f,52.381f}, + {71.6667f,38.0952f}, + {71.6667f,28.5714f}, + {66.9048f,14.2857f}, + {57.381f,4.7619f}, + {43.0952f,0.0f}, + {28.8095f,0.0f}, + {14.5238f,4.7619f}, + {9.7619f,9.5238f}, + {5.0f,19.0476f} +}; + +static const SFG_StrokeStrip ch51st[] = +{ + {15,ch51st0} +}; + +static const SFG_StrokeChar ch51 = {77.0467f,1,ch51st}; + +/* char: 0x34 */ + +static const SFG_StrokeVertex ch52st0[] = +{ + {51.499f,100.0f}, + {3.88f,33.3333f}, + {75.3086f,33.3333f} +}; + +static const SFG_StrokeVertex ch52st1[] = +{ + {51.499f,100.0f}, + {51.499f,0.0f} +}; + +static const SFG_StrokeStrip ch52st[] = +{ + {3,ch52st0}, + {2,ch52st1} +}; + +static const SFG_StrokeChar ch52 = {80.1686f,2,ch52st}; + +/* char: 0x35 */ + +static const SFG_StrokeVertex ch53st0[] = +{ + {62.0029f,100.0f}, + {14.3838f,100.0f}, + {9.6219f,57.1429f}, + {14.3838f,61.9048f}, + {28.6695f,66.6667f}, + {42.9552f,66.6667f}, + {57.241f,61.9048f}, + {66.7648f,52.381f}, + {71.5267f,38.0952f}, + {71.5267f,28.5714f}, + {66.7648f,14.2857f}, + {57.241f,4.7619f}, + {42.9552f,0.0f}, + {28.6695f,0.0f}, + {14.3838f,4.7619f}, + {9.6219f,9.5238f}, + {4.86f,19.0476f} +}; + +static const SFG_StrokeStrip ch53st[] = +{ + {17,ch53st0} +}; + +static const SFG_StrokeChar ch53 = {77.6867f,1,ch53st}; + +/* char: 0x36 */ + +static const SFG_StrokeVertex ch54st0[] = +{ + {62.7229f,85.7143f}, + {57.961f,95.2381f}, + {43.6752f,100.0f}, + {34.1514f,100.0f}, + {19.8657f,95.2381f}, + {10.3419f,80.9524f}, + {5.58f,57.1429f}, + {5.58f,33.3333f}, + {10.3419f,14.2857f}, + {19.8657f,4.7619f}, + {34.1514f,0.0f}, + {38.9133f,0.0f}, + {53.199f,4.7619f}, + {62.7229f,14.2857f}, + {67.4848f,28.5714f}, + {67.4848f,33.3333f}, + {62.7229f,47.619f}, + {53.199f,57.1429f}, + {38.9133f,61.9048f}, + {34.1514f,61.9048f}, + {19.8657f,57.1429f}, + {10.3419f,47.619f}, + {5.58f,33.3333f} +}; + +static const SFG_StrokeStrip ch54st[] = +{ + {23,ch54st0} +}; + +static const SFG_StrokeChar ch54 = {73.8048f,1,ch54st}; + +/* char: 0x37 */ + +static const SFG_StrokeVertex ch55st0[] = +{ + {72.2267f,100.0f}, + {24.6076f,0.0f} +}; + +static const SFG_StrokeVertex ch55st1[] = +{ + {5.56f,100.0f}, + {72.2267f,100.0f} +}; + +static const SFG_StrokeStrip ch55st[] = +{ + {2,ch55st0}, + {2,ch55st1} +}; + +static const SFG_StrokeChar ch55 = {77.2267f,2,ch55st}; + +/* char: 0x38 */ + +static const SFG_StrokeVertex ch56st0[] = +{ + {29.4095f,100.0f}, + {15.1238f,95.2381f}, + {10.3619f,85.7143f}, + {10.3619f,76.1905f}, + {15.1238f,66.6667f}, + {24.6476f,61.9048f}, + {43.6952f,57.1429f}, + {57.981f,52.381f}, + {67.5048f,42.8571f}, + {72.2667f,33.3333f}, + {72.2667f,19.0476f}, + {67.5048f,9.5238f}, + {62.7429f,4.7619f}, + {48.4571f,0.0f}, + {29.4095f,0.0f}, + {15.1238f,4.7619f}, + {10.3619f,9.5238f}, + {5.6f,19.0476f}, + {5.6f,33.3333f}, + {10.3619f,42.8571f}, + {19.8857f,52.381f}, + {34.1714f,57.1429f}, + {53.219f,61.9048f}, + {62.7429f,66.6667f}, + {67.5048f,76.1905f}, + {67.5048f,85.7143f}, + {62.7429f,95.2381f}, + {48.4571f,100.0f}, + {29.4095f,100.0f} +}; + +static const SFG_StrokeStrip ch56st[] = +{ + {29,ch56st0} +}; + +static const SFG_StrokeChar ch56 = {77.6667f,1,ch56st}; + +/* char: 0x39 */ + +static const SFG_StrokeVertex ch57st0[] = +{ + {68.5048f,66.6667f}, + {63.7429f,52.381f}, + {54.219f,42.8571f}, + {39.9333f,38.0952f}, + {35.1714f,38.0952f}, + {20.8857f,42.8571f}, + {11.3619f,52.381f}, + {6.6f,66.6667f}, + {6.6f,71.4286f}, + {11.3619f,85.7143f}, + {20.8857f,95.2381f}, + {35.1714f,100.0f}, + {39.9333f,100.0f}, + {54.219f,95.2381f}, + {63.7429f,85.7143f}, + {68.5048f,66.6667f}, + {68.5048f,42.8571f}, + {63.7429f,19.0476f}, + {54.219f,4.7619f}, + {39.9333f,0.0f}, + {30.4095f,0.0f}, + {16.1238f,4.7619f}, + {11.3619f,14.2857f} +}; + +static const SFG_StrokeStrip ch57st[] = +{ + {23,ch57st0} +}; + +static const SFG_StrokeChar ch57 = {74.0648f,1,ch57st}; + +/* char: 0x3a */ + +static const SFG_StrokeVertex ch58st0[] = +{ + {14.0819f,66.6667f}, + {9.32f,61.9048f}, + {14.0819f,57.1429f}, + {18.8438f,61.9048f}, + {14.0819f,66.6667f} +}; + +static const SFG_StrokeVertex ch58st1[] = +{ + {14.0819f,9.5238f}, + {9.32f,4.7619f}, + {14.0819f,0.0f}, + {18.8438f,4.7619f}, + {14.0819f,9.5238f} +}; + +static const SFG_StrokeStrip ch58st[] = +{ + {5,ch58st0}, + {5,ch58st1} +}; + +static const SFG_StrokeChar ch58 = {26.2238f,2,ch58st}; + +/* char: 0x3b */ + +static const SFG_StrokeVertex ch59st0[] = +{ + {12.9619f,66.6667f}, + {8.2f,61.9048f}, + {12.9619f,57.1429f}, + {17.7238f,61.9048f}, + {12.9619f,66.6667f} +}; + +static const SFG_StrokeVertex ch59st1[] = +{ + {17.7238f,4.7619f}, + {12.9619f,0.0f}, + {8.2f,4.7619f}, + {12.9619f,9.5238f}, + {17.7238f,4.7619f}, + {17.7238f,-4.7619f}, + {12.9619f,-14.2857f}, + {8.2f,-19.0476f} +}; + +static const SFG_StrokeStrip ch59st[] = +{ + {5,ch59st0}, + {8,ch59st1} +}; + +static const SFG_StrokeChar ch59 = {26.3038f,2,ch59st}; + +/* char: 0x3c */ + +static const SFG_StrokeVertex ch60st0[] = +{ + {79.2505f,85.7143f}, + {3.06f,42.8571f}, + {79.2505f,0.0f} +}; + +static const SFG_StrokeStrip ch60st[] = +{ + {3,ch60st0} +}; + +static const SFG_StrokeChar ch60 = {81.6105f,1,ch60st}; + +/* char: 0x3d */ + +static const SFG_StrokeVertex ch61st0[] = +{ + {5.7f,57.1429f}, + {91.4143f,57.1429f} +}; + +static const SFG_StrokeVertex ch61st1[] = +{ + {5.7f,28.5714f}, + {91.4143f,28.5714f} +}; + +static const SFG_StrokeStrip ch61st[] = +{ + {2,ch61st0}, + {2,ch61st1} +}; + +static const SFG_StrokeChar ch61 = {97.2543f,2,ch61st}; + +/* char: 0x3e */ + +static const SFG_StrokeVertex ch62st0[] = +{ + {2.78f,85.7143f}, + {78.9705f,42.8571f}, + {2.78f,0.0f} +}; + +static const SFG_StrokeStrip ch62st[] = +{ + {3,ch62st0} +}; + +static const SFG_StrokeChar ch62 = {81.6105f,1,ch62st}; + +/* char: 0x3f */ + +static const SFG_StrokeVertex ch63st0[] = +{ + {8.42f,76.1905f}, + {8.42f,80.9524f}, + {13.1819f,90.4762f}, + {17.9438f,95.2381f}, + {27.4676f,100.0f}, + {46.5152f,100.0f}, + {56.039f,95.2381f}, + {60.801f,90.4762f}, + {65.5629f,80.9524f}, + {65.5629f,71.4286f}, + {60.801f,61.9048f}, + {56.039f,57.1429f}, + {36.9914f,47.619f}, + {36.9914f,33.3333f} +}; + +static const SFG_StrokeVertex ch63st1[] = +{ + {36.9914f,9.5238f}, + {32.2295f,4.7619f}, + {36.9914f,0.0f}, + {41.7533f,4.7619f}, + {36.9914f,9.5238f} +}; + +static const SFG_StrokeStrip ch63st[] = +{ + {14,ch63st0}, + {5,ch63st1} +}; + +static const SFG_StrokeChar ch63 = {73.9029f,2,ch63st}; + +/* char: 0x40 */ + +static const SFG_StrokeVertex ch64st0[] = +{ + {49.2171f,52.381f}, + {39.6933f,57.1429f}, + {30.1695f,57.1429f}, + {25.4076f,47.619f}, + {25.4076f,42.8571f}, + {30.1695f,33.3333f}, + {39.6933f,33.3333f}, + {49.2171f,38.0952f} +}; + +static const SFG_StrokeVertex ch64st1[] = +{ + {49.2171f,57.1429f}, + {49.2171f,38.0952f}, + {53.979f,33.3333f}, + {63.5029f,33.3333f}, + {68.2648f,42.8571f}, + {68.2648f,47.619f}, + {63.5029f,61.9048f}, + {53.979f,71.4286f}, + {39.6933f,76.1905f}, + {34.9314f,76.1905f}, + {20.6457f,71.4286f}, + {11.1219f,61.9048f}, + {6.36f,47.619f}, + {6.36f,42.8571f}, + {11.1219f,28.5714f}, + {20.6457f,19.0476f}, + {34.9314f,14.2857f}, + {39.6933f,14.2857f}, + {53.979f,19.0476f} +}; + +static const SFG_StrokeStrip ch64st[] = +{ + {8,ch64st0}, + {19,ch64st1} +}; + +static const SFG_StrokeChar ch64 = {74.3648f,2,ch64st}; + +/* char: 0x41 */ + +static const SFG_StrokeVertex ch65st0[] = +{ + {40.5952f,100.0f}, + {2.5f,0.0f} +}; + +static const SFG_StrokeVertex ch65st1[] = +{ + {40.5952f,100.0f}, + {78.6905f,0.0f} +}; + +static const SFG_StrokeVertex ch65st2[] = +{ + {16.7857f,33.3333f}, + {64.4048f,33.3333f} +}; + +static const SFG_StrokeStrip ch65st[] = +{ + {2,ch65st0}, + {2,ch65st1}, + {2,ch65st2} +}; + +static const SFG_StrokeChar ch65 = {80.4905f,3,ch65st}; + +/* char: 0x42 */ + +static const SFG_StrokeVertex ch66st0[] = +{ + {11.42f,100.0f}, + {11.42f,0.0f} +}; + +static const SFG_StrokeVertex ch66st1[] = +{ + {11.42f,100.0f}, + {54.2771f,100.0f}, + {68.5629f,95.2381f}, + {73.3248f,90.4762f}, + {78.0867f,80.9524f}, + {78.0867f,71.4286f}, + {73.3248f,61.9048f}, + {68.5629f,57.1429f}, + {54.2771f,52.381f} +}; + +static const SFG_StrokeVertex ch66st2[] = +{ + {11.42f,52.381f}, + {54.2771f,52.381f}, + {68.5629f,47.619f}, + {73.3248f,42.8571f}, + {78.0867f,33.3333f}, + {78.0867f,19.0476f}, + {73.3248f,9.5238f}, + {68.5629f,4.7619f}, + {54.2771f,0.0f}, + {11.42f,0.0f} +}; + +static const SFG_StrokeStrip ch66st[] = +{ + {2,ch66st0}, + {9,ch66st1}, + {10,ch66st2} +}; + +static const SFG_StrokeChar ch66 = {83.6267f,3,ch66st}; + +/* char: 0x43 */ + +static const SFG_StrokeVertex ch67st0[] = +{ + {78.0886f,76.1905f}, + {73.3267f,85.7143f}, + {63.8029f,95.2381f}, + {54.279f,100.0f}, + {35.2314f,100.0f}, + {25.7076f,95.2381f}, + {16.1838f,85.7143f}, + {11.4219f,76.1905f}, + {6.66f,61.9048f}, + {6.66f,38.0952f}, + {11.4219f,23.8095f}, + {16.1838f,14.2857f}, + {25.7076f,4.7619f}, + {35.2314f,0.0f}, + {54.279f,0.0f}, + {63.8029f,4.7619f}, + {73.3267f,14.2857f}, + {78.0886f,23.8095f} +}; + +static const SFG_StrokeStrip ch67st[] = +{ + {18,ch67st0} +}; + +static const SFG_StrokeChar ch67 = {84.4886f,1,ch67st}; + +/* char: 0x44 */ + +static const SFG_StrokeVertex ch68st0[] = +{ + {11.96f,100.0f}, + {11.96f,0.0f} +}; + +static const SFG_StrokeVertex ch68st1[] = +{ + {11.96f,100.0f}, + {45.2933f,100.0f}, + {59.579f,95.2381f}, + {69.1029f,85.7143f}, + {73.8648f,76.1905f}, + {78.6267f,61.9048f}, + {78.6267f,38.0952f}, + {73.8648f,23.8095f}, + {69.1029f,14.2857f}, + {59.579f,4.7619f}, + {45.2933f,0.0f}, + {11.96f,0.0f} +}; + +static const SFG_StrokeStrip ch68st[] = +{ + {2,ch68st0}, + {12,ch68st1} +}; + +static const SFG_StrokeChar ch68 = {85.2867f,2,ch68st}; + +/* char: 0x45 */ + +static const SFG_StrokeVertex ch69st0[] = +{ + {11.42f,100.0f}, + {11.42f,0.0f} +}; + +static const SFG_StrokeVertex ch69st1[] = +{ + {11.42f,100.0f}, + {73.3248f,100.0f} +}; + +static const SFG_StrokeVertex ch69st2[] = +{ + {11.42f,52.381f}, + {49.5152f,52.381f} +}; + +static const SFG_StrokeVertex ch69st3[] = +{ + {11.42f,0.0f}, + {73.3248f,0.0f} +}; + +static const SFG_StrokeStrip ch69st[] = +{ + {2,ch69st0}, + {2,ch69st1}, + {2,ch69st2}, + {2,ch69st3} +}; + +static const SFG_StrokeChar ch69 = {78.1848f,4,ch69st}; + +/* char: 0x46 */ + +static const SFG_StrokeVertex ch70st0[] = +{ + {11.42f,100.0f}, + {11.42f,0.0f} +}; + +static const SFG_StrokeVertex ch70st1[] = +{ + {11.42f,100.0f}, + {73.3248f,100.0f} +}; + +static const SFG_StrokeVertex ch70st2[] = +{ + {11.42f,52.381f}, + {49.5152f,52.381f} +}; + +static const SFG_StrokeStrip ch70st[] = +{ + {2,ch70st0}, + {2,ch70st1}, + {2,ch70st2} +}; + +static const SFG_StrokeChar ch70 = {78.7448f,3,ch70st}; + +/* char: 0x47 */ + +static const SFG_StrokeVertex ch71st0[] = +{ + {78.4886f,76.1905f}, + {73.7267f,85.7143f}, + {64.2029f,95.2381f}, + {54.679f,100.0f}, + {35.6314f,100.0f}, + {26.1076f,95.2381f}, + {16.5838f,85.7143f}, + {11.8219f,76.1905f}, + {7.06f,61.9048f}, + {7.06f,38.0952f}, + {11.8219f,23.8095f}, + {16.5838f,14.2857f}, + {26.1076f,4.7619f}, + {35.6314f,0.0f}, + {54.679f,0.0f}, + {64.2029f,4.7619f}, + {73.7267f,14.2857f}, + {78.4886f,23.8095f}, + {78.4886f,38.0952f} +}; + +static const SFG_StrokeVertex ch71st1[] = +{ + {54.679f,38.0952f}, + {78.4886f,38.0952f} +}; + +static const SFG_StrokeStrip ch71st[] = +{ + {19,ch71st0}, + {2,ch71st1} +}; + +static const SFG_StrokeChar ch71 = {89.7686f,2,ch71st}; + +/* char: 0x48 */ + +static const SFG_StrokeVertex ch72st0[] = +{ + {11.42f,100.0f}, + {11.42f,0.0f} +}; + +static const SFG_StrokeVertex ch72st1[] = +{ + {78.0867f,100.0f}, + {78.0867f,0.0f} +}; + +static const SFG_StrokeVertex ch72st2[] = +{ + {11.42f,52.381f}, + {78.0867f,52.381f} +}; + +static const SFG_StrokeStrip ch72st[] = +{ + {2,ch72st0}, + {2,ch72st1}, + {2,ch72st2} +}; + +static const SFG_StrokeChar ch72 = {89.0867f,3,ch72st}; + +/* char: 0x49 */ + +static const SFG_StrokeVertex ch73st0[] = +{ + {10.86f,100.0f}, + {10.86f,0.0f} +}; + +static const SFG_StrokeStrip ch73st[] = +{ + {2,ch73st0} +}; + +static const SFG_StrokeChar ch73 = {21.3f,1,ch73st}; + +/* char: 0x4a */ + +static const SFG_StrokeVertex ch74st0[] = +{ + {50.119f,100.0f}, + {50.119f,23.8095f}, + {45.3571f,9.5238f}, + {40.5952f,4.7619f}, + {31.0714f,0.0f}, + {21.5476f,0.0f}, + {12.0238f,4.7619f}, + {7.2619f,9.5238f}, + {2.5f,23.8095f}, + {2.5f,33.3333f} +}; + +static const SFG_StrokeStrip ch74st[] = +{ + {10,ch74st0} +}; + +static const SFG_StrokeChar ch74 = {59.999f,1,ch74st}; + +/* char: 0x4b */ + +static const SFG_StrokeVertex ch75st0[] = +{ + {11.28f,100.0f}, + {11.28f,0.0f} +}; + +static const SFG_StrokeVertex ch75st1[] = +{ + {77.9467f,100.0f}, + {11.28f,33.3333f} +}; + +static const SFG_StrokeVertex ch75st2[] = +{ + {35.0895f,57.1429f}, + {77.9467f,0.0f} +}; + +static const SFG_StrokeStrip ch75st[] = +{ + {2,ch75st0}, + {2,ch75st1}, + {2,ch75st2} +}; + +static const SFG_StrokeChar ch75 = {79.3267f,3,ch75st}; + +/* char: 0x4c */ + +static const SFG_StrokeVertex ch76st0[] = +{ + {11.68f,100.0f}, + {11.68f,0.0f} +}; + +static const SFG_StrokeVertex ch76st1[] = +{ + {11.68f,0.0f}, + {68.8229f,0.0f} +}; + +static const SFG_StrokeStrip ch76st[] = +{ + {2,ch76st0}, + {2,ch76st1} +}; + +static const SFG_StrokeChar ch76 = {71.3229f,2,ch76st}; + +/* char: 0x4d */ + +static const SFG_StrokeVertex ch77st0[] = +{ + {10.86f,100.0f}, + {10.86f,0.0f} +}; + +static const SFG_StrokeVertex ch77st1[] = +{ + {10.86f,100.0f}, + {48.9552f,0.0f} +}; + +static const SFG_StrokeVertex ch77st2[] = +{ + {87.0505f,100.0f}, + {48.9552f,0.0f} +}; + +static const SFG_StrokeVertex ch77st3[] = +{ + {87.0505f,100.0f}, + {87.0505f,0.0f} +}; + +static const SFG_StrokeStrip ch77st[] = +{ + {2,ch77st0}, + {2,ch77st1}, + {2,ch77st2}, + {2,ch77st3} +}; + +static const SFG_StrokeChar ch77 = {97.2105f,4,ch77st}; + +/* char: 0x4e */ + +static const SFG_StrokeVertex ch78st0[] = +{ + {11.14f,100.0f}, + {11.14f,0.0f} +}; + +static const SFG_StrokeVertex ch78st1[] = +{ + {11.14f,100.0f}, + {77.8067f,0.0f} +}; + +static const SFG_StrokeVertex ch78st2[] = +{ + {77.8067f,100.0f}, + {77.8067f,0.0f} +}; + +static const SFG_StrokeStrip ch78st[] = +{ + {2,ch78st0}, + {2,ch78st1}, + {2,ch78st2} +}; + +static const SFG_StrokeChar ch78 = {88.8067f,3,ch78st}; + +/* char: 0x4f */ + +static const SFG_StrokeVertex ch79st0[] = +{ + {34.8114f,100.0f}, + {25.2876f,95.2381f}, + {15.7638f,85.7143f}, + {11.0019f,76.1905f}, + {6.24f,61.9048f}, + {6.24f,38.0952f}, + {11.0019f,23.8095f}, + {15.7638f,14.2857f}, + {25.2876f,4.7619f}, + {34.8114f,0.0f}, + {53.859f,0.0f}, + {63.3829f,4.7619f}, + {72.9067f,14.2857f}, + {77.6686f,23.8095f}, + {82.4305f,38.0952f}, + {82.4305f,61.9048f}, + {77.6686f,76.1905f}, + {72.9067f,85.7143f}, + {63.3829f,95.2381f}, + {53.859f,100.0f}, + {34.8114f,100.0f} +}; + +static const SFG_StrokeStrip ch79st[] = +{ + {21,ch79st0} +}; + +static const SFG_StrokeChar ch79 = {88.8305f,1,ch79st}; + +/* char: 0x50 */ + +static const SFG_StrokeVertex ch80st0[] = +{ + {12.1f,100.0f}, + {12.1f,0.0f} +}; + +static const SFG_StrokeVertex ch80st1[] = +{ + {12.1f,100.0f}, + {54.9571f,100.0f}, + {69.2429f,95.2381f}, + {74.0048f,90.4762f}, + {78.7667f,80.9524f}, + {78.7667f,66.6667f}, + {74.0048f,57.1429f}, + {69.2429f,52.381f}, + {54.9571f,47.619f}, + {12.1f,47.619f} +}; + +static const SFG_StrokeStrip ch80st[] = +{ + {2,ch80st0}, + {10,ch80st1} +}; + +static const SFG_StrokeChar ch80 = {85.6667f,2,ch80st}; + +/* char: 0x51 */ + +static const SFG_StrokeVertex ch81st0[] = +{ + {33.8714f,100.0f}, + {24.3476f,95.2381f}, + {14.8238f,85.7143f}, + {10.0619f,76.1905f}, + {5.3f,61.9048f}, + {5.3f,38.0952f}, + {10.0619f,23.8095f}, + {14.8238f,14.2857f}, + {24.3476f,4.7619f}, + {33.8714f,0.0f}, + {52.919f,0.0f}, + {62.4429f,4.7619f}, + {71.9667f,14.2857f}, + {76.7286f,23.8095f}, + {81.4905f,38.0952f}, + {81.4905f,61.9048f}, + {76.7286f,76.1905f}, + {71.9667f,85.7143f}, + {62.4429f,95.2381f}, + {52.919f,100.0f}, + {33.8714f,100.0f} +}; + +static const SFG_StrokeVertex ch81st1[] = +{ + {48.1571f,19.0476f}, + {76.7286f,-9.5238f} +}; + +static const SFG_StrokeStrip ch81st[] = +{ + {21,ch81st0}, + {2,ch81st1} +}; + +static const SFG_StrokeChar ch81 = {88.0905f,2,ch81st}; + +/* char: 0x52 */ + +static const SFG_StrokeVertex ch82st0[] = +{ + {11.68f,100.0f}, + {11.68f,0.0f} +}; + +static const SFG_StrokeVertex ch82st1[] = +{ + {11.68f,100.0f}, + {54.5371f,100.0f}, + {68.8229f,95.2381f}, + {73.5848f,90.4762f}, + {78.3467f,80.9524f}, + {78.3467f,71.4286f}, + {73.5848f,61.9048f}, + {68.8229f,57.1429f}, + {54.5371f,52.381f}, + {11.68f,52.381f} +}; + +static const SFG_StrokeVertex ch82st2[] = +{ + {45.0133f,52.381f}, + {78.3467f,0.0f} +}; + +static const SFG_StrokeStrip ch82st[] = +{ + {2,ch82st0}, + {10,ch82st1}, + {2,ch82st2} +}; + +static const SFG_StrokeChar ch82 = {82.3667f,3,ch82st}; + +/* char: 0x53 */ + +static const SFG_StrokeVertex ch83st0[] = +{ + {74.6667f,85.7143f}, + {65.1429f,95.2381f}, + {50.8571f,100.0f}, + {31.8095f,100.0f}, + {17.5238f,95.2381f}, + {8.0f,85.7143f}, + {8.0f,76.1905f}, + {12.7619f,66.6667f}, + {17.5238f,61.9048f}, + {27.0476f,57.1429f}, + {55.619f,47.619f}, + {65.1429f,42.8571f}, + {69.9048f,38.0952f}, + {74.6667f,28.5714f}, + {74.6667f,14.2857f}, + {65.1429f,4.7619f}, + {50.8571f,0.0f}, + {31.8095f,0.0f}, + {17.5238f,4.7619f}, + {8.0f,14.2857f} +}; + +static const SFG_StrokeStrip ch83st[] = +{ + {20,ch83st0} +}; + +static const SFG_StrokeChar ch83 = {80.8267f,1,ch83st}; + +/* char: 0x54 */ + +static const SFG_StrokeVertex ch84st0[] = +{ + {35.6933f,100.0f}, + {35.6933f,0.0f} +}; + +static const SFG_StrokeVertex ch84st1[] = +{ + {2.36f,100.0f}, + {69.0267f,100.0f} +}; + +static const SFG_StrokeStrip ch84st[] = +{ + {2,ch84st0}, + {2,ch84st1} +}; + +static const SFG_StrokeChar ch84 = {71.9467f,2,ch84st}; + +/* char: 0x55 */ + +static const SFG_StrokeVertex ch85st0[] = +{ + {11.54f,100.0f}, + {11.54f,28.5714f}, + {16.3019f,14.2857f}, + {25.8257f,4.7619f}, + {40.1114f,0.0f}, + {49.6352f,0.0f}, + {63.921f,4.7619f}, + {73.4448f,14.2857f}, + {78.2067f,28.5714f}, + {78.2067f,100.0f} +}; + +static const SFG_StrokeStrip ch85st[] = +{ + {10,ch85st0} +}; + +static const SFG_StrokeChar ch85 = {89.4867f,1,ch85st}; + +/* char: 0x56 */ + +static const SFG_StrokeVertex ch86st0[] = +{ + {2.36f,100.0f}, + {40.4552f,0.0f} +}; + +static const SFG_StrokeVertex ch86st1[] = +{ + {78.5505f,100.0f}, + {40.4552f,0.0f} +}; + +static const SFG_StrokeStrip ch86st[] = +{ + {2,ch86st0}, + {2,ch86st1} +}; + +static const SFG_StrokeChar ch86 = {81.6105f,2,ch86st}; + +/* char: 0x57 */ + +static const SFG_StrokeVertex ch87st0[] = +{ + {2.22f,100.0f}, + {26.0295f,0.0f} +}; + +static const SFG_StrokeVertex ch87st1[] = +{ + {49.839f,100.0f}, + {26.0295f,0.0f} +}; + +static const SFG_StrokeVertex ch87st2[] = +{ + {49.839f,100.0f}, + {73.6486f,0.0f} +}; + +static const SFG_StrokeVertex ch87st3[] = +{ + {97.4581f,100.0f}, + {73.6486f,0.0f} +}; + +static const SFG_StrokeStrip ch87st[] = +{ + {2,ch87st0}, + {2,ch87st1}, + {2,ch87st2}, + {2,ch87st3} +}; + +static const SFG_StrokeChar ch87 = {100.518f,4,ch87st}; + +/* char: 0x58 */ + +static const SFG_StrokeVertex ch88st0[] = +{ + {2.5f,100.0f}, + {69.1667f,0.0f} +}; + +static const SFG_StrokeVertex ch88st1[] = +{ + {69.1667f,100.0f}, + {2.5f,0.0f} +}; + +static const SFG_StrokeStrip ch88st[] = +{ + {2,ch88st0}, + {2,ch88st1} +}; + +static const SFG_StrokeChar ch88 = {72.3667f,2,ch88st}; + +/* char: 0x59 */ + +static const SFG_StrokeVertex ch89st0[] = +{ + {1.52f,100.0f}, + {39.6152f,52.381f}, + {39.6152f,0.0f} +}; + +static const SFG_StrokeVertex ch89st1[] = +{ + {77.7105f,100.0f}, + {39.6152f,52.381f} +}; + +static const SFG_StrokeStrip ch89st[] = +{ + {3,ch89st0}, + {2,ch89st1} +}; + +static const SFG_StrokeChar ch89 = {79.6505f,2,ch89st}; + +/* char: 0x5a */ + +static const SFG_StrokeVertex ch90st0[] = +{ + {69.1667f,100.0f}, + {2.5f,0.0f} +}; + +static const SFG_StrokeVertex ch90st1[] = +{ + {2.5f,100.0f}, + {69.1667f,100.0f} +}; + +static const SFG_StrokeVertex ch90st2[] = +{ + {2.5f,0.0f}, + {69.1667f,0.0f} +}; + +static const SFG_StrokeStrip ch90st[] = +{ + {2,ch90st0}, + {2,ch90st1}, + {2,ch90st2} +}; + +static const SFG_StrokeChar ch90 = {73.7467f,3,ch90st}; + +/* char: 0x5b */ + +static const SFG_StrokeVertex ch91st0[] = +{ + {7.78f,119.048f}, + {7.78f,-33.3333f} +}; + +static const SFG_StrokeVertex ch91st1[] = +{ + {12.5419f,119.048f}, + {12.5419f,-33.3333f} +}; + +static const SFG_StrokeVertex ch91st2[] = +{ + {7.78f,119.048f}, + {41.1133f,119.048f} +}; + +static const SFG_StrokeVertex ch91st3[] = +{ + {7.78f,-33.3333f}, + {41.1133f,-33.3333f} +}; + +static const SFG_StrokeStrip ch91st[] = +{ + {2,ch91st0}, + {2,ch91st1}, + {2,ch91st2}, + {2,ch91st3} +}; + +static const SFG_StrokeChar ch91 = {46.1133f,4,ch91st}; + +/* char: 0x5c */ + +static const SFG_StrokeVertex ch92st0[] = +{ + {5.84f,100.0f}, + {72.5067f,-14.2857f} +}; + +static const SFG_StrokeStrip ch92st[] = +{ + {2,ch92st0} +}; + +static const SFG_StrokeChar ch92 = {78.2067f,1,ch92st}; + +/* char: 0x5d */ + +static const SFG_StrokeVertex ch93st0[] = +{ + {33.0114f,119.048f}, + {33.0114f,-33.3333f} +}; + +static const SFG_StrokeVertex ch93st1[] = +{ + {37.7733f,119.048f}, + {37.7733f,-33.3333f} +}; + +static const SFG_StrokeVertex ch93st2[] = +{ + {4.44f,119.048f}, + {37.7733f,119.048f} +}; + +static const SFG_StrokeVertex ch93st3[] = +{ + {4.44f,-33.3333f}, + {37.7733f,-33.3333f} +}; + +static const SFG_StrokeStrip ch93st[] = +{ + {2,ch93st0}, + {2,ch93st1}, + {2,ch93st2}, + {2,ch93st3} +}; + +static const SFG_StrokeChar ch93 = {46.3933f,4,ch93st}; + +/* char: 0x5e */ + +static const SFG_StrokeVertex ch94st0[] = +{ + {44.0752f,109.524f}, + {5.98f,42.8571f} +}; + +static const SFG_StrokeVertex ch94st1[] = +{ + {44.0752f,109.524f}, + {82.1705f,42.8571f} +}; + +static const SFG_StrokeStrip ch94st[] = +{ + {2,ch94st0}, + {2,ch94st1} +}; + +static const SFG_StrokeChar ch94 = {90.2305f,2,ch94st}; + +/* char: 0x5f */ + +static const SFG_StrokeVertex ch95st0[] = +{ + {-1.1f,-33.3333f}, + {103.662f,-33.3333f}, + {103.662f,-28.5714f}, + {-1.1f,-28.5714f}, + {-1.1f,-33.3333f} +}; + +static const SFG_StrokeStrip ch95st[] = +{ + {5,ch95st0} +}; + +static const SFG_StrokeChar ch95 = {104.062f,1,ch95st}; + +/* char: 0x60 */ + +static const SFG_StrokeVertex ch96st0[] = +{ + {33.0219f,100.0f}, + {56.8314f,71.4286f} +}; + +static const SFG_StrokeVertex ch96st1[] = +{ + {33.0219f,100.0f}, + {28.26f,95.2381f}, + {56.8314f,71.4286f} +}; + +static const SFG_StrokeStrip ch96st[] = +{ + {2,ch96st0}, + {3,ch96st1} +}; + +static const SFG_StrokeChar ch96 = {83.5714f,2,ch96st}; + +/* char: 0x61 */ + +static const SFG_StrokeVertex ch97st0[] = +{ + {63.8229f,66.6667f}, + {63.8229f,0.0f} +}; + +static const SFG_StrokeVertex ch97st1[] = +{ + {63.8229f,52.381f}, + {54.299f,61.9048f}, + {44.7752f,66.6667f}, + {30.4895f,66.6667f}, + {20.9657f,61.9048f}, + {11.4419f,52.381f}, + {6.68f,38.0952f}, + {6.68f,28.5714f}, + {11.4419f,14.2857f}, + {20.9657f,4.7619f}, + {30.4895f,0.0f}, + {44.7752f,0.0f}, + {54.299f,4.7619f}, + {63.8229f,14.2857f} +}; + +static const SFG_StrokeStrip ch97st[] = +{ + {2,ch97st0}, + {14,ch97st1} +}; + +static const SFG_StrokeChar ch97 = {66.6029f,2,ch97st}; + +/* char: 0x62 */ + +static const SFG_StrokeVertex ch98st0[] = +{ + {8.76f,100.0f}, + {8.76f,0.0f} +}; + +static const SFG_StrokeVertex ch98st1[] = +{ + {8.76f,52.381f}, + {18.2838f,61.9048f}, + {27.8076f,66.6667f}, + {42.0933f,66.6667f}, + {51.6171f,61.9048f}, + {61.141f,52.381f}, + {65.9029f,38.0952f}, + {65.9029f,28.5714f}, + {61.141f,14.2857f}, + {51.6171f,4.7619f}, + {42.0933f,0.0f}, + {27.8076f,0.0f}, + {18.2838f,4.7619f}, + {8.76f,14.2857f} +}; + +static const SFG_StrokeStrip ch98st[] = +{ + {2,ch98st0}, + {14,ch98st1} +}; + +static const SFG_StrokeChar ch98 = {70.4629f,2,ch98st}; + +/* char: 0x63 */ + +static const SFG_StrokeVertex ch99st0[] = +{ + {62.6629f,52.381f}, + {53.139f,61.9048f}, + {43.6152f,66.6667f}, + {29.3295f,66.6667f}, + {19.8057f,61.9048f}, + {10.2819f,52.381f}, + {5.52f,38.0952f}, + {5.52f,28.5714f}, + {10.2819f,14.2857f}, + {19.8057f,4.7619f}, + {29.3295f,0.0f}, + {43.6152f,0.0f}, + {53.139f,4.7619f}, + {62.6629f,14.2857f} +}; + +static const SFG_StrokeStrip ch99st[] = +{ + {14,ch99st0} +}; + +static const SFG_StrokeChar ch99 = {68.9229f,1,ch99st}; + +/* char: 0x64 */ + +static const SFG_StrokeVertex ch100st0[] = +{ + {61.7829f,100.0f}, + {61.7829f,0.0f} +}; + +static const SFG_StrokeVertex ch100st1[] = +{ + {61.7829f,52.381f}, + {52.259f,61.9048f}, + {42.7352f,66.6667f}, + {28.4495f,66.6667f}, + {18.9257f,61.9048f}, + {9.4019f,52.381f}, + {4.64f,38.0952f}, + {4.64f,28.5714f}, + {9.4019f,14.2857f}, + {18.9257f,4.7619f}, + {28.4495f,0.0f}, + {42.7352f,0.0f}, + {52.259f,4.7619f}, + {61.7829f,14.2857f} +}; + +static const SFG_StrokeStrip ch100st[] = +{ + {2,ch100st0}, + {14,ch100st1} +}; + +static const SFG_StrokeChar ch100 = {70.2629f,2,ch100st}; + +/* char: 0x65 */ + +static const SFG_StrokeVertex ch101st0[] = +{ + {5.72f,38.0952f}, + {62.8629f,38.0952f}, + {62.8629f,47.619f}, + {58.101f,57.1429f}, + {53.339f,61.9048f}, + {43.8152f,66.6667f}, + {29.5295f,66.6667f}, + {20.0057f,61.9048f}, + {10.4819f,52.381f}, + {5.72f,38.0952f}, + {5.72f,28.5714f}, + {10.4819f,14.2857f}, + {20.0057f,4.7619f}, + {29.5295f,0.0f}, + {43.8152f,0.0f}, + {53.339f,4.7619f}, + {62.8629f,14.2857f} +}; + +static const SFG_StrokeStrip ch101st[] = +{ + {17,ch101st0} +}; + +static const SFG_StrokeChar ch101 = {68.5229f,1,ch101st}; + +/* char: 0x66 */ + +static const SFG_StrokeVertex ch102st0[] = +{ + {38.7752f,100.0f}, + {29.2514f,100.0f}, + {19.7276f,95.2381f}, + {14.9657f,80.9524f}, + {14.9657f,0.0f} +}; + +static const SFG_StrokeVertex ch102st1[] = +{ + {0.68f,66.6667f}, + {34.0133f,66.6667f} +}; + +static const SFG_StrokeStrip ch102st[] = +{ + {5,ch102st0}, + {2,ch102st1} +}; + +static const SFG_StrokeChar ch102 = {38.6552f,2,ch102st}; + +/* char: 0x67 */ + +static const SFG_StrokeVertex ch103st0[] = +{ + {62.5029f,66.6667f}, + {62.5029f,-9.5238f}, + {57.741f,-23.8095f}, + {52.979f,-28.5714f}, + {43.4552f,-33.3333f}, + {29.1695f,-33.3333f}, + {19.6457f,-28.5714f} +}; + +static const SFG_StrokeVertex ch103st1[] = +{ + {62.5029f,52.381f}, + {52.979f,61.9048f}, + {43.4552f,66.6667f}, + {29.1695f,66.6667f}, + {19.6457f,61.9048f}, + {10.1219f,52.381f}, + {5.36f,38.0952f}, + {5.36f,28.5714f}, + {10.1219f,14.2857f}, + {19.6457f,4.7619f}, + {29.1695f,0.0f}, + {43.4552f,0.0f}, + {52.979f,4.7619f}, + {62.5029f,14.2857f} +}; + +static const SFG_StrokeStrip ch103st[] = +{ + {7,ch103st0}, + {14,ch103st1} +}; + +static const SFG_StrokeChar ch103 = {70.9829f,2,ch103st}; + +/* char: 0x68 */ + +static const SFG_StrokeVertex ch104st0[] = +{ + {9.6f,100.0f}, + {9.6f,0.0f} +}; + +static const SFG_StrokeVertex ch104st1[] = +{ + {9.6f,47.619f}, + {23.8857f,61.9048f}, + {33.4095f,66.6667f}, + {47.6952f,66.6667f}, + {57.219f,61.9048f}, + {61.981f,47.619f}, + {61.981f,0.0f} +}; + +static const SFG_StrokeStrip ch104st[] = +{ + {2,ch104st0}, + {7,ch104st1} +}; + +static const SFG_StrokeChar ch104 = {71.021f,2,ch104st}; + +/* char: 0x69 */ + +static const SFG_StrokeVertex ch105st0[] = +{ + {10.02f,100.0f}, + {14.7819f,95.2381f}, + {19.5438f,100.0f}, + {14.7819f,104.762f}, + {10.02f,100.0f} +}; + +static const SFG_StrokeVertex ch105st1[] = +{ + {14.7819f,66.6667f}, + {14.7819f,0.0f} +}; + +static const SFG_StrokeStrip ch105st[] = +{ + {5,ch105st0}, + {2,ch105st1} +}; + +static const SFG_StrokeChar ch105 = {28.8638f,2,ch105st}; + +/* char: 0x6a */ + +static const SFG_StrokeVertex ch106st0[] = +{ + {17.3876f,100.0f}, + {22.1495f,95.2381f}, + {26.9114f,100.0f}, + {22.1495f,104.762f}, + {17.3876f,100.0f} +}; + +static const SFG_StrokeVertex ch106st1[] = +{ + {22.1495f,66.6667f}, + {22.1495f,-14.2857f}, + {17.3876f,-28.5714f}, + {7.8638f,-33.3333f}, + {-1.66f,-33.3333f} +}; + +static const SFG_StrokeStrip ch106st[] = +{ + {5,ch106st0}, + {5,ch106st1} +}; + +static const SFG_StrokeChar ch106 = {36.2314f,2,ch106st}; + +/* char: 0x6b */ + +static const SFG_StrokeVertex ch107st0[] = +{ + {9.6f,100.0f}, + {9.6f,0.0f} +}; + +static const SFG_StrokeVertex ch107st1[] = +{ + {57.219f,66.6667f}, + {9.6f,19.0476f} +}; + +static const SFG_StrokeVertex ch107st2[] = +{ + {28.6476f,38.0952f}, + {61.981f,0.0f} +}; + +static const SFG_StrokeStrip ch107st[] = +{ + {2,ch107st0}, + {2,ch107st1}, + {2,ch107st2} +}; + +static const SFG_StrokeChar ch107 = {62.521f,3,ch107st}; + +/* char: 0x6c */ + +static const SFG_StrokeVertex ch108st0[] = +{ + {10.02f,100.0f}, + {10.02f,0.0f} +}; + +static const SFG_StrokeStrip ch108st[] = +{ + {2,ch108st0} +}; + +static const SFG_StrokeChar ch108 = {19.34f,1,ch108st}; + +/* char: 0x6d */ + +static const SFG_StrokeVertex ch109st0[] = +{ + {9.6f,66.6667f}, + {9.6f,0.0f} +}; + +static const SFG_StrokeVertex ch109st1[] = +{ + {9.6f,47.619f}, + {23.8857f,61.9048f}, + {33.4095f,66.6667f}, + {47.6952f,66.6667f}, + {57.219f,61.9048f}, + {61.981f,47.619f}, + {61.981f,0.0f} +}; + +static const SFG_StrokeVertex ch109st2[] = +{ + {61.981f,47.619f}, + {76.2667f,61.9048f}, + {85.7905f,66.6667f}, + {100.076f,66.6667f}, + {109.6f,61.9048f}, + {114.362f,47.619f}, + {114.362f,0.0f} +}; + +static const SFG_StrokeStrip ch109st[] = +{ + {2,ch109st0}, + {7,ch109st1}, + {7,ch109st2} +}; + +static const SFG_StrokeChar ch109 = {123.962f,3,ch109st}; + +/* char: 0x6e */ + +static const SFG_StrokeVertex ch110st0[] = +{ + {9.18f,66.6667f}, + {9.18f,0.0f} +}; + +static const SFG_StrokeVertex ch110st1[] = +{ + {9.18f,47.619f}, + {23.4657f,61.9048f}, + {32.9895f,66.6667f}, + {47.2752f,66.6667f}, + {56.799f,61.9048f}, + {61.561f,47.619f}, + {61.561f,0.0f} +}; + +static const SFG_StrokeStrip ch110st[] = +{ + {2,ch110st0}, + {7,ch110st1} +}; + +static const SFG_StrokeChar ch110 = {70.881f,2,ch110st}; + +/* char: 0x6f */ + +static const SFG_StrokeVertex ch111st0[] = +{ + {28.7895f,66.6667f}, + {19.2657f,61.9048f}, + {9.7419f,52.381f}, + {4.98f,38.0952f}, + {4.98f,28.5714f}, + {9.7419f,14.2857f}, + {19.2657f,4.7619f}, + {28.7895f,0.0f}, + {43.0752f,0.0f}, + {52.599f,4.7619f}, + {62.1229f,14.2857f}, + {66.8848f,28.5714f}, + {66.8848f,38.0952f}, + {62.1229f,52.381f}, + {52.599f,61.9048f}, + {43.0752f,66.6667f}, + {28.7895f,66.6667f} +}; + +static const SFG_StrokeStrip ch111st[] = +{ + {17,ch111st0} +}; + +static const SFG_StrokeChar ch111 = {71.7448f,1,ch111st}; + +/* char: 0x70 */ + +static const SFG_StrokeVertex ch112st0[] = +{ + {9.46f,66.6667f}, + {9.46f,-33.3333f} +}; + +static const SFG_StrokeVertex ch112st1[] = +{ + {9.46f,52.381f}, + {18.9838f,61.9048f}, + {28.5076f,66.6667f}, + {42.7933f,66.6667f}, + {52.3171f,61.9048f}, + {61.841f,52.381f}, + {66.6029f,38.0952f}, + {66.6029f,28.5714f}, + {61.841f,14.2857f}, + {52.3171f,4.7619f}, + {42.7933f,0.0f}, + {28.5076f,0.0f}, + {18.9838f,4.7619f}, + {9.46f,14.2857f} +}; + +static const SFG_StrokeStrip ch112st[] = +{ + {2,ch112st0}, + {14,ch112st1} +}; + +static const SFG_StrokeChar ch112 = {70.8029f,2,ch112st}; + +/* char: 0x71 */ + +static const SFG_StrokeVertex ch113st0[] = +{ + {61.9829f,66.6667f}, + {61.9829f,-33.3333f} +}; + +static const SFG_StrokeVertex ch113st1[] = +{ + {61.9829f,52.381f}, + {52.459f,61.9048f}, + {42.9352f,66.6667f}, + {28.6495f,66.6667f}, + {19.1257f,61.9048f}, + {9.6019f,52.381f}, + {4.84f,38.0952f}, + {4.84f,28.5714f}, + {9.6019f,14.2857f}, + {19.1257f,4.7619f}, + {28.6495f,0.0f}, + {42.9352f,0.0f}, + {52.459f,4.7619f}, + {61.9829f,14.2857f} +}; + +static const SFG_StrokeStrip ch113st[] = +{ + {2,ch113st0}, + {14,ch113st1} +}; + +static const SFG_StrokeChar ch113 = {70.7429f,2,ch113st}; + +/* char: 0x72 */ + +static const SFG_StrokeVertex ch114st0[] = +{ + {9.46f,66.6667f}, + {9.46f,0.0f} +}; + +static const SFG_StrokeVertex ch114st1[] = +{ + {9.46f,38.0952f}, + {14.2219f,52.381f}, + {23.7457f,61.9048f}, + {33.2695f,66.6667f}, + {47.5552f,66.6667f} +}; + +static const SFG_StrokeStrip ch114st[] = +{ + {2,ch114st0}, + {5,ch114st1} +}; + +static const SFG_StrokeChar ch114 = {49.4952f,2,ch114st}; + +/* char: 0x73 */ + +static const SFG_StrokeVertex ch115st0[] = +{ + {57.081f,52.381f}, + {52.319f,61.9048f}, + {38.0333f,66.6667f}, + {23.7476f,66.6667f}, + {9.4619f,61.9048f}, + {4.7f,52.381f}, + {9.4619f,42.8571f}, + {18.9857f,38.0952f}, + {42.7952f,33.3333f}, + {52.319f,28.5714f}, + {57.081f,19.0476f}, + {57.081f,14.2857f}, + {52.319f,4.7619f}, + {38.0333f,0.0f}, + {23.7476f,0.0f}, + {9.4619f,4.7619f}, + {4.7f,14.2857f} +}; + +static const SFG_StrokeStrip ch115st[] = +{ + {17,ch115st0} +}; + +static const SFG_StrokeChar ch115 = {62.321f,1,ch115st}; + +/* char: 0x74 */ + +static const SFG_StrokeVertex ch116st0[] = +{ + {14.8257f,100.0f}, + {14.8257f,19.0476f}, + {19.5876f,4.7619f}, + {29.1114f,0.0f}, + {38.6352f,0.0f} +}; + +static const SFG_StrokeVertex ch116st1[] = +{ + {0.54f,66.6667f}, + {33.8733f,66.6667f} +}; + +static const SFG_StrokeStrip ch116st[] = +{ + {5,ch116st0}, + {2,ch116st1} +}; + +static const SFG_StrokeChar ch116 = {39.3152f,2,ch116st}; + +/* char: 0x75 */ + +static const SFG_StrokeVertex ch117st0[] = +{ + {9.46f,66.6667f}, + {9.46f,19.0476f}, + {14.2219f,4.7619f}, + {23.7457f,0.0f}, + {38.0314f,0.0f}, + {47.5552f,4.7619f}, + {61.841f,19.0476f} +}; + +static const SFG_StrokeVertex ch117st1[] = +{ + {61.841f,66.6667f}, + {61.841f,0.0f} +}; + +static const SFG_StrokeStrip ch117st[] = +{ + {7,ch117st0}, + {2,ch117st1} +}; + +static const SFG_StrokeChar ch117 = {71.161f,2,ch117st}; + +/* char: 0x76 */ + +static const SFG_StrokeVertex ch118st0[] = +{ + {1.8f,66.6667f}, + {30.3714f,0.0f} +}; + +static const SFG_StrokeVertex ch118st1[] = +{ + {58.9429f,66.6667f}, + {30.3714f,0.0f} +}; + +static const SFG_StrokeStrip ch118st[] = +{ + {2,ch118st0}, + {2,ch118st1} +}; + +static const SFG_StrokeChar ch118 = {60.6029f,2,ch118st}; + +/* char: 0x77 */ + +static const SFG_StrokeVertex ch119st0[] = +{ + {2.5f,66.6667f}, + {21.5476f,0.0f} +}; + +static const SFG_StrokeVertex ch119st1[] = +{ + {40.5952f,66.6667f}, + {21.5476f,0.0f} +}; + +static const SFG_StrokeVertex ch119st2[] = +{ + {40.5952f,66.6667f}, + {59.6429f,0.0f} +}; + +static const SFG_StrokeVertex ch119st3[] = +{ + {78.6905f,66.6667f}, + {59.6429f,0.0f} +}; + +static const SFG_StrokeStrip ch119st[] = +{ + {2,ch119st0}, + {2,ch119st1}, + {2,ch119st2}, + {2,ch119st3} +}; + +static const SFG_StrokeChar ch119 = {80.4905f,4,ch119st}; + +/* char: 0x78 */ + +static const SFG_StrokeVertex ch120st0[] = +{ + {1.66f,66.6667f}, + {54.041f,0.0f} +}; + +static const SFG_StrokeVertex ch120st1[] = +{ + {54.041f,66.6667f}, + {1.66f,0.0f} +}; + +static const SFG_StrokeStrip ch120st[] = +{ + {2,ch120st0}, + {2,ch120st1} +}; + +static const SFG_StrokeChar ch120 = {56.401f,2,ch120st}; + +/* char: 0x79 */ + +static const SFG_StrokeVertex ch121st0[] = +{ + {6.5619f,66.6667f}, + {35.1333f,0.0f} +}; + +static const SFG_StrokeVertex ch121st1[] = +{ + {63.7048f,66.6667f}, + {35.1333f,0.0f}, + {25.6095f,-19.0476f}, + {16.0857f,-28.5714f}, + {6.5619f,-33.3333f}, + {1.8f,-33.3333f} +}; + +static const SFG_StrokeStrip ch121st[] = +{ + {2,ch121st0}, + {6,ch121st1} +}; + +static const SFG_StrokeChar ch121 = {66.0648f,2,ch121st}; + +/* char: 0x7a */ + +static const SFG_StrokeVertex ch122st0[] = +{ + {56.821f,66.6667f}, + {4.44f,0.0f} +}; + +static const SFG_StrokeVertex ch122st1[] = +{ + {4.44f,66.6667f}, + {56.821f,66.6667f} +}; + +static const SFG_StrokeVertex ch122st2[] = +{ + {4.44f,0.0f}, + {56.821f,0.0f} +}; + +static const SFG_StrokeStrip ch122st[] = +{ + {2,ch122st0}, + {2,ch122st1}, + {2,ch122st2} +}; + +static const SFG_StrokeChar ch122 = {61.821f,3,ch122st}; + +/* char: 0x7b */ + +static const SFG_StrokeVertex ch123st0[] = +{ + {31.1895f,119.048f}, + {21.6657f,114.286f}, + {16.9038f,109.524f}, + {12.1419f,100.0f}, + {12.1419f,90.4762f}, + {16.9038f,80.9524f}, + {21.6657f,76.1905f}, + {26.4276f,66.6667f}, + {26.4276f,57.1429f}, + {16.9038f,47.619f} +}; + +static const SFG_StrokeVertex ch123st1[] = +{ + {21.6657f,114.286f}, + {16.9038f,104.762f}, + {16.9038f,95.2381f}, + {21.6657f,85.7143f}, + {26.4276f,80.9524f}, + {31.1895f,71.4286f}, + {31.1895f,61.9048f}, + {26.4276f,52.381f}, + {7.38f,42.8571f}, + {26.4276f,33.3333f}, + {31.1895f,23.8095f}, + {31.1895f,14.2857f}, + {26.4276f,4.7619f}, + {21.6657f,0.0f}, + {16.9038f,-9.5238f}, + {16.9038f,-19.0476f}, + {21.6657f,-28.5714f} +}; + +static const SFG_StrokeVertex ch123st2[] = +{ + {16.9038f,38.0952f}, + {26.4276f,28.5714f}, + {26.4276f,19.0476f}, + {21.6657f,9.5238f}, + {16.9038f,4.7619f}, + {12.1419f,-4.7619f}, + {12.1419f,-14.2857f}, + {16.9038f,-23.8095f}, + {21.6657f,-28.5714f}, + {31.1895f,-33.3333f} +}; + +static const SFG_StrokeStrip ch123st[] = +{ + {10,ch123st0}, + {17,ch123st1}, + {10,ch123st2} +}; + +static const SFG_StrokeChar ch123 = {41.6295f,3,ch123st}; + +/* char: 0x7c */ + +static const SFG_StrokeVertex ch124st0[] = +{ + {11.54f,119.048f}, + {11.54f,-33.3333f} +}; + +static const SFG_StrokeStrip ch124st[] = +{ + {2,ch124st0} +}; + +static const SFG_StrokeChar ch124 = {23.78f,1,ch124st}; + +/* char: 0x7d */ + +static const SFG_StrokeVertex ch125st0[] = +{ + {9.18f,119.048f}, + {18.7038f,114.286f}, + {23.4657f,109.524f}, + {28.2276f,100.0f}, + {28.2276f,90.4762f}, + {23.4657f,80.9524f}, + {18.7038f,76.1905f}, + {13.9419f,66.6667f}, + {13.9419f,57.1429f}, + {23.4657f,47.619f} +}; + +static const SFG_StrokeVertex ch125st1[] = +{ + {18.7038f,114.286f}, + {23.4657f,104.762f}, + {23.4657f,95.2381f}, + {18.7038f,85.7143f}, + {13.9419f,80.9524f}, + {9.18f,71.4286f}, + {9.18f,61.9048f}, + {13.9419f,52.381f}, + {32.9895f,42.8571f}, + {13.9419f,33.3333f}, + {9.18f,23.8095f}, + {9.18f,14.2857f}, + {13.9419f,4.7619f}, + {18.7038f,0.0f}, + {23.4657f,-9.5238f}, + {23.4657f,-19.0476f}, + {18.7038f,-28.5714f} +}; + +static const SFG_StrokeVertex ch125st2[] = +{ + {23.4657f,38.0952f}, + {13.9419f,28.5714f}, + {13.9419f,19.0476f}, + {18.7038f,9.5238f}, + {23.4657f,4.7619f}, + {28.2276f,-4.7619f}, + {28.2276f,-14.2857f}, + {23.4657f,-23.8095f}, + {18.7038f,-28.5714f}, + {9.18f,-33.3333f} +}; + +static const SFG_StrokeStrip ch125st[] = +{ + {10,ch125st0}, + {17,ch125st1}, + {10,ch125st2} +}; + +static const SFG_StrokeChar ch125 = {41.4695f,3,ch125st}; + +/* char: 0x7e */ + +static const SFG_StrokeVertex ch126st0[] = +{ + {2.92f,28.5714f}, + {2.92f,38.0952f}, + {7.6819f,52.381f}, + {17.2057f,57.1429f}, + {26.7295f,57.1429f}, + {36.2533f,52.381f}, + {55.301f,38.0952f}, + {64.8248f,33.3333f}, + {74.3486f,33.3333f}, + {83.8724f,38.0952f}, + {88.6343f,47.619f} +}; + +static const SFG_StrokeVertex ch126st1[] = +{ + {2.92f,38.0952f}, + {7.6819f,47.619f}, + {17.2057f,52.381f}, + {26.7295f,52.381f}, + {36.2533f,47.619f}, + {55.301f,33.3333f}, + {64.8248f,28.5714f}, + {74.3486f,28.5714f}, + {83.8724f,33.3333f}, + {88.6343f,47.619f}, + {88.6343f,57.1429f} +}; + +static const SFG_StrokeStrip ch126st[] = +{ + {11,ch126st0}, + {11,ch126st1} +}; + +static const SFG_StrokeChar ch126 = {91.2743f,2,ch126st}; + +/* char: 0x7f */ + +static const SFG_StrokeVertex ch127st0[] = +{ + {52.381f,100.0f}, + {14.2857f,-33.3333f} +}; + +static const SFG_StrokeVertex ch127st1[] = +{ + {28.5714f,66.6667f}, + {14.2857f,61.9048f}, + {4.7619f,52.381f}, + {0.0f,38.0952f}, + {0.0f,23.8095f}, + {4.7619f,14.2857f}, + {14.2857f,4.7619f}, + {28.5714f,0.0f}, + {38.0952f,0.0f}, + {52.381f,4.7619f}, + {61.9048f,14.2857f}, + {66.6667f,28.5714f}, + {66.6667f,42.8571f}, + {61.9048f,52.381f}, + {52.381f,61.9048f}, + {38.0952f,66.6667f}, + {28.5714f,66.6667f} +}; + +static const SFG_StrokeStrip ch127st[] = +{ + {2,ch127st0}, + {17,ch127st1} +}; + +static const SFG_StrokeChar ch127 = {66.6667f,2,ch127st}; + +static const SFG_StrokeChar *chars[] = +{ + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + &ch32, &ch33, &ch34, &ch35, &ch36, &ch37, &ch38, &ch39, + &ch40, &ch41, &ch42, &ch43, &ch44, &ch45, &ch46, &ch47, + &ch48, &ch49, &ch50, &ch51, &ch52, &ch53, &ch54, &ch55, + &ch56, &ch57, &ch58, &ch59, &ch60, &ch61, &ch62, &ch63, + &ch64, &ch65, &ch66, &ch67, &ch68, &ch69, &ch70, &ch71, + &ch72, &ch73, &ch74, &ch75, &ch76, &ch77, &ch78, &ch79, + &ch80, &ch81, &ch82, &ch83, &ch84, &ch85, &ch86, &ch87, + &ch88, &ch89, &ch90, &ch91, &ch92, &ch93, &ch94, &ch95, + &ch96, &ch97, &ch98, &ch99, &ch100, &ch101, &ch102, &ch103, + &ch104, &ch105, &ch106, &ch107, &ch108, &ch109, &ch110, &ch111, + &ch112, &ch113, &ch114, &ch115, &ch116, &ch117, &ch118, &ch119, + &ch120, &ch121, &ch122, &ch123, &ch124, &ch125, &ch126, &ch127 +}; + +const SFG_StrokeFont fgStrokeRoman = {"Roman",128,152.381f,chars}; diff --git a/examples/opengl-framework/freeglut/freeglut_structure.c b/examples/opengl-framework/freeglut/freeglut_structure.c new file mode 100644 index 00000000..c5a2d734 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_structure.c @@ -0,0 +1,598 @@ +/* + * freeglut_structure.c + * + * Windows and menus need tree structure + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Sat Dec 18 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* -- GLOBAL EXPORTS ------------------------------------------------------- */ + +/* + * The SFG_Structure container holds information about windows and menus + * created between glutInit() and glutMainLoop() return. + */ + +SFG_Structure fgStructure = { { NULL, NULL }, /* The list of windows */ + { NULL, NULL }, /* The list of menus */ + { NULL, NULL }, /* Windows to Destroy list */ + NULL, /* The current window */ + NULL, /* The current menu */ + NULL, /* The menu OpenGL context */ + NULL, /* The game mode window */ + 0, /* The current new window ID */ + 0 }; /* The current new menu ID */ + + +/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ + +static void fghClearCallBacks( SFG_Window *window ) +{ + if( window ) + { + int i; + for( i = 0; i < TOTAL_CALLBACKS; ++i ) + window->CallBacks[ i ] = NULL; + } +} + +/* + * This private function creates, opens and adds to the hierarchy + * a freeglut window complete with OpenGL context and stuff... + * + * If parent is set to NULL, the window created will be a topmost one. + */ +SFG_Window* fgCreateWindow( SFG_Window* parent, const char* title, + GLboolean positionUse, int x, int y, + GLboolean sizeUse, int w, int h, + GLboolean gameMode, GLboolean isMenu ) +{ + /* Have the window object created */ + SFG_Window *window = (SFG_Window *)calloc( sizeof(SFG_Window), 1 ); + +#if TARGET_HOST_UNIX_X11 + window->Window.FBConfig = NULL; +#endif + fghClearCallBacks( window ); + + /* Initialize the object properties */ + window->ID = ++fgStructure.WindowID; +#if TARGET_HOST_POSIX_X11 + window->State.OldHeight = window->State.OldWidth = -1; +#endif + + fgListInit( &window->Children ); + if( parent ) + { + fgListAppend( &parent->Children, &window->Node ); + window->Parent = parent; + } + else + fgListAppend( &fgStructure.Windows, &window->Node ); + + /* Set the default mouse cursor and reset the modifiers value */ + window->State.Cursor = GLUT_CURSOR_INHERIT; + + window->IsMenu = isMenu; + + window->State.IgnoreKeyRepeat = GL_FALSE; + window->State.KeyRepeating = GL_FALSE; + window->State.IsFullscreen = GL_FALSE; + + /* + * Open the window now. The fgOpenWindow() function is system + * dependant, and resides in freeglut_window.c. Uses fgState. + */ + fgOpenWindow( window, title, positionUse, x, y, sizeUse, w, h, gameMode, + (GLboolean)(parent ? GL_TRUE : GL_FALSE) ); + + return window; +} + +/* + * This private function creates a menu and adds it to the menus list + */ +SFG_Menu* fgCreateMenu( FGCBMenu menuCallback ) +{ + int x = 100, y = 100, w = 1, h = 1; + SFG_Window *current_window = fgStructure.CurrentWindow; + + /* Have the menu object created */ + SFG_Menu* menu = (SFG_Menu *)calloc( sizeof(SFG_Menu), 1 ); + + menu->ParentWindow = NULL; + + /* Create a window for the menu to reside in. */ + + fgCreateWindow( NULL, "freeglut menu", GL_TRUE, x, y, GL_TRUE, w, h, + GL_FALSE, GL_TRUE ); + menu->Window = fgStructure.CurrentWindow; + glutDisplayFunc( fgDisplayMenu ); + + glutHideWindow( ); /* Hide the window for now */ + fgSetWindow( current_window ); + + /* Initialize the object properties: */ + menu->ID = ++fgStructure.MenuID; + menu->Callback = menuCallback; + menu->ActiveEntry = NULL; + + fgListInit( &menu->Entries ); + fgListAppend( &fgStructure.Menus, &menu->Node ); + + /* Newly created menus implicitly become current ones */ + fgStructure.CurrentMenu = menu; + + return menu; +} + +/* + * Function to add a window to the linked list of windows to destroy. + * Subwindows are automatically added because they hang from the window + * structure. + */ +void fgAddToWindowDestroyList( SFG_Window* window ) +{ + SFG_WindowList *new_list_entry = + ( SFG_WindowList* )malloc( sizeof(SFG_WindowList ) ); + new_list_entry->window = window; + fgListAppend( &fgStructure.WindowsToDestroy, &new_list_entry->node ); + + /* Check if the window is the current one... */ + if( fgStructure.CurrentWindow == window ) + fgStructure.CurrentWindow = NULL; + + /* + * Clear all window callbacks except Destroy, which will + * be invoked later. Right now, we are potentially carrying + * out a freeglut operation at the behest of a client callback, + * so we are reluctant to re-enter the client with the Destroy + * callback, right now. The others are all wiped out, however, + * to ensure that they are no longer called after this point. + */ + { + FGCBDestroy destroy = (FGCBDestroy)FETCH_WCB( *window, Destroy ); + fghClearCallBacks( window ); + SET_WCB( *window, Destroy, destroy ); + } +} + +/* + * Function to close down all the windows in the "WindowsToDestroy" list + */ +void fgCloseWindows( ) +{ + while( fgStructure.WindowsToDestroy.First ) + { + SFG_WindowList *window_ptr = fgStructure.WindowsToDestroy.First; + fgDestroyWindow( window_ptr->window ); + fgListRemove( &fgStructure.WindowsToDestroy, &window_ptr->node ); + free( window_ptr ); + } +} + +/* + * This function destroys a window and all of its subwindows. Actually, + * another function, defined in freeglut_window.c is called, but this is + * a whole different story... + */ +void fgDestroyWindow( SFG_Window* window ) +{ + FREEGLUT_INTERNAL_ERROR_EXIT ( window, "Window destroy function called with null window", + "fgDestroyWindow" ); + + while( window->Children.First ) + fgDestroyWindow( ( SFG_Window * )window->Children.First ); + + { + SFG_Window *activeWindow = fgStructure.CurrentWindow; + INVOKE_WCB( *window, Destroy, ( ) ); + fgSetWindow( activeWindow ); + } + + if( window->Parent ) + fgListRemove( &window->Parent->Children, &window->Node ); + else + fgListRemove( &fgStructure.Windows, &window->Node ); + + if( window->ActiveMenu ) + fgDeactivateMenu( window ); + + fghClearCallBacks( window ); + fgCloseWindow( window ); + free( window ); + if( fgStructure.CurrentWindow == window ) + fgStructure.CurrentWindow = NULL; +} + +/* + * This is a helper static function that removes a menu (given its pointer) + * from any windows that can be accessed from a given parent... + */ +static void fghRemoveMenuFromWindow( SFG_Window* window, SFG_Menu* menu ) +{ + SFG_Window *subWindow; + int i; + + /* Check whether this is the active menu in the window */ + if ( menu == window->ActiveMenu ) + window->ActiveMenu = NULL ; + + /* + * Check if the menu is attached to the current window, + * if so, have it detached (by overwriting with a NULL): + */ + for( i = 0; i < FREEGLUT_MAX_MENUS; i++ ) + if( window->Menu[ i ] == menu ) + window->Menu[ i ] = NULL; + + /* Call this function for all of the window's children recursively: */ + for( subWindow = (SFG_Window *)window->Children.First; + subWindow; + subWindow = (SFG_Window *)subWindow->Node.Next) + fghRemoveMenuFromWindow( subWindow, menu ); +} + +/* + * This is a static helper function that removes menu references + * from another menu, given two pointers to them... + */ +static void fghRemoveMenuFromMenu( SFG_Menu* from, SFG_Menu* menu ) +{ + SFG_MenuEntry *entry; + + for( entry = (SFG_MenuEntry *)from->Entries.First; + entry; + entry = ( SFG_MenuEntry * )entry->Node.Next ) + if( entry->SubMenu == menu ) + entry->SubMenu = NULL; +} + +/* + * This function destroys a menu specified by the parameter. All menus + * and windows are updated to make sure no ill pointers hang around. + */ +void fgDestroyMenu( SFG_Menu* menu ) +{ + SFG_Window *window; + SFG_Menu *from; + + FREEGLUT_INTERNAL_ERROR_EXIT ( menu, "Menu destroy function called with null menu", + "fgDestroyMenu" ); + + /* First of all, have all references to this menu removed from all windows: */ + for( window = (SFG_Window *)fgStructure.Windows.First; + window; + window = (SFG_Window *)window->Node.Next ) + fghRemoveMenuFromWindow( window, menu ); + + /* Now proceed with removing menu entries that lead to this menu */ + for( from = ( SFG_Menu * )fgStructure.Menus.First; + from; + from = ( SFG_Menu * )from->Node.Next ) + fghRemoveMenuFromMenu( from, menu ); + + /* + * If the programmer defined a destroy callback, call it + * A. Donev: But first make this the active menu + */ + if( menu->Destroy ) + { + SFG_Menu *activeMenu=fgStructure.CurrentMenu; + fgStructure.CurrentMenu = menu; + menu->Destroy( ); + fgStructure.CurrentMenu = activeMenu; + } + + /* + * Now we are pretty sure the menu is not used anywhere + * and that we can remove all of its entries + */ + while( menu->Entries.First ) + { + SFG_MenuEntry *entry = ( SFG_MenuEntry * ) menu->Entries.First; + + fgListRemove( &menu->Entries, &entry->Node ); + + if( entry->Text ) + free( entry->Text ); + entry->Text = NULL; + + free( entry ); + } + + if( fgStructure.CurrentWindow == menu->Window ) + fgSetWindow( NULL ); + fgDestroyWindow( menu->Window ); + fgListRemove( &fgStructure.Menus, &menu->Node ); + if( fgStructure.CurrentMenu == menu ) + fgStructure.CurrentMenu = NULL; + + free( menu ); +} + +/* + * This function should be called on glutInit(). It will prepare the internal + * structure of freeglut to be used in the application. The structure will be + * destroyed using fgDestroyStructure() on glutMainLoop() return. In that + * case further use of freeglut should be preceeded with a glutInit() call. + */ +void fgCreateStructure( void ) +{ + /* + * We will be needing two lists: the first containing windows, + * and the second containing the user-defined menus. + * Also, no current window/menu is set, as none has been created yet. + */ + + fgListInit(&fgStructure.Windows); + fgListInit(&fgStructure.Menus); + fgListInit(&fgStructure.WindowsToDestroy); + + fgStructure.CurrentWindow = NULL; + fgStructure.CurrentMenu = NULL; + fgStructure.MenuContext = NULL; + fgStructure.GameModeWindow = NULL; + fgStructure.WindowID = 0; + fgStructure.MenuID = 0; +} + +/* + * This function is automatically called on glutMainLoop() return. + * It should deallocate and destroy all remnants of previous + * glutInit()-enforced structure initialization... + */ +void fgDestroyStructure( void ) +{ + /* Clean up the WindowsToDestroy list. */ + fgCloseWindows( ); + + /* Make sure all windows and menus have been deallocated */ + while( fgStructure.Menus.First ) + fgDestroyMenu( ( SFG_Menu * )fgStructure.Menus.First ); + + while( fgStructure.Windows.First ) + fgDestroyWindow( ( SFG_Window * )fgStructure.Windows.First ); +} + +/* + * Helper function to enumerate through all registered top-level windows + */ +void fgEnumWindows( FGCBenumerator enumCallback, SFG_Enumerator* enumerator ) +{ + SFG_Window *window; + + FREEGLUT_INTERNAL_ERROR_EXIT ( enumCallback && enumerator, + "Enumerator or callback missing from window enumerator call", + "fgEnumWindows" ); + + /* Check every of the top-level windows */ + for( window = ( SFG_Window * )fgStructure.Windows.First; + window; + window = ( SFG_Window * )window->Node.Next ) + { + enumCallback( window, enumerator ); + if( enumerator->found ) + return; + } +} + +/* + * Helper function to enumerate through all a window's subwindows + * (single level descent) + */ +void fgEnumSubWindows( SFG_Window* window, FGCBenumerator enumCallback, + SFG_Enumerator* enumerator ) +{ + SFG_Window *child; + + FREEGLUT_INTERNAL_ERROR_EXIT ( enumCallback && enumerator, + "Enumerator or callback missing from subwindow enumerator call", + "fgEnumSubWindows" ); + FREEGLUT_INTERNAL_ERROR_EXIT_IF_NOT_INITIALISED ( "Window Enumeration" ); + + for( child = ( SFG_Window * )window->Children.First; + child; + child = ( SFG_Window * )child->Node.Next ) + { + enumCallback( child, enumerator ); + if( enumerator->found ) + return; + } +} + +/* + * A static helper function to look for a window given its handle + */ +static void fghcbWindowByHandle( SFG_Window *window, + SFG_Enumerator *enumerator ) +{ + if ( enumerator->found ) + return; + + /* Check the window's handle. Hope this works. Looks ugly. That's for sure. */ + if( window->Window.Handle == (SFG_WindowHandleType) (enumerator->data) ) + { + enumerator->found = GL_TRUE; + enumerator->data = window; + + return; + } + + /* Otherwise, check this window's children */ + fgEnumSubWindows( window, fghcbWindowByHandle, enumerator ); +} + +/* + * fgWindowByHandle returns a (SFG_Window *) value pointing to the + * first window in the queue matching the specified window handle. + * The function is defined in freeglut_structure.c file. + */ +SFG_Window* fgWindowByHandle ( SFG_WindowHandleType hWindow ) +{ + SFG_Enumerator enumerator; + + /* This is easy and makes use of the windows enumeration defined above */ + enumerator.found = GL_FALSE; + enumerator.data = (void *)hWindow; + fgEnumWindows( fghcbWindowByHandle, &enumerator ); + + if( enumerator.found ) + return( SFG_Window *) enumerator.data; + return NULL; +} + +/* + * A static helper function to look for a window given its ID + */ +static void fghcbWindowByID( SFG_Window *window, SFG_Enumerator *enumerator ) +{ + /* Make sure we do not overwrite our precious results... */ + if( enumerator->found ) + return; + + /* Check the window's handle. Hope this works. Looks ugly. That's for sure. */ + if( window->ID == *( int *)(enumerator->data) ) + { + enumerator->found = GL_TRUE; + enumerator->data = window; + + return; + } + + /* Otherwise, check this window's children */ + fgEnumSubWindows( window, fghcbWindowByID, enumerator ); +} + +/* + * This function is similiar to the previous one, except it is + * looking for a specified (sub)window identifier. The function + * is defined in freeglut_structure.c file. + */ +SFG_Window* fgWindowByID( int windowID ) +{ + SFG_Enumerator enumerator; + + /* Uses a method very similiar for fgWindowByHandle... */ + enumerator.found = GL_FALSE; + enumerator.data = ( void * )&windowID; + fgEnumWindows( fghcbWindowByID, &enumerator ); + if( enumerator.found ) + return ( SFG_Window * )enumerator.data; + return NULL; +} + +/* + * Looks up a menu given its ID. This is easier that fgWindowByXXX + * as all menus are placed in one doubly linked list... + */ +SFG_Menu* fgMenuByID( int menuID ) +{ + SFG_Menu *menu = NULL; + + /* It's enough to check all entries in fgStructure.Menus... */ + for( menu = (SFG_Menu *)fgStructure.Menus.First; + menu; + menu = (SFG_Menu *)menu->Node.Next ) + if( menu->ID == menuID ) + return menu; + return NULL; +} + +/* + * List functions... + */ +void fgListInit(SFG_List *list) +{ + list->First = NULL; + list->Last = NULL; +} + +void fgListAppend(SFG_List *list, SFG_Node *node) +{ + if ( list->Last ) + { + SFG_Node *ln = (SFG_Node *) list->Last; + ln->Next = node; + node->Prev = ln; + } + else + { + node->Prev = NULL; + list->First = node; + } + + node->Next = NULL; + list->Last = node; +} + +void fgListRemove(SFG_List *list, SFG_Node *node) +{ + if( node->Next ) + ( ( SFG_Node * )node->Next )->Prev = node->Prev; + if( node->Prev ) + ( ( SFG_Node * )node->Prev )->Next = node->Next; + if( ( ( SFG_Node * )list->First ) == node ) + list->First = node->Next; + if( ( ( SFG_Node * )list->Last ) == node ) + list->Last = node->Prev; +} + +int fgListLength(SFG_List *list) +{ + SFG_Node *node; + int length = 0; + + for( node =( SFG_Node * )list->First; + node; + node = ( SFG_Node * )node->Next ) + ++length; + + return length; +} + + +void fgListInsert(SFG_List *list, SFG_Node *next, SFG_Node *node) +{ + SFG_Node *prev; + + if( (node->Next = next) ) + { + prev = next->Prev; + next->Prev = node; + } + else + { + prev = list->Last; + list->Last = node; + } + + if( (node->Prev = prev) ) + prev->Next = node; + else + list->First = node; +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_teapot.c b/examples/opengl-framework/freeglut/freeglut_teapot.c new file mode 100644 index 00000000..722bca61 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_teapot.c @@ -0,0 +1,200 @@ +/* + * freeglut_teapot.c + * + * Teapot(tm) rendering code. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Fri Dec 24 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +/* + * Original teapot code copyright follows: + */ + +/* + * (c) Copyright 1993, Silicon Graphics, Inc. + * + * ALL RIGHTS RESERVED + * + * Permission to use, copy, modify, and distribute this software + * for any purpose and without fee is hereby granted, provided + * that the above copyright notice appear in all copies and that + * both the copyright notice and this permission notice appear in + * supporting documentation, and that the name of Silicon + * Graphics, Inc. not be used in advertising or publicity + * pertaining to distribution of the software without specific, + * written prior permission. + * + * THE MATERIAL EMBODIED ON THIS SOFTWARE IS PROVIDED TO YOU + * "AS-IS" AND WITHOUT WARRANTY OF ANY KIND, EXPRESS, IMPLIED OR + * OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY WARRANTY OF + * MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. IN NO + * EVENT SHALL SILICON GRAPHICS, INC. BE LIABLE TO YOU OR ANYONE + * ELSE FOR ANY DIRECT, SPECIAL, INCIDENTAL, INDIRECT OR + * CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER, + * INCLUDING WITHOUT LIMITATION, LOSS OF PROFIT, LOSS OF USE, + * SAVINGS OR REVENUE, OR THE CLAIMS OF THIRD PARTIES, WHETHER OR + * NOT SILICON GRAPHICS, INC. HAS BEEN ADVISED OF THE POSSIBILITY + * OF SUCH LOSS, HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, + * ARISING OUT OF OR IN CONNECTION WITH THE POSSESSION, USE OR + * PERFORMANCE OF THIS SOFTWARE. + * + * US Government Users Restricted Rights + * + * Use, duplication, or disclosure by the Government is subject to + * restrictions set forth in FAR 52.227.19(c)(2) or subparagraph + * (c)(1)(ii) of the Rights in Technical Data and Computer + * Software clause at DFARS 252.227-7013 and/or in similar or + * successor clauses in the FAR or the DOD or NASA FAR + * Supplement. Unpublished-- rights reserved under the copyright + * laws of the United States. Contractor/manufacturer is Silicon + * Graphics, Inc., 2011 N. Shoreline Blvd., Mountain View, CA + * 94039-7311. + * + * OpenGL(TM) is a trademark of Silicon Graphics, Inc. + */ + +#include +#include "freeglut_internal.h" +#include "freeglut_teapot_data.h" + +/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ + + +static void fghTeapot( GLint grid, GLdouble scale, GLenum type ) +{ +#if defined(_WIN32_WCE) + int i, numV=sizeof(strip_vertices)/4, numI=sizeof(strip_normals)/4; +#else + double p[4][4][3], q[4][4][3], r[4][4][3], s[4][4][3]; + long i, j, k, l; +#endif + + glPushAttrib( GL_ENABLE_BIT | GL_EVAL_BIT ); + glEnable( GL_AUTO_NORMAL ); + glEnable( GL_NORMALIZE ); + glEnable( GL_MAP2_VERTEX_3 ); + glEnable( GL_MAP2_TEXTURE_COORD_2 ); + + glPushMatrix(); + glRotated( 270.0, 1.0, 0.0, 0.0 ); + glScaled( 0.5 * scale, 0.5 * scale, 0.5 * scale ); + glTranslated( 0.0, 0.0, -1.5 ); + +#if defined(_WIN32_WCE) + glRotated( 90.0, 1.0, 0.0, 0.0 ); + glBegin( GL_TRIANGLE_STRIP ); + + for( i = 0; i < numV-1; i++ ) + { + int vidx = strip_vertices[i], + nidx = strip_normals[i]; + + if( vidx != -1 ) + { + glNormal3fv( normals[nidx] ); + glVertex3fv( vertices[vidx] ); + } + else + { + glEnd(); + glBegin( GL_TRIANGLE_STRIP ); + } + } + + glEnd(); +#else + for (i = 0; i < 10; i++) { + for (j = 0; j < 4; j++) { + for (k = 0; k < 4; k++) { + for (l = 0; l < 3; l++) { + p[j][k][l] = cpdata[patchdata[i][j * 4 + k]][l]; + q[j][k][l] = cpdata[patchdata[i][j * 4 + (3 - k)]][l]; + if (l == 1) + q[j][k][l] *= -1.0; + if (i < 6) { + r[j][k][l] = + cpdata[patchdata[i][j * 4 + (3 - k)]][l]; + if (l == 0) + r[j][k][l] *= -1.0; + s[j][k][l] = cpdata[patchdata[i][j * 4 + k]][l]; + if (l == 0) + s[j][k][l] *= -1.0; + if (l == 1) + s[j][k][l] *= -1.0; + } + } + } + } + + glMap2d(GL_MAP2_TEXTURE_COORD_2, 0.0, 1.0, 2, 2, 0.0, 1.0, 4, 2, + &tex[0][0][0]); + glMap2d(GL_MAP2_VERTEX_3, 0.0, 1.0, 3, 4, 0.0, 1.0, 12, 4, + &p[0][0][0]); + glMapGrid2d(grid, 0.0, 1.0, grid, 0.0, 1.0); + glEvalMesh2(type, 0, grid, 0, grid); + glMap2d(GL_MAP2_VERTEX_3, 0.0, 1.0, 3, 4, 0.0, 1.0, 12, 4, + &q[0][0][0]); + glEvalMesh2(type, 0, grid, 0, grid); + if (i < 6) { + glMap2d(GL_MAP2_VERTEX_3, 0.0, 1.0, 3, 4, 0.0, 1.0, 12, 4, + &r[0][0][0]); + glEvalMesh2(type, 0, grid, 0, grid); + glMap2d(GL_MAP2_VERTEX_3, 0.0, 1.0, 3, 4, 0.0, 1.0, 12, 4, + &s[0][0][0]); + glEvalMesh2(type, 0, grid, 0, grid); + } + } +#endif /* defined(_WIN32_WCE) */ + + glPopMatrix(); + glPopAttrib(); +} + + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * Renders a beautiful wired teapot... + */ +void FGAPIENTRY glutWireTeapot( GLdouble size ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWireTeapot" ); + /* We will use the general teapot rendering code */ + fghTeapot( 10, size, GL_LINE ); +} + +/* + * Renders a beautiful filled teapot... + */ +void FGAPIENTRY glutSolidTeapot( GLdouble size ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSolidTeapot" ); + /* We will use the general teapot rendering code */ + fghTeapot( 7, size, GL_FILL ); +} + +/*** END OF FILE ***/ + + + + + diff --git a/examples/opengl-framework/freeglut/freeglut_teapot_data.h b/examples/opengl-framework/freeglut/freeglut_teapot_data.h new file mode 100644 index 00000000..3bf83e11 --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_teapot_data.h @@ -0,0 +1,2429 @@ +/* + * freeglut_teapot_data.h + * + * The freeglut library teapot data include file. + * + * 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 + * PAWEL W. OLSZTA 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 FREEGLUT_TEAPOT_DATA_H +#define FREEGLUT_TEAPOT_DATA_H + +#if defined(_WIN32_WCE) +/* + * Submitted through the kind offices of Daniel Wagner (daniel@ims.tuwien.ac.at) + */ + +/* 530 vertices */ + +const int numVertices = 530; +const float vertices[530][3] = { +2.1f, 3.6f, 0.0f, +2.071f, 3.711f, 0.0f, +2.105f, 3.748f, 0.0f, +2.174f, 3.711f, 0.0f, +2.25f, 3.6f, 0.0f, +1.937f, 3.6f, 0.8242f, +1.91f, 3.711f, 0.8128f, +1.942f, 3.748f, 0.8261f, +2.005f, 3.711f, 0.8532f, +2.076f, 3.6f, 0.8831f, +1.491f, 3.6f, 1.491f, +1.47f, 3.711f, 1.47f, +1.494f, 3.748f, 1.494f, +1.543f, 3.711f, 1.543f, +1.597f, 3.6f, 1.597f, +0.8242f, 3.6f, 1.937f, +0.8128f, 3.711f, 1.91f, +0.8261f, 3.748f, 1.942f, +0.8532f, 3.711f, 2.005f, +0.8831f, 3.6f, 2.076f, +0.0f, 3.6f, 2.1f, +0.0f, 3.711f, 2.071f, +0.0f, 3.748f, 2.105f, +0.0f, 3.711f, 2.174f, +0.0f, 3.6f, 2.25f, +-0.8812f, 3.6f, 1.937f, +-0.8368f, 3.711f, 1.91f, +-0.8332f, 3.748f, 1.942f, +-0.8541f, 3.711f, 2.005f, +-0.8831f, 3.6f, 2.076f, +-1.542f, 3.6f, 1.491f, +-1.492f, 3.711f, 1.47f, +-1.501f, 3.748f, 1.494f, +-1.544f, 3.711f, 1.543f, +-1.597f, 3.6f, 1.597f, +-1.956f, 3.6f, 0.8242f, +-1.918f, 3.711f, 0.8128f, +-1.944f, 3.748f, 0.8261f, +-2.006f, 3.711f, 0.8532f, +-2.076f, 3.6f, 0.8831f, +-2.1f, 3.6f, 0.0f, +-2.071f, 3.711f, 0.0f, +-2.105f, 3.748f, 0.0f, +-2.174f, 3.711f, 0.0f, +-2.25f, 3.6f, 0.0f, +-1.937f, 3.6f, -0.8242f, +-1.91f, 3.711f, -0.8128f, +-1.942f, 3.748f, -0.8261f, +-2.005f, 3.711f, -0.8532f, +-2.076f, 3.6f, -0.8831f, +-1.491f, 3.6f, -1.491f, +-1.47f, 3.711f, -1.47f, +-1.494f, 3.748f, -1.494f, +-1.543f, 3.711f, -1.543f, +-1.597f, 3.6f, -1.597f, +-0.8242f, 3.6f, -1.937f, +-0.8128f, 3.711f, -1.91f, +-0.8261f, 3.748f, -1.942f, +-0.8532f, 3.711f, -2.005f, +-0.8831f, 3.6f, -2.076f, +0.0f, 3.6f, -2.1f, +0.0f, 3.711f, -2.071f, +0.0f, 3.748f, -2.105f, +0.0f, 3.711f, -2.174f, +0.0f, 3.6f, -2.25f, +0.8242f, 3.6f, -1.937f, +0.8128f, 3.711f, -1.91f, +0.8261f, 3.748f, -1.942f, +0.8532f, 3.711f, -2.005f, +0.8831f, 3.6f, -2.076f, +1.491f, 3.6f, -1.491f, +1.47f, 3.711f, -1.47f, +1.494f, 3.748f, -1.494f, +1.543f, 3.711f, -1.543f, +1.597f, 3.6f, -1.597f, +1.937f, 3.6f, -0.8242f, +1.91f, 3.711f, -0.8128f, +1.942f, 3.748f, -0.8261f, +2.005f, 3.711f, -0.8532f, +2.076f, 3.6f, -0.8831f, +2.525f, 3.011f, 0.0f, +2.766f, 2.433f, 0.0f, +2.936f, 1.876f, 0.0f, +3.0f, 1.35f, 0.0f, +2.33f, 3.011f, 0.9912f, +2.551f, 2.433f, 1.086f, +2.708f, 1.876f, 1.152f, +2.767f, 1.35f, 1.178f, +1.793f, 3.011f, 1.793f, +1.964f, 2.433f, 1.964f, +2.084f, 1.876f, 2.084f, +2.13f, 1.35f, 2.13f, +0.9912f, 3.011f, 2.33f, +1.086f, 2.433f, 2.551f, +1.152f, 1.876f, 2.708f, +1.178f, 1.35f, 2.767f, +0.0f, 3.011f, 2.525f, +0.0f, 2.433f, 2.766f, +0.0f, 1.876f, 2.936f, +0.0f, 1.35f, 3.0f, +-0.9912f, 3.011f, 2.33f, +-1.086f, 2.433f, 2.551f, +-1.152f, 1.876f, 2.708f, +-1.178f, 1.35f, 2.767f, +-1.793f, 3.011f, 1.793f, +-1.964f, 2.433f, 1.964f, +-2.084f, 1.876f, 2.084f, +-2.13f, 1.35f, 2.13f, +-2.33f, 3.011f, 0.9912f, +-2.551f, 2.433f, 1.086f, +-2.708f, 1.876f, 1.152f, +-2.767f, 1.35f, 1.178f, +-2.525f, 3.011f, 0.0f, +-2.766f, 2.433f, 0.0f, +-2.936f, 1.876f, 0.0f, +-3.0f, 1.35f, 0.0f, +-2.33f, 3.011f, -0.9912f, +-2.551f, 2.433f, -1.086f, +-2.708f, 1.876f, -1.152f, +-2.767f, 1.35f, -1.178f, +-1.793f, 3.011f, -1.793f, +-1.964f, 2.433f, -1.964f, +-2.084f, 1.876f, -2.084f, +-2.13f, 1.35f, -2.13f, +-0.9912f, 3.011f, -2.33f, +-1.086f, 2.433f, -2.551f, +-1.152f, 1.876f, -2.708f, +-1.178f, 1.35f, -2.767f, +0.0f, 3.011f, -2.525f, +0.0f, 2.433f, -2.766f, +0.0f, 1.876f, -2.936f, +0.0f, 1.35f, -3.0f, +0.9912f, 3.011f, -2.33f, +1.086f, 2.433f, -2.551f, +1.152f, 1.876f, -2.708f, +1.178f, 1.35f, -2.767f, +1.793f, 3.011f, -1.793f, +1.964f, 2.433f, -1.964f, +2.084f, 1.876f, -2.084f, +2.13f, 1.35f, -2.13f, +2.33f, 3.011f, -0.9912f, +2.551f, 2.433f, -1.086f, +2.708f, 1.876f, -1.152f, +2.767f, 1.35f, -1.178f, +2.883f, 0.9053f, 0.0f, +2.625f, 0.5766f, 0.0f, +2.367f, 0.3533f, 0.0f, +2.25f, 0.225f, 0.0f, +2.659f, 0.9053f, 1.132f, +2.422f, 0.5766f, 1.03f, +2.184f, 0.3533f, 0.9291f, +2.076f, 0.225f, 0.8831f, +2.047f, 0.9053f, 2.047f, +1.864f, 0.5766f, 1.864f, +1.681f, 0.3533f, 1.681f, +1.597f, 0.225f, 1.597f, +1.132f, 0.9053f, 2.659f, +1.03f, 0.5766f, 2.422f, +0.9291f, 0.3533f, 2.184f, +0.8831f, 0.225f, 2.076f, +0.0f, 0.9053f, 2.883f, +0.0f, 0.5766f, 2.625f, +0.0f, 0.3533f, 2.367f, +0.0f, 0.225f, 2.25f, +-1.132f, 0.9053f, 2.659f, +-1.03f, 0.5766f, 2.422f, +-0.9291f, 0.3533f, 2.184f, +-0.8831f, 0.225f, 2.076f, +-2.047f, 0.9053f, 2.047f, +-1.864f, 0.5766f, 1.864f, +-1.681f, 0.3533f, 1.681f, +-1.597f, 0.225f, 1.597f, +-2.659f, 0.9053f, 1.132f, +-2.422f, 0.5766f, 1.03f, +-2.184f, 0.3533f, 0.9291f, +-2.076f, 0.225f, 0.8831f, +-2.883f, 0.9053f, 0.0f, +-2.625f, 0.5766f, 0.0f, +-2.367f, 0.3533f, 0.0f, +-2.25f, 0.225f, 0.0f, +-2.659f, 0.9053f, -1.132f, +-2.422f, 0.5766f, -1.03f, +-2.184f, 0.3533f, -0.9291f, +-2.076f, 0.225f, -0.8831f, +-2.047f, 0.9053f, -2.047f, +-1.864f, 0.5766f, -1.864f, +-1.681f, 0.3533f, -1.681f, +-1.597f, 0.225f, -1.597f, +-1.132f, 0.9053f, -2.659f, +-1.03f, 0.5766f, -2.422f, +-0.9291f, 0.3533f, -2.184f, +-0.8831f, 0.225f, -2.076f, +0.0f, 0.9053f, -2.883f, +0.0f, 0.5766f, -2.625f, +0.0f, 0.3533f, -2.367f, +0.0f, 0.225f, -2.25f, +1.132f, 0.9053f, -2.659f, +1.03f, 0.5766f, -2.422f, +0.9291f, 0.3533f, -2.184f, +0.8831f, 0.225f, -2.076f, +2.047f, 0.9053f, -2.047f, +1.864f, 0.5766f, -1.864f, +1.681f, 0.3533f, -1.681f, +1.597f, 0.225f, -1.597f, +2.659f, 0.9053f, -1.132f, +2.422f, 0.5766f, -1.03f, +2.184f, 0.3533f, -0.9291f, +2.076f, 0.225f, -0.8831f, +2.199f, 0.1424f, 0.0f, +1.927f, 0.07031f, 0.0f, +1.253f, 0.01934f, 0.0f, +0.0f, 0.0f, 0.0f, +2.029f, 0.1424f, 0.8631f, +1.777f, 0.07031f, 0.7562f, +1.156f, 0.01934f, 0.4919f, +1.561f, 0.1424f, 1.561f, +1.368f, 0.07031f, 1.368f, +0.8899f, 0.01934f, 0.8899f, +0.8631f, 0.1424f, 2.029f, +0.7562f, 0.07031f, 1.777f, +0.4919f, 0.01934f, 1.156f, +0.0f, 0.1424f, 2.199f, +0.0f, 0.07031f, 1.927f, +0.0f, 0.01934f, 1.253f, +-0.8631f, 0.1424f, 2.029f, +-0.7562f, 0.07031f, 1.777f, +-0.4919f, 0.01934f, 1.156f, +-1.561f, 0.1424f, 1.561f, +-1.368f, 0.07031f, 1.368f, +-0.8899f, 0.01934f, 0.8899f, +-2.029f, 0.1424f, 0.8631f, +-1.777f, 0.07031f, 0.7562f, +-1.156f, 0.01934f, 0.4919f, +-2.199f, 0.1424f, 0.0f, +-1.927f, 0.07031f, 0.0f, +-1.253f, 0.01934f, 0.0f, +-2.029f, 0.1424f, -0.8631f, +-1.777f, 0.07031f, -0.7562f, +-1.156f, 0.01934f, -0.4919f, +-1.561f, 0.1424f, -1.561f, +-1.368f, 0.07031f, -1.368f, +-0.8899f, 0.01934f, -0.8899f, +-0.8631f, 0.1424f, -2.029f, +-0.7562f, 0.07031f, -1.777f, +-0.4919f, 0.01934f, -1.156f, +0.0f, 0.1424f, -2.199f, +0.0f, 0.07031f, -1.927f, +0.0f, 0.01934f, -1.253f, +0.8631f, 0.1424f, -2.029f, +0.7562f, 0.07031f, -1.777f, +0.4919f, 0.01934f, -1.156f, +1.561f, 0.1424f, -1.561f, +1.368f, 0.07031f, -1.368f, +0.8899f, 0.01934f, -0.8899f, +2.029f, 0.1424f, -0.8631f, +1.777f, 0.07031f, -0.7562f, +1.156f, 0.01934f, -0.4919f, +-2.4f, 3.038f, 0.0f, +-3.101f, 3.032f, 0.0f, +-3.619f, 2.995f, 0.0f, +-3.94f, 2.895f, 0.0f, +-4.05f, 2.7f, 0.0f, +-2.377f, 3.09f, 0.2531f, +-3.122f, 3.084f, 0.2531f, +-3.669f, 3.041f, 0.2531f, +-4.005f, 2.926f, 0.2531f, +-4.12f, 2.7f, 0.2531f, +-2.325f, 3.206f, 0.3375f, +-3.168f, 3.198f, 0.3375f, +-3.778f, 3.143f, 0.3375f, +-4.15f, 2.993f, 0.3375f, +-4.275f, 2.7f, 0.3375f, +-2.273f, 3.322f, 0.2531f, +-3.214f, 3.313f, 0.2531f, +-3.888f, 3.244f, 0.2531f, +-4.294f, 3.06f, 0.2531f, +-4.43f, 2.7f, 0.2531f, +-2.25f, 3.375f, 0.0f, +-3.234f, 3.364f, 0.0f, +-3.938f, 3.291f, 0.0f, +-4.359f, 3.09f, 0.0f, +-4.5f, 2.7f, 0.0f, +-2.273f, 3.322f, -0.2531f, +-3.214f, 3.313f, -0.2531f, +-3.888f, 3.244f, -0.2531f, +-4.294f, 3.06f, -0.2531f, +-4.43f, 2.7f, -0.2531f, +-2.325f, 3.206f, -0.3375f, +-3.168f, 3.198f, -0.3375f, +-3.778f, 3.143f, -0.3375f, +-4.15f, 2.993f, -0.3375f, +-4.275f, 2.7f, -0.3375f, +-2.377f, 3.09f, -0.2531f, +-3.122f, 3.084f, -0.2531f, +-3.669f, 3.041f, -0.2531f, +-4.005f, 2.926f, -0.2531f, +-4.12f, 2.7f, -0.2531f, +-3.991f, 2.394f, 0.0f, +-3.806f, 2.025f, 0.0f, +-3.48f, 1.656f, 0.0f, +-3.0f, 1.35f, 0.0f, +-4.055f, 2.365f, 0.2531f, +-3.852f, 1.98f, 0.2531f, +-3.496f, 1.6f, 0.2531f, +-2.977f, 1.28f, 0.2531f, +-4.196f, 2.3f, 0.3375f, +-3.952f, 1.881f, 0.3375f, +-3.531f, 1.478f, 0.3375f, +-2.925f, 1.125f, 0.3375f, +-4.336f, 2.235f, 0.2531f, +-4.051f, 1.782f, 0.2531f, +-3.566f, 1.356f, 0.2531f, +-2.873f, 0.9703f, 0.2531f, +-4.4f, 2.205f, 0.0f, +-4.097f, 1.737f, 0.0f, +-3.582f, 1.3f, 0.0f, +-2.85f, 0.9f, 0.0f, +-4.336f, 2.235f, -0.2531f, +-4.051f, 1.782f, -0.2531f, +-3.566f, 1.356f, -0.2531f, +-2.873f, 0.9703f, -0.2531f, +-4.196f, 2.3f, -0.3375f, +-3.952f, 1.881f, -0.3375f, +-3.531f, 1.478f, -0.3375f, +-2.925f, 1.125f, -0.3375f, +-4.055f, 2.365f, -0.2531f, +-3.852f, 1.98f, -0.2531f, +-3.496f, 1.6f, -0.2531f, +-2.977f, 1.28f, -0.2531f, +2.55f, 2.137f, 0.0f, +3.27f, 2.303f, 0.0f, +3.581f, 2.7f, 0.0f, +3.752f, 3.182f, 0.0f, +4.05f, 3.6f, 0.0f, +2.55f, 1.944f, 0.5569f, +3.324f, 2.159f, 0.5028f, +3.652f, 2.617f, 0.3839f, +3.838f, 3.151f, 0.265f, +4.191f, 3.6f, 0.2109f, +2.55f, 1.519f, 0.7425f, +3.445f, 1.844f, 0.6704f, +3.806f, 2.433f, 0.5119f, +4.027f, 3.085f, 0.3533f, +4.5f, 3.6f, 0.2813f, +2.55f, 1.093f, 0.5569f, +3.566f, 1.529f, 0.5028f, +3.961f, 2.249f, 0.3839f, +4.215f, 3.018f, 0.265f, +4.809f, 3.6f, 0.2109f, +2.55f, 0.9f, 0.0f, +3.621f, 1.385f, 0.0f, +4.031f, 2.166f, 0.0f, +4.301f, 2.988f, 0.0f, +4.95f, 3.6f, 0.0f, +2.55f, 1.093f, -0.5569f, +3.566f, 1.529f, -0.5028f, +3.961f, 2.249f, -0.3839f, +4.215f, 3.018f, -0.265f, +4.809f, 3.6f, -0.2109f, +2.55f, 1.519f, -0.7425f, +3.445f, 1.844f, -0.6704f, +3.806f, 2.433f, -0.5119f, +4.027f, 3.085f, -0.3533f, +4.5f, 3.6f, -0.2813f, +2.55f, 1.944f, -0.5569f, +3.324f, 2.159f, -0.5028f, +3.652f, 2.617f, -0.3839f, +3.838f, 3.151f, -0.265f, +4.191f, 3.6f, -0.2109f, +4.158f, 3.663f, 0.0f, +4.238f, 3.684f, 0.0f, +4.261f, 3.663f, 0.0f, +4.2f, 3.6f, 0.0f, +4.308f, 3.666f, 0.1978f, +4.379f, 3.689f, 0.1687f, +4.381f, 3.668f, 0.1397f, +4.294f, 3.6f, 0.1266f, +4.64f, 3.673f, 0.2637f, +4.69f, 3.7f, 0.225f, +4.645f, 3.677f, 0.1863f, +4.5f, 3.6f, 0.1688f, +4.971f, 3.68f, 0.1978f, +5.001f, 3.711f, 0.1687f, +4.909f, 3.687f, 0.1397f, +4.706f, 3.6f, 0.1266f, +5.122f, 3.683f, 0.0f, +5.142f, 3.716f, 0.0f, +5.029f, 3.691f, 0.0f, +4.8f, 3.6f, 0.0f, +4.971f, 3.68f, -0.1978f, +5.001f, 3.711f, -0.1687f, +4.909f, 3.687f, -0.1397f, +4.706f, 3.6f, -0.1266f, +4.64f, 3.673f, -0.2637f, +4.69f, 3.7f, -0.225f, +4.645f, 3.677f, -0.1863f, +4.5f, 3.6f, -0.1688f, +4.308f, 3.666f, -0.1978f, +4.379f, 3.689f, -0.1687f, +4.381f, 3.668f, -0.1397f, +4.294f, 3.6f, -0.1266f, +0.0f, 4.725f, 0.0f, +0.5109f, 4.651f, 0.0f, +0.4875f, 4.472f, 0.0f, +0.2953f, 4.25f, 0.0f, +0.3f, 4.05f, 0.0f, +0.4715f, 4.651f, 0.2011f, +0.4499f, 4.472f, 0.1918f, +0.2725f, 4.25f, 0.1161f, +0.2768f, 4.05f, 0.1178f, +0.3632f, 4.651f, 0.3632f, +0.3465f, 4.472f, 0.3465f, +0.2098f, 4.25f, 0.2098f, +0.213f, 4.05f, 0.213f, +0.2011f, 4.651f, 0.4715f, +0.1918f, 4.472f, 0.4499f, +0.1161f, 4.25f, 0.2725f, +0.1178f, 4.05f, 0.2768f, +0.0f, 4.651f, 0.5109f, +0.0f, 4.472f, 0.4875f, +0.0f, 4.25f, 0.2953f, +0.0f, 4.05f, 0.3f, +-0.2011f, 4.651f, 0.4715f, +-0.1918f, 4.472f, 0.4499f, +-0.1161f, 4.25f, 0.2725f, +-0.1178f, 4.05f, 0.2768f, +-0.3632f, 4.651f, 0.3632f, +-0.3465f, 4.472f, 0.3465f, +-0.2098f, 4.25f, 0.2098f, +-0.213f, 4.05f, 0.213f, +-0.4715f, 4.651f, 0.2011f, +-0.4499f, 4.472f, 0.1918f, +-0.2725f, 4.25f, 0.1161f, +-0.2768f, 4.05f, 0.1178f, +-0.5109f, 4.651f, 0.0f, +-0.4875f, 4.472f, 0.0f, +-0.2953f, 4.25f, 0.0f, +-0.3f, 4.05f, 0.0f, +-0.4715f, 4.651f, -0.2011f, +-0.4499f, 4.472f, -0.1918f, +-0.2725f, 4.25f, -0.1161f, +-0.2768f, 4.05f, -0.1178f, +-0.3632f, 4.651f, -0.3632f, +-0.3465f, 4.472f, -0.3465f, +-0.2098f, 4.25f, -0.2098f, +-0.213f, 4.05f, -0.213f, +-0.2011f, 4.651f, -0.4715f, +-0.1918f, 4.472f, -0.4499f, +-0.1161f, 4.25f, -0.2725f, +-0.1178f, 4.05f, -0.2768f, +0.0f, 4.651f, -0.5109f, +0.0f, 4.472f, -0.4875f, +0.0f, 4.25f, -0.2953f, +0.0f, 4.05f, -0.3f, +0.2011f, 4.651f, -0.4715f, +0.1918f, 4.472f, -0.4499f, +0.1161f, 4.25f, -0.2725f, +0.1178f, 4.05f, -0.2768f, +0.3632f, 4.651f, -0.3632f, +0.3465f, 4.472f, -0.3465f, +0.2098f, 4.25f, -0.2098f, +0.213f, 4.05f, -0.213f, +0.4715f, 4.651f, -0.2011f, +0.4499f, 4.472f, -0.1918f, +0.2725f, 4.25f, -0.1161f, +0.2768f, 4.05f, -0.1178f, +0.6844f, 3.916f, 0.0f, +1.237f, 3.825f, 0.0f, +1.734f, 3.734f, 0.0f, +1.95f, 3.6f, 0.0f, +0.6313f, 3.916f, 0.2686f, +1.142f, 3.825f, 0.4857f, +1.6f, 3.734f, 0.6807f, +1.799f, 3.6f, 0.7654f, +0.4859f, 3.916f, 0.4859f, +0.8786f, 3.825f, 0.8786f, +1.231f, 3.734f, 1.231f, +1.385f, 3.6f, 1.385f, +0.2686f, 3.916f, 0.6313f, +0.4857f, 3.825f, 1.142f, +0.6807f, 3.734f, 1.6f, +0.7654f, 3.6f, 1.799f, +0.0f, 3.916f, 0.6844f, +0.0f, 3.825f, 1.237f, +0.0f, 3.734f, 1.734f, +0.0f, 3.6f, 1.95f, +-0.2686f, 3.916f, 0.6313f, +-0.4857f, 3.825f, 1.142f, +-0.6807f, 3.734f, 1.6f, +-0.7654f, 3.6f, 1.799f, +-0.4859f, 3.916f, 0.4859f, +-0.8786f, 3.825f, 0.8786f, +-1.231f, 3.734f, 1.231f, +-1.385f, 3.6f, 1.385f, +-0.6313f, 3.916f, 0.2686f, +-1.142f, 3.825f, 0.4857f, +-1.6f, 3.734f, 0.6807f, +-1.799f, 3.6f, 0.7654f, +-0.6844f, 3.916f, 0.0f, +-1.237f, 3.825f, 0.0f, +-1.734f, 3.734f, 0.0f, +-1.95f, 3.6f, 0.0f, +-0.6313f, 3.916f, -0.2686f, +-1.142f, 3.825f, -0.4857f, +-1.6f, 3.734f, -0.6807f, +-1.799f, 3.6f, -0.7654f, +-0.4859f, 3.916f, -0.4859f, +-0.8786f, 3.825f, -0.8786f, +-1.231f, 3.734f, -1.231f, +-1.385f, 3.6f, -1.385f, +-0.2686f, 3.916f, -0.6313f, +-0.4857f, 3.825f, -1.142f, +-0.6807f, 3.734f, -1.6f, +-0.7654f, 3.6f, -1.799f, +0.0f, 3.916f, -0.6844f, +0.0f, 3.825f, -1.237f, +0.0f, 3.734f, -1.734f, +0.0f, 3.6f, -1.95f, +0.2686f, 3.916f, -0.6313f, +0.4857f, 3.825f, -1.142f, +0.6807f, 3.734f, -1.6f, +0.7654f, 3.6f, -1.799f, +0.4859f, 3.916f, -0.4859f, +0.8786f, 3.825f, -0.8786f, +1.231f, 3.734f, -1.231f, +1.385f, 3.6f, -1.385f, +0.6313f, 3.916f, -0.2686f, +1.142f, 3.825f, -0.4857f, +1.6f, 3.734f, -0.6807f, +1.799f, 3.6f, -0.7654f +}; + + +/* 530 normals */ +const int numNormals = 530; +const float normals[530][3] = { +0.0486f, -0.9986f, 0.0168f, +0.9976f, -0.0678f, -0.0008f, +-0.233f, 0.8502f, -0.4719f, +-0.2299f, 0.9679f, 0.1004f, +-0.1648f, 0.985f, 0.0501f, +-0.0117f, 0.7461f, 0.6656f, +-0.0888f, 0.9692f, 0.2294f, +0.6449f, -0.7172f, -0.2637f, +-0.066f, 0.9851f, 0.1583f, +-0.6585f, -0.342f, -0.6703f, +-0.293f, 0.9558f, 0.0209f, +0.179f, 0.9825f, -0.0513f, +-0.0094f, 0.903f, 0.4295f, +-0.0059f, -0.986f, -0.1662f, +-0.7355f, 0.6774f, -0.0026f, +-0.997f, 0.0763f, 0.0019f, +-0.1478f, 0.9333f, 0.3271f, +-0.3014f, -0.6034f, -0.7382f, +-0.7048f, -0.0681f, 0.706f, +-0.3361f, 0.9332f, 0.1263f, +0.3709f, 0.1524f, -0.916f, +-0.3399f, -0.4121f, 0.8453f, +0.1921f, 0.9724f, -0.1316f, +-0.2671f, 0.7429f, 0.6137f, +0.0888f, 0.9692f, -0.2294f, +0.066f, 0.9851f, -0.1583f, +0.9411f, 0.338f, 0.001f, +0.8666f, -0.2559f, 0.4282f, +-0.8029f, 0.4968f, 0.3293f, +-0.0008f, -0.0678f, -0.9976f, +-0.8453f, -0.4121f, -0.3399f, +-0.4801f, -0.8741f, 0.0733f, +0.6355f, -0.772f, 0.0006f, +-0.9215f, -0.0678f, 0.3822f, +-0.6698f, -0.6907f, -0.2723f, +0.3734f, 0.876f, -0.3051f, +0.3548f, -0.4118f, 0.8393f, +-0.3629f, 0.2429f, 0.8995f, +0.9033f, 0.2079f, 0.375f, +-0.2824f, 0.5939f, 0.7532f, +0.8938f, 0.4452f, 0.0532f, +0.1478f, 0.9333f, -0.3271f, +0.0085f, -0.0031f, -0.9999f, +0.3595f, 0.933f, 0.0115f, +0.8995f, 0.2429f, 0.3629f, +0.7048f, -0.0681f, -0.706f, +-0.6428f, -0.7172f, -0.2688f, +0.6366f, -0.447f, 0.6283f, +-0.1213f, -0.9861f, -0.1128f, +0.8003f, 0.4978f, 0.334f, +0.3361f, 0.9332f, -0.1263f, +0.3399f, -0.4121f, -0.8453f, +-0.3909f, 0.4452f, 0.8055f, +0.0117f, 0.7462f, -0.6655f, +0.9215f, -0.0678f, -0.3822f, +0.3582f, -0.7656f, 0.5343f, +-0.9782f, 0.2075f, -0.0011f, +0.2824f, 0.5939f, -0.7532f, +0.035f, -0.8413f, 0.5393f, +-0.8044f, 0.5934f, 0.0262f, +-0.1128f, -0.9861f, 0.1213f, +0.13f, -0.1396f, 0.9816f, +0.6644f, 0.3392f, 0.6659f, +-0.0042f, -0.6898f, -0.7239f, +-0.1587f, 0.9851f, 0.065f, +-0.8719f, -0.3415f, 0.3508f, +0.6486f, 0.4756f, -0.5941f, +-0.4991f, 0.8499f, -0.1684f, +-0.3969f, 0.6342f, -0.6634f, +0.7041f, -0.3863f, -0.5956f, +0.3909f, 0.4452f, -0.8055f, +-0.0391f, -0.0113f, 0.9991f, +-0.3321f, 0.5936f, -0.733f, +0.8523f, -0.5219f, -0.0338f, +0.329f, 0.4978f, 0.8023f, +0.8044f, 0.5934f, -0.0262f, +0.1128f, -0.9861f, -0.1213f, +0.0178f, 0.9861f, -0.1651f, +0.3491f, 0.4045f, 0.8452f, +-0.2727f, 0.8505f, 0.4496f, +0.065f, 0.9851f, 0.1587f, +-0.0005f, 0.4037f, 0.9148f, +-0.0077f, -0.4109f, -0.9116f, +0.5609f, -0.604f, 0.5661f, +0.8236f, 0.5668f, -0.0138f, +0.1587f, 0.9851f, -0.065f, +0.8719f, -0.3415f, -0.3508f, +-0.7382f, -0.6034f, 0.3014f, +0.0346f, 0.8495f, 0.5263f, +-0.4373f, -0.7921f, -0.4257f, +-0.0532f, 0.4452f, 0.8938f, +0.0689f, -0.9861f, 0.1509f, +-0.1509f, -0.9861f, 0.0689f, +0.7706f, -0.2424f, -0.5893f, +-0.7543f, -0.6564f, 0.0105f, +0.0005f, 0.4037f, -0.9148f, +-0.9116f, -0.4109f, 0.0077f, +0.0058f, -0.0438f, 0.999f, +0.1719f, 0.985f, 0.0005f, +-0.1697f, 0.9693f, 0.1774f, +0.5874f, -0.5124f, 0.6263f, +0.7382f, -0.6034f, -0.3014f, +-0.1518f, 0.985f, -0.081f, +0.646f, 0.4051f, 0.6468f, +0.334f, 0.4978f, -0.8003f, +-0.7354f, -0.6034f, -0.3082f, +-0.6919f, 0.2428f, -0.6798f, +0.0532f, 0.4452f, -0.8938f, +0.3547f, -0.3173f, 0.8794f, +0.9879f, -0.1547f, -0.0033f, +-0.0462f, -0.9986f, 0.0223f, +-0.6088f, 0.4806f, 0.6311f, +-0.109f, -0.1969f, -0.9743f, +0.1509f, -0.9861f, -0.0689f, +-0.0568f, 0.9983f, 0.0009f, +0.9074f, -0.3096f, -0.2839f, +0.8677f, 0.4969f, 0.0026f, +-0.2723f, -0.6907f, 0.6698f, +-0.4734f, -0.6798f, 0.5599f, +0.9116f, -0.4109f, -0.0077f, +0.1697f, 0.9693f, -0.1774f, +0.5875f, 0.5937f, 0.5497f, +-0.3232f, 0.6846f, 0.6533f, +-0.5078f, -0.6913f, 0.5139f, +-0.4612f, 0.7474f, -0.478f, +-0.2071f, -0.8049f, 0.556f, +-0.6976f, -0.7164f, -0.0027f, +-0.8697f, 0.3388f, 0.3587f, +0.0462f, -0.9986f, -0.0223f, +0.2723f, -0.6907f, -0.6698f, +-0.829f, -0.4466f, -0.3365f, +0.9148f, 0.4037f, 0.0005f, +-0.1583f, 0.9851f, -0.066f, +0.148f, 0.9838f, 0.1002f, +-0.1717f, 0.985f, -0.0162f, +-0.4282f, -0.2559f, 0.8666f, +0.3094f, -0.2556f, 0.9159f, +0.2803f, -0.6907f, 0.6665f, +-0.6154f, 0.497f, 0.6117f, +-0.0262f, 0.5934f, -0.8044f, +0.0286f, 0.1639f, -0.986f, +-0.6924f, 0.2083f, 0.6907f, +-0.0105f, 0.9975f, -0.0685f, +0.5078f, -0.6913f, -0.5139f, +0.2071f, -0.8049f, -0.556f, +-0.4903f, -0.7178f, -0.4942f, +-0.2637f, -0.7172f, -0.6449f, +-0.3822f, -0.0678f, -0.9215f, +0.8697f, 0.3388f, -0.3587f, +0.2461f, -0.805f, 0.5397f, +-0.2615f, 0.9334f, 0.2452f, +0.6187f, 0.747f, -0.243f, +0.0375f, -0.8401f, -0.5411f, +0.0054f, 0.9691f, 0.2464f, +0.3587f, 0.3388f, 0.8697f, +0.3993f, 0.6582f, -0.6381f, +-0.3476f, -0.4464f, -0.8245f, +0.099f, 0.9692f, 0.2251f, +-0.3666f, -0.3412f, 0.8655f, +0.0396f, 0.153f, -0.9874f, +0.0349f, 0.9969f, -0.0698f, +0.1096f, 0.985f, 0.1324f, +-0.0578f, -0.9861f, 0.1556f, +0.4479f, -0.5145f, -0.7311f, +0.6924f, 0.2083f, -0.6907f, +0.6096f, 0.747f, 0.265f, +-0.3508f, -0.3415f, -0.8719f, +-0.6215f, 0.4454f, -0.6443f, +-0.4942f, -0.7178f, 0.4903f, +-0.9402f, -0.3403f, -0.0085f, +0.0056f, -0.0358f, 0.9993f, +0.2615f, 0.9334f, -0.2452f, +-0.0024f, 0.0291f, -0.9995f, +-0.2667f, 0.9637f, -0.001f, +0.0569f, -0.2712f, -0.9608f, +0.7463f, 0.254f, 0.615f, +0.5153f, 0.6516f, -0.5564f, +0.0223f, -0.9986f, 0.0462f, +0.3666f, -0.3412f, -0.8655f, +0.0578f, -0.9861f, -0.1556f, +0.6111f, 0.4984f, 0.6148f, +-0.243f, 0.747f, -0.6187f, +-0.0092f, 0.2338f, -0.9722f, +0.478f, 0.7474f, -0.4612f, +-0.0058f, -0.4457f, -0.8951f, +-0.4856f, -0.6774f, -0.5524f, +0.54f, 0.6414f, 0.5448f, +-0.3365f, -0.4466f, 0.829f, +-0.2257f, 0.795f, 0.5629f, +0.8055f, 0.4452f, 0.3909f, +0.3729f, 0.208f, 0.9042f, +-0.727f, -0.2562f, 0.6369f, +-0.0514f, -0.9986f, 0.0029f, +0.9159f, 0.1555f, -0.3699f, +0.0019f, -0.2377f, -0.9713f, +0.4942f, -0.7178f, -0.4903f, +0.6497f, -0.4127f, 0.6383f, +0.0089f, 0.0486f, -0.9987f, +-0.0213f, 0.6301f, -0.7761f, +-0.9269f, -0.3751f, 0.0038f, +-0.1215f, 0.9852f, 0.1207f, +-0.5856f, 0.5198f, 0.6218f, +0.8655f, -0.3412f, 0.3666f, +-0.2464f, 0.9691f, 0.0054f, +0.0123f, 0.1386f, 0.9902f, +0.0179f, -0.0369f, 0.9991f, +-0.1207f, 0.9852f, -0.1215f, +-0.0081f, 0.5671f, 0.8235f, +-0.8689f, 0.3387f, -0.3607f, +0.0062f, 0.0309f, -0.9995f, +0.3365f, -0.4466f, -0.829f, +-0.3787f, 0.2424f, -0.8931f, +-0.2904f, 0.4454f, -0.8468f, +-0.8707f, 0.4915f, 0.0133f, +0.163f, -0.8182f, 0.5512f, +0.4337f, -0.8052f, 0.4041f, +0.0514f, -0.9986f, -0.0029f, +-0.0084f, 0.1303f, 0.9914f, +-0.706f, -0.0681f, -0.7048f, +-0.556f, -0.8049f, -0.2071f, +0.8448f, 0.4045f, 0.3501f, +0.4259f, -0.5474f, 0.7203f, +-0.6907f, 0.2083f, -0.6924f, +0.1215f, 0.9852f, -0.1207f, +-0.1263f, 0.9332f, -0.3361f, +0.7711f, -0.0741f, -0.6323f, +0.2464f, 0.9691f, -0.0054f, +0.1774f, 0.9693f, 0.1697f, +-0.9042f, 0.208f, 0.3729f, +-0.8393f, -0.4118f, 0.3548f, +0.6888f, -0.7219f, -0.0648f, +0.1556f, -0.9861f, 0.0578f, +0.3271f, 0.9333f, 0.1478f, +-0.0024f, 0.2379f, 0.9712f, +-0.0026f, 0.4969f, 0.8677f, +0.0f, 1.0f, 0.0f, +0.1912f, -0.9815f, -0.0025f, +-0.3762f, -0.6681f, 0.6418f, +-0.7759f, 0.0432f, 0.6292f, +-0.0208f, -0.8044f, -0.5936f, +-0.2274f, 0.8822f, -0.4122f, +0.7532f, 0.5939f, 0.2824f, +-0.9221f, -0.0681f, -0.3807f, +-0.2198f, 0.8494f, 0.4796f, +0.0065f, -0.7656f, 0.6431f, +-0.5876f, 0.4472f, -0.6742f, +0.7981f, -0.6024f, 0.0036f, +-0.0383f, -0.9986f, -0.0341f, +-0.6369f, -0.2562f, -0.727f, +-0.5497f, 0.5937f, 0.5875f, +0.1084f, 0.9431f, 0.314f, +0.9042f, 0.208f, -0.3729f, +-0.6659f, 0.3392f, 0.6644f, +0.8393f, -0.4118f, -0.3548f, +0.0029f, -0.9986f, 0.0514f, +-0.9647f, -0.2552f, -0.0635f, +-0.2294f, 0.9692f, -0.0888f, +0.0026f, 0.4969f, -0.8677f, +0.2452f, 0.9334f, 0.2615f, +0.5171f, -0.4876f, -0.7033f, +-0.8951f, -0.4457f, 0.0058f, +-0.5936f, -0.8044f, 0.0208f, +0.5642f, -0.5426f, -0.6222f, +0.5938f, 0.4451f, 0.6702f, +0.5497f, 0.5937f, -0.5875f, +0.6657f, 0.4653f, 0.5832f, +0.4857f, -0.6243f, 0.6117f, +-0.0486f, -0.9986f, -0.0168f, +-0.6468f, 0.4051f, 0.646f, +0.6659f, 0.3392f, -0.6644f, +0.1833f, 0.9735f, -0.1365f, +0.3955f, 0.8505f, 0.3465f, +0.5139f, -0.6913f, 0.5078f, +0.8023f, 0.4978f, -0.329f, +-0.001f, 0.338f, 0.9411f, +-0.2496f, 0.8321f, -0.4951f, +0.8951f, -0.4457f, -0.0058f, +0.233f, 0.8502f, 0.4719f, +-0.0168f, -0.9986f, 0.0486f, +0.5936f, -0.8044f, -0.0208f, +-0.05f, 0.3155f, 0.9475f, +0.6585f, -0.342f, 0.6703f, +0.4909f, -0.1864f, -0.8509f, +-0.37f, 0.9238f, -0.0973f, +0.6468f, 0.4051f, -0.646f, +0.0059f, -0.986f, 0.1662f, +-0.3724f, 0.9278f, -0.0202f, +-0.3501f, 0.4045f, 0.8448f, +-0.0425f, 0.8398f, -0.5411f, +-0.1684f, 0.8499f, 0.4991f, +-0.6665f, -0.6907f, 0.2803f, +-0.2251f, 0.9692f, 0.099f, +0.9241f, -0.3816f, -0.0169f, +0.001f, 0.338f, -0.9411f, +-0.9411f, 0.338f, -0.001f, +-0.8666f, -0.2559f, -0.4282f, +0.0262f, 0.5183f, -0.8547f, +0.3014f, -0.6034f, 0.7382f, +0.0168f, -0.9986f, -0.0486f, +-0.3548f, -0.4118f, -0.8393f, +-0.6023f, -0.5297f, 0.5971f, +-0.9033f, 0.2079f, -0.375f, +-0.8938f, 0.4452f, -0.0532f, +0.6044f, 0.7397f, 0.2957f, +0.0008f, -0.0678f, 0.9976f, +0.7058f, 0.0906f, -0.7025f, +0.8453f, -0.4121f, 0.3399f, +-0.3595f, 0.933f, -0.0115f, +0.6698f, -0.6907f, 0.2723f, +-0.8995f, 0.2429f, -0.3629f, +-0.6366f, -0.447f, -0.6283f, +0.3501f, 0.4045f, -0.8448f, +-0.01f, -0.0605f, 0.9981f, +-0.8003f, 0.4978f, -0.334f, +0.1684f, 0.8499f, -0.4991f, +0.6665f, -0.6907f, -0.2803f, +0.2251f, 0.9692f, -0.099f, +-0.0036f, -0.6024f, 0.7981f, +0.6637f, -0.2967f, -0.6865f, +-0.081f, 0.985f, 0.1518f, +0.0084f, 0.2423f, 0.9701f, +0.0071f, -0.9029f, -0.4296f, +-0.8679f, 0.4966f, -0.0026f, +0.0123f, 0.5735f, 0.819f, +-0.0005f, 0.985f, 0.1719f, +0.6428f, -0.7172f, 0.2688f, +0.6588f, -0.3366f, 0.6727f, +0.1213f, -0.9861f, 0.1128f, +-0.8931f, 0.2424f, 0.3787f, +-0.1662f, -0.986f, 0.0059f, +0.9994f, 0.0313f, 0.0095f, +0.762f, -0.146f, 0.6308f, +-0.7731f, 0.0861f, -0.6283f, +-0.6644f, 0.3392f, -0.6659f, +-0.0027f, -0.7164f, 0.6976f, +0.0036f, -0.6024f, -0.7981f, +0.9782f, 0.2075f, 0.0011f, +0.0405f, -0.9991f, -0.0018f, +0.6882f, -0.703f, 0.179f, +-0.0115f, 0.933f, 0.3595f, +0.0911f, 0.0518f, -0.9944f, +0.0005f, 0.985f, -0.1719f, +0.5337f, -0.5852f, -0.6104f, +0.0042f, -0.6898f, 0.7239f, +0.4863f, 0.2366f, 0.8411f, +0.4991f, 0.8499f, 0.1684f, +-0.6543f, 0.7561f, 0.0071f, +0.265f, 0.747f, -0.6096f, +-0.329f, 0.4978f, -0.8023f, +0.1662f, -0.986f, -0.0059f, +-0.3491f, 0.4045f, -0.8452f, +0.3321f, 0.5936f, 0.733f, +-0.065f, 0.9851f, -0.1587f, +-0.6283f, -0.447f, 0.6366f, +0.0027f, -0.7164f, -0.6976f, +-0.1316f, 0.6339f, 0.762f, +-0.5609f, -0.604f, -0.5661f, +-0.8452f, 0.4045f, 0.3491f, +-0.5263f, 0.8495f, 0.0346f, +0.0115f, 0.933f, -0.3595f, +-0.0346f, 0.8495f, -0.5263f, +0.0077f, -0.4109f, 0.9116f, +0.5758f, -0.8175f, -0.0017f, +-0.0011f, 0.2075f, 0.9782f, +-0.0689f, -0.9861f, -0.1509f, +0.2934f, -0.5928f, -0.7499f, +0.0724f, 0.1198f, -0.9901f, +-0.7367f, -0.275f, -0.6176f, +-0.3131f, 0.8154f, 0.4868f, +-0.0114f, 0.0022f, 0.9999f, +0.6283f, -0.447f, -0.6366f, +0.8452f, 0.4045f, -0.3491f, +0.5263f, 0.8495f, -0.0346f, +-0.6383f, -0.4127f, 0.6497f, +-0.1719f, 0.985f, -0.0005f, +-0.6703f, -0.342f, 0.6585f, +-0.0085f, -0.3403f, 0.9402f, +-0.646f, 0.4051f, -0.6468f, +0.0011f, 0.2075f, -0.9782f, +-0.7216f, -0.3071f, 0.6204f, +0.0282f, 0.0023f, -0.9995f, +-0.2483f, 0.6806f, -0.6892f, +0.1518f, 0.985f, 0.081f, +0.047f, 0.0466f, -0.9978f, +0.7354f, -0.6034f, 0.3082f, +0.6919f, 0.2428f, 0.6798f, +0.4086f, -0.3626f, -0.8375f, +0.6383f, -0.4127f, -0.6497f, +-0.5875f, 0.5937f, -0.5497f, +0.6703f, -0.342f, -0.6585f, +-0.8245f, -0.4464f, 0.3476f, +0.0085f, -0.3403f, -0.9402f, +-0.0591f, -0.0663f, 0.996f, +0.0f, -1.0f, 0.0f, +0.4612f, 0.7474f, 0.478f, +0.6976f, -0.7164f, 0.0027f, +-0.9148f, 0.4037f, -0.0005f, +0.173f, -0.8158f, -0.5518f, +-0.3607f, 0.3387f, 0.8689f, +0.7836f, -0.2411f, 0.5724f, +-0.1985f, 0.8026f, -0.5623f, +-0.3094f, -0.2556f, -0.9159f, +-0.2803f, -0.6907f, -0.6665f, +0.8245f, -0.4464f, -0.3476f, +0.829f, -0.4466f, 0.3365f, +-0.4848f, 0.7385f, 0.4683f, +0.1583f, 0.9851f, 0.066f, +-0.0077f, 0.7656f, -0.6432f, +-0.0162f, 0.985f, 0.1717f, +0.1717f, 0.985f, 0.0162f, +0.0244f, 0.9805f, -0.1949f, +-0.2461f, -0.805f, -0.5397f, +0.0262f, 0.5934f, 0.8044f, +0.142f, 0.1881f, 0.9718f, +0.1846f, 0.1002f, 0.9776f, +0.4903f, -0.7178f, 0.4942f, +0.2637f, -0.7172f, 0.6449f, +0.3822f, -0.0678f, 0.9215f, +-0.0054f, 0.9691f, -0.2464f, +0.3607f, 0.3387f, -0.8689f, +-0.3587f, 0.3388f, -0.8697f, +-0.5694f, -0.8219f, 0.0081f, +-0.1324f, 0.985f, 0.1096f, +-0.099f, 0.9692f, -0.2251f, +-0.6702f, 0.4451f, 0.5938f, +0.0077f, -0.9976f, 0.0684f, +-0.5661f, -0.604f, 0.5609f, +-0.1096f, 0.985f, -0.1324f, +-0.6096f, 0.747f, -0.265f, +-0.0015f, 0.0295f, -0.9995f, +0.3476f, -0.4464f, 0.8245f, +-0.0635f, -0.2552f, 0.9647f, +-0.8468f, 0.4454f, 0.2904f, +-0.4719f, 0.8502f, 0.233f, +-0.0502f, 0.8385f, 0.5425f, +-0.6671f, 0.7448f, -0.0116f, +0.3508f, -0.3415f, 0.8719f, +-0.4119f, 0.6135f, -0.6736f, +-0.2688f, -0.7172f, 0.6428f, +-0.4041f, -0.8052f, 0.4337f, +-0.375f, 0.2079f, 0.9033f, +-0.0223f, -0.9986f, -0.0462f, +0.6702f, 0.4451f, -0.5938f, +0.9402f, -0.3403f, 0.0085f, +0.5661f, -0.604f, -0.5609f, +-0.6252f, 0.7406f, 0.246f, +-0.0341f, -0.9986f, 0.0383f, +-0.6111f, 0.4984f, -0.6148f, +0.6655f, 0.7462f, 0.0117f, +0.1233f, 0.199f, 0.9722f, +0.8468f, 0.4454f, -0.2904f, +0.7383f, 0.2702f, -0.6179f, +-0.8055f, 0.4452f, -0.3909f, +-0.3729f, 0.208f, -0.9042f, +0.4719f, 0.8502f, -0.233f, +0.243f, 0.747f, 0.6187f, +-0.6497f, -0.4127f, -0.6383f, +-0.5406f, 0.5651f, -0.623f, +0.0058f, -0.4457f, 0.8951f, +-0.3082f, -0.6034f, 0.7354f, +-0.8655f, -0.3412f, -0.3666f, +0.2688f, -0.7172f, -0.6428f, +0.4041f, -0.8052f, -0.4337f, +0.375f, 0.2079f, -0.9033f, +0.0341f, -0.9986f, -0.0383f, +-0.9701f, 0.2423f, 0.0084f, +-0.3807f, -0.0681f, 0.9221f, +0.9643f, -0.2551f, 0.0705f, +-0.8758f, 0.4808f, 0.0415f, +0.1207f, 0.9852f, 0.1215f, +0.4821f, 0.7724f, 0.4133f, +-0.0522f, 0.9982f, 0.0278f, +-0.4337f, -0.8052f, -0.4041f, +-0.6164f, 0.4198f, 0.6661f, +-0.8448f, 0.4045f, -0.3501f, +0.3082f, -0.6034f, -0.7354f, +0.8689f, 0.3387f, 0.3607f, +0.6894f, -0.7242f, 0.0091f, +0.3787f, 0.2424f, 0.8931f, +0.2904f, 0.4454f, 0.8468f, +0.6148f, 0.4984f, -0.6111f, +0.0501f, 0.985f, 0.1648f, +-0.5397f, -0.805f, 0.2461f, +-0.9159f, -0.2556f, 0.3094f, +0.706f, -0.0681f, 0.7048f, +-0.3341f, 0.4972f, 0.8006f, +0.556f, -0.8049f, 0.2071f, +-0.1774f, 0.9693f, -0.1697f, +0.6907f, 0.2083f, 0.6924f, +0.1263f, 0.9332f, 0.3361f, +0.3807f, -0.0681f, -0.9221f, +-0.1556f, -0.9861f, -0.0578f, +-0.3271f, 0.9333f, -0.1478f, +-0.3465f, 0.8505f, 0.3955f, +0.5315f, 0.8438f, -0.0735f, +0.9737f, 0.2276f, -0.0003f, +0.6441f, 0.7648f, -0.0112f, +-0.7239f, -0.6898f, 0.0042f, +-0.7532f, 0.5939f, -0.2824f, +0.1093f, 0.1415f, -0.9838f, +0.5397f, -0.805f, -0.2461f, +-0.7981f, -0.6024f, -0.0036f, +0.9456f, 0.3251f, -0.0052f, +0.1278f, 0.9696f, -0.2085f, +0.0208f, -0.8044f, 0.5936f, +0.1635f, 0.1348f, -0.9772f, +-0.733f, 0.5936f, 0.3321f, +-0.0505f, 0.9852f, -0.1635f, +0.4089f, -0.9069f, -0.1015f, +-0.0029f, -0.9986f, -0.0514f, +-0.1796f, 0.814f, -0.5522f, +0.9221f, -0.0681f, 0.3807f, +0.0383f, -0.9986f, 0.0341f, +0.6369f, -0.2562f, 0.727f, +0.3465f, 0.8505f, -0.3955f, +-0.2452f, 0.9334f, -0.2615f, +0.4921f, -0.247f, 0.8346f, +-0.9976f, -0.0678f, 0.0008f, +-0.5396f, 0.8418f, -0.0094f, +0.2294f, 0.9692f, 0.0888f, +0.7239f, -0.6898f, -0.0042f, +-0.4472f, 0.5952f, 0.6675f, +-0.6449f, -0.7172f, 0.2637f, +0.4543f, 0.2732f, -0.8478f, +-0.6798f, 0.2428f, 0.6919f, +-0.5938f, 0.4451f, -0.6702f, +0.733f, 0.5936f, -0.3321f, +-0.3955f, 0.8505f, -0.3465f, +-0.5139f, -0.6913f, -0.5078f, +-0.623f, -0.5156f, -0.5881f +}; + +/* 1 color */ +/*255 255 0 */ + +/* 1024 faces */ +/* numIdx fidx0 fidx1 fidx2 nidx0 nidx1 nidx2 coloridx */ + +const int numFaces = 1024; +const int faces[1024][8] = { +3, 0, 5, 6, 255, 295, 309, 0, +3, 6, 1, 0, 309, 465, 255, 0, +3, 1, 6, 7, 465, 309, 134, 0, +3, 7, 2, 1, 134, 4, 465, 0, +3, 2, 7, 8, 4, 134, 165, 0, +3, 8, 3, 2, 165, 448, 4, 0, +3, 3, 8, 9, 448, 165, 49, 0, +3, 9, 4, 3, 49, 116, 448, 0, +3, 5, 10, 11, 295, 248, 106, 0, +3, 11, 6, 5, 106, 309, 295, 0, +3, 6, 11, 12, 309, 106, 102, 0, +3, 12, 7, 6, 102, 134, 309, 0, +3, 7, 12, 13, 134, 102, 394, 0, +3, 13, 8, 7, 394, 165, 134, 0, +3, 8, 13, 14, 165, 394, 180, 0, +3, 14, 9, 8, 180, 49, 165, 0, +3, 10, 15, 16, 248, 401, 211, 0, +3, 16, 11, 10, 211, 106, 248, 0, +3, 11, 16, 17, 106, 211, 427, 0, +3, 17, 12, 11, 427, 102, 106, 0, +3, 12, 17, 18, 102, 427, 455, 0, +3, 18, 13, 12, 455, 394, 102, 0, +3, 13, 18, 19, 394, 455, 74, 0, +3, 19, 14, 13, 74, 180, 394, 0, +3, 15, 20, 21, 401, 174, 182, 0, +3, 21, 16, 15, 182, 211, 401, 0, +3, 16, 21, 22, 211, 182, 507, 0, +3, 22, 17, 16, 507, 427, 211, 0, +3, 17, 22, 23, 427, 507, 5, 0, +3, 23, 18, 17, 5, 455, 427, 0, +3, 18, 23, 24, 455, 5, 234, 0, +3, 24, 19, 18, 234, 74, 455, 0, +3, 20, 25, 26, 174, 386, 20, 0, +3, 26, 21, 20, 20, 182, 174, 0, +3, 21, 26, 27, 182, 20, 410, 0, +3, 27, 22, 21, 410, 507, 182, 0, +3, 22, 27, 28, 507, 410, 23, 0, +3, 28, 23, 22, 23, 5, 507, 0, +3, 23, 28, 29, 5, 23, 485, 0, +3, 29, 24, 23, 485, 234, 5, 0, +3, 25, 30, 31, 386, 69, 305, 0, +3, 31, 26, 25, 305, 20, 386, 0, +3, 26, 31, 32, 20, 305, 503, 0, +3, 32, 27, 26, 503, 410, 20, 0, +3, 27, 32, 33, 410, 503, 405, 0, +3, 33, 28, 27, 405, 23, 410, 0, +3, 28, 33, 34, 23, 405, 138, 0, +3, 34, 29, 28, 138, 485, 23, 0, +3, 30, 35, 36, 69, 115, 193, 0, +3, 36, 31, 30, 193, 305, 69, 0, +3, 31, 36, 37, 305, 193, 270, 0, +3, 37, 32, 31, 270, 503, 305, 0, +3, 32, 37, 38, 503, 270, 445, 0, +3, 38, 33, 32, 445, 405, 503, 0, +3, 33, 38, 39, 405, 445, 28, 0, +3, 39, 34, 33, 28, 138, 405, 0, +3, 35, 40, 41, 115, 467, 495, 0, +3, 41, 36, 35, 495, 193, 115, 0, +3, 36, 41, 42, 193, 495, 11, 0, +3, 42, 37, 36, 11, 270, 193, 0, +3, 37, 42, 43, 270, 11, 435, 0, +3, 43, 38, 37, 435, 445, 270, 0, +3, 38, 43, 44, 445, 435, 322, 0, +3, 44, 39, 38, 322, 28, 445, 0, +3, 40, 45, 46, 467, 27, 44, 0, +3, 46, 41, 40, 44, 495, 467, 0, +3, 41, 46, 47, 495, 44, 409, 0, +3, 47, 42, 41, 409, 11, 495, 0, +3, 42, 47, 48, 11, 409, 428, 0, +3, 48, 43, 42, 428, 435, 11, 0, +3, 43, 48, 49, 435, 428, 313, 0, +3, 49, 44, 43, 313, 322, 435, 0, +3, 45, 50, 51, 27, 513, 385, 0, +3, 51, 46, 45, 385, 44, 27, 0, +3, 46, 51, 52, 44, 385, 382, 0, +3, 52, 47, 46, 382, 409, 44, 0, +3, 47, 52, 53, 409, 382, 124, 0, +3, 53, 48, 47, 124, 428, 409, 0, +3, 48, 53, 54, 428, 124, 447, 0, +3, 54, 49, 48, 447, 313, 428, 0, +3, 50, 55, 56, 513, 136, 478, 0, +3, 56, 51, 50, 478, 385, 513, 0, +3, 51, 56, 57, 385, 478, 161, 0, +3, 57, 52, 51, 161, 382, 385, 0, +3, 52, 57, 58, 382, 161, 181, 0, +3, 58, 53, 52, 181, 124, 382, 0, +3, 53, 58, 59, 124, 181, 348, 0, +3, 59, 54, 53, 348, 447, 124, 0, +3, 55, 60, 61, 136, 431, 320, 0, +3, 61, 56, 55, 320, 478, 136, 0, +3, 56, 61, 62, 478, 320, 481, 0, +3, 62, 57, 56, 481, 161, 478, 0, +3, 57, 62, 63, 161, 481, 53, 0, +3, 63, 58, 57, 53, 181, 161, 0, +3, 58, 63, 64, 181, 53, 257, 0, +3, 64, 59, 58, 257, 348, 181, 0, +3, 60, 65, 66, 431, 135, 37, 0, +3, 66, 61, 60, 37, 320, 431, 0, +3, 61, 66, 67, 320, 37, 408, 0, +3, 67, 62, 61, 408, 481, 320, 0, +3, 62, 67, 68, 481, 408, 347, 0, +3, 68, 63, 62, 347, 53, 481, 0, +3, 63, 68, 69, 53, 347, 104, 0, +3, 69, 64, 63, 104, 257, 53, 0, +3, 65, 70, 71, 135, 191, 524, 0, +3, 71, 66, 65, 524, 37, 135, 0, +3, 66, 71, 72, 37, 524, 319, 0, +3, 72, 67, 66, 319, 408, 37, 0, +3, 67, 72, 73, 408, 319, 183, 0, +3, 73, 68, 67, 183, 347, 408, 0, +3, 68, 73, 74, 347, 183, 480, 0, +3, 74, 69, 68, 480, 104, 347, 0, +3, 70, 75, 76, 191, 483, 328, 0, +3, 76, 71, 70, 328, 524, 191, 0, +3, 71, 76, 77, 524, 328, 422, 0, +3, 77, 72, 71, 422, 319, 524, 0, +3, 72, 77, 78, 319, 422, 151, 0, +3, 78, 73, 72, 151, 183, 319, 0, +3, 73, 78, 79, 183, 151, 273, 0, +3, 79, 74, 73, 273, 480, 183, 0, +3, 75, 0, 1, 483, 255, 465, 0, +3, 1, 76, 75, 465, 328, 483, 0, +3, 76, 1, 2, 328, 465, 4, 0, +3, 2, 77, 76, 4, 422, 328, 0, +3, 77, 2, 3, 422, 4, 448, 0, +3, 3, 78, 77, 448, 151, 422, 0, +3, 78, 3, 4, 151, 448, 116, 0, +3, 4, 79, 78, 116, 273, 151, 0, +3, 4, 9, 84, 116, 49, 220, 0, +3, 84, 80, 4, 220, 131, 116, 0, +3, 80, 84, 85, 131, 220, 476, 0, +3, 85, 81, 80, 476, 26, 131, 0, +3, 81, 85, 86, 26, 476, 38, 0, +3, 86, 82, 81, 38, 336, 26, 0, +3, 82, 86, 87, 336, 38, 511, 0, +3, 87, 83, 82, 511, 1, 336, 0, +3, 9, 14, 88, 49, 180, 103, 0, +3, 88, 84, 9, 103, 220, 49, 0, +3, 84, 88, 89, 220, 103, 62, 0, +3, 89, 85, 84, 62, 476, 220, 0, +3, 85, 89, 90, 476, 62, 488, 0, +3, 90, 86, 85, 488, 38, 476, 0, +3, 86, 90, 91, 38, 488, 484, 0, +3, 91, 87, 86, 484, 511, 38, 0, +3, 14, 19, 92, 180, 74, 78, 0, +3, 92, 88, 14, 78, 103, 180, 0, +3, 88, 92, 93, 103, 78, 154, 0, +3, 93, 89, 88, 154, 62, 103, 0, +3, 89, 93, 94, 62, 154, 190, 0, +3, 94, 90, 89, 190, 488, 62, 0, +3, 90, 94, 95, 488, 190, 417, 0, +3, 95, 91, 90, 417, 484, 488, 0, +3, 19, 24, 96, 74, 234, 81, 0, +3, 96, 92, 19, 81, 78, 74, 0, +3, 92, 96, 97, 78, 81, 274, 0, +3, 97, 93, 92, 274, 154, 78, 0, +3, 93, 97, 98, 154, 274, 363, 0, +3, 98, 94, 93, 363, 190, 154, 0, +3, 94, 98, 99, 190, 363, 304, 0, +3, 99, 95, 94, 304, 417, 190, 0, +3, 24, 29, 100, 234, 485, 287, 0, +3, 100, 96, 24, 287, 81, 234, 0, +3, 96, 100, 101, 81, 287, 398, 0, +3, 101, 97, 96, 398, 274, 81, 0, +3, 97, 101, 102, 274, 398, 440, 0, +3, 102, 98, 97, 440, 363, 274, 0, +3, 98, 102, 103, 363, 440, 466, 0, +3, 103, 99, 98, 466, 304, 363, 0, +3, 29, 34, 104, 485, 138, 268, 0, +3, 104, 100, 29, 268, 287, 485, 0, +3, 100, 104, 105, 287, 268, 252, 0, +3, 105, 101, 100, 252, 398, 287, 0, +3, 101, 105, 106, 398, 252, 141, 0, +3, 106, 102, 101, 141, 440, 398, 0, +3, 102, 106, 107, 440, 141, 18, 0, +3, 107, 103, 102, 18, 466, 440, 0, +3, 34, 39, 108, 138, 28, 357, 0, +3, 108, 104, 34, 357, 268, 138, 0, +3, 104, 108, 109, 268, 357, 127, 0, +3, 109, 105, 104, 127, 252, 268, 0, +3, 105, 109, 110, 252, 127, 228, 0, +3, 110, 106, 105, 228, 141, 252, 0, +3, 106, 110, 111, 141, 228, 33, 0, +3, 111, 107, 106, 33, 18, 141, 0, +3, 39, 44, 112, 28, 322, 396, 0, +3, 112, 108, 39, 396, 357, 28, 0, +3, 108, 112, 113, 357, 396, 294, 0, +3, 113, 109, 108, 294, 127, 357, 0, +3, 109, 113, 114, 127, 294, 56, 0, +3, 114, 110, 109, 56, 228, 127, 0, +3, 110, 114, 115, 228, 56, 517, 0, +3, 115, 111, 110, 517, 33, 228, 0, +3, 44, 49, 116, 322, 313, 474, 0, +3, 116, 112, 44, 474, 396, 322, 0, +3, 112, 116, 117, 396, 474, 208, 0, +3, 117, 113, 112, 208, 294, 396, 0, +3, 113, 117, 118, 294, 208, 301, 0, +3, 118, 114, 113, 301, 56, 294, 0, +3, 114, 118, 119, 56, 301, 242, 0, +3, 119, 115, 114, 242, 517, 56, 0, +3, 49, 54, 120, 313, 447, 377, 0, +3, 120, 116, 49, 377, 474, 313, 0, +3, 116, 120, 121, 474, 377, 333, 0, +3, 121, 117, 116, 333, 208, 474, 0, +3, 117, 121, 122, 208, 333, 222, 0, +3, 122, 118, 117, 222, 301, 208, 0, +3, 118, 122, 123, 301, 222, 218, 0, +3, 123, 119, 118, 218, 242, 301, 0, +3, 54, 59, 124, 447, 348, 350, 0, +3, 124, 120, 54, 350, 377, 447, 0, +3, 120, 124, 125, 377, 350, 420, 0, +3, 125, 121, 120, 420, 333, 377, 0, +3, 121, 125, 126, 333, 420, 453, 0, +3, 126, 122, 121, 453, 222, 333, 0, +3, 122, 126, 127, 222, 453, 147, 0, +3, 127, 123, 122, 147, 218, 222, 0, +3, 59, 64, 128, 348, 257, 95, 0, +3, 128, 124, 59, 95, 350, 348, 0, +3, 124, 128, 129, 350, 95, 293, 0, +3, 129, 125, 124, 293, 420, 350, 0, +3, 125, 129, 130, 420, 293, 378, 0, +3, 130, 126, 125, 378, 453, 420, 0, +3, 126, 130, 131, 453, 378, 29, 0, +3, 131, 127, 126, 29, 147, 453, 0, +3, 64, 69, 132, 257, 104, 311, 0, +3, 132, 128, 64, 311, 95, 257, 0, +3, 128, 132, 133, 95, 311, 419, 0, +3, 133, 129, 128, 419, 293, 95, 0, +3, 129, 133, 134, 293, 419, 463, 0, +3, 134, 130, 129, 463, 378, 293, 0, +3, 130, 134, 135, 378, 463, 490, 0, +3, 135, 131, 130, 490, 29, 378, 0, +3, 69, 74, 136, 104, 480, 284, 0, +3, 136, 132, 69, 284, 311, 104, 0, +3, 132, 136, 137, 311, 284, 269, 0, +3, 137, 133, 132, 269, 419, 311, 0, +3, 133, 137, 138, 419, 269, 164, 0, +3, 138, 134, 133, 164, 463, 419, 0, +3, 134, 138, 139, 463, 164, 45, 0, +3, 139, 135, 134, 45, 490, 463, 0, +3, 74, 79, 140, 480, 273, 371, 0, +3, 140, 136, 74, 371, 284, 480, 0, +3, 136, 140, 141, 284, 371, 148, 0, +3, 141, 137, 136, 148, 269, 284, 0, +3, 137, 141, 142, 269, 148, 251, 0, +3, 142, 138, 137, 251, 164, 269, 0, +3, 138, 142, 143, 164, 251, 54, 0, +3, 143, 139, 138, 54, 45, 164, 0, +3, 79, 4, 80, 273, 116, 131, 0, +3, 80, 140, 79, 131, 371, 273, 0, +3, 140, 80, 81, 371, 131, 26, 0, +3, 81, 141, 140, 26, 148, 371, 0, +3, 141, 81, 82, 148, 26, 336, 0, +3, 82, 142, 141, 336, 251, 148, 0, +3, 142, 82, 83, 251, 336, 1, 0, +3, 83, 143, 142, 1, 54, 251, 0, +3, 83, 87, 148, 1, 511, 404, 0, +3, 148, 144, 83, 404, 276, 1, 0, +3, 144, 148, 149, 276, 404, 308, 0, +3, 149, 145, 144, 308, 520, 276, 0, +3, 145, 149, 150, 520, 308, 325, 0, +3, 150, 146, 145, 325, 395, 520, 0, +3, 146, 150, 151, 395, 325, 384, 0, +3, 151, 147, 146, 384, 246, 395, 0, +3, 87, 91, 152, 511, 484, 47, 0, +3, 152, 148, 87, 47, 404, 511, 0, +3, 148, 152, 153, 404, 47, 272, 0, +3, 153, 149, 148, 272, 308, 404, 0, +3, 149, 153, 154, 308, 272, 415, 0, +3, 154, 150, 149, 415, 325, 308, 0, +3, 150, 154, 155, 325, 415, 83, 0, +3, 155, 151, 150, 83, 384, 325, 0, +3, 91, 95, 156, 484, 417, 430, 0, +3, 156, 152, 91, 430, 47, 484, 0, +3, 152, 156, 157, 47, 430, 137, 0, +3, 157, 153, 152, 137, 272, 47, 0, +3, 153, 157, 158, 272, 137, 416, 0, +3, 158, 154, 153, 416, 415, 272, 0, +3, 154, 158, 159, 415, 416, 297, 0, +3, 159, 155, 154, 297, 83, 415, 0, +3, 95, 99, 160, 417, 304, 458, 0, +3, 160, 156, 95, 458, 430, 417, 0, +3, 156, 160, 161, 430, 458, 343, 0, +3, 161, 157, 156, 343, 137, 430, 0, +3, 157, 161, 162, 137, 343, 334, 0, +3, 162, 158, 157, 334, 416, 137, 0, +3, 158, 162, 163, 416, 334, 317, 0, +3, 163, 159, 158, 317, 297, 416, 0, +3, 99, 103, 164, 304, 466, 187, 0, +3, 164, 160, 99, 187, 458, 304, 0, +3, 160, 164, 165, 458, 187, 117, 0, +3, 165, 161, 160, 117, 343, 458, 0, +3, 161, 165, 166, 343, 117, 438, 0, +3, 166, 162, 161, 438, 334, 343, 0, +3, 162, 166, 167, 334, 438, 459, 0, +3, 167, 163, 162, 459, 317, 334, 0, +3, 103, 107, 168, 466, 18, 353, 0, +3, 168, 164, 103, 353, 187, 466, 0, +3, 164, 168, 169, 187, 353, 123, 0, +3, 169, 165, 164, 123, 117, 187, 0, +3, 165, 169, 170, 117, 123, 168, 0, +3, 170, 166, 165, 168, 438, 117, 0, +3, 166, 170, 171, 438, 168, 426, 0, +3, 171, 167, 166, 426, 459, 438, 0, +3, 107, 111, 172, 18, 33, 390, 0, +3, 172, 168, 107, 390, 353, 18, 0, +3, 168, 172, 173, 353, 390, 290, 0, +3, 173, 169, 168, 290, 123, 353, 0, +3, 169, 173, 174, 123, 290, 522, 0, +3, 174, 170, 169, 522, 168, 123, 0, +3, 170, 174, 175, 168, 522, 87, 0, +3, 175, 171, 170, 87, 426, 168, 0, +3, 111, 115, 176, 33, 517, 260, 0, +3, 176, 172, 111, 260, 390, 33, 0, +3, 172, 176, 177, 390, 260, 497, 0, +3, 177, 173, 172, 497, 290, 390, 0, +3, 173, 177, 178, 290, 497, 126, 0, +3, 178, 174, 173, 126, 522, 290, 0, +3, 174, 178, 179, 522, 126, 501, 0, +3, 179, 175, 174, 501, 87, 522, 0, +3, 115, 119, 180, 517, 242, 130, 0, +3, 180, 176, 115, 130, 260, 517, 0, +3, 176, 180, 181, 260, 130, 34, 0, +3, 181, 177, 176, 34, 497, 260, 0, +3, 177, 181, 182, 497, 34, 46, 0, +3, 182, 178, 177, 46, 126, 497, 0, +3, 178, 182, 183, 126, 46, 105, 0, +3, 183, 179, 178, 105, 501, 126, 0, +3, 119, 123, 184, 242, 218, 310, 0, +3, 184, 180, 119, 310, 130, 242, 0, +3, 180, 184, 185, 130, 310, 528, 0, +3, 185, 181, 180, 528, 34, 130, 0, +3, 181, 185, 186, 34, 528, 145, 0, +3, 186, 182, 181, 145, 46, 34, 0, +3, 182, 186, 187, 46, 145, 356, 0, +3, 187, 183, 182, 356, 105, 46, 0, +3, 123, 127, 188, 218, 147, 156, 0, +3, 188, 184, 123, 156, 310, 218, 0, +3, 184, 188, 189, 310, 156, 402, 0, +3, 189, 185, 184, 402, 528, 310, 0, +3, 185, 189, 190, 528, 402, 146, 0, +3, 190, 186, 185, 146, 145, 528, 0, +3, 186, 190, 191, 145, 146, 17, 0, +3, 191, 187, 186, 17, 356, 145, 0, +3, 127, 131, 192, 147, 29, 184, 0, +3, 192, 188, 127, 184, 156, 147, 0, +3, 188, 192, 193, 156, 184, 63, 0, +3, 193, 189, 188, 63, 402, 156, 0, +3, 189, 193, 194, 402, 63, 354, 0, +3, 194, 190, 189, 354, 146, 402, 0, +3, 190, 194, 195, 146, 354, 335, 0, +3, 195, 191, 190, 335, 17, 146, 0, +3, 131, 135, 196, 29, 490, 210, 0, +3, 196, 192, 131, 210, 184, 29, 0, +3, 192, 196, 197, 184, 210, 129, 0, +3, 197, 193, 192, 129, 63, 184, 0, +3, 193, 197, 198, 63, 129, 461, 0, +3, 198, 194, 193, 461, 354, 63, 0, +3, 194, 198, 199, 354, 461, 475, 0, +3, 199, 195, 194, 475, 335, 354, 0, +3, 135, 139, 200, 490, 45, 370, 0, +3, 200, 196, 135, 370, 210, 490, 0, +3, 196, 200, 201, 210, 370, 143, 0, +3, 201, 197, 196, 143, 129, 210, 0, +3, 197, 201, 202, 129, 143, 195, 0, +3, 202, 198, 197, 195, 461, 129, 0, +3, 198, 202, 203, 461, 195, 444, 0, +3, 203, 199, 198, 444, 475, 461, 0, +3, 139, 143, 204, 45, 54, 403, 0, +3, 204, 200, 139, 403, 370, 45, 0, +3, 200, 204, 205, 370, 403, 315, 0, +3, 205, 201, 200, 315, 143, 370, 0, +3, 201, 205, 206, 143, 315, 7, 0, +3, 206, 202, 201, 7, 195, 143, 0, +3, 202, 206, 207, 195, 7, 101, 0, +3, 207, 203, 202, 101, 444, 195, 0, +3, 143, 83, 144, 54, 1, 276, 0, +3, 144, 204, 143, 276, 403, 54, 0, +3, 204, 144, 145, 403, 276, 520, 0, +3, 145, 205, 204, 520, 315, 403, 0, +3, 205, 145, 146, 315, 520, 395, 0, +3, 146, 206, 205, 395, 7, 315, 0, +3, 206, 146, 147, 7, 395, 246, 0, +3, 147, 207, 206, 246, 101, 7, 0, +3, 147, 151, 212, 246, 384, 486, 0, +3, 212, 208, 147, 486, 279, 246, 0, +3, 208, 212, 213, 279, 486, 231, 0, +3, 213, 209, 208, 231, 349, 279, 0, +3, 209, 213, 214, 349, 231, 0, 0, +3, 214, 210, 209, 0, 216, 349, 0, +3, 210, 214, 211, 216, 0, 393, 0, +3, 211, 211, 210, 393, 393, 216, 0, +3, 151, 155, 215, 384, 83, 215, 0, +3, 215, 212, 151, 215, 486, 384, 0, +3, 212, 215, 216, 486, 215, 327, 0, +3, 216, 213, 212, 327, 231, 486, 0, +3, 213, 216, 217, 231, 327, 512, 0, +3, 217, 214, 213, 512, 0, 231, 0, +3, 214, 217, 211, 0, 512, 393, 0, +3, 211, 211, 214, 393, 393, 0, 0, +3, 155, 159, 218, 83, 297, 149, 0, +3, 218, 215, 155, 149, 215, 83, 0, +3, 215, 218, 219, 215, 149, 91, 0, +3, 219, 216, 215, 91, 327, 215, 0, +3, 216, 219, 220, 327, 91, 177, 0, +3, 220, 217, 216, 177, 512, 327, 0, +3, 217, 220, 211, 512, 177, 393, 0, +3, 211, 211, 217, 393, 393, 512, 0, +3, 159, 163, 221, 297, 317, 504, 0, +3, 221, 218, 159, 504, 149, 297, 0, +3, 218, 221, 222, 149, 504, 285, 0, +3, 222, 219, 218, 285, 91, 149, 0, +3, 219, 222, 223, 91, 285, 254, 0, +3, 223, 220, 219, 254, 177, 91, 0, +3, 220, 223, 211, 177, 254, 393, 0, +3, 211, 211, 220, 393, 393, 177, 0, +3, 163, 167, 224, 317, 459, 125, 0, +3, 224, 221, 163, 125, 504, 317, 0, +3, 221, 224, 225, 504, 125, 162, 0, +3, 225, 222, 221, 162, 285, 504, 0, +3, 222, 225, 226, 285, 162, 278, 0, +3, 226, 223, 222, 278, 254, 285, 0, +3, 223, 226, 211, 254, 278, 393, 0, +3, 211, 211, 223, 393, 393, 254, 0, +3, 167, 171, 227, 459, 426, 439, 0, +3, 227, 224, 167, 439, 125, 459, 0, +3, 224, 227, 228, 125, 439, 60, 0, +3, 228, 225, 224, 60, 162, 125, 0, +3, 225, 228, 229, 162, 60, 446, 0, +3, 229, 226, 225, 446, 278, 162, 0, +3, 226, 229, 211, 278, 446, 393, 0, +3, 211, 211, 226, 393, 393, 278, 0, +3, 171, 175, 230, 426, 87, 482, 0, +3, 230, 227, 171, 482, 439, 426, 0, +3, 227, 230, 231, 439, 482, 92, 0, +3, 231, 228, 227, 92, 60, 439, 0, +3, 228, 231, 232, 60, 92, 110, 0, +3, 232, 229, 228, 110, 446, 60, 0, +3, 229, 232, 211, 446, 110, 393, 0, +3, 211, 211, 229, 393, 393, 446, 0, +3, 175, 179, 233, 87, 501, 261, 0, +3, 233, 230, 175, 261, 482, 87, 0, +3, 230, 233, 234, 482, 261, 329, 0, +3, 234, 231, 230, 329, 92, 482, 0, +3, 231, 234, 235, 92, 329, 192, 0, +3, 235, 232, 231, 192, 110, 92, 0, +3, 232, 235, 211, 110, 192, 393, 0, +3, 211, 211, 232, 393, 393, 110, 0, +3, 179, 183, 236, 501, 105, 219, 0, +3, 236, 233, 179, 219, 261, 501, 0, +3, 233, 236, 237, 261, 219, 491, 0, +3, 237, 234, 233, 491, 329, 261, 0, +3, 234, 237, 238, 329, 491, 267, 0, +3, 238, 235, 234, 267, 192, 329, 0, +3, 235, 238, 211, 192, 267, 393, 0, +3, 211, 211, 235, 393, 393, 192, 0, +3, 183, 187, 239, 105, 356, 472, 0, +3, 239, 236, 183, 472, 219, 105, 0, +3, 236, 239, 240, 219, 472, 48, 0, +3, 240, 237, 236, 48, 491, 219, 0, +3, 237, 240, 241, 491, 48, 247, 0, +3, 241, 238, 237, 247, 267, 491, 0, +3, 238, 241, 211, 267, 247, 393, 0, +3, 211, 211, 238, 393, 393, 267, 0, +3, 187, 191, 242, 356, 17, 411, 0, +3, 242, 239, 187, 411, 472, 356, 0, +3, 239, 242, 243, 472, 411, 364, 0, +3, 243, 240, 239, 364, 48, 472, 0, +3, 240, 243, 244, 48, 364, 441, 0, +3, 244, 241, 240, 441, 247, 48, 0, +3, 241, 244, 211, 247, 441, 393, 0, +3, 211, 211, 241, 393, 393, 247, 0, +3, 191, 195, 245, 17, 335, 239, 0, +3, 245, 242, 191, 239, 411, 17, 0, +3, 242, 245, 246, 411, 239, 13, 0, +3, 246, 243, 242, 13, 364, 411, 0, +3, 243, 246, 247, 364, 13, 509, 0, +3, 247, 244, 243, 509, 441, 364, 0, +3, 244, 247, 211, 441, 509, 393, 0, +3, 211, 211, 244, 393, 393, 441, 0, +3, 195, 199, 248, 335, 475, 144, 0, +3, 248, 245, 195, 144, 239, 335, 0, +3, 245, 248, 249, 239, 144, 179, 0, +3, 249, 246, 245, 179, 13, 239, 0, +3, 246, 249, 250, 13, 179, 298, 0, +3, 250, 247, 246, 298, 509, 13, 0, +3, 247, 250, 211, 509, 298, 393, 0, +3, 211, 211, 247, 393, 393, 509, 0, +3, 199, 203, 251, 475, 444, 462, 0, +3, 251, 248, 199, 462, 144, 475, 0, +3, 248, 251, 252, 144, 462, 76, 0, +3, 252, 249, 248, 76, 179, 144, 0, +3, 249, 252, 253, 179, 76, 464, 0, +3, 253, 250, 249, 464, 298, 179, 0, +3, 250, 253, 211, 298, 464, 393, 0, +3, 211, 211, 250, 393, 393, 298, 0, +3, 203, 207, 254, 444, 101, 500, 0, +3, 254, 251, 203, 500, 462, 444, 0, +3, 251, 254, 255, 462, 500, 113, 0, +3, 255, 252, 251, 113, 76, 462, 0, +3, 252, 255, 256, 76, 113, 128, 0, +3, 256, 253, 252, 128, 464, 76, 0, +3, 253, 256, 211, 464, 128, 393, 0, +3, 211, 211, 253, 393, 393, 464, 0, +3, 207, 147, 208, 101, 246, 279, 0, +3, 208, 254, 207, 279, 500, 101, 0, +3, 254, 208, 209, 500, 279, 349, 0, +3, 209, 255, 254, 349, 113, 500, 0, +3, 255, 209, 210, 113, 349, 216, 0, +3, 210, 256, 255, 216, 128, 113, 0, +3, 256, 210, 211, 128, 216, 393, 0, +3, 211, 211, 256, 393, 393, 128, 0, +3, 257, 262, 263, 425, 244, 58, 0, +3, 263, 258, 257, 58, 337, 425, 0, +3, 258, 263, 264, 337, 58, 214, 0, +3, 264, 259, 258, 214, 236, 337, 0, +3, 259, 264, 265, 236, 214, 266, 0, +3, 265, 260, 259, 266, 32, 236, 0, +3, 260, 265, 266, 32, 266, 331, 0, +3, 266, 261, 260, 331, 109, 32, 0, +3, 262, 267, 268, 244, 233, 369, 0, +3, 268, 263, 262, 369, 58, 244, 0, +3, 263, 268, 269, 58, 369, 71, 0, +3, 269, 264, 263, 71, 214, 58, 0, +3, 264, 269, 270, 214, 71, 392, 0, +3, 270, 265, 264, 392, 266, 214, 0, +3, 265, 270, 271, 266, 392, 312, 0, +3, 271, 266, 265, 312, 331, 266, 0, +3, 267, 272, 273, 233, 12, 434, 0, +3, 273, 268, 267, 434, 369, 233, 0, +3, 268, 273, 274, 369, 434, 188, 0, +3, 274, 269, 268, 188, 71, 369, 0, +3, 269, 274, 275, 71, 188, 201, 0, +3, 275, 270, 269, 201, 392, 71, 0, +3, 270, 275, 276, 392, 201, 238, 0, +3, 276, 271, 270, 238, 312, 392, 0, +3, 272, 277, 278, 12, 142, 114, 0, +3, 278, 273, 272, 114, 434, 12, 0, +3, 273, 278, 279, 434, 114, 173, 0, +3, 279, 274, 273, 173, 188, 434, 0, +3, 274, 279, 280, 188, 173, 14, 0, +3, 280, 275, 274, 14, 201, 188, 0, +3, 275, 280, 281, 201, 14, 15, 0, +3, 281, 276, 275, 15, 238, 201, 0, +3, 277, 282, 283, 142, 407, 288, 0, +3, 283, 278, 277, 288, 114, 142, 0, +3, 278, 283, 284, 114, 288, 400, 0, +3, 284, 279, 278, 400, 173, 114, 0, +3, 279, 284, 285, 173, 400, 457, 0, +3, 285, 280, 279, 457, 14, 173, 0, +3, 280, 285, 286, 14, 457, 332, 0, +3, 286, 281, 280, 332, 15, 14, 0, +3, 282, 287, 288, 407, 194, 42, 0, +3, 288, 283, 282, 42, 288, 407, 0, +3, 283, 288, 289, 288, 42, 380, 0, +3, 289, 284, 283, 380, 400, 288, 0, +3, 284, 289, 290, 400, 380, 383, 0, +3, 290, 285, 284, 383, 457, 400, 0, +3, 285, 290, 291, 457, 383, 197, 0, +3, 291, 286, 285, 197, 332, 457, 0, +3, 287, 292, 293, 194, 321, 152, 0, +3, 293, 288, 287, 152, 42, 194, 0, +3, 288, 293, 294, 42, 152, 397, 0, +3, 294, 289, 288, 397, 380, 42, 0, +3, 289, 294, 295, 380, 397, 342, 0, +3, 295, 290, 289, 342, 383, 380, 0, +3, 290, 295, 296, 383, 342, 225, 0, +3, 296, 291, 290, 225, 197, 383, 0, +3, 292, 257, 258, 321, 425, 337, 0, +3, 258, 293, 292, 337, 152, 321, 0, +3, 293, 258, 259, 152, 337, 236, 0, +3, 259, 294, 293, 236, 397, 152, 0, +3, 294, 259, 260, 397, 236, 32, 0, +3, 260, 295, 294, 32, 342, 397, 0, +3, 295, 260, 261, 342, 32, 109, 0, +3, 261, 296, 295, 109, 225, 342, 0, +3, 261, 266, 301, 109, 331, 175, 0, +3, 301, 297, 261, 175, 502, 109, 0, +3, 297, 301, 302, 502, 175, 265, 0, +3, 302, 298, 297, 265, 84, 502, 0, +3, 298, 302, 303, 84, 265, 186, 0, +3, 303, 299, 298, 186, 496, 84, 0, +3, 299, 303, 304, 496, 186, 470, 0, +3, 304, 300, 299, 470, 494, 496, 0, +3, 266, 271, 305, 331, 312, 170, 0, +3, 305, 301, 266, 170, 175, 331, 0, +3, 301, 305, 306, 175, 170, 97, 0, +3, 306, 302, 301, 97, 265, 175, 0, +3, 302, 306, 307, 265, 97, 205, 0, +3, 307, 303, 302, 205, 186, 265, 0, +3, 303, 307, 308, 186, 205, 449, 0, +3, 308, 304, 303, 449, 470, 186, 0, +3, 271, 276, 309, 312, 238, 379, 0, +3, 309, 305, 271, 379, 170, 312, 0, +3, 305, 309, 310, 170, 379, 300, 0, +3, 310, 306, 305, 300, 97, 170, 0, +3, 306, 310, 311, 97, 300, 118, 0, +3, 311, 307, 306, 118, 205, 97, 0, +3, 307, 311, 312, 205, 118, 237, 0, +3, 312, 308, 307, 237, 449, 205, 0, +3, 276, 281, 313, 238, 15, 199, 0, +3, 313, 309, 276, 199, 379, 238, 0, +3, 309, 313, 314, 379, 199, 94, 0, +3, 314, 310, 309, 94, 300, 379, 0, +3, 310, 314, 315, 300, 94, 421, 0, +3, 315, 311, 310, 421, 118, 300, 0, +3, 311, 315, 316, 118, 421, 31, 0, +3, 316, 312, 311, 31, 237, 118, 0, +3, 281, 286, 317, 15, 332, 367, 0, +3, 317, 313, 281, 367, 199, 15, 0, +3, 313, 317, 318, 199, 367, 529, 0, +3, 318, 314, 313, 529, 94, 199, 0, +3, 314, 318, 319, 94, 529, 185, 0, +3, 319, 315, 314, 185, 421, 94, 0, +3, 315, 319, 320, 421, 185, 89, 0, +3, 320, 316, 315, 89, 31, 421, 0, +3, 286, 291, 321, 332, 197, 172, 0, +3, 321, 317, 286, 172, 367, 332, 0, +3, 317, 321, 322, 367, 172, 209, 0, +3, 322, 318, 317, 209, 529, 367, 0, +3, 318, 322, 323, 529, 209, 429, 0, +3, 323, 319, 318, 429, 185, 529, 0, +3, 319, 323, 324, 185, 429, 112, 0, +3, 324, 320, 319, 112, 89, 185, 0, +3, 291, 296, 325, 197, 225, 451, 0, +3, 325, 321, 291, 451, 172, 197, 0, +3, 321, 325, 326, 172, 451, 66, 0, +3, 326, 322, 321, 66, 209, 172, 0, +3, 322, 326, 327, 209, 66, 176, 0, +3, 327, 323, 322, 176, 429, 209, 0, +3, 323, 327, 328, 429, 176, 155, 0, +3, 328, 324, 323, 155, 112, 429, 0, +3, 296, 261, 297, 225, 109, 502, 0, +3, 297, 325, 296, 502, 451, 225, 0, +3, 325, 297, 298, 451, 502, 84, 0, +3, 298, 326, 325, 84, 66, 451, 0, +3, 326, 298, 299, 66, 84, 496, 0, +3, 299, 327, 326, 496, 176, 66, 0, +3, 327, 299, 300, 176, 496, 494, 0, +3, 300, 328, 327, 494, 155, 176, 0, +3, 329, 334, 335, 3, 355, 122, 0, +3, 335, 330, 329, 122, 518, 3, 0, +3, 330, 335, 336, 518, 122, 111, 0, +3, 336, 331, 330, 111, 213, 518, 0, +3, 331, 336, 337, 213, 111, 473, 0, +3, 337, 332, 331, 473, 468, 213, 0, +3, 332, 337, 338, 468, 473, 521, 0, +3, 338, 333, 332, 521, 346, 468, 0, +3, 334, 339, 340, 355, 61, 414, 0, +3, 340, 335, 334, 414, 122, 355, 0, +3, 335, 340, 341, 122, 414, 413, 0, +3, 341, 336, 335, 413, 111, 122, 0, +3, 336, 341, 342, 111, 413, 204, 0, +3, 342, 337, 336, 204, 473, 111, 0, +3, 337, 342, 343, 473, 204, 217, 0, +3, 343, 338, 337, 217, 521, 473, 0, +3, 339, 344, 345, 61, 55, 100, 0, +3, 345, 340, 339, 100, 414, 61, 0, +3, 340, 345, 346, 414, 100, 399, 0, +3, 346, 341, 340, 399, 413, 414, 0, +3, 341, 346, 347, 413, 399, 326, 0, +3, 347, 342, 341, 326, 204, 413, 0, +3, 342, 347, 348, 204, 326, 221, 0, +3, 348, 343, 342, 221, 217, 204, 0, +3, 344, 349, 350, 55, 508, 477, 0, +3, 350, 345, 344, 477, 100, 55, 0, +3, 345, 350, 351, 100, 477, 292, 0, +3, 351, 346, 345, 292, 399, 100, 0, +3, 346, 351, 352, 399, 292, 73, 0, +3, 352, 347, 346, 73, 326, 399, 0, +3, 347, 352, 353, 326, 73, 362, 0, +3, 353, 348, 347, 362, 221, 326, 0, +3, 349, 354, 355, 508, 365, 262, 0, +3, 355, 350, 349, 262, 477, 508, 0, +3, 350, 355, 356, 477, 262, 93, 0, +3, 356, 351, 350, 93, 292, 477, 0, +3, 351, 356, 357, 292, 93, 318, 0, +3, 357, 352, 351, 318, 73, 292, 0, +3, 352, 357, 358, 73, 318, 163, 0, +3, 358, 353, 352, 163, 362, 73, 0, +3, 354, 359, 360, 365, 140, 340, 0, +3, 360, 355, 354, 340, 262, 365, 0, +3, 355, 360, 361, 262, 340, 505, 0, +3, 361, 356, 355, 505, 93, 262, 0, +3, 356, 361, 362, 93, 505, 499, 0, +3, 362, 357, 356, 499, 318, 93, 0, +3, 357, 362, 363, 318, 499, 159, 0, +3, 363, 358, 357, 159, 163, 318, 0, +3, 359, 364, 365, 140, 510, 68, 0, +3, 365, 360, 359, 68, 340, 140, 0, +3, 360, 365, 366, 340, 68, 167, 0, +3, 366, 361, 360, 167, 505, 340, 0, +3, 361, 366, 367, 505, 167, 245, 0, +3, 367, 362, 361, 245, 499, 505, 0, +3, 362, 367, 368, 499, 245, 437, 0, +3, 368, 363, 362, 437, 159, 499, 0, +3, 364, 329, 330, 510, 3, 518, 0, +3, 330, 365, 364, 518, 68, 510, 0, +3, 365, 330, 331, 68, 518, 213, 0, +3, 331, 366, 365, 213, 167, 68, 0, +3, 366, 331, 332, 167, 213, 468, 0, +3, 332, 367, 366, 468, 245, 167, 0, +3, 367, 332, 333, 245, 468, 346, 0, +3, 333, 368, 367, 346, 437, 245, 0, +3, 333, 338, 373, 346, 521, 79, 0, +3, 373, 369, 333, 79, 286, 346, 0, +3, 369, 373, 374, 286, 79, 77, 0, +3, 374, 370, 369, 77, 22, 286, 0, +3, 370, 374, 375, 22, 77, 523, 0, +3, 375, 371, 370, 523, 330, 22, 0, +3, 371, 375, 376, 330, 523, 259, 0, +3, 376, 372, 371, 259, 338, 330, 0, +3, 338, 343, 377, 521, 217, 207, 0, +3, 377, 373, 338, 207, 79, 521, 0, +3, 373, 377, 378, 79, 207, 471, 0, +3, 378, 374, 373, 471, 77, 79, 0, +3, 374, 378, 379, 77, 471, 198, 0, +3, 379, 375, 374, 198, 523, 77, 0, +3, 375, 379, 380, 523, 198, 366, 0, +3, 380, 376, 375, 366, 259, 523, 0, +3, 343, 348, 381, 217, 221, 516, 0, +3, 381, 377, 343, 516, 207, 217, 0, +3, 377, 381, 382, 207, 516, 250, 0, +3, 382, 378, 377, 250, 471, 207, 0, +3, 378, 382, 383, 471, 250, 240, 0, +3, 383, 379, 378, 240, 198, 471, 0, +3, 379, 383, 384, 198, 240, 381, 0, +3, 384, 380, 379, 381, 366, 198, 0, +3, 348, 353, 385, 221, 362, 230, 0, +3, 385, 381, 348, 230, 516, 221, 0, +3, 381, 385, 386, 516, 230, 303, 0, +3, 386, 382, 381, 303, 250, 516, 0, +3, 382, 386, 387, 250, 303, 10, 0, +3, 387, 383, 382, 10, 240, 250, 0, +3, 383, 387, 388, 240, 10, 283, 0, +3, 388, 384, 383, 283, 381, 240, 0, +3, 353, 358, 389, 362, 163, 282, 0, +3, 389, 385, 353, 282, 230, 362, 0, +3, 385, 389, 390, 230, 282, 35, 0, +3, 390, 386, 385, 35, 303, 230, 0, +3, 386, 390, 391, 303, 35, 243, 0, +3, 391, 387, 386, 243, 10, 303, 0, +3, 387, 391, 392, 10, 243, 368, 0, +3, 392, 388, 387, 368, 283, 10, 0, +3, 358, 363, 393, 163, 159, 296, 0, +3, 393, 389, 358, 296, 282, 163, 0, +3, 389, 393, 394, 282, 296, 160, 0, +3, 394, 390, 389, 160, 35, 282, 0, +3, 390, 394, 395, 35, 160, 323, 0, +3, 395, 391, 390, 323, 243, 35, 0, +3, 391, 395, 396, 243, 323, 280, 0, +3, 396, 392, 391, 280, 368, 243, 0, +3, 363, 368, 397, 159, 437, 275, 0, +3, 397, 393, 363, 275, 296, 159, 0, +3, 393, 397, 398, 296, 275, 133, 0, +3, 398, 394, 393, 133, 160, 296, 0, +3, 394, 398, 399, 160, 133, 344, 0, +3, 399, 395, 394, 344, 323, 160, 0, +3, 395, 399, 400, 323, 344, 108, 0, +3, 400, 396, 395, 108, 280, 323, 0, +3, 368, 333, 369, 437, 346, 286, 0, +3, 369, 397, 368, 286, 275, 437, 0, +3, 397, 369, 370, 275, 286, 22, 0, +3, 370, 398, 397, 22, 133, 275, 0, +3, 398, 370, 371, 133, 22, 330, 0, +3, 371, 399, 398, 330, 344, 133, 0, +3, 399, 371, 372, 344, 330, 338, 0, +3, 372, 400, 399, 338, 108, 344, 0, +3, 401, 401, 406, 235, 235, 189, 0, +3, 406, 402, 401, 189, 40, 235, 0, +3, 402, 406, 407, 40, 189, 306, 0, +3, 407, 403, 402, 306, 119, 40, 0, +3, 403, 407, 408, 119, 306, 202, 0, +3, 408, 404, 403, 202, 443, 119, 0, +3, 404, 408, 409, 443, 202, 241, 0, +3, 409, 405, 404, 241, 75, 443, 0, +3, 401, 401, 410, 235, 235, 263, 0, +3, 410, 406, 401, 263, 189, 235, 0, +3, 406, 410, 411, 189, 263, 196, 0, +3, 411, 407, 406, 196, 306, 189, 0, +3, 407, 411, 412, 306, 196, 281, 0, +3, 412, 408, 407, 281, 202, 306, 0, +3, 408, 412, 413, 202, 281, 121, 0, +3, 413, 409, 408, 121, 241, 202, 0, +3, 401, 401, 414, 235, 235, 479, 0, +3, 414, 410, 401, 479, 263, 235, 0, +3, 410, 414, 415, 263, 479, 36, 0, +3, 415, 411, 410, 36, 196, 263, 0, +3, 411, 415, 416, 196, 36, 436, 0, +3, 416, 412, 411, 436, 281, 196, 0, +3, 412, 416, 417, 281, 436, 351, 0, +3, 417, 413, 412, 351, 121, 281, 0, +3, 401, 401, 418, 235, 235, 90, 0, +3, 418, 414, 401, 90, 479, 235, 0, +3, 414, 418, 419, 479, 90, 361, 0, +3, 419, 415, 414, 361, 36, 479, 0, +3, 415, 419, 420, 36, 361, 376, 0, +3, 420, 416, 415, 376, 436, 36, 0, +3, 416, 420, 421, 436, 376, 412, 0, +3, 421, 417, 416, 412, 351, 436, 0, +3, 401, 401, 422, 235, 235, 52, 0, +3, 422, 418, 401, 52, 90, 235, 0, +3, 418, 422, 423, 90, 52, 21, 0, +3, 423, 419, 418, 21, 361, 90, 0, +3, 419, 423, 424, 361, 21, 158, 0, +3, 424, 420, 419, 158, 376, 361, 0, +3, 420, 424, 425, 376, 158, 39, 0, +3, 425, 421, 420, 39, 412, 376, 0, +3, 401, 401, 426, 235, 235, 424, 0, +3, 426, 422, 401, 424, 52, 235, 0, +3, 422, 426, 427, 52, 424, 373, 0, +3, 427, 423, 422, 373, 21, 52, 0, +3, 423, 427, 428, 21, 373, 375, 0, +3, 428, 424, 423, 375, 158, 21, 0, +3, 424, 428, 429, 158, 375, 249, 0, +3, 429, 425, 424, 249, 39, 158, 0, +3, 401, 401, 430, 235, 235, 432, 0, +3, 430, 426, 401, 432, 424, 235, 0, +3, 426, 430, 431, 424, 432, 229, 0, +3, 431, 427, 426, 229, 373, 424, 0, +3, 427, 431, 432, 373, 229, 65, 0, +3, 432, 428, 427, 65, 375, 373, 0, +3, 428, 432, 433, 375, 65, 506, 0, +3, 433, 429, 428, 506, 249, 375, 0, +3, 401, 401, 434, 235, 235, 302, 0, +3, 434, 430, 401, 302, 432, 235, 0, +3, 430, 434, 435, 432, 302, 96, 0, +3, 435, 431, 430, 96, 229, 432, 0, +3, 431, 435, 436, 229, 96, 169, 0, +3, 436, 432, 431, 169, 65, 229, 0, +3, 432, 436, 437, 65, 169, 59, 0, +3, 437, 433, 432, 59, 506, 65, 0, +3, 401, 401, 438, 235, 235, 452, 0, +3, 438, 434, 401, 452, 302, 235, 0, +3, 434, 438, 439, 302, 452, 30, 0, +3, 439, 435, 434, 30, 96, 302, 0, +3, 435, 439, 440, 96, 30, 460, 0, +3, 440, 436, 435, 460, 169, 96, 0, +3, 436, 440, 441, 169, 460, 498, 0, +3, 441, 437, 436, 498, 59, 169, 0, +3, 401, 401, 442, 235, 235, 525, 0, +3, 442, 438, 401, 525, 452, 235, 0, +3, 438, 442, 443, 452, 525, 456, 0, +3, 443, 439, 438, 456, 30, 452, 0, +3, 439, 443, 444, 30, 456, 9, 0, +3, 444, 440, 439, 9, 460, 30, 0, +3, 440, 444, 445, 460, 9, 388, 0, +3, 445, 441, 440, 388, 498, 460, 0, +3, 401, 401, 446, 235, 235, 212, 0, +3, 446, 442, 401, 212, 525, 235, 0, +3, 442, 446, 447, 525, 212, 299, 0, +3, 447, 443, 442, 299, 456, 525, 0, +3, 443, 447, 448, 456, 299, 166, 0, +3, 448, 444, 443, 166, 9, 456, 0, +3, 444, 448, 449, 9, 166, 72, 0, +3, 449, 445, 444, 72, 388, 9, 0, +3, 401, 401, 450, 235, 235, 107, 0, +3, 450, 446, 401, 107, 212, 235, 0, +3, 446, 450, 451, 212, 107, 82, 0, +3, 451, 447, 446, 82, 299, 212, 0, +3, 447, 451, 452, 299, 82, 391, 0, +3, 452, 448, 447, 391, 166, 299, 0, +3, 448, 452, 453, 166, 391, 139, 0, +3, 453, 449, 448, 139, 72, 166, 0, +3, 401, 401, 454, 235, 235, 70, 0, +3, 454, 450, 401, 70, 107, 235, 0, +3, 450, 454, 455, 107, 70, 51, 0, +3, 455, 451, 450, 51, 82, 107, 0, +3, 451, 455, 456, 82, 51, 178, 0, +3, 456, 452, 451, 178, 391, 82, 0, +3, 452, 456, 457, 391, 178, 57, 0, +3, 457, 453, 452, 57, 139, 391, 0, +3, 401, 401, 458, 235, 235, 442, 0, +3, 458, 454, 401, 442, 70, 235, 0, +3, 454, 458, 459, 70, 442, 387, 0, +3, 459, 455, 454, 387, 51, 70, 0, +3, 455, 459, 460, 51, 387, 389, 0, +3, 460, 456, 455, 389, 178, 51, 0, +3, 456, 460, 461, 178, 389, 264, 0, +3, 461, 457, 456, 264, 57, 178, 0, +3, 401, 401, 462, 235, 235, 450, 0, +3, 462, 458, 401, 450, 442, 235, 0, +3, 458, 462, 463, 442, 450, 253, 0, +3, 463, 459, 458, 253, 387, 442, 0, +3, 459, 463, 464, 387, 253, 86, 0, +3, 464, 460, 459, 86, 389, 387, 0, +3, 460, 464, 465, 389, 86, 526, 0, +3, 465, 461, 460, 526, 264, 389, 0, +3, 401, 401, 402, 235, 235, 40, 0, +3, 402, 462, 401, 40, 450, 235, 0, +3, 462, 402, 403, 450, 40, 119, 0, +3, 403, 463, 462, 119, 253, 450, 0, +3, 463, 403, 404, 253, 119, 443, 0, +3, 404, 464, 463, 443, 86, 253, 0, +3, 464, 404, 405, 86, 443, 75, 0, +3, 405, 465, 464, 75, 526, 86, 0, +3, 405, 409, 470, 75, 241, 519, 0, +3, 470, 466, 405, 519, 226, 75, 0, +3, 466, 470, 471, 226, 519, 406, 0, +3, 471, 467, 466, 406, 98, 226, 0, +3, 467, 471, 472, 98, 406, 232, 0, +3, 472, 468, 467, 232, 43, 98, 0, +3, 468, 472, 473, 43, 232, 345, 0, +3, 473, 469, 468, 345, 372, 43, 0, +3, 409, 413, 474, 241, 121, 227, 0, +3, 474, 470, 409, 227, 519, 241, 0, +3, 470, 474, 475, 519, 227, 469, 0, +3, 475, 471, 470, 469, 406, 519, 0, +3, 471, 475, 476, 406, 469, 258, 0, +3, 476, 472, 471, 258, 232, 406, 0, +3, 472, 476, 477, 232, 258, 271, 0, +3, 477, 473, 472, 271, 345, 232, 0, +3, 413, 417, 478, 121, 351, 157, 0, +3, 478, 474, 413, 157, 227, 121, 0, +3, 474, 478, 479, 227, 157, 80, 0, +3, 479, 475, 474, 80, 469, 227, 0, +3, 475, 479, 480, 469, 80, 489, 0, +3, 480, 476, 475, 489, 258, 469, 0, +3, 476, 480, 481, 258, 489, 277, 0, +3, 481, 477, 476, 277, 271, 258, 0, +3, 417, 421, 482, 351, 412, 153, 0, +3, 482, 478, 417, 153, 157, 351, 0, +3, 478, 482, 483, 157, 153, 324, 0, +3, 483, 479, 478, 324, 80, 157, 0, +3, 479, 483, 484, 80, 324, 339, 0, +3, 484, 480, 479, 339, 489, 80, 0, +3, 480, 484, 485, 489, 339, 88, 0, +3, 485, 481, 480, 88, 277, 489, 0, +3, 421, 425, 486, 412, 39, 6, 0, +3, 486, 482, 421, 6, 153, 412, 0, +3, 482, 486, 487, 153, 6, 8, 0, +3, 487, 483, 482, 8, 324, 153, 0, +3, 483, 487, 488, 324, 8, 16, 0, +3, 488, 484, 483, 16, 339, 324, 0, +3, 484, 488, 489, 339, 16, 289, 0, +3, 489, 485, 484, 289, 88, 339, 0, +3, 425, 429, 490, 39, 249, 99, 0, +3, 490, 486, 425, 99, 6, 39, 0, +3, 486, 490, 491, 6, 99, 200, 0, +3, 491, 487, 486, 200, 8, 6, 0, +3, 487, 491, 492, 8, 200, 150, 0, +3, 492, 488, 487, 150, 16, 8, 0, +3, 488, 492, 493, 16, 150, 493, 0, +3, 493, 489, 488, 493, 289, 16, 0, +3, 429, 433, 494, 249, 506, 291, 0, +3, 494, 490, 429, 291, 99, 249, 0, +3, 490, 494, 495, 99, 291, 64, 0, +3, 495, 491, 490, 64, 200, 99, 0, +3, 491, 495, 496, 200, 64, 19, 0, +3, 496, 492, 491, 19, 150, 200, 0, +3, 492, 496, 497, 150, 19, 433, 0, +3, 497, 493, 492, 433, 493, 150, 0, +3, 433, 437, 498, 506, 59, 203, 0, +3, 498, 494, 433, 203, 291, 506, 0, +3, 494, 498, 499, 291, 203, 374, 0, +3, 499, 495, 494, 374, 64, 291, 0, +3, 495, 499, 500, 64, 374, 307, 0, +3, 500, 496, 495, 307, 19, 64, 0, +3, 496, 500, 501, 19, 307, 358, 0, +3, 501, 497, 496, 358, 433, 19, 0, +3, 437, 441, 502, 59, 498, 256, 0, +3, 502, 498, 437, 256, 203, 59, 0, +3, 498, 502, 503, 203, 256, 132, 0, +3, 503, 499, 498, 132, 374, 203, 0, +3, 499, 503, 504, 374, 132, 492, 0, +3, 504, 500, 499, 492, 307, 374, 0, +3, 500, 504, 505, 307, 492, 67, 0, +3, 505, 501, 500, 67, 358, 307, 0, +3, 441, 445, 506, 498, 388, 487, 0, +3, 506, 502, 441, 487, 256, 498, 0, +3, 502, 506, 507, 256, 487, 206, 0, +3, 507, 503, 502, 206, 132, 256, 0, +3, 503, 507, 508, 132, 206, 515, 0, +3, 508, 504, 503, 515, 492, 132, 0, +3, 504, 508, 509, 492, 515, 527, 0, +3, 509, 505, 504, 527, 67, 492, 0, +3, 445, 449, 510, 388, 72, 423, 0, +3, 510, 506, 445, 423, 487, 388, 0, +3, 506, 510, 511, 487, 423, 352, 0, +3, 511, 507, 506, 352, 206, 487, 0, +3, 507, 511, 512, 206, 352, 224, 0, +3, 512, 508, 507, 224, 515, 206, 0, +3, 508, 512, 513, 515, 224, 2, 0, +3, 513, 509, 508, 2, 527, 515, 0, +3, 449, 453, 514, 72, 139, 418, 0, +3, 514, 510, 449, 418, 423, 72, 0, +3, 510, 514, 515, 423, 418, 341, 0, +3, 515, 511, 510, 341, 352, 423, 0, +3, 511, 515, 516, 352, 341, 359, 0, +3, 516, 512, 511, 359, 224, 352, 0, +3, 512, 516, 517, 224, 359, 360, 0, +3, 517, 513, 512, 360, 2, 224, 0, +3, 453, 457, 518, 139, 57, 24, 0, +3, 518, 514, 453, 24, 418, 139, 0, +3, 514, 518, 519, 418, 24, 25, 0, +3, 519, 515, 514, 25, 341, 418, 0, +3, 515, 519, 520, 341, 25, 41, 0, +3, 520, 516, 515, 41, 359, 341, 0, +3, 516, 520, 521, 359, 41, 314, 0, +3, 521, 517, 516, 314, 360, 359, 0, +3, 457, 461, 522, 57, 264, 120, 0, +3, 522, 518, 457, 120, 24, 57, 0, +3, 518, 522, 523, 24, 120, 223, 0, +3, 523, 519, 518, 223, 25, 24, 0, +3, 519, 523, 524, 25, 223, 171, 0, +3, 524, 520, 519, 171, 41, 25, 0, +3, 520, 524, 525, 41, 171, 514, 0, +3, 525, 521, 520, 514, 314, 41, 0, +3, 461, 465, 526, 264, 526, 316, 0, +3, 526, 522, 461, 316, 120, 264, 0, +3, 522, 526, 527, 120, 316, 85, 0, +3, 527, 523, 522, 85, 223, 120, 0, +3, 523, 527, 528, 223, 85, 50, 0, +3, 528, 524, 523, 50, 171, 223, 0, +3, 524, 528, 529, 171, 50, 454, 0, +3, 529, 525, 524, 454, 514, 171, 0, +3, 465, 405, 466, 526, 75, 226, 0, +3, 466, 526, 465, 226, 316, 526, 0, +3, 526, 466, 467, 316, 226, 98, 0, +3, 467, 527, 526, 98, 85, 316, 0, +3, 527, 467, 468, 85, 98, 43, 0, +3, 468, 528, 527, 43, 50, 85, 0, +3, 528, 468, 469, 50, 43, 372, 0, +3, 469, 529, 528, 372, 454, 50, 0 +}; + + +const int strip_vertices[] = { +508, 508, 504, 509, 504, 505, 500, 501, 496, 497, 492, 493, 488, 489, 484, 485, 480, 481, 476, 477, 472, 473, -1, +476, 475, 480, 479, 484, 483, 488, 487, 492, 491, 496, 495, 500, 499, 504, 499, 503, 498, 502, 437, 441, -1, +527, 526, 467, 466, 471, 470, 475, 474, 479, 478, 483, 482, 487, 486, 491, 490, 495, 494, 499, 494, 498, -1, +490, 490, 425, 486, 421, 482, 417, 478, 413, 474, 409, 470, 405, 466, 465, 526, 465, 461, 460, 456, 455, 451, -1, +405, 465, 464, 460, 459, 455, 454, 450, -1, +455, 451, 450, 446, 450, 401, 454, 458, 459, 463, 464, 404, 405, 404, 409, 408, 413, 412, 417, 416, 421, 420, -1, +421, 420, 425, 420, 424, 419, 423, 418, 422, 418, 401, 414, 410, 415, 411, 416, 411, 412, 407, 408, 403, 404, 403, 463, -1, +418, 418, 414, 419, 415, 420, 416, -1, +407, 403, 402, 462, -1, +403, 463, 462, 458, 462, 401, 402, 406, 407, 406, 411, 406, 410, 401, -1, +494, 494, 498, 433, 437, 432, 436, 431, 435, 430, 434, 430, 401, 426, 422, 427, 423, 428, 424, 429, 425, 490, -1, +430, 430, 426, 431, 427, 432, 428, 433, 429, 494, 490, -1, +437, 437, 441, 436, 440, 435, 439, 434, 438, 401, 442, 446, 447, 451, 452, 456, 457, 461, 522, 526, 527, -1, +452, 448, 447, -1, +510, 445, 449, 444, 448, 443, 447, 443, 442, 443, 438, 443, 439, 444, 440, 445, 441, 506, 502, 507, 503, -1, +510, 506, 445, -1, +507, 506, 511, 510, 515, 510, 514, 449, 453, 448, 453, 452, 457, -1, +527, 523, 522, 518, 457, 518, 453, 518, 514, 519, 515, -1, +523, 519, 518, -1, +504, 503, 508, 507, 512, 511, 516, 515, 520, 519, 524, 523, 528, 527, 468, 467, 472, 471, 476, 475, -1, +472, 473, 468, 469, 528, 529, 524, 525, 520, 521, 516, 517, 512, 513, 508, 509, -1, +211, 211, 214, 210, 209, -1, +212, 215, 216, 219, 220, 223, 220, 211, 217, 214, 213, 209, 213, 208, 212, 147, -1, +220, 217, 216, 213, 212, -1, +251, 251, 248, 252, 249, 253, 250, 253, 211, 256, 210, 255, 209, 254, 208, 207, 147, 206, 147, 146, 147, 151, 212, 215, -1, +206, 206, 202, 207, 203, 254, 251, 255, 252, 256, 253, -1, +223, 223, 222, 219, 218, 215, 155, 151, 150, 146, 145, 146, 205, 206, 201, 202, 197, 202, 198, 203, 199, 251, 248, -1, +145, 149, 150, 154, 155, 159, 218, 221, 222, 225, 226, 229, -1, +204, 204, 145, 144, 149, 148, 149, 153, 154, 158, 159, 163, 221, 224, 225, 228, 229, 232, 229, 211, 226, 223, 222, -1, +224, 224, 167, 163, 162, 158, 157, 153, 152, 148, 87, 148, 83, 144, 143, 204, 139, 200, 135, 196, 131, 192, -1, +82, 83, 142, 143, 138, 139, 134, 135, 130, 131, 126, 127, 122, 123, 118, 123, 119, 184, 180, 185, 181, -1, +81, 82, 141, 142, 137, 138, 133, 134, 129, 130, 125, 126, 121, 122, 117, 118, 113, 114, 109, 110, -1, +80, 81, 140, 141, 136, 137, 132, 133, 128, 129, 124, 125, 120, 121, 116, 117, 112, 113, 108, 109, -1, +4, 80, 79, 140, 74, 136, 69, 132, 64, 128, 59, 124, 54, 120, 49, 116, 44, 112, 39, 108, -1, +79, 79, 73, 74, 68, 69, 63, 64, 58, 59, 53, 54, 48, 49, 48, 43, 42, 37, 36, 31, 30, 31, 25, -1, +42, 42, 48, 47, 53, 52, 58, 57, 63, 62, 68, 67, 73, 72, 78, 77, 3, 2, 8, 7, 13, -1, +36, 36, 42, 41, 47, 46, 52, 51, 57, 56, 62, 61, 67, 66, 72, 71, 77, 76, 2, 1, 7, -1, +66, 66, 60, 61, 55, 56, 50, 51, 45, 46, 40, 41, 35, 36, 30, -1, +31, 31, 25, 26, 20, 21, 15, 16, 10, 11, 5, 6, 0, 1, 75, 76, 70, 71, 65, 66, 60, -1, +1, 1, 7, 6, 12, 11, 17, 16, 22, 21, 27, 26, 32, 31, 32, 37, 38, 43, 44, 49, -1, +7, 7, 13, 12, 18, 17, 23, 22, 28, 27, 33, 32, 33, 38, -1, +44, 44, 38, 39, 33, 34, 28, 29, 23, 24, 18, 19, 13, 14, 8, 9, 3, 4, 78, 79, 73, -1, +39, 108, 34, 104, 29, 100, 24, 96, 19, 92, 14, 88, 9, 84, 4, 84, 80, 85, 81, 86, 81, 82, -1, +108, 109, 104, 105, 100, 101, 96, 97, 92, 93, 88, 89, 84, 85, -1, +109, 110, 105, 106, 101, 102, 97, 98, 93, 94, 89, 90, 85, 86, -1, +118, 119, 114, 115, 110, 111, 106, 107, 102, 103, 98, 99, 94, 95, 90, 91, 86, 87, 82, 83, -1, +111, 115, 176, -1, +107, 111, 172, 176, 177, -1, +103, 107, 168, 172, 173, 177, 178, -1, +99, 103, 164, 168, 169, 173, 174, 178, 179, -1, +95, 99, 160, 164, 165, 169, 170, 174, 175, 179, 233, -1, +91, 95, 156, 160, 161, 165, 166, 170, 171, 175, 230, 233, 234, -1, +87, 91, 152, 156, 157, 161, 162, 166, 167, 171, 227, 230, 231, 234, 235, 234, 238, 234, 237, 233, 236, 179, -1, +185, 185, 181, 186, 182, 187, 183, 239, 236, 240, 237, 241, 238, 211, 235, 232, 231, 228, 227, 224, 167, -1, +236, 179, 183, 178, 182, 177, 181, 176, 180, 115, 119, -1, +131, 192, 127, 188, 123, 188, 184, 189, 185, 190, 186, 191, 187, 242, 239, 243, 240, 244, 241, 244, 211, 247, -1, +192, 192, 188, 193, 189, 194, 190, 195, 191, 245, 242, 246, 243, 247, 244, -1, +211, 247, 250, 246, 249, 245, 248, 195, 199, 194, 198, 193, 197, 192, 197, 196, 201, 200, 205, 204, 145, -1, +393, 393, 394, 398, 399, 371, -1, +399, 395, 394, -1, +363, 363, 393, 397, 398, 370, 371, 375, -1, +379, 375, 374, 370, 369, 397, 368, 363, 362, -1, +396, 395, 400, 399, 372, 371, 376, 375, 380, 379, 384, 383, 388, 387, 392, 391, 396, 391, 395, 390, 394, -1, +374, 378, 379, 378, 383, 382, 387, 386, 391, 386, 390, 385, 389, 353, 358, 352, 357, 351, 356, 350, 355, -1, +341, 341, 347, 346, 352, 346, 351, 345, 350, -1, +335, 334, 340, 339, 345, 344, 350, 349, 355, 354, -1, +390, 390, 394, 389, 393, 358, 363, 357, 362, 356, 361, 355, 360, 354, 360, 359, 365, 364, 330, 329, 335, 334, -1, +345, 346, 340, 341, 335, 336, 330, 331, 365, 366, 360, 366, 361, 367, 362, 367, 368, 333, 369, 373, 374, 378, -1, +353, 353, 348, 385, 381, 386, 381, 382, 377, 378, 377, 373, 338, 333, 332, 367, 332, 366, 332, 331, 337, 336, 342, 341, 347, -1, +332, 337, 338, 343, 377, 343, 381, 343, 348, 342, 348, 347, 353, 352, -1, +337, 342, 343, -1, +314, 314, 319, 318, 323, 322, 323, 327, -1, +309, 309, 314, 313, 318, 317, 322, 321, 322, 326, 327, 299, -1, +271, 271, 309, 276, 313, 281, 317, 286, 321, 291, 321, 325, 326, 298, 299, 303, -1, +265, 265, 271, 270, 276, 275, 281, 280, 286, 285, 291, 290, 291, 296, 325, 297, 298, 302, 303, 307, -1, +259, 259, 265, 264, 270, 269, 275, 274, 280, 279, 285, 284, 290, 289, 290, 295, 296, 261, 297, 301, 302, 306, 307, 311, -1, +293, 293, 259, 258, 264, 263, 269, 268, 274, 273, 279, 278, 284, 283, 289, 288, 289, 294, 295, 260, 261, 266, -1, +309, 305, 271, 266, 265, 260, 259, 294, 293, 288, 287, 288, 282, 283, 277, 278, 272, 273, 267, 268, 262, -1, +268, 268, 262, 263, 257, 258, 292, 293, 287, -1, +261, 266, 301, 305, 306, 310, 311, 315, 316, 320, -1, +316, 316, 311, 312, 307, 308, 303, 304, 299, 300, 327, 328, 323, 324, 319, 320, 319, 315, 314, 310, 309, 305, -1 +}; + + +const int strip_normals[] = { +515, 515, 492, 527, 492, 67, 307, 358, 19, 433, 150, 493, 16, 289, 339, 88, 489, 277, 258, 271, 232, 345, -1, +258, 469, 489, 80, 339, 324, 16, 8, 150, 200, 19, 64, 307, 374, 492, 374, 132, 203, 256, 59, 498, -1, +85, 316, 98, 226, 406, 519, 469, 227, 80, 157, 324, 153, 8, 6, 200, 99, 64, 291, 374, 291, 203, -1, +99, 99, 39, 6, 412, 153, 351, 157, 121, 227, 241, 519, 75, 226, 526, 316, 526, 264, 389, 178, 51, 82, -1, +75, 526, 86, 389, 387, 51, 70, 107, -1, +51, 82, 107, 212, 107, 235, 70, 442, 387, 253, 86, 443, 75, 443, 241, 202, 121, 281, 351, 436, 412, 376, -1, +412, 376, 39, 376, 158, 361, 21, 90, 52, 90, 235, 479, 263, 36, 196, 436, 196, 281, 306, 202, 119, 443, 119, 253, -1, +90, 90, 479, 361, 36, 376, 436, -1, +306, 119, 40, 450, -1, +119, 253, 450, 442, 450, 235, 40, 189, 306, 189, 196, 189, 263, 235, -1, +291, 291, 203, 506, 59, 65, 169, 229, 96, 432, 302, 432, 235, 424, 52, 373, 21, 375, 158, 249, 39, 99, -1, +432, 432, 424, 229, 373, 65, 375, 506, 249, 291, 99, -1, +59, 59, 498, 169, 460, 96, 30, 302, 452, 235, 525, 212, 299, 82, 391, 178, 57, 264, 120, 316, 85, -1, +391, 166, 299, -1, +423, 388, 72, 9, 166, 456, 299, 456, 525, 456, 452, 456, 30, 9, 460, 388, 498, 487, 256, 206, 132, -1, +423, 487, 388, -1, +206, 487, 352, 423, 341, 423, 418, 72, 139, 166, 139, 391, 57, -1, +85, 223, 120, 24, 57, 24, 139, 24, 418, 25, 341, -1, +223, 25, 24, -1, +492, 132, 515, 206, 224, 352, 359, 341, 41, 25, 171, 223, 50, 85, 43, 98, 232, 406, 258, 469, -1, +232, 345, 43, 372, 50, 454, 171, 514, 41, 314, 359, 360, 224, 2, 515, 527, -1, +393, 393, 0, 216, 349, -1, +486, 215, 327, 91, 177, 254, 177, 393, 512, 0, 231, 349, 231, 279, 486, 246, -1, +177, 512, 327, 231, 486, -1, +462, 462, 144, 76, 179, 464, 298, 464, 393, 128, 216, 113, 349, 500, 279, 101, 246, 7, 246, 395, 246, 384, 486, 215, -1, +7, 7, 195, 101, 444, 500, 462, 113, 76, 128, 464, -1, +254, 254, 285, 91, 149, 215, 83, 384, 325, 395, 520, 395, 315, 7, 143, 195, 129, 195, 461, 444, 475, 462, 144, -1, +520, 308, 325, 415, 83, 297, 149, 504, 285, 162, 278, 446, -1, +403, 403, 520, 276, 308, 404, 308, 272, 415, 416, 297, 317, 504, 125, 162, 60, 446, 110, 446, 393, 278, 254, 285, -1, +125, 125, 459, 317, 334, 416, 137, 272, 47, 404, 511, 404, 1, 276, 54, 403, 45, 370, 490, 210, 29, 184, -1, +336, 1, 251, 54, 164, 45, 463, 490, 378, 29, 453, 147, 222, 218, 301, 218, 242, 310, 130, 528, 34, -1, +26, 336, 148, 251, 269, 164, 419, 463, 293, 378, 420, 453, 333, 222, 208, 301, 294, 56, 127, 228, -1, +131, 26, 371, 148, 284, 269, 311, 419, 95, 293, 350, 420, 377, 333, 474, 208, 396, 294, 357, 127, -1, +116, 131, 273, 371, 480, 284, 104, 311, 257, 95, 348, 350, 447, 377, 313, 474, 322, 396, 28, 357, -1, +273, 273, 183, 480, 347, 104, 53, 257, 181, 348, 124, 447, 428, 313, 428, 435, 11, 270, 193, 305, 69, 305, 386, -1, +11, 11, 428, 409, 124, 382, 181, 161, 53, 481, 347, 408, 183, 319, 151, 422, 448, 4, 165, 134, 394, -1, +193, 193, 11, 495, 409, 44, 382, 385, 161, 478, 481, 320, 408, 37, 319, 524, 422, 328, 4, 465, 134, -1, +37, 37, 431, 320, 136, 478, 513, 385, 27, 44, 467, 495, 115, 193, 69, -1, +305, 305, 386, 20, 174, 182, 401, 211, 248, 106, 295, 309, 255, 465, 483, 328, 191, 524, 135, 37, 431, -1, +465, 465, 134, 309, 102, 106, 427, 211, 507, 182, 410, 20, 503, 305, 503, 270, 445, 435, 322, 313, -1, +134, 134, 394, 102, 455, 427, 5, 507, 23, 410, 405, 503, 405, 445, -1, +322, 322, 445, 28, 405, 138, 23, 485, 5, 234, 455, 74, 394, 180, 165, 49, 448, 116, 151, 273, 183, -1, +28, 357, 138, 268, 485, 287, 234, 81, 74, 78, 180, 103, 49, 220, 116, 220, 131, 476, 26, 38, 26, 336, -1, +357, 127, 268, 252, 287, 398, 81, 274, 78, 154, 103, 62, 220, 476, -1, +127, 228, 252, 141, 398, 440, 274, 363, 154, 190, 62, 488, 476, 38, -1, +301, 242, 56, 517, 228, 33, 141, 18, 440, 466, 363, 304, 190, 417, 488, 484, 38, 511, 336, 1, -1, +33, 517, 260, -1, +18, 33, 390, 260, 497, -1, +466, 18, 353, 390, 290, 497, 126, -1, +304, 466, 187, 353, 123, 290, 522, 126, 501, -1, +417, 304, 458, 187, 117, 123, 168, 522, 87, 501, 261, -1, +484, 417, 430, 458, 343, 117, 438, 168, 426, 87, 482, 261, 329, -1, +511, 484, 47, 430, 137, 343, 334, 438, 459, 426, 439, 482, 92, 329, 192, 329, 267, 329, 491, 261, 219, 501, -1, +528, 528, 34, 145, 46, 356, 105, 472, 219, 48, 491, 247, 267, 393, 192, 110, 92, 60, 439, 125, 459, -1, +219, 501, 105, 126, 46, 497, 34, 260, 130, 517, 242, -1, +29, 184, 147, 156, 218, 156, 310, 402, 528, 146, 145, 17, 356, 411, 472, 364, 48, 441, 247, 441, 393, 509, -1, +184, 184, 156, 63, 402, 354, 146, 335, 17, 239, 411, 13, 364, 509, 441, -1, +393, 509, 298, 13, 179, 239, 144, 335, 475, 354, 461, 63, 129, 184, 129, 210, 143, 370, 315, 403, 520, -1, +296, 296, 160, 133, 344, 330, -1, +344, 323, 160, -1, +159, 159, 296, 275, 133, 22, 330, 523, -1, +198, 523, 77, 22, 286, 275, 437, 159, 499, -1, +280, 323, 108, 344, 338, 330, 259, 523, 366, 198, 381, 240, 283, 10, 368, 243, 280, 243, 323, 35, 160, -1, +77, 471, 198, 471, 240, 250, 10, 303, 243, 303, 35, 230, 282, 362, 163, 73, 318, 292, 93, 477, 262, -1, +413, 413, 326, 399, 73, 399, 292, 100, 477, -1, +122, 355, 414, 61, 100, 55, 477, 508, 262, 365, -1, +35, 35, 160, 282, 296, 163, 159, 318, 499, 93, 505, 262, 340, 365, 340, 140, 68, 510, 518, 3, 122, 355, -1, +100, 399, 414, 413, 122, 111, 518, 213, 68, 167, 340, 167, 505, 245, 499, 245, 437, 346, 286, 79, 77, 471, -1, +362, 362, 221, 230, 516, 303, 516, 250, 207, 471, 207, 79, 521, 346, 468, 245, 468, 167, 468, 213, 473, 111, 204, 413, 326, -1, +468, 473, 521, 217, 207, 217, 516, 217, 221, 204, 221, 326, 362, 73, -1, +473, 204, 217, -1, +94, 94, 185, 529, 429, 209, 429, 176, -1, +379, 379, 94, 199, 529, 367, 209, 172, 209, 66, 176, 496, -1, +312, 312, 379, 238, 199, 15, 367, 332, 172, 197, 172, 451, 66, 84, 496, 186, -1, +266, 266, 312, 392, 238, 201, 15, 14, 332, 457, 197, 383, 197, 225, 451, 502, 84, 265, 186, 205, -1, +236, 236, 266, 214, 392, 71, 201, 188, 14, 173, 457, 400, 383, 380, 383, 342, 225, 109, 502, 175, 265, 97, 205, 118, -1, +152, 152, 236, 337, 214, 58, 71, 369, 188, 434, 173, 114, 400, 288, 380, 42, 380, 397, 342, 32, 109, 331, -1, +379, 170, 312, 331, 266, 32, 236, 397, 152, 42, 194, 42, 407, 288, 142, 114, 12, 434, 233, 369, 244, -1, +369, 369, 244, 58, 425, 337, 321, 152, 194, -1, +109, 331, 175, 170, 97, 300, 118, 421, 31, 89, -1, +31, 31, 118, 237, 205, 449, 186, 470, 496, 494, 176, 155, 429, 112, 185, 89, 185, 421, 94, 300, 379, 170, -1 +}; + +#else /* defined(_WIN32_WCE) */ + +/* + * Original teapot code copyright follows: + */ + +/* + * (c) Copyright 1993, Silicon Graphics, Inc. + * + * ALL RIGHTS RESERVED + * + * Permission to use, copy, modify, and distribute this software + * for any purpose and without fee is hereby granted, provided + * that the above copyright notice appear in all copies and that + * both the copyright notice and this permission notice appear in + * supporting documentation, and that the name of Silicon + * Graphics, Inc. not be used in advertising or publicity + * pertaining to distribution of the software without specific, + * written prior permission. + * + * THE MATERIAL EMBODIED ON THIS SOFTWARE IS PROVIDED TO YOU + * "AS-IS" AND WITHOUT WARRANTY OF ANY KIND, EXPRESS, IMPLIED OR + * OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY WARRANTY OF + * MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. IN NO + * EVENT SHALL SILICON GRAPHICS, INC. BE LIABLE TO YOU OR ANYONE + * ELSE FOR ANY DIRECT, SPECIAL, INCIDENTAL, INDIRECT OR + * CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER, + * INCLUDING WITHOUT LIMITATION, LOSS OF PROFIT, LOSS OF USE, + * SAVINGS OR REVENUE, OR THE CLAIMS OF THIRD PARTIES, WHETHER OR + * NOT SILICON GRAPHICS, INC. HAS BEEN ADVISED OF THE POSSIBILITY + * OF SUCH LOSS, HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, + * ARISING OUT OF OR IN CONNECTION WITH THE POSSESSION, USE OR + * PERFORMANCE OF THIS SOFTWARE. + * + * US Government Users Restricted Rights + * + * Use, duplication, or disclosure by the Government is subject to + * restrictions set forth in FAR 52.227.19(c)(2) or subparagraph + * (c)(1)(ii) of the Rights in Technical Data and Computer + * Software clause at DFARS 252.227-7013 and/or in similar or + * successor clauses in the FAR or the DOD or NASA FAR + * Supplement. Unpublished-- rights reserved under the copyright + * laws of the United States. Contractor/manufacturer is Silicon + * Graphics, Inc., 2011 N. Shoreline Blvd., Mountain View, CA + * 94039-7311. + * + * OpenGL(TM) is a trademark of Silicon Graphics, Inc. + */ + +/* + * Rim, body, lid, and bottom data must be reflected in x and y; + * handle and spout data across the y axis only. + */ +static int patchdata[][16] = +{ + { 102, 103, 104, 105, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 }, /* rim */ + { 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27 }, /* body */ + { 24, 25, 26, 27, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40 }, + { 96, 96, 96, 96, 97, 98, 99, 100, 101, 101, 101, 101, 0, 1, 2, 3 }, /* lid */ + { 0, 1, 2, 3, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 117 }, + { 118, 118, 118, 118, 124, 122, 119, 121, 123, 126, 125, 120, 40, 39, 38, 37 }, /* bottom */ + { 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56 }, /* handle */ + { 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 28, 65, 66, 67 }, + { 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83 }, /* spout */ + { 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95 } +}; + +static double cpdata[][3] = +{ + {0.2, 0, 2.7}, {0.2, -0.112, 2.7}, {0.112, -0.2, 2.7}, {0, + -0.2, 2.7}, {1.3375, 0, 2.53125}, {1.3375, -0.749, 2.53125}, + {0.749, -1.3375, 2.53125}, {0, -1.3375, 2.53125}, {1.4375, + 0, 2.53125}, {1.4375, -0.805, 2.53125}, {0.805, -1.4375, + 2.53125}, {0, -1.4375, 2.53125}, {1.5, 0, 2.4}, {1.5, -0.84, + 2.4}, {0.84, -1.5, 2.4}, {0, -1.5, 2.4}, {1.75, 0, 1.875}, + {1.75, -0.98, 1.875}, {0.98, -1.75, 1.875}, {0, -1.75, + 1.875}, {2, 0, 1.35}, {2, -1.12, 1.35}, {1.12, -2, 1.35}, + {0, -2, 1.35}, {2, 0, 0.9}, {2, -1.12, 0.9}, {1.12, -2, + 0.9}, {0, -2, 0.9}, {-2, 0, 0.9}, {2, 0, 0.45}, {2, -1.12, + 0.45}, {1.12, -2, 0.45}, {0, -2, 0.45}, {1.5, 0, 0.225}, + {1.5, -0.84, 0.225}, {0.84, -1.5, 0.225}, {0, -1.5, 0.225}, + {1.5, 0, 0.15}, {1.5, -0.84, 0.15}, {0.84, -1.5, 0.15}, {0, + -1.5, 0.15}, {-1.6, 0, 2.025}, {-1.6, -0.3, 2.025}, {-1.5, + -0.3, 2.25}, {-1.5, 0, 2.25}, {-2.3, 0, 2.025}, {-2.3, -0.3, + 2.025}, {-2.5, -0.3, 2.25}, {-2.5, 0, 2.25}, {-2.7, 0, + 2.025}, {-2.7, -0.3, 2.025}, {-3, -0.3, 2.25}, {-3, 0, + 2.25}, {-2.7, 0, 1.8}, {-2.7, -0.3, 1.8}, {-3, -0.3, 1.8}, + {-3, 0, 1.8}, {-2.7, 0, 1.575}, {-2.7, -0.3, 1.575}, {-3, + -0.3, 1.35}, {-3, 0, 1.35}, {-2.5, 0, 1.125}, {-2.5, -0.3, + 1.125}, {-2.65, -0.3, 0.9375}, {-2.65, 0, 0.9375}, {-2, + -0.3, 0.9}, {-1.9, -0.3, 0.6}, {-1.9, 0, 0.6}, {1.7, 0, + 1.425}, {1.7, -0.66, 1.425}, {1.7, -0.66, 0.6}, {1.7, 0, + 0.6}, {2.6, 0, 1.425}, {2.6, -0.66, 1.425}, {3.1, -0.66, + 0.825}, {3.1, 0, 0.825}, {2.3, 0, 2.1}, {2.3, -0.25, 2.1}, + {2.4, -0.25, 2.025}, {2.4, 0, 2.025}, {2.7, 0, 2.4}, {2.7, + -0.25, 2.4}, {3.3, -0.25, 2.4}, {3.3, 0, 2.4}, {2.8, 0, + 2.475}, {2.8, -0.25, 2.475}, {3.525, -0.25, 2.49375}, + {3.525, 0, 2.49375}, {2.9, 0, 2.475}, {2.9, -0.15, 2.475}, + {3.45, -0.15, 2.5125}, {3.45, 0, 2.5125}, {2.8, 0, 2.4}, + {2.8, -0.15, 2.4}, {3.2, -0.15, 2.4}, {3.2, 0, 2.4}, {0, 0, + 3.15}, {0.8, 0, 3.15}, {0.8, -0.45, 3.15}, {0.45, -0.8, + 3.15}, {0, -0.8, 3.15}, {0, 0, 2.85}, {1.4, 0, 2.4}, {1.4, + -0.784, 2.4}, {0.784, -1.4, 2.4}, {0, -1.4, 2.4}, {0.4, 0, + 2.55}, {0.4, -0.224, 2.55}, {0.224, -0.4, 2.55}, {0, -0.4, + 2.55}, {1.3, 0, 2.55}, {1.3, -0.728, 2.55}, {0.728, -1.3, + 2.55}, {0, -1.3, 2.55}, {1.3, 0, 2.4}, {1.3, -0.728, 2.4}, + {0.728, -1.3, 2.4}, {0, -1.3, 2.4}, {0, 0, 0}, {1.425, + -0.798, 0}, {1.5, 0, 0.075}, {1.425, 0, 0}, {0.798, -1.425, + 0}, {0, -1.5, 0.075}, {0, -1.425, 0}, {1.5, -0.84, 0.075}, + {0.84, -1.5, 0.075} +}; + +static double tex[2][2][2] = +{ + { {0.0, 0.0}, {1.0, 0.0} }, + { {0.0, 1.0}, {1.0, 1.0} } +}; +#endif /* defined(_WIN32_WCE) */ + + +#endif /* FREEGLUT_TEAPOT_DATA_H */ + diff --git a/examples/opengl-framework/freeglut/freeglut_videoresize.c b/examples/opengl-framework/freeglut/freeglut_videoresize.c new file mode 100644 index 00000000..4d00fb9c --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_videoresize.c @@ -0,0 +1,50 @@ +/* + * freeglut_videoresize.c + * + * Video resize functions (as defined by GLUT API) + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Thu Dec 16 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include +#include "freeglut_internal.h" + +/* + * NOTE: functions declared in this file probably will not be implemented. + */ + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +int FGAPIENTRY glutVideoResizeGet( GLenum eWhat ) { return( 0x00 ); } +void FGAPIENTRY glutSetupVideoResizing( void ) { /* Not implemented */ } +void FGAPIENTRY glutStopVideoResizing( void ) { /* Not implemented */ } +void FGAPIENTRY glutVideoResize( int x, int y, int w, int h ) { /* Not implemented */ } +void FGAPIENTRY glutVideoPan( int x, int y, int w, int h ) { /* Not implemented */ } + +/*** END OF FILE ***/ + + + + + + + diff --git a/examples/opengl-framework/freeglut/freeglut_window.c b/examples/opengl-framework/freeglut/freeglut_window.c new file mode 100644 index 00000000..0542017b --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_window.c @@ -0,0 +1,2162 @@ +/* + * freeglut_window.c + * + * Window management methods. + * + * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. + * Written by Pawel W. Olszta, + * Creation date: Fri Dec 3 1999 + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#define FREEGLUT_BUILDING_LIB +#include +#include "freeglut_internal.h" + +#if TARGET_HOST_POSIX_X11 +#include /* LONG_MAX */ +#include /* usleep */ +#endif + +#if defined(_WIN32_WCE) +# include +# ifdef FREEGLUT_LIB_PRAGMAS +# pragma comment( lib, "Aygshell.lib" ) +# endif +#endif /* defined(_WIN32_WCE) */ + + +#if TARGET_HOST_POSIX_X11 +#ifndef GLX_FRAMEBUFFER_SRGB_CAPABLE_ARB +#define GLX_FRAMEBUFFER_SRGB_CAPABLE_ARB 0x20B2 +#endif + +#ifndef GLX_CONTEXT_MAJOR_VERSION_ARB +#define GLX_CONTEXT_MAJOR_VERSION_ARB 0x2091 +#endif + +#ifndef GLX_CONTEXT_MINOR_VERSION_ARB +#define GLX_CONTEXT_MINOR_VERSION_ARB 0x2092 +#endif + +#ifndef GLX_CONTEXT_FLAGS_ARB +#define GLX_CONTEXT_FLAGS_ARB 0x2094 +#endif + +#ifndef GLX_CONTEXT_PROFILE_MASK_ARB +#define GLX_CONTEXT_PROFILE_MASK_ARB 0x9126 +#endif + +#ifndef GLX_CONTEXT_DEBUG_BIT_ARB +#define GLX_CONTEXT_DEBUG_BIT_ARB 0x0001 +#endif + +#ifndef GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB +#define GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB 0x0002 +#endif + +#ifndef GLX_CONTEXT_CORE_PROFILE_BIT_ARB +#define GLX_CONTEXT_CORE_PROFILE_BIT_ARB 0x00000001 +#endif + +#ifndef GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB +#define GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB 0x00000002 +#endif + +#ifndef GLX_RGBA_FLOAT_TYPE +#define GLX_RGBA_FLOAT_TYPE 0x20B9 +#endif + +#ifndef GLX_RGBA_FLOAT_BIT +#define GLX_RGBA_FLOAT_BIT 0x00000004 +#endif +#endif /* TARGET_HOST_POSIX_X11 */ + + +#if TARGET_HOST_MS_WINDOWS +/* The following include file is available from SGI but is not standard: + * #include + * So we copy the necessary parts out of it. + * XXX: should local definitions for extensions be put in a separate include file? + */ +typedef const char * (WINAPI * PFNWGLGETEXTENSIONSSTRINGARBPROC) (HDC hdc); + +typedef BOOL (WINAPI * PFNWGLCHOOSEPIXELFORMATARBPROC) (HDC hdc, const int *piAttribIList, const FLOAT *pfAttribFList, UINT nMaxFormats, int *piFormats, UINT *nNumFormats); + +#define WGL_DRAW_TO_WINDOW_ARB 0x2001 +#define WGL_ACCELERATION_ARB 0x2003 +#define WGL_SUPPORT_OPENGL_ARB 0x2010 +#define WGL_DOUBLE_BUFFER_ARB 0x2011 +#define WGL_COLOR_BITS_ARB 0x2014 +#define WGL_ALPHA_BITS_ARB 0x201B +#define WGL_DEPTH_BITS_ARB 0x2022 +#define WGL_STENCIL_BITS_ARB 0x2023 +#define WGL_FULL_ACCELERATION_ARB 0x2027 + +#define WGL_SAMPLE_BUFFERS_ARB 0x2041 +#define WGL_SAMPLES_ARB 0x2042 + +#define WGL_TYPE_RGBA_FLOAT_ARB 0x21A0 + +#define WGL_FRAMEBUFFER_SRGB_CAPABLE_ARB 0x20A9 + +#ifndef WGL_ARB_create_context +#define WGL_ARB_create_context 1 +#ifdef WGL_WGLEXT_PROTOTYPES +extern HGLRC WINAPI wglCreateContextAttribsARB (HDC, HGLRC, const int *); +#endif /* WGL_WGLEXT_PROTOTYPES */ +typedef HGLRC (WINAPI * PFNWGLCREATECONTEXTATTRIBSARBPROC) (HDC hDC, HGLRC hShareContext, const int *attribList); + +#define WGL_CONTEXT_MAJOR_VERSION_ARB 0x2091 +#define WGL_CONTEXT_MINOR_VERSION_ARB 0x2092 +#define WGL_CONTEXT_LAYER_PLANE_ARB 0x2093 +#define WGL_CONTEXT_FLAGS_ARB 0x2094 +#define WGL_CONTEXT_PROFILE_MASK_ARB 0x9126 + +#define WGL_CONTEXT_DEBUG_BIT_ARB 0x0001 +#define WGL_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB 0x0002 + +#define WGL_CONTEXT_CORE_PROFILE_BIT_ARB 0x00000001 +#define WGL_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB 0x00000002 + +#define ERROR_INVALID_VERSION_ARB 0x2095 +#define ERROR_INVALID_PROFILE_ARB 0x2096 +#endif + +#endif /* TARGET_HOST_MS_WINDOWS */ + +#ifdef WM_TOUCH + typedef BOOL (WINAPI *pRegisterTouchWindow)(HWND,ULONG); + static pRegisterTouchWindow fghRegisterTouchWindow = (pRegisterTouchWindow)0xDEADBEEF; +#endif + +/* pushing attribute/value pairs into an array */ +#define ATTRIB(a) attributes[where++]=(a) +#define ATTRIB_VAL(a,v) {ATTRIB(a); ATTRIB(v);} + +/* + * TODO BEFORE THE STABLE RELEASE: + * + * fgChooseFBConfig() -- OK, but what about glutInitDisplayString()? + * fgSetupPixelFormat -- ignores the display mode settings + * fgOpenWindow() -- check the Win32 version, -iconic handling! + * fgCloseWindow() -- check the Win32 version + * glutCreateWindow() -- Check when default position and size is {-1,-1} + * glutCreateSubWindow() -- Check when default position and size is {-1,-1} + * glutDestroyWindow() -- check the Win32 version + * glutSetWindow() -- check the Win32 version + * glutGetWindow() -- OK + * glutSetWindowTitle() -- check the Win32 version + * glutSetIconTitle() -- check the Win32 version + * glutShowWindow() -- check the Win32 version + * glutHideWindow() -- check the Win32 version + * glutIconifyWindow() -- check the Win32 version + * glutReshapeWindow() -- check the Win32 version + * glutPositionWindow() -- check the Win32 version + * glutPushWindow() -- check the Win32 version + * glutPopWindow() -- check the Win32 version + */ + +/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ + +static int fghIsLegacyContextVersionRequested( void ) +{ + return fgState.MajorVersion < 2 || (fgState.MajorVersion == 2 && fgState.MinorVersion <= 1); +} + +static int fghIsLegacyContextRequested( void ) +{ + return fghIsLegacyContextVersionRequested() && + fgState.ContextFlags == 0 && + fgState.ContextProfile == 0; +} + +static int fghNumberOfAuxBuffersRequested( void ) +{ + if ( fgState.DisplayMode & GLUT_AUX4 ) { + return 4; + } + if ( fgState.DisplayMode & GLUT_AUX3 ) { + return 3; + } + if ( fgState.DisplayMode & GLUT_AUX2 ) { + return 2; + } + if ( fgState.DisplayMode & GLUT_AUX1 ) { /* NOTE: Same as GLUT_AUX! */ + return fgState.AuxiliaryBufferNumber; + } + return 0; +} + +static int fghMapBit( int mask, int from, int to ) +{ + return ( mask & from ) ? to : 0; + +} + +static void fghContextCreationError( void ) +{ + fgError( "Unable to create OpenGL %d.%d context (flags %x, profile %x)", + fgState.MajorVersion, fgState.MinorVersion, fgState.ContextFlags, + fgState.ContextProfile ); +} + + +/* -- SYSTEM-DEPENDENT PRIVATE FUNCTIONS ------------------------------------ */ + +#if TARGET_HOST_POSIX_X11 +/* + * Chooses a visual basing on the current display mode settings + */ + +GLXFBConfig* fgChooseFBConfig( int *numcfgs ) +{ + GLboolean wantIndexedMode = GL_FALSE; + int attributes[ 100 ]; + int where = 0, numAuxBuffers; + + /* First we have to process the display mode settings... */ + if( fgState.DisplayMode & GLUT_INDEX ) { + ATTRIB_VAL( GLX_BUFFER_SIZE, 8 ); + /* Buffer size is selected later. */ + + ATTRIB_VAL( GLX_RENDER_TYPE, GLX_COLOR_INDEX_BIT ); + wantIndexedMode = GL_TRUE; + } else { + ATTRIB_VAL( GLX_RED_SIZE, 1 ); + ATTRIB_VAL( GLX_GREEN_SIZE, 1 ); + ATTRIB_VAL( GLX_BLUE_SIZE, 1 ); + if( fgState.DisplayMode & GLUT_ALPHA ) { + ATTRIB_VAL( GLX_ALPHA_SIZE, 1 ); + } + } + + if( fgState.DisplayMode & GLUT_DOUBLE ) { + ATTRIB_VAL( GLX_DOUBLEBUFFER, True ); + } + + if( fgState.DisplayMode & GLUT_STEREO ) { + ATTRIB_VAL( GLX_STEREO, True ); + } + + if( fgState.DisplayMode & GLUT_DEPTH ) { + ATTRIB_VAL( GLX_DEPTH_SIZE, 1 ); + } + + if( fgState.DisplayMode & GLUT_STENCIL ) { + ATTRIB_VAL( GLX_STENCIL_SIZE, 1 ); + } + + if( fgState.DisplayMode & GLUT_ACCUM ) { + ATTRIB_VAL( GLX_ACCUM_RED_SIZE, 1 ); + ATTRIB_VAL( GLX_ACCUM_GREEN_SIZE, 1 ); + ATTRIB_VAL( GLX_ACCUM_BLUE_SIZE, 1 ); + if( fgState.DisplayMode & GLUT_ALPHA ) { + ATTRIB_VAL( GLX_ACCUM_ALPHA_SIZE, 1 ); + } + } + + numAuxBuffers = fghNumberOfAuxBuffersRequested(); + if ( numAuxBuffers > 0 ) { + ATTRIB_VAL( GLX_AUX_BUFFERS, numAuxBuffers ); + } + + if( fgState.DisplayMode & GLUT_SRGB ) { + ATTRIB_VAL( GLX_FRAMEBUFFER_SRGB_CAPABLE_ARB, True ); + } + + if (fgState.DisplayMode & GLUT_MULTISAMPLE) { + ATTRIB_VAL(GLX_SAMPLE_BUFFERS, 1); + ATTRIB_VAL(GLX_SAMPLES, fgState.SampleNumber); + } + + /* Push a terminator at the end of the list */ + ATTRIB( None ); + + { + GLXFBConfig * fbconfigArray; /* Array of FBConfigs */ + GLXFBConfig * fbconfig; /* The FBConfig we want */ + int fbconfigArraySize; /* Number of FBConfigs in the array */ + + + /* Get all FBConfigs that match "attributes". */ + fbconfigArray = glXChooseFBConfig( fgDisplay.Display, + fgDisplay.Screen, + attributes, + &fbconfigArraySize ); + + if (fbconfigArray != NULL) + { + int result; /* Returned by glXGetFBConfigAttrib, not checked. */ + + + if( wantIndexedMode ) + { + /* + * In index mode, we want the largest buffer size, i.e. visual + * depth. Here, FBConfigs are sorted by increasing buffer size + * first, so FBConfigs with the largest size come last. + */ + + int bufferSizeMin, bufferSizeMax; + + /* Get bufferSizeMin. */ + result = + glXGetFBConfigAttrib( fgDisplay.Display, + fbconfigArray[0], + GLX_BUFFER_SIZE, + &bufferSizeMin ); + /* Get bufferSizeMax. */ + result = + glXGetFBConfigAttrib( fgDisplay.Display, + fbconfigArray[fbconfigArraySize - 1], + GLX_BUFFER_SIZE, + &bufferSizeMax ); + + if (bufferSizeMax > bufferSizeMin) + { + /* + * Free and reallocate fbconfigArray, keeping only FBConfigs + * with the largest buffer size. + */ + XFree(fbconfigArray); + + /* Add buffer size token at the end of the list. */ + where--; + ATTRIB_VAL( GLX_BUFFER_SIZE, bufferSizeMax ); + ATTRIB( None ); + + fbconfigArray = glXChooseFBConfig( fgDisplay.Display, + fgDisplay.Screen, + attributes, + &fbconfigArraySize ); + } + } + + /* + * We now have an array of FBConfigs, the first one being the "best" + * one. So we should return only this FBConfig: + * + * int fbconfigXID; + * + * - pick the XID of the FBConfig we want + * result = glXGetFBConfigAttrib( fgDisplay.Display, + * fbconfigArray[0], + * GLX_FBCONFIG_ID, + * &fbconfigXID ); + * + * - free the array + * XFree(fbconfigArray); + * + * - reset "attributes" with the XID + * where = 0; + * ATTRIB_VAL( GLX_FBCONFIG_ID, fbconfigXID ); + * ATTRIB( None ); + * + * - get our FBConfig only + * fbconfig = glXChooseFBConfig( fgDisplay.Display, + * fgDisplay.Screen, + * attributes, + * &fbconfigArraySize ); + * + * However, for some configurations (for instance multisampling with + * Mesa 6.5.2 and ATI drivers), this does not work: + * glXChooseFBConfig returns NULL, whereas fbconfigXID is a valid + * XID. Further investigation is needed. + * + * So, for now, we return the whole array of FBConfigs. This should + * not produce any side effects elsewhere. + */ + fbconfig = fbconfigArray; + } + else + { + fbconfig = NULL; + } + + if (numcfgs) + *numcfgs = fbconfigArraySize; + + return fbconfig; + } +} + + +static void fghFillContextAttributes( int *attributes ) { + int where = 0, contextFlags, contextProfile; + + if ( !fghIsLegacyContextVersionRequested() ) { + ATTRIB_VAL( GLX_CONTEXT_MAJOR_VERSION_ARB, fgState.MajorVersion ); + ATTRIB_VAL( GLX_CONTEXT_MINOR_VERSION_ARB, fgState.MinorVersion ); + } + + contextFlags = + fghMapBit( fgState.ContextFlags, GLUT_DEBUG, GLX_CONTEXT_DEBUG_BIT_ARB ) | + fghMapBit( fgState.ContextFlags, GLUT_FORWARD_COMPATIBLE, GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB ); + if ( contextFlags != 0 ) { + ATTRIB_VAL( GLX_CONTEXT_FLAGS_ARB, contextFlags ); + } + + contextProfile = + fghMapBit( fgState.ContextProfile, GLUT_CORE_PROFILE, GLX_CONTEXT_CORE_PROFILE_BIT_ARB ) | + fghMapBit( fgState.ContextProfile, GLUT_COMPATIBILITY_PROFILE, GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB ); + if ( contextProfile != 0 ) { + ATTRIB_VAL( GLX_CONTEXT_PROFILE_MASK_ARB, contextProfile ); + } + + ATTRIB( 0 ); +} + +typedef GLXContext (*CreateContextAttribsProc)(Display *dpy, GLXFBConfig config, + GLXContext share_list, Bool direct, + const int *attrib_list); + +static GLXContext fghCreateNewContext( SFG_Window* window ) +{ + /* for color model calculation */ + int menu = ( window->IsMenu && !fgStructure.MenuContext ); + int index_mode = ( fgState.DisplayMode & GLUT_INDEX ); + + /* "classic" context creation */ + Display *dpy = fgDisplay.Display; + GLXFBConfig config = *(window->Window.FBConfig); + int render_type = ( !menu && index_mode ) ? GLX_COLOR_INDEX_TYPE : GLX_RGBA_TYPE; + GLXContext share_list = NULL; + Bool direct = ( fgState.DirectContext != GLUT_FORCE_INDIRECT_CONTEXT ); + GLXContext context; + + /* new context creation */ + int attributes[9]; + CreateContextAttribsProc createContextAttribs = (CreateContextAttribsProc) fghGetProcAddress( "glXCreateContextAttribsARB" ); + + /* glXCreateContextAttribsARB not found, yet the user has requested the new context creation */ + if ( !createContextAttribs && !fghIsLegacyContextRequested() ) { + fgWarning( "OpenGL >2.1 context requested but glXCreateContextAttribsARB is not available! Falling back to legacy context creation" ); + fgState.MajorVersion = 2; + fgState.MinorVersion = 1; + } + + /* If nothing fancy has been required, simply use the old context creation GLX API entry */ + if ( fghIsLegacyContextRequested() || !createContextAttribs ) + { + context = glXCreateNewContext( dpy, config, render_type, share_list, direct ); + if ( context == NULL ) { + fghContextCreationError(); + } + return context; + } + + /* color index mode is not available anymore with OpenGL 3.0 */ + if ( render_type == GLX_COLOR_INDEX_TYPE ) { + fgWarning( "color index mode is deprecated, using RGBA mode" ); + } + + fghFillContextAttributes( attributes ); + + context = createContextAttribs( dpy, config, share_list, direct, attributes ); + if ( context == NULL ) { + fghContextCreationError(); + } + return context; +} + + +#define _NET_WM_STATE_TOGGLE 2 +static int fghResizeFullscrToggle(void) +{ + XWindowAttributes attributes; + + if(glutGet(GLUT_FULL_SCREEN)) { + /* restore original window size */ + SFG_Window *win = fgStructure.CurrentWindow; + fgStructure.CurrentWindow->State.NeedToResize = GL_TRUE; + fgStructure.CurrentWindow->State.Width = win->State.OldWidth; + fgStructure.CurrentWindow->State.Height = win->State.OldHeight; + + } else { + /* resize the window to cover the entire screen */ + XGetWindowAttributes(fgDisplay.Display, + fgStructure.CurrentWindow->Window.Handle, + &attributes); + + /* + * The "x" and "y" members of "attributes" are the window's coordinates + * relative to its parent, i.e. to the decoration window. + */ + XMoveResizeWindow(fgDisplay.Display, + fgStructure.CurrentWindow->Window.Handle, + -attributes.x, + -attributes.y, + fgDisplay.ScreenWidth, + fgDisplay.ScreenHeight); + } + return 0; +} + +static int fghEwmhFullscrToggle(void) +{ + XEvent xev; + long evmask = SubstructureRedirectMask | SubstructureNotifyMask; + + if(!fgDisplay.State || !fgDisplay.StateFullScreen) { + return -1; + } + + xev.type = ClientMessage; + xev.xclient.window = fgStructure.CurrentWindow->Window.Handle; + xev.xclient.message_type = fgDisplay.State; + xev.xclient.format = 32; + xev.xclient.data.l[0] = _NET_WM_STATE_TOGGLE; + xev.xclient.data.l[1] = fgDisplay.StateFullScreen; + xev.xclient.data.l[2] = 0; /* no second property to toggle */ + xev.xclient.data.l[3] = 1; /* source indication: application */ + xev.xclient.data.l[4] = 0; /* unused */ + + if(!XSendEvent(fgDisplay.Display, fgDisplay.RootWindow, 0, evmask, &xev)) { + return -1; + } + return 0; +} + +static int fghToggleFullscreen(void) +{ + /* first try the EWMH (_NET_WM_STATE) method ... */ + if(fghEwmhFullscrToggle() != -1) { + return 0; + } + + /* fall back to resizing the window */ + if(fghResizeFullscrToggle() != -1) { + return 0; + } + return -1; +} + + +#endif /* TARGET_HOST_POSIX_X11 */ + + +#if TARGET_HOST_MS_WINDOWS +/* + * Setup the pixel format for a Win32 window + */ + +#if defined(_WIN32_WCE) +static wchar_t* fghWstrFromStr(const char* str) +{ + int i,len=strlen(str); + wchar_t* wstr = (wchar_t*)malloc(2*len+2); + for(i=0; iWindow.Device, window->Window.Context ); + + if ( !fghIsExtensionSupported( window->Window.Device, "WGL_ARB_create_context" ) ) + { + return; + } + + /* new context creation */ + fghFillContextAttributes( attributes ); + + wglCreateContextAttribsARB = (PFNWGLCREATECONTEXTATTRIBSARBPROC) wglGetProcAddress( "wglCreateContextAttribsARB" ); + if ( wglCreateContextAttribsARB == NULL ) + { + fgError( "wglCreateContextAttribsARB not found" ); + } + + context = wglCreateContextAttribsARB( window->Window.Device, 0, attributes ); + if ( context == NULL ) + { + fghContextCreationError(); + } + + wglMakeCurrent( NULL, NULL ); + wglDeleteContext( window->Window.Context ); + window->Window.Context = context; +} + +#if !defined(_WIN32_WCE) + +static void fghFillPFD( PIXELFORMATDESCRIPTOR *ppfd, HDC hdc, unsigned char layer_type ) +{ + int flags = PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL; + if ( fgState.DisplayMode & GLUT_DOUBLE ) { + flags |= PFD_DOUBLEBUFFER; + } + if ( fgState.DisplayMode & GLUT_STEREO ) { + flags |= PFD_STEREO; + } + +#if defined(_MSC_VER) +#pragma message( "fgSetupPixelFormat(): there is still some work to do here!" ) +#endif + + /* Specify which pixel format do we opt for... */ + ppfd->nSize = sizeof(PIXELFORMATDESCRIPTOR); + ppfd->nVersion = 1; + ppfd->dwFlags = flags; + + if( fgState.DisplayMode & GLUT_INDEX ) { + ppfd->iPixelType = PFD_TYPE_COLORINDEX; + ppfd->cRedBits = 0; + ppfd->cGreenBits = 0; + ppfd->cBlueBits = 0; + ppfd->cAlphaBits = 0; + } else { + ppfd->iPixelType = PFD_TYPE_RGBA; + ppfd->cRedBits = 8; + ppfd->cGreenBits = 8; + ppfd->cBlueBits = 8; + ppfd->cAlphaBits = ( fgState.DisplayMode & GLUT_ALPHA ) ? 8 : 0; + } + + ppfd->cColorBits = 24; + ppfd->cRedShift = 0; + ppfd->cGreenShift = 0; + ppfd->cBlueShift = 0; + ppfd->cAlphaShift = 0; + ppfd->cAccumBits = ( fgState.DisplayMode & GLUT_ACCUM ) ? 1 : 0; + ppfd->cAccumRedBits = 0; + ppfd->cAccumGreenBits = 0; + ppfd->cAccumBlueBits = 0; + ppfd->cAccumAlphaBits = 0; + + /* Hmmm, or 32/0 instead of 24/8? */ + ppfd->cDepthBits = 24; + ppfd->cStencilBits = 8; + + ppfd->cAuxBuffers = fghNumberOfAuxBuffersRequested(); + ppfd->iLayerType = layer_type; + ppfd->bReserved = 0; + ppfd->dwLayerMask = 0; + ppfd->dwVisibleMask = 0; + ppfd->dwDamageMask = 0; + + ppfd->cColorBits = (BYTE) GetDeviceCaps( hdc, BITSPIXEL ); +} + +static void fghFillPixelFormatAttributes( int *attributes, const PIXELFORMATDESCRIPTOR *ppfd ) +{ + int where = 0; + + ATTRIB_VAL( WGL_DRAW_TO_WINDOW_ARB, GL_TRUE ); + ATTRIB_VAL( WGL_SUPPORT_OPENGL_ARB, GL_TRUE ); + ATTRIB_VAL( WGL_ACCELERATION_ARB, WGL_FULL_ACCELERATION_ARB ); + + ATTRIB_VAL( WGL_COLOR_BITS_ARB, ppfd->cColorBits ); + ATTRIB_VAL( WGL_ALPHA_BITS_ARB, ppfd->cAlphaBits ); + ATTRIB_VAL( WGL_DEPTH_BITS_ARB, ppfd->cDepthBits ); + ATTRIB_VAL( WGL_STENCIL_BITS_ARB, ppfd->cStencilBits ); + + ATTRIB_VAL( WGL_DOUBLE_BUFFER_ARB, ( fgState.DisplayMode & GLUT_DOUBLE ) != 0 ); + + if ( fgState.DisplayMode & GLUT_SRGB ) { + ATTRIB_VAL( WGL_FRAMEBUFFER_SRGB_CAPABLE_ARB, TRUE ); + } + + ATTRIB_VAL( WGL_SAMPLE_BUFFERS_ARB, GL_TRUE ); + ATTRIB_VAL( WGL_SAMPLES_ARB, fgState.SampleNumber ); + ATTRIB( 0 ); +} +#endif + +GLboolean fgSetupPixelFormat( SFG_Window* window, GLboolean checkOnly, + unsigned char layer_type ) +{ +#if defined(_WIN32_WCE) + return GL_TRUE; +#else + PIXELFORMATDESCRIPTOR pfd; + PIXELFORMATDESCRIPTOR* ppfd = &pfd; + int pixelformat; + HDC current_hDC; + GLboolean success; + + if (checkOnly) + current_hDC = CreateDC(TEXT("DISPLAY"), NULL ,NULL ,NULL); + else + current_hDC = window->Window.Device; + + fghFillPFD( ppfd, current_hDC, layer_type ); + pixelformat = ChoosePixelFormat( current_hDC, ppfd ); + + /* windows hack for multismapling/sRGB */ + if ( ( fgState.DisplayMode & GLUT_MULTISAMPLE ) || + ( fgState.DisplayMode & GLUT_SRGB ) ) + { + HGLRC rc, rc_before=wglGetCurrentContext(); + HWND hWnd; + HDC hDC, hDC_before=wglGetCurrentDC(); + WNDCLASS wndCls; + + /* create a dummy window */ + ZeroMemory(&wndCls, sizeof(wndCls)); + wndCls.lpfnWndProc = DefWindowProc; + wndCls.hInstance = fgDisplay.Instance; + wndCls.style = CS_OWNDC | CS_HREDRAW | CS_VREDRAW; + wndCls.lpszClassName = _T("FREEGLUT_dummy"); + RegisterClass( &wndCls ); + + hWnd=CreateWindow(_T("FREEGLUT_dummy"), _T(""), WS_CLIPSIBLINGS | WS_CLIPCHILDREN | WS_OVERLAPPEDWINDOW , 0,0,0,0, 0, 0, fgDisplay.Instance, 0 ); + hDC=GetDC(hWnd); + SetPixelFormat( hDC, pixelformat, ppfd ); + + rc = wglCreateContext( hDC ); + wglMakeCurrent(hDC, rc); + + if ( fghIsExtensionSupported( hDC, "WGL_ARB_multisample" ) ) + { + PFNWGLCHOOSEPIXELFORMATARBPROC wglChoosePixelFormatARBProc = + (PFNWGLCHOOSEPIXELFORMATARBPROC) wglGetProcAddress("wglChoosePixelFormatARB"); + if ( wglChoosePixelFormatARBProc ) + { + int attributes[100]; + int iPixelFormat; + BOOL bValid; + float fAttributes[] = { 0, 0 }; + UINT numFormats; + fghFillPixelFormatAttributes( attributes, ppfd ); + bValid = wglChoosePixelFormatARBProc(hDC, attributes, fAttributes, 1, &iPixelFormat, &numFormats); + + if ( bValid && numFormats > 0 ) + { + pixelformat = iPixelFormat; + } + } + } + + wglMakeCurrent( hDC_before, rc_before); + wglDeleteContext(rc); + ReleaseDC(hWnd, hDC); + DestroyWindow(hWnd); + UnregisterClass(_T("FREEGLUT_dummy"), fgDisplay.Instance); + } + + success = ( pixelformat != 0 ) && ( checkOnly || SetPixelFormat( current_hDC, pixelformat, ppfd ) ); + + if (checkOnly) + DeleteDC(current_hDC); + + return success; +#endif /* defined(_WIN32_WCE) */ +} + +#endif /* TARGET_HOST_MS_WINDOWS */ + +/* + * Sets the OpenGL context and the fgStructure "Current Window" pointer to + * the window structure passed in. + */ +void fgSetWindow ( SFG_Window *window ) +{ +#if TARGET_HOST_POSIX_X11 + if ( window ) + { + glXMakeContextCurrent( + fgDisplay.Display, + window->Window.Handle, + window->Window.Handle, + window->Window.Context + ); + } +#elif TARGET_HOST_MS_WINDOWS + if ( window != fgStructure.CurrentWindow ) + { + if( fgStructure.CurrentWindow ) + ReleaseDC( fgStructure.CurrentWindow->Window.Handle, + fgStructure.CurrentWindow->Window.Device ); + + if ( window ) + { + window->Window.Device = GetDC( window->Window.Handle ); + wglMakeCurrent( + window->Window.Device, + window->Window.Context + ); + } + } +#endif + fgStructure.CurrentWindow = window; +} + +#if TARGET_HOST_MS_WINDOWS + +/* Computes position of corners of window Rect (outer position including + * decorations) based on the provided client rect and based on the style + * of the window in question. + * If posIsOutside is set to true, the input client Rect is taken to follow + * freeGLUT's window specification convention in which the top-left corner + * is at the outside of the window, while the size + * (rect.right-rect.left,rect.bottom-rect.top) is the size of the drawable + * area. + */ +void fghComputeWindowRectFromClientArea_UseStyle( const DWORD windowStyle, RECT *clientRect, BOOL posIsOutside ) +{ + int xBorderWidth = 0, yBorderWidth = 0; + + /* If window has title bar, correct rect for it */ + if (windowStyle & WS_MAXIMIZEBOX) /* Need to query for WS_MAXIMIZEBOX to see if we have a title bar, the WS_CAPTION query is also true for a WS_DLGFRAME only... */ + if (posIsOutside) + clientRect->bottom += GetSystemMetrics( SM_CYCAPTION ); + else + clientRect->top -= GetSystemMetrics( SM_CYCAPTION ); + + /* get width of window's borders (frame), correct rect for it. + * Note, borders can be of zero width if style does not specify borders + */ + fghGetBorderWidth(windowStyle, &xBorderWidth, &yBorderWidth); + if (posIsOutside) + { + clientRect->right += xBorderWidth * 2; + clientRect->bottom += yBorderWidth * 2; + } + else + { + clientRect->left -= xBorderWidth; + clientRect->right += xBorderWidth; + clientRect->top -= yBorderWidth; + clientRect->bottom += yBorderWidth; + } +} + +/* Computes position of corners of window Rect (outer position including + * decorations) based on the provided client rect and based on the style + * of the window in question. If the window pointer or the window handle + * is NULL, a fully decorated window (caption and border) is assumed. + * Furthermore, if posIsOutside is set to true, the input client Rect is + * taken to follow freeGLUT's window specification convention in which the + * top-left corner is at the outside of the window, while the size + * (rect.right-rect.left,rect.bottom-rect.top) is the size of the drawable + * area. +*/ +void fghComputeWindowRectFromClientArea_QueryWindow( const SFG_Window *window, RECT *clientRect, BOOL posIsOutside ) +{ + DWORD windowStyle = 0; + + if (window && window->Window.Handle) + windowStyle = GetWindowLong(window->Window.Handle, GWL_STYLE); + else + windowStyle = WS_OVERLAPPEDWINDOW; + + fghComputeWindowRectFromClientArea_UseStyle(windowStyle, clientRect, posIsOutside); +} + +/* Computes position of corners of client area (drawable area) of a window + * based on the provided window Rect (outer position including decorations) + * and based on the style of the window in question. If the window pointer + * or the window handle is NULL, a fully decorated window (caption and + * border) is assumed. + * Furthermore, if wantPosOutside is set to true, the output client Rect + * will follow freeGLUT's window specification convention in which the + * top-left corner is at the outside of the window, the size + * (rect.right-rect.left,rect.bottom-rect.top) is the size of the drawable + * area. + */ +void fghComputeClientAreaFromWindowRect( const SFG_Window *window, RECT *windowRect, BOOL wantPosOutside ) +{ + DWORD windowStyle = 0; + int xBorderWidth = 0, yBorderWidth = 0; + + if (window && window->Window.Handle) + windowStyle = GetWindowLong(window->Window.Handle, GWL_STYLE); + else + windowStyle = WS_OVERLAPPEDWINDOW; + + /* If window has title bar, correct rect for it */ + if (windowStyle & WS_MAXIMIZEBOX) /* Need to query for WS_MAXIMIZEBOX to see if we have a title bar, the WS_CAPTION query is also true for a WS_DLGFRAME only... */ + if (wantPosOutside) + windowRect->bottom -= GetSystemMetrics( SM_CYCAPTION ); + else + windowRect->top += GetSystemMetrics( SM_CYCAPTION ); + + /* get width of window's borders (frame), correct rect for it. + * Note, borders can be of zero width if style does not specify borders + */ + fghGetBorderWidth(windowStyle, &xBorderWidth, &yBorderWidth); + if (wantPosOutside) + { + windowRect->right -= xBorderWidth * 2; + windowRect->bottom -= yBorderWidth * 2; + } + else + { + windowRect->left += xBorderWidth; + windowRect->right -= xBorderWidth; + windowRect->top += yBorderWidth; + windowRect->bottom -= yBorderWidth; + } +} + +/* Gets the rect describing the client area (drawable area) of the + * specified window. + * Returns an empty rect if window pointer or window handle is NULL. + * If wantPosOutside is set to true, the output client Rect + * will follow freeGLUT's window specification convention in which the + * top-left corner is at the outside of the window, while the size + * (rect.right-rect.left,rect.bottom-rect.top) is the size of the drawable + * area. + */ +RECT fghGetClientArea( const SFG_Window *window, BOOL wantPosOutside ) +{ + RECT windowRect = {0,0,0,0}; + + freeglut_return_val_if_fail((window && window->Window.Handle),windowRect); + + /* + * call GetWindowRect() + * (this returns the pixel coordinates of the outside of the window) + */ + GetWindowRect( window->Window.Handle, &windowRect ); + + /* Then correct the results */ + fghComputeClientAreaFromWindowRect(window, &windowRect, wantPosOutside); + + return windowRect; +} + +/* Returns the width of the window borders based on the window's style. + */ +void fghGetBorderWidth(const DWORD windowStyle, int* xBorderWidth, int* yBorderWidth) +{ + if (windowStyle & WS_THICKFRAME) + { + *xBorderWidth = GetSystemMetrics(SM_CXSIZEFRAME); + *yBorderWidth = GetSystemMetrics(SM_CYSIZEFRAME); + } + else if (windowStyle & WS_DLGFRAME) + { + *xBorderWidth = GetSystemMetrics(SM_CXFIXEDFRAME); + *yBorderWidth = GetSystemMetrics(SM_CYFIXEDFRAME); + } + else + { + *xBorderWidth = 0; + *yBorderWidth = 0; + } +} + +#if(WINVER >= 0x500) +typedef struct +{ + int *x; + int *y; + const char *name; +} m_proc_t; + +static BOOL CALLBACK m_proc(HMONITOR mon, + HDC hdc, + LPRECT rect, + LPARAM data) +{ + m_proc_t *dp=(m_proc_t *)data; + MONITORINFOEX info; + BOOL res; + info.cbSize=sizeof(info); + res=GetMonitorInfo(mon,(LPMONITORINFO)&info); + if( res ) + { + if( strcmp(dp->name,info.szDevice)==0 ) + { + *(dp->x)=info.rcMonitor.left; + *(dp->y)=info.rcMonitor.top; + return FALSE; + } + } + return TRUE; +} + +/* + * this function returns the origin of the screen identified by + * fgDisplay.DisplayName, and 0 otherwise. + * This is used in fgOpenWindow to open the gamemode window on the screen + * identified by the -display command line argument. The function should + * not be called otherwise. + */ + +static void get_display_origin(int *xp,int *yp) +{ + *xp = 0; + *yp = 0; + + if( fgDisplay.DisplayName ) + { + m_proc_t st; + st.x=xp; + st.y=yp; + st.name=fgDisplay.DisplayName; + EnumDisplayMonitors(0,0,m_proc,(LPARAM)&st); + } +} +#else +#pragma message( "-display parameter only works if compiled with WINVER >= 0x0500") + +static void get_display_origin(int *xp,int *yp) +{ + *xp = 0; + *yp = 0; + + if( fgDisplay.DisplayName ) + { + fgWarning( "for working -display support FreeGLUT must be compiled with WINVER >= 0x0500"); + } +} +#endif +#endif + + +#if TARGET_HOST_POSIX_X11 +static Bool fghWindowIsVisible( Display *display, XEvent *event, XPointer arg) +{ + Window window = (Window)arg; + return (event->type == MapNotify) && (event->xmap.window == window); +} +#endif + + +/* + * Opens a window. Requires a SFG_Window object created and attached + * to the freeglut structure. OpenGL context is created here. + */ +void fgOpenWindow( SFG_Window* window, const char* title, + GLboolean positionUse, int x, int y, + GLboolean sizeUse, int w, int h, + GLboolean gameMode, GLboolean isSubWindow ) +{ +#if TARGET_HOST_POSIX_X11 + XVisualInfo * visualInfo = NULL; + XSetWindowAttributes winAttr; + XTextProperty textProperty; + XSizeHints sizeHints; + XWMHints wmHints; + XEvent eventReturnBuffer; /* return buffer required for a call */ + unsigned long mask; + int num_FBConfigs, i; + unsigned int current_DisplayMode = fgState.DisplayMode ; + + /* Save the display mode if we are creating a menu window */ + if( window->IsMenu && ( ! fgStructure.MenuContext ) ) + fgState.DisplayMode = GLUT_DOUBLE | GLUT_RGB ; + + window->Window.FBConfig = fgChooseFBConfig( &num_FBConfigs ); + + if( window->IsMenu && ( ! fgStructure.MenuContext ) ) + fgState.DisplayMode = current_DisplayMode ; + + if( ! window->Window.FBConfig ) + { + /* + * The "fgChooseFBConfig" returned a null meaning that the visual + * context is not available. + * Try a couple of variations to see if they will work. + */ + if( !( fgState.DisplayMode & GLUT_DOUBLE ) ) + { + fgState.DisplayMode |= GLUT_DOUBLE ; + window->Window.FBConfig = fgChooseFBConfig( &num_FBConfigs ); + fgState.DisplayMode &= ~GLUT_DOUBLE; + } + + if( fgState.DisplayMode & GLUT_MULTISAMPLE ) + { + fgState.DisplayMode &= ~GLUT_MULTISAMPLE ; + window->Window.FBConfig = fgChooseFBConfig( &num_FBConfigs ); + fgState.DisplayMode |= GLUT_MULTISAMPLE; + } + } + + FREEGLUT_INTERNAL_ERROR_EXIT( window->Window.FBConfig != NULL, + "FBConfig with necessary capabilities not found", "fgOpenWindow" ); + + /* Get the X visual. */ + for (i = 0; i < num_FBConfigs; i++) { + visualInfo = glXGetVisualFromFBConfig( fgDisplay.Display, + window->Window.FBConfig[i] ); + if (visualInfo) + break; + } + + FREEGLUT_INTERNAL_ERROR_EXIT( visualInfo != NULL, + "visualInfo could not be retrieved from FBConfig", "fgOpenWindow" ); + + /* + * XXX HINT: the masks should be updated when adding/removing callbacks. + * XXX This might speed up message processing. Is that true? + * XXX + * XXX A: Not appreciably, but it WILL make it easier to debug. + * XXX Try tracing old GLUT and try tracing freeglut. Old GLUT + * XXX turns off events that it doesn't need and is a whole lot + * XXX more pleasant to trace. (Think mouse-motion! Tons of + * XXX ``bonus'' GUI events stream in.) + */ + winAttr.event_mask = + StructureNotifyMask | SubstructureNotifyMask | ExposureMask | + ButtonPressMask | ButtonReleaseMask | KeyPressMask | KeyReleaseMask | + VisibilityChangeMask | EnterWindowMask | LeaveWindowMask | + PointerMotionMask | ButtonMotionMask; + winAttr.background_pixmap = None; + winAttr.background_pixel = 0; + winAttr.border_pixel = 0; + + winAttr.colormap = XCreateColormap( + fgDisplay.Display, fgDisplay.RootWindow, + visualInfo->visual, AllocNone + ); + + mask = CWBackPixmap | CWBorderPixel | CWColormap | CWEventMask; + + if( window->IsMenu || ( gameMode == GL_TRUE ) ) + { + winAttr.override_redirect = True; + mask |= CWOverrideRedirect; + } + + if( ! positionUse ) + x = y = -1; /* default window position */ + if( ! sizeUse ) + w = h = 300; /* default window size */ + + window->Window.Handle = XCreateWindow( + fgDisplay.Display, + window->Parent == NULL ? fgDisplay.RootWindow : + window->Parent->Window.Handle, + x, y, w, h, 0, + visualInfo->depth, InputOutput, + visualInfo->visual, mask, + &winAttr + ); + + /* + * The GLX context creation, possibly trying the direct context rendering + * or else use the current context if the user has so specified + */ + + if( window->IsMenu ) + { + /* + * If there isn't already an OpenGL rendering context for menu + * windows, make one + */ + if( !fgStructure.MenuContext ) + { + fgStructure.MenuContext = + (SFG_MenuContext *)malloc( sizeof(SFG_MenuContext) ); + fgStructure.MenuContext->MContext = fghCreateNewContext( window ); + } + + /* window->Window.Context = fgStructure.MenuContext->MContext; */ + window->Window.Context = fghCreateNewContext( window ); + } + else if( fgState.UseCurrentContext ) + { + window->Window.Context = glXGetCurrentContext( ); + + if( ! window->Window.Context ) + window->Window.Context = fghCreateNewContext( window ); + } + else + window->Window.Context = fghCreateNewContext( window ); + +#if !defined( __FreeBSD__ ) && !defined( __NetBSD__ ) + if( !glXIsDirect( fgDisplay.Display, window->Window.Context ) ) + { + if( fgState.DirectContext == GLUT_FORCE_DIRECT_CONTEXT ) + fgError( "Unable to force direct context rendering for window '%s'", + title ); + } +#endif + + /* + * XXX Assume the new window is visible by default + * XXX Is this a safe assumption? + */ + window->State.Visible = GL_TRUE; + + sizeHints.flags = 0; + if ( positionUse ) + sizeHints.flags |= USPosition; + if ( sizeUse ) + sizeHints.flags |= USSize; + + /* + * Fill in the size hints values now (the x, y, width and height + * settings are obsolete, are there any more WMs that support them?) + * Unless the X servers actually stop supporting these, we should + * continue to fill them in. It is *not* our place to tell the user + * that they should replace a window manager that they like, and which + * works, just because *we* think that it's not "modern" enough. + */ + sizeHints.x = x; + sizeHints.y = y; + sizeHints.width = w; + sizeHints.height = h; + + wmHints.flags = StateHint; + wmHints.initial_state = fgState.ForceIconic ? IconicState : NormalState; + /* Prepare the window and iconified window names... */ + XStringListToTextProperty( (char **) &title, 1, &textProperty ); + + XSetWMProperties( + fgDisplay.Display, + window->Window.Handle, + &textProperty, + &textProperty, + 0, + 0, + &sizeHints, + &wmHints, + NULL + ); + XFree( textProperty.value ); + + XSetWMProtocols( fgDisplay.Display, window->Window.Handle, + &fgDisplay.DeleteWindow, 1 ); + + glXMakeContextCurrent( + fgDisplay.Display, + window->Window.Handle, + window->Window.Handle, + window->Window.Context + ); + + /* register extension events _before_ window is mapped */ + #ifdef HAVE_X11_EXTENSIONS_XINPUT2_H + fgRegisterDevices( fgDisplay.Display, &(window->Window.Handle) ); + #endif + + XMapWindow( fgDisplay.Display, window->Window.Handle ); + + XFree(visualInfo); + + if( !isSubWindow) + XPeekIfEvent( fgDisplay.Display, &eventReturnBuffer, &fghWindowIsVisible, (XPointer)(window->Window.Handle) ); + +#elif TARGET_HOST_MS_WINDOWS + + WNDCLASS wc; + DWORD flags = 0; + DWORD exFlags = 0; + ATOM atom; + + /* Grab the window class we have registered on glutInit(): */ + atom = GetClassInfo( fgDisplay.Instance, _T("FREEGLUT"), &wc ); + FREEGLUT_INTERNAL_ERROR_EXIT ( atom, "Window Class Info Not Found", + "fgOpenWindow" ); + + /* Determine window style flags*/ + if( gameMode ) + { + FREEGLUT_INTERNAL_ERROR_EXIT ( window->Parent == NULL, + "Game mode being invoked on a subwindow", + "fgOpenWindow" ); + + /* + * Set the window creation flags appropriately to make the window + * entirely visible: + */ + flags = WS_POPUP | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | WS_VISIBLE; + } + else + { + flags = WS_CLIPSIBLINGS | WS_CLIPCHILDREN; + + /* + * There's a small difference between creating the top, child and + * menu windows + */ + if ( window->IsMenu ) + { + flags |= WS_POPUP; + exFlags |= WS_EX_TOOLWINDOW; + } +#if defined(_WIN32_WCE) + /* no decorations for windows CE */ +#else + /* if this is not a subwindow (child), set its style based on the requested display mode */ + else if( window->Parent == NULL ) + if ( fgState.DisplayMode & GLUT_BORDERLESS ) + { + /* no window decorations needed */ + } + else if ( fgState.DisplayMode & GLUT_CAPTIONLESS ) + /* only window decoration is a border, no title bar or buttons */ + flags |= WS_DLGFRAME; + else + /* window decoration are a border, title bar and buttons. + * NB: we later query whether the window has a title bar or + * not by testing for the maximize button, as the test for + * WS_CAPTION can be true without the window having a title + * bar. This style WS_OVERLAPPEDWINDOW gives you a maximize + * button. */ + flags |= WS_OVERLAPPEDWINDOW; +#endif + else + /* subwindows always have no decoration, but are marked as a child window to the OS */ + flags |= WS_CHILD; + } + + /* determine window size and position */ + if( gameMode ) + { + /* if in gamemode, query the origin of specified by the -display + * command line parameter (if any) and offset the upper-left corner + * of the window so we create the window on that screen. + * The -display argument doesn't do anything if not trying to enter + * gamemode. + */ + int xoff=0, yoff=0; + get_display_origin(&xoff,&yoff); + x += xoff; + y += yoff; + } + if( !positionUse ) + { + x = CW_USEDEFAULT; + y = CW_USEDEFAULT; + } + if( !sizeUse ) + { + if( ! window->IsMenu ) + { + w = CW_USEDEFAULT; + h = CW_USEDEFAULT; + } + else /* fail safe - Windows can make a window of size (0, 0) */ + w = h = 300; /* default window size */ + } + /* store requested client area width and height */ + window->State.Width = w; + window->State.Height = h; + +#if !defined(_WIN32_WCE) /* no decorations for windows CE */ + if( sizeUse ) + { + RECT windowRect; + /* + * Update the window dimensions, taking the window decorations + * into account. FreeGLUT is to create the window with the + * topleft outside corner at (x,y) and with client area + * dimensions (w,h). + * note: don't need to do this when w=h=CW_USEDEFAULT, so in the + * if( sizeUse ) here is convenient. + */ + windowRect.left = x; + windowRect.top = y; + windowRect.right = x+w; + windowRect.bottom = y+h; + + fghComputeWindowRectFromClientArea_UseStyle(flags,&windowRect,TRUE); + + w = windowRect.right - windowRect.left; + h = windowRect.bottom- windowRect.top; + } +#endif /* !defined(_WIN32_WCE) */ + +#if defined(_WIN32_WCE) + { + wchar_t* wstr = fghWstrFromStr(title); + + window->Window.Handle = CreateWindow( + _T("FREEGLUT"), + wstr, + WS_VISIBLE | WS_POPUP, + 0,0, 240,320, + NULL, + NULL, + fgDisplay.Instance, + (LPVOID) window + ); + + free(wstr); + + SHFullScreen(window->Window.Handle, SHFS_HIDESTARTICON); + SHFullScreen(window->Window.Handle, SHFS_HIDESIPBUTTON); + SHFullScreen(window->Window.Handle, SHFS_HIDETASKBAR); + MoveWindow(window->Window.Handle, 0, 0, 240, 320, TRUE); + ShowWindow(window->Window.Handle, SW_SHOW); + UpdateWindow(window->Window.Handle); + } +#else + window->Window.Handle = CreateWindowEx( + exFlags, + _T("FREEGLUT"), + title, + flags, + x, y, w, h, + (HWND) window->Parent == NULL ? NULL : window->Parent->Window.Handle, + (HMENU) NULL, + fgDisplay.Instance, + (LPVOID) window + ); +#endif /* defined(_WIN32_WCE) */ + + if( !( window->Window.Handle ) ) + fgError( "Failed to create a window (%s)!", title ); + +#if !defined(_WIN32_WCE) + /* Need to set requested style again, apparently Windows doesn't listen when requesting windows without title bar or borders */ + SetWindowLong(window->Window.Handle, GWL_STYLE, flags); + SetWindowPos(window->Window.Handle, HWND_TOP, 0,0,0,0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_FRAMECHANGED); +#endif /* defined(_WIN32_WCE) */ + + /* Make a menu window always on top - fix Feature Request 947118 */ + if( window->IsMenu || gameMode ) + SetWindowPos( + window->Window.Handle, + HWND_TOPMOST, + 0, 0, 0, 0, + SWP_NOMOVE | SWP_NOSIZE + ); + + /* Enable multitouch: additional flag TWF_FINETOUCH, TWF_WANTPALM */ + #ifdef WM_TOUCH + if (fghRegisterTouchWindow == (pRegisterTouchWindow)0xDEADBEEF) + fghRegisterTouchWindow = (pRegisterTouchWindow)GetProcAddress(GetModuleHandle("user32"),"RegisterTouchWindow"); + if (fghRegisterTouchWindow) + fghRegisterTouchWindow( window->Window.Handle, TWF_FINETOUCH | TWF_WANTPALM ); + #endif + +#if defined(_WIN32_WCE) + ShowWindow( window->Window.Handle, SW_SHOW ); +#else + ShowWindow( window->Window.Handle, + fgState.ForceIconic ? SW_SHOWMINIMIZED : SW_SHOW ); +#endif /* defined(_WIN32_WCE) */ + + UpdateWindow( window->Window.Handle ); + ShowCursor( TRUE ); /* XXX Old comments say "hide cursor"! */ + +#endif + + fgSetWindow( window ); + + window->Window.DoubleBuffered = + ( fgState.DisplayMode & GLUT_DOUBLE ) ? 1 : 0; + + if ( ! window->Window.DoubleBuffered ) + { + glDrawBuffer ( GL_FRONT ); + glReadBuffer ( GL_FRONT ); + } +} + +/* + * Closes a window, destroying the frame and OpenGL context + */ +void fgCloseWindow( SFG_Window* window ) +{ + /* if we're in gamemode and we're closing the gamemode window, + * call glutLeaveGameMode first to make sure the gamemode is + * properly closed before closing the window + */ + if (fgStructure.GameModeWindow != NULL && fgStructure.GameModeWindow->ID==window->ID) + glutLeaveGameMode(); + +#if TARGET_HOST_POSIX_X11 + + if( window->Window.Context ) + glXDestroyContext( fgDisplay.Display, window->Window.Context ); + XFree( window->Window.FBConfig ); + + if( window->Window.Handle ) { + XDestroyWindow( fgDisplay.Display, window->Window.Handle ); + } + /* XFlush( fgDisplay.Display ); */ /* XXX Shouldn't need this */ + +#elif TARGET_HOST_MS_WINDOWS + + /* Make sure we don't close a window with current context active */ + if( fgStructure.CurrentWindow == window ) + wglMakeCurrent( NULL, NULL ); + + /* + * Step through the list of windows. If the rendering context + * is not being used by another window, then we delete it. + */ + { + int used = FALSE ; + SFG_Window *iter ; + + for( iter = (SFG_Window *)fgStructure.Windows.First; + iter; + iter = (SFG_Window *)iter->Node.Next ) + { + if( ( iter->Window.Context == window->Window.Context ) && + ( iter != window ) ) + used = TRUE; + } + + if( ! used ) + wglDeleteContext( window->Window.Context ); + } + + DestroyWindow( window->Window.Handle ); +#endif +} + + +/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ + +/* + * Creates a new top-level freeglut window + */ +int FGAPIENTRY glutCreateWindow( const char* title ) +{ + /* XXX GLUT does not exit; it simply calls "glutInit" quietly if the + * XXX application has not already done so. The "freeglut" community + * XXX decided not to go this route (freeglut-developer e-mail from + * XXX Steve Baker, 12/16/04, 4:22 PM CST, "Re: [Freeglut-developer] + * XXX Desired 'freeglut' behaviour when there is no current window" + */ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutCreateWindow" ); + + return fgCreateWindow( NULL, title, fgState.Position.Use, + fgState.Position.X, fgState.Position.Y, + fgState.Size.Use, fgState.Size.X, fgState.Size.Y, + GL_FALSE, GL_FALSE )->ID; +} + +#if TARGET_HOST_MS_WINDOWS +int FGAPIENTRY __glutCreateWindowWithExit( const char *title, void (__cdecl *exit_function)(int) ) +{ + __glutExitFunc = exit_function; + return glutCreateWindow( title ); +} +#endif + +/* + * This function creates a sub window. + */ +int FGAPIENTRY glutCreateSubWindow( int parentID, int x, int y, int w, int h ) +{ + int ret = 0; + SFG_Window* window = NULL; + SFG_Window* parent = NULL; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutCreateSubWindow" ); + parent = fgWindowByID( parentID ); + freeglut_return_val_if_fail( parent != NULL, 0 ); + if ( x < 0 ) + { + x = parent->State.Width + x ; + if ( w >= 0 ) x -= w ; + } + + if ( w < 0 ) w = parent->State.Width - x + w ; + if ( w < 0 ) + { + x += w ; + w = -w ; + } + + if ( y < 0 ) + { + y = parent->State.Height + y ; + if ( h >= 0 ) y -= h ; + } + + if ( h < 0 ) h = parent->State.Height - y + h ; + if ( h < 0 ) + { + y += h ; + h = -h ; + } + + window = fgCreateWindow( parent, "", GL_TRUE, x, y, GL_TRUE, w, h, GL_FALSE, GL_FALSE ); + ret = window->ID; + + return ret; +} + +/* + * Destroys a window and all of its subwindows + */ +void FGAPIENTRY glutDestroyWindow( int windowID ) +{ + SFG_Window* window; + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDestroyWindow" ); + window = fgWindowByID( windowID ); + freeglut_return_if_fail( window != NULL ); + { + fgExecutionState ExecState = fgState.ExecState; + fgAddToWindowDestroyList( window ); + fgState.ExecState = ExecState; + } +} + +/* + * This function selects the current window + */ +void FGAPIENTRY glutSetWindow( int ID ) +{ + SFG_Window* window = NULL; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetWindow" ); + if( fgStructure.CurrentWindow != NULL ) + if( fgStructure.CurrentWindow->ID == ID ) + return; + + window = fgWindowByID( ID ); + if( window == NULL ) + { + fgWarning( "glutSetWindow(): window ID %d not found!", ID ); + return; + } + + fgSetWindow( window ); +} + +/* + * This function returns the ID number of the current window, 0 if none exists + */ +int FGAPIENTRY glutGetWindow( void ) +{ + SFG_Window *win = fgStructure.CurrentWindow; + /* + * Since GLUT did not throw an error if this function was called without a prior call to + * "glutInit", this function shouldn't do so here. Instead let us return a zero. + * See Feature Request "[ 1307049 ] glutInit check". + */ + if ( ! fgState.Initialised ) + return 0; + + while ( win && win->IsMenu ) + win = win->Parent; + return win ? win->ID : 0; +} + +/* + * This function makes the current window visible + */ +void FGAPIENTRY glutShowWindow( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutShowWindow" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutShowWindow" ); + +#if TARGET_HOST_POSIX_X11 + + XMapWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle ); + XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ + +#elif TARGET_HOST_MS_WINDOWS + + ShowWindow( fgStructure.CurrentWindow->Window.Handle, SW_SHOW ); + +#endif + + fgStructure.CurrentWindow->State.Redisplay = GL_TRUE; +} + +/* + * This function hides the current window + */ +void FGAPIENTRY glutHideWindow( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutHideWindow" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutHideWindow" ); + +#if TARGET_HOST_POSIX_X11 + + if( fgStructure.CurrentWindow->Parent == NULL ) + XWithdrawWindow( fgDisplay.Display, + fgStructure.CurrentWindow->Window.Handle, + fgDisplay.Screen ); + else + XUnmapWindow( fgDisplay.Display, + fgStructure.CurrentWindow->Window.Handle ); + XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ + +#elif TARGET_HOST_MS_WINDOWS + + ShowWindow( fgStructure.CurrentWindow->Window.Handle, SW_HIDE ); + +#endif + + fgStructure.CurrentWindow->State.Redisplay = GL_FALSE; +} + +/* + * Iconify the current window (top-level windows only) + */ +void FGAPIENTRY glutIconifyWindow( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutIconifyWindow" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutIconifyWindow" ); + + fgStructure.CurrentWindow->State.Visible = GL_FALSE; +#if TARGET_HOST_POSIX_X11 + + XIconifyWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle, + fgDisplay.Screen ); + XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ + +#elif TARGET_HOST_MS_WINDOWS + + ShowWindow( fgStructure.CurrentWindow->Window.Handle, SW_MINIMIZE ); + +#endif + + fgStructure.CurrentWindow->State.Redisplay = GL_FALSE; +} + +/* + * Set the current window's title + */ +void FGAPIENTRY glutSetWindowTitle( const char* title ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetWindowTitle" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutSetWindowTitle" ); + if( ! fgStructure.CurrentWindow->Parent ) + { +#if TARGET_HOST_POSIX_X11 + + XTextProperty text; + + text.value = (unsigned char *) title; + text.encoding = XA_STRING; + text.format = 8; + text.nitems = strlen( title ); + + XSetWMName( + fgDisplay.Display, + fgStructure.CurrentWindow->Window.Handle, + &text + ); + + XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ + +#elif TARGET_HOST_MS_WINDOWS +# ifdef _WIN32_WCE + { + wchar_t* wstr = fghWstrFromStr(title); + SetWindowText( fgStructure.CurrentWindow->Window.Handle, wstr ); + free(wstr); + } +# else + SetWindowText( fgStructure.CurrentWindow->Window.Handle, title ); +# endif + +#endif + } +} + +/* + * Set the current window's iconified title + */ +void FGAPIENTRY glutSetIconTitle( const char* title ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetIconTitle" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutSetIconTitle" ); + + if( ! fgStructure.CurrentWindow->Parent ) + { +#if TARGET_HOST_POSIX_X11 + + XTextProperty text; + + text.value = (unsigned char *) title; + text.encoding = XA_STRING; + text.format = 8; + text.nitems = strlen( title ); + + XSetWMIconName( + fgDisplay.Display, + fgStructure.CurrentWindow->Window.Handle, + &text + ); + + XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ + +#elif TARGET_HOST_MS_WINDOWS +# ifdef _WIN32_WCE + { + wchar_t* wstr = fghWstrFromStr(title); + SetWindowText( fgStructure.CurrentWindow->Window.Handle, wstr ); + free(wstr); + } +# else + SetWindowText( fgStructure.CurrentWindow->Window.Handle, title ); +# endif + +#endif + } +} + +/* + * Change the current window's size + */ +void FGAPIENTRY glutReshapeWindow( int width, int height ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutReshapeWindow" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutReshapeWindow" ); + + if (glutGet(GLUT_FULL_SCREEN)) + { + /* Leave full screen state before resizing. */ + glutLeaveFullScreen(); + } + + fgStructure.CurrentWindow->State.NeedToResize = GL_TRUE; + fgStructure.CurrentWindow->State.Width = width ; + fgStructure.CurrentWindow->State.Height = height; +} + +/* + * Change the current window's position + */ +void FGAPIENTRY glutPositionWindow( int x, int y ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPositionWindow" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutPositionWindow" ); + + if (glutGet(GLUT_FULL_SCREEN)) + { + /* Leave full screen state before moving. */ + glutLeaveFullScreen(); + } + +#if TARGET_HOST_POSIX_X11 + + XMoveWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle, + x, y ); + XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ + +#elif TARGET_HOST_MS_WINDOWS + + { + RECT winRect; + + /* "GetWindowRect" returns the pixel coordinates of the outside of the window */ + GetWindowRect( fgStructure.CurrentWindow->Window.Handle, &winRect ); + MoveWindow( + fgStructure.CurrentWindow->Window.Handle, + x, + y, + winRect.right - winRect.left, + winRect.bottom - winRect.top, + TRUE + ); + } + +#endif +} + +/* + * Lowers the current window (by Z order change) + */ +void FGAPIENTRY glutPushWindow( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPushWindow" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutPushWindow" ); + +#if TARGET_HOST_POSIX_X11 + + XLowerWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle ); + +#elif TARGET_HOST_MS_WINDOWS + + SetWindowPos( + fgStructure.CurrentWindow->Window.Handle, + HWND_BOTTOM, + 0, 0, 0, 0, + SWP_NOSIZE | SWP_NOMOVE + ); + +#endif +} + +/* + * Raises the current window (by Z order change) + */ +void FGAPIENTRY glutPopWindow( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPopWindow" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutPopWindow" ); + +#if TARGET_HOST_POSIX_X11 + + XRaiseWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle ); + +#elif TARGET_HOST_MS_WINDOWS + + SetWindowPos( + fgStructure.CurrentWindow->Window.Handle, + HWND_TOP, + 0, 0, 0, 0, + SWP_NOSIZE | SWP_NOMOVE + ); + +#endif +} + +/* + * Resize the current window so that it fits the whole screen + */ +void FGAPIENTRY glutFullScreen( void ) +{ + SFG_Window *win; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutFullScreen" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutFullScreen" ); + + win = fgStructure.CurrentWindow; + + if (win->Parent) + { + /* Child windows cannot be made fullscreen, consistent with GLUT's behavior + * Also, what would it mean for a child window to be fullscreen, given that it + * is confined to its parent? + */ + fgWarning("glutFullScreen called on a child window, ignoring..."); + return; + } + else if (fgStructure.GameModeWindow != NULL && fgStructure.GameModeWindow->ID==win->ID) + { + /* Ignore fullscreen call on GameMode window, those are always fullscreen already */ + return; + } + +#if TARGET_HOST_POSIX_X11 + if(!glutGet(GLUT_FULL_SCREEN)) { + if(fghToggleFullscreen() != -1) { + win->State.IsFullscreen = GL_TRUE; + } + } + +#elif TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) /* FIXME: what about WinCE */ + + if (glutGet(GLUT_FULL_SCREEN)) + { + /* Leave full screen state before entering fullscreen again (resizing?) */ + glutLeaveFullScreen(); + } + + { +#if(WINVER >= 0x0500) /* Windows 2000 or later */ + DWORD s; + RECT rect; + HMONITOR hMonitor; + MONITORINFO mi; + + /* For fullscreen mode, first remove all window decoration + * and set style to popup so it will overlap the taskbar + * then force to maximize on the screen on which it has the most + * overlap. + */ + + + /* store current window rect */ + GetWindowRect( win->Window.Handle, &win->State.OldRect ); + + /* store current window style */ + win->State.OldStyle = s = GetWindowLong(win->Window.Handle, GWL_STYLE); + + /* remove decorations from style and add popup style*/ + s &= ~WS_OVERLAPPEDWINDOW; + s |= WS_POPUP; + SetWindowLong(win->Window.Handle, GWL_STYLE, s); + SetWindowPos(win->Window.Handle, HWND_TOP, 0,0,0,0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_FRAMECHANGED); + + /* For fullscreen mode, find the monitor that is covered the most + * by the window and get its rect as the resize target. + */ + hMonitor= MonitorFromRect(&win->State.OldRect, MONITOR_DEFAULTTONEAREST); + mi.cbSize = sizeof(mi); + GetMonitorInfo(hMonitor, &mi); + rect = mi.rcMonitor; +#else /* if (WINVER >= 0x0500) */ + RECT rect; + + /* For fullscreen mode, force the top-left corner to 0,0 + * and adjust the window rectangle so that the client area + * covers the whole screen. + */ + + rect.left = 0; + rect.top = 0; + rect.right = fgDisplay.ScreenWidth; + rect.bottom = fgDisplay.ScreenHeight; + + AdjustWindowRect ( &rect, WS_OVERLAPPEDWINDOW | WS_CLIPSIBLINGS | + WS_CLIPCHILDREN, FALSE ); +#endif /* (WINVER >= 0x0500) */ + + /* + * then resize window + * SWP_NOACTIVATE Do not activate the window + * SWP_NOOWNERZORDER Do not change position in z-order + * SWP_NOSENDCHANGING Suppress WM_WINDOWPOSCHANGING message + * SWP_NOZORDER Retains the current Z order (ignore 2nd param) + */ + SetWindowPos( fgStructure.CurrentWindow->Window.Handle, + HWND_TOP, + rect.left, + rect.top, + rect.right - rect.left, + rect.bottom - rect.top, + SWP_NOACTIVATE | SWP_NOOWNERZORDER | SWP_NOSENDCHANGING | + SWP_NOZORDER + ); + + win->State.IsFullscreen = GL_TRUE; + } +#endif +} + +/* + * If we are fullscreen, resize the current window back to its original size + */ +void FGAPIENTRY glutLeaveFullScreen( void ) +{ + SFG_Window *win; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutFullScreen" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutFullScreen" ); + + win = fgStructure.CurrentWindow; + +#if TARGET_HOST_POSIX_X11 + if(glutGet(GLUT_FULL_SCREEN)) { + if(fghToggleFullscreen() != -1) { + win->State.IsFullscreen = GL_FALSE; + } + } + +#elif TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) /* FIXME: what about WinCE */ + if (!glutGet(GLUT_FULL_SCREEN)) + { + /* nothing to do */ + return; + } + + /* restore style of window before making it fullscreen */ + SetWindowLong(win->Window.Handle, GWL_STYLE, win->State.OldStyle); + SetWindowPos(win->Window.Handle, HWND_TOP, 0,0,0,0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_FRAMECHANGED); + + /* Then resize */ + SetWindowPos(win->Window.Handle, + HWND_TOP, + win->State.OldRect.left, + win->State.OldRect.top, + win->State.OldRect.right - win->State.OldRect.left, + win->State.OldRect.bottom - win->State.OldRect.top, + SWP_NOACTIVATE | SWP_NOOWNERZORDER | SWP_NOSENDCHANGING | + SWP_NOZORDER + ); + + win->State.IsFullscreen = GL_FALSE; +#endif +} + +/* + * Toggle the window's full screen state. + */ +void FGAPIENTRY glutFullScreenToggle( void ) +{ + SFG_Window *win; + + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutFullScreenToggle" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutFullScreenToggle" ); + + win = fgStructure.CurrentWindow; + +#if TARGET_HOST_POSIX_X11 + if(fghToggleFullscreen() != -1) { + win->State.IsFullscreen = !win->State.IsFullscreen; + } +#elif TARGET_HOST_MS_WINDOWS + if (!win->State.IsFullscreen) + glutFullScreen(); + else + glutLeaveFullScreen(); +#endif +} + +/* + * A.Donev: Set and retrieve the window's user data + */ +void* FGAPIENTRY glutGetWindowData( void ) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetWindowData" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutGetWindowData" ); + return fgStructure.CurrentWindow->UserData; +} + +void FGAPIENTRY glutSetWindowData(void* data) +{ + FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetWindowData" ); + FREEGLUT_EXIT_IF_NO_WINDOW ( "glutSetWindowData" ); + fgStructure.CurrentWindow->UserData = data; +} + +/*** END OF FILE ***/ diff --git a/examples/opengl-framework/freeglut/freeglut_xinput.c b/examples/opengl-framework/freeglut/freeglut_xinput.c new file mode 100644 index 00000000..b0d8cc7a --- /dev/null +++ b/examples/opengl-framework/freeglut/freeglut_xinput.c @@ -0,0 +1,219 @@ +/* Written for XI1 by Nikolas Doerfler (c) 2008 * + * Rewritten for XI2 by Florian Echtler (c) 2009 */ + +#include + +#include "freeglut_internal.h" + +#if TARGET_HOST_POSIX_X11 && HAVE_X11_EXTENSIONS_XINPUT2_H + +#include +#include + +#include +#include + +/* import function from freeglut_main.c */ +int fghGetXModifiers( int state ); + +/* extension opcode for XInput */ +int xi_opcode = -1; + +/** + * \brief Sets window up for XI2 events. + */ +void fgRegisterDevices( Display* dpy, Window* win ) { + + XIEventMask mask; + unsigned char flags[2] = { 0, 0 }; + int event, error; + + /*Display* dpy = fgDisplay.Display; + Window* win = glutGetXWindow();*/ + + /* get XInput extension opcode */ + if (!XQueryExtension( dpy, "XInputExtension", &xi_opcode, &event, &error )) { xi_opcode = -1; } + + /* Select for motion events */ + mask.deviceid = XIAllMasterDevices; + mask.mask_len = 2; + mask.mask = flags; + + XISetMask(mask.mask, XI_Enter); + XISetMask(mask.mask, XI_Motion); + XISetMask(mask.mask, XI_ButtonPress); + XISetMask(mask.mask, XI_ButtonRelease); + XISetMask(mask.mask, XI_Leave); + /*XISetMask(mask.mask, XI_KeyPress); + XISetMask(mask.mask, XI_KeyRelease); + XISetMask(mask.mask, XI_DeviceChanged); + XISetMask(mask.mask, XI_RawEvent); + XISetMask(mask.mask, XI_FocusIn); + XISetMask(mask.mask, XI_FocusOut); + XISetMask(mask.mask, XI_HierarchyChanged);*/ + + XISelectEvents( dpy, *win, &mask, 1 ); +} + + +void fgPrintXILeaveEvent(XILeaveEvent* event) +{ + char* mode = ""; + char* detail = ""; + int i; + + printf(" windows: root 0x%lx event 0x%lx child 0x%ld\n", + event->root, event->event, event->child); + switch(event->mode) + { + case NotifyNormal: mode = "NotifyNormal"; break; + case NotifyGrab: mode = "NotifyGrab"; break; + case NotifyUngrab: mode = "NotifyUngrab"; break; + case NotifyWhileGrabbed: mode = "NotifyWhileGrabbed"; break; + } + switch (event->detail) + { + case NotifyAncestor: detail = "NotifyAncestor"; break; + case NotifyVirtual: detail = "NotifyVirtual"; break; + case NotifyInferior: detail = "NotifyInferior"; break; + case NotifyNonlinear: detail = "NotifyNonlinear"; break; + case NotifyNonlinearVirtual: detail = "NotifyNonlinearVirtual"; break; + case NotifyPointer: detail = "NotifyPointer"; break; + case NotifyPointerRoot: detail = "NotifyPointerRoot"; break; + case NotifyDetailNone: detail = "NotifyDetailNone"; break; + } + printf(" mode: %s (detail %s)\n", mode, detail); + printf(" flags: %s %s\n", event->focus ? "[focus]" : "", + event->same_screen ? "[same screen]" : ""); + printf(" buttons:"); + for (i = 0; i < event->buttons.mask_len * 8; i++) + if (XIMaskIsSet(event->buttons.mask, i)) + printf(" %d", i); + printf("\n"); + + printf(" modifiers: locked 0x%x latched 0x%x base 0x%x\n", + event->mods.locked, event->mods.latched, + event->mods.base); + printf(" group: locked 0x%x latched 0x%x base 0x%x\n", + event->group.locked, event->group.latched, + event->group.base); + + printf(" root x/y: %.2f / %.2f\n", event->root_x, event->root_y); + printf(" event x/y: %.2f / %.2f\n", event->event_x, event->event_y); + +} + + +void fgPrintXIDeviceEvent(XIDeviceEvent* event) +{ + double *val; + int i; + + printf(" device: %d (%d)\n", event->deviceid, event->sourceid); + printf(" detail: %d\n", event->detail); + printf(" buttons:"); + for (i = 0; i < event->buttons.mask_len * 8; i++) + if (XIMaskIsSet(event->buttons.mask, i)) + printf(" %d", i); + printf("\n"); + + printf(" modifiers: locked 0x%x latched 0x%x base 0x%x\n", + event->mods.locked, event->mods.latched, + event->mods.base); + printf(" group: locked 0x%x latched 0x%x base 0x%x\n", + event->group.locked, event->group.latched, + event->group.base); + printf(" valuators:"); + + val = event->valuators.values; + for (i = 0; i < event->valuators.mask_len * 8; i++) + if (XIMaskIsSet(event->valuators.mask, i)) + printf(" %d: %.2f", i, *val++); + printf("\n"); + + printf(" windows: root 0x%lx event 0x%lx child 0x%ld\n", + event->root, event->event, event->child); + printf(" root x/y: %.2f / %.2f\n", event->root_x, event->root_y); + printf(" event x/y: %.2f / %.2f\n", event->event_x, event->event_y); + +} + + +/** + * \brief This function is called when an Extension Event is received + * and calls the corresponding callback functions for these events. + */ +void fgHandleExtensionEvents( XEvent* base_ev ) { + + int i, button = 0; + XGenericEventCookie* cookie = (XGenericEventCookie*)&(base_ev->xcookie); + + if ( XGetEventData( fgDisplay.Display, cookie ) && (cookie->type == GenericEvent) && (cookie->extension == xi_opcode) ) { + + XIDeviceEvent* event = (XIDeviceEvent*)(cookie->data); + /*printf("XI2 event type: %d - %d\n", cookie->evtype, event->type );*/ + + SFG_Window* window = fgWindowByHandle( event->event ); + if (!window) return; + + switch (cookie->evtype) { + + case XI_Enter: + case XI_Leave: + fgState.Modifiers = fghGetXModifiers( ((XIEnterEvent*)event)->mods.base ); + INVOKE_WCB( *window, MultiEntry, ( + event->deviceid, + (event->evtype == XI_Enter ? GLUT_ENTERED : GLUT_LEFT) + )); + #if _DEBUG + fgPrintXILeaveEvent((XILeaveEvent*)event); + #endif + break; + + case XI_ButtonPress: + case XI_ButtonRelease: + fgState.Modifiers = fghGetXModifiers( event->mods.base ); + INVOKE_WCB( *window, MultiButton, ( + event->deviceid, + event->event_x, + event->event_y, + (event->detail)-1, + (event->evtype == XI_ButtonPress ? GLUT_DOWN : GLUT_UP) + )); + INVOKE_WCB( *window, Mouse, ( + (event->detail)-1, + (event->evtype == XI_ButtonPress ? GLUT_DOWN : GLUT_UP), + event->event_x, + event->event_y + )); + break; + + case XI_Motion: + fgState.Modifiers = fghGetXModifiers( event->mods.base ); + for (i = 0; i < event->buttons.mask_len; i++) if (event->buttons.mask[i]) button = 1; + if (button) { + INVOKE_WCB( *window, MultiMotion, ( event->deviceid, event->event_x, event->event_y ) ); + INVOKE_WCB( *window, Motion, ( event->event_x, event->event_y ) ); + } else { + INVOKE_WCB( *window, MultiPassive, ( event->deviceid, event->event_x, event->event_y ) ); + INVOKE_WCB( *window, Passive, ( event->event_x, event->event_y ) ); + } + #if _DEBUG + fgPrintXIDeviceEvent(event); + #endif + break; + + default: + #if _DEBUG + fgWarning( "Unknown XI2 device event:" ); + fgPrintXIDeviceEvent( event ); + #endif + break; + } + fgState.Modifiers = INVALID_MODIFIERS; + } + XFreeEventData( fgDisplay.Display, cookie ); +} + +#endif + diff --git a/examples/opengl-framework/freeglut/glut.h b/examples/opengl-framework/freeglut/glut.h new file mode 100644 index 00000000..6191f77b --- /dev/null +++ b/examples/opengl-framework/freeglut/glut.h @@ -0,0 +1,21 @@ +#ifndef __GLUT_H__ +#define __GLUT_H__ + +/* + * glut.h + * + * The freeglut library include file + * + * 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 + * PAWEL W. OLSZTA 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. + */ + +#include "freeglut_std.h" + +/*** END OF FILE ***/ + +#endif /* __GLUT_H__ */ diff --git a/examples/opengl-framework/src/Camera.cpp b/examples/opengl-framework/src/Camera.cpp new file mode 100644 index 00000000..a032a046 --- /dev/null +++ b/examples/opengl-framework/src/Camera.cpp @@ -0,0 +1,97 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + + +// Libraries +#include "Camera.h" +#include "definitions.h" +#include + +// Namespaces +using namespace openglframework; + +// Constructor +Camera::Camera() : Object3D() { + + // Set default values + mFieldOfView = 45.0f; + mSceneRadius = 1.0f; + mNearPlane = 0.1f; + mFarPlane = 10.0f; + mWidth = 1; + mHeight = 1; + + // Update the projection matrix + updateProjectionMatrix(); +} + +// Destructor +Camera::~Camera() { + +} + +// Update the projection matrix +void Camera::updateProjectionMatrix() { + + // Compute the aspect ratio + float aspect = float(mWidth) / float(mHeight); + + float top = mNearPlane * tan((mFieldOfView / 2.0f) * (float(PI) / 180.0f)); + float bottom = -top; + float left = bottom * aspect; + float right = top * aspect; + + float fx = 2.0f * mNearPlane / (right - left); + float fy = 2.0f * mNearPlane / (top - bottom); + float fz = -(mFarPlane + mNearPlane) / (mFarPlane - mNearPlane); + float fw = -2.0f * mFarPlane * mNearPlane / (mFarPlane - mNearPlane); + + // Recompute the projection matrix + mProjectionMatrix = Matrix4(fx, 0, 0, 0, + 0, fy, 0, 0, + 0, 0, fz, fw, + 0, 0, -1, 0); +} + +// Translate the camera go a given point using the dx, dy fraction +void Camera::translateCamera(float dx, float dy, const Vector3& worldPoint) { + + // Transform the world point into camera coordinates + Vector3 pointCamera = mTransformMatrix.getInverse() * worldPoint; + + // Get the depth + float z = -pointCamera.z; + + // Find the scaling of dx and dy from windows coordinates to near plane coordinates + // and from there to camera coordinates at the object's depth + float aspect = float(mWidth) / float(mHeight); + float top = mNearPlane * tan(mFieldOfView * PI / 360.0f); + float right = top * aspect; + + // Translate the camera + translateLocal(Vector3(2.0f * dx * right / mNearPlane * z, + -2.0f * dy * top / mNearPlane * z, + 0.0f)); +} diff --git a/examples/opengl-framework/src/Camera.h b/examples/opengl-framework/src/Camera.h new file mode 100644 index 00000000..76b24407 --- /dev/null +++ b/examples/opengl-framework/src/Camera.h @@ -0,0 +1,181 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 CAMERA_H +#define CAMERA_H + +// Libraries +#include "Object3D.h" +#include "definitions.h" + +namespace openglframework { + +// Class Camera +class Camera : public Object3D { + + protected : + + // ------------------- Attributes ------------------- // + + // Field of view + float mFieldOfView; + + // Radius of the scene + float mSceneRadius; + + // Near plane + float mNearPlane; + + // Far plane + float mFarPlane; + + // Width of the camera + uint mWidth; + + // Height of the camera + uint mHeight; + + // Projection matrix + Matrix4 mProjectionMatrix; + + // ------------------- Methods ------------------- // + + // Update the projection matrix + void updateProjectionMatrix(); + + public: + + // ------------------- Methods ------------------- // + + // Constructor + Camera(); + + // Destructor + ~Camera(); + + // Get the projection matrix + const Matrix4& getProjectionMatrix() const; + + // Set the dimensions of the camera + void setDimensions(uint width, uint height); + + // Get the radius of the scene the camera should capture + float getSceneRadius() const; + + // Set the radius of the scene the camera should capture + // This will update the clipping planes accordingly + void setSceneRadius(float radius); + + // Set the clipping planes + void setClippingPlanes(float near, float far); + + // Set the field of view + void setFieldOfView(float fov); + + // Set the zoom of the camera (a fraction between 0 and 1) + void setZoom(float fraction); + + // Translate the camera go a given point using the dx, dy fraction + void translateCamera(float dx, float dy, const Vector3& worldPoint); + + // Get the near clipping plane + float getNearClippingPlane() const; + + // Get the far clipping plane + float getFarClippingPlane() const; + + // Get the width + uint getWidth() const; + + // Get the height + uint getHeight() const; +}; + +// Get the projection matrix +inline const Matrix4& Camera::getProjectionMatrix() const { + return mProjectionMatrix; +} + +// Set the dimensions of the camera +inline void Camera::setDimensions(uint width, uint height) { + mWidth = width; + mHeight = height; + updateProjectionMatrix(); +} + +// Get the radius of the scene the camera should capture +inline float Camera::getSceneRadius() const { + return mSceneRadius; +} + +// Set the radius of the scene the camera should capture +// This will update the clipping planes accordingly +inline void Camera::setSceneRadius(float radius) { + mSceneRadius = radius; + setClippingPlanes(0.01f * radius, 10.0f * radius); +} + +// Set the clipping planes +inline void Camera::setClippingPlanes(float near, float far) { + mNearPlane = near; + mFarPlane = far; + updateProjectionMatrix(); +} + +// Set the field of view +inline void Camera::setFieldOfView(float fov) { + mFieldOfView = fov; + updateProjectionMatrix(); +} + +// Set the zoom of the camera (a fraction between 0 and 1) +inline void Camera::setZoom(float fraction) { + Vector3 zoomVector(0, 0, mSceneRadius * fraction * 3.0f); + translateLocal(zoomVector); +} + +// Get the near clipping plane +inline float Camera::getNearClippingPlane() const { + return mNearPlane; +} + +// Get the far clipping plane +inline float Camera::getFarClippingPlane() const { + return mFarPlane; +} + +// Get the width +inline uint Camera::getWidth() const { + return mWidth; +} + +// Get the height +inline uint Camera::getHeight() const { + return mHeight; +} + +} + +#endif diff --git a/examples/opengl-framework/src/FrameBufferObject.cpp b/examples/opengl-framework/src/FrameBufferObject.cpp new file mode 100644 index 00000000..5f67be23 --- /dev/null +++ b/examples/opengl-framework/src/FrameBufferObject.cpp @@ -0,0 +1,115 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "FrameBufferObject.h" +#include + +using namespace openglframework; +using namespace std; + +// Constructor +FrameBufferObject::FrameBufferObject() : mFrameBufferID(0), mRenderBufferID (0) { + +} + +// Destructor +FrameBufferObject::~FrameBufferObject() { + +} + +// Create the frame buffer object +bool FrameBufferObject::create(uint width, uint height, bool needRenderBuffer) { + + // Destroy the current FBO + destroy(); + + // Check that the needed OpenGL extensions are available + bool isExtensionOK = checkOpenGLExtensions(); + if (!isExtensionOK) { + std::cerr << "Error : Impossible to use Framebuffer Object on this platform" << std::endl; + assert(false); + return false; + } + + // Generate a new FBO + glGenFramebuffersEXT(1, &mFrameBufferID); + assert(mFrameBufferID != 0); + + // If we also need to create a render buffer + if (needRenderBuffer) { + + // Generate the render buffer + glGenRenderbuffers(1, &mRenderBufferID); + assert(mRenderBufferID != 0); + + glBindRenderbuffer(GL_RENDERBUFFER, mRenderBufferID); + glRenderbufferStorage(GL_RENDERBUFFER, GL_DEPTH_COMPONENT32, width, height); + glBindFramebuffer(GL_FRAMEBUFFER, mFrameBufferID); + glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_RENDERBUFFER, + mRenderBufferID); + glBindRenderbuffer(GL_RENDERBUFFER, 0); + glBindFramebuffer(GL_FRAMEBUFFER, 0); + } + + // Check the FBO status + GLenum statusFBO = glCheckFramebufferStatus(GL_FRAMEBUFFER); + if(statusFBO != GL_FRAMEBUFFER_COMPLETE) { + std::cerr << "Error : An error occured while creating the Frame Buffer Object !" << endl; + assert(false); + return false; + } + + return true; +} + +// Destroy the FBO +void FrameBufferObject::destroy() { + + // Delete the frame buffer object + if (mFrameBufferID) { + glDeleteFramebuffers(1, &mFrameBufferID); + mFrameBufferID = 0; + } + + // Delete the render buffer + if (mRenderBufferID) { + glDeleteRenderbuffers(1, &mRenderBufferID); + } +} + +// Attach a texture to the frame buffer object +void FrameBufferObject::attachTexture(uint position, uint textureID) { + assert(mFrameBufferID); + + // Bind the current FBO + glBindFramebuffer(GL_FRAMEBUFFER, mFrameBufferID); + + // Bind the texture + glFramebufferTexture2D(GL_FRAMEBUFFER, position, GL_TEXTURE_2D, textureID, 0); + + // Unbind the current FBO + glBindFramebuffer(GL_FRAMEBUFFER, 0); +} diff --git a/examples/opengl-framework/src/FrameBufferObject.h b/examples/opengl-framework/src/FrameBufferObject.h new file mode 100644 index 00000000..a76951c7 --- /dev/null +++ b/examples/opengl-framework/src/FrameBufferObject.h @@ -0,0 +1,105 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 FRAME_BUFFER_OBJECT_H +#define FRAME_BUFFER_OBJECT_H + +// Libraries +#include "definitions.h" +#include +#include +#include + +namespace openglframework { + + +// Class FrameBufferObject +class FrameBufferObject { + + private: + + // -------------------- Attributes -------------------- // + + // Frame buffer ID + uint mFrameBufferID; + + // Render buffer ID + uint mRenderBufferID; + + public: + + // -------------------- Methods -------------------- // + + // Constructor + FrameBufferObject(); + + // Destructor + ~FrameBufferObject(); + + // Create the frame buffer object + bool create(uint width, uint height, bool needRenderBuffer = true); + + // Attach a texture to the frame buffer object + void attachTexture(uint position, uint textureID); + + // Bind the FBO + void bind(uint position) const; + + // Unbind the FBO + void unbind() const; + + // Return true if the needed OpenGL extensions are available for FBO + static bool checkOpenGLExtensions(); + + // Destroy the FBO + void destroy(); +}; + +// Bind the FBO +inline void FrameBufferObject::bind(uint position) const { + assert(mFrameBufferID != 0); + glBindFramebuffer(GL_FRAMEBUFFER, mFrameBufferID); + glDrawBuffer(position); + glReadBuffer(position); +} + +// Unbind the FBO +inline void FrameBufferObject::unbind() const { + assert(mFrameBufferID != 0); + glDrawBuffer(GL_NONE); + glReadBuffer(GL_NONE); + glBindFramebuffer(GL_FRAMEBUFFER, 0); +} + +// Return true if the needed OpenGL extensions are available for FBO +inline bool FrameBufferObject::checkOpenGLExtensions() { + + // Check that OpenGL version is at least 3.0 or there the framebuffer object extension exists + return (GLEW_VERSION_3_0 || GLEW_ARB_framebuffer_object); +} + +} + +#endif diff --git a/examples/opengl-framework/src/GlutViewer.cpp b/examples/opengl-framework/src/GlutViewer.cpp new file mode 100644 index 00000000..d0dc9bed --- /dev/null +++ b/examples/opengl-framework/src/GlutViewer.cpp @@ -0,0 +1,260 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "GlutViewer.h" +#include + +// Namespaces +using namespace openglframework; +using namespace std; + +// Constructor +GlutViewer::GlutViewer() { + + // Initialize the state of the mouse buttons + for (int i=0; i<10; i++) { + mIsButtonDown[i] = false; + } +} + +// Destructor +GlutViewer::~GlutViewer() { + +} + +// Initialize the viewer +bool GlutViewer::init(int argc, char** argv, const string& windowsTitle, + const Vector2& windowsSize, const Vector2& windowsPosition, + bool isMultisamplingActive) { + + // Initialize the GLUT library + bool outputValue = initGLUT(argc, argv, windowsTitle, windowsSize, + windowsPosition, isMultisamplingActive); + + // Active the multi-sampling by default + if (isMultisamplingActive) { + activateMultiSampling(true); + } + + return outputValue; +} + +// Initialize the GLUT library +bool GlutViewer::initGLUT(int argc, char** argv, const string& windowsTitle, + const Vector2& windowsSize, const Vector2& windowsPosition, + bool isMultisamplingActive) { + + // Initialize GLUT + glutInit(&argc, argv); + uint modeWithoutMultiSampling = GLUT_DOUBLE | GLUT_RGBA | GLUT_DEPTH; + uint modeWithMultiSampling = GLUT_DOUBLE | GLUT_RGBA | GLUT_DEPTH | GL_MULTISAMPLE; + uint displayMode = isMultisamplingActive ? modeWithMultiSampling : modeWithoutMultiSampling; + glutInitDisplayMode(displayMode); + + // Initialize the size of the GLUT windows + glutInitWindowSize(static_cast(windowsSize.x), + static_cast(windowsSize.y)); + + // Initialize the position of the GLUT windows + glutInitWindowPosition(static_cast(windowsPosition.x), + static_cast(windowsPosition.y)); + + // Create the GLUT windows + glutCreateWindow(windowsTitle.c_str()); + + // Initialize the GLEW library + GLenum error = glewInit(); + if (error != GLEW_OK) { + + // Problem: glewInit failed, something is wrong + cerr << "GLEW Error : " << glewGetErrorString(error) << std::endl; + assert(false); + return false; + } + + return true; +} + +// Set the camera so that we can view the whole scene +void GlutViewer::resetCameraToViewAll() { + + // Move the camera to the origin of the scene + mCamera.translateWorld(-mCamera.getOrigin()); + + // Move the camera to the center of the scene + mCamera.translateWorld(mCenterScene); + + // Set the zoom of the camera so that the scene center is + // in negative view direction of the camera + mCamera.setZoom(1.0); +} + +// Called when a GLUT mouse button event occurs +void GlutViewer::mouseButtonEvent(int button, int state, int x, int y) { + + // If the mouse button is pressed + if (state == GLUT_DOWN) { + mLastMouseX = x; + mLastMouseY = y; + mIsLastPointOnSphereValid = mapMouseCoordinatesToSphere(x, y, mLastPointOnSphere); + mIsButtonDown[button] = true; + } + else { // If the mouse button is released + mIsLastPointOnSphereValid = false; + mIsButtonDown[button] = false; + + // If it is a mouse wheel click event + if (button == 3) { + zoom(0, (int) (y - 0.05f * mCamera.getWidth())); + } + else if (button == 4) { + zoom(0, (int) (y + 0.05f * mCamera.getHeight())); + } + } + + mModifiers = glutGetModifiers(); + + // Notify GLUT to redisplay + glutPostRedisplay(); +} + +// Called when a GLUT mouse motion event occurs +void GlutViewer::mouseMotionEvent(int xMouse, int yMouse) { + + // Zoom + if ((mIsButtonDown[GLUT_LEFT_BUTTON] && mIsButtonDown[GLUT_MIDDLE_BUTTON]) || + (mIsButtonDown[GLUT_LEFT_BUTTON] && mModifiers == GLUT_ACTIVE_ALT)) { + zoom(xMouse, yMouse); + } + // Translation + else if (mIsButtonDown[GLUT_MIDDLE_BUTTON] || mIsButtonDown[GLUT_RIGHT_BUTTON] || + (mIsButtonDown[GLUT_LEFT_BUTTON] && (mModifiers == GLUT_ACTIVE_ALT))) { + translate(xMouse, yMouse); + } + // Rotation + else if (mIsButtonDown[GLUT_LEFT_BUTTON]) { + rotate(xMouse, yMouse); + } + + // Remember the mouse position + mLastMouseX = xMouse; + mLastMouseY = yMouse; + mIsLastPointOnSphereValid = mapMouseCoordinatesToSphere(xMouse, yMouse, mLastPointOnSphere); + + // Notify GLUT to redisplay + glutPostRedisplay(); +} + +// Map the mouse x,y coordinates to a point on a sphere +bool GlutViewer::mapMouseCoordinatesToSphere(int xMouse, int yMouse, Vector3& spherePoint) const { + + int width = mCamera.getWidth(); + int height = mCamera.getHeight(); + + if ((xMouse >= 0) && (xMouse <= width) && (yMouse >= 0) && (yMouse <= height)) { + float x = float(xMouse - 0.5f * width) / float(width); + float y = float(0.5f * height - yMouse) / float(height); + float sinx = sin(PI * x * 0.5f); + float siny = sin(PI * y * 0.5f); + float sinx2siny2 = sinx * sinx + siny * siny; + + // Compute the point on the sphere + spherePoint.x = sinx; + spherePoint.y = siny; + spherePoint.z = (sinx2siny2 < 1.0) ? sqrt(1.0f - sinx2siny2) : 0.0f; + + return true; + } + + return false; +} + +// Zoom the camera +void GlutViewer::zoom(int xMouse, int yMouse) { + float dy = static_cast(yMouse - mLastMouseY); + float h = static_cast(mCamera.getHeight()); + + // Zoom the camera + mCamera.setZoom(-dy / h); +} + +// Translate the camera +void GlutViewer::translate(int xMouse, int yMouse) { + float dx = static_cast(xMouse - mLastMouseX); + float dy = static_cast(yMouse - mLastMouseY); + + // Translate the camera + mCamera.translateCamera(-dx / float(mCamera.getWidth()), + -dy / float(mCamera.getHeight()), mCenterScene); +} + +// Rotate the camera +void GlutViewer::rotate(int xMouse, int yMouse) { + + if (mIsLastPointOnSphereValid) { + + Vector3 newPoint3D; + bool isNewPointOK = mapMouseCoordinatesToSphere(xMouse, yMouse, newPoint3D); + + if (isNewPointOK) { + Vector3 axis = mLastPointOnSphere.cross(newPoint3D); + float cosAngle = mLastPointOnSphere.dot(newPoint3D); + + float epsilon = std::numeric_limits::epsilon(); + if (fabs(cosAngle) < 1.0f && axis.length() > epsilon) { + axis.normalize(); + float angle = 2.0f * acos(cosAngle); + + // Rotate the camera around the center of the scene + mCamera.rotateAroundLocalPoint(axis, -angle, mCenterScene); + } + } + } +} + +// Check the OpenGL errors +void GlutViewer::checkOpenGLErrors() { + GLenum glError; + + // Get the OpenGL errors + glError = glGetError(); + + // While there are errors + while (glError != GL_NO_ERROR) { + + // Get the error string + const GLubyte* stringError = gluErrorString(glError); + + // Display the error + if (stringError) + cerr << "OpenGL Error #" << glError << "(" << gluErrorString(glError) << endl; + else + cerr << "OpenGL Error #" << glError << " (no message available)" << endl; + + // Get the next error + glError = glGetError(); + } +} diff --git a/examples/opengl-framework/src/GlutViewer.h b/examples/opengl-framework/src/GlutViewer.h new file mode 100644 index 00000000..b5916179 --- /dev/null +++ b/examples/opengl-framework/src/GlutViewer.h @@ -0,0 +1,160 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 GLUT_VIEWER_H +#define GLUT_VIEWER_H + +// Libraries +#include "Shader.h" +#include "Camera.h" +#include "maths/Vector2.h" +#include +#include +#include "GL/freeglut.h" + +namespace openglframework { + +// Class Renderer +class GlutViewer { + + private: + + // -------------------- Attributes -------------------- // + + // Camera + Camera mCamera; + + // Center of the scene + Vector3 mCenterScene; + + // Last mouse coordinates on the windows + int mLastMouseX, mLastMouseY; + + // Last point computed on a sphere (for camera rotation) + Vector3 mLastPointOnSphere; + + // True if the last point computed on a sphere (for camera rotation) is valid + bool mIsLastPointOnSphereValid; + + // State of the mouse buttons + bool mIsButtonDown[10]; + + // GLUT keyboard modifiers + int mModifiers; + + // -------------------- Methods -------------------- // + + // Initialize the GLUT library + bool initGLUT(int argc, char** argv, const std::string& windowsTitle, + const Vector2& windowsSize, const Vector2& windowsPosition, + bool isMultisamplingActive); + + bool mapMouseCoordinatesToSphere(int xMouse, int yMouse, Vector3& spherePoint) const; + + public: + + // -------------------- Methods -------------------- // + + // Constructor + GlutViewer(); + + // Destructor + ~GlutViewer(); + + // Initialize the viewer + bool init(int argc, char** argv, const std::string& windowsTitle, + const Vector2& windowsSize, const Vector2& windowsPosition, + bool isMultisamplingActive = false); + + // Called when the windows is reshaped + void reshape(int width, int height); + + // Set the scene position (where the camera needs to look at) + void setScenePosition(const Vector3& position, float sceneRadius); + + // Set the camera so that we can view the whole scene + void resetCameraToViewAll(); + + // Enable/Disable the multi-sampling for anti-aliasing + void activateMultiSampling(bool isActive) const; + + // Zoom the camera + void zoom(int xMouse, int yMouse); + + // Translate the camera + void translate(int xMouse, int yMouse); + + // Rotate the camera + void rotate(int xMouse, int yMouse); + + // Get the camera + Camera& getCamera(); + + void motion(int x, int y); + + // Called when a GLUT mouse button event occurs + void mouseButtonEvent(int button, int state, int x, int y); + + // Called when a GLUT mouse motion event occurs + void mouseMotionEvent(int xMouse, int yMouse); + + void keyboard(int key, int x, int y); + void special(int key, int x, int y); + + // Check the OpenGL errors + static void checkOpenGLErrors(); +}; + +// Set the dimension of the camera viewport +inline void GlutViewer::reshape(int width, int height) { + mCamera.setDimensions(width, height); + glViewport(0, 0, width, height); + glutPostRedisplay(); +} + +// Set the scene position (where the camera needs to look at) +inline void GlutViewer::setScenePosition(const Vector3& position, float sceneRadius) { + + // Set the position and radius of the scene + mCenterScene = position; + mCamera.setSceneRadius(sceneRadius); + + // Reset the camera position and zoom in order to view all the scene + resetCameraToViewAll(); +} + +// Get the camera +inline Camera& GlutViewer::getCamera() { + return mCamera; +} + +// Enable/Disable the multi-sampling for anti-aliasing +inline void GlutViewer::activateMultiSampling(bool isActive) const { + (isActive) ? glEnable(GL_MULTISAMPLE) : glDisable(GL_MULTISAMPLE); +} + +} + +#endif diff --git a/examples/opengl-framework/src/Light.cpp b/examples/opengl-framework/src/Light.cpp new file mode 100644 index 00000000..cd1831eb --- /dev/null +++ b/examples/opengl-framework/src/Light.cpp @@ -0,0 +1,100 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Light.h" + +// Namespaces +using namespace openglframework; + +// Constructor +Light::Light(GLuint id) + : mLightID(id), mDiffuseColor(Color::white()), + mSpecularColor(Color::white()), mIsActive(false) { + +} + +// Constructor +Light::Light(GLuint id, Color diffuseColor, Color specularColor) + : mLightID(id), mDiffuseColor(diffuseColor), + mSpecularColor(specularColor), mIsActive(false) { + +} + +// Destructor +Light::~Light() { + +} + +// Initialize the light +void Light::init() { + + // Enable the light + enable(); + + // Set the diffuse and specular color + GLfloat diffuseColor[] = {mDiffuseColor.r, mDiffuseColor.g, mDiffuseColor.b, mDiffuseColor.a}; + GLfloat specularColor[] = {mSpecularColor.r,mSpecularColor.g,mSpecularColor.b,mSpecularColor.a}; + glLightfv(mLightID, GL_DIFFUSE, diffuseColor); + glLightfv(mLightID, GL_SPECULAR, specularColor); +} + +// Create a shadow map associated with this light +bool Light::createShadowMap(uint width, uint height) { + + // Destroy the current shadow map + destroyShadowMap(); + + // Create the Framebuffer object to render the shadow map + bool isFBOCreated = mFBOShadowMap.create(width, height, false); + if (!isFBOCreated) { + std::cerr << "Error : Cannot create the Shadow Map !" << std::endl; + destroyShadowMap(); + return false; + } + + // Bind the Framebuffer object + mFBOShadowMap.bind(GL_NONE); + + // Create the shadow map depth texture + mShadowMap.create(width, height, GL_DEPTH_COMPONENT, GL_DEPTH_COMPONENT, GL_UNSIGNED_BYTE); + + // Attache the shadow map texture to the Framebuffer object + mFBOShadowMap.attachTexture(GL_DEPTH_ATTACHMENT_EXT, mShadowMap.getID()); + + // Unbind the Framebuffer object + mFBOShadowMap.unbind(); + + // TODO : Change the path of the shader here so that it does not depend on the build folder + bool isShaderCreated = mDepthShader.create("../../opengl-framework/src/shaders/depth.vert", + "../../opengl-framework/src/shaders/depth.vert"); + if (!isShaderCreated) { + std::cerr << "Error : Cannot create the Shadow Map !" << std::endl; + destroyShadowMap(); + return false; + } + + return true; +} diff --git a/examples/opengl-framework/src/Light.h b/examples/opengl-framework/src/Light.h new file mode 100644 index 00000000..8f8323d9 --- /dev/null +++ b/examples/opengl-framework/src/Light.h @@ -0,0 +1,181 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 LIGHT_H +#define LIGHT_H + +// Libraries +#include "maths/Color.h" +#include "maths/Vector3.h" +#include "Object3D.h" +#include "Texture2D.h" +#include "FrameBufferObject.h" +#include "Shader.h" +#include + +namespace openglframework { + +// Class Light +class Light : public Object3D { + + private: + + // -------------------- Attributes -------------------- // + + // OpenGL light ID + GLuint mLightID; + + // Diffuse color of the light + Color mDiffuseColor; + + // Specular color of the light + Color mSpecularColor; + + // True if the light is active + bool mIsActive; + + // Shadow map associated with this light + Texture2D mShadowMap; + + // Framebuffer object to render the shadow map + FrameBufferObject mFBOShadowMap; + + // Shader to render the depth of the scene into a texture + Shader mDepthShader; + + public: + + // -------------------- Methods -------------------- // + + // Constructor + Light(GLuint id); + + // Constructor + Light(GLuint id, Color diffuseColor, Color specularColor); + + // Destructor + virtual ~Light(); + + // Return the diffuse color + Color getDiffuseColor() const; + + // Set the diffuse color + void setDiffuseColor(const Color& color); + + // Return the specular color + Color getSpecularColor() const; + + // Set the specular color + void setSpecularColor(const Color& color); + + // Return true if the light is active + bool getIsActive() const; + + // Initialize the light + void init(); + + // Enable the light + void enable(); + + // Disable the light + void disable(); + + // Create a shadow map associated with this light + bool createShadowMap(uint width, uint height); + + // Call this method before rendering the scene for the shadow map computation + void startRenderingShadowMap(); + + // Call this method after having rendered the scene for the shadow map computation + void stopRenderingShadowMap(); + + // Destroy the shadow map associated with this light + void destroyShadowMap(); +}; + +// Return the diffuse color +inline Color Light::getDiffuseColor() const { + return mDiffuseColor; +} + +// Set the diffuse color +inline void Light::setDiffuseColor(const Color& color) { + mDiffuseColor = color; +} + +// Return the specular color +inline Color Light::getSpecularColor() const { + return mSpecularColor; +} + +// Set the specular color +inline void Light::setSpecularColor(const Color& color) { + mSpecularColor = color; +} + +// Return true if the light is active +inline bool Light::getIsActive() const { + return mIsActive; +} + +// Enable the light +inline void Light::enable() { + + mIsActive = true; + + // Enable the light + glEnable(GL_LIGHTING); + glEnable(GL_LIGHT0 + mLightID); +} + +// Disable the light +inline void Light::disable() { + + mIsActive = false; + + // Disable the light + glDisable(GL_LIGHT0 + mLightID); +} + +// Destroy the shadow map associated with this light +inline void Light::destroyShadowMap() { + mShadowMap.destroy(); + mFBOShadowMap.destroy(); + mDepthShader.destroy(); +} + +// Call this method before rendering the scene for the shadow map computation +inline void Light::startRenderingShadowMap() { + assert(mShadowMap.getID()); +} + +// Call this method after having rendered the scene for the shadow map computation +inline void Light::stopRenderingShadowMap() { + assert(mShadowMap.getID()); +} + +} + +#endif diff --git a/examples/opengl-framework/src/Mesh.cpp b/examples/opengl-framework/src/Mesh.cpp new file mode 100644 index 00000000..de15254c --- /dev/null +++ b/examples/opengl-framework/src/Mesh.cpp @@ -0,0 +1,177 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Mesh.h" + +// Namespaces +using namespace openglframework; +using namespace std; + +// Constructor +Mesh::Mesh() { + +} + +// Destructor +Mesh::~Mesh() { + +} + +// Destroy the mesh +void Mesh::destroy() { + + mVertices.clear(); + mNormals.clear(); + mTangents.clear(); + mIndices.clear(); + mColors.clear(); + mUVs.clear(); + mTextures.clear(); +} + +// Compute the normals of the mesh +void Mesh::calculateNormals() { + + mNormals = vector(getNbVertices(), Vector3(0, 0, 0)); + + // For each triangular face + for (uint i=0; i 0); + mNormals[i] = mNormals[i].normalize(); + } +} + +// Compute the tangents of the mesh +void Mesh::calculateTangents() { + + mTangents = std::vector(getNbVertices(), Vector3(0, 0, 0)); + + // For each face + for (uint i=0; i::const_iterator it(mVertices.begin()); + + // For each vertex of the mesh + for (; it != mVertices.end(); ++it) { + + if( (*it).x < min.x ) min.x = (*it).x; + else if ( (*it).x > max.x ) max.x = (*it).x; + + if( (*it).y < min.y ) min.y = (*it).y; + else if ( (*it).y > max.y ) max.y = (*it).y; + + if( (*it).z < min.z ) min.z = (*it).z; + else if ( (*it).z > max.z ) max.z = (*it).z; + } + } + else { + std::cerr << "Error : Impossible to calculate the bounding box of the mesh because there" << + "are no vertices !" << std::endl; + assert(false); + } +} + +// Scale of vertices of the mesh using a given factor +void Mesh::scaleVertices(float factor) { + + // For each vertex + for (uint i=0; i +#include +#include +#include "definitions.h" +#include "maths/Vector2.h" +#include "maths/Vector3.h" +#include "maths/Color.h" +#include "Texture2D.h" +#include "Object3D.h" + +namespace openglframework { + +// Class Mesh +// This class represents a 3D triangular mesh +// object that can be loaded from an OBJ file for instance. +class Mesh : public Object3D { + + private: + + // -------------------- Attributes -------------------- // + + // A triplet of vertex indices for each triangle + std::vector > mIndices; + + // Vertices coordinates (local space) + std::vector mVertices; + + // Normals coordinates + std::vector mNormals; + + // Tangents coordinates + std::vector mTangents; + + // Color for each vertex + std::vector mColors; + + // UV texture coordinates + std::vector mUVs; + + // Textures of the mesh (one for each part of the mesh) + std::map mTextures; + + public: + + // -------------------- Methods -------------------- // + + // Constructor + Mesh(); + + // Destructor + virtual ~Mesh(); + + // Destroy the mesh + void destroy(); + + // Compute the normals of the mesh + void calculateNormals(); + + // Compute the tangents of the mesh + void calculateTangents(); + + // Calculate the bounding box of the mesh + void calculateBoundingBox(Vector3& min, Vector3& max) const; + + // Scale of vertices of the mesh using a given factor + void scaleVertices(float factor); + + // Return the number of triangles + uint getNbFaces(uint part = 0) const; + + // Return the number of vertices + uint getNbVertices() const; + + // Return the number of parts in the mesh + uint getNbParts() const; + + // Return a reference to the vertices + const std::vector& getVertices() const; + + // Set the vertices of the mesh + void setVertices(std::vector& vertices); + + // Return a reference to the normals + const std::vector& getNormals() const; + + // set the normals of the mesh + void setNormals(std::vector& normals); + + // Return a reference to the UVs + const std::vector& getUVs() const; + + // Set the UV texture coordinates of the mesh + void setUVs(std::vector& uvs); + + // Return a reference to the vertex indices + const std::vector& getIndices(uint part = 0) const; + + // Set the vertices indices of the mesh + void setIndices(std::vector >& indices); + + // Return the coordinates of a given vertex + const Vector3& getVertex(uint i) const; + + // Set the coordinates of a given vertex + void setVertex(uint i, const Vector3& vertex); + + // Return the coordinates of a given normal + const Vector3& getNormal(uint i) const; + + // Set the coordinates of a given normal + void setNormal(uint i, const Vector3& normal); + + // Return the color of a given vertex + const Color& getColor(uint i) const; + + // Set the color of a given vertex + void setColor(uint i, const Color& color); + + // Set a color to all the vertices + void setColorToAllVertices(const Color& color); + + // Return the UV of a given vertex + const Vector2& getUV(uint i) const; + + // Set the UV of a given vertex + void setUV(uint i, const Vector2& uv); + + // Return the vertex index of the ith (i=0,1,2) vertex of a given face + uint getVertexIndexInFace(uint faceIndex, uint i, uint part = 0) const; + + // Return true if the mesh has normals + bool hasNormals() const; + + // Return true if the mesh has tangents + bool hasTangents() const; + + // Return true if the mesh has vertex colors + bool hasColors() const; + + // Return true if the mesh has UV texture coordinates + bool hasUVTextureCoordinates() const; + + // Return true if the mesh has a texture for a given part of the mesh and if it + // also have texture coordinates + bool hasTextureForPart(uint part = 0) const; + + // Return true if the mesh has a texture (and texture coordinates) for at least one + // part of the mesh + bool hasTexture() const; + + // Return a pointer to the vertices data + void* getVerticesPointer(); + + // Return a pointer to the normals data + void* getNormalsPointer(); + + // Return a pointer to the colors data + void* getColorsPointer(); + + // Return a pointer to the tangents data + void* getTangentsPointer(); + + // Return a pointer to the UV texture coordinates data + void* getUVTextureCoordinatesPointer(); + + // Return a pointer to the vertex indicies data + void* getIndicesPointer(uint part = 0); + + // Return a reference to a texture of the mesh + Texture2D &getTexture(uint part = 0); + + // Set a texture to a part of the mesh + void setTexture(Texture2D &texture, uint part = 0); +}; + +// Return the number of triangles +inline uint Mesh::getNbFaces(uint part) const { + return mIndices[part].size() / 3; +} + +// Return the number of vertices +inline uint Mesh::getNbVertices() const { + return mVertices.size(); +} + +// Return the number of parts in the mesh +inline uint Mesh::getNbParts() const { + return mIndices.size(); +} + +// Return a reference to the vertices +inline const std::vector& Mesh::getVertices() const { + return mVertices; +} + +// Set the vertices of the mesh +inline void Mesh::setVertices(std::vector& vertices) { + mVertices = vertices; +} + +// Return a reference to the normals +inline const std::vector& Mesh::getNormals() const { + return mNormals; +} + +// set the normals of the mesh +inline void Mesh::setNormals(std::vector& normals) { + mNormals = normals; +} + +// Return a reference to the UVs +inline const std::vector& Mesh::getUVs() const { + return mUVs; +} + +// Set the UV texture coordinates of the mesh +inline void Mesh::setUVs(std::vector& uvs) { + mUVs = uvs; +} + +// Return a reference to the vertex indices +inline const std::vector& Mesh::getIndices(uint part) const { + return mIndices[part]; +} + +// Set the vertices indices of the mesh +inline void Mesh::setIndices(std::vector >& indices) { + mIndices = indices; +} + +// Return the coordinates of a given vertex +inline const Vector3& Mesh::getVertex(uint i) const { + assert(i < getNbVertices()); + return mVertices[i]; +} + +// Set the coordinates of a given vertex +inline void Mesh::setVertex(uint i, const Vector3& vertex) { + assert(i < getNbVertices()); + mVertices[i] = vertex; +} + +// Return the coordinates of a given normal +inline const Vector3& Mesh::getNormal(uint i) const { + assert(i < getNbVertices()); + return mNormals[i]; +} + +// Set the coordinates of a given normal +inline void Mesh::setNormal(uint i, const Vector3& normal) { + assert(i < getNbVertices()); + mNormals[i] = normal; +} + +// Return the color of a given vertex +inline const Color& Mesh::getColor(uint i) const { + assert(i < getNbVertices()); + return mColors[i]; +} + +// Set the color of a given vertex +inline void Mesh::setColor(uint i, const Color& color) { + + // If the color array does not have the same size as + // the vertices array + if (mColors.size() != mVertices.size()) { + + // Create the color array with the same size + mColors = std::vector(mVertices.size()); + } + + mColors[i] = color; +} + +// Set a color to all the vertices +inline void Mesh::setColorToAllVertices(const Color& color) { + + // If the color array does not have the same size as + // the vertices array + if (mColors.size() != mVertices.size()) { + + // Create the color array with the same size + mColors = std::vector(mVertices.size()); + } + + for (size_t v=0; v 0); +} + +// Return a pointer to the vertices data +inline void* Mesh::getVerticesPointer() { + return &(mVertices[0]); +} + +// Return a pointer to the normals data +inline void* Mesh::getNormalsPointer() { + return &(mNormals[0]); +} + +// Return a pointer to the colors data +inline void* Mesh::getColorsPointer() { + return &(mColors[0]); +} + +// Return a pointer to the tangents data +inline void* Mesh::getTangentsPointer() { + return &(mTangents[0]); +} + +// Return a pointer to the UV texture coordinates data +inline void* Mesh::getUVTextureCoordinatesPointer() { + return &(mUVs[0]); +} + +// Return a pointer to the vertex indicies data +inline void* Mesh::getIndicesPointer(uint part) { + return &(mIndices[part])[0]; +} + +// Return a reference to a texture of the mesh +inline Texture2D& Mesh::getTexture(uint part) { + return mTextures[part]; +} + +// Set a texture to a part of the mesh +inline void Mesh::setTexture(Texture2D& texture, uint part) { + mTextures[part] = texture; +} + +} + +#endif diff --git a/examples/opengl-framework/src/MeshReaderWriter.cpp b/examples/opengl-framework/src/MeshReaderWriter.cpp new file mode 100644 index 00000000..f720240f --- /dev/null +++ b/examples/opengl-framework/src/MeshReaderWriter.cpp @@ -0,0 +1,397 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "MeshReaderWriter.h" +#include +#include +#include +#include +#include +#include + +using namespace openglframework; +using namespace std; + +// Constructor +MeshReaderWriter::MeshReaderWriter() { + +} + +// Load a mesh from a file and returns true if the mesh has been sucessfully loaded +void MeshReaderWriter::loadMeshFromFile(const std::string& filename, + Mesh& meshToCreate) + throw(std::invalid_argument, std::runtime_error) { + + // Get the extension of the file + uint startPosExtension = filename.find_last_of("."); + string extension = filename.substr(startPosExtension+1); + + // Load the file using the correct method + if (extension == "obj") { + loadOBJFile(filename, meshToCreate); + } + else { + + // Display an error message and throw an exception + string errorMessage("Error : the MeshReaderWriter class cannot load a file with the extension ."); + errorMessage += extension; + std::cerr << errorMessage << std::endl; + throw std::invalid_argument(errorMessage.c_str()); + } +} + +// Write a mesh to a file +void MeshReaderWriter::writeMeshToFile(const std::string& filename, + const Mesh& meshToWrite) + throw(std::invalid_argument, std::runtime_error) { + + // Get the extension of the file + uint startPosExtension = filename.find_last_of("."); + string extension = filename.substr(startPosExtension+1); + + // Load the file using the correct method + if (extension == "obj") { + writeOBJFile(filename, meshToWrite); + } + else { + + // Display an error message and throw an exception + string errorMessage("Error : the MeshReaderWriter class cannot store a mesh file with the extension ."); + errorMessage += extension; + std::cerr << errorMessage << std::endl; + throw std::invalid_argument(errorMessage.c_str()); + } +} + +// Load an OBJ file with a triangular or quad mesh +void MeshReaderWriter::loadOBJFile(const string &filename, Mesh& meshToCreate) { + + // Open the file + std::ifstream meshFile(filename.c_str()); + + // If we cannot open the file + if(!meshFile.is_open()) { + + // Throw an exception and display an error message + string errorMessage("Error : Cannot open the file " + filename); + std::cerr << errorMessage << std::endl; + throw runtime_error(errorMessage); + } + + std::string buffer; + string line, tmp; + int id1, id2, id3, id4; + int nId1, nId2, nId3, nId4; + int tId1, tId2, tId3, tId4; + float v1, v2, v3; + size_t found1, found2; + std::vector isQuad; + std::vector vertices; + std::vector normals; + std::vector uvs; + std::vector verticesIndices; + std::vector normalsIndices; + std::vector uvsIndices; + + // ---------- Collect the data from the file ---------- // + + // For each line of the file + while(std::getline(meshFile, buffer)) { + + std::istringstream lineStream(buffer); + std::string word; + lineStream >> word; + std::transform(word.begin(), word.end(), word.begin(), ::tolower); + if(word == "usemtl") { // Material definition + + // Loading of MTL file is not implemented + + } + else if(word == "v") { // Vertex position + sscanf(buffer.c_str(), "%*s %f %f %f", &v1, &v2, &v3); + vertices.push_back(Vector3(v1, v2, v3)); + } + else if(word == "vt") { // Vertex texture coordinate + sscanf(buffer.c_str(), "%*s %f %f", &v1, &v2); + uvs.push_back(Vector2(v1,v2)); + } + else if(word == "vn") { // Vertex normal + sscanf(buffer.c_str(), "%*s %f %f %f", &v1, &v2, &v3); + normals.push_back(Vector3(v1 ,v2, v3)); + } + else if (word == "f") { // Face + line = buffer; + found1 = (int)line.find("/"); + bool isFaceQuad = false; + int foundNext = (int)line.substr(found1+1).find("/"); + + // If the face definition is of the form "f v1 v2 v3 v4" + if(found1 == string::npos) { + int nbVertices = sscanf(buffer.c_str(), "%*s %d %d %d %d", &id1, &id2, &id3, &id4); + if (nbVertices == 4) isFaceQuad = true; + } + // If the face definition is of the form "f v1// v2// v3// v4//" + else if (foundNext == 0) { + int nbVertices = sscanf(buffer.c_str(), "%*s %d// %d// %d// %d//", &id1, &id2, &id3, &id4); + if (nbVertices == 4) isFaceQuad = true; + } + else { // If the face definition contains vertices and texture coordinates + + //get the part of the string until the second index + tmp = line.substr(found1+1); + found2 = (int)tmp.find(" "); + tmp = tmp.substr(0,found2); + found2 = (int)tmp.find("/"); + + // If the face definition is of the form "f vert1/textcoord1 vert2/textcoord2 ..." + if(found2 == string::npos) { + int n = sscanf(buffer.c_str(), "%*s %d/%d %d/%d %d/%d %d/%d", &id1, &tId1, &id2, &tId2, &id3, &tId3, &id4, &tId4); + if (n == 8) isFaceQuad = true; + uvsIndices.push_back(tId1-1); + uvsIndices.push_back(tId2-1); + uvsIndices.push_back(tId3-1); + if (isFaceQuad) uvsIndices.push_back(tId4-1); + } + else { + tmp = line.substr(found1+1); + found2 = (int)tmp.find("/"); + + // If the face definition is of the form "f vert1/normal1 vert2/normal2 ..." + if(found2 == 0) { + int n = sscanf(buffer.c_str(), "%*s %d//%d %d//%d %d//%d %d//%d", &id1, &nId1, &id2, &nId2, &id3, &nId3, &id4, &nId4); + if (n == 8) isFaceQuad = true; + } + // If the face definition is of the form "f vert1/textcoord1/normal1 ..." + else { + int n = sscanf(buffer.c_str(), "%*s %d/%d/%d %d/%d/%d %d/%d/%d %d/%d/%d", &id1, &tId1, &nId1, &id2, &tId2, &nId2, &id3, &tId3, &nId3, &id4, &tId4, &nId4); + if (n == 12) isFaceQuad = true; + uvsIndices.push_back(tId1-1); + uvsIndices.push_back(tId2-1); + uvsIndices.push_back(tId3-1); + if (isFaceQuad) uvsIndices.push_back(tId4-1); + } + normalsIndices.push_back(nId1-1); + normalsIndices.push_back(nId2-1); + normalsIndices.push_back(nId3-1); + if (isFaceQuad) normalsIndices.push_back(nId4-1); + } + } + verticesIndices.push_back(id1-1); + verticesIndices.push_back(id2-1); + verticesIndices.push_back(id3-1); + if (isFaceQuad) verticesIndices.push_back((id4-1)); + isQuad.push_back(isFaceQuad); + } + } + + assert(!verticesIndices.empty()); + assert(normalsIndices.empty() || normalsIndices.size() == verticesIndices.size()); + assert(uvsIndices.empty() || uvsIndices.size() == verticesIndices.size()); + meshFile.close(); + + // ---------- Merge the data that we have collected from the file ---------- // + + // Destroy the current mesh + meshToCreate.destroy(); + + // Mesh data + vector > meshIndices; + vector meshNormals; + if (!normals.empty()) meshNormals = vector(vertices.size(), Vector3(0, 0, 0)); + vector meshUVs; + if (!uvs.empty()) meshUVs = vector(vertices.size(), Vector2(0, 0)); + + // We cannot load mesh with several parts for the moment + uint meshPart = 0; + + // Fill in the vertex indices + // We also triangulate each quad face + meshIndices.push_back(std::vector()); + for(size_t i = 0, j = 0; i < verticesIndices.size(); j++) { + + // Get the current vertex IDs + uint i1 = verticesIndices[i]; + uint i2 = verticesIndices[i+1]; + uint i3 = verticesIndices[i+2]; + + // Add the vertex normal + if (!normalsIndices.empty() && !normals.empty()) { + meshNormals[i1] = normals[normalsIndices[i]]; + meshNormals[i2] = normals[normalsIndices[i+1]]; + meshNormals[i3] = normals[normalsIndices[i+2]]; + } + + // Add the vertex UV texture coordinates + if (!uvsIndices.empty() && !uvs.empty()) { + meshUVs[i1] = uvs[uvsIndices[i]]; + meshUVs[i2] = uvs[uvsIndices[i+1]]; + meshUVs[i3] = uvs[uvsIndices[i+2]]; + } + + // If the current vertex not in a quad (it is part of a triangle) + if (!isQuad[j]) { + + // Add the vertex indices + meshIndices[meshPart].push_back(i1); + meshIndices[meshPart].push_back(i2); + meshIndices[meshPart].push_back(i3); + + i+=3; + } + else { // If the current vertex is in a quad + + Vector3 v1 = vertices[i1]; + Vector3 v2 = vertices[i2]; + Vector3 v3 = vertices[i3]; + uint i4 = verticesIndices[i+3]; + Vector3 v4 = vertices[i4]; + + Vector3 v13 = v3-v1; + Vector3 v12 = v2-v1; + Vector3 v14 = v4-v1; + + float a1 = v13.dot(v12); + float a2 = v13.dot(v14); + if((a1 >= 0 && a2 <= 0) || (a1 <= 0 && a2 >= 0)) { + meshIndices[meshPart].push_back(i1); + meshIndices[meshPart].push_back(i2); + meshIndices[meshPart].push_back(i3); + meshIndices[meshPart].push_back(i1); + meshIndices[meshPart].push_back(i3); + meshIndices[meshPart].push_back(i4); + } + else { + meshIndices[meshPart].push_back(i1); + meshIndices[meshPart].push_back(i2); + meshIndices[meshPart].push_back(i4); + meshIndices[meshPart].push_back(i2); + meshIndices[meshPart].push_back(i3); + meshIndices[meshPart].push_back(i4); + } + + // Add the vertex normal + if (!normalsIndices.empty() && !normals.empty()) { + meshNormals[i4] = normals[normalsIndices[i]]; + } + + // Add the vertex UV texture coordinates + if (!uvsIndices.empty() && !uvs.empty()) { + meshUVs[i4] = uvs[uvsIndices[i]]; + } + + i+=4; + } + } + + assert(meshNormals.empty() || meshNormals.size() == vertices.size()); + assert(meshUVs.empty() || meshUVs.size() == vertices.size()); + + // Set the data to the mesh + meshToCreate.setIndices(meshIndices); + meshToCreate.setVertices(vertices); + meshToCreate.setNormals(meshNormals); + meshToCreate.setUVs(meshUVs); +} + +// Store a mesh into a OBJ file +void MeshReaderWriter::writeOBJFile(const std::string& filename, const Mesh& meshToWrite) { + std::ofstream file(filename.c_str()); + + // Geth the mesh data + const std::vector& vertices = meshToWrite.getVertices(); + const std::vector& normals = meshToWrite.getNormals(); + const std::vector& uvs = meshToWrite.getUVs(); + + // If we can open the file + if (file.is_open()) { + + assert(meshToWrite.getNbVertices() == vertices.size()); + + // Write the vertices + for (uint v=0; v& indices = meshToWrite.getIndices(p); + + // For each index of the part + for (uint i=0; i +#include +#include "Mesh.h" + +namespace openglframework { + +// Class MeshReaderWriter +// This class is used to read or write any mesh file in order to +// create the corresponding Mesh object. Currently, this class +// is able to read meshes of the current formats : .obj +class MeshReaderWriter { + + private : + + // -------------------- Methods -------------------- // + + // Constructor (private because we do not want instances of this class) + MeshReaderWriter(); + + // Load an OBJ file with a triangular or quad mesh + static void loadOBJFile(const std::string& filename, Mesh& meshToCreate); + + // Store a mesh into a OBJ file + static void writeOBJFile(const std::string& filename, const Mesh &meshToWrite); + + public : + + // -------------------- Methods -------------------- // + + // Read a mesh from a file + static void loadMeshFromFile(const std::string& filename, + Mesh& meshToCreate) + throw(std::invalid_argument, std::runtime_error); + + // Write a mesh to a file + static void writeMeshToFile(const std::string& filename, + const Mesh& meshToWrite) + throw(std::invalid_argument, std::runtime_error); +}; + +// Class VertexMergingData +// This class is used in the method to read a mesh +class VertexMergingData { + + public: + VertexMergingData() : indexPosition(0), indexNormal(0), indexUV(0) {} + unsigned int indexPosition; + unsigned int indexNormal; + unsigned int indexUV; +}; + +// Class VertexMergingDataComparison +// This class is used in the method to read a mesh +class VertexMergingDataComparison { + + public: + bool operator()(const VertexMergingData& x, const VertexMergingData& y) const { + if(x.indexPosition < y.indexPosition) + return true; + if(x.indexPosition == y.indexPosition && x.indexNormal < y.indexNormal) + return true; + if(x.indexPosition == y.indexPosition && x.indexNormal == + y.indexNormal && x.indexUV < y.indexUV) + return true; + return false; + } +}; + +} + +#endif diff --git a/examples/opengl-framework/src/Object3D.cpp b/examples/opengl-framework/src/Object3D.cpp new file mode 100644 index 00000000..6beb8d28 --- /dev/null +++ b/examples/opengl-framework/src/Object3D.cpp @@ -0,0 +1,41 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Object3D.h" + +// Namespaces +using namespace openglframework; + +// Constructor +Object3D::Object3D() { + // Set the transformation matrix to the identity + setToIdentity(); +} + +// Destructor +Object3D::~Object3D() { + +} diff --git a/examples/opengl-framework/src/Object3D.h b/examples/opengl-framework/src/Object3D.h new file mode 100644 index 00000000..24c28fa1 --- /dev/null +++ b/examples/opengl-framework/src/Object3D.h @@ -0,0 +1,149 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 OBJECT3D_H +#define OBJECT3D_H + +// Libraries +#include "maths/Vector3.h" +#include "maths/Matrix4.h" + +namespace openglframework { + +// Class Object3D +// This class represent a generic 3D object on the scene. +class Object3D { + + protected: + + // -------------------- Attributes -------------------- // + + // Transformation matrix that convert local-space + // coordinates to world-space coordinates + Matrix4 mTransformMatrix; + + public: + + // -------------------- Methods -------------------- // + + // Constructor + Object3D(); + + // Destructor + virtual ~Object3D(); + + // Return the transform matrix + const Matrix4& getTransformMatrix() const; + + // Set the transform matrix + void setTransformMatrix(const Matrix4& matrix); + + // Set to the identity transform + void setToIdentity(); + + // Return the origin of object in world-space + Vector3 getOrigin() const; + + // Translate the object in world-space + void translateWorld(const Vector3& v); + + // Translate the object in local-space + void translateLocal(const Vector3& v); + + // Rotate the object in world-space + void rotateWorld(const Vector3& axis, float angle); + + // Rotate the object in local-space + void rotateLocal(const Vector3& axis, float angle); + + // Rotate around a world-space point + void rotateAroundWorldPoint(const Vector3& axis, float angle, const Vector3& point); + + // Rotate around a local-space point + void rotateAroundLocalPoint(const Vector3& axis, float angle, const Vector3& worldPoint); +}; + +// Return the transform matrix +inline const Matrix4& Object3D::getTransformMatrix() const { + return mTransformMatrix; +} + +// Set the transform matrix +inline void Object3D::setTransformMatrix(const Matrix4& matrix) { + mTransformMatrix = matrix; +} + +// Set to the identity transform +inline void Object3D::setToIdentity() { + mTransformMatrix.setToIdentity(); +} + + // Return the origin of object in world-space +inline Vector3 Object3D::getOrigin() const { + return mTransformMatrix * Vector3(0.0, 0.0, 0.0); +} + +// Translate the object in world-space +inline void Object3D::translateWorld(const Vector3& v) { + mTransformMatrix = Matrix4::translationMatrix(v) * mTransformMatrix; +} + +// Translate the object in local-space +inline void Object3D::translateLocal(const Vector3& v) { + mTransformMatrix = mTransformMatrix * Matrix4::translationMatrix(v); +} + +// Rotate the object in world-space +inline void Object3D::rotateWorld(const Vector3& axis, float angle) { + mTransformMatrix = Matrix4::rotationMatrix(axis, angle) * mTransformMatrix; +} + +// Rotate the object in local-space +inline void Object3D::rotateLocal(const Vector3& axis, float angle) { + mTransformMatrix = mTransformMatrix * Matrix4::rotationMatrix(axis, angle); +} + +// Rotate the object around a world-space point +inline void Object3D::rotateAroundWorldPoint(const Vector3& axis, float angle, + const Vector3& worldPoint) { + mTransformMatrix = Matrix4::translationMatrix(worldPoint) * Matrix4::rotationMatrix(axis, angle) + * Matrix4::translationMatrix(-worldPoint) * mTransformMatrix; +} + +// Rotate the object around a local-space point +inline void Object3D::rotateAroundLocalPoint(const Vector3& axis, float angle, + const Vector3& worldPoint) { + + // Convert the world point into the local coordinate system + Vector3 localPoint = mTransformMatrix.getInverse() * worldPoint; + + mTransformMatrix = mTransformMatrix * Matrix4::translationMatrix(localPoint) + * Matrix4::rotationMatrix(axis, angle) + * Matrix4::translationMatrix(-localPoint); +} + +} + +#endif diff --git a/examples/opengl-framework/src/Shader.cpp b/examples/opengl-framework/src/Shader.cpp new file mode 100644 index 00000000..1ec1b4a9 --- /dev/null +++ b/examples/opengl-framework/src/Shader.cpp @@ -0,0 +1,207 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Shader.h" +#include +#include +#include + +// Namespaces +using namespace openglframework; +using namespace std; + +// Constructor +Shader::Shader() : mProgramObjectID(0) { + +} + +// Constructor with arguments +Shader::Shader(const std::string vertexShaderFilename, const std::string fragmentShaderFilename) + : mProgramObjectID(0) { + + // Create the shader + create(vertexShaderFilename, fragmentShaderFilename); +} + +// Destructor +Shader::~Shader() { + +} + +// Create the shader +bool Shader::create(const std::string vertexShaderFilename, + const std::string fragmentShaderFilename) { + + // Set the shader filenames + mFilenameVertexShader = vertexShaderFilename; + mFilenameFragmentShader = fragmentShaderFilename; + + // Check that the needed OpenGL extensions are available + bool isExtensionOK = checkOpenGLExtensions(); + if (!isExtensionOK) { + cerr << "Error : Impossible to use GLSL vertex or fragment shaders on this platform" << endl; + assert(false); + return false; + } + + // Delete the current shader + destroy(); + + assert(!vertexShaderFilename.empty() && !fragmentShaderFilename.empty()); + + // ------------------- Load the vertex shader ------------------- // + GLuint vertexShaderID; + std::ifstream fileVertexShader; + fileVertexShader.open(vertexShaderFilename.c_str(), std::ios::binary); + + if (fileVertexShader.is_open()) { + + // Get the size of the file + fileVertexShader.seekg(0, std::ios::end); + uint fileSize = (uint) (fileVertexShader.tellg()); + assert(fileSize != 0); + + // Read the file + fileVertexShader.seekg(std::ios::beg); + char* bufferVertexShader = new char[fileSize + 1]; + fileVertexShader.read(bufferVertexShader, fileSize); + fileVertexShader.close(); + bufferVertexShader[fileSize] = '\0'; + + // Create the OpenGL vertex shader and compile it + vertexShaderID = glCreateShader(GL_VERTEX_SHADER); + assert(vertexShaderID != 0); + glShaderSource(vertexShaderID, 1, (const char **) (&bufferVertexShader), NULL); + glCompileShader(vertexShaderID); + delete[] bufferVertexShader; + + // Get the compilation information + int compiled; + glGetShaderiv(vertexShaderID, GL_COMPILE_STATUS, &compiled); + + // If the compilation failed + if (compiled == 0) { + + // Get the log of the compilation + int lengthLog; + glGetShaderiv(vertexShaderID, GL_INFO_LOG_LENGTH, &lengthLog); + char* str = new char[lengthLog]; + glGetShaderInfoLog(vertexShaderID, lengthLog, NULL, str); + + // Display the log of the compilation + std::cerr << "Vertex Shader Error : " << str << std::endl; + delete[] str; + assert(false); + return false; + } + } + else { + std::cerr << "Error : Impossible to open the vertex shader file " << + vertexShaderFilename << std::endl; + assert(false); + return false; + } + + // ------------------- Load the fragment shader ------------------- // + GLuint fragmentShaderID; + std::ifstream fileFragmentShader; + fileFragmentShader.open(fragmentShaderFilename.c_str(), std::ios::binary); + assert(fileFragmentShader.is_open()); + + if (fileFragmentShader.is_open()) { + + // Get the size of the file + fileFragmentShader.seekg(0, std::ios::end); + uint fileSize = (uint) (fileFragmentShader.tellg()); + assert(fileSize != 0); + + // Read the file + fileFragmentShader.seekg(std::ios::beg); + char* bufferFragmentShader = new char[fileSize + 1]; + fileFragmentShader.read(bufferFragmentShader, fileSize); + fileFragmentShader.close(); + bufferFragmentShader[fileSize] = '\0'; + + // Create the OpenGL fragment shader and compile it + fragmentShaderID = glCreateShader(GL_FRAGMENT_SHADER); + assert(fragmentShaderID != 0); + glShaderSource(fragmentShaderID, 1, (const char **) (&bufferFragmentShader), NULL); + glCompileShader(fragmentShaderID); + delete[] bufferFragmentShader; + + // Get the compilation information + int compiled; + glGetShaderiv(fragmentShaderID, GL_COMPILE_STATUS, &compiled); + + // If the compilation failed + if (compiled == 0) { + + // Get the log of the compilation + int lengthLog; + glGetShaderiv(fragmentShaderID, GL_INFO_LOG_LENGTH, &lengthLog); + char* str = new char[lengthLog]; + glGetShaderInfoLog(fragmentShaderID, lengthLog, NULL, str); + + // Display the log of the compilation + std::cerr << "Fragment Shader Error : " << str << std::endl; + delete[] str; + assert(false); + return false; + } + } + else { + std::cerr << "Error : Impossible to open the fragment shader file " << + fragmentShaderFilename << std::endl; + assert(false); + return false; + } + + // Create the shader program and attach the shaders + mProgramObjectID = glCreateProgram(); + assert(mProgramObjectID); + glAttachShader(mProgramObjectID, vertexShaderID); + glAttachShader(mProgramObjectID, fragmentShaderID); + glDeleteShader(vertexShaderID); + glDeleteShader(fragmentShaderID); + + // Try to link the program + glLinkProgram(mProgramObjectID); + int linked; + glGetProgramiv(mProgramObjectID, GL_LINK_STATUS, &linked); + if (!linked) { + int logLength; + glGetProgramiv(mProgramObjectID, GL_INFO_LOG_LENGTH, &logLength); + char* strLog = new char[logLength]; + glGetProgramInfoLog(mProgramObjectID, logLength, NULL, strLog); + cerr << "Linker Error in vertex shader " << vertexShaderFilename << + " or in fragment shader " << fragmentShaderFilename << " : " << strLog << endl; + delete[] strLog; + destroy(); + assert(false); + } + + return true; +} diff --git a/examples/opengl-framework/src/Shader.h b/examples/opengl-framework/src/Shader.h new file mode 100644 index 00000000..0b353138 --- /dev/null +++ b/examples/opengl-framework/src/Shader.h @@ -0,0 +1,243 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 SHADER_H +#define SHADER_H + +// Libraries +#include "definitions.h" +#include "maths/Matrix4.h" +#include "maths/Vector2.h" +#include "maths/Vector3.h" +#include "maths/Vector4.h" +#include +#include +#include + +namespace openglframework { + +// Class Shader +class Shader { + + private : + + // -------------------- Attributes -------------------- // + + // Shader object program ID + GLuint mProgramObjectID; + + // Filenames of the vertex and fragment shaders + std::string mFilenameVertexShader, mFilenameFragmentShader; + + public : + + // -------------------- Methods -------------------- // + + // Constructor + Shader(); + + // Constructor with arguments + Shader(const std::string vertexShaderFilename, const std::string fragmentShaderFilename); + + // Destructor + ~Shader(); + + // Create the shader + bool create(const std::string vertexShaderFilename, + const std::string fragmentShaderFilename); + + // Clear the shader + void destroy(); + + // Bind the shader + void bind() const; + + // Unbind the shader + void unbind() const; + + // Return the location of a uniform variable inside a shader program + int getUniformLocation(const std::string& variableName) const; + + // Set a float uniform value to this shader (be careful if the uniform is not + // used in the shader, the compiler will remove it, then when you will try + // to set it, an assert will occur) + void setFloatUniform(const std::string& variableName, float value) const; + + // Set an int uniform value to this shader (be careful if the uniform is not + // used in the shader, the compiler will remove it, then when you will try + // to set it, an assert will occur) + void setIntUniform(const std::string& variableName, int value) const; + + // Set a vector 2 uniform value to this shader (be careful if the uniform is not + // used in the shader, the compiler will remove it, then when you will try + // to set it, an assert will occur) + void setVector2Uniform(const std::string& variableName, const Vector2& v) const; + + // Set a vector 3 uniform value to this shader (be careful if the uniform is not + // used in the shader, the compiler will remove it, then when you will try + // to set it, an assert will occur) + void setVector3Uniform(const std::string& variableName, const Vector3& v) const; + + // Set a vector 4 uniform value to this shader (be careful if the uniform is not + // used in the shader, the compiler will remove it, then when you will try + // to set it, an assert will occur) + void setVector4Uniform(const std::string& variableName, const Vector4 &v) const; + + // Set a 3x3 matrix uniform value to this shader (be careful if the uniform is not + // used in the shader, the compiler will remove it, then when you will try + // to set it, an assert will occur) + void setMatrix3x3Uniform(const std::string& variableName, const float* matrix, + bool transpose = false) const; + + // Set a 4x4 matrix uniform value to this shader (be careful if the uniform is not + // used in the shader, the compiler will remove it, then when you will try + // to set it, an assert will occur) + void setMatrix4x4Uniform(const std::string& variableName, const float* matrix, + bool transpose = false) const; + + // Set a 4x4 matrix uniform value to this shader (be careful if the uniform is not + // used in the shader, the compiler will remove it, then when you will try + // to set it, an assert will occur) + void setMatrix4x4Uniform(const std::string& variableName, const Matrix4& matrix) const; + + // Return true if the needed OpenGL extensions are available + static bool checkOpenGLExtensions(); +}; + +// Bind the shader +inline void Shader::bind() const { + assert(mProgramObjectID != 0); + glUseProgram(mProgramObjectID); +} + +// Unbind the shader +inline void Shader::unbind() const { + assert(mProgramObjectID != 0); + glUseProgram(0); +} + +// Return the location of a uniform variable inside a shader program +inline int Shader::getUniformLocation(const std::string& variableName) const { + assert(mProgramObjectID != 0); + int location = glGetUniformLocation(mProgramObjectID, variableName.c_str()); + if (location == -1) { + std::cerr << "Error in vertex shader " << mFilenameVertexShader << " or in fragment shader" + << mFilenameFragmentShader << " : No Uniform variable : " << variableName + << std::endl; + } + assert(location != -1); + return location; +} + +// Clear the shader +inline void Shader::destroy() { + if (mProgramObjectID != 0) { + glDeleteShader(mProgramObjectID); + mProgramObjectID = 0; + } +} + +// Set a float uniform value to this shader (be careful if the uniform is not +// used in the shader, the compiler will remove it, then when you will try +// to set it, an assert will occur) +inline void Shader::setFloatUniform(const std::string& variableName, float value) const { + assert(mProgramObjectID != 0); + glUniform1f(getUniformLocation(variableName), value); +} + +// Set an int uniform value to this shader (be careful if the uniform is not +// used in the shader, the compiler will remove it, then when you will try +// to set it, an assert will occur) +inline void Shader::setIntUniform(const std::string& variableName, int value) const { + assert(mProgramObjectID != 0); + glUniform1i(getUniformLocation(variableName), value); +} + +// Set a vector 2 uniform value to this shader (be careful if the uniform is not +// used in the shader, the compiler will remove it, then when you will try +// to set it, an assert will occur) +inline void Shader::setVector2Uniform(const std::string& variableName, const Vector2& v) const { + assert(mProgramObjectID != 0); + glUniform2f(getUniformLocation(variableName), v.x, v.y); +} + +// Set a vector 3 uniform value to this shader (be careful if the uniform is not +// used in the shader, the compiler will remove it, then when you will try +// to set it, an assert will occur) +inline void Shader::setVector3Uniform(const std::string& variableName, const Vector3 &v) const { + assert(mProgramObjectID != 0); + glUniform3f(getUniformLocation(variableName), v.x, v.y, v.z); +} + +// Set a vector 4 uniform value to this shader (be careful if the uniform is not +// used in the shader, the compiler will remove it, then when you will try +// to set it, an assert will occur) +inline void Shader::setVector4Uniform(const std::string& variableName, const Vector4& v) const { + assert(mProgramObjectID != 0); + glUniform4f(getUniformLocation(variableName), v.x, v.y, v.z, v.w); +} + +// Set a 4x4 matrix uniform value to this shader (be careful if the uniform is not +// used in the shader, the compiler will remove it, then when you will try +// to set it, an assert will occur) +inline void Shader::setMatrix3x3Uniform(const std::string& variableName, const float* matrix, + bool transpose) const { + assert(mProgramObjectID != 0); + glUniformMatrix3fv(getUniformLocation(variableName), 1, transpose, matrix); +} + +// Set a 4x4 matrix uniform value to this shader (be careful if the uniform is not +// used in the shader, the compiler will remove it, then when you will try +// to set it, an assert will occur) +inline void Shader::setMatrix4x4Uniform(const std::string& variableName, const float* matrix, + bool transpose) const { + assert(mProgramObjectID != 0); + glUniformMatrix4fv(getUniformLocation(variableName), 1, transpose, matrix); +} + +// Set a 4x4 matrix uniform value to this shader (be careful if the uniform is not +// used in the shader, the compiler will remove it, then when you will try +// to set it, an assert will occur) +inline void Shader::setMatrix4x4Uniform(const std::string& variableName, const Matrix4& matrix) const { + assert(mProgramObjectID != 0); + GLfloat mat[16]; + for (int i=0; i<4; i++) { + for (int j=0; j<4; j++) { + mat[i*4 + j] = matrix.m[i][j]; + } + } + glUniformMatrix4fv(getUniformLocation(variableName), 1, true, mat); +} + +// Return true if the needed OpenGL extensions are available for shaders +inline bool Shader::checkOpenGLExtensions() { + + // Check that GLSL vertex and fragment shaders are available on the platform + return (GLEW_VERSION_2_0 || (GLEW_ARB_vertex_shader && GLEW_ARB_fragment_shader)); +} + +} + +#endif diff --git a/examples/opengl-framework/src/Texture2D.cpp b/examples/opengl-framework/src/Texture2D.cpp new file mode 100644 index 00000000..18deb915 --- /dev/null +++ b/examples/opengl-framework/src/Texture2D.cpp @@ -0,0 +1,84 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Texture2D.h" +#include +#include +#include + +// Namespaces +using namespace openglframework; + +// Constructor +Texture2D::Texture2D() : mID(0), mLayer(0), mWidth(0), mHeight(0) { + +} + +// Constructor +Texture2D::Texture2D(uint width, uint height, uint internalFormat, uint format, uint type) + : mID(0), mLayer(0), mWidth(0), mHeight(0){ + + // Create the texture + create(width, height, internalFormat, format, type); +} + +// Destructor +Texture2D::~Texture2D() { + +} + +// Create the texture +void Texture2D::create(uint width, uint height, uint internalFormat, uint format, uint type, + void* data) throw(std::invalid_argument) { + + // Destroy the current texture + destroy(); + + mWidth = width; + mHeight = height; + + // Create the OpenGL texture + glGenTextures(1, &mID); + assert(mID != 0); + glBindTexture(GL_TEXTURE_2D, mID); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); + glTexImage2D(GL_TEXTURE_2D, 0, internalFormat, mWidth, mHeight, 0, format, type, data); + glBindTexture(GL_TEXTURE_2D, 0); +} + +// Destroy the texture +void Texture2D::destroy() { + if (mID != 0) { + glDeleteTextures(1, &mID); + mID = 0; + mLayer = 0; + mWidth = 0; + mHeight = 0; + } +} diff --git a/examples/opengl-framework/src/Texture2D.h b/examples/opengl-framework/src/Texture2D.h new file mode 100644 index 00000000..bb455c16 --- /dev/null +++ b/examples/opengl-framework/src/Texture2D.h @@ -0,0 +1,142 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 TEXTURE2D_H +#define TEXTURE2D_H + +// Libraries +#include +#include +#include "definitions.h" +#include + +namespace openglframework { + +// Class Texture2D +// This class represents a 2D texture +class Texture2D { + + private: + + // -------------------- Attributes -------------------- // + + // OpenGL texture ID + GLuint mID; + + // Layer of the texture + GLuint mLayer; + + // Width + uint mWidth; + + // Height + uint mHeight; + + public: + + // -------------------- Methods -------------------- // + + // Constructor + Texture2D(); + + // Constructor + Texture2D(uint width, uint height, uint internalFormat, uint format, uint type); + + // Destructor + ~Texture2D(); + + // Create the texture + void create(uint width, uint height, uint internalFormat, uint format, uint type, + void* data = NULL) throw(std::invalid_argument); + + // Destroy the texture + void destroy(); + + // Bind the texture + void bind() const; + + // Unbind the texture + void unbind() const; + + // Get the OpenGL texture ID + uint getID() const; + + // Get the layer of the texture + uint getLayer() const; + + // Set the layer of the texture + void setLayer(uint layer); + + // Get the width + uint getWidth() const; + + // Get the height + uint getHeight() const; +}; + +// Bind the texture +inline void Texture2D::bind() const { + assert(mID != 0); + glEnable(GL_TEXTURE_2D); + glActiveTexture(GL_TEXTURE0 + mLayer); + glBindTexture(GL_TEXTURE_2D, mID); +} + +// Unbind the texture +inline void Texture2D::unbind() const { + assert(mID != 0); + glActiveTexture(GL_TEXTURE0 + mLayer); + glBindTexture(GL_TEXTURE_2D, 0); + glDisable(GL_TEXTURE_2D); +} + +// Get the OpenGL texture ID +inline uint Texture2D::getID() const { + return mID; +} + +// Get the layer of the texture +inline uint Texture2D::getLayer() const { + return mLayer; +} + +// Set the layer of the texture +inline void Texture2D::setLayer(uint layer) { + mLayer = layer; +} + +// Get the width +inline uint Texture2D::getWidth() const { + return mWidth; +} + +// Get the height +inline uint Texture2D::getHeight() const { + return mHeight; +} + +} + +#endif diff --git a/examples/opengl-framework/src/TextureReaderWriter.cpp b/examples/opengl-framework/src/TextureReaderWriter.cpp new file mode 100644 index 00000000..7bfc7e6a --- /dev/null +++ b/examples/opengl-framework/src/TextureReaderWriter.cpp @@ -0,0 +1,329 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Librairies +#include "TextureReaderWriter.h" +#include +#include +#include + +using namespace openglframework; +using namespace std; + +// Constants +const uint TextureReaderWriter::JPEG_COMPRESSION_QUALITY = 98; + +// TGA file Header +#pragma pack(push, 1) +typedef struct { + unsigned char identsize; // size of ID field that follows 18 byte header (0 usually) + unsigned char colourmaptype; // type of colour map 0=none, 1=has palette + unsigned char imagetype; // type of image 0=none,1=indexed,2=rgb,3=grey,+8=rle packed + + short colourmapstart; // first colour map entry in palette + short colourmaplength; // number of colours in palette + unsigned char colourmapbits; // number of bits per palette entry 15,16,24,32 + + short xstart; // image x origin + short ystart; // image y origin + short width; // image width in pixels + short height; // image height in pixels + unsigned char bits; // image bits per pixel 8,16,24,32 + unsigned char descriptor; // image descriptor bits (vh flip bits) + + // pixel data follows header +} TGA_HEADER; +#pragma pack(pop) + +// Load a texture from a file +void TextureReaderWriter::loadTextureFromFile(const std::string& filename, + Texture2D& textureToCreate) + throw(runtime_error, invalid_argument){ + + // Get the extension of the file + uint startPosExtension = filename.find_last_of("."); + string extension = filename.substr(startPosExtension+1); + + // Load the file using the correct method + if (extension == "tga") { + readTGAPicture(filename, textureToCreate); + } + else if (extension == "jpg" || extension == "jpeg"){ + readJPEGPicture(filename, textureToCreate); + } + else { + + // Display an error message and throw an exception + string errorMessage("Error : the TextureLoader class cannot load a file with the extension ."); + errorMessage += extension; + std::cerr << errorMessage << std::endl; + throw std::invalid_argument(errorMessage.c_str()); + } +} + +// Write a texture to a file +void TextureReaderWriter::writeTextureToFile(const std::string& filename, + const Texture2D& texture) + throw(runtime_error, invalid_argument){ + + // Get the extension of the file + uint startPosExtension = filename.find_last_of("."); + string extension = filename.substr(startPosExtension+1); + + // Write the file using the correct method + if (extension == "tga") { + writeTGAPicture(filename, texture); + } + else if (extension == "jpg" || extension == "jpeg"){ + writeJPEGPicture(filename, texture); + } + else { + + // Display an error message and throw an exception + string errorMessage("Error : the TextureReaderWriter class cannot write a file with the extension ."); + errorMessage += extension; + std::cerr << errorMessage << std::endl; + throw std::invalid_argument(errorMessage.c_str()); + } +} + +// Load a TGA picture +void TextureReaderWriter::readTGAPicture(const std::string &filename, + Texture2D& textureToCreate) throw(runtime_error) { + + // Open the file + std::ifstream stream(filename.c_str(), std::ios::binary); + + // If we cannot open the file + if(!stream.is_open()) { + + // Throw an exception and display an error message + string errorMessage("Error : Cannot open the file " + filename); + std::cerr << errorMessage << std::endl; + throw std::runtime_error(errorMessage); + } + + TGA_HEADER header; + stream.read((char *)(&header), sizeof(TGA_HEADER)); + assert(header.width <= 4096 && header.width <= 4096 && + header.imagetype == 2 && header.bits == 24); + + // Read the file + uint width = header.width; + uint height = header.width; + uint sizeImg = width*height; + char* data = new char[sizeImg*3]; + assert(data); + stream.read(data, sizeImg*3); + for(uint i = 0; i < sizeImg; i++) { + unsigned pos = i*3; + unsigned char red = data[pos]; + data[pos] = data[pos + 2]; + data[pos + 2] = red; + } + + // Close the stream + stream.close(); + + // Create the OpenGL texture using the picture data from the file + textureToCreate.create(width, height, GL_RGB, GL_RGB, GL_UNSIGNED_BYTE, data); + + // Free the data memory + delete[] data; +} + + +// Write a TGA picture +void TextureReaderWriter::writeTGAPicture(const std::string& filename, + const Texture2D& texture) throw(runtime_error) { + assert(texture.getID() != 0); + + // Bind the corresponding texture + glBindTexture(GL_TEXTURE_2D, texture.getID()); + + uint sizeImg = texture.getWidth() * texture.getHeight(); + + // Get the bytes form the OpenGL texture + char* data = new char[sizeImg * 3]; + glGetTexImage(GL_TEXTURE_2D, 0, GL_RGB, GL_UNSIGNED_BYTE, data); + + // Open the file + std::ofstream stream(filename.c_str(), std::ios::binary); + + // If the file cannot be opened + if(!stream.is_open()) { + + // Throw an exception and display an error message + string errorMessage("Error : Cannot create/access the file " + filename); + std::cerr << errorMessage << std::endl; + delete[] data; + throw std::runtime_error(errorMessage); + } + + // Fill in the TGA header + TGA_HEADER header; + header.identsize = 0; // size of ID field that follows 18 byte header (0 usually) + header.colourmaptype = 0; // type of colour map 0=none, 1=has palette + header.imagetype = 2; // type of image 0=none,1=indexed,2=rgb,3=grey,+8=rle packed + header.colourmapstart = 0; // first colour map entry in palette + header.colourmaplength = 0; // number of colours in palette + header.colourmapbits=0; // number of bits per palette entry 15,16,24,32 + header.xstart = 0; // image x origin + header.ystart = 0; // image y origin + header.width = (short)texture.getWidth(); // image width in pixels + header.height = (short)texture.getHeight(); // image height in pixels + header.bits = 24; // image bits per pixel 8,16,24,32 + header.descriptor = 0; // image descriptor bits (vh flip bits) + + // Write the header to the file + stream.write((char*)(&header), sizeof(TGA_HEADER)); + + // Write the bytes to the file + for(uint i = 0; i < sizeImg; i++) { + unsigned pos = i*3; + unsigned char red = data[pos]; + data[pos] = data[pos + 2]; + data[pos + 2] = red; + } + stream.write(data, sizeImg*3); + + // Close the file + stream.close(); + + // Delete the data + delete[] data; + + // Unbind the corresponding texture + glBindTexture(GL_TEXTURE_2D, 0); +} + +// Read a JPEG picture +void TextureReaderWriter::readJPEGPicture(const std::string& filename, + Texture2D& textureToCreate) throw(std::runtime_error) { + + struct jpeg_decompress_struct info; + struct jpeg_error_mgr error; + + info.err = jpeg_std_error(&error); + jpeg_create_decompress(&info); + + // Open the file + FILE* file = fopen(filename.c_str(), "rb"); + + // If we cannot open the file + if (!file) { + + // Throw an exception and display an error message + string errorMessage("Error : Cannot open the file " + filename); + std::cerr << errorMessage << std::endl; + throw std::runtime_error(errorMessage); + } + + jpeg_stdio_src(&info, file); + jpeg_read_header(&info, true); + jpeg_start_decompress(&info); + + unsigned long x = info.output_width; + unsigned long y = info.output_height; + int channels = info.num_components; + assert(channels == 3); + + unsigned long size = x * y * 3; + + BYTE* data = new BYTE[size]; + + BYTE* p1 = data; + BYTE** p2 = &p1; + int numlines = 0; + + while(info.output_scanline < info.output_height) { + numlines = jpeg_read_scanlines(&info, p2, 1); + *p2 += numlines * 3 * info.output_width; + } + + jpeg_finish_decompress(&info); //finish decompressing this file + + // Close the file + fclose(file); + + // Create the OpenGL texture + textureToCreate.create(x, y, GL_RGB, GL_RGB, GL_UNSIGNED_BYTE, data); + + // Free allocated memory + delete[] data; +} + +// Write a JPEG picture +void TextureReaderWriter::writeJPEGPicture(const std::string& filename, + const Texture2D& texture) throw(std::runtime_error) { + + struct jpeg_compress_struct info; + struct jpeg_error_mgr error; + + info.err = jpeg_std_error(&error); + jpeg_create_compress(&info); + + // Open the file + FILE* file = fopen(filename.c_str(), "wb"); + + if (!file) { + // Throw an exception and display an error message + string errorMessage("Error : Cannot write JPEG picture into the file " + filename); + std::cerr << errorMessage << std::endl; + throw std::runtime_error(errorMessage); + } + + // Get the bytes form the OpenGL texture + char* data = new char[texture.getWidth() * texture.getHeight() * 3]; + glGetTexImage(GL_TEXTURE_2D, 0, GL_RGB, GL_UNSIGNED_BYTE, data); + + jpeg_stdio_dest(&info, file); + + info.image_width = texture.getWidth(); + info.image_height = texture.getHeight(); + info.input_components = 3; + info.in_color_space = JCS_RGB; + jpeg_set_defaults(&info); + jpeg_set_quality(&info, JPEG_COMPRESSION_QUALITY, true); + + jpeg_start_compress(&info, true); + + // Write the data into the file + JSAMPROW rowPointer; + int rowStride = texture.getWidth() * 3; + while (info.next_scanline < info.image_height) { + rowPointer = (JSAMPROW) &data[info.next_scanline * rowStride]; + jpeg_write_scanlines(&info, &rowPointer, 1); + } + + jpeg_finish_compress(&info); + jpeg_destroy_compress(&info); + + // Close the file + fclose(file); + + // Free allocated memory + delete[] data; +} diff --git a/examples/opengl-framework/src/TextureReaderWriter.h b/examples/opengl-framework/src/TextureReaderWriter.h new file mode 100644 index 00000000..72a50baf --- /dev/null +++ b/examples/opengl-framework/src/TextureReaderWriter.h @@ -0,0 +1,87 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 TEXTURE_READER_WRITER_H +#define TEXTURE_READER_WRITER_H + +// Libraries +#include "Texture2D.h" +#include +#include +#include + +namespace openglframework { + +// Class TextureReaderWriter +// This class is used to load or write a texture image in different picture format. +// It currently allows to read and write the following formats : .tga +class TextureReaderWriter { + + private : + + // ------------------- Constants ------------------- // + + // JPEG quality for compression (in percent) + static const uint JPEG_COMPRESSION_QUALITY; + + // -------------------- Methods -------------------- // + + // Constructor (private because we do not want instances of this class) + TextureReaderWriter(); + + // Read a TGA picture + static void readTGAPicture(const std::string& filename, + Texture2D& textureToCreate) throw(std::runtime_error); + + // Write a TGA picture + static void writeTGAPicture(const std::string& filename, + const Texture2D& texture) throw(std::runtime_error); + + // Read a JPEG picture + static void readJPEGPicture(const std::string& filename, + Texture2D& textureToCreate) throw(std::runtime_error); + + // Write a JPEG picture + static void writeJPEGPicture(const std::string& filename, + const Texture2D& texture) throw(std::runtime_error); + + public : + + // -------------------- Methods -------------------- // + + // Load a texture from a file + static void loadTextureFromFile(const std::string& filename, + Texture2D& textureToCreate) + throw(std::runtime_error, std::invalid_argument); + + // Write a texture to a file + static void writeTextureToFile(const std::string& filename, + const Texture2D& texture) + throw(std::runtime_error, std::invalid_argument); +}; + +} + +#endif diff --git a/examples/opengl-framework/src/VertexBufferObject.cpp b/examples/opengl-framework/src/VertexBufferObject.cpp new file mode 100644 index 00000000..8afd1daf --- /dev/null +++ b/examples/opengl-framework/src/VertexBufferObject.cpp @@ -0,0 +1,84 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "VertexBufferObject.h" + +using namespace openglframework; + +// Constructor +VertexBufferObject::VertexBufferObject(GLenum targetData) + : mVertexBufferID(0), mTargetData(targetData) { + +} + +// Destructor +VertexBufferObject::~VertexBufferObject() { + +} + +// Create the vertex buffer object +bool VertexBufferObject::create() { + + // Destroy the current VBO + destroy(); + + // Check that the needed OpenGL extensions are available + bool isExtensionOK = checkOpenGLExtensions(); + if (!isExtensionOK) { + std::cerr << "Error : Impossible to use Vertex Buffer Object on this platform" << std::endl; + assert(false); + return false; + } + + // Generate a new VBO + glGenBuffers(1, &mVertexBufferID); + assert(mVertexBufferID != 0); + + return true; +} + +// Copy data into the VBO +void VertexBufferObject::copyDataIntoVBO(GLsizei size, const void* data, GLenum usage) { + + // Bind the VBO + bind(); + + // Copy the data into the VBO + glBufferData(mTargetData, size, data, usage); + + // Unbind the VBO + unbind(); +} + +// Destroy the VBO +void VertexBufferObject::destroy() { + + // Delete the vertex buffer object + if (mVertexBufferID) { + glDeleteFramebuffers(1, &mVertexBufferID); + mVertexBufferID = 0; + } +} diff --git a/examples/opengl-framework/src/VertexBufferObject.h b/examples/opengl-framework/src/VertexBufferObject.h new file mode 100644 index 00000000..d697c659 --- /dev/null +++ b/examples/opengl-framework/src/VertexBufferObject.h @@ -0,0 +1,106 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 VERTEX_BUFFER_OBJECT_H +#define VERTEX_BUFFER_OBJECT_H + +// Libraries +#include +#include +#include + +namespace openglframework { + + +// Class VertexBufferObject +class VertexBufferObject { + + private : + + // -------------------- Attributes -------------------- // + + /// ID of the Vertex Buffer Object + GLuint mVertexBufferID; + + /// Target data. This variable must be GL_ARRAY_BUFFER if the VBO contains vertex + /// data (vertex coordinates, texture coordinates, normals, colors) or must be + /// GL_ELEMENT_ARRAY_BUFFER if the VBO contains index data (index array). + GLenum mTargetData; + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + VertexBufferObject(GLenum targetData); + + /// Destructor + ~VertexBufferObject(); + + /// Create the vertex buffer object + bool create(); + + /// Copy data into the VBO + void copyDataIntoVBO(GLsizei size, const void* data, GLenum usage); + + /// Bind the VBO + void bind() const; + + /// Unbind the VBO + void unbind() const; + + /// Return true if the needed OpenGL extensions are available for VBO + static bool checkOpenGLExtensions(); + + /// Destroy the VBO + void destroy(); +}; + +// Bind the VBO +inline void VertexBufferObject::bind() const { + assert(mVertexBufferID != 0); + + // Bind the VBO + glBindBuffer(mTargetData, mVertexBufferID); +} + +// Unbind the VBO +inline void VertexBufferObject::unbind() const { + assert(mVertexBufferID != 0); + + // Unbind the VBO + glBindBuffer(mTargetData, 0); +} + +// Return true if the needed OpenGL extensions are available for VBO +inline bool VertexBufferObject::checkOpenGLExtensions() { + + // Check that OpenGL version is at least 1.5 or there the vertex buffer object extension exists + return (GLEW_VERSION_1_5 || GL_ARB_vertex_buffer_object); +} + +} + +#endif diff --git a/examples/opengl-framework/src/definitions.h b/examples/opengl-framework/src/definitions.h new file mode 100644 index 00000000..5cacbdf4 --- /dev/null +++ b/examples/opengl-framework/src/definitions.h @@ -0,0 +1,39 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 DEFINITIONS_H +#define DEFINITIONS_H + +namespace openglframework { + +// ------------------- Type definitions ------------------- // +typedef unsigned int uint; + +// ------------------- Constants ------------------- // +const float PI = 3.141592654f; + +} + +#endif diff --git a/examples/opengl-framework/src/maths/Color.h b/examples/opengl-framework/src/maths/Color.h new file mode 100644 index 00000000..e8ea2c40 --- /dev/null +++ b/examples/opengl-framework/src/maths/Color.h @@ -0,0 +1,76 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 COLOR_H +#define COLOR_H + +namespace openglframework { + +// Structure Color +// This structure represents a RGBA color. +struct Color { + + public: + + // -------------------- Attributes -------------------- // + + // RGBA color components + float r, g, b, a; + + // -------------------- Methods -------------------- // + + // Constructor + Color() : r(1), g(1), b(1), a(1) {} + + // Constructor + Color(float r, float g, float b, float a) : r(r), g(g), b(b), a(a) {} + + // Constructor + Color(const Color& color) : r(color.r), g(color.g), b(color.b), a(color.a) {} + + // Destructor + ~Color() {} + + // Return the black color + static Color black() { return Color(0.0f, 0.0f, 0.0f, 1.0f);} + + // Return the white color + static Color white() { return Color(1.0f, 1.0f, 1.0f, 1.0f);} + + // = operator + Color& operator=(const Color& color) { + if (&color != this) { + r = color.r; + g = color.g; + b = color.b; + a = color.a; + } + return *this; + } +}; + +} + +#endif diff --git a/examples/opengl-framework/src/maths/Matrix3.h b/examples/opengl-framework/src/maths/Matrix3.h new file mode 100644 index 00000000..5460f616 --- /dev/null +++ b/examples/opengl-framework/src/maths/Matrix3.h @@ -0,0 +1,279 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 MATRIX3_H +#define MATRIX3_H + +// Libraries +#include +#include +#include "Vector3.h" + +namespace openglframework { + +// Class Matrix4 +// This class represents a 4x4 matrix +class Matrix3 { + + private : + + // -------------------- Attributes -------------------- // + + // Elements of the matrix + float m[3][3]; + + public : + + // Constructor + Matrix3() { + setToNull(); + } + + // Constructor + Matrix3(float a1, float a2, + float a3, float b1, float b2, float b3, + float c1, float c2, float c3) { + setAllValues(a1, a2, a3, b1, b2, b3, c1, c2, c3); + } + + // Constructor + Matrix3(float n[3][3]) { + m[0][0]=n[0][0]; m[0][1]=n[0][1]; m[0][2]=n[0][2]; + m[1][0]=n[1][0]; m[1][1]=n[1][1]; m[1][2]=n[1][2]; + m[2][0]=n[2][0]; m[2][1]=n[2][1]; m[2][2]=n[2][2]; + } + + // Constructor + Matrix3(const Vector3& a1, const Vector3& a2, const Vector3& a3) { + m[0][0] = a1.x; m[0][1] = a2.x; m[0][2] = a3.x; + m[1][0] = a1.y; m[1][1] = a2.y; m[1][2] = a3.y; + m[2][0] = a1.z; m[2][1] = a2.z; m[2][2] = a3.z; + } + + // Constructor + Matrix3(const Matrix3& matrix) { + setAllValues(matrix.m[0][0], matrix.m[0][1], matrix.m[0][2], + matrix.m[1][0], matrix.m[1][1], matrix.m[1][2], + matrix.m[2][0], matrix.m[2][1], matrix.m[2][2]); + } + + // Method to get a value in the matrix + float getValue(int i, int j) const { + assert(i>=0 && i<3 && j>=0 && j<3); + return m[i][j]; + } + + // Method to set a value in the matrix + void setValue(int i, int j, float value) { + assert(i>=0 && i<3 && j>=0 && j<3); + m[i][j] = value; + } + + // Method to set all the values in the matrix + void setAllValues(float a1, float a2, float a3, float b1, float b2, float b3, + float c1, float c2, float c3) { + m[0][0] = a1; m[0][1] = a2; m[0][2] = a3; + m[1][0] = b1; m[1][1] = b2; m[1][2] = b3; + m[2][0] = c1; m[2][1] = c2; m[2][2] = c3; + } + + // Return a column + Vector3 getColumn(int i) const { + assert(i>= 0 && i<3); + return Vector3(m[0][i], m[1][i], m[2][i]); + } + + // Return the transpose matrix + Matrix3 getTranspose() const { + // Return the transpose matrix + return Matrix3(m[0][0], m[1][0], m[2][0], + m[0][1], m[1][1], m[2][1], + m[0][2], m[1][2], m[2][2]); + } + + // Return the determinant of the matrix + float getDeterminant() const { + // Compute and return the determinant of the matrix + return (m[0][0]*(m[1][1]*m[2][2]-m[2][1]*m[1][2]) - m[0][1]*(m[1][0]*m[2][2]-m[2][0]*m[1][2]) + + m[0][2]*(m[1][0]*m[2][1]-m[2][0]*m[1][1])); + } + + // Return the trace of the matrix + float getTrace() const { + // Compute and return the trace + return (m[0][0] + m[1][1] + m[2][2]); + } + + void setToNull() { + m[0][0] = 0.0; m[0][1] = 0.0; m[0][2] = 0.0; + m[1][0] = 0.0; m[1][1] = 0.0; m[1][2] = 0.0; + m[2][0] = 0.0; m[2][1] = 0.0; m[2][2] = 0.0; + } + + bool isNull() const { + Matrix3 zero; + return *this == zero; + } + + // Set the matrix to the identity matrix + void setToIdentity() { + m[0][0] = 1.0; m[0][1] = 0.0; m[0][2] = 0.0; + m[1][0] = 0.0; m[1][1] = 1.0; m[1][2] = 0.0; + m[2][0] = 0.0; m[2][1] = 0.0; m[2][2] = 1.0; + } + + bool isIdentity() const { + Matrix3 I; + I.setToIdentity(); + return ( *this == I ); + } + + // Return the inverse matrix + Matrix3 getInverse() const { + + // Compute the determinant of the matrix + float determinant = getDeterminant(); + + // Check if the determinant is equal to zero + assert(determinant > std::numeric_limits::epsilon()); + + float invDeterminant = 1.0f / determinant; + Matrix3 tempMatrix((m[1][1]*m[2][2]-m[2][1]*m[1][2]), -(m[0][1]*m[2][2]-m[2][1]*m[0][2]), (m[0][1]*m[1][2]-m[0][2]*m[1][1]), + -(m[1][0]*m[2][2]-m[2][0]*m[1][2]), (m[0][0]*m[2][2]-m[2][0]*m[0][2]), -(m[0][0]*m[1][2]-m[1][0]*m[0][2]), + (m[1][0]*m[2][1]-m[2][0]*m[1][1]), -(m[0][0]*m[2][1]-m[2][0]*m[0][1]), (m[0][0]*m[1][1]-m[0][1]*m[1][0])); + + // Return the inverse matrix + return (tempMatrix * invDeterminant); + } + + // Display the matrix + void print() const { + for (int i=0; i<3; i++) { + for (int j=0; j<3; j++) { + std::cout << m[i][j] << " "; + } + std::cout << std::endl; + } + } + + // Overloaded operator = + Matrix3& operator=(const Matrix3& matrix) { + if (&matrix != this) { + setAllValues(matrix.m[0][0], matrix.m[0][1], matrix.m[0][2], + matrix.m[1][0], matrix.m[1][1], matrix.m[1][2], + matrix.m[2][0], matrix.m[2][1], matrix.m[2][2]); + } + return *this; + } + + + // Overloaded operator for addition + Matrix3 operator+(const Matrix3& matrix2) { + return Matrix3(m[0][0] + matrix2.m[0][0], m[0][1] + matrix2.m[0][1], m[0][2] + matrix2.m[0][2], + m[1][0] + matrix2.m[1][0], m[1][1] + matrix2.m[1][1], m[1][2] + matrix2.m[1][2], + m[2][0] + matrix2.m[2][0], m[2][1] + matrix2.m[2][1], m[2][2] + matrix2.m[2][2]); + } + + // Overloaded operator for substraction + Matrix3 operator-(const Matrix3& matrix2) { + return Matrix3(m[0][0] - matrix2.m[0][0], m[0][1] - matrix2.m[0][1], m[0][2] - matrix2.m[0][2], + m[1][0] - matrix2.m[1][0], m[1][1] - matrix2.m[1][1], m[1][2] - matrix2.m[1][2], + m[2][0] - matrix2.m[2][0], m[2][1] - matrix2.m[2][1], m[2][2] - matrix2.m[2][2]); + } + + // Overloaded operator for the negative of the matrix + Matrix3 operator-() { + return Matrix3(-m[0][0], -m[0][1], -m[0][2], + -m[1][0], -m[1][1], -m[1][2], + -m[2][0], -m[2][1], -m[2][2]); + } + + // Overloaded operator for multiplication with a number + Matrix3 operator*(float nb) { + return Matrix3(m[0][0] * nb, m[0][1] * nb, m[0][2] * nb, + m[1][0] * nb, m[1][1] * nb, m[1][2] * nb, + m[2][0] * nb, m[2][1] * nb, m[2][2] * nb); + } + + // Overloaded operator for matrix multiplication + Matrix3 operator*(const Matrix3& matrix2) { + return Matrix3(m[0][0]*matrix2.m[0][0] + m[0][1]*matrix2.m[1][0] + m[0][2]*matrix2.m[2][0], + m[0][0]*matrix2.m[0][1] + m[0][1]*matrix2.m[1][1] + m[0][2]*matrix2.m[2][1], + m[0][0]*matrix2.m[0][2] + m[0][1]*matrix2.m[1][2] + m[0][2]*matrix2.m[2][2], + m[1][0]*matrix2.m[0][0] + m[1][1]*matrix2.m[1][0] + m[1][2]*matrix2.m[2][0], + m[1][0]*matrix2.m[0][1] + m[1][1]*matrix2.m[1][1] + m[1][2]*matrix2.m[2][1], + m[1][0]*matrix2.m[0][2] + m[1][1]*matrix2.m[1][2] + m[1][2]*matrix2.m[2][2], + m[2][0]*matrix2.m[0][0] + m[2][1]*matrix2.m[1][0] + m[2][2]*matrix2.m[2][0], + m[2][0]*matrix2.m[0][1] + m[2][1]*matrix2.m[1][1] + m[2][2]*matrix2.m[2][1], + m[2][0]*matrix2.m[0][2] + m[2][1]*matrix2.m[1][2] + m[2][2]*matrix2.m[2][2]); + } + + // Overloaded operator for multiplication with a vector + Vector3 operator*(const Vector3& vector) { + return Vector3(m[0][0]*vector.x + m[0][1]*vector.y + m[0][2]*vector.z, + m[1][0]*vector.x + m[1][1]*vector.y + m[1][2]*vector.z, + m[2][0]*vector.x + m[2][1]*vector.y + m[2][2]*vector.z); + } + + // Overloaded operator for equality condition + bool operator==(const Matrix3& matrix) const { + return (m[0][0] == matrix.m[0][0] && m[0][1] == matrix.m[0][1] && m[0][2] == matrix.m[0][2] && + m[1][0] == matrix.m[1][0] && m[1][1] == matrix.m[1][1] && m[1][2] == matrix.m[1][2] && + m[2][0] == matrix.m[2][0] && m[2][1] == matrix.m[2][1] && m[2][2] == matrix.m[2][2]); + } + + // Overloaded operator for the is different condition + bool operator!= (const Matrix3& matrix) const { + return !(*this == matrix); + } + + // Overloaded operator for addition with assignment + Matrix3& operator+=(const Matrix3& matrix) { + m[0][0] += matrix.m[0][0]; m[0][1] += matrix.m[0][1]; m[0][2] += matrix.m[0][2]; + m[1][0] += matrix.m[1][0]; m[1][1] += matrix.m[1][1]; m[1][2] += matrix.m[1][2]; + m[2][0] += matrix.m[2][0]; m[2][1] += matrix.m[2][1]; m[2][2] += matrix.m[2][2]; + return *this; + } + + // Overloaded operator for substraction with assignment + Matrix3& operator-=(const Matrix3& matrix) { + m[0][0] -= matrix.m[0][0]; m[0][1] -= matrix.m[0][1]; m[0][2] -= matrix.m[0][2]; + m[1][0] -= matrix.m[1][0]; m[1][1] -= matrix.m[1][1]; m[1][2] -= matrix.m[1][2]; + m[2][0] -= matrix.m[2][0]; m[2][1] -= matrix.m[2][1]; m[2][2] -= matrix.m[2][2]; + return *this; + } + + // Overloaded operator for multiplication with a number with assignment + Matrix3& operator*=(float nb) { + m[0][0] *= nb; m[0][1] *= nb; m[0][2] *= nb; + m[1][0] *= nb; m[1][1] *= nb; m[1][2] *= nb; + m[2][0] *= nb; m[2][1] *= nb; m[2][2] *= nb; + return *this; + } +}; + +} + +#endif diff --git a/examples/opengl-framework/src/maths/Matrix4.h b/examples/opengl-framework/src/maths/Matrix4.h new file mode 100644 index 00000000..009f6354 --- /dev/null +++ b/examples/opengl-framework/src/maths/Matrix4.h @@ -0,0 +1,419 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 MATRIX4_H +#define MATRIX4_H + +// Libraries +#include +#include +#include +#include "Vector3.h" +#include "Vector4.h" + +namespace openglframework { + +// Class Matrix4 +// This class represents a 4x4 matrix +class Matrix4 { + + public: + + // -------------------- Attributes -------------------- // + + // Elements of the matrix + float m[4][4]; + + // -------------------- Methods -------------------- // + + // Constructor + Matrix4(float m_00=0, float m_01=0, float m_02=0, float m_03=0, + float m_10=0, float m_11=0, float m_12=0, float m_13=0, + float m_20=0, float m_21=0, float m_22=0, float m_23=0, + float m_30=0, float m_31=0, float m_32=0, float m_33=0) { + m[0][0] = m_00; m[0][1] = m_01; m[0][2] = m_02; m[0][3] = m_03; + m[1][0] = m_10; m[1][1] = m_11; m[1][2] = m_12; m[1][3] = m_13; + m[2][0] = m_20; m[2][1] = m_21; m[2][2] = m_22; m[2][3] = m_23; + m[3][0] = m_30; m[3][1] = m_31; m[3][2] = m_32; m[3][3] = m_33; + } + + // Constructor + Matrix4(float n[4][4]) { + m[0][0]=n[0][0]; m[0][1]=n[0][1]; m[0][2]=n[0][2]; m[0][3]=n[0][3]; + m[1][0]=n[1][0]; m[1][1]=n[1][1]; m[1][2]=n[1][2]; m[1][3]=n[1][3]; + m[2][0]=n[2][0]; m[2][1]=n[2][1]; m[2][2]=n[2][2]; m[2][3]=n[2][3]; + m[3][0]=n[3][0]; m[3][1]=n[3][1]; m[3][2]=n[3][2]; m[3][3]=n[3][3]; + } + + // Constructor + Matrix4(const Vector3& a1, const Vector3& a2, const Vector3& a3) { + m[0][0] = a1.x; m[0][1] = a2.x; m[0][2] = a3.x; m[0][3] = 0.f; + m[1][0] = a1.y; m[1][1] = a2.y; m[1][2] = a3.y; m[1][3] = 0.f; + m[2][0] = a1.z; m[2][1] = a2.z; m[2][2] = a3.z; m[2][3] = 0.f; + m[3][0] = 0.f; m[3][1] = 0.f; m[3][2] = 0.f; m[3][3] = 1.f; + } + + // Constructor + Matrix4(const Vector4& a1, const Vector4& a2, const Vector4& a3) { + m[0][0] = a1.x; m[0][1] = a2.x; m[0][2] = a3.x; m[0][3] = 0.f; + m[1][0] = a1.y; m[1][1] = a2.y; m[1][2] = a3.y; m[1][3] = 0.f; + m[2][0] = a1.z; m[2][1] = a2.z; m[2][2] = a3.z; m[2][3] = 0.f; + m[3][0] = a1.w; m[3][1] = a2.w; m[3][2] = a3.w; m[3][3] = 1.f; + } + + // Constructor + Matrix4(const Matrix4& matrix) { + + setAllValues(matrix.m[0][0], matrix.m[0][1], matrix.m[0][2], matrix.m[0][3], + matrix.m[1][0], matrix.m[1][1], matrix.m[1][2], matrix.m[1][3], + matrix.m[2][0], matrix.m[2][1], matrix.m[2][2], matrix.m[2][3], + matrix.m[3][0], matrix.m[3][1], matrix.m[3][2], matrix.m[3][3]); + } + + // + operator + Matrix4 operator+(const Matrix4 &n) const { + return Matrix4(m[0][0]+n.m[0][0], m[0][1]+n.m[0][1], m[0][2]+n.m[0][2], m[0][3]+n.m[0][3], + m[1][0]+n.m[1][0], m[1][1]+n.m[1][1], m[1][2]+n.m[1][2], m[1][3]+n.m[1][3], + m[2][0]+n.m[2][0], m[2][1]+n.m[2][1], m[2][2]+n.m[2][2], m[2][3]+n.m[2][3], + m[3][0]+n.m[3][0], m[3][1]+n.m[3][1], m[3][2]+n.m[3][2], m[3][3]+n.m[3][3]); + } + + // += operator + Matrix4& operator+=(const Matrix4 &n) { + m[0][0]+=n.m[0][0]; m[0][1]+=n.m[0][1]; m[0][2]+=n.m[0][2]; m[0][3]+=n.m[0][3]; + m[1][0]+=n.m[1][0]; m[1][1]+=n.m[1][1]; m[1][2]+=n.m[1][2]; m[1][3]+=n.m[1][3]; + m[2][0]+=n.m[2][0]; m[2][1]+=n.m[2][1]; m[2][2]+=n.m[2][2]; m[2][3]+=n.m[2][3]; + m[3][0]+=n.m[3][0]; m[3][1]+=n.m[3][1]; m[3][2]+=n.m[3][2]; m[3][3]+=n.m[3][3]; + return *this; + } + + // - operator + Matrix4 operator-(const Matrix4 &n) const { + return Matrix4(m[0][0]-n.m[0][0], m[0][1]-n.m[0][1], m[0][2]-n.m[0][2], m[0][3]-n.m[0][3], + m[1][0]-n.m[1][0], m[1][1]-n.m[1][1], m[1][2]-n.m[1][2], m[1][3]-n.m[1][3], + m[2][0]-n.m[2][0], m[2][1]-n.m[2][1], m[2][2]-n.m[2][2], m[2][3]-n.m[2][3], + m[3][0]-n.m[3][0], m[3][1]-n.m[3][1], m[3][2]-n.m[3][2], m[3][3]-n.m[3][3]); + } + + // -= operator + Matrix4& operator-=(const Matrix4 &n) { + m[0][0]-=n.m[0][0]; m[0][1]-=n.m[0][1]; m[0][2]-=n.m[0][2]; m[0][3]-=n.m[0][3]; + m[1][0]-=n.m[1][0]; m[1][1]-=n.m[1][1]; m[1][2]-=n.m[1][2]; m[1][3]-=n.m[1][3]; + m[2][0]-=n.m[2][0]; m[2][1]-=n.m[2][1]; m[2][2]-=n.m[2][2]; m[2][3]-=n.m[2][3]; + m[3][0]-=n.m[3][0]; m[3][1]-=n.m[3][1]; m[3][2]-=n.m[3][2]; m[3][3]-=n.m[3][3]; + return *this; + } + + // = operator + Matrix4& operator=(const Matrix4& matrix) { + if (&matrix != this) { + setAllValues(matrix.m[0][0], matrix.m[0][1], matrix.m[0][2], matrix.m[0][3], + matrix.m[1][0], matrix.m[1][1], matrix.m[1][2], matrix.m[1][3], + matrix.m[2][0], matrix.m[2][1], matrix.m[2][2], matrix.m[2][3], + matrix.m[3][0], matrix.m[3][1], matrix.m[3][2], matrix.m[3][3]); + } + return *this; + } + + // == operator + bool operator==(const Matrix4 &n) const { + return m[0][0]==n.m[0][0] && m[0][1]==n.m[0][1] && m[0][2]==n.m[0][2] && m[0][3]==n.m[0][3] && + m[1][0]==n.m[1][0] && m[1][1]==n.m[1][1] && m[1][2]==n.m[1][2] && m[1][3]==n.m[1][3] && + m[2][0]==n.m[2][0] && m[2][1]==n.m[2][1] && m[2][2]==n.m[2][2] && m[2][3]==n.m[2][3] && + m[3][0]==n.m[3][0] && m[3][1]==n.m[3][1] && m[3][2]==n.m[3][2] && m[3][3]==n.m[3][3]; + } + + // * operator + Matrix4 operator*(const Matrix4 &n) const { + Matrix4 o; + for(int i = 0; i < 4; i++) { + for(int j = 0; j < 4; j++) { + float v = 0; + for(int k = 0; k < 4; k++) { + v += m[i][k] * n.m[k][j]; + } + o.m[i][j] = v; + } + } + return o; + } + + // * operator + Vector3 operator*(const Vector3 &v) const { + Vector3 u =Vector3(m[0][0]*v.x + m[0][1]*v.y + m[0][2]*v.z + m[0][3], + m[1][0]*v.x + m[1][1]*v.y + m[1][2]*v.z + m[1][3], + m[2][0]*v.x + m[2][1]*v.y + m[2][2]*v.z + m[2][3]); + float w = m[3][0]*v.x + m[3][1]*v.y + m[3][2]*v.z + m[3][3]; + return u/w; + } + + // * operator + Vector4 operator*(const Vector4 &v) const { + Vector4 u = Vector4(m[0][0]*v.x + m[0][1]*v.y + m[0][2]*v.z + v.w*m[0][3], + m[1][0]*v.x + m[1][1]*v.y + m[1][2]*v.z + v.w*m[1][3], + m[2][0]*v.x + m[2][1]*v.y + m[2][2]*v.z + v.w*m[2][3], + m[3][0]*v.x + m[3][1]*v.y + m[3][2]*v.z + v.w*m[3][3]); + if(u.w != 0) + return u/u.w; + else + return u; + } + + // * operator + Matrix4 operator*(float f) const { + return Matrix4(m[0][0]*f, m[0][1]*f, m[0][2]*f, m[0][3]*f, + m[1][0]*f, m[1][1]*f, m[1][2]*f, m[1][3]*f, + m[2][0]*f, m[2][1]*f, m[2][2]*f, m[2][3]*f, + m[3][0]*f, m[3][1]*f, m[3][2]*f, m[3][3]*f); + } + + // * operator + Matrix4 &operator*=(float f) { + m[0][0]*=f; m[0][1]*=f; m[0][2]*=f; m[0][3]*=f; + m[1][0]*=f; m[1][1]*=f; m[1][2]*=f; m[1][3]*=f; + m[2][0]*=f; m[2][1]*=f; m[2][2]*=f; m[2][3]*=f; + m[3][0]*=f; m[3][1]*=f; m[3][2]*=f; m[3][3]*=f; + return *this; + } + + // / operator + Matrix4 operator/(float f) const { + assert(f!=0); + return Matrix4(m[0][0]/f, m[0][1]/f, m[0][2]/f, m[0][3]/f, + m[1][0]/f, m[1][1]/f, m[1][2]/f, m[1][3]/f, + m[2][0]/f, m[2][1]/f, m[2][2]/f, m[2][3]/f, + m[3][0]/f, m[3][1]/f, m[3][2]/f, m[3][3]/f); + } + + // /= operator + Matrix4 &operator/=(float f) { + assert(f!=0); + m[0][0]/=f; m[0][1]/=f; m[0][2]/=f; m[0][3]/=f; + m[1][0]/=f; m[1][1]/=f; m[1][2]/=f; m[1][3]/=f; + m[2][0]/=f; m[2][1]/=f; m[2][2]/=f; m[2][3]/=f; + m[3][0]/=f; m[3][1]/=f; m[3][2]/=f; m[3][3]/=f; + return *this; + } + + // - operator + Matrix4 operator-() const { + return Matrix4(-m[0][0], -m[0][1], -m[0][2], -m[0][3], + -m[1][0], -m[1][1], -m[1][2], -m[1][3], + -m[2][0], -m[2][1], -m[2][2], -m[2][3], + -m[3][0], -m[3][1], -m[3][2], -m[3][3]); + } + + // Return the transpose matrix + Matrix4 getTranspose() const { + return Matrix4(m[0][0], m[1][0], m[2][0], m[3][0], + m[0][1], m[1][1], m[2][1], m[3][1], + m[0][2], m[1][2], m[2][2], m[3][2], + m[0][3], m[1][3], m[2][3], m[3][3]); + } + + // Return the inversed matrix + Matrix4 getInverse() const { + int indxc[4], indxr[4]; + int ipiv[4] = { 0, 0, 0, 0 }; + float minv[4][4]; + float temp; + + for (int s=0; s<4; s++) { + for (int t=0; t<4; t++) { + minv[s][t] = m[s][t]; + } + } + + for (int i = 0; i < 4; i++) { + int irow = -1, icol = -1; + float big = 0.; + // Choose pivot + for (int j = 0; j < 4; j++) { + if (ipiv[j] != 1) { + for (int k = 0; k < 4; k++) { + if (ipiv[k] == 0) { + if (fabs(minv[j][k]) >= big) { + big = float(fabs(minv[j][k])); + irow = j; + icol = k; + } + } + else if (ipiv[k] > 1) { + std::cout << "ERROR: Singular matrix in MatrixInvert\n"; + } + } + } + } + ++ipiv[icol]; + // Swap rows _irow_ and _icol_ for pivot + if (irow != icol) { + for (int k = 0; k < 4; ++k){ + temp = minv[irow][k]; + minv[irow][k] = minv[icol][k]; + minv[icol][k] = temp; + } + } + indxr[i] = irow; + indxc[i] = icol; + if (minv[icol][icol] == 0.){ + std::cout << "Singular matrix in MatrixInvert\n"; + } + // Set $m[icol][icol]$ to one by scaling row _icol_ appropriately + float pivinv = 1.f / minv[icol][icol]; + minv[icol][icol] = 1.f; + for (int j = 0; j < 4; j++) { + minv[icol][j] *= pivinv; + } + + // Subtract this row from others to zero out their columns + for (int j = 0; j < 4; j++) { + if (j != icol) { + float save = minv[j][icol]; + minv[j][icol] = 0; + for (int k = 0; k < 4; k++) { + minv[j][k] -= minv[icol][k]*save; + } + } + } + } + // Swap columns to reflect permutation + for (int j = 3; j >= 0; j--) { + if (indxr[j] != indxc[j]) { + for (int k = 0; k < 4; k++){ + temp = minv[k][indxr[j]]; + minv[k][indxr[j]] = minv[k][indxc[j]]; + minv[k][indxc[j]] = temp; + } + } + } + return Matrix4(minv); + } + + // Method to set all the values in the matrix + void setAllValues(float a1, float a2, float a3, float a4, + float b1, float b2, float b3, float b4, + float c1, float c2, float c3, float c4, + float d1, float d2, float d3, float d4) { + m[0][0] = a1; m[0][1] = a2; m[0][2] = a3, m[0][3] = a4; + m[1][0] = b1; m[1][1] = b2; m[1][2] = b3; m[1][3] = b4; + m[2][0] = c1; m[2][1] = c2; m[2][2] = c3; m[2][3] = c4; + m[3][0] = d1; m[3][1] = d2; m[3][2] = d3; m[3][3] = d4; + } + + // Set the matrix to the identity matrix + Matrix4 setToIdentity() { + m[0][0] = 1.f; m[0][1] = 0.f; m[0][2] = 0.f; m[0][3] = 0.f; + m[1][0] = 0.f; m[1][1] = 1.f; m[1][2] = 0.f; m[1][3] = 0.f; + m[2][0] = 0.f; m[2][1] = 0.f; m[2][2] = 1.f; m[2][3] = 0.f; + m[3][0] = 0.f; m[3][1] = 0.f; m[3][2] = 0.f; m[3][3] = 1.f; + return *this; + } + + // Display the matrix + void print() const { + for (int i=0; i<4; i++) { + for (int j=0; j<4; j++) { + std::cout << m[i][j]; + } + std::cout << std::endl; + } + } + + // Return the pointer to the data array of the matrix + float* dataBlock() { + return m[0]; + } + + // Return the constant pointer to the data array of the matrix + const float* dataBlock() const { + return m[0]; + } + + // Return a given value from the matrix + float getValue(int i, int j) const { + assert(i >= 0 && i<4 && j >= 0 && j<4); + return m[i][j]; + } + + // Return the trace of the matrix + float getTrace() const { + // Compute and return the trace + return (m[0][0] + m[1][1] + m[2][2] + m[3][3]); + } + + // Return a 4x4 translation matrix + static Matrix4 translationMatrix(const Vector3& v); + + // Return a 4x4 rotation matrix + static Matrix4 rotationMatrix(const Vector3& axis, float angle); +}; + +// * operator +inline Matrix4 operator*(float f, const Matrix4 & m) { + return (m * f); +} + +// Return a 4x4 translation matrix +inline Matrix4 Matrix4::translationMatrix(const Vector3& v) { + return Matrix4(1, 0, 0, v.x, + 0, 1, 0, v.y, + 0, 0, 1, v.z, + 0, 0, 0, 1); +} + +// Return a 4x4 rotation matrix +inline Matrix4 Matrix4::rotationMatrix(const Vector3& axis, float angle) { + + float cosA = cos(angle); + float sinA = sin(angle); + Matrix4 rotationMatrix; + rotationMatrix.setToIdentity(); + + rotationMatrix.m[0][0] = cosA + (1-cosA) * axis.x * axis.x; + rotationMatrix.m[0][1] = (1-cosA) * axis.x * axis.y - axis.z * sinA; + rotationMatrix.m[0][2] = (1-cosA) * axis.x * axis.z + axis.y * sinA; + rotationMatrix.m[0][3] = 0.f; + + rotationMatrix.m[1][0] = (1-cosA) * axis.x * axis.y + axis.z * sinA; + rotationMatrix.m[1][1] = cosA + (1-cosA) * axis.y * axis.y; + rotationMatrix.m[1][2] = (1-cosA) * axis.y * axis.z - axis.x * sinA; + rotationMatrix.m[1][3] = 0.f; + + rotationMatrix.m[2][0] = (1-cosA) * axis.x * axis.z - axis.y * sinA; + rotationMatrix.m[2][1] = (1-cosA) * axis.y * axis.z + axis.x * sinA; + rotationMatrix.m[2][2] = cosA + (1-cosA) * axis.z * axis.z; + rotationMatrix.m[2][3] = 0.f; + + rotationMatrix.m[3][0] = 0.f; + rotationMatrix.m[3][1] = 0.f; + rotationMatrix.m[3][2] = 0.f; + rotationMatrix.m[3][3] = 1.f; + + return rotationMatrix; +} + +} + +#endif //_MATRIX4_H diff --git a/examples/opengl-framework/src/maths/Vector2.h b/examples/opengl-framework/src/maths/Vector2.h new file mode 100644 index 00000000..0bc0eb9c --- /dev/null +++ b/examples/opengl-framework/src/maths/Vector2.h @@ -0,0 +1,159 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 VECTOR2_H +#define VECTOR2_H + +// Libraries +#include +#include + +namespace openglframework { + +// Class Vector2 +// This class represents a 2D vector. +class Vector2 { + + public: + + // -------------------- Attributes -------------------- // + + // Components of the vector + float x, y; + + + // -------------------- Methods -------------------- // + + // Constructor + Vector2(float x=0, float y=0) : x(x), y(y) {} + + // Constructor + Vector2(const Vector2& vector) : x(vector.x), y(vector.y) {} + + // + operator + Vector2 operator+(const Vector2 &v) const { + return Vector2(x + v.x, y + v.y); + } + + // += operator + Vector2& operator+=(const Vector2 &v) { + x += v.x; y += v.y; + return *this; + } + + // - operator + Vector2 operator-(const Vector2 &v) const { + return Vector2(x - v.x, y - v.y); + } + + // -= operator + Vector2& operator-=(const Vector2 &v) { + x -= v.x; y -= v.y; + return *this; + } + + // = operator + Vector2& operator=(const Vector2& vector) { + if (&vector != this) { + x = vector.x; + y = vector.y; + } + return *this; + } + + // == operator + bool operator==(const Vector2 &v) const { + return x == v.x && y == v.y; + } + + // * operator + Vector2 operator*(float f) const { + return Vector2(f*x, f*y); + } + + // *= operator + Vector2 &operator*=(float f) { + x *= f; y *= f; + return *this; + } + + // / operator + Vector2 operator/(float f) const { + assert(f!=0); + float inv = 1.f / f; + return Vector2(x * inv, y * inv); + } + + // /= operator + Vector2 &operator/=(float f) { + assert(f!=0); + float inv = 1.f / f; + x *= inv; y *= inv; + return *this; + } + + // - operator + Vector2 operator-() const { + return Vector2(-x, -y); + } + + // [] operator + float &operator[](int i) { + assert(i >= 0 && i <= 1); + switch (i) { + case 0: return x; + case 1: return y; + } + return y; + } + + // Normalize the vector and return it + Vector2 normalize() { + float l = length(); + assert(l > 0); + x /= l; + y /= l; + return *this; + } + + // Clamp the vector values between 0 and 1 + Vector2 clamp01() { + if (x>1.f) x=1.f; + else if (x<0.f) x=0.f; + if (y>1.f) y=1.f; + else if (y<0.f) y=0.f; + return *this; + } + + // Return the squared length of the vector + float lengthSquared() const { return x*x + y*y; } + + // Return the length of the vector + float length() const { return sqrt(lengthSquared()); } +}; + +} + +#endif diff --git a/examples/opengl-framework/src/maths/Vector3.h b/examples/opengl-framework/src/maths/Vector3.h new file mode 100644 index 00000000..adeecb7c --- /dev/null +++ b/examples/opengl-framework/src/maths/Vector3.h @@ -0,0 +1,203 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 OPENGLFRAMEWORK_VECTOR3_H +#define OPENGLFRAMEWORK_VECTOR3_H + +// Libraries +#include +#include +#include + +namespace openglframework { + +// Class Vector3 +// This class represents a 3D vector. +class Vector3 { + + public: + + // -------------------- Attributes -------------------- // + + // Components of the vector + float x, y, z; + + // -------------------- Methods -------------------- // + + // Constructor + Vector3(float x=0, float y=0, float z=0) : x(x), y(y), z(z) {} + + // Constructor + Vector3(const Vector3& vector) : x(vector.x), y(vector.y), z(vector.z) {} + + // Constructor + ~Vector3() {} + + // = operator + Vector3& operator=(const Vector3& vector) { + if (&vector != this) { + x = vector.x; + y = vector.y; + z = vector.z; + } + return *this; + } + + // + operator + Vector3 operator+(const Vector3 &v) const { + return Vector3(x + v.x, y + v.y, z + v.z); + } + + // += operator + Vector3& operator+=(const Vector3 &v) { + x += v.x; y += v.y; z += v.z; + return *this; + } + + // - operator + Vector3 operator-(const Vector3 &v) const { + return Vector3(x - v.x, y - v.y, z - v.z); + } + + // -= operator + Vector3& operator-=(const Vector3 &v) { + x -= v.x; y -= v.y; z -= v.z; + return *this; + } + + // == operator + bool operator==(const Vector3 &v) const { + return x == v.x && y == v.y && z == v.z; + } + + // != operator + bool operator!=(const Vector3 &v) const { + return !( *this == v ); + } + + // * operator + Vector3 operator*(float f) const { + return Vector3(f*x, f*y, f*z); + } + + // *= operator + Vector3 &operator*=(float f) { + x *= f; y *= f; z *= f; + return *this; + } + + // / operator + Vector3 operator/(float f) const { + assert(f > std::numeric_limits::epsilon() ); + float inv = 1.f / f; + return Vector3(x * inv, y * inv, z * inv); + } + + // /= operator + Vector3 &operator/=(float f) { + assert(f > std::numeric_limits::epsilon()); + float inv = 1.f / f; + x *= inv; y *= inv; z *= inv; + return *this; + } + + // - operator + Vector3 operator-() const { + return Vector3(-x, -y, -z); + } + + // [] operator + float &operator[](int i) { + assert(i >= 0 && i <= 2); + switch (i) { + case 0: return x; + case 1: return y; + case 2: return z; + } + return z; + } + + // [] operator + const float &operator[](int i) const { + assert(i >= 0 && i <= 2); + switch (i) { + case 0: return x; + case 1: return y; + case 2: return z; + } + return z; + } + + // Cross product operator + Vector3 cross(const Vector3 &v) const{ + return Vector3(y * v.z - z * v.y, z * v.x - x * v.z, x * v.y - y * v.x); + } + + // Dot product operator + float dot(const Vector3 &v) const{ + return x * v.x + y * v.y + z * v.z; + } + + // Normalize the vector and return it + Vector3 normalize() { + float l = length(); + if(l < std::numeric_limits::epsilon() ) { + assert(false); + } + x /= l; + y /= l; + z /= l; + return *this; + } + + bool isNull() const { + return( x == 0. && y == 0. && z == 0. ); + } + + // Clamp the values between 0 and 1 + Vector3 clamp01() { + if (x>1.f) x=1.f; + else if (x<0.f) x=0.f; + if (y>1.f) y=1.f; + else if (y<0.f) y=0.f; + if (z>1.f) z=1.f; + else if (z<0.f) z=0.f; + return *this; + } + + // Return the squared length of the vector + float lengthSquared() const { return x*x + y*y + z*z; } + + // Return the length of the vector + float length() const { return sqrt(lengthSquared()); } +}; + +inline Vector3 operator*(float f, const Vector3 & o) { + return o*f; +} + +} + +#endif diff --git a/examples/opengl-framework/src/maths/Vector4.h b/examples/opengl-framework/src/maths/Vector4.h new file mode 100644 index 00000000..3c4002d8 --- /dev/null +++ b/examples/opengl-framework/src/maths/Vector4.h @@ -0,0 +1,167 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 VECTOR4_H +#define VECTOR4_H + +// Libraries +#include +#include + +namespace openglframework { + +// Class Vector4 +// This class represents a 4D vector. +class Vector4 { + + public: + + // -------------------- Attributes -------------------- // + + // Components of the vector + float x, y, z, w; + + // -------------------- Methods -------------------- // + + // Constructor + Vector4(float x=0, float y=0, float z=0, float w=0) : x(x), y(y), z(z), w(w) {} + + // Constructor + Vector4(const Vector4& vector) : x(vector.x), y(vector.y), z(vector.z), w(vector.w) {} + + // + operator + Vector4 operator+(const Vector4 &v) const { + return Vector4(x + v.x, y + v.y, z + v.z, w + v.w); + } + + // += operator + Vector4& operator+=(const Vector4 &v) { + x += v.x; y += v.y; z += v.z; w += v.w; + return *this; + } + + // - operator + Vector4 operator-(const Vector4 &v) const { + return Vector4(x - v.x, y - v.y, z - v.z, w - v.w); + } + + // -= operator + Vector4& operator-=(const Vector4 &v) { + x -= v.x; y -= v.y; z -= v.z, w -=v.w; + return *this; + } + + // = operator + Vector4& operator=(const Vector4& vector) { + if (&vector != this) { + x = vector.x; + y = vector.y; + z = vector.z; + w = vector.w; + } + return *this; + } + + // == operator + bool operator==(const Vector4 &v) const { + return x == v.x && y == v.y && z == v.z && w == v.w; + } + + // * operator + Vector4 operator*(float f) const { + return Vector4(f*x, f*y, f*z, f*w); + } + + // *= operator + Vector4 &operator*=(float f) { + x *= f; y *= f; z *= f; w *= f; + return *this; + } + + // / operator + Vector4 operator/(float f) const { + assert(f!=0); + float inv = 1.f / f; + return Vector4(x * inv, y * inv, z * inv, w * inv); + } + + // /= operator + Vector4 &operator/=(float f) { + assert(f!=0); + float inv = 1.f / f; + x *= inv; y *= inv; z *= inv; w *= inv; + return *this; + } + + // - operator + Vector4 operator-() const { + return Vector4(-x, -y, -z, -w); + } + + // [] operator + float &operator[](int i) { + assert(i >= 0 && i <= 3); + switch (i) { + case 0: return x; + case 1: return y; + case 2: return z; + case 3: return w; + } + return w; + } + + // Dot product operator + float dot(const Vector4 &v) const { + return x * v.x + y * v.y + z * v.z + w * v.w; + } + + // Multiply two vectors by their components + Vector4 componentMul(const Vector4 &v) const { + return Vector4(x * v.x, y * v.y, z * v.z, w * v.w); + } + + // Clamp the values between 0 and 1 + Vector4 clamp01() { + if (x>1.f) x=1.f; + else if (x<0.f) x=0.f; + if (y>1.f) y=1.f; + else if (y<0.f) y=0.f; + if (z>1.f) z=1.f; + else if (z<0.f) z=0.f; + if (w>1.f) w=1.f; + else if (w<0.f) w=0.f; + return *this; + } + + // Return the squared length of the vector + float lengthSquared() const { return x * x + y * y + z * z + w * w; } + + // Return the length of the vector + float length() const { return sqrt(lengthSquared()); } +}; + +} + +#endif //_VECTOR4_H diff --git a/examples/opengl-framework/src/openglframework.h b/examples/opengl-framework/src/openglframework.h new file mode 100644 index 00000000..b8a0a41a --- /dev/null +++ b/examples/opengl-framework/src/openglframework.h @@ -0,0 +1,49 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 OPENGL_FRAMEWORK_H +#define OPENGL_FRAMEWORK_H + +// Libraries +#include "MeshReaderWriter.h" +#include "TextureReaderWriter.h" +#include "GlutViewer.h" +#include "Camera.h" +#include "Light.h" +#include "Mesh.h" +#include "Shader.h" +#include "Texture2D.h" +#include "FrameBufferObject.h" +#include "VertexBufferObject.h" +#include "Shader.h" +#include "maths/Color.h" +#include "maths/Vector2.h" +#include "maths/Vector3.h" +#include "maths/Vector4.h" +#include "maths/Matrix4.h" +#include "maths/Matrix3.h" +#include "definitions.h" + +#endif diff --git a/examples/opengl-framework/src/shaders/depth.frag b/examples/opengl-framework/src/shaders/depth.frag new file mode 100644 index 00000000..4ef97b32 --- /dev/null +++ b/examples/opengl-framework/src/shaders/depth.frag @@ -0,0 +1,32 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +void main(void) { + + // Compute the depth of the pixel + float depth = + + gl_FragColor = vec4(depth, depth, depth, 1); +} diff --git a/examples/opengl-framework/src/shaders/depth.vert b/examples/opengl-framework/src/shaders/depth.vert new file mode 100644 index 00000000..f320cc72 --- /dev/null +++ b/examples/opengl-framework/src/shaders/depth.vert @@ -0,0 +1,36 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Uniform variables +uniform mat4 modelToWorldMatrix; // Model too world coordinates matrix +uniform mat4 worldToCameraMatrix; // World to camera coordinates matrix +uniform mat4 projectionMatrix; // Projection matrix + +void main(void) { + + // Compute the clip-space vertex coordinates + gl_Position = projectionMatrix * worldToCameraMatrix * + modelToWorldMatrix * gl_Vertex; +} diff --git a/examples/opengl-framework/src/shaders/phong.frag b/examples/opengl-framework/src/shaders/phong.frag new file mode 100644 index 00000000..8995f7f0 --- /dev/null +++ b/examples/opengl-framework/src/shaders/phong.frag @@ -0,0 +1,63 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Uniform variables +uniform vec3 cameraWorldPosition; // World position of the camera +uniform vec3 lightWorldPosition; // World position of the light +uniform vec3 lightAmbientColor; // Lights ambient color +uniform vec3 lightDiffuseColor; // Light diffuse color +uniform vec3 lightSpecularColor; // Light specular color +uniform float shininess; // Shininess +uniform sampler2D texture; // Texture +uniform bool isTexture; // True if we need to use the texture + +// Varying variables +varying vec3 worldPosition; // World position of the vertex +varying vec3 worldNormal; // World surface normalWorld +varying vec2 texCoords; // Texture coordinates + +void main() { + + // Compute the ambient term + vec3 ambient = lightAmbientColor; + + // Get the texture color + + vec3 textureColor = vec3(1); + if (isTexture) textureColor = texture2D(texture, texCoords).rgb; + + // Compute the diffuse term + vec3 L = normalize(lightWorldPosition - worldPosition); + vec3 N = normalize(worldNormal); + vec3 diffuse = lightDiffuseColor * max(dot(N, L), 0.0) * textureColor; + + // Compute the specular term + vec3 V = normalize(cameraWorldPosition - worldPosition); + vec3 H = normalize(V + L); + vec3 specular = lightSpecularColor * pow(max(dot(N, H), 0), shininess); + + // Compute the final color + gl_FragColor = vec4(ambient + diffuse + specular, 1.0); +} diff --git a/examples/opengl-framework/src/shaders/phong.vert b/examples/opengl-framework/src/shaders/phong.vert new file mode 100644 index 00000000..97bca225 --- /dev/null +++ b/examples/opengl-framework/src/shaders/phong.vert @@ -0,0 +1,51 @@ +/******************************************************************************** +* OpenGL-Framework * +* Copyright (c) 2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Uniform variables +uniform mat4 modelToWorldMatrix; // Model too world coordinates matrix +uniform mat4 worldToCameraMatrix; // World to camera coordinates matrix +uniform mat4 projectionMatrix; // Projection matrix + +// Varying variables +varying vec3 worldPosition; // World position of the vertex +varying vec3 worldNormal; // World surface normalWorld +varying vec2 texCoords; // Texture coordinates + +void main() { + + // Compute the vertex position + vec4 worldPos = modelToWorldMatrix * gl_Vertex; + worldPosition = worldPos.xyz; + + // Compute the world surface normal + vec3 bodyNormal = normalize(gl_Normal); + worldNormal = (modelToWorldMatrix * vec4(bodyNormal, 0.0)).xyz; + + // Get the texture coordinates + texCoords = gl_MultiTexCoord0.xy; + + // Compute the clip-space vertex coordinates + gl_Position = projectionMatrix * worldToCameraMatrix * worldPos; +} From 91908c1bbc8dcbdd4ab058231e5de95a4eb2a446 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 18 Apr 2013 22:22:45 +0200 Subject: [PATCH 07/66] Continue to implement the examples using the openg-framework --- documentation/API/ReactPhysics3DLogo.png | Bin .../UserManual/ReactPhysics3D-UserManual.tex | 0 .../UserManual/images/ReactPhysics3DLogo.png | Bin documentation/UserManual/title.tex | 0 examples/CMakeLists.txt | 4 +- examples/common/Box.cpp | 180 ++++++++++++++++++ examples/common/Box.h | 111 +++++++++++ examples/common/CMakeLists.txt | 4 + examples/common/Viewer.cpp | 83 ++++++++ examples/common/Viewer.h | 71 +++++++ .../opengl-framework/CMakeLists.txt | 0 .../opengl-framework/freeglut/CMakeLists.txt | 0 .../opengl-framework/freeglut/COPYING.txt | 0 .../opengl-framework/freeglut/GL/freeglut.h | 0 .../freeglut/GL/freeglut_ext.h | 0 .../freeglut/GL/freeglut_std.h | 0 .../opengl-framework/freeglut/GL/glut.h | 0 .../opengl-framework/freeglut/VERSION.txt | 0 .../freeglut/freeglut_callbacks.c | 0 .../freeglut/freeglut_cursor.c | 0 .../freeglut/freeglut_display.c | 0 .../opengl-framework/freeglut/freeglut_ext.c | 0 .../opengl-framework/freeglut/freeglut_font.c | 0 .../freeglut/freeglut_font_data.c | 0 .../freeglut/freeglut_gamemode.c | 0 .../freeglut/freeglut_geometry.c | 0 .../freeglut/freeglut_glutfont_definitions.c | 0 .../opengl-framework/freeglut/freeglut_init.c | 0 .../freeglut/freeglut_input_devices.c | 0 .../freeglut/freeglut_internal.h | 0 .../freeglut/freeglut_joystick.c | 0 .../opengl-framework/freeglut/freeglut_main.c | 0 .../opengl-framework/freeglut/freeglut_menu.c | 0 .../opengl-framework/freeglut/freeglut_misc.c | 0 .../freeglut/freeglut_overlay.c | 0 .../freeglut/freeglut_spaceball.c | 0 .../freeglut/freeglut_state.c | 0 .../freeglut/freeglut_stroke_mono_roman.c | 0 .../freeglut/freeglut_stroke_roman.c | 0 .../freeglut/freeglut_structure.c | 0 .../freeglut/freeglut_teapot.c | 0 .../freeglut/freeglut_teapot_data.h | 0 .../freeglut/freeglut_videoresize.c | 0 .../freeglut/freeglut_window.c | 0 .../freeglut/freeglut_xinput.c | 0 .../opengl-framework/freeglut/glut.h | 0 .../opengl-framework/src/Camera.cpp | 0 .../opengl-framework/src/Camera.h | 0 .../src/FrameBufferObject.cpp | 0 .../opengl-framework/src/FrameBufferObject.h | 0 .../opengl-framework/src/GlutViewer.cpp | 10 + .../opengl-framework/src/GlutViewer.h | 13 +- .../opengl-framework/src/Light.cpp | 0 .../{ => common}/opengl-framework/src/Light.h | 0 .../opengl-framework/src/Mesh.cpp | 0 .../{ => common}/opengl-framework/src/Mesh.h | 0 .../opengl-framework/src/MeshReaderWriter.cpp | 0 .../opengl-framework/src/MeshReaderWriter.h | 0 .../opengl-framework/src/Object3D.cpp | 0 .../opengl-framework/src/Object3D.h | 0 .../opengl-framework/src/Shader.cpp | 0 .../opengl-framework/src/Shader.h | 0 .../opengl-framework/src/Texture2D.cpp | 0 .../opengl-framework/src/Texture2D.h | 0 .../src/TextureReaderWriter.cpp | 0 .../src/TextureReaderWriter.h | 0 .../src/VertexBufferObject.cpp | 0 .../opengl-framework/src/VertexBufferObject.h | 0 .../opengl-framework/src/definitions.h | 0 .../opengl-framework/src/maths/Color.h | 0 .../opengl-framework/src/maths/Matrix3.h | 0 .../opengl-framework/src/maths/Matrix4.h | 0 .../opengl-framework/src/maths/Vector2.h | 0 .../opengl-framework/src/maths/Vector3.h | 0 .../opengl-framework/src/maths/Vector4.h | 0 .../opengl-framework/src/openglframework.h | 0 .../opengl-framework/src/shaders/depth.frag | 0 .../opengl-framework/src/shaders/depth.vert | 0 .../opengl-framework/src/shaders/phong.frag | 0 .../opengl-framework/src/shaders/phong.vert | 0 examples/fallingcubes/CMakeLists.txt | 6 +- .../{main.cpp => FallingCubes.cpp} | 96 +++------- examples/fallingcubes/Scene.cpp | 45 +++-- examples/fallingcubes/Scene.h | 36 +++- src/engine/DynamicsWorld.h | 1 - 85 files changed, 565 insertions(+), 95 deletions(-) mode change 100755 => 100644 documentation/API/ReactPhysics3DLogo.png mode change 100755 => 100644 documentation/UserManual/ReactPhysics3D-UserManual.tex mode change 100755 => 100644 documentation/UserManual/images/ReactPhysics3DLogo.png mode change 100755 => 100644 documentation/UserManual/title.tex create mode 100644 examples/common/Box.cpp create mode 100644 examples/common/Box.h create mode 100644 examples/common/CMakeLists.txt create mode 100644 examples/common/Viewer.cpp create mode 100644 examples/common/Viewer.h rename examples/{ => common}/opengl-framework/CMakeLists.txt (100%) rename examples/{ => common}/opengl-framework/freeglut/CMakeLists.txt (100%) rename examples/{ => common}/opengl-framework/freeglut/COPYING.txt (100%) rename examples/{ => common}/opengl-framework/freeglut/GL/freeglut.h (100%) rename examples/{ => common}/opengl-framework/freeglut/GL/freeglut_ext.h (100%) rename examples/{ => common}/opengl-framework/freeglut/GL/freeglut_std.h (100%) rename examples/{ => common}/opengl-framework/freeglut/GL/glut.h (100%) rename examples/{ => common}/opengl-framework/freeglut/VERSION.txt (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_callbacks.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_cursor.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_display.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_ext.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_font.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_font_data.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_gamemode.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_geometry.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_glutfont_definitions.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_init.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_input_devices.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_internal.h (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_joystick.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_main.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_menu.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_misc.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_overlay.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_spaceball.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_state.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_stroke_mono_roman.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_stroke_roman.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_structure.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_teapot.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_teapot_data.h (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_videoresize.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_window.c (100%) rename examples/{ => common}/opengl-framework/freeglut/freeglut_xinput.c (100%) rename examples/{ => common}/opengl-framework/freeglut/glut.h (100%) rename examples/{ => common}/opengl-framework/src/Camera.cpp (100%) rename examples/{ => common}/opengl-framework/src/Camera.h (100%) rename examples/{ => common}/opengl-framework/src/FrameBufferObject.cpp (100%) rename examples/{ => common}/opengl-framework/src/FrameBufferObject.h (100%) rename examples/{ => common}/opengl-framework/src/GlutViewer.cpp (97%) rename examples/{ => common}/opengl-framework/src/GlutViewer.h (95%) rename examples/{ => common}/opengl-framework/src/Light.cpp (100%) rename examples/{ => common}/opengl-framework/src/Light.h (100%) rename examples/{ => common}/opengl-framework/src/Mesh.cpp (100%) rename examples/{ => common}/opengl-framework/src/Mesh.h (100%) rename examples/{ => common}/opengl-framework/src/MeshReaderWriter.cpp (100%) rename examples/{ => common}/opengl-framework/src/MeshReaderWriter.h (100%) rename examples/{ => common}/opengl-framework/src/Object3D.cpp (100%) rename examples/{ => common}/opengl-framework/src/Object3D.h (100%) rename examples/{ => common}/opengl-framework/src/Shader.cpp (100%) rename examples/{ => common}/opengl-framework/src/Shader.h (100%) rename examples/{ => common}/opengl-framework/src/Texture2D.cpp (100%) rename examples/{ => common}/opengl-framework/src/Texture2D.h (100%) rename examples/{ => common}/opengl-framework/src/TextureReaderWriter.cpp (100%) rename examples/{ => common}/opengl-framework/src/TextureReaderWriter.h (100%) rename examples/{ => common}/opengl-framework/src/VertexBufferObject.cpp (100%) rename examples/{ => common}/opengl-framework/src/VertexBufferObject.h (100%) rename examples/{ => common}/opengl-framework/src/definitions.h (100%) rename examples/{ => common}/opengl-framework/src/maths/Color.h (100%) rename examples/{ => common}/opengl-framework/src/maths/Matrix3.h (100%) rename examples/{ => common}/opengl-framework/src/maths/Matrix4.h (100%) rename examples/{ => common}/opengl-framework/src/maths/Vector2.h (100%) rename examples/{ => common}/opengl-framework/src/maths/Vector3.h (100%) rename examples/{ => common}/opengl-framework/src/maths/Vector4.h (100%) rename examples/{ => common}/opengl-framework/src/openglframework.h (100%) rename examples/{ => common}/opengl-framework/src/shaders/depth.frag (100%) rename examples/{ => common}/opengl-framework/src/shaders/depth.vert (100%) rename examples/{ => common}/opengl-framework/src/shaders/phong.frag (100%) rename examples/{ => common}/opengl-framework/src/shaders/phong.vert (100%) rename examples/fallingcubes/{main.cpp => FallingCubes.cpp} (72%) diff --git a/documentation/API/ReactPhysics3DLogo.png b/documentation/API/ReactPhysics3DLogo.png old mode 100755 new mode 100644 diff --git a/documentation/UserManual/ReactPhysics3D-UserManual.tex b/documentation/UserManual/ReactPhysics3D-UserManual.tex old mode 100755 new mode 100644 diff --git a/documentation/UserManual/images/ReactPhysics3DLogo.png b/documentation/UserManual/images/ReactPhysics3DLogo.png old mode 100755 new mode 100644 diff --git a/documentation/UserManual/title.tex b/documentation/UserManual/title.tex old mode 100755 new mode 100644 diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt index 4f0cab67..568479fb 100644 --- a/examples/CMakeLists.txt +++ b/examples/CMakeLists.txt @@ -1,5 +1,5 @@ # Minimum cmake version required CMAKE_MINIMUM_REQUIRED(VERSION 2.6) -add_subdirectory(opengl-framework/) -add_subdirectory(fallingcubes/) \ No newline at end of file +add_subdirectory(common/) +add_subdirectory(fallingcubes/) diff --git a/examples/common/Box.cpp b/examples/common/Box.cpp new file mode 100644 index 00000000..d4169c9a --- /dev/null +++ b/examples/common/Box.cpp @@ -0,0 +1,180 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Box.h" + +// Macros +#define MEMBER_OFFSET(s,m) ((char *)NULL + (offsetof(s,m))) + +// Initialize static variables +openglframework::VertexBufferObject Box::mVBOVertices(GL_ARRAY_BUFFER); +openglframework::VertexBufferObject Box::mVBOIndices(GL_ELEMENT_ARRAY_BUFFER); +bool Box::areVBOsCreated = false; +VertexData Box::mCubeVertices[8] = { + {openglframework::Vector3(1,1,1),openglframework::Vector3(1,1,1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(-1,1,1),openglframework::Vector3(-1,1,1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(-1,-1,1),openglframework::Vector3(-1,-1,1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(1,-1,1),openglframework::Vector3(1,-1,1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(1,-1,-1),openglframework::Vector3(1,-1,-1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(-1,-1,-1),openglframework::Vector3(-1,-1,-1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(-1,1,-1),openglframework::Vector3(-1,1,-1),openglframework::Color(0,0,1,1)}, + {openglframework::Vector3(1,1,-1),openglframework::Vector3(1,1,-1),openglframework::Color(0,0,1,1)} +}; +GLuint Box::mCubeIndices[36] = { 0, 1, 2, + 2, 3, 0, + 7, 4, 5, + 5, 6, 7, + 6, 5, 2, + 2, 1, 6, + 7, 0, 3, + 3, 4, 7, + 7, 6, 1, + 1, 0, 7, + 3, 2, 5, + 5, 4, 3}; + +// Constructor +Box::Box(const openglframework::Vector3& size, const openglframework::Vector3 &position, + float mass, reactphysics3d::DynamicsWorld* dynamicsWorld) + : openglframework::Object3D() { + + // Initialize the size of the box + mSize[0] = size.x * 0.5f; + mSize[1] = size.y * 0.5f; + mSize[2] = size.z * 0.5f; + + // Compute the scaling matrix + mScalingMatrix = openglframework::Matrix4(mSize[0], 0, 0, 0, + 0, mSize[1], 0, 0, + 0, 0, mSize[2], 0, + 0, 0, 0, 1); + + // Initialize the position where the cube will be rendered + translateWorld(position); + + // Create the collision shape for the rigid body (box shape) + mCollisionShape = new rp3d::BoxShape(rp3d::Vector3(mSize[0], mSize[1], mSize[2])); + + // Compute the inertia tensor of the body using its collision shape + rp3d::Matrix3x3 inertiaTensor; + mCollisionShape->computeLocalInertiaTensor(inertiaTensor, mass); + + // Initial position and orientation of the rigid body + rp3d::Vector3 initPosition(position.x, position.y, position.z); + rp3d::Quaternion initOrientation = rp3d::Quaternion::identity(); + rp3d::Transform transform(initPosition, initOrientation); + + // Create a rigid body corresponding to the cube in the dynamics world + mRigidBody = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, mCollisionShape); + + // If the Vertex Buffer object has not been created yet + if (!areVBOsCreated) { + // Create the Vertex Buffer + createVBO(); + } +} + +// Destructor +Box::~Box() { + + // Destroy the collision shape + delete mCollisionShape; +} + +// Render the cube at the correct position and with the correct orientation +void Box::render(openglframework::Shader& shader) { + + // Bind the shader + shader.bind(); + + // Set the model to World matrix + shader.setMatrix4x4Uniform("modelToWorldMatrix", mTransformMatrix); + + // Bind the vertices VBO + mVBOVertices.bind(); + + // Enable the vertex, normal and color arrays + glEnableClientState(GL_VERTEX_ARRAY); + glEnableClientState(GL_COLOR_ARRAY); + glEnableClientState(GL_NORMAL_ARRAY); + + // Set the arrays pointers + glVertexPointer(3, GL_FLOAT, sizeof(VertexData), MEMBER_OFFSET(VertexData, position)); + glNormalPointer(GL_FLOAT, sizeof(VertexData), MEMBER_OFFSET(VertexData, normal)); + glColorPointer(3, GL_FLOAT, sizeof(VertexData), MEMBER_OFFSET(VertexData, color)); + + // Bind the indices VBO + mVBOIndices.bind(); + + // Draw the geometry of the box + glDrawElements(GL_TRIANGLES, 36, GL_UNSIGNED_INT, (char*)NULL); + + // Unbind the VBOs + mVBOVertices.unbind(); + mVBOIndices.unbind(); + + // Disable the arrays + glDisableClientState(GL_VERTEX_ARRAY); + glDisableClientState(GL_COLOR_ARRAY); + glDisableClientState(GL_NORMAL_ARRAY); + + // Unbind the shader + shader.unbind(); +} + +// Update the transform matrix of the box +void Box::updateTransform() { + + // Get the interpolated transform of the rigid body + rp3d::Transform transform = mRigidBody->getInterpolatedTransform(); + + // Compute the transform used for rendering the box + float matrix[16]; + transform.getOpenGLMatrix(matrix); + openglframework::Matrix4 newMatrix(matrix[0], matrix[4], matrix[8], matrix[12], + matrix[1], matrix[5], matrix[9], matrix[13], + matrix[2], matrix[6], matrix[10], matrix[14], + matrix[3], matrix[7], matrix[11], matrix[15]); + + // Apply the scaling matrix to have the correct box dimensions + mTransformMatrix = newMatrix * mScalingMatrix; +} + +// Create the Vertex Buffer Objects used to render to box with OpenGL. +/// We create two VBOs (one for vertices and one for indices) to render all the boxes +/// in the simulation. +void Box::createVBO() { + + // Create the VBOs + mVBOVertices.create(); + mVBOIndices.create(); + + // Copy the data into the VBOs + mVBOVertices.copyDataIntoVBO(sizeof(mCubeVertices), mCubeVertices, GL_STATIC_DRAW); + mVBOIndices.copyDataIntoVBO(sizeof(mCubeIndices), mCubeIndices, GL_STATIC_DRAW); + + areVBOsCreated = true; +} diff --git a/examples/common/Box.h b/examples/common/Box.h new file mode 100644 index 00000000..d287c3cb --- /dev/null +++ b/examples/common/Box.h @@ -0,0 +1,111 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 BOX_H +#define BOX_H + +// Libraries +#include "openglframework.h" +#include "reactphysics3d.h" + +// Structure VertexData +struct VertexData { + + /// Vertex position + openglframework::Vector3 position; + + /// Vertex normal + openglframework::Vector3 normal; + + // Vertex color + openglframework::Color color; +}; + +// Class Box +class Box : public openglframework::Object3D { + + private : + + // -------------------- Attributes -------------------- // + + /// Size of each side of the box + float mSize[3]; + + /// Rigid body used to simulate the dynamics of the box + rp3d::RigidBody* mRigidBody; + + /// Collision shape of the rigid body + rp3d::BoxShape* mCollisionShape; + + /// Scaling matrix (applied to a cube to obtain the correct box dimensions) + openglframework::Matrix4 mScalingMatrix; + + /// Vertex Buffer Object for the vertices data used to render the box with OpenGL + static openglframework::VertexBufferObject mVBOVertices; + + /// Vertex Buffer Object for the indices used to render the box with OpenGL + static openglframework::VertexBufferObject mVBOIndices; + + /// Vertex data for each vertex of the cube (used to render the box) + static VertexData mCubeVertices[8]; + + /// Indices of the cube (used to render the box) + static GLuint mCubeIndices[36]; + + /// True if the VBOs have already been created + static bool areVBOsCreated; + + // -------------------- Methods -------------------- // + + /// Create a Vertex Buffer Object to render to box with OpenGL + static void createVBO(); + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + Box(const openglframework::Vector3& size, const openglframework::Vector3& position, + float mass, rp3d::DynamicsWorld* dynamicsWorld); + + /// Destructor + ~Box(); + + /// Return a pointer to the rigid body of the box + rp3d::RigidBody* getRigidBody(); + + /// Update the transform matrix of the box + void updateTransform(); + + /// Render the cube at the correct position and with the correct orientation + void render(openglframework::Shader& shader); +}; + +// Return a pointer to the rigid body of the box +inline rp3d::RigidBody* Box::getRigidBody() { + return mRigidBody; +} + +#endif diff --git a/examples/common/CMakeLists.txt b/examples/common/CMakeLists.txt new file mode 100644 index 00000000..ba4484e0 --- /dev/null +++ b/examples/common/CMakeLists.txt @@ -0,0 +1,4 @@ +# Minimum cmake version required +CMAKE_MINIMUM_REQUIRED(VERSION 2.6) + +add_subdirectory(opengl-framework/) diff --git a/examples/common/Viewer.cpp b/examples/common/Viewer.cpp new file mode 100644 index 00000000..1f99d4dc --- /dev/null +++ b/examples/common/Viewer.cpp @@ -0,0 +1,83 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Viewer.h" +#include "openglframework.h" +#include + +// Constructor +Viewer::Viewer() : openglframework::GlutViewer(), fps(0), nbFrames(0) { + +} + +// Compute the FPS +void Viewer::computeFPS() { + + nbFrames++; + + // Get the number of milliseconds since glutInit called + currentTime = glutGet(GLUT_ELAPSED_TIME); + + // Calculate time passed + int timeInterval = currentTime - previousTime; + + // Update the FPS counter each second + if(timeInterval > 1000){ + + // calculate the number of frames per second + fps = nbFrames / (timeInterval / 1000.0f); + + // Set time + previousTime = currentTime; + + // Reset frame count + nbFrames = 0; + } +} + +// Display the GUI +void Viewer::displayGUI() { + + // Display the FPS + displayFPS(); +} + +// Display the FPS +void Viewer::displayFPS() { + + glMatrixMode(GL_PROJECTION); + glLoadIdentity(); + glOrtho(0, mCamera.getWidth(), mCamera.getHeight(), 0, -1, 1); + + glMatrixMode(GL_MODELVIEW); + glLoadIdentity(); + + glRasterPos2i(10, 20); + glColor4f(1.0f, 1.0f, 1.0f, 1.0f); + std::stringstream ss; + ss << "FPS : " << fps; + glutBitmapString(GLUT_BITMAP_HELVETICA_12, (const unsigned char*)ss.str().c_str()); +} diff --git a/examples/common/Viewer.h b/examples/common/Viewer.h new file mode 100644 index 00000000..572154fa --- /dev/null +++ b/examples/common/Viewer.h @@ -0,0 +1,71 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 VIEWER_H +#define VIEWER_H + +// Libraries +#include "openglframework.h" + +// Class Viewer +class Viewer : public openglframework::GlutViewer { + + private : + + // -------------------- Attributes -------------------- // + + /// Current number of frames per seconds + int fps; + + /// Number of frames during the last second + int nbFrames; + + /// Current time for fps computation + int currentTime; + + /// Previous time for fps computation + int previousTime; + + // -------------------- Methods -------------------- // + + /// Display the FPS + void displayFPS(); + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + Viewer(); + + /// Compute the FPS + void computeFPS(); + + /// Display the GUI + void displayGUI(); + +}; + +#endif diff --git a/examples/opengl-framework/CMakeLists.txt b/examples/common/opengl-framework/CMakeLists.txt similarity index 100% rename from examples/opengl-framework/CMakeLists.txt rename to examples/common/opengl-framework/CMakeLists.txt diff --git a/examples/opengl-framework/freeglut/CMakeLists.txt b/examples/common/opengl-framework/freeglut/CMakeLists.txt similarity index 100% rename from examples/opengl-framework/freeglut/CMakeLists.txt rename to examples/common/opengl-framework/freeglut/CMakeLists.txt diff --git a/examples/opengl-framework/freeglut/COPYING.txt b/examples/common/opengl-framework/freeglut/COPYING.txt similarity index 100% rename from examples/opengl-framework/freeglut/COPYING.txt rename to examples/common/opengl-framework/freeglut/COPYING.txt diff --git a/examples/opengl-framework/freeglut/GL/freeglut.h b/examples/common/opengl-framework/freeglut/GL/freeglut.h similarity index 100% rename from examples/opengl-framework/freeglut/GL/freeglut.h rename to examples/common/opengl-framework/freeglut/GL/freeglut.h diff --git a/examples/opengl-framework/freeglut/GL/freeglut_ext.h b/examples/common/opengl-framework/freeglut/GL/freeglut_ext.h similarity index 100% rename from examples/opengl-framework/freeglut/GL/freeglut_ext.h rename to examples/common/opengl-framework/freeglut/GL/freeglut_ext.h diff --git a/examples/opengl-framework/freeglut/GL/freeglut_std.h b/examples/common/opengl-framework/freeglut/GL/freeglut_std.h similarity index 100% rename from examples/opengl-framework/freeglut/GL/freeglut_std.h rename to examples/common/opengl-framework/freeglut/GL/freeglut_std.h diff --git a/examples/opengl-framework/freeglut/GL/glut.h b/examples/common/opengl-framework/freeglut/GL/glut.h similarity index 100% rename from examples/opengl-framework/freeglut/GL/glut.h rename to examples/common/opengl-framework/freeglut/GL/glut.h diff --git a/examples/opengl-framework/freeglut/VERSION.txt b/examples/common/opengl-framework/freeglut/VERSION.txt similarity index 100% rename from examples/opengl-framework/freeglut/VERSION.txt rename to examples/common/opengl-framework/freeglut/VERSION.txt diff --git a/examples/opengl-framework/freeglut/freeglut_callbacks.c b/examples/common/opengl-framework/freeglut/freeglut_callbacks.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_callbacks.c rename to examples/common/opengl-framework/freeglut/freeglut_callbacks.c diff --git a/examples/opengl-framework/freeglut/freeglut_cursor.c b/examples/common/opengl-framework/freeglut/freeglut_cursor.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_cursor.c rename to examples/common/opengl-framework/freeglut/freeglut_cursor.c diff --git a/examples/opengl-framework/freeglut/freeglut_display.c b/examples/common/opengl-framework/freeglut/freeglut_display.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_display.c rename to examples/common/opengl-framework/freeglut/freeglut_display.c diff --git a/examples/opengl-framework/freeglut/freeglut_ext.c b/examples/common/opengl-framework/freeglut/freeglut_ext.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_ext.c rename to examples/common/opengl-framework/freeglut/freeglut_ext.c diff --git a/examples/opengl-framework/freeglut/freeglut_font.c b/examples/common/opengl-framework/freeglut/freeglut_font.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_font.c rename to examples/common/opengl-framework/freeglut/freeglut_font.c diff --git a/examples/opengl-framework/freeglut/freeglut_font_data.c b/examples/common/opengl-framework/freeglut/freeglut_font_data.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_font_data.c rename to examples/common/opengl-framework/freeglut/freeglut_font_data.c diff --git a/examples/opengl-framework/freeglut/freeglut_gamemode.c b/examples/common/opengl-framework/freeglut/freeglut_gamemode.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_gamemode.c rename to examples/common/opengl-framework/freeglut/freeglut_gamemode.c diff --git a/examples/opengl-framework/freeglut/freeglut_geometry.c b/examples/common/opengl-framework/freeglut/freeglut_geometry.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_geometry.c rename to examples/common/opengl-framework/freeglut/freeglut_geometry.c diff --git a/examples/opengl-framework/freeglut/freeglut_glutfont_definitions.c b/examples/common/opengl-framework/freeglut/freeglut_glutfont_definitions.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_glutfont_definitions.c rename to examples/common/opengl-framework/freeglut/freeglut_glutfont_definitions.c diff --git a/examples/opengl-framework/freeglut/freeglut_init.c b/examples/common/opengl-framework/freeglut/freeglut_init.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_init.c rename to examples/common/opengl-framework/freeglut/freeglut_init.c diff --git a/examples/opengl-framework/freeglut/freeglut_input_devices.c b/examples/common/opengl-framework/freeglut/freeglut_input_devices.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_input_devices.c rename to examples/common/opengl-framework/freeglut/freeglut_input_devices.c diff --git a/examples/opengl-framework/freeglut/freeglut_internal.h b/examples/common/opengl-framework/freeglut/freeglut_internal.h similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_internal.h rename to examples/common/opengl-framework/freeglut/freeglut_internal.h diff --git a/examples/opengl-framework/freeglut/freeglut_joystick.c b/examples/common/opengl-framework/freeglut/freeglut_joystick.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_joystick.c rename to examples/common/opengl-framework/freeglut/freeglut_joystick.c diff --git a/examples/opengl-framework/freeglut/freeglut_main.c b/examples/common/opengl-framework/freeglut/freeglut_main.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_main.c rename to examples/common/opengl-framework/freeglut/freeglut_main.c diff --git a/examples/opengl-framework/freeglut/freeglut_menu.c b/examples/common/opengl-framework/freeglut/freeglut_menu.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_menu.c rename to examples/common/opengl-framework/freeglut/freeglut_menu.c diff --git a/examples/opengl-framework/freeglut/freeglut_misc.c b/examples/common/opengl-framework/freeglut/freeglut_misc.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_misc.c rename to examples/common/opengl-framework/freeglut/freeglut_misc.c diff --git a/examples/opengl-framework/freeglut/freeglut_overlay.c b/examples/common/opengl-framework/freeglut/freeglut_overlay.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_overlay.c rename to examples/common/opengl-framework/freeglut/freeglut_overlay.c diff --git a/examples/opengl-framework/freeglut/freeglut_spaceball.c b/examples/common/opengl-framework/freeglut/freeglut_spaceball.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_spaceball.c rename to examples/common/opengl-framework/freeglut/freeglut_spaceball.c diff --git a/examples/opengl-framework/freeglut/freeglut_state.c b/examples/common/opengl-framework/freeglut/freeglut_state.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_state.c rename to examples/common/opengl-framework/freeglut/freeglut_state.c diff --git a/examples/opengl-framework/freeglut/freeglut_stroke_mono_roman.c b/examples/common/opengl-framework/freeglut/freeglut_stroke_mono_roman.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_stroke_mono_roman.c rename to examples/common/opengl-framework/freeglut/freeglut_stroke_mono_roman.c diff --git a/examples/opengl-framework/freeglut/freeglut_stroke_roman.c b/examples/common/opengl-framework/freeglut/freeglut_stroke_roman.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_stroke_roman.c rename to examples/common/opengl-framework/freeglut/freeglut_stroke_roman.c diff --git a/examples/opengl-framework/freeglut/freeglut_structure.c b/examples/common/opengl-framework/freeglut/freeglut_structure.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_structure.c rename to examples/common/opengl-framework/freeglut/freeglut_structure.c diff --git a/examples/opengl-framework/freeglut/freeglut_teapot.c b/examples/common/opengl-framework/freeglut/freeglut_teapot.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_teapot.c rename to examples/common/opengl-framework/freeglut/freeglut_teapot.c diff --git a/examples/opengl-framework/freeglut/freeglut_teapot_data.h b/examples/common/opengl-framework/freeglut/freeglut_teapot_data.h similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_teapot_data.h rename to examples/common/opengl-framework/freeglut/freeglut_teapot_data.h diff --git a/examples/opengl-framework/freeglut/freeglut_videoresize.c b/examples/common/opengl-framework/freeglut/freeglut_videoresize.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_videoresize.c rename to examples/common/opengl-framework/freeglut/freeglut_videoresize.c diff --git a/examples/opengl-framework/freeglut/freeglut_window.c b/examples/common/opengl-framework/freeglut/freeglut_window.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_window.c rename to examples/common/opengl-framework/freeglut/freeglut_window.c diff --git a/examples/opengl-framework/freeglut/freeglut_xinput.c b/examples/common/opengl-framework/freeglut/freeglut_xinput.c similarity index 100% rename from examples/opengl-framework/freeglut/freeglut_xinput.c rename to examples/common/opengl-framework/freeglut/freeglut_xinput.c diff --git a/examples/opengl-framework/freeglut/glut.h b/examples/common/opengl-framework/freeglut/glut.h similarity index 100% rename from examples/opengl-framework/freeglut/glut.h rename to examples/common/opengl-framework/freeglut/glut.h diff --git a/examples/opengl-framework/src/Camera.cpp b/examples/common/opengl-framework/src/Camera.cpp similarity index 100% rename from examples/opengl-framework/src/Camera.cpp rename to examples/common/opengl-framework/src/Camera.cpp diff --git a/examples/opengl-framework/src/Camera.h b/examples/common/opengl-framework/src/Camera.h similarity index 100% rename from examples/opengl-framework/src/Camera.h rename to examples/common/opengl-framework/src/Camera.h diff --git a/examples/opengl-framework/src/FrameBufferObject.cpp b/examples/common/opengl-framework/src/FrameBufferObject.cpp similarity index 100% rename from examples/opengl-framework/src/FrameBufferObject.cpp rename to examples/common/opengl-framework/src/FrameBufferObject.cpp diff --git a/examples/opengl-framework/src/FrameBufferObject.h b/examples/common/opengl-framework/src/FrameBufferObject.h similarity index 100% rename from examples/opengl-framework/src/FrameBufferObject.h rename to examples/common/opengl-framework/src/FrameBufferObject.h diff --git a/examples/opengl-framework/src/GlutViewer.cpp b/examples/common/opengl-framework/src/GlutViewer.cpp similarity index 97% rename from examples/opengl-framework/src/GlutViewer.cpp rename to examples/common/opengl-framework/src/GlutViewer.cpp index d0dc9bed..45cf639d 100644 --- a/examples/opengl-framework/src/GlutViewer.cpp +++ b/examples/common/opengl-framework/src/GlutViewer.cpp @@ -168,6 +168,16 @@ void GlutViewer::mouseMotionEvent(int xMouse, int yMouse) { glutPostRedisplay(); } +// Called when a GLUT keyboard event occurs +void GlutViewer::keyboardEvent(int key, int xMouse, int yMouse) { + +} + +// Called when a GLUT special keyboard event occurs +void GlutViewer::keyboardSpecialEvent(int key, int xMouse, int yMouse) { + +} + // Map the mouse x,y coordinates to a point on a sphere bool GlutViewer::mapMouseCoordinatesToSphere(int xMouse, int yMouse, Vector3& spherePoint) const { diff --git a/examples/opengl-framework/src/GlutViewer.h b/examples/common/opengl-framework/src/GlutViewer.h similarity index 95% rename from examples/opengl-framework/src/GlutViewer.h rename to examples/common/opengl-framework/src/GlutViewer.h index b5916179..7718a1af 100644 --- a/examples/opengl-framework/src/GlutViewer.h +++ b/examples/common/opengl-framework/src/GlutViewer.h @@ -39,7 +39,7 @@ namespace openglframework { // Class Renderer class GlutViewer { - private: + protected : // -------------------- Attributes -------------------- // @@ -81,7 +81,7 @@ class GlutViewer { GlutViewer(); // Destructor - ~GlutViewer(); + virtual ~GlutViewer(); // Initialize the viewer bool init(int argc, char** argv, const std::string& windowsTitle, @@ -112,16 +112,17 @@ class GlutViewer { // Get the camera Camera& getCamera(); - void motion(int x, int y); - // Called when a GLUT mouse button event occurs void mouseButtonEvent(int button, int state, int x, int y); // Called when a GLUT mouse motion event occurs void mouseMotionEvent(int xMouse, int yMouse); - void keyboard(int key, int x, int y); - void special(int key, int x, int y); + // Called when a GLUT keyboard event occurs + void keyboardEvent(int key, int xMouse, int yMouse); + + // Called when a GLUT special keyboard event occurs + void keyboardSpecialEvent(int key, int xMouse, int yMouse); // Check the OpenGL errors static void checkOpenGLErrors(); diff --git a/examples/opengl-framework/src/Light.cpp b/examples/common/opengl-framework/src/Light.cpp similarity index 100% rename from examples/opengl-framework/src/Light.cpp rename to examples/common/opengl-framework/src/Light.cpp diff --git a/examples/opengl-framework/src/Light.h b/examples/common/opengl-framework/src/Light.h similarity index 100% rename from examples/opengl-framework/src/Light.h rename to examples/common/opengl-framework/src/Light.h diff --git a/examples/opengl-framework/src/Mesh.cpp b/examples/common/opengl-framework/src/Mesh.cpp similarity index 100% rename from examples/opengl-framework/src/Mesh.cpp rename to examples/common/opengl-framework/src/Mesh.cpp diff --git a/examples/opengl-framework/src/Mesh.h b/examples/common/opengl-framework/src/Mesh.h similarity index 100% rename from examples/opengl-framework/src/Mesh.h rename to examples/common/opengl-framework/src/Mesh.h diff --git a/examples/opengl-framework/src/MeshReaderWriter.cpp b/examples/common/opengl-framework/src/MeshReaderWriter.cpp similarity index 100% rename from examples/opengl-framework/src/MeshReaderWriter.cpp rename to examples/common/opengl-framework/src/MeshReaderWriter.cpp diff --git a/examples/opengl-framework/src/MeshReaderWriter.h b/examples/common/opengl-framework/src/MeshReaderWriter.h similarity index 100% rename from examples/opengl-framework/src/MeshReaderWriter.h rename to examples/common/opengl-framework/src/MeshReaderWriter.h diff --git a/examples/opengl-framework/src/Object3D.cpp b/examples/common/opengl-framework/src/Object3D.cpp similarity index 100% rename from examples/opengl-framework/src/Object3D.cpp rename to examples/common/opengl-framework/src/Object3D.cpp diff --git a/examples/opengl-framework/src/Object3D.h b/examples/common/opengl-framework/src/Object3D.h similarity index 100% rename from examples/opengl-framework/src/Object3D.h rename to examples/common/opengl-framework/src/Object3D.h diff --git a/examples/opengl-framework/src/Shader.cpp b/examples/common/opengl-framework/src/Shader.cpp similarity index 100% rename from examples/opengl-framework/src/Shader.cpp rename to examples/common/opengl-framework/src/Shader.cpp diff --git a/examples/opengl-framework/src/Shader.h b/examples/common/opengl-framework/src/Shader.h similarity index 100% rename from examples/opengl-framework/src/Shader.h rename to examples/common/opengl-framework/src/Shader.h diff --git a/examples/opengl-framework/src/Texture2D.cpp b/examples/common/opengl-framework/src/Texture2D.cpp similarity index 100% rename from examples/opengl-framework/src/Texture2D.cpp rename to examples/common/opengl-framework/src/Texture2D.cpp diff --git a/examples/opengl-framework/src/Texture2D.h b/examples/common/opengl-framework/src/Texture2D.h similarity index 100% rename from examples/opengl-framework/src/Texture2D.h rename to examples/common/opengl-framework/src/Texture2D.h diff --git a/examples/opengl-framework/src/TextureReaderWriter.cpp b/examples/common/opengl-framework/src/TextureReaderWriter.cpp similarity index 100% rename from examples/opengl-framework/src/TextureReaderWriter.cpp rename to examples/common/opengl-framework/src/TextureReaderWriter.cpp diff --git a/examples/opengl-framework/src/TextureReaderWriter.h b/examples/common/opengl-framework/src/TextureReaderWriter.h similarity index 100% rename from examples/opengl-framework/src/TextureReaderWriter.h rename to examples/common/opengl-framework/src/TextureReaderWriter.h diff --git a/examples/opengl-framework/src/VertexBufferObject.cpp b/examples/common/opengl-framework/src/VertexBufferObject.cpp similarity index 100% rename from examples/opengl-framework/src/VertexBufferObject.cpp rename to examples/common/opengl-framework/src/VertexBufferObject.cpp diff --git a/examples/opengl-framework/src/VertexBufferObject.h b/examples/common/opengl-framework/src/VertexBufferObject.h similarity index 100% rename from examples/opengl-framework/src/VertexBufferObject.h rename to examples/common/opengl-framework/src/VertexBufferObject.h diff --git a/examples/opengl-framework/src/definitions.h b/examples/common/opengl-framework/src/definitions.h similarity index 100% rename from examples/opengl-framework/src/definitions.h rename to examples/common/opengl-framework/src/definitions.h diff --git a/examples/opengl-framework/src/maths/Color.h b/examples/common/opengl-framework/src/maths/Color.h similarity index 100% rename from examples/opengl-framework/src/maths/Color.h rename to examples/common/opengl-framework/src/maths/Color.h diff --git a/examples/opengl-framework/src/maths/Matrix3.h b/examples/common/opengl-framework/src/maths/Matrix3.h similarity index 100% rename from examples/opengl-framework/src/maths/Matrix3.h rename to examples/common/opengl-framework/src/maths/Matrix3.h diff --git a/examples/opengl-framework/src/maths/Matrix4.h b/examples/common/opengl-framework/src/maths/Matrix4.h similarity index 100% rename from examples/opengl-framework/src/maths/Matrix4.h rename to examples/common/opengl-framework/src/maths/Matrix4.h diff --git a/examples/opengl-framework/src/maths/Vector2.h b/examples/common/opengl-framework/src/maths/Vector2.h similarity index 100% rename from examples/opengl-framework/src/maths/Vector2.h rename to examples/common/opengl-framework/src/maths/Vector2.h diff --git a/examples/opengl-framework/src/maths/Vector3.h b/examples/common/opengl-framework/src/maths/Vector3.h similarity index 100% rename from examples/opengl-framework/src/maths/Vector3.h rename to examples/common/opengl-framework/src/maths/Vector3.h diff --git a/examples/opengl-framework/src/maths/Vector4.h b/examples/common/opengl-framework/src/maths/Vector4.h similarity index 100% rename from examples/opengl-framework/src/maths/Vector4.h rename to examples/common/opengl-framework/src/maths/Vector4.h diff --git a/examples/opengl-framework/src/openglframework.h b/examples/common/opengl-framework/src/openglframework.h similarity index 100% rename from examples/opengl-framework/src/openglframework.h rename to examples/common/opengl-framework/src/openglframework.h diff --git a/examples/opengl-framework/src/shaders/depth.frag b/examples/common/opengl-framework/src/shaders/depth.frag similarity index 100% rename from examples/opengl-framework/src/shaders/depth.frag rename to examples/common/opengl-framework/src/shaders/depth.frag diff --git a/examples/opengl-framework/src/shaders/depth.vert b/examples/common/opengl-framework/src/shaders/depth.vert similarity index 100% rename from examples/opengl-framework/src/shaders/depth.vert rename to examples/common/opengl-framework/src/shaders/depth.vert diff --git a/examples/opengl-framework/src/shaders/phong.frag b/examples/common/opengl-framework/src/shaders/phong.frag similarity index 100% rename from examples/opengl-framework/src/shaders/phong.frag rename to examples/common/opengl-framework/src/shaders/phong.frag diff --git a/examples/opengl-framework/src/shaders/phong.vert b/examples/common/opengl-framework/src/shaders/phong.vert similarity index 100% rename from examples/opengl-framework/src/shaders/phong.vert rename to examples/common/opengl-framework/src/shaders/phong.vert diff --git a/examples/fallingcubes/CMakeLists.txt b/examples/fallingcubes/CMakeLists.txt index 4dcb291f..a7b5ee34 100644 --- a/examples/fallingcubes/CMakeLists.txt +++ b/examples/fallingcubes/CMakeLists.txt @@ -5,13 +5,13 @@ cmake_minimum_required(VERSION 2.6) PROJECT(FallingCubes) # Copy the shaders used for the demo into the build directory -FILE(COPY "../opengl-framework/src/shaders/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/shaders/") +FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/shaders/") # Headers -INCLUDE_DIRECTORIES("../opengl-framework/src/") +INCLUDE_DIRECTORIES("../common/opengl-framework/src/" "../common/") # Create the example executable using the # compiled reactphysics3d static library -ADD_EXECUTABLE(fallingcubes main.cpp Scene.cpp Scene.h Box.cpp Box.h) +ADD_EXECUTABLE(fallingcubes FallingCubes.cpp Scene.cpp Scene.h "../common/Box.cpp" "../common/Box.h" "../common/Viewer.cpp" "../common/Viewer.h") TARGET_LINK_LIBRARIES(fallingcubes reactphysics3d openglframework) diff --git a/examples/fallingcubes/main.cpp b/examples/fallingcubes/FallingCubes.cpp similarity index 72% rename from examples/fallingcubes/main.cpp rename to examples/fallingcubes/FallingCubes.cpp index 6dbfc7f4..65f8f485 100644 --- a/examples/fallingcubes/main.cpp +++ b/examples/fallingcubes/FallingCubes.cpp @@ -25,40 +25,32 @@ // Libraries #include "Scene.h" -#include +#include "Viewer.h" // Declarations void simulate(); void display(); -void displayFPS(); -void computeFPS(); +void finish(); void reshape(int width, int height); void mouseButton(int button, int state, int x, int y); void mouseMotion(int x, int y); -void keyboardSpecial(int key, int x, int y); +void keyboard(unsigned char key, int x, int y); void init(); // Namespaces using namespace openglframework; // Global variables -GlutViewer* viewer; +Viewer* viewer; Scene* scene; -int fps; -int nbFrames; -int currentTime; -int previousTime; -int width, height; // Main function int main(int argc, char** argv) { // Create and initialize the Viewer - viewer = new GlutViewer(); + viewer = new Viewer(); Vector2 windowsSize = Vector2(800, 600); Vector2 windowsPosition = Vector2(100, 100); - width = windowsSize.x; - height = windowsSize.y; bool initOK = viewer->init(argc, argv, "ReactPhysics3D Examples - Falling Cubes", windowsSize, windowsPosition); if (!initOK) return 1; @@ -67,21 +59,18 @@ int main(int argc, char** argv) { init(); - nbFrames = 0; - // Glut Idle function that is continuously called glutIdleFunc(simulate); glutDisplayFunc(display); glutReshapeFunc(reshape); glutMouseFunc(mouseButton); glutMotionFunc(mouseMotion); - glutSpecialFunc(keyboardSpecial); + glutKeyboardFunc(keyboard); // Glut main looop glutMainLoop(); - delete viewer; - delete scene; + finish(); return 0; } @@ -92,7 +81,7 @@ void simulate() { // Physics simulation scene->simulate(); - computeFPS(); + viewer->computeFPS(); // Ask GLUT to render the scene glutPostRedisplay (); @@ -108,8 +97,6 @@ void init() { // Reshape function void reshape(int newWidth, int newHeight) { viewer->reshape(newWidth, newHeight); - width = newWidth; - height = newHeight; } // Called when a mouse button event occurs @@ -123,12 +110,28 @@ void mouseMotion(int x, int y) { } // Called when the user hits a special key on the keyboard -void keyboardSpecial(int key, int x, int y) { - /* - if(key=='0') +void keyboard(unsigned char key, int x, int y) { + switch(key) { + + // Escape key + case 27: + finish(); exit(0); - if(key== GLUT_KEY_RIGHT) { - */ + break; + + // Space bar + case 32: + scene->pauseContinueSimulation(); + break; + } +} + +// End of the application +void finish() { + + // Destroy the viewer and the scene + delete viewer; + delete scene; } // Display the scene @@ -138,7 +141,7 @@ void display() { scene->render(); // Display the FPS - displayFPS(); + viewer->displayGUI(); // Swap the buffers glutSwapBuffers(); @@ -147,43 +150,4 @@ void display() { GlutViewer::checkOpenGLErrors(); } -// Compute the FPS -void computeFPS() { - nbFrames++; - // Get the number of milliseconds since glutInit called - currentTime = glutGet(GLUT_ELAPSED_TIME); - - // Calculate time passed - int timeInterval = currentTime - previousTime; - - // Update the FPS counter each second - if(timeInterval > 1000){ - - // calculate the number of frames per second - fps = nbFrames / (timeInterval / 1000.0f); - - // Set time - previousTime = currentTime; - - // Reset frame count - nbFrames = 0; - } -} - -// Display the FPS -void displayFPS() { - - glMatrixMode(GL_PROJECTION); - glLoadIdentity(); - glOrtho(0, width, height, 0, -1, 1); - - glMatrixMode(GL_MODELVIEW); - glLoadIdentity(); - - glRasterPos2i(10, 20); - glColor4f(1.0f, 1.0f, 1.0f, 1.0f); - std::stringstream ss; - ss << "FPS : " << fps; - glutBitmapString(GLUT_BITMAP_HELVETICA_12, (const unsigned char*)ss.str().c_str()); -} diff --git a/examples/fallingcubes/Scene.cpp b/examples/fallingcubes/Scene.cpp index 71a84821..693a0fa5 100644 --- a/examples/fallingcubes/Scene.cpp +++ b/examples/fallingcubes/Scene.cpp @@ -32,32 +32,40 @@ using namespace openglframework; // Constructor Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), mPhongShader("shaders/phong.vert", - "shaders/phong.frag"){ + "shaders/phong.frag"), mIsRunning(false) { // Move the light 0 mLight0.translateWorld(Vector3(7, 15, 15)); // Compute the radius and the center of the scene - float radius = 10.0f; + float radiusScene = 10.0f; openglframework::Vector3 center(0, 5, 0); // Set the center of the scene - mViewer->setScenePosition(center, radius); + mViewer->setScenePosition(center, radiusScene); // Gravity vector in the dynamics world rp3d::Vector3 gravity(0, -9.81, 0); // Time step for the physics simulation - rp3d::decimal timeStep = 1.0f / 80.0f; + rp3d::decimal timeStep = 1.0f / 60.0f; // Create the dynamics world for the physics simulation mDynamicsWorld = new rp3d::DynamicsWorld(gravity, timeStep); + // Set the number of iterations of the constraint solver + mDynamicsWorld->setNbIterationsSolver(15); + + float radius = 2.0f; + // Create all the cubes of the scene for (int i=0; igetRigidBody()->setRestitution(0.3); // Start the simulation - mDynamicsWorld->start(); + startSimulation(); } // Destructor Scene::~Scene() { // Stop the physics simulation - mDynamicsWorld->stop(); + stopSimulation(); // Destroy the shader mPhongShader.destroy(); @@ -118,17 +126,22 @@ Scene::~Scene() { // Take a step for the simulation void Scene::simulate() { - // Take a simulation step - mDynamicsWorld->update(); + // If the physics simulation is running + if (mIsRunning) { - // Update the position and orientation of the boxes - for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + // Take a simulation step + mDynamicsWorld->update(); + + // Update the position and orientation of the boxes + for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + + // Update the transform used for the rendering + (*it)->updateTransform(); + } + + mFloor->updateTransform(); - // Update the transform used for the rendering - (*it)->updateTransform(); } - - mFloor->updateTransform(); } // Render the scene @@ -136,7 +149,7 @@ void Scene::render() { glEnable(GL_DEPTH_TEST); glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); - glDisable(GL_CULL_FACE); + glEnable(GL_CULL_FACE); // Bind the shader mPhongShader.bind(); diff --git a/examples/fallingcubes/Scene.h b/examples/fallingcubes/Scene.h index a865d8e0..3741962e 100644 --- a/examples/fallingcubes/Scene.h +++ b/examples/fallingcubes/Scene.h @@ -32,7 +32,7 @@ #include "Box.h" // Constants -const int NB_BOXES = 10; // Number of boxes in the scene +const int NB_BOXES = 20; // Number of boxes in the scene const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters const float BOX_MASS = 1.0f; // Box mass in kilograms @@ -63,6 +63,9 @@ class Scene { /// Dynamics world used for the physics simulation rp3d::DynamicsWorld* mDynamicsWorld; + /// True if the physics simulation is running + bool mIsRunning; + public: // -------------------- Methods -------------------- // @@ -76,8 +79,39 @@ class Scene { /// Take a step for the simulation void simulate(); + /// Stop the simulation + void stopSimulation(); + + /// Start the simulation + void startSimulation(); + + /// Pause or continue simulation + void pauseContinueSimulation(); + /// Render the scene void render(); }; +// Stop the simulation +inline void Scene::stopSimulation() { + mDynamicsWorld->stop(); + mIsRunning = false; +} + +// Start the simulation +inline void Scene::startSimulation() { + mDynamicsWorld->start(); + mIsRunning = true; +} + +// Pause or continue simulation +inline void Scene::pauseContinueSimulation() { + if (mIsRunning) { + stopSimulation(); + } + else { + startSimulation(); + } +} + #endif diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index fb66aa7b..1519b369 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -215,7 +215,6 @@ inline void DynamicsWorld::start() { } inline void DynamicsWorld::stop() { - std::cout << "Stop Simulation" << std::endl; mTimer.stop(); } From 5d205c2d7f03137816c7e4b6506c1e01adb479e1 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 18 Apr 2013 22:28:14 +0200 Subject: [PATCH 08/66] Commit documentation modifications --- documentation/API/ReactPhysics3DLogo.png | Bin .../UserManual/ReactPhysics3D-UserManual.tex | 0 .../UserManual/images/ReactPhysics3DLogo.png | Bin documentation/UserManual/title.tex | 0 examples/fallingcubes/CMakeLists.txt | 0 5 files changed, 0 insertions(+), 0 deletions(-) mode change 100755 => 100644 documentation/API/ReactPhysics3DLogo.png mode change 100755 => 100644 documentation/UserManual/ReactPhysics3D-UserManual.tex mode change 100755 => 100644 documentation/UserManual/images/ReactPhysics3DLogo.png mode change 100755 => 100644 documentation/UserManual/title.tex mode change 100755 => 100644 examples/fallingcubes/CMakeLists.txt diff --git a/documentation/API/ReactPhysics3DLogo.png b/documentation/API/ReactPhysics3DLogo.png old mode 100755 new mode 100644 diff --git a/documentation/UserManual/ReactPhysics3D-UserManual.tex b/documentation/UserManual/ReactPhysics3D-UserManual.tex old mode 100755 new mode 100644 diff --git a/documentation/UserManual/images/ReactPhysics3DLogo.png b/documentation/UserManual/images/ReactPhysics3DLogo.png old mode 100755 new mode 100644 diff --git a/documentation/UserManual/title.tex b/documentation/UserManual/title.tex old mode 100755 new mode 100644 diff --git a/examples/fallingcubes/CMakeLists.txt b/examples/fallingcubes/CMakeLists.txt old mode 100755 new mode 100644 From 16c64877965d6fb38979efeeef3ba2fd27a455eb Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 18 Apr 2013 22:54:36 +0200 Subject: [PATCH 09/66] Change #ifndef names in headers to avoid name collision with others libraries --- src/body/Body.h | 4 ++-- src/body/CollisionBody.h | 4 ++-- src/body/RigidBody.h | 4 ++-- src/collision/BroadPhasePair.h | 4 ++-- src/collision/CollisionDetection.h | 4 ++-- src/collision/ContactInfo.h | 4 ++-- src/collision/broadphase/BroadPhaseAlgorithm.h | 4 ++-- src/collision/broadphase/NoBroadPhaseAlgorithm.h | 4 ++-- src/collision/broadphase/PairManager.h | 4 ++-- src/collision/broadphase/SweepAndPruneAlgorithm.h | 4 ++-- src/collision/narrowphase/EPA/EPAAlgorithm.h | 4 ++-- src/collision/narrowphase/EPA/EdgeEPA.h | 4 ++-- src/collision/narrowphase/EPA/TriangleEPA.h | 4 ++-- src/collision/narrowphase/EPA/TrianglesStore.h | 4 ++-- src/collision/narrowphase/GJK/GJKAlgorithm.h | 4 ++-- src/collision/narrowphase/GJK/Simplex.h | 4 ++-- src/collision/narrowphase/NarrowPhaseAlgorithm.h | 4 ++-- src/collision/narrowphase/SphereVsSphereAlgorithm.h | 4 ++-- src/collision/shapes/AABB.h | 4 ++-- src/collision/shapes/BoxShape.h | 4 ++-- src/collision/shapes/CollisionShape.h | 4 ++-- src/collision/shapes/ConeShape.h | 4 ++-- src/collision/shapes/CylinderShape.h | 4 ++-- src/collision/shapes/SphereShape.h | 4 ++-- src/configuration.h | 4 ++-- src/constraint/Constraint.h | 4 ++-- src/constraint/ContactPoint.h | 4 ++-- src/decimal.h | 4 ++-- src/engine/CollisionWorld.h | 4 ++-- src/engine/ContactManifold.h | 4 ++-- src/engine/ContactSolver.h | 4 ++-- src/engine/DynamicsWorld.h | 4 ++-- src/engine/OverlappingPair.h | 4 ++-- src/engine/Profiler.h | 4 ++-- src/engine/Timer.h | 4 ++-- src/mathematics/Matrix3x3.h | 4 ++-- src/mathematics/Quaternion.h | 4 ++-- src/mathematics/Transform.h | 4 ++-- src/mathematics/Vector3.h | 4 ++-- src/mathematics/mathematics.h | 4 ++-- src/mathematics/mathematics_functions.h | 4 ++-- src/memory/MemoryAllocator.h | 4 ++-- 42 files changed, 84 insertions(+), 84 deletions(-) diff --git a/src/body/Body.h b/src/body/Body.h index 6aa9a2eb..d131eec8 100644 --- a/src/body/Body.h +++ b/src/body/Body.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef BODY_H -#define BODY_H +#ifndef REACTPHYSICS3D_BODY_H +#define REACTPHYSICS3D_BODY_H // Libraries #include diff --git a/src/body/CollisionBody.h b/src/body/CollisionBody.h index 57659766..c119925e 100644 --- a/src/body/CollisionBody.h +++ b/src/body/CollisionBody.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef COLLISION_BODY_H -#define COLLISION_BODY_H +#ifndef REACTPHYSICS3D_COLLISION_BODY_H +#define REACTPHYSICS3D_COLLISION_BODY_H // Libraries #include diff --git a/src/body/RigidBody.h b/src/body/RigidBody.h index 0d8652e5..0128bb42 100644 --- a/src/body/RigidBody.h +++ b/src/body/RigidBody.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef RIGID_BODY_H -#define RIGID_BODY_H +#ifndef REACTPHYSICS3D_RIGID_BODY_H +#define REACTPHYSICS3D_RIGID_BODY_H // Libraries #include diff --git a/src/collision/BroadPhasePair.h b/src/collision/BroadPhasePair.h index c3878eda..f0a2f9ca 100644 --- a/src/collision/BroadPhasePair.h +++ b/src/collision/BroadPhasePair.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef BROAD_PHASE_PAIR_H -#define BROAD_PHASE_PAIR_H +#ifndef REACTPHYSICS3D_BROAD_PHASE_PAIR_H +#define REACTPHYSICS3D_BROAD_PHASE_PAIR_H // Libraries #include "../body/CollisionBody.h" diff --git a/src/collision/CollisionDetection.h b/src/collision/CollisionDetection.h index 71dda4bf..897103f9 100644 --- a/src/collision/CollisionDetection.h +++ b/src/collision/CollisionDetection.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef COLLISION_DETECTION_H -#define COLLISION_DETECTION_H +#ifndef REACTPHYSICS3D_COLLISION_DETECTION_H +#define REACTPHYSICS3D_COLLISION_DETECTION_H // Libraries #include "../body/CollisionBody.h" diff --git a/src/collision/ContactInfo.h b/src/collision/ContactInfo.h index c68479d9..a9d8245a 100644 --- a/src/collision/ContactInfo.h +++ b/src/collision/ContactInfo.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef CONTACT_INFO_H -#define CONTACT_INFO_H +#ifndef REACTPHYSICS3D_CONTACT_INFO_H +#define REACTPHYSICS3D_CONTACT_INFO_H // Libraries #include "../collision/shapes/BoxShape.h" diff --git a/src/collision/broadphase/BroadPhaseAlgorithm.h b/src/collision/broadphase/BroadPhaseAlgorithm.h index dfc78c75..79536b05 100644 --- a/src/collision/broadphase/BroadPhaseAlgorithm.h +++ b/src/collision/broadphase/BroadPhaseAlgorithm.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef BROAD_PHASE_ALGORITHM_H -#define BROAD_PHASE_ALGORITHM_H +#ifndef REACTPHYSICS3D_BROAD_PHASE_ALGORITHM_H +#define REACTPHYSICS3D_BROAD_PHASE_ALGORITHM_H // Libraries #include diff --git a/src/collision/broadphase/NoBroadPhaseAlgorithm.h b/src/collision/broadphase/NoBroadPhaseAlgorithm.h index 12a89d85..9c4ea7ca 100644 --- a/src/collision/broadphase/NoBroadPhaseAlgorithm.h +++ b/src/collision/broadphase/NoBroadPhaseAlgorithm.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef NO_BROAD_PHASE_ALGORITHM_H -#define NO_BROAD_PHASE_ALGORITHM_H +#ifndef REACTPHYSICS3D_NO_BROAD_PHASE_ALGORITHM_H +#define REACTPHYSICS3D_NO_BROAD_PHASE_ALGORITHM_H // Libraries #include "BroadPhaseAlgorithm.h" diff --git a/src/collision/broadphase/PairManager.h b/src/collision/broadphase/PairManager.h index 130b1dc6..860ab19a 100644 --- a/src/collision/broadphase/PairManager.h +++ b/src/collision/broadphase/PairManager.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef PAIR_MANAGER_H -#define PAIR_MANAGER_H +#ifndef REACTPHYSICS3D_PAIR_MANAGER_H +#define REACTPHYSICS3D_PAIR_MANAGER_H // Libraries #include "../../body/CollisionBody.h" diff --git a/src/collision/broadphase/SweepAndPruneAlgorithm.h b/src/collision/broadphase/SweepAndPruneAlgorithm.h index e3c9c537..25051509 100644 --- a/src/collision/broadphase/SweepAndPruneAlgorithm.h +++ b/src/collision/broadphase/SweepAndPruneAlgorithm.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef SWEEP_AND_PRUNE_ALGORITHM_H -#define SWEEP_AND_PRUNE_ALGORITHM_H +#ifndef REACTPHYSICS3D_SWEEP_AND_PRUNE_ALGORITHM_H +#define REACTPHYSICS3D_SWEEP_AND_PRUNE_ALGORITHM_H // Libraries #include "BroadPhaseAlgorithm.h" diff --git a/src/collision/narrowphase/EPA/EPAAlgorithm.h b/src/collision/narrowphase/EPA/EPAAlgorithm.h index c9edc3b9..5b11a2bf 100644 --- a/src/collision/narrowphase/EPA/EPAAlgorithm.h +++ b/src/collision/narrowphase/EPA/EPAAlgorithm.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef EPA_ALGORITHM_H -#define EPA_ALGORITHM_H +#ifndef REACTPHYSICS3D_EPA_ALGORITHM_H +#define REACTPHYSICS3D_EPA_ALGORITHM_H // Libraries #include "../GJK/Simplex.h" diff --git a/src/collision/narrowphase/EPA/EdgeEPA.h b/src/collision/narrowphase/EPA/EdgeEPA.h index c3c7ec01..ef6e9873 100644 --- a/src/collision/narrowphase/EPA/EdgeEPA.h +++ b/src/collision/narrowphase/EPA/EdgeEPA.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef EDGE_EPA_H -#define EDGE_EPA_H +#ifndef REACTPHYSICS3D_EDGE_EPA_H +#define REACTPHYSICS3D_EDGE_EPA_H // Libraries diff --git a/src/collision/narrowphase/EPA/TriangleEPA.h b/src/collision/narrowphase/EPA/TriangleEPA.h index 1d4a195c..4e83c2d6 100644 --- a/src/collision/narrowphase/EPA/TriangleEPA.h +++ b/src/collision/narrowphase/EPA/TriangleEPA.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef TRIANGLE_EPA_H -#define TRIANGLE_EPA_H +#ifndef REACTPHYSICS3D_TRIANGLE_EPA_H +#define REACTPHYSICS3D_TRIANGLE_EPA_H // Libraries #include "../../../mathematics/mathematics.h" diff --git a/src/collision/narrowphase/EPA/TrianglesStore.h b/src/collision/narrowphase/EPA/TrianglesStore.h index f0ab8aab..bd273b94 100644 --- a/src/collision/narrowphase/EPA/TrianglesStore.h +++ b/src/collision/narrowphase/EPA/TrianglesStore.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef TRIANGLES_STORE_H -#define TRIANGLES_STORE_H +#ifndef REACTPHYSICS3D_TRIANGLES_STORE_H +#define REACTPHYSICS3D_TRIANGLES_STORE_H #include "TriangleEPA.h" diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.h b/src/collision/narrowphase/GJK/GJKAlgorithm.h index fc4af52e..5e387b68 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.h +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef GJK_ALGORITHM_H -#define GJK_ALGORITHM_H +#ifndef REACTPHYSICS3D_GJK_ALGORITHM_H +#define REACTPHYSICS3D_GJK_ALGORITHM_H // Libraries #include "../NarrowPhaseAlgorithm.h" diff --git a/src/collision/narrowphase/GJK/Simplex.h b/src/collision/narrowphase/GJK/Simplex.h index a8db761a..ab1a56c9 100644 --- a/src/collision/narrowphase/GJK/Simplex.h +++ b/src/collision/narrowphase/GJK/Simplex.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef SIMPLEX_H -#define SIMPLEX_H +#ifndef REACTPHYSICS3D_SIMPLEX_H +#define REACTPHYSICS3D_SIMPLEX_H // Libraries #include "../../../mathematics/mathematics.h" diff --git a/src/collision/narrowphase/NarrowPhaseAlgorithm.h b/src/collision/narrowphase/NarrowPhaseAlgorithm.h index d68f3f28..998583a3 100644 --- a/src/collision/narrowphase/NarrowPhaseAlgorithm.h +++ b/src/collision/narrowphase/NarrowPhaseAlgorithm.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef NARROW_PHASE_ALGORITHM_H -#define NARROW_PHASE_ALGORITHM_H +#ifndef REACTPHYSICS3D_NARROW_PHASE_ALGORITHM_H +#define REACTPHYSICS3D_NARROW_PHASE_ALGORITHM_H // Libraries #include "../../body/Body.h" diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.h b/src/collision/narrowphase/SphereVsSphereAlgorithm.h index 1c50fb1f..e9320619 100644 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.h +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef SPHERE_VS_SPHERE_ALGORITHM_H -#define SPHERE_VS_SPHERE_ALGORITHM_H +#ifndef REACTPHYSICS3D_SPHERE_VS_SPHERE_ALGORITHM_H +#define REACTPHYSICS3D_SPHERE_VS_SPHERE_ALGORITHM_H // Libraries #include "../../body/Body.h" diff --git a/src/collision/shapes/AABB.h b/src/collision/shapes/AABB.h index 037fa2d4..e92e43e1 100644 --- a/src/collision/shapes/AABB.h +++ b/src/collision/shapes/AABB.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef AABB_H -#define AABB_H +#ifndef REACTPHYSICS3D_AABB_H +#define REACTPHYSICS3D_AABB_H // Libraries #include "../../mathematics/mathematics.h" diff --git a/src/collision/shapes/BoxShape.h b/src/collision/shapes/BoxShape.h index 3f6134ba..82b3aadb 100644 --- a/src/collision/shapes/BoxShape.h +++ b/src/collision/shapes/BoxShape.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef BOX_SHAPE_H -#define BOX_SHAPE_H +#ifndef REACTPHYSICS3D_BOX_SHAPE_H +#define REACTPHYSICS3D_BOX_SHAPE_H // Libraries #include diff --git a/src/collision/shapes/CollisionShape.h b/src/collision/shapes/CollisionShape.h index dba56ad7..535332c6 100644 --- a/src/collision/shapes/CollisionShape.h +++ b/src/collision/shapes/CollisionShape.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef COLLISION_SHAPE_H -#define COLLISION_SHAPE_H +#ifndef REACTPHYSICS3D_COLLISION_SHAPE_H +#define REACTPHYSICS3D_COLLISION_SHAPE_H // Libraries #include diff --git a/src/collision/shapes/ConeShape.h b/src/collision/shapes/ConeShape.h index b656d641..2e2614d1 100644 --- a/src/collision/shapes/ConeShape.h +++ b/src/collision/shapes/ConeShape.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef CONE_SHAPE_H -#define CONE_SHAPE_H +#ifndef REACTPHYSICS3D_CONE_SHAPE_H +#define REACTPHYSICS3D_CONE_SHAPE_H // Libraries #include "CollisionShape.h" diff --git a/src/collision/shapes/CylinderShape.h b/src/collision/shapes/CylinderShape.h index fb01add6..9caf7b22 100644 --- a/src/collision/shapes/CylinderShape.h +++ b/src/collision/shapes/CylinderShape.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef CYLINDER_SHAPE_H -#define CYLINDER_SHAPE_H +#ifndef REACTPHYSICS3D_CYLINDER_SHAPE_H +#define REACTPHYSICS3D_CYLINDER_SHAPE_H // Libraries #include "CollisionShape.h" diff --git a/src/collision/shapes/SphereShape.h b/src/collision/shapes/SphereShape.h index 8477d2c7..834250d1 100644 --- a/src/collision/shapes/SphereShape.h +++ b/src/collision/shapes/SphereShape.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef SPHERE_SHAPE_H -#define SPHERE_SHAPE_H +#ifndef REACTPHYSICS3D_SPHERE_SHAPE_H +#define REACTPHYSICS3D_SPHERE_SHAPE_H // Libraries #include "CollisionShape.h" diff --git a/src/configuration.h b/src/configuration.h index c4e73597..59319ae8 100644 --- a/src/configuration.h +++ b/src/configuration.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef CONFIGURATION_H -#define CONFIGURATION_H +#ifndef REACTPHYSICS3D_CONFIGURATION_H +#define REACTPHYSICS3D_CONFIGURATION_H // Libraries #include diff --git a/src/constraint/Constraint.h b/src/constraint/Constraint.h index 9470a3c6..e49b4984 100644 --- a/src/constraint/Constraint.h +++ b/src/constraint/Constraint.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef CONSTRAINT_H -#define CONSTRAINT_H +#ifndef REACTPHYSICS3D_CONSTRAINT_H +#define REACTPHYSICS3D_CONSTRAINT_H // Libraries #include "../body/RigidBody.h" diff --git a/src/constraint/ContactPoint.h b/src/constraint/ContactPoint.h index 0900d480..fa6abe5c 100644 --- a/src/constraint/ContactPoint.h +++ b/src/constraint/ContactPoint.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef CONTACT_POINT_H -#define CONTACT_POINT_H +#ifndef REACTPHYSICS3D_CONTACT_POINT_H +#define REACTPHYSICS3D_CONTACT_POINT_H // Libraries #include "Constraint.h" diff --git a/src/decimal.h b/src/decimal.h index 8b81eb22..52d0a0a8 100644 --- a/src/decimal.h +++ b/src/decimal.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef DECIMAL_H -#define DECIMAL_H +#ifndef REACTPHYSICS3D_DECIMAL_H +#define REACTPHYSICS3D_DECIMAL_H /// ReactPhysiscs3D namespace namespace reactphysics3d { diff --git a/src/engine/CollisionWorld.h b/src/engine/CollisionWorld.h index 00de6cdf..ba4071d6 100644 --- a/src/engine/CollisionWorld.h +++ b/src/engine/CollisionWorld.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef COLLISION_WORLD_H -#define COLLISION_WORLD_H +#ifndef REACTPHYSICS3D_COLLISION_WORLD_H +#define REACTPHYSICS3D_COLLISION_WORLD_H // Libraries #include diff --git a/src/engine/ContactManifold.h b/src/engine/ContactManifold.h index 68c1d2f3..25fd18c1 100644 --- a/src/engine/ContactManifold.h +++ b/src/engine/ContactManifold.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef CONTACT_MANIFOLD_H -#define CONTACT_MANIFOLD_H +#ifndef REACTPHYSICS3D_CONTACT_MANIFOLD_H +#define REACTPHYSICS3D_CONTACT_MANIFOLD_H // Libraries #include diff --git a/src/engine/ContactSolver.h b/src/engine/ContactSolver.h index f7bec64a..36997013 100644 --- a/src/engine/ContactSolver.h +++ b/src/engine/ContactSolver.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef CONTACT_SOLVER_H -#define CONTACT_SOLVER_H +#ifndef REACTPHYSICS3D_CONTACT_SOLVER_H +#define REACTPHYSICS3D_CONTACT_SOLVER_H // Libraries #include "../constraint/ContactPoint.h" diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index 1519b369..b2d11bd9 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef DYNAMICS_WORLD_H -#define DYNAMICS_WORLD_H +#ifndef REACTPHYSICS3D_DYNAMICS_WORLD_H +#define REACTPHYSICS3D_DYNAMICS_WORLD_H // Libraries #include "CollisionWorld.h" diff --git a/src/engine/OverlappingPair.h b/src/engine/OverlappingPair.h index b3c16880..40f1958b 100644 --- a/src/engine/OverlappingPair.h +++ b/src/engine/OverlappingPair.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef OVERLAPPING_PAIR_H -#define OVERLAPPING_PAIR_H +#ifndef REACTPHYSICS3D_OVERLAPPING_PAIR_H +#define REACTPHYSICS3D_OVERLAPPING_PAIR_H // Libraries #include "ContactManifold.h" diff --git a/src/engine/Profiler.h b/src/engine/Profiler.h index 286e909b..7030e685 100644 --- a/src/engine/Profiler.h +++ b/src/engine/Profiler.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef PROFILER_H -#define PROFILER_H +#ifndef REACTPHYSICS3D_PROFILER_H +#define REACTPHYSICS3D_PROFILER_H #ifdef IS_PROFILING_ACTIVE diff --git a/src/engine/Timer.h b/src/engine/Timer.h index 6c90ab05..22728537 100644 --- a/src/engine/Timer.h +++ b/src/engine/Timer.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef TIMER_H -#define TIMER_H +#ifndef REACTPHYSICS3D_TIMER_H +#define REACTPHYSICS3D_TIMER_H // Libraries #include diff --git a/src/mathematics/Matrix3x3.h b/src/mathematics/Matrix3x3.h index faa5f0fc..560d4031 100644 --- a/src/mathematics/Matrix3x3.h +++ b/src/mathematics/Matrix3x3.h @@ -24,8 +24,8 @@ ********************************************************************************/ -#ifndef MATRIX3X3_H -#define MATRIX3X3_H +#ifndef REACTPHYSICS3D_MATRIX3X3_H +#define REACTPHYSICS3D_MATRIX3X3_H // Libraries #include diff --git a/src/mathematics/Quaternion.h b/src/mathematics/Quaternion.h index d93964e0..c6fd64a1 100644 --- a/src/mathematics/Quaternion.h +++ b/src/mathematics/Quaternion.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef QUATERNION_H -#define QUATERNION_H +#ifndef REACTPHYSICS3D_QUATERNION_H +#define REACTPHYSICS3D_QUATERNION_H // Libraries #include diff --git a/src/mathematics/Transform.h b/src/mathematics/Transform.h index 0ead2ec7..2d3933db 100644 --- a/src/mathematics/Transform.h +++ b/src/mathematics/Transform.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef TRANSFORM_H -#define TRANSFORM_H +#ifndef REACTPHYSICS3D_TRANSFORM_H +#define REACTPHYSICS3D_TRANSFORM_H // Libraries #include "Matrix3x3.h" diff --git a/src/mathematics/Vector3.h b/src/mathematics/Vector3.h index 275c861e..3a54b06a 100644 --- a/src/mathematics/Vector3.h +++ b/src/mathematics/Vector3.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef VECTOR3_H -#define VECTOR3_H +#ifndef REACTPHYSICS3D_VECTOR3_H +#define REACTPHYSICS3D_VECTOR3_H // Libraries #include diff --git a/src/mathematics/mathematics.h b/src/mathematics/mathematics.h index 371cfdea..cc5aa467 100644 --- a/src/mathematics/mathematics.h +++ b/src/mathematics/mathematics.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef MATHEMATICS_H -#define MATHEMATICS_H +#ifndef REACTPHYSICS3D_MATHEMATICS_H +#define REACTPHYSICS3D_MATHEMATICS_H // Libraries #include "Matrix3x3.h" diff --git a/src/mathematics/mathematics_functions.h b/src/mathematics/mathematics_functions.h index e529fa8a..d5278abd 100644 --- a/src/mathematics/mathematics_functions.h +++ b/src/mathematics/mathematics_functions.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef MATHEMATICS_FUNCTIONS_H -#define MATHEMATICS_FUNCTIONS_H +#ifndef REACTPHYSICS3D_MATHEMATICS_FUNCTIONS_H +#define REACTPHYSICS3D_MATHEMATICS_FUNCTIONS_H // Libraries #include "../configuration.h" diff --git a/src/memory/MemoryAllocator.h b/src/memory/MemoryAllocator.h index 8ef90e14..46caa3dd 100644 --- a/src/memory/MemoryAllocator.h +++ b/src/memory/MemoryAllocator.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef MEMORY_ALLOCATOR_H -#define MEMORY_ALLOCATOR_H +#ifndef REACTPHYSICS3D_MEMORY_ALLOCATOR_H +#define REACTPHYSICS3D_MEMORY_ALLOCATOR_H // Libraries #include "../configuration.h" From f692f7ef126fd163f0ad37b3a239138b04fd543b Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Mon, 22 Apr 2013 21:25:40 +0200 Subject: [PATCH 10/66] Allocate memory for the collision shapes inside the physics engine --- examples/common/Box.cpp | 10 ++-- examples/common/Box.h | 3 -- src/collision/shapes/BoxShape.cpp | 5 ++ src/collision/shapes/BoxShape.h | 25 ++++++++++ src/collision/shapes/CollisionShape.cpp | 11 ++++- src/collision/shapes/CollisionShape.h | 54 +++++++++++++++++++++ src/collision/shapes/ConeShape.cpp | 7 +++ src/collision/shapes/ConeShape.h | 25 ++++++++++ src/collision/shapes/CylinderShape.cpp | 6 +++ src/collision/shapes/CylinderShape.h | 25 ++++++++++ src/collision/shapes/SphereShape.cpp | 8 +++- src/collision/shapes/SphereShape.h | 25 ++++++++++ src/engine/CollisionWorld.cpp | 63 ++++++++++++++++++++++++- src/engine/CollisionWorld.h | 14 +++++- src/engine/DynamicsWorld.cpp | 10 +++- src/engine/DynamicsWorld.h | 4 +- 16 files changed, 276 insertions(+), 19 deletions(-) diff --git a/examples/common/Box.cpp b/examples/common/Box.cpp index d4169c9a..1c311231 100644 --- a/examples/common/Box.cpp +++ b/examples/common/Box.cpp @@ -76,11 +76,13 @@ Box::Box(const openglframework::Vector3& size, const openglframework::Vector3 &p translateWorld(position); // Create the collision shape for the rigid body (box shape) - mCollisionShape = new rp3d::BoxShape(rp3d::Vector3(mSize[0], mSize[1], mSize[2])); + // ReactPhysics3D will clone this object to create an internal one. Therefore, + // it is OK if this object is destroy right after calling Dynamics::createRigidBody() + const rp3d::BoxShape collisionShape(rp3d::Vector3(mSize[0], mSize[1], mSize[2])); // Compute the inertia tensor of the body using its collision shape rp3d::Matrix3x3 inertiaTensor; - mCollisionShape->computeLocalInertiaTensor(inertiaTensor, mass); + collisionShape.computeLocalInertiaTensor(inertiaTensor, mass); // Initial position and orientation of the rigid body rp3d::Vector3 initPosition(position.x, position.y, position.z); @@ -88,7 +90,7 @@ Box::Box(const openglframework::Vector3& size, const openglframework::Vector3 &p rp3d::Transform transform(initPosition, initOrientation); // Create a rigid body corresponding to the cube in the dynamics world - mRigidBody = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, mCollisionShape); + mRigidBody = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, collisionShape); // If the Vertex Buffer object has not been created yet if (!areVBOsCreated) { @@ -100,8 +102,6 @@ Box::Box(const openglframework::Vector3& size, const openglframework::Vector3 &p // Destructor Box::~Box() { - // Destroy the collision shape - delete mCollisionShape; } // Render the cube at the correct position and with the correct orientation diff --git a/examples/common/Box.h b/examples/common/Box.h index d287c3cb..ec5d75b2 100644 --- a/examples/common/Box.h +++ b/examples/common/Box.h @@ -56,9 +56,6 @@ class Box : public openglframework::Object3D { /// Rigid body used to simulate the dynamics of the box rp3d::RigidBody* mRigidBody; - /// Collision shape of the rigid body - rp3d::BoxShape* mCollisionShape; - /// Scaling matrix (applied to a cube to obtain the correct box dimensions) openglframework::Matrix4 mScalingMatrix; diff --git a/src/collision/shapes/BoxShape.cpp b/src/collision/shapes/BoxShape.cpp index a521acc7..111e8397 100644 --- a/src/collision/shapes/BoxShape.cpp +++ b/src/collision/shapes/BoxShape.cpp @@ -50,6 +50,11 @@ BoxShape::BoxShape(const Vector3& extent) : CollisionShape(BOX), mExtent(extent) } +// Private copy-constructor +BoxShape::BoxShape(const BoxShape& shape) : CollisionShape(shape), mExtent(shape.mExtent) { + +} + // Destructor BoxShape::~BoxShape() { diff --git a/src/collision/shapes/BoxShape.h b/src/collision/shapes/BoxShape.h index 82b3aadb..f8ea72c9 100644 --- a/src/collision/shapes/BoxShape.h +++ b/src/collision/shapes/BoxShape.h @@ -69,6 +69,9 @@ class BoxShape : public CollisionShape { /// Destructor virtual ~BoxShape(); + /// Allocate and return a copy of the object + virtual BoxShape* clone(void* allocatedMemory) const; + /// Return the extents of the box const Vector3& getExtent() const; @@ -81,6 +84,9 @@ class BoxShape : public CollisionShape { /// Return the margin distance around the shape virtual decimal getMargin() const; + /// Return the number of bytes used by the collision shape + virtual size_t getSizeInBytes() const; + /// Return a local support point in a given direction with the object margin virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const; @@ -90,12 +96,20 @@ class BoxShape : public CollisionShape { /// Return the local inertia tensor of the collision shape virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; + /// Test equality between two box shapes + virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const; + #ifdef VISUAL_DEBUG /// Draw the Box (only for testing purpose) virtual void draw() const; #endif }; +// Allocate and return a copy of the object +inline BoxShape* BoxShape::clone(void* allocatedMemory) const { + return new (allocatedMemory) BoxShape(*this); +} + // Return the extents of the box inline const Vector3& BoxShape::getExtent() const { return mExtent; @@ -117,6 +131,11 @@ inline decimal BoxShape::getMargin() const { return OBJECT_MARGIN; } +// Return the number of bytes used by the collision shape +inline size_t BoxShape::getSizeInBytes() const { + return sizeof(BoxShape); +} + // Return a local support point in a given direction with the object margin inline Vector3 BoxShape::getLocalSupportPointWithMargin(const Vector3& direction) const { @@ -136,6 +155,12 @@ inline Vector3 BoxShape::getLocalSupportPointWithoutMargin(const Vector3& direct direction.z < 0.0 ? -mExtent.z : mExtent.z); } +// Test equality between two box shapes +inline bool BoxShape::isEqualTo(const CollisionShape& otherCollisionShape) const { + const BoxShape& otherShape = dynamic_cast(otherCollisionShape); + return (mExtent == otherShape.mExtent); +} + } #endif diff --git a/src/collision/shapes/CollisionShape.cpp b/src/collision/shapes/CollisionShape.cpp index 6775c3a2..aca091d5 100644 --- a/src/collision/shapes/CollisionShape.cpp +++ b/src/collision/shapes/CollisionShape.cpp @@ -30,13 +30,20 @@ using namespace reactphysics3d; // Constructor -CollisionShape::CollisionShape(CollisionShapeType type) : mType(type) { +CollisionShape::CollisionShape(CollisionShapeType type) + : mType(type), mNbSimilarCreatedShapes(0) { } +// Private copy-constructor +CollisionShape::CollisionShape(const CollisionShape& shape) + : mType(shape.mType), mNbSimilarCreatedShapes(shape.mNbSimilarCreatedShapes){ + +} + // Destructor CollisionShape::~CollisionShape() { - + assert(mNbSimilarCreatedShapes == 0); } // Update the AABB of a body using its collision shape diff --git a/src/collision/shapes/CollisionShape.h b/src/collision/shapes/CollisionShape.h index 535332c6..4d865b19 100644 --- a/src/collision/shapes/CollisionShape.h +++ b/src/collision/shapes/CollisionShape.h @@ -54,6 +54,9 @@ class CollisionShape { /// Type of the collision shape CollisionShapeType mType; + + /// Current number of similar created shapes + uint mNbSimilarCreatedShapes; // -------------------- Methods -------------------- // @@ -73,9 +76,18 @@ class CollisionShape { /// Destructor virtual ~CollisionShape(); + /// Allocate and return a copy of the object + virtual CollisionShape* clone(void* allocatedMemory) const=0; + /// Return the type of the collision shapes CollisionShapeType getType() const; + /// Return the number of similar created shapes + uint getNbSimilarCreatedShapes() const; + + /// Return the number of bytes used by the collision shape + virtual size_t getSizeInBytes() const = 0; + /// Return a local support point in a given direction with the object margin virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const=0; @@ -93,6 +105,18 @@ class CollisionShape { /// Update the AABB of a body using its collision shape virtual void updateAABB(AABB& aabb, const Transform& transform); + + /// Increment the number of similar allocated collision shapes + void incrementNbSimilarCreatedShapes(); + + /// Decrement the number of similar allocated collision shapes + void decrementNbSimilarCreatedShapes(); + + /// Equality operator between two collision shapes. + bool operator==(const CollisionShape& otherCollisionShape) const; + + /// Test equality between two collision shapes of the same type (same derived classes). + virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const=0; }; // Return the type of the collision shape @@ -100,6 +124,36 @@ inline CollisionShapeType CollisionShape::getType() const { return mType; } +// Return the number of similar created shapes +inline uint CollisionShape::getNbSimilarCreatedShapes() const { + return mNbSimilarCreatedShapes; +} + +// Increment the number of similar allocated collision shapes +inline void CollisionShape::incrementNbSimilarCreatedShapes() { + mNbSimilarCreatedShapes++; +} + +// Decrement the number of similar allocated collision shapes +inline void CollisionShape::decrementNbSimilarCreatedShapes() { + mNbSimilarCreatedShapes--; +} + +// Equality operator between two collision shapes. +/// This methods returns true only if the two collision shapes are of the same type and +/// of the same dimensions. +inline bool CollisionShape::operator==(const CollisionShape& otherCollisionShape) const { + + // If the two collisions shapes are not of the same type (same derived classes) + // we return false + if (mType != otherCollisionShape.mType) return false; + + assert(typeid(*this) == typeid(otherCollisionShape)); + + // Check if the two shapes are equal + return otherCollisionShape.isEqualTo(*this); +} + } #endif diff --git a/src/collision/shapes/ConeShape.cpp b/src/collision/shapes/ConeShape.cpp index 55f56e35..58439866 100644 --- a/src/collision/shapes/ConeShape.cpp +++ b/src/collision/shapes/ConeShape.cpp @@ -53,6 +53,13 @@ ConeShape::ConeShape(decimal radius, decimal height) mSinTheta = radius / (sqrt(radius * radius + height * height)); } +// Private copy-constructor +ConeShape::ConeShape(const ConeShape& shape) + : CollisionShape(shape), mRadius(shape.mRadius), mHalfHeight(shape.mHalfHeight), + mSinTheta(shape.mSinTheta){ + +} + // Destructor ConeShape::~ConeShape() { diff --git a/src/collision/shapes/ConeShape.h b/src/collision/shapes/ConeShape.h index 2e2614d1..d1721383 100644 --- a/src/collision/shapes/ConeShape.h +++ b/src/collision/shapes/ConeShape.h @@ -75,6 +75,9 @@ class ConeShape : public CollisionShape { /// Destructor virtual ~ConeShape(); + /// Allocate and return a copy of the object + virtual ConeShape* clone(void* allocatedMemory) const; + /// Return the radius decimal getRadius() const; @@ -87,6 +90,9 @@ class ConeShape : public CollisionShape { /// Set the height void setHeight(decimal height); + /// Return the number of bytes used by the collision shape + virtual size_t getSizeInBytes() const; + /// Return a local support point in a given direction with the object margin virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const; @@ -102,12 +108,20 @@ class ConeShape : public CollisionShape { /// Return the margin distance around the shape virtual decimal getMargin() const; + /// Test equality between two cone shapes + virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const; + #ifdef VISUAL_DEBUG /// Draw the sphere (only for testing purpose) virtual void draw() const; #endif }; +// Allocate and return a copy of the object +inline ConeShape* ConeShape::clone(void* allocatedMemory) const { + return new (allocatedMemory) ConeShape(*this); +} + // Return the radius inline decimal ConeShape::getRadius() const { return mRadius; @@ -134,6 +148,11 @@ inline void ConeShape::setHeight(decimal height) { mSinTheta = mRadius / (sqrt(mRadius * mRadius + height * height)); } +// Return the number of bytes used by the collision shape +inline size_t ConeShape::getSizeInBytes() const { + return sizeof(ConeShape); +} + // Return the local extents in x,y and z direction inline Vector3 ConeShape::getLocalExtents(decimal margin) const { return Vector3(mRadius + margin, mHalfHeight + margin, mRadius + margin); @@ -153,6 +172,12 @@ inline decimal ConeShape::getMargin() const { return OBJECT_MARGIN; } +// Test equality between two cone shapes +inline bool ConeShape::isEqualTo(const CollisionShape& otherCollisionShape) const { + const ConeShape& otherShape = dynamic_cast(otherCollisionShape); + return (mRadius == otherShape.mRadius && mHalfHeight == otherShape.mHalfHeight); +} + } #endif diff --git a/src/collision/shapes/CylinderShape.cpp b/src/collision/shapes/CylinderShape.cpp index 07c3cb17..8bb87eaa 100644 --- a/src/collision/shapes/CylinderShape.cpp +++ b/src/collision/shapes/CylinderShape.cpp @@ -48,6 +48,12 @@ CylinderShape::CylinderShape(decimal radius, decimal height) } +// Private copy-constructor +CylinderShape::CylinderShape(const CylinderShape& shape) + : CollisionShape(shape), mRadius(shape.mRadius), mHalfHeight(shape.mHalfHeight) { + +} + // Destructor CylinderShape::~CylinderShape() { diff --git a/src/collision/shapes/CylinderShape.h b/src/collision/shapes/CylinderShape.h index 9caf7b22..3d0b91d4 100644 --- a/src/collision/shapes/CylinderShape.h +++ b/src/collision/shapes/CylinderShape.h @@ -71,6 +71,9 @@ class CylinderShape : public CollisionShape { /// Destructor virtual ~CylinderShape(); + /// Allocate and return a copy of the object + virtual CylinderShape* clone(void* allocatedMemory) const; + /// Return the radius decimal getRadius() const; @@ -83,6 +86,9 @@ class CylinderShape : public CollisionShape { /// Set the height void setHeight(decimal height); + /// Return the number of bytes used by the collision shape + virtual size_t getSizeInBytes() const; + /// Return a local support point in a given direction with the object margin virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const; @@ -98,12 +104,20 @@ class CylinderShape : public CollisionShape { /// Return the margin distance around the shape virtual decimal getMargin() const; + /// Test equality between two cylinder shapes + virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const; + #ifdef VISUAL_DEBUG /// Draw the sphere (only for testing purpose) virtual void draw() const; #endif }; +/// Allocate and return a copy of the object +inline CylinderShape* CylinderShape::clone(void* allocatedMemory) const { + return new (allocatedMemory) CylinderShape(*this); +} + // Return the radius inline decimal CylinderShape::getRadius() const { return mRadius; @@ -124,6 +138,11 @@ inline void CylinderShape::setHeight(decimal height) { mHalfHeight = height * decimal(0.5); } +// Return the number of bytes used by the collision shape +inline size_t CylinderShape::getSizeInBytes() const { + return sizeof(CylinderShape); +} + // Return the local extents in x,y and z direction inline Vector3 CylinderShape::getLocalExtents(decimal margin) const { return Vector3(mRadius + margin, mHalfHeight + margin, mRadius + margin); @@ -143,6 +162,12 @@ inline decimal CylinderShape::getMargin() const { return OBJECT_MARGIN; } +// Test equality between two cylinder shapes +inline bool CylinderShape::isEqualTo(const CollisionShape& otherCollisionShape) const { + const CylinderShape& otherShape = dynamic_cast(otherCollisionShape); + return (mRadius == otherShape.mRadius && mHalfHeight == otherShape.mHalfHeight); +} + } #endif diff --git a/src/collision/shapes/SphereShape.cpp b/src/collision/shapes/SphereShape.cpp index c3afa4a3..6ce64c6f 100644 --- a/src/collision/shapes/SphereShape.cpp +++ b/src/collision/shapes/SphereShape.cpp @@ -45,7 +45,13 @@ using namespace reactphysics3d; using namespace std; // Constructor -SphereShape::SphereShape(decimal radius) : CollisionShape(SPHERE), mRadius(radius) { +SphereShape::SphereShape(decimal radius): CollisionShape(SPHERE), mRadius(radius) { + +} + +// Private copy-constructor +SphereShape::SphereShape(const SphereShape& shape) + : CollisionShape(shape), mRadius(shape.mRadius) { } diff --git a/src/collision/shapes/SphereShape.h b/src/collision/shapes/SphereShape.h index 834250d1..0b7d4392 100644 --- a/src/collision/shapes/SphereShape.h +++ b/src/collision/shapes/SphereShape.h @@ -65,12 +65,18 @@ class SphereShape : public CollisionShape { /// Destructor virtual ~SphereShape(); + /// Allocate and return a copy of the object + virtual SphereShape* clone(void* allocatedMemory) const; + /// Return the radius of the sphere decimal getRadius() const; /// Set the radius of the sphere void setRadius(decimal radius); + /// Return the number of bytes used by the collision shape + virtual size_t getSizeInBytes() const; + /// Return a local support point in a given direction with the object margin virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const; @@ -89,12 +95,20 @@ class SphereShape : public CollisionShape { /// Update the AABB of a body using its collision shape virtual void updateAABB(AABB& aabb, const Transform& transform); + /// Test equality between two sphere shapes + virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const; + #ifdef VISUAL_DEBUG /// Draw the sphere (only for testing purpose) virtual void draw() const; #endif }; +/// Allocate and return a copy of the object +inline SphereShape* SphereShape::clone(void* allocatedMemory) const { + return new (allocatedMemory) SphereShape(*this); +} + // Get the radius of the sphere inline decimal SphereShape::getRadius() const { return mRadius; @@ -105,6 +119,11 @@ inline void SphereShape::setRadius(decimal radius) { mRadius = radius; } +// Return the number of bytes used by the collision shape +inline size_t SphereShape::getSizeInBytes() const { + return sizeof(SphereShape); +} + // Return a local support point in a given direction with the object margin inline Vector3 SphereShape::getLocalSupportPointWithMargin(const Vector3& direction) const { @@ -163,6 +182,12 @@ inline void SphereShape::updateAABB(AABB& aabb, const Transform& transform) { aabb.setMax(maxCoordinates); } +// Test equality between two sphere shapes +inline bool SphereShape::isEqualTo(const CollisionShape& otherCollisionShape) const { + const SphereShape& otherShape = dynamic_cast(otherCollisionShape); + return (mRadius == otherShape.mRadius); +} + } #endif diff --git a/src/engine/CollisionWorld.cpp b/src/engine/CollisionWorld.cpp index 39b9dc74..3a048d09 100644 --- a/src/engine/CollisionWorld.cpp +++ b/src/engine/CollisionWorld.cpp @@ -38,7 +38,8 @@ CollisionWorld::CollisionWorld() // Destructor CollisionWorld::~CollisionWorld() { - + assert(mCollisionShapes.empty()); + assert(mBodies.empty()); } // Notify the world about a new broad-phase overlapping pair @@ -104,7 +105,7 @@ void CollisionWorld::destroyCollisionBody(CollisionBody* collisionBody) { collisionBody->CollisionBody::~CollisionBody(); // Remove the collision body from the list of bodies - mBodies.erase(collisionBody); // TODO : Maybe use a set to make this faster + mBodies.erase(collisionBody); // Free the object from the memory allocator mMemoryAllocator.release(collisionBody, sizeof(CollisionBody)); @@ -127,4 +128,62 @@ bodyindex CollisionWorld::computeNextAvailableBodyID() { return bodyID; } +// Create a new collision shape. +/// First, this methods checks that the new collision shape does not exist yet in the +/// world. If it already exists, we do not allocate memory for a new one but instead +/// we reuse the existing one. The goal is to only allocate memory for a single +/// collision shape if this one is used for several bodies in the world. To allocate +/// memory for a new collision shape, we use the memory allocator. +CollisionShape* CollisionWorld::createCollisionShape(const CollisionShape& collisionShape) { + + // Check if there is already a similar collision shape in the world + std::list::iterator it; + for (it = mCollisionShapes.begin(); it != mCollisionShapes.end(); ++it) { + + if (collisionShape == (*(*it))) { + + // Increment the number of similar created shapes + (*it)->incrementNbSimilarCreatedShapes(); + + // A similar collision shape already exists in the world, so we do not + // create a new one but we simply return a pointer to the existing one + return (*it); + } + } + + // A similar collision shape does not already exist in the world, so we create a + // new one and add it to the world + void* allocatedMemory = mMemoryAllocator.allocate(collisionShape.getSizeInBytes()); + CollisionShape* newCollisionShape = collisionShape.clone(allocatedMemory); + mCollisionShapes.push_back(newCollisionShape); + + newCollisionShape->incrementNbSimilarCreatedShapes(); + + // Return a pointer to the new collision shape + return newCollisionShape; +} + + +// Remove a collision shape. +/// First, we check if another body is still using the same collision shape. If so, +/// we keep the allocated collision shape. If it is not the case, we can deallocate +/// the memory associated with the collision shape. +void CollisionWorld::removeCollisionShape(CollisionShape* collisionShape) { + + assert(collisionShape->getNbSimilarCreatedShapes() != 0); + + // Decrement the number of bodies using the same collision shape + collisionShape->decrementNbSimilarCreatedShapes(); + + // If no other body is using the collision shape in the world + if (collisionShape->getNbSimilarCreatedShapes() == 0) { + + // Remove the shape from the set of shapes in the world + mCollisionShapes.remove(collisionShape); + + // Deallocate the memory used by the collision shape + mMemoryAllocator.release(collisionShape, collisionShape->getSizeInBytes()); + } +} + diff --git a/src/engine/CollisionWorld.h b/src/engine/CollisionWorld.h index ba4071d6..a120956b 100644 --- a/src/engine/CollisionWorld.h +++ b/src/engine/CollisionWorld.h @@ -29,6 +29,7 @@ // Libraries #include #include +#include #include #include "../mathematics/mathematics.h" #include "Profiler.h" @@ -60,6 +61,9 @@ class CollisionWorld { /// All the bodies (rigid and soft) of the world std::set mBodies; + /// All the collision shapes of the world + std::list mCollisionShapes; + /// Broad-phase overlapping pairs of bodies std::map mOverlappingPairs; @@ -95,9 +99,15 @@ class CollisionWorld { /// Return the next available body ID bodyindex computeNextAvailableBodyID(); + /// Create a new collision shape. + CollisionShape* createCollisionShape(const CollisionShape& collisionShape); + + /// Remove a collision shape. + void removeCollisionShape(CollisionShape* collisionShape); + public : - // ----- Methods ----- // + // -------------------- Methods -------------------- // /// Constructor CollisionWorld(); @@ -118,7 +128,7 @@ class CollisionWorld { /// Destroy a collision body void destroyCollisionBody(CollisionBody* collisionBody); - // ----- Friends ----- // + // -------------------- Friends -------------------- // friend class CollisionDetection; }; diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 27184f98..13d79138 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -258,7 +258,7 @@ void DynamicsWorld::applyGravity() { // Create a rigid body into the physics world RigidBody* DynamicsWorld::createRigidBody(const Transform& transform, decimal mass, const Matrix3x3& inertiaTensorLocal, - CollisionShape* collisionShape) { + const CollisionShape& collisionShape) { // Compute the body ID bodyindex bodyID = computeNextAvailableBodyID(); @@ -266,11 +266,14 @@ RigidBody* DynamicsWorld::createRigidBody(const Transform& transform, decimal ma // Largest index cannot be used (it is used for invalid index) assert(bodyID < std::numeric_limits::max()); + // Create a collision shape for the rigid body into the world + CollisionShape* newCollisionShape = createCollisionShape(collisionShape); + // Create the rigid body RigidBody* rigidBody = new (mMemoryAllocator.allocate(sizeof(RigidBody))) RigidBody(transform, mass, inertiaTensorLocal, - collisionShape, + newCollisionShape, bodyID); assert(rigidBody != NULL); @@ -294,6 +297,9 @@ void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { // Add the body ID to the list of free IDs mFreeBodiesIDs.push_back(rigidBody->getID()); + // Remove the collision shape from the world + removeCollisionShape(rigidBody->getCollisionShape()); + // Call the constructor of the rigid body rigidBody->RigidBody::~RigidBody(); diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index b2d11bd9..93b87bf2 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -158,10 +158,10 @@ public : /// Set the isErrorCorrectionActive value void setIsErrorCorrectionActive(bool isErrorCorrectionActive); - /// Create a rigid body into the physics world + /// Create a rigid body into the physics world. RigidBody* createRigidBody(const Transform& transform, decimal mass, const Matrix3x3& inertiaTensorLocal, - CollisionShape* collisionShape); + const CollisionShape& collisionShape); /// Destroy a rigid body void destroyRigidBody(RigidBody* rigidBody); From 471e4f7afcf1b4e0b5de2b3fecd02e24ab21f72b Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Mon, 22 Apr 2013 23:32:52 +0200 Subject: [PATCH 11/66] Start to implement the joints --- src/constraint/BallAndSocketJoint.cpp | 42 +++++++++ src/constraint/BallAndSocketJoint.h | 59 ++++++++++++ src/constraint/Constraint.cpp | 11 +-- src/constraint/Constraint.h | 40 +------- src/constraint/ContactPoint.cpp | 9 +- src/constraint/ContactPoint.h | 59 +++++++++++- src/engine/ConstraintSolver.cpp | 39 ++++++++ src/engine/ConstraintSolver.h | 128 ++++++++++++++++++++++++++ src/engine/ContactSolver.cpp | 12 +-- src/engine/ContactSolver.h | 2 +- 10 files changed, 345 insertions(+), 56 deletions(-) create mode 100644 src/constraint/BallAndSocketJoint.cpp create mode 100644 src/constraint/BallAndSocketJoint.h create mode 100644 src/engine/ConstraintSolver.cpp create mode 100644 src/engine/ConstraintSolver.h diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp new file mode 100644 index 00000000..51b017c7 --- /dev/null +++ b/src/constraint/BallAndSocketJoint.cpp @@ -0,0 +1,42 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "BallAndSocketJoint.h" + +using namespace reactphysics3d; + +// Constructor +BallAndSocketJoint::BallAndSocketJoint(RigidBody* const body1, RigidBody* const body2, + bool active, ConstraintType type) + : Constraint(body1, body2, active, type){ + +} + +// Destructor +BallAndSocketJoint::~BallAndSocketJoint() { + +} + diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h new file mode 100644 index 00000000..7908dc8f --- /dev/null +++ b/src/constraint/BallAndSocketJoint.h @@ -0,0 +1,59 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_BALL_AND_SOCKET_JOINT_H +#define REACTPHYSICS3D_BALL_AND_SOCKET_JOINT_H + +// Libraries +#include "Constraint.h" + +namespace reactphysics3d { + +// Class BallAndSocketJoint +/** + * This class represents a ball-and-socket joint that allows arbitrary rotation + * between two bodies. + */ +class BallAndSocketJoint : public Constraint { + + private : + + // -------------------- Attributes -------------------- // + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + BallAndSocketJoint(RigidBody* const body1, RigidBody* const body2, + bool active, ConstraintType type); + + /// Destructor + virtual ~BallAndSocketJoint(); +}; + +} + +#endif diff --git a/src/constraint/Constraint.cpp b/src/constraint/Constraint.cpp index 7f40b3f7..a1bd2b76 100644 --- a/src/constraint/Constraint.cpp +++ b/src/constraint/Constraint.cpp @@ -30,14 +30,9 @@ using namespace reactphysics3d; // Constructor Constraint::Constraint(RigidBody* const body1, RigidBody* const body2, - uint nbConstraints, bool active, ConstraintType type) - :mBody1(body1), mBody2(body2), mActive(active), - mNbConstraints(nbConstraints), mType(type) { - - // Initialize the cached lambda values - for (uint i=0; i mCachedLambdas; - // -------------------- Methods -------------------- // /// Private copy-constructor @@ -81,7 +74,7 @@ class Constraint { // -------------------- Methods -------------------- // /// Constructor - Constraint(RigidBody* const body1, RigidBody* const body2, uint nbConstraints, + Constraint(RigidBody* const body1, RigidBody* const body2, bool active, ConstraintType type); /// Destructor @@ -98,15 +91,6 @@ class Constraint { /// Return the type of the constraint ConstraintType getType() const; - - /// Return the number of mathematical constraints - unsigned int getNbConstraints() const; - - /// Get one cached lambda value - decimal getCachedLambda(uint index) const; - - /// Set on cached lambda value - void setCachedLambda(uint index, decimal lambda); }; // Return the reference to the body 1 @@ -127,25 +111,7 @@ inline bool Constraint::isActive() const { // Return the type of the constraint inline ConstraintType Constraint::getType() const { return mType; -} - - -// Return the number auxiliary constraints -inline uint Constraint::getNbConstraints() const { - return mNbConstraints; -} - -// Get one previous lambda value -inline decimal Constraint::getCachedLambda(uint index) const { - assert(index < mNbConstraints); - return mCachedLambdas[index]; -} - -// Set on cached lambda value -inline void Constraint::setCachedLambda(uint index, decimal lambda) { - assert(index < mNbConstraints); - mCachedLambdas[index] = lambda; -} +} } diff --git a/src/constraint/ContactPoint.cpp b/src/constraint/ContactPoint.cpp index d2f34781..df38e269 100644 --- a/src/constraint/ContactPoint.cpp +++ b/src/constraint/ContactPoint.cpp @@ -32,13 +32,16 @@ using namespace std; // Constructor ContactPoint::ContactPoint(RigidBody* const body1, RigidBody* const body2, const ContactInfo* contactInfo) - : Constraint(body1, body2, 3, true, CONTACT), mNormal(contactInfo->normal), + : Constraint(body1, body2, true, CONTACT), mNormal(contactInfo->normal), mPenetrationDepth(contactInfo->penetrationDepth), mLocalPointOnBody1(contactInfo->localPoint1), mLocalPointOnBody2(contactInfo->localPoint2), mWorldPointOnBody1(body1->getTransform() * contactInfo->localPoint1), - mWorldPointOnBody2(body2->getTransform() * contactInfo->localPoint2), - mIsRestingContact(false), mFrictionVectors(2, Vector3(0, 0, 0)) { + mWorldPointOnBody2(body2->getTransform() * contactInfo->localPoint2), + mIsRestingContact(false) { + + mFrictionVectors[0] = Vector3(0, 0, 0); + mFrictionVectors[1] = Vector3(0, 0, 0); assert(mPenetrationDepth > 0.0); diff --git a/src/constraint/ContactPoint.h b/src/constraint/ContactPoint.h index fa6abe5c..8ad4eff9 100644 --- a/src/constraint/ContactPoint.h +++ b/src/constraint/ContactPoint.h @@ -84,7 +84,16 @@ class ContactPoint : public Constraint { bool mIsRestingContact; /// Two orthogonal vectors that span the tangential friction plane - std::vector mFrictionVectors; + Vector3 mFrictionVectors[2]; + + /// Cached penetration impulse + decimal mPenetrationImpulse; + + /// Cached first friction impulse + decimal mFrictionImpulse1; + + /// Cached second friction impulse + decimal mFrictionImpulse2; // -------------------- Methods -------------------- // @@ -122,6 +131,24 @@ class ContactPoint : public Constraint { /// Return the contact world point on body 2 Vector3 getWorldPointOnBody2() const; + /// Return the cached penetration impulse + decimal getPenetrationImpulse() const; + + /// Return the cached first friction impulse + decimal getFrictionImpulse1() const; + + /// Return the cached second friction impulse + decimal getFrictionImpulse2() const; + + /// Set the cached penetration impulse + void setPenetrationImpulse(decimal impulse); + + /// Set the first cached friction impulse + void setFrictionImpulse1(decimal impulse); + + /// Set the second cached friction impulse + void setFrictionImpulse2(decimal impulse); + /// Set the contact world point on body 1 void setWorldPointOnBody1(const Vector3& worldPoint); @@ -185,6 +212,36 @@ inline Vector3 ContactPoint::getWorldPointOnBody2() const { return mWorldPointOnBody2; } +// Return the cached penetration impulse +inline decimal ContactPoint::getPenetrationImpulse() const { + return mPenetrationImpulse; +} + +// Return the cached first friction impulse +inline decimal ContactPoint::getFrictionImpulse1() const { + return mFrictionImpulse1; +} + +// Return the cached second friction impulse +inline decimal ContactPoint::getFrictionImpulse2() const { + return mFrictionImpulse2; +} + +// Set the cached penetration impulse +inline void ContactPoint::setPenetrationImpulse(decimal impulse) { + mPenetrationImpulse = impulse; +} + +// Set the first cached friction impulse +inline void ContactPoint::setFrictionImpulse1(decimal impulse) { + mFrictionImpulse1 = impulse; +} + +// Set the second cached friction impulse +inline void ContactPoint::setFrictionImpulse2(decimal impulse) { + mFrictionImpulse2 = impulse; +} + // Set the contact world point on body 1 inline void ContactPoint::setWorldPointOnBody1(const Vector3& worldPoint) { mWorldPointOnBody1 = worldPoint; diff --git a/src/engine/ConstraintSolver.cpp b/src/engine/ConstraintSolver.cpp new file mode 100644 index 00000000..a0131bac --- /dev/null +++ b/src/engine/ConstraintSolver.cpp @@ -0,0 +1,39 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "ConstraintSolver.h" + +using namespace reactphysics3d; + +// Constructor +ConstraintSolver::ConstraintSolver() { + +} + +// Destructor +ConstraintSolver::~ConstraintSolver() { + +} diff --git a/src/engine/ConstraintSolver.h b/src/engine/ConstraintSolver.h new file mode 100644 index 00000000..227c4c9a --- /dev/null +++ b/src/engine/ConstraintSolver.h @@ -0,0 +1,128 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_CONSTRAINT_SOLVER_H +#define REACTPHYSICS3D_CONSTRAINT_SOLVER_H + +// Libraries + +namespace reactphysics3d { + +// Class ConstraintSolver +/** + * This class represents the constraint solver that is used to solve constraints between + * the rigid bodies. The constraint solver is based on the "Sequential Impulse" technique + * described by Erin Catto in his GDC slides (http://code.google.com/p/box2d/downloads/list). + * + * A constraint between two bodies is represented by a function C(x) which is equal to zero + * when the constraint is satisfied. The condition C(x)=0 describes a valid position and the + * condition dC(x)/dt=0 describes a valid velocity. We have dC(x)/dt = Jv + b = 0 where J is + * the Jacobian matrix of the constraint, v is a vector that contains the velocity of both + * bodies and b is the constraint bias. We are looking for a force F_c that will act on the + * bodies to keep the constraint satisfied. Note that from the virtual work principle, we have + * F_c = J^t * lambda where J^t is the transpose of the Jacobian matrix and lambda is a + * Lagrange multiplier. Therefore, finding the force F_c is equivalent to finding the Lagrange + * multiplier lambda. + + * An impulse P = F * dt where F is a force and dt is the timestep. We can apply impulses a + * body to change its velocity. The idea of the Sequential Impulse technique is to apply + * impulses to bodies of each constraints in order to keep the constraint satisfied. + * + * --- Step 1 --- + * + * First, we integrate the applied force F_a acting of each rigid body (like gravity, ...) and + * we obtain some new velocities v2' that tends to violate the constraints. + * + * v2' = v1 + dt * M^-1 * F_a + * + * where M is a matrix that contains mass and inertia tensor information. + * + * --- Step 2 --- + * + * During the second step, we iterate over all the constraints for a certain number of + * iterations and for each constraint we compute the impulse to apply to the bodies needed + * so that the new velocity of the bodies satisfy Jv + b = 0. From the Newton law, we know that + * M * deltaV = P_c where M is the mass of the body, deltaV is the difference of velocity and + * P_c is the constraint impulse to apply to the body. Therefore, we have + * v2 = v2' + M^-1 * P_c. For each constraint, we can compute the Lagrange multiplier lambda + * using : lambda = -m_c (Jv2' + b) where m_c = 1 / (J * M^-1 * J^t). Now that we have the + * Lagrange multiplier lambda, we can compute the impulse P_c = J^t * lambda * dt to apply to + * the bodies to satisfy the constraint. + * + * --- Step 3 --- + * + * In the third step, we integrate the new position x2 of the bodies using the new velocities + * v2 computed in the second step with : x2 = x1 + dt * v2. + * + * Note that in the following code (as it is also explained in the slides from Erin Catto), + * the value lambda is not only the lagrange multiplier but is the multiplication of the + * Lagrange multiplier with the timestep dt. Therefore, in the following code, when we use + * lambda, we mean (lambda * dt). + * + * We are using the accumulated impulse technique that is also described in the slides from + * Erin Catto. + * + * We are also using warm starting. The idea is to warm start the solver at the beginning of + * each step by applying the last impulstes for the constraints that we already existing at the + * previous step. This allows the iterative solver to converge faster towards the solution. + * + * For contact constraints, we are also using split impulses so that the position correction + * that uses Baumgarte stabilization does not change the momentum of the bodies. + * + * There are two ways to apply the friction constraints. Either the friction constraints are + * applied at each contact point or they are applied only at the center of the contact manifold + * between two bodies. If we solve the friction constraints at each contact point, we need + * two constraints (two tangential friction directions) and if we solve the friction + * constraints at the center of the contact manifold, we need two constraints for tangential + * friction but also another twist friction constraint to prevent spin of the body around the + * contact manifold center. + */ +class ConstraintSolver { + + private : + + // -------------------- Attributes -------------------- // + + /// Number of iterations of the contact solver + uint mNbIterations; + + /// Current time step + decimal mTimeStep; + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + ConstraintSolver(); + + /// Destructor + ~ConstraintSolver(); + +}; + +} + +#endif diff --git a/src/engine/ContactSolver.cpp b/src/engine/ContactSolver.cpp index e90ed978..7d786588 100644 --- a/src/engine/ContactSolver.cpp +++ b/src/engine/ContactSolver.cpp @@ -307,9 +307,9 @@ void ContactSolver::initializeContactConstraints() { if (mIsWarmStartingActive) { // Get the cached accumulated impulses from the previous step - contactPoint.penetrationImpulse = externalContact->getCachedLambda(0); - contactPoint.friction1Impulse = externalContact->getCachedLambda(1); - contactPoint.friction2Impulse = externalContact->getCachedLambda(2); + contactPoint.penetrationImpulse = externalContact->getPenetrationImpulse(); + contactPoint.friction1Impulse = externalContact->getFrictionImpulse1(); + contactPoint.friction2Impulse = externalContact->getFrictionImpulse2(); } // Initialize the split impulses to zero @@ -785,9 +785,9 @@ void ContactSolver::storeImpulses() { ContactPointSolver& contactPoint = manifold.contacts[i]; - contactPoint.externalContact->setCachedLambda(0, contactPoint.penetrationImpulse); - contactPoint.externalContact->setCachedLambda(1, contactPoint.friction1Impulse); - contactPoint.externalContact->setCachedLambda(2, contactPoint.friction2Impulse); + contactPoint.externalContact->setPenetrationImpulse(contactPoint.penetrationImpulse); + contactPoint.externalContact->setFrictionImpulse1(contactPoint.friction1Impulse); + contactPoint.externalContact->setFrictionImpulse2(contactPoint.friction2Impulse); contactPoint.externalContact->setFrictionVector1(contactPoint.frictionVector1); contactPoint.externalContact->setFrictionVector2(contactPoint.frictionVector2); diff --git a/src/engine/ContactSolver.h b/src/engine/ContactSolver.h index 36997013..1153f306 100644 --- a/src/engine/ContactSolver.h +++ b/src/engine/ContactSolver.h @@ -345,7 +345,7 @@ class ContactSolver { /// Reference to the world DynamicsWorld& mWorld; - /// Number of iterations of the constraints solver + /// Number of iterations of the contact solver uint mNbIterations; /// Split linear velocities for the position contact solver (split impulse) From 2b2143d41e1bd5472c220ce5bf3d180902e041a6 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 24 Apr 2013 19:24:28 +0200 Subject: [PATCH 12/66] Continue the implementation of the constraint solver --- src/collision/CollisionDetection.cpp | 12 ++- src/collision/CollisionDetection.h | 2 +- src/collision/ContactInfo.cpp | 38 --------- src/collision/ContactInfo.h | 81 ------------------- .../narrowphase/EPA/EPAAlgorithm.cpp | 4 +- src/collision/narrowphase/EPA/EPAAlgorithm.h | 4 +- .../narrowphase/GJK/GJKAlgorithm.cpp | 12 +-- src/collision/narrowphase/GJK/GJKAlgorithm.h | 6 +- .../narrowphase/NarrowPhaseAlgorithm.h | 6 +- .../narrowphase/SphereVsSphereAlgorithm.cpp | 4 +- .../SphereVsSphereAlgorithm.cpp.autosave | 81 +++++++++++++++++++ .../narrowphase/SphereVsSphereAlgorithm.h | 4 +- src/constraint/BallAndSocketJoint.cpp | 5 +- src/constraint/BallAndSocketJoint.h | 36 ++++++++- src/constraint/Constraint.cpp | 8 +- src/constraint/Constraint.h | 39 ++++++++- src/constraint/ContactPoint.cpp | 15 ++-- src/constraint/ContactPoint.h | 56 ++++++++++++- src/engine/CollisionWorld.cpp | 7 +- src/engine/CollisionWorld.h | 2 +- src/engine/ConstraintSolver.cpp | 8 +- src/engine/ConstraintSolver.h | 25 +++++- src/engine/ContactSolver.cpp | 15 ++-- src/engine/ContactSolver.h | 11 +-- src/engine/DynamicsWorld.cpp | 72 +++++++++++++---- src/engine/DynamicsWorld.h | 78 ++++-------------- 26 files changed, 373 insertions(+), 258 deletions(-) delete mode 100644 src/collision/ContactInfo.cpp delete mode 100644 src/collision/ContactInfo.h create mode 100644 src/collision/narrowphase/SphereVsSphereAlgorithm.cpp.autosave diff --git a/src/collision/CollisionDetection.cpp b/src/collision/CollisionDetection.cpp index e5359860..46773a1c 100644 --- a/src/collision/CollisionDetection.cpp +++ b/src/collision/CollisionDetection.cpp @@ -99,7 +99,7 @@ void CollisionDetection::computeNarrowPhase() { // For each possible collision pair of bodies for (it = mOverlappingPairs.begin(); it != mOverlappingPairs.end(); it++) { - ContactInfo* contactInfo = NULL; + ContactPointInfo* contactInfo = NULL; BroadPhasePair* pair = (*it).second; assert(pair != NULL); @@ -125,12 +125,18 @@ void CollisionDetection::computeNarrowPhase() { contactInfo)) { assert(contactInfo != NULL); + // Set the bodies of the contact + contactInfo->body1 = dynamic_cast(body1); + contactInfo->body2 = dynamic_cast(body2); + assert(contactInfo->body1 != NULL); + assert(contactInfo->body2 != NULL); + // Notify the world about the new narrow-phase contact mWorld->notifyNewContact(pair, contactInfo); // Delete and remove the contact info from the memory allocator - contactInfo->ContactInfo::~ContactInfo(); - mMemoryAllocator.release(contactInfo, sizeof(ContactInfo)); + contactInfo->ContactPointInfo::~ContactPointInfo(); + mMemoryAllocator.release(contactInfo, sizeof(ContactPointInfo)); } } } diff --git a/src/collision/CollisionDetection.h b/src/collision/CollisionDetection.h index 897103f9..2cb0879a 100644 --- a/src/collision/CollisionDetection.h +++ b/src/collision/CollisionDetection.h @@ -33,7 +33,7 @@ #include "narrowphase/GJK/GJKAlgorithm.h" #include "narrowphase/SphereVsSphereAlgorithm.h" #include "../memory/MemoryAllocator.h" -#include "ContactInfo.h" +#include "../constraint/ContactPoint.h" #include #include #include diff --git a/src/collision/ContactInfo.cpp b/src/collision/ContactInfo.cpp deleted file mode 100644 index b3d7486a..00000000 --- a/src/collision/ContactInfo.cpp +++ /dev/null @@ -1,38 +0,0 @@ -/******************************************************************************** -* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * -* Copyright (c) 2010-2013 Daniel Chappuis * -********************************************************************************* -* * -* 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. * -* * -********************************************************************************/ - -// Libraries -#include "ContactInfo.h" - -using namespace reactphysics3d; - - -// Constructor -ContactInfo::ContactInfo(const Vector3& normal, decimal penetrationDepth, - const Vector3& localPoint1, const Vector3& localPoint2) - : normal(normal), penetrationDepth(penetrationDepth), localPoint1(localPoint1), - localPoint2(localPoint2) { - -} diff --git a/src/collision/ContactInfo.h b/src/collision/ContactInfo.h deleted file mode 100644 index a9d8245a..00000000 --- a/src/collision/ContactInfo.h +++ /dev/null @@ -1,81 +0,0 @@ -/******************************************************************************** -* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * -* Copyright (c) 2010-2013 Daniel Chappuis * -********************************************************************************* -* * -* 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 REACTPHYSICS3D_CONTACT_INFO_H -#define REACTPHYSICS3D_CONTACT_INFO_H - -// Libraries -#include "../collision/shapes/BoxShape.h" -#include "../mathematics/mathematics.h" - -// ReactPhysics3D namespace -namespace reactphysics3d { - -// Structure ContactInfo -/** - * This structure contains informations about a collision contact - * computed during the narrow-phase collision detection. Those - * informations are used to compute the contact set for a contact - * between two bodies. - */ -struct ContactInfo { - - private: - - // -------------------- Methods -------------------- // - - /// Private copy-constructor - ContactInfo(const ContactInfo& contactInfo); - - /// Private assignment operator - ContactInfo& operator=(const ContactInfo& contactInfo); - - public: - - // -------------------- Attributes -------------------- // - - /// Normal vector the the collision contact in world space - const Vector3 normal; - - /// Penetration depth of the contact - const decimal penetrationDepth; - - /// Contact point of body 1 in local space of body 1 - const Vector3 localPoint1; - - /// Contact point of body 2 in local space of body 2 - const Vector3 localPoint2; - - // -------------------- Methods -------------------- // - - /// Constructor - ContactInfo(const Vector3& normal, decimal penetrationDepth, - const Vector3& localPoint1, const Vector3& localPoint2); -}; - -} - -#endif - diff --git a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp index d14aff7b..2439979a 100644 --- a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp +++ b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp @@ -87,7 +87,7 @@ bool EPAAlgorithm::computePenetrationDepthAndContactPoints(const Simplex& simple const Transform& transform1, const CollisionShape* collisionShape2, const Transform& transform2, - Vector3& v, ContactInfo*& contactInfo) { + Vector3& v, ContactPointInfo*& contactInfo) { Vector3 suppPointsA[MAX_SUPPORT_POINTS]; // Support points of object A in local coordinates Vector3 suppPointsB[MAX_SUPPORT_POINTS]; // Support points of object B in local coordinates @@ -400,7 +400,7 @@ bool EPAAlgorithm::computePenetrationDepthAndContactPoints(const Simplex& simple assert(penetrationDepth > 0.0); // Create the contact info object - contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo(normal, + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactPointInfo))) ContactPointInfo(normal, penetrationDepth, pALocal, pBLocal); diff --git a/src/collision/narrowphase/EPA/EPAAlgorithm.h b/src/collision/narrowphase/EPA/EPAAlgorithm.h index 5b11a2bf..8f13b8a5 100644 --- a/src/collision/narrowphase/EPA/EPAAlgorithm.h +++ b/src/collision/narrowphase/EPA/EPAAlgorithm.h @@ -29,7 +29,7 @@ // Libraries #include "../GJK/Simplex.h" #include "../../shapes/CollisionShape.h" -#include "../../ContactInfo.h" +#include "../../../constraint/ContactPoint.h" #include "../../../mathematics/mathematics.h" #include "TriangleEPA.h" #include "../../../memory/MemoryAllocator.h" @@ -123,7 +123,7 @@ class EPAAlgorithm { const Transform& transform1, const CollisionShape* collisionShape2, const Transform& transform2, - Vector3& v, ContactInfo*& contactInfo); + Vector3& v, ContactPointInfo*& contactInfo); }; // Add a triangle face in the candidate triangle heap in the EPA algorithm diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.cpp b/src/collision/narrowphase/GJK/GJKAlgorithm.cpp index b03f3953..16e8f671 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.cpp +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.cpp @@ -61,7 +61,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, const Transform& transform1, const CollisionShape* collisionShape2, const Transform& transform2, - ContactInfo*& contactInfo) { + ContactPointInfo*& contactInfo) { Vector3 suppA; // Support point of object A Vector3 suppB; // Support point of object B @@ -137,7 +137,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, if (penetrationDepth <= 0.0) return false; // Create the contact info object - contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo(normal, + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactPointInfo))) ContactPointInfo(normal, penetrationDepth, pA, pB); @@ -169,7 +169,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, if (penetrationDepth <= 0.0) return false; // Create the contact info object - contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo(normal, + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactPointInfo))) ContactPointInfo(normal, penetrationDepth, pA, pB); @@ -199,7 +199,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, if (penetrationDepth <= 0.0) return false; // Create the contact info object - contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo(normal, + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactPointInfo))) ContactPointInfo(normal, penetrationDepth, pA, pB); @@ -236,7 +236,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, if (penetrationDepth <= 0.0) return false; // Create the contact info object - contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo(normal, + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactPointInfo))) ContactPointInfo(normal, penetrationDepth, pA, pB); @@ -263,7 +263,7 @@ bool GJKAlgorithm::computePenetrationDepthForEnlargedObjects(const CollisionShap const Transform& transform1, const CollisionShape* collisionShape2, const Transform& transform2, - ContactInfo*& contactInfo, + ContactPointInfo*& contactInfo, Vector3& v) { Simplex simplex; Vector3 suppA; diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.h b/src/collision/narrowphase/GJK/GJKAlgorithm.h index 5e387b68..5ce89f2e 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.h +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.h @@ -28,7 +28,7 @@ // Libraries #include "../NarrowPhaseAlgorithm.h" -#include "../../ContactInfo.h" +#include "../../../constraint/ContactPoint.h" #include "../../../collision/shapes/CollisionShape.h" #include "../EPA/EPAAlgorithm.h" @@ -78,7 +78,7 @@ class GJKAlgorithm : public NarrowPhaseAlgorithm { const Transform& transform1, const CollisionShape* collisionShape2, const Transform& transform2, - ContactInfo*& contactInfo, Vector3& v); + ContactPointInfo*& contactInfo, Vector3& v); public : @@ -95,7 +95,7 @@ class GJKAlgorithm : public NarrowPhaseAlgorithm { const Transform& transform1, const CollisionShape* collisionShape2, const Transform& transform2, - ContactInfo*& contactInfo); + ContactPointInfo*& contactInfo); }; } diff --git a/src/collision/narrowphase/NarrowPhaseAlgorithm.h b/src/collision/narrowphase/NarrowPhaseAlgorithm.h index 998583a3..79ac19d9 100644 --- a/src/collision/narrowphase/NarrowPhaseAlgorithm.h +++ b/src/collision/narrowphase/NarrowPhaseAlgorithm.h @@ -28,7 +28,7 @@ // Libraries #include "../../body/Body.h" -#include "../ContactInfo.h" +#include "../../constraint/ContactPoint.h" #include "../broadphase/PairManager.h" #include "../../memory/MemoryAllocator.h" #include "../BroadPhasePair.h" @@ -36,7 +36,7 @@ /// Namespace ReactPhysics3D namespace reactphysics3d { - + // Class NarrowPhaseAlgorithm /** * This class is an abstract class that represents an algorithm @@ -82,7 +82,7 @@ class NarrowPhaseAlgorithm { const Transform& transform1, const CollisionShape* collisionShape2, const Transform& transform2, - ContactInfo*& contactInfo)=0; + ContactPointInfo*& contactInfo)=0; }; // Set the current overlapping pair of bodies diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp index 2f950e04..1a67782a 100644 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp @@ -45,7 +45,7 @@ bool SphereVsSphereAlgorithm::testCollision(const CollisionShape* collisionShape const Transform& transform1, const CollisionShape* collisionShape2, const Transform& transform2, - ContactInfo*& contactInfo) { + ContactPointInfo*& contactInfo) { // Get the sphere collision shapes const SphereShape* sphereShape1 = dynamic_cast(collisionShape1); @@ -69,7 +69,7 @@ bool SphereVsSphereAlgorithm::testCollision(const CollisionShape* collisionShape decimal penetrationDepth = sumRadius - std::sqrt(squaredDistanceBetweenCenters); // Create the contact info object - contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactInfo))) ContactInfo( + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactPointInfo))) ContactPointInfo( vectorBetweenCenters.getUnit(), penetrationDepth, intersectionOnBody1, intersectionOnBody2); diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp.autosave b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp.autosave new file mode 100644 index 00000000..a4165d64 --- /dev/null +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp.autosave @@ -0,0 +1,81 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "SphereVsSphereAlgorithm.h" +#include "../../collision/shapes/SphereShape.h" + +// We want to use the ReactPhysics3D namespace +using namespace reactphysics3d; + + +// Constructor +SphereVsSphereAlgorithm::SphereVsSphereAlgorithm(MemoryAllocator& memoryAllocator) + :NarrowPhaseAlgorithm(memoryAllocator) { + +} + +// Destructor +SphereVsSphereAlgorithm::~SphereVsSphereAlgorithm() { + +} + +bool SphereVsSphereAlgorithm::testCollision(const CollisionShape* collisionShape1, + const Transform& transform1, + const CollisionShape* collisionShape2, + const Transform& transform2, + ContactPointInfo*& contactInfo) { + + // Get the sphere collision shapes + const SphereShape* sphereShape1 = dynamic_cast(collisionShape1); + const SphereShape* sphereShape2 = dynamic_cast(collisionShape2); + + // Compute the distance between the centers + Vector3 vectorBetweenCenters = transform2.getPosition() - transform1.getPosition(); + decimal squaredDistanceBetweenCenters = vectorBetweenCenters.lengthSquare(); + + // Compute the sum of the radius + decimal sumRadius = sphereShape1->getRadius() + sphereShape2->getRadius(); + + // If the sphere collision shapes intersect + if (squaredDistanceBetweenCenters <= sumRadius * sumRadius) { + Vector3 centerSphere2InBody1LocalSpace = transform1.inverse() * transform2.getPosition(); + Vector3 centerSphere1InBody2LocalSpace = transform2.inverse() * transform1.getPosition(); + Vector3 intersectionOnBody1 = sphereShape1->getRadius() * + centerSphere2InBody1LocalSpace.getUnit(); + Vector3 intersectionOnBody2 = sphereShape2->getRadius() * + centerSphere1InBody2LocalSpace.getUnit(); + decimal penetrationDepth = sumRadius - std::sqrt(squaredDistanceBetweenCenters); + + // Create the contact info object + contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactPointInfo))) ContactPointInfo( + vectorBetweenCenters.getUnit(), penetrationDepth, + intersectionOnBody1, intersectionOnBody2); + + return true; + } + + return false; +} diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.h b/src/collision/narrowphase/SphereVsSphereAlgorithm.h index e9320619..d47f55aa 100644 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.h +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.h @@ -28,7 +28,7 @@ // Libraries #include "../../body/Body.h" -#include "../ContactInfo.h" +#include "../../constraint/ContactPoint.h" #include "NarrowPhaseAlgorithm.h" @@ -67,7 +67,7 @@ class SphereVsSphereAlgorithm : public NarrowPhaseAlgorithm { const Transform& transform1, const CollisionShape* collisionShape2, const Transform& transform2, - ContactInfo*& contactInfo); + ContactPointInfo*& contactInfo); }; } diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index 51b017c7..17de606d 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -29,9 +29,8 @@ using namespace reactphysics3d; // Constructor -BallAndSocketJoint::BallAndSocketJoint(RigidBody* const body1, RigidBody* const body2, - bool active, ConstraintType type) - : Constraint(body1, body2, active, type){ +BallAndSocketJoint::BallAndSocketJoint(const BallAndSocketJointInfo &jointInfo) + : Constraint(jointInfo){ } diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index 7908dc8f..e7e3f5ea 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -28,9 +28,28 @@ // Libraries #include "Constraint.h" +#include "../mathematics/mathematics.h" namespace reactphysics3d { +// Structure BallAndSocketJointInfo +/** + * This structure is used to gather the information needed to create a ball-and-socket + * joint. This structure will be used to create the actual ball-and-socket joint. + */ +struct BallAndSocketJointInfo : public ConstraintInfo { + + public : + + // -------------------- Attributes -------------------- // + + /// Anchor point (in world space coordinates) + Vector3 anchorPoint; + + /// Constructor + BallAndSocketJointInfo() : ConstraintInfo(BALLSOCKETJOINT) {} +}; + // Class BallAndSocketJoint /** * This class represents a ball-and-socket joint that allows arbitrary rotation @@ -42,18 +61,31 @@ class BallAndSocketJoint : public Constraint { // -------------------- Attributes -------------------- // + /// Anchor point of body 1 (in local space coordinates) + Vector3 mLocalAnchorPoint1; + + /// Anchor point of body 2 (in local space coordinates) + Vector3 mLocalAnchorPoint2; + public : // -------------------- Methods -------------------- // /// Constructor - BallAndSocketJoint(RigidBody* const body1, RigidBody* const body2, - bool active, ConstraintType type); + BallAndSocketJoint(const BallAndSocketJointInfo& jointInfo); /// Destructor virtual ~BallAndSocketJoint(); + + /// Return the number of bytes used by the joint + virtual size_t getSizeInBytes() const; }; +// Return the number of bytes used by the joint +inline size_t BallAndSocketJoint::getSizeInBytes() const { + return sizeof(BallAndSocketJoint); +} + } #endif diff --git a/src/constraint/Constraint.cpp b/src/constraint/Constraint.cpp index a1bd2b76..7ecded94 100644 --- a/src/constraint/Constraint.cpp +++ b/src/constraint/Constraint.cpp @@ -29,10 +29,12 @@ using namespace reactphysics3d; // Constructor -Constraint::Constraint(RigidBody* const body1, RigidBody* const body2, - bool active, ConstraintType type) - :mBody1(body1), mBody2(body2), mActive(active), mType(type) { +Constraint::Constraint(const ConstraintInfo& constraintInfo) + :mBody1(constraintInfo.body1), mBody2(constraintInfo.body2), mActive(true), + mType(constraintInfo.type) { + assert(mBody1 != NULL); + assert(mBody2 != NULL); } // Destructor diff --git a/src/constraint/Constraint.h b/src/constraint/Constraint.h index 9399d84b..de09c956 100644 --- a/src/constraint/Constraint.h +++ b/src/constraint/Constraint.h @@ -36,6 +36,39 @@ namespace reactphysics3d { // Enumeration for the type of a constraint enum ConstraintType {CONTACT, BALLSOCKETJOINT}; +// Structure ConstraintInfo +/** + * This structure is used to gather the information needed to create a constraint. + */ +struct ConstraintInfo { + + public : + + // -------------------- Attributes -------------------- // + + /// First rigid body of the constraint + RigidBody* body1; + + /// Second rigid body of the constraint + RigidBody* body2; + + /// Type of the constraint + ConstraintType type; + + /// Constructor + ConstraintInfo(ConstraintType constraintType) + : body1(NULL), body2(NULL), type(constraintType) {} + + /// Constructor + ConstraintInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, ConstraintType constraintType) + : body1(rigidBody1), body2(rigidBody2), type(constraintType) { + } + + /// Destructor + virtual ~ConstraintInfo() {} + +}; + // Class Constraint /** * This abstract class represents a constraint in the physics engine. @@ -74,8 +107,7 @@ class Constraint { // -------------------- Methods -------------------- // /// Constructor - Constraint(RigidBody* const body1, RigidBody* const body2, - bool active, ConstraintType type); + Constraint(const ConstraintInfo& constraintInfo); /// Destructor virtual ~Constraint(); @@ -91,6 +123,9 @@ class Constraint { /// Return the type of the constraint ConstraintType getType() const; + + /// Return the number of bytes used by the constraint + virtual size_t getSizeInBytes() const = 0; }; // Return the reference to the body 1 diff --git a/src/constraint/ContactPoint.cpp b/src/constraint/ContactPoint.cpp index df38e269..93a7ec3d 100644 --- a/src/constraint/ContactPoint.cpp +++ b/src/constraint/ContactPoint.cpp @@ -30,14 +30,13 @@ using namespace reactphysics3d; using namespace std; // Constructor -ContactPoint::ContactPoint(RigidBody* const body1, RigidBody* const body2, - const ContactInfo* contactInfo) - : Constraint(body1, body2, true, CONTACT), mNormal(contactInfo->normal), - mPenetrationDepth(contactInfo->penetrationDepth), - mLocalPointOnBody1(contactInfo->localPoint1), - mLocalPointOnBody2(contactInfo->localPoint2), - mWorldPointOnBody1(body1->getTransform() * contactInfo->localPoint1), - mWorldPointOnBody2(body2->getTransform() * contactInfo->localPoint2), +ContactPoint::ContactPoint(const ContactPointInfo& contactInfo) + : Constraint(contactInfo), mNormal(contactInfo.normal), + mPenetrationDepth(contactInfo.penetrationDepth), + mLocalPointOnBody1(contactInfo.localPoint1), + mLocalPointOnBody2(contactInfo.localPoint2), + mWorldPointOnBody1(contactInfo.body1->getTransform() * contactInfo.localPoint1), + mWorldPointOnBody2(contactInfo.body2->getTransform() * contactInfo.localPoint2), mIsRestingContact(false) { mFrictionVectors[0] = Vector3(0, 0, 0); diff --git a/src/constraint/ContactPoint.h b/src/constraint/ContactPoint.h index 8ad4eff9..14852282 100644 --- a/src/constraint/ContactPoint.h +++ b/src/constraint/ContactPoint.h @@ -28,7 +28,6 @@ // Libraries #include "Constraint.h" -#include "../collision/ContactInfo.h" #include "../body/RigidBody.h" #include "../configuration.h" #include "../mathematics/mathematics.h" @@ -50,6 +49,52 @@ /// ReactPhysics3D namespace namespace reactphysics3d { +// Structure ContactPointInfo +/** + * This structure contains informations about a collision contact + * computed during the narrow-phase collision detection. Those + * informations are used to compute the contact set for a contact + * between two bodies. + */ +struct ContactPointInfo : public ConstraintInfo { + + private: + + // -------------------- Methods -------------------- // + + /// Private copy-constructor + ContactPointInfo(const ContactPointInfo& contactInfo); + + /// Private assignment operator + ContactPointInfo& operator=(const ContactPointInfo& contactInfo); + + public: + + // -------------------- Attributes -------------------- // + + /// Normal vector the the collision contact in world space + const Vector3 normal; + + /// Penetration depth of the contact + const decimal penetrationDepth; + + /// Contact point of body 1 in local space of body 1 + const Vector3 localPoint1; + + /// Contact point of body 2 in local space of body 2 + const Vector3 localPoint2; + + // -------------------- Methods -------------------- // + + /// Constructor + ContactPointInfo(const Vector3& normal, decimal penetrationDepth, + const Vector3& localPoint1, const Vector3& localPoint2) + : ConstraintInfo(CONTACT), normal(normal), penetrationDepth(penetrationDepth), + localPoint1(localPoint1), localPoint2(localPoint2) { + + } +}; + // Class ContactPoint /** * This class represents a collision contact point between two @@ -108,7 +153,7 @@ class ContactPoint : public Constraint { // -------------------- Methods -------------------- // /// Constructor - ContactPoint(RigidBody* const body1, RigidBody* const body2, const ContactInfo* contactInfo); + ContactPoint(const ContactPointInfo& contactInfo); /// Destructor virtual ~ContactPoint(); @@ -176,6 +221,9 @@ class ContactPoint : public Constraint { /// Return the penetration depth decimal getPenetrationDepth() const; + /// Return the number of bytes used by the contact point + virtual size_t getSizeInBytes() const; + #ifdef VISUAL_DEBUG /// Draw the contact (for debugging) void draw() const; @@ -287,6 +335,10 @@ inline decimal ContactPoint::getPenetrationDepth() const { return mPenetrationDepth; } +// Return the number of bytes used by the contact point +inline size_t ContactPoint::getSizeInBytes() const { + return sizeof(ContactPoint); +} #ifdef VISUAL_DEBUG inline void ContactPoint::draw() const { diff --git a/src/engine/CollisionWorld.cpp b/src/engine/CollisionWorld.cpp index 3a048d09..80e73e87 100644 --- a/src/engine/CollisionWorld.cpp +++ b/src/engine/CollisionWorld.cpp @@ -56,7 +56,7 @@ void CollisionWorld::notifyRemovedOverlappingPair(const BroadPhasePair* removedP // Notify the world about a new narrow-phase contact void CollisionWorld::notifyNewContact(const BroadPhasePair* broadPhasePair, - const ContactInfo* contactInfo) { + const ContactPointInfo* contactInfo) { // TODO : Implement this method } @@ -101,7 +101,7 @@ void CollisionWorld::destroyCollisionBody(CollisionBody* collisionBody) { // Add the body ID to the list of free IDs mFreeBodiesIDs.push_back(collisionBody->getID()); - // Call the constructor of the collision body + // Call the destructor of the collision body collisionBody->CollisionBody::~CollisionBody(); // Remove the collision body from the list of bodies @@ -181,6 +181,9 @@ void CollisionWorld::removeCollisionShape(CollisionShape* collisionShape) { // Remove the shape from the set of shapes in the world mCollisionShapes.remove(collisionShape); + // Call the destructor of the collision shape + collisionShape->CollisionShape::~CollisionShape(); + // Deallocate the memory used by the collision shape mMemoryAllocator.release(collisionShape, collisionShape->getSizeInBytes()); } diff --git a/src/engine/CollisionWorld.h b/src/engine/CollisionWorld.h index a120956b..89f38c5c 100644 --- a/src/engine/CollisionWorld.h +++ b/src/engine/CollisionWorld.h @@ -91,7 +91,7 @@ class CollisionWorld { virtual void notifyRemovedOverlappingPair(const BroadPhasePair* removedPair); /// Notify the world about a new narrow-phase contact - virtual void notifyNewContact(const BroadPhasePair* pair, const ContactInfo* contactInfo); + virtual void notifyNewContact(const BroadPhasePair* pair, const ContactPointInfo* contactInfo); /// Update the overlapping pair virtual void updateOverlappingPair(const BroadPhasePair* pair); diff --git a/src/engine/ConstraintSolver.cpp b/src/engine/ConstraintSolver.cpp index a0131bac..5fc41c2a 100644 --- a/src/engine/ConstraintSolver.cpp +++ b/src/engine/ConstraintSolver.cpp @@ -29,7 +29,13 @@ using namespace reactphysics3d; // Constructor -ConstraintSolver::ConstraintSolver() { +ConstraintSolver::ConstraintSolver(std::set& joints, + std::vector& constrainedLinearVelocities, + std::vector& constrainedAngularVelocities, + const std::map& mapBodyToVelocityIndex) + : mJoints(joints), mConstrainedLinearVelocities(constrainedLinearVelocities), + mConstrainedAngularVelocities(constrainedAngularVelocities), + mMapBodyToConstrainedVelocityIndex(mapBodyToVelocityIndex) { } diff --git a/src/engine/ConstraintSolver.h b/src/engine/ConstraintSolver.h index 227c4c9a..5dce0686 100644 --- a/src/engine/ConstraintSolver.h +++ b/src/engine/ConstraintSolver.h @@ -27,6 +27,11 @@ #define REACTPHYSICS3D_CONSTRAINT_SOLVER_H // Libraries +#include "../configuration.h" +#include "mathematics/mathematics.h" +#include "../constraint/Constraint.h" +#include +#include namespace reactphysics3d { @@ -105,6 +110,20 @@ class ConstraintSolver { // -------------------- Attributes -------------------- // + /// Reference to all the joints of the world + std::set& mJoints; + + /// Reference to the array of constrained linear velocities (state of the linear velocities + /// after solving the constraints) + std::vector& mConstrainedLinearVelocities; + + /// Reference to the array of constrained angular velocities (state of the angular velocities + /// after solving the constraints) + std::vector& mConstrainedAngularVelocities; + + /// Reference to the map of rigid body to their index in the constrained velocities array + const std::map& mMapBodyToConstrainedVelocityIndex; + /// Number of iterations of the contact solver uint mNbIterations; @@ -116,11 +135,13 @@ class ConstraintSolver { // -------------------- Methods -------------------- // /// Constructor - ConstraintSolver(); + ConstraintSolver(std::set& joints, + std::vector& constrainedLinearVelocities, + std::vector& constrainedAngularVelocities, + const std::map& mapBodyToVelocityIndex); /// Destructor ~ConstraintSolver(); - }; } diff --git a/src/engine/ContactSolver.cpp b/src/engine/ContactSolver.cpp index 7d786588..94d58943 100644 --- a/src/engine/ContactSolver.cpp +++ b/src/engine/ContactSolver.cpp @@ -39,10 +39,12 @@ const decimal ContactSolver::BETA_SPLIT_IMPULSE = decimal(0.2); const decimal ContactSolver::SLOP = decimal(0.01); // Constructor -ContactSolver::ContactSolver(DynamicsWorld& world,std::vector& constrainedLinearVelocities, +ContactSolver::ContactSolver(std::vector& contactManifolds, + std::vector& constrainedLinearVelocities, std::vector& constrainedAngularVelocities, const std::map& mapBodyToVelocityIndex) - :mWorld(world), mNbIterations(DEFAULT_CONSTRAINTS_SOLVER_NB_ITERATIONS), + :mContactManifolds(contactManifolds), + mNbIterations(DEFAULT_CONSTRAINTS_SOLVER_NB_ITERATIONS), mSplitLinearVelocities(NULL), mSplitAngularVelocities(NULL), mContactConstraints(NULL), mConstrainedLinearVelocities(constrainedLinearVelocities), @@ -62,14 +64,13 @@ ContactSolver::~ContactSolver() { void ContactSolver::initialize() { // TODO : Use better memory allocation here - mContactConstraints = new ContactManifoldSolver[mWorld.getNbContactManifolds()]; + mContactConstraints = new ContactManifoldSolver[mContactManifolds.size()]; mNbContactManifolds = 0; // For each contact manifold of the world vector::iterator it; - for (it = mWorld.getContactManifoldsBeginIterator(); - it != mWorld.getContactManifoldsEndIterator(); ++it) { + for (it = mContactManifolds.begin(); it != mContactManifolds.end(); ++it) { ContactManifold* externalManifold = *it; @@ -174,8 +175,8 @@ void ContactSolver::initialize() { // Allocated memory for split impulse velocities // TODO : Use better memory allocation here - mSplitLinearVelocities = new Vector3[mWorld.getNbRigidBodies()]; - mSplitAngularVelocities = new Vector3[mWorld.getNbRigidBodies()]; + mSplitLinearVelocities = new Vector3[mMapBodyToConstrainedVelocityIndex.size()]; + mSplitAngularVelocities = new Vector3[mMapBodyToConstrainedVelocityIndex.size()]; assert(mSplitLinearVelocities != NULL); assert(mSplitAngularVelocities != NULL); diff --git a/src/engine/ContactSolver.h b/src/engine/ContactSolver.h index 1153f306..97ec39b5 100644 --- a/src/engine/ContactSolver.h +++ b/src/engine/ContactSolver.h @@ -342,8 +342,8 @@ class ContactSolver { // -------------------- Attributes -------------------- // - /// Reference to the world - DynamicsWorld& mWorld; + /// Reference to all the contact manifold of the world + std::vector& mContactManifolds; /// Number of iterations of the contact solver uint mNbIterations; @@ -366,11 +366,11 @@ class ContactSolver { /// Constrained bodies std::set mConstraintBodies; - /// Pointer to the array of constrained linear velocities (state of the linear velocities + /// Reference to the array of constrained linear velocities (state of the linear velocities /// after solving the constraints) std::vector& mConstrainedLinearVelocities; - /// Pointer to the array of constrained angular velocities (state of the angular velocities + /// Reference to the array of constrained angular velocities (state of the angular velocities /// after solving the constraints) std::vector& mConstrainedAngularVelocities; @@ -452,7 +452,8 @@ class ContactSolver { // -------------------- Methods -------------------- // /// Constructor - ContactSolver(DynamicsWorld& mWorld, std::vector& constrainedLinearVelocities, + ContactSolver(std::vector& contactManifolds, + std::vector& constrainedLinearVelocities, std::vector& constrainedAngularVelocities, const std::map& mapBodyToVelocityIndex); diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 13d79138..dd319483 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -25,6 +25,7 @@ // Libraries #include "DynamicsWorld.h" +#include "constraint/BallAndSocketJoint.h" // Namespaces using namespace reactphysics3d; @@ -33,8 +34,10 @@ using namespace std; // Constructor DynamicsWorld::DynamicsWorld(const Vector3 &gravity, decimal timeStep = DEFAULT_TIMESTEP) : CollisionWorld(), mTimer(timeStep), mGravity(gravity), mIsGravityOn(true), - mContactSolver(*this, mConstrainedLinearVelocities, mConstrainedAngularVelocities, + mContactSolver(mContactManifolds, mConstrainedLinearVelocities, mConstrainedAngularVelocities, mMapBodyToConstrainedVelocityIndex), + mConstraintSolver(mJoints, mConstrainedLinearVelocities, mConstrainedAngularVelocities, + mMapBodyToConstrainedVelocityIndex), mIsDeactivationActive(DEACTIVATION_ENABLED) { } @@ -221,6 +224,8 @@ void DynamicsWorld::initConstrainedVelocitiesArray() { i++; } + + assert(mMapBodyToConstrainedVelocityIndex.size() == mRigidBodies.size()); } // Cleanup the constrained velocities array at each step @@ -230,6 +235,9 @@ void DynamicsWorld::cleanupConstrainedVelocitiesArray() { mConstrainedLinearVelocities.clear(); mConstrainedAngularVelocities.clear(); + // Clear the constrained bodies + mConstrainedBodies.clear(); + // Clear the rigid body to velocities array index mapping mMapBodyToConstrainedVelocityIndex.clear(); } @@ -300,7 +308,7 @@ void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { // Remove the collision shape from the world removeCollisionShape(rigidBody->getCollisionShape()); - // Call the constructor of the rigid body + // Call the destructor of the rigid body rigidBody->RigidBody::~RigidBody(); // Remove the rigid body from the list of rigid bodies @@ -311,9 +319,51 @@ void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { mMemoryAllocator.release(rigidBody, sizeof(RigidBody)); } -// Remove all constraints in the physics world -void DynamicsWorld::removeAllConstraints() { - mConstraints.clear(); +// Create a joint between two bodies in the world and return a pointer to the new joint +Constraint* DynamicsWorld::createJoint(const ConstraintInfo& jointInfo) { + + Constraint* newJoint = NULL; + + // Allocate memory to create the new joint + switch(jointInfo.type) { + + // Ball-and-Socket joint + case BALLSOCKETJOINT: + { + void* allocatedMemory = mMemoryAllocator.allocate(sizeof(BallAndSocketJoint)); + const BallAndSocketJointInfo& info = dynamic_cast( + jointInfo); + newJoint = new (allocatedMemory) BallAndSocketJoint(info); + break; + } + + default: + { + assert(false); + return NULL; + } + } + + // Add the joint into the world + mJoints.insert(newJoint); + + // Return the pointer to the created joint + return newJoint; +} + +// Destroy a joint +void DynamicsWorld::destroyJoint(Constraint* joint) { + + assert(joint != NULL); + + // Remove the joint from the world + mJoints.erase(joint); + + // Call the destructor of the joint + joint->Constraint::~Constraint(); + + // Release the allocated memory + mMemoryAllocator.release(joint, joint->getSizeInBytes()); } // Notify the world about a new broad-phase overlapping pair @@ -345,19 +395,11 @@ void DynamicsWorld::notifyRemovedOverlappingPair(const BroadPhasePair* removedPa // Notify the world about a new narrow-phase contact void DynamicsWorld::notifyNewContact(const BroadPhasePair* broadPhasePair, - const ContactInfo* contactInfo) { - - RigidBody* const rigidBody1 = dynamic_cast(broadPhasePair->body1); - RigidBody* const rigidBody2 = dynamic_cast(broadPhasePair->body2); - - assert(rigidBody1 != NULL); - assert(rigidBody2 != NULL); + const ContactPointInfo* contactInfo) { // Create a new contact ContactPoint* contact = new (mMemoryAllocator.allocate(sizeof(ContactPoint))) ContactPoint( - rigidBody1, - rigidBody2, - contactInfo); + *contactInfo); assert(contact != NULL); // Get the corresponding overlapping pair diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index 93b87bf2..de1edd7c 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -30,6 +30,7 @@ #include "CollisionWorld.h" #include "../collision/CollisionDetection.h" #include "ContactSolver.h" +#include "ConstraintSolver.h" #include "../body/RigidBody.h" #include "Timer.h" #include "../configuration.h" @@ -55,6 +56,9 @@ class DynamicsWorld : public CollisionWorld { /// Contact solver ContactSolver mContactSolver; + /// Constraint solver + ConstraintSolver mConstraintSolver; + /// True if the deactivation (sleeping) of inactive bodies is enabled bool mIsDeactivationActive; @@ -64,8 +68,11 @@ class DynamicsWorld : public CollisionWorld { /// All the contact constraints std::vector mContactManifolds; - /// All the constraints (except contact constraints) - std::vector mConstraints; + /// All the joints of the world + std::set mJoints; + + /// All the bodies that are part of contacts or constraints + std::set mConstrainedBodies; /// Gravity vector of the world Vector3 mGravity; @@ -124,7 +131,7 @@ class DynamicsWorld : public CollisionWorld { virtual void notifyRemovedOverlappingPair(const BroadPhasePair* removedPair); /// Notify the world about a new narrow-phase contact - virtual void notifyNewContact(const BroadPhasePair* pair, const ContactInfo* contactInfo); + virtual void notifyNewContact(const BroadPhasePair* pair, const ContactPointInfo* contactInfo); public : @@ -166,6 +173,12 @@ public : /// Destroy a rigid body void destroyRigidBody(RigidBody* rigidBody); + /// Create a joint between two bodies in the world and return a pointer to the new joint + Constraint* createJoint(const ConstraintInfo& jointInfo); + + /// Destroy a joint + void destroyJoint(Constraint* joint); + /// Return the gravity vector of the world Vector3 getGravity() const; @@ -178,30 +191,9 @@ public : /// Return the number of rigid bodies in the world uint getNbRigidBodies() const; - /// Add a constraint - void addConstraint(Constraint* constraint); - - /// Remove a constraint - void removeConstraint(Constraint* constraint); - - /// Remove all constraints and delete them (free their memory) - void removeAllConstraints(); - /// Return the number of contact constraints in the world uint getNbContactManifolds() const; - /// Return a start iterator on the constraint list - std::vector::iterator getConstraintsBeginIterator(); - - /// Return a end iterator on the constraint list - std::vector::iterator getConstraintsEndIterator(); - - /// Return a start iterator on the contact manifolds list - std::vector::iterator getContactManifoldsBeginIterator(); - - /// Return a end iterator on the contact manifolds list - std::vector::iterator getContactManifoldsEndIterator(); - /// Return an iterator to the beginning of the rigid bodies of the physics world std::set::iterator getRigidBodiesBeginIterator(); @@ -259,24 +251,6 @@ inline void DynamicsWorld::updateOverlappingPair(const BroadPhasePair* pair) { overlappingPair->update(); } - -// Add a constraint into the physics world -inline void DynamicsWorld::addConstraint(Constraint* constraint) { - assert(constraint != 0); - mConstraints.push_back(constraint); -} - -// Remove a constraint and free its memory -inline void DynamicsWorld::removeConstraint(Constraint* constraint) { - std::vector::iterator it; - - assert(constraint != NULL); - it = std::find(mConstraints.begin(), mConstraints.end(), constraint); - assert(*it == constraint); - delete *it; - mConstraints.erase(it); -} - // Return the gravity vector of the world inline Vector3 DynamicsWorld::getGravity() const { return mGravity; @@ -312,26 +286,6 @@ inline uint DynamicsWorld::getNbContactManifolds() const { return mContactManifolds.size(); } -// Return a start iterator on the constraint list -inline std::vector::iterator DynamicsWorld::getConstraintsBeginIterator() { - return mConstraints.begin(); -} - -// Return a end iterator on the constraint list -inline std::vector::iterator DynamicsWorld::getConstraintsEndIterator() { - return mConstraints.end(); -} - -// Return a start iterator on the contact manifolds list -inline std::vector::iterator DynamicsWorld::getContactManifoldsBeginIterator() { - return mContactManifolds.begin(); -} - -// Return a end iterator on the contact manifolds list -inline std::vector::iterator DynamicsWorld::getContactManifoldsEndIterator() { - return mContactManifolds.end(); -} - } #endif From ded465c10508f902fdac58506a841e11b97bbd85 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 24 Apr 2013 22:29:31 +0200 Subject: [PATCH 13/66] Remove a file --- .../SphereVsSphereAlgorithm.cpp.autosave | 81 ------------------- 1 file changed, 81 deletions(-) delete mode 100644 src/collision/narrowphase/SphereVsSphereAlgorithm.cpp.autosave diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp.autosave b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp.autosave deleted file mode 100644 index a4165d64..00000000 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp.autosave +++ /dev/null @@ -1,81 +0,0 @@ -/******************************************************************************** -* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * -* Copyright (c) 2010-2013 Daniel Chappuis * -********************************************************************************* -* * -* 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. * -* * -********************************************************************************/ - -// Libraries -#include "SphereVsSphereAlgorithm.h" -#include "../../collision/shapes/SphereShape.h" - -// We want to use the ReactPhysics3D namespace -using namespace reactphysics3d; - - -// Constructor -SphereVsSphereAlgorithm::SphereVsSphereAlgorithm(MemoryAllocator& memoryAllocator) - :NarrowPhaseAlgorithm(memoryAllocator) { - -} - -// Destructor -SphereVsSphereAlgorithm::~SphereVsSphereAlgorithm() { - -} - -bool SphereVsSphereAlgorithm::testCollision(const CollisionShape* collisionShape1, - const Transform& transform1, - const CollisionShape* collisionShape2, - const Transform& transform2, - ContactPointInfo*& contactInfo) { - - // Get the sphere collision shapes - const SphereShape* sphereShape1 = dynamic_cast(collisionShape1); - const SphereShape* sphereShape2 = dynamic_cast(collisionShape2); - - // Compute the distance between the centers - Vector3 vectorBetweenCenters = transform2.getPosition() - transform1.getPosition(); - decimal squaredDistanceBetweenCenters = vectorBetweenCenters.lengthSquare(); - - // Compute the sum of the radius - decimal sumRadius = sphereShape1->getRadius() + sphereShape2->getRadius(); - - // If the sphere collision shapes intersect - if (squaredDistanceBetweenCenters <= sumRadius * sumRadius) { - Vector3 centerSphere2InBody1LocalSpace = transform1.inverse() * transform2.getPosition(); - Vector3 centerSphere1InBody2LocalSpace = transform2.inverse() * transform1.getPosition(); - Vector3 intersectionOnBody1 = sphereShape1->getRadius() * - centerSphere2InBody1LocalSpace.getUnit(); - Vector3 intersectionOnBody2 = sphereShape2->getRadius() * - centerSphere1InBody2LocalSpace.getUnit(); - decimal penetrationDepth = sumRadius - std::sqrt(squaredDistanceBetweenCenters); - - // Create the contact info object - contactInfo = new (mMemoryAllocator.allocate(sizeof(ContactPointInfo))) ContactPointInfo( - vectorBetweenCenters.getUnit(), penetrationDepth, - intersectionOnBody1, intersectionOnBody2); - - return true; - } - - return false; -} From fdda0b26a92c603e8b52b6b9aa4f94fddf3b29d7 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 25 Apr 2013 22:34:20 +0200 Subject: [PATCH 14/66] Modify the contact solver so that its main loop is outside the solver --- examples/fallingcubes/FallingCubes.cpp | 6 +- src/engine/CollisionWorld.cpp | 6 +- src/engine/ConstraintSolver.cpp | 17 ++ src/engine/ConstraintSolver.h | 6 + src/engine/ContactSolver.cpp | 405 ++++++++++++------------- src/engine/ContactSolver.h | 38 +-- src/engine/DynamicsWorld.cpp | 72 +++-- src/engine/DynamicsWorld.h | 19 +- 8 files changed, 301 insertions(+), 268 deletions(-) diff --git a/examples/fallingcubes/FallingCubes.cpp b/examples/fallingcubes/FallingCubes.cpp index 65f8f485..54bbfa62 100644 --- a/examples/fallingcubes/FallingCubes.cpp +++ b/examples/fallingcubes/FallingCubes.cpp @@ -66,12 +66,11 @@ int main(int argc, char** argv) { glutMouseFunc(mouseButton); glutMotionFunc(mouseMotion); glutKeyboardFunc(keyboard); + glutCloseFunc(finish); // Glut main looop glutMainLoop(); - finish(); - return 0; } @@ -115,8 +114,7 @@ void keyboard(unsigned char key, int x, int y) { // Escape key case 27: - finish(); - exit(0); + glutLeaveMainLoop(); break; // Space bar diff --git a/src/engine/CollisionWorld.cpp b/src/engine/CollisionWorld.cpp index 80e73e87..2f2d4d81 100644 --- a/src/engine/CollisionWorld.cpp +++ b/src/engine/CollisionWorld.cpp @@ -154,6 +154,7 @@ CollisionShape* CollisionWorld::createCollisionShape(const CollisionShape& colli // A similar collision shape does not already exist in the world, so we create a // new one and add it to the world void* allocatedMemory = mMemoryAllocator.allocate(collisionShape.getSizeInBytes()); + size_t test = collisionShape.getSizeInBytes(); CollisionShape* newCollisionShape = collisionShape.clone(allocatedMemory); mCollisionShapes.push_back(newCollisionShape); @@ -181,11 +182,14 @@ void CollisionWorld::removeCollisionShape(CollisionShape* collisionShape) { // Remove the shape from the set of shapes in the world mCollisionShapes.remove(collisionShape); + // Compute the size (in bytes) of the collision shape + size_t nbBytesShape = collisionShape->getSizeInBytes(); + // Call the destructor of the collision shape collisionShape->CollisionShape::~CollisionShape(); // Deallocate the memory used by the collision shape - mMemoryAllocator.release(collisionShape, collisionShape->getSizeInBytes()); + mMemoryAllocator.release(collisionShape, nbBytesShape); } } diff --git a/src/engine/ConstraintSolver.cpp b/src/engine/ConstraintSolver.cpp index 5fc41c2a..0291eee4 100644 --- a/src/engine/ConstraintSolver.cpp +++ b/src/engine/ConstraintSolver.cpp @@ -25,6 +25,7 @@ // Libraries #include "ConstraintSolver.h" +#include "Profiler.h" using namespace reactphysics3d; @@ -43,3 +44,19 @@ ConstraintSolver::ConstraintSolver(std::set& joints, ConstraintSolver::~ConstraintSolver() { } + +// Initialize the constraint solver +void ConstraintSolver::initialize(decimal dt) { + + PROFILE("ConstraintSolver::initialize()"); + + // Set the current time step + mTimeStep = dt; +} + +// Solve the constraints +void ConstraintSolver::solve() { + + PROFILE("ConstraintSolver::solve()"); + +} diff --git a/src/engine/ConstraintSolver.h b/src/engine/ConstraintSolver.h index 5dce0686..08e4a08f 100644 --- a/src/engine/ConstraintSolver.h +++ b/src/engine/ConstraintSolver.h @@ -142,6 +142,12 @@ class ConstraintSolver { /// Destructor ~ConstraintSolver(); + + /// Initialize the constraint solver + void initialize(decimal dt); + + /// Solve the constraints + void solve(); }; } diff --git a/src/engine/ContactSolver.cpp b/src/engine/ContactSolver.cpp index 94d58943..3399ac0d 100644 --- a/src/engine/ContactSolver.cpp +++ b/src/engine/ContactSolver.cpp @@ -44,7 +44,6 @@ ContactSolver::ContactSolver(std::vector& contactManifolds, std::vector& constrainedAngularVelocities, const std::map& mapBodyToVelocityIndex) :mContactManifolds(contactManifolds), - mNbIterations(DEFAULT_CONSTRAINTS_SOLVER_NB_ITERATIONS), mSplitLinearVelocities(NULL), mSplitAngularVelocities(NULL), mContactConstraints(NULL), mConstrainedLinearVelocities(constrainedLinearVelocities), @@ -61,7 +60,12 @@ ContactSolver::~ContactSolver() { } // Initialize the constraint solver -void ContactSolver::initialize() { +void ContactSolver::initialize(decimal dt) { + + PROFILE("ContactSolver::initialize()"); + + // Set the current time step + mTimeStep = dt; // TODO : Use better memory allocation here mContactConstraints = new ContactManifoldSolver[mContactManifolds.size()]; @@ -187,6 +191,9 @@ void ContactSolver::initialize() { // Initialize the split impulse velocities initializeSplitImpulseVelocities(); + + // Fill-in all the matrices needed to solve the LCP problem + initializeContactConstraints(); } // Initialize the split impulse velocities @@ -380,6 +387,9 @@ void ContactSolver::initializeContactConstraints() { /// the solution of the linear system void ContactSolver::warmStart() { + // Check that warm starting is active + if (!mIsWarmStartingActive) return; + // For each constraint for (uint c=0; c SLOP) biasPenetrationDepth = -(beta/mTimeStep) * + max(0.0f, float(contactPoint.penetrationDepth - SLOP)); + decimal b = biasPenetrationDepth + contactPoint.restitutionBias; - // Compute the bias "b" of the constraint - decimal beta = mIsSplitImpulseActive ? BETA_SPLIT_IMPULSE : BETA; - decimal biasPenetrationDepth = 0.0; - if (contactPoint.penetrationDepth > SLOP) biasPenetrationDepth = -(beta/mTimeStep) * - max(0.0f, float(contactPoint.penetrationDepth - SLOP)); - decimal b = biasPenetrationDepth + contactPoint.restitutionBias; + // Compute the Lagrange multiplier lambda + if (mIsSplitImpulseActive) { + deltaLambda = - (Jv + contactPoint.restitutionBias) * + contactPoint.inversePenetrationMass; + } + else { + deltaLambda = - (Jv + b) * contactPoint.inversePenetrationMass; + } + lambdaTemp = contactPoint.penetrationImpulse; + contactPoint.penetrationImpulse = std::max(contactPoint.penetrationImpulse + + deltaLambda, decimal(0.0)); + deltaLambda = contactPoint.penetrationImpulse - lambdaTemp; - // Compute the Lagrange multiplier lambda - if (mIsSplitImpulseActive) { - deltaLambda = - (Jv + contactPoint.restitutionBias) * - contactPoint.inversePenetrationMass; - } - else { - deltaLambda = - (Jv + b) * contactPoint.inversePenetrationMass; - } - lambdaTemp = contactPoint.penetrationImpulse; - contactPoint.penetrationImpulse = std::max(contactPoint.penetrationImpulse + - deltaLambda, decimal(0.0)); - deltaLambda = contactPoint.penetrationImpulse - lambdaTemp; + // Compute the impulse P=J^T * lambda + const Impulse impulsePenetration = computePenetrationImpulse(deltaLambda, + contactPoint); + + // Apply the impulse to the bodies of the constraint + applyImpulse(impulsePenetration, contactManifold); + + sumPenetrationImpulse += contactPoint.penetrationImpulse; + + // If the split impulse position correction is active + if (mIsSplitImpulseActive) { + + // Split impulse (position correction) + const Vector3& v1Split = mSplitLinearVelocities[contactManifold.indexBody1]; + const Vector3& w1Split = mSplitAngularVelocities[contactManifold.indexBody1]; + const Vector3& v2Split = mSplitLinearVelocities[contactManifold.indexBody2]; + const Vector3& w2Split = mSplitAngularVelocities[contactManifold.indexBody2]; + Vector3 deltaVSplit = v2Split + w2Split.cross(contactPoint.r2) - + v1Split - w1Split.cross(contactPoint.r1); + decimal JvSplit = deltaVSplit.dot(contactPoint.normal); + decimal deltaLambdaSplit = - (JvSplit + biasPenetrationDepth) * + contactPoint.inversePenetrationMass; + decimal lambdaTempSplit = contactPoint.penetrationSplitImpulse; + contactPoint.penetrationSplitImpulse = std::max( + contactPoint.penetrationSplitImpulse + + deltaLambdaSplit, decimal(0.0)); + deltaLambda = contactPoint.penetrationSplitImpulse - lambdaTempSplit; // Compute the impulse P=J^T * lambda - const Impulse impulsePenetration = computePenetrationImpulse(deltaLambda, - contactPoint); + const Impulse splitImpulsePenetration = computePenetrationImpulse( + deltaLambdaSplit, contactPoint); - // Apply the impulse to the bodies of the constraint - applyImpulse(impulsePenetration, contactManifold); - - sumPenetrationImpulse += contactPoint.penetrationImpulse; - - // If the split impulse position correction is active - if (mIsSplitImpulseActive) { - - // Split impulse (position correction) - const Vector3& v1Split = mSplitLinearVelocities[contactManifold.indexBody1]; - const Vector3& w1Split = mSplitAngularVelocities[contactManifold.indexBody1]; - const Vector3& v2Split = mSplitLinearVelocities[contactManifold.indexBody2]; - const Vector3& w2Split = mSplitAngularVelocities[contactManifold.indexBody2]; - Vector3 deltaVSplit = v2Split + w2Split.cross(contactPoint.r2) - - v1Split - w1Split.cross(contactPoint.r1); - decimal JvSplit = deltaVSplit.dot(contactPoint.normal); - decimal deltaLambdaSplit = - (JvSplit + biasPenetrationDepth) * - contactPoint.inversePenetrationMass; - decimal lambdaTempSplit = contactPoint.penetrationSplitImpulse; - contactPoint.penetrationSplitImpulse = std::max( - contactPoint.penetrationSplitImpulse + - deltaLambdaSplit, decimal(0.0)); - deltaLambda = contactPoint.penetrationSplitImpulse - lambdaTempSplit; - - // Compute the impulse P=J^T * lambda - const Impulse splitImpulsePenetration = computePenetrationImpulse( - deltaLambdaSplit, contactPoint); - - applySplitImpulse(splitImpulsePenetration, contactManifold); - } - - // If we do not solve the friction constraints at the center of the contact manifold - if (!mIsSolveFrictionAtContactManifoldCenterActive) { - - // --------- Friction 1 --------- // - - // Compute J*v - deltaV = v2 + w2.cross(contactPoint.r2) - v1 - w1.cross(contactPoint.r1); - Jv = deltaV.dot(contactPoint.frictionVector1); - - // Compute the Lagrange multiplier lambda - deltaLambda = -Jv; - deltaLambda *= contactPoint.inverseFriction1Mass; - decimal frictionLimit = contactManifold.frictionCoefficient * - contactPoint.penetrationImpulse; - lambdaTemp = contactPoint.friction1Impulse; - contactPoint.friction1Impulse = std::max(-frictionLimit, - std::min(contactPoint.friction1Impulse - + deltaLambda, frictionLimit)); - deltaLambda = contactPoint.friction1Impulse - lambdaTemp; - - // Compute the impulse P=J^T * lambda - const Impulse impulseFriction1 = computeFriction1Impulse(deltaLambda, - contactPoint); - - // Apply the impulses to the bodies of the constraint - applyImpulse(impulseFriction1, contactManifold); - - // --------- Friction 2 --------- // - - // Compute J*v - deltaV = v2 + w2.cross(contactPoint.r2) - v1 - w1.cross(contactPoint.r1); - Jv = deltaV.dot(contactPoint.frictionVector2); - - // Compute the Lagrange multiplier lambda - deltaLambda = -Jv; - deltaLambda *= contactPoint.inverseFriction2Mass; - frictionLimit = contactManifold.frictionCoefficient * - contactPoint.penetrationImpulse; - lambdaTemp = contactPoint.friction2Impulse; - contactPoint.friction2Impulse = std::max(-frictionLimit, - std::min(contactPoint.friction2Impulse - + deltaLambda, frictionLimit)); - deltaLambda = contactPoint.friction2Impulse - lambdaTemp; - - // Compute the impulse P=J^T * lambda - const Impulse impulseFriction2 = computeFriction2Impulse(deltaLambda, - contactPoint); - - // Apply the impulses to the bodies of the constraint - applyImpulse(impulseFriction2, contactManifold); - } + applySplitImpulse(splitImpulsePenetration, contactManifold); } - // If we solve the friction constraints at the center of the contact manifold - if (mIsSolveFrictionAtContactManifoldCenterActive) { + // If we do not solve the friction constraints at the center of the contact manifold + if (!mIsSolveFrictionAtContactManifoldCenterActive) { - // ------ First friction constraint at the center of the contact manifol ------ // + // --------- Friction 1 --------- // // Compute J*v - Vector3 deltaV = v2 + w2.cross(contactManifold.r2Friction) - - v1 - w1.cross(contactManifold.r1Friction); - decimal Jv = deltaV.dot(contactManifold.frictionVector1); + deltaV = v2 + w2.cross(contactPoint.r2) - v1 - w1.cross(contactPoint.r1); + Jv = deltaV.dot(contactPoint.frictionVector1); // Compute the Lagrange multiplier lambda - decimal deltaLambda = -Jv * contactManifold.inverseFriction1Mass; - decimal frictionLimit = contactManifold.frictionCoefficient * sumPenetrationImpulse; - lambdaTemp = contactManifold.friction1Impulse; - contactManifold.friction1Impulse = std::max(-frictionLimit, - std::min(contactManifold.friction1Impulse + - deltaLambda, frictionLimit)); - deltaLambda = contactManifold.friction1Impulse - lambdaTemp; + deltaLambda = -Jv; + deltaLambda *= contactPoint.inverseFriction1Mass; + decimal frictionLimit = contactManifold.frictionCoefficient * + contactPoint.penetrationImpulse; + lambdaTemp = contactPoint.friction1Impulse; + contactPoint.friction1Impulse = std::max(-frictionLimit, + std::min(contactPoint.friction1Impulse + + deltaLambda, frictionLimit)); + deltaLambda = contactPoint.friction1Impulse - lambdaTemp; // Compute the impulse P=J^T * lambda - Vector3 linearImpulseBody1 = -contactManifold.frictionVector1 * deltaLambda; - Vector3 angularImpulseBody1 = -contactManifold.r1CrossT1 * deltaLambda; - Vector3 linearImpulseBody2 = contactManifold.frictionVector1 * deltaLambda; - Vector3 angularImpulseBody2 = contactManifold.r2CrossT1 * deltaLambda; - const Impulse impulseFriction1(linearImpulseBody1, angularImpulseBody1, - linearImpulseBody2, angularImpulseBody2); + const Impulse impulseFriction1 = computeFriction1Impulse(deltaLambda, + contactPoint); // Apply the impulses to the bodies of the constraint applyImpulse(impulseFriction1, contactManifold); - // ------ Second friction constraint at the center of the contact manifol ----- // + // --------- Friction 2 --------- // // Compute J*v - deltaV = v2 + w2.cross(contactManifold.r2Friction) - - v1 - w1.cross(contactManifold.r1Friction); - Jv = deltaV.dot(contactManifold.frictionVector2); + deltaV = v2 + w2.cross(contactPoint.r2) - v1 - w1.cross(contactPoint.r1); + Jv = deltaV.dot(contactPoint.frictionVector2); // Compute the Lagrange multiplier lambda - deltaLambda = -Jv * contactManifold.inverseFriction2Mass; - frictionLimit = contactManifold.frictionCoefficient * sumPenetrationImpulse; - lambdaTemp = contactManifold.friction2Impulse; - contactManifold.friction2Impulse = std::max(-frictionLimit, - std::min(contactManifold.friction2Impulse + - deltaLambda, frictionLimit)); - deltaLambda = contactManifold.friction2Impulse - lambdaTemp; + deltaLambda = -Jv; + deltaLambda *= contactPoint.inverseFriction2Mass; + frictionLimit = contactManifold.frictionCoefficient * + contactPoint.penetrationImpulse; + lambdaTemp = contactPoint.friction2Impulse; + contactPoint.friction2Impulse = std::max(-frictionLimit, + std::min(contactPoint.friction2Impulse + + deltaLambda, frictionLimit)); + deltaLambda = contactPoint.friction2Impulse - lambdaTemp; // Compute the impulse P=J^T * lambda - linearImpulseBody1 = -contactManifold.frictionVector2 * deltaLambda; - angularImpulseBody1 = -contactManifold.r1CrossT2 * deltaLambda; - linearImpulseBody2 = contactManifold.frictionVector2 * deltaLambda; - angularImpulseBody2 = contactManifold.r2CrossT2 * deltaLambda; - const Impulse impulseFriction2(linearImpulseBody1, angularImpulseBody1, - linearImpulseBody2, angularImpulseBody2); + const Impulse impulseFriction2 = computeFriction2Impulse(deltaLambda, + contactPoint); // Apply the impulses to the bodies of the constraint applyImpulse(impulseFriction2, contactManifold); - - // ------ Twist friction constraint at the center of the contact manifol ------ // - - // Compute J*v - deltaV = w2 - w1; - Jv = deltaV.dot(contactManifold.normal); - - deltaLambda = -Jv * (contactManifold.inverseTwistFrictionMass); - frictionLimit = contactManifold.frictionCoefficient * sumPenetrationImpulse; - lambdaTemp = contactManifold.frictionTwistImpulse; - contactManifold.frictionTwistImpulse = std::max(-frictionLimit, - std::min(contactManifold.frictionTwistImpulse - + deltaLambda, frictionLimit)); - deltaLambda = contactManifold.frictionTwistImpulse - lambdaTemp; - - // Compute the impulse P=J^T * lambda - linearImpulseBody1 = Vector3(0.0, 0.0, 0.0); - angularImpulseBody1 = -contactManifold.normal * deltaLambda; - linearImpulseBody2 = Vector3(0.0, 0.0, 0.0);; - angularImpulseBody2 = contactManifold.normal * deltaLambda; - const Impulse impulseTwistFriction(linearImpulseBody1, angularImpulseBody1, - linearImpulseBody2, angularImpulseBody2); - - // Apply the impulses to the bodies of the constraint - applyImpulse(impulseTwistFriction, contactManifold); } } + + // If we solve the friction constraints at the center of the contact manifold + if (mIsSolveFrictionAtContactManifoldCenterActive) { + + // ------ First friction constraint at the center of the contact manifol ------ // + + // Compute J*v + Vector3 deltaV = v2 + w2.cross(contactManifold.r2Friction) + - v1 - w1.cross(contactManifold.r1Friction); + decimal Jv = deltaV.dot(contactManifold.frictionVector1); + + // Compute the Lagrange multiplier lambda + decimal deltaLambda = -Jv * contactManifold.inverseFriction1Mass; + decimal frictionLimit = contactManifold.frictionCoefficient * sumPenetrationImpulse; + lambdaTemp = contactManifold.friction1Impulse; + contactManifold.friction1Impulse = std::max(-frictionLimit, + std::min(contactManifold.friction1Impulse + + deltaLambda, frictionLimit)); + deltaLambda = contactManifold.friction1Impulse - lambdaTemp; + + // Compute the impulse P=J^T * lambda + Vector3 linearImpulseBody1 = -contactManifold.frictionVector1 * deltaLambda; + Vector3 angularImpulseBody1 = -contactManifold.r1CrossT1 * deltaLambda; + Vector3 linearImpulseBody2 = contactManifold.frictionVector1 * deltaLambda; + Vector3 angularImpulseBody2 = contactManifold.r2CrossT1 * deltaLambda; + const Impulse impulseFriction1(linearImpulseBody1, angularImpulseBody1, + linearImpulseBody2, angularImpulseBody2); + + // Apply the impulses to the bodies of the constraint + applyImpulse(impulseFriction1, contactManifold); + + // ------ Second friction constraint at the center of the contact manifol ----- // + + // Compute J*v + deltaV = v2 + w2.cross(contactManifold.r2Friction) + - v1 - w1.cross(contactManifold.r1Friction); + Jv = deltaV.dot(contactManifold.frictionVector2); + + // Compute the Lagrange multiplier lambda + deltaLambda = -Jv * contactManifold.inverseFriction2Mass; + frictionLimit = contactManifold.frictionCoefficient * sumPenetrationImpulse; + lambdaTemp = contactManifold.friction2Impulse; + contactManifold.friction2Impulse = std::max(-frictionLimit, + std::min(contactManifold.friction2Impulse + + deltaLambda, frictionLimit)); + deltaLambda = contactManifold.friction2Impulse - lambdaTemp; + + // Compute the impulse P=J^T * lambda + linearImpulseBody1 = -contactManifold.frictionVector2 * deltaLambda; + angularImpulseBody1 = -contactManifold.r1CrossT2 * deltaLambda; + linearImpulseBody2 = contactManifold.frictionVector2 * deltaLambda; + angularImpulseBody2 = contactManifold.r2CrossT2 * deltaLambda; + const Impulse impulseFriction2(linearImpulseBody1, angularImpulseBody1, + linearImpulseBody2, angularImpulseBody2); + + // Apply the impulses to the bodies of the constraint + applyImpulse(impulseFriction2, contactManifold); + + // ------ Twist friction constraint at the center of the contact manifol ------ // + + // Compute J*v + deltaV = w2 - w1; + Jv = deltaV.dot(contactManifold.normal); + + deltaLambda = -Jv * (contactManifold.inverseTwistFrictionMass); + frictionLimit = contactManifold.frictionCoefficient * sumPenetrationImpulse; + lambdaTemp = contactManifold.frictionTwistImpulse; + contactManifold.frictionTwistImpulse = std::max(-frictionLimit, + std::min(contactManifold.frictionTwistImpulse + + deltaLambda, frictionLimit)); + deltaLambda = contactManifold.frictionTwistImpulse - lambdaTemp; + + // Compute the impulse P=J^T * lambda + linearImpulseBody1 = Vector3(0.0, 0.0, 0.0); + angularImpulseBody1 = -contactManifold.normal * deltaLambda; + linearImpulseBody2 = Vector3(0.0, 0.0, 0.0);; + angularImpulseBody2 = contactManifold.normal * deltaLambda; + const Impulse impulseTwistFriction(linearImpulseBody1, angularImpulseBody1, + linearImpulseBody2, angularImpulseBody2); + + // Apply the impulses to the bodies of the constraint + applyImpulse(impulseTwistFriction, contactManifold); + } } } -// Solve the constraints -void ContactSolver::solve(decimal timeStep) { - - PROFILE("ContactSolver::solve()"); - - // Set the current time step - mTimeStep = timeStep; - - // Initialize the solver - initialize(); - - // Fill-in all the matrices needed to solve the LCP problem - initializeContactConstraints(); - - // Warm start the solver - if (mIsWarmStartingActive) { - warmStart(); - } - - // Solve the contact constraints - solveContactConstraints(); - - // Cache the lambda values in order to use them in the next step - storeImpulses(); -} - // Store the computed impulses to use them to // warm start the solver at the next iteration void ContactSolver::storeImpulses() { diff --git a/src/engine/ContactSolver.h b/src/engine/ContactSolver.h index 97ec39b5..68e0c14a 100644 --- a/src/engine/ContactSolver.h +++ b/src/engine/ContactSolver.h @@ -345,9 +345,6 @@ class ContactSolver { /// Reference to all the contact manifold of the world std::vector& mContactManifolds; - /// Number of iterations of the contact solver - uint mNbIterations; - /// Split linear velocities for the position contact solver (split impulse) Vector3* mSplitLinearVelocities; @@ -389,25 +386,12 @@ class ContactSolver { // -------------------- Methods -------------------- // - /// Initialize the constraint solver - void initialize(); - /// Initialize the split impulse velocities void initializeSplitImpulseVelocities(); /// Initialize the contact constraints before solving the system void initializeContactConstraints(); - /// Store the computed impulses to use them to - /// warm start the solver at the next iteration - void storeImpulses(); - - /// Warm start the solver. - void warmStart(); - - /// Solve the contact constraints by applying sequential impulses - void solveContactConstraints(); - /// Apply an impulse to the two bodies of a constraint void applyImpulse(const Impulse& impulse, const ContactManifoldSolver& manifold); @@ -460,8 +444,18 @@ class ContactSolver { /// Destructor virtual ~ContactSolver(); - /// Solve the constraints - void solve(decimal timeStep); + /// Initialize the constraint solver + void initialize(decimal dt); + + /// Warm start the solver. + void warmStart(); + + /// Store the computed impulses to use them to + /// warm start the solver at the next iteration + void storeImpulses(); + + /// Solve the contacts + void solve(); /// Return true if the body is in at least one constraint bool isConstrainedBody(RigidBody* body) const; @@ -481,9 +475,6 @@ class ContactSolver { /// Clean up the constraint solver void cleanup(); - /// Set the number of iterations of the constraint solver - void setNbIterationsSolver(uint nbIterations); - /// Activate or Deactivate the split impulses for contacts void setIsSplitImpulseActive(bool isActive); @@ -511,11 +502,6 @@ inline Vector3 ContactSolver::getSplitAngularVelocityOfBody(RigidBody* body) { return mSplitAngularVelocities[indexBody]; } -// Set the number of iterations of the constraint solver -inline void ContactSolver::setNbIterationsSolver(uint nbIterations) { - mNbIterations = nbIterations; -} - // Activate or Deactivate the split impulses for contacts inline void ContactSolver::setIsSplitImpulseActive(bool isActive) { mIsSplitImpulseActive = isActive; diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index dd319483..237d5236 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -38,6 +38,7 @@ DynamicsWorld::DynamicsWorld(const Vector3 &gravity, decimal timeStep = DEFAULT_ mMapBodyToConstrainedVelocityIndex), mConstraintSolver(mJoints, mConstrainedLinearVelocities, mConstrainedAngularVelocities, mMapBodyToConstrainedVelocityIndex), + mNbSolverIterations(DEFAULT_CONSTRAINTS_SOLVER_NB_ITERATIONS), mIsDeactivationActive(DEACTIVATION_ENABLED) { } @@ -94,15 +95,11 @@ void DynamicsWorld::update() { // Compute the collision detection mCollisionDetection.computeCollisionDetection(); - // Initialize the constrained velocities - initConstrainedVelocitiesArray(); + // Integrate the velocities + integrateRigidBodiesVelocities(); - // If there are contacts - if (!mContactManifolds.empty()) { - - // Solve the contacts - mContactSolver.solve(static_cast(mTimer.getTimeStep())); - } + // Solve the contacts and constraints + solveContactsAndConstraints(); // Update the timer mTimer.nextStep(); @@ -110,8 +107,8 @@ void DynamicsWorld::update() { // Reset the movement boolean variable of each body to false resetBodiesMovementVariable(); - // Update the position and orientation of each body - updateRigidBodiesPositionAndOrientation(); + // Integrate the position and orientation of each body + integrateRigidBodiesPositions(); // Cleanup of the contact solver mContactSolver.cleanup(); @@ -124,8 +121,8 @@ void DynamicsWorld::update() { setInterpolationFactorToAllBodies(); } -// Update the position and orientation of the rigid bodies -void DynamicsWorld::updateRigidBodiesPositionAndOrientation() { +// Integrate position and orientation of the rigid bodies +void DynamicsWorld::integrateRigidBodiesPositions() { PROFILE("DynamicsWorld::updateRigidBodiesPositionAndOrientation()"); @@ -200,8 +197,8 @@ void DynamicsWorld::setInterpolationFactorToAllBodies() { } } -// Initialize the constrained velocities array at each step -void DynamicsWorld::initConstrainedVelocitiesArray() { +// Integrate the velocities of rigid bodies +void DynamicsWorld::integrateRigidBodiesVelocities() { // TODO : Use better memory allocation here mConstrainedLinearVelocities = std::vector(mRigidBodies.size(), Vector3(0, 0, 0)); @@ -228,6 +225,50 @@ void DynamicsWorld::initConstrainedVelocitiesArray() { assert(mMapBodyToConstrainedVelocityIndex.size() == mRigidBodies.size()); } +// Solve the contacts and constraints +void DynamicsWorld::solveContactsAndConstraints() { + + PROFILE("DynamicsWorld::solveContactsAndConstraints()"); + + // Get the current time step + decimal dt = static_cast(mTimer.getTimeStep()); + + // Check if there are contacts and constraints to solve + bool isConstraintsToSolve = !mJoints.empty(); + bool isContactsToSolve = !mContactManifolds.empty(); + if (!isConstraintsToSolve && !isContactsToSolve) return; + + // If there are contacts + if (isContactsToSolve) { + + // Initialize the solver + mContactSolver.initialize(dt); + + // Warm start the contact solver + mContactSolver.warmStart(); + } + + // If there are constraints + if (isConstraintsToSolve) { + + // Initialize the constraint solver + mConstraintSolver.initialize(dt); + } + + // For each iteration of the solver + for (uint i=0; i mJoints; - /// All the bodies that are part of contacts or constraints - std::set mConstrainedBodies; - /// Gravity vector of the world Vector3 mGravity; @@ -99,8 +99,8 @@ class DynamicsWorld : public CollisionWorld { /// Private assignment operator DynamicsWorld& operator=(const DynamicsWorld& world); - /// Compute the motion of all bodies and update their positions and orientations - void updateRigidBodiesPositionAndOrientation(); + /// Integrate the positions and orientations of rigid bodies + void integrateRigidBodiesPositions(); /// Update the position and orientation of a body void updatePositionAndOrientationOfBody(RigidBody* body, Vector3 newLinVelocity, @@ -109,8 +109,11 @@ class DynamicsWorld : public CollisionWorld { /// Compute and set the interpolation factor to all bodies void setInterpolationFactorToAllBodies(); - /// Initialize the constrained velocities array at each step - void initConstrainedVelocitiesArray(); + /// Integrate the velocities of rigid bodies + void integrateRigidBodiesVelocities(); + + /// Solve the contacts and constraints + void solveContactsAndConstraints(); /// Cleanup the constrained velocities array at each step void cleanupConstrainedVelocitiesArray(); @@ -212,7 +215,7 @@ inline void DynamicsWorld::stop() { // Set the number of iterations of the constraint solver inline void DynamicsWorld::setNbIterationsSolver(uint nbIterations) { - mContactSolver.setNbIterationsSolver(nbIterations); + mNbSolverIterations = nbIterations; } // Activate or Deactivate the split impulses for contacts From 434a1687857a397cc71a6d0288e5fbbf86ff99b1 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Fri, 26 Apr 2013 00:46:09 +0200 Subject: [PATCH 15/66] Add files for the joints example --- examples/CMakeLists.txt | 1 + examples/fallingcubes/Box.cpp | 180 ----------------- examples/joints/CMakeLists.txt | 17 ++ examples/joints/Joints.cpp | 151 +++++++++++++++ examples/joints/Scene.cpp | 182 ++++++++++++++++++ .../{fallingcubes/Box.h => joints/Scene.h} | 114 +++++------ 6 files changed, 411 insertions(+), 234 deletions(-) delete mode 100644 examples/fallingcubes/Box.cpp create mode 100644 examples/joints/CMakeLists.txt create mode 100644 examples/joints/Joints.cpp create mode 100644 examples/joints/Scene.cpp rename examples/{fallingcubes/Box.h => joints/Scene.h} (51%) diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt index 568479fb..0740c9fc 100644 --- a/examples/CMakeLists.txt +++ b/examples/CMakeLists.txt @@ -3,3 +3,4 @@ CMAKE_MINIMUM_REQUIRED(VERSION 2.6) add_subdirectory(common/) add_subdirectory(fallingcubes/) +add_subdirectory(joints/) diff --git a/examples/fallingcubes/Box.cpp b/examples/fallingcubes/Box.cpp deleted file mode 100644 index d4169c9a..00000000 --- a/examples/fallingcubes/Box.cpp +++ /dev/null @@ -1,180 +0,0 @@ -/******************************************************************************** -* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * -* Copyright (c) 2010-2013 Daniel Chappuis * -********************************************************************************* -* * -* 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. * -* * -********************************************************************************/ - -// Libraries -#include "Box.h" - -// Macros -#define MEMBER_OFFSET(s,m) ((char *)NULL + (offsetof(s,m))) - -// Initialize static variables -openglframework::VertexBufferObject Box::mVBOVertices(GL_ARRAY_BUFFER); -openglframework::VertexBufferObject Box::mVBOIndices(GL_ELEMENT_ARRAY_BUFFER); -bool Box::areVBOsCreated = false; -VertexData Box::mCubeVertices[8] = { - {openglframework::Vector3(1,1,1),openglframework::Vector3(1,1,1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(-1,1,1),openglframework::Vector3(-1,1,1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(-1,-1,1),openglframework::Vector3(-1,-1,1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(1,-1,1),openglframework::Vector3(1,-1,1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(1,-1,-1),openglframework::Vector3(1,-1,-1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(-1,-1,-1),openglframework::Vector3(-1,-1,-1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(-1,1,-1),openglframework::Vector3(-1,1,-1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(1,1,-1),openglframework::Vector3(1,1,-1),openglframework::Color(0,0,1,1)} -}; -GLuint Box::mCubeIndices[36] = { 0, 1, 2, - 2, 3, 0, - 7, 4, 5, - 5, 6, 7, - 6, 5, 2, - 2, 1, 6, - 7, 0, 3, - 3, 4, 7, - 7, 6, 1, - 1, 0, 7, - 3, 2, 5, - 5, 4, 3}; - -// Constructor -Box::Box(const openglframework::Vector3& size, const openglframework::Vector3 &position, - float mass, reactphysics3d::DynamicsWorld* dynamicsWorld) - : openglframework::Object3D() { - - // Initialize the size of the box - mSize[0] = size.x * 0.5f; - mSize[1] = size.y * 0.5f; - mSize[2] = size.z * 0.5f; - - // Compute the scaling matrix - mScalingMatrix = openglframework::Matrix4(mSize[0], 0, 0, 0, - 0, mSize[1], 0, 0, - 0, 0, mSize[2], 0, - 0, 0, 0, 1); - - // Initialize the position where the cube will be rendered - translateWorld(position); - - // Create the collision shape for the rigid body (box shape) - mCollisionShape = new rp3d::BoxShape(rp3d::Vector3(mSize[0], mSize[1], mSize[2])); - - // Compute the inertia tensor of the body using its collision shape - rp3d::Matrix3x3 inertiaTensor; - mCollisionShape->computeLocalInertiaTensor(inertiaTensor, mass); - - // Initial position and orientation of the rigid body - rp3d::Vector3 initPosition(position.x, position.y, position.z); - rp3d::Quaternion initOrientation = rp3d::Quaternion::identity(); - rp3d::Transform transform(initPosition, initOrientation); - - // Create a rigid body corresponding to the cube in the dynamics world - mRigidBody = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, mCollisionShape); - - // If the Vertex Buffer object has not been created yet - if (!areVBOsCreated) { - // Create the Vertex Buffer - createVBO(); - } -} - -// Destructor -Box::~Box() { - - // Destroy the collision shape - delete mCollisionShape; -} - -// Render the cube at the correct position and with the correct orientation -void Box::render(openglframework::Shader& shader) { - - // Bind the shader - shader.bind(); - - // Set the model to World matrix - shader.setMatrix4x4Uniform("modelToWorldMatrix", mTransformMatrix); - - // Bind the vertices VBO - mVBOVertices.bind(); - - // Enable the vertex, normal and color arrays - glEnableClientState(GL_VERTEX_ARRAY); - glEnableClientState(GL_COLOR_ARRAY); - glEnableClientState(GL_NORMAL_ARRAY); - - // Set the arrays pointers - glVertexPointer(3, GL_FLOAT, sizeof(VertexData), MEMBER_OFFSET(VertexData, position)); - glNormalPointer(GL_FLOAT, sizeof(VertexData), MEMBER_OFFSET(VertexData, normal)); - glColorPointer(3, GL_FLOAT, sizeof(VertexData), MEMBER_OFFSET(VertexData, color)); - - // Bind the indices VBO - mVBOIndices.bind(); - - // Draw the geometry of the box - glDrawElements(GL_TRIANGLES, 36, GL_UNSIGNED_INT, (char*)NULL); - - // Unbind the VBOs - mVBOVertices.unbind(); - mVBOIndices.unbind(); - - // Disable the arrays - glDisableClientState(GL_VERTEX_ARRAY); - glDisableClientState(GL_COLOR_ARRAY); - glDisableClientState(GL_NORMAL_ARRAY); - - // Unbind the shader - shader.unbind(); -} - -// Update the transform matrix of the box -void Box::updateTransform() { - - // Get the interpolated transform of the rigid body - rp3d::Transform transform = mRigidBody->getInterpolatedTransform(); - - // Compute the transform used for rendering the box - float matrix[16]; - transform.getOpenGLMatrix(matrix); - openglframework::Matrix4 newMatrix(matrix[0], matrix[4], matrix[8], matrix[12], - matrix[1], matrix[5], matrix[9], matrix[13], - matrix[2], matrix[6], matrix[10], matrix[14], - matrix[3], matrix[7], matrix[11], matrix[15]); - - // Apply the scaling matrix to have the correct box dimensions - mTransformMatrix = newMatrix * mScalingMatrix; -} - -// Create the Vertex Buffer Objects used to render to box with OpenGL. -/// We create two VBOs (one for vertices and one for indices) to render all the boxes -/// in the simulation. -void Box::createVBO() { - - // Create the VBOs - mVBOVertices.create(); - mVBOIndices.create(); - - // Copy the data into the VBOs - mVBOVertices.copyDataIntoVBO(sizeof(mCubeVertices), mCubeVertices, GL_STATIC_DRAW); - mVBOIndices.copyDataIntoVBO(sizeof(mCubeIndices), mCubeIndices, GL_STATIC_DRAW); - - areVBOsCreated = true; -} diff --git a/examples/joints/CMakeLists.txt b/examples/joints/CMakeLists.txt new file mode 100644 index 00000000..dc2eed0c --- /dev/null +++ b/examples/joints/CMakeLists.txt @@ -0,0 +1,17 @@ +# Minimum cmake version required +cmake_minimum_required(VERSION 2.6) + +# Project configuration +PROJECT(Joints) + +# Copy the shaders used for the demo into the build directory +FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/shaders/") + +# Headers +INCLUDE_DIRECTORIES("../common/opengl-framework/src/" "../common/") + +# Create the example executable using the +# compiled reactphysics3d static library +ADD_EXECUTABLE(joints Joints.cpp Scene.cpp Scene.h "../common/Box.cpp" "../common/Box.h" "../common/Viewer.cpp" "../common/Viewer.h") + +TARGET_LINK_LIBRARIES(joints reactphysics3d openglframework) diff --git a/examples/joints/Joints.cpp b/examples/joints/Joints.cpp new file mode 100644 index 00000000..f42ffb97 --- /dev/null +++ b/examples/joints/Joints.cpp @@ -0,0 +1,151 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Scene.h" +#include "Viewer.h" + +// Declarations +void simulate(); +void display(); +void finish(); +void reshape(int width, int height); +void mouseButton(int button, int state, int x, int y); +void mouseMotion(int x, int y); +void keyboard(unsigned char key, int x, int y); +void init(); + +// Namespaces +using namespace openglframework; + +// Global variables +Viewer* viewer; +Scene* scene; + +// Main function +int main(int argc, char** argv) { + + // Create and initialize the Viewer + viewer = new Viewer(); + Vector2 windowsSize = Vector2(800, 600); + Vector2 windowsPosition = Vector2(100, 100); + bool initOK = viewer->init(argc, argv, "ReactPhysics3D Examples - Joints", windowsSize, windowsPosition); + if (!initOK) return 1; + + // Create the scene + scene = new Scene(viewer); + + init(); + + // Glut Idle function that is continuously called + glutIdleFunc(simulate); + glutDisplayFunc(display); + glutReshapeFunc(reshape); + glutMouseFunc(mouseButton); + glutMotionFunc(mouseMotion); + glutKeyboardFunc(keyboard); + glutCloseFunc(finish); + + // Glut main looop + glutMainLoop(); + + return 0; +} + +// Simulate function +void simulate() { + + // Physics simulation + scene->simulate(); + + viewer->computeFPS(); + + // Ask GLUT to render the scene + glutPostRedisplay (); +} + +// Initialization +void init() { + + // Define the background color (black) + glClearColor(0.0, 0.0, 0.0, 1.0); +} + +// Reshape function +void reshape(int newWidth, int newHeight) { + viewer->reshape(newWidth, newHeight); +} + +// Called when a mouse button event occurs +void mouseButton(int button, int state, int x, int y) { + viewer->mouseButtonEvent(button, state, x, y); +} + +// Called when a mouse motion event occurs +void mouseMotion(int x, int y) { + viewer->mouseMotionEvent(x, y); +} + +// Called when the user hits a special key on the keyboard +void keyboard(unsigned char key, int x, int y) { + switch(key) { + + // Escape key + case 27: + glutLeaveMainLoop(); + break; + + // Space bar + case 32: + scene->pauseContinueSimulation(); + break; + } +} + +// End of the application +void finish() { + + // Destroy the viewer and the scene + delete viewer; + delete scene; +} + +// Display the scene +void display() { + + // Render the scene + scene->render(); + + // Display the FPS + viewer->displayGUI(); + + // Swap the buffers + glutSwapBuffers(); + + // Check the OpenGL errors + GlutViewer::checkOpenGLErrors(); +} + + diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp new file mode 100644 index 00000000..693a0fa5 --- /dev/null +++ b/examples/joints/Scene.cpp @@ -0,0 +1,182 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Scene.h" + +// Namespaces +using namespace openglframework; + +// Constructor +Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), + mPhongShader("shaders/phong.vert", + "shaders/phong.frag"), mIsRunning(false) { + + // Move the light 0 + mLight0.translateWorld(Vector3(7, 15, 15)); + + // Compute the radius and the center of the scene + float radiusScene = 10.0f; + openglframework::Vector3 center(0, 5, 0); + + // Set the center of the scene + mViewer->setScenePosition(center, radiusScene); + + // Gravity vector in the dynamics world + rp3d::Vector3 gravity(0, -9.81, 0); + + // Time step for the physics simulation + rp3d::decimal timeStep = 1.0f / 60.0f; + + // Create the dynamics world for the physics simulation + mDynamicsWorld = new rp3d::DynamicsWorld(gravity, timeStep); + + // Set the number of iterations of the constraint solver + mDynamicsWorld->setNbIterationsSolver(15); + + float radius = 2.0f; + + // Create all the cubes of the scene + for (int i=0; igetRigidBody()->setIsMotionEnabled(true); + + // Set the bouncing factor of the box + cube->getRigidBody()->setRestitution(0.4); + + // Add the box the list of box in the scene + mBoxes.push_back(cube); + } + + // Create the floor + openglframework::Vector3 floorPosition(0, 0, 0); + mFloor = new Box(FLOOR_SIZE, floorPosition, FLOOR_MASS, mDynamicsWorld); + + // The floor must be a non-moving rigid body + mFloor->getRigidBody()->setIsMotionEnabled(false); + + // Set the bouncing factor of the floor + mFloor->getRigidBody()->setRestitution(0.3); + + // Start the simulation + startSimulation(); +} + +// Destructor +Scene::~Scene() { + + // Stop the physics simulation + stopSimulation(); + + // Destroy the shader + mPhongShader.destroy(); + + // Destroy all the cubes of the scene + for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + + // Destroy the corresponding rigid body from the dynamics world + mDynamicsWorld->destroyRigidBody((*it)->getRigidBody()); + + // Destroy the cube + delete (*it); + } + + // Destroy the rigid body of the floor + mDynamicsWorld->destroyRigidBody(mFloor->getRigidBody()); + + // Destroy the floor + delete mFloor; + + // Destroy the dynamics world + delete mDynamicsWorld; +} + +// Take a step for the simulation +void Scene::simulate() { + + // If the physics simulation is running + if (mIsRunning) { + + // Take a simulation step + mDynamicsWorld->update(); + + // Update the position and orientation of the boxes + for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + + // Update the transform used for the rendering + (*it)->updateTransform(); + } + + mFloor->updateTransform(); + + } +} + +// Render the scene +void Scene::render() { + + glEnable(GL_DEPTH_TEST); + glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); + glEnable(GL_CULL_FACE); + + // Bind the shader + mPhongShader.bind(); + + // Set the variables of the shader + const Camera& camera = mViewer->getCamera(); + Matrix4 matrixIdentity; + matrixIdentity.setToIdentity(); + mPhongShader.setVector3Uniform("cameraWorldPosition", mViewer->getCamera().getOrigin()); + mPhongShader.setMatrix4x4Uniform("worldToCameraMatrix", camera.getTransformMatrix().getInverse()); + mPhongShader.setMatrix4x4Uniform("projectionMatrix", camera.getProjectionMatrix()); + mPhongShader.setVector3Uniform("lightWorldPosition", mLight0.getOrigin()); + mPhongShader.setVector3Uniform("lightAmbientColor", Vector3(0.3f, 0.3f, 0.3f)); + Color& diffCol = mLight0.getDiffuseColor(); + Color& specCol = mLight0.getSpecularColor(); + mPhongShader.setVector3Uniform("lightDiffuseColor", Vector3(diffCol.r, diffCol.g, diffCol.b)); + mPhongShader.setVector3Uniform("lightSpecularColor", Vector3(specCol.r, specCol.g, specCol.b)); + mPhongShader.setFloatUniform("shininess", 60.0f); + + // Render all the cubes of the scene + for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + (*it)->render(mPhongShader); + } + + // Render the floor + mFloor->render(mPhongShader); + + // Unbind the shader + mPhongShader.unbind(); +} diff --git a/examples/fallingcubes/Box.h b/examples/joints/Scene.h similarity index 51% rename from examples/fallingcubes/Box.h rename to examples/joints/Scene.h index d287c3cb..3741962e 100644 --- a/examples/fallingcubes/Box.h +++ b/examples/joints/Scene.h @@ -23,89 +23,95 @@ * * ********************************************************************************/ -#ifndef BOX_H -#define BOX_H +#ifndef SCENE_H +#define SCENE_H // Libraries #include "openglframework.h" #include "reactphysics3d.h" +#include "Box.h" -// Structure VertexData -struct VertexData { +// Constants +const int NB_BOXES = 20; // Number of boxes in the scene +const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters +const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters +const float BOX_MASS = 1.0f; // Box mass in kilograms +const float FLOOR_MASS = 100.0f; // Floor mass in kilograms - /// Vertex position - openglframework::Vector3 position; - - /// Vertex normal - openglframework::Vector3 normal; - - // Vertex color - openglframework::Color color; -}; - -// Class Box -class Box : public openglframework::Object3D { +// Class Scene +class Scene { private : // -------------------- Attributes -------------------- // - /// Size of each side of the box - float mSize[3]; + // Pointer to the viewer + openglframework::GlutViewer* mViewer; - /// Rigid body used to simulate the dynamics of the box - rp3d::RigidBody* mRigidBody; + // Light 0 + openglframework::Light mLight0; - /// Collision shape of the rigid body - rp3d::BoxShape* mCollisionShape; + // Phong shader + openglframework::Shader mPhongShader; - /// Scaling matrix (applied to a cube to obtain the correct box dimensions) - openglframework::Matrix4 mScalingMatrix; + /// All the boxes of the scene + std::vector mBoxes; - /// Vertex Buffer Object for the vertices data used to render the box with OpenGL - static openglframework::VertexBufferObject mVBOVertices; + /// Box for the floor + Box* mFloor; - /// Vertex Buffer Object for the indices used to render the box with OpenGL - static openglframework::VertexBufferObject mVBOIndices; + /// Dynamics world used for the physics simulation + rp3d::DynamicsWorld* mDynamicsWorld; - /// Vertex data for each vertex of the cube (used to render the box) - static VertexData mCubeVertices[8]; + /// True if the physics simulation is running + bool mIsRunning; - /// Indices of the cube (used to render the box) - static GLuint mCubeIndices[36]; - - /// True if the VBOs have already been created - static bool areVBOsCreated; - - // -------------------- Methods -------------------- // - - /// Create a Vertex Buffer Object to render to box with OpenGL - static void createVBO(); - - public : + public: // -------------------- Methods -------------------- // /// Constructor - Box(const openglframework::Vector3& size, const openglframework::Vector3& position, - float mass, rp3d::DynamicsWorld* dynamicsWorld); + Scene(openglframework::GlutViewer* viewer); /// Destructor - ~Box(); + ~Scene(); - /// Return a pointer to the rigid body of the box - rp3d::RigidBody* getRigidBody(); + /// Take a step for the simulation + void simulate(); - /// Update the transform matrix of the box - void updateTransform(); + /// Stop the simulation + void stopSimulation(); - /// Render the cube at the correct position and with the correct orientation - void render(openglframework::Shader& shader); + /// Start the simulation + void startSimulation(); + + /// Pause or continue simulation + void pauseContinueSimulation(); + + /// Render the scene + void render(); }; -// Return a pointer to the rigid body of the box -inline rp3d::RigidBody* Box::getRigidBody() { - return mRigidBody; +// Stop the simulation +inline void Scene::stopSimulation() { + mDynamicsWorld->stop(); + mIsRunning = false; +} + +// Start the simulation +inline void Scene::startSimulation() { + mDynamicsWorld->start(); + mIsRunning = true; +} + +// Pause or continue simulation +inline void Scene::pauseContinueSimulation() { + if (mIsRunning) { + stopSimulation(); + } + else { + startSimulation(); + } } #endif From 44a26bdcab42d9ac115b323d4f83c2d5000b0f63 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sat, 27 Apr 2013 00:15:25 +0200 Subject: [PATCH 16/66] Add cmake configuration file to find GLEW library --- CMakeLists.txt | 3 ++ cmake/FindGLEW.cmake | 65 ++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 68 insertions(+) create mode 100644 cmake/FindGLEW.cmake diff --git a/CMakeLists.txt b/CMakeLists.txt index 7986b1ce..47bc8aaa 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -7,6 +7,9 @@ PROJECT(REACTPHYSICS3D) # Where to build the library SET(LIBRARY_OUTPUT_PATH lib/) +# Where to find the module to find special packages/libraries +set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/") + # Options OPTION(COMPILE_EXAMPLES "Select this if you want to build the examples" OFF) OPTION(COMPILE_TESTS "Select this if you want to build the tests" OFF) diff --git a/cmake/FindGLEW.cmake b/cmake/FindGLEW.cmake new file mode 100644 index 00000000..c29c4eb2 --- /dev/null +++ b/cmake/FindGLEW.cmake @@ -0,0 +1,65 @@ +# +# Try to find GLEW library and include path. +# Once done this will define +# +# GLEW_FOUND +# GLEW_INCLUDE_PATH +# GLEW_LIBRARY +# + +IF (WIN32) +FIND_PATH( GLEW_INCLUDE_PATH GL/glew.h +$ENV{PROGRAMFILES}/GLEW/include +${GLEW_ROOT_DIR}/include +DOC "The directory where GL/glew.h resides") + +IF (NV_SYSTEM_PROCESSOR STREQUAL "AMD64") +FIND_LIBRARY( GLEW_LIBRARY +NAMES glew64 glew64s +PATHS +$ENV{PROGRAMFILES}/GLEW/lib +${PROJECT_SOURCE_DIR}/src/nvgl/glew/bin +${PROJECT_SOURCE_DIR}/src/nvgl/glew/lib +DOC "The GLEW library (64-bit)" +) +ELSE(NV_SYSTEM_PROCESSOR STREQUAL "AMD64") +FIND_LIBRARY( GLEW_LIBRARY +NAMES glew GLEW glew32 glew32s +PATHS +$ENV{PROGRAMFILES}/GLEW/lib +${PROJECT_SOURCE_DIR}/src/nvgl/glew/bin +${PROJECT_SOURCE_DIR}/src/nvgl/glew/lib +DOC "The GLEW library" +) +ENDIF(NV_SYSTEM_PROCESSOR STREQUAL "AMD64") +ELSE (WIN32) +FIND_PATH( GLEW_INCLUDE_PATH GL/glew.h +/usr/include +/usr/local/include +/sw/include +/opt/local/include +${GLEW_ROOT_DIR}/include +DOC "The directory where GL/glew.h resides") + +FIND_LIBRARY( GLEW_LIBRARY +NAMES GLEW glew +PATHS +/usr/lib64 +/usr/lib +/usr/local/lib64 +/usr/local/lib +/sw/lib +/opt/local/lib +${GLEW_ROOT_DIR}/lib +DOC "The GLEW library") +ENDIF (WIN32) + +SET(GLEW_FOUND "NO") +IF (GLEW_INCLUDE_PATH AND GLEW_LIBRARY) +SET(GLEW_LIBRARIES ${GLEW_LIBRARY}) +SET(GLEW_FOUND "YES") +ENDIF (GLEW_INCLUDE_PATH AND GLEW_LIBRARY) + + +include(FindPackageHandleStandardArgs) +find_package_handle_standard_args(GLEW DEFAULT_MSG GLEW_LIBRARY GLEW_INCLUDE_PATH) \ No newline at end of file From c4eee4fb1f42f7d51e4f952ae838539dc0dbc8a7 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 2 May 2013 22:41:57 +0200 Subject: [PATCH 17/66] Implement the Ball-And-Socket joint --- examples/fallingcubes/Scene.h | 6 +- examples/joints/Scene.cpp | 126 +++++++++++------- examples/joints/Scene.h | 25 +++- .../narrowphase/EPA/EPAAlgorithm.cpp | 4 +- .../narrowphase/GJK/GJKAlgorithm.cpp | 12 +- .../narrowphase/SphereVsSphereAlgorithm.cpp | 4 +- src/constraint/BallAndSocketJoint.cpp | 95 ++++++++++++- src/constraint/BallAndSocketJoint.h | 30 ++++- src/constraint/Constraint.h | 15 +++ src/constraint/ContactPoint.cpp | 10 ++ src/constraint/ContactPoint.h | 6 + src/engine/ConstraintSolver.cpp | 43 +++++- src/engine/ConstraintSolver.h | 60 +++++++-- src/engine/ContactSolver.cpp | 32 ++--- src/engine/ContactSolver.h | 45 +------ src/engine/Impulse.h | 65 +++++++++ src/mathematics/Matrix3x3.h | 10 ++ src/mathematics/Quaternion.h | 4 +- src/mathematics/Transform.h | 4 +- src/reactphysics3d.h | 1 + test/tests/mathematics/TestMatrix3x3.h | 9 ++ test/tests/mathematics/TestTransform.h | 2 +- 22 files changed, 461 insertions(+), 147 deletions(-) create mode 100644 src/engine/Impulse.h diff --git a/examples/fallingcubes/Scene.h b/examples/fallingcubes/Scene.h index 3741962e..d2dbfc18 100644 --- a/examples/fallingcubes/Scene.h +++ b/examples/fallingcubes/Scene.h @@ -45,13 +45,13 @@ class Scene { // -------------------- Attributes -------------------- // - // Pointer to the viewer + /// Pointer to the viewer openglframework::GlutViewer* mViewer; - // Light 0 + /// Light 0 openglframework::Light mLight0; - // Phong shader + /// Phong shader openglframework::Shader mPhongShader; /// All the boxes of the scene diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index 693a0fa5..5e93b09c 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -56,39 +56,11 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), // Set the number of iterations of the constraint solver mDynamicsWorld->setNbIterationsSolver(15); - float radius = 2.0f; - - // Create all the cubes of the scene - for (int i=0; igetRigidBody()->setIsMotionEnabled(true); - - // Set the bouncing factor of the box - cube->getRigidBody()->setRestitution(0.4); - - // Add the box the list of box in the scene - mBoxes.push_back(cube); - } + // Create the Ball-and-Socket joint + createBallAndSocketJoints(); // Create the floor - openglframework::Vector3 floorPosition(0, 0, 0); - mFloor = new Box(FLOOR_SIZE, floorPosition, FLOOR_MASS, mDynamicsWorld); - - // The floor must be a non-moving rigid body - mFloor->getRigidBody()->setIsMotionEnabled(false); - - // Set the bouncing factor of the floor - mFloor->getRigidBody()->setRestitution(0.3); + createFloor(); // Start the simulation startSimulation(); @@ -103,20 +75,17 @@ Scene::~Scene() { // Destroy the shader mPhongShader.destroy(); - // Destroy all the cubes of the scene - for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + // Destroy the joints + mDynamicsWorld->destroyJoint(mBallAndSocketJoint); - // Destroy the corresponding rigid body from the dynamics world - mDynamicsWorld->destroyRigidBody((*it)->getRigidBody()); - - // Destroy the cube - delete (*it); - } - - // Destroy the rigid body of the floor - mDynamicsWorld->destroyRigidBody(mFloor->getRigidBody()); + // Destroy all the boxes of the scene + mDynamicsWorld->destroyRigidBody(mBallAndSocketJointBox1->getRigidBody()); + mDynamicsWorld->destroyRigidBody(mBallAndSocketJointBox2->getRigidBody()); + delete mBallAndSocketJointBox1; + delete mBallAndSocketJointBox2; // Destroy the floor + mDynamicsWorld->destroyRigidBody(mFloor->getRigidBody()); delete mFloor; // Destroy the dynamics world @@ -133,12 +102,10 @@ void Scene::simulate() { mDynamicsWorld->update(); // Update the position and orientation of the boxes - for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { - - // Update the transform used for the rendering - (*it)->updateTransform(); - } + mBallAndSocketJointBox1->updateTransform(); + mBallAndSocketJointBox2->updateTransform(); + // Update the position and orientation of the floor mFloor->updateTransform(); } @@ -169,10 +136,9 @@ void Scene::render() { mPhongShader.setVector3Uniform("lightSpecularColor", Vector3(specCol.r, specCol.g, specCol.b)); mPhongShader.setFloatUniform("shininess", 60.0f); - // Render all the cubes of the scene - for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { - (*it)->render(mPhongShader); - } + // Render all the boxes + mBallAndSocketJointBox1->render(mPhongShader); + mBallAndSocketJointBox2->render(mPhongShader); // Render the floor mFloor->render(mPhongShader); @@ -180,3 +146,61 @@ void Scene::render() { // Unbind the shader mPhongShader.unbind(); } + +// Create the boxes and joints for the Ball-and-Socket joint example +void Scene::createBallAndSocketJoints() { + + // --------------- Create the first box --------------- // + + // Position of the box + openglframework::Vector3 positionBox1(0, 15, 0); + + // Create a box and a corresponding rigid in the dynamics world + mBallAndSocketJointBox1 = new Box(BOX_SIZE, positionBox1 , BOX_MASS, mDynamicsWorld); + + // The fist box cannot move + mBallAndSocketJointBox1->getRigidBody()->setIsMotionEnabled(false); + + // Set the bouncing factor of the box + mBallAndSocketJointBox1->getRigidBody()->setRestitution(0.4); + + // --------------- Create the second box --------------- // + + // Position of the box + openglframework::Vector3 positionBox2(0, 10, 0); + + // Create a box and a corresponding rigid in the dynamics world + mBallAndSocketJointBox2 = new Box(BOX_SIZE, positionBox2 , BOX_MASS, mDynamicsWorld); + + // The second box is allowed to move + mBallAndSocketJointBox2->getRigidBody()->setIsMotionEnabled(true); + + // Set the bouncing factor of the box + mBallAndSocketJointBox2->getRigidBody()->setRestitution(0.4); + + // --------------- Create the joint --------------- // + + // Create the joint info object + rp3d::BallAndSocketJointInfo jointInfo; + jointInfo.body1 = mBallAndSocketJointBox1->getRigidBody(); + jointInfo.body2 = mBallAndSocketJointBox2->getRigidBody(); + jointInfo.anchorPointWorldSpace = rp3d::Vector3(0, 12.5, 0); + + // Create the joint in the dynamics world + mBallAndSocketJoint = dynamic_cast( + mDynamicsWorld->createJoint(jointInfo)); +} + +// Create the floor +void Scene::createFloor() { + + // Create the floor + openglframework::Vector3 floorPosition(0, 0, 0); + mFloor = new Box(FLOOR_SIZE, floorPosition, FLOOR_MASS, mDynamicsWorld); + + // The floor must be a non-moving rigid body + mFloor->getRigidBody()->setIsMotionEnabled(false); + + // Set the bouncing factor of the floor + mFloor->getRigidBody()->setRestitution(0.3); +} diff --git a/examples/joints/Scene.h b/examples/joints/Scene.h index 3741962e..7d89dc3a 100644 --- a/examples/joints/Scene.h +++ b/examples/joints/Scene.h @@ -32,7 +32,6 @@ #include "Box.h" // Constants -const int NB_BOXES = 20; // Number of boxes in the scene const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters const float BOX_MASS = 1.0f; // Box mass in kilograms @@ -45,17 +44,23 @@ class Scene { // -------------------- Attributes -------------------- // - // Pointer to the viewer + /// Pointer to the viewer openglframework::GlutViewer* mViewer; - // Light 0 + /// Light 0 openglframework::Light mLight0; - // Phong shader + /// Phong shader openglframework::Shader mPhongShader; - /// All the boxes of the scene - std::vector mBoxes; + /// Box 1 of Ball-And-Socket joint + Box* mBallAndSocketJointBox1; + + /// Box 2 of Ball-And-Socket joint + Box* mBallAndSocketJointBox2; + + /// Ball-and-Socket joint + rp3d::BallAndSocketJoint* mBallAndSocketJoint; /// Box for the floor Box* mFloor; @@ -66,6 +71,14 @@ class Scene { /// True if the physics simulation is running bool mIsRunning; + // -------------------- Methods -------------------- // + + /// Create the boxes and joints for the Ball-and-Socket joint example + void createBallAndSocketJoints(); + + /// Create the floor + void createFloor(); + public: // -------------------- Methods -------------------- // diff --git a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp index 2439979a..b6c70ba5 100644 --- a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp +++ b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp @@ -98,7 +98,7 @@ bool EPAAlgorithm::computePenetrationDepthAndContactPoints(const Simplex& simple // Transform a point from local space of body 2 to local // space of body 1 (the GJK algorithm is done in local space of body 1) - Transform body2Tobody1 = transform1.inverse() * transform2; + Transform body2Tobody1 = transform1.getInverse() * transform2; // Matrix that transform a direction from local // space of body 1 into local space of body 2 @@ -394,7 +394,7 @@ bool EPAAlgorithm::computePenetrationDepthAndContactPoints(const Simplex& simple // Compute the contact info v = transform1.getOrientation().getMatrix() * triangle->getClosestPoint(); Vector3 pALocal = triangle->computeClosestPointOfObject(suppPointsA); - Vector3 pBLocal = body2Tobody1.inverse() * triangle->computeClosestPointOfObject(suppPointsB); + Vector3 pBLocal = body2Tobody1.getInverse() * triangle->computeClosestPointOfObject(suppPointsB); Vector3 normal = v.getUnit(); decimal penetrationDepth = v.length(); assert(penetrationDepth > 0.0); diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.cpp b/src/collision/narrowphase/GJK/GJKAlgorithm.cpp index 16e8f671..392c704d 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.cpp +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.cpp @@ -73,7 +73,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, // Transform a point from local space of body 2 to local // space of body 1 (the GJK algorithm is done in local space of body 1) - Transform body2Tobody1 = transform1.inverse() * transform2; + Transform body2Tobody1 = transform1.getInverse() * transform2; // Matrix that transform a direction from local // space of body 1 into local space of body 2 @@ -127,7 +127,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, decimal dist = sqrt(distSquare); assert(dist > 0.0); pA = (pA - (collisionShape1->getMargin() / dist) * v); - pB = body2Tobody1.inverse() * (pB + (collisionShape2->getMargin() / dist) * v); + pB = body2Tobody1.getInverse() * (pB + (collisionShape2->getMargin() / dist) * v); // Compute the contact info Vector3 normal = transform1.getOrientation().getMatrix() * (-v.getUnit()); @@ -159,7 +159,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, decimal dist = sqrt(distSquare); assert(dist > 0.0); pA = (pA - (collisionShape1->getMargin() / dist) * v); - pB = body2Tobody1.inverse() * (pB + (collisionShape2->getMargin() / dist) * v); + pB = body2Tobody1.getInverse() * (pB + (collisionShape2->getMargin() / dist) * v); // Compute the contact info Vector3 normal = transform1.getOrientation().getMatrix() * (-v.getUnit()); @@ -189,7 +189,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, decimal dist = sqrt(distSquare); assert(dist > 0.0); pA = (pA - (collisionShape1->getMargin() / dist) * v); - pB = body2Tobody1.inverse() * (pB + (collisionShape2->getMargin() / dist) * v); + pB = body2Tobody1.getInverse() * (pB + (collisionShape2->getMargin() / dist) * v); // Compute the contact info Vector3 normal = transform1.getOrientation().getMatrix() * (-v.getUnit()); @@ -226,7 +226,7 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, decimal dist = sqrt(distSquare); assert(dist > 0.0); pA = (pA - (collisionShape1->getMargin() / dist) * v); - pB = body2Tobody1.inverse() * (pB + (collisionShape2->getMargin() / dist) * v); + pB = body2Tobody1.getInverse() * (pB + (collisionShape2->getMargin() / dist) * v); // Compute the contact info Vector3 normal = transform1.getOrientation().getMatrix() * (-v.getUnit()); @@ -275,7 +275,7 @@ bool GJKAlgorithm::computePenetrationDepthForEnlargedObjects(const CollisionShap // Transform a point from local space of body 2 to local space // of body 1 (the GJK algorithm is done in local space of body 1) - Transform body2ToBody1 = transform1.inverse() * transform2; + Transform body2ToBody1 = transform1.getInverse() * transform2; // Matrix that transform a direction from local space of body 1 into local space of body 2 Matrix3x3 rotateToBody2 = transform2.getOrientation().getMatrix().getTranspose() * diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp index 1a67782a..d50fe507 100644 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp @@ -60,8 +60,8 @@ bool SphereVsSphereAlgorithm::testCollision(const CollisionShape* collisionShape // If the sphere collision shapes intersect if (squaredDistanceBetweenCenters <= sumRadius * sumRadius) { - Vector3 centerSphere2InBody1LocalSpace = transform1.inverse() * transform2.getPosition(); - Vector3 centerSphere1InBody2LocalSpace = transform2.inverse() * transform1.getPosition(); + Vector3 centerSphere2InBody1LocalSpace = transform1.getInverse() * transform2.getPosition(); + Vector3 centerSphere1InBody2LocalSpace = transform2.getInverse() * transform1.getPosition(); Vector3 intersectionOnBody1 = sphereShape1->getRadius() * centerSphere2InBody1LocalSpace.getUnit(); Vector3 intersectionOnBody2 = sphereShape2->getRadius() * diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index 17de606d..214b7337 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -25,13 +25,17 @@ // Libraries #include "BallAndSocketJoint.h" +#include "../engine/ConstraintSolver.h" using namespace reactphysics3d; // Constructor BallAndSocketJoint::BallAndSocketJoint(const BallAndSocketJointInfo &jointInfo) - : Constraint(jointInfo){ + : Constraint(jointInfo), mImpulse(Vector3(0, 0, 0)) { + // Compute the local-space anchor point for each body + mLocalAnchorPointBody1 = mBody1->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; + mLocalAnchorPointBody1 = mBody1->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; } // Destructor @@ -39,3 +43,92 @@ BallAndSocketJoint::~BallAndSocketJoint() { } +// Initialize before solving the constraint +void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverData) { + + // Initialize the bodies index in the velocity array + mIndexBody1 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody1)->second; + mIndexBody2 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody2)->second; + + // Get the bodies positions and orientations + const Vector3& x1 = mBody1->getTransform().getPosition(); + const Vector3& x2 = mBody2->getTransform().getPosition(); + const Quaternion& orientationBody1 = mBody1->getTransform().getOrientation(); + const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); + + // Get the inertia tensor of bodies + Matrix3x3 inverseInertiaTensorBody1 = mBody1->getInertiaTensorInverseWorld(); + Matrix3x3 inverseInertiaTensorBody2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the vector from body center to anchor point in local-space + const Vector3 u1Local = mLocalAnchorPointBody1 - x1; + const Vector3 u2Local = mLocalAnchorPointBody2 - x2; + + // Compute the vector from body center to the anchor point in world-space + mU1World = orientationBody1 * u1Local; + mU2World = orientationBody2 * u2Local; + + // Compute the corresponding skew-symmetric matrices + Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mU1World); + Matrix3x3 skewSymmetricMatrixU2= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mU2World); + + // Compute the matrix JM^-1J^t + decimal inverseMassBodies = mBody1->getMassInverse() + mBody2->getMassInverse(); + Matrix3x3 massMatrix= Matrix3x3(inverseMassBodies, 0, 0, + 0, inverseMassBodies, 0, + 0, 0, inverseMassBodies) + + skewSymmetricMatrixU1 * inverseInertiaTensorBody1 * skewSymmetricMatrixU1+ + skewSymmetricMatrixU2 * inverseInertiaTensorBody2 * skewSymmetricMatrixU2; + + // Compute the inverse mass matrix K + mInverseMassMatrix = massMatrix.getInverse(); +} + +// Solve the constraint +void BallAndSocketJoint::solve(const ConstraintSolverData& constraintSolverData) { + + // Get the body positions + const Vector3& x1 = mBody1->getTransform().getPosition(); + const Vector3& x2 = mBody2->getTransform().getPosition(); + + // Get the velocities + Vector3& v1 = constraintSolverData.linearVelocities[mIndexBody1]; + Vector3& v2 = constraintSolverData.linearVelocities[mIndexBody2]; + Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; + Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + decimal inverseMassBody1 = mBody1->getMassInverse(); + decimal inverseMassBody2 = mBody2->getMassInverse(); + Matrix3x3 inverseInertiaTensorBody1 = mBody1->getInertiaTensorInverseWorld(); + Matrix3x3 inverseInertiaTensorBody2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute J*v + Vector3 Jv = -v1 + mU1World.cross(w1) + v2 - mU2World.cross(w2); + + // Compute the bias "b" of the constraint + decimal beta = 0.8; // TODO : Use a constant here + decimal biasFactor = -(beta/constraintSolverData.timeStep); + Vector3 b = biasFactor * (x2 + mU2World - x1 - mU1World); + + // Compute the Lagrange multiplier lambda + Vector3 deltaLambda = mInverseMassMatrix * (-Jv - b); + mImpulse = mImpulse + deltaLambda; + + // Compute the impulse P=J^T * lambda + Vector3 linearImpulseBody1 = -deltaLambda; + Vector3 angularImpulseBody1 = mU1World.cross(deltaLambda); + Vector3 linearImpulseBody2 = deltaLambda; + Vector3 angularImpulseBody2 = -mU2World.cross(deltaLambda); + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + w1 += inverseInertiaTensorBody1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += inverseInertiaTensorBody2 * angularImpulseBody2; + } +} + diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index e7e3f5ea..a69507ca 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -44,7 +44,7 @@ struct BallAndSocketJointInfo : public ConstraintInfo { // -------------------- Attributes -------------------- // /// Anchor point (in world space coordinates) - Vector3 anchorPoint; + Vector3 anchorPointWorldSpace; /// Constructor BallAndSocketJointInfo() : ConstraintInfo(BALLSOCKETJOINT) {} @@ -62,10 +62,28 @@ class BallAndSocketJoint : public Constraint { // -------------------- Attributes -------------------- // /// Anchor point of body 1 (in local space coordinates) - Vector3 mLocalAnchorPoint1; + Vector3 mLocalAnchorPointBody1; /// Anchor point of body 2 (in local space coordinates) - Vector3 mLocalAnchorPoint2; + Vector3 mLocalAnchorPointBody2; + + /// Vector from center of body 2 to anchor point in world-space + Vector3 mU1World; + + /// Vector from center of body 2 to anchor point in world-space + Vector3 mU2World; + + /// Skew-Symmetric matrix for cross product with vector mU1World + Matrix3x3 mSkewSymmetricMatrixU1World; + + /// Skew-Symmetric matrix for cross product with vector mU2World + Matrix3x3 mSkewSymmetricMatrixU2World; + + /// Inverse mass matrix K=JM^-1J^-t of the constraint + Matrix3x3 mInverseMassMatrix; + + /// Accumulated impulse + Vector3 mImpulse; public : @@ -79,6 +97,12 @@ class BallAndSocketJoint : public Constraint { /// Return the number of bytes used by the joint virtual size_t getSizeInBytes() const; + + /// Initialize before solving the constraint + virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); + + /// Solve the constraint + virtual void solve(const ConstraintSolverData& constraintSolverData); }; // Return the number of bytes used by the joint diff --git a/src/constraint/Constraint.h b/src/constraint/Constraint.h index de09c956..ae3918e0 100644 --- a/src/constraint/Constraint.h +++ b/src/constraint/Constraint.h @@ -36,6 +36,9 @@ namespace reactphysics3d { // Enumeration for the type of a constraint enum ConstraintType {CONTACT, BALLSOCKETJOINT}; +// Class declarations +struct ConstraintSolverData; + // Structure ConstraintInfo /** * This structure is used to gather the information needed to create a constraint. @@ -94,6 +97,12 @@ class Constraint { /// Type of the constraint const ConstraintType mType; + /// Body 1 index in the velocity array to solve the constraint + uint mIndexBody1; + + /// Body 2 index in the velocity array to solve the constraint + uint mIndexBody2; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -126,6 +135,12 @@ class Constraint { /// Return the number of bytes used by the constraint virtual size_t getSizeInBytes() const = 0; + + /// Initialize before solving the constraint + virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData) = 0; + + /// Solve the constraint + virtual void solve(const ConstraintSolverData& constraintSolverData) = 0; }; // Return the reference to the body 1 diff --git a/src/constraint/ContactPoint.cpp b/src/constraint/ContactPoint.cpp index 93a7ec3d..b945c8e3 100644 --- a/src/constraint/ContactPoint.cpp +++ b/src/constraint/ContactPoint.cpp @@ -50,3 +50,13 @@ ContactPoint::ContactPoint(const ContactPointInfo& contactInfo) ContactPoint::~ContactPoint() { } + +// Initialize before solving the constraint +void ContactPoint::initBeforeSolve(const ConstraintSolverData& constraintSolverData) { + +} + +// Solve the constraint +void ContactPoint::solve(const ConstraintSolverData& constraintSolverData) { + +} diff --git a/src/constraint/ContactPoint.h b/src/constraint/ContactPoint.h index 14852282..3aff1334 100644 --- a/src/constraint/ContactPoint.h +++ b/src/constraint/ContactPoint.h @@ -224,6 +224,12 @@ class ContactPoint : public Constraint { /// Return the number of bytes used by the contact point virtual size_t getSizeInBytes() const; + /// Initialize before solving the constraint + virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); + + /// Solve the constraint + virtual void solve(const ConstraintSolverData& constraintSolverData); + #ifdef VISUAL_DEBUG /// Draw the contact (for debugging) void draw() const; diff --git a/src/engine/ConstraintSolver.cpp b/src/engine/ConstraintSolver.cpp index 0291eee4..53a271af 100644 --- a/src/engine/ConstraintSolver.cpp +++ b/src/engine/ConstraintSolver.cpp @@ -31,12 +31,14 @@ using namespace reactphysics3d; // Constructor ConstraintSolver::ConstraintSolver(std::set& joints, - std::vector& constrainedLinearVelocities, - std::vector& constrainedAngularVelocities, + std::vector& linearVelocities, + std::vector& angularVelocities, const std::map& mapBodyToVelocityIndex) - : mJoints(joints), mConstrainedLinearVelocities(constrainedLinearVelocities), - mConstrainedAngularVelocities(constrainedAngularVelocities), - mMapBodyToConstrainedVelocityIndex(mapBodyToVelocityIndex) { + : mJoints(joints), mLinearVelocities(linearVelocities), + mAngularVelocities(angularVelocities), + mMapBodyToConstrainedVelocityIndex(mapBodyToVelocityIndex), + mIsWarmStartingActive(false), mConstraintSolverData(linearVelocities, + angularVelocities, mapBodyToVelocityIndex){ } @@ -52,6 +54,28 @@ void ConstraintSolver::initialize(decimal dt) { // Set the current time step mTimeStep = dt; + + // Initialize the constraint solver data used to initialize and solve the constraints + mConstraintSolverData.timeStep = mTimeStep; + mConstraintSolverData.isWarmStartingActive = mIsWarmStartingActive; + + // For each joint + std::set::iterator it; + for (it = mJoints.begin(); it != mJoints.end(); ++it) { + + Constraint* joint = (*it); + + // Get the rigid bodies of the joint + RigidBody* body1 = joint->getBody1(); + RigidBody* body2 = joint->getBody2(); + + // Add the bodies to the set of constrained bodies + mConstraintBodies.insert(body1); + mConstraintBodies.insert(body2); + + // Initialize the constraint before solving it + joint->initBeforeSolve(mConstraintSolverData); + } } // Solve the constraints @@ -59,4 +83,13 @@ void ConstraintSolver::solve() { PROFILE("ConstraintSolver::solve()"); + // For each joint + std::set::iterator it; + for (it = mJoints.begin(); it != mJoints.end(); ++it) { + + Constraint* joint = (*it); + + // Solve the constraint + joint->solve(mConstraintSolverData); + } } diff --git a/src/engine/ConstraintSolver.h b/src/engine/ConstraintSolver.h index 08e4a08f..d72f9e98 100644 --- a/src/engine/ConstraintSolver.h +++ b/src/engine/ConstraintSolver.h @@ -35,6 +35,43 @@ namespace reactphysics3d { +// Structure ConstraintSolverData +/** + * This structure contains data from the constraint solver that are used to solve + * each joint constraint. + */ +struct ConstraintSolverData { + + public : + + /// Current time step of the simulation + decimal timeStep; + + /// Reference to the bodies linear velocities + std::vector& linearVelocities; + + /// Reference to the bodies angular velocities + std::vector& angularVelocities; + + /// Reference to the map that associates rigid body to their index + /// in the constrained velocities array + const std::map& mapBodyToConstrainedVelocityIndex; + + /// True if warm starting of the solver is active + bool isWarmStartingActive; + + /// Constructor + ConstraintSolverData(std::vector& refLinearVelocities, + std::vector& refAngularVelocities, + const std::map& refMapBodyToConstrainedVelocityIndex) + :linearVelocities(refLinearVelocities), + angularVelocities(refAngularVelocities), + mapBodyToConstrainedVelocityIndex(refMapBodyToConstrainedVelocityIndex){ + + } + +}; + // Class ConstraintSolver /** * This class represents the constraint solver that is used to solve constraints between @@ -113,31 +150,38 @@ class ConstraintSolver { /// Reference to all the joints of the world std::set& mJoints; + /// Constrained bodies + std::set mConstraintBodies; + /// Reference to the array of constrained linear velocities (state of the linear velocities /// after solving the constraints) - std::vector& mConstrainedLinearVelocities; + std::vector& mLinearVelocities; /// Reference to the array of constrained angular velocities (state of the angular velocities /// after solving the constraints) - std::vector& mConstrainedAngularVelocities; + std::vector& mAngularVelocities; - /// Reference to the map of rigid body to their index in the constrained velocities array + /// Reference to the map that associates rigid body to their index in + /// the constrained velocities array const std::map& mMapBodyToConstrainedVelocityIndex; - /// Number of iterations of the contact solver - uint mNbIterations; - /// Current time step decimal mTimeStep; + /// True if the warm starting of the solver is active + bool mIsWarmStartingActive; + + /// Constraint solver data used to initialize and solve the constraints + ConstraintSolverData mConstraintSolverData; + public : // -------------------- Methods -------------------- // /// Constructor ConstraintSolver(std::set& joints, - std::vector& constrainedLinearVelocities, - std::vector& constrainedAngularVelocities, + std::vector& linearVelocities, + std::vector& angularVelocities, const std::map& mapBodyToVelocityIndex); /// Destructor diff --git a/src/engine/ContactSolver.cpp b/src/engine/ContactSolver.cpp index 3399ac0d..8467a710 100644 --- a/src/engine/ContactSolver.cpp +++ b/src/engine/ContactSolver.cpp @@ -46,8 +46,8 @@ ContactSolver::ContactSolver(std::vector& contactManifolds, :mContactManifolds(contactManifolds), mSplitLinearVelocities(NULL), mSplitAngularVelocities(NULL), mContactConstraints(NULL), - mConstrainedLinearVelocities(constrainedLinearVelocities), - mConstrainedAngularVelocities(constrainedAngularVelocities), + mLinearVelocities(constrainedLinearVelocities), + mAngularVelocities(constrainedAngularVelocities), mMapBodyToConstrainedVelocityIndex(mapBodyToVelocityIndex), mIsWarmStartingActive(true), mIsSplitImpulseActive(true), mIsSolveFrictionAtContactManifoldCenterActive(true) { @@ -186,8 +186,8 @@ void ContactSolver::initialize(decimal dt) { assert(mConstraintBodies.size() > 0); assert(mMapBodyToConstrainedVelocityIndex.size() >= mConstraintBodies.size()); - assert(mConstrainedLinearVelocities.size() >= mConstraintBodies.size()); - assert(mConstrainedAngularVelocities.size() >= mConstraintBodies.size()); + assert(mLinearVelocities.size() >= mConstraintBodies.size()); + assert(mAngularVelocities.size() >= mConstraintBodies.size()); // Initialize the split impulse velocities initializeSplitImpulseVelocities(); @@ -231,10 +231,10 @@ void ContactSolver::initializeContactConstraints() { } // Get the velocities of the bodies - const Vector3& v1 = mConstrainedLinearVelocities[manifold.indexBody1]; - const Vector3& w1 = mConstrainedAngularVelocities[manifold.indexBody1]; - const Vector3& v2 = mConstrainedLinearVelocities[manifold.indexBody2]; - const Vector3& w2 = mConstrainedAngularVelocities[manifold.indexBody2]; + const Vector3& v1 = mLinearVelocities[manifold.indexBody1]; + const Vector3& w1 = mAngularVelocities[manifold.indexBody1]; + const Vector3& v2 = mLinearVelocities[manifold.indexBody2]; + const Vector3& w2 = mAngularVelocities[manifold.indexBody2]; // For each contact point constraint for (uint i=0; i #include /// ReactPhysics3D namespace namespace reactphysics3d { -// Declarations -class DynamicsWorld; - -// Structure Impulse -/** - * Represents an impulse that we can apply to bodies in the contact or constraint solver. - */ -struct Impulse { - - public: - - /// Linear impulse applied to the first body - const Vector3 linearImpulseBody1; - - /// Linear impulse applied to the second body - const Vector3 linearImpulseBody2; - - /// Angular impulse applied to the first body - const Vector3 angularImpulseBody1; - - /// Angular impulse applied to the second body - const Vector3 angularImpulseBody2; - - /// Constructor - Impulse(const Vector3& linearImpulseBody1, const Vector3& angularImpulseBody1, - const Vector3& linearImpulseBody2, const Vector3& angularImpulseBody2) - : linearImpulseBody1(linearImpulseBody1), angularImpulseBody1(angularImpulseBody1), - linearImpulseBody2(linearImpulseBody2), angularImpulseBody2(angularImpulseBody2) { - - } -}; - // Class Contact Solver /** @@ -85,7 +54,7 @@ struct Impulse { * F_c = J^t * lambda where J^t is the transpose of the Jacobian matrix and lambda is a * Lagrange multiplier. Therefore, finding the force F_c is equivalent to finding the Lagrange * multiplier lambda. - + * * An impulse P = F * dt where F is a force and dt is the timestep. We can apply impulses a * body to change its velocity. The idea of the Sequential Impulse technique is to apply * impulses to bodies of each constraints in order to keep the constraint satisfied. @@ -363,13 +332,11 @@ class ContactSolver { /// Constrained bodies std::set mConstraintBodies; - /// Reference to the array of constrained linear velocities (state of the linear velocities - /// after solving the constraints) - std::vector& mConstrainedLinearVelocities; + /// Reference to the array of linear velocities + std::vector& mLinearVelocities; - /// Reference to the array of constrained angular velocities (state of the angular velocities - /// after solving the constraints) - std::vector& mConstrainedAngularVelocities; + /// Reference to the array of angular velocities + std::vector& mAngularVelocities; /// Reference to the map of rigid body to their index in the constrained velocities array const std::map& mMapBodyToConstrainedVelocityIndex; diff --git a/src/engine/Impulse.h b/src/engine/Impulse.h new file mode 100644 index 00000000..5ed9e228 --- /dev/null +++ b/src/engine/Impulse.h @@ -0,0 +1,65 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_IMPULSE_H +#define REACTPHYSICS3D_IMPULSE_H + +// Libraries +#include "../mathematics/mathematics.h" + +namespace reactphysics3d { + +// Structure Impulse +/** + * Represents an impulse that we can apply to bodies in the contact or constraint solver. + */ +struct Impulse { + + public: + + /// Linear impulse applied to the first body + const Vector3 linearImpulseBody1; + + /// Linear impulse applied to the second body + const Vector3 linearImpulseBody2; + + /// Angular impulse applied to the first body + const Vector3 angularImpulseBody1; + + /// Angular impulse applied to the second body + const Vector3 angularImpulseBody2; + + /// Constructor + Impulse(const Vector3& linearImpulseBody1, const Vector3& angularImpulseBody1, + const Vector3& linearImpulseBody2, const Vector3& angularImpulseBody2) + : linearImpulseBody1(linearImpulseBody1), angularImpulseBody1(angularImpulseBody1), + linearImpulseBody2(linearImpulseBody2), angularImpulseBody2(angularImpulseBody2) { + + } +}; + +} + +#endif diff --git a/src/mathematics/Matrix3x3.h b/src/mathematics/Matrix3x3.h index 560d4031..d9382b04 100644 --- a/src/mathematics/Matrix3x3.h +++ b/src/mathematics/Matrix3x3.h @@ -105,6 +105,10 @@ class Matrix3x3 { /// Return the 3x3 identity matrix static Matrix3x3 identity(); + /// Return a skew-symmetric matrix using a given vector that can be used + /// to compute cross product with another vector using matrix multiplication + static Matrix3x3 computeSkewSymmetricMatrixForCrossProduct(const Vector3& vector); + /// Overloaded operator for addition friend Matrix3x3 operator+(const Matrix3x3& matrix1, const Matrix3x3& matrix2); @@ -215,6 +219,12 @@ inline Matrix3x3 Matrix3x3::identity() { return Matrix3x3(1.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 1.0); } +// Return a skew-symmetric matrix using a given vector that can be used +// to compute cross product with another vector using matrix multiplication +inline Matrix3x3 Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(const Vector3& vector) { + return Matrix3x3(0, -vector.z, vector.y, vector.z, 0, -vector.x, -vector.y, vector.x, 0); +} + // Return the matrix with absolute values inline Matrix3x3 Matrix3x3::getAbsoluteMatrix() const { return Matrix3x3(fabs(mRows[0][0]), fabs(mRows[0][1]), fabs(mRows[0][2]), diff --git a/src/mathematics/Quaternion.h b/src/mathematics/Quaternion.h index c6fd64a1..eb957db7 100644 --- a/src/mathematics/Quaternion.h +++ b/src/mathematics/Quaternion.h @@ -131,7 +131,7 @@ struct Quaternion { Quaternion operator*(const Quaternion& quaternion) const; /// Overloaded operator for the multiplication with a vector - Vector3 operator*(const Vector3& point); + Vector3 operator*(const Vector3& point) const; /// Overloaded operator for assignment Quaternion& operator=(const Quaternion& quaternion); @@ -250,7 +250,7 @@ inline Quaternion Quaternion::operator*(const Quaternion& quaternion) const { // Overloaded operator for the multiplication with a vector. /// This methods rotates a point given the rotation of a quaternion. -inline Vector3 Quaternion::operator*(const Vector3& point) { +inline Vector3 Quaternion::operator*(const Vector3& point) const { Quaternion p(point.x, point.y, point.z, 0.0); return (((*this) * p) * getConjugate()).getVectorV(); } diff --git a/src/mathematics/Transform.h b/src/mathematics/Transform.h index 2d3933db..9f5e07e1 100644 --- a/src/mathematics/Transform.h +++ b/src/mathematics/Transform.h @@ -92,7 +92,7 @@ class Transform { void getOpenGLMatrix(decimal* openglMatrix) const; /// Return the inverse of the transform - Transform inverse() const; + Transform getInverse() const; /// Return an interpolated transform static Transform interpolateTransforms(const Transform& oldTransform, @@ -167,7 +167,7 @@ inline void Transform::getOpenGLMatrix(decimal* openglMatrix) const { } // Return the inverse of the transform -inline Transform Transform::inverse() const { +inline Transform Transform::getInverse() const { const Quaternion& invQuaternion = mOrientation.getInverse(); Matrix3x3 invMatrix = invQuaternion.getMatrix(); return Transform(invMatrix * (-mPosition), invQuaternion); diff --git a/src/reactphysics3d.h b/src/reactphysics3d.h index 09331956..5f463e13 100644 --- a/src/reactphysics3d.h +++ b/src/reactphysics3d.h @@ -47,6 +47,7 @@ #include "collision/shapes/ConeShape.h" #include "collision/shapes/CylinderShape.h" #include "collision/shapes/AABB.h" +#include "constraint/BallAndSocketJoint.h" /// Alias to the ReactPhysics3D namespace namespace rp3d = reactphysics3d; diff --git a/test/tests/mathematics/TestMatrix3x3.h b/test/tests/mathematics/TestMatrix3x3.h index ccdda9a9..ca0a2bb2 100644 --- a/test/tests/mathematics/TestMatrix3x3.h +++ b/test/tests/mathematics/TestMatrix3x3.h @@ -196,6 +196,15 @@ class TestMatrix3x3 : public Test { test(matrix.getAbsoluteMatrix() == Matrix3x3(24, 64, 253, 35, 52, 72, 21, 35, 363)); Matrix3x3 absoluteMatrix = matrix2.getAbsoluteMatrix(); test(absoluteMatrix == Matrix3x3(2, 3, 4, 5, 6, 7, 8, 9, 10)); + + // Test method that computes skew-symmetric matrix for cross product + Vector3 vector1(3, -5, 6); + Vector3 vector2(73, 42, 26); + Matrix3x3 skewMatrix = Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(vector1); + test(skewMatrix == Matrix3x3(0, -6, -5, 6, 0, -3, 5, 3, 0)); + Vector3 crossProduct1 = vector1.cross(vector2); + Vector3 crossProduct2 = skewMatrix * vector2; + test(crossProduct1 == crossProduct2); } /// Test the operators diff --git a/test/tests/mathematics/TestTransform.h b/test/tests/mathematics/TestTransform.h index 4c700a0d..72804053 100644 --- a/test/tests/mathematics/TestTransform.h +++ b/test/tests/mathematics/TestTransform.h @@ -114,7 +114,7 @@ class TestTransform : public Test { /// Test the inverse void testInverse() { - Transform inverseTransform = mTransform1.inverse(); + Transform inverseTransform = mTransform1.getInverse(); Vector3 vector(2, 3, 4); Vector3 tempVector = mTransform1 * vector; Vector3 tempVector2 = inverseTransform * tempVector; From 0071ed16a8d3f89befe1309be491512d215e3506 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 2 May 2013 22:51:31 +0200 Subject: [PATCH 18/66] Add comments in the dynamics world code --- src/engine/DynamicsWorld.cpp | 10 ++++++++-- src/engine/DynamicsWorld.h | 4 ++-- 2 files changed, 10 insertions(+), 4 deletions(-) diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 237d5236..aac339ac 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -121,7 +121,9 @@ void DynamicsWorld::update() { setInterpolationFactorToAllBodies(); } -// Integrate position and orientation of the rigid bodies +// Integrate position and orientation of the rigid bodies. +/// The positions and orientations of the bodies are integrated using +/// the sympletic Euler time stepping scheme. void DynamicsWorld::integrateRigidBodiesPositions() { PROFILE("DynamicsWorld::updateRigidBodiesPositionAndOrientation()"); @@ -197,7 +199,11 @@ void DynamicsWorld::setInterpolationFactorToAllBodies() { } } -// Integrate the velocities of rigid bodies +// Integrate the velocities of rigid bodies. +/// This method only set the temporary velocities but does not update +/// the actual velocitiy of the bodies. The velocities updated in this method +/// might violate the constraints and will be corrected in the constraint and +/// contact solver. void DynamicsWorld::integrateRigidBodiesVelocities() { // TODO : Use better memory allocation here diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index 22fbc495..0503ebe2 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -99,7 +99,7 @@ class DynamicsWorld : public CollisionWorld { /// Private assignment operator DynamicsWorld& operator=(const DynamicsWorld& world); - /// Integrate the positions and orientations of rigid bodies + /// Integrate the positions and orientations of rigid bodies. void integrateRigidBodiesPositions(); /// Update the position and orientation of a body @@ -109,7 +109,7 @@ class DynamicsWorld : public CollisionWorld { /// Compute and set the interpolation factor to all bodies void setInterpolationFactorToAllBodies(); - /// Integrate the velocities of rigid bodies + /// Integrate the velocities of rigid bodies. void integrateRigidBodiesVelocities(); /// Solve the contacts and constraints From af2fcaeb8213c2848b95f0943a2b4ad14af89121 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 2 May 2013 23:55:10 +0200 Subject: [PATCH 19/66] Fix two issues --- src/constraint/BallAndSocketJoint.cpp | 10 ++++++---- src/engine/DynamicsWorld.cpp | 14 +++++++++----- 2 files changed, 15 insertions(+), 9 deletions(-) diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index 214b7337..8d7ae17d 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -34,8 +34,10 @@ BallAndSocketJoint::BallAndSocketJoint(const BallAndSocketJointInfo &jointInfo) : Constraint(jointInfo), mImpulse(Vector3(0, 0, 0)) { // Compute the local-space anchor point for each body - mLocalAnchorPointBody1 = mBody1->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; - mLocalAnchorPointBody1 = mBody1->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; + mLocalAnchorPointBody1 = mBody1->getTransform().getOrientation().getInverse() * + jointInfo.anchorPointWorldSpace; + mLocalAnchorPointBody2 = mBody2->getTransform().getOrientation().getInverse() * + jointInfo.anchorPointWorldSpace; } // Destructor @@ -107,13 +109,13 @@ void BallAndSocketJoint::solve(const ConstraintSolverData& constraintSolverData) Vector3 Jv = -v1 + mU1World.cross(w1) + v2 - mU2World.cross(w2); // Compute the bias "b" of the constraint - decimal beta = 0.8; // TODO : Use a constant here + decimal beta = 0.7; // TODO : Use a constant here decimal biasFactor = -(beta/constraintSolverData.timeStep); Vector3 b = biasFactor * (x2 + mU2World - x1 - mU1World); // Compute the Lagrange multiplier lambda Vector3 deltaLambda = mInverseMassMatrix * (-Jv - b); - mImpulse = mImpulse + deltaLambda; + mImpulse += deltaLambda; // Compute the impulse P=J^T * lambda Vector3 linearImpulseBody1 = -deltaLambda; diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index aac339ac..8dce0ece 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -219,11 +219,15 @@ void DynamicsWorld::integrateRigidBodiesVelocities() { RigidBody* rigidBody = *it; mMapBodyToConstrainedVelocityIndex.insert(std::make_pair(rigidBody, i)); - // Integrate the external force to get the new velocity of the body - mConstrainedLinearVelocities[i] = rigidBody->getLinearVelocity() + - dt * rigidBody->getMassInverse() * rigidBody->getExternalForce(); - mConstrainedAngularVelocities[i] = rigidBody->getAngularVelocity() + - dt * rigidBody->getInertiaTensorInverseWorld() * rigidBody->getExternalTorque(); + // If the body is allowed to move + if (rigidBody->getIsMotionEnabled()) { + + // Integrate the external force to get the new velocity of the body + mConstrainedLinearVelocities[i] = rigidBody->getLinearVelocity() + + dt * rigidBody->getMassInverse() * rigidBody->getExternalForce(); + mConstrainedAngularVelocities[i] = rigidBody->getAngularVelocity() + + dt * rigidBody->getInertiaTensorInverseWorld() * rigidBody->getExternalTorque(); + } i++; } From da78e5d79a547b929341a6fedfa1b0d9d7dd924c Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sat, 4 May 2013 09:20:53 +0200 Subject: [PATCH 20/66] Modifications in the BallAndSocketJoint to make it work --- examples/joints/Scene.cpp | 4 ++-- src/body/RigidBody.h | 4 ++-- src/constraint/BallAndSocketJoint.cpp | 30 +++++++++++---------------- src/engine/DynamicsWorld.cpp | 5 ++++- 4 files changed, 20 insertions(+), 23 deletions(-) diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index 5e93b09c..8c1ea4d0 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -167,7 +167,7 @@ void Scene::createBallAndSocketJoints() { // --------------- Create the second box --------------- // // Position of the box - openglframework::Vector3 positionBox2(0, 10, 0); + openglframework::Vector3 positionBox2(5, 10, 0); // Create a box and a corresponding rigid in the dynamics world mBallAndSocketJointBox2 = new Box(BOX_SIZE, positionBox2 , BOX_MASS, mDynamicsWorld); @@ -184,7 +184,7 @@ void Scene::createBallAndSocketJoints() { rp3d::BallAndSocketJointInfo jointInfo; jointInfo.body1 = mBallAndSocketJointBox1->getRigidBody(); jointInfo.body2 = mBallAndSocketJointBox2->getRigidBody(); - jointInfo.anchorPointWorldSpace = rp3d::Vector3(0, 12.5, 0); + jointInfo.anchorPointWorldSpace = rp3d::Vector3(0, 10, 0); // Create the joint in the dynamics world mBallAndSocketJoint = dynamic_cast( diff --git a/src/body/RigidBody.h b/src/body/RigidBody.h index 0128bb42..1f340961 100644 --- a/src/body/RigidBody.h +++ b/src/body/RigidBody.h @@ -135,7 +135,7 @@ class RigidBody : public CollisionBody { decimal getMassInverse() const; /// Return the local inertia tensor of the body (in body coordinates) - Matrix3x3 getInertiaTensorLocal() const; + const Matrix3x3& getInertiaTensorLocal() const; /// Set the local inertia tensor of the body (in body coordinates) void setInertiaTensorLocal(const Matrix3x3& inertiaTensorLocal); @@ -222,7 +222,7 @@ inline decimal RigidBody::getMassInverse() const { } // Return the local inertia tensor of the body (in body coordinates) -inline Matrix3x3 RigidBody::getInertiaTensorLocal() const { +inline const Matrix3x3& RigidBody::getInertiaTensorLocal() const { return mInertiaTensorLocal; } diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index 8d7ae17d..dab548a8 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -34,10 +34,8 @@ BallAndSocketJoint::BallAndSocketJoint(const BallAndSocketJointInfo &jointInfo) : Constraint(jointInfo), mImpulse(Vector3(0, 0, 0)) { // Compute the local-space anchor point for each body - mLocalAnchorPointBody1 = mBody1->getTransform().getOrientation().getInverse() * - jointInfo.anchorPointWorldSpace; - mLocalAnchorPointBody2 = mBody2->getTransform().getOrientation().getInverse() * - jointInfo.anchorPointWorldSpace; + mLocalAnchorPointBody1 = mBody1->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; + mLocalAnchorPointBody2 = mBody2->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; } // Destructor @@ -53,8 +51,6 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS mIndexBody2 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody2)->second; // Get the bodies positions and orientations - const Vector3& x1 = mBody1->getTransform().getPosition(); - const Vector3& x2 = mBody2->getTransform().getPosition(); const Quaternion& orientationBody1 = mBody1->getTransform().getOrientation(); const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); @@ -62,13 +58,9 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS Matrix3x3 inverseInertiaTensorBody1 = mBody1->getInertiaTensorInverseWorld(); Matrix3x3 inverseInertiaTensorBody2 = mBody2->getInertiaTensorInverseWorld(); - // Compute the vector from body center to anchor point in local-space - const Vector3 u1Local = mLocalAnchorPointBody1 - x1; - const Vector3 u2Local = mLocalAnchorPointBody2 - x2; - // Compute the vector from body center to the anchor point in world-space - mU1World = orientationBody1 * u1Local; - mU2World = orientationBody2 * u2Local; + mU1World = orientationBody1 * mLocalAnchorPointBody1; + mU2World = orientationBody2 * mLocalAnchorPointBody2; // Compute the corresponding skew-symmetric matrices Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mU1World); @@ -79,8 +71,10 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS Matrix3x3 massMatrix= Matrix3x3(inverseMassBodies, 0, 0, 0, inverseMassBodies, 0, 0, 0, inverseMassBodies) + - skewSymmetricMatrixU1 * inverseInertiaTensorBody1 * skewSymmetricMatrixU1+ - skewSymmetricMatrixU2 * inverseInertiaTensorBody2 * skewSymmetricMatrixU2; + skewSymmetricMatrixU1 * inverseInertiaTensorBody1 * + skewSymmetricMatrixU1.getTranspose() + + skewSymmetricMatrixU2 * inverseInertiaTensorBody2 * + skewSymmetricMatrixU2.getTranspose(); // Compute the inverse mass matrix K mInverseMassMatrix = massMatrix.getInverse(); @@ -109,8 +103,8 @@ void BallAndSocketJoint::solve(const ConstraintSolverData& constraintSolverData) Vector3 Jv = -v1 + mU1World.cross(w1) + v2 - mU2World.cross(w2); // Compute the bias "b" of the constraint - decimal beta = 0.7; // TODO : Use a constant here - decimal biasFactor = -(beta/constraintSolverData.timeStep); + decimal beta = decimal(0.2); // TODO : Use a constant here + decimal biasFactor = (beta / constraintSolverData.timeStep); Vector3 b = biasFactor * (x2 + mU2World - x1 - mU1World); // Compute the Lagrange multiplier lambda @@ -119,9 +113,9 @@ void BallAndSocketJoint::solve(const ConstraintSolverData& constraintSolverData) // Compute the impulse P=J^T * lambda Vector3 linearImpulseBody1 = -deltaLambda; - Vector3 angularImpulseBody1 = mU1World.cross(deltaLambda); + Vector3 angularImpulseBody1 = deltaLambda.cross(mU1World); Vector3 linearImpulseBody2 = deltaLambda; - Vector3 angularImpulseBody2 = -mU2World.cross(deltaLambda); + Vector3 angularImpulseBody2 = -deltaLambda.cross(mU2World); // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 8dce0ece..ed69e4a5 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -407,11 +407,14 @@ void DynamicsWorld::destroyJoint(Constraint* joint) { // Remove the joint from the world mJoints.erase(joint); + // Get the size in bytes of the joint + size_t nbBytes = joint->getSizeInBytes(); + // Call the destructor of the joint joint->Constraint::~Constraint(); // Release the allocated memory - mMemoryAllocator.release(joint, joint->getSizeInBytes()); + mMemoryAllocator.release(joint, nbBytes); } // Notify the world about a new broad-phase overlapping pair From b87f981827807581a6142cb05e3b8cc3aaa3258a Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 8 May 2013 23:33:04 +0200 Subject: [PATCH 21/66] Start working on the slider joint --- examples/fallingcubes/Scene.cpp | 2 +- examples/joints/Scene.cpp | 10 +- src/configuration.h | 23 ++++- src/constraint/BallAndSocketJoint.cpp | 32 ++++--- src/constraint/BallAndSocketJoint.h | 14 ++- src/constraint/Constraint.cpp | 3 +- src/constraint/Constraint.h | 23 ++++- src/constraint/ContactPoint.cpp | 9 +- src/constraint/ContactPoint.h | 7 +- src/constraint/SliderJoint.cpp | 105 +++++++++++++++++++++ src/constraint/SliderJoint.h | 131 ++++++++++++++++++++++++++ src/engine/ConstraintSolver.cpp | 28 +++++- src/engine/ConstraintSolver.h | 24 ++++- src/engine/DynamicsWorld.cpp | 38 +++++++- src/engine/DynamicsWorld.h | 58 +++++++++--- 15 files changed, 446 insertions(+), 61 deletions(-) create mode 100644 src/constraint/SliderJoint.cpp create mode 100644 src/constraint/SliderJoint.h diff --git a/examples/fallingcubes/Scene.cpp b/examples/fallingcubes/Scene.cpp index 693a0fa5..b11f3a84 100644 --- a/examples/fallingcubes/Scene.cpp +++ b/examples/fallingcubes/Scene.cpp @@ -54,7 +54,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), mDynamicsWorld = new rp3d::DynamicsWorld(gravity, timeStep); // Set the number of iterations of the constraint solver - mDynamicsWorld->setNbIterationsSolver(15); + mDynamicsWorld->setNbIterationsVelocitySolver(15); float radius = 2.0f; diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index 8c1ea4d0..c1b5060b 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -54,7 +54,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), mDynamicsWorld = new rp3d::DynamicsWorld(gravity, timeStep); // Set the number of iterations of the constraint solver - mDynamicsWorld->setNbIterationsSolver(15); + mDynamicsWorld->setNbIterationsVelocitySolver(15); // Create the Ball-and-Socket joint createBallAndSocketJoints(); @@ -181,10 +181,10 @@ void Scene::createBallAndSocketJoints() { // --------------- Create the joint --------------- // // Create the joint info object - rp3d::BallAndSocketJointInfo jointInfo; - jointInfo.body1 = mBallAndSocketJointBox1->getRigidBody(); - jointInfo.body2 = mBallAndSocketJointBox2->getRigidBody(); - jointInfo.anchorPointWorldSpace = rp3d::Vector3(0, 10, 0); + rp3d::RigidBody* body1 = mBallAndSocketJointBox1->getRigidBody(); + rp3d::RigidBody* body2 = mBallAndSocketJointBox2->getRigidBody(); + const rp3d::Vector3 anchorPointWorldSpace(0, 10, 0); + rp3d::BallAndSocketJointInfo jointInfo(body1, body2, anchorPointWorldSpace); // Create the joint in the dynamics world mBallAndSocketJoint = dynamic_cast( diff --git a/src/configuration.h b/src/configuration.h index 59319ae8..eae264f9 100644 --- a/src/configuration.h +++ b/src/configuration.h @@ -51,6 +51,22 @@ typedef long unsigned int luint; typedef luint bodyindex; typedef std::pair bodyindexpair; +// ------------------- Enumerations ------------------- // + +/// Position correction technique used in the constraint solver (for joints). +/// BAUMGARTE : Faster but can be innacurate in some situations. This is the option +/// used by default. +/// NON_LINEAR_GAUSS_SEIDEL : Slower but more precise. +enum JointsPositionCorrectionTechnique {BAUMGARTE_JOINTS, NON_LINEAR_GAUSS_SEIDEL}; + +/// Position correction technique used in the contact solver (for contacts) +/// BAUMGARTE : Faster but can be innacurate and can lead to unexpected bounciness +/// in some situations (due to error correction factor being added to +/// the bodies momentum). +/// SPLIT_IMPULSES : A bit slower but the error correction factor is not added to the +/// bodies momentum. This is the option used by default. +enum ContactsPositionCorrectionTechnique {BAUMGARTE_CONTACTS, SPLIT_IMPULSES}; + // ------------------- Constants ------------------- // /// Smallest decimal value (negative) @@ -83,8 +99,11 @@ const decimal PERSISTENT_CONTACT_DIST_THRESHOLD = decimal(0.03); /// Velocity threshold for contact velocity restitution const decimal RESTITUTION_VELOCITY_THRESHOLD = decimal(1.0); -/// Number of iterations when solving a LCP problem -const uint DEFAULT_CONSTRAINTS_SOLVER_NB_ITERATIONS = 15; +/// Number of iterations when solving the velocity constraints of the Sequential Impulse technique +const uint DEFAULT_VELOCITY_SOLVER_NB_ITERATIONS = 15; + +/// Number of iterations when solving the position constraints of the Sequential Impulse technique +const uint DEFAULT_POSITION_SOLVER_NB_ITERATIONS = 3; // TODO : Maybe we can use less iterations here } diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index dab548a8..b4fc6575 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -55,8 +55,8 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); // Get the inertia tensor of bodies - Matrix3x3 inverseInertiaTensorBody1 = mBody1->getInertiaTensorInverseWorld(); - Matrix3x3 inverseInertiaTensorBody2 = mBody2->getInertiaTensorInverseWorld(); + const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // Compute the vector from body center to the anchor point in world-space mU1World = orientationBody1 * mLocalAnchorPointBody1; @@ -66,22 +66,20 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mU1World); Matrix3x3 skewSymmetricMatrixU2= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mU2World); - // Compute the matrix JM^-1J^t + // Compute the matrix K=JM^-1J^t (3x3 matrix) decimal inverseMassBodies = mBody1->getMassInverse() + mBody2->getMassInverse(); Matrix3x3 massMatrix= Matrix3x3(inverseMassBodies, 0, 0, 0, inverseMassBodies, 0, 0, 0, inverseMassBodies) + - skewSymmetricMatrixU1 * inverseInertiaTensorBody1 * - skewSymmetricMatrixU1.getTranspose() + - skewSymmetricMatrixU2 * inverseInertiaTensorBody2 * - skewSymmetricMatrixU2.getTranspose(); + skewSymmetricMatrixU1 * I1 * skewSymmetricMatrixU1.getTranspose() + + skewSymmetricMatrixU2 * I2 * skewSymmetricMatrixU2.getTranspose(); - // Compute the inverse mass matrix K + // Compute the inverse mass matrix K^-1 mInverseMassMatrix = massMatrix.getInverse(); } -// Solve the constraint -void BallAndSocketJoint::solve(const ConstraintSolverData& constraintSolverData) { +// Solve the velocity constraint +void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) { // Get the body positions const Vector3& x1 = mBody1->getTransform().getPosition(); @@ -103,9 +101,12 @@ void BallAndSocketJoint::solve(const ConstraintSolverData& constraintSolverData) Vector3 Jv = -v1 + mU1World.cross(w1) + v2 - mU2World.cross(w2); // Compute the bias "b" of the constraint - decimal beta = decimal(0.2); // TODO : Use a constant here - decimal biasFactor = (beta / constraintSolverData.timeStep); - Vector3 b = biasFactor * (x2 + mU2World - x1 - mU1World); + Vector3 b(0, 0, 0); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + decimal beta = decimal(0.2); // TODO : Use a constant here + decimal biasFactor = (beta / constraintSolverData.timeStep); + b = biasFactor * (x2 + mU2World - x1 - mU1World); + } // Compute the Lagrange multiplier lambda Vector3 deltaLambda = mInverseMassMatrix * (-Jv - b); @@ -128,3 +129,8 @@ void BallAndSocketJoint::solve(const ConstraintSolverData& constraintSolverData) } } +// Solve the position constraint +void BallAndSocketJoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { + +} + diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index a69507ca..0280c28c 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -43,11 +43,14 @@ struct BallAndSocketJointInfo : public ConstraintInfo { // -------------------- Attributes -------------------- // - /// Anchor point (in world space coordinates) + /// Anchor point (in world-space coordinates) Vector3 anchorPointWorldSpace; /// Constructor - BallAndSocketJointInfo() : ConstraintInfo(BALLSOCKETJOINT) {} + BallAndSocketJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, + const Vector3& initAnchorPointWorldSpace) + : ConstraintInfo(rigidBody1, rigidBody2, BALLSOCKETJOINT), + anchorPointWorldSpace(initAnchorPointWorldSpace){} }; // Class BallAndSocketJoint @@ -101,8 +104,11 @@ class BallAndSocketJoint : public Constraint { /// Initialize before solving the constraint virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); - /// Solve the constraint - virtual void solve(const ConstraintSolverData& constraintSolverData); + /// Solve the velocity constraint + virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); + + /// Solve the position constraint + virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); }; // Return the number of bytes used by the joint diff --git a/src/constraint/Constraint.cpp b/src/constraint/Constraint.cpp index 7ecded94..234861e3 100644 --- a/src/constraint/Constraint.cpp +++ b/src/constraint/Constraint.cpp @@ -31,7 +31,8 @@ using namespace reactphysics3d; // Constructor Constraint::Constraint(const ConstraintInfo& constraintInfo) :mBody1(constraintInfo.body1), mBody2(constraintInfo.body2), mActive(true), - mType(constraintInfo.type) { + mType(constraintInfo.type), + mPositionCorrectionTechnique(constraintInfo.positionCorrectionTechnique) { assert(mBody1 != NULL); assert(mBody2 != NULL); diff --git a/src/constraint/Constraint.h b/src/constraint/Constraint.h index ae3918e0..58181768 100644 --- a/src/constraint/Constraint.h +++ b/src/constraint/Constraint.h @@ -27,6 +27,7 @@ #define REACTPHYSICS3D_CONSTRAINT_H // Libraries +#include "../configuration.h" #include "../body/RigidBody.h" #include "../mathematics/mathematics.h" @@ -34,7 +35,7 @@ namespace reactphysics3d { // Enumeration for the type of a constraint -enum ConstraintType {CONTACT, BALLSOCKETJOINT}; +enum ConstraintType {CONTACT, BALLSOCKETJOINT, SLIDERJOINT}; // Class declarations struct ConstraintSolverData; @@ -58,13 +59,19 @@ struct ConstraintInfo { /// Type of the constraint ConstraintType type; + /// Position correction technique used for the constraint (used for joints). + /// By default, the BAUMGARTE technique is used + JointsPositionCorrectionTechnique positionCorrectionTechnique; + /// Constructor ConstraintInfo(ConstraintType constraintType) - : body1(NULL), body2(NULL), type(constraintType) {} + : body1(NULL), body2(NULL), type(constraintType), + positionCorrectionTechnique(BAUMGARTE_JOINTS) {} /// Constructor ConstraintInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, ConstraintType constraintType) - : body1(rigidBody1), body2(rigidBody2), type(constraintType) { + : body1(rigidBody1), body2(rigidBody2), type(constraintType), + positionCorrectionTechnique(BAUMGARTE_JOINTS) { } /// Destructor @@ -103,6 +110,9 @@ class Constraint { /// Body 2 index in the velocity array to solve the constraint uint mIndexBody2; + /// Position correction technique used for the constraint (used for joints) + JointsPositionCorrectionTechnique mPositionCorrectionTechnique; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -139,8 +149,11 @@ class Constraint { /// Initialize before solving the constraint virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData) = 0; - /// Solve the constraint - virtual void solve(const ConstraintSolverData& constraintSolverData) = 0; + /// Solve the velocity constraint + virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) = 0; + + /// Solve the position constraint + virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData) = 0; }; // Return the reference to the body 1 diff --git a/src/constraint/ContactPoint.cpp b/src/constraint/ContactPoint.cpp index b945c8e3..178a5e1d 100644 --- a/src/constraint/ContactPoint.cpp +++ b/src/constraint/ContactPoint.cpp @@ -56,7 +56,12 @@ void ContactPoint::initBeforeSolve(const ConstraintSolverData& constraintSolverD } -// Solve the constraint -void ContactPoint::solve(const ConstraintSolverData& constraintSolverData) { +// Solve the velocity constraint +void ContactPoint::solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) { + +} + +// Solve the position constraint +void ContactPoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { } diff --git a/src/constraint/ContactPoint.h b/src/constraint/ContactPoint.h index 3aff1334..0047c020 100644 --- a/src/constraint/ContactPoint.h +++ b/src/constraint/ContactPoint.h @@ -227,8 +227,11 @@ class ContactPoint : public Constraint { /// Initialize before solving the constraint virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); - /// Solve the constraint - virtual void solve(const ConstraintSolverData& constraintSolverData); + /// Solve the velocity constraint + virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); + + /// Solve the position constraint + virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); #ifdef VISUAL_DEBUG /// Draw the contact (for debugging) diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp new file mode 100644 index 00000000..0d763e9f --- /dev/null +++ b/src/constraint/SliderJoint.cpp @@ -0,0 +1,105 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "SliderJoint.h" + +using namespace reactphysics3d; + +// Constructor +SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) : Constraint(jointInfo) { + + // Compute the local-space anchor point for each body + mLocalAnchorPointBody1 = mBody1->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; + mLocalAnchorPointBody2 = mBody2->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; +} + +// Destructor +SliderJoint::~SliderJoint() { + +} + +// Initialize before solving the constraint +void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverData) { + + // Initialize the bodies index in the velocity array + mIndexBody1 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody1)->second; + mIndexBody2 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody2)->second; + + // Get the bodies positions and orientations + const Quaternion& orientationBody1 = mBody1->getTransform().getOrientation(); + const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); + + // Get the inertia tensor of bodies + const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the vector from body center to the anchor point in world-space + mU1World = orientationBody1 * mLocalAnchorPointBody1; + mU2World = orientationBody2 * mLocalAnchorPointBody2; + + // Compute the two orthogonal vectors to vector mU1World in world-space + mN1 = mU1World.getOneUnitOrthogonalVector(); + mN2 = mU1World.cross(mN1); + + // Compute the cross product used in the Jacobian + mU1WorldCrossN1 = mN2; + mU1WorldCrossN2 = mU1World.cross(mN2); + mU2WorldCrossN1 = mU2World.cross(mN1); + mU2WorldCrossN2 = mU2World.cross(mN2); + + // Compute the mass matrix K=JM^-1J^t for the 2 translation constraints (2x2 matrix) + const decimal n1Dotn1 = mN1.lengthSquare(); + const decimal n2Dotn2 = mN2.lengthSquare(); + const decimal sumInverseMass = mBody1->getMassInverse() + mBody2->getMassInverse(); + +} + +// Solve the velocity constraint +void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) { + + // Get the body positions + const Vector3& x1 = mBody1->getTransform().getPosition(); + const Vector3& x2 = mBody2->getTransform().getPosition(); + + // Get the velocities + Vector3& v1 = constraintSolverData.linearVelocities[mIndexBody1]; + Vector3& v2 = constraintSolverData.linearVelocities[mIndexBody2]; + Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; + Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + decimal inverseMassBody1 = mBody1->getMassInverse(); + decimal inverseMassBody2 = mBody2->getMassInverse(); + Matrix3x3 inverseInertiaTensorBody1 = mBody1->getInertiaTensorInverseWorld(); + Matrix3x3 inverseInertiaTensorBody2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute J*v +} + +// Solve the position constraint +void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { + +} diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h new file mode 100644 index 00000000..2d9ae4ec --- /dev/null +++ b/src/constraint/SliderJoint.h @@ -0,0 +1,131 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_SLIDER_JOINT_H +#define REACTPHYSICS3D_SLIDER_JOINT_H + +// Libraries +#include "../mathematics/mathematics.h" +#include "../engine/ConstraintSolver.h" + +namespace reactphysics3d { + +// Structure SliderJointInfo +/** + * This structure is used to gather the information needed to create a slider + * joint. This structure will be used to create the actual slider joint. + */ +struct SliderJointInfo : public ConstraintInfo { + + public : + + // -------------------- Attributes -------------------- // + + /// Anchor point (in world-space coordinates) + Vector3 anchorPointWorldSpace; + + /// Slider axis (in world-space coordinates) + Vector3 axisWorldSpace; + + /// Constructor + SliderJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, + const Vector3& initAnchorPointWorldSpace, + const Vector3& initAxisWorldSpace) + : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), + anchorPointWorldSpace(initAnchorPointWorldSpace), + axisWorldSpace(initAxisWorldSpace) {} +}; + +// Class SliderJoint +/** + * This class represents a slider joint. + */ +class SliderJoint : public Constraint { + + private : + + // -------------------- Attributes -------------------- // + + /// Anchor point of body 1 (in local space coordinates) + Vector3 mLocalAnchorPointBody1; + + /// Anchor point of body 2 (in local space coordinates) + Vector3 mLocalAnchorPointBody2; + + /// Vector from center of body 2 to anchor point in world-space + Vector3 mU1World; + + /// Vector from center of body 2 to anchor point in world-space + Vector3 mU2World; + + /// First vector orthogonal to vector mU1World in world-space + Vector3 mN1; + + /// Second vector orthogonal to vector mU1World and mN1 in world-space + Vector3 mN2; + + /// Cross product of mU1World and mN1 + Vector3 mU1WorldCrossN1; + + /// Cross product of mU1World and mN2 + Vector3 mU1WorldCrossN2; + + /// Cross product of mU2World and mN1 + Vector3 mU2WorldCrossN1; + + /// Cross product of mU2World and mN2 + Vector3 mU2WorldCrossN2; + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + SliderJoint(const SliderJointInfo& jointInfo); + + /// Destructor + virtual ~SliderJoint(); + + /// Return the number of bytes used by the joint + virtual size_t getSizeInBytes() const; + + /// Initialize before solving the constraint + virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); + + /// Solve the velocity constraint + virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); + + /// Solve the position constraint + virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); +}; + +// Return the number of bytes used by the joint +inline size_t SliderJoint::getSizeInBytes() const { + return sizeof(SliderJoint); +} + +} + +#endif diff --git a/src/engine/ConstraintSolver.cpp b/src/engine/ConstraintSolver.cpp index 53a271af..f825e110 100644 --- a/src/engine/ConstraintSolver.cpp +++ b/src/engine/ConstraintSolver.cpp @@ -37,7 +37,9 @@ ConstraintSolver::ConstraintSolver(std::set& joints, : mJoints(joints), mLinearVelocities(linearVelocities), mAngularVelocities(angularVelocities), mMapBodyToConstrainedVelocityIndex(mapBodyToVelocityIndex), - mIsWarmStartingActive(false), mConstraintSolverData(linearVelocities, + mIsWarmStartingActive(false), + mIsNonLinearGaussSeidelPositionCorrectionActive(false), + mConstraintSolverData(linearVelocities, angularVelocities, mapBodyToVelocityIndex){ } @@ -78,10 +80,10 @@ void ConstraintSolver::initialize(decimal dt) { } } -// Solve the constraints -void ConstraintSolver::solve() { +// Solve the velocity constraints +void ConstraintSolver::solveVelocityConstraints() { - PROFILE("ConstraintSolver::solve()"); + PROFILE("ConstraintSolver::solveVelocityConstraints()"); // For each joint std::set::iterator it; @@ -90,6 +92,22 @@ void ConstraintSolver::solve() { Constraint* joint = (*it); // Solve the constraint - joint->solve(mConstraintSolverData); + joint->solveVelocityConstraint(mConstraintSolverData); + } +} + +// Solve the position constraints +void ConstraintSolver::solvePositionConstraints() { + + PROFILE("ConstraintSolver::solvePositionConstraints()"); + + // For each joint + std::set::iterator it; + for (it = mJoints.begin(); it != mJoints.end(); ++it) { + + Constraint* joint = (*it); + + // Solve the constraint + joint->solveVelocityConstraint(mConstraintSolverData); } } diff --git a/src/engine/ConstraintSolver.h b/src/engine/ConstraintSolver.h index d72f9e98..fd9b3249 100644 --- a/src/engine/ConstraintSolver.h +++ b/src/engine/ConstraintSolver.h @@ -171,6 +171,9 @@ class ConstraintSolver { /// True if the warm starting of the solver is active bool mIsWarmStartingActive; + /// True if the Non-Linear-Gauss-Seidel position correction technique is enabled + bool mIsNonLinearGaussSeidelPositionCorrectionActive; + /// Constraint solver data used to initialize and solve the constraints ConstraintSolverData mConstraintSolverData; @@ -191,9 +194,28 @@ class ConstraintSolver { void initialize(decimal dt); /// Solve the constraints - void solve(); + void solveVelocityConstraints(); + + /// Solve the position constraints + void solvePositionConstraints(); + + /// Return true if the Non-Linear-Gauss-Seidel position correction technique is active + bool getIsNonLinearGaussSeidelPositionCorrectionActive() const; + + /// Enable/Disable the Non-Linear-Gauss-Seidel position correction technique. + void setIsNonLinearGaussSeidelPositionCorrectionActive(bool isActive); }; +// Return true if the Non-Linear-Gauss-Seidel position correction technique is active +inline bool ConstraintSolver::getIsNonLinearGaussSeidelPositionCorrectionActive() const { + return mIsNonLinearGaussSeidelPositionCorrectionActive; +} + +// Enable/Disable the Non-Linear-Gauss-Seidel position correction technique. +inline void ConstraintSolver::setIsNonLinearGaussSeidelPositionCorrectionActive(bool isActive) { + mIsNonLinearGaussSeidelPositionCorrectionActive = isActive; +} + } #endif diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index ed69e4a5..2c046e99 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -26,6 +26,7 @@ // Libraries #include "DynamicsWorld.h" #include "constraint/BallAndSocketJoint.h" +#include "constraint/SliderJoint.h" // Namespaces using namespace reactphysics3d; @@ -38,7 +39,8 @@ DynamicsWorld::DynamicsWorld(const Vector3 &gravity, decimal timeStep = DEFAULT_ mMapBodyToConstrainedVelocityIndex), mConstraintSolver(mJoints, mConstrainedLinearVelocities, mConstrainedAngularVelocities, mMapBodyToConstrainedVelocityIndex), - mNbSolverIterations(DEFAULT_CONSTRAINTS_SOLVER_NB_ITERATIONS), + mNbVelocitySolverIterations(DEFAULT_VELOCITY_SOLVER_NB_ITERATIONS), + mNbPositionSolverIterations(DEFAULT_POSITION_SOLVER_NB_ITERATIONS), mIsDeactivationActive(DEACTIVATION_ENABLED) { } @@ -248,6 +250,8 @@ void DynamicsWorld::solveContactsAndConstraints() { bool isContactsToSolve = !mContactManifolds.empty(); if (!isConstraintsToSolve && !isContactsToSolve) return; + // ---------- Solve velocity constraints for joints and contacts ---------- // + // If there are contacts if (isContactsToSolve) { @@ -265,11 +269,11 @@ void DynamicsWorld::solveContactsAndConstraints() { mConstraintSolver.initialize(dt); } - // For each iteration of the solver - for (uint i=0; i(jointInfo); + newJoint = new (allocatedMemory) SliderJoint(info); + break; + } + default: { assert(false); diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index 0503ebe2..e9c7bcbb 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -59,8 +59,11 @@ class DynamicsWorld : public CollisionWorld { /// Constraint solver ConstraintSolver mConstraintSolver; - /// Number of solver iterations for the Sequential Impulses technique - uint mNbSolverIterations; + /// Number of iterations for the velocity solver of the Sequential Impulses technique + uint mNbVelocitySolverIterations; + + /// Number of iterations for the position solver of the Sequential Impulses technique + uint mNbPositionSolverIterations; /// True if the deactivation (sleeping) of inactive bodies is enabled bool mIsDeactivationActive; @@ -155,19 +158,22 @@ public : /// Update the physics simulation void update(); - /// Set the number of iterations of the constraint solver - void setNbIterationsSolver(uint nbIterations); + /// Set the number of iterations for the velocity constraint solver + void setNbIterationsVelocitySolver(uint nbIterations); - /// Activate or Deactivate the split impulses for contacts - void setIsSplitImpulseActive(bool isActive); + /// Set the number of iterations for the position constraint solver + void setNbIterationsPositionSolver(uint nbIterations); + + /// Set the position correction technique used for contacts + void setContactsPositionCorrectionTechnique(ContactsPositionCorrectionTechnique technique); + + /// Set the position correction technique used for joints + void setJointsPositionCorrectionTechnique(JointsPositionCorrectionTechnique technique); /// Activate or deactivate the solving of friction constraints at the center of /// the contact manifold instead of solving them at each contact point void setIsSolveFrictionAtContactManifoldCenterActive(bool isActive); - /// Set the isErrorCorrectionActive value - void setIsErrorCorrectionActive(bool isErrorCorrectionActive); - /// Create a rigid body into the physics world. RigidBody* createRigidBody(const Transform& transform, decimal mass, const Matrix3x3& inertiaTensorLocal, @@ -213,14 +219,36 @@ inline void DynamicsWorld::stop() { mTimer.stop(); } -// Set the number of iterations of the constraint solver -inline void DynamicsWorld::setNbIterationsSolver(uint nbIterations) { - mNbSolverIterations = nbIterations; +// Set the number of iterations for the velocity constraint solver +inline void DynamicsWorld::setNbIterationsVelocitySolver(uint nbIterations) { + mNbVelocitySolverIterations = nbIterations; } -// Activate or Deactivate the split impulses for contacts -inline void DynamicsWorld::setIsSplitImpulseActive(bool isActive) { - mContactSolver.setIsSplitImpulseActive(isActive); +// Set the number of iterations for the position constraint solver +inline void DynamicsWorld::setNbIterationsPositionSolver(uint nbIterations) { + mNbPositionSolverIterations = nbIterations; +} + +// Set the position correction technique used for contacts +inline void DynamicsWorld::setContactsPositionCorrectionTechnique( + ContactsPositionCorrectionTechnique technique) { + if (technique == BAUMGARTE_CONTACTS) { + mContactSolver.setIsSplitImpulseActive(false); + } + else { + mContactSolver.setIsSplitImpulseActive(true); + } +} + +// Set the position correction technique used for joints +inline void DynamicsWorld::setJointsPositionCorrectionTechnique( + JointsPositionCorrectionTechnique technique) { + if (technique == BAUMGARTE_JOINTS) { + mConstraintSolver.setIsNonLinearGaussSeidelPositionCorrectionActive(false); + } + else { + mConstraintSolver.setIsNonLinearGaussSeidelPositionCorrectionActive(true); + } } // Activate or deactivate the solving of friction constraints at the center of From 7a2c2bdbd57db1de5e3db876b2675fff41d85980 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 9 May 2013 19:02:09 +0200 Subject: [PATCH 22/66] Add Vector2 and Matrix2x3 classes --- src/mathematics/Matrix2x2.cpp | 89 +++++++ src/mathematics/Matrix2x2.h | 312 ++++++++++++++++++++++++ src/mathematics/Vector2.cpp | 71 ++++++ src/mathematics/Vector2.h | 296 ++++++++++++++++++++++ src/mathematics/Vector3.h | 3 - test/tests/mathematics/TestMatrix2x2.h | 239 ++++++++++++++++++ test/tests/mathematics/TestMatrix3x3.h | 7 +- test/tests/mathematics/TestQuaternion.h | 6 +- test/tests/mathematics/TestTransform.h | 6 +- test/tests/mathematics/TestVector2.h | 208 ++++++++++++++++ test/tests/mathematics/TestVector3.h | 6 +- 11 files changed, 1223 insertions(+), 20 deletions(-) create mode 100644 src/mathematics/Matrix2x2.cpp create mode 100644 src/mathematics/Matrix2x2.h create mode 100644 src/mathematics/Vector2.cpp create mode 100644 src/mathematics/Vector2.h create mode 100644 test/tests/mathematics/TestMatrix2x2.h create mode 100644 test/tests/mathematics/TestVector2.h diff --git a/src/mathematics/Matrix2x2.cpp b/src/mathematics/Matrix2x2.cpp new file mode 100644 index 00000000..87353a3d --- /dev/null +++ b/src/mathematics/Matrix2x2.cpp @@ -0,0 +1,89 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Matrix2x2.h" + +using namespace reactphysics3d; + +// Constructor of the class Matrix2x2 +Matrix2x2::Matrix2x2() { + + // Initialize all values in the matrix to zero + setAllValues(0.0, 0.0, 0.0, 0.0); +} + +// Constructor +Matrix2x2::Matrix2x2(decimal value) { + setAllValues(value, value, value, value); +} + +// Constructor with arguments +Matrix2x2::Matrix2x2(decimal a1, decimal a2, decimal b1, decimal b2) { + + // Initialize the matrix with the values + setAllValues(a1, a2, b1, b2); +} + +// Destructor +Matrix2x2::~Matrix2x2() { + +} + +// Copy-constructor +Matrix2x2::Matrix2x2(const Matrix2x2& matrix) { + setAllValues(matrix.mRows[0][0], matrix.mRows[0][1], + matrix.mRows[1][0], matrix.mRows[1][1]); +} + +// Assignment operator +Matrix2x2& Matrix2x2::operator=(const Matrix2x2& matrix) { + + // Check for self-assignment + if (&matrix != this) { + setAllValues(matrix.mRows[0][0], matrix.mRows[0][1], + matrix.mRows[1][0], matrix.mRows[1][1]); + } + return *this; +} + +// Return the inverse matrix +Matrix2x2 Matrix2x2::getInverse() const { + + // Compute the determinant of the matrix + decimal determinant = getDeterminant(); + + // Check if the determinant is equal to zero + assert(std::abs(determinant) > MACHINE_EPSILON); + + decimal invDeterminant = decimal(1.0) / determinant; + + // TODO : Implement this + assert(false); + Matrix2x2 tempMatrix; + + // Return the inverse matrix + return (invDeterminant * tempMatrix); +} diff --git a/src/mathematics/Matrix2x2.h b/src/mathematics/Matrix2x2.h new file mode 100644 index 00000000..ec9c28f1 --- /dev/null +++ b/src/mathematics/Matrix2x2.h @@ -0,0 +1,312 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_MATRIX2X2_H +#define REACTPHYSICS3D_MATRIX2X2_H + +// Libraries +#include +#include "Vector2.h" + +/// ReactPhysics3D namespace +namespace reactphysics3d { + +// Class Matrix2x2 +/** + * This class represents a 2x2 matrix. + */ +class Matrix2x2 { + + private : + + // -------------------- Attributes -------------------- // + + /// Rows of the matrix; + Vector2 mRows[3]; + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + Matrix2x2(); + + /// Constructor + Matrix2x2(decimal value); + + /// Constructor + Matrix2x2(decimal a1, decimal a2, decimal b1, decimal b2); + + /// Destructor + virtual ~Matrix2x2(); + + /// Copy-constructor + Matrix2x2(const Matrix2x2& matrix); + + /// Assignment operator + Matrix2x2& operator=(const Matrix2x2& matrix); + + /// Set all the values in the matrix + void setAllValues(decimal a1, decimal a2, decimal b1, decimal b2); + + /// Set the matrix to zero + void setToZero(); + + /// Return a column + Vector2 getColumn(int i) const; + + /// Return a row + Vector2 getRow(int i) const; + + /// Return the transpose matrix + Matrix2x2 getTranspose() const; + + /// Return the determinant of the matrix + decimal getDeterminant() const; + + /// Return the trace of the matrix + decimal getTrace() const; + + /// Return the inverse matrix + Matrix2x2 getInverse() const; + + /// Return the matrix with absolute values + Matrix2x2 getAbsoluteMatrix() const; + + /// Set the matrix to the identity matrix + void setToIdentity(); + + /// Return the 2x2 identity matrix + static Matrix2x2 identity(); + + /// Overloaded operator for addition + friend Matrix2x2 operator+(const Matrix2x2& matrix1, const Matrix2x2& matrix2); + + /// Overloaded operator for substraction + friend Matrix2x2 operator-(const Matrix2x2& matrix1, const Matrix2x2& matrix2); + + /// Overloaded operator for the negative of the matrix + friend Matrix2x2 operator-(const Matrix2x2& matrix); + + /// Overloaded operator for multiplication with a number + friend Matrix2x2 operator*(decimal nb, const Matrix2x2& matrix); + + /// Overloaded operator for multiplication with a matrix + friend Matrix2x2 operator*(const Matrix2x2& matrix, decimal nb); + + /// Overloaded operator for matrix multiplication + friend Matrix2x2 operator*(const Matrix2x2& matrix1, const Matrix2x2& matrix2); + + /// Overloaded operator for multiplication with a vector + friend Vector2 operator*(const Matrix2x2& matrix, const Vector2& vector); + + /// Overloaded operator for equality condition + bool operator==(const Matrix2x2& matrix) const; + + /// Overloaded operator for the is different condition + bool operator!= (const Matrix2x2& matrix) const; + + /// Overloaded operator for addition with assignment + Matrix2x2& operator+=(const Matrix2x2& matrix); + + /// Overloaded operator for substraction with assignment + Matrix2x2& operator-=(const Matrix2x2& matrix); + + /// Overloaded operator for multiplication with a number with assignment + Matrix2x2& operator*=(decimal nb); + + /// Overloaded operator to read element of the matrix. + const Vector2& operator[](int row) const; + + /// Overloaded operator to read/write element of the matrix. + Vector2& operator[](int row); +}; + +// Method to set all the values in the matrix +inline void Matrix2x2::setAllValues(decimal a1, decimal a2, + decimal b1, decimal b2) { + mRows[0][0] = a1; mRows[0][1] = a2; + mRows[1][0] = b1; mRows[1][1] = b2; +} + +// Set the matrix to zero +inline void Matrix2x2::setToZero() { + mRows[0].setToZero(); + mRows[1].setToZero(); +} + +// Return a column +inline Vector2 Matrix2x2::getColumn(int i) const { + assert(i>= 0 && i<2); + return Vector2(mRows[0][i], mRows[1][i]); +} + +// Return a row +inline Vector2 Matrix2x2::getRow(int i) const { + assert(i>= 0 && i<2); + return mRows[i]; +} + +// Return the transpose matrix +inline Matrix2x2 Matrix2x2::getTranspose() const { + + // Return the transpose matrix + return Matrix2x2(mRows[0][0], mRows[1][0], + mRows[0][1], mRows[1][1]); +} + +// Return the determinant of the matrix +inline decimal Matrix2x2::getDeterminant() const { + + // Compute and return the determinant of the matrix + return mRows[0][0] * mRows[1][1] - mRows[1][0] * mRows[0][1]; +} + +// Return the trace of the matrix +inline decimal Matrix2x2::getTrace() const { + + // Compute and return the trace + return (mRows[0][0] + mRows[1][1]); +} + +// Set the matrix to the identity matrix +inline void Matrix2x2::setToIdentity() { + mRows[0][0] = 1.0; mRows[0][1] = 0.0; + mRows[1][0] = 0.0; mRows[1][1] = 1.0; +} + +// Return the 2x2 identity matrix +inline Matrix2x2 Matrix2x2::identity() { + + // Return the isdentity matrix + return Matrix2x2(1.0, 0.0, 0.0, 1.0); +} + +// Return the matrix with absolute values +inline Matrix2x2 Matrix2x2::getAbsoluteMatrix() const { + return Matrix2x2(fabs(mRows[0][0]), fabs(mRows[0][1]), + fabs(mRows[1][0]), fabs(mRows[1][1])); +} + +// Overloaded operator for addition +inline Matrix2x2 operator+(const Matrix2x2& matrix1, const Matrix2x2& matrix2) { + return Matrix2x2(matrix1.mRows[0][0] + matrix2.mRows[0][0], + matrix1.mRows[0][1] + matrix2.mRows[0][1], + matrix1.mRows[1][0] + matrix2.mRows[1][0], + matrix1.mRows[1][1] + matrix2.mRows[1][1]); +} + +// Overloaded operator for substraction +inline Matrix2x2 operator-(const Matrix2x2& matrix1, const Matrix2x2& matrix2) { + return Matrix2x2(matrix1.mRows[0][0] - matrix2.mRows[0][0], + matrix1.mRows[0][1] - matrix2.mRows[0][1], + matrix1.mRows[1][0] - matrix2.mRows[1][0], + matrix1.mRows[1][1] - matrix2.mRows[1][1]); +} + +// Overloaded operator for the negative of the matrix +inline Matrix2x2 operator-(const Matrix2x2& matrix) { + return Matrix2x2(-matrix.mRows[0][0], -matrix.mRows[0][1], + -matrix.mRows[1][0], -matrix.mRows[1][1]); +} + +// Overloaded operator for multiplication with a number +inline Matrix2x2 operator*(decimal nb, const Matrix2x2& matrix) { + return Matrix2x2(matrix.mRows[0][0] * nb, matrix.mRows[0][1] * nb, + matrix.mRows[1][0] * nb, matrix.mRows[1][1] * nb); +} + +// Overloaded operator for multiplication with a matrix +inline Matrix2x2 operator*(const Matrix2x2& matrix, decimal nb) { + return nb * matrix; +} + +// Overloaded operator for matrix multiplication +inline Matrix2x2 operator*(const Matrix2x2& matrix1, const Matrix2x2& matrix2) { + return Matrix2x2(matrix1.mRows[0][0] * matrix2.mRows[0][0] + matrix1.mRows[0][1] * + matrix2.mRows[1][0], + matrix1.mRows[0][0] * matrix2.mRows[0][1] + matrix1.mRows[0][1] * + matrix2.mRows[1][1], + matrix1.mRows[1][0] * matrix2.mRows[0][0] + matrix1.mRows[1][1] * + matrix2.mRows[1][0], + matrix1.mRows[1][0] * matrix2.mRows[0][1] + matrix1.mRows[1][1] * + matrix2.mRows[1][1]); +} + +// Overloaded operator for multiplication with a vector +inline Vector2 operator*(const Matrix2x2& matrix, const Vector2& vector) { + return Vector2(matrix.mRows[0][0]*vector.x + matrix.mRows[0][1]*vector.y, + matrix.mRows[1][0]*vector.x + matrix.mRows[1][1]*vector.y); +} + +// Overloaded operator for equality condition +inline bool Matrix2x2::operator==(const Matrix2x2& matrix) const { + return (mRows[0][0] == matrix.mRows[0][0] && mRows[0][1] == matrix.mRows[0][1] && + mRows[1][0] == matrix.mRows[1][0] && mRows[1][1] == matrix.mRows[1][1]); +} + +// Overloaded operator for the is different condition +inline bool Matrix2x2::operator!= (const Matrix2x2& matrix) const { + return !(*this == matrix); +} + +// Overloaded operator for addition with assignment +inline Matrix2x2& Matrix2x2::operator+=(const Matrix2x2& matrix) { + mRows[0][0] += matrix.mRows[0][0]; mRows[0][1] += matrix.mRows[0][1]; + mRows[1][0] += matrix.mRows[1][0]; mRows[1][1] += matrix.mRows[1][1]; + return *this; +} + +// Overloaded operator for substraction with assignment +inline Matrix2x2& Matrix2x2::operator-=(const Matrix2x2& matrix) { + mRows[0][0] -= matrix.mRows[0][0]; mRows[0][1] -= matrix.mRows[0][1]; + mRows[1][0] -= matrix.mRows[1][0]; mRows[1][1] -= matrix.mRows[1][1]; + return *this; +} + +// Overloaded operator for multiplication with a number with assignment +inline Matrix2x2& Matrix2x2::operator*=(decimal nb) { + mRows[0][0] *= nb; mRows[0][1] *= nb; + mRows[1][0] *= nb; mRows[1][1] *= nb; + return *this; +} + +// Overloaded operator to return a row of the matrix. +/// This operator is also used to access a matrix value using the syntax +/// matrix[row][col]. +inline const Vector2& Matrix2x2::operator[](int row) const { + return mRows[row]; +} + +// Overloaded operator to return a row of the matrix. +/// This operator is also used to access a matrix value using the syntax +/// matrix[row][col]. +inline Vector2& Matrix2x2::operator[](int row) { + return mRows[row]; +} + +} + +#endif diff --git a/src/mathematics/Vector2.cpp b/src/mathematics/Vector2.cpp new file mode 100644 index 00000000..c7a2c9f1 --- /dev/null +++ b/src/mathematics/Vector2.cpp @@ -0,0 +1,71 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Vector2.h" +#include + +// Namespaces +using namespace reactphysics3d; + +// Constructor +Vector2::Vector2() : x(0.0), y(0.0) { + +} + +// Constructor with arguments +Vector2::Vector2(decimal newX, decimal newY) : x(newX), y(newY) { + +} + +// Copy-constructor +Vector2::Vector2(const Vector2& vector) : x(vector.x), y(vector.y) { + +} + +// Destructor +Vector2::~Vector2() { + +} + +// Return the corresponding unit vector +Vector2 Vector2::getUnit() const { + decimal lengthVector = length(); + + assert(lengthVector > MACHINE_EPSILON); + + // Compute and return the unit vector + decimal lengthInv = decimal(1.0) / lengthVector; + return Vector2(x * lengthInv, y * lengthInv); +} + +// Return one unit orthogonal vector of the current vector +Vector2 Vector2::getOneUnitOrthogonalVector() const { + + decimal l = length(); + assert(l > MACHINE_EPSILON); + + return Vector2(-y / l, x / l); +} diff --git a/src/mathematics/Vector2.h b/src/mathematics/Vector2.h new file mode 100644 index 00000000..6f1f518e --- /dev/null +++ b/src/mathematics/Vector2.h @@ -0,0 +1,296 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_VECTOR2_H +#define REACTPHYSICS3D_VECTOR2_H + +// Libraries +#include +#include +#include "mathematics_functions.h" +#include "../decimal.h" + + +/// ReactPhysics3D namespace +namespace reactphysics3d { + +// Class Vector2 +/** + * This class represents a 2D vector. + */ +struct Vector2 { + + public: + + // -------------------- Attributes -------------------- // + + /// Component x + decimal x; + + /// Component y + decimal y; + + // -------------------- Methods -------------------- // + + /// Constructor of the class Vector3D + Vector2(); + + /// Constructor with arguments + Vector2(decimal newX, decimal newY); + + /// Copy-constructor + Vector2(const Vector2& vector); + + /// Destructor + ~Vector2(); + + /// Set all the values of the vector + void setAllValues(decimal newX, decimal newY); + + /// Set the vector to zero + void setToZero(); + + /// Return the length of the vector + decimal length() const; + + /// Return the square of the length of the vector + decimal lengthSquare() const; + + /// Return the corresponding unit vector + Vector2 getUnit() const; + + /// Return one unit orthogonal vector of the current vector + Vector2 getOneUnitOrthogonalVector() const; + + /// Return true if the vector is unit and false otherwise + bool isUnit() const; + + /// Return true if the current vector is the zero vector + bool isZero() const; + + /// Dot product of two vectors + decimal dot(const Vector2& vector) const; + + /// Normalize the vector + void normalize(); + + /// Return the corresponding absolute value vector + Vector2 getAbsoluteVector() const; + + /// Return the axis with the minimal value + int getMinAxis() const; + + /// Return the axis with the maximal value + int getMaxAxis() const; + + /// Overloaded operator for the equality condition + bool operator== (const Vector2& vector) const; + + /// Overloaded operator for the is different condition + bool operator!= (const Vector2& vector) const; + + /// Overloaded operator for addition with assignment + Vector2& operator+=(const Vector2& vector); + + /// Overloaded operator for substraction with assignment + Vector2& operator-=(const Vector2& vector); + + /// Overloaded operator for multiplication with a number with assignment + Vector2& operator*=(decimal number); + + /// Overloaded operator for division by a number with assignment + Vector2& operator/=(decimal number); + + /// Overloaded operator for value access + decimal& operator[] (int index); + + /// Overloaded operator for value access + const decimal& operator[] (int index) const; + + /// Overloaded operator + Vector2& operator=(const Vector2& vector); + + // -------------------- Friends -------------------- // + + friend Vector2 operator+(const Vector2& vector1, const Vector2& vector2); + friend Vector2 operator-(const Vector2& vector1, const Vector2& vector2); + friend Vector2 operator-(const Vector2& vector); + friend Vector2 operator*(const Vector2& vector, decimal number); + friend Vector2 operator*(decimal number, const Vector2& vector); + friend Vector2 operator/(const Vector2& vector, decimal number); +}; + +// Set the vector to zero +inline void Vector2::setToZero() { + x = 0; + y = 0; +} + +// Set all the values of the vector +inline void Vector2::setAllValues(decimal newX, decimal newY) { + x = newX; + y = newY; +} + +// Return the length of the vector +inline decimal Vector2::length() const { + return sqrt(x*x + y*y); +} + +// Return the square of the length of the vector +inline decimal Vector2::lengthSquare() const { + return x*x + y*y; +} + +// Scalar product of two vectors (inline) +inline decimal Vector2::dot(const Vector2& vector) const { + return (x*vector.x + y*vector.y); +} + +// Normalize the vector +inline void Vector2::normalize() { + decimal l = length(); + assert(l > std::numeric_limits::epsilon()); + x /= l; + y /= l; +} + +// Return the corresponding absolute value vector +inline Vector2 Vector2::getAbsoluteVector() const { + return Vector2(std::abs(x), std::abs(y)); +} + +// Return the axis with the minimal value +inline int Vector2::getMinAxis() const { + return (x < y ? 0 : 1); +} + +// Return the axis with the maximal value +inline int Vector2::getMaxAxis() const { + return (x < y ? 1 : 0); +} + +// Return true if the vector is unit and false otherwise +inline bool Vector2::isUnit() const { + return approxEqual(lengthSquare(), 1.0); +} + +// Return true if the vector is the zero vector +inline bool Vector2::isZero() const { + return approxEqual(lengthSquare(), 0.0); +} + +// Overloaded operator for the equality condition +inline bool Vector2::operator== (const Vector2& vector) const { + return (x == vector.x && y == vector.y); +} + +// Overloaded operator for the is different condition +inline bool Vector2::operator!= (const Vector2& vector) const { + return !(*this == vector); +} + +// Overloaded operator for addition with assignment +inline Vector2& Vector2::operator+=(const Vector2& vector) { + x += vector.x; + y += vector.y; + return *this; +} + +// Overloaded operator for substraction with assignment +inline Vector2& Vector2::operator-=(const Vector2& vector) { + x -= vector.x; + y -= vector.y; + return *this; +} + +// Overloaded operator for multiplication with a number with assignment +inline Vector2& Vector2::operator*=(decimal number) { + x *= number; + y *= number; + return *this; +} + +// Overloaded operator for division by a number with assignment +inline Vector2& Vector2::operator/=(decimal number) { + assert(number > std::numeric_limits::epsilon()); + x /= number; + y /= number; + return *this; +} + +// Overloaded operator for value access +inline decimal& Vector2::operator[] (int index) { + return (&x)[index]; +} + +// Overloaded operator for value access +inline const decimal& Vector2::operator[] (int index) const { + return (&x)[index]; +} + +// Overloaded operator for addition +inline Vector2 operator+(const Vector2& vector1, const Vector2& vector2) { + return Vector2(vector1.x + vector2.x, vector1.y + vector2.y); +} + +// Overloaded operator for substraction +inline Vector2 operator-(const Vector2& vector1, const Vector2& vector2) { + return Vector2(vector1.x - vector2.x, vector1.y - vector2.y); +} + +// Overloaded operator for the negative of a vector +inline Vector2 operator-(const Vector2& vector) { + return Vector2(-vector.x, -vector.y); +} + +// Overloaded operator for multiplication with a number +inline Vector2 operator*(const Vector2& vector, decimal number) { + return Vector2(number * vector.x, number * vector.y); +} + +// Overloaded operator for division by a number +inline Vector2 operator/(const Vector2& vector, decimal number) { + assert(number > MACHINE_EPSILON); + return Vector2(vector.x / number, vector.y / number); +} + +// Overloaded operator for multiplication with a number +inline Vector2 operator*(decimal number, const Vector2& vector) { + return vector * number; +} + +// Assignment operator +inline Vector2& Vector2::operator=(const Vector2& vector) { + if (&vector != this) { + x = vector.x; + y = vector.y; + } + return *this; +} + +} + +#endif diff --git a/src/mathematics/Vector3.h b/src/mathematics/Vector3.h index 3a54b06a..24385fb1 100644 --- a/src/mathematics/Vector3.h +++ b/src/mathematics/Vector3.h @@ -111,9 +111,6 @@ struct Vector3 { /// Return the axis with the maximal value int getMaxAxis() const; - /// Return true if two vectors are parallel - bool isParallelWith(const Vector3& vector) const; - /// Overloaded operator for the equality condition bool operator== (const Vector3& vector) const; diff --git a/test/tests/mathematics/TestMatrix2x2.h b/test/tests/mathematics/TestMatrix2x2.h new file mode 100644 index 00000000..082950a8 --- /dev/null +++ b/test/tests/mathematics/TestMatrix2x2.h @@ -0,0 +1,239 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 TEST_MATRIX2X2_H +#define TEST_MATRIX2X2_H + +// Libraries +#include "../../Test.h" +#include "../../../src/mathematics/Matrix2x2.h" + +/// Reactphysics3D namespace +namespace reactphysics3d { + +// Class TestMatrix2x2 +/** + * Unit test for the Matrix2x2 class + */ +class TestMatrix2x2 : public Test { + + private : + + // ---------- Atributes ---------- // + + /// Identity transform + Matrix2x2 mIdentity; + + /// First example matrix + Matrix2x2 mMatrix1; + + public : + + // ---------- Methods ---------- // + + /// Constructor + TestMatrix2x2() : mIdentity(Matrix2x2::identity()), + mMatrix1(2, 24, -4, 5) { + + + } + + /// Run the tests + void run() { + testConstructors(); + testGetSet(); + testIdentity(); + testOthersMethods(); + testOperators(); + } + + /// Test the constructors + void testConstructors() { + + Matrix2x2 test1(5.0); + Matrix2x2 test2(2, 3, 4, 5); + Matrix2x2 test3(mMatrix1); + + test(test1[0][0] == 5); + test(test1[0][1] == 5); + test(test1[1][0] == 5); + test(test1[1][1] == 5); + + test(test2[0][0] == 2); + test(test2[0][1] == 3); + test(test2[1][0] == 4); + test(test2[1][1] == 5); + + test(test3 == mMatrix1); + } + + /// Test the getter and setter methods + void testGetSet() { + + // Test method to set all the values + Matrix2x2 test2; + test2.setAllValues(2, 24, -4, 5); + test(test2 == mMatrix1); + + // Test method to set to zero + test2.setToZero(); + test(test2 == Matrix2x2(0, 0, 0, 0)); + + // Test method that returns a column + Vector2 column1 = mMatrix1.getColumn(0); + Vector2 column2 = mMatrix1.getColumn(1); + test(column1 == Vector2(2, -4)); + test(column2 == Vector2(24, 5)); + + // Test method that returns a row + Vector3 row1 = mMatrix1.getRow(0); + Vector3 row2 = mMatrix1.getRow(1); + test(row1 == Vector3(2, 24)); + test(row2 == Vector3(-4, 5)); + } + + /// Test the identity methods + void testIdentity() { + + Matrix2x2 identity = Matrix2x2::identity(); + Matrix2x2 test1; + test1.setToIdentity(); + + test(identity[0][0] == 1); + test(identity[0][1] == 0); + test(identity[1][0] == 0); + test(identity[1][1] == 1); + + test(test1 == Matrix2x2::identity()); + } + + /// Test others methods + void testOthersMethods() { + + // Test transpose + Matrix2x2 transpose = mMatrix1.getTranspose(); + test(transpose == Matrix2x2(2, -4, 24, 5)); + + // Test trace + test(mMatrix1.getTrace() == 10); + test(Matrix2x2::identity().getTrace() == 2); + + // Test determinant + Matrix2x2 matrix(-24, 64, 253, -35); + test(mMatrix1.getDeterminant() == 106); + test(matrix.getDeterminant() == -15352); + test(mIdentity.getDeterminant() == 1); + + // Test inverse + Matrix2x2 inverseMatrix = matrix.getInverse(); + test(approxEqual(inverseMatrix[0][0], decimal(0.056369), decimal(10e-6))); + test(approxEqual(inverseMatrix[0][1], decimal(-0.049549), decimal(10e-6))); + test(approxEqual(inverseMatrix[1][0], decimal(0.029460), decimal(10e-6))); + test(approxEqual(inverseMatrix[1][1], decimal(0.038575), decimal(10e-6))); + Matrix2x2 inverseMatrix1 = mMatrix1.getInverse(); + test(approxEqual(inverseMatrix1[0][0], decimal(0.030232), decimal(10e-6))); + test(approxEqual(inverseMatrix1[0][1], decimal(0.015676), decimal(10e-6))); + test(approxEqual(inverseMatrix1[1][0], decimal(-0.057410), decimal(10e-6))); + test(approxEqual(inverseMatrix1[1][1], decimal(0.039088), decimal(10e-6))); + + // Test absolute matrix + Matrix2x2 matrix2(-2, -3, -4, -5); + test(matrix.getAbsoluteMatrix() == Matrix2x2(24, 64, 253, 35)); + Matrix2x2 absoluteMatrix = matrix2.getAbsoluteMatrix(); + test(absoluteMatrix == Matrix2x2(2, 3, 4, 5)); + } + + /// Test the operators + void testOperators() { + + // Test addition + Matrix2x2 matrix1(2, 3, 4, 5); + Matrix2x2 matrix2(-2, 3, -5, 10); + Matrix2x2 addition1 = matrix1 + matrix2; + Matrix2x2 addition2(matrix1); + addition2 += matrix2; + test(addition1 == Matrix2x2(0, 6, -1, 15)); + test(addition2 == Matrix2x2(0, 6, -1, 15)); + + // Test substraction + Matrix2x2 substraction1 = matrix1 - matrix2; + Matrix2x2 substraction2(matrix1); + substraction2 -= matrix2; + test(substraction1 == Matrix2x2(4, 0, 9, -5)); + test(substraction2 == Matrix2x2(4, 0, 9, -5)); + + // Test negative operator + Matrix2x2 negative = -matrix1; + test(negative == Matrix2x2(-2, -3, -4, -5)); + + // Test multiplication with a number + Matrix2x2 multiplication1 = 3 * matrix1; + Matrix2x2 multiplication2 = matrix1 * 3; + Matrix2x2 multiplication3(matrix1); + multiplication3 *= 3; + test(multiplication1 == Matrix2x2(6, 9, 12, 15)); + test(multiplication2 == Matrix2x2(6, 9, 12, 15)); + test(multiplication3 == Matrix2x2(6, 9, 12, 15)); + + // Test multiplication with a matrix + Matrix2x2 multiplication4 = matrix1 * matrix2; + Matrix2x2 multiplication5 = matrix2 * matrix1; + test(multiplication4 == Matrix2x2(-19, 36, -33, 62)); + test(multiplication5 == Matrix2x2(8, 9, 30, 35)); + + // Test multiplication with a vector + Vector2 vector1(3, -32); + Vector2 vector2(-31, -422); + Vector2 test1 = matrix1 * vector1; + Vector2 test2 = matrix2 * vector2; + test(test1 == Vector2(-762, -182)); + test(test2 == Vector2(-10190, -1986)); + + // Test equality operators + test(Matrix2x2(34, 38, 43, 64) == + Matrix2x2(34, 38, 43, 64)); + test(Matrix2x2(34, 64, 43, 7) != + Matrix2x2(34, 38, 43, 64)); + + // Test operator to read a value + test(mMatrix1[0][0] == 2); + test(mMatrix1[0][1] == 24); + test(mMatrix1[1][0] == -4); + test(mMatrix1[1][1] == 5); + + // Test operator to set a value + Matrix2x2 test3; + test3[0][0] = 2; + test3[0][1] = 24; + test3[1][0] = -4; + test3[1][1] = 5; + test(test3 == mMatrix1); + } + + }; + +} + +#endif diff --git a/test/tests/mathematics/TestMatrix3x3.h b/test/tests/mathematics/TestMatrix3x3.h index ca0a2bb2..78ae43aa 100644 --- a/test/tests/mathematics/TestMatrix3x3.h +++ b/test/tests/mathematics/TestMatrix3x3.h @@ -1,4 +1,3 @@ - /******************************************************************************** * ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * * Copyright (c) 2010-2013 Daniel Chappuis * @@ -27,14 +26,10 @@ #ifndef TEST_MATRIX3X3_H #define TEST_MATRIX3X3_H -#endif - // Libraries #include "../../Test.h" #include "../../../src/mathematics/Matrix3x3.h" -using namespace reactphysics3d; - /// Reactphysics3D namespace namespace reactphysics3d { @@ -287,3 +282,5 @@ class TestMatrix3x3 : public Test { }; } + +#endif diff --git a/test/tests/mathematics/TestQuaternion.h b/test/tests/mathematics/TestQuaternion.h index 9007f9b4..ac01cb6c 100644 --- a/test/tests/mathematics/TestQuaternion.h +++ b/test/tests/mathematics/TestQuaternion.h @@ -27,14 +27,10 @@ #ifndef TEST_QUATERNION_H #define TEST_QUATERNION_H -#endif - // Libraries #include "../../Test.h" #include "../../../src/mathematics/Quaternion.h" -using namespace reactphysics3d; - /// Reactphysics3D namespace namespace reactphysics3d { @@ -229,3 +225,5 @@ class TestQuaternion : public Test { }; } + +#endif diff --git a/test/tests/mathematics/TestTransform.h b/test/tests/mathematics/TestTransform.h index 72804053..2d24aaa8 100644 --- a/test/tests/mathematics/TestTransform.h +++ b/test/tests/mathematics/TestTransform.h @@ -27,14 +27,10 @@ #ifndef TEST_TRANSFORM_H #define TEST_TRANSFORM_H -#endif - // Libraries #include "../../Test.h" #include "../../../src/mathematics/Transform.h" -using namespace reactphysics3d; - /// Reactphysics3D namespace namespace reactphysics3d { @@ -216,3 +212,5 @@ class TestTransform : public Test { }; } + +#endif diff --git a/test/tests/mathematics/TestVector2.h b/test/tests/mathematics/TestVector2.h new file mode 100644 index 00000000..0cb1f812 --- /dev/null +++ b/test/tests/mathematics/TestVector2.h @@ -0,0 +1,208 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 TEST_VECTOR2_H +#define TEST_VECTOR2_H + +// Libraries +#include "../../Test.h" +#include "../../../src/mathematics/Vector2.h" + +/// Reactphysics3D namespace +namespace reactphysics3d { + +// Class TestVector2 +/** + * Unit test for the Vector2 class + */ +class TestVector2 : public Test { + + private : + + // ---------- Atributes ---------- // + + /// Zero vector + Vector2 mVectorZero; + + // Vector (3, 4) + Vector2 mVector34; + + public : + + // ---------- Methods ---------- // + + /// Constructor + TestVector1() : mVectorZero(0, 0), mVector34(3, 4) {} + + /// Run the tests + void run() { + testConstructors(); + testLengthMethods(); + testDotProduct(); + testOthersMethods(); + testOperators(); + } + + /// Test the constructors, getter and setter + void testConstructors() { + + // Test constructor + test(mVectorZero.x == 0.0); + test(mVectorZero.y == 0.0); + test(mVector34.x == 3.0); + test(mVector34.y == 4.0); + + // Test copy-constructor + Vector2 newVector(mVector34); + test(newVector.x == 3.0); + test(newVector.y == 4.0); + + // Test method to set values + Vector2 newVector2; + newVector2.setAllValues(decimal(6.1), decimal(7.2)); + test(approxEqual(newVector2.x, decimal(6.1))); + test(approxEqual(newVector2.y, decimal(7.2))); + + // Test method to set to zero + newVector2.setToZero(); + test(newVector2 == Vector2(0, 0)); + } + + /// Test the length, unit vector and normalize methods + void testLengthMethods() { + + // Test length methods + test(mVectorZero.length() == 0.0); + test(mVectorZero.lengthSquare() == 0.0); + test(Vector2(1, 0).length() == 1.0); + test(Vector2(0, 1).length() == 1.0); + test(mVector345.lengthSquare() == 50.0); + + // Test unit vector methods + test(Vector2(1, 0).isUnit()); + test(Vector2(0, 1).isUnit()); + test(!mVector34.isUnit()); + test(Vector2(5, 0).getUnit() == Vector3(1, 0)); + test(Vector2(0, 5).getUnit() == Vector3(0, 1)); + + test(!mVector34.isZero()); + test(mVectorZero.isZero()); + + // Test normalization method + Vector2 mVector10(1, 0); + Vector2 mVector01(0, 1); + Vector2 mVector50(5, 0); + Vector2 mVector05(0, 5); + mVector10.normalize(); + mVector01.normalize(); + mVector50.normalize(); + mVector05.normalize(); + test(mVector10 == Vector2(1, 0)); + test(mVector01 == Vector2(0, 1)); + test(mVector50 == Vector2(1, 0)); + test(mVector05 == Vector2(0, 1)); + } + + /// Test the dot product + void testDotProduct() { + + // Test the dot product + test(Vector2(5, 0).dot(Vector2(0, 8)) == 0); + test(Vector2(5, 8).dot(Vector2(0, 0)) == 0); + test(Vector2(12, 45).dot(Vector2(0, 0)) == 0); + test(Vector2(5, 7).dot(Vector2(5, 7)) == 74); + test(Vector2(3, 6).dot(Vector2(-3, -6)) == -45); + test(Vector2(2, 3).dot(Vector2(-7, 4)) == -2); + test(Vector2(4, 3).dot(Vector2(8, 9)) == 59); + } + + /// Test others methods + void testOthersMethods() { + + // Test the method that returns the absolute vector + test(Vector2(4, 5).getAbsoluteVector() == Vector2(4, 5)); + test(Vector2(-7, -24).getAbsoluteVector() == Vector2(7, 24)); + + // Test the method that returns the minimal element + test(Vector2(6, 35).getMinAxis() == 0); + test(Vector2(564, 45).getMinAxis() == 1); + test(Vector2(98, 23).getMinAxis() == 1); + test(Vector2(-53, -25).getMinAxis() == 0); + + // Test the method that returns the maximal element + test(Vector2(6, 35).getMaxAxis() == 1); + test(Vector2(7, 537).getMaxAxis() == 1); + test(Vector2(98, 23).getMaxAxis() == 0); + test(Vector2(-53, -25).getMaxAxis() == 1); + } + + /// Test the operators + void testOperators() { + + // Test the [] operator + test(mVector34[0] == 3); + test(mVector34[1] == 4); + + // Assignment operator + Vector2 newVector(6, 4); + newVector = Vector2(7, 8); + test(newVector == Vector2(7, 8)); + + // Equality, inequality operators + test(Vector2(5, 7) == Vector2(5, 7)); + test(Vector2(63, 64) != Vector2(63, 84)); + test(Vector2(63, 64) != Vector2(12, 64)); + + // Addition, substraction + Vector2 vector1(6, 33); + Vector2 vector2(7, 68); + test(Vector2(63, 24) + Vector2(3, 4) == Vector2(66, 28)); + test(Vector2(63, 24) - Vector2(3, 4) == Vector2(60, 20)); + vector1 += Vector2(5, 10); + vector2 -= Vector2(10, 21); + test(vector1 == Vector2(11, 43)); + test(vector2 == Vector2(-3, 47)); + + // Multiplication, division + Vector2 vector3(6, 33); + Vector2 vector4(15, 60); + test(Vector2(63, 24) * 3 == Vector2(189, 72)); + test(3 * Vector2(63, 24) == Vector2(189, 72)); + test(Vector2(14, 8) / 2 == Vector2(7, 4)); + vector3 *= 10; + vector4 /= 3; + test(vector3 == Vector2(60, 330)); + test(vector4 == Vector2(5, 20)); + + // Negative operator + Vector2 vector5(-34, 5); + Vector2 negative = -vector5; + test(negative == Vector2(34, -5)); + } + }; + +} + +#endif diff --git a/test/tests/mathematics/TestVector3.h b/test/tests/mathematics/TestVector3.h index d8f42599..274624fb 100644 --- a/test/tests/mathematics/TestVector3.h +++ b/test/tests/mathematics/TestVector3.h @@ -26,14 +26,10 @@ #ifndef TEST_VECTOR3_H #define TEST_VECTOR3_H -#endif - // Libraries #include "../../Test.h" #include "../../../src/mathematics/Vector3.h" -using namespace reactphysics3d; - /// Reactphysics3D namespace namespace reactphysics3d { @@ -232,3 +228,5 @@ class TestVector3 : public Test { }; } + +#endif From f23096af503f09b7e57aecf1b200481155bda786 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 9 May 2013 19:47:09 +0200 Subject: [PATCH 23/66] modify code in the Matrix2x2 class --- src/mathematics/Matrix2x2.cpp | 4 +--- test/tests/mathematics/TestMatrix2x2.h | 24 ++++++++++++------------ 2 files changed, 13 insertions(+), 15 deletions(-) diff --git a/src/mathematics/Matrix2x2.cpp b/src/mathematics/Matrix2x2.cpp index 87353a3d..96aa5806 100644 --- a/src/mathematics/Matrix2x2.cpp +++ b/src/mathematics/Matrix2x2.cpp @@ -80,9 +80,7 @@ Matrix2x2 Matrix2x2::getInverse() const { decimal invDeterminant = decimal(1.0) / determinant; - // TODO : Implement this - assert(false); - Matrix2x2 tempMatrix; + Matrix2x2 tempMatrix(mRows[1][1], -mRows[0][1], -mRows[1][0], mRows[0][0]); // Return the inverse matrix return (invDeterminant * tempMatrix); diff --git a/test/tests/mathematics/TestMatrix2x2.h b/test/tests/mathematics/TestMatrix2x2.h index 082950a8..b4251ec6 100644 --- a/test/tests/mathematics/TestMatrix2x2.h +++ b/test/tests/mathematics/TestMatrix2x2.h @@ -57,7 +57,6 @@ class TestMatrix2x2 : public Test { TestMatrix2x2() : mIdentity(Matrix2x2::identity()), mMatrix1(2, 24, -4, 5) { - } /// Run the tests @@ -147,21 +146,22 @@ class TestMatrix2x2 : public Test { test(mIdentity.getDeterminant() == 1); // Test inverse - Matrix2x2 inverseMatrix = matrix.getInverse(); - test(approxEqual(inverseMatrix[0][0], decimal(0.056369), decimal(10e-6))); - test(approxEqual(inverseMatrix[0][1], decimal(-0.049549), decimal(10e-6))); - test(approxEqual(inverseMatrix[1][0], decimal(0.029460), decimal(10e-6))); - test(approxEqual(inverseMatrix[1][1], decimal(0.038575), decimal(10e-6))); + Matrix2x2 matrix2(1, 2, 3, 4); + Matrix2x2 inverseMatrix = matrix2.getInverse(); + test(approxEqual(inverseMatrix[0][0], decimal(-2), decimal(10e-6))); + test(approxEqual(inverseMatrix[0][1], decimal(1), decimal(10e-6))); + test(approxEqual(inverseMatrix[1][0], decimal(-0.75), decimal(10e-6))); + test(approxEqual(inverseMatrix[1][1], decimal(0.25), decimal(10e-6))); Matrix2x2 inverseMatrix1 = mMatrix1.getInverse(); - test(approxEqual(inverseMatrix1[0][0], decimal(0.030232), decimal(10e-6))); - test(approxEqual(inverseMatrix1[0][1], decimal(0.015676), decimal(10e-6))); - test(approxEqual(inverseMatrix1[1][0], decimal(-0.057410), decimal(10e-6))); - test(approxEqual(inverseMatrix1[1][1], decimal(0.039088), decimal(10e-6))); + test(approxEqual(inverseMatrix1[0][0], decimal(0.04716981), decimal(10e-6))); + test(approxEqual(inverseMatrix1[0][1], decimal(-0.2264150), decimal(10e-6))); + test(approxEqual(inverseMatrix1[1][0], decimal(0.0377358), decimal(10e-6))); + test(approxEqual(inverseMatrix1[1][1], decimal(0.0188679), decimal(10e-6))); // Test absolute matrix - Matrix2x2 matrix2(-2, -3, -4, -5); + Matrix2x2 matrix3(-2, -3, -4, -5); test(matrix.getAbsoluteMatrix() == Matrix2x2(24, 64, 253, 35)); - Matrix2x2 absoluteMatrix = matrix2.getAbsoluteMatrix(); + Matrix2x2 absoluteMatrix = matrix3.getAbsoluteMatrix(); test(absoluteMatrix == Matrix2x2(2, 3, 4, 5)); } From 0f17bd1b0b417c66de03da4557107a14ab0c171c Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sun, 12 May 2013 12:43:07 +0200 Subject: [PATCH 24/66] continue working on the slider joint --- src/constraint/BallAndSocketJoint.cpp | 40 +++++++- src/constraint/BallAndSocketJoint.h | 3 + src/constraint/Constraint.h | 3 + src/constraint/ContactPoint.cpp | 5 + src/constraint/ContactPoint.h | 3 + src/constraint/SliderJoint.cpp | 127 ++++++++++++++++++++++++-- src/constraint/SliderJoint.h | 15 +++ src/mathematics/mathematics.h | 2 + 8 files changed, 188 insertions(+), 10 deletions(-) diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index b4fc6575..9b52ef22 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -78,6 +78,38 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS mInverseMassMatrix = massMatrix.getInverse(); } +// Warm start the constraint (apply the previous impulse at the beginning of the step) +void BallAndSocketJoint::warmstart(const ConstraintSolverData& constraintSolverData) { + + // Get the velocities + Vector3& v1 = constraintSolverData.linearVelocities[mIndexBody1]; + Vector3& v2 = constraintSolverData.linearVelocities[mIndexBody2]; + Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; + Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + decimal inverseMassBody1 = mBody1->getMassInverse(); + decimal inverseMassBody2 = mBody2->getMassInverse(); + Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the impulse P=J^T * lambda + Vector3 linearImpulseBody1 = -mImpulse; + Vector3 angularImpulseBody1 = mImpulse.cross(mU1World); + Vector3 linearImpulseBody2 = mImpulse; + Vector3 angularImpulseBody2 = -mImpulse.cross(mU2World); + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += I2 * angularImpulseBody2; + } +} + // Solve the velocity constraint void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) { @@ -94,8 +126,8 @@ void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& con // Get the inverse mass and inverse inertia tensors of the bodies decimal inverseMassBody1 = mBody1->getMassInverse(); decimal inverseMassBody2 = mBody2->getMassInverse(); - Matrix3x3 inverseInertiaTensorBody1 = mBody1->getInertiaTensorInverseWorld(); - Matrix3x3 inverseInertiaTensorBody2 = mBody2->getInertiaTensorInverseWorld(); + Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // Compute J*v Vector3 Jv = -v1 + mU1World.cross(w1) + v2 - mU2World.cross(w2); @@ -121,11 +153,11 @@ void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& con // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { v1 += inverseMassBody1 * linearImpulseBody1; - w1 += inverseInertiaTensorBody1 * angularImpulseBody1; + w1 += I1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { v2 += inverseMassBody2 * linearImpulseBody2; - w2 += inverseInertiaTensorBody2 * angularImpulseBody2; + w2 += I2 * angularImpulseBody2; } } diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index 0280c28c..0c48cdad 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -104,6 +104,9 @@ class BallAndSocketJoint : public Constraint { /// Initialize before solving the constraint virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); + /// Warm start the constraint (apply the previous impulse at the beginning of the step) + virtual void warmstart(const ConstraintSolverData& constraintSolverData); + /// Solve the velocity constraint virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); diff --git a/src/constraint/Constraint.h b/src/constraint/Constraint.h index 58181768..1be00170 100644 --- a/src/constraint/Constraint.h +++ b/src/constraint/Constraint.h @@ -149,6 +149,9 @@ class Constraint { /// Initialize before solving the constraint virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData) = 0; + /// Warm start the constraint (apply the previous impulse at the beginning of the step) + virtual void warmstart(const ConstraintSolverData& constraintSolverData) = 0; + /// Solve the velocity constraint virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) = 0; diff --git a/src/constraint/ContactPoint.cpp b/src/constraint/ContactPoint.cpp index 178a5e1d..1f068df3 100644 --- a/src/constraint/ContactPoint.cpp +++ b/src/constraint/ContactPoint.cpp @@ -56,6 +56,11 @@ void ContactPoint::initBeforeSolve(const ConstraintSolverData& constraintSolverD } +// Warm start the constraint (apply the previous impulse at the beginning of the step) +void ContactPoint::warmstart(const ConstraintSolverData& constraintSolverData) { + +} + // Solve the velocity constraint void ContactPoint::solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) { diff --git a/src/constraint/ContactPoint.h b/src/constraint/ContactPoint.h index 0047c020..dc66b8b0 100644 --- a/src/constraint/ContactPoint.h +++ b/src/constraint/ContactPoint.h @@ -227,6 +227,9 @@ class ContactPoint : public Constraint { /// Initialize before solving the constraint virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); + /// Warm start the constraint (apply the previous impulse at the beginning of the step) + virtual void warmstart(const ConstraintSolverData& constraintSolverData); + /// Solve the velocity constraint virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index 0d763e9f..73222eef 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -29,7 +29,8 @@ using namespace reactphysics3d; // Constructor -SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) : Constraint(jointInfo) { +SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) + : Constraint(jointInfo), mImpulseTranslation(0, 0), mImpulseRotation(0, 0, 0) { // Compute the local-space anchor point for each body mLocalAnchorPointBody1 = mBody1->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; @@ -64,17 +65,76 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa mN1 = mU1World.getOneUnitOrthogonalVector(); mN2 = mU1World.cross(mN1); - // Compute the cross product used in the Jacobian + // Compute the cross products used in the Jacobian mU1WorldCrossN1 = mN2; mU1WorldCrossN2 = mU1World.cross(mN2); mU2WorldCrossN1 = mU2World.cross(mN1); mU2WorldCrossN2 = mU2World.cross(mN2); - // Compute the mass matrix K=JM^-1J^t for the 2 translation constraints (2x2 matrix) + // Compute the inverse of the mass matrix K=JM^-1J^t for the 2 translation + // constraints (2x2 matrix) const decimal n1Dotn1 = mN1.lengthSquare(); const decimal n2Dotn2 = mN2.lengthSquare(); + const decimal n1Dotn2 = mN1.dot(mN2); const decimal sumInverseMass = mBody1->getMassInverse() + mBody2->getMassInverse(); + const Vector3 I1U2CrossN1 = I1 * mU2WorldCrossN1; + const Vector3 I1U2CrossN2 = I1 * mU2WorldCrossN2; + const Vector3 I2U1CrossN1 = I2 * mU1WorldCrossN1; + const Vector3 I2U1CrossN2 = I2 * mU1WorldCrossN2; + const decimal el11 = sumInverseMass * (n1Dotn1) + mU2WorldCrossN1.dot(I1U2CrossN1) + + mU1WorldCrossN1.dot(I2U1CrossN1); + const decimal el12 = sumInverseMass * (n1Dotn2) + mU2WorldCrossN1.dot(I1U2CrossN2) + + mU1WorldCrossN1.dot(I2U1CrossN2); + const decimal el21 = sumInverseMass * (n1Dotn2) + mU2WorldCrossN2.dot(I1U2CrossN1) + + mU1WorldCrossN2.dot(I2U1CrossN1); + const decimal el22 = sumInverseMass * (n2Dotn2) + mU2WorldCrossN2.dot(I1U2CrossN2) + + mU1WorldCrossN2.dot(I2U1CrossN2); + Matrix2x2 matrixKTranslation(el11, el12, el21, el22); + mInverseMassMatrixTranslationConstraint = matrixKTranslation.getInverse(); + // Compute the inverse of the mass matrix K=JM^-1J^t for the 3 rotation + // contraints (3x3 matrix) + mInverseMassMatrixRotationConstraint = I1 + I2; +} + +// Warm start the constraint (apply the previous impulse at the beginning of the step) +void SliderJoint::warmstart(const ConstraintSolverData& constraintSolverData) { + + // Get the velocities + Vector3& v1 = constraintSolverData.linearVelocities[mIndexBody1]; + Vector3& v2 = constraintSolverData.linearVelocities[mIndexBody2]; + Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; + Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + decimal inverseMassBody1 = mBody1->getMassInverse(); + decimal inverseMassBody2 = mBody2->getMassInverse(); + const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the impulse P=J^T * lambda for the 2 translation constraints + Vector3 linearImpulseBody1 = -mN1 * mImpulseTranslation.x - + mN2 * mImpulseTranslation.y; + Vector3 angularImpulseBody1 = mU2WorldCrossN1 * mImpulseTranslation.x + + mU2WorldCrossN2 * mImpulseTranslation.y; + Vector3 linearImpulseBody2 = mN1 * mImpulseTranslation.x + + mN2 * mImpulseTranslation.y; + Vector3 angularImpulseBody2 = -mU1WorldCrossN1 * mImpulseTranslation.x - + mU1WorldCrossN2 * mImpulseTranslation.y; + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + angularImpulseBody1 += -mImpulseRotation; + angularImpulseBody2 += mImpulseRotation; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += I2 * angularImpulseBody2; + } } // Solve the velocity constraint @@ -93,10 +153,65 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint // Get the inverse mass and inverse inertia tensors of the bodies decimal inverseMassBody1 = mBody1->getMassInverse(); decimal inverseMassBody2 = mBody2->getMassInverse(); - Matrix3x3 inverseInertiaTensorBody1 = mBody1->getInertiaTensorInverseWorld(); - Matrix3x3 inverseInertiaTensorBody2 = mBody2->getInertiaTensorInverseWorld(); + const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); - // Compute J*v + // Compute J*v for the 2 translation constraints + const decimal el1 = -mN1.dot(v1) + mU2WorldCrossN1.dot(w1) + + mN1.dot(v2) - mU1WorldCrossN1.dot(w2); + const decimal el2 = -mN2.dot(v1) + mU2WorldCrossN2.dot(w1) + + mN2.dot(v2) - mU1WorldCrossN2.dot(w2); + const Vector2 JvTranslation(el1, el2); + + // Compute J*v for the 3 translation constraints + const Vector3 JvRotation = w2 - w1; + + // Compute the bias "b" of the translation constraint + Vector2 bTranslation(0, 0); + decimal beta = decimal(0.2); // TODO : Use a constant here + decimal biasFactor = (beta / constraintSolverData.timeStep); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + Vector3 deltaV = x2 + mU2World - x1 - mU1World; + bTranslation.x = biasFactor * deltaV.dot(mN1); + bTranslation.y = biasFactor * deltaV.dot(mN2); + } + + // Compute the bias "b" of the translation constraint + Vector3 bRotation(0, 0, 0); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + Quaternion q1 = mBody1->getTransform().getOrientation(); + Quaternion q2 = mBody2->getTransform().getOrientation(); + Quaternion qDiff = q1 * q2.getInverse(); + bRotation = 2.0 * qDiff.getVectorV(); + } + + // Compute the Lagrange multiplier lambda for the 2 translation constraints + Vector2 deltaLambda = mInverseMassMatrixTranslationConstraint * (-JvTranslation - bTranslation); + mImpulseTranslation += deltaLambda; + + // Compute the Lagrange multiplier lambda for the 3 rotation constraints + Vector3 deltaLambda2 = mInverseMassMatrixRotationConstraint * (-JvRotation - bRotation); + mImpulseRotation += deltaLambda2; + + // Compute the impulse P=J^T * lambda for the 2 translation constraints + Vector3 linearImpulseBody1 = -mN1 * deltaLambda.x - mN2 * deltaLambda.y; + Vector3 angularImpulseBody1 = mU2WorldCrossN1 * deltaLambda.x + mU2WorldCrossN2 * deltaLambda.y; + Vector3 linearImpulseBody2 = mN1 * deltaLambda.x + mN2 * deltaLambda.y; + Vector3 angularImpulseBody2 = -mU1WorldCrossN1 * deltaLambda.x -mU1WorldCrossN2 * deltaLambda.y; + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + angularImpulseBody1 += -deltaLambda2; + angularImpulseBody2 += deltaLambda2; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += I2 * angularImpulseBody2; + } } // Solve the position constraint diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h index 2d9ae4ec..d4aab31a 100644 --- a/src/constraint/SliderJoint.h +++ b/src/constraint/SliderJoint.h @@ -98,6 +98,18 @@ class SliderJoint : public Constraint { /// Cross product of mU2World and mN2 Vector3 mU2WorldCrossN2; + /// Inverse of mass matrix K=JM^-1J^t for the translation constraint (2x2 matrix) + Matrix2x2 mInverseMassMatrixTranslationConstraint; + + /// Inverse of mass matrix K=JM^-1J^t for the rotation constraint (3x3 matrix) + Matrix3x3 mInverseMassMatrixRotationConstraint; + + /// Impulse for the 2 translation constraints + Vector2 mImpulseTranslation; + + /// Impulse for the 3 rotation constraints + Vector3 mImpulseRotation; + public : // -------------------- Methods -------------------- // @@ -114,6 +126,9 @@ class SliderJoint : public Constraint { /// Initialize before solving the constraint virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); + /// Warm start the constraint (apply the previous impulse at the beginning of the step) + virtual void warmstart(const ConstraintSolverData& constraintSolverData); + /// Solve the velocity constraint virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); diff --git a/src/mathematics/mathematics.h b/src/mathematics/mathematics.h index cc5aa467..3697634f 100644 --- a/src/mathematics/mathematics.h +++ b/src/mathematics/mathematics.h @@ -28,8 +28,10 @@ // Libraries #include "Matrix3x3.h" +#include "Matrix2x2.h" #include "Quaternion.h" #include "Vector3.h" +#include "Vector2.h" #include "Transform.h" #include "../configuration.h" #include "mathematics_functions.h" From 74070ae400be34de0b273d4ce08797aaee381a22 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sun, 12 May 2013 14:08:30 +0200 Subject: [PATCH 25/66] Modify CMakeLists.txt file --- CMakeLists.txt | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 47bc8aaa..805fb5d3 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -14,7 +14,8 @@ set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/") OPTION(COMPILE_EXAMPLES "Select this if you want to build the examples" OFF) OPTION(COMPILE_TESTS "Select this if you want to build the tests" OFF) OPTION(PROFILING_ENABLED "Select this if you want to compile with enabled profiling" OFF) - +OPTION(DOUBLE_PRECISION_ENABLED "Select this if you want to compile using double precision floating + values" OFF) # Headers INCLUDE_DIRECTORIES(src) @@ -22,6 +23,10 @@ IF (PROFILING_ENABLED) ADD_DEFINITIONS(-DIS_PROFILING_ACTIVE) ENDIF (PROFILING_ENABLED) +IF (DOUBLE_PRECISION_ENABLED) + ADD_DEFINITIONS(-DIS_DOUBLE_PRECISION_ENABLED) +ENDIF (DOUBLE_PRECISION_ENABLED) + # Library configuration file ( GLOB_RECURSE From 8f37d4ac98e94302c57c7db21499a99a3b40f2f5 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 16 May 2013 21:42:13 +0200 Subject: [PATCH 26/66] Fix issues in the unit tests for Vector2 and Matrix2x2 --- src/mathematics/Matrix2x2.h | 2 +- test/main.cpp | 4 ++++ test/tests/mathematics/TestMatrix2x2.h | 26 +++++++++++++------------- test/tests/mathematics/TestVector2.h | 8 ++++---- 4 files changed, 22 insertions(+), 18 deletions(-) diff --git a/src/mathematics/Matrix2x2.h b/src/mathematics/Matrix2x2.h index ec9c28f1..53a313af 100644 --- a/src/mathematics/Matrix2x2.h +++ b/src/mathematics/Matrix2x2.h @@ -60,7 +60,7 @@ class Matrix2x2 { Matrix2x2(decimal a1, decimal a2, decimal b1, decimal b2); /// Destructor - virtual ~Matrix2x2(); + ~Matrix2x2(); /// Copy-constructor Matrix2x2(const Matrix2x2& matrix); diff --git a/test/main.cpp b/test/main.cpp index 11139b6f..76c2048c 100644 --- a/test/main.cpp +++ b/test/main.cpp @@ -25,9 +25,11 @@ // Libraries #include "TestSuite.h" +#include "tests/mathematics/TestVector2.h" #include "tests/mathematics/TestVector3.h" #include "tests/mathematics/TestTransform.h" #include "tests/mathematics/TestQuaternion.h" +#include "tests/mathematics/TestMatrix2x2.h" #include "tests/mathematics/TestMatrix3x3.h" using namespace reactphysics3d; @@ -38,10 +40,12 @@ int main() { // ---------- Mathematics tests ---------- // + testSuite.addTest(new TestVector2); testSuite.addTest(new TestVector3); testSuite.addTest(new TestTransform); testSuite.addTest(new TestQuaternion); testSuite.addTest(new TestMatrix3x3); + testSuite.addTest(new TestMatrix2x2); // ----------------------------- --------- // diff --git a/test/tests/mathematics/TestMatrix2x2.h b/test/tests/mathematics/TestMatrix2x2.h index b4251ec6..f9144cc9 100644 --- a/test/tests/mathematics/TestMatrix2x2.h +++ b/test/tests/mathematics/TestMatrix2x2.h @@ -107,10 +107,10 @@ class TestMatrix2x2 : public Test { test(column2 == Vector2(24, 5)); // Test method that returns a row - Vector3 row1 = mMatrix1.getRow(0); - Vector3 row2 = mMatrix1.getRow(1); - test(row1 == Vector3(2, 24)); - test(row2 == Vector3(-4, 5)); + Vector2 row1 = mMatrix1.getRow(0); + Vector2 row2 = mMatrix1.getRow(1); + test(row1 == Vector2(2, 24)); + test(row2 == Vector2(-4, 5)); } /// Test the identity methods @@ -136,7 +136,7 @@ class TestMatrix2x2 : public Test { test(transpose == Matrix2x2(2, -4, 24, 5)); // Test trace - test(mMatrix1.getTrace() == 10); + test(mMatrix1.getTrace() ==7); test(Matrix2x2::identity().getTrace() == 2); // Test determinant @@ -150,13 +150,13 @@ class TestMatrix2x2 : public Test { Matrix2x2 inverseMatrix = matrix2.getInverse(); test(approxEqual(inverseMatrix[0][0], decimal(-2), decimal(10e-6))); test(approxEqual(inverseMatrix[0][1], decimal(1), decimal(10e-6))); - test(approxEqual(inverseMatrix[1][0], decimal(-0.75), decimal(10e-6))); - test(approxEqual(inverseMatrix[1][1], decimal(0.25), decimal(10e-6))); + test(approxEqual(inverseMatrix[1][0], decimal(1.5), decimal(10e-6))); + test(approxEqual(inverseMatrix[1][1], decimal(-0.5), decimal(10e-6))); Matrix2x2 inverseMatrix1 = mMatrix1.getInverse(); - test(approxEqual(inverseMatrix1[0][0], decimal(0.04716981), decimal(10e-6))); - test(approxEqual(inverseMatrix1[0][1], decimal(-0.2264150), decimal(10e-6))); - test(approxEqual(inverseMatrix1[1][0], decimal(0.0377358), decimal(10e-6))); - test(approxEqual(inverseMatrix1[1][1], decimal(0.0188679), decimal(10e-6))); + test(approxEqual(inverseMatrix1[0][0], decimal(0.047169811), decimal(10e-6))); + test(approxEqual(inverseMatrix1[0][1], decimal(-0.226415094), decimal(10e-6))); + test(approxEqual(inverseMatrix1[1][0], decimal(0.037735849), decimal(10e-6))); + test(approxEqual(inverseMatrix1[1][1], decimal(0.018867925), decimal(10e-6))); // Test absolute matrix Matrix2x2 matrix3(-2, -3, -4, -5); @@ -208,8 +208,8 @@ class TestMatrix2x2 : public Test { Vector2 vector2(-31, -422); Vector2 test1 = matrix1 * vector1; Vector2 test2 = matrix2 * vector2; - test(test1 == Vector2(-762, -182)); - test(test2 == Vector2(-10190, -1986)); + test(test1 == Vector2(-90, -148)); + test(test2 == Vector2(-1204, -4065)); // Test equality operators test(Matrix2x2(34, 38, 43, 64) == diff --git a/test/tests/mathematics/TestVector2.h b/test/tests/mathematics/TestVector2.h index 0cb1f812..f001fa0a 100644 --- a/test/tests/mathematics/TestVector2.h +++ b/test/tests/mathematics/TestVector2.h @@ -54,7 +54,7 @@ class TestVector2 : public Test { // ---------- Methods ---------- // /// Constructor - TestVector1() : mVectorZero(0, 0), mVector34(3, 4) {} + TestVector2() : mVectorZero(0, 0), mVector34(3, 4) {} /// Run the tests void run() { @@ -98,14 +98,14 @@ class TestVector2 : public Test { test(mVectorZero.lengthSquare() == 0.0); test(Vector2(1, 0).length() == 1.0); test(Vector2(0, 1).length() == 1.0); - test(mVector345.lengthSquare() == 50.0); + test(mVector34.lengthSquare() == 25.0); // Test unit vector methods test(Vector2(1, 0).isUnit()); test(Vector2(0, 1).isUnit()); test(!mVector34.isUnit()); - test(Vector2(5, 0).getUnit() == Vector3(1, 0)); - test(Vector2(0, 5).getUnit() == Vector3(0, 1)); + test(Vector2(5, 0).getUnit() == Vector2(1, 0)); + test(Vector2(0, 5).getUnit() == Vector2(0, 1)); test(!mVector34.isZero()); test(mVectorZero.isZero()); From 78abbaac729038a91cb5e0994a3c23e84b0acd47 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 21 May 2013 23:03:14 +0200 Subject: [PATCH 27/66] Fix issues in the BallAndSocketJoint and SliderJoint --- examples/joints/Scene.cpp | 59 ++++++++- examples/joints/Scene.h | 12 ++ src/constraint/BallAndSocketJoint.cpp | 63 ++++++---- src/constraint/BallAndSocketJoint.h | 18 +-- src/constraint/SliderJoint.cpp | 175 ++++++++++++++++---------- src/constraint/SliderJoint.h | 43 ++++--- src/mathematics/Matrix2x2.h | 2 +- src/reactphysics3d.h | 1 + 8 files changed, 253 insertions(+), 120 deletions(-) diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index c1b5060b..419f5c24 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -59,6 +59,9 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), // Create the Ball-and-Socket joint createBallAndSocketJoints(); + // Create the Slider joint + createSliderJoint(); + // Create the floor createFloor(); @@ -77,12 +80,17 @@ Scene::~Scene() { // Destroy the joints mDynamicsWorld->destroyJoint(mBallAndSocketJoint); + mDynamicsWorld->destroyJoint(mSliderJoint); // Destroy all the boxes of the scene mDynamicsWorld->destroyRigidBody(mBallAndSocketJointBox1->getRigidBody()); mDynamicsWorld->destroyRigidBody(mBallAndSocketJointBox2->getRigidBody()); + mDynamicsWorld->destroyRigidBody(mSliderJointBox1->getRigidBody()); + mDynamicsWorld->destroyRigidBody(mSliderJointBox2->getRigidBody()); delete mBallAndSocketJointBox1; delete mBallAndSocketJointBox2; + delete mSliderJointBox1; + delete mSliderJointBox2; // Destroy the floor mDynamicsWorld->destroyRigidBody(mFloor->getRigidBody()); @@ -104,10 +112,11 @@ void Scene::simulate() { // Update the position and orientation of the boxes mBallAndSocketJointBox1->updateTransform(); mBallAndSocketJointBox2->updateTransform(); + mSliderJointBox1->updateTransform(); + mSliderJointBox2->updateTransform(); // Update the position and orientation of the floor mFloor->updateTransform(); - } } @@ -139,6 +148,8 @@ void Scene::render() { // Render all the boxes mBallAndSocketJointBox1->render(mPhongShader); mBallAndSocketJointBox2->render(mPhongShader); + mSliderJointBox1->render(mPhongShader); + mSliderJointBox2->render(mPhongShader); // Render the floor mFloor->render(mPhongShader); @@ -191,6 +202,52 @@ void Scene::createBallAndSocketJoints() { mDynamicsWorld->createJoint(jointInfo)); } +/// Create the boxes and joint for the Slider joint example +void Scene::createSliderJoint() { + + // --------------- Create the first box --------------- // + + // Position of the box + openglframework::Vector3 positionBox1(-4, 6, 0); + + // Create a box and a corresponding rigid in the dynamics world + mSliderJointBox1 = new Box(BOX_SIZE, positionBox1 , BOX_MASS, mDynamicsWorld); + + // The fist box cannot move + mSliderJointBox1->getRigidBody()->setIsMotionEnabled(false); + + // Set the bouncing factor of the box + mSliderJointBox1->getRigidBody()->setRestitution(0.4); + + // --------------- Create the second box --------------- // + + // Position of the box + openglframework::Vector3 positionBox2(2, 4, 0); + + // Create a box and a corresponding rigid in the dynamics world + mSliderJointBox2 = new Box(BOX_SIZE, positionBox2 , BOX_MASS, mDynamicsWorld); + + // The second box is allowed to move + mSliderJointBox2->getRigidBody()->setIsMotionEnabled(true); + + // Set the bouncing factor of the box + mSliderJointBox2->getRigidBody()->setRestitution(0.4); + + // --------------- Create the joint --------------- // + + // Create the joint info object + rp3d::RigidBody* body1 = mSliderJointBox1->getRigidBody(); + rp3d::RigidBody* body2 = mSliderJointBox2->getRigidBody(); + const rp3d::Vector3& body1Position = body1->getTransform().getPosition(); + const rp3d::Vector3& body2Position = body2->getTransform().getPosition(); + const rp3d::Vector3 anchorPointWorldSpace = 0.5 * (body2Position + body1Position); + const rp3d::Vector3 sliderAxisWorldSpace = body2Position - body1Position; + rp3d::SliderJointInfo jointInfo(body1, body2, anchorPointWorldSpace, sliderAxisWorldSpace); + + // Create the joint in the dynamics world + mSliderJoint = dynamic_cast(mDynamicsWorld->createJoint(jointInfo)); +} + // Create the floor void Scene::createFloor() { diff --git a/examples/joints/Scene.h b/examples/joints/Scene.h index 7d89dc3a..3bc5cb5d 100644 --- a/examples/joints/Scene.h +++ b/examples/joints/Scene.h @@ -62,6 +62,15 @@ class Scene { /// Ball-and-Socket joint rp3d::BallAndSocketJoint* mBallAndSocketJoint; + /// Box 1 of Slider joint + Box* mSliderJointBox1; + + /// Box 2 of Slider joint + Box* mSliderJointBox2; + + /// Slider joint + rp3d::SliderJoint* mSliderJoint; + /// Box for the floor Box* mFloor; @@ -76,6 +85,9 @@ class Scene { /// Create the boxes and joints for the Ball-and-Socket joint example void createBallAndSocketJoints(); + /// Create the boxes and joint for the Slider joint example + void createSliderJoint(); + /// Create the floor void createFloor(); diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index 9b52ef22..aa3409f2 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -59,23 +59,36 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // Compute the vector from body center to the anchor point in world-space - mU1World = orientationBody1 * mLocalAnchorPointBody1; - mU2World = orientationBody2 * mLocalAnchorPointBody2; + mR1World = orientationBody1 * mLocalAnchorPointBody1; + mR2World = orientationBody2 * mLocalAnchorPointBody2; // Compute the corresponding skew-symmetric matrices - Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mU1World); - Matrix3x3 skewSymmetricMatrixU2= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mU2World); + Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR1World); + Matrix3x3 skewSymmetricMatrixU2= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR2World); // Compute the matrix K=JM^-1J^t (3x3 matrix) - decimal inverseMassBodies = mBody1->getMassInverse() + mBody2->getMassInverse(); - Matrix3x3 massMatrix= Matrix3x3(inverseMassBodies, 0, 0, + decimal inverseMassBodies = 0.0; + if (mBody1->getIsMotionEnabled()) { + inverseMassBodies += mBody1->getMassInverse(); + } + if (mBody2->getIsMotionEnabled()) { + inverseMassBodies += mBody2->getMassInverse(); + } + Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, 0, inverseMassBodies, 0, - 0, 0, inverseMassBodies) + - skewSymmetricMatrixU1 * I1 * skewSymmetricMatrixU1.getTranspose() + - skewSymmetricMatrixU2 * I2 * skewSymmetricMatrixU2.getTranspose(); + 0, 0, inverseMassBodies); + if (mBody1->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU1 * I1 * skewSymmetricMatrixU1.getTranspose(); + } + if (mBody2->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU2 * I2 * skewSymmetricMatrixU2.getTranspose(); + } // Compute the inverse mass matrix K^-1 - mInverseMassMatrix = massMatrix.getInverse(); + mInverseMassMatrix.setToZero(); + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrix = massMatrix.getInverse(); + } } // Warm start the constraint (apply the previous impulse at the beginning of the step) @@ -88,16 +101,16 @@ void BallAndSocketJoint::warmstart(const ConstraintSolverData& constraintSolverD Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; // Get the inverse mass and inverse inertia tensors of the bodies - decimal inverseMassBody1 = mBody1->getMassInverse(); - decimal inverseMassBody2 = mBody2->getMassInverse(); - Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + const decimal inverseMassBody1 = mBody1->getMassInverse(); + const decimal inverseMassBody2 = mBody2->getMassInverse(); + const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // Compute the impulse P=J^T * lambda - Vector3 linearImpulseBody1 = -mImpulse; - Vector3 angularImpulseBody1 = mImpulse.cross(mU1World); - Vector3 linearImpulseBody2 = mImpulse; - Vector3 angularImpulseBody2 = -mImpulse.cross(mU2World); + const Vector3 linearImpulseBody1 = -mImpulse; + const Vector3 angularImpulseBody1 = mImpulse.cross(mR1World); + const Vector3 linearImpulseBody2 = mImpulse; + const Vector3 angularImpulseBody2 = -mImpulse.cross(mR2World); // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { @@ -130,25 +143,25 @@ void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& con Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // Compute J*v - Vector3 Jv = -v1 + mU1World.cross(w1) + v2 - mU2World.cross(w2); + const Vector3 Jv = -v1 + mR1World.cross(w1) + v2 - mR2World.cross(w2); // Compute the bias "b" of the constraint Vector3 b(0, 0, 0); if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { decimal beta = decimal(0.2); // TODO : Use a constant here decimal biasFactor = (beta / constraintSolverData.timeStep); - b = biasFactor * (x2 + mU2World - x1 - mU1World); + b = biasFactor * (x2 + mR2World - x1 - mR1World); } // Compute the Lagrange multiplier lambda - Vector3 deltaLambda = mInverseMassMatrix * (-Jv - b); + const Vector3 deltaLambda = mInverseMassMatrix * (-Jv - b); mImpulse += deltaLambda; // Compute the impulse P=J^T * lambda - Vector3 linearImpulseBody1 = -deltaLambda; - Vector3 angularImpulseBody1 = deltaLambda.cross(mU1World); - Vector3 linearImpulseBody2 = deltaLambda; - Vector3 angularImpulseBody2 = -deltaLambda.cross(mU2World); + const Vector3 linearImpulseBody1 = -deltaLambda; + const Vector3 angularImpulseBody1 = deltaLambda.cross(mR1World); + const Vector3 linearImpulseBody2 = deltaLambda; + const Vector3 angularImpulseBody2 = -deltaLambda.cross(mR2World); // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index 0c48cdad..556e316b 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -50,7 +50,7 @@ struct BallAndSocketJointInfo : public ConstraintInfo { BallAndSocketJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace) : ConstraintInfo(rigidBody1, rigidBody2, BALLSOCKETJOINT), - anchorPointWorldSpace(initAnchorPointWorldSpace){} + anchorPointWorldSpace(initAnchorPointWorldSpace) {} }; // Class BallAndSocketJoint @@ -64,23 +64,23 @@ class BallAndSocketJoint : public Constraint { // -------------------- Attributes -------------------- // - /// Anchor point of body 1 (in local space coordinates) + /// Anchor point of body 1 (in local-space coordinates of body 1) Vector3 mLocalAnchorPointBody1; - /// Anchor point of body 2 (in local space coordinates) + /// Anchor point of body 2 (in local-space coordinates of body 2) Vector3 mLocalAnchorPointBody2; /// Vector from center of body 2 to anchor point in world-space - Vector3 mU1World; + Vector3 mR1World; /// Vector from center of body 2 to anchor point in world-space - Vector3 mU2World; + Vector3 mR2World; - /// Skew-Symmetric matrix for cross product with vector mU1World - Matrix3x3 mSkewSymmetricMatrixU1World; + /// Skew-Symmetric matrix for cross product with vector mR1World + Matrix3x3 mSkewSymmetricMatrixR1World; - /// Skew-Symmetric matrix for cross product with vector mU2World - Matrix3x3 mSkewSymmetricMatrixU2World; + /// Skew-Symmetric matrix for cross product with vector mR2World + Matrix3x3 mSkewSymmetricMatrixR2World; /// Inverse mass matrix K=JM^-1J^-t of the constraint Matrix3x3 mInverseMassMatrix; diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index 73222eef..8d395b31 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -35,6 +35,11 @@ SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) // Compute the local-space anchor point for each body mLocalAnchorPointBody1 = mBody1->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; mLocalAnchorPointBody2 = mBody2->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; + + // Compute the slider axis in local-space of body 1 + mSliderAxisBody1 = mBody1->getTransform().getOrientation().getInverse() * + jointInfo.sliderAxisWorldSpace; + mSliderAxisBody1.normalize(); } // Destructor @@ -49,6 +54,10 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa mIndexBody1 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody1)->second; mIndexBody2 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody2)->second; + // Get the body positions + const Vector3& x1 = mBody1->getTransform().getPosition(); + const Vector3& x2 = mBody2->getTransform().getPosition(); + // Get the bodies positions and orientations const Quaternion& orientationBody1 = mBody1->getTransform().getOrientation(); const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); @@ -57,44 +66,72 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); - // Compute the vector from body center to the anchor point in world-space - mU1World = orientationBody1 * mLocalAnchorPointBody1; - mU2World = orientationBody2 * mLocalAnchorPointBody2; + // Vector from body center to the anchor point + mR1 = orientationBody1 * mLocalAnchorPointBody1; + mR2 = orientationBody2 * mLocalAnchorPointBody2; - // Compute the two orthogonal vectors to vector mU1World in world-space - mN1 = mU1World.getOneUnitOrthogonalVector(); - mN2 = mU1World.cross(mN1); + // Compute the vector u + const Vector3 u = x2 + mR2 - x1 - mR1; + + // Compute the two orthogonal vectors to the slider axis in local-space of body 1 + Vector3 sliderAxisWorld = orientationBody1 * mSliderAxisBody1; + sliderAxisWorld.normalize(); + mN1 = sliderAxisWorld.getOneUnitOrthogonalVector(); + mN2 = sliderAxisWorld.cross(mN1); // Compute the cross products used in the Jacobian - mU1WorldCrossN1 = mN2; - mU1WorldCrossN2 = mU1World.cross(mN2); - mU2WorldCrossN1 = mU2World.cross(mN1); - mU2WorldCrossN2 = mU2World.cross(mN2); + mR2CrossN1 = mR2.cross(mN1); + mR2CrossN2 = mR2.cross(mN2); + const Vector3 r1PlusU = mR1 + u; + mR1PlusUCrossN1 = (r1PlusU).cross(mN1); + mR1PlusUCrossN2 = (r1PlusU).cross(mN2); // Compute the inverse of the mass matrix K=JM^-1J^t for the 2 translation // constraints (2x2 matrix) const decimal n1Dotn1 = mN1.lengthSquare(); const decimal n2Dotn2 = mN2.lengthSquare(); const decimal n1Dotn2 = mN1.dot(mN2); - const decimal sumInverseMass = mBody1->getMassInverse() + mBody2->getMassInverse(); - const Vector3 I1U2CrossN1 = I1 * mU2WorldCrossN1; - const Vector3 I1U2CrossN2 = I1 * mU2WorldCrossN2; - const Vector3 I2U1CrossN1 = I2 * mU1WorldCrossN1; - const Vector3 I2U1CrossN2 = I2 * mU1WorldCrossN2; - const decimal el11 = sumInverseMass * (n1Dotn1) + mU2WorldCrossN1.dot(I1U2CrossN1) + - mU1WorldCrossN1.dot(I2U1CrossN1); - const decimal el12 = sumInverseMass * (n1Dotn2) + mU2WorldCrossN1.dot(I1U2CrossN2) + - mU1WorldCrossN1.dot(I2U1CrossN2); - const decimal el21 = sumInverseMass * (n1Dotn2) + mU2WorldCrossN2.dot(I1U2CrossN1) + - mU1WorldCrossN2.dot(I2U1CrossN1); - const decimal el22 = sumInverseMass * (n2Dotn2) + mU2WorldCrossN2.dot(I1U2CrossN2) + - mU1WorldCrossN2.dot(I2U1CrossN2); + decimal sumInverseMass = 0.0; + Vector3 I1R1PlusUCrossN1(0, 0, 0); + Vector3 I1R1PlusUCrossN2(0, 0, 0); + Vector3 I2R2CrossN1(0, 0, 0); + Vector3 I2R2CrossN2(0, 0, 0); + if (mBody1->getIsMotionEnabled()) { + sumInverseMass += mBody1->getMassInverse(); + I1R1PlusUCrossN1 = I1 * mR1PlusUCrossN1; + I1R1PlusUCrossN2 = I1 * mR1PlusUCrossN2; + } + if (mBody2->getIsMotionEnabled()) { + sumInverseMass += mBody2->getMassInverse(); + I2R2CrossN1 = I2 * mR2CrossN1; + I2R2CrossN2 = I2 * mR2CrossN2; + } + const decimal el11 = sumInverseMass * (n1Dotn1) + mR1PlusUCrossN1.dot(I1R1PlusUCrossN1) + + mR2CrossN1.dot(I2R2CrossN1); + const decimal el12 = sumInverseMass * (n1Dotn2) + mR1PlusUCrossN1.dot(I1R1PlusUCrossN2) + + mR2CrossN1.dot(I2R2CrossN2); + const decimal el21 = sumInverseMass * (n1Dotn2) + mR1PlusUCrossN2.dot(I1R1PlusUCrossN1) + + mR2CrossN2.dot(I2R2CrossN1); + const decimal el22 = sumInverseMass * (n2Dotn2) + mR1PlusUCrossN2.dot(I1R1PlusUCrossN2) + + mR2CrossN2.dot(I2R2CrossN2); Matrix2x2 matrixKTranslation(el11, el12, el21, el22); - mInverseMassMatrixTranslationConstraint = matrixKTranslation.getInverse(); + mInverseMassMatrixTranslationConstraint.setToZero(); + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixTranslationConstraint = matrixKTranslation.getInverse(); + } // Compute the inverse of the mass matrix K=JM^-1J^t for the 3 rotation // contraints (3x3 matrix) - mInverseMassMatrixRotationConstraint = I1 + I2; + mInverseMassMatrixRotationConstraint.setToZero(); + if (mBody1->getIsMotionEnabled()) { + mInverseMassMatrixRotationConstraint += I1; + } + if (mBody2->getIsMotionEnabled()) { + mInverseMassMatrixRotationConstraint += I2; + } + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixRotationConstraint = mInverseMassMatrixRotationConstraint.getInverse(); + } } // Warm start the constraint (apply the previous impulse at the beginning of the step) @@ -107,20 +144,18 @@ void SliderJoint::warmstart(const ConstraintSolverData& constraintSolverData) { Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; // Get the inverse mass and inverse inertia tensors of the bodies - decimal inverseMassBody1 = mBody1->getMassInverse(); - decimal inverseMassBody2 = mBody2->getMassInverse(); + const decimal inverseMassBody1 = mBody1->getMassInverse(); + const decimal inverseMassBody2 = mBody2->getMassInverse(); const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // Compute the impulse P=J^T * lambda for the 2 translation constraints - Vector3 linearImpulseBody1 = -mN1 * mImpulseTranslation.x - - mN2 * mImpulseTranslation.y; - Vector3 angularImpulseBody1 = mU2WorldCrossN1 * mImpulseTranslation.x + - mU2WorldCrossN2 * mImpulseTranslation.y; - Vector3 linearImpulseBody2 = mN1 * mImpulseTranslation.x + - mN2 * mImpulseTranslation.y; - Vector3 angularImpulseBody2 = -mU1WorldCrossN1 * mImpulseTranslation.x - - mU1WorldCrossN2 * mImpulseTranslation.y; + Vector3 linearImpulseBody1 = -mN1 * mImpulseTranslation.x - mN2 * mImpulseTranslation.y; + Vector3 angularImpulseBody1 = -mR1PlusUCrossN1 * mImpulseTranslation.x - + mR1PlusUCrossN2 * mImpulseTranslation.y; + Vector3 linearImpulseBody2 = -linearImpulseBody1; + Vector3 angularImpulseBody2 = mR2CrossN1 * mImpulseTranslation.x + + mR2CrossN2 * mImpulseTranslation.y; // Compute the impulse P=J^T * lambda for the 3 rotation constraints angularImpulseBody1 += -mImpulseRotation; @@ -156,52 +191,34 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); - // Compute J*v for the 2 translation constraints - const decimal el1 = -mN1.dot(v1) + mU2WorldCrossN1.dot(w1) + - mN1.dot(v2) - mU1WorldCrossN1.dot(w2); - const decimal el2 = -mN2.dot(v1) + mU2WorldCrossN2.dot(w1) + - mN2.dot(v2) - mU1WorldCrossN2.dot(w2); - const Vector2 JvTranslation(el1, el2); + // --------------- Translation Constraints --------------- // - // Compute J*v for the 3 translation constraints - const Vector3 JvRotation = w2 - w1; + // Compute J*v for the 2 translation constraints + const decimal el1 = -mN1.dot(v1) - w1.dot(mR1PlusUCrossN1) + + mN1.dot(v2) + w2.dot(mR2CrossN1); + const decimal el2 = -mN2.dot(v1) - w1.dot(mR1PlusUCrossN2) + + mN2.dot(v2) + w2.dot(mR2CrossN2); + const Vector2 JvTranslation(el1, el2); // Compute the bias "b" of the translation constraint Vector2 bTranslation(0, 0); decimal beta = decimal(0.2); // TODO : Use a constant here decimal biasFactor = (beta / constraintSolverData.timeStep); if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { - Vector3 deltaV = x2 + mU2World - x1 - mU1World; + Vector3 deltaV = x2 + mR2 - x1 - mR1; bTranslation.x = biasFactor * deltaV.dot(mN1); bTranslation.y = biasFactor * deltaV.dot(mN2); } - // Compute the bias "b" of the translation constraint - Vector3 bRotation(0, 0, 0); - if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { - Quaternion q1 = mBody1->getTransform().getOrientation(); - Quaternion q2 = mBody2->getTransform().getOrientation(); - Quaternion qDiff = q1 * q2.getInverse(); - bRotation = 2.0 * qDiff.getVectorV(); - } - // Compute the Lagrange multiplier lambda for the 2 translation constraints Vector2 deltaLambda = mInverseMassMatrixTranslationConstraint * (-JvTranslation - bTranslation); mImpulseTranslation += deltaLambda; - // Compute the Lagrange multiplier lambda for the 3 rotation constraints - Vector3 deltaLambda2 = mInverseMassMatrixRotationConstraint * (-JvRotation - bRotation); - mImpulseRotation += deltaLambda2; - // Compute the impulse P=J^T * lambda for the 2 translation constraints Vector3 linearImpulseBody1 = -mN1 * deltaLambda.x - mN2 * deltaLambda.y; - Vector3 angularImpulseBody1 = mU2WorldCrossN1 * deltaLambda.x + mU2WorldCrossN2 * deltaLambda.y; - Vector3 linearImpulseBody2 = mN1 * deltaLambda.x + mN2 * deltaLambda.y; - Vector3 angularImpulseBody2 = -mU1WorldCrossN1 * deltaLambda.x -mU1WorldCrossN2 * deltaLambda.y; - - // Compute the impulse P=J^T * lambda for the 3 rotation constraints - angularImpulseBody1 += -deltaLambda2; - angularImpulseBody2 += deltaLambda2; + Vector3 angularImpulseBody1 = -mR1PlusUCrossN1 * deltaLambda.x -mR1PlusUCrossN2 * deltaLambda.y; + Vector3 linearImpulseBody2 = -linearImpulseBody1; + Vector3 angularImpulseBody2 = mR2CrossN1 * deltaLambda.x + mR2CrossN2 * deltaLambda.y; // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { @@ -212,6 +229,36 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint v2 += inverseMassBody2 * linearImpulseBody2; w2 += I2 * angularImpulseBody2; } + + // --------------- Rotation Constraints --------------- // + + // Compute J*v for the 3 rotation constraints + const Vector3 JvRotation = w2 - w1; + + // Compute the bias "b" of the rotation constraint + Vector3 bRotation(0, 0, 0); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + Quaternion q1 = mBody1->getTransform().getOrientation(); + Quaternion q2 = mBody2->getTransform().getOrientation(); + Quaternion qDiff = q1 * q2.getInverse(); + bRotation = 2.0 * qDiff.getVectorV(); + } + + // Compute the Lagrange multiplier lambda for the 3 rotation constraints + Vector3 deltaLambda2 = mInverseMassMatrixRotationConstraint * (-JvRotation - bRotation); + mImpulseRotation += deltaLambda2; + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + angularImpulseBody1 = -deltaLambda2; + angularImpulseBody2 = deltaLambda2; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + w2 += I2 * angularImpulseBody2; + } } // Solve the position constraint diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h index d4aab31a..44742a15 100644 --- a/src/constraint/SliderJoint.h +++ b/src/constraint/SliderJoint.h @@ -47,15 +47,15 @@ struct SliderJointInfo : public ConstraintInfo { Vector3 anchorPointWorldSpace; /// Slider axis (in world-space coordinates) - Vector3 axisWorldSpace; + Vector3 sliderAxisWorldSpace; /// Constructor SliderJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace, - const Vector3& initAxisWorldSpace) + const Vector3& initSliderAxisWorldSpace) : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), - axisWorldSpace(initAxisWorldSpace) {} + sliderAxisWorldSpace(initSliderAxisWorldSpace) {} }; // Class SliderJoint @@ -68,35 +68,38 @@ class SliderJoint : public Constraint { // -------------------- Attributes -------------------- // - /// Anchor point of body 1 (in local space coordinates) + /// Anchor point of body 1 (in local-space coordinates of body 1) Vector3 mLocalAnchorPointBody1; - /// Anchor point of body 2 (in local space coordinates) + /// Anchor point of body 2 (in local-space coordinates of body 2) Vector3 mLocalAnchorPointBody2; - /// Vector from center of body 2 to anchor point in world-space - Vector3 mU1World; + /// Slider axis (in local-space coordinates of body 1) + Vector3 mSliderAxisBody1; - /// Vector from center of body 2 to anchor point in world-space - Vector3 mU2World; - - /// First vector orthogonal to vector mU1World in world-space + /// First vector orthogonal to the slider axis local-space of body 1 Vector3 mN1; - /// Second vector orthogonal to vector mU1World and mN1 in world-space + /// Second vector orthogonal to the slider axis and mN1 in local-space of body 1 Vector3 mN2; - /// Cross product of mU1World and mN1 - Vector3 mU1WorldCrossN1; + /// Vector r1 in world-space coordinates + Vector3 mR1; - /// Cross product of mU1World and mN2 - Vector3 mU1WorldCrossN2; + /// Vector r2 in world-space coordinates + Vector3 mR2; - /// Cross product of mU2World and mN1 - Vector3 mU2WorldCrossN1; + /// Cross product of r2 and n1 + Vector3 mR2CrossN1; - /// Cross product of mU2World and mN2 - Vector3 mU2WorldCrossN2; + /// Cross product of r2 and n2 + Vector3 mR2CrossN2; + + /// Cross product of vector (r1 + u) and n1 + Vector3 mR1PlusUCrossN1; + + /// Cross product of vector (r1 + u) and n2 + Vector3 mR1PlusUCrossN2; /// Inverse of mass matrix K=JM^-1J^t for the translation constraint (2x2 matrix) Matrix2x2 mInverseMassMatrixTranslationConstraint; diff --git a/src/mathematics/Matrix2x2.h b/src/mathematics/Matrix2x2.h index 53a313af..d19846bc 100644 --- a/src/mathematics/Matrix2x2.h +++ b/src/mathematics/Matrix2x2.h @@ -44,7 +44,7 @@ class Matrix2x2 { // -------------------- Attributes -------------------- // /// Rows of the matrix; - Vector2 mRows[3]; + Vector2 mRows[2]; public : diff --git a/src/reactphysics3d.h b/src/reactphysics3d.h index 5f463e13..df9e367c 100644 --- a/src/reactphysics3d.h +++ b/src/reactphysics3d.h @@ -48,6 +48,7 @@ #include "collision/shapes/CylinderShape.h" #include "collision/shapes/AABB.h" #include "constraint/BallAndSocketJoint.h" +#include "constraint/SliderJoint.h" /// Alias to the ReactPhysics3D namespace namespace rp3d = reactphysics3d; From c7aa6e7e0e7837d175d9fa40b618856b89c07225 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 22 May 2013 23:38:30 +0200 Subject: [PATCH 28/66] Start working of the SliderJoint limits --- src/constraint/SliderJoint.cpp | 142 +++++++++++++++++++++++---------- src/constraint/SliderJoint.h | 67 +++++++++++++++- 2 files changed, 164 insertions(+), 45 deletions(-) diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index 8d395b31..a9a322a2 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -30,11 +30,23 @@ using namespace reactphysics3d; // Constructor SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) - : Constraint(jointInfo), mImpulseTranslation(0, 0), mImpulseRotation(0, 0, 0) { + : Constraint(jointInfo), mImpulseTranslation(0, 0), mImpulseRotation(0, 0, 0), + mImpulseLowerLimit(0), mIsLimitsActive(jointInfo.isLimitsActive), + mLowerLimit(jointInfo.lowerLimit), mUpperLimit(jointInfo.upperLimit) { + + assert(mUpperLimit >= 0.0); + assert(mLowerLimit <= 0.0); // Compute the local-space anchor point for each body - mLocalAnchorPointBody1 = mBody1->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; - mLocalAnchorPointBody2 = mBody2->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; + const Transform& transform1 = mBody1->getTransform(); + const Transform& transform2 = mBody2->getTransform(); + mLocalAnchorPointBody1 = transform1.getInverse() * jointInfo.anchorPointWorldSpace; + mLocalAnchorPointBody2 = transform2.getInverse() * jointInfo.anchorPointWorldSpace; + + // Compute the initial orientation difference between the two bodies + mInitOrientationDifference = transform2.getOrientation() * + transform1.getOrientation().getInverse(); + mInitOrientationDifference.normalize(); // Compute the slider axis in local-space of body 1 mSliderAxisBody1 = mBody1->getTransform().getOrientation().getInverse() * @@ -54,11 +66,9 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa mIndexBody1 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody1)->second; mIndexBody2 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody2)->second; - // Get the body positions + // Get the bodies positions and orientations const Vector3& x1 = mBody1->getTransform().getPosition(); const Vector3& x2 = mBody2->getTransform().getPosition(); - - // Get the bodies positions and orientations const Quaternion& orientationBody1 = mBody1->getTransform().getOrientation(); const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); @@ -74,23 +84,26 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa const Vector3 u = x2 + mR2 - x1 - mR1; // Compute the two orthogonal vectors to the slider axis in local-space of body 1 - Vector3 sliderAxisWorld = orientationBody1 * mSliderAxisBody1; - sliderAxisWorld.normalize(); - mN1 = sliderAxisWorld.getOneUnitOrthogonalVector(); - mN2 = sliderAxisWorld.cross(mN1); + mSliderAxisWorld = orientationBody1 * mSliderAxisBody1; + mSliderAxisWorld.normalize(); + mN1 = mSliderAxisWorld.getOneUnitOrthogonalVector(); + mN2 = mSliderAxisWorld.cross(mN1); - // Compute the cross products used in the Jacobian + // Check if the limit constraints are violated or not + decimal lowerLimitError = u.dot(mSliderAxisWorld) - mLowerLimit; + mIsLowerLimitViolated = (lowerLimitError <= 0); + + // Compute the cross products used in the Jacobians mR2CrossN1 = mR2.cross(mN1); mR2CrossN2 = mR2.cross(mN2); + mR2CrossSliderAxis = mR2.cross(mSliderAxisWorld); const Vector3 r1PlusU = mR1 + u; mR1PlusUCrossN1 = (r1PlusU).cross(mN1); mR1PlusUCrossN2 = (r1PlusU).cross(mN2); + mR1PlusUCrossSliderAxis = (r1PlusU).cross(mSliderAxisWorld); // Compute the inverse of the mass matrix K=JM^-1J^t for the 2 translation // constraints (2x2 matrix) - const decimal n1Dotn1 = mN1.lengthSquare(); - const decimal n2Dotn2 = mN2.lengthSquare(); - const decimal n1Dotn2 = mN1.dot(mN2); decimal sumInverseMass = 0.0; Vector3 I1R1PlusUCrossN1(0, 0, 0); Vector3 I1R1PlusUCrossN2(0, 0, 0); @@ -106,13 +119,13 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa I2R2CrossN1 = I2 * mR2CrossN1; I2R2CrossN2 = I2 * mR2CrossN2; } - const decimal el11 = sumInverseMass * (n1Dotn1) + mR1PlusUCrossN1.dot(I1R1PlusUCrossN1) + + const decimal el11 = sumInverseMass + mR1PlusUCrossN1.dot(I1R1PlusUCrossN1) + mR2CrossN1.dot(I2R2CrossN1); - const decimal el12 = sumInverseMass * (n1Dotn2) + mR1PlusUCrossN1.dot(I1R1PlusUCrossN2) + + const decimal el12 = mR1PlusUCrossN1.dot(I1R1PlusUCrossN2) + mR2CrossN1.dot(I2R2CrossN2); - const decimal el21 = sumInverseMass * (n1Dotn2) + mR1PlusUCrossN2.dot(I1R1PlusUCrossN1) + + const decimal el21 = mR1PlusUCrossN2.dot(I1R1PlusUCrossN1) + mR2CrossN2.dot(I2R2CrossN1); - const decimal el22 = sumInverseMass * (n2Dotn2) + mR1PlusUCrossN2.dot(I1R1PlusUCrossN2) + + const decimal el22 = sumInverseMass + mR1PlusUCrossN2.dot(I1R1PlusUCrossN2) + mR2CrossN2.dot(I2R2CrossN2); Matrix2x2 matrixKTranslation(el11, el12, el21, el22); mInverseMassMatrixTranslationConstraint.setToZero(); @@ -120,6 +133,16 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa mInverseMassMatrixTranslationConstraint = matrixKTranslation.getInverse(); } + // Compute the bias "b" of the translation constraint + mBTranslation.setToZero(); + decimal beta = decimal(0.2); // TODO : Use a constant here + decimal biasFactor = (beta / constraintSolverData.timeStep); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBTranslation.x = u.dot(mN1); + mBTranslation.y = u.dot(mN2); + mBTranslation *= biasFactor; + } + // Compute the inverse of the mass matrix K=JM^-1J^t for the 3 rotation // contraints (3x3 matrix) mInverseMassMatrixRotationConstraint.setToZero(); @@ -132,6 +155,27 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { mInverseMassMatrixRotationConstraint = mInverseMassMatrixRotationConstraint.getInverse(); } + + // Compute the bias "b" of the rotation constraint + mBRotation.setToZero(); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + Quaternion currentOrientationDifference = orientationBody2 * orientationBody1.getInverse(); + currentOrientationDifference.normalize(); + const Quaternion qError = currentOrientationDifference * + mInitOrientationDifference.getInverse(); + mBRotation = biasFactor * decimal(2.0) * qError.getVectorV(); + } + + // Compute the inverse of the mass matrix K=JM^-1J^t for the lower limit (1x1 matrix) + mInverseMassMatrixLowerLimit = sumInverseMass + + mR1PlusUCrossSliderAxis.dot(I1 * mR1PlusUCrossSliderAxis) + + mR2CrossSliderAxis.dot(I2 * mR2CrossSliderAxis); + + // Compute the bias "b" of the lower limit constraint + mBLowerLimit = 0.0; + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBLowerLimit = biasFactor * lowerLimitError; + } } // Warm start the constraint (apply the previous impulse at the beginning of the step) @@ -175,10 +219,6 @@ void SliderJoint::warmstart(const ConstraintSolverData& constraintSolverData) { // Solve the velocity constraint void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) { - // Get the body positions - const Vector3& x1 = mBody1->getTransform().getPosition(); - const Vector3& x2 = mBody2->getTransform().getPosition(); - // Get the velocities Vector3& v1 = constraintSolverData.linearVelocities[mIndexBody1]; Vector3& v2 = constraintSolverData.linearVelocities[mIndexBody2]; @@ -200,18 +240,8 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint mN2.dot(v2) + w2.dot(mR2CrossN2); const Vector2 JvTranslation(el1, el2); - // Compute the bias "b" of the translation constraint - Vector2 bTranslation(0, 0); - decimal beta = decimal(0.2); // TODO : Use a constant here - decimal biasFactor = (beta / constraintSolverData.timeStep); - if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { - Vector3 deltaV = x2 + mR2 - x1 - mR1; - bTranslation.x = biasFactor * deltaV.dot(mN1); - bTranslation.y = biasFactor * deltaV.dot(mN2); - } - // Compute the Lagrange multiplier lambda for the 2 translation constraints - Vector2 deltaLambda = mInverseMassMatrixTranslationConstraint * (-JvTranslation - bTranslation); + Vector2 deltaLambda = mInverseMassMatrixTranslationConstraint * (-JvTranslation -mBTranslation); mImpulseTranslation += deltaLambda; // Compute the impulse P=J^T * lambda for the 2 translation constraints @@ -235,17 +265,8 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint // Compute J*v for the 3 rotation constraints const Vector3 JvRotation = w2 - w1; - // Compute the bias "b" of the rotation constraint - Vector3 bRotation(0, 0, 0); - if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { - Quaternion q1 = mBody1->getTransform().getOrientation(); - Quaternion q2 = mBody2->getTransform().getOrientation(); - Quaternion qDiff = q1 * q2.getInverse(); - bRotation = 2.0 * qDiff.getVectorV(); - } - // Compute the Lagrange multiplier lambda for the 3 rotation constraints - Vector3 deltaLambda2 = mInverseMassMatrixRotationConstraint * (-JvRotation - bRotation); + Vector3 deltaLambda2 = mInverseMassMatrixRotationConstraint * (-JvRotation - mBRotation); mImpulseRotation += deltaLambda2; // Compute the impulse P=J^T * lambda for the 3 rotation constraints @@ -259,6 +280,41 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint if (mBody2->getIsMotionEnabled()) { w2 += I2 * angularImpulseBody2; } + + // --------------- Limits Constraints --------------- // + + if (mIsLimitsActive) { + + // If the lower limit is violated + if (mIsLowerLimitViolated) { + + // Compute J*v for the lower limit constraint + const decimal JvLowerLimit = mSliderAxisWorld.dot(v2) + mR2CrossSliderAxis.dot(w2) - + mSliderAxisWorld.dot(v1) - mR1PlusUCrossSliderAxis.dot(w1); + + // Compute the Lagrange multiplier lambda for the lower limit constraint + decimal deltaLambdaLower = mInverseMassMatrixLowerLimit * (-JvLowerLimit -mBLowerLimit); + decimal lambdaTemp = mImpulseLowerLimit; + mImpulseLowerLimit = std::max(mImpulseLowerLimit + deltaLambdaLower, decimal(0.0)); + deltaLambdaLower = mImpulseLowerLimit - lambdaTemp; + + // Compute the impulse P=J^T * lambda for the lower limit constraint + Vector3 linearImpulseBody1 = -deltaLambdaLower * mSliderAxisWorld; + Vector3 angularImpulseBody1 = -deltaLambdaLower * mR1PlusUCrossSliderAxis; + Vector3 linearImpulseBody2 = -linearImpulseBody1; + Vector3 angularImpulseBody2 = deltaLambdaLower * mR2CrossSliderAxis; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += I2 * angularImpulseBody2; + } + } + } } // Solve the position constraint diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h index 44742a15..1f9bef7c 100644 --- a/src/constraint/SliderJoint.h +++ b/src/constraint/SliderJoint.h @@ -49,13 +49,34 @@ struct SliderJointInfo : public ConstraintInfo { /// Slider axis (in world-space coordinates) Vector3 sliderAxisWorldSpace; - /// Constructor + /// True if the slider limits are active + bool isLimitsActive; + + /// Lower limit + decimal lowerLimit; + + /// Upper limit + decimal upperLimit; + + /// Constructor without limits SliderJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace, const Vector3& initSliderAxisWorldSpace) : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), - sliderAxisWorldSpace(initSliderAxisWorldSpace) {} + sliderAxisWorldSpace(initSliderAxisWorldSpace), + isLimitsActive(false), lowerLimit(-1.0), upperLimit(1.0) {} + + /// Constructor with limits + SliderJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, + const Vector3& initAnchorPointWorldSpace, + const Vector3& initSliderAxisWorldSpace, + decimal initLowerLimit, decimal initUpperLimit) + : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), + anchorPointWorldSpace(initAnchorPointWorldSpace), + sliderAxisWorldSpace(initSliderAxisWorldSpace), + isLimitsActive(true), lowerLimit(initLowerLimit), + upperLimit(initUpperLimit) {} }; // Class SliderJoint @@ -77,6 +98,9 @@ class SliderJoint : public Constraint { /// Slider axis (in local-space coordinates of body 1) Vector3 mSliderAxisBody1; + /// Initial orientation difference between the two bodies + Quaternion mInitOrientationDifference; + /// First vector orthogonal to the slider axis local-space of body 1 Vector3 mN1; @@ -95,24 +119,63 @@ class SliderJoint : public Constraint { /// Cross product of r2 and n2 Vector3 mR2CrossN2; + /// Cross product of r2 and the slider axis + Vector3 mR2CrossSliderAxis; + /// Cross product of vector (r1 + u) and n1 Vector3 mR1PlusUCrossN1; /// Cross product of vector (r1 + u) and n2 Vector3 mR1PlusUCrossN2; + /// Cross product of vector (r1 + u) and the slider axis + Vector3 mR1PlusUCrossSliderAxis; + + /// Bias of the 2 translation constraints + Vector2 mBTranslation; + + /// Bias of the 3 rotation constraints + Vector3 mBRotation; + + /// Bias of the lower limit constraint + decimal mBLowerLimit; + /// Inverse of mass matrix K=JM^-1J^t for the translation constraint (2x2 matrix) Matrix2x2 mInverseMassMatrixTranslationConstraint; /// Inverse of mass matrix K=JM^-1J^t for the rotation constraint (3x3 matrix) Matrix3x3 mInverseMassMatrixRotationConstraint; + /// Inverse of mass matrix K=JM^-1J^t for the lower limit constraint (1x1 matrix) + decimal mInverseMassMatrixLowerLimit; + /// Impulse for the 2 translation constraints Vector2 mImpulseTranslation; /// Impulse for the 3 rotation constraints Vector3 mImpulseRotation; + /// Impulse for the lower limit constraint + decimal mImpulseLowerLimit; + + /// True if the slider limits are active + bool mIsLimitsActive; + + /// Slider axis in world-space coordinates + Vector3 mSliderAxisWorld; + + /// Lower limit + decimal mLowerLimit; + + /// Upper limit + decimal mUpperLimit; + + /// True if the lower limit is violated + bool mIsLowerLimitViolated; + + /// True if the upper limit is violated + bool mIsUpperLimitViolated; + public : // -------------------- Methods -------------------- // From 61562b3560b8795f37e1cef488432d3a4442c37c Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 23 May 2013 22:52:08 +0200 Subject: [PATCH 29/66] Implement the upper limit for the slider joint --- src/constraint/SliderJoint.cpp | 60 ++++++++++++++++++++++++++++------ src/constraint/SliderJoint.h | 10 ++++-- 2 files changed, 58 insertions(+), 12 deletions(-) diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index a9a322a2..74003790 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -31,8 +31,9 @@ using namespace reactphysics3d; // Constructor SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) : Constraint(jointInfo), mImpulseTranslation(0, 0), mImpulseRotation(0, 0, 0), - mImpulseLowerLimit(0), mIsLimitsActive(jointInfo.isLimitsActive), - mLowerLimit(jointInfo.lowerLimit), mUpperLimit(jointInfo.upperLimit) { + mImpulseLowerLimit(0), mImpulseUpperLimit(0), + mIsLimitsActive(jointInfo.isLimitsActive), mLowerLimit(jointInfo.lowerLimit), + mUpperLimit(jointInfo.upperLimit) { assert(mUpperLimit >= 0.0); assert(mLowerLimit <= 0.0); @@ -83,15 +84,18 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa // Compute the vector u const Vector3 u = x2 + mR2 - x1 - mR1; - // Compute the two orthogonal vectors to the slider axis in local-space of body 1 + // Compute the two orthogonal vectors to the slider axis in world-space mSliderAxisWorld = orientationBody1 * mSliderAxisBody1; mSliderAxisWorld.normalize(); mN1 = mSliderAxisWorld.getOneUnitOrthogonalVector(); mN2 = mSliderAxisWorld.cross(mN1); // Check if the limit constraints are violated or not - decimal lowerLimitError = u.dot(mSliderAxisWorld) - mLowerLimit; + decimal uDotSliderAxis = u.dot(mSliderAxisWorld); + decimal lowerLimitError = uDotSliderAxis - mLowerLimit; + decimal upperLimitError = mUpperLimit - uDotSliderAxis; mIsLowerLimitViolated = (lowerLimitError <= 0); + mIsUpperLimitViolated = (upperLimitError <= 0); // Compute the cross products used in the Jacobians mR2CrossN1 = mR2.cross(mN1); @@ -167,7 +171,7 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa } // Compute the inverse of the mass matrix K=JM^-1J^t for the lower limit (1x1 matrix) - mInverseMassMatrixLowerLimit = sumInverseMass + + mInverseMassMatrixLimit = sumInverseMass + mR1PlusUCrossSliderAxis.dot(I1 * mR1PlusUCrossSliderAxis) + mR2CrossSliderAxis.dot(I2 * mR2CrossSliderAxis); @@ -176,6 +180,12 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { mBLowerLimit = biasFactor * lowerLimitError; } + + // Compute the bias "b" of the upper limit constraint + mBUpperLimit = 0.0; + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBUpperLimit = biasFactor * upperLimitError; + } } // Warm start the constraint (apply the previous impulse at the beginning of the step) @@ -293,16 +303,46 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint mSliderAxisWorld.dot(v1) - mR1PlusUCrossSliderAxis.dot(w1); // Compute the Lagrange multiplier lambda for the lower limit constraint - decimal deltaLambdaLower = mInverseMassMatrixLowerLimit * (-JvLowerLimit -mBLowerLimit); + decimal deltaLambdaLower = mInverseMassMatrixLimit * (-JvLowerLimit -mBLowerLimit); decimal lambdaTemp = mImpulseLowerLimit; mImpulseLowerLimit = std::max(mImpulseLowerLimit + deltaLambdaLower, decimal(0.0)); deltaLambdaLower = mImpulseLowerLimit - lambdaTemp; // Compute the impulse P=J^T * lambda for the lower limit constraint - Vector3 linearImpulseBody1 = -deltaLambdaLower * mSliderAxisWorld; - Vector3 angularImpulseBody1 = -deltaLambdaLower * mR1PlusUCrossSliderAxis; - Vector3 linearImpulseBody2 = -linearImpulseBody1; - Vector3 angularImpulseBody2 = deltaLambdaLower * mR2CrossSliderAxis; + const Vector3 linearImpulseBody1 = -deltaLambdaLower * mSliderAxisWorld; + const Vector3 angularImpulseBody1 = -deltaLambdaLower * mR1PlusUCrossSliderAxis; + const Vector3 linearImpulseBody2 = -linearImpulseBody1; + const Vector3 angularImpulseBody2 = deltaLambdaLower * mR2CrossSliderAxis; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += I2 * angularImpulseBody2; + } + } + + // If the upper limit is violated + if (mIsUpperLimitViolated) { + + // Compute J*v for the upper limit constraint + const decimal JvUpperLimit = mSliderAxisWorld.dot(v1) + mR1PlusUCrossSliderAxis.dot(w1) + - mSliderAxisWorld.dot(v2) - mR2CrossSliderAxis.dot(w2); + + // Compute the Lagrange multiplier lambda for the upper limit constraint + decimal deltaLambdaUpper = mInverseMassMatrixLimit * (-JvUpperLimit -mBUpperLimit); + decimal lambdaTemp = mImpulseUpperLimit; + mImpulseUpperLimit = std::max(mImpulseUpperLimit + deltaLambdaUpper, decimal(0.0)); + deltaLambdaUpper = mImpulseUpperLimit - lambdaTemp; + + // Compute the impulse P=J^T * lambda for the upper limit constraint + const Vector3 linearImpulseBody1 = deltaLambdaUpper * mSliderAxisWorld; + const Vector3 angularImpulseBody1 = deltaLambdaUpper * mR1PlusUCrossSliderAxis; + const Vector3 linearImpulseBody2 = -linearImpulseBody1; + const Vector3 angularImpulseBody2 = -deltaLambdaUpper * mR2CrossSliderAxis; // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h index 1f9bef7c..d0429d25 100644 --- a/src/constraint/SliderJoint.h +++ b/src/constraint/SliderJoint.h @@ -140,14 +140,17 @@ class SliderJoint : public Constraint { /// Bias of the lower limit constraint decimal mBLowerLimit; + /// Bias of the upper limit constraint + decimal mBUpperLimit; + /// Inverse of mass matrix K=JM^-1J^t for the translation constraint (2x2 matrix) Matrix2x2 mInverseMassMatrixTranslationConstraint; /// Inverse of mass matrix K=JM^-1J^t for the rotation constraint (3x3 matrix) Matrix3x3 mInverseMassMatrixRotationConstraint; - /// Inverse of mass matrix K=JM^-1J^t for the lower limit constraint (1x1 matrix) - decimal mInverseMassMatrixLowerLimit; + /// Inverse of mass matrix K=JM^-1J^t for the upper and lower limit constraints (1x1 matrix) + decimal mInverseMassMatrixLimit; /// Impulse for the 2 translation constraints Vector2 mImpulseTranslation; @@ -158,6 +161,9 @@ class SliderJoint : public Constraint { /// Impulse for the lower limit constraint decimal mImpulseLowerLimit; + /// Impulse for the upper limit constraint + decimal mImpulseUpperLimit; + /// True if the slider limits are active bool mIsLimitsActive; From 9c0844cf1b8ec612ae7cf60dc4cc9ca0b7fd1ecf Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 29 May 2013 22:45:02 +0200 Subject: [PATCH 30/66] Finish the implementation of the slider joint --- src/constraint/BallAndSocketJoint.cpp | 37 +++-- src/constraint/BallAndSocketJoint.h | 8 + src/constraint/Constraint.h | 4 +- src/constraint/SliderJoint.cpp | 210 +++++++++++++++++++++++--- src/constraint/SliderJoint.h | 140 +++++++++++++++-- 5 files changed, 352 insertions(+), 47 deletions(-) diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index aa3409f2..edec02b1 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -29,8 +29,11 @@ using namespace reactphysics3d; +// Static variables definition +const decimal BallAndSocketJoint::BETA = 0.2; + // Constructor -BallAndSocketJoint::BallAndSocketJoint(const BallAndSocketJointInfo &jointInfo) +BallAndSocketJoint::BallAndSocketJoint(const BallAndSocketJointInfo& jointInfo) : Constraint(jointInfo), mImpulse(Vector3(0, 0, 0)) { // Compute the local-space anchor point for each body @@ -51,6 +54,8 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS mIndexBody2 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody2)->second; // Get the bodies positions and orientations + const Vector3& x1 = mBody1->getTransform().getPosition(); + const Vector3& x2 = mBody2->getTransform().getPosition(); const Quaternion& orientationBody1 = mBody1->getTransform().getOrientation(); const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); @@ -89,6 +94,20 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { mInverseMassMatrix = massMatrix.getInverse(); } + + // Compute the bias "b" of the constraint + mBiasVector.setToZero(); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + decimal biasFactor = (BETA / constraintSolverData.timeStep); + mBiasVector = biasFactor * (x2 + mR2World - x1 - mR1World); + } + + // If warm-starting is not enabled + if (!constraintSolverData.isWarmStartingActive) { + + // Reset the accumulated impulse + mImpulse.setToZero(); + } } // Warm start the constraint (apply the previous impulse at the beginning of the step) @@ -126,10 +145,6 @@ void BallAndSocketJoint::warmstart(const ConstraintSolverData& constraintSolverD // Solve the velocity constraint void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) { - // Get the body positions - const Vector3& x1 = mBody1->getTransform().getPosition(); - const Vector3& x2 = mBody2->getTransform().getPosition(); - // Get the velocities Vector3& v1 = constraintSolverData.linearVelocities[mIndexBody1]; Vector3& v2 = constraintSolverData.linearVelocities[mIndexBody2]; @@ -143,18 +158,10 @@ void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& con Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // Compute J*v - const Vector3 Jv = -v1 + mR1World.cross(w1) + v2 - mR2World.cross(w2); - - // Compute the bias "b" of the constraint - Vector3 b(0, 0, 0); - if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { - decimal beta = decimal(0.2); // TODO : Use a constant here - decimal biasFactor = (beta / constraintSolverData.timeStep); - b = biasFactor * (x2 + mR2World - x1 - mR1World); - } + const Vector3 Jv = v2 + w2.cross(mR2World) - v1 - w1.cross(mR1World); // Compute the Lagrange multiplier lambda - const Vector3 deltaLambda = mInverseMassMatrix * (-Jv - b); + const Vector3 deltaLambda = mInverseMassMatrix * (-Jv - mBiasVector); mImpulse += deltaLambda; // Compute the impulse P=J^T * lambda diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index 556e316b..dff9210d 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -62,6 +62,11 @@ class BallAndSocketJoint : public Constraint { private : + // -------------------- Constants -------------------- // + + // Beta value for the bias factor of position correction + static const decimal BETA; + // -------------------- Attributes -------------------- // /// Anchor point of body 1 (in local-space coordinates of body 1) @@ -76,6 +81,9 @@ class BallAndSocketJoint : public Constraint { /// Vector from center of body 2 to anchor point in world-space Vector3 mR2World; + /// Bias vector for the constraint + Vector3 mBiasVector; + /// Skew-Symmetric matrix for cross product with vector mR1World Matrix3x3 mSkewSymmetricMatrixR1World; diff --git a/src/constraint/Constraint.h b/src/constraint/Constraint.h index 1be00170..97b70150 100644 --- a/src/constraint/Constraint.h +++ b/src/constraint/Constraint.h @@ -33,9 +33,9 @@ // ReactPhysics3D namespace namespace reactphysics3d { - + // Enumeration for the type of a constraint -enum ConstraintType {CONTACT, BALLSOCKETJOINT, SLIDERJOINT}; +enum ConstraintType {CONTACT, BALLSOCKETJOINT, SLIDERJOINT, HINGEJOINT}; // Class declarations struct ConstraintSolverData; diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index 74003790..7825a1fc 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -28,15 +28,21 @@ using namespace reactphysics3d; +// Static variables definition +const decimal SliderJoint::BETA = decimal(0.2); + // Constructor SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) : Constraint(jointInfo), mImpulseTranslation(0, 0), mImpulseRotation(0, 0, 0), - mImpulseLowerLimit(0), mImpulseUpperLimit(0), - mIsLimitsActive(jointInfo.isLimitsActive), mLowerLimit(jointInfo.lowerLimit), - mUpperLimit(jointInfo.upperLimit) { + mImpulseLowerLimit(0), mImpulseUpperLimit(0), mImpulseMotor(0), + mIsLimitEnabled(jointInfo.isLimitEnabled), mIsMotorEnabled(jointInfo.isMotorEnabled), + mLowerLimit(jointInfo.lowerLimit), mUpperLimit(jointInfo.upperLimit), + mIsLowerLimitViolated(false), mIsUpperLimitViolated(false), + mMotorSpeed(jointInfo.motorSpeed), mMaxMotorForce(jointInfo.maxMotorForce){ assert(mUpperLimit >= 0.0); assert(mLowerLimit <= 0.0); + assert(mMaxMotorForce >= 0.0); // Compute the local-space anchor point for each body const Transform& transform1 = mBody1->getTransform(); @@ -63,7 +69,7 @@ SliderJoint::~SliderJoint() { // Initialize before solving the constraint void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverData) { - // Initialize the bodies index in the velocity array + // Initialize the bodies index in the veloc ity array mIndexBody1 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody1)->second; mIndexBody2 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody2)->second; @@ -94,8 +100,16 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa decimal uDotSliderAxis = u.dot(mSliderAxisWorld); decimal lowerLimitError = uDotSliderAxis - mLowerLimit; decimal upperLimitError = mUpperLimit - uDotSliderAxis; - mIsLowerLimitViolated = (lowerLimitError <= 0); - mIsUpperLimitViolated = (upperLimitError <= 0); + bool oldIsLowerLimitViolated = mIsLowerLimitViolated; + mIsLowerLimitViolated = lowerLimitError <= 0; + if (mIsLowerLimitViolated != oldIsLowerLimitViolated) { + mImpulseLowerLimit = 0.0; + } + bool oldIsUpperLimitViolated = mIsUpperLimitViolated; + mIsUpperLimitViolated = upperLimitError <= 0; + if (mIsUpperLimitViolated != oldIsUpperLimitViolated) { + mImpulseUpperLimit = 0.0; + } // Compute the cross products used in the Jacobians mR2CrossN1 = mR2.cross(mN1); @@ -139,8 +153,7 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa // Compute the bias "b" of the translation constraint mBTranslation.setToZero(); - decimal beta = decimal(0.2); // TODO : Use a constant here - decimal biasFactor = (beta / constraintSolverData.timeStep); + decimal biasFactor = (BETA / constraintSolverData.timeStep); if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { mBTranslation.x = u.dot(mN1); mBTranslation.y = u.dot(mN2); @@ -170,21 +183,54 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa mBRotation = biasFactor * decimal(2.0) * qError.getVectorV(); } - // Compute the inverse of the mass matrix K=JM^-1J^t for the lower limit (1x1 matrix) - mInverseMassMatrixLimit = sumInverseMass + - mR1PlusUCrossSliderAxis.dot(I1 * mR1PlusUCrossSliderAxis) + - mR2CrossSliderAxis.dot(I2 * mR2CrossSliderAxis); + if (mIsLimitEnabled && (mIsLowerLimitViolated || mIsUpperLimitViolated)) { - // Compute the bias "b" of the lower limit constraint - mBLowerLimit = 0.0; - if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { - mBLowerLimit = biasFactor * lowerLimitError; + // Compute the inverse of the mass matrix K=JM^-1J^t for the lower limit (1x1 matrix) + mInverseMassMatrixLimit = 0.0; + if (mBody1->getIsMotionEnabled()) { + mInverseMassMatrixLimit += mBody1->getMassInverse() + + mR1PlusUCrossSliderAxis.dot(I1 * mR1PlusUCrossSliderAxis); + } + if (mBody2->getIsMotionEnabled()) { + mInverseMassMatrixLimit += mBody2->getMassInverse() + + mR2CrossSliderAxis.dot(I2 * mR2CrossSliderAxis); + } + mInverseMassMatrixLimit = (mInverseMassMatrixLimit > 0.0) ? + decimal(1.0) / mInverseMassMatrixLimit : decimal(0.0); + + // Compute the bias "b" of the lower limit constraint + mBLowerLimit = 0.0; + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBLowerLimit = biasFactor * lowerLimitError; + } + + // Compute the bias "b" of the upper limit constraint + mBUpperLimit = 0.0; + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBUpperLimit = biasFactor * upperLimitError; + } } - // Compute the bias "b" of the upper limit constraint - mBUpperLimit = 0.0; - if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { - mBUpperLimit = biasFactor * upperLimitError; + // Compute the inverse of mass matrix K=JM^-1J^t for the motor (1x1 matrix) + mInverseMassMatrixMotor = 0.0; + if (mBody1->getIsMotionEnabled()) { + mInverseMassMatrixMotor += mBody1->getMassInverse(); + } + if (mBody2->getIsMotionEnabled()) { + mInverseMassMatrixMotor += mBody2->getMassInverse(); + } + mInverseMassMatrixMotor = (mInverseMassMatrixMotor > 0.0) ? + decimal(1.0) / mInverseMassMatrixMotor : decimal(0.0); + + // If warm-starting is not enabled + if (!constraintSolverData.isWarmStartingActive) { + + // Reset all the accumulated impulses + mImpulseTranslation.setToZero(); + mImpulseRotation.setToZero(); + mImpulseLowerLimit = 0.0; + mImpulseUpperLimit = 0.0; + mImpulseMotor = 0.0; } } @@ -215,6 +261,19 @@ void SliderJoint::warmstart(const ConstraintSolverData& constraintSolverData) { angularImpulseBody1 += -mImpulseRotation; angularImpulseBody2 += mImpulseRotation; + // Compute the impulse P=J^T * lambda for the lower and upper limits constraints + decimal impulseLimits = mImpulseUpperLimit - mImpulseLowerLimit; + Vector3 linearImpulseLimits = impulseLimits * mSliderAxisWorld; + linearImpulseBody1 += linearImpulseLimits; + angularImpulseBody1 += impulseLimits * mR1PlusUCrossSliderAxis; + linearImpulseBody2 += -linearImpulseLimits; + angularImpulseBody2 += -impulseLimits * mR2CrossSliderAxis; + + // Compute the impulse P=J^T * lambda for the motor constraint + Vector3 impulseMotor = mImpulseMotor * mSliderAxisWorld; + linearImpulseBody1 += impulseMotor; + linearImpulseBody2 += -impulseMotor; + // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { v1 += inverseMassBody1 * linearImpulseBody1; @@ -293,7 +352,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint // --------------- Limits Constraints --------------- // - if (mIsLimitsActive) { + if (mIsLimitEnabled) { // If the lower limit is violated if (mIsLowerLimitViolated) { @@ -355,9 +414,118 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint } } } + + // --------------- Motor --------------- // + + if (mIsMotorEnabled) { + + // Compute J*v for the motor + const decimal JvMotor = mSliderAxisWorld.dot(v1) - mSliderAxisWorld.dot(v2); + + // Compute the Lagrange multiplier lambda for the motor + const decimal maxMotorImpulse = mMaxMotorForce * constraintSolverData.timeStep; + decimal deltaLambdaMotor = mInverseMassMatrixMotor * (-JvMotor -mMotorSpeed); + decimal lambdaTemp = mImpulseMotor; + mImpulseMotor = clamp(mImpulseMotor + deltaLambdaMotor, -maxMotorImpulse, maxMotorImpulse); + deltaLambdaMotor = mImpulseMotor - lambdaTemp; + + // Compute the impulse P=J^T * lambda for the motor + const Vector3 linearImpulseBody1 = deltaLambdaMotor * mSliderAxisWorld; + const Vector3 linearImpulseBody2 = -linearImpulseBody1; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + } + } } // Solve the position constraint void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { } + +// Enable/Disable the limits of the joint +void SliderJoint::enableLimit(bool isLimitEnabled) { + + if (isLimitEnabled != mIsLimitEnabled) { + + mIsLimitEnabled = isLimitEnabled; + + // Reset the limits + resetLimits(); + } +} + +// Enable/Disable the motor of the joint +void SliderJoint::enableMotor(bool isMotorEnabled) { + + mIsMotorEnabled = isMotorEnabled; + mImpulseMotor = 0.0; + + // TODO : Wake up the bodies of the joint here when sleeping is implemented +} + +// Set the lower limit +void SliderJoint::setLowerLimit(decimal lowerLimit) { + + assert(lowerLimit <= mUpperLimit); + + if (lowerLimit != mLowerLimit) { + + mLowerLimit = lowerLimit; + + // Reset the limits + resetLimits(); + } +} + +// Set the upper limit +void SliderJoint::setUpperLimit(decimal upperLimit) { + + assert(mLowerLimit <= upperLimit); + + if (upperLimit != mUpperLimit) { + + mUpperLimit = upperLimit; + + // Reset the limits + resetLimits(); + } +} + +// Reset the limits +void SliderJoint::resetLimits() { + + // Reset the accumulated impulses for the limits + mImpulseLowerLimit = 0.0; + mImpulseUpperLimit = 0.0; + + // TODO : Wake up the bodies of the joint here when sleeping is implemented +} + +// Set the motor speed +void SliderJoint::setMotorSpeed(decimal motorSpeed) { + + if (motorSpeed != mMotorSpeed) { + + mMotorSpeed = motorSpeed; + + // TODO : Wake up the bodies of the joint here when sleeping is implemented + } +} + +// Set the maximum motor force +void SliderJoint::setMaxMotorForce(decimal maxMotorForce) { + + if (maxMotorForce != mMaxMotorForce) { + + assert(mMaxMotorForce >= 0.0); + mMaxMotorForce = maxMotorForce; + + // TODO : Wake up the bodies of the joint here when sleeping is implemented + } +} diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h index d0429d25..fd54b5cb 100644 --- a/src/constraint/SliderJoint.h +++ b/src/constraint/SliderJoint.h @@ -49,8 +49,11 @@ struct SliderJointInfo : public ConstraintInfo { /// Slider axis (in world-space coordinates) Vector3 sliderAxisWorldSpace; - /// True if the slider limits are active - bool isLimitsActive; + /// True if the slider limits are enabled + bool isLimitEnabled; + + /// True if the slider motor is enabled + bool isMotorEnabled; /// Lower limit decimal lowerLimit; @@ -58,16 +61,23 @@ struct SliderJointInfo : public ConstraintInfo { /// Upper limit decimal upperLimit; - /// Constructor without limits + /// Motor speed + decimal motorSpeed; + + /// Maximum motor force (in Newton) that can be applied to reach to desired motor speed + decimal maxMotorForce; + + /// Constructor without limits and without motor SliderJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace, const Vector3& initSliderAxisWorldSpace) : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), sliderAxisWorldSpace(initSliderAxisWorldSpace), - isLimitsActive(false), lowerLimit(-1.0), upperLimit(1.0) {} + isLimitEnabled(false), isMotorEnabled(false), lowerLimit(-1.0), + upperLimit(1.0), motorSpeed(0), maxMotorForce(0) {} - /// Constructor with limits + /// Constructor with limits and no motor SliderJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace, const Vector3& initSliderAxisWorldSpace, @@ -75,8 +85,21 @@ struct SliderJointInfo : public ConstraintInfo { : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), sliderAxisWorldSpace(initSliderAxisWorldSpace), - isLimitsActive(true), lowerLimit(initLowerLimit), - upperLimit(initUpperLimit) {} + isLimitEnabled(true), isMotorEnabled(false), lowerLimit(initLowerLimit), + upperLimit(initUpperLimit), motorSpeed(0), maxMotorForce(0) {} + + /// Constructor with limits and motor + SliderJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, + const Vector3& initAnchorPointWorldSpace, + const Vector3& initSliderAxisWorldSpace, + decimal initLowerLimit, decimal initUpperLimit, + decimal initMotorSpeed, decimal initMaxMotorForce) + : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), + anchorPointWorldSpace(initAnchorPointWorldSpace), + sliderAxisWorldSpace(initSliderAxisWorldSpace), + isLimitEnabled(true), isMotorEnabled(true), lowerLimit(initLowerLimit), + upperLimit(initUpperLimit), motorSpeed(initMotorSpeed), + maxMotorForce(initMaxMotorForce) {} }; // Class SliderJoint @@ -87,6 +110,11 @@ class SliderJoint : public Constraint { private : + // -------------------- Constants -------------------- // + + // Beta value for the position correction bias factor + static const decimal BETA; + // -------------------- Attributes -------------------- // /// Anchor point of body 1 (in local-space coordinates of body 1) @@ -152,6 +180,9 @@ class SliderJoint : public Constraint { /// Inverse of mass matrix K=JM^-1J^t for the upper and lower limit constraints (1x1 matrix) decimal mInverseMassMatrixLimit; + /// Inverse of mass matrix K=JM^-1J^t for the motor + decimal mInverseMassMatrixMotor; + /// Impulse for the 2 translation constraints Vector2 mImpulseTranslation; @@ -164,8 +195,14 @@ class SliderJoint : public Constraint { /// Impulse for the upper limit constraint decimal mImpulseUpperLimit; - /// True if the slider limits are active - bool mIsLimitsActive; + /// Impulse for the motor + decimal mImpulseMotor; + + /// True if the slider limits are enabled + bool mIsLimitEnabled; + + /// True if the motor of the joint in enabled + bool mIsMotorEnabled; /// Slider axis in world-space coordinates Vector3 mSliderAxisWorld; @@ -182,6 +219,17 @@ class SliderJoint : public Constraint { /// True if the upper limit is violated bool mIsUpperLimitViolated; + /// Motor speed + decimal mMotorSpeed; + + /// Maximum motor force (in Newton) that can be applied to reach to desired motor speed + decimal mMaxMotorForce; + + // -------------------- Methods -------------------- // + + /// Reset the limits + void resetLimits(); + public : // -------------------- Methods -------------------- // @@ -192,6 +240,45 @@ class SliderJoint : public Constraint { /// Destructor virtual ~SliderJoint(); + /// Return true if the limits or the joint are enabled + bool isLimitEnabled() const; + + /// Return true if the motor of the joint is enabled + bool isMotorEnabled() const; + + /// Enable/Disable the limits of the joint + void enableLimit(bool isLimitEnabled); + + /// Enable/Disable the motor of the joint + void enableMotor(bool isMotorEnabled); + + /// Return the lower limit + decimal getLowerLimit() const; + + /// Set the lower limit + void setLowerLimit(decimal lowerLimit); + + /// Return the upper limit + decimal getUpperLimit() const; + + /// Set the upper limit + void setUpperLimit(decimal upperLimit); + + /// Return the motor speed + decimal getMotorSpeed() const; + + /// Set the motor speed + void setMotorSpeed(decimal motorSpeed); + + /// Return the maximum motor force + decimal getMaxMotorForce() const; + + /// Set the maximum motor force + void setMaxMotorForce(decimal maxMotorForce); + + /// Return the intensity of the current force applied for the joint motor + decimal getMotorForce(decimal timeStep) const; + /// Return the number of bytes used by the joint virtual size_t getSizeInBytes() const; @@ -208,6 +295,41 @@ class SliderJoint : public Constraint { virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); }; +// Return true if the limits or the joint are enabled +inline bool SliderJoint::isLimitEnabled() const { + return mIsLimitEnabled; +} + +// Return true if the motor of the joint is enabled +inline bool SliderJoint::isMotorEnabled() const { + return mIsMotorEnabled; +} + +// Return the lower limit +inline decimal SliderJoint::getLowerLimit() const { + return mLowerLimit; +} + +// Return the upper limit +inline decimal SliderJoint::getUpperLimit() const { + return mUpperLimit; +} + +// Return the motor speed +inline decimal SliderJoint::getMotorSpeed() const { + return mMotorSpeed; +} + +// Return the maximum motor force +inline decimal SliderJoint::getMaxMotorForce() const { + return mMaxMotorForce; +} + +// Return the intensity of the current force applied for the joint motor +inline decimal SliderJoint::getMotorForce(decimal timeStep) const { + return mImpulseMotor / timeStep; +} + // Return the number of bytes used by the joint inline size_t SliderJoint::getSizeInBytes() const { return sizeof(SliderJoint); From c4d6206ee240c69fd0991013e43381c916ab1648 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sun, 9 Jun 2013 16:31:01 +0200 Subject: [PATCH 31/66] Finish the implementation of the Hinge joint and some others modifications --- .../narrowphase/EPA/EPAAlgorithm.cpp | 2 +- src/configuration.h | 3 + src/constraint/BallAndSocketJoint.cpp | 4 +- src/constraint/HingeJoint.cpp | 591 ++++++++++++++++++ src/constraint/HingeJoint.h | 342 ++++++++++ src/constraint/SliderJoint.cpp | 28 +- src/constraint/SliderJoint.h | 69 +- src/engine/ConstraintSolver.cpp | 7 +- src/engine/DynamicsWorld.cpp | 12 +- src/mathematics/Quaternion.h | 46 +- src/mathematics/mathematics_functions.h | 8 + src/reactphysics3d.h | 1 + test/tests/mathematics/TestQuaternion.h | 15 +- 13 files changed, 1070 insertions(+), 58 deletions(-) create mode 100644 src/constraint/HingeJoint.cpp create mode 100644 src/constraint/HingeJoint.h diff --git a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp index b6c70ba5..2cdfc87b 100644 --- a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp +++ b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp @@ -143,7 +143,7 @@ bool EPAAlgorithm::computePenetrationDepthAndContactPoints(const Simplex& simple int minAxis = d.getAbsoluteVector().getMinAxis(); // Compute sin(60) - const decimal sin60 = sqrt(3.0) * 0.5; + const decimal sin60 = sqrt(3.0) * decimal(0.5); // Create a rotation quaternion to rotate the vector v1 to get the vectors // v2 and v3 diff --git a/src/configuration.h b/src/configuration.h index eae264f9..83e3c817 100644 --- a/src/configuration.h +++ b/src/configuration.h @@ -81,6 +81,9 @@ const decimal MACHINE_EPSILON = std::numeric_limits::epsilon(); /// Pi constant const decimal PI = decimal(3.14159265); +/// 2*Pi constant +const decimal PI_TIMES_2 = decimal(6.28318530); + /// Default internal constant timestep in seconds const decimal DEFAULT_TIMESTEP = decimal(1.0 / 60.0); diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index edec02b1..4a8db2ce 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -27,10 +27,12 @@ #include "BallAndSocketJoint.h" #include "../engine/ConstraintSolver.h" +// TODO : Solve 2x2 or 3x3 linear systems without inverting the A matrix (direct resolution) + using namespace reactphysics3d; // Static variables definition -const decimal BallAndSocketJoint::BETA = 0.2; +const decimal BallAndSocketJoint::BETA = decimal(0.2); // Constructor BallAndSocketJoint::BallAndSocketJoint(const BallAndSocketJointInfo& jointInfo) diff --git a/src/constraint/HingeJoint.cpp b/src/constraint/HingeJoint.cpp new file mode 100644 index 00000000..aa0da0eb --- /dev/null +++ b/src/constraint/HingeJoint.cpp @@ -0,0 +1,591 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "HingeJoint.h" +#include "../engine/ConstraintSolver.h" +#include + +// TODO : Solve 2x2 or 3x3 linear systems without inverting the A matrix (direct resolution) + +using namespace reactphysics3d; + +// Static variables definition +const decimal HingeJoint::BETA = decimal(0.2); + +// Constructor +HingeJoint::HingeJoint(const HingeJointInfo& jointInfo) + : Constraint(jointInfo), mImpulseTranslation(0, 0, 0), mImpulseRotation(0, 0), + mImpulseLowerLimit(0), mImpulseUpperLimit(0), mImpulseMotor(0), + mIsLimitEnabled(jointInfo.isLimitEnabled), mIsMotorEnabled(jointInfo.isMotorEnabled), + mLowerLimit(jointInfo.minAngleLimit), mUpperLimit(jointInfo.maxAngleLimit), + mIsLowerLimitViolated(false), mIsUpperLimitViolated(false), + mMotorSpeed(jointInfo.motorSpeed), mMaxMotorForce(jointInfo.maxMotorForce) { + + assert(mLowerLimit <= 0 && mLowerLimit >= -2.0 * PI); + assert(mUpperLimit >= 0 && mUpperLimit <= 2.0 * PI); + + // Compute the local-space anchor point for each body + Transform transform1 = mBody1->getTransform(); + Transform transform2 = mBody2->getTransform(); + mLocalAnchorPointBody1 = transform1.getInverse() * jointInfo.anchorPointWorldSpace; + mLocalAnchorPointBody2 = transform2.getInverse() * jointInfo.anchorPointWorldSpace; + + // Compute the local-space hinge axis + mHingeLocalAxisBody1 = transform1.getOrientation().getInverse() * jointInfo.rotationAxisWorld; + mHingeLocalAxisBody2 = transform2.getOrientation().getInverse() * jointInfo.rotationAxisWorld; + mHingeLocalAxisBody1.normalize(); + mHingeLocalAxisBody2.normalize(); + + // Compute the inverse of the initial orientation difference between the two bodies + mInitOrientationDifferenceInv = transform2.getOrientation() * + transform1.getOrientation().getInverse(); + mInitOrientationDifferenceInv.normalize(); + mInitOrientationDifferenceInv.inverse(); +} + +// Destructor +HingeJoint::~HingeJoint() { + +} + +// Initialize before solving the constraint +void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverData) { + + // Initialize the bodies index in the velocity array + mIndexBody1 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody1)->second; + mIndexBody2 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody2)->second; + + // Get the bodies positions and orientations + const Vector3& x1 = mBody1->getTransform().getPosition(); + const Vector3& x2 = mBody2->getTransform().getPosition(); + const Quaternion& orientationBody1 = mBody1->getTransform().getOrientation(); + const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); + + // Get the inertia tensor of bodies + const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the vector from body center to the anchor point in world-space + mR1World = orientationBody1 * mLocalAnchorPointBody1; + mR2World = orientationBody2 * mLocalAnchorPointBody2; + + // Compute the current angle around the hinge axis + decimal hingeAngle = computeCurrentHingeAngle(orientationBody1, orientationBody2); + + // Check if the limit constraints are violated or not + decimal lowerLimitError = hingeAngle - mLowerLimit; + decimal upperLimitError = mUpperLimit - hingeAngle; + bool oldIsLowerLimitViolated = mIsLowerLimitViolated; + mIsLowerLimitViolated = lowerLimitError <= 0; + if (mIsLowerLimitViolated != oldIsLowerLimitViolated) { + mImpulseLowerLimit = 0.0; + } + bool oldIsUpperLimitViolated = mIsUpperLimitViolated; + mIsUpperLimitViolated = upperLimitError <= 0; + if (mIsUpperLimitViolated != oldIsUpperLimitViolated) { + mImpulseUpperLimit = 0.0; + } + + decimal testAngle = computeCurrentHingeAngle(orientationBody1, orientationBody2); + + // Compute vectors needed in the Jacobian + mA1 = orientationBody1 * mHingeLocalAxisBody1; + Vector3 a2 = orientationBody2 * mHingeLocalAxisBody2; + mA1.normalize(); + a2.normalize(); + const Vector3 b2 = a2.getOneUnitOrthogonalVector(); + const Vector3 c2 = a2.cross(b2); + mB2CrossA1 = b2.cross(mA1); + mC2CrossA1 = c2.cross(mA1); + + // Compute the corresponding skew-symmetric matrices + Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR1World); + Matrix3x3 skewSymmetricMatrixU2= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR2World); + + // Compute the inverse mass matrix K=JM^-1J^t for the 3 translation constraints (3x3 matrix) + decimal inverseMassBodies = 0.0; + if (mBody1->getIsMotionEnabled()) { + inverseMassBodies += mBody1->getMassInverse(); + } + if (mBody2->getIsMotionEnabled()) { + inverseMassBodies += mBody2->getMassInverse(); + } + Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, + 0, inverseMassBodies, 0, + 0, 0, inverseMassBodies); + if (mBody1->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU1 * I1 * skewSymmetricMatrixU1.getTranspose(); + } + if (mBody2->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU2 * I2 * skewSymmetricMatrixU2.getTranspose(); + } + mInverseMassMatrixTranslation.setToZero(); + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixTranslation = massMatrix.getInverse(); + } + + // Compute the bias "b" of the translation constraints + mBTranslation.setToZero(); + decimal biasFactor = (BETA / constraintSolverData.timeStep); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBTranslation = biasFactor * (x2 + mR2World - x1 - mR1World); + } + + // Compute the inverse mass matrix K=JM^-1J^t for the 2 rotation constraints (2x2 matrix) + Vector3 I1B2CrossA1(0, 0, 0); + Vector3 I1C2CrossA1(0, 0, 0); + Vector3 I2B2CrossA1(0, 0, 0); + Vector3 I2C2CrossA1(0, 0, 0); + if (mBody1->getIsMotionEnabled()) { + I1B2CrossA1 = I1 * mB2CrossA1; + I1C2CrossA1 = I1 * mC2CrossA1; + } + if (mBody2->getIsMotionEnabled()) { + I2B2CrossA1 = I2 * mB2CrossA1; + I2C2CrossA1 = I2 * mC2CrossA1; + } + const decimal el11 = mB2CrossA1.dot(I1B2CrossA1) + + mB2CrossA1.dot(I2B2CrossA1); + const decimal el12 = mB2CrossA1.dot(I1C2CrossA1) + + mB2CrossA1.dot(I2C2CrossA1); + const decimal el21 = mC2CrossA1.dot(I1B2CrossA1) + + mC2CrossA1.dot(I2B2CrossA1); + const decimal el22 = mC2CrossA1.dot(I1C2CrossA1) + + mC2CrossA1.dot(I2C2CrossA1); + const Matrix2x2 matrixKRotation(el11, el12, el21, el22); + mInverseMassMatrixRotation.setToZero(); + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixRotation = matrixKRotation.getInverse(); + } + + // Compute the bias "b" of the rotation constraints + mBRotation.setToZero(); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBRotation = biasFactor * Vector2(mA1.dot(b2), mA1.dot(c2)); + } + + // If warm-starting is not enabled + if (!constraintSolverData.isWarmStartingActive) { + + // Reset all the accumulated impulses + mImpulseTranslation.setToZero(); + mImpulseRotation.setToZero(); + mImpulseLowerLimit = 0.0; + mImpulseUpperLimit = 0.0; + mImpulseMotor = 0.0; + } + + if (mIsLimitEnabled && (mIsLowerLimitViolated || mIsUpperLimitViolated)) { + + // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) + mInverseMassMatrixLimitMotor = 0.0; + if (mBody1->getIsMotionEnabled()) { + mInverseMassMatrixLimitMotor += mA1.dot(I1 * mA1); + } + if (mBody2->getIsMotionEnabled()) { + mInverseMassMatrixLimitMotor += mA1.dot(I2 * mA1); + } + mInverseMassMatrixLimitMotor = (mInverseMassMatrixLimitMotor > 0.0) ? + decimal(1.0) / mInverseMassMatrixLimitMotor : decimal(0.0); + + // Compute the bias "b" of the lower limit constraint + mBLowerLimit = 0.0; + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBLowerLimit = biasFactor * lowerLimitError; + } + + // Compute the bias "b" of the upper limit constraint + mBUpperLimit = 0.0; + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBUpperLimit = biasFactor * upperLimitError; + } + } +} + +// Warm start the constraint (apply the previous impulse at the beginning of the step) +void HingeJoint::warmstart(const ConstraintSolverData& constraintSolverData) { + + // Get the velocities + Vector3& v1 = constraintSolverData.linearVelocities[mIndexBody1]; + Vector3& v2 = constraintSolverData.linearVelocities[mIndexBody2]; + Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; + Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + const decimal inverseMassBody1 = mBody1->getMassInverse(); + const decimal inverseMassBody2 = mBody2->getMassInverse(); + const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the impulse P=J^T * lambda for the 3 translation constraints + Vector3 linearImpulseBody1 = -mImpulseTranslation; + Vector3 angularImpulseBody1 = mImpulseTranslation.cross(mR1World); + Vector3 linearImpulseBody2 = mImpulseTranslation; + Vector3 angularImpulseBody2 = -mImpulseTranslation.cross(mR2World); + + // Compute the impulse P=J^T * lambda for the 2 rotation constraints + Vector3 rotationImpulse = -mB2CrossA1 * mImpulseRotation.x - mC2CrossA1 * mImpulseRotation.y; + angularImpulseBody1 += rotationImpulse; + angularImpulseBody2 += -rotationImpulse; + + // Compute the impulse P=J^T * lambda for the lower and upper limits constraints + const Vector3 limitsImpulse = (mImpulseUpperLimit - mImpulseLowerLimit) * mA1; + angularImpulseBody1 += limitsImpulse; + angularImpulseBody2 += -limitsImpulse; + + // Compute the impulse P=J^T * lambda for the motor constraint + const Vector3 motorImpulse = -mImpulseMotor * mA1; + angularImpulseBody1 += motorImpulse; + angularImpulseBody2 += -motorImpulse; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += I2 * angularImpulseBody2; + } +} + +// Solve the velocity constraint +void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) { + + // Get the velocities + Vector3& v1 = constraintSolverData.linearVelocities[mIndexBody1]; + Vector3& v2 = constraintSolverData.linearVelocities[mIndexBody2]; + Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; + Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + decimal inverseMassBody1 = mBody1->getMassInverse(); + decimal inverseMassBody2 = mBody2->getMassInverse(); + Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + + // --------------- Translation Constraints --------------- // + + // Compute J*v + const Vector3 JvTranslation = v2 + w2.cross(mR2World) - v1 - w1.cross(mR1World); + + // Compute the Lagrange multiplier lambda + const Vector3 deltaLambdaTranslation = mInverseMassMatrixTranslation * + (-JvTranslation - mBTranslation); + mImpulseTranslation += deltaLambdaTranslation; + + // Compute the impulse P=J^T * lambda + Vector3 linearImpulseBody1 = -deltaLambdaTranslation; + Vector3 angularImpulseBody1 = deltaLambdaTranslation.cross(mR1World); + Vector3 linearImpulseBody2 = deltaLambdaTranslation; + Vector3 angularImpulseBody2 = -deltaLambdaTranslation.cross(mR2World); + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += I2 * angularImpulseBody2; + } + + // --------------- Rotation Constraints --------------- // + + // Compute J*v for the 2 rotation constraints + const Vector2 JvRotation(-mB2CrossA1.dot(w1) + mB2CrossA1.dot(w2), + -mC2CrossA1.dot(w1) + mC2CrossA1.dot(w2)); + + // Compute the Lagrange multiplier lambda for the 2 rotation constraints + Vector2 deltaLambdaRotation = mInverseMassMatrixRotation * (-JvRotation - mBRotation); + mImpulseRotation += deltaLambdaRotation; + + // Compute the impulse P=J^T * lambda for the 2 rotation constraints + angularImpulseBody1 = -mB2CrossA1 * deltaLambdaRotation.x - mC2CrossA1 * deltaLambdaRotation.y; + angularImpulseBody2 = -angularImpulseBody1; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + w2 += I2 * angularImpulseBody2; + } + + // --------------- Limits Constraints --------------- // + + if (mIsLimitEnabled) { + + // If the lower limit is violated + if (mIsLowerLimitViolated) { + + // Compute J*v for the lower limit constraint + const decimal JvLowerLimit = (w2 - w1).dot(mA1); + + // Compute the Lagrange multiplier lambda for the lower limit constraint + decimal deltaLambdaLower = mInverseMassMatrixLimitMotor * (-JvLowerLimit - mBLowerLimit); + decimal lambdaTemp = mImpulseLowerLimit; + mImpulseLowerLimit = std::max(mImpulseLowerLimit + deltaLambdaLower, decimal(0.0)); + deltaLambdaLower = mImpulseLowerLimit - lambdaTemp; + + // Compute the impulse P=J^T * lambda for the lower limit constraint + const Vector3 angularImpulseBody1 = -deltaLambdaLower * mA1; + const Vector3 angularImpulseBody2 = -angularImpulseBody1; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + w2 += I2 * angularImpulseBody2; + } + } + + // If the upper limit is violated + if (mIsUpperLimitViolated) { + + // Compute J*v for the upper limit constraint + const decimal JvUpperLimit = -(w2 - w1).dot(mA1); + + // Compute the Lagrange multiplier lambda for the upper limit constraint + decimal deltaLambdaUpper = mInverseMassMatrixLimitMotor * (-JvUpperLimit -mBUpperLimit); + decimal lambdaTemp = mImpulseUpperLimit; + mImpulseUpperLimit = std::max(mImpulseUpperLimit + deltaLambdaUpper, decimal(0.0)); + deltaLambdaUpper = mImpulseUpperLimit - lambdaTemp; + + // Compute the impulse P=J^T * lambda for the upper limit constraint + const Vector3 angularImpulseBody1 = deltaLambdaUpper * mA1; + const Vector3 angularImpulseBody2 = -angularImpulseBody1; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + w2 += I2 * angularImpulseBody2; + } + } + } + + // --------------- Motor --------------- // + + if (mIsMotorEnabled) { + + // Compute J*v for the motor + const decimal JvMotor = mA1.dot(w1 - w2); + + // Compute the Lagrange multiplier lambda for the motor + const decimal maxMotorImpulse = mMaxMotorForce * constraintSolverData.timeStep; + decimal deltaLambdaMotor = mInverseMassMatrixLimitMotor * (-JvMotor - mMotorSpeed); + decimal lambdaTemp = mImpulseMotor; + mImpulseMotor = clamp(mImpulseMotor + deltaLambdaMotor, -maxMotorImpulse, maxMotorImpulse); + deltaLambdaMotor = mImpulseMotor - lambdaTemp; + + // Compute the impulse P=J^T * lambda for the motor + const Vector3 angularImpulseBody1 = -deltaLambdaMotor * mA1; + const Vector3 angularImpulseBody2 = -angularImpulseBody1; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + w2 += I2 * angularImpulseBody2; + } + } +} + +// Solve the position constraint +void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { + +} + + +// Enable/Disable the limits of the joint +void HingeJoint::enableLimit(bool isLimitEnabled) { + + if (isLimitEnabled != mIsLimitEnabled) { + + mIsLimitEnabled = isLimitEnabled; + + // Reset the limits + resetLimits(); + } +} + +// Enable/Disable the motor of the joint +void HingeJoint::enableMotor(bool isMotorEnabled) { + + mIsMotorEnabled = isMotorEnabled; + mImpulseMotor = 0.0; + + // TODO : Wake up the bodies of the joint here when sleeping is implemented +} + +// Set the minimum angle limit +void HingeJoint::setMinAngleLimit(decimal lowerLimit) { + + assert(mLowerLimit <= 0 && mLowerLimit >= -2.0 * PI); + + if (lowerLimit != mLowerLimit) { + + mLowerLimit = lowerLimit; + + // Reset the limits + resetLimits(); + } +} + +// Set the maximum angle limit +void HingeJoint::setMaxAngleLimit(decimal upperLimit) { + + assert(upperLimit >= 0 && upperLimit <= 2.0 * PI); + + if (upperLimit != mUpperLimit) { + + mUpperLimit = upperLimit; + + // Reset the limits + resetLimits(); + } +} + +// Reset the limits +void HingeJoint::resetLimits() { + + // Reset the accumulated impulses for the limits + mImpulseLowerLimit = 0.0; + mImpulseUpperLimit = 0.0; + + // TODO : Wake up the bodies of the joint here when sleeping is implemented +} + +// Set the motor speed +void HingeJoint::setMotorSpeed(decimal motorSpeed) { + + if (motorSpeed != mMotorSpeed) { + + mMotorSpeed = motorSpeed; + + // TODO : Wake up the bodies of the joint here when sleeping is implemented + } +} + +// Set the maximum motor force +void HingeJoint::setMaxMotorForce(decimal maxMotorForce) { + + if (maxMotorForce != mMaxMotorForce) { + + assert(mMaxMotorForce >= 0.0); + mMaxMotorForce = maxMotorForce; + + // TODO : Wake up the bodies of the joint here when sleeping is implemented + } +} + +// Given an angle in radian, this method returns the corresponding angle in the range [-pi; pi] +decimal HingeJoint::computeNormalizedAngle(decimal angle) const { + + // Convert it into the range [-2*pi; 2*pi] + angle = fmod(angle, PI_TIMES_2); + + // Convert it into the range [-pi; pi] + if (angle < -PI) { + return angle + PI_TIMES_2; + } + else if (angle > PI) { + return angle - PI_TIMES_2; + } + else { + return angle; + } +} + +// Given an "inputAngle" in the range [-pi, pi], this method returns an +// angle (modulo 2*pi) in the range [-2*pi; 2*pi] that is closest to one of the +// two angle limits in arguments. +decimal HingeJoint::computeCorrespondingAngleNearLimits(decimal inputAngle, decimal lowerLimitAngle, + decimal upperLimitAngle) const { + if (upperLimitAngle <= lowerLimitAngle) { + return inputAngle; + } + else if (inputAngle > upperLimitAngle) { + decimal diffToUpperLimit = fabs(computeNormalizedAngle(inputAngle - upperLimitAngle)); + decimal diffToLowerLimit = fabs(computeNormalizedAngle(inputAngle - lowerLimitAngle)); + return (diffToUpperLimit > diffToLowerLimit) ? (inputAngle - PI_TIMES_2) : inputAngle; + } + else if (inputAngle < lowerLimitAngle) { + decimal diffToUpperLimit = fabs(computeNormalizedAngle(upperLimitAngle - inputAngle)); + decimal diffToLowerLimit = fabs(computeNormalizedAngle(lowerLimitAngle - inputAngle)); + return (diffToUpperLimit > diffToLowerLimit) ? inputAngle : (inputAngle + PI_TIMES_2); + } + else { + return inputAngle; + } +} + +// Compute the current angle around the hinge axis +decimal HingeJoint::computeCurrentHingeAngle(const Quaternion& orientationBody1, + const Quaternion& orientationBody2) { + + decimal hingeAngle; + + // Compute the current orientation difference between the two bodies + Quaternion currentOrientationDiff = orientationBody2 * orientationBody1.getInverse(); + currentOrientationDiff.normalize(); + + // Compute the relative rotation considering the initial orientation difference + Quaternion relativeRotation = currentOrientationDiff * mInitOrientationDifferenceInv; + relativeRotation.normalize(); + + // A quaternion q = [cos(theta/2); sin(theta/2) * rotAxis] where rotAxis is a unit + // length vector. We can extract cos(theta/2) with q.w and we can extract |sin(theta/2)| with : + // |sin(theta/2)| = q.getVectorV().length() since rotAxis is unit length. Note that any + // rotation can be represented by a quaternion q and -q. Therefore, if the relative rotation + // axis is not pointing in the same direction as the hinge axis, we use the rotation -q which + // has the same |sin(theta/2)| value but the value cos(theta/2) is sign inverted. Some details + // about this trick is explained in the source code of OpenTissue (http://www.opentissue.org). + decimal cosHalfAngle = relativeRotation.w; + decimal sinHalfAngleAbs = relativeRotation.getVectorV().length(); + + // Compute the dot product of the relative rotation axis and the hinge axis + decimal dotProduct = relativeRotation.getVectorV().dot(mA1); + + // If the relative rotation axis and the hinge axis are pointing the same direction + if (dotProduct >= decimal(0.0)) { + hingeAngle = decimal(2.0) * std::atan2(sinHalfAngleAbs, cosHalfAngle); + } + else { + hingeAngle = decimal(2.0) * std::atan2(sinHalfAngleAbs, -cosHalfAngle); + } + + // Convert the angle from range [-2*pi; 2*pi] into the range [-pi; pi] + hingeAngle = computeNormalizedAngle(hingeAngle); + + // Compute and return the corresponding angle near one the two limits + return computeCorrespondingAngleNearLimits(hingeAngle, mLowerLimit, mUpperLimit); +} + diff --git a/src/constraint/HingeJoint.h b/src/constraint/HingeJoint.h new file mode 100644 index 00000000..aa258159 --- /dev/null +++ b/src/constraint/HingeJoint.h @@ -0,0 +1,342 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_HINGE_JOINT_H +#define REACTPHYSICS3D_HINGE_JOINT_H + +// Libraries +#include "Constraint.h" +#include "../mathematics/mathematics.h" + +namespace reactphysics3d { + +// Structure HingeJointInfo +/** + * This structure is used to gather the information needed to create a hinge joint. + * This structure will be used to create the actual hinge joint. + */ +struct HingeJointInfo : public ConstraintInfo { + + public : + + // -------------------- Attributes -------------------- // + + /// Anchor point (in world-space coordinates) + Vector3 anchorPointWorldSpace; + + /// Hinge rotation axis (in world-space coordinates) + Vector3 rotationAxisWorld; + + /// True if the slider limits are enabled + bool isLimitEnabled; + + /// True if the slider motor is enabled + bool isMotorEnabled; + + /// Minimum allowed rotation angle (in radian) if limits are enabled. + /// The angle must be in the range [-2*pi, 0] + decimal minAngleLimit; + + /// Maximum allowed rotation angle (in radian) if limits are enabled. + /// The angle must be in the range [0, 2*pi] + decimal maxAngleLimit; + + /// Motor speed (in radian/second) + decimal motorSpeed; + + /// Maximum motor force (in Newton) that can be applied to reach to desired motor speed + decimal maxMotorForce; + + /// Constructor without limits and without motor + HingeJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, + const Vector3& initAnchorPointWorldSpace, + const Vector3& initRotationAxisWorld) + : ConstraintInfo(rigidBody1, rigidBody2, HINGEJOINT), + anchorPointWorldSpace(initAnchorPointWorldSpace), + rotationAxisWorld(initRotationAxisWorld), isLimitEnabled(false), + isMotorEnabled(false), minAngleLimit(-1), maxAngleLimit(1), motorSpeed(0), + maxMotorForce(0) {} + + /// Constructor with limits but without motor + HingeJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, + const Vector3& initAnchorPointWorldSpace, + const Vector3& initRotationAxisWorld, + decimal initMinAngleLimit, decimal initMaxAngleLimit) + : ConstraintInfo(rigidBody1, rigidBody2, HINGEJOINT), + anchorPointWorldSpace(initAnchorPointWorldSpace), + rotationAxisWorld(initRotationAxisWorld), isLimitEnabled(true), + isMotorEnabled(false), minAngleLimit(initMinAngleLimit), + maxAngleLimit(initMaxAngleLimit), motorSpeed(0), maxMotorForce(0) {} + + /// Constructor with limits and motor + HingeJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, + const Vector3& initAnchorPointWorldSpace, + const Vector3& initRotationAxisWorld, + decimal initMinAngleLimit, decimal initMaxAngleLimit, + decimal initMotorSpeed, decimal initMaxMotorForce) + : ConstraintInfo(rigidBody1, rigidBody2, HINGEJOINT), + anchorPointWorldSpace(initAnchorPointWorldSpace), + rotationAxisWorld(initRotationAxisWorld), isLimitEnabled(true), + isMotorEnabled(false), minAngleLimit(initMinAngleLimit), + maxAngleLimit(initMaxAngleLimit), motorSpeed(initMotorSpeed), + maxMotorForce(initMaxMotorForce) {} +}; + +// Class HingeJoint +/** + * This class represents a hinge joint that allows arbitrary rotation + * between two bodies around a single axis. + */ +class HingeJoint : public Constraint { + + private : + + // -------------------- Constants -------------------- // + + // Beta value for the bias factor of position correction + static const decimal BETA; + + // -------------------- Attributes -------------------- // + + /// Anchor point of body 1 (in local-space coordinates of body 1) + Vector3 mLocalAnchorPointBody1; + + /// Anchor point of body 2 (in local-space coordinates of body 2) + Vector3 mLocalAnchorPointBody2; + + /// Hinge rotation axis (in local-space coordinates of body 1) + Vector3 mHingeLocalAxisBody1; + + /// Hinge rotation axis (in local-space coordiantes of body 2) + Vector3 mHingeLocalAxisBody2; + + /// Hinge rotation axis (in world-space coordinates) computed from body 1 + Vector3 mA1; + + /// Vector from center of body 2 to anchor point in world-space + Vector3 mR1World; + + /// Vector from center of body 2 to anchor point in world-space + Vector3 mR2World; + + /// Cross product of vector b2 and a1 + Vector3 mB2CrossA1; + + /// Cross product of vector c2 and a1; + Vector3 mC2CrossA1; + + /// Impulse for the 3 translation constraints + Vector3 mImpulseTranslation; + + /// Impulse for the 2 rotation constraints + Vector2 mImpulseRotation; + + /// Accumulated impulse for the lower limit constraint + decimal mImpulseLowerLimit; + + /// Accumulated impulse for the upper limit constraint + decimal mImpulseUpperLimit; + + /// Accumulated impulse for the motor constraint; + decimal mImpulseMotor; + + /// Inverse mass matrix K=JM^-1J^t for the 3 translation constraints + Matrix3x3 mInverseMassMatrixTranslation; + + /// Inverse mass matrix K=JM^-1J^t for the 2 rotation constraints + Matrix2x2 mInverseMassMatrixRotation; + + /// Inverse of mass matrix K=JM^-1J^t for the limits and motor constraints (1x1 matrix) + decimal mInverseMassMatrixLimitMotor; + + /// Inverse of mass matrix K=JM^-1J^t for the motor + decimal mInverseMassMatrixMotor; + + /// Bias vector for the error correction for the translation constraints + Vector3 mBTranslation; + + /// Bias vector for the error correction for the rotation constraints + Vector2 mBRotation; + + /// Bias of the lower limit constraint + decimal mBLowerLimit; + + /// Bias of the upper limit constraint + decimal mBUpperLimit; + + /// Inverse of the initial orientation difference between the bodies + Quaternion mInitOrientationDifferenceInv; + + /// True if the joint limits are enabled + bool mIsLimitEnabled; + + /// True if the motor of the joint in enabled + bool mIsMotorEnabled; + + /// Lower limit (minimum allowed rotation angle in radi) + decimal mLowerLimit; + + /// Upper limit (maximum translation distance) + decimal mUpperLimit; + + /// True if the lower limit is violated + bool mIsLowerLimitViolated; + + /// True if the upper limit is violated + bool mIsUpperLimitViolated; + + /// Motor speed + decimal mMotorSpeed; + + /// Maximum motor force (in Newton) that can be applied to reach to desired motor speed + decimal mMaxMotorForce; + + // -------------------- Methods -------------------- // + + /// Reset the limits + void resetLimits(); + + /// Given an angle in radian, this method returns the corresponding angle in the range [-pi; pi] + decimal computeNormalizedAngle(decimal angle) const; + + /// Given an "inputAngle" in the range [-pi, pi], this method returns an + /// angle (modulo 2*pi) in the range [-2*pi; 2*pi] that is closest to one of the + /// two angle limits in arguments. + decimal computeCorrespondingAngleNearLimits(decimal inputAngle, decimal lowerLimitAngle, + decimal upperLimitAngle) const; + + /// Compute the current angle around the hinge axis + decimal computeCurrentHingeAngle(const Quaternion& orientationBody1, + const Quaternion& orientationBody2); + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + HingeJoint(const HingeJointInfo& jointInfo); + + /// Destructor + virtual ~HingeJoint(); + + /// Return true if the limits or the joint are enabled + bool isLimitEnabled() const; + + /// Return true if the motor of the joint is enabled + bool isMotorEnabled() const; + + /// Enable/Disable the limits of the joint + void enableLimit(bool isLimitEnabled); + + /// Enable/Disable the motor of the joint + void enableMotor(bool isMotorEnabled); + + /// Return the minimum angle limit + decimal getMinAngleLimit() const; + + /// Set the minimum angle limit + void setMinAngleLimit(decimal lowerLimit); + + /// Return the maximum angle limit + decimal getMaxAngleLimit() const; + + /// Set the maximum angle limit + void setMaxAngleLimit(decimal upperLimit); + + /// Return the motor speed + decimal getMotorSpeed() const; + + /// Set the motor speed + void setMotorSpeed(decimal motorSpeed); + + /// Return the maximum motor force + decimal getMaxMotorForce() const; + + /// Set the maximum motor force + void setMaxMotorForce(decimal maxMotorForce); + + /// Return the intensity of the current force applied for the joint motor + decimal getMotorForce(decimal timeStep) const; + + /// Return the number of bytes used by the joint + virtual size_t getSizeInBytes() const; + + /// Initialize before solving the constraint + virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); + + /// Warm start the constraint (apply the previous impulse at the beginning of the step) + virtual void warmstart(const ConstraintSolverData& constraintSolverData); + + /// Solve the velocity constraint + virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); + + /// Solve the position constraint + virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); +}; + +// Return true if the limits or the joint are enabled +inline bool HingeJoint::isLimitEnabled() const { + return mIsLimitEnabled; +} + +// Return true if the motor of the joint is enabled +inline bool HingeJoint::isMotorEnabled() const { + return mIsMotorEnabled; +} + +// Return the minimum angle limit +inline decimal HingeJoint::getMinAngleLimit() const { + return mLowerLimit; +} + +// Return the maximum angle limit +inline decimal HingeJoint::getMaxAngleLimit() const { + return mUpperLimit; +} + +// Return the motor speed +inline decimal HingeJoint::getMotorSpeed() const { + return mMotorSpeed; +} + +// Return the maximum motor force +inline decimal HingeJoint::getMaxMotorForce() const { + return mMaxMotorForce; +} + +// Return the intensity of the current force applied for the joint motor +inline decimal HingeJoint::getMotorForce(decimal timeStep) const { + return mImpulseMotor / timeStep; +} + +// Return the number of bytes used by the joint +inline size_t HingeJoint::getSizeInBytes() const { + return sizeof(HingeJoint); +} + +} + + +#endif diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index 7825a1fc..dc148f6f 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -26,6 +26,8 @@ // Libraries #include "SliderJoint.h" +// TODO : Solve 2x2 or 3x3 linear systems without inverting the A matrix (direct resolution) + using namespace reactphysics3d; // Static variables definition @@ -36,7 +38,7 @@ SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) : Constraint(jointInfo), mImpulseTranslation(0, 0), mImpulseRotation(0, 0, 0), mImpulseLowerLimit(0), mImpulseUpperLimit(0), mImpulseMotor(0), mIsLimitEnabled(jointInfo.isLimitEnabled), mIsMotorEnabled(jointInfo.isMotorEnabled), - mLowerLimit(jointInfo.lowerLimit), mUpperLimit(jointInfo.upperLimit), + mLowerLimit(jointInfo.minTranslationLimit), mUpperLimit(jointInfo.maxTranslationLimit), mIsLowerLimitViolated(false), mIsUpperLimitViolated(false), mMotorSpeed(jointInfo.motorSpeed), mMaxMotorForce(jointInfo.maxMotorForce){ @@ -50,10 +52,11 @@ SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) mLocalAnchorPointBody1 = transform1.getInverse() * jointInfo.anchorPointWorldSpace; mLocalAnchorPointBody2 = transform2.getInverse() * jointInfo.anchorPointWorldSpace; - // Compute the initial orientation difference between the two bodies - mInitOrientationDifference = transform2.getOrientation() * + // Compute the inverse of the initial orientation difference between the two bodies + mInitOrientationDifferenceInv = transform2.getOrientation() * transform1.getOrientation().getInverse(); - mInitOrientationDifference.normalize(); + mInitOrientationDifferenceInv.normalize(); + mInitOrientationDifferenceInv.inverse(); // Compute the slider axis in local-space of body 1 mSliderAxisBody1 = mBody1->getTransform().getOrientation().getInverse() * @@ -87,7 +90,7 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa mR1 = orientationBody1 * mLocalAnchorPointBody1; mR2 = orientationBody2 * mLocalAnchorPointBody2; - // Compute the vector u + // Compute the vector u (difference between anchor points) const Vector3 u = x2 + mR2 - x1 - mR1; // Compute the two orthogonal vectors to the slider axis in world-space @@ -178,14 +181,13 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { Quaternion currentOrientationDifference = orientationBody2 * orientationBody1.getInverse(); currentOrientationDifference.normalize(); - const Quaternion qError = currentOrientationDifference * - mInitOrientationDifference.getInverse(); + const Quaternion qError = currentOrientationDifference * mInitOrientationDifferenceInv; mBRotation = biasFactor * decimal(2.0) * qError.getVectorV(); } if (mIsLimitEnabled && (mIsLowerLimitViolated || mIsUpperLimitViolated)) { - // Compute the inverse of the mass matrix K=JM^-1J^t for the lower limit (1x1 matrix) + // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) mInverseMassMatrixLimit = 0.0; if (mBody1->getIsMotionEnabled()) { mInverseMassMatrixLimit += mBody1->getMassInverse() + @@ -424,7 +426,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint // Compute the Lagrange multiplier lambda for the motor const decimal maxMotorImpulse = mMaxMotorForce * constraintSolverData.timeStep; - decimal deltaLambdaMotor = mInverseMassMatrixMotor * (-JvMotor -mMotorSpeed); + decimal deltaLambdaMotor = mInverseMassMatrixMotor * (-JvMotor - mMotorSpeed); decimal lambdaTemp = mImpulseMotor; mImpulseMotor = clamp(mImpulseMotor + deltaLambdaMotor, -maxMotorImpulse, maxMotorImpulse); deltaLambdaMotor = mImpulseMotor - lambdaTemp; @@ -469,8 +471,8 @@ void SliderJoint::enableMotor(bool isMotorEnabled) { // TODO : Wake up the bodies of the joint here when sleeping is implemented } -// Set the lower limit -void SliderJoint::setLowerLimit(decimal lowerLimit) { +// Set the minimum translation limit +void SliderJoint::setMinTranslationLimit(decimal lowerLimit) { assert(lowerLimit <= mUpperLimit); @@ -483,8 +485,8 @@ void SliderJoint::setLowerLimit(decimal lowerLimit) { } } -// Set the upper limit -void SliderJoint::setUpperLimit(decimal upperLimit) { +// Set the maximum translation limit +void SliderJoint::setMaxTranslationLimit(decimal upperLimit) { assert(mLowerLimit <= upperLimit); diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h index fd54b5cb..79834322 100644 --- a/src/constraint/SliderJoint.h +++ b/src/constraint/SliderJoint.h @@ -55,11 +55,11 @@ struct SliderJointInfo : public ConstraintInfo { /// True if the slider motor is enabled bool isMotorEnabled; - /// Lower limit - decimal lowerLimit; + /// Mininum allowed translation if limits are enabled + decimal minTranslationLimit; - /// Upper limit - decimal upperLimit; + /// Maximum allowed translation if limits are enabled + decimal maxTranslationLimit; /// Motor speed decimal motorSpeed; @@ -74,31 +74,34 @@ struct SliderJointInfo : public ConstraintInfo { : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), sliderAxisWorldSpace(initSliderAxisWorldSpace), - isLimitEnabled(false), isMotorEnabled(false), lowerLimit(-1.0), - upperLimit(1.0), motorSpeed(0), maxMotorForce(0) {} + isLimitEnabled(false), isMotorEnabled(false), minTranslationLimit(-1.0), + maxTranslationLimit(1.0), motorSpeed(0), maxMotorForce(0) {} /// Constructor with limits and no motor SliderJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace, const Vector3& initSliderAxisWorldSpace, - decimal initLowerLimit, decimal initUpperLimit) + decimal initMinTranslationLimit, decimal initMaxTranslationLimit) : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), sliderAxisWorldSpace(initSliderAxisWorldSpace), - isLimitEnabled(true), isMotorEnabled(false), lowerLimit(initLowerLimit), - upperLimit(initUpperLimit), motorSpeed(0), maxMotorForce(0) {} + isLimitEnabled(true), isMotorEnabled(false), + minTranslationLimit(initMinTranslationLimit), + maxTranslationLimit(initMaxTranslationLimit), motorSpeed(0), + maxMotorForce(0) {} /// Constructor with limits and motor SliderJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace, const Vector3& initSliderAxisWorldSpace, - decimal initLowerLimit, decimal initUpperLimit, + decimal initMinTranslationLimit, decimal initMaxTranslationLimit, decimal initMotorSpeed, decimal initMaxMotorForce) : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), sliderAxisWorldSpace(initSliderAxisWorldSpace), - isLimitEnabled(true), isMotorEnabled(true), lowerLimit(initLowerLimit), - upperLimit(initUpperLimit), motorSpeed(initMotorSpeed), + isLimitEnabled(true), isMotorEnabled(true), + minTranslationLimit(initMinTranslationLimit), + maxTranslationLimit(initMaxTranslationLimit), motorSpeed(initMotorSpeed), maxMotorForce(initMaxMotorForce) {} }; @@ -126,8 +129,8 @@ class SliderJoint : public Constraint { /// Slider axis (in local-space coordinates of body 1) Vector3 mSliderAxisBody1; - /// Initial orientation difference between the two bodies - Quaternion mInitOrientationDifference; + /// Inverse of the initial orientation difference between the two bodies + Quaternion mInitOrientationDifferenceInv; /// First vector orthogonal to the slider axis local-space of body 1 Vector3 mN1; @@ -183,19 +186,19 @@ class SliderJoint : public Constraint { /// Inverse of mass matrix K=JM^-1J^t for the motor decimal mInverseMassMatrixMotor; - /// Impulse for the 2 translation constraints + /// Accumulated impulse for the 2 translation constraints Vector2 mImpulseTranslation; - /// Impulse for the 3 rotation constraints + /// Accumulated impulse for the 3 rotation constraints Vector3 mImpulseRotation; - /// Impulse for the lower limit constraint + /// Accumulated impulse for the lower limit constraint decimal mImpulseLowerLimit; - /// Impulse for the upper limit constraint + /// Accumulated impulse for the upper limit constraint decimal mImpulseUpperLimit; - /// Impulse for the motor + /// Accumulated impulse for the motor decimal mImpulseMotor; /// True if the slider limits are enabled @@ -207,10 +210,10 @@ class SliderJoint : public Constraint { /// Slider axis in world-space coordinates Vector3 mSliderAxisWorld; - /// Lower limit + /// Lower limit (minimum translation distance) decimal mLowerLimit; - /// Upper limit + /// Upper limit (maximum translation distance) decimal mUpperLimit; /// True if the lower limit is violated @@ -252,17 +255,17 @@ class SliderJoint : public Constraint { /// Enable/Disable the motor of the joint void enableMotor(bool isMotorEnabled); - /// Return the lower limit - decimal getLowerLimit() const; + /// Return the minimum translation limit + decimal getMinTranslationLimit() const; - /// Set the lower limit - void setLowerLimit(decimal lowerLimit); + /// Set the minimum translation limit + void setMinTranslationLimit(decimal lowerLimit); - /// Return the upper limit - decimal getUpperLimit() const; + /// Return the maximum translation limit + decimal getMaxTranslationLimit() const; - /// Set the upper limit - void setUpperLimit(decimal upperLimit); + /// Set the maximum translation limit + void setMaxTranslationLimit(decimal upperLimit); /// Return the motor speed decimal getMotorSpeed() const; @@ -305,13 +308,13 @@ inline bool SliderJoint::isMotorEnabled() const { return mIsMotorEnabled; } -// Return the lower limit -inline decimal SliderJoint::getLowerLimit() const { +// Return the minimum translation limit +inline decimal SliderJoint::getMinTranslationLimit() const { return mLowerLimit; } -// Return the upper limit -inline decimal SliderJoint::getUpperLimit() const { +// Return the maximum translation limit +inline decimal SliderJoint::getMaxTranslationLimit() const { return mUpperLimit; } diff --git a/src/engine/ConstraintSolver.cpp b/src/engine/ConstraintSolver.cpp index f825e110..ff994a28 100644 --- a/src/engine/ConstraintSolver.cpp +++ b/src/engine/ConstraintSolver.cpp @@ -37,7 +37,7 @@ ConstraintSolver::ConstraintSolver(std::set& joints, : mJoints(joints), mLinearVelocities(linearVelocities), mAngularVelocities(angularVelocities), mMapBodyToConstrainedVelocityIndex(mapBodyToVelocityIndex), - mIsWarmStartingActive(false), + mIsWarmStartingActive(true), mIsNonLinearGaussSeidelPositionCorrectionActive(false), mConstraintSolverData(linearVelocities, angularVelocities, mapBodyToVelocityIndex){ @@ -77,6 +77,11 @@ void ConstraintSolver::initialize(decimal dt) { // Initialize the constraint before solving it joint->initBeforeSolve(mConstraintSolverData); + + // Warm-start the constraint if warm-starting is enabled + if (mIsWarmStartingActive) { + joint->warmstart(mConstraintSolverData); + } } } diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 2c046e99..98a9966a 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -27,6 +27,7 @@ #include "DynamicsWorld.h" #include "constraint/BallAndSocketJoint.h" #include "constraint/SliderJoint.h" +#include "constraint/HingeJoint.h" // Namespaces using namespace reactphysics3d; @@ -212,7 +213,7 @@ void DynamicsWorld::integrateRigidBodiesVelocities() { mConstrainedLinearVelocities = std::vector(mRigidBodies.size(), Vector3(0, 0, 0)); mConstrainedAngularVelocities = std::vector(mRigidBodies.size(), Vector3(0, 0, 0)); - double dt = mTimer.getTimeStep(); + decimal dt = static_cast(mTimer.getTimeStep()); // Fill in the mapping of rigid body to their index in the constrained // velocities arrays @@ -413,6 +414,15 @@ Constraint* DynamicsWorld::createJoint(const ConstraintInfo& jointInfo) { break; } + // Hinge joint + case HINGEJOINT: + { + void* allocatedMemory = mMemoryAllocator.allocate(sizeof(HingeJoint)); + const HingeJointInfo& info = dynamic_cast(jointInfo); + newJoint = new (allocatedMemory) HingeJoint(info); + break; + } + default: { assert(false); diff --git a/src/mathematics/Quaternion.h b/src/mathematics/Quaternion.h index eb957db7..246d56b2 100644 --- a/src/mathematics/Quaternion.h +++ b/src/mathematics/Quaternion.h @@ -84,15 +84,24 @@ struct Quaternion { /// Set the quaternion to zero void setToZero(); + /// Set to the identity quaternion + void setToIdentity(); + /// Return the vector v=(x y z) of the quaternion Vector3 getVectorV() const; /// Return the length of the quaternion decimal length() const; + /// Return the square of the length of the quaternion + decimal lengthSquare() const; + /// Normalize the quaternion void normalize(); + /// Inverse the quaternion + void inverse(); + /// Return the unit quaternion Quaternion getUnit() const; @@ -156,6 +165,14 @@ inline void Quaternion::setToZero() { w = 0; } +// Set to the identity quaternion +inline void Quaternion::setToIdentity() { + x = 0; + y = 0; + z = 0; + w = 1; +} + // Return the vector v=(x y z) of the quaternion inline Vector3 Quaternion::getVectorV() const { @@ -168,6 +185,11 @@ inline decimal Quaternion::length() const { return sqrt(x*x + y*y + z*z + w*w); } +// Return the square of the length of the quaternion +inline decimal Quaternion::lengthSquare() const { + return x*x + y*y + z*z + w*w; +} + // Normalize the quaternion inline void Quaternion::normalize() { @@ -182,6 +204,21 @@ inline void Quaternion::normalize() { w /= l; } +// Inverse the quaternion +inline void Quaternion::inverse() { + + // Get the square length of the quaternion + decimal lengthSquareQuaternion = lengthSquare(); + + assert (lengthSquareQuaternion > MACHINE_EPSILON); + + // Compute and return the inverse quaternion + x /= -lengthSquareQuaternion; + y /= -lengthSquareQuaternion; + z /= -lengthSquareQuaternion; + w /= lengthSquareQuaternion; +} + // Return the unit quaternion inline Quaternion Quaternion::getUnit() const { decimal lengthQuaternion = length(); @@ -207,14 +244,13 @@ inline Quaternion Quaternion::getConjugate() const { // Return the inverse of the quaternion (inline) inline Quaternion Quaternion::getInverse() const { - decimal lengthQuaternion = length(); - lengthQuaternion = lengthQuaternion * lengthQuaternion; + decimal lengthSquareQuaternion = lengthSquare(); - assert (lengthQuaternion > MACHINE_EPSILON); + assert (lengthSquareQuaternion > MACHINE_EPSILON); // Compute and return the inverse quaternion - return Quaternion(-x / lengthQuaternion, -y / lengthQuaternion, - -z / lengthQuaternion, w / lengthQuaternion); + return Quaternion(-x / lengthSquareQuaternion, -y / lengthSquareQuaternion, + -z / lengthSquareQuaternion, w / lengthSquareQuaternion); } // Scalar product between two quaternions diff --git a/src/mathematics/mathematics_functions.h b/src/mathematics/mathematics_functions.h index d5278abd..6ac380d8 100644 --- a/src/mathematics/mathematics_functions.h +++ b/src/mathematics/mathematics_functions.h @@ -29,6 +29,7 @@ // Libraries #include "../configuration.h" #include "../decimal.h" +#include /// ReactPhysics3D namespace namespace reactphysics3d { @@ -43,6 +44,13 @@ inline bool approxEqual(decimal a, decimal b, decimal epsilon = MACHINE_EPSILON) return (difference < epsilon && difference > -epsilon); } +/// Function that returns the result of the "value" clamped by +/// two others values "lowerLimit" and "upperLimit" +inline decimal clamp(decimal value, decimal lowerLimit, decimal upperLimit) { + assert(lowerLimit <= upperLimit); + return std::min(std::max(value, lowerLimit), upperLimit); +} + } diff --git a/src/reactphysics3d.h b/src/reactphysics3d.h index df9e367c..98a8b146 100644 --- a/src/reactphysics3d.h +++ b/src/reactphysics3d.h @@ -49,6 +49,7 @@ #include "collision/shapes/AABB.h" #include "constraint/BallAndSocketJoint.h" #include "constraint/SliderJoint.h" +#include "constraint/HingeJoint.h" /// Alias to the ReactPhysics3D namespace namespace rp3d = reactphysics3d; diff --git a/test/tests/mathematics/TestQuaternion.h b/test/tests/mathematics/TestQuaternion.h index ac01cb6c..ac7ad741 100644 --- a/test/tests/mathematics/TestQuaternion.h +++ b/test/tests/mathematics/TestQuaternion.h @@ -120,6 +120,12 @@ class TestQuaternion : public Test { quaternion.setToZero(); test(quaternion == Quaternion(0, 0, 0, 0)); + // Tes the methods to get or set to identity + Quaternion identity1(1, 2, 3, 4); + identity1.setToIdentity(); + test(identity1 == Quaternion(0, 0, 0, 1)); + test(Quaternion::identity() == Quaternion(0, 0, 0, 1)); + // Test the method to get the vector (x, y, z) Vector3 v = mQuaternion1.getVectorV(); test(v.x == mQuaternion1.x); @@ -133,9 +139,12 @@ class TestQuaternion : public Test { test(conjugate.z == -mQuaternion1.z); test(conjugate.w == mQuaternion1.w); - // Test the inverse method - Quaternion inverse = mQuaternion1.getInverse(); - Quaternion product = mQuaternion1 * inverse; + // Test the inverse methods + Quaternion inverse1 = mQuaternion1.getInverse(); + Quaternion inverse2(mQuaternion1); + inverse2.inverse(); + test(inverse2 == inverse1); + Quaternion product = mQuaternion1 * inverse1; test(approxEqual(product.x, mIdentity.x, decimal(10e-6))); test(approxEqual(product.y, mIdentity.y, decimal(10e-6))); test(approxEqual(product.z, mIdentity.z, decimal(10e-6))); From 1c0726d9d68e5c5b272f7db825481043a9f22511 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 12 Jun 2013 20:43:54 +0200 Subject: [PATCH 32/66] Add the fixed joint --- src/body/CollisionBody.h | 2 +- src/constraint/BallAndSocketJoint.cpp | 1 - src/constraint/BallAndSocketJoint.h | 6 - src/constraint/Constraint.h | 2 +- src/constraint/FixedJoint.cpp | 248 ++++++++++++++++++++++++++ src/constraint/FixedJoint.h | 138 ++++++++++++++ src/constraint/HingeJoint.cpp | 2 - src/constraint/SliderJoint.cpp | 2 - src/engine/DynamicsWorld.cpp | 10 ++ src/reactphysics3d.h | 1 + 10 files changed, 399 insertions(+), 13 deletions(-) create mode 100644 src/constraint/FixedJoint.cpp create mode 100644 src/constraint/FixedJoint.h diff --git a/src/body/CollisionBody.h b/src/body/CollisionBody.h index c119925e..22b93c0b 100644 --- a/src/body/CollisionBody.h +++ b/src/body/CollisionBody.h @@ -208,7 +208,7 @@ inline const Transform& CollisionBody::getTransform() const { inline void CollisionBody::setTransform(const Transform& transform) { // Check if the body has moved - if (this->mTransform != transform) { + if (mTransform != transform) { mHasMoved = true; } diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index 4a8db2ce..f5c0c50c 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -27,7 +27,6 @@ #include "BallAndSocketJoint.h" #include "../engine/ConstraintSolver.h" -// TODO : Solve 2x2 or 3x3 linear systems without inverting the A matrix (direct resolution) using namespace reactphysics3d; diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index dff9210d..934f32eb 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -84,12 +84,6 @@ class BallAndSocketJoint : public Constraint { /// Bias vector for the constraint Vector3 mBiasVector; - /// Skew-Symmetric matrix for cross product with vector mR1World - Matrix3x3 mSkewSymmetricMatrixR1World; - - /// Skew-Symmetric matrix for cross product with vector mR2World - Matrix3x3 mSkewSymmetricMatrixR2World; - /// Inverse mass matrix K=JM^-1J^-t of the constraint Matrix3x3 mInverseMassMatrix; diff --git a/src/constraint/Constraint.h b/src/constraint/Constraint.h index 97b70150..0d3cd8f1 100644 --- a/src/constraint/Constraint.h +++ b/src/constraint/Constraint.h @@ -35,7 +35,7 @@ namespace reactphysics3d { // Enumeration for the type of a constraint -enum ConstraintType {CONTACT, BALLSOCKETJOINT, SLIDERJOINT, HINGEJOINT}; +enum ConstraintType {CONTACT, BALLSOCKETJOINT, SLIDERJOINT, HINGEJOINT, FIXEDJOINT}; // Class declarations struct ConstraintSolverData; diff --git a/src/constraint/FixedJoint.cpp b/src/constraint/FixedJoint.cpp new file mode 100644 index 00000000..8fcbe04f --- /dev/null +++ b/src/constraint/FixedJoint.cpp @@ -0,0 +1,248 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "FixedJoint.h" +#include "../engine/ConstraintSolver.h" + +using namespace reactphysics3d; + +// Static variables definition +const decimal FixedJoint::BETA = decimal(0.2); + +// Constructor +FixedJoint::FixedJoint(const FixedJointInfo& jointInfo) + : Constraint(jointInfo), mImpulseTranslation(0, 0, 0), mImpulseRotation(0, 0, 0) { + + // Compute the local-space anchor point for each body + const Transform& transform1 = mBody1->getTransform(); + const Transform& transform2 = mBody2->getTransform(); + mLocalAnchorPointBody1 = transform1.getInverse() * jointInfo.anchorPointWorldSpace; + mLocalAnchorPointBody2 = transform2.getInverse() * jointInfo.anchorPointWorldSpace; + + // Compute the inverse of the initial orientation difference between the two bodies + mInitOrientationDifferenceInv = transform2.getOrientation() * + transform1.getOrientation().getInverse(); + mInitOrientationDifferenceInv.normalize(); + mInitOrientationDifferenceInv.inverse(); +} + +// Destructor +FixedJoint::~FixedJoint() { + +} + +// Initialize before solving the constraint +void FixedJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverData) { + + // Initialize the bodies index in the velocity array + mIndexBody1 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody1)->second; + mIndexBody2 = constraintSolverData.mapBodyToConstrainedVelocityIndex.find(mBody2)->second; + + // Get the bodies positions and orientations + const Vector3& x1 = mBody1->getTransform().getPosition(); + const Vector3& x2 = mBody2->getTransform().getPosition(); + const Quaternion& orientationBody1 = mBody1->getTransform().getOrientation(); + const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); + + // Get the inertia tensor of bodies + const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the vector from body center to the anchor point in world-space + mR1World = orientationBody1 * mLocalAnchorPointBody1; + mR2World = orientationBody2 * mLocalAnchorPointBody2; + + // Compute the corresponding skew-symmetric matrices + Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR1World); + Matrix3x3 skewSymmetricMatrixU2= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR2World); + + // Compute the matrix K=JM^-1J^t (3x3 matrix) + decimal inverseMassBodies = 0.0; + if (mBody1->getIsMotionEnabled()) { + inverseMassBodies += mBody1->getMassInverse(); + } + if (mBody2->getIsMotionEnabled()) { + inverseMassBodies += mBody2->getMassInverse(); + } + Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, + 0, inverseMassBodies, 0, + 0, 0, inverseMassBodies); + if (mBody1->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU1 * I1 * skewSymmetricMatrixU1.getTranspose(); + } + if (mBody2->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU2 * I2 * skewSymmetricMatrixU2.getTranspose(); + } + + // Compute the inverse mass matrix K^-1 for the 3 translation constraints + mInverseMassMatrixTranslation.setToZero(); + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixTranslation = massMatrix.getInverse(); + } + + // Compute the bias "b" of the constraint for the 3 translation constraints + decimal biasFactor = (BETA / constraintSolverData.timeStep); + mBiasTranslation.setToZero(); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBiasTranslation = biasFactor * (x2 + mR2World - x1 - mR1World); + } + + // Compute the inverse of the mass matrix K=JM^-1J^t for the 3 rotation + // contraints (3x3 matrix) + mInverseMassMatrixRotation.setToZero(); + if (mBody1->getIsMotionEnabled()) { + mInverseMassMatrixRotation += I1; + } + if (mBody2->getIsMotionEnabled()) { + mInverseMassMatrixRotation += I2; + } + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixRotation = mInverseMassMatrixRotation.getInverse(); + } + + // Compute the bias "b" for the 3 rotation constraints + mBiasRotation.setToZero(); + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + Quaternion currentOrientationDifference = orientationBody2 * orientationBody1.getInverse(); + currentOrientationDifference.normalize(); + const Quaternion qError = currentOrientationDifference * mInitOrientationDifferenceInv; + mBiasRotation = biasFactor * decimal(2.0) * qError.getVectorV(); + } + + // If warm-starting is not enabled + if (!constraintSolverData.isWarmStartingActive) { + + // Reset the accumulated impulses + mImpulseTranslation.setToZero(); + mImpulseRotation.setToZero(); + } +} + +// Warm start the constraint (apply the previous impulse at the beginning of the step) +void FixedJoint::warmstart(const ConstraintSolverData& constraintSolverData) { + + // Get the velocities + Vector3& v1 = constraintSolverData.linearVelocities[mIndexBody1]; + Vector3& v2 = constraintSolverData.linearVelocities[mIndexBody2]; + Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; + Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + const decimal inverseMassBody1 = mBody1->getMassInverse(); + const decimal inverseMassBody2 = mBody2->getMassInverse(); + const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the impulse P=J^T * lambda for the 3 translation constraints + Vector3 linearImpulseBody1 = -mImpulseTranslation; + Vector3 angularImpulseBody1 = mImpulseTranslation.cross(mR1World); + Vector3 linearImpulseBody2 = mImpulseTranslation; + Vector3 angularImpulseBody2 = -mImpulseTranslation.cross(mR2World); + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + angularImpulseBody1 += -mImpulseRotation; + angularImpulseBody2 += mImpulseRotation; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += I2 * angularImpulseBody2; + } + +} + +// Solve the velocity constraint +void FixedJoint::solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) { + + // Get the velocities + Vector3& v1 = constraintSolverData.linearVelocities[mIndexBody1]; + Vector3& v2 = constraintSolverData.linearVelocities[mIndexBody2]; + Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; + Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + decimal inverseMassBody1 = mBody1->getMassInverse(); + decimal inverseMassBody2 = mBody2->getMassInverse(); + Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); + Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + + // --------------- Translation Constraints --------------- // + + // Compute J*v for the 3 translation constraints + const Vector3 JvTranslation = v2 + w2.cross(mR2World) - v1 - w1.cross(mR1World); + + // Compute the Lagrange multiplier lambda + const Vector3 deltaLambda = mInverseMassMatrixTranslation * + (-JvTranslation - mBiasTranslation); + mImpulseTranslation += deltaLambda; + + // Compute the impulse P=J^T * lambda + Vector3 linearImpulseBody1 = -deltaLambda; + Vector3 angularImpulseBody1 = deltaLambda.cross(mR1World); + Vector3 linearImpulseBody2 = deltaLambda; + Vector3 angularImpulseBody2 = -deltaLambda.cross(mR2World); + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + v1 += inverseMassBody1 * linearImpulseBody1; + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += I2 * angularImpulseBody2; + } + + // --------------- Rotation Constraints --------------- // + + // Compute J*v for the 3 rotation constraints + const Vector3 JvRotation = w2 - w1; + + // Compute the Lagrange multiplier lambda for the 3 rotation constraints + Vector3 deltaLambda2 = mInverseMassMatrixRotation * (-JvRotation - mBiasRotation); + mImpulseRotation += deltaLambda2; + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + angularImpulseBody1 = -deltaLambda2; + angularImpulseBody2 = deltaLambda2; + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + w1 += I1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + w2 += I2 * angularImpulseBody2; + } +} + +// Solve the position constraint +void FixedJoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { + +} + diff --git a/src/constraint/FixedJoint.h b/src/constraint/FixedJoint.h new file mode 100644 index 00000000..f5e179e7 --- /dev/null +++ b/src/constraint/FixedJoint.h @@ -0,0 +1,138 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_FIXED_JOINT_H +#define REACTPHYSICS3D_FIXED_JOINT_H + +// Libraries +#include "Constraint.h" +#include "../mathematics/mathematics.h" + +namespace reactphysics3d { + +// Structure FixedJointInfo +/** + * This structure is used to gather the information needed to create a fixed + * joint. This structure will be used to create the actual fixed joint. + */ +struct FixedJointInfo : public ConstraintInfo { + + public : + + // -------------------- Attributes -------------------- // + + /// Anchor point (in world-space coordinates) + Vector3 anchorPointWorldSpace; + + /// Constructor + FixedJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, + const Vector3& initAnchorPointWorldSpace) + : ConstraintInfo(rigidBody1, rigidBody2, FIXEDJOINT), + anchorPointWorldSpace(initAnchorPointWorldSpace){} +}; + +// Class FixedJoint +/** + * This class represents a fixed joint that is used to forbid any translation or rotation + * between two bodies. + */ +class FixedJoint : public Constraint { + + private : + + // -------------------- Constants -------------------- // + + // Beta value for the bias factor of position correction + static const decimal BETA; + + // -------------------- Attributes -------------------- // + + /// Anchor point of body 1 (in local-space coordinates of body 1) + Vector3 mLocalAnchorPointBody1; + + /// Anchor point of body 2 (in local-space coordinates of body 2) + Vector3 mLocalAnchorPointBody2; + + /// Vector from center of body 2 to anchor point in world-space + Vector3 mR1World; + + /// Vector from center of body 2 to anchor point in world-space + Vector3 mR2World; + + /// Accumulated impulse for the 3 translation constraints + Vector3 mImpulseTranslation; + + /// Accumulate impulse for the 3 rotation constraints + Vector3 mImpulseRotation; + + /// Inverse mass matrix K=JM^-1J^-t of the 3 translation constraints (3x3 matrix) + Matrix3x3 mInverseMassMatrixTranslation; + + /// Inverse mass matrix K=JM^-1J^-t of the 3 rotation constraints (3x3 matrix) + Matrix3x3 mInverseMassMatrixRotation; + + /// Bias vector for the 3 translation constraints + Vector3 mBiasTranslation; + + /// Bias vector for the 3 rotation constraints + Vector3 mBiasRotation; + + /// Inverse of the initial orientation difference between the two bodies + Quaternion mInitOrientationDifferenceInv; + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + FixedJoint(const FixedJointInfo& jointInfo); + + /// Destructor + virtual ~FixedJoint(); + + /// Return the number of bytes used by the joint + virtual size_t getSizeInBytes() const; + + /// Initialize before solving the constraint + virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); + + /// Warm start the constraint (apply the previous impulse at the beginning of the step) + virtual void warmstart(const ConstraintSolverData& constraintSolverData); + + /// Solve the velocity constraint + virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); + + /// Solve the position constraint + virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); +}; + +// Return the number of bytes used by the joint +inline size_t FixedJoint::getSizeInBytes() const { + return sizeof(FixedJoint); +} + +} + +#endif diff --git a/src/constraint/HingeJoint.cpp b/src/constraint/HingeJoint.cpp index aa0da0eb..10b3589e 100644 --- a/src/constraint/HingeJoint.cpp +++ b/src/constraint/HingeJoint.cpp @@ -28,8 +28,6 @@ #include "../engine/ConstraintSolver.h" #include -// TODO : Solve 2x2 or 3x3 linear systems without inverting the A matrix (direct resolution) - using namespace reactphysics3d; // Static variables definition diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index dc148f6f..c9586a61 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -26,8 +26,6 @@ // Libraries #include "SliderJoint.h" -// TODO : Solve 2x2 or 3x3 linear systems without inverting the A matrix (direct resolution) - using namespace reactphysics3d; // Static variables definition diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 98a9966a..53cd3e35 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -28,6 +28,7 @@ #include "constraint/BallAndSocketJoint.h" #include "constraint/SliderJoint.h" #include "constraint/HingeJoint.h" +#include "constraint/FixedJoint.h" // Namespaces using namespace reactphysics3d; @@ -423,6 +424,15 @@ Constraint* DynamicsWorld::createJoint(const ConstraintInfo& jointInfo) { break; } + // Fixed joint + case FIXEDJOINT: + { + void* allocatedMemory = mMemoryAllocator.allocate(sizeof(FixedJoint)); + const FixedJointInfo& info = dynamic_cast(jointInfo); + newJoint = new (allocatedMemory) FixedJoint(info); + break; + } + default: { assert(false); diff --git a/src/reactphysics3d.h b/src/reactphysics3d.h index 98a8b146..e24c2b82 100644 --- a/src/reactphysics3d.h +++ b/src/reactphysics3d.h @@ -50,6 +50,7 @@ #include "constraint/BallAndSocketJoint.h" #include "constraint/SliderJoint.h" #include "constraint/HingeJoint.h" +#include "constraint/FixedJoint.h" /// Alias to the ReactPhysics3D namespace namespace rp3d = reactphysics3d; From ce0078c2a96a38bb41a523560edefa24e92a550f Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 26 Jun 2013 22:28:31 +0200 Subject: [PATCH 33/66] Implement the non-linear-gauss-seidel position error correction --- src/body/CollisionBody.h | 2 - src/collision/BroadPhasePair.h | 15 +- src/collision/CollisionDetection.cpp | 3 + src/collision/CollisionDetection.h | 23 +- .../narrowphase/EPA/EPAAlgorithm.cpp | 2 +- src/configuration.h | 15 +- src/constraint/BallAndSocketJoint.cpp | 152 ++++-- src/constraint/BallAndSocketJoint.h | 8 +- src/constraint/Constraint.cpp | 3 +- src/constraint/Constraint.h | 22 +- src/constraint/FixedJoint.cpp | 243 +++++++-- src/constraint/FixedJoint.h | 8 +- src/constraint/HingeJoint.cpp | 446 +++++++++++++--- src/constraint/HingeJoint.h | 53 +- src/constraint/SliderJoint.cpp | 482 +++++++++++++++--- src/constraint/SliderJoint.h | 15 +- src/engine/ConstraintSolver.cpp | 19 +- src/engine/ConstraintSolver.h | 34 +- src/engine/ContactSolver.h | 10 +- src/engine/DynamicsWorld.cpp | 135 ++++- src/engine/DynamicsWorld.h | 35 +- src/engine/Timer.h | 10 +- src/mathematics/Quaternion.h | 24 + test/tests/mathematics/TestQuaternion.h | 6 + 24 files changed, 1429 insertions(+), 336 deletions(-) diff --git a/src/body/CollisionBody.h b/src/body/CollisionBody.h index 22b93c0b..df484e33 100644 --- a/src/body/CollisionBody.h +++ b/src/body/CollisionBody.h @@ -239,8 +239,6 @@ inline void CollisionBody::updateOldTransform() { // Update the rigid body in order to reflect a change in the body state inline void CollisionBody::updateAABB() { - // TODO : An AABB should not be updated every frame but only if the body has moved - // Update the AABB mCollisionShape->updateAABB(mAabb, mTransform); } diff --git a/src/collision/BroadPhasePair.h b/src/collision/BroadPhasePair.h index f0a2f9ca..4712c9e1 100644 --- a/src/collision/BroadPhasePair.h +++ b/src/collision/BroadPhasePair.h @@ -60,6 +60,9 @@ struct BroadPhasePair { /// Destructor ~BroadPhasePair(); + /// Return the pair of bodies index + static bodyindexpair computeBodiesIndexPair(CollisionBody* body1, CollisionBody* body2); + /// Return the pair of bodies index bodyindexpair getBodiesIndexPair() const; @@ -77,16 +80,22 @@ struct BroadPhasePair { }; // Return the pair of bodies index -inline bodyindexpair BroadPhasePair::getBodiesIndexPair() const { - +inline bodyindexpair BroadPhasePair::computeBodiesIndexPair(CollisionBody* body1, + CollisionBody* body2) { // Construct the pair of body index - bodyindexpair indexPair = body1->getID() < body2->getID() ? + bodyindexpair indexPair = body1->getID() < body2->getID() ? std::make_pair(body1->getID(), body2->getID()) : std::make_pair(body2->getID(), body1->getID()); assert(indexPair.first != indexPair.second); return indexPair; } +// Return the pair of bodies index +inline bodyindexpair BroadPhasePair::getBodiesIndexPair() const { + + return computeBodiesIndexPair(body1, body2); +} + // Smaller than operator inline bool BroadPhasePair::operator<(const BroadPhasePair& broadPhasePair2) const { return (body1 < broadPhasePair2.body1 ? true : (body2 < broadPhasePair2.body2)); diff --git a/src/collision/CollisionDetection.cpp b/src/collision/CollisionDetection.cpp index 46773a1c..5f97d271 100644 --- a/src/collision/CollisionDetection.cpp +++ b/src/collision/CollisionDetection.cpp @@ -109,6 +109,9 @@ void CollisionDetection::computeNarrowPhase() { // Update the contact cache of the overlapping pair mWorld->updateOverlappingPair(pair); + + // Check if the two bodies are allowed to collide, otherwise, we do not test for collision + if (mNoCollisionPairs.count(pair->getBodiesIndexPair()) > 0) continue; // Select the narrow phase algorithm to use according to the two collision shapes NarrowPhaseAlgorithm& narrowPhaseAlgorithm = SelectNarrowPhaseAlgorithm( diff --git a/src/collision/CollisionDetection.h b/src/collision/CollisionDetection.h index 2cb0879a..a2aa3427 100644 --- a/src/collision/CollisionDetection.h +++ b/src/collision/CollisionDetection.h @@ -79,6 +79,9 @@ class CollisionDetection { /// Narrow-phase Sphere vs Sphere algorithm SphereVsSphereAlgorithm mNarrowPhaseSphereVsSphereAlgorithm; + /// Set of pair of bodies that cannot collide between each other + std::set mNoCollisionPairs; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -113,6 +116,12 @@ class CollisionDetection { /// Remove a body from the collision detection void removeBody(CollisionBody* body); + /// Add a pair of bodies that cannot collide with each other + void addNoCollisionPair(CollisionBody* body1, CollisionBody* body2); + + /// Remove a pair of bodies that cannot collide with each other + void removeNoCollisionPair(CollisionBody *body1, CollisionBody *body2); + /// Compute the collision detection void computeCollisionDetection(); @@ -148,7 +157,19 @@ inline void CollisionDetection::removeBody(CollisionBody* body) { // Remove the body from the broad-phase mBroadPhaseAlgorithm->removeObject(body); -} +} + +// Add a pair of bodies that cannot collide with each other +inline void CollisionDetection::addNoCollisionPair(CollisionBody* body1, + CollisionBody* body2) { + mNoCollisionPairs.insert(BroadPhasePair::computeBodiesIndexPair(body1, body2)); +} + +// Remove a pair of bodies that cannot collide with each other +inline void CollisionDetection::removeNoCollisionPair(CollisionBody* body1, + CollisionBody* body2) { + mNoCollisionPairs.erase(BroadPhasePair::computeBodiesIndexPair(body1, body2)); +} } diff --git a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp index 2cdfc87b..a9cf7a38 100644 --- a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp +++ b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp @@ -143,7 +143,7 @@ bool EPAAlgorithm::computePenetrationDepthAndContactPoints(const Simplex& simple int minAxis = d.getAbsoluteVector().getMinAxis(); // Compute sin(60) - const decimal sin60 = sqrt(3.0) * decimal(0.5); + const decimal sin60 = decimal(sqrt(3.0)) * decimal(0.5); // Create a rotation quaternion to rotate the vector v1 to get the vectors // v2 and v3 diff --git a/src/configuration.h b/src/configuration.h index 83e3c817..4873ff96 100644 --- a/src/configuration.h +++ b/src/configuration.h @@ -54,15 +54,14 @@ typedef std::pair bodyindexpair; // ------------------- Enumerations ------------------- // /// Position correction technique used in the constraint solver (for joints). -/// BAUMGARTE : Faster but can be innacurate in some situations. This is the option -/// used by default. -/// NON_LINEAR_GAUSS_SEIDEL : Slower but more precise. +/// BAUMGARTE_JOINTS : Faster but can be innacurate in some situations. +/// NON_LINEAR_GAUSS_SEIDEL : Slower but more precise. This is the option used by default. enum JointsPositionCorrectionTechnique {BAUMGARTE_JOINTS, NON_LINEAR_GAUSS_SEIDEL}; /// Position correction technique used in the contact solver (for contacts) -/// BAUMGARTE : Faster but can be innacurate and can lead to unexpected bounciness -/// in some situations (due to error correction factor being added to -/// the bodies momentum). +/// BAUMGARTE_CONTACTS : Faster but can be innacurate and can lead to unexpected bounciness +/// in some situations (due to error correction factor being added to +/// the bodies momentum). /// SPLIT_IMPULSES : A bit slower but the error correction factor is not added to the /// bodies momentum. This is the option used by default. enum ContactsPositionCorrectionTechnique {BAUMGARTE_CONTACTS, SPLIT_IMPULSES}; @@ -93,7 +92,7 @@ const decimal DEFAULT_FRICTION_COEFFICIENT = decimal(0.3); /// True if the deactivation (sleeping) of inactive bodies is enabled const bool DEACTIVATION_ENABLED = true; -///Object margin for collision detection in cm (For GJK-EPA Algorithm) +/// Object margin for collision detection in cm (For GJK-EPA Algorithm) const decimal OBJECT_MARGIN = decimal(0.04); /// Distance threshold for two contact points for a valid persistent contact (in meters) @@ -106,7 +105,7 @@ const decimal RESTITUTION_VELOCITY_THRESHOLD = decimal(1.0); const uint DEFAULT_VELOCITY_SOLVER_NB_ITERATIONS = 15; /// Number of iterations when solving the position constraints of the Sequential Impulse technique -const uint DEFAULT_POSITION_SOLVER_NB_ITERATIONS = 3; // TODO : Maybe we can use less iterations here +const uint DEFAULT_POSITION_SOLVER_NB_ITERATIONS = 5; } diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index f5c0c50c..6dc016ae 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -27,7 +27,6 @@ #include "BallAndSocketJoint.h" #include "../engine/ConstraintSolver.h" - using namespace reactphysics3d; // Static variables definition @@ -61,8 +60,8 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); // Get the inertia tensor of bodies - const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + mI1 = mBody1->getInertiaTensorInverseWorld(); + mI2 = mBody2->getInertiaTensorInverseWorld(); // Compute the vector from body center to the anchor point in world-space mR1World = orientationBody1 * mLocalAnchorPointBody1; @@ -84,10 +83,10 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS 0, inverseMassBodies, 0, 0, 0, inverseMassBodies); if (mBody1->getIsMotionEnabled()) { - massMatrix += skewSymmetricMatrixU1 * I1 * skewSymmetricMatrixU1.getTranspose(); + massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); } if (mBody2->getIsMotionEnabled()) { - massMatrix += skewSymmetricMatrixU2 * I2 * skewSymmetricMatrixU2.getTranspose(); + massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); } // Compute the inverse mass matrix K^-1 @@ -120,26 +119,29 @@ void BallAndSocketJoint::warmstart(const ConstraintSolverData& constraintSolverD Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; - // Get the inverse mass and inverse inertia tensors of the bodies + // Get the inverse mass of the bodies const decimal inverseMassBody1 = mBody1->getMassInverse(); const decimal inverseMassBody2 = mBody2->getMassInverse(); - const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); - // Compute the impulse P=J^T * lambda - const Vector3 linearImpulseBody1 = -mImpulse; - const Vector3 angularImpulseBody1 = mImpulse.cross(mR1World); - const Vector3 linearImpulseBody2 = mImpulse; - const Vector3 angularImpulseBody2 = -mImpulse.cross(mR2World); - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 linearImpulseBody1 = -mImpulse; + const Vector3 angularImpulseBody1 = mImpulse.cross(mR1World); + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; - w1 += I1 * angularImpulseBody1; + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 linearImpulseBody2 = mImpulse; + const Vector3 angularImpulseBody2 = -mImpulse.cross(mR2World); + + // Apply the impulse to the body v2 += inverseMassBody2 * linearImpulseBody2; - w2 += I2 * angularImpulseBody2; + w2 += mI2 * angularImpulseBody2; } } @@ -152,11 +154,9 @@ void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& con Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; - // Get the inverse mass and inverse inertia tensors of the bodies + // Get the inverse mass of the bodies decimal inverseMassBody1 = mBody1->getMassInverse(); decimal inverseMassBody2 = mBody2->getMassInverse(); - Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // Compute J*v const Vector3 Jv = v2 + w2.cross(mR2World) - v1 - w1.cross(mR1World); @@ -165,25 +165,117 @@ void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& con const Vector3 deltaLambda = mInverseMassMatrix * (-Jv - mBiasVector); mImpulse += deltaLambda; - // Compute the impulse P=J^T * lambda - const Vector3 linearImpulseBody1 = -deltaLambda; - const Vector3 angularImpulseBody1 = deltaLambda.cross(mR1World); - const Vector3 linearImpulseBody2 = deltaLambda; - const Vector3 angularImpulseBody2 = -deltaLambda.cross(mR2World); - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 linearImpulseBody1 = -deltaLambda; + const Vector3 angularImpulseBody1 = deltaLambda.cross(mR1World); + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; - w1 += I1 * angularImpulseBody1; + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 linearImpulseBody2 = deltaLambda; + const Vector3 angularImpulseBody2 = -deltaLambda.cross(mR2World); + + // Apply the impulse to the body v2 += inverseMassBody2 * linearImpulseBody2; - w2 += I2 * angularImpulseBody2; + w2 += mI2 * angularImpulseBody2; } } -// Solve the position constraint +// Solve the position constraint (for position error correction) void BallAndSocketJoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { + // If the error position correction technique is not the non-linear-gauss-seidel, we do + // do not execute this method + if (mPositionCorrectionTechnique != NON_LINEAR_GAUSS_SEIDEL) return; + + // Get the bodies positions and orientations + Vector3& x1 = constraintSolverData.positions[mIndexBody1]; + Vector3& x2 = constraintSolverData.positions[mIndexBody2]; + Quaternion& q1 = constraintSolverData.orientations[mIndexBody1]; + Quaternion& q2 = constraintSolverData.orientations[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + decimal inverseMassBody1 = mBody1->getMassInverse(); + decimal inverseMassBody2 = mBody2->getMassInverse(); + + // Recompute the inverse inertia tensors + mI1 = mBody1->getInertiaTensorInverseWorld(); + mI2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the vector from body center to the anchor point in world-space + mR1World = q1 * mLocalAnchorPointBody1; + mR2World = q2 * mLocalAnchorPointBody2; + + // Compute the corresponding skew-symmetric matrices + Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR1World); + Matrix3x3 skewSymmetricMatrixU2= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR2World); + + // Recompute the inverse mass matrix K=J^TM^-1J of of the 3 translation constraints + decimal inverseMassBodies = 0.0; + if (mBody1->getIsMotionEnabled()) { + inverseMassBodies += inverseMassBody1; + } + if (mBody2->getIsMotionEnabled()) { + inverseMassBodies += inverseMassBody2; + } + Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, + 0, inverseMassBodies, 0, + 0, 0, inverseMassBodies); + if (mBody1->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); + } + if (mBody2->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); + } + mInverseMassMatrix.setToZero(); + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrix = massMatrix.getInverse(); + } + + // Compute the constraint error (value of the C(x) function) + const Vector3 constraintError = (x2 + mR2World - x1 - mR1World); + + // Compute the Lagrange multiplier lambda + // TODO : Do not solve the system by computing the inverse each time and multiplying with the + // right-hand side vector but instead use a method to directly solve the linear system. + const Vector3 lambda = mInverseMassMatrix * (-constraintError); + + // Apply the impulse to the bodies of the joint (directly update the position/orientation) + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse + const Vector3 linearImpulseBody1 = -lambda; + const Vector3 angularImpulseBody1 = lambda.cross(mR1World); + + // Compute the pseudo velocity + const Vector3 v1 = inverseMassBody1 * linearImpulseBody1; + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + x1 += v1; + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse + const Vector3 linearImpulseBody2 = lambda; + const Vector3 angularImpulseBody2 = -lambda.cross(mR2World); + + // Compute the pseudo velocity + const Vector3 v2 = inverseMassBody2 * linearImpulseBody2; + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + x2 += v2; + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); + } } diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index 934f32eb..6cd47fa6 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -81,6 +81,12 @@ class BallAndSocketJoint : public Constraint { /// Vector from center of body 2 to anchor point in world-space Vector3 mR2World; + /// Inertia tensor of body 1 (in world-space coordinates) + Matrix3x3 mI1; + + /// Inertia tensor of body 2 (in world-space coordinates) + Matrix3x3 mI2; + /// Bias vector for the constraint Vector3 mBiasVector; @@ -112,7 +118,7 @@ class BallAndSocketJoint : public Constraint { /// Solve the velocity constraint virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); - /// Solve the position constraint + /// Solve the position constraint (for position error correction) virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); }; diff --git a/src/constraint/Constraint.cpp b/src/constraint/Constraint.cpp index 234861e3..5da1c4eb 100644 --- a/src/constraint/Constraint.cpp +++ b/src/constraint/Constraint.cpp @@ -32,7 +32,8 @@ using namespace reactphysics3d; Constraint::Constraint(const ConstraintInfo& constraintInfo) :mBody1(constraintInfo.body1), mBody2(constraintInfo.body2), mActive(true), mType(constraintInfo.type), - mPositionCorrectionTechnique(constraintInfo.positionCorrectionTechnique) { + mPositionCorrectionTechnique(constraintInfo.positionCorrectionTechnique), + mIsCollisionEnabled(constraintInfo.isCollisionEnabled){ assert(mBody1 != NULL); assert(mBody2 != NULL); diff --git a/src/constraint/Constraint.h b/src/constraint/Constraint.h index 0d3cd8f1..ae7bfa77 100644 --- a/src/constraint/Constraint.h +++ b/src/constraint/Constraint.h @@ -59,6 +59,9 @@ struct ConstraintInfo { /// Type of the constraint ConstraintType type; + /// True if the two bodies of the constraint are allowed to collide with each other + bool isCollisionEnabled; + /// Position correction technique used for the constraint (used for joints). /// By default, the BAUMGARTE technique is used JointsPositionCorrectionTechnique positionCorrectionTechnique; @@ -66,12 +69,14 @@ struct ConstraintInfo { /// Constructor ConstraintInfo(ConstraintType constraintType) : body1(NULL), body2(NULL), type(constraintType), - positionCorrectionTechnique(BAUMGARTE_JOINTS) {} + positionCorrectionTechnique(NON_LINEAR_GAUSS_SEIDEL), + isCollisionEnabled(true) {} /// Constructor ConstraintInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, ConstraintType constraintType) : body1(rigidBody1), body2(rigidBody2), type(constraintType), - positionCorrectionTechnique(BAUMGARTE_JOINTS) { + positionCorrectionTechnique(NON_LINEAR_GAUSS_SEIDEL), + isCollisionEnabled(true) { } /// Destructor @@ -113,6 +118,9 @@ class Constraint { /// Position correction technique used for the constraint (used for joints) JointsPositionCorrectionTechnique mPositionCorrectionTechnique; + /// True if the two bodies of the constraint are allowed to collide with each other + bool mIsCollisionEnabled; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -143,6 +151,9 @@ class Constraint { /// Return the type of the constraint ConstraintType getType() const; + /// Return true if the collision between the two bodies of the constraint is enabled + bool isCollisionEnabled() const; + /// Return the number of bytes used by the constraint virtual size_t getSizeInBytes() const = 0; @@ -177,7 +188,12 @@ inline bool Constraint::isActive() const { // Return the type of the constraint inline ConstraintType Constraint::getType() const { return mType; -} +} + +// Return true if the collision between the two bodies of the constraint is enabled +inline bool Constraint::isCollisionEnabled() const { + return mIsCollisionEnabled; +} } diff --git a/src/constraint/FixedJoint.cpp b/src/constraint/FixedJoint.cpp index 8fcbe04f..7a5acf82 100644 --- a/src/constraint/FixedJoint.cpp +++ b/src/constraint/FixedJoint.cpp @@ -68,8 +68,8 @@ void FixedJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); // Get the inertia tensor of bodies - const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + mI1 = mBody1->getInertiaTensorInverseWorld(); + mI2 = mBody2->getInertiaTensorInverseWorld(); // Compute the vector from body center to the anchor point in world-space mR1World = orientationBody1 * mLocalAnchorPointBody1; @@ -79,7 +79,7 @@ void FixedJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR1World); Matrix3x3 skewSymmetricMatrixU2= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR2World); - // Compute the matrix K=JM^-1J^t (3x3 matrix) + // Compute the matrix K=JM^-1J^t (3x3 matrix) for the 3 translation constraints decimal inverseMassBodies = 0.0; if (mBody1->getIsMotionEnabled()) { inverseMassBodies += mBody1->getMassInverse(); @@ -91,10 +91,10 @@ void FixedJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat 0, inverseMassBodies, 0, 0, 0, inverseMassBodies); if (mBody1->getIsMotionEnabled()) { - massMatrix += skewSymmetricMatrixU1 * I1 * skewSymmetricMatrixU1.getTranspose(); + massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); } if (mBody2->getIsMotionEnabled()) { - massMatrix += skewSymmetricMatrixU2 * I2 * skewSymmetricMatrixU2.getTranspose(); + massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); } // Compute the inverse mass matrix K^-1 for the 3 translation constraints @@ -114,10 +114,10 @@ void FixedJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat // contraints (3x3 matrix) mInverseMassMatrixRotation.setToZero(); if (mBody1->getIsMotionEnabled()) { - mInverseMassMatrixRotation += I1; + mInverseMassMatrixRotation += mI1; } if (mBody2->getIsMotionEnabled()) { - mInverseMassMatrixRotation += I2; + mInverseMassMatrixRotation += mI2; } if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { mInverseMassMatrixRotation = mInverseMassMatrixRotation.getInverse(); @@ -150,32 +150,36 @@ void FixedJoint::warmstart(const ConstraintSolverData& constraintSolverData) { Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; - // Get the inverse mass and inverse inertia tensors of the bodies + // Get the inverse mass of the bodies const decimal inverseMassBody1 = mBody1->getMassInverse(); const decimal inverseMassBody2 = mBody2->getMassInverse(); - const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); - // Compute the impulse P=J^T * lambda for the 3 translation constraints - Vector3 linearImpulseBody1 = -mImpulseTranslation; - Vector3 angularImpulseBody1 = mImpulseTranslation.cross(mR1World); - Vector3 linearImpulseBody2 = mImpulseTranslation; - Vector3 angularImpulseBody2 = -mImpulseTranslation.cross(mR2World); - - // Compute the impulse P=J^T * lambda for the 3 rotation constraints - angularImpulseBody1 += -mImpulseRotation; - angularImpulseBody2 += mImpulseRotation; - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 3 translation constraints + Vector3 linearImpulseBody1 = -mImpulseTranslation; + Vector3 angularImpulseBody1 = mImpulseTranslation.cross(mR1World); + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + angularImpulseBody1 += -mImpulseRotation; + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; - w1 += I1 * angularImpulseBody1; + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { - v2 += inverseMassBody2 * linearImpulseBody2; - w2 += I2 * angularImpulseBody2; - } + // Compute the impulse P=J^T * lambda for the 3 translation constraints + Vector3 linearImpulseBody2 = mImpulseTranslation; + Vector3 angularImpulseBody2 = -mImpulseTranslation.cross(mR2World); + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + angularImpulseBody2 += mImpulseRotation; + + // Apply the impulse to the body + v2 += inverseMassBody2 * linearImpulseBody2; + w2 += mI2 * angularImpulseBody2; + } } // Solve the velocity constraint @@ -187,11 +191,9 @@ void FixedJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS Vector3& w1 = constraintSolverData.angularVelocities[mIndexBody1]; Vector3& w2 = constraintSolverData.angularVelocities[mIndexBody2]; - // Get the inverse mass and inverse inertia tensors of the bodies + // Get the inverse mass of the bodies decimal inverseMassBody1 = mBody1->getMassInverse(); decimal inverseMassBody2 = mBody2->getMassInverse(); - Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // --------------- Translation Constraints --------------- // @@ -203,20 +205,25 @@ void FixedJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS (-JvTranslation - mBiasTranslation); mImpulseTranslation += deltaLambda; - // Compute the impulse P=J^T * lambda - Vector3 linearImpulseBody1 = -deltaLambda; - Vector3 angularImpulseBody1 = deltaLambda.cross(mR1World); - Vector3 linearImpulseBody2 = deltaLambda; - Vector3 angularImpulseBody2 = -deltaLambda.cross(mR2World); - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 linearImpulseBody1 = -deltaLambda; + const Vector3 angularImpulseBody1 = deltaLambda.cross(mR1World); + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; - w1 += I1 * angularImpulseBody1; + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 linearImpulseBody2 = deltaLambda; + const Vector3 angularImpulseBody2 = -deltaLambda.cross(mR2World); + + // Apply the impulse to the body v2 += inverseMassBody2 * linearImpulseBody2; - w2 += I2 * angularImpulseBody2; + w2 += mI2 * angularImpulseBody2; } // --------------- Rotation Constraints --------------- // @@ -228,21 +235,163 @@ void FixedJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS Vector3 deltaLambda2 = mInverseMassMatrixRotation * (-JvRotation - mBiasRotation); mImpulseRotation += deltaLambda2; - // Compute the impulse P=J^T * lambda for the 3 rotation constraints - angularImpulseBody1 = -deltaLambda2; - angularImpulseBody2 = deltaLambda2; + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + const Vector3 angularImpulseBody1 = -deltaLambda2; + + // Apply the impulse to the body + w1 += mI1 * angularImpulseBody1; + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + const Vector3 angularImpulseBody2 = deltaLambda2; + + // Apply the impulse to the body + w2 += mI2 * angularImpulseBody2; + } +} + +// Solve the position constraint (for position error correction) +void FixedJoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { + + // If the error position correction technique is not the non-linear-gauss-seidel, we do + // do not execute this method + if (mPositionCorrectionTechnique != NON_LINEAR_GAUSS_SEIDEL) return; + + // Get the bodies positions and orientations + Vector3& x1 = constraintSolverData.positions[mIndexBody1]; + Vector3& x2 = constraintSolverData.positions[mIndexBody2]; + Quaternion& q1 = constraintSolverData.orientations[mIndexBody1]; + Quaternion& q2 = constraintSolverData.orientations[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + decimal inverseMassBody1 = mBody1->getMassInverse(); + decimal inverseMassBody2 = mBody2->getMassInverse(); + + // Recompute the inverse inertia tensors + mI1 = mBody1->getInertiaTensorInverseWorld(); + mI2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the vector from body center to the anchor point in world-space + mR1World = q1 * mLocalAnchorPointBody1; + mR2World = q2 * mLocalAnchorPointBody2; + + // Compute the corresponding skew-symmetric matrices + Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR1World); + Matrix3x3 skewSymmetricMatrixU2= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR2World); + + // --------------- Translation Constraints --------------- // + + // Compute the matrix K=JM^-1J^t (3x3 matrix) for the 3 translation constraints + decimal inverseMassBodies = 0.0; + if (mBody1->getIsMotionEnabled()) { + inverseMassBodies += mBody1->getMassInverse(); + } + if (mBody2->getIsMotionEnabled()) { + inverseMassBodies += mBody2->getMassInverse(); + } + Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, + 0, inverseMassBodies, 0, + 0, 0, inverseMassBodies); + if (mBody1->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); + } + if (mBody2->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); + } + mInverseMassMatrixTranslation.setToZero(); + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixTranslation = massMatrix.getInverse(); + } + + // Compute position error for the 3 translation constraints + const Vector3 errorTranslation = x2 + mR2World - x1 - mR1World; + + // Compute the Lagrange multiplier lambda + const Vector3 lambdaTranslation = mInverseMassMatrixTranslation * (-errorTranslation); // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { - w1 += I1 * angularImpulseBody1; + + // Compute the impulse + Vector3 linearImpulseBody1 = -lambdaTranslation; + Vector3 angularImpulseBody1 = lambdaTranslation.cross(mR1World); + + // Compute the pseudo velocity + const Vector3 v1 = inverseMassBody1 * linearImpulseBody1; + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + x1 += v1; + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); } if (mBody2->getIsMotionEnabled()) { - w2 += I2 * angularImpulseBody2; + + // Compute the impulse + Vector3 linearImpulseBody2 = lambdaTranslation; + Vector3 angularImpulseBody2 = -lambdaTranslation.cross(mR2World); + + // Compute the pseudo velocity + const Vector3 v2 = inverseMassBody2 * linearImpulseBody2; + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + x2 += v2; + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); + } + + // --------------- Rotation Constraints --------------- // + + // Compute the inverse of the mass matrix K=JM^-1J^t for the 3 rotation + // contraints (3x3 matrix) + mInverseMassMatrixRotation.setToZero(); + if (mBody1->getIsMotionEnabled()) { + mInverseMassMatrixRotation += mI1; + } + if (mBody2->getIsMotionEnabled()) { + mInverseMassMatrixRotation += mI2; + } + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixRotation = mInverseMassMatrixRotation.getInverse(); + } + + // Compute the position error for the 3 rotation constraints + Quaternion currentOrientationDifference = q2 * q1.getInverse(); + currentOrientationDifference.normalize(); + const Quaternion qError = currentOrientationDifference * mInitOrientationDifferenceInv; + const Vector3 errorRotation = decimal(2.0) * qError.getVectorV(); + + // Compute the Lagrange multiplier lambda for the 3 rotation constraints + Vector3 lambdaRotation = mInverseMassMatrixRotation * (-errorRotation); + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + const Vector3 angularImpulseBody1 = -lambdaRotation; + + // Compute the pseudo velocity + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse + const Vector3 angularImpulseBody2 = lambdaRotation; + + // Compute the pseudo velocity + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); } } -// Solve the position constraint -void FixedJoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { - -} - diff --git a/src/constraint/FixedJoint.h b/src/constraint/FixedJoint.h index f5e179e7..7f4f987f 100644 --- a/src/constraint/FixedJoint.h +++ b/src/constraint/FixedJoint.h @@ -81,6 +81,12 @@ class FixedJoint : public Constraint { /// Vector from center of body 2 to anchor point in world-space Vector3 mR2World; + /// Inertia tensor of body 1 (in world-space coordinates) + Matrix3x3 mI1; + + /// Inertia tensor of body 2 (in world-space coordinates) + Matrix3x3 mI2; + /// Accumulated impulse for the 3 translation constraints Vector3 mImpulseTranslation; @@ -124,7 +130,7 @@ class FixedJoint : public Constraint { /// Solve the velocity constraint virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); - /// Solve the position constraint + /// Solve the position constraint (for position error correction) virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); }; diff --git a/src/constraint/HingeJoint.cpp b/src/constraint/HingeJoint.cpp index 10b3589e..57c8764f 100644 --- a/src/constraint/HingeJoint.cpp +++ b/src/constraint/HingeJoint.cpp @@ -40,7 +40,7 @@ HingeJoint::HingeJoint(const HingeJointInfo& jointInfo) mIsLimitEnabled(jointInfo.isLimitEnabled), mIsMotorEnabled(jointInfo.isMotorEnabled), mLowerLimit(jointInfo.minAngleLimit), mUpperLimit(jointInfo.maxAngleLimit), mIsLowerLimitViolated(false), mIsUpperLimitViolated(false), - mMotorSpeed(jointInfo.motorSpeed), mMaxMotorForce(jointInfo.maxMotorForce) { + mMotorSpeed(jointInfo.motorSpeed), mMaxMotorTorque(jointInfo.maxMotorTorque) { assert(mLowerLimit <= 0 && mLowerLimit >= -2.0 * PI); assert(mUpperLimit >= 0 && mUpperLimit <= 2.0 * PI); @@ -83,8 +83,8 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); // Get the inertia tensor of bodies - const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + mI1 = mBody1->getInertiaTensorInverseWorld(); + mI2 = mBody2->getInertiaTensorInverseWorld(); // Compute the vector from body center to the anchor point in world-space mR1World = orientationBody1 * mLocalAnchorPointBody1; @@ -107,8 +107,6 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat mImpulseUpperLimit = 0.0; } - decimal testAngle = computeCurrentHingeAngle(orientationBody1, orientationBody2); - // Compute vectors needed in the Jacobian mA1 = orientationBody1 * mHingeLocalAxisBody1; Vector3 a2 = orientationBody2 * mHingeLocalAxisBody2; @@ -135,10 +133,10 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat 0, inverseMassBodies, 0, 0, 0, inverseMassBodies); if (mBody1->getIsMotionEnabled()) { - massMatrix += skewSymmetricMatrixU1 * I1 * skewSymmetricMatrixU1.getTranspose(); + massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); } if (mBody2->getIsMotionEnabled()) { - massMatrix += skewSymmetricMatrixU2 * I2 * skewSymmetricMatrixU2.getTranspose(); + massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); } mInverseMassMatrixTranslation.setToZero(); if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { @@ -158,12 +156,12 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat Vector3 I2B2CrossA1(0, 0, 0); Vector3 I2C2CrossA1(0, 0, 0); if (mBody1->getIsMotionEnabled()) { - I1B2CrossA1 = I1 * mB2CrossA1; - I1C2CrossA1 = I1 * mC2CrossA1; + I1B2CrossA1 = mI1 * mB2CrossA1; + I1C2CrossA1 = mI1 * mC2CrossA1; } if (mBody2->getIsMotionEnabled()) { - I2B2CrossA1 = I2 * mB2CrossA1; - I2C2CrossA1 = I2 * mC2CrossA1; + I2B2CrossA1 = mI2 * mB2CrossA1; + I2C2CrossA1 = mI2 * mC2CrossA1; } const decimal el11 = mB2CrossA1.dot(I1B2CrossA1) + mB2CrossA1.dot(I2B2CrossA1); @@ -201,10 +199,10 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) mInverseMassMatrixLimitMotor = 0.0; if (mBody1->getIsMotionEnabled()) { - mInverseMassMatrixLimitMotor += mA1.dot(I1 * mA1); + mInverseMassMatrixLimitMotor += mA1.dot(mI1 * mA1); } if (mBody2->getIsMotionEnabled()) { - mInverseMassMatrixLimitMotor += mA1.dot(I2 * mA1); + mInverseMassMatrixLimitMotor += mA1.dot(mI2 * mA1); } mInverseMassMatrixLimitMotor = (mInverseMassMatrixLimitMotor > 0.0) ? decimal(1.0) / mInverseMassMatrixLimitMotor : decimal(0.0); @@ -235,38 +233,53 @@ void HingeJoint::warmstart(const ConstraintSolverData& constraintSolverData) { // Get the inverse mass and inverse inertia tensors of the bodies const decimal inverseMassBody1 = mBody1->getMassInverse(); const decimal inverseMassBody2 = mBody2->getMassInverse(); - const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); - - // Compute the impulse P=J^T * lambda for the 3 translation constraints - Vector3 linearImpulseBody1 = -mImpulseTranslation; - Vector3 angularImpulseBody1 = mImpulseTranslation.cross(mR1World); - Vector3 linearImpulseBody2 = mImpulseTranslation; - Vector3 angularImpulseBody2 = -mImpulseTranslation.cross(mR2World); // Compute the impulse P=J^T * lambda for the 2 rotation constraints Vector3 rotationImpulse = -mB2CrossA1 * mImpulseRotation.x - mC2CrossA1 * mImpulseRotation.y; - angularImpulseBody1 += rotationImpulse; - angularImpulseBody2 += -rotationImpulse; // Compute the impulse P=J^T * lambda for the lower and upper limits constraints const Vector3 limitsImpulse = (mImpulseUpperLimit - mImpulseLowerLimit) * mA1; - angularImpulseBody1 += limitsImpulse; - angularImpulseBody2 += -limitsImpulse; // Compute the impulse P=J^T * lambda for the motor constraint const Vector3 motorImpulse = -mImpulseMotor * mA1; - angularImpulseBody1 += motorImpulse; - angularImpulseBody2 += -motorImpulse; - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 3 translation constraints + Vector3 linearImpulseBody1 = -mImpulseTranslation; + Vector3 angularImpulseBody1 = mImpulseTranslation.cross(mR1World); + + // Compute the impulse P=J^T * lambda for the 2 rotation constraints + angularImpulseBody1 += rotationImpulse; + + // Compute the impulse P=J^T * lambda for the lower and upper limits constraints + angularImpulseBody1 += limitsImpulse; + + // Compute the impulse P=J^T * lambda for the motor constraint + angularImpulseBody1 += motorImpulse; + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; - w1 += I1 * angularImpulseBody1; + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 3 translation constraints + Vector3 linearImpulseBody2 = mImpulseTranslation; + Vector3 angularImpulseBody2 = -mImpulseTranslation.cross(mR2World); + + // Compute the impulse P=J^T * lambda for the 2 rotation constraints + angularImpulseBody2 += -rotationImpulse; + + // Compute the impulse P=J^T * lambda for the lower and upper limits constraints + angularImpulseBody2 += -limitsImpulse; + + // Compute the impulse P=J^T * lambda for the motor constraint + angularImpulseBody2 += -motorImpulse; + + // Apply the impulse to the body v2 += inverseMassBody2 * linearImpulseBody2; - w2 += I2 * angularImpulseBody2; + w2 += mI2 * angularImpulseBody2; } } @@ -282,8 +295,6 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS // Get the inverse mass and inverse inertia tensors of the bodies decimal inverseMassBody1 = mBody1->getMassInverse(); decimal inverseMassBody2 = mBody2->getMassInverse(); - Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // --------------- Translation Constraints --------------- // @@ -295,20 +306,25 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS (-JvTranslation - mBTranslation); mImpulseTranslation += deltaLambdaTranslation; - // Compute the impulse P=J^T * lambda - Vector3 linearImpulseBody1 = -deltaLambdaTranslation; - Vector3 angularImpulseBody1 = deltaLambdaTranslation.cross(mR1World); - Vector3 linearImpulseBody2 = deltaLambdaTranslation; - Vector3 angularImpulseBody2 = -deltaLambdaTranslation.cross(mR2World); - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 linearImpulseBody1 = -deltaLambdaTranslation; + const Vector3 angularImpulseBody1 = deltaLambdaTranslation.cross(mR1World); + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; - w1 += I1 * angularImpulseBody1; + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 linearImpulseBody2 = deltaLambdaTranslation; + const Vector3 angularImpulseBody2 = -deltaLambdaTranslation.cross(mR2World); + + // Apply the impulse to the body v2 += inverseMassBody2 * linearImpulseBody2; - w2 += I2 * angularImpulseBody2; + w2 += mI2 * angularImpulseBody2; } // --------------- Rotation Constraints --------------- // @@ -321,16 +337,23 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS Vector2 deltaLambdaRotation = mInverseMassMatrixRotation * (-JvRotation - mBRotation); mImpulseRotation += deltaLambdaRotation; - // Compute the impulse P=J^T * lambda for the 2 rotation constraints - angularImpulseBody1 = -mB2CrossA1 * deltaLambdaRotation.x - mC2CrossA1 * deltaLambdaRotation.y; - angularImpulseBody2 = -angularImpulseBody1; - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { - w1 += I1 * angularImpulseBody1; + + // Compute the impulse P=J^T * lambda for the 2 rotation constraints + const Vector3 angularImpulseBody1 = -mB2CrossA1 * deltaLambdaRotation.x - + mC2CrossA1 * deltaLambdaRotation.y; + + // Apply the impulse to the body + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { - w2 += I2 * angularImpulseBody2; + + // Compute the impulse P=J^T * lambda for the 2 rotation constraints + const Vector3 angularImpulseBody2 = mB2CrossA1 * deltaLambdaRotation.x + + mC2CrossA1 * deltaLambdaRotation.y; + + // Apply the impulse to the body + w2 += mI2 * angularImpulseBody2; } // --------------- Limits Constraints --------------- // @@ -344,21 +367,26 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS const decimal JvLowerLimit = (w2 - w1).dot(mA1); // Compute the Lagrange multiplier lambda for the lower limit constraint - decimal deltaLambdaLower = mInverseMassMatrixLimitMotor * (-JvLowerLimit - mBLowerLimit); + decimal deltaLambdaLower = mInverseMassMatrixLimitMotor * (-JvLowerLimit -mBLowerLimit); decimal lambdaTemp = mImpulseLowerLimit; mImpulseLowerLimit = std::max(mImpulseLowerLimit + deltaLambdaLower, decimal(0.0)); deltaLambdaLower = mImpulseLowerLimit - lambdaTemp; - // Compute the impulse P=J^T * lambda for the lower limit constraint - const Vector3 angularImpulseBody1 = -deltaLambdaLower * mA1; - const Vector3 angularImpulseBody2 = -angularImpulseBody1; - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { - w1 += I1 * angularImpulseBody1; + + // Compute the impulse P=J^T * lambda for the lower limit constraint + const Vector3 angularImpulseBody1 = -deltaLambdaLower * mA1; + + // Apply the impulse to the body + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { - w2 += I2 * angularImpulseBody2; + + // Compute the impulse P=J^T * lambda for the lower limit constraint + const Vector3 angularImpulseBody2 = deltaLambdaLower * mA1; + + // Apply the impulse to the body + w2 += mI2 * angularImpulseBody2; } } @@ -374,16 +402,21 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS mImpulseUpperLimit = std::max(mImpulseUpperLimit + deltaLambdaUpper, decimal(0.0)); deltaLambdaUpper = mImpulseUpperLimit - lambdaTemp; - // Compute the impulse P=J^T * lambda for the upper limit constraint - const Vector3 angularImpulseBody1 = deltaLambdaUpper * mA1; - const Vector3 angularImpulseBody2 = -angularImpulseBody1; - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { - w1 += I1 * angularImpulseBody1; + + // Compute the impulse P=J^T * lambda for the upper limit constraint + const Vector3 angularImpulseBody1 = deltaLambdaUpper * mA1; + + // Apply the impulse to the body + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { - w2 += I2 * angularImpulseBody2; + + // Compute the impulse P=J^T * lambda for the upper limit constraint + const Vector3 angularImpulseBody2 = -deltaLambdaUpper * mA1; + + // Apply the impulse to the body + w2 += mI2 * angularImpulseBody2; } } } @@ -396,29 +429,288 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS const decimal JvMotor = mA1.dot(w1 - w2); // Compute the Lagrange multiplier lambda for the motor - const decimal maxMotorImpulse = mMaxMotorForce * constraintSolverData.timeStep; + const decimal maxMotorImpulse = mMaxMotorTorque * constraintSolverData.timeStep; decimal deltaLambdaMotor = mInverseMassMatrixLimitMotor * (-JvMotor - mMotorSpeed); decimal lambdaTemp = mImpulseMotor; mImpulseMotor = clamp(mImpulseMotor + deltaLambdaMotor, -maxMotorImpulse, maxMotorImpulse); deltaLambdaMotor = mImpulseMotor - lambdaTemp; - // Compute the impulse P=J^T * lambda for the motor - const Vector3 angularImpulseBody1 = -deltaLambdaMotor * mA1; - const Vector3 angularImpulseBody2 = -angularImpulseBody1; - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { - w1 += I1 * angularImpulseBody1; + + // Compute the impulse P=J^T * lambda for the motor + const Vector3 angularImpulseBody1 = -deltaLambdaMotor * mA1; + + // Apply the impulse to the body + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { - w2 += I2 * angularImpulseBody2; + + // Compute the impulse P=J^T * lambda for the motor + const Vector3 angularImpulseBody2 = deltaLambdaMotor * mA1; + + // Apply the impulse to the body + w2 += mI2 * angularImpulseBody2; } } } -// Solve the position constraint +// Solve the position constraint (for position error correction) void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { + // If the error position correction technique is not the non-linear-gauss-seidel, we do + // do not execute this method + if (mPositionCorrectionTechnique != NON_LINEAR_GAUSS_SEIDEL) return; + + // Get the bodies positions and orientations + Vector3& x1 = constraintSolverData.positions[mIndexBody1]; + Vector3& x2 = constraintSolverData.positions[mIndexBody2]; + Quaternion& q1 = constraintSolverData.orientations[mIndexBody1]; + Quaternion& q2 = constraintSolverData.orientations[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + decimal inverseMassBody1 = mBody1->getMassInverse(); + decimal inverseMassBody2 = mBody2->getMassInverse(); + + // Recompute the inverse inertia tensors + mI1 = mBody1->getInertiaTensorInverseWorld(); + mI2 = mBody2->getInertiaTensorInverseWorld(); + + // Compute the vector from body center to the anchor point in world-space + mR1World = q1 * mLocalAnchorPointBody1; + mR2World = q2 * mLocalAnchorPointBody2; + + // Compute the current angle around the hinge axis + decimal hingeAngle = computeCurrentHingeAngle(q1, q2); + + // Check if the limit constraints are violated or not + decimal lowerLimitError = hingeAngle - mLowerLimit; + decimal upperLimitError = mUpperLimit - hingeAngle; + mIsLowerLimitViolated = lowerLimitError <= 0; + mIsUpperLimitViolated = upperLimitError <= 0; + + // Compute vectors needed in the Jacobian + mA1 = q1 * mHingeLocalAxisBody1; + Vector3 a2 = q2 * mHingeLocalAxisBody2; + mA1.normalize(); + a2.normalize(); + const Vector3 b2 = a2.getOneUnitOrthogonalVector(); + const Vector3 c2 = a2.cross(b2); + mB2CrossA1 = b2.cross(mA1); + mC2CrossA1 = c2.cross(mA1); + + // Compute the corresponding skew-symmetric matrices + Matrix3x3 skewSymmetricMatrixU1= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR1World); + Matrix3x3 skewSymmetricMatrixU2= Matrix3x3::computeSkewSymmetricMatrixForCrossProduct(mR2World); + + // --------------- Translation Constraints --------------- // + + // Compute the matrix K=JM^-1J^t (3x3 matrix) for the 3 translation constraints + decimal inverseMassBodies = 0.0; + if (mBody1->getIsMotionEnabled()) { + inverseMassBodies += mBody1->getMassInverse(); + } + if (mBody2->getIsMotionEnabled()) { + inverseMassBodies += mBody2->getMassInverse(); + } + Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, + 0, inverseMassBodies, 0, + 0, 0, inverseMassBodies); + if (mBody1->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); + } + if (mBody2->getIsMotionEnabled()) { + massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); + } + mInverseMassMatrixTranslation.setToZero(); + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixTranslation = massMatrix.getInverse(); + } + + // Compute position error for the 3 translation constraints + const Vector3 errorTranslation = x2 + mR2World - x1 - mR1World; + + // Compute the Lagrange multiplier lambda + const Vector3 lambdaTranslation = mInverseMassMatrixTranslation * (-errorTranslation); + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse + Vector3 linearImpulseBody1 = -lambdaTranslation; + Vector3 angularImpulseBody1 = lambdaTranslation.cross(mR1World); + + // Compute the pseudo velocity + const Vector3 v1 = inverseMassBody1 * linearImpulseBody1; + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + x1 += v1; + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse + Vector3 linearImpulseBody2 = lambdaTranslation; + Vector3 angularImpulseBody2 = -lambdaTranslation.cross(mR2World); + + // Compute the pseudo velocity + const Vector3 v2 = inverseMassBody2 * linearImpulseBody2; + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + x2 += v2; + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); + } + + // --------------- Rotation Constraints --------------- // + + // Compute the inverse mass matrix K=JM^-1J^t for the 2 rotation constraints (2x2 matrix) + Vector3 I1B2CrossA1(0, 0, 0); + Vector3 I1C2CrossA1(0, 0, 0); + Vector3 I2B2CrossA1(0, 0, 0); + Vector3 I2C2CrossA1(0, 0, 0); + if (mBody1->getIsMotionEnabled()) { + I1B2CrossA1 = mI1 * mB2CrossA1; + I1C2CrossA1 = mI1 * mC2CrossA1; + } + if (mBody2->getIsMotionEnabled()) { + I2B2CrossA1 = mI2 * mB2CrossA1; + I2C2CrossA1 = mI2 * mC2CrossA1; + } + const decimal el11 = mB2CrossA1.dot(I1B2CrossA1) + + mB2CrossA1.dot(I2B2CrossA1); + const decimal el12 = mB2CrossA1.dot(I1C2CrossA1) + + mB2CrossA1.dot(I2C2CrossA1); + const decimal el21 = mC2CrossA1.dot(I1B2CrossA1) + + mC2CrossA1.dot(I2B2CrossA1); + const decimal el22 = mC2CrossA1.dot(I1C2CrossA1) + + mC2CrossA1.dot(I2C2CrossA1); + const Matrix2x2 matrixKRotation(el11, el12, el21, el22); + mInverseMassMatrixRotation.setToZero(); + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixRotation = matrixKRotation.getInverse(); + } + + // Compute the position error for the 3 rotation constraints + const Vector2 errorRotation = Vector2(mA1.dot(b2), mA1.dot(c2)); + + // Compute the Lagrange multiplier lambda for the 3 rotation constraints + Vector2 lambdaRotation = mInverseMassMatrixRotation * (-errorRotation); + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + const Vector3 angularImpulseBody1 = -mB2CrossA1 * lambdaRotation.x - + mC2CrossA1 * lambdaRotation.y; + + // Compute the pseudo velocity + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse + const Vector3 angularImpulseBody2 = mB2CrossA1 * lambdaRotation.x + + mC2CrossA1 * lambdaRotation.y; + + // Compute the pseudo velocity + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); + } + + // --------------- Limits Constraints --------------- // + + if (mIsLimitEnabled) { + + if (mIsLowerLimitViolated || mIsUpperLimitViolated) { + + // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) + mInverseMassMatrixLimitMotor = 0.0; + if (mBody1->getIsMotionEnabled()) { + mInverseMassMatrixLimitMotor += mA1.dot(mI1 * mA1); + } + if (mBody2->getIsMotionEnabled()) { + mInverseMassMatrixLimitMotor += mA1.dot(mI2 * mA1); + } + mInverseMassMatrixLimitMotor = (mInverseMassMatrixLimitMotor > 0.0) ? + decimal(1.0) / mInverseMassMatrixLimitMotor : decimal(0.0); + } + + // If the lower limit is violated + if (mIsLowerLimitViolated) { + + // Compute the Lagrange multiplier lambda for the lower limit constraint + decimal lambdaLowerLimit = mInverseMassMatrixLimitMotor * (-lowerLimitError ); + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 angularImpulseBody1 = -lambdaLowerLimit * mA1; + + // Compute the pseudo velocity + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 angularImpulseBody2 = lambdaLowerLimit * mA1; + + // Compute the pseudo velocity + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); + } + } + + // If the upper limit is violated + if (mIsUpperLimitViolated) { + + // Compute the Lagrange multiplier lambda for the upper limit constraint + decimal lambdaUpperLimit = mInverseMassMatrixLimitMotor * (-upperLimitError); + + // Apply the impulse to the bodies of the joint + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 angularImpulseBody1 = lambdaUpperLimit * mA1; + + // Compute the pseudo velocity + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda + const Vector3 angularImpulseBody2 = -lambdaUpperLimit * mA1; + + // Compute the pseudo velocity + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); + } + } + } } @@ -492,13 +784,13 @@ void HingeJoint::setMotorSpeed(decimal motorSpeed) { } } -// Set the maximum motor force -void HingeJoint::setMaxMotorForce(decimal maxMotorForce) { +// Set the maximum motor torque +void HingeJoint::setMaxMotorTorque(decimal maxMotorTorque) { - if (maxMotorForce != mMaxMotorForce) { + if (maxMotorTorque != mMaxMotorTorque) { - assert(mMaxMotorForce >= 0.0); - mMaxMotorForce = maxMotorForce; + assert(mMaxMotorTorque >= 0.0); + mMaxMotorTorque = maxMotorTorque; // TODO : Wake up the bodies of the joint here when sleeping is implemented } diff --git a/src/constraint/HingeJoint.h b/src/constraint/HingeJoint.h index aa258159..33962f0c 100644 --- a/src/constraint/HingeJoint.h +++ b/src/constraint/HingeJoint.h @@ -66,8 +66,9 @@ struct HingeJointInfo : public ConstraintInfo { /// Motor speed (in radian/second) decimal motorSpeed; - /// Maximum motor force (in Newton) that can be applied to reach to desired motor speed - decimal maxMotorForce; + /// Maximum motor torque (in Newtons * meters) that can be applied to reach + /// to desired motor speed + decimal maxMotorTorque; /// Constructor without limits and without motor HingeJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, @@ -76,8 +77,8 @@ struct HingeJointInfo : public ConstraintInfo { : ConstraintInfo(rigidBody1, rigidBody2, HINGEJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), rotationAxisWorld(initRotationAxisWorld), isLimitEnabled(false), - isMotorEnabled(false), minAngleLimit(-1), maxAngleLimit(1), motorSpeed(0), - maxMotorForce(0) {} + isMotorEnabled(false), minAngleLimit(-1), maxAngleLimit(1), + motorSpeed(0), maxMotorTorque(0) {} /// Constructor with limits but without motor HingeJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, @@ -88,20 +89,21 @@ struct HingeJointInfo : public ConstraintInfo { anchorPointWorldSpace(initAnchorPointWorldSpace), rotationAxisWorld(initRotationAxisWorld), isLimitEnabled(true), isMotorEnabled(false), minAngleLimit(initMinAngleLimit), - maxAngleLimit(initMaxAngleLimit), motorSpeed(0), maxMotorForce(0) {} + maxAngleLimit(initMaxAngleLimit), motorSpeed(0), + maxMotorTorque(0) {} /// Constructor with limits and motor HingeJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace, const Vector3& initRotationAxisWorld, decimal initMinAngleLimit, decimal initMaxAngleLimit, - decimal initMotorSpeed, decimal initMaxMotorForce) + decimal initMotorSpeed, decimal initMaxMotorTorque) : ConstraintInfo(rigidBody1, rigidBody2, HINGEJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), rotationAxisWorld(initRotationAxisWorld), isLimitEnabled(true), isMotorEnabled(false), minAngleLimit(initMinAngleLimit), maxAngleLimit(initMaxAngleLimit), motorSpeed(initMotorSpeed), - maxMotorForce(initMaxMotorForce) {} + maxMotorTorque(initMaxMotorTorque) {} }; // Class HingeJoint @@ -132,6 +134,12 @@ class HingeJoint : public Constraint { /// Hinge rotation axis (in local-space coordiantes of body 2) Vector3 mHingeLocalAxisBody2; + /// Inertia tensor of body 1 (in world-space coordinates) + Matrix3x3 mI1; + + /// Inertia tensor of body 2 (in world-space coordinates) + Matrix3x3 mI2; + /// Hinge rotation axis (in world-space coordinates) computed from body 1 Vector3 mA1; @@ -210,15 +218,16 @@ class HingeJoint : public Constraint { /// Motor speed decimal mMotorSpeed; - /// Maximum motor force (in Newton) that can be applied to reach to desired motor speed - decimal mMaxMotorForce; + /// Maximum motor torque (in Newtons) that can be applied to reach to desired motor speed + decimal mMaxMotorTorque; // -------------------- Methods -------------------- // /// Reset the limits void resetLimits(); - /// Given an angle in radian, this method returns the corresponding angle in the range [-pi; pi] + /// Given an angle in radian, this method returns the corresponding + /// angle in the range [-pi; pi] decimal computeNormalizedAngle(decimal angle) const; /// Given an "inputAngle" in the range [-pi, pi], this method returns an @@ -271,14 +280,14 @@ class HingeJoint : public Constraint { /// Set the motor speed void setMotorSpeed(decimal motorSpeed); - /// Return the maximum motor force - decimal getMaxMotorForce() const; + /// Return the maximum motor torque + decimal getMaxMotorTorque() const; - /// Set the maximum motor force - void setMaxMotorForce(decimal maxMotorForce); + /// Set the maximum motor torque + void setMaxMotorTorque(decimal maxMotorTorque); - /// Return the intensity of the current force applied for the joint motor - decimal getMotorForce(decimal timeStep) const; + /// Return the intensity of the current torque applied for the joint motor + decimal getMotorTorque(decimal timeStep) const; /// Return the number of bytes used by the joint virtual size_t getSizeInBytes() const; @@ -292,7 +301,7 @@ class HingeJoint : public Constraint { /// Solve the velocity constraint virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); - /// Solve the position constraint + /// Solve the position constraint (for position error correction) virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); }; @@ -321,13 +330,13 @@ inline decimal HingeJoint::getMotorSpeed() const { return mMotorSpeed; } -// Return the maximum motor force -inline decimal HingeJoint::getMaxMotorForce() const { - return mMaxMotorForce; +// Return the maximum motor torque +inline decimal HingeJoint::getMaxMotorTorque() const { + return mMaxMotorTorque; } -// Return the intensity of the current force applied for the joint motor -inline decimal HingeJoint::getMotorForce(decimal timeStep) const { +// Return the intensity of the current torque applied for the joint motor +inline decimal HingeJoint::getMotorTorque(decimal timeStep) const { return mImpulseMotor / timeStep; } diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index c9586a61..e1f2daed 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -36,9 +36,10 @@ SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) : Constraint(jointInfo), mImpulseTranslation(0, 0), mImpulseRotation(0, 0, 0), mImpulseLowerLimit(0), mImpulseUpperLimit(0), mImpulseMotor(0), mIsLimitEnabled(jointInfo.isLimitEnabled), mIsMotorEnabled(jointInfo.isMotorEnabled), - mLowerLimit(jointInfo.minTranslationLimit), mUpperLimit(jointInfo.maxTranslationLimit), - mIsLowerLimitViolated(false), mIsUpperLimitViolated(false), - mMotorSpeed(jointInfo.motorSpeed), mMaxMotorForce(jointInfo.maxMotorForce){ + mLowerLimit(jointInfo.minTranslationLimit), + mUpperLimit(jointInfo.maxTranslationLimit), mIsLowerLimitViolated(false), + mIsUpperLimitViolated(false), mMotorSpeed(jointInfo.motorSpeed), + mMaxMotorForce(jointInfo.maxMotorForce){ assert(mUpperLimit >= 0.0); assert(mLowerLimit <= 0.0); @@ -81,8 +82,8 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa const Quaternion& orientationBody2 = mBody2->getTransform().getOrientation(); // Get the inertia tensor of bodies - const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); + mI1 = mBody1->getInertiaTensorInverseWorld(); + mI2 = mBody2->getInertiaTensorInverseWorld(); // Vector from body center to the anchor point mR1 = orientationBody1 * mLocalAnchorPointBody1; @@ -130,13 +131,13 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa Vector3 I2R2CrossN2(0, 0, 0); if (mBody1->getIsMotionEnabled()) { sumInverseMass += mBody1->getMassInverse(); - I1R1PlusUCrossN1 = I1 * mR1PlusUCrossN1; - I1R1PlusUCrossN2 = I1 * mR1PlusUCrossN2; + I1R1PlusUCrossN1 = mI1 * mR1PlusUCrossN1; + I1R1PlusUCrossN2 = mI1 * mR1PlusUCrossN2; } if (mBody2->getIsMotionEnabled()) { sumInverseMass += mBody2->getMassInverse(); - I2R2CrossN1 = I2 * mR2CrossN1; - I2R2CrossN2 = I2 * mR2CrossN2; + I2R2CrossN1 = mI2 * mR2CrossN1; + I2R2CrossN2 = mI2 * mR2CrossN2; } const decimal el11 = sumInverseMass + mR1PlusUCrossN1.dot(I1R1PlusUCrossN1) + mR2CrossN1.dot(I2R2CrossN1); @@ -165,10 +166,10 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa // contraints (3x3 matrix) mInverseMassMatrixRotationConstraint.setToZero(); if (mBody1->getIsMotionEnabled()) { - mInverseMassMatrixRotationConstraint += I1; + mInverseMassMatrixRotationConstraint += mI1; } if (mBody2->getIsMotionEnabled()) { - mInverseMassMatrixRotationConstraint += I2; + mInverseMassMatrixRotationConstraint += mI2; } if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { mInverseMassMatrixRotationConstraint = mInverseMassMatrixRotationConstraint.getInverse(); @@ -189,11 +190,11 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa mInverseMassMatrixLimit = 0.0; if (mBody1->getIsMotionEnabled()) { mInverseMassMatrixLimit += mBody1->getMassInverse() + - mR1PlusUCrossSliderAxis.dot(I1 * mR1PlusUCrossSliderAxis); + mR1PlusUCrossSliderAxis.dot(mI1 * mR1PlusUCrossSliderAxis); } if (mBody2->getIsMotionEnabled()) { mInverseMassMatrixLimit += mBody2->getMassInverse() + - mR2CrossSliderAxis.dot(I2 * mR2CrossSliderAxis); + mR2CrossSliderAxis.dot(mI2 * mR2CrossSliderAxis); } mInverseMassMatrixLimit = (mInverseMassMatrixLimit > 0.0) ? decimal(1.0) / mInverseMassMatrixLimit : decimal(0.0); @@ -246,42 +247,55 @@ void SliderJoint::warmstart(const ConstraintSolverData& constraintSolverData) { // Get the inverse mass and inverse inertia tensors of the bodies const decimal inverseMassBody1 = mBody1->getMassInverse(); const decimal inverseMassBody2 = mBody2->getMassInverse(); - const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); - - // Compute the impulse P=J^T * lambda for the 2 translation constraints - Vector3 linearImpulseBody1 = -mN1 * mImpulseTranslation.x - mN2 * mImpulseTranslation.y; - Vector3 angularImpulseBody1 = -mR1PlusUCrossN1 * mImpulseTranslation.x - - mR1PlusUCrossN2 * mImpulseTranslation.y; - Vector3 linearImpulseBody2 = -linearImpulseBody1; - Vector3 angularImpulseBody2 = mR2CrossN1 * mImpulseTranslation.x + - mR2CrossN2 * mImpulseTranslation.y; - - // Compute the impulse P=J^T * lambda for the 3 rotation constraints - angularImpulseBody1 += -mImpulseRotation; - angularImpulseBody2 += mImpulseRotation; // Compute the impulse P=J^T * lambda for the lower and upper limits constraints decimal impulseLimits = mImpulseUpperLimit - mImpulseLowerLimit; Vector3 linearImpulseLimits = impulseLimits * mSliderAxisWorld; - linearImpulseBody1 += linearImpulseLimits; - angularImpulseBody1 += impulseLimits * mR1PlusUCrossSliderAxis; - linearImpulseBody2 += -linearImpulseLimits; - angularImpulseBody2 += -impulseLimits * mR2CrossSliderAxis; // Compute the impulse P=J^T * lambda for the motor constraint Vector3 impulseMotor = mImpulseMotor * mSliderAxisWorld; - linearImpulseBody1 += impulseMotor; - linearImpulseBody2 += -impulseMotor; - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 2 translation constraints + Vector3 linearImpulseBody1 = -mN1 * mImpulseTranslation.x - mN2 * mImpulseTranslation.y; + Vector3 angularImpulseBody1 = -mR1PlusUCrossN1 * mImpulseTranslation.x - + mR1PlusUCrossN2 * mImpulseTranslation.y; + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + angularImpulseBody1 += -mImpulseRotation; + + // Compute the impulse P=J^T * lambda for the lower and upper limits constraints + linearImpulseBody1 += linearImpulseLimits; + angularImpulseBody1 += impulseLimits * mR1PlusUCrossSliderAxis; + + // Compute the impulse P=J^T * lambda for the motor constraint + linearImpulseBody1 += impulseMotor; + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; - w1 += I1 * angularImpulseBody1; + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 2 translation constraints + Vector3 linearImpulseBody2 = mN1 * mImpulseTranslation.x + mN2 * mImpulseTranslation.y; + Vector3 angularImpulseBody2 = mR2CrossN1 * mImpulseTranslation.x + + mR2CrossN2 * mImpulseTranslation.y; + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + angularImpulseBody2 += mImpulseRotation; + + // Compute the impulse P=J^T * lambda for the lower and upper limits constraints + linearImpulseBody2 += -linearImpulseLimits; + angularImpulseBody2 += -impulseLimits * mR2CrossSliderAxis; + + // Compute the impulse P=J^T * lambda for the motor constraint + linearImpulseBody2 += -impulseMotor; + + // Apply the impulse to the body v2 += inverseMassBody2 * linearImpulseBody2; - w2 += I2 * angularImpulseBody2; + w2 += mI2 * angularImpulseBody2; } } @@ -297,8 +311,6 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint // Get the inverse mass and inverse inertia tensors of the bodies decimal inverseMassBody1 = mBody1->getMassInverse(); decimal inverseMassBody2 = mBody2->getMassInverse(); - const Matrix3x3 I1 = mBody1->getInertiaTensorInverseWorld(); - const Matrix3x3 I2 = mBody2->getInertiaTensorInverseWorld(); // --------------- Translation Constraints --------------- // @@ -313,20 +325,26 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint Vector2 deltaLambda = mInverseMassMatrixTranslationConstraint * (-JvTranslation -mBTranslation); mImpulseTranslation += deltaLambda; - // Compute the impulse P=J^T * lambda for the 2 translation constraints - Vector3 linearImpulseBody1 = -mN1 * deltaLambda.x - mN2 * deltaLambda.y; - Vector3 angularImpulseBody1 = -mR1PlusUCrossN1 * deltaLambda.x -mR1PlusUCrossN2 * deltaLambda.y; - Vector3 linearImpulseBody2 = -linearImpulseBody1; - Vector3 angularImpulseBody2 = mR2CrossN1 * deltaLambda.x + mR2CrossN2 * deltaLambda.y; - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 2 translation constraints + const Vector3 linearImpulseBody1 = -mN1 * deltaLambda.x - mN2 * deltaLambda.y; + const Vector3 angularImpulseBody1 = -mR1PlusUCrossN1 * deltaLambda.x - + mR1PlusUCrossN2 * deltaLambda.y; + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; - w1 += I1 * angularImpulseBody1; + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 2 translation constraints + const Vector3 linearImpulseBody2 = mN1 * deltaLambda.x + mN2 * deltaLambda.y; + const Vector3 angularImpulseBody2 = mR2CrossN1 * deltaLambda.x + mR2CrossN2 * deltaLambda.y; + + // Apply the impulse to the body v2 += inverseMassBody2 * linearImpulseBody2; - w2 += I2 * angularImpulseBody2; + w2 += mI2 * angularImpulseBody2; } // --------------- Rotation Constraints --------------- // @@ -338,16 +356,21 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint Vector3 deltaLambda2 = mInverseMassMatrixRotationConstraint * (-JvRotation - mBRotation); mImpulseRotation += deltaLambda2; - // Compute the impulse P=J^T * lambda for the 3 rotation constraints - angularImpulseBody1 = -deltaLambda2; - angularImpulseBody2 = deltaLambda2; - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { - w1 += I1 * angularImpulseBody1; + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + const Vector3 angularImpulseBody1 = -deltaLambda2; + + // Apply the impulse to the body + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { - w2 += I2 * angularImpulseBody2; + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + const Vector3 angularImpulseBody2 = deltaLambda2; + + // Apply the impulse to the body + w2 += mI2 * angularImpulseBody2; } // --------------- Limits Constraints --------------- // @@ -367,20 +390,25 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint mImpulseLowerLimit = std::max(mImpulseLowerLimit + deltaLambdaLower, decimal(0.0)); deltaLambdaLower = mImpulseLowerLimit - lambdaTemp; - // Compute the impulse P=J^T * lambda for the lower limit constraint - const Vector3 linearImpulseBody1 = -deltaLambdaLower * mSliderAxisWorld; - const Vector3 angularImpulseBody1 = -deltaLambdaLower * mR1PlusUCrossSliderAxis; - const Vector3 linearImpulseBody2 = -linearImpulseBody1; - const Vector3 angularImpulseBody2 = deltaLambdaLower * mR2CrossSliderAxis; - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the lower limit constraint + const Vector3 linearImpulseBody1 = -deltaLambdaLower * mSliderAxisWorld; + const Vector3 angularImpulseBody1 = -deltaLambdaLower * mR1PlusUCrossSliderAxis; + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; - w1 += I1 * angularImpulseBody1; + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the lower limit constraint + const Vector3 linearImpulseBody2 = deltaLambdaLower * mSliderAxisWorld; + const Vector3 angularImpulseBody2 = deltaLambdaLower * mR2CrossSliderAxis; + + // Apply the impulse to the body v2 += inverseMassBody2 * linearImpulseBody2; - w2 += I2 * angularImpulseBody2; + w2 += mI2 * angularImpulseBody2; } } @@ -397,20 +425,25 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint mImpulseUpperLimit = std::max(mImpulseUpperLimit + deltaLambdaUpper, decimal(0.0)); deltaLambdaUpper = mImpulseUpperLimit - lambdaTemp; - // Compute the impulse P=J^T * lambda for the upper limit constraint - const Vector3 linearImpulseBody1 = deltaLambdaUpper * mSliderAxisWorld; - const Vector3 angularImpulseBody1 = deltaLambdaUpper * mR1PlusUCrossSliderAxis; - const Vector3 linearImpulseBody2 = -linearImpulseBody1; - const Vector3 angularImpulseBody2 = -deltaLambdaUpper * mR2CrossSliderAxis; - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the upper limit constraint + const Vector3 linearImpulseBody1 = deltaLambdaUpper * mSliderAxisWorld; + const Vector3 angularImpulseBody1 = deltaLambdaUpper * mR1PlusUCrossSliderAxis; + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; - w1 += I1 * angularImpulseBody1; + w1 += mI1 * angularImpulseBody1; } if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the upper limit constraint + const Vector3 linearImpulseBody2 = -deltaLambdaUpper * mSliderAxisWorld; + const Vector3 angularImpulseBody2 = -deltaLambdaUpper * mR2CrossSliderAxis; + + // Apply the impulse to the body v2 += inverseMassBody2 * linearImpulseBody2; - w2 += I2 * angularImpulseBody2; + w2 += mI2 * angularImpulseBody2; } } } @@ -429,23 +462,292 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint mImpulseMotor = clamp(mImpulseMotor + deltaLambdaMotor, -maxMotorImpulse, maxMotorImpulse); deltaLambdaMotor = mImpulseMotor - lambdaTemp; - // Compute the impulse P=J^T * lambda for the motor - const Vector3 linearImpulseBody1 = deltaLambdaMotor * mSliderAxisWorld; - const Vector3 linearImpulseBody2 = -linearImpulseBody1; - - // Apply the impulse to the bodies of the joint if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the motor + const Vector3 linearImpulseBody1 = deltaLambdaMotor * mSliderAxisWorld; + + // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; } if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the motor + const Vector3 linearImpulseBody2 = -deltaLambdaMotor * mSliderAxisWorld; + + // Apply the impulse to the body v2 += inverseMassBody2 * linearImpulseBody2; } } } -// Solve the position constraint +// Solve the position constraint (for position error correction) void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { + // If the error position correction technique is not the non-linear-gauss-seidel, we do + // do not execute this method + if (mPositionCorrectionTechnique != NON_LINEAR_GAUSS_SEIDEL) return; + + // Get the bodies positions and orientations + Vector3& x1 = constraintSolverData.positions[mIndexBody1]; + Vector3& x2 = constraintSolverData.positions[mIndexBody2]; + Quaternion& q1 = constraintSolverData.orientations[mIndexBody1]; + Quaternion& q2 = constraintSolverData.orientations[mIndexBody2]; + + // Get the inverse mass and inverse inertia tensors of the bodies + decimal inverseMassBody1 = mBody1->getMassInverse(); + decimal inverseMassBody2 = mBody2->getMassInverse(); + + // Recompute the inertia tensor of bodies + mI1 = mBody1->getInertiaTensorInverseWorld(); + mI2 = mBody2->getInertiaTensorInverseWorld(); + + // Vector from body center to the anchor point + mR1 = q1 * mLocalAnchorPointBody1; + mR2 = q2 * mLocalAnchorPointBody2; + + // Compute the vector u (difference between anchor points) + const Vector3 u = x2 + mR2 - x1 - mR1; + + // Compute the two orthogonal vectors to the slider axis in world-space + mSliderAxisWorld = q1 * mSliderAxisBody1; + mSliderAxisWorld.normalize(); + mN1 = mSliderAxisWorld.getOneUnitOrthogonalVector(); + mN2 = mSliderAxisWorld.cross(mN1); + + // Check if the limit constraints are violated or not + decimal uDotSliderAxis = u.dot(mSliderAxisWorld); + decimal lowerLimitError = uDotSliderAxis - mLowerLimit; + decimal upperLimitError = mUpperLimit - uDotSliderAxis; + mIsLowerLimitViolated = lowerLimitError <= 0; + mIsUpperLimitViolated = upperLimitError <= 0; + + // Compute the cross products used in the Jacobians + mR2CrossN1 = mR2.cross(mN1); + mR2CrossN2 = mR2.cross(mN2); + mR2CrossSliderAxis = mR2.cross(mSliderAxisWorld); + const Vector3 r1PlusU = mR1 + u; + mR1PlusUCrossN1 = (r1PlusU).cross(mN1); + mR1PlusUCrossN2 = (r1PlusU).cross(mN2); + mR1PlusUCrossSliderAxis = (r1PlusU).cross(mSliderAxisWorld); + + // --------------- Translation Constraints --------------- // + + // Recompute the inverse of the mass matrix K=JM^-1J^t for the 2 translation + // constraints (2x2 matrix) + decimal sumInverseMass = 0.0; + Vector3 I1R1PlusUCrossN1(0, 0, 0); + Vector3 I1R1PlusUCrossN2(0, 0, 0); + Vector3 I2R2CrossN1(0, 0, 0); + Vector3 I2R2CrossN2(0, 0, 0); + if (mBody1->getIsMotionEnabled()) { + sumInverseMass += mBody1->getMassInverse(); + I1R1PlusUCrossN1 = mI1 * mR1PlusUCrossN1; + I1R1PlusUCrossN2 = mI1 * mR1PlusUCrossN2; + } + if (mBody2->getIsMotionEnabled()) { + sumInverseMass += mBody2->getMassInverse(); + I2R2CrossN1 = mI2 * mR2CrossN1; + I2R2CrossN2 = mI2 * mR2CrossN2; + } + const decimal el11 = sumInverseMass + mR1PlusUCrossN1.dot(I1R1PlusUCrossN1) + + mR2CrossN1.dot(I2R2CrossN1); + const decimal el12 = mR1PlusUCrossN1.dot(I1R1PlusUCrossN2) + + mR2CrossN1.dot(I2R2CrossN2); + const decimal el21 = mR1PlusUCrossN2.dot(I1R1PlusUCrossN1) + + mR2CrossN2.dot(I2R2CrossN1); + const decimal el22 = sumInverseMass + mR1PlusUCrossN2.dot(I1R1PlusUCrossN2) + + mR2CrossN2.dot(I2R2CrossN2); + Matrix2x2 matrixKTranslation(el11, el12, el21, el22); + mInverseMassMatrixTranslationConstraint.setToZero(); + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixTranslationConstraint = matrixKTranslation.getInverse(); + } + + // Compute the position error for the 2 translation constraints + const Vector2 translationError(u.dot(mN1), u.dot(mN2)); + + // Compute the Lagrange multiplier lambda for the 2 translation constraints + Vector2 lambdaTranslation = mInverseMassMatrixTranslationConstraint * (-translationError); + + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 2 translation constraints + const Vector3 linearImpulseBody1 = -mN1 * lambdaTranslation.x - mN2 * lambdaTranslation.y; + const Vector3 angularImpulseBody1 = -mR1PlusUCrossN1 * lambdaTranslation.x - + mR1PlusUCrossN2 * lambdaTranslation.y; + + // Apply the impulse to the body + const Vector3 v1 = inverseMassBody1 * linearImpulseBody1; + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + x1 += v1; + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 2 translation constraints + const Vector3 linearImpulseBody2 = mN1 * lambdaTranslation.x + mN2 * lambdaTranslation.y; + const Vector3 angularImpulseBody2 = mR2CrossN1 * lambdaTranslation.x + + mR2CrossN2 * lambdaTranslation.y; + + // Apply the impulse to the body + const Vector3 v2 = inverseMassBody2 * linearImpulseBody2; + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + x2 += v2; + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); + } + + // --------------- Rotation Constraints --------------- // + + // Compute the inverse of the mass matrix K=JM^-1J^t for the 3 rotation + // contraints (3x3 matrix) + mInverseMassMatrixRotationConstraint.setToZero(); + if (mBody1->getIsMotionEnabled()) { + mInverseMassMatrixRotationConstraint += mI1; + } + if (mBody2->getIsMotionEnabled()) { + mInverseMassMatrixRotationConstraint += mI2; + } + if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + mInverseMassMatrixRotationConstraint = mInverseMassMatrixRotationConstraint.getInverse(); + } + + // Compute the position error for the 3 rotation constraints + Quaternion currentOrientationDifference = q2 * q1.getInverse(); + currentOrientationDifference.normalize(); + const Quaternion qError = currentOrientationDifference * mInitOrientationDifferenceInv; + const Vector3 errorRotation = decimal(2.0) * qError.getVectorV(); + + // Compute the Lagrange multiplier lambda for the 3 rotation constraints + Vector3 lambdaRotation = mInverseMassMatrixRotationConstraint * (-errorRotation); + + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + const Vector3 angularImpulseBody1 = -lambdaRotation; + + // Apply the impulse to the body + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the 3 rotation constraints + const Vector3 angularImpulseBody2 = lambdaRotation; + + // Apply the impulse to the body + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); + } + + // --------------- Limits Constraints --------------- // + + if (mIsLimitEnabled) { + + if (mIsLowerLimitViolated || mIsUpperLimitViolated) { + + // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) + mInverseMassMatrixLimit = 0.0; + if (mBody1->getIsMotionEnabled()) { + mInverseMassMatrixLimit += mBody1->getMassInverse() + + mR1PlusUCrossSliderAxis.dot(mI1 * mR1PlusUCrossSliderAxis); + } + if (mBody2->getIsMotionEnabled()) { + mInverseMassMatrixLimit += mBody2->getMassInverse() + + mR2CrossSliderAxis.dot(mI2 * mR2CrossSliderAxis); + } + mInverseMassMatrixLimit = (mInverseMassMatrixLimit > 0.0) ? + decimal(1.0) / mInverseMassMatrixLimit : decimal(0.0); + } + + // If the lower limit is violated + if (mIsLowerLimitViolated) { + + // Compute the Lagrange multiplier lambda for the lower limit constraint + decimal lambdaLowerLimit = mInverseMassMatrixLimit * (-lowerLimitError); + + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the lower limit constraint + const Vector3 linearImpulseBody1 = -lambdaLowerLimit * mSliderAxisWorld; + const Vector3 angularImpulseBody1 = -lambdaLowerLimit * mR1PlusUCrossSliderAxis; + + // Apply the impulse to the body + const Vector3 v1 = inverseMassBody1 * linearImpulseBody1; + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + x1 += v1; + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the lower limit constraint + const Vector3 linearImpulseBody2 = lambdaLowerLimit * mSliderAxisWorld; + const Vector3 angularImpulseBody2 = lambdaLowerLimit * mR2CrossSliderAxis; + + // Apply the impulse to the body + const Vector3 v2 = inverseMassBody2 * linearImpulseBody2; + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + x2 += v2; + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); + } + } + + // If the upper limit is violated + if (mIsUpperLimitViolated) { + + // Compute the Lagrange multiplier lambda for the upper limit constraint + decimal lambdaUpperLimit = mInverseMassMatrixLimit * (-upperLimitError); + + if (mBody1->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the upper limit constraint + const Vector3 linearImpulseBody1 = lambdaUpperLimit * mSliderAxisWorld; + const Vector3 angularImpulseBody1 = lambdaUpperLimit * mR1PlusUCrossSliderAxis; + + // Apply the impulse to the body + const Vector3 v1 = inverseMassBody1 * linearImpulseBody1; + const Vector3 w1 = mI1 * angularImpulseBody1; + + // Update the body position/orientation + x1 += v1; + q1 += Quaternion(0, w1) * q1 * decimal(0.5); + q1.normalize(); + } + if (mBody2->getIsMotionEnabled()) { + + // Compute the impulse P=J^T * lambda for the upper limit constraint + const Vector3 linearImpulseBody2 = -lambdaUpperLimit * mSliderAxisWorld; + const Vector3 angularImpulseBody2 = -lambdaUpperLimit * mR2CrossSliderAxis; + + // Apply the impulse to the body + const Vector3 v2 = inverseMassBody2 * linearImpulseBody2; + const Vector3 w2 = mI2 * angularImpulseBody2; + + // Update the body position/orientation + x2 += v2; + q2 += Quaternion(0, w2) * q2 * decimal(0.5); + q2.normalize(); + } + } + } } // Enable/Disable the limits of the joint @@ -469,6 +771,30 @@ void SliderJoint::enableMotor(bool isMotorEnabled) { // TODO : Wake up the bodies of the joint here when sleeping is implemented } +// Return the current translation value of the joint +decimal SliderJoint::getTranslation() const { + + // Get the bodies positions and orientations + const Vector3& x1 = mBody1->getTransform().getPosition(); + const Vector3& x2 = mBody2->getTransform().getPosition(); + const Quaternion& q1 = mBody1->getTransform().getOrientation(); + const Quaternion& q2 = mBody2->getTransform().getOrientation(); + + // Compute the two anchor points in world-space coordinates + const Vector3 anchorBody1 = x1 + q1 * mLocalAnchorPointBody1; + const Vector3 anchorBody2 = x2 + q2 * mLocalAnchorPointBody2; + + // Compute the vector u (difference between anchor points) + const Vector3 u = anchorBody2 - anchorBody1; + + // Compute the slider axis in world-space + Vector3 sliderAxisWorld = q1 * mSliderAxisBody1; + sliderAxisWorld.normalize(); + + // Compute and return the translation value + return u.dot(sliderAxisWorld); +} + // Set the minimum translation limit void SliderJoint::setMinTranslationLimit(decimal lowerLimit) { diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h index 79834322..6515ffff 100644 --- a/src/constraint/SliderJoint.h +++ b/src/constraint/SliderJoint.h @@ -64,7 +64,7 @@ struct SliderJointInfo : public ConstraintInfo { /// Motor speed decimal motorSpeed; - /// Maximum motor force (in Newton) that can be applied to reach to desired motor speed + /// Maximum motor force (in Newtons) that can be applied to reach to desired motor speed decimal maxMotorForce; /// Constructor without limits and without motor @@ -129,6 +129,12 @@ class SliderJoint : public Constraint { /// Slider axis (in local-space coordinates of body 1) Vector3 mSliderAxisBody1; + /// Inertia tensor of body 1 (in world-space coordinates) + Matrix3x3 mI1; + + /// Inertia tensor of body 2 (in world-space coordinates) + Matrix3x3 mI2; + /// Inverse of the initial orientation difference between the two bodies Quaternion mInitOrientationDifferenceInv; @@ -225,7 +231,7 @@ class SliderJoint : public Constraint { /// Motor speed decimal mMotorSpeed; - /// Maximum motor force (in Newton) that can be applied to reach to desired motor speed + /// Maximum motor force (in Newtons) that can be applied to reach to desired motor speed decimal mMaxMotorForce; // -------------------- Methods -------------------- // @@ -255,6 +261,9 @@ class SliderJoint : public Constraint { /// Enable/Disable the motor of the joint void enableMotor(bool isMotorEnabled); + /// Return the current translation value of the joint + decimal getTranslation() const; + /// Return the minimum translation limit decimal getMinTranslationLimit() const; @@ -294,7 +303,7 @@ class SliderJoint : public Constraint { /// Solve the velocity constraint virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); - /// Solve the position constraint + /// Solve the position constraint (for position error correction) virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); }; diff --git a/src/engine/ConstraintSolver.cpp b/src/engine/ConstraintSolver.cpp index ff994a28..2154ceb0 100644 --- a/src/engine/ConstraintSolver.cpp +++ b/src/engine/ConstraintSolver.cpp @@ -33,14 +33,15 @@ using namespace reactphysics3d; ConstraintSolver::ConstraintSolver(std::set& joints, std::vector& linearVelocities, std::vector& angularVelocities, + std::vector& positions, + std::vector& orientations, const std::map& mapBodyToVelocityIndex) : mJoints(joints), mLinearVelocities(linearVelocities), - mAngularVelocities(angularVelocities), + mAngularVelocities(angularVelocities), mPositions(positions), + mOrientations(orientations), mMapBodyToConstrainedVelocityIndex(mapBodyToVelocityIndex), - mIsWarmStartingActive(true), - mIsNonLinearGaussSeidelPositionCorrectionActive(false), - mConstraintSolverData(linearVelocities, - angularVelocities, mapBodyToVelocityIndex){ + mIsWarmStartingActive(true), mConstraintSolverData(linearVelocities, + angularVelocities, positions, orientations, mapBodyToVelocityIndex){ } @@ -94,10 +95,8 @@ void ConstraintSolver::solveVelocityConstraints() { std::set::iterator it; for (it = mJoints.begin(); it != mJoints.end(); ++it) { - Constraint* joint = (*it); - // Solve the constraint - joint->solveVelocityConstraint(mConstraintSolverData); + (*it)->solveVelocityConstraint(mConstraintSolverData); } } @@ -110,9 +109,7 @@ void ConstraintSolver::solvePositionConstraints() { std::set::iterator it; for (it = mJoints.begin(); it != mJoints.end(); ++it) { - Constraint* joint = (*it); - // Solve the constraint - joint->solveVelocityConstraint(mConstraintSolverData); + (*it)->solvePositionConstraint(mConstraintSolverData); } } diff --git a/src/engine/ConstraintSolver.h b/src/engine/ConstraintSolver.h index fd9b3249..79cac7f7 100644 --- a/src/engine/ConstraintSolver.h +++ b/src/engine/ConstraintSolver.h @@ -53,6 +53,12 @@ struct ConstraintSolverData { /// Reference to the bodies angular velocities std::vector& angularVelocities; + /// Reference to the bodies positions + std::vector& positions; + + /// Reference to the bodies orientations + std::vector& orientations; + /// Reference to the map that associates rigid body to their index /// in the constrained velocities array const std::map& mapBodyToConstrainedVelocityIndex; @@ -63,9 +69,12 @@ struct ConstraintSolverData { /// Constructor ConstraintSolverData(std::vector& refLinearVelocities, std::vector& refAngularVelocities, + std::vector& refPositions, + std::vector& refOrientations, const std::map& refMapBodyToConstrainedVelocityIndex) :linearVelocities(refLinearVelocities), angularVelocities(refAngularVelocities), + positions(refPositions), orientations(refOrientations), mapBodyToConstrainedVelocityIndex(refMapBodyToConstrainedVelocityIndex){ } @@ -161,6 +170,12 @@ class ConstraintSolver { /// after solving the constraints) std::vector& mAngularVelocities; + /// Reference to the array of bodies positions (for position error correction) + std::vector& mPositions; + + /// Reference to the array of bodies orientations (for position error correction) + std::vector& mOrientations; + /// Reference to the map that associates rigid body to their index in /// the constrained velocities array const std::map& mMapBodyToConstrainedVelocityIndex; @@ -171,9 +186,6 @@ class ConstraintSolver { /// True if the warm starting of the solver is active bool mIsWarmStartingActive; - /// True if the Non-Linear-Gauss-Seidel position correction technique is enabled - bool mIsNonLinearGaussSeidelPositionCorrectionActive; - /// Constraint solver data used to initialize and solve the constraints ConstraintSolverData mConstraintSolverData; @@ -185,6 +197,8 @@ class ConstraintSolver { ConstraintSolver(std::set& joints, std::vector& linearVelocities, std::vector& angularVelocities, + std::vector& positions, + std::vector& orientations, const std::map& mapBodyToVelocityIndex); /// Destructor @@ -204,16 +218,14 @@ class ConstraintSolver { /// Enable/Disable the Non-Linear-Gauss-Seidel position correction technique. void setIsNonLinearGaussSeidelPositionCorrectionActive(bool isActive); + + /// Return true if the body is in at least one constraint + bool isConstrainedBody(RigidBody* body) const; }; -// Return true if the Non-Linear-Gauss-Seidel position correction technique is active -inline bool ConstraintSolver::getIsNonLinearGaussSeidelPositionCorrectionActive() const { - return mIsNonLinearGaussSeidelPositionCorrectionActive; -} - -// Enable/Disable the Non-Linear-Gauss-Seidel position correction technique. -inline void ConstraintSolver::setIsNonLinearGaussSeidelPositionCorrectionActive(bool isActive) { - mIsNonLinearGaussSeidelPositionCorrectionActive = isActive; +// Return true if the body is in at least one constraint +inline bool ConstraintSolver::isConstrainedBody(RigidBody* body) const { + return mConstraintBodies.count(body) == 1; } } diff --git a/src/engine/ContactSolver.h b/src/engine/ContactSolver.h index e40fb977..d1a3971f 100644 --- a/src/engine/ContactSolver.h +++ b/src/engine/ContactSolver.h @@ -442,6 +442,9 @@ class ContactSolver { /// Clean up the constraint solver void cleanup(); + /// Return true if the split impulses position correction technique is used for contacts + bool isSplitImpulseActive() const; + /// Activate or Deactivate the split impulses for contacts void setIsSplitImpulseActive(bool isActive); @@ -469,6 +472,11 @@ inline Vector3 ContactSolver::getSplitAngularVelocityOfBody(RigidBody* body) { return mSplitAngularVelocities[indexBody]; } +// Return true if the split impulses position correction technique is used for contacts +inline bool ContactSolver::isSplitImpulseActive() const { + return mIsSplitImpulseActive; +} + // Activate or Deactivate the split impulses for contacts inline void ContactSolver::setIsSplitImpulseActive(bool isActive) { mIsSplitImpulseActive = isActive; @@ -482,7 +490,7 @@ inline void ContactSolver::setIsSolveFrictionAtContactManifoldCenterActive(bool // Compute the collision restitution factor from the restitution factor of each body inline decimal ContactSolver::computeMixedRestitutionFactor(const RigidBody* body1, - const RigidBody* body2) const { + const RigidBody* body2) const { decimal restitution1 = body1->getRestitution(); decimal restitution2 = body2->getRestitution(); diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 53cd3e35..5a7062e0 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -40,6 +40,7 @@ DynamicsWorld::DynamicsWorld(const Vector3 &gravity, decimal timeStep = DEFAULT_ mContactSolver(mContactManifolds, mConstrainedLinearVelocities, mConstrainedAngularVelocities, mMapBodyToConstrainedVelocityIndex), mConstraintSolver(mJoints, mConstrainedLinearVelocities, mConstrainedAngularVelocities, + mConstrainedPositions, mConstrainedOrientations, mMapBodyToConstrainedVelocityIndex), mNbVelocitySolverIterations(DEFAULT_VELOCITY_SOLVER_NB_ITERATIONS), mNbPositionSolverIterations(DEFAULT_POSITION_SOLVER_NB_ITERATIONS), @@ -102,18 +103,24 @@ void DynamicsWorld::update() { // Integrate the velocities integrateRigidBodiesVelocities(); - // Solve the contacts and constraints - solveContactsAndConstraints(); + // Reset the movement boolean variable of each body to false + resetBodiesMovementVariable(); // Update the timer mTimer.nextStep(); - // Reset the movement boolean variable of each body to false - resetBodiesMovementVariable(); + // Solve the contacts and constraints + solveContactsAndConstraints(); // Integrate the position and orientation of each body integrateRigidBodiesPositions(); + // Solve the position correction for constraints + solvePositionCorrection(); + + // Update the AABBs of the bodies + updateRigidBodiesAABB(); + // Cleanup of the contact solver mContactSolver.cleanup(); @@ -130,7 +137,7 @@ void DynamicsWorld::update() { /// the sympletic Euler time stepping scheme. void DynamicsWorld::integrateRigidBodiesPositions() { - PROFILE("DynamicsWorld::updateRigidBodiesPositionAndOrientation()"); + PROFILE("DynamicsWorld::integrateRigidBodiesPositions()"); decimal dt = static_cast(mTimer.getTimeStep()); @@ -144,9 +151,6 @@ void DynamicsWorld::integrateRigidBodiesPositions() { // If the body is allowed to move if (rigidBody->getIsMotionEnabled()) { - // Update the old Transform of the body - rigidBody->updateOldTransform(); - // Get the constrained velocity uint indexArray = mMapBodyToConstrainedVelocityIndex.find(rigidBody)->second; Vector3 newLinVelocity = mConstrainedLinearVelocities[indexArray]; @@ -157,7 +161,9 @@ void DynamicsWorld::integrateRigidBodiesPositions() { rigidBody->setAngularVelocity(newAngVelocity); // Add the split impulse velocity from Contact Solver (only used to update the position) - if (mContactSolver.isConstrainedBody(rigidBody)) { + if (mContactSolver.isConstrainedBody(rigidBody) && + mContactSolver.isSplitImpulseActive()) { + newLinVelocity += mContactSolver.getSplitLinearVelocityOfBody(rigidBody); newAngVelocity += mContactSolver.getSplitAngularVelocityOfBody(rigidBody); } @@ -168,17 +174,30 @@ void DynamicsWorld::integrateRigidBodiesPositions() { // Compute the new position of the body Vector3 newPosition = currentPosition + newLinVelocity * dt; - Quaternion newOrientation = currentOrientation + Quaternion(newAngVelocity.x, - newAngVelocity.y, - newAngVelocity.z, 0) * - currentOrientation * 0.5 * dt; + Quaternion newOrientation = currentOrientation + Quaternion(0, newAngVelocity) * + currentOrientation * decimal(0.5) * dt; // Update the Transform of the body Transform newTransform(newPosition, newOrientation.getUnit()); rigidBody->setTransform(newTransform); + } + } +} + +// Update the AABBs of the bodies +void DynamicsWorld::updateRigidBodiesAABB() { + + PROFILE("DynamicsWorld::updateRigidBodiesAABB()"); + + // For each rigid body of the world + set::iterator it; + for (it = getRigidBodiesBeginIterator(); it != getRigidBodiesEndIterator(); ++it) { + + // If the body has moved + if ((*it)->getHasMoved()) { // Update the AABB of the rigid body - rigidBody->updateAABB(); + (*it)->updateAABB(); } } } @@ -210,6 +229,8 @@ void DynamicsWorld::setInterpolationFactorToAllBodies() { /// contact solver. void DynamicsWorld::integrateRigidBodiesVelocities() { + PROFILE("DynamicsWorld::integrateRigidBodiesVelocities()"); + // TODO : Use better memory allocation here mConstrainedLinearVelocities = std::vector(mRigidBodies.size(), Vector3(0, 0, 0)); mConstrainedAngularVelocities = std::vector(mRigidBodies.size(), Vector3(0, 0, 0)); @@ -231,6 +252,9 @@ void DynamicsWorld::integrateRigidBodiesVelocities() { dt * rigidBody->getMassInverse() * rigidBody->getExternalForce(); mConstrainedAngularVelocities[i] = rigidBody->getAngularVelocity() + dt * rigidBody->getInertiaTensorInverseWorld() * rigidBody->getExternalTorque(); + + // Update the old Transform of the body + rigidBody->updateOldTransform(); } i++; @@ -274,7 +298,7 @@ void DynamicsWorld::solveContactsAndConstraints() { // For each iteration of the velocity solver for (uint i=0; i(mRigidBodies.size()); + mConstrainedOrientations = std::vector(mRigidBodies.size()); + for (std::set::iterator it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { + + // If it is a constrained bodies (by a joint) + if (mConstraintSolver.isConstrainedBody(*it)) { + + uint index = mMapBodyToConstrainedVelocityIndex.find(*it)->second; + + // Get the position/orientation of the rigid body + const Transform& transform = (*it)->getTransform(); + mConstrainedPositions[index] = transform.getPosition(); + mConstrainedOrientations[index]= transform.getOrientation(); + } + } + + // ---------- Solve the position error correction for the constraints ---------- // + + // For each iteration of the position (error correction) solver + for (uint i=0; i::iterator it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { + + // If it is a constrained bodies (by a joint) + if (mConstraintSolver.isConstrainedBody(*it)) { + + uint index = mMapBodyToConstrainedVelocityIndex.find(*it)->second; + + // Get the new position/orientation of the body + const Vector3& newPosition = mConstrainedPositions[index]; + const Quaternion& newOrientation = mConstrainedOrientations[index]; + + // Update the Transform of the body + Transform newTransform(newPosition, newOrientation.getUnit()); + (*it)->setTransform(newTransform); } } } @@ -365,7 +430,7 @@ RigidBody* DynamicsWorld::createRigidBody(const Transform& transform, decimal ma return rigidBody; } -// Destroy a rigid body +// Destroy a rigid body and all the joints which it belongs void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { // Remove the body from the collision detection @@ -377,6 +442,14 @@ void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { // Remove the collision shape from the world removeCollisionShape(rigidBody->getCollisionShape()); + // Destroy all the joints that contains the rigid body to be destroyed + bodyindex idToRemove = rigidBody->getID(); + for (std::set::iterator it = mJoints.begin(); it != mJoints.end(); ++it) { + if ((*it)->getBody1()->getID() == idToRemove || (*it)->getBody2()->getID() == idToRemove) { + destroyJoint(*it); + } + } + // Call the destructor of the rigid body rigidBody->RigidBody::~RigidBody(); @@ -440,6 +513,13 @@ Constraint* DynamicsWorld::createJoint(const ConstraintInfo& jointInfo) { } } + // If the collision between the two bodies of the constraint is disabled + if (!jointInfo.isCollisionEnabled) { + + // Add the pair of bodies in the set of body pairs that cannot collide with each other + mCollisionDetection.addNoCollisionPair(jointInfo.body1, jointInfo.body2); + } + // Add the joint into the world mJoints.insert(newJoint); @@ -452,6 +532,13 @@ void DynamicsWorld::destroyJoint(Constraint* joint) { assert(joint != NULL); + // If the collision between the two bodies of the constraint was disabled + if (!joint->isCollisionEnabled()) { + + // Remove the pair of bodies from the set of body pairs that cannot collide with each other + mCollisionDetection.removeNoCollisionPair(joint->getBody1(), joint->getBody2()); + } + // Remove the joint from the world mJoints.erase(joint); diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index e9c7bcbb..d5412971 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -91,6 +91,12 @@ class DynamicsWorld : public CollisionWorld { /// after solving the constraints) std::vector mConstrainedAngularVelocities; + /// Array of constrained rigid bodies position (for position error correction) + std::vector mConstrainedPositions; + + /// Array of constrained rigid bodies orientation (for position error correction) + std::vector mConstrainedOrientations; + /// Map body to their index in the constrained velocities array std::map mMapBodyToConstrainedVelocityIndex; @@ -105,6 +111,9 @@ class DynamicsWorld : public CollisionWorld { /// Integrate the positions and orientations of rigid bodies. void integrateRigidBodiesPositions(); + /// Update the AABBs of the bodies + void updateRigidBodiesAABB(); + /// Update the position and orientation of a body void updatePositionAndOrientationOfBody(RigidBody* body, Vector3 newLinVelocity, Vector3 newAngVelocity); @@ -118,6 +127,9 @@ class DynamicsWorld : public CollisionWorld { /// Solve the contacts and constraints void solveContactsAndConstraints(); + /// Solve the position error correction of the constraints + void solvePositionCorrection(); + /// Cleanup the constrained velocities array at each step void cleanupConstrainedVelocitiesArray(); @@ -137,7 +149,8 @@ class DynamicsWorld : public CollisionWorld { virtual void notifyRemovedOverlappingPair(const BroadPhasePair* removedPair); /// Notify the world about a new narrow-phase contact - virtual void notifyNewContact(const BroadPhasePair* pair, const ContactPointInfo* contactInfo); + virtual void notifyNewContact(const BroadPhasePair* pair, + const ContactPointInfo* contactInfo); public : @@ -179,7 +192,7 @@ public : const Matrix3x3& inertiaTensorLocal, const CollisionShape& collisionShape); - /// Destroy a rigid body + /// Destroy a rigid body and all the joints which it belongs void destroyRigidBody(RigidBody* rigidBody); /// Create a joint between two bodies in the world and return a pointer to the new joint @@ -200,9 +213,15 @@ public : /// Return the number of rigid bodies in the world uint getNbRigidBodies() const; - /// Return the number of contact constraints in the world + /// Return the number of joints in the world + uint getNbJoints() const; + + /// Return the number of contact manifolds in the world uint getNbContactManifolds() const; + /// Return the current physics time (in seconds) + long double getPhysicsTime() const; + /// Return an iterator to the beginning of the rigid bodies of the physics world std::set::iterator getRigidBodiesBeginIterator(); @@ -302,6 +321,11 @@ inline uint DynamicsWorld::getNbRigidBodies() const { return mRigidBodies.size(); } +/// Return the number of joints in the world +inline uint DynamicsWorld::getNbJoints() const { + return mJoints.size(); +} + // Return an iterator to the beginning of the bodies of the physics world inline std::set::iterator DynamicsWorld::getRigidBodiesBeginIterator() { return mRigidBodies.begin(); @@ -317,6 +341,11 @@ inline uint DynamicsWorld::getNbContactManifolds() const { return mContactManifolds.size(); } +/// Return the current physics time (in seconds) +inline long double DynamicsWorld::getPhysicsTime() const { + return mTimer.getPhysicsTime(); +} + } #endif diff --git a/src/engine/Timer.h b/src/engine/Timer.h index 22728537..7e1b3b7d 100644 --- a/src/engine/Timer.h +++ b/src/engine/Timer.h @@ -47,7 +47,7 @@ namespace reactphysics3d { // Class Timer /** * This class will take care of the time in the physics engine. It - * uses fuunctions that depend on the current platform to get the + * uses functions that depend on the current platform to get the * current time. */ class Timer { @@ -59,9 +59,6 @@ class Timer { /// Timestep dt of the physics engine (timestep > 0.0) double mTimeStep; - /// Current time of the physics engine - long double mTime; - /// Last time the timer has been updated long double mLastUpdateTime; @@ -139,7 +136,7 @@ inline void Timer::setTimeStep(double timeStep) { // Return the current time inline long double Timer::getPhysicsTime() const { - return mTime; + return mLastUpdateTime; } // Return if the timer is running @@ -173,9 +170,6 @@ inline bool Timer::isPossibleToTakeStep() const { inline void Timer::nextStep() { assert(mIsRunning); - // Update the current time of the physics engine - mTime += mTimeStep; - // Update the accumulator value mAccumulator -= mTimeStep; } diff --git a/src/mathematics/Quaternion.h b/src/mathematics/Quaternion.h index 246d56b2..5bc862da 100644 --- a/src/mathematics/Quaternion.h +++ b/src/mathematics/Quaternion.h @@ -133,6 +133,12 @@ struct Quaternion { /// Overloaded operator for the substraction Quaternion operator-(const Quaternion& quaternion) const; + /// Overloaded operator for addition with assignment + Quaternion& operator+=(const Quaternion& quaternion); + + /// Overloaded operator for substraction with assignment + Quaternion& operator-=(const Quaternion& quaternion); + /// Overloaded operator for the multiplication with a constant Quaternion operator*(decimal nb) const; @@ -272,6 +278,24 @@ inline Quaternion Quaternion::operator-(const Quaternion& quaternion) const { return Quaternion(x - quaternion.x, y - quaternion.y, z - quaternion.z, w - quaternion.w); } +// Overloaded operator for addition with assignment +inline Quaternion& Quaternion::operator+=(const Quaternion& quaternion) { + x += quaternion.x; + y += quaternion.y; + z += quaternion.z; + w += quaternion.w; + return *this; +} + +// Overloaded operator for substraction with assignment +inline Quaternion& Quaternion::operator-=(const Quaternion& quaternion) { + x -= quaternion.x; + y -= quaternion.y; + z -= quaternion.z; + w -= quaternion.w; + return *this; +} + // Overloaded operator for the multiplication with a constant inline Quaternion Quaternion::operator*(decimal nb) const { return Quaternion(nb * x, nb * y, nb * z, nb * w); diff --git a/test/tests/mathematics/TestQuaternion.h b/test/tests/mathematics/TestQuaternion.h index ac7ad741..a7f3d324 100644 --- a/test/tests/mathematics/TestQuaternion.h +++ b/test/tests/mathematics/TestQuaternion.h @@ -197,11 +197,17 @@ class TestQuaternion : public Test { Quaternion quat1(4, 5, 2, 10); Quaternion quat2(-2, 7, 8, 3); Quaternion test1 = quat1 + quat2; + Quaternion test11(-6, 52, 2, 8); + test11 += quat1; test(test1 == Quaternion(2, 12, 10, 13)); + test(test11 == Quaternion(-2, 57, 4, 18)); // Test substraction Quaternion test2 = quat1 - quat2; + Quaternion test22(-73, 62, 25, 9); + test22 -= quat1; test(test2 == Quaternion(6, -2, -6, 7)); + test(test22 == Quaternion(-77, 57, 23, -1)); // Test multiplication with a number Quaternion test3 = quat1 * 3.0; From d265a52b67e76f8b588143a1802f7ca217cbcf0b Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 26 Jun 2013 22:29:23 +0200 Subject: [PATCH 34/66] Modify the joints example --- examples/joints/Scene.cpp | 249 +++++++++++++++++++++++++++++--------- examples/joints/Scene.h | 49 ++++++-- 2 files changed, 229 insertions(+), 69 deletions(-) diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index 419f5c24..ce278b63 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -25,6 +25,7 @@ // Libraries #include "Scene.h" +#include // Namespaces using namespace openglframework; @@ -62,6 +63,12 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), // Create the Slider joint createSliderJoint(); + // Create the Hinge joint + createPropellerHingeJoint(); + + // Create the Fixed joint + createFixedJoints(); + // Create the floor createFloor(); @@ -79,18 +86,32 @@ Scene::~Scene() { mPhongShader.destroy(); // Destroy the joints - mDynamicsWorld->destroyJoint(mBallAndSocketJoint); mDynamicsWorld->destroyJoint(mSliderJoint); + mDynamicsWorld->destroyJoint(mPropellerHingeJoint); + mDynamicsWorld->destroyJoint(mFixedJoint1); + mDynamicsWorld->destroyJoint(mFixedJoint2); + for (int i=0; idestroyJoint(mBallAndSocketJoints[i]); + } - // Destroy all the boxes of the scene - mDynamicsWorld->destroyRigidBody(mBallAndSocketJointBox1->getRigidBody()); - mDynamicsWorld->destroyRigidBody(mBallAndSocketJointBox2->getRigidBody()); - mDynamicsWorld->destroyRigidBody(mSliderJointBox1->getRigidBody()); - mDynamicsWorld->destroyRigidBody(mSliderJointBox2->getRigidBody()); - delete mBallAndSocketJointBox1; - delete mBallAndSocketJointBox2; - delete mSliderJointBox1; - delete mSliderJointBox2; + // Destroy all the rigid bodies of the scene + mDynamicsWorld->destroyRigidBody(mSliderJointBottomBox->getRigidBody()); + mDynamicsWorld->destroyRigidBody(mSliderJointTopBox->getRigidBody()); + mDynamicsWorld->destroyRigidBody(mPropellerBox->getRigidBody()); + mDynamicsWorld->destroyRigidBody(mFixedJointBox1->getRigidBody()); + mDynamicsWorld->destroyRigidBody(mFixedJointBox2->getRigidBody()); + for (int i=0; idestroyRigidBody(mBallAndSocketJointChainBoxes[i]->getRigidBody()); + } + + delete mSliderJointBottomBox; + delete mSliderJointTopBox; + delete mPropellerBox; + delete mFixedJointBox1; + delete mFixedJointBox2; + for (int i=0; idestroyRigidBody(mFloor->getRigidBody()); @@ -106,14 +127,22 @@ void Scene::simulate() { // If the physics simulation is running if (mIsRunning) { + // Update the motor speed of the Slider Joint (to move up and down) + long double motorSpeed = 3 * cos(mDynamicsWorld->getPhysicsTime() * 1.5); + mSliderJoint->setMotorSpeed(motorSpeed); + // Take a simulation step mDynamicsWorld->update(); // Update the position and orientation of the boxes - mBallAndSocketJointBox1->updateTransform(); - mBallAndSocketJointBox2->updateTransform(); - mSliderJointBox1->updateTransform(); - mSliderJointBox2->updateTransform(); + mSliderJointBottomBox->updateTransform(); + mSliderJointTopBox->updateTransform(); + mPropellerBox->updateTransform(); + mFixedJointBox1->updateTransform(); + mFixedJointBox2->updateTransform(); + for (int i=0; iupdateTransform(); + } // Update the position and orientation of the floor mFloor->updateTransform(); @@ -146,10 +175,14 @@ void Scene::render() { mPhongShader.setFloatUniform("shininess", 60.0f); // Render all the boxes - mBallAndSocketJointBox1->render(mPhongShader); - mBallAndSocketJointBox2->render(mPhongShader); - mSliderJointBox1->render(mPhongShader); - mSliderJointBox2->render(mPhongShader); + mSliderJointBottomBox->render(mPhongShader); + mSliderJointTopBox->render(mPhongShader); + mPropellerBox->render(mPhongShader); + mFixedJointBox1->render(mPhongShader); + mFixedJointBox2->render(mPhongShader); + for (int i=0; irender(mPhongShader); + } // Render the floor mFloor->render(mPhongShader); @@ -161,45 +194,44 @@ void Scene::render() { // Create the boxes and joints for the Ball-and-Socket joint example void Scene::createBallAndSocketJoints() { - // --------------- Create the first box --------------- // + // --------------- Create the boxes --------------- // - // Position of the box - openglframework::Vector3 positionBox1(0, 15, 0); + openglframework::Vector3 positionBox(0, 15, 5); + openglframework::Vector3 boxDimension(1, 1, 1); + const float boxMass = 0.5f; - // Create a box and a corresponding rigid in the dynamics world - mBallAndSocketJointBox1 = new Box(BOX_SIZE, positionBox1 , BOX_MASS, mDynamicsWorld); + for (int i=0; igetRigidBody()->setIsMotionEnabled(false); + // Create a box and a corresponding rigid in the dynamics world + mBallAndSocketJointChainBoxes[i] = new Box(boxDimension, positionBox , boxMass, + mDynamicsWorld); - // Set the bouncing factor of the box - mBallAndSocketJointBox1->getRigidBody()->setRestitution(0.4); + // The fist box cannot move + if (i == 0) mBallAndSocketJointChainBoxes[i]->getRigidBody()->setIsMotionEnabled(false); + else mBallAndSocketJointChainBoxes[i]->getRigidBody()->setIsMotionEnabled(true); - // --------------- Create the second box --------------- // + // Set the bouncing factor of the box + mBallAndSocketJointChainBoxes[i]->getRigidBody()->setRestitution(0.4); - // Position of the box - openglframework::Vector3 positionBox2(5, 10, 0); + positionBox.y -= boxDimension.y + 0.5; + } - // Create a box and a corresponding rigid in the dynamics world - mBallAndSocketJointBox2 = new Box(BOX_SIZE, positionBox2 , BOX_MASS, mDynamicsWorld); + // --------------- Create the joints --------------- // - // The second box is allowed to move - mBallAndSocketJointBox2->getRigidBody()->setIsMotionEnabled(true); + for (int i=0; igetRigidBody()->setRestitution(0.4); + // Create the joint info object + rp3d::RigidBody* body1 = mBallAndSocketJointChainBoxes[i]->getRigidBody(); + rp3d::RigidBody* body2 = mBallAndSocketJointChainBoxes[i+1]->getRigidBody(); + rp3d::Vector3 body1Position = body1->getTransform().getPosition(); + rp3d::Vector3 body2Position = body2->getTransform().getPosition(); + const rp3d::Vector3 anchorPointWorldSpace = 0.5 * (body1Position + body2Position); + rp3d::BallAndSocketJointInfo jointInfo(body1, body2, anchorPointWorldSpace); - // --------------- Create the joint --------------- // - - // Create the joint info object - rp3d::RigidBody* body1 = mBallAndSocketJointBox1->getRigidBody(); - rp3d::RigidBody* body2 = mBallAndSocketJointBox2->getRigidBody(); - const rp3d::Vector3 anchorPointWorldSpace(0, 10, 0); - rp3d::BallAndSocketJointInfo jointInfo(body1, body2, anchorPointWorldSpace); - - // Create the joint in the dynamics world - mBallAndSocketJoint = dynamic_cast( - mDynamicsWorld->createJoint(jointInfo)); + // Create the joint in the dynamics world + mBallAndSocketJoints[i] = dynamic_cast( + mDynamicsWorld->createJoint(jointInfo)); + } } /// Create the boxes and joint for the Slider joint example @@ -208,46 +240,145 @@ void Scene::createSliderJoint() { // --------------- Create the first box --------------- // // Position of the box - openglframework::Vector3 positionBox1(-4, 6, 0); + openglframework::Vector3 positionBox1(0, 2.1, 0); // Create a box and a corresponding rigid in the dynamics world - mSliderJointBox1 = new Box(BOX_SIZE, positionBox1 , BOX_MASS, mDynamicsWorld); + openglframework::Vector3 box1Dimension(2, 4, 2); + mSliderJointBottomBox = new Box(box1Dimension, positionBox1 , BOX_MASS, mDynamicsWorld); // The fist box cannot move - mSliderJointBox1->getRigidBody()->setIsMotionEnabled(false); + mSliderJointBottomBox->getRigidBody()->setIsMotionEnabled(false); // Set the bouncing factor of the box - mSliderJointBox1->getRigidBody()->setRestitution(0.4); + mSliderJointBottomBox->getRigidBody()->setRestitution(0.4); // --------------- Create the second box --------------- // // Position of the box - openglframework::Vector3 positionBox2(2, 4, 0); + openglframework::Vector3 positionBox2(0, 4.2, 0); // Create a box and a corresponding rigid in the dynamics world - mSliderJointBox2 = new Box(BOX_SIZE, positionBox2 , BOX_MASS, mDynamicsWorld); + openglframework::Vector3 box2Dimension(1.5, 4, 1.5); + mSliderJointTopBox = new Box(box2Dimension, positionBox2 , BOX_MASS, mDynamicsWorld); // The second box is allowed to move - mSliderJointBox2->getRigidBody()->setIsMotionEnabled(true); + mSliderJointTopBox->getRigidBody()->setIsMotionEnabled(true); // Set the bouncing factor of the box - mSliderJointBox2->getRigidBody()->setRestitution(0.4); + mSliderJointTopBox->getRigidBody()->setRestitution(0.4); // --------------- Create the joint --------------- // // Create the joint info object - rp3d::RigidBody* body1 = mSliderJointBox1->getRigidBody(); - rp3d::RigidBody* body2 = mSliderJointBox2->getRigidBody(); + rp3d::RigidBody* body1 = mSliderJointBottomBox->getRigidBody(); + rp3d::RigidBody* body2 = mSliderJointTopBox->getRigidBody(); const rp3d::Vector3& body1Position = body1->getTransform().getPosition(); const rp3d::Vector3& body2Position = body2->getTransform().getPosition(); const rp3d::Vector3 anchorPointWorldSpace = 0.5 * (body2Position + body1Position); - const rp3d::Vector3 sliderAxisWorldSpace = body2Position - body1Position; - rp3d::SliderJointInfo jointInfo(body1, body2, anchorPointWorldSpace, sliderAxisWorldSpace); + const rp3d::Vector3 sliderAxisWorldSpace = (body2Position - body1Position); + rp3d::SliderJointInfo jointInfo(body1, body2, anchorPointWorldSpace, sliderAxisWorldSpace, + -1.7, 1.7); + jointInfo.isMotorEnabled = true; + jointInfo.motorSpeed = 0.0; + jointInfo.maxMotorForce = 10000.0; + jointInfo.isCollisionEnabled = false; // Create the joint in the dynamics world mSliderJoint = dynamic_cast(mDynamicsWorld->createJoint(jointInfo)); } +/// Create the boxes and joint for the Hinge joint example +void Scene::createPropellerHingeJoint() { + + // --------------- Create the propeller box --------------- // + + // Position of the box + openglframework::Vector3 positionBox1(0, 7, 0); + + // Create a box and a corresponding rigid in the dynamics world + openglframework::Vector3 boxDimension(10, 1, 1); + mPropellerBox = new Box(boxDimension, positionBox1 , BOX_MASS, mDynamicsWorld); + + // The fist box cannot move + mPropellerBox->getRigidBody()->setIsMotionEnabled(true); + + // Set the bouncing factor of the box + mPropellerBox->getRigidBody()->setRestitution(0.4); + + // --------------- Create the Hinge joint --------------- // + + // Create the joint info object + rp3d::RigidBody* body1 = mPropellerBox->getRigidBody(); + rp3d::RigidBody* body2 = mSliderJointTopBox->getRigidBody(); + const rp3d::Vector3& body1Position = body1->getTransform().getPosition(); + const rp3d::Vector3& body2Position = body2->getTransform().getPosition(); + const rp3d::Vector3 anchorPointWorldSpace = 0.5 * (body2Position + body1Position); + const rp3d::Vector3 hingeAxisWorldSpace(0, 1, 0); + rp3d::HingeJointInfo jointInfo(body1, body2, anchorPointWorldSpace, hingeAxisWorldSpace); + jointInfo.isMotorEnabled = true; + jointInfo.motorSpeed = -0.5 * PI; + jointInfo.maxMotorTorque = 60.0; + jointInfo.isCollisionEnabled = false; + + // Create the joint in the dynamics world + mPropellerHingeJoint = dynamic_cast(mDynamicsWorld->createJoint(jointInfo)); +} + +/// Create the boxes and joints for the fixed joints +void Scene::createFixedJoints() { + + // --------------- Create the first box --------------- // + + // Position of the box + openglframework::Vector3 positionBox1(5, 7, 0); + + // Create a box and a corresponding rigid in the dynamics world + openglframework::Vector3 boxDimension(1.5, 1.5, 1.5); + mFixedJointBox1 = new Box(boxDimension, positionBox1 , BOX_MASS, mDynamicsWorld); + + // The fist box cannot move + mFixedJointBox1->getRigidBody()->setIsMotionEnabled(true); + + // Set the bouncing factor of the box + mFixedJointBox1->getRigidBody()->setRestitution(0.4); + + // --------------- Create the second box --------------- // + + // Position of the box + openglframework::Vector3 positionBox2(-5, 7, 0); + + // Create a box and a corresponding rigid in the dynamics world + mFixedJointBox2 = new Box(boxDimension, positionBox2 , BOX_MASS, mDynamicsWorld); + + // The second box is allowed to move + mFixedJointBox2->getRigidBody()->setIsMotionEnabled(true); + + // Set the bouncing factor of the box + mFixedJointBox2->getRigidBody()->setRestitution(0.4); + + // --------------- Create the first fixed joint --------------- // + + // Create the joint info object + rp3d::RigidBody* body1 = mFixedJointBox1->getRigidBody(); + rp3d::RigidBody* propellerBody = mPropellerBox->getRigidBody(); + const rp3d::Vector3 anchorPointWorldSpace1(5, 7, 0); + rp3d::FixedJointInfo jointInfo1(body1, propellerBody, anchorPointWorldSpace1); + + // Create the joint in the dynamics world + mFixedJoint1 = dynamic_cast(mDynamicsWorld->createJoint(jointInfo1)); + + // --------------- Create the second fixed joint --------------- // + + // Create the joint info object + rp3d::RigidBody* body2 = mFixedJointBox2->getRigidBody(); + const rp3d::Vector3 anchorPointWorldSpace2(-5, 7, 0); + rp3d::FixedJointInfo jointInfo2(body2, propellerBody, anchorPointWorldSpace2); + + // Create the joint in the dynamics world + mFixedJoint2 = dynamic_cast(mDynamicsWorld->createJoint(jointInfo2)); +} + + // Create the floor void Scene::createFloor() { diff --git a/examples/joints/Scene.h b/examples/joints/Scene.h index 3bc5cb5d..90430d1d 100644 --- a/examples/joints/Scene.h +++ b/examples/joints/Scene.h @@ -36,6 +36,8 @@ const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters const float BOX_MASS = 1.0f; // Box mass in kilograms const float FLOOR_MASS = 100.0f; // Floor mass in kilograms +const int NB_BALLSOCKETJOINT_BOXES = 7; // Number of Ball-And-Socket chain boxes +const int NB_HINGE_BOXES = 7; // Number of Hinge chain boxes // Class Scene class Scene { @@ -53,24 +55,45 @@ class Scene { /// Phong shader openglframework::Shader mPhongShader; - /// Box 1 of Ball-And-Socket joint - Box* mBallAndSocketJointBox1; + /// Boxes of Ball-And-Socket joint chain + Box* mBallAndSocketJointChainBoxes[NB_BALLSOCKETJOINT_BOXES]; - /// Box 2 of Ball-And-Socket joint - Box* mBallAndSocketJointBox2; + /// Boxes of the Hinge joint chain + Box* mHingeJointChainBoxes[NB_HINGE_BOXES]; - /// Ball-and-Socket joint - rp3d::BallAndSocketJoint* mBallAndSocketJoint; + /// Ball-And-Socket joints of the chain + rp3d::BallAndSocketJoint* mBallAndSocketJoints[NB_BALLSOCKETJOINT_BOXES-1]; - /// Box 1 of Slider joint - Box* mSliderJointBox1; + /// Hinge joints of the chain + rp3d::HingeJoint* mHingeJoints[NB_HINGE_BOXES-1]; - /// Box 2 of Slider joint - Box* mSliderJointBox2; + /// Bottom box of the Slider joint + Box* mSliderJointBottomBox; + + /// Top box of the Slider joint + Box* mSliderJointTopBox; /// Slider joint rp3d::SliderJoint* mSliderJoint; + /// Propeller box + Box* mPropellerBox; + + /// Box 1 of Fixed joint + Box* mFixedJointBox1; + + /// Box 2 of Fixed joint + Box* mFixedJointBox2; + + /// Hinge joint + rp3d::HingeJoint* mPropellerHingeJoint; + + /// First Fixed joint + rp3d::FixedJoint* mFixedJoint1; + + /// Second Fixed joint + rp3d::FixedJoint* mFixedJoint2; + /// Box for the floor Box* mFloor; @@ -88,6 +111,12 @@ class Scene { /// Create the boxes and joint for the Slider joint example void createSliderJoint(); + /// Create the boxes and joint for the Hinge joint example + void createPropellerHingeJoint(); + + /// Create the boxes and joint for the Fixed joint example + void createFixedJoints(); + /// Create the floor void createFloor(); From d58db2e6f2c010f3298ab9d5942282fa31a1eb19 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 27 Jun 2013 19:53:13 +0200 Subject: [PATCH 35/66] Remove the gravity from the rigid body external force and allow the user to enable/disable the gravity for each body --- src/body/RigidBody.cpp | 3 ++- src/body/RigidBody.h | 19 +++++++++++++++++++ src/engine/DynamicsWorld.cpp | 32 ++++++++------------------------ src/engine/DynamicsWorld.h | 3 --- 4 files changed, 29 insertions(+), 28 deletions(-) diff --git a/src/body/RigidBody.cpp b/src/body/RigidBody.cpp index 40139aff..5b2022bd 100644 --- a/src/body/RigidBody.cpp +++ b/src/body/RigidBody.cpp @@ -35,7 +35,8 @@ using namespace reactphysics3d; CollisionShape *collisionShape, bodyindex id) : CollisionBody(transform, collisionShape, id), mInertiaTensorLocal(inertiaTensorLocal), mMass(mass), mInertiaTensorLocalInverse(inertiaTensorLocal.getInverse()), - mMassInverse(decimal(1.0) / mass), mFrictionCoefficient(DEFAULT_FRICTION_COEFFICIENT) { + mMassInverse(decimal(1.0) / mass), mFrictionCoefficient(DEFAULT_FRICTION_COEFFICIENT), + mIsGravityEnabled(true) { mRestitution = decimal(1.0); diff --git a/src/body/RigidBody.h b/src/body/RigidBody.h index 1f340961..66a3f70c 100644 --- a/src/body/RigidBody.h +++ b/src/body/RigidBody.h @@ -79,6 +79,9 @@ class RigidBody : public CollisionBody { /// Friction coefficient decimal mFrictionCoefficient; + /// True if the gravity needs to be applied to this rigid body + bool mIsGravityEnabled; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -160,6 +163,12 @@ class RigidBody : public CollisionBody { /// Set the friction coefficient void setFrictionCoefficient(decimal frictionCoefficient); + + /// Return true if the gravity needs to be applied to this rigid body + bool isGravityEnabled() const; + + /// Set the variable to know if the gravity is applied to this rigid body + void enableGravity(bool isEnabled); }; // Method that return the mass of the body @@ -288,6 +297,16 @@ inline void RigidBody::setFrictionCoefficient(decimal frictionCoefficient) { mFrictionCoefficient = frictionCoefficient; } +// Return true if the gravity needs to be applied to this rigid body +inline bool RigidBody::isGravityEnabled() const { + return mIsGravityEnabled; +} + +// Set the variable to know if the gravity is applied to this rigid body +inline void RigidBody::enableGravity(bool isEnabled) { + mIsGravityEnabled = isEnabled; +} + } #endif diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 5a7062e0..27a8fc12 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -88,9 +88,6 @@ void DynamicsWorld::update() { // Compute the time since the last update() call and update the timer mTimer.update(); - // Apply the gravity force to all bodies - applyGravity(); - // While the time accumulator is not empty while(mTimer.isPossibleToTakeStep()) { @@ -253,6 +250,14 @@ void DynamicsWorld::integrateRigidBodiesVelocities() { mConstrainedAngularVelocities[i] = rigidBody->getAngularVelocity() + dt * rigidBody->getInertiaTensorInverseWorld() * rigidBody->getExternalTorque(); + // If the gravity has to be applied to this rigid body + if (rigidBody->isGravityEnabled() && mIsGravityOn) { + + // Integrate the gravity force + mConstrainedLinearVelocities[i] += dt * rigidBody->getMassInverse() * + rigidBody->getMass() * mGravity; + } + // Update the old Transform of the body rigidBody->updateOldTransform(); } @@ -376,27 +381,6 @@ void DynamicsWorld::cleanupConstrainedVelocitiesArray() { mMapBodyToConstrainedVelocityIndex.clear(); } -// Apply the gravity force to all bodies of the physics world -void DynamicsWorld::applyGravity() { - - PROFILE("DynamicsWorld::applyGravity()"); - - // For each body of the physics world - set::iterator it; - for (it = getRigidBodiesBeginIterator(); it != getRigidBodiesEndIterator(); ++it) { - - RigidBody* rigidBody = dynamic_cast(*it); - assert(rigidBody != NULL); - - // If the gravity force is on - if(mIsGravityOn) { - - // Apply the current gravity force to the body - rigidBody->setExternalForce(rigidBody->getMass() * mGravity); - } - } -} - // Create a rigid body into the physics world RigidBody* DynamicsWorld::createRigidBody(const Transform& transform, decimal mass, const Matrix3x3& inertiaTensorLocal, diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index d5412971..f911bd08 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -133,9 +133,6 @@ class DynamicsWorld : public CollisionWorld { /// Cleanup the constrained velocities array at each step void cleanupConstrainedVelocitiesArray(); - /// Apply the gravity force to all bodies - void applyGravity(); - /// Reset the boolean movement variable of each body void resetBodiesMovementVariable(); From 501b0747d3ba60f4097f023474ad77302eda4c00 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 2 Jul 2013 23:14:08 +0200 Subject: [PATCH 36/66] Modifications in the examples --- examples/CMakeLists.txt | 1 + examples/common/Box.cpp | 14 +- examples/common/Box.h | 2 +- examples/common/Cone.cpp | 136 +++ examples/common/Cone.h | 81 ++ examples/common/Cylinder.cpp | 136 +++ examples/common/Cylinder.h | 81 ++ examples/common/Sphere.cpp | 136 +++ examples/common/Sphere.h | 78 ++ examples/common/VisualContactPoint.cpp | 115 ++ examples/common/VisualContactPoint.h | 73 ++ examples/common/meshes/cone.obj | 101 ++ examples/common/meshes/cylinder.obj | 293 +++++ examples/common/meshes/sphere.obj | 997 ++++++++++++++++++ examples/common/opengl-framework/src/Shader.h | 19 + .../opengl-framework/src/maths/Matrix4.h | 8 + .../opengl-framework/src/shaders/phong.frag | 23 +- .../opengl-framework/src/shaders/phong.vert | 17 +- examples/fallingcubes/Scene.cpp | 19 +- examples/fallingcubes/Scene.h | 4 +- examples/joints/Scene.cpp | 34 +- examples/joints/Scene.h | 2 +- src/engine/DynamicsWorld.h | 10 +- 23 files changed, 2325 insertions(+), 55 deletions(-) create mode 100644 examples/common/Cone.cpp create mode 100644 examples/common/Cone.h create mode 100644 examples/common/Cylinder.cpp create mode 100644 examples/common/Cylinder.h create mode 100644 examples/common/Sphere.cpp create mode 100644 examples/common/Sphere.h create mode 100644 examples/common/VisualContactPoint.cpp create mode 100644 examples/common/VisualContactPoint.h create mode 100644 examples/common/meshes/cone.obj create mode 100644 examples/common/meshes/cylinder.obj create mode 100644 examples/common/meshes/sphere.obj diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt index 0740c9fc..e8d6a1d6 100644 --- a/examples/CMakeLists.txt +++ b/examples/CMakeLists.txt @@ -4,3 +4,4 @@ CMAKE_MINIMUM_REQUIRED(VERSION 2.6) add_subdirectory(common/) add_subdirectory(fallingcubes/) add_subdirectory(joints/) +add_subdirectory(convexmesh/) diff --git a/examples/common/Box.cpp b/examples/common/Box.cpp index 1c311231..34fb6597 100644 --- a/examples/common/Box.cpp +++ b/examples/common/Box.cpp @@ -105,13 +105,21 @@ Box::~Box() { } // Render the cube at the correct position and with the correct orientation -void Box::render(openglframework::Shader& shader) { +void Box::render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix) { // Bind the shader shader.bind(); - // Set the model to World matrix - shader.setMatrix4x4Uniform("modelToWorldMatrix", mTransformMatrix); + // Set the model to camera matrix + const openglframework::Matrix4 localToCameraMatrix = worldToCameraMatrix * mTransformMatrix; + shader.setMatrix4x4Uniform("localToCameraMatrix", localToCameraMatrix); + + // Set the normal matrix (inverse transpose of the 3x3 upper-left sub matrix of the + // model-view matrix) + const openglframework::Matrix3 normalMatrix = + localToCameraMatrix.getUpperLeft3x3Matrix().getInverse().getTranspose(); + shader.setMatrix3x3Uniform("normalMatrix", normalMatrix); // Bind the vertices VBO mVBOVertices.bind(); diff --git a/examples/common/Box.h b/examples/common/Box.h index ec5d75b2..302df1bc 100644 --- a/examples/common/Box.h +++ b/examples/common/Box.h @@ -97,7 +97,7 @@ class Box : public openglframework::Object3D { void updateTransform(); /// Render the cube at the correct position and with the correct orientation - void render(openglframework::Shader& shader); + void render(openglframework::Shader& shader, const openglframework::Matrix4& worldToCameraMatrix); }; // Return a pointer to the rigid body of the box diff --git a/examples/common/Cone.cpp b/examples/common/Cone.cpp new file mode 100644 index 00000000..ca7e1cbb --- /dev/null +++ b/examples/common/Cone.cpp @@ -0,0 +1,136 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Cone.h" + + +// Constructor +Cone::Cone(float radius, float height, const openglframework::Vector3 &position, + float mass, reactphysics3d::DynamicsWorld* dynamicsWorld) + : openglframework::Mesh(), mRadius(radius), mHeight(height) { + + // Load the mesh from a file + openglframework::MeshReaderWriter::loadMeshFromFile("meshes/cone.obj", *this); + + // Calculate the normals of the mesh + calculateNormals(); + + // Compute the scaling matrix + mScalingMatrix = openglframework::Matrix4(mRadius, 0, 0, 0, + 0, mHeight, 0, 0, + 0, 0, mRadius, 0, + 0, 0, 0, 1); + + // Initialize the position where the cone will be rendered + translateWorld(position); + + // Create the collision shape for the rigid body (cone shape) + // ReactPhysics3D will clone this object to create an internal one. Therefore, + // it is OK if this object is destroy right after calling Dynamics::createRigidBody() + const rp3d::ConeShape collisionShape(mRadius, mHeight); + + // Compute the inertia tensor of the body using its collision shape + rp3d::Matrix3x3 inertiaTensor; + collisionShape.computeLocalInertiaTensor(inertiaTensor, mass); + + // Initial position and orientation of the rigid body + rp3d::Vector3 initPosition(position.x, position.y, position.z); + rp3d::Quaternion initOrientation = rp3d::Quaternion::identity(); + rp3d::Transform transform(initPosition, initOrientation); + + // Create a rigid body corresponding to the cone in the dynamics world + mRigidBody = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, collisionShape); +} + +// Destructor +Cone::~Cone() { + + // Destroy the mesh + destroy(); +} + +// Render the cone at the correct position and with the correct orientation +void Cone::render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix) { + + // Bind the shader + shader.bind(); + + // Set the model to camera matrix + const openglframework::Matrix4 localToCameraMatrix = worldToCameraMatrix * mTransformMatrix; + shader.setMatrix4x4Uniform("localToCameraMatrix", localToCameraMatrix); + + // Set the normal matrix (inverse transpose of the 3x3 upper-left sub matrix of the + // model-view matrix) + const openglframework::Matrix3 normalMatrix = + localToCameraMatrix.getUpperLeft3x3Matrix().getInverse().getTranspose(); + shader.setMatrix3x3Uniform("normalMatrix", normalMatrix); + + glEnableClientState(GL_VERTEX_ARRAY); + glEnableClientState(GL_NORMAL_ARRAY); + if (hasTexture()) { + glEnableClientState(GL_TEXTURE_COORD_ARRAY); + } + + glVertexPointer(3, GL_FLOAT, 0, getVerticesPointer()); + glNormalPointer(GL_FLOAT, 0, getNormalsPointer()); + if(hasTexture()) { + glTexCoordPointer(2, GL_FLOAT, 0, getUVTextureCoordinatesPointer()); + } + + // For each part of the mesh + for (unsigned int i=0; igetInterpolatedTransform(); + + // Compute the transform used for rendering the cone + float matrix[16]; + transform.getOpenGLMatrix(matrix); + openglframework::Matrix4 newMatrix(matrix[0], matrix[4], matrix[8], matrix[12], + matrix[1], matrix[5], matrix[9], matrix[13], + matrix[2], matrix[6], matrix[10], matrix[14], + matrix[3], matrix[7], matrix[11], matrix[15]); + + // Apply the scaling matrix to have the correct cone dimensions + mTransformMatrix = newMatrix * mScalingMatrix; +} diff --git a/examples/common/Cone.h b/examples/common/Cone.h new file mode 100644 index 00000000..fabb7157 --- /dev/null +++ b/examples/common/Cone.h @@ -0,0 +1,81 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 CONE_H +#define CONE_H + +// Libraries +#include "openglframework.h" +#include "reactphysics3d.h" + +// Class Cone +class Cone : public openglframework::Mesh { + + private : + + // -------------------- Attributes -------------------- // + + /// Radius of the cone + float mRadius; + + /// Height of the cone + float mHeight; + + /// Rigid body used to simulate the dynamics of the cone + rp3d::RigidBody* mRigidBody; + + /// Scaling matrix (applied to a sphere to obtain the correct cone dimensions) + openglframework::Matrix4 mScalingMatrix; + + // -------------------- Methods -------------------- // + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + Cone(float radius, float height, const openglframework::Vector3& position, + float mass, rp3d::DynamicsWorld* dynamicsWorld); + + /// Destructor + ~Cone(); + + /// Return a pointer to the rigid body of the cone + rp3d::RigidBody* getRigidBody(); + + /// Update the transform matrix of the cone + void updateTransform(); + + /// Render the cone at the correct position and with the correct orientation + void render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix); +}; + +// Return a pointer to the rigid body of the cone +inline rp3d::RigidBody* Cone::getRigidBody() { + return mRigidBody; +} + +#endif diff --git a/examples/common/Cylinder.cpp b/examples/common/Cylinder.cpp new file mode 100644 index 00000000..556d413f --- /dev/null +++ b/examples/common/Cylinder.cpp @@ -0,0 +1,136 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Cylinder.h" + + +// Constructor +Cylinder::Cylinder(float radius, float height, const openglframework::Vector3 &position, + float mass, reactphysics3d::DynamicsWorld* dynamicsWorld) + : openglframework::Mesh(), mRadius(radius), mHeight(height) { + + // Load the mesh from a file + openglframework::MeshReaderWriter::loadMeshFromFile("meshes/cylinder.obj", *this); + + // Calculate the normals of the mesh + calculateNormals(); + + // Compute the scaling matrix + mScalingMatrix = openglframework::Matrix4(mRadius, 0, 0, 0, + 0, mHeight, 0, 0, + 0, 0, mRadius, 0, + 0, 0, 0, 1); + + // Initialize the position where the cylinder will be rendered + translateWorld(position); + + // Create the collision shape for the rigid body (cylinder shape) + // ReactPhysics3D will clone this object to create an internal one. Therefore, + // it is OK if this object is destroy right after calling Dynamics::createRigidBody() + const rp3d::CylinderShape collisionShape(mRadius, mHeight); + + // Compute the inertia tensor of the body using its collision shape + rp3d::Matrix3x3 inertiaTensor; + collisionShape.computeLocalInertiaTensor(inertiaTensor, mass); + + // Initial position and orientation of the rigid body + rp3d::Vector3 initPosition(position.x, position.y, position.z); + rp3d::Quaternion initOrientation = rp3d::Quaternion::identity(); + rp3d::Transform transform(initPosition, initOrientation); + + // Create a rigid body corresponding to the cylinder in the dynamics world + mRigidBody = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, collisionShape); +} + +// Destructor +Cylinder::~Cylinder() { + + // Destroy the mesh + destroy(); +} + +// Render the cylinder at the correct position and with the correct orientation +void Cylinder::render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix) { + + // Bind the shader + shader.bind(); + + // Set the model to camera matrix + const openglframework::Matrix4 localToCameraMatrix = worldToCameraMatrix * mTransformMatrix; + shader.setMatrix4x4Uniform("localToCameraMatrix", localToCameraMatrix); + + // Set the normal matrix (inverse transpose of the 3x3 upper-left sub matrix of the + // model-view matrix) + const openglframework::Matrix3 normalMatrix = + localToCameraMatrix.getUpperLeft3x3Matrix().getInverse().getTranspose(); + shader.setMatrix3x3Uniform("normalMatrix", normalMatrix); + + glEnableClientState(GL_VERTEX_ARRAY); + glEnableClientState(GL_NORMAL_ARRAY); + if (hasTexture()) { + glEnableClientState(GL_TEXTURE_COORD_ARRAY); + } + + glVertexPointer(3, GL_FLOAT, 0, getVerticesPointer()); + glNormalPointer(GL_FLOAT, 0, getNormalsPointer()); + if(hasTexture()) { + glTexCoordPointer(2, GL_FLOAT, 0, getUVTextureCoordinatesPointer()); + } + + // For each part of the mesh + for (unsigned int i=0; igetInterpolatedTransform(); + + // Compute the transform used for rendering the cylinder + float matrix[16]; + transform.getOpenGLMatrix(matrix); + openglframework::Matrix4 newMatrix(matrix[0], matrix[4], matrix[8], matrix[12], + matrix[1], matrix[5], matrix[9], matrix[13], + matrix[2], matrix[6], matrix[10], matrix[14], + matrix[3], matrix[7], matrix[11], matrix[15]); + + // Apply the scaling matrix to have the correct cylinder dimensions + mTransformMatrix = newMatrix * mScalingMatrix; +} diff --git a/examples/common/Cylinder.h b/examples/common/Cylinder.h new file mode 100644 index 00000000..e094580d --- /dev/null +++ b/examples/common/Cylinder.h @@ -0,0 +1,81 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 CYLINDER_H +#define CYLINDER_H + +// Libraries +#include "openglframework.h" +#include "reactphysics3d.h" + +// Class Cylinder +class Cylinder : public openglframework::Mesh { + + private : + + // -------------------- Attributes -------------------- // + + /// Radius of the cylinder + float mRadius; + + /// Height of the cylinder + float mHeight; + + /// Rigid body used to simulate the dynamics of the cylinder + rp3d::RigidBody* mRigidBody; + + /// Scaling matrix (applied to a sphere to obtain the correct cylinder dimensions) + openglframework::Matrix4 mScalingMatrix; + + // -------------------- Methods -------------------- // + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + Cylinder(float radius, float height, const openglframework::Vector3& position, + float mass, rp3d::DynamicsWorld* dynamicsWorld); + + /// Destructor + ~Cylinder(); + + /// Return a pointer to the rigid body of the cylinder + rp3d::RigidBody* getRigidBody(); + + /// Update the transform matrix of the cylinder + void updateTransform(); + + /// Render the cylinder at the correct position and with the correct orientation + void render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix); +}; + +// Return a pointer to the rigid body of the cylinder +inline rp3d::RigidBody* Cylinder::getRigidBody() { + return mRigidBody; +} + +#endif diff --git a/examples/common/Sphere.cpp b/examples/common/Sphere.cpp new file mode 100644 index 00000000..f3d20010 --- /dev/null +++ b/examples/common/Sphere.cpp @@ -0,0 +1,136 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Sphere.h" + + +// Constructor +Sphere::Sphere(float radius, const openglframework::Vector3 &position, + float mass, reactphysics3d::DynamicsWorld* dynamicsWorld) + : openglframework::Mesh(), mRadius(radius) { + + // Load the mesh from a file + openglframework::MeshReaderWriter::loadMeshFromFile("meshes/sphere.obj", *this); + + // Calculate the normals of the mesh + calculateNormals(); + + // Compute the scaling matrix + mScalingMatrix = openglframework::Matrix4(mRadius, 0, 0, 0, + 0, mRadius, 0, 0, + 0, 0, mRadius, 0, + 0, 0, 0, 1); + + // Initialize the position where the sphere will be rendered + translateWorld(position); + + // Create the collision shape for the rigid body (sphere shape) + // ReactPhysics3D will clone this object to create an internal one. Therefore, + // it is OK if this object is destroy right after calling Dynamics::createRigidBody() + const rp3d::SphereShape collisionShape(mRadius); + + // Compute the inertia tensor of the body using its collision shape + rp3d::Matrix3x3 inertiaTensor; + collisionShape.computeLocalInertiaTensor(inertiaTensor, mass); + + // Initial position and orientation of the rigid body + rp3d::Vector3 initPosition(position.x, position.y, position.z); + rp3d::Quaternion initOrientation = rp3d::Quaternion::identity(); + rp3d::Transform transform(initPosition, initOrientation); + + // Create a rigid body corresponding to the sphere in the dynamics world + mRigidBody = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, collisionShape); +} + +// Destructor +Sphere::~Sphere() { + + // Destroy the mesh + destroy(); +} + +// Render the sphere at the correct position and with the correct orientation +void Sphere::render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix) { + + // Bind the shader + shader.bind(); + + // Set the model to camera matrix + const openglframework::Matrix4 localToCameraMatrix = worldToCameraMatrix * mTransformMatrix; + shader.setMatrix4x4Uniform("localToCameraMatrix", localToCameraMatrix); + + // Set the normal matrix (inverse transpose of the 3x3 upper-left sub matrix of the + // model-view matrix) + const openglframework::Matrix3 normalMatrix = + localToCameraMatrix.getUpperLeft3x3Matrix().getInverse().getTranspose(); + shader.setMatrix3x3Uniform("normalMatrix", normalMatrix); + + glEnableClientState(GL_VERTEX_ARRAY); + glEnableClientState(GL_NORMAL_ARRAY); + if (hasTexture()) { + glEnableClientState(GL_TEXTURE_COORD_ARRAY); + } + + glVertexPointer(3, GL_FLOAT, 0, getVerticesPointer()); + glNormalPointer(GL_FLOAT, 0, getNormalsPointer()); + if(hasTexture()) { + glTexCoordPointer(2, GL_FLOAT, 0, getUVTextureCoordinatesPointer()); + } + + // For each part of the mesh + for (unsigned int i=0; igetInterpolatedTransform(); + + // Compute the transform used for rendering the sphere + float matrix[16]; + transform.getOpenGLMatrix(matrix); + openglframework::Matrix4 newMatrix(matrix[0], matrix[4], matrix[8], matrix[12], + matrix[1], matrix[5], matrix[9], matrix[13], + matrix[2], matrix[6], matrix[10], matrix[14], + matrix[3], matrix[7], matrix[11], matrix[15]); + + // Apply the scaling matrix to have the correct sphere dimensions + mTransformMatrix = newMatrix * mScalingMatrix; +} diff --git a/examples/common/Sphere.h b/examples/common/Sphere.h new file mode 100644 index 00000000..dccef0eb --- /dev/null +++ b/examples/common/Sphere.h @@ -0,0 +1,78 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 SPHERE_H +#define SPHERE_H + +// Libraries +#include "openglframework.h" +#include "reactphysics3d.h" + +// Class Sphere +class Sphere : public openglframework::Mesh { + + private : + + // -------------------- Attributes -------------------- // + + /// Radius of the sphere + float mRadius; + + /// Rigid body used to simulate the dynamics of the sphere + rp3d::RigidBody* mRigidBody; + + /// Scaling matrix (applied to a sphere to obtain the correct sphere dimensions) + openglframework::Matrix4 mScalingMatrix; + + // -------------------- Methods -------------------- // + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + Sphere(float radius, const openglframework::Vector3& position, + float mass, rp3d::DynamicsWorld* dynamicsWorld); + + /// Destructor + ~Sphere(); + + /// Return a pointer to the rigid body of the sphere + rp3d::RigidBody* getRigidBody(); + + /// Update the transform matrix of the sphere + void updateTransform(); + + /// Render the sphere at the correct position and with the correct orientation + void render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix); +}; + +// Return a pointer to the rigid body of the sphere +inline rp3d::RigidBody* Sphere::getRigidBody() { + return mRigidBody; +} + +#endif diff --git a/examples/common/VisualContactPoint.cpp b/examples/common/VisualContactPoint.cpp new file mode 100644 index 00000000..8cbfa489 --- /dev/null +++ b/examples/common/VisualContactPoint.cpp @@ -0,0 +1,115 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "VisualContactPoint.h" + +// Initialization of static variables +int VisualContactPoint::mNbTotalPoints = 0; +bool VisualContactPoint::mIsMeshInitialized = false; +openglframework::Mesh VisualContactPoint::mMesh; + +// Constructor +VisualContactPoint::VisualContactPoint(const openglframework::Vector3 &position) { + + assert(mIsMeshInitialized); + + // Initialize the position where the sphere will be rendered + translateWorld(position); +} + +// Destructor +VisualContactPoint::~VisualContactPoint() { + +} + +// Load and initialize the mesh for all the contact points +void VisualContactPoint::createStaticData() { + + if (!mIsMeshInitialized) { + + // Load the mesh from a file + openglframework::MeshReaderWriter::loadMeshFromFile("meshes/sphere.obj", mMesh); + + // Calculate the normals of the mesh + mMesh.calculateNormals(); + + mMesh.scaleVertices(VISUAL_CONTACT_POINT_RADIUS); + + mIsMeshInitialized = true; + } +} + +// Destroy the mesh for the contact points +void VisualContactPoint::destroyStaticData() { + + mMesh.destroy(); + mIsMeshInitialized = false; +} + +// Render the sphere at the correct position and with the correct orientation +void VisualContactPoint::render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix) { + + // Bind the shader + shader.bind(); + + // Set the model to camera matrix + const openglframework::Matrix4 localToCameraMatrix = worldToCameraMatrix * mTransformMatrix; + shader.setMatrix4x4Uniform("localToCameraMatrix", localToCameraMatrix); + + // Set the normal matrix (inverse transpose of the 3x3 upper-left sub matrix of the + // model-view matrix) + const openglframework::Matrix3 normalMatrix = + localToCameraMatrix.getUpperLeft3x3Matrix().getInverse().getTranspose(); + shader.setMatrix3x3Uniform("normalMatrix", normalMatrix); + + glEnableClientState(GL_VERTEX_ARRAY); + glEnableClientState(GL_NORMAL_ARRAY); + if (mMesh.hasTexture()) { + glEnableClientState(GL_TEXTURE_COORD_ARRAY); + } + + glVertexPointer(3, GL_FLOAT, 0, mMesh.getVerticesPointer()); + glNormalPointer(GL_FLOAT, 0, mMesh.getNormalsPointer()); + if(mMesh.hasTexture()) { + glTexCoordPointer(2, GL_FLOAT, 0, mMesh.getUVTextureCoordinatesPointer()); + } + + // For each part of the mesh + for (unsigned int i=0; i #include "Vector3.h" #include "Vector4.h" +#include "Matrix3.h" namespace openglframework { @@ -233,6 +234,13 @@ class Matrix4 { m[0][3], m[1][3], m[2][3], m[3][3]); } + // Return the 3x3 upper-left matrix + Matrix3 getUpperLeft3x3Matrix() const { + return Matrix3(m[0][0], m[0][1], m[0][2], + m[1][0], m[1][1], m[1][2], + m[2][0], m[2][1], m[2][2]); + } + // Return the inversed matrix Matrix4 getInverse() const { int indxc[4], indxr[4]; diff --git a/examples/common/opengl-framework/src/shaders/phong.frag b/examples/common/opengl-framework/src/shaders/phong.frag index 8995f7f0..6f38b4b4 100644 --- a/examples/common/opengl-framework/src/shaders/phong.frag +++ b/examples/common/opengl-framework/src/shaders/phong.frag @@ -24,8 +24,7 @@ ********************************************************************************/ // Uniform variables -uniform vec3 cameraWorldPosition; // World position of the camera -uniform vec3 lightWorldPosition; // World position of the light +uniform vec3 lightPosCameraSpace; // Camera-space position of the light uniform vec3 lightAmbientColor; // Lights ambient color uniform vec3 lightDiffuseColor; // Light diffuse color uniform vec3 lightSpecularColor; // Light specular color @@ -34,9 +33,9 @@ uniform sampler2D texture; // Texture uniform bool isTexture; // True if we need to use the texture // Varying variables -varying vec3 worldPosition; // World position of the vertex -varying vec3 worldNormal; // World surface normalWorld -varying vec2 texCoords; // Texture coordinates +varying vec3 vertexPosCameraSpace; // Camera-space position of the vertex +varying vec3 vertexNormalCameraSpace; // Vertex normal in camera-space +varying vec2 texCoords; // Texture coordinates void main() { @@ -44,19 +43,21 @@ void main() { vec3 ambient = lightAmbientColor; // Get the texture color - vec3 textureColor = vec3(1); if (isTexture) textureColor = texture2D(texture, texCoords).rgb; // Compute the diffuse term - vec3 L = normalize(lightWorldPosition - worldPosition); - vec3 N = normalize(worldNormal); - vec3 diffuse = lightDiffuseColor * max(dot(N, L), 0.0) * textureColor; + vec3 L = normalize(lightPosCameraSpace - vertexPosCameraSpace); + vec3 N = normalize(vertexNormalCameraSpace); + float diffuseFactor = max(dot(N, L), 0.0); + vec3 diffuse = lightDiffuseColor * diffuseFactor * textureColor; // Compute the specular term - vec3 V = normalize(cameraWorldPosition - worldPosition); + vec3 V = normalize(-vertexPosCameraSpace); vec3 H = normalize(V + L); - vec3 specular = lightSpecularColor * pow(max(dot(N, H), 0), shininess); + float specularFactor = pow(max(dot(N, H), 0), shininess); + if (diffuseFactor < 0) specularFactor = 0.0; + vec3 specular = lightSpecularColor * specularFactor; // Compute the final color gl_FragColor = vec4(ambient + diffuse + specular, 1.0); diff --git a/examples/common/opengl-framework/src/shaders/phong.vert b/examples/common/opengl-framework/src/shaders/phong.vert index 97bca225..620344a9 100644 --- a/examples/common/opengl-framework/src/shaders/phong.vert +++ b/examples/common/opengl-framework/src/shaders/phong.vert @@ -24,28 +24,27 @@ ********************************************************************************/ // Uniform variables -uniform mat4 modelToWorldMatrix; // Model too world coordinates matrix -uniform mat4 worldToCameraMatrix; // World to camera coordinates matrix +uniform mat4 localToCameraMatrix; // Local-space to camera-space matrix uniform mat4 projectionMatrix; // Projection matrix +uniform mat3 normalMatrix; // Normal matrix // Varying variables -varying vec3 worldPosition; // World position of the vertex -varying vec3 worldNormal; // World surface normalWorld +varying vec3 vertexPosCameraSpace; // Camera-space position of the vertex +varying vec3 vertexNormalCameraSpace; // Vertex normal in camera-space varying vec2 texCoords; // Texture coordinates void main() { // Compute the vertex position - vec4 worldPos = modelToWorldMatrix * gl_Vertex; - worldPosition = worldPos.xyz; + vec4 positionCameraSpace = localToCameraMatrix * gl_Vertex; + vertexPosCameraSpace = positionCameraSpace.xyz; // Compute the world surface normal - vec3 bodyNormal = normalize(gl_Normal); - worldNormal = (modelToWorldMatrix * vec4(bodyNormal, 0.0)).xyz; + vertexNormalCameraSpace = normalMatrix * gl_Normal; // Get the texture coordinates texCoords = gl_MultiTexCoord0.xy; // Compute the clip-space vertex coordinates - gl_Position = projectionMatrix * worldToCameraMatrix * worldPos; + gl_Position = projectionMatrix * positionCameraSpace; } diff --git a/examples/fallingcubes/Scene.cpp b/examples/fallingcubes/Scene.cpp index b11f3a84..4e596bb5 100644 --- a/examples/fallingcubes/Scene.cpp +++ b/examples/fallingcubes/Scene.cpp @@ -59,7 +59,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), float radius = 2.0f; // Create all the cubes of the scene - for (int i=0; igetRigidBody()->setIsMotionEnabled(true); @@ -151,17 +151,16 @@ void Scene::render() { glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); glEnable(GL_CULL_FACE); + // Get the world-space to camera-space matrix + const Camera& camera = mViewer->getCamera(); + const openglframework::Matrix4 worldToCameraMatrix = camera.getTransformMatrix().getInverse(); + // Bind the shader mPhongShader.bind(); // Set the variables of the shader - const Camera& camera = mViewer->getCamera(); - Matrix4 matrixIdentity; - matrixIdentity.setToIdentity(); - mPhongShader.setVector3Uniform("cameraWorldPosition", mViewer->getCamera().getOrigin()); - mPhongShader.setMatrix4x4Uniform("worldToCameraMatrix", camera.getTransformMatrix().getInverse()); mPhongShader.setMatrix4x4Uniform("projectionMatrix", camera.getProjectionMatrix()); - mPhongShader.setVector3Uniform("lightWorldPosition", mLight0.getOrigin()); + mPhongShader.setVector3Uniform("lightPosCameraSpace",worldToCameraMatrix * mLight0.getOrigin()); mPhongShader.setVector3Uniform("lightAmbientColor", Vector3(0.3f, 0.3f, 0.3f)); Color& diffCol = mLight0.getDiffuseColor(); Color& specCol = mLight0.getSpecularColor(); @@ -171,11 +170,11 @@ void Scene::render() { // Render all the cubes of the scene for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { - (*it)->render(mPhongShader); + (*it)->render(mPhongShader, worldToCameraMatrix); } // Render the floor - mFloor->render(mPhongShader); + mFloor->render(mPhongShader, worldToCameraMatrix); // Unbind the shader mPhongShader.unbind(); diff --git a/examples/fallingcubes/Scene.h b/examples/fallingcubes/Scene.h index d2dbfc18..d76e8c49 100644 --- a/examples/fallingcubes/Scene.h +++ b/examples/fallingcubes/Scene.h @@ -32,10 +32,10 @@ #include "Box.h" // Constants -const int NB_BOXES = 20; // Number of boxes in the scene +const int NB_SPHERES = 20; // Number of boxes in the scene const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters -const float BOX_MASS = 1.0f; // Box mass in kilograms +const float SPHERE_MASS = 1.0f; // Box mass in kilograms const float FLOOR_MASS = 100.0f; // Floor mass in kilograms // Class Scene diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index ce278b63..1adceb36 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -156,15 +156,15 @@ void Scene::render() { glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); glEnable(GL_CULL_FACE); + // Get the world-space to camera-space matrix + const Camera& camera = mViewer->getCamera(); + const openglframework::Matrix4 worldToCameraMatrix = camera.getTransformMatrix().getInverse(); + // Bind the shader mPhongShader.bind(); // Set the variables of the shader - const Camera& camera = mViewer->getCamera(); - Matrix4 matrixIdentity; - matrixIdentity.setToIdentity(); - mPhongShader.setVector3Uniform("cameraWorldPosition", mViewer->getCamera().getOrigin()); - mPhongShader.setMatrix4x4Uniform("worldToCameraMatrix", camera.getTransformMatrix().getInverse()); + mPhongShader.setVector3Uniform("lightPosCameraSpace",worldToCameraMatrix * mLight0.getOrigin()); mPhongShader.setMatrix4x4Uniform("projectionMatrix", camera.getProjectionMatrix()); mPhongShader.setVector3Uniform("lightWorldPosition", mLight0.getOrigin()); mPhongShader.setVector3Uniform("lightAmbientColor", Vector3(0.3f, 0.3f, 0.3f)); @@ -175,17 +175,17 @@ void Scene::render() { mPhongShader.setFloatUniform("shininess", 60.0f); // Render all the boxes - mSliderJointBottomBox->render(mPhongShader); - mSliderJointTopBox->render(mPhongShader); - mPropellerBox->render(mPhongShader); - mFixedJointBox1->render(mPhongShader); - mFixedJointBox2->render(mPhongShader); + mSliderJointBottomBox->render(mPhongShader, worldToCameraMatrix); + mSliderJointTopBox->render(mPhongShader, worldToCameraMatrix); + mPropellerBox->render(mPhongShader, worldToCameraMatrix); + mFixedJointBox1->render(mPhongShader, worldToCameraMatrix); + mFixedJointBox2->render(mPhongShader, worldToCameraMatrix); for (int i=0; irender(mPhongShader); + mBallAndSocketJointChainBoxes[i]->render(mPhongShader, worldToCameraMatrix); } // Render the floor - mFloor->render(mPhongShader); + mFloor->render(mPhongShader, worldToCameraMatrix); // Unbind the shader mPhongShader.unbind(); @@ -244,7 +244,7 @@ void Scene::createSliderJoint() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 box1Dimension(2, 4, 2); - mSliderJointBottomBox = new Box(box1Dimension, positionBox1 , BOX_MASS, mDynamicsWorld); + mSliderJointBottomBox = new Box(box1Dimension, positionBox1 , SPHERE_MASS, mDynamicsWorld); // The fist box cannot move mSliderJointBottomBox->getRigidBody()->setIsMotionEnabled(false); @@ -259,7 +259,7 @@ void Scene::createSliderJoint() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 box2Dimension(1.5, 4, 1.5); - mSliderJointTopBox = new Box(box2Dimension, positionBox2 , BOX_MASS, mDynamicsWorld); + mSliderJointTopBox = new Box(box2Dimension, positionBox2 , SPHERE_MASS, mDynamicsWorld); // The second box is allowed to move mSliderJointTopBox->getRigidBody()->setIsMotionEnabled(true); @@ -297,7 +297,7 @@ void Scene::createPropellerHingeJoint() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 boxDimension(10, 1, 1); - mPropellerBox = new Box(boxDimension, positionBox1 , BOX_MASS, mDynamicsWorld); + mPropellerBox = new Box(boxDimension, positionBox1 , SPHERE_MASS, mDynamicsWorld); // The fist box cannot move mPropellerBox->getRigidBody()->setIsMotionEnabled(true); @@ -334,7 +334,7 @@ void Scene::createFixedJoints() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 boxDimension(1.5, 1.5, 1.5); - mFixedJointBox1 = new Box(boxDimension, positionBox1 , BOX_MASS, mDynamicsWorld); + mFixedJointBox1 = new Box(boxDimension, positionBox1 , SPHERE_MASS, mDynamicsWorld); // The fist box cannot move mFixedJointBox1->getRigidBody()->setIsMotionEnabled(true); @@ -348,7 +348,7 @@ void Scene::createFixedJoints() { openglframework::Vector3 positionBox2(-5, 7, 0); // Create a box and a corresponding rigid in the dynamics world - mFixedJointBox2 = new Box(boxDimension, positionBox2 , BOX_MASS, mDynamicsWorld); + mFixedJointBox2 = new Box(boxDimension, positionBox2 , SPHERE_MASS, mDynamicsWorld); // The second box is allowed to move mFixedJointBox2->getRigidBody()->setIsMotionEnabled(true); diff --git a/examples/joints/Scene.h b/examples/joints/Scene.h index 90430d1d..d65f3ba8 100644 --- a/examples/joints/Scene.h +++ b/examples/joints/Scene.h @@ -34,7 +34,7 @@ // Constants const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters -const float BOX_MASS = 1.0f; // Box mass in kilograms +const float SPHERE_MASS = 1.0f; // Box mass in kilograms const float FLOOR_MASS = 100.0f; // Floor mass in kilograms const int NB_BALLSOCKETJOINT_BOXES = 7; // Number of Ball-And-Socket chain boxes const int NB_HINGE_BOXES = 7; // Number of Hinge chain boxes diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index f911bd08..a1a1f821 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -224,6 +224,9 @@ public : /// Return an iterator to the end of the rigid bodies of the physics world std::set::iterator getRigidBodiesEndIterator(); + + /// Return a reference to the contact manifolds of the world + const std::vector& getContactManifolds() const; }; // Start the physics simulation @@ -333,12 +336,17 @@ inline std::set::iterator DynamicsWorld::getRigidBodiesEndIterator() return mRigidBodies.end(); } +// Return a reference to the contact manifolds of the world +inline const std::vector& DynamicsWorld::getContactManifolds() const { + return mContactManifolds; +} + // Return the number of contact manifolds in the world inline uint DynamicsWorld::getNbContactManifolds() const { return mContactManifolds.size(); } -/// Return the current physics time (in seconds) +// Return the current physics time (in seconds) inline long double DynamicsWorld::getPhysicsTime() const { return mTimer.getPhysicsTime(); } From 84d946ac48b343085079e2ed1324115ec1cbf865 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 3 Jul 2013 22:42:58 +0200 Subject: [PATCH 37/66] Modifications of the examples --- .../opengl-framework/src/shaders/phong.frag | 26 ++++++++++--------- examples/fallingcubes/Scene.cpp | 10 +++---- examples/fallingcubes/Scene.h | 4 +-- examples/joints/Scene.cpp | 19 +++++++------- examples/joints/Scene.h | 2 +- src/collision/shapes/ConeShape.cpp | 10 +++---- 6 files changed, 37 insertions(+), 34 deletions(-) diff --git a/examples/common/opengl-framework/src/shaders/phong.frag b/examples/common/opengl-framework/src/shaders/phong.frag index 6f38b4b4..3102de06 100644 --- a/examples/common/opengl-framework/src/shaders/phong.frag +++ b/examples/common/opengl-framework/src/shaders/phong.frag @@ -6,7 +6,7 @@ * 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: * @@ -24,10 +24,10 @@ ********************************************************************************/ // Uniform variables -uniform vec3 lightPosCameraSpace; // Camera-space position of the light uniform vec3 lightAmbientColor; // Lights ambient color -uniform vec3 lightDiffuseColor; // Light diffuse color -uniform vec3 lightSpecularColor; // Light specular color +uniform vec3 light0PosCameraSpace; // Camera-space position of the light +uniform vec3 light0DiffuseColor; // Light 0 diffuse color +uniform vec3 light0SpecularColor; // Light 0 specular color uniform float shininess; // Shininess uniform sampler2D texture; // Texture uniform bool isTexture; // True if we need to use the texture @@ -46,18 +46,20 @@ void main() { vec3 textureColor = vec3(1); if (isTexture) textureColor = texture2D(texture, texCoords).rgb; - // Compute the diffuse term - vec3 L = normalize(lightPosCameraSpace - vertexPosCameraSpace); + // Compute the surface normal vector vec3 N = normalize(vertexNormalCameraSpace); - float diffuseFactor = max(dot(N, L), 0.0); - vec3 diffuse = lightDiffuseColor * diffuseFactor * textureColor; - // Compute the specular term + // Compute the diffuse term of light 0 + vec3 L0 = normalize(light0PosCameraSpace - vertexPosCameraSpace); + float diffuseFactor = max(dot(N, L0), 0.0); + vec3 diffuse = light0DiffuseColor * diffuseFactor * textureColor; + + // Compute the specular term of light 0 vec3 V = normalize(-vertexPosCameraSpace); - vec3 H = normalize(V + L); - float specularFactor = pow(max(dot(N, H), 0), shininess); + vec3 H0 = normalize(V + L0); + float specularFactor = pow(max(dot(N, H0), 0), shininess); if (diffuseFactor < 0) specularFactor = 0.0; - vec3 specular = lightSpecularColor * specularFactor; + vec3 specular = light0SpecularColor * specularFactor; // Compute the final color gl_FragColor = vec4(ambient + diffuse + specular, 1.0); diff --git a/examples/fallingcubes/Scene.cpp b/examples/fallingcubes/Scene.cpp index 4e596bb5..fbc80ddb 100644 --- a/examples/fallingcubes/Scene.cpp +++ b/examples/fallingcubes/Scene.cpp @@ -59,7 +59,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), float radius = 2.0f; // Create all the cubes of the scene - for (int i=0; igetRigidBody()->setIsMotionEnabled(true); @@ -160,12 +160,12 @@ void Scene::render() { // Set the variables of the shader mPhongShader.setMatrix4x4Uniform("projectionMatrix", camera.getProjectionMatrix()); - mPhongShader.setVector3Uniform("lightPosCameraSpace",worldToCameraMatrix * mLight0.getOrigin()); + mPhongShader.setVector3Uniform("light0PosCameraSpace",worldToCameraMatrix * mLight0.getOrigin()); mPhongShader.setVector3Uniform("lightAmbientColor", Vector3(0.3f, 0.3f, 0.3f)); Color& diffCol = mLight0.getDiffuseColor(); Color& specCol = mLight0.getSpecularColor(); - mPhongShader.setVector3Uniform("lightDiffuseColor", Vector3(diffCol.r, diffCol.g, diffCol.b)); - mPhongShader.setVector3Uniform("lightSpecularColor", Vector3(specCol.r, specCol.g, specCol.b)); + mPhongShader.setVector3Uniform("light0DiffuseColor", Vector3(diffCol.r, diffCol.g, diffCol.b)); + mPhongShader.setVector3Uniform("light0SpecularColor", Vector3(specCol.r, specCol.g, specCol.b)); mPhongShader.setFloatUniform("shininess", 60.0f); // Render all the cubes of the scene diff --git a/examples/fallingcubes/Scene.h b/examples/fallingcubes/Scene.h index d76e8c49..ae4c14d2 100644 --- a/examples/fallingcubes/Scene.h +++ b/examples/fallingcubes/Scene.h @@ -32,10 +32,10 @@ #include "Box.h" // Constants -const int NB_SPHERES = 20; // Number of boxes in the scene +const int NB_CUBES = 20; // Number of boxes in the scene const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters -const float SPHERE_MASS = 1.0f; // Box mass in kilograms +const float CUBE_MASS = 1.0f; // Box mass in kilograms const float FLOOR_MASS = 100.0f; // Floor mass in kilograms // Class Scene diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index 1adceb36..a66b8f30 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -164,14 +164,13 @@ void Scene::render() { mPhongShader.bind(); // Set the variables of the shader - mPhongShader.setVector3Uniform("lightPosCameraSpace",worldToCameraMatrix * mLight0.getOrigin()); + mPhongShader.setVector3Uniform("light0PosCameraSpace",worldToCameraMatrix * mLight0.getOrigin()); mPhongShader.setMatrix4x4Uniform("projectionMatrix", camera.getProjectionMatrix()); - mPhongShader.setVector3Uniform("lightWorldPosition", mLight0.getOrigin()); mPhongShader.setVector3Uniform("lightAmbientColor", Vector3(0.3f, 0.3f, 0.3f)); Color& diffCol = mLight0.getDiffuseColor(); Color& specCol = mLight0.getSpecularColor(); - mPhongShader.setVector3Uniform("lightDiffuseColor", Vector3(diffCol.r, diffCol.g, diffCol.b)); - mPhongShader.setVector3Uniform("lightSpecularColor", Vector3(specCol.r, specCol.g, specCol.b)); + mPhongShader.setVector3Uniform("light0DiffuseColor", Vector3(diffCol.r, diffCol.g, diffCol.b)); + mPhongShader.setVector3Uniform("light0SpecularColor", Vector3(specCol.r, specCol.g, specCol.b)); mPhongShader.setFloatUniform("shininess", 60.0f); // Render all the boxes @@ -244,7 +243,7 @@ void Scene::createSliderJoint() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 box1Dimension(2, 4, 2); - mSliderJointBottomBox = new Box(box1Dimension, positionBox1 , SPHERE_MASS, mDynamicsWorld); + mSliderJointBottomBox = new Box(box1Dimension, positionBox1 , CUBE_MASS, mDynamicsWorld); // The fist box cannot move mSliderJointBottomBox->getRigidBody()->setIsMotionEnabled(false); @@ -259,7 +258,7 @@ void Scene::createSliderJoint() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 box2Dimension(1.5, 4, 1.5); - mSliderJointTopBox = new Box(box2Dimension, positionBox2 , SPHERE_MASS, mDynamicsWorld); + mSliderJointTopBox = new Box(box2Dimension, positionBox2 , CUBE_MASS, mDynamicsWorld); // The second box is allowed to move mSliderJointTopBox->getRigidBody()->setIsMotionEnabled(true); @@ -297,7 +296,7 @@ void Scene::createPropellerHingeJoint() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 boxDimension(10, 1, 1); - mPropellerBox = new Box(boxDimension, positionBox1 , SPHERE_MASS, mDynamicsWorld); + mPropellerBox = new Box(boxDimension, positionBox1 , CUBE_MASS, mDynamicsWorld); // The fist box cannot move mPropellerBox->getRigidBody()->setIsMotionEnabled(true); @@ -334,7 +333,7 @@ void Scene::createFixedJoints() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 boxDimension(1.5, 1.5, 1.5); - mFixedJointBox1 = new Box(boxDimension, positionBox1 , SPHERE_MASS, mDynamicsWorld); + mFixedJointBox1 = new Box(boxDimension, positionBox1 , CUBE_MASS, mDynamicsWorld); // The fist box cannot move mFixedJointBox1->getRigidBody()->setIsMotionEnabled(true); @@ -348,7 +347,7 @@ void Scene::createFixedJoints() { openglframework::Vector3 positionBox2(-5, 7, 0); // Create a box and a corresponding rigid in the dynamics world - mFixedJointBox2 = new Box(boxDimension, positionBox2 , SPHERE_MASS, mDynamicsWorld); + mFixedJointBox2 = new Box(boxDimension, positionBox2 , CUBE_MASS, mDynamicsWorld); // The second box is allowed to move mFixedJointBox2->getRigidBody()->setIsMotionEnabled(true); @@ -363,6 +362,7 @@ void Scene::createFixedJoints() { rp3d::RigidBody* propellerBody = mPropellerBox->getRigidBody(); const rp3d::Vector3 anchorPointWorldSpace1(5, 7, 0); rp3d::FixedJointInfo jointInfo1(body1, propellerBody, anchorPointWorldSpace1); + jointInfo1.isCollisionEnabled = false; // Create the joint in the dynamics world mFixedJoint1 = dynamic_cast(mDynamicsWorld->createJoint(jointInfo1)); @@ -373,6 +373,7 @@ void Scene::createFixedJoints() { rp3d::RigidBody* body2 = mFixedJointBox2->getRigidBody(); const rp3d::Vector3 anchorPointWorldSpace2(-5, 7, 0); rp3d::FixedJointInfo jointInfo2(body2, propellerBody, anchorPointWorldSpace2); + jointInfo2.isCollisionEnabled = false; // Create the joint in the dynamics world mFixedJoint2 = dynamic_cast(mDynamicsWorld->createJoint(jointInfo2)); diff --git a/examples/joints/Scene.h b/examples/joints/Scene.h index d65f3ba8..1c3f72d5 100644 --- a/examples/joints/Scene.h +++ b/examples/joints/Scene.h @@ -34,7 +34,7 @@ // Constants const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters -const float SPHERE_MASS = 1.0f; // Box mass in kilograms +const float CUBE_MASS = 1.0f; // Box mass in kilograms const float FLOOR_MASS = 100.0f; // Floor mass in kilograms const int NB_BALLSOCKETJOINT_BOXES = 7; // Number of Ball-And-Socket chain boxes const int NB_HINGE_BOXES = 7; // Number of Hinge chain boxes diff --git a/src/collision/shapes/ConeShape.cpp b/src/collision/shapes/ConeShape.cpp index 58439866..b0be3dd3 100644 --- a/src/collision/shapes/ConeShape.cpp +++ b/src/collision/shapes/ConeShape.cpp @@ -45,12 +45,12 @@ using namespace reactphysics3d; // Constructor ConeShape::ConeShape(decimal radius, decimal height) - : CollisionShape(CONE), mRadius(radius), mHalfHeight(height / decimal(2.0)) { - assert(radius > 0.0); + : CollisionShape(CONE), mRadius(radius), mHalfHeight(height * decimal(0.5)) { + assert(mRadius > 0.0); assert(mHalfHeight > 0.0); // Compute the sine of the semi-angle at the apex point - mSinTheta = radius / (sqrt(radius * radius + height * height)); + mSinTheta = mRadius / (sqrt(mRadius * mRadius + height * height)); } // Private copy-constructor @@ -88,7 +88,7 @@ inline Vector3 ConeShape::getLocalSupportPointWithoutMargin(const Vector3& direc decimal sinThetaTimesLengthV = mSinTheta * v.length(); Vector3 supportPoint; - if (v.y >= sinThetaTimesLengthV) { + if (v.y > sinThetaTimesLengthV) { supportPoint = Vector3(0.0, mHalfHeight, 0.0); } else { @@ -98,7 +98,7 @@ inline Vector3 ConeShape::getLocalSupportPointWithoutMargin(const Vector3& direc supportPoint = Vector3(v.x * d, -mHalfHeight, v.z * d); } else { - supportPoint = Vector3(mRadius, -mHalfHeight, 0.0); + supportPoint = Vector3(0.0, -mHalfHeight, 0.0); } } From a3ca3598d5b77a87d6cee8b340b9611d3cb9cb5b Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 3 Jul 2013 22:50:00 +0200 Subject: [PATCH 38/66] Remove the margin gap for the BoxShape and make possible to choose the collision margin when creating a collision shape --- src/collision/narrowphase/GJK/GJKAlgorithm.h | 8 +- src/collision/shapes/AABB.cpp | 64 +--------------- src/collision/shapes/AABB.h | 5 -- src/collision/shapes/BoxShape.cpp | 80 +++----------------- src/collision/shapes/BoxShape.h | 40 +++------- src/collision/shapes/CollisionShape.cpp | 9 ++- src/collision/shapes/CollisionShape.h | 22 ++++-- src/collision/shapes/ConeShape.cpp | 36 ++------- src/collision/shapes/ConeShape.h | 43 +---------- src/collision/shapes/CylinderShape.cpp | 36 ++------- src/collision/shapes/CylinderShape.h | 37 +-------- src/collision/shapes/SphereShape.cpp | 29 +------ src/collision/shapes/SphereShape.h | 35 ++------- src/configuration.h | 2 +- src/constraint/ContactPoint.h | 25 ------ 15 files changed, 75 insertions(+), 396 deletions(-) diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.h b/src/collision/narrowphase/GJK/GJKAlgorithm.h index 5ce89f2e..a7c5a940 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.h +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.h @@ -45,12 +45,12 @@ const decimal REL_ERROR_SQUARE = REL_ERROR * REL_ERROR; * This class implements a narrow-phase collision detection algorithm. This * algorithm uses the ISA-GJK algorithm and the EPA algorithm. This * implementation is based on the implementation discussed in the book - * "Collision Detection in 3D Environments". + * "Collision Detection in Interactive 3D Environments" by Gino van den Bergen. * This method implements the Hybrid Technique for calculating the * penetration depth. The two objects are enlarged with a small margin. If - * the object intersection, the penetration depth is quickly computed using - * GJK algorithm on the original objects (without margin). If the - * original objects (without margin) intersect, we run again the GJK + * the object intersects in their margins, the penetration depth is quickly + * computed using the GJK algorithm on the original objects (without margin). + * If the original objects (without margin) intersect, we run again the GJK * algorithm on the enlarged objects (with margin) to compute simplex * polytope that contains the origin and give it to the EPA (Expanding * Polytope Algorithm) to compute the correct penetration depth between the diff --git a/src/collision/shapes/AABB.cpp b/src/collision/shapes/AABB.cpp index 93498d70..4e65edef 100644 --- a/src/collision/shapes/AABB.cpp +++ b/src/collision/shapes/AABB.cpp @@ -28,25 +28,12 @@ #include "../../configuration.h" #include -#if defined(VISUAL_DEBUG) - #if defined(APPLE_OS) - #include - #include - #elif defined(WINDOWS_OS) - #include - #include - #elif defined(LINUX_OS) - #include - #include - #endif -#endif - using namespace reactphysics3d; using namespace std; // Constructor AABB::AABB() { - + } // Constructor @@ -60,52 +47,3 @@ AABB::~AABB() { } -#ifdef VISUAL_DEBUG -// Draw the AABB (only for testing purpose) -void AABB::draw() const { - - // Draw in red - glColor3f(1.0, 0.0, 0.0); - - // Draw the AABB - glBegin(GL_LINES); - glVertex3f(mMaxCoordinates.x, mMinCoordinates.y, mMinCoordinates.z); - glVertex3f(mMaxCoordinates.x, mMaxCoordinates.y, mMinCoordinates.z); - - glVertex3f(mMaxCoordinates.x, mMinCoordinates.y, mMinCoordinates.z); - glVertex3f(mMaxCoordinates.x, mMinCoordinates.y, mMaxCoordinates.z); - - glVertex3f(mMaxCoordinates.x, mMinCoordinates.y, mMaxCoordinates.z); - glVertex3f(mMaxCoordinates.x, mMaxCoordinates.y, mMaxCoordinates.z); - - glVertex3f(mMaxCoordinates.x, mMaxCoordinates.y, mMinCoordinates.z); - glVertex3f(mMaxCoordinates.x, mMaxCoordinates.y, mMaxCoordinates.z); - - glVertex3f(mMinCoordinates.x, mMinCoordinates.y, mMinCoordinates.z); - glVertex3f(mMinCoordinates.x, mMaxCoordinates.y, mMinCoordinates.z); - - glVertex3f(mMinCoordinates.x, mMinCoordinates.y, mMinCoordinates.z); - glVertex3f(mMinCoordinates.x, mMinCoordinates.y, mMaxCoordinates.z); - - glVertex3f(mMinCoordinates.x, mMinCoordinates.y, mMaxCoordinates.z); - glVertex3f(mMinCoordinates.x, mMaxCoordinates.y, mMaxCoordinates.z); - - glVertex3f(mMinCoordinates.x, mMaxCoordinates.y, mMinCoordinates.z); - glVertex3f(mMinCoordinates.x, mMaxCoordinates.y, mMaxCoordinates.z); - - glVertex3f(mMinCoordinates.x, mMinCoordinates.y, mMinCoordinates.z); - glVertex3f(mMaxCoordinates.x, mMinCoordinates.y, mMinCoordinates.z); - - glVertex3f(mMinCoordinates.x, mMaxCoordinates.y, mMinCoordinates.z); - glVertex3f(mMaxCoordinates.x, mMaxCoordinates.y, mMinCoordinates.z); - - glVertex3f(mMinCoordinates.x, mMaxCoordinates.y, mMaxCoordinates.z); - glVertex3f(mMaxCoordinates.x, mMaxCoordinates.y, mMaxCoordinates.z); - - glVertex3f(mMinCoordinates.x, mMinCoordinates.y, mMaxCoordinates.z); - glVertex3f(mMaxCoordinates.x, mMinCoordinates.y, mMaxCoordinates.z); - - glEnd(); -} -#endif - diff --git a/src/collision/shapes/AABB.h b/src/collision/shapes/AABB.h index e92e43e1..ad6dd20b 100644 --- a/src/collision/shapes/AABB.h +++ b/src/collision/shapes/AABB.h @@ -97,11 +97,6 @@ class AABB { /// Return true if the current AABB is overlapping with the AABB in argument bool testCollision(const AABB& aabb) const; - -#ifdef VISUAL_DEBUG - /// Draw the AABB (only for testing purpose) - virtual void draw() const; -#endif }; // Return the center point of the AABB in world coordinates diff --git a/src/collision/shapes/BoxShape.cpp b/src/collision/shapes/BoxShape.cpp index 111e8397..221f5037 100644 --- a/src/collision/shapes/BoxShape.cpp +++ b/src/collision/shapes/BoxShape.cpp @@ -29,25 +29,16 @@ #include #include -#if defined(VISUAL_DEBUG) - #if defined(APPLE_OS) - #include - #include - #elif defined(WINDOWS_OS) - #include - #include - #elif defined(LINUX_OS) - #include - #include - #endif -#endif - using namespace reactphysics3d; using namespace std; // Constructor -BoxShape::BoxShape(const Vector3& extent) : CollisionShape(BOX), mExtent(extent) { - +BoxShape::BoxShape(const Vector3& extent, decimal margin) + : CollisionShape(BOX, margin), mExtent(extent - Vector3(margin, margin, margin)) { + assert(extent.x > decimal(0.0) && extent.x > margin); + assert(extent.y > decimal(0.0) && extent.y > margin); + assert(extent.z > decimal(0.0) && extent.z > margin); + assert(margin > decimal(0.0)); } // Private copy-constructor @@ -63,62 +54,11 @@ BoxShape::~BoxShape() { // Return the local inertia tensor of the collision shape void BoxShape::computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const { decimal factor = (decimal(1.0) / decimal(3.0)) * mass; - decimal xSquare = mExtent.x * mExtent.x; - decimal ySquare = mExtent.y * mExtent.y; - decimal zSquare = mExtent.z * mExtent.z; + Vector3 realExtent = mExtent + Vector3(mMargin, mMargin, mMargin); + decimal xSquare = realExtent.x * realExtent.x; + decimal ySquare = realExtent.y * realExtent.y; + decimal zSquare = realExtent.z * realExtent.z; tensor.setAllValues(factor * (ySquare + zSquare), 0.0, 0.0, 0.0, factor * (xSquare + zSquare), 0.0, 0.0, 0.0, factor * (xSquare + ySquare)); } - -#ifdef VISUAL_DEBUG -// Draw the Box (only for testing purpose) -void BoxShape::draw() const { - decimal e1 = mExtent.x; - decimal e2 = mExtent.y; - decimal e3 = mExtent.z; - - // Draw in red - glColor3f(1.0, 0.0, 0.0); - - // Draw the Box - glBegin(GL_LINES); - glVertex3f(e1, -e2, -e3); - glVertex3f(e1, e2, -e3); - - glVertex3f(e1, -e2, -e3); - glVertex3f(e1, -e2, e3); - - glVertex3f(e1, -e2, e3); - glVertex3f(e1, e2, e3); - - glVertex3f(e1, e2, e3); - glVertex3f(e1, e2, -e3); - - glVertex3f(-e1, -e2, -e3); - glVertex3f(-e1, e2, -e3); - - glVertex3f(-e1, -e2, -e3); - glVertex3f(-e1, -e2, e3); - - glVertex3f(-e1, -e2, e3); - glVertex3f(-e1, e2, e3); - - glVertex3f(-e1, e2, e3); - glVertex3f(-e1, e2, -e3); - - glVertex3f(e1, -e2, -e3); - glVertex3f(-e1, -e2, -e3); - - glVertex3f(e1, -e2, -e3); - glVertex3f(-e1, -e2, -e3); - - glVertex3f(e1, -e2, e3); - glVertex3f(-e1, -e2, e3); - - glVertex3f(e1, e2, e3); - glVertex3f(-e1, e2, e3); - - glEnd(); -} -#endif diff --git a/src/collision/shapes/BoxShape.h b/src/collision/shapes/BoxShape.h index f8ea72c9..66afa348 100644 --- a/src/collision/shapes/BoxShape.h +++ b/src/collision/shapes/BoxShape.h @@ -64,7 +64,7 @@ class BoxShape : public CollisionShape { // -------------------- Methods -------------------- // /// Constructor - BoxShape(const Vector3& extent); + BoxShape(const Vector3& extent, decimal margin = OBJECT_MARGIN); /// Destructor virtual ~BoxShape(); @@ -75,14 +75,8 @@ class BoxShape : public CollisionShape { /// Return the extents of the box const Vector3& getExtent() const; - /// Set the extents of the box - void setExtent(const Vector3& extent); - /// Return the local extents in x,y and z direction. - virtual Vector3 getLocalExtents(decimal margin=0.0) const; - - /// Return the margin distance around the shape - virtual decimal getMargin() const; + virtual Vector3 getLocalExtents() const; /// Return the number of bytes used by the collision shape virtual size_t getSizeInBytes() const; @@ -98,11 +92,6 @@ class BoxShape : public CollisionShape { /// Test equality between two box shapes virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const; - -#ifdef VISUAL_DEBUG - /// Draw the Box (only for testing purpose) - virtual void draw() const; -#endif }; // Allocate and return a copy of the object @@ -112,23 +101,13 @@ inline BoxShape* BoxShape::clone(void* allocatedMemory) const { // Return the extents of the box inline const Vector3& BoxShape::getExtent() const { - return mExtent; -} - - // Set the extents of the box -inline void BoxShape::setExtent(const Vector3& extent) { - this->mExtent = extent; + return mExtent + Vector3(mMargin, mMargin, mMargin); } // Return the local extents of the box (half-width) in x,y and z local direction. /// This method is used to compute the AABB of the box -inline Vector3 BoxShape::getLocalExtents(decimal margin) const { - return mExtent + Vector3(getMargin(), getMargin(), getMargin()); -} - -// Return the margin distance around the shape -inline decimal BoxShape::getMargin() const { - return OBJECT_MARGIN; +inline Vector3 BoxShape::getLocalExtents() const { + return mExtent + Vector3(mMargin, mMargin, mMargin); } // Return the number of bytes used by the collision shape @@ -139,12 +118,11 @@ inline size_t BoxShape::getSizeInBytes() const { // Return a local support point in a given direction with the object margin inline Vector3 BoxShape::getLocalSupportPointWithMargin(const Vector3& direction) const { - decimal margin = getMargin(); - assert(margin >= 0.0); + assert(mMargin > 0.0); - return Vector3(direction.x < 0.0 ? -mExtent.x - margin : mExtent.x + margin, - direction.y < 0.0 ? -mExtent.y - margin : mExtent.y + margin, - direction.z < 0.0 ? -mExtent.z - margin : mExtent.z + margin); + return Vector3(direction.x < 0.0 ? -mExtent.x - mMargin : mExtent.x + mMargin, + direction.y < 0.0 ? -mExtent.y - mMargin : mExtent.y + mMargin, + direction.z < 0.0 ? -mExtent.z - mMargin : mExtent.z + mMargin); } // Return a local support point in a given direction without the objec margin diff --git a/src/collision/shapes/CollisionShape.cpp b/src/collision/shapes/CollisionShape.cpp index aca091d5..45a2efa7 100644 --- a/src/collision/shapes/CollisionShape.cpp +++ b/src/collision/shapes/CollisionShape.cpp @@ -30,14 +30,15 @@ using namespace reactphysics3d; // Constructor -CollisionShape::CollisionShape(CollisionShapeType type) - : mType(type), mNbSimilarCreatedShapes(0) { +CollisionShape::CollisionShape(CollisionShapeType type, decimal margin) + : mType(type), mNbSimilarCreatedShapes(0), mMargin(margin) { } // Private copy-constructor CollisionShape::CollisionShape(const CollisionShape& shape) - : mType(shape.mType), mNbSimilarCreatedShapes(shape.mNbSimilarCreatedShapes){ + : mType(shape.mType), mNbSimilarCreatedShapes(shape.mNbSimilarCreatedShapes), + mMargin(shape.mMargin) { } @@ -50,7 +51,7 @@ CollisionShape::~CollisionShape() { inline void CollisionShape::updateAABB(AABB& aabb, const Transform& transform) { // Get the local extents in x,y and z direction - Vector3 extents = getLocalExtents(OBJECT_MARGIN); + Vector3 extents = getLocalExtents(); // Rotate the local extents according to the orientation of the body Matrix3x3 worldAxis = transform.getOrientation().getMatrix().getAbsoluteMatrix(); diff --git a/src/collision/shapes/CollisionShape.h b/src/collision/shapes/CollisionShape.h index 4d865b19..9436277d 100644 --- a/src/collision/shapes/CollisionShape.h +++ b/src/collision/shapes/CollisionShape.h @@ -36,7 +36,7 @@ namespace reactphysics3d { /// Type of the collision shape -enum CollisionShapeType {BOX, SPHERE, CONE, CYLINDER}; +enum CollisionShapeType {BOX, SPHERE, CONE, CYLINDER, CAPSULE}; // Declarations class Body; @@ -57,6 +57,9 @@ class CollisionShape { /// Current number of similar created shapes uint mNbSimilarCreatedShapes; + + /// Margin used for the GJK collision detection algorithm + decimal mMargin; // -------------------- Methods -------------------- // @@ -71,7 +74,7 @@ class CollisionShape { // -------------------- Methods -------------------- // /// Constructor - CollisionShape(CollisionShapeType type); + CollisionShape(CollisionShapeType type, decimal margin); /// Destructor virtual ~CollisionShape(); @@ -85,6 +88,9 @@ class CollisionShape { /// Return the number of similar created shapes uint getNbSimilarCreatedShapes() const; + /// Return the current object margin + decimal getMargin() const; + /// Return the number of bytes used by the collision shape virtual size_t getSizeInBytes() const = 0; @@ -95,10 +101,7 @@ class CollisionShape { virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const=0; /// Return the local extents in x,y and z direction - virtual Vector3 getLocalExtents(decimal margin=0.0) const=0; - - /// Return the margin distance around the shape - virtual decimal getMargin() const=0; + virtual Vector3 getLocalExtents() const=0; /// Return the local inertia tensor of the collision shapes virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const=0; @@ -129,6 +132,11 @@ inline uint CollisionShape::getNbSimilarCreatedShapes() const { return mNbSimilarCreatedShapes; } +// Return the current object margin +inline decimal CollisionShape::getMargin() const { + return mMargin; +} + // Increment the number of similar allocated collision shapes inline void CollisionShape::incrementNbSimilarCreatedShapes() { mNbSimilarCreatedShapes++; @@ -150,6 +158,8 @@ inline bool CollisionShape::operator==(const CollisionShape& otherCollisionShape assert(typeid(*this) == typeid(otherCollisionShape)); + if (mMargin != otherCollisionShape.mMargin) return false; + // Check if the two shapes are equal return otherCollisionShape.isEqualTo(*this); } diff --git a/src/collision/shapes/ConeShape.cpp b/src/collision/shapes/ConeShape.cpp index b0be3dd3..1cdd0392 100644 --- a/src/collision/shapes/ConeShape.cpp +++ b/src/collision/shapes/ConeShape.cpp @@ -28,26 +28,14 @@ #include "../../configuration.h" #include "ConeShape.h" -#if defined(VISUAL_DEBUG) - #if defined(APPLE_OS) - #include - #include - #elif defined(WINDOWS_OS) - #include - #include - #elif defined(LINUX_OS) - #include - #include - #endif -#endif - using namespace reactphysics3d; // Constructor -ConeShape::ConeShape(decimal radius, decimal height) - : CollisionShape(CONE), mRadius(radius), mHalfHeight(height * decimal(0.5)) { - assert(mRadius > 0.0); - assert(mHalfHeight > 0.0); +ConeShape::ConeShape(decimal radius, decimal height, decimal margin) + : CollisionShape(CONE, margin), mRadius(radius), mHalfHeight(height * decimal(0.5)) { + assert(mRadius > decimal(0.0)); + assert(mHalfHeight > decimal(0.0)); + assert(margin > decimal(0.0)); // Compute the sine of the semi-angle at the apex point mSinTheta = mRadius / (sqrt(mRadius * mRadius + height * height)); @@ -76,7 +64,7 @@ inline Vector3 ConeShape::getLocalSupportPointWithMargin(const Vector3& directio if (direction.lengthSquare() > MACHINE_EPSILON * MACHINE_EPSILON) { unitVec = direction.getUnit(); } - supportPoint += unitVec * getMargin(); + supportPoint += unitVec * mMargin; return supportPoint; } @@ -104,15 +92,3 @@ inline Vector3 ConeShape::getLocalSupportPointWithoutMargin(const Vector3& direc return supportPoint; } - -#ifdef VISUAL_DEBUG -// Draw the cone (only for debuging purpose) -void ConeShape::draw() const { - - // Draw in red - glColor3f(1.0, 0.0, 0.0); - - // Draw the sphere - glutWireCone(mRadius, 2.0 * mHalfHeight, 50, 50); -} -#endif diff --git a/src/collision/shapes/ConeShape.h b/src/collision/shapes/ConeShape.h index d1721383..42b4356f 100644 --- a/src/collision/shapes/ConeShape.h +++ b/src/collision/shapes/ConeShape.h @@ -70,7 +70,7 @@ class ConeShape : public CollisionShape { // -------------------- Methods -------------------- // /// Constructor - ConeShape(decimal mRadius, decimal height); + ConeShape(decimal mRadius, decimal height, decimal margin = OBJECT_MARGIN); /// Destructor virtual ~ConeShape(); @@ -81,15 +81,9 @@ class ConeShape : public CollisionShape { /// Return the radius decimal getRadius() const; - /// Set the radius - void setRadius(decimal radius); - /// Return the height decimal getHeight() const; - /// Set the height - void setHeight(decimal height); - /// Return the number of bytes used by the collision shape virtual size_t getSizeInBytes() const; @@ -100,21 +94,13 @@ class ConeShape : public CollisionShape { virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const; /// Return the local extents in x,y and z direction - virtual Vector3 getLocalExtents(decimal margin=0.0) const; + virtual Vector3 getLocalExtents() const; /// Return the local inertia tensor of the collision shape virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; - /// Return the margin distance around the shape - virtual decimal getMargin() const; - /// Test equality between two cone shapes virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const; - -#ifdef VISUAL_DEBUG - /// Draw the sphere (only for testing purpose) - virtual void draw() const; -#endif }; // Allocate and return a copy of the object @@ -127,35 +113,19 @@ inline decimal ConeShape::getRadius() const { return mRadius; } -// Set the radius -inline void ConeShape::setRadius(decimal radius) { - mRadius = radius; - - // Update sine of the semi-angle at the apex point - mSinTheta = radius / (sqrt(radius * radius + 4 * mHalfHeight * mHalfHeight)); -} - // Return the height inline decimal ConeShape::getHeight() const { return decimal(2.0) * mHalfHeight; } -// Set the height -inline void ConeShape::setHeight(decimal height) { - mHalfHeight = height * decimal(0.5); - - // Update the sine of the semi-angle at the apex point - mSinTheta = mRadius / (sqrt(mRadius * mRadius + height * height)); -} - // Return the number of bytes used by the collision shape inline size_t ConeShape::getSizeInBytes() const { return sizeof(ConeShape); } // Return the local extents in x,y and z direction -inline Vector3 ConeShape::getLocalExtents(decimal margin) const { - return Vector3(mRadius + margin, mHalfHeight + margin, mRadius + margin); +inline Vector3 ConeShape::getLocalExtents() const { + return Vector3(mRadius + mMargin, mHalfHeight + mMargin, mRadius + mMargin); } // Return the local inertia tensor of the collision shape @@ -167,11 +137,6 @@ inline void ConeShape::computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass 0.0, 0.0, 0.0, diagXZ); } -// Return the margin distance around the shape -inline decimal ConeShape::getMargin() const { - return OBJECT_MARGIN; -} - // Test equality between two cone shapes inline bool ConeShape::isEqualTo(const CollisionShape& otherCollisionShape) const { const ConeShape& otherShape = dynamic_cast(otherCollisionShape); diff --git a/src/collision/shapes/CylinderShape.cpp b/src/collision/shapes/CylinderShape.cpp index 8bb87eaa..ac5a413b 100644 --- a/src/collision/shapes/CylinderShape.cpp +++ b/src/collision/shapes/CylinderShape.cpp @@ -27,25 +27,15 @@ #include "CylinderShape.h" #include "../../configuration.h" -#if defined(VISUAL_DEBUG) - #if defined(APPLE_OS) - #include - #include - #elif defined(WINDOWS_OS) - #include - #include - #elif defined(LINUX_OS) - #include - #include - #endif -#endif - using namespace reactphysics3d; // Constructor -CylinderShape::CylinderShape(decimal radius, decimal height) - : CollisionShape(CYLINDER), mRadius(radius), mHalfHeight(height/decimal(2.0)) { - +CylinderShape::CylinderShape(decimal radius, decimal height, decimal margin) + : CollisionShape(CYLINDER, margin), mRadius(radius), + mHalfHeight(height/decimal(2.0)) { + assert(radius > decimal(0.0)); + assert(height > decimal(0.0)); + assert(margin > decimal(0.0)); } // Private copy-constructor @@ -70,7 +60,7 @@ Vector3 CylinderShape::getLocalSupportPointWithMargin(const Vector3& direction) if (direction.lengthSquare() > MACHINE_EPSILON * MACHINE_EPSILON) { unitVec = direction.getUnit(); } - supportPoint += unitVec * getMargin(); + supportPoint += unitVec * mMargin; return supportPoint; } @@ -95,15 +85,3 @@ Vector3 CylinderShape::getLocalSupportPointWithoutMargin(const Vector3& directio return supportPoint; } - -#ifdef VISUAL_DEBUG -// Draw the cone (only for debuging purpose) -void CylinderShape::draw() const { - - // Draw in red - glColor3f(1.0, 0.0, 0.0); - - // Draw the sphere - glutWireSphere(mRadius, 50, 50); -} -#endif diff --git a/src/collision/shapes/CylinderShape.h b/src/collision/shapes/CylinderShape.h index 3d0b91d4..bfa2e41b 100644 --- a/src/collision/shapes/CylinderShape.h +++ b/src/collision/shapes/CylinderShape.h @@ -66,7 +66,7 @@ class CylinderShape : public CollisionShape { // -------------------- Methods -------------------- // /// Constructor - CylinderShape(decimal radius, decimal height); + CylinderShape(decimal radius, decimal height, decimal margin = OBJECT_MARGIN); /// Destructor virtual ~CylinderShape(); @@ -77,15 +77,9 @@ class CylinderShape : public CollisionShape { /// Return the radius decimal getRadius() const; - /// Set the radius - void setRadius(decimal mRadius); - /// Return the height decimal getHeight() const; - /// Set the height - void setHeight(decimal height); - /// Return the number of bytes used by the collision shape virtual size_t getSizeInBytes() const; @@ -96,21 +90,13 @@ class CylinderShape : public CollisionShape { virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const; /// Return the local extents in x,y and z direction - virtual Vector3 getLocalExtents(decimal margin=0.0) const; + virtual Vector3 getLocalExtents() const; /// Return the local inertia tensor of the collision shape virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; - /// Return the margin distance around the shape - virtual decimal getMargin() const; - /// Test equality between two cylinder shapes virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const; - -#ifdef VISUAL_DEBUG - /// Draw the sphere (only for testing purpose) - virtual void draw() const; -#endif }; /// Allocate and return a copy of the object @@ -123,29 +109,19 @@ inline decimal CylinderShape::getRadius() const { return mRadius; } -// Set the radius -inline void CylinderShape::setRadius(decimal radius) { - this->mRadius = radius; -} - // Return the height inline decimal CylinderShape::getHeight() const { return mHalfHeight * decimal(2.0); } -// Set the height -inline void CylinderShape::setHeight(decimal height) { - mHalfHeight = height * decimal(0.5); -} - // Return the number of bytes used by the collision shape inline size_t CylinderShape::getSizeInBytes() const { return sizeof(CylinderShape); } // Return the local extents in x,y and z direction -inline Vector3 CylinderShape::getLocalExtents(decimal margin) const { - return Vector3(mRadius + margin, mHalfHeight + margin, mRadius + margin); +inline Vector3 CylinderShape::getLocalExtents() const { + return Vector3(mRadius + mMargin, mHalfHeight + mMargin, mRadius + mMargin); } // Return the local inertia tensor of the cylinder @@ -157,11 +133,6 @@ inline void CylinderShape::computeLocalInertiaTensor(Matrix3x3& tensor, decimal 0.0, 0.0, diag); } -// Return the margin distance around the shape -inline decimal CylinderShape::getMargin() const { - return OBJECT_MARGIN; -} - // Test equality between two cylinder shapes inline bool CylinderShape::isEqualTo(const CollisionShape& otherCollisionShape) const { const CylinderShape& otherShape = dynamic_cast(otherCollisionShape); diff --git a/src/collision/shapes/SphereShape.cpp b/src/collision/shapes/SphereShape.cpp index 6ce64c6f..09d40f74 100644 --- a/src/collision/shapes/SphereShape.cpp +++ b/src/collision/shapes/SphereShape.cpp @@ -28,25 +28,12 @@ #include "../../configuration.h" #include -#if defined(VISUAL_DEBUG) - #if defined(APPLE_OS) - #include - #include - #elif defined(WINDOWS_OS) - #include - #include - #elif defined(LINUX_OS) - #include - #include - #endif -#endif - using namespace reactphysics3d; using namespace std; // Constructor -SphereShape::SphereShape(decimal radius): CollisionShape(SPHERE), mRadius(radius) { - +SphereShape::SphereShape(decimal radius) : CollisionShape(SPHERE, radius), mRadius(radius) { + assert(radius > decimal(0.0)); } // Private copy-constructor @@ -59,15 +46,3 @@ SphereShape::SphereShape(const SphereShape& shape) SphereShape::~SphereShape() { } - -#ifdef VISUAL_DEBUG -// Draw the sphere (only for testing purpose) -void SphereShape::draw() const { - - // Draw in red - glColor3f(1.0, 0.0, 0.0); - - // Draw the sphere - glutWireSphere(mRadius, 50, 50); -} -#endif diff --git a/src/collision/shapes/SphereShape.h b/src/collision/shapes/SphereShape.h index 0b7d4392..27481e04 100644 --- a/src/collision/shapes/SphereShape.h +++ b/src/collision/shapes/SphereShape.h @@ -71,9 +71,6 @@ class SphereShape : public CollisionShape { /// Return the radius of the sphere decimal getRadius() const; - /// Set the radius of the sphere - void setRadius(decimal radius); - /// Return the number of bytes used by the collision shape virtual size_t getSizeInBytes() const; @@ -84,24 +81,16 @@ class SphereShape : public CollisionShape { virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const; /// Return the local extents in x,y and z direction - virtual Vector3 getLocalExtents(decimal margin=0.0) const; + virtual Vector3 getLocalExtents() const; /// Return the local inertia tensor of the collision shape virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; - /// Return the margin distance around the shape - virtual decimal getMargin() const; - /// Update the AABB of a body using its collision shape virtual void updateAABB(AABB& aabb, const Transform& transform); /// Test equality between two sphere shapes virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const; - -#ifdef VISUAL_DEBUG - /// Draw the sphere (only for testing purpose) - virtual void draw() const; -#endif }; /// Allocate and return a copy of the object @@ -114,11 +103,6 @@ inline decimal SphereShape::getRadius() const { return mRadius; } -// Set the radius of the sphere -inline void SphereShape::setRadius(decimal radius) { - mRadius = radius; -} - // Return the number of bytes used by the collision shape inline size_t SphereShape::getSizeInBytes() const { return sizeof(SphereShape); @@ -127,18 +111,16 @@ inline size_t SphereShape::getSizeInBytes() const { // Return a local support point in a given direction with the object margin inline Vector3 SphereShape::getLocalSupportPointWithMargin(const Vector3& direction) const { - decimal margin = getMargin(); - // If the direction vector is not the zero vector if (direction.lengthSquare() >= MACHINE_EPSILON * MACHINE_EPSILON) { // Return the support point of the sphere in the given direction - return margin * direction.getUnit(); + return mMargin * direction.getUnit(); } // If the direction vector is the zero vector we return a point on the // boundary of the sphere - return Vector3(0, margin, 0); + return Vector3(0, mMargin, 0); } // Return a local support point in a given direction without the object margin @@ -150,8 +132,8 @@ inline Vector3 SphereShape::getLocalSupportPointWithoutMargin(const Vector3& dir // Return the local extents of the collision shape (half-width) in x,y and z local direction // This method is used to compute the AABB of the box -inline Vector3 SphereShape::getLocalExtents(decimal margin) const { - return Vector3(mRadius + margin, mRadius + margin, mRadius + margin); +inline Vector3 SphereShape::getLocalExtents() const { + return Vector3(mRadius, mRadius, mRadius); } // Return the local inertia tensor of the sphere @@ -162,16 +144,11 @@ inline void SphereShape::computeLocalInertiaTensor(Matrix3x3& tensor, decimal ma 0.0, 0.0, diag); } -// Return the margin distance around the shape -inline decimal SphereShape::getMargin() const { - return mRadius + OBJECT_MARGIN; -} - // Update the AABB of a body using its collision shape inline void SphereShape::updateAABB(AABB& aabb, const Transform& transform) { // Get the local extents in x,y and z direction - Vector3 extents = getLocalExtents(OBJECT_MARGIN); + Vector3 extents = getLocalExtents(); // Compute the minimum and maximum coordinates of the rotated extents Vector3 minCoordinates = transform.getPosition() - extents; diff --git a/src/configuration.h b/src/configuration.h index 4873ff96..3c756dab 100644 --- a/src/configuration.h +++ b/src/configuration.h @@ -92,7 +92,7 @@ const decimal DEFAULT_FRICTION_COEFFICIENT = decimal(0.3); /// True if the deactivation (sleeping) of inactive bodies is enabled const bool DEACTIVATION_ENABLED = true; -/// Object margin for collision detection in cm (For GJK-EPA Algorithm) +/// Object margin for collision detection in meters (for the GJK-EPA Algorithm) const decimal OBJECT_MARGIN = decimal(0.04); /// Distance threshold for two contact points for a valid persistent contact (in meters) diff --git a/src/constraint/ContactPoint.h b/src/constraint/ContactPoint.h index dc66b8b0..333fcf45 100644 --- a/src/constraint/ContactPoint.h +++ b/src/constraint/ContactPoint.h @@ -33,19 +33,6 @@ #include "../mathematics/mathematics.h" #include "../configuration.h" -#if defined(VISUAL_DEBUG) - #if defined(APPLE_OS) - #include - #include - #elif defined(WINDOWS_OS) - #include - #include - #elif defined(LINUX_OS) - #include - #include - #endif -#endif - /// ReactPhysics3D namespace namespace reactphysics3d { @@ -235,11 +222,6 @@ class ContactPoint : public Constraint { /// Solve the position constraint virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); - - #ifdef VISUAL_DEBUG - /// Draw the contact (for debugging) - void draw() const; - #endif }; // Return the normal vector of the contact @@ -352,13 +334,6 @@ inline size_t ContactPoint::getSizeInBytes() const { return sizeof(ContactPoint); } -#ifdef VISUAL_DEBUG -inline void ContactPoint::draw() const { - glColor3f(1.0, 0.0, 0.0); - glutSolidSphere(0.3, 20, 20); -} -#endif - } #endif From e081661d8c10dc1d0714fc930107b4ae20a69eac Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 10 Jul 2013 00:17:14 +0200 Subject: [PATCH 39/66] Add the Material class to store the material properties of a rigid body --- examples/common/opengl-framework/src/Mesh.cpp | 2 +- examples/fallingcubes/Scene.cpp | 13 +- examples/fallingcubes/Scene.h | 2 +- examples/joints/Scene.cpp | 35 +++--- src/body/RigidBody.cpp | 5 +- src/body/RigidBody.h | 59 +++------ src/configuration.h | 3 + src/engine/ContactSolver.h | 23 ++-- src/engine/Material.cpp | 46 +++++++ src/engine/Material.h | 119 ++++++++++++++++++ 10 files changed, 231 insertions(+), 76 deletions(-) create mode 100644 src/engine/Material.cpp create mode 100644 src/engine/Material.h diff --git a/examples/common/opengl-framework/src/Mesh.cpp b/examples/common/opengl-framework/src/Mesh.cpp index de15254c..6a993539 100644 --- a/examples/common/opengl-framework/src/Mesh.cpp +++ b/examples/common/opengl-framework/src/Mesh.cpp @@ -84,7 +84,7 @@ void Mesh::calculateNormals() { // Normalize the normal at each vertex for (uint i=0; i 0); + std::cout << "vertex n : " << i << std::endl; mNormals[i] = mNormals[i].normalize(); } } diff --git a/examples/fallingcubes/Scene.cpp b/examples/fallingcubes/Scene.cpp index fbc80ddb..360a0cb9 100644 --- a/examples/fallingcubes/Scene.cpp +++ b/examples/fallingcubes/Scene.cpp @@ -59,7 +59,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), float radius = 2.0f; // Create all the cubes of the scene - for (int i=0; igetRigidBody()->setIsMotionEnabled(true); - // Set the bouncing factor of the box - cube->getRigidBody()->setRestitution(0.4); + // Change the material properties of the rigid body + rp3d::Material& material = cube->getRigidBody()->getMaterial(); + material.setBounciness(0.4); // Add the box the list of box in the scene mBoxes.push_back(cube); @@ -87,8 +87,9 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), // The floor must be a non-moving rigid body mFloor->getRigidBody()->setIsMotionEnabled(false); - // Set the bouncing factor of the floor - mFloor->getRigidBody()->setRestitution(0.3); + // Change the material properties of the floor rigid body + rp3d::Material& material = mFloor->getRigidBody()->getMaterial(); + material.setBounciness(0.3); // Start the simulation startSimulation(); diff --git a/examples/fallingcubes/Scene.h b/examples/fallingcubes/Scene.h index ae4c14d2..45df3c4b 100644 --- a/examples/fallingcubes/Scene.h +++ b/examples/fallingcubes/Scene.h @@ -32,7 +32,7 @@ #include "Box.h" // Constants -const int NB_CUBES = 20; // Number of boxes in the scene +const int NB_SPHERES = 20; // Number of boxes in the scene const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters const float CUBE_MASS = 1.0f; // Box mass in kilograms diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index a66b8f30..44211770 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -209,8 +209,9 @@ void Scene::createBallAndSocketJoints() { if (i == 0) mBallAndSocketJointChainBoxes[i]->getRigidBody()->setIsMotionEnabled(false); else mBallAndSocketJointChainBoxes[i]->getRigidBody()->setIsMotionEnabled(true); - // Set the bouncing factor of the box - mBallAndSocketJointChainBoxes[i]->getRigidBody()->setRestitution(0.4); + // Change the material properties of the rigid body + rp3d::Material& material = mBallAndSocketJointChainBoxes[i]->getRigidBody()->getMaterial(); + material.setBounciness(0.4); positionBox.y -= boxDimension.y + 0.5; } @@ -248,8 +249,9 @@ void Scene::createSliderJoint() { // The fist box cannot move mSliderJointBottomBox->getRigidBody()->setIsMotionEnabled(false); - // Set the bouncing factor of the box - mSliderJointBottomBox->getRigidBody()->setRestitution(0.4); + // Change the material properties of the rigid body + rp3d::Material& material1 = mSliderJointBottomBox->getRigidBody()->getMaterial(); + material1.setBounciness(0.4); // --------------- Create the second box --------------- // @@ -263,8 +265,9 @@ void Scene::createSliderJoint() { // The second box is allowed to move mSliderJointTopBox->getRigidBody()->setIsMotionEnabled(true); - // Set the bouncing factor of the box - mSliderJointTopBox->getRigidBody()->setRestitution(0.4); + // Change the material properties of the rigid body + rp3d::Material& material2 = mSliderJointTopBox->getRigidBody()->getMaterial(); + material2.setBounciness(0.4); // --------------- Create the joint --------------- // @@ -301,8 +304,9 @@ void Scene::createPropellerHingeJoint() { // The fist box cannot move mPropellerBox->getRigidBody()->setIsMotionEnabled(true); - // Set the bouncing factor of the box - mPropellerBox->getRigidBody()->setRestitution(0.4); + // Change the material properties of the rigid body + rp3d::Material& material = mPropellerBox->getRigidBody()->getMaterial(); + material.setBounciness(0.4); // --------------- Create the Hinge joint --------------- // @@ -338,8 +342,9 @@ void Scene::createFixedJoints() { // The fist box cannot move mFixedJointBox1->getRigidBody()->setIsMotionEnabled(true); - // Set the bouncing factor of the box - mFixedJointBox1->getRigidBody()->setRestitution(0.4); + // Change the material properties of the rigid body + rp3d::Material& material1 = mFixedJointBox1->getRigidBody()->getMaterial(); + material1.setBounciness(0.4); // --------------- Create the second box --------------- // @@ -352,8 +357,9 @@ void Scene::createFixedJoints() { // The second box is allowed to move mFixedJointBox2->getRigidBody()->setIsMotionEnabled(true); - // Set the bouncing factor of the box - mFixedJointBox2->getRigidBody()->setRestitution(0.4); + // Change the material properties of the rigid body + rp3d::Material& material2 = mFixedJointBox2->getRigidBody()->getMaterial(); + material2.setBounciness(0.4); // --------------- Create the first fixed joint --------------- // @@ -390,6 +396,7 @@ void Scene::createFloor() { // The floor must be a non-moving rigid body mFloor->getRigidBody()->setIsMotionEnabled(false); - // Set the bouncing factor of the floor - mFloor->getRigidBody()->setRestitution(0.3); + // Change the material properties of the rigid body + rp3d::Material& material = mFloor->getRigidBody()->getMaterial(); + material.setBounciness(0.3); } diff --git a/src/body/RigidBody.cpp b/src/body/RigidBody.cpp index 5b2022bd..e18ac82a 100644 --- a/src/body/RigidBody.cpp +++ b/src/body/RigidBody.cpp @@ -35,10 +35,7 @@ using namespace reactphysics3d; CollisionShape *collisionShape, bodyindex id) : CollisionBody(transform, collisionShape, id), mInertiaTensorLocal(inertiaTensorLocal), mMass(mass), mInertiaTensorLocalInverse(inertiaTensorLocal.getInverse()), - mMassInverse(decimal(1.0) / mass), mFrictionCoefficient(DEFAULT_FRICTION_COEFFICIENT), - mIsGravityEnabled(true) { - - mRestitution = decimal(1.0); + mMassInverse(decimal(1.0) / mass), mIsGravityEnabled(true) { assert(collisionShape); } diff --git a/src/body/RigidBody.h b/src/body/RigidBody.h index 66a3f70c..4af18f37 100644 --- a/src/body/RigidBody.h +++ b/src/body/RigidBody.h @@ -29,6 +29,7 @@ // Libraries #include #include "CollisionBody.h" +#include "../engine/Material.h" #include "../mathematics/mathematics.h" /// Namespace reactphysics3d @@ -73,15 +74,12 @@ class RigidBody : public CollisionBody { /// Inverse of the mass of the body decimal mMassInverse; - /// Coefficient of restitution (between 0 and 1) where 1 is for a very bouncy body - decimal mRestitution; - - /// Friction coefficient - decimal mFrictionCoefficient; - /// True if the gravity needs to be applied to this rigid body bool mIsGravityEnabled; + /// Material properties of the rigid body + Material mMaterial; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -151,24 +149,18 @@ class RigidBody : public CollisionBody { /// Return the inverse of the inertia tensor in world coordinates. Matrix3x3 getInertiaTensorInverseWorld() const; - - /// Get the restitution coefficient - decimal getRestitution() const; - - /// Set the restitution coefficient - void setRestitution(decimal restitution); - - /// Get the friction coefficient - decimal getFrictionCoefficient() const; - - /// Set the friction coefficient - void setFrictionCoefficient(decimal frictionCoefficient); /// Return true if the gravity needs to be applied to this rigid body bool isGravityEnabled() const; /// Set the variable to know if the gravity is applied to this rigid body void enableGravity(bool isEnabled); + + /// Return a reference to the material properties of the rigid body + Material& getMaterial(); + + /// Set a new material for this rigid body + void setMaterial(const Material& material); }; // Method that return the mass of the body @@ -276,27 +268,6 @@ inline void RigidBody::setLinearVelocity(const Vector3& linearVelocity) { } } -// Get the restitution coeffficient of the rigid body -inline decimal RigidBody::getRestitution() const { - return mRestitution; -} - -// Set the restitution coefficient -inline void RigidBody::setRestitution(decimal restitution) { - assert(restitution >= 0.0 && restitution <= 1.0); - mRestitution = restitution; -} - -// Get the friction coefficient -inline decimal RigidBody::getFrictionCoefficient() const { - return mFrictionCoefficient; -} - -// Set the friction coefficient -inline void RigidBody::setFrictionCoefficient(decimal frictionCoefficient) { - mFrictionCoefficient = frictionCoefficient; -} - // Return true if the gravity needs to be applied to this rigid body inline bool RigidBody::isGravityEnabled() const { return mIsGravityEnabled; @@ -307,6 +278,16 @@ inline void RigidBody::enableGravity(bool isEnabled) { mIsGravityEnabled = isEnabled; } +// Return a reference to the material properties of the rigid body +inline Material& RigidBody::getMaterial() { + return mMaterial; +} + +// Set a new material for this rigid body +inline void RigidBody::setMaterial(const Material& material) { + mMaterial = material; +} + } #endif diff --git a/src/configuration.h b/src/configuration.h index 3c756dab..c762f14e 100644 --- a/src/configuration.h +++ b/src/configuration.h @@ -89,6 +89,9 @@ const decimal DEFAULT_TIMESTEP = decimal(1.0 / 60.0); /// Default friction coefficient for a rigid body const decimal DEFAULT_FRICTION_COEFFICIENT = decimal(0.3); +/// Default bounciness factor for a rigid body +const decimal DEFAULT_BOUNCINESS = decimal(0.5); + /// True if the deactivation (sleeping) of inactive bodies is enabled const bool DEACTIVATION_ENABLED = true; diff --git a/src/engine/ContactSolver.h b/src/engine/ContactSolver.h index d1a3971f..2c635fb0 100644 --- a/src/engine/ContactSolver.h +++ b/src/engine/ContactSolver.h @@ -367,12 +367,12 @@ class ContactSolver { const ContactManifoldSolver& manifold); /// Compute the collision restitution factor from the restitution factor of each body - decimal computeMixedRestitutionFactor(const RigidBody* body1, - const RigidBody* body2) const; + decimal computeMixedRestitutionFactor(RigidBody *body1, + RigidBody *body2) const; /// Compute the mixed friction coefficient from the friction coefficient of each body - decimal computeMixedFrictionCoefficient(const RigidBody* body1, - const RigidBody* body2)const; + decimal computeMixedFrictionCoefficient(RigidBody* body1, + RigidBody* body2) const; /// Compute the two unit orthogonal vectors "t1" and "t2" that span the tangential friction /// plane for a contact point. The two vectors have to be @@ -489,20 +489,21 @@ inline void ContactSolver::setIsSolveFrictionAtContactManifoldCenterActive(bool } // Compute the collision restitution factor from the restitution factor of each body -inline decimal ContactSolver::computeMixedRestitutionFactor(const RigidBody* body1, - const RigidBody* body2) const { - decimal restitution1 = body1->getRestitution(); - decimal restitution2 = body2->getRestitution(); +inline decimal ContactSolver::computeMixedRestitutionFactor(RigidBody* body1, + RigidBody* body2) const { + decimal restitution1 = body1->getMaterial().getBounciness(); + decimal restitution2 = body2->getMaterial().getBounciness(); // Return the largest restitution factor return (restitution1 > restitution2) ? restitution1 : restitution2; } // Compute the mixed friction coefficient from the friction coefficient of each body -inline decimal ContactSolver::computeMixedFrictionCoefficient(const RigidBody* body1, - const RigidBody* body2) const { +inline decimal ContactSolver::computeMixedFrictionCoefficient(RigidBody *body1, + RigidBody *body2) const { // Use the geometric mean to compute the mixed friction coefficient - return sqrt(body1->getFrictionCoefficient() * body2->getFrictionCoefficient()); + return sqrt(body1->getMaterial().getFrictionCoefficient() * + body2->getMaterial().getFrictionCoefficient()); } // Compute a penetration constraint impulse diff --git a/src/engine/Material.cpp b/src/engine/Material.cpp new file mode 100644 index 00000000..491a1184 --- /dev/null +++ b/src/engine/Material.cpp @@ -0,0 +1,46 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Material.h" + +using namespace reactphysics3d; + +// Constructor +Material::Material() + : mFrictionCoefficient(DEFAULT_FRICTION_COEFFICIENT), mBounciness(DEFAULT_BOUNCINESS) { + +} + +// Copy-constructor +Material::Material(const Material& material) + : mFrictionCoefficient(material.mFrictionCoefficient), mBounciness(material.mBounciness) { + +} + +// Destructor +Material::~Material() { + +} diff --git a/src/engine/Material.h b/src/engine/Material.h new file mode 100644 index 00000000..4ab4a9c7 --- /dev/null +++ b/src/engine/Material.h @@ -0,0 +1,119 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_MATERIAL_H +#define REACTPHYSICS3D_MATERIAL_H + +// Libraries +#include +#include "../configuration.h" + +namespace reactphysics3d { + +// Class Material +/** + * This class contains the material properties of a rigid body that will be use for + * the dynamics simulation like the friction coefficient or the bounciness of the rigid + * body. + */ +class Material { + + private : + + // -------------------- Attributes -------------------- // + + /// Friction coefficient (positive value) + decimal mFrictionCoefficient; + + /// Bounciness during collisions (between 0 and 1) where 1 is for a very bouncy body + decimal mBounciness; + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + Material(); + + /// Copy-constructor + Material(const Material& material); + + /// Destructor + ~Material(); + + /// Return the bounciness + decimal getBounciness() const; + + /// Set the bounciness + void setBounciness(decimal bounciness); + + /// Return the friction coefficient + decimal getFrictionCoefficient() const; + + /// Set the friction coefficient + void setFrictionCoefficient(decimal frictionCoefficient); + + /// Overloaded assignment operator + Material& operator=(const Material& material); +}; + +// Return the bounciness +inline decimal Material::getBounciness() const { + return mBounciness; +} + +// Set the bounciness +inline void Material::setBounciness(decimal bounciness) { + assert(bounciness >= decimal(0.0) && bounciness <= decimal(1.0)); + mBounciness = bounciness; +} + +// Return the friction coefficient +inline decimal Material::getFrictionCoefficient() const { + return mFrictionCoefficient; +} + +// Set the friction coefficient +inline void Material::setFrictionCoefficient(decimal frictionCoefficient) { + assert(frictionCoefficient >= decimal(0.0)); + mFrictionCoefficient = frictionCoefficient; +} + +// Overloaded assignment operator +inline Material& Material::operator=(const Material& material) { + + // Check for self-assignment + if (this != &material) { + mFrictionCoefficient = material.mFrictionCoefficient; + mBounciness = material.mBounciness; + } + + // Return this material + return *this; +} + +} + +#endif From 9d9142af30ab05fbbe02118102da245c22d442e9 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 10 Jul 2013 00:18:55 +0200 Subject: [PATCH 40/66] Add the CapsuleShape class for the capsule collision shape --- src/collision/shapes/CapsuleShape.cpp | 125 +++++++++++++++++++++++ src/collision/shapes/CapsuleShape.h | 140 ++++++++++++++++++++++++++ src/reactphysics3d.h | 2 + 3 files changed, 267 insertions(+) create mode 100644 src/collision/shapes/CapsuleShape.cpp create mode 100644 src/collision/shapes/CapsuleShape.h diff --git a/src/collision/shapes/CapsuleShape.cpp b/src/collision/shapes/CapsuleShape.cpp new file mode 100644 index 00000000..6397b693 --- /dev/null +++ b/src/collision/shapes/CapsuleShape.cpp @@ -0,0 +1,125 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "CapsuleShape.h" +#include "../../configuration.h" +#include + +using namespace reactphysics3d; + +// Constructor +CapsuleShape::CapsuleShape(decimal radius, decimal height) + : CollisionShape(CAPSULE, radius), mRadius(radius), mHalfHeight(height * decimal(0.5)) { + assert(radius > decimal(0.0)); + assert(height > decimal(0.0)); +} + +// Private copy-constructor +CapsuleShape::CapsuleShape(const CapsuleShape& shape) + : CollisionShape(shape), mRadius(shape.mRadius), mHalfHeight(shape.mHalfHeight) { + +} + +// Destructor +CapsuleShape::~CapsuleShape() { + +} + +// Return a local support point in a given direction with the object margin. +/// A capsule is the convex hull of two spheres S1 and S2. The support point in the direction "d" +/// of the convex hull of a set of convex objects is the support point "p" in the set of all +/// support points from all the convex objects with the maximum dot product with the direction "d". +/// Therefore, in this method, we compute the support points of both top and bottom spheres of +/// the capsule and return the point with the maximum dot product with the direction vector. Note +/// that the object margin is implicitly the radius and height of the capsule. +Vector3 CapsuleShape::getLocalSupportPointWithMargin(const Vector3& direction) const { + + // If the direction vector is not the zero vector + if (direction.lengthSquare() >= MACHINE_EPSILON * MACHINE_EPSILON) { + + Vector3 unitDirection = direction.getUnit(); + + // Support point top sphere + Vector3 centerTopSphere(0, mHalfHeight, 0); + Vector3 topSpherePoint = centerTopSphere + unitDirection * mRadius; + decimal dotProductTop = topSpherePoint.dot(direction); + + // Support point bottom sphere + Vector3 centerBottomSphere(0, -mHalfHeight, 0); + Vector3 bottomSpherePoint = centerBottomSphere + unitDirection * mRadius; + decimal dotProductBottom = bottomSpherePoint.dot(direction); + + // Return the point with the maximum dot product + if (dotProductTop > dotProductBottom) { + return topSpherePoint; + } + else { + return bottomSpherePoint; + } + } + + // If the direction vector is the zero vector we return a point on the + // boundary of the capsule + return Vector3(0, mRadius, 0); +} + +// Return a local support point in a given direction without the object margin. +Vector3 CapsuleShape::getLocalSupportPointWithoutMargin(const Vector3& direction) const { + + // If the dot product of the direction and the local Y axis (dotProduct = direction.y) + // is positive + if (direction.y > 0.0) { + + // Return the top sphere center point + return Vector3(0, mHalfHeight, 0); + } + else { + + // Return the bottom sphere center point + return Vector3(0, -mHalfHeight, 0); + } +} + +// Return the local inertia tensor of the capsule +void CapsuleShape::computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const { + + // The inertia tensor formula for a capsule can be found in : Game Engine Gems, Volume 1 + + decimal height = mHalfHeight + mHalfHeight; + decimal radiusSquare = mRadius * mRadius; + decimal heightSquare = height * height; + decimal radiusSquareDouble = radiusSquare + radiusSquare; + decimal factor1 = decimal(2.0) * mRadius / (decimal(4.0) * mRadius + decimal(3.0) * height); + decimal factor2 = decimal(3.0) * height / (decimal(4.0) * mRadius + decimal(3.0) * height); + decimal sum1 = decimal(0.4) * radiusSquareDouble; + decimal sum2 = decimal(0.75) * height * mRadius + decimal(0.5) * heightSquare; + decimal sum3 = decimal(0.25) * radiusSquare + decimal(1.0 / 12.0) * heightSquare; + decimal IxxAndzz = factor1 * mass * (sum1 + sum2) + factor2 * mass * sum3; + decimal Iyy = factor1 * mass * sum1 + factor2 * mass * decimal(0.25) * radiusSquareDouble; + tensor.setAllValues(IxxAndzz, 0.0, 0.0, + 0.0, Iyy, 0.0, + 0.0, 0.0, IxxAndzz); +} diff --git a/src/collision/shapes/CapsuleShape.h b/src/collision/shapes/CapsuleShape.h new file mode 100644 index 00000000..dc7592e1 --- /dev/null +++ b/src/collision/shapes/CapsuleShape.h @@ -0,0 +1,140 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_CAPSULE_SHAPE_H +#define REACTPHYSICS3D_CAPSULE_SHAPE_H + +// Libraries +#include "CollisionShape.h" +#include "../../mathematics/mathematics.h" + +// ReactPhysics3D namespace +namespace reactphysics3d { + +// Class CapsuleShape +/** + * This class represents a capsule collision shape that is defined around the Y axis. + * A capsule shape can be seen as the convex hull of two spheres. + * The capsule shape is defined by its radius (radius of the two spheres of the capsule) + * and its height (distance between the centers of the two spheres). This collision shape + * does not have an explicit object margin distance. The margin is implicitly the radius + * and height of the shape. Therefore, no need to specify an object margin for a + * capsule shape. + */ +class CapsuleShape : public CollisionShape { + + private : + + // -------------------- Attributes -------------------- // + + /// Radius of the two spheres of the capsule + decimal mRadius; + + /// Half height of the capsule (height = distance between the centers of the two spheres) + decimal mHalfHeight; + + // -------------------- Methods -------------------- // + + /// Private copy-constructor + CapsuleShape(const CapsuleShape& shape); + + /// Private assignment operator + CapsuleShape& operator=(const CapsuleShape& shape); + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + CapsuleShape(decimal radius, decimal height); + + /// Destructor + virtual ~CapsuleShape(); + + /// Allocate and return a copy of the object + virtual CapsuleShape* clone(void* allocatedMemory) const; + + /// Return the radius of the capsule + decimal getRadius() const; + + /// Return the height of the capsule + decimal getHeight() const; + + /// Return the number of bytes used by the collision shape + virtual size_t getSizeInBytes() const; + + /// Return a local support point in a given direction with the object margin. + virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const; + + /// Return a local support point in a given direction without the object margin + virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const; + + /// Return the local extents in x,y and z direction + virtual Vector3 getLocalExtents() const; + + /// Return the local inertia tensor of the collision shape + virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; + + /// Test equality between two capsule shapes + virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const; +}; + +/// Allocate and return a copy of the object +inline CapsuleShape* CapsuleShape::clone(void* allocatedMemory) const { + return new (allocatedMemory) CapsuleShape(*this); +} + +// Get the radius of the capsule +inline decimal CapsuleShape::getRadius() const { + return mRadius; +} + +// Return the height of the capsule +inline decimal CapsuleShape::getHeight() const { + return mHalfHeight + mHalfHeight; +} + +// Return the number of bytes used by the collision shape +inline size_t CapsuleShape::getSizeInBytes() const { + return sizeof(CapsuleShape); +} + +// Return the local extents of the collision shape (half-width) in x,y and z local direction +// This method is used to compute the AABB of the box +inline Vector3 CapsuleShape::getLocalExtents() const { + return Vector3(mRadius, + mHalfHeight + mRadius, + mRadius); +} + +// Test equality between two capsule shapes +inline bool CapsuleShape::isEqualTo(const CollisionShape& otherCollisionShape) const { + const CapsuleShape& otherShape = dynamic_cast(otherCollisionShape); + return (mRadius == otherShape.mRadius && mHalfHeight == otherShape.mHalfHeight); +} + +} + +#endif diff --git a/src/reactphysics3d.h b/src/reactphysics3d.h index e24c2b82..2bf3885c 100644 --- a/src/reactphysics3d.h +++ b/src/reactphysics3d.h @@ -41,11 +41,13 @@ #include "body/RigidBody.h" #include "engine/DynamicsWorld.h" #include "engine/CollisionWorld.h" +#include "engine/Material.h" #include "collision/shapes/CollisionShape.h" #include "collision/shapes/BoxShape.h" #include "collision/shapes/SphereShape.h" #include "collision/shapes/ConeShape.h" #include "collision/shapes/CylinderShape.h" +#include "collision/shapes/CapsuleShape.h" #include "collision/shapes/AABB.h" #include "constraint/BallAndSocketJoint.h" #include "constraint/SliderJoint.h" From 25b6b015b8c6e388ab31d134d64262dfcd5880e6 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 10 Jul 2013 00:21:31 +0200 Subject: [PATCH 41/66] Add some private copy-constructors and private assignment operators --- src/constraint/BallAndSocketJoint.h | 8 ++++++++ src/constraint/FixedJoint.h | 8 ++++++++ src/constraint/HingeJoint.h | 6 ++++++ src/constraint/SliderJoint.h | 6 ++++++ src/engine/Impulse.h | 32 ++++++++++++++++++++++++----- 5 files changed, 55 insertions(+), 5 deletions(-) diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index 6cd47fa6..3d155afd 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -96,6 +96,14 @@ class BallAndSocketJoint : public Constraint { /// Accumulated impulse Vector3 mImpulse; + // -------------------- Methods -------------------- // + + /// Private copy-constructor + BallAndSocketJoint(const BallAndSocketJoint& constraint); + + /// Private assignment operator + BallAndSocketJoint& operator=(const BallAndSocketJoint& constraint); + public : // -------------------- Methods -------------------- // diff --git a/src/constraint/FixedJoint.h b/src/constraint/FixedJoint.h index 7f4f987f..7ce30912 100644 --- a/src/constraint/FixedJoint.h +++ b/src/constraint/FixedJoint.h @@ -108,6 +108,14 @@ class FixedJoint : public Constraint { /// Inverse of the initial orientation difference between the two bodies Quaternion mInitOrientationDifferenceInv; + // -------------------- Methods -------------------- // + + /// Private copy-constructor + FixedJoint(const FixedJoint& constraint); + + /// Private assignment operator + FixedJoint& operator=(const FixedJoint& constraint); + public : // -------------------- Methods -------------------- // diff --git a/src/constraint/HingeJoint.h b/src/constraint/HingeJoint.h index 33962f0c..782cf834 100644 --- a/src/constraint/HingeJoint.h +++ b/src/constraint/HingeJoint.h @@ -223,6 +223,12 @@ class HingeJoint : public Constraint { // -------------------- Methods -------------------- // + /// Private copy-constructor + HingeJoint(const HingeJoint& constraint); + + /// Private assignment operator + HingeJoint& operator=(const HingeJoint& constraint); + /// Reset the limits void resetLimits(); diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h index 6515ffff..454f3385 100644 --- a/src/constraint/SliderJoint.h +++ b/src/constraint/SliderJoint.h @@ -236,6 +236,12 @@ class SliderJoint : public Constraint { // -------------------- Methods -------------------- // + /// Private copy-constructor + SliderJoint(const SliderJoint& constraint); + + /// Private assignment operator + SliderJoint& operator=(const SliderJoint& constraint); + /// Reset the limits void resetLimits(); diff --git a/src/engine/Impulse.h b/src/engine/Impulse.h index 5ed9e228..849d90ed 100644 --- a/src/engine/Impulse.h +++ b/src/engine/Impulse.h @@ -37,8 +37,17 @@ namespace reactphysics3d { */ struct Impulse { + private: + + // -------------------- Methods -------------------- // + + /// Private assignment operator + Impulse& operator=(const Impulse& impulse); + public: + // -------------------- Attributes -------------------- // + /// Linear impulse applied to the first body const Vector3 linearImpulseBody1; @@ -51,12 +60,25 @@ struct Impulse { /// Angular impulse applied to the second body const Vector3 angularImpulseBody2; - /// Constructor - Impulse(const Vector3& linearImpulseBody1, const Vector3& angularImpulseBody1, - const Vector3& linearImpulseBody2, const Vector3& angularImpulseBody2) - : linearImpulseBody1(linearImpulseBody1), angularImpulseBody1(angularImpulseBody1), - linearImpulseBody2(linearImpulseBody2), angularImpulseBody2(angularImpulseBody2) { + // -------------------- Methods -------------------- // + /// Constructor + Impulse(const Vector3& initLinearImpulseBody1, const Vector3& initAngularImpulseBody1, + const Vector3& initLinearImpulseBody2, const Vector3& initAngularImpulseBody2) + : linearImpulseBody1(initLinearImpulseBody1), + angularImpulseBody1(initAngularImpulseBody1), + linearImpulseBody2(initLinearImpulseBody2), + angularImpulseBody2(initAngularImpulseBody2) { + + } + + /// Copy-constructor + Impulse(const Impulse& impulse) + : linearImpulseBody1(impulse.linearImpulseBody1), + angularImpulseBody1(impulse.angularImpulseBody1), + linearImpulseBody2(impulse.linearImpulseBody2), + angularImpulseBody2(impulse.angularImpulseBody2) { +; } }; From 88504bbb4450709279ff215d390dd3935191a05f Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 10 Jul 2013 00:22:40 +0200 Subject: [PATCH 42/66] Add some description comments in the collision shapes --- src/collision/shapes/BoxShape.cpp | 1 - src/collision/shapes/BoxShape.h | 9 ++++++++- src/collision/shapes/ConeShape.h | 8 +++++++- src/collision/shapes/CylinderShape.h | 9 ++++++++- src/collision/shapes/SphereShape.cpp | 1 - src/collision/shapes/SphereShape.h | 5 ++++- 6 files changed, 27 insertions(+), 6 deletions(-) diff --git a/src/collision/shapes/BoxShape.cpp b/src/collision/shapes/BoxShape.cpp index 221f5037..1914f87f 100644 --- a/src/collision/shapes/BoxShape.cpp +++ b/src/collision/shapes/BoxShape.cpp @@ -30,7 +30,6 @@ #include using namespace reactphysics3d; -using namespace std; // Constructor BoxShape::BoxShape(const Vector3& extent, decimal margin) diff --git a/src/collision/shapes/BoxShape.h b/src/collision/shapes/BoxShape.h index 66afa348..6ebb1a81 100644 --- a/src/collision/shapes/BoxShape.h +++ b/src/collision/shapes/BoxShape.h @@ -40,7 +40,14 @@ namespace reactphysics3d { * This class represents a 3D box shape. Those axis are unit length. * The three extents are half-widths of the box along the three * axis x, y, z local axis. The "transform" of the corresponding - * rigid body gives an orientation and a position to the box. + * rigid body will give an orientation and a position to the box. This + * collision shape uses an extra margin distance around it for collision + * detection purpose. The default margin is 4cm (if your units are meters, + * which is recommended). In case, you want to simulate small objects + * (smaller than the margin distance), you might want to reduce the margin by + * specifying your own margin distance using the "margin" parameter in the + * constructor of the box shape. Otherwise, it is recommended to use the + * default margin distance by not using the "margin" parameter in the constructor. */ class BoxShape : public CollisionShape { diff --git a/src/collision/shapes/ConeShape.h b/src/collision/shapes/ConeShape.h index 42b4356f..5ac83d06 100644 --- a/src/collision/shapes/ConeShape.h +++ b/src/collision/shapes/ConeShape.h @@ -40,7 +40,13 @@ namespace reactphysics3d { * by its height and by the radius of its base. The center of the * cone is at the half of the height. The "transform" of the * corresponding rigid body gives an orientation and a position - * to the cone. + * to the cone. This collision shape uses an extra margin distance around + * it for collision detection purpose. The default margin is 4cm (if your + * units are meters, which is recommended). In case, you want to simulate small + * objects (smaller than the margin distance), you might want to reduce the margin + * by specifying your own margin distance using the "margin" parameter in the + * constructor of the cone shape. Otherwise, it is recommended to use the + * default margin distance by not using the "margin" parameter in the constructor. */ class ConeShape : public CollisionShape { diff --git a/src/collision/shapes/CylinderShape.h b/src/collision/shapes/CylinderShape.h index bfa2e41b..8cdcfc3f 100644 --- a/src/collision/shapes/CylinderShape.h +++ b/src/collision/shapes/CylinderShape.h @@ -40,6 +40,13 @@ namespace reactphysics3d { * and centered at the origin. The cylinder is defined by its height * and the radius of its base. The "transform" of the corresponding * rigid body gives an orientation and a position to the cylinder. + * This collision shape uses an extra margin distance around it for collision + * detection purpose. The default margin is 4cm (if your units are meters, + * which is recommended). In case, you want to simulate small objects + * (smaller than the margin distance), you might want to reduce the margin by + * specifying your own margin distance using the "margin" parameter in the + * constructor of the cylinder shape. Otherwise, it is recommended to use the + * default margin distance by not using the "margin" parameter in the constructor. */ class CylinderShape : public CollisionShape { @@ -111,7 +118,7 @@ inline decimal CylinderShape::getRadius() const { // Return the height inline decimal CylinderShape::getHeight() const { - return mHalfHeight * decimal(2.0); + return mHalfHeight + mHalfHeight; } // Return the number of bytes used by the collision shape diff --git a/src/collision/shapes/SphereShape.cpp b/src/collision/shapes/SphereShape.cpp index 09d40f74..c6b3b3e3 100644 --- a/src/collision/shapes/SphereShape.cpp +++ b/src/collision/shapes/SphereShape.cpp @@ -29,7 +29,6 @@ #include using namespace reactphysics3d; -using namespace std; // Constructor SphereShape::SphereShape(decimal radius) : CollisionShape(SPHERE, radius), mRadius(radius) { diff --git a/src/collision/shapes/SphereShape.h b/src/collision/shapes/SphereShape.h index 27481e04..1d46d888 100644 --- a/src/collision/shapes/SphereShape.h +++ b/src/collision/shapes/SphereShape.h @@ -36,7 +36,10 @@ namespace reactphysics3d { // Class SphereShape /** * This class represents a sphere collision shape that is centered - * at the origin and defined by its radius. + * at the origin and defined by its radius. This collision shape does not + * have an explicit object margin distance. The margin is implicitly the + * radius of the sphere. Therefore, no need to specify an object margin + * for a sphere shape. */ class SphereShape : public CollisionShape { From e4d283186f95612b3d5c4ebc09c973fd2eab8395 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 10 Jul 2013 00:23:15 +0200 Subject: [PATCH 43/66] Modify some examples --- examples/common/Capsule.cpp | 136 ++++ examples/common/Capsule.h | 81 +++ examples/common/meshes/capsule.obj | 1060 ++++++++++++++++++++++++++++ 3 files changed, 1277 insertions(+) create mode 100644 examples/common/Capsule.cpp create mode 100644 examples/common/Capsule.h create mode 100644 examples/common/meshes/capsule.obj diff --git a/examples/common/Capsule.cpp b/examples/common/Capsule.cpp new file mode 100644 index 00000000..07216c5e --- /dev/null +++ b/examples/common/Capsule.cpp @@ -0,0 +1,136 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Capsule.h" + + +// Constructor +Capsule::Capsule(float radius, float height, const openglframework::Vector3 &position, + float mass, reactphysics3d::DynamicsWorld* dynamicsWorld) + : openglframework::Mesh(), mRadius(radius), mHeight(height) { + + // Load the mesh from a file + openglframework::MeshReaderWriter::loadMeshFromFile("meshes/capsule.obj", *this); + + // Calculate the normals of the mesh + calculateNormals(); + + // Compute the scaling matrix + mScalingMatrix = openglframework::Matrix4(mRadius, 0, 0, 0, + 0, (mHeight + 2.0 * mRadius) / 3.0, 0,0, + 0, 0, mRadius, 0, + 0, 0, 0, 1); + + // Initialize the position where the sphere will be rendered + translateWorld(position); + + // Create the collision shape for the rigid body (sphere shape) + // ReactPhysics3D will clone this object to create an internal one. Therefore, + // it is OK if this object is destroy right after calling Dynamics::createRigidBody() + const rp3d::CapsuleShape collisionShape(mRadius, mHeight); + + // Compute the inertia tensor of the body using its collision shape + rp3d::Matrix3x3 inertiaTensor; + collisionShape.computeLocalInertiaTensor(inertiaTensor, mass); + + // Initial position and orientation of the rigid body + rp3d::Vector3 initPosition(position.x, position.y, position.z); + rp3d::Quaternion initOrientation = rp3d::Quaternion::identity(); + rp3d::Transform transform(initPosition, initOrientation); + + // Create a rigid body corresponding to the sphere in the dynamics world + mRigidBody = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, collisionShape); +} + +// Destructor +Capsule::~Capsule() { + + // Destroy the mesh + destroy(); +} + +// Render the sphere at the correct position and with the correct orientation +void Capsule::render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix) { + + // Bind the shader + shader.bind(); + + // Set the model to camera matrix + const openglframework::Matrix4 localToCameraMatrix = worldToCameraMatrix * mTransformMatrix; + shader.setMatrix4x4Uniform("localToCameraMatrix", localToCameraMatrix); + + // Set the normal matrix (inverse transpose of the 3x3 upper-left sub matrix of the + // model-view matrix) + const openglframework::Matrix3 normalMatrix = + localToCameraMatrix.getUpperLeft3x3Matrix().getInverse().getTranspose(); + shader.setMatrix3x3Uniform("normalMatrix", normalMatrix); + + glEnableClientState(GL_VERTEX_ARRAY); + glEnableClientState(GL_NORMAL_ARRAY); + if (hasTexture()) { + glEnableClientState(GL_TEXTURE_COORD_ARRAY); + } + + glVertexPointer(3, GL_FLOAT, 0, getVerticesPointer()); + glNormalPointer(GL_FLOAT, 0, getNormalsPointer()); + if(hasTexture()) { + glTexCoordPointer(2, GL_FLOAT, 0, getUVTextureCoordinatesPointer()); + } + + // For each part of the mesh + for (unsigned int i=0; igetInterpolatedTransform(); + + // Compute the transform used for rendering the sphere + float matrix[16]; + transform.getOpenGLMatrix(matrix); + openglframework::Matrix4 newMatrix(matrix[0], matrix[4], matrix[8], matrix[12], + matrix[1], matrix[5], matrix[9], matrix[13], + matrix[2], matrix[6], matrix[10], matrix[14], + matrix[3], matrix[7], matrix[11], matrix[15]); + + // Apply the scaling matrix to have the correct sphere dimensions + mTransformMatrix = newMatrix * mScalingMatrix; +} diff --git a/examples/common/Capsule.h b/examples/common/Capsule.h new file mode 100644 index 00000000..e99f9fa4 --- /dev/null +++ b/examples/common/Capsule.h @@ -0,0 +1,81 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 CAPSULE_H +#define CAPSULE_H + +// Libraries +#include "openglframework.h" +#include "reactphysics3d.h" + +// Class Sphere +class Capsule : public openglframework::Mesh { + + private : + + // -------------------- Attributes -------------------- // + + /// Radius of the capsule + float mRadius; + + /// Height of the capsule + float mHeight; + + /// Rigid body used to simulate the dynamics of the sphere + rp3d::RigidBody* mRigidBody; + + /// Scaling matrix (applied to a sphere to obtain the correct sphere dimensions) + openglframework::Matrix4 mScalingMatrix; + + // -------------------- Methods -------------------- // + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + Capsule(float radius, float height, const openglframework::Vector3& position, + float mass, rp3d::DynamicsWorld* dynamicsWorld); + + /// Destructor + ~Capsule(); + + /// Return a pointer to the rigid body of the sphere + rp3d::RigidBody* getRigidBody(); + + /// Update the transform matrix of the sphere + void updateTransform(); + + /// Render the sphere at the correct position and with the correct orientation + void render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix); +}; + +// Return a pointer to the rigid body of the sphere +inline rp3d::RigidBody* Capsule::getRigidBody() { + return mRigidBody; +} + +#endif diff --git a/examples/common/meshes/capsule.obj b/examples/common/meshes/capsule.obj new file mode 100644 index 00000000..8479fb35 --- /dev/null +++ b/examples/common/meshes/capsule.obj @@ -0,0 +1,1060 @@ +# Blender v2.66 (sub 0) OBJ File: '' +v -0.195090 1.481026 0.000000 +v -0.382683 1.424120 0.000000 +v -0.555570 1.331710 0.000000 +v -0.707107 1.207348 0.000000 +v -0.831470 1.055811 0.000000 +v -0.923880 0.882924 0.000000 +v -0.980785 0.695331 0.000000 +v -1.000000 0.500241 0.000000 +v -0.191342 1.481026 -0.038060 +v -0.375330 1.424120 -0.074658 +v -0.544895 1.331710 -0.108386 +v -0.693520 1.207348 -0.137950 +v -0.815493 1.055811 -0.162212 +v -0.906127 0.882924 -0.180240 +v -0.961940 0.695331 -0.191342 +v -0.980785 0.500241 -0.195090 +v -0.180240 1.481026 -0.074658 +v -0.353553 1.424120 -0.146447 +v -0.513280 1.331710 -0.212608 +v -0.653281 1.207348 -0.270598 +v -0.768178 1.055811 -0.318190 +v -0.853553 0.882924 -0.353553 +v -0.906127 0.695331 -0.375330 +v -0.923880 0.500241 -0.382684 +v -0.162212 1.481026 -0.108387 +v -0.318190 1.424120 -0.212608 +v -0.461940 1.331710 -0.308658 +v -0.587938 1.207348 -0.392848 +v -0.691342 1.055811 -0.461940 +v -0.768178 0.882924 -0.513280 +v -0.815493 0.695331 -0.544895 +v -0.831470 0.500241 -0.555570 +v -0.137950 1.481026 -0.137950 +v -0.270598 1.424120 -0.270598 +v -0.392847 1.331710 -0.392848 +v -0.500000 1.207348 -0.500000 +v -0.587938 1.055811 -0.587938 +v -0.653281 0.882924 -0.653282 +v -0.693520 0.695331 -0.693520 +v -0.707107 0.500241 -0.707107 +v -0.108386 1.481026 -0.162212 +v -0.212607 1.424120 -0.318190 +v -0.308658 1.331710 -0.461940 +v -0.392847 1.207348 -0.587938 +v -0.461939 1.055811 -0.691342 +v -0.513280 0.882924 -0.768178 +v -0.544895 0.695331 -0.815493 +v -0.555570 0.500241 -0.831470 +v -0.074658 1.481026 -0.180240 +v -0.146446 1.424120 -0.353554 +v -0.212607 1.331710 -0.513280 +v -0.270598 1.207348 -0.653282 +v -0.318189 1.055811 -0.768178 +v -0.353553 0.882924 -0.853554 +v -0.375330 0.695331 -0.906128 +v -0.382683 0.500241 -0.923880 +v -0.038060 1.481026 -0.191342 +v -0.074658 1.424120 -0.375331 +v -0.108386 1.331710 -0.544895 +v -0.137949 1.207348 -0.693520 +v -0.162211 1.055811 -0.815493 +v -0.180240 0.882924 -0.906128 +v -0.191341 0.695331 -0.961940 +v -0.195090 0.500241 -0.980785 +v 0.000000 1.481026 -0.195091 +v 0.000000 1.424120 -0.382684 +v 0.000000 1.331710 -0.555570 +v 0.000000 1.207348 -0.707107 +v 0.000000 1.055811 -0.831470 +v 0.000000 0.882924 -0.923880 +v 0.000000 0.695331 -0.980785 +v 0.000000 0.500241 -1.000000 +v 0.038061 1.481026 -0.191342 +v 0.074658 1.424120 -0.375330 +v 0.108387 1.331710 -0.544895 +v 0.137950 1.207348 -0.693520 +v 0.162212 1.055811 -0.815493 +v 0.180240 0.882924 -0.906128 +v 0.191342 0.695331 -0.961940 +v 0.195091 0.500241 -0.980785 +v 0.074658 1.481026 -0.180240 +v 0.146447 1.424120 -0.353554 +v 0.212608 1.331710 -0.513280 +v 0.270599 1.207348 -0.653282 +v 0.318190 1.055811 -0.768178 +v 0.353554 0.882924 -0.853553 +v 0.375331 0.695331 -0.906127 +v 0.382684 0.500241 -0.923880 +v 0.108387 1.481026 -0.162212 +v 0.212608 1.424120 -0.318190 +v 0.308659 1.331710 -0.461940 +v 0.392848 1.207348 -0.587938 +v 0.461940 1.055811 -0.691342 +v 0.513280 0.882924 -0.768178 +v 0.544895 0.695331 -0.815493 +v 0.555571 0.500241 -0.831470 +v 0.137950 1.481026 -0.137950 +v 0.270599 1.424120 -0.270598 +v 0.392848 1.331710 -0.392848 +v 0.500000 1.207348 -0.500000 +v 0.587938 1.055811 -0.587938 +v 0.653282 0.882924 -0.653281 +v 0.693520 0.695331 -0.693520 +v 0.707107 0.500241 -0.707107 +v 0.162212 1.481026 -0.108386 +v 0.318190 1.424120 -0.212608 +v 0.461940 1.331710 -0.308658 +v 0.587938 1.207348 -0.392847 +v 0.691342 1.055811 -0.461940 +v 0.768178 0.882924 -0.513280 +v 0.815493 0.695331 -0.544895 +v 0.831470 0.500241 -0.555570 +v 0.180240 1.481026 -0.074658 +v 0.353554 1.424120 -0.146447 +v 0.513280 1.331710 -0.212608 +v 0.653282 1.207348 -0.270598 +v 0.768178 1.055811 -0.318190 +v 0.853554 0.882924 -0.353553 +v 0.906128 0.695331 -0.375330 +v 0.923880 0.500241 -0.382683 +v 0.191342 1.481026 -0.038060 +v 0.375331 1.424120 -0.074658 +v 0.544896 1.331710 -0.108386 +v 0.693520 1.207348 -0.137950 +v 0.815493 1.055811 -0.162212 +v 0.906128 0.882924 -0.180240 +v 0.961940 0.695331 -0.191342 +v 0.980785 0.500241 -0.195090 +v 0.195091 1.481026 0.000000 +v 0.382684 1.424120 0.000000 +v 0.555571 1.331710 0.000000 +v 0.707107 1.207348 0.000000 +v 0.831470 1.055811 0.000000 +v 0.923880 0.882924 0.000000 +v 0.980785 0.695331 0.000000 +v 1.000000 0.500241 0.000000 +v 0.191342 1.481026 0.038060 +v 0.375331 1.424120 0.074658 +v 0.544896 1.331710 0.108386 +v 0.693520 1.207348 0.137950 +v 0.815493 1.055811 0.162212 +v 0.906128 0.882924 0.180240 +v 0.961940 0.695331 0.191342 +v 0.980785 0.500241 0.195090 +v 0.180240 1.481026 0.074658 +v 0.353554 1.424120 0.146447 +v 0.513280 1.331710 0.212608 +v 0.653282 1.207348 0.270598 +v 0.768178 1.055811 0.318190 +v 0.853554 0.882924 0.353554 +v 0.906128 0.695331 0.375330 +v 0.923880 0.500241 0.382683 +v 0.162212 1.481026 0.108387 +v 0.318190 1.424120 0.212608 +v 0.461940 1.331710 0.308658 +v 0.587938 1.207348 0.392848 +v 0.691342 1.055811 0.461940 +v 0.768178 0.882924 0.513280 +v 0.815493 0.695331 0.544895 +v 0.831470 0.500241 0.555570 +v 0.137950 1.481026 0.137950 +v 0.270598 1.424120 0.270598 +v 0.392848 1.331710 0.392848 +v 0.500000 1.207348 0.500000 +v 0.587938 1.055811 0.587938 +v 0.653282 0.882924 0.653282 +v 0.693520 0.695331 0.693520 +v 0.707107 0.500241 0.707107 +v 0.108387 1.481026 0.162212 +v 0.212608 1.424120 0.318190 +v 0.308659 1.331710 0.461940 +v 0.392848 1.207348 0.587938 +v 0.461940 1.055811 0.691342 +v 0.513280 0.882924 0.768178 +v 0.544895 0.695331 0.815493 +v 0.555570 0.500241 0.831469 +v 0.074658 1.481026 0.180240 +v 0.146447 1.424120 0.353554 +v 0.212608 1.331710 0.513280 +v 0.270598 1.207348 0.653281 +v 0.318190 1.055811 0.768178 +v 0.353554 0.882924 0.853553 +v 0.375330 0.695331 0.906127 +v 0.382683 0.500241 0.923879 +v 0.038061 1.481026 0.191342 +v 0.074658 1.424120 0.375330 +v 0.108387 1.331710 0.544895 +v 0.137950 1.207348 0.693520 +v 0.162212 1.055811 0.815493 +v 0.180240 0.882924 0.906127 +v 0.191342 0.695331 0.961940 +v 0.195090 0.500241 0.980785 +v 0.000000 1.481026 0.195090 +v 0.000000 1.424120 0.382684 +v 0.000000 1.331710 0.555570 +v 0.000000 1.207348 0.707107 +v 0.000000 1.055811 0.831469 +v 0.000000 0.882924 0.923879 +v 0.000000 0.695331 0.980785 +v 0.000000 0.500241 1.000000 +v -0.038060 1.481026 0.191342 +v -0.074658 1.424120 0.375330 +v -0.108386 1.331710 0.544895 +v -0.137949 1.207348 0.693520 +v -0.162211 1.055811 0.815493 +v -0.180240 0.882924 0.906127 +v -0.191342 0.695331 0.961939 +v -0.195090 0.500241 0.980785 +v -0.074658 1.481026 0.180240 +v -0.146446 1.424120 0.353553 +v -0.212607 1.331710 0.513280 +v -0.270598 1.207348 0.653281 +v -0.318189 1.055811 0.768177 +v -0.353553 0.882924 0.853553 +v -0.375330 0.695331 0.906127 +v -0.382683 0.500241 0.923879 +v -0.108386 1.481026 0.162212 +v -0.212607 1.424120 0.318190 +v -0.308658 1.331710 0.461940 +v -0.392847 1.207348 0.587938 +v -0.461939 1.055811 0.691341 +v -0.513280 0.882924 0.768178 +v -0.544895 0.695331 0.815493 +v -0.555570 0.500241 0.831469 +v -0.137949 1.481026 0.137950 +v -0.270598 1.424120 0.270598 +v -0.392847 1.331710 0.392847 +v -0.500000 1.207348 0.500000 +v -0.587937 1.055811 0.587937 +v -0.653281 0.882924 0.653281 +v -0.693519 0.695331 0.693519 +v -0.707106 0.500241 0.707106 +v 0.000000 1.500241 0.000000 +v -0.162211 1.481026 0.108386 +v -0.318189 1.424120 0.212608 +v -0.461939 1.331710 0.308658 +v -0.587937 1.207348 0.392847 +v -0.691341 1.055811 0.461939 +v -0.768177 0.882924 0.513280 +v -0.815493 0.695331 0.544895 +v -0.831469 0.500241 0.555570 +v -0.180240 1.481026 0.074658 +v -0.353553 1.424120 0.146447 +v -0.513280 1.331710 0.212607 +v -0.653281 1.207348 0.270598 +v -0.768177 1.055811 0.318189 +v -0.853553 0.882924 0.353553 +v -0.906127 0.695331 0.375330 +v -0.923879 0.500241 0.382683 +v -0.191341 1.481026 0.038060 +v -0.375330 1.424120 0.074658 +v -0.544895 1.331710 0.108386 +v -0.693520 1.207348 0.137950 +v -0.815492 1.055811 0.162211 +v -0.906127 0.882924 0.180240 +v -0.961939 0.695331 0.191341 +v -0.980784 0.500241 0.195090 +v -1.000000 -0.503485 0.000000 +v -0.980785 -0.698576 0.000000 +v -0.923880 -0.886169 0.000000 +v -0.831470 -1.059056 0.000000 +v -0.707107 -1.210592 0.000000 +v -0.555570 -1.334955 0.000000 +v -0.382683 -1.427365 0.000000 +v -0.195090 -1.484271 0.000000 +v -0.980785 -0.503485 -0.195090 +v -0.961940 -0.698576 -0.191342 +v -0.906127 -0.886169 -0.180240 +v -0.815493 -1.059056 -0.162212 +v -0.693520 -1.210592 -0.137950 +v -0.544895 -1.334955 -0.108386 +v -0.375330 -1.427365 -0.074658 +v -0.191341 -1.484271 -0.038060 +v -0.923880 -0.503485 -0.382684 +v -0.906127 -0.698576 -0.375330 +v -0.853553 -0.886169 -0.353554 +v -0.768178 -1.059056 -0.318190 +v -0.653281 -1.210592 -0.270598 +v -0.513280 -1.334955 -0.212608 +v -0.353553 -1.427365 -0.146447 +v -0.180240 -1.484271 -0.074658 +v -0.831470 -0.503485 -0.555570 +v -0.815493 -0.698576 -0.544895 +v -0.768178 -0.886169 -0.513280 +v -0.691342 -1.059056 -0.461940 +v -0.587938 -1.210592 -0.392848 +v -0.461940 -1.334955 -0.308658 +v -0.318189 -1.427365 -0.212608 +v -0.162211 -1.484271 -0.108386 +v -0.707107 -0.503485 -0.707107 +v -0.693520 -0.698576 -0.693520 +v -0.653281 -0.886169 -0.653282 +v -0.587938 -1.059056 -0.587938 +v -0.500000 -1.210592 -0.500000 +v -0.392847 -1.334955 -0.392848 +v -0.270598 -1.427365 -0.270598 +v -0.137949 -1.484271 -0.137950 +v -0.555570 -0.503485 -0.831470 +v -0.544895 -0.698576 -0.815493 +v -0.513280 -0.886169 -0.768178 +v -0.461939 -1.059056 -0.691342 +v -0.392847 -1.210592 -0.587938 +v -0.308658 -1.334955 -0.461940 +v -0.212607 -1.427365 -0.318190 +v -0.108386 -1.484271 -0.162212 +v -0.382683 -0.503485 -0.923880 +v -0.375330 -0.698576 -0.906128 +v -0.353553 -0.886169 -0.853554 +v -0.318189 -1.059056 -0.768178 +v -0.270598 -1.210592 -0.653282 +v -0.212607 -1.334955 -0.513280 +v -0.146446 -1.427365 -0.353554 +v -0.074658 -1.484271 -0.180240 +v -0.195090 -0.503485 -0.980785 +v -0.191341 -0.698576 -0.961940 +v -0.180240 -0.886169 -0.906128 +v -0.162211 -1.059056 -0.815493 +v -0.137949 -1.210592 -0.693520 +v -0.108386 -1.334955 -0.544895 +v -0.074658 -1.427365 -0.375330 +v -0.038060 -1.484271 -0.191342 +v 0.000000 -0.503485 -1.000000 +v 0.000000 -0.698576 -0.980785 +v 0.000000 -0.886169 -0.923880 +v 0.000000 -1.059056 -0.831470 +v 0.000000 -1.210592 -0.707107 +v 0.000000 -1.334955 -0.555570 +v 0.000000 -1.427365 -0.382684 +v 0.000000 -1.484271 -0.195090 +v 0.195091 -0.503485 -0.980785 +v 0.191342 -0.698576 -0.961940 +v 0.180240 -0.886169 -0.906128 +v 0.162212 -1.059056 -0.815493 +v 0.137950 -1.210592 -0.693520 +v 0.108387 -1.334955 -0.544895 +v 0.074658 -1.427365 -0.375330 +v 0.038061 -1.484271 -0.191342 +v 0.382684 -0.503485 -0.923880 +v 0.375331 -0.698576 -0.906127 +v 0.353554 -0.886169 -0.853554 +v 0.318190 -1.059056 -0.768178 +v 0.270599 -1.210592 -0.653282 +v 0.212608 -1.334955 -0.513280 +v 0.146447 -1.427365 -0.353553 +v 0.074658 -1.484271 -0.180240 +v 0.555571 -0.503485 -0.831470 +v 0.544895 -0.698576 -0.815493 +v 0.513280 -0.886169 -0.768178 +v 0.461940 -1.059056 -0.691342 +v 0.392848 -1.210592 -0.587938 +v 0.308659 -1.334955 -0.461940 +v 0.212608 -1.427365 -0.318190 +v 0.108387 -1.484271 -0.162212 +v 0.707107 -0.503485 -0.707107 +v 0.693520 -0.698576 -0.693520 +v 0.653282 -0.886169 -0.653282 +v 0.587938 -1.059056 -0.587938 +v 0.500000 -1.210592 -0.500000 +v 0.392848 -1.334955 -0.392847 +v 0.270598 -1.427365 -0.270598 +v 0.137950 -1.484271 -0.137950 +v 0.831470 -0.503485 -0.555570 +v 0.815493 -0.698576 -0.544895 +v 0.768178 -0.886169 -0.513280 +v 0.691342 -1.059056 -0.461940 +v 0.587938 -1.210592 -0.392847 +v 0.461940 -1.334955 -0.308658 +v 0.318190 -1.427365 -0.212608 +v 0.162212 -1.484271 -0.108386 +v 0.923880 -0.503485 -0.382683 +v 0.906128 -0.698576 -0.375330 +v 0.853554 -0.886169 -0.353553 +v 0.768178 -1.059056 -0.318190 +v 0.653282 -1.210592 -0.270598 +v 0.513280 -1.334955 -0.212607 +v 0.353554 -1.427365 -0.146447 +v 0.180240 -1.484271 -0.074658 +v 0.980785 -0.503485 -0.195090 +v 0.961940 -0.698576 -0.191342 +v 0.906128 -0.886169 -0.180240 +v 0.815493 -1.059056 -0.162212 +v 0.693520 -1.210592 -0.137950 +v 0.544895 -1.334955 -0.108386 +v 0.375331 -1.427365 -0.074658 +v 0.191342 -1.484271 -0.038060 +v 1.000000 -0.503485 0.000000 +v 0.980785 -0.698576 0.000000 +v 0.923880 -0.886169 0.000000 +v 0.831470 -1.059056 0.000000 +v 0.707107 -1.210592 0.000000 +v 0.555571 -1.334955 0.000000 +v 0.382684 -1.427365 0.000000 +v 0.195091 -1.484271 0.000000 +v 0.980785 -0.503485 0.195090 +v 0.961940 -0.698576 0.191342 +v 0.906128 -0.886169 0.180240 +v 0.815493 -1.059056 0.162212 +v 0.693520 -1.210592 0.137950 +v 0.544895 -1.334955 0.108386 +v 0.375331 -1.427365 0.074658 +v 0.191342 -1.484271 0.038060 +v 0.923880 -0.503485 0.382683 +v 0.906128 -0.698576 0.375330 +v 0.853554 -0.886169 0.353553 +v 0.768178 -1.059056 0.318190 +v 0.653282 -1.210592 0.270598 +v 0.513280 -1.334955 0.212608 +v 0.353554 -1.427365 0.146447 +v 0.180240 -1.484271 0.074658 +v 0.831470 -0.503485 0.555570 +v 0.815493 -0.698576 0.544895 +v 0.768178 -0.886169 0.513280 +v 0.691342 -1.059056 0.461940 +v 0.587938 -1.210592 0.392848 +v 0.461940 -1.334955 0.308658 +v 0.318190 -1.427365 0.212608 +v 0.162212 -1.484271 0.108386 +v 0.707107 -0.503485 0.707107 +v 0.693520 -0.698576 0.693520 +v 0.653282 -0.886169 0.653282 +v 0.587938 -1.059056 0.587938 +v 0.500000 -1.210592 0.500000 +v 0.392848 -1.334955 0.392848 +v 0.270598 -1.427365 0.270598 +v 0.137950 -1.484271 0.137950 +v 0.555570 -0.503485 0.831469 +v 0.544895 -0.698576 0.815493 +v 0.513280 -0.886169 0.768178 +v 0.461940 -1.059056 0.691342 +v 0.392848 -1.210592 0.587938 +v 0.308659 -1.334955 0.461940 +v 0.212608 -1.427365 0.318190 +v 0.108387 -1.484271 0.162212 +v 0.000000 -1.503485 0.000000 +v 0.382683 -0.503485 0.923879 +v 0.375330 -0.698576 0.906127 +v 0.353554 -0.886169 0.853553 +v 0.318190 -1.059056 0.768178 +v 0.270598 -1.210592 0.653281 +v 0.212608 -1.334955 0.513280 +v 0.146447 -1.427365 0.353553 +v 0.074658 -1.484271 0.180240 +v 0.195090 -0.503485 0.980785 +v 0.191342 -0.698576 0.961940 +v 0.180240 -0.886169 0.906128 +v 0.162212 -1.059056 0.815493 +v 0.137950 -1.210592 0.693520 +v 0.108387 -1.334955 0.544895 +v 0.074658 -1.427365 0.375330 +v 0.038061 -1.484271 0.191342 +v 0.000000 -0.503485 1.000000 +v 0.000000 -0.698576 0.980785 +v 0.000000 -0.886169 0.923880 +v 0.000000 -1.059056 0.831469 +v 0.000000 -1.210592 0.707107 +v 0.000000 -1.334955 0.555570 +v 0.000000 -1.427365 0.382683 +v 0.000000 -1.484271 0.195090 +v -0.195090 -0.503485 0.980785 +v -0.191342 -0.698576 0.961939 +v -0.180240 -0.886169 0.906127 +v -0.162211 -1.059056 0.815493 +v -0.137949 -1.210592 0.693520 +v -0.108386 -1.334955 0.544895 +v -0.074658 -1.427365 0.375330 +v -0.038060 -1.484271 0.191342 +v -0.382683 -0.503485 0.923879 +v -0.375330 -0.698576 0.906127 +v -0.353553 -0.886169 0.853553 +v -0.318189 -1.059056 0.768177 +v -0.270598 -1.210592 0.653281 +v -0.212607 -1.334955 0.513280 +v -0.146446 -1.427365 0.353553 +v -0.074657 -1.484271 0.180240 +v -0.555570 -0.503485 0.831469 +v -0.544895 -0.698576 0.815493 +v -0.513280 -0.886169 0.768178 +v -0.461939 -1.059056 0.691341 +v -0.392847 -1.210592 0.587938 +v -0.308658 -1.334955 0.461940 +v -0.212607 -1.427365 0.318190 +v -0.108386 -1.484271 0.162212 +v -0.707106 -0.503485 0.707106 +v -0.693519 -0.698576 0.693519 +v -0.653281 -0.886169 0.653281 +v -0.587937 -1.059056 0.587937 +v -0.500000 -1.210592 0.500000 +v -0.392847 -1.334955 0.392847 +v -0.270598 -1.427365 0.270598 +v -0.137949 -1.484271 0.137950 +v -0.831469 -0.503485 0.555570 +v -0.815493 -0.698576 0.544895 +v -0.768178 -0.886169 0.513280 +v -0.691341 -1.059056 0.461939 +v -0.587937 -1.210592 0.392847 +v -0.461939 -1.334955 0.308658 +v -0.318189 -1.427365 0.212608 +v -0.162211 -1.484271 0.108386 +v -0.923879 -0.503485 0.382683 +v -0.906127 -0.698576 0.375330 +v -0.853553 -0.886169 0.353553 +v -0.768177 -1.059056 0.318189 +v -0.653281 -1.210592 0.270598 +v -0.513280 -1.334955 0.212607 +v -0.353553 -1.427365 0.146447 +v -0.180240 -1.484271 0.074658 +v -0.980784 -0.503485 0.195090 +v -0.961939 -0.698576 0.191341 +v -0.906127 -0.886169 0.180240 +v -0.815492 -1.059056 0.162211 +v -0.693520 -1.210592 0.137950 +v -0.544895 -1.334955 0.108386 +v -0.375330 -1.427365 0.074658 +v -0.191341 -1.484271 0.038060 +s off +f 7 6 14 15 +f 8 7 15 16 +f 2 1 9 10 +f 3 2 10 11 +f 4 3 11 12 +f 5 4 12 13 +f 6 5 13 14 +f 11 10 18 19 +f 12 11 19 20 +f 13 12 20 21 +f 14 13 21 22 +f 15 14 22 23 +f 16 15 23 24 +f 10 9 17 18 +f 23 22 30 31 +f 24 23 31 32 +f 18 17 25 26 +f 19 18 26 27 +f 20 19 27 28 +f 21 20 28 29 +f 22 21 29 30 +f 29 28 36 37 +f 30 29 37 38 +f 31 30 38 39 +f 32 31 39 40 +f 26 25 33 34 +f 27 26 34 35 +f 28 27 35 36 +f 35 34 42 43 +f 36 35 43 44 +f 37 36 44 45 +f 38 37 45 46 +f 39 38 46 47 +f 40 39 47 48 +f 34 33 41 42 +f 47 46 54 55 +f 48 47 55 56 +f 42 41 49 50 +f 43 42 50 51 +f 44 43 51 52 +f 45 44 52 53 +f 46 45 53 54 +f 53 52 60 61 +f 54 53 61 62 +f 55 54 62 63 +f 56 55 63 64 +f 50 49 57 58 +f 51 50 58 59 +f 52 51 59 60 +f 58 57 65 66 +f 59 58 66 67 +f 60 59 67 68 +f 61 60 68 69 +f 62 61 69 70 +f 63 62 70 71 +f 64 63 71 72 +f 70 69 77 78 +f 71 70 78 79 +f 72 71 79 80 +f 66 65 73 74 +f 67 66 74 75 +f 68 67 75 76 +f 69 68 76 77 +f 76 75 83 84 +f 77 76 84 85 +f 78 77 85 86 +f 79 78 86 87 +f 80 79 87 88 +f 74 73 81 82 +f 75 74 82 83 +f 88 87 95 96 +f 82 81 89 90 +f 83 82 90 91 +f 84 83 91 92 +f 85 84 92 93 +f 86 85 93 94 +f 87 86 94 95 +f 94 93 101 102 +f 95 94 102 103 +f 96 95 103 104 +f 90 89 97 98 +f 91 90 98 99 +f 92 91 99 100 +f 93 92 100 101 +f 100 99 107 108 +f 101 100 108 109 +f 102 101 109 110 +f 103 102 110 111 +f 104 103 111 112 +f 98 97 105 106 +f 99 98 106 107 +f 112 111 119 120 +f 106 105 113 114 +f 107 106 114 115 +f 108 107 115 116 +f 109 108 116 117 +f 110 109 117 118 +f 111 110 118 119 +f 118 117 125 126 +f 119 118 126 127 +f 120 119 127 128 +f 114 113 121 122 +f 115 114 122 123 +f 116 115 123 124 +f 117 116 124 125 +f 124 123 131 132 +f 125 124 132 133 +f 126 125 133 134 +f 127 126 134 135 +f 128 127 135 136 +f 122 121 129 130 +f 123 122 130 131 +f 136 135 143 144 +f 130 129 137 138 +f 131 130 138 139 +f 132 131 139 140 +f 133 132 140 141 +f 134 133 141 142 +f 135 134 142 143 +f 141 140 148 149 +f 142 141 149 150 +f 143 142 150 151 +f 144 143 151 152 +f 138 137 145 146 +f 139 138 146 147 +f 140 139 147 148 +f 147 146 154 155 +f 148 147 155 156 +f 149 148 156 157 +f 150 149 157 158 +f 151 150 158 159 +f 152 151 159 160 +f 146 145 153 154 +f 159 158 166 167 +f 160 159 167 168 +f 154 153 161 162 +f 155 154 162 163 +f 156 155 163 164 +f 157 156 164 165 +f 158 157 165 166 +f 165 164 172 173 +f 166 165 173 174 +f 167 166 174 175 +f 168 167 175 176 +f 162 161 169 170 +f 163 162 170 171 +f 164 163 171 172 +f 171 170 178 179 +f 172 171 179 180 +f 173 172 180 181 +f 174 173 181 182 +f 175 174 182 183 +f 176 175 183 184 +f 170 169 177 178 +f 183 182 190 191 +f 184 183 191 192 +f 178 177 185 186 +f 179 178 186 187 +f 180 179 187 188 +f 181 180 188 189 +f 182 181 189 190 +f 189 188 196 197 +f 190 189 197 198 +f 191 190 198 199 +f 192 191 199 200 +f 186 185 193 194 +f 187 186 194 195 +f 188 187 195 196 +f 195 194 202 203 +f 196 195 203 204 +f 197 196 204 205 +f 198 197 205 206 +f 199 198 206 207 +f 200 199 207 208 +f 194 193 201 202 +f 207 206 214 215 +f 208 207 215 216 +f 202 201 209 210 +f 203 202 210 211 +f 204 203 211 212 +f 205 204 212 213 +f 206 205 213 214 +f 212 211 219 220 +f 213 212 220 221 +f 214 213 221 222 +f 215 214 222 223 +f 216 215 223 224 +f 210 209 217 218 +f 211 210 218 219 +f 224 223 231 232 +f 218 217 225 226 +f 219 218 226 227 +f 220 219 227 228 +f 221 220 228 229 +f 222 221 229 230 +f 223 222 230 231 +f 230 229 238 239 +f 231 230 239 240 +f 232 231 240 241 +f 226 225 234 235 +f 227 226 235 236 +f 228 227 236 237 +f 229 228 237 238 +f 237 236 244 245 +f 238 237 245 246 +f 239 238 246 247 +f 240 239 247 248 +f 241 240 248 249 +f 235 234 242 243 +f 236 235 243 244 +f 249 248 256 257 +f 243 242 250 251 +f 244 243 251 252 +f 245 244 252 253 +f 246 245 253 254 +f 247 246 254 255 +f 248 247 255 256 +f 1 233 9 +f 9 233 17 +f 17 233 25 +f 25 233 33 +f 33 233 41 +f 41 233 49 +f 49 233 57 +f 57 233 65 +f 65 233 73 +f 73 233 81 +f 81 233 89 +f 89 233 97 +f 97 233 105 +f 105 233 113 +f 113 233 121 +f 121 233 129 +f 129 233 137 +f 137 233 145 +f 145 233 153 +f 153 233 161 +f 161 233 169 +f 169 233 177 +f 177 233 185 +f 185 233 193 +f 193 233 201 +f 201 233 209 +f 209 233 217 +f 217 233 225 +f 225 233 234 +f 234 233 242 +f 242 233 250 +f 255 254 5 6 +f 256 255 6 7 +f 250 233 1 +f 257 256 7 8 +f 251 250 1 2 +f 252 251 2 3 +f 253 252 3 4 +f 254 253 4 5 +f 263 262 270 271 +f 264 263 271 272 +f 265 264 272 273 +f 259 258 266 267 +f 260 259 267 268 +f 261 260 268 269 +f 262 261 269 270 +f 273 272 280 281 +f 267 266 274 275 +f 268 267 275 276 +f 269 268 276 277 +f 270 269 277 278 +f 271 270 278 279 +f 272 271 279 280 +f 279 278 286 287 +f 280 279 287 288 +f 281 280 288 289 +f 275 274 282 283 +f 276 275 283 284 +f 277 276 284 285 +f 278 277 285 286 +f 285 284 292 293 +f 286 285 293 294 +f 287 286 294 295 +f 288 287 295 296 +f 289 288 296 297 +f 283 282 290 291 +f 284 283 291 292 +f 297 296 304 305 +f 291 290 298 299 +f 292 291 299 300 +f 293 292 300 301 +f 294 293 301 302 +f 295 294 302 303 +f 296 295 303 304 +f 303 302 310 311 +f 304 303 311 312 +f 305 304 312 313 +f 299 298 306 307 +f 300 299 307 308 +f 301 300 308 309 +f 302 301 309 310 +f 309 308 316 317 +f 310 309 317 318 +f 311 310 318 319 +f 312 311 319 320 +f 313 312 320 321 +f 307 306 314 315 +f 308 307 315 316 +f 321 320 328 329 +f 315 314 322 323 +f 316 315 323 324 +f 317 316 324 325 +f 318 317 325 326 +f 319 318 326 327 +f 320 319 327 328 +f 327 326 334 335 +f 328 327 335 336 +f 329 328 336 337 +f 323 322 330 331 +f 324 323 331 332 +f 325 324 332 333 +f 326 325 333 334 +f 332 331 339 340 +f 333 332 340 341 +f 334 333 341 342 +f 335 334 342 343 +f 336 335 343 344 +f 337 336 344 345 +f 331 330 338 339 +f 344 343 351 352 +f 345 344 352 353 +f 339 338 346 347 +f 340 339 347 348 +f 341 340 348 349 +f 342 341 349 350 +f 343 342 350 351 +f 350 349 357 358 +f 351 350 358 359 +f 352 351 359 360 +f 353 352 360 361 +f 347 346 354 355 +f 348 347 355 356 +f 349 348 356 357 +f 356 355 363 364 +f 357 356 364 365 +f 358 357 365 366 +f 359 358 366 367 +f 360 359 367 368 +f 361 360 368 369 +f 355 354 362 363 +f 368 367 375 376 +f 369 368 376 377 +f 363 362 370 371 +f 364 363 371 372 +f 365 364 372 373 +f 366 365 373 374 +f 367 366 374 375 +f 374 373 381 382 +f 375 374 382 383 +f 376 375 383 384 +f 377 376 384 385 +f 371 370 378 379 +f 372 371 379 380 +f 373 372 380 381 +f 380 379 387 388 +f 381 380 388 389 +f 382 381 389 390 +f 383 382 390 391 +f 384 383 391 392 +f 385 384 392 393 +f 379 378 386 387 +f 392 391 399 400 +f 393 392 400 401 +f 387 386 394 395 +f 388 387 395 396 +f 389 388 396 397 +f 390 389 397 398 +f 391 390 398 399 +f 398 397 405 406 +f 399 398 406 407 +f 400 399 407 408 +f 401 400 408 409 +f 395 394 402 403 +f 396 395 403 404 +f 397 396 404 405 +f 403 402 410 411 +f 404 403 411 412 +f 405 404 412 413 +f 406 405 413 414 +f 407 406 414 415 +f 408 407 415 416 +f 409 408 416 417 +f 415 414 422 423 +f 416 415 423 424 +f 417 416 424 425 +f 411 410 418 419 +f 412 411 419 420 +f 413 412 420 421 +f 414 413 421 422 +f 421 420 428 429 +f 422 421 429 430 +f 423 422 430 431 +f 424 423 431 432 +f 425 424 432 433 +f 419 418 426 427 +f 420 419 427 428 +f 433 432 441 442 +f 427 426 435 436 +f 428 427 436 437 +f 429 428 437 438 +f 430 429 438 439 +f 431 430 439 440 +f 432 431 440 441 +f 440 439 447 448 +f 441 440 448 449 +f 442 441 449 450 +f 436 435 443 444 +f 437 436 444 445 +f 438 437 445 446 +f 439 438 446 447 +f 446 445 453 454 +f 447 446 454 455 +f 448 447 455 456 +f 449 448 456 457 +f 450 449 457 458 +f 444 443 451 452 +f 445 444 452 453 +f 458 457 465 466 +f 452 451 459 460 +f 453 452 460 461 +f 454 453 461 462 +f 455 454 462 463 +f 456 455 463 464 +f 457 456 464 465 +f 464 463 471 472 +f 465 464 472 473 +f 466 465 473 474 +f 460 459 467 468 +f 461 460 468 469 +f 462 461 469 470 +f 463 462 470 471 +f 470 469 477 478 +f 471 470 478 479 +f 472 471 479 480 +f 473 472 480 481 +f 474 473 481 482 +f 468 467 475 476 +f 469 468 476 477 +f 482 481 489 490 +f 476 475 483 484 +f 477 476 484 485 +f 478 477 485 486 +f 479 478 486 487 +f 480 479 487 488 +f 481 480 488 489 +f 487 486 494 495 +f 488 487 495 496 +f 489 488 496 497 +f 490 489 497 498 +f 484 483 491 492 +f 485 484 492 493 +f 486 485 493 494 +f 493 492 500 501 +f 494 493 501 502 +f 495 494 502 503 +f 496 495 503 504 +f 497 496 504 505 +f 498 497 505 506 +f 492 491 499 500 +f 505 504 512 513 +f 506 505 513 514 +f 500 499 507 508 +f 501 500 508 509 +f 502 501 509 510 +f 503 502 510 511 +f 504 503 511 512 +f 434 265 273 +f 434 273 281 +f 434 281 289 +f 434 289 297 +f 434 297 305 +f 434 305 313 +f 434 313 321 +f 434 321 329 +f 434 329 337 +f 434 337 345 +f 434 345 353 +f 434 353 361 +f 434 361 369 +f 434 369 377 +f 434 377 385 +f 434 385 393 +f 434 393 401 +f 434 401 409 +f 434 409 417 +f 434 417 425 +f 434 425 433 +f 434 433 442 +f 434 442 450 +f 434 450 458 +f 434 458 466 +f 434 466 474 +f 434 474 482 +f 434 482 490 +f 434 490 498 +f 434 498 506 +f 434 506 514 +f 511 510 261 262 +f 512 511 262 263 +f 513 512 263 264 +f 514 513 264 265 +f 508 507 258 259 +f 434 514 265 +f 509 508 259 260 +f 510 509 260 261 +f 128 378 370 120 +f 120 370 362 112 +f 136 386 378 128 +f 144 394 386 136 +f 152 402 394 144 +f 402 152 160 410 +f 168 418 410 160 +f 176 426 418 168 +f 426 176 184 435 +f 192 443 435 184 +f 200 451 443 192 +f 451 200 208 459 +f 216 467 459 208 +f 224 475 467 216 +f 232 483 475 224 +f 241 491 483 232 +f 249 499 491 241 +f 257 507 499 249 +f 8 258 507 257 +f 16 266 258 8 +f 266 16 24 274 +f 32 282 274 24 +f 40 290 282 32 +f 290 40 48 298 +f 56 306 298 48 +f 64 314 306 56 +f 314 64 72 322 +f 80 330 322 72 +f 88 338 330 80 +f 96 346 338 88 +f 104 354 346 96 +f 112 362 354 104 From c3f4355c256f03295826fd2a5e226852de02699c Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Mon, 15 Jul 2013 19:09:07 +0200 Subject: [PATCH 44/66] Modify the method that return the bounds of a collision shape and make the getSupportPoint() methods non-const --- .../narrowphase/EPA/EPAAlgorithm.cpp | 4 +- src/collision/narrowphase/EPA/EPAAlgorithm.h | 4 +- .../narrowphase/GJK/GJKAlgorithm.cpp | 8 ++-- src/collision/narrowphase/GJK/GJKAlgorithm.h | 8 ++-- .../narrowphase/NarrowPhaseAlgorithm.h | 4 +- .../narrowphase/SphereVsSphereAlgorithm.cpp | 4 +- .../narrowphase/SphereVsSphereAlgorithm.h | 4 +- src/collision/shapes/BoxShape.h | 27 ++++++++------ src/collision/shapes/CapsuleShape.cpp | 4 +- src/collision/shapes/CapsuleShape.h | 25 ++++++++----- src/collision/shapes/CollisionShape.cpp | 23 +++++++----- src/collision/shapes/CollisionShape.h | 10 ++--- src/collision/shapes/ConeShape.cpp | 4 +- src/collision/shapes/ConeShape.h | 23 ++++++++---- src/collision/shapes/CylinderShape.cpp | 4 +- src/collision/shapes/CylinderShape.h | 23 ++++++++---- src/collision/shapes/SphereShape.h | 37 +++++++++++-------- src/engine/CollisionWorld.cpp | 1 - 18 files changed, 128 insertions(+), 89 deletions(-) diff --git a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp index a9cf7a38..6268da6b 100644 --- a/src/collision/narrowphase/EPA/EPAAlgorithm.cpp +++ b/src/collision/narrowphase/EPA/EPAAlgorithm.cpp @@ -83,9 +83,9 @@ int EPAAlgorithm::isOriginInTetrahedron(const Vector3& p1, const Vector3& p2, /// GJK algorithm. The EPA Algorithm will extend this simplex polytope to find /// the correct penetration depth bool EPAAlgorithm::computePenetrationDepthAndContactPoints(const Simplex& simplex, - const CollisionShape* collisionShape1, + CollisionShape* collisionShape1, const Transform& transform1, - const CollisionShape* collisionShape2, + CollisionShape* collisionShape2, const Transform& transform2, Vector3& v, ContactPointInfo*& contactInfo) { diff --git a/src/collision/narrowphase/EPA/EPAAlgorithm.h b/src/collision/narrowphase/EPA/EPAAlgorithm.h index 8f13b8a5..8a03f2cb 100644 --- a/src/collision/narrowphase/EPA/EPAAlgorithm.h +++ b/src/collision/narrowphase/EPA/EPAAlgorithm.h @@ -119,9 +119,9 @@ class EPAAlgorithm { /// Compute the penetration depth with EPA algorithm. bool computePenetrationDepthAndContactPoints(const Simplex& simplex, - const CollisionShape* collisionShape1, + CollisionShape* collisionShape1, const Transform& transform1, - const CollisionShape* collisionShape2, + CollisionShape* collisionShape2, const Transform& transform2, Vector3& v, ContactPointInfo*& contactInfo); }; diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.cpp b/src/collision/narrowphase/GJK/GJKAlgorithm.cpp index 392c704d..6ab3e959 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.cpp +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.cpp @@ -57,9 +57,9 @@ GJKAlgorithm::~GJKAlgorithm() { /// algorithm on the enlarged object to obtain a simplex polytope that contains the /// origin, they we give that simplex polytope to the EPA algorithm which will compute /// the correct penetration depth and contact points between the enlarged objects. -bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, +bool GJKAlgorithm::testCollision(CollisionShape* collisionShape1, const Transform& transform1, - const CollisionShape* collisionShape2, + CollisionShape* collisionShape2, const Transform& transform2, ContactPointInfo*& contactInfo) { @@ -259,9 +259,9 @@ bool GJKAlgorithm::testCollision(const CollisionShape* collisionShape1, /// assumed to intersect in the original objects (without margin). Therefore such /// a polytope must exist. Then, we give that polytope to the EPA algorithm to /// compute the correct penetration depth and contact points of the enlarged objects. -bool GJKAlgorithm::computePenetrationDepthForEnlargedObjects(const CollisionShape* collisionShape1, +bool GJKAlgorithm::computePenetrationDepthForEnlargedObjects(CollisionShape* collisionShape1, const Transform& transform1, - const CollisionShape* collisionShape2, + CollisionShape* collisionShape2, const Transform& transform2, ContactPointInfo*& contactInfo, Vector3& v) { diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.h b/src/collision/narrowphase/GJK/GJKAlgorithm.h index a7c5a940..dae18d66 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.h +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.h @@ -74,9 +74,9 @@ class GJKAlgorithm : public NarrowPhaseAlgorithm { GJKAlgorithm& operator=(const GJKAlgorithm& algorithm); /// Compute the penetration depth for enlarged objects. - bool computePenetrationDepthForEnlargedObjects(const CollisionShape* collisionShape1, + bool computePenetrationDepthForEnlargedObjects(CollisionShape* collisionShape1, const Transform& transform1, - const CollisionShape* collisionShape2, + CollisionShape* collisionShape2, const Transform& transform2, ContactPointInfo*& contactInfo, Vector3& v); @@ -91,9 +91,9 @@ class GJKAlgorithm : public NarrowPhaseAlgorithm { ~GJKAlgorithm(); /// Return true and compute a contact info if the two bounding volumes collide. - virtual bool testCollision(const CollisionShape* collisionShape1, + virtual bool testCollision(CollisionShape *collisionShape1, const Transform& transform1, - const CollisionShape* collisionShape2, + CollisionShape *collisionShape2, const Transform& transform2, ContactPointInfo*& contactInfo); }; diff --git a/src/collision/narrowphase/NarrowPhaseAlgorithm.h b/src/collision/narrowphase/NarrowPhaseAlgorithm.h index 79ac19d9..3dd5d086 100644 --- a/src/collision/narrowphase/NarrowPhaseAlgorithm.h +++ b/src/collision/narrowphase/NarrowPhaseAlgorithm.h @@ -78,9 +78,9 @@ class NarrowPhaseAlgorithm { void setCurrentOverlappingPair(BroadPhasePair* overlappingPair); /// Return true and compute a contact info if the two bounding volume collide - virtual bool testCollision(const CollisionShape* collisionShape1, + virtual bool testCollision(CollisionShape* collisionShape1, const Transform& transform1, - const CollisionShape* collisionShape2, + CollisionShape* collisionShape2, const Transform& transform2, ContactPointInfo*& contactInfo)=0; }; diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp index d50fe507..539cb487 100644 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp @@ -41,9 +41,9 @@ SphereVsSphereAlgorithm::~SphereVsSphereAlgorithm() { } -bool SphereVsSphereAlgorithm::testCollision(const CollisionShape* collisionShape1, +bool SphereVsSphereAlgorithm::testCollision(CollisionShape* collisionShape1, const Transform& transform1, - const CollisionShape* collisionShape2, + CollisionShape* collisionShape2, const Transform& transform2, ContactPointInfo*& contactInfo) { diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.h b/src/collision/narrowphase/SphereVsSphereAlgorithm.h index d47f55aa..f5c6cc18 100644 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.h +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.h @@ -63,9 +63,9 @@ class SphereVsSphereAlgorithm : public NarrowPhaseAlgorithm { virtual ~SphereVsSphereAlgorithm(); /// Return true and compute a contact info if the two bounding volume collide - virtual bool testCollision(const CollisionShape* collisionShape1, + virtual bool testCollision(CollisionShape* collisionShape1, const Transform& transform1, - const CollisionShape* collisionShape2, + CollisionShape* collisionShape2, const Transform& transform2, ContactPointInfo*& contactInfo); }; diff --git a/src/collision/shapes/BoxShape.h b/src/collision/shapes/BoxShape.h index 6ebb1a81..f53ebc70 100644 --- a/src/collision/shapes/BoxShape.h +++ b/src/collision/shapes/BoxShape.h @@ -80,19 +80,19 @@ class BoxShape : public CollisionShape { virtual BoxShape* clone(void* allocatedMemory) const; /// Return the extents of the box - const Vector3& getExtent() const; + Vector3 getExtent() const; - /// Return the local extents in x,y and z direction. - virtual Vector3 getLocalExtents() const; + /// Return the local bounds of the shape in x, y and z directions + virtual void getLocalBounds(Vector3& min, Vector3& max) const; /// Return the number of bytes used by the collision shape virtual size_t getSizeInBytes() const; /// Return a local support point in a given direction with the object margin - virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const; + virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction); /// Return a local support point in a given direction without the object margin - virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const; + virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction); /// Return the local inertia tensor of the collision shape virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; @@ -107,14 +107,19 @@ inline BoxShape* BoxShape::clone(void* allocatedMemory) const { } // Return the extents of the box -inline const Vector3& BoxShape::getExtent() const { +inline Vector3 BoxShape::getExtent() const { return mExtent + Vector3(mMargin, mMargin, mMargin); } -// Return the local extents of the box (half-width) in x,y and z local direction. +// Return the local bounds of the shape in x, y and z directions /// This method is used to compute the AABB of the box -inline Vector3 BoxShape::getLocalExtents() const { - return mExtent + Vector3(mMargin, mMargin, mMargin); +inline void BoxShape::getLocalBounds(Vector3& min, Vector3& max) const { + + // Maximum bounds + max = mExtent + Vector3(mMargin, mMargin, mMargin); + + // Minimum bounds + min = -max; } // Return the number of bytes used by the collision shape @@ -123,7 +128,7 @@ inline size_t BoxShape::getSizeInBytes() const { } // Return a local support point in a given direction with the object margin -inline Vector3 BoxShape::getLocalSupportPointWithMargin(const Vector3& direction) const { +inline Vector3 BoxShape::getLocalSupportPointWithMargin(const Vector3& direction) { assert(mMargin > 0.0); @@ -133,7 +138,7 @@ inline Vector3 BoxShape::getLocalSupportPointWithMargin(const Vector3& direction } // Return a local support point in a given direction without the objec margin -inline Vector3 BoxShape::getLocalSupportPointWithoutMargin(const Vector3& direction) const { +inline Vector3 BoxShape::getLocalSupportPointWithoutMargin(const Vector3& direction) { return Vector3(direction.x < 0.0 ? -mExtent.x : mExtent.x, direction.y < 0.0 ? -mExtent.y : mExtent.y, diff --git a/src/collision/shapes/CapsuleShape.cpp b/src/collision/shapes/CapsuleShape.cpp index 6397b693..ff1103ff 100644 --- a/src/collision/shapes/CapsuleShape.cpp +++ b/src/collision/shapes/CapsuleShape.cpp @@ -55,7 +55,7 @@ CapsuleShape::~CapsuleShape() { /// Therefore, in this method, we compute the support points of both top and bottom spheres of /// the capsule and return the point with the maximum dot product with the direction vector. Note /// that the object margin is implicitly the radius and height of the capsule. -Vector3 CapsuleShape::getLocalSupportPointWithMargin(const Vector3& direction) const { +Vector3 CapsuleShape::getLocalSupportPointWithMargin(const Vector3& direction) { // If the direction vector is not the zero vector if (direction.lengthSquare() >= MACHINE_EPSILON * MACHINE_EPSILON) { @@ -87,7 +87,7 @@ Vector3 CapsuleShape::getLocalSupportPointWithMargin(const Vector3& direction) c } // Return a local support point in a given direction without the object margin. -Vector3 CapsuleShape::getLocalSupportPointWithoutMargin(const Vector3& direction) const { +Vector3 CapsuleShape::getLocalSupportPointWithoutMargin(const Vector3& direction) { // If the dot product of the direction and the local Y axis (dotProduct = direction.y) // is positive diff --git a/src/collision/shapes/CapsuleShape.h b/src/collision/shapes/CapsuleShape.h index dc7592e1..a43de124 100644 --- a/src/collision/shapes/CapsuleShape.h +++ b/src/collision/shapes/CapsuleShape.h @@ -86,13 +86,13 @@ class CapsuleShape : public CollisionShape { virtual size_t getSizeInBytes() const; /// Return a local support point in a given direction with the object margin. - virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const; + virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction); /// Return a local support point in a given direction without the object margin - virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const; + virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction); - /// Return the local extents in x,y and z direction - virtual Vector3 getLocalExtents() const; + /// Return the local bounds of the shape in x, y and z directions + virtual void getLocalBounds(Vector3& min, Vector3& max) const; /// Return the local inertia tensor of the collision shape virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; @@ -121,12 +121,19 @@ inline size_t CapsuleShape::getSizeInBytes() const { return sizeof(CapsuleShape); } -// Return the local extents of the collision shape (half-width) in x,y and z local direction +// Return the local bounds of the shape in x, y and z directions // This method is used to compute the AABB of the box -inline Vector3 CapsuleShape::getLocalExtents() const { - return Vector3(mRadius, - mHalfHeight + mRadius, - mRadius); +inline void CapsuleShape::getLocalBounds(Vector3& min, Vector3& max) const { + + // Maximum bounds + max.x = mRadius; + max.y = mHalfHeight + mRadius; + max.z = mRadius; + + // Minimum bounds + min.x = -mRadius; + min.y = -max.y; + min.z = min.x; } // Test equality between two capsule shapes diff --git a/src/collision/shapes/CollisionShape.cpp b/src/collision/shapes/CollisionShape.cpp index 45a2efa7..e895f9bb 100644 --- a/src/collision/shapes/CollisionShape.cpp +++ b/src/collision/shapes/CollisionShape.cpp @@ -48,20 +48,25 @@ CollisionShape::~CollisionShape() { } // Update the AABB of a body using its collision shape -inline void CollisionShape::updateAABB(AABB& aabb, const Transform& transform) { +void CollisionShape::updateAABB(AABB& aabb, const Transform& transform) { - // Get the local extents in x,y and z direction - Vector3 extents = getLocalExtents(); + // Get the local bounds in x,y and z direction + Vector3 minBounds; + Vector3 maxBounds; + getLocalBounds(minBounds, maxBounds); - // Rotate the local extents according to the orientation of the body + // Rotate the local bounds according to the orientation of the body Matrix3x3 worldAxis = transform.getOrientation().getMatrix().getAbsoluteMatrix(); - Vector3 worldExtents = Vector3(worldAxis.getColumn(0).dot(extents), - worldAxis.getColumn(1).dot(extents), - worldAxis.getColumn(2).dot(extents)); + Vector3 worldMinBounds(worldAxis.getColumn(0).dot(minBounds), + worldAxis.getColumn(1).dot(minBounds), + worldAxis.getColumn(2).dot(minBounds)); + Vector3 worldMaxBounds(worldAxis.getColumn(0).dot(maxBounds), + worldAxis.getColumn(1).dot(maxBounds), + worldAxis.getColumn(2).dot(maxBounds)); // Compute the minimum and maximum coordinates of the rotated extents - Vector3 minCoordinates = transform.getPosition() - worldExtents; - Vector3 maxCoordinates = transform.getPosition() + worldExtents; + Vector3 minCoordinates = transform.getPosition() + worldMinBounds; + Vector3 maxCoordinates = transform.getPosition() + worldMaxBounds; // Update the AABB with the new minimum and maximum coordinates aabb.setMin(minCoordinates); diff --git a/src/collision/shapes/CollisionShape.h b/src/collision/shapes/CollisionShape.h index 9436277d..413015f8 100644 --- a/src/collision/shapes/CollisionShape.h +++ b/src/collision/shapes/CollisionShape.h @@ -36,7 +36,7 @@ namespace reactphysics3d { /// Type of the collision shape -enum CollisionShapeType {BOX, SPHERE, CONE, CYLINDER, CAPSULE}; +enum CollisionShapeType {BOX, SPHERE, CONE, CYLINDER, CAPSULE, CONVEX_MESH}; // Declarations class Body; @@ -95,13 +95,13 @@ class CollisionShape { virtual size_t getSizeInBytes() const = 0; /// Return a local support point in a given direction with the object margin - virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const=0; + virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction)=0; /// Return a local support point in a given direction without the object margin - virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const=0; + virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction)=0; - /// Return the local extents in x,y and z direction - virtual Vector3 getLocalExtents() const=0; + /// Return the local bounds of the shape in x, y and z directions + virtual void getLocalBounds(Vector3& min, Vector3& max) const=0; /// Return the local inertia tensor of the collision shapes virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const=0; diff --git a/src/collision/shapes/ConeShape.cpp b/src/collision/shapes/ConeShape.cpp index 1cdd0392..b674384d 100644 --- a/src/collision/shapes/ConeShape.cpp +++ b/src/collision/shapes/ConeShape.cpp @@ -54,7 +54,7 @@ ConeShape::~ConeShape() { } // Return a local support point in a given direction with the object margin -inline Vector3 ConeShape::getLocalSupportPointWithMargin(const Vector3& direction) const { +Vector3 ConeShape::getLocalSupportPointWithMargin(const Vector3& direction) { // Compute the support point without the margin Vector3 supportPoint = getLocalSupportPointWithoutMargin(direction); @@ -70,7 +70,7 @@ inline Vector3 ConeShape::getLocalSupportPointWithMargin(const Vector3& directio } // Return a local support point in a given direction without the object margin -inline Vector3 ConeShape::getLocalSupportPointWithoutMargin(const Vector3& direction) const { +Vector3 ConeShape::getLocalSupportPointWithoutMargin(const Vector3& direction) { const Vector3& v = direction; decimal sinThetaTimesLengthV = mSinTheta * v.length(); diff --git a/src/collision/shapes/ConeShape.h b/src/collision/shapes/ConeShape.h index 5ac83d06..1c4a9326 100644 --- a/src/collision/shapes/ConeShape.h +++ b/src/collision/shapes/ConeShape.h @@ -94,13 +94,13 @@ class ConeShape : public CollisionShape { virtual size_t getSizeInBytes() const; /// Return a local support point in a given direction with the object margin - virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const; + virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction); /// Return a local support point in a given direction without the object margin - virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const; + virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction); - /// Return the local extents in x,y and z direction - virtual Vector3 getLocalExtents() const; + /// Return the local bounds of the shape in x, y and z directions + virtual void getLocalBounds(Vector3& min, Vector3& max) const; /// Return the local inertia tensor of the collision shape virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; @@ -129,9 +129,18 @@ inline size_t ConeShape::getSizeInBytes() const { return sizeof(ConeShape); } -// Return the local extents in x,y and z direction -inline Vector3 ConeShape::getLocalExtents() const { - return Vector3(mRadius + mMargin, mHalfHeight + mMargin, mRadius + mMargin); +// Return the local bounds of the shape in x, y and z directions +inline void ConeShape::getLocalBounds(Vector3& min, Vector3& max) const { + + // Maximum bounds + max.x = mRadius + mMargin; + max.y = mHalfHeight + mMargin; + max.z = max.x; + + // Minimum bounds + min.x = -max.x; + min.y = -max.y; + min.z = min.x; } // Return the local inertia tensor of the collision shape diff --git a/src/collision/shapes/CylinderShape.cpp b/src/collision/shapes/CylinderShape.cpp index ac5a413b..675dd480 100644 --- a/src/collision/shapes/CylinderShape.cpp +++ b/src/collision/shapes/CylinderShape.cpp @@ -50,7 +50,7 @@ CylinderShape::~CylinderShape() { } // Return a local support point in a given direction with the object margin -Vector3 CylinderShape::getLocalSupportPointWithMargin(const Vector3& direction) const { +Vector3 CylinderShape::getLocalSupportPointWithMargin(const Vector3& direction) { // Compute the support point without the margin Vector3 supportPoint = getLocalSupportPointWithoutMargin(direction); @@ -66,7 +66,7 @@ Vector3 CylinderShape::getLocalSupportPointWithMargin(const Vector3& direction) } // Return a local support point in a given direction without the object margin -Vector3 CylinderShape::getLocalSupportPointWithoutMargin(const Vector3& direction) const { +Vector3 CylinderShape::getLocalSupportPointWithoutMargin(const Vector3& direction) { Vector3 supportPoint(0.0, 0.0, 0.0); decimal uDotv = direction.y; diff --git a/src/collision/shapes/CylinderShape.h b/src/collision/shapes/CylinderShape.h index 8cdcfc3f..748bc4f4 100644 --- a/src/collision/shapes/CylinderShape.h +++ b/src/collision/shapes/CylinderShape.h @@ -91,13 +91,13 @@ class CylinderShape : public CollisionShape { virtual size_t getSizeInBytes() const; /// Return a local support point in a given direction with the object margin - virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const; + virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction); /// Return a local support point in a given direction without the object margin - virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const; + virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction); - /// Return the local extents in x,y and z direction - virtual Vector3 getLocalExtents() const; + /// Return the local bounds of the shape in x, y and z directions + virtual void getLocalBounds(Vector3& min, Vector3& max) const; /// Return the local inertia tensor of the collision shape virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; @@ -126,9 +126,18 @@ inline size_t CylinderShape::getSizeInBytes() const { return sizeof(CylinderShape); } -// Return the local extents in x,y and z direction -inline Vector3 CylinderShape::getLocalExtents() const { - return Vector3(mRadius + mMargin, mHalfHeight + mMargin, mRadius + mMargin); +// Return the local bounds of the shape in x, y and z directions +inline void CylinderShape::getLocalBounds(Vector3& min, Vector3& max) const { + + // Maximum bounds + max.x = mRadius + mMargin; + max.y = mHalfHeight + mMargin; + max.z = max.x; + + // Minimum bounds + min.x = -max.x; + min.y = -max.y; + min.z = min.x; } // Return the local inertia tensor of the cylinder diff --git a/src/collision/shapes/SphereShape.h b/src/collision/shapes/SphereShape.h index 1d46d888..962c0277 100644 --- a/src/collision/shapes/SphereShape.h +++ b/src/collision/shapes/SphereShape.h @@ -78,13 +78,13 @@ class SphereShape : public CollisionShape { virtual size_t getSizeInBytes() const; /// Return a local support point in a given direction with the object margin - virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction) const; + virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction); /// Return a local support point in a given direction without the object margin - virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction) const; + virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction); - /// Return the local extents in x,y and z direction - virtual Vector3 getLocalExtents() const; + /// Return the local bounds of the shape in x, y and z directions. + virtual void getLocalBounds(Vector3& min, Vector3& max) const; /// Return the local inertia tensor of the collision shape virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; @@ -112,7 +112,7 @@ inline size_t SphereShape::getSizeInBytes() const { } // Return a local support point in a given direction with the object margin -inline Vector3 SphereShape::getLocalSupportPointWithMargin(const Vector3& direction) const { +inline Vector3 SphereShape::getLocalSupportPointWithMargin(const Vector3& direction) { // If the direction vector is not the zero vector if (direction.lengthSquare() >= MACHINE_EPSILON * MACHINE_EPSILON) { @@ -127,16 +127,25 @@ inline Vector3 SphereShape::getLocalSupportPointWithMargin(const Vector3& direct } // Return a local support point in a given direction without the object margin -inline Vector3 SphereShape::getLocalSupportPointWithoutMargin(const Vector3& direction) const { +inline Vector3 SphereShape::getLocalSupportPointWithoutMargin(const Vector3& direction) { // Return the center of the sphere (the radius is taken into account in the object margin) return Vector3(0.0, 0.0, 0.0); } -// Return the local extents of the collision shape (half-width) in x,y and z local direction +// Return the local bounds of the shape in x, y and z directions. // This method is used to compute the AABB of the box -inline Vector3 SphereShape::getLocalExtents() const { - return Vector3(mRadius, mRadius, mRadius); +inline void SphereShape::getLocalBounds(Vector3& min, Vector3& max) const { + + // Maximum bounds + max.x = mRadius; + max.y = mRadius; + max.z = mRadius; + + // Minimum bounds + min.x = -mRadius; + min.y = min.x; + min.z = min.x; } // Return the local inertia tensor of the sphere @@ -151,15 +160,11 @@ inline void SphereShape::computeLocalInertiaTensor(Matrix3x3& tensor, decimal ma inline void SphereShape::updateAABB(AABB& aabb, const Transform& transform) { // Get the local extents in x,y and z direction - Vector3 extents = getLocalExtents(); - - // Compute the minimum and maximum coordinates of the rotated extents - Vector3 minCoordinates = transform.getPosition() - extents; - Vector3 maxCoordinates = transform.getPosition() + extents; + Vector3 extents(mRadius, mRadius, mRadius); // Update the AABB with the new minimum and maximum coordinates - aabb.setMin(minCoordinates); - aabb.setMax(maxCoordinates); + aabb.setMin(transform.getPosition() - extents); + aabb.setMax(transform.getPosition() + extents); } // Test equality between two sphere shapes diff --git a/src/engine/CollisionWorld.cpp b/src/engine/CollisionWorld.cpp index 2f2d4d81..8674a289 100644 --- a/src/engine/CollisionWorld.cpp +++ b/src/engine/CollisionWorld.cpp @@ -154,7 +154,6 @@ CollisionShape* CollisionWorld::createCollisionShape(const CollisionShape& colli // A similar collision shape does not already exist in the world, so we create a // new one and add it to the world void* allocatedMemory = mMemoryAllocator.allocate(collisionShape.getSizeInBytes()); - size_t test = collisionShape.getSizeInBytes(); CollisionShape* newCollisionShape = collisionShape.clone(allocatedMemory); mCollisionShapes.push_back(newCollisionShape); From ffd79a89e39f0264e64be7a3dab3e10158173739 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Mon, 15 Jul 2013 19:10:30 +0200 Subject: [PATCH 45/66] Add the convex mesh collision shape --- src/collision/shapes/ConvexMeshShape.cpp | 228 +++++++++++++++++++++++ src/collision/shapes/ConvexMeshShape.h | 227 ++++++++++++++++++++++ src/reactphysics3d.h | 1 + 3 files changed, 456 insertions(+) create mode 100644 src/collision/shapes/ConvexMeshShape.cpp create mode 100644 src/collision/shapes/ConvexMeshShape.h diff --git a/src/collision/shapes/ConvexMeshShape.cpp b/src/collision/shapes/ConvexMeshShape.cpp new file mode 100644 index 00000000..75f819bc --- /dev/null +++ b/src/collision/shapes/ConvexMeshShape.cpp @@ -0,0 +1,228 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include +#include "../../configuration.h" +#include "ConvexMeshShape.h" + +using namespace reactphysics3d; + +// Constructor to initialize with a array of 3D vertices. +/// This method creates an internal copy of the input vertices. +ConvexMeshShape::ConvexMeshShape(const decimal* arrayVertices, uint nbVertices, int stride, + decimal margin) + : CollisionShape(CONVEX_MESH, margin), mNbVertices(nbVertices), mMinBounds(0, 0, 0), + mMaxBounds(0, 0, 0), mIsEdgesInformationUsed(false), mCachedSupportVertex(0) { + assert(nbVertices > 0); + assert(stride > 0); + assert(margin > decimal(0.0)); + + const unsigned char* vertexPointer = (const unsigned char*) arrayVertices; + + // Copy all the vertices into the internal array + for (uint i=0; i decimal(0.0)); +} + +// Private copy-constructor +ConvexMeshShape::ConvexMeshShape(const ConvexMeshShape& shape) + : CollisionShape(shape), mVertices(shape.mVertices), mNbVertices(shape.mNbVertices), + mMinBounds(shape.mMinBounds), mMaxBounds(shape.mMaxBounds), + mIsEdgesInformationUsed(shape.mIsEdgesInformationUsed), + mEdgesAdjacencyList(shape.mEdgesAdjacencyList), + mCachedSupportVertex(shape.mCachedSupportVertex) { + + assert(mNbVertices == mVertices.size()); +} + +// Destructor +ConvexMeshShape::~ConvexMeshShape() { + +} + +// Return a local support point in a given direction with the object margin +Vector3 ConvexMeshShape::getLocalSupportPointWithMargin(const Vector3& direction) { + + // Get the support point without the margin + Vector3 supportPoint = getLocalSupportPointWithoutMargin(direction); + + // Get the unit direction vector + Vector3 unitDirection = direction; + if (direction.lengthSquare() < MACHINE_EPSILON * MACHINE_EPSILON) { + unitDirection.setAllValues(1.0, 1.0, 1.0); + } + unitDirection.normalize(); + + // Add the margin to the support point and return it + return supportPoint + unitDirection * mMargin; +} + +// Return a local support point in a given direction without the object margin. +/// If the edges information is not used for collision detection, this method will go through +/// the whole vertices list and pick up the vertex with the largest dot product in the support +/// direction. This is an O(n) process with "n" being the number of vertices in the mesh. +/// However, if the edges information is used, we can cache the previous support vertex and use +/// it as a start in a hill-climbing (local search) process to find the new support vertex which +/// will be in most of the cases very close to the previous one. Using hill-climbing, this method +/// runs in almost constant time. +Vector3 ConvexMeshShape::getLocalSupportPointWithoutMargin(const Vector3& direction) { + + assert(mNbVertices == mVertices.size()); + + // If the edges information is used to speed up the collision detection + if (mIsEdgesInformationUsed) { + + assert(mEdgesAdjacencyList.size() == mNbVertices); + + uint maxVertex = mCachedSupportVertex; + decimal maxDotProduct = direction.dot(mVertices[maxVertex]); + bool isOptimal; + + // Perform hill-climbing (local search) + do { + isOptimal = true; + + assert(mEdgesAdjacencyList.at(maxVertex).size() > 0); + + // For all neighbors of the current vertex + std::set::const_iterator it; + std::set::const_iterator itBegin = mEdgesAdjacencyList.at(maxVertex).begin(); + std::set::const_iterator itEnd = mEdgesAdjacencyList.at(maxVertex).end(); + for (it = itBegin; it != itEnd; ++it) { + + // Compute the dot product + decimal dotProduct = direction.dot(mVertices[*it]); + + // If the current vertex is a better vertex (larger dot product) + if (dotProduct > maxDotProduct) { + maxVertex = *it; + maxDotProduct = dotProduct; + isOptimal = false; + } + } + + } while(!isOptimal); + + // Cache the support vertex + mCachedSupportVertex = maxVertex; + + // Return the support vertex + return mVertices[maxVertex]; + } + else { // If the edges information is not used + + decimal maxDotProduct = DECIMAL_SMALLEST; + uint indexMaxDotProduct = 0; + + // For each vertex of the mesh + for (uint i=0; i maxDotProduct) { + indexMaxDotProduct = i; + maxDotProduct = dotProduct; + } + } + + assert(maxDotProduct >= decimal(0.0)); + + // Return the vertex with the largest dot product in the support direction + return mVertices[indexMaxDotProduct]; + } +} + +// Recompute the bounds of the mesh +void ConvexMeshShape::recalculateBounds() { + + mMinBounds.setToZero(); + mMaxBounds.setToZero(); + + // For each vertex of the mesh + for (uint i=0; i mMaxBounds.x) mMaxBounds.x = mVertices[i].x; + if (mVertices[i].x < mMinBounds.x) mMinBounds.x = mVertices[i].x; + + if (mVertices[i].y > mMaxBounds.y) mMaxBounds.y = mVertices[i].y; + if (mVertices[i].y < mMinBounds.y) mMinBounds.y = mVertices[i].y; + + if (mVertices[i].z > mMaxBounds.z) mMaxBounds.z = mVertices[i].z; + if (mVertices[i].z < mMinBounds.z) mMinBounds.z = mVertices[i].z; + } + + // Add the object margin to the bounds + mMaxBounds += Vector3(mMargin, mMargin, mMargin); + mMinBounds -= Vector3(mMargin, mMargin, mMargin); +} + +// Test equality between two cone shapes +bool ConvexMeshShape::isEqualTo(const CollisionShape& otherCollisionShape) const { + const ConvexMeshShape& otherShape = dynamic_cast(otherCollisionShape); + + assert(mNbVertices == mVertices.size()); + + if (mNbVertices != otherShape.mNbVertices) return false; + + // If edges information is used, it means that a collison shape object will store + // cached data (previous support vertex) and therefore, we should not reuse the shape + // for another body. Therefore, we consider that all convex mesh shape using edges + // information are different. + if (mIsEdgesInformationUsed) return false; + + if (mEdgesAdjacencyList.size() != otherShape.mEdgesAdjacencyList.size()) return false; + + // Check that the vertices are the same + for (uint i=0; i +#include +#include + +/// ReactPhysics3D namespace +namespace reactphysics3d { + +// Class ConvexMeshShape +/** + * This class represents a convex mesh shape. In order to create a convex mesh shape, you + * need to indicate the local-space position of the mesh vertices. You do it either by + * passing a vertices array to the constructor or using the addVertex() method. Make sure + * that the set of vertices that you use to create the shape are indeed part of a convex + * mesh. The center of mass of the shape will be at the origin of the local-space geometry + * that you use to create the mesh. The method used for collision detection with a convex + * mesh shape has an O(n) running time with "n" beeing the number of vertices in the mesh. + * Therefore, you should try not to use too many vertices. However, it is possible to speed + * up the collision detection by using the edges information of your mesh. The running time + * of the collision detection that uses the edges is almost O(1) constant time at the cost + * of additional memory used to store the vertices. You can indicate edges information + * with the addEdge() method. Then, you must use the setIsEdgesInformationUsed(true) method + * in order to use the edges information for collision detection. + */ +class ConvexMeshShape : public CollisionShape { + + private : + + // -------------------- Attributes -------------------- // + + /// Array with the vertices of the mesh + std::vector mVertices; + + /// Number of vertices in the mesh + uint mNbVertices; + + /// Mesh minimum bounds in the three local x, y and z directions + Vector3 mMinBounds; + + /// Mesh maximum bounds in the three local x, y and z directions + Vector3 mMaxBounds; + + /// True if the shape contains the edges of the convex mesh in order to + /// make the collision detection faster + bool mIsEdgesInformationUsed; + + /// Adjacency list representing the edges of the mesh + std::map > mEdgesAdjacencyList; + + /// Cached support vertex index (previous support vertex) + uint mCachedSupportVertex; + + // -------------------- Methods -------------------- // + + /// Private copy-constructor + ConvexMeshShape(const ConvexMeshShape& shape); + + /// Private assignment operator + ConvexMeshShape& operator=(const ConvexMeshShape& shape); + + /// Recompute the bounds of the mesh + void recalculateBounds(); + + public : + + // -------------------- Methods -------------------- // + + /// Constructor to initialize with a array of 3D vertices. + ConvexMeshShape(const decimal* arrayVertices, uint nbVertices, int stride, + decimal margin = OBJECT_MARGIN); + + /// Constructor. + ConvexMeshShape(decimal margin = OBJECT_MARGIN); + + /// Destructor + virtual ~ConvexMeshShape(); + + /// Allocate and return a copy of the object + virtual ConvexMeshShape* clone(void* allocatedMemory) const; + + /// Return the number of bytes used by the collision shape + virtual size_t getSizeInBytes() const; + + /// Return a local support point in a given direction with the object margin + virtual Vector3 getLocalSupportPointWithMargin(const Vector3& direction); + + /// Return a local support point in a given direction without the object margin. + virtual Vector3 getLocalSupportPointWithoutMargin(const Vector3& direction); + + /// Return the local bounds of the shape in x, y and z directions + virtual void getLocalBounds(Vector3& min, Vector3& max) const; + + /// Return the local inertia tensor of the collision shape. + virtual void computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const; + + /// Test equality between two cone shapes + virtual bool isEqualTo(const CollisionShape& otherCollisionShape) const; + + /// Add a vertex into the convex mesh + void addVertex(const Vector3& vertex); + + /// Add an edge into the convex mesh by specifying the two vertex indices of the edge. + void addEdge(uint v1, uint v2); + + /// Return true if the edges information is used to speed up the collision detection + bool isEdgesInformationUsed() const; + + /// Set the variable to know if the edges information is used to speed up the + /// collision detection + void setIsEdgesInformationUsed(bool isEdgesUsed); +}; + +// Allocate and return a copy of the object +inline ConvexMeshShape* ConvexMeshShape::clone(void* allocatedMemory) const { + return new (allocatedMemory) ConvexMeshShape(*this); +} + +// Return the number of bytes used by the collision shape +inline size_t ConvexMeshShape::getSizeInBytes() const { + return sizeof(ConvexMeshShape); +} + +// Return the local bounds of the shape in x, y and z directions +inline void ConvexMeshShape::getLocalBounds(Vector3& min, Vector3& max) const { + min = mMinBounds; + max = mMaxBounds; +} + +// Return the local inertia tensor of the collision shape. +/// The local inertia tensor of the convex mesh is approximated using the inertia tensor +/// of its bounding box. +inline void ConvexMeshShape::computeLocalInertiaTensor(Matrix3x3& tensor, decimal mass) const { + decimal factor = (decimal(1.0) / decimal(3.0)) * mass; + Vector3 realExtent = decimal(0.5) * (mMaxBounds - mMinBounds); + assert(realExtent.x > 0 && realExtent.y > 0 && realExtent.z > 0); + decimal xSquare = realExtent.x * realExtent.x; + decimal ySquare = realExtent.y * realExtent.y; + decimal zSquare = realExtent.z * realExtent.z; + tensor.setAllValues(factor * (ySquare + zSquare), 0.0, 0.0, + 0.0, factor * (xSquare + zSquare), 0.0, + 0.0, 0.0, factor * (xSquare + ySquare)); +} + +// Add a vertex into the convex mesh +inline void ConvexMeshShape::addVertex(const Vector3& vertex) { + + // Add the vertex in to vertices array + mVertices.push_back(vertex); + mNbVertices++; + + // Update the bounds of the mesh + if (vertex.x > mMaxBounds.x) mMaxBounds.x = vertex.x; + if (vertex.x < mMinBounds.x) mMinBounds.x = vertex.x; + if (vertex.y > mMaxBounds.y) mMaxBounds.y = vertex.y; + if (vertex.y < mMinBounds.y) mMinBounds.y = vertex.y; + if (vertex.z > mMaxBounds.z) mMaxBounds.z = vertex.z; + if (vertex.z < mMinBounds.z) mMinBounds.z = vertex.z; +} + +// Add an edge into the convex mesh by specifying the two vertex indices of the edge. +/// Note that the vertex indices start at zero and need to correspond to the order of +/// the vertices in the vertices array in the constructor or the order of the calls +/// of the addVertex() methods that you use to add vertices into the convex mesh. +inline void ConvexMeshShape::addEdge(uint v1, uint v2) { + + assert(v1 >= 0); + assert(v2 >= 0); + + // If the entry for vertex v1 does not exist in the adjacency list + if (mEdgesAdjacencyList.count(v1) == 0) { + mEdgesAdjacencyList.insert(std::make_pair >(v1, std::set())); + } + + // If the entry for vertex v2 does not exist in the adjacency list + if (mEdgesAdjacencyList.count(v2) == 0) { + mEdgesAdjacencyList.insert(std::make_pair >(v2, std::set())); + } + + // Add the edge in the adjacency list + mEdgesAdjacencyList[v1].insert(v2); + mEdgesAdjacencyList[v2].insert(v1); +} + +// Return true if the edges information is used to speed up the collision detection +inline bool ConvexMeshShape::isEdgesInformationUsed() const { + return mIsEdgesInformationUsed; +} + +// Set the variable to know if the edges information is used to speed up the +// collision detection +inline void ConvexMeshShape::setIsEdgesInformationUsed(bool isEdgesUsed) { + mIsEdgesInformationUsed = isEdgesUsed; +} + +} + +#endif diff --git a/src/reactphysics3d.h b/src/reactphysics3d.h index 2bf3885c..ababb9bc 100644 --- a/src/reactphysics3d.h +++ b/src/reactphysics3d.h @@ -48,6 +48,7 @@ #include "collision/shapes/ConeShape.h" #include "collision/shapes/CylinderShape.h" #include "collision/shapes/CapsuleShape.h" +#include "collision/shapes/ConvexMeshShape.h" #include "collision/shapes/AABB.h" #include "constraint/BallAndSocketJoint.h" #include "constraint/SliderJoint.h" From 828af79bcf8dfa3590a9ac709dd2cb672ac86e69 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 16 Jul 2013 22:28:03 +0200 Subject: [PATCH 46/66] Fix issues in the Sweep-And-Prune algorithm (thanks Aleksi for reporting them) and handle the broad-phase world limits in a better way --- .../broadphase/SweepAndPruneAlgorithm.cpp | 133 ++++++++++++------ .../broadphase/SweepAndPruneAlgorithm.h | 31 +++- 2 files changed, 117 insertions(+), 47 deletions(-) diff --git a/src/collision/broadphase/SweepAndPruneAlgorithm.cpp b/src/collision/broadphase/SweepAndPruneAlgorithm.cpp index 4a303a9e..1c363d7a 100644 --- a/src/collision/broadphase/SweepAndPruneAlgorithm.cpp +++ b/src/collision/broadphase/SweepAndPruneAlgorithm.cpp @@ -35,15 +35,30 @@ using namespace reactphysics3d; using namespace std; // Initialization of static variables -bodyindex SweepAndPruneAlgorithm::INVALID_INDEX = std::numeric_limits::max(); +const bodyindex SweepAndPruneAlgorithm::INVALID_INDEX = std::numeric_limits::max(); +const luint SweepAndPruneAlgorithm::NB_SENTINELS = 2; -// Constructor of AABBInt +// Constructor that takes an AABB as input AABBInt::AABBInt(const AABB& aabb) { - for (int axis=0; axis<3; axis++) { - min[axis] = encodeFloatIntoInteger(aabb.getMin()[axis]); - max[axis] = encodeFloatIntoInteger(aabb.getMax()[axis]); - } -} + min[0] = encodeFloatIntoInteger(aabb.getMin().x); + min[1] = encodeFloatIntoInteger(aabb.getMin().y); + min[2] = encodeFloatIntoInteger(aabb.getMin().z); + + max[0] = encodeFloatIntoInteger(aabb.getMax().x); + max[1] = encodeFloatIntoInteger(aabb.getMax().y); + max[2] = encodeFloatIntoInteger(aabb.getMax().z); +} + +// Constructor that set all the axis with an minimum and maximum value +AABBInt::AABBInt(uint minValue, uint maxValue) { + min[0] = minValue; + min[1] = minValue; + min[2] = minValue; + + max[0] = maxValue; + max[1] = maxValue; + max[2] = maxValue; +} // Constructor SweepAndPruneAlgorithm::SweepAndPruneAlgorithm(CollisionDetection& collisionDetection) @@ -67,8 +82,16 @@ SweepAndPruneAlgorithm::~SweepAndPruneAlgorithm() { // Notify the broad-phase about a new object in the world /// This method adds the AABB of the object ion to broad-phase void SweepAndPruneAlgorithm::addObject(CollisionBody* body, const AABB& aabb) { + bodyindex boxIndex; - + + assert(encodeFloatIntoInteger(aabb.getMin().x) > encodeFloatIntoInteger(-FLT_MAX)); + assert(encodeFloatIntoInteger(aabb.getMin().y) > encodeFloatIntoInteger(-FLT_MAX)); + assert(encodeFloatIntoInteger(aabb.getMin().z) > encodeFloatIntoInteger(-FLT_MAX)); + assert(encodeFloatIntoInteger(aabb.getMax().x) < encodeFloatIntoInteger(FLT_MAX)); + assert(encodeFloatIntoInteger(aabb.getMax().y) < encodeFloatIntoInteger(FLT_MAX)); + assert(encodeFloatIntoInteger(aabb.getMax().z) < encodeFloatIntoInteger(FLT_MAX)); + // If the index of the first free box is valid (means that // there is a bucket in the middle of the array that doesn't // contain a box anymore because it has been removed) @@ -88,12 +111,11 @@ void SweepAndPruneAlgorithm::addObject(CollisionBody* body, const AABB& aabb) { // Move the maximum limit end-point two elements further // at the end-points array in all three axis - const luint nbSentinels = 2; - const luint indexLimitEndPoint = 2 * mNbBoxes + nbSentinels - 1; + const luint indexLimitEndPoint = 2 * mNbBoxes + NB_SENTINELS - 1; for (uint axis=0; axis<3; axis++) { EndPoint* maxLimitEndPoint = &mEndPoints[axis][indexLimitEndPoint]; - assert(mEndPoints[axis][0].boxID == INVALID_INDEX && mEndPoints[axis][0].isMin == true); - assert(maxLimitEndPoint->boxID == INVALID_INDEX && maxLimitEndPoint->isMin == false); + assert(mEndPoints[axis][0].boxID == INVALID_INDEX && mEndPoints[axis][0].isMin); + assert(maxLimitEndPoint->boxID == INVALID_INDEX && !maxLimitEndPoint->isMin); EndPoint* newMaxLimitEndPoint = &mEndPoints[axis][indexLimitEndPoint + 2]; newMaxLimitEndPoint->setValues(maxLimitEndPoint->boxID, maxLimitEndPoint->isMin, maxLimitEndPoint->value); @@ -102,15 +124,15 @@ void SweepAndPruneAlgorithm::addObject(CollisionBody* body, const AABB& aabb) { // Create a new box BoxAABB* box = &mBoxes[boxIndex]; box->body = body; - const uint minEndPointValue = encodeFloatIntoInteger(FLT_MAX - 2.0f); - const uint maxEndPointValue = encodeFloatIntoInteger(FLT_MAX - 1.0f); + const uint maxEndPointValue = encodeFloatIntoInteger(FLT_MAX) - 1; + const uint minEndPointValue = encodeFloatIntoInteger(FLT_MAX) - 2; for (uint axis=0; axis<3; axis++) { box->min[axis] = indexLimitEndPoint; box->max[axis] = indexLimitEndPoint + 1; EndPoint* minimumEndPoint = &mEndPoints[axis][box->min[axis]]; - minimumEndPoint->setValues(body->getID(), true, minEndPointValue); + minimumEndPoint->setValues(boxIndex, true, minEndPointValue); EndPoint* maximumEndPoint = &mEndPoints[axis][box->max[axis]]; - maximumEndPoint->setValues(body->getID(), false, maxEndPointValue); + maximumEndPoint->setValues(boxIndex, false, maxEndPointValue); } // Add the body pointer to box index mapping @@ -125,19 +147,36 @@ void SweepAndPruneAlgorithm::addObject(CollisionBody* body, const AABB& aabb) { updateObject(body, aabb); } -// Notify the broad-phase about a object that has been removed from the world +// Notify the broad-phase about an object that has been removed from the world void SweepAndPruneAlgorithm::removeObject(CollisionBody* body) { + assert(mNbBoxes > 0); + // Call the update method with an AABB that is very far away // in order to remove all overlapping pairs from the pair manager - const decimal max = DECIMAL_LARGEST; - const Vector3 maxVector(max, max, max); - const AABB aabb(maxVector, maxVector); - updateObject(body, aabb); + const uint maxEndPointValue = encodeFloatIntoInteger(FLT_MAX) - 1; + const uint minEndPointValue = encodeFloatIntoInteger(FLT_MAX) - 2; + const AABBInt aabbInt(minEndPointValue, maxEndPointValue); + updateObjectIntegerAABB(body, aabbInt); // Get the corresponding box bodyindex boxIndex = mMapBodyToBoxIndex.find(body)->second; - BoxAABB* box = &mBoxes[boxIndex]; + + // Remove the end-points of the box by moving the maximum end-points two elements back in + // the end-points array + const luint indexLimitEndPoint = 2 * mNbBoxes + NB_SENTINELS - 1; + for (uint axis=0; axis < 3; axis++) { + EndPoint* maxLimitEndPoint = &mEndPoints[axis][indexLimitEndPoint]; + assert(mEndPoints[axis][0].boxID == INVALID_INDEX && mEndPoints[axis][0].isMin); + assert(maxLimitEndPoint->boxID == INVALID_INDEX && !maxLimitEndPoint->isMin); + EndPoint* newMaxLimitEndPoint = &mEndPoints[axis][indexLimitEndPoint - 2]; + assert(mEndPoints[axis][indexLimitEndPoint - 1].boxID == boxIndex); + assert(!mEndPoints[axis][indexLimitEndPoint - 1].isMin); + assert(newMaxLimitEndPoint->boxID == boxIndex); + assert(newMaxLimitEndPoint->isMin); + newMaxLimitEndPoint->setValues(maxLimitEndPoint->boxID, maxLimitEndPoint->isMin, + maxLimitEndPoint->value); + } // Add the box index into the list of free indices mFreeBoxIndices.push_back(boxIndex); @@ -146,11 +185,16 @@ void SweepAndPruneAlgorithm::removeObject(CollisionBody* body) { mNbBoxes--; } -// Notify the broad-phase that the AABB of an object has changed -void SweepAndPruneAlgorithm::updateObject(CollisionBody* body, const AABB& aabb) { - - // Compute the AABB with integer coordinates - AABBInt aabbInt(aabb); +// Notify the broad-phase that the AABB of an object has changed. +/// The input is an AABB with integer coordinates +void SweepAndPruneAlgorithm::updateObjectIntegerAABB(CollisionBody* body, const AABBInt& aabbInt) { + + assert(aabbInt.min[0] > encodeFloatIntoInteger(-FLT_MAX)); + assert(aabbInt.min[1] > encodeFloatIntoInteger(-FLT_MAX)); + assert(aabbInt.min[2] > encodeFloatIntoInteger(-FLT_MAX)); + assert(aabbInt.max[0] < encodeFloatIntoInteger(FLT_MAX)); + assert(aabbInt.max[1] < encodeFloatIntoInteger(FLT_MAX)); + assert(aabbInt.max[2] < encodeFloatIntoInteger(FLT_MAX)); // Get the corresponding box bodyindex boxIndex = mMapBodyToBoxIndex.find(body)->second; @@ -182,7 +226,8 @@ void SweepAndPruneAlgorithm::updateObject(CollisionBody* body, const AABB& aabb) // The minimum end-point is moving left EndPoint savedEndPoint = *currentMinEndPoint; - luint indexEndPoint = (size_t(currentMinEndPoint) - size_t(startEndPointsCurrentAxis)) / sizeof(EndPoint); + luint indexEndPoint = (size_t(currentMinEndPoint) - + size_t(startEndPointsCurrentAxis)) / sizeof(EndPoint); const luint savedEndPointIndex = indexEndPoint; while ((--currentMinEndPoint)->value > limit) { @@ -197,7 +242,8 @@ void SweepAndPruneAlgorithm::updateObject(CollisionBody* body, const AABB& aabb) // for box intersection if (box != id1) { if (testIntersect2D(*box, *id1, otherAxis1, otherAxis2) && - testIntersect1DSortedAABBs(*id1, aabbInt, startEndPointsCurrentAxis, axis)) { + testIntersect1DSortedAABBs(*id1, aabbInt, + startEndPointsCurrentAxis, axis)) { // Add an overlapping pair to the pair manager mPairManager.addPair(body, id1->body); @@ -225,13 +271,14 @@ void SweepAndPruneAlgorithm::updateObject(CollisionBody* body, const AABB& aabb) startEndPointsCurrentAxis[indexEndPoint] = savedEndPoint; } } - else if (limit > currentMinEndPoint->value) { // The minimum of the box has moved to the right + else if (limit > currentMinEndPoint->value){// The minimum of the box has moved to the right currentMinEndPoint->value = limit; // The minimum en-point is moving right EndPoint savedEndPoint = *currentMinEndPoint; - luint indexEndPoint = (size_t(currentMinEndPoint) -size_t(startEndPointsCurrentAxis)) / sizeof(EndPoint); + luint indexEndPoint = (size_t(currentMinEndPoint) - + size_t(startEndPointsCurrentAxis)) / sizeof(EndPoint); const luint savedEndPointIndex = indexEndPoint; // For each end-point between the current position of the minimum @@ -291,7 +338,8 @@ void SweepAndPruneAlgorithm::updateObject(CollisionBody* body, const AABB& aabb) currentMaxEndPoint->value = limit; EndPoint savedEndPoint = *currentMaxEndPoint; - luint indexEndPoint = (size_t(currentMaxEndPoint) -size_t(startEndPointsCurrentAxis)) / sizeof(EndPoint); + luint indexEndPoint = (size_t(currentMaxEndPoint) - + size_t(startEndPointsCurrentAxis)) / sizeof(EndPoint); const luint savedEndPointIndex = indexEndPoint; while ((++currentMaxEndPoint)->value < limit) { @@ -308,7 +356,8 @@ void SweepAndPruneAlgorithm::updateObject(CollisionBody* body, const AABB& aabb) // for box intersection if (box != id1) { if (testIntersect2D(*box, *id1, otherAxis1, otherAxis2) && - testIntersect1DSortedAABBs(*id1, aabbInt, startEndPointsCurrentAxis,axis)) { + testIntersect1DSortedAABBs(*id1, aabbInt, + startEndPointsCurrentAxis,axis)) { // Add an overlapping pair to the pair manager mPairManager.addPair(body, id1->body); @@ -340,7 +389,8 @@ void SweepAndPruneAlgorithm::updateObject(CollisionBody* body, const AABB& aabb) currentMaxEndPoint->value = limit; EndPoint savedEndPoint = *currentMaxEndPoint; - luint indexEndPoint = (size_t(currentMaxEndPoint) -size_t(startEndPointsCurrentAxis)) / sizeof(EndPoint); + luint indexEndPoint = (size_t(currentMaxEndPoint) - + size_t(startEndPointsCurrentAxis)) / sizeof(EndPoint); const luint savedEndPointIndex = indexEndPoint; // For each end-point between the current position of the maximum @@ -390,10 +440,9 @@ void SweepAndPruneAlgorithm::updateObject(CollisionBody* body, const AABB& aabb) void SweepAndPruneAlgorithm::resizeArrays() { // New number of boxes in the array - const luint nbSentinels = 2; const luint newNbMaxBoxes = mNbMaxBoxes ? 2 * mNbMaxBoxes : 100; - const luint nbEndPoints = mNbBoxes * 2 + nbSentinels; - const luint newNbEndPoints = newNbMaxBoxes * 2 + nbSentinels; + const luint nbEndPoints = mNbBoxes * 2 + NB_SENTINELS; + const luint newNbEndPoints = newNbMaxBoxes * 2 + NB_SENTINELS; // Allocate memory for the new boxes and end-points arrays BoxAABB* newBoxesArray = new BoxAABB[newNbMaxBoxes]; @@ -409,18 +458,18 @@ void SweepAndPruneAlgorithm::resizeArrays() { // If the arrays were not empty before if (mNbBoxes > 0) { - // Copy the data in the old arrays into the new one + // Copy the data from the old arrays into the new one memcpy(newBoxesArray, mBoxes, sizeof(BoxAABB) * mNbBoxes); - size_t nbBytesNewEndPoints = sizeof(EndPoint) * nbEndPoints; + const size_t nbBytesNewEndPoints = sizeof(EndPoint) * nbEndPoints; memcpy(newEndPointsXArray, mEndPoints[0], nbBytesNewEndPoints); memcpy(newEndPointsYArray, mEndPoints[1], nbBytesNewEndPoints); memcpy(newEndPointsZArray, mEndPoints[2], nbBytesNewEndPoints); } else { // If the arrays were empty - + // Add the limits endpoints (sentinels) into the array - const uint min = encodeFloatIntoInteger(DECIMAL_SMALLEST); - const uint max = encodeFloatIntoInteger(DECIMAL_LARGEST); + const uint min = encodeFloatIntoInteger(-FLT_MAX); + const uint max = encodeFloatIntoInteger(FLT_MAX); newEndPointsXArray[0].setValues(INVALID_INDEX, true, min); newEndPointsXArray[1].setValues(INVALID_INDEX, false, max); newEndPointsYArray[0].setValues(INVALID_INDEX, true, min); diff --git a/src/collision/broadphase/SweepAndPruneAlgorithm.h b/src/collision/broadphase/SweepAndPruneAlgorithm.h index 25051509..2c72152b 100644 --- a/src/collision/broadphase/SweepAndPruneAlgorithm.h +++ b/src/collision/broadphase/SweepAndPruneAlgorithm.h @@ -35,7 +35,7 @@ /// Namespace ReactPhysics3D namespace reactphysics3d { - + // Structure EndPoint /** * EndPoint structure that represent an end-point of an AABB @@ -94,8 +94,11 @@ struct AABBInt { /// Maximum values on the three axis uint max[3]; - /// Constructor + /// Constructor that takes an AABB as input AABBInt(const AABB& aabb); + + /// Constructor that set all the axis with an minimum and maximum value + AABBInt(uint minValue, uint maxValue); }; // Class SweepAndPruneAlgorithm @@ -109,10 +112,15 @@ class SweepAndPruneAlgorithm : public BroadPhaseAlgorithm { protected : - // -------------------- Attributes -------------------- // + // -------------------- Constants -------------------- // /// Invalid array index - static bodyindex INVALID_INDEX; + const static bodyindex INVALID_INDEX; + + /// Number of sentinel end-points in the array of a given axis + const static luint NB_SENTINELS; + + // -------------------- Attributes -------------------- // /// Array that contains all the AABB boxes of the broad-phase BoxAABB* mBoxes; @@ -153,6 +161,9 @@ class SweepAndPruneAlgorithm : public BroadPhaseAlgorithm { /// Check for 2D box intersection. bool testIntersect2D(const BoxAABB& box1, const BoxAABB& box2, luint axis1, uint axis2) const; + + /// Notify the broad-phase that the AABB of an object has changed. + void updateObjectIntegerAABB(CollisionBody* body, const AABBInt& aabbInt); public : @@ -211,7 +222,17 @@ inline bool SweepAndPruneAlgorithm::testIntersect2D(const BoxAABB& box1, const B luint axis1, uint axis2) const { return !(box2.max[axis1] < box1.min[axis1] || box1.max[axis1] < box2.min[axis1] || box2.max[axis2] < box1.min[axis2] || box1.max[axis2] < box2.min[axis2]); -} +} + +// Notify the broad-phase that the AABB of an object has changed +inline void SweepAndPruneAlgorithm::updateObject(CollisionBody* body, const AABB& aabb) { + + // Compute the corresponding AABB with integer coordinates + AABBInt aabbInt(aabb); + + // Call the update object method that uses an AABB with integer coordinates + updateObjectIntegerAABB(body, aabbInt); +} } From d2f7f6e28c09339436c4aba05180343ff6794c29 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 17 Jul 2013 00:09:15 +0200 Subject: [PATCH 47/66] Release memory in the Sweep-And-Prune algorithm when several objects are removed from the world --- src/collision/broadphase/PairManager.h | 8 +- .../broadphase/SweepAndPruneAlgorithm.cpp | 102 ++++++++++++++++++ .../broadphase/SweepAndPruneAlgorithm.h | 6 +- 3 files changed, 111 insertions(+), 5 deletions(-) diff --git a/src/collision/broadphase/PairManager.h b/src/collision/broadphase/PairManager.h index 860ab19a..336df403 100644 --- a/src/collision/broadphase/PairManager.h +++ b/src/collision/broadphase/PairManager.h @@ -127,9 +127,6 @@ class PairManager { /// This method returns an hash value for a 32 bits key. int computeHash32Bits(int key) const; - /// Return the next power of two - luint computeNextPowerOfTwo(luint number) const; - /// Reallocate memory for more pairs void reallocatePairs(); @@ -171,6 +168,9 @@ class PairManager { /// Find a pair given two body IDs BodyPair* findPair(bodyindex id1, bodyindex id2) const; + /// Return the next power of two + static luint computeNextPowerOfTwo(luint number); + /// Return a pointer to the first overlapping pair (used to /// iterate over the active pairs). BodyPair* beginOverlappingPairsPointer() const; @@ -215,7 +215,7 @@ inline bool PairManager::isDifferentPair(const BodyPair& pair1, bodyindex pair2I } // Return the next power of two of a 32bits integer using a SWAR algorithm -inline luint PairManager::computeNextPowerOfTwo(luint number) const { +inline luint PairManager::computeNextPowerOfTwo(luint number) { number |= (number >> 1); number |= (number >> 2); number |= (number >> 4); diff --git a/src/collision/broadphase/SweepAndPruneAlgorithm.cpp b/src/collision/broadphase/SweepAndPruneAlgorithm.cpp index 1c363d7a..142ec56f 100644 --- a/src/collision/broadphase/SweepAndPruneAlgorithm.cpp +++ b/src/collision/broadphase/SweepAndPruneAlgorithm.cpp @@ -183,6 +183,12 @@ void SweepAndPruneAlgorithm::removeObject(CollisionBody* body) { mMapBodyToBoxIndex.erase(body); mNbBoxes--; + + // Check if we need to shrink the allocated memory + const luint nextPowerOf2 = PairManager::computeNextPowerOfTwo((mNbBoxes-1) / 100 ); + if (nextPowerOf2 * 100 < mNbMaxBoxes) { + shrinkArrays(); + } } // Notify the broad-phase that the AABB of an object has changed. @@ -492,3 +498,99 @@ void SweepAndPruneAlgorithm::resizeArrays() { mNbMaxBoxes = newNbMaxBoxes; } + +// Shrink the boxes and end-points arrays when too much memory is allocated +void SweepAndPruneAlgorithm::shrinkArrays() { + + // New number of boxes and end-points in the array + const luint nextPowerOf2 = PairManager::computeNextPowerOfTwo((mNbBoxes-1) / 100 ); + const luint newNbMaxBoxes = (mNbBoxes > 100) ? nextPowerOf2 * 100 : 100; + const luint nbEndPoints = mNbBoxes * 2 + NB_SENTINELS; + const luint newNbEndPoints = newNbMaxBoxes * 2 + NB_SENTINELS; + + assert(newNbMaxBoxes < mNbMaxBoxes); + + // Sort the list of the free boxes indices in ascending order + mFreeBoxIndices.sort(); + + // Reorganize the boxes inside the boxes array so that all the boxes are at the + // beginning of the array + std::map newMapBodyToBoxIndex; + std::map::const_iterator it; + for (it = mMapBodyToBoxIndex.begin(); it != mMapBodyToBoxIndex.end(); ++it) { + + CollisionBody* body = it->first; + bodyindex boxIndex = it->second; + + // If the box index is outside the range of the current number of boxes + if (boxIndex >= mNbBoxes) { + + assert(!mFreeBoxIndices.empty()); + + // Get a new box index for that body (from the list of free box indices) + bodyindex newBoxIndex = mFreeBoxIndices.front(); + mFreeBoxIndices.pop_front(); + assert(newBoxIndex < mNbBoxes); + + // Copy the box to its new location in the boxes array + BoxAABB* oldBox = &mBoxes[boxIndex]; + BoxAABB* newBox = &mBoxes[newBoxIndex]; + assert(oldBox->body->getID() == body->getID()); + newBox->body = oldBox->body; + for (uint axis=0; axis<3; axis++) { + + // Copy the minimum and maximum end-points indices + newBox->min[axis] = oldBox->min[axis]; + newBox->max[axis] = oldBox->max[axis]; + + // Update the box index of the end-points + EndPoint* minimumEndPoint = &mEndPoints[axis][newBox->min[axis]]; + EndPoint* maximumEndPoint = &mEndPoints[axis][newBox->max[axis]]; + assert(minimumEndPoint->boxID == boxIndex); + assert(maximumEndPoint->boxID == boxIndex); + minimumEndPoint->boxID = newBoxIndex; + maximumEndPoint->boxID = newBoxIndex; + } + + newMapBodyToBoxIndex.insert(pair(body, newBoxIndex)); + } + else { + newMapBodyToBoxIndex.insert(pair(body, boxIndex)); + } + } + + assert(newMapBodyToBoxIndex.size() == mMapBodyToBoxIndex.size()); + mMapBodyToBoxIndex = newMapBodyToBoxIndex; + + // Allocate memory for the new boxes and end-points arrays + BoxAABB* newBoxesArray = new BoxAABB[newNbMaxBoxes]; + EndPoint* newEndPointsXArray = new EndPoint[newNbEndPoints]; + EndPoint* newEndPointsYArray = new EndPoint[newNbEndPoints]; + EndPoint* newEndPointsZArray = new EndPoint[newNbEndPoints]; + + assert(newBoxesArray != NULL); + assert(newEndPointsXArray != NULL); + assert(newEndPointsYArray != NULL); + assert(newEndPointsZArray != NULL); + + // Copy the data from the old arrays into the new one + memcpy(newBoxesArray, mBoxes, sizeof(BoxAABB) * mNbBoxes); + const size_t nbBytesNewEndPoints = sizeof(EndPoint) * nbEndPoints; + memcpy(newEndPointsXArray, mEndPoints[0], nbBytesNewEndPoints); + memcpy(newEndPointsYArray, mEndPoints[1], nbBytesNewEndPoints); + memcpy(newEndPointsZArray, mEndPoints[2], nbBytesNewEndPoints); + + // Delete the old arrays + delete[] mBoxes; + delete[] mEndPoints[0]; + delete[] mEndPoints[1]; + delete[] mEndPoints[2]; + + // Assign the pointer to the new arrays + mBoxes = newBoxesArray; + mEndPoints[0] = newEndPointsXArray; + mEndPoints[1] = newEndPointsYArray; + mEndPoints[2] = newEndPointsZArray; + + mNbMaxBoxes = newNbMaxBoxes; +} diff --git a/src/collision/broadphase/SweepAndPruneAlgorithm.h b/src/collision/broadphase/SweepAndPruneAlgorithm.h index 2c72152b..a4e2594b 100644 --- a/src/collision/broadphase/SweepAndPruneAlgorithm.h +++ b/src/collision/broadphase/SweepAndPruneAlgorithm.h @@ -31,6 +31,7 @@ #include "../../collision/shapes/AABB.h" #include #include +#include /// Namespace ReactPhysics3D @@ -135,7 +136,7 @@ class SweepAndPruneAlgorithm : public BroadPhaseAlgorithm { bodyindex mNbMaxBoxes; /// Indices that are not used by any boxes - std::vector mFreeBoxIndices; + std::list mFreeBoxIndices; /// Map a body pointer to a box index std::map mMapBodyToBoxIndex; @@ -151,6 +152,9 @@ class SweepAndPruneAlgorithm : public BroadPhaseAlgorithm { /// Resize the boxes and end-points arrays when it's full void resizeArrays(); + /// Shrink the boxes and end-points arrays when too much memory is allocated + void shrinkArrays(); + /// Add an overlapping pair of AABBS void addPair(CollisionBody* body1, CollisionBody* body2); From 2e6f571b98f6247cd007ca265d16c1b728897fbc Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 6 Aug 2013 20:51:56 +0200 Subject: [PATCH 48/66] Implement linear and angular velocity damping --- examples/fallingcubes/Scene.cpp | 6 ++--- examples/joints/Scene.cpp | 38 ++++++++++++++++--------------- src/body/RigidBody.cpp | 7 +++--- src/body/RigidBody.h | 40 +++++++++++++++++++++++++++++++++ src/engine/DynamicsWorld.cpp | 22 ++++++++++++++++++ 5 files changed, 89 insertions(+), 24 deletions(-) diff --git a/examples/fallingcubes/Scene.cpp b/examples/fallingcubes/Scene.cpp index 360a0cb9..dc2082cc 100644 --- a/examples/fallingcubes/Scene.cpp +++ b/examples/fallingcubes/Scene.cpp @@ -45,7 +45,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), mViewer->setScenePosition(center, radiusScene); // Gravity vector in the dynamics world - rp3d::Vector3 gravity(0, -9.81, 0); + rp3d::Vector3 gravity(0, rp3d::decimal(-9.81), 0); // Time step for the physics simulation rp3d::decimal timeStep = 1.0f / 60.0f; @@ -74,7 +74,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), // Change the material properties of the rigid body rp3d::Material& material = cube->getRigidBody()->getMaterial(); - material.setBounciness(0.4); + material.setBounciness(rp3d::decimal(0.4)); // Add the box the list of box in the scene mBoxes.push_back(cube); @@ -89,7 +89,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), // Change the material properties of the floor rigid body rp3d::Material& material = mFloor->getRigidBody()->getMaterial(); - material.setBounciness(0.3); + material.setBounciness(rp3d::decimal(0.3)); // Start the simulation startSimulation(); diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index 44211770..8e5e4e5c 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -34,7 +34,6 @@ using namespace openglframework; Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), mPhongShader("shaders/phong.vert", "shaders/phong.frag"), mIsRunning(false) { - // Move the light 0 mLight0.translateWorld(Vector3(7, 15, 15)); @@ -46,7 +45,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), mViewer->setScenePosition(center, radiusScene); // Gravity vector in the dynamics world - rp3d::Vector3 gravity(0, -9.81, 0); + rp3d::Vector3 gravity(0, rp3d::decimal(-9.81), 0); // Time step for the physics simulation rp3d::decimal timeStep = 1.0f / 60.0f; @@ -129,7 +128,7 @@ void Scene::simulate() { // Update the motor speed of the Slider Joint (to move up and down) long double motorSpeed = 3 * cos(mDynamicsWorld->getPhysicsTime() * 1.5); - mSliderJoint->setMotorSpeed(motorSpeed); + mSliderJoint->setMotorSpeed(rp3d::decimal(motorSpeed)); // Take a simulation step mDynamicsWorld->update(); @@ -209,11 +208,14 @@ void Scene::createBallAndSocketJoints() { if (i == 0) mBallAndSocketJointChainBoxes[i]->getRigidBody()->setIsMotionEnabled(false); else mBallAndSocketJointChainBoxes[i]->getRigidBody()->setIsMotionEnabled(true); + // Add some angular velocity damping + mBallAndSocketJointChainBoxes[i]->getRigidBody()->setAngularDamping(rp3d::decimal(0.2)); + // Change the material properties of the rigid body rp3d::Material& material = mBallAndSocketJointChainBoxes[i]->getRigidBody()->getMaterial(); - material.setBounciness(0.4); + material.setBounciness(rp3d::decimal(0.4)); - positionBox.y -= boxDimension.y + 0.5; + positionBox.y -= boxDimension.y + 0.5f; } // --------------- Create the joints --------------- // @@ -240,7 +242,7 @@ void Scene::createSliderJoint() { // --------------- Create the first box --------------- // // Position of the box - openglframework::Vector3 positionBox1(0, 2.1, 0); + openglframework::Vector3 positionBox1(0, 2.1f, 0); // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 box1Dimension(2, 4, 2); @@ -251,15 +253,15 @@ void Scene::createSliderJoint() { // Change the material properties of the rigid body rp3d::Material& material1 = mSliderJointBottomBox->getRigidBody()->getMaterial(); - material1.setBounciness(0.4); + material1.setBounciness(0.4f); // --------------- Create the second box --------------- // // Position of the box - openglframework::Vector3 positionBox2(0, 4.2, 0); + openglframework::Vector3 positionBox2(0, 4.2f, 0); // Create a box and a corresponding rigid in the dynamics world - openglframework::Vector3 box2Dimension(1.5, 4, 1.5); + openglframework::Vector3 box2Dimension(1.5f, 4, 1.5f); mSliderJointTopBox = new Box(box2Dimension, positionBox2 , CUBE_MASS, mDynamicsWorld); // The second box is allowed to move @@ -267,7 +269,7 @@ void Scene::createSliderJoint() { // Change the material properties of the rigid body rp3d::Material& material2 = mSliderJointTopBox->getRigidBody()->getMaterial(); - material2.setBounciness(0.4); + material2.setBounciness(0.4f); // --------------- Create the joint --------------- // @@ -276,10 +278,10 @@ void Scene::createSliderJoint() { rp3d::RigidBody* body2 = mSliderJointTopBox->getRigidBody(); const rp3d::Vector3& body1Position = body1->getTransform().getPosition(); const rp3d::Vector3& body2Position = body2->getTransform().getPosition(); - const rp3d::Vector3 anchorPointWorldSpace = 0.5 * (body2Position + body1Position); + const rp3d::Vector3 anchorPointWorldSpace = rp3d::decimal(0.5) * (body2Position + body1Position); const rp3d::Vector3 sliderAxisWorldSpace = (body2Position - body1Position); rp3d::SliderJointInfo jointInfo(body1, body2, anchorPointWorldSpace, sliderAxisWorldSpace, - -1.7, 1.7); + rp3d::decimal(-1.7), rp3d::decimal(1.7)); jointInfo.isMotorEnabled = true; jointInfo.motorSpeed = 0.0; jointInfo.maxMotorForce = 10000.0; @@ -306,7 +308,7 @@ void Scene::createPropellerHingeJoint() { // Change the material properties of the rigid body rp3d::Material& material = mPropellerBox->getRigidBody()->getMaterial(); - material.setBounciness(0.4); + material.setBounciness(rp3d::decimal(0.4)); // --------------- Create the Hinge joint --------------- // @@ -319,8 +321,8 @@ void Scene::createPropellerHingeJoint() { const rp3d::Vector3 hingeAxisWorldSpace(0, 1, 0); rp3d::HingeJointInfo jointInfo(body1, body2, anchorPointWorldSpace, hingeAxisWorldSpace); jointInfo.isMotorEnabled = true; - jointInfo.motorSpeed = -0.5 * PI; - jointInfo.maxMotorTorque = 60.0; + jointInfo.motorSpeed = - rp3d::decimal(0.5) * PI; + jointInfo.maxMotorTorque = rp3d::decimal(60.0); jointInfo.isCollisionEnabled = false; // Create the joint in the dynamics world @@ -344,7 +346,7 @@ void Scene::createFixedJoints() { // Change the material properties of the rigid body rp3d::Material& material1 = mFixedJointBox1->getRigidBody()->getMaterial(); - material1.setBounciness(0.4); + material1.setBounciness(rp3d::decimal(0.4)); // --------------- Create the second box --------------- // @@ -359,7 +361,7 @@ void Scene::createFixedJoints() { // Change the material properties of the rigid body rp3d::Material& material2 = mFixedJointBox2->getRigidBody()->getMaterial(); - material2.setBounciness(0.4); + material2.setBounciness(rp3d::decimal(0.4)); // --------------- Create the first fixed joint --------------- // @@ -398,5 +400,5 @@ void Scene::createFloor() { // Change the material properties of the rigid body rp3d::Material& material = mFloor->getRigidBody()->getMaterial(); - material.setBounciness(0.3); + material.setBounciness(rp3d::decimal(0.3)); } diff --git a/src/body/RigidBody.cpp b/src/body/RigidBody.cpp index e18ac82a..757af86d 100644 --- a/src/body/RigidBody.cpp +++ b/src/body/RigidBody.cpp @@ -30,12 +30,13 @@ // We want to use the ReactPhysics3D namespace using namespace reactphysics3d; - // Constructor - RigidBody::RigidBody(const Transform& transform, decimal mass, const Matrix3x3& inertiaTensorLocal, +// Constructor +RigidBody::RigidBody(const Transform& transform, decimal mass, const Matrix3x3& inertiaTensorLocal, CollisionShape *collisionShape, bodyindex id) : CollisionBody(transform, collisionShape, id), mInertiaTensorLocal(inertiaTensorLocal), mMass(mass), mInertiaTensorLocalInverse(inertiaTensorLocal.getInverse()), - mMassInverse(decimal(1.0) / mass), mIsGravityEnabled(true) { + mMassInverse(decimal(1.0) / mass), mIsGravityEnabled(true), + mLinearDamping(decimal(0.0)), mAngularDamping(decimal(0.0)) { assert(collisionShape); } diff --git a/src/body/RigidBody.h b/src/body/RigidBody.h index 4af18f37..6136f1dd 100644 --- a/src/body/RigidBody.h +++ b/src/body/RigidBody.h @@ -80,6 +80,12 @@ class RigidBody : public CollisionBody { /// Material properties of the rigid body Material mMaterial; + /// Linear velocity damping factor + decimal mLinearDamping; + + /// Angular velocity damping factor + decimal mAngularDamping; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -161,6 +167,18 @@ class RigidBody : public CollisionBody { /// Set a new material for this rigid body void setMaterial(const Material& material); + + /// Return the linear velocity damping factor + decimal getLinearDamping() const; + + /// Set the linear damping factor + void setLinearDamping(decimal linearDamping); + + /// Return the angular velocity damping factor + decimal getAngularDamping() const; + + /// Set the angular damping factor + void setAngularDamping(decimal angularDamping); }; // Method that return the mass of the body @@ -288,6 +306,28 @@ inline void RigidBody::setMaterial(const Material& material) { mMaterial = material; } +// Return the linear velocity damping factor +inline decimal RigidBody::getLinearDamping() const { + return mLinearDamping; +} + +// Set the linear damping factor +inline void RigidBody::setLinearDamping(decimal linearDamping) { + assert(linearDamping >= decimal(0.0)); + mLinearDamping = linearDamping; +} + +// Return the angular velocity damping factor +inline decimal RigidBody::getAngularDamping() const { + return mAngularDamping; +} + +// Set the angular damping factor +inline void RigidBody::setAngularDamping(decimal angularDamping) { + assert(angularDamping >= decimal(0.0)); + mAngularDamping = angularDamping; +} + } #endif diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 27a8fc12..605aec54 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -258,6 +258,28 @@ void DynamicsWorld::integrateRigidBodiesVelocities() { rigidBody->getMass() * mGravity; } + // Apply the velocity damping + // Damping force : F_c = -c' * v (c=damping factor) + // Equation : m * dv/dt = -c' * v + // => dv/dt = -c * v (with c=c'/m) + // => dv/dt + c * v = 0 + // Solution : v(t) = v0 * e^(-c * t) + // => v(t + dt) = v0 * e^(-c(t + dt)) + // = v0 * e^(-ct) * e^(-c * dt) + // = v(t) * e^(-c * dt) + // => v2 = v1 * e^(-c * dt) + // Using Taylor Serie for e^(-x) : e^x ~ 1 + x + x^2/2! + ... + // => e^(-x) ~ 1 - x + // => v2 = v1 * (1 - c * dt) + decimal linDampingFactor = rigidBody->getLinearDamping(); + decimal angDampingFactor = rigidBody->getAngularDamping(); + decimal linearDamping = clamp(decimal(1.0) - dt * linDampingFactor, + decimal(0.0), decimal(1.0)); + decimal angularDamping = clamp(decimal(1.0) - dt * angDampingFactor, + decimal(0.0), decimal(1.0)); + mConstrainedLinearVelocities[i] *= clamp(linearDamping, decimal(0.0), decimal(1.0)); + mConstrainedAngularVelocities[i] *= clamp(angularDamping, decimal(0.0), decimal(1.0)); + // Update the old Transform of the body rigidBody->updateOldTransform(); } From f1d29b5123f649aae8bb042b13383cec04ae93cb Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Mon, 26 Aug 2013 21:28:48 +0200 Subject: [PATCH 49/66] Implement the islands computation --- src/body/Body.cpp | 5 +- src/body/Body.h | 61 ++++++++ src/body/CollisionBody.cpp | 24 ++- src/body/CollisionBody.h | 22 +++ src/body/RigidBody.cpp | 33 +++- src/body/RigidBody.h | 23 +++ src/configuration.h | 4 +- src/constraint/Constraint.cpp | 2 +- src/constraint/Constraint.h | 47 +++++- src/constraint/HingeJoint.cpp | 2 +- src/engine/ContactManifold.cpp | 28 ++-- src/engine/ContactManifold.h | 75 ++++++++- src/engine/DynamicsWorld.cpp | 267 ++++++++++++++++++++++++++++++++- src/engine/DynamicsWorld.h | 35 ++++- src/engine/Island.cpp | 54 +++++++ src/engine/Island.h | 186 +++++++++++++++++++++++ src/engine/OverlappingPair.h | 8 +- src/memory/MemoryAllocator.cpp | 4 +- 18 files changed, 836 insertions(+), 44 deletions(-) create mode 100644 src/engine/Island.cpp create mode 100644 src/engine/Island.h diff --git a/src/body/Body.cpp b/src/body/Body.cpp index a9c639f3..243b0c3f 100644 --- a/src/body/Body.cpp +++ b/src/body/Body.cpp @@ -23,7 +23,7 @@ * * ********************************************************************************/ - // Libraries +// Libraries #include "Body.h" #include "../collision/shapes/CollisionShape.h" @@ -31,7 +31,8 @@ using namespace reactphysics3d; // Constructor -Body::Body(bodyindex id) : mID(id) { +Body::Body(bodyindex id) + : mID(id), mIsAlreadyInIsland(false), mIsAllowedToSleep(true), mIsSleeping(false) { } diff --git a/src/body/Body.h b/src/body/Body.h index d131eec8..f0358570 100644 --- a/src/body/Body.h +++ b/src/body/Body.h @@ -47,6 +47,15 @@ class Body { /// ID of the body bodyindex mID; + /// True if the body has already been added in an island (for sleeping technique) + bool mIsAlreadyInIsland; + + /// True if the body is allowed to go to sleep for better efficiency + bool mIsAllowedToSleep; + + /// True if the body is sleeping (for sleeping technique) + bool mIsSleeping; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -68,6 +77,24 @@ class Body { /// Return the id of the body bodyindex getID() const; + /// Return true if the body has already been added in an island (for the sleeping technique) + bool isAlreadyInIsland() const; + + /// Set the value of to know if the body has already been added into an island + void setIsAlreadyInIsland(bool isAlreadyInIsland); + + /// Return whether or not the body is allowed to sleep + bool isAllowedToSleep() const; + + /// Set whether or not the body is allowed to go to sleep + void setIsAllowedToSleep(bool isAllowedToSleep); + + /// Return whether or not the body is sleeping + bool isSleeping() const; + + /// Set the variable to know whether or not the body is sleeping + void setIsSleeping(bool isSleeping); + /// Smaller than operator bool operator<(const Body& body2) const; @@ -79,6 +106,10 @@ class Body { /// Not equal operator bool operator!=(const Body& body2) const; + + // -------------------- Friendship -------------------- // + + friend class DynamicsWorld; }; // Return the id of the body @@ -86,6 +117,36 @@ inline bodyindex Body::getID() const { return mID; } +// Return true if the body has already been added in an island (for the sleeping technique) +inline bool Body::isAlreadyInIsland() const { + return mIsAlreadyInIsland; +} + +// Set the value of to know if the body has already been added into an island +inline void Body::setIsAlreadyInIsland(bool isAlreadyInIsland) { + mIsAlreadyInIsland = isAlreadyInIsland; +} + +// Return whether or not the body is allowed to sleep +inline bool Body::isAllowedToSleep() const { + return mIsAllowedToSleep; +} + +// Set whether or not the body is allowed to go to sleep +inline void Body::setIsAllowedToSleep(bool isAllowedToSleep) { + mIsAllowedToSleep = isAllowedToSleep; +} + +// Return whether or not the body is sleeping +inline bool Body::isSleeping() const { + return mIsSleeping; +} + +// Set the variable to know whether or not the body is sleeping +inline void Body::setIsSleeping(bool isSleeping) { + mIsSleeping = isSleeping; +} + // Smaller than operator inline bool Body::operator<(const Body& body2) const { return (mID < body2.mID); diff --git a/src/body/CollisionBody.cpp b/src/body/CollisionBody.cpp index 98e6c600..0ba418b4 100644 --- a/src/body/CollisionBody.cpp +++ b/src/body/CollisionBody.cpp @@ -25,6 +25,7 @@ // Libraries #include "CollisionBody.h" +#include "../engine/ContactManifold.h" // We want to use the ReactPhysics3D namespace using namespace reactphysics3d; @@ -33,7 +34,7 @@ using namespace reactphysics3d; CollisionBody::CollisionBody(const Transform& transform, CollisionShape *collisionShape, bodyindex id) : Body(id), mCollisionShape(collisionShape), mTransform(transform), - mIsActive(true), mHasMoved(false) { + mIsActive(true), mHasMoved(false), mContactManifoldsList(NULL) { assert(collisionShape); @@ -50,5 +51,24 @@ CollisionBody::CollisionBody(const Transform& transform, CollisionShape *collisi // Destructor CollisionBody::~CollisionBody() { - + assert(mContactManifoldsList == NULL); +} + +// Reset the contact manifold lists +void CollisionBody::resetContactManifoldsList(MemoryAllocator& memoryAllocator) { + + // Delete the linked list of contact manifolds of that body + ContactManifoldListElement* currentElement = mContactManifoldsList; + while (currentElement != NULL) { + ContactManifoldListElement* nextElement = currentElement->next; + + // Delete the current element + currentElement->ContactManifoldListElement::~ContactManifoldListElement(); + memoryAllocator.release(currentElement, sizeof(ContactManifoldListElement)); + + currentElement = nextElement; + } + mContactManifoldsList = NULL; + + assert(mContactManifoldsList == NULL); } diff --git a/src/body/CollisionBody.h b/src/body/CollisionBody.h index df484e33..e050f8e3 100644 --- a/src/body/CollisionBody.h +++ b/src/body/CollisionBody.h @@ -33,11 +33,15 @@ #include "../mathematics/Transform.h" #include "../collision/shapes/AABB.h" #include "../collision/shapes/CollisionShape.h" +#include "../memory/MemoryAllocator.h" #include "../configuration.h" /// Namespace reactphysics3d namespace reactphysics3d { +// Class declarations +struct ContactManifoldListElement; + // Class CollisionBody /** * This class represents a body that is able to collide with others @@ -76,6 +80,9 @@ class CollisionBody : public Body { /// True if the body has moved during the last frame bool mHasMoved; + /// First element of the linked list of contact manifolds involving this body + ContactManifoldListElement* mContactManifoldsList; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -84,6 +91,9 @@ class CollisionBody : public Body { /// Private assignment operator CollisionBody& operator=(const CollisionBody& body); + /// Reset the contact manifold lists + void resetContactManifoldsList(MemoryAllocator& memoryAllocator); + public : // -------------------- Methods -------------------- // @@ -144,6 +154,13 @@ class CollisionBody : public Body { /// Update the Axis-Aligned Bounding Box coordinates void updateAABB(); + + /// Return the first element of the linked list of contact manifolds involving this body + const ContactManifoldListElement* getContactManifoldsLists() const; + + // -------------------- Friendship -------------------- // + + friend class DynamicsWorld; }; // Return true if the body has moved during the last frame @@ -243,6 +260,11 @@ inline void CollisionBody::updateAABB() { mCollisionShape->updateAABB(mAabb, mTransform); } +// Return the first element of the linked list of contact manifolds involving this body +inline const ContactManifoldListElement* CollisionBody::getContactManifoldsLists() const { + return mContactManifoldsList; +} + } #endif diff --git a/src/body/RigidBody.cpp b/src/body/RigidBody.cpp index 757af86d..8b61ed6f 100644 --- a/src/body/RigidBody.cpp +++ b/src/body/RigidBody.cpp @@ -25,6 +25,7 @@ // Libraries #include "RigidBody.h" +#include "constraint/Constraint.h" #include "../collision/shapes/CollisionShape.h" // We want to use the ReactPhysics3D namespace @@ -36,13 +37,41 @@ RigidBody::RigidBody(const Transform& transform, decimal mass, const Matrix3x3& : CollisionBody(transform, collisionShape, id), mInertiaTensorLocal(inertiaTensorLocal), mMass(mass), mInertiaTensorLocalInverse(inertiaTensorLocal.getInverse()), mMassInverse(decimal(1.0) / mass), mIsGravityEnabled(true), - mLinearDamping(decimal(0.0)), mAngularDamping(decimal(0.0)) { + mLinearDamping(decimal(0.0)), mAngularDamping(decimal(0.0)), mJointsList(NULL) { assert(collisionShape); } // Destructor RigidBody::~RigidBody() { - + assert(mJointsList == NULL); } +// Remove a joint from the joints list +void RigidBody::removeJointFromJointsList(MemoryAllocator& memoryAllocator, const Constraint* joint) { + + assert(joint != NULL); + assert(mJointsList != NULL); + + // Remove the joint from the linked list of the joints of the first body + if (mJointsList->joint == joint) { // If the first element is the one to remove + JointListElement* elementToRemove = mJointsList; + mJointsList = elementToRemove->next; + elementToRemove->JointListElement::~JointListElement(); + memoryAllocator.release(elementToRemove, sizeof(JointListElement)); + } + else { // If the element to remove is not the first one in the list + JointListElement* currentElement = mJointsList; + while (currentElement->next != NULL) { + if (currentElement->next->joint == joint) { + JointListElement* elementToRemove = currentElement->next; + currentElement->next = elementToRemove->next; + elementToRemove->JointListElement::~JointListElement(); + memoryAllocator.release(elementToRemove, sizeof(JointListElement)); + break; + } + } + } +} + + diff --git a/src/body/RigidBody.h b/src/body/RigidBody.h index 6136f1dd..2856bc64 100644 --- a/src/body/RigidBody.h +++ b/src/body/RigidBody.h @@ -31,10 +31,15 @@ #include "CollisionBody.h" #include "../engine/Material.h" #include "../mathematics/mathematics.h" +#include "../memory/MemoryAllocator.h" /// Namespace reactphysics3d namespace reactphysics3d { +// Class declarations +struct JointListElement; +class Constraint; + // Class RigidBody /** * This class represents a rigid body of the physics @@ -86,6 +91,9 @@ class RigidBody : public CollisionBody { /// Angular velocity damping factor decimal mAngularDamping; + /// First element of the linked list of joints involving this body + JointListElement* mJointsList; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -94,6 +102,9 @@ class RigidBody : public CollisionBody { /// Private assignment operator RigidBody& operator=(const RigidBody& body); + /// Remove a joint from the joints list + void removeJointFromJointsList(MemoryAllocator& memoryAllocator, const Constraint* joint); + public : // -------------------- Methods -------------------- // @@ -179,6 +190,13 @@ class RigidBody : public CollisionBody { /// Set the angular damping factor void setAngularDamping(decimal angularDamping); + + /// Return the first element of the linked list of joints involving this body + const JointListElement* getJointsList() const; + + // -------------------- Friendship -------------------- // + + friend class DynamicsWorld; }; // Method that return the mass of the body @@ -328,6 +346,11 @@ inline void RigidBody::setAngularDamping(decimal angularDamping) { mAngularDamping = angularDamping; } +// Return the first element of the linked list of joints involving this body +inline const JointListElement* RigidBody::getJointsList() const { + return mJointsList; +} + } #endif diff --git a/src/configuration.h b/src/configuration.h index c762f14e..3cb8c3f0 100644 --- a/src/configuration.h +++ b/src/configuration.h @@ -92,8 +92,8 @@ const decimal DEFAULT_FRICTION_COEFFICIENT = decimal(0.3); /// Default bounciness factor for a rigid body const decimal DEFAULT_BOUNCINESS = decimal(0.5); -/// True if the deactivation (sleeping) of inactive bodies is enabled -const bool DEACTIVATION_ENABLED = true; +/// True if the spleeping technique for inactive bodies is enabled +const bool SPLEEPING_ENABLED = true; /// Object margin for collision detection in meters (for the GJK-EPA Algorithm) const decimal OBJECT_MARGIN = decimal(0.04); diff --git a/src/constraint/Constraint.cpp b/src/constraint/Constraint.cpp index 5da1c4eb..ab9c7d99 100644 --- a/src/constraint/Constraint.cpp +++ b/src/constraint/Constraint.cpp @@ -33,7 +33,7 @@ Constraint::Constraint(const ConstraintInfo& constraintInfo) :mBody1(constraintInfo.body1), mBody2(constraintInfo.body2), mActive(true), mType(constraintInfo.type), mPositionCorrectionTechnique(constraintInfo.positionCorrectionTechnique), - mIsCollisionEnabled(constraintInfo.isCollisionEnabled){ + mIsCollisionEnabled(constraintInfo.isCollisionEnabled), mIsAlreadyInIsland(false) { assert(mBody1 != NULL); assert(mBody2 != NULL); diff --git a/src/constraint/Constraint.h b/src/constraint/Constraint.h index ae7bfa77..adc4092a 100644 --- a/src/constraint/Constraint.h +++ b/src/constraint/Constraint.h @@ -39,6 +39,32 @@ enum ConstraintType {CONTACT, BALLSOCKETJOINT, SLIDERJOINT, HINGEJOINT, FIXEDJOI // Class declarations struct ConstraintSolverData; +class Constraint; + +// Structure JointListElement +/** + * This structure represents a single element of a linked list of joints + */ +struct JointListElement { + + public: + + // -------------------- Attributes -------------------- // + + /// Pointer to the actual joint + Constraint* joint; + + /// Next element of the list + JointListElement* next; + + // -------------------- Methods -------------------- // + + /// Constructor + JointListElement(Constraint* initJoint, JointListElement* initNext) + :joint(initJoint), next(initNext){ + + } +}; // Structure ConstraintInfo /** @@ -87,9 +113,8 @@ struct ConstraintInfo { // Class Constraint /** * This abstract class represents a constraint in the physics engine. - * A constraint can be a collision contact or a joint for - * instance. Each constraint can be made of several "mathematical - * constraints" needed to represent the main constraint. + * A constraint can be a collision contact point or a joint for + * instance. */ class Constraint { @@ -121,6 +146,9 @@ class Constraint { /// True if the two bodies of the constraint are allowed to collide with each other bool mIsCollisionEnabled; + /// True if the joint has already been added into an island + bool mIsAlreadyInIsland; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -154,6 +182,9 @@ class Constraint { /// Return true if the collision between the two bodies of the constraint is enabled bool isCollisionEnabled() const; + /// Return true if the joint has already been added into an island + bool isAlreadyInIsland() const; + /// Return the number of bytes used by the constraint virtual size_t getSizeInBytes() const = 0; @@ -168,6 +199,11 @@ class Constraint { /// Solve the position constraint virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData) = 0; + + // -------------------- Friendship -------------------- // + + friend class DynamicsWorld; + friend class Island; }; // Return the reference to the body 1 @@ -195,6 +231,11 @@ inline bool Constraint::isCollisionEnabled() const { return mIsCollisionEnabled; } +// Return true if the joint has already been added into an island +inline bool Constraint::isAlreadyInIsland() const { + return mIsAlreadyInIsland; +} + } #endif diff --git a/src/constraint/HingeJoint.cpp b/src/constraint/HingeJoint.cpp index 57c8764f..30b929b7 100644 --- a/src/constraint/HingeJoint.cpp +++ b/src/constraint/HingeJoint.cpp @@ -59,7 +59,7 @@ HingeJoint::HingeJoint(const HingeJointInfo& jointInfo) // Compute the inverse of the initial orientation difference between the two bodies mInitOrientationDifferenceInv = transform2.getOrientation() * - transform1.getOrientation().getInverse(); + transform1.getOrientation().getInverse(); mInitOrientationDifferenceInv.normalize(); mInitOrientationDifferenceInv.inverse(); } diff --git a/src/engine/ContactManifold.cpp b/src/engine/ContactManifold.cpp index 69a67e97..b2b9987e 100644 --- a/src/engine/ContactManifold.cpp +++ b/src/engine/ContactManifold.cpp @@ -30,10 +30,10 @@ using namespace reactphysics3d; // Constructor -ContactManifold::ContactManifold(Body* const body1, Body* const body2, +ContactManifold::ContactManifold(CollisionBody* body1, CollisionBody* body2, MemoryAllocator& memoryAllocator) : mBody1(body1), mBody2(body2), mNbContactPoints(0), mFrictionImpulse1(0.0), - mFrictionImpulse2(0.0), mFrictionTwistImpulse(0.0), + mFrictionImpulse2(0.0), mFrictionTwistImpulse(0.0), mIsAlreadyInIsland(false), mMemoryAllocator(memoryAllocator) { } @@ -107,9 +107,12 @@ void ContactManifold::update(const Transform& transform1, const Transform& trans // Update the world coordinates and penetration depth of the contact points in the manifold for (uint i=0; isetWorldPointOnBody1(transform1 * mContactPoints[i]->getLocalPointOnBody1()); - mContactPoints[i]->setWorldPointOnBody2(transform2 * mContactPoints[i]->getLocalPointOnBody2()); - mContactPoints[i]->setPenetrationDepth((mContactPoints[i]->getWorldPointOnBody1() - mContactPoints[i]->getWorldPointOnBody2()).dot(mContactPoints[i]->getNormal())); + mContactPoints[i]->setWorldPointOnBody1(transform1 * + mContactPoints[i]->getLocalPointOnBody1()); + mContactPoints[i]->setWorldPointOnBody2(transform2 * + mContactPoints[i]->getLocalPointOnBody2()); + mContactPoints[i]->setPenetrationDepth((mContactPoints[i]->getWorldPointOnBody1() - + mContactPoints[i]->getWorldPointOnBody2()).dot(mContactPoints[i]->getNormal())); } const decimal squarePersistentContactThreshold = PERSISTENT_CONTACT_DIST_THRESHOLD * @@ -172,7 +175,8 @@ int ContactManifold::getIndexOfDeepestPenetration(ContactPoint* newContact) cons /// Area = 0.5 * | AC x BD | where AC and BD form the diagonals of the quadrilateral. Note that /// when we compute this area, we do not calculate it exactly but we /// only estimate it because we do not compute the actual diagonals of the quadrialteral. Therefore, -/// this is only a guess that is faster to compute. +/// this is only a guess that is faster to compute. This idea comes from the Bullet Physics library +/// by Erwin Coumans (http://wwww.bulletphysics.org). int ContactManifold::getIndexToRemove(int indexMaxPenetration, const Vector3& newPoint) const { assert(mNbContactPoints == MAX_CONTACT_POINTS_IN_MANIFOLD); @@ -185,28 +189,32 @@ int ContactManifold::getIndexToRemove(int indexMaxPenetration, const Vector3& ne if (indexMaxPenetration != 0) { // Compute the area Vector3 vector1 = newPoint - mContactPoints[1]->getLocalPointOnBody1(); - Vector3 vector2 = mContactPoints[3]->getLocalPointOnBody1() - mContactPoints[2]->getLocalPointOnBody1(); + Vector3 vector2 = mContactPoints[3]->getLocalPointOnBody1() - + mContactPoints[2]->getLocalPointOnBody1(); Vector3 crossProduct = vector1.cross(vector2); area0 = crossProduct.lengthSquare(); } if (indexMaxPenetration != 1) { // Compute the area Vector3 vector1 = newPoint - mContactPoints[0]->getLocalPointOnBody1(); - Vector3 vector2 = mContactPoints[3]->getLocalPointOnBody1() - mContactPoints[2]->getLocalPointOnBody1(); + Vector3 vector2 = mContactPoints[3]->getLocalPointOnBody1() - + mContactPoints[2]->getLocalPointOnBody1(); Vector3 crossProduct = vector1.cross(vector2); area1 = crossProduct.lengthSquare(); } if (indexMaxPenetration != 2) { // Compute the area Vector3 vector1 = newPoint - mContactPoints[0]->getLocalPointOnBody1(); - Vector3 vector2 = mContactPoints[3]->getLocalPointOnBody1() - mContactPoints[1]->getLocalPointOnBody1(); + Vector3 vector2 = mContactPoints[3]->getLocalPointOnBody1() - + mContactPoints[1]->getLocalPointOnBody1(); Vector3 crossProduct = vector1.cross(vector2); area2 = crossProduct.lengthSquare(); } if (indexMaxPenetration != 3) { // Compute the area Vector3 vector1 = newPoint - mContactPoints[0]->getLocalPointOnBody1(); - Vector3 vector2 = mContactPoints[2]->getLocalPointOnBody1() - mContactPoints[1]->getLocalPointOnBody1(); + Vector3 vector2 = mContactPoints[2]->getLocalPointOnBody1() - + mContactPoints[1]->getLocalPointOnBody1(); Vector3 crossProduct = vector1.cross(vector2); area3 = crossProduct.lengthSquare(); } diff --git a/src/engine/ContactManifold.h b/src/engine/ContactManifold.h index 25fd18c1..70ebaae9 100644 --- a/src/engine/ContactManifold.h +++ b/src/engine/ContactManifold.h @@ -28,7 +28,7 @@ // Libraries #include -#include "../body/Body.h" +#include "../body/CollisionBody.h" #include "../constraint/ContactPoint.h" #include "../memory/MemoryAllocator.h" @@ -38,6 +38,35 @@ namespace reactphysics3d { // Constants const uint MAX_CONTACT_POINTS_IN_MANIFOLD = 4; // Maximum number of contacts in the manifold +// Class declarations +class ContactManifold; + +// Structure ContactManifoldListElement +/** + * This structure represents a single element of a linked list of contact manifolds + */ +struct ContactManifoldListElement { + + public: + + // -------------------- Attributes -------------------- // + + /// Pointer to the actual contact manifold + ContactManifold* contactManifold; + + /// Next element of the list + ContactManifoldListElement* next; + + // -------------------- Methods -------------------- // + + /// Constructor + ContactManifoldListElement(ContactManifold* initContactManifold, + ContactManifoldListElement* initNext) + :contactManifold(initContactManifold), next(initNext) { + + } +}; + // Class ContactManifold /** * This class represents the set of contact points between two bodies. @@ -59,11 +88,11 @@ class ContactManifold { // -------------------- Attributes -------------------- // - /// Pointer to the first body - Body* const mBody1; + /// Pointer to the first body of the contact + CollisionBody* mBody1; - /// Pointer to the second body - Body* const mBody2; + /// Pointer to the second body of the contact + CollisionBody* mBody2; /// Contact points in the manifold ContactPoint* mContactPoints[MAX_CONTACT_POINTS_IN_MANIFOLD]; @@ -86,6 +115,9 @@ class ContactManifold { /// Twist friction constraint accumulated impulse decimal mFrictionTwistImpulse; + /// True if the contact manifold has already been added into an island + bool mIsAlreadyInIsland; + /// Reference to the memory allocator MemoryAllocator& mMemoryAllocator; @@ -97,6 +129,12 @@ class ContactManifold { /// Private assignment operator ContactManifold& operator=(const ContactManifold& contactManifold); + /// Return a pointer to the first body of the contact manifold + CollisionBody* getBody1() const; + + /// Return a pointer to the second body of the contact manifold + CollisionBody* getBody2() const; + /// Return the index of maximum area int getMaxArea(decimal area0, decimal area1, decimal area2, decimal area3) const; @@ -109,15 +147,16 @@ class ContactManifold { /// Remove a contact point from the manifold void removeContactPoint(uint index); - /// Return true if two vectors are approximatively equal - bool isApproxEqual(const Vector3& vector1, const Vector3& vector2) const; + /// Return true if the contact manifold has already been added into an island + bool isAlreadyInIsland() const; public: // -------------------- Methods -------------------- // /// Constructor - ContactManifold(Body* const mBody1, Body* const mBody2, MemoryAllocator& memoryAllocator); + ContactManifold(CollisionBody* body1, CollisionBody* body2, + MemoryAllocator& memoryAllocator); /// Destructor ~ContactManifold(); @@ -166,8 +205,23 @@ class ContactManifold { /// Return a contact point of the manifold ContactPoint* getContactPoint(uint index) const; + + // -------------------- Friendship -------------------- // + + friend class DynamicsWorld; + friend class Island; }; +// Return a pointer to the first body of the contact manifold +inline CollisionBody* ContactManifold::getBody1() const { + return mBody1; +} + +// Return a pointer to the second body of the contact manifold +inline CollisionBody* ContactManifold::getBody2() const { + return mBody2; +} + // Return the number of contact points in the manifold inline uint ContactManifold::getNbContactPoints() const { return mNbContactPoints; @@ -229,6 +283,11 @@ inline ContactPoint* ContactManifold::getContactPoint(uint index) const { return mContactPoints[index]; } +// Return true if the contact manifold has already been added into an island +inline bool ContactManifold::isAlreadyInIsland() const { + return mIsAlreadyInIsland; +} + } #endif diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 605aec54..4ca989a9 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -44,7 +44,8 @@ DynamicsWorld::DynamicsWorld(const Vector3 &gravity, decimal timeStep = DEFAULT_ mMapBodyToConstrainedVelocityIndex), mNbVelocitySolverIterations(DEFAULT_VELOCITY_SOLVER_NB_ITERATIONS), mNbPositionSolverIterations(DEFAULT_POSITION_SOLVER_NB_ITERATIONS), - mIsDeactivationActive(DEACTIVATION_ENABLED) { + mIsSleepingEnabled(SPLEEPING_ENABLED), mNbIslands(0), mNbIslandsCapacity(0), + mIslands(NULL) { } @@ -59,6 +60,19 @@ DynamicsWorld::~DynamicsWorld() { mMemoryAllocator.release((*it).second, sizeof(OverlappingPair)); } + // Release the memory allocated for the islands + for (uint i=0; iIsland::~Island(); + + // Release the allocated memory for the island + mMemoryAllocator.release(mIslands[i], sizeof(Island)); + } + if (mNbIslandsCapacity > 0) { + mMemoryAllocator.release(mIslands, sizeof(Island*) * mNbIslandsCapacity); + } + // Free the allocated memory for the constrained velocities cleanupConstrainedVelocitiesArray(); @@ -93,6 +107,9 @@ void DynamicsWorld::update() { // Remove all contact manifolds mContactManifolds.clear(); + + // Reset all the contact manifolds lists of each body + resetContactManifoldListsOfBodies(); // Compute the collision detection mCollisionDetection.computeCollisionDetection(); @@ -106,6 +123,9 @@ void DynamicsWorld::update() { // Update the timer mTimer.nextStep(); + // Compute the islands (separate groups of bodies with constraints between each others) + computeIslands(); + // Solve the contacts and constraints solveContactsAndConstraints(); @@ -448,7 +468,8 @@ void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { // Remove the collision shape from the world removeCollisionShape(rigidBody->getCollisionShape()); - // Destroy all the joints that contains the rigid body to be destroyed + // Destroy all the joints in which the rigid body to be destroyed is involved + // TODO : Iterate on the mJointList of the rigid body instead over all the joints of the world bodyindex idToRemove = rigidBody->getID(); for (std::set::iterator it = mJoints.begin(); it != mJoints.end(); ++it) { if ((*it)->getBody1()->getID() == idToRemove || (*it)->getBody2()->getID() == idToRemove) { @@ -456,6 +477,9 @@ void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { } } + // Reset the contact manifold list of the body + rigidBody->resetContactManifoldsList(mMemoryAllocator); + // Call the destructor of the rigid body rigidBody->RigidBody::~RigidBody(); @@ -529,6 +553,9 @@ Constraint* DynamicsWorld::createJoint(const ConstraintInfo& jointInfo) { // Add the joint into the world mJoints.insert(newJoint); + // Add the joint into the joint list of the bodies involved in the joint + addJointToBody(newJoint); + // Return the pointer to the created joint return newJoint; } @@ -548,14 +575,235 @@ void DynamicsWorld::destroyJoint(Constraint* joint) { // Remove the joint from the world mJoints.erase(joint); - // Get the size in bytes of the joint - size_t nbBytes = joint->getSizeInBytes(); + // Remove the joint from the joint list of the bodies involved in the joint + joint->mBody1->removeJointFromJointsList(mMemoryAllocator, joint); + joint->mBody2->removeJointFromJointsList(mMemoryAllocator, joint); // Call the destructor of the joint joint->Constraint::~Constraint(); // Release the allocated memory - mMemoryAllocator.release(joint, nbBytes); + mMemoryAllocator.release(joint, joint->getSizeInBytes()); +} + +// Add the joint to the list of joints of the two bodies involved in the joint +void DynamicsWorld::addJointToBody(Constraint* joint) { + + assert(joint != NULL); + + // Add the joint at the beginning of the linked list of joints of the first body + void* allocatedMemory1 = mMemoryAllocator.allocate(sizeof(JointListElement)); + JointListElement* jointListElement1 = new (allocatedMemory1) JointListElement(joint, + joint->mBody1->mJointsList); + joint->mBody1->mJointsList = jointListElement1; + + // Add the joint at the beginning of the linked list of joints of the second body + void* allocatedMemory2 = mMemoryAllocator.allocate(sizeof(JointListElement)); + JointListElement* jointListElement2 = new (allocatedMemory2) JointListElement(joint, + joint->mBody2->mJointsList); + joint->mBody2->mJointsList = jointListElement2; +} + +// Add a contact manifold to the linked list of contact manifolds of the two bodies involed +// in the corresponding contact +void DynamicsWorld::addContactManifoldToBody(ContactManifold* contactManifold, + CollisionBody* body1, CollisionBody* body2) { + + assert(contactManifold != NULL); + + // Add the contact manifold at the beginning of the linked + // list of contact manifolds of the first body + void* allocatedMemory1 = mMemoryAllocator.allocate(sizeof(ContactManifoldListElement)); + ContactManifoldListElement* listElement1 = new (allocatedMemory1) + ContactManifoldListElement(contactManifold, + body1->mContactManifoldsList); + body1->mContactManifoldsList = listElement1; + + // Add the joint at the beginning of the linked list of joints of the second body + void* allocatedMemory2 = mMemoryAllocator.allocate(sizeof(ContactManifoldListElement)); + ContactManifoldListElement* listElement2 = new (allocatedMemory2) + ContactManifoldListElement(contactManifold, + body2->mContactManifoldsList); + body2->mContactManifoldsList = listElement2; +} + +// Reset all the contact manifolds linked list of each body +void DynamicsWorld::resetContactManifoldListsOfBodies() { + + // For each rigid body of the world + for (std::set::iterator it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { + + // Reset the contact manifold list of the body + (*it)->resetContactManifoldsList(mMemoryAllocator); + } +} + +// Compute the islands of awake bodies. +/// An island is an isolated group of rigid bodies that have constraints (joints or contacts) +/// between each other. This method computes the islands at each time step as follows: For each +/// awake rigid body, we run a Depth First Search (DFS) through the constraint graph of that body +/// (graph where nodes are the bodies and where the edges are the constraints between the bodies) to +/// find all the bodies that are connected with it (the bodies that share joints or contacts with +/// it). Then, we create an island with this group of connected bodies. +void DynamicsWorld::computeIslands() { + + PROFILE("DynamicsWorld::computeIslands()"); + + uint nbBodies = mRigidBodies.size(); + + // Clear all the islands + for (uint i=0; iIsland::~Island(); + + // Release the allocated memory for the island + mMemoryAllocator.release(mIslands[i], sizeof(Island)); + } + + // Allocate and create the array of islands + if (mNbIslandsCapacity != nbBodies && nbBodies > 0) { + if (mNbIslandsCapacity > 0) { + mMemoryAllocator.release(mIslands, sizeof(Island*) * mNbIslandsCapacity); + } + mNbIslandsCapacity = nbBodies; + mIslands = (Island**)mMemoryAllocator.allocate(sizeof(Island*) * mNbIslandsCapacity); + } + mNbIslands = 0; + + // Reset all the isAlreadyInIsland variables of bodies, joints and contact manifolds + for (std::set::iterator it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { + (*it)->mIsAlreadyInIsland = false; + } + for (std::vector::iterator it = mContactManifolds.begin(); + it != mContactManifolds.end(); ++it) { + (*it)->mIsAlreadyInIsland = false; + } + for (std::set::iterator it = mJoints.begin(); it != mJoints.end(); ++it) { + (*it)->mIsAlreadyInIsland = false; + } + + // Create a stack (using an array) for the rigid bodies to visit during the Depth First Search + size_t nbBytesStack = sizeof(RigidBody*) * nbBodies; + RigidBody** stackBodiesToVisit = (RigidBody**)mMemoryAllocator.allocate(nbBytesStack); + + uint idIsland = 0; // TODO : REMOVE THIS + + // For each rigid body of the world + for (std::set::iterator it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { + + RigidBody* body = *it; + + // If the body has already been added to an island, we go to the next body + if (body->isAlreadyInIsland()) continue; + + // If the body is not moving, we go to the next body + // TODO : When we will use STATIC bodies, we will need to take care of this case here + if (!body->getIsMotionEnabled()) continue; + + // If the body is sleeping, we go to the next body + if (body->isSleeping()) continue; + + // Reset the stack of bodies to visit + uint stackIndex = 0; + stackBodiesToVisit[stackIndex] = body; + stackIndex++; + body->mIsAlreadyInIsland = true; + + // Create the new island + void* allocatedMemoryIsland = mMemoryAllocator.allocate(sizeof(Island)); + mIslands[mNbIslands] = new (allocatedMemoryIsland) Island(idIsland, nbBodies,mContactManifolds.size(), + mJoints.size(), mMemoryAllocator); + idIsland++; + + // While there are still some bodies to visit in the stack + while (stackIndex > 0) { + + // Get the next body to visit from the stack + stackIndex--; + RigidBody* bodyToVisit = stackBodiesToVisit[stackIndex]; + + // Awake the body if it is slepping + bodyToVisit->setIsSleeping(false); + + // Add the body into the island + mIslands[mNbIslands]->addBody(bodyToVisit); + + // If the current body is not moving, we do not want to perform the DFS + // search across that body + if (!bodyToVisit->getIsMotionEnabled()) continue; + + // For each contact manifold in which the current body is involded + ContactManifoldListElement* contactElement; + for (contactElement = bodyToVisit->mContactManifoldsList; contactElement != NULL; + contactElement = contactElement->next) { + + ContactManifold* contactManifold = contactElement->contactManifold; + + // Check if the current contact manifold has already been added into an island + if (contactManifold->isAlreadyInIsland()) continue; + + // Add the contact manifold into the island + mIslands[mNbIslands]->addContactManifold(contactManifold); + contactManifold->mIsAlreadyInIsland = true; + + // Get the other body of the contact manifold + RigidBody* body1 = dynamic_cast(contactManifold->getBody1()); + RigidBody* body2 = dynamic_cast(contactManifold->getBody2()); + RigidBody* otherBody = (body1->getID() == bodyToVisit->getID()) ? body2 : body1; + + // Check if the other body has already been added to the island + if (otherBody->isAlreadyInIsland()) continue; + + // Insert the other body into the stack of bodies to visit + stackBodiesToVisit[stackIndex] = otherBody; + stackIndex++; + otherBody->mIsAlreadyInIsland = true; + } + + // For each joint in which the current body is involved + JointListElement* jointElement; + for (jointElement = bodyToVisit->mJointsList; jointElement != NULL; + jointElement = jointElement->next) { + + Constraint* joint = jointElement->joint; + + // Check if the current joint has already been added into an island + if (joint->isAlreadyInIsland()) continue; + + // Add the joint into the island + mIslands[mNbIslands]->addJoint(joint); + joint->mIsAlreadyInIsland = true; + + // Get the other body of the contact manifold + RigidBody* body1 = dynamic_cast(joint->getBody1()); + RigidBody* body2 = dynamic_cast(joint->getBody2()); + RigidBody* otherBody = (body1->getID() == bodyToVisit->getID()) ? body2 : body1; + + // Check if the other body has already been added to the island + if (otherBody->isAlreadyInIsland()) continue; + + // Insert the other body into the stack of bodies to visit + stackBodiesToVisit[stackIndex] = otherBody; + stackIndex++; + otherBody->mIsAlreadyInIsland = true; + } + } + + // Reset the isAlreadyIsland variable of the static bodies so that they + // can also be included in the other islands + for (uint i=0; i < mIslands[mNbIslands]->mNbBodies; i++) { + + if (!mIslands[mNbIslands]->mBodies[i]->getIsMotionEnabled()) { + mIslands[mNbIslands]->mBodies[i]->mIsAlreadyInIsland = false; + } + } + + mNbIslands++; + } + + // Release the allocated memory for the stack of bodies to visit + mMemoryAllocator.release(stackBodiesToVisit, nbBytesStack); } // Notify the world about a new broad-phase overlapping pair @@ -565,8 +813,8 @@ void DynamicsWorld::notifyAddedOverlappingPair(const BroadPhasePair* addedPair) bodyindexpair indexPair = addedPair->getBodiesIndexPair(); // Add the pair into the set of overlapping pairs (if not there yet) - OverlappingPair* newPair = new (mMemoryAllocator.allocate(sizeof(OverlappingPair))) OverlappingPair( - addedPair->body1, addedPair->body2, mMemoryAllocator); + OverlappingPair* newPair = new (mMemoryAllocator.allocate(sizeof(OverlappingPair))) + OverlappingPair(addedPair->body1, addedPair->body2, mMemoryAllocator); assert(newPair != NULL); std::pair::iterator, bool> check = mOverlappingPairs.insert(make_pair(indexPair, newPair)); @@ -604,4 +852,9 @@ void DynamicsWorld::notifyNewContact(const BroadPhasePair* broadPhasePair, // Add the contact manifold to the world mContactManifolds.push_back(overlappingPair->getContactManifold()); + + // Add the contact manifold into the list of contact manifolds + // of the two bodies involved in the contact + addContactManifoldToBody(overlappingPair->getContactManifold(), overlappingPair->mBody1, + overlappingPair->mBody2); } diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index a1a1f821..89ebec5d 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -33,6 +33,7 @@ #include "ConstraintSolver.h" #include "../body/RigidBody.h" #include "Timer.h" +#include "Island.h" #include "../configuration.h" /// Namespace ReactPhysics3D @@ -65,13 +66,14 @@ class DynamicsWorld : public CollisionWorld { /// Number of iterations for the position solver of the Sequential Impulses technique uint mNbPositionSolverIterations; - /// True if the deactivation (sleeping) of inactive bodies is enabled - bool mIsDeactivationActive; + /// True if the spleeping technique for inactive bodies is enabled + bool mIsSleepingEnabled; /// All the rigid bodies of the physics world std::set mRigidBodies; /// All the contact constraints + // TODO : Remove this variable (we will use the ones in the island now) std::vector mContactManifolds; /// All the joints of the world @@ -100,6 +102,15 @@ class DynamicsWorld : public CollisionWorld { /// Map body to their index in the constrained velocities array std::map mMapBodyToConstrainedVelocityIndex; + /// Number of islands in the world + uint mNbIslands; + + /// Current allocated capacity for the islands + uint mNbIslandsCapacity; + + /// Array with all the islands of awaken bodies + Island** mIslands; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -136,6 +147,9 @@ class DynamicsWorld : public CollisionWorld { /// Reset the boolean movement variable of each body void resetBodiesMovementVariable(); + /// Compute the islands of awake bodies. + void computeIslands(); + /// Update the overlapping pair virtual void updateOverlappingPair(const BroadPhasePair* pair); @@ -198,6 +212,17 @@ public : /// Destroy a joint void destroyJoint(Constraint* joint); + /// Add the joint to the list of joints of the two bodies involved in the joint + void addJointToBody(Constraint* joint); + + //// Add a contact manifold to the linked list of contact manifolds of the two bodies involed + //// in the corresponding contact. + void addContactManifoldToBody(ContactManifold* contactManifold, + CollisionBody *body1, CollisionBody *body2); + + /// Reset all the contact manifolds linked list of each body + void resetContactManifoldListsOfBodies(); + /// Return the gravity vector of the world Vector3 getGravity() const; @@ -227,6 +252,12 @@ public : /// Return a reference to the contact manifolds of the world const std::vector& getContactManifolds() const; + + // TODO : REMOVE THIS + Island** getIslands() { return mIslands;} + + // TODO : REMOVE THIS + uint getNbIslands() const {return mNbIslands;} }; // Start the physics simulation diff --git a/src/engine/Island.cpp b/src/engine/Island.cpp new file mode 100644 index 00000000..06b3734b --- /dev/null +++ b/src/engine/Island.cpp @@ -0,0 +1,54 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Island.h" + +using namespace reactphysics3d; + +// Constructor +Island::Island(uint id, uint nbMaxBodies, uint nbMaxContactManifolds, uint nbMaxJoints, + MemoryAllocator& memoryAllocator) + : mID(id), mBodies(NULL), mContactManifolds(NULL), mJoints(NULL), mNbBodies(0), + mNbContactManifolds(0), mNbJoints(0), mMemoryAllocator(memoryAllocator) { + + // Allocate memory for the arrays + mNbAllocatedBytesBodies = sizeof(RigidBody*) * nbMaxBodies; + mBodies = (RigidBody**) mMemoryAllocator.allocate(mNbAllocatedBytesBodies); + mNbAllocatedBytesContactManifolds = sizeof(ContactManifold*) * nbMaxContactManifolds; + mContactManifolds = (ContactManifold**) mMemoryAllocator.allocate( + mNbAllocatedBytesContactManifolds); + mNbAllocatedBytesJoints = sizeof(Constraint*) * nbMaxJoints; + mJoints = (Constraint**) mMemoryAllocator.allocate(mNbAllocatedBytesJoints); +} + +// Destructor +Island::~Island() { + + // Release the memory of the arrays + mMemoryAllocator.release(mBodies, mNbAllocatedBytesBodies); + mMemoryAllocator.release(mContactManifolds, mNbAllocatedBytesContactManifolds); + mMemoryAllocator.release(mJoints, mNbAllocatedBytesJoints); +} diff --git a/src/engine/Island.h b/src/engine/Island.h new file mode 100644 index 00000000..bcd9d990 --- /dev/null +++ b/src/engine/Island.h @@ -0,0 +1,186 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_ISLAND_H +#define REACTPHYSICS3D_ISLAND_H + +// Libraries +#include "../memory/MemoryAllocator.h" +#include "../body/RigidBody.h" +#include "../constraint/Constraint.h" +#include "ContactManifold.h" + +namespace reactphysics3d { + +// Class Island +/** + * An island represent an isolated group of awake bodies that are connected with each other by + * some contraints (contacts or joints). + */ +class Island { + + private: + + // -------------------- Attributes -------------------- // + + // TODO : REMOVE THIS + uint mID; + + /// Array with all the bodies of the island + RigidBody** mBodies; + + /// Array with all the contact manifolds between bodies of the island + ContactManifold** mContactManifolds; + + /// Array with all the joints between bodies of the island + Constraint** mJoints; + + /// Current number of bodies in the island + uint mNbBodies; + + /// Current number of contact manifold in the island + uint mNbContactManifolds; + + /// Current number of joints in the island + uint mNbJoints; + + /// Reference to the memory allocator + MemoryAllocator& mMemoryAllocator; + + /// Number of bytes allocated for the bodies array + size_t mNbAllocatedBytesBodies; + + /// Number of bytes allocated for the contact manifolds array + size_t mNbAllocatedBytesContactManifolds; + + /// Number of bytes allocated for the joints array + size_t mNbAllocatedBytesJoints; + + // -------------------- Methods -------------------- // + + /// Private assignment operator + Island& operator=(const Island& island); + + /// Private copy-constructor + Island(const Island& island); + + public: + + // -------------------- Methods -------------------- // + + /// Constructor + Island(uint id, uint nbMaxBodies, uint nbMaxContactManifolds, uint nbMaxJoints, + MemoryAllocator& memoryAllocator); + + /// Destructor + ~Island(); + + /// Add a body into the island + void addBody(RigidBody* body); + + /// Add a contact manifold into the island + void addContactManifold(ContactManifold* contactManifold); + + /// Add a joint into the island + void addJoint(Constraint* joint); + + /// Return the number of bodies in the island + uint getNbBodies() const; + + /// Return the number of contact manifolds in the island + uint getNbContactManifolds() const; + + /// Return the number of joints in the island + uint getNbJoints() const; + + /// Return a pointer to the array of bodies + RigidBody** getBodies(); + + /// Return a pointer to the array of contact manifolds + ContactManifold** getContactManifold(); + + /// Return a pointer to the array of joints + Constraint** getJoints(); + + // TODO : REMOVE THIS + uint getID() const {return mID;} + + // -------------------- Friendship -------------------- // + + friend class DynamicsWorld; +}; + +// Add a body into the island +inline void Island::addBody(RigidBody* body) { + assert(!body->isSleeping()); + mBodies[mNbBodies] = body; + mNbBodies++; +} + +// Add a contact manifold into the island +inline void Island::addContactManifold(ContactManifold* contactManifold) { + mContactManifolds[mNbContactManifolds] = contactManifold; + mNbContactManifolds++; +} + +// Add a joint into the island +inline void Island::addJoint(Constraint* joint) { + mJoints[mNbJoints] = joint; + mNbJoints++; +} + +// Return the number of bodies in the island +inline uint Island::getNbBodies() const { + return mNbBodies; +} + +// Return the number of contact manifolds in the island +inline uint Island::getNbContactManifolds() const { + return mNbContactManifolds; +} + +// Return the number of joints in the island +inline uint Island::getNbJoints() const { + return mNbJoints; +} + +// Return a pointer to the array of bodies +inline RigidBody** Island::getBodies() { + return mBodies; +} + +// Return a pointer to the array of contact manifolds +inline ContactManifold** Island::getContactManifold() { + return mContactManifolds; +} + +// Return a pointer to the array of joints +inline Constraint** Island::getJoints() { + return mJoints; +} + +} + +#endif diff --git a/src/engine/OverlappingPair.h b/src/engine/OverlappingPair.h index 40f1958b..1ecfe0c6 100644 --- a/src/engine/OverlappingPair.h +++ b/src/engine/OverlappingPair.h @@ -47,10 +47,10 @@ class OverlappingPair { // -------------------- Attributes -------------------- // /// Pointer to the first body of the contact - CollisionBody* const mBody1; + CollisionBody* mBody1; /// Pointer to the second body of the contact - CollisionBody* const mBody2; + CollisionBody* mBody2; /// Persistent contact manifold ContactManifold mContactManifold; @@ -100,6 +100,10 @@ class OverlappingPair { /// Return the contact manifold ContactManifold* getContactManifold(); + + // -------------------- Friendship -------------------- // + + friend class DynamicsWorld; }; // Return the pointer to first body diff --git a/src/memory/MemoryAllocator.cpp b/src/memory/MemoryAllocator.cpp index e715a157..ec074042 100644 --- a/src/memory/MemoryAllocator.cpp +++ b/src/memory/MemoryAllocator.cpp @@ -148,10 +148,10 @@ void* MemoryAllocator::allocate(size_t size) { assert(nbUnits * unitSize <= BLOCK_SIZE); for (size_t i=0; i < nbUnits - 1; i++) { MemoryUnit* unit = (MemoryUnit*) ((size_t)newBlock->memoryUnits + unitSize * i); - MemoryUnit* nextUnit = (MemoryUnit*) ((size_t)newBlock->memoryUnits + unitSize * (i + 1)); + MemoryUnit* nextUnit = (MemoryUnit*) ((size_t)newBlock->memoryUnits + unitSize * (i+1)); unit->nextUnit = nextUnit; } - MemoryUnit* lastUnit = (MemoryUnit*) ((size_t)newBlock->memoryUnits + unitSize * (nbUnits - 1)); + MemoryUnit* lastUnit = (MemoryUnit*) ((size_t)newBlock->memoryUnits + unitSize*(nbUnits-1)); lastUnit->nextUnit = NULL; // Add the new allocated block into the list of free memory units in the heap From 475ec5be5fca73c4ce78e3250375631da96fa03b Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sat, 31 Aug 2013 19:03:21 +0200 Subject: [PATCH 50/66] Iterate over the islands to solve the contacts and joints --- src/body/RigidBody.cpp | 1 + src/engine/ConstraintSolver.cpp | 70 ++++++----- src/engine/ConstraintSolver.h | 62 +++++----- src/engine/ContactSolver.cpp | 90 ++++---------- src/engine/ContactSolver.h | 85 ++++++-------- src/engine/DynamicsWorld.cpp | 200 +++++++++++++++++++------------- src/engine/DynamicsWorld.h | 16 ++- 7 files changed, 250 insertions(+), 274 deletions(-) diff --git a/src/body/RigidBody.cpp b/src/body/RigidBody.cpp index 8b61ed6f..82e5c928 100644 --- a/src/body/RigidBody.cpp +++ b/src/body/RigidBody.cpp @@ -70,6 +70,7 @@ void RigidBody::removeJointFromJointsList(MemoryAllocator& memoryAllocator, cons memoryAllocator.release(elementToRemove, sizeof(JointListElement)); break; } + currentElement = currentElement->next; } } } diff --git a/src/engine/ConstraintSolver.cpp b/src/engine/ConstraintSolver.cpp index 2154ceb0..25ef6641 100644 --- a/src/engine/ConstraintSolver.cpp +++ b/src/engine/ConstraintSolver.cpp @@ -30,18 +30,14 @@ using namespace reactphysics3d; // Constructor -ConstraintSolver::ConstraintSolver(std::set& joints, - std::vector& linearVelocities, - std::vector& angularVelocities, - std::vector& positions, +ConstraintSolver::ConstraintSolver(std::vector& positions, std::vector& orientations, const std::map& mapBodyToVelocityIndex) - : mJoints(joints), mLinearVelocities(linearVelocities), - mAngularVelocities(angularVelocities), mPositions(positions), + : mLinearVelocities(NULL), mAngularVelocities(NULL), mPositions(positions), mOrientations(orientations), mMapBodyToConstrainedVelocityIndex(mapBodyToVelocityIndex), - mIsWarmStartingActive(true), mConstraintSolverData(linearVelocities, - angularVelocities, positions, orientations, mapBodyToVelocityIndex){ + mIsWarmStartingActive(true), mConstraintSolverData(positions, orientations, + mapBodyToVelocityIndex){ } @@ -50,10 +46,16 @@ ConstraintSolver::~ConstraintSolver() { } -// Initialize the constraint solver -void ConstraintSolver::initialize(decimal dt) { +// Initialize the constraint solver for a given island +void ConstraintSolver::initializeForIsland(decimal dt, Island* island) { - PROFILE("ConstraintSolver::initialize()"); + PROFILE("ConstraintSolver::initializeForIsland()"); + + assert(mLinearVelocities != NULL); + assert(mAngularVelocities != NULL); + assert(island != NULL); + assert(island->getNbBodies() > 0); + assert(island->getNbJoints() > 0); // Set the current time step mTimeStep = dt; @@ -62,54 +64,50 @@ void ConstraintSolver::initialize(decimal dt) { mConstraintSolverData.timeStep = mTimeStep; mConstraintSolverData.isWarmStartingActive = mIsWarmStartingActive; - // For each joint - std::set::iterator it; - for (it = mJoints.begin(); it != mJoints.end(); ++it) { - - Constraint* joint = (*it); - - // Get the rigid bodies of the joint - RigidBody* body1 = joint->getBody1(); - RigidBody* body2 = joint->getBody2(); - - // Add the bodies to the set of constrained bodies - mConstraintBodies.insert(body1); - mConstraintBodies.insert(body2); + // For each joint of the island + Constraint** joints = island->getJoints(); + for (uint i=0; igetNbJoints(); i++) { // Initialize the constraint before solving it - joint->initBeforeSolve(mConstraintSolverData); + joints[i]->initBeforeSolve(mConstraintSolverData); // Warm-start the constraint if warm-starting is enabled if (mIsWarmStartingActive) { - joint->warmstart(mConstraintSolverData); + joints[i]->warmstart(mConstraintSolverData); } } } // Solve the velocity constraints -void ConstraintSolver::solveVelocityConstraints() { +void ConstraintSolver::solveVelocityConstraints(Island* island) { PROFILE("ConstraintSolver::solveVelocityConstraints()"); - // For each joint - std::set::iterator it; - for (it = mJoints.begin(); it != mJoints.end(); ++it) { + assert(island != NULL); + assert(island->getNbJoints() > 0); + + // For each joint of the island + Constraint** joints = island->getJoints(); + for (uint i=0; igetNbJoints(); i++) { // Solve the constraint - (*it)->solveVelocityConstraint(mConstraintSolverData); + joints[i]->solveVelocityConstraint(mConstraintSolverData); } } // Solve the position constraints -void ConstraintSolver::solvePositionConstraints() { +void ConstraintSolver::solvePositionConstraints(Island* island) { PROFILE("ConstraintSolver::solvePositionConstraints()"); - // For each joint - std::set::iterator it; - for (it = mJoints.begin(); it != mJoints.end(); ++it) { + assert(island != NULL); + assert(island->getNbJoints() > 0); + + // For each joint of the island + Constraint** joints = island->getJoints(); + for (uint i=0; i < island->getNbJoints(); i++) { // Solve the constraint - (*it)->solvePositionConstraint(mConstraintSolverData); + joints[i]->solvePositionConstraint(mConstraintSolverData); } } diff --git a/src/engine/ConstraintSolver.h b/src/engine/ConstraintSolver.h index 79cac7f7..f52c6412 100644 --- a/src/engine/ConstraintSolver.h +++ b/src/engine/ConstraintSolver.h @@ -30,6 +30,7 @@ #include "../configuration.h" #include "mathematics/mathematics.h" #include "../constraint/Constraint.h" +#include "Island.h" #include #include @@ -47,11 +48,11 @@ struct ConstraintSolverData { /// Current time step of the simulation decimal timeStep; - /// Reference to the bodies linear velocities - std::vector& linearVelocities; + /// Array with the bodies linear velocities + Vector3* linearVelocities; - /// Reference to the bodies angular velocities - std::vector& angularVelocities; + /// Array with the bodies angular velocities + Vector3* angularVelocities; /// Reference to the bodies positions std::vector& positions; @@ -67,13 +68,11 @@ struct ConstraintSolverData { bool isWarmStartingActive; /// Constructor - ConstraintSolverData(std::vector& refLinearVelocities, - std::vector& refAngularVelocities, - std::vector& refPositions, + ConstraintSolverData(std::vector& refPositions, std::vector& refOrientations, const std::map& refMapBodyToConstrainedVelocityIndex) - :linearVelocities(refLinearVelocities), - angularVelocities(refAngularVelocities), + :linearVelocities(NULL), + angularVelocities(NULL), positions(refPositions), orientations(refOrientations), mapBodyToConstrainedVelocityIndex(refMapBodyToConstrainedVelocityIndex){ @@ -156,19 +155,13 @@ class ConstraintSolver { // -------------------- Attributes -------------------- // - /// Reference to all the joints of the world - std::set& mJoints; - - /// Constrained bodies - std::set mConstraintBodies; - - /// Reference to the array of constrained linear velocities (state of the linear velocities + /// Array of constrained linear velocities (state of the linear velocities /// after solving the constraints) - std::vector& mLinearVelocities; + Vector3* mLinearVelocities; - /// Reference to the array of constrained angular velocities (state of the angular velocities + /// Array of constrained angular velocities (state of the angular velocities /// after solving the constraints) - std::vector& mAngularVelocities; + Vector3* mAngularVelocities; /// Reference to the array of bodies positions (for position error correction) std::vector& mPositions; @@ -194,24 +187,20 @@ class ConstraintSolver { // -------------------- Methods -------------------- // /// Constructor - ConstraintSolver(std::set& joints, - std::vector& linearVelocities, - std::vector& angularVelocities, - std::vector& positions, - std::vector& orientations, + ConstraintSolver(std::vector& positions, std::vector& orientations, const std::map& mapBodyToVelocityIndex); /// Destructor ~ConstraintSolver(); - /// Initialize the constraint solver - void initialize(decimal dt); + /// Initialize the constraint solver for a given island + void initializeForIsland(decimal dt, Island* island); /// Solve the constraints - void solveVelocityConstraints(); + void solveVelocityConstraints(Island* island); /// Solve the position constraints - void solvePositionConstraints(); + void solvePositionConstraints(Island* island); /// Return true if the Non-Linear-Gauss-Seidel position correction technique is active bool getIsNonLinearGaussSeidelPositionCorrectionActive() const; @@ -219,13 +208,20 @@ class ConstraintSolver { /// Enable/Disable the Non-Linear-Gauss-Seidel position correction technique. void setIsNonLinearGaussSeidelPositionCorrectionActive(bool isActive); - /// Return true if the body is in at least one constraint - bool isConstrainedBody(RigidBody* body) const; + /// Set the constrained velocities arrays + void setConstrainedVelocitiesArrays(Vector3* constrainedLinearVelocities, + Vector3* constrainedAngularVelocities); }; -// Return true if the body is in at least one constraint -inline bool ConstraintSolver::isConstrainedBody(RigidBody* body) const { - return mConstraintBodies.count(body) == 1; +// Set the constrained velocities arrays +inline void ConstraintSolver::setConstrainedVelocitiesArrays(Vector3* constrainedLinearVelocities, + Vector3* constrainedAngularVelocities) { + assert(constrainedLinearVelocities != NULL); + assert(constrainedAngularVelocities != NULL); + mLinearVelocities = constrainedLinearVelocities; + mAngularVelocities = constrainedAngularVelocities; + mConstraintSolverData.linearVelocities = mLinearVelocities; + mConstraintSolverData.angularVelocities = mAngularVelocities; } } diff --git a/src/engine/ContactSolver.cpp b/src/engine/ContactSolver.cpp index 8467a710..f70abbec 100644 --- a/src/engine/ContactSolver.cpp +++ b/src/engine/ContactSolver.cpp @@ -36,18 +36,12 @@ using namespace std; // Constants initialization const decimal ContactSolver::BETA = decimal(0.2); const decimal ContactSolver::BETA_SPLIT_IMPULSE = decimal(0.2); -const decimal ContactSolver::SLOP = decimal(0.01); +const decimal ContactSolver::SLOP= decimal(0.01); // Constructor -ContactSolver::ContactSolver(std::vector& contactManifolds, - std::vector& constrainedLinearVelocities, - std::vector& constrainedAngularVelocities, - const std::map& mapBodyToVelocityIndex) - :mContactManifolds(contactManifolds), - mSplitLinearVelocities(NULL), mSplitAngularVelocities(NULL), - mContactConstraints(NULL), - mLinearVelocities(constrainedLinearVelocities), - mAngularVelocities(constrainedAngularVelocities), +ContactSolver::ContactSolver(const std::map& mapBodyToVelocityIndex) + :mSplitLinearVelocities(NULL), mSplitAngularVelocities(NULL), + mContactConstraints(NULL), mLinearVelocities(NULL), mAngularVelocities(NULL), mMapBodyToConstrainedVelocityIndex(mapBodyToVelocityIndex), mIsWarmStartingActive(true), mIsSplitImpulseActive(true), mIsSolveFrictionAtContactManifoldCenterActive(true) { @@ -59,26 +53,32 @@ ContactSolver::~ContactSolver() { } -// Initialize the constraint solver -void ContactSolver::initialize(decimal dt) { +// Initialize the constraint solver for a given island +void ContactSolver::initializeForIsland(decimal dt, Island* island) { - PROFILE("ContactSolver::initialize()"); + PROFILE("ContactSolver::initializeForIsland()"); + + assert(island != NULL); + assert(island->getNbBodies() > 0); + assert(island->getNbContactManifolds() > 0); + assert(mSplitLinearVelocities != NULL); + assert(mSplitAngularVelocities != NULL); // Set the current time step mTimeStep = dt; - // TODO : Use better memory allocation here - mContactConstraints = new ContactManifoldSolver[mContactManifolds.size()]; + mNbContactManifolds = island->getNbContactManifolds(); - mNbContactManifolds = 0; + mContactConstraints = new ContactManifoldSolver[mNbContactManifolds]; + assert(mContactConstraints != NULL); - // For each contact manifold of the world - vector::iterator it; - for (it = mContactManifolds.begin(); it != mContactManifolds.end(); ++it) { + // For each contact manifold of the island + ContactManifold** contactManifolds = island->getContactManifold(); + for (uint i=0; igetNbContactPoints() > 0); @@ -86,10 +86,6 @@ void ContactSolver::initialize(decimal dt) { RigidBody* body1 = externalManifold->getContactPoint(0)->getBody1(); RigidBody* body2 = externalManifold->getContactPoint(0)->getBody2(); - // Add the two bodies of the constraint in the constraintBodies list - mConstraintBodies.insert(body1); - mConstraintBodies.insert(body2); - // Get the position of the two bodies Vector3 x1 = body1->getTransform().getPosition(); Vector3 x2 = body2->getTransform().getPosition(); @@ -173,46 +169,12 @@ void ContactSolver::initialize(decimal dt) { internalManifold.frictionTwistImpulse = 0.0; } } - - mNbContactManifolds++; } - // Allocated memory for split impulse velocities - // TODO : Use better memory allocation here - mSplitLinearVelocities = new Vector3[mMapBodyToConstrainedVelocityIndex.size()]; - mSplitAngularVelocities = new Vector3[mMapBodyToConstrainedVelocityIndex.size()]; - assert(mSplitLinearVelocities != NULL); - assert(mSplitAngularVelocities != NULL); - - assert(mConstraintBodies.size() > 0); - assert(mMapBodyToConstrainedVelocityIndex.size() >= mConstraintBodies.size()); - assert(mLinearVelocities.size() >= mConstraintBodies.size()); - assert(mAngularVelocities.size() >= mConstraintBodies.size()); - - // Initialize the split impulse velocities - initializeSplitImpulseVelocities(); - // Fill-in all the matrices needed to solve the LCP problem initializeContactConstraints(); } -// Initialize the split impulse velocities -void ContactSolver::initializeSplitImpulseVelocities() { - - // For each current body that is implied in some constraint - set::iterator it; - for (it = mConstraintBodies.begin(); it != mConstraintBodies.end(); ++it) { - RigidBody* rigidBody = *it; - assert(rigidBody); - - uint bodyNumber = mMapBodyToConstrainedVelocityIndex.find(rigidBody)->second; - - // Initialize the split impulse velocities to zero - mSplitLinearVelocities[bodyNumber] = Vector3(0, 0, 0); - mSplitAngularVelocities[bodyNumber] = Vector3(0, 0, 0); - } -} - // Initialize the contact constraints before solving the system void ContactSolver::initializeContactConstraints() { @@ -884,18 +846,8 @@ void ContactSolver::computeFrictionVectors(const Vector3& deltaVelocity, // Clean up the constraint solver void ContactSolver::cleanup() { - mConstraintBodies.clear(); - if (mContactConstraints != NULL) { delete[] mContactConstraints; mContactConstraints = NULL; } - if (mSplitLinearVelocities != NULL) { - delete[] mSplitLinearVelocities; - mSplitLinearVelocities = NULL; - } - if (mSplitAngularVelocities != NULL) { - delete[] mSplitAngularVelocities; - mSplitAngularVelocities = NULL; - } } diff --git a/src/engine/ContactSolver.h b/src/engine/ContactSolver.h index 2c635fb0..979a0abf 100644 --- a/src/engine/ContactSolver.h +++ b/src/engine/ContactSolver.h @@ -31,6 +31,7 @@ #include "../configuration.h" #include "../constraint/Constraint.h" #include "ContactManifold.h" +#include "Island.h" #include "Impulse.h" #include #include @@ -311,9 +312,6 @@ class ContactSolver { // -------------------- Attributes -------------------- // - /// Reference to all the contact manifold of the world - std::vector& mContactManifolds; - /// Split linear velocities for the position contact solver (split impulse) Vector3* mSplitLinearVelocities; @@ -329,14 +327,11 @@ class ContactSolver { /// Number of contact constraints uint mNbContactManifolds; - /// Constrained bodies - std::set mConstraintBodies; + /// Array of linear velocities + Vector3* mLinearVelocities; - /// Reference to the array of linear velocities - std::vector& mLinearVelocities; - - /// Reference to the array of angular velocities - std::vector& mAngularVelocities; + /// Array of angular velocities + Vector3* mAngularVelocities; /// Reference to the map of rigid body to their index in the constrained velocities array const std::map& mMapBodyToConstrainedVelocityIndex; @@ -353,9 +348,6 @@ class ContactSolver { // -------------------- Methods -------------------- // - /// Initialize the split impulse velocities - void initializeSplitImpulseVelocities(); - /// Initialize the contact constraints before solving the system void initializeContactConstraints(); @@ -403,16 +395,21 @@ class ContactSolver { // -------------------- Methods -------------------- // /// Constructor - ContactSolver(std::vector& contactManifolds, - std::vector& constrainedLinearVelocities, - std::vector& constrainedAngularVelocities, - const std::map& mapBodyToVelocityIndex); + ContactSolver(const std::map& mapBodyToVelocityIndex); /// Destructor virtual ~ContactSolver(); - /// Initialize the constraint solver - void initialize(decimal dt); + /// Initialize the constraint solver for a given island + void initializeForIsland(decimal dt, Island* island); + + /// Set the split velocities arrays + void setSplitVelocitiesArrays(Vector3* splitLinearVelocities, + Vector3* splitAngularVelocities); + + /// Set the constrained velocities arrays + void setConstrainedVelocitiesArrays(Vector3* constrainedLinearVelocities, + Vector3* constrainedAngularVelocities); /// Warm start the solver. void warmStart(); @@ -424,24 +421,6 @@ class ContactSolver { /// Solve the contacts void solve(); - /// Return true if the body is in at least one constraint - bool isConstrainedBody(RigidBody* body) const; - - /// Return the constrained linear velocity of a body after solving the constraints - Vector3 getConstrainedLinearVelocityOfBody(RigidBody *body); - - /// Return the split linear velocity - Vector3 getSplitLinearVelocityOfBody(RigidBody* body); - - /// Return the constrained angular velocity of a body after solving the constraints - Vector3 getConstrainedAngularVelocityOfBody(RigidBody* body); - - /// Return the split angular velocity - Vector3 getSplitAngularVelocityOfBody(RigidBody* body); - - /// Clean up the constraint solver - void cleanup(); - /// Return true if the split impulses position correction technique is used for contacts bool isSplitImpulseActive() const; @@ -451,25 +430,27 @@ class ContactSolver { /// Activate or deactivate the solving of friction constraints at the center of /// the contact manifold instead of solving them at each contact point void setIsSolveFrictionAtContactManifoldCenterActive(bool isActive); + + /// Clean up the constraint solver + void cleanup(); }; -// Return true if the body is in at least one constraint -inline bool ContactSolver::isConstrainedBody(RigidBody* body) const { - return mConstraintBodies.count(body) == 1; +// Set the split velocities arrays +inline void ContactSolver::setSplitVelocitiesArrays(Vector3* splitLinearVelocities, + Vector3* splitAngularVelocities) { + assert(splitLinearVelocities != NULL); + assert(splitAngularVelocities != NULL); + mSplitLinearVelocities = splitLinearVelocities; + mSplitAngularVelocities = splitAngularVelocities; } -// Return the split linear velocity -inline Vector3 ContactSolver::getSplitLinearVelocityOfBody(RigidBody* body) { - assert(isConstrainedBody(body)); - const uint indexBody = mMapBodyToConstrainedVelocityIndex.find(body)->second; - return mSplitLinearVelocities[indexBody]; -} - -// Return the split angular velocity -inline Vector3 ContactSolver::getSplitAngularVelocityOfBody(RigidBody* body) { - assert(isConstrainedBody(body)); - const uint indexBody = mMapBodyToConstrainedVelocityIndex.find(body)->second; - return mSplitAngularVelocities[indexBody]; +// Set the constrained velocities arrays +inline void ContactSolver::setConstrainedVelocitiesArrays(Vector3* constrainedLinearVelocities, + Vector3* constrainedAngularVelocities) { + assert(constrainedLinearVelocities != NULL); + assert(constrainedAngularVelocities != NULL); + mLinearVelocities = constrainedLinearVelocities; + mAngularVelocities = constrainedAngularVelocities; } // Return true if the split impulses position correction technique is used for contacts diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 4ca989a9..511f4630 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -37,15 +37,15 @@ using namespace std; // Constructor DynamicsWorld::DynamicsWorld(const Vector3 &gravity, decimal timeStep = DEFAULT_TIMESTEP) : CollisionWorld(), mTimer(timeStep), mGravity(gravity), mIsGravityOn(true), - mContactSolver(mContactManifolds, mConstrainedLinearVelocities, mConstrainedAngularVelocities, - mMapBodyToConstrainedVelocityIndex), - mConstraintSolver(mJoints, mConstrainedLinearVelocities, mConstrainedAngularVelocities, - mConstrainedPositions, mConstrainedOrientations, + mConstrainedLinearVelocities(NULL), mConstrainedAngularVelocities(NULL), + mContactSolver(mMapBodyToConstrainedVelocityIndex), + mConstraintSolver(mConstrainedPositions, mConstrainedOrientations, mMapBodyToConstrainedVelocityIndex), mNbVelocitySolverIterations(DEFAULT_VELOCITY_SOLVER_NB_ITERATIONS), mNbPositionSolverIterations(DEFAULT_POSITION_SOLVER_NB_ITERATIONS), - mIsSleepingEnabled(SPLEEPING_ENABLED), mNbIslands(0), mNbIslandsCapacity(0), - mIslands(NULL) { + mIsSleepingEnabled(SPLEEPING_ENABLED), mSplitLinearVelocities(NULL), + mSplitAngularVelocities(NULL), mNbIslands(0), mNbIslandsCapacity(0), + mIslands(NULL), mNbBodiesCapacity(0) { } @@ -73,8 +73,13 @@ DynamicsWorld::~DynamicsWorld() { mMemoryAllocator.release(mIslands, sizeof(Island*) * mNbIslandsCapacity); } - // Free the allocated memory for the constrained velocities - cleanupConstrainedVelocitiesArray(); + // Release the memory allocated for the bodies velocity arrays + if (mNbBodiesCapacity > 0) { + delete[] mSplitLinearVelocities; + delete[] mSplitAngularVelocities; + delete[] mConstrainedLinearVelocities; + delete[] mConstrainedAngularVelocities; + } #ifdef IS_PROFILING_ACTIVE @@ -137,12 +142,6 @@ void DynamicsWorld::update() { // Update the AABBs of the bodies updateRigidBodiesAABB(); - - // Cleanup of the contact solver - mContactSolver.cleanup(); - - // Cleanup the constrained velocities - cleanupConstrainedVelocitiesArray(); } // Compute and set the interpolation factor to all the bodies @@ -178,11 +177,10 @@ void DynamicsWorld::integrateRigidBodiesPositions() { rigidBody->setAngularVelocity(newAngVelocity); // Add the split impulse velocity from Contact Solver (only used to update the position) - if (mContactSolver.isConstrainedBody(rigidBody) && - mContactSolver.isSplitImpulseActive()) { + if (mContactSolver.isSplitImpulseActive()) { - newLinVelocity += mContactSolver.getSplitLinearVelocityOfBody(rigidBody); - newAngVelocity += mContactSolver.getSplitAngularVelocityOfBody(rigidBody); + newLinVelocity += mSplitLinearVelocities[indexArray]; + newAngVelocity += mSplitAngularVelocities[indexArray]; } // Get current position and orientation of the body @@ -239,6 +237,34 @@ void DynamicsWorld::setInterpolationFactorToAllBodies() { } } +// Initialize the bodies velocities arrays for the next simulation step. +void DynamicsWorld::initVelocityArrays() { + + // Allocate memory for the bodies velocity arrays + uint nbBodies = mRigidBodies.size(); + if (mNbBodiesCapacity != nbBodies && nbBodies > 0) { + if (mNbBodiesCapacity > 0) { + delete[] mSplitLinearVelocities; + delete[] mSplitAngularVelocities; + } + mNbBodiesCapacity = nbBodies; + mSplitLinearVelocities = new Vector3[mNbBodiesCapacity]; + mSplitAngularVelocities = new Vector3[mNbBodiesCapacity]; + mConstrainedLinearVelocities = new Vector3[mNbBodiesCapacity]; + mConstrainedAngularVelocities = new Vector3[mNbBodiesCapacity]; + assert(mSplitLinearVelocities != NULL); + assert(mSplitAngularVelocities != NULL); + assert(mConstrainedLinearVelocities != NULL); + assert(mConstrainedAngularVelocities != NULL); + } + + // Reset the velocities arrays + for (uint i=0; i(mRigidBodies.size(), Vector3(0, 0, 0)); - mConstrainedAngularVelocities = std::vector(mRigidBodies.size(), Vector3(0, 0, 0)); + // Initialize the bodies velocity arrays + initVelocityArrays(); decimal dt = static_cast(mTimer.getTimeStep()); // Fill in the mapping of rigid body to their index in the constrained // velocities arrays uint i = 0; + mMapBodyToConstrainedVelocityIndex.clear(); for (std::set::iterator it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { RigidBody* rigidBody = *it; mMapBodyToConstrainedVelocityIndex.insert(std::make_pair(rigidBody, i)); + assert(mSplitLinearVelocities[i] == Vector3(0, 0, 0)); + assert(mSplitAngularVelocities[i] == Vector3(0, 0, 0)); + // If the body is allowed to move if (rigidBody->getIsMotionEnabled()) { @@ -318,42 +347,59 @@ void DynamicsWorld::solveContactsAndConstraints() { // Get the current time step decimal dt = static_cast(mTimer.getTimeStep()); - // Check if there are contacts and constraints to solve - bool isConstraintsToSolve = !mJoints.empty(); - bool isContactsToSolve = !mContactManifolds.empty(); - if (!isConstraintsToSolve && !isContactsToSolve) return; + // Set the velocities arrays + mContactSolver.setSplitVelocitiesArrays(mSplitLinearVelocities, mSplitAngularVelocities); + mContactSolver.setConstrainedVelocitiesArrays(mConstrainedLinearVelocities, + mConstrainedAngularVelocities); + mConstraintSolver.setConstrainedVelocitiesArrays(mConstrainedLinearVelocities, + mConstrainedAngularVelocities); // ---------- Solve velocity constraints for joints and contacts ---------- // - // If there are contacts - if (isContactsToSolve) { + // For each island of the world + for (uint islandIndex = 0; islandIndex < mNbIslands; islandIndex++) { - // Initialize the solver - mContactSolver.initialize(dt); + // Check if there are contacts and constraints to solve + bool isConstraintsToSolve = mIslands[islandIndex]->getNbJoints() > 0; + bool isContactsToSolve = mIslands[islandIndex]->getNbContactManifolds() > 0; + if (!isConstraintsToSolve && !isContactsToSolve) continue; - // Warm start the contact solver - mContactSolver.warmStart(); + // If there are contacts in the current island + if (isContactsToSolve) { + + // Initialize the solver + mContactSolver.initializeForIsland(dt, mIslands[islandIndex]); + + // Warm start the contact solver + mContactSolver.warmStart(); + } + + // If there are constraints + if (isConstraintsToSolve) { + + // Initialize the constraint solver + mConstraintSolver.initializeForIsland(dt, mIslands[islandIndex]); + } + + // For each iteration of the velocity solver + for (uint i=0; i(mRigidBodies.size()); mConstrainedOrientations = std::vector(mRigidBodies.size()); - for (std::set::iterator it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { - // If it is a constrained bodies (by a joint) - if (mConstraintSolver.isConstrainedBody(*it)) { + // For each island of the world + for (uint islandIndex = 0; islandIndex < mNbIslands; islandIndex++) { - uint index = mMapBodyToConstrainedVelocityIndex.find(*it)->second; + // For each body of the island + RigidBody** bodies = mIslands[islandIndex]->getBodies(); + for (uint b=0; b < mIslands[islandIndex]->getNbBodies(); b++) { + + uint index = mMapBodyToConstrainedVelocityIndex.find(bodies[b])->second; // Get the position/orientation of the rigid body - const Transform& transform = (*it)->getTransform(); + const Transform& transform = bodies[b]->getTransform(); mConstrainedPositions[index] = transform.getPosition(); mConstrainedOrientations[index]= transform.getOrientation(); } - } - // ---------- Solve the position error correction for the constraints ---------- // + // ---------- Solve the position error correction for the constraints ---------- // - // For each iteration of the position (error correction) solver - for (uint i=0; i::iterator it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { + for (uint b=0; b < mIslands[islandIndex]->getNbBodies(); b++) { - // If it is a constrained bodies (by a joint) - if (mConstraintSolver.isConstrainedBody(*it)) { - - uint index = mMapBodyToConstrainedVelocityIndex.find(*it)->second; + uint index = mMapBodyToConstrainedVelocityIndex.find(bodies[b])->second; // Get the new position/orientation of the body const Vector3& newPosition = mConstrainedPositions[index]; @@ -407,22 +452,11 @@ void DynamicsWorld::solvePositionCorrection() { // Update the Transform of the body Transform newTransform(newPosition, newOrientation.getUnit()); - (*it)->setTransform(newTransform); + bodies[b]->setTransform(newTransform); } } } -// Cleanup the constrained velocities array at each step -void DynamicsWorld::cleanupConstrainedVelocitiesArray() { - - // Clear the constrained velocites - mConstrainedLinearVelocities.clear(); - mConstrainedAngularVelocities.clear(); - - // Clear the rigid body to velocities array index mapping - mMapBodyToConstrainedVelocityIndex.clear(); -} - // Create a rigid body into the physics world RigidBody* DynamicsWorld::createRigidBody(const Transform& transform, decimal mass, const Matrix3x3& inertiaTensorLocal, @@ -579,11 +613,13 @@ void DynamicsWorld::destroyJoint(Constraint* joint) { joint->mBody1->removeJointFromJointsList(mMemoryAllocator, joint); joint->mBody2->removeJointFromJointsList(mMemoryAllocator, joint); + size_t nbBytes = joint->getSizeInBytes(); + // Call the destructor of the joint joint->Constraint::~Constraint(); // Release the allocated memory - mMemoryAllocator.release(joint, joint->getSizeInBytes()); + mMemoryAllocator.release(joint, nbBytes); } // Add the joint to the list of joints of the two bodies involved in the joint diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index 89ebec5d..b1bc41ca 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -87,11 +87,17 @@ class DynamicsWorld : public CollisionWorld { /// Array of constrained linear velocities (state of the linear velocities /// after solving the constraints) - std::vector mConstrainedLinearVelocities; + Vector3* mConstrainedLinearVelocities; /// Array of constrained angular velocities (state of the angular velocities /// after solving the constraints) - std::vector mConstrainedAngularVelocities; + Vector3* mConstrainedAngularVelocities; + + /// Split linear velocities for the position contact solver (split impulse) + Vector3* mSplitLinearVelocities; + + /// Split angular velocities for the position contact solver (split impulse) + Vector3* mSplitAngularVelocities; /// Array of constrained rigid bodies position (for position error correction) std::vector mConstrainedPositions; @@ -111,6 +117,9 @@ class DynamicsWorld : public CollisionWorld { /// Array with all the islands of awaken bodies Island** mIslands; + /// Current allocated capacity for the bodies + uint mNbBodiesCapacity; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -132,6 +141,9 @@ class DynamicsWorld : public CollisionWorld { /// Compute and set the interpolation factor to all bodies void setInterpolationFactorToAllBodies(); + /// Initialize the bodies velocities arrays for the next simulation step. + void initVelocityArrays(); + /// Integrate the velocities of rigid bodies. void integrateRigidBodiesVelocities(); From 8db7823433deac41c7ca47212fbc72786beafc17 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 3 Sep 2013 19:30:43 +0200 Subject: [PATCH 51/66] Continue to implement the sleeping technique --- src/body/Body.cpp | 3 +- src/body/Body.h | 17 +- src/body/RigidBody.h | 16 ++ src/configuration.h | 13 +- src/constraint/HingeJoint.cpp | 16 +- src/constraint/SliderJoint.cpp | 16 +- src/engine/DynamicsWorld.cpp | 279 ++++++++++++++++++++++----------- src/engine/DynamicsWorld.h | 102 ++++++++++-- 8 files changed, 350 insertions(+), 112 deletions(-) diff --git a/src/body/Body.cpp b/src/body/Body.cpp index 243b0c3f..5cdff7d2 100644 --- a/src/body/Body.cpp +++ b/src/body/Body.cpp @@ -32,7 +32,8 @@ using namespace reactphysics3d; // Constructor Body::Body(bodyindex id) - : mID(id), mIsAlreadyInIsland(false), mIsAllowedToSleep(true), mIsSleeping(false) { + : mID(id), mIsAlreadyInIsland(false), mIsAllowedToSleep(true), mIsSleeping(false), + mSleepTime(0) { } diff --git a/src/body/Body.h b/src/body/Body.h index f0358570..299de91f 100644 --- a/src/body/Body.h +++ b/src/body/Body.h @@ -56,6 +56,9 @@ class Body { /// True if the body is sleeping (for sleeping technique) bool mIsSleeping; + /// Elapsed time since the body velocity was bellow the sleep velocity + decimal mSleepTime; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -93,7 +96,7 @@ class Body { bool isSleeping() const; /// Set the variable to know whether or not the body is sleeping - void setIsSleeping(bool isSleeping); + virtual void setIsSleeping(bool isSleeping); /// Smaller than operator bool operator<(const Body& body2) const; @@ -135,6 +138,8 @@ inline bool Body::isAllowedToSleep() const { // Set whether or not the body is allowed to go to sleep inline void Body::setIsAllowedToSleep(bool isAllowedToSleep) { mIsAllowedToSleep = isAllowedToSleep; + + if (!mIsAllowedToSleep) setIsSleeping(false); } // Return whether or not the body is sleeping @@ -144,6 +149,16 @@ inline bool Body::isSleeping() const { // Set the variable to know whether or not the body is sleeping inline void Body::setIsSleeping(bool isSleeping) { + + if (isSleeping) { + mSleepTime = decimal(0.0); + } + else { + if (mIsSleeping) { + mSleepTime = decimal(0.0); + } + } + mIsSleeping = isSleeping; } diff --git a/src/body/RigidBody.h b/src/body/RigidBody.h index 2856bc64..1d38d610 100644 --- a/src/body/RigidBody.h +++ b/src/body/RigidBody.h @@ -194,6 +194,9 @@ class RigidBody : public CollisionBody { /// Return the first element of the linked list of joints involving this body const JointListElement* getJointsList() const; + /// Set the variable to know whether or not the body is sleeping + virtual void setIsSleeping(bool isSleeping); + // -------------------- Friendship -------------------- // friend class DynamicsWorld; @@ -351,6 +354,19 @@ inline const JointListElement* RigidBody::getJointsList() const { return mJointsList; } +// Set the variable to know whether or not the body is sleeping +inline void RigidBody::setIsSleeping(bool isSleeping) { + + if (isSleeping) { + mLinearVelocity.setToZero(); + mAngularVelocity.setToZero(); + mExternalForce.setToZero(); + mExternalTorque.setToZero(); + } + + Body::setIsSleeping(isSleeping); +} + } #endif diff --git a/src/configuration.h b/src/configuration.h index 3cb8c3f0..7141ab18 100644 --- a/src/configuration.h +++ b/src/configuration.h @@ -92,7 +92,7 @@ const decimal DEFAULT_FRICTION_COEFFICIENT = decimal(0.3); /// Default bounciness factor for a rigid body const decimal DEFAULT_BOUNCINESS = decimal(0.5); -/// True if the spleeping technique for inactive bodies is enabled +/// True if the spleeping technique is enabled const bool SPLEEPING_ENABLED = true; /// Object margin for collision detection in meters (for the GJK-EPA Algorithm) @@ -110,6 +110,17 @@ const uint DEFAULT_VELOCITY_SOLVER_NB_ITERATIONS = 15; /// Number of iterations when solving the position constraints of the Sequential Impulse technique const uint DEFAULT_POSITION_SOLVER_NB_ITERATIONS = 5; +/// Time (in seconds) that a body must stay still to be considered sleeping +const float DEFAULT_TIME_BEFORE_SLEEP = 1.0f; + +/// A body with a linear velocity smaller than the sleep linear velocity (in m/s) +/// might enter sleeping mode. +const decimal DEFAULT_SLEEP_LINEAR_VELOCITY = decimal(0.02); + +/// A body with angular velocity smaller than the sleep angular velocity (in rad/s) +/// might enter sleeping mode +const decimal DEFAULT_SLEEP_ANGULAR_VELOCITY = decimal(3.0 * (PI / 180.0)); + } #endif diff --git a/src/constraint/HingeJoint.cpp b/src/constraint/HingeJoint.cpp index 30b929b7..b3eb9f9b 100644 --- a/src/constraint/HingeJoint.cpp +++ b/src/constraint/HingeJoint.cpp @@ -732,7 +732,9 @@ void HingeJoint::enableMotor(bool isMotorEnabled) { mIsMotorEnabled = isMotorEnabled; mImpulseMotor = 0.0; - // TODO : Wake up the bodies of the joint here when sleeping is implemented + // Wake up the two bodies of the joint + mBody1->setIsSleeping(false); + mBody2->setIsSleeping(false); } // Set the minimum angle limit @@ -770,7 +772,9 @@ void HingeJoint::resetLimits() { mImpulseLowerLimit = 0.0; mImpulseUpperLimit = 0.0; - // TODO : Wake up the bodies of the joint here when sleeping is implemented + // Wake up the two bodies of the joint + mBody1->setIsSleeping(false); + mBody2->setIsSleeping(false); } // Set the motor speed @@ -780,7 +784,9 @@ void HingeJoint::setMotorSpeed(decimal motorSpeed) { mMotorSpeed = motorSpeed; - // TODO : Wake up the bodies of the joint here when sleeping is implemented + // Wake up the two bodies of the joint + mBody1->setIsSleeping(false); + mBody2->setIsSleeping(false); } } @@ -792,7 +798,9 @@ void HingeJoint::setMaxMotorTorque(decimal maxMotorTorque) { assert(mMaxMotorTorque >= 0.0); mMaxMotorTorque = maxMotorTorque; - // TODO : Wake up the bodies of the joint here when sleeping is implemented + // Wake up the two bodies of the joint + mBody1->setIsSleeping(false); + mBody2->setIsSleeping(false); } } diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index e1f2daed..9a0f1321 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -768,7 +768,9 @@ void SliderJoint::enableMotor(bool isMotorEnabled) { mIsMotorEnabled = isMotorEnabled; mImpulseMotor = 0.0; - // TODO : Wake up the bodies of the joint here when sleeping is implemented + // Wake up the two bodies of the joint + mBody1->setIsSleeping(false); + mBody2->setIsSleeping(false); } // Return the current translation value of the joint @@ -830,7 +832,9 @@ void SliderJoint::resetLimits() { mImpulseLowerLimit = 0.0; mImpulseUpperLimit = 0.0; - // TODO : Wake up the bodies of the joint here when sleeping is implemented + // Wake up the two bodies of the joint + mBody1->setIsSleeping(false); + mBody2->setIsSleeping(false); } // Set the motor speed @@ -840,7 +844,9 @@ void SliderJoint::setMotorSpeed(decimal motorSpeed) { mMotorSpeed = motorSpeed; - // TODO : Wake up the bodies of the joint here when sleeping is implemented + // Wake up the two bodies of the joint + mBody1->setIsSleeping(false); + mBody2->setIsSleeping(false); } } @@ -852,6 +858,8 @@ void SliderJoint::setMaxMotorForce(decimal maxMotorForce) { assert(mMaxMotorForce >= 0.0); mMaxMotorForce = maxMotorForce; - // TODO : Wake up the bodies of the joint here when sleeping is implemented + // Wake up the two bodies of the joint + mBody1->setIsSleeping(false); + mBody2->setIsSleeping(false); } } diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 511f4630..cf585e43 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -36,7 +36,7 @@ using namespace std; // Constructor DynamicsWorld::DynamicsWorld(const Vector3 &gravity, decimal timeStep = DEFAULT_TIMESTEP) - : CollisionWorld(), mTimer(timeStep), mGravity(gravity), mIsGravityOn(true), + : CollisionWorld(), mTimer(timeStep), mGravity(gravity), mIsGravityEnabled(true), mConstrainedLinearVelocities(NULL), mConstrainedAngularVelocities(NULL), mContactSolver(mMapBodyToConstrainedVelocityIndex), mConstraintSolver(mConstrainedPositions, mConstrainedOrientations, @@ -45,7 +45,10 @@ DynamicsWorld::DynamicsWorld(const Vector3 &gravity, decimal timeStep = DEFAULT_ mNbPositionSolverIterations(DEFAULT_POSITION_SOLVER_NB_ITERATIONS), mIsSleepingEnabled(SPLEEPING_ENABLED), mSplitLinearVelocities(NULL), mSplitAngularVelocities(NULL), mNbIslands(0), mNbIslandsCapacity(0), - mIslands(NULL), mNbBodiesCapacity(0) { + mIslands(NULL), mNbBodiesCapacity(0), + mSleepLinearVelocity(DEFAULT_SLEEP_LINEAR_VELOCITY), + mSleepAngularVelocity(DEFAULT_SLEEP_ANGULAR_VELOCITY), + mTimeBeforeSleep(DEFAULT_TIME_BEFORE_SLEEP) { } @@ -119,6 +122,9 @@ void DynamicsWorld::update() { // Compute the collision detection mCollisionDetection.computeCollisionDetection(); + // Compute the islands (separate groups of bodies with constraints between each others) + computeIslands(); + // Integrate the velocities integrateRigidBodiesVelocities(); @@ -128,9 +134,6 @@ void DynamicsWorld::update() { // Update the timer mTimer.nextStep(); - // Compute the islands (separate groups of bodies with constraints between each others) - computeIslands(); - // Solve the contacts and constraints solveContactsAndConstraints(); @@ -140,6 +143,8 @@ void DynamicsWorld::update() { // Solve the position correction for constraints solvePositionCorrection(); + if (mIsSleepingEnabled) updateSleepingBodies(); + // Update the AABBs of the bodies updateRigidBodiesAABB(); } @@ -157,44 +162,47 @@ void DynamicsWorld::integrateRigidBodiesPositions() { decimal dt = static_cast(mTimer.getTimeStep()); - // For each rigid body of the world - set::iterator it; - for (it = getRigidBodiesBeginIterator(); it != getRigidBodiesEndIterator(); ++it) { + // For each island of the world + for (uint i=0; i < mNbIslands; i++) { - RigidBody* rigidBody = *it; - assert(rigidBody != NULL); + RigidBody** bodies = mIslands[i]->getBodies(); - // If the body is allowed to move - if (rigidBody->getIsMotionEnabled()) { + // For each body of the island + for (uint b=0; b < mIslands[i]->getNbBodies(); b++) { - // Get the constrained velocity - uint indexArray = mMapBodyToConstrainedVelocityIndex.find(rigidBody)->second; - Vector3 newLinVelocity = mConstrainedLinearVelocities[indexArray]; - Vector3 newAngVelocity = mConstrainedAngularVelocities[indexArray]; + // If the body is allowed to move + if (bodies[b]->getIsMotionEnabled()) { - // Update the linear and angular velocity of the body - rigidBody->setLinearVelocity(newLinVelocity); - rigidBody->setAngularVelocity(newAngVelocity); + // Get the constrained velocity + uint indexArray = mMapBodyToConstrainedVelocityIndex.find(bodies[b])->second; + Vector3 newLinVelocity = mConstrainedLinearVelocities[indexArray]; + Vector3 newAngVelocity = mConstrainedAngularVelocities[indexArray]; - // Add the split impulse velocity from Contact Solver (only used to update the position) - if (mContactSolver.isSplitImpulseActive()) { + // Update the linear and angular velocity of the body + bodies[b]->setLinearVelocity(newLinVelocity); + bodies[b]->setAngularVelocity(newAngVelocity); - newLinVelocity += mSplitLinearVelocities[indexArray]; - newAngVelocity += mSplitAngularVelocities[indexArray]; + // Add the split impulse velocity from Contact Solver (only used + // to update the position) + if (mContactSolver.isSplitImpulseActive()) { + + newLinVelocity += mSplitLinearVelocities[indexArray]; + newAngVelocity += mSplitAngularVelocities[indexArray]; + } + + // Get current position and orientation of the body + const Vector3& currentPosition = bodies[b]->getTransform().getPosition(); + const Quaternion& currentOrientation = bodies[b]->getTransform().getOrientation(); + + // Compute the new position of the body + Vector3 newPosition = currentPosition + newLinVelocity * dt; + Quaternion newOrientation = currentOrientation + Quaternion(0, newAngVelocity) * + currentOrientation * decimal(0.5) * dt; + + // Update the Transform of the body + Transform newTransform(newPosition, newOrientation.getUnit()); + bodies[b]->setTransform(newTransform); } - - // Get current position and orientation of the body - const Vector3& currentPosition = rigidBody->getTransform().getPosition(); - const Quaternion& currentOrientation = rigidBody->getTransform().getOrientation(); - - // Compute the new position of the body - Vector3 newPosition = currentPosition + newLinVelocity * dt; - Quaternion newOrientation = currentOrientation + Quaternion(0, newAngVelocity) * - currentOrientation * decimal(0.5) * dt; - - // Update the Transform of the body - Transform newTransform(newPosition, newOrientation.getUnit()); - rigidBody->setTransform(newTransform); } } } @@ -206,7 +214,7 @@ void DynamicsWorld::updateRigidBodiesAABB() { // For each rigid body of the world set::iterator it; - for (it = getRigidBodiesBeginIterator(); it != getRigidBodiesEndIterator(); ++it) { + for (it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { // If the body has moved if ((*it)->getHasMoved()) { @@ -228,12 +236,9 @@ void DynamicsWorld::setInterpolationFactorToAllBodies() { // Set the factor to all bodies set::iterator it; - for (it = getRigidBodiesBeginIterator(); it != getRigidBodiesEndIterator(); ++it) { + for (it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { - RigidBody* rigidBody = dynamic_cast(*it); - assert(rigidBody); - - rigidBody->setInterpolationFactor(factor); + (*it)->setInterpolationFactor(factor); } } @@ -263,6 +268,18 @@ void DynamicsWorld::initVelocityArrays() { mSplitLinearVelocities[i].setToZero(); mSplitAngularVelocities[i].setToZero(); } + + // Initialize the map of body indexes in the velocity arrays + mMapBodyToConstrainedVelocityIndex.clear(); + std::set::const_iterator it; + uint indexBody = 0; + for (it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { + + // Add the body into the map + mMapBodyToConstrainedVelocityIndex.insert(std::make_pair(*it, indexBody)); + indexBody++; + } } // Integrate the velocities of rigid bodies. @@ -279,64 +296,69 @@ void DynamicsWorld::integrateRigidBodiesVelocities() { decimal dt = static_cast(mTimer.getTimeStep()); - // Fill in the mapping of rigid body to their index in the constrained - // velocities arrays - uint i = 0; - mMapBodyToConstrainedVelocityIndex.clear(); - for (std::set::iterator it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { - RigidBody* rigidBody = *it; - mMapBodyToConstrainedVelocityIndex.insert(std::make_pair(rigidBody, i)); + // For each island of the world + for (uint i=0; i < mNbIslands; i++) { - assert(mSplitLinearVelocities[i] == Vector3(0, 0, 0)); - assert(mSplitAngularVelocities[i] == Vector3(0, 0, 0)); + RigidBody** bodies = mIslands[i]->getBodies(); - // If the body is allowed to move - if (rigidBody->getIsMotionEnabled()) { + // For each body of the island + for (uint b=0; b < mIslands[i]->getNbBodies(); b++) { - // Integrate the external force to get the new velocity of the body - mConstrainedLinearVelocities[i] = rigidBody->getLinearVelocity() + - dt * rigidBody->getMassInverse() * rigidBody->getExternalForce(); - mConstrainedAngularVelocities[i] = rigidBody->getAngularVelocity() + - dt * rigidBody->getInertiaTensorInverseWorld() * rigidBody->getExternalTorque(); + // Insert the body into the map of constrained velocities + uint indexBody = mMapBodyToConstrainedVelocityIndex.find(bodies[b])->second; - // If the gravity has to be applied to this rigid body - if (rigidBody->isGravityEnabled() && mIsGravityOn) { + assert(mSplitLinearVelocities[indexBody] == Vector3(0, 0, 0)); + assert(mSplitAngularVelocities[indexBody] == Vector3(0, 0, 0)); - // Integrate the gravity force - mConstrainedLinearVelocities[i] += dt * rigidBody->getMassInverse() * - rigidBody->getMass() * mGravity; + // If the body is allowed to move + if (bodies[b]->getIsMotionEnabled()) { + + // Integrate the external force to get the new velocity of the body + mConstrainedLinearVelocities[indexBody] = bodies[b]->getLinearVelocity() + + dt * bodies[b]->getMassInverse() * bodies[b]->getExternalForce(); + mConstrainedAngularVelocities[indexBody] = bodies[b]->getAngularVelocity() + + dt * bodies[b]->getInertiaTensorInverseWorld() * + bodies[b]->getExternalTorque(); + + // If the gravity has to be applied to this rigid body + if (bodies[b]->isGravityEnabled() && mIsGravityEnabled) { + + // Integrate the gravity force + mConstrainedLinearVelocities[indexBody] += dt * bodies[b]->getMassInverse() * + bodies[b]->getMass() * mGravity; + } + + // Apply the velocity damping + // Damping force : F_c = -c' * v (c=damping factor) + // Equation : m * dv/dt = -c' * v + // => dv/dt = -c * v (with c=c'/m) + // => dv/dt + c * v = 0 + // Solution : v(t) = v0 * e^(-c * t) + // => v(t + dt) = v0 * e^(-c(t + dt)) + // = v0 * e^(-ct) * e^(-c * dt) + // = v(t) * e^(-c * dt) + // => v2 = v1 * e^(-c * dt) + // Using Taylor Serie for e^(-x) : e^x ~ 1 + x + x^2/2! + ... + // => e^(-x) ~ 1 - x + // => v2 = v1 * (1 - c * dt) + decimal linDampingFactor = bodies[b]->getLinearDamping(); + decimal angDampingFactor = bodies[b]->getAngularDamping(); + decimal linearDamping = clamp(decimal(1.0) - dt * linDampingFactor, + decimal(0.0), decimal(1.0)); + decimal angularDamping = clamp(decimal(1.0) - dt * angDampingFactor, + decimal(0.0), decimal(1.0)); + mConstrainedLinearVelocities[indexBody] *= clamp(linearDamping, decimal(0.0), + decimal(1.0)); + mConstrainedAngularVelocities[indexBody] *= clamp(angularDamping, decimal(0.0), + decimal(1.0)); + + // Update the old Transform of the body + bodies[b]->updateOldTransform(); } - // Apply the velocity damping - // Damping force : F_c = -c' * v (c=damping factor) - // Equation : m * dv/dt = -c' * v - // => dv/dt = -c * v (with c=c'/m) - // => dv/dt + c * v = 0 - // Solution : v(t) = v0 * e^(-c * t) - // => v(t + dt) = v0 * e^(-c(t + dt)) - // = v0 * e^(-ct) * e^(-c * dt) - // = v(t) * e^(-c * dt) - // => v2 = v1 * e^(-c * dt) - // Using Taylor Serie for e^(-x) : e^x ~ 1 + x + x^2/2! + ... - // => e^(-x) ~ 1 - x - // => v2 = v1 * (1 - c * dt) - decimal linDampingFactor = rigidBody->getLinearDamping(); - decimal angDampingFactor = rigidBody->getAngularDamping(); - decimal linearDamping = clamp(decimal(1.0) - dt * linDampingFactor, - decimal(0.0), decimal(1.0)); - decimal angularDamping = clamp(decimal(1.0) - dt * angDampingFactor, - decimal(0.0), decimal(1.0)); - mConstrainedLinearVelocities[i] *= clamp(linearDamping, decimal(0.0), decimal(1.0)); - mConstrainedAngularVelocities[i] *= clamp(angularDamping, decimal(0.0), decimal(1.0)); - - // Update the old Transform of the body - rigidBody->updateOldTransform(); + indexBody++; } - - i++; } - - assert(mMapBodyToConstrainedVelocityIndex.size() == mRigidBodies.size()); } // Solve the contacts and constraints @@ -606,6 +628,10 @@ void DynamicsWorld::destroyJoint(Constraint* joint) { mCollisionDetection.removeNoCollisionPair(joint->getBody1(), joint->getBody2()); } + // Wake up the two bodies of the joint + joint->getBody1()->setIsSleeping(false); + joint->getBody2()->setIsSleeping(false); + // Remove the joint from the world mJoints.erase(joint); @@ -842,6 +868,61 @@ void DynamicsWorld::computeIslands() { mMemoryAllocator.release(stackBodiesToVisit, nbBytesStack); } +// Put bodies to sleep if needed. +/// For each island, if all the bodies have been almost still for a long enough period of +/// time, we put all the bodies of the island to sleep. +void DynamicsWorld::updateSleepingBodies() { + + PROFILE("DynamicsWorld::updateSleepingBodies()"); + + const decimal dt = static_cast(mTimer.getTimeStep()); + const decimal sleepLinearVelocitySquare = mSleepLinearVelocity * mSleepLinearVelocity; + const decimal sleepAngularVelocitySquare = mSleepAngularVelocity * mSleepAngularVelocity; + + // For each island of the world + for (uint i=0; igetBodies(); + for (uint b=0; b < mIslands[i]->getNbBodies(); b++) { + + // Skip static bodies + if (!bodies[b]->getIsMotionEnabled()) continue; + + // If the body is velocity is large enough to stay awake + if (bodies[b]->getLinearVelocity().lengthSquare() > sleepLinearVelocitySquare || + bodies[b]->getAngularVelocity().lengthSquare() > sleepAngularVelocitySquare || + !bodies[b]->isAllowedToSleep()) { + + // Reset the sleep time of the body + bodies[b]->mSleepTime = decimal(0.0); + minSleepTime = decimal(0.0); + } + else { // If the body velocity is bellow the sleeping velocity threshold + + // Increase the sleep time + bodies[b]->mSleepTime += dt; + if (bodies[b]->mSleepTime < minSleepTime) { + minSleepTime = bodies[b]->mSleepTime; + } + } + } + + // If the velocity of all the bodies of the island is under the + // sleeping velocity threshold for a period of time larger than + // the time required to become a sleeping body + if (minSleepTime >= mTimeBeforeSleep) { + + // Put all the bodies of the island to sleep + for (uint b=0; b < mIslands[i]->getNbBodies(); b++) { + bodies[b]->setIsSleeping(true); + } + } + } +} + // Notify the world about a new broad-phase overlapping pair void DynamicsWorld::notifyAddedOverlappingPair(const BroadPhasePair* addedPair) { @@ -894,3 +975,19 @@ void DynamicsWorld::notifyNewContact(const BroadPhasePair* broadPhasePair, addContactManifoldToBody(overlappingPair->getContactManifold(), overlappingPair->mBody1, overlappingPair->mBody2); } + +// Enable/Disable the sleeping technique +void DynamicsWorld::enableSleeping(bool isSleepingEnabled) { + mIsSleepingEnabled = isSleepingEnabled; + + if (!mIsSleepingEnabled) { + + // For each body of the world + std::set::iterator it; + for (it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { + + // Wake up the rigid body + (*it)->setIsSleeping(false); + } + } +} diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index b1bc41ca..7f783ffd 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -83,7 +83,7 @@ class DynamicsWorld : public CollisionWorld { Vector3 mGravity; /// True if the gravity force is on - bool mIsGravityOn; + bool mIsGravityEnabled; /// Array of constrained linear velocities (state of the linear velocities /// after solving the constraints) @@ -120,6 +120,16 @@ class DynamicsWorld : public CollisionWorld { /// Current allocated capacity for the bodies uint mNbBodiesCapacity; + /// Sleep linear velocity threshold + decimal mSleepLinearVelocity; + + /// Sleep angular velocity threshold + decimal mSleepAngularVelocity; + + /// Time (in seconds) before a body is put to sleep if its velocity + /// becomes smaller than the sleep velocity. + decimal mTimeBeforeSleep; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -162,6 +172,9 @@ class DynamicsWorld : public CollisionWorld { /// Compute the islands of awake bodies. void computeIslands(); + /// Put bodies to sleep if needed. + void updateSleepingBodies(); + /// Update the overlapping pair virtual void updateOverlappingPair(const BroadPhasePair* pair); @@ -239,10 +252,10 @@ public : Vector3 getGravity() const; /// Return if the gravity is on - bool getIsGravityOn() const; + bool isGravityEnabled() const; - /// Set the isGravityOn attribute - void setIsGratityOn(bool isGravityOn); + /// Enable/Disable the gravity + void setIsGratityEnabled(bool isGravityEnabled); /// Return the number of rigid bodies in the world uint getNbRigidBodies() const; @@ -265,6 +278,30 @@ public : /// Return a reference to the contact manifolds of the world const std::vector& getContactManifolds() const; + /// Return true if the sleeping technique is enabled + bool isSleepingEnabled() const; + + /// Enable/Disable the sleeping technique + void enableSleeping(bool isSleepingEnabled); + + /// Return the current sleep linear velocity + decimal getSleepLinearVelocity() const; + + /// Set the sleep linear velocity. + void setSleepLinearVelocity(decimal sleepLinearVelocity); + + /// Return the current sleep angular velocity + decimal getSleepAngularVelocity() const; + + /// Set the sleep angular velocity. + void setSleepAngularVelocity(decimal sleepAngularVelocity); + + /// Return the time a body is required to stay still before sleeping + decimal getTimeBeforeSleep() const; + + /// Set the time a body is required to stay still before sleeping + void setTimeBeforeSleep(decimal timeBeforeSleep); + // TODO : REMOVE THIS Island** getIslands() { return mIslands;} @@ -349,14 +386,14 @@ inline Vector3 DynamicsWorld::getGravity() const { return mGravity; } -// Return if the gravity is on -inline bool DynamicsWorld::getIsGravityOn() const { - return mIsGravityOn; +// Return if the gravity is enaled +inline bool DynamicsWorld::isGravityEnabled() const { + return mIsGravityEnabled; } -// Set the isGravityOn attribute -inline void DynamicsWorld::setIsGratityOn(bool isGravityOn) { - mIsGravityOn = isGravityOn; +// Enable/Disable the gravity +inline void DynamicsWorld::setIsGratityEnabled(bool isGravityEnabled) { + mIsGravityEnabled = isGravityEnabled; } // Return the number of rigid bodies in the world @@ -394,6 +431,51 @@ inline long double DynamicsWorld::getPhysicsTime() const { return mTimer.getPhysicsTime(); } +// Return true if the sleeping technique is enabled +inline bool DynamicsWorld::isSleepingEnabled() const { + return mIsSleepingEnabled; +} + +// Return the current sleep linear velocity +inline decimal DynamicsWorld::getSleepLinearVelocity() const { + return mSleepLinearVelocity; +} + +// Set the sleep linear velocity. +/// When the velocity of a body becomes smaller than the sleep linear/angular +/// velocity for a given amount of time, the body starts sleeping and does not need +/// to be simulated anymore. +inline void DynamicsWorld::setSleepLinearVelocity(decimal sleepLinearVelocity) { + assert(sleepLinearVelocity >= decimal(0.0)); + mSleepLinearVelocity = sleepLinearVelocity; +} + +// Return the current sleep angular velocity +inline decimal DynamicsWorld::getSleepAngularVelocity() const { + return mSleepAngularVelocity; +} + +// Set the sleep angular velocity. +/// When the velocity of a body becomes smaller than the sleep linear/angular +/// velocity for a given amount of time, the body starts sleeping and does not need +/// to be simulated anymore. +inline void DynamicsWorld::setSleepAngularVelocity(decimal sleepAngularVelocity) { + assert(sleepAngularVelocity >= decimal(0.0)); + mSleepAngularVelocity = sleepAngularVelocity; +} + +// Return the time a body is required to stay still before sleeping +inline decimal DynamicsWorld::getTimeBeforeSleep() const { + return mTimeBeforeSleep; +} + + +// Set the time a body is required to stay still before sleeping +inline void DynamicsWorld::setTimeBeforeSleep(decimal timeBeforeSleep) { + assert(timeBeforeSleep >= decimal(0.0)); + mTimeBeforeSleep = timeBeforeSleep; +} + } #endif From a009debf21e2b5cb272de338754db2f1f2a9a817 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 3 Sep 2013 19:31:50 +0200 Subject: [PATCH 52/66] Fix issue in the contact solver that reduces jittering --- src/engine/ContactSolver.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/engine/ContactSolver.cpp b/src/engine/ContactSolver.cpp index f70abbec..b40eb8f2 100644 --- a/src/engine/ContactSolver.cpp +++ b/src/engine/ContactSolver.cpp @@ -266,10 +266,10 @@ void ContactSolver::initializeContactConstraints() { // Compute the restitution velocity bias "b". We compute this here instead // of inside the solve() method because we need to use the velocity difference // at the beginning of the contact. Note that if it is a resting contact (normal - // velocity under a given threshold), we don't add a restitution velocity bias + // velocity bellow a given threshold), we do not add a restitution velocity bias contactPoint.restitutionBias = 0.0; decimal deltaVDotN = deltaV.dot(contactPoint.normal); - if (deltaVDotN < RESTITUTION_VELOCITY_THRESHOLD) { + if (deltaVDotN < -RESTITUTION_VELOCITY_THRESHOLD) { contactPoint.restitutionBias = manifold.restitutionFactor * deltaVDotN; } From d4c7eee1755c1604b4bc52bf9afb872226fb99ed Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sat, 7 Sep 2013 10:57:58 +0200 Subject: [PATCH 53/66] Rename the Constraint class into Joint and do not perform collision detection between two sleeping bodies --- src/body/Body.cpp | 4 +- src/body/Body.h | 11 ++ src/body/CollisionBody.cpp | 4 +- src/body/CollisionBody.h | 19 ---- src/body/RigidBody.cpp | 4 +- src/body/RigidBody.h | 101 ++++++++++++------- src/collision/CollisionDetection.cpp | 3 + src/constraint/BallAndSocketJoint.cpp | 2 +- src/constraint/BallAndSocketJoint.h | 8 +- src/constraint/ContactPoint.cpp | 23 +---- src/constraint/ContactPoint.h | 56 ++++++---- src/constraint/FixedJoint.cpp | 2 +- src/constraint/FixedJoint.h | 8 +- src/constraint/HingeJoint.cpp | 2 +- src/constraint/HingeJoint.h | 12 +-- src/constraint/{Constraint.cpp => Joint.cpp} | 13 ++- src/constraint/{Constraint.h => Joint.h} | 87 ++++++++-------- src/constraint/SliderJoint.cpp | 2 +- src/constraint/SliderJoint.h | 10 +- src/engine/CollisionWorld.h | 2 +- src/engine/ConstraintSolver.cpp | 6 +- src/engine/ConstraintSolver.h | 2 +- src/engine/ContactSolver.h | 2 +- src/engine/DynamicsWorld.cpp | 28 ++--- src/engine/DynamicsWorld.h | 22 +++- src/engine/Island.cpp | 4 +- src/engine/Island.h | 12 +-- 27 files changed, 242 insertions(+), 207 deletions(-) rename src/constraint/{Constraint.cpp => Joint.cpp} (83%) rename src/constraint/{Constraint.h => Joint.h} (74%) diff --git a/src/body/Body.cpp b/src/body/Body.cpp index 5cdff7d2..f34cf5cc 100644 --- a/src/body/Body.cpp +++ b/src/body/Body.cpp @@ -32,8 +32,8 @@ using namespace reactphysics3d; // Constructor Body::Body(bodyindex id) - : mID(id), mIsAlreadyInIsland(false), mIsAllowedToSleep(true), mIsSleeping(false), - mSleepTime(0) { + : mID(id), mIsAlreadyInIsland(false), mIsAllowedToSleep(true), mIsActive(true), + mIsSleeping(false), mSleepTime(0) { } diff --git a/src/body/Body.h b/src/body/Body.h index 299de91f..af8ad635 100644 --- a/src/body/Body.h +++ b/src/body/Body.h @@ -53,6 +53,9 @@ class Body { /// True if the body is allowed to go to sleep for better efficiency bool mIsAllowedToSleep; + /// True if the body is active + bool mIsActive; + /// True if the body is sleeping (for sleeping technique) bool mIsSleeping; @@ -95,6 +98,9 @@ class Body { /// Return whether or not the body is sleeping bool isSleeping() const; + /// Return true if the body is active + bool isActive() const; + /// Set the variable to know whether or not the body is sleeping virtual void setIsSleeping(bool isSleeping); @@ -147,6 +153,11 @@ inline bool Body::isSleeping() const { return mIsSleeping; } +// Return true if the body is active +inline bool Body::isActive() const { + return mIsActive; +} + // Set the variable to know whether or not the body is sleeping inline void Body::setIsSleeping(bool isSleeping) { diff --git a/src/body/CollisionBody.cpp b/src/body/CollisionBody.cpp index 0ba418b4..cebf9559 100644 --- a/src/body/CollisionBody.cpp +++ b/src/body/CollisionBody.cpp @@ -33,8 +33,8 @@ using namespace reactphysics3d; // Constructor CollisionBody::CollisionBody(const Transform& transform, CollisionShape *collisionShape, bodyindex id) - : Body(id), mCollisionShape(collisionShape), mTransform(transform), - mIsActive(true), mHasMoved(false), mContactManifoldsList(NULL) { + : Body(id), mCollisionShape(collisionShape), mTransform(transform), + mHasMoved(false), mContactManifoldsList(NULL) { assert(collisionShape); diff --git a/src/body/CollisionBody.h b/src/body/CollisionBody.h index e050f8e3..399bd971 100644 --- a/src/body/CollisionBody.h +++ b/src/body/CollisionBody.h @@ -65,9 +65,6 @@ class CollisionBody : public Body { /// Interpolation factor used for the state interpolation decimal mInterpolationFactor; - /// True if the body is active (not sleeping) - bool mIsActive; - /// True if the body is able to move bool mIsMotionEnabled; @@ -116,12 +113,6 @@ class CollisionBody : public Body { /// Set the collision shape void setCollisionShape(CollisionShape* collisionShape); - /// Return true for an active body - bool getIsActive() const; - - /// Set the isActive variable - void setIsActive(bool isActive); - /// Return the current position and orientation const Transform& getTransform() const; @@ -185,16 +176,6 @@ inline void CollisionBody::setCollisionShape(CollisionShape* collisionShape) { mCollisionShape = collisionShape; } -// Return true if the body is active -inline bool CollisionBody::getIsActive() const { - return mIsActive; -} - -// Set the isActive variable -inline void CollisionBody::setIsActive(bool isActive) { - mIsActive = isActive; -} - // Return the interpolated transform for rendering inline Transform CollisionBody::getInterpolatedTransform() const { return Transform::interpolateTransforms(mOldTransform, mTransform, mInterpolationFactor); diff --git a/src/body/RigidBody.cpp b/src/body/RigidBody.cpp index 82e5c928..a3d0f8fd 100644 --- a/src/body/RigidBody.cpp +++ b/src/body/RigidBody.cpp @@ -25,7 +25,7 @@ // Libraries #include "RigidBody.h" -#include "constraint/Constraint.h" +#include "constraint/Joint.h" #include "../collision/shapes/CollisionShape.h" // We want to use the ReactPhysics3D namespace @@ -48,7 +48,7 @@ RigidBody::~RigidBody() { } // Remove a joint from the joints list -void RigidBody::removeJointFromJointsList(MemoryAllocator& memoryAllocator, const Constraint* joint) { +void RigidBody::removeJointFromJointsList(MemoryAllocator& memoryAllocator, const Joint* joint) { assert(joint != NULL); assert(mJointsList != NULL); diff --git a/src/body/RigidBody.h b/src/body/RigidBody.h index 1d38d610..c388e214 100644 --- a/src/body/RigidBody.h +++ b/src/body/RigidBody.h @@ -38,7 +38,7 @@ namespace reactphysics3d { // Class declarations struct JointListElement; -class Constraint; +class Joint; // Class RigidBody /** @@ -103,7 +103,7 @@ class RigidBody : public CollisionBody { RigidBody& operator=(const RigidBody& body); /// Remove a joint from the joints list - void removeJointFromJointsList(MemoryAllocator& memoryAllocator, const Constraint* joint); + void removeJointFromJointsList(MemoryAllocator& memoryAllocator, const Joint* joint); public : @@ -137,18 +137,6 @@ class RigidBody : public CollisionBody { /// Set the inverse of the mass void setMassInverse(decimal massInverse); - /// Return the current external force of the body - Vector3 getExternalForce() const; - - /// Set the current external force on the body - void setExternalForce(const Vector3& force); - - /// Return the current external torque of the body - Vector3 getExternalTorque() const; - - /// Set the current external torque of the body - void setExternalTorque(const Vector3& torque); - /// Return the inverse of the mass of the body decimal getMassInverse() const; @@ -197,6 +185,15 @@ class RigidBody : public CollisionBody { /// Set the variable to know whether or not the body is sleeping virtual void setIsSleeping(bool isSleeping); + /// Apply an external force to the body at its gravity center. + void applyForceToCenter(const Vector3& force); + + /// Apply an external force to the body at a given point (in world-coordinates). + void applyForce(const Vector3& force, const Vector3& point); + + /// Apply an external torque to the body. + void applyTorque(const Vector3& torque); + // -------------------- Friendship -------------------- // friend class DynamicsWorld; @@ -236,26 +233,6 @@ inline Matrix3x3 RigidBody::getInertiaTensorLocalInverse() const { return mInertiaTensorLocalInverse; } -// Return the external force on the body -inline Vector3 RigidBody::getExternalForce() const { - return mExternalForce; -} - -// Set the external force on the body -inline void RigidBody::setExternalForce(const Vector3& force) { - mExternalForce = force; -} - -// Return the current external torque on the body -inline Vector3 RigidBody::getExternalTorque() const { - return mExternalTorque; -} - - // Set the current external torque on the body -inline void RigidBody::setExternalTorque(const Vector3& torque) { - mExternalTorque = torque; -} - // Return the inverse of the mass of the body inline decimal RigidBody::getMassInverse() const { return mMassInverse; @@ -367,6 +344,62 @@ inline void RigidBody::setIsSleeping(bool isSleeping) { Body::setIsSleeping(isSleeping); } +// Apply an external force to the body at its gravity center. +/// If the body is sleeping, calling this method will wake it up. Note that the +/// force will we added to the sum of the applied forces and that this sum will be +/// reset to zero at the end of each call of the DynamicsWorld::update() method. +inline void RigidBody::applyForceToCenter(const Vector3& force) { + // If it is a static body, do not apply any force + if (!mIsMotionEnabled) return; + + // Awake the body if it was sleeping + if (mIsSleeping) { + setIsSleeping(false); + } + + // Add the force + mExternalForce += force; +} + +// Apply an external force to the body at a given point (in world-coordinates). +/// If the point is not at the center of gravity of the body, it will also +/// generate some torque and therefore, change the angular velocity of the body. +/// If the body is sleeping, calling this method will wake it up. Note that the +/// force will we added to the sum of the applied forces and that this sum will be +/// reset to zero at the end of each call of the DynamicsWorld::update() method. +inline void RigidBody::applyForce(const Vector3& force, const Vector3& point) { + + // If it is a static body, do not apply any force + if (!mIsMotionEnabled) return; + + // Awake the body if it was sleeping + if (mIsSleeping) { + setIsSleeping(false); + } + + // Add the force and torque + mExternalForce += force; + mExternalTorque += (point - mTransform.getPosition()).cross(force); +} + +// Apply an external torque to the body. +/// If the body is sleeping, calling this method will wake it up. Note that the +/// force will we added to the sum of the applied torques and that this sum will be +/// reset to zero at the end of each call of the DynamicsWorld::update() method. +inline void RigidBody::applyTorque(const Vector3& torque) { + + // If it is a static body, do not apply any force + if (!mIsMotionEnabled) return; + + // Awake the body if it was sleeping + if (mIsSleeping) { + setIsSleeping(false); + } + + // Add the torque + mExternalTorque += torque; +} + } #endif diff --git a/src/collision/CollisionDetection.cpp b/src/collision/CollisionDetection.cpp index 5f97d271..cf006f5b 100644 --- a/src/collision/CollisionDetection.cpp +++ b/src/collision/CollisionDetection.cpp @@ -112,6 +112,9 @@ void CollisionDetection::computeNarrowPhase() { // Check if the two bodies are allowed to collide, otherwise, we do not test for collision if (mNoCollisionPairs.count(pair->getBodiesIndexPair()) > 0) continue; + + // Check if the two bodies are sleeping, if so, we do no test collision between them + if (body1->isSleeping() && body2->isSleeping()) continue; // Select the narrow phase algorithm to use according to the two collision shapes NarrowPhaseAlgorithm& narrowPhaseAlgorithm = SelectNarrowPhaseAlgorithm( diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index 6dc016ae..c707e8cd 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -34,7 +34,7 @@ const decimal BallAndSocketJoint::BETA = decimal(0.2); // Constructor BallAndSocketJoint::BallAndSocketJoint(const BallAndSocketJointInfo& jointInfo) - : Constraint(jointInfo), mImpulse(Vector3(0, 0, 0)) { + : Joint(jointInfo), mImpulse(Vector3(0, 0, 0)) { // Compute the local-space anchor point for each body mLocalAnchorPointBody1 = mBody1->getTransform().getInverse() * jointInfo.anchorPointWorldSpace; diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index 3d155afd..f6bd8d7a 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -27,7 +27,7 @@ #define REACTPHYSICS3D_BALL_AND_SOCKET_JOINT_H // Libraries -#include "Constraint.h" +#include "Joint.h" #include "../mathematics/mathematics.h" namespace reactphysics3d { @@ -37,7 +37,7 @@ namespace reactphysics3d { * This structure is used to gather the information needed to create a ball-and-socket * joint. This structure will be used to create the actual ball-and-socket joint. */ -struct BallAndSocketJointInfo : public ConstraintInfo { +struct BallAndSocketJointInfo : public JointInfo { public : @@ -49,7 +49,7 @@ struct BallAndSocketJointInfo : public ConstraintInfo { /// Constructor BallAndSocketJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace) - : ConstraintInfo(rigidBody1, rigidBody2, BALLSOCKETJOINT), + : JointInfo(rigidBody1, rigidBody2, BALLSOCKETJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace) {} }; @@ -58,7 +58,7 @@ struct BallAndSocketJointInfo : public ConstraintInfo { * This class represents a ball-and-socket joint that allows arbitrary rotation * between two bodies. */ -class BallAndSocketJoint : public Constraint { +class BallAndSocketJoint : public Joint { private : diff --git a/src/constraint/ContactPoint.cpp b/src/constraint/ContactPoint.cpp index 1f068df3..1f7b9dd3 100644 --- a/src/constraint/ContactPoint.cpp +++ b/src/constraint/ContactPoint.cpp @@ -31,7 +31,8 @@ using namespace std; // Constructor ContactPoint::ContactPoint(const ContactPointInfo& contactInfo) - : Constraint(contactInfo), mNormal(contactInfo.normal), + : mBody1(contactInfo.body1), mBody2(contactInfo.body2), + mNormal(contactInfo.normal), mPenetrationDepth(contactInfo.penetrationDepth), mLocalPointOnBody1(contactInfo.localPoint1), mLocalPointOnBody2(contactInfo.localPoint2), @@ -50,23 +51,3 @@ ContactPoint::ContactPoint(const ContactPointInfo& contactInfo) ContactPoint::~ContactPoint() { } - -// Initialize before solving the constraint -void ContactPoint::initBeforeSolve(const ConstraintSolverData& constraintSolverData) { - -} - -// Warm start the constraint (apply the previous impulse at the beginning of the step) -void ContactPoint::warmstart(const ConstraintSolverData& constraintSolverData) { - -} - -// Solve the velocity constraint -void ContactPoint::solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) { - -} - -// Solve the position constraint -void ContactPoint::solvePositionConstraint(const ConstraintSolverData& constraintSolverData) { - -} diff --git a/src/constraint/ContactPoint.h b/src/constraint/ContactPoint.h index 333fcf45..21a6a37f 100644 --- a/src/constraint/ContactPoint.h +++ b/src/constraint/ContactPoint.h @@ -27,7 +27,6 @@ #define REACTPHYSICS3D_CONTACT_POINT_H // Libraries -#include "Constraint.h" #include "../body/RigidBody.h" #include "../configuration.h" #include "../mathematics/mathematics.h" @@ -43,7 +42,7 @@ namespace reactphysics3d { * informations are used to compute the contact set for a contact * between two bodies. */ -struct ContactPointInfo : public ConstraintInfo { +struct ContactPointInfo { private: @@ -59,6 +58,12 @@ struct ContactPointInfo : public ConstraintInfo { // -------------------- Attributes -------------------- // + /// First rigid body of the constraint + RigidBody* body1; + + /// Second rigid body of the constraint + RigidBody* body2; + /// Normal vector the the collision contact in world space const Vector3 normal; @@ -76,7 +81,7 @@ struct ContactPointInfo : public ConstraintInfo { /// Constructor ContactPointInfo(const Vector3& normal, decimal penetrationDepth, const Vector3& localPoint1, const Vector3& localPoint2) - : ConstraintInfo(CONTACT), normal(normal), penetrationDepth(penetrationDepth), + : normal(normal), penetrationDepth(penetrationDepth), localPoint1(localPoint1), localPoint2(localPoint2) { } @@ -85,15 +90,20 @@ struct ContactPointInfo : public ConstraintInfo { // Class ContactPoint /** * This class represents a collision contact point between two - * bodies in the physics engine. The ContactPoint class inherits from - * the Constraint class. + * bodies in the physics engine. */ -class ContactPoint : public Constraint { +class ContactPoint { - protected : + private : // -------------------- Attributes -------------------- // + /// First rigid body of the contact + RigidBody* mBody1; + + /// Second rigid body of the contact + RigidBody* mBody2; + /// Normal vector of the contact (From body1 toward body2) in world space const Vector3 mNormal; @@ -143,7 +153,13 @@ class ContactPoint : public Constraint { ContactPoint(const ContactPointInfo& contactInfo); /// Destructor - virtual ~ContactPoint(); + ~ContactPoint(); + + /// Return the reference to the body 1 + RigidBody* const getBody1() const; + + /// Return the reference to the body 2 + RigidBody* const getBody2() const; /// Return the normal vector of the contact Vector3 getNormal() const; @@ -209,21 +225,19 @@ class ContactPoint : public Constraint { decimal getPenetrationDepth() const; /// Return the number of bytes used by the contact point - virtual size_t getSizeInBytes() const; - - /// Initialize before solving the constraint - virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); - - /// Warm start the constraint (apply the previous impulse at the beginning of the step) - virtual void warmstart(const ConstraintSolverData& constraintSolverData); - - /// Solve the velocity constraint - virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); - - /// Solve the position constraint - virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); + size_t getSizeInBytes() const; }; +// Return the reference to the body 1 +inline RigidBody* const ContactPoint::getBody1() const { + return mBody1; +} + +// Return the reference to the body 2 +inline RigidBody* const ContactPoint::getBody2() const { + return mBody2; +} + // Return the normal vector of the contact inline Vector3 ContactPoint::getNormal() const { return mNormal; diff --git a/src/constraint/FixedJoint.cpp b/src/constraint/FixedJoint.cpp index 7a5acf82..46f7b89d 100644 --- a/src/constraint/FixedJoint.cpp +++ b/src/constraint/FixedJoint.cpp @@ -34,7 +34,7 @@ const decimal FixedJoint::BETA = decimal(0.2); // Constructor FixedJoint::FixedJoint(const FixedJointInfo& jointInfo) - : Constraint(jointInfo), mImpulseTranslation(0, 0, 0), mImpulseRotation(0, 0, 0) { + : Joint(jointInfo), mImpulseTranslation(0, 0, 0), mImpulseRotation(0, 0, 0) { // Compute the local-space anchor point for each body const Transform& transform1 = mBody1->getTransform(); diff --git a/src/constraint/FixedJoint.h b/src/constraint/FixedJoint.h index 7ce30912..1e153085 100644 --- a/src/constraint/FixedJoint.h +++ b/src/constraint/FixedJoint.h @@ -27,7 +27,7 @@ #define REACTPHYSICS3D_FIXED_JOINT_H // Libraries -#include "Constraint.h" +#include "Joint.h" #include "../mathematics/mathematics.h" namespace reactphysics3d { @@ -37,7 +37,7 @@ namespace reactphysics3d { * This structure is used to gather the information needed to create a fixed * joint. This structure will be used to create the actual fixed joint. */ -struct FixedJointInfo : public ConstraintInfo { +struct FixedJointInfo : public JointInfo { public : @@ -49,7 +49,7 @@ struct FixedJointInfo : public ConstraintInfo { /// Constructor FixedJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace) - : ConstraintInfo(rigidBody1, rigidBody2, FIXEDJOINT), + : JointInfo(rigidBody1, rigidBody2, FIXEDJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace){} }; @@ -58,7 +58,7 @@ struct FixedJointInfo : public ConstraintInfo { * This class represents a fixed joint that is used to forbid any translation or rotation * between two bodies. */ -class FixedJoint : public Constraint { +class FixedJoint : public Joint { private : diff --git a/src/constraint/HingeJoint.cpp b/src/constraint/HingeJoint.cpp index b3eb9f9b..2a7a5ee0 100644 --- a/src/constraint/HingeJoint.cpp +++ b/src/constraint/HingeJoint.cpp @@ -35,7 +35,7 @@ const decimal HingeJoint::BETA = decimal(0.2); // Constructor HingeJoint::HingeJoint(const HingeJointInfo& jointInfo) - : Constraint(jointInfo), mImpulseTranslation(0, 0, 0), mImpulseRotation(0, 0), + : Joint(jointInfo), mImpulseTranslation(0, 0, 0), mImpulseRotation(0, 0), mImpulseLowerLimit(0), mImpulseUpperLimit(0), mImpulseMotor(0), mIsLimitEnabled(jointInfo.isLimitEnabled), mIsMotorEnabled(jointInfo.isMotorEnabled), mLowerLimit(jointInfo.minAngleLimit), mUpperLimit(jointInfo.maxAngleLimit), diff --git a/src/constraint/HingeJoint.h b/src/constraint/HingeJoint.h index 782cf834..faf4c2e4 100644 --- a/src/constraint/HingeJoint.h +++ b/src/constraint/HingeJoint.h @@ -27,7 +27,7 @@ #define REACTPHYSICS3D_HINGE_JOINT_H // Libraries -#include "Constraint.h" +#include "Joint.h" #include "../mathematics/mathematics.h" namespace reactphysics3d { @@ -37,7 +37,7 @@ namespace reactphysics3d { * This structure is used to gather the information needed to create a hinge joint. * This structure will be used to create the actual hinge joint. */ -struct HingeJointInfo : public ConstraintInfo { +struct HingeJointInfo : public JointInfo { public : @@ -74,7 +74,7 @@ struct HingeJointInfo : public ConstraintInfo { HingeJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace, const Vector3& initRotationAxisWorld) - : ConstraintInfo(rigidBody1, rigidBody2, HINGEJOINT), + : JointInfo(rigidBody1, rigidBody2, HINGEJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), rotationAxisWorld(initRotationAxisWorld), isLimitEnabled(false), isMotorEnabled(false), minAngleLimit(-1), maxAngleLimit(1), @@ -85,7 +85,7 @@ struct HingeJointInfo : public ConstraintInfo { const Vector3& initAnchorPointWorldSpace, const Vector3& initRotationAxisWorld, decimal initMinAngleLimit, decimal initMaxAngleLimit) - : ConstraintInfo(rigidBody1, rigidBody2, HINGEJOINT), + : JointInfo(rigidBody1, rigidBody2, HINGEJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), rotationAxisWorld(initRotationAxisWorld), isLimitEnabled(true), isMotorEnabled(false), minAngleLimit(initMinAngleLimit), @@ -98,7 +98,7 @@ struct HingeJointInfo : public ConstraintInfo { const Vector3& initRotationAxisWorld, decimal initMinAngleLimit, decimal initMaxAngleLimit, decimal initMotorSpeed, decimal initMaxMotorTorque) - : ConstraintInfo(rigidBody1, rigidBody2, HINGEJOINT), + : JointInfo(rigidBody1, rigidBody2, HINGEJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), rotationAxisWorld(initRotationAxisWorld), isLimitEnabled(true), isMotorEnabled(false), minAngleLimit(initMinAngleLimit), @@ -111,7 +111,7 @@ struct HingeJointInfo : public ConstraintInfo { * This class represents a hinge joint that allows arbitrary rotation * between two bodies around a single axis. */ -class HingeJoint : public Constraint { +class HingeJoint : public Joint { private : diff --git a/src/constraint/Constraint.cpp b/src/constraint/Joint.cpp similarity index 83% rename from src/constraint/Constraint.cpp rename to src/constraint/Joint.cpp index ab9c7d99..189b69d8 100644 --- a/src/constraint/Constraint.cpp +++ b/src/constraint/Joint.cpp @@ -24,22 +24,21 @@ ********************************************************************************/ // Libraries -#include "Constraint.h" +#include "Joint.h" using namespace reactphysics3d; // Constructor -Constraint::Constraint(const ConstraintInfo& constraintInfo) - :mBody1(constraintInfo.body1), mBody2(constraintInfo.body2), mActive(true), - mType(constraintInfo.type), - mPositionCorrectionTechnique(constraintInfo.positionCorrectionTechnique), - mIsCollisionEnabled(constraintInfo.isCollisionEnabled), mIsAlreadyInIsland(false) { +Joint::Joint(const JointInfo& jointInfo) + :mBody1(jointInfo.body1), mBody2(jointInfo.body2), mType(jointInfo.type), + mPositionCorrectionTechnique(jointInfo.positionCorrectionTechnique), + mIsCollisionEnabled(jointInfo.isCollisionEnabled), mIsAlreadyInIsland(false) { assert(mBody1 != NULL); assert(mBody2 != NULL); } // Destructor -Constraint::~Constraint() { +Joint::~Joint() { } diff --git a/src/constraint/Constraint.h b/src/constraint/Joint.h similarity index 74% rename from src/constraint/Constraint.h rename to src/constraint/Joint.h index adc4092a..425ece73 100644 --- a/src/constraint/Constraint.h +++ b/src/constraint/Joint.h @@ -35,11 +35,11 @@ namespace reactphysics3d { // Enumeration for the type of a constraint -enum ConstraintType {CONTACT, BALLSOCKETJOINT, SLIDERJOINT, HINGEJOINT, FIXEDJOINT}; +enum JointType {BALLSOCKETJOINT, SLIDERJOINT, HINGEJOINT, FIXEDJOINT}; // Class declarations struct ConstraintSolverData; -class Constraint; +class Joint; // Structure JointListElement /** @@ -52,7 +52,7 @@ struct JointListElement { // -------------------- Attributes -------------------- // /// Pointer to the actual joint - Constraint* joint; + Joint* joint; /// Next element of the list JointListElement* next; @@ -60,32 +60,32 @@ struct JointListElement { // -------------------- Methods -------------------- // /// Constructor - JointListElement(Constraint* initJoint, JointListElement* initNext) + JointListElement(Joint* initJoint, JointListElement* initNext) :joint(initJoint), next(initNext){ } }; -// Structure ConstraintInfo +// Structure JointInfo /** - * This structure is used to gather the information needed to create a constraint. + * This structure is used to gather the information needed to create a joint. */ -struct ConstraintInfo { +struct JointInfo { public : // -------------------- Attributes -------------------- // - /// First rigid body of the constraint + /// First rigid body of the joint RigidBody* body1; - /// Second rigid body of the constraint + /// Second rigid body of the joint RigidBody* body2; - /// Type of the constraint - ConstraintType type; + /// Type of the joint + JointType type; - /// True if the two bodies of the constraint are allowed to collide with each other + /// True if the two bodies of the joint are allowed to collide with each other bool isCollisionEnabled; /// Position correction technique used for the constraint (used for joints). @@ -93,46 +93,41 @@ struct ConstraintInfo { JointsPositionCorrectionTechnique positionCorrectionTechnique; /// Constructor - ConstraintInfo(ConstraintType constraintType) + JointInfo(JointType constraintType) : body1(NULL), body2(NULL), type(constraintType), positionCorrectionTechnique(NON_LINEAR_GAUSS_SEIDEL), isCollisionEnabled(true) {} /// Constructor - ConstraintInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, ConstraintType constraintType) + JointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, JointType constraintType) : body1(rigidBody1), body2(rigidBody2), type(constraintType), positionCorrectionTechnique(NON_LINEAR_GAUSS_SEIDEL), isCollisionEnabled(true) { } /// Destructor - virtual ~ConstraintInfo() {} + virtual ~JointInfo() {} }; -// Class Constraint +// Class Joint /** - * This abstract class represents a constraint in the physics engine. - * A constraint can be a collision contact point or a joint for - * instance. + * This abstract class represents a joint between two bodies. */ -class Constraint { +class Joint { protected : // -------------------- Attributes -------------------- // - /// Pointer to the first body of the constraint + /// Pointer to the first body of the joint RigidBody* const mBody1; - /// Pointer to the second body of the constraint + /// Pointer to the second body of the joint RigidBody* const mBody2; - /// True if the constraint is active - bool mActive; - - /// Type of the constraint - const ConstraintType mType; + /// Type of the joint + const JointType mType; /// Body 1 index in the velocity array to solve the constraint uint mIndexBody1; @@ -152,20 +147,20 @@ class Constraint { // -------------------- Methods -------------------- // /// Private copy-constructor - Constraint(const Constraint& constraint); + Joint(const Joint& constraint); /// Private assignment operator - Constraint& operator=(const Constraint& constraint); + Joint& operator=(const Joint& constraint); public : // -------------------- Methods -------------------- // /// Constructor - Constraint(const ConstraintInfo& constraintInfo); + Joint(const JointInfo& jointInfo); /// Destructor - virtual ~Constraint(); + virtual ~Joint(); /// Return the reference to the body 1 RigidBody* const getBody1() const; @@ -177,21 +172,21 @@ class Constraint { bool isActive() const; /// Return the type of the constraint - ConstraintType getType() const; + JointType getType() const; - /// Return true if the collision between the two bodies of the constraint is enabled + /// Return true if the collision between the two bodies of the joint is enabled bool isCollisionEnabled() const; /// Return true if the joint has already been added into an island bool isAlreadyInIsland() const; - /// Return the number of bytes used by the constraint + /// Return the number of bytes used by the joint virtual size_t getSizeInBytes() const = 0; - /// Initialize before solving the constraint + /// Initialize before solving the joint virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData) = 0; - /// Warm start the constraint (apply the previous impulse at the beginning of the step) + /// Warm start the joint (apply the previous impulse at the beginning of the step) virtual void warmstart(const ConstraintSolverData& constraintSolverData) = 0; /// Solve the velocity constraint @@ -207,32 +202,32 @@ class Constraint { }; // Return the reference to the body 1 -inline RigidBody* const Constraint::getBody1() const { +inline RigidBody* const Joint::getBody1() const { return mBody1; } // Return the reference to the body 2 -inline RigidBody* const Constraint::getBody2() const { +inline RigidBody* const Joint::getBody2() const { return mBody2; } -// Return true if the constraint is active -inline bool Constraint::isActive() const { - return mActive; +// Return true if the joint is active +inline bool Joint::isActive() const { + return (mBody1->isActive() && mBody2->isActive()); } -// Return the type of the constraint -inline ConstraintType Constraint::getType() const { +// Return the type of the joint +inline JointType Joint::getType() const { return mType; } -// Return true if the collision between the two bodies of the constraint is enabled -inline bool Constraint::isCollisionEnabled() const { +// Return true if the collision between the two bodies of the joint is enabled +inline bool Joint::isCollisionEnabled() const { return mIsCollisionEnabled; } // Return true if the joint has already been added into an island -inline bool Constraint::isAlreadyInIsland() const { +inline bool Joint::isAlreadyInIsland() const { return mIsAlreadyInIsland; } diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index 9a0f1321..0fd912ce 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -33,7 +33,7 @@ const decimal SliderJoint::BETA = decimal(0.2); // Constructor SliderJoint::SliderJoint(const SliderJointInfo& jointInfo) - : Constraint(jointInfo), mImpulseTranslation(0, 0), mImpulseRotation(0, 0, 0), + : Joint(jointInfo), mImpulseTranslation(0, 0), mImpulseRotation(0, 0, 0), mImpulseLowerLimit(0), mImpulseUpperLimit(0), mImpulseMotor(0), mIsLimitEnabled(jointInfo.isLimitEnabled), mIsMotorEnabled(jointInfo.isMotorEnabled), mLowerLimit(jointInfo.minTranslationLimit), diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h index 454f3385..8e06fd26 100644 --- a/src/constraint/SliderJoint.h +++ b/src/constraint/SliderJoint.h @@ -37,7 +37,7 @@ namespace reactphysics3d { * This structure is used to gather the information needed to create a slider * joint. This structure will be used to create the actual slider joint. */ -struct SliderJointInfo : public ConstraintInfo { +struct SliderJointInfo : public JointInfo { public : @@ -71,7 +71,7 @@ struct SliderJointInfo : public ConstraintInfo { SliderJointInfo(RigidBody* rigidBody1, RigidBody* rigidBody2, const Vector3& initAnchorPointWorldSpace, const Vector3& initSliderAxisWorldSpace) - : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), + : JointInfo(rigidBody1, rigidBody2, SLIDERJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), sliderAxisWorldSpace(initSliderAxisWorldSpace), isLimitEnabled(false), isMotorEnabled(false), minTranslationLimit(-1.0), @@ -82,7 +82,7 @@ struct SliderJointInfo : public ConstraintInfo { const Vector3& initAnchorPointWorldSpace, const Vector3& initSliderAxisWorldSpace, decimal initMinTranslationLimit, decimal initMaxTranslationLimit) - : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), + : JointInfo(rigidBody1, rigidBody2, SLIDERJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), sliderAxisWorldSpace(initSliderAxisWorldSpace), isLimitEnabled(true), isMotorEnabled(false), @@ -96,7 +96,7 @@ struct SliderJointInfo : public ConstraintInfo { const Vector3& initSliderAxisWorldSpace, decimal initMinTranslationLimit, decimal initMaxTranslationLimit, decimal initMotorSpeed, decimal initMaxMotorForce) - : ConstraintInfo(rigidBody1, rigidBody2, SLIDERJOINT), + : JointInfo(rigidBody1, rigidBody2, SLIDERJOINT), anchorPointWorldSpace(initAnchorPointWorldSpace), sliderAxisWorldSpace(initSliderAxisWorldSpace), isLimitEnabled(true), isMotorEnabled(true), @@ -109,7 +109,7 @@ struct SliderJointInfo : public ConstraintInfo { /** * This class represents a slider joint. */ -class SliderJoint : public Constraint { +class SliderJoint : public Joint { private : diff --git a/src/engine/CollisionWorld.h b/src/engine/CollisionWorld.h index 89f38c5c..8678ea01 100644 --- a/src/engine/CollisionWorld.h +++ b/src/engine/CollisionWorld.h @@ -36,7 +36,7 @@ #include "../body/CollisionBody.h" #include "OverlappingPair.h" #include "../collision/CollisionDetection.h" -#include "../constraint/Constraint.h" +#include "../constraint/Joint.h" #include "../constraint/ContactPoint.h" #include "../memory/MemoryAllocator.h" diff --git a/src/engine/ConstraintSolver.cpp b/src/engine/ConstraintSolver.cpp index 25ef6641..cdd97b41 100644 --- a/src/engine/ConstraintSolver.cpp +++ b/src/engine/ConstraintSolver.cpp @@ -65,7 +65,7 @@ void ConstraintSolver::initializeForIsland(decimal dt, Island* island) { mConstraintSolverData.isWarmStartingActive = mIsWarmStartingActive; // For each joint of the island - Constraint** joints = island->getJoints(); + Joint** joints = island->getJoints(); for (uint i=0; igetNbJoints(); i++) { // Initialize the constraint before solving it @@ -87,7 +87,7 @@ void ConstraintSolver::solveVelocityConstraints(Island* island) { assert(island->getNbJoints() > 0); // For each joint of the island - Constraint** joints = island->getJoints(); + Joint** joints = island->getJoints(); for (uint i=0; igetNbJoints(); i++) { // Solve the constraint @@ -104,7 +104,7 @@ void ConstraintSolver::solvePositionConstraints(Island* island) { assert(island->getNbJoints() > 0); // For each joint of the island - Constraint** joints = island->getJoints(); + Joint** joints = island->getJoints(); for (uint i=0; i < island->getNbJoints(); i++) { // Solve the constraint diff --git a/src/engine/ConstraintSolver.h b/src/engine/ConstraintSolver.h index f52c6412..f15f72cc 100644 --- a/src/engine/ConstraintSolver.h +++ b/src/engine/ConstraintSolver.h @@ -29,7 +29,7 @@ // Libraries #include "../configuration.h" #include "mathematics/mathematics.h" -#include "../constraint/Constraint.h" +#include "../constraint/Joint.h" #include "Island.h" #include #include diff --git a/src/engine/ContactSolver.h b/src/engine/ContactSolver.h index 979a0abf..55b47c37 100644 --- a/src/engine/ContactSolver.h +++ b/src/engine/ContactSolver.h @@ -29,7 +29,7 @@ // Libraries #include "../constraint/ContactPoint.h" #include "../configuration.h" -#include "../constraint/Constraint.h" +#include "../constraint/Joint.h" #include "ContactManifold.h" #include "Island.h" #include "Impulse.h" diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index cf585e43..190a9228 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -147,6 +147,9 @@ void DynamicsWorld::update() { // Update the AABBs of the bodies updateRigidBodiesAABB(); + + // Reset the external force and torque applied to the bodies + resetBodiesForceAndTorque(); } // Compute and set the interpolation factor to all the bodies @@ -315,10 +318,10 @@ void DynamicsWorld::integrateRigidBodiesVelocities() { // Integrate the external force to get the new velocity of the body mConstrainedLinearVelocities[indexBody] = bodies[b]->getLinearVelocity() + - dt * bodies[b]->getMassInverse() * bodies[b]->getExternalForce(); + dt * bodies[b]->getMassInverse() * bodies[b]->mExternalForce; mConstrainedAngularVelocities[indexBody] = bodies[b]->getAngularVelocity() + dt * bodies[b]->getInertiaTensorInverseWorld() * - bodies[b]->getExternalTorque(); + bodies[b]->mExternalTorque; // If the gravity has to be applied to this rigid body if (bodies[b]->isGravityEnabled() && mIsGravityEnabled) { @@ -527,7 +530,7 @@ void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { // Destroy all the joints in which the rigid body to be destroyed is involved // TODO : Iterate on the mJointList of the rigid body instead over all the joints of the world bodyindex idToRemove = rigidBody->getID(); - for (std::set::iterator it = mJoints.begin(); it != mJoints.end(); ++it) { + for (std::set::iterator it = mJoints.begin(); it != mJoints.end(); ++it) { if ((*it)->getBody1()->getID() == idToRemove || (*it)->getBody2()->getID() == idToRemove) { destroyJoint(*it); } @@ -548,9 +551,9 @@ void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { } // Create a joint between two bodies in the world and return a pointer to the new joint -Constraint* DynamicsWorld::createJoint(const ConstraintInfo& jointInfo) { +Joint* DynamicsWorld::createJoint(const JointInfo& jointInfo) { - Constraint* newJoint = NULL; + Joint* newJoint = NULL; // Allocate memory to create the new joint switch(jointInfo.type) { @@ -617,7 +620,7 @@ Constraint* DynamicsWorld::createJoint(const ConstraintInfo& jointInfo) { } // Destroy a joint -void DynamicsWorld::destroyJoint(Constraint* joint) { +void DynamicsWorld::destroyJoint(Joint* joint) { assert(joint != NULL); @@ -642,14 +645,14 @@ void DynamicsWorld::destroyJoint(Constraint* joint) { size_t nbBytes = joint->getSizeInBytes(); // Call the destructor of the joint - joint->Constraint::~Constraint(); + joint->Joint::~Joint(); // Release the allocated memory mMemoryAllocator.release(joint, nbBytes); } // Add the joint to the list of joints of the two bodies involved in the joint -void DynamicsWorld::addJointToBody(Constraint* joint) { +void DynamicsWorld::addJointToBody(Joint* joint) { assert(joint != NULL); @@ -741,7 +744,7 @@ void DynamicsWorld::computeIslands() { it != mContactManifolds.end(); ++it) { (*it)->mIsAlreadyInIsland = false; } - for (std::set::iterator it = mJoints.begin(); it != mJoints.end(); ++it) { + for (std::set::iterator it = mJoints.begin(); it != mJoints.end(); ++it) { (*it)->mIsAlreadyInIsland = false; } @@ -763,8 +766,8 @@ void DynamicsWorld::computeIslands() { // TODO : When we will use STATIC bodies, we will need to take care of this case here if (!body->getIsMotionEnabled()) continue; - // If the body is sleeping, we go to the next body - if (body->isSleeping()) continue; + // If the body is sleeping or inactive, we go to the next body + if (body->isSleeping() || !body->isActive()) continue; // Reset the stack of bodies to visit uint stackIndex = 0; @@ -784,6 +787,7 @@ void DynamicsWorld::computeIslands() { // Get the next body to visit from the stack stackIndex--; RigidBody* bodyToVisit = stackBodiesToVisit[stackIndex]; + assert(bodyToVisit->isActive()); // Awake the body if it is slepping bodyToVisit->setIsSleeping(false); @@ -828,7 +832,7 @@ void DynamicsWorld::computeIslands() { for (jointElement = bodyToVisit->mJointsList; jointElement != NULL; jointElement = jointElement->next) { - Constraint* joint = jointElement->joint; + Joint* joint = jointElement->joint; // Check if the current joint has already been added into an island if (joint->isAlreadyInIsland()) continue; diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index 7f783ffd..93fd5d7d 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -77,7 +77,7 @@ class DynamicsWorld : public CollisionWorld { std::vector mContactManifolds; /// All the joints of the world - std::set mJoints; + std::set mJoints; /// Gravity vector of the world Vector3 mGravity; @@ -144,6 +144,9 @@ class DynamicsWorld : public CollisionWorld { /// Update the AABBs of the bodies void updateRigidBodiesAABB(); + /// Reset the external force and torque applied to the bodies + void resetBodiesForceAndTorque(); + /// Update the position and orientation of a body void updatePositionAndOrientationOfBody(RigidBody* body, Vector3 newLinVelocity, Vector3 newAngVelocity); @@ -232,13 +235,13 @@ public : void destroyRigidBody(RigidBody* rigidBody); /// Create a joint between two bodies in the world and return a pointer to the new joint - Constraint* createJoint(const ConstraintInfo& jointInfo); + Joint* createJoint(const JointInfo& jointInfo); /// Destroy a joint - void destroyJoint(Constraint* joint); + void destroyJoint(Joint* joint); /// Add the joint to the list of joints of the two bodies involved in the joint - void addJointToBody(Constraint* joint); + void addJointToBody(Joint* joint); //// Add a contact manifold to the linked list of contact manifolds of the two bodies involed //// in the corresponding contact. @@ -309,6 +312,17 @@ public : uint getNbIslands() const {return mNbIslands;} }; +// Reset the external force and torque applied to the bodies +inline void DynamicsWorld::resetBodiesForceAndTorque() { + + // For each body of the world + std::set::iterator it; + for (it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { + (*it)->mExternalForce.setToZero(); + (*it)->mExternalTorque.setToZero(); + } +} + // Start the physics simulation inline void DynamicsWorld::start() { mTimer.start(); diff --git a/src/engine/Island.cpp b/src/engine/Island.cpp index 06b3734b..5f5b36a2 100644 --- a/src/engine/Island.cpp +++ b/src/engine/Island.cpp @@ -40,8 +40,8 @@ Island::Island(uint id, uint nbMaxBodies, uint nbMaxContactManifolds, uint nbMax mNbAllocatedBytesContactManifolds = sizeof(ContactManifold*) * nbMaxContactManifolds; mContactManifolds = (ContactManifold**) mMemoryAllocator.allocate( mNbAllocatedBytesContactManifolds); - mNbAllocatedBytesJoints = sizeof(Constraint*) * nbMaxJoints; - mJoints = (Constraint**) mMemoryAllocator.allocate(mNbAllocatedBytesJoints); + mNbAllocatedBytesJoints = sizeof(Joint*) * nbMaxJoints; + mJoints = (Joint**) mMemoryAllocator.allocate(mNbAllocatedBytesJoints); } // Destructor diff --git a/src/engine/Island.h b/src/engine/Island.h index bcd9d990..283e532d 100644 --- a/src/engine/Island.h +++ b/src/engine/Island.h @@ -29,7 +29,7 @@ // Libraries #include "../memory/MemoryAllocator.h" #include "../body/RigidBody.h" -#include "../constraint/Constraint.h" +#include "../constraint/Joint.h" #include "ContactManifold.h" namespace reactphysics3d { @@ -55,7 +55,7 @@ class Island { ContactManifold** mContactManifolds; /// Array with all the joints between bodies of the island - Constraint** mJoints; + Joint** mJoints; /// Current number of bodies in the island uint mNbBodies; @@ -104,7 +104,7 @@ class Island { void addContactManifold(ContactManifold* contactManifold); /// Add a joint into the island - void addJoint(Constraint* joint); + void addJoint(Joint* joint); /// Return the number of bodies in the island uint getNbBodies() const; @@ -122,7 +122,7 @@ class Island { ContactManifold** getContactManifold(); /// Return a pointer to the array of joints - Constraint** getJoints(); + Joint** getJoints(); // TODO : REMOVE THIS uint getID() const {return mID;} @@ -146,7 +146,7 @@ inline void Island::addContactManifold(ContactManifold* contactManifold) { } // Add a joint into the island -inline void Island::addJoint(Constraint* joint) { +inline void Island::addJoint(Joint* joint) { mJoints[mNbJoints] = joint; mNbJoints++; } @@ -177,7 +177,7 @@ inline ContactManifold** Island::getContactManifold() { } // Return a pointer to the array of joints -inline Constraint** Island::getJoints() { +inline Joint** Island::getJoints() { return mJoints; } From 07df001e8bcc55c5b4915687cd9ddf349ee891f4 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 10 Sep 2013 21:33:52 +0200 Subject: [PATCH 54/66] Add the EventListener class --- .../{fallingcubes => cubes}/CMakeLists.txt | 0 .../FallingCubes.cpp => cubes/Cubes.cpp} | 0 examples/{fallingcubes => cubes}/Scene.cpp | 0 examples/{fallingcubes => cubes}/Scene.h | 0 src/engine/CollisionWorld.cpp | 1 + src/engine/CollisionWorld.h | 4 +- src/engine/DynamicsWorld.cpp | 18 ++++-- src/engine/DynamicsWorld.h | 13 ++++ src/engine/EventListener.h | 62 +++++++++++++++++++ src/reactphysics3d.h | 1 + 10 files changed, 94 insertions(+), 5 deletions(-) rename examples/{fallingcubes => cubes}/CMakeLists.txt (100%) rename examples/{fallingcubes/FallingCubes.cpp => cubes/Cubes.cpp} (100%) rename examples/{fallingcubes => cubes}/Scene.cpp (100%) rename examples/{fallingcubes => cubes}/Scene.h (100%) create mode 100644 src/engine/EventListener.h diff --git a/examples/fallingcubes/CMakeLists.txt b/examples/cubes/CMakeLists.txt similarity index 100% rename from examples/fallingcubes/CMakeLists.txt rename to examples/cubes/CMakeLists.txt diff --git a/examples/fallingcubes/FallingCubes.cpp b/examples/cubes/Cubes.cpp similarity index 100% rename from examples/fallingcubes/FallingCubes.cpp rename to examples/cubes/Cubes.cpp diff --git a/examples/fallingcubes/Scene.cpp b/examples/cubes/Scene.cpp similarity index 100% rename from examples/fallingcubes/Scene.cpp rename to examples/cubes/Scene.cpp diff --git a/examples/fallingcubes/Scene.h b/examples/cubes/Scene.h similarity index 100% rename from examples/fallingcubes/Scene.h rename to examples/cubes/Scene.h diff --git a/src/engine/CollisionWorld.cpp b/src/engine/CollisionWorld.cpp index 8674a289..87dfa76b 100644 --- a/src/engine/CollisionWorld.cpp +++ b/src/engine/CollisionWorld.cpp @@ -34,6 +34,7 @@ using namespace std; // Constructor CollisionWorld::CollisionWorld() : mCollisionDetection(this, mMemoryAllocator), mCurrentBodyID(0) { + } // Destructor diff --git a/src/engine/CollisionWorld.h b/src/engine/CollisionWorld.h index 8678ea01..4d297466 100644 --- a/src/engine/CollisionWorld.h +++ b/src/engine/CollisionWorld.h @@ -39,6 +39,7 @@ #include "../constraint/Joint.h" #include "../constraint/ContactPoint.h" #include "../memory/MemoryAllocator.h" +#include "EventListener.h" /// Namespace reactphysics3d namespace reactphysics3d { @@ -91,7 +92,8 @@ class CollisionWorld { virtual void notifyRemovedOverlappingPair(const BroadPhasePair* removedPair); /// Notify the world about a new narrow-phase contact - virtual void notifyNewContact(const BroadPhasePair* pair, const ContactPointInfo* contactInfo); + virtual void notifyNewContact(const BroadPhasePair* pair, + const ContactPointInfo* contactInfo); /// Update the overlapping pair virtual void updateOverlappingPair(const BroadPhasePair* pair); diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 190a9228..68073562 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -48,7 +48,7 @@ DynamicsWorld::DynamicsWorld(const Vector3 &gravity, decimal timeStep = DEFAULT_ mIslands(NULL), mNbBodiesCapacity(0), mSleepLinearVelocity(DEFAULT_SLEEP_LINEAR_VELOCITY), mSleepAngularVelocity(DEFAULT_SLEEP_ANGULAR_VELOCITY), - mTimeBeforeSleep(DEFAULT_TIME_BEFORE_SLEEP) { + mTimeBeforeSleep(DEFAULT_TIME_BEFORE_SLEEP), mEventListener(NULL) { } @@ -147,11 +147,11 @@ void DynamicsWorld::update() { // Update the AABBs of the bodies updateRigidBodiesAABB(); - - // Reset the external force and torque applied to the bodies - resetBodiesForceAndTorque(); } + // Reset the external force and torque applied to the bodies + resetBodiesForceAndTorque(); + // Compute and set the interpolation factor to all the bodies setInterpolationFactorToAllBodies(); } @@ -968,6 +968,13 @@ void DynamicsWorld::notifyNewContact(const BroadPhasePair* broadPhasePair, OverlappingPair* overlappingPair = mOverlappingPairs.find(indexPair)->second; assert(overlappingPair != NULL); + // If it is the first contact since the pair are overlapping + if (overlappingPair->getNbContactPoints() == 0) { + + // Trigger a callback event + if (mEventListener != NULL) mEventListener->beginContact(*contactInfo); + } + // Add the contact to the contact cache of the corresponding overlapping pair overlappingPair->addContact(contact); @@ -978,6 +985,9 @@ void DynamicsWorld::notifyNewContact(const BroadPhasePair* broadPhasePair, // of the two bodies involved in the contact addContactManifoldToBody(overlappingPair->getContactManifold(), overlappingPair->mBody1, overlappingPair->mBody2); + + // Trigger a callback event for the new contact + if (mEventListener != NULL) mEventListener->newContact(*contactInfo); } // Enable/Disable the sleeping technique diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index 93fd5d7d..36f6ab23 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -130,6 +130,9 @@ class DynamicsWorld : public CollisionWorld { /// becomes smaller than the sleep velocity. decimal mTimeBeforeSleep; + /// Pointer to an event listener object + EventListener* mEventListener; + // -------------------- Methods -------------------- // /// Private copy-constructor @@ -305,6 +308,9 @@ public : /// Set the time a body is required to stay still before sleeping void setTimeBeforeSleep(decimal timeBeforeSleep); + /// Set an event listener object to receive events callbacks. + void setEventListener(EventListener* eventListener); + // TODO : REMOVE THIS Island** getIslands() { return mIslands;} @@ -490,6 +496,13 @@ inline void DynamicsWorld::setTimeBeforeSleep(decimal timeBeforeSleep) { mTimeBeforeSleep = timeBeforeSleep; } +// Set an event listener object to receive events callbacks. +/// If you use NULL as an argument, the events callbacks will be disabled. +inline void DynamicsWorld::setEventListener(EventListener* eventListener) { + mEventListener = eventListener; +} + + } #endif diff --git a/src/engine/EventListener.h b/src/engine/EventListener.h new file mode 100644 index 00000000..97d7c347 --- /dev/null +++ b/src/engine/EventListener.h @@ -0,0 +1,62 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 REACTPHYSICS3D_EVENT_LISTENER_H +#define REACTPHYSICS3D_EVENT_LISTENER_H + +// Libraries +#include "../constraint/ContactPoint.h" + +namespace reactphysics3d { + +// Class EventListener +/** + * This class can be used to receive event callbacks from the physics engine. + * In order to receive callbacks, you need to create a new class that inherits from + * this one and you must override the methods you need. Then, you need to register your + * new event listener class to the physics world using the DynamicsWorld::setEventListener() + * method. + */ +class EventListener { + + public : + + /// Constructor + EventListener() {} + + /// Destructor + virtual ~EventListener() {} + + /// Called when a new contact point is found between two bodies that were separated before + virtual void beginContact(const ContactPointInfo& contact) {} + + /// Called when a new contact point is found between two bodies + virtual void newContact(const ContactPointInfo& contact) {} + +}; + +} + +#endif diff --git a/src/reactphysics3d.h b/src/reactphysics3d.h index ababb9bc..cf85da5e 100644 --- a/src/reactphysics3d.h +++ b/src/reactphysics3d.h @@ -42,6 +42,7 @@ #include "engine/DynamicsWorld.h" #include "engine/CollisionWorld.h" #include "engine/Material.h" +#include "engine/EventListener.h" #include "collision/shapes/CollisionShape.h" #include "collision/shapes/BoxShape.h" #include "collision/shapes/SphereShape.h" From 4bd228e8c0aba28087ff4cf5fee4bc2f6063e59d Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 10 Sep 2013 21:35:56 +0200 Subject: [PATCH 55/66] Add the collision shapes example --- examples/collisionshapes/CMakeLists.txt | 20 + examples/collisionshapes/CollisionShapes.cpp | 152 +++++++ examples/collisionshapes/Scene.cpp | 451 +++++++++++++++++++ examples/collisionshapes/Scene.h | 153 +++++++ examples/common/ConvexMesh.cpp | 149 ++++++ examples/common/ConvexMesh.h | 72 +++ examples/common/meshes/convexmesh.obj | 125 +++++ 7 files changed, 1122 insertions(+) create mode 100644 examples/collisionshapes/CMakeLists.txt create mode 100644 examples/collisionshapes/CollisionShapes.cpp create mode 100644 examples/collisionshapes/Scene.cpp create mode 100644 examples/collisionshapes/Scene.h create mode 100644 examples/common/ConvexMesh.cpp create mode 100644 examples/common/ConvexMesh.h create mode 100644 examples/common/meshes/convexmesh.obj diff --git a/examples/collisionshapes/CMakeLists.txt b/examples/collisionshapes/CMakeLists.txt new file mode 100644 index 00000000..932b10e8 --- /dev/null +++ b/examples/collisionshapes/CMakeLists.txt @@ -0,0 +1,20 @@ +# Minimum cmake version required +cmake_minimum_required(VERSION 2.6) + +# Project configuration +PROJECT(CollisionShapes) + +# Copy the shaders used for the demo into the build directory +FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/shaders/") + +# Copy the meshes used for the demo into the build directory +FILE(COPY "../common/meshes/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/meshes/") + +# Headers +INCLUDE_DIRECTORIES("../common/opengl-framework/src/" "../common/") + +# Create the example executable using the +# compiled reactphysics3d static library +ADD_EXECUTABLE(collisionshapes CollisionShapes.cpp Scene.cpp Scene.h "../common/VisualContactPoint.cpp" "../common/ConvexMesh.cpp" "../common/Capsule.cpp" "../common/Sphere.cpp" "../common/Cylinder.cpp" "../common/Cone.cpp" "../common/Box.cpp" "../common/Viewer.cpp") + +TARGET_LINK_LIBRARIES(collisionshapes reactphysics3d openglframework) diff --git a/examples/collisionshapes/CollisionShapes.cpp b/examples/collisionshapes/CollisionShapes.cpp new file mode 100644 index 00000000..b444c466 --- /dev/null +++ b/examples/collisionshapes/CollisionShapes.cpp @@ -0,0 +1,152 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Scene.h" +#include "Viewer.h" + +// Declarations +void simulate(); +void display(); +void finish(); +void reshape(int width, int height); +void mouseButton(int button, int state, int x, int y); +void mouseMotion(int x, int y); +void keyboard(unsigned char key, int x, int y); +void init(); + +// Namespaces +using namespace openglframework; + +// Global variables +Viewer* viewer; +Scene* scene; + +// Main function +int main(int argc, char** argv) { + + // Create and initialize the Viewer + viewer = new Viewer(); + Vector2 windowsSize = Vector2(800, 600); + Vector2 windowsPosition = Vector2(100, 100); + bool initOK = viewer->init(argc, argv, "ReactPhysics3D Examples - Collision Shapes", + windowsSize, windowsPosition); + if (!initOK) return 1; + + // Create the scene + scene = new Scene(viewer); + + init(); + + // Glut Idle function that is continuously called + glutIdleFunc(simulate); + glutDisplayFunc(display); + glutReshapeFunc(reshape); + glutMouseFunc(mouseButton); + glutMotionFunc(mouseMotion); + glutKeyboardFunc(keyboard); + glutCloseFunc(finish); + + // Glut main looop + glutMainLoop(); + + return 0; +} + +// Simulate function +void simulate() { + + // Physics simulation + scene->simulate(); + + viewer->computeFPS(); + + // Ask GLUT to render the scene + glutPostRedisplay (); +} + +// Initialization +void init() { + + // Define the background color (black) + glClearColor(0.0, 0.0, 0.0, 1.0); +} + +// Reshape function +void reshape(int newWidth, int newHeight) { + viewer->reshape(newWidth, newHeight); +} + +// Called when a mouse button event occurs +void mouseButton(int button, int state, int x, int y) { + viewer->mouseButtonEvent(button, state, x, y); +} + +// Called when a mouse motion event occurs +void mouseMotion(int x, int y) { + viewer->mouseMotionEvent(x, y); +} + +// Called when the user hits a special key on the keyboard +void keyboard(unsigned char key, int x, int y) { + switch(key) { + + // Escape key + case 27: + glutLeaveMainLoop(); + break; + + // Space bar + case 32: + scene->pauseContinueSimulation(); + break; + } +} + +// End of the application +void finish() { + + // Destroy the viewer and the scene + delete viewer; + delete scene; +} + +// Display the scene +void display() { + + // Render the scene + scene->render(); + + // Display the FPS + viewer->displayGUI(); + + // Swap the buffers + glutSwapBuffers(); + + // Check the OpenGL errors + GlutViewer::checkOpenGLErrors(); +} + + diff --git a/examples/collisionshapes/Scene.cpp b/examples/collisionshapes/Scene.cpp new file mode 100644 index 00000000..038567a9 --- /dev/null +++ b/examples/collisionshapes/Scene.cpp @@ -0,0 +1,451 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "Scene.h" + +// Namespaces +using namespace openglframework; + +// Constructor +Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), + mPhongShader("shaders/phong.vert", + "shaders/phong.frag"), mIsRunning(false) { + + // Move the light 0 + mLight0.translateWorld(Vector3(50, 50, 50)); + + // Compute the radius and the center of the scene + float radiusScene = 10.0f; + openglframework::Vector3 center(0, 5, 0); + + // Set the center of the scene + mViewer->setScenePosition(center, radiusScene); + + // Gravity vector in the dynamics world + rp3d::Vector3 gravity(0, rp3d::decimal(-9.81), 0); + + // Time step for the physics simulation + rp3d::decimal timeStep = 1.0f / 60.0f; + + // Create the dynamics world for the physics simulation + mDynamicsWorld = new rp3d::DynamicsWorld(gravity, timeStep); + + // Set the number of iterations of the constraint solver + mDynamicsWorld->setNbIterationsVelocitySolver(15); + + // Create the static data for the visual contact points + VisualContactPoint::createStaticData(); + + float radius = 3.0f; + + // Create all the boxes of the scene + for (int i=0; igetRigidBody()->setIsMotionEnabled(true); + + // Change the material properties of the rigid body + rp3d::Material& material = box->getRigidBody()->getMaterial(); + material.setBounciness(rp3d::decimal(0.2)); + + // Add the sphere the list of sphere in the scene + mBoxes.push_back(box); + } + + // Create all the spheres of the scene + for (int i=0; igetRigidBody()->setIsMotionEnabled(true); + + // Change the material properties of the rigid body + rp3d::Material& material = sphere->getRigidBody()->getMaterial(); + material.setBounciness(rp3d::decimal(0.2)); + + // Add the sphere the list of sphere in the scene + mSpheres.push_back(sphere); + } + + // Create all the cones of the scene + for (int i=0; igetRigidBody()->setIsMotionEnabled(true); + + // Change the material properties of the rigid body + rp3d::Material& material = cone->getRigidBody()->getMaterial(); + material.setBounciness(rp3d::decimal(0.2)); + + // Add the cone the list of sphere in the scene + mCones.push_back(cone); + } + + // Create all the cylinders of the scene + for (int i=0; igetRigidBody()->setIsMotionEnabled(true); + + // Change the material properties of the rigid body + rp3d::Material& material = cylinder->getRigidBody()->getMaterial(); + material.setBounciness(rp3d::decimal(0.2)); + + // Add the cylinder the list of sphere in the scene + mCylinders.push_back(cylinder); + } + + // Create all the capsules of the scene + for (int i=0; igetRigidBody()->setIsMotionEnabled(true); + + // Change the material properties of the rigid body + rp3d::Material& material = capsule->getRigidBody()->getMaterial(); + material.setBounciness(rp3d::decimal(0.2)); + + // Add the cylinder the list of sphere in the scene + mCapsules.push_back(capsule); + } + + // Create all the convex meshes of the scene + for (int i=0; igetRigidBody()->setIsMotionEnabled(true); + + // Change the material properties of the rigid body + rp3d::Material& material = mesh->getRigidBody()->getMaterial(); + material.setBounciness(rp3d::decimal(0.2)); + + // Add the mesh the list of sphere in the scene + mConvexMeshes.push_back(mesh); + } + + // Create the floor + openglframework::Vector3 floorPosition(0, 0, 0); + mFloor = new Box(FLOOR_SIZE, floorPosition, FLOOR_MASS, mDynamicsWorld); + + // The floor must be a non-moving rigid body + mFloor->getRigidBody()->setIsMotionEnabled(false); + + // Change the material properties of the rigid body + rp3d::Material& material = mFloor->getRigidBody()->getMaterial(); + material.setBounciness(rp3d::decimal(0.2)); + + // Start the simulation + startSimulation(); +} + +// Destructor +Scene::~Scene() { + + // Stop the physics simulation + stopSimulation(); + + // Destroy the shader + mPhongShader.destroy(); + + // Destroy all the boxes of the scene + for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + + // Destroy the corresponding rigid body from the dynamics world + mDynamicsWorld->destroyRigidBody((*it)->getRigidBody()); + + // Destroy the box + delete (*it); + } + + // Destroy all the sphere of the scene + for (std::vector::iterator it = mSpheres.begin(); it != mSpheres.end(); ++it) { + + // Destroy the corresponding rigid body from the dynamics world + mDynamicsWorld->destroyRigidBody((*it)->getRigidBody()); + + // Destroy the sphere + delete (*it); + } + + // Destroy all the cones of the scene + for (std::vector::iterator it = mCones.begin(); it != mCones.end(); ++it) { + + // Destroy the corresponding rigid body from the dynamics world + mDynamicsWorld->destroyRigidBody((*it)->getRigidBody()); + + // Destroy the sphere + delete (*it); + } + + // Destroy all the cylinders of the scene + for (std::vector::iterator it = mCylinders.begin(); it != mCylinders.end(); ++it) { + + // Destroy the corresponding rigid body from the dynamics world + mDynamicsWorld->destroyRigidBody((*it)->getRigidBody()); + + // Destroy the sphere + delete (*it); + } + + // Destroy all the capsules of the scene + for (std::vector::iterator it = mCapsules.begin(); it != mCapsules.end(); ++it) { + + // Destroy the corresponding rigid body from the dynamics world + mDynamicsWorld->destroyRigidBody((*it)->getRigidBody()); + + // Destroy the sphere + delete (*it); + } + + // Destroy all the convex meshes of the scene + for (std::vector::iterator it = mConvexMeshes.begin(); + it != mConvexMeshes.end(); ++it) { + + // Destroy the corresponding rigid body from the dynamics world + mDynamicsWorld->destroyRigidBody((*it)->getRigidBody()); + + // Destroy the convex mesh + delete (*it); + } + + // Destroy all the visual contact points + for (std::vector::iterator it = mContactPoints.begin(); + it != mContactPoints.end(); ++it) { + delete (*it); + } + + // Destroy the static data for the visual contact points + VisualContactPoint::destroyStaticData(); + + // Destroy the rigid body of the floor + mDynamicsWorld->destroyRigidBody(mFloor->getRigidBody()); + + // Destroy the floor + delete mFloor; + + // Destroy the dynamics world + delete mDynamicsWorld; +} + +// Take a step for the simulation +void Scene::simulate() { + + // If the physics simulation is running + if (mIsRunning) { + + // Take a simulation step + mDynamicsWorld->update(); + + // Update the position and orientation of the boxes + for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + + // Update the transform used for the rendering + (*it)->updateTransform(); + } + + // Update the position and orientation of the sphere + for (std::vector::iterator it = mSpheres.begin(); it != mSpheres.end(); ++it) { + + // Update the transform used for the rendering + (*it)->updateTransform(); + } + + // Update the position and orientation of the cones + for (std::vector::iterator it = mCones.begin(); it != mCones.end(); ++it) { + + // Update the transform used for the rendering + (*it)->updateTransform(); + } + + // Update the position and orientation of the cylinders + for (std::vector::iterator it = mCylinders.begin(); it != mCylinders.end(); ++it) { + + // Update the transform used for the rendering + (*it)->updateTransform(); + } + + // Update the position and orientation of the capsules + for (std::vector::iterator it = mCapsules.begin(); it != mCapsules.end(); ++it) { + + // Update the transform used for the rendering + (*it)->updateTransform(); + } + + // Update the position and orientation of the convex meshes + for (std::vector::iterator it = mConvexMeshes.begin(); + it != mConvexMeshes.end(); ++it) { + + // Update the transform used for the rendering + (*it)->updateTransform(); + } + + // Destroy all the visual contact points + for (std::vector::iterator it = mContactPoints.begin(); + it != mContactPoints.end(); ++it) { + delete (*it); + } + mContactPoints.clear(); + + // Generate the new visual contact points + const std::vector& manifolds = mDynamicsWorld->getContactManifolds(); + for (std::vector::const_iterator it = manifolds.begin(); + it != manifolds.end(); ++it) { + for (unsigned int i=0; i<(*it)->getNbContactPoints(); i++) { + rp3d::ContactPoint* point = (*it)->getContactPoint(i); + + const rp3d::Vector3 pos = point->getWorldPointOnBody1(); + openglframework::Vector3 position(pos.x, pos.y, pos.z); + VisualContactPoint* visualPoint = new VisualContactPoint(position); + mContactPoints.push_back(visualPoint); + } + } + + mFloor->updateTransform(); + } +} + +// Render the scene +void Scene::render() { + + glEnable(GL_DEPTH_TEST); + glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); + glEnable(GL_CULL_FACE); + + // Get the world-space to camera-space matrix + const Camera& camera = mViewer->getCamera(); + const openglframework::Matrix4 worldToCameraMatrix = camera.getTransformMatrix().getInverse(); + + // Bind the shader + mPhongShader.bind(); + + // Set the variables of the shader + mPhongShader.setMatrix4x4Uniform("projectionMatrix", camera.getProjectionMatrix()); + mPhongShader.setVector3Uniform("light0PosCameraSpace", worldToCameraMatrix * mLight0.getOrigin()); + mPhongShader.setVector3Uniform("lightAmbientColor", Vector3(0.3f, 0.3f, 0.3f)); + Color& diffColLight0 = mLight0.getDiffuseColor(); + Color& specColLight0 = mLight0.getSpecularColor(); + mPhongShader.setVector3Uniform("light0DiffuseColor", Vector3(diffColLight0.r, diffColLight0.g, diffColLight0.b)); + mPhongShader.setVector3Uniform("light0SpecularColor", Vector3(specColLight0.r, specColLight0.g, specColLight0.b)); + mPhongShader.setFloatUniform("shininess", 200.0f); + + // Render all the boxes of the scene + for (std::vector::iterator it = mBoxes.begin(); it != mBoxes.end(); ++it) { + (*it)->render(mPhongShader, worldToCameraMatrix); + } + + // Render all the sphere of the scene + for (std::vector::iterator it = mSpheres.begin(); it != mSpheres.end(); ++it) { + (*it)->render(mPhongShader, worldToCameraMatrix); + } + + // Render all the cones of the scene + for (std::vector::iterator it = mCones.begin(); it != mCones.end(); ++it) { + (*it)->render(mPhongShader, worldToCameraMatrix); + } + + // Render all the cylinders of the scene + for (std::vector::iterator it = mCylinders.begin(); it != mCylinders.end(); ++it) { + (*it)->render(mPhongShader, worldToCameraMatrix); + } + + // Render all the capsules of the scene + for (std::vector::iterator it = mCapsules.begin(); it != mCapsules.end(); ++it) { + (*it)->render(mPhongShader, worldToCameraMatrix); + } + + // Render all the convex meshes of the scene + for (std::vector::iterator it = mConvexMeshes.begin(); + it != mConvexMeshes.end(); ++it) { + (*it)->render(mPhongShader, worldToCameraMatrix); + } + + // Render all the visual contact points + for (std::vector::iterator it = mContactPoints.begin(); + it != mContactPoints.end(); ++it) { + (*it)->render(mPhongShader, worldToCameraMatrix); + } + + // Render the floor + mFloor->render(mPhongShader, worldToCameraMatrix); + + // Unbind the shader + mPhongShader.unbind(); +} diff --git a/examples/collisionshapes/Scene.h b/examples/collisionshapes/Scene.h new file mode 100644 index 00000000..d893b311 --- /dev/null +++ b/examples/collisionshapes/Scene.h @@ -0,0 +1,153 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 SCENE_H +#define SCENE_H + +// Libraries +#include "openglframework.h" +#include "reactphysics3d.h" +#include "Sphere.h" +#include "Box.h" +#include "Cone.h" +#include "Cylinder.h" +#include "Capsule.h" +#include "ConvexMesh.h" +#include "VisualContactPoint.h" + +// Constants +const int NB_BOXES = 3; +const int NB_SPHERES = 3; +const int NB_CONES = 3; +const int NB_CYLINDERS = 3; +const int NB_CAPSULES = 3; +const int NB_MESHES = 3; +const openglframework::Vector3 BOX_SIZE(2, 2, 2); +const float SPHERE_RADIUS = 1.5f; +const float CONE_RADIUS = 2.0f; +const float CONE_HEIGHT = 3.0f; +const float CYLINDER_RADIUS = 1.0f; +const float CYLINDER_HEIGHT = 5.0f; +const float CAPSULE_RADIUS = 1.0f; +const float CAPSULE_HEIGHT = 1.0f; +const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters +const float BOX_MASS = 1.0f; +const float CONE_MASS = 1.0f; +const float CYLINDER_MASS = 1.0f; +const float CAPSULE_MASS = 1.0f; +const float MESH_MASS = 1.0f; +const float FLOOR_MASS = 100.0f; // Floor mass in kilograms + +// Class Scene +class Scene { + + private : + + // -------------------- Attributes -------------------- // + + /// Pointer to the viewer + openglframework::GlutViewer* mViewer; + + /// Light 0 + openglframework::Light mLight0; + + /// Phong shader + openglframework::Shader mPhongShader; + + /// All the spheres of the scene + std::vector mBoxes; + + std::vector mSpheres; + + std::vector mCones; + + std::vector mCylinders; + + std::vector mCapsules; + + /// All the convex meshes of the scene + std::vector mConvexMeshes; + + /// All the visual contact points + std::vector mContactPoints; + + /// Box for the floor + Box* mFloor; + + /// Dynamics world used for the physics simulation + rp3d::DynamicsWorld* mDynamicsWorld; + + /// True if the physics simulation is running + bool mIsRunning; + + public: + + // -------------------- Methods -------------------- // + + /// Constructor + Scene(openglframework::GlutViewer* viewer); + + /// Destructor + ~Scene(); + + /// Take a step for the simulation + void simulate(); + + /// Stop the simulation + void stopSimulation(); + + /// Start the simulation + void startSimulation(); + + /// Pause or continue simulation + void pauseContinueSimulation(); + + /// Render the scene + void render(); +}; + +// Stop the simulation +inline void Scene::stopSimulation() { + mDynamicsWorld->stop(); + mIsRunning = false; +} + +// Start the simulation +inline void Scene::startSimulation() { + mDynamicsWorld->start(); + mIsRunning = true; +} + +// Pause or continue simulation +inline void Scene::pauseContinueSimulation() { + if (mIsRunning) { + stopSimulation(); + } + else { + startSimulation(); + } +} + +#endif diff --git a/examples/common/ConvexMesh.cpp b/examples/common/ConvexMesh.cpp new file mode 100644 index 00000000..2e8307ee --- /dev/null +++ b/examples/common/ConvexMesh.cpp @@ -0,0 +1,149 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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. * +* * +********************************************************************************/ + +// Libraries +#include "ConvexMesh.h" + + +// Constructor +ConvexMesh::ConvexMesh(const openglframework::Vector3 &position, float mass, + reactphysics3d::DynamicsWorld* dynamicsWorld) + : openglframework::Mesh() { + + // Load the mesh from a file + openglframework::MeshReaderWriter::loadMeshFromFile("meshes/convexmesh.obj", *this); + + // Calculate the normals of the mesh + calculateNormals(); + + // Initialize the position where the sphere will be rendered + translateWorld(position); + + // Create the collision shape for the rigid body (convex mesh shape) + // ReactPhysics3D will clone this object to create an internal one. Therefore, + // it is OK if this object is destroyed right after calling Dynamics::createRigidBody() + rp3d::decimal* verticesArray = (rp3d::decimal*) getVerticesPointer(); + rp3d::ConvexMeshShape collisionShape(verticesArray, mVertices.size(), + sizeof(openglframework::Vector3)); + + // Add the edges information of the mesh into the convex mesh collision shape. + // This is optional but it really speed up the convex mesh collision detection at the + // cost of some additional memory to store the edges inside the collision shape. + for (unsigned int i=0; icreateRigidBody(transform, mass, inertiaTensor, collisionShape); +} + +// Destructor +ConvexMesh::~ConvexMesh() { + + // Destroy the mesh + destroy(); +} + +// Render the sphere at the correct position and with the correct orientation +void ConvexMesh::render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix) { + + // Bind the shader + shader.bind(); + + // Set the model to camera matrix + const openglframework::Matrix4 localToCameraMatrix = worldToCameraMatrix * mTransformMatrix; + shader.setMatrix4x4Uniform("localToCameraMatrix", localToCameraMatrix); + + // Set the normal matrix (inverse transpose of the 3x3 upper-left sub matrix of the + // model-view matrix) + const openglframework::Matrix3 normalMatrix = + localToCameraMatrix.getUpperLeft3x3Matrix().getInverse().getTranspose(); + shader.setMatrix3x3Uniform("normalMatrix", normalMatrix); + + glEnableClientState(GL_VERTEX_ARRAY); + glEnableClientState(GL_NORMAL_ARRAY); + if (hasTexture()) { + glEnableClientState(GL_TEXTURE_COORD_ARRAY); + } + + glVertexPointer(3, GL_FLOAT, 0, getVerticesPointer()); + glNormalPointer(GL_FLOAT, 0, getNormalsPointer()); + if(hasTexture()) { + glTexCoordPointer(2, GL_FLOAT, 0, getUVTextureCoordinatesPointer()); + } + + // For each part of the mesh + for (unsigned int i=0; igetInterpolatedTransform(); + + // Compute the transform used for rendering the sphere + float matrix[16]; + transform.getOpenGLMatrix(matrix); + openglframework::Matrix4 newMatrix(matrix[0], matrix[4], matrix[8], matrix[12], + matrix[1], matrix[5], matrix[9], matrix[13], + matrix[2], matrix[6], matrix[10], matrix[14], + matrix[3], matrix[7], matrix[11], matrix[15]); + + // Apply the scaling matrix to have the correct sphere dimensions + mTransformMatrix = newMatrix; +} diff --git a/examples/common/ConvexMesh.h b/examples/common/ConvexMesh.h new file mode 100644 index 00000000..46ae8c09 --- /dev/null +++ b/examples/common/ConvexMesh.h @@ -0,0 +1,72 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://code.google.com/p/reactphysics3d/ * +* Copyright (c) 2010-2013 Daniel Chappuis * +********************************************************************************* +* * +* 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 CONVEX_MESH_H +#define CONVEX_MESH_H + +// Libraries +#include "openglframework.h" +#include "reactphysics3d.h" + +// Class ConvexMesh +class ConvexMesh : public openglframework::Mesh { + + private : + + // -------------------- Attributes -------------------- // + + /// Rigid body used to simulate the dynamics of the mesh + rp3d::RigidBody* mRigidBody; + + // -------------------- Methods -------------------- // + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + ConvexMesh(const openglframework::Vector3& position, float mass, + rp3d::DynamicsWorld* dynamicsWorld); + + /// Destructor + ~ConvexMesh(); + + /// Return a pointer to the rigid body of the mesh + rp3d::RigidBody* getRigidBody(); + + /// Update the transform matrix of the mesh + void updateTransform(); + + /// Render the mesh at the correct position and with the correct orientation + void render(openglframework::Shader& shader, + const openglframework::Matrix4& worldToCameraMatrix); +}; + +// Return a pointer to the rigid body of the mesh +inline rp3d::RigidBody* ConvexMesh::getRigidBody() { + return mRigidBody; +} + +#endif diff --git a/examples/common/meshes/convexmesh.obj b/examples/common/meshes/convexmesh.obj new file mode 100644 index 00000000..7a194f13 --- /dev/null +++ b/examples/common/meshes/convexmesh.obj @@ -0,0 +1,125 @@ +# Blender v2.66 (sub 0) OBJ File: '' +# www.blender.org +v 0.000000 -1.000000 0.000000 +v 0.723607 -0.447220 0.525725 +v -0.276388 -0.447220 0.850649 +v -0.894426 -0.447216 0.000000 +v -0.276388 -0.447220 -0.850649 +v 0.723607 -0.447220 -0.525725 +v 0.276388 0.447220 0.850649 +v -0.723607 0.447220 0.525725 +v -0.723607 0.447220 -0.525725 +v 0.276388 0.447220 -0.850649 +v 0.894426 0.447216 0.000000 +v 0.000000 1.000000 0.000000 +v 0.425323 -0.850654 0.309011 +v 0.262869 -0.525738 0.809012 +v -0.162456 -0.850654 0.499995 +v 0.425323 -0.850654 -0.309011 +v 0.850648 -0.525736 0.000000 +v -0.688189 -0.525736 0.499997 +v -0.525730 -0.850652 0.000000 +v -0.688189 -0.525736 -0.499997 +v -0.162456 -0.850654 -0.499995 +v 0.262869 -0.525738 -0.809012 +v 0.951058 0.000000 -0.309013 +v 0.951058 0.000000 0.309013 +v 0.587786 0.000000 0.809017 +v 0.000000 0.000000 1.000000 +v -0.587786 0.000000 0.809017 +v -0.951058 0.000000 0.309013 +v -0.951058 0.000000 -0.309013 +v -0.587786 0.000000 -0.809017 +v 0.000000 0.000000 -1.000000 +v 0.587786 0.000000 -0.809017 +v 0.688189 0.525736 0.499997 +v -0.262869 0.525738 0.809012 +v -0.850648 0.525736 0.000000 +v -0.262869 0.525738 -0.809012 +v 0.688189 0.525736 -0.499997 +v 0.525730 0.850652 0.000000 +v 0.162456 0.850654 0.499995 +v -0.425323 0.850654 0.309011 +v -0.425323 0.850654 -0.309011 +v 0.162456 0.850654 -0.499995 +s off +f 1 13 15 +f 2 13 17 +f 1 15 19 +f 1 19 21 +f 1 21 16 +f 2 17 24 +f 3 14 26 +f 4 18 28 +f 5 20 30 +f 6 22 32 +f 2 24 25 +f 3 26 27 +f 4 28 29 +f 5 30 31 +f 6 32 23 +f 7 33 39 +f 8 34 40 +f 9 35 41 +f 10 36 42 +f 11 37 38 +f 15 14 3 +f 15 13 14 +f 13 2 14 +f 17 16 6 +f 17 13 16 +f 13 1 16 +f 19 18 4 +f 19 15 18 +f 15 3 18 +f 21 20 5 +f 21 19 20 +f 19 4 20 +f 16 22 6 +f 16 21 22 +f 21 5 22 +f 24 23 11 +f 24 17 23 +f 17 6 23 +f 26 25 7 +f 26 14 25 +f 14 2 25 +f 28 27 8 +f 28 18 27 +f 18 3 27 +f 30 29 9 +f 30 20 29 +f 20 4 29 +f 32 31 10 +f 32 22 31 +f 22 5 31 +f 25 33 7 +f 25 24 33 +f 24 11 33 +f 27 34 8 +f 27 26 34 +f 26 7 34 +f 29 35 9 +f 29 28 35 +f 28 8 35 +f 31 36 10 +f 31 30 36 +f 30 9 36 +f 23 37 11 +f 23 32 37 +f 32 10 37 +f 39 38 12 +f 39 33 38 +f 33 11 38 +f 40 39 12 +f 40 34 39 +f 34 7 39 +f 41 40 12 +f 41 35 40 +f 35 8 40 +f 42 41 12 +f 42 36 41 +f 36 9 41 +f 38 42 12 +f 38 37 42 +f 37 10 42 From f575a46fdded59e66cb70ba5c9554c62e096cb06 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Tue, 10 Sep 2013 21:53:34 +0200 Subject: [PATCH 56/66] Modification of the examples --- examples/CMakeLists.txt | 4 +-- examples/common/Box.cpp | 29 ++++++++++------ examples/common/Box.h | 6 ++++ examples/common/Capsule.cpp | 6 ++-- examples/common/Cone.cpp | 2 +- examples/common/Cylinder.cpp | 2 +- examples/common/Sphere.cpp | 2 +- examples/common/Viewer.cpp | 2 +- .../common/opengl-framework/CMakeLists.txt | 24 ++++++++++---- examples/common/opengl-framework/src/Mesh.cpp | 1 - examples/common/opengl-framework/src/Mesh.h | 2 +- .../opengl-framework/src/MeshReaderWriter.cpp | 8 ++--- .../opengl-framework/src/MeshReaderWriter.h | 8 ++--- .../common/opengl-framework/src/Texture2D.cpp | 2 +- .../common/opengl-framework/src/Texture2D.h | 2 +- .../src/TextureReaderWriter.cpp | 33 ++++++++++--------- .../src/TextureReaderWriter.h | 20 ++++------- .../common/opengl-framework/src/maths/Color.h | 2 +- .../opengl-framework/src/shaders/phong.frag | 6 ++-- examples/cubes/CMakeLists.txt | 6 ++-- examples/cubes/Cubes.cpp | 2 +- examples/cubes/Scene.cpp | 16 +++++++-- examples/cubes/Scene.h | 7 ++-- examples/joints/Scene.cpp | 10 +++--- examples/joints/Scene.h | 2 +- 25 files changed, 116 insertions(+), 88 deletions(-) diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt index e8d6a1d6..6399abeb 100644 --- a/examples/CMakeLists.txt +++ b/examples/CMakeLists.txt @@ -2,6 +2,6 @@ CMAKE_MINIMUM_REQUIRED(VERSION 2.6) add_subdirectory(common/) -add_subdirectory(fallingcubes/) +add_subdirectory(cubes/) add_subdirectory(joints/) -add_subdirectory(convexmesh/) +add_subdirectory(collisionshapes/) diff --git a/examples/common/Box.cpp b/examples/common/Box.cpp index 34fb6597..757a3127 100644 --- a/examples/common/Box.cpp +++ b/examples/common/Box.cpp @@ -34,14 +34,14 @@ openglframework::VertexBufferObject Box::mVBOVertices(GL_ARRAY_BUFFER); openglframework::VertexBufferObject Box::mVBOIndices(GL_ELEMENT_ARRAY_BUFFER); bool Box::areVBOsCreated = false; VertexData Box::mCubeVertices[8] = { - {openglframework::Vector3(1,1,1),openglframework::Vector3(1,1,1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(-1,1,1),openglframework::Vector3(-1,1,1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(-1,-1,1),openglframework::Vector3(-1,-1,1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(1,-1,1),openglframework::Vector3(1,-1,1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(1,-1,-1),openglframework::Vector3(1,-1,-1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(-1,-1,-1),openglframework::Vector3(-1,-1,-1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(-1,1,-1),openglframework::Vector3(-1,1,-1),openglframework::Color(0,0,1,1)}, - {openglframework::Vector3(1,1,-1),openglframework::Vector3(1,1,-1),openglframework::Color(0,0,1,1)} + {openglframework::Vector3(1,1,1),openglframework::Vector3(1,1,1),openglframework::Color(1,0,0,1)}, + {openglframework::Vector3(-1,1,1),openglframework::Vector3(-1,1,1),openglframework::Color(1,0,0,1)}, + {openglframework::Vector3(-1,-1,1),openglframework::Vector3(-1,-1,1),openglframework::Color(1,0,0,1)}, + {openglframework::Vector3(1,-1,1),openglframework::Vector3(1,-1,1),openglframework::Color(1,0,0,1)}, + {openglframework::Vector3(1,-1,-1),openglframework::Vector3(1,-1,-1),openglframework::Color(1,0,0,1)}, + {openglframework::Vector3(-1,-1,-1),openglframework::Vector3(-1,-1,-1),openglframework::Color(1,0,0,1)}, + {openglframework::Vector3(-1,1,-1),openglframework::Vector3(-1,1,-1),openglframework::Color(1,0,0,1)}, + {openglframework::Vector3(1,1,-1),openglframework::Vector3(1,1,-1),openglframework::Color(1,0,0,1)} }; GLuint Box::mCubeIndices[36] = { 0, 1, 2, 2, 3, 0, @@ -59,7 +59,7 @@ GLuint Box::mCubeIndices[36] = { 0, 1, 2, // Constructor Box::Box(const openglframework::Vector3& size, const openglframework::Vector3 &position, float mass, reactphysics3d::DynamicsWorld* dynamicsWorld) - : openglframework::Object3D() { + : openglframework::Object3D(), mColor(0.5f, 0.5f, 0.5f, 1.0f) { // Initialize the size of the box mSize[0] = size.x * 0.5f; @@ -77,7 +77,7 @@ Box::Box(const openglframework::Vector3& size, const openglframework::Vector3 &p // Create the collision shape for the rigid body (box shape) // ReactPhysics3D will clone this object to create an internal one. Therefore, - // it is OK if this object is destroy right after calling Dynamics::createRigidBody() + // it is OK if this object is destroyed right after calling Dynamics::createRigidBody() const rp3d::BoxShape collisionShape(rp3d::Vector3(mSize[0], mSize[1], mSize[2])); // Compute the inertia tensor of the body using its collision shape @@ -121,6 +121,10 @@ void Box::render(openglframework::Shader& shader, localToCameraMatrix.getUpperLeft3x3Matrix().getInverse().getTranspose(); shader.setMatrix3x3Uniform("normalMatrix", normalMatrix); + // TODO : REMOVE THIS + openglframework::Vector4 color(mColor.r, mColor.g, mColor.b, mColor.a); + shader.setVector4Uniform("vertexColor", color); + // Bind the vertices VBO mVBOVertices.bind(); @@ -153,6 +157,11 @@ void Box::render(openglframework::Shader& shader, shader.unbind(); } +// Set the color of the box +void Box::setColor(openglframework::Color& color) { + mColor = color; +} + // Update the transform matrix of the box void Box::updateTransform() { diff --git a/examples/common/Box.h b/examples/common/Box.h index 302df1bc..c507ed4c 100644 --- a/examples/common/Box.h +++ b/examples/common/Box.h @@ -74,6 +74,9 @@ class Box : public openglframework::Object3D { /// True if the VBOs have already been created static bool areVBOsCreated; + // TODO : REMOVE THIS + openglframework::Color mColor; + // -------------------- Methods -------------------- // /// Create a Vertex Buffer Object to render to box with OpenGL @@ -98,6 +101,9 @@ class Box : public openglframework::Object3D { /// Render the cube at the correct position and with the correct orientation void render(openglframework::Shader& shader, const openglframework::Matrix4& worldToCameraMatrix); + + /// Set the color of the box + void setColor(openglframework::Color& color); }; // Return a pointer to the rigid body of the box diff --git a/examples/common/Capsule.cpp b/examples/common/Capsule.cpp index 07216c5e..d069e25f 100644 --- a/examples/common/Capsule.cpp +++ b/examples/common/Capsule.cpp @@ -40,16 +40,16 @@ Capsule::Capsule(float radius, float height, const openglframework::Vector3 &pos // Compute the scaling matrix mScalingMatrix = openglframework::Matrix4(mRadius, 0, 0, 0, - 0, (mHeight + 2.0 * mRadius) / 3.0, 0,0, + 0, (mHeight + 2.0f * mRadius) / 3.0f, 0,0, 0, 0, mRadius, 0, - 0, 0, 0, 1); + 0, 0, 0, 1.0f); // Initialize the position where the sphere will be rendered translateWorld(position); // Create the collision shape for the rigid body (sphere shape) // ReactPhysics3D will clone this object to create an internal one. Therefore, - // it is OK if this object is destroy right after calling Dynamics::createRigidBody() + // it is OK if this object is destroyed right after calling Dynamics::createRigidBody() const rp3d::CapsuleShape collisionShape(mRadius, mHeight); // Compute the inertia tensor of the body using its collision shape diff --git a/examples/common/Cone.cpp b/examples/common/Cone.cpp index ca7e1cbb..6043c5be 100644 --- a/examples/common/Cone.cpp +++ b/examples/common/Cone.cpp @@ -49,7 +49,7 @@ Cone::Cone(float radius, float height, const openglframework::Vector3 &position, // Create the collision shape for the rigid body (cone shape) // ReactPhysics3D will clone this object to create an internal one. Therefore, - // it is OK if this object is destroy right after calling Dynamics::createRigidBody() + // it is OK if this object is destroyed right after calling Dynamics::createRigidBody() const rp3d::ConeShape collisionShape(mRadius, mHeight); // Compute the inertia tensor of the body using its collision shape diff --git a/examples/common/Cylinder.cpp b/examples/common/Cylinder.cpp index 556d413f..b0b089e1 100644 --- a/examples/common/Cylinder.cpp +++ b/examples/common/Cylinder.cpp @@ -49,7 +49,7 @@ Cylinder::Cylinder(float radius, float height, const openglframework::Vector3 &p // Create the collision shape for the rigid body (cylinder shape) // ReactPhysics3D will clone this object to create an internal one. Therefore, - // it is OK if this object is destroy right after calling Dynamics::createRigidBody() + // it is OK if this object is destroyed right after calling Dynamics::createRigidBody() const rp3d::CylinderShape collisionShape(mRadius, mHeight); // Compute the inertia tensor of the body using its collision shape diff --git a/examples/common/Sphere.cpp b/examples/common/Sphere.cpp index f3d20010..4a62b0b1 100644 --- a/examples/common/Sphere.cpp +++ b/examples/common/Sphere.cpp @@ -49,7 +49,7 @@ Sphere::Sphere(float radius, const openglframework::Vector3 &position, // Create the collision shape for the rigid body (sphere shape) // ReactPhysics3D will clone this object to create an internal one. Therefore, - // it is OK if this object is destroy right after calling Dynamics::createRigidBody() + // it is OK if this object is destroyed right after calling Dynamics::createRigidBody() const rp3d::SphereShape collisionShape(mRadius); // Compute the inertia tensor of the body using its collision shape diff --git a/examples/common/Viewer.cpp b/examples/common/Viewer.cpp index 1f99d4dc..e0114f00 100644 --- a/examples/common/Viewer.cpp +++ b/examples/common/Viewer.cpp @@ -48,7 +48,7 @@ void Viewer::computeFPS() { if(timeInterval > 1000){ // calculate the number of frames per second - fps = nbFrames / (timeInterval / 1000.0f); + fps = static_cast(nbFrames / (timeInterval / 1000.0f)); // Set time previousTime = currentTime; diff --git a/examples/common/opengl-framework/CMakeLists.txt b/examples/common/opengl-framework/CMakeLists.txt index 82cba467..7683a289 100644 --- a/examples/common/opengl-framework/CMakeLists.txt +++ b/examples/common/opengl-framework/CMakeLists.txt @@ -4,6 +4,9 @@ CMAKE_MINIMUM_REQUIRED(VERSION 2.6) # Project configuration PROJECT(OPENGLFRAMEWORK) +# Options +OPTION(USE_JPEG_TEXTURES "Select this if you want to use jpeg textures (libjpeg required)" OFF) + # Where to find the module to find special packages/libraries set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/") @@ -23,12 +26,16 @@ else() MESSAGE("GLEW not found") endif() -# Find the LIBJPEG library -FIND_PACKAGE(JPEG REQUIRED) -if(JPEG_FOUND) - MESSAGE("LIBJPEG found") -else() - MESSAGE("LIBJPEG not found") +# If the user wants to use JPEG textures +if(USE_JPEG_TEXTURES) + + # Find the LIBJPEG library + FIND_PACKAGE(JPEG REQUIRED) + if(JPEG_FOUND) + MESSAGE("LIBJPEG found") + else() + MESSAGE("LIBJPEG not found") + endif() endif() # Freeglut @@ -37,6 +44,10 @@ add_subdirectory(freeglut) # Headers INCLUDE_DIRECTORIES(src freeglut ${JPEG_INCLUDE_DIR}) +if(USE_JPEG_TEXTURES) + add_definitions(-DUSE_JPEG_TEXTURE) +endif() + # Library configuration file ( GLOB_RECURSE @@ -44,7 +55,6 @@ file ( src/* ) - # Require the opengl-framework code to be compiled in a static library ADD_LIBRARY ( openglframework diff --git a/examples/common/opengl-framework/src/Mesh.cpp b/examples/common/opengl-framework/src/Mesh.cpp index 6a993539..070c1d07 100644 --- a/examples/common/opengl-framework/src/Mesh.cpp +++ b/examples/common/opengl-framework/src/Mesh.cpp @@ -84,7 +84,6 @@ void Mesh::calculateNormals() { // Normalize the normal at each vertex for (uint i=0; i -#include -#include +#ifdef USE_JPEG_TEXTURE + #include + #include +#endif using namespace openglframework; using namespace std; @@ -59,8 +61,7 @@ typedef struct { // Load a texture from a file void TextureReaderWriter::loadTextureFromFile(const std::string& filename, - Texture2D& textureToCreate) - throw(runtime_error, invalid_argument){ + Texture2D& textureToCreate) { // Get the extension of the file uint startPosExtension = filename.find_last_of("."); @@ -70,9 +71,11 @@ void TextureReaderWriter::loadTextureFromFile(const std::string& filename, if (extension == "tga") { readTGAPicture(filename, textureToCreate); } +#ifdef USE_JPEG_TEXTURE else if (extension == "jpg" || extension == "jpeg"){ readJPEGPicture(filename, textureToCreate); } +#endif else { // Display an error message and throw an exception @@ -84,9 +87,7 @@ void TextureReaderWriter::loadTextureFromFile(const std::string& filename, } // Write a texture to a file -void TextureReaderWriter::writeTextureToFile(const std::string& filename, - const Texture2D& texture) - throw(runtime_error, invalid_argument){ +void TextureReaderWriter::writeTextureToFile(const std::string& filename,const Texture2D& texture) { // Get the extension of the file uint startPosExtension = filename.find_last_of("."); @@ -96,9 +97,11 @@ void TextureReaderWriter::writeTextureToFile(const std::string& filename, if (extension == "tga") { writeTGAPicture(filename, texture); } +#ifdef USE_JPEG_TEXTURE else if (extension == "jpg" || extension == "jpeg"){ writeJPEGPicture(filename, texture); } +#endif else { // Display an error message and throw an exception @@ -110,8 +113,7 @@ void TextureReaderWriter::writeTextureToFile(const std::string& filename, } // Load a TGA picture -void TextureReaderWriter::readTGAPicture(const std::string &filename, - Texture2D& textureToCreate) throw(runtime_error) { +void TextureReaderWriter::readTGAPicture(const std::string &filename, Texture2D& textureToCreate) { // Open the file std::ifstream stream(filename.c_str(), std::ios::binary); @@ -156,8 +158,7 @@ void TextureReaderWriter::readTGAPicture(const std::string &filename, // Write a TGA picture -void TextureReaderWriter::writeTGAPicture(const std::string& filename, - const Texture2D& texture) throw(runtime_error) { +void TextureReaderWriter::writeTGAPicture(const std::string& filename, const Texture2D& texture) { assert(texture.getID() != 0); // Bind the corresponding texture @@ -219,9 +220,10 @@ void TextureReaderWriter::writeTGAPicture(const std::string& filename, glBindTexture(GL_TEXTURE_2D, 0); } +#ifdef USE_JPEG_TEXTURE + // Read a JPEG picture -void TextureReaderWriter::readJPEGPicture(const std::string& filename, - Texture2D& textureToCreate) throw(std::runtime_error) { +void TextureReaderWriter::readJPEGPicture(const std::string& filename, Texture2D& textureToCreate) { struct jpeg_decompress_struct info; struct jpeg_error_mgr error; @@ -276,8 +278,7 @@ void TextureReaderWriter::readJPEGPicture(const std::string& filename, } // Write a JPEG picture -void TextureReaderWriter::writeJPEGPicture(const std::string& filename, - const Texture2D& texture) throw(std::runtime_error) { +void TextureReaderWriter::writeJPEGPicture(const std::string& filename, const Texture2D& texture) { struct jpeg_compress_struct info; struct jpeg_error_mgr error; @@ -327,3 +328,5 @@ void TextureReaderWriter::writeJPEGPicture(const std::string& filename, // Free allocated memory delete[] data; } + +#endif diff --git a/examples/common/opengl-framework/src/TextureReaderWriter.h b/examples/common/opengl-framework/src/TextureReaderWriter.h index 72a50baf..a83b9671 100644 --- a/examples/common/opengl-framework/src/TextureReaderWriter.h +++ b/examples/common/opengl-framework/src/TextureReaderWriter.h @@ -52,34 +52,26 @@ class TextureReaderWriter { TextureReaderWriter(); // Read a TGA picture - static void readTGAPicture(const std::string& filename, - Texture2D& textureToCreate) throw(std::runtime_error); + static void readTGAPicture(const std::string& filename, Texture2D& textureToCreate); // Write a TGA picture - static void writeTGAPicture(const std::string& filename, - const Texture2D& texture) throw(std::runtime_error); + static void writeTGAPicture(const std::string& filename, const Texture2D& texture); // Read a JPEG picture - static void readJPEGPicture(const std::string& filename, - Texture2D& textureToCreate) throw(std::runtime_error); + static void readJPEGPicture(const std::string& filename, Texture2D& textureToCreate); // Write a JPEG picture - static void writeJPEGPicture(const std::string& filename, - const Texture2D& texture) throw(std::runtime_error); + static void writeJPEGPicture(const std::string& filename, const Texture2D& texture); public : // -------------------- Methods -------------------- // // Load a texture from a file - static void loadTextureFromFile(const std::string& filename, - Texture2D& textureToCreate) - throw(std::runtime_error, std::invalid_argument); + static void loadTextureFromFile(const std::string& filename, Texture2D& textureToCreate); // Write a texture to a file - static void writeTextureToFile(const std::string& filename, - const Texture2D& texture) - throw(std::runtime_error, std::invalid_argument); + static void writeTextureToFile(const std::string& filename, const Texture2D& texture); }; } diff --git a/examples/common/opengl-framework/src/maths/Color.h b/examples/common/opengl-framework/src/maths/Color.h index e8ea2c40..a4629f8d 100644 --- a/examples/common/opengl-framework/src/maths/Color.h +++ b/examples/common/opengl-framework/src/maths/Color.h @@ -47,7 +47,7 @@ struct Color { // Constructor Color(float r, float g, float b, float a) : r(r), g(g), b(b), a(a) {} - // Constructor + // Copy-constructor Color(const Color& color) : r(color.r), g(color.g), b(color.b), a(color.a) {} // Destructor diff --git a/examples/common/opengl-framework/src/shaders/phong.frag b/examples/common/opengl-framework/src/shaders/phong.frag index 3102de06..009c5cc7 100644 --- a/examples/common/opengl-framework/src/shaders/phong.frag +++ b/examples/common/opengl-framework/src/shaders/phong.frag @@ -31,6 +31,7 @@ uniform vec3 light0SpecularColor; // Light 0 specular color uniform float shininess; // Shininess uniform sampler2D texture; // Texture uniform bool isTexture; // True if we need to use the texture +uniform vec4 vertexColor; // Vertex color // Varying variables varying vec3 vertexPosCameraSpace; // Camera-space position of the vertex @@ -40,10 +41,11 @@ varying vec2 texCoords; // Texture coordinates void main() { // Compute the ambient term - vec3 ambient = lightAmbientColor; + //vec3 ambient = lightAmbientColor; + vec3 ambient = vertexColor.rgb + 0.0 * lightAmbientColor; // Get the texture color - vec3 textureColor = vec3(1); + vec3 textureColor = vertexColor.rgb; if (isTexture) textureColor = texture2D(texture, texCoords).rgb; // Compute the surface normal vector diff --git a/examples/cubes/CMakeLists.txt b/examples/cubes/CMakeLists.txt index a7b5ee34..f65c428a 100644 --- a/examples/cubes/CMakeLists.txt +++ b/examples/cubes/CMakeLists.txt @@ -2,7 +2,7 @@ cmake_minimum_required(VERSION 2.6) # Project configuration -PROJECT(FallingCubes) +PROJECT(Cubes) # Copy the shaders used for the demo into the build directory FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/shaders/") @@ -12,6 +12,6 @@ INCLUDE_DIRECTORIES("../common/opengl-framework/src/" "../common/") # Create the example executable using the # compiled reactphysics3d static library -ADD_EXECUTABLE(fallingcubes FallingCubes.cpp Scene.cpp Scene.h "../common/Box.cpp" "../common/Box.h" "../common/Viewer.cpp" "../common/Viewer.h") +ADD_EXECUTABLE(cubes Cubes.cpp Scene.cpp Scene.h "../common/Box.cpp" "../common/Box.h" "../common/Viewer.cpp" "../common/Viewer.h") -TARGET_LINK_LIBRARIES(fallingcubes reactphysics3d openglframework) +TARGET_LINK_LIBRARIES(cubes reactphysics3d openglframework) diff --git a/examples/cubes/Cubes.cpp b/examples/cubes/Cubes.cpp index 54bbfa62..e0dc6ad0 100644 --- a/examples/cubes/Cubes.cpp +++ b/examples/cubes/Cubes.cpp @@ -51,7 +51,7 @@ int main(int argc, char** argv) { viewer = new Viewer(); Vector2 windowsSize = Vector2(800, 600); Vector2 windowsPosition = Vector2(100, 100); - bool initOK = viewer->init(argc, argv, "ReactPhysics3D Examples - Falling Cubes", windowsSize, windowsPosition); + bool initOK = viewer->init(argc, argv, "ReactPhysics3D Examples - Cubes", windowsSize, windowsPosition); if (!initOK) return 1; // Create the scene diff --git a/examples/cubes/Scene.cpp b/examples/cubes/Scene.cpp index dc2082cc..6fc35f68 100644 --- a/examples/cubes/Scene.cpp +++ b/examples/cubes/Scene.cpp @@ -65,12 +65,13 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), float angle = i * 30.0f; openglframework::Vector3 position(radius * cos(angle), 1 + i * (BOX_SIZE.y + 0.3f), - radius * sin(angle)); + 0); // Create a cube and a corresponding rigid in the dynamics world - Box* cube = new Box(BOX_SIZE, position , CUBE_MASS, mDynamicsWorld); + Box* cube = new Box(BOX_SIZE, position , BOX_MASS, mDynamicsWorld); cube->getRigidBody()->setIsMotionEnabled(true); + mMapBodyToBox.insert(std::make_pair(cube->getRigidBody(), cube)); // Change the material properties of the rigid body rp3d::Material& material = cube->getRigidBody()->getMaterial(); @@ -87,6 +88,8 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), // The floor must be a non-moving rigid body mFloor->getRigidBody()->setIsMotionEnabled(false); + mMapBodyToBox.insert(std::make_pair(mFloor->getRigidBody(), mFloor)); + // Change the material properties of the floor rigid body rp3d::Material& material = mFloor->getRigidBody()->getMaterial(); material.setBounciness(rp3d::decimal(0.3)); @@ -142,6 +145,15 @@ void Scene::simulate() { mFloor->updateTransform(); + // Set the color of the awake/sleeping bodies + for (uint i=0; igetRigidBody()->isSleeping()) { + mBoxes[i]->setColor(Color(1, 0, 0, 1)); + } + else { + mBoxes[i]->setColor(Color(0, 1, 0, 1)); + } + } } } diff --git a/examples/cubes/Scene.h b/examples/cubes/Scene.h index 45df3c4b..ed2033ef 100644 --- a/examples/cubes/Scene.h +++ b/examples/cubes/Scene.h @@ -34,8 +34,8 @@ // Constants const int NB_SPHERES = 20; // Number of boxes in the scene const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters -const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters -const float CUBE_MASS = 1.0f; // Box mass in kilograms +const openglframework::Vector3 FLOOR_SIZE(50, 0.5f, 50); // Floor dimensions in meters +const float BOX_MASS = 1.0f; // Box mass in kilograms const float FLOOR_MASS = 100.0f; // Floor mass in kilograms // Class Scene @@ -66,6 +66,9 @@ class Scene { /// True if the physics simulation is running bool mIsRunning; + // TODO : REMOVE THIS + std::map mMapBodyToBox; + public: // -------------------- Methods -------------------- // diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index 8e5e4e5c..b7b44159 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -246,7 +246,7 @@ void Scene::createSliderJoint() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 box1Dimension(2, 4, 2); - mSliderJointBottomBox = new Box(box1Dimension, positionBox1 , CUBE_MASS, mDynamicsWorld); + mSliderJointBottomBox = new Box(box1Dimension, positionBox1 , BOX_MASS, mDynamicsWorld); // The fist box cannot move mSliderJointBottomBox->getRigidBody()->setIsMotionEnabled(false); @@ -262,7 +262,7 @@ void Scene::createSliderJoint() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 box2Dimension(1.5f, 4, 1.5f); - mSliderJointTopBox = new Box(box2Dimension, positionBox2 , CUBE_MASS, mDynamicsWorld); + mSliderJointTopBox = new Box(box2Dimension, positionBox2 , BOX_MASS, mDynamicsWorld); // The second box is allowed to move mSliderJointTopBox->getRigidBody()->setIsMotionEnabled(true); @@ -301,7 +301,7 @@ void Scene::createPropellerHingeJoint() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 boxDimension(10, 1, 1); - mPropellerBox = new Box(boxDimension, positionBox1 , CUBE_MASS, mDynamicsWorld); + mPropellerBox = new Box(boxDimension, positionBox1 , BOX_MASS, mDynamicsWorld); // The fist box cannot move mPropellerBox->getRigidBody()->setIsMotionEnabled(true); @@ -339,7 +339,7 @@ void Scene::createFixedJoints() { // Create a box and a corresponding rigid in the dynamics world openglframework::Vector3 boxDimension(1.5, 1.5, 1.5); - mFixedJointBox1 = new Box(boxDimension, positionBox1 , CUBE_MASS, mDynamicsWorld); + mFixedJointBox1 = new Box(boxDimension, positionBox1 , BOX_MASS, mDynamicsWorld); // The fist box cannot move mFixedJointBox1->getRigidBody()->setIsMotionEnabled(true); @@ -354,7 +354,7 @@ void Scene::createFixedJoints() { openglframework::Vector3 positionBox2(-5, 7, 0); // Create a box and a corresponding rigid in the dynamics world - mFixedJointBox2 = new Box(boxDimension, positionBox2 , CUBE_MASS, mDynamicsWorld); + mFixedJointBox2 = new Box(boxDimension, positionBox2 , BOX_MASS, mDynamicsWorld); // The second box is allowed to move mFixedJointBox2->getRigidBody()->setIsMotionEnabled(true); diff --git a/examples/joints/Scene.h b/examples/joints/Scene.h index 1c3f72d5..90430d1d 100644 --- a/examples/joints/Scene.h +++ b/examples/joints/Scene.h @@ -34,7 +34,7 @@ // Constants const openglframework::Vector3 BOX_SIZE(2, 2, 2); // Box dimensions in meters const openglframework::Vector3 FLOOR_SIZE(20, 0.5f, 20); // Floor dimensions in meters -const float CUBE_MASS = 1.0f; // Box mass in kilograms +const float BOX_MASS = 1.0f; // Box mass in kilograms const float FLOOR_MASS = 100.0f; // Floor mass in kilograms const int NB_BALLSOCKETJOINT_BOXES = 7; // Number of Ball-And-Socket chain boxes const int NB_HINGE_BOXES = 7; // Number of Hinge chain boxes From 7ca5b88ce32a7b788f9465ea8122958fd5815453 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Thu, 12 Sep 2013 22:45:43 +0200 Subject: [PATCH 57/66] Fix compilation errors on Mac OS X and remove the compilation of freeglut from the CMake files --- CMakeLists.txt | 4 +- cmake/FindFreeglut.cmake | 32 +++++++++ examples/collisionshapes/Scene.cpp | 4 +- examples/common/Box.cpp | 5 -- examples/common/Box.h | 9 ++- .../common/opengl-framework/CMakeLists.txt | 33 ++++++++-- .../opengl-framework/cmake/FindFREEGLUT.cmake | 32 +++++++++ .../opengl-framework/cmake/FindGLEW.cmake | 65 +++++++++++++++++++ examples/cubes/Scene.cpp | 7 +- examples/cubes/Scene.h | 3 - examples/joints/Scene.cpp | 4 +- src/collision/shapes/CollisionShape.h | 1 + src/memory/MemoryAllocator.h | 1 + 13 files changed, 173 insertions(+), 27 deletions(-) create mode 100644 cmake/FindFreeglut.cmake create mode 100644 examples/common/opengl-framework/cmake/FindFREEGLUT.cmake create mode 100644 examples/common/opengl-framework/cmake/FindGLEW.cmake diff --git a/CMakeLists.txt b/CMakeLists.txt index 805fb5d3..ff4ecfaf 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -7,8 +7,8 @@ PROJECT(REACTPHYSICS3D) # Where to build the library SET(LIBRARY_OUTPUT_PATH lib/) -# Where to find the module to find special packages/libraries -set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/") +# Where to build the executables +SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin) # Options OPTION(COMPILE_EXAMPLES "Select this if you want to build the examples" OFF) diff --git a/cmake/FindFreeglut.cmake b/cmake/FindFreeglut.cmake new file mode 100644 index 00000000..c8ea9d3e --- /dev/null +++ b/cmake/FindFreeglut.cmake @@ -0,0 +1,32 @@ +# This module is used to try to find the Freeglut library and include files + +IF(WIN32) + FIND_PATH(FREEGLUT_INCLUDE_DIR NAMES GL/freeglut.h) + FIND_LIBRARY(FREEGLUT_LIBRARY NAMES freeglut freeglut_static) + PATHS ${OPENGL_LIBRARY_DIR}) +ELSE(WIN32) + + IF(APPLE) + # Do nothing, we do not want to use freeglut on Mac OS X + ELSE(APPLE) + FIND_PATH(FREEGLUT_INCLUDE_DIR GL/freeglut.h /usr/include/GL + /usr/openwin/share/include + /usr/openwin/include + /opt/graphics/OpenGL/include + /opt/graphics/OpenGL/contrib/libglut) + + FIND_LIBRARY(FREEGLUT_LIBRARY NAMES freeglut freeglut_static PATHS /usr/openwin/lib) + FIND_LIBRARY(Xi_LIBRARY Xi /usr/openwin/lib) + FIND_LIBRARY(Xmu_LIBRARY Xmu /usr/openwin/lib) + ENDIF(APPLE) +ENDIF(WIN32) + +INCLUDE(${CMAKE_CURRENT_LIST_DIR}/FindPackageHandleStandardArgs.cmake) +FIND_PACKAGE_HANDLE_STANDARD_ARGS(FREEGLUT REQUIRED_VARS FREEGLUT_LIBRARY FREEGLUT_INCLUDE_DIR) + +IF(FREEGLUT_FOUND) + SET(FREEGLUT_LIBRARIES ${FREEGLUT_LIBRARY} ${Xi_LIBRARY} ${Xmu_LIBRARY}) + SET(FREEGLUT_LIBRARY ${FREEGLUT_LIBRARIES}) +ENDIF(FREEGLUT_FOUND) + +MARK_AS_ADVANCED(FREEGLUT_INCLUDE_DIR FREEGLUT_LIBRARY Xi_LIBRARY Xmu_LIBRARY) diff --git a/examples/collisionshapes/Scene.cpp b/examples/collisionshapes/Scene.cpp index 038567a9..923e4206 100644 --- a/examples/collisionshapes/Scene.cpp +++ b/examples/collisionshapes/Scene.cpp @@ -400,8 +400,8 @@ void Scene::render() { mPhongShader.setMatrix4x4Uniform("projectionMatrix", camera.getProjectionMatrix()); mPhongShader.setVector3Uniform("light0PosCameraSpace", worldToCameraMatrix * mLight0.getOrigin()); mPhongShader.setVector3Uniform("lightAmbientColor", Vector3(0.3f, 0.3f, 0.3f)); - Color& diffColLight0 = mLight0.getDiffuseColor(); - Color& specColLight0 = mLight0.getSpecularColor(); + const Color& diffColLight0 = mLight0.getDiffuseColor(); + const Color& specColLight0 = mLight0.getSpecularColor(); mPhongShader.setVector3Uniform("light0DiffuseColor", Vector3(diffColLight0.r, diffColLight0.g, diffColLight0.b)); mPhongShader.setVector3Uniform("light0SpecularColor", Vector3(specColLight0.r, specColLight0.g, specColLight0.b)); mPhongShader.setFloatUniform("shininess", 200.0f); diff --git a/examples/common/Box.cpp b/examples/common/Box.cpp index 757a3127..a148921b 100644 --- a/examples/common/Box.cpp +++ b/examples/common/Box.cpp @@ -157,11 +157,6 @@ void Box::render(openglframework::Shader& shader, shader.unbind(); } -// Set the color of the box -void Box::setColor(openglframework::Color& color) { - mColor = color; -} - // Update the transform matrix of the box void Box::updateTransform() { diff --git a/examples/common/Box.h b/examples/common/Box.h index c507ed4c..e89368a2 100644 --- a/examples/common/Box.h +++ b/examples/common/Box.h @@ -74,7 +74,7 @@ class Box : public openglframework::Object3D { /// True if the VBOs have already been created static bool areVBOsCreated; - // TODO : REMOVE THIS + /// Main color of the box openglframework::Color mColor; // -------------------- Methods -------------------- // @@ -103,7 +103,7 @@ class Box : public openglframework::Object3D { void render(openglframework::Shader& shader, const openglframework::Matrix4& worldToCameraMatrix); /// Set the color of the box - void setColor(openglframework::Color& color); + void setColor(const openglframework::Color& color); }; // Return a pointer to the rigid body of the box @@ -111,4 +111,9 @@ inline rp3d::RigidBody* Box::getRigidBody() { return mRigidBody; } +// Set the color of the box +inline void Box::setColor(const openglframework::Color& color) { + mColor = color; +} + #endif diff --git a/examples/common/opengl-framework/CMakeLists.txt b/examples/common/opengl-framework/CMakeLists.txt index 7683a289..383c87a6 100644 --- a/examples/common/opengl-framework/CMakeLists.txt +++ b/examples/common/opengl-framework/CMakeLists.txt @@ -8,7 +8,7 @@ PROJECT(OPENGLFRAMEWORK) OPTION(USE_JPEG_TEXTURES "Select this if you want to use jpeg textures (libjpeg required)" OFF) # Where to find the module to find special packages/libraries -set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/") +set(CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake/") # Find OpenGL FIND_PACKAGE(OpenGL REQUIRED) @@ -26,6 +26,30 @@ else() MESSAGE("GLEW not found") endif() +# Find the GLUT/FREEGLUT library +IF(APPLE) + + # Find the GLUT library + FIND_PACKAGE(GLUT REQUIRED) + IF(GLUT_FOUND) + MESSAGE("GLUT found") + ELSE(GLUT_FOUND) + MESSAGE(SEND_ERROR "GLUT not found") + ENDIF(GLUT_FOUND) + +ELSE(APPLE) + + # Find the FREEGLUT library + FIND_PACKAGE(FREEGLUT REQUIRED) + IF(FREEGLUT_FOUND) + MESSAGE("FREEGLUT found") + ADD_DEFINITIONS(-DUSE_FREEGLUT) + ELSE(FREEGLUT_FOUND) + MESSAGE(SEND_ERROR "FREEGLUT not found") + ENDIF(FREEGLUT_FOUND) + +ENDIF(APPLE) + # If the user wants to use JPEG textures if(USE_JPEG_TEXTURES) @@ -38,11 +62,8 @@ if(USE_JPEG_TEXTURES) endif() endif() -# Freeglut -add_subdirectory(freeglut) - # Headers -INCLUDE_DIRECTORIES(src freeglut ${JPEG_INCLUDE_DIR}) +INCLUDE_DIRECTORIES(src ${OPENGL_INCLUDE_DIR} ${GLEW_INCLUDE_PATH} ${FREEGLUT_INCLUDE_DIR} ${GLUT_INCLUDE_DIR} ${JPEG_INCLUDE_DIR}) if(USE_JPEG_TEXTURES) add_definitions(-DUSE_JPEG_TEXTURE) @@ -62,4 +83,4 @@ ADD_LIBRARY ( ${OPENGLFRAMEWORK_SOURCES_FILES} ) -TARGET_LINK_LIBRARIES(openglframework ${GLEW_LIBRARIES} ${OPENGL_LIBRARY} freeglut_static) +TARGET_LINK_LIBRARIES(openglframework ${GLEW_LIBRARIES} ${OPENGL_LIBRARY} ${FREEGLUT_LIBRARY} ${GLUT_LIBRARY}) diff --git a/examples/common/opengl-framework/cmake/FindFREEGLUT.cmake b/examples/common/opengl-framework/cmake/FindFREEGLUT.cmake new file mode 100644 index 00000000..0ca4821e --- /dev/null +++ b/examples/common/opengl-framework/cmake/FindFREEGLUT.cmake @@ -0,0 +1,32 @@ +# This module is used to try to find the Freeglut library and include files + +IF(WIN32) + FIND_PATH(FREEGLUT_INCLUDE_DIR NAMES GL/freeglut.h) + FIND_LIBRARY(FREEGLUT_LIBRARY NAMES freeglut freeglut_static + PATHS ${OPENGL_LIBRARY_DIR}) +ELSE(WIN32) + + IF(APPLE) + # Do nothing, we do not want to use freeglut on Mac OS X + ELSE(APPLE) + FIND_PATH(FREEGLUT_INCLUDE_DIR GL/freeglut.h /usr/include/GL + /usr/openwin/share/include + /usr/openwin/include + /opt/graphics/OpenGL/include + /opt/graphics/OpenGL/contrib/libglut) + + FIND_LIBRARY(FREEGLUT_LIBRARY NAMES glut freeglut freeglut_static PATHS /usr/openwin/lib) + FIND_LIBRARY(Xi_LIBRARY Xi /usr/openwin/lib) + FIND_LIBRARY(Xmu_LIBRARY Xmu /usr/openwin/lib) + ENDIF(APPLE) +ENDIF(WIN32) + +INCLUDE(FindPackageHandleStandardArgs) +FIND_PACKAGE_HANDLE_STANDARD_ARGS(FREEGLUT REQUIRED_VARS FREEGLUT_LIBRARY FREEGLUT_INCLUDE_DIR) + +IF(FREEGLUT_FOUND) + SET(FREEGLUT_LIBRARIES ${FREEGLUT_LIBRARY} ${Xi_LIBRARY} ${Xmu_LIBRARY}) + SET(FREEGLUT_LIBRARY ${FREEGLUT_LIBRARIES}) +ENDIF(FREEGLUT_FOUND) + +MARK_AS_ADVANCED(FREEGLUT_INCLUDE_DIR FREEGLUT_LIBRARY Xi_LIBRARY Xmu_LIBRARY) diff --git a/examples/common/opengl-framework/cmake/FindGLEW.cmake b/examples/common/opengl-framework/cmake/FindGLEW.cmake new file mode 100644 index 00000000..c29c4eb2 --- /dev/null +++ b/examples/common/opengl-framework/cmake/FindGLEW.cmake @@ -0,0 +1,65 @@ +# +# Try to find GLEW library and include path. +# Once done this will define +# +# GLEW_FOUND +# GLEW_INCLUDE_PATH +# GLEW_LIBRARY +# + +IF (WIN32) +FIND_PATH( GLEW_INCLUDE_PATH GL/glew.h +$ENV{PROGRAMFILES}/GLEW/include +${GLEW_ROOT_DIR}/include +DOC "The directory where GL/glew.h resides") + +IF (NV_SYSTEM_PROCESSOR STREQUAL "AMD64") +FIND_LIBRARY( GLEW_LIBRARY +NAMES glew64 glew64s +PATHS +$ENV{PROGRAMFILES}/GLEW/lib +${PROJECT_SOURCE_DIR}/src/nvgl/glew/bin +${PROJECT_SOURCE_DIR}/src/nvgl/glew/lib +DOC "The GLEW library (64-bit)" +) +ELSE(NV_SYSTEM_PROCESSOR STREQUAL "AMD64") +FIND_LIBRARY( GLEW_LIBRARY +NAMES glew GLEW glew32 glew32s +PATHS +$ENV{PROGRAMFILES}/GLEW/lib +${PROJECT_SOURCE_DIR}/src/nvgl/glew/bin +${PROJECT_SOURCE_DIR}/src/nvgl/glew/lib +DOC "The GLEW library" +) +ENDIF(NV_SYSTEM_PROCESSOR STREQUAL "AMD64") +ELSE (WIN32) +FIND_PATH( GLEW_INCLUDE_PATH GL/glew.h +/usr/include +/usr/local/include +/sw/include +/opt/local/include +${GLEW_ROOT_DIR}/include +DOC "The directory where GL/glew.h resides") + +FIND_LIBRARY( GLEW_LIBRARY +NAMES GLEW glew +PATHS +/usr/lib64 +/usr/lib +/usr/local/lib64 +/usr/local/lib +/sw/lib +/opt/local/lib +${GLEW_ROOT_DIR}/lib +DOC "The GLEW library") +ENDIF (WIN32) + +SET(GLEW_FOUND "NO") +IF (GLEW_INCLUDE_PATH AND GLEW_LIBRARY) +SET(GLEW_LIBRARIES ${GLEW_LIBRARY}) +SET(GLEW_FOUND "YES") +ENDIF (GLEW_INCLUDE_PATH AND GLEW_LIBRARY) + + +include(FindPackageHandleStandardArgs) +find_package_handle_standard_args(GLEW DEFAULT_MSG GLEW_LIBRARY GLEW_INCLUDE_PATH) \ No newline at end of file diff --git a/examples/cubes/Scene.cpp b/examples/cubes/Scene.cpp index 6fc35f68..e3bc109a 100644 --- a/examples/cubes/Scene.cpp +++ b/examples/cubes/Scene.cpp @@ -71,7 +71,6 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), Box* cube = new Box(BOX_SIZE, position , BOX_MASS, mDynamicsWorld); cube->getRigidBody()->setIsMotionEnabled(true); - mMapBodyToBox.insert(std::make_pair(cube->getRigidBody(), cube)); // Change the material properties of the rigid body rp3d::Material& material = cube->getRigidBody()->getMaterial(); @@ -88,8 +87,6 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), // The floor must be a non-moving rigid body mFloor->getRigidBody()->setIsMotionEnabled(false); - mMapBodyToBox.insert(std::make_pair(mFloor->getRigidBody(), mFloor)); - // Change the material properties of the floor rigid body rp3d::Material& material = mFloor->getRigidBody()->getMaterial(); material.setBounciness(rp3d::decimal(0.3)); @@ -175,8 +172,8 @@ void Scene::render() { mPhongShader.setMatrix4x4Uniform("projectionMatrix", camera.getProjectionMatrix()); mPhongShader.setVector3Uniform("light0PosCameraSpace",worldToCameraMatrix * mLight0.getOrigin()); mPhongShader.setVector3Uniform("lightAmbientColor", Vector3(0.3f, 0.3f, 0.3f)); - Color& diffCol = mLight0.getDiffuseColor(); - Color& specCol = mLight0.getSpecularColor(); + const Color& diffCol = mLight0.getDiffuseColor(); + const Color& specCol = mLight0.getSpecularColor(); mPhongShader.setVector3Uniform("light0DiffuseColor", Vector3(diffCol.r, diffCol.g, diffCol.b)); mPhongShader.setVector3Uniform("light0SpecularColor", Vector3(specCol.r, specCol.g, specCol.b)); mPhongShader.setFloatUniform("shininess", 60.0f); diff --git a/examples/cubes/Scene.h b/examples/cubes/Scene.h index ed2033ef..a2802043 100644 --- a/examples/cubes/Scene.h +++ b/examples/cubes/Scene.h @@ -66,9 +66,6 @@ class Scene { /// True if the physics simulation is running bool mIsRunning; - // TODO : REMOVE THIS - std::map mMapBodyToBox; - public: // -------------------- Methods -------------------- // diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index b7b44159..5a566f17 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -166,8 +166,8 @@ void Scene::render() { mPhongShader.setVector3Uniform("light0PosCameraSpace",worldToCameraMatrix * mLight0.getOrigin()); mPhongShader.setMatrix4x4Uniform("projectionMatrix", camera.getProjectionMatrix()); mPhongShader.setVector3Uniform("lightAmbientColor", Vector3(0.3f, 0.3f, 0.3f)); - Color& diffCol = mLight0.getDiffuseColor(); - Color& specCol = mLight0.getSpecularColor(); + const Color& diffCol = mLight0.getDiffuseColor(); + const Color& specCol = mLight0.getSpecularColor(); mPhongShader.setVector3Uniform("light0DiffuseColor", Vector3(diffCol.r, diffCol.g, diffCol.b)); mPhongShader.setVector3Uniform("light0SpecularColor", Vector3(specCol.r, specCol.g, specCol.b)); mPhongShader.setFloatUniform("shininess", 60.0f); diff --git a/src/collision/shapes/CollisionShape.h b/src/collision/shapes/CollisionShape.h index 413015f8..4e6072c1 100644 --- a/src/collision/shapes/CollisionShape.h +++ b/src/collision/shapes/CollisionShape.h @@ -28,6 +28,7 @@ // Libraries #include +#include #include "../../mathematics/Vector3.h" #include "../../mathematics/Matrix3x3.h" #include "AABB.h" diff --git a/src/memory/MemoryAllocator.h b/src/memory/MemoryAllocator.h index 46caa3dd..23365949 100644 --- a/src/memory/MemoryAllocator.h +++ b/src/memory/MemoryAllocator.h @@ -27,6 +27,7 @@ #define REACTPHYSICS3D_MEMORY_ALLOCATOR_H // Libraries +#include #include "../configuration.h" /// ReactPhysics3D namespace From 76deef43580dffd40c12b23f59ce0dde36d00192 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sun, 15 Sep 2013 23:32:42 +0200 Subject: [PATCH 58/66] Fix issues on Mac OS X --- CMakeLists.txt | 3 --- examples/collisionshapes/CMakeLists.txt | 7 +++++-- examples/collisionshapes/CollisionShapes.cpp | 9 ++++++++- examples/common/Viewer.cpp | 2 ++ examples/common/opengl-framework/src/GlutViewer.h | 6 +++++- .../common/opengl-framework/src/shaders/phong.frag | 7 +++---- examples/cubes/CMakeLists.txt | 5 ++++- examples/cubes/Cubes.cpp | 10 +++++++++- examples/joints/CMakeLists.txt | 5 ++++- examples/joints/Joints.cpp | 9 ++++++++- 10 files changed, 48 insertions(+), 15 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index ff4ecfaf..5613cb52 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -7,9 +7,6 @@ PROJECT(REACTPHYSICS3D) # Where to build the library SET(LIBRARY_OUTPUT_PATH lib/) -# Where to build the executables -SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin) - # Options OPTION(COMPILE_EXAMPLES "Select this if you want to build the examples" OFF) OPTION(COMPILE_TESTS "Select this if you want to build the tests" OFF) diff --git a/examples/collisionshapes/CMakeLists.txt b/examples/collisionshapes/CMakeLists.txt index 932b10e8..bb0d377d 100644 --- a/examples/collisionshapes/CMakeLists.txt +++ b/examples/collisionshapes/CMakeLists.txt @@ -4,11 +4,14 @@ cmake_minimum_required(VERSION 2.6) # Project configuration PROJECT(CollisionShapes) +# Where to build the executable +SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin/collisionshapes/) + # Copy the shaders used for the demo into the build directory -FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/shaders/") +FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") # Copy the meshes used for the demo into the build directory -FILE(COPY "../common/meshes/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/meshes/") +FILE(COPY "../common/meshes/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/meshes/") # Headers INCLUDE_DIRECTORIES("../common/opengl-framework/src/" "../common/") diff --git a/examples/collisionshapes/CollisionShapes.cpp b/examples/collisionshapes/CollisionShapes.cpp index b444c466..cf1f9fdc 100644 --- a/examples/collisionshapes/CollisionShapes.cpp +++ b/examples/collisionshapes/CollisionShapes.cpp @@ -67,7 +67,11 @@ int main(int argc, char** argv) { glutMouseFunc(mouseButton); glutMotionFunc(mouseMotion); glutKeyboardFunc(keyboard); +#ifdef USE_FREEGLUT glutCloseFunc(finish); +#else + atexit(finish); +#endif // Glut main looop glutMainLoop(); @@ -115,7 +119,10 @@ void keyboard(unsigned char key, int x, int y) { // Escape key case 27: - glutLeaveMainLoop(); + #ifdef USE_FREEGLUT + // TODO : Check if we need to call finish() here + glutLeaveMainLoop(); + #endif break; // Space bar diff --git a/examples/common/Viewer.cpp b/examples/common/Viewer.cpp index e0114f00..1ff19ccc 100644 --- a/examples/common/Viewer.cpp +++ b/examples/common/Viewer.cpp @@ -68,6 +68,7 @@ void Viewer::displayGUI() { // Display the FPS void Viewer::displayFPS() { +#ifdef USE_FREEGLUT glMatrixMode(GL_PROJECTION); glLoadIdentity(); glOrtho(0, mCamera.getWidth(), mCamera.getHeight(), 0, -1, 1); @@ -80,4 +81,5 @@ void Viewer::displayFPS() { std::stringstream ss; ss << "FPS : " << fps; glutBitmapString(GLUT_BITMAP_HELVETICA_12, (const unsigned char*)ss.str().c_str()); +#endif } diff --git a/examples/common/opengl-framework/src/GlutViewer.h b/examples/common/opengl-framework/src/GlutViewer.h index 7718a1af..3d6d3c56 100644 --- a/examples/common/opengl-framework/src/GlutViewer.h +++ b/examples/common/opengl-framework/src/GlutViewer.h @@ -32,7 +32,11 @@ #include "maths/Vector2.h" #include #include -#include "GL/freeglut.h" +#ifdef __APPLE__ + #include "GLUT/glut.h" +#else + #include "GL/freeglut.h" +#endif namespace openglframework { diff --git a/examples/common/opengl-framework/src/shaders/phong.frag b/examples/common/opengl-framework/src/shaders/phong.frag index 009c5cc7..f09777da 100644 --- a/examples/common/opengl-framework/src/shaders/phong.frag +++ b/examples/common/opengl-framework/src/shaders/phong.frag @@ -41,8 +41,7 @@ varying vec2 texCoords; // Texture coordinates void main() { // Compute the ambient term - //vec3 ambient = lightAmbientColor; - vec3 ambient = vertexColor.rgb + 0.0 * lightAmbientColor; + vec3 ambient = lightAmbientColor; // Get the texture color vec3 textureColor = vertexColor.rgb; @@ -59,8 +58,8 @@ void main() { // Compute the specular term of light 0 vec3 V = normalize(-vertexPosCameraSpace); vec3 H0 = normalize(V + L0); - float specularFactor = pow(max(dot(N, H0), 0), shininess); - if (diffuseFactor < 0) specularFactor = 0.0; + float specularFactor = pow(max(dot(N, H0), 0.0), shininess); + if (diffuseFactor < 0.0) specularFactor = 0.0; vec3 specular = light0SpecularColor * specularFactor; // Compute the final color diff --git a/examples/cubes/CMakeLists.txt b/examples/cubes/CMakeLists.txt index f65c428a..90f64abb 100644 --- a/examples/cubes/CMakeLists.txt +++ b/examples/cubes/CMakeLists.txt @@ -4,8 +4,11 @@ cmake_minimum_required(VERSION 2.6) # Project configuration PROJECT(Cubes) +# Where to build the executable +SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin/cubes/) + # Copy the shaders used for the demo into the build directory -FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/shaders/") +FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") # Headers INCLUDE_DIRECTORIES("../common/opengl-framework/src/" "../common/") diff --git a/examples/cubes/Cubes.cpp b/examples/cubes/Cubes.cpp index e0dc6ad0..80c5d501 100644 --- a/examples/cubes/Cubes.cpp +++ b/examples/cubes/Cubes.cpp @@ -66,7 +66,12 @@ int main(int argc, char** argv) { glutMouseFunc(mouseButton); glutMotionFunc(mouseMotion); glutKeyboardFunc(keyboard); + +#ifdef USE_FREEGLUT glutCloseFunc(finish); +#else + atexit(finish); +#endif // Glut main looop glutMainLoop(); @@ -114,7 +119,10 @@ void keyboard(unsigned char key, int x, int y) { // Escape key case 27: - glutLeaveMainLoop(); + #ifdef USE_FREEGLUT + // TODO : Check if we need to call finish() here + glutLeaveMainLoop(); + #endif break; // Space bar diff --git a/examples/joints/CMakeLists.txt b/examples/joints/CMakeLists.txt index dc2eed0c..458a9731 100644 --- a/examples/joints/CMakeLists.txt +++ b/examples/joints/CMakeLists.txt @@ -4,8 +4,11 @@ cmake_minimum_required(VERSION 2.6) # Project configuration PROJECT(Joints) +# Where to build the executable +SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin/joints/) + # Copy the shaders used for the demo into the build directory -FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/shaders/") +FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") # Headers INCLUDE_DIRECTORIES("../common/opengl-framework/src/" "../common/") diff --git a/examples/joints/Joints.cpp b/examples/joints/Joints.cpp index f42ffb97..75c6892a 100644 --- a/examples/joints/Joints.cpp +++ b/examples/joints/Joints.cpp @@ -66,7 +66,11 @@ int main(int argc, char** argv) { glutMouseFunc(mouseButton); glutMotionFunc(mouseMotion); glutKeyboardFunc(keyboard); +#ifdef USE_FREEGLUT glutCloseFunc(finish); +#else + atexit(finish); +#endif // Glut main looop glutMainLoop(); @@ -114,7 +118,10 @@ void keyboard(unsigned char key, int x, int y) { // Escape key case 27: - glutLeaveMainLoop(); + #ifdef USE_FREEGLUT + // TODO : Check if we need to call finish() here + glutLeaveMainLoop(); + #endif break; // Space bar From a76100a37840aa1942d6c3b17b6cb8345fc6fa7d Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Mon, 16 Sep 2013 20:35:23 +0200 Subject: [PATCH 59/66] Modifications of CMake files and remove the freeglut source code --- CMakeLists.txt | 138 +- examples/CMakeLists.txt | 11 +- examples/collisionshapes/CMakeLists.txt | 25 +- examples/common/CMakeLists.txt | 2 +- .../common/opengl-framework/CMakeLists.txt | 78 +- .../opengl-framework/freeglut/CMakeLists.txt | 24 - .../opengl-framework/freeglut/COPYING.txt | 27 - .../opengl-framework/freeglut/GL/freeglut.h | 22 - .../freeglut/GL/freeglut_ext.h | 236 -- .../freeglut/GL/freeglut_std.h | 628 ---- .../opengl-framework/freeglut/GL/glut.h | 21 - .../opengl-framework/freeglut/VERSION.txt | 1 - .../freeglut/freeglut_callbacks.c | 412 --- .../freeglut/freeglut_cursor.c | 282 -- .../freeglut/freeglut_display.c | 103 - .../opengl-framework/freeglut/freeglut_ext.c | 233 -- .../opengl-framework/freeglut/freeglut_font.c | 384 --- .../freeglut/freeglut_font_data.c | 2020 ------------ .../freeglut/freeglut_gamemode.c | 819 ----- .../freeglut/freeglut_geometry.c | 1215 ------- .../freeglut/freeglut_glutfont_definitions.c | 108 - .../opengl-framework/freeglut/freeglut_init.c | 1180 ------- .../freeglut/freeglut_input_devices.c | 378 --- .../freeglut/freeglut_internal.h | 1023 ------ .../freeglut/freeglut_joystick.c | 1800 ----------- .../opengl-framework/freeglut/freeglut_main.c | 2511 --------------- .../opengl-framework/freeglut/freeglut_menu.c | 1002 ------ .../opengl-framework/freeglut/freeglut_misc.c | 214 -- .../freeglut/freeglut_overlay.c | 45 - .../freeglut/freeglut_spaceball.c | 471 --- .../freeglut/freeglut_state.c | 887 ----- .../freeglut/freeglut_stroke_mono_roman.c | 2849 ----------------- .../freeglut/freeglut_stroke_roman.c | 2849 ----------------- .../freeglut/freeglut_structure.c | 598 ---- .../freeglut/freeglut_teapot.c | 200 -- .../freeglut/freeglut_teapot_data.h | 2429 -------------- .../freeglut/freeglut_videoresize.c | 50 - .../freeglut/freeglut_window.c | 2162 ------------- .../freeglut/freeglut_xinput.c | 219 -- .../common/opengl-framework/freeglut/glut.h | 21 - examples/cubes/CMakeLists.txt | 22 +- examples/joints/CMakeLists.txt | 21 +- 42 files changed, 229 insertions(+), 27491 deletions(-) delete mode 100644 examples/common/opengl-framework/freeglut/CMakeLists.txt delete mode 100644 examples/common/opengl-framework/freeglut/COPYING.txt delete mode 100644 examples/common/opengl-framework/freeglut/GL/freeglut.h delete mode 100644 examples/common/opengl-framework/freeglut/GL/freeglut_ext.h delete mode 100644 examples/common/opengl-framework/freeglut/GL/freeglut_std.h delete mode 100644 examples/common/opengl-framework/freeglut/GL/glut.h delete mode 100644 examples/common/opengl-framework/freeglut/VERSION.txt delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_callbacks.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_cursor.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_display.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_ext.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_font.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_font_data.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_gamemode.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_geometry.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_glutfont_definitions.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_init.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_input_devices.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_internal.h delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_joystick.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_main.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_menu.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_misc.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_overlay.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_spaceball.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_state.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_stroke_mono_roman.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_stroke_roman.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_structure.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_teapot.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_teapot_data.h delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_videoresize.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_window.c delete mode 100644 examples/common/opengl-framework/freeglut/freeglut_xinput.c delete mode 100644 examples/common/opengl-framework/freeglut/glut.h diff --git a/CMakeLists.txt b/CMakeLists.txt index 5613cb52..65ce371a 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -16,34 +16,130 @@ OPTION(DOUBLE_PRECISION_ENABLED "Select this if you want to compile using double # Headers INCLUDE_DIRECTORIES(src) -IF (PROFILING_ENABLED) - ADD_DEFINITIONS(-DIS_PROFILING_ACTIVE) -ENDIF (PROFILING_ENABLED) +IF(PROFILING_ENABLED) + ADD_DEFINITIONS(-DIS_PROFILING_ACTIVE) +ENDIF(PROFILING_ENABLED) -IF (DOUBLE_PRECISION_ENABLED) - ADD_DEFINITIONS(-DIS_DOUBLE_PRECISION_ENABLED) -ENDIF (DOUBLE_PRECISION_ENABLED) +IF(DOUBLE_PRECISION_ENABLED) + ADD_DEFINITIONS(-DIS_DOUBLE_PRECISION_ENABLED) +ENDIF(DOUBLE_PRECISION_ENABLED) -# Library configuration -file ( - GLOB_RECURSE - source_files - src/* +# Source files +SET (REACTPHYSICS3D_SOURCES + "src/configuration.h" + "src/decimal.h" + "src/reactphysics3d.h" + "src/body/Body.h" + "src/body/Body.cpp" + "src/body/CollisionBody.h" + "src/body/CollisionBody.cpp" + "src/body/RigidBody.h" + "src/body/RigidBody.cpp" + "src/collision/broadphase/BroadPhaseAlgorithm.h" + "src/collision/broadphase/BroadPhaseAlgorithm.cpp" + "src/collision/broadphase/NoBroadPhaseAlgorithm.h" + "src/collision/broadphase/NoBroadPhaseAlgorithm.cpp" + "src/collision/broadphase/PairManager.h" + "src/collision/broadphase/PairManager.cpp" + "src/collision/broadphase/SweepAndPruneAlgorithm.h" + "src/collision/broadphase/SweepAndPruneAlgorithm.cpp" + "src/collision/narrowphase/EPA/EdgeEPA.h" + "src/collision/narrowphase/EPA/EdgeEPA.cpp" + "src/collision/narrowphase/EPA/EPAAlgorithm.h" + "src/collision/narrowphase/EPA/EPAAlgorithm.cpp" + "src/collision/narrowphase/EPA/TriangleEPA.h" + "src/collision/narrowphase/EPA/TriangleEPA.cpp" + "src/collision/narrowphase/EPA/TrianglesStore.h" + "src/collision/narrowphase/EPA/TrianglesStore.cpp" + "src/collision/narrowphase/GJK/Simplex.h" + "src/collision/narrowphase/GJK/Simplex.cpp" + "src/collision/narrowphase/GJK/GJKAlgorithm.h" + "src/collision/narrowphase/GJK/GJKAlgorithm.cpp" + "src/collision/narrowphase/NarrowPhaseAlgorithm.h" + "src/collision/narrowphase/NarrowPhaseAlgorithm.cpp" + "src/collision/narrowphase/SphereVsSphereAlgorithm.h" + "src/collision/narrowphase/SphereVsSphereAlgorithm.cpp" + "src/collision/shapes/AABB.h" + "src/collision/shapes/AABB.cpp" + "src/collision/shapes/BoxShape.h" + "src/collision/shapes/BoxShape.cpp" + "src/collision/shapes/CapsuleShape.h" + "src/collision/shapes/CapsuleShape.cpp" + "src/collision/shapes/CollisionShape.h" + "src/collision/shapes/CollisionShape.cpp" + "src/collision/shapes/ConeShape.h" + "src/collision/shapes/ConeShape.cpp" + "src/collision/shapes/ConvexMeshShape.h" + "src/collision/shapes/ConvexMeshShape.cpp" + "src/collision/shapes/CylinderShape.h" + "src/collision/shapes/CylinderShape.cpp" + "src/collision/shapes/SphereShape.h" + "src/collision/shapes/SphereShape.cpp" + "src/collision/BroadPhasePair.h" + "src/collision/BroadPhasePair.cpp" + "src/collision/CollisionDetection.h" + "src/collision/CollisionDetection.cpp" + "src/constraint/BallAndSocketJoint.h" + "src/constraint/BallAndSocketJoint.cpp" + "src/constraint/ContactPoint.h" + "src/constraint/ContactPoint.cpp" + "src/constraint/FixedJoint.h" + "src/constraint/FixedJoint.cpp" + "src/constraint/HingeJoint.h" + "src/constraint/HingeJoint.cpp" + "src/constraint/Joint.h" + "src/constraint/Joint.cpp" + "src/constraint/SliderJoint.h" + "src/constraint/SliderJoint.cpp" + "src/engine/CollisionWorld.h" + "src/engine/CollisionWorld.cpp" + "src/engine/ConstraintSolver.h" + "src/engine/ConstraintSolver.cpp" + "src/engine/ContactManifold.h" + "src/engine/ContactManifold.cpp" + "src/engine/ContactSolver.h" + "src/engine/ContactSolver.cpp" + "src/engine/DynamicsWorld.h" + "src/engine/DynamicsWorld.cpp" + "src/engine/EventListener.h" + "src/engine/Impulse.h" + "src/engine/Island.h" + "src/engine/Island.cpp" + "src/engine/Material.h" + "src/engine/Material.cpp" + "src/engine/OverlappingPair.h" + "src/engine/OverlappingPair.cpp" + "src/engine/Profiler.h" + "src/engine/Profiler.cpp" + "src/engine/Timer.h" + "src/engine/Timer.cpp" + "src/mathematics/mathematics.h" + "src/mathematics/mathematics_functions.h" + "src/mathematics/Matrix2x2.h" + "src/mathematics/Matrix2x2.cpp" + "src/mathematics/Matrix3x3.h" + "src/mathematics/Matrix3x3.cpp" + "src/mathematics/Quaternion.h" + "src/mathematics/Quaternion.cpp" + "src/mathematics/Transform.h" + "src/mathematics/Transform.cpp" + "src/mathematics/Vector2.h" + "src/mathematics/Vector2.cpp" + "src/mathematics/Vector3.h" + "src/mathematics/Vector3.cpp" + "src/memory/MemoryAllocator.h" + "src/memory/MemoryAllocator.cpp" ) -# Require the reactphysics3d code to be compiled in a static library -ADD_LIBRARY ( - reactphysics3d - STATIC - ${source_files} -) +# Create the library +ADD_LIBRARY (reactphysics3d STATIC ${REACTPHYSICS3D_SOURCES}) # If we need to compile the examples -IF (COMPILE_EXAMPLES) +IF(COMPILE_EXAMPLES) add_subdirectory(examples/) -ENDIF (COMPILE_EXAMPLES) +ENDIF(COMPILE_EXAMPLES) # If we need to compile the tests -IF (COMPILE_TESTS) +IF(COMPILE_TESTS) add_subdirectory(test/) -ENDIF (COMPILE_TESTS) +ENDIF(COMPILE_TESTS) diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt index 6399abeb..edc71428 100644 --- a/examples/CMakeLists.txt +++ b/examples/CMakeLists.txt @@ -1,7 +1,10 @@ # Minimum cmake version required CMAKE_MINIMUM_REQUIRED(VERSION 2.6) -add_subdirectory(common/) -add_subdirectory(cubes/) -add_subdirectory(joints/) -add_subdirectory(collisionshapes/) +# Set a variable for the directory of the opengl-framework +SET(OPENGLFRAMEWORK_DIR "${CMAKE_CURRENT_SOURCE_DIR}/common/opengl-framework") + +ADD_SUBDIRECTORY(common/) +ADD_SUBDIRECTORY(cubes/) +ADD_SUBDIRECTORY(joints/) +ADD_SUBDIRECTORY(collisionshapes/) diff --git a/examples/collisionshapes/CMakeLists.txt b/examples/collisionshapes/CMakeLists.txt index bb0d377d..ffbdebbe 100644 --- a/examples/collisionshapes/CMakeLists.txt +++ b/examples/collisionshapes/CMakeLists.txt @@ -8,16 +8,31 @@ PROJECT(CollisionShapes) SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin/collisionshapes/) # Copy the shaders used for the demo into the build directory -FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") +FILE(COPY "${OPENGLFRAMEWORK_DIR}/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") # Copy the meshes used for the demo into the build directory FILE(COPY "../common/meshes/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/meshes/") # Headers -INCLUDE_DIRECTORIES("../common/opengl-framework/src/" "../common/") +INCLUDE_DIRECTORIES("${OPENGLFRAMEWORK_DIR}/src/" "../common/") -# Create the example executable using the -# compiled reactphysics3d static library -ADD_EXECUTABLE(collisionshapes CollisionShapes.cpp Scene.cpp Scene.h "../common/VisualContactPoint.cpp" "../common/ConvexMesh.cpp" "../common/Capsule.cpp" "../common/Sphere.cpp" "../common/Cylinder.cpp" "../common/Cone.cpp" "../common/Box.cpp" "../common/Viewer.cpp") +# Source files +SET(COLLISION_SHAPES_SOURCES + CollisionShapes.cpp + Scene.cpp + Scene.h + "../common/VisualContactPoint.cpp" + "../common/ConvexMesh.cpp" + "../common/Capsule.cpp" + "../common/Sphere.cpp" + "../common/Cylinder.cpp" + "../common/Cone.cpp" + "../common/Box.cpp" + "../common/Viewer.cpp" +) +# Create the executable +ADD_EXECUTABLE(collisionshapes ${COLLISION_SHAPES_SOURCES}) + +# Link with libraries TARGET_LINK_LIBRARIES(collisionshapes reactphysics3d openglframework) diff --git a/examples/common/CMakeLists.txt b/examples/common/CMakeLists.txt index ba4484e0..eb9eee29 100644 --- a/examples/common/CMakeLists.txt +++ b/examples/common/CMakeLists.txt @@ -1,4 +1,4 @@ # Minimum cmake version required CMAKE_MINIMUM_REQUIRED(VERSION 2.6) -add_subdirectory(opengl-framework/) +ADD_SUBDIRECTORY(opengl-framework/) diff --git a/examples/common/opengl-framework/CMakeLists.txt b/examples/common/opengl-framework/CMakeLists.txt index 383c87a6..ab8192db 100644 --- a/examples/common/opengl-framework/CMakeLists.txt +++ b/examples/common/opengl-framework/CMakeLists.txt @@ -8,23 +8,23 @@ PROJECT(OPENGLFRAMEWORK) OPTION(USE_JPEG_TEXTURES "Select this if you want to use jpeg textures (libjpeg required)" OFF) # Where to find the module to find special packages/libraries -set(CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake/") +SET(CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake/") # Find OpenGL FIND_PACKAGE(OpenGL REQUIRED) -if(OPENGL_FOUND) +IF(OPENGL_FOUND) MESSAGE("OpenGL found") -else() +ELSE() MESSAGE("OpenGL not found") -endif() +ENDIF() # Find the GLEW library FIND_PACKAGE(GLEW REQUIRED) -if(GLEW_FOUND) +IF(GLEW_FOUND) MESSAGE("GLEW found") -else() +ELSE() MESSAGE("GLEW not found") -endif() +ENDIF() # Find the GLUT/FREEGLUT library IF(APPLE) @@ -51,36 +51,60 @@ ELSE(APPLE) ENDIF(APPLE) # If the user wants to use JPEG textures -if(USE_JPEG_TEXTURES) +IF(USE_JPEG_TEXTURES) # Find the LIBJPEG library FIND_PACKAGE(JPEG REQUIRED) - if(JPEG_FOUND) + IF(JPEG_FOUND) MESSAGE("LIBJPEG found") - else() + ELSE() MESSAGE("LIBJPEG not found") - endif() -endif() + ENDIF() +ENDIF() # Headers INCLUDE_DIRECTORIES(src ${OPENGL_INCLUDE_DIR} ${GLEW_INCLUDE_PATH} ${FREEGLUT_INCLUDE_DIR} ${GLUT_INCLUDE_DIR} ${JPEG_INCLUDE_DIR}) -if(USE_JPEG_TEXTURES) - add_definitions(-DUSE_JPEG_TEXTURE) -endif() - -# Library configuration -file ( - GLOB_RECURSE - OPENGLFRAMEWORK_SOURCES_FILES - src/* +# Source files +SET(OPENGL_FRAMEWORK_SOURCES + "src/maths/Color.h" + "src/maths/Matrix3.h" + "src/maths/Matrix4.h" + "src/maths/Vector2.h" + "src/maths/Vector3.h" + "src/maths/Vector4.h" + "src/Camera.cpp" + "src/Camera.h" + "src/definitions.h" + "src/FrameBufferObject.cpp" + "src/FrameBufferObject.h" + "src/GlutViewer.cpp" + "src/GlutViewer.h" + "src/Light.h" + "src/Light.cpp" + "src/Mesh.h" + "src/Mesh.cpp" + "src/MeshReaderWriter.h" + "src/MeshReaderWriter.cpp" + "src/Object3D.h" + "src/Object3D.cpp" + "src/openglframework.h" + "src/Shader.h" + "src/Shader.cpp" + "src/Texture2D.h" + "src/Texture2D.cpp" + "src/TextureReaderWriter.h" + "src/TextureReaderWriter.cpp" + "src/VertexBufferObject.h" + "src/VertexBufferObject.cpp" ) -# Require the opengl-framework code to be compiled in a static library -ADD_LIBRARY ( - openglframework - STATIC - ${OPENGLFRAMEWORK_SOURCES_FILES} -) +IF(USE_JPEG_TEXTURES) + ADD_DEFINITIONS(-DUSE_JPEG_TEXTURE) +ENDIF() +# Create the library +ADD_LIBRARY (openglframework STATIC ${OPENGL_FRAMEWORK_SOURCES}) + +# Link with others libraries TARGET_LINK_LIBRARIES(openglframework ${GLEW_LIBRARIES} ${OPENGL_LIBRARY} ${FREEGLUT_LIBRARY} ${GLUT_LIBRARY}) diff --git a/examples/common/opengl-framework/freeglut/CMakeLists.txt b/examples/common/opengl-framework/freeglut/CMakeLists.txt deleted file mode 100644 index bfa701ed..00000000 --- a/examples/common/opengl-framework/freeglut/CMakeLists.txt +++ /dev/null @@ -1,24 +0,0 @@ -add_definitions( -DFREEGLUT_EXPORTS -DFREEGLUT_STATIC -D_CRT_SECURE_NO_WARNINGS ) - -if(APPLE) - include_directories( /usr/X11/include ) -endif(APPLE) - -if(UNIX) - add_definitions( -D__unix__ -DHAVE_FCNTL_H -DHAVE_GETTIMEOFDAY ) -endif(UNIX) - -file ( - GLOB_RECURSE - FREEGLUT_SOURCES - ./* -) - -include_directories ( - ${OPENGL_INCLUDE_DIR} - . -) - -add_library(freeglut_static - ${FREEGLUT_SOURCES} -) diff --git a/examples/common/opengl-framework/freeglut/COPYING.txt b/examples/common/opengl-framework/freeglut/COPYING.txt deleted file mode 100644 index fc36ad99..00000000 --- a/examples/common/opengl-framework/freeglut/COPYING.txt +++ /dev/null @@ -1,27 +0,0 @@ - - Freeglut Copyright - ------------------ - - Freeglut code without an explicit copyright is covered by the following - copyright: - - Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - 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 or substantial portions of the Software. - - 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 - PAWEL W. OLSZTA 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. - - Except as contained in this notice, the name of Pawel W. Olszta shall not be - used in advertising or otherwise to promote the sale, use or other dealings - in this Software without prior written authorization from Pawel W. Olszta. diff --git a/examples/common/opengl-framework/freeglut/GL/freeglut.h b/examples/common/opengl-framework/freeglut/GL/freeglut.h deleted file mode 100644 index 0e6f8c6a..00000000 --- a/examples/common/opengl-framework/freeglut/GL/freeglut.h +++ /dev/null @@ -1,22 +0,0 @@ -#ifndef __FREEGLUT_H__ -#define __FREEGLUT_H__ - -/* - * freeglut.h - * - * The freeglut library include file - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include "freeglut_std.h" -#include "freeglut_ext.h" - -/*** END OF FILE ***/ - -#endif /* __FREEGLUT_H__ */ diff --git a/examples/common/opengl-framework/freeglut/GL/freeglut_ext.h b/examples/common/opengl-framework/freeglut/GL/freeglut_ext.h deleted file mode 100644 index 6bf84b86..00000000 --- a/examples/common/opengl-framework/freeglut/GL/freeglut_ext.h +++ /dev/null @@ -1,236 +0,0 @@ -#ifndef __FREEGLUT_EXT_H__ -#define __FREEGLUT_EXT_H__ - -/* - * freeglut_ext.h - * - * The non-GLUT-compatible extensions to the freeglut library include file - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 2 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#ifdef __cplusplus - extern "C" { -#endif - -/* - * Additional GLUT Key definitions for the Special key function - */ -#define GLUT_KEY_NUM_LOCK 0x006D -#define GLUT_KEY_BEGIN 0x006E -#define GLUT_KEY_DELETE 0x006F -#define GLUT_KEY_SHIFT_L 0x0070 -#define GLUT_KEY_SHIFT_R 0x0071 -#define GLUT_KEY_CTRL_L 0x0072 -#define GLUT_KEY_CTRL_R 0x0073 -#define GLUT_KEY_ALT_L 0x0074 -#define GLUT_KEY_ALT_R 0x0075 - -/* - * GLUT API Extension macro definitions -- behaviour when the user clicks on an "x" to close a window - */ -#define GLUT_ACTION_EXIT 0 -#define GLUT_ACTION_GLUTMAINLOOP_RETURNS 1 -#define GLUT_ACTION_CONTINUE_EXECUTION 2 - -/* - * Create a new rendering context when the user opens a new window? - */ -#define GLUT_CREATE_NEW_CONTEXT 0 -#define GLUT_USE_CURRENT_CONTEXT 1 - -/* - * Direct/Indirect rendering context options (has meaning only in Unix/X11) - */ -#define GLUT_FORCE_INDIRECT_CONTEXT 0 -#define GLUT_ALLOW_DIRECT_CONTEXT 1 -#define GLUT_TRY_DIRECT_CONTEXT 2 -#define GLUT_FORCE_DIRECT_CONTEXT 3 - -/* - * GLUT API Extension macro definitions -- the glutGet parameters - */ -#define GLUT_INIT_STATE 0x007C - -#define GLUT_ACTION_ON_WINDOW_CLOSE 0x01F9 - -#define GLUT_WINDOW_BORDER_WIDTH 0x01FA -#define GLUT_WINDOW_HEADER_HEIGHT 0x01FB - -#define GLUT_VERSION 0x01FC - -#define GLUT_RENDERING_CONTEXT 0x01FD -#define GLUT_DIRECT_RENDERING 0x01FE - -#define GLUT_FULL_SCREEN 0x01FF - -/* - * New tokens for glutInitDisplayMode. - * Only one GLUT_AUXn bit may be used at a time. - * Value 0x0400 is defined in OpenGLUT. - */ -#define GLUT_AUX 0x1000 - -#define GLUT_AUX1 0x1000 -#define GLUT_AUX2 0x2000 -#define GLUT_AUX3 0x4000 -#define GLUT_AUX4 0x8000 - -/* - * Context-related flags, see freeglut_state.c - */ -#define GLUT_INIT_MAJOR_VERSION 0x0200 -#define GLUT_INIT_MINOR_VERSION 0x0201 -#define GLUT_INIT_FLAGS 0x0202 -#define GLUT_INIT_PROFILE 0x0203 - -/* - * Flags for glutInitContextFlags, see freeglut_init.c - */ -#define GLUT_DEBUG 0x0001 -#define GLUT_FORWARD_COMPATIBLE 0x0002 - - -/* - * Flags for glutInitContextProfile, see freeglut_init.c - */ -#define GLUT_CORE_PROFILE 0x0001 -#define GLUT_COMPATIBILITY_PROFILE 0x0002 - -/* - * Process loop function, see freeglut_main.c - */ -FGAPI void FGAPIENTRY glutMainLoopEvent( void ); -FGAPI void FGAPIENTRY glutLeaveMainLoop( void ); -FGAPI void FGAPIENTRY glutExit ( void ); - -/* - * Window management functions, see freeglut_window.c - */ -FGAPI void FGAPIENTRY glutFullScreenToggle( void ); -FGAPI void FGAPIENTRY glutLeaveFullScreen( void ); - -/* - * Window-specific callback functions, see freeglut_callbacks.c - */ -FGAPI void FGAPIENTRY glutMouseWheelFunc( void (* callback)( int, int, int, int ) ); -FGAPI void FGAPIENTRY glutCloseFunc( void (* callback)( void ) ); -FGAPI void FGAPIENTRY glutWMCloseFunc( void (* callback)( void ) ); -/* A. Donev: Also a destruction callback for menus */ -FGAPI void FGAPIENTRY glutMenuDestroyFunc( void (* callback)( void ) ); - -/* - * State setting and retrieval functions, see freeglut_state.c - */ -FGAPI void FGAPIENTRY glutSetOption ( GLenum option_flag, int value ); -FGAPI int * FGAPIENTRY glutGetModeValues(GLenum mode, int * size); -/* A.Donev: User-data manipulation */ -FGAPI void* FGAPIENTRY glutGetWindowData( void ); -FGAPI void FGAPIENTRY glutSetWindowData(void* data); -FGAPI void* FGAPIENTRY glutGetMenuData( void ); -FGAPI void FGAPIENTRY glutSetMenuData(void* data); - -/* - * Font stuff, see freeglut_font.c - */ -FGAPI int FGAPIENTRY glutBitmapHeight( void* font ); -FGAPI GLfloat FGAPIENTRY glutStrokeHeight( void* font ); -FGAPI void FGAPIENTRY glutBitmapString( void* font, const unsigned char *string ); -FGAPI void FGAPIENTRY glutStrokeString( void* font, const unsigned char *string ); - -/* - * Geometry functions, see freeglut_geometry.c - */ -FGAPI void FGAPIENTRY glutWireRhombicDodecahedron( void ); -FGAPI void FGAPIENTRY glutSolidRhombicDodecahedron( void ); -FGAPI void FGAPIENTRY glutWireSierpinskiSponge ( int num_levels, GLdouble offset[3], GLdouble scale ); -FGAPI void FGAPIENTRY glutSolidSierpinskiSponge ( int num_levels, GLdouble offset[3], GLdouble scale ); -FGAPI void FGAPIENTRY glutWireCylinder( GLdouble radius, GLdouble height, GLint slices, GLint stacks); -FGAPI void FGAPIENTRY glutSolidCylinder( GLdouble radius, GLdouble height, GLint slices, GLint stacks); - -/* - * Extension functions, see freeglut_ext.c - */ -typedef void (*GLUTproc)(); -FGAPI GLUTproc FGAPIENTRY glutGetProcAddress( const char *procName ); - -/* - * Multi-touch/multi-pointer extensions - */ - -#define GLUT_HAS_MULTI 1 - -FGAPI void FGAPIENTRY glutMultiEntryFunc( void (* callback)( int, int ) ); -FGAPI void FGAPIENTRY glutMultiButtonFunc( void (* callback)( int, int, int, int, int ) ); -FGAPI void FGAPIENTRY glutMultiMotionFunc( void (* callback)( int, int, int ) ); -FGAPI void FGAPIENTRY glutMultiPassiveFunc( void (* callback)( int, int, int ) ); - -/* - * Joystick functions, see freeglut_joystick.c - */ -/* USE OF THESE FUNCTIONS IS DEPRECATED !!!!! */ -/* If you have a serious need for these functions in your application, please either - * contact the "freeglut" developer community at freeglut-developer@lists.sourceforge.net, - * switch to the OpenGLUT library, or else port your joystick functionality over to PLIB's - * "js" library. - */ -int glutJoystickGetNumAxes( int ident ); -int glutJoystickGetNumButtons( int ident ); -int glutJoystickNotWorking( int ident ); -float glutJoystickGetDeadBand( int ident, int axis ); -void glutJoystickSetDeadBand( int ident, int axis, float db ); -float glutJoystickGetSaturation( int ident, int axis ); -void glutJoystickSetSaturation( int ident, int axis, float st ); -void glutJoystickSetMinRange( int ident, float *axes ); -void glutJoystickSetMaxRange( int ident, float *axes ); -void glutJoystickSetCenter( int ident, float *axes ); -void glutJoystickGetMinRange( int ident, float *axes ); -void glutJoystickGetMaxRange( int ident, float *axes ); -void glutJoystickGetCenter( int ident, float *axes ); - -/* - * Initialization functions, see freeglut_init.c - */ -FGAPI void FGAPIENTRY glutInitContextVersion( int majorVersion, int minorVersion ); -FGAPI void FGAPIENTRY glutInitContextFlags( int flags ); -FGAPI void FGAPIENTRY glutInitContextProfile( int profile ); - -/* to get the typedef for va_list */ -#include - -FGAPI void FGAPIENTRY glutInitErrorFunc( void (* vError)( const char *fmt, va_list ap ) ); -FGAPI void FGAPIENTRY glutInitWarningFunc( void (* vWarning)( const char *fmt, va_list ap ) ); - -/* - * GLUT API macro definitions -- the display mode definitions - */ -#define GLUT_CAPTIONLESS 0x0400 -#define GLUT_BORDERLESS 0x0800 -#define GLUT_SRGB 0x1000 - -#ifdef __cplusplus - } -#endif - -/*** END OF FILE ***/ - -#endif /* __FREEGLUT_EXT_H__ */ diff --git a/examples/common/opengl-framework/freeglut/GL/freeglut_std.h b/examples/common/opengl-framework/freeglut/GL/freeglut_std.h deleted file mode 100644 index ba1b7165..00000000 --- a/examples/common/opengl-framework/freeglut/GL/freeglut_std.h +++ /dev/null @@ -1,628 +0,0 @@ -#ifndef __FREEGLUT_STD_H__ -#define __FREEGLUT_STD_H__ - -/* - * freeglut_std.h - * - * The GLUT-compatible part of the freeglut library include file - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 2 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#ifdef __cplusplus - extern "C" { -#endif - -/* - * Under windows, we have to differentiate between static and dynamic libraries - */ -#ifdef _WIN32 -/* #pragma may not be supported by some compilers. - * Discussion by FreeGLUT developers suggests that - * Visual C++ specific code involving pragmas may - * need to move to a separate header. 24th Dec 2003 - */ - -/* Define FREEGLUT_LIB_PRAGMAS to 1 to include library - * pragmas or to 0 to exclude library pragmas. - * The default behavior depends on the compiler/platform. - */ -# ifndef FREEGLUT_LIB_PRAGMAS -# if ( defined(_MSC_VER) || defined(__WATCOMC__) ) && !defined(_WIN32_WCE) -# define FREEGLUT_LIB_PRAGMAS 1 -# else -# define FREEGLUT_LIB_PRAGMAS 0 -# endif -# endif - -# ifndef WIN32_LEAN_AND_MEAN -# define WIN32_LEAN_AND_MEAN 1 -# endif -# ifndef NOMINMAX -# define NOMINMAX -# endif -# include - -/* Windows static library */ -# ifdef FREEGLUT_STATIC - -# define FGAPI -# define FGAPIENTRY - - /* Link with Win32 static freeglut lib */ -# if FREEGLUT_LIB_PRAGMAS -# pragma comment (lib, "freeglut_static.lib") -# endif - -/* Windows shared library (DLL) */ -# else - -# define FGAPIENTRY __stdcall -# if defined(FREEGLUT_EXPORTS) -# define FGAPI __declspec(dllexport) -# else -# define FGAPI __declspec(dllimport) - - /* Link with Win32 shared freeglut lib */ -# if FREEGLUT_LIB_PRAGMAS -# pragma comment (lib, "freeglut.lib") -# endif - -# endif - -# endif - -/* Drag in other Windows libraries as required by FreeGLUT */ -# if FREEGLUT_LIB_PRAGMAS -# pragma comment (lib, "glu32.lib") /* link OpenGL Utility lib */ -# pragma comment (lib, "opengl32.lib") /* link Microsoft OpenGL lib */ -# pragma comment (lib, "gdi32.lib") /* link Windows GDI lib */ -# pragma comment (lib, "winmm.lib") /* link Windows MultiMedia lib */ -# pragma comment (lib, "user32.lib") /* link Windows user lib */ -# endif - -#else - -/* Non-Windows definition of FGAPI and FGAPIENTRY */ -# define FGAPI -# define FGAPIENTRY - -#endif - -/* - * The freeglut and GLUT API versions - */ -#define FREEGLUT 1 -#define GLUT_API_VERSION 4 -#define FREEGLUT_VERSION_2_0 1 -#define GLUT_XLIB_IMPLEMENTATION 13 - -/* - * Always include OpenGL and GLU headers - */ -#include -#include - -/* - * GLUT API macro definitions -- the special key codes: - */ -#define GLUT_KEY_F1 0x0001 -#define GLUT_KEY_F2 0x0002 -#define GLUT_KEY_F3 0x0003 -#define GLUT_KEY_F4 0x0004 -#define GLUT_KEY_F5 0x0005 -#define GLUT_KEY_F6 0x0006 -#define GLUT_KEY_F7 0x0007 -#define GLUT_KEY_F8 0x0008 -#define GLUT_KEY_F9 0x0009 -#define GLUT_KEY_F10 0x000A -#define GLUT_KEY_F11 0x000B -#define GLUT_KEY_F12 0x000C -#define GLUT_KEY_LEFT 0x0064 -#define GLUT_KEY_UP 0x0065 -#define GLUT_KEY_RIGHT 0x0066 -#define GLUT_KEY_DOWN 0x0067 -#define GLUT_KEY_PAGE_UP 0x0068 -#define GLUT_KEY_PAGE_DOWN 0x0069 -#define GLUT_KEY_HOME 0x006A -#define GLUT_KEY_END 0x006B -#define GLUT_KEY_INSERT 0x006C - -/* - * GLUT API macro definitions -- mouse state definitions - */ -#define GLUT_LEFT_BUTTON 0x0000 -#define GLUT_MIDDLE_BUTTON 0x0001 -#define GLUT_RIGHT_BUTTON 0x0002 -#define GLUT_DOWN 0x0000 -#define GLUT_UP 0x0001 -#define GLUT_LEFT 0x0000 -#define GLUT_ENTERED 0x0001 - -/* - * GLUT API macro definitions -- the display mode definitions - */ -#define GLUT_RGB 0x0000 -#define GLUT_RGBA 0x0000 -#define GLUT_INDEX 0x0001 -#define GLUT_SINGLE 0x0000 -#define GLUT_DOUBLE 0x0002 -#define GLUT_ACCUM 0x0004 -#define GLUT_ALPHA 0x0008 -#define GLUT_DEPTH 0x0010 -#define GLUT_STENCIL 0x0020 -#define GLUT_MULTISAMPLE 0x0080 -#define GLUT_STEREO 0x0100 -#define GLUT_LUMINANCE 0x0200 - -/* - * GLUT API macro definitions -- windows and menu related definitions - */ -#define GLUT_MENU_NOT_IN_USE 0x0000 -#define GLUT_MENU_IN_USE 0x0001 -#define GLUT_NOT_VISIBLE 0x0000 -#define GLUT_VISIBLE 0x0001 -#define GLUT_HIDDEN 0x0000 -#define GLUT_FULLY_RETAINED 0x0001 -#define GLUT_PARTIALLY_RETAINED 0x0002 -#define GLUT_FULLY_COVERED 0x0003 - -/* - * GLUT API macro definitions -- fonts definitions - * - * Steve Baker suggested to make it binary compatible with GLUT: - */ -#if defined(_MSC_VER) || defined(__CYGWIN__) || defined(__MINGW32__) || defined(__WATCOMC__) -# define GLUT_STROKE_ROMAN ((void *)0x0000) -# define GLUT_STROKE_MONO_ROMAN ((void *)0x0001) -# define GLUT_BITMAP_9_BY_15 ((void *)0x0002) -# define GLUT_BITMAP_8_BY_13 ((void *)0x0003) -# define GLUT_BITMAP_TIMES_ROMAN_10 ((void *)0x0004) -# define GLUT_BITMAP_TIMES_ROMAN_24 ((void *)0x0005) -# define GLUT_BITMAP_HELVETICA_10 ((void *)0x0006) -# define GLUT_BITMAP_HELVETICA_12 ((void *)0x0007) -# define GLUT_BITMAP_HELVETICA_18 ((void *)0x0008) -#else - /* - * I don't really know if it's a good idea... But here it goes: - */ - extern void* glutStrokeRoman; - extern void* glutStrokeMonoRoman; - extern void* glutBitmap9By15; - extern void* glutBitmap8By13; - extern void* glutBitmapTimesRoman10; - extern void* glutBitmapTimesRoman24; - extern void* glutBitmapHelvetica10; - extern void* glutBitmapHelvetica12; - extern void* glutBitmapHelvetica18; - - /* - * Those pointers will be used by following definitions: - */ -# define GLUT_STROKE_ROMAN ((void *) &glutStrokeRoman) -# define GLUT_STROKE_MONO_ROMAN ((void *) &glutStrokeMonoRoman) -# define GLUT_BITMAP_9_BY_15 ((void *) &glutBitmap9By15) -# define GLUT_BITMAP_8_BY_13 ((void *) &glutBitmap8By13) -# define GLUT_BITMAP_TIMES_ROMAN_10 ((void *) &glutBitmapTimesRoman10) -# define GLUT_BITMAP_TIMES_ROMAN_24 ((void *) &glutBitmapTimesRoman24) -# define GLUT_BITMAP_HELVETICA_10 ((void *) &glutBitmapHelvetica10) -# define GLUT_BITMAP_HELVETICA_12 ((void *) &glutBitmapHelvetica12) -# define GLUT_BITMAP_HELVETICA_18 ((void *) &glutBitmapHelvetica18) -#endif - -/* - * GLUT API macro definitions -- the glutGet parameters - */ -#define GLUT_WINDOW_X 0x0064 -#define GLUT_WINDOW_Y 0x0065 -#define GLUT_WINDOW_WIDTH 0x0066 -#define GLUT_WINDOW_HEIGHT 0x0067 -#define GLUT_WINDOW_BUFFER_SIZE 0x0068 -#define GLUT_WINDOW_STENCIL_SIZE 0x0069 -#define GLUT_WINDOW_DEPTH_SIZE 0x006A -#define GLUT_WINDOW_RED_SIZE 0x006B -#define GLUT_WINDOW_GREEN_SIZE 0x006C -#define GLUT_WINDOW_BLUE_SIZE 0x006D -#define GLUT_WINDOW_ALPHA_SIZE 0x006E -#define GLUT_WINDOW_ACCUM_RED_SIZE 0x006F -#define GLUT_WINDOW_ACCUM_GREEN_SIZE 0x0070 -#define GLUT_WINDOW_ACCUM_BLUE_SIZE 0x0071 -#define GLUT_WINDOW_ACCUM_ALPHA_SIZE 0x0072 -#define GLUT_WINDOW_DOUBLEBUFFER 0x0073 -#define GLUT_WINDOW_RGBA 0x0074 -#define GLUT_WINDOW_PARENT 0x0075 -#define GLUT_WINDOW_NUM_CHILDREN 0x0076 -#define GLUT_WINDOW_COLORMAP_SIZE 0x0077 -#define GLUT_WINDOW_NUM_SAMPLES 0x0078 -#define GLUT_WINDOW_STEREO 0x0079 -#define GLUT_WINDOW_CURSOR 0x007A - -#define GLUT_SCREEN_WIDTH 0x00C8 -#define GLUT_SCREEN_HEIGHT 0x00C9 -#define GLUT_SCREEN_WIDTH_MM 0x00CA -#define GLUT_SCREEN_HEIGHT_MM 0x00CB -#define GLUT_MENU_NUM_ITEMS 0x012C -#define GLUT_DISPLAY_MODE_POSSIBLE 0x0190 -#define GLUT_INIT_WINDOW_X 0x01F4 -#define GLUT_INIT_WINDOW_Y 0x01F5 -#define GLUT_INIT_WINDOW_WIDTH 0x01F6 -#define GLUT_INIT_WINDOW_HEIGHT 0x01F7 -#define GLUT_INIT_DISPLAY_MODE 0x01F8 -#define GLUT_ELAPSED_TIME 0x02BC -#define GLUT_WINDOW_FORMAT_ID 0x007B - -/* - * GLUT API macro definitions -- the glutDeviceGet parameters - */ -#define GLUT_HAS_KEYBOARD 0x0258 -#define GLUT_HAS_MOUSE 0x0259 -#define GLUT_HAS_SPACEBALL 0x025A -#define GLUT_HAS_DIAL_AND_BUTTON_BOX 0x025B -#define GLUT_HAS_TABLET 0x025C -#define GLUT_NUM_MOUSE_BUTTONS 0x025D -#define GLUT_NUM_SPACEBALL_BUTTONS 0x025E -#define GLUT_NUM_BUTTON_BOX_BUTTONS 0x025F -#define GLUT_NUM_DIALS 0x0260 -#define GLUT_NUM_TABLET_BUTTONS 0x0261 -#define GLUT_DEVICE_IGNORE_KEY_REPEAT 0x0262 -#define GLUT_DEVICE_KEY_REPEAT 0x0263 -#define GLUT_HAS_JOYSTICK 0x0264 -#define GLUT_OWNS_JOYSTICK 0x0265 -#define GLUT_JOYSTICK_BUTTONS 0x0266 -#define GLUT_JOYSTICK_AXES 0x0267 -#define GLUT_JOYSTICK_POLL_RATE 0x0268 - -/* - * GLUT API macro definitions -- the glutLayerGet parameters - */ -#define GLUT_OVERLAY_POSSIBLE 0x0320 -#define GLUT_LAYER_IN_USE 0x0321 -#define GLUT_HAS_OVERLAY 0x0322 -#define GLUT_TRANSPARENT_INDEX 0x0323 -#define GLUT_NORMAL_DAMAGED 0x0324 -#define GLUT_OVERLAY_DAMAGED 0x0325 - -/* - * GLUT API macro definitions -- the glutVideoResizeGet parameters - */ -#define GLUT_VIDEO_RESIZE_POSSIBLE 0x0384 -#define GLUT_VIDEO_RESIZE_IN_USE 0x0385 -#define GLUT_VIDEO_RESIZE_X_DELTA 0x0386 -#define GLUT_VIDEO_RESIZE_Y_DELTA 0x0387 -#define GLUT_VIDEO_RESIZE_WIDTH_DELTA 0x0388 -#define GLUT_VIDEO_RESIZE_HEIGHT_DELTA 0x0389 -#define GLUT_VIDEO_RESIZE_X 0x038A -#define GLUT_VIDEO_RESIZE_Y 0x038B -#define GLUT_VIDEO_RESIZE_WIDTH 0x038C -#define GLUT_VIDEO_RESIZE_HEIGHT 0x038D - -/* - * GLUT API macro definitions -- the glutUseLayer parameters - */ -#define GLUT_NORMAL 0x0000 -#define GLUT_OVERLAY 0x0001 - -/* - * GLUT API macro definitions -- the glutGetModifiers parameters - */ -#define GLUT_ACTIVE_SHIFT 0x0001 -#define GLUT_ACTIVE_CTRL 0x0002 -#define GLUT_ACTIVE_ALT 0x0004 - -/* - * GLUT API macro definitions -- the glutSetCursor parameters - */ -#define GLUT_CURSOR_RIGHT_ARROW 0x0000 -#define GLUT_CURSOR_LEFT_ARROW 0x0001 -#define GLUT_CURSOR_INFO 0x0002 -#define GLUT_CURSOR_DESTROY 0x0003 -#define GLUT_CURSOR_HELP 0x0004 -#define GLUT_CURSOR_CYCLE 0x0005 -#define GLUT_CURSOR_SPRAY 0x0006 -#define GLUT_CURSOR_WAIT 0x0007 -#define GLUT_CURSOR_TEXT 0x0008 -#define GLUT_CURSOR_CROSSHAIR 0x0009 -#define GLUT_CURSOR_UP_DOWN 0x000A -#define GLUT_CURSOR_LEFT_RIGHT 0x000B -#define GLUT_CURSOR_TOP_SIDE 0x000C -#define GLUT_CURSOR_BOTTOM_SIDE 0x000D -#define GLUT_CURSOR_LEFT_SIDE 0x000E -#define GLUT_CURSOR_RIGHT_SIDE 0x000F -#define GLUT_CURSOR_TOP_LEFT_CORNER 0x0010 -#define GLUT_CURSOR_TOP_RIGHT_CORNER 0x0011 -#define GLUT_CURSOR_BOTTOM_RIGHT_CORNER 0x0012 -#define GLUT_CURSOR_BOTTOM_LEFT_CORNER 0x0013 -#define GLUT_CURSOR_INHERIT 0x0064 -#define GLUT_CURSOR_NONE 0x0065 -#define GLUT_CURSOR_FULL_CROSSHAIR 0x0066 - -/* - * GLUT API macro definitions -- RGB color component specification definitions - */ -#define GLUT_RED 0x0000 -#define GLUT_GREEN 0x0001 -#define GLUT_BLUE 0x0002 - -/* - * GLUT API macro definitions -- additional keyboard and joystick definitions - */ -#define GLUT_KEY_REPEAT_OFF 0x0000 -#define GLUT_KEY_REPEAT_ON 0x0001 -#define GLUT_KEY_REPEAT_DEFAULT 0x0002 - -#define GLUT_JOYSTICK_BUTTON_A 0x0001 -#define GLUT_JOYSTICK_BUTTON_B 0x0002 -#define GLUT_JOYSTICK_BUTTON_C 0x0004 -#define GLUT_JOYSTICK_BUTTON_D 0x0008 - -/* - * GLUT API macro definitions -- game mode definitions - */ -#define GLUT_GAME_MODE_ACTIVE 0x0000 -#define GLUT_GAME_MODE_POSSIBLE 0x0001 -#define GLUT_GAME_MODE_WIDTH 0x0002 -#define GLUT_GAME_MODE_HEIGHT 0x0003 -#define GLUT_GAME_MODE_PIXEL_DEPTH 0x0004 -#define GLUT_GAME_MODE_REFRESH_RATE 0x0005 -#define GLUT_GAME_MODE_DISPLAY_CHANGED 0x0006 - -/* - * Initialization functions, see fglut_init.c - */ -FGAPI void FGAPIENTRY glutInit( int* pargc, char** argv ); -FGAPI void FGAPIENTRY glutInitWindowPosition( int x, int y ); -FGAPI void FGAPIENTRY glutInitWindowSize( int width, int height ); -FGAPI void FGAPIENTRY glutInitDisplayMode( unsigned int displayMode ); -FGAPI void FGAPIENTRY glutInitDisplayString( const char* displayMode ); - -/* - * Process loop function, see freeglut_main.c - */ -FGAPI void FGAPIENTRY glutMainLoop( void ); - -/* - * Window management functions, see freeglut_window.c - */ -FGAPI int FGAPIENTRY glutCreateWindow( const char* title ); -FGAPI int FGAPIENTRY glutCreateSubWindow( int window, int x, int y, int width, int height ); -FGAPI void FGAPIENTRY glutDestroyWindow( int window ); -FGAPI void FGAPIENTRY glutSetWindow( int window ); -FGAPI int FGAPIENTRY glutGetWindow( void ); -FGAPI void FGAPIENTRY glutSetWindowTitle( const char* title ); -FGAPI void FGAPIENTRY glutSetIconTitle( const char* title ); -FGAPI void FGAPIENTRY glutReshapeWindow( int width, int height ); -FGAPI void FGAPIENTRY glutPositionWindow( int x, int y ); -FGAPI void FGAPIENTRY glutShowWindow( void ); -FGAPI void FGAPIENTRY glutHideWindow( void ); -FGAPI void FGAPIENTRY glutIconifyWindow( void ); -FGAPI void FGAPIENTRY glutPushWindow( void ); -FGAPI void FGAPIENTRY glutPopWindow( void ); -FGAPI void FGAPIENTRY glutFullScreen( void ); - -/* - * Display-connected functions, see freeglut_display.c - */ -FGAPI void FGAPIENTRY glutPostWindowRedisplay( int window ); -FGAPI void FGAPIENTRY glutPostRedisplay( void ); -FGAPI void FGAPIENTRY glutSwapBuffers( void ); - -/* - * Mouse cursor functions, see freeglut_cursor.c - */ -FGAPI void FGAPIENTRY glutWarpPointer( int x, int y ); -FGAPI void FGAPIENTRY glutSetCursor( int cursor ); - -/* - * Overlay stuff, see freeglut_overlay.c - */ -FGAPI void FGAPIENTRY glutEstablishOverlay( void ); -FGAPI void FGAPIENTRY glutRemoveOverlay( void ); -FGAPI void FGAPIENTRY glutUseLayer( GLenum layer ); -FGAPI void FGAPIENTRY glutPostOverlayRedisplay( void ); -FGAPI void FGAPIENTRY glutPostWindowOverlayRedisplay( int window ); -FGAPI void FGAPIENTRY glutShowOverlay( void ); -FGAPI void FGAPIENTRY glutHideOverlay( void ); - -/* - * Menu stuff, see freeglut_menu.c - */ -FGAPI int FGAPIENTRY glutCreateMenu( void (* callback)( int menu ) ); -FGAPI void FGAPIENTRY glutDestroyMenu( int menu ); -FGAPI int FGAPIENTRY glutGetMenu( void ); -FGAPI void FGAPIENTRY glutSetMenu( int menu ); -FGAPI void FGAPIENTRY glutAddMenuEntry( const char* label, int value ); -FGAPI void FGAPIENTRY glutAddSubMenu( const char* label, int subMenu ); -FGAPI void FGAPIENTRY glutChangeToMenuEntry( int item, const char* label, int value ); -FGAPI void FGAPIENTRY glutChangeToSubMenu( int item, const char* label, int value ); -FGAPI void FGAPIENTRY glutRemoveMenuItem( int item ); -FGAPI void FGAPIENTRY glutAttachMenu( int button ); -FGAPI void FGAPIENTRY glutDetachMenu( int button ); - -/* - * Global callback functions, see freeglut_callbacks.c - */ -FGAPI void FGAPIENTRY glutTimerFunc( unsigned int time, void (* callback)( int ), int value ); -FGAPI void FGAPIENTRY glutIdleFunc( void (* callback)( void ) ); - -/* - * Window-specific callback functions, see freeglut_callbacks.c - */ -FGAPI void FGAPIENTRY glutKeyboardFunc( void (* callback)( unsigned char, int, int ) ); -FGAPI void FGAPIENTRY glutSpecialFunc( void (* callback)( int, int, int ) ); -FGAPI void FGAPIENTRY glutReshapeFunc( void (* callback)( int, int ) ); -FGAPI void FGAPIENTRY glutVisibilityFunc( void (* callback)( int ) ); -FGAPI void FGAPIENTRY glutDisplayFunc( void (* callback)( void ) ); -FGAPI void FGAPIENTRY glutMouseFunc( void (* callback)( int, int, int, int ) ); -FGAPI void FGAPIENTRY glutMotionFunc( void (* callback)( int, int ) ); -FGAPI void FGAPIENTRY glutPassiveMotionFunc( void (* callback)( int, int ) ); -FGAPI void FGAPIENTRY glutEntryFunc( void (* callback)( int ) ); - -FGAPI void FGAPIENTRY glutKeyboardUpFunc( void (* callback)( unsigned char, int, int ) ); -FGAPI void FGAPIENTRY glutSpecialUpFunc( void (* callback)( int, int, int ) ); -FGAPI void FGAPIENTRY glutJoystickFunc( void (* callback)( unsigned int, int, int, int ), int pollInterval ); -FGAPI void FGAPIENTRY glutMenuStateFunc( void (* callback)( int ) ); -FGAPI void FGAPIENTRY glutMenuStatusFunc( void (* callback)( int, int, int ) ); -FGAPI void FGAPIENTRY glutOverlayDisplayFunc( void (* callback)( void ) ); -FGAPI void FGAPIENTRY glutWindowStatusFunc( void (* callback)( int ) ); - -FGAPI void FGAPIENTRY glutSpaceballMotionFunc( void (* callback)( int, int, int ) ); -FGAPI void FGAPIENTRY glutSpaceballRotateFunc( void (* callback)( int, int, int ) ); -FGAPI void FGAPIENTRY glutSpaceballButtonFunc( void (* callback)( int, int ) ); -FGAPI void FGAPIENTRY glutButtonBoxFunc( void (* callback)( int, int ) ); -FGAPI void FGAPIENTRY glutDialsFunc( void (* callback)( int, int ) ); -FGAPI void FGAPIENTRY glutTabletMotionFunc( void (* callback)( int, int ) ); -FGAPI void FGAPIENTRY glutTabletButtonFunc( void (* callback)( int, int, int, int ) ); - -/* - * State setting and retrieval functions, see freeglut_state.c - */ -FGAPI int FGAPIENTRY glutGet( GLenum query ); -FGAPI int FGAPIENTRY glutDeviceGet( GLenum query ); -FGAPI int FGAPIENTRY glutGetModifiers( void ); -FGAPI int FGAPIENTRY glutLayerGet( GLenum query ); - -/* - * Font stuff, see freeglut_font.c - */ -FGAPI void FGAPIENTRY glutBitmapCharacter( void* font, int character ); -FGAPI int FGAPIENTRY glutBitmapWidth( void* font, int character ); -FGAPI void FGAPIENTRY glutStrokeCharacter( void* font, int character ); -FGAPI int FGAPIENTRY glutStrokeWidth( void* font, int character ); -FGAPI int FGAPIENTRY glutBitmapLength( void* font, const unsigned char* string ); -FGAPI int FGAPIENTRY glutStrokeLength( void* font, const unsigned char* string ); - -/* - * Geometry functions, see freeglut_geometry.c - */ -FGAPI void FGAPIENTRY glutWireCube( GLdouble size ); -FGAPI void FGAPIENTRY glutSolidCube( GLdouble size ); -FGAPI void FGAPIENTRY glutWireSphere( GLdouble radius, GLint slices, GLint stacks ); -FGAPI void FGAPIENTRY glutSolidSphere( GLdouble radius, GLint slices, GLint stacks ); -FGAPI void FGAPIENTRY glutWireCone( GLdouble base, GLdouble height, GLint slices, GLint stacks ); -FGAPI void FGAPIENTRY glutSolidCone( GLdouble base, GLdouble height, GLint slices, GLint stacks ); - -FGAPI void FGAPIENTRY glutWireTorus( GLdouble innerRadius, GLdouble outerRadius, GLint sides, GLint rings ); -FGAPI void FGAPIENTRY glutSolidTorus( GLdouble innerRadius, GLdouble outerRadius, GLint sides, GLint rings ); -FGAPI void FGAPIENTRY glutWireDodecahedron( void ); -FGAPI void FGAPIENTRY glutSolidDodecahedron( void ); -FGAPI void FGAPIENTRY glutWireOctahedron( void ); -FGAPI void FGAPIENTRY glutSolidOctahedron( void ); -FGAPI void FGAPIENTRY glutWireTetrahedron( void ); -FGAPI void FGAPIENTRY glutSolidTetrahedron( void ); -FGAPI void FGAPIENTRY glutWireIcosahedron( void ); -FGAPI void FGAPIENTRY glutSolidIcosahedron( void ); - -/* - * Teapot rendering functions, found in freeglut_teapot.c - */ -FGAPI void FGAPIENTRY glutWireTeapot( GLdouble size ); -FGAPI void FGAPIENTRY glutSolidTeapot( GLdouble size ); - -/* - * Game mode functions, see freeglut_gamemode.c - */ -FGAPI void FGAPIENTRY glutGameModeString( const char* string ); -FGAPI int FGAPIENTRY glutEnterGameMode( void ); -FGAPI void FGAPIENTRY glutLeaveGameMode( void ); -FGAPI int FGAPIENTRY glutGameModeGet( GLenum query ); - -/* - * Video resize functions, see freeglut_videoresize.c - */ -FGAPI int FGAPIENTRY glutVideoResizeGet( GLenum query ); -FGAPI void FGAPIENTRY glutSetupVideoResizing( void ); -FGAPI void FGAPIENTRY glutStopVideoResizing( void ); -FGAPI void FGAPIENTRY glutVideoResize( int x, int y, int width, int height ); -FGAPI void FGAPIENTRY glutVideoPan( int x, int y, int width, int height ); - -/* - * Colormap functions, see freeglut_misc.c - */ -FGAPI void FGAPIENTRY glutSetColor( int color, GLfloat red, GLfloat green, GLfloat blue ); -FGAPI GLfloat FGAPIENTRY glutGetColor( int color, int component ); -FGAPI void FGAPIENTRY glutCopyColormap( int window ); - -/* - * Misc keyboard and joystick functions, see freeglut_misc.c - */ -FGAPI void FGAPIENTRY glutIgnoreKeyRepeat( int ignore ); -FGAPI void FGAPIENTRY glutSetKeyRepeat( int repeatMode ); -FGAPI void FGAPIENTRY glutForceJoystickFunc( void ); - -/* - * Misc functions, see freeglut_misc.c - */ -FGAPI int FGAPIENTRY glutExtensionSupported( const char* extension ); -FGAPI void FGAPIENTRY glutReportErrors( void ); - -/* Comment from glut.h of classic GLUT: - - Win32 has an annoying issue where there are multiple C run-time - libraries (CRTs). If the executable is linked with a different CRT - from the GLUT DLL, the GLUT DLL will not share the same CRT static - data seen by the executable. In particular, atexit callbacks registered - in the executable will not be called if GLUT calls its (different) - exit routine). GLUT is typically built with the - "/MD" option (the CRT with multithreading DLL support), but the Visual - C++ linker default is "/ML" (the single threaded CRT). - - One workaround to this issue is requiring users to always link with - the same CRT as GLUT is compiled with. That requires users supply a - non-standard option. GLUT 3.7 has its own built-in workaround where - the executable's "exit" function pointer is covertly passed to GLUT. - GLUT then calls the executable's exit function pointer to ensure that - any "atexit" calls registered by the application are called if GLUT - needs to exit. - - Note that the __glut*WithExit routines should NEVER be called directly. - To avoid the atexit workaround, #define GLUT_DISABLE_ATEXIT_HACK. */ - -/* to get the prototype for exit() */ -#include - -#if defined(_WIN32) && !defined(GLUT_DISABLE_ATEXIT_HACK) && !defined(__WATCOMC__) -FGAPI void FGAPIENTRY __glutInitWithExit(int *argcp, char **argv, void (__cdecl *exitfunc)(int)); -FGAPI int FGAPIENTRY __glutCreateWindowWithExit(const char *title, void (__cdecl *exitfunc)(int)); -FGAPI int FGAPIENTRY __glutCreateMenuWithExit(void (* func)(int), void (__cdecl *exitfunc)(int)); -#ifndef FREEGLUT_BUILDING_LIB -#if defined(__GNUC__) -#define FGUNUSED __attribute__((unused)) -#else -#define FGUNUSED -#endif -static void FGAPIENTRY FGUNUSED glutInit_ATEXIT_HACK(int *argcp, char **argv) { __glutInitWithExit(argcp, argv, exit); } -#define glutInit glutInit_ATEXIT_HACK -static int FGAPIENTRY FGUNUSED glutCreateWindow_ATEXIT_HACK(const char *title) { return __glutCreateWindowWithExit(title, exit); } -#define glutCreateWindow glutCreateWindow_ATEXIT_HACK -static int FGAPIENTRY FGUNUSED glutCreateMenu_ATEXIT_HACK(void (* func)(int)) { return __glutCreateMenuWithExit(func, exit); } -#define glutCreateMenu glutCreateMenu_ATEXIT_HACK -#endif -#endif - -#ifdef __cplusplus - } -#endif - -/*** END OF FILE ***/ - -#endif /* __FREEGLUT_STD_H__ */ - diff --git a/examples/common/opengl-framework/freeglut/GL/glut.h b/examples/common/opengl-framework/freeglut/GL/glut.h deleted file mode 100644 index 6191f77b..00000000 --- a/examples/common/opengl-framework/freeglut/GL/glut.h +++ /dev/null @@ -1,21 +0,0 @@ -#ifndef __GLUT_H__ -#define __GLUT_H__ - -/* - * glut.h - * - * The freeglut library include file - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include "freeglut_std.h" - -/*** END OF FILE ***/ - -#endif /* __GLUT_H__ */ diff --git a/examples/common/opengl-framework/freeglut/VERSION.txt b/examples/common/opengl-framework/freeglut/VERSION.txt deleted file mode 100644 index 6533b668..00000000 --- a/examples/common/opengl-framework/freeglut/VERSION.txt +++ /dev/null @@ -1 +0,0 @@ -2.8.0 \ No newline at end of file diff --git a/examples/common/opengl-framework/freeglut/freeglut_callbacks.c b/examples/common/opengl-framework/freeglut/freeglut_callbacks.c deleted file mode 100644 index a40f7daf..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_callbacks.c +++ /dev/null @@ -1,412 +0,0 @@ -/* - * freeglut_callbacks.c - * - * The callbacks setting methods. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Fri Dec 3 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * All of the callbacks setting methods can be generalized to this: - */ -#define SET_CALLBACK(a) \ -do \ -{ \ - if( fgStructure.CurrentWindow == NULL ) \ - return; \ - SET_WCB( ( *( fgStructure.CurrentWindow ) ), a, callback ); \ -} while( 0 ) - -/* - * Sets the Display callback for the current window - */ -void FGAPIENTRY glutDisplayFunc( void (* callback)( void ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDisplayFunc" ); - if( !callback ) - fgError( "Fatal error in program. NULL display callback not " - "permitted in GLUT 3.0+ or freeglut 2.0.1+" ); - SET_CALLBACK( Display ); -} - -/* - * Sets the Reshape callback for the current window - */ -void FGAPIENTRY glutReshapeFunc( void (* callback)( int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutReshapeFunc" ); - SET_CALLBACK( Reshape ); -} - -/* - * Sets the Keyboard callback for the current window - */ -void FGAPIENTRY glutKeyboardFunc( void (* callback) - ( unsigned char, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutKeyboardFunc" ); - SET_CALLBACK( Keyboard ); -} - -/* - * Sets the Special callback for the current window - */ -void FGAPIENTRY glutSpecialFunc( void (* callback)( int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSpecialFunc" ); - SET_CALLBACK( Special ); -} - -/* - * Sets the global idle callback - */ -void FGAPIENTRY glutIdleFunc( void (* callback)( void ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutIdleFunc" ); - fgState.IdleCallback = callback; -} - -/* - * Sets the Timer callback for the current window - */ -void FGAPIENTRY glutTimerFunc( unsigned int timeOut, void (* callback)( int ), - int timerID ) -{ - SFG_Timer *timer, *node; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutTimerFunc" ); - - if( (timer = fgState.FreeTimers.Last) ) - { - fgListRemove( &fgState.FreeTimers, &timer->Node ); - } - else - { - if( ! (timer = malloc(sizeof(SFG_Timer))) ) - fgError( "Fatal error: " - "Memory allocation failure in glutTimerFunc()" ); - } - - timer->Callback = callback; - timer->ID = timerID; - timer->TriggerTime = fgElapsedTime() + timeOut; - - for( node = fgState.Timers.First; node; node = node->Node.Next ) - { - if( node->TriggerTime > timer->TriggerTime ) - break; - } - - fgListInsert( &fgState.Timers, &node->Node, &timer->Node ); -} - -/* - * Sets the Visibility callback for the current window. - */ -static void fghVisibility( int status ) -{ - int glut_status = GLUT_VISIBLE; - - FREEGLUT_INTERNAL_ERROR_EXIT_IF_NOT_INITIALISED ( "Visibility Callback" ); - freeglut_return_if_fail( fgStructure.CurrentWindow ); - - if( ( GLUT_HIDDEN == status ) || ( GLUT_FULLY_COVERED == status ) ) - glut_status = GLUT_NOT_VISIBLE; - INVOKE_WCB( *( fgStructure.CurrentWindow ), Visibility, ( glut_status ) ); -} - -void FGAPIENTRY glutVisibilityFunc( void (* callback)( int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutVisibilityFunc" ); - SET_CALLBACK( Visibility ); - - if( callback ) - glutWindowStatusFunc( fghVisibility ); - else - glutWindowStatusFunc( NULL ); -} - -/* - * Sets the keyboard key release callback for the current window - */ -void FGAPIENTRY glutKeyboardUpFunc( void (* callback) - ( unsigned char, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutKeyboardUpFunc" ); - SET_CALLBACK( KeyboardUp ); -} - -/* - * Sets the special key release callback for the current window - */ -void FGAPIENTRY glutSpecialUpFunc( void (* callback)( int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSpecialUpFunc" ); - SET_CALLBACK( SpecialUp ); -} - -/* - * Sets the joystick callback and polling rate for the current window - */ -void FGAPIENTRY glutJoystickFunc( void (* callback) - ( unsigned int, int, int, int ), - int pollInterval ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickFunc" ); - fgInitialiseJoysticks (); - - if ( ( ( fgStructure.CurrentWindow->State.JoystickPollRate < 0 ) || - !FETCH_WCB(*fgStructure.CurrentWindow,Joystick) ) && /* Joystick callback was disabled */ - ( callback && ( pollInterval >= 0 ) ) ) /* but is now enabled */ - ++fgState.NumActiveJoysticks; - else if ( ( ( fgStructure.CurrentWindow->State.JoystickPollRate >= 0 ) && - FETCH_WCB(*fgStructure.CurrentWindow,Joystick) ) && /* Joystick callback was enabled */ - ( !callback || ( pollInterval < 0 ) ) ) /* but is now disabled */ - --fgState.NumActiveJoysticks; - - SET_CALLBACK( Joystick ); - fgStructure.CurrentWindow->State.JoystickPollRate = pollInterval; - - fgStructure.CurrentWindow->State.JoystickLastPoll = - fgElapsedTime() - fgStructure.CurrentWindow->State.JoystickPollRate; - - if( fgStructure.CurrentWindow->State.JoystickLastPoll < 0 ) - fgStructure.CurrentWindow->State.JoystickLastPoll = 0; -} - -/* - * Sets the mouse callback for the current window - */ -void FGAPIENTRY glutMouseFunc( void (* callback)( int, int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMouseFunc" ); - SET_CALLBACK( Mouse ); -} - -/* - * Sets the mouse wheel callback for the current window - */ -void FGAPIENTRY glutMouseWheelFunc( void (* callback)( int, int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMouseWheelFunc" ); - SET_CALLBACK( MouseWheel ); -} - -/* - * Sets the mouse motion callback for the current window (one or more buttons - * are pressed) - */ -void FGAPIENTRY glutMotionFunc( void (* callback)( int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMotionFunc" ); - SET_CALLBACK( Motion ); -} - -/* - * Sets the passive mouse motion callback for the current window (no mouse - * buttons are pressed) - */ -void FGAPIENTRY glutPassiveMotionFunc( void (* callback)( int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPassiveMotionFunc" ); - SET_CALLBACK( Passive ); -} - -/* - * Window mouse entry/leave callback - */ -void FGAPIENTRY glutEntryFunc( void (* callback)( int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutEntryFunc" ); - SET_CALLBACK( Entry ); -} - -/* - * Window destruction callbacks - */ -void FGAPIENTRY glutCloseFunc( void (* callback)( void ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutCloseFunc" ); - SET_CALLBACK( Destroy ); -} - -void FGAPIENTRY glutWMCloseFunc( void (* callback)( void ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWMCloseFunc" ); - glutCloseFunc( callback ); -} - -/* A. Donev: Destruction callback for menus */ -void FGAPIENTRY glutMenuDestroyFunc( void (* callback)( void ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMenuDestroyFunc" ); - if( fgStructure.CurrentMenu ) - fgStructure.CurrentMenu->Destroy = callback; -} - -/* - * Deprecated version of glutMenuStatusFunc callback setting method - */ -void FGAPIENTRY glutMenuStateFunc( void (* callback)( int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMenuStateFunc" ); - fgState.MenuStateCallback = callback; -} - -/* - * Sets the global menu status callback for the current window - */ -void FGAPIENTRY glutMenuStatusFunc( void (* callback)( int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMenuStatusFunc" ); - fgState.MenuStatusCallback = callback; -} - -/* - * Sets the overlay display callback for the current window - */ -void FGAPIENTRY glutOverlayDisplayFunc( void (* callback)( void ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutOverlayDisplayFunc" ); - SET_CALLBACK( OverlayDisplay ); -} - -/* - * Sets the window status callback for the current window - */ -void FGAPIENTRY glutWindowStatusFunc( void (* callback)( int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWindowStatusFunc" ); - SET_CALLBACK( WindowStatus ); -} - -/* - * Sets the spaceball motion callback for the current window - */ -void FGAPIENTRY glutSpaceballMotionFunc( void (* callback)( int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSpaceballMotionFunc" ); - fgInitialiseSpaceball(); - - SET_CALLBACK( SpaceMotion ); -} - -/* - * Sets the spaceball rotate callback for the current window - */ -void FGAPIENTRY glutSpaceballRotateFunc( void (* callback)( int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSpaceballRotateFunc" ); - fgInitialiseSpaceball(); - - SET_CALLBACK( SpaceRotation ); -} - -/* - * Sets the spaceball button callback for the current window - */ -void FGAPIENTRY glutSpaceballButtonFunc( void (* callback)( int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSpaceballButtonFunc" ); - fgInitialiseSpaceball(); - - SET_CALLBACK( SpaceButton ); -} - -/* - * Sets the button box callback for the current window - */ -void FGAPIENTRY glutButtonBoxFunc( void (* callback)( int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutButtonBoxFunc" ); - SET_CALLBACK( ButtonBox ); -} - -/* - * Sets the dials box callback for the current window - */ -void FGAPIENTRY glutDialsFunc( void (* callback)( int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDialsFunc" ); - SET_CALLBACK( Dials ); -} - -/* - * Sets the tablet motion callback for the current window - */ -void FGAPIENTRY glutTabletMotionFunc( void (* callback)( int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutTabletMotionFunc" ); - SET_CALLBACK( TabletMotion ); -} - -/* - * Sets the tablet buttons callback for the current window - */ -void FGAPIENTRY glutTabletButtonFunc( void (* callback)( int, int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutTabletButtonFunc" ); - SET_CALLBACK( TabletButton ); -} - -/* - * Sets the multi-pointer entry callback for the current window - */ -void FGAPIENTRY glutMultiEntryFunc( void (* callback)(int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMultiEntryFunc" ); - SET_CALLBACK( MultiEntry ); -} - -/* - * Sets the multi-pointer button callback for the current window - */ -void FGAPIENTRY glutMultiButtonFunc( void (* callback)(int, int, int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMultiButtonFunc" ); - SET_CALLBACK( MultiButton ); -} - -/* - * Sets the multi-pointer motion callback for the current window - */ -void FGAPIENTRY glutMultiMotionFunc( void (* callback)(int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMultiMotionFunc" ); - SET_CALLBACK( MultiMotion ); -} - -/* - * Sets the multi-pointer passive motion callback for the current window - */ -void FGAPIENTRY glutMultiPassiveFunc( void (* callback)(int, int, int ) ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMultiPassiveFunc" ); - SET_CALLBACK( MultiPassive ); -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_cursor.c b/examples/common/opengl-framework/freeglut/freeglut_cursor.c deleted file mode 100644 index 33708319..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_cursor.c +++ /dev/null @@ -1,282 +0,0 @@ -/* - * freeglut_cursor.c - * - * The mouse cursor related stuff. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 16 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* - * TODO BEFORE THE STABLE RELEASE: - * glutSetCursor() -- Win32 mappings are incomplete. - * - * It would be good to use custom mouse cursor shapes, and introduce - * an option to display them using glBitmap() and/or texture mapping, - * apart from the windowing system version. - */ - -/* -- PRIVATE FUNCTIONS --------------------------------------------------- */ - -#if TARGET_HOST_POSIX_X11 || TARGET_HOST_MAC_OSX || TARGET_HOST_SOLARIS - #include - -/* - * A factory method for an empty cursor - */ -static Cursor getEmptyCursor( void ) -{ - static Cursor cursorNone = None; - if( cursorNone == None ) { - char cursorNoneBits[ 32 ]; - XColor dontCare; - Pixmap cursorNonePixmap; - memset( cursorNoneBits, 0, sizeof( cursorNoneBits ) ); - memset( &dontCare, 0, sizeof( dontCare ) ); - cursorNonePixmap = XCreateBitmapFromData ( fgDisplay.Display, - fgDisplay.RootWindow, - cursorNoneBits, 16, 16 ); - if( cursorNonePixmap != None ) { - cursorNone = XCreatePixmapCursor( fgDisplay.Display, - cursorNonePixmap, cursorNonePixmap, - &dontCare, &dontCare, 0, 0 ); - XFreePixmap( fgDisplay.Display, cursorNonePixmap ); - } - } - return cursorNone; -} - -typedef struct tag_cursorCacheEntry cursorCacheEntry; -struct tag_cursorCacheEntry { - unsigned int cursorShape; /* an XC_foo value */ - Cursor cachedCursor; /* None if the corresponding cursor has - not been created yet */ -}; - -/* - * Note: The arrangement of the table below depends on the fact that - * the "normal" GLUT_CURSOR_* values start a 0 and are consecutive. - */ -static cursorCacheEntry cursorCache[] = { - { XC_arrow, None }, /* GLUT_CURSOR_RIGHT_ARROW */ - { XC_top_left_arrow, None }, /* GLUT_CURSOR_LEFT_ARROW */ - { XC_hand1, None }, /* GLUT_CURSOR_INFO */ - { XC_pirate, None }, /* GLUT_CURSOR_DESTROY */ - { XC_question_arrow, None }, /* GLUT_CURSOR_HELP */ - { XC_exchange, None }, /* GLUT_CURSOR_CYCLE */ - { XC_spraycan, None }, /* GLUT_CURSOR_SPRAY */ - { XC_watch, None }, /* GLUT_CURSOR_WAIT */ - { XC_xterm, None }, /* GLUT_CURSOR_TEXT */ - { XC_crosshair, None }, /* GLUT_CURSOR_CROSSHAIR */ - { XC_sb_v_double_arrow, None }, /* GLUT_CURSOR_UP_DOWN */ - { XC_sb_h_double_arrow, None }, /* GLUT_CURSOR_LEFT_RIGHT */ - { XC_top_side, None }, /* GLUT_CURSOR_TOP_SIDE */ - { XC_bottom_side, None }, /* GLUT_CURSOR_BOTTOM_SIDE */ - { XC_left_side, None }, /* GLUT_CURSOR_LEFT_SIDE */ - { XC_right_side, None }, /* GLUT_CURSOR_RIGHT_SIDE */ - { XC_top_left_corner, None }, /* GLUT_CURSOR_TOP_LEFT_CORNER */ - { XC_top_right_corner, None }, /* GLUT_CURSOR_TOP_RIGHT_CORNER */ - { XC_bottom_right_corner, None }, /* GLUT_CURSOR_BOTTOM_RIGHT_CORNER */ - { XC_bottom_left_corner, None } /* GLUT_CURSOR_BOTTOM_LEFT_CORNER */ -}; - -static void fghSetCursor ( SFG_Window *window, int cursorID ) -{ - Cursor cursor; - /* - * XXX FULL_CROSSHAIR demotes to plain CROSSHAIR. Old GLUT allows - * for this, but if there is a system that easily supports a full- - * window (or full-screen) crosshair, we might consider it. - */ - int cursorIDToUse = - ( cursorID == GLUT_CURSOR_FULL_CROSSHAIR ) ? GLUT_CURSOR_CROSSHAIR : cursorID; - - if( ( cursorIDToUse >= 0 ) && - ( cursorIDToUse < sizeof( cursorCache ) / sizeof( cursorCache[0] ) ) ) { - cursorCacheEntry *entry = &cursorCache[ cursorIDToUse ]; - if( entry->cachedCursor == None ) { - entry->cachedCursor = - XCreateFontCursor( fgDisplay.Display, entry->cursorShape ); - } - cursor = entry->cachedCursor; - } else { - switch( cursorIDToUse ) - { - case GLUT_CURSOR_NONE: - cursor = getEmptyCursor( ); - break; - - case GLUT_CURSOR_INHERIT: - cursor = None; - break; - - default: - fgError( "Unknown cursor type: %d", cursorIDToUse ); - return; - } - } - - if ( cursorIDToUse == GLUT_CURSOR_INHERIT ) { - XUndefineCursor( fgDisplay.Display, window->Window.Handle ); - } else if ( cursor != None ) { - XDefineCursor( fgDisplay.Display, window->Window.Handle, cursor ); - } else if ( cursorIDToUse != GLUT_CURSOR_NONE ) { - fgError( "Failed to create cursor" ); - } -} - - -static void fghWarpPointer ( int x, int y ) -{ - XWarpPointer( - fgDisplay.Display, - None, - fgStructure.CurrentWindow->Window.Handle, - 0, 0, 0, 0, - x, y - ); - /* Make the warp visible immediately. */ - XFlush( fgDisplay.Display ); -} -#endif - - -#if TARGET_HOST_MS_WINDOWS -static void fghSetCursor ( SFG_Window *window, int cursorID ) -{ - /* - * Joe Krahn is re-writing the following code. - */ - /* Set the cursor AND change it for this window class. */ -#if !defined(__MINGW64__) && _MSC_VER <= 1200 -# define MAP_CURSOR(a,b) \ - case a: \ - SetCursor( LoadCursor( NULL, b ) ); \ - SetClassLong( window->Window.Handle, \ - GCL_HCURSOR, \ - ( LONG )LoadCursor( NULL, b ) ); \ - break; - /* Nuke the cursor AND change it for this window class. */ -# define ZAP_CURSOR(a,b) \ - case a: \ - SetCursor( NULL ); \ - SetClassLong( window->Window.Handle, \ - GCL_HCURSOR, ( LONG )NULL ); \ - break; -#else -# define MAP_CURSOR(a,b) \ - case a: \ - SetCursor( LoadCursor( NULL, b ) ); \ - SetClassLongPtr( window->Window.Handle, \ - GCLP_HCURSOR, \ - ( LONG )( LONG_PTR )LoadCursor( NULL, b ) ); \ - break; - /* Nuke the cursor AND change it for this window class. */ -# define ZAP_CURSOR(a,b) \ - case a: \ - SetCursor( NULL ); \ - SetClassLongPtr( window->Window.Handle, \ - GCLP_HCURSOR, ( LONG )( LONG_PTR )NULL ); \ - break; -#endif - - switch( cursorID ) - { - MAP_CURSOR( GLUT_CURSOR_RIGHT_ARROW, IDC_ARROW ); - MAP_CURSOR( GLUT_CURSOR_LEFT_ARROW, IDC_ARROW ); - MAP_CURSOR( GLUT_CURSOR_INFO, IDC_HELP ); - MAP_CURSOR( GLUT_CURSOR_DESTROY, IDC_CROSS ); - MAP_CURSOR( GLUT_CURSOR_HELP, IDC_HELP ); - MAP_CURSOR( GLUT_CURSOR_CYCLE, IDC_SIZEALL ); - MAP_CURSOR( GLUT_CURSOR_SPRAY, IDC_CROSS ); - MAP_CURSOR( GLUT_CURSOR_WAIT, IDC_WAIT ); - MAP_CURSOR( GLUT_CURSOR_TEXT, IDC_IBEAM ); - MAP_CURSOR( GLUT_CURSOR_CROSSHAIR, IDC_CROSS ); - MAP_CURSOR( GLUT_CURSOR_UP_DOWN, IDC_SIZENS ); - MAP_CURSOR( GLUT_CURSOR_LEFT_RIGHT, IDC_SIZEWE ); - MAP_CURSOR( GLUT_CURSOR_TOP_SIDE, IDC_ARROW ); /* XXX ToDo */ - MAP_CURSOR( GLUT_CURSOR_BOTTOM_SIDE, IDC_ARROW ); /* XXX ToDo */ - MAP_CURSOR( GLUT_CURSOR_LEFT_SIDE, IDC_ARROW ); /* XXX ToDo */ - MAP_CURSOR( GLUT_CURSOR_RIGHT_SIDE, IDC_ARROW ); /* XXX ToDo */ - MAP_CURSOR( GLUT_CURSOR_TOP_LEFT_CORNER, IDC_SIZENWSE ); - MAP_CURSOR( GLUT_CURSOR_TOP_RIGHT_CORNER, IDC_SIZENESW ); - MAP_CURSOR( GLUT_CURSOR_BOTTOM_RIGHT_CORNER, IDC_SIZENWSE ); - MAP_CURSOR( GLUT_CURSOR_BOTTOM_LEFT_CORNER, IDC_SIZENESW ); - MAP_CURSOR( GLUT_CURSOR_INHERIT, IDC_ARROW ); /* XXX ToDo */ - ZAP_CURSOR( GLUT_CURSOR_NONE, NULL ); - MAP_CURSOR( GLUT_CURSOR_FULL_CROSSHAIR, IDC_CROSS ); /* XXX ToDo */ - - default: - fgError( "Unknown cursor type: %d", cursorID ); - break; - } -} - - -static void fghWarpPointer ( int x, int y ) -{ - POINT coords; - coords.x = x; - coords.y = y; - - /* ClientToScreen() translates {coords} for us. */ - ClientToScreen( fgStructure.CurrentWindow->Window.Handle, &coords ); - SetCursorPos( coords.x, coords.y ); -} -#endif - - -/* -- INTERNAL FUNCTIONS ---------------------------------------------------- */ -void fgSetCursor ( SFG_Window *window, int cursorID ) -{ - fghSetCursor ( window, cursorID ); -} - - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * Set the cursor image to be used for the current window - */ -void FGAPIENTRY glutSetCursor( int cursorID ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetCursor" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutSetCursor" ); - - fghSetCursor ( fgStructure.CurrentWindow, cursorID ); - fgStructure.CurrentWindow->State.Cursor = cursorID; -} - -/* - * Moves the mouse pointer to given window coordinates - */ -void FGAPIENTRY glutWarpPointer( int x, int y ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWarpPointer" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutWarpPointer" ); - - fghWarpPointer ( x, y ); -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_display.c b/examples/common/opengl-framework/freeglut/freeglut_display.c deleted file mode 100644 index 2ba0ec54..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_display.c +++ /dev/null @@ -1,103 +0,0 @@ -/* - * freeglut_display.c - * - * Display message posting, context buffer swapping. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Fri Dec 3 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * Marks the current window to have the redisplay performed when possible... - */ -void FGAPIENTRY glutPostRedisplay( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPostRedisplay" ); - if ( ! fgStructure.CurrentWindow ) - { - fgError ( " ERROR: Function <%s> called" - " with no current window defined.", "glutPostRedisplay" ) ; - } - - fgStructure.CurrentWindow->State.Redisplay = GL_TRUE; -} - -/* - * Swaps the buffers for the current window (if any) - */ -void FGAPIENTRY glutSwapBuffers( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSwapBuffers" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutSwapBuffers" ); - - /* - * "glXSwapBuffers" already performs an implicit call to "glFlush". What - * about "SwapBuffers"? - */ - glFlush( ); - if( ! fgStructure.CurrentWindow->Window.DoubleBuffered ) - return; - -#if TARGET_HOST_POSIX_X11 - glXSwapBuffers( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle ); -#elif TARGET_HOST_MS_WINDOWS - SwapBuffers( fgStructure.CurrentWindow->Window.Device ); -#endif - - /* GLUT_FPS env var support */ - if( fgState.FPSInterval ) - { - GLint t = glutGet( GLUT_ELAPSED_TIME ); - fgState.SwapCount++; - if( fgState.SwapTime == 0 ) - fgState.SwapTime = t; - else if( t - fgState.SwapTime > fgState.FPSInterval ) - { - float time = 0.001f * ( t - fgState.SwapTime ); - float fps = ( float )fgState.SwapCount / time; - fprintf( stderr, - "freeglut: %d frames in %.2f seconds = %.2f FPS\n", - fgState.SwapCount, time, fps ); - fgState.SwapTime = t; - fgState.SwapCount = 0; - } - } -} - -/* - * Mark appropriate window to be displayed - */ -void FGAPIENTRY glutPostWindowRedisplay( int windowID ) -{ - SFG_Window* window; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPostWindowRedisplay" ); - window = fgWindowByID( windowID ); - freeglut_return_if_fail( window ); - window->State.Redisplay = GL_TRUE; -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_ext.c b/examples/common/opengl-framework/freeglut/freeglut_ext.c deleted file mode 100644 index d8aca45e..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_ext.c +++ /dev/null @@ -1,233 +0,0 @@ -/* - * freeglut_ext.c - * - * Functions related to OpenGL extensions. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 9 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#define GLX_GLXEXT_PROTOTYPES -#include -#include "freeglut_internal.h" - -static GLUTproc fghGetGLUTProcAddress( const char* procName ) -{ - /* optimization: quick initial check */ - if( strncmp( procName, "glut", 4 ) != 0 ) - return NULL; - -#define CHECK_NAME(x) if( strcmp( procName, #x ) == 0) return (GLUTproc)x; - CHECK_NAME(glutInit); - CHECK_NAME(glutInitDisplayMode); - CHECK_NAME(glutInitDisplayString); - CHECK_NAME(glutInitWindowPosition); - CHECK_NAME(glutInitWindowSize); - CHECK_NAME(glutMainLoop); - CHECK_NAME(glutExit); - CHECK_NAME(glutCreateWindow); - CHECK_NAME(glutCreateSubWindow); - CHECK_NAME(glutDestroyWindow); - CHECK_NAME(glutPostRedisplay); - CHECK_NAME(glutPostWindowRedisplay); - CHECK_NAME(glutSwapBuffers); - CHECK_NAME(glutGetWindow); - CHECK_NAME(glutSetWindow); - CHECK_NAME(glutSetWindowTitle); - CHECK_NAME(glutSetIconTitle); - CHECK_NAME(glutPositionWindow); - CHECK_NAME(glutReshapeWindow); - CHECK_NAME(glutPopWindow); - CHECK_NAME(glutPushWindow); - CHECK_NAME(glutIconifyWindow); - CHECK_NAME(glutShowWindow); - CHECK_NAME(glutHideWindow); - CHECK_NAME(glutFullScreen); - CHECK_NAME(glutSetCursor); - CHECK_NAME(glutWarpPointer); - CHECK_NAME(glutEstablishOverlay); - CHECK_NAME(glutRemoveOverlay); - CHECK_NAME(glutUseLayer); - CHECK_NAME(glutPostOverlayRedisplay); - CHECK_NAME(glutPostWindowOverlayRedisplay); - CHECK_NAME(glutShowOverlay); - CHECK_NAME(glutHideOverlay); - CHECK_NAME(glutCreateMenu); - CHECK_NAME(glutDestroyMenu); - CHECK_NAME(glutGetMenu); - CHECK_NAME(glutSetMenu); - CHECK_NAME(glutAddMenuEntry); - CHECK_NAME(glutAddSubMenu); - CHECK_NAME(glutChangeToMenuEntry); - CHECK_NAME(glutChangeToSubMenu); - CHECK_NAME(glutRemoveMenuItem); - CHECK_NAME(glutAttachMenu); - CHECK_NAME(glutDetachMenu); - CHECK_NAME(glutDisplayFunc); - CHECK_NAME(glutReshapeFunc); - CHECK_NAME(glutKeyboardFunc); - CHECK_NAME(glutMouseFunc); - CHECK_NAME(glutMultiEntryFunc); - CHECK_NAME(glutMultiMotionFunc); - CHECK_NAME(glutMultiButtonFunc); - CHECK_NAME(glutMultiPassiveFunc); - CHECK_NAME(glutMotionFunc); - CHECK_NAME(glutPassiveMotionFunc); - CHECK_NAME(glutEntryFunc); - CHECK_NAME(glutVisibilityFunc); - CHECK_NAME(glutIdleFunc); - CHECK_NAME(glutTimerFunc); - CHECK_NAME(glutMenuStateFunc); - CHECK_NAME(glutSpecialFunc); - CHECK_NAME(glutSpaceballMotionFunc); - CHECK_NAME(glutSpaceballRotateFunc); - CHECK_NAME(glutSpaceballButtonFunc); - CHECK_NAME(glutButtonBoxFunc); - CHECK_NAME(glutDialsFunc); - CHECK_NAME(glutTabletMotionFunc); - CHECK_NAME(glutTabletButtonFunc); - CHECK_NAME(glutMenuStatusFunc); - CHECK_NAME(glutOverlayDisplayFunc); - CHECK_NAME(glutWindowStatusFunc); - CHECK_NAME(glutKeyboardUpFunc); - CHECK_NAME(glutSpecialUpFunc); -#if !defined(_WIN32_WCE) - CHECK_NAME(glutJoystickFunc); -#endif /* !defined(_WIN32_WCE) */ - CHECK_NAME(glutSetColor); - CHECK_NAME(glutGetColor); - CHECK_NAME(glutCopyColormap); - CHECK_NAME(glutGet); - CHECK_NAME(glutDeviceGet); - CHECK_NAME(glutExtensionSupported); - CHECK_NAME(glutGetModifiers); - CHECK_NAME(glutLayerGet); - CHECK_NAME(glutBitmapCharacter); - CHECK_NAME(glutBitmapWidth); - CHECK_NAME(glutStrokeCharacter); - CHECK_NAME(glutStrokeWidth); - CHECK_NAME(glutBitmapLength); - CHECK_NAME(glutStrokeLength); - CHECK_NAME(glutWireSphere); - CHECK_NAME(glutSolidSphere); - CHECK_NAME(glutWireCone); - CHECK_NAME(glutSolidCone); - CHECK_NAME(glutWireCube); - CHECK_NAME(glutSolidCube); - CHECK_NAME(glutWireTorus); - CHECK_NAME(glutSolidTorus); - CHECK_NAME(glutWireDodecahedron); - CHECK_NAME(glutSolidDodecahedron); - CHECK_NAME(glutWireTeapot); - CHECK_NAME(glutSolidTeapot); - CHECK_NAME(glutWireOctahedron); - CHECK_NAME(glutSolidOctahedron); - CHECK_NAME(glutWireTetrahedron); - CHECK_NAME(glutSolidTetrahedron); - CHECK_NAME(glutWireIcosahedron); - CHECK_NAME(glutSolidIcosahedron); - CHECK_NAME(glutVideoResizeGet); - CHECK_NAME(glutSetupVideoResizing); - CHECK_NAME(glutStopVideoResizing); - CHECK_NAME(glutVideoResize); - CHECK_NAME(glutVideoPan); - CHECK_NAME(glutReportErrors); - CHECK_NAME(glutIgnoreKeyRepeat); - CHECK_NAME(glutSetKeyRepeat); -#if !defined(_WIN32_WCE) - CHECK_NAME(glutForceJoystickFunc); - CHECK_NAME(glutGameModeString); - CHECK_NAME(glutEnterGameMode); - CHECK_NAME(glutLeaveGameMode); - CHECK_NAME(glutGameModeGet); -#endif /* !defined(_WIN32_WCE) */ - /* freeglut extensions */ - CHECK_NAME(glutMainLoopEvent); - CHECK_NAME(glutLeaveMainLoop); - CHECK_NAME(glutCloseFunc); - CHECK_NAME(glutWMCloseFunc); - CHECK_NAME(glutMenuDestroyFunc); - CHECK_NAME(glutFullScreenToggle); - CHECK_NAME(glutLeaveFullScreen); - CHECK_NAME(glutSetOption); - CHECK_NAME(glutGetModeValues); - CHECK_NAME(glutSetWindowData); - CHECK_NAME(glutGetWindowData); - CHECK_NAME(glutSetMenuData); - CHECK_NAME(glutGetMenuData); - CHECK_NAME(glutBitmapHeight); - CHECK_NAME(glutStrokeHeight); - CHECK_NAME(glutBitmapString); - CHECK_NAME(glutStrokeString); - CHECK_NAME(glutWireRhombicDodecahedron); - CHECK_NAME(glutSolidRhombicDodecahedron); - CHECK_NAME(glutWireSierpinskiSponge); - CHECK_NAME(glutSolidSierpinskiSponge); - CHECK_NAME(glutWireCylinder); - CHECK_NAME(glutSolidCylinder); - CHECK_NAME(glutGetProcAddress); - CHECK_NAME(glutMouseWheelFunc); - CHECK_NAME(glutJoystickGetNumAxes); - CHECK_NAME(glutJoystickGetNumButtons); - CHECK_NAME(glutJoystickNotWorking); - CHECK_NAME(glutJoystickGetDeadBand); - CHECK_NAME(glutJoystickSetDeadBand); - CHECK_NAME(glutJoystickGetSaturation); - CHECK_NAME(glutJoystickSetSaturation); - CHECK_NAME(glutJoystickSetMinRange); - CHECK_NAME(glutJoystickSetMaxRange); - CHECK_NAME(glutJoystickSetCenter); - CHECK_NAME(glutJoystickGetMinRange); - CHECK_NAME(glutJoystickGetMaxRange); - CHECK_NAME(glutJoystickGetCenter); - CHECK_NAME(glutInitContextVersion); - CHECK_NAME(glutInitContextFlags); - CHECK_NAME(glutInitContextProfile); - CHECK_NAME(glutInitErrorFunc); - CHECK_NAME(glutInitWarningFunc); -#undef CHECK_NAME - - return NULL; -} - - -SFG_Proc fghGetProcAddress( const char *procName ) -{ -#if TARGET_HOST_MS_WINDOWS - return (SFG_Proc)wglGetProcAddress( ( LPCSTR )procName ); -#elif TARGET_HOST_POSIX_X11 && defined( GLX_ARB_get_proc_address ) - return (SFG_Proc)glXGetProcAddressARB( ( const GLubyte * )procName ); -#else - return NULL; -#endif -} - - -GLUTproc FGAPIENTRY -glutGetProcAddress( const char *procName ) -{ - GLUTproc p; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetProcAddress" ); - - /* Try GLUT functions first, then core GL functions */ - p = fghGetGLUTProcAddress( procName ); - return ( p != NULL ) ? p : fghGetProcAddress( procName ); -} diff --git a/examples/common/opengl-framework/freeglut/freeglut_font.c b/examples/common/opengl-framework/freeglut/freeglut_font.c deleted file mode 100644 index e956d5b0..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_font.c +++ /dev/null @@ -1,384 +0,0 @@ -/* - * freeglut_font.c - * - * Bitmap and stroke fonts displaying. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 16 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* - * TODO BEFORE THE STABLE RELEASE: - * - * Test things out ... - */ - -/* -- IMPORT DECLARATIONS -------------------------------------------------- */ - -/* - * These are the font faces defined in freeglut_font_data.c file: - */ -extern SFG_Font fgFontFixed8x13; -extern SFG_Font fgFontFixed9x15; -extern SFG_Font fgFontHelvetica10; -extern SFG_Font fgFontHelvetica12; -extern SFG_Font fgFontHelvetica18; -extern SFG_Font fgFontTimesRoman10; -extern SFG_Font fgFontTimesRoman24; -extern SFG_StrokeFont fgStrokeRoman; -extern SFG_StrokeFont fgStrokeMonoRoman; - - -/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ - -/* - * Matches a font ID with a SFG_Font structure pointer. - * This was changed to match the GLUT header style. - */ -static SFG_Font* fghFontByID( void* font ) -{ - if( font == GLUT_BITMAP_8_BY_13 ) - return &fgFontFixed8x13; - if( font == GLUT_BITMAP_9_BY_15 ) - return &fgFontFixed9x15; - if( font == GLUT_BITMAP_HELVETICA_10 ) - return &fgFontHelvetica10; - if( font == GLUT_BITMAP_HELVETICA_12 ) - return &fgFontHelvetica12; - if( font == GLUT_BITMAP_HELVETICA_18 ) - return &fgFontHelvetica18; - if( font == GLUT_BITMAP_TIMES_ROMAN_10 ) - return &fgFontTimesRoman10; - if( font == GLUT_BITMAP_TIMES_ROMAN_24 ) - return &fgFontTimesRoman24; - - fgWarning( "font 0x%08x not found", font ); - return 0; -} - -/* - * Matches a font ID with a SFG_StrokeFont structure pointer. - * This was changed to match the GLUT header style. - */ -static SFG_StrokeFont* fghStrokeByID( void* font ) -{ - if( font == GLUT_STROKE_ROMAN ) - return &fgStrokeRoman; - if( font == GLUT_STROKE_MONO_ROMAN ) - return &fgStrokeMonoRoman; - - fgWarning( "stroke font 0x%08x not found", font ); - return 0; -} - - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * Draw a bitmap character - */ -void FGAPIENTRY glutBitmapCharacter( void* fontID, int character ) -{ - const GLubyte* face; - SFG_Font* font; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutBitmapCharacter" ); - font = fghFontByID( fontID ); - freeglut_return_if_fail( ( character >= 1 )&&( character < 256 ) ); - freeglut_return_if_fail( font ); - - /* - * Find the character we want to draw (???) - */ - face = font->Characters[ character ]; - - glPushClientAttrib( GL_CLIENT_PIXEL_STORE_BIT ); - glPixelStorei( GL_UNPACK_SWAP_BYTES, GL_FALSE ); - glPixelStorei( GL_UNPACK_LSB_FIRST, GL_FALSE ); - glPixelStorei( GL_UNPACK_ROW_LENGTH, 0 ); - glPixelStorei( GL_UNPACK_SKIP_ROWS, 0 ); - glPixelStorei( GL_UNPACK_SKIP_PIXELS, 0 ); - glPixelStorei( GL_UNPACK_ALIGNMENT, 1 ); - glBitmap( - face[ 0 ], font->Height, /* The bitmap's width and height */ - font->xorig, font->yorig, /* The origin in the font glyph */ - ( float )( face[ 0 ] ), 0.0, /* The raster advance -- inc. x,y */ - ( face + 1 ) /* The packed bitmap data... */ - ); - glPopClientAttrib( ); -} - -void FGAPIENTRY glutBitmapString( void* fontID, const unsigned char *string ) -{ - unsigned char c; - float x = 0.0f ; - SFG_Font* font; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutBitmapString" ); - font = fghFontByID( fontID ); - freeglut_return_if_fail( font ); - if ( !string || ! *string ) - return; - - glPushClientAttrib( GL_CLIENT_PIXEL_STORE_BIT ); - glPixelStorei( GL_UNPACK_SWAP_BYTES, GL_FALSE ); - glPixelStorei( GL_UNPACK_LSB_FIRST, GL_FALSE ); - glPixelStorei( GL_UNPACK_ROW_LENGTH, 0 ); - glPixelStorei( GL_UNPACK_SKIP_ROWS, 0 ); - glPixelStorei( GL_UNPACK_SKIP_PIXELS, 0 ); - glPixelStorei( GL_UNPACK_ALIGNMENT, 1 ); - - /* - * Step through the string, drawing each character. - * A newline will simply translate the next character's insertion - * point back to the start of the line and down one line. - */ - while( ( c = *string++) ) - if( c == '\n' ) - { - glBitmap ( 0, 0, 0, 0, -x, (float) -font->Height, NULL ); - x = 0.0f; - } - else /* Not an EOL, draw the bitmap character */ - { - const GLubyte* face = font->Characters[ c ]; - - glBitmap( - face[ 0 ], font->Height, /* Bitmap's width and height */ - font->xorig, font->yorig, /* The origin in the font glyph */ - ( float )( face[ 0 ] ), 0.0, /* The raster advance; inc. x,y */ - ( face + 1 ) /* The packed bitmap data... */ - ); - - x += ( float )( face[ 0 ] ); - } - - glPopClientAttrib( ); -} - -/* - * Returns the width in pixels of a font's character - */ -int FGAPIENTRY glutBitmapWidth( void* fontID, int character ) -{ - SFG_Font* font; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutBitmapWidth" ); - font = fghFontByID( fontID ); - freeglut_return_val_if_fail( character > 0 && character < 256, 0 ); - freeglut_return_val_if_fail( font, 0 ); - return *( font->Characters[ character ] ); -} - -/* - * Return the width of a string drawn using a bitmap font - */ -int FGAPIENTRY glutBitmapLength( void* fontID, const unsigned char* string ) -{ - unsigned char c; - int length = 0, this_line_length = 0; - SFG_Font* font; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutBitmapLength" ); - font = fghFontByID( fontID ); - freeglut_return_val_if_fail( font, 0 ); - if ( !string || ! *string ) - return 0; - - while( ( c = *string++) ) - { - if( c != '\n' )/* Not an EOL, increment length of line */ - this_line_length += *( font->Characters[ c ]); - else /* EOL; reset the length of this line */ - { - if( length < this_line_length ) - length = this_line_length; - this_line_length = 0; - } - } - if ( length < this_line_length ) - length = this_line_length; - - return length; -} - -/* - * Returns the height of a bitmap font - */ -int FGAPIENTRY glutBitmapHeight( void* fontID ) -{ - SFG_Font* font; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutBitmapHeight" ); - font = fghFontByID( fontID ); - freeglut_return_val_if_fail( font, 0 ); - return font->Height; -} - -/* - * Draw a stroke character - */ -void FGAPIENTRY glutStrokeCharacter( void* fontID, int character ) -{ - const SFG_StrokeChar *schar; - const SFG_StrokeStrip *strip; - int i, j; - SFG_StrokeFont* font; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutStrokeCharacter" ); - font = fghStrokeByID( fontID ); - freeglut_return_if_fail( character >= 0 ); - freeglut_return_if_fail( character < font->Quantity ); - freeglut_return_if_fail( font ); - - schar = font->Characters[ character ]; - freeglut_return_if_fail( schar ); - strip = schar->Strips; - - for( i = 0; i < schar->Number; i++, strip++ ) - { - glBegin( GL_LINE_STRIP ); - for( j = 0; j < strip->Number; j++ ) - glVertex2f( strip->Vertices[ j ].X, strip->Vertices[ j ].Y ); - glEnd( ); - glBegin( GL_POINTS ); - for( j = 0; j < strip->Number; j++ ) - glVertex2f( strip->Vertices[ j ].X, strip->Vertices[ j ].Y ); - glEnd( ); - } - glTranslatef( schar->Right, 0.0, 0.0 ); -} - -void FGAPIENTRY glutStrokeString( void* fontID, const unsigned char *string ) -{ - unsigned char c; - int i, j; - float length = 0.0; - SFG_StrokeFont* font; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutStrokeString" ); - font = fghStrokeByID( fontID ); - freeglut_return_if_fail( font ); - if ( !string || ! *string ) - return; - - /* - * Step through the string, drawing each character. - * A newline will simply translate the next character's insertion - * point back to the start of the line and down one line. - */ - while( ( c = *string++) ) - if( c < font->Quantity ) - { - if( c == '\n' ) - { - glTranslatef ( -length, -( float )( font->Height ), 0.0 ); - length = 0.0; - } - else /* Not an EOL, draw the bitmap character */ - { - const SFG_StrokeChar *schar = font->Characters[ c ]; - if( schar ) - { - const SFG_StrokeStrip *strip = schar->Strips; - - for( i = 0; i < schar->Number; i++, strip++ ) - { - glBegin( GL_LINE_STRIP ); - for( j = 0; j < strip->Number; j++ ) - glVertex2f( strip->Vertices[ j ].X, - strip->Vertices[ j ].Y); - - glEnd( ); - } - - length += schar->Right; - glTranslatef( schar->Right, 0.0, 0.0 ); - } - } - } -} - -/* - * Return the width in pixels of a stroke character - */ -int FGAPIENTRY glutStrokeWidth( void* fontID, int character ) -{ - const SFG_StrokeChar *schar; - SFG_StrokeFont* font; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutStrokeWidth" ); - font = fghStrokeByID( fontID ); - freeglut_return_val_if_fail( ( character >= 0 ) && - ( character < font->Quantity ), - 0 - ); - freeglut_return_val_if_fail( font, 0 ); - schar = font->Characters[ character ]; - freeglut_return_val_if_fail( schar, 0 ); - - return ( int )( schar->Right + 0.5 ); -} - -/* - * Return the width of a string drawn using a stroke font - */ -int FGAPIENTRY glutStrokeLength( void* fontID, const unsigned char* string ) -{ - unsigned char c; - float length = 0.0; - float this_line_length = 0.0; - SFG_StrokeFont* font; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutStrokeLength" ); - font = fghStrokeByID( fontID ); - freeglut_return_val_if_fail( font, 0 ); - if ( !string || ! *string ) - return 0; - - while( ( c = *string++) ) - if( c < font->Quantity ) - { - if( c == '\n' ) /* EOL; reset the length of this line */ - { - if( length < this_line_length ) - length = this_line_length; - this_line_length = 0.0; - } - else /* Not an EOL, increment the length of this line */ - { - const SFG_StrokeChar *schar = font->Characters[ c ]; - if( schar ) - this_line_length += schar->Right; - } - } - if( length < this_line_length ) - length = this_line_length; - return( int )( length + 0.5 ); -} - -/* - * Returns the height of a stroke font - */ -GLfloat FGAPIENTRY glutStrokeHeight( void* fontID ) -{ - SFG_StrokeFont* font; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutStrokeHeight" ); - font = fghStrokeByID( fontID ); - freeglut_return_val_if_fail( font, 0.0 ); - return font->Height; -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_font_data.c b/examples/common/opengl-framework/freeglut/freeglut_font_data.c deleted file mode 100644 index 35dfc797..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_font_data.c +++ /dev/null @@ -1,2020 +0,0 @@ -/* - \file og_font_data.c - \brief Bitmapped font data for OpenGLUT fonts. -*/ - -/* - * This file has been automatically generated by the - * genfonts utility. - * - * The legal status of this file is a bit vague. The font glyphs - * themselves come from XFree86 v4.3.0 (as of this writing), and as - * part of the X server may be subject to the XFree86 copyrights. - * The original freeglut fonts were extracted by a utility written - * by Pawel W. Olszta (see below) and the generated fonts contained - * his copyright exclusively. Steve Baker asserts that Pawel - * assigned intellectual property rights to Steve Baker. Steve - * Baker also asserts that fonts cannot be copyrighted. He has - * neither stripped the copyright from the freeglut fonts nor - * formally retitled anything in his name. Since that time, the - * OpenGLUT project has branched from freeglut, and has made - * necessary modifications to Pawel's ``genfonts'' utility. - * To that extent, OpenGLUT may have some title to this file. - * What is fairly clear is that the font data is licensed under - * the XFree86 license (which is variously termed ``XFree'' and - * ``MIT'' by the freeglut project). It is believed that all - * title holders wish this file to be as useful as possible, and - * that either the ``XFree'' or ``MIT'' license works. - * - * Portions copyright (c) 2004, the OpenGLUT project contributors. - * OpenGLUT branched from freeglut in February, 2004. - * - * Copyright (c) 1999-2000 by Pawel W. Olszta - * Written by Pawel W. Olszta, - * - * 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 Sotware. - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* - * The following bitmapped fonts are defined in this file: - * - * 1. fgFontFixed8x13 - * -misc-fixed-medium-r-normal--13-120-75-75-C-80-iso8859-1 - * 2. fgFontFixed9x15 - * -misc-fixed-medium-r-normal--15-140-75-75-C-90-iso8859-1 - * 3. fgFontHelvetica10 - * -adobe-helvetica-medium-r-normal--10-100-75-75-p-56-iso8859-1 - * 4. fgFontHelvetica12 - * -adobe-helvetica-medium-r-normal--12-120-75-75-p-67-iso8859-1 - * 5. fgFontHelvetica18 - * -adobe-helvetica-medium-r-normal--18-180-75-75-p-98-iso8859-1 - * 6. fgFontTimesRoman10 - * -adobe-times-medium-r-normal--10-100-75-75-p-54-iso8859-1 - * 7. fgFontTimesRoman24 - * -adobe-times-medium-r-normal--24-240-75-75-p-124-iso8859-1 - */ - -static const GLubyte Fixed8x13_Character_000[] = { 8, 0, 0, 0,170, 0,130, 0,130, 0,130, 0,170, 0, 0}; -static const GLubyte Fixed8x13_Character_001[] = { 8, 0, 0, 0, 0, 16, 56,124,254,124, 56, 16, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_002[] = { 8, 0,170, 85,170, 85,170, 85,170, 85,170, 85,170, 85,170}; -static const GLubyte Fixed8x13_Character_003[] = { 8, 0, 0, 0, 4, 4, 4, 4,174,160,224,160,160, 0, 0}; -static const GLubyte Fixed8x13_Character_004[] = { 8, 0, 0, 0, 8, 8, 12, 8,142,128,192,128,224, 0, 0}; -static const GLubyte Fixed8x13_Character_005[] = { 8, 0, 0, 0, 10, 10, 12, 10,108,128,128,128, 96, 0, 0}; -static const GLubyte Fixed8x13_Character_006[] = { 8, 0, 0, 0, 8, 8, 12, 8,238,128,128,128,128, 0, 0}; -static const GLubyte Fixed8x13_Character_007[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 24, 36, 36, 24, 0, 0}; -static const GLubyte Fixed8x13_Character_008[] = { 8, 0, 0, 0, 0,124, 0, 16, 16,124, 16, 16, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_009[] = { 8, 0, 0, 0, 14, 8, 8, 8,168,160,160,160,192, 0, 0}; -static const GLubyte Fixed8x13_Character_010[] = { 8, 0, 0, 0, 4, 4, 4, 4, 46, 80, 80,136,136, 0, 0}; -static const GLubyte Fixed8x13_Character_011[] = { 8, 0, 0, 0, 0, 0, 0, 0,240, 16, 16, 16, 16, 16, 16}; -static const GLubyte Fixed8x13_Character_012[] = { 8, 0, 16, 16, 16, 16, 16, 16,240, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_013[] = { 8, 0, 16, 16, 16, 16, 16, 16, 31, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_014[] = { 8, 0, 0, 0, 0, 0, 0, 0, 31, 16, 16, 16, 16, 16, 16}; -static const GLubyte Fixed8x13_Character_015[] = { 8, 0, 16, 16, 16, 16, 16, 16,255, 16, 16, 16, 16, 16, 16}; -static const GLubyte Fixed8x13_Character_016[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255}; -static const GLubyte Fixed8x13_Character_017[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_018[] = { 8, 0, 0, 0, 0, 0, 0, 0,255, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_019[] = { 8, 0, 0, 0, 0,255, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_020[] = { 8, 0,255, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_021[] = { 8, 0, 16, 16, 16, 16, 16, 16, 31, 16, 16, 16, 16, 16, 16}; -static const GLubyte Fixed8x13_Character_022[] = { 8, 0, 16, 16, 16, 16, 16, 16,240, 16, 16, 16, 16, 16, 16}; -static const GLubyte Fixed8x13_Character_023[] = { 8, 0, 0, 0, 0, 0, 0, 0,255, 16, 16, 16, 16, 16, 16}; -static const GLubyte Fixed8x13_Character_024[] = { 8, 0, 16, 16, 16, 16, 16, 16,255, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_025[] = { 8, 0, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16}; -static const GLubyte Fixed8x13_Character_026[] = { 8, 0, 0, 0,254, 0, 14, 48,192, 48, 14, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_027[] = { 8, 0, 0, 0,254, 0,224, 24, 6, 24,224, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_028[] = { 8, 0, 0, 0, 68, 68, 68, 68, 68,254, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_029[] = { 8, 0, 0, 0, 32, 32,126, 16, 8,126, 4, 4, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_030[] = { 8, 0, 0, 0,220, 98, 32, 32, 32,112, 32, 34, 28, 0, 0}; -static const GLubyte Fixed8x13_Character_031[] = { 8, 0, 0, 0, 0, 0, 0, 0, 24, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_032[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_033[] = { 8, 0, 0, 0, 16, 0, 16, 16, 16, 16, 16, 16, 16, 0, 0}; -static const GLubyte Fixed8x13_Character_034[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 36, 36, 36, 0, 0}; -static const GLubyte Fixed8x13_Character_035[] = { 8, 0, 0, 0, 0, 36, 36,126, 36,126, 36, 36, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_036[] = { 8, 0, 0, 0, 16,120, 20, 20, 56, 80, 80, 60, 16, 0, 0}; -static const GLubyte Fixed8x13_Character_037[] = { 8, 0, 0, 0, 68, 42, 36, 16, 8, 8, 36, 82, 34, 0, 0}; -static const GLubyte Fixed8x13_Character_038[] = { 8, 0, 0, 0, 58, 68, 74, 48, 72, 72, 48, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_039[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 64, 48, 56, 0, 0}; -static const GLubyte Fixed8x13_Character_040[] = { 8, 0, 0, 0, 4, 8, 8, 16, 16, 16, 8, 8, 4, 0, 0}; -static const GLubyte Fixed8x13_Character_041[] = { 8, 0, 0, 0, 32, 16, 16, 8, 8, 8, 16, 16, 32, 0, 0}; -static const GLubyte Fixed8x13_Character_042[] = { 8, 0, 0, 0, 0, 0, 36, 24,126, 24, 36, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_043[] = { 8, 0, 0, 0, 0, 0, 16, 16,124, 16, 16, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_044[] = { 8, 0, 0, 64, 48, 56, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_045[] = { 8, 0, 0, 0, 0, 0, 0, 0,126, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_046[] = { 8, 0, 0, 16, 56, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_047[] = { 8, 0, 0, 0,128,128, 64, 32, 16, 8, 4, 2, 2, 0, 0}; -static const GLubyte Fixed8x13_Character_048[] = { 8, 0, 0, 0, 24, 36, 66, 66, 66, 66, 66, 36, 24, 0, 0}; -static const GLubyte Fixed8x13_Character_049[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16, 80, 48, 16, 0, 0}; -static const GLubyte Fixed8x13_Character_050[] = { 8, 0, 0, 0,126, 64, 32, 24, 4, 2, 66, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_051[] = { 8, 0, 0, 0, 60, 66, 2, 2, 28, 8, 4, 2,126, 0, 0}; -static const GLubyte Fixed8x13_Character_052[] = { 8, 0, 0, 0, 4, 4,126, 68, 68, 36, 20, 12, 4, 0, 0}; -static const GLubyte Fixed8x13_Character_053[] = { 8, 0, 0, 0, 60, 66, 2, 2, 98, 92, 64, 64,126, 0, 0}; -static const GLubyte Fixed8x13_Character_054[] = { 8, 0, 0, 0, 60, 66, 66, 98, 92, 64, 64, 32, 28, 0, 0}; -static const GLubyte Fixed8x13_Character_055[] = { 8, 0, 0, 0, 32, 32, 16, 16, 8, 8, 4, 2,126, 0, 0}; -static const GLubyte Fixed8x13_Character_056[] = { 8, 0, 0, 0, 60, 66, 66, 66, 60, 66, 66, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_057[] = { 8, 0, 0, 0, 56, 4, 2, 2, 58, 70, 66, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_058[] = { 8, 0, 0, 16, 56, 16, 0, 0, 16, 56, 16, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_059[] = { 8, 0, 0, 64, 48, 56, 0, 0, 16, 56, 16, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_060[] = { 8, 0, 0, 0, 2, 4, 8, 16, 32, 16, 8, 4, 2, 0, 0}; -static const GLubyte Fixed8x13_Character_061[] = { 8, 0, 0, 0, 0, 0,126, 0, 0,126, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_062[] = { 8, 0, 0, 0, 64, 32, 16, 8, 4, 8, 16, 32, 64, 0, 0}; -static const GLubyte Fixed8x13_Character_063[] = { 8, 0, 0, 0, 8, 0, 8, 8, 4, 2, 66, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_064[] = { 8, 0, 0, 0, 60, 64, 74, 86, 82, 78, 66, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_065[] = { 8, 0, 0, 0, 66, 66, 66,126, 66, 66, 66, 36, 24, 0, 0}; -static const GLubyte Fixed8x13_Character_066[] = { 8, 0, 0, 0,252, 66, 66, 66,124, 66, 66, 66,252, 0, 0}; -static const GLubyte Fixed8x13_Character_067[] = { 8, 0, 0, 0, 60, 66, 64, 64, 64, 64, 64, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_068[] = { 8, 0, 0, 0,252, 66, 66, 66, 66, 66, 66, 66,252, 0, 0}; -static const GLubyte Fixed8x13_Character_069[] = { 8, 0, 0, 0,126, 64, 64, 64,120, 64, 64, 64,126, 0, 0}; -static const GLubyte Fixed8x13_Character_070[] = { 8, 0, 0, 0, 64, 64, 64, 64,120, 64, 64, 64,126, 0, 0}; -static const GLubyte Fixed8x13_Character_071[] = { 8, 0, 0, 0, 58, 70, 66, 78, 64, 64, 64, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_072[] = { 8, 0, 0, 0, 66, 66, 66, 66,126, 66, 66, 66, 66, 0, 0}; -static const GLubyte Fixed8x13_Character_073[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16, 16, 16,124, 0, 0}; -static const GLubyte Fixed8x13_Character_074[] = { 8, 0, 0, 0, 56, 68, 4, 4, 4, 4, 4, 4, 31, 0, 0}; -static const GLubyte Fixed8x13_Character_075[] = { 8, 0, 0, 0, 66, 68, 72, 80, 96, 80, 72, 68, 66, 0, 0}; -static const GLubyte Fixed8x13_Character_076[] = { 8, 0, 0, 0,126, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0}; -static const GLubyte Fixed8x13_Character_077[] = { 8, 0, 0, 0,130,130,130,146,146,170,198,130,130, 0, 0}; -static const GLubyte Fixed8x13_Character_078[] = { 8, 0, 0, 0, 66, 66, 66, 70, 74, 82, 98, 66, 66, 0, 0}; -static const GLubyte Fixed8x13_Character_079[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_080[] = { 8, 0, 0, 0, 64, 64, 64, 64,124, 66, 66, 66,124, 0, 0}; -static const GLubyte Fixed8x13_Character_081[] = { 8, 0, 0, 2, 60, 74, 82, 66, 66, 66, 66, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_082[] = { 8, 0, 0, 0, 66, 68, 72, 80,124, 66, 66, 66,124, 0, 0}; -static const GLubyte Fixed8x13_Character_083[] = { 8, 0, 0, 0, 60, 66, 2, 2, 60, 64, 64, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_084[] = { 8, 0, 0, 0, 16, 16, 16, 16, 16, 16, 16, 16,254, 0, 0}; -static const GLubyte Fixed8x13_Character_085[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 0}; -static const GLubyte Fixed8x13_Character_086[] = { 8, 0, 0, 0, 16, 40, 40, 40, 68, 68, 68,130,130, 0, 0}; -static const GLubyte Fixed8x13_Character_087[] = { 8, 0, 0, 0, 68,170,146,146,146,130,130,130,130, 0, 0}; -static const GLubyte Fixed8x13_Character_088[] = { 8, 0, 0, 0,130,130, 68, 40, 16, 40, 68,130,130, 0, 0}; -static const GLubyte Fixed8x13_Character_089[] = { 8, 0, 0, 0, 16, 16, 16, 16, 16, 40, 68,130,130, 0, 0}; -static const GLubyte Fixed8x13_Character_090[] = { 8, 0, 0, 0,126, 64, 64, 32, 16, 8, 4, 2,126, 0, 0}; -static const GLubyte Fixed8x13_Character_091[] = { 8, 0, 0, 0, 60, 32, 32, 32, 32, 32, 32, 32, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_092[] = { 8, 0, 0, 0, 2, 2, 4, 8, 16, 32, 64,128,128, 0, 0}; -static const GLubyte Fixed8x13_Character_093[] = { 8, 0, 0, 0,120, 8, 8, 8, 8, 8, 8, 8,120, 0, 0}; -static const GLubyte Fixed8x13_Character_094[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 68, 40, 16, 0, 0}; -static const GLubyte Fixed8x13_Character_095[] = { 8, 0, 0,254, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_096[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4, 24, 56, 0, 0}; -static const GLubyte Fixed8x13_Character_097[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_098[] = { 8, 0, 0, 0, 92, 98, 66, 66, 98, 92, 64, 64, 64, 0, 0}; -static const GLubyte Fixed8x13_Character_099[] = { 8, 0, 0, 0, 60, 66, 64, 64, 66, 60, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_100[] = { 8, 0, 0, 0, 58, 70, 66, 66, 70, 58, 2, 2, 2, 0, 0}; -static const GLubyte Fixed8x13_Character_101[] = { 8, 0, 0, 0, 60, 66, 64,126, 66, 60, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_102[] = { 8, 0, 0, 0, 32, 32, 32, 32,124, 32, 32, 34, 28, 0, 0}; -static const GLubyte Fixed8x13_Character_103[] = { 8, 0, 60, 66, 60, 64, 56, 68, 68, 58, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_104[] = { 8, 0, 0, 0, 66, 66, 66, 66, 98, 92, 64, 64, 64, 0, 0}; -static const GLubyte Fixed8x13_Character_105[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 48, 0, 16, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_106[] = { 8, 0, 56, 68, 68, 4, 4, 4, 4, 12, 0, 4, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_107[] = { 8, 0, 0, 0, 66, 68, 72,112, 72, 68, 64, 64, 64, 0, 0}; -static const GLubyte Fixed8x13_Character_108[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16, 16, 16, 48, 0, 0}; -static const GLubyte Fixed8x13_Character_109[] = { 8, 0, 0, 0,130,146,146,146,146,236, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_110[] = { 8, 0, 0, 0, 66, 66, 66, 66, 98, 92, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_111[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_112[] = { 8, 0, 64, 64, 64, 92, 98, 66, 98, 92, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_113[] = { 8, 0, 2, 2, 2, 58, 70, 66, 70, 58, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_114[] = { 8, 0, 0, 0, 32, 32, 32, 32, 34, 92, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_115[] = { 8, 0, 0, 0, 60, 66, 12, 48, 66, 60, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_116[] = { 8, 0, 0, 0, 28, 34, 32, 32, 32,124, 32, 32, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_117[] = { 8, 0, 0, 0, 58, 68, 68, 68, 68, 68, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_118[] = { 8, 0, 0, 0, 16, 40, 40, 68, 68, 68, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_119[] = { 8, 0, 0, 0, 68,170,146,146,130,130, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_120[] = { 8, 0, 0, 0, 66, 36, 24, 24, 36, 66, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_121[] = { 8, 0, 60, 66, 2, 58, 70, 66, 66, 66, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_122[] = { 8, 0, 0, 0,126, 32, 16, 8, 4,126, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_123[] = { 8, 0, 0, 0, 14, 16, 16, 8, 48, 8, 16, 16, 14, 0, 0}; -static const GLubyte Fixed8x13_Character_124[] = { 8, 0, 0, 0, 16, 16, 16, 16, 16, 16, 16, 16, 16, 0, 0}; -static const GLubyte Fixed8x13_Character_125[] = { 8, 0, 0, 0,112, 8, 8, 16, 12, 16, 8, 8,112, 0, 0}; -static const GLubyte Fixed8x13_Character_126[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 72, 84, 36, 0, 0}; -static const GLubyte Fixed8x13_Character_127[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_128[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_129[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_130[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_131[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_132[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_133[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_134[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_135[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_136[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_137[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_138[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_139[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_140[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_141[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_142[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_143[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_144[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_145[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_146[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_147[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_148[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_149[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_150[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_151[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_152[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_153[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_154[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_155[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_156[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_157[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_158[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_159[] = { 9, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_160[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_161[] = { 8, 0, 0, 0, 16, 16, 16, 16, 16, 16, 16, 0, 16, 0, 0}; -static const GLubyte Fixed8x13_Character_162[] = { 8, 0, 0, 0, 0, 16, 56, 84, 80, 80, 84, 56, 16, 0, 0}; -static const GLubyte Fixed8x13_Character_163[] = { 8, 0, 0, 0,220, 98, 32, 32, 32,112, 32, 34, 28, 0, 0}; -static const GLubyte Fixed8x13_Character_164[] = { 8, 0, 0, 0, 0, 66, 60, 36, 36, 60, 66, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_165[] = { 8, 0, 0, 0, 16, 16,124, 16,124, 40, 68,130,130, 0, 0}; -static const GLubyte Fixed8x13_Character_166[] = { 8, 0, 0, 0, 16, 16, 16, 16, 0, 16, 16, 16, 16, 0, 0}; -static const GLubyte Fixed8x13_Character_167[] = { 8, 0, 0, 0, 24, 36, 4, 24, 36, 36, 24, 32, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_168[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,108, 0, 0}; -static const GLubyte Fixed8x13_Character_169[] = { 8, 0, 0, 0, 0, 56, 68,146,170,162,170,146, 68, 56, 0}; -static const GLubyte Fixed8x13_Character_170[] = { 8, 0, 0, 0, 0, 0,124, 0, 60, 68, 60, 4, 56, 0, 0}; -static const GLubyte Fixed8x13_Character_171[] = { 8, 0, 0, 0, 0, 18, 36, 72,144, 72, 36, 18, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_172[] = { 8, 0, 0, 0, 0, 2, 2, 2,126, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_173[] = { 8, 0, 0, 0, 0, 0, 0, 0, 60, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_174[] = { 8, 0, 0, 0, 0, 56, 68,170,178,170,170,146, 68, 56, 0}; -static const GLubyte Fixed8x13_Character_175[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 0}; -static const GLubyte Fixed8x13_Character_176[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 24, 36, 36, 24, 0, 0}; -static const GLubyte Fixed8x13_Character_177[] = { 8, 0, 0, 0, 0,124, 0, 16, 16,124, 16, 16, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_178[] = { 8, 0, 0, 0, 0, 0, 0, 0,120, 64, 48, 8, 72, 48, 0}; -static const GLubyte Fixed8x13_Character_179[] = { 8, 0, 0, 0, 0, 0, 0, 0, 48, 72, 8, 16, 72, 48, 0}; -static const GLubyte Fixed8x13_Character_180[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_181[] = { 8, 0, 0, 64, 90,102, 66, 66, 66, 66, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_182[] = { 8, 0, 0, 0, 20, 20, 20, 20, 52,116,116,116, 62, 0, 0}; -static const GLubyte Fixed8x13_Character_183[] = { 8, 0, 0, 0, 0, 0, 0, 0, 24, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_184[] = { 8, 0, 24, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_185[] = { 8, 0, 0, 0, 0, 0, 0, 0,112, 32, 32, 32, 96, 32, 0}; -static const GLubyte Fixed8x13_Character_186[] = { 8, 0, 0, 0, 0, 0, 0,120, 0, 48, 72, 72, 48, 0, 0}; -static const GLubyte Fixed8x13_Character_187[] = { 8, 0, 0, 0, 0,144, 72, 36, 18, 36, 72,144, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_188[] = { 8, 0, 0, 0, 6, 26, 18, 10,230, 66, 64, 64,192, 64, 0}; -static const GLubyte Fixed8x13_Character_189[] = { 8, 0, 0, 0, 30, 16, 12, 2,242, 76, 64, 64,192, 64, 0}; -static const GLubyte Fixed8x13_Character_190[] = { 8, 0, 0, 0, 6, 26, 18, 10,102,146, 16, 32,144, 96, 0}; -static const GLubyte Fixed8x13_Character_191[] = { 8, 0, 0, 0, 60, 66, 66, 64, 32, 16, 16, 0, 16, 0, 0}; -static const GLubyte Fixed8x13_Character_192[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 0, 8, 16, 0}; -static const GLubyte Fixed8x13_Character_193[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_194[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 0, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_195[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 0, 76, 50, 0}; -static const GLubyte Fixed8x13_Character_196[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 0, 36, 36, 0}; -static const GLubyte Fixed8x13_Character_197[] = { 8, 0, 0, 0, 66, 66,126, 66, 66, 36, 24, 24, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_198[] = { 8, 0, 0, 0,158,144,144,240,156,144,144,144,110, 0, 0}; -static const GLubyte Fixed8x13_Character_199[] = { 8, 0, 16, 8, 60, 66, 64, 64, 64, 64, 64, 66, 60, 0, 0}; -static const GLubyte Fixed8x13_Character_200[] = { 8, 0, 0, 0,126, 64, 64,120, 64, 64,126, 0, 8, 16, 0}; -static const GLubyte Fixed8x13_Character_201[] = { 8, 0, 0, 0,126, 64, 64,120, 64, 64,126, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_202[] = { 8, 0, 0, 0,126, 64, 64,120, 64, 64,126, 0, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_203[] = { 8, 0, 0, 0,126, 64, 64,120, 64, 64,126, 0, 36, 36, 0}; -static const GLubyte Fixed8x13_Character_204[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16,124, 0, 16, 32, 0}; -static const GLubyte Fixed8x13_Character_205[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16,124, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_206[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16,124, 0, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_207[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 16,124, 0, 40, 40, 0}; -static const GLubyte Fixed8x13_Character_208[] = { 8, 0, 0, 0,120, 68, 66, 66,226, 66, 66, 68,120, 0, 0}; -static const GLubyte Fixed8x13_Character_209[] = { 8, 0, 0, 0,130,134,138,146,162,194,130, 0,152,100, 0}; -static const GLubyte Fixed8x13_Character_210[] = { 8, 0, 0, 0,124,130,130,130,130,130,124, 0, 16, 32, 0}; -static const GLubyte Fixed8x13_Character_211[] = { 8, 0, 0, 0,124,130,130,130,130,130,124, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_212[] = { 8, 0, 0, 0,124,130,130,130,130,130,124, 0, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_213[] = { 8, 0, 0, 0,124,130,130,130,130,130,124, 0,152,100, 0}; -static const GLubyte Fixed8x13_Character_214[] = { 8, 0, 0, 0,124,130,130,130,130,130,124, 0, 40, 40, 0}; -static const GLubyte Fixed8x13_Character_215[] = { 8, 0, 0, 0, 0, 66, 36, 24, 24, 36, 66, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_216[] = { 8, 0, 0, 64, 60, 98, 82, 82, 82, 74, 74, 70, 60, 2, 0}; -static const GLubyte Fixed8x13_Character_217[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 0, 8, 16, 0}; -static const GLubyte Fixed8x13_Character_218[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_219[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 0, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_220[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 0, 36, 36, 0}; -static const GLubyte Fixed8x13_Character_221[] = { 8, 0, 0, 0, 16, 16, 16, 16, 40, 68, 68, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_222[] = { 8, 0, 0, 0, 64, 64, 64,124, 66, 66, 66,124, 64, 0, 0}; -static const GLubyte Fixed8x13_Character_223[] = { 8, 0, 0, 0, 92, 66, 66, 76, 80, 72, 68, 68, 56, 0, 0}; -static const GLubyte Fixed8x13_Character_224[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 8, 16, 0}; -static const GLubyte Fixed8x13_Character_225[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 8, 4, 0}; -static const GLubyte Fixed8x13_Character_226[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_227[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 76, 50, 0}; -static const GLubyte Fixed8x13_Character_228[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 0, 36, 36, 0}; -static const GLubyte Fixed8x13_Character_229[] = { 8, 0, 0, 0, 58, 70, 66, 62, 2, 60, 0, 24, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_230[] = { 8, 0, 0, 0,108,146,144,124, 18,108, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_231[] = { 8, 0, 16, 8, 60, 66, 64, 64, 66, 60, 0, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_232[] = { 8, 0, 0, 0, 60, 66, 64,126, 66, 60, 0, 0, 8, 16, 0}; -static const GLubyte Fixed8x13_Character_233[] = { 8, 0, 0, 0, 60, 66, 64,126, 66, 60, 0, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_234[] = { 8, 0, 0, 0, 60, 66, 64,126, 66, 60, 0, 0, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_235[] = { 8, 0, 0, 0, 60, 66, 64,126, 66, 60, 0, 0, 36, 36, 0}; -static const GLubyte Fixed8x13_Character_236[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 48, 0, 0, 16, 32, 0}; -static const GLubyte Fixed8x13_Character_237[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 48, 0, 0, 32, 16, 0}; -static const GLubyte Fixed8x13_Character_238[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 48, 0, 0, 72, 48, 0}; -static const GLubyte Fixed8x13_Character_239[] = { 8, 0, 0, 0,124, 16, 16, 16, 16, 48, 0, 0, 40, 40, 0}; -static const GLubyte Fixed8x13_Character_240[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 4, 40, 24, 36, 0}; -static const GLubyte Fixed8x13_Character_241[] = { 8, 0, 0, 0, 66, 66, 66, 66, 98, 92, 0, 0, 76, 50, 0}; -static const GLubyte Fixed8x13_Character_242[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 16, 32, 0}; -static const GLubyte Fixed8x13_Character_243[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_244[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_245[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 76, 50, 0}; -static const GLubyte Fixed8x13_Character_246[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 60, 0, 0, 36, 36, 0}; -static const GLubyte Fixed8x13_Character_247[] = { 8, 0, 0, 0, 0, 16, 16, 0,124, 0, 16, 16, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_248[] = { 8, 0, 0, 64, 60, 98, 82, 74, 70, 60, 2, 0, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_249[] = { 8, 0, 0, 0, 58, 68, 68, 68, 68, 68, 0, 0, 16, 32, 0}; -static const GLubyte Fixed8x13_Character_250[] = { 8, 0, 0, 0, 58, 68, 68, 68, 68, 68, 0, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_251[] = { 8, 0, 0, 0, 58, 68, 68, 68, 68, 68, 0, 0, 36, 24, 0}; -static const GLubyte Fixed8x13_Character_252[] = { 8, 0, 0, 0, 58, 68, 68, 68, 68, 68, 0, 0, 40, 40, 0}; -static const GLubyte Fixed8x13_Character_253[] = { 8, 0, 60, 66, 2, 58, 70, 66, 66, 66, 0, 0, 16, 8, 0}; -static const GLubyte Fixed8x13_Character_254[] = { 8, 0, 64, 64, 92, 98, 66, 66, 98, 92, 64, 64, 0, 0, 0}; -static const GLubyte Fixed8x13_Character_255[] = { 8, 0, 60, 66, 2, 58, 70, 66, 66, 66, 0, 0, 36, 36, 0}; - -/* The font characters mapping: */ -static const GLubyte* Fixed8x13_Character_Map[] = {Fixed8x13_Character_000,Fixed8x13_Character_001,Fixed8x13_Character_002,Fixed8x13_Character_003,Fixed8x13_Character_004,Fixed8x13_Character_005,Fixed8x13_Character_006,Fixed8x13_Character_007,Fixed8x13_Character_008,Fixed8x13_Character_009,Fixed8x13_Character_010,Fixed8x13_Character_011,Fixed8x13_Character_012,Fixed8x13_Character_013,Fixed8x13_Character_014,Fixed8x13_Character_015, - Fixed8x13_Character_016,Fixed8x13_Character_017,Fixed8x13_Character_018,Fixed8x13_Character_019,Fixed8x13_Character_020,Fixed8x13_Character_021,Fixed8x13_Character_022,Fixed8x13_Character_023,Fixed8x13_Character_024,Fixed8x13_Character_025,Fixed8x13_Character_026,Fixed8x13_Character_027,Fixed8x13_Character_028,Fixed8x13_Character_029,Fixed8x13_Character_030,Fixed8x13_Character_031, - Fixed8x13_Character_032,Fixed8x13_Character_033,Fixed8x13_Character_034,Fixed8x13_Character_035,Fixed8x13_Character_036,Fixed8x13_Character_037,Fixed8x13_Character_038,Fixed8x13_Character_039,Fixed8x13_Character_040,Fixed8x13_Character_041,Fixed8x13_Character_042,Fixed8x13_Character_043,Fixed8x13_Character_044,Fixed8x13_Character_045,Fixed8x13_Character_046,Fixed8x13_Character_047, - Fixed8x13_Character_048,Fixed8x13_Character_049,Fixed8x13_Character_050,Fixed8x13_Character_051,Fixed8x13_Character_052,Fixed8x13_Character_053,Fixed8x13_Character_054,Fixed8x13_Character_055,Fixed8x13_Character_056,Fixed8x13_Character_057,Fixed8x13_Character_058,Fixed8x13_Character_059,Fixed8x13_Character_060,Fixed8x13_Character_061,Fixed8x13_Character_062,Fixed8x13_Character_063, - Fixed8x13_Character_064,Fixed8x13_Character_065,Fixed8x13_Character_066,Fixed8x13_Character_067,Fixed8x13_Character_068,Fixed8x13_Character_069,Fixed8x13_Character_070,Fixed8x13_Character_071,Fixed8x13_Character_072,Fixed8x13_Character_073,Fixed8x13_Character_074,Fixed8x13_Character_075,Fixed8x13_Character_076,Fixed8x13_Character_077,Fixed8x13_Character_078,Fixed8x13_Character_079, - Fixed8x13_Character_080,Fixed8x13_Character_081,Fixed8x13_Character_082,Fixed8x13_Character_083,Fixed8x13_Character_084,Fixed8x13_Character_085,Fixed8x13_Character_086,Fixed8x13_Character_087,Fixed8x13_Character_088,Fixed8x13_Character_089,Fixed8x13_Character_090,Fixed8x13_Character_091,Fixed8x13_Character_092,Fixed8x13_Character_093,Fixed8x13_Character_094,Fixed8x13_Character_095, - Fixed8x13_Character_096,Fixed8x13_Character_097,Fixed8x13_Character_098,Fixed8x13_Character_099,Fixed8x13_Character_100,Fixed8x13_Character_101,Fixed8x13_Character_102,Fixed8x13_Character_103,Fixed8x13_Character_104,Fixed8x13_Character_105,Fixed8x13_Character_106,Fixed8x13_Character_107,Fixed8x13_Character_108,Fixed8x13_Character_109,Fixed8x13_Character_110,Fixed8x13_Character_111, - Fixed8x13_Character_112,Fixed8x13_Character_113,Fixed8x13_Character_114,Fixed8x13_Character_115,Fixed8x13_Character_116,Fixed8x13_Character_117,Fixed8x13_Character_118,Fixed8x13_Character_119,Fixed8x13_Character_120,Fixed8x13_Character_121,Fixed8x13_Character_122,Fixed8x13_Character_123,Fixed8x13_Character_124,Fixed8x13_Character_125,Fixed8x13_Character_126,Fixed8x13_Character_032, - Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032, - Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032,Fixed8x13_Character_032, - Fixed8x13_Character_160,Fixed8x13_Character_161,Fixed8x13_Character_162,Fixed8x13_Character_163,Fixed8x13_Character_164,Fixed8x13_Character_165,Fixed8x13_Character_166,Fixed8x13_Character_167,Fixed8x13_Character_168,Fixed8x13_Character_169,Fixed8x13_Character_170,Fixed8x13_Character_171,Fixed8x13_Character_172,Fixed8x13_Character_173,Fixed8x13_Character_174,Fixed8x13_Character_175, - Fixed8x13_Character_176,Fixed8x13_Character_177,Fixed8x13_Character_178,Fixed8x13_Character_179,Fixed8x13_Character_180,Fixed8x13_Character_181,Fixed8x13_Character_182,Fixed8x13_Character_183,Fixed8x13_Character_184,Fixed8x13_Character_185,Fixed8x13_Character_186,Fixed8x13_Character_187,Fixed8x13_Character_188,Fixed8x13_Character_189,Fixed8x13_Character_190,Fixed8x13_Character_191, - Fixed8x13_Character_192,Fixed8x13_Character_193,Fixed8x13_Character_194,Fixed8x13_Character_195,Fixed8x13_Character_196,Fixed8x13_Character_197,Fixed8x13_Character_198,Fixed8x13_Character_199,Fixed8x13_Character_200,Fixed8x13_Character_201,Fixed8x13_Character_202,Fixed8x13_Character_203,Fixed8x13_Character_204,Fixed8x13_Character_205,Fixed8x13_Character_206,Fixed8x13_Character_207, - Fixed8x13_Character_208,Fixed8x13_Character_209,Fixed8x13_Character_210,Fixed8x13_Character_211,Fixed8x13_Character_212,Fixed8x13_Character_213,Fixed8x13_Character_214,Fixed8x13_Character_215,Fixed8x13_Character_216,Fixed8x13_Character_217,Fixed8x13_Character_218,Fixed8x13_Character_219,Fixed8x13_Character_220,Fixed8x13_Character_221,Fixed8x13_Character_222,Fixed8x13_Character_223, - Fixed8x13_Character_224,Fixed8x13_Character_225,Fixed8x13_Character_226,Fixed8x13_Character_227,Fixed8x13_Character_228,Fixed8x13_Character_229,Fixed8x13_Character_230,Fixed8x13_Character_231,Fixed8x13_Character_232,Fixed8x13_Character_233,Fixed8x13_Character_234,Fixed8x13_Character_235,Fixed8x13_Character_236,Fixed8x13_Character_237,Fixed8x13_Character_238,Fixed8x13_Character_239, - Fixed8x13_Character_240,Fixed8x13_Character_241,Fixed8x13_Character_242,Fixed8x13_Character_243,Fixed8x13_Character_244,Fixed8x13_Character_245,Fixed8x13_Character_246,Fixed8x13_Character_247,Fixed8x13_Character_248,Fixed8x13_Character_249,Fixed8x13_Character_250,Fixed8x13_Character_251,Fixed8x13_Character_252,Fixed8x13_Character_253,Fixed8x13_Character_254,Fixed8x13_Character_255,NULL}; - -/* The font structure: */ -const SFG_Font fgFontFixed8x13 = { "-misc-fixed-medium-r-normal--13-120-75-75-C-80-iso8859-1", 256, 14, Fixed8x13_Character_Map, 0, 3 }; - -static const GLubyte Fixed9x15_Character_000[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_001[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 28, 0, 62, 0,127, 0,255,128,127, 0, 62, 0, 28, 0, 8, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_002[] = { 9, 0, 0,170,128, 85, 0,170,128, 85, 0,170,128, 85, 0,170,128, 85, 0,170,128, 85, 0,170,128, 85, 0,170,128, 85, 0,170,128}; -static const GLubyte Fixed9x15_Character_003[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 4, 0, 4, 0, 4, 0, 4, 0, 31, 0, 0, 0, 72, 0, 72, 0,120, 0, 72, 0, 72, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_004[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 14, 0, 72, 0, 79, 0, 64, 0,112, 0, 64, 0,120, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_005[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 9, 0, 14, 0, 9, 0, 14, 0, 0, 0, 56, 0, 64, 0, 64, 0, 64, 0, 56, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_006[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 14, 0, 8, 0, 15, 0, 0, 0,120, 0, 64, 0, 64, 0, 64, 0, 64, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_007[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 18, 0, 18, 0, 12, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_008[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 8, 0, 8, 0, 8, 0,127, 0, 8, 0, 8, 0, 8, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_009[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0, 68, 0, 76, 0, 84, 0,100, 0, 68, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_010[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 4, 0, 4, 0, 4, 0, 4, 0, 31, 0, 0, 0, 16, 0, 40, 0, 40, 0, 68, 0, 68, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_011[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,248, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_012[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_013[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_014[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_015[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0,255,128, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_016[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255,128}; -static const GLubyte Fixed9x15_Character_017[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255,128, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_018[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_019[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,255,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_020[] = { 9, 0, 0,255,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_021[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 15,128, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_022[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0,248, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_023[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,255,128, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_024[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0,255,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_025[] = { 9, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_026[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 3, 0, 28, 0, 96, 0, 28, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_027[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 96, 0, 28, 0, 3, 0, 28, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_028[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 34, 0, 34, 0, 34, 0, 34, 0, 34, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_029[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16, 0, 16, 0,127, 0, 8, 0,127, 0, 4, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_030[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 46, 0, 81, 0, 48, 0, 16, 0, 16, 0,124, 0, 16, 0, 16, 0, 17, 0, 14, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_031[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_032[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_033[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_034[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 18, 0, 18, 0, 18, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_035[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 36, 0, 36, 0,126, 0, 36, 0, 36, 0,126, 0, 36, 0, 36, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_036[] = { 9, 0, 0, 0, 0, 0, 0, 8, 0, 62, 0, 73, 0, 9, 0, 9, 0, 10, 0, 28, 0, 40, 0, 72, 0, 73, 0, 62, 0, 8, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_037[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 66, 0, 37, 0, 37, 0, 18, 0, 8, 0, 8, 0, 36, 0, 82, 0, 82, 0, 33, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_038[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 49, 0, 74, 0, 68, 0, 74, 0, 49, 0, 48, 0, 72, 0, 72, 0, 72, 0, 48, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_039[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16, 0, 8, 0, 4, 0, 6, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_040[] = { 9, 0, 0, 0, 0, 0, 0, 4, 0, 8, 0, 8, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 8, 0, 8, 0, 4, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_041[] = { 9, 0, 0, 0, 0, 0, 0, 16, 0, 8, 0, 8, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 8, 0, 8, 0, 16, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_042[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 73, 0, 42, 0, 28, 0, 42, 0, 73, 0, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_043[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0,127, 0, 8, 0, 8, 0, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_044[] = { 9, 0, 0, 8, 0, 4, 0, 4, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_045[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_046[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_047[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 64, 0, 32, 0, 32, 0, 16, 0, 8, 0, 8, 0, 4, 0, 2, 0, 2, 0, 1, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_048[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_049[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 72, 0, 40, 0, 24, 0, 8, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_050[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 64, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_051[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 1, 0, 1, 0, 1, 0, 14, 0, 4, 0, 2, 0, 1, 0,127, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_052[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 2, 0, 2, 0,127, 0, 66, 0, 34, 0, 18, 0, 10, 0, 6, 0, 2, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_053[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 1, 0, 1, 0, 1, 0, 97, 0, 94, 0, 64, 0, 64, 0,127, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_054[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 64, 0, 64, 0, 32, 0, 30, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_055[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 32, 0, 16, 0, 16, 0, 8, 0, 4, 0, 2, 0, 1, 0, 1, 0,127, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_056[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 65, 0, 65, 0, 34, 0, 28, 0, 34, 0, 65, 0, 34, 0, 28, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_057[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0, 2, 0, 1, 0, 1, 0, 61, 0, 67, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_058[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_059[] = { 9, 0, 0, 8, 0, 4, 0, 4, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_060[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 4, 0, 8, 0, 16, 0, 32, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_061[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 0, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_062[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0, 2, 0, 4, 0, 8, 0, 16, 0, 32, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_063[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 0, 0, 8, 0, 8, 0, 4, 0, 2, 0, 1, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_064[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0, 77, 0, 83, 0, 81, 0, 79, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_065[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 65, 0, 34, 0, 20, 0, 8, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_066[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 33, 0, 33, 0, 33, 0, 33, 0,126, 0, 33, 0, 33, 0, 33, 0,126, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_067[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_068[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 33, 0, 33, 0, 33, 0, 33, 0, 33, 0, 33, 0, 33, 0, 33, 0,126, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_069[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0, 32, 0,127, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_070[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 32, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0, 32, 0,127, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_071[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 71, 0, 64, 0, 64, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_072[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_073[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_074[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0, 66, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 15,128, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_075[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 66, 0, 68, 0, 72, 0, 80, 0,112, 0, 72, 0, 68, 0, 66, 0, 65, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_076[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_077[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 73, 0, 73, 0, 85, 0, 85, 0, 99, 0, 65, 0, 65, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_078[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 67, 0, 69, 0, 73, 0, 81, 0, 97, 0, 65, 0, 65, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_079[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_080[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0,126, 0, 65, 0, 65, 0, 65, 0,126, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_081[] = { 9, 0, 0, 0, 0, 3, 0, 4, 0, 62, 0, 73, 0, 81, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_082[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 66, 0, 68, 0, 72, 0,126, 0, 65, 0, 65, 0, 65, 0,126, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_083[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 1, 0, 6, 0, 56, 0, 64, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_084[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0,127, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_085[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_086[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 20, 0, 20, 0, 20, 0, 34, 0, 34, 0, 34, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_087[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 85, 0, 73, 0, 73, 0, 73, 0, 73, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_088[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 34, 0, 20, 0, 8, 0, 8, 0, 20, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_089[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 20, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_090[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 64, 0, 64, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0, 1, 0,127, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_091[] = { 9, 0, 0, 0, 0, 0, 0, 30, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 16, 0, 30, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_092[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 2, 0, 2, 0, 4, 0, 8, 0, 8, 0, 16, 0, 32, 0, 32, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_093[] = { 9, 0, 0, 0, 0, 0, 0, 60, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 4, 0, 60, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_094[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 34, 0, 20, 0, 8, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_095[] = { 9, 0, 0, 0, 0, 0, 0,255, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_096[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4, 0, 8, 0, 16, 0, 48, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_097[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_098[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 94, 0, 97, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 64, 0, 64, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_099[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 64, 0, 64, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_100[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 65, 0, 65, 0, 67, 0, 61, 0, 1, 0, 1, 0, 1, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_101[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0,127, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_102[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 16, 0, 16, 0, 16, 0, 16, 0,124, 0, 16, 0, 16, 0, 17, 0, 17, 0, 14, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_103[] = { 9, 0, 0, 62, 0, 65, 0, 65, 0, 62, 0, 64, 0, 60, 0, 66, 0, 66, 0, 66, 0, 61, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_104[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 64, 0, 64, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_105[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 0, 0, 24, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_106[] = { 9, 0, 0, 60, 0, 66, 0, 66, 0, 66, 0, 2, 0, 2, 0, 2, 0, 2, 0, 2, 0, 14, 0, 0, 0, 0, 0, 6, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_107[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 70, 0, 88, 0, 96, 0, 88, 0, 70, 0, 65, 0, 64, 0, 64, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_108[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_109[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 73, 0, 73, 0, 73, 0, 73, 0, 73, 0,118, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_110[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_111[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_112[] = { 9, 0, 0, 64, 0, 64, 0, 64, 0, 94, 0, 97, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_113[] = { 9, 0, 0, 1, 0, 1, 0, 1, 0, 61, 0, 67, 0, 65, 0, 65, 0, 65, 0, 67, 0, 61, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_114[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 32, 0, 32, 0, 32, 0, 33, 0, 49, 0, 78, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_115[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 1, 0, 62, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_116[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 14, 0, 17, 0, 16, 0, 16, 0, 16, 0, 16, 0,126, 0, 16, 0, 16, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_117[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_118[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 20, 0, 20, 0, 34, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_119[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 85, 0, 73, 0, 73, 0, 73, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_120[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 34, 0, 20, 0, 8, 0, 20, 0, 34, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_121[] = { 9, 0, 0, 60, 0, 66, 0, 2, 0, 58, 0, 70, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_122[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_123[] = { 9, 0, 0, 0, 0, 0, 0, 7, 0, 8, 0, 8, 0, 8, 0, 4, 0, 24, 0, 24, 0, 4, 0, 8, 0, 8, 0, 8, 0, 7, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_124[] = { 9, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_125[] = { 9, 0, 0, 0, 0, 0, 0,112, 0, 8, 0, 8, 0, 8, 0, 16, 0, 12, 0, 12, 0, 16, 0, 8, 0, 8, 0, 8, 0,112, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_126[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 70, 0, 73, 0, 49, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_127[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_128[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_129[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_130[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_131[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_132[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_133[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_134[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_135[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_136[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_137[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_138[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_139[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_140[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_141[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_142[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_143[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_144[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_145[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_146[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_147[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_148[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_149[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_150[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_151[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_152[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_153[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_154[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_155[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_156[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_157[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_158[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_159[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 91, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0, 65, 0, 64, 0, 1, 0,109, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_160[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_161[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0, 0, 0, 8, 0, 8, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_162[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 60, 0, 82, 0, 80, 0, 72, 0, 74, 0, 60, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_163[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 46, 0, 81, 0, 48, 0, 16, 0, 16, 0,124, 0, 16, 0, 16, 0, 17, 0, 14, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_164[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 62, 0, 34, 0, 34, 0, 62, 0, 65, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_165[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 62, 0, 8, 0, 62, 0, 20, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_166[] = { 9, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_167[] = { 9, 0, 0, 0, 0, 0, 0, 24, 0, 36, 0, 4, 0, 24, 0, 36, 0, 36, 0, 36, 0, 24, 0, 32, 0, 36, 0, 24, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_168[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 34, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_169[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0, 66, 0,153, 0,165, 0,161, 0,165, 0,153, 0, 66, 0, 60, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_170[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 0, 0, 60, 0, 72, 0, 56, 0, 72, 0, 48, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_171[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 18, 0, 36, 0, 72, 0, 72, 0, 36, 0, 18, 0, 9, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_172[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 2, 0, 2, 0,126, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_173[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_174[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0, 66, 0,165, 0,169, 0,185, 0,165, 0,185, 0, 66, 0, 60, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_175[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_176[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 18, 0, 18, 0, 12, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_177[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 0, 0, 8, 0, 8, 0, 8, 0,127, 0, 8, 0, 8, 0, 8, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_178[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120, 0, 64, 0, 48, 0, 8, 0, 72, 0, 48, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_179[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 0, 72, 0, 8, 0, 16, 0, 72, 0, 48, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_180[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16, 0, 8, 0, 4, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_181[] = { 9, 0, 0, 0, 0, 64, 0, 64, 0, 93, 0, 99, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_182[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 5, 0, 5, 0, 5, 0, 5, 0, 5, 0, 61, 0, 69, 0, 69, 0, 69, 0, 63, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_183[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_184[] = { 9, 0, 0, 24, 0, 36, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_185[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,112, 0, 32, 0, 32, 0, 32, 0, 96, 0, 32, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_186[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 0, 0, 56, 0, 68, 0, 68, 0, 56, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_187[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 72, 0, 36, 0, 18, 0, 9, 0, 9, 0, 18, 0, 36, 0, 72, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_188[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 13, 0, 9, 0, 5, 0,115, 0, 33, 0, 32, 0, 32, 0, 96, 0, 32, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_189[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 8, 0, 6, 0, 1, 0,121, 0, 38, 0, 32, 0, 32, 0, 96, 0, 32, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_190[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 13, 0, 9, 0, 5, 0, 51, 0, 73, 0, 8, 0, 16, 0, 72, 0, 48, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_191[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 64, 0, 32, 0, 16, 0, 8, 0, 8, 0, 0, 0, 8, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_192[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 8, 0, 16, 0, 32, 0}; -static const GLubyte Fixed9x15_Character_193[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 8, 0, 4, 0, 2, 0}; -static const GLubyte Fixed9x15_Character_194[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 34, 0, 20, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_195[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 78, 0, 49, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_196[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 34, 0, 34, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_197[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 34, 0, 20, 0, 28, 0, 34, 0, 28, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_198[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 79, 0, 72, 0, 72, 0, 72, 0,126, 0, 72, 0, 72, 0, 72, 0, 72, 0, 55, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_199[] = { 9, 0, 0, 24, 0, 36, 0, 12, 0, 62, 0, 65, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_200[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0,127, 0, 0, 0, 8, 0, 16, 0, 32, 0}; -static const GLubyte Fixed9x15_Character_201[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0,127, 0, 0, 0, 8, 0, 4, 0, 2, 0}; -static const GLubyte Fixed9x15_Character_202[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0,127, 0, 0, 0, 34, 0, 20, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_203[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 32, 0, 32, 0, 32, 0, 60, 0, 32, 0, 32, 0,127, 0, 0, 0, 34, 0, 34, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_204[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 62, 0, 0, 0, 8, 0, 16, 0, 32, 0}; -static const GLubyte Fixed9x15_Character_205[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 62, 0, 0, 0, 8, 0, 4, 0, 2, 0}; -static const GLubyte Fixed9x15_Character_206[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 62, 0, 0, 0, 34, 0, 20, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_207[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 62, 0, 0, 0, 34, 0, 34, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_208[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 33, 0, 33, 0, 33, 0, 33, 0,225, 0, 33, 0, 33, 0, 33, 0,124, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_209[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 67, 0, 69, 0, 73, 0, 73, 0, 81, 0, 97, 0, 65, 0, 0, 0, 78, 0, 49, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_210[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 16, 0, 32, 0}; -static const GLubyte Fixed9x15_Character_211[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 4, 0, 2, 0}; -static const GLubyte Fixed9x15_Character_212[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 20, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_213[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 78, 0, 49, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_214[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 34, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_215[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 34, 0, 20, 0, 8, 0, 20, 0, 34, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_216[] = { 9, 0, 0, 0, 0, 0, 0, 64, 0, 62, 0, 97, 0, 81, 0, 81, 0, 73, 0, 73, 0, 69, 0, 69, 0, 67, 0, 62, 0, 1, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_217[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 8, 0, 16, 0, 32, 0}; -static const GLubyte Fixed9x15_Character_218[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 8, 0, 4, 0, 2, 0}; -static const GLubyte Fixed9x15_Character_219[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 34, 0, 20, 0, 8, 0}; -static const GLubyte Fixed9x15_Character_220[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 34, 0, 34, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_221[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 20, 0, 34, 0, 65, 0, 65, 0, 0, 0, 8, 0, 4, 0, 2, 0}; -static const GLubyte Fixed9x15_Character_222[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 64, 0, 64, 0, 64, 0,126, 0, 65, 0, 65, 0, 65, 0,126, 0, 64, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_223[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 44, 0, 34, 0, 34, 0, 34, 0, 36, 0,104, 0, 36, 0, 34, 0, 34, 0, 28, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_224[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 4, 0, 8, 0, 16, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_225[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 8, 0, 4, 0, 2, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_226[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 34, 0, 20, 0, 8, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_227[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 38, 0, 25, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_228[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 34, 0, 34, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_229[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 67, 0, 65, 0, 63, 0, 1, 0, 1, 0, 62, 0, 0, 0, 12, 0, 18, 0, 12, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_230[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 55, 0, 73, 0, 72, 0, 62, 0, 9, 0, 73, 0, 54, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_231[] = { 9, 0, 0, 24, 0, 36, 0, 12, 0, 62, 0, 65, 0, 64, 0, 64, 0, 64, 0, 65, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_232[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0,127, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 16, 0, 32, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_233[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0,127, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 4, 0, 2, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_234[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0,127, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 20, 0, 8, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_235[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 64, 0, 64, 0,127, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 34, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_236[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 8, 0, 16, 0, 32, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_237[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 16, 0, 8, 0, 4, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_238[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 68, 0, 40, 0, 16, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_239[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 8, 0, 8, 0, 8, 0, 8, 0, 8, 0, 56, 0, 0, 0, 36, 0, 36, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_240[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 4, 0, 40, 0, 24, 0, 36, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_241[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 97, 0, 94, 0, 0, 0, 78, 0, 49, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_242[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 16, 0, 32, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_243[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 8, 0, 4, 0, 2, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_244[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 20, 0, 8, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_245[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 78, 0, 49, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_246[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 62, 0, 0, 0, 34, 0, 34, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_247[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 28, 0, 8, 0, 0, 0,127, 0, 0, 0, 8, 0, 28, 0, 8, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_248[] = { 9, 0, 0, 0, 0, 0, 0, 64, 0, 62, 0, 81, 0, 81, 0, 73, 0, 69, 0, 69, 0, 62, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_249[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 8, 0, 16, 0, 32, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_250[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 8, 0, 4, 0, 2, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_251[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 34, 0, 20, 0, 8, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_252[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 61, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 36, 0, 36, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_253[] = { 9, 0, 0, 60, 0, 66, 0, 2, 0, 58, 0, 70, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 16, 0, 8, 0, 4, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_254[] = { 9, 0, 0, 64, 0, 64, 0, 64, 0, 94, 0, 97, 0, 65, 0, 65, 0, 97, 0, 94, 0, 64, 0, 64, 0, 64, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Fixed9x15_Character_255[] = { 9, 0, 0, 60, 0, 66, 0, 2, 0, 58, 0, 70, 0, 66, 0, 66, 0, 66, 0, 66, 0, 66, 0, 0, 0, 36, 0, 36, 0, 0, 0, 0, 0}; - -/* The font characters mapping: */ -static const GLubyte* Fixed9x15_Character_Map[] = {Fixed9x15_Character_000,Fixed9x15_Character_001,Fixed9x15_Character_002,Fixed9x15_Character_003,Fixed9x15_Character_004,Fixed9x15_Character_005,Fixed9x15_Character_006,Fixed9x15_Character_007,Fixed9x15_Character_008,Fixed9x15_Character_009,Fixed9x15_Character_010,Fixed9x15_Character_011,Fixed9x15_Character_012,Fixed9x15_Character_013,Fixed9x15_Character_014,Fixed9x15_Character_015, - Fixed9x15_Character_016,Fixed9x15_Character_017,Fixed9x15_Character_018,Fixed9x15_Character_019,Fixed9x15_Character_020,Fixed9x15_Character_021,Fixed9x15_Character_022,Fixed9x15_Character_023,Fixed9x15_Character_024,Fixed9x15_Character_025,Fixed9x15_Character_026,Fixed9x15_Character_027,Fixed9x15_Character_028,Fixed9x15_Character_029,Fixed9x15_Character_030,Fixed9x15_Character_031, - Fixed9x15_Character_032,Fixed9x15_Character_033,Fixed9x15_Character_034,Fixed9x15_Character_035,Fixed9x15_Character_036,Fixed9x15_Character_037,Fixed9x15_Character_038,Fixed9x15_Character_039,Fixed9x15_Character_040,Fixed9x15_Character_041,Fixed9x15_Character_042,Fixed9x15_Character_043,Fixed9x15_Character_044,Fixed9x15_Character_045,Fixed9x15_Character_046,Fixed9x15_Character_047, - Fixed9x15_Character_048,Fixed9x15_Character_049,Fixed9x15_Character_050,Fixed9x15_Character_051,Fixed9x15_Character_052,Fixed9x15_Character_053,Fixed9x15_Character_054,Fixed9x15_Character_055,Fixed9x15_Character_056,Fixed9x15_Character_057,Fixed9x15_Character_058,Fixed9x15_Character_059,Fixed9x15_Character_060,Fixed9x15_Character_061,Fixed9x15_Character_062,Fixed9x15_Character_063, - Fixed9x15_Character_064,Fixed9x15_Character_065,Fixed9x15_Character_066,Fixed9x15_Character_067,Fixed9x15_Character_068,Fixed9x15_Character_069,Fixed9x15_Character_070,Fixed9x15_Character_071,Fixed9x15_Character_072,Fixed9x15_Character_073,Fixed9x15_Character_074,Fixed9x15_Character_075,Fixed9x15_Character_076,Fixed9x15_Character_077,Fixed9x15_Character_078,Fixed9x15_Character_079, - Fixed9x15_Character_080,Fixed9x15_Character_081,Fixed9x15_Character_082,Fixed9x15_Character_083,Fixed9x15_Character_084,Fixed9x15_Character_085,Fixed9x15_Character_086,Fixed9x15_Character_087,Fixed9x15_Character_088,Fixed9x15_Character_089,Fixed9x15_Character_090,Fixed9x15_Character_091,Fixed9x15_Character_092,Fixed9x15_Character_093,Fixed9x15_Character_094,Fixed9x15_Character_095, - Fixed9x15_Character_096,Fixed9x15_Character_097,Fixed9x15_Character_098,Fixed9x15_Character_099,Fixed9x15_Character_100,Fixed9x15_Character_101,Fixed9x15_Character_102,Fixed9x15_Character_103,Fixed9x15_Character_104,Fixed9x15_Character_105,Fixed9x15_Character_106,Fixed9x15_Character_107,Fixed9x15_Character_108,Fixed9x15_Character_109,Fixed9x15_Character_110,Fixed9x15_Character_111, - Fixed9x15_Character_112,Fixed9x15_Character_113,Fixed9x15_Character_114,Fixed9x15_Character_115,Fixed9x15_Character_116,Fixed9x15_Character_117,Fixed9x15_Character_118,Fixed9x15_Character_119,Fixed9x15_Character_120,Fixed9x15_Character_121,Fixed9x15_Character_122,Fixed9x15_Character_123,Fixed9x15_Character_124,Fixed9x15_Character_125,Fixed9x15_Character_126,Fixed9x15_Character_032, - Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032, - Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032,Fixed9x15_Character_032, - Fixed9x15_Character_160,Fixed9x15_Character_161,Fixed9x15_Character_162,Fixed9x15_Character_163,Fixed9x15_Character_164,Fixed9x15_Character_165,Fixed9x15_Character_166,Fixed9x15_Character_167,Fixed9x15_Character_168,Fixed9x15_Character_169,Fixed9x15_Character_170,Fixed9x15_Character_171,Fixed9x15_Character_172,Fixed9x15_Character_173,Fixed9x15_Character_174,Fixed9x15_Character_175, - Fixed9x15_Character_176,Fixed9x15_Character_177,Fixed9x15_Character_178,Fixed9x15_Character_179,Fixed9x15_Character_180,Fixed9x15_Character_181,Fixed9x15_Character_182,Fixed9x15_Character_183,Fixed9x15_Character_184,Fixed9x15_Character_185,Fixed9x15_Character_186,Fixed9x15_Character_187,Fixed9x15_Character_188,Fixed9x15_Character_189,Fixed9x15_Character_190,Fixed9x15_Character_191, - Fixed9x15_Character_192,Fixed9x15_Character_193,Fixed9x15_Character_194,Fixed9x15_Character_195,Fixed9x15_Character_196,Fixed9x15_Character_197,Fixed9x15_Character_198,Fixed9x15_Character_199,Fixed9x15_Character_200,Fixed9x15_Character_201,Fixed9x15_Character_202,Fixed9x15_Character_203,Fixed9x15_Character_204,Fixed9x15_Character_205,Fixed9x15_Character_206,Fixed9x15_Character_207, - Fixed9x15_Character_208,Fixed9x15_Character_209,Fixed9x15_Character_210,Fixed9x15_Character_211,Fixed9x15_Character_212,Fixed9x15_Character_213,Fixed9x15_Character_214,Fixed9x15_Character_215,Fixed9x15_Character_216,Fixed9x15_Character_217,Fixed9x15_Character_218,Fixed9x15_Character_219,Fixed9x15_Character_220,Fixed9x15_Character_221,Fixed9x15_Character_222,Fixed9x15_Character_223, - Fixed9x15_Character_224,Fixed9x15_Character_225,Fixed9x15_Character_226,Fixed9x15_Character_227,Fixed9x15_Character_228,Fixed9x15_Character_229,Fixed9x15_Character_230,Fixed9x15_Character_231,Fixed9x15_Character_232,Fixed9x15_Character_233,Fixed9x15_Character_234,Fixed9x15_Character_235,Fixed9x15_Character_236,Fixed9x15_Character_237,Fixed9x15_Character_238,Fixed9x15_Character_239, - Fixed9x15_Character_240,Fixed9x15_Character_241,Fixed9x15_Character_242,Fixed9x15_Character_243,Fixed9x15_Character_244,Fixed9x15_Character_245,Fixed9x15_Character_246,Fixed9x15_Character_247,Fixed9x15_Character_248,Fixed9x15_Character_249,Fixed9x15_Character_250,Fixed9x15_Character_251,Fixed9x15_Character_252,Fixed9x15_Character_253,Fixed9x15_Character_254,Fixed9x15_Character_255,NULL}; - -/* The font structure: */ -const SFG_Font fgFontFixed9x15 = { "-misc-fixed-medium-r-normal--15-140-75-75-C-90-iso8859-1", 256, 16, Fixed9x15_Character_Map, 0, 4 }; - -static const GLubyte Helvetica10_Character_000[] = { 8, 0, 0, 0, 84, 0, 68, 0, 68, 0, 84, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_001[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_002[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_003[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_004[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_005[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_006[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_007[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_008[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_009[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_010[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_011[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_012[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_013[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_014[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_015[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_016[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_017[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_018[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_019[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_020[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_021[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_022[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_023[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_024[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_025[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_026[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_027[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_028[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_029[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_030[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_031[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_032[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_033[] = { 3, 0, 0, 0, 64, 0, 64, 64, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica10_Character_034[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 80, 80, 0, 0, 0}; -static const GLubyte Helvetica10_Character_035[] = { 6, 0, 0, 0, 80, 80,248, 40,124, 40, 40, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_036[] = { 6, 0, 0, 32,112,168, 40,112,160,168,112, 32, 0, 0, 0}; -static const GLubyte Helvetica10_Character_037[] = { 9, 0, 0, 0, 0, 0, 0, 38, 0, 41, 0, 22, 0, 16, 0, 8, 0,104, 0,148, 0,100, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_038[] = { 8, 0, 0, 0, 50, 76, 76, 82, 48, 40, 40, 16, 0, 0, 0}; -static const GLubyte Helvetica10_Character_039[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 64, 32, 32, 0, 0, 0}; -static const GLubyte Helvetica10_Character_040[] = { 4, 0, 32, 64, 64,128,128,128,128, 64, 64, 32, 0, 0, 0}; -static const GLubyte Helvetica10_Character_041[] = { 4, 0, 64, 32, 32, 16, 16, 16, 16, 32, 32, 64, 0, 0, 0}; -static const GLubyte Helvetica10_Character_042[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0,160, 64,160, 0, 0, 0}; -static const GLubyte Helvetica10_Character_043[] = { 6, 0, 0, 0, 0, 32, 32,248, 32, 32, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_044[] = { 3, 0,128, 64, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_045[] = { 7, 0, 0, 0, 0, 0, 0,124, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_046[] = { 3, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_047[] = { 3, 0, 0, 0,128,128, 64, 64, 64, 64, 32, 32, 0, 0, 0}; -static const GLubyte Helvetica10_Character_048[] = { 6, 0, 0, 0,112,136,136,136,136,136,136,112, 0, 0, 0}; -static const GLubyte Helvetica10_Character_049[] = { 6, 0, 0, 0, 32, 32, 32, 32, 32, 32, 96, 32, 0, 0, 0}; -static const GLubyte Helvetica10_Character_050[] = { 6, 0, 0, 0,248,128, 64, 48, 8, 8,136,112, 0, 0, 0}; -static const GLubyte Helvetica10_Character_051[] = { 6, 0, 0, 0,112,136, 8, 8, 48, 8,136,112, 0, 0, 0}; -static const GLubyte Helvetica10_Character_052[] = { 6, 0, 0, 0, 16, 16,248,144, 80, 80, 48, 16, 0, 0, 0}; -static const GLubyte Helvetica10_Character_053[] = { 6, 0, 0, 0,112,136, 8, 8,240,128,128,248, 0, 0, 0}; -static const GLubyte Helvetica10_Character_054[] = { 6, 0, 0, 0,112,136,136,200,176,128,136,112, 0, 0, 0}; -static const GLubyte Helvetica10_Character_055[] = { 6, 0, 0, 0, 64, 64, 32, 32, 16, 16, 8,248, 0, 0, 0}; -static const GLubyte Helvetica10_Character_056[] = { 6, 0, 0, 0,112,136,136,136,112,136,136,112, 0, 0, 0}; -static const GLubyte Helvetica10_Character_057[] = { 6, 0, 0, 0,112,136, 8,104,152,136,136,112, 0, 0, 0}; -static const GLubyte Helvetica10_Character_058[] = { 3, 0, 0, 0, 64, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_059[] = { 3, 0,128, 64, 64, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_060[] = { 6, 0, 0, 0, 0, 16, 32, 64, 32, 16, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_061[] = { 5, 0, 0, 0, 0, 0,240, 0,240, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_062[] = { 6, 0, 0, 0, 0, 64, 32, 16, 32, 64, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_063[] = { 6, 0, 0, 0, 32, 0, 32, 32, 16, 8, 72, 48, 0, 0, 0}; -static const GLubyte Helvetica10_Character_064[] = { 11, 0, 0, 62, 0, 64, 0,155, 0,164,128,164,128,162, 64,146, 64, 77, 64, 32,128, 31, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_065[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 0, 0}; -static const GLubyte Helvetica10_Character_066[] = { 7, 0, 0, 0,120, 68, 68, 68,120, 68, 68,120, 0, 0, 0}; -static const GLubyte Helvetica10_Character_067[] = { 8, 0, 0, 0, 60, 66, 64, 64, 64, 64, 66, 60, 0, 0, 0}; -static const GLubyte Helvetica10_Character_068[] = { 8, 0, 0, 0,120, 68, 66, 66, 66, 66, 68,120, 0, 0, 0}; -static const GLubyte Helvetica10_Character_069[] = { 7, 0, 0, 0,124, 64, 64, 64,124, 64, 64,124, 0, 0, 0}; -static const GLubyte Helvetica10_Character_070[] = { 6, 0, 0, 0, 64, 64, 64, 64,120, 64, 64,124, 0, 0, 0}; -static const GLubyte Helvetica10_Character_071[] = { 8, 0, 0, 0, 58, 70, 66, 70, 64, 64, 66, 60, 0, 0, 0}; -static const GLubyte Helvetica10_Character_072[] = { 8, 0, 0, 0, 66, 66, 66, 66,126, 66, 66, 66, 0, 0, 0}; -static const GLubyte Helvetica10_Character_073[] = { 3, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica10_Character_074[] = { 5, 0, 0, 0, 96,144, 16, 16, 16, 16, 16, 16, 0, 0, 0}; -static const GLubyte Helvetica10_Character_075[] = { 7, 0, 0, 0, 68, 68, 72, 72,112, 80, 72, 68, 0, 0, 0}; -static const GLubyte Helvetica10_Character_076[] = { 6, 0, 0, 0,120, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica10_Character_077[] = { 9, 0, 0, 0, 0, 0, 0, 73, 0, 73, 0, 73, 0, 85, 0, 85, 0, 99, 0, 99, 0, 65, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_078[] = { 8, 0, 0, 0, 70, 70, 74, 74, 82, 82, 98, 98, 0, 0, 0}; -static const GLubyte Helvetica10_Character_079[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 0, 0}; -static const GLubyte Helvetica10_Character_080[] = { 7, 0, 0, 0, 64, 64, 64, 64,120, 68, 68,120, 0, 0, 0}; -static const GLubyte Helvetica10_Character_081[] = { 8, 0, 0, 1, 62, 70, 74, 66, 66, 66, 66, 60, 0, 0, 0}; -static const GLubyte Helvetica10_Character_082[] = { 7, 0, 0, 0, 68, 68, 68, 68,120, 68, 68,120, 0, 0, 0}; -static const GLubyte Helvetica10_Character_083[] = { 7, 0, 0, 0, 56, 68, 68, 4, 56, 64, 68, 56, 0, 0, 0}; -static const GLubyte Helvetica10_Character_084[] = { 5, 0, 0, 0, 32, 32, 32, 32, 32, 32, 32,248, 0, 0, 0}; -static const GLubyte Helvetica10_Character_085[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 0, 0, 0}; -static const GLubyte Helvetica10_Character_086[] = { 7, 0, 0, 0, 16, 40, 40, 68, 68, 68,130,130, 0, 0, 0}; -static const GLubyte Helvetica10_Character_087[] = { 9, 0, 0, 0, 0, 0, 0, 34, 0, 34, 0, 34, 0, 85, 0, 73, 0, 73, 0,136,128,136,128, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_088[] = { 7, 0, 0, 0, 68, 68, 40, 40, 16, 40, 68, 68, 0, 0, 0}; -static const GLubyte Helvetica10_Character_089[] = { 7, 0, 0, 0, 16, 16, 16, 40, 40, 68, 68,130, 0, 0, 0}; -static const GLubyte Helvetica10_Character_090[] = { 7, 0, 0, 0,124, 64, 32, 16, 16, 8, 4,124, 0, 0, 0}; -static const GLubyte Helvetica10_Character_091[] = { 3, 0, 96, 64, 64, 64, 64, 64, 64, 64, 64, 96, 0, 0, 0}; -static const GLubyte Helvetica10_Character_092[] = { 3, 0, 0, 0, 32, 32, 64, 64, 64, 64,128,128, 0, 0, 0}; -static const GLubyte Helvetica10_Character_093[] = { 3, 0,192, 64, 64, 64, 64, 64, 64, 64, 64,192, 0, 0, 0}; -static const GLubyte Helvetica10_Character_094[] = { 6, 0, 0, 0, 0, 0, 0,136, 80, 80, 32, 32, 0, 0, 0}; -static const GLubyte Helvetica10_Character_095[] = { 6, 0,252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_096[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 63, 64, 32, 0, 0, 0}; -static const GLubyte Helvetica10_Character_097[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_098[] = { 6, 0, 0, 0,176,200,136,136,200,176,128,128, 0, 0, 0}; -static const GLubyte Helvetica10_Character_099[] = { 5, 0, 0, 0, 96,144,128,128,144, 96, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_100[] = { 6, 0, 0, 0,104,152,136,136,152,104, 8, 8, 0, 0, 0}; -static const GLubyte Helvetica10_Character_101[] = { 5, 0, 0, 0, 96,144,128,240,144, 96, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_102[] = { 4, 0, 0, 0, 64, 64, 64, 64, 64,224, 64, 48, 0, 0, 0}; -static const GLubyte Helvetica10_Character_103[] = { 6, 0,112, 8,104,152,136,136,152,104, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_104[] = { 6, 0, 0, 0,136,136,136,136,200,176,128,128, 0, 0, 0}; -static const GLubyte Helvetica10_Character_105[] = { 2, 0, 0, 0,128,128,128,128,128,128, 0,128, 0, 0, 0}; -static const GLubyte Helvetica10_Character_106[] = { 2, 0, 0,128,128,128,128,128,128,128, 0,128, 0, 0, 0}; -static const GLubyte Helvetica10_Character_107[] = { 5, 0, 0, 0,144,144,160,192,160,144,128,128, 0, 0, 0}; -static const GLubyte Helvetica10_Character_108[] = { 2, 0, 0, 0,128,128,128,128,128,128,128,128, 0, 0, 0}; -static const GLubyte Helvetica10_Character_109[] = { 8, 0, 0, 0,146,146,146,146,146,236, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_110[] = { 6, 0, 0, 0,136,136,136,136,200,176, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_111[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_112[] = { 6, 0,128,128,176,200,136,136,200,176, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_113[] = { 6, 0, 8, 8,104,152,136,136,152,104, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_114[] = { 4, 0, 0, 0,128,128,128,128,192,160, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_115[] = { 5, 0, 0, 0, 96,144, 16, 96,144, 96, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_116[] = { 4, 0, 0, 0, 96, 64, 64, 64, 64,224, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica10_Character_117[] = { 5, 0, 0, 0,112,144,144,144,144,144, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_118[] = { 6, 0, 0, 0, 32, 32, 80, 80,136,136, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_119[] = { 8, 0, 0, 0, 40, 40, 84, 84,146,146, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_120[] = { 6, 0, 0, 0,136,136, 80, 32, 80,136, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_121[] = { 5, 0,128, 64, 64, 96,160,160,144,144, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_122[] = { 5, 0, 0, 0,240,128, 64, 32, 16,240, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_123[] = { 3, 0, 32, 64, 64, 64, 64,128, 64, 64, 64, 32, 0, 0, 0}; -static const GLubyte Helvetica10_Character_124[] = { 3, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica10_Character_125[] = { 3, 0,128, 64, 64, 64, 64, 32, 64, 64, 64,128, 0, 0, 0}; -static const GLubyte Helvetica10_Character_126[] = { 7, 0, 0, 0, 0, 0, 0,152,100, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_127[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_128[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_129[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_130[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_131[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_132[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_133[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_134[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_135[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_136[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_137[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_138[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_139[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_140[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_141[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_142[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_143[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_144[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_145[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_146[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_147[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_148[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_149[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_150[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_151[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_152[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_153[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_154[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_155[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_156[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_157[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_158[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_159[] = { 13, 0, 0, 0, 0, 0, 0, 84, 0, 0, 0, 68, 0, 0, 0, 68, 0, 0, 0, 84, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_160[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_161[] = { 3, 0, 64, 64, 64, 64, 64, 64, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_162[] = { 6, 0, 0, 64,112,168,160,160,168,112, 16, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_163[] = { 6, 0, 0, 0,176, 72, 64, 64,224, 64, 72, 48, 0, 0, 0}; -static const GLubyte Helvetica10_Character_164[] = { 5, 0, 0, 0, 0,144, 96,144,144, 96,144, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_165[] = { 6, 0, 0, 0, 32,248, 32,248, 80, 80,136,136, 0, 0, 0}; -static const GLubyte Helvetica10_Character_166[] = { 3, 0, 64, 64, 64, 64, 0, 0, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica10_Character_167[] = { 6, 0,112,136, 24,112,200,152,112,192,136,112, 0, 0, 0}; -static const GLubyte Helvetica10_Character_168[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,160, 0, 0, 0}; -static const GLubyte Helvetica10_Character_169[] = { 9, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 77, 0, 81, 0, 77, 0, 34, 0, 28, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_170[] = { 4, 0, 0, 0, 0, 0, 0,224, 0,160, 32,224, 0, 0, 0}; -static const GLubyte Helvetica10_Character_171[] = { 6, 0, 0, 0, 40, 80,160, 80, 40, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_172[] = { 7, 0, 0, 0, 0, 0, 4, 4,124, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_173[] = { 4, 0, 0, 0, 0, 0, 0,224, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_174[] = { 9, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 85, 0, 89, 0, 93, 0, 34, 0, 28, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_175[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,224, 0, 0, 0}; -static const GLubyte Helvetica10_Character_176[] = { 4, 0, 0, 0, 0, 0, 0, 96,144,144, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_177[] = { 6, 0, 0, 0,248, 0, 32, 32,248, 32, 32, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_178[] = { 3, 0, 0, 0, 0, 0, 0,224, 64,160, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_179[] = { 3, 0, 0, 0, 0, 0, 0,192, 32, 64,224, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_180[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0,128, 64, 0, 0, 0}; -static const GLubyte Helvetica10_Character_181[] = { 5, 0,128,128,240,144,144,144,144,144, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_182[] = { 6, 0, 40, 40, 40, 40, 40,104,232,232,232,124, 0, 0, 0}; -static const GLubyte Helvetica10_Character_183[] = { 3, 0, 0, 0, 0, 0, 0,192, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_184[] = { 3, 0,192, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_185[] = { 3, 0, 0, 0, 0, 0, 0, 64, 64,192, 64, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_186[] = { 4, 0, 0, 0, 0, 0, 0,224, 0,224,160,224, 0, 0, 0}; -static const GLubyte Helvetica10_Character_187[] = { 6, 0, 0, 0,160, 80, 40, 80,160, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_188[] = { 9, 0, 0, 0, 0, 0, 0, 33, 0, 23,128, 19, 0, 9, 0, 72, 0, 68, 0,196, 0, 66, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_189[] = { 9, 0, 0, 0, 0, 0, 0, 39, 0, 18, 0, 21, 0, 11, 0, 72, 0, 68, 0,196, 0, 66, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_190[] = { 9, 0, 0, 0, 0, 0, 0, 33, 0, 23,128, 19, 0, 9, 0,200, 0, 36, 0, 68, 0,226, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_191[] = { 6, 0, 48, 72, 64, 32, 16, 16, 0, 16, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_192[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 16, 32}; -static const GLubyte Helvetica10_Character_193[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 16, 8}; -static const GLubyte Helvetica10_Character_194[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 40, 16}; -static const GLubyte Helvetica10_Character_195[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 40, 20}; -static const GLubyte Helvetica10_Character_196[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 0, 40, 0}; -static const GLubyte Helvetica10_Character_197[] = { 7, 0, 0, 0,130,130,124, 68, 40, 40, 16, 16, 16, 40, 16}; -static const GLubyte Helvetica10_Character_198[] = { 10, 0, 0, 0, 0, 0, 0,143,128,136, 0,120, 0, 72, 0, 47,128, 40, 0, 24, 0, 31,128, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_199[] = { 8, 0, 24, 8, 60, 66, 64, 64, 64, 64, 66, 60, 0, 0, 0}; -static const GLubyte Helvetica10_Character_200[] = { 7, 0, 0, 0,124, 64, 64, 64,124, 64, 64,124, 0, 16, 32}; -static const GLubyte Helvetica10_Character_201[] = { 7, 0, 0, 0,124, 64, 64, 64,124, 64, 64,124, 0, 16, 8}; -static const GLubyte Helvetica10_Character_202[] = { 7, 0, 0, 0,124, 64, 64,124, 64, 64, 64,124, 0, 40, 16}; -static const GLubyte Helvetica10_Character_203[] = { 7, 0, 0, 0,124, 64, 64, 64,124, 64, 64,124, 0, 40, 0}; -static const GLubyte Helvetica10_Character_204[] = { 3, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64,128}; -static const GLubyte Helvetica10_Character_205[] = { 3, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64, 32}; -static const GLubyte Helvetica10_Character_206[] = { 3, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0,160, 64}; -static const GLubyte Helvetica10_Character_207[] = { 3, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0,160, 0}; -static const GLubyte Helvetica10_Character_208[] = { 8, 0, 0, 0,120, 68, 66, 66,242, 66, 68,120, 0, 0, 0}; -static const GLubyte Helvetica10_Character_209[] = { 8, 0, 0, 0, 70, 70, 74, 74, 82, 82, 98, 98, 0, 40, 20}; -static const GLubyte Helvetica10_Character_210[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 8, 16}; -static const GLubyte Helvetica10_Character_211[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 8, 4}; -static const GLubyte Helvetica10_Character_212[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 20, 8}; -static const GLubyte Helvetica10_Character_213[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 40, 20}; -static const GLubyte Helvetica10_Character_214[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 60, 0, 36, 0}; -static const GLubyte Helvetica10_Character_215[] = { 6, 0, 0, 0, 0,136, 80, 32, 80,136, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_216[] = { 8, 0, 0, 64, 60, 98, 82, 82, 74, 74, 70, 60, 2, 0, 0}; -static const GLubyte Helvetica10_Character_217[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 0, 8, 16}; -static const GLubyte Helvetica10_Character_218[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 0, 16, 8}; -static const GLubyte Helvetica10_Character_219[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 0, 20, 8}; -static const GLubyte Helvetica10_Character_220[] = { 8, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 0, 36, 0}; -static const GLubyte Helvetica10_Character_221[] = { 7, 0, 0, 0, 16, 16, 16, 40, 40, 68, 68,130, 0, 16, 8}; -static const GLubyte Helvetica10_Character_222[] = { 7, 0, 0, 0, 64, 64,120, 68, 68,120, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica10_Character_223[] = { 5, 0, 0, 0,160,144,144,144,160,144,144, 96, 0, 0, 0}; -static const GLubyte Helvetica10_Character_224[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0, 32, 64, 0, 0}; -static const GLubyte Helvetica10_Character_225[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0, 32, 16, 0, 0}; -static const GLubyte Helvetica10_Character_226[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0, 80, 32, 0, 0}; -static const GLubyte Helvetica10_Character_227[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0,160, 80, 0, 0}; -static const GLubyte Helvetica10_Character_228[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 0, 80, 0, 0, 0}; -static const GLubyte Helvetica10_Character_229[] = { 5, 0, 0, 0,104,144,144,112, 16,224, 32, 80, 32, 0, 0}; -static const GLubyte Helvetica10_Character_230[] = { 8, 0, 0, 0,108,146,144,126, 18,236, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_231[] = { 5, 0, 96, 32, 96,144,128,128,144, 96, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_232[] = { 5, 0, 0, 0, 96,144,128,240,144, 96, 0, 32, 64, 0, 0}; -static const GLubyte Helvetica10_Character_233[] = { 5, 0, 0, 0, 96,144,128,240,144, 96, 0, 64, 32, 0, 0}; -static const GLubyte Helvetica10_Character_234[] = { 5, 0, 0, 0, 96,144,128,240,144, 96, 0, 80, 32, 0, 0}; -static const GLubyte Helvetica10_Character_235[] = { 5, 0, 0, 0, 96,144,128,240,144, 96, 0, 80, 0, 0, 0}; -static const GLubyte Helvetica10_Character_236[] = { 2, 0, 0, 0,128,128,128,128,128,128, 0,128, 0, 0, 0}; -static const GLubyte Helvetica10_Character_237[] = { 2, 0, 0, 0,128,128,128,128,128,128, 0,128, 64, 0, 0}; -static const GLubyte Helvetica10_Character_238[] = { 2, 0, 0, 0,128,128,128,128,128,128, 0, 64,128, 0, 0}; -static const GLubyte Helvetica10_Character_239[] = { 2, 0, 0, 0, 64, 64, 64, 64, 64, 64, 0,160, 0, 0, 0}; -static const GLubyte Helvetica10_Character_240[] = { 6, 0, 0, 0,112,136,136,136,136,120,144, 96, 80, 0, 0}; -static const GLubyte Helvetica10_Character_241[] = { 5, 0, 0, 0,144,144,144,144,144,224, 0,160, 80, 0, 0}; -static const GLubyte Helvetica10_Character_242[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 32, 64, 0, 0}; -static const GLubyte Helvetica10_Character_243[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 32, 16, 0, 0}; -static const GLubyte Helvetica10_Character_244[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 80, 32, 0, 0}; -static const GLubyte Helvetica10_Character_245[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 80, 40, 0, 0}; -static const GLubyte Helvetica10_Character_246[] = { 6, 0, 0, 0,112,136,136,136,136,112, 0, 80, 0, 0, 0}; -static const GLubyte Helvetica10_Character_247[] = { 6, 0, 0, 0, 0, 32, 0,248, 0, 32, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_248[] = { 6, 0, 0, 0,112,136,200,168,152,116, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica10_Character_249[] = { 5, 0, 0, 0,112,144,144,144,144,144, 0, 32, 64, 0, 0}; -static const GLubyte Helvetica10_Character_250[] = { 5, 0, 0, 0,112,144,144,144,144,144, 0, 64, 32, 0, 0}; -static const GLubyte Helvetica10_Character_251[] = { 5, 0, 0, 0,112,144,144,144,144,144, 0, 80, 32, 0, 0}; -static const GLubyte Helvetica10_Character_252[] = { 5, 0, 0, 0,112,144,144,144,144,144, 0, 80, 0, 0, 0}; -static const GLubyte Helvetica10_Character_253[] = { 5, 0,128, 64, 64, 96,160,160,144,144, 0, 32, 16, 0, 0}; -static const GLubyte Helvetica10_Character_254[] = { 6, 0,128,128,176,200,136,136,200,176,128,128, 0, 0, 0}; -static const GLubyte Helvetica10_Character_255[] = { 5, 0,128, 64, 64, 96,160,160,144,144, 0, 80, 0, 0, 0}; - -/* The font characters mapping: */ -static const GLubyte* Helvetica10_Character_Map[] = {Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032, - Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032, - Helvetica10_Character_032,Helvetica10_Character_033,Helvetica10_Character_034,Helvetica10_Character_035,Helvetica10_Character_036,Helvetica10_Character_037,Helvetica10_Character_038,Helvetica10_Character_039,Helvetica10_Character_040,Helvetica10_Character_041,Helvetica10_Character_042,Helvetica10_Character_043,Helvetica10_Character_044,Helvetica10_Character_045,Helvetica10_Character_046,Helvetica10_Character_047, - Helvetica10_Character_048,Helvetica10_Character_049,Helvetica10_Character_050,Helvetica10_Character_051,Helvetica10_Character_052,Helvetica10_Character_053,Helvetica10_Character_054,Helvetica10_Character_055,Helvetica10_Character_056,Helvetica10_Character_057,Helvetica10_Character_058,Helvetica10_Character_059,Helvetica10_Character_060,Helvetica10_Character_061,Helvetica10_Character_062,Helvetica10_Character_063, - Helvetica10_Character_064,Helvetica10_Character_065,Helvetica10_Character_066,Helvetica10_Character_067,Helvetica10_Character_068,Helvetica10_Character_069,Helvetica10_Character_070,Helvetica10_Character_071,Helvetica10_Character_072,Helvetica10_Character_073,Helvetica10_Character_074,Helvetica10_Character_075,Helvetica10_Character_076,Helvetica10_Character_077,Helvetica10_Character_078,Helvetica10_Character_079, - Helvetica10_Character_080,Helvetica10_Character_081,Helvetica10_Character_082,Helvetica10_Character_083,Helvetica10_Character_084,Helvetica10_Character_085,Helvetica10_Character_086,Helvetica10_Character_087,Helvetica10_Character_088,Helvetica10_Character_089,Helvetica10_Character_090,Helvetica10_Character_091,Helvetica10_Character_092,Helvetica10_Character_093,Helvetica10_Character_094,Helvetica10_Character_095, - Helvetica10_Character_096,Helvetica10_Character_097,Helvetica10_Character_098,Helvetica10_Character_099,Helvetica10_Character_100,Helvetica10_Character_101,Helvetica10_Character_102,Helvetica10_Character_103,Helvetica10_Character_104,Helvetica10_Character_105,Helvetica10_Character_106,Helvetica10_Character_107,Helvetica10_Character_108,Helvetica10_Character_109,Helvetica10_Character_110,Helvetica10_Character_111, - Helvetica10_Character_112,Helvetica10_Character_113,Helvetica10_Character_114,Helvetica10_Character_115,Helvetica10_Character_116,Helvetica10_Character_117,Helvetica10_Character_118,Helvetica10_Character_119,Helvetica10_Character_120,Helvetica10_Character_121,Helvetica10_Character_122,Helvetica10_Character_123,Helvetica10_Character_124,Helvetica10_Character_125,Helvetica10_Character_126,Helvetica10_Character_032, - Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032, - Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032,Helvetica10_Character_032, - Helvetica10_Character_160,Helvetica10_Character_161,Helvetica10_Character_162,Helvetica10_Character_163,Helvetica10_Character_164,Helvetica10_Character_165,Helvetica10_Character_166,Helvetica10_Character_167,Helvetica10_Character_168,Helvetica10_Character_169,Helvetica10_Character_170,Helvetica10_Character_171,Helvetica10_Character_172,Helvetica10_Character_173,Helvetica10_Character_174,Helvetica10_Character_175, - Helvetica10_Character_176,Helvetica10_Character_177,Helvetica10_Character_178,Helvetica10_Character_179,Helvetica10_Character_180,Helvetica10_Character_181,Helvetica10_Character_182,Helvetica10_Character_183,Helvetica10_Character_184,Helvetica10_Character_185,Helvetica10_Character_186,Helvetica10_Character_187,Helvetica10_Character_188,Helvetica10_Character_189,Helvetica10_Character_190,Helvetica10_Character_191, - Helvetica10_Character_192,Helvetica10_Character_193,Helvetica10_Character_194,Helvetica10_Character_195,Helvetica10_Character_196,Helvetica10_Character_197,Helvetica10_Character_198,Helvetica10_Character_199,Helvetica10_Character_200,Helvetica10_Character_201,Helvetica10_Character_202,Helvetica10_Character_203,Helvetica10_Character_204,Helvetica10_Character_205,Helvetica10_Character_206,Helvetica10_Character_207, - Helvetica10_Character_208,Helvetica10_Character_209,Helvetica10_Character_210,Helvetica10_Character_211,Helvetica10_Character_212,Helvetica10_Character_213,Helvetica10_Character_214,Helvetica10_Character_215,Helvetica10_Character_216,Helvetica10_Character_217,Helvetica10_Character_218,Helvetica10_Character_219,Helvetica10_Character_220,Helvetica10_Character_221,Helvetica10_Character_222,Helvetica10_Character_223, - Helvetica10_Character_224,Helvetica10_Character_225,Helvetica10_Character_226,Helvetica10_Character_227,Helvetica10_Character_228,Helvetica10_Character_229,Helvetica10_Character_230,Helvetica10_Character_231,Helvetica10_Character_232,Helvetica10_Character_233,Helvetica10_Character_234,Helvetica10_Character_235,Helvetica10_Character_236,Helvetica10_Character_237,Helvetica10_Character_238,Helvetica10_Character_239, - Helvetica10_Character_240,Helvetica10_Character_241,Helvetica10_Character_242,Helvetica10_Character_243,Helvetica10_Character_244,Helvetica10_Character_245,Helvetica10_Character_246,Helvetica10_Character_247,Helvetica10_Character_248,Helvetica10_Character_249,Helvetica10_Character_250,Helvetica10_Character_251,Helvetica10_Character_252,Helvetica10_Character_253,Helvetica10_Character_254,Helvetica10_Character_255,NULL}; - -/* The font structure: */ -const SFG_Font fgFontHelvetica10 = { "-adobe-helvetica-medium-r-normal--10-100-75-75-p-56-iso8859-1", 256, 14, Helvetica10_Character_Map, 0, 3 }; - -static const GLubyte Helvetica12_Character_000[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_001[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_002[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_003[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_004[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_005[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_006[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_007[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_008[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_009[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_010[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_011[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_012[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_013[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_014[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_015[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_016[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_017[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_018[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_019[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_020[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_021[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_022[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_023[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_024[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_025[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_026[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_027[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_028[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_029[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_030[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_031[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_032[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_033[] = { 3, 0, 0, 0, 0, 64, 0, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_034[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 80, 80, 80, 0, 0, 0}; -static const GLubyte Helvetica12_Character_035[] = { 7, 0, 0, 0, 0, 80, 80, 80,252, 40,252, 40, 40, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_036[] = { 7, 0, 0, 0, 16, 56, 84, 84, 20, 56, 80, 84, 56, 16, 0, 0, 0}; -static const GLubyte Helvetica12_Character_037[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 17,128, 10, 64, 10, 64, 9,128, 4, 0, 52, 0, 74, 0, 74, 0, 49, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_038[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 57, 0, 70, 0, 66, 0, 69, 0, 40, 0, 24, 0, 36, 0, 36, 0, 24, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_039[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 64, 32, 96, 0, 0, 0}; -static const GLubyte Helvetica12_Character_040[] = { 4, 0, 16, 32, 32, 64, 64, 64, 64, 64, 64, 32, 32, 16, 0, 0, 0}; -static const GLubyte Helvetica12_Character_041[] = { 4, 0,128, 64, 64, 32, 32, 32, 32, 32, 32, 64, 64,128, 0, 0, 0}; -static const GLubyte Helvetica12_Character_042[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 80, 32, 80, 0, 0, 0}; -static const GLubyte Helvetica12_Character_043[] = { 7, 0, 0, 0, 0, 0, 16, 16,124, 16, 16, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_044[] = { 4, 0, 0, 64, 32, 32, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_045[] = { 8, 0, 0, 0, 0, 0, 0, 0,124, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_046[] = { 3, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_047[] = { 4, 0, 0, 0, 0,128,128, 64, 64, 64, 32, 32, 16, 16, 0, 0, 0}; -static const GLubyte Helvetica12_Character_048[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 68, 68, 56, 0, 0, 0}; -static const GLubyte Helvetica12_Character_049[] = { 7, 0, 0, 0, 0, 16, 16, 16, 16, 16, 16, 16,112, 16, 0, 0, 0}; -static const GLubyte Helvetica12_Character_050[] = { 7, 0, 0, 0, 0,124, 64, 64, 32, 16, 8, 4, 68, 56, 0, 0, 0}; -static const GLubyte Helvetica12_Character_051[] = { 7, 0, 0, 0, 0, 56, 68, 68, 4, 4, 24, 4, 68, 56, 0, 0, 0}; -static const GLubyte Helvetica12_Character_052[] = { 7, 0, 0, 0, 0, 8, 8,252,136, 72, 40, 40, 24, 8, 0, 0, 0}; -static const GLubyte Helvetica12_Character_053[] = { 7, 0, 0, 0, 0, 56, 68, 68, 4, 4,120, 64, 64,124, 0, 0, 0}; -static const GLubyte Helvetica12_Character_054[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68,100, 88, 64, 68, 56, 0, 0, 0}; -static const GLubyte Helvetica12_Character_055[] = { 7, 0, 0, 0, 0, 32, 32, 16, 16, 16, 8, 8, 4,124, 0, 0, 0}; -static const GLubyte Helvetica12_Character_056[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 56, 68, 68, 56, 0, 0, 0}; -static const GLubyte Helvetica12_Character_057[] = { 7, 0, 0, 0, 0, 56, 68, 4, 4, 60, 68, 68, 68, 56, 0, 0, 0}; -static const GLubyte Helvetica12_Character_058[] = { 3, 0, 0, 0, 0, 64, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_059[] = { 3, 0, 0,128, 64, 64, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_060[] = { 7, 0, 0, 0, 0, 0, 12, 48,192, 48, 12, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_061[] = { 7, 0, 0, 0, 0, 0, 0,124, 0,124, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_062[] = { 7, 0, 0, 0, 0, 0, 96, 24, 6, 24, 96, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_063[] = { 7, 0, 0, 0, 0, 16, 0, 16, 16, 8, 8, 68, 68, 56, 0, 0, 0}; -static const GLubyte Helvetica12_Character_064[] = { 12, 0, 0, 0, 0, 0, 0, 31, 0, 32, 0, 77,128, 83, 64, 81, 32, 81, 32, 73, 32, 38,160, 48, 64, 15,128, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_065[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 20, 0, 8, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_066[] = { 8, 0, 0, 0, 0,124, 66, 66, 66,124, 66, 66, 66,124, 0, 0, 0}; -static const GLubyte Helvetica12_Character_067[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 33, 0, 30, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_068[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 66, 0, 65, 0, 65, 0, 65, 0, 65, 0, 65, 0, 66, 0,124, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_069[] = { 8, 0, 0, 0, 0,126, 64, 64, 64,126, 64, 64, 64,126, 0, 0, 0}; -static const GLubyte Helvetica12_Character_070[] = { 8, 0, 0, 0, 0, 64, 64, 64, 64,124, 64, 64, 64,126, 0, 0, 0}; -static const GLubyte Helvetica12_Character_071[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 29, 0, 35, 0, 65, 0, 65, 0, 71, 0, 64, 0, 64, 0, 33, 0, 30, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_072[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 65, 0,127, 0, 65, 0, 65, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_073[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_074[] = { 7, 0, 0, 0, 0, 56, 68, 68, 4, 4, 4, 4, 4, 4, 0, 0, 0}; -static const GLubyte Helvetica12_Character_075[] = { 8, 0, 0, 0, 0, 65, 66, 68, 72,112, 80, 72, 68, 66, 0, 0, 0}; -static const GLubyte Helvetica12_Character_076[] = { 7, 0, 0, 0, 0,124, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_077[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 68, 64, 68, 64, 74, 64, 74, 64, 81, 64, 81, 64, 96,192, 96,192, 64, 64, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_078[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 67, 0, 69, 0, 69, 0, 73, 0, 81, 0, 81, 0, 97, 0, 65, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_079[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_080[] = { 8, 0, 0, 0, 0, 64, 64, 64, 64,124, 66, 66, 66,124, 0, 0, 0}; -static const GLubyte Helvetica12_Character_081[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30,128, 33, 0, 66,128, 68,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_082[] = { 8, 0, 0, 0, 0, 66, 66, 66, 68,124, 66, 66, 66,124, 0, 0, 0}; -static const GLubyte Helvetica12_Character_083[] = { 8, 0, 0, 0, 0, 60, 66, 66, 2, 12, 48, 64, 66, 60, 0, 0, 0}; -static const GLubyte Helvetica12_Character_084[] = { 7, 0, 0, 0, 0, 16, 16, 16, 16, 16, 16, 16, 16,254, 0, 0, 0}; -static const GLubyte Helvetica12_Character_085[] = { 8, 0, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 0, 0}; -static const GLubyte Helvetica12_Character_086[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 20, 0, 20, 0, 34, 0, 34, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_087[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 17, 0, 17, 0, 17, 0, 42,128, 42,128, 36,128, 68, 64, 68, 64, 68, 64, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_088[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 34, 0, 34, 0, 20, 0, 8, 0, 20, 0, 34, 0, 34, 0, 65, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_089[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 20, 0, 34, 0, 34, 0, 65, 0, 65, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_090[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,127, 0, 64, 0, 32, 0, 16, 0, 8, 0, 4, 0, 2, 0, 1, 0,127, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_091[] = { 3, 0, 96, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 96, 0, 0, 0}; -static const GLubyte Helvetica12_Character_092[] = { 4, 0, 0, 0, 0, 16, 16, 32, 32, 32, 64, 64,128,128, 0, 0, 0}; -static const GLubyte Helvetica12_Character_093[] = { 3, 0,192, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64,192, 0, 0, 0}; -static const GLubyte Helvetica12_Character_094[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0,136, 80, 32, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_095[] = { 7, 0, 0,254, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_096[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192,128, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_097[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_098[] = { 7, 0, 0, 0, 0, 88,100, 68, 68, 68,100, 88, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_099[] = { 7, 0, 0, 0, 0, 56, 68, 64, 64, 64, 68, 56, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_100[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 76, 52, 4, 4, 0, 0, 0}; -static const GLubyte Helvetica12_Character_101[] = { 7, 0, 0, 0, 0, 56, 68, 64,124, 68, 68, 56, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_102[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64,224, 64, 48, 0, 0, 0}; -static const GLubyte Helvetica12_Character_103[] = { 7, 0, 56, 68, 4, 52, 76, 68, 68, 68, 76, 52, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_104[] = { 7, 0, 0, 0, 0, 68, 68, 68, 68, 68,100, 88, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_105[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 0, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_106[] = { 3, 0,128, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_107[] = { 6, 0, 0, 0, 0, 68, 72, 80, 96, 96, 80, 72, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_108[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_109[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 73, 0, 73, 0, 73, 0, 73, 0, 73, 0,109, 0, 82, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_110[] = { 7, 0, 0, 0, 0, 68, 68, 68, 68, 68,100, 88, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_111[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_112[] = { 7, 0, 64, 64, 64, 88,100, 68, 68, 68,100, 88, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_113[] = { 7, 0, 4, 4, 4, 52, 76, 68, 68, 68, 76, 52, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_114[] = { 4, 0, 0, 0, 0, 64, 64, 64, 64, 64, 96, 80, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_115[] = { 6, 0, 0, 0, 0, 48, 72, 8, 48, 64, 72, 48, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_116[] = { 3, 0, 0, 0, 0, 96, 64, 64, 64, 64, 64,224, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_117[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 68, 68, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_118[] = { 7, 0, 0, 0, 0, 16, 16, 40, 40, 68, 68, 68, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_119[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 34, 0, 85, 0, 73, 0, 73, 0,136,128,136,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_120[] = { 6, 0, 0, 0, 0,132,132, 72, 48, 48, 72,132, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_121[] = { 7, 0, 64, 32, 16, 16, 40, 40, 72, 68, 68, 68, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_122[] = { 6, 0, 0, 0, 0,120, 64, 32, 32, 16, 8,120, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_123[] = { 4, 0, 48, 64, 64, 64, 64, 64,128, 64, 64, 64, 64, 48, 0, 0, 0}; -static const GLubyte Helvetica12_Character_124[] = { 3, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_125[] = { 4, 0,192, 32, 32, 32, 32, 32, 16, 32, 32, 32, 32,192, 0, 0, 0}; -static const GLubyte Helvetica12_Character_126[] = { 7, 0, 0, 0, 0, 0, 0, 0,152,100, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_127[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_128[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_129[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_130[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_131[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_132[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_133[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_134[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_135[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_136[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_137[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_138[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_139[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_140[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_141[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_142[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_143[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_144[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_145[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_146[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_147[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_148[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_149[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_150[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_151[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_152[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_153[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_154[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_155[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_156[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_157[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_158[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_159[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 85, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 65, 0, 0, 0, 85, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_160[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_161[] = { 3, 0, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_162[] = { 7, 0, 0, 0, 32, 56,100, 80, 80, 80, 84, 56, 8, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_163[] = { 7, 0, 0, 0, 0, 88, 36, 16, 16,120, 32, 32, 36, 24, 0, 0, 0}; -static const GLubyte Helvetica12_Character_164[] = { 7, 0, 0, 0, 0, 0,132,120, 72, 72,120,132, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_165[] = { 7, 0, 0, 0, 0, 16, 16,124, 16,124, 16, 40, 68, 68, 0, 0, 0}; -static const GLubyte Helvetica12_Character_166[] = { 3, 0, 0, 64, 64, 64, 64, 0, 0, 0, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_167[] = { 6, 0,112,136, 8, 48, 72,136,136,144, 96,128,136,112, 0, 0, 0}; -static const GLubyte Helvetica12_Character_168[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,160, 0, 0, 0}; -static const GLubyte Helvetica12_Character_169[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 32,128, 78, 64, 81, 64, 80, 64, 81, 64, 78, 64, 32,128, 31, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_170[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0,112, 0, 80, 16,112, 0, 0, 0}; -static const GLubyte Helvetica12_Character_171[] = { 7, 0, 0, 0, 0, 0, 20, 40, 80, 40, 20, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_172[] = { 8, 0, 0, 0, 0, 0, 0, 2, 2, 2,126, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_173[] = { 5, 0, 0, 0, 0, 0, 0, 0,240, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_174[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 32,128, 74, 64, 74, 64, 76, 64, 74, 64, 78, 64, 32,128, 31, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_175[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,240, 0, 0, 0}; -static const GLubyte Helvetica12_Character_176[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 96,144,144, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_177[] = { 7, 0, 0, 0, 0,124, 0, 16, 16,124, 16, 16, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_178[] = { 4, 0, 0, 0, 0, 0, 0, 0,240, 64, 32,144, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_179[] = { 4, 0, 0, 0, 0, 0, 0, 0,192, 32, 64, 32,224, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_180[] = { 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,128, 64, 0, 0}; -static const GLubyte Helvetica12_Character_181[] = { 7, 0, 64, 64, 64,116, 76, 68, 68, 68, 68, 68, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_182[] = { 7, 0, 40, 40, 40, 40, 40, 40,104,232,232,232,104, 60, 0, 0, 0}; -static const GLubyte Helvetica12_Character_183[] = { 3, 0, 0, 0, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_184[] = { 3, 0,192, 32, 32, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_185[] = { 4, 0, 0, 0, 0, 0, 0, 0, 32, 32, 32, 96, 32, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_186[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0,112, 0,112, 80,112, 0, 0, 0}; -static const GLubyte Helvetica12_Character_187[] = { 7, 0, 0, 0, 0, 0, 80, 40, 20, 40, 80, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_188[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 39,128, 21, 0, 19, 0, 73, 0, 68, 0, 68, 0,194, 0, 65, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_189[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 71,128, 34, 0, 17, 0, 20,128, 75, 0, 72, 0, 68, 0,194, 0, 65, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_190[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 33, 0, 23,128, 21, 0, 11, 0,201, 0, 36, 0, 68, 0, 34, 0,225, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_191[] = { 7, 0, 56, 68, 68, 32, 32, 16, 16, 0, 16, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_192[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 0, 0, 8, 0, 16, 0}; -static const GLubyte Helvetica12_Character_193[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 0, 0, 8, 0, 4, 0}; -static const GLubyte Helvetica12_Character_194[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 0, 0, 20, 0, 8, 0}; -static const GLubyte Helvetica12_Character_195[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 0, 0, 20, 0, 10, 0}; -static const GLubyte Helvetica12_Character_196[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 0, 0, 20, 0, 0, 0}; -static const GLubyte Helvetica12_Character_197[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 65, 0, 65, 0, 62, 0, 34, 0, 34, 0, 20, 0, 8, 0, 8, 0, 8, 0, 20, 0, 8, 0}; -static const GLubyte Helvetica12_Character_198[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 71,192, 68, 0, 68, 0, 60, 0, 39,192, 36, 0, 20, 0, 20, 0, 15,192, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_199[] = { 9, 0, 0, 24, 0, 4, 0, 4, 0, 30, 0, 33, 0, 64, 0, 64, 0, 64, 0, 64, 0, 64, 0, 33, 0, 30, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_200[] = { 8, 0, 0, 0, 0,126, 64, 64, 64,126, 64, 64, 64,126, 0, 8, 16}; -static const GLubyte Helvetica12_Character_201[] = { 8, 0, 0, 0, 0,126, 64, 64, 64,126, 64, 64, 64,126, 0, 8, 4}; -static const GLubyte Helvetica12_Character_202[] = { 8, 0, 0, 0, 0,126, 64, 64, 64,126, 64, 64, 64,126, 0, 20, 8}; -static const GLubyte Helvetica12_Character_203[] = { 8, 0, 0, 0, 0,126, 64, 64, 64,126, 64, 64, 64,126, 0, 20, 0}; -static const GLubyte Helvetica12_Character_204[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64,128}; -static const GLubyte Helvetica12_Character_205[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0, 64, 32}; -static const GLubyte Helvetica12_Character_206[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0,160, 64}; -static const GLubyte Helvetica12_Character_207[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 64, 64, 0,160, 0}; -static const GLubyte Helvetica12_Character_208[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 66, 0, 65, 0, 65, 0,241, 0, 65, 0, 65, 0, 66, 0,124, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_209[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 67, 0, 69, 0, 69, 0, 73, 0, 81, 0, 81, 0, 97, 0, 65, 0, 0, 0, 20, 0, 10, 0}; -static const GLubyte Helvetica12_Character_210[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 4, 0, 8, 0}; -static const GLubyte Helvetica12_Character_211[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 4, 0, 2, 0}; -static const GLubyte Helvetica12_Character_212[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 10, 0, 4, 0}; -static const GLubyte Helvetica12_Character_213[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 20, 0, 10, 0}; -static const GLubyte Helvetica12_Character_214[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 33, 0, 64,128, 64,128, 64,128, 64,128, 64,128, 33, 0, 30, 0, 0, 0, 18, 0, 0, 0}; -static const GLubyte Helvetica12_Character_215[] = { 7, 0, 0, 0, 0, 0, 68, 40, 16, 40, 68, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_216[] = { 10, 0, 0, 0, 0, 0, 0,128, 0, 94, 0, 33, 0, 80,128, 72,128, 68,128, 68,128, 66,128, 33, 0, 30,128, 0, 64, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_217[] = { 8, 0, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 8, 16}; -static const GLubyte Helvetica12_Character_218[] = { 8, 0, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 8, 4}; -static const GLubyte Helvetica12_Character_219[] = { 8, 0, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 20, 8}; -static const GLubyte Helvetica12_Character_220[] = { 8, 0, 0, 0, 0, 60, 66, 66, 66, 66, 66, 66, 66, 66, 0, 36, 0}; -static const GLubyte Helvetica12_Character_221[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 8, 0, 8, 0, 20, 0, 34, 0, 34, 0, 65, 0, 65, 0, 0, 0, 8, 0, 4, 0}; -static const GLubyte Helvetica12_Character_222[] = { 8, 0, 0, 0, 0, 64, 64,124, 66, 66, 66,124, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_223[] = { 7, 0, 0, 0, 0, 88, 68, 68, 68, 88, 68, 68, 68, 56, 0, 0, 0}; -static const GLubyte Helvetica12_Character_224[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 8, 16, 0, 0}; -static const GLubyte Helvetica12_Character_225[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 16, 8, 0, 0}; -static const GLubyte Helvetica12_Character_226[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 40, 16, 0, 0}; -static const GLubyte Helvetica12_Character_227[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 40, 20, 0, 0}; -static const GLubyte Helvetica12_Character_228[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 0, 40, 0, 0, 0}; -static const GLubyte Helvetica12_Character_229[] = { 7, 0, 0, 0, 0, 58, 68, 68, 60, 4, 68, 56, 24, 36, 24, 0, 0}; -static const GLubyte Helvetica12_Character_230[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 59,128, 68, 64, 68, 0, 63,192, 4, 64, 68, 64, 59,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_231[] = { 7, 0, 48, 8, 16, 56, 68, 64, 64, 64, 68, 56, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_232[] = { 7, 0, 0, 0, 0, 56, 68, 64,124, 68, 68, 56, 0, 16, 32, 0, 0}; -static const GLubyte Helvetica12_Character_233[] = { 7, 0, 0, 0, 0, 56, 68, 64,124, 68, 68, 56, 0, 16, 8, 0, 0}; -static const GLubyte Helvetica12_Character_234[] = { 7, 0, 0, 0, 0, 56, 68, 64,124, 68, 68, 56, 0, 40, 16, 0, 0}; -static const GLubyte Helvetica12_Character_235[] = { 7, 0, 0, 0, 0, 56, 68, 64,124, 68, 68, 56, 0, 40, 0, 0, 0}; -static const GLubyte Helvetica12_Character_236[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 0, 64,128, 0, 0}; -static const GLubyte Helvetica12_Character_237[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 0, 64, 32, 0, 0}; -static const GLubyte Helvetica12_Character_238[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 0,160, 64, 0, 0}; -static const GLubyte Helvetica12_Character_239[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64, 64, 64, 64, 0,160, 0, 0, 0}; -static const GLubyte Helvetica12_Character_240[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 60, 4, 40, 24, 52, 0, 0}; -static const GLubyte Helvetica12_Character_241[] = { 7, 0, 0, 0, 0, 68, 68, 68, 68, 68,100, 88, 0, 40, 20, 0, 0}; -static const GLubyte Helvetica12_Character_242[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 16, 32, 0, 0}; -static const GLubyte Helvetica12_Character_243[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 16, 8, 0, 0}; -static const GLubyte Helvetica12_Character_244[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 40, 16, 0, 0}; -static const GLubyte Helvetica12_Character_245[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 40, 20, 0, 0}; -static const GLubyte Helvetica12_Character_246[] = { 7, 0, 0, 0, 0, 56, 68, 68, 68, 68, 68, 56, 0, 40, 0, 0, 0}; -static const GLubyte Helvetica12_Character_247[] = { 7, 0, 0, 0, 0, 0, 16, 0,124, 0, 16, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_248[] = { 7, 0, 0, 0, 0,184, 68,100, 84, 76, 68, 58, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica12_Character_249[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 68, 68, 0, 16, 32, 0, 0}; -static const GLubyte Helvetica12_Character_250[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 68, 68, 0, 16, 8, 0, 0}; -static const GLubyte Helvetica12_Character_251[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 68, 68, 0, 40, 16, 0, 0}; -static const GLubyte Helvetica12_Character_252[] = { 7, 0, 0, 0, 0, 52, 76, 68, 68, 68, 68, 68, 0, 40, 0, 0, 0}; -static const GLubyte Helvetica12_Character_253[] = { 7, 0, 64, 32, 16, 16, 40, 40, 72, 68, 68, 68, 0, 16, 8, 0, 0}; -static const GLubyte Helvetica12_Character_254[] = { 7, 0, 64, 64, 64, 88,100, 68, 68, 68,100, 88, 64, 64, 0, 0, 0}; -static const GLubyte Helvetica12_Character_255[] = { 7, 0, 96, 16, 16, 16, 24, 40, 40, 36, 68, 68, 0, 40, 0, 0, 0}; - -/* The font characters mapping: */ -static const GLubyte* Helvetica12_Character_Map[] = {Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032, - Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032, - Helvetica12_Character_032,Helvetica12_Character_033,Helvetica12_Character_034,Helvetica12_Character_035,Helvetica12_Character_036,Helvetica12_Character_037,Helvetica12_Character_038,Helvetica12_Character_039,Helvetica12_Character_040,Helvetica12_Character_041,Helvetica12_Character_042,Helvetica12_Character_043,Helvetica12_Character_044,Helvetica12_Character_045,Helvetica12_Character_046,Helvetica12_Character_047, - Helvetica12_Character_048,Helvetica12_Character_049,Helvetica12_Character_050,Helvetica12_Character_051,Helvetica12_Character_052,Helvetica12_Character_053,Helvetica12_Character_054,Helvetica12_Character_055,Helvetica12_Character_056,Helvetica12_Character_057,Helvetica12_Character_058,Helvetica12_Character_059,Helvetica12_Character_060,Helvetica12_Character_061,Helvetica12_Character_062,Helvetica12_Character_063, - Helvetica12_Character_064,Helvetica12_Character_065,Helvetica12_Character_066,Helvetica12_Character_067,Helvetica12_Character_068,Helvetica12_Character_069,Helvetica12_Character_070,Helvetica12_Character_071,Helvetica12_Character_072,Helvetica12_Character_073,Helvetica12_Character_074,Helvetica12_Character_075,Helvetica12_Character_076,Helvetica12_Character_077,Helvetica12_Character_078,Helvetica12_Character_079, - Helvetica12_Character_080,Helvetica12_Character_081,Helvetica12_Character_082,Helvetica12_Character_083,Helvetica12_Character_084,Helvetica12_Character_085,Helvetica12_Character_086,Helvetica12_Character_087,Helvetica12_Character_088,Helvetica12_Character_089,Helvetica12_Character_090,Helvetica12_Character_091,Helvetica12_Character_092,Helvetica12_Character_093,Helvetica12_Character_094,Helvetica12_Character_095, - Helvetica12_Character_096,Helvetica12_Character_097,Helvetica12_Character_098,Helvetica12_Character_099,Helvetica12_Character_100,Helvetica12_Character_101,Helvetica12_Character_102,Helvetica12_Character_103,Helvetica12_Character_104,Helvetica12_Character_105,Helvetica12_Character_106,Helvetica12_Character_107,Helvetica12_Character_108,Helvetica12_Character_109,Helvetica12_Character_110,Helvetica12_Character_111, - Helvetica12_Character_112,Helvetica12_Character_113,Helvetica12_Character_114,Helvetica12_Character_115,Helvetica12_Character_116,Helvetica12_Character_117,Helvetica12_Character_118,Helvetica12_Character_119,Helvetica12_Character_120,Helvetica12_Character_121,Helvetica12_Character_122,Helvetica12_Character_123,Helvetica12_Character_124,Helvetica12_Character_125,Helvetica12_Character_126,Helvetica12_Character_032, - Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032, - Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032,Helvetica12_Character_032, - Helvetica12_Character_160,Helvetica12_Character_161,Helvetica12_Character_162,Helvetica12_Character_163,Helvetica12_Character_164,Helvetica12_Character_165,Helvetica12_Character_166,Helvetica12_Character_167,Helvetica12_Character_168,Helvetica12_Character_169,Helvetica12_Character_170,Helvetica12_Character_171,Helvetica12_Character_172,Helvetica12_Character_173,Helvetica12_Character_174,Helvetica12_Character_175, - Helvetica12_Character_176,Helvetica12_Character_177,Helvetica12_Character_178,Helvetica12_Character_179,Helvetica12_Character_180,Helvetica12_Character_181,Helvetica12_Character_182,Helvetica12_Character_183,Helvetica12_Character_184,Helvetica12_Character_185,Helvetica12_Character_186,Helvetica12_Character_187,Helvetica12_Character_188,Helvetica12_Character_189,Helvetica12_Character_190,Helvetica12_Character_191, - Helvetica12_Character_192,Helvetica12_Character_193,Helvetica12_Character_194,Helvetica12_Character_195,Helvetica12_Character_196,Helvetica12_Character_197,Helvetica12_Character_198,Helvetica12_Character_199,Helvetica12_Character_200,Helvetica12_Character_201,Helvetica12_Character_202,Helvetica12_Character_203,Helvetica12_Character_204,Helvetica12_Character_205,Helvetica12_Character_206,Helvetica12_Character_207, - Helvetica12_Character_208,Helvetica12_Character_209,Helvetica12_Character_210,Helvetica12_Character_211,Helvetica12_Character_212,Helvetica12_Character_213,Helvetica12_Character_214,Helvetica12_Character_215,Helvetica12_Character_216,Helvetica12_Character_217,Helvetica12_Character_218,Helvetica12_Character_219,Helvetica12_Character_220,Helvetica12_Character_221,Helvetica12_Character_222,Helvetica12_Character_223, - Helvetica12_Character_224,Helvetica12_Character_225,Helvetica12_Character_226,Helvetica12_Character_227,Helvetica12_Character_228,Helvetica12_Character_229,Helvetica12_Character_230,Helvetica12_Character_231,Helvetica12_Character_232,Helvetica12_Character_233,Helvetica12_Character_234,Helvetica12_Character_235,Helvetica12_Character_236,Helvetica12_Character_237,Helvetica12_Character_238,Helvetica12_Character_239, - Helvetica12_Character_240,Helvetica12_Character_241,Helvetica12_Character_242,Helvetica12_Character_243,Helvetica12_Character_244,Helvetica12_Character_245,Helvetica12_Character_246,Helvetica12_Character_247,Helvetica12_Character_248,Helvetica12_Character_249,Helvetica12_Character_250,Helvetica12_Character_251,Helvetica12_Character_252,Helvetica12_Character_253,Helvetica12_Character_254,Helvetica12_Character_255,NULL}; - -/* The font structure: */ -const SFG_Font fgFontHelvetica12 = { "-adobe-helvetica-medium-r-normal--12-120-75-75-p-67-iso8859-1", 256, 16, Helvetica12_Character_Map, 0, 4 }; - -static const GLubyte Helvetica18_Character_000[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 64, 16, 0, 0, 64, 16, 0, 0, 64, 16, 0, 0, 64, 16, 0, 0, 64, 16, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_001[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_002[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_003[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_004[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_005[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_006[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_007[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_008[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_009[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_010[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_011[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_012[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_013[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_014[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_015[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_016[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_017[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_018[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_019[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_020[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_021[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_022[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_023[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_024[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_025[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_026[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_027[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_028[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_029[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_030[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_031[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_032[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_033[] = { 6, 0, 0, 0, 0, 0, 48, 48, 0, 0, 32, 32, 48, 48, 48, 48, 48, 48, 48, 48, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_034[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,144,144,216,216,216, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_035[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 36, 0, 36, 0, 36, 0,255,128,255,128, 18, 0, 18, 0, 18, 0,127,192,127,192, 9, 0, 9, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_036[] = { 10, 0, 0, 0, 0, 0, 0, 4, 0, 4, 0, 31, 0, 63,128,117,192,100,192, 4,192, 7,128, 31, 0, 60, 0,116, 0,100, 0,101,128, 63,128, 31, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_037[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 60, 12,126, 6,102, 6,102, 3,126, 3, 60, 1,128, 61,128,126,192,102,192,102, 96,126, 96, 60, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_038[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 56, 63,112,115,224, 97,192, 97,224, 99, 96,119, 96, 62, 0, 30, 0, 51, 0, 51, 0, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_039[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 64, 32, 32, 96, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_040[] = { 6, 0, 8, 24, 48, 48, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48, 48, 24, 8, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_041[] = { 6, 0, 64, 96, 48, 48, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 48, 48, 96, 64, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_042[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 68, 56, 56,124, 16, 16, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_043[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 12, 0, 12, 0,127,128,127,128, 12, 0, 12, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_044[] = { 5, 0, 0, 64, 32, 32, 96, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_045[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_046[] = { 5, 0, 0, 0, 0, 0, 96, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_047[] = { 5, 0, 0, 0, 0, 0,192,192, 64, 64, 96, 96, 32, 32, 48, 48, 16, 16, 24, 24, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_048[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63, 0, 51, 0, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 51, 0, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_049[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 62, 0, 62, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_050[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,128, 96, 0,112, 0, 56, 0, 28, 0, 14, 0, 7, 0, 3,128, 1,128, 97,128,127, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_051[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63, 0, 99,128, 97,128, 1,128, 3,128, 15, 0, 14, 0, 3, 0, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_052[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,128, 1,128, 1,128,127,192,127,192, 97,128, 49,128, 25,128, 25,128, 13,128, 7,128, 3,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_053[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0,127, 0, 99,128, 97,128, 1,128, 1,128, 99,128,127, 0,126, 0, 96, 0, 96, 0,127, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_054[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63, 0,113,128, 97,128, 97,128, 97,128,127, 0,110, 0, 96, 0, 96, 0, 49,128, 63,128, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_055[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 0, 48, 0, 24, 0, 24, 0, 24, 0, 12, 0, 12, 0, 6, 0, 6, 0, 3, 0, 1,128,127,128,127,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_056[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63, 0,115,128, 97,128, 97,128, 51, 0, 63, 0, 51, 0, 97,128, 97,128,115,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_057[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0,127, 0, 99, 0, 1,128, 1,128, 29,128, 63,128, 97,128, 97,128, 97,128, 99,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_058[] = { 5, 0, 0, 0, 0, 0, 96, 96, 0, 0, 0, 0, 0, 0, 96, 96, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_059[] = { 5, 0, 0, 64, 32, 32, 96, 96, 0, 0, 0, 0, 0, 0, 96, 96, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_060[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,128, 7,128, 30, 0, 56, 0, 96, 0, 56, 0, 30, 0, 7,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_061[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 63,128, 63,128, 0, 0, 0, 0, 63,128, 63,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_062[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0,120, 0, 30, 0, 7, 0, 1,128, 7, 0, 30, 0,120, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_063[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 0, 24, 0, 0, 0, 0, 0, 24, 0, 24, 0, 24, 0, 28, 0, 14, 0, 7, 0, 99, 0, 99, 0,127, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_064[] = { 18, 0, 0, 0, 0, 0, 0, 3,240, 0, 15,248, 0, 28, 0, 0, 56, 0, 0, 51,184, 0,103,252, 0,102,102, 0,102, 51, 0,102, 51, 0,102, 49,128, 99, 25,128, 51,185,128, 49,217,128, 24, 3, 0, 14, 7, 0, 7,254, 0, 1,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_065[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_066[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,224, 96,112, 96, 48, 96, 48, 96,112,127,224,127,192, 96,192, 96, 96, 96, 96, 96,224,127,192,127,128, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_067[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 0, 96, 0, 96, 0, 96, 0, 96, 0,112, 0, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_068[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,192, 96,224, 96, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 96, 96,224,127,192,127,128, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_069[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,192, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_070[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_071[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,216, 31,248, 56, 56, 48, 24,112, 24, 96,248, 96,248, 96, 0, 96, 0,112, 24, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_072[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48,127,240,127,240, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_073[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_074[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63, 0,115,128, 97,128, 97,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_075[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 56, 96,112, 96,224, 97,192, 99,128,103, 0,126, 0,124, 0,110, 0,103, 0, 99,128, 97,192, 96,224, 96,112, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_076[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_077[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,134, 97,134, 99,198, 98, 70,102,102,102,102,108, 54,108, 54,120, 30,120, 30,112, 14,112, 14, 96, 6, 96, 6, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_078[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 48, 96,112, 96,240, 96,240, 97,176, 99, 48, 99, 48,102, 48,102, 48,108, 48,120, 48,120, 48,112, 48, 96, 48, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_079[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_080[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,192, 96,224, 96, 96, 96, 96, 96,224,127,192,127,128, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_081[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 7,216, 31,240, 56,120, 48,216,112,220, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_082[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96,192, 96,192,127,128,127,192, 96,224, 96, 96, 96, 96, 96,224,127,192,127,128, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_083[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31,128, 63,224,112,112, 96, 48, 0, 48, 0,112, 1,224, 15,128, 62, 0,112, 0, 96, 48,112,112, 63,224, 15,128, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_084[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0,127,224,127,224, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_085[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 63,224, 48, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_086[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 7,128, 7,128, 12,192, 12,192, 12,192, 24, 96, 24, 96, 24, 96, 48, 48, 48, 48, 48, 48, 96, 24, 96, 24, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_087[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 12, 0, 12, 12, 0, 14, 28, 0, 26, 22, 0, 27, 54, 0, 27, 54, 0, 51, 51, 0, 51, 51, 0, 49, 35, 0, 49,227, 0, 97,225,128, 96,193,128, 96,193,128, 96,193,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_088[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 48,112,112, 48, 96, 56,224, 24,192, 13,128, 7, 0, 7, 0, 13,128, 24,192, 56,224, 48, 96,112,112, 96, 48, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_089[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 7,128, 12,192, 24, 96, 24, 96, 48, 48, 48, 48, 96, 24, 96, 24, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_090[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,224,127,224, 96, 0, 48, 0, 24, 0, 12, 0, 14, 0, 6, 0, 3, 0, 1,128, 0,192, 0, 96,127,224,127,224, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_091[] = { 5, 0,120,120, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96,120,120, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_092[] = { 5, 0, 0, 0, 0, 0, 24, 24, 16, 16, 48, 48, 32, 32, 96, 96, 64, 64,192,192, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_093[] = { 5, 0,240,240, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,240,240, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_094[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 65, 0, 99, 0, 54, 0, 28, 0, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_095[] = { 10, 0, 0,255,192,255,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_096[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 96, 64, 64, 32, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_097[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_098[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,111, 0,127,128,113,128, 96,192, 96,192, 96,192, 96,192,113,128,127,128,111, 0, 96, 0, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_099[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96, 0, 96, 0, 96, 0, 96, 0, 49,128, 63,128, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_100[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30,192, 63,192, 49,192, 96,192, 96,192, 96,192, 96,192, 49,192, 63,192, 30,192, 0,192, 0,192, 0,192, 0,192, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_101[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63,128,113,128, 96, 0, 96, 0,127,128, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_102[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48,252,252, 48, 48, 60, 28, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_103[] = { 11, 0, 0, 14, 0, 63,128, 49,128, 0,192, 30,192, 63,192, 49,192, 96,192, 96,192, 96,192, 96,192, 48,192, 63,192, 30,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_104[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128,113,128,111,128,103, 0, 96, 0, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_105[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0, 0, 96, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_106[] = { 4, 0,192,224, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0, 0, 96, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_107[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 99,128, 99, 0,103, 0,102, 0,108, 0,124, 0,120, 0,108, 0,102, 0, 99, 0, 96, 0, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_108[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_109[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 99, 24, 99, 24, 99, 24, 99, 24, 99, 24, 99, 24, 99, 24,115,152,111,120,102, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_110[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128,113,128,111,128,103, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_111[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_112[] = { 11, 0, 0, 96, 0, 96, 0, 96, 0, 96, 0,111, 0,127,128,113,128, 96,192, 96,192, 96,192, 96,192,113,128,127,128,111, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_113[] = { 11, 0, 0, 0,192, 0,192, 0,192, 0,192, 30,192, 63,192, 49,192, 96,192, 96,192, 96,192, 96,192, 49,192, 63,192, 30,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_114[] = { 6, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96,112,108,108, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_115[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0,126, 0, 99, 0, 3, 0, 31, 0,126, 0, 96, 0, 99, 0, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_116[] = { 6, 0, 0, 0, 0, 0, 24, 56, 48, 48, 48, 48, 48, 48,252,252, 48, 48, 48, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_117[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 57,128,125,128, 99,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_118[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 30, 0, 18, 0, 51, 0, 51, 0, 51, 0, 97,128, 97,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_119[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12,192, 12,192, 28,224, 20,160, 52,176, 51, 48, 51, 48, 99, 24, 99, 24, 99, 24, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_120[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,128,115,128, 51, 0, 30, 0, 12, 0, 12, 0, 30, 0, 51, 0,115,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_121[] = { 10, 0, 0, 56, 0, 56, 0, 12, 0, 12, 0, 12, 0, 12, 0, 30, 0, 18, 0, 51, 0, 51, 0, 51, 0, 97,128, 97,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_122[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0,127, 0, 96, 0, 48, 0, 24, 0, 12, 0, 6, 0, 3, 0,127, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_123[] = { 6, 0, 12, 24, 48, 48, 48, 48, 48, 48, 96,192, 96, 48, 48, 48, 48, 48, 24, 12, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_124[] = { 4, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_125[] = { 6, 0,192, 96, 48, 48, 48, 48, 48, 48, 24, 12, 24, 48, 48, 48, 48, 48, 96,192, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_126[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,102, 0, 63, 0, 25,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_127[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_128[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_129[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_130[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_131[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_132[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_133[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_134[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_135[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_136[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_137[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_138[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_139[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_140[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_141[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_142[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_143[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_144[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_145[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_146[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_147[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_148[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_149[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_150[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_151[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_152[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_153[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_154[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_155[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_156[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_157[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_158[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_159[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 64, 16, 0, 0, 0, 0, 85, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_160[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_161[] = { 6, 0, 48, 48, 48, 48, 48, 48, 48, 48, 16, 16, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_162[] = { 10, 0, 0, 0, 0, 0, 0, 8, 0, 8, 0, 31, 0, 63,128, 53,128,100, 0,100, 0,100, 0,100, 0, 53,128, 63,128, 31, 0, 2, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_163[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,223, 0,255,128, 96,128, 48, 0, 24, 0, 24, 0,126, 0, 48, 0, 96, 0, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_164[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,128,127,128, 51, 0, 51, 0, 51, 0,127,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_165[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 12, 0, 12, 0,127,128, 12, 0,127,128, 30, 0, 51, 0, 51, 0, 51, 0, 97,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_166[] = { 4, 0, 0, 96, 96, 96, 96, 96, 96, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_167[] = { 10, 0, 0, 30, 0, 63, 0, 97,128, 97,128, 3,128, 7, 0, 31, 0, 57,128,113,128, 97,128, 99,128, 55, 0, 62, 0,120, 0, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_168[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,216,216, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_169[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 24, 48, 32, 8, 35,136, 68, 68, 72, 4, 72, 4, 72, 4, 68, 68, 35,136, 32, 8, 24, 48, 7,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_170[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 52,108, 36, 28,100, 56, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_171[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 0, 27, 0, 54, 0,108, 0,108, 0, 54, 0, 27, 0, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_172[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 0,192, 0,192,127,192,127,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_173[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0,124,124, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_174[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 24, 48, 32, 8, 36, 40, 68, 68, 68,132, 71,196, 68, 36, 68, 36, 39,200, 32, 8, 24, 48, 7,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_175[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,248, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_176[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,108, 68,108, 56, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_177[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,128, 0, 0, 12, 0, 12, 0, 12, 0,127,128,127,128, 12, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_178[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,248,248, 96, 48, 24,152,248,112, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_179[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,112,248,152, 48, 48,152,248,112, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_180[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 96, 48, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_181[] = { 10, 0, 0, 96, 0, 96, 0, 96, 0, 96, 0,109,128,127,128,115,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_182[] = { 10, 0, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 9, 0, 25, 0, 57, 0,121, 0,121, 0,121, 0,121, 0, 57, 0, 31,128, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_183[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_184[] = { 5, 0,240,216, 24,112, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_185[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48,112,112, 48, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_186[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 56,108, 68, 68,108, 56, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_187[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 72, 0,108, 0, 54, 0, 27, 0, 27, 0, 54, 0,108, 0, 72, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_188[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 24, 24, 24, 12,252, 6,216, 6,120, 51, 56, 49, 24, 49,136, 48,192, 48,192,112, 96,112, 48, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_189[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24,124, 24,124, 12, 48, 6, 24, 6, 12, 51, 76, 49,124, 49,184, 48,192, 48,192,112, 96,112, 48, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_190[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 24, 24, 24, 12,252, 6,216, 6,120,115, 56,249, 24,153,136, 48,192, 48,192,152, 96,248, 48,112, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_191[] = { 10, 0, 0, 62, 0,127, 0, 99, 0, 99, 0,112, 0, 56, 0, 28, 0, 12, 0, 12, 0, 12, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_192[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 6, 0, 12, 0, 24, 0}; -static const GLubyte Helvetica18_Character_193[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 6, 0, 3, 0, 1,128}; -static const GLubyte Helvetica18_Character_194[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 25,128, 15, 0, 6, 0}; -static const GLubyte Helvetica18_Character_195[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 19, 0, 22,128, 12,128}; -static const GLubyte Helvetica18_Character_196[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 0, 0, 25,128, 25,128, 0, 0}; -static const GLubyte Helvetica18_Character_197[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192, 48,192, 48, 96, 96, 96, 96,127,224, 63,192, 48,192, 48,192, 25,128, 25,128, 15, 0, 15, 0, 6, 0, 6, 0, 15, 0, 25,128, 25,128, 15, 0}; -static const GLubyte Helvetica18_Character_198[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96,255,128, 96,255,128, 48,192, 0, 48,192, 0, 63,192, 0, 31,192, 0, 24,255, 0, 24,255, 0, 12,192, 0, 12,192, 0, 6,192, 0, 6,192, 0, 3,255,128, 3,255,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_199[] = { 14, 0, 0, 15, 0, 13,128, 1,128, 7, 0, 7,192, 31,240, 56, 56, 48, 24,112, 0, 96, 0, 96, 0, 96, 0, 96, 0,112, 0, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_200[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,192, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 6, 0, 12, 0, 24, 0}; -static const GLubyte Helvetica18_Character_201[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,192, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 6, 0, 3, 0, 1,128}; -static const GLubyte Helvetica18_Character_202[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,192, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 25,128, 15, 0, 6, 0}; -static const GLubyte Helvetica18_Character_203[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,192, 96, 0, 96, 0, 96, 0, 96, 0,127,128,127,128, 96, 0, 96, 0, 96, 0, 96, 0,127,192,127,192, 0, 0, 25,128, 25,128, 0, 0}; -static const GLubyte Helvetica18_Character_204[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0, 48, 96,192}; -static const GLubyte Helvetica18_Character_205[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0, 48, 24, 12}; -static const GLubyte Helvetica18_Character_206[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0,204,120, 48}; -static const GLubyte Helvetica18_Character_207[] = { 6, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0,204,204, 0}; -static const GLubyte Helvetica18_Character_208[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128,127,192, 96,224, 96, 96, 96, 48, 96, 48,252, 48,252, 48, 96, 48, 96, 48, 96, 96, 96,224,127,192,127,128, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_209[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 48, 96,112, 96,240, 96,240, 97,176, 99, 48, 99, 48,102, 48,102, 48,108, 48,108, 48,120, 48,112, 48,112, 48, 0, 0, 9,128, 11, 64, 6, 64}; -static const GLubyte Helvetica18_Character_210[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 1,128, 3, 0, 6, 0}; -static const GLubyte Helvetica18_Character_211[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 1,128, 0,192, 0, 96}; -static const GLubyte Helvetica18_Character_212[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 6, 96, 3,192, 1,128}; -static const GLubyte Helvetica18_Character_213[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 4,192, 5,160, 3, 32}; -static const GLubyte Helvetica18_Character_214[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,192, 31,240, 56, 56, 48, 24,112, 28, 96, 12, 96, 12, 96, 12, 96, 12,112, 28, 48, 24, 56, 56, 31,240, 7,192, 0, 0, 6,192, 6,192, 0, 0}; -static const GLubyte Helvetica18_Character_215[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192,192, 97,128, 51, 0, 30, 0, 12, 0, 30, 0, 51, 0, 97,128,192,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_216[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,199,192,255,240,120, 56, 56, 24,108, 28,110, 12,103, 12, 99,140, 97,204,112,220, 48,120, 56, 56, 31,252, 7,204, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_217[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 63,224, 48, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 3, 0, 6, 0, 12, 0}; -static const GLubyte Helvetica18_Character_218[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 63,224, 48, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 6, 0, 3, 0, 1,128}; -static const GLubyte Helvetica18_Character_219[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 63,224, 48, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 12,192, 7,128, 3, 0}; -static const GLubyte Helvetica18_Character_220[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,128, 63,224, 48, 96, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 0, 0, 12,192, 12,192, 0, 0}; -static const GLubyte Helvetica18_Character_221[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 7,128, 12,192, 24, 96, 24, 96, 48, 48, 48, 48, 96, 24, 96, 24, 0, 0, 3, 0, 1,128, 0,192}; -static const GLubyte Helvetica18_Character_222[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0, 96, 0, 96, 0,127,128,127,192, 96,224, 96, 96, 96, 96, 96,224,127,192,127,128, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_223[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,110, 0,111, 0, 99, 0, 99, 0, 99, 0, 99, 0,110, 0,110, 0, 99, 0, 99, 0, 99, 0, 99, 0, 62, 0, 28, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_224[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 12, 0, 24, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_225[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 24, 0, 12, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_226[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 51, 0, 30, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_227[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 38, 0, 45, 0, 25, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_228[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 0, 0, 54, 0, 54, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_229[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 59, 0,119, 0, 99, 0, 99, 0,115, 0, 63, 0, 7, 0, 99, 0,119, 0, 62, 0, 28, 0, 54, 0, 54, 0, 28, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_230[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 58,240,119,252, 99,140, 99, 0,115, 0, 63,252, 7, 12, 99, 12,119,248, 62,240, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_231[] = { 10, 0, 0, 60, 0, 54, 0, 6, 0, 28, 0, 31, 0, 63,128, 49,128, 96, 0, 96, 0, 96, 0, 96, 0, 49,128, 63,128, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_232[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63,128,113,128, 96, 0, 96, 0,127,128, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 12, 0, 24, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_233[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63,128,113,128, 96, 0, 96, 0,127,128, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 12, 0, 6, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_234[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63,128,113,128, 96, 0, 96, 0,127,128, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 51, 0, 30, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_235[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 0, 63,128,113,128, 96, 0, 96, 0,127,128, 97,128, 97,128, 63, 0, 30, 0, 0, 0, 27, 0, 27, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_236[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0, 48, 96,192, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_237[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0,192, 96, 48, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_238[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0,144,240, 96, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_239[] = { 4, 0, 0, 0, 0, 0, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 0,216,216, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_240[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 38, 0, 28, 0, 27, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_241[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128,113,128,111,128,103, 0, 0, 0, 38, 0, 45, 0, 25, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_242[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 6, 0, 12, 0, 24, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_243[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 12, 0, 6, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_244[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 25,128, 15, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_245[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 19, 0, 22,128, 12,128, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_246[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 31, 0, 63,128, 49,128, 96,192, 96,192, 96,192, 96,192, 49,128, 63,128, 31, 0, 0, 0, 27, 0, 27, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_247[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0,127,128,127,128, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_248[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,206, 0,127,128, 49,128,120,192,108,192,102,192, 99,192, 49,128, 63,192, 14, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_249[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 57,128,125,128, 99,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 6, 0, 12, 0, 24, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_250[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 57,128,125,128, 99,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 12, 0, 6, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_251[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 57,128,125,128, 99,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 51, 0, 30, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_252[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 57,128,125,128, 99,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 97,128, 0, 0, 51, 0, 51, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_253[] = { 10, 0, 0, 56, 0, 56, 0, 12, 0, 12, 0, 12, 0, 12, 0, 30, 0, 18, 0, 51, 0, 51, 0, 51, 0, 97,128, 97,128, 97,128, 0, 0, 12, 0, 6, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_254[] = { 11, 0, 0, 96, 0, 96, 0, 96, 0, 96, 0,111, 0,127,128,113,128, 96,192, 96,192, 96,192, 96,192,113,128,127,128,111, 0, 96, 0, 96, 0, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte Helvetica18_Character_255[] = { 10, 0, 0, 56, 0, 56, 0, 12, 0, 12, 0, 12, 0, 12, 0, 30, 0, 18, 0, 51, 0, 51, 0, 51, 0, 97,128, 97,128, 97,128, 0, 0, 51, 0, 51, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; - -/* The font characters mapping: */ -static const GLubyte* Helvetica18_Character_Map[] = {Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032, - Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032, - Helvetica18_Character_032,Helvetica18_Character_033,Helvetica18_Character_034,Helvetica18_Character_035,Helvetica18_Character_036,Helvetica18_Character_037,Helvetica18_Character_038,Helvetica18_Character_039,Helvetica18_Character_040,Helvetica18_Character_041,Helvetica18_Character_042,Helvetica18_Character_043,Helvetica18_Character_044,Helvetica18_Character_045,Helvetica18_Character_046,Helvetica18_Character_047, - Helvetica18_Character_048,Helvetica18_Character_049,Helvetica18_Character_050,Helvetica18_Character_051,Helvetica18_Character_052,Helvetica18_Character_053,Helvetica18_Character_054,Helvetica18_Character_055,Helvetica18_Character_056,Helvetica18_Character_057,Helvetica18_Character_058,Helvetica18_Character_059,Helvetica18_Character_060,Helvetica18_Character_061,Helvetica18_Character_062,Helvetica18_Character_063, - Helvetica18_Character_064,Helvetica18_Character_065,Helvetica18_Character_066,Helvetica18_Character_067,Helvetica18_Character_068,Helvetica18_Character_069,Helvetica18_Character_070,Helvetica18_Character_071,Helvetica18_Character_072,Helvetica18_Character_073,Helvetica18_Character_074,Helvetica18_Character_075,Helvetica18_Character_076,Helvetica18_Character_077,Helvetica18_Character_078,Helvetica18_Character_079, - Helvetica18_Character_080,Helvetica18_Character_081,Helvetica18_Character_082,Helvetica18_Character_083,Helvetica18_Character_084,Helvetica18_Character_085,Helvetica18_Character_086,Helvetica18_Character_087,Helvetica18_Character_088,Helvetica18_Character_089,Helvetica18_Character_090,Helvetica18_Character_091,Helvetica18_Character_092,Helvetica18_Character_093,Helvetica18_Character_094,Helvetica18_Character_095, - Helvetica18_Character_096,Helvetica18_Character_097,Helvetica18_Character_098,Helvetica18_Character_099,Helvetica18_Character_100,Helvetica18_Character_101,Helvetica18_Character_102,Helvetica18_Character_103,Helvetica18_Character_104,Helvetica18_Character_105,Helvetica18_Character_106,Helvetica18_Character_107,Helvetica18_Character_108,Helvetica18_Character_109,Helvetica18_Character_110,Helvetica18_Character_111, - Helvetica18_Character_112,Helvetica18_Character_113,Helvetica18_Character_114,Helvetica18_Character_115,Helvetica18_Character_116,Helvetica18_Character_117,Helvetica18_Character_118,Helvetica18_Character_119,Helvetica18_Character_120,Helvetica18_Character_121,Helvetica18_Character_122,Helvetica18_Character_123,Helvetica18_Character_124,Helvetica18_Character_125,Helvetica18_Character_126,Helvetica18_Character_032, - Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032, - Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032,Helvetica18_Character_032, - Helvetica18_Character_160,Helvetica18_Character_161,Helvetica18_Character_162,Helvetica18_Character_163,Helvetica18_Character_164,Helvetica18_Character_165,Helvetica18_Character_166,Helvetica18_Character_167,Helvetica18_Character_168,Helvetica18_Character_169,Helvetica18_Character_170,Helvetica18_Character_171,Helvetica18_Character_172,Helvetica18_Character_173,Helvetica18_Character_174,Helvetica18_Character_175, - Helvetica18_Character_176,Helvetica18_Character_177,Helvetica18_Character_178,Helvetica18_Character_179,Helvetica18_Character_180,Helvetica18_Character_181,Helvetica18_Character_182,Helvetica18_Character_183,Helvetica18_Character_184,Helvetica18_Character_185,Helvetica18_Character_186,Helvetica18_Character_187,Helvetica18_Character_188,Helvetica18_Character_189,Helvetica18_Character_190,Helvetica18_Character_191, - Helvetica18_Character_192,Helvetica18_Character_193,Helvetica18_Character_194,Helvetica18_Character_195,Helvetica18_Character_196,Helvetica18_Character_197,Helvetica18_Character_198,Helvetica18_Character_199,Helvetica18_Character_200,Helvetica18_Character_201,Helvetica18_Character_202,Helvetica18_Character_203,Helvetica18_Character_204,Helvetica18_Character_205,Helvetica18_Character_206,Helvetica18_Character_207, - Helvetica18_Character_208,Helvetica18_Character_209,Helvetica18_Character_210,Helvetica18_Character_211,Helvetica18_Character_212,Helvetica18_Character_213,Helvetica18_Character_214,Helvetica18_Character_215,Helvetica18_Character_216,Helvetica18_Character_217,Helvetica18_Character_218,Helvetica18_Character_219,Helvetica18_Character_220,Helvetica18_Character_221,Helvetica18_Character_222,Helvetica18_Character_223, - Helvetica18_Character_224,Helvetica18_Character_225,Helvetica18_Character_226,Helvetica18_Character_227,Helvetica18_Character_228,Helvetica18_Character_229,Helvetica18_Character_230,Helvetica18_Character_231,Helvetica18_Character_232,Helvetica18_Character_233,Helvetica18_Character_234,Helvetica18_Character_235,Helvetica18_Character_236,Helvetica18_Character_237,Helvetica18_Character_238,Helvetica18_Character_239, - Helvetica18_Character_240,Helvetica18_Character_241,Helvetica18_Character_242,Helvetica18_Character_243,Helvetica18_Character_244,Helvetica18_Character_245,Helvetica18_Character_246,Helvetica18_Character_247,Helvetica18_Character_248,Helvetica18_Character_249,Helvetica18_Character_250,Helvetica18_Character_251,Helvetica18_Character_252,Helvetica18_Character_253,Helvetica18_Character_254,Helvetica18_Character_255,NULL}; - -/* The font structure: */ -const SFG_Font fgFontHelvetica18 = { "-adobe-helvetica-medium-r-normal--18-180-75-75-p-98-iso8859-1", 256, 23, Helvetica18_Character_Map, 0, 5 }; - -static const GLubyte TimesRoman10_Character_000[] = { 8, 0, 0, 0, 0,170, 0,130, 0,130, 0,170, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_001[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_002[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_003[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_004[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_005[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_006[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_007[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_008[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_009[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_010[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_011[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_012[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_013[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_014[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_015[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_016[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_017[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_018[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_019[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_020[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_021[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_022[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_023[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_024[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_025[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_026[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_027[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_028[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_029[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_030[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_031[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_032[] = { 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_033[] = { 3, 0, 0, 0, 0, 64, 0, 64, 64, 64, 64, 64, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_034[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0,160,160, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_035[] = { 5, 0, 0, 0, 0, 80, 80,248, 80,248, 80, 80, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_036[] = { 5, 0, 0, 0, 32,224,144, 16, 96,128,144,112, 32, 0, 0}; -static const GLubyte TimesRoman10_Character_037[] = { 8, 0, 0, 0, 0, 68, 42, 42, 86,168,164,126, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_038[] = { 8, 0, 0, 0, 0,118,141,152,116,110, 80, 48, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_039[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 64,192, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_040[] = { 4, 0, 0, 32, 64, 64,128,128,128, 64, 64, 32, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_041[] = { 4, 0, 0,128, 64, 64, 32, 32, 32, 64, 64,128, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_042[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0,160, 64,160, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_043[] = { 6, 0, 0, 0, 0, 32, 32,248, 32, 32, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_044[] = { 3, 0, 0, 64, 64, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_045[] = { 7, 0, 0, 0, 0, 0, 0,120, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_046[] = { 3, 0, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_047[] = { 3, 0, 0, 0, 0,128,128, 64, 64, 64, 32, 32, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_048[] = { 5, 0, 0, 0, 0, 96,144,144,144,144,144, 96, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_049[] = { 5, 0, 0, 0, 0,112, 32, 32, 32, 32, 96, 32, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_050[] = { 5, 0, 0, 0, 0,240, 64, 32, 32, 16,144, 96, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_051[] = { 5, 0, 0, 0, 0,224, 16, 16, 96, 16,144, 96, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_052[] = { 5, 0, 0, 0, 0, 16, 16,248,144, 80, 48, 16, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_053[] = { 5, 0, 0, 0, 0,224,144, 16, 16,224, 64,112, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_054[] = { 5, 0, 0, 0, 0, 96,144,144,144,224, 64, 48, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_055[] = { 5, 0, 0, 0, 0, 64, 64, 64, 32, 32,144,240, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_056[] = { 5, 0, 0, 0, 0, 96,144,144, 96,144,144, 96, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_057[] = { 5, 0, 0, 0, 0,192, 32,112,144,144,144, 96, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_058[] = { 3, 0, 0, 0, 0, 64, 0, 0, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_059[] = { 3, 0, 0, 64, 64, 64, 0, 0, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_060[] = { 5, 0, 0, 0, 0, 16, 32, 64, 32, 16, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_061[] = { 6, 0, 0, 0, 0, 0,248, 0,248, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_062[] = { 5, 0, 0, 0, 0,128, 64, 32, 64,128, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_063[] = { 4, 0, 0, 0, 0, 64, 0, 64, 64, 32,160,224, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_064[] = { 9, 0, 0, 0, 0, 62, 0, 64, 0,146, 0,173, 0,165, 0,165, 0,157, 0, 66, 0, 60, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_065[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_066[] = { 6, 0, 0, 0, 0,240, 72, 72,112, 72, 72,240, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_067[] = { 7, 0, 0, 0, 0,120,196,128,128,128,196,124, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_068[] = { 7, 0, 0, 0, 0,248, 76, 68, 68, 68, 76,248, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_069[] = { 6, 0, 0, 0, 0,248, 72, 64,112, 64, 72,248, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_070[] = { 6, 0, 0, 0, 0,224, 64, 64,112, 64, 72,248, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_071[] = { 7, 0, 0, 0, 0,120,196,132,156,128,196,124, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_072[] = { 8, 0, 0, 0, 0,238, 68, 68,124, 68, 68,238, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_073[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,224, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_074[] = { 4, 0, 0, 0, 0,192,160, 32, 32, 32, 32,112, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_075[] = { 7, 0, 0, 0, 0,236, 72, 80, 96, 80, 72,236, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_076[] = { 6, 0, 0, 0, 0,248, 72, 64, 64, 64, 64,224, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_077[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0,235,128, 73, 0, 85, 0, 85, 0, 99, 0, 99, 0,227,128, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_078[] = { 8, 0, 0, 0, 0,228, 76, 76, 84, 84,100,238, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_079[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_080[] = { 6, 0, 0, 0, 0,224, 64, 64,112, 72, 72,240, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_081[] = { 7, 0, 0, 12, 24,112,204,132,132,132,204,120, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_082[] = { 7, 0, 0, 0, 0,236, 72, 80,112, 72, 72,240, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_083[] = { 5, 0, 0, 0, 0,224,144, 16, 96,192,144,112, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_084[] = { 6, 0, 0, 0, 0,112, 32, 32, 32, 32,168,248, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_085[] = { 8, 0, 0, 0, 0, 56,108, 68, 68, 68, 68,238, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_086[] = { 8, 0, 0, 0, 0, 16, 16, 40, 40,108, 68,238, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_087[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 34, 0, 85, 0, 85, 0,201,128,136,128,221,192, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_088[] = { 8, 0, 0, 0, 0,238, 68, 40, 16, 40, 68,238, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_089[] = { 8, 0, 0, 0, 0, 56, 16, 16, 40, 40, 68,238, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_090[] = { 6, 0, 0, 0, 0,248,136, 64, 32, 16,136,248, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_091[] = { 3, 0, 0,192,128,128,128,128,128,128,128,192, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_092[] = { 3, 0, 0, 0, 0, 32, 32, 64, 64, 64,128,128, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_093[] = { 3, 0, 0,192, 64, 64, 64, 64, 64, 64, 64,192, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_094[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 80, 80, 32, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_095[] = { 5, 0,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_096[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,192,128, 0, 0}; -static const GLubyte TimesRoman10_Character_097[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_098[] = { 5, 0, 0, 0, 0,224,144,144,144,224,128,128, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_099[] = { 4, 0, 0, 0, 0, 96,128,128,128, 96, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_100[] = { 5, 0, 0, 0, 0,104,144,144,144,112, 16, 48, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_101[] = { 4, 0, 0, 0, 0, 96,128,192,160, 96, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_102[] = { 4, 0, 0, 0, 0,224, 64, 64, 64,224, 64, 48, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_103[] = { 5, 0, 0,224,144, 96, 64,160,160,112, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_104[] = { 5, 0, 0, 0, 0,216,144,144,144,224,128,128, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_105[] = { 3, 0, 0, 0, 0, 64, 64, 64, 64,192, 0, 64, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_106[] = { 3, 0, 0,128, 64, 64, 64, 64, 64,192, 0, 64, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_107[] = { 5, 0, 0, 0, 0,152,144,224,160,144,128,128, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_108[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,192, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_109[] = { 8, 0, 0, 0, 0,219,146,146,146,236, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_110[] = { 5, 0, 0, 0, 0,216,144,144,144,224, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_111[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_112[] = { 5, 0, 0,192,128,224,144,144,144,224, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_113[] = { 5, 0, 0, 56, 16,112,144,144,144,112, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_114[] = { 4, 0, 0, 0, 0,224, 64, 64, 96,160, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_115[] = { 4, 0, 0, 0, 0,224, 32, 96,128,224, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_116[] = { 4, 0, 0, 0, 0, 48, 64, 64, 64,224, 64, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_117[] = { 5, 0, 0, 0, 0,104,144,144,144,144, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_118[] = { 5, 0, 0, 0, 0, 32, 96, 80,144,216, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_119[] = { 8, 0, 0, 0, 0, 40,108, 84,146,219, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_120[] = { 6, 0, 0, 0, 0,216, 80, 32, 80,216, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_121[] = { 5, 0, 0,128,128, 64, 96,160,144,184, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_122[] = { 5, 0, 0, 0, 0,240,144, 64, 32,240, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_123[] = { 4, 0, 0, 32, 64, 64, 64,128, 64, 64, 64, 32, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_124[] = { 2, 0, 0,128,128,128,128,128,128,128,128,128, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_125[] = { 4, 0, 0,128, 64, 64, 64, 32, 64, 64, 64,128, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_126[] = { 7, 0, 0, 0, 0, 0, 0,152,100, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_127[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_128[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_129[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_130[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_131[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_132[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_133[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_134[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_135[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_136[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_137[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_138[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_139[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_140[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_141[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_142[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_143[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_144[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_145[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_146[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_147[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_148[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_149[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_150[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_151[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_152[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_153[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_154[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_155[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_156[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_157[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_158[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_159[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0,170, 0, 0, 0,130, 0, 0, 0,130, 0, 0, 0,170, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_160[] = { 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_161[] = { 3, 0, 0, 64, 64, 64, 64, 64, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_162[] = { 5, 0, 0, 0,128,224,144,128,144,112, 16, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_163[] = { 5, 0, 0, 0, 0,240,200, 64,224, 64, 80, 48, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_164[] = { 5, 0, 0, 0, 0, 0,136,112, 80, 80,112,136, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_165[] = { 5, 0, 0, 0, 0,112, 32,248, 32,216, 80,136, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_166[] = { 2, 0, 0, 0, 0,128,128,128, 0,128,128,128, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_167[] = { 5, 0, 0, 0,224,144, 32, 80,144,160, 64,144,112, 0, 0}; -static const GLubyte TimesRoman10_Character_168[] = { 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 80, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_169[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 77, 0, 81, 0, 77, 0, 34, 0, 28, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_170[] = { 4, 0, 0, 0, 0, 0, 0,224, 0,160, 32,192, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_171[] = { 5, 0, 0, 0, 0, 0, 80,160,160, 80, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_172[] = { 7, 0, 0, 0, 0, 0, 4, 4,124, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_173[] = { 4, 0, 0, 0, 0, 0, 0,224, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_174[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 85, 0, 89, 0, 93, 0, 34, 0, 28, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_175[] = { 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,224, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_176[] = { 4, 0, 0, 0, 0, 0, 0, 0, 96,144,144, 96, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_177[] = { 6, 0, 0, 0, 0,248, 0, 32, 32,248, 32, 32, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_178[] = { 3, 0, 0, 0, 0, 0, 0, 0,224, 64,160, 96, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_179[] = { 3, 0, 0, 0, 0, 0, 0, 0,192, 32, 64,224, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_180[] = { 3, 0, 0, 0, 0, 0, 0, 0, 0, 0,128, 64, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_181[] = { 5, 0, 0,128,128,232,144,144,144,144, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_182[] = { 6, 0, 0, 40, 40, 40, 40,104,232,232,232,124, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_183[] = { 2, 0, 0, 0, 0, 0, 0,128, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_184[] = { 4, 0,192, 32, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_185[] = { 3, 0, 0, 0, 0, 0, 0, 0,224, 64,192, 64, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_186[] = { 4, 0, 0, 0, 0, 0, 0,224, 0, 64,160, 64, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_187[] = { 5, 0, 0, 0, 0, 0,160, 80, 80,160, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_188[] = { 8, 0, 0, 0, 0, 68, 62, 44,244, 72,200, 68, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_189[] = { 8, 0, 0, 0, 0, 78, 36, 42,246, 72,200, 68, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_190[] = { 8, 0, 0, 0, 0, 68, 62, 44,212, 40, 72,228, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_191[] = { 4, 0, 0,224,160,128, 64, 64, 0, 64, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_192[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 16, 32}; -static const GLubyte TimesRoman10_Character_193[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 16, 8}; -static const GLubyte TimesRoman10_Character_194[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 40, 16}; -static const GLubyte TimesRoman10_Character_195[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 40, 20}; -static const GLubyte TimesRoman10_Character_196[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 0, 40, 0}; -static const GLubyte TimesRoman10_Character_197[] = { 8, 0, 0, 0, 0,238, 68,124, 40, 40, 56, 16, 16, 40, 16}; -static const GLubyte TimesRoman10_Character_198[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0,239, 0, 73, 0,120, 0, 46, 0, 40, 0, 57, 0, 31, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_199[] = { 7, 0, 96, 16, 32,120,196,128,128,128,196,124, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_200[] = { 6, 0, 0, 0, 0,248, 72, 64,112, 64, 72,248, 0, 32, 64}; -static const GLubyte TimesRoman10_Character_201[] = { 6, 0, 0, 0, 0,248, 72, 64,112, 64, 72,248, 0, 32, 16}; -static const GLubyte TimesRoman10_Character_202[] = { 6, 0, 0, 0, 0,248, 72, 64,112, 64, 72,248, 0, 80, 32}; -static const GLubyte TimesRoman10_Character_203[] = { 6, 0, 0, 0, 0,248, 72, 64,112, 64, 72,248, 0, 80, 0}; -static const GLubyte TimesRoman10_Character_204[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,224, 0, 64,128}; -static const GLubyte TimesRoman10_Character_205[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,224, 0, 64, 32}; -static const GLubyte TimesRoman10_Character_206[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,224, 0,160, 64}; -static const GLubyte TimesRoman10_Character_207[] = { 4, 0, 0, 0, 0,224, 64, 64, 64, 64, 64,224, 0,160, 0}; -static const GLubyte TimesRoman10_Character_208[] = { 7, 0, 0, 0, 0,248, 76, 68,228, 68, 76,248, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_209[] = { 8, 0, 0, 0, 0,228, 76, 76, 84, 84,100,238, 0, 80, 40}; -static const GLubyte TimesRoman10_Character_210[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 32, 64}; -static const GLubyte TimesRoman10_Character_211[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 16, 8}; -static const GLubyte TimesRoman10_Character_212[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 80, 32}; -static const GLubyte TimesRoman10_Character_213[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 80, 40}; -static const GLubyte TimesRoman10_Character_214[] = { 7, 0, 0, 0, 0,120,204,132,132,132,204,120, 0, 80, 0}; -static const GLubyte TimesRoman10_Character_215[] = { 6, 0, 0, 0, 0,136, 80, 32, 80,136, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_216[] = { 8, 0, 0, 0,128,124,102, 82, 82, 74,102, 62, 1, 0, 0}; -static const GLubyte TimesRoman10_Character_217[] = { 8, 0, 0, 0, 0, 56,108, 68, 68, 68, 68,238, 0, 16, 32}; -static const GLubyte TimesRoman10_Character_218[] = { 8, 0, 0, 0, 0, 56,108, 68, 68, 68, 68,238, 0, 16, 8}; -static const GLubyte TimesRoman10_Character_219[] = { 8, 0, 0, 0, 0, 56,108, 68, 68, 68, 68,238, 0, 40, 16}; -static const GLubyte TimesRoman10_Character_220[] = { 8, 0, 0, 0, 0, 56,108, 68, 68, 68, 68,238, 0, 40, 0}; -static const GLubyte TimesRoman10_Character_221[] = { 8, 0, 0, 0, 0, 56, 16, 16, 40, 40, 68,238, 0, 16, 8}; -static const GLubyte TimesRoman10_Character_222[] = { 6, 0, 0, 0, 0,224, 64,112, 72,112, 64,224, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_223[] = { 5, 0, 0, 0, 0,224, 80, 80, 96, 80, 80, 32, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_224[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0, 64,128, 0, 0}; -static const GLubyte TimesRoman10_Character_225[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0, 64, 32, 0, 0}; -static const GLubyte TimesRoman10_Character_226[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0,160, 64, 0, 0}; -static const GLubyte TimesRoman10_Character_227[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0,160, 80, 0, 0}; -static const GLubyte TimesRoman10_Character_228[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 0,160, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_229[] = { 4, 0, 0, 0, 0,224,160, 96, 32,192, 64,160, 64, 0, 0}; -static const GLubyte TimesRoman10_Character_230[] = { 6, 0, 0, 0, 0,216,160,112, 40,216, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_231[] = { 4, 0,192, 32, 64, 96,128,128,128, 96, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_232[] = { 4, 0, 0, 0, 0, 96,128,192,160, 96, 0, 64,128, 0, 0}; -static const GLubyte TimesRoman10_Character_233[] = { 4, 0, 0, 0, 0, 96,128,192,160, 96, 0, 64, 32, 0, 0}; -static const GLubyte TimesRoman10_Character_234[] = { 4, 0, 0, 0, 0, 96,128,192,160, 96, 0,160, 64, 0, 0}; -static const GLubyte TimesRoman10_Character_235[] = { 4, 0, 0, 0, 0, 96,128,192,160, 96, 0,160, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_236[] = { 4, 0, 0, 0, 0,224, 64, 64, 64,192, 0, 64,128, 0, 0}; -static const GLubyte TimesRoman10_Character_237[] = { 4, 0, 0, 0, 0,224, 64, 64, 64,192, 0, 64, 32, 0, 0}; -static const GLubyte TimesRoman10_Character_238[] = { 4, 0, 0, 0, 0,224, 64, 64, 64,192, 0,160, 64, 0, 0}; -static const GLubyte TimesRoman10_Character_239[] = { 4, 0, 0, 0, 0,224, 64, 64, 64,192, 0,160, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_240[] = { 5, 0, 0, 0, 0, 96,144,144,144,112,160,112, 64, 0, 0}; -static const GLubyte TimesRoman10_Character_241[] = { 5, 0, 0, 0, 0,216,144,144,144,224, 0,160, 80, 0, 0}; -static const GLubyte TimesRoman10_Character_242[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0, 32, 64, 0, 0}; -static const GLubyte TimesRoman10_Character_243[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0, 64, 32, 0, 0}; -static const GLubyte TimesRoman10_Character_244[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0,160, 64, 0, 0}; -static const GLubyte TimesRoman10_Character_245[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0,160, 80, 0, 0}; -static const GLubyte TimesRoman10_Character_246[] = { 5, 0, 0, 0, 0, 96,144,144,144, 96, 0,160, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_247[] = { 6, 0, 0, 0, 0, 32, 0,248, 0, 32, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_248[] = { 5, 0, 0, 0, 0,224,144,144,144,112, 8, 0, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_249[] = { 5, 0, 0, 0, 0,104,144,144,144,144, 0, 32, 64, 0, 0}; -static const GLubyte TimesRoman10_Character_250[] = { 5, 0, 0, 0, 0,104,144,144,144,144, 0, 64, 32, 0, 0}; -static const GLubyte TimesRoman10_Character_251[] = { 5, 0, 0, 0, 0,104,144,144,144,144, 0, 80, 32, 0, 0}; -static const GLubyte TimesRoman10_Character_252[] = { 5, 0, 0, 0, 0,104,144,144,144,144, 0, 80, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_253[] = { 5, 0, 0,128,192, 64, 96,160,144,184, 0, 32, 16, 0, 0}; -static const GLubyte TimesRoman10_Character_254[] = { 5, 0, 0,192,128,224,144,144,144,224,128,128, 0, 0, 0}; -static const GLubyte TimesRoman10_Character_255[] = { 5, 0, 0,128,192, 64, 96,160,144,184, 0,160, 0, 0, 0}; - -/* The font characters mapping: */ -static const GLubyte* TimesRoman10_Character_Map[] = {TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032, - TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032, - TimesRoman10_Character_032,TimesRoman10_Character_033,TimesRoman10_Character_034,TimesRoman10_Character_035,TimesRoman10_Character_036,TimesRoman10_Character_037,TimesRoman10_Character_038,TimesRoman10_Character_039,TimesRoman10_Character_040,TimesRoman10_Character_041,TimesRoman10_Character_042,TimesRoman10_Character_043,TimesRoman10_Character_044,TimesRoman10_Character_045,TimesRoman10_Character_046,TimesRoman10_Character_047, - TimesRoman10_Character_048,TimesRoman10_Character_049,TimesRoman10_Character_050,TimesRoman10_Character_051,TimesRoman10_Character_052,TimesRoman10_Character_053,TimesRoman10_Character_054,TimesRoman10_Character_055,TimesRoman10_Character_056,TimesRoman10_Character_057,TimesRoman10_Character_058,TimesRoman10_Character_059,TimesRoman10_Character_060,TimesRoman10_Character_061,TimesRoman10_Character_062,TimesRoman10_Character_063, - TimesRoman10_Character_064,TimesRoman10_Character_065,TimesRoman10_Character_066,TimesRoman10_Character_067,TimesRoman10_Character_068,TimesRoman10_Character_069,TimesRoman10_Character_070,TimesRoman10_Character_071,TimesRoman10_Character_072,TimesRoman10_Character_073,TimesRoman10_Character_074,TimesRoman10_Character_075,TimesRoman10_Character_076,TimesRoman10_Character_077,TimesRoman10_Character_078,TimesRoman10_Character_079, - TimesRoman10_Character_080,TimesRoman10_Character_081,TimesRoman10_Character_082,TimesRoman10_Character_083,TimesRoman10_Character_084,TimesRoman10_Character_085,TimesRoman10_Character_086,TimesRoman10_Character_087,TimesRoman10_Character_088,TimesRoman10_Character_089,TimesRoman10_Character_090,TimesRoman10_Character_091,TimesRoman10_Character_092,TimesRoman10_Character_093,TimesRoman10_Character_094,TimesRoman10_Character_095, - TimesRoman10_Character_096,TimesRoman10_Character_097,TimesRoman10_Character_098,TimesRoman10_Character_099,TimesRoman10_Character_100,TimesRoman10_Character_101,TimesRoman10_Character_102,TimesRoman10_Character_103,TimesRoman10_Character_104,TimesRoman10_Character_105,TimesRoman10_Character_106,TimesRoman10_Character_107,TimesRoman10_Character_108,TimesRoman10_Character_109,TimesRoman10_Character_110,TimesRoman10_Character_111, - TimesRoman10_Character_112,TimesRoman10_Character_113,TimesRoman10_Character_114,TimesRoman10_Character_115,TimesRoman10_Character_116,TimesRoman10_Character_117,TimesRoman10_Character_118,TimesRoman10_Character_119,TimesRoman10_Character_120,TimesRoman10_Character_121,TimesRoman10_Character_122,TimesRoman10_Character_123,TimesRoman10_Character_124,TimesRoman10_Character_125,TimesRoman10_Character_126,TimesRoman10_Character_032, - TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032, - TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032,TimesRoman10_Character_032, - TimesRoman10_Character_160,TimesRoman10_Character_161,TimesRoman10_Character_162,TimesRoman10_Character_163,TimesRoman10_Character_164,TimesRoman10_Character_165,TimesRoman10_Character_166,TimesRoman10_Character_167,TimesRoman10_Character_168,TimesRoman10_Character_169,TimesRoman10_Character_170,TimesRoman10_Character_171,TimesRoman10_Character_172,TimesRoman10_Character_173,TimesRoman10_Character_174,TimesRoman10_Character_175, - TimesRoman10_Character_176,TimesRoman10_Character_177,TimesRoman10_Character_178,TimesRoman10_Character_179,TimesRoman10_Character_180,TimesRoman10_Character_181,TimesRoman10_Character_182,TimesRoman10_Character_183,TimesRoman10_Character_184,TimesRoman10_Character_185,TimesRoman10_Character_186,TimesRoman10_Character_187,TimesRoman10_Character_188,TimesRoman10_Character_189,TimesRoman10_Character_190,TimesRoman10_Character_191, - TimesRoman10_Character_192,TimesRoman10_Character_193,TimesRoman10_Character_194,TimesRoman10_Character_195,TimesRoman10_Character_196,TimesRoman10_Character_197,TimesRoman10_Character_198,TimesRoman10_Character_199,TimesRoman10_Character_200,TimesRoman10_Character_201,TimesRoman10_Character_202,TimesRoman10_Character_203,TimesRoman10_Character_204,TimesRoman10_Character_205,TimesRoman10_Character_206,TimesRoman10_Character_207, - TimesRoman10_Character_208,TimesRoman10_Character_209,TimesRoman10_Character_210,TimesRoman10_Character_211,TimesRoman10_Character_212,TimesRoman10_Character_213,TimesRoman10_Character_214,TimesRoman10_Character_215,TimesRoman10_Character_216,TimesRoman10_Character_217,TimesRoman10_Character_218,TimesRoman10_Character_219,TimesRoman10_Character_220,TimesRoman10_Character_221,TimesRoman10_Character_222,TimesRoman10_Character_223, - TimesRoman10_Character_224,TimesRoman10_Character_225,TimesRoman10_Character_226,TimesRoman10_Character_227,TimesRoman10_Character_228,TimesRoman10_Character_229,TimesRoman10_Character_230,TimesRoman10_Character_231,TimesRoman10_Character_232,TimesRoman10_Character_233,TimesRoman10_Character_234,TimesRoman10_Character_235,TimesRoman10_Character_236,TimesRoman10_Character_237,TimesRoman10_Character_238,TimesRoman10_Character_239, - TimesRoman10_Character_240,TimesRoman10_Character_241,TimesRoman10_Character_242,TimesRoman10_Character_243,TimesRoman10_Character_244,TimesRoman10_Character_245,TimesRoman10_Character_246,TimesRoman10_Character_247,TimesRoman10_Character_248,TimesRoman10_Character_249,TimesRoman10_Character_250,TimesRoman10_Character_251,TimesRoman10_Character_252,TimesRoman10_Character_253,TimesRoman10_Character_254,TimesRoman10_Character_255,NULL}; - -/* The font structure: */ -const SFG_Font fgFontTimesRoman10 = { "-adobe-times-medium-r-normal--10-100-75-75-p-54-iso8859-1", 256, 14, TimesRoman10_Character_Map, 0, 4 }; - -static const GLubyte TimesRoman24_Character_000[] = { 19, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 64, 0, 64, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_001[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_002[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_003[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_004[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_005[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_006[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_007[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_008[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_009[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_010[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_011[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_012[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_013[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_014[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_015[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_016[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_017[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_018[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_019[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_020[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_021[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_022[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_023[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_024[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_025[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_026[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_027[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_028[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_029[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_030[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_031[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_032[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_033[] = { 8, 0, 0, 0, 0, 0, 0, 0, 24, 24, 0, 0, 0, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_034[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 68, 0,102, 0,102, 0,102, 0,102, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_035[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 17, 0, 17, 0, 17, 0, 17, 0, 17, 0,127,224,127,224, 8,128, 8,128, 8,128, 63,240, 63,240, 4, 64, 4, 64, 4, 64, 4, 64, 4, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_036[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4, 0, 4, 0, 63, 0,229,192,196,192,132, 96,132, 96, 4, 96, 4,224, 7,192, 7,128, 30, 0, 60, 0,116, 0,100, 0,100, 32,100, 96, 52,224, 31,128, 4, 0, 4, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_037[] = { 19, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 30, 0, 12, 57, 0, 6, 48,128, 2, 48, 64, 3, 48, 64, 1,152, 64, 0,140,192, 0,199,128, 60, 96, 0,114, 32, 0, 97, 48, 0, 96,152, 0, 96,136, 0, 48,140, 0, 25,254, 0, 15, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_038[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 30, 30, 0, 63,191, 0,112,240,128, 96, 96, 0, 96,224, 0, 96,208, 0, 49,144, 0, 27,136, 0, 15, 12, 0, 7, 31, 0, 7,128, 0, 14,192, 0, 12, 96, 0, 12, 32, 0, 12, 32, 0, 6, 96, 0, 3,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_039[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 12, 4, 28, 24, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_040[] = { 8, 0, 0, 2, 4, 8, 24, 16, 48, 48, 96, 96, 96, 96, 96, 96, 96, 96, 48, 48, 16, 24, 8, 4, 2, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_041[] = { 8, 0, 0, 64, 32, 16, 24, 8, 12, 12, 6, 6, 6, 6, 6, 6, 6, 6, 12, 12, 8, 24, 16, 32, 64, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_042[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 7, 0, 50, 96, 58,224, 7, 0, 58,224, 50, 96, 7, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_043[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0,127,248,127,248, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_044[] = { 7, 0, 0, 0, 0, 48, 24, 8, 56, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_045[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,248,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_046[] = { 6, 0, 0, 0, 0, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_047[] = { 7, 0, 0, 0, 0,192,192,192, 64, 96, 96, 32, 48, 48, 16, 24, 24, 8, 12, 12, 4, 6, 6, 6, 6, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_048[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 25,128, 48,192, 48,192,112,224, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 48,192, 25,128, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_049[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 63,192, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 6, 0, 30, 0, 6, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_050[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,192,127,224, 48, 32, 24, 0, 12, 0, 6, 0, 2, 0, 3, 0, 1,128, 1,128, 0,192, 0,192, 64,192, 64,192, 33,192, 63,128, 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_051[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0,115, 0, 97,128, 0,128, 0,192, 0,192, 0,192, 1,192, 3,128, 15, 0, 6, 0, 3, 0, 65,128, 65,128, 35,128, 63, 0, 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_052[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,128, 1,128, 1,128, 1,128,127,224,127,224, 97,128, 33,128, 49,128, 17,128, 25,128, 9,128, 13,128, 5,128, 3,128, 3,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_053[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 63, 0,113,192, 96,192, 0, 96, 0, 96, 0, 96, 0, 96, 0,224, 1,192, 7,192, 63, 0, 60, 0, 48, 0, 16, 0, 16, 0, 15,192, 15,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_054[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 61,192, 48,192,112, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96,192,121,192,119, 0, 48, 0, 56, 0, 24, 0, 12, 0, 7, 0, 1,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_055[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 6, 0, 6, 0, 6, 0, 2, 0, 3, 0, 3, 0, 1, 0, 1,128, 1,128, 0,128, 0,192, 64,192, 96, 96,127,224, 63,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_056[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192,112,192, 96, 96, 96, 96, 96, 96, 32,224, 48,192, 27,128, 15, 0, 15, 0, 25,128, 48,192, 48,192, 48,192, 25,128, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_057[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120, 0, 14, 0, 3, 0, 1,128, 1,192, 0,192, 14,192, 57,224, 48,224, 96, 96, 96, 96, 96, 96, 96, 96, 96,224, 48,192, 59,192, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_058[] = { 6, 0, 0, 0, 0, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_059[] = { 7, 0, 0, 0, 0, 48, 24, 8, 56, 48, 0, 0, 0, 0, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_060[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 0,224, 3,128, 14, 0, 56, 0, 96, 0, 56, 0, 14, 0, 3,128, 0,224, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_061[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,248,127,248, 0, 0, 0, 0,127,248,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_062[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0, 56, 0, 14, 0, 3,128, 0,224, 0, 48, 0,224, 3,128, 14, 0, 56, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_063[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 12, 0, 12, 0, 0, 0, 0, 0, 4, 0, 4, 0, 4, 0, 6, 0, 6, 0, 3, 0, 3,128, 1,192, 48,192, 48,192, 32,192, 49,128, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_064[] = { 22, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 0, 3,131, 0, 6, 0, 0, 12, 0, 0, 24,119,128, 24,222,192, 49,142, 96, 49,134, 32, 49,134, 48, 49,134, 16, 49,131, 16, 48,195, 16, 48,227, 16, 56,127, 16, 24, 59, 48, 28, 0, 32, 14, 0, 96, 7, 0,192, 3,195,128, 0,254, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_065[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_066[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,240, 24, 60, 24, 12, 24, 6, 24, 6, 24, 6, 24, 12, 24, 28, 31,240, 24, 32, 24, 24, 24, 12, 24, 12, 24, 12, 24, 24, 24, 56,127,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_067[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 15, 28, 28, 4, 48, 2, 48, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 48, 2, 48, 2, 28, 6, 14, 30, 3,242, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_068[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,224, 0, 24, 56, 0, 24, 28, 0, 24, 6, 0, 24, 6, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 6, 0, 24, 6, 0, 24, 28, 0, 24, 56, 0,127,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_069[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 32, 24, 32, 31,224, 24, 32, 24, 32, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_070[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 16, 24, 16, 31,240, 24, 16, 24, 16, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_071[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 15, 28, 0, 28, 14, 0, 48, 6, 0, 48, 6, 0, 96, 6, 0, 96, 6, 0, 96, 31,128, 96, 0, 0, 96, 0, 0, 96, 0, 0, 96, 0, 0, 48, 2, 0, 48, 2, 0, 28, 6, 0, 14, 30, 0, 3,242, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_072[] = { 19, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 15,192, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 31,255, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0,126, 15,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_073[] = { 8, 0, 0, 0, 0, 0, 0, 0,126, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,126, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_074[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 60, 0,102, 0, 99, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 15,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_075[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 15,128, 24, 7, 0, 24, 14, 0, 24, 28, 0, 24, 56, 0, 24,112, 0, 24,224, 0, 25,192, 0, 31,128, 0, 31, 0, 0, 25,128, 0, 24,192, 0, 24, 96, 0, 24, 48, 0, 24, 24, 0, 24, 12, 0,126, 63, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_076[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0,126, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_077[] = { 22, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 16,252, 16, 48, 48, 16, 48, 48, 16,104, 48, 16,104, 48, 16,196, 48, 16,196, 48, 17,132, 48, 17,130, 48, 19, 2, 48, 19, 1, 48, 22, 1, 48, 22, 1, 48, 28, 0,176, 28, 0,176, 24, 0,112,120, 0,124, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_078[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 6, 0, 16, 14, 0, 16, 14, 0, 16, 26, 0, 16, 50, 0, 16, 50, 0, 16, 98, 0, 16,194, 0, 16,194, 0, 17,130, 0, 19, 2, 0, 19, 2, 0, 22, 2, 0, 28, 2, 0, 28, 2, 0, 24, 2, 0,120, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_079[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_080[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 31,224, 24, 56, 24, 24, 24, 12, 24, 12, 24, 12, 24, 24, 24, 56,127,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_081[] = { 18, 0, 0, 0, 0, 0, 0, 0, 7,128, 0, 28, 0, 0, 56, 0, 0,112, 0, 0,224, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_082[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 15, 24, 14, 24, 28, 24, 56, 24, 48, 24, 96, 24,224, 25,192, 31,224, 24, 56, 24, 24, 24, 28, 24, 12, 24, 28, 24, 24, 24, 56,127,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_083[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 79, 0,120,192, 96, 96, 64, 48, 64, 48, 0, 48, 0,112, 1,224, 7,192, 15, 0, 60, 0,112, 0, 96, 32, 96, 32, 96, 96, 49,224, 15, 32, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_084[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,224, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 65,130, 65,130, 97,134,127,254, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_085[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 24, 0, 12, 4, 0, 24, 4, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0,126, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_086[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,128, 0, 1,128, 0, 1,128, 0, 3,192, 0, 3, 64, 0, 3, 96, 0, 6, 32, 0, 6, 32, 0, 6, 48, 0, 12, 16, 0, 12, 24, 0, 24, 8, 0, 24, 8, 0, 24, 12, 0, 48, 4, 0, 48, 6, 0,252, 31,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_087[] = { 23, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,131, 0, 1,131, 0, 1,131,128, 3,135,128, 3, 70,128, 3, 70,192, 6, 70, 64, 6, 76, 64, 6, 76, 96, 12, 44, 96, 12, 44, 32, 24, 44, 32, 24, 24, 48, 24, 24, 16, 48, 24, 16, 48, 24, 24,252,126,126, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_088[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 15,192, 48, 3,128, 24, 7, 0, 8, 14, 0, 4, 12, 0, 6, 24, 0, 2, 56, 0, 1,112, 0, 0,224, 0, 0,192, 0, 1,192, 0, 3,160, 0, 3, 16, 0, 6, 8, 0, 14, 12, 0, 28, 6, 0,126, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_089[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,224, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 3,192, 3, 64, 6, 96, 6, 32, 12, 48, 28, 16, 24, 24, 56, 8, 48, 12,252, 63, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_090[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252,112, 12, 56, 4, 24, 4, 28, 0, 12, 0, 14, 0, 7, 0, 3, 0, 3,128, 1,128, 1,192, 0,224, 64, 96, 64,112, 96, 56,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_091[] = { 8, 0, 0, 0, 62, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 62, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_092[] = { 7, 0, 0, 0, 0, 0, 0, 0, 6, 6, 4, 12, 12, 8, 24, 24, 16, 48, 48, 32, 96, 96, 64,192,192, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_093[] = { 8, 0, 0, 0,124, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12,124, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_094[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 64, 64, 96,192, 32,128, 49,128, 17, 0, 27, 0, 10, 0, 14, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_095[] = { 13, 0, 0, 0, 0,255,248,255,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_096[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48,112, 64, 96, 48, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_097[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_098[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 47, 0, 57,192, 48,192, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48,192, 57,192, 55, 0, 48, 0, 48, 0, 48, 0, 48, 0,112, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_099[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 32,192, 49,192, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_100[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 96, 57,192, 48,192, 96,192, 96,192, 96,192, 96,192, 96,192, 96,192, 48,192, 57,192, 14,192, 0,192, 0,192, 0,192, 0,192, 1,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_101[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0,127,192, 96,192, 32,192, 49,128, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_102[] = { 7, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,254, 48, 48, 48, 22, 14, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_103[] = { 12, 0, 0, 0, 0, 31,128,120,224, 96, 48, 96, 16, 48, 48, 31,224, 63,128, 48, 0, 24, 0, 31, 0, 25,128, 48,192, 48,192, 48,192, 48,192, 25,128, 15,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_104[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120,240, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 56,224, 55,192, 51,128, 48, 0, 48, 0, 48, 0, 48, 0,112, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_105[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_106[] = { 6, 0, 0,192,224, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_107[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,121,240, 48,224, 49,192, 51,128, 55, 0, 54, 0, 60, 0, 52, 0, 50, 0, 51, 0, 49,128, 51,224, 48, 0, 48, 0, 48, 0, 48, 0,112, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_108[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_109[] = { 20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120,241,224, 48, 96,192, 48, 96,192, 48, 96,192, 48, 96,192, 48, 96,192, 48, 96,192, 48, 96,192, 48, 96,192, 56,241,192, 55,207,128,115,135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_110[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120,240, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 56,224, 55,192,115,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_111[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_112[] = { 12, 0, 0, 0, 0,120, 0, 48, 0, 48, 0, 48, 0, 48, 0, 55, 0, 57,192, 48,192, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48,192, 57,192,119, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_113[] = { 12, 0, 0, 0, 0, 1,224, 0,192, 0,192, 0,192, 0,192, 14,192, 57,192, 48,192, 96,192, 96,192, 96,192, 96,192, 96,192, 96,192, 48,192, 57,192, 14,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_114[] = { 8, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 59, 55,115, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_115[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 0, 99, 0, 65,128, 1,128, 3,128, 15, 0, 62, 0, 56, 0,112, 0, 97, 0, 51, 0, 31, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_116[] = { 7, 0, 0, 0, 0, 0, 0, 0, 28, 50, 48, 48, 48, 48, 48, 48, 48, 48, 48,254,112, 48, 16, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_117[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14,112, 31, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_118[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4, 0, 14, 0, 14, 0, 26, 0, 25, 0, 25, 0, 49, 0, 48,128, 48,128, 96,128, 96,192,241,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_119[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 4, 16, 0, 14, 56, 0, 14, 56, 0, 26, 40, 0, 26,100, 0, 25,100, 0, 49,100, 0, 48,194, 0, 48,194, 0, 96,194, 0, 96,195, 0,241,231,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_120[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120,240, 48, 96, 16,192, 25,192, 13,128, 7, 0, 6, 0, 13, 0, 28,128, 24,192, 48, 96,120,240, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_121[] = { 11, 0, 0, 0, 0,224, 0,240, 0, 24, 0, 8, 0, 12, 0, 4, 0, 14, 0, 14, 0, 26, 0, 25, 0, 25, 0, 49, 0, 48,128, 48,128, 96,128, 96,192,241,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_122[] = { 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,128, 97,128, 48,128, 56, 0, 24, 0, 28, 0, 12, 0, 14, 0, 7, 0, 67, 0, 97,128,127,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_123[] = { 10, 0, 0, 0, 0, 3,128, 6, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 8, 0, 24, 0, 16, 0, 96, 0, 16, 0, 24, 0, 8, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 6, 0, 3,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_124[] = { 6, 0, 0, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_125[] = { 10, 0, 0, 0, 0,112, 0, 24, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 4, 0, 6, 0, 2, 0, 1,128, 2, 0, 6, 0, 4, 0, 12, 0, 12, 0, 12, 0, 12, 0, 12, 0, 24, 0,112, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_126[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 65,192, 99,224, 62, 48, 28, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_127[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_128[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_129[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_130[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_131[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_132[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_133[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_134[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_135[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_136[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_137[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_138[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_139[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_140[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_141[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_142[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_143[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_144[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_145[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_146[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_147[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_148[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_149[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_150[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_151[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_152[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_153[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_154[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_155[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_156[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_157[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_158[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_159[] = { 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 64, 0, 64, 0, 0, 0, 0, 0, 85, 85, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_160[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_161[] = { 8, 0, 0, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 0, 0, 0, 12, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_162[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32, 0, 32, 0, 31, 0, 63,128, 56, 64,104, 0,100, 0,100, 0,100, 0, 98, 0, 98, 0, 33,192, 49,192, 15,128, 0,128, 0,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_163[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,115,192, 95, 96, 60, 32, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24, 0,126, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24,192, 12,192, 7,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_164[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 48,119,112, 63,224, 24,192, 48, 96, 48, 96, 48, 96, 48, 96, 24,192, 63,224,119,112, 96, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_165[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15,192, 3, 0, 3, 0, 3, 0, 3, 0, 31,224, 3, 0, 31,224, 3, 0, 7,128, 12,128, 12,192, 24, 64, 24, 96, 48, 32,112, 48,248,124, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_166[] = { 6, 0, 0, 0, 0, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 0, 0, 0, 48, 48, 48, 48, 48, 48, 48, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_167[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14, 0, 25, 0, 24,128, 1,128, 3,128, 7, 0, 14, 0, 29, 0, 56,128, 48,192, 32,192, 33,192, 19,128, 15, 0, 14, 0, 28, 0, 24, 0, 17,128, 9,128, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_168[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,102,102, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_169[] = { 19, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,248, 0, 14, 14, 0, 24, 3, 0, 48,225,128, 35,184,128, 98, 12,192, 70, 0, 64, 68, 0, 64, 68, 0, 64, 68, 0, 64, 70, 0, 64, 98, 12,192, 35,152,128, 48,241,128, 24, 3, 0, 14, 14, 0, 3,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_170[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0,118,204,204,124, 12,204,120, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_171[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 32, 6, 96, 12,192, 25,128, 51, 0, 51, 0, 25,128, 12,192, 6, 96, 2, 32, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_172[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 0, 24, 0, 24, 0, 24, 0, 24,127,248,127,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_173[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127, 0,127, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_174[] = { 19, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,248, 0, 14, 14, 0, 24, 3, 0, 48, 1,128, 35,140,128, 97, 24,192, 65, 16, 64, 65, 32, 64, 65,240, 64, 65, 24, 64, 65, 8, 64, 97, 8,192, 33, 24,128, 51,241,128, 24, 3, 0, 14, 14, 0, 3,248, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_175[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126,126, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_176[] = { 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 28, 0, 34, 0, 65, 0, 65, 0, 65, 0, 34, 0, 28, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_177[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,248,127,248, 0, 0, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0,127,248,127,248, 3, 0, 3, 0, 3, 0, 3, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_178[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 68, 32, 48, 16, 8, 12,140, 76, 56, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_179[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,112,136,140, 12, 8, 48, 8,140, 76, 56, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_180[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32, 24, 14, 6, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_181[] = { 13, 0, 0, 0, 0, 32, 0,112, 0, 96, 0, 32, 0, 32, 0, 46,112, 63, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_182[] = { 11, 0, 0, 0, 0, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 4,128, 12,128, 28,128, 60,128, 60,128,124,128,124,128,124,128, 60,128, 60,128, 28,128, 15,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_183[] = { 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_184[] = { 8, 0, 60,102, 6, 30, 24, 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_185[] = { 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 16, 16, 16, 16, 16, 16, 80, 48, 16, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_186[] = { 8, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 60,102,102,102,102,102, 60, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_187[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 34, 0, 51, 0, 25,128, 12,192, 6, 96, 6, 96, 12,192, 25,128, 51, 0, 34, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_188[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 2, 0, 8, 2, 0, 12,127,128, 4, 34, 0, 6, 50, 0, 3, 18, 0, 1, 10, 0,125,142, 0, 16,134, 0, 16,194, 0, 16, 96, 0, 16, 32, 0, 16, 48, 0, 16, 16, 0, 80, 24, 0, 48, 12, 0, 16, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_189[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 63, 0, 8, 17, 0, 12, 8, 0, 4, 12, 0, 6, 4, 0, 3, 2, 0, 1, 3, 0,125,163, 0, 16,147, 0, 16,206, 0, 16, 96, 0, 16, 32, 0, 16, 48, 0, 16, 16, 0, 80, 24, 0, 48, 12, 0, 16, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_190[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 24, 2, 0, 8, 2, 0, 12,127,128, 4, 34, 0, 6, 50, 0, 3, 18, 0, 1, 10, 0,113,142, 0,136,134, 0,140,194, 0, 12, 96, 0, 8, 32, 0, 48, 48, 0, 8, 16, 0,140, 24, 0, 76, 12, 0, 56, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_191[] = { 11, 0, 0, 0, 0, 31, 0, 49,128, 96,128, 97,128, 97,128,112, 0, 56, 0, 24, 0, 28, 0, 12, 0, 12, 0, 4, 0, 4, 0, 0, 0, 0, 0, 6, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_192[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 0, 32, 0, 0,192, 0, 3,128, 0, 3, 0, 0}; -static const GLubyte TimesRoman24_Character_193[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 1, 0, 0, 0,192, 0, 0,112, 0, 0, 48, 0}; -static const GLubyte TimesRoman24_Character_194[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 8, 16, 0, 6, 96, 0, 3,192, 0, 1,128, 0}; -static const GLubyte TimesRoman24_Character_195[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 7, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 0, 0, 0, 4,224, 0, 3,144, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_196[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 0, 0, 0, 0, 0, 0, 6, 48, 0, 6, 48, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_197[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,252, 31,128, 48, 6, 0, 16, 6, 0, 16, 12, 0, 24, 12, 0, 8, 12, 0, 15,248, 0, 12, 24, 0, 4, 24, 0, 4, 48, 0, 6, 48, 0, 2, 48, 0, 2, 96, 0, 1, 96, 0, 1,192, 0, 1,192, 0, 0,128, 0, 1,192, 0, 2, 32, 0, 2, 32, 0, 1,192, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_198[] = { 21, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,249,255,240, 48, 96, 48, 16, 96, 16, 16, 96, 16, 24, 96, 0, 8, 96, 0, 15,224,128, 12, 96,128, 4,127,128, 4, 96,128, 6, 96,128, 2, 96, 0, 2, 96, 0, 1, 96, 32, 1, 96, 32, 1,224, 96, 3,255,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_199[] = { 16, 0, 0, 3,192, 6, 96, 0, 96, 1,224, 1,128, 0,128, 3,240, 15, 28, 28, 4, 48, 2, 48, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 48, 2, 48, 2, 28, 6, 14, 30, 3,242, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_200[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 32, 24, 32, 31,224, 24, 32, 24, 32, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 0,128, 3, 0, 14, 0, 12, 0}; -static const GLubyte TimesRoman24_Character_201[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 32, 24, 32, 31,224, 24, 32, 24, 32, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 2, 0, 1,128, 0,224, 0, 96}; -static const GLubyte TimesRoman24_Character_202[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 32, 24, 32, 31,224, 24, 32, 24, 32, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 8, 16, 6, 96, 3,192, 1,128}; -static const GLubyte TimesRoman24_Character_203[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,252, 24, 12, 24, 4, 24, 4, 24, 0, 24, 0, 24, 32, 24, 32, 31,224, 24, 32, 24, 32, 24, 0, 24, 0, 24, 8, 24, 8, 24, 24,127,248, 0, 0, 0, 0, 12,192, 12,192, 0, 0}; -static const GLubyte TimesRoman24_Character_204[] = { 8, 0, 0, 0, 0, 0, 0, 0,126, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,126, 0, 4, 24,112, 96}; -static const GLubyte TimesRoman24_Character_205[] = { 8, 0, 0, 0, 0, 0, 0, 0,126, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,126, 0, 32, 24, 14, 6}; -static const GLubyte TimesRoman24_Character_206[] = { 8, 0, 0, 0, 0, 0, 0, 0, 63, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 63, 0, 64, 51, 30, 12}; -static const GLubyte TimesRoman24_Character_207[] = { 8, 0, 0, 0, 0, 0, 0, 0,126, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,126, 0, 0,102,102, 0}; -static const GLubyte TimesRoman24_Character_208[] = { 17, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,127,224, 0, 24, 56, 0, 24, 28, 0, 24, 6, 0, 24, 6, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0,255, 3, 0, 24, 3, 0, 24, 3, 0, 24, 3, 0, 24, 6, 0, 24, 6, 0, 24, 28, 0, 24, 56, 0,127,224, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_209[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,124, 6, 0, 16, 14, 0, 16, 14, 0, 16, 26, 0, 16, 50, 0, 16, 50, 0, 16, 98, 0, 16,194, 0, 16,194, 0, 17,130, 0, 19, 2, 0, 19, 2, 0, 22, 2, 0, 28, 2, 0, 28, 2, 0, 24, 2, 0,120, 15,128, 0, 0, 0, 0, 0, 0, 2,112, 0, 1,200, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_210[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0, 32, 0, 0,192, 0, 3,128, 0, 3, 0, 0}; -static const GLubyte TimesRoman24_Character_211[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0,128, 0, 0, 96, 0, 0, 56, 0, 0, 24, 0}; -static const GLubyte TimesRoman24_Character_212[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 4, 8, 0, 3, 48, 0, 1,224, 0, 0,192, 0}; -static const GLubyte TimesRoman24_Character_213[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0, 0, 0, 2,112, 0, 1,200, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_214[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 28, 0, 28, 14, 0, 48, 3, 0, 48, 3, 0, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 96, 1,128, 48, 3, 0, 48, 3, 0, 28, 14, 0, 14, 28, 0, 3,240, 0, 0, 0, 0, 0, 0, 0, 3, 48, 0, 3, 48, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_215[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32, 16, 48, 48, 24, 96, 12,192, 7,128, 3, 0, 7,128, 12,192, 24, 96, 48, 48, 32, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_216[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16, 0, 0, 19,240, 0, 14, 28, 0, 28, 14, 0, 52, 3, 0, 50, 3, 0, 97, 1,128, 97, 1,128, 96,129,128, 96,129,128, 96, 65,128, 96, 65,128, 96, 33,128, 48, 35, 0, 48, 19, 0, 28, 14, 0, 14, 28, 0, 3,242, 0, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_217[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 24, 0, 12, 4, 0, 24, 4, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0,126, 15,128, 0, 0, 0, 0, 32, 0, 0,192, 0, 3,128, 0, 3, 0, 0}; -static const GLubyte TimesRoman24_Character_218[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 24, 0, 12, 4, 0, 24, 4, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0,126, 15,128, 0, 0, 0, 0,128, 0, 0, 96, 0, 0, 56, 0, 0, 24, 0}; -static const GLubyte TimesRoman24_Character_219[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 24, 0, 12, 4, 0, 24, 4, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0,126, 15,128, 0, 0, 0, 4, 8, 0, 3, 48, 0, 1,224, 0, 0,192, 0}; -static const GLubyte TimesRoman24_Character_220[] = { 18, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3,240, 0, 14, 24, 0, 12, 4, 0, 24, 4, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0, 24, 2, 0,126, 15,128, 0, 0, 0, 0, 0, 0, 3, 24, 0, 3, 24, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_221[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 7,224, 1,128, 1,128, 1,128, 1,128, 1,128, 1,128, 3,192, 3, 64, 6, 96, 6, 32, 12, 48, 28, 16, 24, 24, 56, 8, 48, 12,252, 63, 0, 0, 1, 0, 0,192, 0,112, 0, 48}; -static const GLubyte TimesRoman24_Character_222[] = { 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,126, 0, 24, 0, 24, 0, 24, 0, 31,224, 24, 56, 24, 24, 24, 12, 24, 12, 24, 12, 24, 24, 24, 56, 31,224, 24, 0, 24, 0, 24, 0,126, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_223[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,115,128, 54, 64, 54, 96, 48, 96, 48, 96, 48,224, 48,192, 49,192, 51,128, 54, 0, 49,128, 48,192, 48,192, 48,192, 48,192, 25,128, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_224[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 2, 0, 12, 0, 56, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_225[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 8, 0, 6, 0, 3,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_226[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 33, 0, 18, 0, 30, 0, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_227[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 0, 0, 46, 0, 29, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_228[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 0, 0, 51, 0, 51, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_229[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,192,125,128, 99,128, 97,128, 97,128, 49,128, 29,128, 7,128, 1,128, 49,128, 51,128, 31, 0, 0, 0, 14, 0, 17, 0, 17, 0, 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_230[] = { 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 56,120,125,252, 99,194, 97,128, 97,128, 49,128, 29,128, 7,254, 1,134, 49,134, 51,204, 30,120, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_231[] = { 11, 0, 0, 30, 0, 51, 0, 3, 0, 15, 0, 12, 0, 4, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0, 96, 0, 96, 0, 32,192, 49,192, 15,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_232[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0,127,192, 96,192, 32,192, 49,128, 15, 0, 0, 0, 2, 0, 12, 0, 56, 0, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_233[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0,127,192, 96,192, 32,192, 49,128, 15, 0, 0, 0, 8, 0, 6, 0, 3,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_234[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0,127,192, 96,192, 32,192, 49,128, 15, 0, 0, 0, 16,128, 9, 0, 15, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_235[] = { 11, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 63,128, 56, 64,112, 0, 96, 0, 96, 0, 96, 0,127,192, 96,192, 32,192, 49,128, 15, 0, 0, 0, 0, 0, 25,128, 25,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_236[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 8, 48,224,192, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_237[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 64, 48, 28, 12, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_238[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0,132, 72,120, 48, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_239[] = { 6, 0, 0, 0, 0, 0, 0, 0,120, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48,112, 0, 0,204,204, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_240[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15,128, 99, 0, 30, 0, 15, 0, 56,192, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_241[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,120,240, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 56,224, 55,192,115,128, 0, 0, 0, 0, 19,128, 14, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_242[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 1, 0, 6, 0, 28, 0, 24, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_243[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 4, 0, 3, 0, 1,192, 0,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_244[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 16,128, 9, 0, 15, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_245[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 0, 0, 19,128, 14, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_246[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 57,192, 48,192, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 96, 48,192, 57,192, 15, 0, 0, 0, 0, 0, 25,128, 25,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_247[] = { 14, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 3, 0, 0, 0, 0, 0,127,248,127,248, 0, 0, 0, 0, 3, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_248[] = { 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 0,111, 0, 57,192, 56,192,104, 96,108, 96,100, 96,102, 96, 98, 96, 99, 96, 49,192, 57,192, 15, 96, 0, 96, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_249[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14,112, 31, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 1, 0, 6, 0, 28, 0, 24, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_250[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14,112, 31, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 4, 0, 3, 0, 1,192, 0,192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_251[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14,112, 31, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 16,128, 9, 0, 15, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_252[] = { 13, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 14,112, 31, 96, 56,224, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96,112,224, 0, 0, 0, 0, 25,128, 25,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_253[] = { 11, 0, 0, 0, 0,224, 0,240, 0, 24, 0, 8, 0, 12, 0, 4, 0, 14, 0, 14, 0, 26, 0, 25, 0, 25, 0, 49, 0, 48,128, 48,128, 96,128, 96,192,241,224, 0, 0, 8, 0, 6, 0, 3,128, 1,128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_254[] = { 12, 0, 0, 0, 0,120, 0, 48, 0, 48, 0, 48, 0, 48, 0, 55, 0, 57,192, 48,192, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48, 96, 48,192, 57,192, 55, 0, 48, 0, 48, 0, 48, 0, 48, 0,112, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -static const GLubyte TimesRoman24_Character_255[] = { 11, 0, 0, 0, 0,224, 0,240, 0, 24, 0, 8, 0, 12, 0, 4, 0, 14, 0, 14, 0, 26, 0, 25, 0, 25, 0, 49, 0, 48,128, 48,128, 96,128, 96,192,241,224, 0, 0, 0, 0, 51, 0, 51, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; - -/* The font characters mapping: */ -static const GLubyte* TimesRoman24_Character_Map[] = {TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032, - TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032, - TimesRoman24_Character_032,TimesRoman24_Character_033,TimesRoman24_Character_034,TimesRoman24_Character_035,TimesRoman24_Character_036,TimesRoman24_Character_037,TimesRoman24_Character_038,TimesRoman24_Character_039,TimesRoman24_Character_040,TimesRoman24_Character_041,TimesRoman24_Character_042,TimesRoman24_Character_043,TimesRoman24_Character_044,TimesRoman24_Character_045,TimesRoman24_Character_046,TimesRoman24_Character_047, - TimesRoman24_Character_048,TimesRoman24_Character_049,TimesRoman24_Character_050,TimesRoman24_Character_051,TimesRoman24_Character_052,TimesRoman24_Character_053,TimesRoman24_Character_054,TimesRoman24_Character_055,TimesRoman24_Character_056,TimesRoman24_Character_057,TimesRoman24_Character_058,TimesRoman24_Character_059,TimesRoman24_Character_060,TimesRoman24_Character_061,TimesRoman24_Character_062,TimesRoman24_Character_063, - TimesRoman24_Character_064,TimesRoman24_Character_065,TimesRoman24_Character_066,TimesRoman24_Character_067,TimesRoman24_Character_068,TimesRoman24_Character_069,TimesRoman24_Character_070,TimesRoman24_Character_071,TimesRoman24_Character_072,TimesRoman24_Character_073,TimesRoman24_Character_074,TimesRoman24_Character_075,TimesRoman24_Character_076,TimesRoman24_Character_077,TimesRoman24_Character_078,TimesRoman24_Character_079, - TimesRoman24_Character_080,TimesRoman24_Character_081,TimesRoman24_Character_082,TimesRoman24_Character_083,TimesRoman24_Character_084,TimesRoman24_Character_085,TimesRoman24_Character_086,TimesRoman24_Character_087,TimesRoman24_Character_088,TimesRoman24_Character_089,TimesRoman24_Character_090,TimesRoman24_Character_091,TimesRoman24_Character_092,TimesRoman24_Character_093,TimesRoman24_Character_094,TimesRoman24_Character_095, - TimesRoman24_Character_096,TimesRoman24_Character_097,TimesRoman24_Character_098,TimesRoman24_Character_099,TimesRoman24_Character_100,TimesRoman24_Character_101,TimesRoman24_Character_102,TimesRoman24_Character_103,TimesRoman24_Character_104,TimesRoman24_Character_105,TimesRoman24_Character_106,TimesRoman24_Character_107,TimesRoman24_Character_108,TimesRoman24_Character_109,TimesRoman24_Character_110,TimesRoman24_Character_111, - TimesRoman24_Character_112,TimesRoman24_Character_113,TimesRoman24_Character_114,TimesRoman24_Character_115,TimesRoman24_Character_116,TimesRoman24_Character_117,TimesRoman24_Character_118,TimesRoman24_Character_119,TimesRoman24_Character_120,TimesRoman24_Character_121,TimesRoman24_Character_122,TimesRoman24_Character_123,TimesRoman24_Character_124,TimesRoman24_Character_125,TimesRoman24_Character_126,TimesRoman24_Character_032, - TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032, - TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032,TimesRoman24_Character_032, - TimesRoman24_Character_160,TimesRoman24_Character_161,TimesRoman24_Character_162,TimesRoman24_Character_163,TimesRoman24_Character_164,TimesRoman24_Character_165,TimesRoman24_Character_166,TimesRoman24_Character_167,TimesRoman24_Character_168,TimesRoman24_Character_169,TimesRoman24_Character_170,TimesRoman24_Character_171,TimesRoman24_Character_172,TimesRoman24_Character_173,TimesRoman24_Character_174,TimesRoman24_Character_175, - TimesRoman24_Character_176,TimesRoman24_Character_177,TimesRoman24_Character_178,TimesRoman24_Character_179,TimesRoman24_Character_180,TimesRoman24_Character_181,TimesRoman24_Character_182,TimesRoman24_Character_183,TimesRoman24_Character_184,TimesRoman24_Character_185,TimesRoman24_Character_186,TimesRoman24_Character_187,TimesRoman24_Character_188,TimesRoman24_Character_189,TimesRoman24_Character_190,TimesRoman24_Character_191, - TimesRoman24_Character_192,TimesRoman24_Character_193,TimesRoman24_Character_194,TimesRoman24_Character_195,TimesRoman24_Character_196,TimesRoman24_Character_197,TimesRoman24_Character_198,TimesRoman24_Character_199,TimesRoman24_Character_200,TimesRoman24_Character_201,TimesRoman24_Character_202,TimesRoman24_Character_203,TimesRoman24_Character_204,TimesRoman24_Character_205,TimesRoman24_Character_206,TimesRoman24_Character_207, - TimesRoman24_Character_208,TimesRoman24_Character_209,TimesRoman24_Character_210,TimesRoman24_Character_211,TimesRoman24_Character_212,TimesRoman24_Character_213,TimesRoman24_Character_214,TimesRoman24_Character_215,TimesRoman24_Character_216,TimesRoman24_Character_217,TimesRoman24_Character_218,TimesRoman24_Character_219,TimesRoman24_Character_220,TimesRoman24_Character_221,TimesRoman24_Character_222,TimesRoman24_Character_223, - TimesRoman24_Character_224,TimesRoman24_Character_225,TimesRoman24_Character_226,TimesRoman24_Character_227,TimesRoman24_Character_228,TimesRoman24_Character_229,TimesRoman24_Character_230,TimesRoman24_Character_231,TimesRoman24_Character_232,TimesRoman24_Character_233,TimesRoman24_Character_234,TimesRoman24_Character_235,TimesRoman24_Character_236,TimesRoman24_Character_237,TimesRoman24_Character_238,TimesRoman24_Character_239, - TimesRoman24_Character_240,TimesRoman24_Character_241,TimesRoman24_Character_242,TimesRoman24_Character_243,TimesRoman24_Character_244,TimesRoman24_Character_245,TimesRoman24_Character_246,TimesRoman24_Character_247,TimesRoman24_Character_248,TimesRoman24_Character_249,TimesRoman24_Character_250,TimesRoman24_Character_251,TimesRoman24_Character_252,TimesRoman24_Character_253,TimesRoman24_Character_254,TimesRoman24_Character_255,NULL}; - -/* The font structure: */ -const SFG_Font fgFontTimesRoman24 = { "-adobe-times-medium-r-normal--24-240-75-75-p-124-iso8859-1", 256, 29, TimesRoman24_Character_Map, 0, 7 }; - diff --git a/examples/common/opengl-framework/freeglut/freeglut_gamemode.c b/examples/common/opengl-framework/freeglut/freeglut_gamemode.c deleted file mode 100644 index 803026b0..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_gamemode.c +++ /dev/null @@ -1,819 +0,0 @@ -/* - * freeglut_gamemode.c - * - * The game mode handling code. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 16 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* - * TODO BEFORE THE STABLE RELEASE: - * - * glutGameModeString() -- missing - * glutEnterGameMode() -- X11 version - * glutLeaveGameMode() -- is that correct? - * glutGameModeGet() -- is that correct? - */ - - -/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ - -#if TARGET_HOST_POSIX_X11 -static int xrandr_resize(int xsz, int ysz, int rate, int just_checking) -{ -#ifdef HAVE_X11_EXTENSIONS_XRANDR_H - int event_base, error_base, ver_major, ver_minor, use_rate; - XRRScreenConfiguration *xrr_config = 0; - Status result = -1; - - /* must check at runtime for the availability of the extension */ - if(!XRRQueryExtension(fgDisplay.Display, &event_base, &error_base)) { - return -1; - } - - XRRQueryVersion(fgDisplay.Display, &ver_major, &ver_minor); - - /* we only heed the rate if we CAN actually use it (Xrandr >= 1.1) and - * the user actually cares about it (rate > 0) - */ - use_rate = ( rate > 0 ) && ( ( ver_major >= 1 ) || - ( ( ver_major == 1 ) && ( ver_minor >= 1 ) ) ); - - /* this loop is only so that the whole thing will be repeated if someone - * else changes video mode between our query of the current information and - * the attempt to change it. - */ - do { - XRRScreenSize *ssizes; - short *rates; - Rotation rot; - int i, ssizes_count, rates_count, curr, res_idx = -1; - Time timestamp, cfg_timestamp; - - if(xrr_config) { - XRRFreeScreenConfigInfo(xrr_config); - } - - if(!(xrr_config = XRRGetScreenInfo(fgDisplay.Display, fgDisplay.RootWindow))) { - fgWarning("XRRGetScreenInfo failed"); - break; - } - ssizes = XRRConfigSizes(xrr_config, &ssizes_count); - curr = XRRConfigCurrentConfiguration(xrr_config, &rot); - timestamp = XRRConfigTimes(xrr_config, &cfg_timestamp); - - /* if either of xsz or ysz are unspecified, use the current values */ - if(xsz <= 0) - xsz = fgState.GameModeSize.X = ssizes[curr].width; - if(ysz <= 0) - ysz = fgState.GameModeSize.Y = ssizes[curr].height; - - - if(xsz == ssizes[curr].width && ysz == ssizes[curr].height) { - /* no need to switch, we're already in the requested resolution */ - res_idx = curr; - } else { - for(i=0; i= 1 ) || ( ( RANDR_MAJOR == 1 ) && ( RANDR_MINOR >= 1 ) ) - if(use_rate) { - rate = fgState.GameModeRefresh; - - /* for the selected resolution, let's find out if there is - * a matching refresh rate available. - */ - rates = XRRConfigRates(xrr_config, res_idx, &rates_count); - - for(i=0; i= 1 ) || ( ( RANDR_MAJOR == 1 ) && ( RANDR_MINOR >= 1 ) ) - if(use_rate) - result = XRRSetScreenConfigAndRate(fgDisplay.Display, xrr_config, - fgDisplay.RootWindow, res_idx, rot, rate, timestamp); - else -#endif - result = XRRSetScreenConfig(fgDisplay.Display, xrr_config, - fgDisplay.RootWindow, res_idx, rot, timestamp); - - } while(result == RRSetConfigInvalidTime); - - if(xrr_config) { - XRRFreeScreenConfigInfo(xrr_config); - } - - if(result == 0) { - return 0; - } - -#endif /* HAVE_X11_EXTENSIONS_XRANDR_H */ - return -1; -} -#endif /* TARGET_HOST_POSIX_X11 */ - -/* - * Remembers the current visual settings, so that - * we can change them and restore later... - */ -static void fghRememberState( void ) -{ -#if TARGET_HOST_POSIX_X11 - int event_base, error_base; - - /* - * Remember the current pointer location before going fullscreen - * for restoring it later: - */ - Window junk_window; - unsigned int junk_mask; - - XQueryPointer(fgDisplay.Display, fgDisplay.RootWindow, - &junk_window, &junk_window, - &fgDisplay.DisplayPointerX, &fgDisplay.DisplayPointerY, - &fgDisplay.DisplayPointerX, &fgDisplay.DisplayPointerY, &junk_mask); - -# ifdef HAVE_X11_EXTENSIONS_XRANDR_H - if(XRRQueryExtension(fgDisplay.Display, &event_base, &error_base)) { - XRRScreenConfiguration *xrr_config; - XRRScreenSize *ssizes; - Rotation rot; - int ssize_count, curr; - - if((xrr_config = XRRGetScreenInfo(fgDisplay.Display, fgDisplay.RootWindow))) { - ssizes = XRRConfigSizes(xrr_config, &ssize_count); - curr = XRRConfigCurrentConfiguration(xrr_config, &rot); - - fgDisplay.prev_xsz = ssizes[curr].width; - fgDisplay.prev_ysz = ssizes[curr].height; - fgDisplay.prev_refresh = -1; - -# if ( RANDR_MAJOR >= 1 ) || ( ( RANDR_MAJOR == 1 ) && ( RANDR_MINOR >= 1 ) ) - if(fgState.GameModeRefresh != -1) { - fgDisplay.prev_refresh = XRRConfigCurrentRate(xrr_config); - } -# endif - - fgDisplay.prev_size_valid = 1; - - XRRFreeScreenConfigInfo(xrr_config); - } - } -# endif - - /* - * This highly depends on the XFree86 extensions, - * not approved as X Consortium standards - */ -# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H - if(!XF86VidModeQueryExtension(fgDisplay.Display, &event_base, &error_base)) { - return; - } - - /* - * Remember the current ViewPort location of the screen to be able to - * restore the ViewPort on LeaveGameMode(): - */ - if( !XF86VidModeGetViewPort( - fgDisplay.Display, - fgDisplay.Screen, - &fgDisplay.DisplayViewPortX, - &fgDisplay.DisplayViewPortY ) ) - fgWarning( "XF86VidModeGetViewPort failed" ); - - - /* Query the current display settings: */ - fgDisplay.DisplayModeValid = - XF86VidModeGetModeLine( - fgDisplay.Display, - fgDisplay.Screen, - &fgDisplay.DisplayModeClock, - &fgDisplay.DisplayMode - ); - - if( !fgDisplay.DisplayModeValid ) - fgWarning( "XF86VidModeGetModeLine failed" ); -# endif - -#elif TARGET_HOST_MS_WINDOWS - -/* DEVMODE devMode; */ - - /* Grab the current desktop settings... */ - -/* hack to get around my stupid cross-gcc headers */ -#define FREEGLUT_ENUM_CURRENT_SETTINGS -1 - - EnumDisplaySettings( fgDisplay.DisplayName, FREEGLUT_ENUM_CURRENT_SETTINGS, - &fgDisplay.DisplayMode ); - - /* Make sure we will be restoring all settings needed */ - fgDisplay.DisplayMode.dmFields |= - DM_PELSWIDTH | DM_PELSHEIGHT | DM_BITSPERPEL | DM_DISPLAYFREQUENCY; - -#endif -} - -/* - * Restores the previously remembered visual settings - */ -static void fghRestoreState( void ) -{ -#if TARGET_HOST_POSIX_X11 - /* Restore the remembered pointer position: */ - XWarpPointer( - fgDisplay.Display, None, fgDisplay.RootWindow, 0, 0, 0, 0, - fgDisplay.DisplayPointerX, fgDisplay.DisplayPointerY - ); - - -# ifdef HAVE_X11_EXTENSIONS_XRANDR_H - if(fgDisplay.prev_size_valid) { - if(xrandr_resize(fgDisplay.prev_xsz, fgDisplay.prev_ysz, fgDisplay.prev_refresh, 0) != -1) { - fgDisplay.prev_size_valid = 0; -# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H - fgDisplay.DisplayModeValid = 0; -# endif - return; - } - } -# endif - - - -# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H - /* - * This highly depends on the XFree86 extensions, - * not approved as X Consortium standards - */ - - if( fgDisplay.DisplayModeValid ) - { - XF86VidModeModeInfo** displayModes; - int i, displayModesCount; - - if( !XF86VidModeGetAllModeLines( - fgDisplay.Display, - fgDisplay.Screen, - &displayModesCount, - &displayModes ) ) - { - fgWarning( "XF86VidModeGetAllModeLines failed" ); - return; - } - - - /* - * Check every of the modes looking for one that matches our demands. - * If we find one, switch to it and restore the remembered viewport. - */ - for( i = 0; i < displayModesCount; i++ ) - { - if(displayModes[ i ]->hdisplay == fgDisplay.DisplayMode.hdisplay && - displayModes[ i ]->vdisplay == fgDisplay.DisplayMode.vdisplay && - displayModes[ i ]->dotclock == fgDisplay.DisplayModeClock ) - { - if( !XF86VidModeSwitchToMode( - fgDisplay.Display, - fgDisplay.Screen, - displayModes[ i ] ) ) - { - fgWarning( "XF86VidModeSwitchToMode failed" ); - break; - } - - if( !XF86VidModeSetViewPort( - fgDisplay.Display, - fgDisplay.Screen, - fgDisplay.DisplayViewPortX, - fgDisplay.DisplayViewPortY ) ) - fgWarning( "XF86VidModeSetViewPort failed" ); - - - /* - * For the case this would be the last X11 call the application - * calls exit() we've to flush the X11 output queue to have the - * commands sent to the X server before the application exits. - */ - XFlush( fgDisplay.Display ); - - fgDisplay.DisplayModeValid = 0; -# ifdef HAVE_X11_EXTENSIONS_XRANDR_H - fgDisplay.prev_size_valid = 0; -# endif - - break; - } - } - XFree( displayModes ); - } - -# endif - -#elif TARGET_HOST_MS_WINDOWS - - /* Restore the previously remembered desktop display settings */ - ChangeDisplaySettingsEx( fgDisplay.DisplayName,&fgDisplay.DisplayMode, 0,0,0 ); - -#endif -} - -#if TARGET_HOST_POSIX_X11 -#ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H - -/* - * Checks a single display mode settings against user's preferences. - */ -static GLboolean fghCheckDisplayMode( int width, int height, int depth, int refresh ) -{ - /* The desired values should be stored in fgState structure... */ - return ( width == fgState.GameModeSize.X ) && - ( height == fgState.GameModeSize.Y ) && - ( depth == fgState.GameModeDepth ) && - ( refresh == fgState.GameModeRefresh ); -} - -/* - * Checks all display modes settings against user's preferences. - * Returns the mode number found or -1 if none could be found. - */ -static int fghCheckDisplayModes( GLboolean exactMatch, int displayModesCount, XF86VidModeModeInfo** displayModes ) -{ - int i; - for( i = 0; i < displayModesCount; i++ ) - { - /* Compute the displays refresh rate, dotclock comes in kHz. */ - int refresh = ( displayModes[ i ]->dotclock * 1000 ) / - ( displayModes[ i ]->htotal * displayModes[ i ]->vtotal ); - - if( fghCheckDisplayMode( displayModes[ i ]->hdisplay, - displayModes[ i ]->vdisplay, - fgState.GameModeDepth, - ( exactMatch ? refresh : fgState.GameModeRefresh ) ) ) { - if (!exactMatch) - { - /* Update the chosen refresh rate, otherwise a - * glutGameModeGet(GLUT_GAME_MODE_REFRESH_RATE) would not - * return the right values - */ - fgState.GameModeRefresh = refresh; - } - - return i; - } - } - return -1; -} - -#endif -#endif - -/* - * Changes the current display mode to match user's settings - */ -static GLboolean fghChangeDisplayMode( GLboolean haveToTest ) -{ - GLboolean success = GL_FALSE; -#if TARGET_HOST_POSIX_X11 - - /* first try to use XRandR, then fallback to XF86VidMode */ -# ifdef HAVE_X11_EXTENSIONS_XRANDR_H - if(xrandr_resize(fgState.GameModeSize.X, fgState.GameModeSize.Y, - fgState.GameModeRefresh, haveToTest) != -1) { - return GL_TRUE; - } -# endif - - - /* - * This highly depends on the XFree86 extensions, - * not approved as X Consortium standards - */ -# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H - - /* - * This is also used by applications which check modes by calling - * glutGameModeGet(GLUT_GAME_MODE_POSSIBLE), so allow the check: - */ - if( haveToTest || fgDisplay.DisplayModeValid ) - { - XF86VidModeModeInfo** displayModes; - int i, displayModesCount; - - /* If we don't have a valid modeline in the display structure, which - * can happen if this is called from glutGameModeGet instead of - * glutEnterGameMode, then we need to query the current mode, to make - * unspecified settings to default to their current values. - */ - if(!fgDisplay.DisplayModeValid) { - if(!XF86VidModeGetModeLine(fgDisplay.Display, fgDisplay.Screen, - &fgDisplay.DisplayModeClock, &fgDisplay.DisplayMode)) { - return success; - } - } - - if (fgState.GameModeSize.X == -1) - { - fgState.GameModeSize.X = fgDisplay.DisplayMode.hdisplay; - } - if (fgState.GameModeSize.Y == -1) - { - fgState.GameModeSize.Y = fgDisplay.DisplayMode.vdisplay; - } - if (fgState.GameModeDepth == -1) - { - /* can't get color depth from this, nor can we change it, do nothing - * TODO: get with XGetVisualInfo()? but then how to set? - */ - } - if (fgState.GameModeRefresh == -1) - { - /* Compute the displays refresh rate, dotclock comes in kHz. */ - int refresh = ( fgDisplay.DisplayModeClock * 1000 ) / - ( fgDisplay.DisplayMode.htotal * fgDisplay.DisplayMode.vtotal ); - - fgState.GameModeRefresh = refresh; - } - - /* query all possible display modes */ - if( !XF86VidModeGetAllModeLines( - fgDisplay.Display, - fgDisplay.Screen, - &displayModesCount, - &displayModes ) ) - { - fgWarning( "XF86VidModeGetAllModeLines failed" ); - return success; - } - - - /* - * Check every of the modes looking for one that matches our demands, - * ignoring the refresh rate if no exact match could be found. - */ - i = fghCheckDisplayModes( GL_TRUE, displayModesCount, displayModes ); - if( i < 0 ) { - i = fghCheckDisplayModes( GL_FALSE, displayModesCount, displayModes ); - } - success = ( i < 0 ) ? GL_FALSE : GL_TRUE; - - if( !haveToTest && success ) { - if( !XF86VidModeSwitchToMode( - fgDisplay.Display, - fgDisplay.Screen, - displayModes[ i ] ) ) - fgWarning( "XF86VidModeSwitchToMode failed" ); - } - - XFree( displayModes ); - } - -# endif - - -#elif TARGET_HOST_MS_WINDOWS - - DEVMODE devMode; - char *fggmstr = NULL; - char displayMode[300]; - - success = GL_FALSE; - - EnumDisplaySettings( fgDisplay.DisplayName, -1, &devMode ); - devMode.dmFields = 0; - - if (fgState.GameModeSize.X!=-1) - { - devMode.dmPelsWidth = fgState.GameModeSize.X; - devMode.dmFields |= DM_PELSWIDTH; - } - if (fgState.GameModeSize.Y!=-1) - { - devMode.dmPelsHeight = fgState.GameModeSize.Y; - devMode.dmFields |= DM_PELSHEIGHT; - } - if (fgState.GameModeDepth!=-1) - { - devMode.dmBitsPerPel = fgState.GameModeDepth; - devMode.dmFields |= DM_BITSPERPEL; - } - if (fgState.GameModeRefresh!=-1) - { - devMode.dmDisplayFrequency = fgState.GameModeRefresh; - devMode.dmFields |= DM_DISPLAYFREQUENCY; - } - - switch ( ChangeDisplaySettingsEx(fgDisplay.DisplayName, &devMode, NULL, haveToTest ? CDS_TEST : CDS_FULLSCREEN , NULL) ) - { - case DISP_CHANGE_SUCCESSFUL: - success = GL_TRUE; - - if (!haveToTest) - { - /* update vars in case if windows switched to proper mode */ - EnumDisplaySettings( fgDisplay.DisplayName, FREEGLUT_ENUM_CURRENT_SETTINGS, &devMode ); - fgState.GameModeSize.X = devMode.dmPelsWidth; - fgState.GameModeSize.Y = devMode.dmPelsHeight; - fgState.GameModeDepth = devMode.dmBitsPerPel; - fgState.GameModeRefresh = devMode.dmDisplayFrequency; - } - break; - case DISP_CHANGE_RESTART: - fggmstr = "The computer must be restarted for the graphics mode to work."; - break; - case DISP_CHANGE_BADFLAGS: - fggmstr = "An invalid set of flags was passed in."; - break; - case DISP_CHANGE_BADPARAM: - fggmstr = "An invalid parameter was passed in. This can include an invalid flag or combination of flags."; - break; - case DISP_CHANGE_FAILED: - fggmstr = "The display driver failed the specified graphics mode."; - break; - case DISP_CHANGE_BADMODE: - fggmstr = "The graphics mode is not supported."; - break; - default: - fggmstr = "Unknown error in graphics mode???"; /* dunno if it is possible,MSDN does not mention any other error */ - break; - } - - if ( !success ) - { - /* I'd rather get info whats going on in my program than wonder about */ - /* magic happenings behind my back, its lib for devels at last ;) */ - - /* append display mode to error to make things more informative */ - sprintf(displayMode,"%s Problem with requested mode: %ix%i:%i@%i", fggmstr, devMode.dmPelsWidth, devMode.dmPelsHeight, devMode.dmBitsPerPel, devMode.dmDisplayFrequency); - fgWarning(displayMode); - } -#endif - - return success; -} - - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * Sets the game mode display string - */ -void FGAPIENTRY glutGameModeString( const char* string ) -{ - int width = -1, height = -1, depth = -1, refresh = -1; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGameModeString" ); - - /* - * This one seems a bit easier than glutInitDisplayString. The bad thing - * about it that I was unable to find the game mode string definition, so - * that I assumed it is: "[width]x[height]:[depth]@[refresh rate]", which - * appears in all GLUT game mode programs I have seen to date. - */ - if( sscanf( string, "%ix%i:%i@%i", &width, &height, &depth, &refresh ) != - 4 ) - if( sscanf( string, "%ix%i:%i", &width, &height, &depth ) != 3 ) - if( sscanf( string, "%ix%i@%i", &width, &height, &refresh ) != 3 ) - if( sscanf( string, "%ix%i", &width, &height ) != 2 ) - if( sscanf( string, ":%i@%i", &depth, &refresh ) != 2 ) - if( sscanf( string, ":%i", &depth ) != 1 ) - if( sscanf( string, "@%i", &refresh ) != 1 ) - fgWarning( - "unable to parse game mode string `%s'", - string - ); - - /* All values not specified are now set to -1, which means those - * aspects of the current display mode are not changed in - * fghChangeDisplayMode() above. - */ - fgState.GameModeSize.X = width; - fgState.GameModeSize.Y = height; - fgState.GameModeDepth = depth; - fgState.GameModeRefresh = refresh; -} - - - -/* - * Enters the game mode - */ -int FGAPIENTRY glutEnterGameMode( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutEnterGameMode" ); - - if( fgStructure.GameModeWindow ) - fgAddToWindowDestroyList( fgStructure.GameModeWindow ); - else - fghRememberState( ); - - if( ! fghChangeDisplayMode( GL_FALSE ) ) - { - fgWarning( "failed to change screen settings" ); - return 0; - } - - fgStructure.GameModeWindow = fgCreateWindow( - NULL, "FREEGLUT", GL_TRUE, 0, 0, - GL_TRUE, fgState.GameModeSize.X, fgState.GameModeSize.Y, - GL_TRUE, GL_FALSE - ); - - fgStructure.GameModeWindow->State.Width = fgState.GameModeSize.X; - fgStructure.GameModeWindow->State.Height = fgState.GameModeSize.Y; - fgStructure.GameModeWindow->State.NeedToResize = GL_TRUE; - -#if TARGET_HOST_POSIX_X11 - - /* - * Sync needed to avoid a real race, the Xserver must have really created - * the window before we can grab the pointer into it: - */ - XSync( fgDisplay.Display, False ); - /* - * Grab the pointer to confine it into the window after the calls to - * XWrapPointer() which ensure that the pointer really enters the window. - * - * We also need to wait here until XGrabPointer() returns GrabSuccess, - * otherwise the new window is not viewable yet and if the next function - * (XSetInputFocus) is called with a not yet viewable window, it will exit - * the application which we have to aviod, so wait until it's viewable: - */ - while( GrabSuccess != XGrabPointer( - fgDisplay.Display, fgStructure.GameModeWindow->Window.Handle, - TRUE, - ButtonPressMask | ButtonReleaseMask | ButtonMotionMask - | PointerMotionMask, - GrabModeAsync, GrabModeAsync, - fgStructure.GameModeWindow->Window.Handle, None, CurrentTime) ) - usleep( 100 ); - /* - * Change input focus to the new window. This will exit the application - * if the new window is not viewable yet, see the XGrabPointer loop above. - */ - XSetInputFocus( - fgDisplay.Display, - fgStructure.GameModeWindow->Window.Handle, - RevertToNone, - CurrentTime - ); - - /* Move the Pointer to the middle of the fullscreen window */ - XWarpPointer( - fgDisplay.Display, - None, - fgDisplay.RootWindow, - 0, 0, 0, 0, - fgState.GameModeSize.X/2, fgState.GameModeSize.Y/2 - ); - -# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H - - if( fgDisplay.DisplayModeValid ) - { - int x, y; - Window child; - - /* Change to viewport to the window topleft edge: */ - if( !XF86VidModeSetViewPort( fgDisplay.Display, fgDisplay.Screen, 0, 0 ) ) - fgWarning( "XF86VidModeSetViewPort failed" ); - - /* - * Final window repositioning: It could be avoided using an undecorated - * window using override_redirect, but this * would possily require - * more changes and investigation. - */ - - /* Get the current postion of the drawable area on screen */ - XTranslateCoordinates( - fgDisplay.Display, - fgStructure.CurrentWindow->Window.Handle, - fgDisplay.RootWindow, - 0, 0, &x, &y, - &child - ); - - /* Move the decorataions out of the topleft corner of the display */ - XMoveWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle, - -x, -y); - } - -#endif - - /* Grab the keyboard, too */ - XGrabKeyboard( - fgDisplay.Display, - fgStructure.GameModeWindow->Window.Handle, - FALSE, - GrabModeAsync, GrabModeAsync, - CurrentTime - ); - -#endif - - return fgStructure.GameModeWindow->ID; -} - -/* - * Leaves the game mode - */ -void FGAPIENTRY glutLeaveGameMode( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutLeaveGameMode" ); - - freeglut_return_if_fail( fgStructure.GameModeWindow ); - - fgAddToWindowDestroyList( fgStructure.GameModeWindow ); - fgStructure.GameModeWindow = NULL; - -#if TARGET_HOST_POSIX_X11 - - XUngrabPointer( fgDisplay.Display, CurrentTime ); - XUngrabKeyboard( fgDisplay.Display, CurrentTime ); - -#endif - - fghRestoreState(); -} - -/* - * Returns information concerning the freeglut game mode - */ -int FGAPIENTRY glutGameModeGet( GLenum eWhat ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGameModeGet" ); - - switch( eWhat ) - { - case GLUT_GAME_MODE_ACTIVE: - return !!fgStructure.GameModeWindow; - - case GLUT_GAME_MODE_POSSIBLE: - return fghChangeDisplayMode( GL_TRUE ); - - case GLUT_GAME_MODE_WIDTH: - return fgState.GameModeSize.X; - - case GLUT_GAME_MODE_HEIGHT: - return fgState.GameModeSize.Y; - - case GLUT_GAME_MODE_PIXEL_DEPTH: - return fgState.GameModeDepth; - - case GLUT_GAME_MODE_REFRESH_RATE: - return fgState.GameModeRefresh; - - case GLUT_GAME_MODE_DISPLAY_CHANGED: - /* - * This is true if the game mode has been activated successfully.. - */ - return !!fgStructure.GameModeWindow; - } - - fgWarning( "Unknown gamemode get: %d", eWhat ); - return -1; -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_geometry.c b/examples/common/opengl-framework/freeglut/freeglut_geometry.c deleted file mode 100644 index 0f9f5c53..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_geometry.c +++ /dev/null @@ -1,1215 +0,0 @@ -/* - * freeglut_geometry.c - * - * Freeglut geometry rendering methods. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Fri Dec 3 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* - * TODO BEFORE THE STABLE RELEASE: - * - * Following functions have been contributed by Andreas Umbach. - * - * glutWireCube() -- looks OK - * glutSolidCube() -- OK - * - * Those functions have been implemented by John Fay. - * - * glutWireTorus() -- looks OK - * glutSolidTorus() -- looks OK - * glutWireDodecahedron() -- looks OK - * glutSolidDodecahedron() -- looks OK - * glutWireOctahedron() -- looks OK - * glutSolidOctahedron() -- looks OK - * glutWireTetrahedron() -- looks OK - * glutSolidTetrahedron() -- looks OK - * glutWireIcosahedron() -- looks OK - * glutSolidIcosahedron() -- looks OK - * - * The Following functions have been updated by Nigel Stewart, based - * on FreeGLUT 2.0.0 implementations: - * - * glutWireSphere() -- looks OK - * glutSolidSphere() -- looks OK - * glutWireCone() -- looks OK - * glutSolidCone() -- looks OK - */ - - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * Draws a wireframed cube. Code contributed by Andreas Umbach - */ -void FGAPIENTRY glutWireCube( GLdouble dSize ) -{ - double size = dSize * 0.5; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWireCube" ); - -# define V(a,b,c) glVertex3d( a size, b size, c size ); -# define N(a,b,c) glNormal3d( a, b, c ); - - /* PWO: I dared to convert the code to use macros... */ - glBegin( GL_LINE_LOOP ); N( 1.0, 0.0, 0.0); V(+,-,+); V(+,-,-); V(+,+,-); V(+,+,+); glEnd(); - glBegin( GL_LINE_LOOP ); N( 0.0, 1.0, 0.0); V(+,+,+); V(+,+,-); V(-,+,-); V(-,+,+); glEnd(); - glBegin( GL_LINE_LOOP ); N( 0.0, 0.0, 1.0); V(+,+,+); V(-,+,+); V(-,-,+); V(+,-,+); glEnd(); - glBegin( GL_LINE_LOOP ); N(-1.0, 0.0, 0.0); V(-,-,+); V(-,+,+); V(-,+,-); V(-,-,-); glEnd(); - glBegin( GL_LINE_LOOP ); N( 0.0,-1.0, 0.0); V(-,-,+); V(-,-,-); V(+,-,-); V(+,-,+); glEnd(); - glBegin( GL_LINE_LOOP ); N( 0.0, 0.0,-1.0); V(-,-,-); V(-,+,-); V(+,+,-); V(+,-,-); glEnd(); - -# undef V -# undef N -} - -/* - * Draws a solid cube. Code contributed by Andreas Umbach - */ -void FGAPIENTRY glutSolidCube( GLdouble dSize ) -{ - double size = dSize * 0.5; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSolidCube" ); - -# define V(a,b,c) glVertex3d( a size, b size, c size ); -# define N(a,b,c) glNormal3d( a, b, c ); - - /* PWO: Again, I dared to convert the code to use macros... */ - glBegin( GL_QUADS ); - N( 1.0, 0.0, 0.0); V(+,-,+); V(+,-,-); V(+,+,-); V(+,+,+); - N( 0.0, 1.0, 0.0); V(+,+,+); V(+,+,-); V(-,+,-); V(-,+,+); - N( 0.0, 0.0, 1.0); V(+,+,+); V(-,+,+); V(-,-,+); V(+,-,+); - N(-1.0, 0.0, 0.0); V(-,-,+); V(-,+,+); V(-,+,-); V(-,-,-); - N( 0.0,-1.0, 0.0); V(-,-,+); V(-,-,-); V(+,-,-); V(+,-,+); - N( 0.0, 0.0,-1.0); V(-,-,-); V(-,+,-); V(+,+,-); V(+,-,-); - glEnd(); - -# undef V -# undef N -} - -/* - * Compute lookup table of cos and sin values forming a cirle - * - * Notes: - * It is the responsibility of the caller to free these tables - * The size of the table is (n+1) to form a connected loop - * The last entry is exactly the same as the first - * The sign of n can be flipped to get the reverse loop - */ - -static void fghCircleTable(double **sint,double **cost,const int n) -{ - int i; - - /* Table size, the sign of n flips the circle direction */ - - const int size = abs(n); - - /* Determine the angle between samples */ - - const double angle = 2*M_PI/(double)( ( n == 0 ) ? 1 : n ); - - /* Allocate memory for n samples, plus duplicate of first entry at the end */ - - *sint = (double *) calloc(sizeof(double), size+1); - *cost = (double *) calloc(sizeof(double), size+1); - - /* Bail out if memory allocation fails, fgError never returns */ - - if (!(*sint) || !(*cost)) - { - free(*sint); - free(*cost); - fgError("Failed to allocate memory in fghCircleTable"); - } - - /* Compute cos and sin around the circle */ - - (*sint)[0] = 0.0; - (*cost)[0] = 1.0; - - for (i=1; i0)?1:0]; - r0 = 0.0; - r1 = sint2[(stacks>0)?1:0]; - - glBegin(GL_TRIANGLE_FAN); - - glNormal3d(0,0,1); - glVertex3d(0,0,radius); - - for (j=slices; j>=0; j--) - { - glNormal3d(cost1[j]*r1, sint1[j]*r1, z1 ); - glVertex3d(cost1[j]*r1*radius, sint1[j]*r1*radius, z1*radius); - } - - glEnd(); - - /* Cover each stack with a quad strip, except the top and bottom stacks */ - - for( i=1; i 0 ) ? stacks : 1 ); - const double rStep = base / ( ( stacks > 0 ) ? stacks : 1 ); - - /* Scaling factors for vertex normals */ - - const double cosn = ( height / sqrt ( height * height + base * base )); - const double sinn = ( base / sqrt ( height * height + base * base )); - - /* Pre-computed circle */ - - double *sint,*cost; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSolidCone" ); - - fghCircleTable(&sint,&cost,-slices); - - /* Cover the circular base with a triangle fan... */ - - z0 = 0.0; - z1 = zStep; - - r0 = base; - r1 = r0 - rStep; - - glBegin(GL_TRIANGLE_FAN); - - glNormal3d(0.0,0.0,-1.0); - glVertex3d(0.0,0.0, z0 ); - - for (j=0; j<=slices; j++) - glVertex3d(cost[j]*r0, sint[j]*r0, z0); - - glEnd(); - - /* Cover each stack with a quad strip, except the top stack */ - - for( i=0; i 0 ) ? stacks : 1 ); - const double rStep = base / ( ( stacks > 0 ) ? stacks : 1 ); - - /* Scaling factors for vertex normals */ - - const double cosn = ( height / sqrt ( height * height + base * base )); - const double sinn = ( base / sqrt ( height * height + base * base )); - - /* Pre-computed circle */ - - double *sint,*cost; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWireCone" ); - - fghCircleTable(&sint,&cost,-slices); - - /* Draw the stacks... */ - - for (i=0; i 0 ) ? stacks : 1 ); - - /* Pre-computed circle */ - - double *sint,*cost; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSolidCylinder" ); - - fghCircleTable(&sint,&cost,-slices); - - /* Cover the base and top */ - - glBegin(GL_TRIANGLE_FAN); - glNormal3d(0.0, 0.0, -1.0 ); - glVertex3d(0.0, 0.0, 0.0 ); - for (j=0; j<=slices; j++) - glVertex3d(cost[j]*radius, sint[j]*radius, 0.0); - glEnd(); - - glBegin(GL_TRIANGLE_FAN); - glNormal3d(0.0, 0.0, 1.0 ); - glVertex3d(0.0, 0.0, height); - for (j=slices; j>=0; j--) - glVertex3d(cost[j]*radius, sint[j]*radius, height); - glEnd(); - - /* Do the stacks */ - - z0 = 0.0; - z1 = zStep; - - for (i=1; i<=stacks; i++) - { - if (i==stacks) - z1 = height; - - glBegin(GL_QUAD_STRIP); - for (j=0; j<=slices; j++ ) - { - glNormal3d(cost[j], sint[j], 0.0 ); - glVertex3d(cost[j]*radius, sint[j]*radius, z0 ); - glVertex3d(cost[j]*radius, sint[j]*radius, z1 ); - } - glEnd(); - - z0 = z1; z1 += zStep; - } - - /* Release sin and cos tables */ - - free(sint); - free(cost); -} - -/* - * Draws a wire cylinder - */ -void FGAPIENTRY glutWireCylinder(GLdouble radius, GLdouble height, GLint slices, GLint stacks) -{ - int i,j; - - /* Step in z and radius as stacks are drawn. */ - - double z = 0.0; - const double zStep = height / ( ( stacks > 0 ) ? stacks : 1 ); - - /* Pre-computed circle */ - - double *sint,*cost; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWireCylinder" ); - - fghCircleTable(&sint,&cost,-slices); - - /* Draw the stacks... */ - - for (i=0; i<=stacks; i++) - { - if (i==stacks) - z = height; - - glBegin(GL_LINE_LOOP); - - for( j=0; j 0 ) - { - GLdouble local_offset[3] ; /* Use a local variable to avoid buildup of roundoff errors */ - num_levels -- ; - scale /= 2.0 ; - for ( i = 0 ; i < NUM_TETR_FACES ; i++ ) - { - local_offset[0] = offset[0] + scale * tet_r[i][0] ; - local_offset[1] = offset[1] + scale * tet_r[i][1] ; - local_offset[2] = offset[2] + scale * tet_r[i][2] ; - glutWireSierpinskiSponge ( num_levels, local_offset, scale ) ; - } - } -} - -void FGAPIENTRY glutSolidSierpinskiSponge ( int num_levels, GLdouble offset[3], GLdouble scale ) -{ - int i, j ; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSolidSierpinskiSponge" ); - - if ( num_levels == 0 ) - { - glBegin ( GL_TRIANGLES ) ; - - for ( i = 0 ; i < NUM_TETR_FACES ; i++ ) - { - glNormal3d ( -tet_r[i][0], -tet_r[i][1], -tet_r[i][2] ) ; - for ( j = 0; j < 3; j++ ) - { - double x = offset[0] + scale * tet_r[tet_i[i][j]][0] ; - double y = offset[1] + scale * tet_r[tet_i[i][j]][1] ; - double z = offset[2] + scale * tet_r[tet_i[i][j]][2] ; - glVertex3d ( x, y, z ) ; - } - } - - glEnd () ; - } - else if ( num_levels > 0 ) - { - GLdouble local_offset[3] ; /* Use a local variable to avoid buildup of roundoff errors */ - num_levels -- ; - scale /= 2.0 ; - for ( i = 0 ; i < NUM_TETR_FACES ; i++ ) - { - local_offset[0] = offset[0] + scale * tet_r[i][0] ; - local_offset[1] = offset[1] + scale * tet_r[i][1] ; - local_offset[2] = offset[2] + scale * tet_r[i][2] ; - glutSolidSierpinskiSponge ( num_levels, local_offset, scale ) ; - } - } -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_glutfont_definitions.c b/examples/common/opengl-framework/freeglut/freeglut_glutfont_definitions.c deleted file mode 100644 index 454ecb73..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_glutfont_definitions.c +++ /dev/null @@ -1,108 +0,0 @@ -/* - * freeglut_glutfont_definitions.c - * - * Bitmap and stroke fonts displaying. - * - * Copyright (c) 2003 Stephen J. Baker (whether he wants it or not). - * All Rights Reserved. - * Written by John F. Fay , who releases the - * copyright over to the "freeglut" project lead. - * Creation date: Mon July 21 2003 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -/* - * This file is necessary for the *nix version of "freeglut" because the - * original GLUT defined its font variables in rather an unusual way. - * Publicly, in "glut.h", they were defined as "void *". Privately, - * in one of the source code files, they were defined as pointers to a - * structure. Most compilers and linkers are satisfied with the "void *" - * and don't go any farther, but some of them balked. In particular, - * when compiling with "freeglut" and then trying to run using the GLUT - * ".so" library, some of them would give an error. So we are having to - * create this file to define the variables as pointers to an unusual - * structure to match GLUT. - */ - -/* - * freeglut_internal.h uses some GL types, but including the GL header portably - * is a bit tricky, so we include freeglut_std.h here, which contains the - * necessary machinery. But this poses another problem, caused by the ugly - * original defintion of the font constants in "classic" GLUT: They are defined - * as void* externally, so we move them temporarily out of the way by AN EXTREME - * CPP HACK. - */ - -#define glutStrokeRoman glutStrokeRomanIGNOREME -#define glutStrokeMonoRoman glutStrokeMonoRomanIGNOREME -#define glutBitmap9By15 glutBitmap9By15IGNOREME -#define glutBitmap8By13 glutBitmap8By13IGNOREME -#define glutBitmapTimesRoman10 glutBitmapTimesRoman10IGNOREME -#define glutBitmapTimesRoman24 glutBitmapTimesRoman24IGNOREME -#define glutBitmapHelvetica10 glutBitmapHelvetica10IGNOREME -#define glutBitmapHelvetica12 glutBitmapHelvetica12IGNOREME -#define glutBitmapHelvetica18 glutBitmapHelvetica18IGNOREME - -#include - -#undef glutStrokeRoman -#undef glutStrokeMonoRoman -#undef glutBitmap9By15 -#undef glutBitmap8By13 -#undef glutBitmapTimesRoman10 -#undef glutBitmapTimesRoman24 -#undef glutBitmapHelvetica10 -#undef glutBitmapHelvetica12 -#undef glutBitmapHelvetica18 - -#include "freeglut_internal.h" - -#if TARGET_HOST_POSIX_X11 - -struct freeglutStrokeFont -{ - const char *name ; - int num_chars ; - void *ch ; - float top ; - float bottom ; -}; - -struct freeglutBitmapFont -{ - const char *name ; - const int num_chars ; - const int first ; - const void *ch ; -}; - - -struct freeglutStrokeFont glutStrokeRoman ; -struct freeglutStrokeFont glutStrokeMonoRoman ; - -struct freeglutBitmapFont glutBitmap9By15 ; -struct freeglutBitmapFont glutBitmap8By13 ; -struct freeglutBitmapFont glutBitmapTimesRoman10 ; -struct freeglutBitmapFont glutBitmapTimesRoman24 ; -struct freeglutBitmapFont glutBitmapHelvetica10 ; -struct freeglutBitmapFont glutBitmapHelvetica12 ; -struct freeglutBitmapFont glutBitmapHelvetica18 ; - -#endif - diff --git a/examples/common/opengl-framework/freeglut/freeglut_init.c b/examples/common/opengl-framework/freeglut/freeglut_init.c deleted file mode 100644 index f59a456c..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_init.c +++ /dev/null @@ -1,1180 +0,0 @@ -/* - * freeglut_init.c - * - * Various freeglut initialization functions. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 2 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#define FREEGLUT_BUILDING_LIB -#include -#include "freeglut_internal.h" - -#if TARGET_HOST_POSIX_X11 -#include /* LONG_MAX */ -#endif - -/* - * TODO BEFORE THE STABLE RELEASE: - * - * fgDeinitialize() -- Win32's OK, X11 needs the OS-specific - * deinitialization done - * glutInitDisplayString() -- display mode string parsing - * - * Wouldn't it be cool to use gettext() for error messages? I just love - * bash saying "nie znaleziono pliku" instead of "file not found" :) - * Is gettext easily portable? - */ - -/* -- GLOBAL VARIABLES ----------------------------------------------------- */ - -/* - * A structure pointed by g_pDisplay holds all information - * regarding the display, screen, root window etc. - */ -SFG_Display fgDisplay; - -/* - * The settings for the current freeglut session - */ -SFG_State fgState = { { -1, -1, GL_FALSE }, /* Position */ - { 300, 300, GL_TRUE }, /* Size */ - GLUT_RGBA | GLUT_SINGLE | GLUT_DEPTH, /* DisplayMode */ - GL_FALSE, /* Initialised */ - GLUT_TRY_DIRECT_CONTEXT, /* DirectContext */ - GL_FALSE, /* ForceIconic */ - GL_FALSE, /* UseCurrentContext */ - GL_FALSE, /* GLDebugSwitch */ - GL_FALSE, /* XSyncSwitch */ - GLUT_KEY_REPEAT_ON, /* KeyRepeat */ - INVALID_MODIFIERS, /* Modifiers */ - 0, /* FPSInterval */ - 0, /* SwapCount */ - 0, /* SwapTime */ - 0, /* Time */ - { NULL, NULL }, /* Timers */ - { NULL, NULL }, /* FreeTimers */ - NULL, /* IdleCallback */ - 0, /* ActiveMenus */ - NULL, /* MenuStateCallback */ - NULL, /* MenuStatusCallback */ - { 640, 480, GL_TRUE }, /* GameModeSize */ - 16, /* GameModeDepth */ - 72, /* GameModeRefresh */ - GLUT_ACTION_EXIT, /* ActionOnWindowClose */ - GLUT_EXEC_STATE_INIT, /* ExecState */ - NULL, /* ProgramName */ - GL_FALSE, /* JoysticksInitialised */ - 0, /* NumActiveJoysticks */ - GL_FALSE, /* InputDevsInitialised */ - 0, /* MouseWheelTicks */ - 1, /* AuxiliaryBufferNumber */ - 4, /* SampleNumber */ - 1, /* MajorVersion */ - 0, /* MinorVersion */ - 0, /* ContextFlags */ - 0, /* ContextProfile */ - NULL, /* ErrorFunc */ - NULL /* WarningFunc */ -}; - - -/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ - -#if TARGET_HOST_POSIX_X11 - -/* Return the atom associated with "name". */ -static Atom fghGetAtom(const char * name) -{ - return XInternAtom(fgDisplay.Display, name, False); -} - -/* - * Check if "property" is set on "window". The property's values are returned - * through "data". If the property is set and is of type "type", return the - * number of elements in "data". Return zero otherwise. In both cases, use - * "Xfree()" to free "data". - */ -static int fghGetWindowProperty(Window window, - Atom property, - Atom type, - unsigned char ** data) -{ - /* - * Caller always has to use "Xfree()" to free "data", since - * "XGetWindowProperty() always allocates one extra byte in prop_return - * [i.e. "data"] (even if the property is zero length) [..]". - */ - - int status; /* Returned by "XGetWindowProperty". */ - - Atom type_returned; - int temp_format; /* Not used. */ - unsigned long number_of_elements; - unsigned long temp_bytes_after; /* Not used. */ - - - status = XGetWindowProperty(fgDisplay.Display, - window, - property, - 0, - LONG_MAX, - False, - type, - &type_returned, - &temp_format, - &number_of_elements, - &temp_bytes_after, - data); - - FREEGLUT_INTERNAL_ERROR_EXIT(status == Success, - "XGetWindowProperty failled", - "fghGetWindowProperty"); - - if (type_returned != type) - { - number_of_elements = 0; - } - - return number_of_elements; -} - -/* Check if the window manager is NET WM compliant. */ -static int fghNetWMSupported(void) -{ - Atom wm_check; - Window ** window_ptr_1; - - int number_of_windows; - int net_wm_supported; - - - net_wm_supported = 0; - - wm_check = fghGetAtom("_NET_SUPPORTING_WM_CHECK"); - window_ptr_1 = malloc(sizeof(Window *)); - - /* - * Check that the window manager has set this property on the root window. - * The property must be the ID of a child window. - */ - number_of_windows = fghGetWindowProperty(fgDisplay.RootWindow, - wm_check, - XA_WINDOW, - (unsigned char **) window_ptr_1); - if (number_of_windows == 1) - { - Window ** window_ptr_2; - - window_ptr_2 = malloc(sizeof(Window *)); - - /* Check that the window has the same property set to the same value. */ - number_of_windows = fghGetWindowProperty(**window_ptr_1, - wm_check, - XA_WINDOW, - (unsigned char **) window_ptr_2); - if ((number_of_windows == 1) && (**window_ptr_1 == **window_ptr_2)) - { - /* NET WM compliant */ - net_wm_supported = 1; - } - - XFree(*window_ptr_2); - free(window_ptr_2); - } - - XFree(*window_ptr_1); - free(window_ptr_1); - - return net_wm_supported; -} - -/* Check if "hint" is present in "property" for "window". */ -int fgHintPresent(Window window, Atom property, Atom hint) -{ - Atom *atoms; - int number_of_atoms; - int supported; - int i; - - supported = 0; - - number_of_atoms = fghGetWindowProperty(window, - property, - XA_ATOM, - (unsigned char **) &atoms); - for (i = 0; i < number_of_atoms; i++) - { - if (atoms[i] == hint) - { - supported = 1; - break; - } - } - - XFree(atoms); - return supported; -} - -#endif /* TARGET_HOST_POSIX_X11 */ - - -/* - * A call to this function should initialize all the display stuff... - */ -static void fghInitialize( const char* displayName ) -{ -#if TARGET_HOST_POSIX_X11 - fgDisplay.Display = XOpenDisplay( displayName ); - - if( fgDisplay.Display == NULL ) - fgError( "failed to open display '%s'", XDisplayName( displayName ) ); - - if( !glXQueryExtension( fgDisplay.Display, NULL, NULL ) ) - fgError( "OpenGL GLX extension not supported by display '%s'", - XDisplayName( displayName ) ); - - fgDisplay.Screen = DefaultScreen( fgDisplay.Display ); - fgDisplay.RootWindow = RootWindow( - fgDisplay.Display, - fgDisplay.Screen - ); - - fgDisplay.ScreenWidth = DisplayWidth( - fgDisplay.Display, - fgDisplay.Screen - ); - fgDisplay.ScreenHeight = DisplayHeight( - fgDisplay.Display, - fgDisplay.Screen - ); - - fgDisplay.ScreenWidthMM = DisplayWidthMM( - fgDisplay.Display, - fgDisplay.Screen - ); - fgDisplay.ScreenHeightMM = DisplayHeightMM( - fgDisplay.Display, - fgDisplay.Screen - ); - - fgDisplay.Connection = ConnectionNumber( fgDisplay.Display ); - - /* Create the window deletion atom */ - fgDisplay.DeleteWindow = fghGetAtom("WM_DELETE_WINDOW"); - - /* Create the state and full screen atoms */ - fgDisplay.State = None; - fgDisplay.StateFullScreen = None; - - if (fghNetWMSupported()) - { - const Atom supported = fghGetAtom("_NET_SUPPORTED"); - const Atom state = fghGetAtom("_NET_WM_STATE"); - - /* Check if the state hint is supported. */ - if (fgHintPresent(fgDisplay.RootWindow, supported, state)) - { - const Atom full_screen = fghGetAtom("_NET_WM_STATE_FULLSCREEN"); - - fgDisplay.State = state; - - /* Check if the window manager supports full screen. */ - /** Check "_NET_WM_ALLOWED_ACTIONS" on our window instead? **/ - if (fgHintPresent(fgDisplay.RootWindow, supported, full_screen)) - { - fgDisplay.StateFullScreen = full_screen; - } - } - } - -#elif TARGET_HOST_MS_WINDOWS - - WNDCLASS wc; - ATOM atom; - - /* What we need to do is to initialize the fgDisplay global structure here. */ - fgDisplay.Instance = GetModuleHandle( NULL ); - fgDisplay.DisplayName= displayName ? strdup(displayName) : 0 ; - atom = GetClassInfo( fgDisplay.Instance, _T("FREEGLUT"), &wc ); - - if( atom == 0 ) - { - ZeroMemory( &wc, sizeof(WNDCLASS) ); - - /* - * Each of the windows should have its own device context, and we - * want redraw events during Vertical and Horizontal Resizes by - * the user. - * - * XXX Old code had "| CS_DBCLCKS" commented out. Plans for the - * XXX future? Dead-end idea? - */ - wc.lpfnWndProc = fgWindowProc; - wc.cbClsExtra = 0; - wc.cbWndExtra = 0; - wc.hInstance = fgDisplay.Instance; - wc.hIcon = LoadIcon( fgDisplay.Instance, _T("GLUT_ICON") ); - -#if defined(_WIN32_WCE) - wc.style = CS_HREDRAW | CS_VREDRAW; -#else - wc.style = CS_OWNDC | CS_HREDRAW | CS_VREDRAW; - if (!wc.hIcon) - wc.hIcon = LoadIcon( NULL, IDI_WINLOGO ); -#endif - - wc.hCursor = LoadCursor( NULL, IDC_ARROW ); - wc.hbrBackground = NULL; - wc.lpszMenuName = NULL; - wc.lpszClassName = _T("FREEGLUT"); - - /* Register the window class */ - atom = RegisterClass( &wc ); - FREEGLUT_INTERNAL_ERROR_EXIT ( atom, "Window Class Not Registered", "fghInitialize" ); - } - - /* The screen dimensions can be obtained via GetSystemMetrics() calls */ - fgDisplay.ScreenWidth = GetSystemMetrics( SM_CXSCREEN ); - fgDisplay.ScreenHeight = GetSystemMetrics( SM_CYSCREEN ); - - { - HWND desktop = GetDesktopWindow( ); - HDC context = GetDC( desktop ); - - fgDisplay.ScreenWidthMM = GetDeviceCaps( context, HORZSIZE ); - fgDisplay.ScreenHeightMM = GetDeviceCaps( context, VERTSIZE ); - - ReleaseDC( desktop, context ); - } - /* If we have a DisplayName try to use it for metrics */ - if( fgDisplay.DisplayName ) - { - HDC context = CreateDC(fgDisplay.DisplayName,0,0,0); - if( context ) - { - fgDisplay.ScreenWidth = GetDeviceCaps( context, HORZRES ); - fgDisplay.ScreenHeight = GetDeviceCaps( context, VERTRES ); - fgDisplay.ScreenWidthMM = GetDeviceCaps( context, HORZSIZE ); - fgDisplay.ScreenHeightMM = GetDeviceCaps( context, VERTSIZE ); - DeleteDC(context); - } - else - fgWarning("fghInitialize: " - "CreateDC failed, Screen size info may be incorrect\n" - "This is quite likely caused by a bad '-display' parameter"); - - } - /* Set the timer granularity to 1 ms */ - timeBeginPeriod ( 1 ); - -#endif - - fgState.Initialised = GL_TRUE; - - /* Avoid registering atexit callback on Win32 as it results in an access - * violation due to calling into a module which has been unloaded. */ -#if ( TARGET_HOST_MS_WINDOWS == 0 ) - atexit(fgDeinitialize); -#endif - - /* InputDevice uses GlutTimerFunc(), so fgState.Initialised must be TRUE */ - fgInitialiseInputDevices(); -} - -/* - * Perform the freeglut deinitialization... - */ -void fgDeinitialize( void ) -{ - SFG_Timer *timer; - - if( !fgState.Initialised ) - { - return; - } - - /* If we're in game mode, we want to leave game mode */ - if( fgStructure.GameModeWindow ) { - glutLeaveGameMode(); - } - - /* If there was a menu created, destroy the rendering context */ - if( fgStructure.MenuContext ) - { -#if TARGET_HOST_POSIX_X11 - /* Note that the MVisualInfo is not owned by the MenuContext! */ - glXDestroyContext( fgDisplay.Display, fgStructure.MenuContext->MContext ); -#endif - free( fgStructure.MenuContext ); - fgStructure.MenuContext = NULL; - } - - fgDestroyStructure( ); - - while( ( timer = fgState.Timers.First) ) - { - fgListRemove( &fgState.Timers, &timer->Node ); - free( timer ); - } - - while( ( timer = fgState.FreeTimers.First) ) - { - fgListRemove( &fgState.FreeTimers, &timer->Node ); - free( timer ); - } - -#if !defined(_WIN32_WCE) - if ( fgState.JoysticksInitialised ) - fgJoystickClose( ); - - if ( fgState.InputDevsInitialised ) - fgInputDeviceClose( ); -#endif /* !defined(_WIN32_WCE) */ - fgState.JoysticksInitialised = GL_FALSE; - fgState.InputDevsInitialised = GL_FALSE; - - fgState.MouseWheelTicks = 0; - - fgState.MajorVersion = 1; - fgState.MinorVersion = 0; - fgState.ContextFlags = 0; - fgState.ContextProfile = 0; - - fgState.Initialised = GL_FALSE; - - fgState.Position.X = -1; - fgState.Position.Y = -1; - fgState.Position.Use = GL_FALSE; - - fgState.Size.X = 300; - fgState.Size.Y = 300; - fgState.Size.Use = GL_TRUE; - - fgState.DisplayMode = GLUT_RGBA | GLUT_SINGLE | GLUT_DEPTH; - - fgState.DirectContext = GLUT_TRY_DIRECT_CONTEXT; - fgState.ForceIconic = GL_FALSE; - fgState.UseCurrentContext = GL_FALSE; - fgState.GLDebugSwitch = GL_FALSE; - fgState.XSyncSwitch = GL_FALSE; - fgState.ActionOnWindowClose = GLUT_ACTION_EXIT; - fgState.ExecState = GLUT_EXEC_STATE_INIT; - - fgState.KeyRepeat = GLUT_KEY_REPEAT_ON; - fgState.Modifiers = INVALID_MODIFIERS; - - fgState.GameModeSize.X = 640; - fgState.GameModeSize.Y = 480; - fgState.GameModeDepth = 16; - fgState.GameModeRefresh = 72; - - fgListInit( &fgState.Timers ); - fgListInit( &fgState.FreeTimers ); - - fgState.IdleCallback = NULL; - fgState.MenuStateCallback = ( FGCBMenuState )NULL; - fgState.MenuStatusCallback = ( FGCBMenuStatus )NULL; - - fgState.SwapCount = 0; - fgState.SwapTime = 0; - fgState.FPSInterval = 0; - - if( fgState.ProgramName ) - { - free( fgState.ProgramName ); - fgState.ProgramName = NULL; - } - -#if TARGET_HOST_POSIX_X11 - - /* - * Make sure all X-client data we have created will be destroyed on - * display closing - */ - XSetCloseDownMode( fgDisplay.Display, DestroyAll ); - - /* - * Close the display connection, destroying all windows we have - * created so far - */ - XCloseDisplay( fgDisplay.Display ); - -#elif TARGET_HOST_MS_WINDOWS - if( fgDisplay.DisplayName ) - { - free( fgDisplay.DisplayName ); - fgDisplay.DisplayName = NULL; - } - - /* Reset the timer granularity */ - timeEndPeriod ( 1 ); - -#endif - - fgState.Initialised = GL_FALSE; -} - -/* - * Everything inside the following #ifndef is copied from the X sources. - */ - -#if TARGET_HOST_MS_WINDOWS - -/* - -Copyright 1985, 1986, 1987,1998 The Open Group - -Permission to use, copy, modify, distribute, and sell this software and its -documentation for any purpose is hereby granted without fee, provided that -the above copyright notice appear in all copies and that both that -copyright notice and this permission notice appear in supporting -documentation. - -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 OPEN GROUP 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. - -Except as contained in this notice, the name of The Open Group shall -not be used in advertising or otherwise to promote the sale, use or -other dealings in this Software without prior written authorization -from The Open Group. - -*/ - -#define NoValue 0x0000 -#define XValue 0x0001 -#define YValue 0x0002 -#define WidthValue 0x0004 -#define HeightValue 0x0008 -#define AllValues 0x000F -#define XNegative 0x0010 -#define YNegative 0x0020 - -/* - * XParseGeometry parses strings of the form - * "=x{+-}{+-}", where - * width, height, xoffset, and yoffset are unsigned integers. - * Example: "=80x24+300-49" - * The equal sign is optional. - * It returns a bitmask that indicates which of the four values - * were actually found in the string. For each value found, - * the corresponding argument is updated; for each value - * not found, the corresponding argument is left unchanged. - */ - -static int -ReadInteger(char *string, char **NextString) -{ - register int Result = 0; - int Sign = 1; - - if (*string == '+') - string++; - else if (*string == '-') - { - string++; - Sign = -1; - } - for (; (*string >= '0') && (*string <= '9'); string++) - { - Result = (Result * 10) + (*string - '0'); - } - *NextString = string; - if (Sign >= 0) - return Result; - else - return -Result; -} - -static int XParseGeometry ( - const char *string, - int *x, - int *y, - unsigned int *width, /* RETURN */ - unsigned int *height) /* RETURN */ -{ - int mask = NoValue; - register char *strind; - unsigned int tempWidth = 0, tempHeight = 0; - int tempX = 0, tempY = 0; - char *nextCharacter; - - if ( (string == NULL) || (*string == '\0')) - return mask; - if (*string == '=') - string++; /* ignore possible '=' at beg of geometry spec */ - - strind = (char *)string; - if (*strind != '+' && *strind != '-' && *strind != 'x') { - tempWidth = ReadInteger(strind, &nextCharacter); - if (strind == nextCharacter) - return 0; - strind = nextCharacter; - mask |= WidthValue; - } - - if (*strind == 'x' || *strind == 'X') { - strind++; - tempHeight = ReadInteger(strind, &nextCharacter); - if (strind == nextCharacter) - return 0; - strind = nextCharacter; - mask |= HeightValue; - } - - if ((*strind == '+') || (*strind == '-')) { - if (*strind == '-') { - strind++; - tempX = -ReadInteger(strind, &nextCharacter); - if (strind == nextCharacter) - return 0; - strind = nextCharacter; - mask |= XNegative; - } - else - { - strind++; - tempX = ReadInteger(strind, &nextCharacter); - if (strind == nextCharacter) - return 0; - strind = nextCharacter; - } - mask |= XValue; - if ((*strind == '+') || (*strind == '-')) { - if (*strind == '-') { - strind++; - tempY = -ReadInteger(strind, &nextCharacter); - if (strind == nextCharacter) - return 0; - strind = nextCharacter; - mask |= YNegative; - } - else - { - strind++; - tempY = ReadInteger(strind, &nextCharacter); - if (strind == nextCharacter) - return 0; - strind = nextCharacter; - } - mask |= YValue; - } - } - - /* If strind isn't at the end of the string the it's an invalid - geometry specification. */ - - if (*strind != '\0') return 0; - - if (mask & XValue) - *x = tempX; - if (mask & YValue) - *y = tempY; - if (mask & WidthValue) - *width = tempWidth; - if (mask & HeightValue) - *height = tempHeight; - return mask; -} -#endif - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * Perform initialization. This usually happens on the program startup - * and restarting after glutMainLoop termination... - */ -void FGAPIENTRY glutInit( int* pargc, char** argv ) -{ - char* displayName = NULL; - char* geometry = NULL; - int i, j, argc = *pargc; - - if( fgState.Initialised ) - fgError( "illegal glutInit() reinitialization attempt" ); - - if (pargc && *pargc && argv && *argv && **argv) - { - fgState.ProgramName = strdup (*argv); - - if( !fgState.ProgramName ) - fgError ("Could not allocate space for the program's name."); - } - - fgCreateStructure( ); - - /* Get start time */ - fgState.Time = fgSystemTime(); - - /* check if GLUT_FPS env var is set */ -#ifndef _WIN32_WCE - { - const char *fps = getenv( "GLUT_FPS" ); - - if( fps ) - { - int interval; - sscanf( fps, "%d", &interval ); - - if( interval <= 0 ) - fgState.FPSInterval = 5000; /* 5000 millisecond default */ - else - fgState.FPSInterval = interval; - } - } - - displayName = getenv( "DISPLAY" ); - - for( i = 1; i < argc; i++ ) - { - if( strcmp( argv[ i ], "-display" ) == 0 ) - { - if( ++i >= argc ) - fgError( "-display parameter must be followed by display name" ); - - displayName = argv[ i ]; - - argv[ i - 1 ] = NULL; - argv[ i ] = NULL; - ( *pargc ) -= 2; - } - else if( strcmp( argv[ i ], "-geometry" ) == 0 ) - { - if( ++i >= argc ) - fgError( "-geometry parameter must be followed by window " - "geometry settings" ); - - geometry = argv[ i ]; - - argv[ i - 1 ] = NULL; - argv[ i ] = NULL; - ( *pargc ) -= 2; - } - else if( strcmp( argv[ i ], "-direct" ) == 0) - { - if( fgState.DirectContext == GLUT_FORCE_INDIRECT_CONTEXT ) - fgError( "parameters ambiguity, -direct and -indirect " - "cannot be both specified" ); - - fgState.DirectContext = GLUT_FORCE_DIRECT_CONTEXT; - argv[ i ] = NULL; - ( *pargc )--; - } - else if( strcmp( argv[ i ], "-indirect" ) == 0 ) - { - if( fgState.DirectContext == GLUT_FORCE_DIRECT_CONTEXT ) - fgError( "parameters ambiguity, -direct and -indirect " - "cannot be both specified" ); - - fgState.DirectContext = GLUT_FORCE_INDIRECT_CONTEXT; - argv[ i ] = NULL; - (*pargc)--; - } - else if( strcmp( argv[ i ], "-iconic" ) == 0 ) - { - fgState.ForceIconic = GL_TRUE; - argv[ i ] = NULL; - ( *pargc )--; - } - else if( strcmp( argv[ i ], "-gldebug" ) == 0 ) - { - fgState.GLDebugSwitch = GL_TRUE; - argv[ i ] = NULL; - ( *pargc )--; - } - else if( strcmp( argv[ i ], "-sync" ) == 0 ) - { - fgState.XSyncSwitch = GL_TRUE; - argv[ i ] = NULL; - ( *pargc )--; - } - } - - /* Compact {argv}. */ - for( i = j = 1; i < *pargc; i++, j++ ) - { - /* Guaranteed to end because there are "*pargc" arguments left */ - while ( argv[ j ] == NULL ) - j++; - if ( i != j ) - argv[ i ] = argv[ j ]; - } - -#endif /* _WIN32_WCE */ - - /* - * Have the display created now. If there wasn't a "-display" - * in the program arguments, we will use the DISPLAY environment - * variable for opening the X display (see code above): - */ - fghInitialize( displayName ); - - /* - * Geometry parsing deffered until here because we may need the screen - * size. - */ - - if (geometry ) - { - unsigned int parsedWidth, parsedHeight; - int mask = XParseGeometry( geometry, - &fgState.Position.X, &fgState.Position.Y, - &parsedWidth, &parsedHeight ); - /* TODO: Check for overflow? */ - fgState.Size.X = parsedWidth; - fgState.Size.Y = parsedHeight; - - if( (mask & (WidthValue|HeightValue)) == (WidthValue|HeightValue) ) - fgState.Size.Use = GL_TRUE; - - if( mask & XNegative ) - fgState.Position.X += fgDisplay.ScreenWidth - fgState.Size.X; - - if( mask & YNegative ) - fgState.Position.Y += fgDisplay.ScreenHeight - fgState.Size.Y; - - if( (mask & (XValue|YValue)) == (XValue|YValue) ) - fgState.Position.Use = GL_TRUE; - } -} - -#if TARGET_HOST_MS_WINDOWS -void (__cdecl *__glutExitFunc)( int return_value ) = NULL; - -void FGAPIENTRY __glutInitWithExit( int *pargc, char **argv, void (__cdecl *exit_function)(int) ) -{ - __glutExitFunc = exit_function; - glutInit(pargc, argv); -} -#endif - -/* - * Undoes all the "glutInit" stuff - */ -void FGAPIENTRY glutExit ( void ) -{ - fgDeinitialize (); -} - -/* - * Sets the default initial window position for new windows - */ -void FGAPIENTRY glutInitWindowPosition( int x, int y ) -{ - fgState.Position.X = x; - fgState.Position.Y = y; - - if( ( x >= 0 ) && ( y >= 0 ) ) - fgState.Position.Use = GL_TRUE; - else - fgState.Position.Use = GL_FALSE; -} - -/* - * Sets the default initial window size for new windows - */ -void FGAPIENTRY glutInitWindowSize( int width, int height ) -{ - fgState.Size.X = width; - fgState.Size.Y = height; - - if( ( width > 0 ) && ( height > 0 ) ) - fgState.Size.Use = GL_TRUE; - else - fgState.Size.Use = GL_FALSE; -} - -/* - * Sets the default display mode for all new windows - */ -void FGAPIENTRY glutInitDisplayMode( unsigned int displayMode ) -{ - /* We will make use of this value when creating a new OpenGL context... */ - fgState.DisplayMode = displayMode; -} - - -/* -- INIT DISPLAY STRING PARSING ------------------------------------------ */ - -static char* Tokens[] = -{ - "alpha", "acca", "acc", "blue", "buffer", "conformant", "depth", "double", - "green", "index", "num", "red", "rgba", "rgb", "luminance", "stencil", - "single", "stereo", "samples", "slow", "win32pdf", "win32pfd", "xvisual", - "xstaticgray", "xgrayscale", "xstaticcolor", "xpseudocolor", - "xtruecolor", "xdirectcolor", - "xstaticgrey", "xgreyscale", "xstaticcolour", "xpseudocolour", - "xtruecolour", "xdirectcolour", "borderless", "aux" -}; -#define NUM_TOKENS (sizeof(Tokens) / sizeof(*Tokens)) - -void FGAPIENTRY glutInitDisplayString( const char* displayMode ) -{ - int glut_state_flag = 0 ; - /* - * Unpack a lot of options from a character string. The options are - * delimited by blanks or tabs. - */ - char *token ; - size_t len = strlen ( displayMode ); - char *buffer = (char *)malloc ( (len+1) * sizeof(char) ); - memcpy ( buffer, displayMode, len ); - buffer[len] = '\0'; - - token = strtok ( buffer, " \t" ); - - while ( token ) - { - /* Process this token */ - int i ; - - /* Temporary fix: Ignore any length specifications and at least - * process the basic token - * TODO: Fix this permanently - */ - size_t cleanlength = strcspn ( token, "=<>~!" ); - - for ( i = 0; i < NUM_TOKENS; i++ ) - { - if ( strncmp ( token, Tokens[i], cleanlength ) == 0 ) break ; - } - - switch ( i ) - { - case 0 : /* "alpha": Alpha color buffer precision in bits */ - glut_state_flag |= GLUT_ALPHA ; /* Somebody fix this for me! */ - break ; - - case 1 : /* "acca": Red, green, blue, and alpha accumulation buffer - precision in bits */ - break ; - - case 2 : /* "acc": Red, green, and blue accumulation buffer precision - in bits with zero bits alpha */ - glut_state_flag |= GLUT_ACCUM ; /* Somebody fix this for me! */ - break ; - - case 3 : /* "blue": Blue color buffer precision in bits */ - break ; - - case 4 : /* "buffer": Number of bits in the color index color buffer - */ - break ; - - case 5 : /* "conformant": Boolean indicating if the frame buffer - configuration is conformant or not */ - break ; - - case 6 : /* "depth": Number of bits of precsion in the depth buffer */ - glut_state_flag |= GLUT_DEPTH ; /* Somebody fix this for me! */ - break ; - - case 7 : /* "double": Boolean indicating if the color buffer is - double buffered */ - glut_state_flag |= GLUT_DOUBLE ; - break ; - - case 8 : /* "green": Green color buffer precision in bits */ - break ; - - case 9 : /* "index": Boolean if the color model is color index or not - */ - glut_state_flag |= GLUT_INDEX ; - break ; - - case 10 : /* "num": A special capability name indicating where the - value represents the Nth frame buffer configuration - matching the description string */ - break ; - - case 11 : /* "red": Red color buffer precision in bits */ - break ; - - case 12 : /* "rgba": Number of bits of red, green, blue, and alpha in - the RGBA color buffer */ - glut_state_flag |= GLUT_RGBA ; /* Somebody fix this for me! */ - break ; - - case 13 : /* "rgb": Number of bits of red, green, and blue in the - RGBA color buffer with zero bits alpha */ - glut_state_flag |= GLUT_RGB ; /* Somebody fix this for me! */ - break ; - - case 14 : /* "luminance": Number of bits of red in the RGBA and zero - bits of green, blue (alpha not specified) of color buffer - precision */ - glut_state_flag |= GLUT_LUMINANCE ; /* Somebody fix this for me! */ - break ; - - case 15 : /* "stencil": Number of bits in the stencil buffer */ - glut_state_flag |= GLUT_STENCIL; /* Somebody fix this for me! */ - break ; - - case 16 : /* "single": Boolean indicate the color buffer is single - buffered */ - glut_state_flag |= GLUT_SINGLE ; - break ; - - case 17 : /* "stereo": Boolean indicating the color buffer supports - OpenGL-style stereo */ - glut_state_flag |= GLUT_STEREO ; - break ; - - case 18 : /* "samples": Indicates the number of multisamples to use - based on GLX's SGIS_multisample extension (for - antialiasing) */ - glut_state_flag |= GLUT_MULTISAMPLE ; /*Somebody fix this for me!*/ - break ; - - case 19 : /* "slow": Boolean indicating if the frame buffer - configuration is slow or not */ - break ; - - case 20 : /* "win32pdf": (incorrect spelling but was there before */ - case 21 : /* "win32pfd": matches the Win32 Pixel Format Descriptor by - number */ -#if TARGET_HOST_MS_WINDOWS -#endif - break ; - - case 22 : /* "xvisual": matches the X visual ID by number */ -#if TARGET_HOST_POSIX_X11 -#endif - break ; - - case 23 : /* "xstaticgray": */ - case 29 : /* "xstaticgrey": boolean indicating if the frame buffer - configuration's X visual is of type StaticGray */ -#if TARGET_HOST_POSIX_X11 -#endif - break ; - - case 24 : /* "xgrayscale": */ - case 30 : /* "xgreyscale": boolean indicating if the frame buffer - configuration's X visual is of type GrayScale */ -#if TARGET_HOST_POSIX_X11 -#endif - break ; - - case 25 : /* "xstaticcolor": */ - case 31 : /* "xstaticcolour": boolean indicating if the frame buffer - configuration's X visual is of type StaticColor */ -#if TARGET_HOST_POSIX_X11 -#endif - break ; - - case 26 : /* "xpseudocolor": */ - case 32 : /* "xpseudocolour": boolean indicating if the frame buffer - configuration's X visual is of type PseudoColor */ -#if TARGET_HOST_POSIX_X11 -#endif - break ; - - case 27 : /* "xtruecolor": */ - case 33 : /* "xtruecolour": boolean indicating if the frame buffer - configuration's X visual is of type TrueColor */ -#if TARGET_HOST_POSIX_X11 -#endif - break ; - - case 28 : /* "xdirectcolor": */ - case 34 : /* "xdirectcolour": boolean indicating if the frame buffer - configuration's X visual is of type DirectColor */ -#if TARGET_HOST_POSIX_X11 -#endif - break ; - - case 35 : /* "borderless": windows should not have borders */ -#if TARGET_HOST_POSIX_X11 -#endif - break ; - - case 36 : /* "aux": some number of aux buffers */ - glut_state_flag |= GLUT_AUX; - break ; - - case 37 : /* Unrecognized */ - fgWarning ( "WARNING - Display string token not recognized: %s", - token ); - break ; - } - - token = strtok ( NULL, " \t" ); - } - - free ( buffer ); - - /* We will make use of this value when creating a new OpenGL context... */ - fgState.DisplayMode = glut_state_flag; -} - -/* -- SETTING OPENGL 3.0 CONTEXT CREATION PARAMETERS ---------------------- */ - -void FGAPIENTRY glutInitContextVersion( int majorVersion, int minorVersion ) -{ - /* We will make use of these valuse when creating a new OpenGL context... */ - fgState.MajorVersion = majorVersion; - fgState.MinorVersion = minorVersion; -} - - -void FGAPIENTRY glutInitContextFlags( int flags ) -{ - /* We will make use of this value when creating a new OpenGL context... */ - fgState.ContextFlags = flags; -} - -void FGAPIENTRY glutInitContextProfile( int profile ) -{ - /* We will make use of this value when creating a new OpenGL context... */ - fgState.ContextProfile = profile; -} - -/* -------------- User Defined Error/Warning Handler Support -------------- */ - -/* - * Sets the user error handler (note the use of va_list for the args to the fmt) - */ -void FGAPIENTRY glutInitErrorFunc( void (* vfgError) ( const char *fmt, va_list ap ) ) -{ - /* This allows user programs to handle freeglut errors */ - fgState.ErrorFunc = vfgError; -} - -/* - * Sets the user warning handler (note the use of va_list for the args to the fmt) - */ -void FGAPIENTRY glutInitWarningFunc( void (* vfgWarning) ( const char *fmt, va_list ap ) ) -{ - /* This allows user programs to handle freeglut warnings */ - fgState.WarningFunc = vfgWarning; -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_input_devices.c b/examples/common/opengl-framework/freeglut/freeglut_input_devices.c deleted file mode 100644 index 04f5fd5d..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_input_devices.c +++ /dev/null @@ -1,378 +0,0 @@ -/* - * freeglut_input_devices.c - * - * Handles miscellaneous input devices via direct serial-port access. - * Proper X11 XInput device support is not yet supported. - * Also lacks Mac support. - * - * Written by Joe Krahn 2005 - * - * Copyright (c) 2005 Stephen J. Baker. All Rights Reserved. - * - * 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 - * PAWEL W. OLSZTA OR STEPHEN J. BAKER 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. - */ - -#ifdef HAVE_CONFIG_H -# include "config.h" -#endif - -#include -#include "freeglut_internal.h" - -#if TARGET_HOST_POSIX_X11 -#ifdef HAVE_ERRNO_H -#include -#endif -#include -#include -#include -#include -#include -#include - -typedef struct { - int fd; - struct termios termio, termio_save; -} SERIALPORT; - -#elif TARGET_HOST_MS_WINDOWS -#include -#include -typedef struct { - HANDLE fh; - COMMTIMEOUTS timeouts_save; - DCB dcb_save; -} SERIALPORT; - -#endif - -/********************* Dialbox definitions ***********************/ - -#define DIAL_NUM_VALUATORS 8 - -/* dial parser state machine states */ -#define DIAL_NEW (-1) -#define DIAL_WHICH_DEVICE 0 -#define DIAL_VALUE_HIGH 1 -#define DIAL_VALUE_LOW 2 - -/* dial/button box commands */ -#define DIAL_INITIALIZE 0x20 -#define DIAL_SET_LEDS 0x75 -#define DIAL_SET_TEXT 0x61 -#define DIAL_SET_AUTO_DIALS 0x50 -#define DIAL_SET_AUTO_DELTA_DIALS 0x51 -#define DIAL_SET_FILTER 0x53 -#define DIAL_SET_BUTTONS_MOM_TYPE 0x71 -#define DIAL_SET_AUTO_MOM_BUTTONS 0x73 -#define DIAL_SET_ALL_LEDS 0x4b -#define DIAL_CLEAR_ALL_LEDS 0x4c - -/* dial/button box replies and events */ -#define DIAL_INITIALIZED 0x20 -#define DIAL_BASE 0x30 -#define DIAL_DELTA_BASE 0x40 -#define DIAL_PRESS_BASE 0xc0 -#define DIAL_RELEASE_BASE 0xe0 - -/* macros to determine reply type */ -#define IS_DIAL_EVENT(ch) (((ch)>=DIAL_BASE)&&((ch)=DIAL_PRESS_BASE)&&((ch)=DIAL_RELEASE_BASE)&&((ch)Callbacks[CB_Dials] */ - INVOKE_WCB ( *window,Dials, ( ((int*)enumerator->data)[0], ((int*)enumerator->data)[1]) ); - fgEnumSubWindows ( window, fghcbEnumDialCallbacks, enumerator ); -} - -static void send_dial_event ( int num, int value ) -{ - SFG_Enumerator enumerator; - int data[2]; - data[0] = num; - data[1] = value; - enumerator.found = GL_FALSE; - enumerator.data = data; - fgEnumWindows ( fghcbEnumDialCallbacks, &enumerator ); -} - -/********************************************************************/ -static void poll_dials ( int id ) -{ - int data; - static int dial_state = DIAL_NEW; - static int dial_which; - static int dial_value; - static int dials[8] = { 0, 0, 0, 0, 0, 0, 0, 0 }; - - if ( !dialbox_port ) return; - - while ( (data=serial_getchar(dialbox_port)) != EOF ) - { - if ( ( dial_state > DIAL_WHICH_DEVICE ) || IS_DIAL_EVENT ( data ) ) - { - switch ( dial_state ) - { - case DIAL_WHICH_DEVICE: - dial_which = data - DIAL_BASE; - dial_state++; - break; - case DIAL_VALUE_HIGH: - dial_value = ( data << 8 ); - dial_state++; - break; - case DIAL_VALUE_LOW: - dial_value |= data; - if ( dial_value & 0x8000 ) dial_value -= 0x10000; - dials[dial_which] = dial_value; - send_dial_event ( dial_which + 1, dial_value * 360 / 256 ); - dial_state = DIAL_WHICH_DEVICE; - break; - default: - /* error: Impossible state value! */ - break; - } - } - else if ( data == DIAL_INITIALIZED ) - { - fgState.InputDevsInitialised = GL_TRUE; - dial_state = DIAL_WHICH_DEVICE; - serial_putchar(dialbox_port,DIAL_SET_AUTO_DIALS); - serial_putchar(dialbox_port,0xff); - serial_putchar(dialbox_port,0xff); - } - else /* Unknown data; try flushing. */ - serial_flush(dialbox_port); - } - - glutTimerFunc ( 2, poll_dials, 0 ); -} - - -/******** OS Specific Serial I/O routines *******/ -#if TARGET_HOST_POSIX_X11 /* ==> Linux/BSD/UNIX POSIX serial I/O */ -static SERIALPORT *serial_open ( const char *device ) -{ - int fd; - struct termios termio; - SERIALPORT *port; - - fd = open(device, O_RDWR | O_NONBLOCK ); - if (fd <0) { - perror(device); - return NULL; - } - - port = malloc(sizeof(SERIALPORT)); - memset(port, 0, sizeof(SERIALPORT)); - port->fd = fd; - - /* save current port settings */ - tcgetattr(fd,&port->termio_save); - - memset(&termio, 0, sizeof(termio)); - termio.c_cflag = CS8 | CREAD | HUPCL ; - termio.c_iflag = IGNPAR | IGNBRK ; - termio.c_cc[VTIME] = 0; /* inter-character timer */ - termio.c_cc[VMIN] = 1; /* block read until 1 chars received, when blocking I/O */ - - cfsetispeed(&termio, B9600); - cfsetospeed(&termio, B9600); - tcsetattr(fd,TCSANOW,&termio); - - serial_flush(port); - return port; -} - -static void serial_close(SERIALPORT *port) -{ - if (port) - { - /* restore old port settings */ - tcsetattr(port->fd,TCSANOW,&port->termio_save); - close(port->fd); - free(port); - } -} - -static int serial_getchar(SERIALPORT *port) -{ - unsigned char ch; - if (!port) return EOF; - if (read(port->fd,&ch,1)) return ch; - return EOF; -} - -static int serial_putchar(SERIALPORT *port, unsigned char ch){ - if (!port) return 0; - return write(port->fd,&ch,1); -} - -static void serial_flush ( SERIALPORT *port ) -{ - tcflush ( port->fd, TCIOFLUSH ); -} - -#elif TARGET_HOST_MS_WINDOWS - -static SERIALPORT *serial_open(const char *device){ - HANDLE fh; - DCB dcb={sizeof(DCB)}; - COMMTIMEOUTS timeouts; - SERIALPORT *port; - - fh = CreateFile(device,GENERIC_READ|GENERIC_WRITE,0,NULL, - OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL); - if (!fh) return NULL; - - port = malloc(sizeof(SERIALPORT)); - ZeroMemory(port, sizeof(SERIALPORT)); - port->fh = fh; - - /* save current port settings */ - GetCommState(fh,&port->dcb_save); - GetCommTimeouts(fh,&port->timeouts_save); - - dcb.DCBlength=sizeof(DCB); - BuildCommDCB("96,n,8,1",&dcb); - SetCommState(fh,&dcb); - - ZeroMemory(&timeouts,sizeof(timeouts)); - timeouts.ReadTotalTimeoutConstant=1; - timeouts.WriteTotalTimeoutConstant=1; - SetCommTimeouts(fh,&timeouts); - - serial_flush(port); - - return port; -} - -static void serial_close(SERIALPORT *port){ - if (port){ - /* restore old port settings */ - SetCommState(port->fh,&port->dcb_save); - SetCommTimeouts(port->fh,&port->timeouts_save); - CloseHandle(port->fh); - free(port); - } -} - -static int serial_getchar(SERIALPORT *port){ - DWORD n; - unsigned char ch; - if (!port) return EOF; - if (!ReadFile(port->fh,&ch,1,&n,NULL)) return EOF; - if (n==1) return ch; - return EOF; -} - -static int serial_putchar(SERIALPORT *port, unsigned char ch){ - DWORD n; - if (!port) return 0; - return WriteFile(port->fh,&ch,1,&n,NULL); -} - -static void serial_flush ( SERIALPORT *port ) -{ - FlushFileBuffers(port->fh); -} - -#endif diff --git a/examples/common/opengl-framework/freeglut/freeglut_internal.h b/examples/common/opengl-framework/freeglut/freeglut_internal.h deleted file mode 100644 index 540f75f5..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_internal.h +++ /dev/null @@ -1,1023 +0,0 @@ -/* - * freeglut_internal.h - * - * The freeglut library private include file. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 2 1999 - * - * 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 - * PAWEL W. OLSZTA 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 FREEGLUT_INTERNAL_H -#define FREEGLUT_INTERNAL_H - -#ifdef HAVE_CONFIG_H -# include "config.h" -#endif - -/* XXX Update these for each release! */ -#define VERSION_MAJOR 2 -#define VERSION_MINOR 7 -#define VERSION_PATCH 0 - -/* Freeglut is intended to function under all Unix/X11 and Win32 platforms. */ -/* XXX: Don't all MS-Windows compilers (except Cygwin) have _WIN32 defined? - * XXX: If so, remove the first set of defined()'s below. - */ -#if !defined(TARGET_HOST_POSIX_X11) && !defined(TARGET_HOST_MS_WINDOWS) && !defined(TARGET_HOST_MAC_OSX) && !defined(TARGET_HOST_SOLARIS) -#if defined(_MSC_VER) || defined(__WATCOMC__) || defined(__MINGW32__) \ - || defined(_WIN32) || defined(_WIN32_WCE) \ - || ( defined(__CYGWIN__) && defined(X_DISPLAY_MISSING) ) -# define TARGET_HOST_MS_WINDOWS 1 - -#elif defined(__posix__) || defined(__unix__) || defined(__linux__) || defined(__sun) -# define TARGET_HOST_POSIX_X11 1 - -#elif defined(__APPLE__) -/* This is a placeholder until we get native OSX support ironed out -- JFF 11/18/09 */ -# define TARGET_HOST_POSIX_X11 1 -/* # define TARGET_HOST_MAC_OSX 1 */ - -#else -# error "Unrecognized target host!" - -#endif -#endif - -/* Detect both SunPro and gcc compilers on Sun Solaris */ -#if defined (__SVR4) && defined (__sun) -# define TARGET_HOST_SOLARIS 1 -#endif - -#ifndef TARGET_HOST_MS_WINDOWS -# define TARGET_HOST_MS_WINDOWS 0 -#endif - -#ifndef TARGET_HOST_POSIX_X11 -# define TARGET_HOST_POSIX_X11 0 -#endif - -#ifndef TARGET_HOST_MAC_OSX -# define TARGET_HOST_MAC_OSX 0 -#endif - -#ifndef TARGET_HOST_SOLARIS -# define TARGET_HOST_SOLARIS 0 -#endif - -/* -- FIXED CONFIGURATION LIMITS ------------------------------------------- */ - -#define FREEGLUT_MAX_MENUS 3 - -/* -- PLATFORM-SPECIFIC INCLUDES ------------------------------------------- */ - -/* All Win32 headers depend on the huge windows.h recursive include. - * Note: Lower-case header names are used, for best cross-platform - * compatibility. - */ -#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) -# include -# include -# include -/* CYGWIN does not have tchar.h, but has TEXT(x), defined in winnt.h. */ -# ifndef __CYGWIN__ -# include -# else -# define _TEXT(x) TEXT(x) -# define _T(x) TEXT(x) -# endif - -#elif TARGET_HOST_POSIX_X11 -# include -# include -# include -# include -# include -# ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H -# include -# endif -# ifdef HAVE_X11_EXTENSIONS_XRANDR_H -# include -# endif -/* If GLX is too old, we will fail during runtime when multisampling - is requested, but at least freeglut compiles. */ -# ifndef GLX_SAMPLE_BUFFERS -# define GLX_SAMPLE_BUFFERS 0x80A8 -# endif -# ifndef GLX_SAMPLES -# define GLX_SAMPLES 0x80A9 -# endif - -#endif - -/* These files should be available on every platform. */ -#include -#include -#include -#include -#include - -/* These are included based on autoconf directives. */ -#ifdef HAVE_SYS_TYPES_H -# include -#endif -#ifdef HAVE_UNISTD_H -# include -#endif -#ifdef TIME_WITH_SYS_TIME -# include -# include -#elif defined(HAVE_SYS_TIME_H) -# include -#else -# include -#endif - -/* -- AUTOCONF HACKS --------------------------------------------------------*/ - -/* XXX: Update autoconf to avoid these. - * XXX: Are non-POSIX platforms intended not to use autoconf? - * If so, perhaps there should be a config_guess.h for them. Alternatively, - * config guesses could be placed above, just after the config.h exclusion. - */ -#if defined(__FreeBSD__) || defined(__NetBSD__) -# define HAVE_USB_JS 1 -# if defined(__NetBSD__) || ( defined(__FreeBSD__) && __FreeBSD_version >= 500000) -# define HAVE_USBHID_H 1 -# endif -#endif - -#if TARGET_HOST_MS_WINDOWS -# define HAVE_VFPRINTF 1 -#endif - -/* MinGW may lack a prototype for ChangeDisplaySettingsEx() (depending on the version?) */ -#if TARGET_HOST_MS_WINDOWS && !defined(ChangeDisplaySettingsEx) -LONG WINAPI ChangeDisplaySettingsExA(LPCSTR,LPDEVMODEA,HWND,DWORD,LPVOID); -LONG WINAPI ChangeDisplaySettingsExW(LPCWSTR,LPDEVMODEW,HWND,DWORD,LPVOID); -# ifdef UNICODE -# define ChangeDisplaySettingsEx ChangeDisplaySettingsExW -# else -# define ChangeDisplaySettingsEx ChangeDisplaySettingsExA -# endif -#endif - -#if defined(_MSC_VER) || defined(__WATCOMC__) -/* strdup() is non-standard, for all but POSIX-2001 */ -#define strdup _strdup -#endif - -/* M_PI is non-standard (defined by BSD, not ISO-C) */ -#ifndef M_PI -# define M_PI 3.14159265358979323846 -#endif - -#ifdef HAVE_STDBOOL_H -# include -# ifndef TRUE -# define TRUE true -# endif -# ifndef FALSE -# define FALSE false -# endif -#else -# ifndef TRUE -# define TRUE 1 -# endif -# ifndef FALSE -# define FALSE 0 -# endif -#endif - -/* General defines */ - -#define INVALID_MODIFIERS 0xffffffff - -/* -- GLOBAL TYPE DEFINITIONS ---------------------------------------------- */ - -/* Freeglut callbacks type definitions */ -typedef void (* FGCBDisplay )( void ); -typedef void (* FGCBReshape )( int, int ); -typedef void (* FGCBVisibility )( int ); -typedef void (* FGCBKeyboard )( unsigned char, int, int ); -typedef void (* FGCBSpecial )( int, int, int ); -typedef void (* FGCBMouse )( int, int, int, int ); -typedef void (* FGCBMouseWheel )( int, int, int, int ); -typedef void (* FGCBMotion )( int, int ); -typedef void (* FGCBPassive )( int, int ); -typedef void (* FGCBEntry )( int ); -typedef void (* FGCBWindowStatus )( int ); -typedef void (* FGCBSelect )( int, int, int ); -typedef void (* FGCBJoystick )( unsigned int, int, int, int ); -typedef void (* FGCBKeyboardUp )( unsigned char, int, int ); -typedef void (* FGCBSpecialUp )( int, int, int ); -typedef void (* FGCBOverlayDisplay)( void ); -typedef void (* FGCBSpaceMotion )( int, int, int ); -typedef void (* FGCBSpaceRotation )( int, int, int ); -typedef void (* FGCBSpaceButton )( int, int ); -typedef void (* FGCBDials )( int, int ); -typedef void (* FGCBButtonBox )( int, int ); -typedef void (* FGCBTabletMotion )( int, int ); -typedef void (* FGCBTabletButton )( int, int, int, int ); -typedef void (* FGCBDestroy )( void ); - -typedef void (* FGCBMultiEntry )( int, int ); -typedef void (* FGCBMultiButton )( int, int, int, int, int ); -typedef void (* FGCBMultiMotion )( int, int, int ); -typedef void (* FGCBMultiPassive )( int, int, int ); - -/* The global callbacks type definitions */ -typedef void (* FGCBIdle )( void ); -typedef void (* FGCBTimer )( int ); -typedef void (* FGCBMenuState )( int ); -typedef void (* FGCBMenuStatus )( int, int, int ); - -/* The callback used when creating/using menus */ -typedef void (* FGCBMenu )( int ); - -/* The FreeGLUT error/warning handler type definition */ -typedef void (* FGError ) ( const char *fmt, va_list ap); -typedef void (* FGWarning ) ( const char *fmt, va_list ap); - - -/* A list structure */ -typedef struct tagSFG_List SFG_List; -struct tagSFG_List -{ - void *First; - void *Last; -}; - -/* A list node structure */ -typedef struct tagSFG_Node SFG_Node; -struct tagSFG_Node -{ - void *Next; - void *Prev; -}; - -/* A helper structure holding two ints and a boolean */ -typedef struct tagSFG_XYUse SFG_XYUse; -struct tagSFG_XYUse -{ - GLint X, Y; /* The two integers... */ - GLboolean Use; /* ...and a single boolean. */ -}; - -/* - * An enumeration containing the state of the GLUT execution: - * initializing, running, or stopping - */ -typedef enum -{ - GLUT_EXEC_STATE_INIT, - GLUT_EXEC_STATE_RUNNING, - GLUT_EXEC_STATE_STOP -} fgExecutionState ; - -/* This structure holds different freeglut settings */ -typedef struct tagSFG_State SFG_State; -struct tagSFG_State -{ - SFG_XYUse Position; /* The default windows' position */ - SFG_XYUse Size; /* The default windows' size */ - unsigned int DisplayMode; /* Display mode for new windows */ - - GLboolean Initialised; /* freeglut has been initialised */ - - int DirectContext; /* Direct rendering state */ - - GLboolean ForceIconic; /* New top windows are iconified */ - GLboolean UseCurrentContext; /* New windows share with current */ - - GLboolean GLDebugSwitch; /* OpenGL state debugging switch */ - GLboolean XSyncSwitch; /* X11 sync protocol switch */ - - int KeyRepeat; /* Global key repeat mode. */ - int Modifiers; /* Current ALT/SHIFT/CTRL state */ - - GLuint FPSInterval; /* Interval between FPS printfs */ - GLuint SwapCount; /* Count of glutSwapBuffer calls */ - GLuint SwapTime; /* Time of last SwapBuffers */ - - unsigned long Time; /* Time that glutInit was called */ - SFG_List Timers; /* The freeglut timer hooks */ - SFG_List FreeTimers; /* The unused timer hooks */ - - FGCBIdle IdleCallback; /* The global idle callback */ - - int ActiveMenus; /* Num. of currently active menus */ - FGCBMenuState MenuStateCallback; /* Menu callbacks are global */ - FGCBMenuStatus MenuStatusCallback; - - SFG_XYUse GameModeSize; /* Game mode screen's dimensions */ - int GameModeDepth; /* The pixel depth for game mode */ - int GameModeRefresh; /* The refresh rate for game mode */ - - int ActionOnWindowClose; /* Action when user closes window */ - - fgExecutionState ExecState; /* Used for GLUT termination */ - char *ProgramName; /* Name of the invoking program */ - GLboolean JoysticksInitialised; /* Only initialize if application calls for them */ - int NumActiveJoysticks; /* Number of active joysticks -- if zero, don't poll joysticks */ - GLboolean InputDevsInitialised; /* Only initialize if application calls for them */ - - int MouseWheelTicks; /* Number of ticks the mouse wheel has turned */ - - int AuxiliaryBufferNumber; /* Number of auxiliary buffers */ - int SampleNumber; /* Number of samples per pixel */ - - int MajorVersion; /* Major OpenGL context version */ - int MinorVersion; /* Minor OpenGL context version */ - int ContextFlags; /* OpenGL context flags */ - int ContextProfile; /* OpenGL context profile */ - FGError ErrorFunc; /* User defined error handler */ - FGWarning WarningFunc; /* User defined warning handler */ -}; - -/* The structure used by display initialization in freeglut_init.c */ -typedef struct tagSFG_Display SFG_Display; -struct tagSFG_Display -{ -#if TARGET_HOST_POSIX_X11 - Display* Display; /* The display we are being run in. */ - int Screen; /* The screen we are about to use. */ - Window RootWindow; /* The screen's root window. */ - int Connection; /* The display's connection number */ - Atom DeleteWindow; /* The window deletion atom */ - Atom State; /* The state atom */ - Atom StateFullScreen; /* The full screen atom */ - -#ifdef HAVE_X11_EXTENSIONS_XRANDR_H - int prev_xsz, prev_ysz; - int prev_refresh; - int prev_size_valid; -#endif /* HAVE_X11_EXTENSIONS_XRANDR_H */ - -#ifdef HAVE_X11_EXTENSIONS_XF86VMODE_H - /* - * XF86VidMode may be compilable even if it fails at runtime. Therefore, - * the validity of the VidMode has to be tracked - */ - int DisplayModeValid; /* Flag that indicates runtime status*/ - XF86VidModeModeLine DisplayMode; /* Current screen's display settings */ - int DisplayModeClock; /* The display mode's refresh rate */ - int DisplayViewPortX; /* saved X location of the viewport */ - int DisplayViewPortY; /* saved Y location of the viewport */ -#endif /* HAVE_X11_EXTENSIONS_XF86VMODE_H */ - - int DisplayPointerX; /* saved X location of the pointer */ - int DisplayPointerY; /* saved Y location of the pointer */ - -#elif TARGET_HOST_MS_WINDOWS - HINSTANCE Instance; /* The application's instance */ - DEVMODE DisplayMode; /* Desktop's display settings */ - char *DisplayName; /* Display name for multi display support*/ - -#endif - - int ScreenWidth; /* The screen's width in pixels */ - int ScreenHeight; /* The screen's height in pixels */ - int ScreenWidthMM; /* The screen's width in milimeters */ - int ScreenHeightMM; /* The screen's height in milimeters */ -}; - - -/* The user can create any number of timer hooks */ -typedef struct tagSFG_Timer SFG_Timer; -struct tagSFG_Timer -{ - SFG_Node Node; - int ID; /* The timer ID integer */ - FGCBTimer Callback; /* The timer callback */ - long TriggerTime; /* The timer trigger time */ -}; - -/* - * Make "freeglut" window handle and context types so that we don't need so - * much conditionally-compiled code later in the library. - */ -#if TARGET_HOST_POSIX_X11 - -typedef Window SFG_WindowHandleType ; -typedef GLXContext SFG_WindowContextType ; - -#elif TARGET_HOST_MS_WINDOWS - -typedef HWND SFG_WindowHandleType ; -typedef HGLRC SFG_WindowContextType ; - -#endif - -/* - * A window and its OpenGL context. The contents of this structure - * are highly dependant on the target operating system we aim at... - */ -typedef struct tagSFG_Context SFG_Context; -struct tagSFG_Context -{ - SFG_WindowHandleType Handle; /* The window's handle */ - SFG_WindowContextType Context; /* The window's OpenGL/WGL context */ - -#if TARGET_HOST_POSIX_X11 - GLXFBConfig* FBConfig; /* The window's FBConfig */ -#elif TARGET_HOST_MS_WINDOWS - HDC Device; /* The window's device context */ -#endif - - int DoubleBuffered; /* Treat the window as double-buffered */ -}; - -/* Window's state description. This structure should be kept portable. */ -typedef struct tagSFG_WindowState SFG_WindowState; -struct tagSFG_WindowState -{ - /* Note that on Windows, sizes always refer to the client area, thus without the window decorations */ - int Width; /* Window's width in pixels */ - int Height; /* The same about the height */ -#if TARGET_HOST_POSIX_X11 - int OldWidth; /* Window width from before a resize */ - int OldHeight; /* " height " " " " */ -#elif TARGET_HOST_MS_WINDOWS - RECT OldRect; /* window rect - stored before the window is made fullscreen */ - DWORD OldStyle; /* window style - stored before the window is made fullscreen */ -#endif - - GLboolean Redisplay; /* Do we have to redisplay? */ - GLboolean Visible; /* Is the window visible now */ - - int Cursor; /* The currently selected cursor */ - - long JoystickPollRate; /* The joystick polling rate */ - long JoystickLastPoll; /* When the last poll happened */ - - int MouseX, MouseY; /* The most recent mouse position */ - - GLboolean IgnoreKeyRepeat; /* Whether to ignore key repeat. */ - GLboolean KeyRepeating; /* Currently in repeat mode */ - - GLboolean NeedToResize; /* Do we need to resize the window? */ - - GLboolean IsFullscreen; /* is the window fullscreen? */ -}; - - -/* - * A generic function pointer. We should really use the GLUTproc type - * defined in freeglut_ext.h, but if we include that header in this file - * a bunch of other stuff (font-related) blows up! - */ -typedef void (*SFG_Proc)(); - - -/* - * SET_WCB() is used as: - * - * SET_WCB( window, cbname, func ); - * - * ...where {window} is the freeglut window to set the callback, - * {cbname} is the window-specific callback to set, - * {func} is a function-pointer. - * - * Originally, {FETCH_WCB( ... ) = func} was rather sloppily used, - * but this can cause warnings because the FETCH_WCB() macro type- - * casts its result, and a type-cast value shouldn't be an lvalue. - * - * The {if( FETCH_WCB( ... ) != func )} test is to do type-checking - * and for no other reason. Since it's hidden in the macro, the - * ugliness is felt to be rather benign. - */ -#define SET_WCB(window,cbname,func) \ -do \ -{ \ - if( FETCH_WCB( window, cbname ) != (SFG_Proc)(func) ) \ - (((window).CallBacks[CB_ ## cbname]) = (SFG_Proc)(func)); \ -} while( 0 ) - -/* - * FETCH_WCB() is used as: - * - * FETCH_WCB( window, cbname ); - * - * ...where {window} is the freeglut window to fetch the callback from, - * {cbname} is the window-specific callback to fetch. - * - * The result is correctly type-cast to the callback function pointer - * type. - */ -#define FETCH_WCB(window,cbname) \ - ((window).CallBacks[CB_ ## cbname]) - -/* - * INVOKE_WCB() is used as: - * - * INVOKE_WCB( window, cbname, ( arg_list ) ); - * - * ...where {window} is the freeglut window, - * {cbname} is the window-specific callback to be invoked, - * {(arg_list)} is the parameter list. - * - * The callback is invoked as: - * - * callback( arg_list ); - * - * ...so the parentheses are REQUIRED in the {arg_list}. - * - * NOTE that it does a sanity-check and also sets the - * current window. - * - */ -#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) /* FIXME: also WinCE? */ -#define INVOKE_WCB(window,cbname,arg_list) \ -do \ -{ \ - if( FETCH_WCB( window, cbname ) ) \ - { \ - FGCB ## cbname func = (FGCB ## cbname)(FETCH_WCB( window, cbname )); \ - fgSetWindow( &window ); \ - func arg_list; \ - } \ -} while( 0 ) -#else -#define INVOKE_WCB(window,cbname,arg_list) \ -do \ -{ \ - if( FETCH_WCB( window, cbname ) ) \ - { \ - fgSetWindow( &window ); \ - ((FGCB ## cbname)FETCH_WCB( window, cbname )) arg_list; \ - } \ -} while( 0 ) -#endif - -/* - * The window callbacks the user can supply us with. Should be kept portable. - * - * This enumeration provides the freeglut CallBack numbers. - * The symbolic constants are indices into a window's array of - * function callbacks. The names are formed by splicing a common - * prefix onto the callback's base name. (This was originally - * done so that an early stage of development could live side-by- - * side with the old callback code. The old callback code used - * the bare callback's name as a structure member, so I used a - * prefix for the array index name.) - * - * XXX For consistancy, perhaps the prefix should match the - * XXX FETCH* and INVOKE* macro suffices. I.e., WCB_, rather than - * XXX CB_. - */ -enum -{ - CB_Display, - CB_Reshape, - CB_Keyboard, - CB_KeyboardUp, - CB_Special, - CB_SpecialUp, - CB_Mouse, - CB_MouseWheel, - CB_Motion, - CB_Passive, - CB_Entry, - CB_Visibility, - CB_WindowStatus, - CB_Joystick, - CB_Destroy, - - /* MPX-related */ - CB_MultiEntry, - CB_MultiButton, - CB_MultiMotion, - CB_MultiPassive, - - /* Presently ignored */ - CB_Select, - CB_OverlayDisplay, - CB_SpaceMotion, /* presently implemented only on UNIX/X11 */ - CB_SpaceRotation, /* presently implemented only on UNIX/X11 */ - CB_SpaceButton, /* presently implemented only on UNIX/X11 */ - CB_Dials, - CB_ButtonBox, - CB_TabletMotion, - CB_TabletButton, - - /* Always make this the LAST one */ - TOTAL_CALLBACKS -}; - - -/* This structure holds the OpenGL rendering context for all the menu windows */ -typedef struct tagSFG_MenuContext SFG_MenuContext; -struct tagSFG_MenuContext -{ - SFG_WindowContextType MContext; /* The menu window's WGL context */ -}; - -/* This structure describes a menu */ -typedef struct tagSFG_Window SFG_Window; -typedef struct tagSFG_MenuEntry SFG_MenuEntry; -typedef struct tagSFG_Menu SFG_Menu; -struct tagSFG_Menu -{ - SFG_Node Node; - void *UserData; /* User data passed back at callback */ - int ID; /* The global menu ID */ - SFG_List Entries; /* The menu entries list */ - FGCBMenu Callback; /* The menu callback */ - FGCBDestroy Destroy; /* Destruction callback */ - GLboolean IsActive; /* Is the menu selected? */ - int Width; /* Menu box width in pixels */ - int Height; /* Menu box height in pixels */ - int X, Y; /* Menu box raster position */ - - SFG_MenuEntry *ActiveEntry; /* Currently active entry in the menu */ - SFG_Window *Window; /* Window for menu */ - SFG_Window *ParentWindow; /* Window in which the menu is invoked */ -}; - -/* This is a menu entry */ -struct tagSFG_MenuEntry -{ - SFG_Node Node; - int ID; /* The menu entry ID (local) */ - int Ordinal; /* The menu's ordinal number */ - char* Text; /* The text to be displayed */ - SFG_Menu* SubMenu; /* Optional sub-menu tree */ - GLboolean IsActive; /* Is the entry highlighted? */ - int Width; /* Label's width in pixels */ -}; - -/* - * A window, making part of freeglut windows hierarchy. - * Should be kept portable. - * - * NOTE that ActiveMenu is set to menu itself if the window is a menu. - */ -struct tagSFG_Window -{ - SFG_Node Node; - int ID; /* Window's ID number */ - - SFG_Context Window; /* Window and OpenGL context */ - SFG_WindowState State; /* The window state */ - SFG_Proc CallBacks[ TOTAL_CALLBACKS ]; /* Array of window callbacks */ - void *UserData ; /* For use by user */ - - SFG_Menu* Menu[ FREEGLUT_MAX_MENUS ]; /* Menus appended to window */ - SFG_Menu* ActiveMenu; /* The window's active menu */ - - SFG_Window* Parent; /* The parent to this window */ - SFG_List Children; /* The subwindows d.l. list */ - - GLboolean IsMenu; /* Set to 1 if we are a menu */ -}; - - -/* A linked list structure of windows */ -typedef struct tagSFG_WindowList SFG_WindowList ; -struct tagSFG_WindowList -{ - SFG_Node node; - SFG_Window *window ; -}; - -/* This holds information about all the windows, menus etc. */ -typedef struct tagSFG_Structure SFG_Structure; -struct tagSFG_Structure -{ - SFG_List Windows; /* The global windows list */ - SFG_List Menus; /* The global menus list */ - SFG_List WindowsToDestroy; - - SFG_Window* CurrentWindow; /* The currently set window */ - SFG_Menu* CurrentMenu; /* Same, but menu... */ - - SFG_MenuContext* MenuContext; /* OpenGL rendering context for menus */ - - SFG_Window* GameModeWindow; /* The game mode window */ - - int WindowID; /* The new current window ID */ - int MenuID; /* The new current menu ID */ -}; - -/* - * This structure is used for the enumeration purposes. - * You can easily extend its functionalities by declaring - * a structure containing enumerator's contents and custom - * data, then casting its pointer to (SFG_Enumerator *). - */ -typedef struct tagSFG_Enumerator SFG_Enumerator; -struct tagSFG_Enumerator -{ - GLboolean found; /* Used to terminate search */ - void* data; /* Custom data pointer */ -}; -typedef void (* FGCBenumerator )( SFG_Window *, SFG_Enumerator * ); - -/* The bitmap font structure */ -typedef struct tagSFG_Font SFG_Font; -struct tagSFG_Font -{ - char* Name; /* The source font name */ - int Quantity; /* Number of chars in font */ - int Height; /* Height of the characters */ - const GLubyte** Characters; /* The characters mapping */ - - float xorig, yorig; /* Relative origin of the character */ -}; - -/* The stroke font structures */ - -typedef struct tagSFG_StrokeVertex SFG_StrokeVertex; -struct tagSFG_StrokeVertex -{ - GLfloat X, Y; -}; - -typedef struct tagSFG_StrokeStrip SFG_StrokeStrip; -struct tagSFG_StrokeStrip -{ - int Number; - const SFG_StrokeVertex* Vertices; -}; - -typedef struct tagSFG_StrokeChar SFG_StrokeChar; -struct tagSFG_StrokeChar -{ - GLfloat Right; - int Number; - const SFG_StrokeStrip* Strips; -}; - -typedef struct tagSFG_StrokeFont SFG_StrokeFont; -struct tagSFG_StrokeFont -{ - char* Name; /* The source font name */ - int Quantity; /* Number of chars in font */ - GLfloat Height; /* Height of the characters */ - const SFG_StrokeChar** Characters; /* The characters mapping */ -}; - -/* -- GLOBAL VARIABLES EXPORTS --------------------------------------------- */ - -/* Freeglut display related stuff (initialized once per session) */ -extern SFG_Display fgDisplay; - -/* Freeglut internal structure */ -extern SFG_Structure fgStructure; - -/* The current freeglut settings */ -extern SFG_State fgState; - - -/* -- PRIVATE FUNCTION DECLARATIONS ---------------------------------------- */ - -/* - * A call to this function makes us sure that the Display and Structure - * subsystems have been properly initialized and are ready to be used - */ -#define FREEGLUT_EXIT_IF_NOT_INITIALISED( string ) \ - if ( ! fgState.Initialised ) \ - { \ - fgError ( " ERROR: Function <%s> called" \ - " without first calling 'glutInit'.", (string) ) ; \ - } - -#define FREEGLUT_INTERNAL_ERROR_EXIT_IF_NOT_INITIALISED( string ) \ - if ( ! fgState.Initialised ) \ - { \ - fgError ( " ERROR: Internal <%s> function called" \ - " without first calling 'glutInit'.", (string) ) ; \ - } - -#define FREEGLUT_INTERNAL_ERROR_EXIT( cond, string, function ) \ - if ( ! ( cond ) ) \ - { \ - fgError ( " ERROR: Internal error <%s> in function %s", \ - (string), (function) ) ; \ - } - -/* - * Following definitions are somewhat similiar to GLib's, - * but do not generate any log messages: - */ -#define freeglut_return_if_fail( expr ) \ - if( !(expr) ) \ - return; -#define freeglut_return_val_if_fail( expr, val ) \ - if( !(expr) ) \ - return val ; - -/* - * A call to those macros assures us that there is a current - * window set, respectively: - */ -#define FREEGLUT_EXIT_IF_NO_WINDOW( string ) \ - if ( ! fgStructure.CurrentWindow && \ - ( fgState.ActionOnWindowClose != GLUT_ACTION_CONTINUE_EXECUTION ) ) \ - { \ - fgError ( " ERROR: Function <%s> called" \ - " with no current window defined.", (string) ) ; \ - } - -/* - * The deinitialize function gets called on glutMainLoop() end. It should clean up - * everything inside of the freeglut - */ -void fgDeinitialize( void ); - -/* - * Those two functions are used to create/destroy the freeglut internal - * structures. This actually happens when calling glutInit() and when - * quitting the glutMainLoop() (which actually happens, when all windows - * have been closed). - */ -void fgCreateStructure( void ); -void fgDestroyStructure( void ); - -/* A helper function to check if a display mode is possible to use */ -#if TARGET_HOST_POSIX_X11 -GLXFBConfig* fgChooseFBConfig( int* numcfgs ); -#endif - -/* The window procedure for Win32 events handling */ -#if TARGET_HOST_MS_WINDOWS -LRESULT CALLBACK fgWindowProc( HWND hWnd, UINT uMsg, - WPARAM wParam, LPARAM lParam ); -void fgNewWGLCreateContext( SFG_Window* window ); -GLboolean fgSetupPixelFormat( SFG_Window* window, GLboolean checkOnly, - unsigned char layer_type ); -#endif - -/* - * Window creation, opening, closing and destruction. - * Also CallBack clearing/initialization. - * Defined in freeglut_structure.c, freeglut_window.c. - */ -SFG_Window* fgCreateWindow( SFG_Window* parent, const char* title, - GLboolean positionUse, int x, int y, - GLboolean sizeUse, int w, int h, - GLboolean gameMode, GLboolean isMenu ); -void fgSetWindow ( SFG_Window *window ); -void fgOpenWindow( SFG_Window* window, const char* title, - GLboolean positionUse, int x, int y, - GLboolean sizeUse, int w, int h, - GLboolean gameMode, GLboolean isSubWindow ); -void fgCloseWindow( SFG_Window* window ); -void fgAddToWindowDestroyList ( SFG_Window* window ); -void fgCloseWindows (); -void fgDestroyWindow( SFG_Window* window ); - -/* Menu creation and destruction. Defined in freeglut_structure.c */ -SFG_Menu* fgCreateMenu( FGCBMenu menuCallback ); -void fgDestroyMenu( SFG_Menu* menu ); - -/* Joystick device management functions, defined in freeglut_joystick.c */ -int fgJoystickDetect( void ); -void fgInitialiseJoysticks( void ); -void fgJoystickClose( void ); -void fgJoystickPollWindow( SFG_Window* window ); - -/* InputDevice Initialisation and Closure */ -int fgInputDeviceDetect( void ); -void fgInitialiseInputDevices( void ); -void fgInputDeviceClose( void ); - -/* spaceball device functions, defined in freeglut_spaceball.c */ -void fgInitialiseSpaceball( void ); -void fgSpaceballClose( void ); -void fgSpaceballSetWindow( SFG_Window *window ); - -int fgHasSpaceball( void ); -int fgSpaceballNumButtons( void ); - -#if TARGET_HOST_POSIX_X11 -int fgIsSpaceballXEvent( const XEvent *ev ); -void fgSpaceballHandleXEvent( const XEvent *ev ); -#endif - -/* Setting the cursor for a given window */ -void fgSetCursor ( SFG_Window *window, int cursorID ); - -/* - * Helper function to enumerate through all registered windows - * and one to enumerate all of a window's subwindows... - * - * The GFunc callback for those functions will be defined as: - * - * void enumCallback( gpointer window, gpointer enumerator ); - * - * where window is the enumerated (sub)window pointer (SFG_Window *), - * and userData is the a custom user-supplied pointer. Functions - * are defined and exported from freeglut_structure.c file. - */ -void fgEnumWindows( FGCBenumerator enumCallback, SFG_Enumerator* enumerator ); -void fgEnumSubWindows( SFG_Window* window, FGCBenumerator enumCallback, - SFG_Enumerator* enumerator ); - -#if TARGET_HOST_MS_WINDOWS -/* - * Helper functions for getting client area from the window rect - * and the window rect from the client area given the style of the window - * (or a valid window pointer from which the style can be queried). - */ -void fghComputeWindowRectFromClientArea_UseStyle ( const DWORD windowStyle , RECT *clientRect, BOOL posIsOutside ); -void fghComputeWindowRectFromClientArea_QueryWindow( const SFG_Window *window, RECT *clientRect, BOOL posIsOutside ); -void fghComputeClientAreaFromWindowRect ( const SFG_Window *window, RECT *windowRect, BOOL wantPosOutside ); -RECT fghGetClientArea ( const SFG_Window *window, BOOL wantPosOutside ); -void fghGetBorderWidth(const DWORD windowStyle, int* xBorderWidth, int* yBorderWidth); -#endif - -/* - * fgWindowByHandle returns a (SFG_Window *) value pointing to the - * first window in the queue matching the specified window handle. - * The function is defined in freeglut_structure.c file. - */ -SFG_Window* fgWindowByHandle( SFG_WindowHandleType hWindow ); - -/* - * This function is similiar to the previous one, except it is - * looking for a specified (sub)window identifier. The function - * is defined in freeglut_structure.c file. - */ -SFG_Window* fgWindowByID( int windowID ); - -/* - * Looks up a menu given its ID. This is easier than fgWindowByXXX - * as all menus are placed in a single doubly linked list... - */ -SFG_Menu* fgMenuByID( int menuID ); - -/* - * The menu activation and deactivation the code. This is the meat - * of the menu user interface handling code... - */ -void fgUpdateMenuHighlight ( SFG_Menu *menu ); -GLboolean fgCheckActiveMenu ( SFG_Window *window, int button, GLboolean pressed, - int mouse_x, int mouse_y ); -void fgDeactivateMenu( SFG_Window *window ); - -/* - * This function gets called just before the buffers swap, so that - * freeglut can display the pull-down menus via OpenGL. The function - * is defined in freeglut_menu.c file. - */ -void fgDisplayMenu( void ); - -/* Elapsed time as per glutGet(GLUT_ELAPSED_TIME). */ -long fgElapsedTime( void ); - -/* System time in milliseconds */ -long unsigned fgSystemTime(void); - -/* List functions */ -void fgListInit(SFG_List *list); -void fgListAppend(SFG_List *list, SFG_Node *node); -void fgListRemove(SFG_List *list, SFG_Node *node); -int fgListLength(SFG_List *list); -void fgListInsert(SFG_List *list, SFG_Node *next, SFG_Node *node); - -/* Error Message functions */ -void fgError( const char *fmt, ... ); -void fgWarning( const char *fmt, ... ); - -/* - * Check if "hint" is present in "property" for "window". See freeglut_init.c - */ -#if TARGET_HOST_POSIX_X11 -int fgHintPresent(Window window, Atom property, Atom hint); - -/* Handler for X extension Events */ -#ifdef HAVE_X11_EXTENSIONS_XINPUT2_H - void fgHandleExtensionEvents( XEvent * ev ); - void fgRegisterDevices( Display* dpy, Window* win ); -#endif - -#endif - -SFG_Proc fghGetProcAddress( const char *procName ); - -#if TARGET_HOST_MS_WINDOWS -extern void (__cdecl *__glutExitFunc)( int return_value ); -#endif - -#endif /* FREEGLUT_INTERNAL_H */ - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_joystick.c b/examples/common/opengl-framework/freeglut/freeglut_joystick.c deleted file mode 100644 index f48dc3f2..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_joystick.c +++ /dev/null @@ -1,1800 +0,0 @@ -/* - * freeglut_joystick.c - * - * Joystick handling code - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Steve Baker, - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -/* - * FreeBSD port by Stephen Montgomery-Smith - * - * Redone by John Fay 2/4/04 with another look from the PLIB "js" library. - * Many thanks for Steve Baker for permission to pull from that library. - */ - -#include -#include "freeglut_internal.h" -#ifdef HAVE_SYS_PARAM_H -# include -#endif - -/* - * Initial defines from "js.h" starting around line 33 with the existing "freeglut_joystick.c" - * interspersed - */ - -/* XXX It might be better to poll the operating system for the numbers of buttons and - * XXX axes and then dynamically allocate the arrays. - */ -#define _JS_MAX_BUTTONS 32 - -#if TARGET_HOST_MACINTOSH -# define _JS_MAX_AXES 9 -# include -#endif - -#if TARGET_HOST_MAC_OSX -# define _JS_MAX_AXES 16 -# include -# include -# include -#endif - -#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) -# define _JS_MAX_AXES 8 -# include -# include -# include - -#endif - -#if TARGET_HOST_POSIX_X11 -# define _JS_MAX_AXES 16 -# ifdef HAVE_SYS_IOCTL_H -# include -# endif -# ifdef HAVE_FCNTL_H -# include -# endif -# ifdef HAVE_ERRNO_H -# include -# include -# endif -# if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__NetBSD__) -/* XXX The below hack is done until freeglut's autoconf is updated. */ -# define HAVE_USB_JS 1 - -# if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) -# include -# else -/* - * XXX NetBSD/amd64 systems may find that they have to steal the - * XXX /usr/include/machine/joystick.h from a NetBSD/i386 system. - * XXX I cannot comment whether that works for the interface, but - * XXX it lets you compile...(^& I do not think that we can do away - * XXX with this header. - */ -# include /* For analog joysticks */ -# endif -# define JS_DATA_TYPE joystick -# define JS_RETURN (sizeof(struct JS_DATA_TYPE)) -# endif - -# if defined(__linux__) -# include - -/* check the joystick driver version */ -# if defined(JS_VERSION) && JS_VERSION >= 0x010000 -# define JS_NEW -# endif -# else /* Not BSD or Linux */ -# ifndef JS_RETURN - - /* - * We'll put these values in and that should - * allow the code to at least compile when there is - * no support. The JS open routine should error out - * and shut off all the code downstream anyway and if - * the application doesn't use a joystick we'll be fine. - */ - - struct JS_DATA_TYPE - { - int buttons; - int x; - int y; - }; - -# define JS_RETURN (sizeof(struct JS_DATA_TYPE)) -# endif -# endif -#endif - -#define JS_TRUE 1 -#define JS_FALSE 0 - -/* BSD defines from "jsBSD.cxx" around lines 42-270 */ - -#if defined(__NetBSD__) || defined(__FreeBSD__) || defined(__FreeBSD_kernel__) - -# ifdef HAVE_USB_JS -# if defined(__NetBSD__) -/* XXX The below hack is done until freeglut's autoconf is updated. */ -# define HAVE_USBHID_H 1 -# ifdef HAVE_USBHID_H -# include -# else -# include -# endif -# elif defined(__FreeBSD__) || defined(__FreeBSD_kernel__) -# ifdef HAVE_USBHID_H -# include -# else -# include -# endif -# endif -# include -# include - -/* Compatibility with older usb.h revisions */ -# if !defined(USB_MAX_DEVNAMES) && defined(MAXDEVNAMES) -# define USB_MAX_DEVNAMES MAXDEVNAMES -# endif -# endif - -static int hatmap_x[9] = { 0, 0, 1, 1, 1, 0, -1, -1, -1 }; -static int hatmap_y[9] = { 0, 1, 1, 0, -1, -1, -1, 0, 1 }; -struct os_specific_s { - char fname [128 ]; - int fd; - int is_analog; - /* The following structure members are specific to analog joysticks */ - struct joystick ajs; -# ifdef HAVE_USB_JS - /* The following structure members are specific to USB joysticks */ - struct hid_item *hids; - int hid_dlen; - int hid_offset; - char *hid_data_buf; - int axes_usage [ _JS_MAX_AXES ]; -# endif - /* We keep button and axes state ourselves, as they might not be updated - * on every read of a USB device - */ - int cache_buttons; - float cache_axes [ _JS_MAX_AXES ]; -}; - -/* Idents lower than USB_IDENT_OFFSET are for analog joysticks. */ -# define USB_IDENT_OFFSET 2 - -# define USBDEV "/dev/usb" -# define UHIDDEV "/dev/uhid" -# define AJSDEV "/dev/joy" - -# ifdef HAVE_USB_JS -/* - * fghJoystickFindUSBdev (and its helper, fghJoystickWalkUSBdev) try to locate - * the full name of a USB device. If /dev/usbN isn't readable, we punt and - * return the uhidN device name. We warn the user of this situation once. - */ -static char *fghJoystickWalkUSBdev(int f, char *dev, char *out, int outlen) -{ - struct usb_device_info di; - int i, a; - char *cp; - - for (a = 1; a < USB_MAX_DEVICES; a++) { - di.udi_addr = a; - if (ioctl(f, USB_DEVICEINFO, &di) != 0) - return NULL; - for (i = 0; i < USB_MAX_DEVNAMES; i++) - if (di.udi_devnames[i][0] && - strcmp(di.udi_devnames[i], dev) == 0) { - cp = calloc( 1, strlen(di.udi_vendor) + strlen(di.udi_product) + 2); - strcpy(cp, di.udi_vendor); - strcat(cp, " "); - strcat(cp, di.udi_product); - strncpy(out, cp, outlen - 1); - out[outlen - 1] = 0; - free( cp ); - return out; - } - } - return NULL; -} - -static int fghJoystickFindUSBdev(char *name, char *out, int outlen) -{ - int i, f; - char buf[50]; - char *cp; - static int protection_warned = 0; - - for (i = 0; i < 16; i++) { - snprintf(buf, sizeof(buf), "%s%d", USBDEV, i); - f = open(buf, O_RDONLY); - if (f >= 0) { - cp = fghJoystickWalkUSBdev(f, name, out, outlen); - close(f); - if (cp) - return 1; - } -#ifdef HAVE_ERRNO_H - else if (errno == EACCES) { - if (!protection_warned) { - fgWarning ( "Can't open %s for read!", buf ); - protection_warned = 1; - } - } -#endif - } - return 0; -} - -static int fghJoystickInitializeHID(struct os_specific_s *os, - int *num_axes, int *num_buttons) -{ - int size, is_joystick; -# ifdef HAVE_USBHID_H - int report_id = 0; -# endif - struct hid_data *d; - struct hid_item h; - report_desc_t rd; - - if ( ( rd = hid_get_report_desc( os->fd ) ) == 0 ) - { -#ifdef HAVE_ERRNO_H - fgWarning ( "error: %s: %s", os->fname, strerror( errno ) ); -#else - fgWarning ( "error: %s", os->fname ); -#endif - return FALSE; - } - - os->hids = NULL; - -# ifdef HAVE_USBHID_H - if( ioctl( os->fd, USB_GET_REPORT_ID, &report_id ) < 0) - { - /*** XXX {report_id} may not be the right variable? ***/ -#ifdef HAVE_ERRNO_H - fgWarning ( "error: %s%d: %s", UHIDDEV, report_id, strerror( errno ) ); -#else - fgWarning ( "error: %s%d", UHIDDEV, report_id ); -#endif - return FALSE; - } - - size = hid_report_size( rd, hid_input, report_id ); -# else - size = hid_report_size( rd, 0, hid_input ); -# endif - os->hid_data_buf = calloc( 1, size ); - os->hid_dlen = size; - - is_joystick = 0; -# ifdef HAVE_USBHID_H - d = hid_start_parse( rd, 1 << hid_input, report_id ); -# else - d = hid_start_parse( rd, 1 << hid_input ); -# endif - while( hid_get_item( d, &h ) ) - { - int usage, page, interesting_hid; - - page = HID_PAGE( h.usage ); - usage = HID_USAGE( h.usage ); - - /* This test is somewhat too simplistic, but this is how MicroSoft - * does, so I guess it works for all joysticks/game pads. */ - is_joystick = is_joystick || - ( h.kind == hid_collection && - page == HUP_GENERIC_DESKTOP && - ( usage == HUG_JOYSTICK || usage == HUG_GAME_PAD ) ); - - if( h.kind != hid_input ) - continue; - - if( !is_joystick ) - continue; - - interesting_hid = TRUE; - if( page == HUP_GENERIC_DESKTOP ) - { - switch( usage ) - { - case HUG_X: - case HUG_RX: - case HUG_Y: - case HUG_RY: - case HUG_Z: - case HUG_RZ: - case HUG_SLIDER: - if( *num_axes < _JS_MAX_AXES ) - { - os->axes_usage[ *num_axes ] = usage; - ( *num_axes )++; - } - break; - case HUG_HAT_SWITCH: - /* Allocate two axes for a hat */ - if( *num_axes + 1 < _JS_MAX_AXES ) - { - os->axes_usage[ *num_axes ] = usage; - (*num_axes)++; - os->axes_usage[ *num_axes ] = usage; - (*num_axes)++; - } - break; - default: - interesting_hid = FALSE; - break; - } - } - else if( page == HUP_BUTTON ) - { - interesting_hid = ( usage > 0 ) && - ( usage <= _JS_MAX_BUTTONS ); - - if( interesting_hid && usage - 1 > *num_buttons ) - *num_buttons = usage - 1; - } - - if( interesting_hid ) - { - h.next = os->hids; - os->hids = calloc( 1, sizeof ( struct hid_item ) ); - *os->hids = h; - } - } - hid_end_parse( d ); - - return os->hids != NULL; -} -# endif -#endif - -/* - * Definition of "SFG_Joystick" structure -- based on JS's "jsJoystick" object class. - * See "js.h" lines 80-178. - */ -typedef struct tagSFG_Joystick SFG_Joystick; -struct tagSFG_Joystick -{ -#if TARGET_HOST_MACINTOSH -#define ISP_NUM_AXIS 9 -#define ISP_NUM_NEEDS 41 - ISpElementReference isp_elem [ ISP_NUM_NEEDS ]; - ISpNeed isp_needs [ ISP_NUM_NEEDS ]; -#endif - -#if TARGET_HOST_MAC_OSX - IOHIDDeviceInterface ** hidDev; - IOHIDElementCookie buttonCookies[41]; - IOHIDElementCookie axisCookies[_JS_MAX_AXES]; - long minReport[_JS_MAX_AXES], - maxReport[_JS_MAX_AXES]; -#endif - -#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) - JOYCAPS jsCaps; - JOYINFOEX js; - UINT js_id; -#endif - - -#if TARGET_HOST_POSIX_X11 -# if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__NetBSD__) - struct os_specific_s *os; -# endif - -# ifdef JS_NEW - struct js_event js; - int tmp_buttons; - float tmp_axes [ _JS_MAX_AXES ]; -# else - struct JS_DATA_TYPE js; -# endif - - char fname [ 128 ]; - int fd; -#endif - - int id; - GLboolean error; - char name [ 128 ]; - int num_axes; - int num_buttons; - - float dead_band[ _JS_MAX_AXES ]; - float saturate [ _JS_MAX_AXES ]; - float center [ _JS_MAX_AXES ]; - float max [ _JS_MAX_AXES ]; - float min [ _JS_MAX_AXES ]; -}; - -/* - * Functions associated with the "jsJoystick" class in PLIB - */ -#if TARGET_HOST_MAC_OSX -#define K_NUM_DEVICES 32 -int numDevices; -io_object_t ioDevices[K_NUM_DEVICES]; - -static void fghJoystickFindDevices ( SFG_Joystick* joy, mach_port_t ); -static CFDictionaryRef fghJoystickGetCFProperties ( SFG_Joystick* joy, io_object_t ); - -static void fghJoystickEnumerateElements ( SFG_Joystick* joy, CFTypeRef element ); -/* callback for CFArrayApply */ -static void fghJoystickElementEnumerator ( SFG_Joystick* joy, void *element, void* vjs ); - -static void fghJoystickAddAxisElement ( SFG_Joystick* joy, CFDictionaryRef axis ); -static void fghJoystickAddButtonElement ( SFG_Joystick* joy, CFDictionaryRef button ); -static void fghJoystickAddHatElement ( SFG_Joystick* joy, CFDictionaryRef hat ); -#endif - - -/* - * The static joystick structure pointer - */ -#define MAX_NUM_JOYSTICKS 2 -static SFG_Joystick *fgJoystick [ MAX_NUM_JOYSTICKS ]; - - -/* - * Read the raw joystick data - */ -static void fghJoystickRawRead( SFG_Joystick* joy, int* buttons, float* axes ) -{ -#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) - MMRESULT status; -#else - int status; -#endif - -#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__NetBSD__) - int len; -#endif - - int i; - - /* Defaults */ - if( buttons ) - *buttons = 0; - - if( axes ) - for( i = 0; i < joy->num_axes; i++ ) - axes[ i ] = 1500.0f; - - if( joy->error ) - return; - -#if TARGET_HOST_MACINTOSH - if ( buttons ) - { - *buttons = 0; - - for ( i = 0; i < joy->num_buttons; i++ ) - { - UInt32 state; - int err = ISpElement_GetSimpleState ( isp_elem [ i + isp_num_axis ], &state); - ISP_CHECK_ERR(err) - - *buttons |= state << i; - } - } - - if ( axes ) - { - for ( i = 0; i < joy->num_axes; i++ ) - { - UInt32 state; - int err = ISpElement_GetSimpleState ( isp_elem [ i ], &state ); - ISP_CHECK_ERR(err) - - axes [i] = (float) state; - } - } -#endif - -#if TARGET_HOST_MAC_OSX - if ( buttons != NULL ) - { - *buttons = 0; - - for ( i = 0; i < joy->num_buttons; i++ ) - { - IOHIDEventStruct hidEvent; - (*(joy->hidDev))->getElementValue ( joy->hidDev, buttonCookies[i], &hidEvent ); - if ( hidEvent.value ) - *buttons |= 1 << i; - } - } - - if ( axes != NULL ) - { - for ( i = 0; i < joy->num_axes; i++ ) - { - IOHIDEventStruct hidEvent; - (*(joy->hidDev))->getElementValue ( joy->hidDev, axisCookies[i], &hidEvent ); - axes[i] = hidEvent.value; - } - } -#endif - -#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) - status = joyGetPosEx( joy->js_id, &joy->js ); - - if ( status != JOYERR_NOERROR ) - { - joy->error = GL_TRUE; - return; - } - - if ( buttons ) - *buttons = joy->js.dwButtons; - - if ( axes ) - { - /* - * WARNING - Fall through case clauses!! - */ - switch ( joy->num_axes ) - { - case 8: - /* Generate two POV axes from the POV hat angle. - * Low 16 bits of js.dwPOV gives heading (clockwise from ahead) in - * hundredths of a degree, or 0xFFFF when idle. - */ - if ( ( joy->js.dwPOV & 0xFFFF ) == 0xFFFF ) - { - axes [ 6 ] = 0.0; - axes [ 7 ] = 0.0; - } - else - { - /* This is the contentious bit: how to convert angle to X/Y. - * wk: I know of no define for PI that we could use here: - * SG_PI would pull in sg, M_PI is undefined for MSVC - * But the accuracy of the value of PI is very unimportant at - * this point. - */ - float s = (float) sin ( ( joy->js.dwPOV & 0xFFFF ) * ( 0.01 * 3.1415926535f / 180.0f ) ); - float c = (float) cos ( ( joy->js.dwPOV & 0xFFFF ) * ( 0.01 * 3.1415926535f / 180.0f ) ); - - /* Convert to coordinates on a square so that North-East - * is (1,1) not (.7,.7), etc. - * s and c cannot both be zero so we won't divide by zero. - */ - if ( fabs ( s ) < fabs ( c ) ) - { - axes [ 6 ] = ( c < 0.0 ) ? -s/c : s/c ; - axes [ 7 ] = ( c < 0.0 ) ? -1.0f : 1.0f; - } - else - { - axes [ 6 ] = ( s < 0.0 ) ? -1.0f : 1.0f; - axes [ 7 ] = ( s < 0.0 ) ? -c/s : c/s ; - } - } - - case 6: axes[5] = (float) joy->js.dwVpos; - case 5: axes[4] = (float) joy->js.dwUpos; - case 4: axes[3] = (float) joy->js.dwRpos; - case 3: axes[2] = (float) joy->js.dwZpos; - case 2: axes[1] = (float) joy->js.dwYpos; - case 1: axes[0] = (float) joy->js.dwXpos; - } - } -#endif - -#if TARGET_HOST_POSIX_X11 -# if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__NetBSD__) - if ( joy->os->is_analog ) - { - int status = read ( joy->os->fd, &joy->os->ajs, sizeof(joy->os->ajs) ); - if ( status != sizeof(joy->os->ajs) ) { - perror ( joy->os->fname ); - joy->error = GL_TRUE; - return; - } - if ( buttons != NULL ) - *buttons = ( joy->os->ajs.b1 ? 1 : 0 ) | ( joy->os->ajs.b2 ? 2 : 0 ); - - if ( axes != NULL ) - { - axes[0] = (float) joy->os->ajs.x; - axes[1] = (float) joy->os->ajs.y; - } - - return; - } - -# ifdef HAVE_USB_JS - while ( ( len = read ( joy->os->fd, joy->os->hid_data_buf, joy->os->hid_dlen ) ) == joy->os->hid_dlen ) - { - struct hid_item *h; - - for ( h = joy->os->hids; h; h = h->next ) - { - int d = hid_get_data ( joy->os->hid_data_buf, h ); - - int page = HID_PAGE ( h->usage ); - int usage = HID_USAGE ( h->usage ); - - if ( page == HUP_GENERIC_DESKTOP ) - { - int i; - for ( i = 0; i < joy->num_axes; i++ ) - if (joy->os->axes_usage[i] == usage) - { - if (usage == HUG_HAT_SWITCH) - { - if (d < 0 || d > 8) - d = 0; /* safety */ - joy->os->cache_axes[i] = (float)hatmap_x[d]; - joy->os->cache_axes[i + 1] = (float)hatmap_y[d]; - } - else - { - joy->os->cache_axes[i] = (float)d; - } - break; - } - } - else if (page == HUP_BUTTON) - { - if (usage > 0 && usage < _JS_MAX_BUTTONS + 1) - { - if (d) - joy->os->cache_buttons |= (1 << ( usage - 1 )); - else - joy->os->cache_buttons &= ~(1 << ( usage - 1 )); - } - } - } - } -#ifdef HAVE_ERRNO_H - if ( len < 0 && errno != EAGAIN ) -#else - if ( len < 0 ) -#endif - { - perror( joy->os->fname ); - joy->error = 1; - } - if ( buttons != NULL ) *buttons = joy->os->cache_buttons; - if ( axes != NULL ) - memcpy ( axes, joy->os->cache_axes, sizeof(float) * joy->num_axes ); -# endif -# endif - -# ifdef JS_NEW - - while ( 1 ) - { - status = read ( joy->fd, &joy->js, sizeof(struct js_event) ); - - if ( status != sizeof( struct js_event ) ) - { -#ifdef HAVE_ERRNO_H - if ( errno == EAGAIN ) - { - /* Use the old values */ - if ( buttons ) - *buttons = joy->tmp_buttons; - if ( axes ) - memcpy( axes, joy->tmp_axes, - sizeof( float ) * joy->num_axes ); - return; - } -#endif - - fgWarning ( "%s", joy->fname ); - joy->error = GL_TRUE; - return; - } - - switch ( joy->js.type & ~JS_EVENT_INIT ) - { - case JS_EVENT_BUTTON: - if( joy->js.value == 0 ) /* clear the flag */ - joy->tmp_buttons &= ~( 1 << joy->js.number ); - else - joy->tmp_buttons |= ( 1 << joy->js.number ); - break; - - case JS_EVENT_AXIS: - if ( joy->js.number < joy->num_axes ) - { - joy->tmp_axes[ joy->js.number ] = ( float )joy->js.value; - - if( axes ) - memcpy( axes, joy->tmp_axes, sizeof(float) * joy->num_axes ); - } - break; - - default: - fgWarning ( "PLIB_JS: Unrecognised /dev/js return!?!" ); - - /* use the old values */ - - if ( buttons != NULL ) *buttons = joy->tmp_buttons; - if ( axes != NULL ) - memcpy ( axes, joy->tmp_axes, sizeof(float) * joy->num_axes ); - - return; - } - - if( buttons ) - *buttons = joy->tmp_buttons; - } -# else - - status = read( joy->fd, &joy->js, JS_RETURN ); - - if ( status != JS_RETURN ) - { - fgWarning( "%s", joy->fname ); - joy->error = GL_TRUE; - return; - } - - if ( buttons ) -# if defined( __FreeBSD__ ) || defined(__FreeBSD_kernel__) || defined( __NetBSD__ ) - *buttons = ( joy->js.b1 ? 1 : 0 ) | ( joy->js.b2 ? 2 : 0 ); /* XXX Should not be here -- BSD is handled earlier */ -# else - *buttons = joy->js.buttons; -# endif - - if ( axes ) - { - axes[ 0 ] = (float) joy->js.x; - axes[ 1 ] = (float) joy->js.y; - } -# endif -#endif -} - -/* - * Correct the joystick axis data - */ -static float fghJoystickFudgeAxis( SFG_Joystick* joy, float value, int axis ) -{ - if( value < joy->center[ axis ] ) - { - float xx = ( value - joy->center[ axis ] ) / ( joy->center[ axis ] - - joy->min[ axis ] ); - - if( xx < -joy->saturate[ axis ] ) - return -1.0f; - - if( xx > -joy->dead_band [ axis ] ) - return 0.0f; - - xx = ( xx + joy->dead_band[ axis ] ) / ( joy->saturate[ axis ] - - joy->dead_band[ axis ] ); - - return ( xx < -1.0f ) ? -1.0f : xx; - } - else - { - float xx = ( value - joy->center [ axis ] ) / ( joy->max[ axis ] - - joy->center[ axis ] ); - - if( xx > joy->saturate[ axis ] ) - return 1.0f; - - if( xx < joy->dead_band[ axis ] ) - return 0.0f; - - xx = ( xx - joy->dead_band[ axis ] ) / ( joy->saturate[ axis ] - - joy->dead_band[ axis ] ); - - return ( xx > 1.0f ) ? 1.0f : xx; - } -} - -/* - * Read the corrected joystick data - */ -static void fghJoystickRead( SFG_Joystick* joy, int* buttons, float* axes ) -{ - float raw_axes[ _JS_MAX_AXES ]; - int i; - - if( joy->error ) - { - if( buttons ) - *buttons = 0; - - if( axes ) - for ( i=0; inum_axes; i++ ) - axes[ i ] = 0.0f; - } - - fghJoystickRawRead( joy, buttons, raw_axes ); - - if( axes ) - for( i=0; inum_axes; i++ ) - axes[ i ] = fghJoystickFudgeAxis( joy, raw_axes[ i ], i ); -} - -/* - * Happy happy happy joy joy joy (happy new year toudi :D) - */ - - -#if TARGET_HOST_MAC_OSX -/** open the IOKit connection, enumerate all the HID devices, add their -interface references to the static array. We then use the array index -as the device number when we come to open() the joystick. */ -static int fghJoystickFindDevices ( SFG_Joystick *joy, mach_port_t masterPort ) -{ - CFMutableDictionaryRef hidMatch = NULL; - IOReturn rv = kIOReturnSuccess; - - io_iterator_t hidIterator; - io_object_t ioDev; - - /* build a dictionary matching HID devices */ - hidMatch = IOServiceMatching(kIOHIDDeviceKey); - - rv = IOServiceGetMatchingServices(masterPort, hidMatch, &hidIterator); - if (rv != kIOReturnSuccess || !hidIterator) { - fgWarning( "no joystick (HID) devices found" ); - return; - } - - /* iterate */ - while ((ioDev = IOIteratorNext(hidIterator))) { - /* filter out keyboard and mouse devices */ - CFDictionaryRef properties = getCFProperties(ioDev); - long usage, page; - - CFTypeRef refPage = CFDictionaryGetValue (properties, CFSTR(kIOHIDPrimaryUsagePageKey)); - CFTypeRef refUsage = CFDictionaryGetValue (properties, CFSTR(kIOHIDPrimaryUsageKey)); - CFNumberGetValue((CFNumberRef) refUsage, kCFNumberLongType, &usage); - CFNumberGetValue((CFNumberRef) refPage, kCFNumberLongType, &page); - - /* keep only joystick devices */ - if ( ( page == kHIDPage_GenericDesktop ) && ( - (usage == kHIDUsage_GD_Joystick) - || (usage == kHIDUsage_GD_GamePad) - || (usage == kHIDUsage_GD_MultiAxisController) - || (usage == kHIDUsage_GD_Hatswitch) /* last two necessary ? */ - /* add it to the array */ - ioDevices[numDevices++] = ioDev; - } - - IOObjectRelease(hidIterator); -} - -static CFDictionaryRef fghJoystickGetCFProperties ( SFG_Joystick *joy, io_object_t ioDev ) -{ - IOReturn rv; - CFMutableDictionaryRef cfProperties; - -#if 0 - /* comment copied from darwin/SDL_sysjoystick.c */ - /* Mac OS X currently is not mirroring all USB properties to HID page so need to look at USB device page also - * get dictionary for usb properties: step up two levels and get CF dictionary for USB properties - */ - - io_registry_entry_t parent1, parent2; - - rv = IORegistryEntryGetParentEntry (ioDev, kIOServicePlane, &parent1); - if (rv != kIOReturnSuccess) { - fgWarning ( "error getting device entry parent"); - return NULL; - } - - rv = IORegistryEntryGetParentEntry (parent1, kIOServicePlane, &parent2); - if (rv != kIOReturnSuccess) { - fgWarning ( "error getting device entry parent 2"); - return NULL; - } -#endif - - rv = IORegistryEntryCreateCFProperties( ioDev /*parent2*/, - &cfProperties, kCFAllocatorDefault, kNilOptions); - if (rv != kIOReturnSuccess || !cfProperties) { - fgWarning ( "error getting device properties"); - return NULL; - } - - return cfProperties; -} - -static void fghJoystickElementEnumerator ( SFG_Joystick *joy, void *element, void* vjs ) -{ - if (CFGetTypeID((CFTypeRef) element) != CFDictionaryGetTypeID()) { - fgError ( "%s", "element enumerator passed non-dictionary value"); - return; - } - - static_cast(vjs)->parseElement ( (CFDictionaryRef) element ); -} - -/** element enumerator function : pass NULL for top-level*/ -static void fghJoystickEnumerateElements ( SFG_Joystick *joy, CFTypeRef element ) -{ - FREEGLUT_INTERNAL_ERROR_EXIT( (CFGetTypeID(element) == CFArrayGetTypeID(), - "Joystick element type mismatch", - "fghJoystickEnumerateElements" ); - - CFRange range = {0, CFArrayGetCount ((CFArrayRef)element)}; - CFArrayApplyFunction((CFArrayRef) element, range, - &fghJoystickElementEnumerator, joy ); -} - -static void fghJoystickAddAxisElement ( SFG_Joystick *joy, CFDictionaryRef axis ) -{ - long cookie, lmin, lmax; - int index = joy->num_axes++; - - CFNumberGetValue ((CFNumberRef) - CFDictionaryGetValue ( axis, CFSTR(kIOHIDElementCookieKey) ), - kCFNumberLongType, &cookie); - - axisCookies[index] = (IOHIDElementCookie) cookie; - - CFNumberGetValue ((CFNumberRef) - CFDictionaryGetValue ( axis, CFSTR(kIOHIDElementMinKey) ), - kCFNumberLongType, &lmin); - - CFNumberGetValue ((CFNumberRef) - CFDictionaryGetValue ( axis, CFSTR(kIOHIDElementMaxKey) ), - kCFNumberLongType, &lmax); - - joy->min[index] = lmin; - joy->max[index] = lmax; - joy->dead_band[index] = 0.0; - joy->saturate[index] = 1.0; - joy->center[index] = (lmax + lmin) * 0.5; -} - -static void fghJoystickAddButtonElement ( SFG_Joystick *joy, CFDictionaryRef button ) -{ - long cookie; - CFNumberGetValue ((CFNumberRef) - CFDictionaryGetValue ( button, CFSTR(kIOHIDElementCookieKey) ), - kCFNumberLongType, &cookie); - - joy->buttonCookies[num_buttons++] = (IOHIDElementCookie) cookie; - /* anything else for buttons? */ -} - -static void fghJoystickAddHatElement ( SFG_Joystick *joy, CFDictionaryRef button ) -{ - /* hatCookies[num_hats++] = (IOHIDElementCookie) cookie; */ - /* do we map hats to axes or buttons? */ -} -#endif - -#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) -/* Inspired by - http://msdn.microsoft.com/archive/en-us/dnargame/html/msdn_sidewind3d.asp - */ -# if FREEGLUT_LIB_PRAGMAS -# pragma comment (lib, "advapi32.lib") -# endif - -static int fghJoystickGetOEMProductName ( SFG_Joystick* joy, char *buf, int buf_sz ) -{ - char buffer [ 256 ]; - - char OEMKey [ 256 ]; - - HKEY hKey; - DWORD dwcb; - LONG lr; - - if ( joy->error ) - return 0; - - /* Open .. MediaResources\CurrentJoystickSettings */ - _snprintf ( buffer, sizeof(buffer), "%s\\%s\\%s", - REGSTR_PATH_JOYCONFIG, joy->jsCaps.szRegKey, - REGSTR_KEY_JOYCURR ); - - lr = RegOpenKeyEx ( HKEY_LOCAL_MACHINE, buffer, 0, KEY_QUERY_VALUE, &hKey); - - if ( lr != ERROR_SUCCESS ) return 0; - - /* Get OEM Key name */ - dwcb = sizeof(OEMKey); - - /* JOYSTICKID1-16 is zero-based; registry entries for VJOYD are 1-based. */ - _snprintf ( buffer, sizeof(buffer), "Joystick%d%s", joy->js_id + 1, REGSTR_VAL_JOYOEMNAME ); - - lr = RegQueryValueEx ( hKey, buffer, 0, 0, (LPBYTE) OEMKey, &dwcb); - RegCloseKey ( hKey ); - - if ( lr != ERROR_SUCCESS ) return 0; - - /* Open OEM Key from ...MediaProperties */ - _snprintf ( buffer, sizeof(buffer), "%s\\%s", REGSTR_PATH_JOYOEM, OEMKey ); - - lr = RegOpenKeyEx ( HKEY_LOCAL_MACHINE, buffer, 0, KEY_QUERY_VALUE, &hKey ); - - if ( lr != ERROR_SUCCESS ) return 0; - - /* Get OEM Name */ - dwcb = buf_sz; - - lr = RegQueryValueEx ( hKey, REGSTR_VAL_JOYOEMNAME, 0, 0, (LPBYTE) buf, - &dwcb ); - RegCloseKey ( hKey ); - - if ( lr != ERROR_SUCCESS ) return 0; - - return 1; -} -#endif - - -static void fghJoystickOpen( SFG_Joystick* joy ) -{ - int i = 0; -#if TARGET_HOST_MACINTOSH - OSStatus err; -#endif -#if TARGET_HOST_MAC_OSX - IOReturn rv; - SInt32 score; - IOCFPlugInInterface **plugin; - - HRESULT pluginResult; - - CFDictionaryRef props; - CFTypeRef topLevelElement; -#endif -#if TARGET_HOST_POSIX_X11 -# if defined( __FreeBSD__ ) || defined(__FreeBSD_kernel__) || defined( __NetBSD__ ) - char *cp; -# endif -# ifdef JS_NEW - unsigned char u; -# else -# if defined( __linux__ ) || TARGET_HOST_SOLARIS - int counter = 0; -# endif -# endif -#endif - - /* Silence gcc, the correct #ifdefs would be too fragile... */ - (void)i; - - /* - * Default values (for no joystick -- each conditional will reset the - * error flag) - */ - joy->error = TRUE; - joy->num_axes = joy->num_buttons = 0; - joy->name[ 0 ] = '\0'; - -#if TARGET_HOST_MACINTOSH - /* XXX FIXME: get joystick name in Mac */ - - err = ISpStartup( ); - - if( err == noErr ) - { -#define ISP_CHECK_ERR(x) if( x != noErr ) { joy->error = GL_TRUE; return; } - - joy->error = GL_TRUE; - - /* initialize the needs structure */ - ISpNeed temp_isp_needs[ isp_num_needs ] = - { - { "\pX-Axis", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, - { "\pY-Axis", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, - { "\pZ-Axis", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, - { "\pR-Axis", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, - { "\pAxis 4", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, - { "\pAxis 5", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, - { "\pAxis 6", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, - { "\pAxis 7", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, - { "\pAxis 8", 128, 0, 0, kISpElementKind_Axis, kISpElementLabel_None, 0, 0, 0, 0 }, - - { "\pButton 0", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 1", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 2", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 3", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 4", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 5", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 6", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 7", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 8", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 9", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 10", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 11", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 12", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 13", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 14", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 15", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 16", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 17", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 18", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 19", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 20", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 21", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 22", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 23", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 24", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 25", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 26", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 27", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 28", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 29", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 30", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - { "\pButton 31", 128, 0, 0, kISpElementKind_Button, kISpElementLabel_Btn_Select, 0, 0, 0, 0 }, - }; - - memcpy( joy->isp_needs, temp_isp_needs, sizeof (temp_isp_needs ) ); - - - /* next two calls allow keyboard and mouse to emulate other input - * devices (gamepads, joysticks, etc) - */ - /* - err = ISpDevices_ActivateClass ( kISpDeviceClass_Keyboard ); - ISP_CHECK_ERR(err) - - - err = ISpDevices_ActivateClass ( kISpDeviceClass_Mouse ); - ISP_CHECK_ERR(err) - */ - - err = ISpElement_NewVirtualFromNeeds( joy->isp_num_needs, - joy->isp_needs, joy->isp_elem, - 0 ); - ISP_CHECK_ERR( err ) - - err = ISpInit( joy->isp_num_needs, joy->isp_needs, joy->isp_elem, - 'freeglut', nil, 0, 128, 0 ); - ISP_CHECK_ERR( err ) - - joy->num_buttons = joy->isp_num_needs - joy->isp_num_axis; - joy->num_axes = joy->isp_num_axis; - - for( i = 0; i < joy->num_axes; i++ ) - { - joy->dead_band[ i ] = 0; - joy->saturate [ i ] = 1; - joy->center [ i ] = kISpAxisMiddle; - joy->max [ i ] = kISpAxisMaximum; - joy->min [ i ] = kISpAxisMinimum; - } - - joy->error = GL_FALSE; - } - else - joy->num_buttons = joy->num_axes = 0; -#endif - -#if TARGET_HOST_MAC_OSX - if( joy->id >= numDevices ) - { - fgWarning( "device index out of range in fgJoystickOpen()" ); - return; - } - - /* create device interface */ - rv = IOCreatePlugInInterfaceForService( ioDevices[ joy->id ], - kIOHIDDeviceUserClientTypeID, - kIOCFPlugInInterfaceID, - &plugin, &score ); - - if( rv != kIOReturnSuccess ) - { - fgWarning( "error creating plugin for io device" ); - return; - } - - pluginResult = ( *plugin )->QueryInterface( - plugin, - CFUUIDGetUUIDBytes(kIOHIDDeviceInterfaceID), - &( LPVOID )joy->hidDev - ); - - if( pluginResult != S_OK ) - fgWarning ( "QI-ing IO plugin to HID Device interface failed" ); - - ( *plugin )->Release( plugin ); /* don't leak a ref */ - if( joy->hidDev == NULL ) - return; - - /* store the interface in this instance */ - rv = ( *( joy->hidDev ) )->open( joy->hidDev, 0 ); - if( rv != kIOReturnSuccess ) - { - fgWarning( "error opening device interface"); - return; - } - - props = getCFProperties( ioDevices[ joy->id ] ); - - /* recursively enumerate all the bits */ - CFTypeRef topLevelElement = - CFDictionaryGetValue( props, CFSTR( kIOHIDElementKey ) ); - enumerateElements( topLevelElement ); - - CFRelease( props ); -#endif - -#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) - joy->js.dwFlags = JOY_RETURNALL; - joy->js.dwSize = sizeof( joy->js ); - - memset( &joy->jsCaps, 0, sizeof( joy->jsCaps ) ); - - joy->error = - ( joyGetDevCaps( joy->js_id, &joy->jsCaps, sizeof( joy->jsCaps ) ) != - JOYERR_NOERROR ); - - if( joy->jsCaps.wNumAxes == 0 ) - { - joy->num_axes = 0; - joy->error = GL_TRUE; - } - else - { - /* Device name from jsCaps is often "Microsoft PC-joystick driver", - * at least for USB. Try to get the real name from the registry. - */ - if ( ! fghJoystickGetOEMProductName( joy, joy->name, - sizeof( joy->name ) ) ) - { - fgWarning( "JS: Failed to read joystick name from registry" ); - strncpy( joy->name, joy->jsCaps.szPname, sizeof( joy->name ) ); - } - - /* Windows joystick drivers may provide any combination of - * X,Y,Z,R,U,V,POV - not necessarily the first n of these. - */ - if( joy->jsCaps.wCaps & JOYCAPS_HASPOV ) - { - joy->num_axes = _JS_MAX_AXES; - joy->min[ 7 ] = -1.0; joy->max[ 7 ] = 1.0; /* POV Y */ - joy->min[ 6 ] = -1.0; joy->max[ 6 ] = 1.0; /* POV X */ - } - else - joy->num_axes = 6; - - joy->min[ 5 ] = ( float )joy->jsCaps.wVmin; - joy->max[ 5 ] = ( float )joy->jsCaps.wVmax; - joy->min[ 4 ] = ( float )joy->jsCaps.wUmin; - joy->max[ 4 ] = ( float )joy->jsCaps.wUmax; - joy->min[ 3 ] = ( float )joy->jsCaps.wRmin; - joy->max[ 3 ] = ( float )joy->jsCaps.wRmax; - joy->min[ 2 ] = ( float )joy->jsCaps.wZmin; - joy->max[ 2 ] = ( float )joy->jsCaps.wZmax; - joy->min[ 1 ] = ( float )joy->jsCaps.wYmin; - joy->max[ 1 ] = ( float )joy->jsCaps.wYmax; - joy->min[ 0 ] = ( float )joy->jsCaps.wXmin; - joy->max[ 0 ] = ( float )joy->jsCaps.wXmax; - } - - /* Guess all the rest judging on the axes extremals */ - for( i = 0; i < joy->num_axes; i++ ) - { - joy->center [ i ] = ( joy->max[ i ] + joy->min[ i ] ) * 0.5f; - joy->dead_band[ i ] = 0.0f; - joy->saturate [ i ] = 1.0f; - } -#endif - -#if TARGET_HOST_POSIX_X11 -#if defined( __FreeBSD__ ) || defined(__FreeBSD_kernel__) || defined( __NetBSD__ ) - for( i = 0; i < _JS_MAX_AXES; i++ ) - joy->os->cache_axes[ i ] = 0.0f; - - joy->os->cache_buttons = 0; - - joy->os->fd = open( joy->os->fname, O_RDONLY | O_NONBLOCK); - -#ifdef HAVE_ERRNO_H - if( joy->os->fd < 0 && errno == EACCES ) - fgWarning ( "%s exists but is not readable by you", joy->os->fname ); -#endif - - joy->error =( joy->os->fd < 0 ); - - if( joy->error ) - return; - - joy->num_axes = 0; - joy->num_buttons = 0; - if( joy->os->is_analog ) - { - FILE *joyfile; - char joyfname[ 1024 ]; - int noargs, in_no_axes; - - float axes [ _JS_MAX_AXES ]; - int buttons[ _JS_MAX_AXES ]; - - joy->num_axes = 2; - joy->num_buttons = 32; - - fghJoystickRawRead( joy, buttons, axes ); - joy->error = axes[ 0 ] < -1000000000.0f; - if( joy->error ) - return; - - snprintf( joyfname, sizeof(joyfname), "%s/.joy%drc", getenv( "HOME" ), joy->id ); - - joyfile = fopen( joyfname, "r" ); - joy->error =( joyfile == NULL ); - if( joy->error ) - return; - - noargs = fscanf( joyfile, "%d%f%f%f%f%f%f", &in_no_axes, - &joy->min[ 0 ], &joy->center[ 0 ], &joy->max[ 0 ], - &joy->min[ 1 ], &joy->center[ 1 ], &joy->max[ 1 ] ); - joy->error = noargs != 7 || in_no_axes != _JS_MAX_AXES; - fclose( joyfile ); - if( joy->error ) - return; - - for( i = 0; i < _JS_MAX_AXES; i++ ) - { - joy->dead_band[ i ] = 0.0f; - joy->saturate [ i ] = 1.0f; - } - - return; /* End of analog code */ - } - -# ifdef HAVE_USB_JS - if( ! fghJoystickInitializeHID( joy->os, &joy->num_axes, - &joy->num_buttons ) ) - { - close( joy->os->fd ); - joy->error = GL_TRUE; - return; - } - - cp = strrchr( joy->os->fname, '/' ); - if( cp ) - { - if( fghJoystickFindUSBdev( &cp[1], joy->name, sizeof( joy->name ) ) == - 0 ) - strcpy( joy->name, &cp[1] ); - } - - if( joy->num_axes > _JS_MAX_AXES ) - joy->num_axes = _JS_MAX_AXES; - - for( i = 0; i < _JS_MAX_AXES; i++ ) - { - /* We really should get this from the HID, but that data seems - * to be quite unreliable for analog-to-USB converters. Punt for - * now. - */ - if( joy->os->axes_usage[ i ] == HUG_HAT_SWITCH ) - { - joy->max [ i ] = 1.0f; - joy->center[ i ] = 0.0f; - joy->min [ i ] = -1.0f; - } - else - { - joy->max [ i ] = 255.0f; - joy->center[ i ] = 127.0f; - joy->min [ i ] = 0.0f; - } - - joy->dead_band[ i ] = 0.0f; - joy->saturate[ i ] = 1.0f; - } -# endif -#endif - -#if defined( __linux__ ) || TARGET_HOST_SOLARIS - /* Default for older Linux systems. */ - joy->num_axes = 2; - joy->num_buttons = 32; - -# ifdef JS_NEW - for( i = 0; i < _JS_MAX_AXES; i++ ) - joy->tmp_axes[ i ] = 0.0f; - - joy->tmp_buttons = 0; -# endif - - joy->fd = open( joy->fname, O_RDONLY ); - - joy->error =( joy->fd < 0 ); - - if( joy->error ) - return; - - /* Set the correct number of axes for the linux driver */ -# ifdef JS_NEW - /* Melchior Franz's fixes for big-endian Linuxes since writing - * to the upper byte of an uninitialized word doesn't work. - * 9 April 2003 - */ - ioctl( joy->fd, JSIOCGAXES, &u ); - joy->num_axes = u; - ioctl( joy->fd, JSIOCGBUTTONS, &u ); - joy->num_buttons = u; - ioctl( joy->fd, JSIOCGNAME( sizeof( joy->name ) ), joy->name ); - fcntl( joy->fd, F_SETFL, O_NONBLOCK ); -# endif - - /* - * The Linux driver seems to return 512 for all axes - * when no stick is present - but there is a chance - * that could happen by accident - so it's gotta happen - * on both axes for at least 100 attempts. - * - * PWO: shouldn't be that done somehow wiser on the kernel level? - */ -# ifndef JS_NEW - counter = 0; - - do - { - fghJoystickRawRead( joy, NULL, joy->center ); - counter++; - } while( !joy->error && - counter < 100 && - joy->center[ 0 ] == 512.0f && - joy->center[ 1 ] == 512.0f ); - - if ( counter >= 100 ) - joy->error = GL_TRUE; -# endif - - for( i = 0; i < _JS_MAX_AXES; i++ ) - { -# ifdef JS_NEW - joy->max [ i ] = 32767.0f; - joy->center[ i ] = 0.0f; - joy->min [ i ] = -32767.0f; -# else - joy->max[ i ] = joy->center[ i ] * 2.0f; - joy->min[ i ] = 0.0f; -# endif - joy->dead_band[ i ] = 0.0f; - joy->saturate [ i ] = 1.0f; - } -#endif -#endif -} - -/* - * This function replaces the constructor method in the JS library. - */ -static void fghJoystickInit( int ident ) -{ - if( ident >= MAX_NUM_JOYSTICKS ) - fgError( "Too large a joystick number: %d", ident ); - - if( fgJoystick[ ident ] ) - fgError( "illegal attempt to initialize joystick device again" ); - - fgJoystick[ ident ] = - ( SFG_Joystick * )calloc( sizeof( SFG_Joystick ), 1 ); - - /* Set defaults */ - fgJoystick[ ident ]->num_axes = fgJoystick[ ident ]->num_buttons = 0; - fgJoystick[ ident ]->error = GL_TRUE; - -#if TARGET_HOST_MACINTOSH - fgJoystick[ ident ]->id = ident; - snprintf( fgJoystick[ ident ]->fname, sizeof(fgJoystick[ ident ]->fname), "/dev/js%d", ident ); /* FIXME */ - fgJoystick[ ident ]->error = GL_FALSE; -#endif - -#if TARGET_HOST_MAC_OSX - fgJoystick[ ident ]->id = ident; - fgJoystick[ ident ]->error = GL_FALSE; - fgJoystick[ ident ]->num_axes = 0; - fgJoystick[ ident ]->num_buttons = 0; - - if( numDevices < 0 ) - { - /* do first-time init (since we can't over-ride jsInit, hmm */ - numDevices = 0; - - mach_port_t masterPort; - IOReturn rv = IOMasterPort( bootstrap_port, &masterPort ); - if( rv != kIOReturnSuccess ) - { - fgWarning( "error getting master Mach port" ); - return; - } - fghJoystickFindDevices( masterPort ); - } - - if ( ident >= numDevices ) - { - fgJoystick[ ident ]->error = GL_TRUE; - return; - } - - /* get the name now too */ - CFDictionaryRef properties = getCFProperties( ioDevices[ ident ] ); - CFTypeRef ref = CFDictionaryGetValue( properties, - CFSTR( kIOHIDProductKey ) ); - if (!ref) - ref = CFDictionaryGetValue(properties, CFSTR( "USB Product Name" ) ); - - if( !ref || - !CFStringGetCString( ( CFStringRef )ref, name, 128, - CFStringGetSystemEncoding( ) ) ) - { - fgWarning( "error getting device name" ); - name[ 0 ] = '\0'; - } -#endif - -#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) - switch( ident ) - { - case 0: - fgJoystick[ ident ]->js_id = JOYSTICKID1; - fgJoystick[ ident ]->error = GL_FALSE; - break; - case 1: - fgJoystick[ ident ]->js_id = JOYSTICKID2; - fgJoystick[ ident ]->error = GL_FALSE; - break; - default: - fgJoystick[ ident ]->num_axes = 0; - fgJoystick[ ident ]->error = GL_TRUE; - return; - } -#endif - -#if TARGET_HOST_POSIX_X11 -# if defined( __FreeBSD__ ) || defined(__FreeBSD_kernel__) || defined( __NetBSD__ ) - fgJoystick[ ident ]->id = ident; - fgJoystick[ ident ]->error = GL_FALSE; - - fgJoystick[ ident ]->os = calloc( 1, sizeof( struct os_specific_s ) ); - memset( fgJoystick[ ident ]->os, 0, sizeof( struct os_specific_s ) ); - if( ident < USB_IDENT_OFFSET ) - fgJoystick[ ident ]->os->is_analog = 1; - if( fgJoystick[ ident ]->os->is_analog ) - snprintf( fgJoystick[ ident ]->os->fname, sizeof(fgJoystick[ ident ]->os->fname), "%s%d", AJSDEV, ident ); - else - snprintf( fgJoystick[ ident ]->os->fname, sizeof(fgJoystick[ ident ]->os->fname), "%s%d", UHIDDEV, - ident - USB_IDENT_OFFSET ); -# elif defined( __linux__ ) - fgJoystick[ ident ]->id = ident; - fgJoystick[ ident ]->error = GL_FALSE; - - snprintf( fgJoystick[ident]->fname, sizeof(fgJoystick[ident]->fname), "/dev/input/js%d", ident ); - - if( access( fgJoystick[ ident ]->fname, F_OK ) != 0 ) - snprintf( fgJoystick[ ident ]->fname, sizeof(fgJoystick[ ident ]->fname), "/dev/js%d", ident ); -# endif -#endif - - fghJoystickOpen( fgJoystick[ ident ] ); -} - -/* - * Try initializing all the joysticks (well, both of them) - */ -void fgInitialiseJoysticks ( void ) -{ - if( !fgState.JoysticksInitialised ) - { - int ident ; - for ( ident = 0; ident < MAX_NUM_JOYSTICKS; ident++ ) - fghJoystickInit( ident ); - - fgState.JoysticksInitialised = GL_TRUE; - } -} - -/* - * - */ -void fgJoystickClose( void ) -{ - int ident ; - for( ident = 0; ident < MAX_NUM_JOYSTICKS; ident++ ) - { - if( fgJoystick[ ident ] ) - { - -#if TARGET_HOST_MACINTOSH - ISpSuspend( ); - ISpStop( ); - ISpShutdown( ); -#endif - -#if TARGET_HOST_MAC_OSX - ( *( fgJoystick[ ident ]->hidDev ) )-> - close( fgJoystick[ ident ]->hidDev ); -#endif - -#if TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) - /* Do nothing special */ -#endif - -#if TARGET_HOST_POSIX_X11 -#if defined( __FreeBSD__ ) || defined(__FreeBSD_kernel__) || defined( __NetBSD__ ) - if( fgJoystick[ident]->os ) - { - if( ! fgJoystick[ ident ]->error ) - close( fgJoystick[ ident ]->os->fd ); -#ifdef HAVE_USB_JS - if( fgJoystick[ ident ]->os->hids ) - free (fgJoystick[ ident ]->os->hids); - if( fgJoystick[ ident ]->os->hid_data_buf ) - free( fgJoystick[ ident ]->os->hid_data_buf ); -#endif - free( fgJoystick[ident]->os ); - } -#endif - - if( ! fgJoystick[ident]->error ) - close( fgJoystick[ ident ]->fd ); -#endif - - free( fgJoystick[ ident ] ); - fgJoystick[ ident ] = NULL; - /* show joystick has been deinitialized */ - } - } -} - -/* - * Polls the joystick and executes the joystick callback hooked to the - * window specified in the function's parameter: - */ -void fgJoystickPollWindow( SFG_Window* window ) -{ - float axes[ _JS_MAX_AXES ]; - int buttons; - int ident; - - freeglut_return_if_fail( window ); - freeglut_return_if_fail( FETCH_WCB( *window, Joystick ) ); - - for( ident = 0; ident < MAX_NUM_JOYSTICKS; ident++ ) - { - if( fgJoystick[ident] ) - { - fghJoystickRead( fgJoystick[ident], &buttons, axes ); - - if( !fgJoystick[ident]->error ) - INVOKE_WCB( *window, Joystick, - ( buttons, - (int) ( axes[ 0 ] * 1000.0f ), - (int) ( axes[ 1 ] * 1000.0f ), - (int) ( axes[ 2 ] * 1000.0f ) ) - ); - } - } -} - -/* - * Implementation for glutDeviceGet(GLUT_HAS_JOYSTICK) - */ -int fgJoystickDetect( void ) -{ - int ident; - - fgInitialiseJoysticks (); - - if ( !fgState.JoysticksInitialised ) - return 0; - - for( ident=0; identerror ) - return 1; - - return 0; -} - -/* - * Joystick information functions - */ -int glutJoystickGetNumAxes( int ident ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetNumAxes" ); - return fgJoystick[ ident ]->num_axes; -} -int glutJoystickGetNumButtons( int ident ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetNumButtons" ); - return fgJoystick[ ident ]->num_buttons; -} -int glutJoystickNotWorking( int ident ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickNotWorking" ); - return fgJoystick[ ident ]->error; -} - -float glutJoystickGetDeadBand( int ident, int axis ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetDeadBand" ); - return fgJoystick[ ident ]->dead_band [ axis ]; -} -void glutJoystickSetDeadBand( int ident, int axis, float db ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickSetDeadBand" ); - fgJoystick[ ident ]->dead_band[ axis ] = db; -} - -float glutJoystickGetSaturation( int ident, int axis ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetSaturation" ); - return fgJoystick[ ident ]->saturate[ axis ]; -} -void glutJoystickSetSaturation( int ident, int axis, float st ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickSetSaturation" ); - fgJoystick[ ident ]->saturate [ axis ] = st; -} - -void glutJoystickSetMinRange( int ident, float *axes ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickSetMinRange" ); - memcpy( fgJoystick[ ident ]->min, axes, - fgJoystick[ ident ]->num_axes * sizeof( float ) ); -} -void glutJoystickSetMaxRange( int ident, float *axes ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickSetMaxRange" ); - memcpy( fgJoystick[ ident ]->max, axes, - fgJoystick[ ident ]->num_axes * sizeof( float ) ); -} -void glutJoystickSetCenter( int ident, float *axes ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickSetCenter" ); - memcpy( fgJoystick[ ident ]->center, axes, - fgJoystick[ ident ]->num_axes * sizeof( float ) ); -} - -void glutJoystickGetMinRange( int ident, float *axes ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetMinRange" ); - memcpy( axes, fgJoystick[ ident ]->min, - fgJoystick[ ident ]->num_axes * sizeof( float ) ); -} -void glutJoystickGetMaxRange( int ident, float *axes ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetMaxRange" ); - memcpy( axes, fgJoystick[ ident ]->max, - fgJoystick[ ident ]->num_axes * sizeof( float ) ); -} -void glutJoystickGetCenter( int ident, float *axes ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutJoystickGetCenter" ); - memcpy( axes, fgJoystick[ ident ]->center, - fgJoystick[ ident ]->num_axes * sizeof( float ) ); -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_main.c b/examples/common/opengl-framework/freeglut/freeglut_main.c deleted file mode 100644 index 5f9b0a5d..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_main.c +++ /dev/null @@ -1,2511 +0,0 @@ -/* - * freeglut_main.c - * - * The windows message processing methods. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Fri Dec 3 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" -#ifdef HAVE_ERRNO_H -# include -#endif -#include -#ifdef HAVE_VFPRINTF -# define VFPRINTF(s,f,a) vfprintf((s),(f),(a)) -#elif defined(HAVE__DOPRNT) -# define VFPRINTF(s,f,a) _doprnt((f),(a),(s)) -#else -# define VFPRINTF(s,f,a) -#endif - -#ifdef _WIN32_WCE - -typedef struct GXDisplayProperties GXDisplayProperties; -typedef struct GXKeyList GXKeyList; -#include - -typedef struct GXKeyList (*GXGETDEFAULTKEYS)(int); -typedef int (*GXOPENINPUT)(); - -GXGETDEFAULTKEYS GXGetDefaultKeys_ = NULL; -GXOPENINPUT GXOpenInput_ = NULL; - -struct GXKeyList gxKeyList; - -#endif /* _WIN32_WCE */ - -/* - * Try to get the maximum value allowed for ints, falling back to the minimum - * guaranteed by ISO C99 if there is no suitable header. - */ -#ifdef HAVE_LIMITS_H -# include -#endif -#ifndef INT_MAX -# define INT_MAX 32767 -#endif - -#ifndef MIN -# define MIN(a,b) (((a)<(b)) ? (a) : (b)) -#endif - -#ifdef WM_TOUCH - typedef BOOL (WINAPI *pGetTouchInputInfo)(HTOUCHINPUT,UINT,PTOUCHINPUT,int); - typedef BOOL (WINAPI *pCloseTouchInputHandle)(HTOUCHINPUT); - static pGetTouchInputInfo fghGetTouchInputInfo = (pGetTouchInputInfo)0xDEADBEEF; - static pCloseTouchInputHandle fghCloseTouchInputHandle = (pCloseTouchInputHandle)0xDEADBEEF; -#endif - -/* - * TODO BEFORE THE STABLE RELEASE: - * - * There are some issues concerning window redrawing under X11, and maybe - * some events are not handled. The Win32 version lacks some more features, - * but seems acceptable for not demanding purposes. - * - * Need to investigate why the X11 version breaks out with an error when - * closing a window (using the window manager, not glutDestroyWindow)... - */ - -/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ - -/* - * Handle a window configuration change. When no reshape - * callback is hooked, the viewport size is updated to - * match the new window size. - */ -static void fghReshapeWindow ( SFG_Window *window, int width, int height ) -{ - SFG_Window *current_window = fgStructure.CurrentWindow; - - freeglut_return_if_fail( window != NULL ); - -#if TARGET_HOST_POSIX_X11 - - XResizeWindow( fgDisplay.Display, window->Window.Handle, - width, height ); - XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ - -#elif TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) - { - RECT windowRect; - - /* - * For windowed mode, get the current position of the - * window and resize taking the size of the frame - * decorations into account. - */ - - /* "GetWindowRect" returns the pixel coordinates of the outside of the window */ - GetWindowRect( window->Window.Handle, &windowRect ); - - /* Create rect in FreeGLUT format, (X,Y) topleft outside window, WxH of client area */ - windowRect.right = windowRect.left+width; - windowRect.bottom = windowRect.top+height; - - if (window->Parent == NULL) - /* get the window rect from this to feed to SetWindowPos, correct for window decorations */ - fghComputeWindowRectFromClientArea_QueryWindow(window,&windowRect,TRUE); - else - { - /* correct rect for position client area of parent window - * (SetWindowPos input for child windows is in coordinates - * relative to the parent's client area). - * Child windows don't have decoration, so no need to correct - * for them. - */ - RECT parentRect; - parentRect = fghGetClientArea( window->Parent, FALSE ); - windowRect.left -= parentRect.left; - windowRect.right -= parentRect.left; - windowRect.top -= parentRect.top; - windowRect.bottom -= parentRect.top; - } - - /* Do the actual resizing */ - SetWindowPos( window->Window.Handle, - HWND_TOP, - windowRect.left, windowRect.top, - windowRect.right - windowRect.left, - windowRect.bottom- windowRect.top, - SWP_NOACTIVATE | SWP_NOOWNERZORDER | SWP_NOSENDCHANGING | - SWP_NOZORDER - ); - } -#endif - - if( FETCH_WCB( *window, Reshape ) ) - INVOKE_WCB( *window, Reshape, ( width, height ) ); - else - { - fgSetWindow( window ); - glViewport( 0, 0, width, height ); - } - - /* - * Force a window redraw. In Windows at least this is only a partial - * solution: if the window is increasing in size in either dimension, - * the already-drawn part does not get drawn again and things look funny. - * But without this we get this bad behaviour whenever we resize the - * window. - */ - window->State.Redisplay = GL_TRUE; - - if( window->IsMenu ) - fgSetWindow( current_window ); -} - -/* - * Calls a window's redraw method. This is used when - * a redraw is forced by the incoming window messages. - */ -static void fghRedrawWindow ( SFG_Window *window ) -{ - SFG_Window *current_window = fgStructure.CurrentWindow; - - freeglut_return_if_fail( window ); - freeglut_return_if_fail( FETCH_WCB ( *window, Display ) ); - - window->State.Redisplay = GL_FALSE; - - freeglut_return_if_fail( window->State.Visible ); - - fgSetWindow( window ); - - if( window->State.NeedToResize ) - { - fghReshapeWindow( - window, - window->State.Width, - window->State.Height - ); - - window->State.NeedToResize = GL_FALSE; - } - - INVOKE_WCB( *window, Display, ( ) ); - - fgSetWindow( current_window ); -} - -/* - * A static helper function to execute display callback for a window - */ -static void fghcbDisplayWindow( SFG_Window *window, - SFG_Enumerator *enumerator ) -{ - if( window->State.Redisplay && - window->State.Visible ) - { - window->State.Redisplay = GL_FALSE; - -#if TARGET_HOST_POSIX_X11 - fghRedrawWindow ( window ) ; -#elif TARGET_HOST_MS_WINDOWS - - RedrawWindow( - window->Window.Handle, NULL, NULL, - RDW_NOERASE | RDW_INTERNALPAINT | RDW_INVALIDATE | RDW_UPDATENOW - ); -#endif - } - - fgEnumSubWindows( window, fghcbDisplayWindow, enumerator ); -} - -/* - * Make all windows perform a display call - */ -static void fghDisplayAll( void ) -{ - SFG_Enumerator enumerator; - - enumerator.found = GL_FALSE; - enumerator.data = NULL; - - fgEnumWindows( fghcbDisplayWindow, &enumerator ); -} - -/* - * Window enumerator callback to check for the joystick polling code - */ -static void fghcbCheckJoystickPolls( SFG_Window *window, - SFG_Enumerator *enumerator ) -{ - long int checkTime = fgElapsedTime( ); - - if( window->State.JoystickLastPoll + window->State.JoystickPollRate <= - checkTime ) - { -#if !defined(_WIN32_WCE) - fgJoystickPollWindow( window ); -#endif /* !defined(_WIN32_WCE) */ - window->State.JoystickLastPoll = checkTime; - } - - fgEnumSubWindows( window, fghcbCheckJoystickPolls, enumerator ); -} - -/* - * Check all windows for joystick polling - */ -static void fghCheckJoystickPolls( void ) -{ - SFG_Enumerator enumerator; - - enumerator.found = GL_FALSE; - enumerator.data = NULL; - - fgEnumWindows( fghcbCheckJoystickPolls, &enumerator ); -} - -/* - * Check the global timers - */ -static void fghCheckTimers( void ) -{ - long checkTime = fgElapsedTime( ); - - while( fgState.Timers.First ) - { - SFG_Timer *timer = fgState.Timers.First; - - if( timer->TriggerTime > checkTime ) - break; - - fgListRemove( &fgState.Timers, &timer->Node ); - fgListAppend( &fgState.FreeTimers, &timer->Node ); - - timer->Callback( timer->ID ); - } -} - - -/* Platform-dependent time in milliseconds, as an unsigned 32-bit integer. - * This value wraps every 49.7 days, but integer overflows cancel - * when subtracting an initial start time, unless the total time exceeds - * 32-bit, where the GLUT API return value is also overflowed. - */ -unsigned long fgSystemTime(void) { -#if TARGET_HOST_SOLARIS || HAVE_GETTIMEOFDAY - struct timeval now; - gettimeofday( &now, NULL ); - return now.tv_usec/1000 + now.tv_sec*1000; -#elif TARGET_HOST_MS_WINDOWS -# if defined(_WIN32_WCE) - return GetTickCount(); -# else - return timeGetTime(); -# endif -#endif -} - -/* - * Elapsed Time - */ -long fgElapsedTime( void ) -{ - return (long) (fgSystemTime() - fgState.Time); -} - -/* - * Error Messages. - */ -void fgError( const char *fmt, ... ) -{ - va_list ap; - - if (fgState.ErrorFunc) { - - va_start( ap, fmt ); - - /* call user set error handler here */ - fgState.ErrorFunc(fmt, ap); - - va_end( ap ); - - } else { - - va_start( ap, fmt ); - - fprintf( stderr, "freeglut "); - if( fgState.ProgramName ) - fprintf( stderr, "(%s): ", fgState.ProgramName ); - VFPRINTF( stderr, fmt, ap ); - fprintf( stderr, "\n" ); - - va_end( ap ); - - if ( fgState.Initialised ) - fgDeinitialize (); - - exit( 1 ); - } -} - -void fgWarning( const char *fmt, ... ) -{ - va_list ap; - - if (fgState.WarningFunc) { - - va_start( ap, fmt ); - - /* call user set warning handler here */ - fgState.WarningFunc(fmt, ap); - - va_end( ap ); - - } else { - - va_start( ap, fmt ); - - fprintf( stderr, "freeglut "); - if( fgState.ProgramName ) - fprintf( stderr, "(%s): ", fgState.ProgramName ); - VFPRINTF( stderr, fmt, ap ); - fprintf( stderr, "\n" ); - - va_end( ap ); - } -} - - -/* - * Indicates whether Joystick events are being used by ANY window. - * - * The current mechanism is to walk all of the windows and ask if - * there is a joystick callback. We have a short-circuit early - * return if we find any joystick handler registered. - * - * The real way to do this is to make use of the glutTimer() API - * to more cleanly re-implement the joystick API. Then, this code - * and all other "joystick timer" code can be yanked. - * - */ -static void fghCheckJoystickCallback( SFG_Window* w, SFG_Enumerator* e) -{ - if( FETCH_WCB( *w, Joystick ) ) - { - e->found = GL_TRUE; - e->data = w; - } - fgEnumSubWindows( w, fghCheckJoystickCallback, e ); -} -static int fghHaveJoystick( void ) -{ - SFG_Enumerator enumerator; - - enumerator.found = GL_FALSE; - enumerator.data = NULL; - fgEnumWindows( fghCheckJoystickCallback, &enumerator ); - return !!enumerator.data; -} -static void fghHavePendingRedisplaysCallback( SFG_Window* w, SFG_Enumerator* e) -{ - if( w->State.Redisplay && w->State.Visible ) - { - e->found = GL_TRUE; - e->data = w; - } - fgEnumSubWindows( w, fghHavePendingRedisplaysCallback, e ); -} -static int fghHavePendingRedisplays (void) -{ - SFG_Enumerator enumerator; - - enumerator.found = GL_FALSE; - enumerator.data = NULL; - fgEnumWindows( fghHavePendingRedisplaysCallback, &enumerator ); - return !!enumerator.data; -} -/* - * Returns the number of GLUT ticks (milliseconds) till the next timer event. - */ -static long fghNextTimer( void ) -{ - long ret = INT_MAX; - SFG_Timer *timer = fgState.Timers.First; - - if( timer ) - ret = timer->TriggerTime - fgElapsedTime(); - if( ret < 0 ) - ret = 0; - - return ret; -} -/* - * Does the magic required to relinquish the CPU until something interesting - * happens. - */ -static void fghSleepForEvents( void ) -{ - long msec; - - if( fgState.IdleCallback || fghHavePendingRedisplays( ) ) - return; - - msec = fghNextTimer( ); - /* XXX Use GLUT timers for joysticks... */ - /* XXX Dumb; forces granularity to .01sec */ - if( fghHaveJoystick( ) && ( msec > 10 ) ) - msec = 10; - -#if TARGET_HOST_POSIX_X11 - /* - * Possibly due to aggressive use of XFlush() and friends, - * it is possible to have our socket drained but still have - * unprocessed events. (Or, this may just be normal with - * X, anyway?) We do non-trivial processing of X events - * after the event-reading loop, in any case, so we - * need to allow that we may have an empty socket but non- - * empty event queue. - */ - if( ! XPending( fgDisplay.Display ) ) - { - fd_set fdset; - int err; - int socket; - struct timeval wait; - - socket = ConnectionNumber( fgDisplay.Display ); - FD_ZERO( &fdset ); - FD_SET( socket, &fdset ); - wait.tv_sec = msec / 1000; - wait.tv_usec = (msec % 1000) * 1000; - err = select( socket+1, &fdset, NULL, NULL, &wait ); - -#ifdef HAVE_ERRNO_H - if( ( -1 == err ) && ( errno != EINTR ) ) - fgWarning ( "freeglut select() error: %d", errno ); -#endif - } -#elif TARGET_HOST_MS_WINDOWS - MsgWaitForMultipleObjects( 0, NULL, FALSE, msec, QS_ALLINPUT ); -#endif -} - -#if TARGET_HOST_POSIX_X11 -/* - * Returns GLUT modifier mask for the state field of an X11 event. - */ -int fghGetXModifiers( int state ) -{ - int ret = 0; - - if( state & ( ShiftMask | LockMask ) ) - ret |= GLUT_ACTIVE_SHIFT; - if( state & ControlMask ) - ret |= GLUT_ACTIVE_CTRL; - if( state & Mod1Mask ) - ret |= GLUT_ACTIVE_ALT; - - return ret; -} -#endif - - -#if TARGET_HOST_POSIX_X11 && _DEBUG - -static const char* fghTypeToString( int type ) -{ - switch( type ) { - case KeyPress: return "KeyPress"; - case KeyRelease: return "KeyRelease"; - case ButtonPress: return "ButtonPress"; - case ButtonRelease: return "ButtonRelease"; - case MotionNotify: return "MotionNotify"; - case EnterNotify: return "EnterNotify"; - case LeaveNotify: return "LeaveNotify"; - case FocusIn: return "FocusIn"; - case FocusOut: return "FocusOut"; - case KeymapNotify: return "KeymapNotify"; - case Expose: return "Expose"; - case GraphicsExpose: return "GraphicsExpose"; - case NoExpose: return "NoExpose"; - case VisibilityNotify: return "VisibilityNotify"; - case CreateNotify: return "CreateNotify"; - case DestroyNotify: return "DestroyNotify"; - case UnmapNotify: return "UnmapNotify"; - case MapNotify: return "MapNotify"; - case MapRequest: return "MapRequest"; - case ReparentNotify: return "ReparentNotify"; - case ConfigureNotify: return "ConfigureNotify"; - case ConfigureRequest: return "ConfigureRequest"; - case GravityNotify: return "GravityNotify"; - case ResizeRequest: return "ResizeRequest"; - case CirculateNotify: return "CirculateNotify"; - case CirculateRequest: return "CirculateRequest"; - case PropertyNotify: return "PropertyNotify"; - case SelectionClear: return "SelectionClear"; - case SelectionRequest: return "SelectionRequest"; - case SelectionNotify: return "SelectionNotify"; - case ColormapNotify: return "ColormapNotify"; - case ClientMessage: return "ClientMessage"; - case MappingNotify: return "MappingNotify"; - default: return "UNKNOWN"; - } -} - -static const char* fghBoolToString( Bool b ) -{ - return b == False ? "False" : "True"; -} - -static const char* fghNotifyHintToString( char is_hint ) -{ - switch( is_hint ) { - case NotifyNormal: return "NotifyNormal"; - case NotifyHint: return "NotifyHint"; - default: return "UNKNOWN"; - } -} - -static const char* fghNotifyModeToString( int mode ) -{ - switch( mode ) { - case NotifyNormal: return "NotifyNormal"; - case NotifyGrab: return "NotifyGrab"; - case NotifyUngrab: return "NotifyUngrab"; - case NotifyWhileGrabbed: return "NotifyWhileGrabbed"; - default: return "UNKNOWN"; - } -} - -static const char* fghNotifyDetailToString( int detail ) -{ - switch( detail ) { - case NotifyAncestor: return "NotifyAncestor"; - case NotifyVirtual: return "NotifyVirtual"; - case NotifyInferior: return "NotifyInferior"; - case NotifyNonlinear: return "NotifyNonlinear"; - case NotifyNonlinearVirtual: return "NotifyNonlinearVirtual"; - case NotifyPointer: return "NotifyPointer"; - case NotifyPointerRoot: return "NotifyPointerRoot"; - case NotifyDetailNone: return "NotifyDetailNone"; - default: return "UNKNOWN"; - } -} - -static const char* fghVisibilityToString( int state ) { - switch( state ) { - case VisibilityUnobscured: return "VisibilityUnobscured"; - case VisibilityPartiallyObscured: return "VisibilityPartiallyObscured"; - case VisibilityFullyObscured: return "VisibilityFullyObscured"; - default: return "UNKNOWN"; - } -} - -static const char* fghConfigureDetailToString( int detail ) -{ - switch( detail ) { - case Above: return "Above"; - case Below: return "Below"; - case TopIf: return "TopIf"; - case BottomIf: return "BottomIf"; - case Opposite: return "Opposite"; - default: return "UNKNOWN"; - } -} - -static const char* fghPlaceToString( int place ) -{ - switch( place ) { - case PlaceOnTop: return "PlaceOnTop"; - case PlaceOnBottom: return "PlaceOnBottom"; - default: return "UNKNOWN"; - } -} - -static const char* fghMappingRequestToString( int request ) -{ - switch( request ) { - case MappingModifier: return "MappingModifier"; - case MappingKeyboard: return "MappingKeyboard"; - case MappingPointer: return "MappingPointer"; - default: return "UNKNOWN"; - } -} - -static const char* fghPropertyStateToString( int state ) -{ - switch( state ) { - case PropertyNewValue: return "PropertyNewValue"; - case PropertyDelete: return "PropertyDelete"; - default: return "UNKNOWN"; - } -} - -static const char* fghColormapStateToString( int state ) -{ - switch( state ) { - case ColormapUninstalled: return "ColormapUninstalled"; - case ColormapInstalled: return "ColormapInstalled"; - default: return "UNKNOWN"; - } -} - -static void fghPrintEvent( XEvent *event ) -{ - switch( event->type ) { - - case KeyPress: - case KeyRelease: { - XKeyEvent *e = &event->xkey; - fgWarning( "%s: window=0x%x, root=0x%x, subwindow=0x%x, time=%lu, " - "(x,y)=(%d,%d), (x_root,y_root)=(%d,%d), state=0x%x, " - "keycode=%u, same_screen=%s", fghTypeToString( e->type ), - e->window, e->root, e->subwindow, (unsigned long)e->time, - e->x, e->y, e->x_root, e->y_root, e->state, e->keycode, - fghBoolToString( e->same_screen ) ); - break; - } - - case ButtonPress: - case ButtonRelease: { - XButtonEvent *e = &event->xbutton; - fgWarning( "%s: window=0x%x, root=0x%x, subwindow=0x%x, time=%lu, " - "(x,y)=(%d,%d), (x_root,y_root)=(%d,%d), state=0x%x, " - "button=%u, same_screen=%d", fghTypeToString( e->type ), - e->window, e->root, e->subwindow, (unsigned long)e->time, - e->x, e->y, e->x_root, e->y_root, e->state, e->button, - fghBoolToString( e->same_screen ) ); - break; - } - - case MotionNotify: { - XMotionEvent *e = &event->xmotion; - fgWarning( "%s: window=0x%x, root=0x%x, subwindow=0x%x, time=%lu, " - "(x,y)=(%d,%d), (x_root,y_root)=(%d,%d), state=0x%x, " - "is_hint=%s, same_screen=%d", fghTypeToString( e->type ), - e->window, e->root, e->subwindow, (unsigned long)e->time, - e->x, e->y, e->x_root, e->y_root, e->state, - fghNotifyHintToString( e->is_hint ), - fghBoolToString( e->same_screen ) ); - break; - } - - case EnterNotify: - case LeaveNotify: { - XCrossingEvent *e = &event->xcrossing; - fgWarning( "%s: window=0x%x, root=0x%x, subwindow=0x%x, time=%lu, " - "(x,y)=(%d,%d), mode=%s, detail=%s, same_screen=%d, " - "focus=%d, state=0x%x", fghTypeToString( e->type ), - e->window, e->root, e->subwindow, (unsigned long)e->time, - e->x, e->y, fghNotifyModeToString( e->mode ), - fghNotifyDetailToString( e->detail ), (int)e->same_screen, - (int)e->focus, e->state ); - break; - } - - case FocusIn: - case FocusOut: { - XFocusChangeEvent *e = &event->xfocus; - fgWarning( "%s: window=0x%x, mode=%s, detail=%s", - fghTypeToString( e->type ), e->window, - fghNotifyModeToString( e->mode ), - fghNotifyDetailToString( e->detail ) ); - break; - } - - case KeymapNotify: { - XKeymapEvent *e = &event->xkeymap; - char buf[32 * 2 + 1]; - int i; - for ( i = 0; i < 32; i++ ) { - snprintf( &buf[ i * 2 ], sizeof( buf ) - i * 2, - "%02x", e->key_vector[ i ] ); - } - buf[ i ] = '\0'; - fgWarning( "%s: window=0x%x, %s", fghTypeToString( e->type ), e->window, - buf ); - break; - } - - case Expose: { - XExposeEvent *e = &event->xexpose; - fgWarning( "%s: window=0x%x, (x,y)=(%d,%d), (width,height)=(%d,%d), " - "count=%d", fghTypeToString( e->type ), e->window, e->x, - e->y, e->width, e->height, e->count ); - break; - } - - case GraphicsExpose: { - XGraphicsExposeEvent *e = &event->xgraphicsexpose; - fgWarning( "%s: drawable=0x%x, (x,y)=(%d,%d), (width,height)=(%d,%d), " - "count=%d, (major_code,minor_code)=(%d,%d)", - fghTypeToString( e->type ), e->drawable, e->x, e->y, - e->width, e->height, e->count, e->major_code, - e->minor_code ); - break; - } - - case NoExpose: { - XNoExposeEvent *e = &event->xnoexpose; - fgWarning( "%s: drawable=0x%x, (major_code,minor_code)=(%d,%d)", - fghTypeToString( e->type ), e->drawable, e->major_code, - e->minor_code ); - break; - } - - case VisibilityNotify: { - XVisibilityEvent *e = &event->xvisibility; - fgWarning( "%s: window=0x%x, state=%s", fghTypeToString( e->type ), - e->window, fghVisibilityToString( e->state) ); - break; - } - - case CreateNotify: { - XCreateWindowEvent *e = &event->xcreatewindow; - fgWarning( "%s: (x,y)=(%d,%d), (width,height)=(%d,%d), border_width=%d, " - "window=0x%x, override_redirect=%s", - fghTypeToString( e->type ), e->x, e->y, e->width, e->height, - e->border_width, e->window, - fghBoolToString( e->override_redirect ) ); - break; - } - - case DestroyNotify: { - XDestroyWindowEvent *e = &event->xdestroywindow; - fgWarning( "%s: event=0x%x, window=0x%x", - fghTypeToString( e->type ), e->event, e->window ); - break; - } - - case UnmapNotify: { - XUnmapEvent *e = &event->xunmap; - fgWarning( "%s: event=0x%x, window=0x%x, from_configure=%s", - fghTypeToString( e->type ), e->event, e->window, - fghBoolToString( e->from_configure ) ); - break; - } - - case MapNotify: { - XMapEvent *e = &event->xmap; - fgWarning( "%s: event=0x%x, window=0x%x, override_redirect=%s", - fghTypeToString( e->type ), e->event, e->window, - fghBoolToString( e->override_redirect ) ); - break; - } - - case MapRequest: { - XMapRequestEvent *e = &event->xmaprequest; - fgWarning( "%s: parent=0x%x, window=0x%x", - fghTypeToString( event->type ), e->parent, e->window ); - break; - } - - case ReparentNotify: { - XReparentEvent *e = &event->xreparent; - fgWarning( "%s: event=0x%x, window=0x%x, parent=0x%x, (x,y)=(%d,%d), " - "override_redirect=%s", fghTypeToString( e->type ), - e->event, e->window, e->parent, e->x, e->y, - fghBoolToString( e->override_redirect ) ); - break; - } - - case ConfigureNotify: { - XConfigureEvent *e = &event->xconfigure; - fgWarning( "%s: event=0x%x, window=0x%x, (x,y)=(%d,%d), " - "(width,height)=(%d,%d), border_width=%d, above=0x%x, " - "override_redirect=%s", fghTypeToString( e->type ), e->event, - e->window, e->x, e->y, e->width, e->height, e->border_width, - e->above, fghBoolToString( e->override_redirect ) ); - break; - } - - case ConfigureRequest: { - XConfigureRequestEvent *e = &event->xconfigurerequest; - fgWarning( "%s: parent=0x%x, window=0x%x, (x,y)=(%d,%d), " - "(width,height)=(%d,%d), border_width=%d, above=0x%x, " - "detail=%s, value_mask=%lx", fghTypeToString( e->type ), - e->parent, e->window, e->x, e->y, e->width, e->height, - e->border_width, e->above, - fghConfigureDetailToString( e->detail ), e->value_mask ); - break; - } - - case GravityNotify: { - XGravityEvent *e = &event->xgravity; - fgWarning( "%s: event=0x%x, window=0x%x, (x,y)=(%d,%d)", - fghTypeToString( e->type ), e->event, e->window, e->x, e->y ); - break; - } - - case ResizeRequest: { - XResizeRequestEvent *e = &event->xresizerequest; - fgWarning( "%s: window=0x%x, (width,height)=(%d,%d)", - fghTypeToString( e->type ), e->window, e->width, e->height ); - break; - } - - case CirculateNotify: { - XCirculateEvent *e = &event->xcirculate; - fgWarning( "%s: event=0x%x, window=0x%x, place=%s", - fghTypeToString( e->type ), e->event, e->window, - fghPlaceToString( e->place ) ); - break; - } - - case CirculateRequest: { - XCirculateRequestEvent *e = &event->xcirculaterequest; - fgWarning( "%s: parent=0x%x, window=0x%x, place=%s", - fghTypeToString( e->type ), e->parent, e->window, - fghPlaceToString( e->place ) ); - break; - } - - case PropertyNotify: { - XPropertyEvent *e = &event->xproperty; - fgWarning( "%s: window=0x%x, atom=%lu, time=%lu, state=%s", - fghTypeToString( e->type ), e->window, - (unsigned long)e->atom, (unsigned long)e->time, - fghPropertyStateToString( e->state ) ); - break; - } - - case SelectionClear: { - XSelectionClearEvent *e = &event->xselectionclear; - fgWarning( "%s: window=0x%x, selection=%lu, time=%lu", - fghTypeToString( e->type ), e->window, - (unsigned long)e->selection, (unsigned long)e->time ); - break; - } - - case SelectionRequest: { - XSelectionRequestEvent *e = &event->xselectionrequest; - fgWarning( "%s: owner=0x%x, requestor=0x%x, selection=0x%x, " - "target=0x%x, property=%lu, time=%lu", - fghTypeToString( e->type ), e->owner, e->requestor, - (unsigned long)e->selection, (unsigned long)e->target, - (unsigned long)e->property, (unsigned long)e->time ); - break; - } - - case SelectionNotify: { - XSelectionEvent *e = &event->xselection; - fgWarning( "%s: requestor=0x%x, selection=0x%x, target=0x%x, " - "property=%lu, time=%lu", fghTypeToString( e->type ), - e->requestor, (unsigned long)e->selection, - (unsigned long)e->target, (unsigned long)e->property, - (unsigned long)e->time ); - break; - } - - case ColormapNotify: { - XColormapEvent *e = &event->xcolormap; - fgWarning( "%s: window=0x%x, colormap=%lu, new=%s, state=%s", - fghTypeToString( e->type ), e->window, - (unsigned long)e->colormap, fghBoolToString( e->new ), - fghColormapStateToString( e->state ) ); - break; - } - - case ClientMessage: { - XClientMessageEvent *e = &event->xclient; - char buf[ 61 ]; - char* p = buf; - char* end = buf + sizeof( buf ); - int i; - switch( e->format ) { - case 8: - for ( i = 0; i < 20; i++, p += 3 ) { - snprintf( p, end - p, " %02x", e->data.b[ i ] ); - } - break; - case 16: - for ( i = 0; i < 10; i++, p += 5 ) { - snprintf( p, end - p, " %04x", e->data.s[ i ] ); - } - break; - case 32: - for ( i = 0; i < 5; i++, p += 9 ) { - snprintf( p, end - p, " %08lx", e->data.l[ i ] ); - } - break; - } - *p = '\0'; - fgWarning( "%s: window=0x%x, message_type=%lu, format=%d, data=(%s )", - fghTypeToString( e->type ), e->window, - (unsigned long)e->message_type, e->format, buf ); - break; - } - - case MappingNotify: { - XMappingEvent *e = &event->xmapping; - fgWarning( "%s: window=0x%x, request=%s, first_keycode=%d, count=%d", - fghTypeToString( e->type ), e->window, - fghMappingRequestToString( e->request ), e->first_keycode, - e->count ); - break; - } - - default: { - fgWarning( "%s", fghTypeToString( event->type ) ); - break; - } - } -} - -#endif - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * Executes a single iteration in the freeglut processing loop. - */ -void FGAPIENTRY glutMainLoopEvent( void ) -{ -#if TARGET_HOST_POSIX_X11 - SFG_Window* window; - XEvent event; - - /* This code was repeated constantly, so here it goes into a definition: */ -#define GETWINDOW(a) \ - window = fgWindowByHandle( event.a.window ); \ - if( window == NULL ) \ - break; - -#define GETMOUSE(a) \ - window->State.MouseX = event.a.x; \ - window->State.MouseY = event.a.y; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMainLoopEvent" ); - - while( XPending( fgDisplay.Display ) ) - { - XNextEvent( fgDisplay.Display, &event ); -#if _DEBUG - fghPrintEvent( &event ); -#endif - - switch( event.type ) - { - case ClientMessage: - if(fgIsSpaceballXEvent(&event)) { - fgSpaceballHandleXEvent(&event); - break; - } - /* Destroy the window when the WM_DELETE_WINDOW message arrives */ - if( (Atom) event.xclient.data.l[ 0 ] == fgDisplay.DeleteWindow ) - { - GETWINDOW( xclient ); - - fgDestroyWindow ( window ); - - if( fgState.ActionOnWindowClose == GLUT_ACTION_EXIT ) - { - fgDeinitialize( ); - exit( 0 ); - } - else if( fgState.ActionOnWindowClose == GLUT_ACTION_GLUTMAINLOOP_RETURNS ) - fgState.ExecState = GLUT_EXEC_STATE_STOP; - - return; - } - break; - - /* - * CreateNotify causes a configure-event so that sub-windows are - * handled compatibly with GLUT. Otherwise, your sub-windows - * (in freeglut only) will not get an initial reshape event, - * which can break things. - * - * GLUT presumably does this because it generally tries to treat - * sub-windows the same as windows. - */ - case CreateNotify: - case ConfigureNotify: - { - int width, height; - if( event.type == CreateNotify ) { - GETWINDOW( xcreatewindow ); - width = event.xcreatewindow.width; - height = event.xcreatewindow.height; - } else { - GETWINDOW( xconfigure ); - width = event.xconfigure.width; - height = event.xconfigure.height; - } - - if( ( width != window->State.OldWidth ) || - ( height != window->State.OldHeight ) ) - { - SFG_Window *current_window = fgStructure.CurrentWindow; - - window->State.OldWidth = width; - window->State.OldHeight = height; - if( FETCH_WCB( *window, Reshape ) ) - INVOKE_WCB( *window, Reshape, ( width, height ) ); - else - { - fgSetWindow( window ); - glViewport( 0, 0, width, height ); - } - glutPostRedisplay( ); - if( window->IsMenu ) - fgSetWindow( current_window ); - } - } - break; - - case DestroyNotify: - /* - * This is sent to confirm the XDestroyWindow call. - * - * XXX WHY is this commented out? Should we re-enable it? - */ - /* fgAddToWindowDestroyList ( window ); */ - break; - - case Expose: - /* - * We are too dumb to process partial exposes... - * - * XXX Well, we could do it. However, it seems to only - * XXX be potentially useful for single-buffered (since - * XXX double-buffered does not respect viewport when we - * XXX do a buffer-swap). - * - */ - if( event.xexpose.count == 0 ) - { - GETWINDOW( xexpose ); - window->State.Redisplay = GL_TRUE; - } - break; - - case MapNotify: - break; - - case UnmapNotify: - /* We get this when iconifying a window. */ - GETWINDOW( xunmap ); - INVOKE_WCB( *window, WindowStatus, ( GLUT_HIDDEN ) ); - window->State.Visible = GL_FALSE; - break; - - case MappingNotify: - /* - * Have the client's keyboard knowledge updated (xlib.ps, - * page 206, says that's a good thing to do) - */ - XRefreshKeyboardMapping( (XMappingEvent *) &event ); - break; - - case VisibilityNotify: - { - /* - * Sending this event, the X server can notify us that the window - * has just acquired one of the three possible visibility states: - * VisibilityUnobscured, VisibilityPartiallyObscured or - * VisibilityFullyObscured. Note that we DO NOT receive a - * VisibilityNotify event when iconifying a window, we only get an - * UnmapNotify then. - */ - GETWINDOW( xvisibility ); - switch( event.xvisibility.state ) - { - case VisibilityUnobscured: - INVOKE_WCB( *window, WindowStatus, ( GLUT_FULLY_RETAINED ) ); - window->State.Visible = GL_TRUE; - break; - - case VisibilityPartiallyObscured: - INVOKE_WCB( *window, WindowStatus, - ( GLUT_PARTIALLY_RETAINED ) ); - window->State.Visible = GL_TRUE; - break; - - case VisibilityFullyObscured: - INVOKE_WCB( *window, WindowStatus, ( GLUT_FULLY_COVERED ) ); - window->State.Visible = GL_FALSE; - break; - - default: - fgWarning( "Unknown X visibility state: %d", - event.xvisibility.state ); - break; - } - } - break; - - case EnterNotify: - case LeaveNotify: - GETWINDOW( xcrossing ); - GETMOUSE( xcrossing ); - if( ( event.type == LeaveNotify ) && window->IsMenu && - window->ActiveMenu && window->ActiveMenu->IsActive ) - fgUpdateMenuHighlight( window->ActiveMenu ); - - INVOKE_WCB( *window, Entry, ( ( EnterNotify == event.type ) ? - GLUT_ENTERED : - GLUT_LEFT ) ); - break; - - case MotionNotify: - { - GETWINDOW( xmotion ); - GETMOUSE( xmotion ); - - if( window->ActiveMenu ) - { - if( window == window->ActiveMenu->ParentWindow ) - { - window->ActiveMenu->Window->State.MouseX = - event.xmotion.x_root - window->ActiveMenu->X; - window->ActiveMenu->Window->State.MouseY = - event.xmotion.y_root - window->ActiveMenu->Y; - } - - fgUpdateMenuHighlight( window->ActiveMenu ); - - break; - } - - /* - * XXX For more than 5 buttons, just check {event.xmotion.state}, - * XXX rather than a host of bit-masks? Or maybe we need to - * XXX track ButtonPress/ButtonRelease events in our own - * XXX bit-mask? - */ - fgState.Modifiers = fghGetXModifiers( event.xmotion.state ); - if ( event.xmotion.state & ( Button1Mask | Button2Mask | Button3Mask | Button4Mask | Button5Mask ) ) { - INVOKE_WCB( *window, Motion, ( event.xmotion.x, - event.xmotion.y ) ); - } else { - INVOKE_WCB( *window, Passive, ( event.xmotion.x, - event.xmotion.y ) ); - } - fgState.Modifiers = INVALID_MODIFIERS; - } - break; - - case ButtonRelease: - case ButtonPress: - { - GLboolean pressed = GL_TRUE; - int button; - - if( event.type == ButtonRelease ) - pressed = GL_FALSE ; - - /* - * A mouse button has been pressed or released. Traditionally, - * break if the window was found within the freeglut structures. - */ - GETWINDOW( xbutton ); - GETMOUSE( xbutton ); - - /* - * An X button (at least in XFree86) is numbered from 1. - * A GLUT button is numbered from 0. - * Old GLUT passed through buttons other than just the first - * three, though it only gave symbolic names and official - * support to the first three. - */ - button = event.xbutton.button - 1; - - /* - * Do not execute the application's mouse callback if a menu - * is hooked to this button. In that case an appropriate - * private call should be generated. - */ - if( fgCheckActiveMenu( window, button, pressed, - event.xbutton.x_root, event.xbutton.y_root ) ) - break; - - /* - * Check if there is a mouse or mouse wheel callback hooked to the - * window - */ - if( ! FETCH_WCB( *window, Mouse ) && - ! FETCH_WCB( *window, MouseWheel ) ) - break; - - fgState.Modifiers = fghGetXModifiers( event.xbutton.state ); - - /* Finally execute the mouse or mouse wheel callback */ - if( ( button < glutDeviceGet ( GLUT_NUM_MOUSE_BUTTONS ) ) || ( ! FETCH_WCB( *window, MouseWheel ) ) ) - INVOKE_WCB( *window, Mouse, ( button, - pressed ? GLUT_DOWN : GLUT_UP, - event.xbutton.x, - event.xbutton.y ) - ); - else - { - /* - * Map 4 and 5 to wheel zero; EVEN to +1, ODD to -1 - * " 6 and 7 " " one; ... - * - * XXX This *should* be behind some variables/macros, - * XXX since the order and numbering isn't certain - * XXX See XFree86 configuration docs (even back in the - * XXX 3.x days, and especially with 4.x). - * - * XXX Note that {button} has already been decremented - * XXX in mapping from X button numbering to GLUT. - * - * XXX Should add support for partial wheel turns as Windows does -- 5/27/11 - */ - int wheel_number = (button - glutDeviceGet ( GLUT_NUM_MOUSE_BUTTONS )) / 2; - int direction = -1; - if( button % 2 ) - direction = 1; - - if( pressed ) - INVOKE_WCB( *window, MouseWheel, ( wheel_number, - direction, - event.xbutton.x, - event.xbutton.y ) - ); - } - fgState.Modifiers = INVALID_MODIFIERS; - } - break; - - case KeyRelease: - case KeyPress: - { - FGCBKeyboard keyboard_cb; - FGCBSpecial special_cb; - - GETWINDOW( xkey ); - GETMOUSE( xkey ); - - /* Detect auto repeated keys, if configured globally or per-window */ - - if ( fgState.KeyRepeat==GLUT_KEY_REPEAT_OFF || window->State.IgnoreKeyRepeat==GL_TRUE ) - { - if (event.type==KeyRelease) - { - /* - * Look at X11 keystate to detect repeat mode. - * While X11 says the key is actually held down, we'll ignore KeyRelease/KeyPress pairs. - */ - - char keys[32]; - XQueryKeymap( fgDisplay.Display, keys ); /* Look at X11 keystate to detect repeat mode */ - - if ( event.xkey.keycode<256 ) /* XQueryKeymap is limited to 256 keycodes */ - { - if ( keys[event.xkey.keycode>>3] & (1<<(event.xkey.keycode%8)) ) - window->State.KeyRepeating = GL_TRUE; - else - window->State.KeyRepeating = GL_FALSE; - } - } - } - else - window->State.KeyRepeating = GL_FALSE; - - /* Cease processing this event if it is auto repeated */ - - if (window->State.KeyRepeating) - { - if (event.type == KeyPress) window->State.KeyRepeating = GL_FALSE; - break; - } - - if( event.type == KeyPress ) - { - keyboard_cb = (FGCBKeyboard)( FETCH_WCB( *window, Keyboard )); - special_cb = (FGCBSpecial) ( FETCH_WCB( *window, Special )); - } - else - { - keyboard_cb = (FGCBKeyboard)( FETCH_WCB( *window, KeyboardUp )); - special_cb = (FGCBSpecial) ( FETCH_WCB( *window, SpecialUp )); - } - - /* Is there a keyboard/special callback hooked for this window? */ - if( keyboard_cb || special_cb ) - { - XComposeStatus composeStatus; - char asciiCode[ 32 ]; - KeySym keySym; - int len; - - /* Check for the ASCII/KeySym codes associated with the event: */ - len = XLookupString( &event.xkey, asciiCode, sizeof(asciiCode), - &keySym, &composeStatus - ); - - /* GLUT API tells us to have two separate callbacks... */ - if( len > 0 ) - { - /* ...one for the ASCII translateable keypresses... */ - if( keyboard_cb ) - { - fgSetWindow( window ); - fgState.Modifiers = fghGetXModifiers( event.xkey.state ); - keyboard_cb( asciiCode[ 0 ], - event.xkey.x, event.xkey.y - ); - fgState.Modifiers = INVALID_MODIFIERS; - } - } - else - { - int special = -1; - - /* - * ...and one for all the others, which need to be - * translated to GLUT_KEY_Xs... - */ - switch( keySym ) - { - case XK_F1: special = GLUT_KEY_F1; break; - case XK_F2: special = GLUT_KEY_F2; break; - case XK_F3: special = GLUT_KEY_F3; break; - case XK_F4: special = GLUT_KEY_F4; break; - case XK_F5: special = GLUT_KEY_F5; break; - case XK_F6: special = GLUT_KEY_F6; break; - case XK_F7: special = GLUT_KEY_F7; break; - case XK_F8: special = GLUT_KEY_F8; break; - case XK_F9: special = GLUT_KEY_F9; break; - case XK_F10: special = GLUT_KEY_F10; break; - case XK_F11: special = GLUT_KEY_F11; break; - case XK_F12: special = GLUT_KEY_F12; break; - - case XK_KP_Left: - case XK_Left: special = GLUT_KEY_LEFT; break; - case XK_KP_Right: - case XK_Right: special = GLUT_KEY_RIGHT; break; - case XK_KP_Up: - case XK_Up: special = GLUT_KEY_UP; break; - case XK_KP_Down: - case XK_Down: special = GLUT_KEY_DOWN; break; - - case XK_KP_Prior: - case XK_Prior: special = GLUT_KEY_PAGE_UP; break; - case XK_KP_Next: - case XK_Next: special = GLUT_KEY_PAGE_DOWN; break; - case XK_KP_Home: - case XK_Home: special = GLUT_KEY_HOME; break; - case XK_KP_End: - case XK_End: special = GLUT_KEY_END; break; - case XK_KP_Insert: - case XK_Insert: special = GLUT_KEY_INSERT; break; - - case XK_Num_Lock : special = GLUT_KEY_NUM_LOCK; break; - case XK_KP_Begin : special = GLUT_KEY_BEGIN; break; - case XK_KP_Delete: special = GLUT_KEY_DELETE; break; - - case XK_Shift_L: special = GLUT_KEY_SHIFT_L; break; - case XK_Shift_R: special = GLUT_KEY_SHIFT_R; break; - case XK_Control_L: special = GLUT_KEY_CTRL_L; break; - case XK_Control_R: special = GLUT_KEY_CTRL_R; break; - case XK_Alt_L: special = GLUT_KEY_ALT_L; break; - case XK_Alt_R: special = GLUT_KEY_ALT_R; break; - } - - /* - * Execute the callback (if one has been specified), - * given that the special code seems to be valid... - */ - if( special_cb && (special != -1) ) - { - fgSetWindow( window ); - fgState.Modifiers = fghGetXModifiers( event.xkey.state ); - special_cb( special, event.xkey.x, event.xkey.y ); - fgState.Modifiers = INVALID_MODIFIERS; - } - } - } - } - break; - - case ReparentNotify: - break; /* XXX Should disable this event */ - - /* Not handled */ - case GravityNotify: - break; - - default: - /* enter handling of Extension Events here */ - #ifdef HAVE_X11_EXTENSIONS_XINPUT2_H - fgHandleExtensionEvents( &event ); - #endif - break; - } - } - -#elif TARGET_HOST_MS_WINDOWS - - MSG stMsg; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMainLoopEvent" ); - - while( PeekMessage( &stMsg, NULL, 0, 0, PM_NOREMOVE ) ) - { - if( GetMessage( &stMsg, NULL, 0, 0 ) == 0 ) - { - if( fgState.ActionOnWindowClose == GLUT_ACTION_EXIT ) - { - fgDeinitialize( ); - exit( 0 ); - } - else if( fgState.ActionOnWindowClose == GLUT_ACTION_GLUTMAINLOOP_RETURNS ) - fgState.ExecState = GLUT_EXEC_STATE_STOP; - - return; - } - - TranslateMessage( &stMsg ); - DispatchMessage( &stMsg ); - } -#endif - - if( fgState.Timers.First ) - fghCheckTimers( ); - fghCheckJoystickPolls( ); - fghDisplayAll( ); - - fgCloseWindows( ); -} - -/* - * Enters the freeglut processing loop. - * Stays until the "ExecState" changes to "GLUT_EXEC_STATE_STOP". - */ -void FGAPIENTRY glutMainLoop( void ) -{ - int action; - -#if TARGET_HOST_MS_WINDOWS - SFG_Window *window = (SFG_Window *)fgStructure.Windows.First ; -#endif - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutMainLoop" ); - -#if TARGET_HOST_MS_WINDOWS - /* - * Processing before the main loop: If there is a window which is open and - * which has a visibility callback, call it. I know this is an ugly hack, - * but I'm not sure what else to do about it. Ideally we should leave - * something uninitialized in the create window code and initialize it in - * the main loop, and have that initialization create a "WM_ACTIVATE" - * message. Then we would put the visibility callback code in the - * "case WM_ACTIVATE" block below. - John Fay -- 10/24/02 - */ - while( window ) - { - if ( FETCH_WCB( *window, Visibility ) ) - { - SFG_Window *current_window = fgStructure.CurrentWindow ; - - INVOKE_WCB( *window, Visibility, ( window->State.Visible ) ); - fgSetWindow( current_window ); - } - - window = (SFG_Window *)window->Node.Next ; - } -#endif - - fgState.ExecState = GLUT_EXEC_STATE_RUNNING ; - while( fgState.ExecState == GLUT_EXEC_STATE_RUNNING ) - { - SFG_Window *window; - - glutMainLoopEvent( ); - /* - * Step through the list of windows, seeing if there are any - * that are not menus - */ - for( window = ( SFG_Window * )fgStructure.Windows.First; - window; - window = ( SFG_Window * )window->Node.Next ) - if ( ! ( window->IsMenu ) ) - break; - - if( ! window ) - fgState.ExecState = GLUT_EXEC_STATE_STOP; - else - { - if( fgState.IdleCallback ) - { - if( fgStructure.CurrentWindow && - fgStructure.CurrentWindow->IsMenu ) - /* fail safe */ - fgSetWindow( window ); - fgState.IdleCallback( ); - } - - fghSleepForEvents( ); - } - } - - /* - * When this loop terminates, destroy the display, state and structure - * of a freeglut session, so that another glutInit() call can happen - * - * Save the "ActionOnWindowClose" because "fgDeinitialize" resets it. - */ - action = fgState.ActionOnWindowClose; - fgDeinitialize( ); - if( action == GLUT_ACTION_EXIT ) - exit( 0 ); -} - -/* - * Leaves the freeglut processing loop. - */ -void FGAPIENTRY glutLeaveMainLoop( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutLeaveMainLoop" ); - fgState.ExecState = GLUT_EXEC_STATE_STOP ; -} - - -#if TARGET_HOST_MS_WINDOWS -/* - * Determine a GLUT modifer mask based on MS-WINDOWS system info. - */ -static int fghGetWin32Modifiers (void) -{ - return - ( ( ( GetKeyState( VK_LSHIFT ) < 0 ) || - ( GetKeyState( VK_RSHIFT ) < 0 )) ? GLUT_ACTIVE_SHIFT : 0 ) | - ( ( ( GetKeyState( VK_LCONTROL ) < 0 ) || - ( GetKeyState( VK_RCONTROL ) < 0 )) ? GLUT_ACTIVE_CTRL : 0 ) | - ( ( ( GetKeyState( VK_LMENU ) < 0 ) || - ( GetKeyState( VK_RMENU ) < 0 )) ? GLUT_ACTIVE_ALT : 0 ); -} - -/* - * The window procedure for handling Win32 events - */ -LRESULT CALLBACK fgWindowProc( HWND hWnd, UINT uMsg, WPARAM wParam, - LPARAM lParam ) -{ - static unsigned char lControl = 0, rControl = 0, lShift = 0, - rShift = 0, lAlt = 0, rAlt = 0; - - SFG_Window* window; - PAINTSTRUCT ps; - LRESULT lRet = 1; - - FREEGLUT_INTERNAL_ERROR_EXIT_IF_NOT_INITIALISED ( "Event Handler" ) ; - - window = fgWindowByHandle( hWnd ); - - if ( ( window == NULL ) && ( uMsg != WM_CREATE ) ) - return DefWindowProc( hWnd, uMsg, wParam, lParam ); - - /* printf ( "Window %3d message <%04x> %12d %12d\n", window?window->ID:0, - uMsg, wParam, lParam ); */ - - if ( window ) - { - /* Checking for CTRL, ALT, and SHIFT key positions: Key Down! */ - if ( !lControl && GetAsyncKeyState ( VK_LCONTROL ) ) - { - INVOKE_WCB ( *window, Special, - ( GLUT_KEY_CTRL_L, window->State.MouseX, window->State.MouseY ) - ); - - lControl = 1; - } - - if ( !rControl && GetAsyncKeyState ( VK_RCONTROL ) ) - { - INVOKE_WCB ( *window, Special, - ( GLUT_KEY_CTRL_R, window->State.MouseX, window->State.MouseY ) - ); - - rControl = 1; - } - - if ( !lShift && GetAsyncKeyState ( VK_LSHIFT ) ) - { - INVOKE_WCB ( *window, Special, - ( GLUT_KEY_SHIFT_L, window->State.MouseX, window->State.MouseY ) - ); - - lShift = 1; - } - - if ( !rShift && GetAsyncKeyState ( VK_RSHIFT ) ) - { - INVOKE_WCB ( *window, Special, - ( GLUT_KEY_SHIFT_R, window->State.MouseX, window->State.MouseY ) - ); - - rShift = 1; - } - - if ( !lAlt && GetAsyncKeyState ( VK_LMENU ) ) - { - INVOKE_WCB ( *window, Special, - ( GLUT_KEY_ALT_L, window->State.MouseX, window->State.MouseY ) - ); - - lAlt = 1; - } - - if ( !rAlt && GetAsyncKeyState ( VK_RMENU ) ) - { - INVOKE_WCB ( *window, Special, - ( GLUT_KEY_ALT_R, window->State.MouseX, window->State.MouseY ) - ); - - rAlt = 1; - } - - /* Checking for CTRL, ALT, and SHIFT key positions: Key Up! */ - if ( lControl && !GetAsyncKeyState ( VK_LCONTROL ) ) - { - INVOKE_WCB ( *window, SpecialUp, - ( GLUT_KEY_CTRL_L, window->State.MouseX, window->State.MouseY ) - ); - - lControl = 0; - } - - if ( rControl && !GetAsyncKeyState ( VK_RCONTROL ) ) - { - INVOKE_WCB ( *window, SpecialUp, - ( GLUT_KEY_CTRL_R, window->State.MouseX, window->State.MouseY ) - ); - - rControl = 0; - } - - if ( lShift && !GetAsyncKeyState ( VK_LSHIFT ) ) - { - INVOKE_WCB ( *window, SpecialUp, - ( GLUT_KEY_SHIFT_L, window->State.MouseX, window->State.MouseY ) - ); - - lShift = 0; - } - - if ( rShift && !GetAsyncKeyState ( VK_RSHIFT ) ) - { - INVOKE_WCB ( *window, SpecialUp, - ( GLUT_KEY_SHIFT_R, window->State.MouseX, window->State.MouseY ) - ); - - rShift = 0; - } - - if ( lAlt && !GetAsyncKeyState ( VK_LMENU ) ) - { - INVOKE_WCB ( *window, SpecialUp, - ( GLUT_KEY_ALT_L, window->State.MouseX, window->State.MouseY ) - ); - - lAlt = 0; - } - - if ( rAlt && !GetAsyncKeyState ( VK_RMENU ) ) - { - INVOKE_WCB ( *window, SpecialUp, - ( GLUT_KEY_ALT_R, window->State.MouseX, window->State.MouseY ) - ); - - rAlt = 0; - } - } - - switch( uMsg ) - { - case WM_CREATE: - /* The window structure is passed as the creation structure parameter... */ - window = (SFG_Window *) (((LPCREATESTRUCT) lParam)->lpCreateParams); - FREEGLUT_INTERNAL_ERROR_EXIT ( ( window != NULL ), "Cannot create window", - "fgWindowProc" ); - - window->Window.Handle = hWnd; - window->Window.Device = GetDC( hWnd ); - if( window->IsMenu ) - { - unsigned int current_DisplayMode = fgState.DisplayMode; - fgState.DisplayMode = GLUT_DOUBLE | GLUT_RGB | GLUT_DEPTH; -#if !defined(_WIN32_WCE) - fgSetupPixelFormat( window, GL_FALSE, PFD_MAIN_PLANE ); -#endif - fgState.DisplayMode = current_DisplayMode; - - if( fgStructure.MenuContext ) - wglMakeCurrent( window->Window.Device, - fgStructure.MenuContext->MContext - ); - else - { - fgStructure.MenuContext = - (SFG_MenuContext *)malloc( sizeof(SFG_MenuContext) ); - fgStructure.MenuContext->MContext = - wglCreateContext( window->Window.Device ); - } - - /* window->Window.Context = wglGetCurrentContext (); */ - window->Window.Context = wglCreateContext( window->Window.Device ); - } - else - { -#if !defined(_WIN32_WCE) - fgSetupPixelFormat( window, GL_FALSE, PFD_MAIN_PLANE ); -#endif - - if( ! fgState.UseCurrentContext ) - window->Window.Context = - wglCreateContext( window->Window.Device ); - else - { - window->Window.Context = wglGetCurrentContext( ); - if( ! window->Window.Context ) - window->Window.Context = - wglCreateContext( window->Window.Device ); - } - -#if !defined(_WIN32_WCE) - fgNewWGLCreateContext( window ); -#endif - } - - window->State.NeedToResize = GL_TRUE; - /* if we used CW_USEDEFAULT (thats a negative value) for the size - * of the window, query the window now for the size at which it - * was created. - */ - if( ( window->State.Width < 0 ) || ( window->State.Height < 0 ) ) - { - SFG_Window *current_window = fgStructure.CurrentWindow; - - fgSetWindow( window ); - window->State.Width = glutGet( GLUT_WINDOW_WIDTH ); - window->State.Height = glutGet( GLUT_WINDOW_HEIGHT ); - fgSetWindow( current_window ); - } - - ReleaseDC( window->Window.Handle, window->Window.Device ); - -#if defined(_WIN32_WCE) - /* Take over button handling */ - { - HINSTANCE dxDllLib=LoadLibrary(_T("gx.dll")); - if (dxDllLib) - { - GXGetDefaultKeys_=(GXGETDEFAULTKEYS)GetProcAddress(dxDllLib, _T("?GXGetDefaultKeys@@YA?AUGXKeyList@@H@Z")); - GXOpenInput_=(GXOPENINPUT)GetProcAddress(dxDllLib, _T("?GXOpenInput@@YAHXZ")); - } - - if(GXOpenInput_) - (*GXOpenInput_)(); - if(GXGetDefaultKeys_) - gxKeyList = (*GXGetDefaultKeys_)(GX_LANDSCAPEKEYS); - } - -#endif /* defined(_WIN32_WCE) */ - break; - - case WM_SIZE: - /* - * If the window is visible, then it is the user manually resizing it. - * If it is not, then it is the system sending us a dummy resize with - * zero dimensions on a "glutIconifyWindow" call. - */ - if( window->State.Visible ) - { - window->State.NeedToResize = GL_TRUE; -#if defined(_WIN32_WCE) - window->State.Width = HIWORD(lParam); - window->State.Height = LOWORD(lParam); -#else - window->State.Width = LOWORD(lParam); - window->State.Height = HIWORD(lParam); -#endif /* defined(_WIN32_WCE) */ - } - - break; - - case WM_SETFOCUS: -/* printf("WM_SETFOCUS: %p\n", window ); */ - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - INVOKE_WCB( *window, Entry, ( GLUT_ENTERED ) ); - break; - - case WM_KILLFOCUS: -/* printf("WM_KILLFOCUS: %p\n", window ); */ - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - INVOKE_WCB( *window, Entry, ( GLUT_LEFT ) ); - - if( window->IsMenu && - window->ActiveMenu && window->ActiveMenu->IsActive ) - fgUpdateMenuHighlight( window->ActiveMenu ); - - break; - -#if 0 - case WM_ACTIVATE: - if (LOWORD(wParam) != WA_INACTIVE) - { -/* printf("WM_ACTIVATE: fgSetCursor( %p, %d)\n", window, - window->State.Cursor ); */ - fgSetCursor( window, window->State.Cursor ); - } - - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - break; -#endif - - case WM_SETCURSOR: -/* printf ( "Cursor event %x %x %x %x\n", window, window->State.Cursor, lParam, wParam ) ; */ - if( LOWORD( lParam ) == HTCLIENT ) - fgSetCursor ( window, window->State.Cursor ) ; - else - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - break; - - case WM_SHOWWINDOW: - window->State.Visible = GL_TRUE; - window->State.Redisplay = GL_TRUE; - break; - - case WM_PAINT: - /* Turn on the visibility in case it was turned off somehow */ - window->State.Visible = GL_TRUE; - BeginPaint( hWnd, &ps ); - fghRedrawWindow( window ); - EndPaint( hWnd, &ps ); - break; - - case WM_CLOSE: - fgDestroyWindow ( window ); - if ( fgState.ActionOnWindowClose != GLUT_ACTION_CONTINUE_EXECUTION ) - PostQuitMessage(0); - break; - - case WM_DESTROY: - /* - * The window already got destroyed, so don't bother with it. - */ - return 0; - - case WM_MOUSEMOVE: - { -#if defined(_WIN32_WCE) - window->State.MouseX = 320-HIWORD( lParam ); - window->State.MouseY = LOWORD( lParam ); -#else - window->State.MouseX = LOWORD( lParam ); - window->State.MouseY = HIWORD( lParam ); -#endif /* defined(_WIN32_WCE) */ - /* Restrict to [-32768, 32767] to match X11 behaviour */ - /* See comment in "freeglut_developer" mailing list 10/4/04 */ - if ( window->State.MouseX > 32767 ) window->State.MouseX -= 65536; - if ( window->State.MouseY > 32767 ) window->State.MouseY -= 65536; - - if ( window->ActiveMenu ) - { - fgUpdateMenuHighlight( window->ActiveMenu ); - break; - } - SetFocus(window->Window.Handle); - - fgState.Modifiers = fghGetWin32Modifiers( ); - - if( ( wParam & MK_LBUTTON ) || - ( wParam & MK_MBUTTON ) || - ( wParam & MK_RBUTTON ) ) - INVOKE_WCB( *window, Motion, ( window->State.MouseX, - window->State.MouseY ) ); - else - INVOKE_WCB( *window, Passive, ( window->State.MouseX, - window->State.MouseY ) ); - - fgState.Modifiers = INVALID_MODIFIERS; - } - break; - - case WM_LBUTTONDOWN: - case WM_MBUTTONDOWN: - case WM_RBUTTONDOWN: - case WM_LBUTTONUP: - case WM_MBUTTONUP: - case WM_RBUTTONUP: - { - GLboolean pressed = GL_TRUE; - int button; - -#if defined(_WIN32_WCE) - window->State.MouseX = 320-HIWORD( lParam ); - window->State.MouseY = LOWORD( lParam ); -#else - window->State.MouseX = LOWORD( lParam ); - window->State.MouseY = HIWORD( lParam ); -#endif /* defined(_WIN32_WCE) */ - - /* Restrict to [-32768, 32767] to match X11 behaviour */ - /* See comment in "freeglut_developer" mailing list 10/4/04 */ - if ( window->State.MouseX > 32767 ) window->State.MouseX -= 65536; - if ( window->State.MouseY > 32767 ) window->State.MouseY -= 65536; - - switch( uMsg ) - { - case WM_LBUTTONDOWN: - pressed = GL_TRUE; - button = GLUT_LEFT_BUTTON; - break; - case WM_MBUTTONDOWN: - pressed = GL_TRUE; - button = GLUT_MIDDLE_BUTTON; - break; - case WM_RBUTTONDOWN: - pressed = GL_TRUE; - button = GLUT_RIGHT_BUTTON; - break; - case WM_LBUTTONUP: - pressed = GL_FALSE; - button = GLUT_LEFT_BUTTON; - break; - case WM_MBUTTONUP: - pressed = GL_FALSE; - button = GLUT_MIDDLE_BUTTON; - break; - case WM_RBUTTONUP: - pressed = GL_FALSE; - button = GLUT_RIGHT_BUTTON; - break; - default: - pressed = GL_FALSE; - button = -1; - break; - } - -#if !defined(_WIN32_WCE) - if( GetSystemMetrics( SM_SWAPBUTTON ) ) - { - if( button == GLUT_LEFT_BUTTON ) - button = GLUT_RIGHT_BUTTON; - else - if( button == GLUT_RIGHT_BUTTON ) - button = GLUT_LEFT_BUTTON; - } -#endif /* !defined(_WIN32_WCE) */ - - if( button == -1 ) - return DefWindowProc( hWnd, uMsg, lParam, wParam ); - - /* - * Do not execute the application's mouse callback if a menu - * is hooked to this button. In that case an appropriate - * private call should be generated. - */ - if( fgCheckActiveMenu( window, button, pressed, - window->State.MouseX, window->State.MouseY ) ) - break; - - /* Set capture so that the window captures all the mouse messages */ - /* - * XXX - Multiple button support: Under X11, the mouse is not released - * XXX - from the window until all buttons have been released, even if the - * XXX - user presses a button in another window. This will take more - * XXX - code changes than I am up to at the moment (10/5/04). The present - * XXX - is a 90 percent solution. - */ - if ( pressed == GL_TRUE ) - SetCapture ( window->Window.Handle ) ; - else - ReleaseCapture () ; - - if( ! FETCH_WCB( *window, Mouse ) ) - break; - - fgSetWindow( window ); - fgState.Modifiers = fghGetWin32Modifiers( ); - - INVOKE_WCB( - *window, Mouse, - ( button, - pressed ? GLUT_DOWN : GLUT_UP, - window->State.MouseX, - window->State.MouseY - ) - ); - - fgState.Modifiers = INVALID_MODIFIERS; - } - break; - - case 0x020a: - /* Should be WM_MOUSEWHEEL but my compiler doesn't recognize it */ - { - int wheel_number = LOWORD( wParam ); - short ticks = ( short )HIWORD( wParam ); - fgState.MouseWheelTicks += ticks; - - /* - * XXX Should use WHEEL_DELTA instead of 120 - */ - if ( abs ( fgState.MouseWheelTicks ) > 120 ) - { - int direction = ( fgState.MouseWheelTicks > 0 ) ? 1 : -1; - - if( ! FETCH_WCB( *window, MouseWheel ) && - ! FETCH_WCB( *window, Mouse ) ) - break; - - fgSetWindow( window ); - fgState.Modifiers = fghGetWin32Modifiers( ); - - /* - * XXX Should use WHEEL_DELTA instead of 120 - */ - while( abs ( fgState.MouseWheelTicks ) > 120 ) - { - if( FETCH_WCB( *window, MouseWheel ) ) - INVOKE_WCB( *window, MouseWheel, - ( wheel_number, - direction, - window->State.MouseX, - window->State.MouseY - ) - ); - else /* No mouse wheel, call the mouse button callback twice */ - { - /* - * Map wheel zero to button 3 and 4; +1 to 3, -1 to 4 - * " " one +1 to 5, -1 to 6, ... - * - * XXX The below assumes that you have no more than 3 mouse - * XXX buttons. Sorry. - */ - int button = wheel_number * 2 + 3; - if( direction < 0 ) - ++button; - INVOKE_WCB( *window, Mouse, - ( button, GLUT_DOWN, - window->State.MouseX, window->State.MouseY ) - ); - INVOKE_WCB( *window, Mouse, - ( button, GLUT_UP, - window->State.MouseX, window->State.MouseY ) - ); - } - - /* - * XXX Should use WHEEL_DELTA instead of 120 - */ - fgState.MouseWheelTicks -= 120 * direction; - } - - fgState.Modifiers = INVALID_MODIFIERS; - } - } - break ; - - case WM_SYSKEYDOWN: - case WM_KEYDOWN: - { - int keypress = -1; - POINT mouse_pos ; - - if( ( fgState.KeyRepeat==GLUT_KEY_REPEAT_OFF || window->State.IgnoreKeyRepeat==GL_TRUE ) && (HIWORD(lParam) & KF_REPEAT) ) - break; - - /* - * Remember the current modifiers state. This is done here in order - * to make sure the VK_DELETE keyboard callback is executed properly. - */ - fgState.Modifiers = fghGetWin32Modifiers( ); - - GetCursorPos( &mouse_pos ); - ScreenToClient( window->Window.Handle, &mouse_pos ); - - window->State.MouseX = mouse_pos.x; - window->State.MouseY = mouse_pos.y; - - /* Convert the Win32 keystroke codes to GLUTtish way */ -# define KEY(a,b) case a: keypress = b; break; - - switch( wParam ) - { - KEY( VK_F1, GLUT_KEY_F1 ); - KEY( VK_F2, GLUT_KEY_F2 ); - KEY( VK_F3, GLUT_KEY_F3 ); - KEY( VK_F4, GLUT_KEY_F4 ); - KEY( VK_F5, GLUT_KEY_F5 ); - KEY( VK_F6, GLUT_KEY_F6 ); - KEY( VK_F7, GLUT_KEY_F7 ); - KEY( VK_F8, GLUT_KEY_F8 ); - KEY( VK_F9, GLUT_KEY_F9 ); - KEY( VK_F10, GLUT_KEY_F10 ); - KEY( VK_F11, GLUT_KEY_F11 ); - KEY( VK_F12, GLUT_KEY_F12 ); - KEY( VK_PRIOR, GLUT_KEY_PAGE_UP ); - KEY( VK_NEXT, GLUT_KEY_PAGE_DOWN ); - KEY( VK_HOME, GLUT_KEY_HOME ); - KEY( VK_END, GLUT_KEY_END ); - KEY( VK_LEFT, GLUT_KEY_LEFT ); - KEY( VK_UP, GLUT_KEY_UP ); - KEY( VK_RIGHT, GLUT_KEY_RIGHT ); - KEY( VK_DOWN, GLUT_KEY_DOWN ); - KEY( VK_INSERT, GLUT_KEY_INSERT ); - KEY( VK_LCONTROL, GLUT_KEY_CTRL_L ); - KEY( VK_RCONTROL, GLUT_KEY_CTRL_R ); - KEY( VK_LSHIFT, GLUT_KEY_SHIFT_L ); - KEY( VK_RSHIFT, GLUT_KEY_SHIFT_R ); - KEY( VK_LMENU, GLUT_KEY_ALT_L ); - KEY( VK_RMENU, GLUT_KEY_ALT_R ); - - case VK_DELETE: - /* The delete key should be treated as an ASCII keypress: */ - INVOKE_WCB( *window, Keyboard, - ( 127, window->State.MouseX, window->State.MouseY ) - ); - } - -#if defined(_WIN32_WCE) - if(!(lParam & 0x40000000)) /* Prevent auto-repeat */ - { - if(wParam==(unsigned)gxKeyList.vkRight) - keypress = GLUT_KEY_RIGHT; - else if(wParam==(unsigned)gxKeyList.vkLeft) - keypress = GLUT_KEY_LEFT; - else if(wParam==(unsigned)gxKeyList.vkUp) - keypress = GLUT_KEY_UP; - else if(wParam==(unsigned)gxKeyList.vkDown) - keypress = GLUT_KEY_DOWN; - else if(wParam==(unsigned)gxKeyList.vkA) - keypress = GLUT_KEY_F1; - else if(wParam==(unsigned)gxKeyList.vkB) - keypress = GLUT_KEY_F2; - else if(wParam==(unsigned)gxKeyList.vkC) - keypress = GLUT_KEY_F3; - else if(wParam==(unsigned)gxKeyList.vkStart) - keypress = GLUT_KEY_F4; - } -#endif - - if( keypress != -1 ) - INVOKE_WCB( *window, Special, - ( keypress, - window->State.MouseX, window->State.MouseY ) - ); - - fgState.Modifiers = INVALID_MODIFIERS; - } - break; - - case WM_SYSKEYUP: - case WM_KEYUP: - { - int keypress = -1; - POINT mouse_pos; - - /* - * Remember the current modifiers state. This is done here in order - * to make sure the VK_DELETE keyboard callback is executed properly. - */ - fgState.Modifiers = fghGetWin32Modifiers( ); - - GetCursorPos( &mouse_pos ); - ScreenToClient( window->Window.Handle, &mouse_pos ); - - window->State.MouseX = mouse_pos.x; - window->State.MouseY = mouse_pos.y; - - /* - * Convert the Win32 keystroke codes to GLUTtish way. - * "KEY(a,b)" was defined under "WM_KEYDOWN" - */ - - switch( wParam ) - { - KEY( VK_F1, GLUT_KEY_F1 ); - KEY( VK_F2, GLUT_KEY_F2 ); - KEY( VK_F3, GLUT_KEY_F3 ); - KEY( VK_F4, GLUT_KEY_F4 ); - KEY( VK_F5, GLUT_KEY_F5 ); - KEY( VK_F6, GLUT_KEY_F6 ); - KEY( VK_F7, GLUT_KEY_F7 ); - KEY( VK_F8, GLUT_KEY_F8 ); - KEY( VK_F9, GLUT_KEY_F9 ); - KEY( VK_F10, GLUT_KEY_F10 ); - KEY( VK_F11, GLUT_KEY_F11 ); - KEY( VK_F12, GLUT_KEY_F12 ); - KEY( VK_PRIOR, GLUT_KEY_PAGE_UP ); - KEY( VK_NEXT, GLUT_KEY_PAGE_DOWN ); - KEY( VK_HOME, GLUT_KEY_HOME ); - KEY( VK_END, GLUT_KEY_END ); - KEY( VK_LEFT, GLUT_KEY_LEFT ); - KEY( VK_UP, GLUT_KEY_UP ); - KEY( VK_RIGHT, GLUT_KEY_RIGHT ); - KEY( VK_DOWN, GLUT_KEY_DOWN ); - KEY( VK_INSERT, GLUT_KEY_INSERT ); - KEY( VK_LCONTROL, GLUT_KEY_CTRL_L ); - KEY( VK_RCONTROL, GLUT_KEY_CTRL_R ); - KEY( VK_LSHIFT, GLUT_KEY_SHIFT_L ); - KEY( VK_RSHIFT, GLUT_KEY_SHIFT_R ); - KEY( VK_LMENU, GLUT_KEY_ALT_L ); - KEY( VK_RMENU, GLUT_KEY_ALT_R ); - - case VK_DELETE: - /* The delete key should be treated as an ASCII keypress: */ - INVOKE_WCB( *window, KeyboardUp, - ( 127, window->State.MouseX, window->State.MouseY ) - ); - break; - - default: - { -#if !defined(_WIN32_WCE) - BYTE state[ 256 ]; - WORD code[ 2 ]; - - GetKeyboardState( state ); - - if( ToAscii( (UINT)wParam, 0, state, code, 0 ) == 1 ) - wParam=code[ 0 ]; - - INVOKE_WCB( *window, KeyboardUp, - ( (char)wParam, - window->State.MouseX, window->State.MouseY ) - ); -#endif /* !defined(_WIN32_WCE) */ - } - } - - if( keypress != -1 ) - INVOKE_WCB( *window, SpecialUp, - ( keypress, - window->State.MouseX, window->State.MouseY ) - ); - - fgState.Modifiers = INVALID_MODIFIERS; - } - break; - - case WM_SYSCHAR: - case WM_CHAR: - { - if( (fgState.KeyRepeat==GLUT_KEY_REPEAT_OFF || window->State.IgnoreKeyRepeat==GL_TRUE) && (HIWORD(lParam) & KF_REPEAT) ) - break; - - fgState.Modifiers = fghGetWin32Modifiers( ); - INVOKE_WCB( *window, Keyboard, - ( (char)wParam, - window->State.MouseX, window->State.MouseY ) - ); - fgState.Modifiers = INVALID_MODIFIERS; - } - break; - - case WM_CAPTURECHANGED: - /* User has finished resizing the window, force a redraw */ - INVOKE_WCB( *window, Display, ( ) ); - - /*lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); */ - break; - - /* Other messages that I have seen and which are not handled already */ - case WM_SETTEXT: /* 0x000c */ - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - /* Pass it on to "DefWindowProc" to set the window text */ - break; - - case WM_GETTEXT: /* 0x000d */ - /* Ideally we would copy the title of the window into "lParam" */ - /* strncpy ( (char *)lParam, "Window Title", wParam ); - lRet = ( wParam > 12 ) ? 12 : wParam; */ - /* the number of characters copied */ - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - break; - - case WM_GETTEXTLENGTH: /* 0x000e */ - /* Ideally we would get the length of the title of the window */ - lRet = 12; - /* the number of characters in "Window Title\0" (see above) */ - break; - - case WM_ERASEBKGND: /* 0x0014 */ - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - break; - -#if !defined(_WIN32_WCE) - case WM_SYNCPAINT: /* 0x0088 */ - /* Another window has moved, need to update this one */ - window->State.Redisplay = GL_TRUE; - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - /* Help screen says this message must be passed to "DefWindowProc" */ - break; - - case WM_NCPAINT: /* 0x0085 */ - /* Need to update the border of this window */ - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - /* Pass it on to "DefWindowProc" to repaint a standard border */ - break; - - case WM_SYSCOMMAND : /* 0x0112 */ - { - /* - * We have received a system command message. Try to act on it. - * The commands are passed in through the "wParam" parameter: - * The least significant digit seems to be which edge of the window - * is being used for a resize event: - * 4 3 5 - * 1 2 - * 7 6 8 - * Congratulations and thanks to Richard Rauch for figuring this out.. - */ - switch ( wParam & 0xfff0 ) - { - case SC_SIZE : - break ; - - case SC_MOVE : - break ; - - case SC_MINIMIZE : - /* User has clicked on the "-" to minimize the window */ - /* Turn off the visibility */ - window->State.Visible = GL_FALSE ; - - break ; - - case SC_MAXIMIZE : - break ; - - case SC_NEXTWINDOW : - break ; - - case SC_PREVWINDOW : - break ; - - case SC_CLOSE : - /* Followed very closely by a WM_CLOSE message */ - break ; - - case SC_VSCROLL : - break ; - - case SC_HSCROLL : - break ; - - case SC_MOUSEMENU : - break ; - - case SC_KEYMENU : - break ; - - case SC_ARRANGE : - break ; - - case SC_RESTORE : - break ; - - case SC_TASKLIST : - break ; - - case SC_SCREENSAVE : - break ; - - case SC_HOTKEY : - break ; - -#if(WINVER >= 0x0400) - case SC_DEFAULT : - break ; - - case SC_MONITORPOWER : - break ; - - case SC_CONTEXTHELP : - break ; -#endif /* WINVER >= 0x0400 */ - - default: -#if _DEBUG - fgWarning( "Unknown wParam type 0x%x", wParam ); -#endif - break; - } - } -#endif /* !defined(_WIN32_WCE) */ - - /* We need to pass the message on to the operating system as well */ - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - break; - -#ifdef WM_TOUCH - /* handle multi-touch messages */ - case WM_TOUCH: - { - unsigned int numInputs = (unsigned int)wParam; - unsigned int i = 0; - TOUCHINPUT* ti = (TOUCHINPUT*)malloc( sizeof(TOUCHINPUT)*numInputs); - - if (fghGetTouchInputInfo == (pGetTouchInputInfo)0xDEADBEEF) { - fghGetTouchInputInfo = (pGetTouchInputInfo)GetProcAddress(GetModuleHandle("user32"),"GetTouchInputInfo"); - fghCloseTouchInputHandle = (pCloseTouchInputHandle)GetProcAddress(GetModuleHandle("user32"),"CloseTouchInputHandle"); - } - - if (!fghGetTouchInputInfo) { - free( (void*)ti ); - break; - } - - if (fghGetTouchInputInfo( (HTOUCHINPUT)lParam, numInputs, ti, sizeof(TOUCHINPUT) )) { - /* Handle each contact point */ - for (i = 0; i < numInputs; ++i ) { - - POINT tp; - tp.x = TOUCH_COORD_TO_PIXEL(ti[i].x); - tp.y = TOUCH_COORD_TO_PIXEL(ti[i].y); - ScreenToClient( hWnd, &tp ); - - ti[i].dwID = ti[i].dwID * 2; - - if (ti[i].dwFlags & TOUCHEVENTF_DOWN) { - INVOKE_WCB( *window, MultiEntry, ( ti[i].dwID, GLUT_ENTERED ) ); - INVOKE_WCB( *window, MultiButton, ( ti[i].dwID, tp.x, tp.y, 0, GLUT_DOWN ) ); - } else if (ti[i].dwFlags & TOUCHEVENTF_MOVE) { - INVOKE_WCB( *window, MultiMotion, ( ti[i].dwID, tp.x, tp.y ) ); - } else if (ti[i].dwFlags & TOUCHEVENTF_UP) { - INVOKE_WCB( *window, MultiButton, ( ti[i].dwID, tp.x, tp.y, 0, GLUT_UP ) ); - INVOKE_WCB( *window, MultiEntry, ( ti[i].dwID, GLUT_LEFT ) ); - } - } - } - fghCloseTouchInputHandle((HTOUCHINPUT)lParam); - free( (void*)ti ); - lRet = 0; /*DefWindowProc( hWnd, uMsg, wParam, lParam );*/ - break; - } -#endif - default: - /* Handle unhandled messages */ - lRet = DefWindowProc( hWnd, uMsg, wParam, lParam ); - break; - } - - return lRet; -} -#endif - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_menu.c b/examples/common/opengl-framework/freeglut/freeglut_menu.c deleted file mode 100644 index ea5837e8..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_menu.c +++ /dev/null @@ -1,1002 +0,0 @@ -/* - * freeglut_menu.c - * - * Pull-down menu creation and handling. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 16 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#define FREEGLUT_BUILDING_LIB -#include -#include "freeglut_internal.h" - -/* -- DEFINITIONS ---------------------------------------------------------- */ - -/* - * FREEGLUT_MENU_FONT can be any freeglut bitmapped font. - * (Stroked fonts would not be out of the question, but we'd need to alter - * code, since GLUT (hence freeglut) does not quite unify stroked and - * bitmapped font handling.) - * Old UNIX/X11 GLUT (BSD, UNIX, IRIX, LINUX, HPUX, ...) used a system - * font best approximated by an 18-pixel HELVETICA, I think. MS-WINDOWS - * GLUT used something closest to the 8x13 fixed-width font. (Old - * GLUT apparently uses host-system menus rather than building its own. - * freeglut is building its own menus from scratch.) - * - * FREEGLUT_MENU_HEIGHT gives the height of ONE menu box. This should be - * the distances between two adjacent menu entries. It should scale - * automatically with the font choice, so you needn't alter it---unless you - * use a stroked font. - * - * FREEGLUT_MENU_BORDER says how many pixels to allow around the edge of a - * menu. (It also seems to be the same as the number of pixels used as - * a border around *items* to separate them from neighbors. John says - * that that wasn't the original intent...if not, perhaps we need another - * symbolic constant, FREEGLUT_MENU_ITEM_BORDER, or such.) - */ -#if TARGET_HOST_MS_WINDOWS -#define FREEGLUT_MENU_FONT GLUT_BITMAP_8_BY_13 -#else -#define FREEGLUT_MENU_FONT GLUT_BITMAP_HELVETICA_18 -#endif - -#define FREEGLUT_MENU_HEIGHT (glutBitmapHeight(FREEGLUT_MENU_FONT) + \ - FREEGLUT_MENU_BORDER) -#define FREEGLUT_MENU_BORDER 2 - - -/* - * These variables are for rendering the freeglut menu items. - * - * The choices are fore- and background, with and without h for Highlighting. - * Old GLUT appeared to be system-dependant for its colors (sigh) so we are - * too. These variables should be stuffed into global state and initialized - * via the glutInit*() system. - */ -#if TARGET_HOST_MS_WINDOWS -static float menu_pen_fore [4] = {0.0f, 0.0f, 0.0f, 1.0f}; -static float menu_pen_back [4] = {0.85f, 0.85f, 0.85f, 1.0f}; -static float menu_pen_hfore [4] = {1.0f, 1.0f, 1.0f, 1.0f}; -static float menu_pen_hback [4] = {0.15f, 0.15f, 0.45f, 1.0f}; -#else -static float menu_pen_fore [4] = {0.0f, 0.0f, 0.0f, 1.0f}; -static float menu_pen_back [4] = {0.70f, 0.70f, 0.70f, 1.0f}; -static float menu_pen_hfore [4] = {0.0f, 0.0f, 0.0f, 1.0f}; -static float menu_pen_hback [4] = {1.0f, 1.0f, 1.0f, 1.0f}; -#endif - -/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ - -/* - * Private function to find a menu entry by index - */ -static SFG_MenuEntry *fghFindMenuEntry( SFG_Menu* menu, int index ) -{ - SFG_MenuEntry *entry; - int i = 1; - - for( entry = (SFG_MenuEntry *)menu->Entries.First; - entry; - entry = (SFG_MenuEntry *)entry->Node.Next ) - { - if( i == index ) - break; - ++i; - } - - return entry; -} - -/* - * Deactivates a menu pointed by the function argument. - */ -static void fghDeactivateSubMenu( SFG_MenuEntry *menuEntry ) -{ - SFG_MenuEntry *subMenuIter; - /* Hide the present menu's window */ - fgSetWindow( menuEntry->SubMenu->Window ); - glutHideWindow( ); - - /* Forget about having that menu active anymore, now: */ - menuEntry->SubMenu->Window->ActiveMenu = NULL; - menuEntry->SubMenu->IsActive = GL_FALSE; - menuEntry->SubMenu->ActiveEntry = NULL; - - /* Hide all submenu windows, and the root menu's window. */ - for ( subMenuIter = (SFG_MenuEntry *)menuEntry->SubMenu->Entries.First; - subMenuIter; - subMenuIter = (SFG_MenuEntry *)subMenuIter->Node.Next ) - { - subMenuIter->IsActive = GL_FALSE; - - /* Is that an active submenu by any case? */ - if( subMenuIter->SubMenu ) - fghDeactivateSubMenu( subMenuIter ); - } - - fgSetWindow ( menuEntry->SubMenu->ParentWindow ) ; -} - -/* - * Private function to get the virtual maximum screen extent - */ -static GLvoid fghGetVMaxExtent( SFG_Window* window, int* x, int* y ) -{ - if( fgStructure.GameModeWindow ) - { -#if TARGET_HOST_POSIX_X11 - int wx, wy; - Window w; - - XTranslateCoordinates( - fgDisplay.Display, - window->Window.Handle, - fgDisplay.RootWindow, - 0, 0, &wx, &wy, &w); - - *x = fgState.GameModeSize.X + wx; - *y = fgState.GameModeSize.Y + wy; -#else - *x = glutGet ( GLUT_SCREEN_WIDTH ); - *y = glutGet ( GLUT_SCREEN_HEIGHT ); -#endif - } - else - { - *x = fgDisplay.ScreenWidth; - *y = fgDisplay.ScreenHeight; - } -} - -/* - * Private function to check for the current menu/sub menu activity state - */ -static GLboolean fghCheckMenuStatus( SFG_Menu* menu ) -{ - SFG_MenuEntry* menuEntry; - int x, y; - - /* First of all check any of the active sub menus... */ - for( menuEntry = (SFG_MenuEntry *)menu->Entries.First; - menuEntry; - menuEntry = (SFG_MenuEntry *)menuEntry->Node.Next ) - { - if( menuEntry->SubMenu && menuEntry->IsActive ) - { - /* - * OK, have the sub-menu checked, too. If it returns GL_TRUE, it - * will mean that it caught the mouse cursor and we do not need - * to regenerate the activity list, and so our parents do... - */ - GLboolean return_status; - - menuEntry->SubMenu->Window->State.MouseX = - menu->Window->State.MouseX + menu->X - menuEntry->SubMenu->X; - menuEntry->SubMenu->Window->State.MouseY = - menu->Window->State.MouseY + menu->Y - menuEntry->SubMenu->Y; - return_status = fghCheckMenuStatus( menuEntry->SubMenu ); - - if ( return_status ) - return GL_TRUE; - } - } - - /* That much about our sub menus, let's get to checking the current menu: */ - x = menu->Window->State.MouseX; - y = menu->Window->State.MouseY; - - /* Check if the mouse cursor is contained within the current menu box */ - if( ( x >= FREEGLUT_MENU_BORDER ) && - ( x < menu->Width - FREEGLUT_MENU_BORDER ) && - ( y >= FREEGLUT_MENU_BORDER ) && - ( y < menu->Height - FREEGLUT_MENU_BORDER ) ) - { - int menuID = ( y - FREEGLUT_MENU_BORDER ) / FREEGLUT_MENU_HEIGHT; - - /* The mouse cursor is somewhere over our box, check it out. */ - menuEntry = fghFindMenuEntry( menu, menuID + 1 ); - FREEGLUT_INTERNAL_ERROR_EXIT( menuEntry, "Cannot find menu entry", - "fghCheckMenuStatus" ); - - menuEntry->IsActive = GL_TRUE; - menuEntry->Ordinal = menuID; - - /* - * If this is not the same as the last active menu entry, deactivate - * the previous entry. Specifically, if the previous active entry - * was a submenu then deactivate it. - */ - if( menu->ActiveEntry && ( menuEntry != menu->ActiveEntry ) ) - if( menu->ActiveEntry->SubMenu ) - fghDeactivateSubMenu( menu->ActiveEntry ); - - if( menuEntry != menu->ActiveEntry ) - { - menu->Window->State.Redisplay = GL_TRUE; - if( menu->ActiveEntry ) - menu->ActiveEntry->IsActive = GL_FALSE; - } - - menu->ActiveEntry = menuEntry; - menu->IsActive = GL_TRUE; /* XXX Do we need this? */ - - /* - * OKi, we have marked that entry as active, but it would be also - * nice to have its contents updated, in case it's a sub menu. - * Also, ignore the return value of the check function: - */ - if( menuEntry->SubMenu ) - { - if ( ! menuEntry->SubMenu->IsActive ) - { - int max_x, max_y; - SFG_Window *current_window = fgStructure.CurrentWindow; - - /* Set up the initial menu position now... */ - menuEntry->SubMenu->IsActive = GL_TRUE; - - /* Set up the initial submenu position now: */ - fghGetVMaxExtent(menu->ParentWindow, &max_x, &max_y); - menuEntry->SubMenu->X = menu->X + menu->Width; - menuEntry->SubMenu->Y = menu->Y + - menuEntry->Ordinal * FREEGLUT_MENU_HEIGHT; - - if( menuEntry->SubMenu->X + menuEntry->SubMenu->Width > max_x ) - menuEntry->SubMenu->X = menu->X - menuEntry->SubMenu->Width; - - if( menuEntry->SubMenu->Y + menuEntry->SubMenu->Height > max_y ) - { - menuEntry->SubMenu->Y -= ( menuEntry->SubMenu->Height - - FREEGLUT_MENU_HEIGHT - - 2 * FREEGLUT_MENU_BORDER ); - if( menuEntry->SubMenu->Y < 0 ) - menuEntry->SubMenu->Y = 0; - } - - fgSetWindow( menuEntry->SubMenu->Window ); - glutPositionWindow( menuEntry->SubMenu->X, - menuEntry->SubMenu->Y ); - glutReshapeWindow( menuEntry->SubMenu->Width, - menuEntry->SubMenu->Height ); - glutPopWindow( ); - glutShowWindow( ); - menuEntry->SubMenu->Window->ActiveMenu = menuEntry->SubMenu; - fgSetWindow( current_window ); - menuEntry->SubMenu->Window->State.MouseX = - x + menu->X - menuEntry->SubMenu->X; - menuEntry->SubMenu->Window->State.MouseY = - y + menu->Y - menuEntry->SubMenu->Y; - fghCheckMenuStatus( menuEntry->SubMenu ); - } - - /* Activate it because its parent entry is active */ - menuEntry->SubMenu->IsActive = GL_TRUE; /* XXX Do we need this? */ - } - - /* Report back that we have caught the menu cursor */ - return GL_TRUE; - } - - /* Looks like the menu cursor is somewhere else... */ - if( menu->ActiveEntry && menu->ActiveEntry->IsActive && - ( !menu->ActiveEntry->SubMenu || - !menu->ActiveEntry->SubMenu->IsActive ) ) - { - menu->Window->State.Redisplay = GL_TRUE; - menu->ActiveEntry->IsActive = GL_FALSE; - menu->ActiveEntry = NULL; - } - - return GL_FALSE; -} - -/* - * Displays a menu box and all of its submenus (if they are active) - */ -static void fghDisplayMenuBox( SFG_Menu* menu ) -{ - SFG_MenuEntry *menuEntry; - int i; - int border = FREEGLUT_MENU_BORDER; - - /* - * Have the menu box drawn first. The +- values are - * here just to make it more nice-looking... - */ - /* a non-black dark version of the below. */ - glColor4f( 1.0f, 1.0f, 1.0f, 1.0f ); - glBegin( GL_QUAD_STRIP ); - glVertex2i( menu->Width , 0 ); - glVertex2i( menu->Width - border, border); - glVertex2i( 0 , 0 ); - glVertex2i( border, border); - glVertex2i( 0 , menu->Height ); - glVertex2i( border, menu->Height - border); - glEnd( ); - - /* a non-black dark version of the below. */ - glColor4f( 0.5f, 0.5f, 0.5f, 1.0f ); - glBegin( GL_QUAD_STRIP ); - glVertex2i( 0 , menu->Height ); - glVertex2i( border, menu->Height - border); - glVertex2i( menu->Width , menu->Height ); - glVertex2i( menu->Width - border, menu->Height - border); - glVertex2i( menu->Width , 0 ); - glVertex2i( menu->Width - border, border); - glEnd( ); - - glColor4fv( menu_pen_back ); - glBegin( GL_QUADS ); - glVertex2i( border, border); - glVertex2i( menu->Width - border, border); - glVertex2i( menu->Width - border, menu->Height - border); - glVertex2i( border, menu->Height - border); - glEnd( ); - - /* Check if any of the submenus is currently active... */ - for( menuEntry = (SFG_MenuEntry *)menu->Entries.First; - menuEntry; - menuEntry = (SFG_MenuEntry *)menuEntry->Node.Next ) - { - /* Has the menu been marked as active, maybe? */ - if( menuEntry->IsActive ) - { - /* - * That's truly right, and we need to have it highlighted. - * There is an assumption that mouse cursor didn't move - * since the last check of menu activity state: - */ - int menuID = menuEntry->Ordinal; - - /* So have the highlight drawn... */ - glColor4fv( menu_pen_hback ); - glBegin( GL_QUADS ); - glVertex2i( border, - (menuID + 0)*FREEGLUT_MENU_HEIGHT + border ); - glVertex2i( menu->Width - border, - (menuID + 0)*FREEGLUT_MENU_HEIGHT + border ); - glVertex2i( menu->Width - border, - (menuID + 1)*FREEGLUT_MENU_HEIGHT + border ); - glVertex2i( border, - (menuID + 1)*FREEGLUT_MENU_HEIGHT + border ); - glEnd( ); - } - } - - /* Print the menu entries now... */ - - glColor4fv( menu_pen_fore ); - - for( menuEntry = (SFG_MenuEntry *)menu->Entries.First, i = 0; - menuEntry; - menuEntry = (SFG_MenuEntry *)menuEntry->Node.Next, ++i ) - { - /* If the menu entry is active, set the color to white */ - if( menuEntry->IsActive ) - glColor4fv( menu_pen_hfore ); - - /* Move the raster into position... */ - /* Try to center the text - JCJ 31 July 2003*/ - glRasterPos2i( - 2 * border, - ( i + 1 )*FREEGLUT_MENU_HEIGHT - - ( int )( FREEGLUT_MENU_HEIGHT*0.3 - border ) - ); - - /* Have the label drawn, character after character: */ - glutBitmapString( FREEGLUT_MENU_FONT, - (unsigned char *)menuEntry->Text); - - /* If it's a submenu, draw a right arrow */ - if( menuEntry->SubMenu ) - { - int width = glutBitmapWidth( FREEGLUT_MENU_FONT, '_' ); - int x_base = menu->Width - 2 - width; - int y_base = i*FREEGLUT_MENU_HEIGHT + border; - glBegin( GL_TRIANGLES ); - glVertex2i( x_base, y_base + 2*border); - glVertex2i( menu->Width - 2, y_base + - ( FREEGLUT_MENU_HEIGHT + border) / 2 ); - glVertex2i( x_base, y_base + FREEGLUT_MENU_HEIGHT - border ); - glEnd( ); - } - - /* If the menu entry is active, reset the color */ - if( menuEntry->IsActive ) - glColor4fv( menu_pen_fore ); - } -} - -/* - * Private static function to set the parent window of a submenu and all - * of its submenus - */ -static void fghSetMenuParentWindow( SFG_Window *window, SFG_Menu *menu ) -{ - SFG_MenuEntry *menuEntry; - - menu->ParentWindow = window; - - for( menuEntry = ( SFG_MenuEntry * )menu->Entries.First; - menuEntry; - menuEntry = ( SFG_MenuEntry * )menuEntry->Node.Next ) - if( menuEntry->SubMenu ) - fghSetMenuParentWindow( window, menuEntry->SubMenu ); -} - -/* - * Function to check for menu entry selection on menu deactivation - */ -static void fghExecuteMenuCallback( SFG_Menu* menu ) -{ - SFG_MenuEntry *menuEntry; - - /* First of all check any of the active sub menus... */ - for( menuEntry = (SFG_MenuEntry *)menu->Entries.First; - menuEntry; - menuEntry = (SFG_MenuEntry *)menuEntry->Node.Next) - { - if( menuEntry->IsActive ) - { - if( menuEntry->SubMenu ) - fghExecuteMenuCallback( menuEntry->SubMenu ); - else - if( menu->Callback ) - { - SFG_Menu *save_menu = fgStructure.CurrentMenu; - fgStructure.CurrentMenu = menu; - menu->Callback( menuEntry->ID ); - fgStructure.CurrentMenu = save_menu; - } - - return; - } - } -} - - -/* - * Displays the currently active menu for the current window - */ -void fgDisplayMenu( void ) -{ - SFG_Window* window = fgStructure.CurrentWindow; - SFG_Menu* menu = NULL; - - FREEGLUT_INTERNAL_ERROR_EXIT ( fgStructure.CurrentWindow, "Displaying menu in nonexistent window", - "fgDisplayMenu" ); - - /* Check if there is an active menu attached to this window... */ - menu = window->ActiveMenu; - freeglut_return_if_fail( menu ); - - fgSetWindow( menu->Window ); - - glPushAttrib( GL_DEPTH_BUFFER_BIT | GL_TEXTURE_BIT | GL_LIGHTING_BIT | - GL_POLYGON_BIT ); - - glDisable( GL_DEPTH_TEST ); - glDisable( GL_TEXTURE_2D ); - glDisable( GL_LIGHTING ); - glDisable( GL_CULL_FACE ); - - glMatrixMode( GL_PROJECTION ); - glPushMatrix( ); - glLoadIdentity( ); - glOrtho( - 0, glutGet( GLUT_WINDOW_WIDTH ), - glutGet( GLUT_WINDOW_HEIGHT ), 0, - -1, 1 - ); - - glMatrixMode( GL_MODELVIEW ); - glPushMatrix( ); - glLoadIdentity( ); - - fghDisplayMenuBox( menu ); - - glPopAttrib( ); - - glMatrixMode( GL_PROJECTION ); - glPopMatrix( ); - glMatrixMode( GL_MODELVIEW ); - glPopMatrix( ); - - glutSwapBuffers( ); - - fgSetWindow ( window ); -} - -/* - * Activates a menu pointed by the function argument - */ -static void fghActivateMenu( SFG_Window* window, int button ) -{ - int max_x, max_y; - - /* We'll be referencing this menu a lot, so remember its address: */ - SFG_Menu* menu = window->Menu[ button ]; - SFG_Window* current_window = fgStructure.CurrentWindow; - - /* If the menu is already active in another window, deactivate it there */ - if ( menu->ParentWindow ) - menu->ParentWindow->ActiveMenu = NULL ; - - /* Mark the menu as active, so that it gets displayed: */ - window->ActiveMenu = menu; - menu->IsActive = GL_TRUE; - fghSetMenuParentWindow ( window, menu ); - fgState.ActiveMenus++; - - /* Set up the initial menu position now: */ - fghGetVMaxExtent(menu->ParentWindow, &max_x, &max_y); - fgSetWindow( window ); - menu->X = window->State.MouseX + glutGet( GLUT_WINDOW_X ); - menu->Y = window->State.MouseY + glutGet( GLUT_WINDOW_Y ); - - if( menu->X + menu->Width > max_x ) - menu->X -=menu->Width; - - if( menu->Y + menu->Height > max_y ) - { - menu->Y -=menu->Height; - if( menu->Y < 0 ) - menu->Y = 0; - } - - menu->Window->State.MouseX = - window->State.MouseX + glutGet( GLUT_WINDOW_X ) - menu->X; - menu->Window->State.MouseY = - window->State.MouseY + glutGet( GLUT_WINDOW_Y ) - menu->Y; - - fgSetWindow( menu->Window ); - glutPositionWindow( menu->X, menu->Y ); - glutReshapeWindow( menu->Width, menu->Height ); - glutPopWindow( ); - glutShowWindow( ); - menu->Window->ActiveMenu = menu; - fghCheckMenuStatus( menu ); - fgSetWindow( current_window ); -} - -/* - * Update Highlight states of the menu - * - * Current mouse position is in menu->Window->State.MouseX/Y. - */ -void fgUpdateMenuHighlight ( SFG_Menu *menu ) -{ - fghCheckMenuStatus( menu ); -} - -/* - * Check whether an active menu absorbs a mouse click - */ -GLboolean fgCheckActiveMenu ( SFG_Window *window, int button, GLboolean pressed, - int mouse_x, int mouse_y ) -{ - /* - * Near as I can tell, this is the menu behaviour: - * - Down-click the menu button, menu not active: activate - * the menu with its upper left-hand corner at the mouse - * location. - * - Down-click any button outside the menu, menu active: - * deactivate the menu - * - Down-click any button inside the menu, menu active: - * select the menu entry and deactivate the menu - * - Up-click the menu button, menu not active: nothing happens - * - Up-click the menu button outside the menu, menu active: - * nothing happens - * - Up-click the menu button inside the menu, menu active: - * select the menu entry and deactivate the menu - * Since menus can have submenus, we need to check this recursively. - */ - if( window->ActiveMenu ) - { - if( window == window->ActiveMenu->ParentWindow ) - { - window->ActiveMenu->Window->State.MouseX = - mouse_x - window->ActiveMenu->X; - window->ActiveMenu->Window->State.MouseY = - mouse_y - window->ActiveMenu->Y; - } - - /* In the menu, invoke the callback and deactivate the menu */ - if( fghCheckMenuStatus( window->ActiveMenu ) ) - { - /* - * Save the current window and menu and set the current - * window to the window whose menu this is - */ - SFG_Window *save_window = fgStructure.CurrentWindow; - SFG_Menu *save_menu = fgStructure.CurrentMenu; - SFG_Window *parent_window = window->ActiveMenu->ParentWindow; - fgSetWindow( parent_window ); - fgStructure.CurrentMenu = window->ActiveMenu; - - /* Execute the menu callback */ - fghExecuteMenuCallback( window->ActiveMenu ); - fgDeactivateMenu( parent_window ); - - /* Restore the current window and menu */ - fgSetWindow( save_window ); - fgStructure.CurrentMenu = save_menu; - } - else if( pressed ) - /* - * Outside the menu, deactivate if it's a downclick - * - * XXX This isn't enough. A downclick outside of - * XXX the interior of our freeglut windows should also - * XXX deactivate the menu. This is more complicated. - */ - fgDeactivateMenu( window->ActiveMenu->ParentWindow ); - - /* - * XXX Why does an active menu require a redisplay at - * XXX this point? If this can come out cleanly, then - * XXX it probably should do so; if not, a comment should - * XXX explain it. - */ - if( ! window->IsMenu ) - window->State.Redisplay = GL_TRUE; - - return GL_TRUE; - } - - /* No active menu, let's check whether we need to activate one. */ - if( ( 0 <= button ) && - ( FREEGLUT_MAX_MENUS > button ) && - ( window->Menu[ button ] ) && - pressed ) - { - /* XXX Posting a requisite Redisplay seems bogus. */ - window->State.Redisplay = GL_TRUE; - fghActivateMenu( window, button ); - return GL_TRUE; - } - - return GL_FALSE; -} - -/* - * Deactivates a menu pointed by the function argument. - */ -void fgDeactivateMenu( SFG_Window *window ) -{ - SFG_Window *parent_window = NULL; - - /* Check if there is an active menu attached to this window... */ - SFG_Menu* menu = window->ActiveMenu; - SFG_MenuEntry *menuEntry; - - /* Did we find an active window? */ - freeglut_return_if_fail( menu ); - - parent_window = menu->ParentWindow; - - /* Hide the present menu's window */ - fgSetWindow( menu->Window ); - glutHideWindow( ); - - /* Forget about having that menu active anymore, now: */ - menu->Window->ActiveMenu = NULL; - menu->ParentWindow->ActiveMenu = NULL; - fghSetMenuParentWindow ( NULL, menu ); - menu->IsActive = GL_FALSE; - menu->ActiveEntry = NULL; - - fgState.ActiveMenus--; - - /* Hide all submenu windows, and the root menu's window. */ - for ( menuEntry = ( SFG_MenuEntry * )menu->Entries.First; - menuEntry; - menuEntry = ( SFG_MenuEntry * )menuEntry->Node.Next ) - { - menuEntry->IsActive = GL_FALSE; - - /* Is that an active submenu by any case? */ - if( menuEntry->SubMenu ) - fghDeactivateSubMenu( menuEntry ); - } - - fgSetWindow ( parent_window ) ; -} - -/* - * Recalculates current menu's box size - */ -void fghCalculateMenuBoxSize( void ) -{ - SFG_MenuEntry* menuEntry; - int width = 0, height = 0; - - /* Make sure there is a current menu set */ - freeglut_return_if_fail( fgStructure.CurrentMenu ); - - /* The menu's box size depends on the menu entries: */ - for( menuEntry = ( SFG_MenuEntry * )fgStructure.CurrentMenu->Entries.First; - menuEntry; - menuEntry = ( SFG_MenuEntry * )menuEntry->Node.Next ) - { - /* Update the menu entry's width value */ - menuEntry->Width = glutBitmapLength( - FREEGLUT_MENU_FONT, - (unsigned char *)menuEntry->Text - ); - - /* - * If the entry is a submenu, then it needs to be wider to - * accomodate the arrow. JCJ 31 July 2003 - */ - if (menuEntry->SubMenu ) - menuEntry->Width += glutBitmapLength( - FREEGLUT_MENU_FONT, - (unsigned char *)"_" - ); - - /* Check if it's the biggest we've found */ - if( menuEntry->Width > width ) - width = menuEntry->Width; - - height += FREEGLUT_MENU_HEIGHT; - } - - /* Store the menu's box size now: */ - fgStructure.CurrentMenu->Height = height + 2 * FREEGLUT_MENU_BORDER; - fgStructure.CurrentMenu->Width = width + 4 * FREEGLUT_MENU_BORDER; -} - - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * Creates a new menu object, adding it to the freeglut structure - */ -int FGAPIENTRY glutCreateMenu( void(* callback)( int ) ) -{ - /* The menu object creation code resides in freeglut_structure.c */ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutCreateMenu" ); - return fgCreateMenu( callback )->ID; -} - -#if TARGET_HOST_MS_WINDOWS -int FGAPIENTRY __glutCreateMenuWithExit( void(* callback)( int ), void (__cdecl *exit_function)(int) ) -{ - __glutExitFunc = exit_function; - return glutCreateMenu( callback ); -} -#endif - -/* - * Destroys a menu object, removing all references to it - */ -void FGAPIENTRY glutDestroyMenu( int menuID ) -{ - SFG_Menu* menu; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDestroyMenu" ); - menu = fgMenuByID( menuID ); - - freeglut_return_if_fail( menu ); - - /* The menu object destruction code resides in freeglut_structure.c */ - fgDestroyMenu( menu ); -} - -/* - * Returns the ID number of the currently active menu - */ -int FGAPIENTRY glutGetMenu( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetMenu" ); - - if( fgStructure.CurrentMenu ) - return fgStructure.CurrentMenu->ID; - - return 0; -} - -/* - * Sets the current menu given its menu ID - */ -void FGAPIENTRY glutSetMenu( int menuID ) -{ - SFG_Menu* menu; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetMenu" ); - menu = fgMenuByID( menuID ); - - freeglut_return_if_fail( menu ); - - fgStructure.CurrentMenu = menu; -} - -/* - * Adds a menu entry to the bottom of the current menu - */ -void FGAPIENTRY glutAddMenuEntry( const char* label, int value ) -{ - SFG_MenuEntry* menuEntry; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutAddMenuEntry" ); - menuEntry = (SFG_MenuEntry *)calloc( sizeof(SFG_MenuEntry), 1 ); - freeglut_return_if_fail( fgStructure.CurrentMenu ); - - menuEntry->Text = strdup( label ); - menuEntry->ID = value; - - /* Have the new menu entry attached to the current menu */ - fgListAppend( &fgStructure.CurrentMenu->Entries, &menuEntry->Node ); - - fghCalculateMenuBoxSize( ); -} - -/* - * Add a sub menu to the bottom of the current menu - */ -void FGAPIENTRY glutAddSubMenu( const char *label, int subMenuID ) -{ - SFG_MenuEntry *menuEntry; - SFG_Menu *subMenu; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutAddSubMenu" ); - menuEntry = ( SFG_MenuEntry * )calloc( sizeof( SFG_MenuEntry ), 1 ); - subMenu = fgMenuByID( subMenuID ); - - freeglut_return_if_fail( fgStructure.CurrentMenu ); - freeglut_return_if_fail( subMenu ); - - menuEntry->Text = strdup( label ); - menuEntry->SubMenu = subMenu; - menuEntry->ID = -1; - - fgListAppend( &fgStructure.CurrentMenu->Entries, &menuEntry->Node ); - fghCalculateMenuBoxSize( ); -} - -/* - * Changes the specified menu item in the current menu into a menu entry - */ -void FGAPIENTRY glutChangeToMenuEntry( int item, const char* label, int value ) -{ - SFG_MenuEntry* menuEntry = NULL; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutChangeToMenuEntry" ); - freeglut_return_if_fail( fgStructure.CurrentMenu ); - - /* Get n-th menu entry in the current menu, starting from one: */ - menuEntry = fghFindMenuEntry( fgStructure.CurrentMenu, item ); - - freeglut_return_if_fail( menuEntry ); - - /* We want it to become a normal menu entry, so: */ - if( menuEntry->Text ) - free( menuEntry->Text ); - - menuEntry->Text = strdup( label ); - menuEntry->ID = value; - menuEntry->SubMenu = NULL; - fghCalculateMenuBoxSize( ); -} - -/* - * Changes the specified menu item in the current menu into a sub-menu trigger. - */ -void FGAPIENTRY glutChangeToSubMenu( int item, const char* label, - int subMenuID ) -{ - SFG_Menu* subMenu; - SFG_MenuEntry* menuEntry; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutChangeToSubMenu" ); - subMenu = fgMenuByID( subMenuID ); - menuEntry = NULL; - - freeglut_return_if_fail( fgStructure.CurrentMenu ); - freeglut_return_if_fail( subMenu ); - - /* Get n-th menu entry in the current menu, starting from one: */ - menuEntry = fghFindMenuEntry( fgStructure.CurrentMenu, item ); - - freeglut_return_if_fail( menuEntry ); - - /* We want it to become a sub menu entry, so: */ - if( menuEntry->Text ) - free( menuEntry->Text ); - - menuEntry->Text = strdup( label ); - menuEntry->SubMenu = subMenu; - menuEntry->ID = -1; - fghCalculateMenuBoxSize( ); -} - -/* - * Removes the specified menu item from the current menu - */ -void FGAPIENTRY glutRemoveMenuItem( int item ) -{ - SFG_MenuEntry* menuEntry; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutRemoveMenuItem" ); - freeglut_return_if_fail( fgStructure.CurrentMenu ); - - /* Get n-th menu entry in the current menu, starting from one: */ - menuEntry = fghFindMenuEntry( fgStructure.CurrentMenu, item ); - - freeglut_return_if_fail( menuEntry ); - - fgListRemove( &fgStructure.CurrentMenu->Entries, &menuEntry->Node ); - if ( menuEntry->Text ) - free( menuEntry->Text ); - - free( menuEntry ); - fghCalculateMenuBoxSize( ); -} - -/* - * Attaches a menu to the current window - */ -void FGAPIENTRY glutAttachMenu( int button ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutAttachMenu" ); - - freeglut_return_if_fail( fgStructure.CurrentWindow ); - freeglut_return_if_fail( fgStructure.CurrentMenu ); - - freeglut_return_if_fail( button >= 0 ); - freeglut_return_if_fail( button < FREEGLUT_MAX_MENUS ); - - fgStructure.CurrentWindow->Menu[ button ] = fgStructure.CurrentMenu; -} - -/* - * Detaches a menu from the current window - */ -void FGAPIENTRY glutDetachMenu( int button ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDetachMenu" ); - - freeglut_return_if_fail( fgStructure.CurrentWindow ); - freeglut_return_if_fail( fgStructure.CurrentMenu ); - - freeglut_return_if_fail( button >= 0 ); - freeglut_return_if_fail( button < FREEGLUT_MAX_MENUS ); - - fgStructure.CurrentWindow->Menu[ button ] = NULL; -} - -/* - * A.Donev: Set and retrieve the menu's user data - */ -void* FGAPIENTRY glutGetMenuData( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetMenuData" ); - return fgStructure.CurrentMenu->UserData; -} - -void FGAPIENTRY glutSetMenuData(void* data) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetMenuData" ); - fgStructure.CurrentMenu->UserData=data; -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_misc.c b/examples/common/opengl-framework/freeglut/freeglut_misc.c deleted file mode 100644 index 4aa809ac..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_misc.c +++ /dev/null @@ -1,214 +0,0 @@ -/* - * freeglut_misc.c - * - * Functions that didn't fit anywhere else... - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 9 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* - * TODO BEFORE THE STABLE RELEASE: - * - * glutSetColor() -- - * glutGetColor() -- - * glutCopyColormap() -- - * glutSetKeyRepeat() -- this is evil and should be removed from API - */ - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * This functions checks if an OpenGL extension is supported or not - * - * XXX Wouldn't this be simpler and clearer if we used strtok()? - */ -int FGAPIENTRY glutExtensionSupported( const char* extension ) -{ - const char *extensions, *start; - const size_t len = strlen( extension ); - - /* Make sure there is a current window, and thus a current context available */ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutExtensionSupported" ); - freeglut_return_val_if_fail( fgStructure.CurrentWindow != NULL, 0 ); - - if (strchr(extension, ' ')) - return 0; - start = extensions = (const char *) glGetString(GL_EXTENSIONS); - - /* XXX consider printing a warning to stderr that there's no current - * rendering context. - */ - freeglut_return_val_if_fail( extensions != NULL, 0 ); - - while (1) { - const char *p = strstr(extensions, extension); - if (!p) - return 0; /* not found */ - /* check that the match isn't a super string */ - if ((p == start || p[-1] == ' ') && (p[len] == ' ' || p[len] == 0)) - return 1; - /* skip the false match and continue */ - extensions = p + len; - } - - return 0 ; -} - -#ifndef GL_INVALID_FRAMEBUFFER_OPERATION -#ifdef GL_INVALID_FRAMEBUFFER_OPERATION_EXT -#define GL_INVALID_FRAMEBUFFER_OPERATION GL_INVALID_FRAMEBUFFER_OPERATION_EXT -#else -#define GL_INVALID_FRAMEBUFFER_OPERATION 0x0506 -#endif -#endif - -#ifndef GL_TABLE_TOO_LARGE -#ifdef GL_TABLE_TOO_LARGE_EXT -#define GL_TABLE_TOO_LARGE GL_TABLE_TOO_LARGE_EXT -#else -#define GL_TABLE_TOO_LARGE 0x8031 -#endif -#endif - -#ifndef GL_TEXTURE_TOO_LARGE -#ifdef GL_TEXTURE_TOO_LARGE_EXT -#define GL_TEXTURE_TOO_LARGE GL_TEXTURE_TOO_LARGE_EXT -#else -#define GL_TEXTURE_TOO_LARGE 0x8065 -#endif -#endif - -/* - * A cut-down local version of gluErrorString to avoid depending on GLU. - */ -static const char* fghErrorString( GLenum error ) -{ - switch ( error ) { - case GL_INVALID_ENUM: return "invalid enumerant"; - case GL_INVALID_VALUE: return "invalid value"; - case GL_INVALID_OPERATION: return "invalid operation"; - case GL_STACK_OVERFLOW: return "stack overflow"; - case GL_STACK_UNDERFLOW: return "stack underflow"; - case GL_OUT_OF_MEMORY: return "out of memory"; - case GL_TABLE_TOO_LARGE: return "table too large"; - case GL_INVALID_FRAMEBUFFER_OPERATION: return "invalid framebuffer operation"; - case GL_TEXTURE_TOO_LARGE: return "texture too large"; - default: return "unknown GL error"; - } -} - -/* - * This function reports all the OpenGL errors that happened till now - */ -void FGAPIENTRY glutReportErrors( void ) -{ - GLenum error; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutReportErrors" ); - while( ( error = glGetError() ) != GL_NO_ERROR ) - fgWarning( "GL error: %s", fghErrorString( error ) ); -} - -/* - * Control the auto-repeat of keystrokes to the current window - */ -void FGAPIENTRY glutIgnoreKeyRepeat( int ignore ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutIgnoreKeyRepeat" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutIgnoreKeyRepeat" ); - - fgStructure.CurrentWindow->State.IgnoreKeyRepeat = ignore ? GL_TRUE : GL_FALSE; -} - -/* - * Set global auto-repeat of keystrokes - * - * RepeatMode should be either: - * GLUT_KEY_REPEAT_OFF - * GLUT_KEY_REPEAT_ON - * GLUT_KEY_REPEAT_DEFAULT - */ -void FGAPIENTRY glutSetKeyRepeat( int repeatMode ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetKeyRepeat" ); - - switch( repeatMode ) - { - case GLUT_KEY_REPEAT_OFF: - case GLUT_KEY_REPEAT_ON: - fgState.KeyRepeat = repeatMode; - break; - - case GLUT_KEY_REPEAT_DEFAULT: - fgState.KeyRepeat = GLUT_KEY_REPEAT_ON; - break; - - default: - fgError ("Invalid glutSetKeyRepeat mode: %d", repeatMode); - break; - } -} - -/* - * Forces the joystick callback to be executed - */ -void FGAPIENTRY glutForceJoystickFunc( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutForceJoystickFunc" ); -#if !defined(_WIN32_WCE) - freeglut_return_if_fail( fgStructure.CurrentWindow != NULL ); - freeglut_return_if_fail( FETCH_WCB( *( fgStructure.CurrentWindow ), Joystick ) ); - fgJoystickPollWindow( fgStructure.CurrentWindow ); -#endif /* !defined(_WIN32_WCE) */ -} - -/* - * - */ -void FGAPIENTRY glutSetColor( int nColor, GLfloat red, GLfloat green, GLfloat blue ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetColor" ); - /* We really need to do something here. */ -} - -/* - * - */ -GLfloat FGAPIENTRY glutGetColor( int color, int component ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetColor" ); - /* We really need to do something here. */ - return( 0.0f ); -} - -/* - * - */ -void FGAPIENTRY glutCopyColormap( int window ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutCopyColormap" ); - /* We really need to do something here. */ -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_overlay.c b/examples/common/opengl-framework/freeglut/freeglut_overlay.c deleted file mode 100644 index 2a1314a4..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_overlay.c +++ /dev/null @@ -1,45 +0,0 @@ -/* - * freeglut_overlay.c - * - * Overlay management functions (as defined by GLUT API) - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 16 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* - * NOTE: functions declared in this file probably will not be implemented. - */ - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -void FGAPIENTRY glutEstablishOverlay( void ) { /* Not implemented */ } -void FGAPIENTRY glutRemoveOverlay( void ) { /* Not implemented */ } -void FGAPIENTRY glutUseLayer( GLenum layer ) { /* Not implemented */ } -void FGAPIENTRY glutPostOverlayRedisplay( void ) { /* Not implemented */ } -void FGAPIENTRY glutPostWindowOverlayRedisplay( int ID ) { /* Not implemented */ } -void FGAPIENTRY glutShowOverlay( void ) { /* Not implemented */ } -void FGAPIENTRY glutHideOverlay( void ) { /* Not implemented */ } - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_spaceball.c b/examples/common/opengl-framework/freeglut/freeglut_spaceball.c deleted file mode 100644 index be0021f0..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_spaceball.c +++ /dev/null @@ -1,471 +0,0 @@ -/* Spaceball support for Linux. - * Written by John Tsiombikas - * - * This code supports 3Dconnexion's 6-dof space-whatever devices. - * It can communicate with either the proprietary 3Dconnexion daemon (3dxsrv) - * free spacenavd (http://spacenav.sourceforge.net), through the "standard" - * magellan X-based protocol. - */ - -#include -#include "freeglut_internal.h" - -/* -- PRIVATE FUNCTIONS --------------------------------------------------- */ - -#if TARGET_HOST_POSIX_X11 -#include - -enum { - SPNAV_EVENT_ANY, /* used by spnav_remove_events() */ - SPNAV_EVENT_MOTION, - SPNAV_EVENT_BUTTON /* includes both press and release */ -}; - -struct spnav_event_motion { - int type; - int x, y, z; - int rx, ry, rz; - unsigned int period; - int *data; -}; - -struct spnav_event_button { - int type; - int press; - int bnum; -}; - -typedef union spnav_event { - int type; - struct spnav_event_motion motion; - struct spnav_event_button button; -} spnav_event; - - -static int spnav_x11_open(Display *dpy, Window win); -static int spnav_x11_window(Window win); -static int spnav_x11_event(const XEvent *xev, spnav_event *event); -static int spnav_close(void); -static int spnav_fd(void); -static int spnav_remove_events(int type); - -static SFG_Window *spnav_win; -#endif - -/* Flag telling whether we have a spaceball: - * 0 - haven't tried initializing - * 1 - have successfully initialized - * -1 - have tried to initialize but not succeeded - */ -static int sball_initialized = 0; - - -void fgInitialiseSpaceball(void) -{ - if(sball_initialized != 0) { - return; - } - -#if TARGET_HOST_POSIX_X11 - { - Window w; - - if(!fgStructure.CurrentWindow) - { - sball_initialized = -1; - return; - } - - w = fgStructure.CurrentWindow->Window.Handle; - if(spnav_x11_open(fgDisplay.Display, w) == -1) - { - sball_initialized = -1; - return; - } - } -#endif - - sball_initialized = 1; -} - -void fgSpaceballClose(void) -{ -#if TARGET_HOST_POSIX_X11 - spnav_close(); -#endif -} - -int fgHasSpaceball(void) -{ - if(sball_initialized == 0) { - fgInitialiseSpaceball(); - if(sball_initialized != 1) { - fgWarning("fgInitialiseSpaceball failed\n"); - return 0; - } - } - -#if TARGET_HOST_POSIX_X11 - /* XXX this function should somehow query the driver if there's a device - * plugged in, as opposed to just checking if there's a driver to talk to. - */ - return spnav_fd() == -1 ? 0 : 1; -#else - return 0; -#endif -} - -int fgSpaceballNumButtons(void) -{ - if(sball_initialized == 0) { - fgInitialiseSpaceball(); - if(sball_initialized != 1) { - fgWarning("fgInitialiseSpaceball failed\n"); - return 0; - } - } - -#if TARGET_HOST_POSIX_X11 - return 2; /* TODO implement this properly */ -#else - return 0; -#endif -} - -void fgSpaceballSetWindow(SFG_Window *window) -{ - if(sball_initialized == 0) { - fgInitialiseSpaceball(); - if(sball_initialized != 1) { - return; - } - } - -#if TARGET_HOST_POSIX_X11 - if(spnav_win != window) { - spnav_x11_window(window->Window.Handle); - spnav_win = window; - } -#endif -} - - -#if TARGET_HOST_POSIX_X11 -int fgIsSpaceballXEvent(const XEvent *xev) -{ - spnav_event sev; - - if(spnav_win != fgStructure.CurrentWindow) { - /* this will also initialize spaceball if needed (first call) */ - fgSpaceballSetWindow(fgStructure.CurrentWindow); - } - - if(sball_initialized != 1) { - return 0; - } - - return spnav_x11_event(xev, &sev); -} - -void fgSpaceballHandleXEvent(const XEvent *xev) -{ - spnav_event sev; - - if(sball_initialized == 0) { - fgInitialiseSpaceball(); - if(sball_initialized != 1) { - return; - } - } - - if(spnav_x11_event(xev, &sev)) { - switch(sev.type) { - case SPNAV_EVENT_MOTION: - if(sev.motion.x | sev.motion.y | sev.motion.z) { - INVOKE_WCB(*spnav_win, SpaceMotion, (sev.motion.x, sev.motion.y, sev.motion.z)); - } - if(sev.motion.rx | sev.motion.ry | sev.motion.rz) { - INVOKE_WCB(*spnav_win, SpaceRotation, (sev.motion.rx, sev.motion.ry, sev.motion.rz)); - } - spnav_remove_events(SPNAV_EVENT_MOTION); - break; - - case SPNAV_EVENT_BUTTON: - INVOKE_WCB(*spnav_win, SpaceButton, (sev.button.bnum, sev.button.press ? GLUT_DOWN : GLUT_UP)); - break; - - default: - break; - } - } -} - -/* -The following code is part of libspnav, part of the spacenav project (spacenav.sf.net) -Copyright (C) 2007-2009 John Tsiombikas - -Redistribution and use in source and binary forms, with or without -modification, are permitted provided that the following conditions are met: - -1. Redistributions of source code must retain the above copyright notice, this - list of conditions and the following disclaimer. -2. Redistributions in binary form must reproduce the above copyright notice, - this list of conditions and the following disclaimer in the documentation - and/or other materials provided with the distribution. -3. The name of the author may not be used to endorse or promote products - derived from this software without specific prior written permission. - -THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED -WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF -MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO -EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, -EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT -OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS -INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN -CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING -IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY -OF SUCH DAMAGE. -*/ -#include -#include -#include - -#ifdef HAVE_ERRNO_H -#include -#endif - -#include -#include - -static Window get_daemon_window(Display *dpy); -static int catch_badwin(Display *dpy, XErrorEvent *err); - -static Display *dpy; -static Window app_win; -static Atom motion_event, button_press_event, button_release_event, command_event; - -enum { - CMD_APP_WINDOW = 27695, - CMD_APP_SENS -}; - -#define IS_OPEN dpy - -struct event_node { - spnav_event event; - struct event_node *next; -}; - -static int spnav_x11_open(Display *display, Window win) -{ - if(IS_OPEN) { - return -1; - } - - dpy = display; - - motion_event = XInternAtom(dpy, "MotionEvent", True); - button_press_event = XInternAtom(dpy, "ButtonPressEvent", True); - button_release_event = XInternAtom(dpy, "ButtonReleaseEvent", True); - command_event = XInternAtom(dpy, "CommandEvent", True); - - if(!motion_event || !button_press_event || !button_release_event || !command_event) { - dpy = 0; - return -1; /* daemon not started */ - } - - if(spnav_x11_window(win) == -1) { - dpy = 0; - return -1; /* daemon not started */ - } - - app_win = win; - return 0; -} - -static int spnav_close(void) -{ - if(dpy) { - spnav_x11_window(DefaultRootWindow(dpy)); - app_win = 0; - dpy = 0; - return 0; - } - return -1; -} - -static int spnav_x11_window(Window win) -{ - int (*prev_xerr_handler)(Display*, XErrorEvent*); - XEvent xev; - Window daemon_win; - - if(!IS_OPEN) { - return -1; - } - - if(!(daemon_win = get_daemon_window(dpy))) { - return -1; - } - - prev_xerr_handler = XSetErrorHandler(catch_badwin); - - xev.type = ClientMessage; - xev.xclient.send_event = False; - xev.xclient.display = dpy; - xev.xclient.window = win; - xev.xclient.message_type = command_event; - xev.xclient.format = 16; - xev.xclient.data.s[0] = ((unsigned int)win & 0xffff0000) >> 16; - xev.xclient.data.s[1] = (unsigned int)win & 0xffff; - xev.xclient.data.s[2] = CMD_APP_WINDOW; - - XSendEvent(dpy, daemon_win, False, 0, &xev); - XSync(dpy, False); - - XSetErrorHandler(prev_xerr_handler); - return 0; -} - -static int spnav_fd(void) -{ - if(dpy) { - return ConnectionNumber(dpy); - } - return -1; -} - -/*static int spnav_wait_event(spnav_event *event) -{ - if(dpy) { - for(;;) { - XEvent xev; - XNextEvent(dpy, &xev); - - if(spnav_x11_event(&xev, event) > 0) { - return event->type; - } - } - } - return 0; -} - -static int spnav_poll_event(spnav_event *event) -{ - if(dpy) { - if(XPending(dpy)) { - XEvent xev; - XNextEvent(dpy, &xev); - - return spnav_x11_event(&xev, event); - } - } - return 0; -}*/ - -static Bool match_events(Display *dpy, XEvent *xev, char *arg) -{ - int evtype = *(int*)arg; - - if(xev->type != ClientMessage) { - return False; - } - - if(xev->xclient.message_type == motion_event) { - return !evtype || evtype == SPNAV_EVENT_MOTION ? True : False; - } - if(xev->xclient.message_type == button_press_event || - xev->xclient.message_type == button_release_event) { - return !evtype || evtype == SPNAV_EVENT_BUTTON ? True : False; - } - return False; -} - -static int spnav_remove_events(int type) -{ - int rm_count = 0; - - if(dpy) { - XEvent xev; - - while(XCheckIfEvent(dpy, &xev, match_events, (char*)&type)) { - rm_count++; - } - return rm_count; - } - return 0; -} - -static int spnav_x11_event(const XEvent *xev, spnav_event *event) -{ - int i; - int xmsg_type; - - if(xev->type != ClientMessage) { - return 0; - } - - xmsg_type = xev->xclient.message_type; - - if(xmsg_type != motion_event && xmsg_type != button_press_event && - xmsg_type != button_release_event) { - return 0; - } - - if(xmsg_type == motion_event) { - event->type = SPNAV_EVENT_MOTION; - event->motion.data = &event->motion.x; - - for(i=0; i<6; i++) { - event->motion.data[i] = xev->xclient.data.s[i + 2]; - } - event->motion.period = xev->xclient.data.s[8]; - } else { - event->type = SPNAV_EVENT_BUTTON; - event->button.press = xmsg_type == button_press_event ? 1 : 0; - event->button.bnum = xev->xclient.data.s[2]; - } - return event->type; -} - - -static Window get_daemon_window(Display *dpy) -{ - Window win, root_win; - XTextProperty wname; - Atom type; - int fmt; - unsigned long nitems, bytes_after; - unsigned char *prop; - - root_win = DefaultRootWindow(dpy); - - XGetWindowProperty(dpy, root_win, command_event, 0, 1, False, AnyPropertyType, &type, &fmt, &nitems, &bytes_after, &prop); - if(!prop) { - return 0; - } - - win = *(Window*)prop; - XFree(prop); - - if(!XGetWMName(dpy, win, &wname) || strcmp("Magellan Window", (char*)wname.value) != 0) { - return 0; - } - - return win; -} - -static int catch_badwin(Display *dpy, XErrorEvent *err) -{ - char buf[256]; - - if(err->error_code == BadWindow) { - /* do nothing? */ - } else { - XGetErrorText(dpy, err->error_code, buf, sizeof buf); - fprintf(stderr, "Caught unexpected X error: %s\n", buf); - } - return 0; -} - -#endif /* TARGET_HOST_POSIX_X11 */ diff --git a/examples/common/opengl-framework/freeglut/freeglut_state.c b/examples/common/opengl-framework/freeglut/freeglut_state.c deleted file mode 100644 index 9e3a627d..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_state.c +++ /dev/null @@ -1,887 +0,0 @@ -/* - * freeglut_state.c - * - * Freeglut state query methods. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 16 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* - * TODO BEFORE THE STABLE RELEASE: - * - * glutGet() -- X11 tests passed, but check if all enums - * handled (what about Win32?) - * glutDeviceGet() -- X11 tests passed, but check if all enums - * handled (what about Win32?) - * glutGetModifiers() -- OK, but could also remove the limitation - * glutLayerGet() -- what about GLUT_NORMAL_DAMAGED? - * - * The fail-on-call policy will help adding the most needed things imho. - */ - -/* -- LOCAL DEFINITIONS ---------------------------------------------------- */ - -/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ - -#if TARGET_HOST_POSIX_X11 -/* - * Queries the GL context about some attributes - */ -static int fghGetConfig( int attribute ) -{ - int returnValue = 0; - int result; /* Not checked */ - - if( fgStructure.CurrentWindow ) - result = glXGetFBConfigAttrib( fgDisplay.Display, - *(fgStructure.CurrentWindow->Window.FBConfig), - attribute, - &returnValue ); - - return returnValue; -} -#endif - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * General settings assignment method - */ -void FGAPIENTRY glutSetOption( GLenum eWhat, int value ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetOption" ); - - /* - * XXX In chronological code add order. (WHY in that order?) - */ - switch( eWhat ) - { - case GLUT_INIT_WINDOW_X: - fgState.Position.X = (GLint)value; - break; - - case GLUT_INIT_WINDOW_Y: - fgState.Position.Y = (GLint)value; - break; - - case GLUT_INIT_WINDOW_WIDTH: - fgState.Size.X = (GLint)value; - break; - - case GLUT_INIT_WINDOW_HEIGHT: - fgState.Size.Y = (GLint)value; - break; - - case GLUT_INIT_DISPLAY_MODE: - fgState.DisplayMode = (unsigned int)value; - break; - - case GLUT_ACTION_ON_WINDOW_CLOSE: - fgState.ActionOnWindowClose = value; - break; - - case GLUT_RENDERING_CONTEXT: - fgState.UseCurrentContext = - ( value == GLUT_USE_CURRENT_CONTEXT ) ? GL_TRUE : GL_FALSE; - break; - - case GLUT_DIRECT_RENDERING: - fgState.DirectContext = value; - break; - - case GLUT_WINDOW_CURSOR: - if( fgStructure.CurrentWindow != NULL ) - fgStructure.CurrentWindow->State.Cursor = value; - break; - - case GLUT_AUX: - fgState.AuxiliaryBufferNumber = value; - break; - - case GLUT_MULTISAMPLE: - fgState.SampleNumber = value; - break; - - default: - fgWarning( "glutSetOption(): missing enum handle %d", eWhat ); - break; - } -} - -#if TARGET_HOST_MS_WINDOWS -/* The following include file is available from SGI but is not standard: - * #include - * So we copy the necessary parts out of it to support the multisampling query - */ -#define WGL_SAMPLES_ARB 0x2042 -#endif - - -/* - * General settings query method - */ -int FGAPIENTRY glutGet( GLenum eWhat ) -{ -#if TARGET_HOST_MS_WINDOWS - int returnValue ; - GLboolean boolValue ; -#endif - - int nsamples = 0; - - switch (eWhat) - { - case GLUT_INIT_STATE: - return fgState.Initialised; - - case GLUT_ELAPSED_TIME: - return fgElapsedTime(); - } - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGet" ); - - /* XXX In chronological code add order. (WHY in that order?) */ - switch( eWhat ) - { - /* Following values are stored in fgState and fgDisplay global structures */ - case GLUT_SCREEN_WIDTH: return fgDisplay.ScreenWidth ; - case GLUT_SCREEN_HEIGHT: return fgDisplay.ScreenHeight ; - case GLUT_SCREEN_WIDTH_MM: return fgDisplay.ScreenWidthMM ; - case GLUT_SCREEN_HEIGHT_MM: return fgDisplay.ScreenHeightMM; - case GLUT_INIT_WINDOW_X: return fgState.Position.Use ? - fgState.Position.X : -1 ; - case GLUT_INIT_WINDOW_Y: return fgState.Position.Use ? - fgState.Position.Y : -1 ; - case GLUT_INIT_WINDOW_WIDTH: return fgState.Size.Use ? - fgState.Size.X : -1 ; - case GLUT_INIT_WINDOW_HEIGHT: return fgState.Size.Use ? - fgState.Size.Y : -1 ; - case GLUT_INIT_DISPLAY_MODE: return fgState.DisplayMode ; - case GLUT_INIT_MAJOR_VERSION: return fgState.MajorVersion ; - case GLUT_INIT_MINOR_VERSION: return fgState.MinorVersion ; - case GLUT_INIT_FLAGS: return fgState.ContextFlags ; - case GLUT_INIT_PROFILE: return fgState.ContextProfile ; - -#if TARGET_HOST_POSIX_X11 - /* - * The window/context specific queries are handled mostly by - * fghGetConfig(). - */ - case GLUT_WINDOW_NUM_SAMPLES: -#ifdef GLX_VERSION_1_3 - glGetIntegerv(GL_SAMPLES, &nsamples); -#endif - return nsamples; - - /* - * The rest of GLX queries under X are general enough to use a macro to - * check them - */ -# define GLX_QUERY(a,b) case a: return fghGetConfig( b ); - - GLX_QUERY( GLUT_WINDOW_RGBA, GLX_RGBA ); - GLX_QUERY( GLUT_WINDOW_DOUBLEBUFFER, GLX_DOUBLEBUFFER ); - GLX_QUERY( GLUT_WINDOW_BUFFER_SIZE, GLX_BUFFER_SIZE ); - GLX_QUERY( GLUT_WINDOW_STENCIL_SIZE, GLX_STENCIL_SIZE ); - GLX_QUERY( GLUT_WINDOW_DEPTH_SIZE, GLX_DEPTH_SIZE ); - GLX_QUERY( GLUT_WINDOW_RED_SIZE, GLX_RED_SIZE ); - GLX_QUERY( GLUT_WINDOW_GREEN_SIZE, GLX_GREEN_SIZE ); - GLX_QUERY( GLUT_WINDOW_BLUE_SIZE, GLX_BLUE_SIZE ); - GLX_QUERY( GLUT_WINDOW_ALPHA_SIZE, GLX_ALPHA_SIZE ); - GLX_QUERY( GLUT_WINDOW_ACCUM_RED_SIZE, GLX_ACCUM_RED_SIZE ); - GLX_QUERY( GLUT_WINDOW_ACCUM_GREEN_SIZE, GLX_ACCUM_GREEN_SIZE ); - GLX_QUERY( GLUT_WINDOW_ACCUM_BLUE_SIZE, GLX_ACCUM_BLUE_SIZE ); - GLX_QUERY( GLUT_WINDOW_ACCUM_ALPHA_SIZE, GLX_ACCUM_ALPHA_SIZE ); - GLX_QUERY( GLUT_WINDOW_STEREO, GLX_STEREO ); - -# undef GLX_QUERY - - /* Colormap size is handled in a bit different way than all the rest */ - case GLUT_WINDOW_COLORMAP_SIZE: - if( (fghGetConfig( GLX_RGBA )) || (fgStructure.CurrentWindow == NULL) ) - { - /* - * We've got a RGBA visual, so there is no colormap at all. - * The other possibility is that we have no current window set. - */ - return 0; - } - else - { - const GLXFBConfig * fbconfig = - fgStructure.CurrentWindow->Window.FBConfig; - - XVisualInfo * visualInfo = - glXGetVisualFromFBConfig( fgDisplay.Display, *fbconfig ); - - const int result = visualInfo->visual->map_entries; - - XFree(visualInfo); - - return result; - } - - /* - * Those calls are somewhat similiar, as they use XGetWindowAttributes() - * function - */ - case GLUT_WINDOW_X: - case GLUT_WINDOW_Y: - case GLUT_WINDOW_BORDER_WIDTH: - case GLUT_WINDOW_HEADER_HEIGHT: - { - int x, y; - Window w; - - if( fgStructure.CurrentWindow == NULL ) - return 0; - - XTranslateCoordinates( - fgDisplay.Display, - fgStructure.CurrentWindow->Window.Handle, - fgDisplay.RootWindow, - 0, 0, &x, &y, &w); - - switch ( eWhat ) - { - case GLUT_WINDOW_X: return x; - case GLUT_WINDOW_Y: return y; - } - - if ( w == 0 ) - return 0; - XTranslateCoordinates( - fgDisplay.Display, - fgStructure.CurrentWindow->Window.Handle, - w, 0, 0, &x, &y, &w); - - switch ( eWhat ) - { - case GLUT_WINDOW_BORDER_WIDTH: return x; - case GLUT_WINDOW_HEADER_HEIGHT: return y; - } - } - - case GLUT_WINDOW_WIDTH: - case GLUT_WINDOW_HEIGHT: - { - XWindowAttributes winAttributes; - - if( fgStructure.CurrentWindow == NULL ) - return 0; - XGetWindowAttributes( - fgDisplay.Display, - fgStructure.CurrentWindow->Window.Handle, - &winAttributes - ); - switch ( eWhat ) - { - case GLUT_WINDOW_WIDTH: return winAttributes.width ; - case GLUT_WINDOW_HEIGHT: return winAttributes.height ; - } - } - - /* I do not know yet if there will be a fgChooseVisual() function for Win32 */ - case GLUT_DISPLAY_MODE_POSSIBLE: - { - /* We should not have to call fgChooseFBConfig again here. */ - GLXFBConfig * fbconfig; - int isPossible; - - fbconfig = fgChooseFBConfig(NULL); - - if (fbconfig == NULL) - { - isPossible = 0; - } - else - { - isPossible = 1; - XFree(fbconfig); - } - - return isPossible; - } - - /* This is system-dependant */ - case GLUT_WINDOW_FORMAT_ID: - if( fgStructure.CurrentWindow == NULL ) - return 0; - - return fghGetConfig( GLX_VISUAL_ID ); - -#elif TARGET_HOST_MS_WINDOWS - - case GLUT_WINDOW_NUM_SAMPLES: - glGetIntegerv(WGL_SAMPLES_ARB, &nsamples); - return nsamples; - - /* Handle the OpenGL inquiries */ - case GLUT_WINDOW_RGBA: -#if defined(_WIN32_WCE) - boolValue = (GLboolean)0; /* WinCE doesn't support this feature */ -#else - glGetBooleanv ( GL_RGBA_MODE, &boolValue ); - returnValue = boolValue ? 1 : 0; -#endif - return returnValue; - case GLUT_WINDOW_DOUBLEBUFFER: -#if defined(_WIN32_WCE) - boolValue = (GLboolean)0; /* WinCE doesn't support this feature */ -#else - glGetBooleanv ( GL_DOUBLEBUFFER, &boolValue ); - returnValue = boolValue ? 1 : 0; -#endif - return returnValue; - case GLUT_WINDOW_STEREO: -#if defined(_WIN32_WCE) - boolValue = (GLboolean)0; /* WinCE doesn't support this feature */ -#else - glGetBooleanv ( GL_STEREO, &boolValue ); - returnValue = boolValue ? 1 : 0; -#endif - return returnValue; - - case GLUT_WINDOW_RED_SIZE: - glGetIntegerv ( GL_RED_BITS, &returnValue ); - return returnValue; - case GLUT_WINDOW_GREEN_SIZE: - glGetIntegerv ( GL_GREEN_BITS, &returnValue ); - return returnValue; - case GLUT_WINDOW_BLUE_SIZE: - glGetIntegerv ( GL_BLUE_BITS, &returnValue ); - return returnValue; - case GLUT_WINDOW_ALPHA_SIZE: - glGetIntegerv ( GL_ALPHA_BITS, &returnValue ); - return returnValue; - case GLUT_WINDOW_ACCUM_RED_SIZE: -#if defined(_WIN32_WCE) - returnValue = 0; /* WinCE doesn't support this feature */ -#else - glGetIntegerv ( GL_ACCUM_RED_BITS, &returnValue ); -#endif - return returnValue; - case GLUT_WINDOW_ACCUM_GREEN_SIZE: -#if defined(_WIN32_WCE) - returnValue = 0; /* WinCE doesn't support this feature */ -#else - glGetIntegerv ( GL_ACCUM_GREEN_BITS, &returnValue ); -#endif - return returnValue; - case GLUT_WINDOW_ACCUM_BLUE_SIZE: -#if defined(_WIN32_WCE) - returnValue = 0; /* WinCE doesn't support this feature */ -#else - glGetIntegerv ( GL_ACCUM_BLUE_BITS, &returnValue ); -#endif - return returnValue; - case GLUT_WINDOW_ACCUM_ALPHA_SIZE: -#if defined(_WIN32_WCE) - returnValue = 0; /* WinCE doesn't support this feature */ -#else - glGetIntegerv ( GL_ACCUM_ALPHA_BITS, &returnValue ); -#endif - return returnValue; - case GLUT_WINDOW_DEPTH_SIZE: - glGetIntegerv ( GL_DEPTH_BITS, &returnValue ); - return returnValue; - - case GLUT_WINDOW_BUFFER_SIZE: - returnValue = 1 ; /* ????? */ - return returnValue; - case GLUT_WINDOW_STENCIL_SIZE: - returnValue = 0 ; /* ????? */ - return returnValue; - - case GLUT_WINDOW_X: - case GLUT_WINDOW_Y: - case GLUT_WINDOW_WIDTH: - case GLUT_WINDOW_HEIGHT: - { - /* - * There is considerable confusion about the "right thing to - * do" concerning window size and position. GLUT itself is - * not consistent between Windows and UNIX/X11; since - * platform independence is a virtue for "freeglut", we - * decided to break with GLUT's behaviour. - * - * Under UNIX/X11, it is apparently not possible to get the - * window border sizes in order to subtract them off the - * window's initial position until some time after the window - * has been created. Therefore we decided on the following - * behaviour, both under Windows and under UNIX/X11: - * - When you create a window with position (x,y) and size - * (w,h), the upper left hand corner of the outside of the - * window is at (x,y) and the size of the drawable area is - * (w,h). - * - When you query the size and position of the window--as - * is happening here for Windows--"freeglut" will return - * the size of the drawable area--the (w,h) that you - * specified when you created the window--and the coordinates - * of the upper left hand corner of the drawable - * area--which is NOT the (x,y) you specified. - */ - - RECT winRect; - - freeglut_return_val_if_fail( fgStructure.CurrentWindow != NULL, 0 ); - -#if defined(_WIN32_WCE) - GetWindowRect( fgStructure.CurrentWindow->Window.Handle, &winRect ); -#else - winRect = fghGetClientArea(fgStructure.CurrentWindow, FALSE); -#endif /* defined(_WIN32_WCE) */ - - switch( eWhat ) - { - case GLUT_WINDOW_X: return winRect.left ; - case GLUT_WINDOW_Y: return winRect.top ; - case GLUT_WINDOW_WIDTH: return winRect.right - winRect.left; - case GLUT_WINDOW_HEIGHT: return winRect.bottom - winRect.top; - } - } - break; - - case GLUT_WINDOW_BORDER_WIDTH : - case GLUT_WINDOW_HEADER_HEIGHT : -#if defined(_WIN32_WCE) - return 0; -#else - { - DWORD windowStyle; - - if (fgStructure.CurrentWindow && fgStructure.CurrentWindow->Window.Handle) - windowStyle = GetWindowLong(fgStructure.CurrentWindow->Window.Handle, GWL_STYLE); - else - /* If no window, return sizes for a default window with title bar and border */ - windowStyle = WS_OVERLAPPEDWINDOW; - - switch( eWhat ) - { - case GLUT_WINDOW_BORDER_WIDTH: - { - int xBorderWidth, yBorderWidth; - fghGetBorderWidth(windowStyle, &xBorderWidth, &yBorderWidth); - return xBorderWidth; - } - case GLUT_WINDOW_HEADER_HEIGHT: - /* Need to query for WS_MAXIMIZEBOX to see if we have a title bar, the WS_CAPTION query is also true for a WS_DLGFRAME only... */ - return (windowStyle & WS_MAXIMIZEBOX)? GetSystemMetrics( SM_CYCAPTION ) : 0; - } - } -#endif /* defined(_WIN32_WCE) */ - - case GLUT_DISPLAY_MODE_POSSIBLE: -#if defined(_WIN32_WCE) - return 0; -#else - return fgSetupPixelFormat( fgStructure.CurrentWindow, GL_TRUE, - PFD_MAIN_PLANE ); -#endif /* defined(_WIN32_WCE) */ - - - case GLUT_WINDOW_FORMAT_ID: -#if !defined(_WIN32_WCE) - if( fgStructure.CurrentWindow != NULL ) - return GetPixelFormat( fgStructure.CurrentWindow->Window.Device ); -#endif /* defined(_WIN32_WCE) */ - return 0; - -#endif - - /* The window structure queries */ - case GLUT_WINDOW_PARENT: - if( fgStructure.CurrentWindow == NULL ) return 0; - if( fgStructure.CurrentWindow->Parent == NULL ) return 0; - return fgStructure.CurrentWindow->Parent->ID; - - case GLUT_WINDOW_NUM_CHILDREN: - if( fgStructure.CurrentWindow == NULL ) - return 0; - return fgListLength( &fgStructure.CurrentWindow->Children ); - - case GLUT_WINDOW_CURSOR: - if( fgStructure.CurrentWindow == NULL ) - return 0; - return fgStructure.CurrentWindow->State.Cursor; - - case GLUT_MENU_NUM_ITEMS: - if( fgStructure.CurrentMenu == NULL ) - return 0; - return fgListLength( &fgStructure.CurrentMenu->Entries ); - - case GLUT_ACTION_ON_WINDOW_CLOSE: - return fgState.ActionOnWindowClose; - - case GLUT_VERSION : - return VERSION_MAJOR * 10000 + VERSION_MINOR * 100 + VERSION_PATCH; - - case GLUT_RENDERING_CONTEXT: - return fgState.UseCurrentContext ? GLUT_USE_CURRENT_CONTEXT - : GLUT_CREATE_NEW_CONTEXT; - - case GLUT_DIRECT_RENDERING: - return fgState.DirectContext; - - case GLUT_FULL_SCREEN: - return fgStructure.CurrentWindow->State.IsFullscreen; - - case GLUT_AUX: - return fgState.AuxiliaryBufferNumber; - - case GLUT_MULTISAMPLE: - return fgState.SampleNumber; - - default: - fgWarning( "glutGet(): missing enum handle %d", eWhat ); - break; - } - return -1; -} - -/* - * Returns various device information. - */ -int FGAPIENTRY glutDeviceGet( GLenum eWhat ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDeviceGet" ); - - /* XXX WARNING: we are mostly lying in this function. */ - switch( eWhat ) - { - case GLUT_HAS_KEYBOARD: - /* - * Win32 is assumed a keyboard, and this cannot be queried, - * except for WindowsCE. - * - * X11 has a core keyboard by definition, although it can - * be present as a virtual/dummy keyboard. For now, there - * is no reliable way to tell if a real keyboard is present. - */ -#if defined(_WIN32_CE) - return ( GetKeyboardStatus() & KBDI_KEYBOARD_PRESENT ) ? 1 : 0; -# if FREEGLUT_LIB_PRAGMAS -# pragma comment (lib,"Kbdui.lib") -# endif - -#else - return 1; -#endif - -#if TARGET_HOST_POSIX_X11 - - /* X11 has a mouse by definition */ - case GLUT_HAS_MOUSE: - return 1 ; - - case GLUT_NUM_MOUSE_BUTTONS: - /* We should be able to pass NULL when the last argument is zero, - * but at least one X server has a bug where this causes a segfault. - * - * In XFree86/Xorg servers, a mouse wheel is seen as two buttons - * rather than an Axis; "freeglut_main.c" expects this when - * checking for a wheel event. - */ - { - unsigned char map; - int nbuttons = XGetPointerMapping(fgDisplay.Display, &map,0); - return nbuttons; - } - -#elif TARGET_HOST_MS_WINDOWS - - case GLUT_HAS_MOUSE: - /* - * MS Windows can be booted without a mouse. - */ - return GetSystemMetrics( SM_MOUSEPRESENT ); - - case GLUT_NUM_MOUSE_BUTTONS: -# if defined(_WIN32_WCE) - return 1; -# else - return GetSystemMetrics( SM_CMOUSEBUTTONS ); -# endif -#endif - - case GLUT_HAS_JOYSTICK: - return fgJoystickDetect (); - - case GLUT_OWNS_JOYSTICK: - return fgState.JoysticksInitialised; - - case GLUT_JOYSTICK_POLL_RATE: - return fgStructure.CurrentWindow ? fgStructure.CurrentWindow->State.JoystickPollRate : 0; - - /* XXX The following two are only for Joystick 0 but this is an improvement */ - case GLUT_JOYSTICK_BUTTONS: - return glutJoystickGetNumButtons ( 0 ); - - case GLUT_JOYSTICK_AXES: - return glutJoystickGetNumAxes ( 0 ); - - case GLUT_HAS_DIAL_AND_BUTTON_BOX: - return fgInputDeviceDetect (); - - case GLUT_NUM_DIALS: - if ( fgState.InputDevsInitialised ) return 8; - return 0; - - case GLUT_NUM_BUTTON_BOX_BUTTONS: - return 0; - - case GLUT_HAS_SPACEBALL: - return fgHasSpaceball(); - - case GLUT_HAS_TABLET: - return 0; - - case GLUT_NUM_SPACEBALL_BUTTONS: - return fgSpaceballNumButtons(); - - case GLUT_NUM_TABLET_BUTTONS: - return 0; - - case GLUT_DEVICE_IGNORE_KEY_REPEAT: - return fgStructure.CurrentWindow ? fgStructure.CurrentWindow->State.IgnoreKeyRepeat : 0; - - case GLUT_DEVICE_KEY_REPEAT: - return fgState.KeyRepeat; - - default: - fgWarning( "glutDeviceGet(): missing enum handle %d", eWhat ); - break; - } - - /* And now -- the failure. */ - return -1; -} - -/* - * This should return the current state of ALT, SHIFT and CTRL keys. - */ -int FGAPIENTRY glutGetModifiers( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetModifiers" ); - if( fgState.Modifiers == INVALID_MODIFIERS ) - { - fgWarning( "glutGetModifiers() called outside an input callback" ); - return 0; - } - - return fgState.Modifiers; -} - -/* - * Return the state of the GLUT API overlay subsystem. A misery ;-) - */ -int FGAPIENTRY glutLayerGet( GLenum eWhat ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutLayerGet" ); - - /* - * This is easy as layers are not implemented ;-) - * - * XXX Can we merge the UNIX/X11 and WIN32 sections? Or - * XXX is overlay support planned? - */ - switch( eWhat ) - { - -#if TARGET_HOST_POSIX_X11 - - case GLUT_OVERLAY_POSSIBLE: - return 0; - - case GLUT_LAYER_IN_USE: - return GLUT_NORMAL; - - case GLUT_HAS_OVERLAY: - return 0; - - case GLUT_TRANSPARENT_INDEX: - /* - * Return just anything, which is always defined as zero - * - * XXX HUH? - */ - return 0; - - case GLUT_NORMAL_DAMAGED: - /* XXX Actually I do not know. Maybe. */ - return 0; - - case GLUT_OVERLAY_DAMAGED: - return -1; - -#elif TARGET_HOST_MS_WINDOWS - - case GLUT_OVERLAY_POSSIBLE: -/* return fgSetupPixelFormat( fgStructure.CurrentWindow, GL_TRUE, - PFD_OVERLAY_PLANE ); */ - return 0 ; - - case GLUT_LAYER_IN_USE: - return GLUT_NORMAL; - - case GLUT_HAS_OVERLAY: - return 0; - - case GLUT_TRANSPARENT_INDEX: - /* - * Return just anything, which is always defined as zero - * - * XXX HUH? - */ - return 0; - - case GLUT_NORMAL_DAMAGED: - /* XXX Actually I do not know. Maybe. */ - return 0; - - case GLUT_OVERLAY_DAMAGED: - return -1; -#endif - - default: - fgWarning( "glutLayerGet(): missing enum handle %d", eWhat ); - break; - } - - /* And fail. That's good. Programs do love failing. */ - return -1; -} - -int * FGAPIENTRY glutGetModeValues(GLenum eWhat, int * size) -{ - int * array; - -#if TARGET_HOST_POSIX_X11 - int attributes[9]; - GLXFBConfig * fbconfigArray; /* Array of FBConfigs */ - int fbconfigArraySize; /* Number of FBConfigs in the array */ - int attribute_name = 0; -#endif - - FREEGLUT_EXIT_IF_NOT_INITIALISED("glutGetModeValues"); - - array = NULL; - *size = 0; - - switch (eWhat) - { -#if TARGET_HOST_POSIX_X11 - case GLUT_AUX: - case GLUT_MULTISAMPLE: - - attributes[0] = GLX_BUFFER_SIZE; - attributes[1] = GLX_DONT_CARE; - - switch (eWhat) - { - case GLUT_AUX: - /* - FBConfigs are now sorted by increasing number of auxiliary - buffers. We want at least one buffer. - */ - attributes[2] = GLX_AUX_BUFFERS; - attributes[3] = 1; - attributes[4] = None; - - attribute_name = GLX_AUX_BUFFERS; - - break; - - - case GLUT_MULTISAMPLE: - attributes[2] = GLX_AUX_BUFFERS; - attributes[3] = GLX_DONT_CARE; - attributes[4] = GLX_SAMPLE_BUFFERS; - attributes[5] = 1; - /* - FBConfigs are now sorted by increasing number of samples per - pixel. We want at least one sample. - */ - attributes[6] = GLX_SAMPLES; - attributes[7] = 1; - attributes[8] = None; - - attribute_name = GLX_SAMPLES; - - break; - } - - fbconfigArray = glXChooseFBConfig(fgDisplay.Display, - fgDisplay.Screen, - attributes, - &fbconfigArraySize); - - if (fbconfigArray != NULL) - { - int * temp_array; - int result; /* Returned by glXGetFBConfigAttrib. Not checked. */ - int previous_value; - int i; - - temp_array = malloc(sizeof(int) * fbconfigArraySize); - previous_value = 0; - - for (i = 0; i < fbconfigArraySize; i++) - { - int value; - - result = glXGetFBConfigAttrib(fgDisplay.Display, - fbconfigArray[i], - attribute_name, - &value); - if (value > previous_value) - { - temp_array[*size] = value; - previous_value = value; - (*size)++; - } - } - - array = malloc(sizeof(int) * (*size)); - for (i = 0; i < *size; i++) - { - array[i] = temp_array[i]; - } - - free(temp_array); - XFree(fbconfigArray); - } - - break; -#endif - - default: - break; - } - - return array; -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_stroke_mono_roman.c b/examples/common/opengl-framework/freeglut/freeglut_stroke_mono_roman.c deleted file mode 100644 index ec86d56f..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_stroke_mono_roman.c +++ /dev/null @@ -1,2849 +0,0 @@ -/* - * freeglut_stroke_mono_roman.c - * - * freeglut Monospace Roman stroke font definition - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 16 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - - -/* This file has been automatically generated by the genstroke utility. */ - -#include -#include "freeglut_internal.h" - -/* char: 0x20 */ - -static const SFG_StrokeStrip ch32st[] = -{ - { 0, NULL } -}; - -static const SFG_StrokeChar ch32 = {104.762f,0,ch32st}; - -/* char: 0x21 */ - -static const SFG_StrokeVertex ch33st0[] = -{ - {52.381f,100.0f}, - {52.381f,33.3333f} -}; - -static const SFG_StrokeVertex ch33st1[] = -{ - {52.381f,9.5238f}, - {47.6191f,4.7619f}, - {52.381f,0.0f}, - {57.1429f,4.7619f}, - {52.381f,9.5238f} -}; - -static const SFG_StrokeStrip ch33st[] = -{ - {2,ch33st0}, - {5,ch33st1} -}; - -static const SFG_StrokeChar ch33 = {104.762f,2,ch33st}; - -/* char: 0x22 */ - -static const SFG_StrokeVertex ch34st0[] = -{ - {33.3334f,100.0f}, - {33.3334f,66.6667f} -}; - -static const SFG_StrokeVertex ch34st1[] = -{ - {71.4286f,100.0f}, - {71.4286f,66.6667f} -}; - -static const SFG_StrokeStrip ch34st[] = -{ - {2,ch34st0}, - {2,ch34st1} -}; - -static const SFG_StrokeChar ch34 = {104.762f,2,ch34st}; - -/* char: 0x23 */ - -static const SFG_StrokeVertex ch35st0[] = -{ - {54.7619f,119.048f}, - {21.4286f,-33.3333f} -}; - -static const SFG_StrokeVertex ch35st1[] = -{ - {83.3334f,119.048f}, - {50.0f,-33.3333f} -}; - -static const SFG_StrokeVertex ch35st2[] = -{ - {21.4286f,57.1429f}, - {88.0952f,57.1429f} -}; - -static const SFG_StrokeVertex ch35st3[] = -{ - {16.6667f,28.5714f}, - {83.3334f,28.5714f} -}; - -static const SFG_StrokeStrip ch35st[] = -{ - {2,ch35st0}, - {2,ch35st1}, - {2,ch35st2}, - {2,ch35st3} -}; - -static const SFG_StrokeChar ch35 = {104.762f,4,ch35st}; - -/* char: 0x24 */ - -static const SFG_StrokeVertex ch36st0[] = -{ - {42.8571f,119.048f}, - {42.8571f,-19.0476f} -}; - -static const SFG_StrokeVertex ch36st1[] = -{ - {61.9047f,119.048f}, - {61.9047f,-19.0476f} -}; - -static const SFG_StrokeVertex ch36st2[] = -{ - {85.7143f,85.7143f}, - {76.1905f,95.2381f}, - {61.9047f,100.0f}, - {42.8571f,100.0f}, - {28.5714f,95.2381f}, - {19.0476f,85.7143f}, - {19.0476f,76.1905f}, - {23.8095f,66.6667f}, - {28.5714f,61.9048f}, - {38.0952f,57.1429f}, - {66.6666f,47.619f}, - {76.1905f,42.8571f}, - {80.9524f,38.0952f}, - {85.7143f,28.5714f}, - {85.7143f,14.2857f}, - {76.1905f,4.7619f}, - {61.9047f,0.0f}, - {42.8571f,0.0f}, - {28.5714f,4.7619f}, - {19.0476f,14.2857f} -}; - -static const SFG_StrokeStrip ch36st[] = -{ - {2,ch36st0}, - {2,ch36st1}, - {20,ch36st2} -}; - -static const SFG_StrokeChar ch36 = {104.762f,3,ch36st}; - -/* char: 0x25 */ - -static const SFG_StrokeVertex ch37st0[] = -{ - {95.2381f,100.0f}, - {9.5238f,0.0f} -}; - -static const SFG_StrokeVertex ch37st1[] = -{ - {33.3333f,100.0f}, - {42.8571f,90.4762f}, - {42.8571f,80.9524f}, - {38.0952f,71.4286f}, - {28.5714f,66.6667f}, - {19.0476f,66.6667f}, - {9.5238f,76.1905f}, - {9.5238f,85.7143f}, - {14.2857f,95.2381f}, - {23.8095f,100.0f}, - {33.3333f,100.0f}, - {42.8571f,95.2381f}, - {57.1428f,90.4762f}, - {71.4286f,90.4762f}, - {85.7143f,95.2381f}, - {95.2381f,100.0f} -}; - -static const SFG_StrokeVertex ch37st2[] = -{ - {76.1905f,33.3333f}, - {66.6667f,28.5714f}, - {61.9048f,19.0476f}, - {61.9048f,9.5238f}, - {71.4286f,0.0f}, - {80.9524f,0.0f}, - {90.4762f,4.7619f}, - {95.2381f,14.2857f}, - {95.2381f,23.8095f}, - {85.7143f,33.3333f}, - {76.1905f,33.3333f} -}; - -static const SFG_StrokeStrip ch37st[] = -{ - {2,ch37st0}, - {16,ch37st1}, - {11,ch37st2} -}; - -static const SFG_StrokeChar ch37 = {104.762f,3,ch37st}; - -/* char: 0x26 */ - -static const SFG_StrokeVertex ch38st0[] = -{ - {100.0f,57.1429f}, - {100.0f,61.9048f}, - {95.2381f,66.6667f}, - {90.4762f,66.6667f}, - {85.7143f,61.9048f}, - {80.9524f,52.381f}, - {71.4286f,28.5714f}, - {61.9048f,14.2857f}, - {52.3809f,4.7619f}, - {42.8571f,0.0f}, - {23.8095f,0.0f}, - {14.2857f,4.7619f}, - {9.5238f,9.5238f}, - {4.7619f,19.0476f}, - {4.7619f,28.5714f}, - {9.5238f,38.0952f}, - {14.2857f,42.8571f}, - {47.619f,61.9048f}, - {52.3809f,66.6667f}, - {57.1429f,76.1905f}, - {57.1429f,85.7143f}, - {52.3809f,95.2381f}, - {42.8571f,100.0f}, - {33.3333f,95.2381f}, - {28.5714f,85.7143f}, - {28.5714f,76.1905f}, - {33.3333f,61.9048f}, - {42.8571f,47.619f}, - {66.6667f,14.2857f}, - {76.1905f,4.7619f}, - {85.7143f,0.0f}, - {95.2381f,0.0f}, - {100.0f,4.7619f}, - {100.0f,9.5238f} -}; - -static const SFG_StrokeStrip ch38st[] = -{ - {34,ch38st0} -}; - -static const SFG_StrokeChar ch38 = {104.762f,1,ch38st}; - -/* char: 0x27 */ - -static const SFG_StrokeVertex ch39st0[] = -{ - {52.381f,100.0f}, - {52.381f,66.6667f} -}; - -static const SFG_StrokeStrip ch39st[] = -{ - {2,ch39st0} -}; - -static const SFG_StrokeChar ch39 = {104.762f,1,ch39st}; - -/* char: 0x28 */ - -static const SFG_StrokeVertex ch40st0[] = -{ - {69.0476f,119.048f}, - {59.5238f,109.524f}, - {50.0f,95.2381f}, - {40.4762f,76.1905f}, - {35.7143f,52.381f}, - {35.7143f,33.3333f}, - {40.4762f,9.5238f}, - {50.0f,-9.5238f}, - {59.5238f,-23.8095f}, - {69.0476f,-33.3333f} -}; - -static const SFG_StrokeStrip ch40st[] = -{ - {10,ch40st0} -}; - -static const SFG_StrokeChar ch40 = {104.762f,1,ch40st}; - -/* char: 0x29 */ - -static const SFG_StrokeVertex ch41st0[] = -{ - {35.7143f,119.048f}, - {45.2381f,109.524f}, - {54.7619f,95.2381f}, - {64.2857f,76.1905f}, - {69.0476f,52.381f}, - {69.0476f,33.3333f}, - {64.2857f,9.5238f}, - {54.7619f,-9.5238f}, - {45.2381f,-23.8095f}, - {35.7143f,-33.3333f} -}; - -static const SFG_StrokeStrip ch41st[] = -{ - {10,ch41st0} -}; - -static const SFG_StrokeChar ch41 = {104.762f,1,ch41st}; - -/* char: 0x2a */ - -static const SFG_StrokeVertex ch42st0[] = -{ - {52.381f,71.4286f}, - {52.381f,14.2857f} -}; - -static const SFG_StrokeVertex ch42st1[] = -{ - {28.5715f,57.1429f}, - {76.1905f,28.5714f} -}; - -static const SFG_StrokeVertex ch42st2[] = -{ - {76.1905f,57.1429f}, - {28.5715f,28.5714f} -}; - -static const SFG_StrokeStrip ch42st[] = -{ - {2,ch42st0}, - {2,ch42st1}, - {2,ch42st2} -}; - -static const SFG_StrokeChar ch42 = {104.762f,3,ch42st}; - -/* char: 0x2b */ - -static const SFG_StrokeVertex ch43st0[] = -{ - {52.3809f,85.7143f}, - {52.3809f,0.0f} -}; - -static const SFG_StrokeVertex ch43st1[] = -{ - {9.5238f,42.8571f}, - {95.2381f,42.8571f} -}; - -static const SFG_StrokeStrip ch43st[] = -{ - {2,ch43st0}, - {2,ch43st1} -}; - -static const SFG_StrokeChar ch43 = {104.762f,2,ch43st}; - -/* char: 0x2c */ - -static const SFG_StrokeVertex ch44st0[] = -{ - {57.1429f,4.7619f}, - {52.381f,0.0f}, - {47.6191f,4.7619f}, - {52.381f,9.5238f}, - {57.1429f,4.7619f}, - {57.1429f,-4.7619f}, - {52.381f,-14.2857f}, - {47.6191f,-19.0476f} -}; - -static const SFG_StrokeStrip ch44st[] = -{ - {8,ch44st0} -}; - -static const SFG_StrokeChar ch44 = {104.762f,1,ch44st}; - -/* char: 0x2d */ - -static const SFG_StrokeVertex ch45st0[] = -{ - {9.5238f,42.8571f}, - {95.2381f,42.8571f} -}; - -static const SFG_StrokeStrip ch45st[] = -{ - {2,ch45st0} -}; - -static const SFG_StrokeChar ch45 = {104.762f,1,ch45st}; - -/* char: 0x2e */ - -static const SFG_StrokeVertex ch46st0[] = -{ - {52.381f,9.5238f}, - {47.6191f,4.7619f}, - {52.381f,0.0f}, - {57.1429f,4.7619f}, - {52.381f,9.5238f} -}; - -static const SFG_StrokeStrip ch46st[] = -{ - {5,ch46st0} -}; - -static const SFG_StrokeChar ch46 = {104.762f,1,ch46st}; - -/* char: 0x2f */ - -static const SFG_StrokeVertex ch47st0[] = -{ - {19.0476f,-14.2857f}, - {85.7143f,100.0f} -}; - -static const SFG_StrokeStrip ch47st[] = -{ - {2,ch47st0} -}; - -static const SFG_StrokeChar ch47 = {104.762f,1,ch47st}; - -/* char: 0x30 */ - -static const SFG_StrokeVertex ch48st0[] = -{ - {47.619f,100.0f}, - {33.3333f,95.2381f}, - {23.8095f,80.9524f}, - {19.0476f,57.1429f}, - {19.0476f,42.8571f}, - {23.8095f,19.0476f}, - {33.3333f,4.7619f}, - {47.619f,0.0f}, - {57.1428f,0.0f}, - {71.4286f,4.7619f}, - {80.9524f,19.0476f}, - {85.7143f,42.8571f}, - {85.7143f,57.1429f}, - {80.9524f,80.9524f}, - {71.4286f,95.2381f}, - {57.1428f,100.0f}, - {47.619f,100.0f} -}; - -static const SFG_StrokeStrip ch48st[] = -{ - {17,ch48st0} -}; - -static const SFG_StrokeChar ch48 = {104.762f,1,ch48st}; - -/* char: 0x31 */ - -static const SFG_StrokeVertex ch49st0[] = -{ - {40.4762f,80.9524f}, - {50.0f,85.7143f}, - {64.2857f,100.0f}, - {64.2857f,0.0f} -}; - -static const SFG_StrokeStrip ch49st[] = -{ - {4,ch49st0} -}; - -static const SFG_StrokeChar ch49 = {104.762f,1,ch49st}; - -/* char: 0x32 */ - -static const SFG_StrokeVertex ch50st0[] = -{ - {23.8095f,76.1905f}, - {23.8095f,80.9524f}, - {28.5714f,90.4762f}, - {33.3333f,95.2381f}, - {42.8571f,100.0f}, - {61.9047f,100.0f}, - {71.4286f,95.2381f}, - {76.1905f,90.4762f}, - {80.9524f,80.9524f}, - {80.9524f,71.4286f}, - {76.1905f,61.9048f}, - {66.6666f,47.619f}, - {19.0476f,0.0f}, - {85.7143f,0.0f} -}; - -static const SFG_StrokeStrip ch50st[] = -{ - {14,ch50st0} -}; - -static const SFG_StrokeChar ch50 = {104.762f,1,ch50st}; - -/* char: 0x33 */ - -static const SFG_StrokeVertex ch51st0[] = -{ - {28.5714f,100.0f}, - {80.9524f,100.0f}, - {52.3809f,61.9048f}, - {66.6666f,61.9048f}, - {76.1905f,57.1429f}, - {80.9524f,52.381f}, - {85.7143f,38.0952f}, - {85.7143f,28.5714f}, - {80.9524f,14.2857f}, - {71.4286f,4.7619f}, - {57.1428f,0.0f}, - {42.8571f,0.0f}, - {28.5714f,4.7619f}, - {23.8095f,9.5238f}, - {19.0476f,19.0476f} -}; - -static const SFG_StrokeStrip ch51st[] = -{ - {15,ch51st0} -}; - -static const SFG_StrokeChar ch51 = {104.762f,1,ch51st}; - -/* char: 0x34 */ - -static const SFG_StrokeVertex ch52st0[] = -{ - {64.2857f,100.0f}, - {16.6667f,33.3333f}, - {88.0952f,33.3333f} -}; - -static const SFG_StrokeVertex ch52st1[] = -{ - {64.2857f,100.0f}, - {64.2857f,0.0f} -}; - -static const SFG_StrokeStrip ch52st[] = -{ - {3,ch52st0}, - {2,ch52st1} -}; - -static const SFG_StrokeChar ch52 = {104.762f,2,ch52st}; - -/* char: 0x35 */ - -static const SFG_StrokeVertex ch53st0[] = -{ - {76.1905f,100.0f}, - {28.5714f,100.0f}, - {23.8095f,57.1429f}, - {28.5714f,61.9048f}, - {42.8571f,66.6667f}, - {57.1428f,66.6667f}, - {71.4286f,61.9048f}, - {80.9524f,52.381f}, - {85.7143f,38.0952f}, - {85.7143f,28.5714f}, - {80.9524f,14.2857f}, - {71.4286f,4.7619f}, - {57.1428f,0.0f}, - {42.8571f,0.0f}, - {28.5714f,4.7619f}, - {23.8095f,9.5238f}, - {19.0476f,19.0476f} -}; - -static const SFG_StrokeStrip ch53st[] = -{ - {17,ch53st0} -}; - -static const SFG_StrokeChar ch53 = {104.762f,1,ch53st}; - -/* char: 0x36 */ - -static const SFG_StrokeVertex ch54st0[] = -{ - {78.5714f,85.7143f}, - {73.8096f,95.2381f}, - {59.5238f,100.0f}, - {50.0f,100.0f}, - {35.7143f,95.2381f}, - {26.1905f,80.9524f}, - {21.4286f,57.1429f}, - {21.4286f,33.3333f}, - {26.1905f,14.2857f}, - {35.7143f,4.7619f}, - {50.0f,0.0f}, - {54.7619f,0.0f}, - {69.0476f,4.7619f}, - {78.5714f,14.2857f}, - {83.3334f,28.5714f}, - {83.3334f,33.3333f}, - {78.5714f,47.619f}, - {69.0476f,57.1429f}, - {54.7619f,61.9048f}, - {50.0f,61.9048f}, - {35.7143f,57.1429f}, - {26.1905f,47.619f}, - {21.4286f,33.3333f} -}; - -static const SFG_StrokeStrip ch54st[] = -{ - {23,ch54st0} -}; - -static const SFG_StrokeChar ch54 = {104.762f,1,ch54st}; - -/* char: 0x37 */ - -static const SFG_StrokeVertex ch55st0[] = -{ - {85.7143f,100.0f}, - {38.0952f,0.0f} -}; - -static const SFG_StrokeVertex ch55st1[] = -{ - {19.0476f,100.0f}, - {85.7143f,100.0f} -}; - -static const SFG_StrokeStrip ch55st[] = -{ - {2,ch55st0}, - {2,ch55st1} -}; - -static const SFG_StrokeChar ch55 = {104.762f,2,ch55st}; - -/* char: 0x38 */ - -static const SFG_StrokeVertex ch56st0[] = -{ - {42.8571f,100.0f}, - {28.5714f,95.2381f}, - {23.8095f,85.7143f}, - {23.8095f,76.1905f}, - {28.5714f,66.6667f}, - {38.0952f,61.9048f}, - {57.1428f,57.1429f}, - {71.4286f,52.381f}, - {80.9524f,42.8571f}, - {85.7143f,33.3333f}, - {85.7143f,19.0476f}, - {80.9524f,9.5238f}, - {76.1905f,4.7619f}, - {61.9047f,0.0f}, - {42.8571f,0.0f}, - {28.5714f,4.7619f}, - {23.8095f,9.5238f}, - {19.0476f,19.0476f}, - {19.0476f,33.3333f}, - {23.8095f,42.8571f}, - {33.3333f,52.381f}, - {47.619f,57.1429f}, - {66.6666f,61.9048f}, - {76.1905f,66.6667f}, - {80.9524f,76.1905f}, - {80.9524f,85.7143f}, - {76.1905f,95.2381f}, - {61.9047f,100.0f}, - {42.8571f,100.0f} -}; - -static const SFG_StrokeStrip ch56st[] = -{ - {29,ch56st0} -}; - -static const SFG_StrokeChar ch56 = {104.762f,1,ch56st}; - -/* char: 0x39 */ - -static const SFG_StrokeVertex ch57st0[] = -{ - {83.3334f,66.6667f}, - {78.5714f,52.381f}, - {69.0476f,42.8571f}, - {54.7619f,38.0952f}, - {50.0f,38.0952f}, - {35.7143f,42.8571f}, - {26.1905f,52.381f}, - {21.4286f,66.6667f}, - {21.4286f,71.4286f}, - {26.1905f,85.7143f}, - {35.7143f,95.2381f}, - {50.0f,100.0f}, - {54.7619f,100.0f}, - {69.0476f,95.2381f}, - {78.5714f,85.7143f}, - {83.3334f,66.6667f}, - {83.3334f,42.8571f}, - {78.5714f,19.0476f}, - {69.0476f,4.7619f}, - {54.7619f,0.0f}, - {45.2381f,0.0f}, - {30.9524f,4.7619f}, - {26.1905f,14.2857f} -}; - -static const SFG_StrokeStrip ch57st[] = -{ - {23,ch57st0} -}; - -static const SFG_StrokeChar ch57 = {104.762f,1,ch57st}; - -/* char: 0x3a */ - -static const SFG_StrokeVertex ch58st0[] = -{ - {52.381f,66.6667f}, - {47.6191f,61.9048f}, - {52.381f,57.1429f}, - {57.1429f,61.9048f}, - {52.381f,66.6667f} -}; - -static const SFG_StrokeVertex ch58st1[] = -{ - {52.381f,9.5238f}, - {47.6191f,4.7619f}, - {52.381f,0.0f}, - {57.1429f,4.7619f}, - {52.381f,9.5238f} -}; - -static const SFG_StrokeStrip ch58st[] = -{ - {5,ch58st0}, - {5,ch58st1} -}; - -static const SFG_StrokeChar ch58 = {104.762f,2,ch58st}; - -/* char: 0x3b */ - -static const SFG_StrokeVertex ch59st0[] = -{ - {52.381f,66.6667f}, - {47.6191f,61.9048f}, - {52.381f,57.1429f}, - {57.1429f,61.9048f}, - {52.381f,66.6667f} -}; - -static const SFG_StrokeVertex ch59st1[] = -{ - {57.1429f,4.7619f}, - {52.381f,0.0f}, - {47.6191f,4.7619f}, - {52.381f,9.5238f}, - {57.1429f,4.7619f}, - {57.1429f,-4.7619f}, - {52.381f,-14.2857f}, - {47.6191f,-19.0476f} -}; - -static const SFG_StrokeStrip ch59st[] = -{ - {5,ch59st0}, - {8,ch59st1} -}; - -static const SFG_StrokeChar ch59 = {104.762f,2,ch59st}; - -/* char: 0x3c */ - -static const SFG_StrokeVertex ch60st0[] = -{ - {90.4762f,85.7143f}, - {14.2857f,42.8571f}, - {90.4762f,0.0f} -}; - -static const SFG_StrokeStrip ch60st[] = -{ - {3,ch60st0} -}; - -static const SFG_StrokeChar ch60 = {104.762f,1,ch60st}; - -/* char: 0x3d */ - -static const SFG_StrokeVertex ch61st0[] = -{ - {9.5238f,57.1429f}, - {95.2381f,57.1429f} -}; - -static const SFG_StrokeVertex ch61st1[] = -{ - {9.5238f,28.5714f}, - {95.2381f,28.5714f} -}; - -static const SFG_StrokeStrip ch61st[] = -{ - {2,ch61st0}, - {2,ch61st1} -}; - -static const SFG_StrokeChar ch61 = {104.762f,2,ch61st}; - -/* char: 0x3e */ - -static const SFG_StrokeVertex ch62st0[] = -{ - {14.2857f,85.7143f}, - {90.4762f,42.8571f}, - {14.2857f,0.0f} -}; - -static const SFG_StrokeStrip ch62st[] = -{ - {3,ch62st0} -}; - -static const SFG_StrokeChar ch62 = {104.762f,1,ch62st}; - -/* char: 0x3f */ - -static const SFG_StrokeVertex ch63st0[] = -{ - {23.8095f,76.1905f}, - {23.8095f,80.9524f}, - {28.5714f,90.4762f}, - {33.3333f,95.2381f}, - {42.8571f,100.0f}, - {61.9047f,100.0f}, - {71.4285f,95.2381f}, - {76.1905f,90.4762f}, - {80.9524f,80.9524f}, - {80.9524f,71.4286f}, - {76.1905f,61.9048f}, - {71.4285f,57.1429f}, - {52.3809f,47.619f}, - {52.3809f,33.3333f} -}; - -static const SFG_StrokeVertex ch63st1[] = -{ - {52.3809f,9.5238f}, - {47.619f,4.7619f}, - {52.3809f,0.0f}, - {57.1428f,4.7619f}, - {52.3809f,9.5238f} -}; - -static const SFG_StrokeStrip ch63st[] = -{ - {14,ch63st0}, - {5,ch63st1} -}; - -static const SFG_StrokeChar ch63 = {104.762f,2,ch63st}; - -/* char: 0x40 */ - -static const SFG_StrokeVertex ch64st0[] = -{ - {64.2857f,52.381f}, - {54.7619f,57.1429f}, - {45.2381f,57.1429f}, - {40.4762f,47.619f}, - {40.4762f,42.8571f}, - {45.2381f,33.3333f}, - {54.7619f,33.3333f}, - {64.2857f,38.0952f} -}; - -static const SFG_StrokeVertex ch64st1[] = -{ - {64.2857f,57.1429f}, - {64.2857f,38.0952f}, - {69.0476f,33.3333f}, - {78.5714f,33.3333f}, - {83.3334f,42.8571f}, - {83.3334f,47.619f}, - {78.5714f,61.9048f}, - {69.0476f,71.4286f}, - {54.7619f,76.1905f}, - {50.0f,76.1905f}, - {35.7143f,71.4286f}, - {26.1905f,61.9048f}, - {21.4286f,47.619f}, - {21.4286f,42.8571f}, - {26.1905f,28.5714f}, - {35.7143f,19.0476f}, - {50.0f,14.2857f}, - {54.7619f,14.2857f}, - {69.0476f,19.0476f} -}; - -static const SFG_StrokeStrip ch64st[] = -{ - {8,ch64st0}, - {19,ch64st1} -}; - -static const SFG_StrokeChar ch64 = {104.762f,2,ch64st}; - -/* char: 0x41 */ - -static const SFG_StrokeVertex ch65st0[] = -{ - {52.3809f,100.0f}, - {14.2857f,0.0f} -}; - -static const SFG_StrokeVertex ch65st1[] = -{ - {52.3809f,100.0f}, - {90.4762f,0.0f} -}; - -static const SFG_StrokeVertex ch65st2[] = -{ - {28.5714f,33.3333f}, - {76.1905f,33.3333f} -}; - -static const SFG_StrokeStrip ch65st[] = -{ - {2,ch65st0}, - {2,ch65st1}, - {2,ch65st2} -}; - -static const SFG_StrokeChar ch65 = {104.762f,3,ch65st}; - -/* char: 0x42 */ - -static const SFG_StrokeVertex ch66st0[] = -{ - {19.0476f,100.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeVertex ch66st1[] = -{ - {19.0476f,100.0f}, - {61.9047f,100.0f}, - {76.1905f,95.2381f}, - {80.9524f,90.4762f}, - {85.7143f,80.9524f}, - {85.7143f,71.4286f}, - {80.9524f,61.9048f}, - {76.1905f,57.1429f}, - {61.9047f,52.381f} -}; - -static const SFG_StrokeVertex ch66st2[] = -{ - {19.0476f,52.381f}, - {61.9047f,52.381f}, - {76.1905f,47.619f}, - {80.9524f,42.8571f}, - {85.7143f,33.3333f}, - {85.7143f,19.0476f}, - {80.9524f,9.5238f}, - {76.1905f,4.7619f}, - {61.9047f,0.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeStrip ch66st[] = -{ - {2,ch66st0}, - {9,ch66st1}, - {10,ch66st2} -}; - -static const SFG_StrokeChar ch66 = {104.762f,3,ch66st}; - -/* char: 0x43 */ - -static const SFG_StrokeVertex ch67st0[] = -{ - {88.0952f,76.1905f}, - {83.3334f,85.7143f}, - {73.8096f,95.2381f}, - {64.2857f,100.0f}, - {45.2381f,100.0f}, - {35.7143f,95.2381f}, - {26.1905f,85.7143f}, - {21.4286f,76.1905f}, - {16.6667f,61.9048f}, - {16.6667f,38.0952f}, - {21.4286f,23.8095f}, - {26.1905f,14.2857f}, - {35.7143f,4.7619f}, - {45.2381f,0.0f}, - {64.2857f,0.0f}, - {73.8096f,4.7619f}, - {83.3334f,14.2857f}, - {88.0952f,23.8095f} -}; - -static const SFG_StrokeStrip ch67st[] = -{ - {18,ch67st0} -}; - -static const SFG_StrokeChar ch67 = {104.762f,1,ch67st}; - -/* char: 0x44 */ - -static const SFG_StrokeVertex ch68st0[] = -{ - {19.0476f,100.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeVertex ch68st1[] = -{ - {19.0476f,100.0f}, - {52.3809f,100.0f}, - {66.6666f,95.2381f}, - {76.1905f,85.7143f}, - {80.9524f,76.1905f}, - {85.7143f,61.9048f}, - {85.7143f,38.0952f}, - {80.9524f,23.8095f}, - {76.1905f,14.2857f}, - {66.6666f,4.7619f}, - {52.3809f,0.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeStrip ch68st[] = -{ - {2,ch68st0}, - {12,ch68st1} -}; - -static const SFG_StrokeChar ch68 = {104.762f,2,ch68st}; - -/* char: 0x45 */ - -static const SFG_StrokeVertex ch69st0[] = -{ - {21.4286f,100.0f}, - {21.4286f,0.0f} -}; - -static const SFG_StrokeVertex ch69st1[] = -{ - {21.4286f,100.0f}, - {83.3334f,100.0f} -}; - -static const SFG_StrokeVertex ch69st2[] = -{ - {21.4286f,52.381f}, - {59.5238f,52.381f} -}; - -static const SFG_StrokeVertex ch69st3[] = -{ - {21.4286f,0.0f}, - {83.3334f,0.0f} -}; - -static const SFG_StrokeStrip ch69st[] = -{ - {2,ch69st0}, - {2,ch69st1}, - {2,ch69st2}, - {2,ch69st3} -}; - -static const SFG_StrokeChar ch69 = {104.762f,4,ch69st}; - -/* char: 0x46 */ - -static const SFG_StrokeVertex ch70st0[] = -{ - {21.4286f,100.0f}, - {21.4286f,0.0f} -}; - -static const SFG_StrokeVertex ch70st1[] = -{ - {21.4286f,100.0f}, - {83.3334f,100.0f} -}; - -static const SFG_StrokeVertex ch70st2[] = -{ - {21.4286f,52.381f}, - {59.5238f,52.381f} -}; - -static const SFG_StrokeStrip ch70st[] = -{ - {2,ch70st0}, - {2,ch70st1}, - {2,ch70st2} -}; - -static const SFG_StrokeChar ch70 = {104.762f,3,ch70st}; - -/* char: 0x47 */ - -static const SFG_StrokeVertex ch71st0[] = -{ - {88.0952f,76.1905f}, - {83.3334f,85.7143f}, - {73.8096f,95.2381f}, - {64.2857f,100.0f}, - {45.2381f,100.0f}, - {35.7143f,95.2381f}, - {26.1905f,85.7143f}, - {21.4286f,76.1905f}, - {16.6667f,61.9048f}, - {16.6667f,38.0952f}, - {21.4286f,23.8095f}, - {26.1905f,14.2857f}, - {35.7143f,4.7619f}, - {45.2381f,0.0f}, - {64.2857f,0.0f}, - {73.8096f,4.7619f}, - {83.3334f,14.2857f}, - {88.0952f,23.8095f}, - {88.0952f,38.0952f} -}; - -static const SFG_StrokeVertex ch71st1[] = -{ - {64.2857f,38.0952f}, - {88.0952f,38.0952f} -}; - -static const SFG_StrokeStrip ch71st[] = -{ - {19,ch71st0}, - {2,ch71st1} -}; - -static const SFG_StrokeChar ch71 = {104.762f,2,ch71st}; - -/* char: 0x48 */ - -static const SFG_StrokeVertex ch72st0[] = -{ - {19.0476f,100.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeVertex ch72st1[] = -{ - {85.7143f,100.0f}, - {85.7143f,0.0f} -}; - -static const SFG_StrokeVertex ch72st2[] = -{ - {19.0476f,52.381f}, - {85.7143f,52.381f} -}; - -static const SFG_StrokeStrip ch72st[] = -{ - {2,ch72st0}, - {2,ch72st1}, - {2,ch72st2} -}; - -static const SFG_StrokeChar ch72 = {104.762f,3,ch72st}; - -/* char: 0x49 */ - -static const SFG_StrokeVertex ch73st0[] = -{ - {52.381f,100.0f}, - {52.381f,0.0f} -}; - -static const SFG_StrokeStrip ch73st[] = -{ - {2,ch73st0} -}; - -static const SFG_StrokeChar ch73 = {104.762f,1,ch73st}; - -/* char: 0x4a */ - -static const SFG_StrokeVertex ch74st0[] = -{ - {76.1905f,100.0f}, - {76.1905f,23.8095f}, - {71.4286f,9.5238f}, - {66.6667f,4.7619f}, - {57.1429f,0.0f}, - {47.6191f,0.0f}, - {38.0953f,4.7619f}, - {33.3334f,9.5238f}, - {28.5715f,23.8095f}, - {28.5715f,33.3333f} -}; - -static const SFG_StrokeStrip ch74st[] = -{ - {10,ch74st0} -}; - -static const SFG_StrokeChar ch74 = {104.762f,1,ch74st}; - -/* char: 0x4b */ - -static const SFG_StrokeVertex ch75st0[] = -{ - {19.0476f,100.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeVertex ch75st1[] = -{ - {85.7143f,100.0f}, - {19.0476f,33.3333f} -}; - -static const SFG_StrokeVertex ch75st2[] = -{ - {42.8571f,57.1429f}, - {85.7143f,0.0f} -}; - -static const SFG_StrokeStrip ch75st[] = -{ - {2,ch75st0}, - {2,ch75st1}, - {2,ch75st2} -}; - -static const SFG_StrokeChar ch75 = {104.762f,3,ch75st}; - -/* char: 0x4c */ - -static const SFG_StrokeVertex ch76st0[] = -{ - {23.8095f,100.0f}, - {23.8095f,0.0f} -}; - -static const SFG_StrokeVertex ch76st1[] = -{ - {23.8095f,0.0f}, - {80.9524f,0.0f} -}; - -static const SFG_StrokeStrip ch76st[] = -{ - {2,ch76st0}, - {2,ch76st1} -}; - -static const SFG_StrokeChar ch76 = {104.762f,2,ch76st}; - -/* char: 0x4d */ - -static const SFG_StrokeVertex ch77st0[] = -{ - {14.2857f,100.0f}, - {14.2857f,0.0f} -}; - -static const SFG_StrokeVertex ch77st1[] = -{ - {14.2857f,100.0f}, - {52.3809f,0.0f} -}; - -static const SFG_StrokeVertex ch77st2[] = -{ - {90.4762f,100.0f}, - {52.3809f,0.0f} -}; - -static const SFG_StrokeVertex ch77st3[] = -{ - {90.4762f,100.0f}, - {90.4762f,0.0f} -}; - -static const SFG_StrokeStrip ch77st[] = -{ - {2,ch77st0}, - {2,ch77st1}, - {2,ch77st2}, - {2,ch77st3} -}; - -static const SFG_StrokeChar ch77 = {104.762f,4,ch77st}; - -/* char: 0x4e */ - -static const SFG_StrokeVertex ch78st0[] = -{ - {19.0476f,100.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeVertex ch78st1[] = -{ - {19.0476f,100.0f}, - {85.7143f,0.0f} -}; - -static const SFG_StrokeVertex ch78st2[] = -{ - {85.7143f,100.0f}, - {85.7143f,0.0f} -}; - -static const SFG_StrokeStrip ch78st[] = -{ - {2,ch78st0}, - {2,ch78st1}, - {2,ch78st2} -}; - -static const SFG_StrokeChar ch78 = {104.762f,3,ch78st}; - -/* char: 0x4f */ - -static const SFG_StrokeVertex ch79st0[] = -{ - {42.8571f,100.0f}, - {33.3333f,95.2381f}, - {23.8095f,85.7143f}, - {19.0476f,76.1905f}, - {14.2857f,61.9048f}, - {14.2857f,38.0952f}, - {19.0476f,23.8095f}, - {23.8095f,14.2857f}, - {33.3333f,4.7619f}, - {42.8571f,0.0f}, - {61.9047f,0.0f}, - {71.4286f,4.7619f}, - {80.9524f,14.2857f}, - {85.7143f,23.8095f}, - {90.4762f,38.0952f}, - {90.4762f,61.9048f}, - {85.7143f,76.1905f}, - {80.9524f,85.7143f}, - {71.4286f,95.2381f}, - {61.9047f,100.0f}, - {42.8571f,100.0f} -}; - -static const SFG_StrokeStrip ch79st[] = -{ - {21,ch79st0} -}; - -static const SFG_StrokeChar ch79 = {104.762f,1,ch79st}; - -/* char: 0x50 */ - -static const SFG_StrokeVertex ch80st0[] = -{ - {19.0476f,100.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeVertex ch80st1[] = -{ - {19.0476f,100.0f}, - {61.9047f,100.0f}, - {76.1905f,95.2381f}, - {80.9524f,90.4762f}, - {85.7143f,80.9524f}, - {85.7143f,66.6667f}, - {80.9524f,57.1429f}, - {76.1905f,52.381f}, - {61.9047f,47.619f}, - {19.0476f,47.619f} -}; - -static const SFG_StrokeStrip ch80st[] = -{ - {2,ch80st0}, - {10,ch80st1} -}; - -static const SFG_StrokeChar ch80 = {104.762f,2,ch80st}; - -/* char: 0x51 */ - -static const SFG_StrokeVertex ch81st0[] = -{ - {42.8571f,100.0f}, - {33.3333f,95.2381f}, - {23.8095f,85.7143f}, - {19.0476f,76.1905f}, - {14.2857f,61.9048f}, - {14.2857f,38.0952f}, - {19.0476f,23.8095f}, - {23.8095f,14.2857f}, - {33.3333f,4.7619f}, - {42.8571f,0.0f}, - {61.9047f,0.0f}, - {71.4286f,4.7619f}, - {80.9524f,14.2857f}, - {85.7143f,23.8095f}, - {90.4762f,38.0952f}, - {90.4762f,61.9048f}, - {85.7143f,76.1905f}, - {80.9524f,85.7143f}, - {71.4286f,95.2381f}, - {61.9047f,100.0f}, - {42.8571f,100.0f} -}; - -static const SFG_StrokeVertex ch81st1[] = -{ - {57.1428f,19.0476f}, - {85.7143f,-9.5238f} -}; - -static const SFG_StrokeStrip ch81st[] = -{ - {21,ch81st0}, - {2,ch81st1} -}; - -static const SFG_StrokeChar ch81 = {104.762f,2,ch81st}; - -/* char: 0x52 */ - -static const SFG_StrokeVertex ch82st0[] = -{ - {19.0476f,100.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeVertex ch82st1[] = -{ - {19.0476f,100.0f}, - {61.9047f,100.0f}, - {76.1905f,95.2381f}, - {80.9524f,90.4762f}, - {85.7143f,80.9524f}, - {85.7143f,71.4286f}, - {80.9524f,61.9048f}, - {76.1905f,57.1429f}, - {61.9047f,52.381f}, - {19.0476f,52.381f} -}; - -static const SFG_StrokeVertex ch82st2[] = -{ - {52.3809f,52.381f}, - {85.7143f,0.0f} -}; - -static const SFG_StrokeStrip ch82st[] = -{ - {2,ch82st0}, - {10,ch82st1}, - {2,ch82st2} -}; - -static const SFG_StrokeChar ch82 = {104.762f,3,ch82st}; - -/* char: 0x53 */ - -static const SFG_StrokeVertex ch83st0[] = -{ - {85.7143f,85.7143f}, - {76.1905f,95.2381f}, - {61.9047f,100.0f}, - {42.8571f,100.0f}, - {28.5714f,95.2381f}, - {19.0476f,85.7143f}, - {19.0476f,76.1905f}, - {23.8095f,66.6667f}, - {28.5714f,61.9048f}, - {38.0952f,57.1429f}, - {66.6666f,47.619f}, - {76.1905f,42.8571f}, - {80.9524f,38.0952f}, - {85.7143f,28.5714f}, - {85.7143f,14.2857f}, - {76.1905f,4.7619f}, - {61.9047f,0.0f}, - {42.8571f,0.0f}, - {28.5714f,4.7619f}, - {19.0476f,14.2857f} -}; - -static const SFG_StrokeStrip ch83st[] = -{ - {20,ch83st0} -}; - -static const SFG_StrokeChar ch83 = {104.762f,1,ch83st}; - -/* char: 0x54 */ - -static const SFG_StrokeVertex ch84st0[] = -{ - {52.3809f,100.0f}, - {52.3809f,0.0f} -}; - -static const SFG_StrokeVertex ch84st1[] = -{ - {19.0476f,100.0f}, - {85.7143f,100.0f} -}; - -static const SFG_StrokeStrip ch84st[] = -{ - {2,ch84st0}, - {2,ch84st1} -}; - -static const SFG_StrokeChar ch84 = {104.762f,2,ch84st}; - -/* char: 0x55 */ - -static const SFG_StrokeVertex ch85st0[] = -{ - {19.0476f,100.0f}, - {19.0476f,28.5714f}, - {23.8095f,14.2857f}, - {33.3333f,4.7619f}, - {47.619f,0.0f}, - {57.1428f,0.0f}, - {71.4286f,4.7619f}, - {80.9524f,14.2857f}, - {85.7143f,28.5714f}, - {85.7143f,100.0f} -}; - -static const SFG_StrokeStrip ch85st[] = -{ - {10,ch85st0} -}; - -static const SFG_StrokeChar ch85 = {104.762f,1,ch85st}; - -/* char: 0x56 */ - -static const SFG_StrokeVertex ch86st0[] = -{ - {14.2857f,100.0f}, - {52.3809f,0.0f} -}; - -static const SFG_StrokeVertex ch86st1[] = -{ - {90.4762f,100.0f}, - {52.3809f,0.0f} -}; - -static const SFG_StrokeStrip ch86st[] = -{ - {2,ch86st0}, - {2,ch86st1} -}; - -static const SFG_StrokeChar ch86 = {104.762f,2,ch86st}; - -/* char: 0x57 */ - -static const SFG_StrokeVertex ch87st0[] = -{ - {4.7619f,100.0f}, - {28.5714f,0.0f} -}; - -static const SFG_StrokeVertex ch87st1[] = -{ - {52.3809f,100.0f}, - {28.5714f,0.0f} -}; - -static const SFG_StrokeVertex ch87st2[] = -{ - {52.3809f,100.0f}, - {76.1905f,0.0f} -}; - -static const SFG_StrokeVertex ch87st3[] = -{ - {100.0f,100.0f}, - {76.1905f,0.0f} -}; - -static const SFG_StrokeStrip ch87st[] = -{ - {2,ch87st0}, - {2,ch87st1}, - {2,ch87st2}, - {2,ch87st3} -}; - -static const SFG_StrokeChar ch87 = {104.762f,4,ch87st}; - -/* char: 0x58 */ - -static const SFG_StrokeVertex ch88st0[] = -{ - {19.0476f,100.0f}, - {85.7143f,0.0f} -}; - -static const SFG_StrokeVertex ch88st1[] = -{ - {85.7143f,100.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeStrip ch88st[] = -{ - {2,ch88st0}, - {2,ch88st1} -}; - -static const SFG_StrokeChar ch88 = {104.762f,2,ch88st}; - -/* char: 0x59 */ - -static const SFG_StrokeVertex ch89st0[] = -{ - {14.2857f,100.0f}, - {52.3809f,52.381f}, - {52.3809f,0.0f} -}; - -static const SFG_StrokeVertex ch89st1[] = -{ - {90.4762f,100.0f}, - {52.3809f,52.381f} -}; - -static const SFG_StrokeStrip ch89st[] = -{ - {3,ch89st0}, - {2,ch89st1} -}; - -static const SFG_StrokeChar ch89 = {104.762f,2,ch89st}; - -/* char: 0x5a */ - -static const SFG_StrokeVertex ch90st0[] = -{ - {85.7143f,100.0f}, - {19.0476f,0.0f} -}; - -static const SFG_StrokeVertex ch90st1[] = -{ - {19.0476f,100.0f}, - {85.7143f,100.0f} -}; - -static const SFG_StrokeVertex ch90st2[] = -{ - {19.0476f,0.0f}, - {85.7143f,0.0f} -}; - -static const SFG_StrokeStrip ch90st[] = -{ - {2,ch90st0}, - {2,ch90st1}, - {2,ch90st2} -}; - -static const SFG_StrokeChar ch90 = {104.762f,3,ch90st}; - -/* char: 0x5b */ - -static const SFG_StrokeVertex ch91st0[] = -{ - {35.7143f,119.048f}, - {35.7143f,-33.3333f} -}; - -static const SFG_StrokeVertex ch91st1[] = -{ - {40.4762f,119.048f}, - {40.4762f,-33.3333f} -}; - -static const SFG_StrokeVertex ch91st2[] = -{ - {35.7143f,119.048f}, - {69.0476f,119.048f} -}; - -static const SFG_StrokeVertex ch91st3[] = -{ - {35.7143f,-33.3333f}, - {69.0476f,-33.3333f} -}; - -static const SFG_StrokeStrip ch91st[] = -{ - {2,ch91st0}, - {2,ch91st1}, - {2,ch91st2}, - {2,ch91st3} -}; - -static const SFG_StrokeChar ch91 = {104.762f,4,ch91st}; - -/* char: 0x5c */ - -static const SFG_StrokeVertex ch92st0[] = -{ - {19.0476f,100.0f}, - {85.7143f,-14.2857f} -}; - -static const SFG_StrokeStrip ch92st[] = -{ - {2,ch92st0} -}; - -static const SFG_StrokeChar ch92 = {104.762f,1,ch92st}; - -/* char: 0x5d */ - -static const SFG_StrokeVertex ch93st0[] = -{ - {64.2857f,119.048f}, - {64.2857f,-33.3333f} -}; - -static const SFG_StrokeVertex ch93st1[] = -{ - {69.0476f,119.048f}, - {69.0476f,-33.3333f} -}; - -static const SFG_StrokeVertex ch93st2[] = -{ - {35.7143f,119.048f}, - {69.0476f,119.048f} -}; - -static const SFG_StrokeVertex ch93st3[] = -{ - {35.7143f,-33.3333f}, - {69.0476f,-33.3333f} -}; - -static const SFG_StrokeStrip ch93st[] = -{ - {2,ch93st0}, - {2,ch93st1}, - {2,ch93st2}, - {2,ch93st3} -}; - -static const SFG_StrokeChar ch93 = {104.762f,4,ch93st}; - -/* char: 0x5e */ - -static const SFG_StrokeVertex ch94st0[] = -{ - {52.3809f,109.524f}, - {14.2857f,42.8571f} -}; - -static const SFG_StrokeVertex ch94st1[] = -{ - {52.3809f,109.524f}, - {90.4762f,42.8571f} -}; - -static const SFG_StrokeStrip ch94st[] = -{ - {2,ch94st0}, - {2,ch94st1} -}; - -static const SFG_StrokeChar ch94 = {104.762f,2,ch94st}; - -/* char: 0x5f */ - -static const SFG_StrokeVertex ch95st0[] = -{ - {0,-33.3333f}, - {104.762f,-33.3333f}, - {104.762f,-28.5714f}, - {0,-28.5714f}, - {0,-33.3333f} -}; - -static const SFG_StrokeStrip ch95st[] = -{ - {5,ch95st0} -}; - -static const SFG_StrokeChar ch95 = {104.762f,1,ch95st}; - -/* char: 0x60 */ - -static const SFG_StrokeVertex ch96st0[] = -{ - {42.8572f,100.0f}, - {66.6667f,71.4286f} -}; - -static const SFG_StrokeVertex ch96st1[] = -{ - {42.8572f,100.0f}, - {38.0953f,95.2381f}, - {66.6667f,71.4286f} -}; - -static const SFG_StrokeStrip ch96st[] = -{ - {2,ch96st0}, - {3,ch96st1} -}; - -static const SFG_StrokeChar ch96 = {104.762f,2,ch96st}; - -/* char: 0x61 */ - -static const SFG_StrokeVertex ch97st0[] = -{ - {80.9524f,66.6667f}, - {80.9524f,0.0f} -}; - -static const SFG_StrokeVertex ch97st1[] = -{ - {80.9524f,52.381f}, - {71.4285f,61.9048f}, - {61.9047f,66.6667f}, - {47.619f,66.6667f}, - {38.0952f,61.9048f}, - {28.5714f,52.381f}, - {23.8095f,38.0952f}, - {23.8095f,28.5714f}, - {28.5714f,14.2857f}, - {38.0952f,4.7619f}, - {47.619f,0.0f}, - {61.9047f,0.0f}, - {71.4285f,4.7619f}, - {80.9524f,14.2857f} -}; - -static const SFG_StrokeStrip ch97st[] = -{ - {2,ch97st0}, - {14,ch97st1} -}; - -static const SFG_StrokeChar ch97 = {104.762f,2,ch97st}; - -/* char: 0x62 */ - -static const SFG_StrokeVertex ch98st0[] = -{ - {23.8095f,100.0f}, - {23.8095f,0.0f} -}; - -static const SFG_StrokeVertex ch98st1[] = -{ - {23.8095f,52.381f}, - {33.3333f,61.9048f}, - {42.8571f,66.6667f}, - {57.1428f,66.6667f}, - {66.6666f,61.9048f}, - {76.1905f,52.381f}, - {80.9524f,38.0952f}, - {80.9524f,28.5714f}, - {76.1905f,14.2857f}, - {66.6666f,4.7619f}, - {57.1428f,0.0f}, - {42.8571f,0.0f}, - {33.3333f,4.7619f}, - {23.8095f,14.2857f} -}; - -static const SFG_StrokeStrip ch98st[] = -{ - {2,ch98st0}, - {14,ch98st1} -}; - -static const SFG_StrokeChar ch98 = {104.762f,2,ch98st}; - -/* char: 0x63 */ - -static const SFG_StrokeVertex ch99st0[] = -{ - {80.9524f,52.381f}, - {71.4285f,61.9048f}, - {61.9047f,66.6667f}, - {47.619f,66.6667f}, - {38.0952f,61.9048f}, - {28.5714f,52.381f}, - {23.8095f,38.0952f}, - {23.8095f,28.5714f}, - {28.5714f,14.2857f}, - {38.0952f,4.7619f}, - {47.619f,0.0f}, - {61.9047f,0.0f}, - {71.4285f,4.7619f}, - {80.9524f,14.2857f} -}; - -static const SFG_StrokeStrip ch99st[] = -{ - {14,ch99st0} -}; - -static const SFG_StrokeChar ch99 = {104.762f,1,ch99st}; - -/* char: 0x64 */ - -static const SFG_StrokeVertex ch100st0[] = -{ - {80.9524f,100.0f}, - {80.9524f,0.0f} -}; - -static const SFG_StrokeVertex ch100st1[] = -{ - {80.9524f,52.381f}, - {71.4285f,61.9048f}, - {61.9047f,66.6667f}, - {47.619f,66.6667f}, - {38.0952f,61.9048f}, - {28.5714f,52.381f}, - {23.8095f,38.0952f}, - {23.8095f,28.5714f}, - {28.5714f,14.2857f}, - {38.0952f,4.7619f}, - {47.619f,0.0f}, - {61.9047f,0.0f}, - {71.4285f,4.7619f}, - {80.9524f,14.2857f} -}; - -static const SFG_StrokeStrip ch100st[] = -{ - {2,ch100st0}, - {14,ch100st1} -}; - -static const SFG_StrokeChar ch100 = {104.762f,2,ch100st}; - -/* char: 0x65 */ - -static const SFG_StrokeVertex ch101st0[] = -{ - {23.8095f,38.0952f}, - {80.9524f,38.0952f}, - {80.9524f,47.619f}, - {76.1905f,57.1429f}, - {71.4285f,61.9048f}, - {61.9047f,66.6667f}, - {47.619f,66.6667f}, - {38.0952f,61.9048f}, - {28.5714f,52.381f}, - {23.8095f,38.0952f}, - {23.8095f,28.5714f}, - {28.5714f,14.2857f}, - {38.0952f,4.7619f}, - {47.619f,0.0f}, - {61.9047f,0.0f}, - {71.4285f,4.7619f}, - {80.9524f,14.2857f} -}; - -static const SFG_StrokeStrip ch101st[] = -{ - {17,ch101st0} -}; - -static const SFG_StrokeChar ch101 = {104.762f,1,ch101st}; - -/* char: 0x66 */ - -static const SFG_StrokeVertex ch102st0[] = -{ - {71.4286f,100.0f}, - {61.9048f,100.0f}, - {52.381f,95.2381f}, - {47.6191f,80.9524f}, - {47.6191f,0.0f} -}; - -static const SFG_StrokeVertex ch102st1[] = -{ - {33.3334f,66.6667f}, - {66.6667f,66.6667f} -}; - -static const SFG_StrokeStrip ch102st[] = -{ - {5,ch102st0}, - {2,ch102st1} -}; - -static const SFG_StrokeChar ch102 = {104.762f,2,ch102st}; - -/* char: 0x67 */ - -static const SFG_StrokeVertex ch103st0[] = -{ - {80.9524f,66.6667f}, - {80.9524f,-9.5238f}, - {76.1905f,-23.8095f}, - {71.4285f,-28.5714f}, - {61.9047f,-33.3333f}, - {47.619f,-33.3333f}, - {38.0952f,-28.5714f} -}; - -static const SFG_StrokeVertex ch103st1[] = -{ - {80.9524f,52.381f}, - {71.4285f,61.9048f}, - {61.9047f,66.6667f}, - {47.619f,66.6667f}, - {38.0952f,61.9048f}, - {28.5714f,52.381f}, - {23.8095f,38.0952f}, - {23.8095f,28.5714f}, - {28.5714f,14.2857f}, - {38.0952f,4.7619f}, - {47.619f,0.0f}, - {61.9047f,0.0f}, - {71.4285f,4.7619f}, - {80.9524f,14.2857f} -}; - -static const SFG_StrokeStrip ch103st[] = -{ - {7,ch103st0}, - {14,ch103st1} -}; - -static const SFG_StrokeChar ch103 = {104.762f,2,ch103st}; - -/* char: 0x68 */ - -static const SFG_StrokeVertex ch104st0[] = -{ - {26.1905f,100.0f}, - {26.1905f,0.0f} -}; - -static const SFG_StrokeVertex ch104st1[] = -{ - {26.1905f,47.619f}, - {40.4762f,61.9048f}, - {50.0f,66.6667f}, - {64.2857f,66.6667f}, - {73.8095f,61.9048f}, - {78.5715f,47.619f}, - {78.5715f,0.0f} -}; - -static const SFG_StrokeStrip ch104st[] = -{ - {2,ch104st0}, - {7,ch104st1} -}; - -static const SFG_StrokeChar ch104 = {104.762f,2,ch104st}; - -/* char: 0x69 */ - -static const SFG_StrokeVertex ch105st0[] = -{ - {47.6191f,100.0f}, - {52.381f,95.2381f}, - {57.1429f,100.0f}, - {52.381f,104.762f}, - {47.6191f,100.0f} -}; - -static const SFG_StrokeVertex ch105st1[] = -{ - {52.381f,66.6667f}, - {52.381f,0.0f} -}; - -static const SFG_StrokeStrip ch105st[] = -{ - {5,ch105st0}, - {2,ch105st1} -}; - -static const SFG_StrokeChar ch105 = {104.762f,2,ch105st}; - -/* char: 0x6a */ - -static const SFG_StrokeVertex ch106st0[] = -{ - {57.1429f,100.0f}, - {61.9048f,95.2381f}, - {66.6667f,100.0f}, - {61.9048f,104.762f}, - {57.1429f,100.0f} -}; - -static const SFG_StrokeVertex ch106st1[] = -{ - {61.9048f,66.6667f}, - {61.9048f,-14.2857f}, - {57.1429f,-28.5714f}, - {47.6191f,-33.3333f}, - {38.0953f,-33.3333f} -}; - -static const SFG_StrokeStrip ch106st[] = -{ - {5,ch106st0}, - {5,ch106st1} -}; - -static const SFG_StrokeChar ch106 = {104.762f,2,ch106st}; - -/* char: 0x6b */ - -static const SFG_StrokeVertex ch107st0[] = -{ - {26.1905f,100.0f}, - {26.1905f,0.0f} -}; - -static const SFG_StrokeVertex ch107st1[] = -{ - {73.8095f,66.6667f}, - {26.1905f,19.0476f} -}; - -static const SFG_StrokeVertex ch107st2[] = -{ - {45.2381f,38.0952f}, - {78.5715f,0.0f} -}; - -static const SFG_StrokeStrip ch107st[] = -{ - {2,ch107st0}, - {2,ch107st1}, - {2,ch107st2} -}; - -static const SFG_StrokeChar ch107 = {104.762f,3,ch107st}; - -/* char: 0x6c */ - -static const SFG_StrokeVertex ch108st0[] = -{ - {52.381f,100.0f}, - {52.381f,0.0f} -}; - -static const SFG_StrokeStrip ch108st[] = -{ - {2,ch108st0} -}; - -static const SFG_StrokeChar ch108 = {104.762f,1,ch108st}; - -/* char: 0x6d */ - -static const SFG_StrokeVertex ch109st0[] = -{ - {0,66.6667f}, - {0,0.0f} -}; - -static const SFG_StrokeVertex ch109st1[] = -{ - {0,47.619f}, - {14.2857f,61.9048f}, - {23.8095f,66.6667f}, - {38.0952f,66.6667f}, - {47.619f,61.9048f}, - {52.381f,47.619f}, - {52.381f,0.0f} -}; - -static const SFG_StrokeVertex ch109st2[] = -{ - {52.381f,47.619f}, - {66.6667f,61.9048f}, - {76.1905f,66.6667f}, - {90.4762f,66.6667f}, - {100.0f,61.9048f}, - {104.762f,47.619f}, - {104.762f,0.0f} -}; - -static const SFG_StrokeStrip ch109st[] = -{ - {2,ch109st0}, - {7,ch109st1}, - {7,ch109st2} -}; - -static const SFG_StrokeChar ch109 = {104.762f,3,ch109st}; - -/* char: 0x6e */ - -static const SFG_StrokeVertex ch110st0[] = -{ - {26.1905f,66.6667f}, - {26.1905f,0.0f} -}; - -static const SFG_StrokeVertex ch110st1[] = -{ - {26.1905f,47.619f}, - {40.4762f,61.9048f}, - {50.0f,66.6667f}, - {64.2857f,66.6667f}, - {73.8095f,61.9048f}, - {78.5715f,47.619f}, - {78.5715f,0.0f} -}; - -static const SFG_StrokeStrip ch110st[] = -{ - {2,ch110st0}, - {7,ch110st1} -}; - -static const SFG_StrokeChar ch110 = {104.762f,2,ch110st}; - -/* char: 0x6f */ - -static const SFG_StrokeVertex ch111st0[] = -{ - {45.2381f,66.6667f}, - {35.7143f,61.9048f}, - {26.1905f,52.381f}, - {21.4286f,38.0952f}, - {21.4286f,28.5714f}, - {26.1905f,14.2857f}, - {35.7143f,4.7619f}, - {45.2381f,0.0f}, - {59.5238f,0.0f}, - {69.0476f,4.7619f}, - {78.5714f,14.2857f}, - {83.3334f,28.5714f}, - {83.3334f,38.0952f}, - {78.5714f,52.381f}, - {69.0476f,61.9048f}, - {59.5238f,66.6667f}, - {45.2381f,66.6667f} -}; - -static const SFG_StrokeStrip ch111st[] = -{ - {17,ch111st0} -}; - -static const SFG_StrokeChar ch111 = {104.762f,1,ch111st}; - -/* char: 0x70 */ - -static const SFG_StrokeVertex ch112st0[] = -{ - {23.8095f,66.6667f}, - {23.8095f,-33.3333f} -}; - -static const SFG_StrokeVertex ch112st1[] = -{ - {23.8095f,52.381f}, - {33.3333f,61.9048f}, - {42.8571f,66.6667f}, - {57.1428f,66.6667f}, - {66.6666f,61.9048f}, - {76.1905f,52.381f}, - {80.9524f,38.0952f}, - {80.9524f,28.5714f}, - {76.1905f,14.2857f}, - {66.6666f,4.7619f}, - {57.1428f,0.0f}, - {42.8571f,0.0f}, - {33.3333f,4.7619f}, - {23.8095f,14.2857f} -}; - -static const SFG_StrokeStrip ch112st[] = -{ - {2,ch112st0}, - {14,ch112st1} -}; - -static const SFG_StrokeChar ch112 = {104.762f,2,ch112st}; - -/* char: 0x71 */ - -static const SFG_StrokeVertex ch113st0[] = -{ - {80.9524f,66.6667f}, - {80.9524f,-33.3333f} -}; - -static const SFG_StrokeVertex ch113st1[] = -{ - {80.9524f,52.381f}, - {71.4285f,61.9048f}, - {61.9047f,66.6667f}, - {47.619f,66.6667f}, - {38.0952f,61.9048f}, - {28.5714f,52.381f}, - {23.8095f,38.0952f}, - {23.8095f,28.5714f}, - {28.5714f,14.2857f}, - {38.0952f,4.7619f}, - {47.619f,0.0f}, - {61.9047f,0.0f}, - {71.4285f,4.7619f}, - {80.9524f,14.2857f} -}; - -static const SFG_StrokeStrip ch113st[] = -{ - {2,ch113st0}, - {14,ch113st1} -}; - -static const SFG_StrokeChar ch113 = {104.762f,2,ch113st}; - -/* char: 0x72 */ - -static const SFG_StrokeVertex ch114st0[] = -{ - {33.3334f,66.6667f}, - {33.3334f,0.0f} -}; - -static const SFG_StrokeVertex ch114st1[] = -{ - {33.3334f,38.0952f}, - {38.0953f,52.381f}, - {47.6191f,61.9048f}, - {57.1429f,66.6667f}, - {71.4286f,66.6667f} -}; - -static const SFG_StrokeStrip ch114st[] = -{ - {2,ch114st0}, - {5,ch114st1} -}; - -static const SFG_StrokeChar ch114 = {104.762f,2,ch114st}; - -/* char: 0x73 */ - -static const SFG_StrokeVertex ch115st0[] = -{ - {78.5715f,52.381f}, - {73.8095f,61.9048f}, - {59.5238f,66.6667f}, - {45.2381f,66.6667f}, - {30.9524f,61.9048f}, - {26.1905f,52.381f}, - {30.9524f,42.8571f}, - {40.4762f,38.0952f}, - {64.2857f,33.3333f}, - {73.8095f,28.5714f}, - {78.5715f,19.0476f}, - {78.5715f,14.2857f}, - {73.8095f,4.7619f}, - {59.5238f,0.0f}, - {45.2381f,0.0f}, - {30.9524f,4.7619f}, - {26.1905f,14.2857f} -}; - -static const SFG_StrokeStrip ch115st[] = -{ - {17,ch115st0} -}; - -static const SFG_StrokeChar ch115 = {104.762f,1,ch115st}; - -/* char: 0x74 */ - -static const SFG_StrokeVertex ch116st0[] = -{ - {47.6191f,100.0f}, - {47.6191f,19.0476f}, - {52.381f,4.7619f}, - {61.9048f,0.0f}, - {71.4286f,0.0f} -}; - -static const SFG_StrokeVertex ch116st1[] = -{ - {33.3334f,66.6667f}, - {66.6667f,66.6667f} -}; - -static const SFG_StrokeStrip ch116st[] = -{ - {5,ch116st0}, - {2,ch116st1} -}; - -static const SFG_StrokeChar ch116 = {104.762f,2,ch116st}; - -/* char: 0x75 */ - -static const SFG_StrokeVertex ch117st0[] = -{ - {26.1905f,66.6667f}, - {26.1905f,19.0476f}, - {30.9524f,4.7619f}, - {40.4762f,0.0f}, - {54.7619f,0.0f}, - {64.2857f,4.7619f}, - {78.5715f,19.0476f} -}; - -static const SFG_StrokeVertex ch117st1[] = -{ - {78.5715f,66.6667f}, - {78.5715f,0.0f} -}; - -static const SFG_StrokeStrip ch117st[] = -{ - {7,ch117st0}, - {2,ch117st1} -}; - -static const SFG_StrokeChar ch117 = {104.762f,2,ch117st}; - -/* char: 0x76 */ - -static const SFG_StrokeVertex ch118st0[] = -{ - {23.8095f,66.6667f}, - {52.3809f,0.0f} -}; - -static const SFG_StrokeVertex ch118st1[] = -{ - {80.9524f,66.6667f}, - {52.3809f,0.0f} -}; - -static const SFG_StrokeStrip ch118st[] = -{ - {2,ch118st0}, - {2,ch118st1} -}; - -static const SFG_StrokeChar ch118 = {104.762f,2,ch118st}; - -/* char: 0x77 */ - -static const SFG_StrokeVertex ch119st0[] = -{ - {14.2857f,66.6667f}, - {33.3333f,0.0f} -}; - -static const SFG_StrokeVertex ch119st1[] = -{ - {52.3809f,66.6667f}, - {33.3333f,0.0f} -}; - -static const SFG_StrokeVertex ch119st2[] = -{ - {52.3809f,66.6667f}, - {71.4286f,0.0f} -}; - -static const SFG_StrokeVertex ch119st3[] = -{ - {90.4762f,66.6667f}, - {71.4286f,0.0f} -}; - -static const SFG_StrokeStrip ch119st[] = -{ - {2,ch119st0}, - {2,ch119st1}, - {2,ch119st2}, - {2,ch119st3} -}; - -static const SFG_StrokeChar ch119 = {104.762f,4,ch119st}; - -/* char: 0x78 */ - -static const SFG_StrokeVertex ch120st0[] = -{ - {26.1905f,66.6667f}, - {78.5715f,0.0f} -}; - -static const SFG_StrokeVertex ch120st1[] = -{ - {78.5715f,66.6667f}, - {26.1905f,0.0f} -}; - -static const SFG_StrokeStrip ch120st[] = -{ - {2,ch120st0}, - {2,ch120st1} -}; - -static const SFG_StrokeChar ch120 = {104.762f,2,ch120st}; - -/* char: 0x79 */ - -static const SFG_StrokeVertex ch121st0[] = -{ - {26.1905f,66.6667f}, - {54.7619f,0.0f} -}; - -static const SFG_StrokeVertex ch121st1[] = -{ - {83.3334f,66.6667f}, - {54.7619f,0.0f}, - {45.2381f,-19.0476f}, - {35.7143f,-28.5714f}, - {26.1905f,-33.3333f}, - {21.4286f,-33.3333f} -}; - -static const SFG_StrokeStrip ch121st[] = -{ - {2,ch121st0}, - {6,ch121st1} -}; - -static const SFG_StrokeChar ch121 = {104.762f,2,ch121st}; - -/* char: 0x7a */ - -static const SFG_StrokeVertex ch122st0[] = -{ - {78.5715f,66.6667f}, - {26.1905f,0.0f} -}; - -static const SFG_StrokeVertex ch122st1[] = -{ - {26.1905f,66.6667f}, - {78.5715f,66.6667f} -}; - -static const SFG_StrokeVertex ch122st2[] = -{ - {26.1905f,0.0f}, - {78.5715f,0.0f} -}; - -static const SFG_StrokeStrip ch122st[] = -{ - {2,ch122st0}, - {2,ch122st1}, - {2,ch122st2} -}; - -static const SFG_StrokeChar ch122 = {104.762f,3,ch122st}; - -/* char: 0x7b */ - -static const SFG_StrokeVertex ch123st0[] = -{ - {64.2857f,119.048f}, - {54.7619f,114.286f}, - {50.0f,109.524f}, - {45.2381f,100.0f}, - {45.2381f,90.4762f}, - {50.0f,80.9524f}, - {54.7619f,76.1905f}, - {59.5238f,66.6667f}, - {59.5238f,57.1429f}, - {50.0f,47.619f} -}; - -static const SFG_StrokeVertex ch123st1[] = -{ - {54.7619f,114.286f}, - {50.0f,104.762f}, - {50.0f,95.2381f}, - {54.7619f,85.7143f}, - {59.5238f,80.9524f}, - {64.2857f,71.4286f}, - {64.2857f,61.9048f}, - {59.5238f,52.381f}, - {40.4762f,42.8571f}, - {59.5238f,33.3333f}, - {64.2857f,23.8095f}, - {64.2857f,14.2857f}, - {59.5238f,4.7619f}, - {54.7619f,0.0f}, - {50.0f,-9.5238f}, - {50.0f,-19.0476f}, - {54.7619f,-28.5714f} -}; - -static const SFG_StrokeVertex ch123st2[] = -{ - {50.0f,38.0952f}, - {59.5238f,28.5714f}, - {59.5238f,19.0476f}, - {54.7619f,9.5238f}, - {50.0f,4.7619f}, - {45.2381f,-4.7619f}, - {45.2381f,-14.2857f}, - {50.0f,-23.8095f}, - {54.7619f,-28.5714f}, - {64.2857f,-33.3333f} -}; - -static const SFG_StrokeStrip ch123st[] = -{ - {10,ch123st0}, - {17,ch123st1}, - {10,ch123st2} -}; - -static const SFG_StrokeChar ch123 = {104.762f,3,ch123st}; - -/* char: 0x7c */ - -static const SFG_StrokeVertex ch124st0[] = -{ - {52.381f,119.048f}, - {52.381f,-33.3333f} -}; - -static const SFG_StrokeStrip ch124st[] = -{ - {2,ch124st0} -}; - -static const SFG_StrokeChar ch124 = {104.762f,1,ch124st}; - -/* char: 0x7d */ - -static const SFG_StrokeVertex ch125st0[] = -{ - {40.4762f,119.048f}, - {50.0f,114.286f}, - {54.7619f,109.524f}, - {59.5238f,100.0f}, - {59.5238f,90.4762f}, - {54.7619f,80.9524f}, - {50.0f,76.1905f}, - {45.2381f,66.6667f}, - {45.2381f,57.1429f}, - {54.7619f,47.619f} -}; - -static const SFG_StrokeVertex ch125st1[] = -{ - {50.0f,114.286f}, - {54.7619f,104.762f}, - {54.7619f,95.2381f}, - {50.0f,85.7143f}, - {45.2381f,80.9524f}, - {40.4762f,71.4286f}, - {40.4762f,61.9048f}, - {45.2381f,52.381f}, - {64.2857f,42.8571f}, - {45.2381f,33.3333f}, - {40.4762f,23.8095f}, - {40.4762f,14.2857f}, - {45.2381f,4.7619f}, - {50.0f,0.0f}, - {54.7619f,-9.5238f}, - {54.7619f,-19.0476f}, - {50.0f,-28.5714f} -}; - -static const SFG_StrokeVertex ch125st2[] = -{ - {54.7619f,38.0952f}, - {45.2381f,28.5714f}, - {45.2381f,19.0476f}, - {50.0f,9.5238f}, - {54.7619f,4.7619f}, - {59.5238f,-4.7619f}, - {59.5238f,-14.2857f}, - {54.7619f,-23.8095f}, - {50.0f,-28.5714f}, - {40.4762f,-33.3333f} -}; - -static const SFG_StrokeStrip ch125st[] = -{ - {10,ch125st0}, - {17,ch125st1}, - {10,ch125st2} -}; - -static const SFG_StrokeChar ch125 = {104.762f,3,ch125st}; - -/* char: 0x7e */ - -static const SFG_StrokeVertex ch126st0[] = -{ - {9.5238f,28.5714f}, - {9.5238f,38.0952f}, - {14.2857f,52.381f}, - {23.8095f,57.1429f}, - {33.3333f,57.1429f}, - {42.8571f,52.381f}, - {61.9048f,38.0952f}, - {71.4286f,33.3333f}, - {80.9524f,33.3333f}, - {90.4762f,38.0952f}, - {95.2381f,47.619f} -}; - -static const SFG_StrokeVertex ch126st1[] = -{ - {9.5238f,38.0952f}, - {14.2857f,47.619f}, - {23.8095f,52.381f}, - {33.3333f,52.381f}, - {42.8571f,47.619f}, - {61.9048f,33.3333f}, - {71.4286f,28.5714f}, - {80.9524f,28.5714f}, - {90.4762f,33.3333f}, - {95.2381f,47.619f}, - {95.2381f,57.1429f} -}; - -static const SFG_StrokeStrip ch126st[] = -{ - {11,ch126st0}, - {11,ch126st1} -}; - -static const SFG_StrokeChar ch126 = {104.762f,2,ch126st}; - -/* char: 0x7f */ - -static const SFG_StrokeVertex ch127st0[] = -{ - {71.4286f,100.0f}, - {33.3333f,-33.3333f} -}; - -static const SFG_StrokeVertex ch127st1[] = -{ - {47.619f,66.6667f}, - {33.3333f,61.9048f}, - {23.8095f,52.381f}, - {19.0476f,38.0952f}, - {19.0476f,23.8095f}, - {23.8095f,14.2857f}, - {33.3333f,4.7619f}, - {47.619f,0.0f}, - {57.1428f,0.0f}, - {71.4286f,4.7619f}, - {80.9524f,14.2857f}, - {85.7143f,28.5714f}, - {85.7143f,42.8571f}, - {80.9524f,52.381f}, - {71.4286f,61.9048f}, - {57.1428f,66.6667f}, - {47.619f,66.6667f} -}; - -static const SFG_StrokeStrip ch127st[] = -{ - {2,ch127st0}, - {17,ch127st1} -}; - -static const SFG_StrokeChar ch127 = {104.762f,2,ch127st}; - -static const SFG_StrokeChar *chars[] = -{ - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - &ch32, &ch33, &ch34, &ch35, &ch36, &ch37, &ch38, &ch39, - &ch40, &ch41, &ch42, &ch43, &ch44, &ch45, &ch46, &ch47, - &ch48, &ch49, &ch50, &ch51, &ch52, &ch53, &ch54, &ch55, - &ch56, &ch57, &ch58, &ch59, &ch60, &ch61, &ch62, &ch63, - &ch64, &ch65, &ch66, &ch67, &ch68, &ch69, &ch70, &ch71, - &ch72, &ch73, &ch74, &ch75, &ch76, &ch77, &ch78, &ch79, - &ch80, &ch81, &ch82, &ch83, &ch84, &ch85, &ch86, &ch87, - &ch88, &ch89, &ch90, &ch91, &ch92, &ch93, &ch94, &ch95, - &ch96, &ch97, &ch98, &ch99, &ch100, &ch101, &ch102, &ch103, - &ch104, &ch105, &ch106, &ch107, &ch108, &ch109, &ch110, &ch111, - &ch112, &ch113, &ch114, &ch115, &ch116, &ch117, &ch118, &ch119, - &ch120, &ch121, &ch122, &ch123, &ch124, &ch125, &ch126, &ch127 -}; - -const SFG_StrokeFont fgStrokeMonoRoman = {"MonoRoman",128,152.381f,chars}; diff --git a/examples/common/opengl-framework/freeglut/freeglut_stroke_roman.c b/examples/common/opengl-framework/freeglut/freeglut_stroke_roman.c deleted file mode 100644 index da4e0302..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_stroke_roman.c +++ /dev/null @@ -1,2849 +0,0 @@ -/* - * freeglut_stroke_roman.c - * - * freeglut Roman stroke font definition - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 16 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - - -/* This file has been automatically generated by the genstroke utility. */ - -#include -#include "freeglut_internal.h" - -/* char: 0x20 */ - -static const SFG_StrokeStrip ch32st[] = -{ - { 0, NULL } -}; - -static const SFG_StrokeChar ch32 = {104.762f,0,ch32st}; - -/* char: 0x21 */ - -static const SFG_StrokeVertex ch33st0[] = -{ - {13.3819f,100.0f}, - {13.3819f,33.3333f} -}; - -static const SFG_StrokeVertex ch33st1[] = -{ - {13.3819f,9.5238f}, - {8.62f,4.7619f}, - {13.3819f,0.0f}, - {18.1438f,4.7619f}, - {13.3819f,9.5238f} -}; - -static const SFG_StrokeStrip ch33st[] = -{ - {2,ch33st0}, - {5,ch33st1} -}; - -static const SFG_StrokeChar ch33 = {26.6238f,2,ch33st}; - -/* char: 0x22 */ - -static const SFG_StrokeVertex ch34st0[] = -{ - {4.02f,100.0f}, - {4.02f,66.6667f} -}; - -static const SFG_StrokeVertex ch34st1[] = -{ - {42.1152f,100.0f}, - {42.1152f,66.6667f} -}; - -static const SFG_StrokeStrip ch34st[] = -{ - {2,ch34st0}, - {2,ch34st1} -}; - -static const SFG_StrokeChar ch34 = {51.4352f,2,ch34st}; - -/* char: 0x23 */ - -static const SFG_StrokeVertex ch35st0[] = -{ - {41.2952f,119.048f}, - {7.9619f,-33.3333f} -}; - -static const SFG_StrokeVertex ch35st1[] = -{ - {69.8667f,119.048f}, - {36.5333f,-33.3333f} -}; - -static const SFG_StrokeVertex ch35st2[] = -{ - {7.9619f,57.1429f}, - {74.6286f,57.1429f} -}; - -static const SFG_StrokeVertex ch35st3[] = -{ - {3.2f,28.5714f}, - {69.8667f,28.5714f} -}; - -static const SFG_StrokeStrip ch35st[] = -{ - {2,ch35st0}, - {2,ch35st1}, - {2,ch35st2}, - {2,ch35st3} -}; - -static const SFG_StrokeChar ch35 = {79.4886f,4,ch35st}; - -/* char: 0x24 */ - -static const SFG_StrokeVertex ch36st0[] = -{ - {28.6295f,119.048f}, - {28.6295f,-19.0476f} -}; - -static const SFG_StrokeVertex ch36st1[] = -{ - {47.6771f,119.048f}, - {47.6771f,-19.0476f} -}; - -static const SFG_StrokeVertex ch36st2[] = -{ - {71.4867f,85.7143f}, - {61.9629f,95.2381f}, - {47.6771f,100.0f}, - {28.6295f,100.0f}, - {14.3438f,95.2381f}, - {4.82f,85.7143f}, - {4.82f,76.1905f}, - {9.5819f,66.6667f}, - {14.3438f,61.9048f}, - {23.8676f,57.1429f}, - {52.439f,47.619f}, - {61.9629f,42.8571f}, - {66.7248f,38.0952f}, - {71.4867f,28.5714f}, - {71.4867f,14.2857f}, - {61.9629f,4.7619f}, - {47.6771f,0.0f}, - {28.6295f,0.0f}, - {14.3438f,4.7619f}, - {4.82f,14.2857f} -}; - -static const SFG_StrokeStrip ch36st[] = -{ - {2,ch36st0}, - {2,ch36st1}, - {20,ch36st2} -}; - -static const SFG_StrokeChar ch36 = {76.2067f,3,ch36st}; - -/* char: 0x25 */ - -static const SFG_StrokeVertex ch37st0[] = -{ - {92.0743f,100.0f}, - {6.36f,0.0f} -}; - -static const SFG_StrokeVertex ch37st1[] = -{ - {30.1695f,100.0f}, - {39.6933f,90.4762f}, - {39.6933f,80.9524f}, - {34.9314f,71.4286f}, - {25.4076f,66.6667f}, - {15.8838f,66.6667f}, - {6.36f,76.1905f}, - {6.36f,85.7143f}, - {11.1219f,95.2381f}, - {20.6457f,100.0f}, - {30.1695f,100.0f}, - {39.6933f,95.2381f}, - {53.979f,90.4762f}, - {68.2648f,90.4762f}, - {82.5505f,95.2381f}, - {92.0743f,100.0f} -}; - -static const SFG_StrokeVertex ch37st2[] = -{ - {73.0267f,33.3333f}, - {63.5029f,28.5714f}, - {58.741f,19.0476f}, - {58.741f,9.5238f}, - {68.2648f,0.0f}, - {77.7886f,0.0f}, - {87.3124f,4.7619f}, - {92.0743f,14.2857f}, - {92.0743f,23.8095f}, - {82.5505f,33.3333f}, - {73.0267f,33.3333f} -}; - -static const SFG_StrokeStrip ch37st[] = -{ - {2,ch37st0}, - {16,ch37st1}, - {11,ch37st2} -}; - -static const SFG_StrokeChar ch37 = {96.5743f,3,ch37st}; - -/* char: 0x26 */ - -static const SFG_StrokeVertex ch38st0[] = -{ - {101.218f,57.1429f}, - {101.218f,61.9048f}, - {96.4562f,66.6667f}, - {91.6943f,66.6667f}, - {86.9324f,61.9048f}, - {82.1705f,52.381f}, - {72.6467f,28.5714f}, - {63.1229f,14.2857f}, - {53.599f,4.7619f}, - {44.0752f,0.0f}, - {25.0276f,0.0f}, - {15.5038f,4.7619f}, - {10.7419f,9.5238f}, - {5.98f,19.0476f}, - {5.98f,28.5714f}, - {10.7419f,38.0952f}, - {15.5038f,42.8571f}, - {48.8371f,61.9048f}, - {53.599f,66.6667f}, - {58.361f,76.1905f}, - {58.361f,85.7143f}, - {53.599f,95.2381f}, - {44.0752f,100.0f}, - {34.5514f,95.2381f}, - {29.7895f,85.7143f}, - {29.7895f,76.1905f}, - {34.5514f,61.9048f}, - {44.0752f,47.619f}, - {67.8848f,14.2857f}, - {77.4086f,4.7619f}, - {86.9324f,0.0f}, - {96.4562f,0.0f}, - {101.218f,4.7619f}, - {101.218f,9.5238f} -}; - -static const SFG_StrokeStrip ch38st[] = -{ - {34,ch38st0} -}; - -static const SFG_StrokeChar ch38 = {101.758f,1,ch38st}; - -/* char: 0x27 */ - -static const SFG_StrokeVertex ch39st0[] = -{ - {4.44f,100.0f}, - {4.44f,66.6667f} -}; - -static const SFG_StrokeStrip ch39st[] = -{ - {2,ch39st0} -}; - -static const SFG_StrokeChar ch39 = {13.62f,1,ch39st}; - -/* char: 0x28 */ - -static const SFG_StrokeVertex ch40st0[] = -{ - {40.9133f,119.048f}, - {31.3895f,109.524f}, - {21.8657f,95.2381f}, - {12.3419f,76.1905f}, - {7.58f,52.381f}, - {7.58f,33.3333f}, - {12.3419f,9.5238f}, - {21.8657f,-9.5238f}, - {31.3895f,-23.8095f}, - {40.9133f,-33.3333f} -}; - -static const SFG_StrokeStrip ch40st[] = -{ - {10,ch40st0} -}; - -static const SFG_StrokeChar ch40 = {47.1733f,1,ch40st}; - -/* char: 0x29 */ - -static const SFG_StrokeVertex ch41st0[] = -{ - {5.28f,119.048f}, - {14.8038f,109.524f}, - {24.3276f,95.2381f}, - {33.8514f,76.1905f}, - {38.6133f,52.381f}, - {38.6133f,33.3333f}, - {33.8514f,9.5238f}, - {24.3276f,-9.5238f}, - {14.8038f,-23.8095f}, - {5.28f,-33.3333f} -}; - -static const SFG_StrokeStrip ch41st[] = -{ - {10,ch41st0} -}; - -static const SFG_StrokeChar ch41 = {47.5333f,1,ch41st}; - -/* char: 0x2a */ - -static const SFG_StrokeVertex ch42st0[] = -{ - {30.7695f,71.4286f}, - {30.7695f,14.2857f} -}; - -static const SFG_StrokeVertex ch42st1[] = -{ - {6.96f,57.1429f}, - {54.579f,28.5714f} -}; - -static const SFG_StrokeVertex ch42st2[] = -{ - {54.579f,57.1429f}, - {6.96f,28.5714f} -}; - -static const SFG_StrokeStrip ch42st[] = -{ - {2,ch42st0}, - {2,ch42st1}, - {2,ch42st2} -}; - -static const SFG_StrokeChar ch42 = {59.439f,3,ch42st}; - -/* char: 0x2b */ - -static const SFG_StrokeVertex ch43st0[] = -{ - {48.8371f,85.7143f}, - {48.8371f,0.0f} -}; - -static const SFG_StrokeVertex ch43st1[] = -{ - {5.98f,42.8571f}, - {91.6943f,42.8571f} -}; - -static const SFG_StrokeStrip ch43st[] = -{ - {2,ch43st0}, - {2,ch43st1} -}; - -static const SFG_StrokeChar ch43 = {97.2543f,2,ch43st}; - -/* char: 0x2c */ - -static const SFG_StrokeVertex ch44st0[] = -{ - {18.2838f,4.7619f}, - {13.5219f,0.0f}, - {8.76f,4.7619f}, - {13.5219f,9.5238f}, - {18.2838f,4.7619f}, - {18.2838f,-4.7619f}, - {13.5219f,-14.2857f}, - {8.76f,-19.0476f} -}; - -static const SFG_StrokeStrip ch44st[] = -{ - {8,ch44st0} -}; - -static const SFG_StrokeChar ch44 = {26.0638f,1,ch44st}; - -/* char: 0x2d */ - -static const SFG_StrokeVertex ch45st0[] = -{ - {7.38f,42.8571f}, - {93.0943f,42.8571f} -}; - -static const SFG_StrokeStrip ch45st[] = -{ - {2,ch45st0} -}; - -static const SFG_StrokeChar ch45 = {100.754f,1,ch45st}; - -/* char: 0x2e */ - -static const SFG_StrokeVertex ch46st0[] = -{ - {13.1019f,9.5238f}, - {8.34f,4.7619f}, - {13.1019f,0.0f}, - {17.8638f,4.7619f}, - {13.1019f,9.5238f} -}; - -static const SFG_StrokeStrip ch46st[] = -{ - {5,ch46st0} -}; - -static const SFG_StrokeChar ch46 = {26.4838f,1,ch46st}; - -/* char: 0x2f */ - -static const SFG_StrokeVertex ch47st0[] = -{ - {7.24f,-14.2857f}, - {73.9067f,100.0f} -}; - -static const SFG_StrokeStrip ch47st[] = -{ - {2,ch47st0} -}; - -static const SFG_StrokeChar ch47 = {82.1067f,1,ch47st}; - -/* char: 0x30 */ - -static const SFG_StrokeVertex ch48st0[] = -{ - {33.5514f,100.0f}, - {19.2657f,95.2381f}, - {9.7419f,80.9524f}, - {4.98f,57.1429f}, - {4.98f,42.8571f}, - {9.7419f,19.0476f}, - {19.2657f,4.7619f}, - {33.5514f,0.0f}, - {43.0752f,0.0f}, - {57.361f,4.7619f}, - {66.8848f,19.0476f}, - {71.6467f,42.8571f}, - {71.6467f,57.1429f}, - {66.8848f,80.9524f}, - {57.361f,95.2381f}, - {43.0752f,100.0f}, - {33.5514f,100.0f} -}; - -static const SFG_StrokeStrip ch48st[] = -{ - {17,ch48st0} -}; - -static const SFG_StrokeChar ch48 = {77.0667f,1,ch48st}; - -/* char: 0x31 */ - -static const SFG_StrokeVertex ch49st0[] = -{ - {11.82f,80.9524f}, - {21.3438f,85.7143f}, - {35.6295f,100.0f}, - {35.6295f,0.0f} -}; - -static const SFG_StrokeStrip ch49st[] = -{ - {4,ch49st0} -}; - -static const SFG_StrokeChar ch49 = {66.5295f,1,ch49st}; - -/* char: 0x32 */ - -static const SFG_StrokeVertex ch50st0[] = -{ - {10.1819f,76.1905f}, - {10.1819f,80.9524f}, - {14.9438f,90.4762f}, - {19.7057f,95.2381f}, - {29.2295f,100.0f}, - {48.2771f,100.0f}, - {57.801f,95.2381f}, - {62.5629f,90.4762f}, - {67.3248f,80.9524f}, - {67.3248f,71.4286f}, - {62.5629f,61.9048f}, - {53.039f,47.619f}, - {5.42f,0.0f}, - {72.0867f,0.0f} -}; - -static const SFG_StrokeStrip ch50st[] = -{ - {14,ch50st0} -}; - -static const SFG_StrokeChar ch50 = {77.6467f,1,ch50st}; - -/* char: 0x33 */ - -static const SFG_StrokeVertex ch51st0[] = -{ - {14.5238f,100.0f}, - {66.9048f,100.0f}, - {38.3333f,61.9048f}, - {52.619f,61.9048f}, - {62.1429f,57.1429f}, - {66.9048f,52.381f}, - {71.6667f,38.0952f}, - {71.6667f,28.5714f}, - {66.9048f,14.2857f}, - {57.381f,4.7619f}, - {43.0952f,0.0f}, - {28.8095f,0.0f}, - {14.5238f,4.7619f}, - {9.7619f,9.5238f}, - {5.0f,19.0476f} -}; - -static const SFG_StrokeStrip ch51st[] = -{ - {15,ch51st0} -}; - -static const SFG_StrokeChar ch51 = {77.0467f,1,ch51st}; - -/* char: 0x34 */ - -static const SFG_StrokeVertex ch52st0[] = -{ - {51.499f,100.0f}, - {3.88f,33.3333f}, - {75.3086f,33.3333f} -}; - -static const SFG_StrokeVertex ch52st1[] = -{ - {51.499f,100.0f}, - {51.499f,0.0f} -}; - -static const SFG_StrokeStrip ch52st[] = -{ - {3,ch52st0}, - {2,ch52st1} -}; - -static const SFG_StrokeChar ch52 = {80.1686f,2,ch52st}; - -/* char: 0x35 */ - -static const SFG_StrokeVertex ch53st0[] = -{ - {62.0029f,100.0f}, - {14.3838f,100.0f}, - {9.6219f,57.1429f}, - {14.3838f,61.9048f}, - {28.6695f,66.6667f}, - {42.9552f,66.6667f}, - {57.241f,61.9048f}, - {66.7648f,52.381f}, - {71.5267f,38.0952f}, - {71.5267f,28.5714f}, - {66.7648f,14.2857f}, - {57.241f,4.7619f}, - {42.9552f,0.0f}, - {28.6695f,0.0f}, - {14.3838f,4.7619f}, - {9.6219f,9.5238f}, - {4.86f,19.0476f} -}; - -static const SFG_StrokeStrip ch53st[] = -{ - {17,ch53st0} -}; - -static const SFG_StrokeChar ch53 = {77.6867f,1,ch53st}; - -/* char: 0x36 */ - -static const SFG_StrokeVertex ch54st0[] = -{ - {62.7229f,85.7143f}, - {57.961f,95.2381f}, - {43.6752f,100.0f}, - {34.1514f,100.0f}, - {19.8657f,95.2381f}, - {10.3419f,80.9524f}, - {5.58f,57.1429f}, - {5.58f,33.3333f}, - {10.3419f,14.2857f}, - {19.8657f,4.7619f}, - {34.1514f,0.0f}, - {38.9133f,0.0f}, - {53.199f,4.7619f}, - {62.7229f,14.2857f}, - {67.4848f,28.5714f}, - {67.4848f,33.3333f}, - {62.7229f,47.619f}, - {53.199f,57.1429f}, - {38.9133f,61.9048f}, - {34.1514f,61.9048f}, - {19.8657f,57.1429f}, - {10.3419f,47.619f}, - {5.58f,33.3333f} -}; - -static const SFG_StrokeStrip ch54st[] = -{ - {23,ch54st0} -}; - -static const SFG_StrokeChar ch54 = {73.8048f,1,ch54st}; - -/* char: 0x37 */ - -static const SFG_StrokeVertex ch55st0[] = -{ - {72.2267f,100.0f}, - {24.6076f,0.0f} -}; - -static const SFG_StrokeVertex ch55st1[] = -{ - {5.56f,100.0f}, - {72.2267f,100.0f} -}; - -static const SFG_StrokeStrip ch55st[] = -{ - {2,ch55st0}, - {2,ch55st1} -}; - -static const SFG_StrokeChar ch55 = {77.2267f,2,ch55st}; - -/* char: 0x38 */ - -static const SFG_StrokeVertex ch56st0[] = -{ - {29.4095f,100.0f}, - {15.1238f,95.2381f}, - {10.3619f,85.7143f}, - {10.3619f,76.1905f}, - {15.1238f,66.6667f}, - {24.6476f,61.9048f}, - {43.6952f,57.1429f}, - {57.981f,52.381f}, - {67.5048f,42.8571f}, - {72.2667f,33.3333f}, - {72.2667f,19.0476f}, - {67.5048f,9.5238f}, - {62.7429f,4.7619f}, - {48.4571f,0.0f}, - {29.4095f,0.0f}, - {15.1238f,4.7619f}, - {10.3619f,9.5238f}, - {5.6f,19.0476f}, - {5.6f,33.3333f}, - {10.3619f,42.8571f}, - {19.8857f,52.381f}, - {34.1714f,57.1429f}, - {53.219f,61.9048f}, - {62.7429f,66.6667f}, - {67.5048f,76.1905f}, - {67.5048f,85.7143f}, - {62.7429f,95.2381f}, - {48.4571f,100.0f}, - {29.4095f,100.0f} -}; - -static const SFG_StrokeStrip ch56st[] = -{ - {29,ch56st0} -}; - -static const SFG_StrokeChar ch56 = {77.6667f,1,ch56st}; - -/* char: 0x39 */ - -static const SFG_StrokeVertex ch57st0[] = -{ - {68.5048f,66.6667f}, - {63.7429f,52.381f}, - {54.219f,42.8571f}, - {39.9333f,38.0952f}, - {35.1714f,38.0952f}, - {20.8857f,42.8571f}, - {11.3619f,52.381f}, - {6.6f,66.6667f}, - {6.6f,71.4286f}, - {11.3619f,85.7143f}, - {20.8857f,95.2381f}, - {35.1714f,100.0f}, - {39.9333f,100.0f}, - {54.219f,95.2381f}, - {63.7429f,85.7143f}, - {68.5048f,66.6667f}, - {68.5048f,42.8571f}, - {63.7429f,19.0476f}, - {54.219f,4.7619f}, - {39.9333f,0.0f}, - {30.4095f,0.0f}, - {16.1238f,4.7619f}, - {11.3619f,14.2857f} -}; - -static const SFG_StrokeStrip ch57st[] = -{ - {23,ch57st0} -}; - -static const SFG_StrokeChar ch57 = {74.0648f,1,ch57st}; - -/* char: 0x3a */ - -static const SFG_StrokeVertex ch58st0[] = -{ - {14.0819f,66.6667f}, - {9.32f,61.9048f}, - {14.0819f,57.1429f}, - {18.8438f,61.9048f}, - {14.0819f,66.6667f} -}; - -static const SFG_StrokeVertex ch58st1[] = -{ - {14.0819f,9.5238f}, - {9.32f,4.7619f}, - {14.0819f,0.0f}, - {18.8438f,4.7619f}, - {14.0819f,9.5238f} -}; - -static const SFG_StrokeStrip ch58st[] = -{ - {5,ch58st0}, - {5,ch58st1} -}; - -static const SFG_StrokeChar ch58 = {26.2238f,2,ch58st}; - -/* char: 0x3b */ - -static const SFG_StrokeVertex ch59st0[] = -{ - {12.9619f,66.6667f}, - {8.2f,61.9048f}, - {12.9619f,57.1429f}, - {17.7238f,61.9048f}, - {12.9619f,66.6667f} -}; - -static const SFG_StrokeVertex ch59st1[] = -{ - {17.7238f,4.7619f}, - {12.9619f,0.0f}, - {8.2f,4.7619f}, - {12.9619f,9.5238f}, - {17.7238f,4.7619f}, - {17.7238f,-4.7619f}, - {12.9619f,-14.2857f}, - {8.2f,-19.0476f} -}; - -static const SFG_StrokeStrip ch59st[] = -{ - {5,ch59st0}, - {8,ch59st1} -}; - -static const SFG_StrokeChar ch59 = {26.3038f,2,ch59st}; - -/* char: 0x3c */ - -static const SFG_StrokeVertex ch60st0[] = -{ - {79.2505f,85.7143f}, - {3.06f,42.8571f}, - {79.2505f,0.0f} -}; - -static const SFG_StrokeStrip ch60st[] = -{ - {3,ch60st0} -}; - -static const SFG_StrokeChar ch60 = {81.6105f,1,ch60st}; - -/* char: 0x3d */ - -static const SFG_StrokeVertex ch61st0[] = -{ - {5.7f,57.1429f}, - {91.4143f,57.1429f} -}; - -static const SFG_StrokeVertex ch61st1[] = -{ - {5.7f,28.5714f}, - {91.4143f,28.5714f} -}; - -static const SFG_StrokeStrip ch61st[] = -{ - {2,ch61st0}, - {2,ch61st1} -}; - -static const SFG_StrokeChar ch61 = {97.2543f,2,ch61st}; - -/* char: 0x3e */ - -static const SFG_StrokeVertex ch62st0[] = -{ - {2.78f,85.7143f}, - {78.9705f,42.8571f}, - {2.78f,0.0f} -}; - -static const SFG_StrokeStrip ch62st[] = -{ - {3,ch62st0} -}; - -static const SFG_StrokeChar ch62 = {81.6105f,1,ch62st}; - -/* char: 0x3f */ - -static const SFG_StrokeVertex ch63st0[] = -{ - {8.42f,76.1905f}, - {8.42f,80.9524f}, - {13.1819f,90.4762f}, - {17.9438f,95.2381f}, - {27.4676f,100.0f}, - {46.5152f,100.0f}, - {56.039f,95.2381f}, - {60.801f,90.4762f}, - {65.5629f,80.9524f}, - {65.5629f,71.4286f}, - {60.801f,61.9048f}, - {56.039f,57.1429f}, - {36.9914f,47.619f}, - {36.9914f,33.3333f} -}; - -static const SFG_StrokeVertex ch63st1[] = -{ - {36.9914f,9.5238f}, - {32.2295f,4.7619f}, - {36.9914f,0.0f}, - {41.7533f,4.7619f}, - {36.9914f,9.5238f} -}; - -static const SFG_StrokeStrip ch63st[] = -{ - {14,ch63st0}, - {5,ch63st1} -}; - -static const SFG_StrokeChar ch63 = {73.9029f,2,ch63st}; - -/* char: 0x40 */ - -static const SFG_StrokeVertex ch64st0[] = -{ - {49.2171f,52.381f}, - {39.6933f,57.1429f}, - {30.1695f,57.1429f}, - {25.4076f,47.619f}, - {25.4076f,42.8571f}, - {30.1695f,33.3333f}, - {39.6933f,33.3333f}, - {49.2171f,38.0952f} -}; - -static const SFG_StrokeVertex ch64st1[] = -{ - {49.2171f,57.1429f}, - {49.2171f,38.0952f}, - {53.979f,33.3333f}, - {63.5029f,33.3333f}, - {68.2648f,42.8571f}, - {68.2648f,47.619f}, - {63.5029f,61.9048f}, - {53.979f,71.4286f}, - {39.6933f,76.1905f}, - {34.9314f,76.1905f}, - {20.6457f,71.4286f}, - {11.1219f,61.9048f}, - {6.36f,47.619f}, - {6.36f,42.8571f}, - {11.1219f,28.5714f}, - {20.6457f,19.0476f}, - {34.9314f,14.2857f}, - {39.6933f,14.2857f}, - {53.979f,19.0476f} -}; - -static const SFG_StrokeStrip ch64st[] = -{ - {8,ch64st0}, - {19,ch64st1} -}; - -static const SFG_StrokeChar ch64 = {74.3648f,2,ch64st}; - -/* char: 0x41 */ - -static const SFG_StrokeVertex ch65st0[] = -{ - {40.5952f,100.0f}, - {2.5f,0.0f} -}; - -static const SFG_StrokeVertex ch65st1[] = -{ - {40.5952f,100.0f}, - {78.6905f,0.0f} -}; - -static const SFG_StrokeVertex ch65st2[] = -{ - {16.7857f,33.3333f}, - {64.4048f,33.3333f} -}; - -static const SFG_StrokeStrip ch65st[] = -{ - {2,ch65st0}, - {2,ch65st1}, - {2,ch65st2} -}; - -static const SFG_StrokeChar ch65 = {80.4905f,3,ch65st}; - -/* char: 0x42 */ - -static const SFG_StrokeVertex ch66st0[] = -{ - {11.42f,100.0f}, - {11.42f,0.0f} -}; - -static const SFG_StrokeVertex ch66st1[] = -{ - {11.42f,100.0f}, - {54.2771f,100.0f}, - {68.5629f,95.2381f}, - {73.3248f,90.4762f}, - {78.0867f,80.9524f}, - {78.0867f,71.4286f}, - {73.3248f,61.9048f}, - {68.5629f,57.1429f}, - {54.2771f,52.381f} -}; - -static const SFG_StrokeVertex ch66st2[] = -{ - {11.42f,52.381f}, - {54.2771f,52.381f}, - {68.5629f,47.619f}, - {73.3248f,42.8571f}, - {78.0867f,33.3333f}, - {78.0867f,19.0476f}, - {73.3248f,9.5238f}, - {68.5629f,4.7619f}, - {54.2771f,0.0f}, - {11.42f,0.0f} -}; - -static const SFG_StrokeStrip ch66st[] = -{ - {2,ch66st0}, - {9,ch66st1}, - {10,ch66st2} -}; - -static const SFG_StrokeChar ch66 = {83.6267f,3,ch66st}; - -/* char: 0x43 */ - -static const SFG_StrokeVertex ch67st0[] = -{ - {78.0886f,76.1905f}, - {73.3267f,85.7143f}, - {63.8029f,95.2381f}, - {54.279f,100.0f}, - {35.2314f,100.0f}, - {25.7076f,95.2381f}, - {16.1838f,85.7143f}, - {11.4219f,76.1905f}, - {6.66f,61.9048f}, - {6.66f,38.0952f}, - {11.4219f,23.8095f}, - {16.1838f,14.2857f}, - {25.7076f,4.7619f}, - {35.2314f,0.0f}, - {54.279f,0.0f}, - {63.8029f,4.7619f}, - {73.3267f,14.2857f}, - {78.0886f,23.8095f} -}; - -static const SFG_StrokeStrip ch67st[] = -{ - {18,ch67st0} -}; - -static const SFG_StrokeChar ch67 = {84.4886f,1,ch67st}; - -/* char: 0x44 */ - -static const SFG_StrokeVertex ch68st0[] = -{ - {11.96f,100.0f}, - {11.96f,0.0f} -}; - -static const SFG_StrokeVertex ch68st1[] = -{ - {11.96f,100.0f}, - {45.2933f,100.0f}, - {59.579f,95.2381f}, - {69.1029f,85.7143f}, - {73.8648f,76.1905f}, - {78.6267f,61.9048f}, - {78.6267f,38.0952f}, - {73.8648f,23.8095f}, - {69.1029f,14.2857f}, - {59.579f,4.7619f}, - {45.2933f,0.0f}, - {11.96f,0.0f} -}; - -static const SFG_StrokeStrip ch68st[] = -{ - {2,ch68st0}, - {12,ch68st1} -}; - -static const SFG_StrokeChar ch68 = {85.2867f,2,ch68st}; - -/* char: 0x45 */ - -static const SFG_StrokeVertex ch69st0[] = -{ - {11.42f,100.0f}, - {11.42f,0.0f} -}; - -static const SFG_StrokeVertex ch69st1[] = -{ - {11.42f,100.0f}, - {73.3248f,100.0f} -}; - -static const SFG_StrokeVertex ch69st2[] = -{ - {11.42f,52.381f}, - {49.5152f,52.381f} -}; - -static const SFG_StrokeVertex ch69st3[] = -{ - {11.42f,0.0f}, - {73.3248f,0.0f} -}; - -static const SFG_StrokeStrip ch69st[] = -{ - {2,ch69st0}, - {2,ch69st1}, - {2,ch69st2}, - {2,ch69st3} -}; - -static const SFG_StrokeChar ch69 = {78.1848f,4,ch69st}; - -/* char: 0x46 */ - -static const SFG_StrokeVertex ch70st0[] = -{ - {11.42f,100.0f}, - {11.42f,0.0f} -}; - -static const SFG_StrokeVertex ch70st1[] = -{ - {11.42f,100.0f}, - {73.3248f,100.0f} -}; - -static const SFG_StrokeVertex ch70st2[] = -{ - {11.42f,52.381f}, - {49.5152f,52.381f} -}; - -static const SFG_StrokeStrip ch70st[] = -{ - {2,ch70st0}, - {2,ch70st1}, - {2,ch70st2} -}; - -static const SFG_StrokeChar ch70 = {78.7448f,3,ch70st}; - -/* char: 0x47 */ - -static const SFG_StrokeVertex ch71st0[] = -{ - {78.4886f,76.1905f}, - {73.7267f,85.7143f}, - {64.2029f,95.2381f}, - {54.679f,100.0f}, - {35.6314f,100.0f}, - {26.1076f,95.2381f}, - {16.5838f,85.7143f}, - {11.8219f,76.1905f}, - {7.06f,61.9048f}, - {7.06f,38.0952f}, - {11.8219f,23.8095f}, - {16.5838f,14.2857f}, - {26.1076f,4.7619f}, - {35.6314f,0.0f}, - {54.679f,0.0f}, - {64.2029f,4.7619f}, - {73.7267f,14.2857f}, - {78.4886f,23.8095f}, - {78.4886f,38.0952f} -}; - -static const SFG_StrokeVertex ch71st1[] = -{ - {54.679f,38.0952f}, - {78.4886f,38.0952f} -}; - -static const SFG_StrokeStrip ch71st[] = -{ - {19,ch71st0}, - {2,ch71st1} -}; - -static const SFG_StrokeChar ch71 = {89.7686f,2,ch71st}; - -/* char: 0x48 */ - -static const SFG_StrokeVertex ch72st0[] = -{ - {11.42f,100.0f}, - {11.42f,0.0f} -}; - -static const SFG_StrokeVertex ch72st1[] = -{ - {78.0867f,100.0f}, - {78.0867f,0.0f} -}; - -static const SFG_StrokeVertex ch72st2[] = -{ - {11.42f,52.381f}, - {78.0867f,52.381f} -}; - -static const SFG_StrokeStrip ch72st[] = -{ - {2,ch72st0}, - {2,ch72st1}, - {2,ch72st2} -}; - -static const SFG_StrokeChar ch72 = {89.0867f,3,ch72st}; - -/* char: 0x49 */ - -static const SFG_StrokeVertex ch73st0[] = -{ - {10.86f,100.0f}, - {10.86f,0.0f} -}; - -static const SFG_StrokeStrip ch73st[] = -{ - {2,ch73st0} -}; - -static const SFG_StrokeChar ch73 = {21.3f,1,ch73st}; - -/* char: 0x4a */ - -static const SFG_StrokeVertex ch74st0[] = -{ - {50.119f,100.0f}, - {50.119f,23.8095f}, - {45.3571f,9.5238f}, - {40.5952f,4.7619f}, - {31.0714f,0.0f}, - {21.5476f,0.0f}, - {12.0238f,4.7619f}, - {7.2619f,9.5238f}, - {2.5f,23.8095f}, - {2.5f,33.3333f} -}; - -static const SFG_StrokeStrip ch74st[] = -{ - {10,ch74st0} -}; - -static const SFG_StrokeChar ch74 = {59.999f,1,ch74st}; - -/* char: 0x4b */ - -static const SFG_StrokeVertex ch75st0[] = -{ - {11.28f,100.0f}, - {11.28f,0.0f} -}; - -static const SFG_StrokeVertex ch75st1[] = -{ - {77.9467f,100.0f}, - {11.28f,33.3333f} -}; - -static const SFG_StrokeVertex ch75st2[] = -{ - {35.0895f,57.1429f}, - {77.9467f,0.0f} -}; - -static const SFG_StrokeStrip ch75st[] = -{ - {2,ch75st0}, - {2,ch75st1}, - {2,ch75st2} -}; - -static const SFG_StrokeChar ch75 = {79.3267f,3,ch75st}; - -/* char: 0x4c */ - -static const SFG_StrokeVertex ch76st0[] = -{ - {11.68f,100.0f}, - {11.68f,0.0f} -}; - -static const SFG_StrokeVertex ch76st1[] = -{ - {11.68f,0.0f}, - {68.8229f,0.0f} -}; - -static const SFG_StrokeStrip ch76st[] = -{ - {2,ch76st0}, - {2,ch76st1} -}; - -static const SFG_StrokeChar ch76 = {71.3229f,2,ch76st}; - -/* char: 0x4d */ - -static const SFG_StrokeVertex ch77st0[] = -{ - {10.86f,100.0f}, - {10.86f,0.0f} -}; - -static const SFG_StrokeVertex ch77st1[] = -{ - {10.86f,100.0f}, - {48.9552f,0.0f} -}; - -static const SFG_StrokeVertex ch77st2[] = -{ - {87.0505f,100.0f}, - {48.9552f,0.0f} -}; - -static const SFG_StrokeVertex ch77st3[] = -{ - {87.0505f,100.0f}, - {87.0505f,0.0f} -}; - -static const SFG_StrokeStrip ch77st[] = -{ - {2,ch77st0}, - {2,ch77st1}, - {2,ch77st2}, - {2,ch77st3} -}; - -static const SFG_StrokeChar ch77 = {97.2105f,4,ch77st}; - -/* char: 0x4e */ - -static const SFG_StrokeVertex ch78st0[] = -{ - {11.14f,100.0f}, - {11.14f,0.0f} -}; - -static const SFG_StrokeVertex ch78st1[] = -{ - {11.14f,100.0f}, - {77.8067f,0.0f} -}; - -static const SFG_StrokeVertex ch78st2[] = -{ - {77.8067f,100.0f}, - {77.8067f,0.0f} -}; - -static const SFG_StrokeStrip ch78st[] = -{ - {2,ch78st0}, - {2,ch78st1}, - {2,ch78st2} -}; - -static const SFG_StrokeChar ch78 = {88.8067f,3,ch78st}; - -/* char: 0x4f */ - -static const SFG_StrokeVertex ch79st0[] = -{ - {34.8114f,100.0f}, - {25.2876f,95.2381f}, - {15.7638f,85.7143f}, - {11.0019f,76.1905f}, - {6.24f,61.9048f}, - {6.24f,38.0952f}, - {11.0019f,23.8095f}, - {15.7638f,14.2857f}, - {25.2876f,4.7619f}, - {34.8114f,0.0f}, - {53.859f,0.0f}, - {63.3829f,4.7619f}, - {72.9067f,14.2857f}, - {77.6686f,23.8095f}, - {82.4305f,38.0952f}, - {82.4305f,61.9048f}, - {77.6686f,76.1905f}, - {72.9067f,85.7143f}, - {63.3829f,95.2381f}, - {53.859f,100.0f}, - {34.8114f,100.0f} -}; - -static const SFG_StrokeStrip ch79st[] = -{ - {21,ch79st0} -}; - -static const SFG_StrokeChar ch79 = {88.8305f,1,ch79st}; - -/* char: 0x50 */ - -static const SFG_StrokeVertex ch80st0[] = -{ - {12.1f,100.0f}, - {12.1f,0.0f} -}; - -static const SFG_StrokeVertex ch80st1[] = -{ - {12.1f,100.0f}, - {54.9571f,100.0f}, - {69.2429f,95.2381f}, - {74.0048f,90.4762f}, - {78.7667f,80.9524f}, - {78.7667f,66.6667f}, - {74.0048f,57.1429f}, - {69.2429f,52.381f}, - {54.9571f,47.619f}, - {12.1f,47.619f} -}; - -static const SFG_StrokeStrip ch80st[] = -{ - {2,ch80st0}, - {10,ch80st1} -}; - -static const SFG_StrokeChar ch80 = {85.6667f,2,ch80st}; - -/* char: 0x51 */ - -static const SFG_StrokeVertex ch81st0[] = -{ - {33.8714f,100.0f}, - {24.3476f,95.2381f}, - {14.8238f,85.7143f}, - {10.0619f,76.1905f}, - {5.3f,61.9048f}, - {5.3f,38.0952f}, - {10.0619f,23.8095f}, - {14.8238f,14.2857f}, - {24.3476f,4.7619f}, - {33.8714f,0.0f}, - {52.919f,0.0f}, - {62.4429f,4.7619f}, - {71.9667f,14.2857f}, - {76.7286f,23.8095f}, - {81.4905f,38.0952f}, - {81.4905f,61.9048f}, - {76.7286f,76.1905f}, - {71.9667f,85.7143f}, - {62.4429f,95.2381f}, - {52.919f,100.0f}, - {33.8714f,100.0f} -}; - -static const SFG_StrokeVertex ch81st1[] = -{ - {48.1571f,19.0476f}, - {76.7286f,-9.5238f} -}; - -static const SFG_StrokeStrip ch81st[] = -{ - {21,ch81st0}, - {2,ch81st1} -}; - -static const SFG_StrokeChar ch81 = {88.0905f,2,ch81st}; - -/* char: 0x52 */ - -static const SFG_StrokeVertex ch82st0[] = -{ - {11.68f,100.0f}, - {11.68f,0.0f} -}; - -static const SFG_StrokeVertex ch82st1[] = -{ - {11.68f,100.0f}, - {54.5371f,100.0f}, - {68.8229f,95.2381f}, - {73.5848f,90.4762f}, - {78.3467f,80.9524f}, - {78.3467f,71.4286f}, - {73.5848f,61.9048f}, - {68.8229f,57.1429f}, - {54.5371f,52.381f}, - {11.68f,52.381f} -}; - -static const SFG_StrokeVertex ch82st2[] = -{ - {45.0133f,52.381f}, - {78.3467f,0.0f} -}; - -static const SFG_StrokeStrip ch82st[] = -{ - {2,ch82st0}, - {10,ch82st1}, - {2,ch82st2} -}; - -static const SFG_StrokeChar ch82 = {82.3667f,3,ch82st}; - -/* char: 0x53 */ - -static const SFG_StrokeVertex ch83st0[] = -{ - {74.6667f,85.7143f}, - {65.1429f,95.2381f}, - {50.8571f,100.0f}, - {31.8095f,100.0f}, - {17.5238f,95.2381f}, - {8.0f,85.7143f}, - {8.0f,76.1905f}, - {12.7619f,66.6667f}, - {17.5238f,61.9048f}, - {27.0476f,57.1429f}, - {55.619f,47.619f}, - {65.1429f,42.8571f}, - {69.9048f,38.0952f}, - {74.6667f,28.5714f}, - {74.6667f,14.2857f}, - {65.1429f,4.7619f}, - {50.8571f,0.0f}, - {31.8095f,0.0f}, - {17.5238f,4.7619f}, - {8.0f,14.2857f} -}; - -static const SFG_StrokeStrip ch83st[] = -{ - {20,ch83st0} -}; - -static const SFG_StrokeChar ch83 = {80.8267f,1,ch83st}; - -/* char: 0x54 */ - -static const SFG_StrokeVertex ch84st0[] = -{ - {35.6933f,100.0f}, - {35.6933f,0.0f} -}; - -static const SFG_StrokeVertex ch84st1[] = -{ - {2.36f,100.0f}, - {69.0267f,100.0f} -}; - -static const SFG_StrokeStrip ch84st[] = -{ - {2,ch84st0}, - {2,ch84st1} -}; - -static const SFG_StrokeChar ch84 = {71.9467f,2,ch84st}; - -/* char: 0x55 */ - -static const SFG_StrokeVertex ch85st0[] = -{ - {11.54f,100.0f}, - {11.54f,28.5714f}, - {16.3019f,14.2857f}, - {25.8257f,4.7619f}, - {40.1114f,0.0f}, - {49.6352f,0.0f}, - {63.921f,4.7619f}, - {73.4448f,14.2857f}, - {78.2067f,28.5714f}, - {78.2067f,100.0f} -}; - -static const SFG_StrokeStrip ch85st[] = -{ - {10,ch85st0} -}; - -static const SFG_StrokeChar ch85 = {89.4867f,1,ch85st}; - -/* char: 0x56 */ - -static const SFG_StrokeVertex ch86st0[] = -{ - {2.36f,100.0f}, - {40.4552f,0.0f} -}; - -static const SFG_StrokeVertex ch86st1[] = -{ - {78.5505f,100.0f}, - {40.4552f,0.0f} -}; - -static const SFG_StrokeStrip ch86st[] = -{ - {2,ch86st0}, - {2,ch86st1} -}; - -static const SFG_StrokeChar ch86 = {81.6105f,2,ch86st}; - -/* char: 0x57 */ - -static const SFG_StrokeVertex ch87st0[] = -{ - {2.22f,100.0f}, - {26.0295f,0.0f} -}; - -static const SFG_StrokeVertex ch87st1[] = -{ - {49.839f,100.0f}, - {26.0295f,0.0f} -}; - -static const SFG_StrokeVertex ch87st2[] = -{ - {49.839f,100.0f}, - {73.6486f,0.0f} -}; - -static const SFG_StrokeVertex ch87st3[] = -{ - {97.4581f,100.0f}, - {73.6486f,0.0f} -}; - -static const SFG_StrokeStrip ch87st[] = -{ - {2,ch87st0}, - {2,ch87st1}, - {2,ch87st2}, - {2,ch87st3} -}; - -static const SFG_StrokeChar ch87 = {100.518f,4,ch87st}; - -/* char: 0x58 */ - -static const SFG_StrokeVertex ch88st0[] = -{ - {2.5f,100.0f}, - {69.1667f,0.0f} -}; - -static const SFG_StrokeVertex ch88st1[] = -{ - {69.1667f,100.0f}, - {2.5f,0.0f} -}; - -static const SFG_StrokeStrip ch88st[] = -{ - {2,ch88st0}, - {2,ch88st1} -}; - -static const SFG_StrokeChar ch88 = {72.3667f,2,ch88st}; - -/* char: 0x59 */ - -static const SFG_StrokeVertex ch89st0[] = -{ - {1.52f,100.0f}, - {39.6152f,52.381f}, - {39.6152f,0.0f} -}; - -static const SFG_StrokeVertex ch89st1[] = -{ - {77.7105f,100.0f}, - {39.6152f,52.381f} -}; - -static const SFG_StrokeStrip ch89st[] = -{ - {3,ch89st0}, - {2,ch89st1} -}; - -static const SFG_StrokeChar ch89 = {79.6505f,2,ch89st}; - -/* char: 0x5a */ - -static const SFG_StrokeVertex ch90st0[] = -{ - {69.1667f,100.0f}, - {2.5f,0.0f} -}; - -static const SFG_StrokeVertex ch90st1[] = -{ - {2.5f,100.0f}, - {69.1667f,100.0f} -}; - -static const SFG_StrokeVertex ch90st2[] = -{ - {2.5f,0.0f}, - {69.1667f,0.0f} -}; - -static const SFG_StrokeStrip ch90st[] = -{ - {2,ch90st0}, - {2,ch90st1}, - {2,ch90st2} -}; - -static const SFG_StrokeChar ch90 = {73.7467f,3,ch90st}; - -/* char: 0x5b */ - -static const SFG_StrokeVertex ch91st0[] = -{ - {7.78f,119.048f}, - {7.78f,-33.3333f} -}; - -static const SFG_StrokeVertex ch91st1[] = -{ - {12.5419f,119.048f}, - {12.5419f,-33.3333f} -}; - -static const SFG_StrokeVertex ch91st2[] = -{ - {7.78f,119.048f}, - {41.1133f,119.048f} -}; - -static const SFG_StrokeVertex ch91st3[] = -{ - {7.78f,-33.3333f}, - {41.1133f,-33.3333f} -}; - -static const SFG_StrokeStrip ch91st[] = -{ - {2,ch91st0}, - {2,ch91st1}, - {2,ch91st2}, - {2,ch91st3} -}; - -static const SFG_StrokeChar ch91 = {46.1133f,4,ch91st}; - -/* char: 0x5c */ - -static const SFG_StrokeVertex ch92st0[] = -{ - {5.84f,100.0f}, - {72.5067f,-14.2857f} -}; - -static const SFG_StrokeStrip ch92st[] = -{ - {2,ch92st0} -}; - -static const SFG_StrokeChar ch92 = {78.2067f,1,ch92st}; - -/* char: 0x5d */ - -static const SFG_StrokeVertex ch93st0[] = -{ - {33.0114f,119.048f}, - {33.0114f,-33.3333f} -}; - -static const SFG_StrokeVertex ch93st1[] = -{ - {37.7733f,119.048f}, - {37.7733f,-33.3333f} -}; - -static const SFG_StrokeVertex ch93st2[] = -{ - {4.44f,119.048f}, - {37.7733f,119.048f} -}; - -static const SFG_StrokeVertex ch93st3[] = -{ - {4.44f,-33.3333f}, - {37.7733f,-33.3333f} -}; - -static const SFG_StrokeStrip ch93st[] = -{ - {2,ch93st0}, - {2,ch93st1}, - {2,ch93st2}, - {2,ch93st3} -}; - -static const SFG_StrokeChar ch93 = {46.3933f,4,ch93st}; - -/* char: 0x5e */ - -static const SFG_StrokeVertex ch94st0[] = -{ - {44.0752f,109.524f}, - {5.98f,42.8571f} -}; - -static const SFG_StrokeVertex ch94st1[] = -{ - {44.0752f,109.524f}, - {82.1705f,42.8571f} -}; - -static const SFG_StrokeStrip ch94st[] = -{ - {2,ch94st0}, - {2,ch94st1} -}; - -static const SFG_StrokeChar ch94 = {90.2305f,2,ch94st}; - -/* char: 0x5f */ - -static const SFG_StrokeVertex ch95st0[] = -{ - {-1.1f,-33.3333f}, - {103.662f,-33.3333f}, - {103.662f,-28.5714f}, - {-1.1f,-28.5714f}, - {-1.1f,-33.3333f} -}; - -static const SFG_StrokeStrip ch95st[] = -{ - {5,ch95st0} -}; - -static const SFG_StrokeChar ch95 = {104.062f,1,ch95st}; - -/* char: 0x60 */ - -static const SFG_StrokeVertex ch96st0[] = -{ - {33.0219f,100.0f}, - {56.8314f,71.4286f} -}; - -static const SFG_StrokeVertex ch96st1[] = -{ - {33.0219f,100.0f}, - {28.26f,95.2381f}, - {56.8314f,71.4286f} -}; - -static const SFG_StrokeStrip ch96st[] = -{ - {2,ch96st0}, - {3,ch96st1} -}; - -static const SFG_StrokeChar ch96 = {83.5714f,2,ch96st}; - -/* char: 0x61 */ - -static const SFG_StrokeVertex ch97st0[] = -{ - {63.8229f,66.6667f}, - {63.8229f,0.0f} -}; - -static const SFG_StrokeVertex ch97st1[] = -{ - {63.8229f,52.381f}, - {54.299f,61.9048f}, - {44.7752f,66.6667f}, - {30.4895f,66.6667f}, - {20.9657f,61.9048f}, - {11.4419f,52.381f}, - {6.68f,38.0952f}, - {6.68f,28.5714f}, - {11.4419f,14.2857f}, - {20.9657f,4.7619f}, - {30.4895f,0.0f}, - {44.7752f,0.0f}, - {54.299f,4.7619f}, - {63.8229f,14.2857f} -}; - -static const SFG_StrokeStrip ch97st[] = -{ - {2,ch97st0}, - {14,ch97st1} -}; - -static const SFG_StrokeChar ch97 = {66.6029f,2,ch97st}; - -/* char: 0x62 */ - -static const SFG_StrokeVertex ch98st0[] = -{ - {8.76f,100.0f}, - {8.76f,0.0f} -}; - -static const SFG_StrokeVertex ch98st1[] = -{ - {8.76f,52.381f}, - {18.2838f,61.9048f}, - {27.8076f,66.6667f}, - {42.0933f,66.6667f}, - {51.6171f,61.9048f}, - {61.141f,52.381f}, - {65.9029f,38.0952f}, - {65.9029f,28.5714f}, - {61.141f,14.2857f}, - {51.6171f,4.7619f}, - {42.0933f,0.0f}, - {27.8076f,0.0f}, - {18.2838f,4.7619f}, - {8.76f,14.2857f} -}; - -static const SFG_StrokeStrip ch98st[] = -{ - {2,ch98st0}, - {14,ch98st1} -}; - -static const SFG_StrokeChar ch98 = {70.4629f,2,ch98st}; - -/* char: 0x63 */ - -static const SFG_StrokeVertex ch99st0[] = -{ - {62.6629f,52.381f}, - {53.139f,61.9048f}, - {43.6152f,66.6667f}, - {29.3295f,66.6667f}, - {19.8057f,61.9048f}, - {10.2819f,52.381f}, - {5.52f,38.0952f}, - {5.52f,28.5714f}, - {10.2819f,14.2857f}, - {19.8057f,4.7619f}, - {29.3295f,0.0f}, - {43.6152f,0.0f}, - {53.139f,4.7619f}, - {62.6629f,14.2857f} -}; - -static const SFG_StrokeStrip ch99st[] = -{ - {14,ch99st0} -}; - -static const SFG_StrokeChar ch99 = {68.9229f,1,ch99st}; - -/* char: 0x64 */ - -static const SFG_StrokeVertex ch100st0[] = -{ - {61.7829f,100.0f}, - {61.7829f,0.0f} -}; - -static const SFG_StrokeVertex ch100st1[] = -{ - {61.7829f,52.381f}, - {52.259f,61.9048f}, - {42.7352f,66.6667f}, - {28.4495f,66.6667f}, - {18.9257f,61.9048f}, - {9.4019f,52.381f}, - {4.64f,38.0952f}, - {4.64f,28.5714f}, - {9.4019f,14.2857f}, - {18.9257f,4.7619f}, - {28.4495f,0.0f}, - {42.7352f,0.0f}, - {52.259f,4.7619f}, - {61.7829f,14.2857f} -}; - -static const SFG_StrokeStrip ch100st[] = -{ - {2,ch100st0}, - {14,ch100st1} -}; - -static const SFG_StrokeChar ch100 = {70.2629f,2,ch100st}; - -/* char: 0x65 */ - -static const SFG_StrokeVertex ch101st0[] = -{ - {5.72f,38.0952f}, - {62.8629f,38.0952f}, - {62.8629f,47.619f}, - {58.101f,57.1429f}, - {53.339f,61.9048f}, - {43.8152f,66.6667f}, - {29.5295f,66.6667f}, - {20.0057f,61.9048f}, - {10.4819f,52.381f}, - {5.72f,38.0952f}, - {5.72f,28.5714f}, - {10.4819f,14.2857f}, - {20.0057f,4.7619f}, - {29.5295f,0.0f}, - {43.8152f,0.0f}, - {53.339f,4.7619f}, - {62.8629f,14.2857f} -}; - -static const SFG_StrokeStrip ch101st[] = -{ - {17,ch101st0} -}; - -static const SFG_StrokeChar ch101 = {68.5229f,1,ch101st}; - -/* char: 0x66 */ - -static const SFG_StrokeVertex ch102st0[] = -{ - {38.7752f,100.0f}, - {29.2514f,100.0f}, - {19.7276f,95.2381f}, - {14.9657f,80.9524f}, - {14.9657f,0.0f} -}; - -static const SFG_StrokeVertex ch102st1[] = -{ - {0.68f,66.6667f}, - {34.0133f,66.6667f} -}; - -static const SFG_StrokeStrip ch102st[] = -{ - {5,ch102st0}, - {2,ch102st1} -}; - -static const SFG_StrokeChar ch102 = {38.6552f,2,ch102st}; - -/* char: 0x67 */ - -static const SFG_StrokeVertex ch103st0[] = -{ - {62.5029f,66.6667f}, - {62.5029f,-9.5238f}, - {57.741f,-23.8095f}, - {52.979f,-28.5714f}, - {43.4552f,-33.3333f}, - {29.1695f,-33.3333f}, - {19.6457f,-28.5714f} -}; - -static const SFG_StrokeVertex ch103st1[] = -{ - {62.5029f,52.381f}, - {52.979f,61.9048f}, - {43.4552f,66.6667f}, - {29.1695f,66.6667f}, - {19.6457f,61.9048f}, - {10.1219f,52.381f}, - {5.36f,38.0952f}, - {5.36f,28.5714f}, - {10.1219f,14.2857f}, - {19.6457f,4.7619f}, - {29.1695f,0.0f}, - {43.4552f,0.0f}, - {52.979f,4.7619f}, - {62.5029f,14.2857f} -}; - -static const SFG_StrokeStrip ch103st[] = -{ - {7,ch103st0}, - {14,ch103st1} -}; - -static const SFG_StrokeChar ch103 = {70.9829f,2,ch103st}; - -/* char: 0x68 */ - -static const SFG_StrokeVertex ch104st0[] = -{ - {9.6f,100.0f}, - {9.6f,0.0f} -}; - -static const SFG_StrokeVertex ch104st1[] = -{ - {9.6f,47.619f}, - {23.8857f,61.9048f}, - {33.4095f,66.6667f}, - {47.6952f,66.6667f}, - {57.219f,61.9048f}, - {61.981f,47.619f}, - {61.981f,0.0f} -}; - -static const SFG_StrokeStrip ch104st[] = -{ - {2,ch104st0}, - {7,ch104st1} -}; - -static const SFG_StrokeChar ch104 = {71.021f,2,ch104st}; - -/* char: 0x69 */ - -static const SFG_StrokeVertex ch105st0[] = -{ - {10.02f,100.0f}, - {14.7819f,95.2381f}, - {19.5438f,100.0f}, - {14.7819f,104.762f}, - {10.02f,100.0f} -}; - -static const SFG_StrokeVertex ch105st1[] = -{ - {14.7819f,66.6667f}, - {14.7819f,0.0f} -}; - -static const SFG_StrokeStrip ch105st[] = -{ - {5,ch105st0}, - {2,ch105st1} -}; - -static const SFG_StrokeChar ch105 = {28.8638f,2,ch105st}; - -/* char: 0x6a */ - -static const SFG_StrokeVertex ch106st0[] = -{ - {17.3876f,100.0f}, - {22.1495f,95.2381f}, - {26.9114f,100.0f}, - {22.1495f,104.762f}, - {17.3876f,100.0f} -}; - -static const SFG_StrokeVertex ch106st1[] = -{ - {22.1495f,66.6667f}, - {22.1495f,-14.2857f}, - {17.3876f,-28.5714f}, - {7.8638f,-33.3333f}, - {-1.66f,-33.3333f} -}; - -static const SFG_StrokeStrip ch106st[] = -{ - {5,ch106st0}, - {5,ch106st1} -}; - -static const SFG_StrokeChar ch106 = {36.2314f,2,ch106st}; - -/* char: 0x6b */ - -static const SFG_StrokeVertex ch107st0[] = -{ - {9.6f,100.0f}, - {9.6f,0.0f} -}; - -static const SFG_StrokeVertex ch107st1[] = -{ - {57.219f,66.6667f}, - {9.6f,19.0476f} -}; - -static const SFG_StrokeVertex ch107st2[] = -{ - {28.6476f,38.0952f}, - {61.981f,0.0f} -}; - -static const SFG_StrokeStrip ch107st[] = -{ - {2,ch107st0}, - {2,ch107st1}, - {2,ch107st2} -}; - -static const SFG_StrokeChar ch107 = {62.521f,3,ch107st}; - -/* char: 0x6c */ - -static const SFG_StrokeVertex ch108st0[] = -{ - {10.02f,100.0f}, - {10.02f,0.0f} -}; - -static const SFG_StrokeStrip ch108st[] = -{ - {2,ch108st0} -}; - -static const SFG_StrokeChar ch108 = {19.34f,1,ch108st}; - -/* char: 0x6d */ - -static const SFG_StrokeVertex ch109st0[] = -{ - {9.6f,66.6667f}, - {9.6f,0.0f} -}; - -static const SFG_StrokeVertex ch109st1[] = -{ - {9.6f,47.619f}, - {23.8857f,61.9048f}, - {33.4095f,66.6667f}, - {47.6952f,66.6667f}, - {57.219f,61.9048f}, - {61.981f,47.619f}, - {61.981f,0.0f} -}; - -static const SFG_StrokeVertex ch109st2[] = -{ - {61.981f,47.619f}, - {76.2667f,61.9048f}, - {85.7905f,66.6667f}, - {100.076f,66.6667f}, - {109.6f,61.9048f}, - {114.362f,47.619f}, - {114.362f,0.0f} -}; - -static const SFG_StrokeStrip ch109st[] = -{ - {2,ch109st0}, - {7,ch109st1}, - {7,ch109st2} -}; - -static const SFG_StrokeChar ch109 = {123.962f,3,ch109st}; - -/* char: 0x6e */ - -static const SFG_StrokeVertex ch110st0[] = -{ - {9.18f,66.6667f}, - {9.18f,0.0f} -}; - -static const SFG_StrokeVertex ch110st1[] = -{ - {9.18f,47.619f}, - {23.4657f,61.9048f}, - {32.9895f,66.6667f}, - {47.2752f,66.6667f}, - {56.799f,61.9048f}, - {61.561f,47.619f}, - {61.561f,0.0f} -}; - -static const SFG_StrokeStrip ch110st[] = -{ - {2,ch110st0}, - {7,ch110st1} -}; - -static const SFG_StrokeChar ch110 = {70.881f,2,ch110st}; - -/* char: 0x6f */ - -static const SFG_StrokeVertex ch111st0[] = -{ - {28.7895f,66.6667f}, - {19.2657f,61.9048f}, - {9.7419f,52.381f}, - {4.98f,38.0952f}, - {4.98f,28.5714f}, - {9.7419f,14.2857f}, - {19.2657f,4.7619f}, - {28.7895f,0.0f}, - {43.0752f,0.0f}, - {52.599f,4.7619f}, - {62.1229f,14.2857f}, - {66.8848f,28.5714f}, - {66.8848f,38.0952f}, - {62.1229f,52.381f}, - {52.599f,61.9048f}, - {43.0752f,66.6667f}, - {28.7895f,66.6667f} -}; - -static const SFG_StrokeStrip ch111st[] = -{ - {17,ch111st0} -}; - -static const SFG_StrokeChar ch111 = {71.7448f,1,ch111st}; - -/* char: 0x70 */ - -static const SFG_StrokeVertex ch112st0[] = -{ - {9.46f,66.6667f}, - {9.46f,-33.3333f} -}; - -static const SFG_StrokeVertex ch112st1[] = -{ - {9.46f,52.381f}, - {18.9838f,61.9048f}, - {28.5076f,66.6667f}, - {42.7933f,66.6667f}, - {52.3171f,61.9048f}, - {61.841f,52.381f}, - {66.6029f,38.0952f}, - {66.6029f,28.5714f}, - {61.841f,14.2857f}, - {52.3171f,4.7619f}, - {42.7933f,0.0f}, - {28.5076f,0.0f}, - {18.9838f,4.7619f}, - {9.46f,14.2857f} -}; - -static const SFG_StrokeStrip ch112st[] = -{ - {2,ch112st0}, - {14,ch112st1} -}; - -static const SFG_StrokeChar ch112 = {70.8029f,2,ch112st}; - -/* char: 0x71 */ - -static const SFG_StrokeVertex ch113st0[] = -{ - {61.9829f,66.6667f}, - {61.9829f,-33.3333f} -}; - -static const SFG_StrokeVertex ch113st1[] = -{ - {61.9829f,52.381f}, - {52.459f,61.9048f}, - {42.9352f,66.6667f}, - {28.6495f,66.6667f}, - {19.1257f,61.9048f}, - {9.6019f,52.381f}, - {4.84f,38.0952f}, - {4.84f,28.5714f}, - {9.6019f,14.2857f}, - {19.1257f,4.7619f}, - {28.6495f,0.0f}, - {42.9352f,0.0f}, - {52.459f,4.7619f}, - {61.9829f,14.2857f} -}; - -static const SFG_StrokeStrip ch113st[] = -{ - {2,ch113st0}, - {14,ch113st1} -}; - -static const SFG_StrokeChar ch113 = {70.7429f,2,ch113st}; - -/* char: 0x72 */ - -static const SFG_StrokeVertex ch114st0[] = -{ - {9.46f,66.6667f}, - {9.46f,0.0f} -}; - -static const SFG_StrokeVertex ch114st1[] = -{ - {9.46f,38.0952f}, - {14.2219f,52.381f}, - {23.7457f,61.9048f}, - {33.2695f,66.6667f}, - {47.5552f,66.6667f} -}; - -static const SFG_StrokeStrip ch114st[] = -{ - {2,ch114st0}, - {5,ch114st1} -}; - -static const SFG_StrokeChar ch114 = {49.4952f,2,ch114st}; - -/* char: 0x73 */ - -static const SFG_StrokeVertex ch115st0[] = -{ - {57.081f,52.381f}, - {52.319f,61.9048f}, - {38.0333f,66.6667f}, - {23.7476f,66.6667f}, - {9.4619f,61.9048f}, - {4.7f,52.381f}, - {9.4619f,42.8571f}, - {18.9857f,38.0952f}, - {42.7952f,33.3333f}, - {52.319f,28.5714f}, - {57.081f,19.0476f}, - {57.081f,14.2857f}, - {52.319f,4.7619f}, - {38.0333f,0.0f}, - {23.7476f,0.0f}, - {9.4619f,4.7619f}, - {4.7f,14.2857f} -}; - -static const SFG_StrokeStrip ch115st[] = -{ - {17,ch115st0} -}; - -static const SFG_StrokeChar ch115 = {62.321f,1,ch115st}; - -/* char: 0x74 */ - -static const SFG_StrokeVertex ch116st0[] = -{ - {14.8257f,100.0f}, - {14.8257f,19.0476f}, - {19.5876f,4.7619f}, - {29.1114f,0.0f}, - {38.6352f,0.0f} -}; - -static const SFG_StrokeVertex ch116st1[] = -{ - {0.54f,66.6667f}, - {33.8733f,66.6667f} -}; - -static const SFG_StrokeStrip ch116st[] = -{ - {5,ch116st0}, - {2,ch116st1} -}; - -static const SFG_StrokeChar ch116 = {39.3152f,2,ch116st}; - -/* char: 0x75 */ - -static const SFG_StrokeVertex ch117st0[] = -{ - {9.46f,66.6667f}, - {9.46f,19.0476f}, - {14.2219f,4.7619f}, - {23.7457f,0.0f}, - {38.0314f,0.0f}, - {47.5552f,4.7619f}, - {61.841f,19.0476f} -}; - -static const SFG_StrokeVertex ch117st1[] = -{ - {61.841f,66.6667f}, - {61.841f,0.0f} -}; - -static const SFG_StrokeStrip ch117st[] = -{ - {7,ch117st0}, - {2,ch117st1} -}; - -static const SFG_StrokeChar ch117 = {71.161f,2,ch117st}; - -/* char: 0x76 */ - -static const SFG_StrokeVertex ch118st0[] = -{ - {1.8f,66.6667f}, - {30.3714f,0.0f} -}; - -static const SFG_StrokeVertex ch118st1[] = -{ - {58.9429f,66.6667f}, - {30.3714f,0.0f} -}; - -static const SFG_StrokeStrip ch118st[] = -{ - {2,ch118st0}, - {2,ch118st1} -}; - -static const SFG_StrokeChar ch118 = {60.6029f,2,ch118st}; - -/* char: 0x77 */ - -static const SFG_StrokeVertex ch119st0[] = -{ - {2.5f,66.6667f}, - {21.5476f,0.0f} -}; - -static const SFG_StrokeVertex ch119st1[] = -{ - {40.5952f,66.6667f}, - {21.5476f,0.0f} -}; - -static const SFG_StrokeVertex ch119st2[] = -{ - {40.5952f,66.6667f}, - {59.6429f,0.0f} -}; - -static const SFG_StrokeVertex ch119st3[] = -{ - {78.6905f,66.6667f}, - {59.6429f,0.0f} -}; - -static const SFG_StrokeStrip ch119st[] = -{ - {2,ch119st0}, - {2,ch119st1}, - {2,ch119st2}, - {2,ch119st3} -}; - -static const SFG_StrokeChar ch119 = {80.4905f,4,ch119st}; - -/* char: 0x78 */ - -static const SFG_StrokeVertex ch120st0[] = -{ - {1.66f,66.6667f}, - {54.041f,0.0f} -}; - -static const SFG_StrokeVertex ch120st1[] = -{ - {54.041f,66.6667f}, - {1.66f,0.0f} -}; - -static const SFG_StrokeStrip ch120st[] = -{ - {2,ch120st0}, - {2,ch120st1} -}; - -static const SFG_StrokeChar ch120 = {56.401f,2,ch120st}; - -/* char: 0x79 */ - -static const SFG_StrokeVertex ch121st0[] = -{ - {6.5619f,66.6667f}, - {35.1333f,0.0f} -}; - -static const SFG_StrokeVertex ch121st1[] = -{ - {63.7048f,66.6667f}, - {35.1333f,0.0f}, - {25.6095f,-19.0476f}, - {16.0857f,-28.5714f}, - {6.5619f,-33.3333f}, - {1.8f,-33.3333f} -}; - -static const SFG_StrokeStrip ch121st[] = -{ - {2,ch121st0}, - {6,ch121st1} -}; - -static const SFG_StrokeChar ch121 = {66.0648f,2,ch121st}; - -/* char: 0x7a */ - -static const SFG_StrokeVertex ch122st0[] = -{ - {56.821f,66.6667f}, - {4.44f,0.0f} -}; - -static const SFG_StrokeVertex ch122st1[] = -{ - {4.44f,66.6667f}, - {56.821f,66.6667f} -}; - -static const SFG_StrokeVertex ch122st2[] = -{ - {4.44f,0.0f}, - {56.821f,0.0f} -}; - -static const SFG_StrokeStrip ch122st[] = -{ - {2,ch122st0}, - {2,ch122st1}, - {2,ch122st2} -}; - -static const SFG_StrokeChar ch122 = {61.821f,3,ch122st}; - -/* char: 0x7b */ - -static const SFG_StrokeVertex ch123st0[] = -{ - {31.1895f,119.048f}, - {21.6657f,114.286f}, - {16.9038f,109.524f}, - {12.1419f,100.0f}, - {12.1419f,90.4762f}, - {16.9038f,80.9524f}, - {21.6657f,76.1905f}, - {26.4276f,66.6667f}, - {26.4276f,57.1429f}, - {16.9038f,47.619f} -}; - -static const SFG_StrokeVertex ch123st1[] = -{ - {21.6657f,114.286f}, - {16.9038f,104.762f}, - {16.9038f,95.2381f}, - {21.6657f,85.7143f}, - {26.4276f,80.9524f}, - {31.1895f,71.4286f}, - {31.1895f,61.9048f}, - {26.4276f,52.381f}, - {7.38f,42.8571f}, - {26.4276f,33.3333f}, - {31.1895f,23.8095f}, - {31.1895f,14.2857f}, - {26.4276f,4.7619f}, - {21.6657f,0.0f}, - {16.9038f,-9.5238f}, - {16.9038f,-19.0476f}, - {21.6657f,-28.5714f} -}; - -static const SFG_StrokeVertex ch123st2[] = -{ - {16.9038f,38.0952f}, - {26.4276f,28.5714f}, - {26.4276f,19.0476f}, - {21.6657f,9.5238f}, - {16.9038f,4.7619f}, - {12.1419f,-4.7619f}, - {12.1419f,-14.2857f}, - {16.9038f,-23.8095f}, - {21.6657f,-28.5714f}, - {31.1895f,-33.3333f} -}; - -static const SFG_StrokeStrip ch123st[] = -{ - {10,ch123st0}, - {17,ch123st1}, - {10,ch123st2} -}; - -static const SFG_StrokeChar ch123 = {41.6295f,3,ch123st}; - -/* char: 0x7c */ - -static const SFG_StrokeVertex ch124st0[] = -{ - {11.54f,119.048f}, - {11.54f,-33.3333f} -}; - -static const SFG_StrokeStrip ch124st[] = -{ - {2,ch124st0} -}; - -static const SFG_StrokeChar ch124 = {23.78f,1,ch124st}; - -/* char: 0x7d */ - -static const SFG_StrokeVertex ch125st0[] = -{ - {9.18f,119.048f}, - {18.7038f,114.286f}, - {23.4657f,109.524f}, - {28.2276f,100.0f}, - {28.2276f,90.4762f}, - {23.4657f,80.9524f}, - {18.7038f,76.1905f}, - {13.9419f,66.6667f}, - {13.9419f,57.1429f}, - {23.4657f,47.619f} -}; - -static const SFG_StrokeVertex ch125st1[] = -{ - {18.7038f,114.286f}, - {23.4657f,104.762f}, - {23.4657f,95.2381f}, - {18.7038f,85.7143f}, - {13.9419f,80.9524f}, - {9.18f,71.4286f}, - {9.18f,61.9048f}, - {13.9419f,52.381f}, - {32.9895f,42.8571f}, - {13.9419f,33.3333f}, - {9.18f,23.8095f}, - {9.18f,14.2857f}, - {13.9419f,4.7619f}, - {18.7038f,0.0f}, - {23.4657f,-9.5238f}, - {23.4657f,-19.0476f}, - {18.7038f,-28.5714f} -}; - -static const SFG_StrokeVertex ch125st2[] = -{ - {23.4657f,38.0952f}, - {13.9419f,28.5714f}, - {13.9419f,19.0476f}, - {18.7038f,9.5238f}, - {23.4657f,4.7619f}, - {28.2276f,-4.7619f}, - {28.2276f,-14.2857f}, - {23.4657f,-23.8095f}, - {18.7038f,-28.5714f}, - {9.18f,-33.3333f} -}; - -static const SFG_StrokeStrip ch125st[] = -{ - {10,ch125st0}, - {17,ch125st1}, - {10,ch125st2} -}; - -static const SFG_StrokeChar ch125 = {41.4695f,3,ch125st}; - -/* char: 0x7e */ - -static const SFG_StrokeVertex ch126st0[] = -{ - {2.92f,28.5714f}, - {2.92f,38.0952f}, - {7.6819f,52.381f}, - {17.2057f,57.1429f}, - {26.7295f,57.1429f}, - {36.2533f,52.381f}, - {55.301f,38.0952f}, - {64.8248f,33.3333f}, - {74.3486f,33.3333f}, - {83.8724f,38.0952f}, - {88.6343f,47.619f} -}; - -static const SFG_StrokeVertex ch126st1[] = -{ - {2.92f,38.0952f}, - {7.6819f,47.619f}, - {17.2057f,52.381f}, - {26.7295f,52.381f}, - {36.2533f,47.619f}, - {55.301f,33.3333f}, - {64.8248f,28.5714f}, - {74.3486f,28.5714f}, - {83.8724f,33.3333f}, - {88.6343f,47.619f}, - {88.6343f,57.1429f} -}; - -static const SFG_StrokeStrip ch126st[] = -{ - {11,ch126st0}, - {11,ch126st1} -}; - -static const SFG_StrokeChar ch126 = {91.2743f,2,ch126st}; - -/* char: 0x7f */ - -static const SFG_StrokeVertex ch127st0[] = -{ - {52.381f,100.0f}, - {14.2857f,-33.3333f} -}; - -static const SFG_StrokeVertex ch127st1[] = -{ - {28.5714f,66.6667f}, - {14.2857f,61.9048f}, - {4.7619f,52.381f}, - {0.0f,38.0952f}, - {0.0f,23.8095f}, - {4.7619f,14.2857f}, - {14.2857f,4.7619f}, - {28.5714f,0.0f}, - {38.0952f,0.0f}, - {52.381f,4.7619f}, - {61.9048f,14.2857f}, - {66.6667f,28.5714f}, - {66.6667f,42.8571f}, - {61.9048f,52.381f}, - {52.381f,61.9048f}, - {38.0952f,66.6667f}, - {28.5714f,66.6667f} -}; - -static const SFG_StrokeStrip ch127st[] = -{ - {2,ch127st0}, - {17,ch127st1} -}; - -static const SFG_StrokeChar ch127 = {66.6667f,2,ch127st}; - -static const SFG_StrokeChar *chars[] = -{ - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - &ch32, &ch33, &ch34, &ch35, &ch36, &ch37, &ch38, &ch39, - &ch40, &ch41, &ch42, &ch43, &ch44, &ch45, &ch46, &ch47, - &ch48, &ch49, &ch50, &ch51, &ch52, &ch53, &ch54, &ch55, - &ch56, &ch57, &ch58, &ch59, &ch60, &ch61, &ch62, &ch63, - &ch64, &ch65, &ch66, &ch67, &ch68, &ch69, &ch70, &ch71, - &ch72, &ch73, &ch74, &ch75, &ch76, &ch77, &ch78, &ch79, - &ch80, &ch81, &ch82, &ch83, &ch84, &ch85, &ch86, &ch87, - &ch88, &ch89, &ch90, &ch91, &ch92, &ch93, &ch94, &ch95, - &ch96, &ch97, &ch98, &ch99, &ch100, &ch101, &ch102, &ch103, - &ch104, &ch105, &ch106, &ch107, &ch108, &ch109, &ch110, &ch111, - &ch112, &ch113, &ch114, &ch115, &ch116, &ch117, &ch118, &ch119, - &ch120, &ch121, &ch122, &ch123, &ch124, &ch125, &ch126, &ch127 -}; - -const SFG_StrokeFont fgStrokeRoman = {"Roman",128,152.381f,chars}; diff --git a/examples/common/opengl-framework/freeglut/freeglut_structure.c b/examples/common/opengl-framework/freeglut/freeglut_structure.c deleted file mode 100644 index c5a2d734..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_structure.c +++ /dev/null @@ -1,598 +0,0 @@ -/* - * freeglut_structure.c - * - * Windows and menus need tree structure - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Sat Dec 18 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* -- GLOBAL EXPORTS ------------------------------------------------------- */ - -/* - * The SFG_Structure container holds information about windows and menus - * created between glutInit() and glutMainLoop() return. - */ - -SFG_Structure fgStructure = { { NULL, NULL }, /* The list of windows */ - { NULL, NULL }, /* The list of menus */ - { NULL, NULL }, /* Windows to Destroy list */ - NULL, /* The current window */ - NULL, /* The current menu */ - NULL, /* The menu OpenGL context */ - NULL, /* The game mode window */ - 0, /* The current new window ID */ - 0 }; /* The current new menu ID */ - - -/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ - -static void fghClearCallBacks( SFG_Window *window ) -{ - if( window ) - { - int i; - for( i = 0; i < TOTAL_CALLBACKS; ++i ) - window->CallBacks[ i ] = NULL; - } -} - -/* - * This private function creates, opens and adds to the hierarchy - * a freeglut window complete with OpenGL context and stuff... - * - * If parent is set to NULL, the window created will be a topmost one. - */ -SFG_Window* fgCreateWindow( SFG_Window* parent, const char* title, - GLboolean positionUse, int x, int y, - GLboolean sizeUse, int w, int h, - GLboolean gameMode, GLboolean isMenu ) -{ - /* Have the window object created */ - SFG_Window *window = (SFG_Window *)calloc( sizeof(SFG_Window), 1 ); - -#if TARGET_HOST_UNIX_X11 - window->Window.FBConfig = NULL; -#endif - fghClearCallBacks( window ); - - /* Initialize the object properties */ - window->ID = ++fgStructure.WindowID; -#if TARGET_HOST_POSIX_X11 - window->State.OldHeight = window->State.OldWidth = -1; -#endif - - fgListInit( &window->Children ); - if( parent ) - { - fgListAppend( &parent->Children, &window->Node ); - window->Parent = parent; - } - else - fgListAppend( &fgStructure.Windows, &window->Node ); - - /* Set the default mouse cursor and reset the modifiers value */ - window->State.Cursor = GLUT_CURSOR_INHERIT; - - window->IsMenu = isMenu; - - window->State.IgnoreKeyRepeat = GL_FALSE; - window->State.KeyRepeating = GL_FALSE; - window->State.IsFullscreen = GL_FALSE; - - /* - * Open the window now. The fgOpenWindow() function is system - * dependant, and resides in freeglut_window.c. Uses fgState. - */ - fgOpenWindow( window, title, positionUse, x, y, sizeUse, w, h, gameMode, - (GLboolean)(parent ? GL_TRUE : GL_FALSE) ); - - return window; -} - -/* - * This private function creates a menu and adds it to the menus list - */ -SFG_Menu* fgCreateMenu( FGCBMenu menuCallback ) -{ - int x = 100, y = 100, w = 1, h = 1; - SFG_Window *current_window = fgStructure.CurrentWindow; - - /* Have the menu object created */ - SFG_Menu* menu = (SFG_Menu *)calloc( sizeof(SFG_Menu), 1 ); - - menu->ParentWindow = NULL; - - /* Create a window for the menu to reside in. */ - - fgCreateWindow( NULL, "freeglut menu", GL_TRUE, x, y, GL_TRUE, w, h, - GL_FALSE, GL_TRUE ); - menu->Window = fgStructure.CurrentWindow; - glutDisplayFunc( fgDisplayMenu ); - - glutHideWindow( ); /* Hide the window for now */ - fgSetWindow( current_window ); - - /* Initialize the object properties: */ - menu->ID = ++fgStructure.MenuID; - menu->Callback = menuCallback; - menu->ActiveEntry = NULL; - - fgListInit( &menu->Entries ); - fgListAppend( &fgStructure.Menus, &menu->Node ); - - /* Newly created menus implicitly become current ones */ - fgStructure.CurrentMenu = menu; - - return menu; -} - -/* - * Function to add a window to the linked list of windows to destroy. - * Subwindows are automatically added because they hang from the window - * structure. - */ -void fgAddToWindowDestroyList( SFG_Window* window ) -{ - SFG_WindowList *new_list_entry = - ( SFG_WindowList* )malloc( sizeof(SFG_WindowList ) ); - new_list_entry->window = window; - fgListAppend( &fgStructure.WindowsToDestroy, &new_list_entry->node ); - - /* Check if the window is the current one... */ - if( fgStructure.CurrentWindow == window ) - fgStructure.CurrentWindow = NULL; - - /* - * Clear all window callbacks except Destroy, which will - * be invoked later. Right now, we are potentially carrying - * out a freeglut operation at the behest of a client callback, - * so we are reluctant to re-enter the client with the Destroy - * callback, right now. The others are all wiped out, however, - * to ensure that they are no longer called after this point. - */ - { - FGCBDestroy destroy = (FGCBDestroy)FETCH_WCB( *window, Destroy ); - fghClearCallBacks( window ); - SET_WCB( *window, Destroy, destroy ); - } -} - -/* - * Function to close down all the windows in the "WindowsToDestroy" list - */ -void fgCloseWindows( ) -{ - while( fgStructure.WindowsToDestroy.First ) - { - SFG_WindowList *window_ptr = fgStructure.WindowsToDestroy.First; - fgDestroyWindow( window_ptr->window ); - fgListRemove( &fgStructure.WindowsToDestroy, &window_ptr->node ); - free( window_ptr ); - } -} - -/* - * This function destroys a window and all of its subwindows. Actually, - * another function, defined in freeglut_window.c is called, but this is - * a whole different story... - */ -void fgDestroyWindow( SFG_Window* window ) -{ - FREEGLUT_INTERNAL_ERROR_EXIT ( window, "Window destroy function called with null window", - "fgDestroyWindow" ); - - while( window->Children.First ) - fgDestroyWindow( ( SFG_Window * )window->Children.First ); - - { - SFG_Window *activeWindow = fgStructure.CurrentWindow; - INVOKE_WCB( *window, Destroy, ( ) ); - fgSetWindow( activeWindow ); - } - - if( window->Parent ) - fgListRemove( &window->Parent->Children, &window->Node ); - else - fgListRemove( &fgStructure.Windows, &window->Node ); - - if( window->ActiveMenu ) - fgDeactivateMenu( window ); - - fghClearCallBacks( window ); - fgCloseWindow( window ); - free( window ); - if( fgStructure.CurrentWindow == window ) - fgStructure.CurrentWindow = NULL; -} - -/* - * This is a helper static function that removes a menu (given its pointer) - * from any windows that can be accessed from a given parent... - */ -static void fghRemoveMenuFromWindow( SFG_Window* window, SFG_Menu* menu ) -{ - SFG_Window *subWindow; - int i; - - /* Check whether this is the active menu in the window */ - if ( menu == window->ActiveMenu ) - window->ActiveMenu = NULL ; - - /* - * Check if the menu is attached to the current window, - * if so, have it detached (by overwriting with a NULL): - */ - for( i = 0; i < FREEGLUT_MAX_MENUS; i++ ) - if( window->Menu[ i ] == menu ) - window->Menu[ i ] = NULL; - - /* Call this function for all of the window's children recursively: */ - for( subWindow = (SFG_Window *)window->Children.First; - subWindow; - subWindow = (SFG_Window *)subWindow->Node.Next) - fghRemoveMenuFromWindow( subWindow, menu ); -} - -/* - * This is a static helper function that removes menu references - * from another menu, given two pointers to them... - */ -static void fghRemoveMenuFromMenu( SFG_Menu* from, SFG_Menu* menu ) -{ - SFG_MenuEntry *entry; - - for( entry = (SFG_MenuEntry *)from->Entries.First; - entry; - entry = ( SFG_MenuEntry * )entry->Node.Next ) - if( entry->SubMenu == menu ) - entry->SubMenu = NULL; -} - -/* - * This function destroys a menu specified by the parameter. All menus - * and windows are updated to make sure no ill pointers hang around. - */ -void fgDestroyMenu( SFG_Menu* menu ) -{ - SFG_Window *window; - SFG_Menu *from; - - FREEGLUT_INTERNAL_ERROR_EXIT ( menu, "Menu destroy function called with null menu", - "fgDestroyMenu" ); - - /* First of all, have all references to this menu removed from all windows: */ - for( window = (SFG_Window *)fgStructure.Windows.First; - window; - window = (SFG_Window *)window->Node.Next ) - fghRemoveMenuFromWindow( window, menu ); - - /* Now proceed with removing menu entries that lead to this menu */ - for( from = ( SFG_Menu * )fgStructure.Menus.First; - from; - from = ( SFG_Menu * )from->Node.Next ) - fghRemoveMenuFromMenu( from, menu ); - - /* - * If the programmer defined a destroy callback, call it - * A. Donev: But first make this the active menu - */ - if( menu->Destroy ) - { - SFG_Menu *activeMenu=fgStructure.CurrentMenu; - fgStructure.CurrentMenu = menu; - menu->Destroy( ); - fgStructure.CurrentMenu = activeMenu; - } - - /* - * Now we are pretty sure the menu is not used anywhere - * and that we can remove all of its entries - */ - while( menu->Entries.First ) - { - SFG_MenuEntry *entry = ( SFG_MenuEntry * ) menu->Entries.First; - - fgListRemove( &menu->Entries, &entry->Node ); - - if( entry->Text ) - free( entry->Text ); - entry->Text = NULL; - - free( entry ); - } - - if( fgStructure.CurrentWindow == menu->Window ) - fgSetWindow( NULL ); - fgDestroyWindow( menu->Window ); - fgListRemove( &fgStructure.Menus, &menu->Node ); - if( fgStructure.CurrentMenu == menu ) - fgStructure.CurrentMenu = NULL; - - free( menu ); -} - -/* - * This function should be called on glutInit(). It will prepare the internal - * structure of freeglut to be used in the application. The structure will be - * destroyed using fgDestroyStructure() on glutMainLoop() return. In that - * case further use of freeglut should be preceeded with a glutInit() call. - */ -void fgCreateStructure( void ) -{ - /* - * We will be needing two lists: the first containing windows, - * and the second containing the user-defined menus. - * Also, no current window/menu is set, as none has been created yet. - */ - - fgListInit(&fgStructure.Windows); - fgListInit(&fgStructure.Menus); - fgListInit(&fgStructure.WindowsToDestroy); - - fgStructure.CurrentWindow = NULL; - fgStructure.CurrentMenu = NULL; - fgStructure.MenuContext = NULL; - fgStructure.GameModeWindow = NULL; - fgStructure.WindowID = 0; - fgStructure.MenuID = 0; -} - -/* - * This function is automatically called on glutMainLoop() return. - * It should deallocate and destroy all remnants of previous - * glutInit()-enforced structure initialization... - */ -void fgDestroyStructure( void ) -{ - /* Clean up the WindowsToDestroy list. */ - fgCloseWindows( ); - - /* Make sure all windows and menus have been deallocated */ - while( fgStructure.Menus.First ) - fgDestroyMenu( ( SFG_Menu * )fgStructure.Menus.First ); - - while( fgStructure.Windows.First ) - fgDestroyWindow( ( SFG_Window * )fgStructure.Windows.First ); -} - -/* - * Helper function to enumerate through all registered top-level windows - */ -void fgEnumWindows( FGCBenumerator enumCallback, SFG_Enumerator* enumerator ) -{ - SFG_Window *window; - - FREEGLUT_INTERNAL_ERROR_EXIT ( enumCallback && enumerator, - "Enumerator or callback missing from window enumerator call", - "fgEnumWindows" ); - - /* Check every of the top-level windows */ - for( window = ( SFG_Window * )fgStructure.Windows.First; - window; - window = ( SFG_Window * )window->Node.Next ) - { - enumCallback( window, enumerator ); - if( enumerator->found ) - return; - } -} - -/* - * Helper function to enumerate through all a window's subwindows - * (single level descent) - */ -void fgEnumSubWindows( SFG_Window* window, FGCBenumerator enumCallback, - SFG_Enumerator* enumerator ) -{ - SFG_Window *child; - - FREEGLUT_INTERNAL_ERROR_EXIT ( enumCallback && enumerator, - "Enumerator or callback missing from subwindow enumerator call", - "fgEnumSubWindows" ); - FREEGLUT_INTERNAL_ERROR_EXIT_IF_NOT_INITIALISED ( "Window Enumeration" ); - - for( child = ( SFG_Window * )window->Children.First; - child; - child = ( SFG_Window * )child->Node.Next ) - { - enumCallback( child, enumerator ); - if( enumerator->found ) - return; - } -} - -/* - * A static helper function to look for a window given its handle - */ -static void fghcbWindowByHandle( SFG_Window *window, - SFG_Enumerator *enumerator ) -{ - if ( enumerator->found ) - return; - - /* Check the window's handle. Hope this works. Looks ugly. That's for sure. */ - if( window->Window.Handle == (SFG_WindowHandleType) (enumerator->data) ) - { - enumerator->found = GL_TRUE; - enumerator->data = window; - - return; - } - - /* Otherwise, check this window's children */ - fgEnumSubWindows( window, fghcbWindowByHandle, enumerator ); -} - -/* - * fgWindowByHandle returns a (SFG_Window *) value pointing to the - * first window in the queue matching the specified window handle. - * The function is defined in freeglut_structure.c file. - */ -SFG_Window* fgWindowByHandle ( SFG_WindowHandleType hWindow ) -{ - SFG_Enumerator enumerator; - - /* This is easy and makes use of the windows enumeration defined above */ - enumerator.found = GL_FALSE; - enumerator.data = (void *)hWindow; - fgEnumWindows( fghcbWindowByHandle, &enumerator ); - - if( enumerator.found ) - return( SFG_Window *) enumerator.data; - return NULL; -} - -/* - * A static helper function to look for a window given its ID - */ -static void fghcbWindowByID( SFG_Window *window, SFG_Enumerator *enumerator ) -{ - /* Make sure we do not overwrite our precious results... */ - if( enumerator->found ) - return; - - /* Check the window's handle. Hope this works. Looks ugly. That's for sure. */ - if( window->ID == *( int *)(enumerator->data) ) - { - enumerator->found = GL_TRUE; - enumerator->data = window; - - return; - } - - /* Otherwise, check this window's children */ - fgEnumSubWindows( window, fghcbWindowByID, enumerator ); -} - -/* - * This function is similiar to the previous one, except it is - * looking for a specified (sub)window identifier. The function - * is defined in freeglut_structure.c file. - */ -SFG_Window* fgWindowByID( int windowID ) -{ - SFG_Enumerator enumerator; - - /* Uses a method very similiar for fgWindowByHandle... */ - enumerator.found = GL_FALSE; - enumerator.data = ( void * )&windowID; - fgEnumWindows( fghcbWindowByID, &enumerator ); - if( enumerator.found ) - return ( SFG_Window * )enumerator.data; - return NULL; -} - -/* - * Looks up a menu given its ID. This is easier that fgWindowByXXX - * as all menus are placed in one doubly linked list... - */ -SFG_Menu* fgMenuByID( int menuID ) -{ - SFG_Menu *menu = NULL; - - /* It's enough to check all entries in fgStructure.Menus... */ - for( menu = (SFG_Menu *)fgStructure.Menus.First; - menu; - menu = (SFG_Menu *)menu->Node.Next ) - if( menu->ID == menuID ) - return menu; - return NULL; -} - -/* - * List functions... - */ -void fgListInit(SFG_List *list) -{ - list->First = NULL; - list->Last = NULL; -} - -void fgListAppend(SFG_List *list, SFG_Node *node) -{ - if ( list->Last ) - { - SFG_Node *ln = (SFG_Node *) list->Last; - ln->Next = node; - node->Prev = ln; - } - else - { - node->Prev = NULL; - list->First = node; - } - - node->Next = NULL; - list->Last = node; -} - -void fgListRemove(SFG_List *list, SFG_Node *node) -{ - if( node->Next ) - ( ( SFG_Node * )node->Next )->Prev = node->Prev; - if( node->Prev ) - ( ( SFG_Node * )node->Prev )->Next = node->Next; - if( ( ( SFG_Node * )list->First ) == node ) - list->First = node->Next; - if( ( ( SFG_Node * )list->Last ) == node ) - list->Last = node->Prev; -} - -int fgListLength(SFG_List *list) -{ - SFG_Node *node; - int length = 0; - - for( node =( SFG_Node * )list->First; - node; - node = ( SFG_Node * )node->Next ) - ++length; - - return length; -} - - -void fgListInsert(SFG_List *list, SFG_Node *next, SFG_Node *node) -{ - SFG_Node *prev; - - if( (node->Next = next) ) - { - prev = next->Prev; - next->Prev = node; - } - else - { - prev = list->Last; - list->Last = node; - } - - if( (node->Prev = prev) ) - prev->Next = node; - else - list->First = node; -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_teapot.c b/examples/common/opengl-framework/freeglut/freeglut_teapot.c deleted file mode 100644 index 722bca61..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_teapot.c +++ /dev/null @@ -1,200 +0,0 @@ -/* - * freeglut_teapot.c - * - * Teapot(tm) rendering code. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Fri Dec 24 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -/* - * Original teapot code copyright follows: - */ - -/* - * (c) Copyright 1993, Silicon Graphics, Inc. - * - * ALL RIGHTS RESERVED - * - * Permission to use, copy, modify, and distribute this software - * for any purpose and without fee is hereby granted, provided - * that the above copyright notice appear in all copies and that - * both the copyright notice and this permission notice appear in - * supporting documentation, and that the name of Silicon - * Graphics, Inc. not be used in advertising or publicity - * pertaining to distribution of the software without specific, - * written prior permission. - * - * THE MATERIAL EMBODIED ON THIS SOFTWARE IS PROVIDED TO YOU - * "AS-IS" AND WITHOUT WARRANTY OF ANY KIND, EXPRESS, IMPLIED OR - * OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY WARRANTY OF - * MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. IN NO - * EVENT SHALL SILICON GRAPHICS, INC. BE LIABLE TO YOU OR ANYONE - * ELSE FOR ANY DIRECT, SPECIAL, INCIDENTAL, INDIRECT OR - * CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER, - * INCLUDING WITHOUT LIMITATION, LOSS OF PROFIT, LOSS OF USE, - * SAVINGS OR REVENUE, OR THE CLAIMS OF THIRD PARTIES, WHETHER OR - * NOT SILICON GRAPHICS, INC. HAS BEEN ADVISED OF THE POSSIBILITY - * OF SUCH LOSS, HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, - * ARISING OUT OF OR IN CONNECTION WITH THE POSSESSION, USE OR - * PERFORMANCE OF THIS SOFTWARE. - * - * US Government Users Restricted Rights - * - * Use, duplication, or disclosure by the Government is subject to - * restrictions set forth in FAR 52.227.19(c)(2) or subparagraph - * (c)(1)(ii) of the Rights in Technical Data and Computer - * Software clause at DFARS 252.227-7013 and/or in similar or - * successor clauses in the FAR or the DOD or NASA FAR - * Supplement. Unpublished-- rights reserved under the copyright - * laws of the United States. Contractor/manufacturer is Silicon - * Graphics, Inc., 2011 N. Shoreline Blvd., Mountain View, CA - * 94039-7311. - * - * OpenGL(TM) is a trademark of Silicon Graphics, Inc. - */ - -#include -#include "freeglut_internal.h" -#include "freeglut_teapot_data.h" - -/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ - - -static void fghTeapot( GLint grid, GLdouble scale, GLenum type ) -{ -#if defined(_WIN32_WCE) - int i, numV=sizeof(strip_vertices)/4, numI=sizeof(strip_normals)/4; -#else - double p[4][4][3], q[4][4][3], r[4][4][3], s[4][4][3]; - long i, j, k, l; -#endif - - glPushAttrib( GL_ENABLE_BIT | GL_EVAL_BIT ); - glEnable( GL_AUTO_NORMAL ); - glEnable( GL_NORMALIZE ); - glEnable( GL_MAP2_VERTEX_3 ); - glEnable( GL_MAP2_TEXTURE_COORD_2 ); - - glPushMatrix(); - glRotated( 270.0, 1.0, 0.0, 0.0 ); - glScaled( 0.5 * scale, 0.5 * scale, 0.5 * scale ); - glTranslated( 0.0, 0.0, -1.5 ); - -#if defined(_WIN32_WCE) - glRotated( 90.0, 1.0, 0.0, 0.0 ); - glBegin( GL_TRIANGLE_STRIP ); - - for( i = 0; i < numV-1; i++ ) - { - int vidx = strip_vertices[i], - nidx = strip_normals[i]; - - if( vidx != -1 ) - { - glNormal3fv( normals[nidx] ); - glVertex3fv( vertices[vidx] ); - } - else - { - glEnd(); - glBegin( GL_TRIANGLE_STRIP ); - } - } - - glEnd(); -#else - for (i = 0; i < 10; i++) { - for (j = 0; j < 4; j++) { - for (k = 0; k < 4; k++) { - for (l = 0; l < 3; l++) { - p[j][k][l] = cpdata[patchdata[i][j * 4 + k]][l]; - q[j][k][l] = cpdata[patchdata[i][j * 4 + (3 - k)]][l]; - if (l == 1) - q[j][k][l] *= -1.0; - if (i < 6) { - r[j][k][l] = - cpdata[patchdata[i][j * 4 + (3 - k)]][l]; - if (l == 0) - r[j][k][l] *= -1.0; - s[j][k][l] = cpdata[patchdata[i][j * 4 + k]][l]; - if (l == 0) - s[j][k][l] *= -1.0; - if (l == 1) - s[j][k][l] *= -1.0; - } - } - } - } - - glMap2d(GL_MAP2_TEXTURE_COORD_2, 0.0, 1.0, 2, 2, 0.0, 1.0, 4, 2, - &tex[0][0][0]); - glMap2d(GL_MAP2_VERTEX_3, 0.0, 1.0, 3, 4, 0.0, 1.0, 12, 4, - &p[0][0][0]); - glMapGrid2d(grid, 0.0, 1.0, grid, 0.0, 1.0); - glEvalMesh2(type, 0, grid, 0, grid); - glMap2d(GL_MAP2_VERTEX_3, 0.0, 1.0, 3, 4, 0.0, 1.0, 12, 4, - &q[0][0][0]); - glEvalMesh2(type, 0, grid, 0, grid); - if (i < 6) { - glMap2d(GL_MAP2_VERTEX_3, 0.0, 1.0, 3, 4, 0.0, 1.0, 12, 4, - &r[0][0][0]); - glEvalMesh2(type, 0, grid, 0, grid); - glMap2d(GL_MAP2_VERTEX_3, 0.0, 1.0, 3, 4, 0.0, 1.0, 12, 4, - &s[0][0][0]); - glEvalMesh2(type, 0, grid, 0, grid); - } - } -#endif /* defined(_WIN32_WCE) */ - - glPopMatrix(); - glPopAttrib(); -} - - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * Renders a beautiful wired teapot... - */ -void FGAPIENTRY glutWireTeapot( GLdouble size ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutWireTeapot" ); - /* We will use the general teapot rendering code */ - fghTeapot( 10, size, GL_LINE ); -} - -/* - * Renders a beautiful filled teapot... - */ -void FGAPIENTRY glutSolidTeapot( GLdouble size ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSolidTeapot" ); - /* We will use the general teapot rendering code */ - fghTeapot( 7, size, GL_FILL ); -} - -/*** END OF FILE ***/ - - - - - diff --git a/examples/common/opengl-framework/freeglut/freeglut_teapot_data.h b/examples/common/opengl-framework/freeglut/freeglut_teapot_data.h deleted file mode 100644 index 3bf83e11..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_teapot_data.h +++ /dev/null @@ -1,2429 +0,0 @@ -/* - * freeglut_teapot_data.h - * - * The freeglut library teapot data include file. - * - * 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 - * PAWEL W. OLSZTA 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 FREEGLUT_TEAPOT_DATA_H -#define FREEGLUT_TEAPOT_DATA_H - -#if defined(_WIN32_WCE) -/* - * Submitted through the kind offices of Daniel Wagner (daniel@ims.tuwien.ac.at) - */ - -/* 530 vertices */ - -const int numVertices = 530; -const float vertices[530][3] = { -2.1f, 3.6f, 0.0f, -2.071f, 3.711f, 0.0f, -2.105f, 3.748f, 0.0f, -2.174f, 3.711f, 0.0f, -2.25f, 3.6f, 0.0f, -1.937f, 3.6f, 0.8242f, -1.91f, 3.711f, 0.8128f, -1.942f, 3.748f, 0.8261f, -2.005f, 3.711f, 0.8532f, -2.076f, 3.6f, 0.8831f, -1.491f, 3.6f, 1.491f, -1.47f, 3.711f, 1.47f, -1.494f, 3.748f, 1.494f, -1.543f, 3.711f, 1.543f, -1.597f, 3.6f, 1.597f, -0.8242f, 3.6f, 1.937f, -0.8128f, 3.711f, 1.91f, -0.8261f, 3.748f, 1.942f, -0.8532f, 3.711f, 2.005f, -0.8831f, 3.6f, 2.076f, -0.0f, 3.6f, 2.1f, -0.0f, 3.711f, 2.071f, -0.0f, 3.748f, 2.105f, -0.0f, 3.711f, 2.174f, -0.0f, 3.6f, 2.25f, --0.8812f, 3.6f, 1.937f, --0.8368f, 3.711f, 1.91f, --0.8332f, 3.748f, 1.942f, --0.8541f, 3.711f, 2.005f, --0.8831f, 3.6f, 2.076f, --1.542f, 3.6f, 1.491f, --1.492f, 3.711f, 1.47f, --1.501f, 3.748f, 1.494f, --1.544f, 3.711f, 1.543f, --1.597f, 3.6f, 1.597f, --1.956f, 3.6f, 0.8242f, --1.918f, 3.711f, 0.8128f, --1.944f, 3.748f, 0.8261f, --2.006f, 3.711f, 0.8532f, --2.076f, 3.6f, 0.8831f, --2.1f, 3.6f, 0.0f, --2.071f, 3.711f, 0.0f, --2.105f, 3.748f, 0.0f, --2.174f, 3.711f, 0.0f, --2.25f, 3.6f, 0.0f, --1.937f, 3.6f, -0.8242f, --1.91f, 3.711f, -0.8128f, --1.942f, 3.748f, -0.8261f, --2.005f, 3.711f, -0.8532f, --2.076f, 3.6f, -0.8831f, --1.491f, 3.6f, -1.491f, --1.47f, 3.711f, -1.47f, --1.494f, 3.748f, -1.494f, --1.543f, 3.711f, -1.543f, --1.597f, 3.6f, -1.597f, --0.8242f, 3.6f, -1.937f, --0.8128f, 3.711f, -1.91f, --0.8261f, 3.748f, -1.942f, --0.8532f, 3.711f, -2.005f, --0.8831f, 3.6f, -2.076f, -0.0f, 3.6f, -2.1f, -0.0f, 3.711f, -2.071f, -0.0f, 3.748f, -2.105f, -0.0f, 3.711f, -2.174f, -0.0f, 3.6f, -2.25f, -0.8242f, 3.6f, -1.937f, -0.8128f, 3.711f, -1.91f, -0.8261f, 3.748f, -1.942f, -0.8532f, 3.711f, -2.005f, -0.8831f, 3.6f, -2.076f, -1.491f, 3.6f, -1.491f, -1.47f, 3.711f, -1.47f, -1.494f, 3.748f, -1.494f, -1.543f, 3.711f, -1.543f, -1.597f, 3.6f, -1.597f, -1.937f, 3.6f, -0.8242f, -1.91f, 3.711f, -0.8128f, -1.942f, 3.748f, -0.8261f, -2.005f, 3.711f, -0.8532f, -2.076f, 3.6f, -0.8831f, -2.525f, 3.011f, 0.0f, -2.766f, 2.433f, 0.0f, -2.936f, 1.876f, 0.0f, -3.0f, 1.35f, 0.0f, -2.33f, 3.011f, 0.9912f, -2.551f, 2.433f, 1.086f, -2.708f, 1.876f, 1.152f, -2.767f, 1.35f, 1.178f, -1.793f, 3.011f, 1.793f, -1.964f, 2.433f, 1.964f, -2.084f, 1.876f, 2.084f, -2.13f, 1.35f, 2.13f, -0.9912f, 3.011f, 2.33f, -1.086f, 2.433f, 2.551f, -1.152f, 1.876f, 2.708f, -1.178f, 1.35f, 2.767f, -0.0f, 3.011f, 2.525f, -0.0f, 2.433f, 2.766f, -0.0f, 1.876f, 2.936f, -0.0f, 1.35f, 3.0f, --0.9912f, 3.011f, 2.33f, --1.086f, 2.433f, 2.551f, --1.152f, 1.876f, 2.708f, --1.178f, 1.35f, 2.767f, --1.793f, 3.011f, 1.793f, --1.964f, 2.433f, 1.964f, --2.084f, 1.876f, 2.084f, --2.13f, 1.35f, 2.13f, --2.33f, 3.011f, 0.9912f, --2.551f, 2.433f, 1.086f, --2.708f, 1.876f, 1.152f, --2.767f, 1.35f, 1.178f, --2.525f, 3.011f, 0.0f, --2.766f, 2.433f, 0.0f, --2.936f, 1.876f, 0.0f, --3.0f, 1.35f, 0.0f, --2.33f, 3.011f, -0.9912f, --2.551f, 2.433f, -1.086f, --2.708f, 1.876f, -1.152f, --2.767f, 1.35f, -1.178f, --1.793f, 3.011f, -1.793f, --1.964f, 2.433f, -1.964f, --2.084f, 1.876f, -2.084f, --2.13f, 1.35f, -2.13f, --0.9912f, 3.011f, -2.33f, --1.086f, 2.433f, -2.551f, --1.152f, 1.876f, -2.708f, --1.178f, 1.35f, -2.767f, -0.0f, 3.011f, -2.525f, -0.0f, 2.433f, -2.766f, -0.0f, 1.876f, -2.936f, -0.0f, 1.35f, -3.0f, -0.9912f, 3.011f, -2.33f, -1.086f, 2.433f, -2.551f, -1.152f, 1.876f, -2.708f, -1.178f, 1.35f, -2.767f, -1.793f, 3.011f, -1.793f, -1.964f, 2.433f, -1.964f, -2.084f, 1.876f, -2.084f, -2.13f, 1.35f, -2.13f, -2.33f, 3.011f, -0.9912f, -2.551f, 2.433f, -1.086f, -2.708f, 1.876f, -1.152f, -2.767f, 1.35f, -1.178f, -2.883f, 0.9053f, 0.0f, -2.625f, 0.5766f, 0.0f, -2.367f, 0.3533f, 0.0f, -2.25f, 0.225f, 0.0f, -2.659f, 0.9053f, 1.132f, -2.422f, 0.5766f, 1.03f, -2.184f, 0.3533f, 0.9291f, -2.076f, 0.225f, 0.8831f, -2.047f, 0.9053f, 2.047f, -1.864f, 0.5766f, 1.864f, -1.681f, 0.3533f, 1.681f, -1.597f, 0.225f, 1.597f, -1.132f, 0.9053f, 2.659f, -1.03f, 0.5766f, 2.422f, -0.9291f, 0.3533f, 2.184f, -0.8831f, 0.225f, 2.076f, -0.0f, 0.9053f, 2.883f, -0.0f, 0.5766f, 2.625f, -0.0f, 0.3533f, 2.367f, -0.0f, 0.225f, 2.25f, --1.132f, 0.9053f, 2.659f, --1.03f, 0.5766f, 2.422f, --0.9291f, 0.3533f, 2.184f, --0.8831f, 0.225f, 2.076f, --2.047f, 0.9053f, 2.047f, --1.864f, 0.5766f, 1.864f, --1.681f, 0.3533f, 1.681f, --1.597f, 0.225f, 1.597f, --2.659f, 0.9053f, 1.132f, --2.422f, 0.5766f, 1.03f, --2.184f, 0.3533f, 0.9291f, --2.076f, 0.225f, 0.8831f, --2.883f, 0.9053f, 0.0f, --2.625f, 0.5766f, 0.0f, --2.367f, 0.3533f, 0.0f, --2.25f, 0.225f, 0.0f, --2.659f, 0.9053f, -1.132f, --2.422f, 0.5766f, -1.03f, --2.184f, 0.3533f, -0.9291f, --2.076f, 0.225f, -0.8831f, --2.047f, 0.9053f, -2.047f, --1.864f, 0.5766f, -1.864f, --1.681f, 0.3533f, -1.681f, --1.597f, 0.225f, -1.597f, --1.132f, 0.9053f, -2.659f, --1.03f, 0.5766f, -2.422f, --0.9291f, 0.3533f, -2.184f, --0.8831f, 0.225f, -2.076f, -0.0f, 0.9053f, -2.883f, -0.0f, 0.5766f, -2.625f, -0.0f, 0.3533f, -2.367f, -0.0f, 0.225f, -2.25f, -1.132f, 0.9053f, -2.659f, -1.03f, 0.5766f, -2.422f, -0.9291f, 0.3533f, -2.184f, -0.8831f, 0.225f, -2.076f, -2.047f, 0.9053f, -2.047f, -1.864f, 0.5766f, -1.864f, -1.681f, 0.3533f, -1.681f, -1.597f, 0.225f, -1.597f, -2.659f, 0.9053f, -1.132f, -2.422f, 0.5766f, -1.03f, -2.184f, 0.3533f, -0.9291f, -2.076f, 0.225f, -0.8831f, -2.199f, 0.1424f, 0.0f, -1.927f, 0.07031f, 0.0f, -1.253f, 0.01934f, 0.0f, -0.0f, 0.0f, 0.0f, -2.029f, 0.1424f, 0.8631f, -1.777f, 0.07031f, 0.7562f, -1.156f, 0.01934f, 0.4919f, -1.561f, 0.1424f, 1.561f, -1.368f, 0.07031f, 1.368f, -0.8899f, 0.01934f, 0.8899f, -0.8631f, 0.1424f, 2.029f, -0.7562f, 0.07031f, 1.777f, -0.4919f, 0.01934f, 1.156f, -0.0f, 0.1424f, 2.199f, -0.0f, 0.07031f, 1.927f, -0.0f, 0.01934f, 1.253f, --0.8631f, 0.1424f, 2.029f, --0.7562f, 0.07031f, 1.777f, --0.4919f, 0.01934f, 1.156f, --1.561f, 0.1424f, 1.561f, --1.368f, 0.07031f, 1.368f, --0.8899f, 0.01934f, 0.8899f, --2.029f, 0.1424f, 0.8631f, --1.777f, 0.07031f, 0.7562f, --1.156f, 0.01934f, 0.4919f, --2.199f, 0.1424f, 0.0f, --1.927f, 0.07031f, 0.0f, --1.253f, 0.01934f, 0.0f, --2.029f, 0.1424f, -0.8631f, --1.777f, 0.07031f, -0.7562f, --1.156f, 0.01934f, -0.4919f, --1.561f, 0.1424f, -1.561f, --1.368f, 0.07031f, -1.368f, --0.8899f, 0.01934f, -0.8899f, --0.8631f, 0.1424f, -2.029f, --0.7562f, 0.07031f, -1.777f, --0.4919f, 0.01934f, -1.156f, -0.0f, 0.1424f, -2.199f, -0.0f, 0.07031f, -1.927f, -0.0f, 0.01934f, -1.253f, -0.8631f, 0.1424f, -2.029f, -0.7562f, 0.07031f, -1.777f, -0.4919f, 0.01934f, -1.156f, -1.561f, 0.1424f, -1.561f, -1.368f, 0.07031f, -1.368f, -0.8899f, 0.01934f, -0.8899f, -2.029f, 0.1424f, -0.8631f, -1.777f, 0.07031f, -0.7562f, -1.156f, 0.01934f, -0.4919f, --2.4f, 3.038f, 0.0f, --3.101f, 3.032f, 0.0f, --3.619f, 2.995f, 0.0f, --3.94f, 2.895f, 0.0f, --4.05f, 2.7f, 0.0f, --2.377f, 3.09f, 0.2531f, --3.122f, 3.084f, 0.2531f, --3.669f, 3.041f, 0.2531f, --4.005f, 2.926f, 0.2531f, --4.12f, 2.7f, 0.2531f, --2.325f, 3.206f, 0.3375f, --3.168f, 3.198f, 0.3375f, --3.778f, 3.143f, 0.3375f, --4.15f, 2.993f, 0.3375f, --4.275f, 2.7f, 0.3375f, --2.273f, 3.322f, 0.2531f, --3.214f, 3.313f, 0.2531f, --3.888f, 3.244f, 0.2531f, --4.294f, 3.06f, 0.2531f, --4.43f, 2.7f, 0.2531f, --2.25f, 3.375f, 0.0f, --3.234f, 3.364f, 0.0f, --3.938f, 3.291f, 0.0f, --4.359f, 3.09f, 0.0f, --4.5f, 2.7f, 0.0f, --2.273f, 3.322f, -0.2531f, --3.214f, 3.313f, -0.2531f, --3.888f, 3.244f, -0.2531f, --4.294f, 3.06f, -0.2531f, --4.43f, 2.7f, -0.2531f, --2.325f, 3.206f, -0.3375f, --3.168f, 3.198f, -0.3375f, --3.778f, 3.143f, -0.3375f, --4.15f, 2.993f, -0.3375f, --4.275f, 2.7f, -0.3375f, --2.377f, 3.09f, -0.2531f, --3.122f, 3.084f, -0.2531f, --3.669f, 3.041f, -0.2531f, --4.005f, 2.926f, -0.2531f, --4.12f, 2.7f, -0.2531f, --3.991f, 2.394f, 0.0f, --3.806f, 2.025f, 0.0f, --3.48f, 1.656f, 0.0f, --3.0f, 1.35f, 0.0f, --4.055f, 2.365f, 0.2531f, --3.852f, 1.98f, 0.2531f, --3.496f, 1.6f, 0.2531f, --2.977f, 1.28f, 0.2531f, --4.196f, 2.3f, 0.3375f, --3.952f, 1.881f, 0.3375f, --3.531f, 1.478f, 0.3375f, --2.925f, 1.125f, 0.3375f, --4.336f, 2.235f, 0.2531f, --4.051f, 1.782f, 0.2531f, --3.566f, 1.356f, 0.2531f, --2.873f, 0.9703f, 0.2531f, --4.4f, 2.205f, 0.0f, --4.097f, 1.737f, 0.0f, --3.582f, 1.3f, 0.0f, --2.85f, 0.9f, 0.0f, --4.336f, 2.235f, -0.2531f, --4.051f, 1.782f, -0.2531f, --3.566f, 1.356f, -0.2531f, --2.873f, 0.9703f, -0.2531f, --4.196f, 2.3f, -0.3375f, --3.952f, 1.881f, -0.3375f, --3.531f, 1.478f, -0.3375f, --2.925f, 1.125f, -0.3375f, --4.055f, 2.365f, -0.2531f, --3.852f, 1.98f, -0.2531f, --3.496f, 1.6f, -0.2531f, --2.977f, 1.28f, -0.2531f, -2.55f, 2.137f, 0.0f, -3.27f, 2.303f, 0.0f, -3.581f, 2.7f, 0.0f, -3.752f, 3.182f, 0.0f, -4.05f, 3.6f, 0.0f, -2.55f, 1.944f, 0.5569f, -3.324f, 2.159f, 0.5028f, -3.652f, 2.617f, 0.3839f, -3.838f, 3.151f, 0.265f, -4.191f, 3.6f, 0.2109f, -2.55f, 1.519f, 0.7425f, -3.445f, 1.844f, 0.6704f, -3.806f, 2.433f, 0.5119f, -4.027f, 3.085f, 0.3533f, -4.5f, 3.6f, 0.2813f, -2.55f, 1.093f, 0.5569f, -3.566f, 1.529f, 0.5028f, -3.961f, 2.249f, 0.3839f, -4.215f, 3.018f, 0.265f, -4.809f, 3.6f, 0.2109f, -2.55f, 0.9f, 0.0f, -3.621f, 1.385f, 0.0f, -4.031f, 2.166f, 0.0f, -4.301f, 2.988f, 0.0f, -4.95f, 3.6f, 0.0f, -2.55f, 1.093f, -0.5569f, -3.566f, 1.529f, -0.5028f, -3.961f, 2.249f, -0.3839f, -4.215f, 3.018f, -0.265f, -4.809f, 3.6f, -0.2109f, -2.55f, 1.519f, -0.7425f, -3.445f, 1.844f, -0.6704f, -3.806f, 2.433f, -0.5119f, -4.027f, 3.085f, -0.3533f, -4.5f, 3.6f, -0.2813f, -2.55f, 1.944f, -0.5569f, -3.324f, 2.159f, -0.5028f, -3.652f, 2.617f, -0.3839f, -3.838f, 3.151f, -0.265f, -4.191f, 3.6f, -0.2109f, -4.158f, 3.663f, 0.0f, -4.238f, 3.684f, 0.0f, -4.261f, 3.663f, 0.0f, -4.2f, 3.6f, 0.0f, -4.308f, 3.666f, 0.1978f, -4.379f, 3.689f, 0.1687f, -4.381f, 3.668f, 0.1397f, -4.294f, 3.6f, 0.1266f, -4.64f, 3.673f, 0.2637f, -4.69f, 3.7f, 0.225f, -4.645f, 3.677f, 0.1863f, -4.5f, 3.6f, 0.1688f, -4.971f, 3.68f, 0.1978f, -5.001f, 3.711f, 0.1687f, -4.909f, 3.687f, 0.1397f, -4.706f, 3.6f, 0.1266f, -5.122f, 3.683f, 0.0f, -5.142f, 3.716f, 0.0f, -5.029f, 3.691f, 0.0f, -4.8f, 3.6f, 0.0f, -4.971f, 3.68f, -0.1978f, -5.001f, 3.711f, -0.1687f, -4.909f, 3.687f, -0.1397f, -4.706f, 3.6f, -0.1266f, -4.64f, 3.673f, -0.2637f, -4.69f, 3.7f, -0.225f, -4.645f, 3.677f, -0.1863f, -4.5f, 3.6f, -0.1688f, -4.308f, 3.666f, -0.1978f, -4.379f, 3.689f, -0.1687f, -4.381f, 3.668f, -0.1397f, -4.294f, 3.6f, -0.1266f, -0.0f, 4.725f, 0.0f, -0.5109f, 4.651f, 0.0f, -0.4875f, 4.472f, 0.0f, -0.2953f, 4.25f, 0.0f, -0.3f, 4.05f, 0.0f, -0.4715f, 4.651f, 0.2011f, -0.4499f, 4.472f, 0.1918f, -0.2725f, 4.25f, 0.1161f, -0.2768f, 4.05f, 0.1178f, -0.3632f, 4.651f, 0.3632f, -0.3465f, 4.472f, 0.3465f, -0.2098f, 4.25f, 0.2098f, -0.213f, 4.05f, 0.213f, -0.2011f, 4.651f, 0.4715f, -0.1918f, 4.472f, 0.4499f, -0.1161f, 4.25f, 0.2725f, -0.1178f, 4.05f, 0.2768f, -0.0f, 4.651f, 0.5109f, -0.0f, 4.472f, 0.4875f, -0.0f, 4.25f, 0.2953f, -0.0f, 4.05f, 0.3f, --0.2011f, 4.651f, 0.4715f, --0.1918f, 4.472f, 0.4499f, --0.1161f, 4.25f, 0.2725f, --0.1178f, 4.05f, 0.2768f, --0.3632f, 4.651f, 0.3632f, --0.3465f, 4.472f, 0.3465f, --0.2098f, 4.25f, 0.2098f, --0.213f, 4.05f, 0.213f, --0.4715f, 4.651f, 0.2011f, --0.4499f, 4.472f, 0.1918f, --0.2725f, 4.25f, 0.1161f, --0.2768f, 4.05f, 0.1178f, --0.5109f, 4.651f, 0.0f, --0.4875f, 4.472f, 0.0f, --0.2953f, 4.25f, 0.0f, --0.3f, 4.05f, 0.0f, --0.4715f, 4.651f, -0.2011f, --0.4499f, 4.472f, -0.1918f, --0.2725f, 4.25f, -0.1161f, --0.2768f, 4.05f, -0.1178f, --0.3632f, 4.651f, -0.3632f, --0.3465f, 4.472f, -0.3465f, --0.2098f, 4.25f, -0.2098f, --0.213f, 4.05f, -0.213f, --0.2011f, 4.651f, -0.4715f, --0.1918f, 4.472f, -0.4499f, --0.1161f, 4.25f, -0.2725f, --0.1178f, 4.05f, -0.2768f, -0.0f, 4.651f, -0.5109f, -0.0f, 4.472f, -0.4875f, -0.0f, 4.25f, -0.2953f, -0.0f, 4.05f, -0.3f, -0.2011f, 4.651f, -0.4715f, -0.1918f, 4.472f, -0.4499f, -0.1161f, 4.25f, -0.2725f, -0.1178f, 4.05f, -0.2768f, -0.3632f, 4.651f, -0.3632f, -0.3465f, 4.472f, -0.3465f, -0.2098f, 4.25f, -0.2098f, -0.213f, 4.05f, -0.213f, -0.4715f, 4.651f, -0.2011f, -0.4499f, 4.472f, -0.1918f, -0.2725f, 4.25f, -0.1161f, -0.2768f, 4.05f, -0.1178f, -0.6844f, 3.916f, 0.0f, -1.237f, 3.825f, 0.0f, -1.734f, 3.734f, 0.0f, -1.95f, 3.6f, 0.0f, -0.6313f, 3.916f, 0.2686f, -1.142f, 3.825f, 0.4857f, -1.6f, 3.734f, 0.6807f, -1.799f, 3.6f, 0.7654f, -0.4859f, 3.916f, 0.4859f, -0.8786f, 3.825f, 0.8786f, -1.231f, 3.734f, 1.231f, -1.385f, 3.6f, 1.385f, -0.2686f, 3.916f, 0.6313f, -0.4857f, 3.825f, 1.142f, -0.6807f, 3.734f, 1.6f, -0.7654f, 3.6f, 1.799f, -0.0f, 3.916f, 0.6844f, -0.0f, 3.825f, 1.237f, -0.0f, 3.734f, 1.734f, -0.0f, 3.6f, 1.95f, --0.2686f, 3.916f, 0.6313f, --0.4857f, 3.825f, 1.142f, --0.6807f, 3.734f, 1.6f, --0.7654f, 3.6f, 1.799f, --0.4859f, 3.916f, 0.4859f, --0.8786f, 3.825f, 0.8786f, --1.231f, 3.734f, 1.231f, --1.385f, 3.6f, 1.385f, --0.6313f, 3.916f, 0.2686f, --1.142f, 3.825f, 0.4857f, --1.6f, 3.734f, 0.6807f, --1.799f, 3.6f, 0.7654f, --0.6844f, 3.916f, 0.0f, --1.237f, 3.825f, 0.0f, --1.734f, 3.734f, 0.0f, --1.95f, 3.6f, 0.0f, --0.6313f, 3.916f, -0.2686f, --1.142f, 3.825f, -0.4857f, --1.6f, 3.734f, -0.6807f, --1.799f, 3.6f, -0.7654f, --0.4859f, 3.916f, -0.4859f, --0.8786f, 3.825f, -0.8786f, --1.231f, 3.734f, -1.231f, --1.385f, 3.6f, -1.385f, --0.2686f, 3.916f, -0.6313f, --0.4857f, 3.825f, -1.142f, --0.6807f, 3.734f, -1.6f, --0.7654f, 3.6f, -1.799f, -0.0f, 3.916f, -0.6844f, -0.0f, 3.825f, -1.237f, -0.0f, 3.734f, -1.734f, -0.0f, 3.6f, -1.95f, -0.2686f, 3.916f, -0.6313f, -0.4857f, 3.825f, -1.142f, -0.6807f, 3.734f, -1.6f, -0.7654f, 3.6f, -1.799f, -0.4859f, 3.916f, -0.4859f, -0.8786f, 3.825f, -0.8786f, -1.231f, 3.734f, -1.231f, -1.385f, 3.6f, -1.385f, -0.6313f, 3.916f, -0.2686f, -1.142f, 3.825f, -0.4857f, -1.6f, 3.734f, -0.6807f, -1.799f, 3.6f, -0.7654f -}; - - -/* 530 normals */ -const int numNormals = 530; -const float normals[530][3] = { -0.0486f, -0.9986f, 0.0168f, -0.9976f, -0.0678f, -0.0008f, --0.233f, 0.8502f, -0.4719f, --0.2299f, 0.9679f, 0.1004f, --0.1648f, 0.985f, 0.0501f, --0.0117f, 0.7461f, 0.6656f, --0.0888f, 0.9692f, 0.2294f, -0.6449f, -0.7172f, -0.2637f, --0.066f, 0.9851f, 0.1583f, --0.6585f, -0.342f, -0.6703f, --0.293f, 0.9558f, 0.0209f, -0.179f, 0.9825f, -0.0513f, --0.0094f, 0.903f, 0.4295f, --0.0059f, -0.986f, -0.1662f, --0.7355f, 0.6774f, -0.0026f, --0.997f, 0.0763f, 0.0019f, --0.1478f, 0.9333f, 0.3271f, --0.3014f, -0.6034f, -0.7382f, --0.7048f, -0.0681f, 0.706f, --0.3361f, 0.9332f, 0.1263f, -0.3709f, 0.1524f, -0.916f, --0.3399f, -0.4121f, 0.8453f, -0.1921f, 0.9724f, -0.1316f, --0.2671f, 0.7429f, 0.6137f, -0.0888f, 0.9692f, -0.2294f, -0.066f, 0.9851f, -0.1583f, -0.9411f, 0.338f, 0.001f, -0.8666f, -0.2559f, 0.4282f, --0.8029f, 0.4968f, 0.3293f, --0.0008f, -0.0678f, -0.9976f, --0.8453f, -0.4121f, -0.3399f, --0.4801f, -0.8741f, 0.0733f, -0.6355f, -0.772f, 0.0006f, --0.9215f, -0.0678f, 0.3822f, --0.6698f, -0.6907f, -0.2723f, -0.3734f, 0.876f, -0.3051f, -0.3548f, -0.4118f, 0.8393f, --0.3629f, 0.2429f, 0.8995f, -0.9033f, 0.2079f, 0.375f, --0.2824f, 0.5939f, 0.7532f, -0.8938f, 0.4452f, 0.0532f, -0.1478f, 0.9333f, -0.3271f, -0.0085f, -0.0031f, -0.9999f, -0.3595f, 0.933f, 0.0115f, -0.8995f, 0.2429f, 0.3629f, -0.7048f, -0.0681f, -0.706f, --0.6428f, -0.7172f, -0.2688f, -0.6366f, -0.447f, 0.6283f, --0.1213f, -0.9861f, -0.1128f, -0.8003f, 0.4978f, 0.334f, -0.3361f, 0.9332f, -0.1263f, -0.3399f, -0.4121f, -0.8453f, --0.3909f, 0.4452f, 0.8055f, -0.0117f, 0.7462f, -0.6655f, -0.9215f, -0.0678f, -0.3822f, -0.3582f, -0.7656f, 0.5343f, --0.9782f, 0.2075f, -0.0011f, -0.2824f, 0.5939f, -0.7532f, -0.035f, -0.8413f, 0.5393f, --0.8044f, 0.5934f, 0.0262f, --0.1128f, -0.9861f, 0.1213f, -0.13f, -0.1396f, 0.9816f, -0.6644f, 0.3392f, 0.6659f, --0.0042f, -0.6898f, -0.7239f, --0.1587f, 0.9851f, 0.065f, --0.8719f, -0.3415f, 0.3508f, -0.6486f, 0.4756f, -0.5941f, --0.4991f, 0.8499f, -0.1684f, --0.3969f, 0.6342f, -0.6634f, -0.7041f, -0.3863f, -0.5956f, -0.3909f, 0.4452f, -0.8055f, --0.0391f, -0.0113f, 0.9991f, --0.3321f, 0.5936f, -0.733f, -0.8523f, -0.5219f, -0.0338f, -0.329f, 0.4978f, 0.8023f, -0.8044f, 0.5934f, -0.0262f, -0.1128f, -0.9861f, -0.1213f, -0.0178f, 0.9861f, -0.1651f, -0.3491f, 0.4045f, 0.8452f, --0.2727f, 0.8505f, 0.4496f, -0.065f, 0.9851f, 0.1587f, --0.0005f, 0.4037f, 0.9148f, --0.0077f, -0.4109f, -0.9116f, -0.5609f, -0.604f, 0.5661f, -0.8236f, 0.5668f, -0.0138f, -0.1587f, 0.9851f, -0.065f, -0.8719f, -0.3415f, -0.3508f, --0.7382f, -0.6034f, 0.3014f, -0.0346f, 0.8495f, 0.5263f, --0.4373f, -0.7921f, -0.4257f, --0.0532f, 0.4452f, 0.8938f, -0.0689f, -0.9861f, 0.1509f, --0.1509f, -0.9861f, 0.0689f, -0.7706f, -0.2424f, -0.5893f, --0.7543f, -0.6564f, 0.0105f, -0.0005f, 0.4037f, -0.9148f, --0.9116f, -0.4109f, 0.0077f, -0.0058f, -0.0438f, 0.999f, -0.1719f, 0.985f, 0.0005f, --0.1697f, 0.9693f, 0.1774f, -0.5874f, -0.5124f, 0.6263f, -0.7382f, -0.6034f, -0.3014f, --0.1518f, 0.985f, -0.081f, -0.646f, 0.4051f, 0.6468f, -0.334f, 0.4978f, -0.8003f, --0.7354f, -0.6034f, -0.3082f, --0.6919f, 0.2428f, -0.6798f, -0.0532f, 0.4452f, -0.8938f, -0.3547f, -0.3173f, 0.8794f, -0.9879f, -0.1547f, -0.0033f, --0.0462f, -0.9986f, 0.0223f, --0.6088f, 0.4806f, 0.6311f, --0.109f, -0.1969f, -0.9743f, -0.1509f, -0.9861f, -0.0689f, --0.0568f, 0.9983f, 0.0009f, -0.9074f, -0.3096f, -0.2839f, -0.8677f, 0.4969f, 0.0026f, --0.2723f, -0.6907f, 0.6698f, --0.4734f, -0.6798f, 0.5599f, -0.9116f, -0.4109f, -0.0077f, -0.1697f, 0.9693f, -0.1774f, -0.5875f, 0.5937f, 0.5497f, --0.3232f, 0.6846f, 0.6533f, --0.5078f, -0.6913f, 0.5139f, --0.4612f, 0.7474f, -0.478f, --0.2071f, -0.8049f, 0.556f, --0.6976f, -0.7164f, -0.0027f, --0.8697f, 0.3388f, 0.3587f, -0.0462f, -0.9986f, -0.0223f, -0.2723f, -0.6907f, -0.6698f, --0.829f, -0.4466f, -0.3365f, -0.9148f, 0.4037f, 0.0005f, --0.1583f, 0.9851f, -0.066f, -0.148f, 0.9838f, 0.1002f, --0.1717f, 0.985f, -0.0162f, --0.4282f, -0.2559f, 0.8666f, -0.3094f, -0.2556f, 0.9159f, -0.2803f, -0.6907f, 0.6665f, --0.6154f, 0.497f, 0.6117f, --0.0262f, 0.5934f, -0.8044f, -0.0286f, 0.1639f, -0.986f, --0.6924f, 0.2083f, 0.6907f, --0.0105f, 0.9975f, -0.0685f, -0.5078f, -0.6913f, -0.5139f, -0.2071f, -0.8049f, -0.556f, --0.4903f, -0.7178f, -0.4942f, --0.2637f, -0.7172f, -0.6449f, --0.3822f, -0.0678f, -0.9215f, -0.8697f, 0.3388f, -0.3587f, -0.2461f, -0.805f, 0.5397f, --0.2615f, 0.9334f, 0.2452f, -0.6187f, 0.747f, -0.243f, -0.0375f, -0.8401f, -0.5411f, -0.0054f, 0.9691f, 0.2464f, -0.3587f, 0.3388f, 0.8697f, -0.3993f, 0.6582f, -0.6381f, --0.3476f, -0.4464f, -0.8245f, -0.099f, 0.9692f, 0.2251f, --0.3666f, -0.3412f, 0.8655f, -0.0396f, 0.153f, -0.9874f, -0.0349f, 0.9969f, -0.0698f, -0.1096f, 0.985f, 0.1324f, --0.0578f, -0.9861f, 0.1556f, -0.4479f, -0.5145f, -0.7311f, -0.6924f, 0.2083f, -0.6907f, -0.6096f, 0.747f, 0.265f, --0.3508f, -0.3415f, -0.8719f, --0.6215f, 0.4454f, -0.6443f, --0.4942f, -0.7178f, 0.4903f, --0.9402f, -0.3403f, -0.0085f, -0.0056f, -0.0358f, 0.9993f, -0.2615f, 0.9334f, -0.2452f, --0.0024f, 0.0291f, -0.9995f, --0.2667f, 0.9637f, -0.001f, -0.0569f, -0.2712f, -0.9608f, -0.7463f, 0.254f, 0.615f, -0.5153f, 0.6516f, -0.5564f, -0.0223f, -0.9986f, 0.0462f, -0.3666f, -0.3412f, -0.8655f, -0.0578f, -0.9861f, -0.1556f, -0.6111f, 0.4984f, 0.6148f, --0.243f, 0.747f, -0.6187f, --0.0092f, 0.2338f, -0.9722f, -0.478f, 0.7474f, -0.4612f, --0.0058f, -0.4457f, -0.8951f, --0.4856f, -0.6774f, -0.5524f, -0.54f, 0.6414f, 0.5448f, --0.3365f, -0.4466f, 0.829f, --0.2257f, 0.795f, 0.5629f, -0.8055f, 0.4452f, 0.3909f, -0.3729f, 0.208f, 0.9042f, --0.727f, -0.2562f, 0.6369f, --0.0514f, -0.9986f, 0.0029f, -0.9159f, 0.1555f, -0.3699f, -0.0019f, -0.2377f, -0.9713f, -0.4942f, -0.7178f, -0.4903f, -0.6497f, -0.4127f, 0.6383f, -0.0089f, 0.0486f, -0.9987f, --0.0213f, 0.6301f, -0.7761f, --0.9269f, -0.3751f, 0.0038f, --0.1215f, 0.9852f, 0.1207f, --0.5856f, 0.5198f, 0.6218f, -0.8655f, -0.3412f, 0.3666f, --0.2464f, 0.9691f, 0.0054f, -0.0123f, 0.1386f, 0.9902f, -0.0179f, -0.0369f, 0.9991f, --0.1207f, 0.9852f, -0.1215f, --0.0081f, 0.5671f, 0.8235f, --0.8689f, 0.3387f, -0.3607f, -0.0062f, 0.0309f, -0.9995f, -0.3365f, -0.4466f, -0.829f, --0.3787f, 0.2424f, -0.8931f, --0.2904f, 0.4454f, -0.8468f, --0.8707f, 0.4915f, 0.0133f, -0.163f, -0.8182f, 0.5512f, -0.4337f, -0.8052f, 0.4041f, -0.0514f, -0.9986f, -0.0029f, --0.0084f, 0.1303f, 0.9914f, --0.706f, -0.0681f, -0.7048f, --0.556f, -0.8049f, -0.2071f, -0.8448f, 0.4045f, 0.3501f, -0.4259f, -0.5474f, 0.7203f, --0.6907f, 0.2083f, -0.6924f, -0.1215f, 0.9852f, -0.1207f, --0.1263f, 0.9332f, -0.3361f, -0.7711f, -0.0741f, -0.6323f, -0.2464f, 0.9691f, -0.0054f, -0.1774f, 0.9693f, 0.1697f, --0.9042f, 0.208f, 0.3729f, --0.8393f, -0.4118f, 0.3548f, -0.6888f, -0.7219f, -0.0648f, -0.1556f, -0.9861f, 0.0578f, -0.3271f, 0.9333f, 0.1478f, --0.0024f, 0.2379f, 0.9712f, --0.0026f, 0.4969f, 0.8677f, -0.0f, 1.0f, 0.0f, -0.1912f, -0.9815f, -0.0025f, --0.3762f, -0.6681f, 0.6418f, --0.7759f, 0.0432f, 0.6292f, --0.0208f, -0.8044f, -0.5936f, --0.2274f, 0.8822f, -0.4122f, -0.7532f, 0.5939f, 0.2824f, --0.9221f, -0.0681f, -0.3807f, --0.2198f, 0.8494f, 0.4796f, -0.0065f, -0.7656f, 0.6431f, --0.5876f, 0.4472f, -0.6742f, -0.7981f, -0.6024f, 0.0036f, --0.0383f, -0.9986f, -0.0341f, --0.6369f, -0.2562f, -0.727f, --0.5497f, 0.5937f, 0.5875f, -0.1084f, 0.9431f, 0.314f, -0.9042f, 0.208f, -0.3729f, --0.6659f, 0.3392f, 0.6644f, -0.8393f, -0.4118f, -0.3548f, -0.0029f, -0.9986f, 0.0514f, --0.9647f, -0.2552f, -0.0635f, --0.2294f, 0.9692f, -0.0888f, -0.0026f, 0.4969f, -0.8677f, -0.2452f, 0.9334f, 0.2615f, -0.5171f, -0.4876f, -0.7033f, --0.8951f, -0.4457f, 0.0058f, --0.5936f, -0.8044f, 0.0208f, -0.5642f, -0.5426f, -0.6222f, -0.5938f, 0.4451f, 0.6702f, -0.5497f, 0.5937f, -0.5875f, -0.6657f, 0.4653f, 0.5832f, -0.4857f, -0.6243f, 0.6117f, --0.0486f, -0.9986f, -0.0168f, --0.6468f, 0.4051f, 0.646f, -0.6659f, 0.3392f, -0.6644f, -0.1833f, 0.9735f, -0.1365f, -0.3955f, 0.8505f, 0.3465f, -0.5139f, -0.6913f, 0.5078f, -0.8023f, 0.4978f, -0.329f, --0.001f, 0.338f, 0.9411f, --0.2496f, 0.8321f, -0.4951f, -0.8951f, -0.4457f, -0.0058f, -0.233f, 0.8502f, 0.4719f, --0.0168f, -0.9986f, 0.0486f, -0.5936f, -0.8044f, -0.0208f, --0.05f, 0.3155f, 0.9475f, -0.6585f, -0.342f, 0.6703f, -0.4909f, -0.1864f, -0.8509f, --0.37f, 0.9238f, -0.0973f, -0.6468f, 0.4051f, -0.646f, -0.0059f, -0.986f, 0.1662f, --0.3724f, 0.9278f, -0.0202f, --0.3501f, 0.4045f, 0.8448f, --0.0425f, 0.8398f, -0.5411f, --0.1684f, 0.8499f, 0.4991f, --0.6665f, -0.6907f, 0.2803f, --0.2251f, 0.9692f, 0.099f, -0.9241f, -0.3816f, -0.0169f, -0.001f, 0.338f, -0.9411f, --0.9411f, 0.338f, -0.001f, --0.8666f, -0.2559f, -0.4282f, -0.0262f, 0.5183f, -0.8547f, -0.3014f, -0.6034f, 0.7382f, -0.0168f, -0.9986f, -0.0486f, --0.3548f, -0.4118f, -0.8393f, --0.6023f, -0.5297f, 0.5971f, --0.9033f, 0.2079f, -0.375f, --0.8938f, 0.4452f, -0.0532f, -0.6044f, 0.7397f, 0.2957f, -0.0008f, -0.0678f, 0.9976f, -0.7058f, 0.0906f, -0.7025f, -0.8453f, -0.4121f, 0.3399f, --0.3595f, 0.933f, -0.0115f, -0.6698f, -0.6907f, 0.2723f, --0.8995f, 0.2429f, -0.3629f, --0.6366f, -0.447f, -0.6283f, -0.3501f, 0.4045f, -0.8448f, --0.01f, -0.0605f, 0.9981f, --0.8003f, 0.4978f, -0.334f, -0.1684f, 0.8499f, -0.4991f, -0.6665f, -0.6907f, -0.2803f, -0.2251f, 0.9692f, -0.099f, --0.0036f, -0.6024f, 0.7981f, -0.6637f, -0.2967f, -0.6865f, --0.081f, 0.985f, 0.1518f, -0.0084f, 0.2423f, 0.9701f, -0.0071f, -0.9029f, -0.4296f, --0.8679f, 0.4966f, -0.0026f, -0.0123f, 0.5735f, 0.819f, --0.0005f, 0.985f, 0.1719f, -0.6428f, -0.7172f, 0.2688f, -0.6588f, -0.3366f, 0.6727f, -0.1213f, -0.9861f, 0.1128f, --0.8931f, 0.2424f, 0.3787f, --0.1662f, -0.986f, 0.0059f, -0.9994f, 0.0313f, 0.0095f, -0.762f, -0.146f, 0.6308f, --0.7731f, 0.0861f, -0.6283f, --0.6644f, 0.3392f, -0.6659f, --0.0027f, -0.7164f, 0.6976f, -0.0036f, -0.6024f, -0.7981f, -0.9782f, 0.2075f, 0.0011f, -0.0405f, -0.9991f, -0.0018f, -0.6882f, -0.703f, 0.179f, --0.0115f, 0.933f, 0.3595f, -0.0911f, 0.0518f, -0.9944f, -0.0005f, 0.985f, -0.1719f, -0.5337f, -0.5852f, -0.6104f, -0.0042f, -0.6898f, 0.7239f, -0.4863f, 0.2366f, 0.8411f, -0.4991f, 0.8499f, 0.1684f, --0.6543f, 0.7561f, 0.0071f, -0.265f, 0.747f, -0.6096f, --0.329f, 0.4978f, -0.8023f, -0.1662f, -0.986f, -0.0059f, --0.3491f, 0.4045f, -0.8452f, -0.3321f, 0.5936f, 0.733f, --0.065f, 0.9851f, -0.1587f, --0.6283f, -0.447f, 0.6366f, -0.0027f, -0.7164f, -0.6976f, --0.1316f, 0.6339f, 0.762f, --0.5609f, -0.604f, -0.5661f, --0.8452f, 0.4045f, 0.3491f, --0.5263f, 0.8495f, 0.0346f, -0.0115f, 0.933f, -0.3595f, --0.0346f, 0.8495f, -0.5263f, -0.0077f, -0.4109f, 0.9116f, -0.5758f, -0.8175f, -0.0017f, --0.0011f, 0.2075f, 0.9782f, --0.0689f, -0.9861f, -0.1509f, -0.2934f, -0.5928f, -0.7499f, -0.0724f, 0.1198f, -0.9901f, --0.7367f, -0.275f, -0.6176f, --0.3131f, 0.8154f, 0.4868f, --0.0114f, 0.0022f, 0.9999f, -0.6283f, -0.447f, -0.6366f, -0.8452f, 0.4045f, -0.3491f, -0.5263f, 0.8495f, -0.0346f, --0.6383f, -0.4127f, 0.6497f, --0.1719f, 0.985f, -0.0005f, --0.6703f, -0.342f, 0.6585f, --0.0085f, -0.3403f, 0.9402f, --0.646f, 0.4051f, -0.6468f, -0.0011f, 0.2075f, -0.9782f, --0.7216f, -0.3071f, 0.6204f, -0.0282f, 0.0023f, -0.9995f, --0.2483f, 0.6806f, -0.6892f, -0.1518f, 0.985f, 0.081f, -0.047f, 0.0466f, -0.9978f, -0.7354f, -0.6034f, 0.3082f, -0.6919f, 0.2428f, 0.6798f, -0.4086f, -0.3626f, -0.8375f, -0.6383f, -0.4127f, -0.6497f, --0.5875f, 0.5937f, -0.5497f, -0.6703f, -0.342f, -0.6585f, --0.8245f, -0.4464f, 0.3476f, -0.0085f, -0.3403f, -0.9402f, --0.0591f, -0.0663f, 0.996f, -0.0f, -1.0f, 0.0f, -0.4612f, 0.7474f, 0.478f, -0.6976f, -0.7164f, 0.0027f, --0.9148f, 0.4037f, -0.0005f, -0.173f, -0.8158f, -0.5518f, --0.3607f, 0.3387f, 0.8689f, -0.7836f, -0.2411f, 0.5724f, --0.1985f, 0.8026f, -0.5623f, --0.3094f, -0.2556f, -0.9159f, --0.2803f, -0.6907f, -0.6665f, -0.8245f, -0.4464f, -0.3476f, -0.829f, -0.4466f, 0.3365f, --0.4848f, 0.7385f, 0.4683f, -0.1583f, 0.9851f, 0.066f, --0.0077f, 0.7656f, -0.6432f, --0.0162f, 0.985f, 0.1717f, -0.1717f, 0.985f, 0.0162f, -0.0244f, 0.9805f, -0.1949f, --0.2461f, -0.805f, -0.5397f, -0.0262f, 0.5934f, 0.8044f, -0.142f, 0.1881f, 0.9718f, -0.1846f, 0.1002f, 0.9776f, -0.4903f, -0.7178f, 0.4942f, -0.2637f, -0.7172f, 0.6449f, -0.3822f, -0.0678f, 0.9215f, --0.0054f, 0.9691f, -0.2464f, -0.3607f, 0.3387f, -0.8689f, --0.3587f, 0.3388f, -0.8697f, --0.5694f, -0.8219f, 0.0081f, --0.1324f, 0.985f, 0.1096f, --0.099f, 0.9692f, -0.2251f, --0.6702f, 0.4451f, 0.5938f, -0.0077f, -0.9976f, 0.0684f, --0.5661f, -0.604f, 0.5609f, --0.1096f, 0.985f, -0.1324f, --0.6096f, 0.747f, -0.265f, --0.0015f, 0.0295f, -0.9995f, -0.3476f, -0.4464f, 0.8245f, --0.0635f, -0.2552f, 0.9647f, --0.8468f, 0.4454f, 0.2904f, --0.4719f, 0.8502f, 0.233f, --0.0502f, 0.8385f, 0.5425f, --0.6671f, 0.7448f, -0.0116f, -0.3508f, -0.3415f, 0.8719f, --0.4119f, 0.6135f, -0.6736f, --0.2688f, -0.7172f, 0.6428f, --0.4041f, -0.8052f, 0.4337f, --0.375f, 0.2079f, 0.9033f, --0.0223f, -0.9986f, -0.0462f, -0.6702f, 0.4451f, -0.5938f, -0.9402f, -0.3403f, 0.0085f, -0.5661f, -0.604f, -0.5609f, --0.6252f, 0.7406f, 0.246f, --0.0341f, -0.9986f, 0.0383f, --0.6111f, 0.4984f, -0.6148f, -0.6655f, 0.7462f, 0.0117f, -0.1233f, 0.199f, 0.9722f, -0.8468f, 0.4454f, -0.2904f, -0.7383f, 0.2702f, -0.6179f, --0.8055f, 0.4452f, -0.3909f, --0.3729f, 0.208f, -0.9042f, -0.4719f, 0.8502f, -0.233f, -0.243f, 0.747f, 0.6187f, --0.6497f, -0.4127f, -0.6383f, --0.5406f, 0.5651f, -0.623f, -0.0058f, -0.4457f, 0.8951f, --0.3082f, -0.6034f, 0.7354f, --0.8655f, -0.3412f, -0.3666f, -0.2688f, -0.7172f, -0.6428f, -0.4041f, -0.8052f, -0.4337f, -0.375f, 0.2079f, -0.9033f, -0.0341f, -0.9986f, -0.0383f, --0.9701f, 0.2423f, 0.0084f, --0.3807f, -0.0681f, 0.9221f, -0.9643f, -0.2551f, 0.0705f, --0.8758f, 0.4808f, 0.0415f, -0.1207f, 0.9852f, 0.1215f, -0.4821f, 0.7724f, 0.4133f, --0.0522f, 0.9982f, 0.0278f, --0.4337f, -0.8052f, -0.4041f, --0.6164f, 0.4198f, 0.6661f, --0.8448f, 0.4045f, -0.3501f, -0.3082f, -0.6034f, -0.7354f, -0.8689f, 0.3387f, 0.3607f, -0.6894f, -0.7242f, 0.0091f, -0.3787f, 0.2424f, 0.8931f, -0.2904f, 0.4454f, 0.8468f, -0.6148f, 0.4984f, -0.6111f, -0.0501f, 0.985f, 0.1648f, --0.5397f, -0.805f, 0.2461f, --0.9159f, -0.2556f, 0.3094f, -0.706f, -0.0681f, 0.7048f, --0.3341f, 0.4972f, 0.8006f, -0.556f, -0.8049f, 0.2071f, --0.1774f, 0.9693f, -0.1697f, -0.6907f, 0.2083f, 0.6924f, -0.1263f, 0.9332f, 0.3361f, -0.3807f, -0.0681f, -0.9221f, --0.1556f, -0.9861f, -0.0578f, --0.3271f, 0.9333f, -0.1478f, --0.3465f, 0.8505f, 0.3955f, -0.5315f, 0.8438f, -0.0735f, -0.9737f, 0.2276f, -0.0003f, -0.6441f, 0.7648f, -0.0112f, --0.7239f, -0.6898f, 0.0042f, --0.7532f, 0.5939f, -0.2824f, -0.1093f, 0.1415f, -0.9838f, -0.5397f, -0.805f, -0.2461f, --0.7981f, -0.6024f, -0.0036f, -0.9456f, 0.3251f, -0.0052f, -0.1278f, 0.9696f, -0.2085f, -0.0208f, -0.8044f, 0.5936f, -0.1635f, 0.1348f, -0.9772f, --0.733f, 0.5936f, 0.3321f, --0.0505f, 0.9852f, -0.1635f, -0.4089f, -0.9069f, -0.1015f, --0.0029f, -0.9986f, -0.0514f, --0.1796f, 0.814f, -0.5522f, -0.9221f, -0.0681f, 0.3807f, -0.0383f, -0.9986f, 0.0341f, -0.6369f, -0.2562f, 0.727f, -0.3465f, 0.8505f, -0.3955f, --0.2452f, 0.9334f, -0.2615f, -0.4921f, -0.247f, 0.8346f, --0.9976f, -0.0678f, 0.0008f, --0.5396f, 0.8418f, -0.0094f, -0.2294f, 0.9692f, 0.0888f, -0.7239f, -0.6898f, -0.0042f, --0.4472f, 0.5952f, 0.6675f, --0.6449f, -0.7172f, 0.2637f, -0.4543f, 0.2732f, -0.8478f, --0.6798f, 0.2428f, 0.6919f, --0.5938f, 0.4451f, -0.6702f, -0.733f, 0.5936f, -0.3321f, --0.3955f, 0.8505f, -0.3465f, --0.5139f, -0.6913f, -0.5078f, --0.623f, -0.5156f, -0.5881f -}; - -/* 1 color */ -/*255 255 0 */ - -/* 1024 faces */ -/* numIdx fidx0 fidx1 fidx2 nidx0 nidx1 nidx2 coloridx */ - -const int numFaces = 1024; -const int faces[1024][8] = { -3, 0, 5, 6, 255, 295, 309, 0, -3, 6, 1, 0, 309, 465, 255, 0, -3, 1, 6, 7, 465, 309, 134, 0, -3, 7, 2, 1, 134, 4, 465, 0, -3, 2, 7, 8, 4, 134, 165, 0, -3, 8, 3, 2, 165, 448, 4, 0, -3, 3, 8, 9, 448, 165, 49, 0, -3, 9, 4, 3, 49, 116, 448, 0, -3, 5, 10, 11, 295, 248, 106, 0, -3, 11, 6, 5, 106, 309, 295, 0, -3, 6, 11, 12, 309, 106, 102, 0, -3, 12, 7, 6, 102, 134, 309, 0, -3, 7, 12, 13, 134, 102, 394, 0, -3, 13, 8, 7, 394, 165, 134, 0, -3, 8, 13, 14, 165, 394, 180, 0, -3, 14, 9, 8, 180, 49, 165, 0, -3, 10, 15, 16, 248, 401, 211, 0, -3, 16, 11, 10, 211, 106, 248, 0, -3, 11, 16, 17, 106, 211, 427, 0, -3, 17, 12, 11, 427, 102, 106, 0, -3, 12, 17, 18, 102, 427, 455, 0, -3, 18, 13, 12, 455, 394, 102, 0, -3, 13, 18, 19, 394, 455, 74, 0, -3, 19, 14, 13, 74, 180, 394, 0, -3, 15, 20, 21, 401, 174, 182, 0, -3, 21, 16, 15, 182, 211, 401, 0, -3, 16, 21, 22, 211, 182, 507, 0, -3, 22, 17, 16, 507, 427, 211, 0, -3, 17, 22, 23, 427, 507, 5, 0, -3, 23, 18, 17, 5, 455, 427, 0, -3, 18, 23, 24, 455, 5, 234, 0, -3, 24, 19, 18, 234, 74, 455, 0, -3, 20, 25, 26, 174, 386, 20, 0, -3, 26, 21, 20, 20, 182, 174, 0, -3, 21, 26, 27, 182, 20, 410, 0, -3, 27, 22, 21, 410, 507, 182, 0, -3, 22, 27, 28, 507, 410, 23, 0, -3, 28, 23, 22, 23, 5, 507, 0, -3, 23, 28, 29, 5, 23, 485, 0, -3, 29, 24, 23, 485, 234, 5, 0, -3, 25, 30, 31, 386, 69, 305, 0, -3, 31, 26, 25, 305, 20, 386, 0, -3, 26, 31, 32, 20, 305, 503, 0, -3, 32, 27, 26, 503, 410, 20, 0, -3, 27, 32, 33, 410, 503, 405, 0, -3, 33, 28, 27, 405, 23, 410, 0, -3, 28, 33, 34, 23, 405, 138, 0, -3, 34, 29, 28, 138, 485, 23, 0, -3, 30, 35, 36, 69, 115, 193, 0, -3, 36, 31, 30, 193, 305, 69, 0, -3, 31, 36, 37, 305, 193, 270, 0, -3, 37, 32, 31, 270, 503, 305, 0, -3, 32, 37, 38, 503, 270, 445, 0, -3, 38, 33, 32, 445, 405, 503, 0, -3, 33, 38, 39, 405, 445, 28, 0, -3, 39, 34, 33, 28, 138, 405, 0, -3, 35, 40, 41, 115, 467, 495, 0, -3, 41, 36, 35, 495, 193, 115, 0, -3, 36, 41, 42, 193, 495, 11, 0, -3, 42, 37, 36, 11, 270, 193, 0, -3, 37, 42, 43, 270, 11, 435, 0, -3, 43, 38, 37, 435, 445, 270, 0, -3, 38, 43, 44, 445, 435, 322, 0, -3, 44, 39, 38, 322, 28, 445, 0, -3, 40, 45, 46, 467, 27, 44, 0, -3, 46, 41, 40, 44, 495, 467, 0, -3, 41, 46, 47, 495, 44, 409, 0, -3, 47, 42, 41, 409, 11, 495, 0, -3, 42, 47, 48, 11, 409, 428, 0, -3, 48, 43, 42, 428, 435, 11, 0, -3, 43, 48, 49, 435, 428, 313, 0, -3, 49, 44, 43, 313, 322, 435, 0, -3, 45, 50, 51, 27, 513, 385, 0, -3, 51, 46, 45, 385, 44, 27, 0, -3, 46, 51, 52, 44, 385, 382, 0, -3, 52, 47, 46, 382, 409, 44, 0, -3, 47, 52, 53, 409, 382, 124, 0, -3, 53, 48, 47, 124, 428, 409, 0, -3, 48, 53, 54, 428, 124, 447, 0, -3, 54, 49, 48, 447, 313, 428, 0, -3, 50, 55, 56, 513, 136, 478, 0, -3, 56, 51, 50, 478, 385, 513, 0, -3, 51, 56, 57, 385, 478, 161, 0, -3, 57, 52, 51, 161, 382, 385, 0, -3, 52, 57, 58, 382, 161, 181, 0, -3, 58, 53, 52, 181, 124, 382, 0, -3, 53, 58, 59, 124, 181, 348, 0, -3, 59, 54, 53, 348, 447, 124, 0, -3, 55, 60, 61, 136, 431, 320, 0, -3, 61, 56, 55, 320, 478, 136, 0, -3, 56, 61, 62, 478, 320, 481, 0, -3, 62, 57, 56, 481, 161, 478, 0, -3, 57, 62, 63, 161, 481, 53, 0, -3, 63, 58, 57, 53, 181, 161, 0, -3, 58, 63, 64, 181, 53, 257, 0, -3, 64, 59, 58, 257, 348, 181, 0, -3, 60, 65, 66, 431, 135, 37, 0, -3, 66, 61, 60, 37, 320, 431, 0, -3, 61, 66, 67, 320, 37, 408, 0, -3, 67, 62, 61, 408, 481, 320, 0, -3, 62, 67, 68, 481, 408, 347, 0, -3, 68, 63, 62, 347, 53, 481, 0, -3, 63, 68, 69, 53, 347, 104, 0, -3, 69, 64, 63, 104, 257, 53, 0, -3, 65, 70, 71, 135, 191, 524, 0, -3, 71, 66, 65, 524, 37, 135, 0, -3, 66, 71, 72, 37, 524, 319, 0, -3, 72, 67, 66, 319, 408, 37, 0, -3, 67, 72, 73, 408, 319, 183, 0, -3, 73, 68, 67, 183, 347, 408, 0, -3, 68, 73, 74, 347, 183, 480, 0, -3, 74, 69, 68, 480, 104, 347, 0, -3, 70, 75, 76, 191, 483, 328, 0, -3, 76, 71, 70, 328, 524, 191, 0, -3, 71, 76, 77, 524, 328, 422, 0, -3, 77, 72, 71, 422, 319, 524, 0, -3, 72, 77, 78, 319, 422, 151, 0, -3, 78, 73, 72, 151, 183, 319, 0, -3, 73, 78, 79, 183, 151, 273, 0, -3, 79, 74, 73, 273, 480, 183, 0, -3, 75, 0, 1, 483, 255, 465, 0, -3, 1, 76, 75, 465, 328, 483, 0, -3, 76, 1, 2, 328, 465, 4, 0, -3, 2, 77, 76, 4, 422, 328, 0, -3, 77, 2, 3, 422, 4, 448, 0, -3, 3, 78, 77, 448, 151, 422, 0, -3, 78, 3, 4, 151, 448, 116, 0, -3, 4, 79, 78, 116, 273, 151, 0, -3, 4, 9, 84, 116, 49, 220, 0, -3, 84, 80, 4, 220, 131, 116, 0, -3, 80, 84, 85, 131, 220, 476, 0, -3, 85, 81, 80, 476, 26, 131, 0, -3, 81, 85, 86, 26, 476, 38, 0, -3, 86, 82, 81, 38, 336, 26, 0, -3, 82, 86, 87, 336, 38, 511, 0, -3, 87, 83, 82, 511, 1, 336, 0, -3, 9, 14, 88, 49, 180, 103, 0, -3, 88, 84, 9, 103, 220, 49, 0, -3, 84, 88, 89, 220, 103, 62, 0, -3, 89, 85, 84, 62, 476, 220, 0, -3, 85, 89, 90, 476, 62, 488, 0, -3, 90, 86, 85, 488, 38, 476, 0, -3, 86, 90, 91, 38, 488, 484, 0, -3, 91, 87, 86, 484, 511, 38, 0, -3, 14, 19, 92, 180, 74, 78, 0, -3, 92, 88, 14, 78, 103, 180, 0, -3, 88, 92, 93, 103, 78, 154, 0, -3, 93, 89, 88, 154, 62, 103, 0, -3, 89, 93, 94, 62, 154, 190, 0, -3, 94, 90, 89, 190, 488, 62, 0, -3, 90, 94, 95, 488, 190, 417, 0, -3, 95, 91, 90, 417, 484, 488, 0, -3, 19, 24, 96, 74, 234, 81, 0, -3, 96, 92, 19, 81, 78, 74, 0, -3, 92, 96, 97, 78, 81, 274, 0, -3, 97, 93, 92, 274, 154, 78, 0, -3, 93, 97, 98, 154, 274, 363, 0, -3, 98, 94, 93, 363, 190, 154, 0, -3, 94, 98, 99, 190, 363, 304, 0, -3, 99, 95, 94, 304, 417, 190, 0, -3, 24, 29, 100, 234, 485, 287, 0, -3, 100, 96, 24, 287, 81, 234, 0, -3, 96, 100, 101, 81, 287, 398, 0, -3, 101, 97, 96, 398, 274, 81, 0, -3, 97, 101, 102, 274, 398, 440, 0, -3, 102, 98, 97, 440, 363, 274, 0, -3, 98, 102, 103, 363, 440, 466, 0, -3, 103, 99, 98, 466, 304, 363, 0, -3, 29, 34, 104, 485, 138, 268, 0, -3, 104, 100, 29, 268, 287, 485, 0, -3, 100, 104, 105, 287, 268, 252, 0, -3, 105, 101, 100, 252, 398, 287, 0, -3, 101, 105, 106, 398, 252, 141, 0, -3, 106, 102, 101, 141, 440, 398, 0, -3, 102, 106, 107, 440, 141, 18, 0, -3, 107, 103, 102, 18, 466, 440, 0, -3, 34, 39, 108, 138, 28, 357, 0, -3, 108, 104, 34, 357, 268, 138, 0, -3, 104, 108, 109, 268, 357, 127, 0, -3, 109, 105, 104, 127, 252, 268, 0, -3, 105, 109, 110, 252, 127, 228, 0, -3, 110, 106, 105, 228, 141, 252, 0, -3, 106, 110, 111, 141, 228, 33, 0, -3, 111, 107, 106, 33, 18, 141, 0, -3, 39, 44, 112, 28, 322, 396, 0, -3, 112, 108, 39, 396, 357, 28, 0, -3, 108, 112, 113, 357, 396, 294, 0, -3, 113, 109, 108, 294, 127, 357, 0, -3, 109, 113, 114, 127, 294, 56, 0, -3, 114, 110, 109, 56, 228, 127, 0, -3, 110, 114, 115, 228, 56, 517, 0, -3, 115, 111, 110, 517, 33, 228, 0, -3, 44, 49, 116, 322, 313, 474, 0, -3, 116, 112, 44, 474, 396, 322, 0, -3, 112, 116, 117, 396, 474, 208, 0, -3, 117, 113, 112, 208, 294, 396, 0, -3, 113, 117, 118, 294, 208, 301, 0, -3, 118, 114, 113, 301, 56, 294, 0, -3, 114, 118, 119, 56, 301, 242, 0, -3, 119, 115, 114, 242, 517, 56, 0, -3, 49, 54, 120, 313, 447, 377, 0, -3, 120, 116, 49, 377, 474, 313, 0, -3, 116, 120, 121, 474, 377, 333, 0, -3, 121, 117, 116, 333, 208, 474, 0, -3, 117, 121, 122, 208, 333, 222, 0, -3, 122, 118, 117, 222, 301, 208, 0, -3, 118, 122, 123, 301, 222, 218, 0, -3, 123, 119, 118, 218, 242, 301, 0, -3, 54, 59, 124, 447, 348, 350, 0, -3, 124, 120, 54, 350, 377, 447, 0, -3, 120, 124, 125, 377, 350, 420, 0, -3, 125, 121, 120, 420, 333, 377, 0, -3, 121, 125, 126, 333, 420, 453, 0, -3, 126, 122, 121, 453, 222, 333, 0, -3, 122, 126, 127, 222, 453, 147, 0, -3, 127, 123, 122, 147, 218, 222, 0, -3, 59, 64, 128, 348, 257, 95, 0, -3, 128, 124, 59, 95, 350, 348, 0, -3, 124, 128, 129, 350, 95, 293, 0, -3, 129, 125, 124, 293, 420, 350, 0, -3, 125, 129, 130, 420, 293, 378, 0, -3, 130, 126, 125, 378, 453, 420, 0, -3, 126, 130, 131, 453, 378, 29, 0, -3, 131, 127, 126, 29, 147, 453, 0, -3, 64, 69, 132, 257, 104, 311, 0, -3, 132, 128, 64, 311, 95, 257, 0, -3, 128, 132, 133, 95, 311, 419, 0, -3, 133, 129, 128, 419, 293, 95, 0, -3, 129, 133, 134, 293, 419, 463, 0, -3, 134, 130, 129, 463, 378, 293, 0, -3, 130, 134, 135, 378, 463, 490, 0, -3, 135, 131, 130, 490, 29, 378, 0, -3, 69, 74, 136, 104, 480, 284, 0, -3, 136, 132, 69, 284, 311, 104, 0, -3, 132, 136, 137, 311, 284, 269, 0, -3, 137, 133, 132, 269, 419, 311, 0, -3, 133, 137, 138, 419, 269, 164, 0, -3, 138, 134, 133, 164, 463, 419, 0, -3, 134, 138, 139, 463, 164, 45, 0, -3, 139, 135, 134, 45, 490, 463, 0, -3, 74, 79, 140, 480, 273, 371, 0, -3, 140, 136, 74, 371, 284, 480, 0, -3, 136, 140, 141, 284, 371, 148, 0, -3, 141, 137, 136, 148, 269, 284, 0, -3, 137, 141, 142, 269, 148, 251, 0, -3, 142, 138, 137, 251, 164, 269, 0, -3, 138, 142, 143, 164, 251, 54, 0, -3, 143, 139, 138, 54, 45, 164, 0, -3, 79, 4, 80, 273, 116, 131, 0, -3, 80, 140, 79, 131, 371, 273, 0, -3, 140, 80, 81, 371, 131, 26, 0, -3, 81, 141, 140, 26, 148, 371, 0, -3, 141, 81, 82, 148, 26, 336, 0, -3, 82, 142, 141, 336, 251, 148, 0, -3, 142, 82, 83, 251, 336, 1, 0, -3, 83, 143, 142, 1, 54, 251, 0, -3, 83, 87, 148, 1, 511, 404, 0, -3, 148, 144, 83, 404, 276, 1, 0, -3, 144, 148, 149, 276, 404, 308, 0, -3, 149, 145, 144, 308, 520, 276, 0, -3, 145, 149, 150, 520, 308, 325, 0, -3, 150, 146, 145, 325, 395, 520, 0, -3, 146, 150, 151, 395, 325, 384, 0, -3, 151, 147, 146, 384, 246, 395, 0, -3, 87, 91, 152, 511, 484, 47, 0, -3, 152, 148, 87, 47, 404, 511, 0, -3, 148, 152, 153, 404, 47, 272, 0, -3, 153, 149, 148, 272, 308, 404, 0, -3, 149, 153, 154, 308, 272, 415, 0, -3, 154, 150, 149, 415, 325, 308, 0, -3, 150, 154, 155, 325, 415, 83, 0, -3, 155, 151, 150, 83, 384, 325, 0, -3, 91, 95, 156, 484, 417, 430, 0, -3, 156, 152, 91, 430, 47, 484, 0, -3, 152, 156, 157, 47, 430, 137, 0, -3, 157, 153, 152, 137, 272, 47, 0, -3, 153, 157, 158, 272, 137, 416, 0, -3, 158, 154, 153, 416, 415, 272, 0, -3, 154, 158, 159, 415, 416, 297, 0, -3, 159, 155, 154, 297, 83, 415, 0, -3, 95, 99, 160, 417, 304, 458, 0, -3, 160, 156, 95, 458, 430, 417, 0, -3, 156, 160, 161, 430, 458, 343, 0, -3, 161, 157, 156, 343, 137, 430, 0, -3, 157, 161, 162, 137, 343, 334, 0, -3, 162, 158, 157, 334, 416, 137, 0, -3, 158, 162, 163, 416, 334, 317, 0, -3, 163, 159, 158, 317, 297, 416, 0, -3, 99, 103, 164, 304, 466, 187, 0, -3, 164, 160, 99, 187, 458, 304, 0, -3, 160, 164, 165, 458, 187, 117, 0, -3, 165, 161, 160, 117, 343, 458, 0, -3, 161, 165, 166, 343, 117, 438, 0, -3, 166, 162, 161, 438, 334, 343, 0, -3, 162, 166, 167, 334, 438, 459, 0, -3, 167, 163, 162, 459, 317, 334, 0, -3, 103, 107, 168, 466, 18, 353, 0, -3, 168, 164, 103, 353, 187, 466, 0, -3, 164, 168, 169, 187, 353, 123, 0, -3, 169, 165, 164, 123, 117, 187, 0, -3, 165, 169, 170, 117, 123, 168, 0, -3, 170, 166, 165, 168, 438, 117, 0, -3, 166, 170, 171, 438, 168, 426, 0, -3, 171, 167, 166, 426, 459, 438, 0, -3, 107, 111, 172, 18, 33, 390, 0, -3, 172, 168, 107, 390, 353, 18, 0, -3, 168, 172, 173, 353, 390, 290, 0, -3, 173, 169, 168, 290, 123, 353, 0, -3, 169, 173, 174, 123, 290, 522, 0, -3, 174, 170, 169, 522, 168, 123, 0, -3, 170, 174, 175, 168, 522, 87, 0, -3, 175, 171, 170, 87, 426, 168, 0, -3, 111, 115, 176, 33, 517, 260, 0, -3, 176, 172, 111, 260, 390, 33, 0, -3, 172, 176, 177, 390, 260, 497, 0, -3, 177, 173, 172, 497, 290, 390, 0, -3, 173, 177, 178, 290, 497, 126, 0, -3, 178, 174, 173, 126, 522, 290, 0, -3, 174, 178, 179, 522, 126, 501, 0, -3, 179, 175, 174, 501, 87, 522, 0, -3, 115, 119, 180, 517, 242, 130, 0, -3, 180, 176, 115, 130, 260, 517, 0, -3, 176, 180, 181, 260, 130, 34, 0, -3, 181, 177, 176, 34, 497, 260, 0, -3, 177, 181, 182, 497, 34, 46, 0, -3, 182, 178, 177, 46, 126, 497, 0, -3, 178, 182, 183, 126, 46, 105, 0, -3, 183, 179, 178, 105, 501, 126, 0, -3, 119, 123, 184, 242, 218, 310, 0, -3, 184, 180, 119, 310, 130, 242, 0, -3, 180, 184, 185, 130, 310, 528, 0, -3, 185, 181, 180, 528, 34, 130, 0, -3, 181, 185, 186, 34, 528, 145, 0, -3, 186, 182, 181, 145, 46, 34, 0, -3, 182, 186, 187, 46, 145, 356, 0, -3, 187, 183, 182, 356, 105, 46, 0, -3, 123, 127, 188, 218, 147, 156, 0, -3, 188, 184, 123, 156, 310, 218, 0, -3, 184, 188, 189, 310, 156, 402, 0, -3, 189, 185, 184, 402, 528, 310, 0, -3, 185, 189, 190, 528, 402, 146, 0, -3, 190, 186, 185, 146, 145, 528, 0, -3, 186, 190, 191, 145, 146, 17, 0, -3, 191, 187, 186, 17, 356, 145, 0, -3, 127, 131, 192, 147, 29, 184, 0, -3, 192, 188, 127, 184, 156, 147, 0, -3, 188, 192, 193, 156, 184, 63, 0, -3, 193, 189, 188, 63, 402, 156, 0, -3, 189, 193, 194, 402, 63, 354, 0, -3, 194, 190, 189, 354, 146, 402, 0, -3, 190, 194, 195, 146, 354, 335, 0, -3, 195, 191, 190, 335, 17, 146, 0, -3, 131, 135, 196, 29, 490, 210, 0, -3, 196, 192, 131, 210, 184, 29, 0, -3, 192, 196, 197, 184, 210, 129, 0, -3, 197, 193, 192, 129, 63, 184, 0, -3, 193, 197, 198, 63, 129, 461, 0, -3, 198, 194, 193, 461, 354, 63, 0, -3, 194, 198, 199, 354, 461, 475, 0, -3, 199, 195, 194, 475, 335, 354, 0, -3, 135, 139, 200, 490, 45, 370, 0, -3, 200, 196, 135, 370, 210, 490, 0, -3, 196, 200, 201, 210, 370, 143, 0, -3, 201, 197, 196, 143, 129, 210, 0, -3, 197, 201, 202, 129, 143, 195, 0, -3, 202, 198, 197, 195, 461, 129, 0, -3, 198, 202, 203, 461, 195, 444, 0, -3, 203, 199, 198, 444, 475, 461, 0, -3, 139, 143, 204, 45, 54, 403, 0, -3, 204, 200, 139, 403, 370, 45, 0, -3, 200, 204, 205, 370, 403, 315, 0, -3, 205, 201, 200, 315, 143, 370, 0, -3, 201, 205, 206, 143, 315, 7, 0, -3, 206, 202, 201, 7, 195, 143, 0, -3, 202, 206, 207, 195, 7, 101, 0, -3, 207, 203, 202, 101, 444, 195, 0, -3, 143, 83, 144, 54, 1, 276, 0, -3, 144, 204, 143, 276, 403, 54, 0, -3, 204, 144, 145, 403, 276, 520, 0, -3, 145, 205, 204, 520, 315, 403, 0, -3, 205, 145, 146, 315, 520, 395, 0, -3, 146, 206, 205, 395, 7, 315, 0, -3, 206, 146, 147, 7, 395, 246, 0, -3, 147, 207, 206, 246, 101, 7, 0, -3, 147, 151, 212, 246, 384, 486, 0, -3, 212, 208, 147, 486, 279, 246, 0, -3, 208, 212, 213, 279, 486, 231, 0, -3, 213, 209, 208, 231, 349, 279, 0, -3, 209, 213, 214, 349, 231, 0, 0, -3, 214, 210, 209, 0, 216, 349, 0, -3, 210, 214, 211, 216, 0, 393, 0, -3, 211, 211, 210, 393, 393, 216, 0, -3, 151, 155, 215, 384, 83, 215, 0, -3, 215, 212, 151, 215, 486, 384, 0, -3, 212, 215, 216, 486, 215, 327, 0, -3, 216, 213, 212, 327, 231, 486, 0, -3, 213, 216, 217, 231, 327, 512, 0, -3, 217, 214, 213, 512, 0, 231, 0, -3, 214, 217, 211, 0, 512, 393, 0, -3, 211, 211, 214, 393, 393, 0, 0, -3, 155, 159, 218, 83, 297, 149, 0, -3, 218, 215, 155, 149, 215, 83, 0, -3, 215, 218, 219, 215, 149, 91, 0, -3, 219, 216, 215, 91, 327, 215, 0, -3, 216, 219, 220, 327, 91, 177, 0, -3, 220, 217, 216, 177, 512, 327, 0, -3, 217, 220, 211, 512, 177, 393, 0, -3, 211, 211, 217, 393, 393, 512, 0, -3, 159, 163, 221, 297, 317, 504, 0, -3, 221, 218, 159, 504, 149, 297, 0, -3, 218, 221, 222, 149, 504, 285, 0, -3, 222, 219, 218, 285, 91, 149, 0, -3, 219, 222, 223, 91, 285, 254, 0, -3, 223, 220, 219, 254, 177, 91, 0, -3, 220, 223, 211, 177, 254, 393, 0, -3, 211, 211, 220, 393, 393, 177, 0, -3, 163, 167, 224, 317, 459, 125, 0, -3, 224, 221, 163, 125, 504, 317, 0, -3, 221, 224, 225, 504, 125, 162, 0, -3, 225, 222, 221, 162, 285, 504, 0, -3, 222, 225, 226, 285, 162, 278, 0, -3, 226, 223, 222, 278, 254, 285, 0, -3, 223, 226, 211, 254, 278, 393, 0, -3, 211, 211, 223, 393, 393, 254, 0, -3, 167, 171, 227, 459, 426, 439, 0, -3, 227, 224, 167, 439, 125, 459, 0, -3, 224, 227, 228, 125, 439, 60, 0, -3, 228, 225, 224, 60, 162, 125, 0, -3, 225, 228, 229, 162, 60, 446, 0, -3, 229, 226, 225, 446, 278, 162, 0, -3, 226, 229, 211, 278, 446, 393, 0, -3, 211, 211, 226, 393, 393, 278, 0, -3, 171, 175, 230, 426, 87, 482, 0, -3, 230, 227, 171, 482, 439, 426, 0, -3, 227, 230, 231, 439, 482, 92, 0, -3, 231, 228, 227, 92, 60, 439, 0, -3, 228, 231, 232, 60, 92, 110, 0, -3, 232, 229, 228, 110, 446, 60, 0, -3, 229, 232, 211, 446, 110, 393, 0, -3, 211, 211, 229, 393, 393, 446, 0, -3, 175, 179, 233, 87, 501, 261, 0, -3, 233, 230, 175, 261, 482, 87, 0, -3, 230, 233, 234, 482, 261, 329, 0, -3, 234, 231, 230, 329, 92, 482, 0, -3, 231, 234, 235, 92, 329, 192, 0, -3, 235, 232, 231, 192, 110, 92, 0, -3, 232, 235, 211, 110, 192, 393, 0, -3, 211, 211, 232, 393, 393, 110, 0, -3, 179, 183, 236, 501, 105, 219, 0, -3, 236, 233, 179, 219, 261, 501, 0, -3, 233, 236, 237, 261, 219, 491, 0, -3, 237, 234, 233, 491, 329, 261, 0, -3, 234, 237, 238, 329, 491, 267, 0, -3, 238, 235, 234, 267, 192, 329, 0, -3, 235, 238, 211, 192, 267, 393, 0, -3, 211, 211, 235, 393, 393, 192, 0, -3, 183, 187, 239, 105, 356, 472, 0, -3, 239, 236, 183, 472, 219, 105, 0, -3, 236, 239, 240, 219, 472, 48, 0, -3, 240, 237, 236, 48, 491, 219, 0, -3, 237, 240, 241, 491, 48, 247, 0, -3, 241, 238, 237, 247, 267, 491, 0, -3, 238, 241, 211, 267, 247, 393, 0, -3, 211, 211, 238, 393, 393, 267, 0, -3, 187, 191, 242, 356, 17, 411, 0, -3, 242, 239, 187, 411, 472, 356, 0, -3, 239, 242, 243, 472, 411, 364, 0, -3, 243, 240, 239, 364, 48, 472, 0, -3, 240, 243, 244, 48, 364, 441, 0, -3, 244, 241, 240, 441, 247, 48, 0, -3, 241, 244, 211, 247, 441, 393, 0, -3, 211, 211, 241, 393, 393, 247, 0, -3, 191, 195, 245, 17, 335, 239, 0, -3, 245, 242, 191, 239, 411, 17, 0, -3, 242, 245, 246, 411, 239, 13, 0, -3, 246, 243, 242, 13, 364, 411, 0, -3, 243, 246, 247, 364, 13, 509, 0, -3, 247, 244, 243, 509, 441, 364, 0, -3, 244, 247, 211, 441, 509, 393, 0, -3, 211, 211, 244, 393, 393, 441, 0, -3, 195, 199, 248, 335, 475, 144, 0, -3, 248, 245, 195, 144, 239, 335, 0, -3, 245, 248, 249, 239, 144, 179, 0, -3, 249, 246, 245, 179, 13, 239, 0, -3, 246, 249, 250, 13, 179, 298, 0, -3, 250, 247, 246, 298, 509, 13, 0, -3, 247, 250, 211, 509, 298, 393, 0, -3, 211, 211, 247, 393, 393, 509, 0, -3, 199, 203, 251, 475, 444, 462, 0, -3, 251, 248, 199, 462, 144, 475, 0, -3, 248, 251, 252, 144, 462, 76, 0, -3, 252, 249, 248, 76, 179, 144, 0, -3, 249, 252, 253, 179, 76, 464, 0, -3, 253, 250, 249, 464, 298, 179, 0, -3, 250, 253, 211, 298, 464, 393, 0, -3, 211, 211, 250, 393, 393, 298, 0, -3, 203, 207, 254, 444, 101, 500, 0, -3, 254, 251, 203, 500, 462, 444, 0, -3, 251, 254, 255, 462, 500, 113, 0, -3, 255, 252, 251, 113, 76, 462, 0, -3, 252, 255, 256, 76, 113, 128, 0, -3, 256, 253, 252, 128, 464, 76, 0, -3, 253, 256, 211, 464, 128, 393, 0, -3, 211, 211, 253, 393, 393, 464, 0, -3, 207, 147, 208, 101, 246, 279, 0, -3, 208, 254, 207, 279, 500, 101, 0, -3, 254, 208, 209, 500, 279, 349, 0, -3, 209, 255, 254, 349, 113, 500, 0, -3, 255, 209, 210, 113, 349, 216, 0, -3, 210, 256, 255, 216, 128, 113, 0, -3, 256, 210, 211, 128, 216, 393, 0, -3, 211, 211, 256, 393, 393, 128, 0, -3, 257, 262, 263, 425, 244, 58, 0, -3, 263, 258, 257, 58, 337, 425, 0, -3, 258, 263, 264, 337, 58, 214, 0, -3, 264, 259, 258, 214, 236, 337, 0, -3, 259, 264, 265, 236, 214, 266, 0, -3, 265, 260, 259, 266, 32, 236, 0, -3, 260, 265, 266, 32, 266, 331, 0, -3, 266, 261, 260, 331, 109, 32, 0, -3, 262, 267, 268, 244, 233, 369, 0, -3, 268, 263, 262, 369, 58, 244, 0, -3, 263, 268, 269, 58, 369, 71, 0, -3, 269, 264, 263, 71, 214, 58, 0, -3, 264, 269, 270, 214, 71, 392, 0, -3, 270, 265, 264, 392, 266, 214, 0, -3, 265, 270, 271, 266, 392, 312, 0, -3, 271, 266, 265, 312, 331, 266, 0, -3, 267, 272, 273, 233, 12, 434, 0, -3, 273, 268, 267, 434, 369, 233, 0, -3, 268, 273, 274, 369, 434, 188, 0, -3, 274, 269, 268, 188, 71, 369, 0, -3, 269, 274, 275, 71, 188, 201, 0, -3, 275, 270, 269, 201, 392, 71, 0, -3, 270, 275, 276, 392, 201, 238, 0, -3, 276, 271, 270, 238, 312, 392, 0, -3, 272, 277, 278, 12, 142, 114, 0, -3, 278, 273, 272, 114, 434, 12, 0, -3, 273, 278, 279, 434, 114, 173, 0, -3, 279, 274, 273, 173, 188, 434, 0, -3, 274, 279, 280, 188, 173, 14, 0, -3, 280, 275, 274, 14, 201, 188, 0, -3, 275, 280, 281, 201, 14, 15, 0, -3, 281, 276, 275, 15, 238, 201, 0, -3, 277, 282, 283, 142, 407, 288, 0, -3, 283, 278, 277, 288, 114, 142, 0, -3, 278, 283, 284, 114, 288, 400, 0, -3, 284, 279, 278, 400, 173, 114, 0, -3, 279, 284, 285, 173, 400, 457, 0, -3, 285, 280, 279, 457, 14, 173, 0, -3, 280, 285, 286, 14, 457, 332, 0, -3, 286, 281, 280, 332, 15, 14, 0, -3, 282, 287, 288, 407, 194, 42, 0, -3, 288, 283, 282, 42, 288, 407, 0, -3, 283, 288, 289, 288, 42, 380, 0, -3, 289, 284, 283, 380, 400, 288, 0, -3, 284, 289, 290, 400, 380, 383, 0, -3, 290, 285, 284, 383, 457, 400, 0, -3, 285, 290, 291, 457, 383, 197, 0, -3, 291, 286, 285, 197, 332, 457, 0, -3, 287, 292, 293, 194, 321, 152, 0, -3, 293, 288, 287, 152, 42, 194, 0, -3, 288, 293, 294, 42, 152, 397, 0, -3, 294, 289, 288, 397, 380, 42, 0, -3, 289, 294, 295, 380, 397, 342, 0, -3, 295, 290, 289, 342, 383, 380, 0, -3, 290, 295, 296, 383, 342, 225, 0, -3, 296, 291, 290, 225, 197, 383, 0, -3, 292, 257, 258, 321, 425, 337, 0, -3, 258, 293, 292, 337, 152, 321, 0, -3, 293, 258, 259, 152, 337, 236, 0, -3, 259, 294, 293, 236, 397, 152, 0, -3, 294, 259, 260, 397, 236, 32, 0, -3, 260, 295, 294, 32, 342, 397, 0, -3, 295, 260, 261, 342, 32, 109, 0, -3, 261, 296, 295, 109, 225, 342, 0, -3, 261, 266, 301, 109, 331, 175, 0, -3, 301, 297, 261, 175, 502, 109, 0, -3, 297, 301, 302, 502, 175, 265, 0, -3, 302, 298, 297, 265, 84, 502, 0, -3, 298, 302, 303, 84, 265, 186, 0, -3, 303, 299, 298, 186, 496, 84, 0, -3, 299, 303, 304, 496, 186, 470, 0, -3, 304, 300, 299, 470, 494, 496, 0, -3, 266, 271, 305, 331, 312, 170, 0, -3, 305, 301, 266, 170, 175, 331, 0, -3, 301, 305, 306, 175, 170, 97, 0, -3, 306, 302, 301, 97, 265, 175, 0, -3, 302, 306, 307, 265, 97, 205, 0, -3, 307, 303, 302, 205, 186, 265, 0, -3, 303, 307, 308, 186, 205, 449, 0, -3, 308, 304, 303, 449, 470, 186, 0, -3, 271, 276, 309, 312, 238, 379, 0, -3, 309, 305, 271, 379, 170, 312, 0, -3, 305, 309, 310, 170, 379, 300, 0, -3, 310, 306, 305, 300, 97, 170, 0, -3, 306, 310, 311, 97, 300, 118, 0, -3, 311, 307, 306, 118, 205, 97, 0, -3, 307, 311, 312, 205, 118, 237, 0, -3, 312, 308, 307, 237, 449, 205, 0, -3, 276, 281, 313, 238, 15, 199, 0, -3, 313, 309, 276, 199, 379, 238, 0, -3, 309, 313, 314, 379, 199, 94, 0, -3, 314, 310, 309, 94, 300, 379, 0, -3, 310, 314, 315, 300, 94, 421, 0, -3, 315, 311, 310, 421, 118, 300, 0, -3, 311, 315, 316, 118, 421, 31, 0, -3, 316, 312, 311, 31, 237, 118, 0, -3, 281, 286, 317, 15, 332, 367, 0, -3, 317, 313, 281, 367, 199, 15, 0, -3, 313, 317, 318, 199, 367, 529, 0, -3, 318, 314, 313, 529, 94, 199, 0, -3, 314, 318, 319, 94, 529, 185, 0, -3, 319, 315, 314, 185, 421, 94, 0, -3, 315, 319, 320, 421, 185, 89, 0, -3, 320, 316, 315, 89, 31, 421, 0, -3, 286, 291, 321, 332, 197, 172, 0, -3, 321, 317, 286, 172, 367, 332, 0, -3, 317, 321, 322, 367, 172, 209, 0, -3, 322, 318, 317, 209, 529, 367, 0, -3, 318, 322, 323, 529, 209, 429, 0, -3, 323, 319, 318, 429, 185, 529, 0, -3, 319, 323, 324, 185, 429, 112, 0, -3, 324, 320, 319, 112, 89, 185, 0, -3, 291, 296, 325, 197, 225, 451, 0, -3, 325, 321, 291, 451, 172, 197, 0, -3, 321, 325, 326, 172, 451, 66, 0, -3, 326, 322, 321, 66, 209, 172, 0, -3, 322, 326, 327, 209, 66, 176, 0, -3, 327, 323, 322, 176, 429, 209, 0, -3, 323, 327, 328, 429, 176, 155, 0, -3, 328, 324, 323, 155, 112, 429, 0, -3, 296, 261, 297, 225, 109, 502, 0, -3, 297, 325, 296, 502, 451, 225, 0, -3, 325, 297, 298, 451, 502, 84, 0, -3, 298, 326, 325, 84, 66, 451, 0, -3, 326, 298, 299, 66, 84, 496, 0, -3, 299, 327, 326, 496, 176, 66, 0, -3, 327, 299, 300, 176, 496, 494, 0, -3, 300, 328, 327, 494, 155, 176, 0, -3, 329, 334, 335, 3, 355, 122, 0, -3, 335, 330, 329, 122, 518, 3, 0, -3, 330, 335, 336, 518, 122, 111, 0, -3, 336, 331, 330, 111, 213, 518, 0, -3, 331, 336, 337, 213, 111, 473, 0, -3, 337, 332, 331, 473, 468, 213, 0, -3, 332, 337, 338, 468, 473, 521, 0, -3, 338, 333, 332, 521, 346, 468, 0, -3, 334, 339, 340, 355, 61, 414, 0, -3, 340, 335, 334, 414, 122, 355, 0, -3, 335, 340, 341, 122, 414, 413, 0, -3, 341, 336, 335, 413, 111, 122, 0, -3, 336, 341, 342, 111, 413, 204, 0, -3, 342, 337, 336, 204, 473, 111, 0, -3, 337, 342, 343, 473, 204, 217, 0, -3, 343, 338, 337, 217, 521, 473, 0, -3, 339, 344, 345, 61, 55, 100, 0, -3, 345, 340, 339, 100, 414, 61, 0, -3, 340, 345, 346, 414, 100, 399, 0, -3, 346, 341, 340, 399, 413, 414, 0, -3, 341, 346, 347, 413, 399, 326, 0, -3, 347, 342, 341, 326, 204, 413, 0, -3, 342, 347, 348, 204, 326, 221, 0, -3, 348, 343, 342, 221, 217, 204, 0, -3, 344, 349, 350, 55, 508, 477, 0, -3, 350, 345, 344, 477, 100, 55, 0, -3, 345, 350, 351, 100, 477, 292, 0, -3, 351, 346, 345, 292, 399, 100, 0, -3, 346, 351, 352, 399, 292, 73, 0, -3, 352, 347, 346, 73, 326, 399, 0, -3, 347, 352, 353, 326, 73, 362, 0, -3, 353, 348, 347, 362, 221, 326, 0, -3, 349, 354, 355, 508, 365, 262, 0, -3, 355, 350, 349, 262, 477, 508, 0, -3, 350, 355, 356, 477, 262, 93, 0, -3, 356, 351, 350, 93, 292, 477, 0, -3, 351, 356, 357, 292, 93, 318, 0, -3, 357, 352, 351, 318, 73, 292, 0, -3, 352, 357, 358, 73, 318, 163, 0, -3, 358, 353, 352, 163, 362, 73, 0, -3, 354, 359, 360, 365, 140, 340, 0, -3, 360, 355, 354, 340, 262, 365, 0, -3, 355, 360, 361, 262, 340, 505, 0, -3, 361, 356, 355, 505, 93, 262, 0, -3, 356, 361, 362, 93, 505, 499, 0, -3, 362, 357, 356, 499, 318, 93, 0, -3, 357, 362, 363, 318, 499, 159, 0, -3, 363, 358, 357, 159, 163, 318, 0, -3, 359, 364, 365, 140, 510, 68, 0, -3, 365, 360, 359, 68, 340, 140, 0, -3, 360, 365, 366, 340, 68, 167, 0, -3, 366, 361, 360, 167, 505, 340, 0, -3, 361, 366, 367, 505, 167, 245, 0, -3, 367, 362, 361, 245, 499, 505, 0, -3, 362, 367, 368, 499, 245, 437, 0, -3, 368, 363, 362, 437, 159, 499, 0, -3, 364, 329, 330, 510, 3, 518, 0, -3, 330, 365, 364, 518, 68, 510, 0, -3, 365, 330, 331, 68, 518, 213, 0, -3, 331, 366, 365, 213, 167, 68, 0, -3, 366, 331, 332, 167, 213, 468, 0, -3, 332, 367, 366, 468, 245, 167, 0, -3, 367, 332, 333, 245, 468, 346, 0, -3, 333, 368, 367, 346, 437, 245, 0, -3, 333, 338, 373, 346, 521, 79, 0, -3, 373, 369, 333, 79, 286, 346, 0, -3, 369, 373, 374, 286, 79, 77, 0, -3, 374, 370, 369, 77, 22, 286, 0, -3, 370, 374, 375, 22, 77, 523, 0, -3, 375, 371, 370, 523, 330, 22, 0, -3, 371, 375, 376, 330, 523, 259, 0, -3, 376, 372, 371, 259, 338, 330, 0, -3, 338, 343, 377, 521, 217, 207, 0, -3, 377, 373, 338, 207, 79, 521, 0, -3, 373, 377, 378, 79, 207, 471, 0, -3, 378, 374, 373, 471, 77, 79, 0, -3, 374, 378, 379, 77, 471, 198, 0, -3, 379, 375, 374, 198, 523, 77, 0, -3, 375, 379, 380, 523, 198, 366, 0, -3, 380, 376, 375, 366, 259, 523, 0, -3, 343, 348, 381, 217, 221, 516, 0, -3, 381, 377, 343, 516, 207, 217, 0, -3, 377, 381, 382, 207, 516, 250, 0, -3, 382, 378, 377, 250, 471, 207, 0, -3, 378, 382, 383, 471, 250, 240, 0, -3, 383, 379, 378, 240, 198, 471, 0, -3, 379, 383, 384, 198, 240, 381, 0, -3, 384, 380, 379, 381, 366, 198, 0, -3, 348, 353, 385, 221, 362, 230, 0, -3, 385, 381, 348, 230, 516, 221, 0, -3, 381, 385, 386, 516, 230, 303, 0, -3, 386, 382, 381, 303, 250, 516, 0, -3, 382, 386, 387, 250, 303, 10, 0, -3, 387, 383, 382, 10, 240, 250, 0, -3, 383, 387, 388, 240, 10, 283, 0, -3, 388, 384, 383, 283, 381, 240, 0, -3, 353, 358, 389, 362, 163, 282, 0, -3, 389, 385, 353, 282, 230, 362, 0, -3, 385, 389, 390, 230, 282, 35, 0, -3, 390, 386, 385, 35, 303, 230, 0, -3, 386, 390, 391, 303, 35, 243, 0, -3, 391, 387, 386, 243, 10, 303, 0, -3, 387, 391, 392, 10, 243, 368, 0, -3, 392, 388, 387, 368, 283, 10, 0, -3, 358, 363, 393, 163, 159, 296, 0, -3, 393, 389, 358, 296, 282, 163, 0, -3, 389, 393, 394, 282, 296, 160, 0, -3, 394, 390, 389, 160, 35, 282, 0, -3, 390, 394, 395, 35, 160, 323, 0, -3, 395, 391, 390, 323, 243, 35, 0, -3, 391, 395, 396, 243, 323, 280, 0, -3, 396, 392, 391, 280, 368, 243, 0, -3, 363, 368, 397, 159, 437, 275, 0, -3, 397, 393, 363, 275, 296, 159, 0, -3, 393, 397, 398, 296, 275, 133, 0, -3, 398, 394, 393, 133, 160, 296, 0, -3, 394, 398, 399, 160, 133, 344, 0, -3, 399, 395, 394, 344, 323, 160, 0, -3, 395, 399, 400, 323, 344, 108, 0, -3, 400, 396, 395, 108, 280, 323, 0, -3, 368, 333, 369, 437, 346, 286, 0, -3, 369, 397, 368, 286, 275, 437, 0, -3, 397, 369, 370, 275, 286, 22, 0, -3, 370, 398, 397, 22, 133, 275, 0, -3, 398, 370, 371, 133, 22, 330, 0, -3, 371, 399, 398, 330, 344, 133, 0, -3, 399, 371, 372, 344, 330, 338, 0, -3, 372, 400, 399, 338, 108, 344, 0, -3, 401, 401, 406, 235, 235, 189, 0, -3, 406, 402, 401, 189, 40, 235, 0, -3, 402, 406, 407, 40, 189, 306, 0, -3, 407, 403, 402, 306, 119, 40, 0, -3, 403, 407, 408, 119, 306, 202, 0, -3, 408, 404, 403, 202, 443, 119, 0, -3, 404, 408, 409, 443, 202, 241, 0, -3, 409, 405, 404, 241, 75, 443, 0, -3, 401, 401, 410, 235, 235, 263, 0, -3, 410, 406, 401, 263, 189, 235, 0, -3, 406, 410, 411, 189, 263, 196, 0, -3, 411, 407, 406, 196, 306, 189, 0, -3, 407, 411, 412, 306, 196, 281, 0, -3, 412, 408, 407, 281, 202, 306, 0, -3, 408, 412, 413, 202, 281, 121, 0, -3, 413, 409, 408, 121, 241, 202, 0, -3, 401, 401, 414, 235, 235, 479, 0, -3, 414, 410, 401, 479, 263, 235, 0, -3, 410, 414, 415, 263, 479, 36, 0, -3, 415, 411, 410, 36, 196, 263, 0, -3, 411, 415, 416, 196, 36, 436, 0, -3, 416, 412, 411, 436, 281, 196, 0, -3, 412, 416, 417, 281, 436, 351, 0, -3, 417, 413, 412, 351, 121, 281, 0, -3, 401, 401, 418, 235, 235, 90, 0, -3, 418, 414, 401, 90, 479, 235, 0, -3, 414, 418, 419, 479, 90, 361, 0, -3, 419, 415, 414, 361, 36, 479, 0, -3, 415, 419, 420, 36, 361, 376, 0, -3, 420, 416, 415, 376, 436, 36, 0, -3, 416, 420, 421, 436, 376, 412, 0, -3, 421, 417, 416, 412, 351, 436, 0, -3, 401, 401, 422, 235, 235, 52, 0, -3, 422, 418, 401, 52, 90, 235, 0, -3, 418, 422, 423, 90, 52, 21, 0, -3, 423, 419, 418, 21, 361, 90, 0, -3, 419, 423, 424, 361, 21, 158, 0, -3, 424, 420, 419, 158, 376, 361, 0, -3, 420, 424, 425, 376, 158, 39, 0, -3, 425, 421, 420, 39, 412, 376, 0, -3, 401, 401, 426, 235, 235, 424, 0, -3, 426, 422, 401, 424, 52, 235, 0, -3, 422, 426, 427, 52, 424, 373, 0, -3, 427, 423, 422, 373, 21, 52, 0, -3, 423, 427, 428, 21, 373, 375, 0, -3, 428, 424, 423, 375, 158, 21, 0, -3, 424, 428, 429, 158, 375, 249, 0, -3, 429, 425, 424, 249, 39, 158, 0, -3, 401, 401, 430, 235, 235, 432, 0, -3, 430, 426, 401, 432, 424, 235, 0, -3, 426, 430, 431, 424, 432, 229, 0, -3, 431, 427, 426, 229, 373, 424, 0, -3, 427, 431, 432, 373, 229, 65, 0, -3, 432, 428, 427, 65, 375, 373, 0, -3, 428, 432, 433, 375, 65, 506, 0, -3, 433, 429, 428, 506, 249, 375, 0, -3, 401, 401, 434, 235, 235, 302, 0, -3, 434, 430, 401, 302, 432, 235, 0, -3, 430, 434, 435, 432, 302, 96, 0, -3, 435, 431, 430, 96, 229, 432, 0, -3, 431, 435, 436, 229, 96, 169, 0, -3, 436, 432, 431, 169, 65, 229, 0, -3, 432, 436, 437, 65, 169, 59, 0, -3, 437, 433, 432, 59, 506, 65, 0, -3, 401, 401, 438, 235, 235, 452, 0, -3, 438, 434, 401, 452, 302, 235, 0, -3, 434, 438, 439, 302, 452, 30, 0, -3, 439, 435, 434, 30, 96, 302, 0, -3, 435, 439, 440, 96, 30, 460, 0, -3, 440, 436, 435, 460, 169, 96, 0, -3, 436, 440, 441, 169, 460, 498, 0, -3, 441, 437, 436, 498, 59, 169, 0, -3, 401, 401, 442, 235, 235, 525, 0, -3, 442, 438, 401, 525, 452, 235, 0, -3, 438, 442, 443, 452, 525, 456, 0, -3, 443, 439, 438, 456, 30, 452, 0, -3, 439, 443, 444, 30, 456, 9, 0, -3, 444, 440, 439, 9, 460, 30, 0, -3, 440, 444, 445, 460, 9, 388, 0, -3, 445, 441, 440, 388, 498, 460, 0, -3, 401, 401, 446, 235, 235, 212, 0, -3, 446, 442, 401, 212, 525, 235, 0, -3, 442, 446, 447, 525, 212, 299, 0, -3, 447, 443, 442, 299, 456, 525, 0, -3, 443, 447, 448, 456, 299, 166, 0, -3, 448, 444, 443, 166, 9, 456, 0, -3, 444, 448, 449, 9, 166, 72, 0, -3, 449, 445, 444, 72, 388, 9, 0, -3, 401, 401, 450, 235, 235, 107, 0, -3, 450, 446, 401, 107, 212, 235, 0, -3, 446, 450, 451, 212, 107, 82, 0, -3, 451, 447, 446, 82, 299, 212, 0, -3, 447, 451, 452, 299, 82, 391, 0, -3, 452, 448, 447, 391, 166, 299, 0, -3, 448, 452, 453, 166, 391, 139, 0, -3, 453, 449, 448, 139, 72, 166, 0, -3, 401, 401, 454, 235, 235, 70, 0, -3, 454, 450, 401, 70, 107, 235, 0, -3, 450, 454, 455, 107, 70, 51, 0, -3, 455, 451, 450, 51, 82, 107, 0, -3, 451, 455, 456, 82, 51, 178, 0, -3, 456, 452, 451, 178, 391, 82, 0, -3, 452, 456, 457, 391, 178, 57, 0, -3, 457, 453, 452, 57, 139, 391, 0, -3, 401, 401, 458, 235, 235, 442, 0, -3, 458, 454, 401, 442, 70, 235, 0, -3, 454, 458, 459, 70, 442, 387, 0, -3, 459, 455, 454, 387, 51, 70, 0, -3, 455, 459, 460, 51, 387, 389, 0, -3, 460, 456, 455, 389, 178, 51, 0, -3, 456, 460, 461, 178, 389, 264, 0, -3, 461, 457, 456, 264, 57, 178, 0, -3, 401, 401, 462, 235, 235, 450, 0, -3, 462, 458, 401, 450, 442, 235, 0, -3, 458, 462, 463, 442, 450, 253, 0, -3, 463, 459, 458, 253, 387, 442, 0, -3, 459, 463, 464, 387, 253, 86, 0, -3, 464, 460, 459, 86, 389, 387, 0, -3, 460, 464, 465, 389, 86, 526, 0, -3, 465, 461, 460, 526, 264, 389, 0, -3, 401, 401, 402, 235, 235, 40, 0, -3, 402, 462, 401, 40, 450, 235, 0, -3, 462, 402, 403, 450, 40, 119, 0, -3, 403, 463, 462, 119, 253, 450, 0, -3, 463, 403, 404, 253, 119, 443, 0, -3, 404, 464, 463, 443, 86, 253, 0, -3, 464, 404, 405, 86, 443, 75, 0, -3, 405, 465, 464, 75, 526, 86, 0, -3, 405, 409, 470, 75, 241, 519, 0, -3, 470, 466, 405, 519, 226, 75, 0, -3, 466, 470, 471, 226, 519, 406, 0, -3, 471, 467, 466, 406, 98, 226, 0, -3, 467, 471, 472, 98, 406, 232, 0, -3, 472, 468, 467, 232, 43, 98, 0, -3, 468, 472, 473, 43, 232, 345, 0, -3, 473, 469, 468, 345, 372, 43, 0, -3, 409, 413, 474, 241, 121, 227, 0, -3, 474, 470, 409, 227, 519, 241, 0, -3, 470, 474, 475, 519, 227, 469, 0, -3, 475, 471, 470, 469, 406, 519, 0, -3, 471, 475, 476, 406, 469, 258, 0, -3, 476, 472, 471, 258, 232, 406, 0, -3, 472, 476, 477, 232, 258, 271, 0, -3, 477, 473, 472, 271, 345, 232, 0, -3, 413, 417, 478, 121, 351, 157, 0, -3, 478, 474, 413, 157, 227, 121, 0, -3, 474, 478, 479, 227, 157, 80, 0, -3, 479, 475, 474, 80, 469, 227, 0, -3, 475, 479, 480, 469, 80, 489, 0, -3, 480, 476, 475, 489, 258, 469, 0, -3, 476, 480, 481, 258, 489, 277, 0, -3, 481, 477, 476, 277, 271, 258, 0, -3, 417, 421, 482, 351, 412, 153, 0, -3, 482, 478, 417, 153, 157, 351, 0, -3, 478, 482, 483, 157, 153, 324, 0, -3, 483, 479, 478, 324, 80, 157, 0, -3, 479, 483, 484, 80, 324, 339, 0, -3, 484, 480, 479, 339, 489, 80, 0, -3, 480, 484, 485, 489, 339, 88, 0, -3, 485, 481, 480, 88, 277, 489, 0, -3, 421, 425, 486, 412, 39, 6, 0, -3, 486, 482, 421, 6, 153, 412, 0, -3, 482, 486, 487, 153, 6, 8, 0, -3, 487, 483, 482, 8, 324, 153, 0, -3, 483, 487, 488, 324, 8, 16, 0, -3, 488, 484, 483, 16, 339, 324, 0, -3, 484, 488, 489, 339, 16, 289, 0, -3, 489, 485, 484, 289, 88, 339, 0, -3, 425, 429, 490, 39, 249, 99, 0, -3, 490, 486, 425, 99, 6, 39, 0, -3, 486, 490, 491, 6, 99, 200, 0, -3, 491, 487, 486, 200, 8, 6, 0, -3, 487, 491, 492, 8, 200, 150, 0, -3, 492, 488, 487, 150, 16, 8, 0, -3, 488, 492, 493, 16, 150, 493, 0, -3, 493, 489, 488, 493, 289, 16, 0, -3, 429, 433, 494, 249, 506, 291, 0, -3, 494, 490, 429, 291, 99, 249, 0, -3, 490, 494, 495, 99, 291, 64, 0, -3, 495, 491, 490, 64, 200, 99, 0, -3, 491, 495, 496, 200, 64, 19, 0, -3, 496, 492, 491, 19, 150, 200, 0, -3, 492, 496, 497, 150, 19, 433, 0, -3, 497, 493, 492, 433, 493, 150, 0, -3, 433, 437, 498, 506, 59, 203, 0, -3, 498, 494, 433, 203, 291, 506, 0, -3, 494, 498, 499, 291, 203, 374, 0, -3, 499, 495, 494, 374, 64, 291, 0, -3, 495, 499, 500, 64, 374, 307, 0, -3, 500, 496, 495, 307, 19, 64, 0, -3, 496, 500, 501, 19, 307, 358, 0, -3, 501, 497, 496, 358, 433, 19, 0, -3, 437, 441, 502, 59, 498, 256, 0, -3, 502, 498, 437, 256, 203, 59, 0, -3, 498, 502, 503, 203, 256, 132, 0, -3, 503, 499, 498, 132, 374, 203, 0, -3, 499, 503, 504, 374, 132, 492, 0, -3, 504, 500, 499, 492, 307, 374, 0, -3, 500, 504, 505, 307, 492, 67, 0, -3, 505, 501, 500, 67, 358, 307, 0, -3, 441, 445, 506, 498, 388, 487, 0, -3, 506, 502, 441, 487, 256, 498, 0, -3, 502, 506, 507, 256, 487, 206, 0, -3, 507, 503, 502, 206, 132, 256, 0, -3, 503, 507, 508, 132, 206, 515, 0, -3, 508, 504, 503, 515, 492, 132, 0, -3, 504, 508, 509, 492, 515, 527, 0, -3, 509, 505, 504, 527, 67, 492, 0, -3, 445, 449, 510, 388, 72, 423, 0, -3, 510, 506, 445, 423, 487, 388, 0, -3, 506, 510, 511, 487, 423, 352, 0, -3, 511, 507, 506, 352, 206, 487, 0, -3, 507, 511, 512, 206, 352, 224, 0, -3, 512, 508, 507, 224, 515, 206, 0, -3, 508, 512, 513, 515, 224, 2, 0, -3, 513, 509, 508, 2, 527, 515, 0, -3, 449, 453, 514, 72, 139, 418, 0, -3, 514, 510, 449, 418, 423, 72, 0, -3, 510, 514, 515, 423, 418, 341, 0, -3, 515, 511, 510, 341, 352, 423, 0, -3, 511, 515, 516, 352, 341, 359, 0, -3, 516, 512, 511, 359, 224, 352, 0, -3, 512, 516, 517, 224, 359, 360, 0, -3, 517, 513, 512, 360, 2, 224, 0, -3, 453, 457, 518, 139, 57, 24, 0, -3, 518, 514, 453, 24, 418, 139, 0, -3, 514, 518, 519, 418, 24, 25, 0, -3, 519, 515, 514, 25, 341, 418, 0, -3, 515, 519, 520, 341, 25, 41, 0, -3, 520, 516, 515, 41, 359, 341, 0, -3, 516, 520, 521, 359, 41, 314, 0, -3, 521, 517, 516, 314, 360, 359, 0, -3, 457, 461, 522, 57, 264, 120, 0, -3, 522, 518, 457, 120, 24, 57, 0, -3, 518, 522, 523, 24, 120, 223, 0, -3, 523, 519, 518, 223, 25, 24, 0, -3, 519, 523, 524, 25, 223, 171, 0, -3, 524, 520, 519, 171, 41, 25, 0, -3, 520, 524, 525, 41, 171, 514, 0, -3, 525, 521, 520, 514, 314, 41, 0, -3, 461, 465, 526, 264, 526, 316, 0, -3, 526, 522, 461, 316, 120, 264, 0, -3, 522, 526, 527, 120, 316, 85, 0, -3, 527, 523, 522, 85, 223, 120, 0, -3, 523, 527, 528, 223, 85, 50, 0, -3, 528, 524, 523, 50, 171, 223, 0, -3, 524, 528, 529, 171, 50, 454, 0, -3, 529, 525, 524, 454, 514, 171, 0, -3, 465, 405, 466, 526, 75, 226, 0, -3, 466, 526, 465, 226, 316, 526, 0, -3, 526, 466, 467, 316, 226, 98, 0, -3, 467, 527, 526, 98, 85, 316, 0, -3, 527, 467, 468, 85, 98, 43, 0, -3, 468, 528, 527, 43, 50, 85, 0, -3, 528, 468, 469, 50, 43, 372, 0, -3, 469, 529, 528, 372, 454, 50, 0 -}; - - -const int strip_vertices[] = { -508, 508, 504, 509, 504, 505, 500, 501, 496, 497, 492, 493, 488, 489, 484, 485, 480, 481, 476, 477, 472, 473, -1, -476, 475, 480, 479, 484, 483, 488, 487, 492, 491, 496, 495, 500, 499, 504, 499, 503, 498, 502, 437, 441, -1, -527, 526, 467, 466, 471, 470, 475, 474, 479, 478, 483, 482, 487, 486, 491, 490, 495, 494, 499, 494, 498, -1, -490, 490, 425, 486, 421, 482, 417, 478, 413, 474, 409, 470, 405, 466, 465, 526, 465, 461, 460, 456, 455, 451, -1, -405, 465, 464, 460, 459, 455, 454, 450, -1, -455, 451, 450, 446, 450, 401, 454, 458, 459, 463, 464, 404, 405, 404, 409, 408, 413, 412, 417, 416, 421, 420, -1, -421, 420, 425, 420, 424, 419, 423, 418, 422, 418, 401, 414, 410, 415, 411, 416, 411, 412, 407, 408, 403, 404, 403, 463, -1, -418, 418, 414, 419, 415, 420, 416, -1, -407, 403, 402, 462, -1, -403, 463, 462, 458, 462, 401, 402, 406, 407, 406, 411, 406, 410, 401, -1, -494, 494, 498, 433, 437, 432, 436, 431, 435, 430, 434, 430, 401, 426, 422, 427, 423, 428, 424, 429, 425, 490, -1, -430, 430, 426, 431, 427, 432, 428, 433, 429, 494, 490, -1, -437, 437, 441, 436, 440, 435, 439, 434, 438, 401, 442, 446, 447, 451, 452, 456, 457, 461, 522, 526, 527, -1, -452, 448, 447, -1, -510, 445, 449, 444, 448, 443, 447, 443, 442, 443, 438, 443, 439, 444, 440, 445, 441, 506, 502, 507, 503, -1, -510, 506, 445, -1, -507, 506, 511, 510, 515, 510, 514, 449, 453, 448, 453, 452, 457, -1, -527, 523, 522, 518, 457, 518, 453, 518, 514, 519, 515, -1, -523, 519, 518, -1, -504, 503, 508, 507, 512, 511, 516, 515, 520, 519, 524, 523, 528, 527, 468, 467, 472, 471, 476, 475, -1, -472, 473, 468, 469, 528, 529, 524, 525, 520, 521, 516, 517, 512, 513, 508, 509, -1, -211, 211, 214, 210, 209, -1, -212, 215, 216, 219, 220, 223, 220, 211, 217, 214, 213, 209, 213, 208, 212, 147, -1, -220, 217, 216, 213, 212, -1, -251, 251, 248, 252, 249, 253, 250, 253, 211, 256, 210, 255, 209, 254, 208, 207, 147, 206, 147, 146, 147, 151, 212, 215, -1, -206, 206, 202, 207, 203, 254, 251, 255, 252, 256, 253, -1, -223, 223, 222, 219, 218, 215, 155, 151, 150, 146, 145, 146, 205, 206, 201, 202, 197, 202, 198, 203, 199, 251, 248, -1, -145, 149, 150, 154, 155, 159, 218, 221, 222, 225, 226, 229, -1, -204, 204, 145, 144, 149, 148, 149, 153, 154, 158, 159, 163, 221, 224, 225, 228, 229, 232, 229, 211, 226, 223, 222, -1, -224, 224, 167, 163, 162, 158, 157, 153, 152, 148, 87, 148, 83, 144, 143, 204, 139, 200, 135, 196, 131, 192, -1, -82, 83, 142, 143, 138, 139, 134, 135, 130, 131, 126, 127, 122, 123, 118, 123, 119, 184, 180, 185, 181, -1, -81, 82, 141, 142, 137, 138, 133, 134, 129, 130, 125, 126, 121, 122, 117, 118, 113, 114, 109, 110, -1, -80, 81, 140, 141, 136, 137, 132, 133, 128, 129, 124, 125, 120, 121, 116, 117, 112, 113, 108, 109, -1, -4, 80, 79, 140, 74, 136, 69, 132, 64, 128, 59, 124, 54, 120, 49, 116, 44, 112, 39, 108, -1, -79, 79, 73, 74, 68, 69, 63, 64, 58, 59, 53, 54, 48, 49, 48, 43, 42, 37, 36, 31, 30, 31, 25, -1, -42, 42, 48, 47, 53, 52, 58, 57, 63, 62, 68, 67, 73, 72, 78, 77, 3, 2, 8, 7, 13, -1, -36, 36, 42, 41, 47, 46, 52, 51, 57, 56, 62, 61, 67, 66, 72, 71, 77, 76, 2, 1, 7, -1, -66, 66, 60, 61, 55, 56, 50, 51, 45, 46, 40, 41, 35, 36, 30, -1, -31, 31, 25, 26, 20, 21, 15, 16, 10, 11, 5, 6, 0, 1, 75, 76, 70, 71, 65, 66, 60, -1, -1, 1, 7, 6, 12, 11, 17, 16, 22, 21, 27, 26, 32, 31, 32, 37, 38, 43, 44, 49, -1, -7, 7, 13, 12, 18, 17, 23, 22, 28, 27, 33, 32, 33, 38, -1, -44, 44, 38, 39, 33, 34, 28, 29, 23, 24, 18, 19, 13, 14, 8, 9, 3, 4, 78, 79, 73, -1, -39, 108, 34, 104, 29, 100, 24, 96, 19, 92, 14, 88, 9, 84, 4, 84, 80, 85, 81, 86, 81, 82, -1, -108, 109, 104, 105, 100, 101, 96, 97, 92, 93, 88, 89, 84, 85, -1, -109, 110, 105, 106, 101, 102, 97, 98, 93, 94, 89, 90, 85, 86, -1, -118, 119, 114, 115, 110, 111, 106, 107, 102, 103, 98, 99, 94, 95, 90, 91, 86, 87, 82, 83, -1, -111, 115, 176, -1, -107, 111, 172, 176, 177, -1, -103, 107, 168, 172, 173, 177, 178, -1, -99, 103, 164, 168, 169, 173, 174, 178, 179, -1, -95, 99, 160, 164, 165, 169, 170, 174, 175, 179, 233, -1, -91, 95, 156, 160, 161, 165, 166, 170, 171, 175, 230, 233, 234, -1, -87, 91, 152, 156, 157, 161, 162, 166, 167, 171, 227, 230, 231, 234, 235, 234, 238, 234, 237, 233, 236, 179, -1, -185, 185, 181, 186, 182, 187, 183, 239, 236, 240, 237, 241, 238, 211, 235, 232, 231, 228, 227, 224, 167, -1, -236, 179, 183, 178, 182, 177, 181, 176, 180, 115, 119, -1, -131, 192, 127, 188, 123, 188, 184, 189, 185, 190, 186, 191, 187, 242, 239, 243, 240, 244, 241, 244, 211, 247, -1, -192, 192, 188, 193, 189, 194, 190, 195, 191, 245, 242, 246, 243, 247, 244, -1, -211, 247, 250, 246, 249, 245, 248, 195, 199, 194, 198, 193, 197, 192, 197, 196, 201, 200, 205, 204, 145, -1, -393, 393, 394, 398, 399, 371, -1, -399, 395, 394, -1, -363, 363, 393, 397, 398, 370, 371, 375, -1, -379, 375, 374, 370, 369, 397, 368, 363, 362, -1, -396, 395, 400, 399, 372, 371, 376, 375, 380, 379, 384, 383, 388, 387, 392, 391, 396, 391, 395, 390, 394, -1, -374, 378, 379, 378, 383, 382, 387, 386, 391, 386, 390, 385, 389, 353, 358, 352, 357, 351, 356, 350, 355, -1, -341, 341, 347, 346, 352, 346, 351, 345, 350, -1, -335, 334, 340, 339, 345, 344, 350, 349, 355, 354, -1, -390, 390, 394, 389, 393, 358, 363, 357, 362, 356, 361, 355, 360, 354, 360, 359, 365, 364, 330, 329, 335, 334, -1, -345, 346, 340, 341, 335, 336, 330, 331, 365, 366, 360, 366, 361, 367, 362, 367, 368, 333, 369, 373, 374, 378, -1, -353, 353, 348, 385, 381, 386, 381, 382, 377, 378, 377, 373, 338, 333, 332, 367, 332, 366, 332, 331, 337, 336, 342, 341, 347, -1, -332, 337, 338, 343, 377, 343, 381, 343, 348, 342, 348, 347, 353, 352, -1, -337, 342, 343, -1, -314, 314, 319, 318, 323, 322, 323, 327, -1, -309, 309, 314, 313, 318, 317, 322, 321, 322, 326, 327, 299, -1, -271, 271, 309, 276, 313, 281, 317, 286, 321, 291, 321, 325, 326, 298, 299, 303, -1, -265, 265, 271, 270, 276, 275, 281, 280, 286, 285, 291, 290, 291, 296, 325, 297, 298, 302, 303, 307, -1, -259, 259, 265, 264, 270, 269, 275, 274, 280, 279, 285, 284, 290, 289, 290, 295, 296, 261, 297, 301, 302, 306, 307, 311, -1, -293, 293, 259, 258, 264, 263, 269, 268, 274, 273, 279, 278, 284, 283, 289, 288, 289, 294, 295, 260, 261, 266, -1, -309, 305, 271, 266, 265, 260, 259, 294, 293, 288, 287, 288, 282, 283, 277, 278, 272, 273, 267, 268, 262, -1, -268, 268, 262, 263, 257, 258, 292, 293, 287, -1, -261, 266, 301, 305, 306, 310, 311, 315, 316, 320, -1, -316, 316, 311, 312, 307, 308, 303, 304, 299, 300, 327, 328, 323, 324, 319, 320, 319, 315, 314, 310, 309, 305, -1 -}; - - -const int strip_normals[] = { -515, 515, 492, 527, 492, 67, 307, 358, 19, 433, 150, 493, 16, 289, 339, 88, 489, 277, 258, 271, 232, 345, -1, -258, 469, 489, 80, 339, 324, 16, 8, 150, 200, 19, 64, 307, 374, 492, 374, 132, 203, 256, 59, 498, -1, -85, 316, 98, 226, 406, 519, 469, 227, 80, 157, 324, 153, 8, 6, 200, 99, 64, 291, 374, 291, 203, -1, -99, 99, 39, 6, 412, 153, 351, 157, 121, 227, 241, 519, 75, 226, 526, 316, 526, 264, 389, 178, 51, 82, -1, -75, 526, 86, 389, 387, 51, 70, 107, -1, -51, 82, 107, 212, 107, 235, 70, 442, 387, 253, 86, 443, 75, 443, 241, 202, 121, 281, 351, 436, 412, 376, -1, -412, 376, 39, 376, 158, 361, 21, 90, 52, 90, 235, 479, 263, 36, 196, 436, 196, 281, 306, 202, 119, 443, 119, 253, -1, -90, 90, 479, 361, 36, 376, 436, -1, -306, 119, 40, 450, -1, -119, 253, 450, 442, 450, 235, 40, 189, 306, 189, 196, 189, 263, 235, -1, -291, 291, 203, 506, 59, 65, 169, 229, 96, 432, 302, 432, 235, 424, 52, 373, 21, 375, 158, 249, 39, 99, -1, -432, 432, 424, 229, 373, 65, 375, 506, 249, 291, 99, -1, -59, 59, 498, 169, 460, 96, 30, 302, 452, 235, 525, 212, 299, 82, 391, 178, 57, 264, 120, 316, 85, -1, -391, 166, 299, -1, -423, 388, 72, 9, 166, 456, 299, 456, 525, 456, 452, 456, 30, 9, 460, 388, 498, 487, 256, 206, 132, -1, -423, 487, 388, -1, -206, 487, 352, 423, 341, 423, 418, 72, 139, 166, 139, 391, 57, -1, -85, 223, 120, 24, 57, 24, 139, 24, 418, 25, 341, -1, -223, 25, 24, -1, -492, 132, 515, 206, 224, 352, 359, 341, 41, 25, 171, 223, 50, 85, 43, 98, 232, 406, 258, 469, -1, -232, 345, 43, 372, 50, 454, 171, 514, 41, 314, 359, 360, 224, 2, 515, 527, -1, -393, 393, 0, 216, 349, -1, -486, 215, 327, 91, 177, 254, 177, 393, 512, 0, 231, 349, 231, 279, 486, 246, -1, -177, 512, 327, 231, 486, -1, -462, 462, 144, 76, 179, 464, 298, 464, 393, 128, 216, 113, 349, 500, 279, 101, 246, 7, 246, 395, 246, 384, 486, 215, -1, -7, 7, 195, 101, 444, 500, 462, 113, 76, 128, 464, -1, -254, 254, 285, 91, 149, 215, 83, 384, 325, 395, 520, 395, 315, 7, 143, 195, 129, 195, 461, 444, 475, 462, 144, -1, -520, 308, 325, 415, 83, 297, 149, 504, 285, 162, 278, 446, -1, -403, 403, 520, 276, 308, 404, 308, 272, 415, 416, 297, 317, 504, 125, 162, 60, 446, 110, 446, 393, 278, 254, 285, -1, -125, 125, 459, 317, 334, 416, 137, 272, 47, 404, 511, 404, 1, 276, 54, 403, 45, 370, 490, 210, 29, 184, -1, -336, 1, 251, 54, 164, 45, 463, 490, 378, 29, 453, 147, 222, 218, 301, 218, 242, 310, 130, 528, 34, -1, -26, 336, 148, 251, 269, 164, 419, 463, 293, 378, 420, 453, 333, 222, 208, 301, 294, 56, 127, 228, -1, -131, 26, 371, 148, 284, 269, 311, 419, 95, 293, 350, 420, 377, 333, 474, 208, 396, 294, 357, 127, -1, -116, 131, 273, 371, 480, 284, 104, 311, 257, 95, 348, 350, 447, 377, 313, 474, 322, 396, 28, 357, -1, -273, 273, 183, 480, 347, 104, 53, 257, 181, 348, 124, 447, 428, 313, 428, 435, 11, 270, 193, 305, 69, 305, 386, -1, -11, 11, 428, 409, 124, 382, 181, 161, 53, 481, 347, 408, 183, 319, 151, 422, 448, 4, 165, 134, 394, -1, -193, 193, 11, 495, 409, 44, 382, 385, 161, 478, 481, 320, 408, 37, 319, 524, 422, 328, 4, 465, 134, -1, -37, 37, 431, 320, 136, 478, 513, 385, 27, 44, 467, 495, 115, 193, 69, -1, -305, 305, 386, 20, 174, 182, 401, 211, 248, 106, 295, 309, 255, 465, 483, 328, 191, 524, 135, 37, 431, -1, -465, 465, 134, 309, 102, 106, 427, 211, 507, 182, 410, 20, 503, 305, 503, 270, 445, 435, 322, 313, -1, -134, 134, 394, 102, 455, 427, 5, 507, 23, 410, 405, 503, 405, 445, -1, -322, 322, 445, 28, 405, 138, 23, 485, 5, 234, 455, 74, 394, 180, 165, 49, 448, 116, 151, 273, 183, -1, -28, 357, 138, 268, 485, 287, 234, 81, 74, 78, 180, 103, 49, 220, 116, 220, 131, 476, 26, 38, 26, 336, -1, -357, 127, 268, 252, 287, 398, 81, 274, 78, 154, 103, 62, 220, 476, -1, -127, 228, 252, 141, 398, 440, 274, 363, 154, 190, 62, 488, 476, 38, -1, -301, 242, 56, 517, 228, 33, 141, 18, 440, 466, 363, 304, 190, 417, 488, 484, 38, 511, 336, 1, -1, -33, 517, 260, -1, -18, 33, 390, 260, 497, -1, -466, 18, 353, 390, 290, 497, 126, -1, -304, 466, 187, 353, 123, 290, 522, 126, 501, -1, -417, 304, 458, 187, 117, 123, 168, 522, 87, 501, 261, -1, -484, 417, 430, 458, 343, 117, 438, 168, 426, 87, 482, 261, 329, -1, -511, 484, 47, 430, 137, 343, 334, 438, 459, 426, 439, 482, 92, 329, 192, 329, 267, 329, 491, 261, 219, 501, -1, -528, 528, 34, 145, 46, 356, 105, 472, 219, 48, 491, 247, 267, 393, 192, 110, 92, 60, 439, 125, 459, -1, -219, 501, 105, 126, 46, 497, 34, 260, 130, 517, 242, -1, -29, 184, 147, 156, 218, 156, 310, 402, 528, 146, 145, 17, 356, 411, 472, 364, 48, 441, 247, 441, 393, 509, -1, -184, 184, 156, 63, 402, 354, 146, 335, 17, 239, 411, 13, 364, 509, 441, -1, -393, 509, 298, 13, 179, 239, 144, 335, 475, 354, 461, 63, 129, 184, 129, 210, 143, 370, 315, 403, 520, -1, -296, 296, 160, 133, 344, 330, -1, -344, 323, 160, -1, -159, 159, 296, 275, 133, 22, 330, 523, -1, -198, 523, 77, 22, 286, 275, 437, 159, 499, -1, -280, 323, 108, 344, 338, 330, 259, 523, 366, 198, 381, 240, 283, 10, 368, 243, 280, 243, 323, 35, 160, -1, -77, 471, 198, 471, 240, 250, 10, 303, 243, 303, 35, 230, 282, 362, 163, 73, 318, 292, 93, 477, 262, -1, -413, 413, 326, 399, 73, 399, 292, 100, 477, -1, -122, 355, 414, 61, 100, 55, 477, 508, 262, 365, -1, -35, 35, 160, 282, 296, 163, 159, 318, 499, 93, 505, 262, 340, 365, 340, 140, 68, 510, 518, 3, 122, 355, -1, -100, 399, 414, 413, 122, 111, 518, 213, 68, 167, 340, 167, 505, 245, 499, 245, 437, 346, 286, 79, 77, 471, -1, -362, 362, 221, 230, 516, 303, 516, 250, 207, 471, 207, 79, 521, 346, 468, 245, 468, 167, 468, 213, 473, 111, 204, 413, 326, -1, -468, 473, 521, 217, 207, 217, 516, 217, 221, 204, 221, 326, 362, 73, -1, -473, 204, 217, -1, -94, 94, 185, 529, 429, 209, 429, 176, -1, -379, 379, 94, 199, 529, 367, 209, 172, 209, 66, 176, 496, -1, -312, 312, 379, 238, 199, 15, 367, 332, 172, 197, 172, 451, 66, 84, 496, 186, -1, -266, 266, 312, 392, 238, 201, 15, 14, 332, 457, 197, 383, 197, 225, 451, 502, 84, 265, 186, 205, -1, -236, 236, 266, 214, 392, 71, 201, 188, 14, 173, 457, 400, 383, 380, 383, 342, 225, 109, 502, 175, 265, 97, 205, 118, -1, -152, 152, 236, 337, 214, 58, 71, 369, 188, 434, 173, 114, 400, 288, 380, 42, 380, 397, 342, 32, 109, 331, -1, -379, 170, 312, 331, 266, 32, 236, 397, 152, 42, 194, 42, 407, 288, 142, 114, 12, 434, 233, 369, 244, -1, -369, 369, 244, 58, 425, 337, 321, 152, 194, -1, -109, 331, 175, 170, 97, 300, 118, 421, 31, 89, -1, -31, 31, 118, 237, 205, 449, 186, 470, 496, 494, 176, 155, 429, 112, 185, 89, 185, 421, 94, 300, 379, 170, -1 -}; - -#else /* defined(_WIN32_WCE) */ - -/* - * Original teapot code copyright follows: - */ - -/* - * (c) Copyright 1993, Silicon Graphics, Inc. - * - * ALL RIGHTS RESERVED - * - * Permission to use, copy, modify, and distribute this software - * for any purpose and without fee is hereby granted, provided - * that the above copyright notice appear in all copies and that - * both the copyright notice and this permission notice appear in - * supporting documentation, and that the name of Silicon - * Graphics, Inc. not be used in advertising or publicity - * pertaining to distribution of the software without specific, - * written prior permission. - * - * THE MATERIAL EMBODIED ON THIS SOFTWARE IS PROVIDED TO YOU - * "AS-IS" AND WITHOUT WARRANTY OF ANY KIND, EXPRESS, IMPLIED OR - * OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY WARRANTY OF - * MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. IN NO - * EVENT SHALL SILICON GRAPHICS, INC. BE LIABLE TO YOU OR ANYONE - * ELSE FOR ANY DIRECT, SPECIAL, INCIDENTAL, INDIRECT OR - * CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER, - * INCLUDING WITHOUT LIMITATION, LOSS OF PROFIT, LOSS OF USE, - * SAVINGS OR REVENUE, OR THE CLAIMS OF THIRD PARTIES, WHETHER OR - * NOT SILICON GRAPHICS, INC. HAS BEEN ADVISED OF THE POSSIBILITY - * OF SUCH LOSS, HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, - * ARISING OUT OF OR IN CONNECTION WITH THE POSSESSION, USE OR - * PERFORMANCE OF THIS SOFTWARE. - * - * US Government Users Restricted Rights - * - * Use, duplication, or disclosure by the Government is subject to - * restrictions set forth in FAR 52.227.19(c)(2) or subparagraph - * (c)(1)(ii) of the Rights in Technical Data and Computer - * Software clause at DFARS 252.227-7013 and/or in similar or - * successor clauses in the FAR or the DOD or NASA FAR - * Supplement. Unpublished-- rights reserved under the copyright - * laws of the United States. Contractor/manufacturer is Silicon - * Graphics, Inc., 2011 N. Shoreline Blvd., Mountain View, CA - * 94039-7311. - * - * OpenGL(TM) is a trademark of Silicon Graphics, Inc. - */ - -/* - * Rim, body, lid, and bottom data must be reflected in x and y; - * handle and spout data across the y axis only. - */ -static int patchdata[][16] = -{ - { 102, 103, 104, 105, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 }, /* rim */ - { 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27 }, /* body */ - { 24, 25, 26, 27, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40 }, - { 96, 96, 96, 96, 97, 98, 99, 100, 101, 101, 101, 101, 0, 1, 2, 3 }, /* lid */ - { 0, 1, 2, 3, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 117 }, - { 118, 118, 118, 118, 124, 122, 119, 121, 123, 126, 125, 120, 40, 39, 38, 37 }, /* bottom */ - { 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56 }, /* handle */ - { 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 28, 65, 66, 67 }, - { 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83 }, /* spout */ - { 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95 } -}; - -static double cpdata[][3] = -{ - {0.2, 0, 2.7}, {0.2, -0.112, 2.7}, {0.112, -0.2, 2.7}, {0, - -0.2, 2.7}, {1.3375, 0, 2.53125}, {1.3375, -0.749, 2.53125}, - {0.749, -1.3375, 2.53125}, {0, -1.3375, 2.53125}, {1.4375, - 0, 2.53125}, {1.4375, -0.805, 2.53125}, {0.805, -1.4375, - 2.53125}, {0, -1.4375, 2.53125}, {1.5, 0, 2.4}, {1.5, -0.84, - 2.4}, {0.84, -1.5, 2.4}, {0, -1.5, 2.4}, {1.75, 0, 1.875}, - {1.75, -0.98, 1.875}, {0.98, -1.75, 1.875}, {0, -1.75, - 1.875}, {2, 0, 1.35}, {2, -1.12, 1.35}, {1.12, -2, 1.35}, - {0, -2, 1.35}, {2, 0, 0.9}, {2, -1.12, 0.9}, {1.12, -2, - 0.9}, {0, -2, 0.9}, {-2, 0, 0.9}, {2, 0, 0.45}, {2, -1.12, - 0.45}, {1.12, -2, 0.45}, {0, -2, 0.45}, {1.5, 0, 0.225}, - {1.5, -0.84, 0.225}, {0.84, -1.5, 0.225}, {0, -1.5, 0.225}, - {1.5, 0, 0.15}, {1.5, -0.84, 0.15}, {0.84, -1.5, 0.15}, {0, - -1.5, 0.15}, {-1.6, 0, 2.025}, {-1.6, -0.3, 2.025}, {-1.5, - -0.3, 2.25}, {-1.5, 0, 2.25}, {-2.3, 0, 2.025}, {-2.3, -0.3, - 2.025}, {-2.5, -0.3, 2.25}, {-2.5, 0, 2.25}, {-2.7, 0, - 2.025}, {-2.7, -0.3, 2.025}, {-3, -0.3, 2.25}, {-3, 0, - 2.25}, {-2.7, 0, 1.8}, {-2.7, -0.3, 1.8}, {-3, -0.3, 1.8}, - {-3, 0, 1.8}, {-2.7, 0, 1.575}, {-2.7, -0.3, 1.575}, {-3, - -0.3, 1.35}, {-3, 0, 1.35}, {-2.5, 0, 1.125}, {-2.5, -0.3, - 1.125}, {-2.65, -0.3, 0.9375}, {-2.65, 0, 0.9375}, {-2, - -0.3, 0.9}, {-1.9, -0.3, 0.6}, {-1.9, 0, 0.6}, {1.7, 0, - 1.425}, {1.7, -0.66, 1.425}, {1.7, -0.66, 0.6}, {1.7, 0, - 0.6}, {2.6, 0, 1.425}, {2.6, -0.66, 1.425}, {3.1, -0.66, - 0.825}, {3.1, 0, 0.825}, {2.3, 0, 2.1}, {2.3, -0.25, 2.1}, - {2.4, -0.25, 2.025}, {2.4, 0, 2.025}, {2.7, 0, 2.4}, {2.7, - -0.25, 2.4}, {3.3, -0.25, 2.4}, {3.3, 0, 2.4}, {2.8, 0, - 2.475}, {2.8, -0.25, 2.475}, {3.525, -0.25, 2.49375}, - {3.525, 0, 2.49375}, {2.9, 0, 2.475}, {2.9, -0.15, 2.475}, - {3.45, -0.15, 2.5125}, {3.45, 0, 2.5125}, {2.8, 0, 2.4}, - {2.8, -0.15, 2.4}, {3.2, -0.15, 2.4}, {3.2, 0, 2.4}, {0, 0, - 3.15}, {0.8, 0, 3.15}, {0.8, -0.45, 3.15}, {0.45, -0.8, - 3.15}, {0, -0.8, 3.15}, {0, 0, 2.85}, {1.4, 0, 2.4}, {1.4, - -0.784, 2.4}, {0.784, -1.4, 2.4}, {0, -1.4, 2.4}, {0.4, 0, - 2.55}, {0.4, -0.224, 2.55}, {0.224, -0.4, 2.55}, {0, -0.4, - 2.55}, {1.3, 0, 2.55}, {1.3, -0.728, 2.55}, {0.728, -1.3, - 2.55}, {0, -1.3, 2.55}, {1.3, 0, 2.4}, {1.3, -0.728, 2.4}, - {0.728, -1.3, 2.4}, {0, -1.3, 2.4}, {0, 0, 0}, {1.425, - -0.798, 0}, {1.5, 0, 0.075}, {1.425, 0, 0}, {0.798, -1.425, - 0}, {0, -1.5, 0.075}, {0, -1.425, 0}, {1.5, -0.84, 0.075}, - {0.84, -1.5, 0.075} -}; - -static double tex[2][2][2] = -{ - { {0.0, 0.0}, {1.0, 0.0} }, - { {0.0, 1.0}, {1.0, 1.0} } -}; -#endif /* defined(_WIN32_WCE) */ - - -#endif /* FREEGLUT_TEAPOT_DATA_H */ - diff --git a/examples/common/opengl-framework/freeglut/freeglut_videoresize.c b/examples/common/opengl-framework/freeglut/freeglut_videoresize.c deleted file mode 100644 index 4d00fb9c..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_videoresize.c +++ /dev/null @@ -1,50 +0,0 @@ -/* - * freeglut_videoresize.c - * - * Video resize functions (as defined by GLUT API) - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Thu Dec 16 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include -#include "freeglut_internal.h" - -/* - * NOTE: functions declared in this file probably will not be implemented. - */ - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -int FGAPIENTRY glutVideoResizeGet( GLenum eWhat ) { return( 0x00 ); } -void FGAPIENTRY glutSetupVideoResizing( void ) { /* Not implemented */ } -void FGAPIENTRY glutStopVideoResizing( void ) { /* Not implemented */ } -void FGAPIENTRY glutVideoResize( int x, int y, int w, int h ) { /* Not implemented */ } -void FGAPIENTRY glutVideoPan( int x, int y, int w, int h ) { /* Not implemented */ } - -/*** END OF FILE ***/ - - - - - - - diff --git a/examples/common/opengl-framework/freeglut/freeglut_window.c b/examples/common/opengl-framework/freeglut/freeglut_window.c deleted file mode 100644 index 0542017b..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_window.c +++ /dev/null @@ -1,2162 +0,0 @@ -/* - * freeglut_window.c - * - * Window management methods. - * - * Copyright (c) 1999-2000 Pawel W. Olszta. All Rights Reserved. - * Written by Pawel W. Olszta, - * Creation date: Fri Dec 3 1999 - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#define FREEGLUT_BUILDING_LIB -#include -#include "freeglut_internal.h" - -#if TARGET_HOST_POSIX_X11 -#include /* LONG_MAX */ -#include /* usleep */ -#endif - -#if defined(_WIN32_WCE) -# include -# ifdef FREEGLUT_LIB_PRAGMAS -# pragma comment( lib, "Aygshell.lib" ) -# endif -#endif /* defined(_WIN32_WCE) */ - - -#if TARGET_HOST_POSIX_X11 -#ifndef GLX_FRAMEBUFFER_SRGB_CAPABLE_ARB -#define GLX_FRAMEBUFFER_SRGB_CAPABLE_ARB 0x20B2 -#endif - -#ifndef GLX_CONTEXT_MAJOR_VERSION_ARB -#define GLX_CONTEXT_MAJOR_VERSION_ARB 0x2091 -#endif - -#ifndef GLX_CONTEXT_MINOR_VERSION_ARB -#define GLX_CONTEXT_MINOR_VERSION_ARB 0x2092 -#endif - -#ifndef GLX_CONTEXT_FLAGS_ARB -#define GLX_CONTEXT_FLAGS_ARB 0x2094 -#endif - -#ifndef GLX_CONTEXT_PROFILE_MASK_ARB -#define GLX_CONTEXT_PROFILE_MASK_ARB 0x9126 -#endif - -#ifndef GLX_CONTEXT_DEBUG_BIT_ARB -#define GLX_CONTEXT_DEBUG_BIT_ARB 0x0001 -#endif - -#ifndef GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB -#define GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB 0x0002 -#endif - -#ifndef GLX_CONTEXT_CORE_PROFILE_BIT_ARB -#define GLX_CONTEXT_CORE_PROFILE_BIT_ARB 0x00000001 -#endif - -#ifndef GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB -#define GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB 0x00000002 -#endif - -#ifndef GLX_RGBA_FLOAT_TYPE -#define GLX_RGBA_FLOAT_TYPE 0x20B9 -#endif - -#ifndef GLX_RGBA_FLOAT_BIT -#define GLX_RGBA_FLOAT_BIT 0x00000004 -#endif -#endif /* TARGET_HOST_POSIX_X11 */ - - -#if TARGET_HOST_MS_WINDOWS -/* The following include file is available from SGI but is not standard: - * #include - * So we copy the necessary parts out of it. - * XXX: should local definitions for extensions be put in a separate include file? - */ -typedef const char * (WINAPI * PFNWGLGETEXTENSIONSSTRINGARBPROC) (HDC hdc); - -typedef BOOL (WINAPI * PFNWGLCHOOSEPIXELFORMATARBPROC) (HDC hdc, const int *piAttribIList, const FLOAT *pfAttribFList, UINT nMaxFormats, int *piFormats, UINT *nNumFormats); - -#define WGL_DRAW_TO_WINDOW_ARB 0x2001 -#define WGL_ACCELERATION_ARB 0x2003 -#define WGL_SUPPORT_OPENGL_ARB 0x2010 -#define WGL_DOUBLE_BUFFER_ARB 0x2011 -#define WGL_COLOR_BITS_ARB 0x2014 -#define WGL_ALPHA_BITS_ARB 0x201B -#define WGL_DEPTH_BITS_ARB 0x2022 -#define WGL_STENCIL_BITS_ARB 0x2023 -#define WGL_FULL_ACCELERATION_ARB 0x2027 - -#define WGL_SAMPLE_BUFFERS_ARB 0x2041 -#define WGL_SAMPLES_ARB 0x2042 - -#define WGL_TYPE_RGBA_FLOAT_ARB 0x21A0 - -#define WGL_FRAMEBUFFER_SRGB_CAPABLE_ARB 0x20A9 - -#ifndef WGL_ARB_create_context -#define WGL_ARB_create_context 1 -#ifdef WGL_WGLEXT_PROTOTYPES -extern HGLRC WINAPI wglCreateContextAttribsARB (HDC, HGLRC, const int *); -#endif /* WGL_WGLEXT_PROTOTYPES */ -typedef HGLRC (WINAPI * PFNWGLCREATECONTEXTATTRIBSARBPROC) (HDC hDC, HGLRC hShareContext, const int *attribList); - -#define WGL_CONTEXT_MAJOR_VERSION_ARB 0x2091 -#define WGL_CONTEXT_MINOR_VERSION_ARB 0x2092 -#define WGL_CONTEXT_LAYER_PLANE_ARB 0x2093 -#define WGL_CONTEXT_FLAGS_ARB 0x2094 -#define WGL_CONTEXT_PROFILE_MASK_ARB 0x9126 - -#define WGL_CONTEXT_DEBUG_BIT_ARB 0x0001 -#define WGL_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB 0x0002 - -#define WGL_CONTEXT_CORE_PROFILE_BIT_ARB 0x00000001 -#define WGL_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB 0x00000002 - -#define ERROR_INVALID_VERSION_ARB 0x2095 -#define ERROR_INVALID_PROFILE_ARB 0x2096 -#endif - -#endif /* TARGET_HOST_MS_WINDOWS */ - -#ifdef WM_TOUCH - typedef BOOL (WINAPI *pRegisterTouchWindow)(HWND,ULONG); - static pRegisterTouchWindow fghRegisterTouchWindow = (pRegisterTouchWindow)0xDEADBEEF; -#endif - -/* pushing attribute/value pairs into an array */ -#define ATTRIB(a) attributes[where++]=(a) -#define ATTRIB_VAL(a,v) {ATTRIB(a); ATTRIB(v);} - -/* - * TODO BEFORE THE STABLE RELEASE: - * - * fgChooseFBConfig() -- OK, but what about glutInitDisplayString()? - * fgSetupPixelFormat -- ignores the display mode settings - * fgOpenWindow() -- check the Win32 version, -iconic handling! - * fgCloseWindow() -- check the Win32 version - * glutCreateWindow() -- Check when default position and size is {-1,-1} - * glutCreateSubWindow() -- Check when default position and size is {-1,-1} - * glutDestroyWindow() -- check the Win32 version - * glutSetWindow() -- check the Win32 version - * glutGetWindow() -- OK - * glutSetWindowTitle() -- check the Win32 version - * glutSetIconTitle() -- check the Win32 version - * glutShowWindow() -- check the Win32 version - * glutHideWindow() -- check the Win32 version - * glutIconifyWindow() -- check the Win32 version - * glutReshapeWindow() -- check the Win32 version - * glutPositionWindow() -- check the Win32 version - * glutPushWindow() -- check the Win32 version - * glutPopWindow() -- check the Win32 version - */ - -/* -- PRIVATE FUNCTIONS ---------------------------------------------------- */ - -static int fghIsLegacyContextVersionRequested( void ) -{ - return fgState.MajorVersion < 2 || (fgState.MajorVersion == 2 && fgState.MinorVersion <= 1); -} - -static int fghIsLegacyContextRequested( void ) -{ - return fghIsLegacyContextVersionRequested() && - fgState.ContextFlags == 0 && - fgState.ContextProfile == 0; -} - -static int fghNumberOfAuxBuffersRequested( void ) -{ - if ( fgState.DisplayMode & GLUT_AUX4 ) { - return 4; - } - if ( fgState.DisplayMode & GLUT_AUX3 ) { - return 3; - } - if ( fgState.DisplayMode & GLUT_AUX2 ) { - return 2; - } - if ( fgState.DisplayMode & GLUT_AUX1 ) { /* NOTE: Same as GLUT_AUX! */ - return fgState.AuxiliaryBufferNumber; - } - return 0; -} - -static int fghMapBit( int mask, int from, int to ) -{ - return ( mask & from ) ? to : 0; - -} - -static void fghContextCreationError( void ) -{ - fgError( "Unable to create OpenGL %d.%d context (flags %x, profile %x)", - fgState.MajorVersion, fgState.MinorVersion, fgState.ContextFlags, - fgState.ContextProfile ); -} - - -/* -- SYSTEM-DEPENDENT PRIVATE FUNCTIONS ------------------------------------ */ - -#if TARGET_HOST_POSIX_X11 -/* - * Chooses a visual basing on the current display mode settings - */ - -GLXFBConfig* fgChooseFBConfig( int *numcfgs ) -{ - GLboolean wantIndexedMode = GL_FALSE; - int attributes[ 100 ]; - int where = 0, numAuxBuffers; - - /* First we have to process the display mode settings... */ - if( fgState.DisplayMode & GLUT_INDEX ) { - ATTRIB_VAL( GLX_BUFFER_SIZE, 8 ); - /* Buffer size is selected later. */ - - ATTRIB_VAL( GLX_RENDER_TYPE, GLX_COLOR_INDEX_BIT ); - wantIndexedMode = GL_TRUE; - } else { - ATTRIB_VAL( GLX_RED_SIZE, 1 ); - ATTRIB_VAL( GLX_GREEN_SIZE, 1 ); - ATTRIB_VAL( GLX_BLUE_SIZE, 1 ); - if( fgState.DisplayMode & GLUT_ALPHA ) { - ATTRIB_VAL( GLX_ALPHA_SIZE, 1 ); - } - } - - if( fgState.DisplayMode & GLUT_DOUBLE ) { - ATTRIB_VAL( GLX_DOUBLEBUFFER, True ); - } - - if( fgState.DisplayMode & GLUT_STEREO ) { - ATTRIB_VAL( GLX_STEREO, True ); - } - - if( fgState.DisplayMode & GLUT_DEPTH ) { - ATTRIB_VAL( GLX_DEPTH_SIZE, 1 ); - } - - if( fgState.DisplayMode & GLUT_STENCIL ) { - ATTRIB_VAL( GLX_STENCIL_SIZE, 1 ); - } - - if( fgState.DisplayMode & GLUT_ACCUM ) { - ATTRIB_VAL( GLX_ACCUM_RED_SIZE, 1 ); - ATTRIB_VAL( GLX_ACCUM_GREEN_SIZE, 1 ); - ATTRIB_VAL( GLX_ACCUM_BLUE_SIZE, 1 ); - if( fgState.DisplayMode & GLUT_ALPHA ) { - ATTRIB_VAL( GLX_ACCUM_ALPHA_SIZE, 1 ); - } - } - - numAuxBuffers = fghNumberOfAuxBuffersRequested(); - if ( numAuxBuffers > 0 ) { - ATTRIB_VAL( GLX_AUX_BUFFERS, numAuxBuffers ); - } - - if( fgState.DisplayMode & GLUT_SRGB ) { - ATTRIB_VAL( GLX_FRAMEBUFFER_SRGB_CAPABLE_ARB, True ); - } - - if (fgState.DisplayMode & GLUT_MULTISAMPLE) { - ATTRIB_VAL(GLX_SAMPLE_BUFFERS, 1); - ATTRIB_VAL(GLX_SAMPLES, fgState.SampleNumber); - } - - /* Push a terminator at the end of the list */ - ATTRIB( None ); - - { - GLXFBConfig * fbconfigArray; /* Array of FBConfigs */ - GLXFBConfig * fbconfig; /* The FBConfig we want */ - int fbconfigArraySize; /* Number of FBConfigs in the array */ - - - /* Get all FBConfigs that match "attributes". */ - fbconfigArray = glXChooseFBConfig( fgDisplay.Display, - fgDisplay.Screen, - attributes, - &fbconfigArraySize ); - - if (fbconfigArray != NULL) - { - int result; /* Returned by glXGetFBConfigAttrib, not checked. */ - - - if( wantIndexedMode ) - { - /* - * In index mode, we want the largest buffer size, i.e. visual - * depth. Here, FBConfigs are sorted by increasing buffer size - * first, so FBConfigs with the largest size come last. - */ - - int bufferSizeMin, bufferSizeMax; - - /* Get bufferSizeMin. */ - result = - glXGetFBConfigAttrib( fgDisplay.Display, - fbconfigArray[0], - GLX_BUFFER_SIZE, - &bufferSizeMin ); - /* Get bufferSizeMax. */ - result = - glXGetFBConfigAttrib( fgDisplay.Display, - fbconfigArray[fbconfigArraySize - 1], - GLX_BUFFER_SIZE, - &bufferSizeMax ); - - if (bufferSizeMax > bufferSizeMin) - { - /* - * Free and reallocate fbconfigArray, keeping only FBConfigs - * with the largest buffer size. - */ - XFree(fbconfigArray); - - /* Add buffer size token at the end of the list. */ - where--; - ATTRIB_VAL( GLX_BUFFER_SIZE, bufferSizeMax ); - ATTRIB( None ); - - fbconfigArray = glXChooseFBConfig( fgDisplay.Display, - fgDisplay.Screen, - attributes, - &fbconfigArraySize ); - } - } - - /* - * We now have an array of FBConfigs, the first one being the "best" - * one. So we should return only this FBConfig: - * - * int fbconfigXID; - * - * - pick the XID of the FBConfig we want - * result = glXGetFBConfigAttrib( fgDisplay.Display, - * fbconfigArray[0], - * GLX_FBCONFIG_ID, - * &fbconfigXID ); - * - * - free the array - * XFree(fbconfigArray); - * - * - reset "attributes" with the XID - * where = 0; - * ATTRIB_VAL( GLX_FBCONFIG_ID, fbconfigXID ); - * ATTRIB( None ); - * - * - get our FBConfig only - * fbconfig = glXChooseFBConfig( fgDisplay.Display, - * fgDisplay.Screen, - * attributes, - * &fbconfigArraySize ); - * - * However, for some configurations (for instance multisampling with - * Mesa 6.5.2 and ATI drivers), this does not work: - * glXChooseFBConfig returns NULL, whereas fbconfigXID is a valid - * XID. Further investigation is needed. - * - * So, for now, we return the whole array of FBConfigs. This should - * not produce any side effects elsewhere. - */ - fbconfig = fbconfigArray; - } - else - { - fbconfig = NULL; - } - - if (numcfgs) - *numcfgs = fbconfigArraySize; - - return fbconfig; - } -} - - -static void fghFillContextAttributes( int *attributes ) { - int where = 0, contextFlags, contextProfile; - - if ( !fghIsLegacyContextVersionRequested() ) { - ATTRIB_VAL( GLX_CONTEXT_MAJOR_VERSION_ARB, fgState.MajorVersion ); - ATTRIB_VAL( GLX_CONTEXT_MINOR_VERSION_ARB, fgState.MinorVersion ); - } - - contextFlags = - fghMapBit( fgState.ContextFlags, GLUT_DEBUG, GLX_CONTEXT_DEBUG_BIT_ARB ) | - fghMapBit( fgState.ContextFlags, GLUT_FORWARD_COMPATIBLE, GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB ); - if ( contextFlags != 0 ) { - ATTRIB_VAL( GLX_CONTEXT_FLAGS_ARB, contextFlags ); - } - - contextProfile = - fghMapBit( fgState.ContextProfile, GLUT_CORE_PROFILE, GLX_CONTEXT_CORE_PROFILE_BIT_ARB ) | - fghMapBit( fgState.ContextProfile, GLUT_COMPATIBILITY_PROFILE, GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB ); - if ( contextProfile != 0 ) { - ATTRIB_VAL( GLX_CONTEXT_PROFILE_MASK_ARB, contextProfile ); - } - - ATTRIB( 0 ); -} - -typedef GLXContext (*CreateContextAttribsProc)(Display *dpy, GLXFBConfig config, - GLXContext share_list, Bool direct, - const int *attrib_list); - -static GLXContext fghCreateNewContext( SFG_Window* window ) -{ - /* for color model calculation */ - int menu = ( window->IsMenu && !fgStructure.MenuContext ); - int index_mode = ( fgState.DisplayMode & GLUT_INDEX ); - - /* "classic" context creation */ - Display *dpy = fgDisplay.Display; - GLXFBConfig config = *(window->Window.FBConfig); - int render_type = ( !menu && index_mode ) ? GLX_COLOR_INDEX_TYPE : GLX_RGBA_TYPE; - GLXContext share_list = NULL; - Bool direct = ( fgState.DirectContext != GLUT_FORCE_INDIRECT_CONTEXT ); - GLXContext context; - - /* new context creation */ - int attributes[9]; - CreateContextAttribsProc createContextAttribs = (CreateContextAttribsProc) fghGetProcAddress( "glXCreateContextAttribsARB" ); - - /* glXCreateContextAttribsARB not found, yet the user has requested the new context creation */ - if ( !createContextAttribs && !fghIsLegacyContextRequested() ) { - fgWarning( "OpenGL >2.1 context requested but glXCreateContextAttribsARB is not available! Falling back to legacy context creation" ); - fgState.MajorVersion = 2; - fgState.MinorVersion = 1; - } - - /* If nothing fancy has been required, simply use the old context creation GLX API entry */ - if ( fghIsLegacyContextRequested() || !createContextAttribs ) - { - context = glXCreateNewContext( dpy, config, render_type, share_list, direct ); - if ( context == NULL ) { - fghContextCreationError(); - } - return context; - } - - /* color index mode is not available anymore with OpenGL 3.0 */ - if ( render_type == GLX_COLOR_INDEX_TYPE ) { - fgWarning( "color index mode is deprecated, using RGBA mode" ); - } - - fghFillContextAttributes( attributes ); - - context = createContextAttribs( dpy, config, share_list, direct, attributes ); - if ( context == NULL ) { - fghContextCreationError(); - } - return context; -} - - -#define _NET_WM_STATE_TOGGLE 2 -static int fghResizeFullscrToggle(void) -{ - XWindowAttributes attributes; - - if(glutGet(GLUT_FULL_SCREEN)) { - /* restore original window size */ - SFG_Window *win = fgStructure.CurrentWindow; - fgStructure.CurrentWindow->State.NeedToResize = GL_TRUE; - fgStructure.CurrentWindow->State.Width = win->State.OldWidth; - fgStructure.CurrentWindow->State.Height = win->State.OldHeight; - - } else { - /* resize the window to cover the entire screen */ - XGetWindowAttributes(fgDisplay.Display, - fgStructure.CurrentWindow->Window.Handle, - &attributes); - - /* - * The "x" and "y" members of "attributes" are the window's coordinates - * relative to its parent, i.e. to the decoration window. - */ - XMoveResizeWindow(fgDisplay.Display, - fgStructure.CurrentWindow->Window.Handle, - -attributes.x, - -attributes.y, - fgDisplay.ScreenWidth, - fgDisplay.ScreenHeight); - } - return 0; -} - -static int fghEwmhFullscrToggle(void) -{ - XEvent xev; - long evmask = SubstructureRedirectMask | SubstructureNotifyMask; - - if(!fgDisplay.State || !fgDisplay.StateFullScreen) { - return -1; - } - - xev.type = ClientMessage; - xev.xclient.window = fgStructure.CurrentWindow->Window.Handle; - xev.xclient.message_type = fgDisplay.State; - xev.xclient.format = 32; - xev.xclient.data.l[0] = _NET_WM_STATE_TOGGLE; - xev.xclient.data.l[1] = fgDisplay.StateFullScreen; - xev.xclient.data.l[2] = 0; /* no second property to toggle */ - xev.xclient.data.l[3] = 1; /* source indication: application */ - xev.xclient.data.l[4] = 0; /* unused */ - - if(!XSendEvent(fgDisplay.Display, fgDisplay.RootWindow, 0, evmask, &xev)) { - return -1; - } - return 0; -} - -static int fghToggleFullscreen(void) -{ - /* first try the EWMH (_NET_WM_STATE) method ... */ - if(fghEwmhFullscrToggle() != -1) { - return 0; - } - - /* fall back to resizing the window */ - if(fghResizeFullscrToggle() != -1) { - return 0; - } - return -1; -} - - -#endif /* TARGET_HOST_POSIX_X11 */ - - -#if TARGET_HOST_MS_WINDOWS -/* - * Setup the pixel format for a Win32 window - */ - -#if defined(_WIN32_WCE) -static wchar_t* fghWstrFromStr(const char* str) -{ - int i,len=strlen(str); - wchar_t* wstr = (wchar_t*)malloc(2*len+2); - for(i=0; iWindow.Device, window->Window.Context ); - - if ( !fghIsExtensionSupported( window->Window.Device, "WGL_ARB_create_context" ) ) - { - return; - } - - /* new context creation */ - fghFillContextAttributes( attributes ); - - wglCreateContextAttribsARB = (PFNWGLCREATECONTEXTATTRIBSARBPROC) wglGetProcAddress( "wglCreateContextAttribsARB" ); - if ( wglCreateContextAttribsARB == NULL ) - { - fgError( "wglCreateContextAttribsARB not found" ); - } - - context = wglCreateContextAttribsARB( window->Window.Device, 0, attributes ); - if ( context == NULL ) - { - fghContextCreationError(); - } - - wglMakeCurrent( NULL, NULL ); - wglDeleteContext( window->Window.Context ); - window->Window.Context = context; -} - -#if !defined(_WIN32_WCE) - -static void fghFillPFD( PIXELFORMATDESCRIPTOR *ppfd, HDC hdc, unsigned char layer_type ) -{ - int flags = PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL; - if ( fgState.DisplayMode & GLUT_DOUBLE ) { - flags |= PFD_DOUBLEBUFFER; - } - if ( fgState.DisplayMode & GLUT_STEREO ) { - flags |= PFD_STEREO; - } - -#if defined(_MSC_VER) -#pragma message( "fgSetupPixelFormat(): there is still some work to do here!" ) -#endif - - /* Specify which pixel format do we opt for... */ - ppfd->nSize = sizeof(PIXELFORMATDESCRIPTOR); - ppfd->nVersion = 1; - ppfd->dwFlags = flags; - - if( fgState.DisplayMode & GLUT_INDEX ) { - ppfd->iPixelType = PFD_TYPE_COLORINDEX; - ppfd->cRedBits = 0; - ppfd->cGreenBits = 0; - ppfd->cBlueBits = 0; - ppfd->cAlphaBits = 0; - } else { - ppfd->iPixelType = PFD_TYPE_RGBA; - ppfd->cRedBits = 8; - ppfd->cGreenBits = 8; - ppfd->cBlueBits = 8; - ppfd->cAlphaBits = ( fgState.DisplayMode & GLUT_ALPHA ) ? 8 : 0; - } - - ppfd->cColorBits = 24; - ppfd->cRedShift = 0; - ppfd->cGreenShift = 0; - ppfd->cBlueShift = 0; - ppfd->cAlphaShift = 0; - ppfd->cAccumBits = ( fgState.DisplayMode & GLUT_ACCUM ) ? 1 : 0; - ppfd->cAccumRedBits = 0; - ppfd->cAccumGreenBits = 0; - ppfd->cAccumBlueBits = 0; - ppfd->cAccumAlphaBits = 0; - - /* Hmmm, or 32/0 instead of 24/8? */ - ppfd->cDepthBits = 24; - ppfd->cStencilBits = 8; - - ppfd->cAuxBuffers = fghNumberOfAuxBuffersRequested(); - ppfd->iLayerType = layer_type; - ppfd->bReserved = 0; - ppfd->dwLayerMask = 0; - ppfd->dwVisibleMask = 0; - ppfd->dwDamageMask = 0; - - ppfd->cColorBits = (BYTE) GetDeviceCaps( hdc, BITSPIXEL ); -} - -static void fghFillPixelFormatAttributes( int *attributes, const PIXELFORMATDESCRIPTOR *ppfd ) -{ - int where = 0; - - ATTRIB_VAL( WGL_DRAW_TO_WINDOW_ARB, GL_TRUE ); - ATTRIB_VAL( WGL_SUPPORT_OPENGL_ARB, GL_TRUE ); - ATTRIB_VAL( WGL_ACCELERATION_ARB, WGL_FULL_ACCELERATION_ARB ); - - ATTRIB_VAL( WGL_COLOR_BITS_ARB, ppfd->cColorBits ); - ATTRIB_VAL( WGL_ALPHA_BITS_ARB, ppfd->cAlphaBits ); - ATTRIB_VAL( WGL_DEPTH_BITS_ARB, ppfd->cDepthBits ); - ATTRIB_VAL( WGL_STENCIL_BITS_ARB, ppfd->cStencilBits ); - - ATTRIB_VAL( WGL_DOUBLE_BUFFER_ARB, ( fgState.DisplayMode & GLUT_DOUBLE ) != 0 ); - - if ( fgState.DisplayMode & GLUT_SRGB ) { - ATTRIB_VAL( WGL_FRAMEBUFFER_SRGB_CAPABLE_ARB, TRUE ); - } - - ATTRIB_VAL( WGL_SAMPLE_BUFFERS_ARB, GL_TRUE ); - ATTRIB_VAL( WGL_SAMPLES_ARB, fgState.SampleNumber ); - ATTRIB( 0 ); -} -#endif - -GLboolean fgSetupPixelFormat( SFG_Window* window, GLboolean checkOnly, - unsigned char layer_type ) -{ -#if defined(_WIN32_WCE) - return GL_TRUE; -#else - PIXELFORMATDESCRIPTOR pfd; - PIXELFORMATDESCRIPTOR* ppfd = &pfd; - int pixelformat; - HDC current_hDC; - GLboolean success; - - if (checkOnly) - current_hDC = CreateDC(TEXT("DISPLAY"), NULL ,NULL ,NULL); - else - current_hDC = window->Window.Device; - - fghFillPFD( ppfd, current_hDC, layer_type ); - pixelformat = ChoosePixelFormat( current_hDC, ppfd ); - - /* windows hack for multismapling/sRGB */ - if ( ( fgState.DisplayMode & GLUT_MULTISAMPLE ) || - ( fgState.DisplayMode & GLUT_SRGB ) ) - { - HGLRC rc, rc_before=wglGetCurrentContext(); - HWND hWnd; - HDC hDC, hDC_before=wglGetCurrentDC(); - WNDCLASS wndCls; - - /* create a dummy window */ - ZeroMemory(&wndCls, sizeof(wndCls)); - wndCls.lpfnWndProc = DefWindowProc; - wndCls.hInstance = fgDisplay.Instance; - wndCls.style = CS_OWNDC | CS_HREDRAW | CS_VREDRAW; - wndCls.lpszClassName = _T("FREEGLUT_dummy"); - RegisterClass( &wndCls ); - - hWnd=CreateWindow(_T("FREEGLUT_dummy"), _T(""), WS_CLIPSIBLINGS | WS_CLIPCHILDREN | WS_OVERLAPPEDWINDOW , 0,0,0,0, 0, 0, fgDisplay.Instance, 0 ); - hDC=GetDC(hWnd); - SetPixelFormat( hDC, pixelformat, ppfd ); - - rc = wglCreateContext( hDC ); - wglMakeCurrent(hDC, rc); - - if ( fghIsExtensionSupported( hDC, "WGL_ARB_multisample" ) ) - { - PFNWGLCHOOSEPIXELFORMATARBPROC wglChoosePixelFormatARBProc = - (PFNWGLCHOOSEPIXELFORMATARBPROC) wglGetProcAddress("wglChoosePixelFormatARB"); - if ( wglChoosePixelFormatARBProc ) - { - int attributes[100]; - int iPixelFormat; - BOOL bValid; - float fAttributes[] = { 0, 0 }; - UINT numFormats; - fghFillPixelFormatAttributes( attributes, ppfd ); - bValid = wglChoosePixelFormatARBProc(hDC, attributes, fAttributes, 1, &iPixelFormat, &numFormats); - - if ( bValid && numFormats > 0 ) - { - pixelformat = iPixelFormat; - } - } - } - - wglMakeCurrent( hDC_before, rc_before); - wglDeleteContext(rc); - ReleaseDC(hWnd, hDC); - DestroyWindow(hWnd); - UnregisterClass(_T("FREEGLUT_dummy"), fgDisplay.Instance); - } - - success = ( pixelformat != 0 ) && ( checkOnly || SetPixelFormat( current_hDC, pixelformat, ppfd ) ); - - if (checkOnly) - DeleteDC(current_hDC); - - return success; -#endif /* defined(_WIN32_WCE) */ -} - -#endif /* TARGET_HOST_MS_WINDOWS */ - -/* - * Sets the OpenGL context and the fgStructure "Current Window" pointer to - * the window structure passed in. - */ -void fgSetWindow ( SFG_Window *window ) -{ -#if TARGET_HOST_POSIX_X11 - if ( window ) - { - glXMakeContextCurrent( - fgDisplay.Display, - window->Window.Handle, - window->Window.Handle, - window->Window.Context - ); - } -#elif TARGET_HOST_MS_WINDOWS - if ( window != fgStructure.CurrentWindow ) - { - if( fgStructure.CurrentWindow ) - ReleaseDC( fgStructure.CurrentWindow->Window.Handle, - fgStructure.CurrentWindow->Window.Device ); - - if ( window ) - { - window->Window.Device = GetDC( window->Window.Handle ); - wglMakeCurrent( - window->Window.Device, - window->Window.Context - ); - } - } -#endif - fgStructure.CurrentWindow = window; -} - -#if TARGET_HOST_MS_WINDOWS - -/* Computes position of corners of window Rect (outer position including - * decorations) based on the provided client rect and based on the style - * of the window in question. - * If posIsOutside is set to true, the input client Rect is taken to follow - * freeGLUT's window specification convention in which the top-left corner - * is at the outside of the window, while the size - * (rect.right-rect.left,rect.bottom-rect.top) is the size of the drawable - * area. - */ -void fghComputeWindowRectFromClientArea_UseStyle( const DWORD windowStyle, RECT *clientRect, BOOL posIsOutside ) -{ - int xBorderWidth = 0, yBorderWidth = 0; - - /* If window has title bar, correct rect for it */ - if (windowStyle & WS_MAXIMIZEBOX) /* Need to query for WS_MAXIMIZEBOX to see if we have a title bar, the WS_CAPTION query is also true for a WS_DLGFRAME only... */ - if (posIsOutside) - clientRect->bottom += GetSystemMetrics( SM_CYCAPTION ); - else - clientRect->top -= GetSystemMetrics( SM_CYCAPTION ); - - /* get width of window's borders (frame), correct rect for it. - * Note, borders can be of zero width if style does not specify borders - */ - fghGetBorderWidth(windowStyle, &xBorderWidth, &yBorderWidth); - if (posIsOutside) - { - clientRect->right += xBorderWidth * 2; - clientRect->bottom += yBorderWidth * 2; - } - else - { - clientRect->left -= xBorderWidth; - clientRect->right += xBorderWidth; - clientRect->top -= yBorderWidth; - clientRect->bottom += yBorderWidth; - } -} - -/* Computes position of corners of window Rect (outer position including - * decorations) based on the provided client rect and based on the style - * of the window in question. If the window pointer or the window handle - * is NULL, a fully decorated window (caption and border) is assumed. - * Furthermore, if posIsOutside is set to true, the input client Rect is - * taken to follow freeGLUT's window specification convention in which the - * top-left corner is at the outside of the window, while the size - * (rect.right-rect.left,rect.bottom-rect.top) is the size of the drawable - * area. -*/ -void fghComputeWindowRectFromClientArea_QueryWindow( const SFG_Window *window, RECT *clientRect, BOOL posIsOutside ) -{ - DWORD windowStyle = 0; - - if (window && window->Window.Handle) - windowStyle = GetWindowLong(window->Window.Handle, GWL_STYLE); - else - windowStyle = WS_OVERLAPPEDWINDOW; - - fghComputeWindowRectFromClientArea_UseStyle(windowStyle, clientRect, posIsOutside); -} - -/* Computes position of corners of client area (drawable area) of a window - * based on the provided window Rect (outer position including decorations) - * and based on the style of the window in question. If the window pointer - * or the window handle is NULL, a fully decorated window (caption and - * border) is assumed. - * Furthermore, if wantPosOutside is set to true, the output client Rect - * will follow freeGLUT's window specification convention in which the - * top-left corner is at the outside of the window, the size - * (rect.right-rect.left,rect.bottom-rect.top) is the size of the drawable - * area. - */ -void fghComputeClientAreaFromWindowRect( const SFG_Window *window, RECT *windowRect, BOOL wantPosOutside ) -{ - DWORD windowStyle = 0; - int xBorderWidth = 0, yBorderWidth = 0; - - if (window && window->Window.Handle) - windowStyle = GetWindowLong(window->Window.Handle, GWL_STYLE); - else - windowStyle = WS_OVERLAPPEDWINDOW; - - /* If window has title bar, correct rect for it */ - if (windowStyle & WS_MAXIMIZEBOX) /* Need to query for WS_MAXIMIZEBOX to see if we have a title bar, the WS_CAPTION query is also true for a WS_DLGFRAME only... */ - if (wantPosOutside) - windowRect->bottom -= GetSystemMetrics( SM_CYCAPTION ); - else - windowRect->top += GetSystemMetrics( SM_CYCAPTION ); - - /* get width of window's borders (frame), correct rect for it. - * Note, borders can be of zero width if style does not specify borders - */ - fghGetBorderWidth(windowStyle, &xBorderWidth, &yBorderWidth); - if (wantPosOutside) - { - windowRect->right -= xBorderWidth * 2; - windowRect->bottom -= yBorderWidth * 2; - } - else - { - windowRect->left += xBorderWidth; - windowRect->right -= xBorderWidth; - windowRect->top += yBorderWidth; - windowRect->bottom -= yBorderWidth; - } -} - -/* Gets the rect describing the client area (drawable area) of the - * specified window. - * Returns an empty rect if window pointer or window handle is NULL. - * If wantPosOutside is set to true, the output client Rect - * will follow freeGLUT's window specification convention in which the - * top-left corner is at the outside of the window, while the size - * (rect.right-rect.left,rect.bottom-rect.top) is the size of the drawable - * area. - */ -RECT fghGetClientArea( const SFG_Window *window, BOOL wantPosOutside ) -{ - RECT windowRect = {0,0,0,0}; - - freeglut_return_val_if_fail((window && window->Window.Handle),windowRect); - - /* - * call GetWindowRect() - * (this returns the pixel coordinates of the outside of the window) - */ - GetWindowRect( window->Window.Handle, &windowRect ); - - /* Then correct the results */ - fghComputeClientAreaFromWindowRect(window, &windowRect, wantPosOutside); - - return windowRect; -} - -/* Returns the width of the window borders based on the window's style. - */ -void fghGetBorderWidth(const DWORD windowStyle, int* xBorderWidth, int* yBorderWidth) -{ - if (windowStyle & WS_THICKFRAME) - { - *xBorderWidth = GetSystemMetrics(SM_CXSIZEFRAME); - *yBorderWidth = GetSystemMetrics(SM_CYSIZEFRAME); - } - else if (windowStyle & WS_DLGFRAME) - { - *xBorderWidth = GetSystemMetrics(SM_CXFIXEDFRAME); - *yBorderWidth = GetSystemMetrics(SM_CYFIXEDFRAME); - } - else - { - *xBorderWidth = 0; - *yBorderWidth = 0; - } -} - -#if(WINVER >= 0x500) -typedef struct -{ - int *x; - int *y; - const char *name; -} m_proc_t; - -static BOOL CALLBACK m_proc(HMONITOR mon, - HDC hdc, - LPRECT rect, - LPARAM data) -{ - m_proc_t *dp=(m_proc_t *)data; - MONITORINFOEX info; - BOOL res; - info.cbSize=sizeof(info); - res=GetMonitorInfo(mon,(LPMONITORINFO)&info); - if( res ) - { - if( strcmp(dp->name,info.szDevice)==0 ) - { - *(dp->x)=info.rcMonitor.left; - *(dp->y)=info.rcMonitor.top; - return FALSE; - } - } - return TRUE; -} - -/* - * this function returns the origin of the screen identified by - * fgDisplay.DisplayName, and 0 otherwise. - * This is used in fgOpenWindow to open the gamemode window on the screen - * identified by the -display command line argument. The function should - * not be called otherwise. - */ - -static void get_display_origin(int *xp,int *yp) -{ - *xp = 0; - *yp = 0; - - if( fgDisplay.DisplayName ) - { - m_proc_t st; - st.x=xp; - st.y=yp; - st.name=fgDisplay.DisplayName; - EnumDisplayMonitors(0,0,m_proc,(LPARAM)&st); - } -} -#else -#pragma message( "-display parameter only works if compiled with WINVER >= 0x0500") - -static void get_display_origin(int *xp,int *yp) -{ - *xp = 0; - *yp = 0; - - if( fgDisplay.DisplayName ) - { - fgWarning( "for working -display support FreeGLUT must be compiled with WINVER >= 0x0500"); - } -} -#endif -#endif - - -#if TARGET_HOST_POSIX_X11 -static Bool fghWindowIsVisible( Display *display, XEvent *event, XPointer arg) -{ - Window window = (Window)arg; - return (event->type == MapNotify) && (event->xmap.window == window); -} -#endif - - -/* - * Opens a window. Requires a SFG_Window object created and attached - * to the freeglut structure. OpenGL context is created here. - */ -void fgOpenWindow( SFG_Window* window, const char* title, - GLboolean positionUse, int x, int y, - GLboolean sizeUse, int w, int h, - GLboolean gameMode, GLboolean isSubWindow ) -{ -#if TARGET_HOST_POSIX_X11 - XVisualInfo * visualInfo = NULL; - XSetWindowAttributes winAttr; - XTextProperty textProperty; - XSizeHints sizeHints; - XWMHints wmHints; - XEvent eventReturnBuffer; /* return buffer required for a call */ - unsigned long mask; - int num_FBConfigs, i; - unsigned int current_DisplayMode = fgState.DisplayMode ; - - /* Save the display mode if we are creating a menu window */ - if( window->IsMenu && ( ! fgStructure.MenuContext ) ) - fgState.DisplayMode = GLUT_DOUBLE | GLUT_RGB ; - - window->Window.FBConfig = fgChooseFBConfig( &num_FBConfigs ); - - if( window->IsMenu && ( ! fgStructure.MenuContext ) ) - fgState.DisplayMode = current_DisplayMode ; - - if( ! window->Window.FBConfig ) - { - /* - * The "fgChooseFBConfig" returned a null meaning that the visual - * context is not available. - * Try a couple of variations to see if they will work. - */ - if( !( fgState.DisplayMode & GLUT_DOUBLE ) ) - { - fgState.DisplayMode |= GLUT_DOUBLE ; - window->Window.FBConfig = fgChooseFBConfig( &num_FBConfigs ); - fgState.DisplayMode &= ~GLUT_DOUBLE; - } - - if( fgState.DisplayMode & GLUT_MULTISAMPLE ) - { - fgState.DisplayMode &= ~GLUT_MULTISAMPLE ; - window->Window.FBConfig = fgChooseFBConfig( &num_FBConfigs ); - fgState.DisplayMode |= GLUT_MULTISAMPLE; - } - } - - FREEGLUT_INTERNAL_ERROR_EXIT( window->Window.FBConfig != NULL, - "FBConfig with necessary capabilities not found", "fgOpenWindow" ); - - /* Get the X visual. */ - for (i = 0; i < num_FBConfigs; i++) { - visualInfo = glXGetVisualFromFBConfig( fgDisplay.Display, - window->Window.FBConfig[i] ); - if (visualInfo) - break; - } - - FREEGLUT_INTERNAL_ERROR_EXIT( visualInfo != NULL, - "visualInfo could not be retrieved from FBConfig", "fgOpenWindow" ); - - /* - * XXX HINT: the masks should be updated when adding/removing callbacks. - * XXX This might speed up message processing. Is that true? - * XXX - * XXX A: Not appreciably, but it WILL make it easier to debug. - * XXX Try tracing old GLUT and try tracing freeglut. Old GLUT - * XXX turns off events that it doesn't need and is a whole lot - * XXX more pleasant to trace. (Think mouse-motion! Tons of - * XXX ``bonus'' GUI events stream in.) - */ - winAttr.event_mask = - StructureNotifyMask | SubstructureNotifyMask | ExposureMask | - ButtonPressMask | ButtonReleaseMask | KeyPressMask | KeyReleaseMask | - VisibilityChangeMask | EnterWindowMask | LeaveWindowMask | - PointerMotionMask | ButtonMotionMask; - winAttr.background_pixmap = None; - winAttr.background_pixel = 0; - winAttr.border_pixel = 0; - - winAttr.colormap = XCreateColormap( - fgDisplay.Display, fgDisplay.RootWindow, - visualInfo->visual, AllocNone - ); - - mask = CWBackPixmap | CWBorderPixel | CWColormap | CWEventMask; - - if( window->IsMenu || ( gameMode == GL_TRUE ) ) - { - winAttr.override_redirect = True; - mask |= CWOverrideRedirect; - } - - if( ! positionUse ) - x = y = -1; /* default window position */ - if( ! sizeUse ) - w = h = 300; /* default window size */ - - window->Window.Handle = XCreateWindow( - fgDisplay.Display, - window->Parent == NULL ? fgDisplay.RootWindow : - window->Parent->Window.Handle, - x, y, w, h, 0, - visualInfo->depth, InputOutput, - visualInfo->visual, mask, - &winAttr - ); - - /* - * The GLX context creation, possibly trying the direct context rendering - * or else use the current context if the user has so specified - */ - - if( window->IsMenu ) - { - /* - * If there isn't already an OpenGL rendering context for menu - * windows, make one - */ - if( !fgStructure.MenuContext ) - { - fgStructure.MenuContext = - (SFG_MenuContext *)malloc( sizeof(SFG_MenuContext) ); - fgStructure.MenuContext->MContext = fghCreateNewContext( window ); - } - - /* window->Window.Context = fgStructure.MenuContext->MContext; */ - window->Window.Context = fghCreateNewContext( window ); - } - else if( fgState.UseCurrentContext ) - { - window->Window.Context = glXGetCurrentContext( ); - - if( ! window->Window.Context ) - window->Window.Context = fghCreateNewContext( window ); - } - else - window->Window.Context = fghCreateNewContext( window ); - -#if !defined( __FreeBSD__ ) && !defined( __NetBSD__ ) - if( !glXIsDirect( fgDisplay.Display, window->Window.Context ) ) - { - if( fgState.DirectContext == GLUT_FORCE_DIRECT_CONTEXT ) - fgError( "Unable to force direct context rendering for window '%s'", - title ); - } -#endif - - /* - * XXX Assume the new window is visible by default - * XXX Is this a safe assumption? - */ - window->State.Visible = GL_TRUE; - - sizeHints.flags = 0; - if ( positionUse ) - sizeHints.flags |= USPosition; - if ( sizeUse ) - sizeHints.flags |= USSize; - - /* - * Fill in the size hints values now (the x, y, width and height - * settings are obsolete, are there any more WMs that support them?) - * Unless the X servers actually stop supporting these, we should - * continue to fill them in. It is *not* our place to tell the user - * that they should replace a window manager that they like, and which - * works, just because *we* think that it's not "modern" enough. - */ - sizeHints.x = x; - sizeHints.y = y; - sizeHints.width = w; - sizeHints.height = h; - - wmHints.flags = StateHint; - wmHints.initial_state = fgState.ForceIconic ? IconicState : NormalState; - /* Prepare the window and iconified window names... */ - XStringListToTextProperty( (char **) &title, 1, &textProperty ); - - XSetWMProperties( - fgDisplay.Display, - window->Window.Handle, - &textProperty, - &textProperty, - 0, - 0, - &sizeHints, - &wmHints, - NULL - ); - XFree( textProperty.value ); - - XSetWMProtocols( fgDisplay.Display, window->Window.Handle, - &fgDisplay.DeleteWindow, 1 ); - - glXMakeContextCurrent( - fgDisplay.Display, - window->Window.Handle, - window->Window.Handle, - window->Window.Context - ); - - /* register extension events _before_ window is mapped */ - #ifdef HAVE_X11_EXTENSIONS_XINPUT2_H - fgRegisterDevices( fgDisplay.Display, &(window->Window.Handle) ); - #endif - - XMapWindow( fgDisplay.Display, window->Window.Handle ); - - XFree(visualInfo); - - if( !isSubWindow) - XPeekIfEvent( fgDisplay.Display, &eventReturnBuffer, &fghWindowIsVisible, (XPointer)(window->Window.Handle) ); - -#elif TARGET_HOST_MS_WINDOWS - - WNDCLASS wc; - DWORD flags = 0; - DWORD exFlags = 0; - ATOM atom; - - /* Grab the window class we have registered on glutInit(): */ - atom = GetClassInfo( fgDisplay.Instance, _T("FREEGLUT"), &wc ); - FREEGLUT_INTERNAL_ERROR_EXIT ( atom, "Window Class Info Not Found", - "fgOpenWindow" ); - - /* Determine window style flags*/ - if( gameMode ) - { - FREEGLUT_INTERNAL_ERROR_EXIT ( window->Parent == NULL, - "Game mode being invoked on a subwindow", - "fgOpenWindow" ); - - /* - * Set the window creation flags appropriately to make the window - * entirely visible: - */ - flags = WS_POPUP | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | WS_VISIBLE; - } - else - { - flags = WS_CLIPSIBLINGS | WS_CLIPCHILDREN; - - /* - * There's a small difference between creating the top, child and - * menu windows - */ - if ( window->IsMenu ) - { - flags |= WS_POPUP; - exFlags |= WS_EX_TOOLWINDOW; - } -#if defined(_WIN32_WCE) - /* no decorations for windows CE */ -#else - /* if this is not a subwindow (child), set its style based on the requested display mode */ - else if( window->Parent == NULL ) - if ( fgState.DisplayMode & GLUT_BORDERLESS ) - { - /* no window decorations needed */ - } - else if ( fgState.DisplayMode & GLUT_CAPTIONLESS ) - /* only window decoration is a border, no title bar or buttons */ - flags |= WS_DLGFRAME; - else - /* window decoration are a border, title bar and buttons. - * NB: we later query whether the window has a title bar or - * not by testing for the maximize button, as the test for - * WS_CAPTION can be true without the window having a title - * bar. This style WS_OVERLAPPEDWINDOW gives you a maximize - * button. */ - flags |= WS_OVERLAPPEDWINDOW; -#endif - else - /* subwindows always have no decoration, but are marked as a child window to the OS */ - flags |= WS_CHILD; - } - - /* determine window size and position */ - if( gameMode ) - { - /* if in gamemode, query the origin of specified by the -display - * command line parameter (if any) and offset the upper-left corner - * of the window so we create the window on that screen. - * The -display argument doesn't do anything if not trying to enter - * gamemode. - */ - int xoff=0, yoff=0; - get_display_origin(&xoff,&yoff); - x += xoff; - y += yoff; - } - if( !positionUse ) - { - x = CW_USEDEFAULT; - y = CW_USEDEFAULT; - } - if( !sizeUse ) - { - if( ! window->IsMenu ) - { - w = CW_USEDEFAULT; - h = CW_USEDEFAULT; - } - else /* fail safe - Windows can make a window of size (0, 0) */ - w = h = 300; /* default window size */ - } - /* store requested client area width and height */ - window->State.Width = w; - window->State.Height = h; - -#if !defined(_WIN32_WCE) /* no decorations for windows CE */ - if( sizeUse ) - { - RECT windowRect; - /* - * Update the window dimensions, taking the window decorations - * into account. FreeGLUT is to create the window with the - * topleft outside corner at (x,y) and with client area - * dimensions (w,h). - * note: don't need to do this when w=h=CW_USEDEFAULT, so in the - * if( sizeUse ) here is convenient. - */ - windowRect.left = x; - windowRect.top = y; - windowRect.right = x+w; - windowRect.bottom = y+h; - - fghComputeWindowRectFromClientArea_UseStyle(flags,&windowRect,TRUE); - - w = windowRect.right - windowRect.left; - h = windowRect.bottom- windowRect.top; - } -#endif /* !defined(_WIN32_WCE) */ - -#if defined(_WIN32_WCE) - { - wchar_t* wstr = fghWstrFromStr(title); - - window->Window.Handle = CreateWindow( - _T("FREEGLUT"), - wstr, - WS_VISIBLE | WS_POPUP, - 0,0, 240,320, - NULL, - NULL, - fgDisplay.Instance, - (LPVOID) window - ); - - free(wstr); - - SHFullScreen(window->Window.Handle, SHFS_HIDESTARTICON); - SHFullScreen(window->Window.Handle, SHFS_HIDESIPBUTTON); - SHFullScreen(window->Window.Handle, SHFS_HIDETASKBAR); - MoveWindow(window->Window.Handle, 0, 0, 240, 320, TRUE); - ShowWindow(window->Window.Handle, SW_SHOW); - UpdateWindow(window->Window.Handle); - } -#else - window->Window.Handle = CreateWindowEx( - exFlags, - _T("FREEGLUT"), - title, - flags, - x, y, w, h, - (HWND) window->Parent == NULL ? NULL : window->Parent->Window.Handle, - (HMENU) NULL, - fgDisplay.Instance, - (LPVOID) window - ); -#endif /* defined(_WIN32_WCE) */ - - if( !( window->Window.Handle ) ) - fgError( "Failed to create a window (%s)!", title ); - -#if !defined(_WIN32_WCE) - /* Need to set requested style again, apparently Windows doesn't listen when requesting windows without title bar or borders */ - SetWindowLong(window->Window.Handle, GWL_STYLE, flags); - SetWindowPos(window->Window.Handle, HWND_TOP, 0,0,0,0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_FRAMECHANGED); -#endif /* defined(_WIN32_WCE) */ - - /* Make a menu window always on top - fix Feature Request 947118 */ - if( window->IsMenu || gameMode ) - SetWindowPos( - window->Window.Handle, - HWND_TOPMOST, - 0, 0, 0, 0, - SWP_NOMOVE | SWP_NOSIZE - ); - - /* Enable multitouch: additional flag TWF_FINETOUCH, TWF_WANTPALM */ - #ifdef WM_TOUCH - if (fghRegisterTouchWindow == (pRegisterTouchWindow)0xDEADBEEF) - fghRegisterTouchWindow = (pRegisterTouchWindow)GetProcAddress(GetModuleHandle("user32"),"RegisterTouchWindow"); - if (fghRegisterTouchWindow) - fghRegisterTouchWindow( window->Window.Handle, TWF_FINETOUCH | TWF_WANTPALM ); - #endif - -#if defined(_WIN32_WCE) - ShowWindow( window->Window.Handle, SW_SHOW ); -#else - ShowWindow( window->Window.Handle, - fgState.ForceIconic ? SW_SHOWMINIMIZED : SW_SHOW ); -#endif /* defined(_WIN32_WCE) */ - - UpdateWindow( window->Window.Handle ); - ShowCursor( TRUE ); /* XXX Old comments say "hide cursor"! */ - -#endif - - fgSetWindow( window ); - - window->Window.DoubleBuffered = - ( fgState.DisplayMode & GLUT_DOUBLE ) ? 1 : 0; - - if ( ! window->Window.DoubleBuffered ) - { - glDrawBuffer ( GL_FRONT ); - glReadBuffer ( GL_FRONT ); - } -} - -/* - * Closes a window, destroying the frame and OpenGL context - */ -void fgCloseWindow( SFG_Window* window ) -{ - /* if we're in gamemode and we're closing the gamemode window, - * call glutLeaveGameMode first to make sure the gamemode is - * properly closed before closing the window - */ - if (fgStructure.GameModeWindow != NULL && fgStructure.GameModeWindow->ID==window->ID) - glutLeaveGameMode(); - -#if TARGET_HOST_POSIX_X11 - - if( window->Window.Context ) - glXDestroyContext( fgDisplay.Display, window->Window.Context ); - XFree( window->Window.FBConfig ); - - if( window->Window.Handle ) { - XDestroyWindow( fgDisplay.Display, window->Window.Handle ); - } - /* XFlush( fgDisplay.Display ); */ /* XXX Shouldn't need this */ - -#elif TARGET_HOST_MS_WINDOWS - - /* Make sure we don't close a window with current context active */ - if( fgStructure.CurrentWindow == window ) - wglMakeCurrent( NULL, NULL ); - - /* - * Step through the list of windows. If the rendering context - * is not being used by another window, then we delete it. - */ - { - int used = FALSE ; - SFG_Window *iter ; - - for( iter = (SFG_Window *)fgStructure.Windows.First; - iter; - iter = (SFG_Window *)iter->Node.Next ) - { - if( ( iter->Window.Context == window->Window.Context ) && - ( iter != window ) ) - used = TRUE; - } - - if( ! used ) - wglDeleteContext( window->Window.Context ); - } - - DestroyWindow( window->Window.Handle ); -#endif -} - - -/* -- INTERFACE FUNCTIONS -------------------------------------------------- */ - -/* - * Creates a new top-level freeglut window - */ -int FGAPIENTRY glutCreateWindow( const char* title ) -{ - /* XXX GLUT does not exit; it simply calls "glutInit" quietly if the - * XXX application has not already done so. The "freeglut" community - * XXX decided not to go this route (freeglut-developer e-mail from - * XXX Steve Baker, 12/16/04, 4:22 PM CST, "Re: [Freeglut-developer] - * XXX Desired 'freeglut' behaviour when there is no current window" - */ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutCreateWindow" ); - - return fgCreateWindow( NULL, title, fgState.Position.Use, - fgState.Position.X, fgState.Position.Y, - fgState.Size.Use, fgState.Size.X, fgState.Size.Y, - GL_FALSE, GL_FALSE )->ID; -} - -#if TARGET_HOST_MS_WINDOWS -int FGAPIENTRY __glutCreateWindowWithExit( const char *title, void (__cdecl *exit_function)(int) ) -{ - __glutExitFunc = exit_function; - return glutCreateWindow( title ); -} -#endif - -/* - * This function creates a sub window. - */ -int FGAPIENTRY glutCreateSubWindow( int parentID, int x, int y, int w, int h ) -{ - int ret = 0; - SFG_Window* window = NULL; - SFG_Window* parent = NULL; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutCreateSubWindow" ); - parent = fgWindowByID( parentID ); - freeglut_return_val_if_fail( parent != NULL, 0 ); - if ( x < 0 ) - { - x = parent->State.Width + x ; - if ( w >= 0 ) x -= w ; - } - - if ( w < 0 ) w = parent->State.Width - x + w ; - if ( w < 0 ) - { - x += w ; - w = -w ; - } - - if ( y < 0 ) - { - y = parent->State.Height + y ; - if ( h >= 0 ) y -= h ; - } - - if ( h < 0 ) h = parent->State.Height - y + h ; - if ( h < 0 ) - { - y += h ; - h = -h ; - } - - window = fgCreateWindow( parent, "", GL_TRUE, x, y, GL_TRUE, w, h, GL_FALSE, GL_FALSE ); - ret = window->ID; - - return ret; -} - -/* - * Destroys a window and all of its subwindows - */ -void FGAPIENTRY glutDestroyWindow( int windowID ) -{ - SFG_Window* window; - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutDestroyWindow" ); - window = fgWindowByID( windowID ); - freeglut_return_if_fail( window != NULL ); - { - fgExecutionState ExecState = fgState.ExecState; - fgAddToWindowDestroyList( window ); - fgState.ExecState = ExecState; - } -} - -/* - * This function selects the current window - */ -void FGAPIENTRY glutSetWindow( int ID ) -{ - SFG_Window* window = NULL; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetWindow" ); - if( fgStructure.CurrentWindow != NULL ) - if( fgStructure.CurrentWindow->ID == ID ) - return; - - window = fgWindowByID( ID ); - if( window == NULL ) - { - fgWarning( "glutSetWindow(): window ID %d not found!", ID ); - return; - } - - fgSetWindow( window ); -} - -/* - * This function returns the ID number of the current window, 0 if none exists - */ -int FGAPIENTRY glutGetWindow( void ) -{ - SFG_Window *win = fgStructure.CurrentWindow; - /* - * Since GLUT did not throw an error if this function was called without a prior call to - * "glutInit", this function shouldn't do so here. Instead let us return a zero. - * See Feature Request "[ 1307049 ] glutInit check". - */ - if ( ! fgState.Initialised ) - return 0; - - while ( win && win->IsMenu ) - win = win->Parent; - return win ? win->ID : 0; -} - -/* - * This function makes the current window visible - */ -void FGAPIENTRY glutShowWindow( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutShowWindow" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutShowWindow" ); - -#if TARGET_HOST_POSIX_X11 - - XMapWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle ); - XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ - -#elif TARGET_HOST_MS_WINDOWS - - ShowWindow( fgStructure.CurrentWindow->Window.Handle, SW_SHOW ); - -#endif - - fgStructure.CurrentWindow->State.Redisplay = GL_TRUE; -} - -/* - * This function hides the current window - */ -void FGAPIENTRY glutHideWindow( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutHideWindow" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutHideWindow" ); - -#if TARGET_HOST_POSIX_X11 - - if( fgStructure.CurrentWindow->Parent == NULL ) - XWithdrawWindow( fgDisplay.Display, - fgStructure.CurrentWindow->Window.Handle, - fgDisplay.Screen ); - else - XUnmapWindow( fgDisplay.Display, - fgStructure.CurrentWindow->Window.Handle ); - XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ - -#elif TARGET_HOST_MS_WINDOWS - - ShowWindow( fgStructure.CurrentWindow->Window.Handle, SW_HIDE ); - -#endif - - fgStructure.CurrentWindow->State.Redisplay = GL_FALSE; -} - -/* - * Iconify the current window (top-level windows only) - */ -void FGAPIENTRY glutIconifyWindow( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutIconifyWindow" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutIconifyWindow" ); - - fgStructure.CurrentWindow->State.Visible = GL_FALSE; -#if TARGET_HOST_POSIX_X11 - - XIconifyWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle, - fgDisplay.Screen ); - XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ - -#elif TARGET_HOST_MS_WINDOWS - - ShowWindow( fgStructure.CurrentWindow->Window.Handle, SW_MINIMIZE ); - -#endif - - fgStructure.CurrentWindow->State.Redisplay = GL_FALSE; -} - -/* - * Set the current window's title - */ -void FGAPIENTRY glutSetWindowTitle( const char* title ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetWindowTitle" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutSetWindowTitle" ); - if( ! fgStructure.CurrentWindow->Parent ) - { -#if TARGET_HOST_POSIX_X11 - - XTextProperty text; - - text.value = (unsigned char *) title; - text.encoding = XA_STRING; - text.format = 8; - text.nitems = strlen( title ); - - XSetWMName( - fgDisplay.Display, - fgStructure.CurrentWindow->Window.Handle, - &text - ); - - XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ - -#elif TARGET_HOST_MS_WINDOWS -# ifdef _WIN32_WCE - { - wchar_t* wstr = fghWstrFromStr(title); - SetWindowText( fgStructure.CurrentWindow->Window.Handle, wstr ); - free(wstr); - } -# else - SetWindowText( fgStructure.CurrentWindow->Window.Handle, title ); -# endif - -#endif - } -} - -/* - * Set the current window's iconified title - */ -void FGAPIENTRY glutSetIconTitle( const char* title ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetIconTitle" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutSetIconTitle" ); - - if( ! fgStructure.CurrentWindow->Parent ) - { -#if TARGET_HOST_POSIX_X11 - - XTextProperty text; - - text.value = (unsigned char *) title; - text.encoding = XA_STRING; - text.format = 8; - text.nitems = strlen( title ); - - XSetWMIconName( - fgDisplay.Display, - fgStructure.CurrentWindow->Window.Handle, - &text - ); - - XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ - -#elif TARGET_HOST_MS_WINDOWS -# ifdef _WIN32_WCE - { - wchar_t* wstr = fghWstrFromStr(title); - SetWindowText( fgStructure.CurrentWindow->Window.Handle, wstr ); - free(wstr); - } -# else - SetWindowText( fgStructure.CurrentWindow->Window.Handle, title ); -# endif - -#endif - } -} - -/* - * Change the current window's size - */ -void FGAPIENTRY glutReshapeWindow( int width, int height ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutReshapeWindow" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutReshapeWindow" ); - - if (glutGet(GLUT_FULL_SCREEN)) - { - /* Leave full screen state before resizing. */ - glutLeaveFullScreen(); - } - - fgStructure.CurrentWindow->State.NeedToResize = GL_TRUE; - fgStructure.CurrentWindow->State.Width = width ; - fgStructure.CurrentWindow->State.Height = height; -} - -/* - * Change the current window's position - */ -void FGAPIENTRY glutPositionWindow( int x, int y ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPositionWindow" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutPositionWindow" ); - - if (glutGet(GLUT_FULL_SCREEN)) - { - /* Leave full screen state before moving. */ - glutLeaveFullScreen(); - } - -#if TARGET_HOST_POSIX_X11 - - XMoveWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle, - x, y ); - XFlush( fgDisplay.Display ); /* XXX Shouldn't need this */ - -#elif TARGET_HOST_MS_WINDOWS - - { - RECT winRect; - - /* "GetWindowRect" returns the pixel coordinates of the outside of the window */ - GetWindowRect( fgStructure.CurrentWindow->Window.Handle, &winRect ); - MoveWindow( - fgStructure.CurrentWindow->Window.Handle, - x, - y, - winRect.right - winRect.left, - winRect.bottom - winRect.top, - TRUE - ); - } - -#endif -} - -/* - * Lowers the current window (by Z order change) - */ -void FGAPIENTRY glutPushWindow( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPushWindow" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutPushWindow" ); - -#if TARGET_HOST_POSIX_X11 - - XLowerWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle ); - -#elif TARGET_HOST_MS_WINDOWS - - SetWindowPos( - fgStructure.CurrentWindow->Window.Handle, - HWND_BOTTOM, - 0, 0, 0, 0, - SWP_NOSIZE | SWP_NOMOVE - ); - -#endif -} - -/* - * Raises the current window (by Z order change) - */ -void FGAPIENTRY glutPopWindow( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutPopWindow" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutPopWindow" ); - -#if TARGET_HOST_POSIX_X11 - - XRaiseWindow( fgDisplay.Display, fgStructure.CurrentWindow->Window.Handle ); - -#elif TARGET_HOST_MS_WINDOWS - - SetWindowPos( - fgStructure.CurrentWindow->Window.Handle, - HWND_TOP, - 0, 0, 0, 0, - SWP_NOSIZE | SWP_NOMOVE - ); - -#endif -} - -/* - * Resize the current window so that it fits the whole screen - */ -void FGAPIENTRY glutFullScreen( void ) -{ - SFG_Window *win; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutFullScreen" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutFullScreen" ); - - win = fgStructure.CurrentWindow; - - if (win->Parent) - { - /* Child windows cannot be made fullscreen, consistent with GLUT's behavior - * Also, what would it mean for a child window to be fullscreen, given that it - * is confined to its parent? - */ - fgWarning("glutFullScreen called on a child window, ignoring..."); - return; - } - else if (fgStructure.GameModeWindow != NULL && fgStructure.GameModeWindow->ID==win->ID) - { - /* Ignore fullscreen call on GameMode window, those are always fullscreen already */ - return; - } - -#if TARGET_HOST_POSIX_X11 - if(!glutGet(GLUT_FULL_SCREEN)) { - if(fghToggleFullscreen() != -1) { - win->State.IsFullscreen = GL_TRUE; - } - } - -#elif TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) /* FIXME: what about WinCE */ - - if (glutGet(GLUT_FULL_SCREEN)) - { - /* Leave full screen state before entering fullscreen again (resizing?) */ - glutLeaveFullScreen(); - } - - { -#if(WINVER >= 0x0500) /* Windows 2000 or later */ - DWORD s; - RECT rect; - HMONITOR hMonitor; - MONITORINFO mi; - - /* For fullscreen mode, first remove all window decoration - * and set style to popup so it will overlap the taskbar - * then force to maximize on the screen on which it has the most - * overlap. - */ - - - /* store current window rect */ - GetWindowRect( win->Window.Handle, &win->State.OldRect ); - - /* store current window style */ - win->State.OldStyle = s = GetWindowLong(win->Window.Handle, GWL_STYLE); - - /* remove decorations from style and add popup style*/ - s &= ~WS_OVERLAPPEDWINDOW; - s |= WS_POPUP; - SetWindowLong(win->Window.Handle, GWL_STYLE, s); - SetWindowPos(win->Window.Handle, HWND_TOP, 0,0,0,0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_FRAMECHANGED); - - /* For fullscreen mode, find the monitor that is covered the most - * by the window and get its rect as the resize target. - */ - hMonitor= MonitorFromRect(&win->State.OldRect, MONITOR_DEFAULTTONEAREST); - mi.cbSize = sizeof(mi); - GetMonitorInfo(hMonitor, &mi); - rect = mi.rcMonitor; -#else /* if (WINVER >= 0x0500) */ - RECT rect; - - /* For fullscreen mode, force the top-left corner to 0,0 - * and adjust the window rectangle so that the client area - * covers the whole screen. - */ - - rect.left = 0; - rect.top = 0; - rect.right = fgDisplay.ScreenWidth; - rect.bottom = fgDisplay.ScreenHeight; - - AdjustWindowRect ( &rect, WS_OVERLAPPEDWINDOW | WS_CLIPSIBLINGS | - WS_CLIPCHILDREN, FALSE ); -#endif /* (WINVER >= 0x0500) */ - - /* - * then resize window - * SWP_NOACTIVATE Do not activate the window - * SWP_NOOWNERZORDER Do not change position in z-order - * SWP_NOSENDCHANGING Suppress WM_WINDOWPOSCHANGING message - * SWP_NOZORDER Retains the current Z order (ignore 2nd param) - */ - SetWindowPos( fgStructure.CurrentWindow->Window.Handle, - HWND_TOP, - rect.left, - rect.top, - rect.right - rect.left, - rect.bottom - rect.top, - SWP_NOACTIVATE | SWP_NOOWNERZORDER | SWP_NOSENDCHANGING | - SWP_NOZORDER - ); - - win->State.IsFullscreen = GL_TRUE; - } -#endif -} - -/* - * If we are fullscreen, resize the current window back to its original size - */ -void FGAPIENTRY glutLeaveFullScreen( void ) -{ - SFG_Window *win; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutFullScreen" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutFullScreen" ); - - win = fgStructure.CurrentWindow; - -#if TARGET_HOST_POSIX_X11 - if(glutGet(GLUT_FULL_SCREEN)) { - if(fghToggleFullscreen() != -1) { - win->State.IsFullscreen = GL_FALSE; - } - } - -#elif TARGET_HOST_MS_WINDOWS && !defined(_WIN32_WCE) /* FIXME: what about WinCE */ - if (!glutGet(GLUT_FULL_SCREEN)) - { - /* nothing to do */ - return; - } - - /* restore style of window before making it fullscreen */ - SetWindowLong(win->Window.Handle, GWL_STYLE, win->State.OldStyle); - SetWindowPos(win->Window.Handle, HWND_TOP, 0,0,0,0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_FRAMECHANGED); - - /* Then resize */ - SetWindowPos(win->Window.Handle, - HWND_TOP, - win->State.OldRect.left, - win->State.OldRect.top, - win->State.OldRect.right - win->State.OldRect.left, - win->State.OldRect.bottom - win->State.OldRect.top, - SWP_NOACTIVATE | SWP_NOOWNERZORDER | SWP_NOSENDCHANGING | - SWP_NOZORDER - ); - - win->State.IsFullscreen = GL_FALSE; -#endif -} - -/* - * Toggle the window's full screen state. - */ -void FGAPIENTRY glutFullScreenToggle( void ) -{ - SFG_Window *win; - - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutFullScreenToggle" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutFullScreenToggle" ); - - win = fgStructure.CurrentWindow; - -#if TARGET_HOST_POSIX_X11 - if(fghToggleFullscreen() != -1) { - win->State.IsFullscreen = !win->State.IsFullscreen; - } -#elif TARGET_HOST_MS_WINDOWS - if (!win->State.IsFullscreen) - glutFullScreen(); - else - glutLeaveFullScreen(); -#endif -} - -/* - * A.Donev: Set and retrieve the window's user data - */ -void* FGAPIENTRY glutGetWindowData( void ) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutGetWindowData" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutGetWindowData" ); - return fgStructure.CurrentWindow->UserData; -} - -void FGAPIENTRY glutSetWindowData(void* data) -{ - FREEGLUT_EXIT_IF_NOT_INITIALISED ( "glutSetWindowData" ); - FREEGLUT_EXIT_IF_NO_WINDOW ( "glutSetWindowData" ); - fgStructure.CurrentWindow->UserData = data; -} - -/*** END OF FILE ***/ diff --git a/examples/common/opengl-framework/freeglut/freeglut_xinput.c b/examples/common/opengl-framework/freeglut/freeglut_xinput.c deleted file mode 100644 index b0d8cc7a..00000000 --- a/examples/common/opengl-framework/freeglut/freeglut_xinput.c +++ /dev/null @@ -1,219 +0,0 @@ -/* Written for XI1 by Nikolas Doerfler (c) 2008 * - * Rewritten for XI2 by Florian Echtler (c) 2009 */ - -#include - -#include "freeglut_internal.h" - -#if TARGET_HOST_POSIX_X11 && HAVE_X11_EXTENSIONS_XINPUT2_H - -#include -#include - -#include -#include - -/* import function from freeglut_main.c */ -int fghGetXModifiers( int state ); - -/* extension opcode for XInput */ -int xi_opcode = -1; - -/** - * \brief Sets window up for XI2 events. - */ -void fgRegisterDevices( Display* dpy, Window* win ) { - - XIEventMask mask; - unsigned char flags[2] = { 0, 0 }; - int event, error; - - /*Display* dpy = fgDisplay.Display; - Window* win = glutGetXWindow();*/ - - /* get XInput extension opcode */ - if (!XQueryExtension( dpy, "XInputExtension", &xi_opcode, &event, &error )) { xi_opcode = -1; } - - /* Select for motion events */ - mask.deviceid = XIAllMasterDevices; - mask.mask_len = 2; - mask.mask = flags; - - XISetMask(mask.mask, XI_Enter); - XISetMask(mask.mask, XI_Motion); - XISetMask(mask.mask, XI_ButtonPress); - XISetMask(mask.mask, XI_ButtonRelease); - XISetMask(mask.mask, XI_Leave); - /*XISetMask(mask.mask, XI_KeyPress); - XISetMask(mask.mask, XI_KeyRelease); - XISetMask(mask.mask, XI_DeviceChanged); - XISetMask(mask.mask, XI_RawEvent); - XISetMask(mask.mask, XI_FocusIn); - XISetMask(mask.mask, XI_FocusOut); - XISetMask(mask.mask, XI_HierarchyChanged);*/ - - XISelectEvents( dpy, *win, &mask, 1 ); -} - - -void fgPrintXILeaveEvent(XILeaveEvent* event) -{ - char* mode = ""; - char* detail = ""; - int i; - - printf(" windows: root 0x%lx event 0x%lx child 0x%ld\n", - event->root, event->event, event->child); - switch(event->mode) - { - case NotifyNormal: mode = "NotifyNormal"; break; - case NotifyGrab: mode = "NotifyGrab"; break; - case NotifyUngrab: mode = "NotifyUngrab"; break; - case NotifyWhileGrabbed: mode = "NotifyWhileGrabbed"; break; - } - switch (event->detail) - { - case NotifyAncestor: detail = "NotifyAncestor"; break; - case NotifyVirtual: detail = "NotifyVirtual"; break; - case NotifyInferior: detail = "NotifyInferior"; break; - case NotifyNonlinear: detail = "NotifyNonlinear"; break; - case NotifyNonlinearVirtual: detail = "NotifyNonlinearVirtual"; break; - case NotifyPointer: detail = "NotifyPointer"; break; - case NotifyPointerRoot: detail = "NotifyPointerRoot"; break; - case NotifyDetailNone: detail = "NotifyDetailNone"; break; - } - printf(" mode: %s (detail %s)\n", mode, detail); - printf(" flags: %s %s\n", event->focus ? "[focus]" : "", - event->same_screen ? "[same screen]" : ""); - printf(" buttons:"); - for (i = 0; i < event->buttons.mask_len * 8; i++) - if (XIMaskIsSet(event->buttons.mask, i)) - printf(" %d", i); - printf("\n"); - - printf(" modifiers: locked 0x%x latched 0x%x base 0x%x\n", - event->mods.locked, event->mods.latched, - event->mods.base); - printf(" group: locked 0x%x latched 0x%x base 0x%x\n", - event->group.locked, event->group.latched, - event->group.base); - - printf(" root x/y: %.2f / %.2f\n", event->root_x, event->root_y); - printf(" event x/y: %.2f / %.2f\n", event->event_x, event->event_y); - -} - - -void fgPrintXIDeviceEvent(XIDeviceEvent* event) -{ - double *val; - int i; - - printf(" device: %d (%d)\n", event->deviceid, event->sourceid); - printf(" detail: %d\n", event->detail); - printf(" buttons:"); - for (i = 0; i < event->buttons.mask_len * 8; i++) - if (XIMaskIsSet(event->buttons.mask, i)) - printf(" %d", i); - printf("\n"); - - printf(" modifiers: locked 0x%x latched 0x%x base 0x%x\n", - event->mods.locked, event->mods.latched, - event->mods.base); - printf(" group: locked 0x%x latched 0x%x base 0x%x\n", - event->group.locked, event->group.latched, - event->group.base); - printf(" valuators:"); - - val = event->valuators.values; - for (i = 0; i < event->valuators.mask_len * 8; i++) - if (XIMaskIsSet(event->valuators.mask, i)) - printf(" %d: %.2f", i, *val++); - printf("\n"); - - printf(" windows: root 0x%lx event 0x%lx child 0x%ld\n", - event->root, event->event, event->child); - printf(" root x/y: %.2f / %.2f\n", event->root_x, event->root_y); - printf(" event x/y: %.2f / %.2f\n", event->event_x, event->event_y); - -} - - -/** - * \brief This function is called when an Extension Event is received - * and calls the corresponding callback functions for these events. - */ -void fgHandleExtensionEvents( XEvent* base_ev ) { - - int i, button = 0; - XGenericEventCookie* cookie = (XGenericEventCookie*)&(base_ev->xcookie); - - if ( XGetEventData( fgDisplay.Display, cookie ) && (cookie->type == GenericEvent) && (cookie->extension == xi_opcode) ) { - - XIDeviceEvent* event = (XIDeviceEvent*)(cookie->data); - /*printf("XI2 event type: %d - %d\n", cookie->evtype, event->type );*/ - - SFG_Window* window = fgWindowByHandle( event->event ); - if (!window) return; - - switch (cookie->evtype) { - - case XI_Enter: - case XI_Leave: - fgState.Modifiers = fghGetXModifiers( ((XIEnterEvent*)event)->mods.base ); - INVOKE_WCB( *window, MultiEntry, ( - event->deviceid, - (event->evtype == XI_Enter ? GLUT_ENTERED : GLUT_LEFT) - )); - #if _DEBUG - fgPrintXILeaveEvent((XILeaveEvent*)event); - #endif - break; - - case XI_ButtonPress: - case XI_ButtonRelease: - fgState.Modifiers = fghGetXModifiers( event->mods.base ); - INVOKE_WCB( *window, MultiButton, ( - event->deviceid, - event->event_x, - event->event_y, - (event->detail)-1, - (event->evtype == XI_ButtonPress ? GLUT_DOWN : GLUT_UP) - )); - INVOKE_WCB( *window, Mouse, ( - (event->detail)-1, - (event->evtype == XI_ButtonPress ? GLUT_DOWN : GLUT_UP), - event->event_x, - event->event_y - )); - break; - - case XI_Motion: - fgState.Modifiers = fghGetXModifiers( event->mods.base ); - for (i = 0; i < event->buttons.mask_len; i++) if (event->buttons.mask[i]) button = 1; - if (button) { - INVOKE_WCB( *window, MultiMotion, ( event->deviceid, event->event_x, event->event_y ) ); - INVOKE_WCB( *window, Motion, ( event->event_x, event->event_y ) ); - } else { - INVOKE_WCB( *window, MultiPassive, ( event->deviceid, event->event_x, event->event_y ) ); - INVOKE_WCB( *window, Passive, ( event->event_x, event->event_y ) ); - } - #if _DEBUG - fgPrintXIDeviceEvent(event); - #endif - break; - - default: - #if _DEBUG - fgWarning( "Unknown XI2 device event:" ); - fgPrintXIDeviceEvent( event ); - #endif - break; - } - fgState.Modifiers = INVALID_MODIFIERS; - } - XFreeEventData( fgDisplay.Display, cookie ); -} - -#endif - diff --git a/examples/common/opengl-framework/freeglut/glut.h b/examples/common/opengl-framework/freeglut/glut.h deleted file mode 100644 index 6191f77b..00000000 --- a/examples/common/opengl-framework/freeglut/glut.h +++ /dev/null @@ -1,21 +0,0 @@ -#ifndef __GLUT_H__ -#define __GLUT_H__ - -/* - * glut.h - * - * The freeglut library include file - * - * 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 - * PAWEL W. OLSZTA 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. - */ - -#include "freeglut_std.h" - -/*** END OF FILE ***/ - -#endif /* __GLUT_H__ */ diff --git a/examples/cubes/CMakeLists.txt b/examples/cubes/CMakeLists.txt index 90f64abb..f6b44f3a 100644 --- a/examples/cubes/CMakeLists.txt +++ b/examples/cubes/CMakeLists.txt @@ -8,13 +8,25 @@ PROJECT(Cubes) SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin/cubes/) # Copy the shaders used for the demo into the build directory -FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") +FILE(COPY "${OPENGLFRAMEWORK_DIR}/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") # Headers -INCLUDE_DIRECTORIES("../common/opengl-framework/src/" "../common/") +INCLUDE_DIRECTORIES("${OPENGLFRAMEWORK_DIR}/src/" "../common/") -# Create the example executable using the -# compiled reactphysics3d static library -ADD_EXECUTABLE(cubes Cubes.cpp Scene.cpp Scene.h "../common/Box.cpp" "../common/Box.h" "../common/Viewer.cpp" "../common/Viewer.h") +# Source files +SET(CUBES_SOURCES + Cubes.cpp + Scene.cpp + Scene.cpp + Scene.h + "../common/Box.cpp" + "../common/Box.h" + "../common/Viewer.cpp" + "../common/Viewer.h" +) +# Create the executable +ADD_EXECUTABLE(cubes ${CUBES_SOURCES}) + +# Link with libraries TARGET_LINK_LIBRARIES(cubes reactphysics3d openglframework) diff --git a/examples/joints/CMakeLists.txt b/examples/joints/CMakeLists.txt index 458a9731..9b4f71e6 100644 --- a/examples/joints/CMakeLists.txt +++ b/examples/joints/CMakeLists.txt @@ -8,13 +8,24 @@ PROJECT(Joints) SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin/joints/) # Copy the shaders used for the demo into the build directory -FILE(COPY "../common/opengl-framework/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") +FILE(COPY "${OPENGLFRAMEWORK_DIR}/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") # Headers -INCLUDE_DIRECTORIES("../common/opengl-framework/src/" "../common/") +INCLUDE_DIRECTORIES("${OPENGLFRAMEWORK_DIR}/src/" "../common/") -# Create the example executable using the -# compiled reactphysics3d static library -ADD_EXECUTABLE(joints Joints.cpp Scene.cpp Scene.h "../common/Box.cpp" "../common/Box.h" "../common/Viewer.cpp" "../common/Viewer.h") +# Source files +SET(JOINTS_SOURCES + Joints.cpp + Scene.cpp + Scene.h + "../common/Box.cpp" + "../common/Box.h" + "../common/Viewer.cpp" + "../common/Viewer.h" +) +# Create the executable +ADD_EXECUTABLE(joints ${JOINTS_SOURCES}) + +# Link with libraries TARGET_LINK_LIBRARIES(joints reactphysics3d openglframework) From 25a00a6b5a2529589bfaae5138b0837f63e25044 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 25 Sep 2013 00:02:09 +0200 Subject: [PATCH 60/66] Improve the user manual documentation --- .../UserManual/ReactPhysics3D-UserManual.tex | 679 +++++++++++++++--- documentation/UserManual/images/boxshape.png | Bin 0 -> 6623 bytes .../UserManual/images/capsuleshape.png | Bin 0 -> 7763 bytes documentation/UserManual/images/coneshape.png | Bin 0 -> 16565 bytes .../UserManual/images/convexshape.png | Bin 0 -> 21090 bytes .../UserManual/images/cylindershape.png | Bin 0 -> 7347 bytes .../UserManual/images/sphereshape.png | Bin 0 -> 10826 bytes 7 files changed, 573 insertions(+), 106 deletions(-) create mode 100644 documentation/UserManual/images/boxshape.png create mode 100644 documentation/UserManual/images/capsuleshape.png create mode 100644 documentation/UserManual/images/coneshape.png create mode 100644 documentation/UserManual/images/convexshape.png create mode 100644 documentation/UserManual/images/cylindershape.png create mode 100644 documentation/UserManual/images/sphereshape.png diff --git a/documentation/UserManual/ReactPhysics3D-UserManual.tex b/documentation/UserManual/ReactPhysics3D-UserManual.tex index 1a71aa4b..3ff12d20 100644 --- a/documentation/UserManual/ReactPhysics3D-UserManual.tex +++ b/documentation/UserManual/ReactPhysics3D-UserManual.tex @@ -9,9 +9,29 @@ \usepackage{graphicx} \usepackage{listings} \usepackage{url} -\usepackage{color} +\usepackage[usenames]{xcolor} \usepackage[top=3cm, bottom=3cm, left=3cm, right=3cm]{geometry} +% Path to the pictures +\graphicspath{ {images/} } + +% Customized style to display c++ source code +\lstdefinestyle{customcpp}{ + backgroundcolor=\color{gray!15}, + belowcaptionskip=1\baselineskip, + breaklines=true, + frame=L, + xleftmargin=\parindent, + language=C++, + showstringspaces=false, + basicstyle=\footnotesize\ttfamily, + keywordstyle=\bfseries\color{green!40!black}, + commentstyle=\itshape\color{purple!40!black}, + identifierstyle=\color{blue}, + stringstyle=\color{orange}, +} +\lstset{style=customcpp} + \input{title} \begin{document} @@ -36,156 +56,603 @@ \begin{itemize} \item Rigid body dynamics \item Discrete collision detection - \item Collision shapes (Sphere, Box, Cone, Cylinder) + \item Collision shapes (Sphere, Box, Cone, Cylinder, Capsule, Convex Mesh) \item Broadphase collision detection (Sweep and Prune using AABBs) \item Narrowphase collision detection (GJK/EPA) \item Collision response and friction (Sequential Impulses Solver) + \item Joints (Ball and Socket, Hinge, Slider, Fixed) + \item Sleeping technique for inactive bodies + \item Integrated Profiler \item Multi-platform (Windows, Linux, Mac OS X) \item Documentation (User manual and Doxygen API) + \item Examples \item Unit tests \end{itemize} - \section{What's new in this version} - - The current version is version 0.3.0 The following things have been - added or improved in this new version : \\ - - \begin{itemize} - \item The Sweep-and-Prune broad-phase collision detection - algorithm has been rewritten according to the technique - described by Pierre Terdiman at - \url{http://www.codercorner.com/SAP.pdf} to be much more efficient - than the previous naive implementation. - \item The contact solver has been rewritten to use the Sequential - Impulses technique from Erin Catto which is mathematically - equivalent to the Projected Gauss Seidel technique that was used - before. The Sequential Impulse technique is more - intuitive. - \item Implementation of a dedicated collision detection algorithm for spheres - against spheres instead of using GJK/EPA algorithm. - \item Make GJK/EPA algorithm more robust for spheres. - \item Make possible to use a unique instance of a collision shape - for multiple rigid bodies. - \item Change the structure of the code for a better separation - between the collision detection and the dynamics simulation code. - \item Create the API documentation using Doxygen - \item Add Unit tests - \end{itemize} - \section{License} - The ReactPhysics3D library is released under the ZLib license. For more information, read the "LICENSE" file. + The ReactPhysics3D library is released under the open-source ZLib license. For more information, read the "LICENSE" file. - \section{Compilation} - \label{sec:compilation} + \section{Building the library} + \label{sec:building} - You can use the CMake software to generate the makefiles or the + You should use the CMake software to generate the makefiles or the project files for your IDE. CMake can be downloaded at \url{http://www.cmake.org} or using you package-management program (apt, yum, \dots) on Linux. Then, you will be able to compile the library to create the static library - file. In order to use ReactPhysics3D in - your application, you can link your program with this static library. + file. In order to use ReactPhysics3D in your application, you can link your program with this static library. + If you have never runned cmake, you should read the page \url{http://www.cmake.org/cmake/help/runningcmake.html} as + it contains many useful information. - \subsection{Linux and Mac OS X} + \subsection{CMake using the command line (Linux and Mac OS X)} - To build the ReactPhysics3D library, create a folder into which - you want to build the library. Then go into that folder and run - the \texttt{cmake} command : \\ + Now, we will see how to build the ReactPhysics3D library using the CMake tool on the command line. + First, create a folder into which you want to build the library. Then go into that folder and run + the \texttt{ccmake} command : \\ - \texttt{cmake \textless path\_to\_library\textgreater \ -G "Unix Makefiles" -DCMAKE\_BUILD\_TYPE=Release} \\ + \texttt{ccmake \textless path\_to\_library\_source\textgreater} \\ - where \texttt{\textless path\_to\_library\textgreater} must be replaced - by the path to the library which is the path to the - \texttt{reactphysics3d-0.3.0/} folder. It is the folder that - contains the \texttt{CMakeLists.txt} file. This will generate the - makefiles into your build directory you have created before. Notice that the previous command generates the makefiles to - compile the library in Release mode. If you wan to generate the makefiles to - compile in Debug mode, you will have to use the following command : \\ + where \texttt{\textless path\_to\_library\_source\textgreater} must be replaced + by the path the path to the \texttt{reactphysics3d-0.4.0/} folder. It is the folder that + contains the \texttt{CMakeLists.txt} file. Running this command will launch the CMake command line interface. + Hit the 'c' key to configure the project. There, you can also change some predefined variables (see section \ref{sec:cmakevariables} for more details) + and then, hit the 'c' key again. Once you have set all the values as you like, you can hit the 'g' key to generate the makefiles in the build directory + that you have created before and exit. \\ - \texttt{cmake \textless path\_to\_library\textgreater \ -G "Unix Makefiles" -DCMAKE\_BUILD\_TYPE=Debug} \\ - - If you also want to build the examples at the same time, you can - use the following command : \\ - - \texttt{cmake \textless path\_to\_library\textgreater \ -G "Unix Makefiles" -DCMAKE\_BUILD\_TYPE=Release \\ - -DCOMPILE\_EXAMPLES=True } \\ - - In order to compile the examples, the GLUT or FREEGLUT library needs to be - installed on your system if you use Mac OS X. \\ - - If you want to build the unit tests, you can use the following - command : \\ - - \texttt{cmake \textless path\_to\_library\textgreater \ -G "Unix Makefiles" -DCMAKE\_BUILD\_TYPE=Debug \\ - -DCOMPILE\_TESTS=True } \\ - - Now that you have generated the makefiles with the CMake software, - you can compile the code to build the static library in the - \texttt{/lib} folder with the following command in your build directory : \\ + Now that you have generated the makefiles with the CMake software, you can compile the code to build the static library in the + \texttt{/lib} folder with the following command in your build directory : \\ - \texttt{make} \\ + \texttt{make} - This will also compile the examples if you have used the above - option to do it. + \subsection{CMake using the graphical interface (Linux, Mac OS X and Windows)} - \subsection{Windows} - - On Windows, you can also use the CMake software to generate the - files that you will be able to use in your favorite IDE to - compile the library and examples. \\ - - First, download the CMake - software for Windows and install it. Then, run the - \texttt{cmake-gui} program. The program will ask you for the - source folder which is the \texttt{reactphysics3d-0.3.0/} folder of + Here, we will see how to build the ReactPhysics3D library using the CMake graphical interface. + First, run the \texttt{cmake-gui} program. The program will ask you for the + source folder which is the \texttt{reactphysics3d-0.4.0/} folder of the library. You will also have to select a folder where you want to build the library and the examples. Select any empty folder that - is on your system. Then, you can click on - \texttt{Configure}. CMake will ask you to choose an IDE that is on - your system. For instance, you can select Visual Studio. Then you - can change the compilation options. For instance, if you want to - generate the files to compile the examples, you have to check the option - \texttt{COMPILE\_EXAMPLES}. If you want to compile the unit - tests, you need to check the option \texttt{COMPILE\_TESTS}. - Note that compiling the examples - requires the GLUT or FREEGLUT library to be installed on your system. Once - this is done, you can click on \texttt{Configure} again and - finally on \texttt{Generate}. \\ + is on your system. Then, you can click on \texttt{Configure}. CMake will ask you to choose an IDE that is on + your system. For instance, you can select Visual Studio, Qt Creator, XCode, ... Then you + can change the compilation options. See section \ref{sec:cmakevariables} to see what are the possible options. + Once this is done, you can click on \texttt{Configure} again and finally on \texttt{Generate}. \\ Now, if you go into the folder you have chosen to build the - library, you should be able to open the - Visual Studio Project that CMake has created and compile the - ReactPhysics3D library and the examples. + library, you should be able to open the project file that corresponds to your IDE and compile + the library. + + \subsection{CMake Variables} + \label{sec:cmakevariables} + + You can find bellow the different CMake variables that you can set before generating the makefiles. + + \begin{description} + \item[CMAKE\_BUILD\_TYPE] If this variable is set to DEBUG, the library will be compiled in debugging mode. + This mode should be used during development stage to know where things might crash. + In debugging mode, the library might run a bit slow due to all the debugging information + that are used. However, if this variable is set to RELEASE, no debugging information is stored + and therefore, it will run much faster. This mode must be used when you compile for the final + release of you application. + + \item[COMPILE\_EXAMPLES] If this variable is ON, the examples of the reactphysics3d library will be compiled. + Note that you will need to have the Freeglut library installed on your system if you use + Windows or Linux and you will need to have the Glut library on Mac OS X if you want to + run those examples. + + \item[COMPILE\_TESTS] If this variable is ON, the unit tests of the reactphysics3d library will be compiled. You will then + be able to launch the tests to make sure that they are running fine on your system. + + \item[PROFILING\_ENABLED] If this variable is ON, the integrated profiler will collect data will the application is running + and the profiling report will be display on the console at the end of the application (in the + destructor of the DynamicsWorld). This might be useful to see what part of the reactphysics3d + library takes time during its execution. This variable must be set to OFF when you compile + for the final release of your application. + + \item[DOUBLE\_PRECISION\_ENABLED] If this variable is ON, the reactphysics3d library will be compile with double floating point precision. + Otherwise, the library will be compile with single precision. + \end{description} + \section{Using ReactPhysics3D in your application} In order to use the library in your own application, first build the static library of ReactPhysics3d as described above to get the - static library file in the \texttt{/lib} folder. Then, in your code, you have to include + static library file in the \texttt{lib/} folder. Then, in your code, you have to include the ReactPhysics3D header file with the line : \\ - \texttt{\#include "reactphysics3d.h"} \\ + \begin{lstlisting} + // Include the ReactPhysics3D header file + #include "reactphysics3d.h" + \end{lstlisting} + + \vspace{0.6cm} Note that the \texttt{reactphysics3d.h} header file can be found in the - \texttt{/src} folder of the library. Do not forget to add the - \texttt{/src} folder in your include directories in order that the + \texttt{src/} folder of the library. Do not forget to add the + \texttt{src/} folder in your include directories in order that the \texttt{reactphysics3d.h} file is accessible in your code. \\ - Don't forget to also link your application with the ReactPhysics3D + Do not forget to also link your application with the ReactPhysics3D static library. \\ Then, you should be able to compile your application using the - ReactPhysics3D library. You should also take a look at the - examples and the API documentation to get a better idea of how to use the + ReactPhysics3D library. \\ + + All the classes of the library are available in the \texttt{reactphysics3d} namespace or its shorter alias + \texttt{rp3d}. Therefore, you need to include this namespace into your code with the following declaration : \\ + + \begin{lstlisting} + // Use the ReactPhysics3D namespace + using namespace reactphysics3d; + \end{lstlisting} + + \vspace{0.6cm} + + You should also take a look at the examples and the API documentation to get a better idea of how to use the ReactPhysics3D library. + \section{The Physics World} + + The physics world will contain the bodies that you create and simulate them across time. + + \subsection{Creating the Physics World} + + \subsection{Customizing the Physics World} + + \subsection{Updating the Physics World} + + \subsection{Destroying the Physics World} + + \section{Rigid Bodies} + + \subsection{Creating a Rigid Body} + + \subsection{Customizing a Rigid Body} + + \subsection{Updating a Rigid Body} + + \subsection{Destroying a Rigid Body} + + \section{Collision Shapes} + + When you create a rigid body, you need to specify a collision shape. This shape will be used to test collision between the body and its environment. + This section describes all the collision shapes available in the ReactPhysics3D library and how to use them. \\ + + Every collision shapes use a \emph{collision margin} which is a small distance around the shape that is used internally in the collision detection. + Some collision shapes have their collision margin integrated into the shape that you define and therefore you do not have to worry about it. + However, for some collision shapes, the collision margin is added around the shape that you define and therefore, you might have to compensate + for this small margin with the way you render the object. \\ + + Once you have created a collision shape object, you need to used it when you create a rigid body in the physics world using the + \texttt{DynamicsWorld::createRigidBody()} method. Note that during the rigid body creating, the collision shape object that you gave as a parameter + will be copied internally. Therefore, you can destroy the collision shape object right after the rigid body creation. + + \subsection{Box Shape} + + \begin{figure}[h] + \centering + \includegraphics{boxshape.png} + \label{fig:boxshape} + \end{figure} + + The class \texttt{BoxShape} class describes a box collision shape centered at the origin of the body local space. The box is aligned with the local x, y and z axis. + In order to create a box shape, you only need to specify the three half extents dimensions of the box in the three X, Y and Z directions. \\ + + For instance, if you want to create a box shape with dimensions of 4 meters, 6 meters and 10 meters along the X, Y and Z axis respectively, you need to use the following code : \\ + + \begin{lstlisting} + // Half extents of the box in the x, y and z directions + const rp3d::Vector3 halfExtents(2.0, 3.0, 5.0); + + // Create the box shape + const rp3d::BoxShape boxShape(halfExtents); + \end{lstlisting} + + \vspace{0.6cm} + + The \texttt{BoxShape} has a collision margin that is added to the box dimension you define. Therefore, the actual box shape will be a little bit larger that the one you define. + It is recommended that you use the default margin. In case, you really need to change the collision margin of your box shape (if the dimension of your box is small compared + to the default collision margin for instance), you can pass the length of the new collision margin (in meters) as a second parameter of the BoxShape constructor. \\ + + For instance, if you want to use a collision margin of 1 centimeter for your box shape, you can do it like this : \\ + + \begin{lstlisting} + // Create the box shape with a custom collision margin + const rp3d::BoxShape boxShape(halfExtents, 0.01); + \end{lstlisting} + + \subsection{Sphere Shape} + + \begin{figure}[h] + \centering + \includegraphics{sphereshape.png} + \label{fig:sphereshape} + \end{figure} + + The \texttt{SphereShape} class describes a sphere collision shape centered at the origin of the body local space. You only need to specify the radius of sphere to create it. \\ + + For instance, if you want to create a sphere shape with a radius of 2 meters, you need to use the following code : \\ + + \begin{lstlisting} + // Create the sphere shape with a radius of 2m + const rp3d::SphereShape sphereShape(2.0); + \end{lstlisting} + + \vspace{0.6cm} + + The collision margin of the \texttt{SphereShape} is integrated into the sphere you define. Therefore, you do not need to worry about it and you cannot change it. + + \subsection{Cone Shape} + + \begin{figure}[h] + \centering + \includegraphics{coneshape.png} + \label{fig:coneshape} + \end{figure} + + The \texttt{ConeShape} class describes a cone collision shape centered at the origin of the body local-space. The cone is aligned along the Y axis. + In order to create a cone shape, you need to give the radius of the base of the cone and the height of the cone (along the Y axis). \\ + + For instance, if you want to create a cone shape with a radius of 1 meter and the height of 3 meters, you need to use the following code : \\ + + \begin{lstlisting} + // Create the cone shape + const rp3d::ConeShape coneShape(1.0, 3.0); + \end{lstlisting} + + \vspace{0.6cm} + + The \texttt{ConeShape} has a collision margin that is added to the cone dimension that you define. Therefore, the actual cone shape will be a little bit larger that the one you define. + It is recommended that you use the default margin. In case, you really need to change the collision margin of your cone shape (if the dimension of your cone is small compared + to the default collision margin for instance), you can pass the length of the new collision margin (in meters) as a third parameter of the \texttt{ConeShape} constructor. \\ + + For instance, if you want to use a collision margin of 1 centimeter for your cone shape, you can do it like this : \\ + + \begin{lstlisting} + // Create the cone shape with a custom collision margin + const rp3d::ConeShape coneShape(1.0, 3.0, 0.01); + \end{lstlisting} + + \subsection{Cylinder Shape} + + \begin{figure}[h] + \centering + \includegraphics{cylindershape.png} + \label{fig:cylindershape} + \end{figure} + + The \texttt{CylinderShape} class describes a cylinder collision shape centered at the origin of the body local-space. The cylinder is aligned along the Y axis. + In order to create a cylinder shape, you need to specify the radius of the base and the height of the cylinder (along the Y axis). \\ + + For instance, if you want to create a cylinder shape with a radius of 1 meter and the height of 3 meters, you need to use the following code : \\ + + \begin{lstlisting} + // Create the cylinder shape + const rp3d::Cylinder cylinderShape(1.0, 3.0); + \end{lstlisting} + + \vspace{0.6cm} + + The \texttt{CylinderShape} has a collision margin that is added to the cylinder dimension that you define. Therefore, the actual cylinder shape will be a little bit larger that the one you define. + It is recommended that you use the default margin. In case, you really need to change the collision margin of your cylinder shape (if the dimension of your cylinder is small compared + to the default collision margin for instance), you can pass the length of the new collision margin (in meters) as a third parameter of the \texttt{CylinderShape} constructor. \\ + + For instance, if you want to use a collision margin of 1 centimeter for your cylinder shape, you can do it like this : \\ + + \begin{lstlisting} + // Create the cylinder shape with a custom collision margin + const rp3d::CylinderShape cylinderShape(1.0, 3.0, 0.01); + \end{lstlisting} + + \subsection{Capsule Shape} + + \begin{figure}[h] + \centering + \includegraphics{capsuleshape.png} + \label{fig:capsuleshape} + \end{figure} + + The \texttt{CapsuleShape} class describes a capsule collision shape around the Y axis and centered at the origin of the body local space. It is the convex hull of two + spheres. It can also be seen as an elongated sphere. In order to create it, you only need to specify the radius of the two spheres and the height of the + capsule (distance between the centers of the two spheres). \\ + + For instance, if you want to create a capsule shape with a radius of 1 meter and the height of 2 meters, you need to use the following code : \\ + + \begin{lstlisting} + // Create the capsule shape + const rp3d::CapsuleShape capsuleShape(1.0, 2.0); + \end{lstlisting} + + \vspace{0.6cm} + + As for the \texttt{SphereShape}, the collision margin of the \texttt{CapsuleShape} is integrated into the capsule you define. + Therefore, you do not need to worry about it and you cannot change it. + + \subsection{Convex Mesh Shape} + + \begin{figure}[h] + \centering + \includegraphics{convexshape.png} + \label{fig:convexshape} + \end{figure} + + The class \texttt{ConvexMeshShape} can be used to describe the shape of a convex mesh. In order to create a convex mesh shape, you need to supply the array with the coordinates of + the vertices of the mesh. The array is supposed to start with the three X, Y and Z coordinates of the first vertex, then the X, Y and Z coordinates of the second vertex and so on. + The first parameter of the \texttt{ConvexMeshShape} constructor is a pointer to the array of vertices coordinates, the second parameter is the number of vertices in the array and + the third parameter is the size (in bytes) of the data needed for a single vertex in the array (data used by all the three coordinates of a single vertex). + + \begin{lstlisting} + // Construct a convex mesh shape + rp3d::ConvexMeshShape shape(verticesArray, nbVertices, 3 * sizeof(float)); + \end{lstlisting} + + \vspace{0.6cm} + + You need to make sure that the mesh you provide is indeed convex and also that the its origin of its local-space is inside the mesh. \\ + + The collision detection test with a convex mesh shape runs in $O(n)$ where $n$ is the number of vertices in the mesh. Collision detection can become expensive if there are + too many vertices in the mesh. It is possible to speed up the collision detection by providing information about the edges of the convex mesh. If you provide edges information + about the convex mesh, the collision detection will run in almost constant time at the cost of a little extra memory to store the edges information. In order to provide the edges + information, you need to call the \texttt{ConvexMeshShape::addEdge()} method for each edge of the mesh. The first parameter is the index of the first vertex of the edge and the + second parameter is the index of the second vertex. Do not worry about calling this method multiple times for the same edge, the edge information will be added only + once. \\ + + For instance, the following code adds the edges information into a convex mesh shape : \\ + + \begin{lstlisting} + // Add the edges information of the mesh into the shape + for (unsigned int i=0; i(world->createJoint(jointInfo)); + \end{lstlisting} + + \vspace{0.6cm} + + \subsection{Hinge Joint} + + The class \texttt{HingeJoint} describes a hinge joint (or revolute joint) between two rigid bodies. The hinge joint only allows rotation around an anchor point and + around a single axis (the hinge axis). This joint can be used to simulate doors or pendulums for instance. \\ + + In order to create a hinge joint, you first need to create a \texttt{HingeJointInfo} object with the necessary information. You need to provide the pointers to the + two rigid bodies, the coordinates of the anchor point (in world-space) and also the hinge rotation axis (in world-space). The two bodies need to be in a correct position + when the joint is created. \\ + + Here is the code to create the \texttt{HingeJointInfo} object : \\ + + \begin{lstlisting} + // Anchor point in world-space + const rp3d::Vector3 anchorPoint(2.0, 4.0, 0.0); + + // Hinge rotation axis in world-space + const rp3d::Vector3 axis(0.0, 0.0, 1.0); + + // Create the joint info object + rp3d::HingeJointInfo jointInfo(body1, body2, anchorPoint, axis); + \end{lstlisting} + + \vspace{0.6cm} + + Now, it is time to create the actual joint in the dynamics world using the \texttt{DynamicsWorld::createJoint()} method. + Note that this method will also return a pointer to the \texttt{HingeJoint} object that has been created internally. You will then + be able to use that pointer to change properties of the joint and also to destroy it at the end. \\ + + Here is how to create the joint in the world : \\ + + \begin{lstlisting} + // Create the hinge joint in the dynamics world + rp3d::HingeJoint* joint; + joint = dynamic_cast(world->createJoint(jointInfo)); + \end{lstlisting} + + \vspace{0.6cm} + + \subsubsection{Limits} + + With the hinge joint, you can constraint the motion range using limits. The limits of the hinge joint are the minimum and maximum angle of rotation allowed with respect to the initial + angle between the bodies when the joint is created. The limits are disabled by default. If you want to use the limits, you first need to enable them by setting the + \texttt{isLimitEnabled} variable of the \texttt{HingeJointInfo} object to \emph{true} before you create the joint. You also have to specify the minimum and maximum limit + angles (in radians) using the \texttt{minAngleLimit} and \texttt{maxAngleLimit} variables of the joint info object. Note that the minimum limit angle must be in the + range $[ -2 \pi; 0 ]$ and the maximum limit angle must be in the range $[ 0; 2 \pi ]$. \\ + + For instance, here is the way to use the limits for a hinge joint when the joint is created : \\ + + \begin{lstlisting} + // Create the joint info object + rp3d::HingeJointInfo jointInfo(body1, body2, anchorPoint, axis); + + // Enable the limits of the joint + jointInfo.isLimitEnabled = true; + + // Minimum limit angle + jointInfo.minAngleLimit = -PI / 2.0; + + // Maximum limit angle + jointInfo.maxAngleLimit = PI / 2.0; + + // Create the hinge joint in the dynamics world + rp3d::HingeJoint* joint; + joint = dynamic_cast(world->createJoint(jointInfo)); + \end{lstlisting} + + \vspace{0.6cm} + + You can also use the \texttt{HingeJoint::enableLimit()}, \texttt{HingeJoint::setMinAngleLimit()} and \texttt{HingeJoint::setMaxAngleLimit()} to specify the limits of the joint after its + creation. See the API documentation for more information. + + \subsubsection{Motor} + + A motor is also available for the hinge joint. It can be used rotate the bodies around the hinge axis at a given angular speed and such that the torque applied to + rotate the bodies does not exceed the maximum allowed torque. The motor is disabled by default. If you want to use it, you first have to activate it using the + \texttt{isMotorEnabled} boolean variable of the \texttt{HingeJointInfo} object before you create the joint. Then, you need to specify the angular motor speed (in radians/seconds) + using the \texttt{motorSpeed} variable and also the maximum allowed torque (in Newton $\cdot$ meters) with the \texttt{maxMotorTorque} variable. \\ + + For instance, here is how to enable the motor of the hinge joint when the joint is created : \\ + + \begin{lstlisting} + // Create the joint info object + rp3d::HingeJointInfo jointInfo(body1, body2, anchorPoint, axis); + + // Enable the motor of the joint + jointInfo.isMotorEnabled = true; + + // Motor angular speed + jointInfo.motorSpeed = PI / 4.0; + + // Maximum allowed torque + jointInfo.maxMotorTorque = 10.0; + + // Create the hinge joint in the dynamics world + rp3d::HingeJoint* joint; + joint = dynamic_cast(world->createJoint(jointInfo)); + \end{lstlisting} + + \vspace{0.6cm} + + You can also use the \texttt{HingeJoint::enableMotor()}, \texttt{HingeJoint::setMotorSpeed()} and \texttt{HingeJoint::setMaxMotorTorque()} to enabled the motor of the joint after its + creation. See the API documentation for more information. + + \subsection{Slider Joint} + + \subsection{Fixed Joint} + + \subsection{Collision between the bodies of a Joint} + + By default the two bodies involved in a joint are able to collide with each other. However, it is possible to disable the collision between the two bodies that are part + of the joint. To do it, you simply need to set the variable \texttt{isCollisionEnabled} of the joint info object to \emph{false} when you create the joint. \\ + + For instance, when you create a \texttt{HingeJointInfo} object in order to construct a hinge joint, you can disable collision between the two bodies of the joint as in the + following example : \\ + + \begin{lstlisting} + // Create the joint info object + rp3d::HingeJointInfo jointInfo(body1, body2, anchorPoint, axis); + + // Disable the collision between the bodies + jointInfo.isCollisionEnabled = false; + + // Create the joint in the dynamics world + rp3d::HingeJoint* joint; + joint = dynamic_cast(world->createJoint(jointInfo)); + \end{lstlisting} + + \subsection{Destroying a Joint} + + In order to destroy a joint, you simply need to call the \texttt{DynamicsWorld::destroyJoint()} method using the pointer to + a previously created joint object as argument as shown in the following code : \\ + + \begin{lstlisting} + // rp3d::BallAndSocketJoint* joint is a previously + // created joint + + // Destroy the joint + world->destroyJoint(joint); + \end{lstlisting} + + \vspace{0.6cm} + + It is important that you destroy all the joints that you have created at the end of the simulation. Also note that destroying a + rigid body that is involved in a joint will automatically destroy that joint. + \section{Examples} - You can find some OpenGL demos in the \texttt{/examples} folder of - the library. Follow the instructions described in section - \ref{sec:compilation} to - compile the examples. Note that the GLUT or FREEGLUT library is required to - compile and run the examples. + You can find some demos in the \texttt{examples/} folder of + the reactphysics3d library. Follow the instructions described in section \ref{sec:building} to + compile the examples. Note that the FREEGLUT library is required on Linux and Windows + and the GLUT library is required on Mac OS X to run those examples. Studying the examples is a + good way to understand how to use the reactphysics3d library. + + \subsection{Cubes} + + In this examples, you will see how to create a floor and some cubes using the Box Shape for collision detection. Because of gravity, + the cubes will fall down on the floor. After falling down, the cubes will come to rest and start sleeping (become inactive). In this demo, + the cubes are green when they are inactive and become red as they get inactive (sleeping). + + \subsection{Collision Shapes} + + In this example, you will see how to create a floor (using the Box Shape) and some other bodies using the different collision shapes available + in the reactphysics3d library like Cylinders, Capsules, Spheres, Convex Meshes and Cones. Those bodies will fall down to the floor. + + \subsection{Joints} + + In this example, you will learn how to create different joints (Ball and Socket, Hinge, Slider, Fixed) into the dynamics world. You can also see how + to set the motor or limits of the joints. + + \section{Profiler} \section{API Documentation} diff --git a/documentation/UserManual/images/boxshape.png b/documentation/UserManual/images/boxshape.png new file mode 100644 index 0000000000000000000000000000000000000000..0889b34c1e8023290a145d07cb87166e90d3af0a GIT binary patch literal 6623 zcmV<586f6~P)|00009a7bBm000id z000id0mpBsWB>pekx4{BRCwC$oqLd6Rh`Gb=f3-P&t#GqLV!SkfP%2Fy8(ekdAO^T zf-DwVYPni~wW2Ev)ULQbR$W(IMRt7zajh*Z6(OPpDk3ft@Uam}StKt5aRUL85c1As zX8LhoXaD&1J#*6Y?3wPKzI}h+KT@}EXJ)!jpHKfD=XZWbNGZW$u`(`$hk(Uml36Sk zlgwhVm}C}<#U!&>Oma_al3YHg`B_~ni-&*^6{3DE1U_>gw(o2FKTGE0U#{e#Vs)tm zuP2l4T7#>;0mlM>a%~**D+GMEemydb+N%109{1~j#bT1%YjOen=8vF9VT=nu1Qb*6 z#GCf00diP-9OC0w;Z0|b-Z<}MhMdJDw?W@{1iyX;06Yudd`Nr3YtbJFfq3-M)o(;K zs$UOPaQ{Ct*eoV_rsnteTk+>N!S(q_+1(G~TeS(U*7KCHRkbGuSL9dA!-b zW;3cG02CaY|3RPt05Ac+`(Ang)J7j!16cDZ3{~q7;}~x|gE?pA$tM47E&g*g3`O%=@V|YzLS8O<|f2=`^5;NdKmxDSKu|v8EjThkgU($10z`Qt~5v0;K%mN{GRwt zz2S5`^iigol_`>cy9K)j0ieHti~b>9`4$i*h}=qg?)^CJ4DP>Xx5<8V1t5!2A-vcimzCH<2(HY$mz2 z#F7mmSti7rfl}&Ba{XwlT=YbHSb@BMbj%F_!{G3nNuH@TZo|fHfIy#zOFuhmQbZrP z(5$^5Ga!eQb;dg`!C$W!9rMNWaM7teNKEn=>5g1A#JQhDp9dkZc_()5p=kx!J(Q!$ zoZ`bQw&&xV<@J#JU>WPbpj&qSka9qBgG*Kokw^gW9f-@)@+raFG_v4|)7PtzfIvvj zd4f>lFXW->9g-U|pUZEor%Cq=;l-Wlgs{@Y6)Q6sOIQL4Uw$>kb?7yt%*h_`9MmA))w^Dt(FTm4D^K?uvn<3E8+ASDd59xNHW zevny`^@z6;;P!{%Sg=j3d5uF(ocdHkVB51W48Xv%zk^}Iv|*S$PQ5@f zdCp5ei2ecqRAW51BZbyrhdKK(umP9?PppP%0RrWN^{;KGF!k>Vk|^RO!iN?x2InTpG*#9zQj2Ab+upp*qdnZuH202?cdDQBP zG_56p1=@5Wi(LiwX#bN9^_+`VfGvUDyRdBogn>9jR0G6p0A^*9j4l0 z+f$es>6r(>fRHc*Oo9J>q~5$YNcSi+z^t((AK8Z)?nn~j{XPpg36MYx>_}nO3-{R% zA0Oa!W#0fsh9Cumf!{v>(}HQ&pW^+8tR~{**(*s6^06tp6`%?TU|5T>jeJwByU->e z1)kebFSpwEhkEGi`aDxvb4T*n(ArJC_fmS(0AA1xwUIM`rM3`It782lFl~fs>rLC^ z3Ce1kWGT_!{r~?CVSuoTY89J)S+BmbY(xP&&#YOI zd~`p~KLu;Hc9tBW8Dt48O|J*jmr(_V(*J2>8DrWuCcL|v=2hC^UP`` z$^9PAIvQ)X&fRO#^WpvJMG06402of!x{#@&6so?GfF-c;F_Z@(B@)%djFZo-HpI!@ zSePeL!u!*;E&;p@L=;s@H6=|IBJD$=xfO`KpZc zi1g-gdXw-LbL58;!?f^g2|V)@ARr9vdKM5cOjr(2HCE?Hj+2Z~t&-t-fT=QPfwqDz z+Dxyi+(-Z+@Ys)FIit?A$<8yY<0MOouf7F$ufx7_Mqay^;GqjFPm$h}`scM)6s5?+ z&Hny9Fa)H54G-6$wH(NVooCiul1$UDN6uhP(l>WhdZ(I()(5ns63r$Vwt)l)LOk~r zObff8ugAS^k!KQ%1BR1=4v zoU|lGx(rwXk3NpDg7P5<0U==X8FQ>mk^I`@)An6?lS&ZyfDIhj1>36o(kxcji?
u7j9IhQ;DI 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 dELQjH{{c_^h)Ds00009a7bBm000id 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 NKO!>1)q;8f{|6Y7BEbLv literal 0 HcmV?d00001 diff --git a/documentation/UserManual/images/convexshape.png b/documentation/UserManual/images/convexshape.png new file mode 100644 index 0000000000000000000000000000000000000000..4451da8355f7eee76d43c3b4e789f6adfbef9b7d GIT binary patch literal 21090 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 literal 0 HcmV?d00001 diff --git a/documentation/UserManual/images/cylindershape.png b/documentation/UserManual/images/cylindershape.png new file mode 100644 index 0000000000000000000000000000000000000000..6cfdf03ace96a309207825452cc657047fa5dc7e GIT binary patch literal 7347 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 Z{{s;3%X4eP(Z&D(002ovPDHLkV1nzCqi+BJ literal 0 HcmV?d00001 diff --git a/documentation/UserManual/images/sphereshape.png b/documentation/UserManual/images/sphereshape.png new file mode 100644 index 0000000000000000000000000000000000000000..cbdb3278e5ada63bcbc48cb25ac6246460271adc GIT binary patch literal 10826 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 Ue;~15tpET307*qoM6N<$g8r1bjsO4v literal 0 HcmV?d00001 From 97b0903a9dfa5c61291f2dd2344c7b198274dadd Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 25 Sep 2013 00:11:23 +0200 Subject: [PATCH 61/66] Modify comments --- src/collision/shapes/CylinderShape.h | 2 +- src/constraint/HingeJoint.h | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/collision/shapes/CylinderShape.h b/src/collision/shapes/CylinderShape.h index 748bc4f4..21cda294 100644 --- a/src/collision/shapes/CylinderShape.h +++ b/src/collision/shapes/CylinderShape.h @@ -57,7 +57,7 @@ class CylinderShape : public CollisionShape { /// Radius of the base decimal mRadius; - /// Half height of the cone + /// Half height of the cylinder decimal mHalfHeight; // -------------------- Methods -------------------- // diff --git a/src/constraint/HingeJoint.h b/src/constraint/HingeJoint.h index faf4c2e4..f9505992 100644 --- a/src/constraint/HingeJoint.h +++ b/src/constraint/HingeJoint.h @@ -49,10 +49,10 @@ struct HingeJointInfo : public JointInfo { /// Hinge rotation axis (in world-space coordinates) Vector3 rotationAxisWorld; - /// True if the slider limits are enabled + /// True if the hinge joint limits are enabled bool isLimitEnabled; - /// True if the slider motor is enabled + /// True if the hinge joint motor is enabled bool isMotorEnabled; /// Minimum allowed rotation angle (in radian) if limits are enabled. From b5ae655b0aa1c0f59104b1672b2662c7c87f1b1b Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Fri, 27 Sep 2013 18:43:45 +0200 Subject: [PATCH 62/66] Add comments, modify method names, change solver nb of iterations, ... --- examples/CMakeLists.txt | 5 ++ examples/collisionshapes/CollisionShapes.cpp | 1 - examples/collisionshapes/Scene.cpp | 14 ++-- examples/common/Box.cpp | 2 +- .../common/opengl-framework/CMakeLists.txt | 27 +++---- examples/cubes/Cubes.cpp | 1 - examples/cubes/Scene.cpp | 4 +- examples/joints/Joints.cpp | 1 - examples/joints/Scene.cpp | 16 ++-- src/body/Body.h | 16 ---- src/body/CollisionBody.h | 57 +++++-------- src/body/RigidBody.h | 6 +- src/collision/CollisionDetection.cpp | 2 +- src/collision/CollisionDetection.h | 2 - src/configuration.h | 2 +- src/constraint/BallAndSocketJoint.cpp | 32 ++++---- src/constraint/BallAndSocketJoint.h | 23 +++--- src/constraint/FixedJoint.cpp | 52 ++++++------ src/constraint/FixedJoint.h | 20 ++--- src/constraint/HingeJoint.cpp | 80 +++++++++---------- src/constraint/HingeJoint.h | 33 ++++---- src/constraint/Joint.h | 37 ++++----- src/constraint/SliderJoint.cpp | 76 +++++++++--------- src/constraint/SliderJoint.h | 34 ++++---- src/engine/ContactSolver.cpp | 4 +- src/engine/DynamicsWorld.cpp | 34 ++++---- src/engine/DynamicsWorld.h | 14 +--- src/engine/Island.cpp | 4 +- src/engine/Island.h | 8 +- src/reactphysics3d.h | 2 +- 30 files changed, 282 insertions(+), 327 deletions(-) diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt index edc71428..450ceb1e 100644 --- a/examples/CMakeLists.txt +++ b/examples/CMakeLists.txt @@ -4,6 +4,11 @@ CMAKE_MINIMUM_REQUIRED(VERSION 2.6) # Set a variable for the directory of the opengl-framework SET(OPENGLFRAMEWORK_DIR "${CMAKE_CURRENT_SOURCE_DIR}/common/opengl-framework") +# If we will use FREEGLUT +IF(NOT APPLE) + ADD_DEFINITIONS(-DUSE_FREEGLUT) +ENDIF() + ADD_SUBDIRECTORY(common/) ADD_SUBDIRECTORY(cubes/) ADD_SUBDIRECTORY(joints/) diff --git a/examples/collisionshapes/CollisionShapes.cpp b/examples/collisionshapes/CollisionShapes.cpp index cf1f9fdc..6f12deeb 100644 --- a/examples/collisionshapes/CollisionShapes.cpp +++ b/examples/collisionshapes/CollisionShapes.cpp @@ -120,7 +120,6 @@ void keyboard(unsigned char key, int x, int y) { // Escape key case 27: #ifdef USE_FREEGLUT - // TODO : Check if we need to call finish() here glutLeaveMainLoop(); #endif break; diff --git a/examples/collisionshapes/Scene.cpp b/examples/collisionshapes/Scene.cpp index 923e4206..360dba9e 100644 --- a/examples/collisionshapes/Scene.cpp +++ b/examples/collisionshapes/Scene.cpp @@ -74,7 +74,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), Box* box = new Box(BOX_SIZE, position , BOX_MASS, mDynamicsWorld); // The sphere is a moving rigid body - box->getRigidBody()->setIsMotionEnabled(true); + box->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material = box->getRigidBody()->getMaterial(); @@ -97,7 +97,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), Sphere* sphere = new Sphere(SPHERE_RADIUS, position , BOX_MASS, mDynamicsWorld); // The sphere is a moving rigid body - sphere->getRigidBody()->setIsMotionEnabled(true); + sphere->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material = sphere->getRigidBody()->getMaterial(); @@ -120,7 +120,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), Cone* cone = new Cone(CONE_RADIUS, CONE_HEIGHT, position , CONE_MASS, mDynamicsWorld); // The cone is a moving rigid body - cone->getRigidBody()->setIsMotionEnabled(true); + cone->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material = cone->getRigidBody()->getMaterial(); @@ -144,7 +144,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), CYLINDER_MASS, mDynamicsWorld); // The cylinder is a moving rigid body - cylinder->getRigidBody()->setIsMotionEnabled(true); + cylinder->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material = cylinder->getRigidBody()->getMaterial(); @@ -168,7 +168,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), CAPSULE_MASS, mDynamicsWorld); // The cylinder is a moving rigid body - capsule->getRigidBody()->setIsMotionEnabled(true); + capsule->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material = capsule->getRigidBody()->getMaterial(); @@ -191,7 +191,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), ConvexMesh* mesh = new ConvexMesh(position, MESH_MASS, mDynamicsWorld); // The mesh is a moving rigid body - mesh->getRigidBody()->setIsMotionEnabled(true); + mesh->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material = mesh->getRigidBody()->getMaterial(); @@ -206,7 +206,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), mFloor = new Box(FLOOR_SIZE, floorPosition, FLOOR_MASS, mDynamicsWorld); // The floor must be a non-moving rigid body - mFloor->getRigidBody()->setIsMotionEnabled(false); + mFloor->getRigidBody()->enableMotion(false); // Change the material properties of the rigid body rp3d::Material& material = mFloor->getRigidBody()->getMaterial(); diff --git a/examples/common/Box.cpp b/examples/common/Box.cpp index a148921b..e5a2bab1 100644 --- a/examples/common/Box.cpp +++ b/examples/common/Box.cpp @@ -121,7 +121,7 @@ void Box::render(openglframework::Shader& shader, localToCameraMatrix.getUpperLeft3x3Matrix().getInverse().getTranspose(); shader.setMatrix3x3Uniform("normalMatrix", normalMatrix); - // TODO : REMOVE THIS + // Set the vertex color openglframework::Vector4 color(mColor.r, mColor.g, mColor.b, mColor.a); shader.setVector4Uniform("vertexColor", color); diff --git a/examples/common/opengl-framework/CMakeLists.txt b/examples/common/opengl-framework/CMakeLists.txt index ab8192db..e8e3c5f1 100644 --- a/examples/common/opengl-framework/CMakeLists.txt +++ b/examples/common/opengl-framework/CMakeLists.txt @@ -28,24 +28,23 @@ ENDIF() # Find the GLUT/FREEGLUT library IF(APPLE) - - # Find the GLUT library - FIND_PACKAGE(GLUT REQUIRED) - IF(GLUT_FOUND) - MESSAGE("GLUT found") - ELSE(GLUT_FOUND) - MESSAGE(SEND_ERROR "GLUT not found") + + # Find the GLUT library + FIND_PACKAGE(GLUT REQUIRED) + IF(GLUT_FOUND) + MESSAGE("GLUT found") + ELSE(GLUT_FOUND) + MESSAGE(SEND_ERROR "GLUT not found") ENDIF(GLUT_FOUND) ELSE(APPLE) - # Find the FREEGLUT library - FIND_PACKAGE(FREEGLUT REQUIRED) - IF(FREEGLUT_FOUND) - MESSAGE("FREEGLUT found") - ADD_DEFINITIONS(-DUSE_FREEGLUT) - ELSE(FREEGLUT_FOUND) - MESSAGE(SEND_ERROR "FREEGLUT not found") + # Find the FREEGLUT library + FIND_PACKAGE(FREEGLUT REQUIRED) + IF(FREEGLUT_FOUND) + MESSAGE("FREEGLUT found") + ELSE(FREEGLUT_FOUND) + MESSAGE(SEND_ERROR "FREEGLUT not found") ENDIF(FREEGLUT_FOUND) ENDIF(APPLE) diff --git a/examples/cubes/Cubes.cpp b/examples/cubes/Cubes.cpp index 80c5d501..bd148434 100644 --- a/examples/cubes/Cubes.cpp +++ b/examples/cubes/Cubes.cpp @@ -120,7 +120,6 @@ void keyboard(unsigned char key, int x, int y) { // Escape key case 27: #ifdef USE_FREEGLUT - // TODO : Check if we need to call finish() here glutLeaveMainLoop(); #endif break; diff --git a/examples/cubes/Scene.cpp b/examples/cubes/Scene.cpp index e3bc109a..aa417739 100644 --- a/examples/cubes/Scene.cpp +++ b/examples/cubes/Scene.cpp @@ -70,7 +70,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), // Create a cube and a corresponding rigid in the dynamics world Box* cube = new Box(BOX_SIZE, position , BOX_MASS, mDynamicsWorld); - cube->getRigidBody()->setIsMotionEnabled(true); + cube->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material = cube->getRigidBody()->getMaterial(); @@ -85,7 +85,7 @@ Scene::Scene(GlutViewer* viewer) : mViewer(viewer), mLight0(0), mFloor = new Box(FLOOR_SIZE, floorPosition, FLOOR_MASS, mDynamicsWorld); // The floor must be a non-moving rigid body - mFloor->getRigidBody()->setIsMotionEnabled(false); + mFloor->getRigidBody()->enableMotion(false); // Change the material properties of the floor rigid body rp3d::Material& material = mFloor->getRigidBody()->getMaterial(); diff --git a/examples/joints/Joints.cpp b/examples/joints/Joints.cpp index 75c6892a..abb58d8a 100644 --- a/examples/joints/Joints.cpp +++ b/examples/joints/Joints.cpp @@ -119,7 +119,6 @@ void keyboard(unsigned char key, int x, int y) { // Escape key case 27: #ifdef USE_FREEGLUT - // TODO : Check if we need to call finish() here glutLeaveMainLoop(); #endif break; diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index 5a566f17..a651334d 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -205,8 +205,8 @@ void Scene::createBallAndSocketJoints() { mDynamicsWorld); // The fist box cannot move - if (i == 0) mBallAndSocketJointChainBoxes[i]->getRigidBody()->setIsMotionEnabled(false); - else mBallAndSocketJointChainBoxes[i]->getRigidBody()->setIsMotionEnabled(true); + if (i == 0) mBallAndSocketJointChainBoxes[i]->getRigidBody()->enableMotion(false); + else mBallAndSocketJointChainBoxes[i]->getRigidBody()->enableMotion(true); // Add some angular velocity damping mBallAndSocketJointChainBoxes[i]->getRigidBody()->setAngularDamping(rp3d::decimal(0.2)); @@ -249,7 +249,7 @@ void Scene::createSliderJoint() { mSliderJointBottomBox = new Box(box1Dimension, positionBox1 , BOX_MASS, mDynamicsWorld); // The fist box cannot move - mSliderJointBottomBox->getRigidBody()->setIsMotionEnabled(false); + mSliderJointBottomBox->getRigidBody()->enableMotion(false); // Change the material properties of the rigid body rp3d::Material& material1 = mSliderJointBottomBox->getRigidBody()->getMaterial(); @@ -265,7 +265,7 @@ void Scene::createSliderJoint() { mSliderJointTopBox = new Box(box2Dimension, positionBox2 , BOX_MASS, mDynamicsWorld); // The second box is allowed to move - mSliderJointTopBox->getRigidBody()->setIsMotionEnabled(true); + mSliderJointTopBox->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material2 = mSliderJointTopBox->getRigidBody()->getMaterial(); @@ -304,7 +304,7 @@ void Scene::createPropellerHingeJoint() { mPropellerBox = new Box(boxDimension, positionBox1 , BOX_MASS, mDynamicsWorld); // The fist box cannot move - mPropellerBox->getRigidBody()->setIsMotionEnabled(true); + mPropellerBox->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material = mPropellerBox->getRigidBody()->getMaterial(); @@ -342,7 +342,7 @@ void Scene::createFixedJoints() { mFixedJointBox1 = new Box(boxDimension, positionBox1 , BOX_MASS, mDynamicsWorld); // The fist box cannot move - mFixedJointBox1->getRigidBody()->setIsMotionEnabled(true); + mFixedJointBox1->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material1 = mFixedJointBox1->getRigidBody()->getMaterial(); @@ -357,7 +357,7 @@ void Scene::createFixedJoints() { mFixedJointBox2 = new Box(boxDimension, positionBox2 , BOX_MASS, mDynamicsWorld); // The second box is allowed to move - mFixedJointBox2->getRigidBody()->setIsMotionEnabled(true); + mFixedJointBox2->getRigidBody()->enableMotion(true); // Change the material properties of the rigid body rp3d::Material& material2 = mFixedJointBox2->getRigidBody()->getMaterial(); @@ -396,7 +396,7 @@ void Scene::createFloor() { mFloor = new Box(FLOOR_SIZE, floorPosition, FLOOR_MASS, mDynamicsWorld); // The floor must be a non-moving rigid body - mFloor->getRigidBody()->setIsMotionEnabled(false); + mFloor->getRigidBody()->enableMotion(false); // Change the material properties of the rigid body rp3d::Material& material = mFloor->getRigidBody()->getMaterial(); diff --git a/src/body/Body.h b/src/body/Body.h index af8ad635..a7351cd0 100644 --- a/src/body/Body.h +++ b/src/body/Body.h @@ -83,12 +83,6 @@ class Body { /// Return the id of the body bodyindex getID() const; - /// Return true if the body has already been added in an island (for the sleeping technique) - bool isAlreadyInIsland() const; - - /// Set the value of to know if the body has already been added into an island - void setIsAlreadyInIsland(bool isAlreadyInIsland); - /// Return whether or not the body is allowed to sleep bool isAllowedToSleep() const; @@ -126,16 +120,6 @@ inline bodyindex Body::getID() const { return mID; } -// Return true if the body has already been added in an island (for the sleeping technique) -inline bool Body::isAlreadyInIsland() const { - return mIsAlreadyInIsland; -} - -// Set the value of to know if the body has already been added into an island -inline void Body::setIsAlreadyInIsland(bool isAlreadyInIsland) { - mIsAlreadyInIsland = isAlreadyInIsland; -} - // Return whether or not the body is allowed to sleep inline bool Body::isAllowedToSleep() const { return mIsAllowedToSleep; diff --git a/src/body/CollisionBody.h b/src/body/CollisionBody.h index 399bd971..0cd9df84 100644 --- a/src/body/CollisionBody.h +++ b/src/body/CollisionBody.h @@ -91,6 +91,12 @@ class CollisionBody : public Body { /// Reset the contact manifold lists void resetContactManifoldsList(MemoryAllocator& memoryAllocator); + /// Update the old transform with the current one. + void updateOldTransform(); + + /// Update the Axis-Aligned Bounding Box coordinates + void updateAABB(); + public : // -------------------- Methods -------------------- // @@ -101,12 +107,6 @@ class CollisionBody : public Body { /// Destructor virtual ~CollisionBody(); - /// Return true if the body has moved during the last frame - bool getHasMoved() const; - - /// Set the hasMoved variable (true if the body has moved during the last frame) - void setHasMoved(bool hasMoved); - /// Return the collision shape CollisionShape* getCollisionShape() const; @@ -128,23 +128,17 @@ class CollisionBody : public Body { /// Set the interpolation factor of the body void setInterpolationFactor(decimal factor); - /// Return if the rigid body can move - bool getIsMotionEnabled() const; + /// Return true if the rigid body is allowed to move + bool isMotionEnabled() const; - /// Set the value to true if the body can move - void setIsMotionEnabled(bool isMotionEnabled); + /// Enable/disable the motion of the body + void enableMotion(bool isMotionEnabled); /// Return true if the body can collide with others bodies - bool getIsCollisionEnabled() const; + bool isCollisionEnabled() const; - /// Set the isCollisionEnabled value - void setIsCollisionEnabled(bool isCollisionEnabled); - - /// Update the old transform with the current one. - void updateOldTransform(); - - /// Update the Axis-Aligned Bounding Box coordinates - void updateAABB(); + /// Enable/disable the collision with this body + void enableCollision(bool isCollisionEnabled); /// Return the first element of the linked list of contact manifolds involving this body const ContactManifoldListElement* getContactManifoldsLists() const; @@ -152,18 +146,9 @@ class CollisionBody : public Body { // -------------------- Friendship -------------------- // friend class DynamicsWorld; + friend class CollisionDetection; }; -// Return true if the body has moved during the last frame -inline bool CollisionBody::getHasMoved() const { - return mHasMoved; -} - -// Set the hasMoved variable (true if the body has moved during the last frame) -inline void CollisionBody::setHasMoved(bool hasMoved) { - mHasMoved = hasMoved; -} - // Return the collision shape inline CollisionShape* CollisionBody::getCollisionShape() const { assert(mCollisionShape); @@ -187,13 +172,13 @@ inline void CollisionBody::setInterpolationFactor(decimal factor) { mInterpolationFactor = factor; } -// Return if the rigid body can move -inline bool CollisionBody::getIsMotionEnabled() const { +// Return true if the rigid body is allowed to move +inline bool CollisionBody::isMotionEnabled() const { return mIsMotionEnabled; } -// Set the value to true if the body can move -inline void CollisionBody::setIsMotionEnabled(bool isMotionEnabled) { +// Enable/disable the motion of the body +inline void CollisionBody::enableMotion(bool isMotionEnabled) { mIsMotionEnabled = isMotionEnabled; } @@ -219,12 +204,12 @@ inline const AABB& CollisionBody::getAABB() const { } // Return true if the body can collide with others bodies -inline bool CollisionBody::getIsCollisionEnabled() const { +inline bool CollisionBody::isCollisionEnabled() const { return mIsCollisionEnabled; } -// Set the isCollisionEnabled value -inline void CollisionBody::setIsCollisionEnabled(bool isCollisionEnabled) { +// Enable/disable the collision with this body +inline void CollisionBody::enableCollision(bool isCollisionEnabled) { mIsCollisionEnabled = isCollisionEnabled; } diff --git a/src/body/RigidBody.h b/src/body/RigidBody.h index c388e214..4b7980e8 100644 --- a/src/body/RigidBody.h +++ b/src/body/RigidBody.h @@ -105,6 +105,9 @@ class RigidBody : public CollisionBody { /// Remove a joint from the joints list void removeJointFromJointsList(MemoryAllocator& memoryAllocator, const Joint* joint); + /// Set the inverse of the mass + void setMassInverse(decimal massInverse); + public : // -------------------- Methods -------------------- // @@ -134,9 +137,6 @@ class RigidBody : public CollisionBody { /// Set the angular velocity void setAngularVelocity(const Vector3& angularVelocity); - /// Set the inverse of the mass - void setMassInverse(decimal massInverse); - /// Return the inverse of the mass of the body decimal getMassInverse() const; diff --git a/src/collision/CollisionDetection.cpp b/src/collision/CollisionDetection.cpp index cf006f5b..c98407bf 100644 --- a/src/collision/CollisionDetection.cpp +++ b/src/collision/CollisionDetection.cpp @@ -82,7 +82,7 @@ void CollisionDetection::computeBroadPhase() { it != mWorld->getBodiesEndIterator(); it++) { // If the body has moved - if ((*it)->getHasMoved()) { + if ((*it)->mHasMoved) { // Notify the broad-phase that the body has moved mBroadPhaseAlgorithm->updateObject(*it, (*it)->getAABB()); diff --git a/src/collision/CollisionDetection.h b/src/collision/CollisionDetection.h index a2aa3427..eee6f1a2 100644 --- a/src/collision/CollisionDetection.h +++ b/src/collision/CollisionDetection.h @@ -38,8 +38,6 @@ #include #include #include -#include // TODO : Delete this - /// ReactPhysics3D namespace namespace reactphysics3d { diff --git a/src/configuration.h b/src/configuration.h index 7141ab18..de4d293c 100644 --- a/src/configuration.h +++ b/src/configuration.h @@ -105,7 +105,7 @@ const decimal PERSISTENT_CONTACT_DIST_THRESHOLD = decimal(0.03); const decimal RESTITUTION_VELOCITY_THRESHOLD = decimal(1.0); /// Number of iterations when solving the velocity constraints of the Sequential Impulse technique -const uint DEFAULT_VELOCITY_SOLVER_NB_ITERATIONS = 15; +const uint DEFAULT_VELOCITY_SOLVER_NB_ITERATIONS = 10; /// Number of iterations when solving the position constraints of the Sequential Impulse technique const uint DEFAULT_POSITION_SOLVER_NB_ITERATIONS = 5; diff --git a/src/constraint/BallAndSocketJoint.cpp b/src/constraint/BallAndSocketJoint.cpp index c707e8cd..3db0e8ba 100644 --- a/src/constraint/BallAndSocketJoint.cpp +++ b/src/constraint/BallAndSocketJoint.cpp @@ -73,25 +73,25 @@ void BallAndSocketJoint::initBeforeSolve(const ConstraintSolverData& constraintS // Compute the matrix K=JM^-1J^t (3x3 matrix) decimal inverseMassBodies = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { inverseMassBodies += mBody1->getMassInverse(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { inverseMassBodies += mBody2->getMassInverse(); } Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, 0, inverseMassBodies, 0, 0, 0, inverseMassBodies); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); } // Compute the inverse mass matrix K^-1 mInverseMassMatrix.setToZero(); - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrix = massMatrix.getInverse(); } @@ -123,7 +123,7 @@ void BallAndSocketJoint::warmstart(const ConstraintSolverData& constraintSolverD const decimal inverseMassBody1 = mBody1->getMassInverse(); const decimal inverseMassBody2 = mBody2->getMassInverse(); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 linearImpulseBody1 = -mImpulse; @@ -133,7 +133,7 @@ void BallAndSocketJoint::warmstart(const ConstraintSolverData& constraintSolverD v1 += inverseMassBody1 * linearImpulseBody1; w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 linearImpulseBody2 = mImpulse; @@ -165,7 +165,7 @@ void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& con const Vector3 deltaLambda = mInverseMassMatrix * (-Jv - mBiasVector); mImpulse += deltaLambda; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 linearImpulseBody1 = -deltaLambda; @@ -175,7 +175,7 @@ void BallAndSocketJoint::solveVelocityConstraint(const ConstraintSolverData& con v1 += inverseMassBody1 * linearImpulseBody1; w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 linearImpulseBody2 = deltaLambda; @@ -218,23 +218,23 @@ void BallAndSocketJoint::solvePositionConstraint(const ConstraintSolverData& con // Recompute the inverse mass matrix K=J^TM^-1J of of the 3 translation constraints decimal inverseMassBodies = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { inverseMassBodies += inverseMassBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { inverseMassBodies += inverseMassBody2; } Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, 0, inverseMassBodies, 0, 0, 0, inverseMassBodies); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); } mInverseMassMatrix.setToZero(); - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrix = massMatrix.getInverse(); } @@ -247,7 +247,7 @@ void BallAndSocketJoint::solvePositionConstraint(const ConstraintSolverData& con const Vector3 lambda = mInverseMassMatrix * (-constraintError); // Apply the impulse to the bodies of the joint (directly update the position/orientation) - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse const Vector3 linearImpulseBody1 = -lambda; @@ -262,7 +262,7 @@ void BallAndSocketJoint::solvePositionConstraint(const ConstraintSolverData& con q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse const Vector3 linearImpulseBody2 = lambda; diff --git a/src/constraint/BallAndSocketJoint.h b/src/constraint/BallAndSocketJoint.h index f6bd8d7a..2030627f 100644 --- a/src/constraint/BallAndSocketJoint.h +++ b/src/constraint/BallAndSocketJoint.h @@ -56,7 +56,8 @@ struct BallAndSocketJointInfo : public JointInfo { // Class BallAndSocketJoint /** * This class represents a ball-and-socket joint that allows arbitrary rotation - * between two bodies. + * between two bodies. This joint has three degrees of freedom. It can be used to + * create a chain of bodies for instance. */ class BallAndSocketJoint : public Joint { @@ -104,16 +105,6 @@ class BallAndSocketJoint : public Joint { /// Private assignment operator BallAndSocketJoint& operator=(const BallAndSocketJoint& constraint); - public : - - // -------------------- Methods -------------------- // - - /// Constructor - BallAndSocketJoint(const BallAndSocketJointInfo& jointInfo); - - /// Destructor - virtual ~BallAndSocketJoint(); - /// Return the number of bytes used by the joint virtual size_t getSizeInBytes() const; @@ -128,6 +119,16 @@ class BallAndSocketJoint : public Joint { /// Solve the position constraint (for position error correction) virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + BallAndSocketJoint(const BallAndSocketJointInfo& jointInfo); + + /// Destructor + virtual ~BallAndSocketJoint(); }; // Return the number of bytes used by the joint diff --git a/src/constraint/FixedJoint.cpp b/src/constraint/FixedJoint.cpp index 46f7b89d..b649d88b 100644 --- a/src/constraint/FixedJoint.cpp +++ b/src/constraint/FixedJoint.cpp @@ -81,25 +81,25 @@ void FixedJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat // Compute the matrix K=JM^-1J^t (3x3 matrix) for the 3 translation constraints decimal inverseMassBodies = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { inverseMassBodies += mBody1->getMassInverse(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { inverseMassBodies += mBody2->getMassInverse(); } Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, 0, inverseMassBodies, 0, 0, 0, inverseMassBodies); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); } // Compute the inverse mass matrix K^-1 for the 3 translation constraints mInverseMassMatrixTranslation.setToZero(); - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixTranslation = massMatrix.getInverse(); } @@ -113,13 +113,13 @@ void FixedJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat // Compute the inverse of the mass matrix K=JM^-1J^t for the 3 rotation // contraints (3x3 matrix) mInverseMassMatrixRotation.setToZero(); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { mInverseMassMatrixRotation += mI1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { mInverseMassMatrixRotation += mI2; } - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixRotation = mInverseMassMatrixRotation.getInverse(); } @@ -154,7 +154,7 @@ void FixedJoint::warmstart(const ConstraintSolverData& constraintSolverData) { const decimal inverseMassBody1 = mBody1->getMassInverse(); const decimal inverseMassBody2 = mBody2->getMassInverse(); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 translation constraints Vector3 linearImpulseBody1 = -mImpulseTranslation; @@ -167,7 +167,7 @@ void FixedJoint::warmstart(const ConstraintSolverData& constraintSolverData) { v1 += inverseMassBody1 * linearImpulseBody1; w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 translation constraints Vector3 linearImpulseBody2 = mImpulseTranslation; @@ -205,7 +205,7 @@ void FixedJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS (-JvTranslation - mBiasTranslation); mImpulseTranslation += deltaLambda; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 linearImpulseBody1 = -deltaLambda; @@ -215,7 +215,7 @@ void FixedJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS v1 += inverseMassBody1 * linearImpulseBody1; w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 linearImpulseBody2 = deltaLambda; @@ -235,7 +235,7 @@ void FixedJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS Vector3 deltaLambda2 = mInverseMassMatrixRotation * (-JvRotation - mBiasRotation); mImpulseRotation += deltaLambda2; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 rotation constraints const Vector3 angularImpulseBody1 = -deltaLambda2; @@ -243,7 +243,7 @@ void FixedJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS // Apply the impulse to the body w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 rotation constraints const Vector3 angularImpulseBody2 = deltaLambda2; @@ -286,23 +286,23 @@ void FixedJoint::solvePositionConstraint(const ConstraintSolverData& constraintS // Compute the matrix K=JM^-1J^t (3x3 matrix) for the 3 translation constraints decimal inverseMassBodies = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { inverseMassBodies += mBody1->getMassInverse(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { inverseMassBodies += mBody2->getMassInverse(); } Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, 0, inverseMassBodies, 0, 0, 0, inverseMassBodies); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); } mInverseMassMatrixTranslation.setToZero(); - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixTranslation = massMatrix.getInverse(); } @@ -313,7 +313,7 @@ void FixedJoint::solvePositionConstraint(const ConstraintSolverData& constraintS const Vector3 lambdaTranslation = mInverseMassMatrixTranslation * (-errorTranslation); // Apply the impulse to the bodies of the joint - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse Vector3 linearImpulseBody1 = -lambdaTranslation; @@ -328,7 +328,7 @@ void FixedJoint::solvePositionConstraint(const ConstraintSolverData& constraintS q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse Vector3 linearImpulseBody2 = lambdaTranslation; @@ -349,13 +349,13 @@ void FixedJoint::solvePositionConstraint(const ConstraintSolverData& constraintS // Compute the inverse of the mass matrix K=JM^-1J^t for the 3 rotation // contraints (3x3 matrix) mInverseMassMatrixRotation.setToZero(); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { mInverseMassMatrixRotation += mI1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { mInverseMassMatrixRotation += mI2; } - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixRotation = mInverseMassMatrixRotation.getInverse(); } @@ -369,7 +369,7 @@ void FixedJoint::solvePositionConstraint(const ConstraintSolverData& constraintS Vector3 lambdaRotation = mInverseMassMatrixRotation * (-errorRotation); // Apply the impulse to the bodies of the joint - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 rotation constraints const Vector3 angularImpulseBody1 = -lambdaRotation; @@ -381,7 +381,7 @@ void FixedJoint::solvePositionConstraint(const ConstraintSolverData& constraintS q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse const Vector3 angularImpulseBody2 = lambdaRotation; diff --git a/src/constraint/FixedJoint.h b/src/constraint/FixedJoint.h index 1e153085..9fe3b055 100644 --- a/src/constraint/FixedJoint.h +++ b/src/constraint/FixedJoint.h @@ -116,16 +116,6 @@ class FixedJoint : public Joint { /// Private assignment operator FixedJoint& operator=(const FixedJoint& constraint); - public : - - // -------------------- Methods -------------------- // - - /// Constructor - FixedJoint(const FixedJointInfo& jointInfo); - - /// Destructor - virtual ~FixedJoint(); - /// Return the number of bytes used by the joint virtual size_t getSizeInBytes() const; @@ -140,6 +130,16 @@ class FixedJoint : public Joint { /// Solve the position constraint (for position error correction) virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); + + public : + + // -------------------- Methods -------------------- // + + /// Constructor + FixedJoint(const FixedJointInfo& jointInfo); + + /// Destructor + virtual ~FixedJoint(); }; // Return the number of bytes used by the joint diff --git a/src/constraint/HingeJoint.cpp b/src/constraint/HingeJoint.cpp index 2a7a5ee0..9e6a132c 100644 --- a/src/constraint/HingeJoint.cpp +++ b/src/constraint/HingeJoint.cpp @@ -123,23 +123,23 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat // Compute the inverse mass matrix K=JM^-1J^t for the 3 translation constraints (3x3 matrix) decimal inverseMassBodies = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { inverseMassBodies += mBody1->getMassInverse(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { inverseMassBodies += mBody2->getMassInverse(); } Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, 0, inverseMassBodies, 0, 0, 0, inverseMassBodies); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); } mInverseMassMatrixTranslation.setToZero(); - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixTranslation = massMatrix.getInverse(); } @@ -155,11 +155,11 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat Vector3 I1C2CrossA1(0, 0, 0); Vector3 I2B2CrossA1(0, 0, 0); Vector3 I2C2CrossA1(0, 0, 0); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { I1B2CrossA1 = mI1 * mB2CrossA1; I1C2CrossA1 = mI1 * mC2CrossA1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { I2B2CrossA1 = mI2 * mB2CrossA1; I2C2CrossA1 = mI2 * mC2CrossA1; } @@ -173,7 +173,7 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat mC2CrossA1.dot(I2C2CrossA1); const Matrix2x2 matrixKRotation(el11, el12, el21, el22); mInverseMassMatrixRotation.setToZero(); - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixRotation = matrixKRotation.getInverse(); } @@ -198,10 +198,10 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) mInverseMassMatrixLimitMotor = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { mInverseMassMatrixLimitMotor += mA1.dot(mI1 * mA1); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { mInverseMassMatrixLimitMotor += mA1.dot(mI2 * mA1); } mInverseMassMatrixLimitMotor = (mInverseMassMatrixLimitMotor > 0.0) ? @@ -243,7 +243,7 @@ void HingeJoint::warmstart(const ConstraintSolverData& constraintSolverData) { // Compute the impulse P=J^T * lambda for the motor constraint const Vector3 motorImpulse = -mImpulseMotor * mA1; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 translation constraints Vector3 linearImpulseBody1 = -mImpulseTranslation; @@ -262,7 +262,7 @@ void HingeJoint::warmstart(const ConstraintSolverData& constraintSolverData) { v1 += inverseMassBody1 * linearImpulseBody1; w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 translation constraints Vector3 linearImpulseBody2 = mImpulseTranslation; @@ -306,7 +306,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS (-JvTranslation - mBTranslation); mImpulseTranslation += deltaLambdaTranslation; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 linearImpulseBody1 = -deltaLambdaTranslation; @@ -316,7 +316,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS v1 += inverseMassBody1 * linearImpulseBody1; w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 linearImpulseBody2 = deltaLambdaTranslation; @@ -337,7 +337,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS Vector2 deltaLambdaRotation = mInverseMassMatrixRotation * (-JvRotation - mBRotation); mImpulseRotation += deltaLambdaRotation; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 2 rotation constraints const Vector3 angularImpulseBody1 = -mB2CrossA1 * deltaLambdaRotation.x - @@ -346,7 +346,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS // Apply the impulse to the body w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 2 rotation constraints const Vector3 angularImpulseBody2 = mB2CrossA1 * deltaLambdaRotation.x + @@ -372,7 +372,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS mImpulseLowerLimit = std::max(mImpulseLowerLimit + deltaLambdaLower, decimal(0.0)); deltaLambdaLower = mImpulseLowerLimit - lambdaTemp; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the lower limit constraint const Vector3 angularImpulseBody1 = -deltaLambdaLower * mA1; @@ -380,7 +380,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS // Apply the impulse to the body w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the lower limit constraint const Vector3 angularImpulseBody2 = deltaLambdaLower * mA1; @@ -402,7 +402,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS mImpulseUpperLimit = std::max(mImpulseUpperLimit + deltaLambdaUpper, decimal(0.0)); deltaLambdaUpper = mImpulseUpperLimit - lambdaTemp; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the upper limit constraint const Vector3 angularImpulseBody1 = deltaLambdaUpper * mA1; @@ -410,7 +410,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS // Apply the impulse to the body w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the upper limit constraint const Vector3 angularImpulseBody2 = -deltaLambdaUpper * mA1; @@ -435,7 +435,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS mImpulseMotor = clamp(mImpulseMotor + deltaLambdaMotor, -maxMotorImpulse, maxMotorImpulse); deltaLambdaMotor = mImpulseMotor - lambdaTemp; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the motor const Vector3 angularImpulseBody1 = -deltaLambdaMotor * mA1; @@ -443,7 +443,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS // Apply the impulse to the body w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the motor const Vector3 angularImpulseBody2 = deltaLambdaMotor * mA1; @@ -506,23 +506,23 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS // Compute the matrix K=JM^-1J^t (3x3 matrix) for the 3 translation constraints decimal inverseMassBodies = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { inverseMassBodies += mBody1->getMassInverse(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { inverseMassBodies += mBody2->getMassInverse(); } Matrix3x3 massMatrix = Matrix3x3(inverseMassBodies, 0, 0, 0, inverseMassBodies, 0, 0, 0, inverseMassBodies); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU1 * mI1 * skewSymmetricMatrixU1.getTranspose(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { massMatrix += skewSymmetricMatrixU2 * mI2 * skewSymmetricMatrixU2.getTranspose(); } mInverseMassMatrixTranslation.setToZero(); - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixTranslation = massMatrix.getInverse(); } @@ -533,7 +533,7 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS const Vector3 lambdaTranslation = mInverseMassMatrixTranslation * (-errorTranslation); // Apply the impulse to the bodies of the joint - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse Vector3 linearImpulseBody1 = -lambdaTranslation; @@ -548,7 +548,7 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse Vector3 linearImpulseBody2 = lambdaTranslation; @@ -571,11 +571,11 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS Vector3 I1C2CrossA1(0, 0, 0); Vector3 I2B2CrossA1(0, 0, 0); Vector3 I2C2CrossA1(0, 0, 0); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { I1B2CrossA1 = mI1 * mB2CrossA1; I1C2CrossA1 = mI1 * mC2CrossA1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { I2B2CrossA1 = mI2 * mB2CrossA1; I2C2CrossA1 = mI2 * mC2CrossA1; } @@ -589,7 +589,7 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS mC2CrossA1.dot(I2C2CrossA1); const Matrix2x2 matrixKRotation(el11, el12, el21, el22); mInverseMassMatrixRotation.setToZero(); - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixRotation = matrixKRotation.getInverse(); } @@ -600,7 +600,7 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS Vector2 lambdaRotation = mInverseMassMatrixRotation * (-errorRotation); // Apply the impulse to the bodies of the joint - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 rotation constraints const Vector3 angularImpulseBody1 = -mB2CrossA1 * lambdaRotation.x - @@ -613,7 +613,7 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse const Vector3 angularImpulseBody2 = mB2CrossA1 * lambdaRotation.x + @@ -635,10 +635,10 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) mInverseMassMatrixLimitMotor = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { mInverseMassMatrixLimitMotor += mA1.dot(mI1 * mA1); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { mInverseMassMatrixLimitMotor += mA1.dot(mI2 * mA1); } mInverseMassMatrixLimitMotor = (mInverseMassMatrixLimitMotor > 0.0) ? @@ -652,7 +652,7 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS decimal lambdaLowerLimit = mInverseMassMatrixLimitMotor * (-lowerLimitError ); // Apply the impulse to the bodies of the joint - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 angularImpulseBody1 = -lambdaLowerLimit * mA1; @@ -664,7 +664,7 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 angularImpulseBody2 = lambdaLowerLimit * mA1; @@ -685,7 +685,7 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS decimal lambdaUpperLimit = mInverseMassMatrixLimitMotor * (-upperLimitError); // Apply the impulse to the bodies of the joint - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 angularImpulseBody1 = lambdaUpperLimit * mA1; @@ -697,7 +697,7 @@ void HingeJoint::solvePositionConstraint(const ConstraintSolverData& constraintS q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda const Vector3 angularImpulseBody2 = -lambdaUpperLimit * mA1; diff --git a/src/constraint/HingeJoint.h b/src/constraint/HingeJoint.h index f9505992..00dacd29 100644 --- a/src/constraint/HingeJoint.h +++ b/src/constraint/HingeJoint.h @@ -109,7 +109,8 @@ struct HingeJointInfo : public JointInfo { // Class HingeJoint /** * This class represents a hinge joint that allows arbitrary rotation - * between two bodies around a single axis. + * between two bodies around a single axis. This joint has one degree of freedom. It + * can be useful to simulate doors or pendulumns. */ class HingeJoint : public Joint { @@ -246,6 +247,21 @@ class HingeJoint : public Joint { decimal computeCurrentHingeAngle(const Quaternion& orientationBody1, const Quaternion& orientationBody2); + /// Return the number of bytes used by the joint + virtual size_t getSizeInBytes() const; + + /// Initialize before solving the constraint + virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); + + /// Warm start the constraint (apply the previous impulse at the beginning of the step) + virtual void warmstart(const ConstraintSolverData& constraintSolverData); + + /// Solve the velocity constraint + virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); + + /// Solve the position constraint (for position error correction) + virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); + public : // -------------------- Methods -------------------- // @@ -294,21 +310,6 @@ class HingeJoint : public Joint { /// Return the intensity of the current torque applied for the joint motor decimal getMotorTorque(decimal timeStep) const; - - /// Return the number of bytes used by the joint - virtual size_t getSizeInBytes() const; - - /// Initialize before solving the constraint - virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); - - /// Warm start the constraint (apply the previous impulse at the beginning of the step) - virtual void warmstart(const ConstraintSolverData& constraintSolverData); - - /// Solve the velocity constraint - virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); - - /// Solve the position constraint (for position error correction) - virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); }; // Return true if the limits or the joint are enabled diff --git a/src/constraint/Joint.h b/src/constraint/Joint.h index 425ece73..9d57aa08 100644 --- a/src/constraint/Joint.h +++ b/src/constraint/Joint.h @@ -152,6 +152,24 @@ class Joint { /// Private assignment operator Joint& operator=(const Joint& constraint); + /// Return true if the joint has already been added into an island + bool isAlreadyInIsland() const; + + /// Return the number of bytes used by the joint + virtual size_t getSizeInBytes() const = 0; + + /// Initialize before solving the joint + virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData) = 0; + + /// Warm start the joint (apply the previous impulse at the beginning of the step) + virtual void warmstart(const ConstraintSolverData& constraintSolverData) = 0; + + /// Solve the velocity constraint + virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) = 0; + + /// Solve the position constraint + virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData) = 0; + public : // -------------------- Methods -------------------- // @@ -177,28 +195,11 @@ class Joint { /// Return true if the collision between the two bodies of the joint is enabled bool isCollisionEnabled() const; - /// Return true if the joint has already been added into an island - bool isAlreadyInIsland() const; - - /// Return the number of bytes used by the joint - virtual size_t getSizeInBytes() const = 0; - - /// Initialize before solving the joint - virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData) = 0; - - /// Warm start the joint (apply the previous impulse at the beginning of the step) - virtual void warmstart(const ConstraintSolverData& constraintSolverData) = 0; - - /// Solve the velocity constraint - virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData) = 0; - - /// Solve the position constraint - virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData) = 0; - // -------------------- Friendship -------------------- // friend class DynamicsWorld; friend class Island; + friend class ConstraintSolver; }; // Return the reference to the body 1 diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index 0fd912ce..9a3cdf97 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -129,12 +129,12 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa Vector3 I1R1PlusUCrossN2(0, 0, 0); Vector3 I2R2CrossN1(0, 0, 0); Vector3 I2R2CrossN2(0, 0, 0); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { sumInverseMass += mBody1->getMassInverse(); I1R1PlusUCrossN1 = mI1 * mR1PlusUCrossN1; I1R1PlusUCrossN2 = mI1 * mR1PlusUCrossN2; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { sumInverseMass += mBody2->getMassInverse(); I2R2CrossN1 = mI2 * mR2CrossN1; I2R2CrossN2 = mI2 * mR2CrossN2; @@ -149,7 +149,7 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa mR2CrossN2.dot(I2R2CrossN2); Matrix2x2 matrixKTranslation(el11, el12, el21, el22); mInverseMassMatrixTranslationConstraint.setToZero(); - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixTranslationConstraint = matrixKTranslation.getInverse(); } @@ -165,13 +165,13 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa // Compute the inverse of the mass matrix K=JM^-1J^t for the 3 rotation // contraints (3x3 matrix) mInverseMassMatrixRotationConstraint.setToZero(); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { mInverseMassMatrixRotationConstraint += mI1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { mInverseMassMatrixRotationConstraint += mI2; } - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixRotationConstraint = mInverseMassMatrixRotationConstraint.getInverse(); } @@ -188,11 +188,11 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) mInverseMassMatrixLimit = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { mInverseMassMatrixLimit += mBody1->getMassInverse() + mR1PlusUCrossSliderAxis.dot(mI1 * mR1PlusUCrossSliderAxis); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { mInverseMassMatrixLimit += mBody2->getMassInverse() + mR2CrossSliderAxis.dot(mI2 * mR2CrossSliderAxis); } @@ -214,10 +214,10 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa // Compute the inverse of mass matrix K=JM^-1J^t for the motor (1x1 matrix) mInverseMassMatrixMotor = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { mInverseMassMatrixMotor += mBody1->getMassInverse(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { mInverseMassMatrixMotor += mBody2->getMassInverse(); } mInverseMassMatrixMotor = (mInverseMassMatrixMotor > 0.0) ? @@ -255,7 +255,7 @@ void SliderJoint::warmstart(const ConstraintSolverData& constraintSolverData) { // Compute the impulse P=J^T * lambda for the motor constraint Vector3 impulseMotor = mImpulseMotor * mSliderAxisWorld; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 2 translation constraints Vector3 linearImpulseBody1 = -mN1 * mImpulseTranslation.x - mN2 * mImpulseTranslation.y; @@ -276,7 +276,7 @@ void SliderJoint::warmstart(const ConstraintSolverData& constraintSolverData) { v1 += inverseMassBody1 * linearImpulseBody1; w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 2 translation constraints Vector3 linearImpulseBody2 = mN1 * mImpulseTranslation.x + mN2 * mImpulseTranslation.y; @@ -325,7 +325,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint Vector2 deltaLambda = mInverseMassMatrixTranslationConstraint * (-JvTranslation -mBTranslation); mImpulseTranslation += deltaLambda; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 2 translation constraints const Vector3 linearImpulseBody1 = -mN1 * deltaLambda.x - mN2 * deltaLambda.y; @@ -336,7 +336,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint v1 += inverseMassBody1 * linearImpulseBody1; w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 2 translation constraints const Vector3 linearImpulseBody2 = mN1 * deltaLambda.x + mN2 * deltaLambda.y; @@ -356,7 +356,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint Vector3 deltaLambda2 = mInverseMassMatrixRotationConstraint * (-JvRotation - mBRotation); mImpulseRotation += deltaLambda2; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 rotation constraints const Vector3 angularImpulseBody1 = -deltaLambda2; @@ -364,7 +364,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint // Apply the impulse to the body w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 rotation constraints const Vector3 angularImpulseBody2 = deltaLambda2; @@ -390,7 +390,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint mImpulseLowerLimit = std::max(mImpulseLowerLimit + deltaLambdaLower, decimal(0.0)); deltaLambdaLower = mImpulseLowerLimit - lambdaTemp; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the lower limit constraint const Vector3 linearImpulseBody1 = -deltaLambdaLower * mSliderAxisWorld; @@ -400,7 +400,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint v1 += inverseMassBody1 * linearImpulseBody1; w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the lower limit constraint const Vector3 linearImpulseBody2 = deltaLambdaLower * mSliderAxisWorld; @@ -425,7 +425,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint mImpulseUpperLimit = std::max(mImpulseUpperLimit + deltaLambdaUpper, decimal(0.0)); deltaLambdaUpper = mImpulseUpperLimit - lambdaTemp; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the upper limit constraint const Vector3 linearImpulseBody1 = deltaLambdaUpper * mSliderAxisWorld; @@ -435,7 +435,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint v1 += inverseMassBody1 * linearImpulseBody1; w1 += mI1 * angularImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the upper limit constraint const Vector3 linearImpulseBody2 = -deltaLambdaUpper * mSliderAxisWorld; @@ -462,7 +462,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint mImpulseMotor = clamp(mImpulseMotor + deltaLambdaMotor, -maxMotorImpulse, maxMotorImpulse); deltaLambdaMotor = mImpulseMotor - lambdaTemp; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the motor const Vector3 linearImpulseBody1 = deltaLambdaMotor * mSliderAxisWorld; @@ -470,7 +470,7 @@ void SliderJoint::solveVelocityConstraint(const ConstraintSolverData& constraint // Apply the impulse to the body v1 += inverseMassBody1 * linearImpulseBody1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the motor const Vector3 linearImpulseBody2 = -deltaLambdaMotor * mSliderAxisWorld; @@ -540,12 +540,12 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint Vector3 I1R1PlusUCrossN2(0, 0, 0); Vector3 I2R2CrossN1(0, 0, 0); Vector3 I2R2CrossN2(0, 0, 0); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { sumInverseMass += mBody1->getMassInverse(); I1R1PlusUCrossN1 = mI1 * mR1PlusUCrossN1; I1R1PlusUCrossN2 = mI1 * mR1PlusUCrossN2; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { sumInverseMass += mBody2->getMassInverse(); I2R2CrossN1 = mI2 * mR2CrossN1; I2R2CrossN2 = mI2 * mR2CrossN2; @@ -560,7 +560,7 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint mR2CrossN2.dot(I2R2CrossN2); Matrix2x2 matrixKTranslation(el11, el12, el21, el22); mInverseMassMatrixTranslationConstraint.setToZero(); - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixTranslationConstraint = matrixKTranslation.getInverse(); } @@ -570,7 +570,7 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint // Compute the Lagrange multiplier lambda for the 2 translation constraints Vector2 lambdaTranslation = mInverseMassMatrixTranslationConstraint * (-translationError); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 2 translation constraints const Vector3 linearImpulseBody1 = -mN1 * lambdaTranslation.x - mN2 * lambdaTranslation.y; @@ -586,7 +586,7 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 2 translation constraints const Vector3 linearImpulseBody2 = mN1 * lambdaTranslation.x + mN2 * lambdaTranslation.y; @@ -608,13 +608,13 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint // Compute the inverse of the mass matrix K=JM^-1J^t for the 3 rotation // contraints (3x3 matrix) mInverseMassMatrixRotationConstraint.setToZero(); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { mInverseMassMatrixRotationConstraint += mI1; } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { mInverseMassMatrixRotationConstraint += mI2; } - if (mBody1->getIsMotionEnabled() || mBody2->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled() || mBody2->isMotionEnabled()) { mInverseMassMatrixRotationConstraint = mInverseMassMatrixRotationConstraint.getInverse(); } @@ -627,7 +627,7 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint // Compute the Lagrange multiplier lambda for the 3 rotation constraints Vector3 lambdaRotation = mInverseMassMatrixRotationConstraint * (-errorRotation); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 rotation constraints const Vector3 angularImpulseBody1 = -lambdaRotation; @@ -639,7 +639,7 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the 3 rotation constraints const Vector3 angularImpulseBody2 = lambdaRotation; @@ -660,11 +660,11 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) mInverseMassMatrixLimit = 0.0; - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { mInverseMassMatrixLimit += mBody1->getMassInverse() + mR1PlusUCrossSliderAxis.dot(mI1 * mR1PlusUCrossSliderAxis); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { mInverseMassMatrixLimit += mBody2->getMassInverse() + mR2CrossSliderAxis.dot(mI2 * mR2CrossSliderAxis); } @@ -678,7 +678,7 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint // Compute the Lagrange multiplier lambda for the lower limit constraint decimal lambdaLowerLimit = mInverseMassMatrixLimit * (-lowerLimitError); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the lower limit constraint const Vector3 linearImpulseBody1 = -lambdaLowerLimit * mSliderAxisWorld; @@ -693,7 +693,7 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the lower limit constraint const Vector3 linearImpulseBody2 = lambdaLowerLimit * mSliderAxisWorld; @@ -716,7 +716,7 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint // Compute the Lagrange multiplier lambda for the upper limit constraint decimal lambdaUpperLimit = mInverseMassMatrixLimit * (-upperLimitError); - if (mBody1->getIsMotionEnabled()) { + if (mBody1->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the upper limit constraint const Vector3 linearImpulseBody1 = lambdaUpperLimit * mSliderAxisWorld; @@ -731,7 +731,7 @@ void SliderJoint::solvePositionConstraint(const ConstraintSolverData& constraint q1 += Quaternion(0, w1) * q1 * decimal(0.5); q1.normalize(); } - if (mBody2->getIsMotionEnabled()) { + if (mBody2->isMotionEnabled()) { // Compute the impulse P=J^T * lambda for the upper limit constraint const Vector3 linearImpulseBody2 = -lambdaUpperLimit * mSliderAxisWorld; diff --git a/src/constraint/SliderJoint.h b/src/constraint/SliderJoint.h index 8e06fd26..645c2e47 100644 --- a/src/constraint/SliderJoint.h +++ b/src/constraint/SliderJoint.h @@ -107,7 +107,9 @@ struct SliderJointInfo : public JointInfo { // Class SliderJoint /** - * This class represents a slider joint. + * This class represents a slider joint. This joint has a one degree of freedom. + * It only allows relative translation of the bodies along a single direction and no + * rotation. */ class SliderJoint : public Joint { @@ -245,6 +247,21 @@ class SliderJoint : public Joint { /// Reset the limits void resetLimits(); + /// Return the number of bytes used by the joint + virtual size_t getSizeInBytes() const; + + /// Initialize before solving the constraint + virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); + + /// Warm start the constraint (apply the previous impulse at the beginning of the step) + virtual void warmstart(const ConstraintSolverData& constraintSolverData); + + /// Solve the velocity constraint + virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); + + /// Solve the position constraint (for position error correction) + virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); + public : // -------------------- Methods -------------------- // @@ -296,21 +313,6 @@ class SliderJoint : public Joint { /// Return the intensity of the current force applied for the joint motor decimal getMotorForce(decimal timeStep) const; - - /// Return the number of bytes used by the joint - virtual size_t getSizeInBytes() const; - - /// Initialize before solving the constraint - virtual void initBeforeSolve(const ConstraintSolverData& constraintSolverData); - - /// Warm start the constraint (apply the previous impulse at the beginning of the step) - virtual void warmstart(const ConstraintSolverData& constraintSolverData); - - /// Solve the velocity constraint - virtual void solveVelocityConstraint(const ConstraintSolverData& constraintSolverData); - - /// Solve the position constraint (for position error correction) - virtual void solvePositionConstraint(const ConstraintSolverData& constraintSolverData); }; // Return true if the limits or the joint are enabled diff --git a/src/engine/ContactSolver.cpp b/src/engine/ContactSolver.cpp index b40eb8f2..3bb660b3 100644 --- a/src/engine/ContactSolver.cpp +++ b/src/engine/ContactSolver.cpp @@ -96,8 +96,8 @@ void ContactSolver::initializeForIsland(decimal dt, Island* island) { internalManifold.indexBody2 = mMapBodyToConstrainedVelocityIndex.find(body2)->second; internalManifold.inverseInertiaTensorBody1 = body1->getInertiaTensorInverseWorld(); internalManifold.inverseInertiaTensorBody2 = body2->getInertiaTensorInverseWorld(); - internalManifold.isBody1Moving = body1->getIsMotionEnabled(); - internalManifold.isBody2Moving = body2->getIsMotionEnabled(); + internalManifold.isBody1Moving = body1->isMotionEnabled(); + internalManifold.isBody2Moving = body2->isMotionEnabled(); internalManifold.massInverseBody1 = body1->getMassInverse(); internalManifold.massInverseBody2 = body2->getMassInverse(); internalManifold.nbContacts = externalManifold->getNbContactPoints(); diff --git a/src/engine/DynamicsWorld.cpp b/src/engine/DynamicsWorld.cpp index 68073562..7688c5ee 100644 --- a/src/engine/DynamicsWorld.cpp +++ b/src/engine/DynamicsWorld.cpp @@ -174,7 +174,7 @@ void DynamicsWorld::integrateRigidBodiesPositions() { for (uint b=0; b < mIslands[i]->getNbBodies(); b++) { // If the body is allowed to move - if (bodies[b]->getIsMotionEnabled()) { + if (bodies[b]->isMotionEnabled()) { // Get the constrained velocity uint indexArray = mMapBodyToConstrainedVelocityIndex.find(bodies[b])->second; @@ -220,7 +220,7 @@ void DynamicsWorld::updateRigidBodiesAABB() { for (it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { // If the body has moved - if ((*it)->getHasMoved()) { + if ((*it)->mHasMoved) { // Update the AABB of the rigid body (*it)->updateAABB(); @@ -314,7 +314,7 @@ void DynamicsWorld::integrateRigidBodiesVelocities() { assert(mSplitAngularVelocities[indexBody] == Vector3(0, 0, 0)); // If the body is allowed to move - if (bodies[b]->getIsMotionEnabled()) { + if (bodies[b]->isMotionEnabled()) { // Integrate the external force to get the new velocity of the body mConstrainedLinearVelocities[indexBody] = bodies[b]->getLinearVelocity() + @@ -528,12 +528,9 @@ void DynamicsWorld::destroyRigidBody(RigidBody* rigidBody) { removeCollisionShape(rigidBody->getCollisionShape()); // Destroy all the joints in which the rigid body to be destroyed is involved - // TODO : Iterate on the mJointList of the rigid body instead over all the joints of the world - bodyindex idToRemove = rigidBody->getID(); - for (std::set::iterator it = mJoints.begin(); it != mJoints.end(); ++it) { - if ((*it)->getBody1()->getID() == idToRemove || (*it)->getBody2()->getID() == idToRemove) { - destroyJoint(*it); - } + JointListElement* element; + for (element = rigidBody->mJointsList; element != NULL; element = element->next) { + destroyJoint(element->joint); } // Reset the contact manifold list of the body @@ -752,19 +749,17 @@ void DynamicsWorld::computeIslands() { size_t nbBytesStack = sizeof(RigidBody*) * nbBodies; RigidBody** stackBodiesToVisit = (RigidBody**)mMemoryAllocator.allocate(nbBytesStack); - uint idIsland = 0; // TODO : REMOVE THIS - // For each rigid body of the world for (std::set::iterator it = mRigidBodies.begin(); it != mRigidBodies.end(); ++it) { RigidBody* body = *it; // If the body has already been added to an island, we go to the next body - if (body->isAlreadyInIsland()) continue; + if (body->mIsAlreadyInIsland) continue; // If the body is not moving, we go to the next body // TODO : When we will use STATIC bodies, we will need to take care of this case here - if (!body->getIsMotionEnabled()) continue; + if (!body->isMotionEnabled()) continue; // If the body is sleeping or inactive, we go to the next body if (body->isSleeping() || !body->isActive()) continue; @@ -777,9 +772,8 @@ void DynamicsWorld::computeIslands() { // Create the new island void* allocatedMemoryIsland = mMemoryAllocator.allocate(sizeof(Island)); - mIslands[mNbIslands] = new (allocatedMemoryIsland) Island(idIsland, nbBodies,mContactManifolds.size(), + mIslands[mNbIslands] = new (allocatedMemoryIsland) Island(nbBodies,mContactManifolds.size(), mJoints.size(), mMemoryAllocator); - idIsland++; // While there are still some bodies to visit in the stack while (stackIndex > 0) { @@ -797,7 +791,7 @@ void DynamicsWorld::computeIslands() { // If the current body is not moving, we do not want to perform the DFS // search across that body - if (!bodyToVisit->getIsMotionEnabled()) continue; + if (!bodyToVisit->isMotionEnabled()) continue; // For each contact manifold in which the current body is involded ContactManifoldListElement* contactElement; @@ -819,7 +813,7 @@ void DynamicsWorld::computeIslands() { RigidBody* otherBody = (body1->getID() == bodyToVisit->getID()) ? body2 : body1; // Check if the other body has already been added to the island - if (otherBody->isAlreadyInIsland()) continue; + if (otherBody->mIsAlreadyInIsland) continue; // Insert the other body into the stack of bodies to visit stackBodiesToVisit[stackIndex] = otherBody; @@ -847,7 +841,7 @@ void DynamicsWorld::computeIslands() { RigidBody* otherBody = (body1->getID() == bodyToVisit->getID()) ? body2 : body1; // Check if the other body has already been added to the island - if (otherBody->isAlreadyInIsland()) continue; + if (otherBody->mIsAlreadyInIsland) continue; // Insert the other body into the stack of bodies to visit stackBodiesToVisit[stackIndex] = otherBody; @@ -860,7 +854,7 @@ void DynamicsWorld::computeIslands() { // can also be included in the other islands for (uint i=0; i < mIslands[mNbIslands]->mNbBodies; i++) { - if (!mIslands[mNbIslands]->mBodies[i]->getIsMotionEnabled()) { + if (!mIslands[mNbIslands]->mBodies[i]->isMotionEnabled()) { mIslands[mNbIslands]->mBodies[i]->mIsAlreadyInIsland = false; } } @@ -893,7 +887,7 @@ void DynamicsWorld::updateSleepingBodies() { for (uint b=0; b < mIslands[i]->getNbBodies(); b++) { // Skip static bodies - if (!bodies[b]->getIsMotionEnabled()) continue; + if (!bodies[b]->isMotionEnabled()) continue; // If the body is velocity is large enough to stay awake if (bodies[b]->getLinearVelocity().lengthSquare() > sleepLinearVelocitySquare || diff --git a/src/engine/DynamicsWorld.h b/src/engine/DynamicsWorld.h index 36f6ab23..d1a9b158 100644 --- a/src/engine/DynamicsWorld.h +++ b/src/engine/DynamicsWorld.h @@ -194,7 +194,7 @@ class DynamicsWorld : public CollisionWorld { virtual void notifyNewContact(const BroadPhasePair* pair, const ContactPointInfo* contactInfo); -public : + public : // -------------------- Methods -------------------- // @@ -246,8 +246,8 @@ public : /// Add the joint to the list of joints of the two bodies involved in the joint void addJointToBody(Joint* joint); - //// Add a contact manifold to the linked list of contact manifolds of the two bodies involed - //// in the corresponding contact. + /// Add a contact manifold to the linked list of contact manifolds of the two bodies + /// involed in the corresponding contact. void addContactManifoldToBody(ContactManifold* contactManifold, CollisionBody *body1, CollisionBody *body2); @@ -310,12 +310,6 @@ public : /// Set an event listener object to receive events callbacks. void setEventListener(EventListener* eventListener); - - // TODO : REMOVE THIS - Island** getIslands() { return mIslands;} - - // TODO : REMOVE THIS - uint getNbIslands() const {return mNbIslands;} }; // Reset the external force and torque applied to the bodies @@ -384,7 +378,7 @@ inline void DynamicsWorld::resetBodiesMovementVariable() { it != getRigidBodiesEndIterator(); it++) { // Set the hasMoved variable to false - (*it)->setHasMoved(false); + (*it)->mHasMoved = false; } } diff --git a/src/engine/Island.cpp b/src/engine/Island.cpp index 5f5b36a2..116f652a 100644 --- a/src/engine/Island.cpp +++ b/src/engine/Island.cpp @@ -29,9 +29,9 @@ using namespace reactphysics3d; // Constructor -Island::Island(uint id, uint nbMaxBodies, uint nbMaxContactManifolds, uint nbMaxJoints, +Island::Island(uint nbMaxBodies, uint nbMaxContactManifolds, uint nbMaxJoints, MemoryAllocator& memoryAllocator) - : mID(id), mBodies(NULL), mContactManifolds(NULL), mJoints(NULL), mNbBodies(0), + : mBodies(NULL), mContactManifolds(NULL), mJoints(NULL), mNbBodies(0), mNbContactManifolds(0), mNbJoints(0), mMemoryAllocator(memoryAllocator) { // Allocate memory for the arrays diff --git a/src/engine/Island.h b/src/engine/Island.h index 283e532d..2aa067f3 100644 --- a/src/engine/Island.h +++ b/src/engine/Island.h @@ -45,9 +45,6 @@ class Island { // -------------------- Attributes -------------------- // - // TODO : REMOVE THIS - uint mID; - /// Array with all the bodies of the island RigidBody** mBodies; @@ -91,7 +88,7 @@ class Island { // -------------------- Methods -------------------- // /// Constructor - Island(uint id, uint nbMaxBodies, uint nbMaxContactManifolds, uint nbMaxJoints, + Island(uint nbMaxBodies, uint nbMaxContactManifolds, uint nbMaxJoints, MemoryAllocator& memoryAllocator); /// Destructor @@ -124,9 +121,6 @@ class Island { /// Return a pointer to the array of joints Joint** getJoints(); - // TODO : REMOVE THIS - uint getID() const {return mID;} - // -------------------- Friendship -------------------- // friend class DynamicsWorld; diff --git a/src/reactphysics3d.h b/src/reactphysics3d.h index cf85da5e..4724eab0 100644 --- a/src/reactphysics3d.h +++ b/src/reactphysics3d.h @@ -26,7 +26,7 @@ /******************************************************************************** * ReactPhysics3D * -* Version 0.3.0 * +* Version 0.4.0 * * http://code.google.com/p/reactphysics3d/ * * Daniel Chappuis * ********************************************************************************/ From 3173b368c7e67271c2eb925902eff519aa1e029c Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Mon, 30 Sep 2013 23:34:47 +0200 Subject: [PATCH 63/66] Fix compilation error on Mac OS X and errors in SliderJoint and HingeJoint --- examples/joints/Scene.cpp | 2 +- src/constraint/HingeJoint.cpp | 27 ++++++++++++++++----------- src/constraint/SliderJoint.cpp | 23 ++++++++++++++--------- src/engine/Profiler.cpp | 2 +- 4 files changed, 32 insertions(+), 22 deletions(-) diff --git a/examples/joints/Scene.cpp b/examples/joints/Scene.cpp index a651334d..c5c9312e 100644 --- a/examples/joints/Scene.cpp +++ b/examples/joints/Scene.cpp @@ -127,7 +127,7 @@ void Scene::simulate() { if (mIsRunning) { // Update the motor speed of the Slider Joint (to move up and down) - long double motorSpeed = 3 * cos(mDynamicsWorld->getPhysicsTime() * 1.5); + long double motorSpeed = 2 * cos(mDynamicsWorld->getPhysicsTime() * 1.5); mSliderJoint->setMotorSpeed(rp3d::decimal(motorSpeed)); // Take a simulation step diff --git a/src/constraint/HingeJoint.cpp b/src/constraint/HingeJoint.cpp index 9e6a132c..9e84b2bc 100644 --- a/src/constraint/HingeJoint.cpp +++ b/src/constraint/HingeJoint.cpp @@ -194,9 +194,10 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat mImpulseMotor = 0.0; } - if (mIsLimitEnabled && (mIsLowerLimitViolated || mIsUpperLimitViolated)) { + // If the motor or limits are enabled + if (mIsMotorEnabled || (mIsLimitEnabled && (mIsLowerLimitViolated || mIsUpperLimitViolated))) { - // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) + // Compute the inverse of the mass matrix K=JM^-1J^t for the limits and motor (1x1 matrix) mInverseMassMatrixLimitMotor = 0.0; if (mBody1->isMotionEnabled()) { mInverseMassMatrixLimitMotor += mA1.dot(mI1 * mA1); @@ -207,16 +208,19 @@ void HingeJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDat mInverseMassMatrixLimitMotor = (mInverseMassMatrixLimitMotor > 0.0) ? decimal(1.0) / mInverseMassMatrixLimitMotor : decimal(0.0); - // Compute the bias "b" of the lower limit constraint - mBLowerLimit = 0.0; - if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { - mBLowerLimit = biasFactor * lowerLimitError; - } + if (mIsLimitEnabled) { - // Compute the bias "b" of the upper limit constraint - mBUpperLimit = 0.0; - if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { - mBUpperLimit = biasFactor * upperLimitError; + // Compute the bias "b" of the lower limit constraint + mBLowerLimit = 0.0; + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBLowerLimit = biasFactor * lowerLimitError; + } + + // Compute the bias "b" of the upper limit constraint + mBUpperLimit = 0.0; + if (mPositionCorrectionTechnique == BAUMGARTE_JOINTS) { + mBUpperLimit = biasFactor * upperLimitError; + } } } } @@ -423,6 +427,7 @@ void HingeJoint::solveVelocityConstraint(const ConstraintSolverData& constraintS // --------------- Motor --------------- // + // If the motor is enabled if (mIsMotorEnabled) { // Compute J*v for the motor diff --git a/src/constraint/SliderJoint.cpp b/src/constraint/SliderJoint.cpp index 9a3cdf97..4a9764fc 100644 --- a/src/constraint/SliderJoint.cpp +++ b/src/constraint/SliderJoint.cpp @@ -184,6 +184,7 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa mBRotation = biasFactor * decimal(2.0) * qError.getVectorV(); } + // If the limits are enabled if (mIsLimitEnabled && (mIsLowerLimitViolated || mIsUpperLimitViolated)) { // Compute the inverse of the mass matrix K=JM^-1J^t for the limits (1x1 matrix) @@ -212,16 +213,20 @@ void SliderJoint::initBeforeSolve(const ConstraintSolverData& constraintSolverDa } } - // Compute the inverse of mass matrix K=JM^-1J^t for the motor (1x1 matrix) - mInverseMassMatrixMotor = 0.0; - if (mBody1->isMotionEnabled()) { - mInverseMassMatrixMotor += mBody1->getMassInverse(); + // If the motor is enabled + if (mIsMotorEnabled) { + + // Compute the inverse of mass matrix K=JM^-1J^t for the motor (1x1 matrix) + mInverseMassMatrixMotor = 0.0; + if (mBody1->isMotionEnabled()) { + mInverseMassMatrixMotor += mBody1->getMassInverse(); + } + if (mBody2->isMotionEnabled()) { + mInverseMassMatrixMotor += mBody2->getMassInverse(); + } + mInverseMassMatrixMotor = (mInverseMassMatrixMotor > 0.0) ? + decimal(1.0) / mInverseMassMatrixMotor : decimal(0.0); } - if (mBody2->isMotionEnabled()) { - mInverseMassMatrixMotor += mBody2->getMassInverse(); - } - mInverseMassMatrixMotor = (mInverseMassMatrixMotor > 0.0) ? - decimal(1.0) / mInverseMassMatrixMotor : decimal(0.0); // If warm-starting is not enabled if (!constraintSolverData.isWarmStartingActive) { diff --git a/src/engine/Profiler.cpp b/src/engine/Profiler.cpp index 2759976a..ad86b557 100644 --- a/src/engine/Profiler.cpp +++ b/src/engine/Profiler.cpp @@ -232,7 +232,7 @@ void Profiler::printRecursiveNodeReport(ProfileNodeIterator* iterator, (currentTotalTime / parentTime) * 100.0 : 0.0; for (int j=0; jgetCurrentName() << " : " << - fraction << " % | " << (currentTotalTime / long double(nbFrames)) << + fraction << " % | " << (currentTotalTime / (long double) (nbFrames)) << " ms/frame (" << iterator->getCurrentNbTotalCalls() << " calls)" << std::endl; totalTime += currentTotalTime; From 6e1521e1610d510440839eead877b69eede17b79 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Mon, 30 Sep 2013 23:55:11 +0200 Subject: [PATCH 64/66] Improve documentation --- documentation/API/Doxyfile | 2 +- .../UserManual/ReactPhysics3D-UserManual.tex | 336 +++++++++++++++++- documentation/UserManual/title.tex | 2 +- 3 files changed, 323 insertions(+), 17 deletions(-) diff --git a/documentation/API/Doxyfile b/documentation/API/Doxyfile index cd2b6b93..5373c5f0 100644 --- a/documentation/API/Doxyfile +++ b/documentation/API/Doxyfile @@ -32,7 +32,7 @@ PROJECT_NAME = "ReactPhysics3D" # This could be handy for archiving the generated documentation or # if some version control system is used. -PROJECT_NUMBER = "0.3.0" +PROJECT_NUMBER = "0.4.0" # Using the PROJECT_BRIEF tag one can provide an optional one line description # for a project that appears at the top of each page and should give viewer diff --git a/documentation/UserManual/ReactPhysics3D-UserManual.tex b/documentation/UserManual/ReactPhysics3D-UserManual.tex index 3ff12d20..79c9112e 100644 --- a/documentation/UserManual/ReactPhysics3D-UserManual.tex +++ b/documentation/UserManual/ReactPhysics3D-UserManual.tex @@ -192,27 +192,198 @@ \section{The Physics World} - The physics world will contain the bodies that you create and simulate them across time. + The physics world will contain the bodies and joints that you create. You will then be able run the simulation across time by updating the world. + The class \texttt{DynamicsWorld} represents the physics world in the ReactPhysics3D library. \subsection{Creating the Physics World} + The first thing you have to do when you want to simulate dynamics of rigid bodies in time with the ReactPhysics3D library is to create an instance + of the \texttt{DynamicsWorld}. You need to specify two parameters when constructing the world. The first one is the gravity acceleration vector (in $m / s^2$) in the world and + the second one is the simulation time step (in seconds). Note that gravity is activated by default when you create the world. The time step is the fixed amount of time that will be simulated + each time a simulation step will be perform when updating the world. For real-time application, a time step of $\frac{1}{60}$ seconds (60 Hz) is usually used. Using a smaller time step + makes the simulation more precise but also more expensive to compute. \\ + + Here is how to create the world : \\ + + \begin{lstlisting} + // Gravity vector + rp3d::Vector3 gravity(0.0, -9.81, 0.0); + + // Time step (in seconds) + rp3d::decimal timeStep = 1.0 / 60.0; + + // Create the dynamics world + rp3d::DynamicsWorld world(gravity, timeStep); + \end{lstlisting} + \subsection{Customizing the Physics World} + \subsubsection{Solver parameters} + + ReactPhysics3D uses an iterative solver to solve contacts and joints. For contacts, there is a unique velocity solver and for joints there are a velocity and a + position solver. By default, the number of iterations of the velocity solver is 10 and the number of iterations for the position solver is 5. It is possible to + change the number of iterations for both solvers. + + To do this, you need to use the following two methods : \\ + + \begin{lstlisting} + // Change the number of iterations of the velocity solver + world.setNbIterationsVelocitySolver(15); + + // Change the number of iterations of the position solver + world.setNbIterationsPositionSolver(8); + \end{lstlisting} + + \vspace{0.6cm} + + Increasing the number of iterations of the solvers will make the simulation more precise but also more expensive to compute. Therefore, you need to change + those values only if needed. + + \subsubsection{Sleeping} + + The purpose of the sleeping technique is to deactivate resting bodies so that they are not simulated anymore. This is used to save computation because simulating many bodies is costly. + A body (or group of bodies) is awaken as soon as another body collides with it or a joint in which it is involed is enabled. The sleeping technique is enabled by default. You can disable it + using the following method : \\ + + \begin{lstlisting} + // Disable the sleeping technique + world.enableSleeping(false); + \end{lstlisting} + + \vspace{0.6cm} + + Note that it is not recommended to disable the sleeping technique because the simulating will become slower. It is also possible to deactivate the sleeping technique on a + per body basis. + + // TODO : setSleepAngularVelocity and setSleepLinearVelocity, setTimeBeforeSleep() + \subsection{Updating the Physics World} + The first thing you have to do to simulate the dynamics of your world is to start the simulation using the following method : \\ + + \begin{lstlisting} + // Start the simulation + world.start(); + \end{lstlisting} + + \vspace{0.6cm} + + Then, each time you have to compute the next frame to render in your application, you need to update the state of the world. To do that, you simply need to call this method : \\ + + \begin{lstlisting} + // Update the world by taking a simulation step + world.update(); + \end{lstlisting} + + \vspace{0.6cm} + + When the \texttt{DynamicsWorld::update()} method is called, collision detection is performed and the position and orientation of the bodies are updated accordingly. + After updating the world, you will be able to get the updated position and orientation of the bodies to render them in the next frame. Make sure that you call + the \texttt{DynamicsWorld::start()} method before calling the \texttt{DynamicsWorld::update()} method. \\ + + You can also use the \texttt{DynamicsWorld::stop()} method to stop the simulation. You will then be able to start it again and to continue updating it. \\ + + Note that you can get the elapsed time (in seconds) from the beginning of the physics simulation using the \texttt{DynamicsWorld::getPhysicsTime()} method. This can be useful to + create some animations. + \subsection{Destroying the Physics World} + Do not forget to destroy the \texttt{DynamicsWorld} instance at the end of your program in order to release the allocated memory. If the object has been created statically, it will + automatically be destroy at the end of the scope in which it has been created. If the object has been created dynamically (using the \texttt{new} operator), you need to destroy + it with the \texttt{delete} operator. + \section{Rigid Bodies} + Once the physics world has been created, you can create rigid bodies into the world. A rigid body will represent the objects you want to simulate in the physics world. + A rigid body has a mass, a collision shape, a position and an orientation. The physics world will compute collision between the bodies and will update their position and + orientation accordingly at each time step. You can also create joints between the bodies in the world. In ReactPhysics3D, the class \texttt{RigidBody} is used to describe a rigid body. + \subsection{Creating a Rigid Body} + In order to create a rigid body, you need to specify a transform, its mass, its inertia tensor and a collision shape. The transform describes the initial + position and orientation of the body in the world. To do that, you need to create an instance of the \texttt{Transform} with a vector describing the + initial position and a quaternion for the initial orientation of the body. \\ + + In order that your rigid body can collide with other bodies in the world, you need to specify a collision shape. Take a look at section \ref{sec:collisionshapes} to learn about the + different collision shapes and how to create them. \\ + + To create a rigid body, you also need to give the mass of the body (in kilograms) and inertia tensor. The inertia tensor is a $3 \times 3$ matrix decribing how the mass is + distributed inside the rigid body which will be used to calculate the rotation of the body. The inertia tensor can be calculate from the collision shape that you have created for the + body. You can find more information about this in section \ref{sec:inertiacollisionshape}. \\ + + You need to call the \texttt{DynamicsWorld::createRigidBody()} method to create a rigid body in the world previously created. This method will return a pointer to the instance + of the \texttt{RigidBody} class that has been created internally. You will then be able to use that pointer to get or set values to the body. \\ + + You can see in the following code how to create a rigid body with a box collision shape : \\ + + \begin{lstlisting} + // Create the collision shape of the rigid body + const rp3d::BoxShape collisionShape(rp3d::Vector3(1.0, 1.0, 1.0)); + + // Compute the inertia tensor of the body + rp3d::Matrix3x3 inertiaTensor; + collisionShape.computeLocalInertiaTensor(inertiaTensor, mass); + + // Initial position and orientation of the rigid body + rp3d::Vector3 initPosition(0.0, 3.0, 0.0); + rp3d::Quaternion initOrientation = rp3d::Quaternion::identity(); + rp3d::Transform transform(initPosition, initOrientation); + + // Create a rigid body in the world + rp3d::RigidBody* body; + body = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, collisionShape); + \end{lstlisting} + \subsection{Customizing a Rigid Body} + TODO : Damping + \subsection{Updating a Rigid Body} + When you call the \texttt{DynamicsWorld::update()} method, the collision between the bodies are computed and the joints are evaluated. Then, the bodies position and orientation + are updated accordingly. After calling this method, you can get the updated position and orientation of each body to render it. To do that, you simply need to use the + \texttt{RigidBody::getInterpolatedTransform()} method to get the interpolated transform. This transform represents the current local-to-world-space transformation. \\ + + Here is how to get the interpolated transform of a rigid body : \\ + + \begin{lstlisting} + // Here, body is a RigidBody* pointer previously created + + // Get the interpolated transform of the rigid body + rp3d::Transform transform = body->getInterpolatedTransform(); + \end{lstlisting} + + \vspace{0.6cm} + + If you need the array with the corresponding $4 \times 4$ OpenGL transformation matrix, you can use the \texttt{Transform::getOpenGLMatrix()} method as in the following code : \\ + + \begin{lstlisting} + // Get the OpenGL matrix array of the transform + float matrix[16]; + transform.getOpenGLMatrix(matrix); + \end{lstlisting} + \subsection{Destroying a Rigid Body} + \begin{sloppypar} + It is really simple to destroy a rigid body. You simply need to use the \texttt{DynamicsWorld::destroyRigidBody()} method. You need to use the pointer to the body you + want to destroy as argument. Note that after calling that method, the pointer will not be valid anymore and therefore, you should not use it. Note that you must + destroy all the rigid bodies at the end of the simulation before you destroy the world. When you destroy a rigid body that was part of a joint, that joint will be automatically + destroyed as well. \\ + \end{sloppypar} + + Here is how to destroy a rigid body : \\ + + \begin{lstlisting} + // Here, world is an instance of the DynamicsWorld class + // and body is a RigidBody* pointer + + // Destroy the rigid body + world.destroyRigidBody(body); + \end{lstlisting} + \section{Collision Shapes} + \label{sec:collisionshapes} When you create a rigid body, you need to specify a collision shape. This shape will be used to test collision between the body and its environment. This section describes all the collision shapes available in the ReactPhysics3D library and how to use them. \\ @@ -426,8 +597,9 @@ \end{lstlisting} \subsection{Inertia Tensor of a Collision Shape} + \label{sec:inertiacollisionshape} - When you create a rigid body, you need to specify its inertia tensor. The inertia tensor is a $3 \times 3$ matrix decribing how the mass is distributed inside the rigid body and this + When you create a rigid body, you need to specify its inertia tensor. The inertia tensor is a $3 \times 3$ matrix decribing how the mass is distributed inside the rigid body which will be used to calculate the rotation of the body. The inertia tensor depends on the mass and the shape of the body. \\ You can use the collision shape of a rigid body to compute its inertia tensor. To do that, you need to use the \texttt{CollisionShape::computeLocalInertiaTensor()} method of your collision @@ -478,7 +650,7 @@ \begin{lstlisting} // Create the joint in the dynamics world rp3d::BallAndSocketJoint* joint; - joint = dynamic_cast(world->createJoint(jointInfo)); + joint = dynamic_cast(world.createJoint(jointInfo)); \end{lstlisting} \vspace{0.6cm} @@ -516,11 +688,9 @@ \begin{lstlisting} // Create the hinge joint in the dynamics world rp3d::HingeJoint* joint; - joint = dynamic_cast(world->createJoint(jointInfo)); + joint = dynamic_cast(world.createJoint(jointInfo)); \end{lstlisting} - \vspace{0.6cm} - \subsubsection{Limits} With the hinge joint, you can constraint the motion range using limits. The limits of the hinge joint are the minimum and maximum angle of rotation allowed with respect to the initial @@ -546,18 +716,20 @@ // Create the hinge joint in the dynamics world rp3d::HingeJoint* joint; - joint = dynamic_cast(world->createJoint(jointInfo)); + joint = dynamic_cast(world.createJoint(jointInfo)); \end{lstlisting} \vspace{0.6cm} - You can also use the \texttt{HingeJoint::enableLimit()}, \texttt{HingeJoint::setMinAngleLimit()} and \texttt{HingeJoint::setMaxAngleLimit()} to specify the limits of the joint after its - creation. See the API documentation for more information. + \begin{sloppypar} + It is also possible to use the \texttt{HingeJoint::enableLimit()}, \texttt{HingeJoint::setMinAngleLimit()} and \texttt{HingeJoint::setMaxAngleLimit()} methods to specify + the limits of the joint after its creation. See the API documentation for more information. + \end{sloppypar} \subsubsection{Motor} A motor is also available for the hinge joint. It can be used rotate the bodies around the hinge axis at a given angular speed and such that the torque applied to - rotate the bodies does not exceed the maximum allowed torque. The motor is disabled by default. If you want to use it, you first have to activate it using the + rotate the bodies does not exceed a maximum allowed torque. The motor is disabled by default. If you want to use it, you first have to activate it using the \texttt{isMotorEnabled} boolean variable of the \texttt{HingeJointInfo} object before you create the joint. Then, you need to specify the angular motor speed (in radians/seconds) using the \texttt{motorSpeed} variable and also the maximum allowed torque (in Newton $\cdot$ meters) with the \texttt{maxMotorTorque} variable. \\ @@ -578,18 +750,148 @@ // Create the hinge joint in the dynamics world rp3d::HingeJoint* joint; - joint = dynamic_cast(world->createJoint(jointInfo)); + joint = dynamic_cast(world.createJoint(jointInfo)); \end{lstlisting} \vspace{0.6cm} - You can also use the \texttt{HingeJoint::enableMotor()}, \texttt{HingeJoint::setMotorSpeed()} and \texttt{HingeJoint::setMaxMotorTorque()} to enabled the motor of the joint after its - creation. See the API documentation for more information. + \begin{sloppypar} + It is also possible to use the \texttt{HingeJoint::enableMotor()}, \texttt{HingeJoint::setMotorSpeed()} and \texttt{HingeJoint::setMaxMotorTorque()} methods to + enable the motor of the joint after its creation. See the API documentation for more information. + \end{sloppypar} \subsection{Slider Joint} + The class \texttt{SliderJoint} describes a slider joint (or prismatic joint) that only allows relative translation along a single direction. It has a single degree of freedom and allows no + relative rotation. In order to create a slider joint, you first need to specify the anchor point (in world-space) and the slider axis direction (in world-space). The constructor of the + \texttt{SliderJointInfo} object needs two pointer to the bodies of the joint, the anchor point and the axis direction. Note that the two bodies have to be in a correct initial position when + the joint is created. \\ + + You can see in the following code how to specify the information to create a slider joint : \\ + + \begin{lstlisting} + // Anchor point in world-space + const rp3d::Vector3 anchorPoint = rp3d::decimal(0.5) * (body2Position + body1Position); + + // Slider axis in world-space + const rp3d::Vector3 axis = (body2Position - body1Position); + + // Create the joint info object + rp3d::SliderJointInfo jointInfo(body1, body2, anchorPoint, axis); + \end{lstlisting} + + \vspace{0.6cm} + + Now, it is possible to create the actual joint in the dynamics world using the \texttt{DynamicsWorld::createJoint()} method. + Note that this method will also return a pointer to the \texttt{SliderJoint} object that has been created internally. You will then + be able to use that pointer to change properties of the joint and also to destroy it at the end. \\ + + Here is how to create the joint in the world : \\ + + \begin{lstlisting} + // Create the slider joint in the dynamics world + rp3d::SliderJoint* joint; + joint = dynamic_cast(world.createJoint(jointInfo)); + \end{lstlisting} + + \subsubsection{Limits} + + It is also possible to control the range of the slider joint motion using limits. The limits are disabled by default. In order to use the limits when the joint is created, you first + need to activate them using the \texttt{isLimitEnabled} variable of the \texttt{SliderJointInfo} class. Then, you need to specify the minimum and maximum translation limits + (in meters) using the \texttt{minTranslationLimit} and \texttt{maxTranslation\-Limit} variables. Note that the initial position of the two bodies when the joint is created + corresponds to a translation of zero. Therefore, the minimum limit must be smaller or equal to zero and the maximum limit must be larger or equal to zero. \\ + + You can see in the following example how to set the limits when the slider joint is created : \\ + + \begin{lstlisting} + // Create the joint info object + rp3d::SliderJointInfo jointInfo(body1, body2, anchorPoint, axis); + + // Enable the limits of the joint + jointInfo.isLimitEnabled = true; + + // Minimum translation limit + jointInfo.minTranslationLimit = -1.7; + + // Maximum translation limit + jointInfo.maxTranslationLimit = 1.7; + + // Create the hinge joint in the dynamics world + rp3d::SliderJoint* joint; + joint = dynamic_cast(world.createJoint(jointInfo)); + \end{lstlisting} + + \vspace{0.6cm} + + \begin{sloppypar} + You can also use the \texttt{SliderJoint::enableLimit()}, \texttt{SliderJoint::\-setMinTranslationLimit()} and \texttt{SliderJoint::setMaxTranslationLimit()} methods to enable the + limits of the joint after its creation. See the API documentation for more information. + \end{sloppypar} + + \subsubsection{Motor} + + The slider joint also has a motor. You can use it to translate the bodies along the slider axis at a given linear speed and such that the force applied to + move the bodies does not exceed a maximum allowed force. The motor is disabled by default. If you want to use it when the joint is created, you first have to activate it using the + \texttt{isMotorEnabled} boolean variable of the \texttt{SliderJointInfo} object before you create the joint. Then, you need to specify the linear motor speed (in meters/seconds) + using the \texttt{motorSpeed} variable and also the maximum allowed force (in Newtons) with the \texttt{maxMotorForce} variable. \\ + + For instance, here is how to enable the motor of the slider joint when the joint is created : \\ + + \begin{lstlisting} + // Create the joint info object + rp3d::SliderJointInfo jointInfo(body1, body2, anchorPoint, axis); + + // Enable the motor of the joint + jointInfo.isMotorEnabled = true; + + // Motor linear speed + jointInfo.motorSpeed = 2.0; + + // Maximum allowed force + jointInfo.maxMotorForce = 10.0; + + // Create the slider joint in the dynamics world + rp3d::SliderJoint* joint; + joint = dynamic_cast(world.createJoint(jointInfo)); + \end{lstlisting} + + \vspace{0.6cm} + + \begin{sloppypar} + It is also possible to use the \texttt{SliderJoint::enableMotor()}, \texttt{SliderJoint::setMotorSpeed()} and \texttt{SliderJoint::setMaxMotorForce()} methods to enable the + motor of the joint after its creation. See the API documentation for more information. + \end{sloppypar} + \subsection{Fixed Joint} + The class \texttt{FixedJoint} describes a fixed joint between two bodies. In a fixed joint, there is no degree of freedom, the bodies are not allowed to translate + or rotate with respect to each other. In order to create a fixed joint, you simply need to specify an anchor point (in world-space) to create the \texttt{FixedJointInfo} + object. \\ + + For instance, here is how to create the joint info object for a fixed joint : \\ + + \begin{lstlisting} + // Anchor point in world-space + rp3d::Vector3 anchorPoint(2.0, 3.0, 4.0); + + // Create the joint info object + rp3d::FixedJointInfo jointInfo1(body1, body2, anchorPoint); + \end{lstlisting} + + \vspace{0.6cm} + + Now, it is possible to create the actual joint in the dynamics world using the \texttt{DynamicsWorld::createJoint()} method. + Note that this method will also return a pointer to the \texttt{FixedJoint} object that has been created internally. You will then + be able to use that pointer to change properties of the joint and also to destroy it at the end. \\ + + Here is how to create the joint in the world : \\ + + \begin{lstlisting} + // Create the fixed joint in the dynamics world + rp3d::FixedJoint* joint; + joint = dynamic_cast(world.createJoint(jointInfo)); + \end{lstlisting} + \subsection{Collision between the bodies of a Joint} By default the two bodies involved in a joint are able to collide with each other. However, it is possible to disable the collision between the two bodies that are part @@ -607,7 +909,7 @@ // Create the joint in the dynamics world rp3d::HingeJoint* joint; - joint = dynamic_cast(world->createJoint(jointInfo)); + joint = dynamic_cast(world.createJoint(jointInfo)); \end{lstlisting} \subsection{Destroying a Joint} @@ -620,7 +922,7 @@ // created joint // Destroy the joint - world->destroyJoint(joint); + world.destroyJoint(joint); \end{lstlisting} \vspace{0.6cm} @@ -652,6 +954,10 @@ In this example, you will learn how to create different joints (Ball and Socket, Hinge, Slider, Fixed) into the dynamics world. You can also see how to set the motor or limits of the joints. + \section{Receiving Feedback} + + \subsection{Contacts} + \section{Profiler} \section{API Documentation} diff --git a/documentation/UserManual/title.tex b/documentation/UserManual/title.tex index b9379893..506eb65b 100644 --- a/documentation/UserManual/title.tex +++ b/documentation/UserManual/title.tex @@ -10,7 +10,7 @@ \vskip 1.3cm {\Huge \@title\par}% \vskip 0.3cm - {\Large Version: 0.3.0\par}% + {\Large Version: 0.4.0\par}% \vskip 0.3cm {\Large \@author\par}% \vskip 2cm From 7432a871496e185b793351a7c251b45b7bf68424 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sun, 6 Oct 2013 16:03:25 +0200 Subject: [PATCH 65/66] Modify cmake files and add comments in the code --- CMakeLists.txt | 8 +++++++- VERSION | 2 +- examples/collisionshapes/CMakeLists.txt | 7 +++++-- examples/cubes/CMakeLists.txt | 7 +++++-- examples/joints/CMakeLists.txt | 7 +++++-- src/body/RigidBody.h | 4 ++-- src/engine/Material.h | 12 ++++++++---- 7 files changed, 33 insertions(+), 14 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 65ce371a..f538a201 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -4,8 +4,14 @@ CMAKE_MINIMUM_REQUIRED(VERSION 2.6) # Project configuration PROJECT(REACTPHYSICS3D) +# Default build type +SET(CMAKE_BUILD_TYPE "Debug") + # Where to build the library -SET(LIBRARY_OUTPUT_PATH lib/) +SET(LIBRARY_OUTPUT_PATH "lib") + +# Where to build the executables +SET(OUR_EXECUTABLE_OUTPUT_PATH "${PROJECT_BINARY_DIR}/bin") # Options OPTION(COMPILE_EXAMPLES "Select this if you want to build the examples" OFF) diff --git a/VERSION b/VERSION index 9325c3cc..60a2d3e9 100644 --- a/VERSION +++ b/VERSION @@ -1 +1 @@ -0.3.0 \ No newline at end of file +0.4.0 \ No newline at end of file diff --git a/examples/collisionshapes/CMakeLists.txt b/examples/collisionshapes/CMakeLists.txt index ffbdebbe..2bd5a930 100644 --- a/examples/collisionshapes/CMakeLists.txt +++ b/examples/collisionshapes/CMakeLists.txt @@ -4,8 +4,11 @@ cmake_minimum_required(VERSION 2.6) # Project configuration PROJECT(CollisionShapes) -# Where to build the executable -SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin/collisionshapes/) +# Where to build the executables +SET(EXECUTABLE_OUTPUT_PATH "${OUR_EXECUTABLE_OUTPUT_PATH}/collisionshapes") +SET(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${EXECUTABLE_OUTPUT_PATH}) +SET(CMAKE_RUNTIME_OUTPUT_DIRECTORY_RELEASE ${EXECUTABLE_OUTPUT_PATH}) +SET(CMAKE_RUNTIME_OUTPUT_DIRECTORY_DEBUG ${EXECUTABLE_OUTPUT_PATH}) # Copy the shaders used for the demo into the build directory FILE(COPY "${OPENGLFRAMEWORK_DIR}/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") diff --git a/examples/cubes/CMakeLists.txt b/examples/cubes/CMakeLists.txt index f6b44f3a..efdcf263 100644 --- a/examples/cubes/CMakeLists.txt +++ b/examples/cubes/CMakeLists.txt @@ -4,8 +4,11 @@ cmake_minimum_required(VERSION 2.6) # Project configuration PROJECT(Cubes) -# Where to build the executable -SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin/cubes/) +# Where to build the executables +SET(EXECUTABLE_OUTPUT_PATH "${OUR_EXECUTABLE_OUTPUT_PATH}/cubes") +SET(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${EXECUTABLE_OUTPUT_PATH}) +SET(CMAKE_RUNTIME_OUTPUT_DIRECTORY_RELEASE ${EXECUTABLE_OUTPUT_PATH}) +SET(CMAKE_RUNTIME_OUTPUT_DIRECTORY_DEBUG ${EXECUTABLE_OUTPUT_PATH}) # Copy the shaders used for the demo into the build directory FILE(COPY "${OPENGLFRAMEWORK_DIR}/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") diff --git a/examples/joints/CMakeLists.txt b/examples/joints/CMakeLists.txt index 9b4f71e6..343f5c7d 100644 --- a/examples/joints/CMakeLists.txt +++ b/examples/joints/CMakeLists.txt @@ -4,8 +4,11 @@ cmake_minimum_required(VERSION 2.6) # Project configuration PROJECT(Joints) -# Where to build the executable -SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin/joints/) +# Where to build the executables +SET(EXECUTABLE_OUTPUT_PATH "${OUR_EXECUTABLE_OUTPUT_PATH}/joints") +SET(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${EXECUTABLE_OUTPUT_PATH}) +SET(CMAKE_RUNTIME_OUTPUT_DIRECTORY_RELEASE ${EXECUTABLE_OUTPUT_PATH}) +SET(CMAKE_RUNTIME_OUTPUT_DIRECTORY_DEBUG ${EXECUTABLE_OUTPUT_PATH}) # Copy the shaders used for the demo into the build directory FILE(COPY "${OPENGLFRAMEWORK_DIR}/src/shaders/" DESTINATION "${EXECUTABLE_OUTPUT_PATH}/shaders/") diff --git a/src/body/RigidBody.h b/src/body/RigidBody.h index 4b7980e8..cbd123e1 100644 --- a/src/body/RigidBody.h +++ b/src/body/RigidBody.h @@ -188,7 +188,7 @@ class RigidBody : public CollisionBody { /// Apply an external force to the body at its gravity center. void applyForceToCenter(const Vector3& force); - /// Apply an external force to the body at a given point (in world-coordinates). + /// Apply an external force to the body at a given point (in world-space coordinates). void applyForce(const Vector3& force, const Vector3& point); /// Apply an external torque to the body. @@ -361,7 +361,7 @@ inline void RigidBody::applyForceToCenter(const Vector3& force) { mExternalForce += force; } -// Apply an external force to the body at a given point (in world-coordinates). +// Apply an external force to the body at a given point (in world-space coordinates). /// If the point is not at the center of gravity of the body, it will also /// generate some torque and therefore, change the angular velocity of the body. /// If the body is sleeping, calling this method will wake it up. Note that the diff --git a/src/engine/Material.h b/src/engine/Material.h index 4ab4a9c7..4d06f1a9 100644 --- a/src/engine/Material.h +++ b/src/engine/Material.h @@ -66,13 +66,13 @@ class Material { /// Return the bounciness decimal getBounciness() const; - /// Set the bounciness + /// Set the bounciness. void setBounciness(decimal bounciness); /// Return the friction coefficient decimal getFrictionCoefficient() const; - /// Set the friction coefficient + /// Set the friction coefficient. void setFrictionCoefficient(decimal frictionCoefficient); /// Overloaded assignment operator @@ -84,7 +84,9 @@ inline decimal Material::getBounciness() const { return mBounciness; } -// Set the bounciness +// Set the bounciness. +/// The bounciness should be a value between 0 and 1. The value 1 is used for a +/// very bouncy body and zero is used for a body that is not bouncy at all. inline void Material::setBounciness(decimal bounciness) { assert(bounciness >= decimal(0.0) && bounciness <= decimal(1.0)); mBounciness = bounciness; @@ -95,7 +97,9 @@ inline decimal Material::getFrictionCoefficient() const { return mFrictionCoefficient; } -// Set the friction coefficient +// Set the friction coefficient. +/// The friction coefficient has to be a positive value. The value zero is used for no +/// friction at all. inline void Material::setFrictionCoefficient(decimal frictionCoefficient) { assert(frictionCoefficient >= decimal(0.0)); mFrictionCoefficient = frictionCoefficient; From bd40e49704adb1c2074981eaf7afcb9794005e38 Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Sun, 6 Oct 2013 18:46:32 +0200 Subject: [PATCH 66/66] Improve the documentation --- .../UserManual/ReactPhysics3D-UserManual.pdf | Bin 0 -> 475624 bytes .../UserManual/ReactPhysics3D-UserManual.tex | 249 ++++++++++++++---- 2 files changed, 205 insertions(+), 44 deletions(-) create mode 100644 documentation/UserManual/ReactPhysics3D-UserManual.pdf diff --git a/documentation/UserManual/ReactPhysics3D-UserManual.pdf b/documentation/UserManual/ReactPhysics3D-UserManual.pdf new file mode 100644 index 0000000000000000000000000000000000000000..c6edca836837fcc5e6711bb12bc85a769b853a13 GIT binary patch literal 475624 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

_ij;|MUXqmxMQY9;KhU+m#g!4Zh=oDh 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

J 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

fFf!lYos$)66G5JIWSi&LV;E!VqEEi8g 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 QuK-vOn~_mgT@L$y00=sa`v3p{ literal 0 HcmV?d00001 diff --git a/documentation/UserManual/ReactPhysics3D-UserManual.tex b/documentation/UserManual/ReactPhysics3D-UserManual.tex index 79c9112e..1fe9f83d 100644 --- a/documentation/UserManual/ReactPhysics3D-UserManual.tex +++ b/documentation/UserManual/ReactPhysics3D-UserManual.tex @@ -81,8 +81,12 @@ \url{http://www.cmake.org} or using you package-management program (apt, yum, \dots) on Linux. Then, you will be able to compile the library to create the static library file. In order to use ReactPhysics3D in your application, you can link your program with this static library. - If you have never runned cmake, you should read the page \url{http://www.cmake.org/cmake/help/runningcmake.html} as - it contains many useful information. + If you have never used cmake before, you should read the page \url{http://www.cmake.org/cmake/help/runningcmake.html} as + it contains many useful information. \\ + + Note that by default, the library is built in \emph{debugging} mode. In this mode, a lot of debugging information is compiled together with the code. This might cause the application to + run much slower that it should be in \emph{release} mode. Therefore, you should not forget to build the library in \emph{release} mode when releasing your final + application. \subsection{CMake using the command line (Linux and Mac OS X)} @@ -118,7 +122,11 @@ Now, if you go into the folder you have chosen to build the library, you should be able to open the project file that corresponds to your IDE and compile - the library. + the library. \\ + + If your want to run the examples within the Microsoft Visual Studio IDE, you need to make sure that in the + \emph{Debugging} section of the \emph{Configuration Properties} of the example projects, the \emph{Working Directory} is set to \texttt{\$(OutDir)}. + Otherwise, you might have problems to run the examples. \subsection{CMake Variables} \label{sec:cmakevariables} @@ -126,28 +134,28 @@ You can find bellow the different CMake variables that you can set before generating the makefiles. \begin{description} - \item[CMAKE\_BUILD\_TYPE] If this variable is set to DEBUG, the library will be compiled in debugging mode. + \item[CMAKE\_BUILD\_TYPE] If this variable is set to \texttt{Debug}, the library will be compiled in debugging mode. This mode should be used during development stage to know where things might crash. In debugging mode, the library might run a bit slow due to all the debugging information - that are used. However, if this variable is set to RELEASE, no debugging information is stored + that are used. However, if this variable is set to \texttt{Release}, no debugging information is stored and therefore, it will run much faster. This mode must be used when you compile for the final release of you application. - \item[COMPILE\_EXAMPLES] If this variable is ON, the examples of the reactphysics3d library will be compiled. + \item[COMPILE\_EXAMPLES] If this variable is \texttt{ON}, the examples of the reactphysics3d library will be compiled. Note that you will need to have the Freeglut library installed on your system if you use Windows or Linux and you will need to have the Glut library on Mac OS X if you want to run those examples. - \item[COMPILE\_TESTS] If this variable is ON, the unit tests of the reactphysics3d library will be compiled. You will then + \item[COMPILE\_TESTS] If this variable is \texttt{ON}, the unit tests of the reactphysics3d library will be compiled. You will then be able to launch the tests to make sure that they are running fine on your system. - \item[PROFILING\_ENABLED] If this variable is ON, the integrated profiler will collect data will the application is running - and the profiling report will be display on the console at the end of the application (in the - destructor of the DynamicsWorld). This might be useful to see what part of the reactphysics3d - library takes time during its execution. This variable must be set to OFF when you compile + \item[PROFILING\_ENABLED] If this variable is \texttt{ON}, the integrated profiler will collect data while the application is running + and the profiling report will be displayed in the console at the end of the application (in the + destructor of the \texttt{DynamicsWorld} class). This might be useful to see what part of the reactphysics3d + library takes time during its execution. This variable must be set to \texttt{OFF} when you compile for the final release of your application. - \item[DOUBLE\_PRECISION\_ENABLED] If this variable is ON, the reactphysics3d library will be compile with double floating point precision. + \item[DOUBLE\_PRECISION\_ENABLED] If this variable is \texttt{ON}, the reactphysics3d library will be compile with double floating point precision. Otherwise, the library will be compile with single precision. \end{description} @@ -197,7 +205,7 @@ \subsection{Creating the Physics World} - The first thing you have to do when you want to simulate dynamics of rigid bodies in time with the ReactPhysics3D library is to create an instance + The first thing you have to do when you want to simulate the dynamics of rigid bodies in time with the ReactPhysics3D library is to create an instance of the \texttt{DynamicsWorld}. You need to specify two parameters when constructing the world. The first one is the gravity acceleration vector (in $m / s^2$) in the world and the second one is the simulation time step (in seconds). Note that gravity is activated by default when you create the world. The time step is the fixed amount of time that will be simulated each time a simulation step will be perform when updating the world. For real-time application, a time step of $\frac{1}{60}$ seconds (60 Hz) is usually used. Using a smaller time step @@ -220,9 +228,9 @@ \subsubsection{Solver parameters} - ReactPhysics3D uses an iterative solver to solve contacts and joints. For contacts, there is a unique velocity solver and for joints there are a velocity and a + ReactPhysics3D uses an iterative solver to solve the contacts and joints. For contacts, there is a unique velocity solver and for joints there are a velocity and a position solver. By default, the number of iterations of the velocity solver is 10 and the number of iterations for the position solver is 5. It is possible to - change the number of iterations for both solvers. + change the number of iterations for both solvers. \\ To do this, you need to use the following two methods : \\ @@ -240,10 +248,11 @@ those values only if needed. \subsubsection{Sleeping} + \label{sec:sleeping} - The purpose of the sleeping technique is to deactivate resting bodies so that they are not simulated anymore. This is used to save computation because simulating many bodies is costly. - A body (or group of bodies) is awaken as soon as another body collides with it or a joint in which it is involed is enabled. The sleeping technique is enabled by default. You can disable it - using the following method : \\ + The purpose of the sleeping technique is to deactivate resting bodies so that they are not simulated anymore. This is used to save computation time because simulating many bodies is costly. + A sleeping body (or group of sleeping bodies) is awaken as soon as another body collides with it or a joint in which it is involed is enabled. The sleeping technique is enabled by + default. You can disable it using the following method : \\ \begin{lstlisting} // Disable the sleeping technique @@ -252,10 +261,15 @@ \vspace{0.6cm} - Note that it is not recommended to disable the sleeping technique because the simulating will become slower. It is also possible to deactivate the sleeping technique on a - per body basis. + Note that it is not recommended to disable the sleeping technique because the simulation will become slower. It is also possible to deactivate the sleeping technique on a + per body basis. See section \ref{sec:rigidbodysleeping} for more information. \\ - // TODO : setSleepAngularVelocity and setSleepLinearVelocity, setTimeBeforeSleep() + \begin{sloppypar} + A body is put to sleep when its linear and angular velocity stay under a given velocity threshold for a certain amount of time (one second by default). It is possible to change the two + linear and angular velocity thresholds using the two methods \texttt{DynamicsWorld::setSleepLinearVelocity()} and \texttt{Dynamics::setSleepAngularVelocity()}. Note that the velocities must + be specified in meters per second. You can also change the amount of time (in seconds) the velocity of a body needs to stay under the threshold to be considered sleeping. To do this, use the + \texttt{DynamicsWorld::setTimeBeforeSleep()} method. + \end{sloppypar} \subsection{Updating the Physics World} @@ -294,25 +308,25 @@ \section{Rigid Bodies} - Once the physics world has been created, you can create rigid bodies into the world. A rigid body will represent the objects you want to simulate in the physics world. + Once the physics world has been created, you can create rigid bodies into the world. A rigid body will represent an object you want to simulate in the physics world. A rigid body has a mass, a collision shape, a position and an orientation. The physics world will compute collision between the bodies and will update their position and orientation accordingly at each time step. You can also create joints between the bodies in the world. In ReactPhysics3D, the class \texttt{RigidBody} is used to describe a rigid body. \subsection{Creating a Rigid Body} - In order to create a rigid body, you need to specify a transform, its mass, its inertia tensor and a collision shape. The transform describes the initial - position and orientation of the body in the world. To do that, you need to create an instance of the \texttt{Transform} with a vector describing the + In order to create a rigid body, you need to specify its transform, its mass, its inertia tensor and a collision shape. The transform describes the initial + position and orientation of the body in the world. You need to create an instance of the \texttt{Transform} with a vector describing the initial position and a quaternion for the initial orientation of the body. \\ In order that your rigid body can collide with other bodies in the world, you need to specify a collision shape. Take a look at section \ref{sec:collisionshapes} to learn about the different collision shapes and how to create them. \\ - To create a rigid body, you also need to give the mass of the body (in kilograms) and inertia tensor. The inertia tensor is a $3 \times 3$ matrix decribing how the mass is - distributed inside the rigid body which will be used to calculate the rotation of the body. The inertia tensor can be calculate from the collision shape that you have created for the + To create a rigid body, you also need to give the mass of the body (in kilograms) and its inertia tensor. The inertia tensor is a $3 \times 3$ matrix decribing how the mass is + distributed inside the rigid body which will be used to calculate the rotation of the body. The inertia tensor can be calculated from the collision shape that you have created for the body. You can find more information about this in section \ref{sec:inertiacollisionshape}. \\ You need to call the \texttt{DynamicsWorld::createRigidBody()} method to create a rigid body in the world previously created. This method will return a pointer to the instance - of the \texttt{RigidBody} class that has been created internally. You will then be able to use that pointer to get or set values to the body. \\ + of the \texttt{RigidBody} class that has been created internally. You will then be able to use that pointer to get or set values of the body. \\ You can see in the following code how to create a rigid body with a box collision shape : \\ @@ -331,12 +345,134 @@ // Create a rigid body in the world rp3d::RigidBody* body; - body = dynamicsWorld->createRigidBody(transform, mass, inertiaTensor, collisionShape); + body = dynamicsWorld.createRigidBody(transform, mass, inertiaTensor, collisionShape); \end{lstlisting} \subsection{Customizing a Rigid Body} - TODO : Damping + Once a rigid body has been created, you can change some of its properties. + + \subsubsection{Static Rigid Body} + + \begin{sloppypar} + By default, the bodies you create in the world are not static. If the rigid body is static and is not supposed to move, you need to specify it using the \texttt{RigidBody::enableMotion()} + method as follows : \\ + \end{sloppypar} + + \begin{lstlisting} + // Specify that the body cannot move + rigidBody->enableMotion(false); + \end{lstlisting} + + \subsubsection{Gravity} + + By default, all the rigid bodies with react to the gravity force of the world. If you do not want the gravity to be applied to a given body, you can disable + it using the \texttt{RigidBody::enableGravity()} method as in the following example : \\ + + \begin{lstlisting} + // Disable gravity for this body + rigidBody->enableGravity(false); + \end{lstlisting} + + \subsubsection{Material of a Rigid Body} + + The material of a rigid body is used to describe its different physical properties. The class \texttt{Material} represents the material of a body. Each body that + you create will have a default material. You can get the material of the rigid body using the \texttt{RigidBody::getMaterial()} method. Then, you will be able to change some + properties. \\ + + For instance, you can change the bounciness of the rigid body. The bounciness is a value between 0 and 1. The value 1 is used for a very bouncy object and the value 0 means that + the body will not be bouncy at all. To change the bounciness of the material, you can use the \texttt{Material::setBounciness()} method. \\ + + You are also able to change the friction coefficient of the body. This value needs to be between 0 and 1. If the value is 0, no friction will be applied when the body is in contact with + another body. However, if the value is 1, the friction force will be high. You can change the friction coefficient of the material with the + \texttt{Material::setFrictionCoefficient()} method. \\ + + Here is how to get the material of a rigid body and how to modify some of its properties : \\ + + \begin{lstlisting} + // Get the current material of the body + rp3d::Material& material = rigidBody->getMaterial(); + + // Change the bounciness of the body + material.setBounciness(rp3d::decimal(0.4)); + + // Change the friction coefficient of the body + material.setFrictionCoefficient(rp3d::decimal(0.2)); + \end{lstlisting} + + \subsubsection{Velocity Damping} + + \begin{sloppypar} + Damping is the effect of reducing the velocity of the rigid body during the simulation. By default, no damping is applied. However, you can choose to damp + the linear or/and the angular velocity of a rigid body. For instance, without angular damping a pendulum will never come to rest. You need to use the + \texttt{RigidBody::setLinearDamping()} and \texttt{RigidBody::setAngularDamping()} methods to change the damping values. The damping value has to be positive and + a value of zero means no damping at all. + \end{sloppypar} + + \subsubsection{Sleeping} + \label{sec:rigidbodysleeping} + + As described in section \ref{sec:sleeping}, the sleeping technique is used to disable the simulation of the resting bodies. By default the bodies are allowed to sleep when they + come to rest. However, if you do not want a given body to be put to sleep, you can use the \texttt{Body::setIsAllowedToSleep()} method as in the next example : \\ + + \begin{lstlisting} + // This rigid body cannot sleep + rigidBody->setIsAllowedToSleep(false); + \end{lstlisting} + + \subsubsection{Applying Force or Torque to a Rigid Body} + + During the simulation, you can apply a force or a torque to a given rigid body. First, you can apply a force to the center of mass of the rigid body using the + \texttt{RigidBody::applyForceToCenter()} method. You need to specifiy the force vector (in Newton) as a parameter. If the force is applied to the center of mass, no + torque will be created and only the linear motion of the body will be affected. \\ + + \begin{lstlisting} + // Force vector (in Newton) + rp3d::Vector3 force(2.0, 0.0, 0.0); + + // Apply a force to the center of the body + rigidBody->applyForceToCenter(force); + \end{lstlisting} + + \vspace{0.6cm} + + \begin{sloppypar} + You can also apply a force to any given point (in world-space) using the \texttt{RigidBody::applyForce()} method. You need to specify the force vector (in Newton) and the point + (in world-space) where to apply the given force. Note that if the point is not the center of mass of the body, applying a force will generate some torque and therefore, the + angular motion of the body will be affected as well. \\ + \end{sloppypar} + + \begin{lstlisting} + // Force vector (in Newton) + rp3d::Vector3 force(2.0, 0.0, 0.0); + + // Point where the force is applied + rp3d::Vector3 point(4.0, 5.0, 6.0); + + // Apply a force to the body + rigidBody->applyForce(force, point); + \end{lstlisting} + + \vspace{0.6cm} + + \begin{sloppypar} + It is also possible to apply a torque to a given body using the \texttt{RigidBody::applyTorque()} method. You simply need to specify the torque vector (in Newton $\cdot$ meter) as + in the following example : \\ + \end{sloppypar} + + \begin{lstlisting} + // Torque vector + rp3d::Vector3 torque(0.0, 3.0, 0.0); + + // Apply a torque to the body + rigidBody->applyTorque(torque); + \end{lstlisting} + + \vspace{0.6cm} + + Note that when you call the previous methods, the specified force/torque will be added to the total force/torque applied to the rigid body and that at the end of each call to the + \texttt{DynamicsWorld::update()}, the total force/torque of all the rigid bodies will be reset to zero. Therefore, you might need to call the previous methods during several frames + if you want the force/torque to be applied during a certain amount of time. \subsection{Updating a Rigid Body} @@ -379,7 +515,7 @@ // and body is a RigidBody* pointer // Destroy the rigid body - world.destroyRigidBody(body); + world.destroyRigidBody(body); \end{lstlisting} \section{Collision Shapes} @@ -394,7 +530,7 @@ for this small margin with the way you render the object. \\ Once you have created a collision shape object, you need to used it when you create a rigid body in the physics world using the - \texttt{DynamicsWorld::createRigidBody()} method. Note that during the rigid body creating, the collision shape object that you gave as a parameter + \texttt{DynamicsWorld::createRigidBody()} method. Note that during the rigid body creation, the collision shape object that you gave as a parameter will be copied internally. Therefore, you can destroy the collision shape object right after the rigid body creation. \subsection{Box Shape} @@ -439,7 +575,7 @@ \label{fig:sphereshape} \end{figure} - The \texttt{SphereShape} class describes a sphere collision shape centered at the origin of the body local space. You only need to specify the radius of sphere to create it. \\ + The \texttt{SphereShape} class describes a sphere collision shape centered at the origin of the body local space. You only need to specify the radius of the sphere to create it. \\ For instance, if you want to create a sphere shape with a radius of 2 meters, you need to use the following code : \\ @@ -548,7 +684,7 @@ The class \texttt{ConvexMeshShape} can be used to describe the shape of a convex mesh. In order to create a convex mesh shape, you need to supply the array with the coordinates of the vertices of the mesh. The array is supposed to start with the three X, Y and Z coordinates of the first vertex, then the X, Y and Z coordinates of the second vertex and so on. - The first parameter of the \texttt{ConvexMeshShape} constructor is a pointer to the array of vertices coordinates, the second parameter is the number of vertices in the array and + The first parameter of the \texttt{ConvexMeshShape} constructor is a pointer to the array of the vertices coordinates, the second parameter is the number of vertices in the array and the third parameter is the size (in bytes) of the data needed for a single vertex in the array (data used by all the three coordinates of a single vertex). \begin{lstlisting} @@ -558,7 +694,7 @@ \vspace{0.6cm} - You need to make sure that the mesh you provide is indeed convex and also that the its origin of its local-space is inside the mesh. \\ + You need to make sure that the mesh you provide is indeed convex and also that the origin of its local-space is inside the mesh. \\ The collision detection test with a convex mesh shape runs in $O(n)$ where $n$ is the number of vertices in the mesh. Collision detection can become expensive if there are too many vertices in the mesh. It is possible to speed up the collision detection by providing information about the edges of the convex mesh. If you provide edges information @@ -599,11 +735,11 @@ \subsection{Inertia Tensor of a Collision Shape} \label{sec:inertiacollisionshape} - When you create a rigid body, you need to specify its inertia tensor. The inertia tensor is a $3 \times 3$ matrix decribing how the mass is distributed inside the rigid body which + When you create a rigid body, you need to specify its inertia tensor. The inertia tensor is a $3 \times 3$ matrix describing how the mass is distributed inside the rigid body which will be used to calculate the rotation of the body. The inertia tensor depends on the mass and the shape of the body. \\ You can use the collision shape of a rigid body to compute its inertia tensor. To do that, you need to use the \texttt{CollisionShape::computeLocalInertiaTensor()} method of your collision - shape. This method takes two parameters. The first one if the inertia tensor matrix that need to be computed and the second one is the mass of the rigid body (in kilograms). For instance, + shape. This method takes two parameters. The first one is the inertia tensor matrix that has to be computed and the second one is the mass of the rigid body (in kilograms). For instance, if you want to compute the inertia tensor matrix of a capsule shape with a mass of 3 kilograms, here is what the code looks like : \\ \begin{lstlisting} @@ -618,14 +754,14 @@ When the motion of the first body of the joint is known, the relative motion of the second body has at most six degrees of freedom (three for the translation and three for the rotation). The different joints can reduce the number of degrees of freedom between two rigid bodies. \\ - Some joints have limits to control the range of motion and some joints have motors to automatically move the the bodies of the joint at a given speed. \\ + Some joints have limits to control the range of motion and some joints have motors to automatically move the bodies of the joint at a given speed. \\ \subsection{Ball and Socket Joint} The \texttt{BallAndSocketJoint} class describes a ball and socket joint between two bodies. In a ball and socket joint, the two bodies cannot translate with respect to each other. - However, they can rotate freely around a common anchor point. This joint has three degrees of freedom. This joint can be used to simulate a chain of bodies for instance. \\ + However, they can rotate freely around a common anchor point. This joint has three degrees of freedom and can be used to simulate a chain of bodies for instance. \\ - In order to create a ball and socket joint, you first need to create an object of the \texttt{BallAndSocketJointInfo} class with the necessary information. You need to provide the pointers to the + In order to create a ball and socket joint, you first need to create an instance of the \texttt{BallAndSocketJointInfo} class with the necessary information. You need to provide the pointers to the two rigid bodies and also the coordinates of the anchor point (in world-space). At the joint creation, the world-space anchor point will be converted into the local-space of the two rigid bodies and then, the joint will make sure that the two local-space anchor points match in world-space. Therefore, the two bodies need to be in a correct position at the joint creation. \\ @@ -728,7 +864,7 @@ \subsubsection{Motor} - A motor is also available for the hinge joint. It can be used rotate the bodies around the hinge axis at a given angular speed and such that the torque applied to + A motor is also available for the hinge joint. It can be used to rotate the bodies around the hinge axis at a given angular speed and such that the torque applied to rotate the bodies does not exceed a maximum allowed torque. The motor is disabled by default. If you want to use it, you first have to activate it using the \texttt{isMotorEnabled} boolean variable of the \texttt{HingeJointInfo} object before you create the joint. Then, you need to specify the angular motor speed (in radians/seconds) using the \texttt{motorSpeed} variable and also the maximum allowed torque (in Newton $\cdot$ meters) with the \texttt{maxMotorTorque} variable. \\ @@ -764,7 +900,7 @@ The class \texttt{SliderJoint} describes a slider joint (or prismatic joint) that only allows relative translation along a single direction. It has a single degree of freedom and allows no relative rotation. In order to create a slider joint, you first need to specify the anchor point (in world-space) and the slider axis direction (in world-space). The constructor of the - \texttt{SliderJointInfo} object needs two pointer to the bodies of the joint, the anchor point and the axis direction. Note that the two bodies have to be in a correct initial position when + \texttt{SliderJointInfo} object needs two pointers to the bodies of the joint, the anchor point and the axis direction. Note that the two bodies have to be in a correct initial position when the joint is created. \\ You can see in the following code how to specify the information to create a slider joint : \\ @@ -897,7 +1033,7 @@ By default the two bodies involved in a joint are able to collide with each other. However, it is possible to disable the collision between the two bodies that are part of the joint. To do it, you simply need to set the variable \texttt{isCollisionEnabled} of the joint info object to \emph{false} when you create the joint. \\ - For instance, when you create a \texttt{HingeJointInfo} object in order to construct a hinge joint, you can disable collision between the two bodies of the joint as in the + For instance, when you create a \texttt{HingeJointInfo} object in order to construct a hinge joint, you can disable the collision between the two bodies of the joint as in the following example : \\ \begin{lstlisting} @@ -914,8 +1050,10 @@ \subsection{Destroying a Joint} + \begin{sloppypar} In order to destroy a joint, you simply need to call the \texttt{DynamicsWorld::destroyJoint()} method using the pointer to a previously created joint object as argument as shown in the following code : \\ + \end{sloppypar} \begin{lstlisting} // rp3d::BallAndSocketJoint* joint is a previously @@ -940,9 +1078,9 @@ \subsection{Cubes} - In this examples, you will see how to create a floor and some cubes using the Box Shape for collision detection. Because of gravity, + In this example, you will see how to create a floor and some cubes using the Box Shape for collision detection. Because of gravity, the cubes will fall down on the floor. After falling down, the cubes will come to rest and start sleeping (become inactive). In this demo, - the cubes are green when they are inactive and become red as they get inactive (sleeping). + the cubes are green when they are active and become red as they get inactive (sleeping). \subsection{Collision Shapes} @@ -956,10 +1094,33 @@ \section{Receiving Feedback} + Sometimes, you want to receive notifications from the physics engine when a given event happened. The \texttt{EventListener} class can be used for that purpose. In order to use + it, you need to create a new class that inherits from the \texttt{EventListener} class and overrides some methods that will be called by the ReactPhysics3D library when some events + occur. You also need to register your class in the physics world using the \texttt{DynamicsWorld::setEventListener()} as in the following code : \\ + + \begin{lstlisting} + // Here, YourEventListener is a class that inherits + // from the EventListener class of reactphysics3d + YourEventListener listener; + + // Register your event listener class + world.setEventListener(&listener); + \end{lstlisting} + \subsection{Contacts} + If you want to be notified when two bodies that were separated before become in contact, you need to override the \texttt{EventListener::beginContact()} method in your event + listener class. Then, this method will be called when the two separated bodies becomes in contact. \\ + + If you receive a notification when a new contact between two bodies is found, you need to override the \texttt{EventListener::newContact()} method in your event listener class. Then, this + method will be called when a new contact is found. + \section{Profiler} + If you build the library with the \texttt{PROFILING\_ENABLED} variable enabled (see section \ref{sec:cmakevariables}), a real-time profiler will collect information while the application + is running. Then, at the end of your application, when the destructor of the \texttt{DynamicsWorld} class is called, information about the running time of the library will be displayed in the + standard output. This can be useful to know where time is spent in the different parts of the ReactPhysics3D library in case your application is too slow. + \section{API Documentation} Some documentation about the API of the code has been generated

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�fNk;N=?4E<+41j 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