This commit was manufactured by cvs2svn to create branch 'chise-r21-4-19'.
[chise/xemacs-chise.git] / lwlib / lwlib-Xlw.c
diff --git a/lwlib/lwlib-Xlw.c b/lwlib/lwlib-Xlw.c
new file mode 100644 (file)
index 0000000..1e909b6
--- /dev/null
@@ -0,0 +1,628 @@
+/* The lwlib interface to "xlwmenu" menus.
+   Copyright (C) 1992, 1994 Lucid, Inc.
+
+This file is part of the Lucid Widget Library.
+
+The Lucid Widget Library 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.
+
+The Lucid Widget Library 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.  */
+
+#include <config.h>
+#include <stdlib.h> /* for abort () */
+#include <stdio.h> /* for abort () */
+#include <limits.h>
+
+#include "lwlib-Xlw.h"
+#include "lwlib-utils.h"
+#include <X11/StringDefs.h>
+#include <X11/IntrinsicP.h>
+#include <X11/ObjectP.h>
+#include <X11/CompositeP.h>
+#include <X11/Shell.h>
+#ifdef HAVE_WIDGETS
+#include "../src/EmacsManager.h"
+#endif
+#ifdef LWLIB_MENUBARS_LUCID
+#include "xlwmenu.h"
+#endif
+#ifdef LWLIB_SCROLLBARS_LUCID
+#include "xlwscrollbar.h"
+#endif
+#ifdef LWLIB_TABS_LUCID
+#ifdef NEED_MOTIF
+#include "lwlib-Xm.h"
+#endif
+#ifdef NEED_ATHENA
+#include "lwlib-Xaw.h"
+#endif
+#include "../src/xmu.h"
+#include "xlwtabs.h"
+#endif
+\f
+
+#include <X11/Intrinsic.h>
+#include <X11/StringDefs.h>
+#include <X11/Shell.h>
+#include <X11/Xaw/Form.h>
+#include <X11/Xaw/Command.h>
+
+
+#ifdef LWLIB_MENUBARS_LUCID
+
+/* Menu callbacks */
+
+static void
+pre_hook (Widget w, XtPointer client_data, XtPointer call_data)
+{
+  widget_instance* instance = (widget_instance*)client_data;
+  widget_value* val;
+  
+  if (w->core.being_destroyed)
+    return;
+
+  val = lw_get_widget_value_for_widget (instance, w);
+#if 0
+  /* #### - this code used to (for some random back_asswards reason) pass
+  the expression below in the call_data slot.  For incremental menu
+  construction, this needs to go.  I can't even figure out why it was done
+  this way in the first place...it's just a historical weirdism. --Stig */
+  call_data = (val ? val->call_data : NULL);
+#endif 
+  if (val && val->call_data)
+    abort();                   /* #### - the call_data for the top_level
+                                  "menubar" widget_value used to be passed
+                                  back to the pre_hook. */
+
+  if (instance->info->pre_activate_cb)
+    instance->info->pre_activate_cb (w, instance->info->id, call_data);
+}
+
+static void
+pick_hook (Widget w, XtPointer client_data, XtPointer call_data)
+{
+  widget_instance* instance = (widget_instance*)client_data;
+  widget_value* contents_val = (widget_value*)call_data;
+  widget_value* widget_val;
+  XtPointer widget_arg;
+  LWLIB_ID id;
+  lw_callback post_activate_cb;
+
+  if (w->core.being_destroyed)
+    return;
+
+  /* Grab these values before running any functions, in case running
+     the selection_cb causes the widget to be destroyed. */
+  id = instance->info->id;
+  post_activate_cb = instance->info->post_activate_cb;
+
+  widget_val = lw_get_widget_value_for_widget (instance, w);
+  widget_arg = widget_val ? widget_val->call_data : NULL;
+
+  if (instance->info->selection_cb &&
+      contents_val &&
+      contents_val->enabled &&
+      !contents_val->contents)
+    instance->info->selection_cb (w, id, contents_val->call_data);
+
+  if (post_activate_cb)
+    post_activate_cb (w, id, widget_arg);
+}
+
+\f
+
+/* creation functions */
+static Widget
+xlw_create_menubar (widget_instance* instance)
+{
+  Arg al [1];
+  Widget widget;
+  
+  XtSetArg (al [0], XtNmenu, instance->info->val);
+  widget = XtCreateWidget (instance->info->name, xlwMenuWidgetClass,
+                          instance->parent, al, 1);
+  XtAddCallback (widget, XtNopen,   pre_hook,  (XtPointer)instance);
+  XtAddCallback (widget, XtNselect, pick_hook, (XtPointer)instance);
+  return widget;
+}
+
+static Widget
+xlw_create_popup_menu (widget_instance* instance)
+{
+  Arg al [2];
+  Widget popup_shell, widget;
+  
+  popup_shell = XtCreatePopupShell (instance->info->name,
+                                   overrideShellWidgetClass,
+                                   instance->parent, NULL, 0);
+  XtSetArg (al [0], XtNmenu, instance->info->val);
+  XtSetArg (al [1], XtNhorizontal, False);
+  widget = XtCreateManagedWidget ("popup", xlwMenuWidgetClass,
+                        popup_shell, al, 2);
+  XtAddCallback (widget, XtNselect, pick_hook, (XtPointer)instance);  
+  return popup_shell;
+}
+#endif /* LWLIB_MENUBARS_LUCID */
+
+#ifdef LWLIB_SCROLLBARS_LUCID
+static void
+xlw_scrollbar_callback (Widget widget, XtPointer closure, XtPointer call_data)
+{
+  widget_instance *instance = (widget_instance *) closure;
+  LWLIB_ID id;
+  XlwScrollBarCallbackStruct *data =
+    (XlwScrollBarCallbackStruct *) call_data;
+  scroll_event event_data;
+  scrollbar_values *val =
+    (scrollbar_values *) instance->info->val->scrollbar_data;
+  double percent;
+
+  if (!instance || widget->core.being_destroyed)
+    return;
+
+  id = instance->info->id;
+
+  percent = (double) (data->value - 1) / (double) (INT_MAX - 1);
+  event_data.slider_value =
+    (int) (percent * (double) (val->maximum - val->minimum)) + val->minimum;
+
+  if (event_data.slider_value > val->maximum - val->slider_size)
+      event_data.slider_value = val->maximum - val->slider_size;
+  else if (event_data.slider_value < val->minimum)
+           event_data.slider_value = val->minimum;
+
+  if (data->event)
+    {
+      switch (data->event->type)
+       {
+       case KeyPress:
+       case KeyRelease:
+         event_data.time = data->event->xkey.time;
+         break;
+       case ButtonPress:
+       case ButtonRelease:
+         event_data.time = data->event->xbutton.time;
+         break;
+       case MotionNotify:
+         event_data.time = data->event->xmotion.time;
+         break;
+       case EnterNotify:
+       case LeaveNotify:
+         event_data.time = data->event->xcrossing.time;
+         break;
+       default:
+         event_data.time = 0;
+         break;
+       }
+    }
+  else
+    event_data.time = 0;
+
+  switch (data->reason)
+    {
+    case XmCR_DECREMENT:       event_data.action = SCROLLBAR_LINE_UP;   break;
+    case XmCR_INCREMENT:       event_data.action = SCROLLBAR_LINE_DOWN; break;
+    case XmCR_PAGE_DECREMENT:  event_data.action = SCROLLBAR_PAGE_UP;   break;
+    case XmCR_PAGE_INCREMENT:  event_data.action = SCROLLBAR_PAGE_DOWN; break;
+    case XmCR_TO_TOP:          event_data.action = SCROLLBAR_TOP;       break;
+    case XmCR_TO_BOTTOM:       event_data.action = SCROLLBAR_BOTTOM;    break;
+    case XmCR_DRAG:            event_data.action = SCROLLBAR_DRAG;      break;
+    case XmCR_VALUE_CHANGED:   event_data.action = SCROLLBAR_CHANGE;    break;
+    default:                   event_data.action = SCROLLBAR_CHANGE;    break;
+    }
+
+  if (instance->info->pre_activate_cb)
+    instance->info->pre_activate_cb (widget, id, (XtPointer) &event_data);
+}
+
+#define add_scrollbar_callback(resource) \
+XtAddCallback (scrollbar, resource, xlw_scrollbar_callback, (XtPointer) instance)
+
+/* #### Does not yet support horizontal scrollbars. */
+static Widget
+xlw_create_scrollbar (widget_instance *instance, int vertical)
+{
+  Arg al[20];
+  int ac = 0;
+  static XtCallbackRec callbacks[2] =
+  { {xlw_scrollbar_callback, NULL}, {NULL, NULL} };
+
+  callbacks[0].closure  = (XtPointer) instance;
+
+  XtSetArg (al[ac], XmNminimum,       1); ac++;
+  XtSetArg (al[ac], XmNmaximum, INT_MAX); ac++;
+  XtSetArg (al[ac], XmNincrement,     1); ac++;
+  XtSetArg (al[ac], XmNpageIncrement, 1); ac++;
+  XtSetArg (al[ac], XmNorientation, (vertical ? XmVERTICAL : XmHORIZONTAL)); ac++;
+
+  XtSetArg (al[ac], XmNdecrementCallback,      callbacks); ac++;
+  XtSetArg (al[ac], XmNdragCallback,           callbacks); ac++;
+  XtSetArg (al[ac], XmNincrementCallback,      callbacks); ac++;
+  XtSetArg (al[ac], XmNpageDecrementCallback,  callbacks); ac++;
+  XtSetArg (al[ac], XmNpageIncrementCallback,  callbacks); ac++;
+  XtSetArg (al[ac], XmNtoBottomCallback,       callbacks); ac++;
+  XtSetArg (al[ac], XmNtoTopCallback,          callbacks); ac++;
+  XtSetArg (al[ac], XmNvalueChangedCallback,   callbacks); ac++;
+
+  return XtCreateWidget (instance->info->name, xlwScrollBarWidgetClass,
+                        instance->parent, al, ac);
+}
+
+static Widget
+xlw_create_vertical_scrollbar (widget_instance *instance)
+{
+  return xlw_create_scrollbar (instance, 1);
+}
+
+static Widget
+xlw_create_horizontal_scrollbar (widget_instance *instance)
+{
+  return xlw_create_scrollbar (instance, 0);
+}
+
+static void
+xlw_update_scrollbar (widget_instance *instance, Widget widget,
+                     widget_value *val)
+{
+  if (val->scrollbar_data)
+    {
+      scrollbar_values *data = val->scrollbar_data;
+      int widget_sliderSize, widget_val;
+      int new_sliderSize, new_value;
+      double percent;
+      Arg al [4];
+
+      /* First size and position the scrollbar widget. */
+      XtSetArg (al [0], XtNx,      data->scrollbar_x);
+      XtSetArg (al [1], XtNy,      data->scrollbar_y);
+      XtSetArg (al [2], XtNwidth,  data->scrollbar_width);
+      XtSetArg (al [3], XtNheight, data->scrollbar_height);
+      XtSetValues (widget, al, 4);
+
+      /* Now size the scrollbar's slider. */
+      XtSetArg (al [0], XmNsliderSize, &widget_sliderSize);
+      XtSetArg (al [1], XmNvalue,      &widget_val);
+      XtGetValues (widget, al, 2);
+
+      percent = (double) data->slider_size /
+       (double) (data->maximum - data->minimum);
+      percent = (percent > 1.0 ? 1.0 : percent);
+      new_sliderSize = (int) ((double) (INT_MAX - 1) * percent);
+
+      percent = (double) (data->slider_position - data->minimum) /
+       (double) (data->maximum - data->minimum);
+      percent = (percent > 1.0 ? 1.0 : percent);
+      new_value = (int) ((double) (INT_MAX - 1) * percent);
+
+      if (new_sliderSize > INT_MAX - 1)
+       new_sliderSize = INT_MAX - 1;
+      else if (new_sliderSize < 1)
+       new_sliderSize = 1;
+
+      if (new_value > (INT_MAX - new_sliderSize))
+       new_value = INT_MAX - new_sliderSize;
+      else if (new_value < 1)
+       new_value = 1;
+
+      if (new_sliderSize != widget_sliderSize || new_value != widget_val)
+       XlwScrollBarSetValues (widget, new_value, new_sliderSize, 1, 1, False);
+    }
+}
+
+#endif /* LWLIB_SCROLLBARS_LUCID */
+
+#ifdef LWLIB_TABS_LUCID
+/* tab control
+   
+   [[ lwlib is such an incredible hairy crock. I just cannot believe
+   it! There are random dependencies between functions, there is a
+   total lack of genericity, even though it initially appears to be
+   generic. It should all be junked and begun again. Building tabs are
+   an example - in theory we should be able to reuse a lot of the
+   general stuff because we want to put labels of whatever toolkit we
+   are using in the tab. Instead we have to hack it by hand. ]]
+   While lwlib is a hairy crock, whoever wrote that seems to misunderstand
+   Falk's tab control widget.  The tab control widget has *two* kinds of
+   children: *widgets*, which all occupy a *single* pane below the row of
+   tabs---this is where the labels created in build_tabs_in_widget go, and
+   *gadgets*, the tabs themselves, which do *not* draw themselves, but
+   rather are drawn by the control.  In fact, in XEmacs the true widget
+   children are *never* visible!  So this case is not a problem in the
+   design of lwlib, but rather of Falk's widget. -- sjt */
+static void
+xlw_tab_control_callback (Widget w, XtPointer client_data, XtPointer call_data)
+{
+  /* call data is the topmost widget */
+  widget_instance* instance = (widget_instance*)client_data;
+  Widget top = (Widget)call_data;
+  char *name = XtName (top);
+  widget_value* widget_val;
+  XtPointer widget_arg;
+  LWLIB_ID id;
+  lw_callback post_activate_cb;
+
+  if (w->core.being_destroyed)
+    return;
+
+  /* Grab these values before running any functions, in case running
+     the selection_cb causes the widget to be destroyed. */
+  id = instance->info->id;
+  post_activate_cb = instance->info->post_activate_cb;
+
+  /* search for the widget_val for the selected tab */
+  for (widget_val = instance->info->val->contents; widget_val; 
+       widget_val = widget_val->next)
+    {
+      if (!strcmp (widget_val->name, name))
+       break;
+    }
+
+  widget_arg = widget_val ? widget_val->call_data : NULL;
+
+  if (instance->info->selection_cb &&
+      widget_val &&
+      widget_val->enabled &&
+      !widget_val->contents)
+    instance->info->selection_cb (w, id, widget_arg);
+
+  if (post_activate_cb)
+    post_activate_cb (w, id, widget_arg);
+}
+
+static Widget
+xlw_create_tab_control (widget_instance *instance)
+{
+  Arg al[20];
+  int ac = 0;
+  Widget tab = 0;
+  widget_value* val = instance->info->val;
+
+  XtSetArg (al [ac], XtNsensitive, val->enabled);              ac++;
+  XtSetArg (al [ac], XtNmappedWhenManaged, False);             ac++;
+  XtSetArg (al [ac], XtNorientation, XtorientHorizontal);      ac++;
+
+  /* add any args the user supplied for creation time */
+  lw_add_value_args_to_args (val, al, &ac);
+
+  tab = XtCreateManagedWidget (val->name, tabsWidgetClass,
+                              instance->parent, al, ac);
+  XtRemoveAllCallbacks (tab, XtNcallback);
+  XtAddCallback (tab, XtNcallback, xlw_tab_control_callback, (XtPointer)instance);
+
+  XtManageChild (tab);
+
+  return tab;
+}
+
+static void build_tabs_in_widget (widget_instance* instance, Widget widget, 
+                                 widget_value* val)
+{
+  widget_value* cur = val;
+  Arg al[1];
+
+  /* Children are always invisible, don't permit resizing. */
+  XtSetArg (al[0], XtNresizable, False);
+
+  for (cur = val; cur; cur = cur->next)
+    {
+      if (cur->value)
+       {
+         Widget w;
+#ifdef LWLIB_WIDGETS_MOTIF
+         w = xm_create_label (widget, cur);
+#else
+         w = xaw_create_label (widget, cur);
+#endif
+         XtSetValues (w, al, 1);
+       }
+      cur->change = NO_CHANGE;
+    }
+}
+
+static void
+xlw_update_tab_control (widget_instance* instance, Widget widget, widget_value* val)
+{
+  Widget* children;
+  unsigned int num_children;
+  int i;
+  widget_value *cur = 0;
+
+  XtRemoveAllCallbacks (widget, XtNcallback);
+  XtAddCallback (widget, XtNcallback, xlw_tab_control_callback, (XtPointer)instance);
+
+  if (val->change == STRUCTURAL_CHANGE
+      ||
+      (val->contents && val->contents->change == STRUCTURAL_CHANGE))
+    {
+      destroy_all_children (widget);
+      build_tabs_in_widget (instance, widget, val->contents);
+    }
+
+  children = XtCompositeChildren (widget, &num_children);
+  if (children)
+    {
+      for (i = 0, cur = val->contents; i < num_children; i++)
+       {
+         if (!cur)
+           abort ();
+         if (children [i]->core.being_destroyed
+             || strcmp (XtName (children [i]), cur->name))
+           continue;
+#ifdef NEED_MOTIF
+         if (lw_motif_widget_p (children [i]))
+           xm_update_one_widget (instance, children [i], cur, False);
+#endif
+#ifdef NEED_ATHENA
+         if (lw_xaw_widget_p (children [i]))
+           xaw_update_one_widget (instance, children [i], cur, False);
+#endif
+         cur = cur->next;
+       }
+      XtFree ((char *) children);
+    }
+  if (cur)
+    abort ();
+}
+#endif /* LWLIB_TABS_LUCID */
+
+#ifdef HAVE_WIDGETS
+static Widget
+xlw_create_clip_window (widget_instance *instance)
+{
+  Arg al[20];
+  int ac = 0;
+  Widget clip = 0;
+  widget_value* val = instance->info->val;
+
+  XtSetArg (al [ac], XtNmappedWhenManaged, FALSE);     ac++;
+  XtSetArg (al [ac], XtNsensitive, TRUE);              ac++;
+  /* add any args the user supplied for creation time */
+  lw_add_value_args_to_args (val, al, &ac);
+
+  /* Create a clip window to contain the subwidget. Incredibly the
+     XEmacs manager seems to be the most appropriate widget for
+     this. Nothing else is simple enough and yet does what is
+     required. */
+  clip = XtCreateManagedWidget (val->name,
+                               emacsManagerWidgetClass,
+                               instance->parent, al, ac);
+
+  return clip;
+}
+#endif
+
+const widget_creation_entry 
+xlw_creation_table [] =
+{
+#ifdef LWLIB_MENUBARS_LUCID
+  {"menubar", xlw_create_menubar},
+  {"popup", xlw_create_popup_menu},
+#endif
+#ifdef LWLIB_SCROLLBARS_LUCID
+  {"vertical-scrollbar",       xlw_create_vertical_scrollbar},
+  {"horizontal-scrollbar",     xlw_create_horizontal_scrollbar},
+#endif
+#ifdef LWLIB_TABS_LUCID
+  {"tab-control",      xlw_create_tab_control},
+#endif
+#ifdef HAVE_WIDGETS
+  {"clip-window", xlw_create_clip_window},
+#endif
+  {NULL, NULL}
+};
+
+Boolean
+lw_lucid_widget_p (Widget widget)
+{
+  WidgetClass the_class = XtClass (widget);
+#ifdef LWLIB_MENUBARS_LUCID
+  if (the_class == xlwMenuWidgetClass)
+    return True;
+#endif
+#ifdef LWLIB_SCROLLBARS_LUCID
+  if (the_class == xlwScrollBarWidgetClass)
+    return True;
+#endif
+#ifdef LWLIB_TABS_LUCID
+  if (the_class == tabsWidgetClass)
+    return True;
+#endif
+#ifdef LWLIB_MENUBARS_LUCID
+  if (the_class == overrideShellWidgetClass)
+    return
+      XtClass (((CompositeWidget)widget)->composite.children [0])
+       == xlwMenuWidgetClass;
+#endif
+#ifdef HAVE_WIDGETS
+  if (the_class == emacsManagerWidgetClass)
+    return True;
+#endif
+  return False;
+}
+
+void
+xlw_update_one_widget (widget_instance* instance, Widget widget,
+                      widget_value* val, Boolean deep_p)
+{
+  WidgetClass class = XtClass (widget);
+
+  if (0)
+    ;
+#ifdef LWLIB_MENUBARS_LUCID
+  else if (class == xlwMenuWidgetClass)
+    {
+      XlwMenuWidget mw;
+      Arg al [1];
+      if (XtIsShell (widget))
+       mw = (XlwMenuWidget)((CompositeWidget)widget)->composite.children [0];
+      else
+       mw = (XlwMenuWidget)widget;
+      XtSetArg (al [0], XtNmenu, val);
+      XtSetValues (widget, al, 1); /* #### mw unused! */
+    }
+#endif
+#ifdef LWLIB_SCROLLBARS_LUCID
+  else if (class == xlwScrollBarWidgetClass)
+    {
+      xlw_update_scrollbar (instance, widget, val);
+    }
+#endif
+#ifdef LWLIB_TABS_LUCID
+  else if (class == tabsWidgetClass)
+    {
+      xlw_update_tab_control (instance, widget, val);
+    }
+#endif
+  /* Lastly update our global arg values. */
+  if (val->args && val->args->nargs)
+    XtSetValues (widget, val->args->args, val->args->nargs);
+}
+
+void
+xlw_update_one_value (widget_instance* instance, Widget widget,
+                     widget_value* val)
+{
+  return;
+}
+
+void
+xlw_pop_instance (widget_instance* instance, Boolean up)
+{
+}
+
+#ifdef LWLIB_MENUBARS_LUCID
+void
+xlw_popup_menu (Widget widget, XEvent *event)
+{
+  XlwMenuWidget mw;
+
+  if (!XtIsShell (widget))
+    return;
+
+  if (event->type == ButtonPress || event->type == ButtonRelease)
+    {
+      mw = (XlwMenuWidget)((CompositeWidget)widget)->composite.children [0];
+      xlw_pop_up_menu (mw, (XButtonPressedEvent *)event);
+    }
+  else
+    abort ();
+}
+#endif /* LWLIB_MENUBARS_LUCID */
+
+\f/* Destruction of instances */
+void
+xlw_destroy_instance (widget_instance* instance)
+{
+  if (instance->widget)
+    XtDestroyWidget (instance->widget);
+}