mirror of
https://github.com/NGSolve/netgen.git
synced 2025-01-12 06:00:33 +05:00
4034 lines
120 KiB
C
4034 lines
120 KiB
C
/* $Id: togl.c,v 1.73 2005/10/26 07:40:22 gregcouch Exp $ */
|
|
|
|
/* vi:set sw=4: */
|
|
|
|
/*
|
|
* Togl - a Tk OpenGL widget
|
|
*
|
|
* Copyright (C) 1996-2002 Brian Paul and Ben Bederson
|
|
* See the LICENSE file for copyright details.
|
|
*/
|
|
|
|
/*
|
|
* Currently we support X11, Win32 and Macintosh only
|
|
*/
|
|
|
|
#include "togl.h"
|
|
|
|
/* Use TCL_STUPID to cast (const char *) to (char *) where the Tcl function
|
|
* prototype argument should really be const */
|
|
#define TCL_STUPID (char *)
|
|
|
|
/* Use WIDGREC to cast widgRec arguments */
|
|
#define WIDGREC (char *)
|
|
|
|
/*** Windows headers ***/
|
|
#if defined(TOGL_WGL)
|
|
# define WIN32_LEAN_AND_MEAN
|
|
# include <windows.h>
|
|
# undef WIN32_LEAN_AND_MEAN
|
|
# include <winnt.h>
|
|
|
|
/*** X Window System headers ***/
|
|
#elif defined(TOGL_X11)
|
|
# include <X11/Xlib.h>
|
|
# include <X11/Xutil.h>
|
|
# include <X11/Xatom.h> /* for XA_RGB_DEFAULT_MAP atom */
|
|
# if defined(__vms)
|
|
# include <X11/StdCmap.h> /* for XmuLookupStandardColormap */
|
|
# else
|
|
# include <X11/Xmu/StdCmap.h> /* for XmuLookupStandardColormap */
|
|
# endif
|
|
# include <GL/glx.h>
|
|
|
|
/*** Mac headers ***/
|
|
#elif defined(TOGL_AGL_CLASSIC)
|
|
# include <Gestalt.h>
|
|
# include <Traps.h>
|
|
# include <agl.h>
|
|
# include <tclMacCommonPch.h>
|
|
|
|
#elif defined(TOGL_AGL)
|
|
# define Cursor QDCursor
|
|
# include <AGL/agl.h>
|
|
# undef Cursor
|
|
# include "tkMacOSX.h"
|
|
# include <tkMacOSXInt.h> /* usa MacDrawable */
|
|
# include <ApplicationServices/ApplicationServices.h>
|
|
|
|
#else /* make sure only one platform defined */
|
|
# error Unsupported platform, or confused platform defines...
|
|
#endif
|
|
|
|
/*** Standard C headers ***/
|
|
#include <assert.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#ifdef TOGL_WGL
|
|
# include <tkPlatDecls.h>
|
|
#endif
|
|
|
|
#if TK_MAJOR_VERSION < 8
|
|
# error Sorry Togl requires Tcl/Tk ver 8.0 or higher.
|
|
#endif
|
|
|
|
#if defined(TOGL_AGL_CLASSIC)
|
|
# if TK_MAJOR_VERSION < 8 || (TK_MAJOR_VERSION == 8 && TK_MINOR_VERSION < 3)
|
|
# error Sorry Mac classic version requires Tcl/Tk ver 8.3.0 or higher.
|
|
# endif
|
|
#endif /* TOGL_AGL_CLASSIC */
|
|
|
|
#if defined(TOGL_AGL)
|
|
# if TK_MAJOR_VERSION < 8 || (TK_MAJOR_VERSION == 8 && TK_MINOR_VERSION < 4)
|
|
# error Sorry Mac Aqua version requires Tcl/Tk ver 8.4.0 or higher.
|
|
# endif
|
|
#endif /* TOGL_AGL */
|
|
|
|
/* workaround for bug #123153 in tcl ver8.4a2 (tcl.h) */
|
|
#if defined(Tcl_InitHashTable) && defined(USE_TCL_STUBS)
|
|
# undef Tcl_InitHashTable
|
|
# define Tcl_InitHashTable (tclStubsPtr->tcl_InitHashTable)
|
|
#endif
|
|
#if TK_MAJOR_VERSION > 8 || (TK_MAJOR_VERSION == 8 && TK_MINOR_VERSION >= 4)
|
|
# define HAVE_TK_SETCLASSPROCS
|
|
/* pointer to Tk_SetClassProcs function in the stub table */
|
|
|
|
static void (*SetClassProcsPtr)
|
|
_ANSI_ARGS_((Tk_Window, Tk_ClassProcs *, ClientData));
|
|
#endif
|
|
|
|
/*
|
|
* Copy of TkClassProcs declarations form tkInt.h
|
|
* (this is needed for Tcl ver =< 8.4a3)
|
|
*/
|
|
|
|
typedef Window (TkClassCreateProc) _ANSI_ARGS_((Tk_Window tkwin,
|
|
Window parent, ClientData instanceData));
|
|
typedef void (TkClassGeometryProc) _ANSI_ARGS_((ClientData instanceData));
|
|
typedef void (TkClassModalProc) _ANSI_ARGS_((Tk_Window tkwin,
|
|
XEvent *eventPtr));
|
|
typedef struct TkClassProcs
|
|
{
|
|
TkClassCreateProc *createProc;
|
|
TkClassGeometryProc *geometryProc;
|
|
TkClassModalProc *modalProc;
|
|
} TkClassProcs;
|
|
|
|
|
|
/* Defaults */
|
|
#define DEFAULT_WIDTH "400"
|
|
#define DEFAULT_HEIGHT "400"
|
|
#define DEFAULT_IDENT ""
|
|
#define DEFAULT_FONTNAME "fixed"
|
|
#define DEFAULT_TIME "1"
|
|
|
|
|
|
#ifdef TOGL_WGL
|
|
/* Maximum size of a logical palette corresponding to a colormap in color index
|
|
* mode. */
|
|
# define MAX_CI_COLORMAP_SIZE 4096
|
|
|
|
# if TOGL_USE_FONTS != 1
|
|
/*
|
|
* copy of TkWinColormap from tkWinInt.h
|
|
*/
|
|
|
|
typedef struct
|
|
{
|
|
HPALETTE palette; /* Palette handle used when drawing. */
|
|
UINT size; /* Number of entries in the palette. */
|
|
int stale; /* 1 if palette needs to be realized, otherwise
|
|
* 0. If the palette is stale, then an idle
|
|
* handler is scheduled to realize the palette. */
|
|
Tcl_HashTable refCounts; /* Hash table of palette entry reference counts
|
|
* indexed by pixel value. */
|
|
} TkWinColormap;
|
|
# else
|
|
# include "tkWinInt.h"
|
|
# endif
|
|
|
|
static LRESULT(CALLBACK *tkWinChildProc) (HWND hwnd, UINT message,
|
|
WPARAM wParam, LPARAM lParam) = NULL;
|
|
|
|
# define TK_WIN_CHILD_CLASS_NAME "TkChild"
|
|
|
|
#endif /* TOGL_WGL */
|
|
|
|
|
|
#define MAX(a,b) (((a)>(b))?(a):(b))
|
|
|
|
#define TCL_ERR(interp, string) \
|
|
do { \
|
|
Tcl_ResetResult(interp); \
|
|
Tcl_AppendResult(interp, string, NULL); \
|
|
return TCL_ERROR; \
|
|
} while (0)
|
|
|
|
/* The constant DUMMY_WINDOW is used to signal window creation failure from the
|
|
* Togl_CreateWindow() */
|
|
#define DUMMY_WINDOW ((Window) -1)
|
|
|
|
#define ALL_EVENTS_MASK \
|
|
(KeyPressMask | \
|
|
KeyReleaseMask | \
|
|
ButtonPressMask | \
|
|
ButtonReleaseMask | \
|
|
EnterWindowMask | \
|
|
LeaveWindowMask | \
|
|
PointerMotionMask | \
|
|
ExposureMask | \
|
|
VisibilityChangeMask | \
|
|
FocusChangeMask | \
|
|
PropertyChangeMask | \
|
|
ColormapChangeMask)
|
|
|
|
struct Togl
|
|
{
|
|
Togl *Next; /* next in linked list */
|
|
|
|
#if defined(TOGL_WGL)
|
|
HDC tglGLHdc; /* Device context of device that OpenGL calls
|
|
* will be drawn on */
|
|
HGLRC tglGLHglrc; /* OpenGL rendering context to be made current */
|
|
int CiColormapSize; /* (Maximum) size of colormap in color index
|
|
* mode */
|
|
#elif defined(TOGL_X11)
|
|
GLXContext GlCtx; /* Normal planes GLX context */
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
AGLContext aglCtx;
|
|
#endif /* TOGL_WGL */
|
|
|
|
Display *display; /* X's token for the window's display. */
|
|
Tk_Window TkWin; /* Tk window structure */
|
|
Tcl_Interp *Interp; /* Tcl interpreter */
|
|
Tcl_Command widgetCmd; /* Token for togl's widget command */
|
|
#ifndef NO_TK_CURSOR
|
|
Tk_Cursor Cursor; /* The widget's cursor */
|
|
#endif
|
|
int Width, Height; /* Dimensions of window */
|
|
int SetGrid; /* positive is grid size for window manager */
|
|
int TimerInterval; /* Time interval for timer in milliseconds */
|
|
#if (TCL_MAJOR_VERSION * 100 + TCL_MINOR_VERSION) >= 705
|
|
Tcl_TimerToken timerHandler; /* Token for togl's timer handler */
|
|
#else
|
|
Tk_TimerToken timerHandler; /* Token for togl's timer handler */
|
|
#endif
|
|
Bool RgbaFlag; /* configuration flags (ala GLX parameters) */
|
|
int RgbaRed;
|
|
int RgbaGreen;
|
|
int RgbaBlue;
|
|
Bool DoubleFlag;
|
|
Bool DepthFlag;
|
|
int DepthSize;
|
|
Bool AccumFlag;
|
|
int AccumRed;
|
|
int AccumGreen;
|
|
int AccumBlue;
|
|
int AccumAlpha;
|
|
Bool AlphaFlag;
|
|
int AlphaSize;
|
|
Bool StencilFlag;
|
|
int StencilSize;
|
|
Bool PrivateCmapFlag;
|
|
Bool OverlayFlag;
|
|
Bool StereoFlag;
|
|
#ifdef __sgi
|
|
Bool OldStereoFlag;
|
|
#endif
|
|
int AuxNumber;
|
|
Bool Indirect;
|
|
int PixelFormat;
|
|
const char *ShareList; /* name (ident) of Togl to share dlists with */
|
|
const char *ShareContext; /* name (ident) to share OpenGL context with */
|
|
|
|
const char *Ident; /* User's identification string */
|
|
ClientData Client_Data; /* Pointer to user data */
|
|
|
|
Bool UpdatePending; /* Should normal planes be redrawn? */
|
|
|
|
Togl_Callback *CreateProc; /* Callback when widget is created */
|
|
Togl_Callback *DisplayProc; /* Callback when widget is rendered */
|
|
Togl_Callback *ReshapeProc; /* Callback when window size changes */
|
|
Togl_Callback *DestroyProc; /* Callback when widget is destroyed */
|
|
Togl_Callback *TimerProc; /* Callback when widget is idle */
|
|
|
|
/* Overlay stuff */
|
|
#if defined(TOGL_X11)
|
|
GLXContext OverlayCtx; /* Overlay planes OpenGL context */
|
|
#elif defined(TOGL_WGL)
|
|
HGLRC tglGLOverlayHglrc;
|
|
#endif /* TOGL_X11 */
|
|
|
|
Window OverlayWindow; /* The overlay window, or 0 */
|
|
Togl_Callback *OverlayDisplayProc; /* Overlay redraw proc */
|
|
Bool OverlayUpdatePending; /* Should overlay be redrawn? */
|
|
Colormap OverlayCmap; /* colormap for overlay is created */
|
|
int OverlayTransparentPixel; /* transparent pixel */
|
|
Bool OverlayIsMapped;
|
|
|
|
/* for DumpToEpsFile: Added by Miguel A. de Riera Pasenau 10.01.1997 */
|
|
XVisualInfo *VisInfo; /* Visual info of the current */
|
|
/* context needed for DumpToEpsFile */
|
|
GLfloat *EpsRedMap; /* Index2RGB Maps for Color index modes */
|
|
GLfloat *EpsGreenMap;
|
|
GLfloat *EpsBlueMap;
|
|
GLint EpsMapSize; /* = Number of indices in our Togl */
|
|
};
|
|
|
|
|
|
/* NTNTNT need to change to handle Windows Data Types */
|
|
/*
|
|
* Prototypes for functions local to this file
|
|
*/
|
|
static int Togl_Cmd(ClientData clientData, Tcl_Interp *interp,
|
|
int argc, CONST84 char **argv);
|
|
static void Togl_EventProc(ClientData clientData, XEvent *eventPtr);
|
|
static Window Togl_CreateWindow(Tk_Window, Window, ClientData);
|
|
static void Togl_WorldChanged(ClientData);
|
|
|
|
#ifdef MESA_COLOR_HACK
|
|
static int get_free_color_cells(Display *display, int screen,
|
|
Colormap colormap);
|
|
static void free_default_color_cells(Display *display, Colormap colormap);
|
|
#endif
|
|
static void ToglCmdDeletedProc(ClientData);
|
|
|
|
|
|
|
|
#if defined(__sgi)
|
|
/* SGI-only stereo */
|
|
static void oldStereoMakeCurrent(Display *dpy, Window win, GLXContext ctx);
|
|
static void oldStereoInit(Togl *togl, int stereoEnabled);
|
|
#endif
|
|
|
|
#if defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
static void SetMacBufRect(Togl *togl);
|
|
#endif
|
|
|
|
|
|
/*
|
|
* Setup Togl widget configuration options:
|
|
*/
|
|
|
|
static Tk_ConfigSpec configSpecs[] = {
|
|
{TK_CONFIG_PIXELS, TCL_STUPID "-height", "height", "Height",
|
|
DEFAULT_HEIGHT, Tk_Offset(Togl, Height), 0, NULL},
|
|
|
|
{TK_CONFIG_PIXELS, TCL_STUPID "-width", "width", "Width",
|
|
DEFAULT_WIDTH, Tk_Offset(Togl, Width), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-setgrid", "setGrid", "SetGrid",
|
|
"0", Tk_Offset(Togl, SetGrid), 0},
|
|
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-rgba", "rgba", "Rgba",
|
|
"true", Tk_Offset(Togl, RgbaFlag), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-redsize", "redsize", "RedSize",
|
|
"1", Tk_Offset(Togl, RgbaRed), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-greensize", "greensize", "GreenSize",
|
|
"1", Tk_Offset(Togl, RgbaGreen), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-bluesize", "bluesize", "BlueSize",
|
|
"1", Tk_Offset(Togl, RgbaBlue), 0, NULL},
|
|
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-double", "double", "Double",
|
|
"false", Tk_Offset(Togl, DoubleFlag), 0, NULL},
|
|
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-depth", "depth", "Depth",
|
|
"false", Tk_Offset(Togl, DepthFlag), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-depthsize", "depthsize", "DepthSize",
|
|
"1", Tk_Offset(Togl, DepthSize), 0, NULL},
|
|
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-accum", "accum", "Accum",
|
|
"false", Tk_Offset(Togl, AccumFlag), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-accumredsize", "accumredsize", "AccumRedSize",
|
|
"1", Tk_Offset(Togl, AccumRed), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-accumgreensize", "accumgreensize",
|
|
"AccumGreenSize",
|
|
"1", Tk_Offset(Togl, AccumGreen), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-accumbluesize", "accumbluesize",
|
|
"AccumBlueSize",
|
|
"1", Tk_Offset(Togl, AccumBlue), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-accumalphasize", "accumalphasize",
|
|
"AccumAlphaSize",
|
|
"1", Tk_Offset(Togl, AccumAlpha), 0, NULL},
|
|
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-alpha", "alpha", "Alpha",
|
|
"false", Tk_Offset(Togl, AlphaFlag), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-alphasize", "alphasize", "AlphaSize",
|
|
"1", Tk_Offset(Togl, AlphaSize), 0, NULL},
|
|
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-stencil", "stencil", "Stencil",
|
|
"false", Tk_Offset(Togl, StencilFlag), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-stencilsize", "stencilsize", "StencilSize",
|
|
"1", Tk_Offset(Togl, StencilSize), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-auxbuffers", "auxbuffers", "AuxBuffers",
|
|
"0", Tk_Offset(Togl, AuxNumber), 0, NULL},
|
|
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-privatecmap", "privateCmap", "PrivateCmap",
|
|
"false", Tk_Offset(Togl, PrivateCmapFlag), 0, NULL},
|
|
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-overlay", "overlay", "Overlay",
|
|
"false", Tk_Offset(Togl, OverlayFlag), 0, NULL},
|
|
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-stereo", "stereo", "Stereo",
|
|
"false", Tk_Offset(Togl, StereoFlag), 0, NULL},
|
|
|
|
#ifdef __sgi
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-oldstereo", "oldstereo", "OldStereo",
|
|
"false", Tk_Offset(Togl, OldStereoFlag), 0, NULL},
|
|
#endif
|
|
|
|
#ifndef NO_TK_CURSOR
|
|
{TK_CONFIG_ACTIVE_CURSOR, TCL_STUPID "-cursor", "cursor", "Cursor",
|
|
"", Tk_Offset(Togl, Cursor), TK_CONFIG_NULL_OK},
|
|
#endif
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-time", "time", "Time",
|
|
DEFAULT_TIME, Tk_Offset(Togl, TimerInterval), 0, NULL},
|
|
|
|
{TK_CONFIG_STRING, TCL_STUPID "-sharelist", "sharelist", "ShareList",
|
|
NULL, Tk_Offset(Togl, ShareList), 0, NULL},
|
|
|
|
{TK_CONFIG_STRING, TCL_STUPID "-sharecontext", "sharecontext",
|
|
"ShareContext", NULL, Tk_Offset(Togl, ShareContext), 0, NULL},
|
|
|
|
{TK_CONFIG_STRING, TCL_STUPID "-ident", "ident", "Ident",
|
|
DEFAULT_IDENT, Tk_Offset(Togl, Ident), 0, NULL},
|
|
|
|
{TK_CONFIG_BOOLEAN, TCL_STUPID "-indirect", "indirect", "Indirect",
|
|
"false", Tk_Offset(Togl, Indirect), 0, NULL},
|
|
|
|
{TK_CONFIG_INT, TCL_STUPID "-pixelformat", "pixelFormat", "PixelFormat",
|
|
"0", Tk_Offset(Togl, PixelFormat), 0, NULL},
|
|
|
|
{TK_CONFIG_END, NULL, NULL, NULL, NULL, 0, 0, NULL}
|
|
};
|
|
|
|
|
|
/*
|
|
* Default callback pointers. When a new Togl widget is created it
|
|
* will be assigned these initial callbacks.
|
|
*/
|
|
static Togl_Callback *DefaultCreateProc = NULL;
|
|
static Togl_Callback *DefaultDisplayProc = NULL;
|
|
static Togl_Callback *DefaultReshapeProc = NULL;
|
|
static Togl_Callback *DefaultDestroyProc = NULL;
|
|
static Togl_Callback *DefaultOverlayDisplayProc = NULL;
|
|
static Togl_Callback *DefaultTimerProc = NULL;
|
|
static ClientData DefaultClientData = NULL;
|
|
static Tcl_HashTable CommandTable;
|
|
|
|
/*
|
|
* Head of linked list of all Togl widgets
|
|
*/
|
|
static Togl *ToglHead = NULL;
|
|
|
|
/*
|
|
* Add given togl widget to linked list.
|
|
*/
|
|
static void
|
|
AddToList(Togl *t)
|
|
{
|
|
t->Next = ToglHead;
|
|
ToglHead = t;
|
|
}
|
|
|
|
/*
|
|
* Remove given togl widget from linked list.
|
|
*/
|
|
static void
|
|
RemoveFromList(Togl *t)
|
|
{
|
|
Togl *prev = NULL;
|
|
Togl *pos = ToglHead;
|
|
|
|
while (pos) {
|
|
if (pos == t) {
|
|
if (prev) {
|
|
prev->Next = pos->Next;
|
|
} else {
|
|
ToglHead = pos->Next;
|
|
}
|
|
return;
|
|
}
|
|
prev = pos;
|
|
pos = pos->Next;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Return pointer to togl widget given a user identifier string.
|
|
*/
|
|
static Togl *
|
|
FindTogl(const char *ident)
|
|
{
|
|
Togl *t = ToglHead;
|
|
|
|
while (t) {
|
|
if (strcmp(t->Ident, ident) == 0)
|
|
return t;
|
|
t = t->Next;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
#if defined(TOGL_X11)
|
|
/*
|
|
* Return pointer to another togl widget with same OpenGL context.
|
|
*/
|
|
static Togl *
|
|
FindToglWithSameContext(Togl *togl)
|
|
{
|
|
Togl *t;
|
|
|
|
for (t = ToglHead; t != NULL; t = t->Next) {
|
|
if (t == togl)
|
|
continue;
|
|
# if defined(TOGL_WGL)
|
|
if (t->tglGLHglrc == togl->tglGLHglrc)
|
|
# elif defined(TOGL_X11)
|
|
if (t->GlCtx == togl->GlCtx)
|
|
# elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
if (t->aglCtx == togl->aglCtx)
|
|
# endif
|
|
return t;
|
|
}
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
#ifdef USE_OVERLAY
|
|
/*
|
|
* Return pointer to another togl widget with same OpenGL overlay context.
|
|
*/
|
|
static Togl *
|
|
FindToglWithSameOverlayContext(Togl *togl)
|
|
{
|
|
Togl *t;
|
|
|
|
for (t = ToglHead; t != NULL; t = t->Next) {
|
|
if (t == togl)
|
|
continue;
|
|
# if defined(TOGL_X11)
|
|
if (t->OverlayCtx == togl->OverlayCtx)
|
|
# elif defined(TOGL_WGL)
|
|
if (t->tglGLOverlayHglrc == togl->tglGLOverlayHglrc)
|
|
# endif
|
|
return t;
|
|
}
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
#if defined(TOGL_X11)
|
|
/*
|
|
* Return an X colormap to use for OpenGL RGB-mode rendering.
|
|
* Input: dpy - the X display
|
|
* scrnum - the X screen number
|
|
* visinfo - the XVisualInfo as returned by glXChooseVisual()
|
|
* Return: an X Colormap or 0 if there's a _serious_ error.
|
|
*/
|
|
static Colormap
|
|
get_rgb_colormap(Display *dpy,
|
|
int scrnum, const XVisualInfo *visinfo, Tk_Window tkwin)
|
|
{
|
|
Atom hp_cr_maps;
|
|
Status status;
|
|
int numCmaps;
|
|
int i;
|
|
XStandardColormap *standardCmaps;
|
|
Window root = XRootWindow(dpy, scrnum);
|
|
Bool using_mesa;
|
|
|
|
/*
|
|
* First check if visinfo's visual matches the default/root visual.
|
|
*/
|
|
if (visinfo->visual == Tk_Visual(tkwin)) {
|
|
/* use the default/root colormap */
|
|
Colormap cmap;
|
|
|
|
cmap = Tk_Colormap(tkwin);
|
|
# ifdef MESA_COLOR_HACK
|
|
(void) get_free_color_cells(dpy, scrnum, cmap);
|
|
# endif
|
|
return cmap;
|
|
}
|
|
|
|
/*
|
|
* Check if we're using Mesa.
|
|
*/
|
|
if (strstr(glXQueryServerString(dpy, scrnum, GLX_VERSION), "Mesa")) {
|
|
using_mesa = True;
|
|
} else {
|
|
using_mesa = False;
|
|
}
|
|
|
|
/*
|
|
* Next, if we're using Mesa and displaying on an HP with the "Color
|
|
* Recovery" feature and the visual is 8-bit TrueColor, search for a
|
|
* special colormap initialized for dithering. Mesa will know how to
|
|
* dither using this colormap.
|
|
*/
|
|
if (using_mesa) {
|
|
hp_cr_maps = XInternAtom(dpy, "_HP_RGB_SMOOTH_MAP_LIST", True);
|
|
if (hp_cr_maps
|
|
# ifdef __cplusplus
|
|
&& visinfo->visual->c_class == TrueColor
|
|
# else
|
|
&& visinfo->visual->class == TrueColor
|
|
# endif
|
|
&& visinfo->depth == 8) {
|
|
status = XGetRGBColormaps(dpy, root, &standardCmaps,
|
|
&numCmaps, hp_cr_maps);
|
|
if (status) {
|
|
for (i = 0; i < numCmaps; i++) {
|
|
if (standardCmaps[i].visualid == visinfo->visual->visualid) {
|
|
Colormap cmap = standardCmaps[i].colormap;
|
|
|
|
(void) XFree(standardCmaps);
|
|
return cmap;
|
|
}
|
|
}
|
|
(void) XFree(standardCmaps);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Next, try to find a standard X colormap.
|
|
*/
|
|
# if !HP && !SUN
|
|
# ifndef SOLARIS_BUG
|
|
status = XmuLookupStandardColormap(dpy, visinfo->screen,
|
|
visinfo->visualid, visinfo->depth, XA_RGB_DEFAULT_MAP,
|
|
/* replace */ False, /* retain */ True);
|
|
if (status == 1) {
|
|
status = XGetRGBColormaps(dpy, root, &standardCmaps,
|
|
&numCmaps, XA_RGB_DEFAULT_MAP);
|
|
if (status == 1) {
|
|
for (i = 0; i < numCmaps; i++) {
|
|
if (standardCmaps[i].visualid == visinfo->visualid) {
|
|
Colormap cmap = standardCmaps[i].colormap;
|
|
|
|
(void) XFree(standardCmaps);
|
|
return cmap;
|
|
}
|
|
}
|
|
(void) XFree(standardCmaps);
|
|
}
|
|
}
|
|
# endif
|
|
# endif
|
|
|
|
/*
|
|
* If we get here, give up and just allocate a new colormap.
|
|
*/
|
|
return XCreateColormap(dpy, root, visinfo->visual, AllocNone);
|
|
}
|
|
#elif defined(TOGL_WGL)
|
|
|
|
/* Code to create RGB palette is taken from the GENGL sample program of Win32
|
|
* SDK */
|
|
|
|
static unsigned char threeto8[8] = {
|
|
0, 0111 >> 1, 0222 >> 1, 0333 >> 1, 0444 >> 1, 0555 >> 1, 0666 >> 1, 0377
|
|
};
|
|
|
|
static unsigned char twoto8[4] = {
|
|
0, 0x55, 0xaa, 0xff
|
|
};
|
|
|
|
static unsigned char oneto8[2] = {
|
|
0, 255
|
|
};
|
|
|
|
static int defaultOverride[13] = {
|
|
0, 3, 24, 27, 64, 67, 88, 173, 181, 236, 247, 164, 91
|
|
};
|
|
|
|
static PALETTEENTRY defaultPalEntry[20] = {
|
|
{0, 0, 0, 0},
|
|
{0x80, 0, 0, 0},
|
|
{0, 0x80, 0, 0},
|
|
{0x80, 0x80, 0, 0},
|
|
{0, 0, 0x80, 0},
|
|
{0x80, 0, 0x80, 0},
|
|
{0, 0x80, 0x80, 0},
|
|
{0xC0, 0xC0, 0xC0, 0},
|
|
|
|
{192, 220, 192, 0},
|
|
{166, 202, 240, 0},
|
|
{255, 251, 240, 0},
|
|
{160, 160, 164, 0},
|
|
|
|
{0x80, 0x80, 0x80, 0},
|
|
{0xFF, 0, 0, 0},
|
|
{0, 0xFF, 0, 0},
|
|
{0xFF, 0xFF, 0, 0},
|
|
{0, 0, 0xFF, 0},
|
|
{0xFF, 0, 0xFF, 0},
|
|
{0, 0xFF, 0xFF, 0},
|
|
{0xFF, 0xFF, 0xFF, 0}
|
|
};
|
|
|
|
static unsigned char
|
|
ComponentFromIndex(int i, UINT nbits, UINT shift)
|
|
{
|
|
unsigned char val;
|
|
|
|
val = (unsigned char) (i >> shift);
|
|
switch (nbits) {
|
|
|
|
case 1:
|
|
val &= 0x1;
|
|
return oneto8[val];
|
|
|
|
case 2:
|
|
val &= 0x3;
|
|
return twoto8[val];
|
|
|
|
case 3:
|
|
val &= 0x7;
|
|
return threeto8[val];
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
static Colormap
|
|
Win32CreateRgbColormap(PIXELFORMATDESCRIPTOR pfd)
|
|
{
|
|
TkWinColormap *cmap = (TkWinColormap *) ckalloc(sizeof (TkWinColormap));
|
|
LOGPALETTE *pPal;
|
|
int n, i;
|
|
|
|
n = 1 << pfd.cColorBits;
|
|
pPal = (PLOGPALETTE) LocalAlloc(LMEM_FIXED, sizeof (LOGPALETTE)
|
|
+ n * sizeof (PALETTEENTRY));
|
|
pPal->palVersion = 0x300;
|
|
pPal->palNumEntries = n;
|
|
for (i = 0; i < n; i++) {
|
|
pPal->palPalEntry[i].peRed =
|
|
ComponentFromIndex(i, pfd.cRedBits, pfd.cRedShift);
|
|
pPal->palPalEntry[i].peGreen =
|
|
ComponentFromIndex(i, pfd.cGreenBits, pfd.cGreenShift);
|
|
pPal->palPalEntry[i].peBlue =
|
|
ComponentFromIndex(i, pfd.cBlueBits, pfd.cBlueShift);
|
|
pPal->palPalEntry[i].peFlags = 0;
|
|
}
|
|
|
|
/* fix up the palette to include the default GDI palette */
|
|
if ((pfd.cColorBits == 8)
|
|
&& (pfd.cRedBits == 3) && (pfd.cRedShift == 0)
|
|
&& (pfd.cGreenBits == 3) && (pfd.cGreenShift == 3)
|
|
&& (pfd.cBlueBits == 2) && (pfd.cBlueShift == 6)) {
|
|
for (i = 1; i <= 12; i++)
|
|
pPal->palPalEntry[defaultOverride[i]] = defaultPalEntry[i];
|
|
}
|
|
|
|
cmap->palette = CreatePalette(pPal);
|
|
LocalFree(pPal);
|
|
cmap->size = n;
|
|
cmap->stale = 0;
|
|
|
|
/* Since this is a private colormap of a fix size, we do not need a valid
|
|
* hash table, but a dummy one */
|
|
|
|
Tcl_InitHashTable(&cmap->refCounts, TCL_ONE_WORD_KEYS);
|
|
return (Colormap) cmap;
|
|
}
|
|
|
|
static Colormap
|
|
Win32CreateCiColormap(Togl *togl)
|
|
{
|
|
/* Create a colormap with size of togl->CiColormapSize and set all entries
|
|
* to black */
|
|
|
|
LOGPALETTE logPalette;
|
|
TkWinColormap *cmap = (TkWinColormap *) ckalloc(sizeof (TkWinColormap));
|
|
|
|
logPalette.palVersion = 0x300;
|
|
logPalette.palNumEntries = 1;
|
|
logPalette.palPalEntry[0].peRed = 0;
|
|
logPalette.palPalEntry[0].peGreen = 0;
|
|
logPalette.palPalEntry[0].peBlue = 0;
|
|
logPalette.palPalEntry[0].peFlags = 0;
|
|
|
|
cmap->palette = CreatePalette(&logPalette);
|
|
cmap->size = togl->CiColormapSize;
|
|
ResizePalette(cmap->palette, cmap->size); /* sets new entries to black */
|
|
cmap->stale = 0;
|
|
|
|
/* Since this is a private colormap of a fix size, we do not need a valid
|
|
* hash table, but a dummy one */
|
|
|
|
Tcl_InitHashTable(&cmap->refCounts, TCL_ONE_WORD_KEYS);
|
|
return (Colormap) cmap;
|
|
}
|
|
#endif /* TOGL_X11 */
|
|
|
|
|
|
|
|
/*
|
|
* Togl_Init
|
|
*
|
|
* Called upon system startup to create Togl command.
|
|
*/
|
|
int
|
|
Togl_Init(Tcl_Interp *interp)
|
|
{
|
|
int major, minor, patchLevel, releaseType;
|
|
|
|
#ifdef USE_TCL_STUBS
|
|
if (Tcl_InitStubs(interp, "8.1", 0) == NULL) {
|
|
return TCL_ERROR;
|
|
}
|
|
#endif
|
|
#ifdef USE_TK_STUBS
|
|
if (Tk_InitStubs(interp, TCL_STUPID "8.1", 0) == NULL) {
|
|
return TCL_ERROR;
|
|
}
|
|
#endif
|
|
|
|
/* Skip all this on Tcl/Tk 8.0 or older. Seems to work */
|
|
#if TCL_MAJOR_VERSION * 100 + TCL_MINOR_VERSION > 800
|
|
Tcl_GetVersion(&major, &minor, &patchLevel, &releaseType);
|
|
|
|
# ifdef HAVE_TK_SETCLASSPROCS
|
|
if (major > 8
|
|
|| (major == 8
|
|
&& (minor > 4
|
|
|| (minor == 4 && (releaseType > 0
|
|
|| patchLevel >= 2))))) {
|
|
# ifdef USE_TK_STUBS
|
|
SetClassProcsPtr = tkStubsPtr->tk_SetClassProcs;
|
|
# else
|
|
SetClassProcsPtr = Tk_SetClassProcs;
|
|
# endif
|
|
} else {
|
|
SetClassProcsPtr = NULL;
|
|
}
|
|
# else
|
|
if (major > 8
|
|
|| (major == 8
|
|
&& (minor > 4
|
|
|| (minor == 4 && (releaseType > 0
|
|
|| patchLevel >= 2))))) {
|
|
TCL_ERR(interp,
|
|
"Sorry, this instance of Togl was not compiled to work with Tcl/Tk 8.4a2 or higher.");
|
|
}
|
|
# endif
|
|
|
|
#endif
|
|
|
|
if (Tcl_PkgProvide(interp, "Togl", TOGL_VERSION) != TCL_OK) {
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
if (Tcl_CreateCommand(interp, "togl", Togl_Cmd,
|
|
(ClientData) Tk_MainWindow(interp), NULL) == NULL)
|
|
return TCL_ERROR;
|
|
|
|
Tcl_InitHashTable(&CommandTable, TCL_STRING_KEYS);
|
|
|
|
return TCL_OK;
|
|
}
|
|
|
|
|
|
/*
|
|
* Register a C function to be called when an Togl widget is realized.
|
|
*/
|
|
void
|
|
Togl_CreateFunc(Togl_Callback *proc)
|
|
{
|
|
DefaultCreateProc = proc;
|
|
}
|
|
|
|
|
|
/*
|
|
* Register a C function to be called when an Togl widget must be redrawn.
|
|
*/
|
|
void
|
|
Togl_DisplayFunc(Togl_Callback *proc)
|
|
{
|
|
DefaultDisplayProc = proc;
|
|
}
|
|
|
|
|
|
/*
|
|
* Register a C function to be called when an Togl widget is resized.
|
|
*/
|
|
void
|
|
Togl_ReshapeFunc(Togl_Callback *proc)
|
|
{
|
|
DefaultReshapeProc = proc;
|
|
}
|
|
|
|
|
|
/*
|
|
* Register a C function to be called when an Togl widget is destroyed.
|
|
*/
|
|
void
|
|
Togl_DestroyFunc(Togl_Callback *proc)
|
|
{
|
|
DefaultDestroyProc = proc;
|
|
}
|
|
|
|
|
|
/*
|
|
* Register a C function to be called from TimerEventHandler.
|
|
*/
|
|
void
|
|
Togl_TimerFunc(Togl_Callback *proc)
|
|
{
|
|
DefaultTimerProc = proc;
|
|
}
|
|
|
|
|
|
/*
|
|
* Reset default callback pointers to NULL.
|
|
*/
|
|
void
|
|
Togl_ResetDefaultCallbacks(void)
|
|
{
|
|
DefaultCreateProc = NULL;
|
|
DefaultDisplayProc = NULL;
|
|
DefaultReshapeProc = NULL;
|
|
DefaultDestroyProc = NULL;
|
|
DefaultOverlayDisplayProc = NULL;
|
|
DefaultTimerProc = NULL;
|
|
DefaultClientData = NULL;
|
|
}
|
|
|
|
|
|
/*
|
|
* Chnage the create callback for a specific Togl widget.
|
|
*/
|
|
void
|
|
Togl_SetCreateFunc(Togl *togl, Togl_Callback *proc)
|
|
{
|
|
togl->CreateProc = proc;
|
|
}
|
|
|
|
|
|
/*
|
|
* Change the display/redraw callback for a specific Togl widget.
|
|
*/
|
|
void
|
|
Togl_SetDisplayFunc(Togl *togl, Togl_Callback *proc)
|
|
{
|
|
togl->DisplayProc = proc;
|
|
}
|
|
|
|
|
|
/*
|
|
* Change the reshape callback for a specific Togl widget.
|
|
*/
|
|
void
|
|
Togl_SetReshapeFunc(Togl *togl, Togl_Callback *proc)
|
|
{
|
|
togl->ReshapeProc = proc;
|
|
}
|
|
|
|
|
|
/*
|
|
* Change the destroy callback for a specific Togl widget.
|
|
*/
|
|
void
|
|
Togl_SetDestroyFunc(Togl *togl, Togl_Callback *proc)
|
|
{
|
|
togl->DestroyProc = proc;
|
|
}
|
|
|
|
|
|
/*
|
|
* Togl_Timer
|
|
*
|
|
* Gets called from Tk_CreateTimerHandler.
|
|
*/
|
|
static void
|
|
Togl_Timer(ClientData clientData)
|
|
{
|
|
Togl *togl = (Togl *) clientData;
|
|
|
|
if (togl->TimerProc) {
|
|
togl->TimerProc(togl);
|
|
|
|
/* Re-register this callback since Tcl/Tk timers are "one-shot". That
|
|
* is, after the timer callback is called it not normally called again.
|
|
* * * * * * * * * That's not the behavior we want for Togl. */
|
|
#if (TK_MAJOR_VERSION * 100 + TK_MINOR_VERSION) >= 401
|
|
togl->timerHandler =
|
|
Tcl_CreateTimerHandler(togl->TimerInterval, Togl_Timer,
|
|
(ClientData) togl);
|
|
#else
|
|
togl->timerHandler =
|
|
Tk_CreateTimerHandler(togl->TimeInterval, Togl_Timer,
|
|
(ClientData) togl);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Change the timer callback for a specific Togl widget.
|
|
* Pass NULL to disable the callback.
|
|
*/
|
|
void
|
|
Togl_SetTimerFunc(Togl *togl, Togl_Callback *proc)
|
|
{
|
|
togl->TimerProc = proc;
|
|
if (proc) {
|
|
#if (TK_MAJOR_VERSION * 100 + TK_MINOR_VERSION) >= 401
|
|
togl->timerHandler =
|
|
Tcl_CreateTimerHandler(togl->TimerInterval, Togl_Timer,
|
|
(ClientData) togl);
|
|
#else
|
|
togl->timerHandler =
|
|
Tk_CreateTimerHandler(togl->TimeInterval, Togl_Timer,
|
|
(ClientData) togl);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Togl_CreateCommand
|
|
*
|
|
* Declares a new C sub-command of Togl callable from Tcl.
|
|
* Every time the sub-command is called from Tcl, the
|
|
* C routine will be called with all the arguments from Tcl.
|
|
*/
|
|
void
|
|
Togl_CreateCommand(char *cmd_name, Togl_CmdProc *cmd_proc)
|
|
{
|
|
int new_item;
|
|
Tcl_HashEntry *entry;
|
|
|
|
entry = Tcl_CreateHashEntry(&CommandTable, cmd_name, &new_item);
|
|
Tcl_SetHashValue(entry, cmd_proc);
|
|
}
|
|
|
|
|
|
/*
|
|
* Togl_MakeCurrent
|
|
*
|
|
* Bind the OpenGL rendering context to the specified
|
|
* Togl widget.
|
|
*/
|
|
void
|
|
Togl_MakeCurrent(const Togl *togl)
|
|
{
|
|
#if defined(TOGL_WGL)
|
|
int res = wglMakeCurrent(togl->tglGLHdc, togl->tglGLHglrc);
|
|
|
|
assert(res == TRUE);
|
|
|
|
#elif defined(TOGL_X11)
|
|
if (!togl->GlCtx)
|
|
return;
|
|
(void) glXMakeCurrent(togl->display,
|
|
togl->TkWin ? Tk_WindowId(togl->TkWin) : None, togl->GlCtx);
|
|
# if defined(__sgi)
|
|
if (togl->OldStereoFlag)
|
|
oldStereoMakeCurrent(togl->display,
|
|
togl->TkWin ? Tk_WindowId(togl->TkWin) : None, togl->GlCtx);
|
|
|
|
# endif /*__sgi STEREO */
|
|
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
if (!togl->aglCtx)
|
|
return;
|
|
aglSetCurrentContext(togl->aglCtx);
|
|
#endif
|
|
}
|
|
|
|
|
|
#ifdef TOGL_AGL_CLASSIC
|
|
/* tell OpenGL which part of the Mac window to render to */
|
|
static void
|
|
SetMacBufRect(Togl *togl)
|
|
{
|
|
GLint wrect[4];
|
|
|
|
/* set wrect[0,1] to lower left corner of widget */
|
|
wrect[2] = ((TkWindow *) (togl->TkWin))->changes.width;
|
|
wrect[3] = ((TkWindow *) (togl->TkWin))->changes.height;
|
|
wrect[0] = ((TkWindow *) (togl->TkWin))->privatePtr->xOff;
|
|
wrect[1] =
|
|
((TkWindow *) (togl->TkWin))->privatePtr->toplevel->portPtr->
|
|
portRect.bottom - wrect[3] -
|
|
((TkWindow *) (togl->TkWin))->privatePtr->yOff;
|
|
aglSetInteger(togl->aglCtx, AGL_BUFFER_RECT, wrect);
|
|
aglEnable(togl->aglCtx, AGL_BUFFER_RECT);
|
|
aglUpdateContext(togl->aglCtx);
|
|
}
|
|
#elif defined(TOGL_AGL)
|
|
/* tell OpenGL which part of the Mac window to render to */
|
|
static void
|
|
SetMacBufRect(Togl *togl)
|
|
{
|
|
GLint wrect[4];
|
|
|
|
/* set wrect[0,1] to lower left corner of widget */
|
|
wrect[2] = Tk_Width(togl->TkWin);
|
|
wrect[3] = Tk_Height(togl->TkWin);
|
|
wrect[0] = ((TkWindow *) (togl->TkWin))->privatePtr->xOff;
|
|
|
|
Rect r;
|
|
|
|
GetPortBounds(((TkWindow *) (togl->TkWin))->privatePtr->toplevel->grafPtr,
|
|
&r);
|
|
|
|
wrect[1] = r.bottom -
|
|
wrect[3] - ((TkWindow *) (togl->TkWin))->privatePtr->yOff;
|
|
|
|
aglSetInteger(togl->aglCtx, AGL_BUFFER_RECT, wrect);
|
|
aglEnable(togl->aglCtx, AGL_BUFFER_RECT);
|
|
aglUpdateContext(togl->aglCtx);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Called when the widget's contents must be redrawn. Basically, we
|
|
* just call the user's render callback function.
|
|
*
|
|
* Note that the parameter type is ClientData so this function can be
|
|
* passed to Tk_DoWhenIdle().
|
|
*/
|
|
static void
|
|
Togl_Render(ClientData clientData)
|
|
{
|
|
Togl *togl = (Togl *) clientData;
|
|
|
|
if (togl->DisplayProc) {
|
|
|
|
#ifdef TOGL_AGL_CLASSIC
|
|
/* Mac is complicated here because OpenGL needs to know what part of
|
|
* the parent window to render into, and it seems that region need to
|
|
* be invalidated before drawing, so that QuickDraw will allow OpenGL
|
|
* to transfer pixels into that part of the window. I'm not even
|
|
* totally sure how or why this works as it does, since this aspect of
|
|
* Mac OpenGL seems to be totally undocumented. This was put together
|
|
* by trial and error! (thiessen) */
|
|
MacRegion r;
|
|
RgnPtr rp = &r;
|
|
GrafPtr curPort, parentWin;
|
|
|
|
parentWin = (GrafPtr)
|
|
(((MacDrawable *) (Tk_WindowId(togl->TkWin)))->toplevel->
|
|
portPtr);
|
|
if (!parentWin)
|
|
return;
|
|
#endif
|
|
|
|
Togl_MakeCurrent(togl);
|
|
|
|
#ifdef TOGL_AGL_CLASSIC
|
|
/* Set QuickDraw port and clipping region */
|
|
GetPort(&curPort);
|
|
SetPort(parentWin);
|
|
r.rgnBBox.left = ((TkWindow *) (togl->TkWin))->privatePtr->xOff;
|
|
r.rgnBBox.right =
|
|
r.rgnBBox.left + ((TkWindow *) (togl->TkWin))->changes.width -
|
|
1;
|
|
r.rgnBBox.top = ((TkWindow *) (togl->TkWin))->privatePtr->yOff;
|
|
r.rgnBBox.bottom =
|
|
r.rgnBBox.top + ((TkWindow *) (togl->TkWin))->changes.height -
|
|
1;
|
|
r.rgnSize = sizeof (Region);
|
|
InvalRgn(&rp);
|
|
SetClip(&rp);
|
|
/* this may seem an odd place to put this, with possibly redundant
|
|
* calls to aglSetInteger(AGL_BUFFER_RECT...), but for some reason
|
|
* performance is actually a lot better if this is called before every
|
|
* render... */
|
|
SetMacBufRect(togl);
|
|
#endif
|
|
|
|
#ifdef TOGL_AGL
|
|
SetMacBufRect(togl);
|
|
#endif
|
|
|
|
togl->DisplayProc(togl);
|
|
|
|
#ifdef TOGL_AGL_CLASSIC
|
|
SetPort(curPort); /* restore previous port */
|
|
#endif
|
|
|
|
}
|
|
#if defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
else {
|
|
/* Always need to update on resize */
|
|
SetMacBufRect(togl);
|
|
}
|
|
#endif
|
|
togl->UpdatePending = False;
|
|
}
|
|
|
|
|
|
static void
|
|
RenderOverlay(ClientData clientData)
|
|
{
|
|
Togl *togl = (Togl *) clientData;
|
|
|
|
if (togl->OverlayFlag && togl->OverlayDisplayProc) {
|
|
|
|
#if defined(TOGL_WGL)
|
|
int res = wglMakeCurrent(togl->tglGLHdc, togl->tglGLHglrc);
|
|
|
|
assert(res == TRUE);
|
|
|
|
#elif defined(TOGL_X11)
|
|
(void) glXMakeCurrent(Tk_Display(togl->TkWin),
|
|
togl->OverlayWindow, togl->OverlayCtx);
|
|
# if defined(__sgi)
|
|
if (togl->OldStereoFlag)
|
|
oldStereoMakeCurrent(Tk_Display(togl->TkWin),
|
|
togl->OverlayWindow, togl->OverlayCtx);
|
|
|
|
# endif /*__sgi STEREO */
|
|
|
|
#endif /* TOGL_WGL */
|
|
|
|
togl->OverlayDisplayProc(togl);
|
|
}
|
|
togl->OverlayUpdatePending = False;
|
|
}
|
|
|
|
|
|
/*
|
|
* It's possible to change with this function or in a script some
|
|
* options like RGBA - ColorIndex ; Z-buffer and so on
|
|
*/
|
|
int
|
|
Togl_Configure(Tcl_Interp *interp, Togl *togl,
|
|
int argc, const char *argv[], int flags)
|
|
{
|
|
Bool oldRgbaFlag = togl->RgbaFlag;
|
|
int oldRgbaRed = togl->RgbaRed;
|
|
int oldRgbaGreen = togl->RgbaGreen;
|
|
int oldRgbaBlue = togl->RgbaBlue;
|
|
Bool oldDoubleFlag = togl->DoubleFlag;
|
|
Bool oldDepthFlag = togl->DepthFlag;
|
|
int oldDepthSize = togl->DepthSize;
|
|
Bool oldAccumFlag = togl->AccumFlag;
|
|
int oldAccumRed = togl->AccumRed;
|
|
int oldAccumGreen = togl->AccumGreen;
|
|
int oldAccumBlue = togl->AccumBlue;
|
|
int oldAccumAlpha = togl->AccumAlpha;
|
|
Bool oldAlphaFlag = togl->AlphaFlag;
|
|
int oldAlphaSize = togl->AlphaSize;
|
|
Bool oldStencilFlag = togl->StencilFlag;
|
|
int oldStencilSize = togl->StencilSize;
|
|
int oldAuxNumber = togl->AuxNumber;
|
|
int oldWidth = togl->Width;
|
|
int oldHeight = togl->Height;
|
|
int oldSetGrid = togl->SetGrid;
|
|
|
|
if (Tk_ConfigureWidget(interp, togl->TkWin, configSpecs,
|
|
argc, argv, WIDGREC togl, flags) == TCL_ERROR) {
|
|
return (TCL_ERROR);
|
|
}
|
|
#ifndef USE_OVERLAY
|
|
if (togl->OverlayFlag) {
|
|
TCL_ERR(interp, "Sorry, overlay was disabled");
|
|
}
|
|
#endif
|
|
|
|
|
|
if (togl->Width != oldWidth || togl->Height != oldHeight
|
|
|| togl->SetGrid != oldSetGrid) {
|
|
Togl_WorldChanged((ClientData) togl);
|
|
/* this added per Lou Arata <arata@enya.picker.com> */
|
|
Tk_ResizeWindow(togl->TkWin, togl->Width, togl->Height);
|
|
|
|
if (togl->ReshapeProc &&
|
|
#if defined(TOGL_WGL)
|
|
togl->tglGLHglrc
|
|
#elif defined(TOGL_X11)
|
|
togl->GlCtx
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
togl->aglCtx
|
|
#endif
|
|
) {
|
|
Togl_MakeCurrent(togl);
|
|
togl->ReshapeProc(togl);
|
|
}
|
|
}
|
|
|
|
if (togl->RgbaFlag != oldRgbaFlag
|
|
|| togl->RgbaRed != oldRgbaRed
|
|
|| togl->RgbaGreen != oldRgbaGreen
|
|
|| togl->RgbaBlue != oldRgbaBlue
|
|
|| togl->DoubleFlag != oldDoubleFlag
|
|
|| togl->DepthFlag != oldDepthFlag
|
|
|| togl->DepthSize != oldDepthSize
|
|
|| togl->AccumFlag != oldAccumFlag
|
|
|| togl->AccumRed != oldAccumRed
|
|
|| togl->AccumGreen != oldAccumGreen
|
|
|| togl->AccumBlue != oldAccumBlue
|
|
|| togl->AccumAlpha != oldAccumAlpha
|
|
|| togl->AlphaFlag != oldAlphaFlag
|
|
|| togl->AlphaSize != oldAlphaSize
|
|
|| togl->StencilFlag != oldStencilFlag
|
|
|| togl->StencilSize != oldStencilSize
|
|
|| togl->AuxNumber != oldAuxNumber) {
|
|
#ifdef MESA_COLOR_HACK
|
|
free_default_color_cells(Tk_Display(togl->TkWin),
|
|
Tk_Colormap(togl->TkWin));
|
|
#endif
|
|
}
|
|
#if defined(__sgi)
|
|
oldStereoInit(togl, togl->OldStereoFlag);
|
|
#endif
|
|
|
|
return TCL_OK;
|
|
}
|
|
|
|
|
|
static int
|
|
Togl_Widget(ClientData clientData, Tcl_Interp *interp, int argc,
|
|
CONST84 char *argv[])
|
|
{
|
|
Togl *togl = (Togl *) clientData;
|
|
int result = TCL_OK;
|
|
Tcl_HashEntry *entry;
|
|
Tcl_HashSearch search;
|
|
Togl_CmdProc *cmd_proc;
|
|
|
|
if (argc < 2) {
|
|
Tcl_AppendResult(interp, "wrong # args: should be \"",
|
|
argv[0], " ?options?\"", NULL);
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
Tk_Preserve((ClientData) togl);
|
|
|
|
if (!strncmp(argv[1], "configure", MAX(1, strlen(argv[1])))) {
|
|
if (argc == 2) {
|
|
/* Return list of all configuration parameters */
|
|
result = Tk_ConfigureInfo(interp, togl->TkWin, configSpecs,
|
|
WIDGREC togl, (char *) NULL, 0);
|
|
} else if (argc == 3) {
|
|
if (strcmp(argv[2], "-extensions") == 0) {
|
|
/* Return a list of OpenGL extensions available */
|
|
const char *extensions;
|
|
|
|
extensions = (const char *) glGetString(GL_EXTENSIONS);
|
|
Tcl_SetResult(interp, TCL_STUPID extensions, TCL_STATIC);
|
|
result = TCL_OK;
|
|
} else {
|
|
/* Return a specific configuration parameter */
|
|
result = Tk_ConfigureInfo(interp, togl->TkWin, configSpecs,
|
|
WIDGREC togl, argv[2], 0);
|
|
}
|
|
} else {
|
|
/* Execute a configuration change */
|
|
result = Togl_Configure(interp, togl, argc - 2, argv + 2,
|
|
TK_CONFIG_ARGV_ONLY);
|
|
}
|
|
} else if (!strncmp(argv[1], "render", MAX(1, strlen(argv[1])))) {
|
|
/* force the widget to be redrawn */
|
|
Togl_Render((ClientData) togl);
|
|
} else if (!strncmp(argv[1], "swapbuffers", MAX(1, strlen(argv[1])))) {
|
|
/* force the widget to be redrawn */
|
|
Togl_SwapBuffers(togl);
|
|
} else if (!strncmp(argv[1], "makecurrent", MAX(1, strlen(argv[1])))) {
|
|
/* force the widget to be redrawn */
|
|
Togl_MakeCurrent(togl);
|
|
}
|
|
#if TOGL_USE_FONTS == 1
|
|
else if (!strncmp(argv[1], "loadbitmapfont", MAX(1, strlen(argv[1])))) {
|
|
if (argc == 3) {
|
|
GLuint fontbase;
|
|
Tcl_Obj *fontbaseAsTclObject;
|
|
|
|
fontbase = Togl_LoadBitmapFont(togl, argv[2]);
|
|
if (fontbase) {
|
|
fontbaseAsTclObject = Tcl_NewIntObj(fontbase);
|
|
Tcl_SetObjResult(interp, fontbaseAsTclObject);
|
|
result = TCL_OK;
|
|
} else {
|
|
Tcl_AppendResult(interp, "Could not allocate font", NULL);
|
|
result = TCL_ERROR;
|
|
}
|
|
} else {
|
|
Tcl_AppendResult(interp, "wrong # args", NULL);
|
|
result = TCL_ERROR;
|
|
}
|
|
} else if (!strncmp(argv[1], "unloadbitmapfont", MAX(1, strlen(argv[1])))) {
|
|
if (argc == 3) {
|
|
Togl_UnloadBitmapFont(togl, atoi(argv[2]));
|
|
result = TCL_OK;
|
|
} else {
|
|
Tcl_AppendResult(interp, "wrong # args", NULL);
|
|
result = TCL_ERROR;
|
|
}
|
|
}
|
|
#endif /* TOGL_USE_FONTS */
|
|
else {
|
|
/* Probably a user-defined function */
|
|
entry = Tcl_FindHashEntry(&CommandTable, argv[1]);
|
|
if (entry != NULL) {
|
|
cmd_proc = (Togl_CmdProc *) Tcl_GetHashValue(entry);
|
|
result = cmd_proc(togl, argc, argv);
|
|
} else {
|
|
Tcl_AppendResult(interp, "Togl: Unknown option: ", argv[1], "\n",
|
|
"Try: configure or render\n",
|
|
"or one of the user-defined commands:\n", NULL);
|
|
entry = Tcl_FirstHashEntry(&CommandTable, &search);
|
|
while (entry) {
|
|
Tcl_AppendResult(interp, " ",
|
|
Tcl_GetHashKey(&CommandTable, entry), "\n", NULL);
|
|
entry = Tcl_NextHashEntry(&search);
|
|
}
|
|
result = TCL_ERROR;
|
|
}
|
|
}
|
|
|
|
Tk_Release((ClientData) togl);
|
|
return result;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Togl_Cmd
|
|
*
|
|
* Called when Togl is executed - creation of a Togl widget.
|
|
* * Creates a new window
|
|
* * Creates an 'Togl' data structure
|
|
* * Creates an event handler for this window
|
|
* * Creates a command that handles this object
|
|
* * Configures this Togl for the given arguments
|
|
*/
|
|
static int
|
|
Togl_Cmd(ClientData clientData, Tcl_Interp *interp, int argc,
|
|
CONST84 char **argv)
|
|
{
|
|
const char *name;
|
|
Tk_Window mainwin = (Tk_Window) clientData;
|
|
Tk_Window tkwin;
|
|
Togl *togl;
|
|
|
|
if (argc <= 1) {
|
|
TCL_ERR(interp, "wrong # args: should be \"pathName read filename\"");
|
|
}
|
|
|
|
/* Create the window. */
|
|
name = argv[1];
|
|
tkwin = Tk_CreateWindowFromPath(interp, mainwin, name, (char *) NULL);
|
|
if (tkwin == NULL) {
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
Tk_SetClass(tkwin, "Togl");
|
|
|
|
/* Create Togl data structure */
|
|
togl = (Togl *) malloc(sizeof (Togl));
|
|
if (!togl) {
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
togl->Next = NULL;
|
|
#if defined(TOGL_WGL)
|
|
togl->tglGLHdc = NULL;
|
|
togl->tglGLHglrc = NULL;
|
|
#elif defined(TOGL_X11)
|
|
togl->GlCtx = NULL;
|
|
togl->OverlayCtx = NULL;
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
togl->aglCtx = NULL;
|
|
#endif /* TOGL_WGL */
|
|
togl->display = Tk_Display(tkwin);
|
|
togl->TkWin = tkwin;
|
|
togl->Interp = interp;
|
|
#ifndef NO_TK_CURSOR
|
|
togl->Cursor = None;
|
|
#endif
|
|
togl->Width = 0;
|
|
togl->Height = 0;
|
|
togl->SetGrid = 0;
|
|
togl->TimerInterval = 0;
|
|
togl->RgbaFlag = True;
|
|
togl->RgbaRed = 1;
|
|
togl->RgbaGreen = 1;
|
|
togl->RgbaBlue = 1;
|
|
togl->DoubleFlag = False;
|
|
togl->DepthFlag = False;
|
|
togl->DepthSize = 1;
|
|
togl->AccumFlag = False;
|
|
togl->AccumRed = 1;
|
|
togl->AccumGreen = 1;
|
|
togl->AccumBlue = 1;
|
|
togl->AccumAlpha = 1;
|
|
togl->AlphaFlag = False;
|
|
togl->AlphaSize = 1;
|
|
togl->StencilFlag = False;
|
|
togl->StencilSize = 1;
|
|
togl->OverlayFlag = False;
|
|
togl->StereoFlag = False;
|
|
#ifdef __sgi
|
|
togl->OldStereoFlag = False;
|
|
#endif
|
|
togl->AuxNumber = 0;
|
|
togl->Indirect = False;
|
|
togl->PixelFormat = 0;
|
|
togl->UpdatePending = False;
|
|
togl->OverlayUpdatePending = False;
|
|
togl->CreateProc = DefaultCreateProc;
|
|
togl->DisplayProc = DefaultDisplayProc;
|
|
togl->ReshapeProc = DefaultReshapeProc;
|
|
togl->DestroyProc = DefaultDestroyProc;
|
|
togl->TimerProc = DefaultTimerProc;
|
|
togl->OverlayDisplayProc = DefaultOverlayDisplayProc;
|
|
togl->ShareList = NULL;
|
|
togl->ShareContext = NULL;
|
|
togl->Ident = NULL;
|
|
togl->Client_Data = DefaultClientData;
|
|
|
|
/* for EPS Output */
|
|
togl->EpsRedMap = togl->EpsGreenMap = togl->EpsBlueMap = NULL;
|
|
togl->EpsMapSize = 0;
|
|
|
|
/* Create command event handler */
|
|
togl->widgetCmd = Tcl_CreateCommand(interp, Tk_PathName(tkwin),
|
|
Togl_Widget, (ClientData) togl,
|
|
(Tcl_CmdDeleteProc *) ToglCmdDeletedProc);
|
|
/*
|
|
* Setup the Tk_ClassProcs callbacks to point at our own window creation
|
|
* function
|
|
*
|
|
* We need to check at runtime if we should use the new Tk_SetClassProcs()
|
|
* API or if we need to modify the window structure directly */
|
|
|
|
|
|
#ifdef HAVE_TK_SETCLASSPROCS
|
|
|
|
if (SetClassProcsPtr != NULL) { /* use public API (Tk 8.4+) */
|
|
Tk_ClassProcs *procsPtr;
|
|
|
|
procsPtr = (Tk_ClassProcs *) Tcl_Alloc(sizeof (Tk_ClassProcs));
|
|
procsPtr->size = sizeof (Tk_ClassProcs);
|
|
procsPtr->createProc = Togl_CreateWindow;
|
|
procsPtr->worldChangedProc = Togl_WorldChanged;
|
|
procsPtr->modalProc = NULL;
|
|
/* Tk_SetClassProcs(togl->TkWin,procsPtr,(ClientData)togl); */
|
|
(SetClassProcsPtr) (togl->TkWin, procsPtr, (ClientData) togl);
|
|
} else
|
|
#endif
|
|
{ /* use private API */
|
|
/*
|
|
* We need to set these fields in the Tk_FakeWin structure: dummy17 =
|
|
* classProcsPtr dummy18 = instanceData */
|
|
TkClassProcs *procsPtr;
|
|
Tk_FakeWin *winPtr = (Tk_FakeWin *) (togl->TkWin);
|
|
|
|
procsPtr = (TkClassProcs *) Tcl_Alloc(sizeof (TkClassProcs));
|
|
procsPtr->createProc = Togl_CreateWindow;
|
|
procsPtr->geometryProc = Togl_WorldChanged;
|
|
procsPtr->modalProc = NULL;
|
|
winPtr->dummy17 = (char *) procsPtr;
|
|
winPtr->dummy18 = (ClientData) togl;
|
|
}
|
|
|
|
Tk_CreateEventHandler(tkwin,
|
|
ExposureMask | StructureNotifyMask, Togl_EventProc,
|
|
(ClientData) togl);
|
|
|
|
/* Configure Togl widget */
|
|
if (Togl_Configure(interp, togl, argc - 2, argv + 2, 0) == TCL_ERROR) {
|
|
Tk_DestroyWindow(tkwin);
|
|
Tcl_AppendResult(interp, "Couldn't configure togl widget\n", NULL);
|
|
goto error;
|
|
}
|
|
|
|
/*
|
|
* If OpenGL window wasn't already created by Togl_Configure() we
|
|
* create it now. We can tell by checking if the GLX context has
|
|
* been initialized.
|
|
*/
|
|
if (!
|
|
#if defined(TOGL_WGL)
|
|
togl->tglGLHdc
|
|
#elif defined(TOGL_X11)
|
|
togl->GlCtx
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
togl->aglCtx
|
|
#endif
|
|
) {
|
|
Tk_MakeWindowExist(togl->TkWin);
|
|
if (Tk_WindowId(togl->TkWin) == DUMMY_WINDOW) {
|
|
return TCL_ERROR;
|
|
}
|
|
Togl_MakeCurrent(togl);
|
|
}
|
|
|
|
/* If defined, call create callback */
|
|
if (togl->CreateProc) {
|
|
togl->CreateProc(togl);
|
|
}
|
|
|
|
/* If defined, call reshape proc */
|
|
if (togl->ReshapeProc) {
|
|
togl->ReshapeProc(togl);
|
|
}
|
|
|
|
/* If defined, setup timer */
|
|
if (togl->TimerProc) {
|
|
(void) Tk_CreateTimerHandler(togl->TimerInterval, Togl_Timer,
|
|
(ClientData) togl);
|
|
}
|
|
|
|
Tcl_AppendResult(interp, Tk_PathName(tkwin), NULL);
|
|
|
|
/* Add to linked list */
|
|
AddToList(togl);
|
|
|
|
return TCL_OK;
|
|
|
|
error:
|
|
(void) Tcl_DeleteCommand(interp, "togl");
|
|
/* free(togl); Don't free it, if we do a crash occurs later... */
|
|
return TCL_ERROR;
|
|
}
|
|
|
|
|
|
#ifdef USE_OVERLAY
|
|
|
|
/*
|
|
* Do all the setup for overlay planes
|
|
* Return: TCL_OK or TCL_ERROR
|
|
*/
|
|
static int
|
|
SetupOverlay(Togl *togl)
|
|
{
|
|
# if defined(TOGL_X11)
|
|
|
|
# ifdef GLX_TRANSPARENT_TYPE_EXT
|
|
static int ovAttributeList[] = {
|
|
GLX_BUFFER_SIZE, 2,
|
|
GLX_LEVEL, 1,
|
|
GLX_TRANSPARENT_TYPE_EXT, GLX_TRANSPARENT_INDEX_EXT,
|
|
None
|
|
};
|
|
# else
|
|
static int ovAttributeList[] = {
|
|
GLX_BUFFER_SIZE, 2,
|
|
GLX_LEVEL, 1,
|
|
None
|
|
};
|
|
# endif
|
|
|
|
Display *dpy;
|
|
XVisualInfo *visinfo;
|
|
TkWindow *winPtr = (TkWindow *) togl->TkWin;
|
|
|
|
XSetWindowAttributes swa;
|
|
Tcl_HashEntry *hPtr;
|
|
int new_flag;
|
|
|
|
dpy = Tk_Display(togl->TkWin);
|
|
|
|
visinfo = glXChooseVisual(dpy, Tk_ScreenNumber(winPtr), ovAttributeList);
|
|
if (!visinfo) {
|
|
Tcl_AppendResult(togl->Interp, Tk_PathName(winPtr),
|
|
": No suitable overlay index visual available", (char *) NULL);
|
|
togl->OverlayCtx = 0;
|
|
togl->OverlayWindow = 0;
|
|
togl->OverlayCmap = 0;
|
|
return TCL_ERROR;
|
|
}
|
|
# ifdef GLX_TRANSPARENT_INDEX_EXT
|
|
{
|
|
int fail =
|
|
glXGetConfig(dpy, visinfo, GLX_TRANSPARENT_INDEX_VALUE_EXT,
|
|
&togl->OverlayTransparentPixel);
|
|
|
|
if (fail)
|
|
togl->OverlayTransparentPixel = 0; /* maybe, maybe ... */
|
|
}
|
|
# else
|
|
togl->OverlayTransparentPixel = 0; /* maybe, maybe ... */
|
|
# endif
|
|
|
|
/* share display lists with normal layer context */
|
|
togl->OverlayCtx =
|
|
glXCreateContext(dpy, visinfo, togl->GlCtx, !togl->Indirect);
|
|
|
|
swa.colormap = XCreateColormap(dpy, XRootWindow(dpy, visinfo->screen),
|
|
visinfo->visual, AllocNone);
|
|
togl->OverlayCmap = swa.colormap;
|
|
|
|
swa.border_pixel = 0;
|
|
swa.event_mask = ALL_EVENTS_MASK;
|
|
togl->OverlayWindow = XCreateWindow(dpy, Tk_WindowId(togl->TkWin), 0, 0,
|
|
togl->Width, togl->Height, 0,
|
|
visinfo->depth, InputOutput,
|
|
visinfo->visual, CWBorderPixel | CWColormap | CWEventMask, &swa);
|
|
|
|
hPtr = Tcl_CreateHashEntry(&winPtr->dispPtr->winTable,
|
|
(char *) togl->OverlayWindow, &new_flag);
|
|
Tcl_SetHashValue(hPtr, winPtr);
|
|
|
|
/* XMapWindow( dpy, togl->OverlayWindow ); */
|
|
togl->OverlayIsMapped = False;
|
|
|
|
/* Make sure window manager installs our colormap */
|
|
XSetWMColormapWindows(dpy, togl->OverlayWindow, &togl->OverlayWindow, 1);
|
|
|
|
return TCL_OK;
|
|
|
|
# elif defined(TOGL_WGL) || defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
/* not yet implemented on these */
|
|
return TCL_ERROR;
|
|
# endif
|
|
}
|
|
|
|
#endif /* USE_OVERLAY */
|
|
|
|
|
|
|
|
#ifdef TOGL_WGL
|
|
# define TOGL_CLASS_NAME "Togl Class"
|
|
static Bool ToglClassInitialized = False;
|
|
|
|
static LRESULT CALLBACK
|
|
Win32WinProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LONG result;
|
|
Togl *togl = (Togl *) GetWindowLong(hwnd, 0);
|
|
WNDCLASS childClass;
|
|
|
|
switch (message) {
|
|
case WM_WINDOWPOSCHANGED:
|
|
/* Should be processed by DefWindowProc, otherwise a double buffered
|
|
* context is not properly resized when the corresponding window is
|
|
* resized. */
|
|
break;
|
|
case WM_DESTROY:
|
|
if (togl->tglGLHglrc) {
|
|
wglDeleteContext(togl->tglGLHglrc);
|
|
}
|
|
if (togl->tglGLHdc) {
|
|
ReleaseDC(hwnd, togl->tglGLHdc);
|
|
}
|
|
free(togl);
|
|
break;
|
|
default:
|
|
# if USE_STATIC_LIB
|
|
return TkWinChildProc(hwnd, message, wParam, lParam);
|
|
# else
|
|
/*
|
|
* OK, since TkWinChildProc is not explicitly exported in the
|
|
* dynamic libraries, we have to retrieve it from the class info
|
|
* registered with windows.
|
|
*
|
|
*/
|
|
if (tkWinChildProc == NULL) {
|
|
GetClassInfo(Tk_GetHINSTANCE(), TK_WIN_CHILD_CLASS_NAME,
|
|
&childClass);
|
|
tkWinChildProc = childClass.lpfnWndProc;
|
|
}
|
|
return tkWinChildProc(hwnd, message, wParam, lParam);
|
|
# endif
|
|
}
|
|
result = DefWindowProc(hwnd, message, wParam, lParam);
|
|
Tcl_ServiceAll();
|
|
return result;
|
|
}
|
|
#endif /* TOGL_WGL */
|
|
|
|
|
|
|
|
/*
|
|
* Togl_CreateWindow
|
|
*
|
|
* Window creation function, invoked as a callback from Tk_MakeWindowExist.
|
|
* Creates an OpenGL window for the Togl widget.
|
|
*/
|
|
static Window
|
|
Togl_CreateWindow(Tk_Window tkwin, Window parent, ClientData instanceData)
|
|
{
|
|
|
|
Togl *togl = (Togl *) instanceData;
|
|
XVisualInfo *visinfo = NULL;
|
|
Display *dpy;
|
|
Colormap cmap;
|
|
int scrnum;
|
|
Window window;
|
|
|
|
#if defined(TOGL_X11)
|
|
Bool directCtx = True;
|
|
int attrib_list[1000];
|
|
int attrib_count;
|
|
int dummy;
|
|
XSetWindowAttributes swa;
|
|
|
|
# define MAX_ATTEMPTS 12
|
|
static int ci_depths[MAX_ATTEMPTS] = {
|
|
8, 4, 2, 1, 12, 16, 8, 4, 2, 1, 12, 16
|
|
};
|
|
static int dbl_flags[MAX_ATTEMPTS] = {
|
|
0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1
|
|
};
|
|
#elif defined(TOGL_WGL)
|
|
HWND hwnd, parentWin;
|
|
int pixelformat;
|
|
HANDLE hInstance;
|
|
WNDCLASS ToglClass;
|
|
PIXELFORMATDESCRIPTOR pfd;
|
|
XVisualInfo VisInf;
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
GLint attribs[20];
|
|
int na;
|
|
AGLPixelFormat fmt;
|
|
XVisualInfo VisInf;
|
|
#endif /* TOGL_X11 */
|
|
|
|
|
|
dpy = Tk_Display(togl->TkWin);
|
|
|
|
#if defined(TOGL_X11)
|
|
/* Make sure OpenGL's GLX extension supported */
|
|
if (!glXQueryExtension(dpy, &dummy, &dummy)) {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "Togl: X server has no OpenGL GLX extension",
|
|
TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
|
|
if (togl->ShareContext && FindTogl(togl->ShareContext)) {
|
|
/* share OpenGL context with existing Togl widget */
|
|
Togl *shareWith = FindTogl(togl->ShareContext);
|
|
|
|
assert(shareWith != NULL);
|
|
assert(shareWith->GlCtx != NULL);
|
|
togl->GlCtx = shareWith->GlCtx;
|
|
togl->VisInfo = shareWith->VisInfo;
|
|
visinfo = togl->VisInfo;
|
|
} else {
|
|
if (togl->PixelFormat) {
|
|
XVisualInfo template;
|
|
int count = 1;
|
|
|
|
template.visualid = togl->PixelFormat;
|
|
visinfo = XGetVisualInfo(dpy, VisualIDMask, &template, &count);
|
|
if (visinfo == NULL) {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "Togl: couldn't choose pixel format",
|
|
TCL_STATIC);
|
|
|
|
return DUMMY_WINDOW;
|
|
}
|
|
/* fill in flags normally passed in that affect behavior */
|
|
(void) glXGetConfig(dpy, visinfo, GLX_RGBA, &togl->RgbaFlag);
|
|
(void) glXGetConfig(dpy, visinfo, GLX_DOUBLEBUFFER,
|
|
&togl->DoubleFlag);
|
|
(void) glXGetConfig(dpy, visinfo, GLX_STEREO, &togl->StereoFlag);
|
|
} else {
|
|
int attempt;
|
|
|
|
/* It may take a few tries to get a visual */
|
|
for (attempt = 0; attempt < MAX_ATTEMPTS; attempt++) {
|
|
attrib_count = 0;
|
|
attrib_list[attrib_count++] = GLX_USE_GL;
|
|
if (togl->RgbaFlag) {
|
|
/* RGB[A] mode */
|
|
attrib_list[attrib_count++] = GLX_RGBA;
|
|
attrib_list[attrib_count++] = GLX_RED_SIZE;
|
|
attrib_list[attrib_count++] = togl->RgbaRed;
|
|
attrib_list[attrib_count++] = GLX_GREEN_SIZE;
|
|
attrib_list[attrib_count++] = togl->RgbaGreen;
|
|
attrib_list[attrib_count++] = GLX_BLUE_SIZE;
|
|
attrib_list[attrib_count++] = togl->RgbaBlue;
|
|
if (togl->AlphaFlag) {
|
|
attrib_list[attrib_count++] = GLX_ALPHA_SIZE;
|
|
attrib_list[attrib_count++] = togl->AlphaSize;
|
|
}
|
|
|
|
/* for EPS Output */
|
|
if (togl->EpsRedMap)
|
|
free(togl->EpsRedMap);
|
|
if (togl->EpsGreenMap)
|
|
free(togl->EpsGreenMap);
|
|
if (togl->EpsBlueMap)
|
|
free(togl->EpsBlueMap);
|
|
togl->EpsRedMap = togl->EpsGreenMap = togl->EpsBlueMap =
|
|
NULL;
|
|
togl->EpsMapSize = 0;
|
|
} else {
|
|
/* Color index mode */
|
|
int depth;
|
|
|
|
attrib_list[attrib_count++] = GLX_BUFFER_SIZE;
|
|
depth = ci_depths[attempt];
|
|
attrib_list[attrib_count++] = depth;
|
|
}
|
|
if (togl->DepthFlag) {
|
|
attrib_list[attrib_count++] = GLX_DEPTH_SIZE;
|
|
attrib_list[attrib_count++] = togl->DepthSize;
|
|
}
|
|
if (togl->DoubleFlag || dbl_flags[attempt]) {
|
|
attrib_list[attrib_count++] = GLX_DOUBLEBUFFER;
|
|
}
|
|
if (togl->StencilFlag) {
|
|
attrib_list[attrib_count++] = GLX_STENCIL_SIZE;
|
|
attrib_list[attrib_count++] = togl->StencilSize;
|
|
}
|
|
if (togl->AccumFlag) {
|
|
attrib_list[attrib_count++] = GLX_ACCUM_RED_SIZE;
|
|
attrib_list[attrib_count++] = togl->AccumRed;
|
|
attrib_list[attrib_count++] = GLX_ACCUM_GREEN_SIZE;
|
|
attrib_list[attrib_count++] = togl->AccumGreen;
|
|
attrib_list[attrib_count++] = GLX_ACCUM_BLUE_SIZE;
|
|
attrib_list[attrib_count++] = togl->AccumBlue;
|
|
if (togl->AlphaFlag) {
|
|
attrib_list[attrib_count++] = GLX_ACCUM_ALPHA_SIZE;
|
|
attrib_list[attrib_count++] = togl->AccumAlpha;
|
|
}
|
|
}
|
|
if (togl->AuxNumber != 0) {
|
|
attrib_list[attrib_count++] = GLX_AUX_BUFFERS;
|
|
attrib_list[attrib_count++] = togl->AuxNumber;
|
|
}
|
|
if (togl->Indirect) {
|
|
directCtx = False;
|
|
}
|
|
|
|
if (togl->StereoFlag) {
|
|
attrib_list[attrib_count++] = GLX_STEREO;
|
|
}
|
|
attrib_list[attrib_count++] = None;
|
|
|
|
visinfo = glXChooseVisual(dpy, Tk_ScreenNumber(togl->TkWin),
|
|
attrib_list);
|
|
if (visinfo) {
|
|
/* found a GLX visual! */
|
|
break;
|
|
}
|
|
}
|
|
|
|
togl->VisInfo = visinfo;
|
|
|
|
if (visinfo == NULL) {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "Togl: couldn't get visual", TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
|
|
/*
|
|
* Create a new OpenGL rendering context.
|
|
*/
|
|
if (togl->ShareList) {
|
|
/* share display lists with existing togl widget */
|
|
Togl *shareWith = FindTogl(togl->ShareList);
|
|
GLXContext shareCtx;
|
|
|
|
if (shareWith)
|
|
shareCtx = shareWith->GlCtx;
|
|
else
|
|
shareCtx = None;
|
|
togl->GlCtx =
|
|
glXCreateContext(dpy, visinfo, shareCtx, directCtx);
|
|
} else {
|
|
/* don't share display lists */
|
|
togl->GlCtx = glXCreateContext(dpy, visinfo, None, directCtx);
|
|
}
|
|
|
|
if (togl->GlCtx == NULL) {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "could not create rendering context",
|
|
TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
|
|
#endif /* TOGL_X11 */
|
|
|
|
#ifdef TOGL_WGL
|
|
parentWin = Tk_GetHWND(parent);
|
|
hInstance = Tk_GetHINSTANCE();
|
|
if (!ToglClassInitialized) {
|
|
ToglClassInitialized = True;
|
|
ToglClass.style = CS_HREDRAW | CS_VREDRAW;
|
|
ToglClass.cbClsExtra = 0;
|
|
ToglClass.cbWndExtra = 4; /* to save struct Togl* */
|
|
ToglClass.hInstance = hInstance;
|
|
ToglClass.hbrBackground = NULL;
|
|
ToglClass.lpszMenuName = NULL;
|
|
ToglClass.lpszClassName = TOGL_CLASS_NAME;
|
|
ToglClass.lpfnWndProc = Win32WinProc;
|
|
ToglClass.hIcon = NULL;
|
|
ToglClass.hCursor = NULL;
|
|
if (!RegisterClass(&ToglClass)) {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "unable register Togl window class", TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
}
|
|
|
|
hwnd = CreateWindow(TOGL_CLASS_NAME, NULL,
|
|
WS_CHILD | WS_CLIPCHILDREN | WS_CLIPSIBLINGS, 0, 0,
|
|
togl->Width, togl->Height, parentWin, NULL, hInstance, NULL);
|
|
SetWindowLong(hwnd, 0, (LONG) togl);
|
|
SetWindowPos(hwnd, HWND_TOP, 0, 0, 0, 0,
|
|
SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOSIZE);
|
|
|
|
togl->tglGLHdc = GetDC(hwnd);
|
|
|
|
pfd.nSize = sizeof (PIXELFORMATDESCRIPTOR);
|
|
pfd.nVersion = 1;
|
|
pfd.dwFlags = PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL;
|
|
if (togl->DoubleFlag) {
|
|
pfd.dwFlags |= PFD_DOUBLEBUFFER;
|
|
}
|
|
/* The stereo flag is not supported in the current generic OpenGL
|
|
* implementation, but may be supported by specific hardware devices. */
|
|
if (togl->StereoFlag) {
|
|
pfd.dwFlags |= PFD_STEREO;
|
|
}
|
|
|
|
if (togl->PixelFormat) {
|
|
pixelformat = togl->PixelFormat;
|
|
} else {
|
|
pfd.cColorBits = togl->RgbaRed + togl->RgbaGreen + togl->RgbaBlue;
|
|
pfd.iPixelType = togl->RgbaFlag ? PFD_TYPE_RGBA : PFD_TYPE_COLORINDEX;
|
|
/* Alpha bitplanes are not supported in the current generic OpenGL
|
|
* implementation, but may be supported by specific hardware devices. */
|
|
pfd.cAlphaBits = togl->AlphaFlag ? togl->AlphaSize : 0;
|
|
pfd.cAccumBits = togl->AccumFlag ? (togl->AccumRed + togl->AccumGreen +
|
|
togl->AccumBlue + togl->AccumAlpha) : 0;
|
|
pfd.cDepthBits = togl->DepthFlag ? togl->DepthSize : 0;
|
|
pfd.cStencilBits = togl->StencilFlag ? togl->StencilSize : 0;
|
|
/* Auxiliary buffers are not supported in the current generic OpenGL
|
|
* implementation, but may be supported by specific hardware devices. */
|
|
pfd.cAuxBuffers = togl->AuxNumber;
|
|
pfd.iLayerType = PFD_MAIN_PLANE;
|
|
|
|
if ((pixelformat = ChoosePixelFormat(togl->tglGLHdc, &pfd)) == 0) {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "Togl: couldn't choose pixel format",
|
|
TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
}
|
|
if (SetPixelFormat(togl->tglGLHdc, pixelformat, &pfd) == FALSE) {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "Togl: couldn't choose pixel format", TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
|
|
/* Get the actual pixel format */
|
|
DescribePixelFormat(togl->tglGLHdc, pixelformat, sizeof (pfd), &pfd);
|
|
if (togl->PixelFormat) {
|
|
/* fill in flags normally passed in that affect behavior */
|
|
togl->RgbaFlag = pfd.iPixelType == PFD_TYPE_RGBA;
|
|
togl->DoubleFlag = pfd.cDepthBits > 0;
|
|
togl->StereoFlag = (pfd.dwFlags & PFD_STEREO) != 0;
|
|
// TODO: set depth flag, and more
|
|
} else if (togl->StereoFlag && (pfd.dwFlags & PFD_STEREO) == 0) {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "Togl: couldn't choose stereo pixel format",
|
|
TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
|
|
if (togl->ShareContext && FindTogl(togl->ShareContext)) {
|
|
/* share OpenGL context with existing Togl widget */
|
|
Togl *shareWith = FindTogl(togl->ShareContext);
|
|
|
|
assert(shareWith);
|
|
assert(shareWith->tglGLHglrc);
|
|
togl->tglGLHglrc = shareWith->tglGLHglrc;
|
|
togl->VisInfo = shareWith->VisInfo;
|
|
visinfo = togl->VisInfo;
|
|
} else {
|
|
/*
|
|
* Create a new OpenGL rendering context. And check to share lists.
|
|
*/
|
|
togl->tglGLHglrc = wglCreateContext(togl->tglGLHdc);
|
|
|
|
if (togl->ShareList) {
|
|
/* share display lists with existing togl widget */
|
|
Togl *shareWith = FindTogl(togl->ShareList);
|
|
|
|
if (shareWith)
|
|
wglShareLists(shareWith->tglGLHglrc, togl->tglGLHglrc);
|
|
}
|
|
|
|
if (!togl->tglGLHglrc) {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "could not create rendering context",
|
|
TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
|
|
/* Just for portability, define the simplest visinfo */
|
|
visinfo = &VisInf;
|
|
visinfo->visual = DefaultVisual(dpy, DefaultScreen(dpy));
|
|
visinfo->depth = visinfo->visual->bits_per_rgb;
|
|
togl->VisInfo = visinfo;
|
|
}
|
|
|
|
#endif /* TOGL_WGL */
|
|
|
|
|
|
/*
|
|
* find a colormap
|
|
*/
|
|
scrnum = Tk_ScreenNumber(togl->TkWin);
|
|
if (togl->RgbaFlag) {
|
|
/* Colormap for RGB mode */
|
|
#if defined(TOGL_X11)
|
|
cmap = get_rgb_colormap(dpy, scrnum, visinfo, togl->TkWin);
|
|
|
|
#elif defined(TOGL_WGL)
|
|
if (pfd.dwFlags & PFD_NEED_PALETTE) {
|
|
cmap = Win32CreateRgbColormap(pfd);
|
|
} else {
|
|
cmap = DefaultColormap(dpy, scrnum);
|
|
}
|
|
/* for EPS Output */
|
|
if (togl->EpsRedMap)
|
|
free(togl->EpsRedMap);
|
|
if (togl->EpsGreenMap)
|
|
free(togl->EpsGreenMap);
|
|
if (togl->EpsBlueMap)
|
|
free(togl->EpsBlueMap);
|
|
togl->EpsRedMap = togl->EpsGreenMap = togl->EpsBlueMap = NULL;
|
|
togl->EpsMapSize = 0;
|
|
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
cmap = DefaultColormap(dpy, scrnum);
|
|
/* for EPS Output */
|
|
if (togl->EpsRedMap)
|
|
free(togl->EpsRedMap);
|
|
if (togl->EpsGreenMap)
|
|
free(togl->EpsGreenMap);
|
|
if (togl->EpsBlueMap)
|
|
free(togl->EpsBlueMap);
|
|
togl->EpsRedMap = togl->EpsGreenMap = togl->EpsBlueMap = NULL;
|
|
togl->EpsMapSize = 0;
|
|
#endif /* TOGL_X11 */
|
|
} else {
|
|
/* Colormap for CI mode */
|
|
#ifdef TOGL_WGL
|
|
togl->CiColormapSize = 1 << pfd.cColorBits;
|
|
togl->CiColormapSize = togl->CiColormapSize < MAX_CI_COLORMAP_SIZE ?
|
|
togl->CiColormapSize : MAX_CI_COLORMAP_SIZE;
|
|
|
|
#endif /* TOGL_WGL */
|
|
if (togl->PrivateCmapFlag) {
|
|
/* need read/write colormap so user can store own color entries */
|
|
#if defined(TOGL_X11)
|
|
cmap = XCreateColormap(dpy, XRootWindow(dpy, visinfo->screen),
|
|
visinfo->visual, AllocAll);
|
|
#elif defined(TOGL_WGL)
|
|
cmap = Win32CreateCiColormap(togl);
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
/* need to figure out how to do this correctly on Mac... */
|
|
cmap = DefaultColormap(dpy, scrnum);
|
|
#endif /* TOGL_X11 */
|
|
} else {
|
|
if (visinfo->visual == DefaultVisual(dpy, scrnum)) {
|
|
/* share default/root colormap */
|
|
cmap = Tk_Colormap(togl->TkWin);
|
|
} else {
|
|
/* make a new read-only colormap */
|
|
cmap = XCreateColormap(dpy, XRootWindow(dpy, visinfo->screen),
|
|
visinfo->visual, AllocNone);
|
|
}
|
|
}
|
|
}
|
|
|
|
#if !defined(TOGL_AGL)
|
|
/* Make sure Tk knows to switch to the new colormap when the cursor is over
|
|
* this window when running in color index mode. */
|
|
(void) Tk_SetWindowVisual(togl->TkWin, visinfo->visual, visinfo->depth,
|
|
cmap);
|
|
#endif
|
|
|
|
#ifdef TOGL_WGL
|
|
/* Install the colormap */
|
|
SelectPalette(togl->tglGLHdc, ((TkWinColormap *) cmap)->palette, TRUE);
|
|
RealizePalette(togl->tglGLHdc);
|
|
#endif /* TOGL_WGL */
|
|
|
|
#if defined(TOGL_X11)
|
|
swa.colormap = cmap;
|
|
swa.border_pixel = 0;
|
|
swa.event_mask = ALL_EVENTS_MASK;
|
|
window = XCreateWindow(dpy, parent,
|
|
0, 0, togl->Width, togl->Height,
|
|
0, visinfo->depth,
|
|
InputOutput, visinfo->visual,
|
|
CWBorderPixel | CWColormap | CWEventMask, &swa);
|
|
/* Make sure window manager installs our colormap */
|
|
(void) XSetWMColormapWindows(dpy, window, &window, 1);
|
|
|
|
#elif defined(TOGL_WGL)
|
|
window = Tk_AttachHWND(togl->TkWin, hwnd);
|
|
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
{
|
|
TkWindow *winPtr = (TkWindow *) togl->TkWin;
|
|
|
|
window = TkpMakeWindow(winPtr, parent);
|
|
}
|
|
#endif /* TOGL_X11 */
|
|
|
|
#ifdef USE_OVERLAY
|
|
if (togl->OverlayFlag) {
|
|
if (SetupOverlay(togl) == TCL_ERROR) {
|
|
fprintf(stderr, "Warning: couldn't setup overlay.\n");
|
|
togl->OverlayFlag = False;
|
|
}
|
|
}
|
|
#endif /* USE_OVERLAY */
|
|
|
|
/* Request the X window to be displayed */
|
|
(void) XMapWindow(dpy, window);
|
|
|
|
#if defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
if (togl->ShareContext && FindTogl(togl->ShareContext)) {
|
|
/* share OpenGL context with existing Togl widget */
|
|
Togl *shareWith = FindTogl(togl->ShareContext);
|
|
|
|
assert(shareWith);
|
|
assert(shareWith->aglCtx);
|
|
togl->aglCtx = shareWith->aglCtx;
|
|
togl->VisInfo = shareWith->VisInfo;
|
|
visinfo = togl->VisInfo;
|
|
|
|
} else {
|
|
AGLContext shareCtx = NULL;
|
|
|
|
if (togl->PixelFormat) {
|
|
/* fill in RgbaFlag, DoubleFlag, and StereoFlag */
|
|
fmt = (AGLPixelFormat) togl->PixelFormat;
|
|
GLint has_rgba, has_doublebuf, has_stereo;
|
|
|
|
if (aglDescribePixelFormat(fmt, AGL_RGBA, &has_rgba) &&
|
|
aglDescribePixelFormat(fmt, AGL_DOUBLEBUFFER,
|
|
&has_doublebuf)
|
|
&& aglDescribePixelFormat(fmt, AGL_STEREO, &has_stereo)) {
|
|
togl->RgbaFlag = (has_rgba ? True : False);
|
|
togl->DoubleFlag = (has_doublebuf ? True : False);
|
|
togl->StereoFlag = (has_stereo ? True : False);
|
|
} else {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID
|
|
"Togl: failed querying pixel format attributes",
|
|
TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
} else {
|
|
|
|
/* Need to do this after mapping window, so MacDrawable structure
|
|
* is more completely filled in */
|
|
na = 0;
|
|
attribs[na++] = AGL_MINIMUM_POLICY;
|
|
attribs[na++] = AGL_ROBUST;
|
|
if (togl->RgbaFlag) {
|
|
/* RGB[A] mode */
|
|
attribs[na++] = AGL_RGBA;
|
|
attribs[na++] = AGL_RED_SIZE;
|
|
attribs[na++] = togl->RgbaRed;
|
|
attribs[na++] = AGL_GREEN_SIZE;
|
|
attribs[na++] = togl->RgbaGreen;
|
|
attribs[na++] = AGL_BLUE_SIZE;
|
|
attribs[na++] = togl->RgbaBlue;
|
|
if (togl->AlphaFlag) {
|
|
attribs[na++] = AGL_ALPHA_SIZE;
|
|
attribs[na++] = togl->AlphaSize;
|
|
}
|
|
} else {
|
|
/* Color index mode */
|
|
attribs[na++] = AGL_BUFFER_SIZE;
|
|
attribs[na++] = 8;
|
|
}
|
|
if (togl->DepthFlag) {
|
|
attribs[na++] = AGL_DEPTH_SIZE;
|
|
attribs[na++] = togl->DepthSize;
|
|
}
|
|
if (togl->DoubleFlag) {
|
|
attribs[na++] = AGL_DOUBLEBUFFER;
|
|
}
|
|
if (togl->StencilFlag) {
|
|
attribs[na++] = AGL_STENCIL_SIZE;
|
|
attribs[na++] = togl->StencilSize;
|
|
}
|
|
if (togl->AccumFlag) {
|
|
attribs[na++] = AGL_ACCUM_RED_SIZE;
|
|
attribs[na++] = togl->AccumRed;
|
|
attribs[na++] = AGL_ACCUM_GREEN_SIZE;
|
|
attribs[na++] = togl->AccumGreen;
|
|
attribs[na++] = AGL_ACCUM_BLUE_SIZE;
|
|
attribs[na++] = togl->AccumBlue;
|
|
if (togl->AlphaFlag) {
|
|
attribs[na++] = AGL_ACCUM_ALPHA_SIZE;
|
|
attribs[na++] = togl->AccumAlpha;
|
|
}
|
|
}
|
|
if (togl->AuxNumber != 0) {
|
|
attribs[na++] = AGL_AUX_BUFFERS;
|
|
attribs[na++] = togl->AuxNumber;
|
|
}
|
|
attribs[na++] = AGL_NONE;
|
|
|
|
if ((fmt = aglChoosePixelFormat(NULL, 0, attribs)) == NULL) {
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "Togl: couldn't choose pixel format",
|
|
TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Check whether to share lists.
|
|
*/
|
|
if (togl->ShareList) {
|
|
/* share display lists with existing togl widget */
|
|
Togl *shareWith = FindTogl(togl->ShareList);
|
|
|
|
if (shareWith)
|
|
shareCtx = shareWith->aglCtx;
|
|
}
|
|
if ((togl->aglCtx = aglCreateContext(fmt, shareCtx)) == NULL) {
|
|
GLenum err = aglGetError();
|
|
|
|
aglDestroyPixelFormat(fmt);
|
|
if (err == AGL_BAD_MATCH)
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID
|
|
"Togl: couldn't create context, shared context doesn't match",
|
|
TCL_STATIC);
|
|
else if (err == AGL_BAD_CONTEXT)
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID
|
|
"Togl: couldn't create context, bad shared context",
|
|
TCL_STATIC);
|
|
else if (err == AGL_BAD_PIXELFMT)
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID
|
|
"Togl: couldn't create context, bad pixel format",
|
|
TCL_STATIC);
|
|
else
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID
|
|
"Togl: couldn't create context, unknown reason",
|
|
TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
|
|
aglDestroyPixelFormat(fmt);
|
|
if (!aglSetDrawable(togl->aglCtx,
|
|
# if defined(TOGL_AGL)
|
|
((MacDrawable *) (window))->toplevel->grafPtr
|
|
# else
|
|
((MacDrawable *) (window))->toplevel->portPtr
|
|
# endif
|
|
)) {
|
|
aglDestroyContext(togl->aglCtx);
|
|
Tcl_SetResult(togl->Interp,
|
|
TCL_STUPID "Togl: couldn't set drawable", TCL_STATIC);
|
|
return DUMMY_WINDOW;
|
|
}
|
|
|
|
/* Just for portability, define the simplest visinfo */
|
|
visinfo = &VisInf;
|
|
visinfo->visual = DefaultVisual(dpy, DefaultScreen(dpy));
|
|
visinfo->depth = visinfo->visual->bits_per_rgb;
|
|
|
|
Tk_SetWindowVisual(togl->TkWin, visinfo->visual, visinfo->depth, cmap);
|
|
}
|
|
#endif /* TOGL_AGL_CLASSIC || TOGL_AGL */
|
|
|
|
#if defined(TOGL_X11)
|
|
/* Check for a single/double buffering snafu */
|
|
{
|
|
int dbl_flag;
|
|
|
|
if (glXGetConfig(dpy, visinfo, GLX_DOUBLEBUFFER, &dbl_flag)) {
|
|
if (!togl->DoubleFlag && dbl_flag) {
|
|
/* We requested single buffering but had to accept a */
|
|
/* double buffered visual. Set the GL draw buffer to */
|
|
/* be the front buffer to simulate single buffering. */
|
|
glDrawBuffer(GL_FRONT);
|
|
}
|
|
}
|
|
}
|
|
#endif /* TOGL_X11 */
|
|
|
|
/* for EPS Output */
|
|
if (!togl->RgbaFlag) {
|
|
int index_size;
|
|
|
|
#if defined(TOGL_X11) || defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
GLint index_bits;
|
|
|
|
glGetIntegerv(GL_INDEX_BITS, &index_bits);
|
|
index_size = 1 << index_bits;
|
|
#elif defined(TOGL_WGL)
|
|
index_size = togl->CiColormapSize;
|
|
#endif /* TOGL_X11 */
|
|
if (togl->EpsMapSize != index_size) {
|
|
if (togl->EpsRedMap)
|
|
free(togl->EpsRedMap);
|
|
if (togl->EpsGreenMap)
|
|
free(togl->EpsGreenMap);
|
|
if (togl->EpsBlueMap)
|
|
free(togl->EpsBlueMap);
|
|
togl->EpsMapSize = index_size;
|
|
togl->EpsRedMap = (GLfloat *) calloc(index_size, sizeof (GLfloat));
|
|
togl->EpsGreenMap =
|
|
(GLfloat *) calloc(index_size, sizeof (GLfloat));
|
|
togl->EpsBlueMap = (GLfloat *) calloc(index_size, sizeof (GLfloat));
|
|
}
|
|
}
|
|
|
|
return window;
|
|
}
|
|
|
|
/*
|
|
* Togl_WorldChanged
|
|
*
|
|
* Add support for setgrid option.
|
|
*/
|
|
static void
|
|
Togl_WorldChanged(ClientData instanceData)
|
|
{
|
|
Togl *togl = (Togl *) instanceData;
|
|
|
|
Tk_GeometryRequest(togl->TkWin, togl->Width, togl->Height);
|
|
Tk_SetInternalBorder(togl->TkWin, 0);
|
|
if (togl->SetGrid > 0) {
|
|
Tk_SetGrid(togl->TkWin, togl->Width / togl->SetGrid,
|
|
togl->Height / togl->SetGrid, togl->SetGrid, togl->SetGrid);
|
|
} else {
|
|
Tk_UnsetGrid(togl->TkWin);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* ToglCmdDeletedProc
|
|
*
|
|
* This procedure is invoked when a widget command is deleted. If
|
|
* the widget isn't already in the process of being destroyed,
|
|
* this command destroys it.
|
|
*
|
|
* Results:
|
|
* None.
|
|
*
|
|
* Side effects:
|
|
* The widget is destroyed.
|
|
*
|
|
*----------------------------------------------------------------------
|
|
*/
|
|
static void
|
|
ToglCmdDeletedProc(ClientData clientData)
|
|
{
|
|
Togl *togl = (Togl *) clientData;
|
|
Tk_Window tkwin = togl->TkWin;
|
|
|
|
/*
|
|
* This procedure could be invoked either because the window was
|
|
* destroyed and the command was then deleted (in which case tkwin
|
|
* is NULL) or because the command was deleted, and then this procedure
|
|
* destroys the widget.
|
|
*/
|
|
|
|
if (togl && tkwin) {
|
|
Tk_DeleteEventHandler(tkwin,
|
|
ExposureMask | StructureNotifyMask,
|
|
Togl_EventProc, (ClientData) togl);
|
|
}
|
|
#if defined(TOGL_X11)
|
|
if (togl->GlCtx) {
|
|
if (FindToglWithSameContext(togl) == NULL)
|
|
glXDestroyContext(togl->display, togl->GlCtx);
|
|
togl->GlCtx = NULL;
|
|
}
|
|
# ifdef USE_OVERLAY
|
|
if (togl->OverlayCtx) {
|
|
Tcl_HashEntry *entryPtr;
|
|
TkWindow *winPtr = (TkWindow *) togl->TkWin;
|
|
|
|
if (winPtr) {
|
|
entryPtr = Tcl_FindHashEntry(&winPtr->dispPtr->winTable,
|
|
(char *) togl->OverlayWindow);
|
|
Tcl_DeleteHashEntry(entryPtr);
|
|
}
|
|
if (FindToglWithSameOverlayContext(togl) == NULL)
|
|
glXDestroyContext(togl->display, togl->OverlayCtx);
|
|
togl->OverlayCtx = NULL;
|
|
}
|
|
# endif /* USE_OVERLAY */
|
|
#endif
|
|
/* TODO: delete contexts on other platforms */
|
|
|
|
if (tkwin != NULL) {
|
|
if (togl->SetGrid > 0) {
|
|
Tk_UnsetGrid(tkwin);
|
|
}
|
|
togl->TkWin = NULL;
|
|
Tk_DestroyWindow(tkwin);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Togl_Destroy
|
|
*
|
|
* Gets called when an Togl widget is destroyed.
|
|
*/
|
|
static void
|
|
Togl_Destroy(
|
|
#if (TK_MAJOR_VERSION * 100 + TK_MINOR_VERSION) >= 401
|
|
char *
|
|
#else
|
|
ClientData
|
|
#endif
|
|
clientData)
|
|
{
|
|
Togl *togl = (Togl *) clientData;
|
|
|
|
Tk_FreeOptions(configSpecs, WIDGREC togl, togl->display, 0);
|
|
|
|
#ifndef NO_TK_CURSOR
|
|
if (togl->Cursor != None) {
|
|
Tk_FreeCursor(togl->display, togl->Cursor);
|
|
}
|
|
#endif
|
|
if (togl->DestroyProc) {
|
|
togl->DestroyProc(togl);
|
|
}
|
|
|
|
/* remove from linked list */
|
|
RemoveFromList(togl);
|
|
|
|
#if !defined(TOGL_WGL)
|
|
/* TODO: why not on Windows? */
|
|
free(togl);
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* This gets called to handle Togl window configuration events
|
|
*/
|
|
static void
|
|
Togl_EventProc(ClientData clientData, XEvent *eventPtr)
|
|
{
|
|
Togl *togl = (Togl *) clientData;
|
|
|
|
switch (eventPtr->type) {
|
|
case Expose:
|
|
if (eventPtr->xexpose.count == 0) {
|
|
if (!togl->UpdatePending
|
|
&& eventPtr->xexpose.window == Tk_WindowId(togl->TkWin)) {
|
|
Togl_PostRedisplay(togl);
|
|
}
|
|
#if defined(TOGL_X11)
|
|
if (!togl->OverlayUpdatePending && togl->OverlayFlag
|
|
&& togl->OverlayIsMapped
|
|
&& eventPtr->xexpose.window == togl->OverlayWindow) {
|
|
Togl_PostOverlayRedisplay(togl);
|
|
}
|
|
#endif /* TOGL_X11 */
|
|
}
|
|
break;
|
|
case ConfigureNotify:
|
|
if (togl->Width != Tk_Width(togl->TkWin)
|
|
|| togl->Height != Tk_Height(togl->TkWin)) {
|
|
togl->Width = Tk_Width(togl->TkWin);
|
|
togl->Height = Tk_Height(togl->TkWin);
|
|
(void) XResizeWindow(Tk_Display(togl->TkWin),
|
|
Tk_WindowId(togl->TkWin), togl->Width, togl->Height);
|
|
#if defined(TOGL_X11)
|
|
if (togl->OverlayFlag) {
|
|
(void) XResizeWindow(Tk_Display(togl->TkWin),
|
|
togl->OverlayWindow, togl->Width, togl->Height);
|
|
(void) XRaiseWindow(Tk_Display(togl->TkWin),
|
|
togl->OverlayWindow);
|
|
}
|
|
#endif /* TOGL_X11 */
|
|
Togl_MakeCurrent(togl);
|
|
if (togl->ReshapeProc) {
|
|
togl->ReshapeProc(togl);
|
|
} else {
|
|
glViewport(0, 0, togl->Width, togl->Height);
|
|
#if defined(TOGL_X11)
|
|
if (togl->OverlayFlag) {
|
|
Togl_UseLayer(togl, TOGL_OVERLAY);
|
|
glViewport(0, 0, togl->Width, togl->Height);
|
|
Togl_UseLayer(togl, TOGL_NORMAL);
|
|
}
|
|
#endif /* TOGL_X11 */
|
|
}
|
|
#ifndef TOGL_WGL /* causes double redisplay on Win32 platform */
|
|
Togl_PostRedisplay(togl);
|
|
#endif /* TOGL_WGL */
|
|
}
|
|
break;
|
|
case MapNotify:
|
|
#if defined(TOGL_AGL)
|
|
{
|
|
/*
|
|
* See comment for the UnmapNotify case below.
|
|
*/
|
|
AGLDrawable d = TkMacOSXGetDrawablePort(Tk_WindowId(togl->TkWin));
|
|
|
|
aglSetDrawable(togl->aglCtx, d);
|
|
}
|
|
#endif /* TOGL_AGL */
|
|
break;
|
|
case UnmapNotify:
|
|
#if defined(TOGL_AGL)
|
|
{
|
|
/*
|
|
* For Mac OS X Aqua, Tk subwindows are not implemented as
|
|
* separate Aqua windows. They are just different regions of
|
|
* a single Aqua window. To unmap them they are just not drawn.
|
|
* Have to disconnect the AGL context otherwise they will continue
|
|
* to be displayed directly by Aqua.
|
|
*/
|
|
aglSetDrawable(togl->aglCtx, NULL);
|
|
}
|
|
#endif /* TOGL_AGL */
|
|
break;
|
|
case DestroyNotify:
|
|
if (togl->TkWin != NULL) {
|
|
if (togl->SetGrid > 0) {
|
|
Tk_UnsetGrid(togl->TkWin);
|
|
}
|
|
togl->TkWin = NULL;
|
|
#if (TCL_MAJOR_VERSION * 100 + TCL_MINOR_VERSION) >= 800
|
|
/* This function new in Tcl/Tk 8.0 */
|
|
(void) Tcl_DeleteCommandFromToken(togl->Interp, togl->widgetCmd);
|
|
#endif
|
|
}
|
|
if (togl->TimerProc != NULL) {
|
|
#if (TK_MAJOR_VERSION * 100 + TK_MINOR_VERSION) >= 401
|
|
Tcl_DeleteTimerHandler(togl->timerHandler);
|
|
#else
|
|
Tk_DeleteTimerHandler(togl->timerHandler);
|
|
#endif
|
|
|
|
}
|
|
if (togl->UpdatePending) {
|
|
#if (TCL_MAJOR_VERSION * 100 + TCL_MINOR_VERSION) >= 705
|
|
Tcl_CancelIdleCall(Togl_Render, (ClientData) togl);
|
|
#else
|
|
Tk_CancelIdleCall(Togl_Render, (ClientData) togl);
|
|
#endif
|
|
}
|
|
#if (TK_MAJOR_VERSION * 100 + TK_MINOR_VERSION) >= 401
|
|
Tcl_EventuallyFree((ClientData) togl, Togl_Destroy);
|
|
#else
|
|
Tk_EventuallyFree((ClientData) togl, Togl_Destroy);
|
|
#endif
|
|
|
|
break;
|
|
default:
|
|
/* nothing */
|
|
;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void
|
|
Togl_PostRedisplay(Togl *togl)
|
|
{
|
|
if (!togl->UpdatePending) {
|
|
togl->UpdatePending = True;
|
|
Tk_DoWhenIdle(Togl_Render, (ClientData) togl);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void
|
|
Togl_SwapBuffers(const Togl *togl)
|
|
{
|
|
if (togl->DoubleFlag) {
|
|
#if defined(TOGL_WGL)
|
|
int res = SwapBuffers(togl->tglGLHdc);
|
|
|
|
assert(res == TRUE);
|
|
#elif defined(TOGL_X11)
|
|
glXSwapBuffers(Tk_Display(togl->TkWin), Tk_WindowId(togl->TkWin));
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
aglSwapBuffers(togl->aglCtx);
|
|
#endif /* TOGL_WGL */
|
|
} else {
|
|
glFlush();
|
|
}
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
Togl_Ident(const Togl *togl)
|
|
{
|
|
return togl->Ident;
|
|
}
|
|
|
|
|
|
int
|
|
Togl_Width(const Togl *togl)
|
|
{
|
|
return togl->Width;
|
|
}
|
|
|
|
|
|
int
|
|
Togl_Height(const Togl *togl)
|
|
{
|
|
return togl->Height;
|
|
}
|
|
|
|
|
|
Tcl_Interp *
|
|
Togl_Interp(const Togl *togl)
|
|
{
|
|
return togl->Interp;
|
|
}
|
|
|
|
|
|
Tk_Window
|
|
Togl_TkWin(const Togl *togl)
|
|
{
|
|
return togl->TkWin;
|
|
}
|
|
|
|
|
|
#if defined(TOGL_X11)
|
|
/*
|
|
* A replacement for XAllocColor. This function should never
|
|
* fail to allocate a color. When XAllocColor fails, we return
|
|
* the nearest matching color. If we have to allocate many colors
|
|
* this function isn't too efficient; the XQueryColors() could be
|
|
* done just once.
|
|
* Written by Michael Pichler, Brian Paul, Mark Kilgard
|
|
* Input: dpy - X display
|
|
* cmap - X colormap
|
|
* cmapSize - size of colormap
|
|
* In/Out: color - the XColor struct
|
|
* Output: exact - 1=exact color match, 0=closest match
|
|
*/
|
|
static void
|
|
noFaultXAllocColor(Display *dpy, Colormap cmap, int cmapSize,
|
|
XColor *color, int *exact)
|
|
{
|
|
XColor *ctable, subColor;
|
|
int i, bestmatch;
|
|
double mindist; /* 3*2^16^2 exceeds long int precision. */
|
|
|
|
/* First try just using XAllocColor. */
|
|
if (XAllocColor(dpy, cmap, color)) {
|
|
*exact = 1;
|
|
return;
|
|
}
|
|
|
|
/* Retrieve color table entries. */
|
|
/* XXX alloca candidate. */
|
|
ctable = (XColor *) malloc(cmapSize * sizeof (XColor));
|
|
for (i = 0; i < cmapSize; i++) {
|
|
ctable[i].pixel = i;
|
|
}
|
|
(void) XQueryColors(dpy, cmap, ctable, cmapSize);
|
|
|
|
/* Find best match. */
|
|
bestmatch = -1;
|
|
mindist = 0;
|
|
for (i = 0; i < cmapSize; i++) {
|
|
double dr = (double) color->red - (double) ctable[i].red;
|
|
double dg = (double) color->green - (double) ctable[i].green;
|
|
double db = (double) color->blue - (double) ctable[i].blue;
|
|
double dist = dr * dr + dg * dg + db * db;
|
|
|
|
if (bestmatch < 0 || dist < mindist) {
|
|
bestmatch = i;
|
|
mindist = dist;
|
|
}
|
|
}
|
|
|
|
/* Return result. */
|
|
subColor.red = ctable[bestmatch].red;
|
|
subColor.green = ctable[bestmatch].green;
|
|
subColor.blue = ctable[bestmatch].blue;
|
|
free(ctable);
|
|
/* Try to allocate the closest match color. This should only fail if the
|
|
* cell is read/write. Otherwise, we're incrementing the cell's reference
|
|
* count. */
|
|
if (!XAllocColor(dpy, cmap, &subColor)) {
|
|
/* do this to work around a problem reported by Frank Ortega */
|
|
subColor.pixel = (unsigned long) bestmatch;
|
|
subColor.red = ctable[bestmatch].red;
|
|
subColor.green = ctable[bestmatch].green;
|
|
subColor.blue = ctable[bestmatch].blue;
|
|
subColor.flags = DoRed | DoGreen | DoBlue;
|
|
}
|
|
*color = subColor;
|
|
}
|
|
|
|
#elif defined(TOGL_WGL)
|
|
|
|
static UINT
|
|
Win32AllocColor(const Togl *togl, float red, float green, float blue)
|
|
{
|
|
/* Modified version of XAllocColor emulation of Tk. - returns index,
|
|
* instead of color itself - allocates logical palette entry even for
|
|
* non-palette devices */
|
|
|
|
TkWinColormap *cmap = (TkWinColormap *) Tk_Colormap(togl->TkWin);
|
|
UINT index;
|
|
COLORREF newColor, closeColor;
|
|
PALETTEENTRY entry, closeEntry;
|
|
int new, refCount;
|
|
Tcl_HashEntry *entryPtr;
|
|
|
|
entry.peRed = (unsigned char) (red * 255 + .5);
|
|
entry.peGreen = (unsigned char) (green * 255 + .5);
|
|
entry.peBlue = (unsigned char) (blue * 255 + .5);
|
|
entry.peFlags = 0;
|
|
|
|
/*
|
|
* Find the nearest existing palette entry.
|
|
*/
|
|
|
|
newColor = RGB(entry.peRed, entry.peGreen, entry.peBlue);
|
|
index = GetNearestPaletteIndex(cmap->palette, newColor);
|
|
GetPaletteEntries(cmap->palette, index, 1, &closeEntry);
|
|
closeColor = RGB(closeEntry.peRed, closeEntry.peGreen, closeEntry.peBlue);
|
|
|
|
/*
|
|
* If this is not a duplicate and colormap is not full, allocate a new entry.
|
|
*/
|
|
|
|
if (newColor != closeColor) {
|
|
if (cmap->size == (unsigned int) togl->CiColormapSize) {
|
|
entry = closeEntry;
|
|
} else {
|
|
cmap->size++;
|
|
ResizePalette(cmap->palette, cmap->size);
|
|
index = cmap->size - 1;
|
|
SetPaletteEntries(cmap->palette, index, 1, &entry);
|
|
SelectPalette(togl->tglGLHdc, cmap->palette, TRUE);
|
|
RealizePalette(togl->tglGLHdc);
|
|
}
|
|
}
|
|
newColor = PALETTERGB(entry.peRed, entry.peGreen, entry.peBlue);
|
|
entryPtr = Tcl_CreateHashEntry(&cmap->refCounts, (char *) newColor, &new);
|
|
if (new) {
|
|
refCount = 1;
|
|
} else {
|
|
refCount = ((int) Tcl_GetHashValue(entryPtr)) + 1;
|
|
}
|
|
Tcl_SetHashValue(entryPtr, (ClientData) refCount);
|
|
|
|
/* for EPS output */
|
|
togl->EpsRedMap[index] = (GLfloat) (entry.peRed / 255.0);
|
|
togl->EpsGreenMap[index] = (GLfloat) (entry.peGreen / 255.0);
|
|
togl->EpsBlueMap[index] = (GLfloat) (entry.peBlue / 255.0);
|
|
return index;
|
|
}
|
|
|
|
static void
|
|
Win32FreeColor(const Togl *togl, unsigned long index)
|
|
{
|
|
TkWinColormap *cmap = (TkWinColormap *) Tk_Colormap(togl->TkWin);
|
|
COLORREF cref;
|
|
UINT count, refCount;
|
|
PALETTEENTRY entry, *entries;
|
|
Tcl_HashEntry *entryPtr;
|
|
|
|
if (index >= cmap->size) {
|
|
panic("Tried to free a color that isn't allocated.");
|
|
}
|
|
GetPaletteEntries(cmap->palette, index, 1, &entry);
|
|
cref = PALETTERGB(entry.peRed, entry.peGreen, entry.peBlue);
|
|
entryPtr = Tcl_FindHashEntry(&cmap->refCounts, (char *) cref);
|
|
if (!entryPtr) {
|
|
panic("Tried to free a color that isn't allocated.");
|
|
}
|
|
refCount = (int) Tcl_GetHashValue(entryPtr) - 1;
|
|
if (refCount == 0) {
|
|
count = cmap->size - index;
|
|
entries = (PALETTEENTRY *) ckalloc(sizeof (PALETTEENTRY) * count);
|
|
GetPaletteEntries(cmap->palette, index + 1, count, entries);
|
|
SetPaletteEntries(cmap->palette, index, count, entries);
|
|
SelectPalette(togl->tglGLHdc, cmap->palette, TRUE);
|
|
RealizePalette(togl->tglGLHdc);
|
|
ckfree((char *) entries);
|
|
cmap->size--;
|
|
Tcl_DeleteHashEntry(entryPtr);
|
|
} else {
|
|
Tcl_SetHashValue(entryPtr, (ClientData) refCount);
|
|
}
|
|
}
|
|
|
|
static void
|
|
Win32SetColor(const Togl *togl,
|
|
unsigned long index, float red, float green, float blue)
|
|
{
|
|
TkWinColormap *cmap = (TkWinColormap *) Tk_Colormap(togl->TkWin);
|
|
PALETTEENTRY entry;
|
|
|
|
entry.peRed = (unsigned char) (red * 255 + .5);
|
|
entry.peGreen = (unsigned char) (green * 255 + .5);
|
|
entry.peBlue = (unsigned char) (blue * 255 + .5);
|
|
entry.peFlags = 0;
|
|
SetPaletteEntries(cmap->palette, index, 1, &entry);
|
|
SelectPalette(togl->tglGLHdc, cmap->palette, TRUE);
|
|
RealizePalette(togl->tglGLHdc);
|
|
|
|
/* for EPS output */
|
|
togl->EpsRedMap[index] = (GLfloat) (entry.peRed / 255.0);
|
|
togl->EpsGreenMap[index] = (GLfloat) (entry.peGreen / 255.0);
|
|
togl->EpsBlueMap[index] = (GLfloat) (entry.peBlue / 255.0);
|
|
}
|
|
#endif /* TOGL_X11 */
|
|
|
|
|
|
unsigned long
|
|
Togl_AllocColor(const Togl *togl, float red, float green, float blue)
|
|
{
|
|
if (togl->RgbaFlag) {
|
|
(void) fprintf(stderr,
|
|
"Error: Togl_AllocColor illegal in RGBA mode.\n");
|
|
return 0;
|
|
}
|
|
/* TODO: maybe not... */
|
|
if (togl->PrivateCmapFlag) {
|
|
(void) fprintf(stderr,
|
|
"Error: Togl_FreeColor illegal with private colormap\n");
|
|
return 0;
|
|
}
|
|
#if defined(TOGL_X11)
|
|
{
|
|
XColor xcol;
|
|
int exact;
|
|
|
|
xcol.red = (short) (red * 65535.0);
|
|
xcol.green = (short) (green * 65535.0);
|
|
xcol.blue = (short) (blue * 65535.0);
|
|
|
|
noFaultXAllocColor(Tk_Display(togl->TkWin), Tk_Colormap(togl->TkWin),
|
|
Tk_Visual(togl->TkWin)->map_entries, &xcol, &exact);
|
|
/* for EPS output */
|
|
togl->EpsRedMap[xcol.pixel] = (float) xcol.red / 65535.0;
|
|
togl->EpsGreenMap[xcol.pixel] = (float) xcol.green / 65535.0;
|
|
togl->EpsBlueMap[xcol.pixel] = (float) xcol.blue / 65535.0;
|
|
|
|
return xcol.pixel;
|
|
}
|
|
|
|
#elif defined(TOGL_WGL)
|
|
return Win32AllocColor(togl, red, green, blue);
|
|
|
|
#elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
/* still need to implement this on Mac... */
|
|
return 0;
|
|
|
|
#endif /* TOGL_X11 */
|
|
}
|
|
|
|
|
|
|
|
void
|
|
Togl_FreeColor(const Togl *togl, unsigned long pixel)
|
|
{
|
|
if (togl->RgbaFlag) {
|
|
(void) fprintf(stderr,
|
|
"Error: Togl_AllocColor illegal in RGBA mode.\n");
|
|
return;
|
|
}
|
|
/* TODO: maybe not... */
|
|
if (togl->PrivateCmapFlag) {
|
|
(void) fprintf(stderr,
|
|
"Error: Togl_FreeColor illegal with private colormap\n");
|
|
return;
|
|
}
|
|
#if defined(TOGL_X11)
|
|
(void) XFreeColors(Tk_Display(togl->TkWin), Tk_Colormap(togl->TkWin),
|
|
&pixel, 1, 0);
|
|
#elif defined(TOGL_WGL)
|
|
Win32FreeColor(togl, pixel);
|
|
#endif /* TOGL_X11 */
|
|
}
|
|
|
|
|
|
|
|
void
|
|
Togl_SetColor(const Togl *togl,
|
|
unsigned long index, float red, float green, float blue)
|
|
{
|
|
|
|
if (togl->RgbaFlag) {
|
|
(void) fprintf(stderr,
|
|
"Error: Togl_AllocColor illegal in RGBA mode.\n");
|
|
return;
|
|
}
|
|
if (!togl->PrivateCmapFlag) {
|
|
(void) fprintf(stderr,
|
|
"Error: Togl_SetColor requires a private colormap\n");
|
|
return;
|
|
}
|
|
#if defined(TOGL_X11)
|
|
{
|
|
XColor xcol;
|
|
|
|
xcol.pixel = index;
|
|
xcol.red = (short) (red * 65535.0);
|
|
xcol.green = (short) (green * 65535.0);
|
|
xcol.blue = (short) (blue * 65535.0);
|
|
xcol.flags = DoRed | DoGreen | DoBlue;
|
|
|
|
(void) XStoreColor(Tk_Display(togl->TkWin), Tk_Colormap(togl->TkWin),
|
|
&xcol);
|
|
|
|
/* for EPS output */
|
|
togl->EpsRedMap[xcol.pixel] = (float) xcol.red / 65535.0;
|
|
togl->EpsGreenMap[xcol.pixel] = (float) xcol.green / 65535.0;
|
|
togl->EpsBlueMap[xcol.pixel] = (float) xcol.blue / 65535.0;
|
|
}
|
|
#elif defined(TOGL_WGL)
|
|
Win32SetColor(togl, index, red, green, blue);
|
|
#endif /* TOGL_X11 */
|
|
}
|
|
|
|
|
|
#if TOGL_USE_FONTS == 1
|
|
|
|
# if defined(TOGL_WGL)
|
|
# include "tkWinInt.h"
|
|
# include "tkFont.h"
|
|
|
|
/*
|
|
* The following structure represents Windows' implementation of a font.
|
|
*/
|
|
|
|
typedef struct WinFont
|
|
{
|
|
TkFont font; /* Stuff used by generic font package. Must be
|
|
* first in structure. */
|
|
HFONT hFont; /* Windows information about font. */
|
|
HWND hwnd; /* Toplevel window of application that owns
|
|
* this font, used for getting HDC. */
|
|
int widths[256]; /* Widths of first 256 chars in this font. */
|
|
} WinFont;
|
|
# endif /* TOGL_WGL */
|
|
|
|
|
|
# define MAX_FONTS 1000
|
|
static GLuint ListBase[MAX_FONTS];
|
|
static GLuint ListCount[MAX_FONTS];
|
|
|
|
|
|
|
|
/*
|
|
* Load the named bitmap font as a sequence of bitmaps in a display list.
|
|
* fontname may be one of the predefined fonts like TOGL_BITMAP_8_BY_13
|
|
* or an X font name, or a Windows font name, etc.
|
|
*/
|
|
GLuint
|
|
Togl_LoadBitmapFont(const Togl *togl, const char *fontname)
|
|
{
|
|
static Bool FirstTime = True;
|
|
|
|
# if defined(TOGL_X11)
|
|
XFontStruct *fontinfo;
|
|
# elif defined(TOGL_WGL)
|
|
WinFont *winfont;
|
|
HFONT oldFont;
|
|
TEXTMETRIC tm;
|
|
# endif
|
|
/* TOGL_X11 */
|
|
int first, last, count;
|
|
GLuint fontbase;
|
|
const char *name;
|
|
|
|
/* Initialize the ListBase and ListCount arrays */
|
|
if (FirstTime) {
|
|
int i;
|
|
|
|
for (i = 0; i < MAX_FONTS; i++) {
|
|
ListBase[i] = ListCount[i] = 0;
|
|
}
|
|
FirstTime = False;
|
|
}
|
|
|
|
/*
|
|
* This method of selecting X fonts according to a TOGL_ font name
|
|
* is a kludge. To be fixed when I find time...
|
|
*/
|
|
if (fontname == TOGL_BITMAP_8_BY_13) {
|
|
name = "8x13";
|
|
} else if (fontname == TOGL_BITMAP_9_BY_15) {
|
|
name = "9x15";
|
|
} else if (fontname == TOGL_BITMAP_TIMES_ROMAN_10) {
|
|
name = "-adobe-times-medium-r-normal--10-100-75-75-p-54-iso8859-1";
|
|
} else if (fontname == TOGL_BITMAP_TIMES_ROMAN_24) {
|
|
name = "-adobe-times-medium-r-normal--24-240-75-75-p-124-iso8859-1";
|
|
} else if (fontname == TOGL_BITMAP_HELVETICA_10) {
|
|
name = "-adobe-helvetica-medium-r-normal--10-100-75-75-p-57-iso8859-1";
|
|
} else if (fontname == TOGL_BITMAP_HELVETICA_12) {
|
|
name = "-adobe-helvetica-medium-r-normal--12-120-75-75-p-67-iso8859-1";
|
|
} else if (fontname == TOGL_BITMAP_HELVETICA_18) {
|
|
name = "-adobe-helvetica-medium-r-normal--18-180-75-75-p-98-iso8859-1";
|
|
} else if (!fontname) {
|
|
name = DEFAULT_FONTNAME;
|
|
} else {
|
|
name = (const char *) fontname;
|
|
}
|
|
|
|
assert(name);
|
|
|
|
# if defined(TOGL_X11)
|
|
fontinfo = (XFontStruct *) XLoadQueryFont(Tk_Display(togl->TkWin), name);
|
|
if (!fontinfo) {
|
|
return 0;
|
|
}
|
|
first = fontinfo->min_char_or_byte2;
|
|
last = fontinfo->max_char_or_byte2;
|
|
# elif defined(TOGL_WGL)
|
|
winfont = (WinFont *) Tk_GetFont(togl->Interp, togl->TkWin, name);
|
|
if (!winfont) {
|
|
return 0;
|
|
}
|
|
oldFont = SelectObject(togl->tglGLHdc, winfont->hFont);
|
|
GetTextMetrics(togl->tglGLHdc, &tm);
|
|
first = tm.tmFirstChar;
|
|
last = tm.tmLastChar;
|
|
# elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
first = 10; /* don't know how to determine font range on
|
|
* Mac... */
|
|
last = 127;
|
|
# endif
|
|
/* TOGL_X11 */
|
|
|
|
count = last - first + 1;
|
|
fontbase = glGenLists((GLuint) (last + 1));
|
|
if (fontbase == 0) {
|
|
# ifdef TOGL_WGL
|
|
SelectObject(togl->tglGLHdc, oldFont);
|
|
Tk_FreeFont((Tk_Font) winfont);
|
|
# endif
|
|
/* TOGL_WGL */
|
|
return 0;
|
|
}
|
|
# if defined(TOGL_WGL)
|
|
wglUseFontBitmaps(togl->tglGLHdc, first, count, (int) fontbase + first);
|
|
SelectObject(togl->tglGLHdc, oldFont);
|
|
Tk_FreeFont((Tk_Font) winfont);
|
|
# elif defined(TOGL_X11)
|
|
glXUseXFont(fontinfo->fid, first, count, (int) fontbase + first);
|
|
# elif defined(TOGL_AGL_CLASSIC) || defined(TOGL_AGL)
|
|
aglUseFont(togl->aglCtx, 1, 0, 14, /* for now, only app font, regular
|
|
* 14-point */
|
|
10, 118, fontbase + first);
|
|
# endif
|
|
|
|
/* Record the list base and number of display lists for
|
|
* Togl_UnloadBitmapFont(). */
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < MAX_FONTS; i++) {
|
|
if (ListBase[i] == 0) {
|
|
ListBase[i] = fontbase;
|
|
ListCount[i] = last + 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return fontbase;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Release the display lists which were generated by Togl_LoadBitmapFont().
|
|
*/
|
|
void
|
|
Togl_UnloadBitmapFont(const Togl *togl, GLuint fontbase)
|
|
{
|
|
int i;
|
|
|
|
(void) togl;
|
|
for (i = 0; i < MAX_FONTS; i++) {
|
|
if (ListBase[i] == fontbase) {
|
|
glDeleteLists(ListBase[i], ListCount[i]);
|
|
ListBase[i] = ListCount[i] = 0;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif /* TOGL_USE_FONTS */
|
|
|
|
|
|
/*
|
|
* Overlay functions
|
|
*/
|
|
|
|
|
|
void
|
|
Togl_UseLayer(Togl *togl, int layer)
|
|
{
|
|
if (!togl->OverlayWindow)
|
|
return;
|
|
if (layer == TOGL_OVERLAY) {
|
|
#if defined(TOGL_WGL)
|
|
int res = wglMakeCurrent(togl->tglGLHdc, togl->tglGLOverlayHglrc);
|
|
|
|
assert(res == TRUE);
|
|
#elif defined(TOGL_X11)
|
|
(void) glXMakeCurrent(Tk_Display(togl->TkWin),
|
|
togl->OverlayWindow, togl->OverlayCtx);
|
|
# if defined(__sgi)
|
|
if (togl->OldStereoFlag)
|
|
oldStereoMakeCurrent(Tk_Display(togl->TkWin),
|
|
togl->OverlayWindow, togl->OverlayCtx);
|
|
# endif
|
|
/* __sgi STEREO */
|
|
#endif /* TOGL_WGL */
|
|
} else if (layer == TOGL_NORMAL) {
|
|
#if defined(TOGL_WGL)
|
|
int res = wglMakeCurrent(togl->tglGLHdc, togl->tglGLHglrc);
|
|
|
|
assert(res == TRUE);
|
|
#elif defined(TOGL_X11)
|
|
(void) glXMakeCurrent(Tk_Display(togl->TkWin),
|
|
Tk_WindowId(togl->TkWin), togl->GlCtx);
|
|
# if defined(__sgi)
|
|
if (togl->OldStereoFlag)
|
|
oldStereoMakeCurrent(Tk_Display(togl->TkWin),
|
|
Tk_WindowId(togl->TkWin), togl->GlCtx);
|
|
# endif
|
|
/* __sgi STEREO */
|
|
#endif /* TOGL_WGL */
|
|
} else {
|
|
/* error */
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
Togl_ShowOverlay(Togl *togl)
|
|
{
|
|
#if defined(TOGL_X11) /* not yet implemented on Windows */
|
|
if (togl->OverlayWindow) {
|
|
(void) XMapWindow(Tk_Display(togl->TkWin), togl->OverlayWindow);
|
|
(void) XInstallColormap(Tk_Display(togl->TkWin), togl->OverlayCmap);
|
|
togl->OverlayIsMapped = True;
|
|
}
|
|
#endif /* TOGL_X11 */
|
|
}
|
|
|
|
|
|
void
|
|
Togl_HideOverlay(Togl *togl)
|
|
{
|
|
if (togl->OverlayWindow && togl->OverlayIsMapped) {
|
|
(void) XUnmapWindow(Tk_Display(togl->TkWin), togl->OverlayWindow);
|
|
togl->OverlayIsMapped = False;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
Togl_PostOverlayRedisplay(Togl *togl)
|
|
{
|
|
if (!togl->OverlayUpdatePending
|
|
&& togl->OverlayWindow && togl->OverlayDisplayProc) {
|
|
Tk_DoWhenIdle(RenderOverlay, (ClientData) togl);
|
|
togl->OverlayUpdatePending = True;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
Togl_OverlayDisplayFunc(Togl_Callback *proc)
|
|
{
|
|
DefaultOverlayDisplayProc = proc;
|
|
}
|
|
|
|
|
|
int
|
|
Togl_ExistsOverlay(const Togl *togl)
|
|
{
|
|
return togl->OverlayFlag;
|
|
}
|
|
|
|
|
|
int
|
|
Togl_GetOverlayTransparentValue(const Togl *togl)
|
|
{
|
|
return togl->OverlayTransparentPixel;
|
|
}
|
|
|
|
|
|
int
|
|
Togl_IsMappedOverlay(const Togl *togl)
|
|
{
|
|
return togl->OverlayFlag && togl->OverlayIsMapped;
|
|
}
|
|
|
|
|
|
unsigned long
|
|
Togl_AllocColorOverlay(const Togl *togl, float red, float green, float blue)
|
|
{
|
|
#if defined(TOGL_X11) /* not yet implemented on Windows */
|
|
if (togl->OverlayFlag && togl->OverlayCmap) {
|
|
XColor xcol;
|
|
|
|
xcol.red = (short) (red * 65535.0);
|
|
xcol.green = (short) (green * 65535.0);
|
|
xcol.blue = (short) (blue * 65535.0);
|
|
if (!XAllocColor(Tk_Display(togl->TkWin), togl->OverlayCmap, &xcol))
|
|
return (unsigned long) -1;
|
|
return xcol.pixel;
|
|
}
|
|
#endif /* TOGL_X11 */
|
|
return (unsigned long) -1;
|
|
}
|
|
|
|
|
|
void
|
|
Togl_FreeColorOverlay(const Togl *togl, unsigned long pixel)
|
|
{
|
|
#if defined(TOGL_X11) /* not yet implemented on Windows */
|
|
if (togl->OverlayFlag && togl->OverlayCmap) {
|
|
(void) XFreeColors(Tk_Display(togl->TkWin), togl->OverlayCmap, &pixel,
|
|
1, 0);
|
|
}
|
|
#endif /* TOGL_X11 */
|
|
}
|
|
|
|
|
|
/*
|
|
* User client data
|
|
*/
|
|
|
|
void
|
|
Togl_ClientData(ClientData clientData)
|
|
{
|
|
DefaultClientData = clientData;
|
|
}
|
|
|
|
|
|
ClientData
|
|
Togl_GetClientData(const Togl *togl)
|
|
{
|
|
return togl->Client_Data;
|
|
}
|
|
|
|
|
|
void
|
|
Togl_SetClientData(Togl *togl, ClientData clientData)
|
|
{
|
|
togl->Client_Data = clientData;
|
|
}
|
|
|
|
|
|
/*
|
|
* X11-only functions
|
|
* Contributed by Miguel A. De Riera Pasenau (miguel@DALILA.UPC.ES)
|
|
*/
|
|
|
|
Display *
|
|
Togl_Display(const Togl *togl)
|
|
{
|
|
return Tk_Display(togl->TkWin);
|
|
}
|
|
|
|
Screen *
|
|
Togl_Screen(const Togl *togl)
|
|
{
|
|
return Tk_Screen(togl->TkWin);
|
|
}
|
|
|
|
int
|
|
Togl_ScreenNumber(const Togl *togl)
|
|
{
|
|
return Tk_ScreenNumber(togl->TkWin);
|
|
}
|
|
|
|
Colormap
|
|
Togl_Colormap(const Togl *togl)
|
|
{
|
|
return Tk_Colormap(togl->TkWin);
|
|
}
|
|
|
|
|
|
|
|
#ifdef MESA_COLOR_HACK
|
|
/*
|
|
* Let's know how many free colors do we have
|
|
*/
|
|
# if 0
|
|
static unsigned char rojo[] = { 4, 39, 74, 110, 145, 181, 216, 251 }, verde[] = {
|
|
4, 39, 74, 110, 145, 181, 216, 251}, azul[] = {
|
|
4, 39, 74, 110, 145, 181, 216, 251};
|
|
|
|
unsigned char rojo[] = { 4, 36, 72, 109, 145, 182, 218, 251 }, verde[] = {
|
|
4, 36, 72, 109, 145, 182, 218, 251}, azul[] = {
|
|
4, 36, 72, 109, 145, 182, 218, 251};
|
|
|
|
azul[] = {
|
|
0, 85, 170, 255};
|
|
# endif
|
|
|
|
# define RLEVELS 5
|
|
# define GLEVELS 9
|
|
# define BLEVELS 5
|
|
|
|
/* to free dithered_rgb_colormap pixels allocated by Mesa */
|
|
static unsigned long *ToglMesaUsedPixelCells = NULL;
|
|
static int ToglMesaUsedFreeCells = 0;
|
|
|
|
static int
|
|
get_free_color_cells(Display *display, int screen, Colormap colormap)
|
|
{
|
|
if (!ToglMesaUsedPixelCells) {
|
|
XColor xcol;
|
|
int i;
|
|
int colorsfailed, ncolors = XDisplayCells(display, screen);
|
|
|
|
long r, g, b;
|
|
|
|
ToglMesaUsedPixelCells =
|
|
(unsigned long *) calloc(ncolors, sizeof (unsigned long));
|
|
|
|
/* Allocate X colors and initialize color_table[], red_table[], etc */
|
|
/* de Mesa 2.1: xmesa1.c setup_dithered_(...) */
|
|
i = colorsfailed = 0;
|
|
for (r = 0; r < RLEVELS; r++)
|
|
for (g = 0; g < GLEVELS; g++)
|
|
for (b = 0; b < BLEVELS; b++) {
|
|
int exact;
|
|
|
|
xcol.red = (r * 65535) / (RLEVELS - 1);
|
|
xcol.green = (g * 65535) / (GLEVELS - 1);
|
|
xcol.blue = (b * 65535) / (BLEVELS - 1);
|
|
noFaultXAllocColor(display, colormap, ncolors,
|
|
&xcol, &exact);
|
|
ToglMesaUsedPixelCells[i++] = xcol.pixel;
|
|
if (!exact) {
|
|
colorsfailed++;
|
|
}
|
|
}
|
|
ToglMesaUsedFreeCells = i;
|
|
|
|
XFreeColors(display, colormap, ToglMesaUsedPixelCells,
|
|
ToglMesaUsedFreeCells, 0x00000000);
|
|
}
|
|
return ToglMesaUsedFreeCells;
|
|
}
|
|
|
|
|
|
static void
|
|
free_default_color_cells(Display *display, Colormap colormap)
|
|
{
|
|
if (ToglMesaUsedPixelCells) {
|
|
XFreeColors(display, colormap, ToglMesaUsedPixelCells,
|
|
ToglMesaUsedFreeCells, 0x00000000);
|
|
free(ToglMesaUsedPixelCells);
|
|
ToglMesaUsedPixelCells = NULL;
|
|
ToglMesaUsedFreeCells = 0;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
/*
|
|
* Generate EPS file.
|
|
* Contributed by Miguel A. De Riera Pasenau (miguel@DALILA.UPC.ES)
|
|
*/
|
|
|
|
/* Function that creates a EPS File from a created pixmap on the current
|
|
* context. Based on the code from Copyright (c) Mark J. Kilgard, 1996.
|
|
* Parameters: name_file, b&w / Color flag, redraw function. The redraw
|
|
* function is needed in order to draw things into the new created pixmap. */
|
|
|
|
/* Copyright (c) Mark J. Kilgard, 1996. */
|
|
|
|
static GLvoid *
|
|
grabPixels(int inColor, unsigned int width, unsigned int height)
|
|
{
|
|
GLvoid *buffer;
|
|
GLint swapbytes, lsbfirst, rowlength;
|
|
GLint skiprows, skippixels, alignment;
|
|
GLenum format;
|
|
unsigned int size;
|
|
|
|
if (inColor) {
|
|
format = GL_RGB;
|
|
size = width * height * 3;
|
|
} else {
|
|
format = GL_LUMINANCE;
|
|
size = width * height * 1;
|
|
}
|
|
|
|
buffer = (GLvoid *) malloc(size);
|
|
if (buffer == NULL)
|
|
return NULL;
|
|
|
|
/* Save current modes. */
|
|
glGetIntegerv(GL_PACK_SWAP_BYTES, &swapbytes);
|
|
glGetIntegerv(GL_PACK_LSB_FIRST, &lsbfirst);
|
|
glGetIntegerv(GL_PACK_ROW_LENGTH, &rowlength);
|
|
glGetIntegerv(GL_PACK_SKIP_ROWS, &skiprows);
|
|
glGetIntegerv(GL_PACK_SKIP_PIXELS, &skippixels);
|
|
glGetIntegerv(GL_PACK_ALIGNMENT, &alignment);
|
|
/* Little endian machines (DEC Alpha for example) could benefit from
|
|
* setting GL_PACK_LSB_FIRST to GL_TRUE instead of GL_FALSE, but this would
|
|
* * * * * * * * * require changing the generated bitmaps too. */
|
|
glPixelStorei(GL_PACK_SWAP_BYTES, GL_FALSE);
|
|
glPixelStorei(GL_PACK_LSB_FIRST, GL_FALSE);
|
|
glPixelStorei(GL_PACK_ROW_LENGTH, 0);
|
|
glPixelStorei(GL_PACK_SKIP_ROWS, 0);
|
|
glPixelStorei(GL_PACK_SKIP_PIXELS, 0);
|
|
glPixelStorei(GL_PACK_ALIGNMENT, 1);
|
|
|
|
/* Actually read the pixels. */
|
|
glReadPixels(0, 0, width, height, format,
|
|
GL_UNSIGNED_BYTE, (GLvoid *) buffer);
|
|
|
|
/* Restore saved modes. */
|
|
glPixelStorei(GL_PACK_SWAP_BYTES, swapbytes);
|
|
glPixelStorei(GL_PACK_LSB_FIRST, lsbfirst);
|
|
glPixelStorei(GL_PACK_ROW_LENGTH, rowlength);
|
|
glPixelStorei(GL_PACK_SKIP_ROWS, skiprows);
|
|
glPixelStorei(GL_PACK_SKIP_PIXELS, skippixels);
|
|
glPixelStorei(GL_PACK_ALIGNMENT, alignment);
|
|
return buffer;
|
|
}
|
|
|
|
|
|
static int
|
|
generateEPS(const char *filename, int inColor,
|
|
unsigned int width, unsigned int height)
|
|
{
|
|
FILE *fp;
|
|
GLvoid *pixels;
|
|
unsigned char *curpix;
|
|
unsigned int components, i;
|
|
int pos;
|
|
unsigned int bitpixel;
|
|
|
|
pixels = grabPixels(inColor, width, height);
|
|
if (pixels == NULL)
|
|
return 1;
|
|
if (inColor)
|
|
components = 3; /* Red, green, blue. */
|
|
else
|
|
components = 1; /* Luminance. */
|
|
|
|
fp = fopen(filename, "w");
|
|
if (fp == NULL) {
|
|
return 2;
|
|
}
|
|
(void) fprintf(fp, "%%!PS-Adobe-2.0 EPSF-1.2\n");
|
|
(void) fprintf(fp, "%%%%Creator: OpenGL pixmap render output\n");
|
|
(void) fprintf(fp, "%%%%BoundingBox: 0 0 %d %d\n", width, height);
|
|
(void) fprintf(fp, "%%%%EndComments\n");
|
|
|
|
i = (((width * height) + 7) / 8) / 40; /* # of lines, 40 bytes per
|
|
* line */
|
|
(void) fprintf(fp, "%%%%BeginPreview: %d %d %d %d\n%%", width, height, 1,
|
|
i);
|
|
pos = 0;
|
|
curpix = (unsigned char *) pixels;
|
|
for (i = 0; i < width * height * components;) {
|
|
bitpixel = 0;
|
|
if (inColor) {
|
|
double pix = 0;
|
|
|
|
pix = 0.30 * (double) curpix[i] + 0.59 * (double) curpix[i + 1] +
|
|
0.11 * (double) curpix[i + 2];
|
|
i += 3;
|
|
if (pix > 127.0)
|
|
bitpixel |= 0x80;
|
|
pix = 0.30 * (double) curpix[i] + 0.59 * (double) curpix[i + 1] +
|
|
0.11 * (double) curpix[i + 2];
|
|
i += 3;
|
|
if (pix > 127.0)
|
|
bitpixel |= 0x40;
|
|
pix = 0.30 * (double) curpix[i] + 0.59 * (double) curpix[i + 1] +
|
|
0.11 * (double) curpix[i + 2];
|
|
i += 3;
|
|
if (pix > 127.0)
|
|
bitpixel |= 0x20;
|
|
pix = 0.30 * (double) curpix[i] + 0.59 * (double) curpix[i + 1] +
|
|
0.11 * (double) curpix[i + 2];
|
|
i += 3;
|
|
if (pix > 127.0)
|
|
bitpixel |= 0x10;
|
|
pix = 0.30 * (double) curpix[i] + 0.59 * (double) curpix[i + 1] +
|
|
0.11 * (double) curpix[i + 2];
|
|
i += 3;
|
|
if (pix > 127.0)
|
|
bitpixel |= 0x08;
|
|
pix = 0.30 * (double) curpix[i] + 0.59 * (double) curpix[i + 1] +
|
|
0.11 * (double) curpix[i + 2];
|
|
i += 3;
|
|
if (pix > 127.0)
|
|
bitpixel |= 0x04;
|
|
pix = 0.30 * (double) curpix[i] + 0.59 * (double) curpix[i + 1] +
|
|
0.11 * (double) curpix[i + 2];
|
|
i += 3;
|
|
if (pix > 127.0)
|
|
bitpixel |= 0x02;
|
|
pix = 0.30 * (double) curpix[i] + 0.59 * (double) curpix[i + 1] +
|
|
0.11 * (double) curpix[i + 2];
|
|
i += 3;
|
|
if (pix > 127.0)
|
|
bitpixel |= 0x01;
|
|
} else {
|
|
if (curpix[i++] > 0x7f)
|
|
bitpixel |= 0x80;
|
|
if (curpix[i++] > 0x7f)
|
|
bitpixel |= 0x40;
|
|
if (curpix[i++] > 0x7f)
|
|
bitpixel |= 0x20;
|
|
if (curpix[i++] > 0x7f)
|
|
bitpixel |= 0x10;
|
|
if (curpix[i++] > 0x7f)
|
|
bitpixel |= 0x08;
|
|
if (curpix[i++] > 0x7f)
|
|
bitpixel |= 0x04;
|
|
if (curpix[i++] > 0x7f)
|
|
bitpixel |= 0x02;
|
|
if (curpix[i++] > 0x7f)
|
|
bitpixel |= 0x01;
|
|
}
|
|
(void) fprintf(fp, "%02x", bitpixel);
|
|
if (++pos >= 40) {
|
|
(void) fprintf(fp, "\n%%");
|
|
pos = 0;
|
|
}
|
|
}
|
|
if (pos)
|
|
(void) fprintf(fp, "\n%%%%EndPreview\n");
|
|
else
|
|
(void) fprintf(fp, "%%EndPreview\n");
|
|
|
|
(void) fprintf(fp, "gsave\n");
|
|
(void) fprintf(fp, "/bwproc {\n");
|
|
(void) fprintf(fp, " rgbproc\n");
|
|
(void) fprintf(fp, " dup length 3 idiv string 0 3 0\n");
|
|
(void) fprintf(fp, " 5 -1 roll {\n");
|
|
(void) fprintf(fp, " add 2 1 roll 1 sub dup 0 eq\n");
|
|
(void) fprintf(fp, " { pop 3 idiv 3 -1 roll dup 4 -1 roll dup\n");
|
|
(void) fprintf(fp, " 3 1 roll 5 -1 roll put 1 add 3 0 }\n");
|
|
(void) fprintf(fp, " { 2 1 roll } ifelse\n");
|
|
(void) fprintf(fp, " } forall\n");
|
|
(void) fprintf(fp, " pop pop pop\n");
|
|
(void) fprintf(fp, "} def\n");
|
|
(void) fprintf(fp, "systemdict /colorimage known not {\n");
|
|
(void) fprintf(fp, " /colorimage {\n");
|
|
(void) fprintf(fp, " pop\n");
|
|
(void) fprintf(fp, " pop\n");
|
|
(void) fprintf(fp, " /rgbproc exch def\n");
|
|
(void) fprintf(fp, " { bwproc } image\n");
|
|
(void) fprintf(fp, " } def\n");
|
|
(void) fprintf(fp, "} if\n");
|
|
(void) fprintf(fp, "/picstr %d string def\n", width * components);
|
|
(void) fprintf(fp, "%d %d scale\n", width, height);
|
|
(void) fprintf(fp, "%d %d %d\n", width, height, 8);
|
|
(void) fprintf(fp, "[%d 0 0 %d 0 0]\n", width, height);
|
|
(void) fprintf(fp, "{currentfile picstr readhexstring pop}\n");
|
|
(void) fprintf(fp, "false %d\n", components);
|
|
(void) fprintf(fp, "colorimage\n");
|
|
|
|
curpix = (unsigned char *) pixels;
|
|
pos = 0;
|
|
for (i = width * height * components; i != 0; i--) {
|
|
(void) fprintf(fp, "%02hx", *curpix++);
|
|
if (++pos >= 40) {
|
|
(void) fprintf(fp, "\n");
|
|
pos = 0;
|
|
}
|
|
}
|
|
if (pos)
|
|
(void) fprintf(fp, "\n");
|
|
|
|
(void) fprintf(fp, "grestore\n");
|
|
free(pixels);
|
|
if (fclose(fp) != 0)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* int Togl_DumpToEpsFile( const Togl *togl, const char *filename, int inColor,
|
|
* void (*user_redraw)(void)) */
|
|
/* changed by GG */
|
|
int
|
|
Togl_DumpToEpsFile(const Togl *togl, const char *filename,
|
|
int inColor, void (*user_redraw) (const Togl *))
|
|
{
|
|
Bool using_mesa = False;
|
|
|
|
#if 0
|
|
Pixmap eps_pixmap;
|
|
GLXPixmap eps_glxpixmap;
|
|
XVisualInfo *vi = togl->VisInfo;
|
|
Window win = Tk_WindowId(togl->TkWin);
|
|
#endif
|
|
int retval;
|
|
unsigned int width = togl->Width, height = togl->Height;
|
|
|
|
#if defined(TOGL_X11)
|
|
Display *dpy = Tk_Display(togl->TkWin);
|
|
int scrnum = Tk_ScreenNumber(togl->TkWin);
|
|
|
|
if (strstr(glXQueryServerString(dpy, scrnum, GLX_VERSION), "Mesa"))
|
|
using_mesa = True;
|
|
else
|
|
#endif /* TOGL_X11 */
|
|
using_mesa = False;
|
|
/* I don't use Pixmap do drawn into, because the code should link with Mesa
|
|
* libraries and OpenGL libraries, and the which library we use at run time
|
|
* should not matter, but the name of the calls differs one from another:
|
|
* MesaGl: glXCreateGLXPixmapMESA( dpy, vi, eps_pixmap,
|
|
* Tk_Colormap(togl->TkWin)) OpenGl: glXCreateGLXPixmap( dpy, vi,
|
|
* eps_pixmap); instead of this I read direct from back buffer of the
|
|
* screeen. */
|
|
#if 0
|
|
eps_pixmap = XCreatePixmap(dpy, win, width, height, vi->depth);
|
|
if (using_mesa)
|
|
eps_glxpixmap =
|
|
glXCreateGLXPixmapMESA(dpy, vi, eps_pixmap,
|
|
Tk_Colormap(togl->TkWin));
|
|
else
|
|
eps_glxpixmap = glXCreateGLXPixmap(dpy, vi, eps_pixmap);
|
|
|
|
glXMakeCurrent(dpy, eps_glxpixmap, togl->GlCtx);
|
|
user_redraw();
|
|
#endif
|
|
if (!togl->RgbaFlag) {
|
|
|
|
#if defined(TOGL_WGL)
|
|
/* Due to the lack of a unique inverse mapping from the frame buffer to
|
|
* the logical palette we need a translation map from the complete
|
|
* logical palette. */
|
|
{
|
|
int n, i;
|
|
TkWinColormap *cmap = (TkWinColormap *) Tk_Colormap(togl->TkWin);
|
|
LPPALETTEENTRY entry =
|
|
malloc(togl->EpsMapSize * sizeof (PALETTEENTRY));
|
|
n = GetPaletteEntries(cmap->palette, 0, togl->EpsMapSize, entry);
|
|
for (i = 0; i < n; i++) {
|
|
togl->EpsRedMap[i] = (GLfloat) (entry[i].peRed / 255.0);
|
|
togl->EpsGreenMap[i] = (GLfloat) (entry[i].peGreen / 255.0);
|
|
togl->EpsBlueMap[i] = (GLfloat) (entry[i].peBlue / 255.0);
|
|
}
|
|
free(entry);
|
|
}
|
|
#endif /* TOGL_WGL */
|
|
|
|
glPixelMapfv(GL_PIXEL_MAP_I_TO_R, togl->EpsMapSize, togl->EpsRedMap);
|
|
glPixelMapfv(GL_PIXEL_MAP_I_TO_G, togl->EpsMapSize, togl->EpsGreenMap);
|
|
glPixelMapfv(GL_PIXEL_MAP_I_TO_B, togl->EpsMapSize, togl->EpsBlueMap);
|
|
}
|
|
/* user_redraw(); */
|
|
user_redraw(togl); /* changed by GG */
|
|
/* glReadBuffer( GL_FRONT); */
|
|
/* by default it read GL_BACK in double buffer mode */
|
|
glFlush();
|
|
retval = generateEPS(filename, inColor, width, height);
|
|
#if 0
|
|
glXMakeCurrent(dpy, win, togl->GlCtx);
|
|
glXDestroyGLXPixmap(dpy, eps_glxpixmap);
|
|
XFreePixmap(dpy, eps_pixmap);
|
|
#endif
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Full screen stereo for SGI graphics
|
|
* Contributed by Ben Evans (Ben.Evans@anusf.anu.edu.au)
|
|
* This code was based on SGI's /usr/share/src/OpenGL/teach/stereo
|
|
*/
|
|
|
|
#if defined(__sgi)
|
|
|
|
static struct stereoStateRec
|
|
{
|
|
Bool useSGIStereo;
|
|
Display *currentDisplay;
|
|
Window currentWindow;
|
|
GLXContext currentContext;
|
|
GLenum currentDrawBuffer;
|
|
int currentStereoBuffer;
|
|
Bool enabled;
|
|
char *stereoCommand;
|
|
char *restoreCommand;
|
|
} stereo;
|
|
|
|
/* call instead of glDrawBuffer */
|
|
void
|
|
Togl_OldStereoDrawBuffer(GLenum mode)
|
|
{
|
|
if (stereo.useSGIStereo) {
|
|
stereo.currentDrawBuffer = mode;
|
|
switch (mode) {
|
|
case GL_FRONT:
|
|
case GL_BACK:
|
|
case GL_FRONT_AND_BACK:
|
|
/*
|
|
** Simultaneous drawing to both left and right buffers isn't
|
|
** really possible if we don't have a stereo capable visual.
|
|
** For now just fall through and use the left buffer.
|
|
*/
|
|
case GL_LEFT:
|
|
case GL_FRONT_LEFT:
|
|
case GL_BACK_LEFT:
|
|
stereo.currentStereoBuffer = STEREO_BUFFER_LEFT;
|
|
break;
|
|
case GL_RIGHT:
|
|
case GL_FRONT_RIGHT:
|
|
stereo.currentStereoBuffer = STEREO_BUFFER_RIGHT;
|
|
mode = GL_FRONT;
|
|
break;
|
|
case GL_BACK_RIGHT:
|
|
stereo.currentStereoBuffer = STEREO_BUFFER_RIGHT;
|
|
mode = GL_BACK;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (stereo.currentDisplay && stereo.currentWindow) {
|
|
glXWaitGL(); /* sync with GL command stream before calling X
|
|
*/
|
|
XSGISetStereoBuffer(stereo.currentDisplay,
|
|
stereo.currentWindow, stereo.currentStereoBuffer);
|
|
glXWaitX(); /* sync with X command stream before calling GL
|
|
*/
|
|
}
|
|
}
|
|
glDrawBuffer(mode);
|
|
}
|
|
|
|
/* call instead of glClear */
|
|
void
|
|
Togl_OldStereoClear(GLbitfield mask)
|
|
{
|
|
GLenum drawBuffer;
|
|
|
|
if (stereo.useSGIStereo) {
|
|
drawBuffer = stereo.currentDrawBuffer;
|
|
switch (drawBuffer) {
|
|
case GL_FRONT:
|
|
Togl_OldStereoDrawBuffer(GL_FRONT_RIGHT);
|
|
glClear(mask);
|
|
Togl_OldStereoDrawBuffer(drawBuffer);
|
|
break;
|
|
case GL_BACK:
|
|
Togl_OldStereoDrawBuffer(GL_BACK_RIGHT);
|
|
glClear(mask);
|
|
Togl_OldStereoDrawBuffer(drawBuffer);
|
|
break;
|
|
case GL_FRONT_AND_BACK:
|
|
Togl_OldStereoDrawBuffer(GL_RIGHT);
|
|
glClear(mask);
|
|
Togl_OldStereoDrawBuffer(drawBuffer);
|
|
break;
|
|
case GL_LEFT:
|
|
case GL_FRONT_LEFT:
|
|
case GL_BACK_LEFT:
|
|
case GL_RIGHT:
|
|
case GL_FRONT_RIGHT:
|
|
case GL_BACK_RIGHT:
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
glClear(mask);
|
|
}
|
|
|
|
static void
|
|
oldStereoMakeCurrent(Display *dpy, Window win, GLXContext ctx)
|
|
{
|
|
|
|
if (dpy && (dpy != stereo.currentDisplay)) {
|
|
int event, error;
|
|
|
|
/* Make sure new Display supports SGIStereo */
|
|
if (XSGIStereoQueryExtension(dpy, &event, &error) == False) {
|
|
dpy = NULL;
|
|
}
|
|
}
|
|
if (dpy && win && (win != stereo.currentWindow)) {
|
|
/* Make sure new Window supports SGIStereo */
|
|
if (XSGIQueryStereoMode(dpy, win) == X_STEREO_UNSUPPORTED) {
|
|
win = None;
|
|
}
|
|
}
|
|
if (ctx && (ctx != stereo.currentContext)) {
|
|
GLint drawBuffer;
|
|
|
|
glGetIntegerv(GL_DRAW_BUFFER, &drawBuffer);
|
|
Togl_OldStereoDrawBuffer((GLenum) drawBuffer);
|
|
}
|
|
stereo.currentDisplay = dpy;
|
|
stereo.currentWindow = win;
|
|
stereo.currentContext = ctx;
|
|
}
|
|
|
|
|
|
/* call before using stereo */
|
|
static void
|
|
oldStereoInit(Togl *togl, int stereoEnabled)
|
|
{
|
|
stereo.useSGIStereo = stereoEnabled;
|
|
stereo.currentDisplay = NULL;
|
|
stereo.currentWindow = None;
|
|
stereo.currentContext = NULL;
|
|
stereo.currentDrawBuffer = GL_NONE;
|
|
stereo.currentStereoBuffer = STEREO_BUFFER_NONE;
|
|
stereo.enabled = False;
|
|
}
|
|
|
|
#endif /* __sgi STEREO */
|
|
|
|
|
|
void
|
|
Togl_StereoFrustum(GLfloat left, GLfloat right, GLfloat bottom, GLfloat top,
|
|
GLfloat zNear, GLfloat zFar, GLfloat eyeDist, GLfloat eyeOffset)
|
|
{
|
|
GLfloat eyeShift = (eyeDist - zNear) * (eyeOffset / eyeDist);
|
|
|
|
glFrustum(left + eyeShift, right + eyeShift, bottom, top, zNear, zFar);
|
|
glTranslatef(-eyeShift, 0, 0);
|
|
}
|
|
|
|
|
|
#ifdef TOGL_AGL_CLASSIC
|
|
/* needed to make shared library on Mac with CodeWarrior; should be overridden
|
|
* by user app */
|
|
/*
|
|
* int main(int argc, char *argv[]) { return -1; } */
|
|
|
|
/* the following code is borrowed from tkMacAppInit.c */
|
|
|
|
/*
|
|
*----------------------------------------------------------------------
|
|
*
|
|
* MacintoshInit --
|
|
*
|
|
* This procedure calls Mac specific initilization calls. Most of
|
|
* these calls must be made as soon as possible in the startup
|
|
* process.
|
|
*
|
|
* Results:
|
|
* Returns TCL_OK if everything went fine. If it didn't the
|
|
* application should probably fail.
|
|
*
|
|
* Side effects:
|
|
* Inits the application.
|
|
*
|
|
*----------------------------------------------------------------------
|
|
*/
|
|
|
|
int
|
|
Togl_MacInit(void)
|
|
{
|
|
int i;
|
|
long result, mask = 0x0700; /* mask = system 7.x */
|
|
|
|
# if GENERATING68K && !GENERATINGCFM
|
|
SetApplLimit(GetApplLimit() - (TK_MAC_68K_STACK_GROWTH));
|
|
# endif
|
|
MaxApplZone();
|
|
for (i = 0; i < 4; i++) {
|
|
(void) MoreMasters();
|
|
}
|
|
|
|
/*
|
|
* Tk needs us to set the qd pointer it uses. This is needed
|
|
* so Tk doesn't have to assume the availability of the qd global
|
|
* variable. Which in turn allows Tk to be used in code resources.
|
|
*/
|
|
tcl_macQdPtr = &qd;
|
|
|
|
/*
|
|
* If appearance is present, then register Tk as an Appearance client
|
|
* This means that the mapping from non-Appearance to Appearance cdefs
|
|
* will be done for Tk regardless of the setting in the Appearance
|
|
* control panel.
|
|
*/
|
|
if (TkMacHaveAppearance()) {
|
|
RegisterAppearanceClient();
|
|
}
|
|
|
|
InitGraf(&tcl_macQdPtr->thePort);
|
|
InitFonts();
|
|
InitWindows();
|
|
InitMenus();
|
|
InitDialogs((long) NULL);
|
|
InitCursor();
|
|
|
|
/*
|
|
* Make sure we are running on system 7 or higher
|
|
*/
|
|
if ((NGetTrapAddress(_Gestalt, ToolTrap) ==
|
|
NGetTrapAddress(_Unimplemented, ToolTrap))
|
|
|| (((Gestalt(gestaltSystemVersion, &result) != noErr)
|
|
|| (result < mask)))) {
|
|
panic("Tcl/Tk requires System 7 or higher.");
|
|
}
|
|
|
|
/*
|
|
* Make sure we have color quick draw
|
|
* (this means we can't run on 68000 macs)
|
|
*/
|
|
if (((Gestalt(gestaltQuickdrawVersion, &result) != noErr)
|
|
|| (result < gestalt32BitQD13))) {
|
|
panic("Tk requires Color QuickDraw.");
|
|
}
|
|
|
|
FlushEvents(everyEvent, 0);
|
|
SetEventMask(everyEvent);
|
|
|
|
Tcl_MacSetEventProc(TkMacConvertEvent);
|
|
return TCL_OK;
|
|
}
|
|
|
|
int
|
|
Togl_MacSetupMainInterp(Tcl_Interp *interp)
|
|
{
|
|
TkMacInitAppleEvents(interp);
|
|
TkMacInitMenus(interp);
|
|
return TCL_OK;
|
|
}
|
|
|
|
#endif /* TOGL_AGL_CLASSIC */
|