15c6c1daeSBarry Smith 25c6c1daeSBarry Smith /* 35c6c1daeSBarry Smith This file contains routines to open an X window display and window 45c6c1daeSBarry Smith This consists of a number of routines that set the various 55c6c1daeSBarry Smith fields in the Window structure, which is passed to 65c6c1daeSBarry Smith all of these routines. 75c6c1daeSBarry Smith 85c6c1daeSBarry Smith Note that if you use the default visual and colormap, then you 95c6c1daeSBarry Smith can use these routines with any X toolkit that will give you the 105c6c1daeSBarry Smith Window id of the window that it is managing. Use that instead of the 115c6c1daeSBarry Smith call to PetscDrawXiCreateWindow . Similarly for the Display. 125c6c1daeSBarry Smith */ 135c6c1daeSBarry Smith 145c6c1daeSBarry Smith #include <../src/sys/classes/draw/impls/x/ximpl.h> 155c6c1daeSBarry Smith 1645f3bb6eSLisandro Dalcin PETSC_INTERN PetscErrorCode PetscDrawSetColormap_X(PetscDraw_X*,Colormap); 175c6c1daeSBarry Smith 185c6c1daeSBarry Smith /* 1915d5bc79SLisandro Dalcin PetscDrawXiOpenDisplay - Open and setup a display 205c6c1daeSBarry Smith */ 2109440f25SLisandro Dalcin static PetscErrorCode PetscDrawXiOpenDisplay(PetscDraw_X *XiWin,const char display[]) 225c6c1daeSBarry Smith { 235c6c1daeSBarry Smith PetscFunctionBegin; 2415d5bc79SLisandro Dalcin XiWin->disp = XOpenDisplay(display); 255c6c1daeSBarry Smith if (!XiWin->disp) { 2698921bdaSJacob Faibussowitsch SETERRQ(PETSC_COMM_SELF,PETSC_ERR_LIB,"Unable to open display on %s\n\ 2715d5bc79SLisandro Dalcin Make sure your COMPUTE NODES are authorized to connect \n\ 285c6c1daeSBarry Smith to this X server and either your DISPLAY variable\n\ 2915d5bc79SLisandro Dalcin is set or you use the -display name option\n",display); 305c6c1daeSBarry Smith } 315c6c1daeSBarry Smith XiWin->screen = DefaultScreen(XiWin->disp); 32481cee7bSLisandro Dalcin XiWin->vis = DefaultVisual(XiWin->disp,XiWin->screen); 33481cee7bSLisandro Dalcin XiWin->depth = DefaultDepth(XiWin->disp,XiWin->screen); 3415d5bc79SLisandro Dalcin XiWin->cmap = DefaultColormap(XiWin->disp,XiWin->screen); 3515d5bc79SLisandro Dalcin XiWin->background = WhitePixel(XiWin->disp,XiWin->screen); 3615d5bc79SLisandro Dalcin XiWin->foreground = BlackPixel(XiWin->disp,XiWin->screen); 375c6c1daeSBarry Smith PetscFunctionReturn(0); 385c6c1daeSBarry Smith } 395c6c1daeSBarry Smith 40815f00f0SLisandro Dalcin PetscErrorCode PetscDrawXiClose(PetscDraw_X *XiWin) 41815f00f0SLisandro Dalcin { 42815f00f0SLisandro Dalcin PetscFunctionBegin; 43815f00f0SLisandro Dalcin if (!XiWin) PetscFunctionReturn(0); 445f80ce2aSJacob Faibussowitsch CHKERRQ(PetscFree(XiWin->font)); 45815f00f0SLisandro Dalcin if (XiWin->disp) { 46815f00f0SLisandro Dalcin #if defined(PETSC_HAVE_SETJMP_H) 47815f00f0SLisandro Dalcin jmp_buf jmpbuf; 48815f00f0SLisandro Dalcin PetscXIOErrorHandler xioerrhdl; 495f80ce2aSJacob Faibussowitsch CHKERRQ(PetscMemcpy(&jmpbuf,&PetscXIOErrorHandlerJumpBuf,sizeof(jmpbuf))); 50815f00f0SLisandro Dalcin xioerrhdl = PetscSetXIOErrorHandler(PetscXIOErrorHandlerJump); 51815f00f0SLisandro Dalcin if (!setjmp(PetscXIOErrorHandlerJumpBuf)) 52815f00f0SLisandro Dalcin #endif 53815f00f0SLisandro Dalcin { 54815f00f0SLisandro Dalcin XFreeGC(XiWin->disp,XiWin->gc.set); 55815f00f0SLisandro Dalcin XCloseDisplay(XiWin->disp); 56815f00f0SLisandro Dalcin } 57815f00f0SLisandro Dalcin XiWin->disp = NULL; 58815f00f0SLisandro Dalcin #if defined(PETSC_HAVE_SETJMP_H) 59815f00f0SLisandro Dalcin (void)PetscSetXIOErrorHandler(xioerrhdl); 605f80ce2aSJacob Faibussowitsch CHKERRQ(PetscMemcpy(&PetscXIOErrorHandlerJumpBuf,&jmpbuf,sizeof(jmpbuf))); 61815f00f0SLisandro Dalcin #endif 62815f00f0SLisandro Dalcin } 63815f00f0SLisandro Dalcin PetscFunctionReturn(0); 64815f00f0SLisandro Dalcin } 65815f00f0SLisandro Dalcin 665c6c1daeSBarry Smith /* 6709440f25SLisandro Dalcin PetscDrawXiCreateGC - setup the GC structure 685c6c1daeSBarry Smith */ 6909440f25SLisandro Dalcin static PetscErrorCode PetscDrawXiCreateGC(PetscDraw_X *XiWin,PetscDrawXiPixVal fg) 705c6c1daeSBarry Smith { 715c6c1daeSBarry Smith XGCValues gcvalues; /* window graphics context values */ 725c6c1daeSBarry Smith 735c6c1daeSBarry Smith PetscFunctionBegin; 745c6c1daeSBarry Smith /* Set the graphics contexts */ 755c6c1daeSBarry Smith /* create a gc for the ROP_SET operation (writing the fg value to a pixel) */ 765c6c1daeSBarry Smith /* (do this with function GXcopy; GXset will automatically write 1) */ 775c6c1daeSBarry Smith gcvalues.function = GXcopy; 785c6c1daeSBarry Smith gcvalues.foreground = fg; 795c6c1daeSBarry Smith XiWin->gc.cur_pix = fg; 805c6c1daeSBarry Smith XiWin->gc.set = XCreateGC(XiWin->disp,RootWindow(XiWin->disp,XiWin->screen),GCFunction|GCForeground,&gcvalues); 812c71b3e2SJacob Faibussowitsch PetscCheckFalse(!XiWin->gc.set,PETSC_COMM_SELF,PETSC_ERR_LIB,"Unable to create X graphics context"); 8215d5bc79SLisandro Dalcin PetscFunctionReturn(0); 8315d5bc79SLisandro Dalcin } 8415d5bc79SLisandro Dalcin 8515d5bc79SLisandro Dalcin /* 86815f00f0SLisandro Dalcin PetscDrawXiInit - basic setup the draw (display, graphics context, font) 8715d5bc79SLisandro Dalcin */ 8865d9662dSLisandro Dalcin PetscErrorCode PetscDrawXiInit(PetscDraw_X *XiWin,const char display[]) 8915d5bc79SLisandro Dalcin { 9015d5bc79SLisandro Dalcin PetscFunctionBegin; 915f80ce2aSJacob Faibussowitsch CHKERRQ(PetscDrawXiOpenDisplay(XiWin,display)); 925f80ce2aSJacob Faibussowitsch CHKERRQ(PetscDrawXiCreateGC(XiWin,XiWin->foreground)); 935f80ce2aSJacob Faibussowitsch CHKERRQ(PetscDrawXiFontFixed(XiWin,6,10,&XiWin->font)); 945c6c1daeSBarry Smith PetscFunctionReturn(0); 955c6c1daeSBarry Smith } 965c6c1daeSBarry Smith 975c6c1daeSBarry Smith /* 9809440f25SLisandro Dalcin This routine waits until the window is actually created or destroyed 9909440f25SLisandro Dalcin Returns 0 if window is mapped; 1 if window is destroyed. 10009440f25SLisandro Dalcin */ 10109440f25SLisandro Dalcin static PetscErrorCode PetscDrawXiWaitMap(PetscDraw_X *XiWin) 10209440f25SLisandro Dalcin { 10309440f25SLisandro Dalcin XEvent event; 10409440f25SLisandro Dalcin 10509440f25SLisandro Dalcin PetscFunctionBegin; 10609440f25SLisandro Dalcin while (1) { 10709440f25SLisandro Dalcin XMaskEvent(XiWin->disp,ExposureMask|StructureNotifyMask,&event); 10809440f25SLisandro Dalcin if (event.xany.window != XiWin->win) break; 10909440f25SLisandro Dalcin else { 11009440f25SLisandro Dalcin switch (event.type) { 11109440f25SLisandro Dalcin case ConfigureNotify: 11209440f25SLisandro Dalcin /* window has been moved or resized */ 11309440f25SLisandro Dalcin XiWin->w = event.xconfigure.width - 2 * event.xconfigure.border_width; 11409440f25SLisandro Dalcin XiWin->h = event.xconfigure.height - 2 * event.xconfigure.border_width; 11509440f25SLisandro Dalcin break; 11609440f25SLisandro Dalcin case DestroyNotify: 11709440f25SLisandro Dalcin PetscFunctionReturn(1); 11809440f25SLisandro Dalcin case Expose: 11909440f25SLisandro Dalcin PetscFunctionReturn(0); 12009440f25SLisandro Dalcin /* else ignore event */ 12109440f25SLisandro Dalcin } 12209440f25SLisandro Dalcin } 12309440f25SLisandro Dalcin } 12409440f25SLisandro Dalcin PetscFunctionReturn(0); 12509440f25SLisandro Dalcin } 12609440f25SLisandro Dalcin 12709440f25SLisandro Dalcin /* 1285c6c1daeSBarry Smith Actually display a window at [x,y] with sizes (w,h) 1295c6c1daeSBarry Smith */ 13009440f25SLisandro Dalcin static PetscErrorCode PetscDrawXiDisplayWindow(PetscDraw_X *XiWin,char *label,int x,int y,int w,int h) 1315c6c1daeSBarry Smith { 1325c6c1daeSBarry Smith unsigned int wavail,havail; 1335c6c1daeSBarry Smith XSizeHints size_hints; 1345c6c1daeSBarry Smith XWindowAttributes in_window_attributes; 1355c6c1daeSBarry Smith XSetWindowAttributes window_attributes; 13615d5bc79SLisandro Dalcin unsigned int border_width = 0; 13715d5bc79SLisandro Dalcin unsigned long backgnd_pixel = WhitePixel(XiWin->disp,XiWin->screen); 1385c6c1daeSBarry Smith unsigned long wmask; 1395c6c1daeSBarry Smith 1405c6c1daeSBarry Smith PetscFunctionBegin; 1415c6c1daeSBarry Smith /* get the available widths */ 1425c6c1daeSBarry Smith wavail = DisplayWidth(XiWin->disp,XiWin->screen); 1435c6c1daeSBarry Smith havail = DisplayHeight(XiWin->disp,XiWin->screen); 1442c71b3e2SJacob Faibussowitsch PetscCheckFalse(w <= 0 || h <= 0,PETSC_COMM_SELF,PETSC_ERR_LIB,"X Window display has invalid height or width"); 1455c6c1daeSBarry Smith if ((unsigned int)w > wavail) w = wavail; 1465c6c1daeSBarry Smith if ((unsigned int)h > havail) h = havail; 1475c6c1daeSBarry Smith 14815d5bc79SLisandro Dalcin if (x < 0) x = (int)(wavail - (unsigned int)w + (unsigned int)x); 14915d5bc79SLisandro Dalcin if (y < 0) y = (int)(havail - (unsigned int)h + (unsigned int)y); 15015d5bc79SLisandro Dalcin x = ((unsigned int)x + w > wavail) ? (int)(wavail - (unsigned int)w) : x; 15115d5bc79SLisandro Dalcin y = ((unsigned int)y + h > havail) ? (int)(havail - (unsigned int)h) : y; 1525c6c1daeSBarry Smith 1535c6c1daeSBarry Smith /* We need XCreateWindow since we may need an visual other than the default one */ 1545c6c1daeSBarry Smith XGetWindowAttributes(XiWin->disp,RootWindow(XiWin->disp,XiWin->screen),&in_window_attributes); 1555c6c1daeSBarry Smith window_attributes.background_pixmap = None; 1565c6c1daeSBarry Smith window_attributes.background_pixel = backgnd_pixel; 1575c6c1daeSBarry Smith /* No border for now */ 1585c6c1daeSBarry Smith window_attributes.border_pixmap = None; 1595c6c1daeSBarry Smith /* 1605c6c1daeSBarry Smith window_attributes.border_pixel = border_pixel; 1615c6c1daeSBarry Smith */ 1625c6c1daeSBarry Smith window_attributes.bit_gravity = in_window_attributes.bit_gravity; 1635c6c1daeSBarry Smith window_attributes.win_gravity = in_window_attributes.win_gravity; 1645c6c1daeSBarry Smith /* Backing store is too slow in color systems */ 16515d5bc79SLisandro Dalcin window_attributes.backing_store = NotUseful; 1665c6c1daeSBarry Smith window_attributes.backing_pixel = backgnd_pixel; 1675c6c1daeSBarry Smith window_attributes.save_under = 1; 1685c6c1daeSBarry Smith window_attributes.event_mask = 0; 1695c6c1daeSBarry Smith window_attributes.do_not_propagate_mask = 0; 1705c6c1daeSBarry Smith window_attributes.override_redirect = 0; 1715c6c1daeSBarry Smith window_attributes.colormap = XiWin->cmap; 1725c6c1daeSBarry Smith /* None for cursor does NOT mean none, it means cursor of Parent */ 1735c6c1daeSBarry Smith window_attributes.cursor = None; 174a297a907SKarl Rupp 1755c6c1daeSBarry Smith wmask = CWBackPixmap | CWBackPixel | CWBorderPixmap | CWBitGravity | 1765c6c1daeSBarry Smith CWWinGravity | CWBackingStore | CWBackingPixel | CWOverrideRedirect | 1775c6c1daeSBarry Smith CWSaveUnder | CWEventMask | CWDontPropagate | 1785c6c1daeSBarry Smith CWCursor | CWColormap; 1795c6c1daeSBarry Smith 18015d5bc79SLisandro Dalcin XiWin->win = XCreateWindow(XiWin->disp,RootWindow(XiWin->disp,XiWin->screen),x,y,w,h,border_width,XiWin->depth,InputOutput,XiWin->vis,wmask,&window_attributes); 181*28b400f6SJacob Faibussowitsch PetscCheck(XiWin->win,PETSC_COMM_SELF,PETSC_ERR_LIB,"Unable to open X window"); 1825c6c1daeSBarry Smith 1835c6c1daeSBarry Smith /* set window manager hints */ 1845c6c1daeSBarry Smith { 1855c6c1daeSBarry Smith XWMHints wm_hints; 1865c6c1daeSBarry Smith XClassHint class_hints; 1875c6c1daeSBarry Smith XTextProperty windowname,iconname; 1885c6c1daeSBarry Smith 189a297a907SKarl Rupp if (label) XStringListToTextProperty(&label,1,&windowname); 190a297a907SKarl Rupp else XStringListToTextProperty(&label,0,&windowname); 191a297a907SKarl Rupp if (label) XStringListToTextProperty(&label,1,&iconname); 192a297a907SKarl Rupp else XStringListToTextProperty(&label,0,&iconname); 1935c6c1daeSBarry Smith 1945c6c1daeSBarry Smith wm_hints.initial_state = NormalState; 1955c6c1daeSBarry Smith wm_hints.input = True; 1965c6c1daeSBarry Smith wm_hints.flags = StateHint|InputHint; 1975c6c1daeSBarry Smith 1985c6c1daeSBarry Smith /* These properties can be used by window managers to decide how to display a window */ 1995c6c1daeSBarry Smith class_hints.res_name = (char*)"petsc"; 2005c6c1daeSBarry Smith class_hints.res_class = (char*)"PETSc"; 2015c6c1daeSBarry Smith 2025c6c1daeSBarry Smith size_hints.x = x; 2035c6c1daeSBarry Smith size_hints.y = y; 2045c6c1daeSBarry Smith size_hints.min_width = 4*border_width; 2055c6c1daeSBarry Smith size_hints.min_height = 4*border_width; 2065c6c1daeSBarry Smith size_hints.width = w; 2075c6c1daeSBarry Smith size_hints.height = h; 2085c6c1daeSBarry Smith size_hints.flags = USPosition | USSize | PMinSize; 2095c6c1daeSBarry Smith 21002c9f0b5SLisandro Dalcin XSetWMProperties(XiWin->disp,XiWin->win,&windowname,&iconname,NULL,0,&size_hints,&wm_hints,&class_hints); 2115c6c1daeSBarry Smith XFree((void*)windowname.value); 2125c6c1daeSBarry Smith XFree((void*)iconname.value); 2135c6c1daeSBarry Smith } 214481cee7bSLisandro Dalcin 2155c6c1daeSBarry Smith /* make the window visible */ 2165c6c1daeSBarry Smith XSelectInput(XiWin->disp,XiWin->win,ExposureMask|StructureNotifyMask); 2175c6c1daeSBarry Smith XMapWindow(XiWin->disp,XiWin->win); 2185c6c1daeSBarry Smith /* some window systems are cruel and interfere with the placement of 2195c6c1daeSBarry Smith windows. We wait here for the window to be created or to die */ 2202c71b3e2SJacob Faibussowitsch PetscCheckFalse(PetscDrawXiWaitMap(XiWin),PETSC_COMM_SELF,PETSC_ERR_LIB,"Wait for X window failed"); 22173ecf448SLisandro Dalcin XSelectInput(XiWin->disp,XiWin->win,NoEventMask); 2225c6c1daeSBarry Smith PetscFunctionReturn(0); 2235c6c1daeSBarry Smith } 2245c6c1daeSBarry Smith 22515d5bc79SLisandro Dalcin PetscErrorCode PetscDrawXiQuickWindow(PetscDraw_X *XiWin,char *name,int x,int y,int nx,int ny) 2265c6c1daeSBarry Smith { 2275c6c1daeSBarry Smith PetscFunctionBegin; 2285f80ce2aSJacob Faibussowitsch CHKERRQ(PetscDrawSetColormap_X(XiWin,(Colormap)0)); 2295f80ce2aSJacob Faibussowitsch CHKERRQ(PetscDrawXiDisplayWindow(XiWin,name,x,y,nx,ny)); 23015d5bc79SLisandro Dalcin XSetWindowBackground(XiWin->disp,XiWin->win,XiWin->background); 23115d5bc79SLisandro Dalcin XClearWindow(XiWin->disp,XiWin->win); 2325c6c1daeSBarry Smith PetscFunctionReturn(0); 2335c6c1daeSBarry Smith } 2345c6c1daeSBarry Smith 2355c6c1daeSBarry Smith /* 2365c6c1daeSBarry Smith A version from an already defined window 2375c6c1daeSBarry Smith */ 23815d5bc79SLisandro Dalcin PetscErrorCode PetscDrawXiQuickWindowFromWindow(PetscDraw_X *XiWin,Window win) 2395c6c1daeSBarry Smith { 2405c6c1daeSBarry Smith XWindowAttributes attributes; 2415c6c1daeSBarry Smith 2425c6c1daeSBarry Smith PetscFunctionBegin; 24315d5bc79SLisandro Dalcin XiWin->win = win; 24415d5bc79SLisandro Dalcin XGetWindowAttributes(XiWin->disp,XiWin->win,&attributes); 2455f80ce2aSJacob Faibussowitsch CHKERRQ(PetscDrawSetColormap_X(XiWin,attributes.colormap)); 2465c6c1daeSBarry Smith PetscFunctionReturn(0); 2475c6c1daeSBarry Smith } 2485c6c1daeSBarry Smith 24909440f25SLisandro Dalcin PetscErrorCode PetscDrawXiQuickPixmap(PetscDraw_X* XiWin) 25009440f25SLisandro Dalcin { 2515c6c1daeSBarry Smith PetscFunctionBegin; 25209440f25SLisandro Dalcin if (XiWin->drw) XFreePixmap(XiWin->disp,XiWin->drw); 25309440f25SLisandro Dalcin XiWin->drw = XCreatePixmap(XiWin->disp,RootWindow(XiWin->disp,XiWin->screen),XiWin->w,XiWin->h,XiWin->depth); 25409440f25SLisandro Dalcin PetscDrawXiSetPixVal(XiWin,XiWin->background); 25509440f25SLisandro Dalcin XFillRectangle(XiWin->disp,XiWin->drw,XiWin->gc.set,0,0,XiWin->w,XiWin->h); 25609440f25SLisandro Dalcin XSync(XiWin->disp,False); 2575c6c1daeSBarry Smith PetscFunctionReturn(0); 2585c6c1daeSBarry Smith } 25948db01dbSLisandro Dalcin 260bf780456SLisandro Dalcin PetscErrorCode PetscDrawXiResizeWindow(PetscDraw_X* XiWin,int w,int h) 261bf780456SLisandro Dalcin { 262bf780456SLisandro Dalcin XEvent event; 263bf780456SLisandro Dalcin PetscFunctionBegin; 264bf780456SLisandro Dalcin XSelectInput(XiWin->disp,XiWin->win,StructureNotifyMask); 265bf780456SLisandro Dalcin XResizeWindow(XiWin->disp,XiWin->win,(unsigned int)w,(unsigned int)h); 266bf780456SLisandro Dalcin XWindowEvent(XiWin->disp,XiWin->win,StructureNotifyMask,&event); 267bf780456SLisandro Dalcin XSelectInput(XiWin->disp,XiWin->win,NoEventMask); 268bf780456SLisandro Dalcin PetscFunctionReturn(0); 269bf780456SLisandro Dalcin } 270bf780456SLisandro Dalcin 27148db01dbSLisandro Dalcin PetscErrorCode PetscDrawXiGetGeometry(PetscDraw_X *XiWin,int *x,int *y,int *w,int *h) 27248db01dbSLisandro Dalcin { 27348db01dbSLisandro Dalcin XWindowAttributes attributes; 27448db01dbSLisandro Dalcin Window root,parent,child; 27548db01dbSLisandro Dalcin int xx=0,yy=0; 27648db01dbSLisandro Dalcin unsigned int ww=0,hh=0,dummy; 27748db01dbSLisandro Dalcin PetscFunctionBegin; 27848db01dbSLisandro Dalcin if (XiWin->win) { 27948db01dbSLisandro Dalcin XGetGeometry(XiWin->disp,XiWin->win,&parent,&xx,&yy,&ww,&hh,&dummy,&dummy); 28048db01dbSLisandro Dalcin root = RootWindow(XiWin->disp,XiWin->screen); 28148db01dbSLisandro Dalcin if (!XTranslateCoordinates(XiWin->disp,XiWin->win,root,0,0,&xx,&yy,&child)) { 28248db01dbSLisandro Dalcin XGetWindowAttributes(XiWin->disp,XiWin->win,&attributes); 28348db01dbSLisandro Dalcin root = attributes.screen->root; 28448db01dbSLisandro Dalcin (void)XTranslateCoordinates(XiWin->disp,XiWin->win,root,0,0,&xx,&yy,&child); 28548db01dbSLisandro Dalcin } 28648db01dbSLisandro Dalcin } else if (XiWin->drw) { 28748db01dbSLisandro Dalcin XGetGeometry(XiWin->disp,XiWin->drw,&root,&xx,&yy,&ww,&hh,&dummy,&dummy); 28848db01dbSLisandro Dalcin } 28948db01dbSLisandro Dalcin if (x) *x = xx; 29048db01dbSLisandro Dalcin if (y) *y = yy; 29148db01dbSLisandro Dalcin if (w) *w = (int)ww; 29248db01dbSLisandro Dalcin if (h) *h = (int)hh; 29348db01dbSLisandro Dalcin PetscFunctionReturn(0); 29448db01dbSLisandro Dalcin } 295