@frilix
Иногда "творю"

Есть ли возможность подружить SDL2 & GTK?

Добрый день, разрабатываю редактор кода на линуксе. Нужно каким-то образом прикрутить SDL к gtk , чтобы использовать некоторые возможности gtk . Прошу помощи, так как уже совсем отчаялся в поисках решения проблемы.

То есть надо использовать gtk для отрисовки меню и кнопочек, а SDL для работы с остальной графикой

#include <gtk/gtk.h> 
#include <gdk/gdkx.h> 
#include <SDL2/SDL.h> 

GtkWidget *window; 
GtkWidget *sdlArea; 
SDL_Window *SDLwin; 
SDL_Renderer *SDLgfx; 

void doDrawing(GtkWidget *widget, gpointer *data) { 
    SDL_Rect myRect = {10, 10, 20, 20}; 
    SDL_SetRenderDrawColor(SDLgfx, 255, 0, 0, SDL_ALPHA_OPAQUE); 
    SDL_RenderClear(SDLgfx); 
    SDL_SetRenderDrawColor(SDLgfx, 0, 255, 0, SDL_ALPHA_OPAQUE); 
    SDL_RenderFillRect(SDLgfx, &myRect); 
    SDL_RenderPresent(SDLgfx); 
} 

int main( int argc, char *argv[] ) { 
    
    SDL_Init(SDL_INIT_VIDEO); 
    gtk_init(&argc, &argv); 

    window = gtk_window_new(GTK_WINDOW_TOPLEVEL); 
    g_signal_connect( window, "destroy", G_CALLBACK (gtk_main_quit), NULL ); 
    
    sdlArea = gtk_drawing_area_new(); 
    g_signal_connect (sdlArea, "draw", G_CALLBACK (doDrawing), NULL ); 
    gtk_widget_set_app_paintable(sdlArea, True); 
    gtk_container_add( GTK_CONTAINER(window), sdlArea ); 
    
    gtk_widget_show_all(window); 

    GdkWindow *gdk_window = gtk_widget_get_window(sdlArea); 
    Window x11_window = gdk_x11_window_get_xid(GDK_X11_WINDOW(gdk_window)); 
    
    SDLwin = SDL_CreateWindowFrom((const void*) x11_window); 
    SDLgfx = SDL_CreateRenderer(SDLwin, -1, SDL_RENDERER_SOFTWARE | SDL_RENDERER_TARGETTEXTURE); 
    
    SDL_Rect myRect = {10, 10, 20, 20}; 
    SDL_SetRenderDrawColor(SDLgfx, 255, 0, 0, SDL_ALPHA_OPAQUE); 
    SDL_RenderClear(SDLgfx); 
    SDL_SetRenderDrawColor(SDLgfx, 0, 255, 0, SDL_ALPHA_OPAQUE); 
    SDL_RenderFillRect(SDLgfx, &myRect); 
    SDL_RenderPresent(SDLgfx); 
    
    gtk_main(); 
    
    SDL_Quit(); 
    return 0; 
}
  • Вопрос задан
  • 835 просмотров
Пригласить эксперта
Ответы на вопрос 2
GTK-demo 1.2
/* A simple example of using SDL with GTk */

#include <stdlib.h>
#include <stdio.h>
#include <time.h>

#include <gtk/gtk.h>
#include <gdk/gdkx.h>

#ifdef USE_XSHAPE
#include <X11/Xlib.h>
#include <X11/extensions/shape.h>
#endif

#include "SDL.h"

#define WINSIZEX 320
#define WINSIZEY 200

/* The main display surface */
GtkWidget *mainwin;
SDL_Surface *screen = NULL;


#ifdef USE_XSHAPE
int use_shape = 0;
Pixmap shape_mask = 0;
GC shape_gc;
int shape_x = 0;
int shape_y = 0;
int shape_dx = 0;
int shape_dy = 0;

void ResizeShape(void)
{
	Display *dpy;
	XGCValues GCvalue;

	if ( ! use_shape ) {
		return;
	}
	dpy = GDK_WINDOW_XDISPLAY(mainwin->window);
	if ( shape_mask ) {
                XFreePixmap(dpy, shape_mask);
                XFreeGC(dpy, shape_gc);
	}
	shape_mask = XCreatePixmap(dpy, DefaultRootWindow(dpy),
	                           screen->w, screen->h, 1);
	GCvalue.function = GXcopy;
        shape_gc = XCreateGC(dpy, shape_mask, GCFunction, &GCvalue);
	XSync(dpy, False);
}

void ShapeWindow(void)
{
#ifdef SQUARE_HOLE
#define RADIUS 16
#else
#define RADIUS 24
#endif
	Display *dpy;
	Window win;
	int x, y, i;
	void (*set_bit)(XImage *image, int x, int y);
	int (*get_bit)(XImage *image, int x, int y);

	if ( ! use_shape ) {
		return;
	}

	/* Only move the shape every 100'th time through */
	{ static int step = 0;
	  if ( (step++)%100 != 0 ) {
		return;
	  }
	}

	/* Set the window and display */
	dpy = GDK_WINDOW_XDISPLAY(mainwin->window);
	win = GDK_WINDOW_XWINDOW(mainwin->window);

	/* Bounce the hole when it hits the edge */
	if ( shape_x == 0 ) {
		shape_dx = 1;
	}
	if ( shape_x >= (screen->w-(2*RADIUS)) ) {
		shape_x = (screen->w-(2*RADIUS));
		shape_dx = -1;
	}
	if ( shape_y == 0 ) {
		shape_dy = 1;
	}
	if ( shape_y >= (screen->h-(2*RADIUS)) ) {
		shape_y = (screen->h-(2*RADIUS));
		shape_dy = -1;
	}

	/* Move the hole around */
	shape_x += shape_dx;
	shape_y += shape_dy;

	/* Now put the hole in the mask
	   Note: you can also set the mask directly by getting an XImage
	         of it, and modifying the data member and then putting
	         the image to the mask pixmap.  You have to think about
	         the bit ordering in the image though.  Using X calls is
	         better for simple shapes because the X server will take
	         care of the bit ordering, and possibly use hardware
	         acceleration when possible.
	*/
	dpy = GDK_WINDOW_XDISPLAY(mainwin->window);
	XSetForeground(dpy, shape_gc, ~0);
	XFillRectangle(dpy, shape_mask, shape_gc, 0, 0, screen->w, screen->h);
	XSetForeground(dpy, shape_gc, 0);
#ifdef SQUARE_HOLE
	XFillRectangle(dpy, shape_mask, shape_gc, shape_x, shape_y, 2*RADIUS, 2*RADIUS);
#else
	XFillArc(dpy, shape_mask, shape_gc, shape_x, shape_y, 2*RADIUS, 2*RADIUS, 0, 360*64);
#endif

	/* Set the mask on the window */
	XShapeCombineMask(dpy, win, ShapeBounding, 0, 0, shape_mask, ShapeSet);

	/* Update the whole screen */
	SDL_UpdateRect(screen, 0, 0, screen->w, screen->h);
}

/* Check for the X shaped window extension */
void CheckShape(void)
{
	int ev_base, er_base;
	Display *dpy;

	use_shape = 0;
	dpy = GDK_WINDOW_XDISPLAY(mainwin->window);
	if ( XShapeQueryExtension(dpy, &ev_base, &er_base) ) {
		use_shape = 1;
	}
	ResizeShape();
printf("Shape extension%s available\n", use_shape ? "" : " not");
}

#endif /* USE_SHAPE */


/* Event handlers -- the configure_event handler is very important! */

gint
button_press_event (GtkWidget *widget,  GdkEventButton *event)
{
	SDL_Rect rect;

	rect.x = event->x-1;
	rect.y = event->y-1;
	rect.w = 2;
	rect.h = 2;
	SDL_FillRect(screen, &rect, SDL_MapRGB(screen->format, 255, 255, 255));
	SDL_UpdateRects(screen, 1, &rect);

	return TRUE ;
}

gint
configure_event (GtkWidget *widget, GdkEventConfigure *event)
{
	screen = SDL_SetVideoMode(event->width, event->height, 0, 0);
#ifdef USE_XSHAPE
	ResizeShape();
#endif
	return TRUE ;
}

gint
delete_event (GtkWidget *widget, GdkEventConfigure *event)
{
	gtk_main_quit();
	return TRUE ;
}

/* Idle function -- called when GTk isn't busy */
gint
idle_loop (gpointer data)
{
	int x, y;

	/* Make a random spot black (well, color 0 anyway :) */
	x = rand()%screen->w;
	y = rand()%screen->h;
	switch (screen->format->BytesPerPixel) {
		case 1:
			*((Uint8 *)screen->pixels+y*screen->pitch+x) = 0;
			break;
		case 2:
			*((Uint16 *)screen->pixels+y*screen->pitch/2+x) = 0;
			break;
		case 3:  /* Yuck... */
			((Uint8 *)screen->pixels+y*screen->pitch+x)[0] = 0;
			((Uint8 *)screen->pixels+y*screen->pitch+x)[1] = 0;
			((Uint8 *)screen->pixels+y*screen->pitch+x)[2] = 0;
			break;
		case 4:
			*((Uint32 *)screen->pixels+y*screen->pitch/4+x) = 0;
			break;
	}
#ifdef USE_XSHAPE
	ShapeWindow();
#endif
	SDL_UpdateRect(screen, x, y, 1, 1);
	return TRUE ;
}

/* Menu callbacks */

void NewScreen(gpointer cb_data, guint cb_action, GtkWidget *widget)
{
	SDL_FillRect(screen, NULL, SDL_MapRGB(screen->format, 255, 255, 255));
	SDL_UpdateRect(screen, 0, 0, 0, 0);
}

void QuitGame(gpointer callback_data, guint callback_action, GtkWidget *widget)
{
	gtk_main_quit();
}


static GtkItemFactoryEntry menu_items[] = {
   {"/_File",			NULL,		0,		0, "<Branch>" },
   {"/_File/_New",		NULL,		NewScreen,	0 },
   {"/_File/_Quit",		"<control>Q",	QuitGame,	0 },
};

void CreateMenus(GtkWidget *window)
{
	GtkAccelGroup *accel_group;
	GtkItemFactory *item_factory;
	GtkWidget *box1;

	accel_group = gtk_accel_group_new();
	item_factory = gtk_item_factory_new(GTK_TYPE_MENU_BAR, "<menu>", accel_group);
	gtk_item_factory_create_items(item_factory, 3, menu_items, NULL);
	gtk_accel_group_attach(accel_group, GTK_OBJECT(window));
	gtk_window_set_title(GTK_WINDOW(window), "Star Field");
	gtk_container_border_width(GTK_CONTAINER(window), 0);
	box1 = gtk_vbox_new(FALSE, 0);
	gtk_container_add(GTK_CONTAINER(window), box1);
	gtk_box_pack_start(GTK_BOX(box1), gtk_item_factory_get_widget(item_factory, "<menu>"), FALSE, FALSE, 0);
	gtk_widget_show_all(window);
}

int main(int argc, char *argv[])
{
	gtk_init(&argc, &argv);

	/* Create a main window */
	mainwin = gtk_window_new(GTK_WINDOW_TOPLEVEL);
	gtk_widget_set_usize(mainwin, WINSIZEX, WINSIZEY);
	gtk_widget_realize(mainwin);

	/* Add event handlers -- the configure_event handler is important! */
	gtk_widget_add_events(mainwin, GDK_BUTTON_PRESS_MASK);
	gtk_signal_connect(GTK_OBJECT(mainwin), "configure_event", GTK_SIGNAL_FUNC(configure_event), 0);
	gtk_signal_connect(GTK_OBJECT(mainwin), "button_press_event", GTK_SIGNAL_FUNC(button_press_event), 0);
	gtk_signal_connect(GTK_OBJECT(mainwin), "delete_event", GTK_SIGNAL_FUNC(delete_event), 0);

	/* Hack to get SDL to use GTK window */
	{ char SDL_windowhack[32];
		sprintf(SDL_windowhack,"SDL_WINDOWID=%ld",
			GDK_WINDOW_XWINDOW(mainwin->window));
		putenv(SDL_windowhack);
	}

	/* Initialize SDL */
	if ( SDL_Init(SDL_INIT_VIDEO) < 0 ) {
		fprintf(stderr,"Couldn't initialize SDL: %s\n",SDL_GetError());
		gtk_main_quit();
	}

	screen = SDL_SetVideoMode(WINSIZEX, WINSIZEY, 0, 0);

#ifdef USE_XSHAPE
	/* Check for the X shaped window extension */
	CheckShape();
#endif

	/* Create menus for the main window */
	CreateMenus( mainwin );

	/* Add an idle function -- game main loop? */
	srand(time(NULL));
	gtk_idle_add(idle_loop, mainwin);

	/* The last thing to get called */
	gtk_main();
	SDL_Quit();
	return 0;
}
Ответ написан
@frilix Автор вопроса
Иногда "творю"
Используется SDL2
Ответ написан
Ваш ответ на вопрос

Войдите, чтобы написать ответ

Войти через центр авторизации
Похожие вопросы