/* * (c) Copyright 1989, 1990, 1991, 1992 OPEN SOFTWARE FOUNDATION, INC. * ALL RIGHTS RESERVED */ /* * Motif Release 1.2 */ #ifdef REV_INFO #ifndef lint static char rcsid[] = "$RCSfile: DNDDemo.c,v $ $Revision: 1.2 $ $Date: 92/03/13 15:29:20 $" #endif #endif /* * (c) Copyright 1987, 1988, 1989 HEWLETT-PACKARD COMPANY */ /* * file: DNDDemo.c * * A demo program showing the basic Drag And Drop operations. */ #include "DNDDemo.h" /* * The folowing character arrays are for use with the drop help * dialogs. For internationalization, message catalogs should * replace these static declarations. */ char HELP_MSG1[] = "This drop action will change the color\n\ of the rectangle that the paint palette\n\ icon is dropped on. To accept this drop\n\ press the o.k. button, otherwise press\n\ cancel"; char HELP_MSG2[] = "This drop action will move the rectangle\n\ to the new position. To accept this drop\n\ press the o.k. button, otherwise press\n\ cancel"; char HELP_MSG3[] = "This drop action will copy the rectangle\n\ to the new position. To accept this drop\n\ press the o.k. button, otherwise press\n\ cancel"; char HELP_MSG4[] = "This drop action can either copy or\n\ move the rectangle to the new position.\n\ Select the operation that you desire.\n\ In the future, use Ctrl with Btn2 to\n\ perform copy operations. The default\n\ operation is move. To accept this drop\n\ press the o.k. button, otherwise press\n\ cancel"; char HELP_MSG5[] = "This drop action is at an Invalid drop\n\ position. Please cancel this drop \n\ by pressing the cancel button."; /* Globals variables */ Widget topLevel; Widget drawingArea; Widget helpDialog = NULL; Widget helpLabel, helpMenu; Widget myDC; XtAppContext appContext; /* This function creates the Drag Icon. */ static Widget #ifdef _NO_PROTO GetDragIcon(w, icon, iconMask, width, height, background, foreground) Widget w; Pixmap icon; Pixmap iconMask; Dimension width; Dimension height; Pixel background; Pixel foreground; #else GetDragIcon(Widget w, Pixmap icon, Pixmap iconMask, Dimension width, Dimension height, Pixel background, Pixel foreground) #endif /* _NO_PROTO */ { Widget dragIcon; Arg args[10]; int n = 0; XtSetArg(args[n], XmNhotX, ICON_X_HOT); n++; XtSetArg(args[n], XmNhotY, ICON_Y_HOT); n++; XtSetArg(args[n], XmNwidth, width); n++; XtSetArg(args[n], XmNheight, height); n++; XtSetArg(args[n], XmNmaxWidth, appInfo->maxCursorWidth); n++; XtSetArg(args[n], XmNmaxHeight, appInfo->maxCursorHeight); n++; XtSetArg(args[n], XmNbackground, background); n++; XtSetArg(args[n], XmNforeground, foreground); n++; XtSetArg(args[n], XmNpixmap, icon); n++; XtSetArg(args[n], XmNmask, iconMask); n++; dragIcon = XmCreateDragIcon(w, "dragIcon", args, n); return(dragIcon); } /* This function creates the bitmaps for the icon and the mask * and then calls GetDragIcon() to create the drag icon. */ static Widget #ifdef _NO_PROTO GetDragIconFromBits(w, bits, mask, width, height, background, foreground) Widget w; char *bits; char *mask; Dimension width; Dimension height; Pixel background; Pixel foreground; #else GetDragIconFromBits(Widget w, char *bits, char *mask, Dimension width, Dimension height, Pixel background, Pixel foreground) #endif /* _NO_PROTO */ { Pixmap icon, iconMask; Display *display = XtDisplay(w); icon = XCreateBitmapFromData(display, DefaultRootWindow(display), bits, width, height); iconMask = XCreateBitmapFromData(display, DefaultRootWindow(display), mask, width, height); return(GetDragIcon(w, icon, iconMask, width, height, background, foreground)); } /* This function creates the rectangle bitmaps for the icon and * the mask based on the maximum server allowable cursor size * and then calls GetDragIcon() to create the drag icon. */ static Widget #ifdef _NO_PROTO GetDragIconFromRect(w, rect, background) Widget w; RectPtr rect; Pixel background; #else GetDragIconFromRect(Widget w, RectPtr rect, Pixel background) #endif /* _NO_PROTO */ { Pixmap icon, icon_mask; Pixel foreground = RectGetColor(rect); Dimension width, height; /* Create a depth 1 pixmap (bitmap) for use with the drag icon */ icon = icon_mask = GetBitmapFromRect(w, rect, background, foreground, &width, &height); /* use bitmap for both the bitmap and mask */ return(GetDragIcon(w, icon, icon_mask, width, height, background, foreground)); } /* This is a selection conversion function that is used in * converting drag/drop export background color targets. * The return types follow ICCC statndards. */ /* ARGSUSED */ Boolean #ifdef _NO_PROTO ColorConvert(w, selection, target, type, value, length, format) Widget w ; Atom *selection ; Atom *target ; Atom *type ; XtPointer *value ; unsigned long *length ; int *format ; #else ColorConvert(Widget w, Atom *selection, Atom *target, Atom *type, XtPointer *value, unsigned long *length, int *format) #endif /* _NO_PROTO */ { Display *display = XtDisplay(w); Atom BACKGROUND = XmInternAtom(display, "BACKGROUND", False); Atom PIXEL = XmInternAtom(display, "PIXEL", False); Atom TARGETS = XmInternAtom(display, "TARGETS", False); Atom MULTIPLE = XmInternAtom(display, "MULTIPLE", False); Atom TIMESTAMP = XmInternAtom(display, "TIMESTAMP", False); int MAX_TARGS = 5; Widget widget; XtPointer client; Arg args[1]; /* get the widget that initiated the drag */ XtSetArg(args[0], XmNclientData, &client); XtGetValues(w, args, 1); widget = (Widget) client; /* Make sure we are doing a motif drag by checking if the widget that * is passed in is a drag context. Make sure the widget in the client * data is not NULL. */ if (!XmIsDragContext(w) || widget == NULL) return False; if (*target == BACKGROUND) { /* Get widget's background */ Pixel *background; background = (Pixel *) XtMalloc(sizeof(Pixel)); XtSetArg(args[0], XmNbackground, background); XtGetValues(widget, args, 1); /* value, type, length, and format must be set */ *value = (XtPointer) background; *type = PIXEL; *length = sizeof(Pixel); *format = 32; } else if (*target == TARGETS) { /* This target is required by ICCC */ Atom *targs = (Atom *)XtMalloc((unsigned) (MAX_TARGS * sizeof(Atom))); int target_count = 0; *value = (XtPointer) targs; *targs++ = BACKGROUND; target_count++; *targs++ = TARGETS; target_count++; *targs++ = MULTIPLE; target_count++; /* supported in the Intrinsics */ *targs++ = TIMESTAMP; target_count++; /* supported in the Intrinsics */ *type = XA_ATOM; *length = (target_count * sizeof(Atom)) >> 2; *format = 32; } else return False; return True; } /* This callback procedure resets the drag icon cursor to show * when the drag is in a valid region . It cause the * state icon to become visible when a drop is at a valid * position for drag over effects. */ static void #ifdef _NO_PROTO DragMotionCallback(w, client, call) Widget w; XtPointer client; XtPointer call; #else DragMotionCallback(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { XmDragMotionCallback cb = (XmDragMotionCallback) call; Arg args[2]; Widget stateIcon, invalidIcon; if (cb->dropSiteStatus == XmVALID_DROP_SITE) { stateIcon = (Widget) client; XtSetArg(args[0], XmNblendModel, XmBLEND_STATE_SOURCE); XtSetArg(args[1], XmNstateCursorIcon, stateIcon); XtSetValues(w, args, 2); } else if (cb->dropSiteStatus == XmINVALID_DROP_SITE) { XtSetArg(args[0], XmNdefaultInvalidCursorIcon, &invalidIcon); XtGetValues(XmGetXmScreen(XtScreen(w)), args, 1); XtSetArg(args[0], XmNblendModel, XmBLEND_STATE_SOURCE); XtSetArg(args[1], XmNstateCursorIcon, invalidIcon); XtSetValues(w, args, 2); } else { XtSetArg(args[0], XmNblendModel, XmBLEND_JUST_SOURCE); XtSetValues(w, args, 1); } } /* This callback procedure resets the drag icon cursor to show * when the drag is in a valid region . It cause the * state icon to become visible when a drop is at a valid * position for drag over effects. */ /* ARGSUSED */ static void #ifdef _NO_PROTO DropSiteLeaveCallback(w, client, call) Widget w; XtPointer client; XtPointer call; #else DropSiteLeaveCallback(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { Arg args[1]; XtSetArg(args[0], XmNblendModel, XmBLEND_JUST_SOURCE); XtSetValues(w, args, 1); } /* This callback procedure removes the icons when the drop is complete */ /* ARGSUSED */ static void #ifdef _NO_PROTO ColorDragDropFinishCB(w, client, call) Widget w; XtPointer client; XtPointer call; #else ColorDragDropFinishCB(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { Widget sourceIcon; Widget stateIcon = (Widget) client; Arg args[1]; XtSetArg(args[0], XmNsourceCursorIcon, &sourceIcon); XtGetValues(w, args, 1); XtDestroyWidget(sourceIcon); XtDestroyWidget(stateIcon); } /* This action procedure sets up the drag data and begins the drag operation */ /* ARGSUSED */ void #ifdef _NO_PROTO ColorRect(w, event, params, num_params) Widget w; XEvent *event; String *params; Cardinal *num_params; #else ColorRect(Widget w, XEvent *event, String *params, Cardinal *num_params) #endif /* _NO_PROTO */ { static XtCallbackRec dragDropFinishCB[] = { {ColorDragDropFinishCB, NULL}, {NULL, NULL} }; static XtCallbackRec dragMotionCB[] = { {DragMotionCallback, NULL}, {NULL, NULL} }; static XtCallbackRec dropSiteLeaveCB[] = { {DropSiteLeaveCallback, NULL}, {NULL, NULL} }; Atom targets[1]; Widget sourceIcon, stateIcon; Pixel background, foreground; char *source_bits, *source_mask; char *state_bits, *state_mask; Dimension width, height; Arg args[16]; int n = 0; n = 0; XtSetArg(args[n], XmNbackground, &background); n++; XtSetArg(args[n], XmNforeground, &foreground); n++; XtGetValues(w, args, n); /* If the server will handle a large icon, create one */ if (appInfo->maxCursorWidth >= ICON_WIDTH && appInfo->maxCursorHeight >= ICON_HEIGHT) { source_bits = SOURCE_ICON_BITS; source_mask = SOURCE_ICON_MASK; state_bits = STATE_ICON_BITS; state_mask = STATE_ICON_MASK; width = ICON_WIDTH; height = ICON_HEIGHT; } else { /* If the server will handle a small icon, create one */ source_bits = SMALL_SOURCE_ICON_BITS; source_mask = SMALL_SOURCE_ICON_MASK; state_bits = SMALL_STATE_ICON_BITS; state_mask = SMALL_STATE_ICON_MASK; width = SMALL_ICON_WIDTH; height = SMALL_ICON_HEIGHT; } /* Create the drag cursor icons */ sourceIcon = GetDragIconFromBits(w, source_bits, source_mask, width, height, background, foreground); stateIcon = GetDragIconFromBits(w, state_bits, state_mask, width, height, background, foreground); /* Setup the arglist for the drag context that is created at drag start */ n = 0; /* initially only show the source icon */ XtSetArg(args[n], XmNblendModel, XmBLEND_JUST_SOURCE); n++; /* set cursor colors for the drag states */ XtSetArg(args[n], XmNvalidCursorForeground, GetColor(VALID_CURSOR_FG_COLOR)); n++; XtSetArg(args[n], XmNinvalidCursorForeground, GetColor(INVALID_CURSOR_FG_COLOR)); n++; XtSetArg(args[n], XmNnoneCursorForeground, GetColor(NONE_CURSOR_FG_COLOR)); n++; /* set args for the drag cursor icons */ XtSetArg(args[n], XmNcursorBackground, background); n++; XtSetArg(args[n], XmNcursorForeground, foreground); n++; XtSetArg(args[n], XmNsourceCursorIcon, sourceIcon); n++; XtSetArg(args[n], XmNstateCursorIcon, stateIcon); n++; /* * set up the available export targets. These are targets that we * wish to provide data on */ targets[0] = XmInternAtom(XtDisplay(w), "BACKGROUND", False); XtSetArg(args[n], XmNexportTargets, targets); n++; XtSetArg(args[n], XmNnumExportTargets, 1); n++; /* * identify the conversion procedure and * the client data passed to the procedure */ XtSetArg(args[n], XmNclientData, w); n++; XtSetArg(args[n], XmNconvertProc, ColorConvert); n++; /* identify the necessary callbacks */ dragDropFinishCB[0].closure = (XtPointer) stateIcon; XtSetArg(args[n], XmNdragDropFinishCallback, dragDropFinishCB); n++; dragMotionCB[0].closure = (XtPointer) stateIcon; XtSetArg(args[n], XmNdragMotionCallback, dragMotionCB); n++; XtSetArg(args[n], XmNdropSiteLeaveCallback, dragMotionCB); n++; /* set the drag operations that are supported */ XtSetArg(args[n], XmNdragOperations, XmDROP_COPY); n++; /* start the drag. This creates a drag context. */ myDC = XmDragStart(w, event, args, n); } /* * This is a selection conversion function that is used in * converting requests for rectangle targets. The return types follow * ICCC statndards. */ /* ARGSUSED */ Boolean #ifdef _NO_PROTO RectConvert( w, selection, target, type, value, length, format) Widget w ; Atom *selection ; Atom *target ; Atom *type ; XtPointer *value ; unsigned long *length ; int *format ; #else RectConvert(Widget w, Atom *selection, Atom *target, Atom *type, XtPointer *value, unsigned long *length, int *format) #endif /* _NO_PROTO */ { Display *display = XtDisplay(w); Atom MY_RECT = XmInternAtom(display, "_MY_RECTANGLE", False); Atom RECT_INFO = XmInternAtom(display, "RECT_INFO", False); Atom DELETE = XmInternAtom(display, "DELETE", False); Atom TARGETS = XmInternAtom(display, "TARGETS", False); Atom MULTIPLE = XmInternAtom(display, "MULTIPLE", False); Atom TIMESTAMP = XmInternAtom(display, "TIMESTAMP", False); Atom *targs; int MAX_TARGS = 6; int target_count; DragConvertPtr conv; Widget widget; Arg args[1]; RectPtr rect, oldRect; /* get the widget that initiated the drag */ XtSetArg(args[0], XmNclientData, &conv); XtGetValues(w, args, 1); widget = (Widget) conv->widget; /* Make sure we are doing a motif drag by checking if the widget that * is passed in is a drag context. Make sure the widget in the client * data is not NULL. */ if (!XmIsDragContext(w) || widget == NULL) return False; if (*target == MY_RECT) { /* Create a new rectangle using information from the old retangle */ oldRect = conv->rect; /* We create create a new rectangle and wait for a delete target on the * old rectangle instead of just moving the old rectangle because * the rectangle movement might be an interclient move. */ rect = RectCreate(oldRect->x, oldRect->y, oldRect->width, oldRect->height, oldRect->color, oldRect->pixmap); /* value, type, length, and format must be assigned values */ *value = (XtPointer) rect; *type = RECT_INFO; *length = sizeof(Pixel); *format = 32; } else if (*target == DELETE) { /* delete the old rectangle */ RectHide(XtDisplay(widget), XtWindow(widget), conv->rect); RectFree(conv->rect); conv->rect = NULL; /* * DELETE target return parameters MUST be assigned as follows to * ICCC compliant. */ *value = NULL; *type = XmInternAtom(XtDisplay(w), "NULL", False); *length = 0; *format = 8; } else if (*target == TARGETS) { /* This target is required by ICCC */ targs = (Atom *)XtMalloc((unsigned) (MAX_TARGS * sizeof(Atom))); target_count = 0; *value = (XtPointer) targs; *targs++ = MY_RECT; target_count++; *targs++ = DELETE; target_count++; *targs++ = TARGETS; target_count++; *targs++ = MULTIPLE; target_count++; /* supported in the Intrinsics */ *targs++ = TIMESTAMP; target_count++; /* supported in the Intrinsics */ *type = XA_ATOM; *length = (target_count * sizeof(Atom)) >> 2; *format = 32; } else return False; return True; } /* This callback procedure removes the old cursor icon */ /* ARGSUSED */ static void #ifdef _NO_PROTO RectDragDropFinishCB(w, client, call) Widget w; XtPointer client; XtPointer call; #else RectDragDropFinishCB(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { DragConvertPtr conv = (DragConvertPtr) client; Widget sourceCursorIcon; Arg args[1]; XtSetArg(args[0], XmNsourceCursorIcon, &sourceCursorIcon); XtGetValues(w, args, 1); XtFree((char *) conv); XtDestroyWidget(sourceCursorIcon); } /* This callback procedure redraws the rectangles once the drop is completed */ /* ARGSUSED */ static void #ifdef _NO_PROTO RectDropFinishCB(w, client, call) Widget w; XtPointer client; XtPointer call; #else RectDropFinishCB(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { DragConvertPtr conv = (DragConvertPtr) client; appInfo->clearRect = NULL; appInfo->doMove = True; RedrawRectangles(conv->widget); } /* This callback procedure handle the drawing of the target * rectangle depending of the dropSiteStatus for drag over * effects. */ /* ARGSUSED */ static void #ifdef _NO_PROTO RectDragMotionCB(w, client, call) Widget w; XtPointer client; XtPointer call; #else RectDragMotionCB(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { XmDragMotionCallback cb = (XmDragMotionCallback) call; DragConvertPtr conv = (DragConvertPtr) client; Display *display; Window window; RectPtr rect; if (cb->dropSiteStatus == XmVALID_DROP_SITE) { /* re-stipple the rectangle when the poitner is inside the drop site */ if ( appInfo->clearRect == NULL && appInfo->doMove) { display = XtDisplay(conv->widget); window = XtWindow(conv->widget); rect = conv->rect; RectHide(display, window, rect); RectDrawStippled(display, window, rect); } } else { /* re-fill the rectangle when the poitner is outside the drop site */ if (appInfo->clearRect != NULL && appInfo->doMove) { appInfo->clearRect = NULL; RedrawRectangles(conv->widget); } } } /* This callback procedure handle the drawing of the target * rectangle When the operation changes. */ /* ARGSUSED */ static void #ifdef _NO_PROTO RectOperationChangedCB(w, client, call) Widget w; XtPointer client; XtPointer call; #else RectOperationChangedCB(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { XmDragMotionCallback cb = (XmDragMotionCallback) call; DragConvertPtr conv = (DragConvertPtr) client; Display *display; Window window; RectPtr rect; /* re-stipple the rectangle when the poitner is inside the drop site */ if ( appInfo->clearRect == NULL && appInfo->doMove) { display = XtDisplay(conv->widget); window = XtWindow(conv->widget); rect = conv->rect; RectHide(display, window, rect); RectDrawStippled(display, window, rect); } /* re-fill the rectangle when the operation changes to copy */ if (appInfo->clearRect != NULL && !appInfo->doMove) { appInfo->clearRect = NULL; RedrawRectangles(conv->widget); } } /* This action procedure sets up the drag data and begins the drag operation */ /* ARGSUSED */ static void #ifdef _NO_PROTO StartMove(w, event, params, num_params) Widget w; XEvent *event; String *params; Cardinal *num_params; #else StartMove(Widget w, XEvent *event, String *params, Cardinal *num_params) #endif /* _NO_PROTO */ { RectPtr rect; Position x = event->xbutton.x; Position y = event->xbutton.y; static XtCallbackRec dragDropFinishCB[] = { {RectDragDropFinishCB, NULL}, {NULL, NULL} }; static XtCallbackRec dropFinishCB[] = { {RectDropFinishCB, NULL}, {NULL, NULL} }; static XtCallbackRec dragMotionCB[] = { {RectDragMotionCB, NULL}, {NULL, NULL} }; static XtCallbackRec operationChangedCB[] = { {RectOperationChangedCB, NULL}, {NULL, NULL} }; Atom targets[1]; Display *display = XtDisplay(w); Widget sourceCursorIcon; DragConvertPtr conv; Pixel background, foreground; Arg args[16]; int n = 0; /* find a rectangle at the given x,y position */ rect = RectFind(x, y); /* start move only if it begins on a rectangle */ if (rect) { XtSetArg(args[0], XmNbackground, &background); XtGetValues(w, args, 1); foreground = RectGetColor(rect); sourceCursorIcon = GetDragIconFromRect(w, rect, background); /* * Set up information to pass to the convert * function and callback procs. */ conv = (DragConvertPtr) XtMalloc(sizeof(DragConvertRec)); conv->widget = w; conv->rect = rect; /* On a move operation, draw the current * rectangle as a stippled outline. */ if (!(event->xbutton.state & ControlMask)) { RectHide(display, XtWindow(w), rect); RectDrawStippled(display, XtWindow(w), rect); } else appInfo->doMove = False; /* Setup arglist for the drag context that is created at drag start */ n = 0; /* initially only show the source icon */ XtSetArg(args[n], XmNblendModel, XmBLEND_JUST_SOURCE); n++; /* set args for the drag cursor icons */ XtSetArg(args[n], XmNcursorBackground, background); n++; XtSetArg(args[n], XmNcursorForeground, foreground); n++; XtSetArg(args[n], XmNsourceCursorIcon, sourceCursorIcon); n++; /* * set up the available export targets. These are targets that we * wish to provide data on */ targets[0] = XmInternAtom(display, "_MY_RECTANGLE", False); XtSetArg(args[n], XmNexportTargets, targets); n++; XtSetArg(args[n], XmNnumExportTargets, 1); n++; /* * identify the conversion procedure and * the client data passed to the procedure */ XtSetArg(args[n], XmNclientData, conv); n++; XtSetArg(args[n], XmNconvertProc, RectConvert); n++; /* identify the necessary callbacks and the client data to be passed */ dragDropFinishCB[0].closure = (XtPointer) conv; XtSetArg(args[n], XmNdragDropFinishCallback, dragDropFinishCB); n++; dropFinishCB[0].closure = (XtPointer) conv; XtSetArg(args[n], XmNdropFinishCallback, dropFinishCB); n++; dragMotionCB[0].closure = (XtPointer) conv; XtSetArg(args[n], XmNdragMotionCallback, dragMotionCB); n++; operationChangedCB[0].closure = (XtPointer) conv; XtSetArg(args[n], XmNoperationChangedCallback, operationChangedCB); n++; /* set the drag operations that are supported */ XtSetArg(args[n], XmNdragOperations, XmDROP_COPY | XmDROP_MOVE); n++; /* start the drag. This creates a drag context. */ myDC = XmDragStart(w, event, args, n); } } /* This procedure searches through the export targets and * returns flags to indicate which targets were found */ /* ARGSUSED */ static void #ifdef _NO_PROTO CheckTargets(w, display, rectFound, bgFound, pixFound) Widget w; Display *display; Boolean *rectFound; Boolean *bgFound; Boolean *pixFound; #else CheckTargets(Widget w, Display *display, Boolean *rectFound, Boolean *bgFound, Boolean *pixFound) #endif /* _NO_PROTO */ { Atom MY_RECT = XmInternAtom(display, "_MY_RECTANGLE", False); Atom BACKGROUND = XmInternAtom(display, "BACKGROUND", False); Atom PIXMAP = XmInternAtom(display, "PIXMAP", False); Atom *exportTargets; Cardinal numExportTargets; Arg args[2]; int n; /* Get list of transfer targets */ n = 0; XtSetArg(args[0], XmNexportTargets, &exportTargets); XtSetArg(args[1], XmNnumExportTargets, &numExportTargets); XtGetValues(w, args, 2); /* initialize targets found flags */ *rectFound = *bgFound = *pixFound = False; /* search through the export targets */ for (n = 0; n < numExportTargets; n++) { if (exportTargets[n] == MY_RECT) *rectFound = True; else if (exportTargets[n] == BACKGROUND) *bgFound = True; else if (exportTargets[n] == PIXMAP) *pixFound = True; } } /* This procedure handles drop site messages and performs the * appropriate drag under effects. */ /* ARGSUSED */ static void #ifdef _NO_PROTO DragProcCallback(w, client, call) Widget w; XtPointer client; XtPointer call; #else DragProcCallback(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { XmDragProcCallbackStruct *cb = (XmDragProcCallbackStruct *) call; Display *display = XtDisplay(w); Boolean rectFound, bgFound, pixFound; static unsigned char initial_operations; static unsigned char initial_operation; RectPtr rect; CheckTargets(cb->dragContext, display, &rectFound, &bgFound, &pixFound); switch(cb->reason) { case XmCR_DROP_SITE_ENTER_MESSAGE: /* save the value of the operations and operation fields */ initial_operations = cb->operations; initial_operation = cb->operation; rect = RectFind(cb->x, cb->y); /* Remove any operations for the operations field which do not * apply to the simulated drop site. */ if (rect) { if (bgFound || pixFound) { cb->operations = XmDROP_COPY; RectHighlight(w, rect); } else if (rectFound) { cb->operations = cb->operations & (XmDROP_COPY | XmDROP_MOVE); RectUnhighlight(w); } } else { cb->operations = initial_operations & (XmDROP_COPY | XmDROP_MOVE); RectUnhighlight(w); } /* Set operation to the valid operation preferred by the simulated * drop site or to XmDROP_NOOP if the operations list does not * contain the preferred operation. */ if (rect) { if (bgFound || pixFound) { if (cb->operations & XmDROP_COPY) cb->operation = XmDROP_COPY; else cb->operation = XmDROP_NOOP; } else if (rectFound) { if (cb->operations & XmDROP_MOVE) cb->operation = XmDROP_MOVE; else if (cb->operations & XmDROP_COPY) cb->operation = XmDROP_COPY; else cb->operation = XmDROP_NOOP; } } else { if (rectFound) { if (cb->operations & XmDROP_MOVE) cb->operation = XmDROP_MOVE; else if (cb->operations & XmDROP_COPY) cb->operation = XmDROP_COPY; else cb->operation = XmDROP_NOOP; } else cb->operation = initial_operation; } /* * Set dropSiteStatus to XmDROP_SITE_INVALID if the operation * field is XmDROP_NOOP, or if there are no common targets * between the source and the nested drop site. Otherwise, set * dropSiteStatus to XmDROP_SITE_VALID. */ if (cb->operation == XmDROP_NOOP || (rect && (!rectFound && !bgFound && !pixFound)) || (!rect && !rectFound)) cb->dropSiteStatus = XmINVALID_DROP_SITE; else cb->dropSiteStatus = XmVALID_DROP_SITE; /* * Display appropriate drag under visuals. Only highlight * the rectangle if we are changing rectangle attributes. */ if (rect && bgFound || pixFound && cb->dropSiteStatus == XmVALID_DROP_SITE) RectHighlight(w, rect); break; case XmCR_DROP_SITE_LEAVE_MESSAGE: /* Only unhighlight the rectangle if previously highlighted */ if (appInfo->highlightRect != NULL) RectUnhighlight(w); break; case XmCR_DROP_SITE_MOTION_MESSAGE: rect = RectFind(cb->x, cb->y); /* * Remove any operations for the operations field which do not * apply to the simulated drop site. */ if (rect) { if (bgFound || pixFound) { cb->operations = XmDROP_COPY; RectHighlight(w, rect); } else if (rectFound) { cb->operations = cb->operations & (XmDROP_COPY | XmDROP_MOVE); RectUnhighlight(w); } } else { cb->operations = initial_operations & (XmDROP_COPY | XmDROP_MOVE); RectUnhighlight(w); } /* * Set operation to the valid operation preferred by the simulated * drop site or to XmDROP_NOOP if the operations list does not * contain the preferred operation. */ if (rect) { if (bgFound || pixFound) { if (cb->operations & XmDROP_COPY) cb->operation = XmDROP_COPY; else cb->operation = XmDROP_NOOP; } else if (rectFound) { if (cb->operations & XmDROP_MOVE) cb->operation = XmDROP_MOVE; else if (cb->operations & XmDROP_COPY) cb->operation = XmDROP_COPY; else cb->operation = XmDROP_NOOP; } } else { if (rectFound) { if (cb->operations & XmDROP_MOVE) cb->operation = XmDROP_MOVE; else if (cb->operations & XmDROP_COPY) cb->operation = XmDROP_COPY; else cb->operation = XmDROP_NOOP; } else cb->operation = initial_operation; } /* * Set dropSiteStatus to XmDROP_SITE_INVALID if the operation * field is XmDROP_NOOP, or if there are no common targets * between the source and the nested drop site. Otherwise, * set dropSiteStatus to XmDROP_SITE_VALID. */ if (cb->operation == XmDROP_NOOP || (rect && (!rectFound && !bgFound && !pixFound)) || (!rect && !rectFound)) cb->dropSiteStatus = XmINVALID_DROP_SITE; else cb->dropSiteStatus = XmVALID_DROP_SITE; /* * Display appropriate drag under visuals. Only highlight * the rectangle if we are changing rectangle attributes. */ if (rect && bgFound || pixFound && cb->dropSiteStatus == XmVALID_DROP_SITE) RectHighlight(w, rect); break; case XmCR_OPERATION_CHANGED: if (rectFound) { if (cb->operation == XmDROP_MOVE) appInfo->doMove = True; else appInfo->doMove = False; } break; default: /* other messages we consider invalid */ cb->dropSiteStatus = XmINVALID_DROP_SITE; break; } /* allow animation to be performed */ cb->animate = True; } /* This procedure handles the data that is being transfer */ /* ARGSUSED */ static void #ifdef _NO_PROTO TransferProcCallback(w, closure, seltype, type, value, length, format) Widget w; XtPointer closure ; Atom *seltype ; Atom *type ; XtPointer value ; unsigned long *length ; int *format ; #else TransferProcCallback(Widget w, XtPointer closure, Atom *seltype, Atom *type, XtPointer value, unsigned long *length, int *format) #endif /* _NO_PROTO */ { DropTransfer transferRec = (DropTransfer) closure; Widget wid = transferRec->widget; Display *display = XtDisplay(wid); Atom RECT_INFO = XmInternAtom(display, "RECT_INFO", False); Atom PIXEL = XmInternAtom(display, "PIXEL", False); Atom NULL_ATOM = XmInternAtom(display, "NULL", False); Arg args[10]; RectPtr rect; int n; /* * The delete target returns a NULL_ATOM type and value equal to NULL * so it isn't a failure. Otherwise, check for NULL value or targets * that wee don't support and set transfer failure. */ if (*type != NULL_ATOM && (!value || (*type != RECT_INFO && *type != PIXEL && *type != XA_DRAWABLE))) { n = 0; /* * On failures set both transferStatus to XmTRANSFER_FAILURE and * numDropTransfers to 0. */ XtSetArg(args[n], XmNtransferStatus, XmTRANSFER_FAILURE); n++; XtSetArg(args[n], XmNnumDropTransfers, 0); n++; XtSetValues(w, args, n); /* Free the value if there is one, or we would have a memory leak */ if (value) XtFree(value); return; } /* Handle pixel type (i.e. change in background) */ if (*type == PIXEL) { rect = RectFind(transferRec->x, transferRec->y); RectSetColor(rect, display, XtWindow(wid), *((Pixel*)value)); } /* Handle drawable type (i.e. change in pixmap) */ else if (*type == XA_DRAWABLE) { rect = RectFind(transferRec->x, transferRec->y); RectSetPixmap(rect, display, XtWindow(wid), *((Pixmap *)value)); } /* Handle rect_info type (i.e. new rectangle) */ else if (*type == RECT_INFO) { rect = (RectPtr) value; RectRegister(rect, transferRec->x, transferRec->y); value = NULL; /* No need to free, it is being stored in RecTable */ } /* Free the value if there is one, or we would have a memory leak */ if (value) XtFree(value); } /* This procedure frees the data used the data transfer proc that * was passed from the drop procedure. */ /* ARGSUSED */ static void #ifdef _NO_PROTO DropDestroyCB(w, clientData, callData) Widget w; XtPointer clientData; XtPointer callData; #else DropDestroyCB(Widget w, XtPointer clientData, XtPointer callData) #endif /* NO_PROTO */ { XtFree((char *)clientData); } /* This procedure initiates the drop transfer. */ /* ARGSUSED */ static void #ifdef _NO_PROTO HandleDrop(w, call) Widget w; XtPointer call; #else HandleDrop(Widget w, XtPointer call) #endif /* _NO_PROTO */ { static XtCallbackRec dropDestroyCB[] = { {DropDestroyCB, NULL}, {NULL, NULL} }; XmDropProcCallbackStruct *cb = (XmDropProcCallbackStruct *)call; Display *display = XtDisplay(w); Arg args[10]; int n; Boolean rectFound, bgFound, pixFound; DropTransfer transferRec; XmDropTransferEntryRec transferEntries[2]; XmDropTransferEntryRec *transferList = NULL; Cardinal numTransfers = 0; Boolean transferValid = False; RectPtr rect; /* Cancel the drop on invalid drop operations */ if (!(cb->operations & XmDROP_MOVE || cb->operations & XmDROP_COPY)) { n = 0; cb->operation = XmDROP_NOOP; cb->dropSiteStatus = XmINVALID_DROP_SITE; XtSetArg(args[n], XmNtransferStatus, XmTRANSFER_FAILURE); n++; XtSetArg(args[n], XmNnumDropTransfers, 0); n++; } else { /* Find out which nested dropsite contains the pointer */ rect = RectFind(cb->x, cb->y); CheckTargets(cb->dragContext, display, &rectFound, &bgFound, &pixFound); /* rect !NULL indicates we are within a nested dropsite */ if (rect) { /* MY_RECT is a possible target, we'll support it first */ if (rectFound) transferValid = True; else if (bgFound || pixFound) { /* support only copy with the BACKGROUND and PIXMAP targets */ if (cb->operation != XmDROP_COPY) cb->operation = XmDROP_COPY; transferValid = True; } } else { if (rectFound) transferValid = True; } if (transferValid) { /* intialize data to send to drop transfer callback */ transferRec = (DropTransfer) XtMalloc(sizeof(DropTransferRec)); transferRec->widget = w; transferRec->x = cb->x; transferRec->y = cb->y; /* order of support is MY_RECT, then BACKGROUND, then PIXMAP */ if (rectFound) transferEntries[0].target = XmInternAtom(display, "_MY_RECTANGLE", False); else if (bgFound) transferEntries[0].target = XmInternAtom(display, "BACKGROUND", False); else if (pixFound) transferEntries[0].target = XmInternAtom(display, "PIXMAP", False); transferEntries[0].client_data = (XtPointer) transferRec; /* Set up move targets */ if (cb->operation == XmDROP_MOVE) { transferEntries[1].client_data = (XtPointer) transferRec; transferEntries[1].target = XmInternAtom(display, "DELETE", False); numTransfers = 2; } else if (cb->operation == XmDROP_COPY) numTransfers = 1; transferList = transferEntries; /* Setup transfer list */ n = 0; cb->dropSiteStatus = XmVALID_DROP_SITE; XtSetArg(args[n], XmNdropTransfers, transferList); n++; XtSetArg(args[n], XmNnumDropTransfers, numTransfers); n++; /* Setup destroy callback to free transferRec */ dropDestroyCB[0].closure = (XtPointer) transferRec; XtSetArg(args[n], XmNdestroyCallback, dropDestroyCB); n++; /* Setup transfer proc to accept the drop transfer data */ XtSetArg(args[n], XmNtransferProc, TransferProcCallback); n++; } else { n = 0; cb->operation = XmDROP_NOOP; cb->dropSiteStatus = XmINVALID_DROP_SITE; XtSetArg(args[n], XmNtransferStatus, XmTRANSFER_FAILURE); n++; XtSetArg(args[n], XmNnumDropTransfers, 0); n++; } } XmDropTransferStart(cb->dragContext, args, n); } /* This procedure is used with the drop help dialog to continue with the drop */ /* ARGSUSED */ static void #ifdef _NO_PROTO HandleOK(w, client, call) Widget w; XtPointer client; XtPointer call; #else HandleOK(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { XmDropProcCallbackStruct *cb = (XmDropProcCallbackStruct *)client; cb->operation = appInfo->operation; HandleDrop(w, (XtPointer) cb); } /* This procedure is used with the drop help dialog to cancel the drop */ /* ARGSUSED */ static void #ifdef _NO_PROTO CancelDrop(w, client, call) Widget w; XtPointer client; XtPointer call; #else CancelDrop(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { XmDropProcCallbackStruct *cb = (XmDropProcCallbackStruct *)client; Arg args[2]; /* On help, we need to cancel the drop transfer */ XtSetArg(args[0], XmNtransferStatus, XmTRANSFER_FAILURE); XtSetArg(args[1], XmNnumDropTransfers, 0); /* we need to start the drop transfer to cancel the transfer */ XmDropTransferStart(cb->dragContext, args, 2); } #ifdef _NO_PROTO void ChangeOperation(widget, client_data, call_data) Widget widget; XtPointer client_data; XmAnyCallbackStruct *call_data; #else XtCallbackProc ChangeOperation(Widget widget, XtPointer client_data, XmAnyCallbackStruct *call_data) #endif { if (client_data == 0) appInfo->operation = XmDROP_MOVE; else appInfo->operation = XmDROP_COPY; } /* This procedure manages the help dialog and determines which * message is displayed in the dialog depending on the position * and the type of drop. */ /* ARGSUSED */ static void #ifdef _NO_PROTO HandleHelp(w, call) Widget w; XtPointer call; #else HandleHelp(Widget w, XtPointer call) #endif /* _NO_PROTO */ { XmDropProcCallbackStruct *cb = (XmDropProcCallbackStruct *)call; static XmDropProcCallbackStruct client; Boolean rectFound, bgFound, pixFound; XmString helpStr; RectPtr rect; Arg args[5]; XmString tempStr, buttonArray[2]; int n = 0; /* the drop is valid until it is determined invalid */ cb->dropSiteStatus = XmVALID_DROP_SITE; /* if we haven't created a help dialog, create one now */ if (helpDialog == NULL) { XtSetArg(args[n], XmNdialogStyle, XmDIALOG_FULL_APPLICATION_MODAL); n++; XtSetArg(args[n], XmNtitle, "Drop Help"); n++; helpDialog = XmCreateMessageDialog(topLevel, "Help", args, n); n = 0; buttonArray[0] = XmStringCreateSimple("Move"); buttonArray[1] = XmStringCreateSimple("Copy"); XtSetArg(args[n], XmNbuttons, buttonArray); n++; XtSetArg(args[n], XmNbuttonCount, 2); n++; XtSetArg(args[n], XmNbuttonSet, 0); n++; XtSetArg(args[n], XmNsimpleCallback, ChangeOperation); n++; tempStr = XmStringCreateSimple("Operations:"); XtSetArg(args[n], XmNoptionLabel, tempStr); n++; helpMenu = XmCreateSimpleOptionMenu(helpDialog, "helpMenu", args, n); XmStringFree(tempStr); XmStringFree(buttonArray[0]); XmStringFree(buttonArray[1]); XtAddCallback(helpDialog, XmNokCallback, (XtCallbackProc) HandleOK, (XtPointer) &client); XtAddCallback(helpDialog, XmNcancelCallback, (XtCallbackProc) CancelDrop, (XtPointer) &client); XtUnmanageChild(XmMessageBoxGetChild(helpDialog, XmDIALOG_HELP_BUTTON)); XtRealizeWidget(helpDialog); } /* pass the necessary callback information along */ client.dragContext = cb->dragContext; client.x = cb->x; client.y = cb->y; client.dropSiteStatus = cb->dropSiteStatus; client.operation = cb->operation; client.operations = cb->operations; /* find the valid targets */ CheckTargets(cb->dragContext, XtDisplay(w), &rectFound, &bgFound, &pixFound); /* determine the appropriate help message */ if (rectFound) { if (cb->operations == XmDROP_MOVE | XmDROP_COPY) { XtManageChild(helpMenu); helpStr = XmStringCreateLtoR(HELP_MSG4, XmFONTLIST_DEFAULT_TAG); XtManageChild(XmMessageBoxGetChild(helpDialog, XmDIALOG_OK_BUTTON)); } else if (cb->operation == XmDROP_MOVE) { XtUnmanageChild(helpMenu); helpStr = XmStringCreateLtoR(HELP_MSG2, XmFONTLIST_DEFAULT_TAG); XtManageChild(XmMessageBoxGetChild(helpDialog, XmDIALOG_OK_BUTTON)); } else if (cb->operation == XmDROP_COPY) { XtUnmanageChild(helpMenu); helpStr = XmStringCreateLtoR(HELP_MSG3, XmFONTLIST_DEFAULT_TAG); XtManageChild(XmMessageBoxGetChild(helpDialog, XmDIALOG_OK_BUTTON)); } } else if (bgFound || pixFound && cb->operation == XmDROP_COPY) { XtUnmanageChild(helpMenu); rect = RectFind(cb->x, cb->y); if (rect) { helpStr = XmStringCreateLtoR(HELP_MSG1, XmFONTLIST_DEFAULT_TAG); XtManageChild(XmMessageBoxGetChild(helpDialog, XmDIALOG_OK_BUTTON)); } else { helpStr = XmStringCreateLtoR(HELP_MSG5, XmFONTLIST_DEFAULT_TAG); XtUnmanageChild(XmMessageBoxGetChild(helpDialog, XmDIALOG_OK_BUTTON)); } } else { XtUnmanageChild(helpMenu); helpStr = XmStringCreateLtoR(HELP_MSG5, XmFONTLIST_DEFAULT_TAG); XtUnmanageChild(XmMessageBoxGetChild(helpDialog, XmDIALOG_OK_BUTTON)); } /* set the help message into the dialog */ XtSetArg(args[0], XmNmessageString, helpStr); XtSetValues(helpDialog, args, 1); /* Free the XmString */ XmStringFree(helpStr); /* map the help dialog */ XtManageChild(helpDialog); } /* The procedure either begins the drop of initiates the help dialog * depending on the dropAction. */ /* ARGSUSED */ static void #ifdef _NO_PROTO DropProcCallback(w, client, call) Widget w; XtPointer client; XtPointer call; #else DropProcCallback(Widget w, XtPointer client, XtPointer call) #endif /* _NO_PROTO */ { XmDropProcCallbackStruct *cb = (XmDropProcCallbackStruct *)call; if (appInfo->highlightRect != NULL) RectUnhighlight(w); if (cb->dropAction != XmDROP_HELP) HandleDrop(w, call); else HandleHelp(w, call); } /* This procedure registers the drop taragets and the drop site */ static void #ifdef _NO_PROTO RegisterDropSite(w) Widget w; #else RegisterDropSite(Widget w) #endif /* _NO_PROTO */ { Display *display = XtDisplay(w); Atom targets[3]; Arg args[5]; int n = 0; /* Only accept moves or copies */ XtSetArg(args[n], XmNdragOperations, XmDROP_COPY | XmDROP_MOVE); n++; /* set all possible targets for any of the nested drop sites */ targets[0] = XmInternAtom(display, "_MY_RECTANGLE", False); targets[1] = XmInternAtom(display, "BACKGROUND", False); targets[2] = XmInternAtom(display, "PIXMAP", False); XtSetArg(args[n], XmNimportTargets, targets); n++; XtSetArg(args[n], XmNnumImportTargets, 3); n++; /* register a dragProc - necessary for simulating nested drop sites */ XtSetArg(args[n], XmNdragProc, DragProcCallback); n++; /* register a dropProc */ XtSetArg(args[n], XmNdropProc, DropProcCallback); n++; XmDropSiteRegister(w, args, n); } static void #ifdef _NO_PROTO SetInvalidIcon(w) Widget w; #else SetInvalidIcon(Widget w) #endif /* _NO_PROTO */ { Widget invalidIcon; char *invalid_bits; Dimension width, height; Arg args[1]; if (appInfo->maxCursorWidth >= ICON_WIDTH && appInfo->maxCursorHeight >= ICON_HEIGHT) { invalid_bits = INVALID_ICON_BITS; width = ICON_WIDTH; height = ICON_HEIGHT; } else { /* If the server will handle a small icon, create one */ invalid_bits = SMALL_INVALID_ICON_BITS; width = SMALL_ICON_WIDTH; height = SMALL_ICON_HEIGHT; } invalidIcon = GetDragIconFromBits(w, invalid_bits, invalid_bits, width, height, GetColor(DRAW_AREA_FG_COLOR), GetColor(DRAW_AREA_BG_COLOR)); XtSetArg(args[0], XmNdefaultInvalidCursorIcon, invalidIcon); XtSetValues(XmGetXmScreen(XtScreen(w)), args, 1); } /* This procedure initalizes the toolkit and other application information */ static void #ifdef _NO_PROTO InitializeApplication(argc, argv) int *argc; String *argv; #else InitializeApplication(int *argc, String *argv) #endif /* _NO_PROTO */ { static XtActionsRec new_actions[] = { {"StartRect", StartRect}, {"ExtendRect", ExtendRect}, {"EndRect", EndRect}, {"StartMove", StartMove}, {"ColorRect", ColorRect}, }; Arg args[5]; Cardinal n = 0; /* Ininialize struct that hold global information */ InitializeAppInfo(); /* Initialize Toolkit and create shell */ XtSetArg(args[n], XmNwidth, 295); n++; XtSetArg(args[n], XmNheight, 270); n++; topLevel = XtAppInitialize(&appContext, "DNDDemo", NULL, 0, argc, argv, NULL, args, n); /* Set drag protocol styles */ n = 0; XtSetArg(args[n], XmNdragInitiatorProtocolStyle, XmDRAG_PREFER_RECEIVER); n++; XtSetArg(args[n], XmNdragReceiverProtocolStyle, XmDRAG_DYNAMIC); n++; XtSetValues(XmGetXmDisplay(XtDisplay(topLevel)), args, n); /* Initialize tables for holding rectangle information */ InitializeRectDpyTable(); /* Add new actions for use with translation tables */ XtAppAddActions(appContext, new_actions, 5); /* Get the display server's best cursor size */ XQueryBestCursor(XtDisplay(topLevel), RootWindowOfScreen(XtScreen(topLevel)), 64, 64, &appInfo->maxCursorWidth, &appInfo->maxCursorHeight); } /* This the program start procedure */ #ifdef DEC_MOTIF_BUG_FIX int #else void #endif #ifdef _NO_PROTO main (argc, argv) int argc; String *argv; #else main (int argc, String *argv) #endif /* _NO_PROTO */ { /* Initialize toolkit and application global values */ InitializeApplication(&argc, argv); /* Create main window, drawing area, and color labels */ CreateLayout(); /* Register the drawing area as a drop site */ RegisterDropSite(drawingArea); SetInvalidIcon(drawingArea); /* Realize and map widget hiearchy */ XtRealizeWidget(topLevel); /* Create GC for drawing rectangles */ CreateRectGC(); /* Begin event loop processing */ XtAppMainLoop(appContext); }