Mercurial > hg > xemacs-beta
view src/balloon_help.c @ 1330:4542b72c005e
[xemacs-hg @ 2003-03-01 07:25:26 by ben]
build patch
Makefile.in.in: Move src deletions to src/Makefile.in.in.
dump-paths.el, dumped-lisp.el: Delete. Combine stuff into setup-paths.el.
find-paths.el: Removed.
Make this file contain generic routines only. Move stuff to
compute Emacs roots to setup-paths.el.
startup.el: Removed.
Move these variables into setup-paths.el.
setup-paths.el, startup.el: Removed.
Combine all high-level code for computing the paths into
setup-paths.el. Create new function startup-find-load-path to
encapsulate all logic for computing `load-path'. Eliminate
invocation-directory and invocation-name parameters since
there is no point (false generality) -- the code references
other globals, which cannot be specified. Eliminate some code
duplicated between setup-paths.el and startup.el. Clean up
the debug-paths code and output load-path in addition.
Add logic to paths-emacs-root-p to support separated source
and build trees.
loadup.el, make-docfile.el, update-elc-2.el, update-elc.el: Rewrite to allow for separated source and build trees, as may occur
in MS Windows.
NOTE TO BUILD HACKERS:
loadup.el, make-docfile.el, update-elc.el and update-elc-2.el made two
assumptions that are no longer correct:
(1) The source and build trees are in the same place.
(2) They can make assumptions about where `.' is.
These files now compute the locations of the source and build
roots at the top of the file. *ALL* constant file names or path
snippets must now be made absolute using expand-file-name and one
of these roots.
dumped-lisp.el, packages.el: Removed.
Remove some unused lists of Lisp files. packages-hardcoded-lisp
(empty, in any case) moved to dumped-lisp.el.
startup.el: When a compiled init file is out-of-date wrt the uncompiled
version, load the uncompiled version and issue a nasty warning.
update-elc-2.el: Force touching of auto-autoloads files when REBUILD_AUTOLOADS
was set.
update-elc.el: Fix code that checks whether dumping is necessary to check against
xemacs.dmp, not xemacs.exe, when Unix and pdump.
lwlib-Xm.c: Fix compile warning.
README, config.inc.samp, xemacs.mak: -- Major reorganization and cleanup.
-- Add support for separated build tree and source tree.
-- Delete all support for X Windows building, since it's
totally bit-rotten and will never be fixed up. Instruct
people to use Cygwin if they want such support.
make-build-dir: New script to create a skeleton build tree for use with
separated build and source tree compilation.
m/acorn.h, m/alliant-2800.h, m/alliant.h, m/altos.h, m/amdahl.h, m/arm.h, m/att3b.h, m/aviion.h, m/clipper.h, m/cnvrgnt.h, m/convex.h, m/cydra5.h, m/delta.h, m/delta88k.h, m/dpx2.h, m/elxsi.h, m/ews4800r.h, m/gould.h, m/hp800.h, m/hp9000s300.h, m/i860.h, m/ibmps2-aix.h, m/ibmrs6000.h, m/ibmrt-aix.h, m/ibmrt.h, m/intel386.h, m/iris4d.h, m/iris5d.h, m/iris6d.h, m/irist.h, m/m68k.h, m/masscomp.h, m/mg1.h, m/mips-nec.h, m/mips-siemens.h, m/mips.h, m/nh3000.h, m/nh4000.h, m/ns32000.h, m/plexus.h, m/powerpc.h, m/sequent-ptx.h, m/sequent.h, m/sgi-challenge.h, m/stride.h, m/tad68k.h, m/targon31.h, m/tekxd88.h, m/template.h, m/tower32.h, m/tower32v3.h, m/ustation.h, m/wicat.h, m/xps100.h, data.c, doc.c, editfns.c, emacs.c, lrecord.h, ntheap.c, process-unix.c, sysdep.c, unexec.c: Delete all support for bit-rotten CANNOT_DUMP. Just use pdump.
Makefile.in.in: Lots o' cleanup. Use names like LISP, SRC instead of
lispdir, srcdir, for consistency with xemacs.mak and the
conventions in the rest of the file. Eliminate use of ${...}
in favor of $(...), to make it easier to move code between
this file and xemacs.mak. Fix dependency handling wrt
NEEDTODUMP to eliminate problems some people (e.g. Vin) have
been seeing with non-GNU makes. Write a long section about
the subtle but oh-so-important differences in dependency
processing between nmake, make, and GNU make. Add
unicode-encapsulate target, from xemacs.mak.
chartab.c, lrecord.h: Fix crash due to attempt to free objects across dump/undump.
author | ben |
---|---|
date | Sat, 01 Mar 2003 07:25:56 +0000 |
parents | 943eaba38521 |
children | 04bc9d2f42c7 |
line wrap: on
line source
/* Balloon Help Copyright (c) 1997 Douglas Keller This file is part of XEmacs. XEmacs is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. XEmacs is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with XEmacs; see the file COPYING. If not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /* Synched up with: Not in FSF. */ /* !!#### Needs to be Mule-ized. */ /* * Balloon Help * * Version: 1.337 (Sun Apr 13 04:52:10 1997) * * Written by Douglas Keller <dkeller@vnet.ibm.com> * * */ #include <config.h> #include <string.h> #include <stdlib.h> #include <stdio.h> #include <assert.h> #include <X11/Xlib.h> #include <X11/Xutil.h> #include <X11/extensions/shape.h> #include "xintrinsic.h" #include "balloon_help.h" #ifndef max #define max(x,y) (x>y?x:y) #endif #undef bool #define bool int #define MARGIN_WIDTH 4 #define POINTER_OFFSET 8 #define BORDER_WIDTH 2 #define BORDER_WIDTH_HALF 1 #define CONE_HEIGHT 20 #define CONE_WIDTH 50 #define SHAPE_CONE_TOP (1<<0) #define SHAPE_CONE_LEFT (1<<1) #define SHAPE_CONE_TOP_LEFT (SHAPE_CONE_TOP | SHAPE_CONE_LEFT) #define SHAPE_CONE_TOP_RIGHT (SHAPE_CONE_TOP) #define SHAPE_CONE_BOTTOM_LEFT (SHAPE_CONE_LEFT) #define SHAPE_CONE_BOTTOM_RIGHT (0) #define SHAPE_CONE_FREE (-1) static Display* b_dpy; static XFontStruct* b_fontStruct; static GC b_gc; static GC b_shineGC; static GC b_shadowGC; static Window b_win; static bool b_winMapped; static Pixmap b_mask; static int b_maskWidth, b_maskHeight; static GC b_maskGC; static const char* b_text; static int b_width, b_height; static XtIntervalId b_timer; static unsigned long b_delay; static int b_screenWidth, b_screenHeight; static int b_lastShape; /*============================================================================ ============================================================================*/ static GC create_gc (Display* dpy, Window win, unsigned long fg, unsigned long bg, XFontStruct* fontStruct) { XGCValues gcv; unsigned long mask; gcv.foreground = fg; gcv.background = bg; gcv.font = fontStruct->fid; gcv.join_style = JoinMiter; gcv.line_width = BORDER_WIDTH; mask = GCFont | GCBackground | GCForeground | GCJoinStyle | GCLineWidth; return XCreateGC (dpy, win, mask, &gcv); } static void destroy_gc (Display* dpy, GC gc) { if (gc) { XFreeGC (dpy, gc); } } /*============================================================================ ============================================================================*/ static Window create_window (Display* dpy, unsigned long bg) { Window win; XSetWindowAttributes attr; unsigned long attr_mask; attr_mask = CWOverrideRedirect | CWBackPixel | CWSaveUnder; attr.override_redirect = True; attr.background_pixel = bg; attr.save_under = True; win = XCreateWindow (dpy, DefaultRootWindow (dpy), 0, 0, 1, 1, 0, CopyFromParent, InputOutput, CopyFromParent, attr_mask, &attr); XSelectInput (dpy, win, SubstructureRedirectMask | SubstructureNotifyMask | ExposureMask | EnterWindowMask | LeaveWindowMask); return win; } static void destroy_window (Display* dpy, Window win) { if (win) { XDestroyWindow (dpy, win); } } /*============================================================================ ============================================================================*/ static void get_pointer_xy (Display* dpy, int* x_return, int* y_return) { int dummy; unsigned int mask; Window dummy_win; XQueryPointer (dpy, RootWindow(dpy, DefaultScreen(dpy)), &dummy_win, &dummy_win, x_return, y_return, &dummy, &dummy, &mask); } /*============================================================================ ============================================================================*/ static void create_pixmap_mask (int width, int height) { b_maskWidth = width; b_maskHeight = height; b_mask = XCreatePixmap (b_dpy, b_win, width, height, 1); } static void destroy_pixmap_mask(void) { XFreePixmap (b_dpy, b_mask); } static void grow_pixmap_mask (int width, int height) { if (width > b_maskWidth || height > b_maskHeight) { destroy_pixmap_mask (); create_pixmap_mask (width, height); } } /*============================================================================ ============================================================================*/ static void text_extent (XFontStruct* fontStruct, const char* text, int len, int* width, int* height) { XCharStruct extent; int dummy; XTextExtents (fontStruct, text, len, &dummy, &dummy, &dummy, &extent); *width = extent.width; *height = fontStruct->ascent + fontStruct->descent; } static void get_text_size (Display* dpy, XFontStruct* fontStruct, const char* text, int* max_width, int* max_height) { int width; int height; const char* start; const char* end; *max_width = *max_height = 0; start = text; while ((end = strchr(start, '\n'))) { text_extent (fontStruct, start, end - start, &width, &height); *max_width = max (width, *max_width); *max_height += height; start = end + 1; } text_extent (fontStruct, start, strlen (start), &width, &height); *max_width = max (width, *max_width); *max_height += height; /* Min width */ *max_width = max (*max_width, CONE_WIDTH / 2 * 3); } static void draw_text (Display* dpy, Window win, GC gc, XFontStruct* fontStruct, int x, int y, const char* text) { const char* start; const char* end; int font_height; y += fontStruct->ascent; font_height = fontStruct->ascent + fontStruct->descent; start = text; while ((end = strchr(start, '\n'))) { XDrawString (dpy, win, gc, x, y, start, end - start); start = end + 1; y += font_height; } XDrawString (dpy, win, gc, x, y, start, strlen (start)); } /*============================================================================ ============================================================================*/ static int get_shape (int last_shape, int x, int y, int width, int height, int screen_width, int screen_height) { /* Can we use last_shape? */ if (((last_shape == SHAPE_CONE_TOP_LEFT) && (x + width < screen_width) && (y + height < screen_height)) || ((last_shape == SHAPE_CONE_TOP_RIGHT) && (x - width > 0) && (y + height < screen_height)) || ((last_shape == SHAPE_CONE_BOTTOM_LEFT) && (x + width < screen_width) && (y - height > 0)) || ((last_shape == SHAPE_CONE_BOTTOM_RIGHT) && (x - width > 0) && (y - height > 0))) return last_shape; /* Try to pick a shape that will not get changed, e.g. if top left quadrant, top_left */ return (x < screen_width / 2) ? (y < screen_height / 2 ? SHAPE_CONE_TOP_LEFT: SHAPE_CONE_BOTTOM_LEFT) : (y < screen_height / 2 ? SHAPE_CONE_TOP_RIGHT: SHAPE_CONE_BOTTOM_RIGHT); } static void make_mask (int shape, int x, int y, int width, int height) { XPoint cone[ 3 ]; grow_pixmap_mask (width, height); /* Clear mask */ XSetForeground (b_dpy, b_maskGC, 0); XFillRectangle (b_dpy, b_mask, b_maskGC, 0, 0, width, height); /* Enable text area */ XSetForeground (b_dpy, b_maskGC, 1); XFillRectangle (b_dpy, b_mask, b_maskGC, 0, shape & SHAPE_CONE_TOP ? CONE_HEIGHT : 0, width, height - CONE_HEIGHT); /* Enable for cone area */ cone[0].x = (shape & SHAPE_CONE_LEFT) ? CONE_WIDTH / 2 : width - (CONE_WIDTH / 2); cone[0].y = (shape & SHAPE_CONE_TOP) ? CONE_HEIGHT : height - CONE_HEIGHT; cone[1].x = (shape & SHAPE_CONE_LEFT) ? 0 : width; cone[1].y = (shape & SHAPE_CONE_TOP) ? 0 : height; cone[2].x = (shape & SHAPE_CONE_LEFT) ? CONE_WIDTH : width - CONE_WIDTH; cone[2].y = (shape & SHAPE_CONE_TOP) ? CONE_HEIGHT : height - CONE_HEIGHT; XFillPolygon (b_dpy, b_mask, b_maskGC, cone, 3, Nonconvex, CoordModeOrigin); } static void show_help (XtPointer data, XtIntervalId* id) { int x, y; int shape; XPoint border[ 3 ]; if (id == NULL || ((id && b_timer) && b_text)) { b_timer = None; /* size */ get_text_size (b_dpy, b_fontStruct, b_text, &b_width, &b_height); b_width += 2 * MARGIN_WIDTH + 2 * BORDER_WIDTH; b_height += 2 * MARGIN_WIDTH + 2 * BORDER_WIDTH + CONE_HEIGHT; /* origin */ get_pointer_xy (b_dpy, &x, &y); /* guess at shape */ shape = get_shape(b_lastShape, x, y, b_width, b_height, b_screenWidth, b_screenHeight); x += (shape & SHAPE_CONE_LEFT) ? POINTER_OFFSET : -POINTER_OFFSET; y += (shape & SHAPE_CONE_TOP) ? POINTER_OFFSET : -POINTER_OFFSET; /* make sure it is still ok with offset */ shape = get_shape (shape, x, y, b_width, b_height, b_screenWidth, b_screenHeight); b_lastShape = shape; make_mask (shape, x, y, b_width, b_height); XShapeCombineMask (b_dpy, b_win, ShapeBounding, 0, 0, b_mask, ShapeSet); XMoveResizeWindow(b_dpy, b_win, (shape & SHAPE_CONE_LEFT) ? x : x - b_width, (shape & SHAPE_CONE_TOP) ? y : y - b_height, b_width, b_height); XClearWindow (b_dpy, b_win); XMapRaised (b_dpy, b_win); b_winMapped = True; draw_text (b_dpy, b_win, b_gc, b_fontStruct, BORDER_WIDTH + MARGIN_WIDTH, BORDER_WIDTH + MARGIN_WIDTH + ((shape & SHAPE_CONE_TOP) ? CONE_HEIGHT : 0), b_text); /* 3d border */ /* shine- top left */ border[0].x = 0 + BORDER_WIDTH_HALF; border[0].y = ((shape & SHAPE_CONE_TOP) ? b_height : b_height - CONE_HEIGHT) - BORDER_WIDTH_HALF; border[1].x = 0 + BORDER_WIDTH_HALF; border[1].y = ((shape & SHAPE_CONE_TOP) ? CONE_HEIGHT : 0) + BORDER_WIDTH_HALF; border[2].x = b_width - BORDER_WIDTH_HALF; border[2].y = border[1].y; XDrawLines (b_dpy, b_win, b_shineGC, border, 3, CoordModeOrigin); /* shadow- bottom right */ border[0].x = 0 + BORDER_WIDTH_HALF; border[0].y = ((shape & SHAPE_CONE_TOP) ? b_height : b_height - CONE_HEIGHT) - BORDER_WIDTH_HALF; border[1].x = b_width - BORDER_WIDTH_HALF; border[1].y = border[0].y; border[2].x = b_width - BORDER_WIDTH_HALF; border[2].y = ((shape & SHAPE_CONE_TOP) ? CONE_HEIGHT : 0) + BORDER_WIDTH_HALF; XDrawLines (b_dpy, b_win, b_shadowGC, border, 3, CoordModeOrigin); /* cone */ if (SHAPE_CONE_TOP_LEFT == shape) { XClearArea (b_dpy, b_win, CONE_WIDTH / 2 + BORDER_WIDTH, CONE_HEIGHT, CONE_WIDTH / 2 - BORDER_WIDTH, BORDER_WIDTH, False); XDrawLine (b_dpy, b_win, b_shadowGC, 0, 0, CONE_WIDTH / 2 + BORDER_WIDTH_HALF, CONE_HEIGHT); XDrawLine (b_dpy, b_win, b_shineGC, 0, 0, CONE_WIDTH - BORDER_WIDTH_HALF, CONE_HEIGHT); } else if (SHAPE_CONE_TOP_RIGHT == shape) { XClearArea (b_dpy, b_win, b_width - CONE_WIDTH + BORDER_WIDTH, CONE_HEIGHT, CONE_WIDTH / 2 - BORDER_WIDTH, BORDER_WIDTH, False); XDrawLine (b_dpy, b_win, b_shadowGC, b_width, 0, b_width - CONE_WIDTH / 2 - BORDER_WIDTH_HALF, CONE_HEIGHT); XDrawLine (b_dpy, b_win, b_shineGC, b_width, 0, b_width - CONE_WIDTH + BORDER_WIDTH_HALF, CONE_HEIGHT); } else if (SHAPE_CONE_BOTTOM_LEFT == shape) { XClearArea (b_dpy, b_win, CONE_WIDTH / 2 + BORDER_WIDTH, b_height - CONE_HEIGHT - BORDER_WIDTH, CONE_WIDTH / 2 - BORDER_WIDTH, BORDER_WIDTH, False); XDrawLine (b_dpy, b_win, b_shadowGC, 0, b_height - 1, CONE_WIDTH, b_height - 1 - CONE_HEIGHT); XDrawLine (b_dpy, b_win, b_shineGC, 0, b_height - 1, CONE_WIDTH / 2 + BORDER_WIDTH, b_height - 1 - CONE_HEIGHT); } else if (SHAPE_CONE_BOTTOM_RIGHT == shape) { XClearArea (b_dpy, b_win, b_width - 1 - CONE_WIDTH + BORDER_WIDTH, b_height - CONE_HEIGHT - BORDER_WIDTH, CONE_WIDTH / 2 - BORDER_WIDTH - 1, BORDER_WIDTH, False); XDrawLine (b_dpy, b_win, b_shadowGC, b_width - 1, b_height - 1, b_width - 1 - CONE_WIDTH, b_height - 1 - CONE_HEIGHT); XDrawLine (b_dpy, b_win, b_shineGC, b_width - 1, b_height - 1, b_width - 1 - CONE_WIDTH / 2 - BORDER_WIDTH, b_height - 1 - CONE_HEIGHT); } } } /*============================================================================ ============================================================================*/ static void balloon_help_destroy (void) { assert (b_dpy != NULL); b_dpy = NULL; destroy_window (b_dpy, b_win); destroy_gc (b_dpy, b_gc); destroy_gc (b_dpy, b_shineGC); destroy_gc (b_dpy, b_shadowGC); destroy_pixmap_mask (); destroy_gc (b_dpy, b_maskGC); if (b_timer) XtRemoveTimeOut (b_timer); } void balloon_help_create (Display* dpy, Pixel fg, Pixel bg, Pixel shine, Pixel shadow, XFontStruct* font) { if (b_dpy) balloon_help_destroy (); b_dpy = dpy; b_fontStruct = font; b_win = create_window (dpy, bg); b_gc = create_gc (dpy, b_win, fg, bg, b_fontStruct); b_shineGC = create_gc (dpy, b_win, shine, bg, b_fontStruct); b_shadowGC = create_gc (dpy, b_win, shadow, bg, b_fontStruct); create_pixmap_mask (1, 1); b_maskGC = create_gc (dpy, b_mask, bg, fg, b_fontStruct); b_winMapped = False; b_timer = None; b_delay = 500; b_screenWidth = DisplayWidth (b_dpy, DefaultScreen(b_dpy)); b_screenHeight = DisplayHeight (b_dpy, DefaultScreen(b_dpy)); b_lastShape = SHAPE_CONE_FREE; } void balloon_help_set_delay (unsigned long milliseconds) { b_delay = milliseconds; } void balloon_help_show (const char* text) { assert (b_dpy != NULL); /* We don't copy the text */ b_text = text; b_lastShape = SHAPE_CONE_FREE; if (b_winMapped) { /* If help is already being shown, don't delay just update */ show_help (NULL, NULL); } else { b_timer = XtAppAddTimeOut (XtDisplayToApplicationContext(b_dpy), b_delay, show_help, NULL); } } void balloon_help_hide (void) { assert (b_dpy != NULL); b_text = NULL; XUnmapWindow (b_dpy, b_win); b_winMapped = False; if (b_timer) { XtRemoveTimeOut (b_timer); b_timer = None; } } void balloon_help_move_to_pointer (void) { assert (b_dpy != NULL); if (b_winMapped) { int x, y; int shape = b_lastShape; get_pointer_xy (b_dpy, &x, &y); x += (shape & SHAPE_CONE_LEFT) ? POINTER_OFFSET : -POINTER_OFFSET; y += (shape & SHAPE_CONE_TOP) ? POINTER_OFFSET : -POINTER_OFFSET; shape = get_shape (shape, x, y, b_width, b_height, b_screenWidth, b_screenHeight); if (shape == b_lastShape) { XMoveWindow (b_dpy, b_win, shape & SHAPE_CONE_LEFT ? x : x - b_width, shape & SHAPE_CONE_TOP ? y : y - b_height); } else { /* text would be off screen, rebuild with new shape */ b_lastShape = SHAPE_CONE_FREE; show_help (NULL, NULL); } } }