This commit was manufactured by cvs2svn to create branch 'XEmacs-21_4'.
authortomo <tomo>
Thu, 12 Aug 1999 02:35:33 +0000 (02:35 +0000)
committertomo <tomo>
Thu, 12 Aug 1999 02:35:33 +0000 (02:35 +0000)
etc/custom/example-themes/europe-theme.el [new file with mode: 0644]
etc/custom/example-themes/ex-custom-file [new file with mode: 0644]
etc/custom/example-themes/example-theme.el [new file with mode: 0644]
lisp/gutter-items.el [new file with mode: 0644]
src/gutter.c [new file with mode: 0644]
src/gutter.h [new file with mode: 0644]
src/s/sco7.h [new file with mode: 0644]
tests/gutter-test.el [new file with mode: 0644]

diff --git a/etc/custom/example-themes/europe-theme.el b/etc/custom/example-themes/europe-theme.el
new file mode 100644 (file)
index 0000000..28ec23c
--- /dev/null
@@ -0,0 +1,13 @@
+;;;autoload 
+(deftheme europe
+  "Settings for European users."
+  :set-variable-settings
+    "This variable has a value appropriate for European users."
+  :set-variable-settings
+    "This has been forceed to the value appropriate for European users.")
+
+(custom-theme-set-variables 'europe
+   '(sentence-end-double-space nil)
+   '(ps-paper-type (quote a4)))
+
+(provide-theme 'europe)
diff --git a/etc/custom/example-themes/ex-custom-file b/etc/custom/example-themes/ex-custom-file
new file mode 100644 (file)
index 0000000..5db279a
--- /dev/null
@@ -0,0 +1,21 @@
+;; Sample User custom-file. All forms below are autogenerated.
+
+
+
+
+
+
+
+
+
+(custom-load-themes
+   'example )
+(custom-reset-variables
+ '(sentence-end-double-space standard))
+(custom-set-variables
+ '(package-get-remove-copy t)
+ '(package-get-dir "/scratch/incoming")
+ '(sgml-public-map (quote ("%S" "/scratch/xemacs/xemacs-20/xemacs-packages/etc/psgml/%o/%c/%d" "/usr/lib/sgml/%o/%c/%d")))
+ '(paren-mode (quote paren) nil (paren)))
+(custom-set-faces
+ '(default ((t (:background "oldlace"))) t))
diff --git a/etc/custom/example-themes/example-theme.el b/etc/custom/example-themes/example-theme.el
new file mode 100644 (file)
index 0000000..d35221b
--- /dev/null
@@ -0,0 +1,15 @@
+;;;autoload 
+(deftheme example
+  "A sample theme for customize theme support."
+  :variable-set-string "This variable has been made an example.")
+
+(custom-theme-load-themes 'example
+    'europe)
+
+(custom-theme-set-variables 'example
+ '(iswitchb-prompt-newbuffer nil))
+
+(provide-theme 'example)
+
+
+  
\ No newline at end of file
diff --git a/lisp/gutter-items.el b/lisp/gutter-items.el
new file mode 100644 (file)
index 0000000..3d83dd1
--- /dev/null
@@ -0,0 +1,160 @@
+;;; gutter-items.el --- Gutter content for XEmacs.
+
+;; Copyright (C) 1999 Free Software Foundation, Inc.
+;; Copyright (C) 1999 Andy Piper.
+
+;; Maintainer: XEmacs Development Team
+;; Keywords: frames, extensions, internal, dumped
+
+;; 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 Xmacs; see the file COPYING.  If not, write to the
+;; Free Software Foundation, 59 Temple Place - Suite 330,
+;; Boston, MA 02111-1307, USA.
+
+;; Some of this is taken from the buffer-menu stuff in menubar-items.el
+;; and the custom specs in toolbar.el.
+
+(defgroup gutter nil
+  "Input from the gutters."
+  :group 'environment)
+
+(defcustom gutter-visible-p 
+  (specifier-instance default-gutter-visible-p)
+  "Whether the default gutter is globally visible. This option can be
+customized through the options menu."
+  :group 'display
+  :type 'boolean
+  :set #'(lambda (var val)
+          (set-specifier default-gutter-visible-p val)
+          (setq gutter-visible-p val)))
+
+(defcustom default-gutter-position
+  (default-gutter-position)
+  "The location of the default gutter. It can be 'top, 'bottom, 'left or
+'right. This option can be customized through the options menu."
+  :group 'display
+  :type '(choice (const :tag "top" 'top)
+                (const :tag "bottom" 'bottom)
+                (const :tag "left" 'left)
+                (const :tag "right" 'right))
+  :set #'(lambda (var val)
+          (set-default-gutter-position val)
+          (setq default-gutter-position val)))
+
+;;; The Buffers tab
+
+(defgroup buffers-tab nil
+  "Customization of `Buffers' tab."
+  :group 'gutter)
+
+(defvar gutter-buffers-tab nil
+  "A tab widget in the gutter for displaying buffers.
+Do not set this. Use `glyph-image-instance' and
+`set-image-instance-property' to change the properties of the tab.")
+
+(defcustom buffers-tab-max-size 6
+  "*Maximum number of entries which may appear on the \"Buffers\" tab.
+If this is 10, then only the ten most-recently-selected buffers will be
+shown.  If this is nil, then all buffers will be shown.  Setting this to
+a large number or nil will slow down tab responsiveness."
+  :type '(choice (const :tag "Show all" nil)
+                (integer 10))
+  :group 'buffers-tab)
+
+(defcustom buffers-tab-switch-to-buffer-function 'buffers-tab-switch-to-buffer
+  "*The function to call to select a buffer from the buffers tab.
+`switch-to-buffer' is a good choice, as is `pop-to-buffer'."
+  :type '(radio (function-item switch-to-buffer)
+               (function-item pop-to-buffer)
+               (function :tag "Other"))
+  :group 'buffers-tab)
+
+(defcustom buffers-tab-omit-function 'buffers-menu-omit-invisible-buffers
+  "*If non-nil, a function specifying the buffers to omit from the buffers tab.
+This is passed a buffer and should return non-nil if the buffer should be
+omitted.  The default value `buffers-tab-omit-invisible-buffers' omits
+buffers that are normally considered \"invisible\" (those whose name
+begins with a space)."
+  :type '(choice (const :tag "None" nil)
+                function)
+  :group 'buffers-tab)
+
+(defcustom buffers-tab-format-buffer-line-function 'format-buffers-menu-line
+  "*The function to call to return a string to represent a buffer in the
+buffers tab.  The function is passed a buffer and should return a string.
+The default value `format-buffers-menu-line' just returns the name of
+the buffer.  Also check out `slow-format-buffers-menu-line' which
+returns a whole bunch of info about a buffer."
+  :type 'function
+  :group 'buffers-tab)
+
+(defun buffers-tab-switch-to-buffer (buffer)
+  "For use as a value for `buffers-tab-switch-to-buffer-function'."
+  (switch-to-buffer buffer t))
+
+(defsubst build-buffers-tab-internal (buffers)
+  (let (line)
+    (mapcar
+     #'(lambda (buffer)
+        (setq line (funcall buffers-tab-format-buffer-line-function
+                            buffer))
+        (vector line (list buffers-tab-switch-to-buffer-function
+                           (buffer-name buffer))))
+     buffers)))
+
+(defun buffers-tab-items ()
+  "This is the tab filter for the top-level buffers \"Buffers\" tab.
+It dynamically creates a list of buffers to use as the contents of the tab.
+Only the most-recently-used few buffers will be listed on the tab, for
+efficiency reasons.  You can control how many buffers will be shown by
+setting `buffers-tab-max-size'.  You can control the text of the tab
+items by redefining the function `format-buffers-menu-line'."
+  (let ((buffers (delete-if buffers-tab-omit-function (buffer-list))))
+    (and (integerp buffers-tab-max-size)
+        (> buffers-tab-max-size 1)
+        (> (length buffers) buffers-tab-max-size)
+        ;; shorten list of buffers
+        (setcdr (nthcdr buffers-tab-max-size buffers) nil))
+    (setq buffers (build-buffers-tab-internal buffers))
+    buffers))
+
+(defun add-tab-to-gutter ()
+  "Put a tab control in the gutter area to hold the most recent buffers."
+  (let ((gutter-string ""))
+    (set-extent-begin-glyph 
+     (make-extent 0 0 gutter-string)
+     (setq gutter-buffers-tab 
+          (make-glyph 
+           (vector 'tab-control :descriptor "Buffers"
+                   :properties (list :items (buffers-tab-items))))))
+    ;; This looks better than a 3d border
+    (set-specifier default-gutter-border-width 0 'global 'mswindows)
+    (set-specifier default-gutter gutter-string 'global 'mswindows)))
+
+(defun update-tab-in-gutter (&optional notused)
+  "Update the tab control in the gutter area."
+  (when (valid-image-instantiator-format-p 'tab-control)
+    (set-image-instance-property (glyph-image-instance gutter-buffers-tab)
+                                :items
+                                (buffers-tab-items))
+    (resize-subwindow (glyph-image-instance gutter-buffers-tab)
+                     (gutter-pixel-width) nil)))
+
+(add-tab-to-gutter)
+(add-hook 'switch-to-buffer-hooks 'update-tab-in-gutter)
+(add-hook 'create-frame-hook 'update-tab-in-gutter)
+
+(provide 'gutter-items)
+;;; gutter-items.el ends here.
diff --git a/src/gutter.c b/src/gutter.c
new file mode 100644 (file)
index 0000000..64d0df3
--- /dev/null
@@ -0,0 +1,1111 @@
+/* Gutter implementation.
+   Copyright (C) 1999 Andy Piper.
+
+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. */
+
+/* Specifers ripped-off from toolbar.c */
+
+#include <config.h>
+#include "lisp.h"
+
+#include "buffer.h"
+#include "frame.h"
+#include "device.h"
+#include "faces.h"
+#include "glyphs.h"
+#include "redisplay.h"
+#include "window.h"
+#include "gutter.h"
+
+Lisp_Object Vgutter[4];
+Lisp_Object Vgutter_size[4];
+Lisp_Object Vgutter_visible_p[4];
+Lisp_Object Vgutter_border_width[4];
+
+Lisp_Object Vdefault_gutter, Vdefault_gutter_visible_p;
+Lisp_Object Vdefault_gutter_width, Vdefault_gutter_height;
+Lisp_Object Vdefault_gutter_border_width;
+
+Lisp_Object Vdefault_gutter_position;
+
+#define SET_GUTTER_WAS_VISIBLE_FLAG(frame, pos, flag)                  \
+  do {                                                                 \
+    switch (pos)                                                       \
+      {                                                                        \
+      case TOP_GUTTER:                                                 \
+       (frame)->top_gutter_was_visible = flag;                 \
+       break;                                                          \
+      case BOTTOM_GUTTER:                                              \
+       (frame)->bottom_gutter_was_visible = flag;                      \
+       break;                                                          \
+      case LEFT_GUTTER:                                                \
+       (frame)->left_gutter_was_visible = flag;                        \
+       break;                                                          \
+      case RIGHT_GUTTER:                                               \
+       (frame)->right_gutter_was_visible = flag;                       \
+       break;                                                          \
+      default:                                                         \
+       abort ();                                                       \
+      }                                                                        \
+  } while (0)
+
+static int gutter_was_visible (struct frame* frame, enum gutter_pos pos)
+{
+  switch (pos)
+    {
+    case TOP_GUTTER:
+      return (frame)->top_gutter_was_visible;
+    case BOTTOM_GUTTER:
+      return (frame)->bottom_gutter_was_visible;
+    case LEFT_GUTTER:
+      return (frame)->left_gutter_was_visible;
+    case RIGHT_GUTTER:
+      return (frame)->right_gutter_was_visible;
+    default:
+      abort ();
+    }
+}
+
+static Lisp_Object
+frame_topmost_window (struct frame *f)
+{
+  Lisp_Object w = FRAME_ROOT_WINDOW (f);
+
+  do {
+    while (!NILP (XWINDOW (w)->vchild))
+      {
+       w = XWINDOW (w)->vchild;
+      }
+  } while (!NILP (XWINDOW (w)->hchild) && !NILP (w = XWINDOW (w)->hchild));
+
+  return w;
+}
+
+static Lisp_Object
+frame_bottommost_window (struct frame *f)
+{
+  Lisp_Object w = FRAME_ROOT_WINDOW (f);
+
+  do {
+    while (!NILP (XWINDOW (w)->vchild))
+      {
+       w = XWINDOW (w)->vchild;
+       while (!NILP (XWINDOW (w)->next))
+         {
+           w = XWINDOW (w)->next;
+         }
+      }
+  } while (!NILP (XWINDOW (w)->hchild) && !NILP (w = XWINDOW (w)->hchild));
+
+  return w;
+}
+
+#if 0
+static Lisp_Object
+frame_leftmost_window (struct frame *f)
+{
+  Lisp_Object w = FRAME_ROOT_WINDOW (f);
+
+  do {
+    while (!NILP (XWINDOW (w)->hchild))
+      {
+       w = XWINDOW (w)->hchild;
+      }
+  } while (!NILP (XWINDOW (w)->vchild) && !NILP (w = XWINDOW (w)->vchild));
+
+  return w;
+}
+
+static Lisp_Object
+frame_rightmost_window (struct frame *f)
+{
+  Lisp_Object w = FRAME_ROOT_WINDOW (f);
+
+  do {
+    while (!NILP (XWINDOW (w)->hchild))
+      {
+       w = XWINDOW (w)->hchild;
+       while (!NILP (XWINDOW (w)->next))
+         {
+           w = XWINDOW (w)->next;
+         }
+      }
+  } while (!NILP (XWINDOW (w)->vchild) && !NILP (w = XWINDOW (w)->vchild));
+  return w;
+}
+#endif
+
+/* calculate the coordinates of a gutter for the current frame and
+   selected window. we have to be careful in calculating this as we
+   need to use *two* windows, the currently selected window will give
+   us the actual height, width and contents of the gutter, but if we
+   use this for calculating the gutter positions we run into trouble
+   if it is not the window nearest the gutter. Instead we predetermine
+   the nearest window and then use that.*/
+void
+get_gutter_coords (struct frame *f, enum gutter_pos pos, int *x, int *y,
+                  int *width, int *height)
+{
+  struct window
+    * top = XWINDOW (frame_topmost_window (f)),
+    * bot = XWINDOW (frame_bottommost_window (f));
+  /* The top and bottom gutters take precedence over the left and
+     right. */
+  switch (pos)
+    {
+    case TOP_GUTTER:
+      *x = FRAME_LEFT_BORDER_END (f);
+      *y = FRAME_TOP_BORDER_END (f);
+      *width = FRAME_RIGHT_BORDER_START (f) 
+       - FRAME_LEFT_BORDER_END (f);
+      *height = FRAME_TOP_GUTTER_BOUNDS (f);
+      break;
+
+    case BOTTOM_GUTTER:
+      *x = FRAME_LEFT_BORDER_END (f);
+      *y = WINDOW_BOTTOM (bot)
+       - FRAME_BOTTOM_GUTTER_BOUNDS (f);
+      *width = FRAME_RIGHT_BORDER_START (f) 
+       - FRAME_LEFT_BORDER_END (f);
+      *height = FRAME_BOTTOM_GUTTER_BOUNDS (f);
+      break;
+
+    case LEFT_GUTTER:
+      *x = FRAME_LEFT_BORDER_END (f);
+      *y = WINDOW_TEXT_TOP (top);
+      *width = FRAME_LEFT_GUTTER_BOUNDS (f);
+      *height = WINDOW_BOTTOM (bot)
+       - (WINDOW_TEXT_TOP (top)
+          + FRAME_BOTTOM_GUTTER_BOUNDS (f));
+      break;
+      
+    case RIGHT_GUTTER:
+      *x = FRAME_RIGHT_BORDER_START (f)
+       - FRAME_RIGHT_GUTTER_BOUNDS (f);
+      *y = WINDOW_TEXT_TOP (top);
+      *width = FRAME_RIGHT_GUTTER_BOUNDS (f);
+      *height = WINDOW_BOTTOM (bot)
+       - (WINDOW_TEXT_TOP (top) 
+          + FRAME_BOTTOM_GUTTER_BOUNDS (f));
+      break;
+
+    default:
+      abort ();
+    }
+}
+
+static void
+output_gutter (struct frame *f, enum gutter_pos pos)
+{
+  Lisp_Object frame;
+  Lisp_Object window = FRAME_LAST_NONMINIBUF_WINDOW (f);
+  struct device *d = XDEVICE (f->device);
+  struct window* w = XWINDOW (window);
+  int x, y, width, height, ypos;
+  int line;
+  int border_width = FRAME_GUTTER_BORDER_WIDTH (f, pos);
+  face_index findex = get_builtin_face_cache_index (w, Vgui_element_face);
+  display_line_dynarr* ddla, *cdla;
+  struct display_line *dl;
+
+  if (!f->current_display_lines)
+    f->current_display_lines = Dynarr_new (display_line);
+  if (!f->desired_display_lines)
+    f->desired_display_lines = Dynarr_new (display_line);
+  
+  ddla = f->desired_display_lines;
+  cdla = f->current_display_lines;
+
+  XSETFRAME (frame, f);
+
+  get_gutter_coords (f, pos, &x, &y, &width, &height);
+  /* clear out what we want to cover */
+  /* generate some display lines */
+  generate_displayable_area (w, WINDOW_GUTTER (w, pos),
+                            x + border_width, y + border_width,
+                            width - 2 * border_width, 
+                            height - 2 * border_width, ddla, 0, findex);
+  /* Output each line. */
+  for (line = 0; line < Dynarr_length (ddla); line++)
+    {
+      output_display_line (w, cdla, ddla, line, -1, -1);
+    }
+
+  /* grab coordinates of last line and blank after it. */
+  dl = Dynarr_atp (ddla, Dynarr_length (ddla) - 1);
+  ypos = dl->ypos + dl->descent - dl->clip;
+  redisplay_clear_region (window, findex, x + border_width , ypos,
+                         width - 2 * border_width, height - (ypos - y));
+
+  /* bevel the gutter area if so desired */
+  if (border_width != 0)
+    {
+      MAYBE_DEVMETH (d, bevel_area, 
+                    (w, findex, x, y, width, height, border_width));
+    }
+}
+
+/* sizing gutters is a pain so we try and help the user by detemining
+   what height will accomodate all lines. This is useless on left and
+   right gutters as we always have a maximal number of lines. */
+static Lisp_Object
+calculate_gutter_size (struct window *w, enum gutter_pos pos)
+{
+  struct frame* f = XFRAME (WINDOW_FRAME (w));
+  int ypos;
+  display_line_dynarr* ddla;
+  struct display_line *dl;
+
+  /* we cannot autodetect gutter sizes for the left and right as there
+     is no reasonable metric to use */
+  assert (pos == TOP_GUTTER || pos == BOTTOM_GUTTER);
+  /* degenerate case */
+  if (NILP (WINDOW_GUTTER (w, pos))
+      ||
+      !FRAME_VISIBLE_P (f))
+    return Qnil;
+
+  ddla = Dynarr_new (display_line);
+  /* generate some display lines */
+  generate_displayable_area (w, WINDOW_GUTTER (w, pos),
+                            FRAME_LEFT_BORDER_END (f),
+                            0,
+                            FRAME_RIGHT_BORDER_START (f)
+                            - FRAME_LEFT_BORDER_END (f),
+                            200,
+                            ddla, 0, 0);
+  /* grab coordinates of last line  */
+  if (Dynarr_length (ddla))
+    {
+      dl = Dynarr_atp (ddla, Dynarr_length (ddla) - 1);
+      ypos = dl->ypos + dl->descent - dl->clip;
+      Dynarr_free (ddla);
+      return make_int (ypos);
+    }
+  else
+    {
+      Dynarr_free (ddla);
+      return Qnil;
+    }
+}
+
+static void
+clear_gutter (struct frame *f, enum gutter_pos pos)
+{
+  int x, y, width, height;
+  Lisp_Object window = FRAME_LAST_NONMINIBUF_WINDOW (f);
+  face_index findex = get_builtin_face_cache_index (XWINDOW (window),
+                                                   Vgui_element_face);
+  get_gutter_coords (f, pos, &x, &y, &width, &height);
+
+  SET_GUTTER_WAS_VISIBLE_FLAG (f, pos, 0);
+
+  redisplay_clear_region (window, findex, x, y, width, height);
+}
+
+void
+update_frame_gutters (struct frame *f)
+{
+  if (f->gutter_changed || f->frame_changed || f->clear)
+    {
+      int pos;
+      /* and output */
+
+      for (pos = 0; pos < 4; pos++)
+       {
+         if (FRAME_GUTTER_VISIBLE (f, pos))
+           output_gutter (f, pos);
+         else if (gutter_was_visible (f, pos))
+           clear_gutter (f, pos);
+       }
+
+    }
+  f->gutter_changed = 0;
+}
+
+static void
+redraw_exposed_gutter (struct frame *f, enum gutter_pos pos, int x, int y,
+                      int width, int height)
+{
+  int g_x, g_y, g_width, g_height;
+  int newx, newy;
+
+  get_gutter_coords (f, pos, &g_x, &g_y, &g_width, &g_height);
+
+  if (((y + height) < g_y) || (y > (g_y + g_height)))
+    return;
+  if (((x + width) < g_x) || (x > (g_x + g_width)))
+    return;
+
+  /* #### optimize this - redrawing the whole gutter for every expose
+     is very expensive. We reset the current display lines because if
+     they're being exposed they are no longer current. */
+  if (f->current_display_lines)
+    Dynarr_reset (f->current_display_lines);
+  /* we have to do this in-case there were subwindows where we are
+     redrawing, unfortunately sometimes this also generates expose
+     events resulting in an endless cycle of redsplay. */
+  newx = max (x, g_x);
+  newy = max (y, g_y);
+  width = min (x + width - newx, g_x + g_width - newx);
+  height = min (y + height - newy, g_y + g_height - newy);
+  redisplay_unmap_subwindows_maybe (f, newx, newy, width, height);
+
+  /* Even if none of the gutter is in the area, the blank region at
+     the very least must be because the first thing we did is verify
+     that some portion of the gutter is in the exposed region. */
+  output_gutter (f, pos);
+}
+
+void
+redraw_exposed_gutters (struct frame *f, int x, int y, int width,
+                       int height)
+{
+      int pos;
+      for (pos = 0; pos < 4; pos++)
+       {
+         if (FRAME_GUTTER_VISIBLE (f, pos))
+           redraw_exposed_gutter (f, pos, x, y, width, height);
+       }
+}
+
+void
+free_frame_gutters (struct frame *f)
+{
+  if (f->current_display_lines)
+    Dynarr_free (f->current_display_lines);
+  if (f->desired_display_lines)
+    Dynarr_free (f->desired_display_lines);
+}
+
+static enum gutter_pos
+decode_gutter_position (Lisp_Object position)
+{
+  if (EQ (position, Qtop))    return TOP_GUTTER;
+  if (EQ (position, Qbottom)) return BOTTOM_GUTTER;
+  if (EQ (position, Qleft))   return LEFT_GUTTER;
+  if (EQ (position, Qright))  return RIGHT_GUTTER;
+  signal_simple_error ("Invalid gutter position", position);
+
+  return TOP_GUTTER; /* not reached */
+}
+
+DEFUN ("set-default-gutter-position", Fset_default_gutter_position, 1, 1, 0, /*
+Set the position that the `default-gutter' will be displayed at.
+Valid positions are 'top, 'bottom, 'left and 'right.
+See `default-gutter-position'.
+*/
+       (position))
+{
+  enum gutter_pos cur = decode_gutter_position (Vdefault_gutter_position);
+  enum gutter_pos new = decode_gutter_position (position);
+
+  if (cur != new)
+    {
+      /* The following calls will automatically cause the dirty
+        flags to be set; we delay frame size changes to avoid
+        lots of frame flickering. */
+      /* #### I think this should be GC protected. -sb */
+      hold_frame_size_changes ();
+      set_specifier_fallback (Vgutter[cur], list1 (Fcons (Qnil, Qnil)));
+      set_specifier_fallback (Vgutter[new], Vdefault_gutter);
+      set_specifier_fallback (Vgutter_size[cur], list1 (Fcons (Qnil, Qzero)));
+      set_specifier_fallback (Vgutter_size[new],
+                             new == TOP_GUTTER || new == BOTTOM_GUTTER
+                             ? Vdefault_gutter_height
+                             : Vdefault_gutter_width);
+      set_specifier_fallback (Vgutter_border_width[cur],
+                             list1 (Fcons (Qnil, Qzero)));
+      set_specifier_fallback (Vgutter_border_width[new],
+                             Vdefault_gutter_border_width);
+      set_specifier_fallback (Vgutter_visible_p[cur],
+                             list1 (Fcons (Qnil, Qt)));
+      set_specifier_fallback (Vgutter_visible_p[new],
+                             Vdefault_gutter_visible_p);
+      Vdefault_gutter_position = position;
+      unhold_frame_size_changes ();
+    }
+
+  return position;
+}
+
+DEFUN ("default-gutter-position", Fdefault_gutter_position, 0, 0, 0, /*
+Return the position that the `default-gutter' will be displayed at.
+The `default-gutter' will only be displayed here if the corresponding
+position-specific gutter specifier does not provide a value.
+*/
+       ())
+{
+  return Vdefault_gutter_position;
+}
+
+DEFUN ("gutter-pixel-width", Fgutter_pixel_width, 0, 2, 0, /*
+Return the pixel width of the gutter at POS in LOCALE.
+POS defaults to the default gutter position. LOCALE defaults to
+the current window.
+*/
+       (pos, locale))
+{
+  int x, y, width, height;
+  enum gutter_pos p = TOP_GUTTER;
+  struct frame *f = decode_frame (FW_FRAME (locale));
+
+  if (NILP (pos))
+    pos = Vdefault_gutter_position;
+  p = decode_gutter_position (pos);
+
+  get_gutter_coords (f, p, &x, &y, &width, &height);
+  width -= (FRAME_GUTTER_BORDER_WIDTH (f, p) * 2);
+
+  return make_int (width);
+}
+
+DEFUN ("gutter-pixel-height", Fgutter_pixel_height, 0, 2, 0, /*
+Return the pixel height of the gutter at POS in LOCALE.
+POS defaults to the default gutter position. LOCALE defaults to
+the current window.
+*/
+       (pos, locale))
+{
+  int x, y, width, height;
+  enum gutter_pos p = TOP_GUTTER;
+  struct frame *f = decode_frame (FW_FRAME (locale));
+
+  if (NILP (pos))
+    pos = Vdefault_gutter_position;
+  p = decode_gutter_position (pos);
+
+  get_gutter_coords (f, p, &x, &y, &width, &height);
+  height -= (FRAME_GUTTER_BORDER_WIDTH (f, p) * 2);
+
+  return make_int (height);
+}
+
+DEFINE_SPECIFIER_TYPE (gutter);
+
+static void
+gutter_after_change (Lisp_Object specifier, Lisp_Object locale)
+{
+  MARK_GUTTER_CHANGED;
+}
+
+static void
+gutter_validate (Lisp_Object instantiator)
+{
+  if (NILP (instantiator))
+    return;
+
+  if (!STRINGP (instantiator))
+    signal_simple_error ("Gutter spec must be string or nil", instantiator);
+}
+
+DEFUN ("gutter-specifier-p", Fgutter_specifier_p, 1, 1, 0, /*
+Return non-nil if OBJECT is a gutter specifier.
+Gutter specifiers are used to specify the format of a gutter.
+The values of the variables `default-gutter', `top-gutter',
+`left-gutter', `right-gutter', and `bottom-gutter' are always
+gutter specifiers.
+
+Valid gutter instantiators are called "gutter descriptors"
+and are lists of vectors.  See `default-gutter' for a description
+of the exact format.
+*/
+       (object))
+{
+  return GUTTER_SPECIFIERP (object) ? Qt : Qnil;
+}
+
+\f
+/*
+  Helper for invalidating the real specifier when default
+  specifier caching changes
+*/
+static void
+recompute_overlaying_specifier (Lisp_Object real_one[4])
+{
+  enum gutter_pos pos = decode_gutter_position (Vdefault_gutter_position);
+  Fset_specifier_dirty_flag (real_one[pos]);
+}
+
+static void
+gutter_specs_changed (Lisp_Object specifier, struct window *w,
+                      Lisp_Object oldval)
+{
+  int pos;
+  for (pos = 0; pos< 4; pos++)
+    {
+      w->real_gutter_size[pos] = w->gutter_size[pos];
+      if (EQ (w->real_gutter_size[pos], Qautodetect)
+         && !NILP (w->gutter_visible_p[pos]))
+       {
+         w->real_gutter_size [pos] = calculate_gutter_size (w, pos);
+       }
+    }
+  MARK_GUTTER_CHANGED;
+  MARK_WINDOWS_CHANGED (w);
+}
+
+static void
+default_gutter_specs_changed (Lisp_Object specifier, struct window *w,
+                              Lisp_Object oldval)
+{
+  recompute_overlaying_specifier (Vgutter);
+}
+
+static void
+gutter_geometry_changed_in_window (Lisp_Object specifier, struct window *w,
+                                   Lisp_Object oldval)
+{
+  int pos;
+  for (pos = 0; pos< 4; pos++)
+    {
+      w->real_gutter_size[pos] = w->gutter_size[pos];
+      if (EQ (w->real_gutter_size[pos], Qautodetect)
+         && !NILP (w->gutter_visible_p[pos]))
+       {
+         w->real_gutter_size [pos] = calculate_gutter_size (w, pos);
+       }
+    }
+  
+  MARK_GUTTER_CHANGED;
+  MARK_WINDOWS_CHANGED (w);
+}
+
+static void
+default_gutter_size_changed_in_window (Lisp_Object specifier, struct window *w,
+                                       Lisp_Object oldval)
+{
+  recompute_overlaying_specifier (Vgutter_size);
+}
+
+static void
+default_gutter_border_width_changed_in_window (Lisp_Object specifier,
+                                               struct window *w,
+                                               Lisp_Object oldval)
+{
+  recompute_overlaying_specifier (Vgutter_border_width);
+}
+
+static void
+default_gutter_visible_p_changed_in_window (Lisp_Object specifier,
+                                            struct window *w,
+                                            Lisp_Object oldval)
+{
+  recompute_overlaying_specifier (Vgutter_visible_p);
+}
+
+void
+init_frame_gutters (struct frame *f)
+{
+  int pos;
+  struct window* w = XWINDOW (FRAME_LAST_NONMINIBUF_WINDOW (f));
+  /* We are here as far in frame creation so cached specifiers are
+     already recomputed, and possibly modified by resource
+     initialization. We need to recalculate autodetected gutters. */
+  for (pos = 0; pos< 4; pos++)
+    {
+      w->real_gutter_size[pos] = w->gutter_size[pos];
+      if (EQ (w->gutter_size[pos], Qautodetect)
+         && !NILP (w->gutter_visible_p[pos]))
+       {
+         w->real_gutter_size [pos] = calculate_gutter_size (w, pos);
+         MARK_GUTTER_CHANGED;
+         MARK_WINDOWS_CHANGED (w);
+       }
+    }
+}
+
+void
+syms_of_gutter (void)
+{
+  DEFSUBR (Fgutter_specifier_p);
+  DEFSUBR (Fset_default_gutter_position);
+  DEFSUBR (Fdefault_gutter_position);
+  DEFSUBR (Fgutter_pixel_height);
+  DEFSUBR (Fgutter_pixel_width);
+}
+
+void
+vars_of_gutter (void)
+{
+  staticpro (&Vdefault_gutter_position);
+  Vdefault_gutter_position = Qtop;
+
+  Fprovide (Qgutter);
+}
+
+void
+specifier_type_create_gutter (void)
+{
+  INITIALIZE_SPECIFIER_TYPE (gutter, "gutter", "gutter-specifier-p");
+
+  SPECIFIER_HAS_METHOD (gutter, validate);
+  SPECIFIER_HAS_METHOD (gutter, after_change);
+}
+
+void
+specifier_vars_of_gutter (void)
+{
+  Lisp_Object fb;
+
+  DEFVAR_SPECIFIER ("default-gutter", &Vdefault_gutter /*
+Specifier for a fallback gutter.
+Use `set-specifier' to change this.
+
+The position of this gutter is specified in the function
+`default-gutter-position'.  If the corresponding position-specific
+gutter (e.g. `top-gutter' if `default-gutter-position' is 'top)
+does not specify a gutter in a particular domain (usually a window),
+then the value of `default-gutter' in that domain, if any, will be
+used instead.
+
+Note that the gutter at any particular position will not be
+displayed unless its visibility flag is true and its thickness
+\(width or height, depending on orientation) is non-zero.  The
+visibility is controlled by the specifiers `top-gutter-visible-p',
+`bottom-gutter-visible-p', `left-gutter-visible-p', and
+`right-gutter-visible-p', and the thickness is controlled by the
+specifiers `top-gutter-height', `bottom-gutter-height',
+`left-gutter-width', and `right-gutter-width'.
+
+Note that one of the four visibility specifiers inherits from
+`default-gutter-visibility' and one of the four thickness
+specifiers inherits from either `default-gutter-width' or
+`default-gutter-height' (depending on orientation), just
+like for the gutter description specifiers (e.g. `top-gutter')
+mentioned above.
+
+Therefore, if you are setting `default-gutter', you should control
+the visibility and thickness using `default-gutter-visible-p',
+`default-gutter-width', and `default-gutter-height', rather than
+using position-specific specifiers.  That way, you will get sane
+behavior if the user changes the default gutter position.
+
+The gutter value should be a string or nil. You can attach extents and
+glyphs to the string and hence display glyphs and text in other fonts
+in the gutter area.
+
+*/ );
+
+  Vdefault_gutter = Fmake_specifier (Qgutter);
+  /* #### It would be even nicer if the specifier caching
+     automatically knew about specifier fallbacks, so we didn't
+     have to do it ourselves. */
+  set_specifier_caching (Vdefault_gutter,
+                        slot_offset (struct window,
+                                     default_gutter),
+                        default_gutter_specs_changed,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("top-gutter",
+                   &Vgutter[TOP_GUTTER] /*
+Specifier for the gutter at the top of the frame.
+Use `set-specifier' to change this.
+See `default-gutter' for a description of a valid gutter instantiator.
+*/ );
+  Vgutter[TOP_GUTTER] = Fmake_specifier (Qgutter);
+  set_specifier_caching (Vgutter[TOP_GUTTER],
+                        slot_offset (struct window,
+                                     gutter[TOP_GUTTER]),
+                        gutter_specs_changed,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("bottom-gutter",
+                   &Vgutter[BOTTOM_GUTTER] /*
+Specifier for the gutter at the bottom of the frame.
+Use `set-specifier' to change this.
+See `default-gutter' for a description of a valid gutter instantiator.
+
+Note that, unless the `default-gutter-position' is `bottom', by
+default the height of the bottom gutter (controlled by
+`bottom-gutter-height') is 0; thus, a bottom gutter will not be
+displayed even if you provide a value for `bottom-gutter'.
+*/ );
+  Vgutter[BOTTOM_GUTTER] = Fmake_specifier (Qgutter);
+  set_specifier_caching (Vgutter[BOTTOM_GUTTER],
+                        slot_offset (struct window,
+                                     gutter[BOTTOM_GUTTER]),
+                        gutter_specs_changed,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("left-gutter",
+                   &Vgutter[LEFT_GUTTER] /*
+Specifier for the gutter at the left edge of the frame.
+Use `set-specifier' to change this.
+See `default-gutter' for a description of a valid gutter instantiator.
+
+Note that, unless the `default-gutter-position' is `left', by
+default the height of the left gutter (controlled by
+`left-gutter-width') is 0; thus, a left gutter will not be
+displayed even if you provide a value for `left-gutter'.
+*/ );
+  Vgutter[LEFT_GUTTER] = Fmake_specifier (Qgutter);
+  set_specifier_caching (Vgutter[LEFT_GUTTER],
+                        slot_offset (struct window,
+                                     gutter[LEFT_GUTTER]),
+                        gutter_specs_changed,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("right-gutter",
+                   &Vgutter[RIGHT_GUTTER] /*
+Specifier for the gutter at the right edge of the frame.
+Use `set-specifier' to change this.
+See `default-gutter' for a description of a valid gutter instantiator.
+
+Note that, unless the `default-gutter-position' is `right', by
+default the height of the right gutter (controlled by
+`right-gutter-width') is 0; thus, a right gutter will not be
+displayed even if you provide a value for `right-gutter'.
+*/ );
+  Vgutter[RIGHT_GUTTER] = Fmake_specifier (Qgutter);
+  set_specifier_caching (Vgutter[RIGHT_GUTTER],
+                        slot_offset (struct window,
+                                     gutter[RIGHT_GUTTER]),
+                        gutter_specs_changed,
+                        0, 0);
+
+  /* initially, top inherits from default; this can be
+     changed with `set-default-gutter-position'. */
+  fb = list1 (Fcons (Qnil, Qnil));
+  set_specifier_fallback (Vdefault_gutter, fb);
+  set_specifier_fallback (Vgutter[TOP_GUTTER], Vdefault_gutter);
+  set_specifier_fallback (Vgutter[BOTTOM_GUTTER], fb);
+  set_specifier_fallback (Vgutter[LEFT_GUTTER],   fb);
+  set_specifier_fallback (Vgutter[RIGHT_GUTTER],  fb);
+
+  DEFVAR_SPECIFIER ("default-gutter-height", &Vdefault_gutter_height /*
+*Height of the default gutter, if it's oriented horizontally.
+This is a specifier; use `set-specifier' to change it.
+
+The position of the default gutter is specified by the function
+`set-default-gutter-position'.  If the corresponding position-specific
+gutter thickness specifier (e.g. `top-gutter-height' if
+`default-gutter-position' is 'top) does not specify a thickness in a
+particular domain (a window or a frame), then the value of
+`default-gutter-height' or `default-gutter-width' (depending on the
+gutter orientation) in that domain, if any, will be used instead.
+
+Note that `default-gutter-height' is only used when
+`default-gutter-position' is 'top or 'bottom, and `default-gutter-width'
+is only used when `default-gutter-position' is 'left or 'right.
+
+Note that all of the position-specific gutter thickness specifiers
+have a fallback value of zero when they do not correspond to the
+default gutter.  Therefore, you will have to set a non-zero thickness
+value if you want a position-specific gutter to be displayed.
+
+If you set the height to 'autodetect the size of the gutter will be
+calculated to be large enough to hold the contents of the gutter. This
+is the default.
+*/ );
+  Vdefault_gutter_height = Fmake_specifier (Qgeneric);
+  set_specifier_caching (Vdefault_gutter_height,
+                        slot_offset (struct window,
+                                     default_gutter_height),
+                        default_gutter_size_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("default-gutter-width", &Vdefault_gutter_width /*
+*Width of the default gutter, if it's oriented vertically.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-height' for more information.
+*/ );
+  Vdefault_gutter_width = Fmake_specifier (Qnatnum);
+  set_specifier_caching (Vdefault_gutter_width,
+                        slot_offset (struct window,
+                                     default_gutter_width),
+                        default_gutter_size_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("top-gutter-height",
+                   &Vgutter_size[TOP_GUTTER] /*
+*Height of the top gutter.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-height' for more information.
+*/ );
+  Vgutter_size[TOP_GUTTER] = Fmake_specifier (Qgeneric);
+  set_specifier_caching (Vgutter_size[TOP_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_size[TOP_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("bottom-gutter-height",
+                   &Vgutter_size[BOTTOM_GUTTER] /*
+*Height of the bottom gutter.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-height' for more information.
+*/ );
+  Vgutter_size[BOTTOM_GUTTER] = Fmake_specifier (Qgeneric);
+  set_specifier_caching (Vgutter_size[BOTTOM_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_size[BOTTOM_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("left-gutter-width",
+                   &Vgutter_size[LEFT_GUTTER] /*
+*Width of left gutter.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-height' for more information.
+*/ );
+  Vgutter_size[LEFT_GUTTER] = Fmake_specifier (Qnatnum);
+  set_specifier_caching (Vgutter_size[LEFT_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_size[LEFT_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("right-gutter-width",
+                   &Vgutter_size[RIGHT_GUTTER] /*
+*Width of right gutter.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-height' for more information.
+*/ );
+  Vgutter_size[RIGHT_GUTTER] = Fmake_specifier (Qnatnum);
+  set_specifier_caching (Vgutter_size[RIGHT_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_size[RIGHT_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  fb = Qnil;
+#ifdef HAVE_TTY
+  fb = Fcons (Fcons (list1 (Qtty), Qautodetect), fb);
+#endif
+#ifdef HAVE_X_WINDOWS
+  fb = Fcons (Fcons (list1 (Qx), Qautodetect), fb);
+#endif
+#ifdef HAVE_MS_WINDOWS
+  fb = Fcons (Fcons (list1 (Qmswindows), Qautodetect), fb);
+#endif
+  if (!NILP (fb))
+    set_specifier_fallback (Vdefault_gutter_height, fb);
+
+  fb = Qnil;
+#ifdef HAVE_TTY
+  fb = Fcons (Fcons (list1 (Qtty), Qzero), fb);
+#endif
+#ifdef HAVE_X_WINDOWS
+  fb = Fcons (Fcons (list1 (Qx), make_int (DEFAULT_GUTTER_WIDTH)), fb);
+#endif
+#ifdef HAVE_MS_WINDOWS
+  fb = Fcons (Fcons (list1 (Qmswindows), 
+                    make_int (DEFAULT_GUTTER_WIDTH)), fb);
+#endif
+  if (!NILP (fb))
+    set_specifier_fallback (Vdefault_gutter_width, fb);
+
+  set_specifier_fallback (Vgutter_size[TOP_GUTTER], Vdefault_gutter_height);
+  fb = list1 (Fcons (Qnil, Qzero));
+  set_specifier_fallback (Vgutter_size[BOTTOM_GUTTER], fb);
+  set_specifier_fallback (Vgutter_size[LEFT_GUTTER],   fb);
+  set_specifier_fallback (Vgutter_size[RIGHT_GUTTER],  fb);
+
+  DEFVAR_SPECIFIER ("default-gutter-border-width",
+                   &Vdefault_gutter_border_width /*
+*Width of the border around the default gutter.
+This is a specifier; use `set-specifier' to change it.
+
+The position of the default gutter is specified by the function
+`set-default-gutter-position'.  If the corresponding position-specific
+gutter border width specifier (e.g. `top-gutter-border-width' if
+`default-gutter-position' is 'top) does not specify a border width in a
+particular domain (a window or a frame), then the value of
+`default-gutter-border-width' in that domain, if any, will be used
+instead.
+
+*/ );
+  Vdefault_gutter_border_width = Fmake_specifier (Qnatnum);
+  set_specifier_caching (Vdefault_gutter_border_width,
+                        slot_offset (struct window,
+                                     default_gutter_border_width),
+                        default_gutter_border_width_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("top-gutter-border-width",
+                   &Vgutter_border_width[TOP_GUTTER] /*
+*Border width of the top gutter.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-height' for more information.
+*/ );
+  Vgutter_border_width[TOP_GUTTER] = Fmake_specifier (Qnatnum);
+  set_specifier_caching (Vgutter_border_width[TOP_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_border_width[TOP_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("bottom-gutter-border-width",
+                   &Vgutter_border_width[BOTTOM_GUTTER] /*
+*Border width of the bottom gutter.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-height' for more information.
+*/ );
+  Vgutter_border_width[BOTTOM_GUTTER] = Fmake_specifier (Qnatnum);
+  set_specifier_caching (Vgutter_border_width[BOTTOM_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_border_width[BOTTOM_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("left-gutter-border-width",
+                   &Vgutter_border_width[LEFT_GUTTER] /*
+*Border width of left gutter.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-height' for more information.
+*/ );
+  Vgutter_border_width[LEFT_GUTTER] = Fmake_specifier (Qnatnum);
+  set_specifier_caching (Vgutter_border_width[LEFT_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_border_width[LEFT_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("right-gutter-border-width",
+                   &Vgutter_border_width[RIGHT_GUTTER] /*
+*Border width of right gutter.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-height' for more information.
+*/ );
+  Vgutter_border_width[RIGHT_GUTTER] = Fmake_specifier (Qnatnum);
+  set_specifier_caching (Vgutter_border_width[RIGHT_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_border_width[RIGHT_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  fb = Qnil;
+#ifdef HAVE_TTY
+  fb = Fcons (Fcons (list1 (Qtty), Qzero), fb);
+#endif
+#ifdef HAVE_X_WINDOWS
+  fb = Fcons (Fcons (list1 (Qx), make_int (DEFAULT_GUTTER_BORDER_WIDTH)), fb);
+#endif
+#ifdef HAVE_MS_WINDOWS
+  fb = Fcons (Fcons (list1 (Qmswindows), make_int (DEFAULT_GUTTER_BORDER_WIDTH)), fb);
+#endif
+  if (!NILP (fb))
+    set_specifier_fallback (Vdefault_gutter_border_width, fb);
+
+  set_specifier_fallback (Vgutter_border_width[TOP_GUTTER], Vdefault_gutter_border_width);
+  fb = list1 (Fcons (Qnil, Qzero));
+  set_specifier_fallback (Vgutter_border_width[BOTTOM_GUTTER], fb);
+  set_specifier_fallback (Vgutter_border_width[LEFT_GUTTER],   fb);
+  set_specifier_fallback (Vgutter_border_width[RIGHT_GUTTER],  fb);
+
+  DEFVAR_SPECIFIER ("default-gutter-visible-p", &Vdefault_gutter_visible_p /*
+*Whether the default gutter is visible.
+This is a specifier; use `set-specifier' to change it.
+
+The position of the default gutter is specified by the function
+`set-default-gutter-position'.  If the corresponding position-specific
+gutter visibility specifier (e.g. `top-gutter-visible-p' if
+`default-gutter-position' is 'top) does not specify a visible-p value
+in a particular domain (a window or a frame), then the value of
+`default-gutter-visible-p' in that domain, if any, will be used
+instead.
+
+`default-gutter-visible-p' and all of the position-specific gutter
+visibility specifiers have a fallback value of true.
+*/ );
+  Vdefault_gutter_visible_p = Fmake_specifier (Qboolean);
+  set_specifier_caching (Vdefault_gutter_visible_p,
+                        slot_offset (struct window,
+                                     default_gutter_visible_p),
+                        default_gutter_visible_p_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("top-gutter-visible-p",
+                   &Vgutter_visible_p[TOP_GUTTER] /*
+*Whether the top gutter is visible.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-visible-p' for more information.
+*/ );
+  Vgutter_visible_p[TOP_GUTTER] = Fmake_specifier (Qboolean);
+  set_specifier_caching (Vgutter_visible_p[TOP_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_visible_p[TOP_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("bottom-gutter-visible-p",
+                   &Vgutter_visible_p[BOTTOM_GUTTER] /*
+*Whether the bottom gutter is visible.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-visible-p' for more information.
+*/ );
+  Vgutter_visible_p[BOTTOM_GUTTER] = Fmake_specifier (Qboolean);
+  set_specifier_caching (Vgutter_visible_p[BOTTOM_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_visible_p[BOTTOM_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("left-gutter-visible-p",
+                   &Vgutter_visible_p[LEFT_GUTTER] /*
+*Whether the left gutter is visible.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-visible-p' for more information.
+*/ );
+  Vgutter_visible_p[LEFT_GUTTER] = Fmake_specifier (Qboolean);
+  set_specifier_caching (Vgutter_visible_p[LEFT_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_visible_p[LEFT_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  DEFVAR_SPECIFIER ("right-gutter-visible-p",
+                   &Vgutter_visible_p[RIGHT_GUTTER] /*
+*Whether the right gutter is visible.
+This is a specifier; use `set-specifier' to change it.
+
+See `default-gutter-visible-p' for more information.
+*/ );
+  Vgutter_visible_p[RIGHT_GUTTER] = Fmake_specifier (Qboolean);
+  set_specifier_caching (Vgutter_visible_p[RIGHT_GUTTER],
+                        slot_offset (struct window,
+                                     gutter_visible_p[RIGHT_GUTTER]),
+                        gutter_geometry_changed_in_window,
+                        0, 0);
+
+  /* initially, top inherits from default; this can be
+     changed with `set-default-gutter-position'. */
+  fb = list1 (Fcons (Qnil, Qt));
+  set_specifier_fallback (Vdefault_gutter_visible_p, fb);
+  set_specifier_fallback (Vgutter_visible_p[TOP_GUTTER],
+                         Vdefault_gutter_visible_p);
+  set_specifier_fallback (Vgutter_visible_p[BOTTOM_GUTTER], fb);
+  set_specifier_fallback (Vgutter_visible_p[LEFT_GUTTER],   fb);
+  set_specifier_fallback (Vgutter_visible_p[RIGHT_GUTTER],  fb);
+
+}
diff --git a/src/gutter.h b/src/gutter.h
new file mode 100644 (file)
index 0000000..d025865
--- /dev/null
@@ -0,0 +1,123 @@
+/* Define general gutter support.
+   Copyright (C) 1999 Andy Piper.
+
+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. */
+
+#ifndef _XEMACS_GUTTER_H_
+#define _XEMACS_GUTTER_H_
+
+#include "specifier.h"
+
+#define DEVICE_SUPPORTS_GUTTERS_P(d)           \
+  (HAS_DEVMETH_P ((d), output_frame_gutters))
+
+DECLARE_SPECIFIER_TYPE (gutter);
+#define XGUTTER_SPECIFIER(x) XSPECIFIER_TYPE (x, gutter)
+#define XSETGUTTER_SPECIFIER(x, p) XSETSPECIFIER_TYPE (x, p, gutter)
+#define GUTTER_SPECIFIERP(x) SPECIFIER_TYPEP (x, gutter)
+#define CHECK_GUTTER_SPECIFIER(x) CHECK_SPECIFIER_TYPE (x, gutter)
+#define CONCHECK_GUTTER_SPECIFIER(x) CONCHECK_SPECIFIER_TYPE (x, gutter)
+
+#define DEFAULT_GUTTER_WIDTH           40
+#define DEFAULT_GUTTER_BORDER_WIDTH    2
+
+enum gutter_pos
+{
+  TOP_GUTTER,
+  BOTTOM_GUTTER,
+  LEFT_GUTTER,
+  RIGHT_GUTTER
+};
+
+extern Lisp_Object Qgutter;
+
+extern Lisp_Object Vgutter_size[4];
+extern Lisp_Object Vgutter_border_width[4];
+void update_frame_gutters (struct frame *f);
+void init_frame_gutters (struct frame *f);
+void init_device_gutters (struct device *d);
+void init_global_gutters (struct device *d);
+void free_frame_gutters (struct frame *f);
+void redraw_exposed_gutters (struct frame *f, int x, int y, int width,
+                            int height);
+
+#define WINDOW_GUTTER_BORDER_WIDTH(w, pos) \
+(NILP ((w)->gutter_border_width[pos]) ? 0 : XINT ((w)->gutter_border_width[pos]))
+#define WINDOW_GUTTER_SIZE(w, pos) \
+(NILP ((w)->gutter_size[pos]) ? 0 : XINT ((w)->gutter_size[pos]))
+#define WINDOW_GUTTER_SIZE_INTERNAL(w, pos) \
+(NILP ((w)->real_gutter_size[pos]) ? 0 : XINT ((w)->real_gutter_size[pos]))
+#define WINDOW_GUTTER_VISIBLE(w, pos) \
+((w)->gutter_visible_p[pos])
+#define WINDOW_GUTTER(w, pos) \
+((w)->gutter[pos])
+
+#define WINDOW_REAL_GUTTER_SIZE(w, pos)        \
+  (!NILP (WINDOW_GUTTER_VISIBLE (w, pos))              \
+   ? WINDOW_GUTTER_SIZE_INTERNAL (w, pos)      \
+   : 0)
+#define WINDOW_REAL_GUTTER_VISIBLE(f, pos)     \
+   (WINDOW_REAL_GUTTER_SIZE (f, pos) > 0)
+#define WINDOW_REAL_GUTTER_BORDER_WIDTH(f, pos)        \
+  ((!NILP (WINDOW_GUTTER_VISIBLE (f, pos))             \
+   && WINDOW_GUTTER_SIZE_INTERNAL (f,pos) > 0) \
+   ? WINDOW_GUTTER_BORDER_WIDTH (f, pos)       \
+   : 0)
+#define WINDOW_REAL_GUTTER_BOUNDS(f, pos)      \
+   (WINDOW_REAL_GUTTER_SIZE (f,pos) +          \
+    2 * WINDOW_REAL_GUTTER_BORDER_WIDTH (f,pos))
+
+/* these macros predicate size on position and type of window */
+#define WINDOW_REAL_TOP_GUTTER_BOUNDS(w)       \
+   ((!MINI_WINDOW_P (w) && window_is_highest (w)) ?    \
+   WINDOW_REAL_GUTTER_BOUNDS (w,TOP_GUTTER) : 0)
+#define WINDOW_REAL_BOTTOM_GUTTER_BOUNDS(w)    \
+   ((!MINI_WINDOW_P (w) && window_is_lowest (w)) ?     \
+   WINDOW_REAL_GUTTER_BOUNDS (w,BOTTOM_GUTTER) : 0)
+#define WINDOW_REAL_LEFT_GUTTER_BOUNDS(w)      \
+   ((!MINI_WINDOW_P (w) && window_is_leftmost (w)) ?   \
+   WINDOW_REAL_GUTTER_BOUNDS (w,LEFT_GUTTER) : 0)
+#define WINDOW_REAL_RIGHT_GUTTER_BOUNDS(w)     \
+   ((!MINI_WINDOW_P (w) && window_is_rightmost (w)) ?  \
+   WINDOW_REAL_GUTTER_BOUNDS (w,RIGHT_GUTTER) : 0)
+
+#define FRAME_GUTTER_VISIBLE(f, pos) \
+   WINDOW_REAL_GUTTER_VISIBLE (XWINDOW (FRAME_LAST_NONMINIBUF_WINDOW (f)), pos)
+#define FRAME_GUTTER_SIZE(f, pos) \
+   WINDOW_REAL_GUTTER_SIZE (XWINDOW (FRAME_LAST_NONMINIBUF_WINDOW (f)), pos)
+#define FRAME_GUTTER_BOUNDS(f, pos) \
+   WINDOW_REAL_GUTTER_BOUNDS (XWINDOW (FRAME_LAST_NONMINIBUF_WINDOW (f)), pos)
+#define FRAME_GUTTER_BORDER_WIDTH(f, pos) \
+   WINDOW_REAL_GUTTER_BORDER_WIDTH (XWINDOW (FRAME_LAST_NONMINIBUF_WINDOW (f)), pos)
+
+#define FRAME_GUTTER(f, pos) \
+WINDOW_GUTTER (XWINDOW (FRAME_LAST_NONMINIBUF_WINDOW (f)), pos)
+
+/* these macros predicate size on position and type of window */
+#define FRAME_TOP_GUTTER_BOUNDS(f) \
+   WINDOW_REAL_GUTTER_BOUNDS (XWINDOW (FRAME_LAST_NONMINIBUF_WINDOW (f)), TOP_GUTTER)
+#define FRAME_BOTTOM_GUTTER_BOUNDS(f) \
+   WINDOW_REAL_GUTTER_BOUNDS (XWINDOW (FRAME_LAST_NONMINIBUF_WINDOW (f)), BOTTOM_GUTTER)
+#define FRAME_LEFT_GUTTER_BOUNDS(f) \
+   WINDOW_REAL_GUTTER_BOUNDS (XWINDOW (FRAME_LAST_NONMINIBUF_WINDOW (f)), LEFT_GUTTER)
+#define FRAME_RIGHT_GUTTER_BOUNDS(f) \
+   WINDOW_REAL_GUTTER_BOUNDS (XWINDOW (FRAME_LAST_NONMINIBUF_WINDOW (f)), RIGHT_GUTTER)
+
+#endif /* _XEMACS_GUTTER_H_ */
diff --git a/src/s/sco7.h b/src/s/sco7.h
new file mode 100644 (file)
index 0000000..988600c
--- /dev/null
@@ -0,0 +1,14 @@
+/* Synched up with: FSF 19.31. */
+
+/* s/ file for System V release 4.2.  */
+
+#include "usg5-4.h"
+
+/* Motif needs -lgen.  */
+#undef LIBS_SYSTEM
+#define LIBS_SYSTEM "-lsocket -lnsl -lelf -lgen"
+
+#define VFORK_RETURN_TYPE pid_t
+
+/* XEmacs change: communicate to m/intel386.h */
+#define USG5_4_2
diff --git a/tests/gutter-test.el b/tests/gutter-test.el
new file mode 100644 (file)
index 0000000..41a870a
--- /dev/null
@@ -0,0 +1,17 @@
+(setq str "Hello There\nHello Again")
+(set-extent-begin-glyph 
+ (make-extent 0 0 str)
+ (make-glyph [xpm :file "../etc/xemacs-icon.xpm"]))
+
+(set-extent-begin-glyph
+ (make-extent 3 3 str)
+ (make-glyph 
+  [button :width 5 :height 1
+         :face modeline-mousable
+         :descriptor "ok" :selected t]))
+
+(set-specifier default-gutter-height 'autodetect)
+(set-specifier default-gutter-width 40)
+(set-specifier default-gutter-border-width 2)
+(set-specifier default-gutter str)
+(set-default-gutter-position 'bottom)