mirror of
https://github.com/UberGames/GtkRadiant.git
synced 2025-01-22 17:31:39 +00:00
1f99ab9e81
This commit: 1. Reverts to using GtkR-deps-1.6-3.zip (instead of GtkR-deps-1.6-4.zip). Some font-related DLLs are therefore no longer used with this patch. 2. Because of the above #1, now using the really old OpenGL font rendering code on Windows systems. This is controlled via "#ifdef _WIN32" blocks. Linux still uses the new and improved OpenGL font rendering. Note that this old font rendering that Windows now uses (err, has used in the past too) makes use of gdk_gl_font_use_pango_font(), which is really old and crufty. So, with this commit Radiant is fully working (modulo bugs) on Linux and Windows, and OpenGL fonts work too. I WOULD NOT SUGGEST MERGING THIS CHANGE INTO TRUNK. THE CORRECT APPROACH IS TO INSTEAD FIX THE DLL LIBRARY DEPENDENCIES IN TRUNK. I AM ONLY COMMITTING THIS SO THAT I CAN CONTINUE FIXING BUGS IN RADIANT IN THE MEANTIME. There are still a zillion bugs on Windows. For example, I can't even save a .map file at this point. That is my next task, to fix saving of .map files. git-svn-id: svn://svn.icculus.org/gtkradiant/GtkRadiant/branches/Rambetter-temp-fixes@350 8a3a26a2-13c4-0310-b231-cf6edde360e5
507 lines
14 KiB
C++
507 lines
14 KiB
C++
/*
|
|
Copyright (c) 2001, Loki software, inc.
|
|
All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without modification,
|
|
are permitted provided that the following conditions are met:
|
|
|
|
Redistributions of source code must retain the above copyright notice, this list
|
|
of conditions and the following disclaimer.
|
|
|
|
Redistributions in binary form must reproduce the above copyright notice, this
|
|
list of conditions and the following disclaimer in the documentation and/or
|
|
other materials provided with the distribution.
|
|
|
|
Neither the name of Loki software nor the names of its contributors may be used
|
|
to endorse or promote products derived from this software without specific prior
|
|
written permission.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS ``AS IS''
|
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR ANY
|
|
DIRECT,INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
// OpenGL widget based on GtkGLExt
|
|
|
|
#include "stdafx.h"
|
|
|
|
#include <gtk/gtkgl.h>
|
|
|
|
#ifndef _WIN32
|
|
#include <pango/pangoft2.h>
|
|
#endif
|
|
|
|
#include "glwidget.h"
|
|
#include "qgl.h"
|
|
|
|
typedef int* attribs_t;
|
|
typedef const attribs_t* configs_iterator;
|
|
|
|
int config_rgba32[] = {
|
|
GDK_GL_RGBA,
|
|
GDK_GL_DOUBLEBUFFER,
|
|
GDK_GL_RED_SIZE, 8,
|
|
GDK_GL_BLUE_SIZE, 8,
|
|
GDK_GL_GREEN_SIZE, 8,
|
|
GDK_GL_ALPHA_SIZE, 8,
|
|
GDK_GL_ATTRIB_LIST_NONE,
|
|
};
|
|
|
|
int config_rgba[] = {
|
|
GDK_GL_RGBA,
|
|
GDK_GL_DOUBLEBUFFER,
|
|
GDK_GL_RED_SIZE, 1,
|
|
GDK_GL_BLUE_SIZE, 1,
|
|
GDK_GL_GREEN_SIZE, 1,
|
|
GDK_GL_ALPHA_SIZE, 1,
|
|
GDK_GL_ATTRIB_LIST_NONE,
|
|
};
|
|
|
|
const attribs_t configs[] = {
|
|
config_rgba32,
|
|
config_rgba,
|
|
};
|
|
|
|
GdkGLConfig* glconfig_new()
|
|
{
|
|
GdkGLConfig* glconfig = NULL;
|
|
|
|
for(configs_iterator i = configs, end = configs + 2; i != end && glconfig == NULL; ++i)
|
|
{
|
|
glconfig = gdk_gl_config_new(*i);
|
|
}
|
|
|
|
if(glconfig == NULL)
|
|
{
|
|
return gdk_gl_config_new_by_mode((GdkGLConfigMode)(GDK_GL_MODE_RGBA | GDK_GL_MODE_DOUBLE));
|
|
}
|
|
|
|
return glconfig;
|
|
}
|
|
|
|
int config_rgba32_depth32[] = {
|
|
GDK_GL_RGBA,
|
|
GDK_GL_DOUBLEBUFFER,
|
|
GDK_GL_RED_SIZE, 8,
|
|
GDK_GL_BLUE_SIZE, 8,
|
|
GDK_GL_GREEN_SIZE, 8,
|
|
GDK_GL_ALPHA_SIZE, 8,
|
|
GDK_GL_DEPTH_SIZE, 32,
|
|
GDK_GL_ATTRIB_LIST_NONE,
|
|
};
|
|
|
|
int config_rgba32_depth24[] = {
|
|
GDK_GL_RGBA,
|
|
GDK_GL_DOUBLEBUFFER,
|
|
GDK_GL_RED_SIZE, 8,
|
|
GDK_GL_BLUE_SIZE, 8,
|
|
GDK_GL_GREEN_SIZE, 8,
|
|
GDK_GL_ALPHA_SIZE, 8,
|
|
GDK_GL_DEPTH_SIZE, 24,
|
|
GDK_GL_ATTRIB_LIST_NONE,
|
|
};
|
|
|
|
int config_rgba32_depth16[] = {
|
|
GDK_GL_RGBA,
|
|
GDK_GL_DOUBLEBUFFER,
|
|
GDK_GL_RED_SIZE, 8,
|
|
GDK_GL_BLUE_SIZE, 8,
|
|
GDK_GL_GREEN_SIZE, 8,
|
|
GDK_GL_ALPHA_SIZE, 8,
|
|
GDK_GL_DEPTH_SIZE, 16,
|
|
GDK_GL_ATTRIB_LIST_NONE,
|
|
};
|
|
|
|
int config_rgba32_depth[] = {
|
|
GDK_GL_RGBA,
|
|
GDK_GL_DOUBLEBUFFER,
|
|
GDK_GL_RED_SIZE, 8,
|
|
GDK_GL_BLUE_SIZE, 8,
|
|
GDK_GL_GREEN_SIZE, 8,
|
|
GDK_GL_ALPHA_SIZE, 8,
|
|
GDK_GL_DEPTH_SIZE, 1,
|
|
GDK_GL_ATTRIB_LIST_NONE,
|
|
};
|
|
|
|
int config_rgba_depth16[] = {
|
|
GDK_GL_RGBA,
|
|
GDK_GL_DOUBLEBUFFER,
|
|
GDK_GL_RED_SIZE, 1,
|
|
GDK_GL_BLUE_SIZE, 1,
|
|
GDK_GL_GREEN_SIZE, 1,
|
|
GDK_GL_ALPHA_SIZE, 1,
|
|
GDK_GL_DEPTH_SIZE, 16,
|
|
GDK_GL_ATTRIB_LIST_NONE,
|
|
};
|
|
|
|
int config_rgba_depth[] = {
|
|
GDK_GL_RGBA,
|
|
GDK_GL_DOUBLEBUFFER,
|
|
GDK_GL_RED_SIZE, 1,
|
|
GDK_GL_BLUE_SIZE, 1,
|
|
GDK_GL_GREEN_SIZE, 1,
|
|
GDK_GL_ALPHA_SIZE, 1,
|
|
GDK_GL_DEPTH_SIZE, 1,
|
|
GDK_GL_ATTRIB_LIST_NONE,
|
|
};
|
|
|
|
const attribs_t configs_with_depth[] =
|
|
{
|
|
config_rgba32_depth32,
|
|
config_rgba32_depth24,
|
|
config_rgba32_depth16,
|
|
config_rgba32_depth,
|
|
config_rgba_depth16,
|
|
config_rgba_depth,
|
|
};
|
|
|
|
GdkGLConfig* glconfig_new_with_depth()
|
|
{
|
|
GdkGLConfig* glconfig = NULL;
|
|
|
|
for(configs_iterator i = configs_with_depth, end = configs_with_depth + 6; i != end && glconfig == NULL; ++i)
|
|
{
|
|
glconfig = gdk_gl_config_new(*i);
|
|
}
|
|
|
|
if(glconfig == NULL)
|
|
{
|
|
return gdk_gl_config_new_by_mode((GdkGLConfigMode)(GDK_GL_MODE_RGBA | GDK_GL_MODE_DOUBLE | GDK_GL_MODE_DEPTH));
|
|
}
|
|
|
|
return glconfig;
|
|
}
|
|
|
|
GtkWidget* WINAPI gtk_glwidget_new (gboolean zbuffer, GtkWidget* share)
|
|
{
|
|
GtkWidget* drawing_area = gtk_drawing_area_new();
|
|
GdkGLConfig* glconfig = (zbuffer) ? glconfig_new_with_depth() : glconfig_new();
|
|
GdkGLContext* shared_context = (share) ? gtk_widget_get_gl_context(share) : NULL;
|
|
|
|
gtk_widget_set_gl_capability (drawing_area, glconfig, shared_context, TRUE, GDK_GL_RGBA_TYPE);
|
|
|
|
return drawing_area;
|
|
}
|
|
|
|
void WINAPI gtk_glwidget_destroy_context (GtkWidget *widget)
|
|
{
|
|
}
|
|
|
|
void WINAPI gtk_glwidget_create_context (GtkWidget *widget)
|
|
{
|
|
}
|
|
|
|
void WINAPI gtk_glwidget_swap_buffers (GtkWidget *widget)
|
|
{
|
|
GdkGLDrawable *gldrawable = gtk_widget_get_gl_drawable (widget);
|
|
gdk_gl_drawable_swap_buffers (gldrawable);
|
|
}
|
|
|
|
gboolean WINAPI gtk_glwidget_make_current (GtkWidget *widget)
|
|
{
|
|
GdkGLContext *glcontext = gtk_widget_get_gl_context (widget);
|
|
GdkGLDrawable *gldrawable = gtk_widget_get_gl_drawable (widget);
|
|
return gdk_gl_drawable_gl_begin (gldrawable, glcontext);
|
|
}
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
GLuint font_list_base;
|
|
static gchar font_string[] = "courier 8";
|
|
static gint font_height;
|
|
static int font_created = 0;
|
|
|
|
#else
|
|
|
|
// Think about rewriting this font stuff to use OpenGL display lists and glBitmap().
|
|
// Bit maps together with display lists may offer a performance increase, but
|
|
// they would not allow antialiased fonts.
|
|
static const char font_string[] = "Monospace";
|
|
static const int font_height = 10;
|
|
static int font_ascent = -1;
|
|
static int font_descent = -1;
|
|
static int y_offset_bitmap_render_pango_units = -1;
|
|
static PangoContext *ft2_context = NULL;
|
|
static int _debug_font_created = 0;
|
|
|
|
#endif
|
|
|
|
|
|
// Units are pixels. Returns a positive value [most likely].
|
|
int gtk_glwidget_font_ascent()
|
|
{
|
|
#ifdef _WIN32
|
|
|
|
return 6; // Approximation.
|
|
|
|
#else
|
|
|
|
if (!_debug_font_created) {
|
|
Error("Programming error: gtk_glwidget_font_ascent() called but font does not exist; "
|
|
"you should have called gtk_glwidget_create_font() first");
|
|
}
|
|
|
|
return font_ascent;
|
|
|
|
#endif
|
|
}
|
|
|
|
// Units are pixels. Returns a positive value [most likely].
|
|
int gtk_glwidget_font_descent()
|
|
{
|
|
#ifdef _WIN32
|
|
|
|
return 0; // Approximation.
|
|
|
|
#else
|
|
|
|
if (!_debug_font_created) {
|
|
Error("Programming error: gtk_glwidget_font_descent() called but font does not exist; "
|
|
"you should have called gtk_glwidget_create_font() first");
|
|
}
|
|
|
|
return font_descent;
|
|
|
|
#endif
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
|
|
void gtk_glwidget_create_font_win_internal()
|
|
{
|
|
if (font_created) return;
|
|
font_created = 1;
|
|
|
|
PangoFontDescription *font_desc;
|
|
PangoFont *font;
|
|
PangoFontMetrics *font_metrics;
|
|
|
|
font_list_base = qglGenLists (256);
|
|
|
|
font_desc = pango_font_description_from_string (font_string);
|
|
|
|
font = gdk_gl_font_use_pango_font (font_desc, 0, 256, font_list_base);
|
|
|
|
if(font != NULL)
|
|
{
|
|
font_metrics = pango_font_get_metrics (font, NULL);
|
|
|
|
font_height = pango_font_metrics_get_ascent (font_metrics) +
|
|
pango_font_metrics_get_descent (font_metrics);
|
|
font_height = PANGO_PIXELS (font_height);
|
|
|
|
pango_font_metrics_unref (font_metrics);
|
|
}
|
|
|
|
pango_font_description_free (font_desc);
|
|
}
|
|
|
|
#endif
|
|
|
|
void gtk_glwidget_create_font()
|
|
{
|
|
#ifdef _WIN32
|
|
|
|
// Do nothing.
|
|
|
|
#else
|
|
|
|
PangoFontDescription *font_desc;
|
|
PangoLayout *layout;
|
|
PangoRectangle log_rect;
|
|
int font_ascent_pango_units;
|
|
int font_descent_pango_units;
|
|
|
|
if (_debug_font_created) {
|
|
Error("Programming error: gtk_glwidget_create_font() was already called; "
|
|
"you must call gtk_glwidget_destroy_font() before creating font again");
|
|
}
|
|
_debug_font_created = 1;
|
|
|
|
// This call is deprecated so we'll have to fix it sometime.
|
|
ft2_context = pango_ft2_get_context(72, 72);
|
|
|
|
font_desc = pango_font_description_from_string(font_string);
|
|
pango_font_description_set_size(font_desc, font_height * PANGO_SCALE);
|
|
pango_context_set_font_description(ft2_context, font_desc);
|
|
pango_font_description_free(font_desc);
|
|
|
|
layout = pango_layout_new(ft2_context);
|
|
|
|
#ifdef _WIN32
|
|
|
|
PangoLayoutIter *iter;
|
|
iter = pango_layout_get_iter(layout);
|
|
font_ascent_pango_units = pango_layout_iter_get_baseline(iter);
|
|
pango_layout_iter_free(iter);
|
|
|
|
#else
|
|
|
|
// I don't believe that's standard preprocessor syntax?
|
|
#if !PANGO_VERSION_CHECK(1,22,0)
|
|
PangoLayoutIter *iter;
|
|
iter = pango_layout_get_iter(layout);
|
|
font_ascent_pango_units = pango_layout_iter_get_baseline(iter);
|
|
pango_layout_iter_free(iter);
|
|
#else
|
|
font_ascent_pango_units = pango_layout_get_baseline(layout);
|
|
#endif
|
|
|
|
#endif
|
|
|
|
pango_layout_get_extents(layout, NULL, &log_rect);
|
|
g_object_unref(G_OBJECT(layout));
|
|
font_descent_pango_units = log_rect.height - font_ascent_pango_units;
|
|
|
|
font_ascent = PANGO_PIXELS_CEIL(font_ascent_pango_units);
|
|
font_descent = PANGO_PIXELS_CEIL(font_descent_pango_units);
|
|
y_offset_bitmap_render_pango_units = (font_ascent * PANGO_SCALE) - font_ascent_pango_units;
|
|
|
|
#endif
|
|
}
|
|
|
|
void gtk_glwidget_destroy_font()
|
|
{
|
|
#ifdef _WIN32
|
|
|
|
// Do nothing.
|
|
|
|
#else
|
|
|
|
if (!_debug_font_created) {
|
|
Error("Programming error: gtk_glwidget_destroy_font() called when font "
|
|
"does not exist");
|
|
}
|
|
|
|
font_ascent = -1;
|
|
font_descent = -1;
|
|
y_offset_bitmap_render_pango_units = -1;
|
|
g_object_unref(G_OBJECT(ft2_context));
|
|
_debug_font_created = 0;
|
|
|
|
#endif
|
|
}
|
|
|
|
|
|
// Renders the input text at the current location with the current color.
|
|
// The X position of the current location is used to place the left edge of the text image,
|
|
// where the text image bounds are defined as the logical extents of the line of text.
|
|
// The Y position of the current location is used to place the bottom of the text image.
|
|
// You should offset the Y position by the amount returned by gtk_glwidget_font_descent()
|
|
// if you want to place the baseline of the text image at the current Y position.
|
|
// Note: A problem with this function is that if the lower left corner of the text falls
|
|
// just a hair outside of the viewport (meaning the current raster position is invalid),
|
|
// then no text will be rendered. The solution to this is a very hacky one. You can search
|
|
// Google for "glDrawPixels clipping".
|
|
void gtk_glwidget_print_string(const char *s)
|
|
{
|
|
#ifdef _WIN32
|
|
|
|
gtk_glwidget_create_font_win_internal();
|
|
qglListBase(font_list_base);
|
|
qglCallLists(strlen(s), GL_UNSIGNED_BYTE, (unsigned char *)s);
|
|
|
|
#else
|
|
|
|
// The idea for this code initially came from the font-pangoft2.c example that comes with GtkGLExt.
|
|
|
|
PangoLayout *layout;
|
|
PangoRectangle log_rect;
|
|
FT_Bitmap bitmap;
|
|
unsigned char *begin_bitmap_buffer;
|
|
GLfloat color[4];
|
|
GLint previous_unpack_alignment;
|
|
GLboolean previous_blend_enabled;
|
|
GLint previous_blend_func_src;
|
|
GLint previous_blend_func_dst;
|
|
GLfloat previous_red_bias;
|
|
GLfloat previous_green_bias;
|
|
GLfloat previous_blue_bias;
|
|
GLfloat previous_alpha_scale;
|
|
|
|
if (!_debug_font_created) {
|
|
Error("Programming error: gtk_glwidget_print_string() called but font does not exist; "
|
|
"you should have called gtk_glwidget_create_font() first");
|
|
}
|
|
|
|
layout = pango_layout_new(ft2_context);
|
|
pango_layout_set_width(layout, -1); // -1 no wrapping. All text on one line.
|
|
pango_layout_set_text(layout, s, -1); // -1 null-terminated string.
|
|
pango_layout_get_extents(layout, NULL, &log_rect);
|
|
|
|
if (log_rect.width > 0 && log_rect.height > 0) {
|
|
bitmap.rows = font_ascent + font_descent;
|
|
bitmap.width = PANGO_PIXELS_CEIL(log_rect.width);
|
|
bitmap.pitch = -bitmap.width; // Rendering it "upside down" for OpenGL.
|
|
begin_bitmap_buffer = (unsigned char *) g_malloc(bitmap.rows * bitmap.width);
|
|
memset(begin_bitmap_buffer, 0, bitmap.rows * bitmap.width);
|
|
bitmap.buffer = begin_bitmap_buffer + (bitmap.rows - 1) * bitmap.width; // See pitch above.
|
|
bitmap.num_grays = 0xff;
|
|
bitmap.pixel_mode = FT_PIXEL_MODE_GRAY;
|
|
pango_ft2_render_layout_subpixel(&bitmap, layout, -log_rect.x,
|
|
y_offset_bitmap_render_pango_units);
|
|
qglGetFloatv(GL_CURRENT_COLOR, color);
|
|
|
|
// Save state. I didn't see any OpenGL push/pop operations for these.
|
|
// Question: Is saving/restoring this state necessary? Being safe.
|
|
qglGetIntegerv(GL_UNPACK_ALIGNMENT, &previous_unpack_alignment);
|
|
previous_blend_enabled = qglIsEnabled(GL_BLEND);
|
|
qglGetIntegerv(GL_BLEND_SRC, &previous_blend_func_src);
|
|
qglGetIntegerv(GL_BLEND_DST, &previous_blend_func_dst);
|
|
qglGetFloatv(GL_RED_BIAS, &previous_red_bias);
|
|
qglGetFloatv(GL_GREEN_BIAS, &previous_green_bias);
|
|
qglGetFloatv(GL_BLUE_BIAS, &previous_blue_bias);
|
|
qglGetFloatv(GL_ALPHA_SCALE, &previous_alpha_scale);
|
|
|
|
qglPixelStorei(GL_UNPACK_ALIGNMENT, 1);
|
|
qglEnable(GL_BLEND);
|
|
qglBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
qglPixelTransferf(GL_RED_BIAS, color[0]);
|
|
qglPixelTransferf(GL_GREEN_BIAS, color[1]);
|
|
qglPixelTransferf(GL_BLUE_BIAS, color[2]);
|
|
qglPixelTransferf(GL_ALPHA_SCALE, color[3]);
|
|
|
|
qglDrawPixels(bitmap.width, bitmap.rows,
|
|
GL_ALPHA, GL_UNSIGNED_BYTE, begin_bitmap_buffer);
|
|
g_free(begin_bitmap_buffer);
|
|
|
|
// Restore state in reverse order of how we set it.
|
|
qglPixelTransferf(GL_ALPHA_SCALE, previous_alpha_scale);
|
|
qglPixelTransferf(GL_BLUE_BIAS, previous_blue_bias);
|
|
qglPixelTransferf(GL_GREEN_BIAS, previous_green_bias);
|
|
qglPixelTransferf(GL_RED_BIAS, previous_red_bias);
|
|
qglBlendFunc(previous_blend_func_src, previous_blend_func_dst);
|
|
if (!previous_blend_enabled) { qglDisable(GL_BLEND); }
|
|
qglPixelStorei(GL_UNPACK_ALIGNMENT, previous_unpack_alignment);
|
|
}
|
|
|
|
g_object_unref(G_OBJECT(layout));
|
|
|
|
#endif
|
|
}
|
|
|
|
void gtk_glwidget_print_char(char s)
|
|
{
|
|
#ifdef _WIN32
|
|
|
|
gtk_glwidget_create_font_win_internal();
|
|
qglListBase(font_list_base);
|
|
qglCallLists(1, GL_UNSIGNED_BYTE, (unsigned char *) &s);
|
|
|
|
#else
|
|
|
|
char str[2];
|
|
str[0] = s;
|
|
str[1] = '\0';
|
|
gtk_glwidget_print_string(str);
|
|
|
|
#endif
|
|
}
|