mirror of
https://github.com/glfw/glfw.git
synced 2024-11-10 09:01:46 +00:00
1314 lines
54 KiB
TeX
1314 lines
54 KiB
TeX
%-------------------------------------------------------------------------
|
||
% GLFW Users Guide
|
||
% API Version: 3.0
|
||
%-------------------------------------------------------------------------
|
||
|
||
% Document class
|
||
\documentclass[a4paper,11pt,oneside]{report}
|
||
|
||
% Document title and API version
|
||
\newcommand{\glfwdoctype}[1][0]{Users Guide}
|
||
\newcommand{\glfwapiver}[1][0]{3.0}
|
||
|
||
% Common document settings and macros
|
||
\input{glfwdoc.sty}
|
||
|
||
% PDF specific document settings
|
||
\hypersetup{pdftitle={GLFW Users Guide}}
|
||
\hypersetup{pdfauthor={Marcus Geelnard}}
|
||
\hypersetup{pdfkeywords={GLFW,OpenGL,guide,manual}}
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
% Document body
|
||
%-------------------------------------------------------------------------
|
||
|
||
\begin{document}
|
||
|
||
\pagestyle{plain}
|
||
|
||
% Title page
|
||
\glfwmaketitle
|
||
|
||
% Summary, trademarks and table of contents
|
||
\pagenumbering{roman}
|
||
\setcounter{page}{1}
|
||
|
||
%-------------------------------------------------------------------------
|
||
% Summary and Trademarks
|
||
%-------------------------------------------------------------------------
|
||
\chapter*{Summary}
|
||
|
||
This document is a users guide for the \GLFW\ API that gives a practical
|
||
introduction to using \GLFW . For a more detailed description of the
|
||
\GLFW\ API you should refer to the \textit{GLFW Reference Manual}.
|
||
\vspace{5cm}
|
||
|
||
\large
|
||
Trademarks
|
||
|
||
\small
|
||
OpenGL and IRIX are registered trademarks of Silicon Graphics, Inc.\linebreak
|
||
Microsoft and Windows are registered trademarks of Microsoft Corporation.\linebreak
|
||
Mac OS is a registered trademark of Apple Computer, Inc.\linebreak
|
||
Linux is a registered trademark of Linus Torvalds.\linebreak
|
||
FreeBSD is a registered trademark of Wind River Systems, Inc.\linebreak
|
||
Solaris is a trademark of Sun Microsystems, Inc.\linebreak
|
||
UNIX is a registered trademark of The Open Group.\linebreak
|
||
X Window System is a trademark of The Open Group.\linebreak
|
||
POSIX is a trademark of IEEE.\linebreak
|
||
Truevision, TARGA and TGA are registered trademarks of Truevision, Inc.\linebreak
|
||
IBM is a registered trademark of IBM Corporation.\linebreak
|
||
|
||
All other trademarks mentioned in this document are the property of their respective owners.
|
||
\normalsize
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
% Table of contents
|
||
%-------------------------------------------------------------------------
|
||
\tableofcontents
|
||
\pagebreak
|
||
|
||
|
||
% Document chapters starts here...
|
||
\pagenumbering{arabic}
|
||
\setcounter{page}{1}
|
||
|
||
\pagestyle{fancy}
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
% Introduction
|
||
%-------------------------------------------------------------------------
|
||
\chapter{Introduction}
|
||
\thispagestyle{fancy}
|
||
\GLFW\ is a portable API (Application Program Interface) that handles
|
||
operating system specific tasks related to \OpenGL\ programming. While
|
||
\OpenGL\ in general is portable, easy to use and often results in tidy and
|
||
compact code, the operating system specific mechanisms that are required
|
||
to set up and manage an \OpenGL\ window are quite the opposite. \GLFW\ tries
|
||
to remedy this by providing the following functionality:
|
||
|
||
\begin{itemize}
|
||
\item Opening and managing an \OpenGL\ window.
|
||
\item Keyboard, mouse and joystick input.
|
||
\item High precision time input.
|
||
\item Support for querying and using \OpenGL\ extensions.
|
||
\end{itemize}
|
||
\vspace{18pt}
|
||
|
||
All this functionality is implemented as a set of easy-to-use functions,
|
||
which makes it possible to write an \OpenGL\ application framework in just a
|
||
few lines of code. The \GLFW\ API is operating system and platform independent,
|
||
making it very simple to port \GLFW\ based \OpenGL\ applications between the
|
||
supported platforms.
|
||
|
||
Currently supported platforms are:
|
||
\begin{itemize}
|
||
\item Microsoft Windows\textsuperscript{\textregistered}
|
||
\item Unix\textsuperscript{\textregistered} or Unix<69>-like systems running the
|
||
X Window System\texttrademark with GLX version 1.3 or later
|
||
\item Mac OS X\textsuperscript{\textregistered} 10.5 and later, using Cocoa\footnote{Support for joysticks missing at the time of writing.}
|
||
\end{itemize}
|
||
|
||
There is also deprecated support for Mac OS X versions 10.3 and 10.4, using the Carbon API.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
% Getting Started
|
||
%-------------------------------------------------------------------------
|
||
\chapter{Getting Started}
|
||
\thispagestyle{fancy}
|
||
In this chapter you will learn how to write a simple \OpenGL\ application
|
||
using \GLFW . We start by initializing \GLFW , then we open a window and
|
||
read some user keyboard input.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Initializing GLFW}
|
||
Before using any of the \GLFW\ functions, it is necessary to call
|
||
\textbf{glfwInit}. It initializes the parts of \GLFW\ that are not dependent on
|
||
a window, such as time and joystick input. The C syntax is:
|
||
|
||
\begin{lstlisting}
|
||
int glfwInit(void)
|
||
\end{lstlisting}
|
||
|
||
\textbf{glfwInit} returns GL\_TRUE if initialization succeeded, or
|
||
GL\_FALSE if it failed.
|
||
|
||
When your application is done using \GLFW , typically at the very end of
|
||
the program, you should call \textbf{glfwTerminate}. The C syntax is:
|
||
|
||
\begin{lstlisting}
|
||
void glfwTerminate(void)
|
||
\end{lstlisting}
|
||
|
||
This releases any resources allocated by GLFW and closes the window if it is
|
||
open. After this call, you must call \textbf{glfwInit} again before using any
|
||
\GLFW\ functions).
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Opening An OpenGL Window}
|
||
Opening an \OpenGL\ window is done with the \textbf{glfwOpenWindow} function.
|
||
The function takes nine arguments, which are used to describe the following
|
||
properties of the requested window:
|
||
|
||
\begin{itemize}
|
||
\item Window dimensions (width and height) in pixels.
|
||
\item Color and alpha buffer bit depth.
|
||
\item Depth buffer (Z-buffer) bit depth.
|
||
\item Stencil buffer bit depth.
|
||
\item Whether to use fullscreen or windowed mode.
|
||
\end{itemize}
|
||
|
||
The C language syntax for \textbf{glfwOpenWindow} is:
|
||
\begin{lstlisting}
|
||
int glfwOpenWindow(int width, int height,
|
||
int redbits, int greenbits, int bluebits,
|
||
int alphabits, int depthbits, int stencilbits,
|
||
int mode)
|
||
\end{lstlisting}
|
||
|
||
\textbf{glfwOpenWindow} returns GL\_TRUE if the window was opened
|
||
correctly, or GL\_FALSE if \GLFW\ failed to open the window.
|
||
|
||
\GLFW\ tries to open a window that best matches the requested parameters.
|
||
Some parameters may be omitted by setting them to zero, which will result
|
||
in \GLFW\ either using a default value, or the related functionality to be
|
||
disabled. For instance, if \textit{width} and \textit{height} are both
|
||
zero, \GLFW\ will use a window resolution of 640x480. If
|
||
\textit{depthbits} is zero, the opened window may not have a depth buffer.
|
||
|
||
The \textit{mode} argument is used to specify if the window is to be a
|
||
fullscreen window or a regular window.
|
||
|
||
If \textit{mode} is GLFW\_FULLSCREEN, the window will cover the entire
|
||
screen and no window border or decorations will be visible. If possible, the
|
||
video mode will be changed to the mode that closest matches the \textit{width},
|
||
\textit{height}, \textit{redbits}, \textit{greenbits}, \textit{bluebits} and
|
||
\textit{alphabits} arguments. Furthermore, the mouse pointer will be hidden,
|
||
and screensavers are prohibited. This is usually the best mode for games and
|
||
demos.
|
||
|
||
If \textit{mode} is GLFW\_WINDOW, the window will be opened as a normal,
|
||
decorated window on the desktop. The mouse pointer will not be hidden and
|
||
screensavers are allowed to be activated.
|
||
|
||
To close the window, you can either use \textbf{glfwTerminate}, as
|
||
described earlier, or you can use the more explicit approach by calling
|
||
\textbf{glfwCloseWindow}, which has the C syntax:
|
||
|
||
\begin{lstlisting}
|
||
void glfwCloseWindow(void)
|
||
\end{lstlisting}
|
||
|
||
Note that you do not need to call \textbf{glfwTerminate} and \textbf{glfwInit}
|
||
before opening a new window after having closed the current one using
|
||
\textbf{glfwCloseWindow}.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Using Keyboard Input}
|
||
\GLFW\ provides several means for receiving user input, which will be
|
||
discussed in more detail in chapter \ref{par:inputhandling}. One of the
|
||
simplest ways of checking for keyboard input is to use the function
|
||
\textbf{glfwGetKey}:
|
||
|
||
\begin{lstlisting}
|
||
int glfwGetKey(int key)
|
||
\end{lstlisting}
|
||
|
||
It queries the current status of individual keyboard keys. The argument
|
||
\textit{key} specifies which key to check, and it must be a valid GLFW key code
|
||
(see the \textit{GLFW Reference Manual} for a list of key codes).
|
||
\textbf{glfwGetKey} returns GLFW\_PRESS if the key is currently held down, or
|
||
GLFW\_RELEASE if the key is not being held down. For example:
|
||
|
||
\begin{lstlisting}
|
||
A_pressed = glfwGetKey(GLFW_KEY_A);
|
||
esc_pressed = glfwGetKey(GLFW_KEY_ESCAPE);
|
||
\end{lstlisting}
|
||
|
||
In order for \textbf{glfwGetKey} to have any effect, you need to poll for
|
||
input events on a regular basis. This can be done in one of two ways:
|
||
|
||
\begin{enumerate}
|
||
\item Implicitly by calling \textbf{glfwSwapBuffers} often.
|
||
\item Explicitly by calling \textbf{glfwPollEvents} often.
|
||
\end{enumerate}
|
||
|
||
In general you do not have to care about this, since you will normally
|
||
call \textbf{glfwSwapBuffers} to swap front and back rendering buffers
|
||
every animation frame anyway. If, however, this is not the case, you
|
||
should call \textbf{glfwPollEvents} in the order of 10-100 times per
|
||
second in order for \GLFW\ to maintain an up to date input state.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Putting It Together: A Minimal GLFW Application}
|
||
Now that you know how to initialize \GLFW , open a window and poll for
|
||
keyboard input, let us exemplify this with a simple \OpenGL\ program:
|
||
|
||
\begin{lstlisting}
|
||
#include <GL/glfw.h>
|
||
#include <stdlib.h>
|
||
|
||
int main(void)
|
||
{
|
||
int running = GL_TRUE;
|
||
|
||
// Initialize GLFW
|
||
if (!glfwInit())
|
||
exit(EXIT_FAILURE);
|
||
|
||
// Open an OpenGL window
|
||
if (!glfwOpenWindow(300,300, 0,0,0,0,0,0, GLFW_WINDOW))
|
||
{
|
||
glfwTerminate();
|
||
exit(EXIT_FAILURE);
|
||
}
|
||
|
||
// Main loop
|
||
while (running)
|
||
{
|
||
// OpenGL rendering goes here...
|
||
glClear(GL_COLOR_BUFFER_BIT);
|
||
|
||
// Swap front and back rendering buffers
|
||
glfwSwapBuffers();
|
||
|
||
// Check if ESC key was pressed or window was closed
|
||
running = !glfwGetKey(GLFW_KEY_ESCAPE) &&
|
||
glfwGetWindowParam(GLFW_OPENED);
|
||
}
|
||
|
||
// Close window and terminate GLFW
|
||
glfwTerminate();
|
||
|
||
// Exit program
|
||
exit(EXIT_SUCCESS);
|
||
}
|
||
\end{lstlisting}
|
||
|
||
The program opens a 300x300 window and runs in a loop until the escape key
|
||
is pressed, or the window was closed. All the \OpenGL\ ``rendering'' that
|
||
is done in this example is to clear the window.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
% Window Operations
|
||
%-------------------------------------------------------------------------
|
||
\chapter{Window Operations}
|
||
\thispagestyle{fancy}
|
||
In this chapter, you will learn more about window related \GLFW\
|
||
functionality, including setting and getting window properties, buffer
|
||
swap control and video mode querying.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Setting Window Properties}
|
||
In the previous chapter the \textbf{glfwOpenWindow} function was
|
||
described, which specifies the sizes of the color, alpha, depth and
|
||
stencil buffers. It is also possible to request a specific minimum OpenGL
|
||
version, multisampling anti-aliasing, an accumulation buffer, stereo
|
||
rendering and more by using the \textbf{glfwOpenWindowHint} function:
|
||
|
||
\begin{lstlisting}
|
||
void glfwOpenWindowHint(int target, int hint)
|
||
\end{lstlisting}
|
||
|
||
The \textit{target} argument can be one of the constants listed in table~
|
||
\ref{tab:winhints}, and \textit{hint} is the value to assign to the
|
||
specified target.
|
||
|
||
%-------------------------------------------------------------------------
|
||
\begin{table}[p]
|
||
\begin{center}
|
||
\begin{tabular}{|l|l|p{7.0cm}|} \hline \raggedright
|
||
\textbf{Name} & \textbf{Default} & \textbf{Description} \\ \hline
|
||
GLFW\_REFRESH\_RATE & 0 & Vertical monitor refresh rate in Hz (only used for fullscreen windows). Zero means system default.\\ \hline
|
||
GLFW\_ACCUM\_RED\_BITS & 0 & Number of bits for the red channel of the accumulation buffer.\\ \hline
|
||
GLFW\_ACCUM\_GREEN\_BITS & 0 & Number of bits for the green channel of the accumulation buffer.\\ \hline
|
||
GLFW\_ACCUM\_BLUE\_BITS & 0 & Number of bits for the blue channel of the accumulation buffer.\\ \hline
|
||
GLFW\_ACCUM\_ALPHA\_BITS & 0 & Number of bits for the alpha channel of the accumulation buffer.\\ \hline
|
||
GLFW\_AUX\_BUFFERS & 0 & Number of auxiliary buffers.\\ \hline
|
||
GLFW\_STEREO & GL\_FALSE & Specify if stereo rendering should be supported (can be GL\_TRUE or GL\_FALSE).\\ \hline
|
||
GLFW\_WINDOW\_NO\_RESIZE & GL\_FALSE & Specify whether the window can be resized by the user (not used for fullscreen windows).\\ \hline
|
||
GLFW\_FSAA\_SAMPLES & 0 & Number of samples to use for the multisampling buffer. Zero disables multisampling.\\ \hline
|
||
GLFW\_OPENGL\_VERSION\_MAJOR & 1 & Major number of the desired minimum OpenGL version.\\ \hline
|
||
GLFW\_OPENGL\_VERSION\_MINOR & 1 & Minor number of the desired minimum OpenGL version.\\ \hline
|
||
GLFW\_OPENGL\_FORWARD\_COMPAT & GL\_FALSE & Specify whether the OpenGL context should be forward-compatible (i.e. disallow legacy functionality).
|
||
This should only be used when requesting OpenGL version 3.0 or above.\\ \hline
|
||
GLFW\_OPENGL\_DEBUG\_CONTEXT & GL\_FALSE & Specify whether a debug context should be created.\\ \hline
|
||
GLFW\_OPENGL\_PROFILE & 0 & The OpenGL profile the context should implement, or zero to let the system choose.
|
||
Available profiles are GLFW\_OPENGL\_CORE\_PROFILE and GLFW\_OPENGL\_COMPAT\_PROFILE.\\ \hline
|
||
\end{tabular}
|
||
\end{center}
|
||
\caption{Targets for \textbf{glfwOpenWindowHint}}
|
||
\label{tab:winhints}
|
||
\end{table}
|
||
%-------------------------------------------------------------------------
|
||
|
||
For a hint to have any effect, the \textbf{glfwOpenWindowHint} function
|
||
must be called before opening the window with the \textbf{glfwOpenWindow}
|
||
function.
|
||
|
||
To request an accumulation buffer, set the GLFW\_ACCUM\_x\_BITS targets to
|
||
values greater than zero (usually eight or sixteen bits per component).
|
||
To request auxiliary buffers, set the GLFW\_AUX\_BUFFERS target to a value
|
||
greater than zero. To request a stereo rendering capable window, set the
|
||
GLFW\_STEREO target to GL\_TRUE.
|
||
|
||
If you want to enable fullscreen antialiasing, set the GLFW\_FSAA\_SAMPLES
|
||
target to a value greater than zero. If the windowing system is unable to
|
||
fulfil the request, \GLFW\ will degrade gracefully and disable FSAA if necessary.
|
||
|
||
The GLFW\_REFRESH\_RATE target should be used with caution, since it may
|
||
result in suboptimal operation, or even a blank or damaged screen.
|
||
|
||
If you want to create a forward-compatible \OpenGL\ context, set the
|
||
GLFW\_OPENGL\_FORWARD\_COMPAT hint to GL\_TRUE. Note that such contexts are
|
||
only available for \OpenGL\ version 3.0 and above, so you will need to specify
|
||
a valid minimum version using the GLFW\_OPENGL\_VERSION\_MAJOR and
|
||
GLFW\_OPENGL\_VERSION\_MINOR hints.
|
||
|
||
If you want to create a context using the core profile as available in \OpenGL\
|
||
version 3.2 and above, set the GLFW\_OPENGL\_PROFILE hint accordingly. Note that
|
||
as above you have to set a valid minimum version for this to work.
|
||
|
||
Also note that at the time of this release, Mac OS X did not support \OpenGL\
|
||
version 3.0 or above; thus GLFW cannot create contexts of versions above 2.1
|
||
on that platform.
|
||
|
||
Besides the parameters that are given with the \textbf{glfwOpenWindow} and
|
||
\textbf{glfwOpenWindowHint} functions, a few more properties of a window
|
||
can be changed after the window has been opened, namely the window title,
|
||
window size, and window position.
|
||
|
||
To change the window title of an open window, use the
|
||
\textbf{glfwSetWindowTitle} function:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSetWindowTitle(const char* title)
|
||
\end{lstlisting}
|
||
|
||
\textit{title} is a null terminated ISO~8859-1 (8-bit Latin~1) string that
|
||
will be used as the window title. It will also be used as the application
|
||
name (for instance in the application list when using \texttt{Alt+Tab}
|
||
under Windows, or as the icon name when the window is iconified under
|
||
the X Window System). The default window name is ``GLFW Window'', which
|
||
will be used unless \textbf{glfwSetWindowTitle} is called after the window
|
||
has been opened.
|
||
|
||
To change the size of a window, call \textbf{glfwSetWindowSize}:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSetWindowSize(int width, int height)
|
||
\end{lstlisting}
|
||
|
||
Where \textit{width} and \textit{height} are the new dimensions of the
|
||
window.
|
||
|
||
To change the position of a window, call \textbf{glfwSetWindowPos}:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSetWindowPos(int x, int y)
|
||
\end{lstlisting}
|
||
|
||
Where \textit{x} and \textit{y} are the new desktop coordinates of the
|
||
window. This function does not have any effect when in fullscreen mode.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Getting Window Properties}
|
||
When opening a window, the opened window will not necessarily have the
|
||
requested properties, so you should always check the parameters that your
|
||
application relies on (e.g. number of stencil bits) using
|
||
\textbf{glfwGetWindowParam}, which has the C syntax:
|
||
|
||
\begin{lstlisting}
|
||
int glfwGetWindowParam(int param)
|
||
\end{lstlisting}
|
||
|
||
The argument \textit{param} can be one of the tokens listed in table
|
||
\ref{tab:winparams}, and the return value is an integer holding the
|
||
requested value.
|
||
|
||
%-------------------------------------------------------------------------
|
||
\begin{table}[p]
|
||
\begin{center}
|
||
\begin{tabular}{|l|p{9.5cm}|} \hline \raggedright
|
||
\textbf{Name} & \textbf{Description} \\ \hline
|
||
GLFW\_OPENED & GL\_TRUE if window is opened, else GL\_FALSE.\\ \hline
|
||
GLFW\_ACTIVE & GL\_TRUE if window has focus, else GL\_FALSE.\\ \hline
|
||
GLFW\_ICONIFIED & GL\_TRUE if window is iconified, else GL\_FALSE.\\ \hline
|
||
GLFW\_ACCELERATED & GL\_TRUE if window is hardware accelerated, else GL\_FALSE.\\ \hline
|
||
GLFW\_RED\_BITS & Number of bits for the red color component.\\ \hline
|
||
GLFW\_GREEN\_BITS & Number of bits for the green color component.\\ \hline
|
||
GLFW\_BLUE\_BITS & Number of bits for the blue color component.\\ \hline
|
||
GLFW\_ALPHA\_BITS & Number of bits for the alpha buffer.\\ \hline
|
||
GLFW\_DEPTH\_BITS & Number of bits for the depth buffer.\\ \hline
|
||
GLFW\_STENCIL\_BITS & Number of bits for the stencil buffer.\\ \hline
|
||
GLFW\_REFRESH\_RATE & Vertical monitor refresh rate in Hz. Zero indicates an unknown or a default refresh rate.\\ \hline
|
||
GLFW\_ACCUM\_RED\_BITS & Number of bits for the red channel of the accumulation buffer.\\ \hline
|
||
GLFW\_ACCUM\_GREEN\_BITS & Number of bits for the green channel of the accumulation buffer.\\ \hline
|
||
GLFW\_ACCUM\_BLUE\_BITS & Number of bits for the blue channel of the accumulation buffer.\\ \hline
|
||
GLFW\_ACCUM\_ALPHA\_BITS & Number of bits for the alpha channel of the accumulation buffer.\\ \hline
|
||
GLFW\_AUX\_BUFFERS & Number of auxiliary buffers.\\ \hline
|
||
GLFW\_STEREO & GL\_TRUE if stereo rendering is supported, else GL\_FALSE.\\ \hline
|
||
GLFW\_WINDOW\_NO\_RESIZE & GL\_TRUE if the window cannot be resized by the user, else GL\_FALSE.\\ \hline
|
||
GLFW\_FSAA\_SAMPLES & Number of multisampling buffer samples. Zero indicated multisampling is disabled.\\ \hline
|
||
GLFW\_OPENGL\_VERSION\_MAJOR & Major number of the actual version of the context.\\ \hline
|
||
GLFW\_OPENGL\_VERSION\_MINOR & Minor number of the actual version of the context.\\ \hline
|
||
GLFW\_OPENGL\_FORWARD\_COMPAT & GL\_TRUE if the context is forward-compatible, else GL\_FALSE.\\ \hline
|
||
GLFW\_OPENGL\_DEBUG\_CONTEXT & GL\_TRUE if the context is a debug context.\\ \hline
|
||
GLFW\_OPENGL\_PROFILE & The profile implemented by the context, or zero.\\ \hline
|
||
\end{tabular}
|
||
\end{center}
|
||
\caption{Window parameters for \textbf{glfwGetWindowParam}}
|
||
\label{tab:winparams}
|
||
\end{table}
|
||
%-------------------------------------------------------------------------
|
||
|
||
Another useful function is \textbf{glfwSetWindowSizeCallback}, which
|
||
specifies a user function that will be called every time the window size
|
||
has changed. The C syntax is:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSetWindowSizeCallback(GLFWwindowsizefun cbfun)
|
||
\end{lstlisting}
|
||
|
||
The user function \textit{fun} should be of the type:
|
||
|
||
\begin{lstlisting}
|
||
void fun(int width, int height)
|
||
\end{lstlisting}
|
||
|
||
The first argument passed to the user function is the width of the window,
|
||
and the second argument is the height of the window. Here is an example
|
||
of how to use a window size callback function:
|
||
|
||
\begin{lstlisting}
|
||
int windowWidth, windowHeight;
|
||
|
||
void WindowResize(int width, int height)
|
||
{
|
||
windowWidth = width;
|
||
windowHeight = height;
|
||
}
|
||
|
||
int main(void)
|
||
{
|
||
...
|
||
glfwSetWindowSizeCallback(WindowResize);
|
||
...
|
||
}
|
||
\end{lstlisting}
|
||
|
||
Using a callback function for getting the window size is mostly useful for
|
||
windowed applications, since the window size may be changed at any time by
|
||
the user. It can also be used to determine the actual fullscreen
|
||
resolution.
|
||
|
||
An alternative to using a callback function for getting the window size,
|
||
is to use the function \textbf{glfwGetWindowSize}:
|
||
|
||
\begin{lstlisting}
|
||
void glfwGetWindowSize(int* width, int* height)
|
||
\end{lstlisting}
|
||
|
||
The variables pointed to by \textit{width} and \textit{height} are set to the
|
||
current window dimensions. Note that either of these may be NULL; that
|
||
argument is then ignored.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Buffer Swapping}
|
||
\GLFW\ windows are always double buffered. That means that you have two
|
||
rendering buffers; a front buffer and a back buffer. The front buffer is
|
||
the buffer that is being displayed, and the back buffer is not displayed.
|
||
\OpenGL\ lets you select which of these two buffers you want to render to
|
||
(with the \textbf{glDrawBuffer} command), but the default (and preferred)
|
||
rendering buffer is the back buffer. This way you will avoid flickering
|
||
and artifacts caused by graphics being only partly drawn at the same time
|
||
as the video raster beam is displaying the graphics on the monitor.
|
||
|
||
When an entire frame has been rendered to the back buffer, it is time to
|
||
swap the back and the front buffers in order to display the rendered
|
||
frame, and begin rendering a new frame. This is done with the command
|
||
\textbf{glfwSwapBuffers}. The C syntax is:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSwapBuffers(void)
|
||
\end{lstlisting}
|
||
|
||
After swapping the front and back rendering buffers, \textbf{glfwSwapBuffers}
|
||
by default calls \textbf{glfwPollEvents}\footnote{This behavior can be disabled
|
||
by calling \textbf{glfwDisable} with the argument GLFW\_AUTO\_POLL\_EVENTS.}.
|
||
This is to ensure frequent polling of events, such as keyboard and mouse input,
|
||
and window reshaping events. Even if a given application does not use input
|
||
events, without frequent polling of events (at \emph{least} once every few
|
||
seconds), most modern window systems will flag the application as unresponsive
|
||
and may suggest that the user terminate it.
|
||
|
||
Sometimes it can be useful to select when the buffer swap will occur. With
|
||
the function \textbf{glfwSwapInterval} it is possible to select the
|
||
minimum number of vertical retraces the video raster line should do before
|
||
swapping the buffers:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSwapInterval(int interval)
|
||
\end{lstlisting}
|
||
|
||
If \textit{interval} is zero, the swap will take place immediately when
|
||
\textbf{glfwSwapBuffers} is called, without waiting for a vertical retrace
|
||
(also known as ``vsync off''). Otherwise at least \textit{interval}
|
||
retraces will pass between each buffer swap (also known as ``vsync on'').
|
||
Using a swap interval of zero can be useful for benchmarking purposes,
|
||
when it is not desirable to measure the time it takes to wait for the
|
||
vertical retrace. However, a swap interval of 1 generally gives better
|
||
visual quality.
|
||
|
||
It should be noted that not all \OpenGL\ implementations and hardware support
|
||
this function, in which case \textbf{glfwSwapInterval} will have no effect. ATI
|
||
Radeon cards under Microsoft Windows are especially notorious in this regard.
|
||
Sometimes it is only possible to affect the swap interval through driver
|
||
settings (e.g. the display settings under Windows, or as an environment
|
||
variable setting under Unix).
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Querying Video Modes}
|
||
Although \GLFW\ generally does a good job at selecting a suitable video
|
||
mode for you when you open a fullscreen window, it is sometimes useful to
|
||
know exactly which modes are available on a certain system. For example,
|
||
you may want to present the user with a list of video modes to select
|
||
from. To get a list of available video modes, you can use the function
|
||
\textbf{glfwGetVideoModes}:
|
||
|
||
\begin{lstlisting}
|
||
int glfwGetVideoModes(GLFWvidmode* list, int maxcount)
|
||
\end{lstlisting}
|
||
|
||
The argument \textit{list} is a vector of GLFWvidmode structures, and
|
||
\textit{maxcount} is the maximum number of video modes that your vector can
|
||
hold. \textbf{glfwGetVideoModes} will return the number of video modes detected
|
||
on the system, up to \textit{maxcount}.
|
||
|
||
The GLFWvidmode structure looks like this:
|
||
|
||
\begin{lstlisting}
|
||
typedef struct {
|
||
int Width, Height; // Video resolution
|
||
int RedBits; // Red bits per pixel
|
||
int GreenBits; // Green bits per pixel
|
||
int BlueBits; // Blue bits per pixel
|
||
} GLFWvidmode;
|
||
\end{lstlisting}
|
||
|
||
Here is an example of retrieving all available video modes:
|
||
|
||
\begin{lstlisting}
|
||
int nummodes;
|
||
GLFWvidmode list[200];
|
||
nummodes = glfwGetVideoModes(list, 200);
|
||
\end{lstlisting}
|
||
|
||
The returned list is sorted, first by color depth ($RedBits + GreenBits +
|
||
BlueBits$), and then by resolution ($Width\times Height$), with the
|
||
lowest resolution, fewest bits per pixel mode first.
|
||
|
||
To get the desktop video mode, use the function
|
||
\textbf{glfwGetDesktopMode}:
|
||
|
||
\begin{lstlisting}
|
||
void glfwGetDesktopMode(GLFWvidmode* mode)
|
||
\end{lstlisting}
|
||
|
||
The function returns the resolution and color depth of the user desktop in
|
||
the mode structure. Note that the user desktop mode is independent of the
|
||
current video mode if a \GLFW\ fullscreen window has been opened.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
% Input Handling
|
||
%-------------------------------------------------------------------------
|
||
\chapter{Input Handling}
|
||
\label{par:inputhandling}
|
||
\thispagestyle{fancy}
|
||
In this chapter you will learn how to use keyboard, mouse and joystick
|
||
input, using either polling or callback functions.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Event Polling}
|
||
The first thing to know about input handling in \GLFW\ is that all
|
||
keyboard and mouse input is collected by checking for input events. This
|
||
has do be done manually by calling either \textbf{glfwPollEvents} or
|
||
\textbf{glfwSwapBuffers} (which implicitly calls \textbf{glfwPollEvents}
|
||
for you). Normally this is not a cause for concern, as
|
||
\textbf{glfwSwapBuffers} is called every frame, which should be often
|
||
enough (about 10-100 times per second for a normal \OpenGL\ application) that
|
||
the window will feel responsive.
|
||
|
||
One exception is when an application is updating its view only in response to input.
|
||
In this case the \textbf{glfwWaitEvents} is useful, as it blocks the calling
|
||
thread until an event arrives. The refresh callback, set with
|
||
\textbf{glfwSetWindowRefreshCallback}, may also be useful for such
|
||
applications, especially on unbuffered window systems.
|
||
|
||
If it is not desirable that \textbf{glfwPollEvents is} called implicitly
|
||
from \textbf{glfwSwapBuffers}, call \textbf{glfwDisable} with the argument
|
||
GLFW\_AUTO\_POLL\_EVENTS.
|
||
|
||
Note that event polling is not needed for joystick input, since all
|
||
relevant joystick state is gathered every time a joystick function is
|
||
called.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Keyboard Input}
|
||
\GLFW\ provides three mechanisms for getting keyboard input:
|
||
|
||
\begin{itemize}
|
||
\item Manually polling the state of individual keys.
|
||
\item Automatically receive new key state for any key, using a callback
|
||
function.
|
||
\item Automatically receive characters, using a callback function.
|
||
\end{itemize}
|
||
|
||
Depending on what the keyboard input will be used for, different methods may be
|
||
preferred. The main difference between the two last methods is that while
|
||
characters are affected by modifier keys (such as shift), key state is
|
||
independent of any modifier keys. Also, special keys (such as function keys,
|
||
cursor keys and modifier keys) are not reported to the character callback
|
||
function.
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Key state}
|
||
To check if a key is held down or not at any given moment, use the
|
||
function \textbf{glfwGetKey}:
|
||
|
||
\begin{lstlisting}
|
||
int glfwGetKey(int key)
|
||
\end{lstlisting}
|
||
|
||
It queries the current status of individual keyboard keys. The argument
|
||
\textit{key} specifies which key to check, which must be a valid GLFW key code.
|
||
\textbf{glfwGetKey} returns GLFW\_PRESS (or 1) if the key is currently
|
||
held down, or GLFW\_RELEASE (or 0) if the key is not being held down.
|
||
|
||
In most situations, it may be useful to know if a key has been pressed and
|
||
released between two calls to \textbf{glfwGetKey} (especially if the
|
||
animation is fairly slow, which may allow the user to press and release a
|
||
key between two calls to \textbf{glfwGetKey}). This can be accomplished by
|
||
enabling sticky keys, which is done by calling \textbf{glfwEnable} with
|
||
the argument GLFW\_STICKY\_KEYS, as in the following example:
|
||
|
||
\begin{lstlisting}
|
||
glfwEnable(GLFW_STICKY_KEYS);
|
||
\end{lstlisting}
|
||
|
||
When sticky keys are enabled, a key will not be released until it is
|
||
checked with \textbf{glfwGetKey}. To disable sticky keys, call
|
||
\textbf{glfwDisable} witht the argument GLFW\_STICKY\_KEYS. Then all keys
|
||
that are not currently held down will be released and future key releases
|
||
will take place immediately when the user releases the key without
|
||
waiting for \textbf{glfwGetKey} to check the key. By default sticky keys
|
||
are disabled.
|
||
|
||
Sticky keys are often very useful and should be used in most cases where
|
||
\textbf{glfwGetKey} is used. There is however a danger involved with
|
||
enabling sticky keys, and that is that keys that are pressed by the user
|
||
but are not checked with \textbf{glfwGetKey}, may remain ``pressed'' for a
|
||
very long time. A typical situation where this may be dangerous is in a
|
||
program that consists of two or more sections (e.g. a menu section and a
|
||
game section). If the first section enables sticky keys but does not check
|
||
for keys which the second section checks for, there is a potential of
|
||
recording many key presses in the first section that will be detected in
|
||
the second section. To avoid this problem, always disable sticky keys
|
||
before leaving a section of a program.
|
||
|
||
A usually better alternative to using \textbf{glfwGetKey} is to register a
|
||
keyboard input callback function with \textbf{glfwSetKeyCallback}:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSetKeyCallback(GLFWkeyfun cbfun)
|
||
\end{lstlisting}
|
||
|
||
The argument \textit{fun} is a pointer to a callback function. The
|
||
callback function shall take two integer arguments. The first is the key
|
||
identifier, and the second is the new key state, which can be GLFW\_PRESS
|
||
or GLFW\_RELEASE. To unregister a callback function, call
|
||
\textbf{glfwSetKeyCallback} with \textit{fun} = NULL.
|
||
|
||
Using the callback function, you can be sure not to miss any key press or
|
||
release events, regardless of how many may have occurred during the last frame.
|
||
It also encourages event-based design, where the application responds only to
|
||
actual events instead of having to poll for every supported event.
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Character input}
|
||
If the keyboard is to be used as a text input device (e.g. in a user
|
||
dialog) rather than as a set of independent buttons, a character callback
|
||
function is more suitable. To register a character callback function, use
|
||
\textbf{glfwSetCharCallback}:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSetCharCallback(GLFWcharfun cbfun)
|
||
\end{lstlisting}
|
||
|
||
The argument \textit{fun} is a pointer to a callback function. The
|
||
callback function shall take two integer arguments. The first is a Unicode
|
||
code point, and the second is GLFW\_PRESS if the key that generated
|
||
the character was pressed, or GLFW\_RELEASE if it was released. To
|
||
unregister a callback function, call \textbf{glfwSetCharCallback} with
|
||
\textit{fun} = NULL.
|
||
|
||
The Unicode character set is an international standard for encoding
|
||
characters. It is much more comprehensive than seven or eight bit
|
||
character sets (e.g. US-ASCII and Latin~1), and includes characters for
|
||
most written languages in the world. It should be noted that Unicode
|
||
character codes 0 to 255 are the same as for ISO~8859-1 (Latin~1), so as
|
||
long as a proper range check is performed on the Unicode character code,
|
||
it can be used just as an eight bit Latin~1 character code (which can be
|
||
useful if full Unicode support is not possible).
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Key repeat}
|
||
By default, \GLFW\ does not report key repeats when a key is held down.
|
||
To activate key repeat, call \textbf{glfwEnable} with the argument
|
||
GLFW\_KEY\_REPEAT:
|
||
|
||
\begin{lstlisting}
|
||
glfwEnable(GLFW_KEY_REPEAT);
|
||
\end{lstlisting}
|
||
|
||
This will let a registered key or character callback function receive key
|
||
repeat events when a key is held down.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Special system keys}
|
||
On most systems there are some special system keys that are normally not
|
||
intercepted by an application. For instance, under Windows it is possible
|
||
to switch programs by pressing \texttt{ALT+TAB}, which brings up a list of
|
||
running programs to select from. In certain situations it can be desirable
|
||
to prevent such special system keys from interfering with the program.
|
||
With \GLFW\ it is possible to do by calling \textbf{glfwDisable} with the
|
||
argument GLFW\_SYSTEM\_KEYS:
|
||
|
||
\begin{lstlisting}
|
||
glfwDisable(GLFW_SYSTEM_KEYS);
|
||
\end{lstlisting}
|
||
|
||
By doing so, most system keys will have no effect and will not interfere
|
||
with your program. System keys can be re-enabled by calling
|
||
\textbf{glfwEnable} with the argument GLFW\_SYSTEM\_KEYS. By default,
|
||
system keys are enabled.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Mouse Input}
|
||
Just like for keyboard input, mouse input can be realized with either
|
||
polling or callback functions.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Mouse position}
|
||
To query the position of the mouse cursor, call \textbf{glfwGetMousePos}:
|
||
|
||
\begin{lstlisting}
|
||
void glfwGetMousePos(int* x, int* y)
|
||
\end{lstlisting}
|
||
|
||
The variables pointed to by \textit{x} and \textit{y} will be updated with the
|
||
current position of the mouse cursor relative to the upper-left corner of the
|
||
client area of the \GLFW\ window.
|
||
|
||
An alternative is to use a callback function, which can be set with
|
||
\textbf{glfwSetMousePosCallback}:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSetMousePosCallback(GLFWmouseposfun cbfun)
|
||
\end{lstlisting}
|
||
|
||
The function that \textit{fun} points to will be called every time the
|
||
mouse cursor moves. The first argument to the callback function is
|
||
the cursor x-coordinate and the second the cursor y-coordinate, both relative
|
||
to the upper-left corner of the client area of the \GLFW\ window.
|
||
|
||
Note that while the \textbf{glfwGetMousePos} function only reports the final
|
||
position after cursor movement events have been processed, using a callback
|
||
function lets the application see each and every such event.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Mouse buttons}
|
||
To query the state of a mouse button, call \textbf{glfwGetMouseButton}:
|
||
|
||
\begin{lstlisting}
|
||
int glfwGetMouseButton(int button)
|
||
\end{lstlisting}
|
||
|
||
The argument \textit{button} can be any \GLFW\ mouse button token, i.e.
|
||
GLFW\_MOUSE\_BUTTON\_1 through GLFW\_MOUSE\_BUTTON\_8 or one of
|
||
GLFW\_MOUSE\_BUTTON\_LEFT, GLFW\_MOUSE\_BUTTON\_RIGHT or
|
||
GLFW\_MOUSE\_BUTTON\_MIDDLE. \textbf{glfwGetMouseButton} will return
|
||
GLFW\_PRESS (which is a non-zero value) if the corresponding mouse button is
|
||
held down, otherwise it will return GLFW\_RELEASE (which is equal to zero).
|
||
|
||
Just as it is possible to make keys ``sticky'', it is also possible to make
|
||
each mouse button appear as held down until it is checked with
|
||
\textbf{glfwGetMouseButton}. To enable sticky mouse buttons, call
|
||
\textbf{glfwEnable} with the argument GLFW\_STICKY\_MOUSE\_BUTTONS.
|
||
|
||
When sticky mouse buttons are enabled, a mouse button will not be released
|
||
until it is checked with \textbf{glfwGetMouseButton}. To disable sticky
|
||
mouse buttons, call \textbf{glfwDisable} with the argument
|
||
GLFW\_STICKY\_MOUSE\_BUTTONS. Then all mouse buttons that are not
|
||
currently held down will be released and future mouse button releases
|
||
will take place immediately when the user releases the mouse button
|
||
without waiting for \textbf{glfwGetMouseButton} to check for the mouse
|
||
button. By default sticky mouse buttons are disabled.
|
||
|
||
There is also a callback function for mouse button activities, which can
|
||
be set with \textbf{glfwSetMouseButtonCallback}:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSetMouseButtonCallback(GLFWmousebuttonfun fun)
|
||
\end{lstlisting}
|
||
|
||
The argument \textit{fun} specifies a function that will be called
|
||
whenever a mouse button is pressed or released, or NULL to unregister a
|
||
callback function. The first argument to the callback function is a mouse
|
||
button identifier, and the second is either GLFW\_PRESS or GLFW\_RELEASE,
|
||
depending on the new state of the corresponding mouse button.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Mouse wheel}
|
||
Some mice have a mouse wheel, most commonly used for vertical scrolling. Also,
|
||
most modern touchpads allow the user to scroll at least vertically, either by
|
||
reserving an area for scrolling or through multi-finger gestures. To get the
|
||
position of the mouse wheel, call \textbf{glfwGetMouseWheel}:
|
||
|
||
\begin{lstlisting}
|
||
int glfwGetMouseWheel(void)
|
||
\end{lstlisting}
|
||
|
||
The function returns an integer that represents the position of the mouse
|
||
wheel. When the user turns the wheel, the wheel position will increase or
|
||
decrease. Note that since scrolling hardware has no absolute position, \GLFW\
|
||
simply sets the position to zero when the window is opened.
|
||
|
||
It is also possible to register a callback function for mouse wheel events
|
||
with the \textbf{glfwSetMouseWheelCallback} function:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSetMouseWheelCallback(GLFWmousewheelfun fun)
|
||
\end{lstlisting}
|
||
|
||
The argument \textit{fun} specifies a function that will be called
|
||
whenever the mouse wheel is moved, or NULL to unregister a callback
|
||
function. The argument to the callback function is the position of the
|
||
mouse wheel.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Hiding the mouse cursor}
|
||
It is possible to hide the mouse cursor with the function call:
|
||
|
||
\begin{lstlisting}
|
||
glfwDisable(GLFW_MOUSE_CURSOR);
|
||
\end{lstlisting}
|
||
|
||
Hiding the mouse cursor has three effects:
|
||
|
||
\begin{enumerate}
|
||
\item The cursor becomes invisible.
|
||
\item The cursor is guaranteed to be confined to the window.
|
||
\item Mouse coordinates are not limited to the window size.
|
||
\end{enumerate}
|
||
|
||
To show the mouse cursor again, call \textbf{glfwEnable} with the
|
||
argument GLFW\_MOUSE\_CURSOR:
|
||
|
||
\begin{lstlisting}
|
||
glfwEnable(GLFW_MOUSE_CURSOR);
|
||
\end{lstlisting}
|
||
|
||
By default the mouse cursor is hidden if a window is opened in fullscreen
|
||
mode, otherwise it is not hidden.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Joystick Input}
|
||
\GLFW\ has support for up to sixteen joysticks, and an infinite\footnote{%
|
||
There are of course actual limitations posed by the underlying hardware,
|
||
drivers and operation system.} number of axes and buttons per joystick.
|
||
Unlike keyboard and mouse input, joystick input does not need an opened
|
||
window, and \textbf{glfwPollEvents} or \textbf{glfwSwapBuffers} does not
|
||
have to be called in order for joystick state to be updated.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Joystick capabilities}
|
||
First, it is often necessary to determine if a joystick is connected and
|
||
what its capabilities are. To get this information the function
|
||
\textbf{glfwGetJoystickParam} can be used:
|
||
|
||
\begin{lstlisting}
|
||
int glfwGetJoystickParam(int joy, int param)
|
||
\end{lstlisting}
|
||
|
||
The \textit{joy} argument specifies which joystick to retrieve the
|
||
parameter from, and it should be GLFW\_JOYSTICK\_\textit{n}, where
|
||
\textit{n} is in the range 1 to 16. The \textit{param} argument specifies
|
||
which parameter to retrieve. To determine if a joystick is connected,
|
||
\textit{param} should be GLFW\_PRESENT, which will cause the function to
|
||
return GL\_TRUE if the joystick is connected, or GL\_FALSE if it is not.
|
||
To determine the number of axes or buttons that are supported by the
|
||
joystick, \textit{param} should be GLFW\_AXES or GLFW\_BUTTONS,
|
||
respectively.
|
||
|
||
Note that \GLFW\ supports both D-pads and POVs, even though they are not
|
||
explicitly mentioned in the API. D-pads are exposed as a set of four buttons
|
||
and POVs are as two axes.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Joystick position}
|
||
To get the current axis positions of the joystick, the
|
||
\textbf{glfwGetJoystickPos} is used:
|
||
|
||
\begin{lstlisting}
|
||
int glfwGetJoystickPos(int joy, float* pos, int numaxes)
|
||
\end{lstlisting}
|
||
|
||
As with \textbf{glfwGetJoystickParam}, the \textit{joy} argument
|
||
specifies which joystick to retrieve information from. The
|
||
\textit{numaxes} argument specifies how many axes to return positions for and the
|
||
\textit{pos} argument specifies an array in which they
|
||
are stored. The function returns the actual number of axes that were
|
||
returned, which could be less than \textit{numaxes} if the joystick does
|
||
not support all the requested axes, or if the joystick is not connected.
|
||
|
||
For instance, to get the position of the first two axes (the X and Y axes)
|
||
of joystick 1, the following code can be used:
|
||
|
||
\begin{lstlisting}
|
||
float position[2];
|
||
|
||
glfwGetJoystickPos(GLFW_JOYSTICK_1, position, 2);
|
||
\end{lstlisting}
|
||
|
||
After this call, the first element of the \textit{position} array will
|
||
hold the X axis position of the joystick, and the second element will hold
|
||
the Y axis position. In this example we do not use the information about
|
||
how many axes were really returned.
|
||
|
||
The position of an axis can be in the range -1.0 to 1.0, where positive
|
||
values represent right, forward or up directions, while negative values
|
||
represent left, back or down directions. If a requested axis is not
|
||
supported by the joystick, the corresponding array element will be set
|
||
to zero. The same goes for the situation when the joystick is not
|
||
connected (all axes are treated as unsupported).
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Joystick buttons}
|
||
A function similar to the \textbf{glfwGetJoystickPos} function is
|
||
available for querying the state of joystick buttons, namely the
|
||
\textbf{glfwGetJoystickButtons} function:
|
||
|
||
\begin{lstlisting}
|
||
int glfwGetJoystickButtons(int joy, unsigned char* buttons,
|
||
int numbuttons)
|
||
\end{lstlisting}
|
||
|
||
The function works just like the \textbf{glfwGetJoystickAxis} function, except
|
||
that it returns the state of joystick buttons instead of axis positions. Each
|
||
button in the array specified by the \textit{buttons} argument can be either
|
||
GLFW\_PRESS or GLFW\_RELEASE, indicating whether the corresponding button is
|
||
currently held down or not. Unsupported buttons will have the value
|
||
GLFW\_RELEASE.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
% Timing
|
||
%-------------------------------------------------------------------------
|
||
\chapter{Timing}
|
||
\thispagestyle{fancy}
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{High Resolution Timer}
|
||
In most applications, it is useful to know exactly how much time has
|
||
passed between point $A$ and point $B$ in a program. A typical situation
|
||
is in a game, where you need to know how much time has passed between two
|
||
rendered frames in order to calculate the correct movement and physics
|
||
etc. Another example is when you want to benchmark a certain piece of
|
||
code.
|
||
|
||
\GLFW\ provides a high-resolution timer, which reports a double precision
|
||
floating point value representing the number of seconds that have passed
|
||
since \textbf{glfwInit} was called. The timer is accessed with the
|
||
function \textbf{glfwGetTime}:
|
||
|
||
\begin{lstlisting}
|
||
double glfwGetTime(void)
|
||
\end{lstlisting}
|
||
|
||
The precision of the timer depends on which computer and operating
|
||
system you are running, but it is almost guaranteed to be better than
|
||
$10~ms$, and in most cases it is much better than $1~ms$ (on a modern PC
|
||
you can get resolutions in the order of $1~ns$).
|
||
|
||
It is possible to set the value of the high precision timer using the
|
||
\textbf{glfwSetTime} function:
|
||
|
||
\begin{lstlisting}
|
||
void glfwSetTime(double time)
|
||
\end{lstlisting}
|
||
|
||
The argument \textit{time} is the time, in seconds, that the timer should
|
||
be set to.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
% OpenGL Extension Support
|
||
%-------------------------------------------------------------------------
|
||
\chapter{OpenGL Extension Support}
|
||
\thispagestyle{fancy}
|
||
One of the benefits of \OpenGL\ is that it is extensible. Independent
|
||
hardware vendors (IHVs) may include functionality in their \OpenGL\
|
||
implementations that exceed that of the \OpenGL\ standard.
|
||
|
||
An extension is defined by:
|
||
|
||
\begin{enumerate}
|
||
\item An extension name (e.g. GL\_ARB\_multitexture).
|
||
\item New OpenGL tokens (e.g. GL\_TEXTURE1\_ARB).
|
||
\item New OpenGL functions (e.g. \textbf{glActiveTextureARB}).
|
||
\end{enumerate}
|
||
|
||
A list of official extensions, together with their definitions, can be
|
||
found at the \textit{OpenGL Registry}
|
||
(\url{http://www.opengl.org/registry/}).
|
||
|
||
To use a certain extension, the following steps must be performed:
|
||
|
||
\begin{enumerate}
|
||
\item A compile time check for the support of the extension.
|
||
\item A run time check for the support of the extension.
|
||
\item Fetch function pointers for the extended \OpenGL\ functions (done at
|
||
run time).
|
||
\end{enumerate}
|
||
|
||
How this is done using \GLFW\ is described in the following sections.
|
||
Please note that this chapter covers some advanced topics, and is quite
|
||
specific to the C programming language.
|
||
|
||
For a much easier way to get access to \OpenGL\ extensions, you should probably
|
||
use a dedicated extension loading library such as GLEW or GLee. This kind of
|
||
library greatly reduces the amount of work necessary to use \OpenGL\
|
||
extensions. GLEW in particular has been extensively tested with and works well
|
||
with \GLFW .
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Compile Time Check}
|
||
The compile time check is necessary to perform in order to know if the
|
||
compiler include files have defined the necessary tokens. It is very easy
|
||
to do. The include file \texttt{GL/gl.h} will define a constant with the
|
||
same name as the extension, if all the extension tokens are defined. Here
|
||
is an example of how to check for the extension GL\_ARB\_multitexture:
|
||
|
||
\begin{lstlisting}
|
||
#ifdef GL_ARB_multitexture
|
||
// Extension is supported by the include files
|
||
#else
|
||
// Extension is not supported by the include files
|
||
// Get a more up-to-date <GL/gl.h> file!
|
||
#endif
|
||
\end{lstlisting}
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Runtime Check}
|
||
Even if the compiler include files have defined all the necessary tokens, a
|
||
given machine may not actually support the extension (it may have a graphics
|
||
card with a different \OpenGL\ implementation, or an older driver). That is why
|
||
it is necessary to do a run time check for the extension support as well. This
|
||
is done with the \GLFW\ function \textbf{glfwExtensionSupported}, which has the
|
||
C syntax:
|
||
|
||
\begin{lstlisting}
|
||
int glfwExtensionSupported(const char* extension)
|
||
\end{lstlisting}
|
||
|
||
The argument \textit{extension} is a null terminated ASCII string
|
||
with the extension name. \textbf{glfwExtensionSupported} returns GL\_TRUE
|
||
if the extension is supported, otherwise it returns GL\_FALSE.
|
||
|
||
Let us extend the previous example of checking for support of the
|
||
extension GL\_ARB\_multitexture. This time we add a run time check, and a
|
||
variable which we set to GL\_TRUE if the extension is supported, or
|
||
GL\_FALSE if it is not supported.
|
||
|
||
\begin{lstlisting}
|
||
int multitexture_supported;
|
||
|
||
#ifdef GL_ARB_multitexture
|
||
// Check if extension is supported at run time
|
||
multitexture_supported = glfwExtensionSupported("GL_ARB_multitexture");
|
||
#else
|
||
// Extension is not supported by the include files
|
||
// Get a more up-to-date <GL/gl.h> file!
|
||
multitexture_supported = GL_FALSE;
|
||
#endif
|
||
\end{lstlisting}
|
||
|
||
Now it is easy to check for the extension within the program, simply do:
|
||
|
||
\begin{lstlisting}
|
||
if (multitexture_supported)
|
||
{
|
||
// Use multi texturing
|
||
}
|
||
else
|
||
{
|
||
// Use some other solution (or fail)
|
||
}
|
||
\end{lstlisting}
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\section{Fetching Function Pointers}
|
||
Some extensions, though not all, require the use of new \OpenGL\ functions.
|
||
These entry points are not necessarily exposed by your link libraries, making
|
||
it necessary to find them dynamically at run time. You can retrieve these
|
||
entry points using the \textbf{glfwGetProcAddress} function:
|
||
|
||
\begin{lstlisting}
|
||
void* glfwGetProcAddress(const char* procname)
|
||
\end{lstlisting}
|
||
|
||
The argument \textit{procname} is a null terminated ASCII string
|
||
holding the name of the \OpenGL\ function. \textbf{glfwGetProcAddress}
|
||
returns the address to the function if the function is available,
|
||
otherwise NULL is returned.
|
||
|
||
Obviously, fetching the function pointer is trivial. For instance, if we
|
||
want to obtain the pointer to \textbf{glActiveTextureARB}, we simply call:
|
||
|
||
\begin{lstlisting}
|
||
glActiveTextureARB = glfwGetProcAddress("glActiveTextureARB");
|
||
\end{lstlisting}
|
||
|
||
However, there are many possible naming and type definition conflicts
|
||
involved with such an operation, which may result in compiler warnings or
|
||
errors. My proposed solution is the following:
|
||
|
||
\begin{itemize}
|
||
\item Do not use the function name for the variable name. Use something
|
||
similar, perhaps by adding a prefix or suffix, and then use
|
||
\texttt{\#define} to map the function name to your variable.
|
||
\item The standard type definition naming convention for function pointers
|
||
is \texttt{PFN\textit{xxxx}PROC}, where \texttt{\textit{xxxx}} is
|
||
the uppercase version of the function name (e.g.
|
||
\texttt{PFNGLACTIVETEXTUREARBPROC}). Either make sure your compiler uses
|
||
a compatible \texttt{gl.h} and/or \texttt{glext.h} file and rely on it to
|
||
define these types, or use define the types yourself using a different
|
||
naming convention (for example \texttt{\textit{xxxx}\_T}) and do the
|
||
type definitions yourself.
|
||
\end{itemize}
|
||
|
||
Here is a slightly longer example of how to use an extension, this time using
|
||
our own function pointer type definition):
|
||
|
||
\begin{lstlisting}
|
||
// Type definition of the function pointer
|
||
typedef void (APIENTRY * GLACTIVETEXTUREARB_T) (GLenum texture);
|
||
|
||
// Function pointer
|
||
GLACTIVETEXTUREARB_T _ActiveTextureARB;
|
||
#define glActiveTextureARB _ActiveTextureARB
|
||
|
||
// Extension availability flag
|
||
int multitexture_supported;
|
||
|
||
#ifdef GL_ARB_multitexture
|
||
// Check if extension is supported at run time
|
||
if (glfwExtensionSupported("GL_ARB_multitexture"))
|
||
{
|
||
// Get the function pointer
|
||
glActiveTextureARB = (GLACTIVETEXTUREARB_T)
|
||
glfwGetProcAddress("glActiveTextureARB");
|
||
|
||
multitexture_supported = GL_TRUE;
|
||
}
|
||
else
|
||
{
|
||
multitexture_supported = GL_FALSE;
|
||
}
|
||
#else
|
||
// Extension is not supported by the include files
|
||
multitexture_supported = GL_FALSE;
|
||
#endif
|
||
\end{lstlisting}
|
||
|
||
Even this example leaves some things to be desired. First of all, the
|
||
GL\_ARB\_multitexture extension defines many more functions than the single
|
||
function used above. Secondly, checking if an extension is supported using
|
||
\textbf{glfwExtensionSupported} is not enough to ensure that the corresponding
|
||
functions will be valid. You also need to check that the all function pointers
|
||
returned by \textbf{glfwGetProcAddress} are non-NULL.
|
||
|
||
|
||
%-------------------------------------------------------------------------
|
||
\subsection{Function pointer type definitions}
|
||
To make a function pointer type definition, you need to know the function
|
||
prototype. This can often be found in the extension definitions (e.g. at
|
||
the \textit{OpenGL Registry}). All the entry points that are defined by an
|
||
extension are listed with their C prototype definitions under the section
|
||
\textit{New Procedures and Functions} in the extension definition.
|
||
|
||
For instance, if we look at the definition of the
|
||
GL\_ARB\_texture\_compression extension, we find a list of new functions.
|
||
One of these is declared like this:
|
||
|
||
\begin{lstlisting}
|
||
void GetCompressedTexImageARB(enum target, int lod, void* img);
|
||
\end{lstlisting}
|
||
|
||
Like in most official \OpenGL\ documentation, all the \texttt{GL} and
|
||
\texttt{gl} prefixes have been left out. In other words, the real function
|
||
prototype would look like this:
|
||
|
||
\begin{lstlisting}
|
||
void glGetCompressedTexImageARB(GLenum target, GLint lod, void* img);
|
||
\end{lstlisting}
|
||
|
||
All we have to do to turn this prototype definition into a function
|
||
pointer type definition, is to replace the function name with
|
||
\texttt{(APIENTRY * \textit{xxxx}\_T)}, where \textit{xxxx} is the
|
||
uppercase version of the name (according to the proposed naming
|
||
convention). The keyword \texttt{APIENTRY} is needed to be compatible
|
||
between different platforms. The \GLFW\ header file \texttt{GL/glfw.h}
|
||
ensures that \texttt{APIENTRY} is properly defined on all supported platforms.
|
||
|
||
In other words, for the function \textbf{glGetCompressedTexImageARB} we
|
||
get:
|
||
|
||
\begin{lstlisting}
|
||
typedef void (APIENTRY * GLGETCOMPRESSEDTEXIMAGEARB_T)
|
||
(GLenum target, GLint level, void* img);
|
||
\end{lstlisting}
|
||
|
||
|
||
\end{document}
|